From f9f95e168a5d18b341b77efa66f3cebdb830bc17 Mon Sep 17 00:00:00 2001 From: unknown Date: Wed, 5 Apr 2006 10:30:18 +0200 Subject: post-commit: Removed commit messages to public lists (for now) BitKeeper/triggers/post-commit: Removed commit messages to public lists (for now) --- BitKeeper/triggers/post-commit | 45 ------------------------------------------ 1 file changed, 45 deletions(-) diff --git a/BitKeeper/triggers/post-commit b/BitKeeper/triggers/post-commit index ce5e03b0ac9..782a4c4dca8 100755 --- a/BitKeeper/triggers/post-commit +++ b/BitKeeper/triggers/post-commit @@ -66,51 +66,6 @@ EOF bk cset -r+ -d ) | $SENDMAIL -t -#++ -# commits@ mail -#-- - echo "Notifying commits list at $COMMITS" - ( - cat < -From: $FROM -To: $COMMITS -Subject: bk commit into $VERSION tree ($CHANGESET)$BS -X-CSetKey: <$CSETKEY> -$BH -Below is the list of changes that have just been committed into a local -$VERSION repository of $USER. When $USER does a push these changes will -be propagated to the main repository and, within 24 hours after the -push, to the public repository. -For information on how to access the public repository -see http://dev.mysql.com/doc/mysql/en/installing-source-tree.html - -EOF - bk changes -v -r+ - bk cset -r+ -d - ) | head -n $LIMIT | $SENDMAIL -t - -#++ -# docs-commit@ mail -# Picks up anything under the Docs subdirectory (relevant for docs team). -#-- - bk changes -v -r+ | grep -q " Docs/" - if [ $? -eq 0 ] - then - echo "Notifying docs list at $DOCS" - ( - cat < -From: $FROM -To: $DOCS -Subject: bk commit - $VERSION tree (Manual) ($CHANGESET)$BS - -EOF - bk changes -v -r+ - bk cset -r+ -d - ) | $SENDMAIL -t - fi - else echo "commit failed because '$BK_STATUS', you may need to re-clone..." fi -- cgit v1.2.1 From d2b8d744d9f9ab32478416885d7929ad00d3b276 Mon Sep 17 00:00:00 2001 From: unknown Date: Tue, 11 Apr 2006 16:45:10 +0300 Subject: Added storage/maria (based on MyISAM). WL#3245 Moved things into ft_global.h, my_handler.h and myisamchk.h to allow MyISAM and Maria to share code and defines Rename of not properly renamed functions in MyISAM and my_handler.c Renamed some MI_ defines to HA_ to allow MyISAM and Maria to share Added maria variables to mysqld.cc and set_var.cc Fixed compiler warnings BitKeeper/etc/ignore: added storage/maria/*.MAI BUILD/SETUP.sh: Compile maria by default BitKeeper/triggers/post-commit: No public maria emails. Mark changesets emails with 'maria' configure.in: Add maria include/ft_global.h: Move defines needed by maria and MyISAM here include/keycache.h: Add support for default key_cache if cache not found include/my_base.h: Add invalidator_by_filename include/my_handler.h: Remove duplicate header files Add defines that are typical for handlers (MyISAM and Maria) include/myisam.h: Move things to my_handler.h to allow Maria and MyISAM to share things (Some things needed to be shared to allow sharing of HA_CHECK structure) libmysqld/Makefile.am: Added ha_maria.cc mysys/mf_keycaches.c: Added default value for multi_key_cache_search mysys/my_handler.c: mi_compare_text -> ha_compare_text Removed compiler warnings sql/ha_myisam.cc: MI_CHECK -> HA_CHECK MI_MAX_KEY_LENGTH -> HA_MAX_KEY_LENGTH sql/ha_myisam.h: MI_CHECK -> HA_CHECK MI_MAX_KEY_LENGTH -> HA_MAX_KEY_LENGTH sql/ha_myisammrg.h: MI_CHECK -> HA_CHECK MI_MAX_KEY_LENGTH -> HA_MAX_KEY_LENGTH sql/handler.h: Added MARIA Added inclusion of my_handler.h sql/item_func.h: Remove duplicate include sql/mysql_priv.h: Added maria variables sql/mysqld.cc: Added maria sql/set_var.cc: Added maria status variables sql/set_var.h: Added maria sql/sql_class.h: Added maria status variables sql/sql_sort.h: Remove duplicate BUFFPEK struct storage/Makefile.am: Added maria storage/csv/ha_tina.cc: Removed compiler warning storage/myisam/Makefile.am: Added ft_myisam.c storage/myisam/ft_boolean_search.c: mi_compare_text -> ha_compare_text MI_MAX_KEY_BUFF -> HA_MAX_KEY_BUFF Remove compiler warnings storage/myisam/ft_nlq_search.c: mi_compare_text -> ha_compare_text storage/myisam/ft_parser.c: mi_compare_text -> ha_compare_text storage/myisam/ft_static.c: Move ft_init_search() to ft_myisam.c to make ft_static.c independent of MyISAM storage/myisam/ft_stopwords.c: mi_compare_text -> ha_compare_text storage/myisam/ft_update.c: mi_compare_text -> ha_compare_text storage/myisam/fulltext.h: Move things to ft_global.h to allow to share more things between MyISAM and Maria storage/myisam/mi_check.c: MI_CHECK -> HA_CHECK storage/myisam/mi_create.c: MI_MAX_POSSIBLE_KEY -> HA_MAX_POSSIBLE_KEY MI_MAX_KEY_BLOCK_SIZE -> HA_MAX_KEY_BLOCK_SIZE MI_MAX_KEY_SEG -> HA_MAX_KEY_SEG MI_MAX_KEY_BUFF -> HA_MAX_KEY_BUFF storage/myisam/mi_delete.c: MI_MAX_KEY_BUFF -> HA_MAX_KEY_BUFF storage/myisam/mi_delete_all.c: Remove not used variable storage/myisam/mi_dynrec.c: _my_calc_total_blob_length -> _mi_calc_total_blob_length storage/myisam/mi_key.c: _my_store_blob_length -> _mi_store_blob_length storage/myisam/mi_log.c: _my_calc_total_blob_length -> _mi_calc_total_blob_length storage/myisam/mi_open.c: MI_MAX_POSSIBLE_KEY -> HA_MAX_POSSIBLE_KEY MI_MAX_KEY_SEG -> HA_MAX_KEY_SEG MI_MAX_KEY_BUFF -> HA_MAX_KEY_BUFF my_n_base_info_read -> mi_n_base_info_read storage/myisam/mi_packrec.c: Made read_pack_length static _my_store_blob_length -> _mi_store_blob_length Remove not used variable storage/myisam/mi_range.c: MI_MAX_KEY_BUFF -> HA_MAX_KEY_BUFF storage/myisam/mi_search.c: MI_MAX_KEY_BUFF -> HA_MAX_KEY_BUFF storage/myisam/mi_test1.c: MI_MAX_KEY_LENGTH -> HA_MAX_KEY_LENGTH storage/myisam/mi_test2.c: Fixed compiler warning storage/myisam/mi_unique.c: Fixed compiler warning mi_compare_text -> ha_compare_text storage/myisam/mi_update.c: MI_MAX_KEY_BUFF -> HA_MAX_KEY_BUFF storage/myisam/mi_write.c: Rename of defines and functions storage/myisam/myisamchk.c: Rename of defines and functions storage/myisam/myisamdef.h: Remove tabs Indentation fixes (Large changes as I did run indent-ex on the file) Move some things to myisamchk.h Added missing functions that gave compiler warnings storage/myisam/myisamlog.c: Rename of defines and functions storage/myisam/myisampack.c: Remove compiler warning storage/myisam/rt_index.c: Rename of defines and functions storage/myisam/sort.c: Rename of defines, functions and structures config/ac-macros/ha_maria.m4: New BitKeeper file ``config/ac-macros/ha_maria.m4'' include/maria.h: New BitKeeper file ``include/maria.h'' include/myisamchk.h: New BitKeeper file ``include/myisamchk.h'' libmysqld/ha_maria.cc: New BitKeeper file ``libmysqld/ha_maria.cc'' mysql-test/include/have_maria.inc: New BitKeeper file ``mysql-test/include/have_maria.inc'' mysql-test/r/have_maria.require: New BitKeeper file ``mysql-test/r/have_maria.require'' mysql-test/r/maria.result: New BitKeeper file ``mysql-test/r/maria.result'' mysql-test/r/ps_maria.result: New BitKeeper file ``mysql-test/r/ps_maria.result'' mysql-test/t/maria.test: New BitKeeper file ``mysql-test/t/maria.test'' mysql-test/t/ps_maria.test: New BitKeeper file ``mysql-test/t/ps_maria.test'' sql/ha_maria.cc: New BitKeeper file ``sql/ha_maria.cc'' sql/ha_maria.h: New BitKeeper file ``sql/ha_maria.h'' storage/maria/Makefile.am: New BitKeeper file ``storage/maria/Makefile.am'' storage/maria/cmakelists.txt: New BitKeeper file ``storage/maria/cmakelists.txt'' storage/maria/ft_maria.c: New BitKeeper file ``storage/maria/ft_maria.c'' storage/maria/ma_cache.c: New BitKeeper file ``storage/maria/ma_cache.c'' storage/maria/ma_changed.c: New BitKeeper file ``storage/maria/ma_changed.c'' storage/maria/ma_check.c: New BitKeeper file ``storage/maria/ma_check.c'' storage/maria/ma_checksum.c: New BitKeeper file ``storage/maria/ma_checksum.c'' storage/maria/ma_close.c: New BitKeeper file ``storage/maria/ma_close.c'' storage/maria/ma_create.c: New BitKeeper file ``storage/maria/ma_create.c'' storage/maria/ma_dbug.c: New BitKeeper file ``storage/maria/ma_dbug.c'' storage/maria/ma_delete.c: New BitKeeper file ``storage/maria/ma_delete.c'' storage/maria/ma_delete_all.c: New BitKeeper file ``storage/maria/ma_delete_all.c'' storage/maria/ma_delete_table.c: New BitKeeper file ``storage/maria/ma_delete_table.c'' storage/maria/ma_dynrec.c: New BitKeeper file ``storage/maria/ma_dynrec.c'' storage/maria/ma_extra.c: New BitKeeper file ``storage/maria/ma_extra.c'' storage/maria/ma_ft_boolean_search.c: New BitKeeper file ``storage/maria/ma_ft_boolean_search.c'' storage/maria/ma_ft_eval.c: New BitKeeper file ``storage/maria/ma_ft_eval.c'' storage/maria/ma_ft_eval.h: New BitKeeper file ``storage/maria/ma_ft_eval.h'' storage/maria/ma_ft_nlq_search.c: New BitKeeper file ``storage/maria/ma_ft_nlq_search.c'' storage/maria/ma_ft_parser.c: New BitKeeper file ``storage/maria/ma_ft_parser.c'' storage/maria/ma_ft_stem.c: New BitKeeper file ``storage/maria/ma_ft_stem.c'' storage/maria/ma_ft_test1.c: New BitKeeper file ``storage/maria/ma_ft_test1.c'' storage/maria/ma_ft_test1.h: New BitKeeper file ``storage/maria/ma_ft_test1.h'' storage/maria/ma_ft_update.c: New BitKeeper file ``storage/maria/ma_ft_update.c'' storage/maria/ma_ftdefs.h: New BitKeeper file ``storage/maria/ma_ftdefs.h'' storage/maria/ma_fulltext.h: New BitKeeper file ``storage/maria/ma_fulltext.h'' storage/maria/ma_info.c: New BitKeeper file ``storage/maria/ma_info.c'' storage/maria/ma_init.c: New BitKeeper file ``storage/maria/ma_init.c'' storage/maria/ma_key.c: New BitKeeper file ``storage/maria/ma_key.c'' storage/maria/ma_keycache.c: New BitKeeper file ``storage/maria/ma_keycache.c'' storage/maria/ma_locking.c: New BitKeeper file ``storage/maria/ma_locking.c'' storage/maria/ma_log.c: New BitKeeper file ``storage/maria/ma_log.c'' storage/maria/ma_open.c: New BitKeeper file ``storage/maria/ma_open.c'' storage/maria/ma_packrec.c: New BitKeeper file ``storage/maria/ma_packrec.c'' storage/maria/ma_page.c: New BitKeeper file ``storage/maria/ma_page.c'' storage/maria/ma_panic.c: New BitKeeper file ``storage/maria/ma_panic.c'' storage/maria/ma_preload.c: New BitKeeper file ``storage/maria/ma_preload.c'' storage/maria/ma_range.c: New BitKeeper file ``storage/maria/ma_range.c'' storage/maria/ma_rename.c: New BitKeeper file ``storage/maria/ma_rename.c'' storage/maria/ma_rfirst.c: New BitKeeper file ``storage/maria/ma_rfirst.c'' storage/maria/ma_rkey.c: New BitKeeper file ``storage/maria/ma_rkey.c'' storage/maria/ma_rlast.c: New BitKeeper file ``storage/maria/ma_rlast.c'' storage/maria/ma_rnext.c: New BitKeeper file ``storage/maria/ma_rnext.c'' storage/maria/ma_rnext_same.c: New BitKeeper file ``storage/maria/ma_rnext_same.c'' storage/maria/ma_rprev.c: New BitKeeper file ``storage/maria/ma_rprev.c'' storage/maria/ma_rrnd.c: New BitKeeper file ``storage/maria/ma_rrnd.c'' storage/maria/ma_rsame.c: New BitKeeper file ``storage/maria/ma_rsame.c'' storage/maria/ma_rsamepos.c: New BitKeeper file ``storage/maria/ma_rsamepos.c'' storage/maria/ma_rt_index.c: New BitKeeper file ``storage/maria/ma_rt_index.c'' storage/maria/ma_rt_index.h: New BitKeeper file ``storage/maria/ma_rt_index.h'' storage/maria/ma_rt_key.c: New BitKeeper file ``storage/maria/ma_rt_key.c'' storage/maria/ma_rt_key.h: New BitKeeper file ``storage/maria/ma_rt_key.h'' storage/maria/ma_rt_mbr.c: New BitKeeper file ``storage/maria/ma_rt_mbr.c'' storage/maria/ma_rt_mbr.h: New BitKeeper file ``storage/maria/ma_rt_mbr.h'' storage/maria/ma_rt_split.c: New BitKeeper file ``storage/maria/ma_rt_split.c'' storage/maria/ma_rt_test.c: New BitKeeper file ``storage/maria/ma_rt_test.c'' storage/maria/ma_scan.c: New BitKeeper file ``storage/maria/ma_scan.c'' storage/maria/ma_search.c: New BitKeeper file ``storage/maria/ma_search.c'' storage/maria/ma_sort.c: New BitKeeper file ``storage/maria/ma_sort.c'' storage/maria/ma_sp_defs.h: New BitKeeper file ``storage/maria/ma_sp_defs.h'' storage/maria/ma_sp_key.c: New BitKeeper file ``storage/maria/ma_sp_key.c'' storage/maria/ma_sp_test.c: New BitKeeper file ``storage/maria/ma_sp_test.c'' storage/maria/ma_static.c: New BitKeeper file ``storage/maria/ma_static.c'' storage/maria/ma_statrec.c: New BitKeeper file ``storage/maria/ma_statrec.c'' storage/maria/ma_test1.c: New BitKeeper file ``storage/maria/ma_test1.c'' storage/maria/ma_test2.c: New BitKeeper file ``storage/maria/ma_test2.c'' storage/maria/ma_test3.c: New BitKeeper file ``storage/maria/ma_test3.c'' storage/maria/ma_test_all.sh: New BitKeeper file ``storage/maria/ma_test_all.sh'' storage/maria/ma_unique.c: New BitKeeper file ``storage/maria/ma_unique.c'' storage/maria/ma_update.c: New BitKeeper file ``storage/maria/ma_update.c'' storage/maria/ma_write.c: New BitKeeper file ``storage/maria/ma_write.c'' storage/maria/maria_chk.c: New BitKeeper file ``storage/maria/maria_chk.c'' storage/maria/maria_def.h: New BitKeeper file ``storage/maria/maria_def.h'' storage/maria/maria_ftdump.c: New BitKeeper file ``storage/maria/maria_ftdump.c'' storage/maria/maria_log.c: New BitKeeper file ``storage/maria/maria_log.c'' storage/maria/maria_pack.c: New BitKeeper file ``storage/maria/maria_pack.c'' storage/maria/maria_rename.sh: New BitKeeper file ``storage/maria/maria_rename.sh'' storage/maria/test_pack: New BitKeeper file ``storage/maria/test_pack'' storage/myisam/ft_myisam.c: New BitKeeper file ``storage/myisam/ft_myisam.c'' --- .bzrignore | 13 + BUILD/SETUP.sh | 2 +- BitKeeper/triggers/post-commit | 47 +- config/ac-macros/ha_maria.m4 | 29 + configure.in | 4 + include/ft_global.h | 11 + include/keycache.h | 3 +- include/maria.h | 418 ++++ include/my_base.h | 4 +- include/my_handler.h | 27 +- include/myisam.h | 258 +- include/myisamchk.h | 160 ++ libmysqld/Makefile.am | 4 +- mysql-test/include/have_maria.inc | 4 + mysql-test/r/have_maria.require | 2 + mysql-test/r/maria.result | 1417 +++++++++++ mysql-test/r/ps_maria.result | 3137 +++++++++++++++++++++++++ mysql-test/t/maria.test | 802 +++++++ mysql-test/t/ps_maria.test | 46 + mysys/mf_keycaches.c | 14 +- mysys/my_handler.c | 30 +- sql/ha_maria.cc | 1836 +++++++++++++++ sql/ha_maria.h | 141 ++ sql/ha_myisam.cc | 38 +- sql/ha_myisam.h | 7 +- sql/ha_myisammrg.h | 4 +- sql/handler.h | 2 + sql/item_func.h | 1 - sql/mysql_priv.h | 6 + sql/mysqld.cc | 81 +- sql/set_var.cc | 36 +- sql/set_var.h | 1 + sql/sql_class.h | 5 +- sql/sql_sort.h | 11 +- storage/Makefile.am | 2 +- storage/csv/ha_tina.cc | 3 +- storage/maria/Makefile.am | 106 + storage/maria/cmakelists.txt | 26 + storage/maria/ft_maria.c | 49 + storage/maria/ma_cache.c | 108 + storage/maria/ma_changed.c | 34 + storage/maria/ma_check.c | 4301 ++++++++++++++++++++++++++++++++++ storage/maria/ma_checksum.c | 65 + storage/maria/ma_close.c | 124 + storage/maria/ma_create.c | 816 +++++++ storage/maria/ma_dbug.c | 193 ++ storage/maria/ma_delete.c | 890 +++++++ storage/maria/ma_delete_all.c | 79 + storage/maria/ma_delete_table.c | 58 + storage/maria/ma_dynrec.c | 1811 ++++++++++++++ storage/maria/ma_extra.c | 426 ++++ storage/maria/ma_ft_boolean_search.c | 955 ++++++++ storage/maria/ma_ft_eval.c | 254 ++ storage/maria/ma_ft_eval.h | 42 + storage/maria/ma_ft_nlq_search.c | 366 +++ storage/maria/ma_ft_parser.c | 394 ++++ storage/maria/ma_ft_stem.c | 19 + storage/maria/ma_ft_test1.c | 317 +++ storage/maria/ma_ft_test1.h | 421 ++++ storage/maria/ma_ft_update.c | 359 +++ storage/maria/ma_ftdefs.h | 149 ++ storage/maria/ma_fulltext.h | 28 + storage/maria/ma_info.c | 133 ++ storage/maria/ma_init.c | 59 + storage/maria/ma_key.c | 592 +++++ storage/maria/ma_keycache.c | 163 ++ storage/maria/ma_locking.c | 554 +++++ storage/maria/ma_log.c | 164 ++ storage/maria/ma_open.c | 1288 ++++++++++ storage/maria/ma_packrec.c | 1346 +++++++++++ storage/maria/ma_page.c | 160 ++ storage/maria/ma_panic.c | 124 + storage/maria/ma_preload.c | 117 + storage/maria/ma_range.c | 244 ++ storage/maria/ma_rename.c | 61 + storage/maria/ma_rfirst.c | 27 + storage/maria/ma_rkey.c | 144 ++ storage/maria/ma_rlast.c | 27 + storage/maria/ma_rnext.c | 122 + storage/maria/ma_rnext_same.c | 105 + storage/maria/ma_rprev.c | 88 + storage/maria/ma_rrnd.c | 60 + storage/maria/ma_rsame.c | 66 + storage/maria/ma_rsamepos.c | 56 + storage/maria/ma_rt_index.c | 1081 +++++++++ storage/maria/ma_rt_index.h | 47 + storage/maria/ma_rt_key.c | 100 + storage/maria/ma_rt_key.h | 33 + storage/maria/ma_rt_mbr.c | 801 +++++++ storage/maria/ma_rt_mbr.h | 38 + storage/maria/ma_rt_split.c | 350 +++ storage/maria/ma_rt_test.c | 473 ++++ storage/maria/ma_scan.c | 46 + storage/maria/ma_search.c | 1894 +++++++++++++++ storage/maria/ma_sort.c | 1021 ++++++++ storage/maria/ma_sp_defs.h | 48 + storage/maria/ma_sp_key.c | 300 +++ storage/maria/ma_sp_test.c | 568 +++++ storage/maria/ma_static.c | 65 + storage/maria/ma_statrec.c | 301 +++ storage/maria/ma_test1.c | 681 ++++++ storage/maria/ma_test2.c | 1050 +++++++++ storage/maria/ma_test3.c | 502 ++++ storage/maria/ma_test_all.sh | 147 ++ storage/maria/ma_unique.c | 234 ++ storage/maria/ma_update.c | 232 ++ storage/maria/ma_write.c | 1033 ++++++++ storage/maria/maria_chk.c | 1824 ++++++++++++++ storage/maria/maria_def.h | 751 ++++++ storage/maria/maria_ftdump.c | 279 +++ storage/maria/maria_log.c | 848 +++++++ storage/maria/maria_pack.c | 3202 +++++++++++++++++++++++++ storage/maria/maria_rename.sh | 17 + storage/maria/test_pack | 10 + storage/myisam/Makefile.am | 3 +- storage/myisam/ft_boolean_search.c | 16 +- storage/myisam/ft_myisam.c | 36 + storage/myisam/ft_nlq_search.c | 2 +- storage/myisam/ft_parser.c | 2 +- storage/myisam/ft_static.c | 13 - storage/myisam/ft_stopwords.c | 2 +- storage/myisam/ft_update.c | 4 +- storage/myisam/fulltext.h | 10 - storage/myisam/mi_check.c | 112 +- storage/myisam/mi_create.c | 8 +- storage/myisam/mi_delete.c | 12 +- storage/myisam/mi_delete_all.c | 1 - storage/myisam/mi_dynrec.c | 10 +- storage/myisam/mi_key.c | 2 +- storage/myisam/mi_log.c | 2 +- storage/myisam/mi_open.c | 15 +- storage/myisam/mi_packrec.c | 6 +- storage/myisam/mi_range.c | 2 +- storage/myisam/mi_search.c | 10 +- storage/myisam/mi_test1.c | 2 +- storage/myisam/mi_test2.c | 2 +- storage/myisam/mi_unique.c | 4 +- storage/myisam/mi_update.c | 2 +- storage/myisam/mi_write.c | 12 +- storage/myisam/myisamchk.c | 33 +- storage/myisam/myisamdef.h | 871 ++++--- storage/myisam/myisamlog.c | 2 +- storage/myisam/myisampack.c | 1 + storage/myisam/rt_index.c | 4 +- storage/myisam/sort.c | 7 +- 145 files changed, 44963 insertions(+), 917 deletions(-) create mode 100644 config/ac-macros/ha_maria.m4 create mode 100644 include/maria.h create mode 100644 include/myisamchk.h create mode 100644 mysql-test/include/have_maria.inc create mode 100644 mysql-test/r/have_maria.require create mode 100644 mysql-test/r/maria.result create mode 100644 mysql-test/r/ps_maria.result create mode 100644 mysql-test/t/maria.test create mode 100644 mysql-test/t/ps_maria.test create mode 100644 sql/ha_maria.cc create mode 100644 sql/ha_maria.h create mode 100644 storage/maria/Makefile.am create mode 100644 storage/maria/cmakelists.txt create mode 100644 storage/maria/ft_maria.c create mode 100644 storage/maria/ma_cache.c create mode 100644 storage/maria/ma_changed.c create mode 100644 storage/maria/ma_check.c create mode 100644 storage/maria/ma_checksum.c create mode 100644 storage/maria/ma_close.c create mode 100644 storage/maria/ma_create.c create mode 100644 storage/maria/ma_dbug.c create mode 100644 storage/maria/ma_delete.c create mode 100644 storage/maria/ma_delete_all.c create mode 100644 storage/maria/ma_delete_table.c create mode 100644 storage/maria/ma_dynrec.c create mode 100644 storage/maria/ma_extra.c create mode 100644 storage/maria/ma_ft_boolean_search.c create mode 100644 storage/maria/ma_ft_eval.c create mode 100644 storage/maria/ma_ft_eval.h create mode 100644 storage/maria/ma_ft_nlq_search.c create mode 100644 storage/maria/ma_ft_parser.c create mode 100644 storage/maria/ma_ft_stem.c create mode 100644 storage/maria/ma_ft_test1.c create mode 100644 storage/maria/ma_ft_test1.h create mode 100644 storage/maria/ma_ft_update.c create mode 100644 storage/maria/ma_ftdefs.h create mode 100644 storage/maria/ma_fulltext.h create mode 100644 storage/maria/ma_info.c create mode 100644 storage/maria/ma_init.c create mode 100644 storage/maria/ma_key.c create mode 100644 storage/maria/ma_keycache.c create mode 100644 storage/maria/ma_locking.c create mode 100644 storage/maria/ma_log.c create mode 100644 storage/maria/ma_open.c create mode 100644 storage/maria/ma_packrec.c create mode 100644 storage/maria/ma_page.c create mode 100644 storage/maria/ma_panic.c create mode 100644 storage/maria/ma_preload.c create mode 100644 storage/maria/ma_range.c create mode 100644 storage/maria/ma_rename.c create mode 100644 storage/maria/ma_rfirst.c create mode 100644 storage/maria/ma_rkey.c create mode 100644 storage/maria/ma_rlast.c create mode 100644 storage/maria/ma_rnext.c create mode 100644 storage/maria/ma_rnext_same.c create mode 100644 storage/maria/ma_rprev.c create mode 100644 storage/maria/ma_rrnd.c create mode 100644 storage/maria/ma_rsame.c create mode 100644 storage/maria/ma_rsamepos.c create mode 100644 storage/maria/ma_rt_index.c create mode 100644 storage/maria/ma_rt_index.h create mode 100644 storage/maria/ma_rt_key.c create mode 100644 storage/maria/ma_rt_key.h create mode 100644 storage/maria/ma_rt_mbr.c create mode 100644 storage/maria/ma_rt_mbr.h create mode 100644 storage/maria/ma_rt_split.c create mode 100644 storage/maria/ma_rt_test.c create mode 100644 storage/maria/ma_scan.c create mode 100644 storage/maria/ma_search.c create mode 100644 storage/maria/ma_sort.c create mode 100644 storage/maria/ma_sp_defs.h create mode 100644 storage/maria/ma_sp_key.c create mode 100644 storage/maria/ma_sp_test.c create mode 100644 storage/maria/ma_static.c create mode 100644 storage/maria/ma_statrec.c create mode 100644 storage/maria/ma_test1.c create mode 100644 storage/maria/ma_test2.c create mode 100644 storage/maria/ma_test3.c create mode 100755 storage/maria/ma_test_all.sh create mode 100644 storage/maria/ma_unique.c create mode 100644 storage/maria/ma_update.c create mode 100644 storage/maria/ma_write.c create mode 100644 storage/maria/maria_chk.c create mode 100644 storage/maria/maria_def.h create mode 100644 storage/maria/maria_ftdump.c create mode 100644 storage/maria/maria_log.c create mode 100644 storage/maria/maria_pack.c create mode 100755 storage/maria/maria_rename.sh create mode 100755 storage/maria/test_pack create mode 100644 storage/myisam/ft_myisam.c diff --git a/.bzrignore b/.bzrignore index 4b6b0411a47..2969dfab09c 100644 --- a/.bzrignore +++ b/.bzrignore @@ -1759,3 +1759,16 @@ vio/viotest.cpp zlib/*.ds? zlib/*.vcproj mysql-test/r/*.log +storage/maria/ma_test1 +storage/maria/ma_test2 +storage/maria/ma_test3 +storage/maria/maria_ftdump +storage/maria/ma_sp_test +storage/maria/ma_rt_test +storage/maria/maria_log +storage/maria/maria_pack +storage/maria/maria_chk +storage/maria/ma_test_all +storage/maria/maria.log +storage/maria/*.MAD +storage/maria/*.MAI diff --git a/BUILD/SETUP.sh b/BUILD/SETUP.sh index a6e18e26bfc..035f297be7e 100755 --- a/BUILD/SETUP.sh +++ b/BUILD/SETUP.sh @@ -84,7 +84,7 @@ debug_extra_cflags="-O1 -Wuninitialized" base_cxxflags="-felide-constructors -fno-exceptions -fno-rtti" amd64_cxxflags="" # If dropping '--with-big-tables', add here "-DBIG_TABLES" -base_configs="$prefix_configs --enable-assembler --with-extra-charsets=complex --enable-thread-safe-client --with-readline --with-big-tables" +base_configs="$prefix_configs --enable-assembler --with-extra-charsets=complex --enable-thread-safe-client --with-readline --with-big-tables --with-maria-storage-engine" static_link="--with-mysqld-ldflags=-all-static --with-client-ldflags=-all-static" amd64_configs="" alpha_configs="" # Not used yet diff --git a/BitKeeper/triggers/post-commit b/BitKeeper/triggers/post-commit index ce5e03b0ac9..d6d678a7c22 100755 --- a/BitKeeper/triggers/post-commit +++ b/BitKeeper/triggers/post-commit @@ -5,7 +5,7 @@ FROM=$USER@mysql.com COMMITS=commits@lists.mysql.com DOCS=docs-commit@mysql.com LIMIT=10000 -VERSION="5.1" +VERSION="maria" if [ "$REAL_EMAIL" = "" ] then @@ -66,51 +66,6 @@ EOF bk cset -r+ -d ) | $SENDMAIL -t -#++ -# commits@ mail -#-- - echo "Notifying commits list at $COMMITS" - ( - cat < -From: $FROM -To: $COMMITS -Subject: bk commit into $VERSION tree ($CHANGESET)$BS -X-CSetKey: <$CSETKEY> -$BH -Below is the list of changes that have just been committed into a local -$VERSION repository of $USER. When $USER does a push these changes will -be propagated to the main repository and, within 24 hours after the -push, to the public repository. -For information on how to access the public repository -see http://dev.mysql.com/doc/mysql/en/installing-source-tree.html - -EOF - bk changes -v -r+ - bk cset -r+ -d - ) | head -n $LIMIT | $SENDMAIL -t - -#++ -# docs-commit@ mail -# Picks up anything under the Docs subdirectory (relevant for docs team). -#-- - bk changes -v -r+ | grep -q " Docs/" - if [ $? -eq 0 ] - then - echo "Notifying docs list at $DOCS" - ( - cat < -From: $FROM -To: $DOCS -Subject: bk commit - $VERSION tree (Manual) ($CHANGESET)$BS - -EOF - bk changes -v -r+ - bk cset -r+ -d - ) | $SENDMAIL -t - fi - else echo "commit failed because '$BK_STATUS', you may need to re-clone..." fi diff --git a/config/ac-macros/ha_maria.m4 b/config/ac-macros/ha_maria.m4 new file mode 100644 index 00000000000..56ec8720a09 --- /dev/null +++ b/config/ac-macros/ha_maria.m4 @@ -0,0 +1,29 @@ +dnl --------------------------------------------------------------------------- +dnl Macro: MYSQL_CHECK_MARIA +dnl Sets HAVE_MARIA_DB if --with-maria-storage-engine is used +dnl --------------------------------------------------------------------------- +AC_DEFUN([MYSQL_CHECK_MARIA], [ + AC_ARG_WITH([maria-storage-engine], + [ + --with-maria-storage-engine + Enable the Maria Storage Engine], + [mariadb="$withval"], + [mariadb=no]) + AC_MSG_CHECKING([for Maria storage engine]) + + case "$mariadb" in + yes ) + AC_DEFINE([HAVE_MARIA_DB], [1], [Builds Maria Storage Engine]) + AC_MSG_RESULT([yes]) + [mariadb=yes] + ;; + * ) + AC_MSG_RESULT([no]) + [mariadb=no] + ;; + esac + +]) +dnl --------------------------------------------------------------------------- +dnl END OF MYSQL_CHECK_MARIA SECTION +dnl --------------------------------------------------------------------------- diff --git a/configure.in b/configure.in index 0253a3a6771..03c68d65137 100644 --- a/configure.in +++ b/configure.in @@ -2483,6 +2483,10 @@ MYSQL_STORAGE_ENGINE(ndbcluster,,ndbcluster,,,,storage/ndb,,,[ MYSQL_SETUP_NDBCLUSTER ]) MYSQL_STORAGE_ENGINE(partition,,partition) +MYSQL_STORAGE_ENGINE(maria,,,,,,storage/maria,, + \$(top_builddir)/storage/maria/libmaria.a, [ + AC_CONFIG_FILES(storage/maria/Makefile) +]) # If we have threads generate some library functions and test programs sql_server_dirs= diff --git a/include/ft_global.h b/include/ft_global.h index 8f02e48f61d..494e89af8aa 100644 --- a/include/ft_global.h +++ b/include/ft_global.h @@ -66,6 +66,17 @@ void ft_free_stopwords(void); FT_INFO *ft_init_search(uint,void *, uint, byte *, uint,CHARSET_INFO *, byte *); my_bool ft_boolean_check_syntax_string(const byte *); +/* Internal symbols for fulltext between maria and MyISAM */ + +#define HA_FT_WTYPE HA_KEYTYPE_FLOAT +#define HA_FT_WLEN 4 +#define FT_SEGS 2 + +#define ft_sintXkorr(A) mi_sint4korr(A) +#define ft_intXstore(T,A) mi_int4store(T,A) + +extern const HA_KEYSEG ft_keysegs[FT_SEGS]; + #ifdef __cplusplus } #endif diff --git a/include/keycache.h b/include/keycache.h index 9fe1cce5da5..ba5ef1c3560 100644 --- a/include/keycache.h +++ b/include/keycache.h @@ -128,7 +128,8 @@ extern void end_key_cache(KEY_CACHE *keycache, my_bool cleanup); /* Functions to handle multiple key caches */ extern my_bool multi_keycache_init(void); extern void multi_keycache_free(void); -extern KEY_CACHE *multi_key_cache_search(byte *key, uint length); +extern KEY_CACHE *multi_key_cache_search(byte *key, uint length, + KEY_CACHE *def); extern my_bool multi_key_cache_set(const byte *key, uint length, KEY_CACHE *key_cache); extern void multi_key_cache_change(KEY_CACHE *old_data, diff --git a/include/maria.h b/include/maria.h new file mode 100644 index 00000000000..c7586143fe7 --- /dev/null +++ b/include/maria.h @@ -0,0 +1,418 @@ +/* Copyright (C) 2006 MySQL AB + + 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; either version 2 of the License, or + (at your option) any later version. + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ + +/* This file should be included when using maria_funktions */ + +#ifndef _maria_h +#define _maria_h +#ifdef __cplusplus +extern "C" { +#endif +#ifndef _my_base_h +#include +#endif +#ifndef _m_ctype_h +#include +#endif +#ifndef _keycache_h +#include "keycache.h" +#endif +#include "my_handler.h" +#include "ft_global.h" +#include +#include + +/* + Limit max keys according to HA_MAX_POSSIBLE_KEY; See myisamchk.h for details +*/ + +#if MAX_INDEXES > HA_MAX_POSSIBLE_KEY +#define MARIA_MAX_KEY HA_MAX_POSSIBLE_KEY /* Max allowed keys */ +#else +#define MARIA_MAX_KEY MAX_INDEXES /* Max allowed keys */ +#endif + +#define MARIA_MAX_MSG_BUF 1024 /* used in CHECK TABLE, REPAIR TABLE */ +#define MARIA_NAME_IEXT ".MAI" +#define MARIA_NAME_DEXT ".MAD" +/* Max extra space to use when sorting keys */ +#define MARIA_MAX_TEMP_LENGTH 2*1024L*1024L*1024L +/* Possible values for maria_block_size (must be power of 2) */ +#define MARIA_KEY_BLOCK_LENGTH 8192 /* default key block length */ +#define MARIA_MIN_KEY_BLOCK_LENGTH 1024 /* Min key block length */ +#define MARIA_MAX_KEY_BLOCK_LENGTH 32768 +#define maria_portable_sizeof_char_ptr 8 + +/* + In the following macros '_keyno_' is 0 .. keys-1. + If there can be more keys than bits in the key_map, the highest bit + is for all upper keys. They cannot be switched individually. + This means that clearing of high keys is ignored, setting one high key + sets all high keys. +*/ +#define MARIA_KEYMAP_BITS (8 * SIZEOF_LONG_LONG) +#define MARIA_KEYMAP_HIGH_MASK (ULL(1) << (MARIA_KEYMAP_BITS - 1)) +#define maria_get_mask_all_keys_active(_keys_) \ + (((_keys_) < MARIA_KEYMAP_BITS) ? \ + ((ULL(1) << (_keys_)) - ULL(1)) : \ + (~ ULL(0))) +#if MARIA_MAX_KEY > MARIA_KEYMAP_BITS +#define maria_is_key_active(_keymap_,_keyno_) \ + (((_keyno_) < MARIA_KEYMAP_BITS) ? \ + test((_keymap_) & (ULL(1) << (_keyno_))) : \ + test((_keymap_) & MARIA_KEYMAP_HIGH_MASK)) +#define maria_set_key_active(_keymap_,_keyno_) \ + (_keymap_)|= (((_keyno_) < MARIA_KEYMAP_BITS) ? \ + (ULL(1) << (_keyno_)) : \ + MARIA_KEYMAP_HIGH_MASK) +#define maria_clear_key_active(_keymap_,_keyno_) \ + (_keymap_)&= (((_keyno_) < MARIA_KEYMAP_BITS) ? \ + (~ (ULL(1) << (_keyno_))) : \ + (~ (ULL(0))) /*ignore*/ ) +#else +#define maria_is_key_active(_keymap_,_keyno_) \ + test((_keymap_) & (ULL(1) << (_keyno_))) +#define maria_set_key_active(_keymap_,_keyno_) \ + (_keymap_)|= (ULL(1) << (_keyno_)) +#define maria_clear_key_active(_keymap_,_keyno_) \ + (_keymap_)&= (~ (ULL(1) << (_keyno_))) +#endif +#define maria_is_any_key_active(_keymap_) \ + test((_keymap_)) +#define maria_is_all_keys_active(_keymap_,_keys_) \ + ((_keymap_) == maria_get_mask_all_keys_active(_keys_)) +#define maria_set_all_keys_active(_keymap_,_keys_) \ + (_keymap_)= maria_get_mask_all_keys_active(_keys_) +#define maria_clear_all_keys_active(_keymap_) \ + (_keymap_)= 0 +#define maria_intersect_keys_active(_to_,_from_) \ + (_to_)&= (_from_) +#define maria_is_any_intersect_keys_active(_keymap1_,_keys_,_keymap2_) \ + ((_keymap1_) & (_keymap2_) & \ + maria_get_mask_all_keys_active(_keys_)) +#define maria_copy_keys_active(_to_,_maxkeys_,_from_) \ + (_to_)= (maria_get_mask_all_keys_active(_maxkeys_) & \ + (_from_)) + + /* Param to/from maria_info */ + +typedef struct st_maria_isaminfo /* Struct from h_info */ +{ + ha_rows records; /* Records in database */ + ha_rows deleted; /* Deleted records in database */ + my_off_t recpos; /* Pos for last used record */ + my_off_t newrecpos; /* Pos if we write new record */ + my_off_t dupp_key_pos; /* Position to record with dup key */ + my_off_t data_file_length; /* Length of data file */ + my_off_t max_data_file_length, index_file_length; + my_off_t max_index_file_length, delete_length; + ulong reclength; /* Recordlength */ + ulong mean_reclength; /* Mean recordlength (if packed) */ + ulonglong auto_increment; + ulonglong key_map; /* Which keys are used */ + char *data_file_name, *index_file_name; + uint keys; /* Number of keys in use */ + uint options; /* HA_OPTION_... used */ + int errkey, /* With key was dupplicated on err */ + sortkey; /* clustered by this key */ + File filenr; /* (uniq) filenr for datafile */ + time_t create_time; /* When table was created */ + time_t check_time; + time_t update_time; + uint reflength; + ulong record_offset; + ulong *rec_per_key; /* for sql optimizing */ +} MARIA_INFO; + + +typedef struct st_maria_create_info +{ + const char *index_file_name, *data_file_name; /* If using symlinks */ + ha_rows max_rows; + ha_rows reloc_rows; + ulonglong auto_increment; + ulonglong data_file_length; + ulonglong key_file_length; + uint old_options; + uint8 language; + my_bool with_auto_increment; +} MARIA_CREATE_INFO; + +struct st_maria_info; /* For referense */ +struct st_maria_share; +typedef struct st_maria_info MARIA_HA; +struct st_maria_s_param; + +typedef struct st_maria_keydef /* Key definition with open & info */ +{ + struct st_maria_share *share; /* Pointer to base (set in open) */ + uint16 keysegs; /* Number of key-segment */ + uint16 flag; /* NOSAME, PACK_USED */ + + uint8 key_alg; /* BTREE, RTREE */ + uint16 block_length; /* Length of keyblock (auto) */ + uint16 underflow_block_length; /* When to execute underflow */ + uint16 keylength; /* Tot length of keyparts (auto) */ + uint16 minlength; /* min length of (packed) key (auto) */ + uint16 maxlength; /* max length of (packed) key (auto) */ + uint16 block_size; /* block_size (auto) */ + uint32 version; /* For concurrent read/write */ + uint32 ftparser_nr; /* distinct ftparser number */ + + HA_KEYSEG *seg, *end; + struct st_mysql_ftparser *parser; /* Fulltext [pre]parser */ + int(*bin_search) (struct st_maria_info *info, + struct st_maria_keydef *keyinfo, uchar *page, uchar *key, + uint key_len, uint comp_flag, uchar **ret_pos, + uchar *buff, my_bool *was_last_key); + uint(*get_key) (struct st_maria_keydef *keyinfo, uint nod_flag, + uchar **page, uchar *key); + int(*pack_key) (struct st_maria_keydef *keyinfo, uint nod_flag, + uchar *next_key, uchar *org_key, uchar *prev_key, + uchar *key, struct st_maria_s_param *s_temp); + void(*store_key) (struct st_maria_keydef *keyinfo, uchar *key_pos, + struct st_maria_s_param *s_temp); + int(*ck_insert) (struct st_maria_info *inf, uint k_nr, uchar *k, uint klen); + int(*ck_delete) (struct st_maria_info *inf, uint k_nr, uchar *k, uint klen); +} MARIA_KEYDEF; + + +#define MARIA_UNIQUE_HASH_LENGTH 4 + +typedef struct st_maria_unique_def /* Segment definition of unique */ +{ + uint16 keysegs; /* Number of key-segment */ + uchar key; /* Mapped to which key */ + uint8 null_are_equal; + HA_KEYSEG *seg, *end; +} MARIA_UNIQUEDEF; + +typedef struct st_maria_decode_tree /* Decode huff-table */ +{ + uint16 *table; + uint quick_table_bits; + byte *intervalls; +} MARIA_DECODE_TREE; + + +struct st_maria_bit_buff; + +/* + Note that null markers should always be first in a row ! + When creating a column, one should only specify: + type, length, null_bit and null_pos +*/ + +typedef struct st_maria_columndef /* column information */ +{ + int16 type; /* en_fieldtype */ + uint16 length; /* length of field */ + uint32 offset; /* Offset to position in row */ + uint8 null_bit; /* If column may be 0 */ + uint16 null_pos; /* position for null marker */ + +#ifndef NOT_PACKED_DATABASES + void(*unpack) (struct st_maria_columndef *rec, + struct st_maria_bit_buff *buff, + uchar *start, uchar *end); + enum en_fieldtype base_type; + uint space_length_bits, pack_type; + MARIA_DECODE_TREE *huff_tree; +#endif +} MARIA_COLUMNDEF; + + +extern my_string maria_log_filename; /* Name of logfile */ +extern ulong maria_block_size; +extern ulong maria_concurrent_insert; +extern my_bool maria_flush, maria_delay_key_write, maria_single_user; +extern my_off_t maria_max_temp_length; +extern ulong maria_bulk_insert_tree_size, maria_data_pointer_size; +extern KEY_CACHE maria_key_cache_var, *maria_key_cache; + + + /* Prototypes for maria-functions */ + +extern int maria_init(void); +extern void maria_end(void); +extern int maria_close(struct st_maria_info *file); +extern int maria_delete(struct st_maria_info *file, const byte *buff); +extern struct st_maria_info *maria_open(const char *name, int mode, + uint wait_if_locked); +extern int maria_panic(enum ha_panic_function function); +extern int maria_rfirst(struct st_maria_info *file, byte *buf, int inx); +extern int maria_rkey(struct st_maria_info *file, byte *buf, int inx, + const byte *key, + uint key_len, enum ha_rkey_function search_flag); +extern int maria_rlast(struct st_maria_info *file, byte *buf, int inx); +extern int maria_rnext(struct st_maria_info *file, byte *buf, int inx); +extern int maria_rnext_same(struct st_maria_info *info, byte *buf); +extern int maria_rprev(struct st_maria_info *file, byte *buf, int inx); +extern int maria_rrnd(struct st_maria_info *file, byte *buf, my_off_t pos); +extern int maria_scan_init(struct st_maria_info *file); +extern int maria_scan(struct st_maria_info *file, byte *buf); +extern int maria_rsame(struct st_maria_info *file, byte *record, int inx); +extern int maria_rsame_with_pos(struct st_maria_info *file, byte *record, + int inx, my_off_t pos); +extern int maria_update(struct st_maria_info *file, const byte *old, + byte *new_record); +extern int maria_write(struct st_maria_info *file, byte *buff); +extern my_off_t maria_position(struct st_maria_info *file); +extern int maria_status(struct st_maria_info *info, MARIA_INFO *x, uint flag); +extern int maria_lock_database(struct st_maria_info *file, int lock_type); +extern int maria_create(const char *name, uint keys, MARIA_KEYDEF *keydef, + uint columns, MARIA_COLUMNDEF *columndef, + uint uniques, MARIA_UNIQUEDEF *uniquedef, + MARIA_CREATE_INFO *create_info, uint flags); +extern int maria_delete_table(const char *name); +extern int maria_rename(const char *from, const char *to); +extern int maria_extra(struct st_maria_info *file, + enum ha_extra_function function, void *extra_arg); +extern ha_rows maria_records_in_range(struct st_maria_info *info, int inx, + key_range *min_key, key_range *max_key); +extern int maria_logging(int activate_log); +extern int maria_is_changed(struct st_maria_info *info); +extern int maria_delete_all_rows(struct st_maria_info *info); +extern uint maria_get_pointer_length(ulonglong file_length, uint def); + + +/* this is used to pass to mysql_mariachk_table */ + +#define MARIA_CHK_REPAIR 1 /* equivalent to mariachk -r */ +#define MARIA_CHK_VERIFY 2 /* Verify, run repair if failure */ + +typedef struct st_maria_sort_info +{ +#ifdef THREAD + /* sync things */ + pthread_mutex_t mutex; + pthread_cond_t cond; +#endif + MARIA_HA *info; + HA_CHECK *param; + char *buff; + SORT_KEY_BLOCKS *key_block, *key_block_end; + SORT_FT_BUF *ft_buf; + + my_off_t filelength, dupp, buff_length; + ha_rows max_records; + uint current_key, total_keys; + uint got_error, threads_running; + myf myf_rw; + enum data_file_type new_data_file_type; +} MARIA_SORT_INFO; + + +typedef struct st_maria_sort_param +{ + pthread_t thr; + IO_CACHE read_cache, tempfile, tempfile_for_exceptions; + DYNAMIC_ARRAY buffpek; + + MARIA_KEYDEF *keyinfo; + MARIA_SORT_INFO *sort_info; + HA_KEYSEG *seg; + uchar **sort_keys; + byte *rec_buff; + void *wordlist, *wordptr; + char *record; + MY_TMPDIR *tmpdir; + + /* + The next two are used to collect statistics, see maria_update_key_parts for + description. + */ + ulonglong unique[HA_MAX_KEY_SEG+1]; + ulonglong notnull[HA_MAX_KEY_SEG+1]; + + my_off_t pos,max_pos,filepos,start_recpos; + uint key, key_length,real_key_length,sortbuff_size; + uint maxbuffers, keys, find_length, sort_keys_length; + my_bool fix_datafile, master; + + int (*key_cmp)(struct st_maria_sort_param *, const void *, const void *); + int (*key_read)(struct st_maria_sort_param *,void *); + int (*key_write)(struct st_maria_sort_param *, const void *); + void (*lock_in_memory)(HA_CHECK *); + NEAR int (*write_keys)(struct st_maria_sort_param *, register uchar **, + uint , struct st_buffpek *, IO_CACHE *); + NEAR uint (*read_to_buffer)(IO_CACHE *,struct st_buffpek *, uint); + NEAR int (*write_key)(struct st_maria_sort_param *, IO_CACHE *,char *, + uint, uint); +} MARIA_SORT_PARAM; + + +/* functions in maria_check */ +void mariachk_init(HA_CHECK *param); +int maria_chk_status(HA_CHECK *param, MARIA_HA *info); +int maria_chk_del(HA_CHECK *param, register MARIA_HA *info, uint test_flag); +int maria_chk_size(HA_CHECK *param, MARIA_HA *info); +int maria_chk_key(HA_CHECK *param, MARIA_HA *info); +int maria_chk_data_link(HA_CHECK *param, MARIA_HA *info, int extend); +int maria_repair(HA_CHECK *param, register MARIA_HA *info, + my_string name, int rep_quick); +int maria_sort_index(HA_CHECK *param, register MARIA_HA *info, + my_string name); +int maria_repair_by_sort(HA_CHECK *param, register MARIA_HA *info, + const char *name, int rep_quick); +int maria_repair_parallel(HA_CHECK *param, register MARIA_HA *info, + const char *name, int rep_quick); +int maria_change_to_newfile(const char *filename, const char *old_ext, + const char *new_ext, uint raid_chunks, myf myflags); +int maria_lock_file(HA_CHECK *param, File file, my_off_t start, int lock_type, + const char *filetype, const char *filename); +void maria_lock_memory(HA_CHECK *param); +int maria_update_state_info(HA_CHECK *param, MARIA_HA *info, uint update); +void maria_update_key_parts(MARIA_KEYDEF *keyinfo, ulong *rec_per_key_part, + ulonglong *unique, ulonglong *notnull, + ulonglong records); +int maria_filecopy(HA_CHECK *param, File to, File from, my_off_t start, + my_off_t length, const char *type); +int maria_movepoint(MARIA_HA *info, byte *record, my_off_t oldpos, + my_off_t newpos, uint prot_key); +int maria_write_data_suffix(MARIA_SORT_INFO *sort_info, my_bool fix_datafile); +int maria_test_if_almost_full(MARIA_HA *info); +int maria_recreate_table(HA_CHECK *param, MARIA_HA ** org_info, char *filename); +int maria_disable_indexes(MARIA_HA *info); +int maria_enable_indexes(MARIA_HA *info); +int maria_indexes_are_disabled(MARIA_HA *info); +void maria_disable_non_unique_index(MARIA_HA *info, ha_rows rows); +my_bool maria_test_if_sort_rep(MARIA_HA *info, ha_rows rows, ulonglong key_map, + my_bool force); + +int maria_init_bulk_insert(MARIA_HA *info, ulong cache_size, ha_rows rows); +void maria_flush_bulk_insert(MARIA_HA *info, uint inx); +void maria_end_bulk_insert(MARIA_HA *info); +int maria_assign_to_key_cache(MARIA_HA *info, ulonglong key_map, + KEY_CACHE *key_cache); +void maria_change_key_cache(KEY_CACHE *old_key_cache, + KEY_CACHE *new_key_cache); +int maria_preload(MARIA_HA *info, ulonglong key_map, my_bool ignore_leaves); + +/* fulltext functions */ +FT_INFO *maria_ft_init_search(uint,void *, uint, byte *, uint, + CHARSET_INFO *, byte *); + +/* 'Almost-internal' Maria functions */ + +void _ma_update_auto_increment_key(HA_CHECK *param, MARIA_HA *info, + my_bool repair); + + +#ifdef __cplusplus +} +#endif +#endif diff --git a/include/my_base.h b/include/my_base.h index b9a806cc02f..5378f4f470b 100644 --- a/include/my_base.h +++ b/include/my_base.h @@ -15,7 +15,6 @@ Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ /* This file includes constants used with all databases */ -/* Author: Michael Widenius */ #ifndef _my_base_h #define _my_base_h @@ -487,4 +486,7 @@ typedef ulong ha_rows; #define HA_VARCHAR_PACKLENGTH(field_length) ((field_length) < 256 ? 1 :2) +/* invalidator function reference for Query Cache */ +typedef void (* invalidator_by_filename)(const char * filename); + #endif /* _my_base_h */ diff --git a/include/my_handler.h b/include/my_handler.h index d531e0fb3e1..9b086036389 100644 --- a/include/my_handler.h +++ b/include/my_handler.h @@ -18,11 +18,30 @@ #ifndef _my_handler_h #define _my_handler_h -#include "my_global.h" -#include "my_base.h" -#include "m_ctype.h" #include "myisampack.h" +/* + There is a hard limit for the maximum number of keys as there are only + 8 bits in the index file header for the number of keys in a table. + This means that 0..255 keys can exist for a table. The idea of + HA_MAX_POSSIBLE_KEY is to ensure that one can use myisamchk & tools on + a MyISAM table for which one has more keys than MyISAM is normally + compiled for. If you don't have this, you will get a core dump when + running myisamchk compiled for 128 keys on a table with 255 keys. +*/ + +#define HA_MAX_POSSIBLE_KEY 255 /* For myisamchk */ +/* + The following defines can be increased if necessary. + But beware the dependency of HA_MAX_POSSIBLE_KEY_BUFF and HA_MAX_KEY_LENGTH. +*/ + +#define HA_MAX_KEY_LENGTH 1000 /* Max length in bytes */ +#define HA_MAX_KEY_SEG 16 /* Max segments for key */ + +#define HA_MAX_POSSIBLE_KEY_BUFF (HA_MAX_KEY_LENGTH + 24+ 6+6) +#define HA_MAX_KEY_BUFF (HA_MAX_KEY_LENGTH+HA_MAX_KEY_SEG*6+8+8) + typedef struct st_HA_KEYSEG /* Key-portion */ { CHARSET_INFO *charset; @@ -82,7 +101,7 @@ typedef struct st_HA_KEYSEG /* Key-portion */ #define clr_rec_bits(bit_ptr, bit_ofs, bit_len) \ set_rec_bits(0, bit_ptr, bit_ofs, bit_len) -extern int mi_compare_text(CHARSET_INFO *, uchar *, uint, uchar *, uint , +extern int ha_compare_text(CHARSET_INFO *, uchar *, uint, uchar *, uint , my_bool, my_bool); extern int ha_key_cmp(register HA_KEYSEG *keyseg, register uchar *a, register uchar *b, uint key_length, uint nextflag, diff --git a/include/myisam.h b/include/myisam.h index 5116f48eeb5..b5f29741be0 100644 --- a/include/myisam.h +++ b/include/myisam.h @@ -32,33 +32,19 @@ extern "C" { #include "keycache.h" #endif #include "my_handler.h" +#include #include /* - There is a hard limit for the maximum number of keys as there are only - 8 bits in the index file header for the number of keys in a table. - This means that 0..255 keys can exist for a table. The idea of - MI_MAX_POSSIBLE_KEY is to ensure that one can use myisamchk & tools on - a MyISAM table for which one has more keys than MyISAM is normally - compiled for. If you don't have this, you will get a core dump when - running myisamchk compiled for 128 keys on a table with 255 keys. + Limit max keys according to HA_MAX_POSSIBLE_KEY; See myisamchk.h for details */ -#define MI_MAX_POSSIBLE_KEY 255 /* For myisam_chk */ -#if MAX_INDEXES > MI_MAX_POSSIBLE_KEY -#define MI_MAX_KEY MI_MAX_POSSIBLE_KEY /* Max allowed keys */ + +#if MAX_INDEXES > HA_MAX_POSSIBLE_KEY +#define MI_MAX_KEY HA_MAX_POSSIBLE_KEY /* Max allowed keys */ #else #define MI_MAX_KEY MAX_INDEXES /* Max allowed keys */ #endif -#define MI_MAX_POSSIBLE_KEY_BUFF (1024+6+6) /* For myisam_chk */ -/* - The following defines can be increased if necessary. - But beware the dependency of MI_MAX_POSSIBLE_KEY_BUFF and MI_MAX_KEY_LENGTH. -*/ -#define MI_MAX_KEY_LENGTH 1000 /* Max length in bytes */ -#define MI_MAX_KEY_SEG 16 /* Max segments for key */ - -#define MI_MAX_KEY_BUFF (MI_MAX_KEY_LENGTH+MI_MAX_KEY_SEG*6+8+8) #define MI_MAX_MSG_BUF 1024 /* used in CHECK TABLE, REPAIR TABLE */ #define MI_NAME_IEXT ".MYI" #define MI_NAME_DEXT ".MYD" @@ -257,9 +243,6 @@ typedef struct st_columndef /* column information */ #endif } MI_COLUMNDEF; -/* invalidator function reference for Query Cache */ -typedef void (* invalidator_by_filename)(const char * filename); - extern my_string myisam_log_filename; /* Name of logfile */ extern ulong myisam_block_size; extern ulong myisam_concurrent_insert; @@ -311,195 +294,105 @@ extern int mi_delete_all_rows(struct st_myisam_info *info); extern ulong _mi_calc_blob_length(uint length , const byte *pos); extern uint mi_get_pointer_length(ulonglong file_length, uint def); -/* this is used to pass to mysql_myisamchk_table -- by Sasha Pachev */ +/* this is used to pass to mysql_myisamchk_table */ #define MYISAMCHK_REPAIR 1 /* equivalent to myisamchk -r */ #define MYISAMCHK_VERIFY 2 /* Verify, run repair if failure */ -/* - Definitions needed for myisamchk.c - - Entries marked as "QQ to be removed" are NOT used to - pass check/repair options to mi_check.c. They are used - internally by myisamchk.c or/and ha_myisam.cc and should NOT - be stored together with other flags. They should be removed - from the following list to make addition of new flags possible. -*/ - -#define T_AUTO_INC 1 -#define T_AUTO_REPAIR 2 /* QQ to be removed */ -#define T_BACKUP_DATA 4 -#define T_CALC_CHECKSUM 8 -#define T_CHECK 16 /* QQ to be removed */ -#define T_CHECK_ONLY_CHANGED 32 /* QQ to be removed */ -#define T_CREATE_MISSING_KEYS 64 -#define T_DESCRIPT 128 -#define T_DONT_CHECK_CHECKSUM 256 -#define T_EXTEND 512 -#define T_FAST (1L << 10) /* QQ to be removed */ -#define T_FORCE_CREATE (1L << 11) /* QQ to be removed */ -#define T_FORCE_UNIQUENESS (1L << 12) -#define T_INFO (1L << 13) -#define T_MEDIUM (1L << 14) -#define T_QUICK (1L << 15) /* QQ to be removed */ -#define T_READONLY (1L << 16) /* QQ to be removed */ -#define T_REP (1L << 17) -#define T_REP_BY_SORT (1L << 18) /* QQ to be removed */ -#define T_REP_PARALLEL (1L << 19) /* QQ to be removed */ -#define T_RETRY_WITHOUT_QUICK (1L << 20) -#define T_SAFE_REPAIR (1L << 21) -#define T_SILENT (1L << 22) -#define T_SORT_INDEX (1L << 23) /* QQ to be removed */ -#define T_SORT_RECORDS (1L << 24) /* QQ to be removed */ -#define T_STATISTICS (1L << 25) -#define T_UNPACK (1L << 26) -#define T_UPDATE_STATE (1L << 27) -#define T_VERBOSE (1L << 28) -#define T_VERY_SILENT (1L << 29) -#define T_WAIT_FOREVER (1L << 30) -#define T_WRITE_LOOP ((ulong) 1L << 31) - -#define T_REP_ANY (T_REP | T_REP_BY_SORT | T_REP_PARALLEL) - -/* - Flags used by myisamchk.c or/and ha_myisam.cc that are NOT passed - to mi_check.c follows: -*/ - -#define TT_USEFRM 1 -#define TT_FOR_UPGRADE 2 - -#define O_NEW_INDEX 1 /* Bits set in out_flag */ -#define O_NEW_DATA 2 -#define O_DATA_LOST 4 - -/* these struct is used by my_check to tell it what to do */ - -typedef struct st_sort_key_blocks /* Used when sorting */ +typedef struct st_sort_info { - uchar *buff,*end_pos; - uchar lastkey[MI_MAX_POSSIBLE_KEY_BUFF]; - uint last_length; - int inited; -} SORT_KEY_BLOCKS; +#ifdef THREAD + /* sync things */ + pthread_mutex_t mutex; + pthread_cond_t cond; +#endif + MI_INFO *info; + HA_CHECK *param; + char *buff; + SORT_KEY_BLOCKS *key_block, *key_block_end; + SORT_FT_BUF *ft_buf; + my_off_t filelength, dupp, buff_length; + ha_rows max_records; + uint current_key, total_keys; + uint got_error, threads_running; + myf myf_rw; + enum data_file_type new_data_file_type; +} MI_SORT_INFO; -/* - MyISAM supports several statistics collection methods. Currently statistics - collection method is not stored in MyISAM file and has to be specified for - each table analyze/repair operation in MI_CHECK::stats_method. -*/ -typedef enum -{ - /* Treat NULLs as inequal when collecting statistics (default for 4.1/5.0) */ - MI_STATS_METHOD_NULLS_NOT_EQUAL, - /* Treat NULLs as equal when collecting statistics (like 4.0 did) */ - MI_STATS_METHOD_NULLS_EQUAL, - /* Ignore NULLs - count only tuples without NULLs in the index components */ - MI_STATS_METHOD_IGNORE_NULLS -} enum_mi_stats_method; - -typedef struct st_mi_check_param +typedef struct st_mi_sort_param { - ulonglong auto_increment_value; - ulonglong max_data_file_length; - ulonglong keys_in_use; - ulonglong max_record_length; - my_off_t search_after_block; - my_off_t new_file_pos,key_file_blocks; - my_off_t keydata,totaldata,key_blocks,start_check_pos; - ha_rows total_records,total_deleted; - ha_checksum record_checksum,glob_crc; - ulong use_buffers,read_buffer_length,write_buffer_length, - sort_buffer_length,sort_key_blocks; - uint out_flag,warning_printed,error_printed,verbose; - uint opt_sort_key,total_files,max_level; - uint testflag, key_cache_block_size; - uint8 language; - my_bool using_global_keycache, opt_lock_memory, opt_follow_links; - my_bool retry_repair, force_sort, calc_checksum; - char temp_filename[FN_REFLEN],*isam_file_name; - MY_TMPDIR *tmpdir; - int tmpfile_createflag; - myf myf_rw; - IO_CACHE read_cache; + pthread_t thr; + IO_CACHE read_cache, tempfile, tempfile_for_exceptions; + DYNAMIC_ARRAY buffpek; + MI_KEYDEF *keyinfo; + MI_SORT_INFO *sort_info; + HA_KEYSEG *seg; + uchar **sort_keys; + byte *rec_buff; + void *wordlist, *wordptr; + char *record; + MY_TMPDIR *tmpdir; + /* The next two are used to collect statistics, see update_key_parts for description. */ - ulonglong unique_count[MI_MAX_KEY_SEG+1]; - ulonglong notnull_count[MI_MAX_KEY_SEG+1]; - - ha_checksum key_crc[MI_MAX_POSSIBLE_KEY]; - ulong rec_per_key_part[MI_MAX_KEY_SEG*MI_MAX_POSSIBLE_KEY]; - void *thd; - const char *db_name, *table_name; - const char *op_name; - enum_mi_stats_method stats_method; -} MI_CHECK; - -typedef struct st_sort_ft_buf -{ - uchar *buf, *end; - int count; - uchar lastkey[MI_MAX_KEY_BUFF]; -} SORT_FT_BUF; + ulonglong unique[HA_MAX_KEY_SEG+1]; + ulonglong notnull[HA_MAX_KEY_SEG+1]; + + my_off_t pos,max_pos,filepos,start_recpos; + uint key, key_length,real_key_length,sortbuff_size; + uint maxbuffers, keys, find_length, sort_keys_length; + my_bool fix_datafile, master; + + int (*key_cmp)(struct st_mi_sort_param *, const void *, const void *); + int (*key_read)(struct st_mi_sort_param *,void *); + int (*key_write)(struct st_mi_sort_param *, const void *); + void (*lock_in_memory)(HA_CHECK *); + NEAR int (*write_keys)(struct st_mi_sort_param *, register uchar **, + uint , struct st_buffpek *, IO_CACHE *); + NEAR uint (*read_to_buffer)(IO_CACHE *,struct st_buffpek *, uint); + NEAR int (*write_key)(struct st_mi_sort_param *, IO_CACHE *,char *, + uint, uint); +} MI_SORT_PARAM; -typedef struct st_sort_info -{ - my_off_t filelength,dupp,buff_length; - ha_rows max_records; - uint current_key, total_keys; - myf myf_rw; - enum data_file_type new_data_file_type; - MI_INFO *info; - MI_CHECK *param; - char *buff; - SORT_KEY_BLOCKS *key_block,*key_block_end; - SORT_FT_BUF *ft_buf; - /* sync things */ - uint got_error, threads_running; -#ifdef THREAD - pthread_mutex_t mutex; - pthread_cond_t cond; -#endif -} SORT_INFO; /* functions in mi_check */ -void myisamchk_init(MI_CHECK *param); -int chk_status(MI_CHECK *param, MI_INFO *info); -int chk_del(MI_CHECK *param, register MI_INFO *info, uint test_flag); -int chk_size(MI_CHECK *param, MI_INFO *info); -int chk_key(MI_CHECK *param, MI_INFO *info); -int chk_data_link(MI_CHECK *param, MI_INFO *info,int extend); -int mi_repair(MI_CHECK *param, register MI_INFO *info, +void myisamchk_init(HA_CHECK *param); +int chk_status(HA_CHECK *param, MI_INFO *info); +int chk_del(HA_CHECK *param, register MI_INFO *info, uint test_flag); +int chk_size(HA_CHECK *param, MI_INFO *info); +int chk_key(HA_CHECK *param, MI_INFO *info); +int chk_data_link(HA_CHECK *param, MI_INFO *info,int extend); +int mi_repair(HA_CHECK *param, register MI_INFO *info, my_string name, int rep_quick); -int mi_sort_index(MI_CHECK *param, register MI_INFO *info, my_string name); -int mi_repair_by_sort(MI_CHECK *param, register MI_INFO *info, +int mi_sort_index(HA_CHECK *param, register MI_INFO *info, my_string name); +int mi_repair_by_sort(HA_CHECK *param, register MI_INFO *info, const char * name, int rep_quick); -int mi_repair_parallel(MI_CHECK *param, register MI_INFO *info, +int mi_repair_parallel(HA_CHECK *param, register MI_INFO *info, const char * name, int rep_quick); int change_to_newfile(const char * filename, const char * old_ext, const char * new_ext, uint raid_chunks, myf myflags); -int lock_file(MI_CHECK *param, File file, my_off_t start, int lock_type, +int lock_file(HA_CHECK *param, File file, my_off_t start, int lock_type, const char *filetype, const char *filename); -void lock_memory(MI_CHECK *param); -void update_auto_increment_key(MI_CHECK *param, MI_INFO *info, +void lock_memory(HA_CHECK *param); +void update_auto_increment_key(HA_CHECK *param, MI_INFO *info, my_bool repair); -int update_state_info(MI_CHECK *param, MI_INFO *info,uint update); +int update_state_info(HA_CHECK *param, MI_INFO *info,uint update); void update_key_parts(MI_KEYDEF *keyinfo, ulong *rec_per_key_part, ulonglong *unique, ulonglong *notnull, ulonglong records); -int filecopy(MI_CHECK *param, File to,File from,my_off_t start, +int filecopy(HA_CHECK *param, File to,File from,my_off_t start, my_off_t length, const char *type); int movepoint(MI_INFO *info,byte *record,my_off_t oldpos, my_off_t newpos, uint prot_key); -int write_data_suffix(SORT_INFO *sort_info, my_bool fix_datafile); +int write_data_suffix(MI_SORT_INFO *sort_info, my_bool fix_datafile); int test_if_almost_full(MI_INFO *info); -int recreate_table(MI_CHECK *param, MI_INFO **org_info, char *filename); +int recreate_table(HA_CHECK *param, MI_INFO **org_info, char *filename); void mi_disable_non_unique_index(MI_INFO *info, ha_rows rows); my_bool mi_test_if_sort_rep(MI_INFO *info, ha_rows rows, ulonglong key_map, my_bool force); @@ -513,6 +406,13 @@ void mi_change_key_cache(KEY_CACHE *old_key_cache, KEY_CACHE *new_key_cache); int mi_preload(MI_INFO *info, ulonglong key_map, my_bool ignore_leaves); +int write_data_suffix(MI_SORT_INFO *sort_info, my_bool fix_datafile); +int flush_pending_blocks(MI_SORT_PARAM *param); +int sort_ft_buf_flush(MI_SORT_PARAM *sort_param); +int thr_write_keys(MI_SORT_PARAM *sort_param); +int sort_write_record(MI_SORT_PARAM *sort_param); +int _create_index_by_sort(MI_SORT_PARAM *info,my_bool no_messages, ulong); + #ifdef __cplusplus } #endif diff --git a/include/myisamchk.h b/include/myisamchk.h new file mode 100644 index 00000000000..442fe5f7ece --- /dev/null +++ b/include/myisamchk.h @@ -0,0 +1,160 @@ +/* Copyright (C) 2006 MySQL AB + + 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; either version 2 of the License, or + (at your option) any later version. + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ + +/* Definitions needed for myisamchk/mariachk.c */ + +/* + Entries marked as "QQ to be removed" are NOT used to + pass check/repair options to xxx_check.c. They are used + internally by xxxchk.c or/and ha_xxxx.cc and should NOT + be stored together with other flags. They should be removed + from the following list to make addition of new flags possible. +*/ + +#ifndef _myisamchk_h +#define _myisamchk_h + +#define T_AUTO_INC 1 +#define T_AUTO_REPAIR 2 /* QQ to be removed */ +#define T_BACKUP_DATA 4 +#define T_CALC_CHECKSUM 8 +#define T_CHECK 16 /* QQ to be removed */ +#define T_CHECK_ONLY_CHANGED 32 /* QQ to be removed */ +#define T_CREATE_MISSING_KEYS 64 +#define T_DESCRIPT 128 +#define T_DONT_CHECK_CHECKSUM 256 +#define T_EXTEND 512 +#define T_FAST (1L << 10) /* QQ to be removed */ +#define T_FORCE_CREATE (1L << 11) /* QQ to be removed */ +#define T_FORCE_UNIQUENESS (1L << 12) +#define T_INFO (1L << 13) +#define T_MEDIUM (1L << 14) +#define T_QUICK (1L << 15) /* QQ to be removed */ +#define T_READONLY (1L << 16) /* QQ to be removed */ +#define T_REP (1L << 17) +#define T_REP_BY_SORT (1L << 18) /* QQ to be removed */ +#define T_REP_PARALLEL (1L << 19) /* QQ to be removed */ +#define T_RETRY_WITHOUT_QUICK (1L << 20) +#define T_SAFE_REPAIR (1L << 21) +#define T_SILENT (1L << 22) +#define T_SORT_INDEX (1L << 23) /* QQ to be removed */ +#define T_SORT_RECORDS (1L << 24) /* QQ to be removed */ +#define T_STATISTICS (1L << 25) +#define T_UNPACK (1L << 26) +#define T_UPDATE_STATE (1L << 27) +#define T_VERBOSE (1L << 28) +#define T_VERY_SILENT (1L << 29) +#define T_WAIT_FOREVER (1L << 30) +#define T_WRITE_LOOP ((ulong) 1L << 31) + +#define T_REP_ANY (T_REP | T_REP_BY_SORT | T_REP_PARALLEL) + +/* + Flags used by xxxxchk.c or/and ha_xxxx.cc that are NOT passed + to xxxcheck.c follows: +*/ + +#define TT_USEFRM 1 +#define TT_FOR_UPGRADE 2 + +#define O_NEW_INDEX 1 /* Bits set in out_flag */ +#define O_NEW_DATA 2 +#define O_DATA_LOST 4 + +typedef struct st_sort_key_blocks /* Used when sorting */ +{ + uchar *buff, *end_pos; + uchar lastkey[HA_MAX_POSSIBLE_KEY_BUFF]; + uint last_length; + int inited; +} SORT_KEY_BLOCKS; + + +/* + MARIA/MYISAM supports several statistics collection + methods. Currently statistics collection method is not stored in + MARIA file and has to be specified for each table analyze/repair + operation in MI_CHECK::stats_method. +*/ + +typedef enum +{ + /* Treat NULLs as inequal when collecting statistics (default for 4.1/5.0) */ + MI_STATS_METHOD_NULLS_NOT_EQUAL, + /* Treat NULLs as equal when collecting statistics (like 4.0 did) */ + MI_STATS_METHOD_NULLS_EQUAL, + /* Ignore NULLs - count only tuples without NULLs in the index components */ + MI_STATS_METHOD_IGNORE_NULLS +} enum_handler_stats_method; + + +typedef struct st_handler_check_param +{ + char *isam_file_name; + MY_TMPDIR *tmpdir; + void *thd; + const char *db_name, *table_name, *op_name; + ulonglong auto_increment_value; + ulonglong max_data_file_length; + ulonglong keys_in_use; + ulonglong max_record_length; + /* + The next two are used to collect statistics, see update_key_parts for + description. + */ + ulonglong unique_count[HA_MAX_KEY_SEG + 1]; + ulonglong notnull_count[HA_MAX_KEY_SEG + 1]; + + my_off_t search_after_block; + my_off_t new_file_pos, key_file_blocks; + my_off_t keydata, totaldata, key_blocks, start_check_pos; + ha_rows total_records, total_deleted; + ha_checksum record_checksum, glob_crc; + ha_checksum key_crc[HA_MAX_POSSIBLE_KEY]; + ulong use_buffers, read_buffer_length, write_buffer_length; + ulong sort_buffer_length, sort_key_blocks; + ulong rec_per_key_part[HA_MAX_KEY_SEG * HA_MAX_POSSIBLE_KEY]; + uint out_flag, warning_printed, error_printed, verbose; + uint opt_sort_key, total_files, max_level; + uint testflag, key_cache_block_size; + int tmpfile_createflag; + myf myf_rw; + uint8 language; + my_bool using_global_keycache, opt_lock_memory, opt_follow_links; + my_bool retry_repair, force_sort, calc_checksum; + char temp_filename[FN_REFLEN]; + IO_CACHE read_cache; + enum_handler_stats_method stats_method; +} HA_CHECK; + + +typedef struct st_sort_ftbuf +{ + uchar *buf, *end; + int count; + uchar lastkey[HA_MAX_KEY_BUFF]; +} SORT_FT_BUF; + + +typedef struct st_buffpek { + my_off_t file_pos; /* Where we are in the sort file */ + uchar *base,*key; /* Key pointers */ + ha_rows count; /* Number of rows in table */ + ulong mem_count; /* numbers of keys in memory */ + ulong max_keys; /* Max keys in buffert */ +} BUFFPEK; + +#endif /* _myisamchk_h */ diff --git a/libmysqld/Makefile.am b/libmysqld/Makefile.am index b5d437918ae..e4430f23a4c 100644 --- a/libmysqld/Makefile.am +++ b/libmysqld/Makefile.am @@ -73,8 +73,8 @@ sqlsources = derror.cc field.cc field_conv.cc strfunc.cc filesort.cc \ libmysqld_int_a_SOURCES= $(libmysqld_sources) $(libmysqlsources) $(sqlsources) EXTRA_libmysqld_a_SOURCES = ha_innodb.cc ha_berkeley.cc ha_archive.cc \ ha_blackhole.cc ha_federated.cc ha_ndbcluster.cc \ - ha_ndbcluster_binlog.cc \ - ha_partition.cc + ha_ndbcluster_binlog.cc ha_partition.cc \ + ha_maria.cc libmysqld_a_DEPENDENCIES= @mysql_se_objs@ libmysqld_a_SOURCES= diff --git a/mysql-test/include/have_maria.inc b/mysql-test/include/have_maria.inc new file mode 100644 index 00000000000..955e2305ca5 --- /dev/null +++ b/mysql-test/include/have_maria.inc @@ -0,0 +1,4 @@ +-- require r/have_maria.require +disable_query_log; +show variables like "have_maria"; +enable_query_log; diff --git a/mysql-test/r/have_maria.require b/mysql-test/r/have_maria.require new file mode 100644 index 00000000000..02988af6976 --- /dev/null +++ b/mysql-test/r/have_maria.require @@ -0,0 +1,2 @@ +Variable_name Value +have_maria YES diff --git a/mysql-test/r/maria.result b/mysql-test/r/maria.result new file mode 100644 index 00000000000..3b360873c36 --- /dev/null +++ b/mysql-test/r/maria.result @@ -0,0 +1,1417 @@ +drop table if exists t1,t2; +SET SQL_WARNINGS=1; +CREATE TABLE t1 ( +STRING_DATA char(255) default NULL, +KEY string_data (STRING_DATA) +) ENGINE=MARIA; +INSERT INTO t1 VALUES ('AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA'); +INSERT INTO t1 VALUES ('DDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDD'); +INSERT INTO t1 VALUES ('FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF'); +INSERT INTO t1 VALUES ('FGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGG'); +INSERT INTO t1 VALUES ('HHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHH'); +INSERT INTO t1 VALUES ('WWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWW'); +CHECK TABLE t1; +Table Op Msg_type Msg_text +test.t1 check status OK +drop table t1; +create table t1 (a tinyint not null auto_increment, b blob not null, primary key (a)) engine=maria; +check table t1; +Table Op Msg_type Msg_text +test.t1 check status OK +repair table t1; +Table Op Msg_type Msg_text +test.t1 repair status OK +delete from t1 where (a & 1); +check table t1; +Table Op Msg_type Msg_text +test.t1 check status OK +repair table t1; +Table Op Msg_type Msg_text +test.t1 repair status OK +check table t1; +Table Op Msg_type Msg_text +test.t1 check status OK +drop table t1; +create table t1 (a int not null auto_increment, b int not null, primary key (a), index(b)) engine=maria; +insert into t1 (b) values (1),(2),(2),(2),(2); +optimize table t1; +Table Op Msg_type Msg_text +test.t1 optimize status OK +show index from t1; +Table Non_unique Key_name Seq_in_index Column_name Collation Cardinality Sub_part Packed Null Index_type Comment +t1 0 PRIMARY 1 a A 5 NULL NULL BTREE +t1 1 b 1 b A 1 NULL NULL BTREE +optimize table t1; +Table Op Msg_type Msg_text +test.t1 optimize status Table is already up to date +show index from t1; +Table Non_unique Key_name Seq_in_index Column_name Collation Cardinality Sub_part Packed Null Index_type Comment +t1 0 PRIMARY 1 a A 5 NULL NULL BTREE +t1 1 b 1 b A 1 NULL NULL BTREE +drop table t1; +create table t1 (a int not null, b int not null, c int not null, primary key (a),key(b)) engine=maria; +insert into t1 values (3,3,3),(1,1,1),(2,2,2),(4,4,4); +explain select * from t1 order by a; +id select_type table type possible_keys key key_len ref rows Extra +1 SIMPLE t1 ALL NULL NULL NULL NULL 4 Using filesort +explain select * from t1 order by b; +id select_type table type possible_keys key key_len ref rows Extra +1 SIMPLE t1 ALL NULL NULL NULL NULL 4 Using filesort +explain select * from t1 order by c; +id select_type table type possible_keys key key_len ref rows Extra +1 SIMPLE t1 ALL NULL NULL NULL NULL 4 Using filesort +explain select a from t1 order by a; +id select_type table type possible_keys key key_len ref rows Extra +1 SIMPLE t1 index NULL PRIMARY 4 NULL 4 Using index +explain select b from t1 order by b; +id select_type table type possible_keys key key_len ref rows Extra +1 SIMPLE t1 index NULL b 4 NULL 4 Using index +explain select a,b from t1 order by b; +id select_type table type possible_keys key key_len ref rows Extra +1 SIMPLE t1 ALL NULL NULL NULL NULL 4 Using filesort +explain select a,b from t1; +id select_type table type possible_keys key key_len ref rows Extra +1 SIMPLE t1 ALL NULL NULL NULL NULL 4 +explain select a,b,c from t1; +id select_type table type possible_keys key key_len ref rows Extra +1 SIMPLE t1 ALL NULL NULL NULL NULL 4 +drop table t1; +CREATE TABLE t1 (a INT) engine=maria; +INSERT INTO t1 VALUES (1), (2), (3); +LOCK TABLES t1 WRITE; +INSERT INTO t1 VALUES (1), (2), (3); +OPTIMIZE TABLE t1; +Table Op Msg_type Msg_text +test.t1 optimize status OK +DROP TABLE t1; +create table t1 ( t1 char(255), key(t1(250))) engine=maria; +insert t1 values ('137513751375137513751375137513751375137569516951695169516951695169516951695169'); +insert t1 values ('178417841784178417841784178417841784178403420342034203420342034203420342034203'); +insert t1 values ('213872387238723872387238723872387238723867376737673767376737673767376737673767'); +insert t1 values ('242624262426242624262426242624262426242607890789078907890789078907890789078907'); +insert t1 values ('256025602560256025602560256025602560256011701170117011701170117011701170117011'); +insert t1 values ('276027602760276027602760276027602760276001610161016101610161016101610161016101'); +insert t1 values ('281528152815281528152815281528152815281564956495649564956495649564956495649564'); +insert t1 values ('292129212921292129212921292129212921292102100210021002100210021002100210021002'); +insert t1 values ('380638063806380638063806380638063806380634483448344834483448344834483448344834'); +insert t1 values ('411641164116411641164116411641164116411616301630163016301630163016301630163016'); +insert t1 values ('420842084208420842084208420842084208420899889988998899889988998899889988998899'); +insert t1 values ('438443844384438443844384438443844384438482448244824482448244824482448244824482'); +insert t1 values ('443244324432443244324432443244324432443239613961396139613961396139613961396139'); +insert t1 values ('485448544854485448544854485448544854485477847784778477847784778477847784778477'); +insert t1 values ('494549454945494549454945494549454945494555275527552755275527552755275527552755'); +insert t1 values ('538647864786478647864786478647864786478688918891889188918891889188918891889188'); +insert t1 values ('565556555655565556555655565556555655565554845484548454845484548454845484548454'); +insert t1 values ('607860786078607860786078607860786078607856665666566656665666566656665666566656'); +insert t1 values ('640164016401640164016401640164016401640141274127412741274127412741274127412741'); +insert t1 values ('719471947194719471947194719471947194719478717871787178717871787178717871787178'); +insert t1 values ('742574257425742574257425742574257425742549604960496049604960496049604960496049'); +insert t1 values ('887088708870887088708870887088708870887035963596359635963596359635963596359635'); +insert t1 values ('917791779177917791779177917791779177917773857385738573857385738573857385738573'); +insert t1 values ('933293329332933293329332933293329332933278987898789878987898789878987898789878'); +insert t1 values ('963896389638963896389638963896389638963877807780778077807780778077807780778077'); +delete from t1 where t1>'2'; +insert t1 values ('70'), ('84'), ('60'), ('20'), ('76'), ('89'), ('49'), ('50'), +('88'), ('61'), ('42'), ('98'), ('39'), ('30'), ('25'), ('66'), ('61'), ('48'), +('80'), ('84'), ('98'), ('19'), ('91'), ('42'), ('47'); +optimize table t1; +Table Op Msg_type Msg_text +test.t1 optimize status OK +check table t1; +Table Op Msg_type Msg_text +test.t1 check status OK +drop table t1; +create table t1 (i1 int, i2 int, i3 int, i4 int, i5 int, i6 int, i7 int, i8 +int, i9 int, i10 int, i11 int, i12 int, i13 int, i14 int, i15 int, i16 int, i17 +int, i18 int, i19 int, i20 int, i21 int, i22 int, i23 int, i24 int, i25 int, +i26 int, i27 int, i28 int, i29 int, i30 int, i31 int, i32 int, i33 int, i34 +int, i35 int, i36 int, i37 int, i38 int, i39 int, i40 int, i41 int, i42 int, +i43 int, i44 int, i45 int, i46 int, i47 int, i48 int, i49 int, i50 int, i51 +int, i52 int, i53 int, i54 int, i55 int, i56 int, i57 int, i58 int, i59 int, +i60 int, i61 int, i62 int, i63 int, i64 int, i65 int, i66 int, i67 int, i68 +int, i69 int, i70 int, i71 int, i72 int, i73 int, i74 int, i75 int, i76 int, +i77 int, i78 int, i79 int, i80 int, i81 int, i82 int, i83 int, i84 int, i85 +int, i86 int, i87 int, i88 int, i89 int, i90 int, i91 int, i92 int, i93 int, +i94 int, i95 int, i96 int, i97 int, i98 int, i99 int, i100 int, i101 int, i102 +int, i103 int, i104 int, i105 int, i106 int, i107 int, i108 int, i109 int, i110 +int, i111 int, i112 int, i113 int, i114 int, i115 int, i116 int, i117 int, i118 +int, i119 int, i120 int, i121 int, i122 int, i123 int, i124 int, i125 int, i126 +int, i127 int, i128 int, i129 int, i130 int, i131 int, i132 int, i133 int, i134 +int, i135 int, i136 int, i137 int, i138 int, i139 int, i140 int, i141 int, i142 +int, i143 int, i144 int, i145 int, i146 int, i147 int, i148 int, i149 int, i150 +int, i151 int, i152 int, i153 int, i154 int, i155 int, i156 int, i157 int, i158 +int, i159 int, i160 int, i161 int, i162 int, i163 int, i164 int, i165 int, i166 +int, i167 int, i168 int, i169 int, i170 int, i171 int, i172 int, i173 int, i174 +int, i175 int, i176 int, i177 int, i178 int, i179 int, i180 int, i181 int, i182 +int, i183 int, i184 int, i185 int, i186 int, i187 int, i188 int, i189 int, i190 +int, i191 int, i192 int, i193 int, i194 int, i195 int, i196 int, i197 int, i198 +int, i199 int, i200 int, i201 int, i202 int, i203 int, i204 int, i205 int, i206 +int, i207 int, i208 int, i209 int, i210 int, i211 int, i212 int, i213 int, i214 +int, i215 int, i216 int, i217 int, i218 int, i219 int, i220 int, i221 int, i222 +int, i223 int, i224 int, i225 int, i226 int, i227 int, i228 int, i229 int, i230 +int, i231 int, i232 int, i233 int, i234 int, i235 int, i236 int, i237 int, i238 +int, i239 int, i240 int, i241 int, i242 int, i243 int, i244 int, i245 int, i246 +int, i247 int, i248 int, i249 int, i250 int, i251 int, i252 int, i253 int, i254 +int, i255 int, i256 int, i257 int, i258 int, i259 int, i260 int, i261 int, i262 +int, i263 int, i264 int, i265 int, i266 int, i267 int, i268 int, i269 int, i270 +int, i271 int, i272 int, i273 int, i274 int, i275 int, i276 int, i277 int, i278 +int, i279 int, i280 int, i281 int, i282 int, i283 int, i284 int, i285 int, i286 +int, i287 int, i288 int, i289 int, i290 int, i291 int, i292 int, i293 int, i294 +int, i295 int, i296 int, i297 int, i298 int, i299 int, i300 int, i301 int, i302 +int, i303 int, i304 int, i305 int, i306 int, i307 int, i308 int, i309 int, i310 +int, i311 int, i312 int, i313 int, i314 int, i315 int, i316 int, i317 int, i318 +int, i319 int, i320 int, i321 int, i322 int, i323 int, i324 int, i325 int, i326 +int, i327 int, i328 int, i329 int, i330 int, i331 int, i332 int, i333 int, i334 +int, i335 int, i336 int, i337 int, i338 int, i339 int, i340 int, i341 int, i342 +int, i343 int, i344 int, i345 int, i346 int, i347 int, i348 int, i349 int, i350 +int, i351 int, i352 int, i353 int, i354 int, i355 int, i356 int, i357 int, i358 +int, i359 int, i360 int, i361 int, i362 int, i363 int, i364 int, i365 int, i366 +int, i367 int, i368 int, i369 int, i370 int, i371 int, i372 int, i373 int, i374 +int, i375 int, i376 int, i377 int, i378 int, i379 int, i380 int, i381 int, i382 +int, i383 int, i384 int, i385 int, i386 int, i387 int, i388 int, i389 int, i390 +int, i391 int, i392 int, i393 int, i394 int, i395 int, i396 int, i397 int, i398 +int, i399 int, i400 int, i401 int, i402 int, i403 int, i404 int, i405 int, i406 +int, i407 int, i408 int, i409 int, i410 int, i411 int, i412 int, i413 int, i414 +int, i415 int, i416 int, i417 int, i418 int, i419 int, i420 int, i421 int, i422 +int, i423 int, i424 int, i425 int, i426 int, i427 int, i428 int, i429 int, i430 +int, i431 int, i432 int, i433 int, i434 int, i435 int, i436 int, i437 int, i438 +int, i439 int, i440 int, i441 int, i442 int, i443 int, i444 int, i445 int, i446 +int, i447 int, i448 int, i449 int, i450 int, i451 int, i452 int, i453 int, i454 +int, i455 int, i456 int, i457 int, i458 int, i459 int, i460 int, i461 int, i462 +int, i463 int, i464 int, i465 int, i466 int, i467 int, i468 int, i469 int, i470 +int, i471 int, i472 int, i473 int, i474 int, i475 int, i476 int, i477 int, i478 +int, i479 int, i480 int, i481 int, i482 int, i483 int, i484 int, i485 int, i486 +int, i487 int, i488 int, i489 int, i490 int, i491 int, i492 int, i493 int, i494 +int, i495 int, i496 int, i497 int, i498 int, i499 int, i500 int, i501 int, i502 +int, i503 int, i504 int, i505 int, i506 int, i507 int, i508 int, i509 int, i510 +int, i511 int, i512 int, i513 int, i514 int, i515 int, i516 int, i517 int, i518 +int, i519 int, i520 int, i521 int, i522 int, i523 int, i524 int, i525 int, i526 +int, i527 int, i528 int, i529 int, i530 int, i531 int, i532 int, i533 int, i534 +int, i535 int, i536 int, i537 int, i538 int, i539 int, i540 int, i541 int, i542 +int, i543 int, i544 int, i545 int, i546 int, i547 int, i548 int, i549 int, i550 +int, i551 int, i552 int, i553 int, i554 int, i555 int, i556 int, i557 int, i558 +int, i559 int, i560 int, i561 int, i562 int, i563 int, i564 int, i565 int, i566 +int, i567 int, i568 int, i569 int, i570 int, i571 int, i572 int, i573 int, i574 +int, i575 int, i576 int, i577 int, i578 int, i579 int, i580 int, i581 int, i582 +int, i583 int, i584 int, i585 int, i586 int, i587 int, i588 int, i589 int, i590 +int, i591 int, i592 int, i593 int, i594 int, i595 int, i596 int, i597 int, i598 +int, i599 int, i600 int, i601 int, i602 int, i603 int, i604 int, i605 int, i606 +int, i607 int, i608 int, i609 int, i610 int, i611 int, i612 int, i613 int, i614 +int, i615 int, i616 int, i617 int, i618 int, i619 int, i620 int, i621 int, i622 +int, i623 int, i624 int, i625 int, i626 int, i627 int, i628 int, i629 int, i630 +int, i631 int, i632 int, i633 int, i634 int, i635 int, i636 int, i637 int, i638 +int, i639 int, i640 int, i641 int, i642 int, i643 int, i644 int, i645 int, i646 +int, i647 int, i648 int, i649 int, i650 int, i651 int, i652 int, i653 int, i654 +int, i655 int, i656 int, i657 int, i658 int, i659 int, i660 int, i661 int, i662 +int, i663 int, i664 int, i665 int, i666 int, i667 int, i668 int, i669 int, i670 +int, i671 int, i672 int, i673 int, i674 int, i675 int, i676 int, i677 int, i678 +int, i679 int, i680 int, i681 int, i682 int, i683 int, i684 int, i685 int, i686 +int, i687 int, i688 int, i689 int, i690 int, i691 int, i692 int, i693 int, i694 +int, i695 int, i696 int, i697 int, i698 int, i699 int, i700 int, i701 int, i702 +int, i703 int, i704 int, i705 int, i706 int, i707 int, i708 int, i709 int, i710 +int, i711 int, i712 int, i713 int, i714 int, i715 int, i716 int, i717 int, i718 +int, i719 int, i720 int, i721 int, i722 int, i723 int, i724 int, i725 int, i726 +int, i727 int, i728 int, i729 int, i730 int, i731 int, i732 int, i733 int, i734 +int, i735 int, i736 int, i737 int, i738 int, i739 int, i740 int, i741 int, i742 +int, i743 int, i744 int, i745 int, i746 int, i747 int, i748 int, i749 int, i750 +int, i751 int, i752 int, i753 int, i754 int, i755 int, i756 int, i757 int, i758 +int, i759 int, i760 int, i761 int, i762 int, i763 int, i764 int, i765 int, i766 +int, i767 int, i768 int, i769 int, i770 int, i771 int, i772 int, i773 int, i774 +int, i775 int, i776 int, i777 int, i778 int, i779 int, i780 int, i781 int, i782 +int, i783 int, i784 int, i785 int, i786 int, i787 int, i788 int, i789 int, i790 +int, i791 int, i792 int, i793 int, i794 int, i795 int, i796 int, i797 int, i798 +int, i799 int, i800 int, i801 int, i802 int, i803 int, i804 int, i805 int, i806 +int, i807 int, i808 int, i809 int, i810 int, i811 int, i812 int, i813 int, i814 +int, i815 int, i816 int, i817 int, i818 int, i819 int, i820 int, i821 int, i822 +int, i823 int, i824 int, i825 int, i826 int, i827 int, i828 int, i829 int, i830 +int, i831 int, i832 int, i833 int, i834 int, i835 int, i836 int, i837 int, i838 +int, i839 int, i840 int, i841 int, i842 int, i843 int, i844 int, i845 int, i846 +int, i847 int, i848 int, i849 int, i850 int, i851 int, i852 int, i853 int, i854 +int, i855 int, i856 int, i857 int, i858 int, i859 int, i860 int, i861 int, i862 +int, i863 int, i864 int, i865 int, i866 int, i867 int, i868 int, i869 int, i870 +int, i871 int, i872 int, i873 int, i874 int, i875 int, i876 int, i877 int, i878 +int, i879 int, i880 int, i881 int, i882 int, i883 int, i884 int, i885 int, i886 +int, i887 int, i888 int, i889 int, i890 int, i891 int, i892 int, i893 int, i894 +int, i895 int, i896 int, i897 int, i898 int, i899 int, i900 int, i901 int, i902 +int, i903 int, i904 int, i905 int, i906 int, i907 int, i908 int, i909 int, i910 +int, i911 int, i912 int, i913 int, i914 int, i915 int, i916 int, i917 int, i918 +int, i919 int, i920 int, i921 int, i922 int, i923 int, i924 int, i925 int, i926 +int, i927 int, i928 int, i929 int, i930 int, i931 int, i932 int, i933 int, i934 +int, i935 int, i936 int, i937 int, i938 int, i939 int, i940 int, i941 int, i942 +int, i943 int, i944 int, i945 int, i946 int, i947 int, i948 int, i949 int, i950 +int, i951 int, i952 int, i953 int, i954 int, i955 int, i956 int, i957 int, i958 +int, i959 int, i960 int, i961 int, i962 int, i963 int, i964 int, i965 int, i966 +int, i967 int, i968 int, i969 int, i970 int, i971 int, i972 int, i973 int, i974 +int, i975 int, i976 int, i977 int, i978 int, i979 int, i980 int, i981 int, i982 +int, i983 int, i984 int, i985 int, i986 int, i987 int, i988 int, i989 int, i990 +int, i991 int, i992 int, i993 int, i994 int, i995 int, i996 int, i997 int, i998 +int, i999 int, i1000 int, b blob) row_format=dynamic engine=maria; +insert into t1 values (1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, +1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, +1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, +1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, +1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, +1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, +1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, +1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, +1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, +1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, +1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, +1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, +1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, +1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, +1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, +1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, +1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, +1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, +1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, +1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, +1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, +1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, +1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, +1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, +1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, +1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, +1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, +1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, +1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, +1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, +1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, +1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, +1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, +1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, +1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, +1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, +1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, +1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, +1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, "Sergei"); +update t1 set b=repeat('a',256); +update t1 set i1=0, i2=0, i3=0, i4=0, i5=0, i6=0, i7=0; +check table t1; +Table Op Msg_type Msg_text +test.t1 check status OK +delete from t1 where i8=1; +select i1,i2 from t1; +i1 i2 +check table t1; +Table Op Msg_type Msg_text +test.t1 check status OK +drop table t1; +CREATE TABLE `t1` ( +`post_id` mediumint(8) unsigned NOT NULL auto_increment, +`topic_id` mediumint(8) unsigned NOT NULL default '0', +`post_time` datetime NOT NULL default '0000-00-00 00:00:00', +`post_text` text NOT NULL, +`icon_url` varchar(10) NOT NULL default '', +`sign` tinyint(1) unsigned NOT NULL default '0', +`post_edit` varchar(150) NOT NULL default '', +`poster_login` varchar(35) NOT NULL default '', +`ip` varchar(15) NOT NULL default '', +PRIMARY KEY (`post_id`), +KEY `post_time` (`post_time`), +KEY `ip` (`ip`), +KEY `poster_login` (`poster_login`), +KEY `topic_id` (`topic_id`), +FULLTEXT KEY `post_text` (`post_text`) +) ENGINE=MARIA; +INSERT INTO t1 (post_text) VALUES ('ceci est un test'),('ceci est un test'),('ceci est un test'),('ceci est un test'),('ceci est un test'); +REPAIR TABLE t1; +Table Op Msg_type Msg_text +test.t1 repair status OK +CHECK TABLE t1; +Table Op Msg_type Msg_text +test.t1 check status OK +drop table t1; +CREATE TABLE t1 (a varchar(255), b varchar(255), c varchar(255), d varchar(255), e varchar(255), KEY t1 (a, b, c, d, e)) engine=maria; +ERROR 42000: Specified key was too long; max key length is 1000 bytes +CREATE TABLE t1 (a varchar(255), b varchar(255), c varchar(255), d varchar(255), e varchar(255)) engine=maria; +ALTER TABLE t1 ADD INDEX t1 (a, b, c, d, e); +ERROR 42000: Specified key was too long; max key length is 1000 bytes +DROP TABLE t1; +CREATE TABLE t1 (a int not null, b int, c int, key(b), key(c), key(a,b), key(c,a)) engine=maria; +INSERT into t1 values (0, null, 0), (0, null, 1), (0, null, 2), (0, null,3), (1,1,4); +create table t2 (a int not null, b int, c int, key(b), key(c), key(a)) engine=maria; +INSERT into t2 values (1,1,1), (2,2,2); +optimize table t1; +Table Op Msg_type Msg_text +test.t1 optimize status OK +show index from t1; +Table Non_unique Key_name Seq_in_index Column_name Collation Cardinality Sub_part Packed Null Index_type Comment +t1 1 b 1 b A 5 NULL NULL YES BTREE +t1 1 c 1 c A 5 NULL NULL YES BTREE +t1 1 a 1 a A 1 NULL NULL BTREE +t1 1 a 2 b A 5 NULL NULL YES BTREE +t1 1 c_2 1 c A 5 NULL NULL YES BTREE +t1 1 c_2 2 a A 5 NULL NULL BTREE +explain select * from t1,t2 where t1.a=t2.a; +id select_type table type possible_keys key key_len ref rows Extra +1 SIMPLE t2 ALL a NULL NULL NULL 2 +1 SIMPLE t1 ALL a NULL NULL NULL 4 Using where +explain select * from t1,t2 force index(a) where t1.a=t2.a; +id select_type table type possible_keys key key_len ref rows Extra +1 SIMPLE t2 ALL a NULL NULL NULL 2 +1 SIMPLE t1 ALL a NULL NULL NULL 4 Using where +explain select * from t1 force index(a),t2 force index(a) where t1.a=t2.a; +id select_type table type possible_keys key key_len ref rows Extra +1 SIMPLE t2 ALL a NULL NULL NULL 2 +1 SIMPLE t1 ref a a 4 test.t2.a 3 +explain select * from t1,t2 where t1.b=t2.b; +id select_type table type possible_keys key key_len ref rows Extra +1 SIMPLE t2 ALL b NULL NULL NULL 2 +1 SIMPLE t1 ref b b 5 test.t2.b 1 Using where +explain select * from t1,t2 force index(c) where t1.a=t2.a; +id select_type table type possible_keys key key_len ref rows Extra +1 SIMPLE t2 ALL NULL NULL NULL NULL 2 +1 SIMPLE t1 ALL a NULL NULL NULL 4 Using where +explain select * from t1 where a=0 or a=2; +id select_type table type possible_keys key key_len ref rows Extra +1 SIMPLE t1 ALL a NULL NULL NULL 5 Using where +explain select * from t1 force index (a) where a=0 or a=2; +id select_type table type possible_keys key key_len ref rows Extra +1 SIMPLE t1 range a a 4 NULL 4 Using where +explain select * from t1 where c=1; +id select_type table type possible_keys key key_len ref rows Extra +1 SIMPLE t1 ref c,c_2 c 5 const 1 Using where +explain select * from t1 use index() where c=1; +id select_type table type possible_keys key key_len ref rows Extra +1 SIMPLE t1 ALL NULL NULL NULL NULL 5 Using where +drop table t1,t2; +create table t1 (a int not null auto_increment primary key, b varchar(255)) engine=maria; +insert into t1 (b) values (repeat('a',100)),(repeat('b',100)),(repeat('c',100)); +update t1 set b=repeat(left(b,1),200) where a=1; +delete from t1 where (a & 1)= 0; +update t1 set b=repeat('e',200) where a=1; +flush tables; +check table t1; +Table Op Msg_type Msg_text +test.t1 check status OK +update t1 set b=repeat(left(b,1),255) where a between 1 and 5; +update t1 set b=repeat(left(b,1),10) where a between 32 and 43; +update t1 set b=repeat(left(b,1),2) where a between 64 and 66; +update t1 set b=repeat(left(b,1),65) where a between 67 and 70; +check table t1; +Table Op Msg_type Msg_text +test.t1 check status OK +insert into t1 (b) values (repeat('z',100)); +update t1 set b="test" where left(b,1) > 'n'; +check table t1; +Table Op Msg_type Msg_text +test.t1 check status OK +drop table t1; +create table t1 ( a text not null, key a (a(20))) engine=maria; +insert into t1 values ('aaa '),('aaa'),('aa'); +check table t1; +Table Op Msg_type Msg_text +test.t1 check status OK +repair table t1; +Table Op Msg_type Msg_text +test.t1 repair status OK +select concat(a,'.') from t1 where a='aaa'; +concat(a,'.') +aaa . +aaa. +select concat(a,'.') from t1 where binary a='aaa'; +concat(a,'.') +aaa. +update t1 set a='bbb' where a='aaa'; +select concat(a,'.') from t1; +concat(a,'.') +bbb. +bbb. +aa. +drop table t1; +create table t1(a text not null, b text not null, c text not null, index (a(10),b(10),c(10))) engine=maria; +insert into t1 values('807780', '477', '165'); +insert into t1 values('807780', '477', '162'); +insert into t1 values('807780', '472', '162'); +select * from t1 where a='807780' and b='477' and c='165'; +a b c +807780 477 165 +drop table t1; +DROP TABLE IF EXISTS t1; +Warnings: +Note 1051 Unknown table 't1' +CREATE TABLE t1 (a varchar(150) NOT NULL, KEY (a)) engine=maria; +INSERT t1 VALUES ("can \tcan"); +INSERT t1 VALUES ("can can"); +INSERT t1 VALUES ("can"); +SELECT * FROM t1; +a +can can +can +can can +CHECK TABLE t1; +Table Op Msg_type Msg_text +test.t1 check status OK +DROP TABLE t1; +create table t1 (a blob) engine=maria; +insert into t1 values('a '),('a'); +select concat(a,'.') from t1 where a='a'; +concat(a,'.') +a. +select concat(a,'.') from t1 where a='a '; +concat(a,'.') +a . +alter table t1 add key(a(2)); +select concat(a,'.') from t1 where a='a'; +concat(a,'.') +a. +select concat(a,'.') from t1 where a='a '; +concat(a,'.') +a . +drop table t1; +create table t1 (a int not null auto_increment primary key, b text not null, unique b (b(20))) engine=maria; +insert into t1 (b) values ('a'),('b'),('c'); +select concat(b,'.') from t1; +concat(b,'.') +a. +b. +c. +update t1 set b='b ' where a=2; +update t1 set b='b ' where a > 1; +ERROR 23000: Duplicate entry 'b ' for key 'b' +insert into t1 (b) values ('b'); +ERROR 23000: Duplicate entry 'b' for key 'b' +select * from t1; +a b +1 a +2 b +3 c +delete from t1 where b='b'; +select a,concat(b,'.') from t1; +a concat(b,'.') +1 a. +3 c. +drop table t1; +create table t1 (a int not null) engine=maria; +create table t2 (a int not null, primary key (a)) engine=maria; +insert into t1 values (1); +insert into t2 values (1),(2); +select sql_big_result distinct t1.a from t1,t2 order by t2.a; +a +1 +select distinct t1.a from t1,t2 order by t2.a; +a +1 +select sql_big_result distinct t1.a from t1,t2; +a +1 +explain select sql_big_result distinct t1.a from t1,t2 order by t2.a; +id select_type table type possible_keys key key_len ref rows Extra +1 SIMPLE t1 system NULL NULL NULL NULL 1 Using temporary +1 SIMPLE t2 index NULL PRIMARY 4 NULL 2 Using index; Distinct +explain select distinct t1.a from t1,t2 order by t2.a; +id select_type table type possible_keys key key_len ref rows Extra +1 SIMPLE t1 system NULL NULL NULL NULL 1 Using temporary +1 SIMPLE t2 index NULL PRIMARY 4 NULL 2 Using index; Distinct +drop table t1,t2; +create table t1 ( +c1 varchar(32), +key (c1) +) engine=maria; +alter table t1 disable keys; +insert into t1 values ('a'), ('b'); +select c1 from t1 order by c1 limit 1; +c1 +a +drop table t1; +CREATE TABLE t1 (`a` int(11) NOT NULL default '0', `b` int(11) NOT NULL default '0', UNIQUE KEY `a` USING RTREE (`a`,`b`)) ENGINE=MARIA; +Got one of the listed errors +create table t1 (a int, b varchar(200), c text not null) checksum=1 engine=maria; +create table t2 (a int, b varchar(200), c text not null) checksum=0 engine=maria; +insert t1 values (1, "aaa", "bbb"), (NULL, "", "ccccc"), (0, NULL, ""); +insert t2 select * from t1; +checksum table t1, t2, t3 quick; +Table Checksum +test.t1 2948697075 +test.t2 NULL +test.t3 NULL +Warnings: +Error 1146 Table 'test.t3' doesn't exist +checksum table t1, t2, t3; +Table Checksum +test.t1 2948697075 +test.t2 2948697075 +test.t3 NULL +Warnings: +Error 1146 Table 'test.t3' doesn't exist +checksum table t1, t2, t3 extended; +Table Checksum +test.t1 2948697075 +test.t2 2948697075 +test.t3 NULL +Warnings: +Error 1146 Table 'test.t3' doesn't exist +drop table t1,t2; +create table t1 (a int, key (a)) engine=maria; +show keys from t1; +Table Non_unique Key_name Seq_in_index Column_name Collation Cardinality Sub_part Packed Null Index_type Comment +t1 1 a 1 a A NULL NULL NULL YES BTREE +alter table t1 disable keys; +show keys from t1; +Table Non_unique Key_name Seq_in_index Column_name Collation Cardinality Sub_part Packed Null Index_type Comment +t1 1 a 1 a A NULL NULL NULL YES BTREE disabled +create table t2 (a int) engine=maria; +set @@rand_seed1=31415926,@@rand_seed2=2718281828; +insert t1 select * from t2; +show keys from t1; +Table Non_unique Key_name Seq_in_index Column_name Collation Cardinality Sub_part Packed Null Index_type Comment +t1 1 a 1 a A NULL NULL NULL YES BTREE disabled +alter table t1 enable keys; +show keys from t1; +Table Non_unique Key_name Seq_in_index Column_name Collation Cardinality Sub_part Packed Null Index_type Comment +t1 1 a 1 a A 1000 NULL NULL YES BTREE +alter table t1 engine=heap; +alter table t1 disable keys; +Warnings: +Note 1031 Table storage engine for 't1' doesn't have this option +show keys from t1; +Table Non_unique Key_name Seq_in_index Column_name Collation Cardinality Sub_part Packed Null Index_type Comment +t1 1 a 1 a NULL 500 NULL NULL YES HASH +drop table t1,t2; +create table t1 ( a tinytext, b char(1), index idx (a(1),b)) engine=maria; +insert into t1 values (null,''), (null,''); +explain select count(*) from t1 where a is null; +id select_type table type possible_keys key key_len ref rows Extra +1 SIMPLE t1 ref idx idx 4 const 1 Using where +select count(*) from t1 where a is null; +count(*) +2 +drop table t1; +create table t1 (c1 int, c2 varchar(4) not null default '', +key(c2(3))) default charset=utf8 engine=maria; +insert into t1 values (1,'A'), (2, 'B'), (3, 'A'); +update t1 set c2='A B' where c1=2; +check table t1; +Table Op Msg_type Msg_text +test.t1 check status OK +drop table t1; +create table t1 (c1 int) engine=maria; +insert into t1 values (1),(2),(3),(4); +checksum table t1; +Table Checksum +test.t1 149057747 +delete from t1 where c1 = 1; +create table t2 engine=maria as select * from t1; +checksum table t1; +Table Checksum +test.t1 984116287 +checksum table t2; +Table Checksum +test.t2 984116287 +drop table t1, t2; +show variables like 'maria_stats_method'; +Variable_name Value +maria_stats_method nulls_unequal +create table t1 (a int, key(a)) engine=maria; +insert into t1 values (0),(1),(2),(3),(4); +insert into t1 select NULL from t1; +analyze table t1; +Table Op Msg_type Msg_text +test.t1 analyze status OK +show index from t1; +Table Non_unique Key_name Seq_in_index Column_name Collation Cardinality Sub_part Packed Null Index_type Comment +t1 1 a 1 a A 10 NULL NULL YES BTREE +insert into t1 values (11); +delete from t1 where a=11; +check table t1; +Table Op Msg_type Msg_text +test.t1 check status OK +show index from t1; +Table Non_unique Key_name Seq_in_index Column_name Collation Cardinality Sub_part Packed Null Index_type Comment +t1 1 a 1 a A 10 NULL NULL YES BTREE +set maria_stats_method=nulls_equal; +show variables like 'maria_stats_method'; +Variable_name Value +maria_stats_method nulls_equal +insert into t1 values (11); +delete from t1 where a=11; +analyze table t1; +Table Op Msg_type Msg_text +test.t1 analyze status OK +show index from t1; +Table Non_unique Key_name Seq_in_index Column_name Collation Cardinality Sub_part Packed Null Index_type Comment +t1 1 a 1 a A 5 NULL NULL YES BTREE +insert into t1 values (11); +delete from t1 where a=11; +check table t1; +Table Op Msg_type Msg_text +test.t1 check status OK +show index from t1; +Table Non_unique Key_name Seq_in_index Column_name Collation Cardinality Sub_part Packed Null Index_type Comment +t1 1 a 1 a A 5 NULL NULL YES BTREE +set maria_stats_method=DEFAULT; +show variables like 'maria_stats_method'; +Variable_name Value +maria_stats_method nulls_unequal +insert into t1 values (11); +delete from t1 where a=11; +analyze table t1; +Table Op Msg_type Msg_text +test.t1 analyze status OK +show index from t1; +Table Non_unique Key_name Seq_in_index Column_name Collation Cardinality Sub_part Packed Null Index_type Comment +t1 1 a 1 a A 10 NULL NULL YES BTREE +insert into t1 values (11); +delete from t1 where a=11; +check table t1; +Table Op Msg_type Msg_text +test.t1 check status OK +show index from t1; +Table Non_unique Key_name Seq_in_index Column_name Collation Cardinality Sub_part Packed Null Index_type Comment +t1 1 a 1 a A 10 NULL NULL YES BTREE +drop table t1; +set maria_stats_method=nulls_ignored; +show variables like 'maria_stats_method'; +Variable_name Value +maria_stats_method nulls_ignored +create table t1 ( +a char(3), b char(4), c char(5), d char(6), +key(a,b,c,d) +) engine=maria; +insert into t1 values ('bcd','def1', NULL, 'zz'); +insert into t1 values ('bcd','def2', NULL, 'zz'); +insert into t1 values ('bce','def1', 'yuu', NULL); +insert into t1 values ('bce','def2', NULL, 'quux'); +analyze table t1; +Table Op Msg_type Msg_text +test.t1 analyze status OK +show index from t1; +Table Non_unique Key_name Seq_in_index Column_name Collation Cardinality Sub_part Packed Null Index_type Comment +t1 1 a 1 a A 2 NULL NULL YES BTREE +t1 1 a 2 b A 4 NULL NULL YES BTREE +t1 1 a 3 c A 4 NULL NULL YES BTREE +t1 1 a 4 d A 4 NULL NULL YES BTREE +delete from t1; +analyze table t1; +Table Op Msg_type Msg_text +test.t1 analyze status OK +show index from t1; +Table Non_unique Key_name Seq_in_index Column_name Collation Cardinality Sub_part Packed Null Index_type Comment +t1 1 a 1 a A 0 NULL NULL YES BTREE +t1 1 a 2 b A 0 NULL NULL YES BTREE +t1 1 a 3 c A 0 NULL NULL YES BTREE +t1 1 a 4 d A 0 NULL NULL YES BTREE +set maria_stats_method=DEFAULT; +drop table t1; +create table t1( +cip INT NOT NULL, +time TIME NOT NULL, +score INT NOT NULL DEFAULT 0, +bob TINYBLOB +) engine=maria; +insert into t1 (cip, time) VALUES (1, '00:01'), (2, '00:02'), (3,'00:03'); +insert into t1 (cip, bob, time) VALUES (4, 'a', '00:04'), (5, 'b', '00:05'), +(6, 'c', '00:06'); +select * from t1 where bob is null and cip=1; +cip time score bob +1 00:01:00 0 NULL +create index bug on t1 (bob(22), cip, time); +select * from t1 where bob is null and cip=1; +cip time score bob +1 00:01:00 0 NULL +drop table t1; +create table t1 ( +id1 int not null auto_increment, +id2 int not null default '0', +t text not null, +primary key (id1), +key x (id2, t(32)) +) engine=maria; +insert into t1 (id2, t) values +(10, 'abc'), (10, 'abc'), (10, 'abc'), +(20, 'abc'), (20, 'abc'), (20, 'def'), +(10, 'abc'), (10, 'abc'); +select count(*) from t1 where id2 = 10; +count(*) +5 +select count(id1) from t1 where id2 = 10; +count(id1) +5 +drop table t1; +set storage_engine=MARIA; +drop table if exists t1,t2,t3; +--- Testing varchar --- +--- Testing varchar --- +create table t1 (v varchar(10), c char(10), t text); +insert into t1 values('+ ', '+ ', '+ '); +set @a=repeat(' ',20); +insert into t1 values (concat('+',@a),concat('+',@a),concat('+',@a)); +Warnings: +Note 1265 Data truncated for column 'v' at row 1 +select concat('*',v,'*',c,'*',t,'*') from t1; +concat('*',v,'*',c,'*',t,'*') +*+ *+*+ * +*+ *+*+ * +show create table t1; +Table Create Table +t1 CREATE TABLE `t1` ( + `v` varchar(10) DEFAULT NULL, + `c` char(10) DEFAULT NULL, + `t` text +) ENGINE=MARIA DEFAULT CHARSET=latin1 +create table t2 like t1; +show create table t2; +Table Create Table +t2 CREATE TABLE `t2` ( + `v` varchar(10) DEFAULT NULL, + `c` char(10) DEFAULT NULL, + `t` text +) ENGINE=MARIA DEFAULT CHARSET=latin1 +create table t3 select * from t1; +show create table t3; +Table Create Table +t3 CREATE TABLE `t3` ( + `v` varchar(10) DEFAULT NULL, + `c` char(10) DEFAULT NULL, + `t` text +) ENGINE=MARIA DEFAULT CHARSET=latin1 +alter table t1 modify c varchar(10); +show create table t1; +Table Create Table +t1 CREATE TABLE `t1` ( + `v` varchar(10) DEFAULT NULL, + `c` varchar(10) DEFAULT NULL, + `t` text +) ENGINE=MARIA DEFAULT CHARSET=latin1 +alter table t1 modify v char(10); +show create table t1; +Table Create Table +t1 CREATE TABLE `t1` ( + `v` char(10) DEFAULT NULL, + `c` varchar(10) DEFAULT NULL, + `t` text +) ENGINE=MARIA DEFAULT CHARSET=latin1 +alter table t1 modify t varchar(10); +Warnings: +Note 1265 Data truncated for column 't' at row 2 +show create table t1; +Table Create Table +t1 CREATE TABLE `t1` ( + `v` char(10) DEFAULT NULL, + `c` varchar(10) DEFAULT NULL, + `t` varchar(10) DEFAULT NULL +) ENGINE=MARIA DEFAULT CHARSET=latin1 +select concat('*',v,'*',c,'*',t,'*') from t1; +concat('*',v,'*',c,'*',t,'*') +*+*+*+ * +*+*+*+ * +drop table t1,t2,t3; +create table t1 (v varchar(10), c char(10), t text, key(v), key(c), key(t(10))); +show create table t1; +Table Create Table +t1 CREATE TABLE `t1` ( + `v` varchar(10) DEFAULT NULL, + `c` char(10) DEFAULT NULL, + `t` text, + KEY `v` (`v`), + KEY `c` (`c`), + KEY `t` (`t`(10)) +) ENGINE=MARIA DEFAULT CHARSET=latin1 +select count(*) from t1; +count(*) +270 +insert into t1 values(concat('a',char(1)),concat('a',char(1)),concat('a',char(1))); +select count(*) from t1 where v='a'; +count(*) +10 +select count(*) from t1 where c='a'; +count(*) +10 +select count(*) from t1 where t='a'; +count(*) +10 +select count(*) from t1 where v='a '; +count(*) +10 +select count(*) from t1 where c='a '; +count(*) +10 +select count(*) from t1 where t='a '; +count(*) +10 +select count(*) from t1 where v between 'a' and 'a '; +count(*) +10 +select count(*) from t1 where v between 'a' and 'a ' and v between 'a ' and 'b\n'; +count(*) +10 +select count(*) from t1 where v like 'a%'; +count(*) +11 +select count(*) from t1 where c like 'a%'; +count(*) +11 +select count(*) from t1 where t like 'a%'; +count(*) +11 +select count(*) from t1 where v like 'a %'; +count(*) +9 +explain select count(*) from t1 where v='a '; +id select_type table type possible_keys key key_len ref rows Extra +1 SIMPLE t1 ref v v 13 const # Using where; Using index +explain select count(*) from t1 where c='a '; +id select_type table type possible_keys key key_len ref rows Extra +1 SIMPLE t1 ref c c 11 const # Using where; Using index +explain select count(*) from t1 where t='a '; +id select_type table type possible_keys key key_len ref rows Extra +1 SIMPLE t1 range t t 13 NULL # Using where +explain select count(*) from t1 where v like 'a%'; +id select_type table type possible_keys key key_len ref rows Extra +1 SIMPLE t1 range v v 13 NULL # Using where; Using index +explain select count(*) from t1 where v between 'a' and 'a '; +id select_type table type possible_keys key key_len ref rows Extra +1 SIMPLE t1 ref v v 13 const # Using where; Using index +explain select count(*) from t1 where v between 'a' and 'a ' and v between 'a ' and 'b\n'; +id select_type table type possible_keys key key_len ref rows Extra +1 SIMPLE t1 ref v v 13 const # Using where; Using index +alter table t1 add unique(v); +ERROR 23000: Duplicate entry '{ ' for key 'v_2' +alter table t1 add key(v); +select concat('*',v,'*',c,'*',t,'*') as qq from t1 where v='a'; +qq +*a*a*a* +*a *a*a * +*a *a*a * +*a *a*a * +*a *a*a * +*a *a*a * +*a *a*a * +*a *a*a * +*a *a*a * +*a *a*a * +explain select * from t1 where v='a'; +id select_type table type possible_keys key key_len ref rows Extra +1 SIMPLE t1 ref v,v_2 # 13 const # Using where +select v,count(*) from t1 group by v limit 10; +v count(*) +a 1 +a 10 +b 10 +c 10 +d 10 +e 10 +f 10 +g 10 +h 10 +i 10 +select v,count(t) from t1 group by v limit 10; +v count(t) +a 1 +a 10 +b 10 +c 10 +d 10 +e 10 +f 10 +g 10 +h 10 +i 10 +select v,count(c) from t1 group by v limit 10; +v count(c) +a 1 +a 10 +b 10 +c 10 +d 10 +e 10 +f 10 +g 10 +h 10 +i 10 +select sql_big_result v,count(t) from t1 group by v limit 10; +v count(t) +a 1 +a 10 +b 10 +c 10 +d 10 +e 10 +f 10 +g 10 +h 10 +i 10 +select sql_big_result v,count(c) from t1 group by v limit 10; +v count(c) +a 1 +a 10 +b 10 +c 10 +d 10 +e 10 +f 10 +g 10 +h 10 +i 10 +select c,count(*) from t1 group by c limit 10; +c count(*) +a 1 +a 10 +b 10 +c 10 +d 10 +e 10 +f 10 +g 10 +h 10 +i 10 +select c,count(t) from t1 group by c limit 10; +c count(t) +a 1 +a 10 +b 10 +c 10 +d 10 +e 10 +f 10 +g 10 +h 10 +i 10 +select sql_big_result c,count(t) from t1 group by c limit 10; +c count(t) +a 1 +a 10 +b 10 +c 10 +d 10 +e 10 +f 10 +g 10 +h 10 +i 10 +select t,count(*) from t1 group by t limit 10; +t count(*) +a 1 +a 10 +b 10 +c 10 +d 10 +e 10 +f 10 +g 10 +h 10 +i 10 +select t,count(t) from t1 group by t limit 10; +t count(t) +a 1 +a 10 +b 10 +c 10 +d 10 +e 10 +f 10 +g 10 +h 10 +i 10 +select sql_big_result t,count(t) from t1 group by t limit 10; +t count(t) +a 1 +a 10 +b 10 +c 10 +d 10 +e 10 +f 10 +g 10 +h 10 +i 10 +alter table t1 modify v varchar(300), drop key v, drop key v_2, add key v (v); +show create table t1; +Table Create Table +t1 CREATE TABLE `t1` ( + `v` varchar(300) DEFAULT NULL, + `c` char(10) DEFAULT NULL, + `t` text, + KEY `c` (`c`), + KEY `t` (`t`(10)), + KEY `v` (`v`) +) ENGINE=MARIA DEFAULT CHARSET=latin1 +select count(*) from t1 where v='a'; +count(*) +10 +select count(*) from t1 where v='a '; +count(*) +10 +select count(*) from t1 where v between 'a' and 'a '; +count(*) +10 +select count(*) from t1 where v between 'a' and 'a ' and v between 'a ' and 'b\n'; +count(*) +10 +select count(*) from t1 where v like 'a%'; +count(*) +11 +select count(*) from t1 where v like 'a %'; +count(*) +9 +explain select count(*) from t1 where v='a '; +id select_type table type possible_keys key key_len ref rows Extra +1 SIMPLE t1 ref v v 303 const # Using where; Using index +explain select count(*) from t1 where v like 'a%'; +id select_type table type possible_keys key key_len ref rows Extra +1 SIMPLE t1 range v v 303 NULL # Using where; Using index +explain select count(*) from t1 where v between 'a' and 'a '; +id select_type table type possible_keys key key_len ref rows Extra +1 SIMPLE t1 ref v v 303 const # Using where; Using index +explain select count(*) from t1 where v between 'a' and 'a ' and v between 'a ' and 'b\n'; +id select_type table type possible_keys key key_len ref rows Extra +1 SIMPLE t1 ref v v 303 const # Using where; Using index +explain select * from t1 where v='a'; +id select_type table type possible_keys key key_len ref rows Extra +1 SIMPLE t1 ref v v 303 const # Using where +select v,count(*) from t1 group by v limit 10; +v count(*) +a 1 +a 10 +b 10 +c 10 +d 10 +e 10 +f 10 +g 10 +h 10 +i 10 +select v,count(t) from t1 group by v limit 10; +v count(t) +a 1 +a 10 +b 10 +c 10 +d 10 +e 10 +f 10 +g 10 +h 10 +i 10 +select sql_big_result v,count(t) from t1 group by v limit 10; +v count(t) +a 1 +a 10 +b 10 +c 10 +d 10 +e 10 +f 10 +g 10 +h 10 +i 10 +alter table t1 drop key v, add key v (v(30)); +show create table t1; +Table Create Table +t1 CREATE TABLE `t1` ( + `v` varchar(300) DEFAULT NULL, + `c` char(10) DEFAULT NULL, + `t` text, + KEY `c` (`c`), + KEY `t` (`t`(10)), + KEY `v` (`v`(30)) +) ENGINE=MARIA DEFAULT CHARSET=latin1 +select count(*) from t1 where v='a'; +count(*) +10 +select count(*) from t1 where v='a '; +count(*) +10 +select count(*) from t1 where v between 'a' and 'a '; +count(*) +10 +select count(*) from t1 where v between 'a' and 'a ' and v between 'a ' and 'b\n'; +count(*) +10 +select count(*) from t1 where v like 'a%'; +count(*) +11 +select count(*) from t1 where v like 'a %'; +count(*) +9 +explain select count(*) from t1 where v='a '; +id select_type table type possible_keys key key_len ref rows Extra +1 SIMPLE t1 ref v v 33 const # Using where +explain select count(*) from t1 where v like 'a%'; +id select_type table type possible_keys key key_len ref rows Extra +1 SIMPLE t1 range v v 33 NULL # Using where +explain select count(*) from t1 where v between 'a' and 'a '; +id select_type table type possible_keys key key_len ref rows Extra +1 SIMPLE t1 ref v v 33 const # Using where +explain select count(*) from t1 where v between 'a' and 'a ' and v between 'a ' and 'b\n'; +id select_type table type possible_keys key key_len ref rows Extra +1 SIMPLE t1 ref v v 33 const # Using where +explain select * from t1 where v='a'; +id select_type table type possible_keys key key_len ref rows Extra +1 SIMPLE t1 ref v v 33 const # Using where +select v,count(*) from t1 group by v limit 10; +v count(*) +a 1 +a 10 +b 10 +c 10 +d 10 +e 10 +f 10 +g 10 +h 10 +i 10 +select v,count(t) from t1 group by v limit 10; +v count(t) +a 1 +a 10 +b 10 +c 10 +d 10 +e 10 +f 10 +g 10 +h 10 +i 10 +select sql_big_result v,count(t) from t1 group by v limit 10; +v count(t) +a 1 +a 10 +b 10 +c 10 +d 10 +e 10 +f 10 +g 10 +h 10 +i 10 +alter table t1 modify v varchar(600), drop key v, add key v (v); +show create table t1; +Table Create Table +t1 CREATE TABLE `t1` ( + `v` varchar(600) DEFAULT NULL, + `c` char(10) DEFAULT NULL, + `t` text, + KEY `c` (`c`), + KEY `t` (`t`(10)), + KEY `v` (`v`) +) ENGINE=MARIA DEFAULT CHARSET=latin1 +select v,count(*) from t1 group by v limit 10; +v count(*) +a 1 +a 10 +b 10 +c 10 +d 10 +e 10 +f 10 +g 10 +h 10 +i 10 +select v,count(t) from t1 group by v limit 10; +v count(t) +a 1 +a 10 +b 10 +c 10 +d 10 +e 10 +f 10 +g 10 +h 10 +i 10 +select sql_big_result v,count(t) from t1 group by v limit 10; +v count(t) +a 1 +a 10 +b 10 +c 10 +d 10 +e 10 +f 10 +g 10 +h 10 +i 10 +drop table t1; +create table t1 (a char(10), unique (a)); +insert into t1 values ('a '); +insert into t1 values ('a '); +ERROR 23000: Duplicate entry 'a' for key 'a' +alter table t1 modify a varchar(10); +insert into t1 values ('a '),('a '),('a '),('a '); +ERROR 23000: Duplicate entry 'a ' for key 'a' +insert into t1 values ('a '); +ERROR 23000: Duplicate entry 'a ' for key 'a' +insert into t1 values ('a '); +ERROR 23000: Duplicate entry 'a ' for key 'a' +insert into t1 values ('a '); +ERROR 23000: Duplicate entry 'a ' for key 'a' +update t1 set a='a ' where a like 'a%'; +select concat(a,'.') from t1; +concat(a,'.') +a . +update t1 set a='abc ' where a like 'a '; +select concat(a,'.') from t1; +concat(a,'.') +a . +update t1 set a='a ' where a like 'a %'; +select concat(a,'.') from t1; +concat(a,'.') +a . +update t1 set a='a ' where a like 'a '; +select concat(a,'.') from t1; +concat(a,'.') +a . +drop table t1; +create table t1 (v varchar(10), c char(10), t text, key(v(5)), key(c(5)), key(t(5))); +show create table t1; +Table Create Table +t1 CREATE TABLE `t1` ( + `v` varchar(10) DEFAULT NULL, + `c` char(10) DEFAULT NULL, + `t` text, + KEY `v` (`v`(5)), + KEY `c` (`c`(5)), + KEY `t` (`t`(5)) +) ENGINE=MARIA DEFAULT CHARSET=latin1 +drop table t1; +create table t1 (v char(10) character set utf8); +show create table t1; +Table Create Table +t1 CREATE TABLE `t1` ( + `v` char(10) CHARACTER SET utf8 DEFAULT NULL +) ENGINE=MARIA DEFAULT CHARSET=latin1 +drop table t1; +create table t1 (v varchar(10), c char(10)) row_format=fixed; +show create table t1; +Table Create Table +t1 CREATE TABLE `t1` ( + `v` varchar(10) DEFAULT NULL, + `c` char(10) DEFAULT NULL +) ENGINE=MARIA DEFAULT CHARSET=latin1 ROW_FORMAT=FIXED +insert into t1 values('a','a'),('a ','a '); +select concat('*',v,'*',c,'*') from t1; +concat('*',v,'*',c,'*') +*a*a* +*a *a* +drop table t1; +create table t1 (v varchar(65530), key(v(10))); +insert into t1 values(repeat('a',65530)); +select length(v) from t1 where v=repeat('a',65530); +length(v) +65530 +drop table t1; +create table t1(a int, b varchar(12), key ba(b, a)); +insert into t1 values (1, 'A'), (20, NULL); +explain select * from t1 where a=20 and b is null; +id select_type table type possible_keys key key_len ref rows Extra +1 SIMPLE t1 ref ba ba 20 const,const 1 Using where; Using index +select * from t1 where a=20 and b is null; +a b +20 NULL +drop table t1; +create table t1 (v varchar(65530), key(v)) engine=maria; +Warnings: +Warning 1071 Specified key was too long; max key length is 1000 bytes +drop table if exists t1; +create table t1 (v varchar(65536)) engine=maria; +Warnings: +Note 1246 Converting column 'v' from VARCHAR to TEXT +show create table t1; +Table Create Table +t1 CREATE TABLE `t1` ( + `v` mediumtext +) ENGINE=MARIA DEFAULT CHARSET=latin1 +drop table t1; +create table t1 (v varchar(65530) character set utf8) engine=maria; +Warnings: +Note 1246 Converting column 'v' from VARCHAR to TEXT +show create table t1; +Table Create Table +t1 CREATE TABLE `t1` ( + `v` mediumtext CHARACTER SET utf8 +) ENGINE=MARIA DEFAULT CHARSET=latin1 +drop table t1; +create table t1 (v varchar(65535)) engine=maria; +ERROR 42000: Row size too large. The maximum row size for the used table type, not counting BLOBs, is 65535. You have to change some columns to TEXT or BLOBs +set storage_engine=MyISAM; +create table t1 (a int) engine=maria; +drop table if exists t1; +Warnings: +Error 2 Can't find file: 't1' (errno: 2) +create table t1 (a int) engine=maria; +drop table t1; +Got one of the listed errors +create table t1 (a int) engine=maria; +drop table t1; +Got one of the listed errors +drop table t1; +ERROR 42S02: Unknown table 't1' +set @save_concurrent_insert=@@concurrent_insert; +set global concurrent_insert=1; +create table t1 (a int) engine=maria; +insert into t1 values (1),(2),(3),(4),(5); +lock table t1 read local; +insert into t1 values(6),(7); +unlock tables; +delete from t1 where a>=3 and a<=4; +lock table t1 read local; +set global concurrent_insert=2; +insert into t1 values (8),(9); +unlock tables; +insert into t1 values (10),(11),(12); +select * from t1; +a +1 +2 +11 +10 +5 +6 +7 +8 +9 +12 +check table t1; +Table Op Msg_type Msg_text +test.t1 check status OK +drop table t1; +create table t1 (a int, b varchar(30) default "hello") engine=maria; +insert into t1 (a) values (1),(2),(3),(4),(5); +lock table t1 read local; +insert into t1 (a) values(6),(7); +unlock tables; +delete from t1 where a>=3 and a<=4; +lock table t1 read local; +set global concurrent_insert=2; +insert into t1 (a) values (8),(9); +unlock tables; +insert into t1 (a) values (10),(11),(12); +select a from t1; +a +1 +2 +11 +10 +5 +6 +7 +8 +9 +12 +check table t1; +Table Op Msg_type Msg_text +test.t1 check status OK +drop table t1; +set global concurrent_insert=@save_concurrent_insert; +create table t1 (a int, key(a)) engine=maria; +insert into t1 values (1),(2),(3),(4),(NULL),(NULL),(NULL),(NULL); +analyze table t1; +Table Op Msg_type Msg_text +test.t1 analyze status OK +show keys from t1; +Table Non_unique Key_name Seq_in_index Column_name Collation Cardinality Sub_part Packed Null Index_type Comment +t1 1 a 1 a A 8 NULL NULL YES BTREE +alter table t1 disable keys; +alter table t1 enable keys; +show keys from t1; +Table Non_unique Key_name Seq_in_index Column_name Collation Cardinality Sub_part Packed Null Index_type Comment +t1 1 a 1 a A 8 NULL NULL YES BTREE +drop table t1; +create table t1 (c1 int) engine=maria pack_keys=0; +create table t2 (c1 int) engine=maria pack_keys=1; +create table t3 (c1 int) engine=maria pack_keys=default; +create table t4 (c1 int) engine=maria pack_keys=2; +ERROR 42000: You have an error in your SQL syntax; check the manual that corresponds to your MySQL server version for the right syntax to use near '2' at line 1 +drop table t1, t2, t3; diff --git a/mysql-test/r/ps_maria.result b/mysql-test/r/ps_maria.result new file mode 100644 index 00000000000..a5cb3050dcb --- /dev/null +++ b/mysql-test/r/ps_maria.result @@ -0,0 +1,3137 @@ +use test; +drop table if exists t1, t9 ; +create table t1 +( +a int, b varchar(30), +primary key(a) +) engine = 'MARIA' ; +create table t9 +( +c1 tinyint, c2 smallint, c3 mediumint, c4 int, +c5 integer, c6 bigint, c7 float, c8 double, +c9 double precision, c10 real, c11 decimal(7, 4), c12 numeric(8, 4), +c13 date, c14 datetime, c15 timestamp, c16 time, +c17 year, c18 tinyint, c19 bool, c20 char, +c21 char(10), c22 varchar(30), c23 tinyblob, c24 tinytext, +c25 blob, c26 text, c27 mediumblob, c28 mediumtext, +c29 longblob, c30 longtext, c31 enum('one', 'two', 'three'), +c32 set('monday', 'tuesday', 'wednesday'), +primary key(c1) +) engine = 'MARIA' ; +delete from t1 ; +insert into t1 values (1,'one'); +insert into t1 values (2,'two'); +insert into t1 values (3,'three'); +insert into t1 values (4,'four'); +commit ; +delete from t9 ; +insert into t9 +set c1= 1, c2= 1, c3= 1, c4= 1, c5= 1, c6= 1, c7= 1, c8= 1, c9= 1, +c10= 1, c11= 1, c12 = 1, +c13= '2004-02-29', c14= '2004-02-29 11:11:11', c15= '2004-02-29 11:11:11', +c16= '11:11:11', c17= '2004', +c18= 1, c19=true, c20= 'a', c21= '123456789a', +c22= '123456789a123456789b123456789c', c23= 'tinyblob', c24= 'tinytext', +c25= 'blob', c26= 'text', c27= 'mediumblob', c28= 'mediumtext', +c29= 'longblob', c30= 'longtext', c31='one', c32= 'monday'; +insert into t9 +set c1= 9, c2= 9, c3= 9, c4= 9, c5= 9, c6= 9, c7= 9, c8= 9, c9= 9, +c10= 9, c11= 9, c12 = 9, +c13= '2004-02-29', c14= '2004-02-29 11:11:11', c15= '2004-02-29 11:11:11', +c16= '11:11:11', c17= '2004', +c18= 1, c19=false, c20= 'a', c21= '123456789a', +c22= '123456789a123456789b123456789c', c23= 'tinyblob', c24= 'tinytext', +c25= 'blob', c26= 'text', c27= 'mediumblob', c28= 'mediumtext', +c29= 'longblob', c30= 'longtext', c31='two', c32= 'tuesday'; +commit ; +test_sequence +------ simple select tests ------ +prepare stmt1 from ' select * from t9 order by c1 ' ; +execute stmt1; +Catalog Database Table Table_alias Column Column_alias Type Length Max length Is_null Flags Decimals Charsetnr +def test t9 t9 c1 c1 1 4 1 N 49155 0 63 +def test t9 t9 c2 c2 2 6 1 Y 32768 0 63 +def test t9 t9 c3 c3 9 9 1 Y 32768 0 63 +def test t9 t9 c4 c4 3 11 1 Y 32768 0 63 +def test t9 t9 c5 c5 3 11 1 Y 32768 0 63 +def test t9 t9 c6 c6 8 20 1 Y 32768 0 63 +def test t9 t9 c7 c7 4 12 1 Y 32768 31 63 +def test t9 t9 c8 c8 5 22 1 Y 32768 31 63 +def test t9 t9 c9 c9 5 22 1 Y 32768 31 63 +def test t9 t9 c10 c10 5 22 1 Y 32768 31 63 +def test t9 t9 c11 c11 246 9 6 Y 0 4 63 +def test t9 t9 c12 c12 246 10 6 Y 0 4 63 +def test t9 t9 c13 c13 10 10 10 Y 128 0 63 +def test t9 t9 c14 c14 12 19 19 Y 128 0 63 +def test t9 t9 c15 c15 7 19 19 N 1249 0 63 +def test t9 t9 c16 c16 11 8 8 Y 128 0 63 +def test t9 t9 c17 c17 13 4 4 Y 32864 0 63 +def test t9 t9 c18 c18 1 4 1 Y 32768 0 63 +def test t9 t9 c19 c19 1 1 1 Y 32768 0 63 +def test t9 t9 c20 c20 254 1 1 Y 0 0 8 +def test t9 t9 c21 c21 254 10 10 Y 0 0 8 +def test t9 t9 c22 c22 253 30 30 Y 0 0 8 +def test t9 t9 c23 c23 252 255 8 Y 144 0 63 +def test t9 t9 c24 c24 252 255 8 Y 16 0 8 +def test t9 t9 c25 c25 252 65535 4 Y 144 0 63 +def test t9 t9 c26 c26 252 65535 4 Y 16 0 8 +def test t9 t9 c27 c27 252 16777215 10 Y 144 0 63 +def test t9 t9 c28 c28 252 16777215 10 Y 16 0 8 +def test t9 t9 c29 c29 252 4294967295 8 Y 144 0 63 +def test t9 t9 c30 c30 252 4294967295 8 Y 16 0 8 +def test t9 t9 c31 c31 254 5 3 Y 256 0 8 +def test t9 t9 c32 c32 254 24 7 Y 2048 0 8 +c1 c2 c3 c4 c5 c6 c7 c8 c9 c10 c11 c12 c13 c14 c15 c16 c17 c18 c19 c20 c21 c22 c23 c24 c25 c26 c27 c28 c29 c30 c31 c32 +1 1 1 1 1 1 1 1 1 1 1.0000 1.0000 2004-02-29 2004-02-29 11:11:11 2004-02-29 11:11:11 11:11:11 2004 1 1 a 123456789a 123456789a123456789b123456789c tinyblob tinytext blob text mediumblob mediumtext longblob longtext one monday +9 9 9 9 9 9 9 9 9 9 9.0000 9.0000 2004-02-29 2004-02-29 11:11:11 2004-02-29 11:11:11 11:11:11 2004 1 0 a 123456789a 123456789a123456789b123456789c tinyblob tinytext blob text mediumblob mediumtext longblob longtext two tuesday +set @arg00='SELECT' ; +@arg00 a from t1 where a=1; +ERROR 42000: You have an error in your SQL syntax; check the manual that corresponds to your MySQL server version for the right syntax to use near '@arg00 a from t1 where a=1' at line 1 +prepare stmt1 from ' ? a from t1 where a=1 '; +ERROR 42000: You have an error in your SQL syntax; check the manual that corresponds to your MySQL server version for the right syntax to use near '? a from t1 where a=1' at line 1 +set @arg00=1 ; +select @arg00, b from t1 where a=1 ; +@arg00 b +1 one +prepare stmt1 from ' select ?, b from t1 where a=1 ' ; +execute stmt1 using @arg00 ; +? b +1 one +set @arg00='lion' ; +select @arg00, b from t1 where a=1 ; +@arg00 b +lion one +prepare stmt1 from ' select ?, b from t1 where a=1 ' ; +execute stmt1 using @arg00 ; +? b +lion one +set @arg00=NULL ; +select @arg00, b from t1 where a=1 ; +@arg00 b +NULL one +prepare stmt1 from ' select ?, b from t1 where a=1 ' ; +execute stmt1 using @arg00 ; +? b +NULL one +set @arg00=1 ; +select b, a - @arg00 from t1 where a=1 ; +b a - @arg00 +one 0 +prepare stmt1 from ' select b, a - ? from t1 where a=1 ' ; +execute stmt1 using @arg00 ; +b a - ? +one 0 +set @arg00=null ; +select @arg00 as my_col ; +my_col +NULL +prepare stmt1 from ' select ? as my_col'; +execute stmt1 using @arg00 ; +my_col +NULL +select @arg00 + 1 as my_col ; +my_col +NULL +prepare stmt1 from ' select ? + 1 as my_col'; +execute stmt1 using @arg00 ; +my_col +NULL +select 1 + @arg00 as my_col ; +my_col +NULL +prepare stmt1 from ' select 1 + ? as my_col'; +execute stmt1 using @arg00 ; +my_col +NULL +set @arg00='MySQL' ; +select substr(@arg00,1,2) from t1 where a=1 ; +substr(@arg00,1,2) +My +prepare stmt1 from ' select substr(?,1,2) from t1 where a=1 ' ; +execute stmt1 using @arg00 ; +substr(?,1,2) +My +set @arg00=3 ; +select substr('MySQL',@arg00,5) from t1 where a=1 ; +substr('MySQL',@arg00,5) +SQL +prepare stmt1 from ' select substr(''MySQL'',?,5) from t1 where a=1 ' ; +execute stmt1 using @arg00 ; +substr('MySQL',?,5) +SQL +select substr('MySQL',1,@arg00) from t1 where a=1 ; +substr('MySQL',1,@arg00) +MyS +prepare stmt1 from ' select substr(''MySQL'',1,?) from t1 where a=1 ' ; +execute stmt1 using @arg00 ; +substr('MySQL',1,?) +MyS +set @arg00='MySQL' ; +select a , concat(@arg00,b) from t1 order by a; +a concat(@arg00,b) +1 MySQLone +2 MySQLtwo +3 MySQLthree +4 MySQLfour +prepare stmt1 from ' select a , concat(?,b) from t1 order by a ' ; +execute stmt1 using @arg00; +a concat(?,b) +1 MySQLone +2 MySQLtwo +3 MySQLthree +4 MySQLfour +select a , concat(b,@arg00) from t1 order by a ; +a concat(b,@arg00) +1 oneMySQL +2 twoMySQL +3 threeMySQL +4 fourMySQL +prepare stmt1 from ' select a , concat(b,?) from t1 order by a ' ; +execute stmt1 using @arg00; +a concat(b,?) +1 oneMySQL +2 twoMySQL +3 threeMySQL +4 fourMySQL +set @arg00='MySQL' ; +select group_concat(@arg00,b order by a) from t1 +group by 'a' ; +group_concat(@arg00,b order by a) +MySQLone,MySQLtwo,MySQLthree,MySQLfour +prepare stmt1 from ' select group_concat(?,b order by a) from t1 +group by ''a'' ' ; +execute stmt1 using @arg00; +group_concat(?,b order by a) +MySQLone,MySQLtwo,MySQLthree,MySQLfour +select group_concat(b,@arg00 order by a) from t1 +group by 'a' ; +group_concat(b,@arg00 order by a) +oneMySQL,twoMySQL,threeMySQL,fourMySQL +prepare stmt1 from ' select group_concat(b,? order by a) from t1 +group by ''a'' ' ; +execute stmt1 using @arg00; +group_concat(b,? order by a) +oneMySQL,twoMySQL,threeMySQL,fourMySQL +set @arg00='first' ; +set @arg01='second' ; +set @arg02=NULL; +select @arg00, @arg01 from t1 where a=1 ; +@arg00 @arg01 +first second +prepare stmt1 from ' select ?, ? from t1 where a=1 ' ; +execute stmt1 using @arg00, @arg01 ; +? ? +first second +execute stmt1 using @arg02, @arg01 ; +? ? +NULL second +execute stmt1 using @arg00, @arg02 ; +? ? +first NULL +execute stmt1 using @arg02, @arg02 ; +? ? +NULL NULL +drop table if exists t5 ; +create table t5 (id1 int(11) not null default '0', +value2 varchar(100), value1 varchar(100)) ; +insert into t5 values (1,'hh','hh'),(2,'hh','hh'), +(1,'ii','ii'),(2,'ii','ii') ; +prepare stmt1 from ' select id1,value1 from t5 where id1=? or value1=? order by id1,value1 ' ; +set @arg00=1 ; +set @arg01='hh' ; +execute stmt1 using @arg00, @arg01 ; +id1 value1 +1 hh +1 ii +2 hh +drop table t5 ; +drop table if exists t5 ; +create table t5(session_id char(9) not null) ; +insert into t5 values ('abc') ; +prepare stmt1 from ' select * from t5 +where ?=''1111'' and session_id = ''abc'' ' ; +set @arg00='abc' ; +execute stmt1 using @arg00 ; +session_id +set @arg00='1111' ; +execute stmt1 using @arg00 ; +session_id +abc +set @arg00='abc' ; +execute stmt1 using @arg00 ; +session_id +drop table t5 ; +set @arg00='FROM' ; +select a @arg00 t1 where a=1 ; +ERROR 42000: You have an error in your SQL syntax; check the manual that corresponds to your MySQL server version for the right syntax to use near '@arg00 t1 where a=1' at line 1 +prepare stmt1 from ' select a ? t1 where a=1 ' ; +ERROR 42000: You have an error in your SQL syntax; check the manual that corresponds to your MySQL server version for the right syntax to use near '? t1 where a=1' at line 1 +set @arg00='t1' ; +select a from @arg00 where a=1 ; +ERROR 42000: You have an error in your SQL syntax; check the manual that corresponds to your MySQL server version for the right syntax to use near '@arg00 where a=1' at line 1 +prepare stmt1 from ' select a from ? where a=1 ' ; +ERROR 42000: You have an error in your SQL syntax; check the manual that corresponds to your MySQL server version for the right syntax to use near '? where a=1' at line 1 +set @arg00='WHERE' ; +select a from t1 @arg00 a=1 ; +ERROR 42000: You have an error in your SQL syntax; check the manual that corresponds to your MySQL server version for the right syntax to use near '@arg00 a=1' at line 1 +prepare stmt1 from ' select a from t1 ? a=1 ' ; +ERROR 42000: You have an error in your SQL syntax; check the manual that corresponds to your MySQL server version for the right syntax to use near '? a=1' at line 1 +set @arg00=1 ; +select a FROM t1 where a=@arg00 ; +a +1 +prepare stmt1 from ' select a FROM t1 where a=? ' ; +execute stmt1 using @arg00 ; +a +1 +set @arg00=1000 ; +execute stmt1 using @arg00 ; +a +set @arg00=NULL ; +select a FROM t1 where a=@arg00 ; +a +prepare stmt1 from ' select a FROM t1 where a=? ' ; +execute stmt1 using @arg00 ; +a +set @arg00=4 ; +select a FROM t1 where a=sqrt(@arg00) ; +a +2 +prepare stmt1 from ' select a FROM t1 where a=sqrt(?) ' ; +execute stmt1 using @arg00 ; +a +2 +set @arg00=NULL ; +select a FROM t1 where a=sqrt(@arg00) ; +a +prepare stmt1 from ' select a FROM t1 where a=sqrt(?) ' ; +execute stmt1 using @arg00 ; +a +set @arg00=2 ; +set @arg01=3 ; +select a FROM t1 where a in (@arg00,@arg01) order by a; +a +2 +3 +prepare stmt1 from ' select a FROM t1 where a in (?,?) order by a '; +execute stmt1 using @arg00, @arg01; +a +2 +3 +set @arg00= 'one' ; +set @arg01= 'two' ; +set @arg02= 'five' ; +prepare stmt1 from ' select b FROM t1 where b in (?,?,?) order by b ' ; +execute stmt1 using @arg00, @arg01, @arg02 ; +b +one +two +prepare stmt1 from ' select b FROM t1 where b like ? '; +set @arg00='two' ; +execute stmt1 using @arg00 ; +b +two +set @arg00='tw%' ; +execute stmt1 using @arg00 ; +b +two +set @arg00='%wo' ; +execute stmt1 using @arg00 ; +b +two +set @arg00=null ; +insert into t9 set c1= 0, c5 = NULL ; +select c5 from t9 where c5 > NULL ; +c5 +prepare stmt1 from ' select c5 from t9 where c5 > ? '; +execute stmt1 using @arg00 ; +c5 +select c5 from t9 where c5 < NULL ; +c5 +prepare stmt1 from ' select c5 from t9 where c5 < ? '; +execute stmt1 using @arg00 ; +c5 +select c5 from t9 where c5 = NULL ; +c5 +prepare stmt1 from ' select c5 from t9 where c5 = ? '; +execute stmt1 using @arg00 ; +c5 +select c5 from t9 where c5 <=> NULL ; +c5 +NULL +prepare stmt1 from ' select c5 from t9 where c5 <=> ? '; +execute stmt1 using @arg00 ; +c5 +NULL +delete from t9 where c1= 0 ; +set @arg00='>' ; +select a FROM t1 where a @arg00 1 ; +ERROR 42000: You have an error in your SQL syntax; check the manual that corresponds to your MySQL server version for the right syntax to use near '@arg00 1' at line 1 +prepare stmt1 from ' select a FROM t1 where a ? 1 ' ; +ERROR 42000: You have an error in your SQL syntax; check the manual that corresponds to your MySQL server version for the right syntax to use near '? 1' at line 1 +set @arg00=1 ; +select a,b FROM t1 where a is not NULL +AND b is not NULL group by a - @arg00 ; +a b +1 one +2 two +3 three +4 four +prepare stmt1 from ' select a,b FROM t1 where a is not NULL +AND b is not NULL group by a - ? ' ; +execute stmt1 using @arg00 ; +a b +1 one +2 two +3 three +4 four +set @arg00='two' ; +select a,b FROM t1 where a is not NULL +AND b is not NULL having b <> @arg00 order by a ; +a b +1 one +3 three +4 four +prepare stmt1 from ' select a,b FROM t1 where a is not NULL +AND b is not NULL having b <> ? order by a ' ; +execute stmt1 using @arg00 ; +a b +1 one +3 three +4 four +set @arg00=1 ; +select a,b FROM t1 where a is not NULL +AND b is not NULL order by a - @arg00 ; +a b +1 one +2 two +3 three +4 four +prepare stmt1 from ' select a,b FROM t1 where a is not NULL +AND b is not NULL order by a - ? ' ; +execute stmt1 using @arg00 ; +a b +1 one +2 two +3 three +4 four +set @arg00=2 ; +select a,b from t1 order by 2 ; +a b +4 four +1 one +3 three +2 two +prepare stmt1 from ' select a,b from t1 +order by ? '; +execute stmt1 using @arg00; +a b +4 four +1 one +3 three +2 two +set @arg00=1 ; +execute stmt1 using @arg00; +a b +1 one +2 two +3 three +4 four +set @arg00=0 ; +execute stmt1 using @arg00; +ERROR 42S22: Unknown column '?' in 'order clause' +set @arg00=1; +prepare stmt1 from ' select a,b from t1 order by a +limit 1 '; +execute stmt1 ; +a b +1 one +prepare stmt1 from ' select a,b from t1 order by a limit ? '; +execute stmt1 using @arg00; +a b +1 one +set @arg00='b' ; +set @arg01=0 ; +set @arg02=2 ; +set @arg03=2 ; +select sum(a), @arg00 from t1 where a > @arg01 +and b is not null group by substr(b,@arg02) +having sum(a) <> @arg03 ; +sum(a) @arg00 +3 b +1 b +4 b +prepare stmt1 from ' select sum(a), ? from t1 where a > ? +and b is not null group by substr(b,?) +having sum(a) <> ? '; +execute stmt1 using @arg00, @arg01, @arg02, @arg03; +sum(a) ? +3 b +1 b +4 b +test_sequence +------ join tests ------ +select first.a as a1, second.a as a2 +from t1 first, t1 second +where first.a = second.a order by a1 ; +a1 a2 +1 1 +2 2 +3 3 +4 4 +prepare stmt1 from ' select first.a as a1, second.a as a2 + from t1 first, t1 second + where first.a = second.a order by a1 '; +execute stmt1 ; +a1 a2 +1 1 +2 2 +3 3 +4 4 +set @arg00='ABC'; +set @arg01='two'; +set @arg02='one'; +select first.a, @arg00, second.a FROM t1 first, t1 second +where @arg01 = first.b or first.a = second.a or second.b = @arg02 +order by second.a, first.a; +a @arg00 a +1 ABC 1 +2 ABC 1 +3 ABC 1 +4 ABC 1 +2 ABC 2 +2 ABC 3 +3 ABC 3 +2 ABC 4 +4 ABC 4 +prepare stmt1 from ' select first.a, ?, second.a FROM t1 first, t1 second + where ? = first.b or first.a = second.a or second.b = ? + order by second.a, first.a'; +execute stmt1 using @arg00, @arg01, @arg02; +a ? a +1 ABC 1 +2 ABC 1 +3 ABC 1 +4 ABC 1 +2 ABC 2 +2 ABC 3 +3 ABC 3 +2 ABC 4 +4 ABC 4 +drop table if exists t2 ; +create table t2 as select * from t1 ; +set @query1= 'SELECT * FROM t2 join t1 on (t1.a=t2.a) order by t2.a ' ; +set @query2= 'SELECT * FROM t2 natural join t1 order by t2.a ' ; +set @query3= 'SELECT * FROM t2 join t1 using(a) order by t2.a ' ; +set @query4= 'SELECT * FROM t2 left join t1 on(t1.a=t2.a) order by t2.a ' ; +set @query5= 'SELECT * FROM t2 natural left join t1 order by t2.a ' ; +set @query6= 'SELECT * FROM t2 left join t1 using(a) order by t2.a ' ; +set @query7= 'SELECT * FROM t2 right join t1 on(t1.a=t2.a) order by t2.a ' ; +set @query8= 'SELECT * FROM t2 natural right join t1 order by t2.a ' ; +set @query9= 'SELECT * FROM t2 right join t1 using(a) order by t2.a ' ; +the join statement is: +SELECT * FROM t2 right join t1 using(a) order by t2.a +prepare stmt1 from @query9 ; +execute stmt1 ; +a b b +1 one one +2 two two +3 three three +4 four four +execute stmt1 ; +a b b +1 one one +2 two two +3 three three +4 four four +execute stmt1 ; +a b b +1 one one +2 two two +3 three three +4 four four +the join statement is: +SELECT * FROM t2 natural right join t1 order by t2.a +prepare stmt1 from @query8 ; +execute stmt1 ; +a b +1 one +2 two +3 three +4 four +execute stmt1 ; +a b +1 one +2 two +3 three +4 four +execute stmt1 ; +a b +1 one +2 two +3 three +4 four +the join statement is: +SELECT * FROM t2 right join t1 on(t1.a=t2.a) order by t2.a +prepare stmt1 from @query7 ; +execute stmt1 ; +a b a b +1 one 1 one +2 two 2 two +3 three 3 three +4 four 4 four +execute stmt1 ; +a b a b +1 one 1 one +2 two 2 two +3 three 3 three +4 four 4 four +execute stmt1 ; +a b a b +1 one 1 one +2 two 2 two +3 three 3 three +4 four 4 four +the join statement is: +SELECT * FROM t2 left join t1 using(a) order by t2.a +prepare stmt1 from @query6 ; +execute stmt1 ; +a b b +1 one one +2 two two +3 three three +4 four four +execute stmt1 ; +a b b +1 one one +2 two two +3 three three +4 four four +execute stmt1 ; +a b b +1 one one +2 two two +3 three three +4 four four +the join statement is: +SELECT * FROM t2 natural left join t1 order by t2.a +prepare stmt1 from @query5 ; +execute stmt1 ; +a b +1 one +2 two +3 three +4 four +execute stmt1 ; +a b +1 one +2 two +3 three +4 four +execute stmt1 ; +a b +1 one +2 two +3 three +4 four +the join statement is: +SELECT * FROM t2 left join t1 on(t1.a=t2.a) order by t2.a +prepare stmt1 from @query4 ; +execute stmt1 ; +a b a b +1 one 1 one +2 two 2 two +3 three 3 three +4 four 4 four +execute stmt1 ; +a b a b +1 one 1 one +2 two 2 two +3 three 3 three +4 four 4 four +execute stmt1 ; +a b a b +1 one 1 one +2 two 2 two +3 three 3 three +4 four 4 four +the join statement is: +SELECT * FROM t2 join t1 using(a) order by t2.a +prepare stmt1 from @query3 ; +execute stmt1 ; +a b b +1 one one +2 two two +3 three three +4 four four +execute stmt1 ; +a b b +1 one one +2 two two +3 three three +4 four four +execute stmt1 ; +a b b +1 one one +2 two two +3 three three +4 four four +the join statement is: +SELECT * FROM t2 natural join t1 order by t2.a +prepare stmt1 from @query2 ; +execute stmt1 ; +a b +1 one +2 two +3 three +4 four +execute stmt1 ; +a b +1 one +2 two +3 three +4 four +execute stmt1 ; +a b +1 one +2 two +3 three +4 four +the join statement is: +SELECT * FROM t2 join t1 on (t1.a=t2.a) order by t2.a +prepare stmt1 from @query1 ; +execute stmt1 ; +a b a b +1 one 1 one +2 two 2 two +3 three 3 three +4 four 4 four +execute stmt1 ; +a b a b +1 one 1 one +2 two 2 two +3 three 3 three +4 four 4 four +execute stmt1 ; +a b a b +1 one 1 one +2 two 2 two +3 three 3 three +4 four 4 four +drop table t2 ; +test_sequence +------ subquery tests ------ +prepare stmt1 from ' select a, b FROM t1 outer_table where + a = (select a from t1 where b = ''two'') '; +execute stmt1 ; +a b +2 two +set @arg00='two' ; +select a, b FROM t1 outer_table where +a = (select a from t1 where b = 'two' ) and b=@arg00 ; +a b +2 two +prepare stmt1 from ' select a, b FROM t1 outer_table where + a = (select a from t1 where b = ''two'') and b=? '; +execute stmt1 using @arg00; +a b +2 two +set @arg00='two' ; +select a, b FROM t1 outer_table where +a = (select a from t1 where b = @arg00 ) and b='two' ; +a b +2 two +prepare stmt1 from ' select a, b FROM t1 outer_table where + a = (select a from t1 where b = ? ) and b=''two'' ' ; +execute stmt1 using @arg00; +a b +2 two +set @arg00=3 ; +set @arg01='three' ; +select a,b FROM t1 where (a,b) in (select 3, 'three'); +a b +3 three +select a FROM t1 where (a,b) in (select @arg00,@arg01); +a +3 +prepare stmt1 from ' select a FROM t1 where (a,b) in (select ?, ?) '; +execute stmt1 using @arg00, @arg01; +a +3 +set @arg00=1 ; +set @arg01='two' ; +set @arg02=2 ; +set @arg03='two' ; +select a, @arg00, b FROM t1 outer_table where +b=@arg01 and a = (select @arg02 from t1 where b = @arg03 ) ; +a @arg00 b +2 1 two +prepare stmt1 from ' select a, ?, b FROM t1 outer_table where + b=? and a = (select ? from t1 where b = ? ) ' ; +execute stmt1 using @arg00, @arg01, @arg02, @arg03 ; +a ? b +2 1 two +prepare stmt1 from 'select c4 FROM t9 where + c13 = (select MAX(b) from t1 where a = ?) and c22 = ? ' ; +execute stmt1 using @arg01, @arg02; +c4 +prepare stmt1 from ' select a, b FROM t1 outer_table where + a = (select a from t1 where b = outer_table.b ) order by a '; +execute stmt1 ; +a b +1 one +2 two +3 three +4 four +prepare stmt1 from ' SELECT a as ccc from t1 where a+1= + (SELECT 1+ccc from t1 where ccc+1=a+1 and a=1) '; +execute stmt1 ; +ccc +1 +deallocate prepare stmt1 ; +prepare stmt1 from ' SELECT a as ccc from t1 where a+1= + (SELECT 1+ccc from t1 where ccc+1=a+1 and a=1) '; +execute stmt1 ; +ccc +1 +deallocate prepare stmt1 ; +prepare stmt1 from ' SELECT a as ccc from t1 where a+1= + (SELECT 1+ccc from t1 where ccc+1=a+1 and a=1) '; +execute stmt1 ; +ccc +1 +deallocate prepare stmt1 ; +set @arg00='two' ; +select a, b FROM t1 outer_table where +a = (select a from t1 where b = outer_table.b ) and b=@arg00 ; +a b +2 two +prepare stmt1 from ' select a, b FROM t1 outer_table where + a = (select a from t1 where b = outer_table.b) and b=? '; +execute stmt1 using @arg00; +a b +2 two +set @arg00=2 ; +select a, b FROM t1 outer_table where +a = (select a from t1 where a = @arg00 and b = outer_table.b) and b='two' ; +a b +2 two +prepare stmt1 from ' select a, b FROM t1 outer_table where + a = (select a from t1 where a = ? and b = outer_table.b) and b=''two'' ' ; +execute stmt1 using @arg00; +a b +2 two +set @arg00=2 ; +select a, b FROM t1 outer_table where +a = (select a from t1 where outer_table.a = @arg00 and a=2) and b='two' ; +a b +2 two +prepare stmt1 from ' select a, b FROM t1 outer_table where + a = (select a from t1 where outer_table.a = ? and a=2) and b=''two'' ' ; +execute stmt1 using @arg00; +a b +2 two +set @arg00=1 ; +set @arg01='two' ; +set @arg02=2 ; +set @arg03='two' ; +select a, @arg00, b FROM t1 outer_table where +b=@arg01 and a = (select @arg02 from t1 where outer_table.b = @arg03 +and outer_table.a=a ) ; +a @arg00 b +2 1 two +prepare stmt1 from ' select a, ?, b FROM t1 outer_table where + b=? and a = (select ? from t1 where outer_table.b = ? + and outer_table.a=a ) ' ; +execute stmt1 using @arg00, @arg01, @arg02, @arg03 ; +a ? b +2 1 two +set @arg00=1 ; +set @arg01=0 ; +select a, @arg00 +from ( select a - @arg00 as a from t1 where a=@arg00 ) as t2 +where a=@arg01; +a @arg00 +0 1 +prepare stmt1 from ' select a, ? + from ( select a - ? as a from t1 where a=? ) as t2 + where a=? '; +execute stmt1 using @arg00, @arg00, @arg00, @arg01 ; +a ? +0 1 +drop table if exists t2 ; +create table t2 as select * from t1; +prepare stmt1 from ' select a in (select a from t2) from t1 ' ; +execute stmt1 ; +a in (select a from t2) +1 +1 +1 +1 +drop table if exists t5, t6, t7 ; +create table t5 (a int , b int) ; +create table t6 like t5 ; +create table t7 like t5 ; +insert into t5 values (0, 100), (1, 2), (1, 3), (2, 2), (2, 7), +(2, -1), (3, 10) ; +insert into t6 values (0, 0), (1, 1), (2, 1), (3, 1), (4, 1) ; +insert into t7 values (3, 3), (2, 2), (1, 1) ; +prepare stmt1 from ' select a, (select count(distinct t5.b) as sum from t5, t6 + where t5.a=t6.a and t6.b > 0 and t5.a <= t7.b + group by t5.a order by sum limit 1) from t7 ' ; +execute stmt1 ; +a (select count(distinct t5.b) as sum from t5, t6 + where t5.a=t6.a and t6.b > 0 and t5.a <= t7.b + group by t5.a order by sum limit 1) +3 1 +2 2 +1 2 +execute stmt1 ; +a (select count(distinct t5.b) as sum from t5, t6 + where t5.a=t6.a and t6.b > 0 and t5.a <= t7.b + group by t5.a order by sum limit 1) +3 1 +2 2 +1 2 +execute stmt1 ; +a (select count(distinct t5.b) as sum from t5, t6 + where t5.a=t6.a and t6.b > 0 and t5.a <= t7.b + group by t5.a order by sum limit 1) +3 1 +2 2 +1 2 +drop table t5, t6, t7 ; +drop table if exists t2 ; +create table t2 as select * from t9; +set @stmt= ' SELECT + (SELECT SUM(c1 + c12 + 0.0) FROM t2 + where (t9.c2 - 0e-3) = t2.c2 + GROUP BY t9.c15 LIMIT 1) as scalar_s, + exists (select 1.0e+0 from t2 + where t2.c3 * 9.0000000000 = t9.c4) as exists_s, + c5 * 4 in (select c6 + 0.3e+1 from t2) as in_s, + (c7 - 4, c8 - 4) in (select c9 + 4.0, c10 + 40e-1 from t2) as in_row_s +FROM t9, +(select c25 x, c32 y from t2) tt WHERE x = c25 ' ; +prepare stmt1 from @stmt ; +execute stmt1 ; +execute stmt1 ; +set @stmt= concat('explain ',@stmt); +prepare stmt1 from @stmt ; +execute stmt1 ; +execute stmt1 ; +set @stmt= ' SELECT + (SELECT SUM(c1+c12+?) FROM t2 where (t9.c2-?)=t2.c2 + GROUP BY t9.c15 LIMIT 1) as scalar_s, + exists (select ? from t2 + where t2.c3*?=t9.c4) as exists_s, + c5*? in (select c6+? from t2) as in_s, + (c7-?, c8-?) in (select c9+?, c10+? from t2) as in_row_s +FROM t9, +(select c25 x, c32 y from t2) tt WHERE x =c25 ' ; +set @arg00= 0.0 ; +set @arg01= 0e-3 ; +set @arg02= 1.0e+0 ; +set @arg03= 9.0000000000 ; +set @arg04= 4 ; +set @arg05= 0.3e+1 ; +set @arg06= 4 ; +set @arg07= 4 ; +set @arg08= 4.0 ; +set @arg09= 40e-1 ; +prepare stmt1 from @stmt ; +execute stmt1 using @arg00, @arg01, @arg02, @arg03, @arg04, @arg05, @arg06, +@arg07, @arg08, @arg09 ; +execute stmt1 using @arg00, @arg01, @arg02, @arg03, @arg04, @arg05, @arg06, +@arg07, @arg08, @arg09 ; +set @stmt= concat('explain ',@stmt); +prepare stmt1 from @stmt ; +execute stmt1 using @arg00, @arg01, @arg02, @arg03, @arg04, @arg05, @arg06, +@arg07, @arg08, @arg09 ; +execute stmt1 using @arg00, @arg01, @arg02, @arg03, @arg04, @arg05, @arg06, +@arg07, @arg08, @arg09 ; +drop table t2 ; +select 1 < (select a from t1) ; +ERROR 21000: Subquery returns more than 1 row +prepare stmt1 from ' select 1 < (select a from t1) ' ; +execute stmt1 ; +ERROR 21000: Subquery returns more than 1 row +select 1 as my_col ; +my_col +1 +test_sequence +------ union tests ------ +prepare stmt1 from ' select a FROM t1 where a=1 + union distinct + select a FROM t1 where a=1 '; +execute stmt1 ; +a +1 +execute stmt1 ; +a +1 +prepare stmt1 from ' select a FROM t1 where a=1 + union all + select a FROM t1 where a=1 '; +execute stmt1 ; +a +1 +1 +prepare stmt1 from ' SELECT 1, 2 union SELECT 1 ' ; +ERROR 21000: The used SELECT statements have a different number of columns +prepare stmt1 from ' SELECT 1 union SELECT 1, 2 ' ; +ERROR 21000: The used SELECT statements have a different number of columns +prepare stmt1 from ' SELECT * from t1 union SELECT 1 ' ; +ERROR 21000: The used SELECT statements have a different number of columns +prepare stmt1 from ' SELECT 1 union SELECT * from t1 ' ; +ERROR 21000: The used SELECT statements have a different number of columns +set @arg00=1 ; +select @arg00 FROM t1 where a=1 +union distinct +select 1 FROM t1 where a=1; +@arg00 +1 +prepare stmt1 from ' select ? FROM t1 where a=1 + union distinct + select 1 FROM t1 where a=1 ' ; +execute stmt1 using @arg00; +? +1 +set @arg00=1 ; +select 1 FROM t1 where a=1 +union distinct +select @arg00 FROM t1 where a=1; +1 +1 +prepare stmt1 from ' select 1 FROM t1 where a=1 + union distinct + select ? FROM t1 where a=1 ' ; +execute stmt1 using @arg00; +1 +1 +set @arg00='a' ; +select @arg00 FROM t1 where a=1 +union distinct +select @arg00 FROM t1 where a=1; +@arg00 +a +prepare stmt1 from ' select ? FROM t1 where a=1 + union distinct + select ? FROM t1 where a=1 '; +execute stmt1 using @arg00, @arg00; +? +a +prepare stmt1 from ' select ? + union distinct + select ? '; +execute stmt1 using @arg00, @arg00; +? +a +set @arg00='a' ; +set @arg01=1 ; +set @arg02='a' ; +set @arg03=2 ; +select @arg00 FROM t1 where a=@arg01 +union distinct +select @arg02 FROM t1 where a=@arg03; +@arg00 +a +prepare stmt1 from ' select ? FROM t1 where a=? + union distinct + select ? FROM t1 where a=? ' ; +execute stmt1 using @arg00, @arg01, @arg02, @arg03; +? +a +set @arg00=1 ; +prepare stmt1 from ' select sum(a) + 200, ? from t1 +union distinct +select sum(a) + 200, 1 from t1 +group by b ' ; +execute stmt1 using @arg00; +sum(a) + 200 ? +210 1 +204 1 +201 1 +203 1 +202 1 +set @Oporto='Oporto' ; +set @Lisboa='Lisboa' ; +set @0=0 ; +set @1=1 ; +set @2=2 ; +set @3=3 ; +set @4=4 ; +select @Oporto,@Lisboa,@0,@1,@2,@3,@4 ; +@Oporto @Lisboa @0 @1 @2 @3 @4 +Oporto Lisboa 0 1 2 3 4 +select sum(a) + 200 as the_sum, @Oporto as the_town from t1 +group by b +union distinct +select sum(a) + 200, @Lisboa from t1 +group by b ; +the_sum the_town +204 Oporto +201 Oporto +203 Oporto +202 Oporto +204 Lisboa +201 Lisboa +203 Lisboa +202 Lisboa +prepare stmt1 from ' select sum(a) + 200 as the_sum, ? as the_town from t1 + group by b + union distinct + select sum(a) + 200, ? from t1 + group by b ' ; +execute stmt1 using @Oporto, @Lisboa; +the_sum the_town +204 Oporto +201 Oporto +203 Oporto +202 Oporto +204 Lisboa +201 Lisboa +203 Lisboa +202 Lisboa +select sum(a) + 200 as the_sum, @Oporto as the_town from t1 +where a > @1 +group by b +union distinct +select sum(a) + 200, @Lisboa from t1 +where a > @2 +group by b ; +the_sum the_town +204 Oporto +203 Oporto +202 Oporto +204 Lisboa +203 Lisboa +prepare stmt1 from ' select sum(a) + 200 as the_sum, ? as the_town from t1 + where a > ? + group by b + union distinct + select sum(a) + 200, ? from t1 + where a > ? + group by b ' ; +execute stmt1 using @Oporto, @1, @Lisboa, @2; +the_sum the_town +204 Oporto +203 Oporto +202 Oporto +204 Lisboa +203 Lisboa +select sum(a) + 200 as the_sum, @Oporto as the_town from t1 +where a > @1 +group by b +having avg(a) > @2 +union distinct +select sum(a) + 200, @Lisboa from t1 +where a > @2 +group by b +having avg(a) > @3; +the_sum the_town +204 Oporto +203 Oporto +204 Lisboa +prepare stmt1 from ' select sum(a) + 200 as the_sum, ? as the_town from t1 + where a > ? + group by b + having avg(a) > ? + union distinct + select sum(a) + 200, ? from t1 + where a > ? + group by b + having avg(a) > ? '; +execute stmt1 using @Oporto, @1, @2, @Lisboa, @2, @3; +the_sum the_town +204 Oporto +203 Oporto +204 Lisboa +test_sequence +------ explain select tests ------ +prepare stmt1 from ' explain select * from t9 ' ; +execute stmt1; +Catalog Database Table Table_alias Column Column_alias Type Length Max length Is_null Flags Decimals Charsetnr +def id 8 3 1 N 32929 0 63 +def select_type 253 19 6 N 1 31 8 +def table 253 64 2 Y 0 31 8 +def type 253 10 3 Y 0 31 8 +def possible_keys 253 4096 0 Y 0 31 8 +def key 253 64 0 Y 0 31 8 +def key_len 253 4096 0 Y 128 31 63 +def ref 253 1024 0 Y 0 31 8 +def rows 8 10 1 Y 32928 0 63 +def Extra 253 255 0 N 1 31 8 +id select_type table type possible_keys key key_len ref rows Extra +1 SIMPLE t9 ALL NULL NULL NULL NULL 2 +drop table if exists t2 ; +create table t2 (s varchar(25), fulltext(s)) +ENGINE = 'MARIA' ; +insert into t2 values ('Gravedigger'), ('Greed'),('Hollow Dogs') ; +commit ; +prepare stmt1 from ' select s from t2 where match (s) against (?) ' ; +set @arg00='Dogs' ; +execute stmt1 using @arg00 ; +s +Hollow Dogs +prepare stmt1 from ' SELECT s FROM t2 +where match (s) against (concat(?,''digger'')) '; +set @arg00='Grave' ; +execute stmt1 using @arg00 ; +s +Gravedigger +drop table t2 ; +test_sequence +------ delete tests ------ +delete from t1 ; +insert into t1 values (1,'one'); +insert into t1 values (2,'two'); +insert into t1 values (3,'three'); +insert into t1 values (4,'four'); +commit ; +delete from t9 ; +insert into t9 +set c1= 1, c2= 1, c3= 1, c4= 1, c5= 1, c6= 1, c7= 1, c8= 1, c9= 1, +c10= 1, c11= 1, c12 = 1, +c13= '2004-02-29', c14= '2004-02-29 11:11:11', c15= '2004-02-29 11:11:11', +c16= '11:11:11', c17= '2004', +c18= 1, c19=true, c20= 'a', c21= '123456789a', +c22= '123456789a123456789b123456789c', c23= 'tinyblob', c24= 'tinytext', +c25= 'blob', c26= 'text', c27= 'mediumblob', c28= 'mediumtext', +c29= 'longblob', c30= 'longtext', c31='one', c32= 'monday'; +insert into t9 +set c1= 9, c2= 9, c3= 9, c4= 9, c5= 9, c6= 9, c7= 9, c8= 9, c9= 9, +c10= 9, c11= 9, c12 = 9, +c13= '2004-02-29', c14= '2004-02-29 11:11:11', c15= '2004-02-29 11:11:11', +c16= '11:11:11', c17= '2004', +c18= 1, c19=false, c20= 'a', c21= '123456789a', +c22= '123456789a123456789b123456789c', c23= 'tinyblob', c24= 'tinytext', +c25= 'blob', c26= 'text', c27= 'mediumblob', c28= 'mediumtext', +c29= 'longblob', c30= 'longtext', c31='two', c32= 'tuesday'; +commit ; +prepare stmt1 from 'delete from t1 where a=2' ; +execute stmt1; +select a,b from t1 where a=2; +a b +execute stmt1; +insert into t1 values(0,NULL); +set @arg00=NULL; +prepare stmt1 from 'delete from t1 where b=?' ; +execute stmt1 using @arg00; +select a,b from t1 where b is NULL ; +a b +0 NULL +set @arg00='one'; +execute stmt1 using @arg00; +select a,b from t1 where b=@arg00; +a b +prepare stmt1 from 'truncate table t1' ; +test_sequence +------ update tests ------ +delete from t1 ; +insert into t1 values (1,'one'); +insert into t1 values (2,'two'); +insert into t1 values (3,'three'); +insert into t1 values (4,'four'); +commit ; +delete from t9 ; +insert into t9 +set c1= 1, c2= 1, c3= 1, c4= 1, c5= 1, c6= 1, c7= 1, c8= 1, c9= 1, +c10= 1, c11= 1, c12 = 1, +c13= '2004-02-29', c14= '2004-02-29 11:11:11', c15= '2004-02-29 11:11:11', +c16= '11:11:11', c17= '2004', +c18= 1, c19=true, c20= 'a', c21= '123456789a', +c22= '123456789a123456789b123456789c', c23= 'tinyblob', c24= 'tinytext', +c25= 'blob', c26= 'text', c27= 'mediumblob', c28= 'mediumtext', +c29= 'longblob', c30= 'longtext', c31='one', c32= 'monday'; +insert into t9 +set c1= 9, c2= 9, c3= 9, c4= 9, c5= 9, c6= 9, c7= 9, c8= 9, c9= 9, +c10= 9, c11= 9, c12 = 9, +c13= '2004-02-29', c14= '2004-02-29 11:11:11', c15= '2004-02-29 11:11:11', +c16= '11:11:11', c17= '2004', +c18= 1, c19=false, c20= 'a', c21= '123456789a', +c22= '123456789a123456789b123456789c', c23= 'tinyblob', c24= 'tinytext', +c25= 'blob', c26= 'text', c27= 'mediumblob', c28= 'mediumtext', +c29= 'longblob', c30= 'longtext', c31='two', c32= 'tuesday'; +commit ; +prepare stmt1 from 'update t1 set b=''a=two'' where a=2' ; +execute stmt1; +select a,b from t1 where a=2; +a b +2 a=two +execute stmt1; +select a,b from t1 where a=2; +a b +2 a=two +set @arg00=NULL; +prepare stmt1 from 'update t1 set b=? where a=2' ; +execute stmt1 using @arg00; +select a,b from t1 where a=2; +a b +2 NULL +set @arg00='two'; +execute stmt1 using @arg00; +select a,b from t1 where a=2; +a b +2 two +set @arg00=2; +prepare stmt1 from 'update t1 set b=NULL where a=?' ; +execute stmt1 using @arg00; +select a,b from t1 where a=@arg00; +a b +2 NULL +update t1 set b='two' where a=@arg00; +set @arg00=2000; +execute stmt1 using @arg00; +select a,b from t1 where a=@arg00; +a b +set @arg00=2; +set @arg01=22; +prepare stmt1 from 'update t1 set a=? where a=?' ; +execute stmt1 using @arg00, @arg00; +select a,b from t1 where a=@arg00; +a b +2 two +execute stmt1 using @arg01, @arg00; +select a,b from t1 where a=@arg01; +a b +22 two +execute stmt1 using @arg00, @arg01; +select a,b from t1 where a=@arg00; +a b +2 two +set @arg00=NULL; +set @arg01=2; +execute stmt1 using @arg00, @arg01; +Warnings: +Warning 1048 Column 'a' cannot be null +select a,b from t1 order by a; +a b +0 two +1 one +3 three +4 four +set @arg00=0; +execute stmt1 using @arg01, @arg00; +select a,b from t1 order by a; +a b +1 one +2 two +3 three +4 four +set @arg00=23; +set @arg01='two'; +set @arg02=2; +set @arg03='two'; +set @arg04=2; +drop table if exists t2; +create table t2 as select a,b from t1 ; +prepare stmt1 from 'update t1 set a=? where b=? + and a in (select ? from t2 + where b = ? or a = ?)'; +execute stmt1 using @arg00, @arg01, @arg02, @arg03, @arg04 ; +affected rows: 1 +info: Rows matched: 1 Changed: 1 Warnings: 0 +select a,b from t1 where a = @arg00 ; +a b +23 two +prepare stmt1 from 'update t1 set a=? where b=? + and a not in (select ? from t2 + where b = ? or a = ?)'; +execute stmt1 using @arg04, @arg01, @arg02, @arg03, @arg00 ; +affected rows: 1 +info: Rows matched: 1 Changed: 1 Warnings: 0 +select a,b from t1 order by a ; +a b +1 one +2 two +3 three +4 four +drop table t2 ; +create table t2 +( +a int, b varchar(30), +primary key(a) +) engine = 'MARIA' ; +insert into t2(a,b) select a, b from t1 ; +prepare stmt1 from 'update t1 set a=? where b=? + and a in (select ? from t2 + where b = ? or a = ?)'; +execute stmt1 using @arg00, @arg01, @arg02, @arg03, @arg04 ; +affected rows: 1 +info: Rows matched: 1 Changed: 1 Warnings: 0 +select a,b from t1 where a = @arg00 ; +a b +23 two +prepare stmt1 from 'update t1 set a=? where b=? + and a not in (select ? from t2 + where b = ? or a = ?)'; +execute stmt1 using @arg04, @arg01, @arg02, @arg03, @arg00 ; +affected rows: 1 +info: Rows matched: 1 Changed: 1 Warnings: 0 +select a,b from t1 order by a ; +a b +1 one +2 two +3 three +4 four +drop table t2 ; +set @arg00=1; +prepare stmt1 from 'update t1 set b=''bla'' +where a=2 +limit 1'; +execute stmt1 ; +select a,b from t1 where b = 'bla' ; +a b +2 bla +prepare stmt1 from 'update t1 set b=''bla'' where a=2 limit ?'; +execute stmt1 using @arg00; +test_sequence +------ insert tests ------ +delete from t1 ; +insert into t1 values (1,'one'); +insert into t1 values (2,'two'); +insert into t1 values (3,'three'); +insert into t1 values (4,'four'); +commit ; +delete from t9 ; +insert into t9 +set c1= 1, c2= 1, c3= 1, c4= 1, c5= 1, c6= 1, c7= 1, c8= 1, c9= 1, +c10= 1, c11= 1, c12 = 1, +c13= '2004-02-29', c14= '2004-02-29 11:11:11', c15= '2004-02-29 11:11:11', +c16= '11:11:11', c17= '2004', +c18= 1, c19=true, c20= 'a', c21= '123456789a', +c22= '123456789a123456789b123456789c', c23= 'tinyblob', c24= 'tinytext', +c25= 'blob', c26= 'text', c27= 'mediumblob', c28= 'mediumtext', +c29= 'longblob', c30= 'longtext', c31='one', c32= 'monday'; +insert into t9 +set c1= 9, c2= 9, c3= 9, c4= 9, c5= 9, c6= 9, c7= 9, c8= 9, c9= 9, +c10= 9, c11= 9, c12 = 9, +c13= '2004-02-29', c14= '2004-02-29 11:11:11', c15= '2004-02-29 11:11:11', +c16= '11:11:11', c17= '2004', +c18= 1, c19=false, c20= 'a', c21= '123456789a', +c22= '123456789a123456789b123456789c', c23= 'tinyblob', c24= 'tinytext', +c25= 'blob', c26= 'text', c27= 'mediumblob', c28= 'mediumtext', +c29= 'longblob', c30= 'longtext', c31='two', c32= 'tuesday'; +commit ; +prepare stmt1 from 'insert into t1 values(5, ''five'' )'; +execute stmt1; +select a,b from t1 where a = 5; +a b +5 five +set @arg00='six' ; +prepare stmt1 from 'insert into t1 values(6, ? )'; +execute stmt1 using @arg00; +select a,b from t1 where b = @arg00; +a b +6 six +execute stmt1 using @arg00; +ERROR 23000: Duplicate entry '6' for key 'PRIMARY' +set @arg00=NULL ; +prepare stmt1 from 'insert into t1 values(0, ? )'; +execute stmt1 using @arg00; +select a,b from t1 where b is NULL; +a b +0 NULL +set @arg00=8 ; +set @arg01='eight' ; +prepare stmt1 from 'insert into t1 values(?, ? )'; +execute stmt1 using @arg00, @arg01 ; +select a,b from t1 where b = @arg01; +a b +8 eight +set @NULL= null ; +set @arg00= 'abc' ; +execute stmt1 using @NULL, @NULL ; +ERROR 23000: Column 'a' cannot be null +execute stmt1 using @NULL, @NULL ; +ERROR 23000: Column 'a' cannot be null +execute stmt1 using @NULL, @arg00 ; +ERROR 23000: Column 'a' cannot be null +execute stmt1 using @NULL, @arg00 ; +ERROR 23000: Column 'a' cannot be null +set @arg01= 10000 + 2 ; +execute stmt1 using @arg01, @arg00 ; +set @arg01= 10000 + 1 ; +execute stmt1 using @arg01, @arg00 ; +select * from t1 where a > 10000 order by a ; +a b +10001 abc +10002 abc +delete from t1 where a > 10000 ; +set @arg01= 10000 + 2 ; +execute stmt1 using @arg01, @NULL ; +set @arg01= 10000 + 1 ; +execute stmt1 using @arg01, @NULL ; +select * from t1 where a > 10000 order by a ; +a b +10001 NULL +10002 NULL +delete from t1 where a > 10000 ; +set @arg01= 10000 + 10 ; +execute stmt1 using @arg01, @arg01 ; +set @arg01= 10000 + 9 ; +execute stmt1 using @arg01, @arg01 ; +set @arg01= 10000 + 8 ; +execute stmt1 using @arg01, @arg01 ; +set @arg01= 10000 + 7 ; +execute stmt1 using @arg01, @arg01 ; +set @arg01= 10000 + 6 ; +execute stmt1 using @arg01, @arg01 ; +set @arg01= 10000 + 5 ; +execute stmt1 using @arg01, @arg01 ; +set @arg01= 10000 + 4 ; +execute stmt1 using @arg01, @arg01 ; +set @arg01= 10000 + 3 ; +execute stmt1 using @arg01, @arg01 ; +set @arg01= 10000 + 2 ; +execute stmt1 using @arg01, @arg01 ; +set @arg01= 10000 + 1 ; +execute stmt1 using @arg01, @arg01 ; +select * from t1 where a > 10000 order by a ; +a b +10001 10001 +10002 10002 +10003 10003 +10004 10004 +10005 10005 +10006 10006 +10007 10007 +10008 10008 +10009 10009 +10010 10010 +delete from t1 where a > 10000 ; +set @arg00=81 ; +set @arg01='8-1' ; +set @arg02=82 ; +set @arg03='8-2' ; +prepare stmt1 from 'insert into t1 values(?,?),(?,?)'; +execute stmt1 using @arg00, @arg01, @arg02, @arg03 ; +select a,b from t1 where a in (@arg00,@arg02) ; +a b +81 8-1 +82 8-2 +set @arg00=9 ; +set @arg01='nine' ; +prepare stmt1 from 'insert into t1 set a=?, b=? '; +execute stmt1 using @arg00, @arg01 ; +select a,b from t1 where a = @arg00 ; +a b +9 nine +set @arg00=6 ; +set @arg01=1 ; +prepare stmt1 from 'insert into t1 set a=?, b=''sechs'' + on duplicate key update a=a + ?, b=concat(b,''modified'') '; +execute stmt1 using @arg00, @arg01; +select * from t1 order by a; +a b +0 NULL +1 one +2 two +3 three +4 four +5 five +7 sixmodified +8 eight +9 nine +81 8-1 +82 8-2 +set @arg00=81 ; +set @arg01=1 ; +execute stmt1 using @arg00, @arg01; +ERROR 23000: Duplicate entry '82' for key 'PRIMARY' +drop table if exists t2 ; +create table t2 (id int auto_increment primary key) +ENGINE= 'MARIA' ; +prepare stmt1 from ' select last_insert_id() ' ; +insert into t2 values (NULL) ; +execute stmt1 ; +last_insert_id() +1 +insert into t2 values (NULL) ; +execute stmt1 ; +last_insert_id() +2 +drop table t2 ; +set @1000=1000 ; +set @x1000_2="x1000_2" ; +set @x1000_3="x1000_3" ; +set @x1000="x1000" ; +set @1100=1100 ; +set @x1100="x1100" ; +set @100=100 ; +set @updated="updated" ; +insert into t1 values(1000,'x1000_1') ; +insert into t1 values(@1000,@x1000_2),(@1000,@x1000_3) +on duplicate key update a = a + @100, b = concat(b,@updated) ; +select a,b from t1 where a >= 1000 order by a ; +a b +1000 x1000_3 +1100 x1000_1updated +delete from t1 where a >= 1000 ; +insert into t1 values(1000,'x1000_1') ; +prepare stmt1 from ' insert into t1 values(?,?),(?,?) + on duplicate key update a = a + ?, b = concat(b,?) '; +execute stmt1 using @1000, @x1000_2, @1000, @x1000_3, @100, @updated ; +select a,b from t1 where a >= 1000 order by a ; +a b +1000 x1000_3 +1100 x1000_1updated +delete from t1 where a >= 1000 ; +insert into t1 values(1000,'x1000_1') ; +execute stmt1 using @1000, @x1000_2, @1100, @x1000_3, @100, @updated ; +select a,b from t1 where a >= 1000 order by a ; +a b +1200 x1000_1updatedupdated +delete from t1 where a >= 1000 ; +prepare stmt1 from ' replace into t1 (a,b) select 100, ''hundred'' '; +execute stmt1; +execute stmt1; +execute stmt1; +test_sequence +------ multi table tests ------ +delete from t1 ; +delete from t9 ; +insert into t1(a,b) values (1, 'one'), (2, 'two'), (3, 'three') ; +insert into t9 (c1,c21) +values (1, 'one'), (2, 'two'), (3, 'three') ; +prepare stmt_delete from " delete t1, t9 + from t1, t9 where t1.a=t9.c1 and t1.b='updated' "; +prepare stmt_update from " update t1, t9 + set t1.b='updated', t9.c21='updated' + where t1.a=t9.c1 and t1.a=? "; +prepare stmt_select1 from " select a, b from t1 order by a" ; +prepare stmt_select2 from " select c1, c21 from t9 order by c1" ; +set @arg00= 1 ; +execute stmt_update using @arg00 ; +execute stmt_delete ; +execute stmt_select1 ; +a b +2 two +3 three +execute stmt_select2 ; +c1 c21 +2 two +3 three +set @arg00= @arg00 + 1 ; +execute stmt_update using @arg00 ; +execute stmt_delete ; +execute stmt_select1 ; +a b +3 three +execute stmt_select2 ; +c1 c21 +3 three +set @arg00= @arg00 + 1 ; +execute stmt_update using @arg00 ; +execute stmt_delete ; +execute stmt_select1 ; +a b +execute stmt_select2 ; +c1 c21 +set @arg00= @arg00 + 1 ; +delete from t1 ; +insert into t1 values (1,'one'); +insert into t1 values (2,'two'); +insert into t1 values (3,'three'); +insert into t1 values (4,'four'); +commit ; +delete from t9 ; +insert into t9 +set c1= 1, c2= 1, c3= 1, c4= 1, c5= 1, c6= 1, c7= 1, c8= 1, c9= 1, +c10= 1, c11= 1, c12 = 1, +c13= '2004-02-29', c14= '2004-02-29 11:11:11', c15= '2004-02-29 11:11:11', +c16= '11:11:11', c17= '2004', +c18= 1, c19=true, c20= 'a', c21= '123456789a', +c22= '123456789a123456789b123456789c', c23= 'tinyblob', c24= 'tinytext', +c25= 'blob', c26= 'text', c27= 'mediumblob', c28= 'mediumtext', +c29= 'longblob', c30= 'longtext', c31='one', c32= 'monday'; +insert into t9 +set c1= 9, c2= 9, c3= 9, c4= 9, c5= 9, c6= 9, c7= 9, c8= 9, c9= 9, +c10= 9, c11= 9, c12 = 9, +c13= '2004-02-29', c14= '2004-02-29 11:11:11', c15= '2004-02-29 11:11:11', +c16= '11:11:11', c17= '2004', +c18= 1, c19=false, c20= 'a', c21= '123456789a', +c22= '123456789a123456789b123456789c', c23= 'tinyblob', c24= 'tinytext', +c25= 'blob', c26= 'text', c27= 'mediumblob', c28= 'mediumtext', +c29= 'longblob', c30= 'longtext', c31='two', c32= 'tuesday'; +commit ; +insert into t1 values(0,NULL) ; +set @duplicate='duplicate ' ; +set @1000=1000 ; +set @5=5 ; +select a,b from t1 where a < 5 order by a ; +a b +0 NULL +1 one +2 two +3 three +4 four +insert into t1 select a + @1000, concat(@duplicate,b) from t1 +where a < @5 ; +affected rows: 5 +info: Records: 5 Duplicates: 0 Warnings: 0 +select a,b from t1 where a >= 1000 order by a ; +a b +1000 NULL +1001 duplicate one +1002 duplicate two +1003 duplicate three +1004 duplicate four +delete from t1 where a >= 1000 ; +prepare stmt1 from ' insert into t1 select a + ?, concat(?,b) from t1 +where a < ? ' ; +execute stmt1 using @1000, @duplicate, @5; +affected rows: 5 +info: Records: 5 Duplicates: 0 Warnings: 0 +select a,b from t1 where a >= 1000 order by a ; +a b +1000 NULL +1001 duplicate one +1002 duplicate two +1003 duplicate three +1004 duplicate four +delete from t1 where a >= 1000 ; +set @1=1 ; +set @2=2 ; +set @100=100 ; +set @float=1.00; +set @five='five' ; +drop table if exists t2; +create table t2 like t1 ; +insert into t2 (b,a) +select @duplicate, sum(first.a) from t1 first, t1 second +where first.a <> @5 and second.b = first.b +and second.b <> @five +group by second.b +having sum(second.a) > @2 +union +select b, a + @100 from t1 +where (a,b) in ( select sqrt(a+@1)+CAST(@float AS signed),b +from t1); +affected rows: 3 +info: Records: 3 Duplicates: 0 Warnings: 0 +select a,b from t2 order by a ; +a b +3 duplicate +4 duplicate +103 three +delete from t2 ; +prepare stmt1 from ' insert into t2 (b,a) +select ?, sum(first.a) + from t1 first, t1 second + where first.a <> ? and second.b = first.b and second.b <> ? + group by second.b + having sum(second.a) > ? +union +select b, a + ? from t1 + where (a,b) in ( select sqrt(a+?)+CAST(? AS signed),b + from t1 ) ' ; +execute stmt1 using @duplicate, @5, @five, @2, @100, @1, @float ; +affected rows: 3 +info: Records: 3 Duplicates: 0 Warnings: 0 +select a,b from t2 order by a ; +a b +3 duplicate +4 duplicate +103 three +drop table t2; +drop table if exists t5 ; +set @arg01= 8; +set @arg02= 8.0; +set @arg03= 80.00000000000e-1; +set @arg04= 'abc' ; +set @arg05= CAST('abc' as binary) ; +set @arg06= '1991-08-05' ; +set @arg07= CAST('1991-08-05' as date); +set @arg08= '1991-08-05 01:01:01' ; +set @arg09= CAST('1991-08-05 01:01:01' as datetime) ; +set @arg10= unix_timestamp('1991-01-01 01:01:01'); +set @arg11= YEAR('1991-01-01 01:01:01'); +set @arg12= 8 ; +set @arg12= NULL ; +set @arg13= 8.0 ; +set @arg13= NULL ; +set @arg14= 'abc'; +set @arg14= NULL ; +set @arg15= CAST('abc' as binary) ; +set @arg15= NULL ; +create table t5 as select +8 as const01, @arg01 as param01, +8.0 as const02, @arg02 as param02, +80.00000000000e-1 as const03, @arg03 as param03, +'abc' as const04, @arg04 as param04, +CAST('abc' as binary) as const05, @arg05 as param05, +'1991-08-05' as const06, @arg06 as param06, +CAST('1991-08-05' as date) as const07, @arg07 as param07, +'1991-08-05 01:01:01' as const08, @arg08 as param08, +CAST('1991-08-05 01:01:01' as datetime) as const09, @arg09 as param09, +unix_timestamp('1991-01-01 01:01:01') as const10, @arg10 as param10, +YEAR('1991-01-01 01:01:01') as const11, @arg11 as param11, +NULL as const12, @arg12 as param12, +@arg13 as param13, +@arg14 as param14, +@arg15 as param15; +show create table t5 ; +Table Create Table +t5 CREATE TABLE `t5` ( + `const01` bigint(1) NOT NULL DEFAULT '0', + `param01` bigint(20) DEFAULT NULL, + `const02` decimal(2,1) unsigned NOT NULL DEFAULT '0.0', + `param02` decimal(65,30) DEFAULT NULL, + `const03` double NOT NULL DEFAULT '0', + `param03` double DEFAULT NULL, + `const04` varchar(3) NOT NULL DEFAULT '', + `param04` longtext, + `const05` varbinary(3) NOT NULL DEFAULT '', + `param05` longblob, + `const06` varchar(10) NOT NULL DEFAULT '', + `param06` longtext, + `const07` date DEFAULT NULL, + `param07` longblob, + `const08` varchar(19) NOT NULL DEFAULT '', + `param08` longtext, + `const09` datetime DEFAULT NULL, + `param09` longblob, + `const10` int(10) NOT NULL DEFAULT '0', + `param10` bigint(20) DEFAULT NULL, + `const11` int(4) DEFAULT NULL, + `param11` bigint(20) DEFAULT NULL, + `const12` binary(0) DEFAULT NULL, + `param12` bigint(20) DEFAULT NULL, + `param13` decimal(65,30) DEFAULT NULL, + `param14` longtext, + `param15` longblob +) ENGINE=MyISAM DEFAULT CHARSET=latin1 +select * from t5 ; +Catalog Database Table Table_alias Column Column_alias Type Length Max length Is_null Flags Decimals Charsetnr +def test t5 t5 const01 const01 8 1 1 N 32769 0 63 +def test t5 t5 param01 param01 8 20 1 Y 32768 0 63 +def test t5 t5 const02 const02 246 3 3 N 33 1 63 +def test t5 t5 param02 param02 246 67 32 Y 0 30 63 +def test t5 t5 const03 const03 5 17 1 N 32769 31 63 +def test t5 t5 param03 param03 5 23 1 Y 32768 31 63 +def test t5 t5 const04 const04 253 3 3 N 1 0 8 +def test t5 t5 param04 param04 252 4294967295 3 Y 16 0 8 +def test t5 t5 const05 const05 253 3 3 N 129 0 63 +def test t5 t5 param05 param05 252 4294967295 3 Y 144 0 63 +def test t5 t5 const06 const06 253 10 10 N 1 0 8 +def test t5 t5 param06 param06 252 4294967295 10 Y 16 0 8 +def test t5 t5 const07 const07 10 10 10 Y 128 0 63 +def test t5 t5 param07 param07 252 4294967295 10 Y 144 0 63 +def test t5 t5 const08 const08 253 19 19 N 1 0 8 +def test t5 t5 param08 param08 252 4294967295 19 Y 16 0 8 +def test t5 t5 const09 const09 12 19 19 Y 128 0 63 +def test t5 t5 param09 param09 252 4294967295 19 Y 144 0 63 +def test t5 t5 const10 const10 3 10 9 N 32769 0 63 +def test t5 t5 param10 param10 8 20 9 Y 32768 0 63 +def test t5 t5 const11 const11 3 4 4 Y 32768 0 63 +def test t5 t5 param11 param11 8 20 4 Y 32768 0 63 +def test t5 t5 const12 const12 254 0 0 Y 128 0 63 +def test t5 t5 param12 param12 8 20 0 Y 32768 0 63 +def test t5 t5 param13 param13 246 67 0 Y 0 30 63 +def test t5 t5 param14 param14 252 4294967295 0 Y 16 0 8 +def test t5 t5 param15 param15 252 4294967295 0 Y 144 0 63 +const01 8 +param01 8 +const02 8.0 +param02 8.000000000000000000000000000000 +const03 8 +param03 8 +const04 abc +param04 abc +const05 abc +param05 abc +const06 1991-08-05 +param06 1991-08-05 +const07 1991-08-05 +param07 1991-08-05 +const08 1991-08-05 01:01:01 +param08 1991-08-05 01:01:01 +const09 1991-08-05 01:01:01 +param09 1991-08-05 01:01:01 +const10 662680861 +param10 662680861 +const11 1991 +param11 1991 +const12 NULL +param12 NULL +param13 NULL +param14 NULL +param15 NULL +drop table t5 ; +test_sequence +------ data type conversion tests ------ +delete from t1 ; +insert into t1 values (1,'one'); +insert into t1 values (2,'two'); +insert into t1 values (3,'three'); +insert into t1 values (4,'four'); +commit ; +delete from t9 ; +insert into t9 +set c1= 1, c2= 1, c3= 1, c4= 1, c5= 1, c6= 1, c7= 1, c8= 1, c9= 1, +c10= 1, c11= 1, c12 = 1, +c13= '2004-02-29', c14= '2004-02-29 11:11:11', c15= '2004-02-29 11:11:11', +c16= '11:11:11', c17= '2004', +c18= 1, c19=true, c20= 'a', c21= '123456789a', +c22= '123456789a123456789b123456789c', c23= 'tinyblob', c24= 'tinytext', +c25= 'blob', c26= 'text', c27= 'mediumblob', c28= 'mediumtext', +c29= 'longblob', c30= 'longtext', c31='one', c32= 'monday'; +insert into t9 +set c1= 9, c2= 9, c3= 9, c4= 9, c5= 9, c6= 9, c7= 9, c8= 9, c9= 9, +c10= 9, c11= 9, c12 = 9, +c13= '2004-02-29', c14= '2004-02-29 11:11:11', c15= '2004-02-29 11:11:11', +c16= '11:11:11', c17= '2004', +c18= 1, c19=false, c20= 'a', c21= '123456789a', +c22= '123456789a123456789b123456789c', c23= 'tinyblob', c24= 'tinytext', +c25= 'blob', c26= 'text', c27= 'mediumblob', c28= 'mediumtext', +c29= 'longblob', c30= 'longtext', c31='two', c32= 'tuesday'; +commit ; +insert into t9 set c1= 0, c15= '1991-01-01 01:01:01' ; +select * from t9 order by c1 ; +c1 c2 c3 c4 c5 c6 c7 c8 c9 c10 c11 c12 c13 c14 c15 c16 c17 c18 c19 c20 c21 c22 c23 c24 c25 c26 c27 c28 c29 c30 c31 c32 +0 NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL 1991-01-01 01:01:01 NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL +1 1 1 1 1 1 1 1 1 1 1.0000 1.0000 2004-02-29 2004-02-29 11:11:11 2004-02-29 11:11:11 11:11:11 2004 1 1 a 123456789a 123456789a123456789b123456789c tinyblob tinytext blob text mediumblob mediumtext longblob longtext one monday +9 9 9 9 9 9 9 9 9 9 9.0000 9.0000 2004-02-29 2004-02-29 11:11:11 2004-02-29 11:11:11 11:11:11 2004 1 0 a 123456789a 123456789a123456789b123456789c tinyblob tinytext blob text mediumblob mediumtext longblob longtext two tuesday +test_sequence +------ select @parameter:= column ------ +prepare full_info from "select @arg01, @arg02, @arg03, @arg04, + @arg05, @arg06, @arg07, @arg08, + @arg09, @arg10, @arg11, @arg12, + @arg13, @arg14, @arg15, @arg16, + @arg17, @arg18, @arg19, @arg20, + @arg21, @arg22, @arg23, @arg24, + @arg25, @arg26, @arg27, @arg28, + @arg29, @arg30, @arg31, @arg32" ; +select @arg01:= c1, @arg02:= c2, @arg03:= c3, @arg04:= c4, +@arg05:= c5, @arg06:= c6, @arg07:= c7, @arg08:= c8, +@arg09:= c9, @arg10:= c10, @arg11:= c11, @arg12:= c12, +@arg13:= c13, @arg14:= c14, @arg15:= c15, @arg16:= c16, +@arg17:= c17, @arg18:= c18, @arg19:= c19, @arg20:= c20, +@arg21:= c21, @arg22:= c22, @arg23:= c23, @arg24:= c24, +@arg25:= c25, @arg26:= c26, @arg27:= c27, @arg28:= c28, +@arg29:= c29, @arg30:= c30, @arg31:= c31, @arg32:= c32 +from t9 where c1= 1 ; +@arg01:= c1 @arg02:= c2 @arg03:= c3 @arg04:= c4 @arg05:= c5 @arg06:= c6 @arg07:= c7 @arg08:= c8 @arg09:= c9 @arg10:= c10 @arg11:= c11 @arg12:= c12 @arg13:= c13 @arg14:= c14 @arg15:= c15 @arg16:= c16 @arg17:= c17 @arg18:= c18 @arg19:= c19 @arg20:= c20 @arg21:= c21 @arg22:= c22 @arg23:= c23 @arg24:= c24 @arg25:= c25 @arg26:= c26 @arg27:= c27 @arg28:= c28 @arg29:= c29 @arg30:= c30 @arg31:= c31 @arg32:= c32 +1 1 1 1 1 1 1 1 1 1 1.0000 1.0000 2004-02-29 2004-02-29 11:11:11 2004-02-29 11:11:11 11:11:11 2004 1 1 a 123456789a 123456789a123456789b123456789c tinyblob tinytext blob text mediumblob mediumtext longblob longtext one monday +execute full_info ; +Catalog Database Table Table_alias Column Column_alias Type Length Max length Is_null Flags Decimals Charsetnr +def @arg01 253 20 1 Y 128 0 63 +def @arg02 253 20 1 Y 128 0 63 +def @arg03 253 20 1 Y 128 0 63 +def @arg04 253 20 1 Y 128 0 63 +def @arg05 253 20 1 Y 128 0 63 +def @arg06 253 20 1 Y 128 0 63 +def @arg07 253 23 1 Y 128 31 63 +def @arg08 253 23 1 Y 128 31 63 +def @arg09 253 23 1 Y 128 31 63 +def @arg10 253 23 1 Y 128 31 63 +def @arg11 253 67 6 Y 128 30 63 +def @arg12 253 67 6 Y 128 30 63 +def @arg13 253 8192 10 Y 128 31 63 +def @arg14 253 8192 19 Y 128 31 63 +def @arg15 253 8192 19 Y 128 31 63 +def @arg16 253 8192 8 Y 128 31 63 +def @arg17 253 20 4 Y 128 0 63 +def @arg18 253 20 1 Y 128 0 63 +def @arg19 253 20 1 Y 128 0 63 +def @arg20 253 8192 1 Y 0 31 8 +def @arg21 253 8192 10 Y 0 31 8 +def @arg22 253 8192 30 Y 0 31 8 +def @arg23 253 8192 8 Y 128 31 63 +def @arg24 253 8192 8 Y 0 31 8 +def @arg25 253 8192 4 Y 128 31 63 +def @arg26 253 8192 4 Y 0 31 8 +def @arg27 253 8192 10 Y 128 31 63 +def @arg28 253 8192 10 Y 0 31 8 +def @arg29 253 8192 8 Y 128 31 63 +def @arg30 253 8192 8 Y 0 31 8 +def @arg31 253 8192 3 Y 0 31 8 +def @arg32 253 8192 6 Y 0 31 8 +@arg01 @arg02 @arg03 @arg04 @arg05 @arg06 @arg07 @arg08 @arg09 @arg10 @arg11 @arg12 @arg13 @arg14 @arg15 @arg16 @arg17 @arg18 @arg19 @arg20 @arg21 @arg22 @arg23 @arg24 @arg25 @arg26 @arg27 @arg28 @arg29 @arg30 @arg31 @arg32 +1 1 1 1 1 1 1 1 1 1 1.0000 1.0000 2004-02-29 2004-02-29 11:11:11 2004-02-29 11:11:11 11:11:11 2004 1 1 a 123456789a 123456789a123456789b123456789c tinyblob tinytext blob text mediumblob mediumtext longblob longtext one monday +select @arg01:= c1, @arg02:= c2, @arg03:= c3, @arg04:= c4, +@arg05:= c5, @arg06:= c6, @arg07:= c7, @arg08:= c8, +@arg09:= c9, @arg10:= c10, @arg11:= c11, @arg12:= c12, +@arg13:= c13, @arg14:= c14, @arg15:= c15, @arg16:= c16, +@arg17:= c17, @arg18:= c18, @arg19:= c19, @arg20:= c20, +@arg21:= c21, @arg22:= c22, @arg23:= c23, @arg24:= c24, +@arg25:= c25, @arg26:= c26, @arg27:= c27, @arg28:= c28, +@arg29:= c29, @arg30:= c30, @arg31:= c31, @arg32:= c32 +from t9 where c1= 0 ; +@arg01:= c1 @arg02:= c2 @arg03:= c3 @arg04:= c4 @arg05:= c5 @arg06:= c6 @arg07:= c7 @arg08:= c8 @arg09:= c9 @arg10:= c10 @arg11:= c11 @arg12:= c12 @arg13:= c13 @arg14:= c14 @arg15:= c15 @arg16:= c16 @arg17:= c17 @arg18:= c18 @arg19:= c19 @arg20:= c20 @arg21:= c21 @arg22:= c22 @arg23:= c23 @arg24:= c24 @arg25:= c25 @arg26:= c26 @arg27:= c27 @arg28:= c28 @arg29:= c29 @arg30:= c30 @arg31:= c31 @arg32:= c32 +0 NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL 1991-01-01 01:01:01 NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL +execute full_info ; +Catalog Database Table Table_alias Column Column_alias Type Length Max length Is_null Flags Decimals Charsetnr +def @arg01 253 20 1 Y 128 0 63 +def @arg02 253 20 0 Y 128 0 63 +def @arg03 253 20 0 Y 128 0 63 +def @arg04 253 20 0 Y 128 0 63 +def @arg05 253 20 0 Y 128 0 63 +def @arg06 253 20 0 Y 128 0 63 +def @arg07 253 23 0 Y 128 31 63 +def @arg08 253 23 0 Y 128 31 63 +def @arg09 253 23 0 Y 128 31 63 +def @arg10 253 23 0 Y 128 31 63 +def @arg11 253 67 0 Y 128 30 63 +def @arg12 253 67 0 Y 128 30 63 +def @arg13 253 8192 0 Y 128 31 63 +def @arg14 253 8192 0 Y 128 31 63 +def @arg15 253 8192 19 Y 128 31 63 +def @arg16 253 8192 0 Y 128 31 63 +def @arg17 253 20 0 Y 128 0 63 +def @arg18 253 20 0 Y 128 0 63 +def @arg19 253 20 0 Y 128 0 63 +def @arg20 253 8192 0 Y 0 31 8 +def @arg21 253 8192 0 Y 0 31 8 +def @arg22 253 8192 0 Y 0 31 8 +def @arg23 253 8192 0 Y 128 31 63 +def @arg24 253 8192 0 Y 0 31 8 +def @arg25 253 8192 0 Y 128 31 63 +def @arg26 253 8192 0 Y 0 31 8 +def @arg27 253 8192 0 Y 128 31 63 +def @arg28 253 8192 0 Y 0 31 8 +def @arg29 253 8192 0 Y 128 31 63 +def @arg30 253 8192 0 Y 0 31 8 +def @arg31 253 8192 0 Y 0 31 8 +def @arg32 253 8192 0 Y 0 31 8 +@arg01 @arg02 @arg03 @arg04 @arg05 @arg06 @arg07 @arg08 @arg09 @arg10 @arg11 @arg12 @arg13 @arg14 @arg15 @arg16 @arg17 @arg18 @arg19 @arg20 @arg21 @arg22 @arg23 @arg24 @arg25 @arg26 @arg27 @arg28 @arg29 @arg30 @arg31 @arg32 +0 NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL 1991-01-01 01:01:01 NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL +prepare stmt1 from "select + @arg01:= c1, @arg02:= c2, @arg03:= c3, @arg04:= c4, + @arg05:= c5, @arg06:= c6, @arg07:= c7, @arg08:= c8, + @arg09:= c9, @arg10:= c10, @arg11:= c11, @arg12:= c12, + @arg13:= c13, @arg14:= c14, @arg15:= c15, @arg16:= c16, + @arg17:= c17, @arg18:= c18, @arg19:= c19, @arg20:= c20, + @arg21:= c21, @arg22:= c22, @arg23:= c23, @arg24:= c24, + @arg25:= c25, @arg26:= c26, @arg27:= c27, @arg28:= c28, + @arg29:= c29, @arg30:= c30, @arg31:= c31, @arg32:= c32 +from t9 where c1= ?" ; +set @my_key= 1 ; +execute stmt1 using @my_key ; +@arg01:= c1 @arg02:= c2 @arg03:= c3 @arg04:= c4 @arg05:= c5 @arg06:= c6 @arg07:= c7 @arg08:= c8 @arg09:= c9 @arg10:= c10 @arg11:= c11 @arg12:= c12 @arg13:= c13 @arg14:= c14 @arg15:= c15 @arg16:= c16 @arg17:= c17 @arg18:= c18 @arg19:= c19 @arg20:= c20 @arg21:= c21 @arg22:= c22 @arg23:= c23 @arg24:= c24 @arg25:= c25 @arg26:= c26 @arg27:= c27 @arg28:= c28 @arg29:= c29 @arg30:= c30 @arg31:= c31 @arg32:= c32 +1 1 1 1 1 1 1 1 1 1 1.0000 1.0000 2004-02-29 2004-02-29 11:11:11 2004-02-29 11:11:11 11:11:11 2004 1 1 a 123456789a 123456789a123456789b123456789c tinyblob tinytext blob text mediumblob mediumtext longblob longtext one monday +execute full_info ; +Catalog Database Table Table_alias Column Column_alias Type Length Max length Is_null Flags Decimals Charsetnr +def @arg01 253 20 1 Y 128 0 63 +def @arg02 253 20 1 Y 128 0 63 +def @arg03 253 20 1 Y 128 0 63 +def @arg04 253 20 1 Y 128 0 63 +def @arg05 253 20 1 Y 128 0 63 +def @arg06 253 20 1 Y 128 0 63 +def @arg07 253 23 1 Y 128 31 63 +def @arg08 253 23 1 Y 128 31 63 +def @arg09 253 23 1 Y 128 31 63 +def @arg10 253 23 1 Y 128 31 63 +def @arg11 253 67 6 Y 128 30 63 +def @arg12 253 67 6 Y 128 30 63 +def @arg13 253 8192 10 Y 128 31 63 +def @arg14 253 8192 19 Y 128 31 63 +def @arg15 253 8192 19 Y 128 31 63 +def @arg16 253 8192 8 Y 128 31 63 +def @arg17 253 20 4 Y 128 0 63 +def @arg18 253 20 1 Y 128 0 63 +def @arg19 253 20 1 Y 128 0 63 +def @arg20 253 8192 1 Y 0 31 8 +def @arg21 253 8192 10 Y 0 31 8 +def @arg22 253 8192 30 Y 0 31 8 +def @arg23 253 8192 8 Y 128 31 63 +def @arg24 253 8192 8 Y 0 31 8 +def @arg25 253 8192 4 Y 128 31 63 +def @arg26 253 8192 4 Y 0 31 8 +def @arg27 253 8192 10 Y 128 31 63 +def @arg28 253 8192 10 Y 0 31 8 +def @arg29 253 8192 8 Y 128 31 63 +def @arg30 253 8192 8 Y 0 31 8 +def @arg31 253 8192 3 Y 0 31 8 +def @arg32 253 8192 6 Y 0 31 8 +@arg01 @arg02 @arg03 @arg04 @arg05 @arg06 @arg07 @arg08 @arg09 @arg10 @arg11 @arg12 @arg13 @arg14 @arg15 @arg16 @arg17 @arg18 @arg19 @arg20 @arg21 @arg22 @arg23 @arg24 @arg25 @arg26 @arg27 @arg28 @arg29 @arg30 @arg31 @arg32 +1 1 1 1 1 1 1 1 1 1 1.0000 1.0000 2004-02-29 2004-02-29 11:11:11 2004-02-29 11:11:11 11:11:11 2004 1 1 a 123456789a 123456789a123456789b123456789c tinyblob tinytext blob text mediumblob mediumtext longblob longtext one monday +set @my_key= 0 ; +execute stmt1 using @my_key ; +@arg01:= c1 @arg02:= c2 @arg03:= c3 @arg04:= c4 @arg05:= c5 @arg06:= c6 @arg07:= c7 @arg08:= c8 @arg09:= c9 @arg10:= c10 @arg11:= c11 @arg12:= c12 @arg13:= c13 @arg14:= c14 @arg15:= c15 @arg16:= c16 @arg17:= c17 @arg18:= c18 @arg19:= c19 @arg20:= c20 @arg21:= c21 @arg22:= c22 @arg23:= c23 @arg24:= c24 @arg25:= c25 @arg26:= c26 @arg27:= c27 @arg28:= c28 @arg29:= c29 @arg30:= c30 @arg31:= c31 @arg32:= c32 +0 NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL 1991-01-01 01:01:01 NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL +execute full_info ; +Catalog Database Table Table_alias Column Column_alias Type Length Max length Is_null Flags Decimals Charsetnr +def @arg01 253 20 1 Y 128 0 63 +def @arg02 253 20 0 Y 128 0 63 +def @arg03 253 20 0 Y 128 0 63 +def @arg04 253 20 0 Y 128 0 63 +def @arg05 253 20 0 Y 128 0 63 +def @arg06 253 20 0 Y 128 0 63 +def @arg07 253 23 0 Y 128 31 63 +def @arg08 253 23 0 Y 128 31 63 +def @arg09 253 23 0 Y 128 31 63 +def @arg10 253 23 0 Y 128 31 63 +def @arg11 253 67 0 Y 128 30 63 +def @arg12 253 67 0 Y 128 30 63 +def @arg13 253 8192 0 Y 128 31 63 +def @arg14 253 8192 0 Y 128 31 63 +def @arg15 253 8192 19 Y 128 31 63 +def @arg16 253 8192 0 Y 128 31 63 +def @arg17 253 20 0 Y 128 0 63 +def @arg18 253 20 0 Y 128 0 63 +def @arg19 253 20 0 Y 128 0 63 +def @arg20 253 8192 0 Y 0 31 8 +def @arg21 253 8192 0 Y 0 31 8 +def @arg22 253 8192 0 Y 0 31 8 +def @arg23 253 8192 0 Y 128 31 63 +def @arg24 253 8192 0 Y 0 31 8 +def @arg25 253 8192 0 Y 128 31 63 +def @arg26 253 8192 0 Y 0 31 8 +def @arg27 253 8192 0 Y 128 31 63 +def @arg28 253 8192 0 Y 0 31 8 +def @arg29 253 8192 0 Y 128 31 63 +def @arg30 253 8192 0 Y 0 31 8 +def @arg31 253 8192 0 Y 0 31 8 +def @arg32 253 8192 0 Y 0 31 8 +@arg01 @arg02 @arg03 @arg04 @arg05 @arg06 @arg07 @arg08 @arg09 @arg10 @arg11 @arg12 @arg13 @arg14 @arg15 @arg16 @arg17 @arg18 @arg19 @arg20 @arg21 @arg22 @arg23 @arg24 @arg25 @arg26 @arg27 @arg28 @arg29 @arg30 @arg31 @arg32 +0 NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL 1991-01-01 01:01:01 NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL +prepare stmt1 from "select ? := c1 from t9 where c1= 1" ; +ERROR 42000: You have an error in your SQL syntax; check the manual that corresponds to your MySQL server version for the right syntax to use near ':= c1 from t9 where c1= 1' at line 1 +test_sequence +------ select column, .. into @parm,.. ------ +select c1, c2, c3, c4, c5, c6, c7, c8, c9, c10, c11, c12, +c13, c14, c15, c16, c17, c18, c19, c20, c21, c22, c23, c24, +c25, c26, c27, c28, c29, c30, c31, c32 +into @arg01, @arg02, @arg03, @arg04, @arg05, @arg06, @arg07, @arg08, +@arg09, @arg10, @arg11, @arg12, @arg13, @arg14, @arg15, @arg16, +@arg17, @arg18, @arg19, @arg20, @arg21, @arg22, @arg23, @arg24, +@arg25, @arg26, @arg27, @arg28, @arg29, @arg30, @arg31, @arg32 +from t9 where c1= 1 ; +execute full_info ; +Catalog Database Table Table_alias Column Column_alias Type Length Max length Is_null Flags Decimals Charsetnr +def @arg01 253 20 1 Y 128 0 63 +def @arg02 253 20 1 Y 128 0 63 +def @arg03 253 20 1 Y 128 0 63 +def @arg04 253 20 1 Y 128 0 63 +def @arg05 253 20 1 Y 128 0 63 +def @arg06 253 20 1 Y 128 0 63 +def @arg07 253 23 1 Y 128 31 63 +def @arg08 253 23 1 Y 128 31 63 +def @arg09 253 23 1 Y 128 31 63 +def @arg10 253 23 1 Y 128 31 63 +def @arg11 253 67 6 Y 128 30 63 +def @arg12 253 67 6 Y 128 30 63 +def @arg13 253 8192 10 Y 128 31 63 +def @arg14 253 8192 19 Y 128 31 63 +def @arg15 253 8192 19 Y 128 31 63 +def @arg16 253 8192 8 Y 128 31 63 +def @arg17 253 20 4 Y 128 0 63 +def @arg18 253 20 1 Y 128 0 63 +def @arg19 253 20 1 Y 128 0 63 +def @arg20 253 8192 1 Y 0 31 8 +def @arg21 253 8192 10 Y 0 31 8 +def @arg22 253 8192 30 Y 0 31 8 +def @arg23 253 8192 8 Y 128 31 63 +def @arg24 253 8192 8 Y 0 31 8 +def @arg25 253 8192 4 Y 128 31 63 +def @arg26 253 8192 4 Y 0 31 8 +def @arg27 253 8192 10 Y 128 31 63 +def @arg28 253 8192 10 Y 0 31 8 +def @arg29 253 8192 8 Y 128 31 63 +def @arg30 253 8192 8 Y 0 31 8 +def @arg31 253 8192 3 Y 0 31 8 +def @arg32 253 8192 6 Y 0 31 8 +@arg01 @arg02 @arg03 @arg04 @arg05 @arg06 @arg07 @arg08 @arg09 @arg10 @arg11 @arg12 @arg13 @arg14 @arg15 @arg16 @arg17 @arg18 @arg19 @arg20 @arg21 @arg22 @arg23 @arg24 @arg25 @arg26 @arg27 @arg28 @arg29 @arg30 @arg31 @arg32 +1 1 1 1 1 1 1 1 1 1 1.0000 1.0000 2004-02-29 2004-02-29 11:11:11 2004-02-29 11:11:11 11:11:11 2004 1 1 a 123456789a 123456789a123456789b123456789c tinyblob tinytext blob text mediumblob mediumtext longblob longtext one monday +select c1, c2, c3, c4, c5, c6, c7, c8, c9, c10, c11, c12, +c13, c14, c15, c16, c17, c18, c19, c20, c21, c22, c23, c24, +c25, c26, c27, c28, c29, c30, c31, c32 +into @arg01, @arg02, @arg03, @arg04, @arg05, @arg06, @arg07, @arg08, +@arg09, @arg10, @arg11, @arg12, @arg13, @arg14, @arg15, @arg16, +@arg17, @arg18, @arg19, @arg20, @arg21, @arg22, @arg23, @arg24, +@arg25, @arg26, @arg27, @arg28, @arg29, @arg30, @arg31, @arg32 +from t9 where c1= 0 ; +execute full_info ; +Catalog Database Table Table_alias Column Column_alias Type Length Max length Is_null Flags Decimals Charsetnr +def @arg01 253 20 1 Y 128 0 63 +def @arg02 253 20 0 Y 128 0 63 +def @arg03 253 20 0 Y 128 0 63 +def @arg04 253 20 0 Y 128 0 63 +def @arg05 253 20 0 Y 128 0 63 +def @arg06 253 20 0 Y 128 0 63 +def @arg07 253 23 0 Y 128 31 63 +def @arg08 253 23 0 Y 128 31 63 +def @arg09 253 23 0 Y 128 31 63 +def @arg10 253 23 0 Y 128 31 63 +def @arg11 253 67 0 Y 128 30 63 +def @arg12 253 67 0 Y 128 30 63 +def @arg13 253 8192 0 Y 128 31 63 +def @arg14 253 8192 0 Y 128 31 63 +def @arg15 253 8192 19 Y 128 31 63 +def @arg16 253 8192 0 Y 128 31 63 +def @arg17 253 20 0 Y 128 0 63 +def @arg18 253 20 0 Y 128 0 63 +def @arg19 253 20 0 Y 128 0 63 +def @arg20 253 8192 0 Y 0 31 8 +def @arg21 253 8192 0 Y 0 31 8 +def @arg22 253 8192 0 Y 0 31 8 +def @arg23 253 8192 0 Y 128 31 63 +def @arg24 253 8192 0 Y 0 31 8 +def @arg25 253 8192 0 Y 128 31 63 +def @arg26 253 8192 0 Y 0 31 8 +def @arg27 253 8192 0 Y 128 31 63 +def @arg28 253 8192 0 Y 0 31 8 +def @arg29 253 8192 0 Y 128 31 63 +def @arg30 253 8192 0 Y 0 31 8 +def @arg31 253 8192 0 Y 0 31 8 +def @arg32 253 8192 0 Y 0 31 8 +@arg01 @arg02 @arg03 @arg04 @arg05 @arg06 @arg07 @arg08 @arg09 @arg10 @arg11 @arg12 @arg13 @arg14 @arg15 @arg16 @arg17 @arg18 @arg19 @arg20 @arg21 @arg22 @arg23 @arg24 @arg25 @arg26 @arg27 @arg28 @arg29 @arg30 @arg31 @arg32 +0 NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL 1991-01-01 01:01:01 NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL +prepare stmt1 from "select c1, c2, c3, c4, c5, c6, c7, c8, c9, c10, c11, c12, + c13, c14, c15, c16, c17, c18, c19, c20, c21, c22, c23, c24, + c25, c26, c27, c28, c29, c30, c31, c32 +into @arg01, @arg02, @arg03, @arg04, @arg05, @arg06, @arg07, @arg08, + @arg09, @arg10, @arg11, @arg12, @arg13, @arg14, @arg15, @arg16, + @arg17, @arg18, @arg19, @arg20, @arg21, @arg22, @arg23, @arg24, + @arg25, @arg26, @arg27, @arg28, @arg29, @arg30, @arg31, @arg32 +from t9 where c1= ?" ; +set @my_key= 1 ; +execute stmt1 using @my_key ; +execute full_info ; +Catalog Database Table Table_alias Column Column_alias Type Length Max length Is_null Flags Decimals Charsetnr +def @arg01 253 20 1 Y 128 0 63 +def @arg02 253 20 1 Y 128 0 63 +def @arg03 253 20 1 Y 128 0 63 +def @arg04 253 20 1 Y 128 0 63 +def @arg05 253 20 1 Y 128 0 63 +def @arg06 253 20 1 Y 128 0 63 +def @arg07 253 23 1 Y 128 31 63 +def @arg08 253 23 1 Y 128 31 63 +def @arg09 253 23 1 Y 128 31 63 +def @arg10 253 23 1 Y 128 31 63 +def @arg11 253 67 6 Y 128 30 63 +def @arg12 253 67 6 Y 128 30 63 +def @arg13 253 8192 10 Y 128 31 63 +def @arg14 253 8192 19 Y 128 31 63 +def @arg15 253 8192 19 Y 128 31 63 +def @arg16 253 8192 8 Y 128 31 63 +def @arg17 253 20 4 Y 128 0 63 +def @arg18 253 20 1 Y 128 0 63 +def @arg19 253 20 1 Y 128 0 63 +def @arg20 253 8192 1 Y 0 31 8 +def @arg21 253 8192 10 Y 0 31 8 +def @arg22 253 8192 30 Y 0 31 8 +def @arg23 253 8192 8 Y 128 31 63 +def @arg24 253 8192 8 Y 0 31 8 +def @arg25 253 8192 4 Y 128 31 63 +def @arg26 253 8192 4 Y 0 31 8 +def @arg27 253 8192 10 Y 128 31 63 +def @arg28 253 8192 10 Y 0 31 8 +def @arg29 253 8192 8 Y 128 31 63 +def @arg30 253 8192 8 Y 0 31 8 +def @arg31 253 8192 3 Y 0 31 8 +def @arg32 253 8192 6 Y 0 31 8 +@arg01 @arg02 @arg03 @arg04 @arg05 @arg06 @arg07 @arg08 @arg09 @arg10 @arg11 @arg12 @arg13 @arg14 @arg15 @arg16 @arg17 @arg18 @arg19 @arg20 @arg21 @arg22 @arg23 @arg24 @arg25 @arg26 @arg27 @arg28 @arg29 @arg30 @arg31 @arg32 +1 1 1 1 1 1 1 1 1 1 1.0000 1.0000 2004-02-29 2004-02-29 11:11:11 2004-02-29 11:11:11 11:11:11 2004 1 1 a 123456789a 123456789a123456789b123456789c tinyblob tinytext blob text mediumblob mediumtext longblob longtext one monday +set @my_key= 0 ; +execute stmt1 using @my_key ; +execute full_info ; +Catalog Database Table Table_alias Column Column_alias Type Length Max length Is_null Flags Decimals Charsetnr +def @arg01 253 20 1 Y 128 0 63 +def @arg02 253 20 0 Y 128 0 63 +def @arg03 253 20 0 Y 128 0 63 +def @arg04 253 20 0 Y 128 0 63 +def @arg05 253 20 0 Y 128 0 63 +def @arg06 253 20 0 Y 128 0 63 +def @arg07 253 23 0 Y 128 31 63 +def @arg08 253 23 0 Y 128 31 63 +def @arg09 253 23 0 Y 128 31 63 +def @arg10 253 23 0 Y 128 31 63 +def @arg11 253 67 0 Y 128 30 63 +def @arg12 253 67 0 Y 128 30 63 +def @arg13 253 8192 0 Y 128 31 63 +def @arg14 253 8192 0 Y 128 31 63 +def @arg15 253 8192 19 Y 128 31 63 +def @arg16 253 8192 0 Y 128 31 63 +def @arg17 253 20 0 Y 128 0 63 +def @arg18 253 20 0 Y 128 0 63 +def @arg19 253 20 0 Y 128 0 63 +def @arg20 253 8192 0 Y 0 31 8 +def @arg21 253 8192 0 Y 0 31 8 +def @arg22 253 8192 0 Y 0 31 8 +def @arg23 253 8192 0 Y 128 31 63 +def @arg24 253 8192 0 Y 0 31 8 +def @arg25 253 8192 0 Y 128 31 63 +def @arg26 253 8192 0 Y 0 31 8 +def @arg27 253 8192 0 Y 128 31 63 +def @arg28 253 8192 0 Y 0 31 8 +def @arg29 253 8192 0 Y 128 31 63 +def @arg30 253 8192 0 Y 0 31 8 +def @arg31 253 8192 0 Y 0 31 8 +def @arg32 253 8192 0 Y 0 31 8 +@arg01 @arg02 @arg03 @arg04 @arg05 @arg06 @arg07 @arg08 @arg09 @arg10 @arg11 @arg12 @arg13 @arg14 @arg15 @arg16 @arg17 @arg18 @arg19 @arg20 @arg21 @arg22 @arg23 @arg24 @arg25 @arg26 @arg27 @arg28 @arg29 @arg30 @arg31 @arg32 +0 NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL 1991-01-01 01:01:01 NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL +prepare stmt1 from "select c1 into ? from t9 where c1= 1" ; +ERROR 42000: You have an error in your SQL syntax; check the manual that corresponds to your MySQL server version for the right syntax to use near '? from t9 where c1= 1' at line 1 +test_sequence +-- insert into numeric columns -- +insert into t9 +( c1, c2, c3, c4, c5, c6, c7, c8, c9, c10, c12 ) +values +( 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20 ) ; +set @arg00= 21 ; +insert into t9 +( c1, c2, c3, c4, c5, c6, c7, c8, c9, c10, c12 ) +values +( @arg00, @arg00, @arg00, @arg00, @arg00, @arg00, +@arg00, @arg00, @arg00, @arg00, @arg00 ) ; +prepare stmt1 from "insert into t9 + ( c1, c2, c3, c4, c5, c6, c7, c8, c9, c10, c12 ) +values + ( 22, 22, 22, 22, 22, 22, 22, 22, 22, 22, 22 )" ; +execute stmt1 ; +set @arg00= 23; +prepare stmt2 from "insert into t9 + ( c1, c2, c3, c4, c5, c6, c7, c8, c9, c10, c12 ) +values + ( ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ? )" ; +execute stmt2 using @arg00, @arg00, @arg00, @arg00, @arg00, @arg00, @arg00, +@arg00, @arg00, @arg00, @arg00 ; +insert into t9 +( c1, c2, c3, c4, c5, c6, c7, c8, c9, c10, c12 ) +values +( 30.0, 30.0, 30.0, 30.0, 30.0, 30.0, 30.0, 30.0, +30.0, 30.0, 30.0 ) ; +set @arg00= 31.0 ; +insert into t9 +( c1, c2, c3, c4, c5, c6, c7, c8, c9, c10, c12 ) +values +( @arg00, @arg00, @arg00, @arg00, @arg00, @arg00, +@arg00, @arg00, @arg00, @arg00, @arg00 ) ; +prepare stmt1 from "insert into t9 + ( c1, c2, c3, c4, c5, c6, c7, c8, c9, c10, c12 ) +values + ( 32.0, 32.0, 32.0, 32.0, 32.0, 32.0, 32.0, 32.0, + 32.0, 32.0, 32.0 )" ; +execute stmt1 ; +set @arg00= 33.0; +prepare stmt2 from "insert into t9 + ( c1, c2, c3, c4, c5, c6, c7, c8, c9, c10, c12 ) +values + ( ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ? )" ; +execute stmt2 using @arg00, @arg00, @arg00, @arg00, @arg00, @arg00, @arg00, +@arg00, @arg00, @arg00, @arg00 ; +insert into t9 +( c1, c2, c3, c4, c5, c6, c7, c8, c9, c10, c12 ) +values +( '40', '40', '40', '40', '40', '40', '40', '40', +'40', '40', '40' ) ; +set @arg00= '41' ; +insert into t9 +( c1, c2, c3, c4, c5, c6, c7, c8, c9, c10, c12 ) +values +( @arg00, @arg00, @arg00, @arg00, @arg00, @arg00, +@arg00, @arg00, @arg00, @arg00, @arg00 ) ; +prepare stmt1 from "insert into t9 + ( c1, c2, c3, c4, c5, c6, c7, c8, c9, c10, c12 ) +values + ( '42', '42', '42', '42', '42', '42', '42', '42', + '42', '42', '42' )" ; +execute stmt1 ; +set @arg00= '43'; +prepare stmt2 from "insert into t9 + ( c1, c2, c3, c4, c5, c6, c7, c8, c9, c10, c12 ) +values + ( ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ? )" ; +execute stmt2 using @arg00, @arg00, @arg00, @arg00, @arg00, @arg00, @arg00, +@arg00, @arg00, @arg00, @arg00 ; +insert into t9 +( c1, c2, c3, c4, c5, c6, c7, c8, c9, c10, c12 ) +values +( CAST('50' as binary), CAST('50' as binary), +CAST('50' as binary), CAST('50' as binary), CAST('50' as binary), +CAST('50' as binary), CAST('50' as binary), CAST('50' as binary), +CAST('50' as binary), CAST('50' as binary), CAST('50' as binary) ) ; +set @arg00= CAST('51' as binary) ; +insert into t9 +( c1, c2, c3, c4, c5, c6, c7, c8, c9, c10, c12 ) +values +( @arg00, @arg00, @arg00, @arg00, @arg00, @arg00, +@arg00, @arg00, @arg00, @arg00, @arg00 ) ; +prepare stmt1 from "insert into t9 + ( c1, c2, c3, c4, c5, c6, c7, c8, c9, c10, c12 ) +values + ( CAST('52' as binary), CAST('52' as binary), + CAST('52' as binary), CAST('52' as binary), CAST('52' as binary), + CAST('52' as binary), CAST('52' as binary), CAST('52' as binary), + CAST('52' as binary), CAST('52' as binary), CAST('52' as binary) )" ; +execute stmt1 ; +set @arg00= CAST('53' as binary) ; +prepare stmt2 from "insert into t9 + ( c1, c2, c3, c4, c5, c6, c7, c8, c9, c10, c12 ) +values + ( ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ? )" ; +execute stmt2 using @arg00, @arg00, @arg00, @arg00, @arg00, @arg00, @arg00, +@arg00, @arg00, @arg00, @arg00 ; +set @arg00= 2 ; +set @arg00= NULL ; +insert into t9 +( c1, c2, c3, c4, c5, c6, c7, c8, c9, c10, c12 ) +values +( 60, NULL, NULL, NULL, NULL, NULL, NULL, NULL, +NULL, NULL, NULL ) ; +insert into t9 +( c1, c2, c3, c4, c5, c6, c7, c8, c9, c10, c12 ) +values +( 61, @arg00, @arg00, @arg00, @arg00, @arg00, +@arg00, @arg00, @arg00, @arg00, @arg00 ) ; +prepare stmt1 from "insert into t9 + ( c1, c2, c3, c4, c5, c6, c7, c8, c9, c10, c12 ) +values + ( 62, NULL, NULL, NULL, NULL, NULL, NULL, NULL, + NULL, NULL, NULL )" ; +execute stmt1 ; +prepare stmt2 from "insert into t9 + ( c1, c2, c3, c4, c5, c6, c7, c8, c9, c10, c12 ) +values + ( 63, ?, ?, ?, ?, ?, ?, ?, ?, ?, ? )" ; +execute stmt2 using @arg00, @arg00, @arg00, @arg00, @arg00, @arg00, +@arg00, @arg00, @arg00, @arg00 ; +set @arg00= 8.0 ; +set @arg00= NULL ; +insert into t9 +( c1, c2, c3, c4, c5, c6, c7, c8, c9, c10, c12 ) +values +( 71, @arg00, @arg00, @arg00, @arg00, @arg00, +@arg00, @arg00, @arg00, @arg00, @arg00 ) ; +prepare stmt2 from "insert into t9 + ( c1, c2, c3, c4, c5, c6, c7, c8, c9, c10, c12 ) +values + ( 73, ?, ?, ?, ?, ?, ?, ?, ?, ?, ? )" ; +execute stmt2 using @arg00, @arg00, @arg00, @arg00, @arg00, @arg00, +@arg00, @arg00, @arg00, @arg00 ; +set @arg00= 'abc' ; +set @arg00= NULL ; +insert into t9 +( c1, c2, c3, c4, c5, c6, c7, c8, c9, c10, c12 ) +values +( 81, @arg00, @arg00, @arg00, @arg00, @arg00, +@arg00, @arg00, @arg00, @arg00, @arg00 ) ; +prepare stmt2 from "insert into t9 + ( c1, c2, c3, c4, c5, c6, c7, c8, c9, c10, c12 ) +values + ( 83, ?, ?, ?, ?, ?, ?, ?, ?, ?, ? )" ; +execute stmt2 using @arg00, @arg00, @arg00, @arg00, @arg00, @arg00, +@arg00, @arg00, @arg00, @arg00 ; +select c1, c2, c3, c4, c5, c6, c7, c8, c9, c10, c12 +from t9 where c1 >= 20 +order by c1 ; +c1 c2 c3 c4 c5 c6 c7 c8 c9 c10 c12 +20 20 20 20 20 20 20 20 20 20 20.0000 +21 21 21 21 21 21 21 21 21 21 21.0000 +22 22 22 22 22 22 22 22 22 22 22.0000 +23 23 23 23 23 23 23 23 23 23 23.0000 +30 30 30 30 30 30 30 30 30 30 30.0000 +31 31 31 31 31 31 31 31 31 31 31.0000 +32 32 32 32 32 32 32 32 32 32 32.0000 +33 33 33 33 33 33 33 33 33 33 33.0000 +40 40 40 40 40 40 40 40 40 40 40.0000 +41 41 41 41 41 41 41 41 41 41 41.0000 +42 42 42 42 42 42 42 42 42 42 42.0000 +43 43 43 43 43 43 43 43 43 43 43.0000 +50 50 50 50 50 50 50 50 50 50 50.0000 +51 51 51 51 51 51 51 51 51 51 51.0000 +52 52 52 52 52 52 52 52 52 52 52.0000 +53 53 53 53 53 53 53 53 53 53 53.0000 +60 NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL +61 NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL +62 NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL +63 NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL +71 NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL +73 NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL +81 NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL +83 NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL +test_sequence +-- select .. where numeric column = .. -- +set @arg00= 20; +select 'true' as found from t9 +where c1= 20 and c2= 20 and c3= 20 and c4= 20 and c5= 20 and c6= 20 and c7= 20 +and c8= 20 and c9= 20 and c10= 20 and c12= 20; +found +true +select 'true' as found from t9 +where c1= @arg00 and c2= @arg00 and c3= @arg00 and c4= @arg00 and c5= @arg00 +and c6= @arg00 and c7= @arg00 and c8= @arg00 and c9= @arg00 and c10= @arg00 +and c12= @arg00; +found +true +prepare stmt1 from "select 'true' as found from t9 +where c1= 20 and c2= 20 and c3= 20 and c4= 20 and c5= 20 and c6= 20 and c7= 20 + and c8= 20 and c9= 20 and c10= 20 and c12= 20 "; +execute stmt1 ; +found +true +prepare stmt1 from "select 'true' as found from t9 +where c1= ? and c2= ? and c3= ? and c4= ? and c5= ? + and c6= ? and c7= ? and c8= ? and c9= ? and c10= ? + and c12= ? "; +execute stmt1 using @arg00, @arg00, @arg00, @arg00, @arg00, @arg00, @arg00, +@arg00, @arg00, @arg00, @arg00 ; +found +true +set @arg00= 20.0; +select 'true' as found from t9 +where c1= 20.0 and c2= 20.0 and c3= 20.0 and c4= 20.0 and c5= 20.0 and c6= 20.0 +and c7= 20.0 and c8= 20.0 and c9= 20.0 and c10= 20.0 and c12= 20.0; +found +true +select 'true' as found from t9 +where c1= @arg00 and c2= @arg00 and c3= @arg00 and c4= @arg00 and c5= @arg00 +and c6= @arg00 and c7= @arg00 and c8= @arg00 and c9= @arg00 and c10= @arg00 +and c12= @arg00; +found +true +prepare stmt1 from "select 'true' as found from t9 +where c1= 20.0 and c2= 20.0 and c3= 20.0 and c4= 20.0 and c5= 20.0 and c6= 20.0 + and c7= 20.0 and c8= 20.0 and c9= 20.0 and c10= 20.0 and c12= 20.0 "; +execute stmt1 ; +found +true +prepare stmt1 from "select 'true' as found from t9 +where c1= ? and c2= ? and c3= ? and c4= ? and c5= ? + and c6= ? and c7= ? and c8= ? and c9= ? and c10= ? + and c12= ? "; +execute stmt1 using @arg00, @arg00, @arg00, @arg00, @arg00, @arg00, @arg00, +@arg00, @arg00, @arg00, @arg00 ; +found +true +select 'true' as found from t9 +where c1= '20' and c2= '20' and c3= '20' and c4= '20' and c5= '20' and c6= '20' + and c7= '20' and c8= '20' and c9= '20' and c10= '20' and c12= '20'; +found +true +prepare stmt1 from "select 'true' as found from t9 +where c1= '20' and c2= '20' and c3= '20' and c4= '20' and c5= '20' and c6= '20' + and c7= '20' and c8= '20' and c9= '20' and c10= '20' and c12= '20' "; +execute stmt1 ; +found +true +set @arg00= '20'; +select 'true' as found from t9 +where c1= @arg00 and c2= @arg00 and c3= @arg00 and c4= @arg00 and c5= @arg00 +and c6= @arg00 and c7= @arg00 and c8= @arg00 and c9= @arg00 and c10= @arg00 +and c12= @arg00; +found +true +prepare stmt1 from "select 'true' as found from t9 +where c1= ? and c2= ? and c3= ? and c4= ? and c5= ? + and c6= ? and c7= ? and c8= ? and c9= ? and c10= ? + and c12= ? "; +execute stmt1 using @arg00, @arg00, @arg00, @arg00, @arg00, @arg00, @arg00, +@arg00, @arg00, @arg00, @arg00 ; +found +true +select 'true' as found from t9 +where c1= CAST('20' as binary) and c2= CAST('20' as binary) and +c3= CAST('20' as binary) and c4= CAST('20' as binary) and +c5= CAST('20' as binary) and c6= CAST('20' as binary) and +c7= CAST('20' as binary) and c8= CAST('20' as binary) and +c9= CAST('20' as binary) and c10= CAST('20' as binary) and +c12= CAST('20' as binary); +found +true +prepare stmt1 from "select 'true' as found from t9 +where c1= CAST('20' as binary) and c2= CAST('20' as binary) and + c3= CAST('20' as binary) and c4= CAST('20' as binary) and + c5= CAST('20' as binary) and c6= CAST('20' as binary) and + c7= CAST('20' as binary) and c8= CAST('20' as binary) and + c9= CAST('20' as binary) and c10= CAST('20' as binary) and + c12= CAST('20' as binary) "; +execute stmt1 ; +found +true +set @arg00= CAST('20' as binary) ; +select 'true' as found from t9 +where c1= @arg00 and c2= @arg00 and c3= @arg00 and c4= @arg00 and c5= @arg00 +and c6= @arg00 and c7= @arg00 and c8= @arg00 and c9= @arg00 and c10= @arg00 +and c12= @arg00; +found +true +prepare stmt1 from "select 'true' as found from t9 +where c1= ? and c2= ? and c3= ? and c4= ? and c5= ? + and c6= ? and c7= ? and c8= ? and c9= ? and c10= ? + and c12= ? "; +execute stmt1 using @arg00, @arg00, @arg00, @arg00, @arg00, @arg00, @arg00, +@arg00, @arg00, @arg00, @arg00 ; +found +true +delete from t9 ; +test_sequence +-- some numeric overflow experiments -- +prepare my_insert from "insert into t9 + ( c21, c1, c2, c3, c4, c5, c6, c7, c8, c9, c10, c12 ) +values + ( 'O', ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ? )" ; +prepare my_select from "select c1, c2, c3, c4, c5, c6, c7, c8, c9, c10, c12 +from t9 where c21 = 'O' "; +prepare my_delete from "delete from t9 where c21 = 'O' "; +set @arg00= 9223372036854775807 ; +execute my_insert using @arg00, @arg00, @arg00, @arg00, @arg00, @arg00, +@arg00, @arg00, @arg00, @arg00, @arg00 ; +Warnings: +Warning 1264 Out of range value for column 'c1' at row 1 +Warning 1264 Out of range value for column 'c2' at row 1 +Warning 1264 Out of range value for column 'c3' at row 1 +Warning 1264 Out of range value for column 'c4' at row 1 +Warning 1264 Out of range value for column 'c5' at row 1 +Warning 1264 Out of range value for column 'c12' at row 1 +execute my_select ; +c1 127 +c2 32767 +c3 8388607 +c4 2147483647 +c5 2147483647 +c6 9223372036854775807 +c7 9.22337e+18 +c8 9.22337203685478e+18 +c9 9.22337203685478e+18 +c10 9.22337203685478e+18 +c12 9999.9999 +execute my_delete ; +set @arg00= '9223372036854775807' ; +execute my_insert using @arg00, @arg00, @arg00, @arg00, @arg00, @arg00, +@arg00, @arg00, @arg00, @arg00, @arg00 ; +Warnings: +Warning 1264 Out of range value for column 'c1' at row 1 +Warning 1264 Out of range value for column 'c2' at row 1 +Warning 1264 Out of range value for column 'c3' at row 1 +Warning 1264 Out of range value for column 'c4' at row 1 +Warning 1264 Out of range value for column 'c5' at row 1 +Warning 1264 Out of range value for column 'c12' at row 1 +execute my_select ; +c1 127 +c2 32767 +c3 8388607 +c4 2147483647 +c5 2147483647 +c6 9223372036854775807 +c7 9.22337e+18 +c8 9.22337203685478e+18 +c9 9.22337203685478e+18 +c10 9.22337203685478e+18 +c12 9999.9999 +execute my_delete ; +set @arg00= -9223372036854775808 ; +execute my_insert using @arg00, @arg00, @arg00, @arg00, @arg00, @arg00, +@arg00, @arg00, @arg00, @arg00, @arg00 ; +Warnings: +Warning 1264 Out of range value for column 'c1' at row 1 +Warning 1264 Out of range value for column 'c2' at row 1 +Warning 1264 Out of range value for column 'c3' at row 1 +Warning 1264 Out of range value for column 'c4' at row 1 +Warning 1264 Out of range value for column 'c5' at row 1 +Warning 1264 Out of range value for column 'c12' at row 1 +execute my_select ; +c1 -128 +c2 -32768 +c3 -8388608 +c4 -2147483648 +c5 -2147483648 +c6 -9223372036854775808 +c7 -9.22337e+18 +c8 -9.22337203685478e+18 +c9 -9.22337203685478e+18 +c10 -9.22337203685478e+18 +c12 -9999.9999 +execute my_delete ; +set @arg00= '-9223372036854775808' ; +execute my_insert using @arg00, @arg00, @arg00, @arg00, @arg00, @arg00, +@arg00, @arg00, @arg00, @arg00, @arg00 ; +Warnings: +Warning 1264 Out of range value for column 'c1' at row 1 +Warning 1264 Out of range value for column 'c2' at row 1 +Warning 1264 Out of range value for column 'c3' at row 1 +Warning 1264 Out of range value for column 'c4' at row 1 +Warning 1264 Out of range value for column 'c5' at row 1 +Warning 1264 Out of range value for column 'c12' at row 1 +execute my_select ; +c1 -128 +c2 -32768 +c3 -8388608 +c4 -2147483648 +c5 -2147483648 +c6 -9223372036854775808 +c7 -9.22337e+18 +c8 -9.22337203685478e+18 +c9 -9.22337203685478e+18 +c10 -9.22337203685478e+18 +c12 -9999.9999 +execute my_delete ; +set @arg00= 1.11111111111111111111e+50 ; +execute my_insert using @arg00, @arg00, @arg00, @arg00, @arg00, @arg00, +@arg00, @arg00, @arg00, @arg00, @arg00 ; +Warnings: +Warning 1264 Out of range value for column 'c1' at row 1 +Warning 1264 Out of range value for column 'c2' at row 1 +Warning 1264 Out of range value for column 'c3' at row 1 +Warning 1264 Out of range value for column 'c4' at row 1 +Warning 1264 Out of range value for column 'c5' at row 1 +Warning 1264 Out of range value for column 'c6' at row 1 +Warning 1264 Out of range value for column 'c7' at row 1 +Warning 1264 Out of range value for column 'c12' at row 1 +execute my_select ; +c1 127 +c2 32767 +c3 8388607 +c4 2147483647 +c5 2147483647 +c6 9223372036854775807 +c7 3.40282e+38 +c8 1.11111111111111e+50 +c9 1.11111111111111e+50 +c10 1.11111111111111e+50 +c12 9999.9999 +execute my_delete ; +set @arg00= '1.11111111111111111111e+50' ; +execute my_insert using @arg00, @arg00, @arg00, @arg00, @arg00, @arg00, +@arg00, @arg00, @arg00, @arg00, @arg00 ; +Warnings: +Warning 1265 Data truncated for column 'c1' at row 1 +Warning 1265 Data truncated for column 'c2' at row 1 +Warning 1265 Data truncated for column 'c3' at row 1 +Warning 1265 Data truncated for column 'c4' at row 1 +Warning 1265 Data truncated for column 'c5' at row 1 +Warning 1265 Data truncated for column 'c6' at row 1 +Warning 1264 Out of range value for column 'c7' at row 1 +Warning 1264 Out of range value for column 'c12' at row 1 +execute my_select ; +c1 1 +c2 1 +c3 1 +c4 1 +c5 1 +c6 1 +c7 3.40282e+38 +c8 1.11111111111111e+50 +c9 1.11111111111111e+50 +c10 1.11111111111111e+50 +c12 9999.9999 +execute my_delete ; +set @arg00= -1.11111111111111111111e+50 ; +execute my_insert using @arg00, @arg00, @arg00, @arg00, @arg00, @arg00, +@arg00, @arg00, @arg00, @arg00, @arg00 ; +Warnings: +Warning 1264 Out of range value for column 'c1' at row 1 +Warning 1264 Out of range value for column 'c2' at row 1 +Warning 1264 Out of range value for column 'c3' at row 1 +Warning 1264 Out of range value for column 'c4' at row 1 +Warning 1264 Out of range value for column 'c5' at row 1 +Warning 1264 Out of range value for column 'c6' at row 1 +Warning 1264 Out of range value for column 'c7' at row 1 +Warning 1264 Out of range value for column 'c12' at row 1 +execute my_select ; +c1 -128 +c2 -32768 +c3 -8388608 +c4 -2147483648 +c5 -2147483648 +c6 -9223372036854775808 +c7 -3.40282e+38 +c8 -1.11111111111111e+50 +c9 -1.11111111111111e+50 +c10 -1.11111111111111e+50 +c12 -9999.9999 +execute my_delete ; +set @arg00= '-1.11111111111111111111e+50' ; +execute my_insert using @arg00, @arg00, @arg00, @arg00, @arg00, @arg00, +@arg00, @arg00, @arg00, @arg00, @arg00 ; +Warnings: +Warning 1265 Data truncated for column 'c1' at row 1 +Warning 1265 Data truncated for column 'c2' at row 1 +Warning 1265 Data truncated for column 'c3' at row 1 +Warning 1265 Data truncated for column 'c4' at row 1 +Warning 1265 Data truncated for column 'c5' at row 1 +Warning 1265 Data truncated for column 'c6' at row 1 +Warning 1264 Out of range value for column 'c7' at row 1 +Warning 1264 Out of range value for column 'c12' at row 1 +execute my_select ; +c1 -1 +c2 -1 +c3 -1 +c4 -1 +c5 -1 +c6 -1 +c7 -3.40282e+38 +c8 -1.11111111111111e+50 +c9 -1.11111111111111e+50 +c10 -1.11111111111111e+50 +c12 -9999.9999 +execute my_delete ; +test_sequence +-- insert into string columns -- +Warnings: +Warning 1265 Data truncated for column 'c20' at row 1 +Warnings: +Warning 1265 Data truncated for column 'c20' at row 1 +Warnings: +Warning 1265 Data truncated for column 'c20' at row 1 +Warnings: +Warning 1265 Data truncated for column 'c20' at row 1 +Warnings: +Warning 1265 Data truncated for column 'c20' at row 1 +Warnings: +Warning 1265 Data truncated for column 'c20' at row 1 +Warnings: +Warning 1265 Data truncated for column 'c20' at row 1 +Warnings: +Warning 1265 Data truncated for column 'c20' at row 1 +Warnings: +Warning 1265 Data truncated for column 'c20' at row 1 +Warnings: +Warning 1265 Data truncated for column 'c20' at row 1 +Warnings: +Warning 1265 Data truncated for column 'c20' at row 1 +Warnings: +Warning 1265 Data truncated for column 'c20' at row 1 +Warnings: +Warning 1265 Data truncated for column 'c20' at row 1 +Warnings: +Warning 1265 Data truncated for column 'c20' at row 1 +Warnings: +Warning 1265 Data truncated for column 'c20' at row 1 +Warnings: +Warning 1265 Data truncated for column 'c20' at row 1 +Warnings: +Warning 1265 Data truncated for column 'c20' at row 1 +Warnings: +Warning 1265 Data truncated for column 'c20' at row 1 +Warnings: +Warning 1265 Data truncated for column 'c20' at row 1 +Warnings: +Warning 1265 Data truncated for column 'c20' at row 1 +select c1, c20, c21, c22, c23, c24, c25, c26, c27, c28, c29, c30 +from t9 where c1 >= 20 +order by c1 ; +c1 c20 c21 c22 c23 c24 c25 c26 c27 c28 c29 c30 +20 2 20 20 20 20 20 20 20 20 20 20 +21 2 21 21 21 21 21 21 21 21 21 21 +22 2 22 22 22 22 22 22 22 22 22 22 +23 2 23 23 23 23 23 23 23 23 23 23 +30 3 30 30 30 30 30 30 30 30 30 30 +31 3 31 31 31 31 31 31 31 31 31 31 +32 3 32 32 32 32 32 32 32 32 32 32 +33 3 33 33 33 33 33 33 33 33 33 33 +40 4 40 40 40 40 40 40 40 40 40 40 +41 4 41 41 41 41 41 41 41 41 41 41 +42 4 42 42 42 42 42 42 42 42 42 42 +43 4 43 43 43 43 43 43 43 43 43 43 +50 5 50.0 50.0 50.0 50.0 50.0 50.0 50.0 50.0 50.0 50.0 +51 5 51.0 51.0 51.0 51.0 51.0 51.0 51.0 51.0 51.0 51.0 +52 5 52.0 52.0 52.0 52.0 52.0 52.0 52.0 52.0 52.0 52.0 +53 5 53.0 53.0 53.0 53.0 53.0 53.0 53.0 53.0 53.0 53.0 +54 5 54 54 54.00 54.00 54.00 54.00 54.00 54.00 54.00 54.00 +55 5 55 55 55 55 55 55 55 55 55 55 +56 6 56 56 56.00 56.00 56.00 56.00 56.00 56.00 56.00 56.00 +57 6 57 57 57.00 57.00 57.00 57.00 57.00 57.00 57.00 57.00 +60 NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL +61 NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL +62 NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL +63 NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL +71 NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL +73 NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL +81 NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL +83 NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL +test_sequence +-- select .. where string column = .. -- +set @arg00= '20'; +select 'true' as found from t9 +where c1= 20 and concat(c20,substr('20',1+length(c20)))= '20' and c21= '20' and +c22= '20' and c23= '20' and c24= '20' and c25= '20' and c26= '20' and +c27= '20' and c28= '20' and c29= '20' and c30= '20' ; +found +true +select 'true' as found from t9 +where c1= 20 and concat(c20,substr(@arg00,1+length(c20)))= @arg00 and +c21= @arg00 and c22= @arg00 and c23= @arg00 and c25= @arg00 and +c26= @arg00 and c27= @arg00 and c28= @arg00 and c29= @arg00 and c30= @arg00; +found +true +prepare stmt1 from "select 'true' as found from t9 +where c1= 20 and concat(c20,substr('20',1+length(c20)))= '20' and c21= '20' and + c22= '20' and c23= '20' and c24= '20' and c25= '20' and c26= '20' and + c27= '20' and c28= '20' and c29= '20' and c30= '20'" ; +execute stmt1 ; +found +true +prepare stmt1 from "select 'true' as found from t9 +where c1= 20 and concat(c20,substr(?,1+length(c20)))= ? and + c21= ? and c22= ? and c23= ? and c25= ? and + c26= ? and c27= ? and c28= ? and c29= ? and c30= ?" ; +execute stmt1 using @arg00, @arg00, @arg00, @arg00, @arg00, @arg00, +@arg00, @arg00, @arg00, @arg00, @arg00 ; +found +true +set @arg00= CAST('20' as binary); +select 'true' as found from t9 +where c1= 20 and concat(c20,substr(CAST('20' as binary),1+length(c20))) += CAST('20' as binary) and c21= CAST('20' as binary) +and c22= CAST('20' as binary) and c23= CAST('20' as binary) and +c24= CAST('20' as binary) and c25= CAST('20' as binary) and +c26= CAST('20' as binary) and c27= CAST('20' as binary) and +c28= CAST('20' as binary) and c29= CAST('20' as binary) and +c30= CAST('20' as binary) ; +found +true +select 'true' as found from t9 +where c1= 20 and concat(c20,substr(@arg00,1+length(c20))) = @arg00 and +c21= @arg00 and c22= @arg00 and c23= @arg00 and c25= @arg00 and +c26= @arg00 and c27= @arg00 and c28= @arg00 and c29= @arg00 and +c30= @arg00; +found +true +prepare stmt1 from "select 'true' as found from t9 +where c1= 20 and concat(c20,substr(CAST('20' as binary),1+length(c20))) + = CAST('20' as binary) and c21= CAST('20' as binary) + and c22= CAST('20' as binary) and c23= CAST('20' as binary) and + c24= CAST('20' as binary) and c25= CAST('20' as binary) and + c26= CAST('20' as binary) and c27= CAST('20' as binary) and + c28= CAST('20' as binary) and c29= CAST('20' as binary) and + c30= CAST('20' as binary)" ; +execute stmt1 ; +found +true +prepare stmt1 from "select 'true' as found from t9 +where c1= 20 and concat(c20,substr(?,1+length(c20))) = ? and c21= ? and + c22= ? and c23= ? and c25= ? and c26= ? and c27= ? and c28= ? and + c29= ? and c30= ?"; +execute stmt1 using @arg00, @arg00, @arg00, @arg00, @arg00, @arg00, +@arg00, @arg00, @arg00, @arg00, @arg00 ; +found +true +set @arg00= 20; +select 'true' as found from t9 +where c1= 20 and concat(c20,substr(20,1+length(c20)))= 20 and c21= 20 and +c22= 20 and c23= 20 and c24= 20 and c25= 20 and c26= 20 and +c27= 20 and c28= 20 and c29= 20 and c30= 20 ; +found +true +select 'true' as found from t9 +where c1= 20 and concat(c20,substr(@arg00,1+length(c20)))= @arg00 and +c21= @arg00 and c22= @arg00 and c23= @arg00 and c25= @arg00 and +c26= @arg00 and c27= @arg00 and c28= @arg00 and c29= @arg00 and c30= @arg00; +found +true +prepare stmt1 from "select 'true' as found from t9 +where c1= 20 and concat(c20,substr(20,1+length(c20)))= 20 and c21= 20 and + c22= 20 and c23= 20 and c24= 20 and c25= 20 and c26= 20 and + c27= 20 and c28= 20 and c29= 20 and c30= 20" ; +execute stmt1 ; +found +true +prepare stmt1 from "select 'true' as found from t9 +where c1= 20 and concat(c20,substr(?,1+length(c20)))= ? and + c21= ? and c22= ? and c23= ? and c25= ? and + c26= ? and c27= ? and c28= ? and c29= ? and c30= ?" ; +execute stmt1 using @arg00, @arg00, @arg00, @arg00, @arg00, @arg00, +@arg00, @arg00, @arg00, @arg00, @arg00 ; +found +true +set @arg00= 20.0; +select 'true' as found from t9 +where c1= 20 and concat(c20,substr(20.0,1+length(c20)))= 20.0 and c21= 20.0 and +c22= 20.0 and c23= 20.0 and c24= 20.0 and c25= 20.0 and c26= 20.0 and +c27= 20.0 and c28= 20.0 and c29= 20.0 and c30= 20.0 ; +found +true +select 'true' as found from t9 +where c1= 20 and concat(c20,substr(@arg00,1+length(c20)))= @arg00 and +c21= @arg00 and c22= @arg00 and c23= @arg00 and c25= @arg00 and +c26= @arg00 and c27= @arg00 and c28= @arg00 and c29= @arg00 and c30= @arg00; +found +true +prepare stmt1 from "select 'true' as found from t9 +where c1= 20 and concat(c20,substr(20.0,1+length(c20)))= 20.0 and c21= 20.0 and + c22= 20.0 and c23= 20.0 and c24= 20.0 and c25= 20.0 and c26= 20.0 and + c27= 20.0 and c28= 20.0 and c29= 20.0 and c30= 20.0" ; +execute stmt1 ; +found +true +prepare stmt1 from "select 'true' as found from t9 +where c1= 20 and concat(c20,substr(?,1+length(c20)))= ? and + c21= ? and c22= ? and c23= ? and c25= ? and + c26= ? and c27= ? and c28= ? and c29= ? and c30= ?" ; +execute stmt1 using @arg00, @arg00, @arg00, @arg00, @arg00, @arg00, +@arg00, @arg00, @arg00, @arg00, @arg00 ; +found +true +delete from t9 ; +test_sequence +-- insert into date/time columns -- +Warnings: +Warning 1265 Data truncated for column 'c17' at row 1 +Warnings: +Warning 1265 Data truncated for column 'c17' at row 1 +Warnings: +Warning 1265 Data truncated for column 'c17' at row 1 +Warnings: +Warning 1265 Data truncated for column 'c17' at row 1 +Warnings: +Warning 1265 Data truncated for column 'c17' at row 1 +Warnings: +Warning 1265 Data truncated for column 'c17' at row 1 +Warnings: +Warning 1265 Data truncated for column 'c17' at row 1 +Warnings: +Warning 1265 Data truncated for column 'c17' at row 1 +Warnings: +Warning 1264 Out of range value for column 'c13' at row 1 +Warning 1264 Out of range value for column 'c14' at row 1 +Warning 1265 Data truncated for column 'c15' at row 1 +Warning 1264 Out of range value for column 'c16' at row 1 +Warning 1264 Out of range value for column 'c17' at row 1 +Warnings: +Warning 1264 Out of range value for column 'c13' at row 1 +Warning 1264 Out of range value for column 'c14' at row 1 +Warning 1265 Data truncated for column 'c15' at row 1 +Warning 1264 Out of range value for column 'c16' at row 1 +Warning 1264 Out of range value for column 'c17' at row 1 +Warnings: +Warning 1264 Out of range value for column 'c13' at row 1 +Warning 1264 Out of range value for column 'c14' at row 1 +Warning 1265 Data truncated for column 'c15' at row 1 +Warning 1264 Out of range value for column 'c16' at row 1 +Warning 1264 Out of range value for column 'c17' at row 1 +Warnings: +Warning 1264 Out of range value for column 'c13' at row 1 +Warning 1264 Out of range value for column 'c14' at row 1 +Warning 1265 Data truncated for column 'c15' at row 1 +Warning 1264 Out of range value for column 'c16' at row 1 +Warning 1264 Out of range value for column 'c17' at row 1 +Warnings: +Warning 1265 Data truncated for column 'c15' at row 1 +Warning 1264 Out of range value for column 'c16' at row 1 +Warning 1264 Out of range value for column 'c17' at row 1 +Warnings: +Warning 1265 Data truncated for column 'c15' at row 1 +Warning 1264 Out of range value for column 'c16' at row 1 +Warning 1264 Out of range value for column 'c17' at row 1 +Warnings: +Warning 1265 Data truncated for column 'c15' at row 1 +Warning 1264 Out of range value for column 'c16' at row 1 +Warning 1264 Out of range value for column 'c17' at row 1 +Warnings: +Warning 1265 Data truncated for column 'c15' at row 1 +Warning 1264 Out of range value for column 'c16' at row 1 +Warning 1264 Out of range value for column 'c17' at row 1 +select c1, c13, c14, c15, c16, c17 from t9 order by c1 ; +c1 c13 c14 c15 c16 c17 +20 1991-01-01 1991-01-01 01:01:01 1991-01-01 01:01:01 01:01:01 1991 +21 1991-01-01 1991-01-01 01:01:01 1991-01-01 01:01:01 01:01:01 1991 +22 1991-01-01 1991-01-01 01:01:01 1991-01-01 01:01:01 01:01:01 1991 +23 1991-01-01 1991-01-01 01:01:01 1991-01-01 01:01:01 01:01:01 1991 +30 1991-01-01 1991-01-01 01:01:01 1991-01-01 01:01:01 01:01:01 1991 +31 1991-01-01 1991-01-01 01:01:01 1991-01-01 01:01:01 01:01:01 1991 +32 1991-01-01 1991-01-01 01:01:01 1991-01-01 01:01:01 01:01:01 1991 +33 1991-01-01 1991-01-01 01:01:01 1991-01-01 01:01:01 01:01:01 1991 +40 0000-00-00 0000-00-00 00:00:00 0000-00-00 00:00:00 838:59:59 0000 +41 0000-00-00 0000-00-00 00:00:00 0000-00-00 00:00:00 838:59:59 0000 +42 0000-00-00 0000-00-00 00:00:00 0000-00-00 00:00:00 838:59:59 0000 +43 0000-00-00 0000-00-00 00:00:00 0000-00-00 00:00:00 838:59:59 0000 +50 2001-00-00 2001-00-00 00:00:00 0000-00-00 00:00:00 838:59:59 0000 +51 2010-00-00 2010-00-00 00:00:00 0000-00-00 00:00:00 838:59:59 0000 +52 2001-00-00 2001-00-00 00:00:00 0000-00-00 00:00:00 838:59:59 0000 +53 2001-00-00 2001-00-00 00:00:00 0000-00-00 00:00:00 838:59:59 0000 +60 NULL NULL 1991-01-01 01:01:01 NULL NULL +61 NULL NULL 1991-01-01 01:01:01 NULL NULL +62 NULL NULL 1991-01-01 01:01:01 NULL NULL +63 NULL NULL 1991-01-01 01:01:01 NULL NULL +71 NULL NULL 1991-01-01 01:01:01 NULL NULL +73 NULL NULL 1991-01-01 01:01:01 NULL NULL +81 NULL NULL 1991-01-01 01:01:01 NULL NULL +83 NULL NULL 1991-01-01 01:01:01 NULL NULL +test_sequence +-- select .. where date/time column = .. -- +set @arg00= '1991-01-01 01:01:01' ; +select 'true' as found from t9 +where c1= 20 and c13= '1991-01-01 01:01:01' and c14= '1991-01-01 01:01:01' and +c15= '1991-01-01 01:01:01' and c16= '1991-01-01 01:01:01' and +c17= '1991-01-01 01:01:01' ; +found +true +select 'true' as found from t9 +where c1= 20 and c13= @arg00 and c14= @arg00 and c15= @arg00 and c16= @arg00 +and c17= @arg00 ; +found +true +prepare stmt1 from "select 'true' as found from t9 +where c1= 20 and c13= '1991-01-01 01:01:01' and c14= '1991-01-01 01:01:01' and + c15= '1991-01-01 01:01:01' and c16= '1991-01-01 01:01:01' and + c17= '1991-01-01 01:01:01'" ; +execute stmt1 ; +found +true +prepare stmt1 from "select 'true' as found from t9 +where c1= 20 and c13= ? and c14= ? and c15= ? and c16= ? and c17= ?" ; +execute stmt1 using @arg00, @arg00, @arg00, @arg00, @arg00 ; +found +true +set @arg00= CAST('1991-01-01 01:01:01' as datetime) ; +select 'true' as found from t9 +where c1= 20 and c13= CAST('1991-01-01 01:01:01' as datetime) and +c14= CAST('1991-01-01 01:01:01' as datetime) and +c15= CAST('1991-01-01 01:01:01' as datetime) and +c16= CAST('1991-01-01 01:01:01' as datetime) and +c17= CAST('1991-01-01 01:01:01' as datetime) ; +found +true +select 'true' as found from t9 +where c1= 20 and c13= @arg00 and c14= @arg00 and c15= @arg00 and c16= @arg00 +and c17= @arg00 ; +found +true +prepare stmt1 from "select 'true' as found from t9 +where c1= 20 and c13= CAST('1991-01-01 01:01:01' as datetime) and + c14= CAST('1991-01-01 01:01:01' as datetime) and + c15= CAST('1991-01-01 01:01:01' as datetime) and + c16= CAST('1991-01-01 01:01:01' as datetime) and + c17= CAST('1991-01-01 01:01:01' as datetime)" ; +execute stmt1 ; +found +true +prepare stmt1 from "select 'true' as found from t9 +where c1= 20 and c13= ? and c14= ? and c15= ? and c16= ? and c17= ?" ; +execute stmt1 using @arg00, @arg00, @arg00, @arg00, @arg00 ; +found +true +set @arg00= 1991 ; +select 'true' as found from t9 +where c1= 20 and c17= 1991 ; +found +true +select 'true' as found from t9 +where c1= 20 and c17= @arg00 ; +found +true +prepare stmt1 from "select 'true' as found from t9 +where c1= 20 and c17= 1991" ; +execute stmt1 ; +found +true +prepare stmt1 from "select 'true' as found from t9 +where c1= 20 and c17= ?" ; +execute stmt1 using @arg00 ; +found +true +set @arg00= 1.991e+3 ; +select 'true' as found from t9 +where c1= 20 and abs(c17 - 1.991e+3) < 0.01 ; +found +true +select 'true' as found from t9 +where c1= 20 and abs(c17 - @arg00) < 0.01 ; +found +true +prepare stmt1 from "select 'true' as found from t9 +where c1= 20 and abs(c17 - 1.991e+3) < 0.01" ; +execute stmt1 ; +found +true +prepare stmt1 from "select 'true' as found from t9 +where c1= 20 and abs(c17 - ?) < 0.01" ; +execute stmt1 using @arg00 ; +found +true +drop table t1, t9; diff --git a/mysql-test/t/maria.test b/mysql-test/t/maria.test new file mode 100644 index 00000000000..59d36206039 --- /dev/null +++ b/mysql-test/t/maria.test @@ -0,0 +1,802 @@ +# +# Test bugs in the MARIA code +# +-- source include/have_maria.inc + +# Initialise +--disable_warnings +drop table if exists t1,t2; +--enable_warnings +SET SQL_WARNINGS=1; + +# +# Test problem with CHECK TABLE; +# + +CREATE TABLE t1 ( + STRING_DATA char(255) default NULL, + KEY string_data (STRING_DATA) +) ENGINE=MARIA; + +INSERT INTO t1 VALUES ('AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA'); +INSERT INTO t1 VALUES ('DDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDD'); +INSERT INTO t1 VALUES ('FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF'); +INSERT INTO t1 VALUES ('FGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGG'); +INSERT INTO t1 VALUES ('HHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHH'); +INSERT INTO t1 VALUES ('WWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWWW'); +CHECK TABLE t1; +drop table t1; + +# +# Test problem with rows that are 65517-65520 bytes long +# + +create table t1 (a tinyint not null auto_increment, b blob not null, primary key (a)) engine=maria; + +let $1=100; +disable_query_log; +--disable_warnings +SET SQL_WARNINGS=0; +while ($1) +{ + eval insert into t1 (b) values(repeat(char(65+$1),65550-$1)); + dec $1; +} +SET SQL_WARNINGS=1; +--enable_warnings +enable_query_log; +check table t1; +repair table t1; +delete from t1 where (a & 1); +check table t1; +repair table t1; +check table t1; +drop table t1; + +# +# Test bug: Two optimize in a row reset index cardinality +# + +create table t1 (a int not null auto_increment, b int not null, primary key (a), index(b)) engine=maria; +insert into t1 (b) values (1),(2),(2),(2),(2); +optimize table t1; +show index from t1; +optimize table t1; +show index from t1; +drop table t1; + +# +# Test of how ORDER BY works when doing it on the whole table +# + +create table t1 (a int not null, b int not null, c int not null, primary key (a),key(b)) engine=maria; +insert into t1 values (3,3,3),(1,1,1),(2,2,2),(4,4,4); +explain select * from t1 order by a; +explain select * from t1 order by b; +explain select * from t1 order by c; +explain select a from t1 order by a; +explain select b from t1 order by b; +explain select a,b from t1 order by b; +explain select a,b from t1; +explain select a,b,c from t1; +drop table t1; + +# +# Test of OPTIMIZE of locked and modified tables +# +CREATE TABLE t1 (a INT) engine=maria; +INSERT INTO t1 VALUES (1), (2), (3); +LOCK TABLES t1 WRITE; +INSERT INTO t1 VALUES (1), (2), (3); +OPTIMIZE TABLE t1; +DROP TABLE t1; + +# +# Test of optimize, when only mi_sort_index (but not mi_repair*) is done +# in ha_maria::repair, and index size is changed (decreased). +# + +create table t1 ( t1 char(255), key(t1(250))) engine=maria; +insert t1 values ('137513751375137513751375137513751375137569516951695169516951695169516951695169'); +insert t1 values ('178417841784178417841784178417841784178403420342034203420342034203420342034203'); +insert t1 values ('213872387238723872387238723872387238723867376737673767376737673767376737673767'); +insert t1 values ('242624262426242624262426242624262426242607890789078907890789078907890789078907'); +insert t1 values ('256025602560256025602560256025602560256011701170117011701170117011701170117011'); +insert t1 values ('276027602760276027602760276027602760276001610161016101610161016101610161016101'); +insert t1 values ('281528152815281528152815281528152815281564956495649564956495649564956495649564'); +insert t1 values ('292129212921292129212921292129212921292102100210021002100210021002100210021002'); +insert t1 values ('380638063806380638063806380638063806380634483448344834483448344834483448344834'); +insert t1 values ('411641164116411641164116411641164116411616301630163016301630163016301630163016'); +insert t1 values ('420842084208420842084208420842084208420899889988998899889988998899889988998899'); +insert t1 values ('438443844384438443844384438443844384438482448244824482448244824482448244824482'); +insert t1 values ('443244324432443244324432443244324432443239613961396139613961396139613961396139'); +insert t1 values ('485448544854485448544854485448544854485477847784778477847784778477847784778477'); +insert t1 values ('494549454945494549454945494549454945494555275527552755275527552755275527552755'); +insert t1 values ('538647864786478647864786478647864786478688918891889188918891889188918891889188'); +insert t1 values ('565556555655565556555655565556555655565554845484548454845484548454845484548454'); +insert t1 values ('607860786078607860786078607860786078607856665666566656665666566656665666566656'); +insert t1 values ('640164016401640164016401640164016401640141274127412741274127412741274127412741'); +insert t1 values ('719471947194719471947194719471947194719478717871787178717871787178717871787178'); +insert t1 values ('742574257425742574257425742574257425742549604960496049604960496049604960496049'); +insert t1 values ('887088708870887088708870887088708870887035963596359635963596359635963596359635'); +insert t1 values ('917791779177917791779177917791779177917773857385738573857385738573857385738573'); +insert t1 values ('933293329332933293329332933293329332933278987898789878987898789878987898789878'); +insert t1 values ('963896389638963896389638963896389638963877807780778077807780778077807780778077'); +delete from t1 where t1>'2'; +insert t1 values ('70'), ('84'), ('60'), ('20'), ('76'), ('89'), ('49'), ('50'), +('88'), ('61'), ('42'), ('98'), ('39'), ('30'), ('25'), ('66'), ('61'), ('48'), +('80'), ('84'), ('98'), ('19'), ('91'), ('42'), ('47'); +optimize table t1; +check table t1; +drop table t1; + +# +# test of maria with huge number of packed fields +# + +create table t1 (i1 int, i2 int, i3 int, i4 int, i5 int, i6 int, i7 int, i8 +int, i9 int, i10 int, i11 int, i12 int, i13 int, i14 int, i15 int, i16 int, i17 +int, i18 int, i19 int, i20 int, i21 int, i22 int, i23 int, i24 int, i25 int, +i26 int, i27 int, i28 int, i29 int, i30 int, i31 int, i32 int, i33 int, i34 +int, i35 int, i36 int, i37 int, i38 int, i39 int, i40 int, i41 int, i42 int, +i43 int, i44 int, i45 int, i46 int, i47 int, i48 int, i49 int, i50 int, i51 +int, i52 int, i53 int, i54 int, i55 int, i56 int, i57 int, i58 int, i59 int, +i60 int, i61 int, i62 int, i63 int, i64 int, i65 int, i66 int, i67 int, i68 +int, i69 int, i70 int, i71 int, i72 int, i73 int, i74 int, i75 int, i76 int, +i77 int, i78 int, i79 int, i80 int, i81 int, i82 int, i83 int, i84 int, i85 +int, i86 int, i87 int, i88 int, i89 int, i90 int, i91 int, i92 int, i93 int, +i94 int, i95 int, i96 int, i97 int, i98 int, i99 int, i100 int, i101 int, i102 +int, i103 int, i104 int, i105 int, i106 int, i107 int, i108 int, i109 int, i110 +int, i111 int, i112 int, i113 int, i114 int, i115 int, i116 int, i117 int, i118 +int, i119 int, i120 int, i121 int, i122 int, i123 int, i124 int, i125 int, i126 +int, i127 int, i128 int, i129 int, i130 int, i131 int, i132 int, i133 int, i134 +int, i135 int, i136 int, i137 int, i138 int, i139 int, i140 int, i141 int, i142 +int, i143 int, i144 int, i145 int, i146 int, i147 int, i148 int, i149 int, i150 +int, i151 int, i152 int, i153 int, i154 int, i155 int, i156 int, i157 int, i158 +int, i159 int, i160 int, i161 int, i162 int, i163 int, i164 int, i165 int, i166 +int, i167 int, i168 int, i169 int, i170 int, i171 int, i172 int, i173 int, i174 +int, i175 int, i176 int, i177 int, i178 int, i179 int, i180 int, i181 int, i182 +int, i183 int, i184 int, i185 int, i186 int, i187 int, i188 int, i189 int, i190 +int, i191 int, i192 int, i193 int, i194 int, i195 int, i196 int, i197 int, i198 +int, i199 int, i200 int, i201 int, i202 int, i203 int, i204 int, i205 int, i206 +int, i207 int, i208 int, i209 int, i210 int, i211 int, i212 int, i213 int, i214 +int, i215 int, i216 int, i217 int, i218 int, i219 int, i220 int, i221 int, i222 +int, i223 int, i224 int, i225 int, i226 int, i227 int, i228 int, i229 int, i230 +int, i231 int, i232 int, i233 int, i234 int, i235 int, i236 int, i237 int, i238 +int, i239 int, i240 int, i241 int, i242 int, i243 int, i244 int, i245 int, i246 +int, i247 int, i248 int, i249 int, i250 int, i251 int, i252 int, i253 int, i254 +int, i255 int, i256 int, i257 int, i258 int, i259 int, i260 int, i261 int, i262 +int, i263 int, i264 int, i265 int, i266 int, i267 int, i268 int, i269 int, i270 +int, i271 int, i272 int, i273 int, i274 int, i275 int, i276 int, i277 int, i278 +int, i279 int, i280 int, i281 int, i282 int, i283 int, i284 int, i285 int, i286 +int, i287 int, i288 int, i289 int, i290 int, i291 int, i292 int, i293 int, i294 +int, i295 int, i296 int, i297 int, i298 int, i299 int, i300 int, i301 int, i302 +int, i303 int, i304 int, i305 int, i306 int, i307 int, i308 int, i309 int, i310 +int, i311 int, i312 int, i313 int, i314 int, i315 int, i316 int, i317 int, i318 +int, i319 int, i320 int, i321 int, i322 int, i323 int, i324 int, i325 int, i326 +int, i327 int, i328 int, i329 int, i330 int, i331 int, i332 int, i333 int, i334 +int, i335 int, i336 int, i337 int, i338 int, i339 int, i340 int, i341 int, i342 +int, i343 int, i344 int, i345 int, i346 int, i347 int, i348 int, i349 int, i350 +int, i351 int, i352 int, i353 int, i354 int, i355 int, i356 int, i357 int, i358 +int, i359 int, i360 int, i361 int, i362 int, i363 int, i364 int, i365 int, i366 +int, i367 int, i368 int, i369 int, i370 int, i371 int, i372 int, i373 int, i374 +int, i375 int, i376 int, i377 int, i378 int, i379 int, i380 int, i381 int, i382 +int, i383 int, i384 int, i385 int, i386 int, i387 int, i388 int, i389 int, i390 +int, i391 int, i392 int, i393 int, i394 int, i395 int, i396 int, i397 int, i398 +int, i399 int, i400 int, i401 int, i402 int, i403 int, i404 int, i405 int, i406 +int, i407 int, i408 int, i409 int, i410 int, i411 int, i412 int, i413 int, i414 +int, i415 int, i416 int, i417 int, i418 int, i419 int, i420 int, i421 int, i422 +int, i423 int, i424 int, i425 int, i426 int, i427 int, i428 int, i429 int, i430 +int, i431 int, i432 int, i433 int, i434 int, i435 int, i436 int, i437 int, i438 +int, i439 int, i440 int, i441 int, i442 int, i443 int, i444 int, i445 int, i446 +int, i447 int, i448 int, i449 int, i450 int, i451 int, i452 int, i453 int, i454 +int, i455 int, i456 int, i457 int, i458 int, i459 int, i460 int, i461 int, i462 +int, i463 int, i464 int, i465 int, i466 int, i467 int, i468 int, i469 int, i470 +int, i471 int, i472 int, i473 int, i474 int, i475 int, i476 int, i477 int, i478 +int, i479 int, i480 int, i481 int, i482 int, i483 int, i484 int, i485 int, i486 +int, i487 int, i488 int, i489 int, i490 int, i491 int, i492 int, i493 int, i494 +int, i495 int, i496 int, i497 int, i498 int, i499 int, i500 int, i501 int, i502 +int, i503 int, i504 int, i505 int, i506 int, i507 int, i508 int, i509 int, i510 +int, i511 int, i512 int, i513 int, i514 int, i515 int, i516 int, i517 int, i518 +int, i519 int, i520 int, i521 int, i522 int, i523 int, i524 int, i525 int, i526 +int, i527 int, i528 int, i529 int, i530 int, i531 int, i532 int, i533 int, i534 +int, i535 int, i536 int, i537 int, i538 int, i539 int, i540 int, i541 int, i542 +int, i543 int, i544 int, i545 int, i546 int, i547 int, i548 int, i549 int, i550 +int, i551 int, i552 int, i553 int, i554 int, i555 int, i556 int, i557 int, i558 +int, i559 int, i560 int, i561 int, i562 int, i563 int, i564 int, i565 int, i566 +int, i567 int, i568 int, i569 int, i570 int, i571 int, i572 int, i573 int, i574 +int, i575 int, i576 int, i577 int, i578 int, i579 int, i580 int, i581 int, i582 +int, i583 int, i584 int, i585 int, i586 int, i587 int, i588 int, i589 int, i590 +int, i591 int, i592 int, i593 int, i594 int, i595 int, i596 int, i597 int, i598 +int, i599 int, i600 int, i601 int, i602 int, i603 int, i604 int, i605 int, i606 +int, i607 int, i608 int, i609 int, i610 int, i611 int, i612 int, i613 int, i614 +int, i615 int, i616 int, i617 int, i618 int, i619 int, i620 int, i621 int, i622 +int, i623 int, i624 int, i625 int, i626 int, i627 int, i628 int, i629 int, i630 +int, i631 int, i632 int, i633 int, i634 int, i635 int, i636 int, i637 int, i638 +int, i639 int, i640 int, i641 int, i642 int, i643 int, i644 int, i645 int, i646 +int, i647 int, i648 int, i649 int, i650 int, i651 int, i652 int, i653 int, i654 +int, i655 int, i656 int, i657 int, i658 int, i659 int, i660 int, i661 int, i662 +int, i663 int, i664 int, i665 int, i666 int, i667 int, i668 int, i669 int, i670 +int, i671 int, i672 int, i673 int, i674 int, i675 int, i676 int, i677 int, i678 +int, i679 int, i680 int, i681 int, i682 int, i683 int, i684 int, i685 int, i686 +int, i687 int, i688 int, i689 int, i690 int, i691 int, i692 int, i693 int, i694 +int, i695 int, i696 int, i697 int, i698 int, i699 int, i700 int, i701 int, i702 +int, i703 int, i704 int, i705 int, i706 int, i707 int, i708 int, i709 int, i710 +int, i711 int, i712 int, i713 int, i714 int, i715 int, i716 int, i717 int, i718 +int, i719 int, i720 int, i721 int, i722 int, i723 int, i724 int, i725 int, i726 +int, i727 int, i728 int, i729 int, i730 int, i731 int, i732 int, i733 int, i734 +int, i735 int, i736 int, i737 int, i738 int, i739 int, i740 int, i741 int, i742 +int, i743 int, i744 int, i745 int, i746 int, i747 int, i748 int, i749 int, i750 +int, i751 int, i752 int, i753 int, i754 int, i755 int, i756 int, i757 int, i758 +int, i759 int, i760 int, i761 int, i762 int, i763 int, i764 int, i765 int, i766 +int, i767 int, i768 int, i769 int, i770 int, i771 int, i772 int, i773 int, i774 +int, i775 int, i776 int, i777 int, i778 int, i779 int, i780 int, i781 int, i782 +int, i783 int, i784 int, i785 int, i786 int, i787 int, i788 int, i789 int, i790 +int, i791 int, i792 int, i793 int, i794 int, i795 int, i796 int, i797 int, i798 +int, i799 int, i800 int, i801 int, i802 int, i803 int, i804 int, i805 int, i806 +int, i807 int, i808 int, i809 int, i810 int, i811 int, i812 int, i813 int, i814 +int, i815 int, i816 int, i817 int, i818 int, i819 int, i820 int, i821 int, i822 +int, i823 int, i824 int, i825 int, i826 int, i827 int, i828 int, i829 int, i830 +int, i831 int, i832 int, i833 int, i834 int, i835 int, i836 int, i837 int, i838 +int, i839 int, i840 int, i841 int, i842 int, i843 int, i844 int, i845 int, i846 +int, i847 int, i848 int, i849 int, i850 int, i851 int, i852 int, i853 int, i854 +int, i855 int, i856 int, i857 int, i858 int, i859 int, i860 int, i861 int, i862 +int, i863 int, i864 int, i865 int, i866 int, i867 int, i868 int, i869 int, i870 +int, i871 int, i872 int, i873 int, i874 int, i875 int, i876 int, i877 int, i878 +int, i879 int, i880 int, i881 int, i882 int, i883 int, i884 int, i885 int, i886 +int, i887 int, i888 int, i889 int, i890 int, i891 int, i892 int, i893 int, i894 +int, i895 int, i896 int, i897 int, i898 int, i899 int, i900 int, i901 int, i902 +int, i903 int, i904 int, i905 int, i906 int, i907 int, i908 int, i909 int, i910 +int, i911 int, i912 int, i913 int, i914 int, i915 int, i916 int, i917 int, i918 +int, i919 int, i920 int, i921 int, i922 int, i923 int, i924 int, i925 int, i926 +int, i927 int, i928 int, i929 int, i930 int, i931 int, i932 int, i933 int, i934 +int, i935 int, i936 int, i937 int, i938 int, i939 int, i940 int, i941 int, i942 +int, i943 int, i944 int, i945 int, i946 int, i947 int, i948 int, i949 int, i950 +int, i951 int, i952 int, i953 int, i954 int, i955 int, i956 int, i957 int, i958 +int, i959 int, i960 int, i961 int, i962 int, i963 int, i964 int, i965 int, i966 +int, i967 int, i968 int, i969 int, i970 int, i971 int, i972 int, i973 int, i974 +int, i975 int, i976 int, i977 int, i978 int, i979 int, i980 int, i981 int, i982 +int, i983 int, i984 int, i985 int, i986 int, i987 int, i988 int, i989 int, i990 +int, i991 int, i992 int, i993 int, i994 int, i995 int, i996 int, i997 int, i998 +int, i999 int, i1000 int, b blob) row_format=dynamic engine=maria; +insert into t1 values (1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, +1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, +1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, +1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, +1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, +1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, +1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, +1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, +1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, +1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, +1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, +1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, +1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, +1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, +1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, +1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, +1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, +1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, +1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, +1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, +1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, +1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, +1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, +1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, +1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, +1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, +1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, +1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, +1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, +1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, +1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, +1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, +1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, +1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, +1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, +1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, +1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, +1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, +1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, "Sergei"); +update t1 set b=repeat('a',256); +update t1 set i1=0, i2=0, i3=0, i4=0, i5=0, i6=0, i7=0; +check table t1; +delete from t1 where i8=1; +select i1,i2 from t1; +check table t1; +drop table t1; + +# +# Test of REPAIR that once failed +# +CREATE TABLE `t1` ( + `post_id` mediumint(8) unsigned NOT NULL auto_increment, + `topic_id` mediumint(8) unsigned NOT NULL default '0', + `post_time` datetime NOT NULL default '0000-00-00 00:00:00', + `post_text` text NOT NULL, + `icon_url` varchar(10) NOT NULL default '', + `sign` tinyint(1) unsigned NOT NULL default '0', + `post_edit` varchar(150) NOT NULL default '', + `poster_login` varchar(35) NOT NULL default '', + `ip` varchar(15) NOT NULL default '', + PRIMARY KEY (`post_id`), + KEY `post_time` (`post_time`), + KEY `ip` (`ip`), + KEY `poster_login` (`poster_login`), + KEY `topic_id` (`topic_id`), + FULLTEXT KEY `post_text` (`post_text`) +) ENGINE=MARIA; + +INSERT INTO t1 (post_text) VALUES ('ceci est un test'),('ceci est un test'),('ceci est un test'),('ceci est un test'),('ceci est un test'); + +REPAIR TABLE t1; +CHECK TABLE t1; +drop table t1; + +# +# Test of creating table with too long key +# + +--error 1071 +CREATE TABLE t1 (a varchar(255), b varchar(255), c varchar(255), d varchar(255), e varchar(255), KEY t1 (a, b, c, d, e)) engine=maria; +CREATE TABLE t1 (a varchar(255), b varchar(255), c varchar(255), d varchar(255), e varchar(255)) engine=maria; +--error 1071 +ALTER TABLE t1 ADD INDEX t1 (a, b, c, d, e); +DROP TABLE t1; + +# +# Test of cardinality of keys with NULL +# + +CREATE TABLE t1 (a int not null, b int, c int, key(b), key(c), key(a,b), key(c,a)) engine=maria; +INSERT into t1 values (0, null, 0), (0, null, 1), (0, null, 2), (0, null,3), (1,1,4); +create table t2 (a int not null, b int, c int, key(b), key(c), key(a)) engine=maria; +INSERT into t2 values (1,1,1), (2,2,2); +optimize table t1; +show index from t1; +explain select * from t1,t2 where t1.a=t2.a; +explain select * from t1,t2 force index(a) where t1.a=t2.a; +explain select * from t1 force index(a),t2 force index(a) where t1.a=t2.a; +explain select * from t1,t2 where t1.b=t2.b; +explain select * from t1,t2 force index(c) where t1.a=t2.a; +explain select * from t1 where a=0 or a=2; +explain select * from t1 force index (a) where a=0 or a=2; +explain select * from t1 where c=1; +explain select * from t1 use index() where c=1; +drop table t1,t2; + +# +# Test bug when updating a split dynamic row where keys are not changed +# + +create table t1 (a int not null auto_increment primary key, b varchar(255)) engine=maria; +insert into t1 (b) values (repeat('a',100)),(repeat('b',100)),(repeat('c',100)); +update t1 set b=repeat(left(b,1),200) where a=1; +delete from t1 where (a & 1)= 0; +update t1 set b=repeat('e',200) where a=1; +flush tables; +check table t1; + +# +# check updating with keys +# + +disable_query_log; +let $1 = 100; +while ($1) +{ + eval insert into t1 (b) values (repeat(char(($1 & 32)+65), $1)); + dec $1; +} +enable_query_log; +update t1 set b=repeat(left(b,1),255) where a between 1 and 5; +update t1 set b=repeat(left(b,1),10) where a between 32 and 43; +update t1 set b=repeat(left(b,1),2) where a between 64 and 66; +update t1 set b=repeat(left(b,1),65) where a between 67 and 70; +check table t1; +insert into t1 (b) values (repeat('z',100)); +update t1 set b="test" where left(b,1) > 'n'; +check table t1; +drop table t1; + +# +# two bugs in maria-space-stripping feature +# +create table t1 ( a text not null, key a (a(20))) engine=maria; +insert into t1 values ('aaa '),('aaa'),('aa'); +check table t1; +repair table t1; +select concat(a,'.') from t1 where a='aaa'; +select concat(a,'.') from t1 where binary a='aaa'; +update t1 set a='bbb' where a='aaa'; +select concat(a,'.') from t1; +drop table t1; + +# +# Third bug in the same code (BUG#2295) +# + +create table t1(a text not null, b text not null, c text not null, index (a(10),b(10),c(10))) engine=maria; +insert into t1 values('807780', '477', '165'); +insert into t1 values('807780', '477', '162'); +insert into t1 values('807780', '472', '162'); +select * from t1 where a='807780' and b='477' and c='165'; +drop table t1; + +# +# space-stripping in _mi_prefix_search: BUG#5284 +# +DROP TABLE IF EXISTS t1; +CREATE TABLE t1 (a varchar(150) NOT NULL, KEY (a)) engine=maria; +INSERT t1 VALUES ("can \tcan"); +INSERT t1 VALUES ("can can"); +INSERT t1 VALUES ("can"); +SELECT * FROM t1; +CHECK TABLE t1; +DROP TABLE t1; + +# +# Verify blob handling +# +create table t1 (a blob) engine=maria; +insert into t1 values('a '),('a'); +select concat(a,'.') from t1 where a='a'; +select concat(a,'.') from t1 where a='a '; +alter table t1 add key(a(2)); +select concat(a,'.') from t1 where a='a'; +select concat(a,'.') from t1 where a='a '; +drop table t1; + +# +# Test text and unique +# +create table t1 (a int not null auto_increment primary key, b text not null, unique b (b(20))) engine=maria; +insert into t1 (b) values ('a'),('b'),('c'); +select concat(b,'.') from t1; +update t1 set b='b ' where a=2; +--error 1062 +update t1 set b='b ' where a > 1; +--error 1062 +insert into t1 (b) values ('b'); +select * from t1; +delete from t1 where b='b'; +select a,concat(b,'.') from t1; +drop table t1; + +# +# Test keys with 0 segments. (Bug #3203) +# +create table t1 (a int not null) engine=maria; +create table t2 (a int not null, primary key (a)) engine=maria; +insert into t1 values (1); +insert into t2 values (1),(2); +select sql_big_result distinct t1.a from t1,t2 order by t2.a; +select distinct t1.a from t1,t2 order by t2.a; +select sql_big_result distinct t1.a from t1,t2; +explain select sql_big_result distinct t1.a from t1,t2 order by t2.a; +explain select distinct t1.a from t1,t2 order by t2.a; +drop table t1,t2; + +# +# Bug#14616 - Freshly imported table returns error 124 when using LIMIT +# +create table t1 ( + c1 varchar(32), + key (c1) +) engine=maria; +alter table t1 disable keys; +insert into t1 values ('a'), ('b'); +select c1 from t1 order by c1 limit 1; +drop table t1; + +# +# Test RTREE index +# +--error 1235, 1289 +CREATE TABLE t1 (`a` int(11) NOT NULL default '0', `b` int(11) NOT NULL default '0', UNIQUE KEY `a` USING RTREE (`a`,`b`)) ENGINE=MARIA; +# INSERT INTO t1 VALUES (1,1),(1,1); +# DELETE FROM rt WHERE a<1; +# DROP TABLE IF EXISTS t1; + +create table t1 (a int, b varchar(200), c text not null) checksum=1 engine=maria; +create table t2 (a int, b varchar(200), c text not null) checksum=0 engine=maria; +insert t1 values (1, "aaa", "bbb"), (NULL, "", "ccccc"), (0, NULL, ""); +insert t2 select * from t1; +checksum table t1, t2, t3 quick; +checksum table t1, t2, t3; +checksum table t1, t2, t3 extended; +#show table status; +drop table t1,t2; + +create table t1 (a int, key (a)) engine=maria; +show keys from t1; +alter table t1 disable keys; +show keys from t1; +create table t2 (a int) engine=maria; +let $i=1000; +set @@rand_seed1=31415926,@@rand_seed2=2718281828; +--disable_query_log +while ($i) +{ + dec $i; + insert t2 values (rand()*100000); +} +--enable_query_log +insert t1 select * from t2; +show keys from t1; +alter table t1 enable keys; +show keys from t1; +alter table t1 engine=heap; +alter table t1 disable keys; +show keys from t1; +drop table t1,t2; + +# +# index search for NULL in blob. Bug #4816 +# +create table t1 ( a tinytext, b char(1), index idx (a(1),b)) engine=maria; +insert into t1 values (null,''), (null,''); +explain select count(*) from t1 where a is null; +select count(*) from t1 where a is null; +drop table t1; + +# +# bug9188 - Corruption Can't open file: 'table.MYI' (errno: 145) +# +create table t1 (c1 int, c2 varchar(4) not null default '', + key(c2(3))) default charset=utf8 engine=maria; +insert into t1 values (1,'A'), (2, 'B'), (3, 'A'); +update t1 set c2='A B' where c1=2; +check table t1; +drop table t1; + + +# +# Bug#12296 - CHECKSUM TABLE reports 0 for the table +# This happened if the first record was marked as deleted. +# +create table t1 (c1 int) engine=maria; +insert into t1 values (1),(2),(3),(4); +checksum table t1; +delete from t1 where c1 = 1; +create table t2 engine=maria as select * from t1; +# The following returns 0 with the bug in place. +checksum table t1; +# The above should give the same number as the following. +checksum table t2; +drop table t1, t2; + +# +# BUG#12232: New maria_stats_method variable. +# + +show variables like 'maria_stats_method'; + +create table t1 (a int, key(a)) engine=maria; +insert into t1 values (0),(1),(2),(3),(4); +insert into t1 select NULL from t1; + +# default: NULLs considered inequal +analyze table t1; +show index from t1; +insert into t1 values (11); +delete from t1 where a=11; +check table t1; +show index from t1; + +# Set nulls to be equal: +set maria_stats_method=nulls_equal; +show variables like 'maria_stats_method'; +insert into t1 values (11); +delete from t1 where a=11; + +analyze table t1; +show index from t1; + +insert into t1 values (11); +delete from t1 where a=11; + +check table t1; +show index from t1; + +# Set nulls back to be equal +set maria_stats_method=DEFAULT; +show variables like 'maria_stats_method'; +insert into t1 values (11); +delete from t1 where a=11; + +analyze table t1; +show index from t1; + +insert into t1 values (11); +delete from t1 where a=11; + +check table t1; +show index from t1; + +drop table t1; + +# WL#2609, CSC#XXXX: MARIA +set maria_stats_method=nulls_ignored; +show variables like 'maria_stats_method'; + +create table t1 ( + a char(3), b char(4), c char(5), d char(6), + key(a,b,c,d) +) engine=maria; +insert into t1 values ('bcd','def1', NULL, 'zz'); +insert into t1 values ('bcd','def2', NULL, 'zz'); +insert into t1 values ('bce','def1', 'yuu', NULL); +insert into t1 values ('bce','def2', NULL, 'quux'); +analyze table t1; +show index from t1; +delete from t1; +analyze table t1; +show index from t1; + +set maria_stats_method=DEFAULT; +drop table t1; + +# BUG#13814 - key value packed incorrectly for TINYBLOBs + +create table t1( + cip INT NOT NULL, + time TIME NOT NULL, + score INT NOT NULL DEFAULT 0, + bob TINYBLOB +) engine=maria; + +insert into t1 (cip, time) VALUES (1, '00:01'), (2, '00:02'), (3,'00:03'); +insert into t1 (cip, bob, time) VALUES (4, 'a', '00:04'), (5, 'b', '00:05'), + (6, 'c', '00:06'); +select * from t1 where bob is null and cip=1; +create index bug on t1 (bob(22), cip, time); +select * from t1 where bob is null and cip=1; +drop table t1; + +# +# Bug#14980 - COUNT(*) incorrect on MARIA table with certain INDEX +# +create table t1 ( + id1 int not null auto_increment, + id2 int not null default '0', + t text not null, + primary key (id1), + key x (id2, t(32)) +) engine=maria; +insert into t1 (id2, t) values +(10, 'abc'), (10, 'abc'), (10, 'abc'), +(20, 'abc'), (20, 'abc'), (20, 'def'), +(10, 'abc'), (10, 'abc'); +select count(*) from t1 where id2 = 10; +select count(id1) from t1 where id2 = 10; +drop table t1; + +# End of 4.1 tests + +# +# Test varchar +# + +let $default=`select @@storage_engine`; +set storage_engine=MARIA; +source include/varchar.inc; + +# +# Some errors/warnings on create +# + +create table t1 (v varchar(65530), key(v)) engine=maria; +drop table if exists t1; +create table t1 (v varchar(65536)) engine=maria; +show create table t1; +drop table t1; +create table t1 (v varchar(65530) character set utf8) engine=maria; +show create table t1; +drop table t1; + +# MARIA specific varchar tests +--error 1118 +create table t1 (v varchar(65535)) engine=maria; + +eval set storage_engine=$default; + +# +# Test how DROP TABLE works if the index or data file doesn't exists + +create table t1 (a int) engine=maria; +system rm $MYSQLTEST_VARDIR/master-data/test/t1.MAI ; +drop table if exists t1; +create table t1 (a int) engine=maria; +system rm $MYSQLTEST_VARDIR/master-data/test/t1.MAI ; +--error 1051,6 +drop table t1; +create table t1 (a int) engine=maria; +system rm $MYSQLTEST_VARDIR/master-data/test/t1.MAD ; +--error 1105,6,29 +drop table t1; +--error 1051 +drop table t1; + +# +# Test concurrent insert +# First with static record length +# +set @save_concurrent_insert=@@concurrent_insert; +set global concurrent_insert=1; +create table t1 (a int) engine=maria; +insert into t1 values (1),(2),(3),(4),(5); +lock table t1 read local; +connect (con1,localhost,root,,); +connection con1; +# Insert in table without hole +insert into t1 values(6),(7); +connection default; +unlock tables; +delete from t1 where a>=3 and a<=4; +lock table t1 read local; +connection con1; +set global concurrent_insert=2; +# Insert in table with hole -> Should insert at end +insert into t1 values (8),(9); +connection default; +unlock tables; +# Insert into hole +insert into t1 values (10),(11),(12); +select * from t1; +check table t1; +drop table t1; +disconnect con1; + +# Same test with dynamic record length +create table t1 (a int, b varchar(30) default "hello") engine=maria; +insert into t1 (a) values (1),(2),(3),(4),(5); +lock table t1 read local; +connect (con1,localhost,root,,); +connection con1; +# Insert in table without hole +insert into t1 (a) values(6),(7); +connection default; +unlock tables; +delete from t1 where a>=3 and a<=4; +lock table t1 read local; +connection con1; +set global concurrent_insert=2; +# Insert in table with hole -> Should insert at end +insert into t1 (a) values (8),(9); +connection default; +unlock tables; +# Insert into hole +insert into t1 (a) values (10),(11),(12); +select a from t1; +check table t1; +drop table t1; +disconnect con1; +set global concurrent_insert=@save_concurrent_insert; + + +# BUG#9622 - ANALYZE TABLE and ALTER TABLE .. ENABLE INDEX produce +# different statistics on the same table with NULL values. +create table t1 (a int, key(a)) engine=maria; + +insert into t1 values (1),(2),(3),(4),(NULL),(NULL),(NULL),(NULL); +analyze table t1; +show keys from t1; + +alter table t1 disable keys; +alter table t1 enable keys; +show keys from t1; + +drop table t1; + +# +# Bug#10056 - PACK_KEYS option take values greater than 1 while creating table +# +create table t1 (c1 int) engine=maria pack_keys=0; +create table t2 (c1 int) engine=maria pack_keys=1; +create table t3 (c1 int) engine=maria pack_keys=default; +--error 1064 +create table t4 (c1 int) engine=maria pack_keys=2; +drop table t1, t2, t3; + +# End of 5.2 tests diff --git a/mysql-test/t/ps_maria.test b/mysql-test/t/ps_maria.test new file mode 100644 index 00000000000..2be99842132 --- /dev/null +++ b/mysql-test/t/ps_maria.test @@ -0,0 +1,46 @@ +############################################### +# # +# Prepared Statements test on MARIA tables # +# # +############################################### + +# +# NOTE: PLEASE SEE ps_1general.test (bottom) +# BEFORE ADDING NEW TEST CASES HERE !!! + +use test; + +-- source include/have_maria.inc + +let $type= 'MARIA' ; +-- source include/ps_create.inc +-- source include/ps_renew.inc + +-- source include/ps_query.inc + +# parameter in SELECT ... MATCH/AGAINST +# case derived from client_test.c: test_bug1500() +--disable_warnings +drop table if exists t2 ; +--enable_warnings +eval create table t2 (s varchar(25), fulltext(s)) +ENGINE = $type ; +insert into t2 values ('Gravedigger'), ('Greed'),('Hollow Dogs') ; +commit ; + +prepare stmt1 from ' select s from t2 where match (s) against (?) ' ; +set @arg00='Dogs' ; +execute stmt1 using @arg00 ; +prepare stmt1 from ' SELECT s FROM t2 +where match (s) against (concat(?,''digger'')) '; +set @arg00='Grave' ; +execute stmt1 using @arg00 ; +drop table t2 ; + +-- source include/ps_modify.inc +-- source include/ps_modify1.inc +-- source include/ps_conv.inc + +drop table t1, t9; + +# End of 4.1 tests diff --git a/mysys/mf_keycaches.c b/mysys/mf_keycaches.c index 38fef31fdd4..fdac198697d 100644 --- a/mysys/mf_keycaches.c +++ b/mysys/mf_keycaches.c @@ -148,7 +148,8 @@ static void safe_hash_free(SAFE_HASH *hash) Return the value stored for a key or default value if no key */ -static byte *safe_hash_search(SAFE_HASH *hash, const byte *key, uint length) +static byte *safe_hash_search(SAFE_HASH *hash, const byte *key, uint length, + byte *def) { byte *result; DBUG_ENTER("safe_hash_search"); @@ -156,7 +157,7 @@ static byte *safe_hash_search(SAFE_HASH *hash, const byte *key, uint length) result= hash_search(&hash->hash, key, length); rw_unlock(&hash->mutex); if (!result) - result= hash->default_value; + result= def; else result= ((SAFE_HASH_ENTRY*) result)->data; DBUG_PRINT("exit",("data: 0x%lx", result)); @@ -316,6 +317,7 @@ void multi_keycache_free(void) multi_key_cache_search() key key to find (usually table path) uint length Length of key. + def Default value if no key cache NOTES This function is coded in such a way that we will return the @@ -326,11 +328,13 @@ void multi_keycache_free(void) key cache to use */ -KEY_CACHE *multi_key_cache_search(byte *key, uint length) +KEY_CACHE *multi_key_cache_search(byte *key, uint length, + KEY_CACHE *def) { if (!key_cache_hash.hash.records) - return dflt_key_cache; - return (KEY_CACHE*) safe_hash_search(&key_cache_hash, key, length); + return def; + return (KEY_CACHE*) safe_hash_search(&key_cache_hash, key, length, + (void*) def); } diff --git a/mysys/my_handler.c b/mysys/my_handler.c index 56f2298a9f0..5f7bf5ff1a8 100644 --- a/mysys/my_handler.c +++ b/mysys/my_handler.c @@ -15,9 +15,12 @@ Software Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA */ -#include "my_handler.h" +#include +#include +#include +#include -int mi_compare_text(CHARSET_INFO *charset_info, uchar *a, uint a_length, +int ha_compare_text(CHARSET_INFO *charset_info, uchar *a, uint a_length, uchar *b, uint b_length, my_bool part_key, my_bool skip_end_space) { @@ -173,7 +176,7 @@ int ha_key_cmp(register HA_KEYSEG *keyseg, register uchar *a, next_key_length=key_length-b_length-pack_length; if (piks && - (flag=mi_compare_text(keyseg->charset,a,a_length,b,b_length, + (flag=ha_compare_text(keyseg->charset,a,a_length,b,b_length, (my_bool) ((nextflag & SEARCH_PREFIX) && next_key_length <= 0), (my_bool)!(nextflag & SEARCH_PREFIX)))) @@ -186,7 +189,7 @@ int ha_key_cmp(register HA_KEYSEG *keyseg, register uchar *a, { uint length=(uint) (end-a), a_length=length, b_length=length; if (piks && - (flag= mi_compare_text(keyseg->charset, a, a_length, b, b_length, + (flag= ha_compare_text(keyseg->charset, a, a_length, b, b_length, (my_bool) ((nextflag & SEARCH_PREFIX) && next_key_length <= 0), (my_bool)!(nextflag & SEARCH_PREFIX)))) @@ -234,7 +237,7 @@ int ha_key_cmp(register HA_KEYSEG *keyseg, register uchar *a, next_key_length=key_length-b_length-pack_length; if (piks && - (flag= mi_compare_text(keyseg->charset,a,a_length,b,b_length, + (flag= ha_compare_text(keyseg->charset,a,a_length,b,b_length, (my_bool) ((nextflag & SEARCH_PREFIX) && next_key_length <= 0), (my_bool) ((nextflag & (SEARCH_FIND | @@ -481,12 +484,15 @@ end: DESCRIPTION Find the first NULL value in index-suffix values tuple. - TODO Consider optimizing this fuction or its use so we don't search for - NULL values in completely NOT NULL index suffixes. + + TODO + Consider optimizing this function or its use so we don't search for + NULL values in completely NOT NULL index suffixes. RETURN - First key part that has NULL as value in values tuple, or the last key part - (with keyseg->type==HA_TYPE_END) if values tuple doesn't contain NULLs. + First key part that has NULL as value in values tuple, or the last key + part (with keyseg->type==HA_TYPE_END) if values tuple doesn't contain + NULLs. */ HA_KEYSEG *ha_find_null(HA_KEYSEG *keyseg, uchar *a) @@ -504,6 +510,7 @@ HA_KEYSEG *ha_find_null(HA_KEYSEG *keyseg, uchar *a) switch ((enum ha_base_keytype) keyseg->type) { case HA_KEYTYPE_TEXT: case HA_KEYTYPE_BINARY: + case HA_KEYTYPE_BIT: if (keyseg->flag & HA_SPACE_PACK) { int a_length; @@ -516,6 +523,8 @@ HA_KEYSEG *ha_find_null(HA_KEYSEG *keyseg, uchar *a) break; case HA_KEYTYPE_VARTEXT1: case HA_KEYTYPE_VARBINARY1: + case HA_KEYTYPE_VARTEXT2: + case HA_KEYTYPE_VARBINARY2: { int a_length; get_key_length(a_length, a); @@ -545,6 +554,9 @@ HA_KEYSEG *ha_find_null(HA_KEYSEG *keyseg, uchar *a) case HA_KEYTYPE_DOUBLE: a= end; break; + case HA_KEYTYPE_END: + DBUG_ASSERT(0); + break; } } return keyseg; diff --git a/sql/ha_maria.cc b/sql/ha_maria.cc new file mode 100644 index 00000000000..46324d0b188 --- /dev/null +++ b/sql/ha_maria.cc @@ -0,0 +1,1836 @@ +/* Copyright (C) 2006,2004 MySQL AB & MySQL Finland AB & TCX DataKonsult AB + + 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; either version 2 of the License, or + (at your option) any later version. + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ + + +#ifdef USE_PRAGMA_IMPLEMENTATION +#pragma implementation // gcc: Class implementation +#endif + +#include "mysql_priv.h" +#include +#include +#include "ha_maria.h" +#ifndef MASTER +#include "../srclib/maria/maria_def.h" +#else +#include "../storage/maria/maria_def.h" +#include "../storage/maria/ma_rt_index.h" +#endif + +ulong maria_recover_options= HA_RECOVER_NONE; +static bool ha_maria_init(); + +/* bits in maria_recover_options */ +const char *maria_recover_names[]= +{ + "DEFAULT", "BACKUP", "FORCE", "QUICK", NullS +}; +TYPELIB maria_recover_typelib= +{ + array_elements(maria_recover_names) - 1, "", + maria_recover_names, NULL +}; + +const char *maria_stats_method_names[]= +{ + "nulls_unequal", "nulls_equal", + "nulls_ignored", NullS +}; +TYPELIB maria_stats_method_typelib= +{ + array_elements(maria_stats_method_names) - 1, "", + maria_stats_method_names, NULL +}; + + +/***************************************************************************** +** MARIA tables +*****************************************************************************/ + +static handler *maria_create_handler(TABLE_SHARE * table); + +/* MARIA handlerton */ + +handlerton maria_hton= +{ + MYSQL_HANDLERTON_INTERFACE_VERSION, + "MARIA", + SHOW_OPTION_YES, + "Transactional storage engine, optimized for long running transactions", + DB_TYPE_MARIA, + ha_maria_init, + 0, /* slot */ + 0, /* savepoint size. */ + NULL, /* close_connection */ + NULL, /* savepoint */ + NULL, /* rollback to savepoint */ + NULL, /* release savepoint */ + NULL, /* commit */ + NULL, /* rollback */ + NULL, /* prepare */ + NULL, /* recover */ + NULL, /* commit_by_xid */ + NULL, /* rollback_by_xid */ + NULL, /* create_cursor_read_view */ + NULL, /* set_cursor_read_view */ + NULL, /* close_cursor_read_view */ + /* + MARIA doesn't support transactions yet and doesn't have + transaction-dependent context: cursors can survive a commit. + */ + maria_create_handler, /* Create a new handler */ + NULL, /* Drop a database */ + maria_panic, /* Panic call */ + NULL, /* Start Consistent Snapshot */ + NULL, /* Flush logs */ + NULL, /* Show status */ + NULL, /* Partition flags */ + NULL, /* Alter table flags */ + NULL, /* Alter Tablespace */ + NULL, /* Fill Files Table */ + HTON_CAN_RECREATE, + NULL, /* binlog_func */ + NULL, /* binlog_log_query */ + NULL /* release_temporary_latches */ +}; + + +static bool ha_maria_init() +{ + return test(maria_init()); +} + +static handler *maria_create_handler(TABLE_SHARE *table) +{ + return new ha_maria(table); +} + + +// collect errors printed by maria_check routines + +static void _ma_check_print_msg(HA_CHECK *param, const char *msg_type, + const char *fmt, va_list args) +{ + THD *thd= (THD *) param->thd; + Protocol *protocol= thd->protocol; + uint length, msg_length; + char msgbuf[MARIA_MAX_MSG_BUF]; + char name[NAME_LEN * 2 + 2]; + + msg_length= my_vsnprintf(msgbuf, sizeof(msgbuf), fmt, args); + msgbuf[sizeof(msgbuf) - 1]= 0; // healthy paranoia + + DBUG_PRINT(msg_type, ("message: %s", msgbuf)); + + if (!thd->vio_ok()) + { + sql_print_error(msgbuf); + return; + } + + if (param->testflag & + (T_CREATE_MISSING_KEYS | T_SAFE_REPAIR | T_AUTO_REPAIR)) + { + my_message(ER_NOT_KEYFILE, msgbuf, MYF(MY_WME)); + return; + } + length= (uint) (strxmov(name, param->db_name, ".", param->table_name, + NullS) - name); + protocol->prepare_for_resend(); + protocol->store(name, length, system_charset_info); + protocol->store(param->op_name, system_charset_info); + protocol->store(msg_type, system_charset_info); + protocol->store(msgbuf, msg_length, system_charset_info); + if (protocol->write()) + sql_print_error("Failed on my_net_write, writing to stderr instead: %s\n", + msgbuf); + return; +} + +extern "C" { + +volatile int *_ma_killed_ptr(HA_CHECK *param) +{ + /* In theory Unsafe conversion, but should be ok for now */ + return (int*) &(((THD *) (param->thd))->killed); +} + + +void _ma_check_print_error(HA_CHECK *param, const char *fmt, ...) +{ + param->error_printed |= 1; + param->out_flag |= O_DATA_LOST; + va_list args; + va_start(args, fmt); + _ma_check_print_msg(param, "error", fmt, args); + va_end(args); +} + + +void _ma_check_print_info(HA_CHECK *param, const char *fmt, ...) +{ + va_list args; + va_start(args, fmt); + _ma_check_print_msg(param, "info", fmt, args); + va_end(args); +} + + +void _ma_check_print_warning(HA_CHECK *param, const char *fmt, ...) +{ + param->warning_printed= 1; + param->out_flag |= O_DATA_LOST; + va_list args; + va_start(args, fmt); + _ma_check_print_msg(param, "warning", fmt, args); + va_end(args); +} + +} + + +ha_maria::ha_maria(TABLE_SHARE *table_arg): +handler(&maria_hton, table_arg), file(0), +int_table_flags(HA_NULL_IN_KEY | HA_CAN_FULLTEXT | HA_CAN_SQL_HANDLER | + HA_DUPP_POS | HA_CAN_INDEX_BLOBS | HA_AUTO_PART_KEY | + HA_FILE_BASED | HA_CAN_GEOMETRY | HA_READ_RND_SAME | + HA_CAN_INSERT_DELAYED | HA_CAN_BIT_FIELD), +can_enable_indexes(1) +{} + + +static const char *ha_maria_exts[]= +{ + MARIA_NAME_IEXT, + MARIA_NAME_DEXT, + NullS +}; + + +const char **ha_maria::bas_ext() const +{ + return ha_maria_exts; +} + + +const char *ha_maria::index_type(uint key_number) +{ + return ((table->key_info[key_number].flags & HA_FULLTEXT) ? + "FULLTEXT" : + (table->key_info[key_number].flags & HA_SPATIAL) ? + "SPATIAL" : + (table->key_info[key_number].algorithm == HA_KEY_ALG_RTREE) ? + "RTREE" : "BTREE"); +} + + +#ifdef HAVE_REPLICATION +int ha_maria::net_read_dump(NET * net) +{ + int data_fd= file->dfile; + int error= 0; + + my_seek(data_fd, 0L, MY_SEEK_SET, MYF(MY_WME)); + for (;;) + { + ulong packet_len= my_net_read(net); + if (!packet_len) + break; // end of file + if (packet_len == packet_error) + { + sql_print_error("ha_maria::net_read_dump - read error "); + error= -1; + goto err; + } + if (my_write(data_fd, (byte *) net->read_pos, (uint) packet_len, + MYF(MY_WME | MY_FNABP))) + { + error= errno; + goto err; + } + } +err: + return error; +} + + +int ha_maria::dump(THD * thd, int fd) +{ + MARIA_SHARE *share= file->s; + NET *net= &thd->net; + uint blocksize= share->blocksize; + my_off_t bytes_to_read= share->state.state.data_file_length; + int data_fd= file->dfile; + byte *buf= (byte *) my_malloc(blocksize, MYF(MY_WME)); + if (!buf) + return ENOMEM; + + int error= 0; + my_seek(data_fd, 0L, MY_SEEK_SET, MYF(MY_WME)); + for (; bytes_to_read > 0;) + { + uint bytes= my_read(data_fd, buf, blocksize, MYF(MY_WME)); + if (bytes == MY_FILE_ERROR) + { + error= errno; + goto err; + } + + if (fd >= 0) + { + if (my_write(fd, buf, bytes, MYF(MY_WME | MY_FNABP))) + { + error= errno ? errno : EPIPE; + goto err; + } + } + else + { + if (my_net_write(net, (char*) buf, bytes)) + { + error= errno ? errno : EPIPE; + goto err; + } + } + bytes_to_read -= bytes; + } + + if (fd < 0) + { + if (my_net_write(net, "", 0)) + error= errno ? errno : EPIPE; + net_flush(net); + } + +err: + my_free((gptr) buf, MYF(0)); + return error; +} +#endif /* HAVE_REPLICATION */ + + +bool ha_maria::check_if_locking_is_allowed(uint sql_command, + ulong type, TABLE *table, + uint count, + bool called_by_logger_thread) +{ + /* + To be able to open and lock for reading system tables like 'mysql.proc', + when we already have some tables opened and locked, and avoid deadlocks + we have to disallow write-locking of these tables with any other tables. + */ + if (table->s->system_table && + table->reginfo.lock_type >= TL_WRITE_ALLOW_WRITE && count != 1) + { + my_error(ER_WRONG_LOCK_OF_SYSTEM_TABLE, MYF(0), table->s->db.str, + table->s->table_name.str); + return FALSE; + } + return TRUE; +} + + + /* Name is here without an extension */ + +int ha_maria::open(const char *name, int mode, uint test_if_locked) +{ + uint i; + if (!(file= maria_open(name, mode, test_if_locked | HA_OPEN_FROM_SQL_LAYER))) + return (my_errno ? my_errno : -1); + + if (test_if_locked & (HA_OPEN_IGNORE_IF_LOCKED | HA_OPEN_TMP_TABLE)) + VOID(maria_extra(file, HA_EXTRA_NO_WAIT_LOCK, 0)); + +#ifdef NOT_USED + if (!(test_if_locked & HA_OPEN_TMP_TABLE) && opt_maria_use_mmap) + VOID(maria_extra(file, HA_EXTRA_MMAP, 0)); +#endif + + info(HA_STATUS_NO_LOCK | HA_STATUS_VARIABLE | HA_STATUS_CONST); + if (!(test_if_locked & HA_OPEN_WAIT_IF_LOCKED)) + VOID(maria_extra(file, HA_EXTRA_WAIT_LOCK, 0)); + if (!table->s->db_record_offset) + int_table_flags |= HA_REC_NOT_IN_SEQ; + if (file->s->options & (HA_OPTION_CHECKSUM | HA_OPTION_COMPRESS_RECORD)) + int_table_flags |= HA_HAS_CHECKSUM; + + for (i= 0; i < table->s->keys; i++) + { + struct st_plugin_int *parser= table->key_info[i].parser; + if (table->key_info[i].flags & HA_USES_PARSER) + file->s->keyinfo[i].parser= + (struct st_mysql_ftparser *) parser->plugin->info; + } + return (0); +} + + +int ha_maria::close(void) +{ + MARIA_HA *tmp= file; + file= 0; + return maria_close(tmp); +} + + +int ha_maria::write_row(byte * buf) +{ + statistic_increment(table->in_use->status_var.ha_write_count, &LOCK_status); + + /* If we have a timestamp column, update it to the current time */ + if (table->timestamp_field_type & TIMESTAMP_AUTO_SET_ON_INSERT) + table->timestamp_field->set_time(); + + /* + If we have an auto_increment column and we are writing a changed row + or a new row, then update the auto_increment value in the record. + */ + if (table->next_number_field && buf == table->record[0]) + update_auto_increment(); + return maria_write(file, buf); +} + + +int ha_maria::check(THD * thd, HA_CHECK_OPT * check_opt) +{ + if (!file) + return HA_ADMIN_INTERNAL_ERROR; + int error; + HA_CHECK param; + MARIA_SHARE *share= file->s; + const char *old_proc_info= thd->proc_info; + + thd->proc_info= "Checking table"; + mariachk_init(¶m); + param.thd= thd; + param.op_name= "check"; + param.db_name= table->s->db.str; + param.table_name= table->alias; + param.testflag= check_opt->flags | T_CHECK | T_SILENT; + param.stats_method= (enum_handler_stats_method) thd->variables. + maria_stats_method; + + if (!(table->db_stat & HA_READ_ONLY)) + param.testflag |= T_STATISTICS; + param.using_global_keycache= 1; + + if (!maria_is_crashed(file) && + (((param.testflag & T_CHECK_ONLY_CHANGED) && + !(share->state.changed & (STATE_CHANGED | STATE_CRASHED | + STATE_CRASHED_ON_REPAIR)) && + share->state.open_count == 0) || + ((param.testflag & T_FAST) && (share->state.open_count == + (uint) (share->global_changed ? 1 : + 0))))) + return HA_ADMIN_ALREADY_DONE; + + error= maria_chk_status(¶m, file); // Not fatal + error= maria_chk_size(¶m, file); + if (!error) + error |= maria_chk_del(¶m, file, param.testflag); + if (!error) + error= maria_chk_key(¶m, file); + if (!error) + { + if ((!(param.testflag & T_QUICK) && + ((share->options & + (HA_OPTION_PACK_RECORD | HA_OPTION_COMPRESS_RECORD)) || + (param.testflag & (T_EXTEND | T_MEDIUM)))) || maria_is_crashed(file)) + { + uint old_testflag= param.testflag; + param.testflag |= T_MEDIUM; + if (!(error= init_io_cache(¶m.read_cache, file->dfile, + my_default_record_cache_size, READ_CACHE, + share->pack.header_length, 1, MYF(MY_WME)))) + { + error= maria_chk_data_link(¶m, file, param.testflag & T_EXTEND); + end_io_cache(&(param.read_cache)); + } + param.testflag= old_testflag; + } + } + if (!error) + { + if ((share->state.changed & (STATE_CHANGED | + STATE_CRASHED_ON_REPAIR | + STATE_CRASHED | STATE_NOT_ANALYZED)) || + (param.testflag & T_STATISTICS) || maria_is_crashed(file)) + { + file->update |= HA_STATE_CHANGED | HA_STATE_ROW_CHANGED; + pthread_mutex_lock(&share->intern_lock); + share->state.changed &= ~(STATE_CHANGED | STATE_CRASHED | + STATE_CRASHED_ON_REPAIR); + if (!(table->db_stat & HA_READ_ONLY)) + error= maria_update_state_info(¶m, file, UPDATE_TIME | UPDATE_OPEN_COUNT | + UPDATE_STAT); + pthread_mutex_unlock(&share->intern_lock); + info(HA_STATUS_NO_LOCK | HA_STATUS_TIME | HA_STATUS_VARIABLE | + HA_STATUS_CONST); + } + } + else if (!maria_is_crashed(file) && !thd->killed) + { + maria_mark_crashed(file); + file->update |= HA_STATE_CHANGED | HA_STATE_ROW_CHANGED; + } + + thd->proc_info= old_proc_info; + return error ? HA_ADMIN_CORRUPT : HA_ADMIN_OK; +} + + +/* + Analyze the key distribution in the table + As the table may be only locked for read, we have to take into account that + two threads may do an analyze at the same time! +*/ + +int ha_maria::analyze(THD *thd, HA_CHECK_OPT * check_opt) +{ + int error= 0; + HA_CHECK param; + MARIA_SHARE *share= file->s; + + mariachk_init(¶m); + param.thd= thd; + param.op_name= "analyze"; + param.db_name= table->s->db.str; + param.table_name= table->alias; + param.testflag= (T_FAST | T_CHECK | T_SILENT | T_STATISTICS | + T_DONT_CHECK_CHECKSUM); + param.using_global_keycache= 1; + param.stats_method= (enum_handler_stats_method) thd->variables. + maria_stats_method; + + if (!(share->state.changed & STATE_NOT_ANALYZED)) + return HA_ADMIN_ALREADY_DONE; + + error= maria_chk_key(¶m, file); + if (!error) + { + pthread_mutex_lock(&share->intern_lock); + error= maria_update_state_info(¶m, file, UPDATE_STAT); + pthread_mutex_unlock(&share->intern_lock); + } + else if (!maria_is_crashed(file) && !thd->killed) + maria_mark_crashed(file); + return error ? HA_ADMIN_CORRUPT : HA_ADMIN_OK; +} + + +int ha_maria::restore(THD * thd, HA_CHECK_OPT *check_opt) +{ + HA_CHECK_OPT tmp_check_opt; + char *backup_dir= thd->lex->backup_dir; + char src_path[FN_REFLEN], dst_path[FN_REFLEN]; + const char *table_name= table->s->table_name.str; + int error; + const char *errmsg; + DBUG_ENTER("restore"); + + if (fn_format_relative_to_data_home(src_path, table_name, backup_dir, + MARIA_NAME_DEXT)) + DBUG_RETURN(HA_ADMIN_INVALID); + + strxmov(dst_path, table->s->normalized_path.str, MARIA_NAME_DEXT, NullS); + if (my_copy(src_path, dst_path, MYF(MY_WME))) + { + error= HA_ADMIN_FAILED; + errmsg= "Failed in my_copy (Error %d)"; + goto err; + } + + tmp_check_opt.init(); + tmp_check_opt.flags |= T_VERY_SILENT | T_CALC_CHECKSUM | T_QUICK; + DBUG_RETURN(repair(thd, &tmp_check_opt)); + +err: + { + HA_CHECK param; + mariachk_init(¶m); + param.thd= thd; + param.op_name= "restore"; + param.db_name= table->s->db.str; + param.table_name= table->s->table_name.str; + param.testflag= 0; + _ma_check_print_error(¶m, errmsg, my_errno); + DBUG_RETURN(error); + } +} + + +int ha_maria::backup(THD * thd, HA_CHECK_OPT *check_opt) +{ + char *backup_dir= thd->lex->backup_dir; + char src_path[FN_REFLEN], dst_path[FN_REFLEN]; + const char *table_name= table->s->table_name.str; + int error; + const char *errmsg; + DBUG_ENTER("ha_maria::backup"); + + if (fn_format_relative_to_data_home(dst_path, table_name, backup_dir, + reg_ext)) + { + errmsg= "Failed in fn_format() for .frm file (errno: %d)"; + error= HA_ADMIN_INVALID; + goto err; + } + + strxmov(src_path, table->s->normalized_path.str, reg_ext, NullS); + if (my_copy(src_path, dst_path, + MYF(MY_WME | MY_HOLD_ORIGINAL_MODES | MY_DONT_OVERWRITE_FILE))) + { + error= HA_ADMIN_FAILED; + errmsg= "Failed copying .frm file (errno: %d)"; + goto err; + } + + /* Change extension */ + if (fn_format_relative_to_data_home(dst_path, table_name, backup_dir, + MARIA_NAME_DEXT)) + { + errmsg= "Failed in fn_format() for .MYD file (errno: %d)"; + error= HA_ADMIN_INVALID; + goto err; + } + + strxmov(src_path, table->s->normalized_path.str, MARIA_NAME_DEXT, NullS); + if (my_copy(src_path, dst_path, + MYF(MY_WME | MY_HOLD_ORIGINAL_MODES | MY_DONT_OVERWRITE_FILE))) + { + errmsg= "Failed copying .MYD file (errno: %d)"; + error= HA_ADMIN_FAILED; + goto err; + } + DBUG_RETURN(HA_ADMIN_OK); + +err: + { + HA_CHECK param; + mariachk_init(¶m); + param.thd= thd; + param.op_name= "backup"; + param.db_name= table->s->db.str; + param.table_name= table->s->table_name.str; + param.testflag= 0; + _ma_check_print_error(¶m, errmsg, my_errno); + DBUG_RETURN(error); + } +} + + +int ha_maria::repair(THD * thd, HA_CHECK_OPT *check_opt) +{ + int error; + HA_CHECK param; + ha_rows start_records; + + if (!file) + return HA_ADMIN_INTERNAL_ERROR; + + mariachk_init(¶m); + param.thd= thd; + param.op_name= "repair"; + param.testflag= ((check_opt->flags & ~(T_EXTEND)) | + T_SILENT | T_FORCE_CREATE | T_CALC_CHECKSUM | + (check_opt->flags & T_EXTEND ? T_REP : T_REP_BY_SORT)); + param.sort_buffer_length= check_opt->sort_buffer_size; + start_records= file->state->records; + while ((error= repair(thd, param, 0)) && param.retry_repair) + { + param.retry_repair= 0; + if (test_all_bits(param.testflag, + (uint) (T_RETRY_WITHOUT_QUICK | T_QUICK))) + { + param.testflag &= ~T_RETRY_WITHOUT_QUICK; + sql_print_information("Retrying repair of: '%s' without quick", + table->s->path); + continue; + } + param.testflag &= ~T_QUICK; + if ((param.testflag & T_REP_BY_SORT)) + { + param.testflag= (param.testflag & ~T_REP_BY_SORT) | T_REP; + sql_print_information("Retrying repair of: '%s' with keycache", + table->s->path); + continue; + } + break; + } + if (!error && start_records != file->state->records && + !(check_opt->flags & T_VERY_SILENT)) + { + char llbuff[22], llbuff2[22]; + sql_print_information("Found %s of %s rows when repairing '%s'", + llstr(file->state->records, llbuff), + llstr(start_records, llbuff2), table->s->path); + } + return error; +} + +int ha_maria::optimize(THD * thd, HA_CHECK_OPT *check_opt) +{ + int error; + if (!file) + return HA_ADMIN_INTERNAL_ERROR; + HA_CHECK param; + + mariachk_init(¶m); + param.thd= thd; + param.op_name= "optimize"; + param.testflag= (check_opt->flags | T_SILENT | T_FORCE_CREATE | + T_REP_BY_SORT | T_STATISTICS | T_SORT_INDEX); + param.sort_buffer_length= check_opt->sort_buffer_size; + if ((error= repair(thd, param, 1)) && param.retry_repair) + { + sql_print_warning("Warning: Optimize table got errno %d, retrying", + my_errno); + param.testflag &= ~T_REP_BY_SORT; + error= repair(thd, param, 1); + } + return error; +} + + +int ha_maria::repair(THD *thd, HA_CHECK ¶m, bool optimize) +{ + int error= 0; + uint local_testflag= param.testflag; + bool optimize_done= !optimize, statistics_done= 0; + const char *old_proc_info= thd->proc_info; + char fixed_name[FN_REFLEN]; + MARIA_SHARE *share= file->s; + ha_rows rows= file->state->records; + DBUG_ENTER("ha_maria::repair"); + + param.db_name= table->s->db.str; + param.table_name= table->alias; + param.tmpfile_createflag= O_RDWR | O_TRUNC; + param.using_global_keycache= 1; + param.thd= thd; + param.tmpdir= &mysql_tmpdir_list; + param.out_flag= 0; + strmov(fixed_name, file->filename); + + // Don't lock tables if we have used LOCK TABLE + if (!thd->locked_tables && + maria_lock_database(file, table->s->tmp_table ? F_EXTRA_LCK : F_WRLCK)) + { + _ma_check_print_error(¶m, ER(ER_CANT_LOCK), my_errno); + DBUG_RETURN(HA_ADMIN_FAILED); + } + + if (!optimize || + ((file->state->del || share->state.split != file->state->records) && + (!(param.testflag & T_QUICK) || + !(share->state.changed & STATE_NOT_OPTIMIZED_KEYS)))) + { + ulonglong key_map= ((local_testflag & T_CREATE_MISSING_KEYS) ? + maria_get_mask_all_keys_active(share->base.keys) : + share->state.key_map); + uint testflag= param.testflag; + if (maria_test_if_sort_rep(file, file->state->records, key_map, 0) && + (local_testflag & T_REP_BY_SORT)) + { + local_testflag |= T_STATISTICS; + param.testflag |= T_STATISTICS; // We get this for free + statistics_done= 1; + if (thd->variables.maria_repair_threads > 1) + { + char buf[40]; + /* TODO: respect maria_repair_threads variable */ + my_snprintf(buf, 40, "Repair with %d threads", my_count_bits(key_map)); + thd->proc_info= buf; + error= maria_repair_parallel(¶m, file, fixed_name, + param.testflag & T_QUICK); + thd->proc_info= "Repair done"; // to reset proc_info, as + // it was pointing to local buffer + } + else + { + thd->proc_info= "Repair by sorting"; + error= maria_repair_by_sort(¶m, file, fixed_name, + param.testflag & T_QUICK); + } + } + else + { + thd->proc_info= "Repair with keycache"; + param.testflag &= ~T_REP_BY_SORT; + error= maria_repair(¶m, file, fixed_name, param.testflag & T_QUICK); + } + param.testflag= testflag; + optimize_done= 1; + } + if (!error) + { + if ((local_testflag & T_SORT_INDEX) && + (share->state.changed & STATE_NOT_SORTED_PAGES)) + { + optimize_done= 1; + thd->proc_info= "Sorting index"; + error= maria_sort_index(¶m, file, fixed_name); + } + if (!statistics_done && (local_testflag & T_STATISTICS)) + { + if (share->state.changed & STATE_NOT_ANALYZED) + { + optimize_done= 1; + thd->proc_info= "Analyzing"; + error= maria_chk_key(¶m, file); + } + else + local_testflag &= ~T_STATISTICS; // Don't update statistics + } + } + thd->proc_info= "Saving state"; + if (!error) + { + if ((share->state.changed & STATE_CHANGED) || maria_is_crashed(file)) + { + share->state.changed &= ~(STATE_CHANGED | STATE_CRASHED | + STATE_CRASHED_ON_REPAIR); + file->update |= HA_STATE_CHANGED | HA_STATE_ROW_CHANGED; + } + /* + the following 'if', thought conceptually wrong, + is a useful optimization nevertheless. + */ + if (file->state != &file->s->state.state) + file->s->state.state= *file->state; + if (file->s->base.auto_key) + _ma_update_auto_increment_key(¶m, file, 1); + if (optimize_done) + error= maria_update_state_info(¶m, file, + UPDATE_TIME | UPDATE_OPEN_COUNT | + (local_testflag & + T_STATISTICS ? UPDATE_STAT : 0)); + info(HA_STATUS_NO_LOCK | HA_STATUS_TIME | HA_STATUS_VARIABLE | + HA_STATUS_CONST); + if (rows != file->state->records && !(param.testflag & T_VERY_SILENT)) + { + char llbuff[22], llbuff2[22]; + _ma_check_print_warning(¶m, "Number of rows changed from %s to %s", + llstr(rows, llbuff), + llstr(file->state->records, llbuff2)); + } + } + else + { + maria_mark_crashed_on_repair(file); + file->update |= HA_STATE_CHANGED | HA_STATE_ROW_CHANGED; + maria_update_state_info(¶m, file, 0); + } + thd->proc_info= old_proc_info; + if (!thd->locked_tables) + maria_lock_database(file, F_UNLCK); + DBUG_RETURN(error ? HA_ADMIN_FAILED : + !optimize_done ? HA_ADMIN_ALREADY_DONE : HA_ADMIN_OK); +} + + +/* + Assign table indexes to a specific key cache. +*/ + +int ha_maria::assign_to_keycache(THD * thd, HA_CHECK_OPT *check_opt) +{ + KEY_CACHE *new_key_cache= check_opt->key_cache; + const char *errmsg= 0; + int error= HA_ADMIN_OK; + ulonglong map= ~(ulonglong) 0; + TABLE_LIST *table_list= table->pos_in_table_list; + DBUG_ENTER("ha_maria::assign_to_keycache"); + + /* Check validity of the index references */ + if (table_list->use_index) + { + /* We only come here when the user did specify an index map */ + key_map kmap; + if (get_key_map_from_key_list(&kmap, table, table_list->use_index)) + { + errmsg= thd->net.last_error; + error= HA_ADMIN_FAILED; + goto err; + } + map= kmap.to_ulonglong(); + } + + if ((error= maria_assign_to_key_cache(file, map, new_key_cache))) + { + char buf[STRING_BUFFER_USUAL_SIZE]; + my_snprintf(buf, sizeof(buf), + "Failed to flush to index file (errno: %d)", error); + errmsg= buf; + error= HA_ADMIN_CORRUPT; + } + +err: + if (error != HA_ADMIN_OK) + { + /* Send error to user */ + HA_CHECK param; + mariachk_init(¶m); + param.thd= thd; + param.op_name= "assign_to_keycache"; + param.db_name= table->s->db.str; + param.table_name= table->s->table_name.str; + param.testflag= 0; + _ma_check_print_error(¶m, errmsg); + } + DBUG_RETURN(error); +} + + +/* + Preload pages of the index file for a table into the key cache. +*/ + +int ha_maria::preload_keys(THD * thd, HA_CHECK_OPT *check_opt) +{ + int error; + const char *errmsg; + ulonglong map= ~(ulonglong) 0; + TABLE_LIST *table_list= table->pos_in_table_list; + my_bool ignore_leaves= table_list->ignore_leaves; + + DBUG_ENTER("ha_maria::preload_keys"); + + /* Check validity of the index references */ + if (table_list->use_index) + { + key_map kmap; + get_key_map_from_key_list(&kmap, table, table_list->use_index); + if (kmap.is_set_all()) + { + errmsg= thd->net.last_error; + error= HA_ADMIN_FAILED; + goto err; + } + if (!kmap.is_clear_all()) + map= kmap.to_ulonglong(); + } + + maria_extra(file, HA_EXTRA_PRELOAD_BUFFER_SIZE, + (void*) &thd->variables.preload_buff_size); + + if ((error= maria_preload(file, map, ignore_leaves))) + { + switch (error) { + case HA_ERR_NON_UNIQUE_BLOCK_SIZE: + errmsg= "Indexes use different block sizes"; + break; + case HA_ERR_OUT_OF_MEM: + errmsg= "Failed to allocate buffer"; + break; + default: + char buf[ERRMSGSIZE + 20]; + my_snprintf(buf, ERRMSGSIZE, + "Failed to read from index file (errno: %d)", my_errno); + errmsg= buf; + } + error= HA_ADMIN_FAILED; + goto err; + } + + DBUG_RETURN(HA_ADMIN_OK); + +err: + { + HA_CHECK param; + mariachk_init(¶m); + param.thd= thd; + param.op_name= "preload_keys"; + param.db_name= table->s->db.str; + param.table_name= table->s->table_name.str; + param.testflag= 0; + _ma_check_print_error(¶m, errmsg); + DBUG_RETURN(error); + } +} + + +/* + Disable indexes, making it persistent if requested. + + SYNOPSIS + disable_indexes() + mode mode of operation: + HA_KEY_SWITCH_NONUNIQ disable all non-unique keys + HA_KEY_SWITCH_ALL disable all keys + HA_KEY_SWITCH_NONUNIQ_SAVE dis. non-uni. and make persistent + HA_KEY_SWITCH_ALL_SAVE dis. all keys and make persistent + + IMPLEMENTATION + HA_KEY_SWITCH_NONUNIQ is not implemented. + HA_KEY_SWITCH_ALL_SAVE is not implemented. + + RETURN + 0 ok + HA_ERR_WRONG_COMMAND mode not implemented. +*/ + +int ha_maria::disable_indexes(uint mode) +{ + int error; + + if (mode == HA_KEY_SWITCH_ALL) + { + /* call a storage engine function to switch the key map */ + error= maria_disable_indexes(file); + } + else if (mode == HA_KEY_SWITCH_NONUNIQ_SAVE) + { + maria_extra(file, HA_EXTRA_NO_KEYS, 0); + info(HA_STATUS_CONST); // Read new key info + error= 0; + } + else + { + /* mode not implemented */ + error= HA_ERR_WRONG_COMMAND; + } + return error; +} + + +/* + Enable indexes, making it persistent if requested. + + SYNOPSIS + enable_indexes() + mode mode of operation: + HA_KEY_SWITCH_NONUNIQ enable all non-unique keys + HA_KEY_SWITCH_ALL enable all keys + HA_KEY_SWITCH_NONUNIQ_SAVE en. non-uni. and make persistent + HA_KEY_SWITCH_ALL_SAVE en. all keys and make persistent + + DESCRIPTION + Enable indexes, which might have been disabled by disable_index() before. + The modes without _SAVE work only if both data and indexes are empty, + since the MARIA repair would enable them persistently. + To be sure in these cases, call handler::delete_all_rows() before. + + IMPLEMENTATION + HA_KEY_SWITCH_NONUNIQ is not implemented. + HA_KEY_SWITCH_ALL_SAVE is not implemented. + + RETURN + 0 ok + !=0 Error, among others: + HA_ERR_CRASHED data or index is non-empty. Delete all rows and retry. + HA_ERR_WRONG_COMMAND mode not implemented. +*/ + +int ha_maria::enable_indexes(uint mode) +{ + int error; + + if (maria_is_all_keys_active(file->s->state.key_map, file->s->base.keys)) + { + /* All indexes are enabled already. */ + return 0; + } + + if (mode == HA_KEY_SWITCH_ALL) + { + error= maria_enable_indexes(file); + /* + Do not try to repair on error, + as this could make the enabled state persistent, + but mode==HA_KEY_SWITCH_ALL forbids it. + */ + } + else if (mode == HA_KEY_SWITCH_NONUNIQ_SAVE) + { + THD *thd= current_thd; + HA_CHECK param; + const char *save_proc_info= thd->proc_info; + thd->proc_info= "Creating index"; + mariachk_init(¶m); + param.op_name= "recreating_index"; + param.testflag= (T_SILENT | T_REP_BY_SORT | T_QUICK | + T_CREATE_MISSING_KEYS); + param.myf_rw &= ~MY_WAIT_IF_FULL; + param.sort_buffer_length= thd->variables.maria_sort_buff_size; + param.stats_method= (enum_handler_stats_method) thd->variables. + maria_stats_method; + param.tmpdir= &mysql_tmpdir_list; + if ((error= (repair(thd, param, 0) != HA_ADMIN_OK)) && param.retry_repair) + { + sql_print_warning("Warning: Enabling keys got errno %d, retrying", + my_errno); + /* Repairing by sort failed. Now try standard repair method. */ + param.testflag &= ~(T_REP_BY_SORT | T_QUICK); + error= (repair(thd, param, 0) != HA_ADMIN_OK); + /* + If the standard repair succeeded, clear all error messages which + might have been set by the first repair. They can still be seen + with SHOW WARNINGS then. + */ + if (!error) + thd->clear_error(); + } + info(HA_STATUS_CONST); + thd->proc_info= save_proc_info; + } + else + { + /* mode not implemented */ + error= HA_ERR_WRONG_COMMAND; + } + return error; +} + + +/* + Test if indexes are disabled. + + + SYNOPSIS + indexes_are_disabled() + no parameters + + + RETURN + 0 indexes are not disabled + 1 all indexes are disabled + [2 non-unique indexes are disabled - NOT YET IMPLEMENTED] +*/ + +int ha_maria::indexes_are_disabled(void) +{ + return maria_indexes_are_disabled(file); +} + + +/* + prepare for a many-rows insert operation + e.g. - disable indexes (if they can be recreated fast) or + activate special bulk-insert optimizations + + SYNOPSIS + start_bulk_insert(rows) + rows Rows to be inserted + 0 if we don't know + + NOTICE + Do not forget to call end_bulk_insert() later! +*/ + +void ha_maria::start_bulk_insert(ha_rows rows) +{ + DBUG_ENTER("ha_maria::start_bulk_insert"); + THD *thd= current_thd; + ulong size= min(thd->variables.read_buff_size, + table->s->avg_row_length * rows); + DBUG_PRINT("info", ("start_bulk_insert: rows %lu size %lu", + (ulong) rows, size)); + + /* don't enable row cache if too few rows */ + if (!rows || (rows > MARIA_MIN_ROWS_TO_USE_WRITE_CACHE)) + maria_extra(file, HA_EXTRA_WRITE_CACHE, (void*) &size); + + can_enable_indexes= maria_is_all_keys_active(file->s->state.key_map, + file->s->base.keys); + + if (!(specialflag & SPECIAL_SAFE_MODE)) + { + /* + Only disable old index if the table was empty and we are inserting + a lot of rows. + We should not do this for only a few rows as this is slower and + we don't want to update the key statistics based of only a few rows. + */ + if (file->state->records == 0 && can_enable_indexes && + (!rows || rows >= MARIA_MIN_ROWS_TO_DISABLE_INDEXES)) + maria_disable_non_unique_index(file, rows); + else + if (!file->bulk_insert && + (!rows || rows >= MARIA_MIN_ROWS_TO_USE_BULK_INSERT)) + { + maria_init_bulk_insert(file, thd->variables.bulk_insert_buff_size, rows); + } + } + DBUG_VOID_RETURN; +} + + +/* + end special bulk-insert optimizations, + which have been activated by start_bulk_insert(). + + SYNOPSIS + end_bulk_insert() + no arguments + + RETURN + 0 OK + != 0 Error +*/ + +int ha_maria::end_bulk_insert() +{ + maria_end_bulk_insert(file); + int err= maria_extra(file, HA_EXTRA_NO_CACHE, 0); + return err ? err : can_enable_indexes ? + enable_indexes(HA_KEY_SWITCH_NONUNIQ_SAVE) : 0; +} + + +bool ha_maria::check_and_repair(THD *thd) +{ + int error= 0; + int marked_crashed; + char *old_query; + uint old_query_length; + HA_CHECK_OPT check_opt; + DBUG_ENTER("ha_maria::check_and_repair"); + + check_opt.init(); + check_opt.flags= T_MEDIUM | T_AUTO_REPAIR; + // Don't use quick if deleted rows + if (!file->state->del && (maria_recover_options & HA_RECOVER_QUICK)) + check_opt.flags |= T_QUICK; + sql_print_warning("Checking table: '%s'", table->s->path); + + old_query= thd->query; + old_query_length= thd->query_length; + pthread_mutex_lock(&LOCK_thread_count); + thd->query= table->s->table_name.str; + thd->query_length= table->s->table_name.length; + pthread_mutex_unlock(&LOCK_thread_count); + + if ((marked_crashed= maria_is_crashed(file)) || check(thd, &check_opt)) + { + sql_print_warning("Recovering table: '%s'", table->s->path); + check_opt.flags= + ((maria_recover_options & HA_RECOVER_BACKUP ? T_BACKUP_DATA : 0) | + (marked_crashed ? 0 : T_QUICK) | + (maria_recover_options & HA_RECOVER_FORCE ? 0 : T_SAFE_REPAIR) | + T_AUTO_REPAIR); + if (repair(thd, &check_opt)) + error= 1; + } + pthread_mutex_lock(&LOCK_thread_count); + thd->query= old_query; + thd->query_length= old_query_length; + pthread_mutex_unlock(&LOCK_thread_count); + DBUG_RETURN(error); +} + + +bool ha_maria::is_crashed() const +{ + return (file->s->state.changed & STATE_CRASHED || + (my_disable_locking && file->s->state.open_count)); +} + + +int ha_maria::update_row(const byte * old_data, byte * new_data) +{ + statistic_increment(table->in_use->status_var.ha_update_count, &LOCK_status); + if (table->timestamp_field_type & TIMESTAMP_AUTO_SET_ON_UPDATE) + table->timestamp_field->set_time(); + return maria_update(file, old_data, new_data); +} + + +int ha_maria::delete_row(const byte * buf) +{ + statistic_increment(table->in_use->status_var.ha_delete_count, &LOCK_status); + return maria_delete(file, buf); +} + + +int ha_maria::index_read(byte * buf, const byte * key, + uint key_len, enum ha_rkey_function find_flag) +{ + DBUG_ASSERT(inited == INDEX); + statistic_increment(table->in_use->status_var.ha_read_key_count, + &LOCK_status); + int error= maria_rkey(file, buf, active_index, key, key_len, find_flag); + table->status= error ? STATUS_NOT_FOUND : 0; + return error; +} + + +int ha_maria::index_read_idx(byte * buf, uint index, const byte * key, + uint key_len, enum ha_rkey_function find_flag) +{ + statistic_increment(table->in_use->status_var.ha_read_key_count, + &LOCK_status); + int error= maria_rkey(file, buf, index, key, key_len, find_flag); + table->status= error ? STATUS_NOT_FOUND : 0; + return error; +} + + +int ha_maria::index_read_last(byte * buf, const byte * key, uint key_len) +{ + DBUG_ASSERT(inited == INDEX); + statistic_increment(table->in_use->status_var.ha_read_key_count, + &LOCK_status); + int error= maria_rkey(file, buf, active_index, key, key_len, + HA_READ_PREFIX_LAST); + table->status= error ? STATUS_NOT_FOUND : 0; + return error; +} + + +int ha_maria::index_next(byte * buf) +{ + DBUG_ASSERT(inited == INDEX); + statistic_increment(table->in_use->status_var.ha_read_next_count, + &LOCK_status); + int error= maria_rnext(file, buf, active_index); + table->status= error ? STATUS_NOT_FOUND : 0; + return error; +} + + +int ha_maria::index_prev(byte * buf) +{ + DBUG_ASSERT(inited == INDEX); + statistic_increment(table->in_use->status_var.ha_read_prev_count, + &LOCK_status); + int error= maria_rprev(file, buf, active_index); + table->status= error ? STATUS_NOT_FOUND : 0; + return error; +} + + +int ha_maria::index_first(byte * buf) +{ + DBUG_ASSERT(inited == INDEX); + statistic_increment(table->in_use->status_var.ha_read_first_count, + &LOCK_status); + int error= maria_rfirst(file, buf, active_index); + table->status= error ? STATUS_NOT_FOUND : 0; + return error; +} + + +int ha_maria::index_last(byte * buf) +{ + DBUG_ASSERT(inited == INDEX); + statistic_increment(table->in_use->status_var.ha_read_last_count, + &LOCK_status); + int error= maria_rlast(file, buf, active_index); + table->status= error ? STATUS_NOT_FOUND : 0; + return error; +} + + +int ha_maria::index_next_same(byte * buf, + const byte *key __attribute__ ((unused)), + uint length __attribute__ ((unused))) +{ + DBUG_ASSERT(inited == INDEX); + statistic_increment(table->in_use->status_var.ha_read_next_count, + &LOCK_status); + int error= maria_rnext_same(file, buf); + table->status= error ? STATUS_NOT_FOUND : 0; + return error; +} + + +int ha_maria::rnd_init(bool scan) +{ + if (scan) + return maria_scan_init(file); + return maria_extra(file, HA_EXTRA_RESET, 0); +} + + +int ha_maria::rnd_next(byte *buf) +{ + statistic_increment(table->in_use->status_var.ha_read_rnd_next_count, + &LOCK_status); + int error= maria_scan(file, buf); + table->status= error ? STATUS_NOT_FOUND : 0; + return error; +} + + +int ha_maria::restart_rnd_next(byte *buf, byte *pos) +{ + return rnd_pos(buf, pos); +} + + +int ha_maria::rnd_pos(byte * buf, byte *pos) +{ + statistic_increment(table->in_use->status_var.ha_read_rnd_count, + &LOCK_status); + int error= maria_rrnd(file, buf, my_get_ptr(pos, ref_length)); + table->status= error ? STATUS_NOT_FOUND : 0; + return error; +} + + +void ha_maria::position(const byte * record) +{ + my_off_t position= maria_position(file); + my_store_ptr(ref, ref_length, position); +} + + +void ha_maria::info(uint flag) +{ + MARIA_INFO info; + char name_buff[FN_REFLEN]; + + (void) maria_status(file, &info, flag); + if (flag & HA_STATUS_VARIABLE) + { + records= info.records; + deleted= info.deleted; + data_file_length= info.data_file_length; + index_file_length= info.index_file_length; + delete_length= info.delete_length; + check_time= info.check_time; + mean_rec_length= info.mean_reclength; + } + if (flag & HA_STATUS_CONST) + { + TABLE_SHARE *share= table->s; + max_data_file_length= info.max_data_file_length; + max_index_file_length= info.max_index_file_length; + create_time= info.create_time; + sortkey= info.sortkey; + ref_length= info.reflength; + share->db_options_in_use= info.options; + block_size= maria_block_size; + + /* Update share */ + if (share->tmp_table == NO_TMP_TABLE) + pthread_mutex_lock(&share->mutex); + share->keys_in_use.set_prefix(share->keys); + share->keys_in_use.intersect_extended(info.key_map); + share->keys_for_keyread.intersect(share->keys_in_use); + share->db_record_offset= info.record_offset; + if (share->key_parts) + memcpy((char*) table->key_info[0].rec_per_key, + (char*) info.rec_per_key, + sizeof(table->key_info[0].rec_per_key) * share->key_parts); + if (share->tmp_table == NO_TMP_TABLE) + pthread_mutex_unlock(&share->mutex); + + /* + Set data_file_name and index_file_name to point at the symlink value + if table is symlinked (Ie; Real name is not same as generated name) + */ + data_file_name= index_file_name= 0; + fn_format(name_buff, file->filename, "", MARIA_NAME_DEXT, MY_APPEND_EXT); + if (strcmp(name_buff, info.data_file_name)) + data_file_name= info.data_file_name; + fn_format(name_buff, file->filename, "", MARIA_NAME_IEXT, MY_APPEND_EXT); + if (strcmp(name_buff, info.index_file_name)) + index_file_name= info.index_file_name; + } + if (flag & HA_STATUS_ERRKEY) + { + errkey= info.errkey; + my_store_ptr(dupp_ref, ref_length, info.dupp_key_pos); + } + /* Faster to always update, than to do it based on flag */ + update_time= info.update_time; + auto_increment_value= info.auto_increment; +} + + +int ha_maria::extra(enum ha_extra_function operation) +{ + if ((specialflag & SPECIAL_SAFE_MODE) && operation == HA_EXTRA_KEYREAD) + return 0; + return maria_extra(file, operation, 0); +} + + +/* To be used with WRITE_CACHE and EXTRA_CACHE */ + +int ha_maria::extra_opt(enum ha_extra_function operation, ulong cache_size) +{ + if ((specialflag & SPECIAL_SAFE_MODE) && operation == HA_EXTRA_WRITE_CACHE) + return 0; + return maria_extra(file, operation, (void*) &cache_size); +} + + +int ha_maria::delete_all_rows() +{ + return maria_delete_all_rows(file); +} + + +int ha_maria::delete_table(const char *name) +{ + return maria_delete_table(name); +} + + +int ha_maria::external_lock(THD *thd, int lock_type) +{ + return maria_lock_database(file, !table->s->tmp_table ? + lock_type : ((lock_type == F_UNLCK) ? + F_UNLCK : F_EXTRA_LCK)); +} + + +THR_LOCK_DATA **ha_maria::store_lock(THD *thd, + THR_LOCK_DATA **to, + enum thr_lock_type lock_type) +{ + if (lock_type != TL_IGNORE && file->lock.type == TL_UNLOCK) + file->lock.type= lock_type; + *to++= &file->lock; + return to; +} + + +void ha_maria::update_create_info(HA_CREATE_INFO *create_info) +{ + ha_maria::info(HA_STATUS_AUTO | HA_STATUS_CONST); + if (!(create_info->used_fields & HA_CREATE_USED_AUTO)) + { + create_info->auto_increment_value= auto_increment_value; + } + create_info->data_file_name= data_file_name; + create_info->index_file_name= index_file_name; +} + + +int ha_maria::create(const char *name, register TABLE *table_arg, + HA_CREATE_INFO *info) +{ + int error; + uint i, j, recpos, minpos, fieldpos, temp_length, length, create_flags= 0; + bool found_real_auto_increment= 0; + enum ha_base_keytype type; + char buff[FN_REFLEN]; + KEY *pos; + MARIA_KEYDEF *keydef; + MARIA_COLUMNDEF *recinfo, *recinfo_pos; + HA_KEYSEG *keyseg; + TABLE_SHARE *share= table_arg->s; + uint options= share->db_options_in_use; + DBUG_ENTER("ha_maria::create"); + + type= HA_KEYTYPE_BINARY; // Keep compiler happy + if (!(my_multi_malloc(MYF(MY_WME), + &recinfo, (share->fields * 2 + 2) * + sizeof(MARIA_COLUMNDEF), + &keydef, share->keys * sizeof(MARIA_KEYDEF), + &keyseg, + ((share->key_parts + share->keys) * + sizeof(HA_KEYSEG)), NullS))) + DBUG_RETURN(HA_ERR_OUT_OF_MEM); + + pos= table_arg->key_info; + for (i= 0; i < share->keys; i++, pos++) + { + if (pos->flags & HA_USES_PARSER) + create_flags |= HA_CREATE_RELIES_ON_SQL_LAYER; + keydef[i].flag= (pos->flags & (HA_NOSAME | HA_FULLTEXT | HA_SPATIAL)); + keydef[i].key_alg= pos->algorithm == HA_KEY_ALG_UNDEF ? + (pos->flags & HA_SPATIAL ? HA_KEY_ALG_RTREE : HA_KEY_ALG_BTREE) : + pos->algorithm; + keydef[i].seg= keyseg; + keydef[i].keysegs= pos->key_parts; + for (j= 0; j < pos->key_parts; j++) + { + Field *field= pos->key_part[j].field; + type= field->key_type(); + keydef[i].seg[j].flag= pos->key_part[j].key_part_flag; + + if (options & HA_OPTION_PACK_KEYS || + (pos->flags & (HA_PACK_KEY | HA_BINARY_PACK_KEY | + HA_SPACE_PACK_USED))) + { + if (pos->key_part[j].length > 8 && + (type == HA_KEYTYPE_TEXT || + type == HA_KEYTYPE_NUM || + (type == HA_KEYTYPE_BINARY && !field->zero_pack()))) + { + /* No blobs here */ + if (j == 0) + keydef[i].flag |= HA_PACK_KEY; + if (!(field->flags & ZEROFILL_FLAG) && + (field->type() == MYSQL_TYPE_STRING || + field->type() == MYSQL_TYPE_VAR_STRING || + ((int) (pos->key_part[j].length - field->decimals())) >= 4)) + keydef[i].seg[j].flag |= HA_SPACE_PACK; + } + else if (j == 0 && (!(pos->flags & HA_NOSAME) || pos->key_length > 16)) + keydef[i].flag |= HA_BINARY_PACK_KEY; + } + keydef[i].seg[j].type= (int) type; + keydef[i].seg[j].start= pos->key_part[j].offset; + keydef[i].seg[j].length= pos->key_part[j].length; + keydef[i].seg[j].bit_start= keydef[i].seg[j].bit_end= + keydef[i].seg[j].bit_length= 0; + keydef[i].seg[j].bit_pos= 0; + keydef[i].seg[j].language= field->charset()->number; + + if (field->null_ptr) + { + keydef[i].seg[j].null_bit= field->null_bit; + keydef[i].seg[j].null_pos= (uint) (field->null_ptr - + (uchar *) table_arg->record[0]); + } + else + { + keydef[i].seg[j].null_bit= 0; + keydef[i].seg[j].null_pos= 0; + } + if (field->type() == FIELD_TYPE_BLOB || + field->type() == FIELD_TYPE_GEOMETRY) + { + keydef[i].seg[j].flag |= HA_BLOB_PART; + /* save number of bytes used to pack length */ + keydef[i].seg[j].bit_start= (uint) (field->pack_length() - + share->blob_ptr_size); + } + else if (field->type() == FIELD_TYPE_BIT) + { + keydef[i].seg[j].bit_length= ((Field_bit *) field)->bit_len; + keydef[i].seg[j].bit_start= ((Field_bit *) field)->bit_ofs; + keydef[i].seg[j].bit_pos= (uint) (((Field_bit *) field)->bit_ptr - + (uchar *) table_arg->record[0]); + } + } + keyseg += pos->key_parts; + } + + if (table_arg->found_next_number_field) + { + keydef[share->next_number_index].flag |= HA_AUTO_KEY; + found_real_auto_increment= share->next_number_key_offset == 0; + } + + recpos= 0; + recinfo_pos= recinfo; + while (recpos < (uint) share->reclength) + { + Field **field, *found= 0; + minpos= share->reclength; + length= 0; + + for (field= table_arg->field; *field; field++) + { + if ((fieldpos= (*field)->offset()) >= recpos && fieldpos <= minpos) + { + /* skip null fields */ + if (!(temp_length= (*field)->pack_length_in_rec())) + continue; /* Skip null-fields */ + if (!found || fieldpos < minpos || + (fieldpos == minpos && temp_length < length)) + { + minpos= fieldpos; + found= *field; + length= temp_length; + } + } + } + DBUG_PRINT("loop", ("found: 0x%lx recpos: %d minpos: %d length: %d", + found, recpos, minpos, length)); + if (recpos != minpos) + { // Reserved space (Null bits?) + bzero((char*) recinfo_pos, sizeof(*recinfo_pos)); + recinfo_pos->type= (int) FIELD_NORMAL; + recinfo_pos++->length= (uint16) (minpos - recpos); + } + if (!found) + break; + + if (found->flags & BLOB_FLAG) + recinfo_pos->type= (int) FIELD_BLOB; + else if (found->type() == MYSQL_TYPE_VARCHAR) + recinfo_pos->type= FIELD_VARCHAR; + else if (!(options & HA_OPTION_PACK_RECORD)) + recinfo_pos->type= (int) FIELD_NORMAL; + else if (found->zero_pack()) + recinfo_pos->type= (int) FIELD_SKIP_ZERO; + else + recinfo_pos->type= (int) ((length <= 3 || + (found->flags & ZEROFILL_FLAG)) ? + FIELD_NORMAL : + found->type() == MYSQL_TYPE_STRING || + found->type() == MYSQL_TYPE_VAR_STRING ? + FIELD_SKIP_ENDSPACE : FIELD_SKIP_PRESPACE); + if (found->null_ptr) + { + recinfo_pos->null_bit= found->null_bit; + recinfo_pos->null_pos= (uint) (found->null_ptr - + (uchar *) table_arg->record[0]); + } + else + { + recinfo_pos->null_bit= 0; + recinfo_pos->null_pos= 0; + } + (recinfo_pos++)->length= (uint16) length; + recpos= minpos + length; + DBUG_PRINT("loop", ("length: %d type: %d", + recinfo_pos[-1].length, recinfo_pos[-1].type)); + + } + MARIA_CREATE_INFO create_info; + bzero((char*) &create_info, sizeof(create_info)); + create_info.max_rows= share->max_rows; + create_info.reloc_rows= share->min_rows; + create_info.with_auto_increment= found_real_auto_increment; + create_info.auto_increment= (info->auto_increment_value ? + info->auto_increment_value - 1 : (ulonglong) 0); + create_info.data_file_length= ((ulonglong) share->max_rows * + share->avg_row_length); + create_info.data_file_name= info->data_file_name; + create_info.index_file_name= info->index_file_name; + + if (info->options & HA_LEX_CREATE_TMP_TABLE) + create_flags |= HA_CREATE_TMP_TABLE; + if (options & HA_OPTION_PACK_RECORD) + create_flags |= HA_PACK_RECORD; + if (options & HA_OPTION_CHECKSUM) + create_flags |= HA_CREATE_CHECKSUM; + if (options & HA_OPTION_DELAY_KEY_WRITE) + create_flags |= HA_CREATE_DELAY_KEY_WRITE; + + /* TODO: Check that the following fn_format is really needed */ + error= + maria_create(fn_format + (buff, name, "", "", MY_UNPACK_FILENAME | MY_APPEND_EXT), + share->keys, keydef, (uint) (recinfo_pos - recinfo), recinfo, + 0, (MARIA_UNIQUEDEF *) 0, &create_info, create_flags); + + my_free((gptr) recinfo, MYF(0)); + DBUG_RETURN(error); +} + + +int ha_maria::rename_table(const char *from, const char *to) +{ + return maria_rename(from, to); +} + + +ulonglong ha_maria::get_auto_increment() +{ + ulonglong nr; + int error; + byte key[HA_MAX_KEY_LENGTH]; + + if (!table->s->next_number_key_offset) + { // Autoincrement at key-start + ha_maria::info(HA_STATUS_AUTO); + return auto_increment_value; + } + + /* it's safe to call the following if bulk_insert isn't on */ + maria_flush_bulk_insert(file, table->s->next_number_index); + + (void) extra(HA_EXTRA_KEYREAD); + key_copy(key, table->record[0], + table->key_info + table->s->next_number_index, + table->s->next_number_key_offset); + error= maria_rkey(file, table->record[1], (int) table->s->next_number_index, + key, table->s->next_number_key_offset, HA_READ_PREFIX_LAST); + if (error) + nr= 1; + else + { + /* Get data from record[1] */ + nr= ((ulonglong) table->next_number_field-> + val_int_offset(table->s->rec_buff_length) + 1); + } + extra(HA_EXTRA_NO_KEYREAD); + return nr; +} + + +/* + Find out how many rows there is in the given range + + SYNOPSIS + records_in_range() + inx Index to use + min_key Start of range. Null pointer if from first key + max_key End of range. Null pointer if to last key + + NOTES + min_key.flag can have one of the following values: + HA_READ_KEY_EXACT Include the key in the range + HA_READ_AFTER_KEY Don't include key in range + + max_key.flag can have one of the following values: + HA_READ_BEFORE_KEY Don't include key in range + HA_READ_AFTER_KEY Include all 'end_key' values in the range + + RETURN + HA_POS_ERROR Something is wrong with the index tree. + 0 There is no matching keys in the given range + number > 0 There is approximately 'number' matching rows in + the range. +*/ + +ha_rows ha_maria::records_in_range(uint inx, key_range *min_key, + key_range *max_key) +{ + return (ha_rows) maria_records_in_range(file, (int) inx, min_key, max_key); +} + + +int ha_maria::ft_read(byte * buf) +{ + int error; + + if (!ft_handler) + return -1; + + thread_safe_increment(table->in_use->status_var.ha_read_next_count, + &LOCK_status); // why ? + + error= ft_handler->please->read_next(ft_handler, (char*) buf); + + table->status= error ? STATUS_NOT_FOUND : 0; + return error; +} + + +uint ha_maria::checksum() const +{ + return (uint) file->state->checksum; +} + + +bool ha_maria::check_if_incompatible_data(HA_CREATE_INFO *info, + uint table_changes) +{ + uint options= table->s->db_options_in_use; + + if (info->auto_increment_value != auto_increment_value || + info->data_file_name != data_file_name || + info->index_file_name != index_file_name || table_changes == IS_EQUAL_NO) + return COMPATIBLE_DATA_NO; + + if ((options & (HA_OPTION_PACK_RECORD | HA_OPTION_CHECKSUM | + HA_OPTION_DELAY_KEY_WRITE)) != + (info->table_options & (HA_OPTION_PACK_RECORD | HA_OPTION_CHECKSUM | + HA_OPTION_DELAY_KEY_WRITE))) + return COMPATIBLE_DATA_NO; + return COMPATIBLE_DATA_YES; +} diff --git a/sql/ha_maria.h b/sql/ha_maria.h new file mode 100644 index 00000000000..5387820edc6 --- /dev/null +++ b/sql/ha_maria.h @@ -0,0 +1,141 @@ +/* Copyright (C) 2006,2004 MySQL AB & MySQL Finland AB & TCX DataKonsult AB + + 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; either version 2 of the License, or + (at your option) any later version. + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ + + +#ifdef USE_PRAGMA_INTERFACE +#pragma interface /* gcc class implementation */ +#endif + +/* class for the the maria handler */ + +#include + +#define HA_RECOVER_NONE 0 /* No automatic recover */ +#define HA_RECOVER_DEFAULT 1 /* Automatic recover active */ +#define HA_RECOVER_BACKUP 2 /* Make a backupfile on recover */ +#define HA_RECOVER_FORCE 4 /* Recover even if we loose rows */ +#define HA_RECOVER_QUICK 8 /* Don't check rows in data file */ + +extern ulong maria_sort_buffer_size; +extern TYPELIB maria_recover_typelib; +extern ulong maria_recover_options; + +class ha_maria :public handler +{ + MARIA_HA *file; + ulong int_table_flags; + char *data_file_name, *index_file_name; + bool can_enable_indexes; + int repair(THD * thd, HA_CHECK ¶m, bool optimize); + +public: + ha_maria(TABLE_SHARE * table_arg); + ~ha_maria() + {} + const char *table_type() const + { return "MARIA"; } + const char *index_type(uint key_number); + const char **bas_ext() const; + ulong table_flags() const + { return int_table_flags; } + ulong index_flags(uint inx, uint part, bool all_parts) const + { + return ((table_share->key_info[inx].algorithm == HA_KEY_ALG_FULLTEXT) ? + 0 : HA_READ_NEXT | HA_READ_PREV | HA_READ_RANGE | + HA_READ_ORDER | HA_KEYREAD_ONLY); + } + uint max_supported_keys() const + { return MARIA_MAX_KEY; } + uint max_supported_key_length() const + { return HA_MAX_KEY_LENGTH; } + uint max_supported_key_part_length() const + { return HA_MAX_KEY_LENGTH; } + uint checksum() const; + + virtual bool check_if_locking_is_allowed(uint sql_command, + ulong type, TABLE * table, + uint count, + bool called_by_logger_thread); + int open(const char *name, int mode, uint test_if_locked); + int close(void); + int write_row(byte * buf); + int update_row(const byte * old_data, byte * new_data); + int delete_row(const byte * buf); + int index_read(byte * buf, const byte * key, + uint key_len, enum ha_rkey_function find_flag); + int index_read_idx(byte * buf, uint idx, const byte * key, + uint key_len, enum ha_rkey_function find_flag); + int index_read_last(byte * buf, const byte * key, uint key_len); + int index_next(byte * buf); + int index_prev(byte * buf); + int index_first(byte * buf); + int index_last(byte * buf); + int index_next_same(byte * buf, const byte * key, uint keylen); + int ft_init() + { + if (!ft_handler) + return 1; + ft_handler->please->reinit_search(ft_handler); + return 0; + } + FT_INFO *ft_init_ext(uint flags, uint inx, String * key) + { + return maria_ft_init_search(flags, file, inx, + (byte *) key->ptr(), key->length(), + key->charset(), table->record[0]); + } + int ft_read(byte * buf); + int rnd_init(bool scan); + int rnd_next(byte * buf); + int rnd_pos(byte * buf, byte * pos); + int restart_rnd_next(byte * buf, byte * pos); + void position(const byte * record); + void info(uint); + int extra(enum ha_extra_function operation); + int extra_opt(enum ha_extra_function operation, ulong cache_size); + int external_lock(THD * thd, int lock_type); + int delete_all_rows(void); + int disable_indexes(uint mode); + int enable_indexes(uint mode); + int indexes_are_disabled(void); + void start_bulk_insert(ha_rows rows); + int end_bulk_insert(); + ha_rows records_in_range(uint inx, key_range * min_key, key_range * max_key); + void update_create_info(HA_CREATE_INFO * create_info); + int create(const char *name, TABLE * form, HA_CREATE_INFO * create_info); + THR_LOCK_DATA **store_lock(THD * thd, THR_LOCK_DATA ** to, + enum thr_lock_type lock_type); + ulonglong get_auto_increment(); + int rename_table(const char *from, const char *to); + int delete_table(const char *name); + int check(THD * thd, HA_CHECK_OPT * check_opt); + int analyze(THD * thd, HA_CHECK_OPT * check_opt); + int repair(THD * thd, HA_CHECK_OPT * check_opt); + bool check_and_repair(THD * thd); + bool is_crashed() const; + bool auto_repair() const + { return maria_recover_options != 0; } + int optimize(THD * thd, HA_CHECK_OPT * check_opt); + int restore(THD * thd, HA_CHECK_OPT * check_opt); + int backup(THD * thd, HA_CHECK_OPT * check_opt); + int assign_to_keycache(THD * thd, HA_CHECK_OPT * check_opt); + int preload_keys(THD * thd, HA_CHECK_OPT * check_opt); + bool check_if_incompatible_data(HA_CREATE_INFO * info, uint table_changes); +#ifdef HAVE_REPLICATION + int dump(THD * thd, int fd); + int net_read_dump(NET * net); +#endif +}; diff --git a/sql/ha_myisam.cc b/sql/ha_myisam.cc index ec39ee00efc..c347bc64f61 100644 --- a/sql/ha_myisam.cc +++ b/sql/ha_myisam.cc @@ -105,7 +105,7 @@ static handler *myisam_create_handler(TABLE_SHARE *table) // collect errors printed by mi_check routines -static void mi_check_print_msg(MI_CHECK *param, const char* msg_type, +static void mi_check_print_msg(HA_CHECK *param, const char* msg_type, const char *fmt, va_list args) { THD* thd = (THD*)param->thd; @@ -146,13 +146,13 @@ static void mi_check_print_msg(MI_CHECK *param, const char* msg_type, extern "C" { -volatile int *killed_ptr(MI_CHECK *param) +volatile int *killed_ptr(HA_CHECK *param) { /* In theory Unsafe conversion, but should be ok for now */ return (int*) &(((THD *)(param->thd))->killed); } -void mi_check_print_error(MI_CHECK *param, const char *fmt,...) +void mi_check_print_error(HA_CHECK *param, const char *fmt,...) { param->error_printed|=1; param->out_flag|= O_DATA_LOST; @@ -162,7 +162,7 @@ void mi_check_print_error(MI_CHECK *param, const char *fmt,...) va_end(args); } -void mi_check_print_info(MI_CHECK *param, const char *fmt,...) +void mi_check_print_info(HA_CHECK *param, const char *fmt,...) { va_list args; va_start(args, fmt); @@ -170,7 +170,7 @@ void mi_check_print_info(MI_CHECK *param, const char *fmt,...) va_end(args); } -void mi_check_print_warning(MI_CHECK *param, const char *fmt,...) +void mi_check_print_warning(HA_CHECK *param, const char *fmt,...) { param->warning_printed=1; param->out_flag|= O_DATA_LOST; @@ -382,7 +382,7 @@ int ha_myisam::check(THD* thd, HA_CHECK_OPT* check_opt) { if (!file) return HA_ADMIN_INTERNAL_ERROR; int error; - MI_CHECK param; + HA_CHECK param; MYISAM_SHARE* share = file->s; const char *old_proc_info=thd->proc_info; @@ -393,7 +393,7 @@ int ha_myisam::check(THD* thd, HA_CHECK_OPT* check_opt) param.db_name= table->s->db.str; param.table_name= table->alias; param.testflag = check_opt->flags | T_CHECK | T_SILENT; - param.stats_method= (enum_mi_stats_method)thd->variables.myisam_stats_method; + param.stats_method= (enum_handler_stats_method)thd->variables.myisam_stats_method; if (!(table->db_stat & HA_READ_ONLY)) param.testflag|= T_STATISTICS; @@ -474,7 +474,7 @@ int ha_myisam::check(THD* thd, HA_CHECK_OPT* check_opt) int ha_myisam::analyze(THD *thd, HA_CHECK_OPT* check_opt) { int error=0; - MI_CHECK param; + HA_CHECK param; MYISAM_SHARE* share = file->s; myisamchk_init(¶m); @@ -485,7 +485,7 @@ int ha_myisam::analyze(THD *thd, HA_CHECK_OPT* check_opt) param.testflag= (T_FAST | T_CHECK | T_SILENT | T_STATISTICS | T_DONT_CHECK_CHECKSUM); param.using_global_keycache = 1; - param.stats_method= (enum_mi_stats_method)thd->variables.myisam_stats_method; + param.stats_method= (enum_handler_stats_method)thd->variables.myisam_stats_method; if (!(share->state.changed & STATE_NOT_ANALYZED)) return HA_ADMIN_ALREADY_DONE; @@ -531,7 +531,7 @@ int ha_myisam::restore(THD* thd, HA_CHECK_OPT *check_opt) err: { - MI_CHECK param; + HA_CHECK param; myisamchk_init(¶m); param.thd= thd; param.op_name= "restore"; @@ -591,7 +591,7 @@ int ha_myisam::backup(THD* thd, HA_CHECK_OPT *check_opt) err: { - MI_CHECK param; + HA_CHECK param; myisamchk_init(¶m); param.thd= thd; param.op_name= "backup"; @@ -607,7 +607,7 @@ int ha_myisam::backup(THD* thd, HA_CHECK_OPT *check_opt) int ha_myisam::repair(THD* thd, HA_CHECK_OPT *check_opt) { int error; - MI_CHECK param; + HA_CHECK param; ha_rows start_records; if (!file) return HA_ADMIN_INTERNAL_ERROR; @@ -657,7 +657,7 @@ int ha_myisam::optimize(THD* thd, HA_CHECK_OPT *check_opt) { int error; if (!file) return HA_ADMIN_INTERNAL_ERROR; - MI_CHECK param; + HA_CHECK param; myisamchk_init(¶m); param.thd = thd; @@ -676,7 +676,7 @@ int ha_myisam::optimize(THD* thd, HA_CHECK_OPT *check_opt) } -int ha_myisam::repair(THD *thd, MI_CHECK ¶m, bool optimize) +int ha_myisam::repair(THD *thd, HA_CHECK ¶m, bool optimize) { int error=0; uint local_testflag=param.testflag; @@ -854,7 +854,7 @@ int ha_myisam::assign_to_keycache(THD* thd, HA_CHECK_OPT *check_opt) if (error != HA_ADMIN_OK) { /* Send error to user */ - MI_CHECK param; + HA_CHECK param; myisamchk_init(¶m); param.thd= thd; param.op_name= "assign_to_keycache"; @@ -922,7 +922,7 @@ int ha_myisam::preload_keys(THD* thd, HA_CHECK_OPT *check_opt) err: { - MI_CHECK param; + HA_CHECK param; myisamchk_init(¶m); param.thd= thd; param.op_name= "preload_keys"; @@ -1029,7 +1029,7 @@ int ha_myisam::enable_indexes(uint mode) else if (mode == HA_KEY_SWITCH_NONUNIQ_SAVE) { THD *thd=current_thd; - MI_CHECK param; + HA_CHECK param; const char *save_proc_info=thd->proc_info; thd->proc_info="Creating index"; myisamchk_init(¶m); @@ -1038,7 +1038,7 @@ int ha_myisam::enable_indexes(uint mode) T_CREATE_MISSING_KEYS); param.myf_rw&= ~MY_WAIT_IF_FULL; param.sort_buffer_length= thd->variables.myisam_sort_buff_size; - param.stats_method= (enum_mi_stats_method)thd->variables.myisam_stats_method; + param.stats_method= (enum_handler_stats_method)thd->variables.myisam_stats_method; param.tmpdir=&mysql_tmpdir_list; if ((error= (repair(thd,param,0) != HA_ADMIN_OK)) && param.retry_repair) { @@ -1685,7 +1685,7 @@ ulonglong ha_myisam::get_auto_increment() { ulonglong nr; int error; - byte key[MI_MAX_KEY_LENGTH]; + byte key[HA_MAX_KEY_LENGTH]; if (!table->s->next_number_key_offset) { // Autoincrement at key-start diff --git a/sql/ha_myisam.h b/sql/ha_myisam.h index 86efed27478..7a6522dd3ef 100644 --- a/sql/ha_myisam.h +++ b/sql/ha_myisam.h @@ -22,6 +22,7 @@ /* class for the the myisam handler */ #include +#include #include #define HA_RECOVER_NONE 0 /* No automatic recover */ @@ -40,7 +41,7 @@ class ha_myisam: public handler ulong int_table_flags; char *data_file_name, *index_file_name; bool can_enable_indexes; - int repair(THD *thd, MI_CHECK ¶m, bool optimize); + int repair(THD *thd, HA_CHECK ¶m, bool optimize); public: ha_myisam(TABLE_SHARE *table_arg); @@ -56,8 +57,8 @@ class ha_myisam: public handler HA_READ_ORDER | HA_KEYREAD_ONLY); } uint max_supported_keys() const { return MI_MAX_KEY; } - uint max_supported_key_length() const { return MI_MAX_KEY_LENGTH; } - uint max_supported_key_part_length() const { return MI_MAX_KEY_LENGTH; } + uint max_supported_key_length() const { return HA_MAX_KEY_LENGTH; } + uint max_supported_key_part_length() const { return HA_MAX_KEY_LENGTH; } uint checksum() const; virtual bool check_if_locking_is_allowed(uint sql_command, diff --git a/sql/ha_myisammrg.h b/sql/ha_myisammrg.h index 4327b1c17b9..db8002ceccf 100644 --- a/sql/ha_myisammrg.h +++ b/sql/ha_myisammrg.h @@ -47,8 +47,8 @@ class ha_myisammrg: public handler HA_READ_ORDER | HA_KEYREAD_ONLY); } uint max_supported_keys() const { return MI_MAX_KEY; } - uint max_supported_key_length() const { return MI_MAX_KEY_LENGTH; } - uint max_supported_key_part_length() const { return MI_MAX_KEY_LENGTH; } + uint max_supported_key_length() const { return HA_MAX_KEY_LENGTH; } + uint max_supported_key_part_length() const { return HA_MAX_KEY_LENGTH; } double scan_time() { return ulonglong2double(data_file_length) / IO_SIZE + file->tables; } diff --git a/sql/handler.h b/sql/handler.h index 261a813bbfa..83f8671df93 100644 --- a/sql/handler.h +++ b/sql/handler.h @@ -21,6 +21,7 @@ #pragma interface /* gcc class implementation */ #endif +#include #include #include @@ -233,6 +234,7 @@ enum legacy_db_type DB_TYPE_BLACKHOLE_DB, DB_TYPE_PARTITION_DB, DB_TYPE_BINLOG, + DB_TYPE_MARIA, DB_TYPE_DEFAULT=127 // Must be last }; diff --git a/sql/item_func.h b/sql/item_func.h index ccbbbab1df4..89a2b9a779b 100644 --- a/sql/item_func.h +++ b/sql/item_func.h @@ -1261,7 +1261,6 @@ public: /* for fulltext search */ -#include class Item_func_match :public Item_real_func { diff --git a/sql/mysql_priv.h b/sql/mysql_priv.h index f79472a54f8..41b73378d47 100644 --- a/sql/mysql_priv.h +++ b/sql/mysql_priv.h @@ -1440,6 +1440,12 @@ extern handlerton partition_hton; #else extern SHOW_COMP_OPTION have_partition_db; #endif +#ifdef WITH_MARIA_STORAGE_ENGINE +extern handlerton maria_hton; +#define have_maria maria_hton.state +#else +extern SHOW_COMP_OPTION have_maria; +#endif extern handlerton myisam_hton; extern handlerton myisammrg_hton; diff --git a/sql/mysqld.cc b/sql/mysqld.cc index 71067630535..98378ed7eb4 100644 --- a/sql/mysqld.cc +++ b/sql/mysqld.cc @@ -27,6 +27,9 @@ #include "event.h" #include "ha_myisam.h" +#ifdef WITH_MARIA_STORAGE_ENGINE +#include "ha_maria.h" +#endif #ifdef WITH_INNOBASE_STORAGE_ENGINE #define OPT_INNODB_DEFAULT 1 @@ -529,6 +532,7 @@ char *mysqld_unix_port, *opt_mysql_tmpdir; const char **errmesg; /* Error messages */ const char *myisam_recover_options_str="OFF"; const char *myisam_stats_method_str="nulls_unequal"; +const char *maria_stats_method_str="nulls_unequal"; /* name of reference on left espression in rewritten IN subquery */ const char *in_left_expr_name= ""; /* name of additional condition */ @@ -4729,10 +4733,17 @@ enum options_mysqld OPT_MAX_LENGTH_FOR_SORT_DATA, OPT_MAX_WRITE_LOCK_COUNT, OPT_BULK_INSERT_BUFFER_SIZE, OPT_MAX_ERROR_COUNT, OPT_MULTI_RANGE_COUNT, OPT_MYISAM_DATA_POINTER_SIZE, + OPT_MYISAM_BLOCK_SIZE, OPT_MYISAM_MAX_EXTRA_SORT_FILE_SIZE, OPT_MYISAM_MAX_SORT_FILE_SIZE, OPT_MYISAM_SORT_BUFFER_SIZE, - OPT_MYISAM_USE_MMAP, + OPT_MYISAM_USE_MMAP, OPT_MYISAM_REPAIR_THREADS, OPT_MYISAM_STATS_METHOD, + + OPT_MARIA_BLOCK_SIZE, + OPT_MARIA_MAX_SORT_FILE_SIZE, OPT_MARIA_SORT_BUFFER_SIZE, + OPT_MARIA_USE_MMAP, OPT_MARIA_REPAIR_THREADS, + OPT_MARIA_STATS_METHOD, + OPT_NET_BUFFER_LENGTH, OPT_NET_RETRY_COUNT, OPT_NET_READ_TIMEOUT, OPT_NET_WRITE_TIMEOUT, OPT_OPEN_FILES_LIMIT, @@ -4746,7 +4757,7 @@ enum options_mysqld OPT_SORT_BUFFER, OPT_TABLE_OPEN_CACHE, OPT_TABLE_DEF_CACHE, OPT_THREAD_CONCURRENCY, OPT_THREAD_CACHE_SIZE, OPT_TMP_TABLE_SIZE, OPT_THREAD_STACK, - OPT_WAIT_TIMEOUT, OPT_MYISAM_REPAIR_THREADS, + OPT_WAIT_TIMEOUT, OPT_INNODB_MIRRORED_LOG_GROUPS, OPT_INNODB_LOG_FILES_IN_GROUP, OPT_INNODB_LOG_FILE_SIZE, @@ -5932,6 +5943,50 @@ log and this option does nothing anymore.", 0 #endif , 0, 2, 0, 1, 0}, +#ifdef WITH_MARIA_STORAGE_ENGINE + {"maria_block_size", OPT_MARIA_BLOCK_SIZE, + "Block size to be used for MARIA index pages.", + (gptr*) &maria_block_size, + (gptr*) &maria_block_size, 0, GET_ULONG, REQUIRED_ARG, + MARIA_KEY_BLOCK_LENGTH, MARIA_MIN_KEY_BLOCK_LENGTH, + MARIA_MAX_KEY_BLOCK_LENGTH, + 0, MARIA_MIN_KEY_BLOCK_LENGTH, 0}, + {"maria_key_buffer_size", OPT_KEY_BUFFER_SIZE, + "The size of the buffer used for index blocks for Maria tables. Increase " + "this to get better index handling (for all reads and multiple writes) to " + "as much as you can afford; 64M on a 256M machine that mainly runs MySQL " + "is quite common.", + (gptr*) &maria_key_cache_var.param_buff_size, (gptr*) 0, + 0, (GET_ULL | GET_ASK_ADDR), + REQUIRED_ARG, KEY_CACHE_SIZE, MALLOC_OVERHEAD, ~(ulong) 0, MALLOC_OVERHEAD, + IO_SIZE, 0}, + {"maria_max_sort_file_size", OPT_MARIA_MAX_SORT_FILE_SIZE, + "Don't use the fast sort index method to created index if the temporary " + "file would get bigger than this.", + (gptr*) &global_system_variables.maria_max_sort_file_size, + (gptr*) &max_system_variables.maria_max_sort_file_size, 0, + GET_ULL, REQUIRED_ARG, (longlong) LONG_MAX, 0, (ulonglong) MAX_FILE_SIZE, + 0, 1024*1024, 0}, + {"maria_repair_threads", OPT_MARIA_REPAIR_THREADS, + "Number of threads to use when repairing maria tables. The value of 1 " + "disables parallel repair.", + (gptr*) &global_system_variables.maria_repair_threads, + (gptr*) &max_system_variables.maria_repair_threads, 0, + GET_ULONG, REQUIRED_ARG, 1, 1, ~0L, 0, 1, 0}, + {"maria_sort_buffer_size", OPT_MARIA_SORT_BUFFER_SIZE, + "The buffer that is allocated when sorting the index when doing a REPAIR " + "or when creating indexes with CREATE INDEX or ALTER TABLE.", + (gptr*) &global_system_variables.maria_sort_buff_size, + (gptr*) &max_system_variables.maria_sort_buff_size, 0, + GET_ULONG, REQUIRED_ARG, 8192*1024, 4, ~0L, 0, 1, 0}, + {"maria_stats_method", OPT_MARIA_STATS_METHOD, + "Specifies how maria index statistics collection code should threat NULLs. " + "Possible values of name are \"nulls_unequal\" (default behavior for 4.1/5.0), " + "\"nulls_equal\" (emulate 4.0 behavior), and \"nulls_ignored\".", + (gptr*) &maria_stats_method_str, (gptr*) &maria_stats_method_str, 0, + GET_STR, REQUIRED_ARG, 0, 0, 0, 0, 0, 0}, +#endif + {"max_allowed_packet", OPT_MAX_ALLOWED_PACKET, "Max packetlength to send/receive from to server.", (gptr*) &global_system_variables.max_allowed_packet, @@ -6028,12 +6083,6 @@ The minimum value for this variable is 4096.", (gptr*) &myisam_data_pointer_size, (gptr*) &myisam_data_pointer_size, 0, GET_ULONG, REQUIRED_ARG, 6, 2, 7, 0, 1, 0}, - {"myisam_max_extra_sort_file_size", OPT_MYISAM_MAX_EXTRA_SORT_FILE_SIZE, - "Deprecated option", - (gptr*) &global_system_variables.myisam_max_extra_sort_file_size, - (gptr*) &max_system_variables.myisam_max_extra_sort_file_size, - 0, GET_ULL, REQUIRED_ARG, (ulonglong) MI_MAX_TEMP_LENGTH, - 0, (ulonglong) MAX_FILE_SIZE, 0, 1, 0}, {"myisam_max_sort_file_size", OPT_MYISAM_MAX_SORT_FILE_SIZE, "Don't use the fast sort index method to created index if the temporary file would get bigger than this.", (gptr*) &global_system_variables.myisam_max_sort_file_size, @@ -6997,15 +7046,24 @@ static void mysql_init_variables(void) query_id= thread_id= 1L; strmov(server_version, MYSQL_SERVER_VERSION); myisam_recover_options_str= sql_mode_str= "OFF"; - myisam_stats_method_str= "nulls_unequal"; + myisam_stats_method_str= maria_stats_method_str= "nulls_unequal"; my_bind_addr = htonl(INADDR_ANY); threads.empty(); thread_cache.empty(); key_caches.empty(); if (!(dflt_key_cache= get_or_create_key_cache(default_key_cache_base.str, - default_key_cache_base.length))) + default_key_cache_base.length))) + exit(1); +#ifdef WITH_MARIA_STORAGE_ENGINE + if (!(maria_key_cache= get_or_create_key_cache(maria_key_cache_base.str, + maria_key_cache_base.length))) exit(1); - multi_keycache_init(); /* set key_cache_hash.default_value = dflt_key_cache */ + maria_key_cache->param_buff_size= maria_key_cache_var.param_buff_size; + maria_key_cache->param_block_size= maria_block_size; +#endif + + /* set key_cache_hash.default_value = dflt_key_cache */ + multi_keycache_init(); /* Set directory paths */ strmake(language, LANGUAGE, sizeof(language)-1); @@ -7692,7 +7750,6 @@ get_one_option(int optid, const struct my_option *opt __attribute__((unused)), int method; LINT_INIT(method_conv); - myisam_stats_method_str= argument; if ((method=find_type(argument, &myisam_stats_method_typelib, 2)) <= 0) { fprintf(stderr, "Invalid value of myisam_stats_method: %s.\n", argument); diff --git a/sql/set_var.cc b/sql/set_var.cc index f2694f651f4..c8c35d52099 100644 --- a/sql/set_var.cc +++ b/sql/set_var.cc @@ -55,6 +55,9 @@ #include #include #include +#ifdef WITH_MARIA_STORAGE_ENGINE +#include +#endif /* WITH_BERKELEY_STORAGE_ENGINE */ extern bool berkeley_shared_data; @@ -150,6 +153,7 @@ static void fix_max_join_size(THD *thd, enum_var_type type); static void fix_query_cache_size(THD *thd, enum_var_type type); static void fix_query_cache_min_res_unit(THD *thd, enum_var_type type); static void fix_myisam_max_sort_file_size(THD *thd, enum_var_type type); +static void fix_maria_max_sort_file_size(THD *thd, enum_var_type type); static void fix_max_binlog_size(THD *thd, enum_var_type type); static void fix_max_relay_log_size(THD *thd, enum_var_type type); static void fix_max_connections(THD *thd, enum_var_type type); @@ -338,6 +342,14 @@ sys_var_thd_enum sys_myisam_stats_method("myisam_stats_method", &myisam_stats_method_typelib, NULL); +sys_var_thd_ulonglong sys_maria_max_sort_file_size("maria_max_sort_file_size", &SV::maria_max_sort_file_size, fix_maria_max_sort_file_size, 1); +sys_var_thd_ulong sys_maria_repair_threads("maria_repair_threads", &SV::maria_repair_threads); +sys_var_thd_ulong sys_maria_sort_buffer_size("maria_sort_buffer_size", &SV::maria_sort_buff_size); +sys_var_thd_enum sys_maria_stats_method("maria_stats_method", + &SV::maria_stats_method, + &myisam_stats_method_typelib, + NULL); + sys_var_thd_ulong sys_net_buffer_length("net_buffer_length", &SV::net_buffer_length); sys_var_thd_ulong sys_net_read_timeout("net_read_timeout", @@ -637,6 +649,7 @@ sys_var_have_variable sys_have_federated_db("have_federated_engine", &have_federated_db); sys_var_have_variable sys_have_geometry("have_geometry", &have_geometry); sys_var_have_variable sys_have_innodb("have_innodb", &have_innodb); +sys_var_have_variable sys_have_maria("have_maria", &have_maria); sys_var_have_variable sys_have_ndbcluster("have_ndbcluster", &have_ndbcluster); sys_var_have_variable sys_have_openssl("have_openssl", &have_openssl); sys_var_have_variable sys_have_partition_db("have_partitioning", @@ -760,6 +773,7 @@ SHOW_VAR init_vars[]= { {sys_have_federated_db.name,(char*) &have_federated_db, SHOW_HAVE}, {sys_have_geometry.name, (char*) &have_geometry, SHOW_HAVE}, {sys_have_innodb.name, (char*) &have_innodb, SHOW_HAVE}, + {sys_have_maria.name, (char*) &have_maria, SHOW_HAVE}, {sys_have_ndbcluster.name, (char*) &have_ndbcluster, SHOW_HAVE}, {sys_have_openssl.name, (char*) &have_openssl, SHOW_HAVE}, {sys_have_partition_db.name,(char*) &have_partition_db, SHOW_HAVE}, @@ -834,6 +848,15 @@ SHOW_VAR init_vars[]= { {sys_low_priority_updates.name, (char*) &sys_low_priority_updates, SHOW_SYS}, {"lower_case_file_system", (char*) &lower_case_file_system, SHOW_MY_BOOL}, {"lower_case_table_names", (char*) &lower_case_table_names, SHOW_INT}, + + {sys_maria_max_sort_file_size.name, (char*) &sys_maria_max_sort_file_size, + SHOW_SYS}, + {sys_maria_repair_threads.name, (char*) &sys_maria_repair_threads, + SHOW_SYS}, + {sys_maria_sort_buffer_size.name, (char*) &sys_maria_sort_buffer_size, + SHOW_SYS}, + {sys_maria_stats_method.name, (char*) &sys_maria_stats_method, SHOW_SYS}, + {sys_max_allowed_packet.name,(char*) &sys_max_allowed_packet, SHOW_SYS}, {sys_max_binlog_cache_size.name,(char*) &sys_max_binlog_cache_size, SHOW_SYS}, {sys_max_binlog_size.name, (char*) &sys_max_binlog_size, SHOW_SYS}, @@ -1116,6 +1139,16 @@ fix_myisam_max_sort_file_size(THD *thd, enum_var_type type) (my_off_t) global_system_variables.myisam_max_sort_file_size; } +static void +fix_maria_max_sort_file_size(THD *thd, enum_var_type type) +{ +#ifdef WITH_MARIA_STORAGE_ENGINE + maria_max_temp_length= + (my_off_t) global_system_variables.myisam_max_sort_file_size; +#endif +} + + /* Set the OPTION_BIG_SELECTS flag if max_join_size == HA_POS_ERROR */ @@ -2299,6 +2332,7 @@ void sys_var_collation_server::set_default(THD *thd, enum_var_type type) LEX_STRING default_key_cache_base= {(char *) "default", 7 }; +LEX_STRING maria_key_cache_base= {(char *) "maria", 5 }; static KEY_CACHE zero_key_cache; @@ -2308,7 +2342,7 @@ KEY_CACHE *get_key_cache(LEX_STRING *cache_name) if (!cache_name || ! cache_name->length) cache_name= &default_key_cache_base; return ((KEY_CACHE*) find_named(&key_caches, - cache_name->str, cache_name->length, 0)); + cache_name->str, cache_name->length, 0)); } diff --git a/sql/set_var.h b/sql/set_var.h index f62d6ce8d2a..53b14369116 100644 --- a/sql/set_var.h +++ b/sql/set_var.h @@ -985,6 +985,7 @@ public: extern sys_var_thd_bool sys_old_alter_table; extern sys_var_thd_bool sys_old_passwords; extern LEX_STRING default_key_cache_base; +extern LEX_STRING maria_key_cache_base; /* For sql_yacc */ struct sys_var_with_base diff --git a/sql/sql_class.h b/sql/sql_class.h index cc1c5488ae5..580a651b2c2 100644 --- a/sql/sql_class.h +++ b/sql/sql_class.h @@ -180,7 +180,7 @@ class Time_zone; struct system_variables { - ulonglong myisam_max_extra_sort_file_size; + ulonglong maria_max_sort_file_size; ulonglong myisam_max_sort_file_size; ha_rows select_limit; ha_rows max_join_size; @@ -195,6 +195,9 @@ struct system_variables ulong max_sort_length; ulong max_tmp_tables; ulong max_insert_delayed_threads; + ulong maria_repair_threads; + ulong maria_sort_buff_size; + ulong maria_stats_method; ulong multi_range_count; ulong myisam_repair_threads; ulong myisam_sort_buff_size; diff --git a/sql/sql_sort.h b/sql/sql_sort.h index 1831c4a2f3d..e39ea1aaf38 100644 --- a/sql/sql_sort.h +++ b/sql/sql_sort.h @@ -35,7 +35,9 @@ the callback function 'unpack_addon_fields'. */ -typedef struct st_sort_addon_field { /* Sort addon packed field */ +typedef struct st_sort_addon_field +{ + /* Sort addon packed field */ Field *field; /* Original field */ uint offset; /* Offset from the last sorted field */ uint null_offset; /* Offset to to null bit from the last sorted field */ @@ -43,13 +45,6 @@ typedef struct st_sort_addon_field { /* Sort addon packed field */ uint8 null_bit; /* Null bit mask for the field */ } SORT_ADDON_FIELD; -typedef struct st_buffpek { /* Struktur om sorteringsbuffrarna */ - my_off_t file_pos; /* Where we are in the sort file */ - uchar *base,*key; /* key pointers */ - ha_rows count; /* Number of rows in table */ - ulong mem_count; /* numbers of keys in memory */ - ulong max_keys; /* Max keys in buffert */ -} BUFFPEK; typedef struct st_sort_param { uint rec_length; /* Length of sorted records */ diff --git a/storage/Makefile.am b/storage/Makefile.am index 95c49b50890..8c68f105a16 100644 --- a/storage/Makefile.am +++ b/storage/Makefile.am @@ -21,7 +21,7 @@ AUTOMAKE_OPTIONS = foreign # These are built from source in the Docs directory EXTRA_DIST = SUBDIRS = -DIST_SUBDIRS = . csv example bdb heap innobase myisam myisammrg ndb archive +DIST_SUBDIRS = . csv example bdb heap innobase myisam myisammrg maria ndb archive # Don't update the files from bitkeeper %::SCCS/s.% diff --git a/storage/csv/ha_tina.cc b/storage/csv/ha_tina.cc index 38575d26242..e9f3d382595 100644 --- a/storage/csv/ha_tina.cc +++ b/storage/csv/ha_tina.cc @@ -112,7 +112,8 @@ handlerton tina_hton= { NULL, /* Fill FILES Table */ HTON_CAN_RECREATE, NULL, /* binlog_func */ - NULL /* binlog_log_query */ + NULL, /* binlog_log_query */ + NULL /* release_temporary_latches */ }; /***************************************************************************** diff --git a/storage/maria/Makefile.am b/storage/maria/Makefile.am new file mode 100644 index 00000000000..bf22428c18f --- /dev/null +++ b/storage/maria/Makefile.am @@ -0,0 +1,106 @@ +# Copyright (C) 2000 MySQL AB & MySQL Finland AB & TCX DataKonsult AB +# +# 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; either version 2 of the License, or +# (at your option) any later version. +# +# 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA + +EXTRA_DIST = ma_test_all.sh ma_test_all.res ma_ft_stem.c cmakelists.txt +pkgdata_DATA = ma_test_all ma_test_all.res + +INCLUDES = -I$(top_builddir)/include -I$(top_srcdir)/include +LDADD = @CLIENT_EXTRA_LDFLAGS@ libmaria.a \ + $(top_builddir)/storage/myisam/libmyisam.a \ + $(top_builddir)/mysys/libmysys.a \ + $(top_builddir)/dbug/libdbug.a \ + $(top_builddir)/strings/libmystrings.a @ZLIB_LIBS@ +pkglib_LIBRARIES = libmaria.a +bin_PROGRAMS = maria_chk maria_log maria_pack maria_ftdump +maria_chk_DEPENDENCIES= $(LIBRARIES) +maria_log_DEPENDENCIES= $(LIBRARIES) +maria_pack_DEPENDENCIES=$(LIBRARIES) +noinst_PROGRAMS = ma_test1 ma_test2 ma_test3 ma_rt_test ma_sp_test +noinst_HEADERS = maria_def.h ma_rt_index.h ma_rt_key.h ma_rt_mbr.h ma_sp_defs.h ma_fulltext.h ma_ftdefs.h ma_ft_test1.h ma_ft_eval.h +ma_test1_DEPENDENCIES= $(LIBRARIES) +ma_test2_DEPENDENCIES= $(LIBRARIES) +ma_test3_DEPENDENCIES= $(LIBRARIES) +#ma_ft_test1_DEPENDENCIES= $(LIBRARIES) +#ma_ft_eval_DEPENDENCIES= $(LIBRARIES) +maria_ftdump_DEPENDENCIES= $(LIBRARIES) +ma_rt_test_DEPENDENCIES= $(LIBRARIES) +ma_sp_test_DEPENDENCIES= $(LIBRARIES) +libmaria_a_SOURCES = ma_init.c ma_open.c ma_extra.c ma_info.c ma_rkey.c \ + ma_rnext.c ma_rnext_same.c \ + ma_search.c ma_page.c ma_key.c ma_locking.c \ + ma_rrnd.c ma_scan.c ma_cache.c \ + ma_statrec.c ma_packrec.c ma_dynrec.c \ + ma_update.c ma_write.c ma_unique.c \ + ma_delete.c \ + ma_rprev.c ma_rfirst.c ma_rlast.c ma_rsame.c \ + ma_rsamepos.c ma_panic.c ma_close.c ma_create.c\ + ma_range.c ma_dbug.c ma_checksum.c ma_log.c \ + ma_changed.c ma_static.c ma_delete_all.c \ + ma_delete_table.c ma_rename.c ma_check.c \ + ma_keycache.c ma_preload.c ma_ft_parser.c \ + ma_ft_update.c ma_ft_boolean_search.c \ + ma_ft_nlq_search.c ft_maria.c ma_sort.c \ + ma_rt_index.c ma_rt_key.c ma_rt_mbr.c ma_rt_split.c \ + ma_sp_key.c +CLEANFILES = test?.MA? FT?.MA? isam.log ma_test_all ma_rt_test.MA? sp_test.MA? +DEFS = + +SUFFIXES = .sh + +.sh: + @RM@ -f $@ $@-t + @SED@ \ + -e 's!@''bindir''@!$(bindir)!g' \ + -e 's!@''scriptdir''@!$(bindir)!g' \ + -e 's!@''prefix''@!$(prefix)!g' \ + -e 's!@''datadir''@!$(datadir)!g' \ + -e 's!@''localstatedir''@!$(localstatedir)!g' \ + -e 's!@''libexecdir''@!$(libexecdir)!g' \ + -e 's!@''CC''@!@CC@!'\ + -e 's!@''CXX''@!@CXX@!'\ + -e 's!@''GXX''@!@GXX@!'\ + -e 's!@''PERL''@!@PERL@!' \ + -e 's!@''CFLAGS''@!@SAVE_CFLAGS@!'\ + -e 's!@''CXXFLAGS''@!@SAVE_CXXFLAGS@!'\ + -e 's!@''LDFLAGS''@!@SAVE_LDFLAGS@!'\ + -e 's!@''VERSION''@!@VERSION@!' \ + -e 's!@''MYSQL_SERVER_SUFFIX''@!@MYSQL_SERVER_SUFFIX@!' \ + -e 's!@''COMPILATION_COMMENT''@!@COMPILATION_COMMENT@!' \ + -e 's!@''MACHINE_TYPE''@!@MACHINE_TYPE@!' \ + -e 's!@''HOSTNAME''@!@HOSTNAME@!' \ + -e 's!@''SYSTEM_TYPE''@!@SYSTEM_TYPE@!' \ + -e 's!@''CHECK_PID''@!@CHECK_PID@!' \ + -e 's!@''FIND_PROC''@!@FIND_PROC@!' \ + -e 's!@''MYSQLD_DEFAULT_SWITCHES''@!@MYSQLD_DEFAULT_SWITCHES@!' \ + -e 's!@''MYSQL_UNIX_ADDR''@!@MYSQL_UNIX_ADDR@!' \ + -e 's!@''TARGET_LINUX''@!@TARGET_LINUX@!' \ + -e "s!@""CONF_COMMAND""@!@CONF_COMMAND@!" \ + -e 's!@''MYSQLD_USER''@!@MYSQLD_USER@!' \ + -e 's!@''sysconfdir''@!@sysconfdir@!' \ + -e 's!@''SHORT_MYSQL_INTRO''@!@SHORT_MYSQL_INTRO@!' \ + -e 's!@''SHARED_LIB_VERSION''@!@SHARED_LIB_VERSION@!' \ + -e 's!@''MYSQL_BASE_VERSION''@!@MYSQL_BASE_VERSION@!' \ + -e 's!@''MYSQL_NO_DASH_VERSION''@!@MYSQL_NO_DASH_VERSION@!' \ + -e 's!@''MYSQL_TCP_PORT''@!@MYSQL_TCP_PORT@!' \ + -e 's!@''PERL_DBI_VERSION''@!@PERL_DBI_VERSION@!' \ + -e 's!@''PERL_DBD_VERSION''@!@PERL_DBD_VERSION@!' \ + -e 's!@''PERL_DATA_DUMPER''@!@PERL_DATA_DUMPER@!' \ + $< > $@-t + @CHMOD@ +x $@-t + @MV@ $@-t $@ + +# Don't update the files from bitkeeper +%::SCCS/s.% diff --git a/storage/maria/cmakelists.txt b/storage/maria/cmakelists.txt new file mode 100644 index 00000000000..3ba7aba4555 --- /dev/null +++ b/storage/maria/cmakelists.txt @@ -0,0 +1,26 @@ +SET(CMAKE_CXX_FLAGS_DEBUG "${CMAKE_CXX_FLAGS_DEBUG} -DSAFEMALLOC -DSAFE_MUTEX") +SET(CMAKE_C_FLAGS_DEBUG "${CMAKE_C_FLAGS_DEBUG} -DSAFEMALLOC -DSAFE_MUTEX") + +INCLUDE_DIRECTORIES(${CMAKE_SOURCE_DIR}/include) +ADD_LIBRARY(myisam ft_boolean_search.c ft_nlq_search.c ft_parser.c ft_static.c ft_stem.c + ft_stopwords.c ft_update.c mi_cache.c mi_changed.c mi_check.c + mi_checksum.c mi_close.c mi_create.c mi_dbug.c mi_delete.c + mi_delete_all.c mi_delete_table.c mi_dynrec.c mi_extra.c mi_info.c + mi_key.c mi_keycache.c mi_locking.c mi_log.c mi_open.c + mi_packrec.c mi_page.c mi_panic.c mi_preload.c mi_range.c mi_rename.c + mi_rfirst.c mi_rlast.c mi_rnext.c mi_rnext_same.c mi_rprev.c mi_rrnd.c + mi_rsame.c mi_rsamepos.c mi_scan.c mi_search.c mi_static.c mi_statrec.c + mi_unique.c mi_update.c mi_write.c rt_index.c rt_key.c rt_mbr.c + rt_split.c sort.c sp_key.c ft_eval.h myisamdef.h rt_index.h mi_rkey.c) + +ADD_EXECUTABLE(myisam_ftdump myisam_ftdump.c) +TARGET_LINK_LIBRARIES(myisam_ftdump myisam mysys dbug strings zlib wsock32) + +ADD_EXECUTABLE(myisamchk myisamchk.c) +TARGET_LINK_LIBRARIES(myisamchk myisam mysys dbug strings zlib wsock32) + +ADD_EXECUTABLE(myisamlog myisamlog.c) +TARGET_LINK_LIBRARIES(myisamlog myisam mysys dbug strings zlib wsock32) + +ADD_EXECUTABLE(myisampack myisampack.c) +TARGET_LINK_LIBRARIES(myisampack myisam mysys dbug strings zlib wsock32) diff --git a/storage/maria/ft_maria.c b/storage/maria/ft_maria.c new file mode 100644 index 00000000000..7104c6704ba --- /dev/null +++ b/storage/maria/ft_maria.c @@ -0,0 +1,49 @@ +/* Copyright (C) 2006 MySQL AB & MySQL Finland AB & TCX DataKonsult AB + + 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; either version 2 of the License, or + (at your option) any later version. + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ + +/* Written by Sergei A. Golubchik, who has a shared copyright to this code */ + +/* + This function is for interface functions between fulltext and maria +*/ + +#include "ma_ftdefs.h" + +FT_INFO *maria_ft_init_search(uint flags, void *info, uint keynr, + byte *query, uint query_len, CHARSET_INFO *cs, + byte *record) +{ + FT_INFO *res; + if (flags & FT_BOOL) + res= maria_ft_init_boolean_search((MARIA_HA *) info, keynr, query, + query_len, cs); + else + res= maria_ft_init_nlq_search((MARIA_HA *) info, keynr, query, query_len, + flags, record); + return res; +} + +const struct _ft_vft _ma_ft_vft_nlq = { + maria_ft_nlq_read_next, maria_ft_nlq_find_relevance, + maria_ft_nlq_close_search, maria_ft_nlq_get_relevance, + maria_ft_nlq_reinit_search +}; +const struct _ft_vft _ma_ft_vft_boolean = { + maria_ft_boolean_read_next, maria_ft_boolean_find_relevance, + maria_ft_boolean_close_search, maria_ft_boolean_get_relevance, + maria_ft_boolean_reinit_search +}; + diff --git a/storage/maria/ma_cache.c b/storage/maria/ma_cache.c new file mode 100644 index 00000000000..d6061c647ec --- /dev/null +++ b/storage/maria/ma_cache.c @@ -0,0 +1,108 @@ +/* Copyright (C) 2006 MySQL AB & MySQL Finland AB & TCX DataKonsult AB + + 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; either version 2 of the License, or + (at your option) any later version. + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ + +/* + Functions for read record cacheing with maria + Used for reading dynamic/compressed records from datafile. + + Can fetch data directly from file (outside cache), + if reading a small chunk straight before the cached part (with possible + overlap). + + Can be explicitly asked not to use cache (by not setting READING_NEXT in + flag) - useful for occasional out-of-cache reads, when the next read is + expected to hit the cache again. + + Allows "partial read" errors in the record header (when READING_HEADER flag + is set) - unread part is bzero'ed + + Note: out-of-cache reads are enabled for shared IO_CACHE's too, + as these reads will be cached by OS cache (and my_pread is always atomic) +*/ + + +#include "maria_def.h" + +int _ma_read_cache(IO_CACHE *info, byte *buff, my_off_t pos, uint length, + int flag) +{ + uint read_length,in_buff_length; + my_off_t offset; + char *in_buff_pos; + DBUG_ENTER("_ma_read_cache"); + + if (pos < info->pos_in_file) + { + read_length=length; + if ((my_off_t) read_length > (my_off_t) (info->pos_in_file-pos)) + read_length=(uint) (info->pos_in_file-pos); + info->seek_not_done=1; + if (my_pread(info->file,buff,read_length,pos,MYF(MY_NABP))) + DBUG_RETURN(1); + if (!(length-=read_length)) + DBUG_RETURN(0); + pos+=read_length; + buff+=read_length; + } + if (pos >= info->pos_in_file && + (offset= (my_off_t) (pos - info->pos_in_file)) < + (my_off_t) (info->read_end - info->request_pos)) + { + in_buff_pos=info->request_pos+(uint) offset; + in_buff_length= min(length,(uint) (info->read_end-in_buff_pos)); + memcpy(buff,info->request_pos+(uint) offset,(size_t) in_buff_length); + if (!(length-=in_buff_length)) + DBUG_RETURN(0); + pos+=in_buff_length; + buff+=in_buff_length; + } + else + in_buff_length=0; + if (flag & READING_NEXT) + { + if (pos != (info->pos_in_file + + (uint) (info->read_end - info->request_pos))) + { + info->pos_in_file=pos; /* Force start here */ + info->read_pos=info->read_end=info->request_pos; /* Everything used */ + info->seek_not_done=1; + } + else + info->read_pos=info->read_end; /* All block used */ + if (!(*info->read_function)(info,buff,length)) + DBUG_RETURN(0); + read_length=info->error; + } + else + { + info->seek_not_done=1; + if ((read_length=my_pread(info->file,buff,length,pos,MYF(0))) == length) + DBUG_RETURN(0); + } + if (!(flag & READING_HEADER) || (int) read_length == -1 || + read_length+in_buff_length < 3) + { + DBUG_PRINT("error", + ("Error %d reading next-multi-part block (Got %d bytes)", + my_errno, (int) read_length)); + if (!my_errno || my_errno == -1) + my_errno=HA_ERR_WRONG_IN_RECORD; + DBUG_RETURN(1); + } + bzero(buff+read_length,MARIA_BLOCK_INFO_HEADER_LENGTH - in_buff_length - + read_length); + DBUG_RETURN(0); +} /* _ma_read_cache */ diff --git a/storage/maria/ma_changed.c b/storage/maria/ma_changed.c new file mode 100644 index 00000000000..9e86212baa6 --- /dev/null +++ b/storage/maria/ma_changed.c @@ -0,0 +1,34 @@ +/* Copyright (C) 2006 MySQL AB & MySQL Finland AB & TCX DataKonsult AB + + 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; either version 2 of the License, or + (at your option) any later version. + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ + +/* Check if somebody has changed table since last check. */ + +#include "maria_def.h" + + /* Return 0 if table isn't changed */ + +int maria_is_changed(MARIA_HA *info) +{ + int result; + DBUG_ENTER("maria_is_changed"); + if (fast_ma_readinfo(info)) + DBUG_RETURN(-1); + VOID(_ma_writeinfo(info,0)); + result=(int) info->data_changed; + info->data_changed=0; + DBUG_PRINT("exit",("result: %d",result)); + DBUG_RETURN(result); +} diff --git a/storage/maria/ma_check.c b/storage/maria/ma_check.c new file mode 100644 index 00000000000..4680caed2da --- /dev/null +++ b/storage/maria/ma_check.c @@ -0,0 +1,4301 @@ +/* Copyright (C) 2006 MySQL AB & MySQL Finland AB & TCX DataKonsult AB + + 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; either version 2 of the License, or + (at your option) any later version. + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ + +/* Describe, check and repair of MARIA tables */ + +#include "ma_ftdefs.h" +#include +#include +#include +#include +#ifdef HAVE_SYS_VADVISE_H +#include +#endif +#ifdef HAVE_SYS_MMAN_H +#include +#endif +#include "ma_rt_index.h" + +#ifndef USE_RAID +#define my_raid_create(A,B,C,D,E,F,G) my_create(A,B,C,G) +#define my_raid_delete(A,B,C) my_delete(A,B) +#endif + + /* Functions defined in this file */ + +static int check_k_link(HA_CHECK *param, MARIA_HA *info,uint nr); +static int chk_index(HA_CHECK *param, MARIA_HA *info,MARIA_KEYDEF *keyinfo, + my_off_t page, uchar *buff, ha_rows *keys, + ha_checksum *key_checksum, uint level); +static uint isam_key_length(MARIA_HA *info,MARIA_KEYDEF *keyinfo); +static ha_checksum calc_checksum(ha_rows count); +static int writekeys(HA_CHECK *param, MARIA_HA *info,byte *buff, + my_off_t filepos); +static int sort_one_index(HA_CHECK *param, MARIA_HA *info,MARIA_KEYDEF *keyinfo, + my_off_t pagepos, File new_file); +static int sort_key_read(MARIA_SORT_PARAM *sort_param,void *key); +static int sort_maria_ft_key_read(MARIA_SORT_PARAM *sort_param,void *key); +static int sort_get_next_record(MARIA_SORT_PARAM *sort_param); +static int sort_key_cmp(MARIA_SORT_PARAM *sort_param, const void *a,const void *b); +static int sort_maria_ft_key_write(MARIA_SORT_PARAM *sort_param, const void *a); +static int sort_key_write(MARIA_SORT_PARAM *sort_param, const void *a); +static my_off_t get_record_for_key(MARIA_HA *info,MARIA_KEYDEF *keyinfo, + uchar *key); +static int sort_insert_key(MARIA_SORT_PARAM *sort_param, + reg1 SORT_KEY_BLOCKS *key_block, + uchar *key, my_off_t prev_block); +static int sort_delete_record(MARIA_SORT_PARAM *sort_param); +/*static int _ma_flush_pending_blocks(HA_CHECK *param);*/ +static SORT_KEY_BLOCKS *alloc_key_blocks(HA_CHECK *param, uint blocks, + uint buffer_length); +static ha_checksum maria_byte_checksum(const byte *buf, uint length); +static void set_data_file_type(MARIA_SORT_INFO *sort_info, MARIA_SHARE *share); + +void mariachk_init(HA_CHECK *param) +{ + bzero((gptr) param,sizeof(*param)); + param->opt_follow_links=1; + param->keys_in_use= ~(ulonglong) 0; + param->search_after_block=HA_OFFSET_ERROR; + param->auto_increment_value= 0; + param->use_buffers=USE_BUFFER_INIT; + param->read_buffer_length=READ_BUFFER_INIT; + param->write_buffer_length=READ_BUFFER_INIT; + param->sort_buffer_length=SORT_BUFFER_INIT; + param->sort_key_blocks=BUFFERS_WHEN_SORTING; + param->tmpfile_createflag=O_RDWR | O_TRUNC | O_EXCL; + param->myf_rw=MYF(MY_NABP | MY_WME | MY_WAIT_IF_FULL); + param->start_check_pos=0; + param->max_record_length= LONGLONG_MAX; + param->key_cache_block_size= KEY_CACHE_BLOCK_SIZE; + param->stats_method= MI_STATS_METHOD_NULLS_NOT_EQUAL; +} + + /* Check the status flags for the table */ + +int maria_chk_status(HA_CHECK *param, register MARIA_HA *info) +{ + MARIA_SHARE *share=info->s; + + if (maria_is_crashed_on_repair(info)) + _ma_check_print_warning(param, + "Table is marked as crashed and last repair failed"); + else if (maria_is_crashed(info)) + _ma_check_print_warning(param, + "Table is marked as crashed"); + if (share->state.open_count != (uint) (info->s->global_changed ? 1 : 0)) + { + /* Don't count this as a real warning, as check can correct this ! */ + uint save=param->warning_printed; + _ma_check_print_warning(param, + share->state.open_count==1 ? + "%d client is using or hasn't closed the table properly" : + "%d clients are using or haven't closed the table properly", + share->state.open_count); + /* If this will be fixed by the check, forget the warning */ + if (param->testflag & T_UPDATE_STATE) + param->warning_printed=save; + } + return 0; +} + + /* Check delete links */ + +int maria_chk_del(HA_CHECK *param, register MARIA_HA *info, uint test_flag) +{ + reg2 ha_rows i; + uint delete_link_length; + my_off_t empty,next_link,old_link; + char buff[22],buff2[22]; + DBUG_ENTER("maria_chk_del"); + + LINT_INIT(old_link); + param->record_checksum=0; + delete_link_length=((info->s->options & HA_OPTION_PACK_RECORD) ? 20 : + info->s->rec_reflength+1); + + if (!(test_flag & T_SILENT)) + puts("- check record delete-chain"); + + next_link=info->s->state.dellink; + if (info->state->del == 0) + { + if (test_flag & T_VERBOSE) + { + puts("No recordlinks"); + } + } + else + { + if (test_flag & T_VERBOSE) + printf("Recordlinks: "); + empty=0; + for (i= info->state->del ; i > 0L && next_link != HA_OFFSET_ERROR ; i--) + { + if (*_ma_killed_ptr(param)) + DBUG_RETURN(1); + if (test_flag & T_VERBOSE) + printf(" %9s",llstr(next_link,buff)); + if (next_link >= info->state->data_file_length) + goto wrong; + if (my_pread(info->dfile,(char*) buff,delete_link_length, + next_link,MYF(MY_NABP))) + { + if (test_flag & T_VERBOSE) puts(""); + _ma_check_print_error(param,"Can't read delete-link at filepos: %s", + llstr(next_link,buff)); + DBUG_RETURN(1); + } + if (*buff != '\0') + { + if (test_flag & T_VERBOSE) puts(""); + _ma_check_print_error(param,"Record at pos: %s is not remove-marked", + llstr(next_link,buff)); + goto wrong; + } + if (info->s->options & HA_OPTION_PACK_RECORD) + { + my_off_t prev_link=mi_sizekorr(buff+12); + if (empty && prev_link != old_link) + { + if (test_flag & T_VERBOSE) puts(""); + _ma_check_print_error(param,"Deleted block at %s doesn't point back at previous delete link",llstr(next_link,buff2)); + goto wrong; + } + old_link=next_link; + next_link=mi_sizekorr(buff+4); + empty+=mi_uint3korr(buff+1); + } + else + { + param->record_checksum+=(ha_checksum) next_link; + next_link= _ma_rec_pos(info->s,(uchar*) buff+1); + empty+=info->s->base.pack_reclength; + } + } + if (test_flag & T_VERBOSE) + puts("\n"); + if (empty != info->state->empty) + { + _ma_check_print_warning(param, + "Found %s deleted space in delete link chain. Should be %s", + llstr(empty,buff2), + llstr(info->state->empty,buff)); + } + if (next_link != HA_OFFSET_ERROR) + { + _ma_check_print_error(param, + "Found more than the expected %s deleted rows in delete link chain", + llstr(info->state->del, buff)); + goto wrong; + } + if (i != 0) + { + _ma_check_print_error(param, + "Found %s deleted rows in delete link chain. Should be %s", + llstr(info->state->del - i, buff2), + llstr(info->state->del, buff)); + goto wrong; + } + } + DBUG_RETURN(0); + +wrong: + param->testflag|=T_RETRY_WITHOUT_QUICK; + if (test_flag & T_VERBOSE) puts(""); + _ma_check_print_error(param,"record delete-link-chain corrupted"); + DBUG_RETURN(1); +} /* maria_chk_del */ + + + /* Check delete links in index file */ + +static int check_k_link(HA_CHECK *param, register MARIA_HA *info, uint nr) +{ + my_off_t next_link; + uint block_size=(nr+1)*MARIA_MIN_KEY_BLOCK_LENGTH; + ha_rows records; + char llbuff[21],*buff; + DBUG_ENTER("check_k_link"); + + if (param->testflag & T_VERBOSE) + printf("block_size %4d:",block_size); + + next_link=info->s->state.key_del[nr]; + records= (ha_rows) (info->state->key_file_length / block_size); + while (next_link != HA_OFFSET_ERROR && records > 0) + { + if (*_ma_killed_ptr(param)) + DBUG_RETURN(1); + if (param->testflag & T_VERBOSE) + printf("%16s",llstr(next_link,llbuff)); + if (next_link > info->state->key_file_length || + next_link & (info->s->blocksize-1)) + DBUG_RETURN(1); + if (!(buff=key_cache_read(info->s->key_cache, + info->s->kfile, next_link, DFLT_INIT_HITS, + (byte*) info->buff, + maria_block_size, block_size, 1))) + DBUG_RETURN(1); + next_link=mi_sizekorr(buff); + records--; + param->key_file_blocks+=block_size; + } + if (param->testflag & T_VERBOSE) + { + if (next_link != HA_OFFSET_ERROR) + printf("%16s\n",llstr(next_link,llbuff)); + else + puts(""); + } + DBUG_RETURN (next_link != HA_OFFSET_ERROR); +} /* check_k_link */ + + + /* Check sizes of files */ + +int maria_chk_size(HA_CHECK *param, register MARIA_HA *info) +{ + int error=0; + register my_off_t skr,size; + char buff[22],buff2[22]; + DBUG_ENTER("maria_chk_size"); + + if (!(param->testflag & T_SILENT)) puts("- check file-size"); + + /* The following is needed if called externally (not from mariachk) */ + flush_key_blocks(info->s->key_cache, + info->s->kfile, FLUSH_FORCE_WRITE); + + size=my_seek(info->s->kfile,0L,MY_SEEK_END,MYF(0)); + if ((skr=(my_off_t) info->state->key_file_length) != size) + { + /* Don't give error if file generated by mariapack */ + if (skr > size && maria_is_any_key_active(info->s->state.key_map)) + { + error=1; + _ma_check_print_error(param, + "Size of indexfile is: %-8s Should be: %s", + llstr(size,buff), llstr(skr,buff2)); + } + else + _ma_check_print_warning(param, + "Size of indexfile is: %-8s Should be: %s", + llstr(size,buff), llstr(skr,buff2)); + } + if (!(param->testflag & T_VERY_SILENT) && + ! (info->s->options & HA_OPTION_COMPRESS_RECORD) && + ulonglong2double(info->state->key_file_length) > + ulonglong2double(info->s->base.margin_key_file_length)*0.9) + _ma_check_print_warning(param,"Keyfile is almost full, %10s of %10s used", + llstr(info->state->key_file_length,buff), + llstr(info->s->base.max_key_file_length-1,buff)); + + size=my_seek(info->dfile,0L,MY_SEEK_END,MYF(0)); + skr=(my_off_t) info->state->data_file_length; + if (info->s->options & HA_OPTION_COMPRESS_RECORD) + skr+= MEMMAP_EXTRA_MARGIN; +#ifdef USE_RELOC + if (info->data_file_type == STATIC_RECORD && + skr < (my_off_t) info->s->base.reloc*info->s->base.min_pack_length) + skr=(my_off_t) info->s->base.reloc*info->s->base.min_pack_length; +#endif + if (skr != size) + { + info->state->data_file_length=size; /* Skip other errors */ + if (skr > size && skr != size + MEMMAP_EXTRA_MARGIN) + { + error=1; + _ma_check_print_error(param,"Size of datafile is: %-9s Should be: %s", + llstr(size,buff), llstr(skr,buff2)); + param->testflag|=T_RETRY_WITHOUT_QUICK; + } + else + { + _ma_check_print_warning(param, + "Size of datafile is: %-9s Should be: %s", + llstr(size,buff), llstr(skr,buff2)); + } + } + if (!(param->testflag & T_VERY_SILENT) && + !(info->s->options & HA_OPTION_COMPRESS_RECORD) && + ulonglong2double(info->state->data_file_length) > + (ulonglong2double(info->s->base.max_data_file_length)*0.9)) + _ma_check_print_warning(param, "Datafile is almost full, %10s of %10s used", + llstr(info->state->data_file_length,buff), + llstr(info->s->base.max_data_file_length-1,buff2)); + DBUG_RETURN(error); +} /* maria_chk_size */ + + + /* Check keys */ + +int maria_chk_key(HA_CHECK *param, register MARIA_HA *info) +{ + uint key,found_keys=0,full_text_keys=0,result=0; + ha_rows keys; + ha_checksum old_record_checksum,init_checksum; + my_off_t all_keydata,all_totaldata,key_totlength,length; + ulong *rec_per_key_part; + MARIA_SHARE *share=info->s; + MARIA_KEYDEF *keyinfo; + char buff[22],buff2[22]; + DBUG_ENTER("maria_chk_key"); + + if (!(param->testflag & T_SILENT)) + puts("- check key delete-chain"); + + param->key_file_blocks=info->s->base.keystart; + for (key=0 ; key < info->s->state.header.max_block_size ; key++) + if (check_k_link(param,info,key)) + { + if (param->testflag & T_VERBOSE) puts(""); + _ma_check_print_error(param,"key delete-link-chain corrupted"); + DBUG_RETURN(-1); + } + + if (!(param->testflag & T_SILENT)) puts("- check index reference"); + + all_keydata=all_totaldata=key_totlength=0; + old_record_checksum=0; + init_checksum=param->record_checksum; + if (!(share->options & + (HA_OPTION_PACK_RECORD | HA_OPTION_COMPRESS_RECORD))) + old_record_checksum=calc_checksum(info->state->records+info->state->del-1)* + share->base.pack_reclength; + rec_per_key_part= param->rec_per_key_part; + for (key= 0,keyinfo= &share->keyinfo[0]; key < share->base.keys ; + rec_per_key_part+=keyinfo->keysegs, key++, keyinfo++) + { + param->key_crc[key]=0; + if (! maria_is_key_active(share->state.key_map, key)) + { + /* Remember old statistics for key */ + memcpy((char*) rec_per_key_part, + (char*) (share->state.rec_per_key_part + + (uint) (rec_per_key_part - param->rec_per_key_part)), + keyinfo->keysegs*sizeof(*rec_per_key_part)); + continue; + } + found_keys++; + + param->record_checksum=init_checksum; + + bzero((char*) ¶m->unique_count,sizeof(param->unique_count)); + bzero((char*) ¶m->notnull_count,sizeof(param->notnull_count)); + + if ((!(param->testflag & T_SILENT))) + printf ("- check data record references index: %d\n",key+1); + if (keyinfo->flag & HA_FULLTEXT) + full_text_keys++; + if (share->state.key_root[key] == HA_OFFSET_ERROR && + (info->state->records == 0 || keyinfo->flag & HA_FULLTEXT)) + goto do_stat; + if (!_ma_fetch_keypage(info,keyinfo,share->state.key_root[key], + DFLT_INIT_HITS,info->buff,0)) + { + _ma_check_print_error(param,"Can't read indexpage from filepos: %s", + llstr(share->state.key_root[key],buff)); + if (!(param->testflag & T_INFO)) + DBUG_RETURN(-1); + result= -1; + continue; + } + param->key_file_blocks+=keyinfo->block_length; + keys=0; + param->keydata=param->totaldata=0; + param->key_blocks=0; + param->max_level=0; + if (chk_index(param,info,keyinfo,share->state.key_root[key],info->buff, + &keys, param->key_crc+key,1)) + DBUG_RETURN(-1); + if(!(keyinfo->flag & (HA_FULLTEXT | HA_SPATIAL))) + { + if (keys != info->state->records) + { + _ma_check_print_error(param,"Found %s keys of %s",llstr(keys,buff), + llstr(info->state->records,buff2)); + if (!(param->testflag & T_INFO)) + DBUG_RETURN(-1); + result= -1; + continue; + } + if (found_keys - full_text_keys == 1 && + ((share->options & + (HA_OPTION_PACK_RECORD | HA_OPTION_COMPRESS_RECORD)) || + (param->testflag & T_DONT_CHECK_CHECKSUM))) + old_record_checksum=param->record_checksum; + else if (old_record_checksum != param->record_checksum) + { + if (key) + _ma_check_print_error(param,"Key %u doesn't point at same records that key 1", + key+1); + else + _ma_check_print_error(param,"Key 1 doesn't point at all records"); + if (!(param->testflag & T_INFO)) + DBUG_RETURN(-1); + result= -1; + continue; + } + } + if ((uint) share->base.auto_key -1 == key) + { + /* Check that auto_increment key is bigger than max key value */ + ulonglong save_auto_value=info->s->state.auto_increment; + info->s->state.auto_increment=0; + info->lastinx=key; + _ma_read_key_record(info, 0L, info->rec_buff); + _ma_update_auto_increment(info, info->rec_buff); + if (info->s->state.auto_increment > save_auto_value) + { + _ma_check_print_warning(param, + "Auto-increment value: %s is smaller than max used value: %s", + llstr(save_auto_value,buff2), + llstr(info->s->state.auto_increment, buff)); + } + if (param->testflag & T_AUTO_INC) + { + set_if_bigger(info->s->state.auto_increment, + param->auto_increment_value); + } + else + info->s->state.auto_increment=save_auto_value; + + /* Check that there isn't a row with auto_increment = 0 in the table */ + maria_extra(info,HA_EXTRA_KEYREAD,0); + bzero(info->lastkey,keyinfo->seg->length); + if (!maria_rkey(info, info->rec_buff, key, (const byte*) info->lastkey, + keyinfo->seg->length, HA_READ_KEY_EXACT)) + { + /* Don't count this as a real warning, as mariachk can't correct it */ + uint save=param->warning_printed; + _ma_check_print_warning(param, + "Found row where the auto_increment column has the value 0"); + param->warning_printed=save; + } + maria_extra(info,HA_EXTRA_NO_KEYREAD,0); + } + + length=(my_off_t) isam_key_length(info,keyinfo)*keys + param->key_blocks*2; + if (param->testflag & T_INFO && param->totaldata != 0L && keys != 0L) + printf("Key: %2d: Keyblocks used: %3d%% Packed: %4d%% Max levels: %2d\n", + key+1, + (int) (my_off_t2double(param->keydata)*100.0/my_off_t2double(param->totaldata)), + (int) ((my_off_t2double(length) - my_off_t2double(param->keydata))*100.0/ + my_off_t2double(length)), + param->max_level); + all_keydata+=param->keydata; all_totaldata+=param->totaldata; key_totlength+=length; + +do_stat: + if (param->testflag & T_STATISTICS) + maria_update_key_parts(keyinfo, rec_per_key_part, param->unique_count, + param->stats_method == MI_STATS_METHOD_IGNORE_NULLS? + param->notnull_count: NULL, + (ulonglong)info->state->records); + } + if (param->testflag & T_INFO) + { + if (all_totaldata != 0L && found_keys > 0) + printf("Total: Keyblocks used: %3d%% Packed: %4d%%\n\n", + (int) (my_off_t2double(all_keydata)*100.0/ + my_off_t2double(all_totaldata)), + (int) ((my_off_t2double(key_totlength) - + my_off_t2double(all_keydata))*100.0/ + my_off_t2double(key_totlength))); + else if (all_totaldata != 0L && maria_is_any_key_active(share->state.key_map)) + puts(""); + } + if (param->key_file_blocks != info->state->key_file_length && + param->keys_in_use != ~(ulonglong) 0) + _ma_check_print_warning(param, "Some data are unreferenced in keyfile"); + if (found_keys != full_text_keys) + param->record_checksum=old_record_checksum-init_checksum; /* Remove delete links */ + else + param->record_checksum=0; + DBUG_RETURN(result); +} /* maria_chk_key */ + + +static int chk_index_down(HA_CHECK *param, MARIA_HA *info, MARIA_KEYDEF *keyinfo, + my_off_t page, uchar *buff, ha_rows *keys, + ha_checksum *key_checksum, uint level) +{ + char llbuff[22],llbuff2[22]; + if (page > info->state->key_file_length || (page & (info->s->blocksize -1))) + { + my_off_t max_length=my_seek(info->s->kfile,0L,MY_SEEK_END,MYF(0)); + _ma_check_print_error(param,"Wrong pagepointer: %s at page: %s", + llstr(page,llbuff),llstr(page,llbuff2)); + + if (page+info->s->blocksize > max_length) + goto err; + info->state->key_file_length=(max_length & + ~ (my_off_t) (info->s->blocksize-1)); + } + if (!_ma_fetch_keypage(info,keyinfo,page, DFLT_INIT_HITS,buff,0)) + { + _ma_check_print_error(param,"Can't read key from filepos: %s", + llstr(page,llbuff)); + goto err; + } + param->key_file_blocks+=keyinfo->block_length; + if (chk_index(param,info,keyinfo,page,buff,keys,key_checksum,level)) + goto err; + + return 0; +err: + return 1; +} + + +/* + "Ignore NULLs" statistics collection method: process first index tuple. + + SYNOPSIS + maria_collect_stats_nonulls_first() + keyseg IN Array of key part descriptions + notnull INOUT Array, notnull[i] = (number of {keypart1...keypart_i} + tuples that don't contain NULLs) + key IN Key values tuple + + DESCRIPTION + Process the first index tuple - find out which prefix tuples don't + contain NULLs, and update the array of notnull counters accordingly. +*/ + +static +void maria_collect_stats_nonulls_first(HA_KEYSEG *keyseg, ulonglong *notnull, + uchar *key) +{ + uint first_null, kp; + first_null= ha_find_null(keyseg, key) - keyseg; + /* + All prefix tuples that don't include keypart_{first_null} are not-null + tuples (and all others aren't), increment counters for them. + */ + for (kp= 0; kp < first_null; kp++) + notnull[kp]++; +} + + +/* + "Ignore NULLs" statistics collection method: process next index tuple. + + SYNOPSIS + maria_collect_stats_nonulls_next() + keyseg IN Array of key part descriptions + notnull INOUT Array, notnull[i] = (number of {keypart1...keypart_i} + tuples that don't contain NULLs) + prev_key IN Previous key values tuple + last_key IN Next key values tuple + + DESCRIPTION + Process the next index tuple: + 1. Find out which prefix tuples of last_key don't contain NULLs, and + update the array of notnull counters accordingly. + 2. Find the first keypart number where the prev_key and last_key tuples + are different(A), or last_key has NULL value(B), and return it, so the + caller can count number of unique tuples for each key prefix. We don't + need (B) to be counted, and that is compensated back in + maria_update_key_parts(). + + RETURN + 1 + number of first keypart where values differ or last_key tuple has NULL +*/ + +static +int maria_collect_stats_nonulls_next(HA_KEYSEG *keyseg, ulonglong *notnull, + uchar *prev_key, uchar *last_key) +{ + uint diffs[2]; + uint first_null_seg, kp; + HA_KEYSEG *seg; + + /* + Find the first keypart where values are different or either of them is + NULL. We get results in diffs array: + diffs[0]= 1 + number of first different keypart + diffs[1]=offset: (last_key + diffs[1]) points to first value in + last_key that is NULL or different from corresponding + value in prev_key. + */ + ha_key_cmp(keyseg, prev_key, last_key, USE_WHOLE_KEY, + SEARCH_FIND | SEARCH_NULL_ARE_NOT_EQUAL, diffs); + seg= keyseg + diffs[0] - 1; + + /* Find first NULL in last_key */ + first_null_seg= ha_find_null(seg, last_key + diffs[1]) - keyseg; + for (kp= 0; kp < first_null_seg; kp++) + notnull[kp]++; + + /* + Return 1+ number of first key part where values differ. Don't care if + these were NULLs and not .... We compensate for that in + maria_update_key_parts. + */ + return diffs[0]; +} + + + /* Check if index is ok */ + +static int chk_index(HA_CHECK *param, MARIA_HA *info, MARIA_KEYDEF *keyinfo, + my_off_t page, uchar *buff, ha_rows *keys, + ha_checksum *key_checksum, uint level) +{ + int flag; + uint used_length,comp_flag,nod_flag,key_length=0; + uchar key[HA_MAX_POSSIBLE_KEY_BUFF],*temp_buff,*keypos,*old_keypos,*endpos; + my_off_t next_page,record; + char llbuff[22]; + uint diff_pos[2]; + DBUG_ENTER("chk_index"); + DBUG_DUMP("buff",(byte*) buff,maria_getint(buff)); + + /* TODO: implement appropriate check for RTree keys */ + if (keyinfo->flag & HA_SPATIAL) + DBUG_RETURN(0); + + if (!(temp_buff=(uchar*) my_alloca((uint) keyinfo->block_length))) + { + _ma_check_print_error(param,"Not enough memory for keyblock"); + DBUG_RETURN(-1); + } + + if (keyinfo->flag & HA_NOSAME) + comp_flag=SEARCH_FIND | SEARCH_UPDATE; /* Not real duplicates */ + else + comp_flag=SEARCH_SAME; /* Keys in positionorder */ + nod_flag=_ma_test_if_nod(buff); + used_length=maria_getint(buff); + keypos=buff+2+nod_flag; + endpos=buff+used_length; + + param->keydata+=used_length; param->totaldata+=keyinfo->block_length; /* INFO */ + param->key_blocks++; + if (level > param->max_level) + param->max_level=level; + + if (used_length > keyinfo->block_length) + { + _ma_check_print_error(param,"Wrong pageinfo at page: %s", + llstr(page,llbuff)); + goto err; + } + for ( ;; ) + { + if (*_ma_killed_ptr(param)) + goto err; + memcpy((char*) info->lastkey,(char*) key,key_length); + info->lastkey_length=key_length; + if (nod_flag) + { + next_page= _ma_kpos(nod_flag,keypos); + if (chk_index_down(param,info,keyinfo,next_page, + temp_buff,keys,key_checksum,level+1)) + goto err; + } + old_keypos=keypos; + if (keypos >= endpos || + (key_length=(*keyinfo->get_key)(keyinfo,nod_flag,&keypos,key)) == 0) + break; + if (keypos > endpos) + { + _ma_check_print_error(param,"Wrong key block length at page: %s",llstr(page,llbuff)); + goto err; + } + if ((*keys)++ && + (flag=ha_key_cmp(keyinfo->seg,info->lastkey,key,key_length, + comp_flag, diff_pos)) >=0) + { + DBUG_DUMP("old",(byte*) info->lastkey, info->lastkey_length); + DBUG_DUMP("new",(byte*) key, key_length); + DBUG_DUMP("new_in_page",(char*) old_keypos,(uint) (keypos-old_keypos)); + + if (comp_flag & SEARCH_FIND && flag == 0) + _ma_check_print_error(param,"Found duplicated key at page %s",llstr(page,llbuff)); + else + _ma_check_print_error(param,"Key in wrong position at page %s",llstr(page,llbuff)); + goto err; + } + if (param->testflag & T_STATISTICS) + { + if (*keys != 1L) /* not first_key */ + { + if (param->stats_method == MI_STATS_METHOD_NULLS_NOT_EQUAL) + ha_key_cmp(keyinfo->seg,info->lastkey,key,USE_WHOLE_KEY, + SEARCH_FIND | SEARCH_NULL_ARE_NOT_EQUAL, + diff_pos); + else if (param->stats_method == MI_STATS_METHOD_IGNORE_NULLS) + { + diff_pos[0]= maria_collect_stats_nonulls_next(keyinfo->seg, + param->notnull_count, + info->lastkey, key); + } + param->unique_count[diff_pos[0]-1]++; + } + else + { + if (param->stats_method == MI_STATS_METHOD_IGNORE_NULLS) + maria_collect_stats_nonulls_first(keyinfo->seg, param->notnull_count, + key); + } + } + (*key_checksum)+= maria_byte_checksum((byte*) key, + key_length- info->s->rec_reflength); + record= _ma_dpos(info,0,key+key_length); + if (keyinfo->flag & HA_FULLTEXT) /* special handling for ft2 */ + { + uint off; + int subkeys; + get_key_full_length_rdonly(off, key); + subkeys=ft_sintXkorr(key+off); + if (subkeys < 0) + { + ha_rows tmp_keys=0; + if (chk_index_down(param,info,&info->s->ft2_keyinfo,record, + temp_buff,&tmp_keys,key_checksum,1)) + goto err; + if (tmp_keys + subkeys) + { + _ma_check_print_error(param, + "Number of words in the 2nd level tree " + "does not match the number in the header. " + "Parent word in on the page %s, offset %u", + llstr(page,llbuff), (uint) (old_keypos-buff)); + goto err; + } + (*keys)+=tmp_keys-1; + continue; + } + /* fall through */ + } + if (record >= info->state->data_file_length) + { +#ifndef DBUG_OFF + char llbuff2[22], llbuff3[22]; +#endif + _ma_check_print_error(param,"Found key at page %s that points to record outside datafile",llstr(page,llbuff)); + DBUG_PRINT("test",("page: %s record: %s filelength: %s", + llstr(page,llbuff),llstr(record,llbuff2), + llstr(info->state->data_file_length,llbuff3))); + DBUG_DUMP("key",(byte*) key,key_length); + DBUG_DUMP("new_in_page",(char*) old_keypos,(uint) (keypos-old_keypos)); + goto err; + } + param->record_checksum+=(ha_checksum) record; + } + if (keypos != endpos) + { + _ma_check_print_error(param,"Keyblock size at page %s is not correct. Block length: %d key length: %d", + llstr(page,llbuff), used_length, (keypos - buff)); + goto err; + } + my_afree((byte*) temp_buff); + DBUG_RETURN(0); + err: + my_afree((byte*) temp_buff); + DBUG_RETURN(1); +} /* chk_index */ + + + /* Calculate a checksum of 1+2+3+4...N = N*(N+1)/2 without overflow */ + +static ha_checksum calc_checksum(ha_rows count) +{ + ulonglong sum,a,b; + DBUG_ENTER("calc_checksum"); + + sum=0; + a=count; b=count+1; + if (a & 1) + b>>=1; + else + a>>=1; + while (b) + { + if (b & 1) + sum+=a; + a<<=1; b>>=1; + } + DBUG_PRINT("exit",("sum: %lx",(ulong) sum)); + DBUG_RETURN((ha_checksum) sum); +} /* calc_checksum */ + + + /* Calc length of key in normal isam */ + +static uint isam_key_length(MARIA_HA *info, register MARIA_KEYDEF *keyinfo) +{ + uint length; + HA_KEYSEG *keyseg; + DBUG_ENTER("isam_key_length"); + + length= info->s->rec_reflength; + for (keyseg=keyinfo->seg ; keyseg->type ; keyseg++) + length+= keyseg->length; + + DBUG_PRINT("exit",("length: %d",length)); + DBUG_RETURN(length); +} /* key_length */ + + + /* Check that record-link is ok */ + +int maria_chk_data_link(HA_CHECK *param, MARIA_HA *info,int extend) +{ + int error,got_error,flag; + uint key,left_length,b_type,field; + ha_rows records,del_blocks; + my_off_t used,empty,pos,splits,start_recpos, + del_length,link_used,start_block; + byte *record,*to; + char llbuff[22],llbuff2[22],llbuff3[22]; + ha_checksum intern_record_checksum; + ha_checksum key_checksum[HA_MAX_POSSIBLE_KEY]; + my_bool static_row_size; + MARIA_KEYDEF *keyinfo; + MARIA_BLOCK_INFO block_info; + DBUG_ENTER("maria_chk_data_link"); + + if (!(param->testflag & T_SILENT)) + { + if (extend) + puts("- check records and index references"); + else + puts("- check record links"); + } + + if (!(record= (byte*) my_malloc(info->s->base.pack_reclength,MYF(0)))) + { + _ma_check_print_error(param,"Not enough memory for record"); + DBUG_RETURN(-1); + } + records=del_blocks=0; + used=link_used=splits=del_length=0; + intern_record_checksum=param->glob_crc=0; + LINT_INIT(left_length); LINT_INIT(start_recpos); LINT_INIT(to); + got_error=error=0; + empty=info->s->pack.header_length; + + /* Check how to calculate checksum of rows */ + static_row_size=1; + if (info->s->data_file_type == COMPRESSED_RECORD) + { + for (field=0 ; field < info->s->base.fields ; field++) + { + if (info->s->rec[field].base_type == FIELD_BLOB || + info->s->rec[field].base_type == FIELD_VARCHAR) + { + static_row_size=0; + break; + } + } + } + + pos=my_b_tell(¶m->read_cache); + bzero((char*) key_checksum, info->s->base.keys * sizeof(key_checksum[0])); + while (pos < info->state->data_file_length) + { + if (*_ma_killed_ptr(param)) + goto err2; + switch (info->s->data_file_type) { + case STATIC_RECORD: + if (my_b_read(¶m->read_cache,(byte*) record, + info->s->base.pack_reclength)) + goto err; + start_recpos=pos; + pos+=info->s->base.pack_reclength; + splits++; + if (*record == '\0') + { + del_blocks++; + del_length+=info->s->base.pack_reclength; + continue; /* Record removed */ + } + param->glob_crc+= _ma_static_checksum(info,record); + used+=info->s->base.pack_reclength; + break; + case DYNAMIC_RECORD: + flag=block_info.second_read=0; + block_info.next_filepos=pos; + do + { + if (_ma_read_cache(¶m->read_cache,(byte*) block_info.header, + (start_block=block_info.next_filepos), + sizeof(block_info.header), + (flag ? 0 : READING_NEXT) | READING_HEADER)) + goto err; + if (start_block & (MARIA_DYN_ALIGN_SIZE-1)) + { + _ma_check_print_error(param,"Wrong aligned block at %s", + llstr(start_block,llbuff)); + goto err2; + } + b_type= _ma_get_block_info(&block_info,-1,start_block); + if (b_type & (BLOCK_DELETED | BLOCK_ERROR | BLOCK_SYNC_ERROR | + BLOCK_FATAL_ERROR)) + { + if (b_type & BLOCK_SYNC_ERROR) + { + if (flag) + { + _ma_check_print_error(param,"Unexpected byte: %d at link: %s", + (int) block_info.header[0], + llstr(start_block,llbuff)); + goto err2; + } + pos=block_info.filepos+block_info.block_len; + goto next; + } + if (b_type & BLOCK_DELETED) + { + if (block_info.block_len < info->s->base.min_block_length) + { + _ma_check_print_error(param, + "Deleted block with impossible length %lu at %s", + block_info.block_len,llstr(pos,llbuff)); + goto err2; + } + if ((block_info.next_filepos != HA_OFFSET_ERROR && + block_info.next_filepos >= info->state->data_file_length) || + (block_info.prev_filepos != HA_OFFSET_ERROR && + block_info.prev_filepos >= info->state->data_file_length)) + { + _ma_check_print_error(param,"Delete link points outside datafile at %s", + llstr(pos,llbuff)); + goto err2; + } + del_blocks++; + del_length+=block_info.block_len; + pos=block_info.filepos+block_info.block_len; + splits++; + goto next; + } + _ma_check_print_error(param,"Wrong bytesec: %d-%d-%d at linkstart: %s", + block_info.header[0],block_info.header[1], + block_info.header[2], + llstr(start_block,llbuff)); + goto err2; + } + if (info->state->data_file_length < block_info.filepos+ + block_info.block_len) + { + _ma_check_print_error(param, + "Recordlink that points outside datafile at %s", + llstr(pos,llbuff)); + got_error=1; + break; + } + splits++; + if (!flag++) /* First block */ + { + start_recpos=pos; + pos=block_info.filepos+block_info.block_len; + if (block_info.rec_len > (uint) info->s->base.max_pack_length) + { + _ma_check_print_error(param,"Found too long record (%lu) at %s", + (ulong) block_info.rec_len, + llstr(start_recpos,llbuff)); + got_error=1; + break; + } + if (info->s->base.blobs) + { + if (!(to= _ma_alloc_rec_buff(info, block_info.rec_len, + &info->rec_buff))) + { + _ma_check_print_error(param, + "Not enough memory (%lu) for blob at %s", + (ulong) block_info.rec_len, + llstr(start_recpos,llbuff)); + got_error=1; + break; + } + } + else + to= info->rec_buff; + left_length=block_info.rec_len; + } + if (left_length < block_info.data_len) + { + _ma_check_print_error(param,"Found too long record (%lu) at %s", + (ulong) block_info.data_len, + llstr(start_recpos,llbuff)); + got_error=1; + break; + } + if (_ma_read_cache(¶m->read_cache,(byte*) to,block_info.filepos, + (uint) block_info.data_len, + flag == 1 ? READING_NEXT : 0)) + goto err; + to+=block_info.data_len; + link_used+= block_info.filepos-start_block; + used+= block_info.filepos - start_block + block_info.data_len; + empty+=block_info.block_len-block_info.data_len; + left_length-=block_info.data_len; + if (left_length) + { + if (b_type & BLOCK_LAST) + { + _ma_check_print_error(param, + "Wrong record length %s of %s at %s", + llstr(block_info.rec_len-left_length,llbuff), + llstr(block_info.rec_len, llbuff2), + llstr(start_recpos,llbuff3)); + got_error=1; + break; + } + if (info->state->data_file_length < block_info.next_filepos) + { + _ma_check_print_error(param, + "Found next-recordlink that points outside datafile at %s", + llstr(block_info.filepos,llbuff)); + got_error=1; + break; + } + } + } while (left_length); + if (! got_error) + { + if (_ma_rec_unpack(info,record,info->rec_buff,block_info.rec_len) == + MY_FILE_ERROR) + { + _ma_check_print_error(param,"Found wrong record at %s", + llstr(start_recpos,llbuff)); + got_error=1; + } + else + { + info->checksum=_ma_checksum(info,record); + if (param->testflag & (T_EXTEND | T_MEDIUM | T_VERBOSE)) + { + if (_ma_rec_check(info,record, info->rec_buff,block_info.rec_len, + test(info->s->calc_checksum))) + { + _ma_check_print_error(param,"Found wrong packed record at %s", + llstr(start_recpos,llbuff)); + got_error=1; + } + } + if (!got_error) + param->glob_crc+= info->checksum; + } + } + else if (!flag) + pos=block_info.filepos+block_info.block_len; + break; + case COMPRESSED_RECORD: + if (_ma_read_cache(¶m->read_cache,(byte*) block_info.header, pos, + info->s->pack.ref_length, READING_NEXT)) + goto err; + start_recpos=pos; + splits++; + VOID(_ma_pack_get_block_info(info,&block_info, -1, start_recpos)); + pos=block_info.filepos+block_info.rec_len; + if (block_info.rec_len < (uint) info->s->min_pack_length || + block_info.rec_len > (uint) info->s->max_pack_length) + { + _ma_check_print_error(param, + "Found block with wrong recordlength: %d at %s", + block_info.rec_len, llstr(start_recpos,llbuff)); + got_error=1; + break; + } + if (_ma_read_cache(¶m->read_cache,(byte*) info->rec_buff, + block_info.filepos, block_info.rec_len, READING_NEXT)) + goto err; + if (_ma_pack_rec_unpack(info,record,info->rec_buff,block_info.rec_len)) + { + _ma_check_print_error(param,"Found wrong record at %s", + llstr(start_recpos,llbuff)); + got_error=1; + } + if (static_row_size) + param->glob_crc+= _ma_static_checksum(info,record); + else + param->glob_crc+= _ma_checksum(info,record); + link_used+= (block_info.filepos - start_recpos); + used+= (pos-start_recpos); + } /* switch */ + if (! got_error) + { + intern_record_checksum+=(ha_checksum) start_recpos; + records++; + if (param->testflag & T_WRITE_LOOP && records % WRITE_COUNT == 0) + { + printf("%s\r", llstr(records,llbuff)); VOID(fflush(stdout)); + } + + /* Check if keys match the record */ + + for (key=0,keyinfo= info->s->keyinfo; key < info->s->base.keys; + key++,keyinfo++) + { + if (maria_is_key_active(info->s->state.key_map, key)) + { + if(!(keyinfo->flag & HA_FULLTEXT)) + { + uint key_length= _ma_make_key(info,key,info->lastkey,record, + start_recpos); + if (extend) + { + /* We don't need to lock the key tree here as we don't allow + concurrent threads when running mariachk + */ + int search_result= +#ifdef HAVE_RTREE_KEYS + (keyinfo->flag & HA_SPATIAL) ? + maria_rtree_find_first(info, key, info->lastkey, key_length, + SEARCH_SAME) : +#endif + _ma_search(info,keyinfo,info->lastkey,key_length, + SEARCH_SAME, info->s->state.key_root[key]); + if (search_result) + { + _ma_check_print_error(param,"Record at: %10s Can't find key for index: %2d", + llstr(start_recpos,llbuff),key+1); + if (error++ > MAXERR || !(param->testflag & T_VERBOSE)) + goto err2; + } + } + else + key_checksum[key]+=maria_byte_checksum((byte*) info->lastkey, + key_length); + } + } + } + } + else + { + got_error=0; + if (error++ > MAXERR || !(param->testflag & T_VERBOSE)) + goto err2; + } + next:; /* Next record */ + } + if (param->testflag & T_WRITE_LOOP) + { + VOID(fputs(" \r",stdout)); VOID(fflush(stdout)); + } + if (records != info->state->records) + { + _ma_check_print_error(param,"Record-count is not ok; is %-10s Should be: %s", + llstr(records,llbuff), llstr(info->state->records,llbuff2)); + error=1; + } + else if (param->record_checksum && + param->record_checksum != intern_record_checksum) + { + _ma_check_print_error(param, + "Keypointers and record positions doesn't match"); + error=1; + } + else if (param->glob_crc != info->state->checksum && + (info->s->options & + (HA_OPTION_CHECKSUM | HA_OPTION_COMPRESS_RECORD))) + { + _ma_check_print_warning(param, + "Record checksum is not the same as checksum stored in the index file\n"); + error=1; + } + else if (!extend) + { + for (key=0 ; key < info->s->base.keys; key++) + { + if (key_checksum[key] != param->key_crc[key] && + !(info->s->keyinfo[key].flag & (HA_FULLTEXT | HA_SPATIAL))) + { + _ma_check_print_error(param,"Checksum for key: %2d doesn't match checksum for records", + key+1); + error=1; + } + } + } + + if (del_length != info->state->empty) + { + _ma_check_print_warning(param, + "Found %s deleted space. Should be %s", + llstr(del_length,llbuff2), + llstr(info->state->empty,llbuff)); + } + if (used+empty+del_length != info->state->data_file_length) + { + _ma_check_print_warning(param, + "Found %s record-data and %s unused data and %s deleted-data", + llstr(used,llbuff),llstr(empty,llbuff2), + llstr(del_length,llbuff3)); + _ma_check_print_warning(param, + "Total %s, Should be: %s", + llstr((used+empty+del_length),llbuff), + llstr(info->state->data_file_length,llbuff2)); + } + if (del_blocks != info->state->del) + { + _ma_check_print_warning(param, + "Found %10s deleted blocks Should be: %s", + llstr(del_blocks,llbuff), + llstr(info->state->del,llbuff2)); + } + if (splits != info->s->state.split) + { + _ma_check_print_warning(param, + "Found %10s parts Should be: %s parts", + llstr(splits,llbuff), + llstr(info->s->state.split,llbuff2)); + } + if (param->testflag & T_INFO) + { + if (param->warning_printed || param->error_printed) + puts(""); + if (used != 0 && ! param->error_printed) + { + printf("Records:%18s M.recordlength:%9lu Packed:%14.0f%%\n", + llstr(records,llbuff), (long)((used-link_used)/records), + (info->s->base.blobs ? 0.0 : + (ulonglong2double((ulonglong) info->s->base.reclength*records)- + my_off_t2double(used))/ + ulonglong2double((ulonglong) info->s->base.reclength*records)*100.0)); + printf("Recordspace used:%9.0f%% Empty space:%12d%% Blocks/Record: %6.2f\n", + (ulonglong2double(used-link_used)/ulonglong2double(used-link_used+empty)*100.0), + (!records ? 100 : (int) (ulonglong2double(del_length+empty)/ + my_off_t2double(used)*100.0)), + ulonglong2double(splits - del_blocks) / records); + } + printf("Record blocks:%12s Delete blocks:%10s\n", + llstr(splits-del_blocks,llbuff),llstr(del_blocks,llbuff2)); + printf("Record data: %12s Deleted data: %10s\n", + llstr(used-link_used,llbuff),llstr(del_length,llbuff2)); + printf("Lost space: %12s Linkdata: %10s\n", + llstr(empty,llbuff),llstr(link_used,llbuff2)); + } + my_free((gptr) record,MYF(0)); + DBUG_RETURN (error); + err: + _ma_check_print_error(param,"got error: %d when reading datafile at record: %s",my_errno, llstr(records,llbuff)); + err2: + my_free((gptr) record,MYF(0)); + param->testflag|=T_RETRY_WITHOUT_QUICK; + DBUG_RETURN(1); +} /* maria_chk_data_link */ + + + /* Recover old table by reading each record and writing all keys */ + /* Save new datafile-name in temp_filename */ + +int maria_repair(HA_CHECK *param, register MARIA_HA *info, + my_string name, int rep_quick) +{ + int error,got_error; + uint i; + ha_rows start_records,new_header_length; + my_off_t del; + File new_file; + MARIA_SHARE *share=info->s; + char llbuff[22],llbuff2[22]; + MARIA_SORT_INFO sort_info; + MARIA_SORT_PARAM sort_param; + DBUG_ENTER("maria_repair"); + + bzero((char *)&sort_info, sizeof(sort_info)); + bzero((char *)&sort_param, sizeof(sort_param)); + start_records=info->state->records; + new_header_length=(param->testflag & T_UNPACK) ? 0L : + share->pack.header_length; + got_error=1; + new_file= -1; + sort_param.sort_info=&sort_info; + + if (!(param->testflag & T_SILENT)) + { + printf("- recovering (with keycache) MARIA-table '%s'\n",name); + printf("Data records: %s\n", llstr(info->state->records,llbuff)); + } + param->testflag|=T_REP; /* for easy checking */ + + if (info->s->options & (HA_OPTION_CHECKSUM | HA_OPTION_COMPRESS_RECORD)) + param->testflag|=T_CALC_CHECKSUM; + + if (!param->using_global_keycache) + VOID(init_key_cache(maria_key_cache, param->key_cache_block_size, + param->use_buffers, 0, 0)); + + if (init_io_cache(¶m->read_cache,info->dfile, + (uint) param->read_buffer_length, + READ_CACHE,share->pack.header_length,1,MYF(MY_WME))) + { + bzero(&info->rec_cache,sizeof(info->rec_cache)); + goto err; + } + if (!rep_quick) + if (init_io_cache(&info->rec_cache,-1,(uint) param->write_buffer_length, + WRITE_CACHE, new_header_length, 1, + MYF(MY_WME | MY_WAIT_IF_FULL))) + goto err; + info->opt_flag|=WRITE_CACHE_USED; + if (!(sort_param.record=(byte*) my_malloc((uint) share->base.pack_reclength, + MYF(0))) || + !_ma_alloc_rec_buff(info, -1, &sort_param.rec_buff)) + { + _ma_check_print_error(param, "Not enough memory for extra record"); + goto err; + } + + if (!rep_quick) + { + /* Get real path for data file */ + if ((new_file=my_raid_create(fn_format(param->temp_filename, + share->data_file_name, "", + DATA_TMP_EXT, 2+4), + 0,param->tmpfile_createflag, + share->base.raid_type, + share->base.raid_chunks, + share->base.raid_chunksize, + MYF(0))) < 0) + { + _ma_check_print_error(param,"Can't create new tempfile: '%s'", + param->temp_filename); + goto err; + } + if (maria_filecopy(param,new_file,info->dfile,0L,new_header_length, + "datafile-header")) + goto err; + info->s->state.dellink= HA_OFFSET_ERROR; + info->rec_cache.file=new_file; + if (param->testflag & T_UNPACK) + { + share->options&= ~HA_OPTION_COMPRESS_RECORD; + mi_int2store(share->state.header.options,share->options); + } + } + sort_info.info=info; + sort_info.param = param; + sort_param.read_cache=param->read_cache; + sort_param.pos=sort_param.max_pos=share->pack.header_length; + sort_param.filepos=new_header_length; + param->read_cache.end_of_file=sort_info.filelength= + my_seek(info->dfile,0L,MY_SEEK_END,MYF(0)); + sort_info.dupp=0; + sort_param.fix_datafile= (my_bool) (! rep_quick); + sort_param.master=1; + sort_info.max_records= ~(ha_rows) 0; + + set_data_file_type(&sort_info, share); + del=info->state->del; + info->state->records=info->state->del=share->state.split=0; + info->state->empty=0; + param->glob_crc=0; + if (param->testflag & T_CALC_CHECKSUM) + param->calc_checksum=1; + + info->update= (short) (HA_STATE_CHANGED | HA_STATE_ROW_CHANGED); + + /* + Clear all keys. Note that all key blocks allocated until now remain + "dead" parts of the key file. (Bug #4692) + */ + for (i=0 ; i < info->s->base.keys ; i++) + share->state.key_root[i]= HA_OFFSET_ERROR; + + /* Drop the delete chain. */ + for (i=0 ; i < share->state.header.max_block_size ; i++) + share->state.key_del[i]= HA_OFFSET_ERROR; + + /* + If requested, activate (enable) all keys in key_map. In this case, + all indexes will be (re-)built. + */ + if (param->testflag & T_CREATE_MISSING_KEYS) + maria_set_all_keys_active(share->state.key_map, share->base.keys); + + info->state->key_file_length=share->base.keystart; + + maria_lock_memory(param); /* Everything is alloced */ + + /* Re-create all keys, which are set in key_map. */ + while (!(error=sort_get_next_record(&sort_param))) + { + if (writekeys(param,info,(byte*)sort_param.record,sort_param.filepos)) + { + if (my_errno != HA_ERR_FOUND_DUPP_KEY) + goto err; + DBUG_DUMP("record",(byte*) sort_param.record,share->base.pack_reclength); + _ma_check_print_info(param,"Duplicate key %2d for record at %10s against new record at %10s", + info->errkey+1, + llstr(sort_param.start_recpos,llbuff), + llstr(info->dupp_key_pos,llbuff2)); + if (param->testflag & T_VERBOSE) + { + VOID(_ma_make_key(info,(uint) info->errkey,info->lastkey, + sort_param.record,0L)); + _ma_print_key(stdout,share->keyinfo[info->errkey].seg,info->lastkey, + USE_WHOLE_KEY); + } + sort_info.dupp++; + if ((param->testflag & (T_FORCE_UNIQUENESS|T_QUICK)) == T_QUICK) + { + param->testflag|=T_RETRY_WITHOUT_QUICK; + param->error_printed=1; + goto err; + } + continue; + } + if (_ma_sort_write_record(&sort_param)) + goto err; + } + if (error > 0 || maria_write_data_suffix(&sort_info, (my_bool)!rep_quick) || + flush_io_cache(&info->rec_cache) || param->read_cache.error < 0) + goto err; + + if (param->testflag & T_WRITE_LOOP) + { + VOID(fputs(" \r",stdout)); VOID(fflush(stdout)); + } + if (my_chsize(share->kfile,info->state->key_file_length,0,MYF(0))) + { + _ma_check_print_warning(param, + "Can't change size of indexfile, error: %d", + my_errno); + goto err; + } + + if (rep_quick && del+sort_info.dupp != info->state->del) + { + _ma_check_print_error(param,"Couldn't fix table with quick recovery: Found wrong number of deleted records"); + _ma_check_print_error(param,"Run recovery again without -q"); + got_error=1; + param->retry_repair=1; + param->testflag|=T_RETRY_WITHOUT_QUICK; + goto err; + } + if (param->testflag & T_SAFE_REPAIR) + { + /* Don't repair if we loosed more than one row */ + if (info->state->records+1 < start_records) + { + info->state->records=start_records; + got_error=1; + goto err; + } + } + + if (!rep_quick) + { + my_close(info->dfile,MYF(0)); + info->dfile=new_file; + info->state->data_file_length=sort_param.filepos; + share->state.version=(ulong) time((time_t*) 0); /* Force reopen */ + } + else + { + info->state->data_file_length=sort_param.max_pos; + } + if (param->testflag & T_CALC_CHECKSUM) + info->state->checksum=param->glob_crc; + + if (!(param->testflag & T_SILENT)) + { + if (start_records != info->state->records) + printf("Data records: %s\n", llstr(info->state->records,llbuff)); + if (sort_info.dupp) + _ma_check_print_warning(param, + "%s records have been removed", + llstr(sort_info.dupp,llbuff)); + } + + got_error=0; + /* If invoked by external program that uses thr_lock */ + if (&share->state.state != info->state) + memcpy( &share->state.state, info->state, sizeof(*info->state)); + +err: + if (!got_error) + { + /* Replace the actual file with the temporary file */ + if (new_file >= 0) + { + my_close(new_file,MYF(0)); + info->dfile=new_file= -1; + if (maria_change_to_newfile(share->data_file_name,MARIA_NAME_DEXT, + DATA_TMP_EXT, share->base.raid_chunks, + (param->testflag & T_BACKUP_DATA ? + MYF(MY_REDEL_MAKE_BACKUP): MYF(0))) || + _ma_open_datafile(info,share,-1)) + got_error=1; + } + } + if (got_error) + { + if (! param->error_printed) + _ma_check_print_error(param,"%d for record at pos %s",my_errno, + llstr(sort_param.start_recpos,llbuff)); + if (new_file >= 0) + { + VOID(my_close(new_file,MYF(0))); + VOID(my_raid_delete(param->temp_filename,info->s->base.raid_chunks, + MYF(MY_WME))); + info->rec_cache.file=-1; /* don't flush data to new_file, it's closed */ + } + maria_mark_crashed_on_repair(info); + } + my_free(_ma_get_rec_buff_ptr(info, sort_param.rec_buff), + MYF(MY_ALLOW_ZERO_PTR)); + my_free(sort_param.record,MYF(MY_ALLOW_ZERO_PTR)); + my_free(sort_info.buff,MYF(MY_ALLOW_ZERO_PTR)); + VOID(end_io_cache(¶m->read_cache)); + info->opt_flag&= ~(READ_CACHE_USED | WRITE_CACHE_USED); + VOID(end_io_cache(&info->rec_cache)); + got_error|=_ma_flush_blocks(param, share->key_cache, share->kfile); + if (!got_error && param->testflag & T_UNPACK) + { + share->state.header.options[0]&= (uchar) ~HA_OPTION_COMPRESS_RECORD; + share->pack.header_length=0; + share->data_file_type=sort_info.new_data_file_type; + } + share->state.changed|= (STATE_NOT_OPTIMIZED_KEYS | STATE_NOT_SORTED_PAGES | + STATE_NOT_ANALYZED); + DBUG_RETURN(got_error); +} + + +/* Uppate keyfile when doing repair */ + +static int writekeys(HA_CHECK *param, register MARIA_HA *info, byte *buff, + my_off_t filepos) +{ + register uint i; + uchar *key; + DBUG_ENTER("writekeys"); + + key=info->lastkey+info->s->base.max_key_length; + for (i=0 ; i < info->s->base.keys ; i++) + { + if (maria_is_key_active(info->s->state.key_map, i)) + { + if (info->s->keyinfo[i].flag & HA_FULLTEXT ) + { + if (_ma_ft_add(info,i,(char*) key,buff,filepos)) + goto err; + } +#ifdef HAVE_SPATIAL + else if (info->s->keyinfo[i].flag & HA_SPATIAL) + { + uint key_length= _ma_make_key(info,i,key,buff,filepos); + if (maria_rtree_insert(info, i, key, key_length)) + goto err; + } +#endif /*HAVE_SPATIAL*/ + else + { + uint key_length= _ma_make_key(info,i,key,buff,filepos); + if (_ma_ck_write(info,i,key,key_length)) + goto err; + } + } + } + DBUG_RETURN(0); + + err: + if (my_errno == HA_ERR_FOUND_DUPP_KEY) + { + info->errkey=(int) i; /* This key was found */ + while ( i-- > 0 ) + { + if (maria_is_key_active(info->s->state.key_map, i)) + { + if (info->s->keyinfo[i].flag & HA_FULLTEXT) + { + if (_ma_ft_del(info,i,(char*) key,buff,filepos)) + break; + } + else + { + uint key_length= _ma_make_key(info,i,key,buff,filepos); + if (_ma_ck_delete(info,i,key,key_length)) + break; + } + } + } + } + /* Remove checksum that was added to glob_crc in sort_get_next_record */ + if (param->calc_checksum) + param->glob_crc-= info->checksum; + DBUG_PRINT("error",("errno: %d",my_errno)); + DBUG_RETURN(-1); +} /* writekeys */ + + + /* Change all key-pointers that points to a records */ + +int maria_movepoint(register MARIA_HA *info, byte *record, my_off_t oldpos, + my_off_t newpos, uint prot_key) +{ + register uint i; + uchar *key; + uint key_length; + DBUG_ENTER("maria_movepoint"); + + key=info->lastkey+info->s->base.max_key_length; + for (i=0 ; i < info->s->base.keys; i++) + { + if (i != prot_key && maria_is_key_active(info->s->state.key_map, i)) + { + key_length= _ma_make_key(info,i,key,record,oldpos); + if (info->s->keyinfo[i].flag & HA_NOSAME) + { /* Change pointer direct */ + uint nod_flag; + MARIA_KEYDEF *keyinfo; + keyinfo=info->s->keyinfo+i; + if (_ma_search(info,keyinfo,key,USE_WHOLE_KEY, + (uint) (SEARCH_SAME | SEARCH_SAVE_BUFF), + info->s->state.key_root[i])) + DBUG_RETURN(-1); + nod_flag=_ma_test_if_nod(info->buff); + _ma_dpointer(info,info->int_keypos-nod_flag- + info->s->rec_reflength,newpos); + if (_ma_write_keypage(info,keyinfo,info->last_keypage, + DFLT_INIT_HITS,info->buff)) + DBUG_RETURN(-1); + } + else + { /* Change old key to new */ + if (_ma_ck_delete(info,i,key,key_length)) + DBUG_RETURN(-1); + key_length= _ma_make_key(info,i,key,record,newpos); + if (_ma_ck_write(info,i,key,key_length)) + DBUG_RETURN(-1); + } + } + } + DBUG_RETURN(0); +} /* maria_movepoint */ + + + /* Tell system that we want all memory for our cache */ + +void maria_lock_memory(HA_CHECK *param __attribute__((unused))) +{ +#ifdef SUN_OS /* Key-cacheing thrases on sun 4.1 */ + if (param->opt_maria_lock_memory) + { + int success = mlockall(MCL_CURRENT); /* or plock(DATLOCK); */ + if (geteuid() == 0 && success != 0) + _ma_check_print_warning(param, + "Failed to lock memory. errno %d",my_errno); + } +#endif +} /* maria_lock_memory */ + + + /* Flush all changed blocks to disk */ + +int _ma_flush_blocks(HA_CHECK *param, KEY_CACHE *key_cache, File file) +{ + if (flush_key_blocks(key_cache, file, FLUSH_RELEASE)) + { + _ma_check_print_error(param,"%d when trying to write bufferts",my_errno); + return(1); + } + if (!param->using_global_keycache) + end_key_cache(key_cache,1); + return 0; +} /* _ma_flush_blocks */ + + + /* Sort index for more efficent reads */ + +int maria_sort_index(HA_CHECK *param, register MARIA_HA *info, my_string name) +{ + reg2 uint key; + reg1 MARIA_KEYDEF *keyinfo; + File new_file; + my_off_t index_pos[HA_MAX_POSSIBLE_KEY]; + uint r_locks,w_locks; + int old_lock; + MARIA_SHARE *share=info->s; + MARIA_STATE_INFO old_state; + DBUG_ENTER("maria_sort_index"); + + if (!(param->testflag & T_SILENT)) + printf("- Sorting index for MARIA-table '%s'\n",name); + + /* Get real path for index file */ + fn_format(param->temp_filename,name,"", MARIA_NAME_IEXT,2+4+32); + if ((new_file=my_create(fn_format(param->temp_filename,param->temp_filename, + "", INDEX_TMP_EXT,2+4), + 0,param->tmpfile_createflag,MYF(0))) <= 0) + { + _ma_check_print_error(param,"Can't create new tempfile: '%s'", + param->temp_filename); + DBUG_RETURN(-1); + } + if (maria_filecopy(param, new_file,share->kfile,0L, + (ulong) share->base.keystart, "headerblock")) + goto err; + + param->new_file_pos=share->base.keystart; + for (key= 0,keyinfo= &share->keyinfo[0]; key < share->base.keys ; + key++,keyinfo++) + { + if (! maria_is_key_active(info->s->state.key_map, key)) + continue; + + if (share->state.key_root[key] != HA_OFFSET_ERROR) + { + index_pos[key]=param->new_file_pos; /* Write first block here */ + if (sort_one_index(param,info,keyinfo,share->state.key_root[key], + new_file)) + goto err; + } + else + index_pos[key]= HA_OFFSET_ERROR; /* No blocks */ + } + + /* Flush key cache for this file if we are calling this outside mariachk */ + flush_key_blocks(share->key_cache,share->kfile, FLUSH_IGNORE_CHANGED); + + share->state.version=(ulong) time((time_t*) 0); + old_state= share->state; /* save state if not stored */ + r_locks= share->r_locks; + w_locks= share->w_locks; + old_lock= info->lock_type; + + /* Put same locks as old file */ + share->r_locks= share->w_locks= share->tot_locks= 0; + (void) _ma_writeinfo(info,WRITEINFO_UPDATE_KEYFILE); + VOID(my_close(share->kfile,MYF(MY_WME))); + share->kfile = -1; + VOID(my_close(new_file,MYF(MY_WME))); + if (maria_change_to_newfile(share->index_file_name,MARIA_NAME_IEXT,INDEX_TMP_EXT,0, + MYF(0)) || + _ma_open_keyfile(share)) + goto err2; + info->lock_type= F_UNLCK; /* Force maria_readinfo to lock */ + _ma_readinfo(info,F_WRLCK,0); /* Will lock the table */ + info->lock_type= old_lock; + share->r_locks= r_locks; + share->w_locks= w_locks; + share->tot_locks= r_locks+w_locks; + share->state= old_state; /* Restore old state */ + + info->state->key_file_length=param->new_file_pos; + info->update= (short) (HA_STATE_CHANGED | HA_STATE_ROW_CHANGED); + for (key=0 ; key < info->s->base.keys ; key++) + info->s->state.key_root[key]=index_pos[key]; + for (key=0 ; key < info->s->state.header.max_block_size ; key++) + info->s->state.key_del[key]= HA_OFFSET_ERROR; + + info->s->state.changed&= ~STATE_NOT_SORTED_PAGES; + DBUG_RETURN(0); + +err: + VOID(my_close(new_file,MYF(MY_WME))); +err2: + VOID(my_delete(param->temp_filename,MYF(MY_WME))); + DBUG_RETURN(-1); +} /* maria_sort_index */ + + + /* Sort records recursive using one index */ + +static int sort_one_index(HA_CHECK *param, MARIA_HA *info, MARIA_KEYDEF *keyinfo, + my_off_t pagepos, File new_file) +{ + uint length,nod_flag,used_length, key_length; + uchar *buff,*keypos,*endpos; + uchar key[HA_MAX_POSSIBLE_KEY_BUFF]; + my_off_t new_page_pos,next_page; + char llbuff[22]; + DBUG_ENTER("sort_one_index"); + + new_page_pos=param->new_file_pos; + param->new_file_pos+=keyinfo->block_length; + + if (!(buff=(uchar*) my_alloca((uint) keyinfo->block_length))) + { + _ma_check_print_error(param,"Not enough memory for key block"); + DBUG_RETURN(-1); + } + if (!_ma_fetch_keypage(info,keyinfo,pagepos,DFLT_INIT_HITS,buff,0)) + { + _ma_check_print_error(param,"Can't read key block from filepos: %s", + llstr(pagepos,llbuff)); + goto err; + } + if ((nod_flag=_ma_test_if_nod(buff)) || keyinfo->flag & HA_FULLTEXT) + { + used_length=maria_getint(buff); + keypos=buff+2+nod_flag; + endpos=buff+used_length; + for ( ;; ) + { + if (nod_flag) + { + next_page= _ma_kpos(nod_flag,keypos); + /* Save new pos */ + _ma_kpointer(info,keypos-nod_flag,param->new_file_pos); + if (sort_one_index(param,info,keyinfo,next_page,new_file)) + { + DBUG_PRINT("error", + ("From page: %ld, keyoffset: %lu used_length: %d", + (ulong) pagepos, (ulong) (keypos - buff), + (int) used_length)); + DBUG_DUMP("buff",(byte*) buff,used_length); + goto err; + } + } + if (keypos >= endpos || + (key_length=(*keyinfo->get_key)(keyinfo,nod_flag,&keypos,key)) == 0) + break; + DBUG_ASSERT(keypos <= endpos); + if (keyinfo->flag & HA_FULLTEXT) + { + uint off; + int subkeys; + get_key_full_length_rdonly(off, key); + subkeys=ft_sintXkorr(key+off); + if (subkeys < 0) + { + next_page= _ma_dpos(info,0,key+key_length); + _ma_dpointer(info,keypos-nod_flag-info->s->rec_reflength, + param->new_file_pos); /* Save new pos */ + if (sort_one_index(param,info,&info->s->ft2_keyinfo, + next_page,new_file)) + goto err; + } + } + } + } + + /* Fill block with zero and write it to the new index file */ + length=maria_getint(buff); + bzero((byte*) buff+length,keyinfo->block_length-length); + if (my_pwrite(new_file,(byte*) buff,(uint) keyinfo->block_length, + new_page_pos,MYF(MY_NABP | MY_WAIT_IF_FULL))) + { + _ma_check_print_error(param,"Can't write indexblock, error: %d",my_errno); + goto err; + } + my_afree((gptr) buff); + DBUG_RETURN(0); +err: + my_afree((gptr) buff); + DBUG_RETURN(1); +} /* sort_one_index */ + + + /* + Let temporary file replace old file. + This assumes that the new file was created in the same + directory as given by realpath(filename). + This will ensure that any symlinks that are used will still work. + Copy stats from old file to new file, deletes orignal and + changes new file name to old file name + */ + +int maria_change_to_newfile(const char * filename, const char * old_ext, + const char * new_ext, + uint raid_chunks __attribute__((unused)), + myf MyFlags) +{ + char old_filename[FN_REFLEN],new_filename[FN_REFLEN]; +#ifdef USE_RAID + if (raid_chunks) + return my_raid_redel(fn_format(old_filename,filename,"",old_ext,2+4), + fn_format(new_filename,filename,"",new_ext,2+4), + raid_chunks, + MYF(MY_WME | MY_LINK_WARNING | MyFlags)); +#endif + /* Get real path to filename */ + (void) fn_format(old_filename,filename,"",old_ext,2+4+32); + return my_redel(old_filename, + fn_format(new_filename,old_filename,"",new_ext,2+4), + MYF(MY_WME | MY_LINK_WARNING | MyFlags)); +} /* maria_change_to_newfile */ + + + /* Locks a whole file */ + /* Gives an error-message if file can't be locked */ + +int maria_lock_file(HA_CHECK *param, File file, my_off_t start, int lock_type, + const char *filetype, const char *filename) +{ + if (my_lock(file,lock_type,start,F_TO_EOF, + param->testflag & T_WAIT_FOREVER ? MYF(MY_SEEK_NOT_DONE) : + MYF(MY_SEEK_NOT_DONE | MY_DONT_WAIT))) + { + _ma_check_print_error(param," %d when locking %s '%s'",my_errno,filetype,filename); + param->error_printed=2; /* Don't give that data is crashed */ + return 1; + } + return 0; +} /* maria_lock_file */ + + + /* Copy a block between two files */ + +int maria_filecopy(HA_CHECK *param, File to,File from,my_off_t start, + my_off_t length, const char *type) +{ + char tmp_buff[IO_SIZE],*buff; + ulong buff_length; + DBUG_ENTER("maria_filecopy"); + + buff_length=(ulong) min(param->write_buffer_length,length); + if (!(buff=my_malloc(buff_length,MYF(0)))) + { + buff=tmp_buff; buff_length=IO_SIZE; + } + + VOID(my_seek(from,start,MY_SEEK_SET,MYF(0))); + while (length > buff_length) + { + if (my_read(from,(byte*) buff,buff_length,MYF(MY_NABP)) || + my_write(to,(byte*) buff,buff_length,param->myf_rw)) + goto err; + length-= buff_length; + } + if (my_read(from,(byte*) buff,(uint) length,MYF(MY_NABP)) || + my_write(to,(byte*) buff,(uint) length,param->myf_rw)) + goto err; + if (buff != tmp_buff) + my_free(buff,MYF(0)); + DBUG_RETURN(0); +err: + if (buff != tmp_buff) + my_free(buff,MYF(0)); + _ma_check_print_error(param,"Can't copy %s to tempfile, error %d", + type,my_errno); + DBUG_RETURN(1); +} + + +/* + Repair table or given index using sorting + + SYNOPSIS + maria_repair_by_sort() + param Repair parameters + info MARIA handler to repair + name Name of table (for warnings) + rep_quick set to <> 0 if we should not change data file + + RESULT + 0 ok + <>0 Error +*/ + +int maria_repair_by_sort(HA_CHECK *param, register MARIA_HA *info, + const char * name, int rep_quick) +{ + int got_error; + uint i; + ulong length; + ha_rows start_records; + my_off_t new_header_length,del; + File new_file; + MARIA_SORT_PARAM sort_param; + MARIA_SHARE *share=info->s; + HA_KEYSEG *keyseg; + ulong *rec_per_key_part; + char llbuff[22]; + MARIA_SORT_INFO sort_info; + ulonglong key_map=share->state.key_map; + DBUG_ENTER("maria_repair_by_sort"); + + start_records=info->state->records; + got_error=1; + new_file= -1; + new_header_length=(param->testflag & T_UNPACK) ? 0 : + share->pack.header_length; + if (!(param->testflag & T_SILENT)) + { + printf("- recovering (with sort) MARIA-table '%s'\n",name); + printf("Data records: %s\n", llstr(start_records,llbuff)); + } + param->testflag|=T_REP; /* for easy checking */ + + if (info->s->options & (HA_OPTION_CHECKSUM | HA_OPTION_COMPRESS_RECORD)) + param->testflag|=T_CALC_CHECKSUM; + + bzero((char*)&sort_info,sizeof(sort_info)); + bzero((char *)&sort_param, sizeof(sort_param)); + if (!(sort_info.key_block= + alloc_key_blocks(param, + (uint) param->sort_key_blocks, + share->base.max_key_block_length)) + || init_io_cache(¶m->read_cache,info->dfile, + (uint) param->read_buffer_length, + READ_CACHE,share->pack.header_length,1,MYF(MY_WME)) || + (! rep_quick && + init_io_cache(&info->rec_cache,info->dfile, + (uint) param->write_buffer_length, + WRITE_CACHE,new_header_length,1, + MYF(MY_WME | MY_WAIT_IF_FULL) & param->myf_rw))) + goto err; + sort_info.key_block_end=sort_info.key_block+param->sort_key_blocks; + info->opt_flag|=WRITE_CACHE_USED; + info->rec_cache.file=info->dfile; /* for sort_delete_record */ + + if (!(sort_param.record=(byte*) my_malloc((uint) share->base.pack_reclength, + MYF(0))) || + !_ma_alloc_rec_buff(info, -1, &sort_param.rec_buff)) + { + _ma_check_print_error(param, "Not enough memory for extra record"); + goto err; + } + if (!rep_quick) + { + /* Get real path for data file */ + if ((new_file=my_raid_create(fn_format(param->temp_filename, + share->data_file_name, "", + DATA_TMP_EXT, 2+4), + 0,param->tmpfile_createflag, + share->base.raid_type, + share->base.raid_chunks, + share->base.raid_chunksize, + MYF(0))) < 0) + { + _ma_check_print_error(param,"Can't create new tempfile: '%s'", + param->temp_filename); + goto err; + } + if (maria_filecopy(param, new_file,info->dfile,0L,new_header_length, + "datafile-header")) + goto err; + if (param->testflag & T_UNPACK) + { + share->options&= ~HA_OPTION_COMPRESS_RECORD; + mi_int2store(share->state.header.options,share->options); + } + share->state.dellink= HA_OFFSET_ERROR; + info->rec_cache.file=new_file; + } + + info->update= (short) (HA_STATE_CHANGED | HA_STATE_ROW_CHANGED); + if (!(param->testflag & T_CREATE_MISSING_KEYS)) + { + /* + Flush key cache for this file if we are calling this outside + mariachk + */ + flush_key_blocks(share->key_cache,share->kfile, FLUSH_IGNORE_CHANGED); + /* Clear the pointers to the given rows */ + for (i=0 ; i < share->base.keys ; i++) + share->state.key_root[i]= HA_OFFSET_ERROR; + for (i=0 ; i < share->state.header.max_block_size ; i++) + share->state.key_del[i]= HA_OFFSET_ERROR; + info->state->key_file_length=share->base.keystart; + } + else + { + if (flush_key_blocks(share->key_cache,share->kfile, FLUSH_FORCE_WRITE)) + goto err; + key_map= ~key_map; /* Create the missing keys */ + } + + sort_info.info=info; + sort_info.param = param; + + set_data_file_type(&sort_info, share); + sort_param.filepos=new_header_length; + sort_info.dupp=0; + sort_info.buff=0; + param->read_cache.end_of_file=sort_info.filelength= + my_seek(param->read_cache.file,0L,MY_SEEK_END,MYF(0)); + + sort_param.wordlist=NULL; + + if (share->data_file_type == DYNAMIC_RECORD) + length=max(share->base.min_pack_length+1,share->base.min_block_length); + else if (share->data_file_type == COMPRESSED_RECORD) + length=share->base.min_block_length; + else + length=share->base.pack_reclength; + sort_info.max_records= + ((param->testflag & T_CREATE_MISSING_KEYS) ? info->state->records : + (ha_rows) (sort_info.filelength/length+1)); + sort_param.key_cmp=sort_key_cmp; + sort_param.lock_in_memory=maria_lock_memory; + sort_param.tmpdir=param->tmpdir; + sort_param.sort_info=&sort_info; + sort_param.fix_datafile= (my_bool) (! rep_quick); + sort_param.master =1; + + del=info->state->del; + param->glob_crc=0; + if (param->testflag & T_CALC_CHECKSUM) + param->calc_checksum=1; + + rec_per_key_part= param->rec_per_key_part; + for (sort_param.key=0 ; sort_param.key < share->base.keys ; + rec_per_key_part+=sort_param.keyinfo->keysegs, sort_param.key++) + { + sort_param.read_cache=param->read_cache; + sort_param.keyinfo=share->keyinfo+sort_param.key; + sort_param.seg=sort_param.keyinfo->seg; + if (! maria_is_key_active(key_map, sort_param.key)) + { + /* Remember old statistics for key */ + memcpy((char*) rec_per_key_part, + (char*) (share->state.rec_per_key_part + + (uint) (rec_per_key_part - param->rec_per_key_part)), + sort_param.keyinfo->keysegs*sizeof(*rec_per_key_part)); + continue; + } + + if ((!(param->testflag & T_SILENT))) + printf ("- Fixing index %d\n",sort_param.key+1); + sort_param.max_pos=sort_param.pos=share->pack.header_length; + keyseg=sort_param.seg; + bzero((char*) sort_param.unique,sizeof(sort_param.unique)); + sort_param.key_length=share->rec_reflength; + for (i=0 ; keyseg[i].type != HA_KEYTYPE_END; i++) + { + sort_param.key_length+=keyseg[i].length; + if (keyseg[i].flag & HA_SPACE_PACK) + sort_param.key_length+=get_pack_length(keyseg[i].length); + if (keyseg[i].flag & (HA_BLOB_PART | HA_VAR_LENGTH_PART)) + sort_param.key_length+=2 + test(keyseg[i].length >= 127); + if (keyseg[i].flag & HA_NULL_PART) + sort_param.key_length++; + } + info->state->records=info->state->del=share->state.split=0; + info->state->empty=0; + + if (sort_param.keyinfo->flag & HA_FULLTEXT) + { + uint ft_max_word_len_for_sort=FT_MAX_WORD_LEN_FOR_SORT* + sort_param.keyinfo->seg->charset->mbmaxlen; + sort_info.max_records= + (ha_rows) (sort_info.filelength/ft_min_word_len+1); + + sort_param.key_read=sort_maria_ft_key_read; + sort_param.key_write=sort_maria_ft_key_write; + sort_param.key_length+=ft_max_word_len_for_sort-HA_FT_MAXBYTELEN; + } + else + { + sort_param.key_read=sort_key_read; + sort_param.key_write=sort_key_write; + } + + if (_ma_create_index_by_sort(&sort_param, + (my_bool) (!(param->testflag & T_VERBOSE)), + (uint) param->sort_buffer_length)) + { + param->retry_repair=1; + goto err; + } + param->calc_checksum=0; /* No need to calc glob_crc */ + + /* Set for next loop */ + sort_info.max_records= (ha_rows) info->state->records; + + if (param->testflag & T_STATISTICS) + maria_update_key_parts(sort_param.keyinfo, rec_per_key_part, sort_param.unique, + param->stats_method == MI_STATS_METHOD_IGNORE_NULLS? + sort_param.notnull: NULL,(ulonglong) info->state->records); + maria_set_key_active(share->state.key_map, sort_param.key); + + if (sort_param.fix_datafile) + { + param->read_cache.end_of_file=sort_param.filepos; + if (maria_write_data_suffix(&sort_info,1) || end_io_cache(&info->rec_cache)) + goto err; + if (param->testflag & T_SAFE_REPAIR) + { + /* Don't repair if we loosed more than one row */ + if (info->state->records+1 < start_records) + { + info->state->records=start_records; + goto err; + } + } + share->state.state.data_file_length = info->state->data_file_length= + sort_param.filepos; + /* Only whole records */ + share->state.version=(ulong) time((time_t*) 0); + my_close(info->dfile,MYF(0)); + info->dfile=new_file; + share->data_file_type=sort_info.new_data_file_type; + share->pack.header_length=(ulong) new_header_length; + sort_param.fix_datafile=0; + } + else + info->state->data_file_length=sort_param.max_pos; + + param->read_cache.file=info->dfile; /* re-init read cache */ + reinit_io_cache(¶m->read_cache,READ_CACHE,share->pack.header_length, + 1,1); + } + + if (param->testflag & T_WRITE_LOOP) + { + VOID(fputs(" \r",stdout)); VOID(fflush(stdout)); + } + + if (rep_quick && del+sort_info.dupp != info->state->del) + { + _ma_check_print_error(param,"Couldn't fix table with quick recovery: Found wrong number of deleted records"); + _ma_check_print_error(param,"Run recovery again without -q"); + got_error=1; + param->retry_repair=1; + param->testflag|=T_RETRY_WITHOUT_QUICK; + goto err; + } + + if (rep_quick & T_FORCE_UNIQUENESS) + { + my_off_t skr=info->state->data_file_length+ + (share->options & HA_OPTION_COMPRESS_RECORD ? + MEMMAP_EXTRA_MARGIN : 0); +#ifdef USE_RELOC + if (share->data_file_type == STATIC_RECORD && + skr < share->base.reloc*share->base.min_pack_length) + skr=share->base.reloc*share->base.min_pack_length; +#endif + if (skr != sort_info.filelength && !info->s->base.raid_type) + if (my_chsize(info->dfile,skr,0,MYF(0))) + _ma_check_print_warning(param, + "Can't change size of datafile, error: %d", + my_errno); + } + if (param->testflag & T_CALC_CHECKSUM) + info->state->checksum=param->glob_crc; + + if (my_chsize(share->kfile,info->state->key_file_length,0,MYF(0))) + _ma_check_print_warning(param, + "Can't change size of indexfile, error: %d", + my_errno); + + if (!(param->testflag & T_SILENT)) + { + if (start_records != info->state->records) + printf("Data records: %s\n", llstr(info->state->records,llbuff)); + if (sort_info.dupp) + _ma_check_print_warning(param, + "%s records have been removed", + llstr(sort_info.dupp,llbuff)); + } + got_error=0; + + if (&share->state.state != info->state) + memcpy( &share->state.state, info->state, sizeof(*info->state)); + +err: + got_error|= _ma_flush_blocks(param, share->key_cache, share->kfile); + VOID(end_io_cache(&info->rec_cache)); + if (!got_error) + { + /* Replace the actual file with the temporary file */ + if (new_file >= 0) + { + my_close(new_file,MYF(0)); + info->dfile=new_file= -1; + if (maria_change_to_newfile(share->data_file_name,MARIA_NAME_DEXT, + DATA_TMP_EXT, share->base.raid_chunks, + (param->testflag & T_BACKUP_DATA ? + MYF(MY_REDEL_MAKE_BACKUP): MYF(0))) || + _ma_open_datafile(info,share,-1)) + got_error=1; + } + } + if (got_error) + { + if (! param->error_printed) + _ma_check_print_error(param,"%d when fixing table",my_errno); + if (new_file >= 0) + { + VOID(my_close(new_file,MYF(0))); + VOID(my_raid_delete(param->temp_filename,share->base.raid_chunks, + MYF(MY_WME))); + if (info->dfile == new_file) + info->dfile= -1; + } + maria_mark_crashed_on_repair(info); + } + else if (key_map == share->state.key_map) + share->state.changed&= ~STATE_NOT_OPTIMIZED_KEYS; + share->state.changed|=STATE_NOT_SORTED_PAGES; + + my_free(_ma_get_rec_buff_ptr(info, sort_param.rec_buff), + MYF(MY_ALLOW_ZERO_PTR)); + my_free(sort_param.record,MYF(MY_ALLOW_ZERO_PTR)); + my_free((gptr) sort_info.key_block,MYF(MY_ALLOW_ZERO_PTR)); + my_free((gptr) sort_info.ft_buf, MYF(MY_ALLOW_ZERO_PTR)); + my_free(sort_info.buff,MYF(MY_ALLOW_ZERO_PTR)); + VOID(end_io_cache(¶m->read_cache)); + info->opt_flag&= ~(READ_CACHE_USED | WRITE_CACHE_USED); + if (!got_error && (param->testflag & T_UNPACK)) + { + share->state.header.options[0]&= (uchar) ~HA_OPTION_COMPRESS_RECORD; + share->pack.header_length=0; + } + DBUG_RETURN(got_error); +} + +/* + Threaded repair of table using sorting + + SYNOPSIS + maria_repair_parallel() + param Repair parameters + info MARIA handler to repair + name Name of table (for warnings) + rep_quick set to <> 0 if we should not change data file + + DESCRIPTION + Same as maria_repair_by_sort but do it multithreaded + Each key is handled by a separate thread. + TODO: make a number of threads a parameter + + RESULT + 0 ok + <>0 Error +*/ + +int maria_repair_parallel(HA_CHECK *param, register MARIA_HA *info, + const char * name, int rep_quick) +{ +#ifndef THREAD + return maria_repair_by_sort(param, info, name, rep_quick); +#else + int got_error; + uint i,key, total_key_length, istep; + ulong rec_length; + ha_rows start_records; + my_off_t new_header_length,del; + File new_file; + MARIA_SORT_PARAM *sort_param=0; + MARIA_SHARE *share=info->s; + ulong *rec_per_key_part; + HA_KEYSEG *keyseg; + char llbuff[22]; + IO_CACHE_SHARE io_share; + MARIA_SORT_INFO sort_info; + ulonglong key_map=share->state.key_map; + pthread_attr_t thr_attr; + DBUG_ENTER("maria_repair_parallel"); + + start_records=info->state->records; + got_error=1; + new_file= -1; + new_header_length=(param->testflag & T_UNPACK) ? 0 : + share->pack.header_length; + if (!(param->testflag & T_SILENT)) + { + printf("- parallel recovering (with sort) MARIA-table '%s'\n",name); + printf("Data records: %s\n", llstr(start_records,llbuff)); + } + param->testflag|=T_REP; /* for easy checking */ + + if (info->s->options & (HA_OPTION_CHECKSUM | HA_OPTION_COMPRESS_RECORD)) + param->testflag|=T_CALC_CHECKSUM; + + bzero((char*)&sort_info,sizeof(sort_info)); + if (!(sort_info.key_block= + alloc_key_blocks(param, + (uint) param->sort_key_blocks, + share->base.max_key_block_length)) + || init_io_cache(¶m->read_cache,info->dfile, + (uint) param->read_buffer_length, + READ_CACHE,share->pack.header_length,1,MYF(MY_WME)) || + (! rep_quick && + init_io_cache(&info->rec_cache,info->dfile, + (uint) param->write_buffer_length, + WRITE_CACHE,new_header_length,1, + MYF(MY_WME | MY_WAIT_IF_FULL) & param->myf_rw))) + goto err; + sort_info.key_block_end=sort_info.key_block+param->sort_key_blocks; + info->opt_flag|=WRITE_CACHE_USED; + info->rec_cache.file=info->dfile; /* for sort_delete_record */ + + if (!rep_quick) + { + /* Get real path for data file */ + if ((new_file=my_raid_create(fn_format(param->temp_filename, + share->data_file_name, "", + DATA_TMP_EXT, + 2+4), + 0,param->tmpfile_createflag, + share->base.raid_type, + share->base.raid_chunks, + share->base.raid_chunksize, + MYF(0))) < 0) + { + _ma_check_print_error(param,"Can't create new tempfile: '%s'", + param->temp_filename); + goto err; + } + if (maria_filecopy(param, new_file,info->dfile,0L,new_header_length, + "datafile-header")) + goto err; + if (param->testflag & T_UNPACK) + { + share->options&= ~HA_OPTION_COMPRESS_RECORD; + mi_int2store(share->state.header.options,share->options); + } + share->state.dellink= HA_OFFSET_ERROR; + info->rec_cache.file=new_file; + } + + info->update= (short) (HA_STATE_CHANGED | HA_STATE_ROW_CHANGED); + if (!(param->testflag & T_CREATE_MISSING_KEYS)) + { + /* + Flush key cache for this file if we are calling this outside + mariachk + */ + flush_key_blocks(share->key_cache,share->kfile, FLUSH_IGNORE_CHANGED); + /* Clear the pointers to the given rows */ + for (i=0 ; i < share->base.keys ; i++) + share->state.key_root[i]= HA_OFFSET_ERROR; + for (i=0 ; i < share->state.header.max_block_size ; i++) + share->state.key_del[i]= HA_OFFSET_ERROR; + info->state->key_file_length=share->base.keystart; + } + else + { + if (flush_key_blocks(share->key_cache,share->kfile, FLUSH_FORCE_WRITE)) + goto err; + key_map= ~key_map; /* Create the missing keys */ + } + + sort_info.info=info; + sort_info.param = param; + + set_data_file_type(&sort_info, share); + sort_info.dupp=0; + sort_info.buff=0; + param->read_cache.end_of_file=sort_info.filelength= + my_seek(param->read_cache.file,0L,MY_SEEK_END,MYF(0)); + + if (share->data_file_type == DYNAMIC_RECORD) + rec_length=max(share->base.min_pack_length+1,share->base.min_block_length); + else if (share->data_file_type == COMPRESSED_RECORD) + rec_length=share->base.min_block_length; + else + rec_length=share->base.pack_reclength; + /* + +1 below is required hack for parallel repair mode. + The info->state->records value, that is compared later + to sort_info.max_records and cannot exceed it, is + increased in sort_key_write. In maria_repair_by_sort, sort_key_write + is called after sort_key_read, where the comparison is performed, + but in parallel mode master thread can call sort_key_write + before some other repair thread calls sort_key_read. + Furthermore I'm not even sure +1 would be enough. + May be sort_info.max_records shold be always set to max value in + parallel mode. + */ + sort_info.max_records= + ((param->testflag & T_CREATE_MISSING_KEYS) ? info->state->records + 1: + (ha_rows) (sort_info.filelength/rec_length+1)); + + del=info->state->del; + param->glob_crc=0; + if (param->testflag & T_CALC_CHECKSUM) + param->calc_checksum=1; + + if (!(sort_param=(MARIA_SORT_PARAM *) + my_malloc((uint) share->base.keys * + (sizeof(MARIA_SORT_PARAM) + share->base.pack_reclength), + MYF(MY_ZEROFILL)))) + { + _ma_check_print_error(param,"Not enough memory for key!"); + goto err; + } + total_key_length=0; + rec_per_key_part= param->rec_per_key_part; + info->state->records=info->state->del=share->state.split=0; + info->state->empty=0; + + for (i=key=0, istep=1 ; key < share->base.keys ; + rec_per_key_part+=sort_param[i].keyinfo->keysegs, i+=istep, key++) + { + sort_param[i].key=key; + sort_param[i].keyinfo=share->keyinfo+key; + sort_param[i].seg=sort_param[i].keyinfo->seg; + if (! maria_is_key_active(key_map, key)) + { + /* Remember old statistics for key */ + memcpy((char*) rec_per_key_part, + (char*) (share->state.rec_per_key_part+ + (uint) (rec_per_key_part - param->rec_per_key_part)), + sort_param[i].keyinfo->keysegs*sizeof(*rec_per_key_part)); + istep=0; + continue; + } + istep=1; + if ((!(param->testflag & T_SILENT))) + printf ("- Fixing index %d\n",key+1); + if (sort_param[i].keyinfo->flag & HA_FULLTEXT) + { + sort_param[i].key_read=sort_maria_ft_key_read; + sort_param[i].key_write=sort_maria_ft_key_write; + } + else + { + sort_param[i].key_read=sort_key_read; + sort_param[i].key_write=sort_key_write; + } + sort_param[i].key_cmp=sort_key_cmp; + sort_param[i].lock_in_memory=maria_lock_memory; + sort_param[i].tmpdir=param->tmpdir; + sort_param[i].sort_info=&sort_info; + sort_param[i].master=0; + sort_param[i].fix_datafile=0; + + sort_param[i].filepos=new_header_length; + sort_param[i].max_pos=sort_param[i].pos=share->pack.header_length; + + sort_param[i].record= (((char *)(sort_param+share->base.keys))+ + (share->base.pack_reclength * i)); + if (!_ma_alloc_rec_buff(info, -1, &sort_param[i].rec_buff)) + { + _ma_check_print_error(param,"Not enough memory!"); + goto err; + } + + sort_param[i].key_length=share->rec_reflength; + for (keyseg=sort_param[i].seg; keyseg->type != HA_KEYTYPE_END; + keyseg++) + { + sort_param[i].key_length+=keyseg->length; + if (keyseg->flag & HA_SPACE_PACK) + sort_param[i].key_length+=get_pack_length(keyseg->length); + if (keyseg->flag & (HA_BLOB_PART | HA_VAR_LENGTH_PART)) + sort_param[i].key_length+=2 + test(keyseg->length >= 127); + if (keyseg->flag & HA_NULL_PART) + sort_param[i].key_length++; + } + total_key_length+=sort_param[i].key_length; + + if (sort_param[i].keyinfo->flag & HA_FULLTEXT) + { + uint ft_max_word_len_for_sort=FT_MAX_WORD_LEN_FOR_SORT* + sort_param[i].keyinfo->seg->charset->mbmaxlen; + sort_param[i].key_length+=ft_max_word_len_for_sort-HA_FT_MAXBYTELEN; + } + } + sort_info.total_keys=i; + sort_param[0].master= 1; + sort_param[0].fix_datafile= (my_bool)(! rep_quick); + + sort_info.got_error=0; + pthread_mutex_init(&sort_info.mutex, MY_MUTEX_INIT_FAST); + pthread_cond_init(&sort_info.cond, 0); + pthread_mutex_lock(&sort_info.mutex); + + init_io_cache_share(¶m->read_cache, &io_share, i); + (void) pthread_attr_init(&thr_attr); + (void) pthread_attr_setdetachstate(&thr_attr,PTHREAD_CREATE_DETACHED); + + for (i=0 ; i < sort_info.total_keys ; i++) + { + sort_param[i].read_cache=param->read_cache; + /* + two approaches: the same amount of memory for each thread + or the memory for the same number of keys for each thread... + In the second one all the threads will fill their sort_buffers + (and call write_keys) at the same time, putting more stress on i/o. + */ + sort_param[i].sortbuff_size= +#ifndef USING_SECOND_APPROACH + param->sort_buffer_length/sort_info.total_keys; +#else + param->sort_buffer_length*sort_param[i].key_length/total_key_length; +#endif + if (pthread_create(&sort_param[i].thr, &thr_attr, + _ma_thr_find_all_keys, + (void *) (sort_param+i))) + { + _ma_check_print_error(param,"Cannot start a repair thread"); + remove_io_thread(¶m->read_cache); + sort_info.got_error=1; + } + else + sort_info.threads_running++; + } + (void) pthread_attr_destroy(&thr_attr); + + /* waiting for all threads to finish */ + while (sort_info.threads_running) + pthread_cond_wait(&sort_info.cond, &sort_info.mutex); + pthread_mutex_unlock(&sort_info.mutex); + + if ((got_error= _ma_thr_write_keys(sort_param))) + { + param->retry_repair=1; + goto err; + } + got_error=1; /* Assume the following may go wrong */ + + if (sort_param[0].fix_datafile) + { + if (maria_write_data_suffix(&sort_info,1) || end_io_cache(&info->rec_cache)) + goto err; + if (param->testflag & T_SAFE_REPAIR) + { + /* Don't repair if we loosed more than one row */ + if (info->state->records+1 < start_records) + { + info->state->records=start_records; + goto err; + } + } + share->state.state.data_file_length= info->state->data_file_length= + sort_param->filepos; + /* Only whole records */ + share->state.version=(ulong) time((time_t*) 0); + my_close(info->dfile,MYF(0)); + info->dfile=new_file; + share->data_file_type=sort_info.new_data_file_type; + share->pack.header_length=(ulong) new_header_length; + } + else + info->state->data_file_length=sort_param->max_pos; + + if (rep_quick && del+sort_info.dupp != info->state->del) + { + _ma_check_print_error(param,"Couldn't fix table with quick recovery: Found wrong number of deleted records"); + _ma_check_print_error(param,"Run recovery again without -q"); + param->retry_repair=1; + param->testflag|=T_RETRY_WITHOUT_QUICK; + goto err; + } + + if (rep_quick & T_FORCE_UNIQUENESS) + { + my_off_t skr=info->state->data_file_length+ + (share->options & HA_OPTION_COMPRESS_RECORD ? + MEMMAP_EXTRA_MARGIN : 0); +#ifdef USE_RELOC + if (share->data_file_type == STATIC_RECORD && + skr < share->base.reloc*share->base.min_pack_length) + skr=share->base.reloc*share->base.min_pack_length; +#endif + if (skr != sort_info.filelength && !info->s->base.raid_type) + if (my_chsize(info->dfile,skr,0,MYF(0))) + _ma_check_print_warning(param, + "Can't change size of datafile, error: %d", + my_errno); + } + if (param->testflag & T_CALC_CHECKSUM) + info->state->checksum=param->glob_crc; + + if (my_chsize(share->kfile,info->state->key_file_length,0,MYF(0))) + _ma_check_print_warning(param, + "Can't change size of indexfile, error: %d", my_errno); + + if (!(param->testflag & T_SILENT)) + { + if (start_records != info->state->records) + printf("Data records: %s\n", llstr(info->state->records,llbuff)); + if (sort_info.dupp) + _ma_check_print_warning(param, + "%s records have been removed", + llstr(sort_info.dupp,llbuff)); + } + got_error=0; + + if (&share->state.state != info->state) + memcpy(&share->state.state, info->state, sizeof(*info->state)); + +err: + got_error|= _ma_flush_blocks(param, share->key_cache, share->kfile); + VOID(end_io_cache(&info->rec_cache)); + if (!got_error) + { + /* Replace the actual file with the temporary file */ + if (new_file >= 0) + { + my_close(new_file,MYF(0)); + info->dfile=new_file= -1; + if (maria_change_to_newfile(share->data_file_name,MARIA_NAME_DEXT, + DATA_TMP_EXT, share->base.raid_chunks, + (param->testflag & T_BACKUP_DATA ? + MYF(MY_REDEL_MAKE_BACKUP): MYF(0))) || + _ma_open_datafile(info,share,-1)) + got_error=1; + } + } + if (got_error) + { + if (! param->error_printed) + _ma_check_print_error(param,"%d when fixing table",my_errno); + if (new_file >= 0) + { + VOID(my_close(new_file,MYF(0))); + VOID(my_raid_delete(param->temp_filename,share->base.raid_chunks, + MYF(MY_WME))); + if (info->dfile == new_file) + info->dfile= -1; + } + maria_mark_crashed_on_repair(info); + } + else if (key_map == share->state.key_map) + share->state.changed&= ~STATE_NOT_OPTIMIZED_KEYS; + share->state.changed|=STATE_NOT_SORTED_PAGES; + + pthread_cond_destroy (&sort_info.cond); + pthread_mutex_destroy(&sort_info.mutex); + + my_free((gptr) sort_info.ft_buf, MYF(MY_ALLOW_ZERO_PTR)); + my_free((gptr) sort_info.key_block,MYF(MY_ALLOW_ZERO_PTR)); + my_free((gptr) sort_param,MYF(MY_ALLOW_ZERO_PTR)); + my_free(sort_info.buff,MYF(MY_ALLOW_ZERO_PTR)); + VOID(end_io_cache(¶m->read_cache)); + info->opt_flag&= ~(READ_CACHE_USED | WRITE_CACHE_USED); + if (!got_error && (param->testflag & T_UNPACK)) + { + share->state.header.options[0]&= (uchar) ~HA_OPTION_COMPRESS_RECORD; + share->pack.header_length=0; + } + DBUG_RETURN(got_error); +#endif /* THREAD */ +} + + /* Read next record and return next key */ + +static int sort_key_read(MARIA_SORT_PARAM *sort_param, void *key) +{ + int error; + MARIA_SORT_INFO *sort_info=sort_param->sort_info; + MARIA_HA *info=sort_info->info; + DBUG_ENTER("sort_key_read"); + + if ((error=sort_get_next_record(sort_param))) + DBUG_RETURN(error); + if (info->state->records == sort_info->max_records) + { + _ma_check_print_error(sort_info->param, + "Key %d - Found too many records; Can't continue", + sort_param->key+1); + DBUG_RETURN(1); + } + sort_param->real_key_length= + (info->s->rec_reflength+ + _ma_make_key(info, sort_param->key, (uchar*) key, + sort_param->record, sort_param->filepos)); +#ifdef HAVE_purify + bzero(key+sort_param->real_key_length, + (sort_param->key_length-sort_param->real_key_length)); +#endif + DBUG_RETURN(_ma_sort_write_record(sort_param)); +} /* sort_key_read */ + +static int sort_maria_ft_key_read(MARIA_SORT_PARAM *sort_param, void *key) +{ + int error; + MARIA_SORT_INFO *sort_info=sort_param->sort_info; + MARIA_HA *info=sort_info->info; + FT_WORD *wptr=0; + DBUG_ENTER("sort_maria_ft_key_read"); + + if (!sort_param->wordlist) + { + for (;;) + { + my_free((char*) wptr, MYF(MY_ALLOW_ZERO_PTR)); + if ((error=sort_get_next_record(sort_param))) + DBUG_RETURN(error); + if (!(wptr= _ma_ft_parserecord(info,sort_param->key,sort_param->record))) + DBUG_RETURN(1); + if (wptr->pos) + break; + error=_ma_sort_write_record(sort_param); + } + sort_param->wordptr=sort_param->wordlist=wptr; + } + else + { + error=0; + wptr=(FT_WORD*)(sort_param->wordptr); + } + + sort_param->real_key_length=(info->s->rec_reflength+ + _ma_ft_make_key(info, sort_param->key, + key, wptr++, sort_param->filepos)); +#ifdef HAVE_purify + if (sort_param->key_length > sort_param->real_key_length) + bzero(key+sort_param->real_key_length, + (sort_param->key_length-sort_param->real_key_length)); +#endif + if (!wptr->pos) + { + my_free((char*) sort_param->wordlist, MYF(0)); + sort_param->wordlist=0; + error=_ma_sort_write_record(sort_param); + } + else + sort_param->wordptr=(void*)wptr; + + DBUG_RETURN(error); +} /* sort_maria_ft_key_read */ + + + /* Read next record from file using parameters in sort_info */ + /* Return -1 if end of file, 0 if ok and > 0 if error */ + +static int sort_get_next_record(MARIA_SORT_PARAM *sort_param) +{ + int searching; + uint found_record,b_type,left_length; + my_off_t pos; + byte *to; + MARIA_BLOCK_INFO block_info; + MARIA_SORT_INFO *sort_info=sort_param->sort_info; + HA_CHECK *param=sort_info->param; + MARIA_HA *info=sort_info->info; + MARIA_SHARE *share=info->s; + char llbuff[22],llbuff2[22]; + DBUG_ENTER("sort_get_next_record"); + + if (*_ma_killed_ptr(param)) + DBUG_RETURN(1); + + switch (share->data_file_type) { + case STATIC_RECORD: + for (;;) + { + if (my_b_read(&sort_param->read_cache,sort_param->record, + share->base.pack_reclength)) + { + if (sort_param->read_cache.error) + param->out_flag |= O_DATA_LOST; + param->retry_repair=1; + param->testflag|=T_RETRY_WITHOUT_QUICK; + DBUG_RETURN(-1); + } + sort_param->start_recpos=sort_param->pos; + if (!sort_param->fix_datafile) + { + sort_param->filepos=sort_param->pos; + if (sort_param->master) + share->state.split++; + } + sort_param->max_pos=(sort_param->pos+=share->base.pack_reclength); + if (*sort_param->record) + { + if (param->calc_checksum) + param->glob_crc+= (info->checksum= + _ma_static_checksum(info,sort_param->record)); + DBUG_RETURN(0); + } + if (!sort_param->fix_datafile && sort_param->master) + { + info->state->del++; + info->state->empty+=share->base.pack_reclength; + } + } + case DYNAMIC_RECORD: + LINT_INIT(to); + pos=sort_param->pos; + searching=(sort_param->fix_datafile && (param->testflag & T_EXTEND)); + for (;;) + { + found_record=block_info.second_read= 0; + left_length=1; + if (searching) + { + pos=MY_ALIGN(pos,MARIA_DYN_ALIGN_SIZE); + param->testflag|=T_RETRY_WITHOUT_QUICK; + sort_param->start_recpos=pos; + } + do + { + if (pos > sort_param->max_pos) + sort_param->max_pos=pos; + if (pos & (MARIA_DYN_ALIGN_SIZE-1)) + { + if ((param->testflag & T_VERBOSE) || searching == 0) + _ma_check_print_info(param,"Wrong aligned block at %s", + llstr(pos,llbuff)); + if (searching) + goto try_next; + } + if (found_record && pos == param->search_after_block) + _ma_check_print_info(param,"Block: %s used by record at %s", + llstr(param->search_after_block,llbuff), + llstr(sort_param->start_recpos,llbuff2)); + if (_ma_read_cache(&sort_param->read_cache, + (byte*) block_info.header,pos, + MARIA_BLOCK_INFO_HEADER_LENGTH, + (! found_record ? READING_NEXT : 0) | + READING_HEADER)) + { + if (found_record) + { + _ma_check_print_info(param, + "Can't read whole record at %s (errno: %d)", + llstr(sort_param->start_recpos,llbuff),errno); + goto try_next; + } + DBUG_RETURN(-1); + } + if (searching && ! sort_param->fix_datafile) + { + param->error_printed=1; + param->retry_repair=1; + param->testflag|=T_RETRY_WITHOUT_QUICK; + DBUG_RETURN(1); /* Something wrong with data */ + } + b_type= _ma_get_block_info(&block_info,-1,pos); + if ((b_type & (BLOCK_ERROR | BLOCK_FATAL_ERROR)) || + ((b_type & BLOCK_FIRST) && + (block_info.rec_len < (uint) share->base.min_pack_length || + block_info.rec_len > (uint) share->base.max_pack_length))) + { + uint i; + if (param->testflag & T_VERBOSE || searching == 0) + _ma_check_print_info(param, + "Wrong bytesec: %3d-%3d-%3d at %10s; Skipped", + block_info.header[0],block_info.header[1], + block_info.header[2],llstr(pos,llbuff)); + if (found_record) + goto try_next; + block_info.second_read=0; + searching=1; + /* Search after block in read header string */ + for (i=MARIA_DYN_ALIGN_SIZE ; + i < MARIA_BLOCK_INFO_HEADER_LENGTH ; + i+= MARIA_DYN_ALIGN_SIZE) + if (block_info.header[i] >= 1 && + block_info.header[i] <= MARIA_MAX_DYN_HEADER_BYTE) + break; + pos+=(ulong) i; + sort_param->start_recpos=pos; + continue; + } + if (b_type & BLOCK_DELETED) + { + bool error=0; + if (block_info.block_len+ (uint) (block_info.filepos-pos) < + share->base.min_block_length) + { + if (!searching) + _ma_check_print_info(param, + "Deleted block with impossible length %u at %s", + block_info.block_len,llstr(pos,llbuff)); + error=1; + } + else + { + if ((block_info.next_filepos != HA_OFFSET_ERROR && + block_info.next_filepos >= + info->state->data_file_length) || + (block_info.prev_filepos != HA_OFFSET_ERROR && + block_info.prev_filepos >= info->state->data_file_length)) + { + if (!searching) + _ma_check_print_info(param, + "Delete link points outside datafile at %s", + llstr(pos,llbuff)); + error=1; + } + } + if (error) + { + if (found_record) + goto try_next; + searching=1; + pos+= MARIA_DYN_ALIGN_SIZE; + sort_param->start_recpos=pos; + block_info.second_read=0; + continue; + } + } + else + { + if (block_info.block_len+ (uint) (block_info.filepos-pos) < + share->base.min_block_length || + block_info.block_len > (uint) share->base.max_pack_length+ + MARIA_SPLIT_LENGTH) + { + if (!searching) + _ma_check_print_info(param, + "Found block with impossible length %u at %s; Skipped", + block_info.block_len+ (uint) (block_info.filepos-pos), + llstr(pos,llbuff)); + if (found_record) + goto try_next; + searching=1; + pos+= MARIA_DYN_ALIGN_SIZE; + sort_param->start_recpos=pos; + block_info.second_read=0; + continue; + } + } + if (b_type & (BLOCK_DELETED | BLOCK_SYNC_ERROR)) + { + if (!sort_param->fix_datafile && sort_param->master && + (b_type & BLOCK_DELETED)) + { + info->state->empty+=block_info.block_len; + info->state->del++; + share->state.split++; + } + if (found_record) + goto try_next; + if (searching) + { + pos+=MARIA_DYN_ALIGN_SIZE; + sort_param->start_recpos=pos; + } + else + pos=block_info.filepos+block_info.block_len; + block_info.second_read=0; + continue; + } + + if (!sort_param->fix_datafile && sort_param->master) + share->state.split++; + if (! found_record++) + { + sort_param->find_length=left_length=block_info.rec_len; + sort_param->start_recpos=pos; + if (!sort_param->fix_datafile) + sort_param->filepos=sort_param->start_recpos; + if (sort_param->fix_datafile && (param->testflag & T_EXTEND)) + sort_param->pos=block_info.filepos+1; + else + sort_param->pos=block_info.filepos+block_info.block_len; + if (share->base.blobs) + { + if (!(to=_ma_alloc_rec_buff(info,block_info.rec_len, + &(sort_param->rec_buff)))) + { + if (param->max_record_length >= block_info.rec_len) + { + _ma_check_print_error(param,"Not enough memory for blob at %s (need %lu)", + llstr(sort_param->start_recpos,llbuff), + (ulong) block_info.rec_len); + DBUG_RETURN(1); + } + else + { + _ma_check_print_info(param,"Not enough memory for blob at %s (need %lu); Row skipped", + llstr(sort_param->start_recpos,llbuff), + (ulong) block_info.rec_len); + goto try_next; + } + } + } + else + to= sort_param->rec_buff; + } + if (left_length < block_info.data_len || ! block_info.data_len) + { + _ma_check_print_info(param, + "Found block with too small length at %s; Skipped", + llstr(sort_param->start_recpos,llbuff)); + goto try_next; + } + if (block_info.filepos + block_info.data_len > + sort_param->read_cache.end_of_file) + { + _ma_check_print_info(param, + "Found block that points outside data file at %s", + llstr(sort_param->start_recpos,llbuff)); + goto try_next; + } + if (_ma_read_cache(&sort_param->read_cache,to,block_info.filepos, + block_info.data_len, + (found_record == 1 ? READING_NEXT : 0))) + { + _ma_check_print_info(param, + "Read error for block at: %s (error: %d); Skipped", + llstr(block_info.filepos,llbuff),my_errno); + goto try_next; + } + left_length-=block_info.data_len; + to+=block_info.data_len; + pos=block_info.next_filepos; + if (pos == HA_OFFSET_ERROR && left_length) + { + _ma_check_print_info(param,"Wrong block with wrong total length starting at %s", + llstr(sort_param->start_recpos,llbuff)); + goto try_next; + } + if (pos + MARIA_BLOCK_INFO_HEADER_LENGTH > sort_param->read_cache.end_of_file) + { + _ma_check_print_info(param,"Found link that points at %s (outside data file) at %s", + llstr(pos,llbuff2), + llstr(sort_param->start_recpos,llbuff)); + goto try_next; + } + } while (left_length); + + if (_ma_rec_unpack(info,sort_param->record,sort_param->rec_buff, + sort_param->find_length) != MY_FILE_ERROR) + { + if (sort_param->read_cache.error < 0) + DBUG_RETURN(1); + if (info->s->calc_checksum) + info->checksum=_ma_checksum(info,sort_param->record); + if ((param->testflag & (T_EXTEND | T_REP)) || searching) + { + if (_ma_rec_check(info, sort_param->record, sort_param->rec_buff, + sort_param->find_length, + (param->testflag & T_QUICK) && + test(info->s->calc_checksum))) + { + _ma_check_print_info(param,"Found wrong packed record at %s", + llstr(sort_param->start_recpos,llbuff)); + goto try_next; + } + } + if (param->calc_checksum) + param->glob_crc+= info->checksum; + DBUG_RETURN(0); + } + if (!searching) + _ma_check_print_info(param,"Key %d - Found wrong stored record at %s", + sort_param->key+1, + llstr(sort_param->start_recpos,llbuff)); + try_next: + pos=(sort_param->start_recpos+=MARIA_DYN_ALIGN_SIZE); + searching=1; + } + case COMPRESSED_RECORD: + for (searching=0 ;; searching=1, sort_param->pos++) + { + if (_ma_read_cache(&sort_param->read_cache,(byte*) block_info.header, + sort_param->pos, + share->pack.ref_length,READING_NEXT)) + DBUG_RETURN(-1); + if (searching && ! sort_param->fix_datafile) + { + param->error_printed=1; + param->retry_repair=1; + param->testflag|=T_RETRY_WITHOUT_QUICK; + DBUG_RETURN(1); /* Something wrong with data */ + } + sort_param->start_recpos=sort_param->pos; + if (_ma_pack_get_block_info(info,&block_info,-1,sort_param->pos)) + DBUG_RETURN(-1); + if (!block_info.rec_len && + sort_param->pos + MEMMAP_EXTRA_MARGIN == + sort_param->read_cache.end_of_file) + DBUG_RETURN(-1); + if (block_info.rec_len < (uint) share->min_pack_length || + block_info.rec_len > (uint) share->max_pack_length) + { + if (! searching) + _ma_check_print_info(param,"Found block with wrong recordlength: %d at %s\n", + block_info.rec_len, + llstr(sort_param->pos,llbuff)); + continue; + } + if (_ma_read_cache(&sort_param->read_cache,(byte*) sort_param->rec_buff, + block_info.filepos, block_info.rec_len, + READING_NEXT)) + { + if (! searching) + _ma_check_print_info(param,"Couldn't read whole record from %s", + llstr(sort_param->pos,llbuff)); + continue; + } + if (_ma_pack_rec_unpack(info,sort_param->record,sort_param->rec_buff, + block_info.rec_len)) + { + if (! searching) + _ma_check_print_info(param,"Found wrong record at %s", + llstr(sort_param->pos,llbuff)); + continue; + } + info->checksum=_ma_checksum(info,sort_param->record); + if (!sort_param->fix_datafile) + { + sort_param->filepos=sort_param->pos; + if (sort_param->master) + share->state.split++; + } + sort_param->max_pos=(sort_param->pos=block_info.filepos+ + block_info.rec_len); + info->packed_length=block_info.rec_len; + if (param->calc_checksum) + param->glob_crc+= info->checksum; + DBUG_RETURN(0); + } + } + DBUG_RETURN(1); /* Impossible */ +} + + + /* Write record to new file */ + +int _ma_sort_write_record(MARIA_SORT_PARAM *sort_param) +{ + int flag; + uint length; + ulong block_length,reclength; + byte *from; + byte block_buff[8]; + MARIA_SORT_INFO *sort_info=sort_param->sort_info; + HA_CHECK *param=sort_info->param; + MARIA_HA *info=sort_info->info; + MARIA_SHARE *share=info->s; + DBUG_ENTER("_ma_sort_write_record"); + + if (sort_param->fix_datafile) + { + switch (sort_info->new_data_file_type) { + case STATIC_RECORD: + if (my_b_write(&info->rec_cache,sort_param->record, + share->base.pack_reclength)) + { + _ma_check_print_error(param,"%d when writing to datafile",my_errno); + DBUG_RETURN(1); + } + sort_param->filepos+=share->base.pack_reclength; + info->s->state.split++; + /* sort_info->param->glob_crc+=_ma_static_checksum(info, sort_param->record); */ + break; + case DYNAMIC_RECORD: + if (! info->blobs) + from=sort_param->rec_buff; + else + { + /* must be sure that local buffer is big enough */ + reclength=info->s->base.pack_reclength+ + _ma_calc_total_blob_length(info,sort_param->record)+ + ALIGN_SIZE(MARIA_MAX_DYN_BLOCK_HEADER)+MARIA_SPLIT_LENGTH+ + MARIA_DYN_DELETE_BLOCK_HEADER; + if (sort_info->buff_length < reclength) + { + if (!(sort_info->buff=my_realloc(sort_info->buff, (uint) reclength, + MYF(MY_FREE_ON_ERROR | + MY_ALLOW_ZERO_PTR)))) + DBUG_RETURN(1); + sort_info->buff_length=reclength; + } + from=sort_info->buff+ALIGN_SIZE(MARIA_MAX_DYN_BLOCK_HEADER); + } + info->checksum=_ma_checksum(info,sort_param->record); + reclength= _ma_rec_pack(info,from,sort_param->record); + flag=0; + /* sort_info->param->glob_crc+=info->checksum; */ + + do + { + block_length=reclength+ 3 + test(reclength >= (65520-3)); + if (block_length < share->base.min_block_length) + block_length=share->base.min_block_length; + info->update|=HA_STATE_WRITE_AT_END; + block_length=MY_ALIGN(block_length,MARIA_DYN_ALIGN_SIZE); + if (block_length > MARIA_MAX_BLOCK_LENGTH) + block_length=MARIA_MAX_BLOCK_LENGTH; + if (_ma_write_part_record(info,0L,block_length, + sort_param->filepos+block_length, + &from,&reclength,&flag)) + { + _ma_check_print_error(param,"%d when writing to datafile",my_errno); + DBUG_RETURN(1); + } + sort_param->filepos+=block_length; + info->s->state.split++; + } while (reclength); + /* sort_info->param->glob_crc+=info->checksum; */ + break; + case COMPRESSED_RECORD: + reclength=info->packed_length; + length= _ma_save_pack_length((uint) share->pack.version, block_buff, + reclength); + if (info->s->base.blobs) + length+= _ma_save_pack_length((uint) share->pack.version, + block_buff + length, info->blob_length); + if (my_b_write(&info->rec_cache,block_buff,length) || + my_b_write(&info->rec_cache,(byte*) sort_param->rec_buff,reclength)) + { + _ma_check_print_error(param,"%d when writing to datafile",my_errno); + DBUG_RETURN(1); + } + /* sort_info->param->glob_crc+=info->checksum; */ + sort_param->filepos+=reclength+length; + info->s->state.split++; + break; + } + } + if (sort_param->master) + { + info->state->records++; + if ((param->testflag & T_WRITE_LOOP) && + (info->state->records % WRITE_COUNT) == 0) + { + char llbuff[22]; + printf("%s\r", llstr(info->state->records,llbuff)); + VOID(fflush(stdout)); + } + } + DBUG_RETURN(0); +} /* _ma_sort_write_record */ + + + /* Compare two keys from _ma_create_index_by_sort */ + +static int sort_key_cmp(MARIA_SORT_PARAM *sort_param, const void *a, + const void *b) +{ + uint not_used[2]; + return (ha_key_cmp(sort_param->seg, *((uchar**) a), *((uchar**) b), + USE_WHOLE_KEY, SEARCH_SAME, not_used)); +} /* sort_key_cmp */ + + +static int sort_key_write(MARIA_SORT_PARAM *sort_param, const void *a) +{ + uint diff_pos[2]; + char llbuff[22],llbuff2[22]; + MARIA_SORT_INFO *sort_info=sort_param->sort_info; + HA_CHECK *param= sort_info->param; + int cmp; + + if (sort_info->key_block->inited) + { + cmp=ha_key_cmp(sort_param->seg,sort_info->key_block->lastkey, + (uchar*) a, USE_WHOLE_KEY,SEARCH_FIND | SEARCH_UPDATE, + diff_pos); + if (param->stats_method == MI_STATS_METHOD_NULLS_NOT_EQUAL) + ha_key_cmp(sort_param->seg,sort_info->key_block->lastkey, + (uchar*) a, USE_WHOLE_KEY, + SEARCH_FIND | SEARCH_NULL_ARE_NOT_EQUAL, diff_pos); + else if (param->stats_method == MI_STATS_METHOD_IGNORE_NULLS) + { + diff_pos[0]= maria_collect_stats_nonulls_next(sort_param->seg, + sort_param->notnull, + sort_info->key_block->lastkey, + (uchar*)a); + } + sort_param->unique[diff_pos[0]-1]++; + } + else + { + cmp= -1; + if (param->stats_method == MI_STATS_METHOD_IGNORE_NULLS) + maria_collect_stats_nonulls_first(sort_param->seg, sort_param->notnull, + (uchar*)a); + } + if ((sort_param->keyinfo->flag & HA_NOSAME) && cmp == 0) + { + sort_info->dupp++; + sort_info->info->lastpos=get_record_for_key(sort_info->info, + sort_param->keyinfo, + (uchar*) a); + _ma_check_print_warning(param, + "Duplicate key for record at %10s against record at %10s", + llstr(sort_info->info->lastpos,llbuff), + llstr(get_record_for_key(sort_info->info, + sort_param->keyinfo, + sort_info->key_block-> + lastkey), + llbuff2)); + param->testflag|=T_RETRY_WITHOUT_QUICK; + if (sort_info->param->testflag & T_VERBOSE) + _ma_print_key(stdout,sort_param->seg,(uchar*) a, USE_WHOLE_KEY); + return (sort_delete_record(sort_param)); + } +#ifndef DBUG_OFF + if (cmp > 0) + { + _ma_check_print_error(param, + "Internal error: Keys are not in order from sort"); + return(1); + } +#endif + return (sort_insert_key(sort_param,sort_info->key_block, + (uchar*) a, HA_OFFSET_ERROR)); +} /* sort_key_write */ + +int _ma_sort_ft_buf_flush(MARIA_SORT_PARAM *sort_param) +{ + MARIA_SORT_INFO *sort_info=sort_param->sort_info; + SORT_KEY_BLOCKS *key_block=sort_info->key_block; + MARIA_SHARE *share=sort_info->info->s; + uint val_off, val_len; + int error; + SORT_FT_BUF *maria_ft_buf=sort_info->ft_buf; + uchar *from, *to; + + val_len=share->ft2_keyinfo.keylength; + get_key_full_length_rdonly(val_off, maria_ft_buf->lastkey); + to=maria_ft_buf->lastkey+val_off; + + if (maria_ft_buf->buf) + { + /* flushing first-level tree */ + error=sort_insert_key(sort_param,key_block,maria_ft_buf->lastkey, + HA_OFFSET_ERROR); + for (from=to+val_len; + !error && from < maria_ft_buf->buf; + from+= val_len) + { + memcpy(to, from, val_len); + error=sort_insert_key(sort_param,key_block,maria_ft_buf->lastkey, + HA_OFFSET_ERROR); + } + return error; + } + /* flushing second-level tree keyblocks */ + error=_ma_flush_pending_blocks(sort_param); + /* updating lastkey with second-level tree info */ + ft_intXstore(maria_ft_buf->lastkey+val_off, -maria_ft_buf->count); + _ma_dpointer(sort_info->info, maria_ft_buf->lastkey+val_off+HA_FT_WLEN, + share->state.key_root[sort_param->key]); + /* restoring first level tree data in sort_info/sort_param */ + sort_info->key_block=sort_info->key_block_end- sort_info->param->sort_key_blocks; + sort_param->keyinfo=share->keyinfo+sort_param->key; + share->state.key_root[sort_param->key]=HA_OFFSET_ERROR; + /* writing lastkey in first-level tree */ + return error ? error : + sort_insert_key(sort_param,sort_info->key_block, + maria_ft_buf->lastkey,HA_OFFSET_ERROR); +} + +static int sort_maria_ft_key_write(MARIA_SORT_PARAM *sort_param, const void *a) +{ + uint a_len, val_off, val_len, error; + uchar *p; + MARIA_SORT_INFO *sort_info= sort_param->sort_info; + SORT_FT_BUF *ft_buf= sort_info->ft_buf; + SORT_KEY_BLOCKS *key_block= sort_info->key_block; + + val_len=HA_FT_WLEN+sort_info->info->s->base.rec_reflength; + get_key_full_length_rdonly(a_len, (uchar *)a); + + if (!ft_buf) + { + /* + use two-level tree only if key_reflength fits in rec_reflength place + and row format is NOT static - for _ma_dpointer not to garble offsets + */ + if ((sort_info->info->s->base.key_reflength <= + sort_info->info->s->base.rec_reflength) && + (sort_info->info->s->options & + (HA_OPTION_PACK_RECORD | HA_OPTION_COMPRESS_RECORD))) + ft_buf= (SORT_FT_BUF *)my_malloc(sort_param->keyinfo->block_length + + sizeof(SORT_FT_BUF), MYF(MY_WME)); + + if (!ft_buf) + { + sort_param->key_write=sort_key_write; + return sort_key_write(sort_param, a); + } + sort_info->ft_buf= ft_buf; + goto word_init_ft_buf; /* no need to duplicate the code */ + } + get_key_full_length_rdonly(val_off, ft_buf->lastkey); + + if (ha_compare_text(sort_param->seg->charset, + ((uchar *)a)+1,a_len-1, + ft_buf->lastkey+1,val_off-1, 0, 0)==0) + { + if (!ft_buf->buf) /* store in second-level tree */ + { + ft_buf->count++; + return sort_insert_key(sort_param,key_block, + ((uchar *)a)+a_len, HA_OFFSET_ERROR); + } + + /* storing the key in the buffer. */ + memcpy (ft_buf->buf, (char *)a+a_len, val_len); + ft_buf->buf+=val_len; + if (ft_buf->buf < ft_buf->end) + return 0; + + /* converting to two-level tree */ + p=ft_buf->lastkey+val_off; + + while (key_block->inited) + key_block++; + sort_info->key_block=key_block; + sort_param->keyinfo=& sort_info->info->s->ft2_keyinfo; + ft_buf->count=(ft_buf->buf - p)/val_len; + + /* flushing buffer to second-level tree */ + for (error=0; !error && p < ft_buf->buf; p+= val_len) + error=sort_insert_key(sort_param,key_block,p,HA_OFFSET_ERROR); + ft_buf->buf=0; + return error; + } + + /* flushing buffer */ + if ((error=_ma_sort_ft_buf_flush(sort_param))) + return error; + +word_init_ft_buf: + a_len+=val_len; + memcpy(ft_buf->lastkey, a, a_len); + ft_buf->buf=ft_buf->lastkey+a_len; + /* + 32 is just a safety margin here + (at least max(val_len, sizeof(nod_flag)) should be there). + May be better performance could be achieved if we'd put + (sort_info->keyinfo->block_length-32)/XXX + instead. + TODO: benchmark the best value for XXX. + */ + ft_buf->end= ft_buf->lastkey+ (sort_param->keyinfo->block_length-32); + return 0; +} /* sort_maria_ft_key_write */ + + + /* get pointer to record from a key */ + +static my_off_t get_record_for_key(MARIA_HA *info, MARIA_KEYDEF *keyinfo, + uchar *key) +{ + return _ma_dpos(info,0,key + _ma_keylength(keyinfo,key)); +} /* get_record_for_key */ + + + /* Insert a key in sort-key-blocks */ + +static int sort_insert_key(MARIA_SORT_PARAM *sort_param, + register SORT_KEY_BLOCKS *key_block, uchar *key, + my_off_t prev_block) +{ + uint a_length,t_length,nod_flag; + my_off_t filepos,key_file_length; + uchar *anc_buff,*lastkey; + MARIA_KEY_PARAM s_temp; + MARIA_HA *info; + MARIA_KEYDEF *keyinfo=sort_param->keyinfo; + MARIA_SORT_INFO *sort_info= sort_param->sort_info; + HA_CHECK *param=sort_info->param; + DBUG_ENTER("sort_insert_key"); + + anc_buff=key_block->buff; + info=sort_info->info; + lastkey=key_block->lastkey; + nod_flag= (key_block == sort_info->key_block ? 0 : + info->s->base.key_reflength); + + if (!key_block->inited) + { + key_block->inited=1; + if (key_block == sort_info->key_block_end) + { + _ma_check_print_error(param,"To many key-block-levels; Try increasing sort_key_blocks"); + DBUG_RETURN(1); + } + a_length=2+nod_flag; + key_block->end_pos=anc_buff+2; + lastkey=0; /* No previous key in block */ + } + else + a_length=maria_getint(anc_buff); + + /* Save pointer to previous block */ + if (nod_flag) + _ma_kpointer(info,key_block->end_pos,prev_block); + + t_length=(*keyinfo->pack_key)(keyinfo,nod_flag, + (uchar*) 0,lastkey,lastkey,key, + &s_temp); + (*keyinfo->store_key)(keyinfo, key_block->end_pos+nod_flag,&s_temp); + a_length+=t_length; + maria_putint(anc_buff,a_length,nod_flag); + key_block->end_pos+=t_length; + if (a_length <= keyinfo->block_length) + { + VOID(_ma_move_key(keyinfo,key_block->lastkey,key)); + key_block->last_length=a_length-t_length; + DBUG_RETURN(0); + } + + /* Fill block with end-zero and write filled block */ + maria_putint(anc_buff,key_block->last_length,nod_flag); + bzero((byte*) anc_buff+key_block->last_length, + keyinfo->block_length- key_block->last_length); + key_file_length=info->state->key_file_length; + if ((filepos= _ma_new(info,keyinfo,DFLT_INIT_HITS)) == HA_OFFSET_ERROR) + DBUG_RETURN(1); + + /* If we read the page from the key cache, we have to write it back to it */ + if (key_file_length == info->state->key_file_length) + { + if (_ma_write_keypage(info, keyinfo, filepos, DFLT_INIT_HITS, anc_buff)) + DBUG_RETURN(1); + } + else if (my_pwrite(info->s->kfile,(byte*) anc_buff, + (uint) keyinfo->block_length,filepos, param->myf_rw)) + DBUG_RETURN(1); + DBUG_DUMP("buff",(byte*) anc_buff,maria_getint(anc_buff)); + + /* Write separator-key to block in next level */ + if (sort_insert_key(sort_param,key_block+1,key_block->lastkey,filepos)) + DBUG_RETURN(1); + + /* clear old block and write new key in it */ + key_block->inited=0; + DBUG_RETURN(sort_insert_key(sort_param, key_block,key,prev_block)); +} /* sort_insert_key */ + + + /* Delete record when we found a duplicated key */ + +static int sort_delete_record(MARIA_SORT_PARAM *sort_param) +{ + uint i; + int old_file,error; + uchar *key; + MARIA_SORT_INFO *sort_info=sort_param->sort_info; + HA_CHECK *param=sort_info->param; + MARIA_HA *info=sort_info->info; + DBUG_ENTER("sort_delete_record"); + + if ((param->testflag & (T_FORCE_UNIQUENESS|T_QUICK)) == T_QUICK) + { + _ma_check_print_error(param, + "Quick-recover aborted; Run recovery without switch -q or with switch -qq"); + DBUG_RETURN(1); + } + if (info->s->options & HA_OPTION_COMPRESS_RECORD) + { + _ma_check_print_error(param, + "Recover aborted; Can't run standard recovery on compressed tables with errors in data-file. Use switch 'mariachk --safe-recover' to fix it\n",stderr);; + DBUG_RETURN(1); + } + + old_file=info->dfile; + info->dfile=info->rec_cache.file; + if (sort_info->current_key) + { + key=info->lastkey+info->s->base.max_key_length; + if ((error=(*info->s->read_rnd)(info,sort_param->record,info->lastpos,0)) && + error != HA_ERR_RECORD_DELETED) + { + _ma_check_print_error(param,"Can't read record to be removed"); + info->dfile=old_file; + DBUG_RETURN(1); + } + + for (i=0 ; i < sort_info->current_key ; i++) + { + uint key_length= _ma_make_key(info,i,key,sort_param->record,info->lastpos); + if (_ma_ck_delete(info,i,key,key_length)) + { + _ma_check_print_error(param,"Can't delete key %d from record to be removed",i+1); + info->dfile=old_file; + DBUG_RETURN(1); + } + } + if (param->calc_checksum) + param->glob_crc-=(*info->s->calc_checksum)(info, sort_param->record); + } + error=flush_io_cache(&info->rec_cache) || (*info->s->delete_record)(info); + info->dfile=old_file; /* restore actual value */ + info->state->records--; + DBUG_RETURN(error); +} /* sort_delete_record */ + + /* Fix all pending blocks and flush everything to disk */ + +int _ma_flush_pending_blocks(MARIA_SORT_PARAM *sort_param) +{ + uint nod_flag,length; + my_off_t filepos,key_file_length; + SORT_KEY_BLOCKS *key_block; + MARIA_SORT_INFO *sort_info= sort_param->sort_info; + myf myf_rw=sort_info->param->myf_rw; + MARIA_HA *info=sort_info->info; + MARIA_KEYDEF *keyinfo=sort_param->keyinfo; + DBUG_ENTER("_ma_flush_pending_blocks"); + + filepos= HA_OFFSET_ERROR; /* if empty file */ + nod_flag=0; + for (key_block=sort_info->key_block ; key_block->inited ; key_block++) + { + key_block->inited=0; + length=maria_getint(key_block->buff); + if (nod_flag) + _ma_kpointer(info,key_block->end_pos,filepos); + key_file_length=info->state->key_file_length; + bzero((byte*) key_block->buff+length, keyinfo->block_length-length); + if ((filepos= _ma_new(info,keyinfo,DFLT_INIT_HITS)) == HA_OFFSET_ERROR) + DBUG_RETURN(1); + + /* If we read the page from the key cache, we have to write it back */ + if (key_file_length == info->state->key_file_length) + { + if (_ma_write_keypage(info, keyinfo, filepos, + DFLT_INIT_HITS, key_block->buff)) + DBUG_RETURN(1); + } + else if (my_pwrite(info->s->kfile,(byte*) key_block->buff, + (uint) keyinfo->block_length,filepos, myf_rw)) + DBUG_RETURN(1); + DBUG_DUMP("buff",(byte*) key_block->buff,length); + nod_flag=1; + } + info->s->state.key_root[sort_param->key]=filepos; /* Last is root for tree */ + DBUG_RETURN(0); +} /* _ma_flush_pending_blocks */ + + /* alloc space and pointers for key_blocks */ + +static SORT_KEY_BLOCKS *alloc_key_blocks(HA_CHECK *param, uint blocks, + uint buffer_length) +{ + reg1 uint i; + SORT_KEY_BLOCKS *block; + DBUG_ENTER("alloc_key_blocks"); + + if (!(block=(SORT_KEY_BLOCKS*) my_malloc((sizeof(SORT_KEY_BLOCKS)+ + buffer_length+IO_SIZE)*blocks, + MYF(0)))) + { + _ma_check_print_error(param,"Not enough memory for sort-key-blocks"); + return(0); + } + for (i=0 ; i < blocks ; i++) + { + block[i].inited=0; + block[i].buff=(uchar*) (block+blocks)+(buffer_length+IO_SIZE)*i; + } + DBUG_RETURN(block); +} /* alloc_key_blocks */ + + + /* Check if file is almost full */ + +int maria_test_if_almost_full(MARIA_HA *info) +{ + if (info->s->options & HA_OPTION_COMPRESS_RECORD) + return 0; + return (my_seek(info->s->kfile,0L,MY_SEEK_END,MYF(0))/10*9 > + (my_off_t) (info->s->base.max_key_file_length) || + my_seek(info->dfile,0L,MY_SEEK_END,MYF(0))/10*9 > + (my_off_t) info->s->base.max_data_file_length); +} + + /* Recreate table with bigger more alloced record-data */ + +int maria_recreate_table(HA_CHECK *param, MARIA_HA **org_info, char *filename) +{ + int error; + MARIA_HA info; + MARIA_SHARE share; + MARIA_KEYDEF *keyinfo,*key,*key_end; + HA_KEYSEG *keysegs,*keyseg; + MARIA_COLUMNDEF *recdef,*rec,*end; + MARIA_UNIQUEDEF *uniquedef,*u_ptr,*u_end; + MARIA_STATUS_INFO status_info; + uint unpack,key_parts; + ha_rows max_records; + ulonglong file_length,tmp_length; + MARIA_CREATE_INFO create_info; + + error=1; /* Default error */ + info= **org_info; + status_info= (*org_info)->state[0]; + info.state= &status_info; + share= *(*org_info)->s; + unpack= (share.options & HA_OPTION_COMPRESS_RECORD) && + (param->testflag & T_UNPACK); + if (!(keyinfo=(MARIA_KEYDEF*) my_alloca(sizeof(MARIA_KEYDEF)*share.base.keys))) + return 0; + memcpy((byte*) keyinfo,(byte*) share.keyinfo, + (size_t) (sizeof(MARIA_KEYDEF)*share.base.keys)); + + key_parts= share.base.all_key_parts; + if (!(keysegs=(HA_KEYSEG*) my_alloca(sizeof(HA_KEYSEG)* + (key_parts+share.base.keys)))) + { + my_afree((gptr) keyinfo); + return 1; + } + if (!(recdef=(MARIA_COLUMNDEF*) + my_alloca(sizeof(MARIA_COLUMNDEF)*(share.base.fields+1)))) + { + my_afree((gptr) keyinfo); + my_afree((gptr) keysegs); + return 1; + } + if (!(uniquedef=(MARIA_UNIQUEDEF*) + my_alloca(sizeof(MARIA_UNIQUEDEF)*(share.state.header.uniques+1)))) + { + my_afree((gptr) recdef); + my_afree((gptr) keyinfo); + my_afree((gptr) keysegs); + return 1; + } + + /* Copy the column definitions */ + memcpy((byte*) recdef,(byte*) share.rec, + (size_t) (sizeof(MARIA_COLUMNDEF)*(share.base.fields+1))); + for (rec=recdef,end=recdef+share.base.fields; rec != end ; rec++) + { + if (unpack && !(share.options & HA_OPTION_PACK_RECORD) && + rec->type != FIELD_BLOB && + rec->type != FIELD_VARCHAR && + rec->type != FIELD_CHECK) + rec->type=(int) FIELD_NORMAL; + } + + /* Change the new key to point at the saved key segments */ + memcpy((byte*) keysegs,(byte*) share.keyparts, + (size_t) (sizeof(HA_KEYSEG)*(key_parts+share.base.keys+ + share.state.header.uniques))); + keyseg=keysegs; + for (key=keyinfo,key_end=keyinfo+share.base.keys; key != key_end ; key++) + { + key->seg=keyseg; + for (; keyseg->type ; keyseg++) + { + if (param->language) + keyseg->language=param->language; /* change language */ + } + keyseg++; /* Skip end pointer */ + } + + /* Copy the unique definitions and change them to point at the new key + segments*/ + memcpy((byte*) uniquedef,(byte*) share.uniqueinfo, + (size_t) (sizeof(MARIA_UNIQUEDEF)*(share.state.header.uniques))); + for (u_ptr=uniquedef,u_end=uniquedef+share.state.header.uniques; + u_ptr != u_end ; u_ptr++) + { + u_ptr->seg=keyseg; + keyseg+=u_ptr->keysegs+1; + } + if (share.options & HA_OPTION_COMPRESS_RECORD) + share.base.records=max_records=info.state->records; + else if (share.base.min_pack_length) + max_records=(ha_rows) (my_seek(info.dfile,0L,MY_SEEK_END,MYF(0)) / + (ulong) share.base.min_pack_length); + else + max_records=0; + unpack= (share.options & HA_OPTION_COMPRESS_RECORD) && + (param->testflag & T_UNPACK); + share.options&= ~HA_OPTION_TEMP_COMPRESS_RECORD; + + file_length=(ulonglong) my_seek(info.dfile,0L,MY_SEEK_END,MYF(0)); + tmp_length= file_length+file_length/10; + set_if_bigger(file_length,param->max_data_file_length); + set_if_bigger(file_length,tmp_length); + set_if_bigger(file_length,(ulonglong) share.base.max_data_file_length); + + VOID(maria_close(*org_info)); + bzero((char*) &create_info,sizeof(create_info)); + create_info.max_rows=max(max_records,share.base.records); + create_info.reloc_rows=share.base.reloc; + create_info.old_options=(share.options | + (unpack ? HA_OPTION_TEMP_COMPRESS_RECORD : 0)); + + create_info.data_file_length=file_length; + create_info.auto_increment=share.state.auto_increment; + create_info.language = (param->language ? param->language : + share.state.header.language); + create_info.key_file_length= status_info.key_file_length; + /* We don't have to handle symlinks here because we are using + HA_DONT_TOUCH_DATA */ + if (maria_create(filename, + share.base.keys - share.state.header.uniques, + keyinfo, share.base.fields, recdef, + share.state.header.uniques, uniquedef, + &create_info, + HA_DONT_TOUCH_DATA)) + { + _ma_check_print_error(param,"Got error %d when trying to recreate indexfile",my_errno); + goto end; + } + *org_info=maria_open(filename,O_RDWR, + (param->testflag & T_WAIT_FOREVER) ? HA_OPEN_WAIT_IF_LOCKED : + (param->testflag & T_DESCRIPT) ? HA_OPEN_IGNORE_IF_LOCKED : + HA_OPEN_ABORT_IF_LOCKED); + if (!*org_info) + { + _ma_check_print_error(param,"Got error %d when trying to open re-created indexfile", + my_errno); + goto end; + } + /* We are modifing */ + (*org_info)->s->options&= ~HA_OPTION_READ_ONLY_DATA; + VOID(_ma_readinfo(*org_info,F_WRLCK,0)); + (*org_info)->state->records=info.state->records; + if (share.state.create_time) + (*org_info)->s->state.create_time=share.state.create_time; + (*org_info)->s->state.unique=(*org_info)->this_unique= + share.state.unique; + (*org_info)->state->checksum=info.state->checksum; + (*org_info)->state->del=info.state->del; + (*org_info)->s->state.dellink=share.state.dellink; + (*org_info)->state->empty=info.state->empty; + (*org_info)->state->data_file_length=info.state->data_file_length; + if (maria_update_state_info(param,*org_info,UPDATE_TIME | UPDATE_STAT | + UPDATE_OPEN_COUNT)) + goto end; + error=0; +end: + my_afree((gptr) uniquedef); + my_afree((gptr) keyinfo); + my_afree((gptr) recdef); + my_afree((gptr) keysegs); + return error; +} + + + /* write suffix to data file if neaded */ + +int maria_write_data_suffix(MARIA_SORT_INFO *sort_info, my_bool fix_datafile) +{ + MARIA_HA *info=sort_info->info; + + if (info->s->options & HA_OPTION_COMPRESS_RECORD && fix_datafile) + { + char buff[MEMMAP_EXTRA_MARGIN]; + bzero(buff,sizeof(buff)); + if (my_b_write(&info->rec_cache,buff,sizeof(buff))) + { + _ma_check_print_error(sort_info->param, + "%d when writing to datafile",my_errno); + return 1; + } + sort_info->param->read_cache.end_of_file+=sizeof(buff); + } + return 0; +} + + /* Update state and mariachk_time of indexfile */ + +int maria_update_state_info(HA_CHECK *param, MARIA_HA *info,uint update) +{ + MARIA_SHARE *share=info->s; + + if (update & UPDATE_OPEN_COUNT) + { + share->state.open_count=0; + share->global_changed=0; + } + if (update & UPDATE_STAT) + { + uint i, key_parts= mi_uint2korr(share->state.header.key_parts); + share->state.rec_per_key_rows=info->state->records; + share->state.changed&= ~STATE_NOT_ANALYZED; + if (info->state->records) + { + for (i=0; istate.rec_per_key_part[i]=param->rec_per_key_part[i])) + share->state.changed|= STATE_NOT_ANALYZED; + } + } + } + if (update & (UPDATE_STAT | UPDATE_SORT | UPDATE_TIME | UPDATE_AUTO_INC)) + { + if (update & UPDATE_TIME) + { + share->state.check_time= (long) time((time_t*) 0); + if (!share->state.create_time) + share->state.create_time=share->state.check_time; + } + /* + When tables are locked we haven't synched the share state and the + real state for a while so we better do it here before synching + the share state to disk. Only when table is write locked is it + necessary to perform this synch. + */ + if (info->lock_type == F_WRLCK) + share->state.state= *info->state; + if (_ma_state_info_write(share->kfile,&share->state,1+2)) + goto err; + share->changed=0; + } + { /* Force update of status */ + int error; + uint r_locks=share->r_locks,w_locks=share->w_locks; + share->r_locks= share->w_locks= share->tot_locks= 0; + error= _ma_writeinfo(info,WRITEINFO_NO_UNLOCK); + share->r_locks=r_locks; + share->w_locks=w_locks; + share->tot_locks=r_locks+w_locks; + if (!error) + return 0; + } +err: + _ma_check_print_error(param,"%d when updating keyfile",my_errno); + return 1; +} + + /* + Update auto increment value for a table + When setting the 'repair_only' flag we only want to change the + old auto_increment value if its wrong (smaller than some given key). + The reason is that we shouldn't change the auto_increment value + for a table without good reason when only doing a repair; If the + user have inserted and deleted rows, the auto_increment value + may be bigger than the biggest current row and this is ok. + + If repair_only is not set, we will update the flag to the value in + param->auto_increment is bigger than the biggest key. + */ + +void _ma_update_auto_increment_key(HA_CHECK *param, MARIA_HA *info, + my_bool repair_only) +{ + byte *record; + if (!info->s->base.auto_key || + ! maria_is_key_active(info->s->state.key_map, info->s->base.auto_key - 1)) + { + if (!(param->testflag & T_VERY_SILENT)) + _ma_check_print_info(param, + "Table: %s doesn't have an auto increment key\n", + param->isam_file_name); + return; + } + if (!(param->testflag & T_SILENT) && + !(param->testflag & T_REP)) + printf("Updating MARIA file: %s\n", param->isam_file_name); + /* + We have to use an allocated buffer instead of info->rec_buff as + _ma_put_key_in_record() may use info->rec_buff + */ + if (!(record= (byte*) my_malloc((uint) info->s->base.pack_reclength, + MYF(0)))) + { + _ma_check_print_error(param,"Not enough memory for extra record"); + return; + } + + maria_extra(info,HA_EXTRA_KEYREAD,0); + if (maria_rlast(info, record, info->s->base.auto_key-1)) + { + if (my_errno != HA_ERR_END_OF_FILE) + { + maria_extra(info,HA_EXTRA_NO_KEYREAD,0); + my_free((char*) record, MYF(0)); + _ma_check_print_error(param,"%d when reading last record",my_errno); + return; + } + if (!repair_only) + info->s->state.auto_increment=param->auto_increment_value; + } + else + { + ulonglong auto_increment= (repair_only ? info->s->state.auto_increment : + param->auto_increment_value); + info->s->state.auto_increment=0; + _ma_update_auto_increment(info, record); + set_if_bigger(info->s->state.auto_increment,auto_increment); + } + maria_extra(info,HA_EXTRA_NO_KEYREAD,0); + my_free((char*) record, MYF(0)); + maria_update_state_info(param, info, UPDATE_AUTO_INC); + return; +} + + +/* + Update statistics for each part of an index + + SYNOPSIS + maria_update_key_parts() + keyinfo IN Index information (only key->keysegs used) + rec_per_key_part OUT Store statistics here + unique IN Array of (#distinct tuples) + notnull_tuples IN Array of (#tuples), or NULL + records Number of records in the table + + DESCRIPTION + This function is called produce index statistics values from unique and + notnull_tuples arrays after these arrays were produced with sequential + index scan (the scan is done in two places: chk_index() and + sort_key_write()). + + This function handles all 3 index statistics collection methods. + + Unique is an array: + unique[0]= (#different values of {keypart1}) - 1 + unique[1]= (#different values of {keypart1,keypart2} tuple)-unique[0]-1 + ... + + For MI_STATS_METHOD_IGNORE_NULLS method, notnull_tuples is an array too: + notnull_tuples[0]= (#of {keypart1} tuples such that keypart1 is not NULL) + notnull_tuples[1]= (#of {keypart1,keypart2} tuples such that all + keypart{i} are not NULL) + ... + For all other statistics collection methods notnull_tuples==NULL. + + Output is an array: + rec_per_key_part[k] = + = E(#records in the table such that keypart_1=c_1 AND ... AND + keypart_k=c_k for arbitrary constants c_1 ... c_k) + + = {assuming that values have uniform distribution and index contains all + tuples from the domain (or that {c_1, ..., c_k} tuple is choosen from + index tuples} + + = #tuples-in-the-index / #distinct-tuples-in-the-index. + + The #tuples-in-the-index and #distinct-tuples-in-the-index have different + meaning depending on which statistics collection method is used: + + MI_STATS_METHOD_* how are nulls compared? which tuples are counted? + NULLS_EQUAL NULL == NULL all tuples in table + NULLS_NOT_EQUAL NULL != NULL all tuples in table + IGNORE_NULLS n/a tuples that don't have NULLs +*/ + +void maria_update_key_parts(MARIA_KEYDEF *keyinfo, ulong *rec_per_key_part, + ulonglong *unique, ulonglong *notnull, + ulonglong records) +{ + ulonglong count=0,tmp, unique_tuples; + ulonglong tuples= records; + uint parts; + for (parts=0 ; parts < keyinfo->keysegs ; parts++) + { + count+=unique[parts]; + unique_tuples= count + 1; + if (notnull) + { + tuples= notnull[parts]; + /* + #(unique_tuples not counting tuples with NULLs) = + #(unique_tuples counting tuples with NULLs as different) - + #(tuples with NULLs) + */ + unique_tuples -= (records - notnull[parts]); + } + + if (unique_tuples == 0) + tmp= 1; + else if (count == 0) + tmp= tuples; /* 1 unique tuple */ + else + tmp= (tuples + unique_tuples/2) / unique_tuples; + + /* + for some weird keys (e.g. FULLTEXT) tmp can be <1 here. + let's ensure it is not + */ + set_if_bigger(tmp,1); + if (tmp >= (ulonglong) ~(ulong) 0) + tmp=(ulonglong) ~(ulong) 0; + + *rec_per_key_part=(ulong) tmp; + rec_per_key_part++; + } +} + + +static ha_checksum maria_byte_checksum(const byte *buf, uint length) +{ + ha_checksum crc; + const byte *end=buf+length; + for (crc=0; buf != end; buf++) + crc=((crc << 1) + *((uchar*) buf)) + + test(crc & (((ha_checksum) 1) << (8*sizeof(ha_checksum)-1))); + return crc; +} + +static my_bool maria_too_big_key_for_sort(MARIA_KEYDEF *key, ha_rows rows) +{ + uint key_maxlength=key->maxlength; + if (key->flag & HA_FULLTEXT) + { + uint ft_max_word_len_for_sort=FT_MAX_WORD_LEN_FOR_SORT* + key->seg->charset->mbmaxlen; + key_maxlength+=ft_max_word_len_for_sort-HA_FT_MAXBYTELEN; + } + return (key->flag & HA_SPATIAL) || + (key->flag & (HA_BINARY_PACK_KEY | HA_VAR_LENGTH_KEY | HA_FULLTEXT) && + ((ulonglong) rows * key_maxlength > + (ulonglong) maria_max_temp_length)); +} + +/* + Deactivate all not unique index that can be recreated fast + These include packed keys on which sorting will use more temporary + space than the max allowed file length or for which the unpacked keys + will take much more space than packed keys. + Note that 'rows' may be zero for the case when we don't know how many + rows we will put into the file. + */ + +void maria_disable_non_unique_index(MARIA_HA *info, ha_rows rows) +{ + MARIA_SHARE *share=info->s; + MARIA_KEYDEF *key=share->keyinfo; + uint i; + + DBUG_ASSERT(info->state->records == 0 && + (!rows || rows >= MARIA_MIN_ROWS_TO_DISABLE_INDEXES)); + for (i=0 ; i < share->base.keys ; i++,key++) + { + if (!(key->flag & (HA_NOSAME | HA_SPATIAL | HA_AUTO_KEY)) && + ! maria_too_big_key_for_sort(key,rows) && info->s->base.auto_key != i+1) + { + maria_clear_key_active(share->state.key_map, i); + info->update|= HA_STATE_CHANGED; + } + } +} + + +/* + Return TRUE if we can use repair by sorting + One can set the force argument to force to use sorting + even if the temporary file would be quite big! +*/ + +my_bool maria_test_if_sort_rep(MARIA_HA *info, ha_rows rows, + ulonglong key_map, my_bool force) +{ + MARIA_SHARE *share=info->s; + MARIA_KEYDEF *key=share->keyinfo; + uint i; + + /* + maria_repair_by_sort only works if we have at least one key. If we don't + have any keys, we should use the normal repair. + */ + if (! maria_is_any_key_active(key_map)) + return FALSE; /* Can't use sort */ + for (i=0 ; i < share->base.keys ; i++,key++) + { + if (!force && maria_too_big_key_for_sort(key,rows)) + return FALSE; + } + return TRUE; +} + + +static void +set_data_file_type(MARIA_SORT_INFO *sort_info, MARIA_SHARE *share) +{ + if ((sort_info->new_data_file_type=share->data_file_type) == + COMPRESSED_RECORD && sort_info->param->testflag & T_UNPACK) + { + MARIA_SHARE tmp; + + if (share->options & HA_OPTION_PACK_RECORD) + sort_info->new_data_file_type = DYNAMIC_RECORD; + else + sort_info->new_data_file_type = STATIC_RECORD; + + /* Set delete_function for sort_delete_record() */ + memcpy((char*) &tmp, share, sizeof(*share)); + tmp.options= ~HA_OPTION_COMPRESS_RECORD; + _ma_setup_functions(&tmp); + share->delete_record=tmp.delete_record; + } +} diff --git a/storage/maria/ma_checksum.c b/storage/maria/ma_checksum.c new file mode 100644 index 00000000000..054873706a4 --- /dev/null +++ b/storage/maria/ma_checksum.c @@ -0,0 +1,65 @@ +/* Copyright (C) 2006 MySQL AB & MySQL Finland AB & TCX DataKonsult AB + + 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; either version 2 of the License, or + (at your option) any later version. + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ + +/* Calculate a checksum for a row */ + +#include "maria_def.h" + +ha_checksum _ma_checksum(MARIA_HA *info, const byte *buf) +{ + uint i; + ha_checksum crc=0; + MARIA_COLUMNDEF *rec=info->s->rec; + + for (i=info->s->base.fields ; i-- ; buf+=(rec++)->length) + { + const byte *pos; + ulong length; + switch (rec->type) { + case FIELD_BLOB: + { + length= _ma_calc_blob_length(rec->length- + maria_portable_sizeof_char_ptr, + buf); + memcpy((char*) &pos, buf+rec->length- maria_portable_sizeof_char_ptr, + sizeof(char*)); + break; + } + case FIELD_VARCHAR: + { + uint pack_length= HA_VARCHAR_PACKLENGTH(rec->length-1); + if (pack_length == 1) + length= (ulong) *(uchar*) buf; + else + length= uint2korr(buf); + pos= buf+pack_length; + break; + } + default: + length=rec->length; + pos=buf; + break; + } + crc=my_checksum(crc, pos ? pos : "", length); + } + return crc; +} + + +ha_checksum _ma_static_checksum(MARIA_HA *info, const byte *pos) +{ + return my_checksum(0, pos, info->s->base.reclength); +} diff --git a/storage/maria/ma_close.c b/storage/maria/ma_close.c new file mode 100644 index 00000000000..f3a1b2ba261 --- /dev/null +++ b/storage/maria/ma_close.c @@ -0,0 +1,124 @@ +/* Copyright (C) 2006 MySQL AB & MySQL Finland AB & TCX DataKonsult AB + + 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; either version 2 of the License, or + (at your option) any later version. + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ + +/* close a isam-database */ +/* + TODO: + We need to have a separate mutex on the closed file to allow other threads + to open other files during the time we flush the cache and close this file +*/ + +#include "maria_def.h" + +int maria_close(register MARIA_HA *info) +{ + int error=0,flag; + MARIA_SHARE *share=info->s; + DBUG_ENTER("maria_close"); + DBUG_PRINT("enter",("base: %lx reopen: %u locks: %u", + info,(uint) share->reopen, (uint) share->tot_locks)); + + pthread_mutex_lock(&THR_LOCK_maria); + if (info->lock_type == F_EXTRA_LCK) + info->lock_type=F_UNLCK; /* HA_EXTRA_NO_USER_CHANGE */ + + if (share->reopen == 1 && share->kfile >= 0) + _ma_decrement_open_count(info); + + if (info->lock_type != F_UNLCK) + { + if (maria_lock_database(info,F_UNLCK)) + error=my_errno; + } + pthread_mutex_lock(&share->intern_lock); + + if (share->options & HA_OPTION_READ_ONLY_DATA) + { + share->r_locks--; + share->tot_locks--; + } + if (info->opt_flag & (READ_CACHE_USED | WRITE_CACHE_USED)) + { + if (end_io_cache(&info->rec_cache)) + error=my_errno; + info->opt_flag&= ~(READ_CACHE_USED | WRITE_CACHE_USED); + } + flag= !--share->reopen; + maria_open_list=list_delete(maria_open_list,&info->open_list); + pthread_mutex_unlock(&share->intern_lock); + + my_free(_ma_get_rec_buff_ptr(info, info->rec_buff), MYF(MY_ALLOW_ZERO_PTR)); + if (flag) + { + if (share->kfile >= 0 && + flush_key_blocks(share->key_cache, share->kfile, + share->temporary ? FLUSH_IGNORE_CHANGED : + FLUSH_RELEASE)) + error=my_errno; + if (share->kfile >= 0) + { + /* + If we are crashed, we can safely flush the current state as it will + not change the crashed state. + We can NOT write the state in other cases as other threads + may be using the file at this point + */ + if (share->mode != O_RDONLY && maria_is_crashed(info)) + _ma_state_info_write(share->kfile, &share->state, 1); + if (my_close(share->kfile,MYF(0))) + error = my_errno; + } +#ifdef HAVE_MMAP + if (share->file_map) + _ma_unmap_file(info); +#endif + if (share->decode_trees) + { + my_free((gptr) share->decode_trees,MYF(0)); + my_free((gptr) share->decode_tables,MYF(0)); + } +#ifdef THREAD + thr_lock_delete(&share->lock); + VOID(pthread_mutex_destroy(&share->intern_lock)); + { + int i,keys; + keys = share->state.header.keys; + VOID(rwlock_destroy(&share->mmap_lock)); + for(i=0; ikey_root_lock[i])); + } + } +#endif + my_free((gptr) info->s,MYF(0)); + } + pthread_mutex_unlock(&THR_LOCK_maria); + if (info->ftparser_param) + { + my_free((gptr)info->ftparser_param, MYF(0)); + info->ftparser_param= 0; + } + if (info->dfile >= 0 && my_close(info->dfile,MYF(0))) + error = my_errno; + + maria_log_command(MARIA_LOG_CLOSE,info,NULL,0,error); + my_free((gptr) info,MYF(0)); + + if (error) + { + DBUG_RETURN(my_errno=error); + } + DBUG_RETURN(0); +} /* maria_close */ diff --git a/storage/maria/ma_create.c b/storage/maria/ma_create.c new file mode 100644 index 00000000000..b15b5b0ae02 --- /dev/null +++ b/storage/maria/ma_create.c @@ -0,0 +1,816 @@ +/* Copyright (C) 2006 MySQL AB & MySQL Finland AB & TCX DataKonsult AB + + 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; either version 2 of the License, or + (at your option) any later version. + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ + +/* Create a MARIA table */ + +#include "ma_ftdefs.h" +#include "ma_sp_defs.h" + +#if defined(MSDOS) || defined(__WIN__) +#ifdef __WIN__ +#include +#else +#include /* Prototype for getpid */ +#endif +#endif +#include + + /* + ** Old options is used when recreating database, from isamchk + */ + +int maria_create(const char *name,uint keys,MARIA_KEYDEF *keydefs, + uint columns, MARIA_COLUMNDEF *recinfo, + uint uniques, MARIA_UNIQUEDEF *uniquedefs, + MARIA_CREATE_INFO *ci,uint flags) +{ + register uint i,j; + File dfile,file; + int errpos,save_errno, create_mode= O_RDWR | O_TRUNC; + myf create_flag; + uint fields,length,max_key_length,packed,pointer,real_length_diff, + key_length,info_length,key_segs,options,min_key_length_skip, + base_pos,long_varchar_count,varchar_length, + max_key_block_length,unique_key_parts,fulltext_keys,offset; + ulong reclength, real_reclength,min_pack_length; + char filename[FN_REFLEN],linkname[FN_REFLEN], *linkname_ptr; + ulong pack_reclength; + ulonglong tot_length,max_rows, tmp; + enum en_fieldtype type; + MARIA_SHARE share; + MARIA_KEYDEF *keydef,tmp_keydef; + MARIA_UNIQUEDEF *uniquedef; + HA_KEYSEG *keyseg,tmp_keyseg; + MARIA_COLUMNDEF *rec; + ulong *rec_per_key_part; + my_off_t key_root[HA_MAX_POSSIBLE_KEY],key_del[MARIA_MAX_KEY_BLOCK_SIZE]; + MARIA_CREATE_INFO tmp_create_info; + DBUG_ENTER("maria_create"); + + if (!ci) + { + bzero((char*) &tmp_create_info,sizeof(tmp_create_info)); + ci=&tmp_create_info; + } + + if (keys + uniques > MARIA_MAX_KEY || columns == 0) + { + DBUG_RETURN(my_errno=HA_WRONG_CREATE_OPTION); + } + LINT_INIT(dfile); + LINT_INIT(file); + errpos=0; + options=0; + bzero((byte*) &share,sizeof(share)); + + if (flags & HA_DONT_TOUCH_DATA) + { + if (!(ci->old_options & HA_OPTION_TEMP_COMPRESS_RECORD)) + options=ci->old_options & + (HA_OPTION_COMPRESS_RECORD | HA_OPTION_PACK_RECORD | + HA_OPTION_READ_ONLY_DATA | HA_OPTION_CHECKSUM | + HA_OPTION_TMP_TABLE | HA_OPTION_DELAY_KEY_WRITE); + else + options=ci->old_options & + (HA_OPTION_CHECKSUM | HA_OPTION_TMP_TABLE | HA_OPTION_DELAY_KEY_WRITE); + } + + if (ci->reloc_rows > ci->max_rows) + ci->reloc_rows=ci->max_rows; /* Check if wrong parameter */ + + if (!(rec_per_key_part= + (ulong*) my_malloc((keys + uniques)*HA_MAX_KEY_SEG*sizeof(long), + MYF(MY_WME | MY_ZEROFILL)))) + DBUG_RETURN(my_errno); + + /* Start by checking fields and field-types used */ + + reclength=varchar_length=long_varchar_count=packed= + min_pack_length=pack_reclength=0; + for (rec=recinfo, fields=0 ; + fields != columns ; + rec++,fields++) + { + reclength+=rec->length; + if ((type=(enum en_fieldtype) rec->type) != FIELD_NORMAL && + type != FIELD_CHECK) + { + packed++; + if (type == FIELD_BLOB) + { + share.base.blobs++; + if (pack_reclength != INT_MAX32) + { + if (rec->length == 4+maria_portable_sizeof_char_ptr) + pack_reclength= INT_MAX32; + else + pack_reclength+=(1 << ((rec->length-maria_portable_sizeof_char_ptr)*8)); /* Max blob length */ + } + } + else if (type == FIELD_SKIP_PRESPACE || + type == FIELD_SKIP_ENDSPACE) + { + if (pack_reclength != INT_MAX32) + pack_reclength+= rec->length > 255 ? 2 : 1; + min_pack_length++; + } + else if (type == FIELD_VARCHAR) + { + varchar_length+= rec->length-1; /* Used for min_pack_length */ + packed--; + pack_reclength++; + min_pack_length++; + /* We must test for 257 as length includes pack-length */ + if (test(rec->length >= 257)) + { + long_varchar_count++; + pack_reclength+= 2; /* May be packed on 3 bytes */ + } + } + else if (type != FIELD_SKIP_ZERO) + { + min_pack_length+=rec->length; + packed--; /* Not a pack record type */ + } + } + else /* FIELD_NORMAL */ + min_pack_length+=rec->length; + } + if ((packed & 7) == 1) + { /* Bad packing, try to remove a zero-field */ + while (rec != recinfo) + { + rec--; + if (rec->type == (int) FIELD_SKIP_ZERO && rec->length == 1) + { + rec->type=(int) FIELD_NORMAL; + packed--; + min_pack_length++; + break; + } + } + } + + if (packed || (flags & HA_PACK_RECORD)) + options|=HA_OPTION_PACK_RECORD; /* Must use packed records */ + /* We can't use checksum with static length rows */ + if (!(options & HA_OPTION_PACK_RECORD)) + options&= ~HA_OPTION_CHECKSUM; + if (!(options & (HA_OPTION_PACK_RECORD | HA_OPTION_COMPRESS_RECORD))) + min_pack_length+= varchar_length; + if (flags & HA_CREATE_TMP_TABLE) + { + options|= HA_OPTION_TMP_TABLE; + create_mode|= O_EXCL | O_NOFOLLOW; + } + if (flags & HA_CREATE_CHECKSUM || (options & HA_OPTION_CHECKSUM)) + { + options|= HA_OPTION_CHECKSUM; + min_pack_length++; + } + if (flags & HA_CREATE_DELAY_KEY_WRITE) + options|= HA_OPTION_DELAY_KEY_WRITE; + if (flags & HA_CREATE_RELIES_ON_SQL_LAYER) + options|= HA_OPTION_RELIES_ON_SQL_LAYER; + + packed=(packed+7)/8; + if (pack_reclength != INT_MAX32) + pack_reclength+= reclength+packed + + test(test_all_bits(options, HA_OPTION_CHECKSUM | HA_PACK_RECORD)); + min_pack_length+=packed; + + if (!ci->data_file_length && ci->max_rows) + { + if (pack_reclength == INT_MAX32 || + (~(ulonglong) 0)/ci->max_rows < (ulonglong) pack_reclength) + ci->data_file_length= ~(ulonglong) 0; + else + ci->data_file_length=(ulonglong) ci->max_rows*pack_reclength; + } + else if (!ci->max_rows) + ci->max_rows=(ha_rows) (ci->data_file_length/(min_pack_length + + ((options & HA_OPTION_PACK_RECORD) ? + 3 : 0))); + + if (options & (HA_OPTION_COMPRESS_RECORD | HA_OPTION_PACK_RECORD)) + pointer=maria_get_pointer_length(ci->data_file_length,maria_data_pointer_size); + else + pointer=maria_get_pointer_length(ci->max_rows,maria_data_pointer_size); + if (!(max_rows=(ulonglong) ci->max_rows)) + max_rows= ((((ulonglong) 1 << (pointer*8)) -1) / min_pack_length); + + + real_reclength=reclength; + if (!(options & (HA_OPTION_COMPRESS_RECORD | HA_OPTION_PACK_RECORD))) + { + if (reclength <= pointer) + reclength=pointer+1; /* reserve place for delete link */ + } + else + reclength+= long_varchar_count; /* We need space for varchar! */ + + max_key_length=0; tot_length=0 ; key_segs=0; + fulltext_keys=0; + max_key_block_length=0; + share.state.rec_per_key_part=rec_per_key_part; + share.state.key_root=key_root; + share.state.key_del=key_del; + if (uniques) + { + max_key_block_length= maria_block_size; + max_key_length= MARIA_UNIQUE_HASH_LENGTH + pointer; + } + + for (i=0, keydef=keydefs ; i < keys ; i++ , keydef++) + { + + share.state.key_root[i]= HA_OFFSET_ERROR; + min_key_length_skip=length=real_length_diff=0; + key_length=pointer; + if (keydef->flag & HA_SPATIAL) + { +#ifdef HAVE_SPATIAL + /* BAR TODO to support 3D and more dimensions in the future */ + uint sp_segs=SPDIMS*2; + keydef->flag=HA_SPATIAL; + + if (flags & HA_DONT_TOUCH_DATA) + { + /* + called by mariachk - i.e. table structure was taken from + MYI file and SPATIAL key *does have* additional sp_segs keysegs. + keydef->seg here points right at the GEOMETRY segment, + so we only need to decrease keydef->keysegs. + (see maria_recreate_table() in _ma_check.c) + */ + keydef->keysegs-=sp_segs-1; + } + + for (j=0, keyseg=keydef->seg ; (int) j < keydef->keysegs ; + j++, keyseg++) + { + if (keyseg->type != HA_KEYTYPE_BINARY && + keyseg->type != HA_KEYTYPE_VARBINARY1 && + keyseg->type != HA_KEYTYPE_VARBINARY2) + { + my_errno=HA_WRONG_CREATE_OPTION; + goto err; + } + } + keydef->keysegs+=sp_segs; + key_length+=SPLEN*sp_segs; + length++; /* At least one length byte */ + min_key_length_skip+=SPLEN*2*SPDIMS; +#else + my_errno= HA_ERR_UNSUPPORTED; + goto err; +#endif /*HAVE_SPATIAL*/ + } + else if (keydef->flag & HA_FULLTEXT) + { + keydef->flag=HA_FULLTEXT | HA_PACK_KEY | HA_VAR_LENGTH_KEY; + options|=HA_OPTION_PACK_KEYS; /* Using packed keys */ + + for (j=0, keyseg=keydef->seg ; (int) j < keydef->keysegs ; + j++, keyseg++) + { + if (keyseg->type != HA_KEYTYPE_TEXT && + keyseg->type != HA_KEYTYPE_VARTEXT1 && + keyseg->type != HA_KEYTYPE_VARTEXT2) + { + my_errno=HA_WRONG_CREATE_OPTION; + goto err; + } + if (!(keyseg->flag & HA_BLOB_PART) && + (keyseg->type == HA_KEYTYPE_VARTEXT1 || + keyseg->type == HA_KEYTYPE_VARTEXT2)) + { + /* Make a flag that this is a VARCHAR */ + keyseg->flag|= HA_VAR_LENGTH_PART; + /* Store in bit_start number of bytes used to pack the length */ + keyseg->bit_start= ((keyseg->type == HA_KEYTYPE_VARTEXT1)? + 1 : 2); + } + } + + fulltext_keys++; + key_length+= HA_FT_MAXBYTELEN+HA_FT_WLEN; + length++; /* At least one length byte */ + min_key_length_skip+=HA_FT_MAXBYTELEN; + real_length_diff=HA_FT_MAXBYTELEN-FT_MAX_WORD_LEN_FOR_SORT; + } + else + { + /* Test if prefix compression */ + if (keydef->flag & HA_PACK_KEY) + { + /* Can't use space_compression on number keys */ + if ((keydef->seg[0].flag & HA_SPACE_PACK) && + keydef->seg[0].type == (int) HA_KEYTYPE_NUM) + keydef->seg[0].flag&= ~HA_SPACE_PACK; + + /* Only use HA_PACK_KEY when first segment is a variable length key */ + if (!(keydef->seg[0].flag & (HA_SPACE_PACK | HA_BLOB_PART | + HA_VAR_LENGTH_PART))) + { + /* pack relative to previous key */ + keydef->flag&= ~HA_PACK_KEY; + keydef->flag|= HA_BINARY_PACK_KEY | HA_VAR_LENGTH_KEY; + } + else + { + keydef->seg[0].flag|=HA_PACK_KEY; /* for easyer intern test */ + keydef->flag|=HA_VAR_LENGTH_KEY; + options|=HA_OPTION_PACK_KEYS; /* Using packed keys */ + } + } + if (keydef->flag & HA_BINARY_PACK_KEY) + options|=HA_OPTION_PACK_KEYS; /* Using packed keys */ + + if (keydef->flag & HA_AUTO_KEY && ci->with_auto_increment) + share.base.auto_key=i+1; + for (j=0, keyseg=keydef->seg ; j < keydef->keysegs ; j++, keyseg++) + { + /* numbers are stored with high by first to make compression easier */ + switch (keyseg->type) { + case HA_KEYTYPE_SHORT_INT: + case HA_KEYTYPE_LONG_INT: + case HA_KEYTYPE_FLOAT: + case HA_KEYTYPE_DOUBLE: + case HA_KEYTYPE_USHORT_INT: + case HA_KEYTYPE_ULONG_INT: + case HA_KEYTYPE_LONGLONG: + case HA_KEYTYPE_ULONGLONG: + case HA_KEYTYPE_INT24: + case HA_KEYTYPE_UINT24: + case HA_KEYTYPE_INT8: + keyseg->flag|= HA_SWAP_KEY; + break; + case HA_KEYTYPE_VARTEXT1: + case HA_KEYTYPE_VARTEXT2: + case HA_KEYTYPE_VARBINARY1: + case HA_KEYTYPE_VARBINARY2: + if (!(keyseg->flag & HA_BLOB_PART)) + { + /* Make a flag that this is a VARCHAR */ + keyseg->flag|= HA_VAR_LENGTH_PART; + /* Store in bit_start number of bytes used to pack the length */ + keyseg->bit_start= ((keyseg->type == HA_KEYTYPE_VARTEXT1 || + keyseg->type == HA_KEYTYPE_VARBINARY1) ? + 1 : 2); + } + break; + default: + break; + } + if (keyseg->flag & HA_SPACE_PACK) + { + DBUG_ASSERT(!(keyseg->flag & HA_VAR_LENGTH_PART)); + keydef->flag |= HA_SPACE_PACK_USED | HA_VAR_LENGTH_KEY; + options|=HA_OPTION_PACK_KEYS; /* Using packed keys */ + length++; /* At least one length byte */ + min_key_length_skip+=keyseg->length; + if (keyseg->length >= 255) + { /* prefix may be 3 bytes */ + min_key_length_skip+=2; + length+=2; + } + } + if (keyseg->flag & (HA_VAR_LENGTH_PART | HA_BLOB_PART)) + { + DBUG_ASSERT(!test_all_bits(keyseg->flag, + (HA_VAR_LENGTH_PART | HA_BLOB_PART))); + keydef->flag|=HA_VAR_LENGTH_KEY; + length++; /* At least one length byte */ + options|=HA_OPTION_PACK_KEYS; /* Using packed keys */ + min_key_length_skip+=keyseg->length; + if (keyseg->length >= 255) + { /* prefix may be 3 bytes */ + min_key_length_skip+=2; + length+=2; + } + } + key_length+= keyseg->length; + if (keyseg->null_bit) + { + key_length++; + options|=HA_OPTION_PACK_KEYS; + keyseg->flag|=HA_NULL_PART; + keydef->flag|=HA_VAR_LENGTH_KEY | HA_NULL_PART_KEY; + } + } + } /* if HA_FULLTEXT */ + key_segs+=keydef->keysegs; + if (keydef->keysegs > HA_MAX_KEY_SEG) + { + my_errno=HA_WRONG_CREATE_OPTION; + goto err; + } + /* + key_segs may be 0 in the case when we only want to be able to + add on row into the table. This can happen with some DISTINCT queries + in MySQL + */ + if ((keydef->flag & (HA_NOSAME | HA_NULL_PART_KEY)) == HA_NOSAME && + key_segs) + share.state.rec_per_key_part[key_segs-1]=1L; + length+=key_length; + keydef->block_length= MARIA_BLOCK_SIZE(length-real_length_diff, + pointer,MARIA_MAX_KEYPTR_SIZE); + if (keydef->block_length > MARIA_MAX_KEY_BLOCK_LENGTH || + length >= HA_MAX_KEY_BUFF) + { + my_errno=HA_WRONG_CREATE_OPTION; + goto err; + } + set_if_bigger(max_key_block_length,keydef->block_length); + keydef->keylength= (uint16) key_length; + keydef->minlength= (uint16) (length-min_key_length_skip); + keydef->maxlength= (uint16) length; + + if (length > max_key_length) + max_key_length= length; + tot_length+= (max_rows/(ulong) (((uint) keydef->block_length-5)/ + (length*2)))* + (ulong) keydef->block_length; + } + for (i=max_key_block_length/MARIA_MIN_KEY_BLOCK_LENGTH ; i-- ; ) + key_del[i]=HA_OFFSET_ERROR; + + unique_key_parts=0; + offset=reclength-uniques*MARIA_UNIQUE_HASH_LENGTH; + for (i=0, uniquedef=uniquedefs ; i < uniques ; i++ , uniquedef++) + { + uniquedef->key=keys+i; + unique_key_parts+=uniquedef->keysegs; + share.state.key_root[keys+i]= HA_OFFSET_ERROR; + tot_length+= (max_rows/(ulong) (((uint) maria_block_size-5)/ + ((MARIA_UNIQUE_HASH_LENGTH + pointer)*2)))* + (ulong) maria_block_size; + } + keys+=uniques; /* Each unique has 1 key */ + key_segs+=uniques; /* Each unique has 1 key seg */ + + base_pos=(MARIA_STATE_INFO_SIZE + keys * MARIA_STATE_KEY_SIZE + + max_key_block_length/MARIA_MIN_KEY_BLOCK_LENGTH* + MARIA_STATE_KEYBLOCK_SIZE+ + key_segs*MARIA_STATE_KEYSEG_SIZE); + info_length=base_pos+(uint) (MARIA_BASE_INFO_SIZE+ + keys * MARIA_KEYDEF_SIZE+ + uniques * MARIA_UNIQUEDEF_SIZE + + (key_segs + unique_key_parts)*HA_KEYSEG_SIZE+ + columns*MARIA_COLUMNDEF_SIZE); + + bmove(share.state.header.file_version,(byte*) maria_file_magic,4); + ci->old_options=options| (ci->old_options & HA_OPTION_TEMP_COMPRESS_RECORD ? + HA_OPTION_COMPRESS_RECORD | + HA_OPTION_TEMP_COMPRESS_RECORD: 0); + mi_int2store(share.state.header.options,ci->old_options); + mi_int2store(share.state.header.header_length,info_length); + mi_int2store(share.state.header.state_info_length,MARIA_STATE_INFO_SIZE); + mi_int2store(share.state.header.base_info_length,MARIA_BASE_INFO_SIZE); + mi_int2store(share.state.header.base_pos,base_pos); + share.state.header.language= (ci->language ? + ci->language : default_charset_info->number); + share.state.header.max_block_size=max_key_block_length/MARIA_MIN_KEY_BLOCK_LENGTH; + + share.state.dellink = HA_OFFSET_ERROR; + share.state.process= (ulong) getpid(); + share.state.unique= (ulong) 0; + share.state.update_count=(ulong) 0; + share.state.version= (ulong) time((time_t*) 0); + share.state.sortkey= (ushort) ~0; + share.state.auto_increment=ci->auto_increment; + share.options=options; + share.base.rec_reflength=pointer; + /* Get estimate for index file length (this may be wrong for FT keys) */ + tmp= (tot_length + max_key_block_length * keys * + MARIA_INDEX_BLOCK_MARGIN) / MARIA_MIN_KEY_BLOCK_LENGTH; + /* + use maximum of key_file_length we calculated and key_file_length value we + got from MYI file header (see also mariapack.c:save_state) + */ + share.base.key_reflength= + maria_get_pointer_length(max(ci->key_file_length,tmp),3); + share.base.keys= share.state.header.keys= keys; + share.state.header.uniques= uniques; + share.state.header.fulltext_keys= fulltext_keys; + mi_int2store(share.state.header.key_parts,key_segs); + mi_int2store(share.state.header.unique_key_parts,unique_key_parts); + + maria_set_all_keys_active(share.state.key_map, keys); + share.base.keystart = share.state.state.key_file_length= + MY_ALIGN(info_length, maria_block_size); + share.base.max_key_block_length=max_key_block_length; + share.base.max_key_length=ALIGN_SIZE(max_key_length+4); + share.base.records=ci->max_rows; + share.base.reloc= ci->reloc_rows; + share.base.reclength=real_reclength; + share.base.pack_reclength=reclength+ test(options & HA_OPTION_CHECKSUM); + share.base.max_pack_length=pack_reclength; + share.base.min_pack_length=min_pack_length; + share.base.pack_bits=packed; + share.base.fields=fields; + share.base.pack_fields=packed; +#ifdef USE_RAID + share.base.raid_type=ci->raid_type; + share.base.raid_chunks=ci->raid_chunks; + share.base.raid_chunksize=ci->raid_chunksize; +#endif + + /* max_data_file_length and max_key_file_length are recalculated on open */ + if (options & HA_OPTION_TMP_TABLE) + share.base.max_data_file_length=(my_off_t) ci->data_file_length; + + share.base.min_block_length= + (share.base.pack_reclength+3 < MARIA_EXTEND_BLOCK_LENGTH && + ! share.base.blobs) ? + max(share.base.pack_reclength,MARIA_MIN_BLOCK_LENGTH) : + MARIA_EXTEND_BLOCK_LENGTH; + if (! (flags & HA_DONT_TOUCH_DATA)) + share.state.create_time= (long) time((time_t*) 0); + + pthread_mutex_lock(&THR_LOCK_maria); + + if (ci->index_file_name) + { + char *iext= strrchr(ci->index_file_name, '.'); + int have_iext= iext && !strcmp(iext, MARIA_NAME_IEXT); + + fn_format(filename, ci->index_file_name, "", MARIA_NAME_IEXT, + MY_UNPACK_FILENAME| (have_iext ? MY_REPLACE_EXT :MY_APPEND_EXT)); + fn_format(linkname, name, "", MARIA_NAME_IEXT, + MY_UNPACK_FILENAME|MY_APPEND_EXT); + linkname_ptr=linkname; + /* + Don't create the table if the link or file exists to ensure that one + doesn't accidently destroy another table. + */ + create_flag=0; + } + else + { + fn_format(filename, name, "", MARIA_NAME_IEXT, + (MY_UNPACK_FILENAME | + (flags & HA_DONT_TOUCH_DATA) ? MY_RETURN_REAL_PATH : 0) | + MY_APPEND_EXT); + linkname_ptr=0; + /* Replace the current file */ + create_flag=MY_DELETE_OLD; + } + + /* + If a MRG_MARIA table is in use, the mapped MARIA tables are open, + but no entry is made in the table cache for them. + A TRUNCATE command checks for the table in the cache only and could + be fooled to believe, the table is not open. + Pull the emergency brake in this situation. (Bug #8306) + */ + if (_ma_test_if_reopen(filename)) + { + my_printf_error(0, "MARIA table '%s' is in use " + "(most likely by a MERGE table). Try FLUSH TABLES.", + MYF(0), name + dirname_length(name)); + goto err; + } + + if ((file= my_create_with_symlink(linkname_ptr, filename, 0, create_mode, + MYF(MY_WME | create_flag))) < 0) + goto err; + errpos=1; + + if (!(flags & HA_DONT_TOUCH_DATA)) + { +#ifdef USE_RAID + if (share.base.raid_type) + { + (void) fn_format(filename, name, "", MARIA_NAME_DEXT, + MY_UNPACK_FILENAME | MY_APPEND_EXT); + if ((dfile=my_raid_create(filename, 0, create_mode, + share.base.raid_type, + share.base.raid_chunks, + share.base.raid_chunksize, + MYF(MY_WME | MY_RAID))) < 0) + goto err; + } + else +#endif + { + if (ci->data_file_name) + { + char *dext= strrchr(ci->data_file_name, '.'); + int have_dext= dext && !strcmp(dext, MARIA_NAME_DEXT); + + fn_format(filename, ci->data_file_name, "", MARIA_NAME_DEXT, + MY_UNPACK_FILENAME | + (have_dext ? MY_REPLACE_EXT : MY_APPEND_EXT)); + fn_format(linkname, name, "",MARIA_NAME_DEXT, + MY_UNPACK_FILENAME | MY_APPEND_EXT); + linkname_ptr=linkname; + create_flag=0; + } + else + { + fn_format(filename,name,"", MARIA_NAME_DEXT, + MY_UNPACK_FILENAME | MY_APPEND_EXT); + linkname_ptr=0; + create_flag=MY_DELETE_OLD; + } + if ((dfile= + my_create_with_symlink(linkname_ptr, filename, 0, create_mode, + MYF(MY_WME | create_flag))) < 0) + goto err; + } + errpos=3; + } + + if (_ma_state_info_write(file, &share.state, 2) || + _ma_base_info_write(file, &share.base)) + goto err; +#ifndef DBUG_OFF + if ((uint) my_tell(file,MYF(0)) != base_pos+ MARIA_BASE_INFO_SIZE) + { + uint pos=(uint) my_tell(file,MYF(0)); + DBUG_PRINT("warning",("base_length: %d != used_length: %d", + base_pos+ MARIA_BASE_INFO_SIZE, pos)); + } +#endif + + /* Write key and keyseg definitions */ + for (i=0 ; i < share.base.keys - uniques; i++) + { + uint sp_segs=(keydefs[i].flag & HA_SPATIAL) ? 2*SPDIMS : 0; + + if (_ma_keydef_write(file, &keydefs[i])) + goto err; + for (j=0 ; j < keydefs[i].keysegs-sp_segs ; j++) + if (_ma_keyseg_write(file, &keydefs[i].seg[j])) + goto err; +#ifdef HAVE_SPATIAL + for (j=0 ; j < sp_segs ; j++) + { + HA_KEYSEG sseg; + sseg.type=SPTYPE; + sseg.language= 7; /* Binary */ + sseg.null_bit=0; + sseg.bit_start=0; + sseg.bit_end=0; + sseg.bit_length= 0; + sseg.bit_pos= 0; + sseg.length=SPLEN; + sseg.null_pos=0; + sseg.start=j*SPLEN; + sseg.flag= HA_SWAP_KEY; + if (_ma_keyseg_write(file, &sseg)) + goto err; + } +#endif + } + /* Create extra keys for unique definitions */ + offset=reclength-uniques*MARIA_UNIQUE_HASH_LENGTH; + bzero((char*) &tmp_keydef,sizeof(tmp_keydef)); + bzero((char*) &tmp_keyseg,sizeof(tmp_keyseg)); + for (i=0; i < uniques ; i++) + { + tmp_keydef.keysegs=1; + tmp_keydef.flag= HA_UNIQUE_CHECK; + tmp_keydef.block_length= (uint16)maria_block_size; + tmp_keydef.keylength= MARIA_UNIQUE_HASH_LENGTH + pointer; + tmp_keydef.minlength=tmp_keydef.maxlength=tmp_keydef.keylength; + tmp_keyseg.type= MARIA_UNIQUE_HASH_TYPE; + tmp_keyseg.length= MARIA_UNIQUE_HASH_LENGTH; + tmp_keyseg.start= offset; + offset+= MARIA_UNIQUE_HASH_LENGTH; + if (_ma_keydef_write(file,&tmp_keydef) || + _ma_keyseg_write(file,(&tmp_keyseg))) + goto err; + } + + /* Save unique definition */ + for (i=0 ; i < share.state.header.uniques ; i++) + { + HA_KEYSEG *keyseg_end; + keyseg= uniquedefs[i].seg; + if (_ma_uniquedef_write(file, &uniquedefs[i])) + goto err; + for (keyseg= uniquedefs[i].seg, keyseg_end= keyseg+ uniquedefs[i].keysegs; + keyseg < keyseg_end; + keyseg++) + { + switch (keyseg->type) { + case HA_KEYTYPE_VARTEXT1: + case HA_KEYTYPE_VARTEXT2: + case HA_KEYTYPE_VARBINARY1: + case HA_KEYTYPE_VARBINARY2: + if (!(keyseg->flag & HA_BLOB_PART)) + { + keyseg->flag|= HA_VAR_LENGTH_PART; + keyseg->bit_start= ((keyseg->type == HA_KEYTYPE_VARTEXT1 || + keyseg->type == HA_KEYTYPE_VARBINARY1) ? + 1 : 2); + } + break; + default: + break; + } + if (_ma_keyseg_write(file, keyseg)) + goto err; + } + } + for (i=0 ; i < share.base.fields ; i++) + if (_ma_recinfo_write(file, &recinfo[i])) + goto err; + +#ifndef DBUG_OFF + if ((uint) my_tell(file,MYF(0)) != info_length) + { + uint pos= (uint) my_tell(file,MYF(0)); + DBUG_PRINT("warning",("info_length: %d != used_length: %d", + info_length, pos)); + } +#endif + + /* Enlarge files */ + if (my_chsize(file,(ulong) share.base.keystart,0,MYF(0))) + goto err; + + if (! (flags & HA_DONT_TOUCH_DATA)) + { +#ifdef USE_RELOC + if (my_chsize(dfile,share.base.min_pack_length*ci->reloc_rows,0,MYF(0))) + goto err; +#endif + errpos=2; + if (my_close(dfile,MYF(0))) + goto err; + } + errpos=0; + pthread_mutex_unlock(&THR_LOCK_maria); + if (my_close(file,MYF(0))) + goto err; + my_free((char*) rec_per_key_part,MYF(0)); + DBUG_RETURN(0); + +err: + pthread_mutex_unlock(&THR_LOCK_maria); + save_errno=my_errno; + switch (errpos) { + case 3: + VOID(my_close(dfile,MYF(0))); + /* fall through */ + case 2: + /* QQ: Tõnu should add a call to my_raid_delete() here */ + if (! (flags & HA_DONT_TOUCH_DATA)) + my_delete_with_symlink(fn_format(filename,name,"",MARIA_NAME_DEXT, + MY_UNPACK_FILENAME | MY_APPEND_EXT), + MYF(0)); + /* fall through */ + case 1: + VOID(my_close(file,MYF(0))); + if (! (flags & HA_DONT_TOUCH_DATA)) + my_delete_with_symlink(fn_format(filename,name,"",MARIA_NAME_IEXT, + MY_UNPACK_FILENAME | MY_APPEND_EXT), + MYF(0)); + } + my_free((char*) rec_per_key_part, MYF(0)); + DBUG_RETURN(my_errno=save_errno); /* return the fatal errno */ +} + + +uint maria_get_pointer_length(ulonglong file_length, uint def) +{ + DBUG_ASSERT(def >= 2 && def <= 7); + if (file_length) /* If not default */ + { +#ifdef NOT_YET_READY_FOR_8_BYTE_POINTERS + if (file_length >= (longlong) 1 << 56) + def=8; +#endif + if (file_length >= (longlong) 1 << 48) + def=7; + if (file_length >= (longlong) 1 << 40) + def=6; + else if (file_length >= (longlong) 1 << 32) + def=5; + else if (file_length >= (1L << 24)) + def=4; + else if (file_length >= (1L << 16)) + def=3; + else + def=2; + } + return def; +} diff --git a/storage/maria/ma_dbug.c b/storage/maria/ma_dbug.c new file mode 100644 index 00000000000..7f2bff85047 --- /dev/null +++ b/storage/maria/ma_dbug.c @@ -0,0 +1,193 @@ +/* Copyright (C) 2006 MySQL AB & MySQL Finland AB & TCX DataKonsult AB + + 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; either version 2 of the License, or + (at your option) any later version. + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ + +/* Support rutiner with are using with dbug */ + +#include "maria_def.h" + + /* Print a key in user understandable format */ + +void _ma_print_key(FILE *stream, register HA_KEYSEG *keyseg, + const uchar *key, uint length) +{ + int flag; + short int s_1; + long int l_1; + float f_1; + double d_1; + const uchar *end; + const uchar *key_end=key+length; + + VOID(fputs("Key: \"",stream)); + flag=0; + for (; keyseg->type && key < key_end ;keyseg++) + { + if (flag++) + VOID(putc('-',stream)); + end= key+ keyseg->length; + if (keyseg->flag & HA_NULL_PART) + { + /* A NULL value is encoded by a 1-byte flag. Zero means NULL. */ + if (! *(key++)) + { + fprintf(stream,"NULL"); + continue; + } + } + + switch (keyseg->type) { + case HA_KEYTYPE_BINARY: + if (!(keyseg->flag & HA_SPACE_PACK) && keyseg->length == 1) + { /* packed binary digit */ + VOID(fprintf(stream,"%d",(uint) *key++)); + break; + } + /* fall through */ + case HA_KEYTYPE_TEXT: + case HA_KEYTYPE_NUM: + if (keyseg->flag & HA_SPACE_PACK) + { + VOID(fprintf(stream,"%.*s",(int) *key,key+1)); + key+= (int) *key+1; + } + else + { + VOID(fprintf(stream,"%.*s",(int) keyseg->length,key)); + key=end; + } + break; + case HA_KEYTYPE_INT8: + VOID(fprintf(stream,"%d",(int) *((signed char*) key))); + key=end; + break; + case HA_KEYTYPE_SHORT_INT: + s_1= mi_sint2korr(key); + VOID(fprintf(stream,"%d",(int) s_1)); + key=end; + break; + case HA_KEYTYPE_USHORT_INT: + { + ushort u_1; + u_1= mi_uint2korr(key); + VOID(fprintf(stream,"%u",(uint) u_1)); + key=end; + break; + } + case HA_KEYTYPE_LONG_INT: + l_1=mi_sint4korr(key); + VOID(fprintf(stream,"%ld",l_1)); + key=end; + break; + case HA_KEYTYPE_ULONG_INT: + l_1=mi_sint4korr(key); + VOID(fprintf(stream,"%lu",(ulong) l_1)); + key=end; + break; + case HA_KEYTYPE_INT24: + VOID(fprintf(stream,"%ld",(long) mi_sint3korr(key))); + key=end; + break; + case HA_KEYTYPE_UINT24: + VOID(fprintf(stream,"%lu",(ulong) mi_uint3korr(key))); + key=end; + break; + case HA_KEYTYPE_FLOAT: + mi_float4get(f_1,key); + VOID(fprintf(stream,"%g",(double) f_1)); + key=end; + break; + case HA_KEYTYPE_DOUBLE: + mi_float8get(d_1,key); + VOID(fprintf(stream,"%g",d_1)); + key=end; + break; +#ifdef HAVE_LONG_LONG + case HA_KEYTYPE_LONGLONG: + { + char buff[21]; + longlong2str(mi_sint8korr(key),buff,-10); + VOID(fprintf(stream,"%s",buff)); + key=end; + break; + } + case HA_KEYTYPE_ULONGLONG: + { + char buff[21]; + longlong2str(mi_sint8korr(key),buff,10); + VOID(fprintf(stream,"%s",buff)); + key=end; + break; + } + case HA_KEYTYPE_BIT: + { + uint i; + fputs("0x",stream); + for (i=0 ; i < keyseg->length ; i++) + fprintf(stream, "%02x", (uint) *key++); + key= end; + break; + } + +#endif + case HA_KEYTYPE_VARTEXT1: /* VARCHAR and TEXT */ + case HA_KEYTYPE_VARTEXT2: /* VARCHAR and TEXT */ + case HA_KEYTYPE_VARBINARY1: /* VARBINARY and BLOB */ + case HA_KEYTYPE_VARBINARY2: /* VARBINARY and BLOB */ + { + uint tmp_length; + get_key_length(tmp_length,key); + /* + The following command sometimes gives a warning from valgrind. + Not yet sure if the bug is in valgrind, glibc or mysqld + */ + VOID(fprintf(stream,"%.*s",(int) tmp_length,key)); + key+=tmp_length; + break; + } + default: break; /* This never happens */ + } + } + VOID(fputs("\"\n",stream)); + return; +} /* print_key */ + + +#ifdef EXTRA_DEBUG + +my_bool _ma_check_table_is_closed(const char *name, const char *where) +{ + char filename[FN_REFLEN]; + LIST *pos; + DBUG_ENTER("_ma_check_table_is_closed"); + + (void) fn_format(filename,name,"",MARIA_NAME_IEXT,4+16+32); + for (pos=maria_open_list ; pos ; pos=pos->next) + { + MARIA_HA *info=(MARIA_HA*) pos->data; + MARIA_SHARE *share=info->s; + if (!strcmp(share->unique_file_name,filename)) + { + if (share->last_version) + { + fprintf(stderr,"Warning: Table: %s is open on %s\n", name,where); + DBUG_PRINT("warning",("Table: %s is open on %s", name,where)); + DBUG_RETURN(1); + } + } + } + DBUG_RETURN(0); +} +#endif /* EXTRA_DEBUG */ diff --git a/storage/maria/ma_delete.c b/storage/maria/ma_delete.c new file mode 100644 index 00000000000..9e06b633171 --- /dev/null +++ b/storage/maria/ma_delete.c @@ -0,0 +1,890 @@ +/* Copyright (C) 2006 MySQL AB & MySQL Finland AB & TCX DataKonsult AB + + 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; either version 2 of the License, or + (at your option) any later version. + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ + +/* Remove a row from a MARIA table */ + +#include "ma_fulltext.h" +#include "ma_rt_index.h" + +static int d_search(MARIA_HA *info,MARIA_KEYDEF *keyinfo,uint comp_flag, + uchar *key,uint key_length,my_off_t page,uchar *anc_buff); +static int del(MARIA_HA *info,MARIA_KEYDEF *keyinfo,uchar *key,uchar *anc_buff, + my_off_t leaf_page,uchar *leaf_buff,uchar *keypos, + my_off_t next_block,uchar *ret_key); +static int underflow(MARIA_HA *info,MARIA_KEYDEF *keyinfo,uchar *anc_buff, + my_off_t leaf_page,uchar *leaf_buff,uchar *keypos); +static uint remove_key(MARIA_KEYDEF *keyinfo,uint nod_flag,uchar *keypos, + uchar *lastkey,uchar *page_end, + my_off_t *next_block); +static int _ma_ck_real_delete(register MARIA_HA *info,MARIA_KEYDEF *keyinfo, + uchar *key, uint key_length, my_off_t *root); + + +int maria_delete(MARIA_HA *info,const byte *record) +{ + uint i; + uchar *old_key; + int save_errno; + char lastpos[8]; + + MARIA_SHARE *share=info->s; + DBUG_ENTER("maria_delete"); + + /* Test if record is in datafile */ + + DBUG_EXECUTE_IF("maria_pretend_crashed_table_on_usage", + maria_print_error(info->s, HA_ERR_CRASHED); + DBUG_RETURN(my_errno= HA_ERR_CRASHED);); + DBUG_EXECUTE_IF("my_error_test_undefined_error", + maria_print_error(info->s, INT_MAX); + DBUG_RETURN(my_errno= INT_MAX);); + if (!(info->update & HA_STATE_AKTIV)) + { + DBUG_RETURN(my_errno=HA_ERR_KEY_NOT_FOUND); /* No database read */ + } + if (share->options & HA_OPTION_READ_ONLY_DATA) + { + DBUG_RETURN(my_errno=EACCES); + } + if (_ma_readinfo(info,F_WRLCK,1)) + DBUG_RETURN(my_errno); + if (info->s->calc_checksum) + info->checksum=(*info->s->calc_checksum)(info,record); + if ((*share->compare_record)(info,record)) + goto err; /* Error on read-check */ + + if (_ma_mark_file_changed(info)) + goto err; + + /* Remove all keys from the .ISAM file */ + + old_key=info->lastkey2; + for (i=0 ; i < share->base.keys ; i++ ) + { + if (maria_is_key_active(info->s->state.key_map, i)) + { + info->s->keyinfo[i].version++; + if (info->s->keyinfo[i].flag & HA_FULLTEXT ) + { + if (_ma_ft_del(info,i,(char*) old_key,record,info->lastpos)) + goto err; + } + else + { + if (info->s->keyinfo[i].ck_delete(info,i,old_key, + _ma_make_key(info,i,old_key,record,info->lastpos))) + goto err; + } + /* The above changed info->lastkey2. Inform maria_rnext_same(). */ + info->update&= ~HA_STATE_RNEXT_SAME; + } + } + + if ((*share->delete_record)(info)) + goto err; /* Remove record from database */ + info->state->checksum-=info->checksum; + + info->update= HA_STATE_CHANGED+HA_STATE_DELETED+HA_STATE_ROW_CHANGED; + info->state->records--; + + mi_sizestore(lastpos,info->lastpos); + maria_log_command(MARIA_LOG_DELETE,info,(byte*) lastpos,sizeof(lastpos),0); + VOID(_ma_writeinfo(info,WRITEINFO_UPDATE_KEYFILE)); + allow_break(); /* Allow SIGHUP & SIGINT */ + if (info->invalidator != 0) + { + DBUG_PRINT("info", ("invalidator... '%s' (delete)", info->filename)); + (*info->invalidator)(info->filename); + info->invalidator=0; + } + DBUG_RETURN(0); + +err: + save_errno=my_errno; + mi_sizestore(lastpos,info->lastpos); + maria_log_command(MARIA_LOG_DELETE,info,(byte*) lastpos, sizeof(lastpos),0); + if (save_errno != HA_ERR_RECORD_CHANGED) + { + maria_print_error(info->s, HA_ERR_CRASHED); + maria_mark_crashed(info); /* mark table crashed */ + } + VOID(_ma_writeinfo(info,WRITEINFO_UPDATE_KEYFILE)); + info->update|=HA_STATE_WRITTEN; /* Buffer changed */ + allow_break(); /* Allow SIGHUP & SIGINT */ + my_errno=save_errno; + if (save_errno == HA_ERR_KEY_NOT_FOUND) + { + maria_print_error(info->s, HA_ERR_CRASHED); + my_errno=HA_ERR_CRASHED; + } + + DBUG_RETURN(my_errno); +} /* maria_delete */ + + + /* Remove a key from the btree index */ + +int _ma_ck_delete(register MARIA_HA *info, uint keynr, uchar *key, + uint key_length) +{ + return _ma_ck_real_delete(info, info->s->keyinfo+keynr, key, key_length, + &info->s->state.key_root[keynr]); +} /* _ma_ck_delete */ + + +static int _ma_ck_real_delete(register MARIA_HA *info, MARIA_KEYDEF *keyinfo, + uchar *key, uint key_length, my_off_t *root) +{ + int error; + uint nod_flag; + my_off_t old_root; + uchar *root_buff; + DBUG_ENTER("_ma_ck_real_delete"); + + if ((old_root=*root) == HA_OFFSET_ERROR) + { + maria_print_error(info->s, HA_ERR_CRASHED); + DBUG_RETURN(my_errno=HA_ERR_CRASHED); + } + if (!(root_buff= (uchar*) my_alloca((uint) keyinfo->block_length+ + HA_MAX_KEY_BUFF*2))) + { + DBUG_PRINT("error",("Couldn't allocate memory")); + DBUG_RETURN(my_errno=ENOMEM); + } + DBUG_PRINT("info",("root_page: %ld",old_root)); + if (!_ma_fetch_keypage(info,keyinfo,old_root,DFLT_INIT_HITS,root_buff,0)) + { + error= -1; + goto err; + } + if ((error=d_search(info,keyinfo, + (keyinfo->flag & HA_FULLTEXT ? SEARCH_FIND | SEARCH_UPDATE + : SEARCH_SAME), + key,key_length,old_root,root_buff)) >0) + { + if (error == 2) + { + DBUG_PRINT("test",("Enlarging of root when deleting")); + error= _ma_enlarge_root(info,keyinfo,key,root); + } + else /* error == 1 */ + { + if (maria_getint(root_buff) <= (nod_flag=_ma_test_if_nod(root_buff))+3) + { + error=0; + if (nod_flag) + *root= _ma_kpos(nod_flag,root_buff+2+nod_flag); + else + *root=HA_OFFSET_ERROR; + if (_ma_dispose(info,keyinfo,old_root,DFLT_INIT_HITS)) + error= -1; + } + else + error= _ma_write_keypage(info,keyinfo,old_root, + DFLT_INIT_HITS,root_buff); + } + } +err: + my_afree((gptr) root_buff); + DBUG_PRINT("exit",("Return: %d",error)); + DBUG_RETURN(error); +} /* _ma_ck_real_delete */ + + + /* + ** Remove key below key root + ** Return values: + ** 1 if there are less buffers; In this case anc_buff is not saved + ** 2 if there are more buffers + ** -1 on errors + */ + +static int d_search(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, + uint comp_flag, uchar *key, uint key_length, + my_off_t page, uchar *anc_buff) +{ + int flag,ret_value,save_flag; + uint length,nod_flag,search_key_length; + my_bool last_key; + uchar *leaf_buff,*keypos; + my_off_t leaf_page,next_block; + uchar lastkey[HA_MAX_KEY_BUFF]; + DBUG_ENTER("d_search"); + DBUG_DUMP("page",(byte*) anc_buff,maria_getint(anc_buff)); + + search_key_length= (comp_flag & SEARCH_FIND) ? key_length : USE_WHOLE_KEY; + flag=(*keyinfo->bin_search)(info,keyinfo,anc_buff,key, search_key_length, + comp_flag, &keypos, lastkey, &last_key); + if (flag == MARIA_FOUND_WRONG_KEY) + { + DBUG_PRINT("error",("Found wrong key")); + DBUG_RETURN(-1); + } + nod_flag=_ma_test_if_nod(anc_buff); + + if (!flag && keyinfo->flag & HA_FULLTEXT) + { + uint off; + int subkeys; + + get_key_full_length_rdonly(off, lastkey); + subkeys=ft_sintXkorr(lastkey+off); + DBUG_ASSERT(info->ft1_to_ft2==0 || subkeys >=0); + comp_flag=SEARCH_SAME; + if (subkeys >= 0) + { + /* normal word, one-level tree structure */ + if (info->ft1_to_ft2) + { + /* we're in ft1->ft2 conversion mode. Saving key data */ + insert_dynamic(info->ft1_to_ft2, (char*) (lastkey+off)); + } + else + { + /* we need exact match only if not in ft1->ft2 conversion mode */ + flag=(*keyinfo->bin_search)(info,keyinfo,anc_buff,key,USE_WHOLE_KEY, + comp_flag, &keypos, lastkey, &last_key); + } + /* fall through to normal delete */ + } + else + { + /* popular word. two-level tree. going down */ + uint tmp_key_length; + my_off_t root; + uchar *kpos=keypos; + + if (!(tmp_key_length=(*keyinfo->get_key)(keyinfo,nod_flag,&kpos,lastkey))) + { + maria_print_error(info->s, HA_ERR_CRASHED); + my_errno= HA_ERR_CRASHED; + DBUG_RETURN(-1); + } + root= _ma_dpos(info,nod_flag,kpos); + if (subkeys == -1) + { + /* the last entry in sub-tree */ + if (_ma_dispose(info, keyinfo, root,DFLT_INIT_HITS)) + DBUG_RETURN(-1); + /* fall through to normal delete */ + } + else + { + keyinfo=&info->s->ft2_keyinfo; + kpos-=keyinfo->keylength+nod_flag; /* we'll modify key entry 'in vivo' */ + get_key_full_length_rdonly(off, key); + key+=off; + ret_value= _ma_ck_real_delete(info, &info->s->ft2_keyinfo, + key, HA_FT_WLEN, &root); + _ma_dpointer(info, kpos+HA_FT_WLEN, root); + subkeys++; + ft_intXstore(kpos, subkeys); + if (!ret_value) + ret_value= _ma_write_keypage(info,keyinfo,page, + DFLT_INIT_HITS,anc_buff); + DBUG_PRINT("exit",("Return: %d",ret_value)); + DBUG_RETURN(ret_value); + } + } + } + leaf_buff=0; + LINT_INIT(leaf_page); + if (nod_flag) + { + leaf_page= _ma_kpos(nod_flag,keypos); + if (!(leaf_buff= (uchar*) my_alloca((uint) keyinfo->block_length+ + HA_MAX_KEY_BUFF*2))) + { + DBUG_PRINT("error",("Couldn't allocate memory")); + my_errno=ENOMEM; + DBUG_PRINT("exit",("Return: %d",-1)); + DBUG_RETURN(-1); + } + if (!_ma_fetch_keypage(info,keyinfo,leaf_page,DFLT_INIT_HITS,leaf_buff,0)) + goto err; + } + + if (flag != 0) + { + if (!nod_flag) + { + DBUG_PRINT("error",("Didn't find key")); + maria_print_error(info->s, HA_ERR_CRASHED); + my_errno=HA_ERR_CRASHED; /* This should newer happend */ + goto err; + } + save_flag=0; + ret_value=d_search(info,keyinfo,comp_flag,key,key_length, + leaf_page,leaf_buff); + } + else + { /* Found key */ + uint tmp; + length=maria_getint(anc_buff); + if (!(tmp= remove_key(keyinfo,nod_flag,keypos,lastkey,anc_buff+length, + &next_block))) + goto err; + + length-= tmp; + + maria_putint(anc_buff,length,nod_flag); + if (!nod_flag) + { /* On leaf page */ + if (_ma_write_keypage(info,keyinfo,page,DFLT_INIT_HITS,anc_buff)) + { + DBUG_PRINT("exit",("Return: %d",-1)); + DBUG_RETURN(-1); + } + /* Page will be update later if we return 1 */ + DBUG_RETURN(test(length <= (info->quick_mode ? MARIA_MIN_KEYBLOCK_LENGTH : + (uint) keyinfo->underflow_block_length))); + } + save_flag=1; + ret_value=del(info,keyinfo,key,anc_buff,leaf_page,leaf_buff,keypos, + next_block,lastkey); + } + if (ret_value >0) + { + save_flag=1; + if (ret_value == 1) + ret_value= underflow(info,keyinfo,anc_buff,leaf_page,leaf_buff,keypos); + else + { /* This happens only with packed keys */ + DBUG_PRINT("test",("Enlarging of key when deleting")); + if (!_ma_get_last_key(info,keyinfo,anc_buff,lastkey,keypos,&length)) + goto err; + ret_value= _ma_insert(info,keyinfo,key,anc_buff,keypos,lastkey, + (uchar*) 0,(uchar*) 0,(my_off_t) 0,(my_bool) 0); + } + } + if (ret_value == 0 && maria_getint(anc_buff) > keyinfo->block_length) + { + save_flag=1; + ret_value= _ma_split_page(info,keyinfo,key,anc_buff,lastkey,0) | 2; + } + if (save_flag && ret_value != 1) + ret_value|= _ma_write_keypage(info,keyinfo,page,DFLT_INIT_HITS,anc_buff); + else + { + DBUG_DUMP("page",(byte*) anc_buff,maria_getint(anc_buff)); + } + my_afree((byte*) leaf_buff); + DBUG_PRINT("exit",("Return: %d",ret_value)); + DBUG_RETURN(ret_value); + +err: + my_afree((byte*) leaf_buff); + DBUG_PRINT("exit",("Error: %d",my_errno)); + DBUG_RETURN (-1); +} /* d_search */ + + + /* Remove a key that has a page-reference */ + +static int del(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, uchar *key, + uchar *anc_buff, my_off_t leaf_page, uchar *leaf_buff, + uchar *keypos, /* Pos to where deleted key was */ + my_off_t next_block, + uchar *ret_key) /* key before keypos in anc_buff */ +{ + int ret_value,length; + uint a_length,nod_flag,tmp; + my_off_t next_page; + uchar keybuff[HA_MAX_KEY_BUFF],*endpos,*next_buff,*key_start, *prev_key; + MARIA_SHARE *share=info->s; + MARIA_KEY_PARAM s_temp; + DBUG_ENTER("del"); + DBUG_PRINT("enter",("leaf_page: %ld keypos: 0x%lx", leaf_page, + (ulong) keypos)); + DBUG_DUMP("leaf_buff",(byte*) leaf_buff,maria_getint(leaf_buff)); + + endpos=leaf_buff+maria_getint(leaf_buff); + if (!(key_start= _ma_get_last_key(info,keyinfo,leaf_buff,keybuff,endpos, + &tmp))) + DBUG_RETURN(-1); + + if ((nod_flag=_ma_test_if_nod(leaf_buff))) + { + next_page= _ma_kpos(nod_flag,endpos); + if (!(next_buff= (uchar*) my_alloca((uint) keyinfo->block_length+ + HA_MAX_KEY_BUFF*2))) + DBUG_RETURN(-1); + if (!_ma_fetch_keypage(info,keyinfo,next_page,DFLT_INIT_HITS,next_buff,0)) + ret_value= -1; + else + { + DBUG_DUMP("next_page",(byte*) next_buff,maria_getint(next_buff)); + if ((ret_value=del(info,keyinfo,key,anc_buff,next_page,next_buff, + keypos,next_block,ret_key)) >0) + { + endpos=leaf_buff+maria_getint(leaf_buff); + if (ret_value == 1) + { + ret_value=underflow(info,keyinfo,leaf_buff,next_page, + next_buff,endpos); + if (ret_value == 0 && maria_getint(leaf_buff) > keyinfo->block_length) + { + ret_value= _ma_split_page(info,keyinfo,key,leaf_buff,ret_key,0) | 2; + } + } + else + { + DBUG_PRINT("test",("Inserting of key when deleting")); + if (_ma_get_last_key(info,keyinfo,leaf_buff,keybuff,endpos, + &tmp)) + goto err; + ret_value= _ma_insert(info,keyinfo,key,leaf_buff,endpos,keybuff, + (uchar*) 0,(uchar*) 0,(my_off_t) 0,0); + } + } + if (_ma_write_keypage(info,keyinfo,leaf_page,DFLT_INIT_HITS,leaf_buff)) + goto err; + } + my_afree((byte*) next_buff); + DBUG_RETURN(ret_value); + } + + /* Remove last key from leaf page */ + + maria_putint(leaf_buff,key_start-leaf_buff,nod_flag); + if (_ma_write_keypage(info,keyinfo,leaf_page,DFLT_INIT_HITS,leaf_buff)) + goto err; + + /* Place last key in ancestor page on deleted key position */ + + a_length=maria_getint(anc_buff); + endpos=anc_buff+a_length; + if (keypos != anc_buff+2+share->base.key_reflength && + !_ma_get_last_key(info,keyinfo,anc_buff,ret_key,keypos,&tmp)) + goto err; + prev_key=(keypos == anc_buff+2+share->base.key_reflength ? + 0 : ret_key); + length=(*keyinfo->pack_key)(keyinfo,share->base.key_reflength, + keypos == endpos ? (uchar*) 0 : keypos, + prev_key, prev_key, + keybuff,&s_temp); + if (length > 0) + bmove_upp((byte*) endpos+length,(byte*) endpos,(uint) (endpos-keypos)); + else + bmove(keypos,keypos-length, (int) (endpos-keypos)+length); + (*keyinfo->store_key)(keyinfo,keypos,&s_temp); + /* Save pointer to next leaf */ + if (!(*keyinfo->get_key)(keyinfo,share->base.key_reflength,&keypos,ret_key)) + goto err; + _ma_kpointer(info,keypos - share->base.key_reflength,next_block); + maria_putint(anc_buff,a_length+length,share->base.key_reflength); + + DBUG_RETURN( maria_getint(leaf_buff) <= + (info->quick_mode ? MARIA_MIN_KEYBLOCK_LENGTH : + (uint) keyinfo->underflow_block_length)); +err: + DBUG_RETURN(-1); +} /* del */ + + + /* Balances adjacent pages if underflow occours */ + +static int underflow(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, + uchar *anc_buff, + my_off_t leaf_page,/* Ancestor page and underflow page */ + uchar *leaf_buff, + uchar *keypos) /* Position to pos after key */ +{ + int t_length; + uint length,anc_length,buff_length,leaf_length,p_length,s_length,nod_flag, + key_reflength,key_length; + my_off_t next_page; + uchar anc_key[HA_MAX_KEY_BUFF],leaf_key[HA_MAX_KEY_BUFF], + *buff,*endpos,*next_keypos,*anc_pos,*half_pos,*temp_pos,*prev_key, + *after_key; + MARIA_KEY_PARAM s_temp; + MARIA_SHARE *share=info->s; + DBUG_ENTER("underflow"); + DBUG_PRINT("enter",("leaf_page: %ld keypos: 0x%lx",(long) leaf_page, + (ulong) keypos)); + DBUG_DUMP("anc_buff",(byte*) anc_buff,maria_getint(anc_buff)); + DBUG_DUMP("leaf_buff",(byte*) leaf_buff,maria_getint(leaf_buff)); + + buff=info->buff; + info->buff_used=1; + next_keypos=keypos; + nod_flag=_ma_test_if_nod(leaf_buff); + p_length=nod_flag+2; + anc_length=maria_getint(anc_buff); + leaf_length=maria_getint(leaf_buff); + key_reflength=share->base.key_reflength; + if (info->s->keyinfo+info->lastinx == keyinfo) + info->page_changed=1; + + if ((keypos < anc_buff+anc_length && (info->state->records & 1)) || + keypos == anc_buff+2+key_reflength) + { /* Use page right of anc-page */ + DBUG_PRINT("test",("use right page")); + + if (keyinfo->flag & HA_BINARY_PACK_KEY) + { + if (!(next_keypos= _ma_get_key(info, keyinfo, + anc_buff, buff, keypos, &length))) + goto err; + } + else + { + /* Got to end of found key */ + buff[0]=buff[1]=0; /* Avoid length error check if packed key */ + if (!(*keyinfo->get_key)(keyinfo,key_reflength,&next_keypos, + buff)) + goto err; + } + next_page= _ma_kpos(key_reflength,next_keypos); + if (!_ma_fetch_keypage(info,keyinfo,next_page,DFLT_INIT_HITS,buff,0)) + goto err; + buff_length=maria_getint(buff); + DBUG_DUMP("next",(byte*) buff,buff_length); + + /* find keys to make a big key-page */ + bmove((byte*) next_keypos-key_reflength,(byte*) buff+2, + key_reflength); + if (!_ma_get_last_key(info,keyinfo,anc_buff,anc_key,next_keypos,&length) + || !_ma_get_last_key(info,keyinfo,leaf_buff,leaf_key, + leaf_buff+leaf_length,&length)) + goto err; + + /* merge pages and put parting key from anc_buff between */ + prev_key=(leaf_length == p_length ? (uchar*) 0 : leaf_key); + t_length=(*keyinfo->pack_key)(keyinfo,nod_flag,buff+p_length, + prev_key, prev_key, + anc_key, &s_temp); + length=buff_length-p_length; + endpos=buff+length+leaf_length+t_length; + /* buff will always be larger than before !*/ + bmove_upp((byte*) endpos, (byte*) buff+buff_length,length); + memcpy((byte*) buff, (byte*) leaf_buff,(size_t) leaf_length); + (*keyinfo->store_key)(keyinfo,buff+leaf_length,&s_temp); + buff_length=(uint) (endpos-buff); + maria_putint(buff,buff_length,nod_flag); + + /* remove key from anc_buff */ + + if (!(s_length=remove_key(keyinfo,key_reflength,keypos,anc_key, + anc_buff+anc_length,(my_off_t *) 0))) + goto err; + + anc_length-=s_length; + maria_putint(anc_buff,anc_length,key_reflength); + + if (buff_length <= keyinfo->block_length) + { /* Keys in one page */ + memcpy((byte*) leaf_buff,(byte*) buff,(size_t) buff_length); + if (_ma_dispose(info,keyinfo,next_page,DFLT_INIT_HITS)) + goto err; + } + else + { /* Page is full */ + endpos=anc_buff+anc_length; + DBUG_PRINT("test",("anc_buff: %lx endpos: %lx",anc_buff,endpos)); + if (keypos != anc_buff+2+key_reflength && + !_ma_get_last_key(info,keyinfo,anc_buff,anc_key,keypos,&length)) + goto err; + if (!(half_pos= _ma_find_half_pos(nod_flag, keyinfo, buff, leaf_key, + &key_length, &after_key))) + goto err; + length=(uint) (half_pos-buff); + memcpy((byte*) leaf_buff,(byte*) buff,(size_t) length); + maria_putint(leaf_buff,length,nod_flag); + + /* Correct new keypointer to leaf_page */ + half_pos=after_key; + _ma_kpointer(info,leaf_key+key_length,next_page); + /* Save key in anc_buff */ + prev_key=(keypos == anc_buff+2+key_reflength ? (uchar*) 0 : anc_key), + t_length=(*keyinfo->pack_key)(keyinfo,key_reflength, + (keypos == endpos ? (uchar*) 0 : + keypos), + prev_key, prev_key, + leaf_key, &s_temp); + if (t_length >= 0) + bmove_upp((byte*) endpos+t_length,(byte*) endpos, + (uint) (endpos-keypos)); + else + bmove(keypos,keypos-t_length,(uint) (endpos-keypos)+t_length); + (*keyinfo->store_key)(keyinfo,keypos,&s_temp); + maria_putint(anc_buff,(anc_length+=t_length),key_reflength); + + /* Store key first in new page */ + if (nod_flag) + bmove((byte*) buff+2,(byte*) half_pos-nod_flag,(size_t) nod_flag); + if (!(*keyinfo->get_key)(keyinfo,nod_flag,&half_pos,leaf_key)) + goto err; + t_length=(int) (*keyinfo->pack_key)(keyinfo, nod_flag, (uchar*) 0, + (uchar*) 0, (uchar *) 0, + leaf_key, &s_temp); + /* t_length will always be > 0 for a new page !*/ + length=(uint) ((buff+maria_getint(buff))-half_pos); + bmove((byte*) buff+p_length+t_length,(byte*) half_pos,(size_t) length); + (*keyinfo->store_key)(keyinfo,buff+p_length,&s_temp); + maria_putint(buff,length+t_length+p_length,nod_flag); + + if (_ma_write_keypage(info,keyinfo,next_page,DFLT_INIT_HITS,buff)) + goto err; + } + if (_ma_write_keypage(info,keyinfo,leaf_page,DFLT_INIT_HITS,leaf_buff)) + goto err; + DBUG_RETURN(anc_length <= ((info->quick_mode ? MARIA_MIN_BLOCK_LENGTH : + (uint) keyinfo->underflow_block_length))); + } + + DBUG_PRINT("test",("use left page")); + + keypos= _ma_get_last_key(info,keyinfo,anc_buff,anc_key,keypos,&length); + if (!keypos) + goto err; + next_page= _ma_kpos(key_reflength,keypos); + if (!_ma_fetch_keypage(info,keyinfo,next_page,DFLT_INIT_HITS,buff,0)) + goto err; + buff_length=maria_getint(buff); + endpos=buff+buff_length; + DBUG_DUMP("prev",(byte*) buff,buff_length); + + /* find keys to make a big key-page */ + bmove((byte*) next_keypos - key_reflength,(byte*) leaf_buff+2, + key_reflength); + next_keypos=keypos; + if (!(*keyinfo->get_key)(keyinfo,key_reflength,&next_keypos, + anc_key)) + goto err; + if (!_ma_get_last_key(info,keyinfo,buff,leaf_key,endpos,&length)) + goto err; + + /* merge pages and put parting key from anc_buff between */ + prev_key=(leaf_length == p_length ? (uchar*) 0 : leaf_key); + t_length=(*keyinfo->pack_key)(keyinfo,nod_flag, + (leaf_length == p_length ? + (uchar*) 0 : leaf_buff+p_length), + prev_key, prev_key, + anc_key, &s_temp); + if (t_length >= 0) + bmove((byte*) endpos+t_length,(byte*) leaf_buff+p_length, + (size_t) (leaf_length-p_length)); + else /* We gained space */ + bmove((byte*) endpos,(byte*) leaf_buff+((int) p_length-t_length), + (size_t) (leaf_length-p_length+t_length)); + + (*keyinfo->store_key)(keyinfo,endpos,&s_temp); + buff_length=buff_length+leaf_length-p_length+t_length; + maria_putint(buff,buff_length,nod_flag); + + /* remove key from anc_buff */ + if (!(s_length= remove_key(keyinfo,key_reflength,keypos,anc_key, + anc_buff+anc_length,(my_off_t *) 0))) + goto err; + + anc_length-=s_length; + maria_putint(anc_buff,anc_length,key_reflength); + + if (buff_length <= keyinfo->block_length) + { /* Keys in one page */ + if (_ma_dispose(info,keyinfo,leaf_page,DFLT_INIT_HITS)) + goto err; + } + else + { /* Page is full */ + if (keypos == anc_buff+2+key_reflength) + anc_pos=0; /* First key */ + else if (!_ma_get_last_key(info,keyinfo,anc_buff,anc_pos=anc_key,keypos, + &length)) + goto err; + endpos= _ma_find_half_pos(nod_flag,keyinfo,buff,leaf_key, + &key_length, &half_pos); + if (!endpos) + goto err; + _ma_kpointer(info,leaf_key+key_length,leaf_page); + /* Save key in anc_buff */ + DBUG_DUMP("anc_buff",(byte*) anc_buff,anc_length); + DBUG_DUMP("key_to_anc",(byte*) leaf_key,key_length); + + temp_pos=anc_buff+anc_length; + t_length=(*keyinfo->pack_key)(keyinfo,key_reflength, + keypos == temp_pos ? (uchar*) 0 + : keypos, + anc_pos, anc_pos, + leaf_key,&s_temp); + if (t_length > 0) + bmove_upp((byte*) temp_pos+t_length,(byte*) temp_pos, + (uint) (temp_pos-keypos)); + else + bmove(keypos,keypos-t_length,(uint) (temp_pos-keypos)+t_length); + (*keyinfo->store_key)(keyinfo,keypos,&s_temp); + maria_putint(anc_buff,(anc_length+=t_length),key_reflength); + + /* Store first key on new page */ + if (nod_flag) + bmove((byte*) leaf_buff+2,(byte*) half_pos-nod_flag,(size_t) nod_flag); + if (!(length=(*keyinfo->get_key)(keyinfo,nod_flag,&half_pos,leaf_key))) + goto err; + DBUG_DUMP("key_to_leaf",(byte*) leaf_key,length); + t_length=(*keyinfo->pack_key)(keyinfo,nod_flag, (uchar*) 0, + (uchar*) 0, (uchar*) 0, leaf_key, &s_temp); + length=(uint) ((buff+buff_length)-half_pos); + DBUG_PRINT("info",("t_length: %d length: %d",t_length,(int) length)); + bmove((byte*) leaf_buff+p_length+t_length,(byte*) half_pos, + (size_t) length); + (*keyinfo->store_key)(keyinfo,leaf_buff+p_length,&s_temp); + maria_putint(leaf_buff,length+t_length+p_length,nod_flag); + if (_ma_write_keypage(info,keyinfo,leaf_page,DFLT_INIT_HITS,leaf_buff)) + goto err; + maria_putint(buff,endpos-buff,nod_flag); + } + if (_ma_write_keypage(info,keyinfo,next_page,DFLT_INIT_HITS,buff)) + goto err; + DBUG_RETURN(anc_length <= (uint) keyinfo->block_length/2); + +err: + DBUG_RETURN(-1); +} /* underflow */ + + + /* + remove a key from packed buffert + The current code doesn't handle the case that the next key may be + packed better against the previous key if there is a case difference + returns how many chars was removed or 0 on error + */ + +static uint remove_key(MARIA_KEYDEF *keyinfo, uint nod_flag, + uchar *keypos, /* Where key starts */ + uchar *lastkey, /* key to be removed */ + uchar *page_end, /* End of page */ + my_off_t *next_block) /* ptr to next block */ +{ + int s_length; + uchar *start; + DBUG_ENTER("remove_key"); + DBUG_PRINT("enter",("keypos: %lx page_end: %lx",keypos,page_end)); + + start=keypos; + if (!(keyinfo->flag & + (HA_PACK_KEY | HA_SPACE_PACK_USED | HA_VAR_LENGTH_KEY | + HA_BINARY_PACK_KEY))) + { + s_length=(int) (keyinfo->keylength+nod_flag); + if (next_block && nod_flag) + *next_block= _ma_kpos(nod_flag,keypos+s_length); + } + else + { /* Let keypos point at next key */ + /* Calculate length of key */ + if (!(*keyinfo->get_key)(keyinfo,nod_flag,&keypos,lastkey)) + DBUG_RETURN(0); /* Error */ + + if (next_block && nod_flag) + *next_block= _ma_kpos(nod_flag,keypos); + s_length=(int) (keypos-start); + if (keypos != page_end) + { + if (keyinfo->flag & HA_BINARY_PACK_KEY) + { + uchar *old_key=start; + uint next_length,prev_length,prev_pack_length; + get_key_length(next_length,keypos); + get_key_pack_length(prev_length,prev_pack_length,old_key); + if (next_length > prev_length) + { + /* We have to copy data from the current key to the next key */ + bmove_upp((char*) keypos,(char*) (lastkey+next_length), + (next_length-prev_length)); + keypos-=(next_length-prev_length)+prev_pack_length; + store_key_length(keypos,prev_length); + s_length=(int) (keypos-start); + } + } + else + { + /* Check if a variable length first key part */ + if ((keyinfo->seg->flag & HA_PACK_KEY) && *keypos & 128) + { + /* Next key is packed against the current one */ + uint next_length,prev_length,prev_pack_length,lastkey_length, + rest_length; + if (keyinfo->seg[0].length >= 127) + { + if (!(prev_length=mi_uint2korr(start) & 32767)) + goto end; + next_length=mi_uint2korr(keypos) & 32767; + keypos+=2; + prev_pack_length=2; + } + else + { + if (!(prev_length= *start & 127)) + goto end; /* Same key as previous*/ + next_length= *keypos & 127; + keypos++; + prev_pack_length=1; + } + if (!(*start & 128)) + prev_length=0; /* prev key not packed */ + if (keyinfo->seg[0].flag & HA_NULL_PART) + lastkey++; /* Skip null marker */ + get_key_length(lastkey_length,lastkey); + if (!next_length) /* Same key after */ + { + next_length=lastkey_length; + rest_length=0; + } + else + get_key_length(rest_length,keypos); + + if (next_length >= prev_length) + { /* Key after is based on deleted key */ + uint pack_length,tmp; + bmove_upp((char*) keypos,(char*) (lastkey+next_length), + tmp=(next_length-prev_length)); + rest_length+=tmp; + pack_length= prev_length ? get_pack_length(rest_length): 0; + keypos-=tmp+pack_length+prev_pack_length; + s_length=(int) (keypos-start); + if (prev_length) /* Pack against prev key */ + { + *keypos++= start[0]; + if (prev_pack_length == 2) + *keypos++= start[1]; + store_key_length(keypos,rest_length); + } + else + { + /* Next key is not packed anymore */ + if (keyinfo->seg[0].flag & HA_NULL_PART) + { + rest_length++; /* Mark not null */ + } + if (prev_pack_length == 2) + { + mi_int2store(keypos,rest_length); + } + else + *keypos= rest_length; + } + } + } + } + } + } + end: + bmove((byte*) start,(byte*) start+s_length, + (uint) (page_end-start-s_length)); + DBUG_RETURN((uint) s_length); +} /* remove_key */ diff --git a/storage/maria/ma_delete_all.c b/storage/maria/ma_delete_all.c new file mode 100644 index 00000000000..d71e4d7dce7 --- /dev/null +++ b/storage/maria/ma_delete_all.c @@ -0,0 +1,79 @@ +/* Copyright (C) 2006 MySQL AB & MySQL Finland AB & TCX DataKonsult AB + + 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; either version 2 of the License, or + (at your option) any later version. + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ + +/* Remove all rows from a MARIA table */ +/* This clears the status information and truncates files */ + +#include "maria_def.h" + +int maria_delete_all_rows(MARIA_HA *info) +{ + uint i; + char buf[22]; + MARIA_SHARE *share=info->s; + MARIA_STATE_INFO *state=&share->state; + DBUG_ENTER("maria_delete_all_rows"); + + if (share->options & HA_OPTION_READ_ONLY_DATA) + { + DBUG_RETURN(my_errno=EACCES); + } + if (_ma_readinfo(info,F_WRLCK,1)) + DBUG_RETURN(my_errno); + if (_ma_mark_file_changed(info)) + goto err; + + info->state->records=info->state->del=state->split=0; + state->dellink = HA_OFFSET_ERROR; + state->sortkey= (ushort) ~0; + info->state->key_file_length=share->base.keystart; + info->state->data_file_length=0; + info->state->empty=info->state->key_empty=0; + info->state->checksum=0; + + for (i=share->base.max_key_block_length/MARIA_MIN_KEY_BLOCK_LENGTH ; i-- ; ) + state->key_del[i]= HA_OFFSET_ERROR; + for (i=0 ; i < share->base.keys ; i++) + state->key_root[i]= HA_OFFSET_ERROR; + + maria_log_command(MARIA_LOG_DELETE_ALL,info,(byte*) 0,0,0); + /* + If we are using delayed keys or if the user has done changes to the tables + since it was locked then there may be key blocks in the key cache + */ + flush_key_blocks(share->key_cache, share->kfile, FLUSH_IGNORE_CHANGED); + if (my_chsize(info->dfile, 0, 0, MYF(MY_WME)) || + my_chsize(share->kfile, share->base.keystart, 0, MYF(MY_WME)) ) + goto err; + VOID(_ma_writeinfo(info,WRITEINFO_UPDATE_KEYFILE)); +#ifdef HAVE_MMAP + /* Resize mmaped area */ + rw_wrlock(&info->s->mmap_lock); + _ma_remap_file(info, (my_off_t)0); + rw_unlock(&info->s->mmap_lock); +#endif + allow_break(); /* Allow SIGHUP & SIGINT */ + DBUG_RETURN(0); + +err: + { + int save_errno=my_errno; + VOID(_ma_writeinfo(info,WRITEINFO_UPDATE_KEYFILE)); + info->update|=HA_STATE_WRITTEN; /* Buffer changed */ + allow_break(); /* Allow SIGHUP & SIGINT */ + DBUG_RETURN(my_errno=save_errno); + } +} /* maria_delete */ diff --git a/storage/maria/ma_delete_table.c b/storage/maria/ma_delete_table.c new file mode 100644 index 00000000000..a9af9a62c99 --- /dev/null +++ b/storage/maria/ma_delete_table.c @@ -0,0 +1,58 @@ +/* Copyright (C) 2006 MySQL AB & MySQL Finland AB & TCX DataKonsult AB + + 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; either version 2 of the License, or + (at your option) any later version. + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ + +/* + deletes a table +*/ + +#include "ma_fulltext.h" + +int maria_delete_table(const char *name) +{ + char from[FN_REFLEN]; +#ifdef USE_RAID + uint raid_type=0,raid_chunks=0; +#endif + DBUG_ENTER("maria_delete_table"); + +#ifdef EXTRA_DEBUG + _ma_check_table_is_closed(name,"delete"); +#endif +#ifdef USE_RAID + { + MARIA_HA *info; + /* we use 'open_for_repair' to be able to delete a crashed table */ + if (!(info=maria_open(name, O_RDONLY, HA_OPEN_FOR_REPAIR))) + DBUG_RETURN(my_errno); + raid_type = info->s->base.raid_type; + raid_chunks = info->s->base.raid_chunks; + maria_close(info); + } +#ifdef EXTRA_DEBUG + _ma_check_table_is_closed(name,"delete"); +#endif +#endif /* USE_RAID */ + + fn_format(from,name,"",MARIA_NAME_IEXT,MY_UNPACK_FILENAME|MY_APPEND_EXT); + if (my_delete_with_symlink(from, MYF(MY_WME))) + DBUG_RETURN(my_errno); + fn_format(from,name,"",MARIA_NAME_DEXT,MY_UNPACK_FILENAME|MY_APPEND_EXT); +#ifdef USE_RAID + if (raid_type) + DBUG_RETURN(my_raid_delete(from, raid_chunks, MYF(MY_WME)) ? my_errno : 0); +#endif + DBUG_RETURN(my_delete_with_symlink(from, MYF(MY_WME)) ? my_errno : 0); +} diff --git a/storage/maria/ma_dynrec.c b/storage/maria/ma_dynrec.c new file mode 100644 index 00000000000..0fbf28b949a --- /dev/null +++ b/storage/maria/ma_dynrec.c @@ -0,0 +1,1811 @@ +/* Copyright (C) 2006 MySQL AB & MySQL Finland AB & TCX DataKonsult AB + + 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; either version 2 of the License, or + (at your option) any later version. + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ + +/* + Functions to handle space-packed-records and blobs + + A row may be stored in one or more linked blocks. + The block size is between MARIA_MIN_BLOCK_LENGTH and MARIA_MAX_BLOCK_LENGTH. + Each block is aligned on MARIA_DYN_ALIGN_SIZE. + The reson for the max block size is to not have too many different types + of blocks. For the differnet block types, look at _ma_get_block_info() +*/ + +#include "maria_def.h" + +/* Enough for comparing if number is zero */ +static char zero_string[]={0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0}; + +static int write_dynamic_record(MARIA_HA *info,const byte *record, + ulong reclength); +static int _ma_find_writepos(MARIA_HA *info,ulong reclength,my_off_t *filepos, + ulong *length); +static int update_dynamic_record(MARIA_HA *info,my_off_t filepos,byte *record, + ulong reclength); +static int delete_dynamic_record(MARIA_HA *info,my_off_t filepos, + uint second_read); +static int _ma_cmp_buffer(File file, const byte *buff, my_off_t filepos, + uint length); + +#ifdef THREAD +/* Play it safe; We have a small stack when using threads */ +#undef my_alloca +#undef my_afree +#define my_alloca(A) my_malloc((A),MYF(0)) +#define my_afree(A) my_free((A),MYF(0)) +#endif + + /* Interface function from MARIA_HA */ + +#ifdef HAVE_MMAP + +/* + Create mmaped area for MARIA handler + + SYNOPSIS + _ma_dynmap_file() + info MARIA handler + + RETURN + 0 ok + 1 error. +*/ + +my_bool _ma_dynmap_file(MARIA_HA *info, my_off_t size) +{ + DBUG_ENTER("_ma_dynmap_file"); + info->s->file_map= (byte*) + my_mmap(0, (size_t)(size + MEMMAP_EXTRA_MARGIN), + info->s->mode==O_RDONLY ? PROT_READ : + PROT_READ | PROT_WRITE, + MAP_SHARED | MAP_NORESERVE, + info->dfile, 0L); + if (info->s->file_map == (byte*) MAP_FAILED) + { + info->s->file_map= NULL; + DBUG_RETURN(1); + } +#if defined(HAVE_MADVISE) + madvise(info->s->file_map, size, MADV_RANDOM); +#endif + info->s->mmaped_length= size; + DBUG_RETURN(0); +} + + +/* + Resize mmaped area for MARIA handler + + SYNOPSIS + _ma_remap_file() + info MARIA handler + + RETURN +*/ + +void _ma_remap_file(MARIA_HA *info, my_off_t size) +{ + if (info->s->file_map) + { + VOID(my_munmap(info->s->file_map, + (size_t) info->s->mmaped_length + MEMMAP_EXTRA_MARGIN)); + _ma_dynmap_file(info, size); + } +} +#endif + + +/* + Read bytes from MySAM handler, using mmap or pread + + SYNOPSIS + _ma_mmap_pread() + info MARIA handler + Buffer Input buffer + Count Count of bytes for read + offset Start position + MyFlags + + RETURN + 0 ok +*/ + +uint _ma_mmap_pread(MARIA_HA *info, byte *Buffer, + uint Count, my_off_t offset, myf MyFlags) +{ + DBUG_PRINT("info", ("maria_read with mmap %d\n", info->dfile)); + if (info->s->concurrent_insert) + rw_rdlock(&info->s->mmap_lock); + + /* + The following test may fail in the following cases: + - We failed to remap a memory area (fragmented memory?) + - This thread has done some writes, but not yet extended the + memory mapped area. + */ + + if (info->s->mmaped_length >= offset + Count) + { + memcpy(Buffer, info->s->file_map + offset, Count); + if (info->s->concurrent_insert) + rw_unlock(&info->s->mmap_lock); + return 0; + } + else + { + if (info->s->concurrent_insert) + rw_unlock(&info->s->mmap_lock); + return my_pread(info->dfile, Buffer, Count, offset, MyFlags); + } +} + + + /* wrapper for my_pread in case if mmap isn't used */ + +uint _ma_nommap_pread(MARIA_HA *info, byte *Buffer, + uint Count, my_off_t offset, myf MyFlags) +{ + return my_pread(info->dfile, Buffer, Count, offset, MyFlags); +} + + +/* + Write bytes to MySAM handler, using mmap or pwrite + + SYNOPSIS + _ma_mmap_pwrite() + info MARIA handler + Buffer Output buffer + Count Count of bytes for write + offset Start position + MyFlags + + RETURN + 0 ok + !=0 error. In this case return error from pwrite +*/ + +uint _ma_mmap_pwrite(MARIA_HA *info, byte *Buffer, + uint Count, my_off_t offset, myf MyFlags) +{ + DBUG_PRINT("info", ("maria_write with mmap %d\n", info->dfile)); + if (info->s->concurrent_insert) + rw_rdlock(&info->s->mmap_lock); + + /* + The following test may fail in the following cases: + - We failed to remap a memory area (fragmented memory?) + - This thread has done some writes, but not yet extended the + memory mapped area. + */ + + if (info->s->mmaped_length >= offset + Count) + { + memcpy(info->s->file_map + offset, Buffer, Count); + if (info->s->concurrent_insert) + rw_unlock(&info->s->mmap_lock); + return 0; + } + else + { + info->s->nonmmaped_inserts++; + if (info->s->concurrent_insert) + rw_unlock(&info->s->mmap_lock); + return my_pwrite(info->dfile, Buffer, Count, offset, MyFlags); + } + +} + + + /* wrapper for my_pwrite in case if mmap isn't used */ + +uint _ma_nommap_pwrite(MARIA_HA *info, byte *Buffer, + uint Count, my_off_t offset, myf MyFlags) +{ + return my_pwrite(info->dfile, Buffer, Count, offset, MyFlags); +} + + +int _ma_write_dynamic_record(MARIA_HA *info, const byte *record) +{ + ulong reclength= _ma_rec_pack(info,info->rec_buff,record); + return (write_dynamic_record(info,info->rec_buff,reclength)); +} + +int _ma_update_dynamic_record(MARIA_HA *info, my_off_t pos, const byte *record) +{ + uint length= _ma_rec_pack(info,info->rec_buff,record); + return (update_dynamic_record(info,pos,info->rec_buff,length)); +} + +int _ma_write_blob_record(MARIA_HA *info, const byte *record) +{ + byte *rec_buff; + int error; + ulong reclength,reclength2,extra; + + extra= (ALIGN_SIZE(MARIA_MAX_DYN_BLOCK_HEADER)+MARIA_SPLIT_LENGTH+ + MARIA_DYN_DELETE_BLOCK_HEADER+1); + reclength= (info->s->base.pack_reclength + + _ma_calc_total_blob_length(info,record)+ extra); +#ifdef NOT_USED /* We now support big rows */ + if (reclength > MARIA_DYN_MAX_ROW_LENGTH) + { + my_errno=HA_ERR_TO_BIG_ROW; + return -1; + } +#endif + if (!(rec_buff=(byte*) my_alloca(reclength))) + { + my_errno=ENOMEM; + return(-1); + } + reclength2= _ma_rec_pack(info,rec_buff+ALIGN_SIZE(MARIA_MAX_DYN_BLOCK_HEADER), + record); + DBUG_PRINT("info",("reclength: %lu reclength2: %lu", + reclength, reclength2)); + DBUG_ASSERT(reclength2 <= reclength); + error=write_dynamic_record(info,rec_buff+ALIGN_SIZE(MARIA_MAX_DYN_BLOCK_HEADER), + reclength2); + my_afree(rec_buff); + return(error); +} + + +int _ma_update_blob_record(MARIA_HA *info, my_off_t pos, const byte *record) +{ + byte *rec_buff; + int error; + ulong reclength,extra; + + extra= (ALIGN_SIZE(MARIA_MAX_DYN_BLOCK_HEADER)+MARIA_SPLIT_LENGTH+ + MARIA_DYN_DELETE_BLOCK_HEADER); + reclength= (info->s->base.pack_reclength+ + _ma_calc_total_blob_length(info,record)+ extra); +#ifdef NOT_USED /* We now support big rows */ + if (reclength > MARIA_DYN_MAX_ROW_LENGTH) + { + my_errno=HA_ERR_TO_BIG_ROW; + return -1; + } +#endif + if (!(rec_buff=(byte*) my_alloca(reclength))) + { + my_errno=ENOMEM; + return(-1); + } + reclength= _ma_rec_pack(info,rec_buff+ALIGN_SIZE(MARIA_MAX_DYN_BLOCK_HEADER), + record); + error=update_dynamic_record(info,pos, + rec_buff+ALIGN_SIZE(MARIA_MAX_DYN_BLOCK_HEADER), + reclength); + my_afree(rec_buff); + return(error); +} + + +int _ma_delete_dynamic_record(MARIA_HA *info) +{ + return delete_dynamic_record(info,info->lastpos,0); +} + + + /* Write record to data-file */ + +static int write_dynamic_record(MARIA_HA *info, const byte *record, + ulong reclength) +{ + int flag; + ulong length; + my_off_t filepos; + DBUG_ENTER("write_dynamic_record"); + + flag=0; + do + { + if (_ma_find_writepos(info,reclength,&filepos,&length)) + goto err; + if (_ma_write_part_record(info,filepos,length, + (info->append_insert_at_end ? + HA_OFFSET_ERROR : info->s->state.dellink), + (byte**) &record,&reclength,&flag)) + goto err; + } while (reclength); + + DBUG_RETURN(0); +err: + DBUG_RETURN(1); +} + + + /* Get a block for data ; The given data-area must be used !! */ + +static int _ma_find_writepos(MARIA_HA *info, + ulong reclength, /* record length */ + my_off_t *filepos, /* Return file pos */ + ulong *length) /* length of block at filepos */ +{ + MARIA_BLOCK_INFO block_info; + ulong tmp; + DBUG_ENTER("_ma_find_writepos"); + + if (info->s->state.dellink != HA_OFFSET_ERROR && + !info->append_insert_at_end) + { + /* Deleted blocks exists; Get last used block */ + *filepos=info->s->state.dellink; + block_info.second_read=0; + info->rec_cache.seek_not_done=1; + if (!(_ma_get_block_info(&block_info,info->dfile,info->s->state.dellink) & + BLOCK_DELETED)) + { + DBUG_PRINT("error",("Delete link crashed")); + my_errno=HA_ERR_WRONG_IN_RECORD; + DBUG_RETURN(-1); + } + info->s->state.dellink=block_info.next_filepos; + info->state->del--; + info->state->empty-= block_info.block_len; + *length= block_info.block_len; + } + else + { + /* No deleted blocks; Allocate a new block */ + *filepos=info->state->data_file_length; + if ((tmp=reclength+3 + test(reclength >= (65520-3))) < + info->s->base.min_block_length) + tmp= info->s->base.min_block_length; + else + tmp= ((tmp+MARIA_DYN_ALIGN_SIZE-1) & + (~ (ulong) (MARIA_DYN_ALIGN_SIZE-1))); + if (info->state->data_file_length > + (info->s->base.max_data_file_length - tmp)) + { + my_errno=HA_ERR_RECORD_FILE_FULL; + DBUG_RETURN(-1); + } + if (tmp > MARIA_MAX_BLOCK_LENGTH) + tmp=MARIA_MAX_BLOCK_LENGTH; + *length= tmp; + info->state->data_file_length+= tmp; + info->s->state.split++; + info->update|=HA_STATE_WRITE_AT_END; + } + DBUG_RETURN(0); +} /* _ma_find_writepos */ + + + +/* + Unlink a deleted block from the deleted list. + This block will be combined with the preceding or next block to form + a big block. +*/ + +static bool unlink_deleted_block(MARIA_HA *info, MARIA_BLOCK_INFO *block_info) +{ + DBUG_ENTER("unlink_deleted_block"); + if (block_info->filepos == info->s->state.dellink) + { + /* First deleted block; We can just use this ! */ + info->s->state.dellink=block_info->next_filepos; + } + else + { + MARIA_BLOCK_INFO tmp; + tmp.second_read=0; + /* Unlink block from the previous block */ + if (!(_ma_get_block_info(&tmp,info->dfile,block_info->prev_filepos) + & BLOCK_DELETED)) + DBUG_RETURN(1); /* Something is wrong */ + mi_sizestore(tmp.header+4,block_info->next_filepos); + if (info->s->file_write(info,(char*) tmp.header+4,8, + block_info->prev_filepos+4, MYF(MY_NABP))) + DBUG_RETURN(1); + /* Unlink block from next block */ + if (block_info->next_filepos != HA_OFFSET_ERROR) + { + if (!(_ma_get_block_info(&tmp,info->dfile,block_info->next_filepos) + & BLOCK_DELETED)) + DBUG_RETURN(1); /* Something is wrong */ + mi_sizestore(tmp.header+12,block_info->prev_filepos); + if (info->s->file_write(info,(char*) tmp.header+12,8, + block_info->next_filepos+12, + MYF(MY_NABP))) + DBUG_RETURN(1); + } + } + /* We now have one less deleted block */ + info->state->del--; + info->state->empty-= block_info->block_len; + info->s->state.split--; + + /* + If this was a block that we where accessing through table scan + (maria_rrnd() or maria_scan(), then ensure that we skip over this block + when doing next maria_rrnd() or maria_scan(). + */ + if (info->nextpos == block_info->filepos) + info->nextpos+=block_info->block_len; + DBUG_RETURN(0); +} + + +/* + Add a backward link to delete block + + SYNOPSIS + update_backward_delete_link() + info MARIA handler + delete_block Position to delete block to update. + If this is 'HA_OFFSET_ERROR', nothing will be done + filepos Position to block that 'delete_block' should point to + + RETURN + 0 ok + 1 error. In this case my_error is set. +*/ + +static int update_backward_delete_link(MARIA_HA *info, my_off_t delete_block, + my_off_t filepos) +{ + MARIA_BLOCK_INFO block_info; + DBUG_ENTER("update_backward_delete_link"); + + if (delete_block != HA_OFFSET_ERROR) + { + block_info.second_read=0; + if (_ma_get_block_info(&block_info,info->dfile,delete_block) + & BLOCK_DELETED) + { + char buff[8]; + mi_sizestore(buff,filepos); + if (info->s->file_write(info,buff, 8, delete_block+12, MYF(MY_NABP))) + DBUG_RETURN(1); /* Error on write */ + } + else + { + my_errno=HA_ERR_WRONG_IN_RECORD; + DBUG_RETURN(1); /* Wrong delete link */ + } + } + DBUG_RETURN(0); +} + + /* Delete datarecord from database */ + /* info->rec_cache.seek_not_done is updated in cmp_record */ + +static int delete_dynamic_record(MARIA_HA *info, my_off_t filepos, + uint second_read) +{ + uint length,b_type; + MARIA_BLOCK_INFO block_info,del_block; + int error; + my_bool remove_next_block; + DBUG_ENTER("delete_dynamic_record"); + + /* First add a link from the last block to the new one */ + error= update_backward_delete_link(info, info->s->state.dellink, filepos); + + block_info.second_read=second_read; + do + { + /* Remove block at 'filepos' */ + if ((b_type= _ma_get_block_info(&block_info,info->dfile,filepos)) + & (BLOCK_DELETED | BLOCK_ERROR | BLOCK_SYNC_ERROR | + BLOCK_FATAL_ERROR) || + (length=(uint) (block_info.filepos-filepos) +block_info.block_len) < + MARIA_MIN_BLOCK_LENGTH) + { + my_errno=HA_ERR_WRONG_IN_RECORD; + DBUG_RETURN(1); + } + /* Check if next block is a delete block */ + del_block.second_read=0; + remove_next_block=0; + if (_ma_get_block_info(&del_block,info->dfile,filepos+length) & + BLOCK_DELETED && del_block.block_len+length < MARIA_DYN_MAX_BLOCK_LENGTH) + { + /* We can't remove this yet as this block may be the head block */ + remove_next_block=1; + length+=del_block.block_len; + } + + block_info.header[0]=0; + mi_int3store(block_info.header+1,length); + mi_sizestore(block_info.header+4,info->s->state.dellink); + if (b_type & BLOCK_LAST) + bfill(block_info.header+12,8,255); + else + mi_sizestore(block_info.header+12,block_info.next_filepos); + if (info->s->file_write(info,(byte*) block_info.header,20,filepos, + MYF(MY_NABP))) + DBUG_RETURN(1); + info->s->state.dellink = filepos; + info->state->del++; + info->state->empty+=length; + filepos=block_info.next_filepos; + + /* Now it's safe to unlink the deleted block directly after this one */ + if (remove_next_block && unlink_deleted_block(info,&del_block)) + error=1; + } while (!(b_type & BLOCK_LAST)); + + DBUG_RETURN(error); +} + + + /* Write a block to datafile */ + +int _ma_write_part_record(MARIA_HA *info, + my_off_t filepos, /* points at empty block */ + ulong length, /* length of block */ + my_off_t next_filepos,/* Next empty block */ + byte **record, /* pointer to record ptr */ + ulong *reclength, /* length of *record */ + int *flag) /* *flag == 0 if header */ +{ + ulong head_length,res_length,extra_length,long_block,del_length; + byte *pos,*record_end; + my_off_t next_delete_block; + uchar temp[MARIA_SPLIT_LENGTH+MARIA_DYN_DELETE_BLOCK_HEADER]; + DBUG_ENTER("_ma_write_part_record"); + + next_delete_block=HA_OFFSET_ERROR; + + res_length=extra_length=0; + if (length > *reclength + MARIA_SPLIT_LENGTH) + { /* Splitt big block */ + res_length=MY_ALIGN(length- *reclength - MARIA_EXTEND_BLOCK_LENGTH, + MARIA_DYN_ALIGN_SIZE); + length-= res_length; /* Use this for first part */ + } + long_block= (length < 65520L && *reclength < 65520L) ? 0 : 1; + if (length == *reclength+ 3 + long_block) + { + /* Block is exactly of the right length */ + temp[0]=(uchar) (1+ *flag)+(uchar) long_block; /* Flag is 0 or 6 */ + if (long_block) + { + mi_int3store(temp+1,*reclength); + head_length=4; + } + else + { + mi_int2store(temp+1,*reclength); + head_length=3; + } + } + else if (length-long_block < *reclength+4) + { /* To short block */ + if (next_filepos == HA_OFFSET_ERROR) + next_filepos= (info->s->state.dellink != HA_OFFSET_ERROR && + !info->append_insert_at_end ? + info->s->state.dellink : info->state->data_file_length); + if (*flag == 0) /* First block */ + { + if (*reclength > MARIA_MAX_BLOCK_LENGTH) + { + head_length= 16; + temp[0]=13; + mi_int4store(temp+1,*reclength); + mi_int3store(temp+5,length-head_length); + mi_sizestore((byte*) temp+8,next_filepos); + } + else + { + head_length=5+8+long_block*2; + temp[0]=5+(uchar) long_block; + if (long_block) + { + mi_int3store(temp+1,*reclength); + mi_int3store(temp+4,length-head_length); + mi_sizestore((byte*) temp+7,next_filepos); + } + else + { + mi_int2store(temp+1,*reclength); + mi_int2store(temp+3,length-head_length); + mi_sizestore((byte*) temp+5,next_filepos); + } + } + } + else + { + head_length=3+8+long_block; + temp[0]=11+(uchar) long_block; + if (long_block) + { + mi_int3store(temp+1,length-head_length); + mi_sizestore((byte*) temp+4,next_filepos); + } + else + { + mi_int2store(temp+1,length-head_length); + mi_sizestore((byte*) temp+3,next_filepos); + } + } + } + else + { /* Block with empty info last */ + head_length=4+long_block; + extra_length= length- *reclength-head_length; + temp[0]= (uchar) (3+ *flag)+(uchar) long_block; /* 3,4 or 9,10 */ + if (long_block) + { + mi_int3store(temp+1,*reclength); + temp[4]= (uchar) (extra_length); + } + else + { + mi_int2store(temp+1,*reclength); + temp[3]= (uchar) (extra_length); + } + length= *reclength+head_length; /* Write only what is needed */ + } + DBUG_DUMP("header",(byte*) temp,head_length); + + /* Make a long block for one write */ + record_end= *record+length-head_length; + del_length=(res_length ? MARIA_DYN_DELETE_BLOCK_HEADER : 0); + bmove((byte*) (*record-head_length),(byte*) temp,head_length); + memcpy(temp,record_end,(size_t) (extra_length+del_length)); + bzero((byte*) record_end,extra_length); + + if (res_length) + { + /* Check first if we can join this block with the next one */ + MARIA_BLOCK_INFO del_block; + my_off_t next_block=filepos+length+extra_length+res_length; + + del_block.second_read=0; + if (next_block < info->state->data_file_length && + info->s->state.dellink != HA_OFFSET_ERROR) + { + if ((_ma_get_block_info(&del_block,info->dfile,next_block) + & BLOCK_DELETED) && + res_length + del_block.block_len < MARIA_DYN_MAX_BLOCK_LENGTH) + { + if (unlink_deleted_block(info,&del_block)) + goto err; + res_length+=del_block.block_len; + } + } + + /* Create a delete link of the last part of the block */ + pos=record_end+extra_length; + pos[0]= '\0'; + mi_int3store(pos+1,res_length); + mi_sizestore(pos+4,info->s->state.dellink); + bfill(pos+12,8,255); /* End link */ + next_delete_block=info->s->state.dellink; + info->s->state.dellink= filepos+length+extra_length; + info->state->del++; + info->state->empty+=res_length; + info->s->state.split++; + } + if (info->opt_flag & WRITE_CACHE_USED && + info->update & HA_STATE_WRITE_AT_END) + { + if (info->update & HA_STATE_EXTEND_BLOCK) + { + info->update&= ~HA_STATE_EXTEND_BLOCK; + if (my_block_write(&info->rec_cache,(byte*) *record-head_length, + length+extra_length+del_length,filepos)) + goto err; + } + else if (my_b_write(&info->rec_cache,(byte*) *record-head_length, + length+extra_length+del_length)) + goto err; + } + else + { + info->rec_cache.seek_not_done=1; + if (info->s->file_write(info,(byte*) *record-head_length,length+extra_length+ + del_length,filepos,info->s->write_flag)) + goto err; + } + memcpy(record_end,temp,(size_t) (extra_length+del_length)); + *record=record_end; + *reclength-=(length-head_length); + *flag=6; + + if (del_length) + { + /* link the next delete block to this */ + if (update_backward_delete_link(info, next_delete_block, + info->s->state.dellink)) + goto err; + } + + DBUG_RETURN(0); +err: + DBUG_PRINT("exit",("errno: %d",my_errno)); + DBUG_RETURN(1); +} /* _ma_write_part_record */ + + + /* update record from datafile */ + +static int update_dynamic_record(MARIA_HA *info, my_off_t filepos, byte *record, + ulong reclength) +{ + int flag; + uint error; + ulong length; + MARIA_BLOCK_INFO block_info; + DBUG_ENTER("update_dynamic_record"); + + flag=block_info.second_read=0; + while (reclength > 0) + { + if (filepos != info->s->state.dellink) + { + block_info.next_filepos= HA_OFFSET_ERROR; + if ((error= _ma_get_block_info(&block_info,info->dfile,filepos)) + & (BLOCK_DELETED | BLOCK_ERROR | BLOCK_SYNC_ERROR | + BLOCK_FATAL_ERROR)) + { + DBUG_PRINT("error",("Got wrong block info")); + if (!(error & BLOCK_FATAL_ERROR)) + my_errno=HA_ERR_WRONG_IN_RECORD; + goto err; + } + length=(ulong) (block_info.filepos-filepos) + block_info.block_len; + if (length < reclength) + { + uint tmp=MY_ALIGN(reclength - length + 3 + + test(reclength >= 65520L),MARIA_DYN_ALIGN_SIZE); + /* Don't create a block bigger than MARIA_MAX_BLOCK_LENGTH */ + tmp= min(length+tmp, MARIA_MAX_BLOCK_LENGTH)-length; + /* Check if we can extend this block */ + if (block_info.filepos + block_info.block_len == + info->state->data_file_length && + info->state->data_file_length < + info->s->base.max_data_file_length-tmp) + { + /* extend file */ + DBUG_PRINT("info",("Extending file with %d bytes",tmp)); + if (info->nextpos == info->state->data_file_length) + info->nextpos+= tmp; + info->state->data_file_length+= tmp; + info->update|= HA_STATE_WRITE_AT_END | HA_STATE_EXTEND_BLOCK; + length+=tmp; + } + else if (length < MARIA_MAX_BLOCK_LENGTH - MARIA_MIN_BLOCK_LENGTH) + { + /* + Check if next block is a deleted block + Above we have MARIA_MIN_BLOCK_LENGTH to avoid the problem where + the next block is so small it can't be splited which could + casue problems + */ + + MARIA_BLOCK_INFO del_block; + del_block.second_read=0; + if (_ma_get_block_info(&del_block,info->dfile, + block_info.filepos + block_info.block_len) & + BLOCK_DELETED) + { + /* Use; Unlink it and extend the current block */ + DBUG_PRINT("info",("Extending current block")); + if (unlink_deleted_block(info,&del_block)) + goto err; + if ((length+=del_block.block_len) > MARIA_MAX_BLOCK_LENGTH) + { + /* + New block was too big, link overflow part back to + delete list + */ + my_off_t next_pos; + ulong rest_length= length-MARIA_MAX_BLOCK_LENGTH; + set_if_bigger(rest_length, MARIA_MIN_BLOCK_LENGTH); + next_pos= del_block.filepos+ del_block.block_len - rest_length; + + if (update_backward_delete_link(info, info->s->state.dellink, + next_pos)) + DBUG_RETURN(1); + + /* create delete link for data that didn't fit into the page */ + del_block.header[0]=0; + mi_int3store(del_block.header+1, rest_length); + mi_sizestore(del_block.header+4,info->s->state.dellink); + bfill(del_block.header+12,8,255); + if (info->s->file_write(info,(byte*) del_block.header,20, next_pos, + MYF(MY_NABP))) + DBUG_RETURN(1); + info->s->state.dellink= next_pos; + info->s->state.split++; + info->state->del++; + info->state->empty+= rest_length; + length-= rest_length; + } + } + } + } + } + else + { + if (_ma_find_writepos(info,reclength,&filepos,&length)) + goto err; + } + if (_ma_write_part_record(info,filepos,length,block_info.next_filepos, + &record,&reclength,&flag)) + goto err; + if ((filepos=block_info.next_filepos) == HA_OFFSET_ERROR) + { + /* Start writing data on deleted blocks */ + filepos=info->s->state.dellink; + } + } + + if (block_info.next_filepos != HA_OFFSET_ERROR) + if (delete_dynamic_record(info,block_info.next_filepos,1)) + goto err; + DBUG_RETURN(0); +err: + DBUG_RETURN(1); +} + + + /* Pack a record. Return new reclength */ + +uint _ma_rec_pack(MARIA_HA *info, register byte *to, register const byte *from) +{ + uint length,new_length,flag,bit,i; + char *pos,*end,*startpos,*packpos; + enum en_fieldtype type; + reg3 MARIA_COLUMNDEF *rec; + MARIA_BLOB *blob; + DBUG_ENTER("_ma_rec_pack"); + + flag=0 ; bit=1; + startpos=packpos=to; to+= info->s->base.pack_bits; blob=info->blobs; + rec=info->s->rec; + + for (i=info->s->base.fields ; i-- > 0; from+= length,rec++) + { + length=(uint) rec->length; + if ((type = (enum en_fieldtype) rec->type) != FIELD_NORMAL) + { + if (type == FIELD_BLOB) + { + if (!blob->length) + flag|=bit; + else + { + char *temp_pos; + size_t tmp_length=length-maria_portable_sizeof_char_ptr; + memcpy((byte*) to,from,tmp_length); + memcpy_fixed(&temp_pos,from+tmp_length,sizeof(char*)); + memcpy(to+tmp_length,temp_pos,(size_t) blob->length); + to+=tmp_length+blob->length; + } + blob++; + } + else if (type == FIELD_SKIP_ZERO) + { + if (memcmp((byte*) from,zero_string,length) == 0) + flag|=bit; + else + { + memcpy((byte*) to,from,(size_t) length); to+=length; + } + } + else if (type == FIELD_SKIP_ENDSPACE || + type == FIELD_SKIP_PRESPACE) + { + pos= (byte*) from; end= (byte*) from + length; + if (type == FIELD_SKIP_ENDSPACE) + { /* Pack trailing spaces */ + while (end > from && *(end-1) == ' ') + end--; + } + else + { /* Pack pref-spaces */ + while (pos < end && *pos == ' ') + pos++; + } + new_length=(uint) (end-pos); + if (new_length +1 + test(rec->length > 255 && new_length > 127) + < length) + { + if (rec->length > 255 && new_length > 127) + { + to[0]=(char) ((new_length & 127)+128); + to[1]=(char) (new_length >> 7); + to+=2; + } + else + *to++= (char) new_length; + memcpy((byte*) to,pos,(size_t) new_length); to+=new_length; + flag|=bit; + } + else + { + memcpy(to,from,(size_t) length); to+=length; + } + } + else if (type == FIELD_VARCHAR) + { + uint pack_length= HA_VARCHAR_PACKLENGTH(rec->length -1); + uint tmp_length; + if (pack_length == 1) + { + tmp_length= (uint) *(uchar*) from; + *to++= *from; + } + else + { + tmp_length= uint2korr(from); + store_key_length_inc(to,tmp_length); + } + memcpy(to, from+pack_length,tmp_length); + to+= tmp_length; + continue; + } + else + { + memcpy(to,from,(size_t) length); to+=length; + continue; /* Normal field */ + } + if ((bit= bit << 1) >= 256) + { + *packpos++ = (char) (uchar) flag; + bit=1; flag=0; + } + } + else + { + memcpy(to,from,(size_t) length); to+=length; + } + } + if (bit != 1) + *packpos= (char) (uchar) flag; + if (info->s->calc_checksum) + *to++=(char) info->checksum; + DBUG_PRINT("exit",("packed length: %d",(int) (to-startpos))); + DBUG_RETURN((uint) (to-startpos)); +} /* _ma_rec_pack */ + + + +/* + Check if a record was correctly packed. Used only by mariachk + Returns 0 if record is ok. +*/ + +my_bool _ma_rec_check(MARIA_HA *info,const char *record, byte *rec_buff, + ulong packed_length, my_bool with_checksum) +{ + uint length,new_length,flag,bit,i; + char *pos,*end,*packpos,*to; + enum en_fieldtype type; + reg3 MARIA_COLUMNDEF *rec; + DBUG_ENTER("_ma_rec_check"); + + packpos=rec_buff; to= rec_buff+info->s->base.pack_bits; + rec=info->s->rec; + flag= *packpos; bit=1; + + for (i=info->s->base.fields ; i-- > 0; record+= length, rec++) + { + length=(uint) rec->length; + if ((type = (enum en_fieldtype) rec->type) != FIELD_NORMAL) + { + if (type == FIELD_BLOB) + { + uint blob_length= + _ma_calc_blob_length(length-maria_portable_sizeof_char_ptr,record); + if (!blob_length && !(flag & bit)) + goto err; + if (blob_length) + to+=length - maria_portable_sizeof_char_ptr+ blob_length; + } + else if (type == FIELD_SKIP_ZERO) + { + if (memcmp((byte*) record,zero_string,length) == 0) + { + if (!(flag & bit)) + goto err; + } + else + to+=length; + } + else if (type == FIELD_SKIP_ENDSPACE || + type == FIELD_SKIP_PRESPACE) + { + pos= (byte*) record; end= (byte*) record + length; + if (type == FIELD_SKIP_ENDSPACE) + { /* Pack trailing spaces */ + while (end > record && *(end-1) == ' ') + end--; + } + else + { /* Pack pre-spaces */ + while (pos < end && *pos == ' ') + pos++; + } + new_length=(uint) (end-pos); + if (new_length +1 + test(rec->length > 255 && new_length > 127) + < length) + { + if (!(flag & bit)) + goto err; + if (rec->length > 255 && new_length > 127) + { + if (to[0] != (char) ((new_length & 127)+128) || + to[1] != (char) (new_length >> 7)) + goto err; + to+=2; + } + else if (*to++ != (char) new_length) + goto err; + to+=new_length; + } + else + to+=length; + } + else if (type == FIELD_VARCHAR) + { + uint pack_length= HA_VARCHAR_PACKLENGTH(rec->length -1); + uint tmp_length; + if (pack_length == 1) + { + tmp_length= (uint) *(uchar*) record; + to+= 1+ tmp_length; + continue; + } + else + { + tmp_length= uint2korr(record); + to+= get_pack_length(tmp_length)+tmp_length; + } + continue; + } + else + { + to+=length; + continue; /* Normal field */ + } + if ((bit= bit << 1) >= 256) + { + flag= *++packpos; + bit=1; + } + } + else + to+= length; + } + if (packed_length != (uint) (to - rec_buff) + test(info->s->calc_checksum) || + (bit != 1 && (flag & ~(bit - 1)))) + goto err; + if (with_checksum && ((uchar) info->checksum != (uchar) *to)) + { + DBUG_PRINT("error",("wrong checksum for row")); + goto err; + } + DBUG_RETURN(0); + +err: + DBUG_RETURN(1); +} + + + + /* Unpacks a record */ + /* Returns -1 and my_errno =HA_ERR_RECORD_DELETED if reclength isn't */ + /* right. Returns reclength (>0) if ok */ + +ulong _ma_rec_unpack(register MARIA_HA *info, register byte *to, byte *from, + ulong found_length) +{ + uint flag,bit,length,rec_length,min_pack_length; + enum en_fieldtype type; + byte *from_end,*to_end,*packpos; + reg3 MARIA_COLUMNDEF *rec,*end_field; + DBUG_ENTER("_ma_rec_unpack"); + + to_end=to + info->s->base.reclength; + from_end=from+found_length; + flag= (uchar) *from; bit=1; packpos=from; + if (found_length < info->s->base.min_pack_length) + goto err; + from+= info->s->base.pack_bits; + min_pack_length=info->s->base.min_pack_length - info->s->base.pack_bits; + + for (rec=info->s->rec , end_field=rec+info->s->base.fields ; + rec < end_field ; to+= rec_length, rec++) + { + rec_length=rec->length; + if ((type = (enum en_fieldtype) rec->type) != FIELD_NORMAL && + (type != FIELD_CHECK)) + { + if (type == FIELD_VARCHAR) + { + uint pack_length= HA_VARCHAR_PACKLENGTH(rec_length-1); + if (pack_length == 1) + { + length= (uint) *(uchar*) from; + if (length > rec_length-1) + goto err; + *to= *from++; + } + else + { + get_key_length(length, from); + if (length > rec_length-2) + goto err; + int2store(to,length); + } + if (from+length > from_end) + goto err; + memcpy(to+pack_length, from, length); + from+= length; + min_pack_length--; + continue; + } + if (flag & bit) + { + if (type == FIELD_BLOB || type == FIELD_SKIP_ZERO) + bzero((byte*) to,rec_length); + else if (type == FIELD_SKIP_ENDSPACE || + type == FIELD_SKIP_PRESPACE) + { + if (rec->length > 255 && *from & 128) + { + if (from + 1 >= from_end) + goto err; + length= (*from & 127)+ ((uint) (uchar) *(from+1) << 7); from+=2; + } + else + { + if (from == from_end) + goto err; + length= (uchar) *from++; + } + min_pack_length--; + if (length >= rec_length || + min_pack_length + length > (uint) (from_end - from)) + goto err; + if (type == FIELD_SKIP_ENDSPACE) + { + memcpy(to,(byte*) from,(size_t) length); + bfill((byte*) to+length,rec_length-length,' '); + } + else + { + bfill((byte*) to,rec_length-length,' '); + memcpy(to+rec_length-length,(byte*) from,(size_t) length); + } + from+=length; + } + } + else if (type == FIELD_BLOB) + { + uint size_length=rec_length- maria_portable_sizeof_char_ptr; + ulong blob_length= _ma_calc_blob_length(size_length,from); + if ((ulong) (from_end-from) - size_length < blob_length || + min_pack_length > (uint) (from_end -(from+size_length+blob_length))) + goto err; + memcpy((byte*) to,(byte*) from,(size_t) size_length); + from+=size_length; + memcpy_fixed((byte*) to+size_length,(byte*) &from,sizeof(char*)); + from+=blob_length; + } + else + { + if (type == FIELD_SKIP_ENDSPACE || type == FIELD_SKIP_PRESPACE) + min_pack_length--; + if (min_pack_length + rec_length > (uint) (from_end - from)) + goto err; + memcpy(to,(byte*) from,(size_t) rec_length); from+=rec_length; + } + if ((bit= bit << 1) >= 256) + { + flag= (uchar) *++packpos; bit=1; + } + } + else + { + if (min_pack_length > (uint) (from_end - from)) + goto err; + min_pack_length-=rec_length; + memcpy(to, (byte*) from, (size_t) rec_length); + from+=rec_length; + } + } + if (info->s->calc_checksum) + from++; + if (to == to_end && from == from_end && (bit == 1 || !(flag & ~(bit-1)))) + DBUG_RETURN(found_length); + +err: + my_errno= HA_ERR_WRONG_IN_RECORD; + DBUG_PRINT("error",("to_end: %lx -> %lx from_end: %lx -> %lx", + to,to_end,from,from_end)); + DBUG_DUMP("from",(byte*) info->rec_buff,info->s->base.min_pack_length); + DBUG_RETURN(MY_FILE_ERROR); +} /* _ma_rec_unpack */ + + + /* Calc length of blob. Update info in blobs->length */ + +ulong _ma_calc_total_blob_length(MARIA_HA *info, const byte *record) +{ + ulong length; + MARIA_BLOB *blob,*end; + + for (length=0, blob= info->blobs, end=blob+info->s->base.blobs ; + blob != end; + blob++) + { + blob->length= _ma_calc_blob_length(blob->pack_length,record + blob->offset); + length+=blob->length; + } + return length; +} + + +ulong _ma_calc_blob_length(uint length, const byte *pos) +{ + switch (length) { + case 1: + return (uint) (uchar) *pos; + case 2: + return (uint) uint2korr(pos); + case 3: + return uint3korr(pos); + case 4: + return uint4korr(pos); + default: + break; + } + return 0; /* Impossible */ +} + + +void _ma_store_blob_length(byte *pos,uint pack_length,uint length) +{ + switch (pack_length) { + case 1: + *pos= (uchar) length; + break; + case 2: + int2store(pos,length); + break; + case 3: + int3store(pos,length); + break; + case 4: + int4store(pos,length); + default: + break; + } + return; +} + + + /* Read record from datafile */ + /* Returns 0 if ok, -1 if error */ + +int _ma_read_dynamic_record(MARIA_HA *info, my_off_t filepos, byte *buf) +{ + int flag; + uint b_type,left_length; + byte *to; + MARIA_BLOCK_INFO block_info; + File file; + DBUG_ENTER("maria_read_dynamic_record"); + + if (filepos != HA_OFFSET_ERROR) + { + LINT_INIT(to); + LINT_INIT(left_length); + file=info->dfile; + block_info.next_filepos=filepos; /* for easyer loop */ + flag=block_info.second_read=0; + do + { + if (info->opt_flag & WRITE_CACHE_USED && + info->rec_cache.pos_in_file <= block_info.next_filepos && + flush_io_cache(&info->rec_cache)) + goto err; + info->rec_cache.seek_not_done=1; + if ((b_type= _ma_get_block_info(&block_info,file, + block_info.next_filepos)) + & (BLOCK_DELETED | BLOCK_ERROR | BLOCK_SYNC_ERROR | + BLOCK_FATAL_ERROR)) + { + if (b_type & (BLOCK_SYNC_ERROR | BLOCK_DELETED)) + my_errno=HA_ERR_RECORD_DELETED; + goto err; + } + if (flag == 0) /* First block */ + { + flag=1; + if (block_info.rec_len > (uint) info->s->base.max_pack_length) + goto panic; + if (info->s->base.blobs) + { + if (!(to=_ma_alloc_rec_buff(info, block_info.rec_len, + &info->rec_buff))) + goto err; + } + else + to= info->rec_buff; + left_length=block_info.rec_len; + } + if (left_length < block_info.data_len || ! block_info.data_len) + goto panic; /* Wrong linked record */ + if (info->s->file_read(info,(byte*) to,block_info.data_len,block_info.filepos, + MYF(MY_NABP))) + goto panic; + left_length-=block_info.data_len; + to+=block_info.data_len; + } while (left_length); + + info->update|= HA_STATE_AKTIV; /* We have a aktive record */ + fast_ma_writeinfo(info); + DBUG_RETURN(_ma_rec_unpack(info,buf,info->rec_buff,block_info.rec_len) != + MY_FILE_ERROR ? 0 : -1); + } + fast_ma_writeinfo(info); + DBUG_RETURN(-1); /* Wrong data to read */ + +panic: + my_errno=HA_ERR_WRONG_IN_RECORD; +err: + VOID(_ma_writeinfo(info,0)); + DBUG_RETURN(-1); +} + + /* compare unique constraint between stored rows */ + +int _ma_cmp_dynamic_unique(MARIA_HA *info, MARIA_UNIQUEDEF *def, + const byte *record, my_off_t pos) +{ + byte *rec_buff,*old_record; + int error; + DBUG_ENTER("_ma_cmp_dynamic_unique"); + + if (!(old_record=my_alloca(info->s->base.reclength))) + DBUG_RETURN(1); + + /* Don't let the compare destroy blobs that may be in use */ + rec_buff=info->rec_buff; + if (info->s->base.blobs) + info->rec_buff=0; + error= _ma_read_dynamic_record(info,pos,old_record); + if (!error) + error=_ma_unique_comp(def, record, old_record, def->null_are_equal); + if (info->s->base.blobs) + { + my_free(_ma_get_rec_buff_ptr(info, info->rec_buff), MYF(MY_ALLOW_ZERO_PTR)); + info->rec_buff=rec_buff; + } + my_afree(old_record); + DBUG_RETURN(error); +} + + + /* Compare of record one disk with packed record in memory */ + +int _ma_cmp_dynamic_record(register MARIA_HA *info, register const byte *record) +{ + uint flag,reclength,b_type; + my_off_t filepos; + byte *buffer; + MARIA_BLOCK_INFO block_info; + DBUG_ENTER("_ma_cmp_dynamic_record"); + + /* We are going to do changes; dont let anybody disturb */ + dont_break(); /* Dont allow SIGHUP or SIGINT */ + + if (info->opt_flag & WRITE_CACHE_USED) + { + info->update&= ~(HA_STATE_WRITE_AT_END | HA_STATE_EXTEND_BLOCK); + if (flush_io_cache(&info->rec_cache)) + DBUG_RETURN(-1); + } + info->rec_cache.seek_not_done=1; + + /* If nobody have touched the database we don't have to test rec */ + + buffer=info->rec_buff; + if ((info->opt_flag & READ_CHECK_USED)) + { /* If check isn't disabled */ + if (info->s->base.blobs) + { + if (!(buffer=(byte*) my_alloca(info->s->base.pack_reclength+ + _ma_calc_total_blob_length(info,record)))) + DBUG_RETURN(-1); + } + reclength= _ma_rec_pack(info,buffer,record); + record= buffer; + + filepos=info->lastpos; + flag=block_info.second_read=0; + block_info.next_filepos=filepos; + while (reclength > 0) + { + if ((b_type= _ma_get_block_info(&block_info,info->dfile, + block_info.next_filepos)) + & (BLOCK_DELETED | BLOCK_ERROR | BLOCK_SYNC_ERROR | + BLOCK_FATAL_ERROR)) + { + if (b_type & (BLOCK_SYNC_ERROR | BLOCK_DELETED)) + my_errno=HA_ERR_RECORD_CHANGED; + goto err; + } + if (flag == 0) /* First block */ + { + flag=1; + if (reclength != block_info.rec_len) + { + my_errno=HA_ERR_RECORD_CHANGED; + goto err; + } + } else if (reclength < block_info.data_len) + { + my_errno=HA_ERR_WRONG_IN_RECORD; + goto err; + } + reclength-=block_info.data_len; + if (_ma_cmp_buffer(info->dfile,record,block_info.filepos, + block_info.data_len)) + { + my_errno=HA_ERR_RECORD_CHANGED; + goto err; + } + flag=1; + record+=block_info.data_len; + } + } + my_errno=0; +err: + if (buffer != info->rec_buff) + my_afree((gptr) buffer); + DBUG_RETURN(my_errno); +} + + + /* Compare file to buffert */ + +static int _ma_cmp_buffer(File file, const byte *buff, my_off_t filepos, + uint length) +{ + uint next_length; + char temp_buff[IO_SIZE*2]; + DBUG_ENTER("_ma_cmp_buffer"); + + next_length= IO_SIZE*2 - (uint) (filepos & (IO_SIZE-1)); + + while (length > IO_SIZE*2) + { + if (my_pread(file,temp_buff,next_length,filepos, MYF(MY_NABP)) || + memcmp((byte*) buff,temp_buff,next_length)) + goto err; + filepos+=next_length; + buff+=next_length; + length-= next_length; + next_length=IO_SIZE*2; + } + if (my_pread(file,temp_buff,length,filepos,MYF(MY_NABP))) + goto err; + DBUG_RETURN(memcmp((byte*) buff,temp_buff,length)); +err: + DBUG_RETURN(1); +} + + +int _ma_read_rnd_dynamic_record(MARIA_HA *info, byte *buf, + register my_off_t filepos, + my_bool skip_deleted_blocks) +{ + int flag,info_read,save_errno; + uint left_len,b_type; + byte *to; + MARIA_BLOCK_INFO block_info; + MARIA_SHARE *share=info->s; + DBUG_ENTER("_ma_read_rnd_dynamic_record"); + + info_read=0; + LINT_INIT(to); + + if (info->lock_type == F_UNLCK) + { +#ifndef UNSAFE_LOCKING + if (share->tot_locks == 0) + { + if (my_lock(share->kfile,F_RDLCK,0L,F_TO_EOF, + MYF(MY_SEEK_NOT_DONE) | info->lock_wait)) + DBUG_RETURN(my_errno); + } +#else + info->tmp_lock_type=F_RDLCK; +#endif + } + else + info_read=1; /* memory-keyinfoblock is ok */ + + flag=block_info.second_read=0; + left_len=1; + do + { + if (filepos >= info->state->data_file_length) + { + if (!info_read) + { /* Check if changed */ + info_read=1; + info->rec_cache.seek_not_done=1; + if (_ma_state_info_read_dsk(share->kfile,&share->state,1)) + goto panic; + } + if (filepos >= info->state->data_file_length) + { + my_errno= HA_ERR_END_OF_FILE; + goto err; + } + } + if (info->opt_flag & READ_CACHE_USED) + { + if (_ma_read_cache(&info->rec_cache,(byte*) block_info.header,filepos, + sizeof(block_info.header), + (!flag && skip_deleted_blocks ? READING_NEXT : 0) | + READING_HEADER)) + goto panic; + b_type= _ma_get_block_info(&block_info,-1,filepos); + } + else + { + if (info->opt_flag & WRITE_CACHE_USED && + info->rec_cache.pos_in_file <= filepos && + flush_io_cache(&info->rec_cache)) + DBUG_RETURN(my_errno); + info->rec_cache.seek_not_done=1; + b_type= _ma_get_block_info(&block_info,info->dfile,filepos); + } + + if (b_type & (BLOCK_DELETED | BLOCK_ERROR | BLOCK_SYNC_ERROR | + BLOCK_FATAL_ERROR)) + { + if ((b_type & (BLOCK_DELETED | BLOCK_SYNC_ERROR)) + && skip_deleted_blocks) + { + filepos=block_info.filepos+block_info.block_len; + block_info.second_read=0; + continue; /* Search after next_record */ + } + if (b_type & (BLOCK_DELETED | BLOCK_SYNC_ERROR)) + { + my_errno=HA_ERR_RECORD_DELETED; + info->lastpos=block_info.filepos; + info->nextpos=block_info.filepos+block_info.block_len; + } + goto err; + } + if (flag == 0) /* First block */ + { + if (block_info.rec_len > (uint) share->base.max_pack_length) + goto panic; + info->lastpos=filepos; + if (share->base.blobs) + { + if (!(to= _ma_alloc_rec_buff(info, block_info.rec_len, + &info->rec_buff))) + goto err; + } + else + to= info->rec_buff; + left_len=block_info.rec_len; + } + if (left_len < block_info.data_len) + goto panic; /* Wrong linked record */ + + /* copy information that is already read */ + { + uint offset=(uint) (block_info.filepos - filepos); + uint tmp_length= (sizeof(block_info.header) - offset); + filepos=block_info.filepos; + + if (tmp_length > block_info.data_len) + tmp_length= block_info.data_len; + if (tmp_length) + { + memcpy((byte*) to, block_info.header+offset,tmp_length); + block_info.data_len-=tmp_length; + left_len-=tmp_length; + to+=tmp_length; + filepos+=tmp_length; + } + } + /* read rest of record from file */ + if (block_info.data_len) + { + if (info->opt_flag & READ_CACHE_USED) + { + if (_ma_read_cache(&info->rec_cache,(byte*) to,filepos, + block_info.data_len, + (!flag && skip_deleted_blocks) ? READING_NEXT :0)) + goto panic; + } + else + { + /* VOID(my_seek(info->dfile,filepos,MY_SEEK_SET,MYF(0))); */ + if (my_read(info->dfile,(byte*) to,block_info.data_len,MYF(MY_NABP))) + { + if (my_errno == -1) + my_errno= HA_ERR_WRONG_IN_RECORD; /* Unexpected end of file */ + goto err; + } + } + } + if (flag++ == 0) + { + info->nextpos=block_info.filepos+block_info.block_len; + skip_deleted_blocks=0; + } + left_len-=block_info.data_len; + to+=block_info.data_len; + filepos=block_info.next_filepos; + } while (left_len); + + info->update|= HA_STATE_AKTIV | HA_STATE_KEY_CHANGED; + fast_ma_writeinfo(info); + if (_ma_rec_unpack(info,buf,info->rec_buff,block_info.rec_len) != + MY_FILE_ERROR) + DBUG_RETURN(0); + DBUG_RETURN(my_errno); /* Wrong record */ + +panic: + my_errno=HA_ERR_WRONG_IN_RECORD; /* Something is fatal wrong */ +err: + save_errno=my_errno; + VOID(_ma_writeinfo(info,0)); + DBUG_RETURN(my_errno=save_errno); +} + + + /* Read and process header from a dynamic-record-file */ + +uint _ma_get_block_info(MARIA_BLOCK_INFO *info, File file, my_off_t filepos) +{ + uint return_val=0; + uchar *header=info->header; + + if (file >= 0) + { + VOID(my_seek(file,filepos,MY_SEEK_SET,MYF(0))); + if (my_read(file,(char*) header,sizeof(info->header),MYF(0)) != + sizeof(info->header)) + goto err; + } + DBUG_DUMP("header",(byte*) header,MARIA_BLOCK_INFO_HEADER_LENGTH); + if (info->second_read) + { + if (info->header[0] <= 6 || info->header[0] == 13) + return_val=BLOCK_SYNC_ERROR; + } + else + { + if (info->header[0] > 6 && info->header[0] != 13) + return_val=BLOCK_SYNC_ERROR; + } + info->next_filepos= HA_OFFSET_ERROR; /* Dummy if no next block */ + + switch (info->header[0]) { + case 0: + if ((info->block_len=(uint) mi_uint3korr(header+1)) < + MARIA_MIN_BLOCK_LENGTH || + (info->block_len & (MARIA_DYN_ALIGN_SIZE -1))) + goto err; + info->filepos=filepos; + info->next_filepos=mi_sizekorr(header+4); + info->prev_filepos=mi_sizekorr(header+12); +#if SIZEOF_OFF_T == 4 + if ((mi_uint4korr(header+4) != 0 && + (mi_uint4korr(header+4) != (ulong) ~0 || + info->next_filepos != (ulong) ~0)) || + (mi_uint4korr(header+12) != 0 && + (mi_uint4korr(header+12) != (ulong) ~0 || + info->prev_filepos != (ulong) ~0))) + goto err; +#endif + return return_val | BLOCK_DELETED; /* Deleted block */ + + case 1: + info->rec_len=info->data_len=info->block_len=mi_uint2korr(header+1); + info->filepos=filepos+3; + return return_val | BLOCK_FIRST | BLOCK_LAST; + case 2: + info->rec_len=info->data_len=info->block_len=mi_uint3korr(header+1); + info->filepos=filepos+4; + return return_val | BLOCK_FIRST | BLOCK_LAST; + + case 13: + info->rec_len=mi_uint4korr(header+1); + info->block_len=info->data_len=mi_uint3korr(header+5); + info->next_filepos=mi_sizekorr(header+8); + info->second_read=1; + info->filepos=filepos+16; + return return_val | BLOCK_FIRST; + + case 3: + info->rec_len=info->data_len=mi_uint2korr(header+1); + info->block_len=info->rec_len+ (uint) header[3]; + info->filepos=filepos+4; + return return_val | BLOCK_FIRST | BLOCK_LAST; + case 4: + info->rec_len=info->data_len=mi_uint3korr(header+1); + info->block_len=info->rec_len+ (uint) header[4]; + info->filepos=filepos+5; + return return_val | BLOCK_FIRST | BLOCK_LAST; + + case 5: + info->rec_len=mi_uint2korr(header+1); + info->block_len=info->data_len=mi_uint2korr(header+3); + info->next_filepos=mi_sizekorr(header+5); + info->second_read=1; + info->filepos=filepos+13; + return return_val | BLOCK_FIRST; + case 6: + info->rec_len=mi_uint3korr(header+1); + info->block_len=info->data_len=mi_uint3korr(header+4); + info->next_filepos=mi_sizekorr(header+7); + info->second_read=1; + info->filepos=filepos+15; + return return_val | BLOCK_FIRST; + + /* The following blocks are identical to 1-6 without rec_len */ + case 7: + info->data_len=info->block_len=mi_uint2korr(header+1); + info->filepos=filepos+3; + return return_val | BLOCK_LAST; + case 8: + info->data_len=info->block_len=mi_uint3korr(header+1); + info->filepos=filepos+4; + return return_val | BLOCK_LAST; + + case 9: + info->data_len=mi_uint2korr(header+1); + info->block_len=info->data_len+ (uint) header[3]; + info->filepos=filepos+4; + return return_val | BLOCK_LAST; + case 10: + info->data_len=mi_uint3korr(header+1); + info->block_len=info->data_len+ (uint) header[4]; + info->filepos=filepos+5; + return return_val | BLOCK_LAST; + + case 11: + info->data_len=info->block_len=mi_uint2korr(header+1); + info->next_filepos=mi_sizekorr(header+3); + info->second_read=1; + info->filepos=filepos+11; + return return_val; + case 12: + info->data_len=info->block_len=mi_uint3korr(header+1); + info->next_filepos=mi_sizekorr(header+4); + info->second_read=1; + info->filepos=filepos+12; + return return_val; + } + +err: + my_errno=HA_ERR_WRONG_IN_RECORD; /* Garbage */ + return BLOCK_ERROR; +} diff --git a/storage/maria/ma_extra.c b/storage/maria/ma_extra.c new file mode 100644 index 00000000000..bd5d4280c9d --- /dev/null +++ b/storage/maria/ma_extra.c @@ -0,0 +1,426 @@ +/* Copyright (C) 2006 MySQL AB & MySQL Finland AB & TCX DataKonsult AB + + 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; either version 2 of the License, or + (at your option) any later version. + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ + +#include "maria_def.h" +#ifdef HAVE_SYS_MMAN_H +#include +#endif + +static void maria_extra_keyflag(MARIA_HA *info, enum ha_extra_function function); + + +/* + Set options and buffers to optimize table handling + + SYNOPSIS + maria_extra() + info open table + function operation + extra_arg Pointer to extra argument (normally pointer to ulong) + Used when function is one of: + HA_EXTRA_WRITE_CACHE + HA_EXTRA_CACHE + RETURN VALUES + 0 ok + # error +*/ + +int maria_extra(MARIA_HA *info, enum ha_extra_function function, void *extra_arg) +{ + int error=0; + ulong cache_size; + MARIA_SHARE *share=info->s; + DBUG_ENTER("maria_extra"); + DBUG_PRINT("enter",("function: %d",(int) function)); + + switch (function) { + case HA_EXTRA_RESET: + /* + Free buffers and reset the following flags: + EXTRA_CACHE, EXTRA_WRITE_CACHE, EXTRA_KEYREAD, EXTRA_QUICK + + If the row buffer cache is large (for dynamic tables), reduce it + to save memory. + */ + if (info->opt_flag & (READ_CACHE_USED | WRITE_CACHE_USED)) + { + info->opt_flag&= ~(READ_CACHE_USED | WRITE_CACHE_USED); + error=end_io_cache(&info->rec_cache); + } + if (share->base.blobs) + _ma_alloc_rec_buff(info, -1, &info->rec_buff); +#if defined(HAVE_MMAP) && defined(HAVE_MADVISE) + if (info->opt_flag & MEMMAP_USED) + madvise(share->file_map,share->state.state.data_file_length,MADV_RANDOM); +#endif + info->opt_flag&= ~(KEY_READ_USED | REMEMBER_OLD_POS); + info->quick_mode=0; + /* Fall through */ + + case HA_EXTRA_RESET_STATE: /* Reset state (don't free buffers) */ + info->lastinx= 0; /* Use first index as def */ + info->last_search_keypage=info->lastpos= HA_OFFSET_ERROR; + info->page_changed=1; + /* Next/prev gives first/last */ + if (info->opt_flag & READ_CACHE_USED) + { + reinit_io_cache(&info->rec_cache,READ_CACHE,0, + (pbool) (info->lock_type != F_UNLCK), + (pbool) test(info->update & HA_STATE_ROW_CHANGED) + ); + } + info->update= ((info->update & HA_STATE_CHANGED) | HA_STATE_NEXT_FOUND | + HA_STATE_PREV_FOUND); + break; + case HA_EXTRA_CACHE: + if (info->lock_type == F_UNLCK && + (share->options & HA_OPTION_PACK_RECORD)) + { + error=1; /* Not possibly if not locked */ + my_errno=EACCES; + break; + } + if (info->s->file_map) /* Don't use cache if mmap */ + break; +#if defined(HAVE_MMAP) && defined(HAVE_MADVISE) + if ((share->options & HA_OPTION_COMPRESS_RECORD)) + { + pthread_mutex_lock(&share->intern_lock); + if (_ma_memmap_file(info)) + { + /* We don't nead MADV_SEQUENTIAL if small file */ + madvise(share->file_map,share->state.state.data_file_length, + share->state.state.data_file_length <= RECORD_CACHE_SIZE*16 ? + MADV_RANDOM : MADV_SEQUENTIAL); + pthread_mutex_unlock(&share->intern_lock); + break; + } + pthread_mutex_unlock(&share->intern_lock); + } +#endif + if (info->opt_flag & WRITE_CACHE_USED) + { + info->opt_flag&= ~WRITE_CACHE_USED; + if ((error=end_io_cache(&info->rec_cache))) + break; + } + if (!(info->opt_flag & + (READ_CACHE_USED | WRITE_CACHE_USED | MEMMAP_USED))) + { + cache_size= (extra_arg ? *(ulong*) extra_arg : + my_default_record_cache_size); + if (!(init_io_cache(&info->rec_cache,info->dfile, + (uint) min(info->state->data_file_length+1, + cache_size), + READ_CACHE,0L,(pbool) (info->lock_type != F_UNLCK), + MYF(share->write_flag & MY_WAIT_IF_FULL)))) + { + info->opt_flag|=READ_CACHE_USED; + info->update&= ~HA_STATE_ROW_CHANGED; + } + if (share->concurrent_insert) + info->rec_cache.end_of_file=info->state->data_file_length; + } + break; + case HA_EXTRA_REINIT_CACHE: + if (info->opt_flag & READ_CACHE_USED) + { + reinit_io_cache(&info->rec_cache,READ_CACHE,info->nextpos, + (pbool) (info->lock_type != F_UNLCK), + (pbool) test(info->update & HA_STATE_ROW_CHANGED)); + info->update&= ~HA_STATE_ROW_CHANGED; + if (share->concurrent_insert) + info->rec_cache.end_of_file=info->state->data_file_length; + } + break; + case HA_EXTRA_WRITE_CACHE: + if (info->lock_type == F_UNLCK) + { + error=1; /* Not possibly if not locked */ + break; + } + + cache_size= (extra_arg ? *(ulong*) extra_arg : + my_default_record_cache_size); + if (!(info->opt_flag & + (READ_CACHE_USED | WRITE_CACHE_USED | OPT_NO_ROWS)) && + !share->state.header.uniques) + if (!(init_io_cache(&info->rec_cache,info->dfile, cache_size, + WRITE_CACHE,info->state->data_file_length, + (pbool) (info->lock_type != F_UNLCK), + MYF(share->write_flag & MY_WAIT_IF_FULL)))) + { + info->opt_flag|=WRITE_CACHE_USED; + info->update&= ~(HA_STATE_ROW_CHANGED | + HA_STATE_WRITE_AT_END | + HA_STATE_EXTEND_BLOCK); + } + break; + case HA_EXTRA_PREPARE_FOR_UPDATE: + if (info->s->data_file_type != DYNAMIC_RECORD) + break; + /* Remove read/write cache if dynamic rows */ + case HA_EXTRA_NO_CACHE: + if (info->opt_flag & (READ_CACHE_USED | WRITE_CACHE_USED)) + { + info->opt_flag&= ~(READ_CACHE_USED | WRITE_CACHE_USED); + error=end_io_cache(&info->rec_cache); + /* Sergei will insert full text index caching here */ + } +#if defined(HAVE_MMAP) && defined(HAVE_MADVISE) + if (info->opt_flag & MEMMAP_USED) + madvise(share->file_map,share->state.state.data_file_length,MADV_RANDOM); +#endif + break; + case HA_EXTRA_FLUSH_CACHE: + if (info->opt_flag & WRITE_CACHE_USED) + { + if ((error=flush_io_cache(&info->rec_cache))) + { + maria_print_error(info->s, HA_ERR_CRASHED); + maria_mark_crashed(info); /* Fatal error found */ + } + } + break; + case HA_EXTRA_NO_READCHECK: + info->opt_flag&= ~READ_CHECK_USED; /* No readcheck */ + break; + case HA_EXTRA_READCHECK: + info->opt_flag|= READ_CHECK_USED; + break; + case HA_EXTRA_KEYREAD: /* Read only keys to record */ + case HA_EXTRA_REMEMBER_POS: + info->opt_flag |= REMEMBER_OLD_POS; + bmove((byte*) info->lastkey+share->base.max_key_length*2, + (byte*) info->lastkey,info->lastkey_length); + info->save_update= info->update; + info->save_lastinx= info->lastinx; + info->save_lastpos= info->lastpos; + info->save_lastkey_length=info->lastkey_length; + if (function == HA_EXTRA_REMEMBER_POS) + break; + /* fall through */ + case HA_EXTRA_KEYREAD_CHANGE_POS: + info->opt_flag |= KEY_READ_USED; + info->read_record= _ma_read_key_record; + break; + case HA_EXTRA_NO_KEYREAD: + case HA_EXTRA_RESTORE_POS: + if (info->opt_flag & REMEMBER_OLD_POS) + { + bmove((byte*) info->lastkey, + (byte*) info->lastkey+share->base.max_key_length*2, + info->save_lastkey_length); + info->update= info->save_update | HA_STATE_WRITTEN; + info->lastinx= info->save_lastinx; + info->lastpos= info->save_lastpos; + info->lastkey_length=info->save_lastkey_length; + } + info->read_record= share->read_record; + info->opt_flag&= ~(KEY_READ_USED | REMEMBER_OLD_POS); + break; + case HA_EXTRA_NO_USER_CHANGE: /* Database is somehow locked agains changes */ + info->lock_type= F_EXTRA_LCK; /* Simulate as locked */ + break; + case HA_EXTRA_WAIT_LOCK: + info->lock_wait=0; + break; + case HA_EXTRA_NO_WAIT_LOCK: + info->lock_wait=MY_DONT_WAIT; + break; + case HA_EXTRA_NO_KEYS: + if (info->lock_type == F_UNLCK) + { + error=1; /* Not possibly if not lock */ + break; + } + if (maria_is_any_key_active(share->state.key_map)) + { + MARIA_KEYDEF *key=share->keyinfo; + uint i; + for (i=0 ; i < share->base.keys ; i++,key++) + { + if (!(key->flag & HA_NOSAME) && info->s->base.auto_key != i+1) + { + maria_clear_key_active(share->state.key_map, i); + info->update|= HA_STATE_CHANGED; + } + } + + if (!share->changed) + { + share->state.changed|= STATE_CHANGED | STATE_NOT_ANALYZED; + share->changed=1; /* Update on close */ + if (!share->global_changed) + { + share->global_changed=1; + share->state.open_count++; + } + } + share->state.state= *info->state; + error=_ma_state_info_write(share->kfile,&share->state,1 | 2); + } + break; + case HA_EXTRA_FORCE_REOPEN: + pthread_mutex_lock(&THR_LOCK_maria); + share->last_version= 0L; /* Impossible version */ + pthread_mutex_unlock(&THR_LOCK_maria); + break; + case HA_EXTRA_PREPARE_FOR_DELETE: + pthread_mutex_lock(&THR_LOCK_maria); + share->last_version= 0L; /* Impossible version */ +#ifdef __WIN__ + /* Close the isam and data files as Win32 can't drop an open table */ + pthread_mutex_lock(&share->intern_lock); + if (flush_key_blocks(share->key_cache, share->kfile, + (function == HA_EXTRA_FORCE_REOPEN ? + FLUSH_RELEASE : FLUSH_IGNORE_CHANGED))) + { + error=my_errno; + share->changed=1; + maria_print_error(info->s, HA_ERR_CRASHED); + maria_mark_crashed(info); /* Fatal error found */ + } + if (info->opt_flag & (READ_CACHE_USED | WRITE_CACHE_USED)) + { + info->opt_flag&= ~(READ_CACHE_USED | WRITE_CACHE_USED); + error=end_io_cache(&info->rec_cache); + } + if (info->lock_type != F_UNLCK && ! info->was_locked) + { + info->was_locked=info->lock_type; + if (maria_lock_database(info,F_UNLCK)) + error=my_errno; + info->lock_type = F_UNLCK; + } + if (share->kfile >= 0) + _ma_decrement_open_count(info); + if (share->kfile >= 0 && my_close(share->kfile,MYF(0))) + error=my_errno; + { + LIST *list_element ; + for (list_element=maria_open_list ; + list_element ; + list_element=list_element->next) + { + MARIA_HA *tmpinfo=(MARIA_HA*) list_element->data; + if (tmpinfo->s == info->s) + { + if (tmpinfo->dfile >= 0 && my_close(tmpinfo->dfile,MYF(0))) + error = my_errno; + tmpinfo->dfile= -1; + } + } + } + share->kfile= -1; /* Files aren't open anymore */ + pthread_mutex_unlock(&share->intern_lock); +#endif + pthread_mutex_unlock(&THR_LOCK_maria); + break; + case HA_EXTRA_FLUSH: + if (!share->temporary) + flush_key_blocks(share->key_cache, share->kfile, FLUSH_KEEP); +#ifdef HAVE_PWRITE + _ma_decrement_open_count(info); +#endif + if (share->not_flushed) + { + share->not_flushed=0; + if (my_sync(share->kfile, MYF(0))) + error= my_errno; + if (my_sync(info->dfile, MYF(0))) + error= my_errno; + if (error) + { + share->changed=1; + maria_print_error(info->s, HA_ERR_CRASHED); + maria_mark_crashed(info); /* Fatal error found */ + } + } + if (share->base.blobs) + _ma_alloc_rec_buff(info, -1, &info->rec_buff); + break; + case HA_EXTRA_NORMAL: /* Theese isn't in use */ + info->quick_mode=0; + break; + case HA_EXTRA_QUICK: + info->quick_mode=1; + break; + case HA_EXTRA_NO_ROWS: + if (!share->state.header.uniques) + info->opt_flag|= OPT_NO_ROWS; + break; + case HA_EXTRA_PRELOAD_BUFFER_SIZE: + info->preload_buff_size= *((ulong *) extra_arg); + break; + case HA_EXTRA_CHANGE_KEY_TO_UNIQUE: + case HA_EXTRA_CHANGE_KEY_TO_DUP: + maria_extra_keyflag(info, function); + break; + case HA_EXTRA_MMAP: +#ifdef HAVE_MMAP + pthread_mutex_lock(&share->intern_lock); + if (!share->file_map) + { + if (_ma_dynmap_file(info, share->state.state.data_file_length)) + { + DBUG_PRINT("warning",("mmap failed: errno: %d",errno)); + error= my_errno= errno; + } + else + { + share->file_read= _ma_mmap_pread; + share->file_write= _ma_mmap_pwrite; + } + } + pthread_mutex_unlock(&share->intern_lock); +#endif + break; + case HA_EXTRA_KEY_CACHE: + case HA_EXTRA_NO_KEY_CACHE: + default: + break; + } + { + char tmp[1]; + tmp[0]=function; + maria_log_command(MARIA_LOG_EXTRA,info,(byte*) tmp,1,error); + } + DBUG_RETURN(error); +} /* maria_extra */ + + +/* + Start/Stop Inserting Duplicates Into a Table, WL#1648. + */ +static void maria_extra_keyflag(MARIA_HA *info, enum ha_extra_function function) +{ + uint idx; + + for (idx= 0; idx< info->s->base.keys; idx++) + { + switch (function) { + case HA_EXTRA_CHANGE_KEY_TO_UNIQUE: + info->s->keyinfo[idx].flag|= HA_NOSAME; + break; + case HA_EXTRA_CHANGE_KEY_TO_DUP: + info->s->keyinfo[idx].flag&= ~(HA_NOSAME); + break; + default: + break; + } + } +} diff --git a/storage/maria/ma_ft_boolean_search.c b/storage/maria/ma_ft_boolean_search.c new file mode 100644 index 00000000000..2b8e0d8b97a --- /dev/null +++ b/storage/maria/ma_ft_boolean_search.c @@ -0,0 +1,955 @@ +/* Copyright (C) 2006 MySQL AB & MySQL Finland AB & TCX DataKonsult AB + + 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; either version 2 of the License, or + (at your option) any later version. + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ + +/* Written by Sergei A. Golubchik, who has a shared copyright to this code */ + +/* TODO: add caching - pre-read several index entries at once */ + +/* + Added optimization for full-text queries with plus-words. It was + implemented by sharing maximal document id (max_docid) variable + inside plus subtree. max_docid could be used by any word in plus + subtree, but it could be updated by plus-word only. + + The idea is: there is no need to search for docid smaller than + biggest docid inside current plus subtree. + + Examples: + +word1 word2 + share same max_docid + max_docid updated by word1 + +word1 +(word2 word3) + share same max_docid + max_docid updated by word1 + +(word1 -word2) +(+word3 word4) + share same max_docid + max_docid updated by word3 +*/ + +#define FT_CORE +#include "ma_ftdefs.h" + +/* search with boolean queries */ + +static double _wghts[11]= +{ + 0.131687242798354, + 0.197530864197531, + 0.296296296296296, + 0.444444444444444, + 0.666666666666667, + 1.000000000000000, + 1.500000000000000, + 2.250000000000000, + 3.375000000000000, + 5.062500000000000, + 7.593750000000000}; +static double *wghts=_wghts+5; /* wghts[i] = 1.5**i */ + +static double _nwghts[11]= +{ + -0.065843621399177, + -0.098765432098766, + -0.148148148148148, + -0.222222222222222, + -0.333333333333334, + -0.500000000000000, + -0.750000000000000, + -1.125000000000000, + -1.687500000000000, + -2.531250000000000, + -3.796875000000000}; +static double *nwghts=_nwghts+5; /* nwghts[i] = -0.5*1.5**i */ + +#define FTB_FLAG_TRUNC 1 +/* At most one of the following flags can be set */ +#define FTB_FLAG_YES 2 +#define FTB_FLAG_NO 4 +#define FTB_FLAG_WONLY 8 + +typedef struct st_ftb_expr FTB_EXPR; +struct st_ftb_expr +{ + FTB_EXPR *up; + uint flags; +/* ^^^^^^^^^^^^^^^^^^ FTB_{EXPR,WORD} common section */ + my_off_t docid[2]; + my_off_t max_docid; + float weight; + float cur_weight; + LIST *phrase; /* phrase words */ + LIST *document; /* for phrase search */ + uint yesses; /* number of "yes" words matched */ + uint nos; /* number of "no" words matched */ + uint ythresh; /* number of "yes" words in expr */ + uint yweaks; /* number of "yes" words for scan only */ +}; + +typedef struct st_ftb_word +{ + FTB_EXPR *up; + uint flags; +/* ^^^^^^^^^^^^^^^^^^ FTB_{EXPR,WORD} common section */ + my_off_t docid[2]; /* for index search and for scan */ + my_off_t key_root; + my_off_t *max_docid; + MARIA_KEYDEF *keyinfo; + struct st_ftb_word *prev; + float weight; + uint ndepth; + uint len; + uchar off; + byte word[1]; +} FTB_WORD; + +typedef struct st_ft_info +{ + struct _ft_vft *please; + MARIA_HA *info; + CHARSET_INFO *charset; + FTB_EXPR *root; + FTB_WORD **list; + FTB_WORD *last_word; + MEM_ROOT mem_root; + QUEUE queue; + TREE no_dupes; + my_off_t lastpos; + uint keynr; + uchar with_scan; + enum { UNINITIALIZED, READY, INDEX_SEARCH, INDEX_DONE } state; +} FTB; + +static int FTB_WORD_cmp(my_off_t *v, FTB_WORD *a, FTB_WORD *b) +{ + int i; + + /* if a==curdoc, take it as a < b */ + if (v && a->docid[0] == *v) + return -1; + + /* ORDER BY docid, ndepth DESC */ + i=CMP_NUM(a->docid[0], b->docid[0]); + if (!i) + i=CMP_NUM(b->ndepth,a->ndepth); + return i; +} + +static int FTB_WORD_cmp_list(CHARSET_INFO *cs, FTB_WORD **a, FTB_WORD **b) +{ + /* ORDER BY word DESC, ndepth DESC */ + int i= ha_compare_text(cs, (uchar*) (*b)->word+1,(*b)->len-1, + (uchar*) (*a)->word+1,(*a)->len-1,0,0); + if (!i) + i=CMP_NUM((*b)->ndepth,(*a)->ndepth); + return i; +} + + +typedef struct st_my_ftb_param +{ + FTB *ftb; + FTB_EXPR *ftbe; + byte *up_quot; + uint depth; +} MY_FTB_PARAM; + + +static int ftb_query_add_word(void *param, char *word, int word_len, + MYSQL_FTPARSER_BOOLEAN_INFO *info) +{ + MY_FTB_PARAM *ftb_param= (MY_FTB_PARAM *)param; + FTB_WORD *ftbw; + FTB_EXPR *ftbe, *tmp_expr; + FT_WORD *phrase_word; + LIST *tmp_element; + int r= info->weight_adjust; + float weight= (float) + (info->wasign ? nwghts : wghts)[(r>5)?5:((r<-5)?-5:r)]; + + switch (info->type) { + case FT_TOKEN_WORD: + ftbw= (FTB_WORD *)alloc_root(&ftb_param->ftb->mem_root, + sizeof(FTB_WORD) + + (info->trunc ? HA_MAX_KEY_BUFF : + word_len * ftb_param->ftb->charset->mbmaxlen + + HA_FT_WLEN + + ftb_param->ftb->info->s->rec_reflength)); + ftbw->len= word_len + 1; + ftbw->flags= 0; + ftbw->off= 0; + if (info->yesno > 0) ftbw->flags|= FTB_FLAG_YES; + if (info->yesno < 0) ftbw->flags|= FTB_FLAG_NO; + if (info->trunc) ftbw->flags|= FTB_FLAG_TRUNC; + ftbw->weight= weight; + ftbw->up= ftb_param->ftbe; + ftbw->docid[0]= ftbw->docid[1]= HA_OFFSET_ERROR; + ftbw->ndepth= (info->yesno < 0) + ftb_param->depth; + ftbw->key_root= HA_OFFSET_ERROR; + memcpy(ftbw->word + 1, word, word_len); + ftbw->word[0]= word_len; + if (info->yesno > 0) ftbw->up->ythresh++; + ftb_param->ftb->queue.max_elements++; + ftbw->prev= ftb_param->ftb->last_word; + ftb_param->ftb->last_word= ftbw; + ftb_param->ftb->with_scan|= (info->trunc & FTB_FLAG_TRUNC); + for (tmp_expr= ftb_param->ftbe; tmp_expr->up; tmp_expr= tmp_expr->up) + if (! (tmp_expr->flags & FTB_FLAG_YES)) + break; + ftbw->max_docid= &tmp_expr->max_docid; + /* fall through */ + case FT_TOKEN_STOPWORD: + if (! ftb_param->up_quot) break; + phrase_word= (FT_WORD *)alloc_root(&ftb_param->ftb->mem_root, sizeof(FT_WORD)); + tmp_element= (LIST *)alloc_root(&ftb_param->ftb->mem_root, sizeof(LIST)); + phrase_word->pos= word; + phrase_word->len= word_len; + tmp_element->data= (void *)phrase_word; + ftb_param->ftbe->phrase= list_add(ftb_param->ftbe->phrase, tmp_element); + /* Allocate document list at this point. + It allows to avoid huge amount of allocs/frees for each row.*/ + tmp_element= (LIST *)alloc_root(&ftb_param->ftb->mem_root, sizeof(LIST)); + tmp_element->data= alloc_root(&ftb_param->ftb->mem_root, sizeof(FT_WORD)); + ftb_param->ftbe->document= + list_add(ftb_param->ftbe->document, tmp_element); + break; + case FT_TOKEN_LEFT_PAREN: + ftbe=(FTB_EXPR *)alloc_root(&ftb_param->ftb->mem_root, sizeof(FTB_EXPR)); + ftbe->flags= 0; + if (info->yesno > 0) ftbe->flags|= FTB_FLAG_YES; + if (info->yesno < 0) ftbe->flags|= FTB_FLAG_NO; + ftbe->weight= weight; + ftbe->up= ftb_param->ftbe; + ftbe->max_docid= ftbe->ythresh= ftbe->yweaks= 0; + ftbe->docid[0]= ftbe->docid[1]= HA_OFFSET_ERROR; + ftbe->phrase= NULL; + ftbe->document= 0; + if (info->quot) ftb_param->ftb->with_scan|= 2; + if (info->yesno > 0) ftbe->up->ythresh++; + ftb_param->ftbe= ftbe; + ftb_param->depth++; + ftb_param->up_quot= info->quot; + break; + case FT_TOKEN_RIGHT_PAREN: + if (ftb_param->ftbe->document) + { + /* Circuit document list */ + for (tmp_element= ftb_param->ftbe->document; + tmp_element->next; tmp_element= tmp_element->next) /* no-op */; + tmp_element->next= ftb_param->ftbe->document; + ftb_param->ftbe->document->prev= tmp_element; + } + info->quot= 0; + if (ftb_param->ftbe->up) + { + DBUG_ASSERT(ftb_param->depth); + ftb_param->ftbe= ftb_param->ftbe->up; + ftb_param->depth--; + ftb_param->up_quot= 0; + } + break; + case FT_TOKEN_EOF: + default: + break; + } + return(0); +} + + +static int ftb_parse_query_internal(void *param, char *query, int len) +{ + MY_FTB_PARAM *ftb_param= (MY_FTB_PARAM *)param; + MYSQL_FTPARSER_BOOLEAN_INFO info; + CHARSET_INFO *cs= ftb_param->ftb->charset; + char **start= &query; + char *end= query + len; + FT_WORD w; + + info.prev= ' '; + info.quot= 0; + while (maria_ft_get_word(cs, start, end, &w, &info)) + ftb_query_add_word(param, w.pos, w.len, &info); + return(0); +} + + +static void _ftb_parse_query(FTB *ftb, byte *query, uint len, + struct st_mysql_ftparser *parser) +{ + MYSQL_FTPARSER_PARAM *param; + MY_FTB_PARAM ftb_param; + DBUG_ENTER("_ftb_parse_query"); + DBUG_ASSERT(parser); + + if (ftb->state != UNINITIALIZED) + DBUG_VOID_RETURN; + + ftb_param.ftb= ftb; + ftb_param.depth= 0; + ftb_param.ftbe= ftb->root; + ftb_param.up_quot= 0; + + if (! (param= maria_ftparser_call_initializer(ftb->info, ftb->keynr))) + DBUG_VOID_RETURN; + param->mysql_parse= ftb_parse_query_internal; + param->mysql_add_word= ftb_query_add_word; + param->mysql_ftparam= (void *)&ftb_param; + param->cs= ftb->charset; + param->doc= query; + param->length= len; + param->mode= MYSQL_FTPARSER_FULL_BOOLEAN_INFO; + parser->parse(param); + DBUG_VOID_RETURN; +} + + +static int _ftb_no_dupes_cmp(void* not_used __attribute__((unused)), + const void *a,const void *b) +{ + return CMP_NUM((*((my_off_t*)a)), (*((my_off_t*)b))); +} + +/* returns 1 if the search was finished (must-word wasn't found) */ +static int _ft2_search(FTB *ftb, FTB_WORD *ftbw, my_bool init_search) +{ + int r; + int subkeys=1; + my_bool can_go_down; + MARIA_HA *info=ftb->info; + uint off, extra=HA_FT_WLEN+info->s->base.rec_reflength; + byte *lastkey_buf=ftbw->word+ftbw->off; + LINT_INIT(off); + + if (ftbw->flags & FTB_FLAG_TRUNC) + lastkey_buf+=ftbw->len; + + if (init_search) + { + ftbw->key_root=info->s->state.key_root[ftb->keynr]; + ftbw->keyinfo=info->s->keyinfo+ftb->keynr; + + r= _ma_search(info, ftbw->keyinfo, (uchar*) ftbw->word, ftbw->len, + SEARCH_FIND | SEARCH_BIGGER, ftbw->key_root); + } + else + { + uint sflag= SEARCH_BIGGER; + if (ftbw->docid[0] < *ftbw->max_docid) + { + sflag|= SEARCH_SAME; + _ma_dpointer(info, (uchar *)(ftbw->word + ftbw->len + HA_FT_WLEN), + *ftbw->max_docid); + } + r= _ma_search(info, ftbw->keyinfo, (uchar*) lastkey_buf, + USE_WHOLE_KEY, sflag, ftbw->key_root); + } + + can_go_down=(!ftbw->off && (init_search || (ftbw->flags & FTB_FLAG_TRUNC))); + /* Skip rows inserted by concurrent insert */ + while (!r) + { + if (can_go_down) + { + /* going down ? */ + off=info->lastkey_length-extra; + subkeys=ft_sintXkorr(info->lastkey+off); + } + if (subkeys<0 || info->lastpos < info->state->data_file_length) + break; + r= _ma_search_next(info, ftbw->keyinfo, info->lastkey, + info->lastkey_length, + SEARCH_BIGGER, ftbw->key_root); + } + + if (!r && !ftbw->off) + { + r= ha_compare_text(ftb->charset, + info->lastkey+1, + info->lastkey_length-extra-1, + (uchar*) ftbw->word+1, + ftbw->len-1, + (my_bool) (ftbw->flags & FTB_FLAG_TRUNC),0); + } + + if (r) /* not found */ + { + if (!ftbw->off || !(ftbw->flags & FTB_FLAG_TRUNC)) + { + ftbw->docid[0]=HA_OFFSET_ERROR; + if ((ftbw->flags & FTB_FLAG_YES) && ftbw->up->up==0) + { + /* + This word MUST BE present in every document returned, + so we can stop the search right now + */ + ftb->state=INDEX_DONE; + return 1; /* search is done */ + } + else + return 0; + } + + /* going up to the first-level tree to continue search there */ + _ma_dpointer(info, (uchar*) (lastkey_buf+HA_FT_WLEN), ftbw->key_root); + ftbw->key_root=info->s->state.key_root[ftb->keynr]; + ftbw->keyinfo=info->s->keyinfo+ftb->keynr; + ftbw->off=0; + return _ft2_search(ftb, ftbw, 0); + } + + /* matching key found */ + memcpy(lastkey_buf, info->lastkey, info->lastkey_length); + if (lastkey_buf == ftbw->word) + ftbw->len=info->lastkey_length-extra; + + /* going down ? */ + if (subkeys<0) + { + /* + yep, going down, to the second-level tree + TODO here: subkey-based optimization + */ + ftbw->off=off; + ftbw->key_root=info->lastpos; + ftbw->keyinfo=& info->s->ft2_keyinfo; + r= _ma_search_first(info, ftbw->keyinfo, ftbw->key_root); + DBUG_ASSERT(r==0); /* found something */ + memcpy(lastkey_buf+off, info->lastkey, info->lastkey_length); + } + ftbw->docid[0]=info->lastpos; + if (ftbw->flags & FTB_FLAG_YES) + *ftbw->max_docid= info->lastpos; + return 0; +} + +static void _ftb_init_index_search(FT_INFO *ftb) +{ + int i; + FTB_WORD *ftbw; + + if ((ftb->state != READY && ftb->state !=INDEX_DONE) || + ftb->keynr == NO_SUCH_KEY) + return; + ftb->state=INDEX_SEARCH; + + for (i=ftb->queue.elements; i; i--) + { + ftbw=(FTB_WORD *)(ftb->queue.root[i]); + + if (ftbw->flags & FTB_FLAG_TRUNC) + { + /* + special treatment for truncation operator + 1. there are some (besides this) +words + | no need to search in the index, it can never ADD new rows + | to the result, and to remove half-matched rows we do scan anyway + 2. -trunc* + | same as 1. + 3. in 1 and 2, +/- need not be on the same expr. level, + but can be on any upper level, as in +word +(trunc1* trunc2*) + 4. otherwise + | We have to index-search for this prefix. + | It may cause duplicates, as in the index (sorted by ) + | + | + | + | Searching for "aa*" will find row1 twice... + */ + FTB_EXPR *ftbe; + for (ftbe=(FTB_EXPR*)ftbw; + ftbe->up && !(ftbe->up->flags & FTB_FLAG_TRUNC); + ftbe->up->flags|= FTB_FLAG_TRUNC, ftbe=ftbe->up) + { + if (ftbe->flags & FTB_FLAG_NO || /* 2 */ + ftbe->up->ythresh - ftbe->up->yweaks >1) /* 1 */ + { + FTB_EXPR *top_ftbe=ftbe->up; + ftbw->docid[0]=HA_OFFSET_ERROR; + for (ftbe=(FTB_EXPR *)ftbw; + ftbe != top_ftbe && !(ftbe->flags & FTB_FLAG_NO); + ftbe=ftbe->up) + ftbe->up->yweaks++; + ftbe=0; + break; + } + } + if (!ftbe) + continue; + /* 4 */ + if (!is_tree_inited(& ftb->no_dupes)) + init_tree(& ftb->no_dupes,0,0,sizeof(my_off_t), + _ftb_no_dupes_cmp,0,0,0); + else + reset_tree(& ftb->no_dupes); + } + + ftbw->off=0; /* in case of reinit */ + if (_ft2_search(ftb, ftbw, 1)) + return; + } + queue_fix(& ftb->queue); +} + + +FT_INFO * maria_ft_init_boolean_search(MARIA_HA *info, uint keynr, byte *query, + uint query_len, CHARSET_INFO *cs) +{ + FTB *ftb; + FTB_EXPR *ftbe; + FTB_WORD *ftbw; + + if (!(ftb=(FTB *)my_malloc(sizeof(FTB), MYF(MY_WME)))) + return 0; + ftb->please= (struct _ft_vft *) & _ma_ft_vft_boolean; + ftb->state=UNINITIALIZED; + ftb->info=info; + ftb->keynr=keynr; + ftb->charset=cs; + DBUG_ASSERT(keynr==NO_SUCH_KEY || cs == info->s->keyinfo[keynr].seg->charset); + ftb->with_scan=0; + ftb->lastpos=HA_OFFSET_ERROR; + bzero(& ftb->no_dupes, sizeof(TREE)); + ftb->last_word= 0; + + init_alloc_root(&ftb->mem_root, 1024, 1024); + ftb->queue.max_elements= 0; + if (!(ftbe=(FTB_EXPR *)alloc_root(&ftb->mem_root, sizeof(FTB_EXPR)))) + goto err; + ftbe->weight=1; + ftbe->flags=FTB_FLAG_YES; + ftbe->nos=1; + ftbe->up=0; + ftbe->max_docid= ftbe->ythresh= ftbe->yweaks= 0; + ftbe->docid[0]=ftbe->docid[1]=HA_OFFSET_ERROR; + ftbe->phrase= NULL; + ftbe->document= 0; + ftb->root=ftbe; + _ftb_parse_query(ftb, query, query_len, keynr == NO_SUCH_KEY ? + &ft_default_parser : + info->s->keyinfo[keynr].parser); + /* + Hack: instead of init_queue, we'll use reinit queue to be able + to alloc queue with alloc_root() + */ + if (! (ftb->queue.root= (byte **)alloc_root(&ftb->mem_root, + (ftb->queue.max_elements + 1) * + sizeof(void *)))) + goto err; + reinit_queue(&ftb->queue, ftb->queue.max_elements, 0, 0, + (int (*)(void*, byte*, byte*))FTB_WORD_cmp, 0); + for (ftbw= ftb->last_word; ftbw; ftbw= ftbw->prev) + queue_insert(&ftb->queue, (byte *)ftbw); + ftb->list=(FTB_WORD **)alloc_root(&ftb->mem_root, + sizeof(FTB_WORD *)*ftb->queue.elements); + memcpy(ftb->list, ftb->queue.root+1, sizeof(FTB_WORD *)*ftb->queue.elements); + qsort2(ftb->list, ftb->queue.elements, sizeof(FTB_WORD *), + (qsort2_cmp)FTB_WORD_cmp_list, ftb->charset); + if (ftb->queue.elements<2) ftb->with_scan &= ~FTB_FLAG_TRUNC; + ftb->state=READY; + return ftb; +err: + free_root(& ftb->mem_root, MYF(0)); + my_free((gptr)ftb,MYF(0)); + return 0; +} + + +typedef struct st_my_ftb_phrase_param +{ + LIST *phrase; + LIST *document; + CHARSET_INFO *cs; + uint phrase_length; + uint document_length; + uint match; +} MY_FTB_PHRASE_PARAM; + + +static int ftb_phrase_add_word(void *param, char *word, int word_len, + MYSQL_FTPARSER_BOOLEAN_INFO *boolean_info __attribute__((unused))) +{ + MY_FTB_PHRASE_PARAM *phrase_param= (MY_FTB_PHRASE_PARAM *)param; + FT_WORD *w= (FT_WORD *)phrase_param->document->data; + LIST *phrase, *document; + w->pos= word; + w->len= word_len; + phrase_param->document= phrase_param->document->prev; + if (phrase_param->phrase_length > phrase_param->document_length) + { + phrase_param->document_length++; + return 0; + } + /* TODO: rewrite phrase search to avoid + comparing the same word twice. */ + for (phrase= phrase_param->phrase, document= phrase_param->document->next; + phrase; phrase= phrase->next, document= document->next) + { + FT_WORD *phrase_word= (FT_WORD *)phrase->data; + FT_WORD *document_word= (FT_WORD *)document->data; + if (my_strnncoll(phrase_param->cs, + (uchar*) phrase_word->pos, phrase_word->len, + (uchar*) document_word->pos, document_word->len)) + return 0; + } + phrase_param->match++; + return 0; +} + + +static int ftb_check_phrase_internal(void *param, char *document, int len) +{ + FT_WORD word; + MY_FTB_PHRASE_PARAM *phrase_param= (MY_FTB_PHRASE_PARAM *)param; + const char *docend= document + len; + while (maria_ft_simple_get_word(phrase_param->cs, &document, docend, &word, FALSE)) + { + ftb_phrase_add_word(param, word.pos, word.len, 0); + if (phrase_param->match) + return 1; + } + return 0; +} + + +/* + Checks if given buffer matches phrase list. + + SYNOPSIS + _ftb_check_phrase() + s0 start of buffer + e0 end of buffer + phrase broken into list phrase + cs charset info + + RETURN VALUE + 1 is returned if phrase found, 0 else. +*/ + +static int _ftb_check_phrase(FTB *ftb, const byte *document, uint len, + FTB_EXPR *ftbe, struct st_mysql_ftparser *parser) +{ + MY_FTB_PHRASE_PARAM ftb_param; + MYSQL_FTPARSER_PARAM *param; + DBUG_ENTER("_ftb_check_phrase"); + DBUG_ASSERT(parser); + if (! (param= maria_ftparser_call_initializer(ftb->info, ftb->keynr))) + DBUG_RETURN(0); + ftb_param.phrase= ftbe->phrase; + ftb_param.document= ftbe->document; + ftb_param.cs= ftb->charset; + ftb_param.phrase_length= list_length(ftbe->phrase); + ftb_param.document_length= 1; + ftb_param.match= 0; + + param->mysql_parse= ftb_check_phrase_internal; + param->mysql_add_word= ftb_phrase_add_word; + param->mysql_ftparam= (void *)&ftb_param; + param->cs= ftb->charset; + param->doc= (byte *)document; + param->length= len; + param->mode= MYSQL_FTPARSER_WITH_STOPWORDS; + parser->parse(param); + DBUG_RETURN(ftb_param.match ? 1 : 0); +} + + +static void _ftb_climb_the_tree(FTB *ftb, FTB_WORD *ftbw, FT_SEG_ITERATOR *ftsi_orig) +{ + FT_SEG_ITERATOR ftsi; + FTB_EXPR *ftbe; + float weight=ftbw->weight; + int yn=ftbw->flags, ythresh, mode=(ftsi_orig != 0); + my_off_t curdoc=ftbw->docid[mode]; + struct st_mysql_ftparser *parser= ftb->keynr == NO_SUCH_KEY ? + &ft_default_parser : + ftb->info->s->keyinfo[ftb->keynr].parser; + + for (ftbe=ftbw->up; ftbe; ftbe=ftbe->up) + { + ythresh = ftbe->ythresh - (mode ? 0 : ftbe->yweaks); + if (ftbe->docid[mode] != curdoc) + { + ftbe->cur_weight=0; + ftbe->yesses=ftbe->nos=0; + ftbe->docid[mode]=curdoc; + } + if (ftbe->nos) + break; + if (yn & FTB_FLAG_YES) + { + weight /= ftbe->ythresh; + ftbe->cur_weight += weight; + if ((int) ++ftbe->yesses == ythresh) + { + yn=ftbe->flags; + weight=ftbe->cur_weight*ftbe->weight; + if (mode && ftbe->phrase) + { + int not_found=1; + + memcpy(&ftsi, ftsi_orig, sizeof(ftsi)); + while (_ma_ft_segiterator(&ftsi) && not_found) + { + if (!ftsi.pos) + continue; + not_found = ! _ftb_check_phrase(ftb, ftsi.pos, ftsi.len, + ftbe, parser); + } + if (not_found) break; + } /* ftbe->quot */ + } + else + break; + } + else + if (yn & FTB_FLAG_NO) + { + /* + NOTE: special sort function of queue assures that all + (yn & FTB_FLAG_NO) != 0 + events for every particular subexpression will + "auto-magically" happen BEFORE all the + (yn & FTB_FLAG_YES) != 0 events. So no + already matched expression can become not-matched again. + */ + ++ftbe->nos; + break; + } + else + { + if (ftbe->ythresh) + weight/=3; + ftbe->cur_weight += weight; + if ((int) ftbe->yesses < ythresh) + break; + if (!(yn & FTB_FLAG_WONLY)) + yn= ((int) ftbe->yesses++ == ythresh) ? ftbe->flags : FTB_FLAG_WONLY ; + weight*= ftbe->weight; + } + } +} + + +int maria_ft_boolean_read_next(FT_INFO *ftb, char *record) +{ + FTB_EXPR *ftbe; + FTB_WORD *ftbw; + MARIA_HA *info=ftb->info; + my_off_t curdoc; + + if (ftb->state != INDEX_SEARCH && ftb->state != INDEX_DONE) + return -1; + + /* black magic ON */ + if ((int) _ma_check_index(info, ftb->keynr) < 0) + return my_errno; + if (_ma_readinfo(info, F_RDLCK, 1)) + return my_errno; + /* black magic OFF */ + + if (!ftb->queue.elements) + return my_errno=HA_ERR_END_OF_FILE; + + /* Attention!!! Address of a local variable is used here! See err: label */ + ftb->queue.first_cmp_arg=(void *)&curdoc; + + while (ftb->state == INDEX_SEARCH && + (curdoc=((FTB_WORD *)queue_top(& ftb->queue))->docid[0]) != + HA_OFFSET_ERROR) + { + while (curdoc == (ftbw=(FTB_WORD *)queue_top(& ftb->queue))->docid[0]) + { + _ftb_climb_the_tree(ftb, ftbw, 0); + + /* update queue */ + _ft2_search(ftb, ftbw, 0); + queue_replaced(& ftb->queue); + } + + ftbe=ftb->root; + if (ftbe->docid[0]==curdoc && ftbe->cur_weight>0 && + ftbe->yesses>=(ftbe->ythresh-ftbe->yweaks) && !ftbe->nos) + { + /* curdoc matched ! */ + if (is_tree_inited(&ftb->no_dupes) && + tree_insert(&ftb->no_dupes, &curdoc, 0, + ftb->no_dupes.custom_arg)->count >1) + /* but it managed already to get past this line once */ + continue; + + info->lastpos=curdoc; + /* Clear all states, except that the table was updated */ + info->update&= (HA_STATE_CHANGED | HA_STATE_ROW_CHANGED); + + if (!(*info->read_record)(info,curdoc,record)) + { + info->update|= HA_STATE_AKTIV; /* Record is read */ + if (ftb->with_scan && maria_ft_boolean_find_relevance(ftb,record,0)==0) + continue; /* no match */ + my_errno=0; + goto err; + } + goto err; + } + } + ftb->state=INDEX_DONE; + my_errno=HA_ERR_END_OF_FILE; +err: + ftb->queue.first_cmp_arg=(void *)0; + return my_errno; +} + + +typedef struct st_my_ftb_find_param +{ + FT_INFO *ftb; + FT_SEG_ITERATOR *ftsi; +} MY_FTB_FIND_PARAM; + + +static int ftb_find_relevance_add_word(void *param, char *word, int len, + MYSQL_FTPARSER_BOOLEAN_INFO *boolean_info __attribute__((unused))) +{ + MY_FTB_FIND_PARAM *ftb_param= (MY_FTB_FIND_PARAM *)param; + FT_INFO *ftb= ftb_param->ftb; + FTB_WORD *ftbw; + int a, b, c; + for (a= 0, b= ftb->queue.elements, c= (a+b)/2; b-a>1; c= (a+b)/2) + { + ftbw= ftb->list[c]; + if (ha_compare_text(ftb->charset, (uchar*)word, len, + (uchar*)ftbw->word+1, ftbw->len-1, + (my_bool)(ftbw->flags&FTB_FLAG_TRUNC), 0) > 0) + b= c; + else + a= c; + } + for (; c >= 0; c--) + { + ftbw= ftb->list[c]; + if (ha_compare_text(ftb->charset, (uchar*)word, len, + (uchar*)ftbw->word + 1,ftbw->len - 1, + (my_bool)(ftbw->flags & FTB_FLAG_TRUNC), 0)) + break; + if (ftbw->docid[1] == ftb->info->lastpos) + continue; + ftbw->docid[1]= ftb->info->lastpos; + _ftb_climb_the_tree(ftb, ftbw, ftb_param->ftsi); + } + return(0); +} + + +static int ftb_find_relevance_parse(void *param, char *doc, int len) +{ + FT_INFO *ftb= ((MY_FTB_FIND_PARAM *)param)->ftb; + char *end= doc + len; + FT_WORD w; + while (maria_ft_simple_get_word(ftb->charset, &doc, end, &w, TRUE)) + ftb_find_relevance_add_word(param, w.pos, w.len, 0); + return(0); +} + + +float maria_ft_boolean_find_relevance(FT_INFO *ftb, byte *record, uint length) +{ + FTB_EXPR *ftbe; + FT_SEG_ITERATOR ftsi, ftsi2; + my_off_t docid=ftb->info->lastpos; + MY_FTB_FIND_PARAM ftb_param; + MYSQL_FTPARSER_PARAM *param; + struct st_mysql_ftparser *parser= ftb->keynr == NO_SUCH_KEY ? + &ft_default_parser : + ftb->info->s->keyinfo[ftb->keynr].parser; + + if (docid == HA_OFFSET_ERROR) + return -2.0; + if (!ftb->queue.elements) + return 0; + if (! (param= maria_ftparser_call_initializer(ftb->info, ftb->keynr))) + return 0; + + if (ftb->state != INDEX_SEARCH && docid <= ftb->lastpos) + { + FTB_EXPR *x; + uint i; + + for (i=0; i < ftb->queue.elements; i++) + { + ftb->list[i]->docid[1]=HA_OFFSET_ERROR; + for (x=ftb->list[i]->up; x; x=x->up) + x->docid[1]=HA_OFFSET_ERROR; + } + } + + ftb->lastpos=docid; + + if (ftb->keynr==NO_SUCH_KEY) + _ma_ft_segiterator_dummy_init(record, length, &ftsi); + else + _ma_ft_segiterator_init(ftb->info, ftb->keynr, record, &ftsi); + memcpy(&ftsi2, &ftsi, sizeof(ftsi)); + + ftb_param.ftb= ftb; + ftb_param.ftsi= &ftsi2; + while (_ma_ft_segiterator(&ftsi)) + { + if (!ftsi.pos) + continue; + /* Since subsequent call to _ftb_check_phrase overwrites param elements, + it must be reinitialized at each iteration _inside_ the loop. */ + param->mysql_parse= ftb_find_relevance_parse; + param->mysql_add_word= ftb_find_relevance_add_word; + param->mysql_ftparam= (void *)&ftb_param; + param->cs= ftb->charset; + param->mode= MYSQL_FTPARSER_SIMPLE_MODE; + param->doc= (byte *)ftsi.pos; + param->length= ftsi.len; + parser->parse(param); + } + ftbe=ftb->root; + if (ftbe->docid[1]==docid && ftbe->cur_weight>0 && + ftbe->yesses>=ftbe->ythresh && !ftbe->nos) + { /* row matched ! */ + return ftbe->cur_weight; + } + else + { /* match failed ! */ + return 0.0; + } +} + + +void maria_ft_boolean_close_search(FT_INFO *ftb) +{ + if (is_tree_inited(& ftb->no_dupes)) + { + delete_tree(& ftb->no_dupes); + } + free_root(& ftb->mem_root, MYF(0)); + my_free((gptr)ftb,MYF(0)); +} + + +float maria_ft_boolean_get_relevance(FT_INFO *ftb) +{ + return ftb->root->cur_weight; +} + + +void maria_ft_boolean_reinit_search(FT_INFO *ftb) +{ + _ftb_init_index_search(ftb); +} diff --git a/storage/maria/ma_ft_eval.c b/storage/maria/ma_ft_eval.c new file mode 100644 index 00000000000..b9b496fc268 --- /dev/null +++ b/storage/maria/ma_ft_eval.c @@ -0,0 +1,254 @@ +/* Copyright (C) 2006 MySQL AB & MySQL Finland AB & TCX DataKonsult AB + 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; either version 2 of the License, or + (at your option) any later version. + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ + +/* Written by Sergei A. Golubchik, who has a shared copyright to this code + added support for long options (my_getopt) 22.5.2002 by Jani Tolonen */ + +#include "ma_ftdefs.h" +#include "maria_ft_eval.h" +#include +#include + +static void print_error(int exit_code, const char *fmt,...); +static void get_options(int argc, char *argv[]); +static int create_record(char *pos, FILE *file); +static void usage(); + +static struct my_option my_long_options[] = +{ + {"", 's', "", 0, 0, 0, GET_STR, REQUIRED_ARG, 0, 0, 0, 0, 0, 0}, + {"", 'q', "", 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, + {"", 'S', "", 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, + {"", '#', "", 0, 0, 0, GET_STR, REQUIRED_ARG, 0, 0, 0, 0, 0, 0}, + {"", 'V', "", 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, + {"", '?', "", 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, + {"", 'h', "", 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, + { 0, 0, 0, 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0} +}; + +int main(int argc, char *argv[]) +{ + MARIA_HA *file; + int i,j; + + MY_INIT(argv[0]); + get_options(argc,argv); + bzero((char*)recinfo,sizeof(recinfo)); + + maria_init(); + /* First define 2 columns */ + recinfo[0].type=FIELD_SKIP_ENDSPACE; + recinfo[0].length=docid_length; + recinfo[1].type=FIELD_BLOB; + recinfo[1].length= 4+maria_portable_sizeof_char_ptr; + + /* Define a key over the first column */ + keyinfo[0].seg=keyseg; + keyinfo[0].keysegs=1; + keyinfo[0].seg[0].type= HA_KEYTYPE_TEXT; + keyinfo[0].seg[0].flag= HA_BLOB_PART; + keyinfo[0].seg[0].start=recinfo[0].length; + keyinfo[0].seg[0].length=key_length; + keyinfo[0].seg[0].null_bit=0; + keyinfo[0].seg[0].null_pos=0; + keyinfo[0].seg[0].bit_start=4; + keyinfo[0].seg[0].language=MY_CHARSET_CURRENT; + keyinfo[0].flag = HA_FULLTEXT; + + if (!silent) + printf("- Creating isam-file\n"); + if (maria_create(filename,1,keyinfo,2,recinfo,0,NULL,(MARIA_CREATE_INFO*) 0,0)) + goto err; + if (!(file=maria_open(filename,2,0))) + goto err; + if (!silent) + printf("Initializing stopwords\n"); + maria_ft_init_stopwords(stopwordlist); + + if (!silent) + printf("- Writing key:s\n"); + + my_errno=0; + i=0; + while (create_record(record,df)) + { + error=maria_write(file,record); + if (error) + printf("I= %2d maria_write: %d errno: %d\n",i,error,my_errno); + i++; + } + fclose(df); + + if (maria_close(file)) goto err; + if (!silent) + printf("- Reopening file\n"); + if (!(file=maria_open(filename,2,0))) goto err; + if (!silent) + printf("- Reading rows with key\n"); + for (i=1;create_record(record,qf);i++) + { + FT_DOCLIST *result; + double w; + int t, err; + + result=maria_ft_nlq_init_search(file,0,blob_record,(uint) strlen(blob_record),1); + if (!result) + { + printf("Query %d failed with errno %3d\n",i,my_errno); + goto err; + } + if (!silent) + printf("Query %d. Found: %d.\n",i,result->ndocs); + for (j=0;(err=maria_ft_nlq_read_next(result, read_record))==0;j++) + { + t=uint2korr(read_record); + w=maria_ft_nlq_get_relevance(result); + printf("%d %.*s %f\n",i,t,read_record+2,w); + } + if (err != HA_ERR_END_OF_FILE) + { + printf("maria_ft_read_next %d failed with errno %3d\n",j,my_errno); + goto err; + } + maria_ft_nlq_close_search(result); + } + + if (maria_close(file)) goto err; + maria_end(); + my_end(MY_CHECK_ERROR); + + return (0); + + err: + printf("got error: %3d when using maria-database\n",my_errno); + return 1; /* skip warning */ + +} + + +static my_bool +get_one_option(int optid, const struct my_option *opt __attribute__((unused)), + char *argument) +{ + switch (optid) { + case 's': + if (stopwordlist && stopwordlist != maria_ft_precompiled_stopwords) + break; + { + FILE *f; char s[HA_FT_MAXLEN]; int i=0,n=SWL_INIT; + + if (!(stopwordlist=(const char**) malloc(n*sizeof(char *)))) + print_error(1,"malloc(%d)",n*sizeof(char *)); + if (!(f=fopen(argument,"r"))) + print_error(1,"fopen(%s)",argument); + while (!feof(f)) + { + if (!(fgets(s,HA_FT_MAXLEN,f))) + print_error(1,"fgets(s,%d,%s)",HA_FT_MAXLEN,argument); + if (!(stopwordlist[i++]=strdup(s))) + print_error(1,"strdup(%s)",s); + if (i >= n) + { + n+=SWL_PLUS; + if (!(stopwordlist=(const char**) realloc((char*) stopwordlist, + n*sizeof(char *)))) + print_error(1,"realloc(%d)",n*sizeof(char *)); + } + } + fclose(f); + stopwordlist[i]=NULL; + break; + } + case 'q': silent=1; break; + case 'S': if (stopwordlist==maria_ft_precompiled_stopwords) stopwordlist=NULL; break; + case '#': + DBUG_PUSH (argument); + break; + case 'V': + case '?': + case 'h': + usage(); + exit(1); + } + return 0; +} + + +static void get_options(int argc, char *argv[]) +{ + int ho_error; + + if ((ho_error=handle_options(&argc, &argv, my_long_options, get_one_option))) + exit(ho_error); + + if (!(d_file=argv[optind])) print_error(1,"No d_file"); + if (!(df=fopen(d_file,"r"))) + print_error(1,"fopen(%s)",d_file); + if (!(q_file=argv[optind+1])) print_error(1,"No q_file"); + if (!(qf=fopen(q_file,"r"))) + print_error(1,"fopen(%s)",q_file); + return; +} /* get options */ + + +static int create_record(char *pos, FILE *file) +{ + uint tmp; char *ptr; + + bzero((char *)pos,MAX_REC_LENGTH); + + /* column 1 - VARCHAR */ + if (!(fgets(pos+2,MAX_REC_LENGTH-32,file))) + { + if (feof(file)) + return 0; + else + print_error(1,"fgets(docid) - 1"); + } + tmp=(uint) strlen(pos+2)-1; + int2store(pos,tmp); + pos+=recinfo[0].length; + + /* column 2 - BLOB */ + + if (!(fgets(blob_record,MAX_BLOB_LENGTH,file))) + print_error(1,"fgets(docid) - 2"); + tmp=(uint) strlen(blob_record); + int4store(pos,tmp); + ptr=blob_record; + memcpy_fixed(pos+4,&ptr,sizeof(char*)); + return 1; +} + +/* VARARGS */ + +static void print_error(int exit_code, const char *fmt,...) +{ + va_list args; + + va_start(args,fmt); + fprintf(stderr,"%s: error: ",my_progname); + VOID(vfprintf(stderr, fmt, args)); + VOID(fputc('\n',stderr)); + fflush(stderr); + va_end(args); + exit(exit_code); +} + + +static void usage() +{ + printf("%s [options]\n", my_progname); + my_print_help(my_long_options); + my_print_variables(my_long_options); +} diff --git a/storage/maria/ma_ft_eval.h b/storage/maria/ma_ft_eval.h new file mode 100644 index 00000000000..d9b5c51642c --- /dev/null +++ b/storage/maria/ma_ft_eval.h @@ -0,0 +1,42 @@ +/* Copyright (C) 2006 MySQL AB & Sergei A. Golubchik + + 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; either version 2 of the License, or + (at your option) any later version. + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ + +/* Written by Sergei A. Golubchik, who has a shared copyright to this code */ + +const char **stopwordlist=maria_ft_precompiled_stopwords; + +#define MAX_REC_LENGTH 128 +#define MAX_BLOB_LENGTH 60000 +char record[MAX_REC_LENGTH], read_record[MAX_REC_LENGTH+MAX_BLOB_LENGTH]; +char blob_record[MAX_BLOB_LENGTH+20*20]; + +char *filename= (char*) "EVAL"; + +int silent=0, error=0; + +uint key_length=MAX_BLOB_LENGTH,docid_length=32; +char *d_file, *q_file; +FILE *df,*qf; + +MARIA_COLUMNDEF recinfo[3]; +MARIA_KEYDEF keyinfo[2]; +HA_KEYSEG keyseg[10]; + +#define SWL_INIT 500 +#define SWL_PLUS 50 + +#define MAX_LINE_LENGTH 128 +char line[MAX_LINE_LENGTH]; diff --git a/storage/maria/ma_ft_nlq_search.c b/storage/maria/ma_ft_nlq_search.c new file mode 100644 index 00000000000..a9741787fc9 --- /dev/null +++ b/storage/maria/ma_ft_nlq_search.c @@ -0,0 +1,366 @@ +/* Copyright (C) 2006 MySQL AB & MySQL Finland AB & TCX DataKonsult AB + + 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; either version 2 of the License, or + (at your option) any later version. + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ + +/* Written by Sergei A. Golubchik, who has a shared copyright to this code */ + +#define FT_CORE +#include "ma_ftdefs.h" + +/* search with natural language queries */ + +typedef struct ft_doc_rec +{ + my_off_t dpos; + double weight; +} FT_DOC; + +struct st_ft_info +{ + struct _ft_vft *please; + MARIA_HA *info; + int ndocs; + int curdoc; + FT_DOC doc[1]; +}; + +typedef struct st_all_in_one +{ + MARIA_HA *info; + uint keynr; + CHARSET_INFO *charset; + uchar *keybuff; + TREE dtree; +} ALL_IN_ONE; + +typedef struct st_ft_superdoc +{ + FT_DOC doc; + FT_WORD *word_ptr; + double tmp_weight; +} FT_SUPERDOC; + +static int FT_SUPERDOC_cmp(void* cmp_arg __attribute__((unused)), + FT_SUPERDOC *p1, FT_SUPERDOC *p2) +{ + if (p1->doc.dpos < p2->doc.dpos) + return -1; + if (p1->doc.dpos == p2->doc.dpos) + return 0; + return 1; +} + +static int walk_and_match(FT_WORD *word, uint32 count, ALL_IN_ONE *aio) +{ + int subkeys, r; + uint keylen, doc_cnt; + FT_SUPERDOC sdoc, *sptr; + TREE_ELEMENT *selem; + double gweight=1; + MARIA_HA *info=aio->info; + uchar *keybuff=aio->keybuff; + MARIA_KEYDEF *keyinfo=info->s->keyinfo+aio->keynr; + my_off_t key_root=info->s->state.key_root[aio->keynr]; + uint extra=HA_FT_WLEN+info->s->base.rec_reflength; +#if HA_FT_WTYPE == HA_KEYTYPE_FLOAT + float tmp_weight; +#else +#error +#endif + + DBUG_ENTER("walk_and_match"); + + word->weight=LWS_FOR_QUERY; + + keylen= _ma_ft_make_key(info,aio->keynr,(char*) keybuff,word,0); + keylen-=HA_FT_WLEN; + doc_cnt=0; + + /* Skip rows inserted by current inserted */ + for (r= _ma_search(info, keyinfo, keybuff, keylen, SEARCH_FIND, key_root) ; + !r && + (subkeys=ft_sintXkorr(info->lastkey+info->lastkey_length-extra)) > 0 && + info->lastpos >= info->state->data_file_length ; + r= _ma_search_next(info, keyinfo, info->lastkey, + info->lastkey_length, SEARCH_BIGGER, key_root)) + ; + + info->update|= HA_STATE_AKTIV; /* for _ma_test_if_changed() */ + + /* The following should be safe, even if we compare doubles */ + while (!r && gweight) + { + + if (keylen && + ha_compare_text(aio->charset,info->lastkey+1, + info->lastkey_length-extra-1, keybuff+1,keylen-1,0,0)) + break; + + if (subkeys<0) + { + if (doc_cnt) + DBUG_RETURN(1); /* index is corrupted */ + /* + TODO here: unsafe optimization, should this word + be skipped (based on subkeys) ? + */ + keybuff+=keylen; + keyinfo=& info->s->ft2_keyinfo; + key_root=info->lastpos; + keylen=0; + r= _ma_search_first(info, keyinfo, key_root); + goto do_skip; + } +#if HA_FT_WTYPE == HA_KEYTYPE_FLOAT + tmp_weight=*(float*)&subkeys; +#else +#error +#endif + /* The following should be safe, even if we compare doubles */ + if (tmp_weight==0) + DBUG_RETURN(doc_cnt); /* stopword, doc_cnt should be 0 */ + + sdoc.doc.dpos=info->lastpos; + + /* saving document matched into dtree */ + if (!(selem=tree_insert(&aio->dtree, &sdoc, 0, aio->dtree.custom_arg))) + DBUG_RETURN(1); + + sptr=(FT_SUPERDOC *)ELEMENT_KEY((&aio->dtree), selem); + + if (selem->count==1) /* document's first match */ + sptr->doc.weight=0; + else + sptr->doc.weight+=sptr->tmp_weight*sptr->word_ptr->weight; + + sptr->word_ptr=word; + sptr->tmp_weight=tmp_weight; + + doc_cnt++; + + gweight=word->weight*GWS_IN_USE; + if (gweight < 0 || doc_cnt > 2000000) + gweight=0; + + if (_ma_test_if_changed(info) == 0) + r= _ma_search_next(info, keyinfo, info->lastkey, info->lastkey_length, + SEARCH_BIGGER, key_root); + else + r= _ma_search(info, keyinfo, info->lastkey, info->lastkey_length, + SEARCH_BIGGER, key_root); +do_skip: + while ((subkeys=ft_sintXkorr(info->lastkey+info->lastkey_length-extra)) > 0 && + !r && info->lastpos >= info->state->data_file_length) + r= _ma_search_next(info, keyinfo, info->lastkey, info->lastkey_length, + SEARCH_BIGGER, key_root); + + } + word->weight=gweight; + + DBUG_RETURN(0); +} + + +static int walk_and_copy(FT_SUPERDOC *from, + uint32 count __attribute__((unused)), FT_DOC **to) +{ + DBUG_ENTER("walk_and_copy"); + from->doc.weight+=from->tmp_weight*from->word_ptr->weight; + (*to)->dpos=from->doc.dpos; + (*to)->weight=from->doc.weight; + (*to)++; + DBUG_RETURN(0); +} + +static int walk_and_push(FT_SUPERDOC *from, + uint32 count __attribute__((unused)), QUEUE *best) +{ + DBUG_ENTER("walk_and_copy"); + from->doc.weight+=from->tmp_weight*from->word_ptr->weight; + set_if_smaller(best->elements, ft_query_expansion_limit-1); + queue_insert(best, (byte *)& from->doc); + DBUG_RETURN(0); +} + + +static int FT_DOC_cmp(void *unused __attribute__((unused)), + FT_DOC *a, FT_DOC *b) +{ + return sgn(b->weight - a->weight); +} + + +FT_INFO *maria_ft_init_nlq_search(MARIA_HA *info, uint keynr, byte *query, + uint query_len, uint flags, byte *record) +{ + TREE wtree; + ALL_IN_ONE aio; + FT_DOC *dptr; + FT_INFO *dlist=NULL; + my_off_t saved_lastpos=info->lastpos; + struct st_mysql_ftparser *parser; + MYSQL_FTPARSER_PARAM *ftparser_param; + DBUG_ENTER("maria_ft_init_nlq_search"); + +/* black magic ON */ + if ((int) (keynr = _ma_check_index(info,keynr)) < 0) + DBUG_RETURN(NULL); + if (_ma_readinfo(info,F_RDLCK,1)) + DBUG_RETURN(NULL); +/* black magic OFF */ + + aio.info=info; + aio.keynr=keynr; + aio.charset=info->s->keyinfo[keynr].seg->charset; + aio.keybuff=info->lastkey+info->s->base.max_key_length; + parser= info->s->keyinfo[keynr].parser; + if (! (ftparser_param= maria_ftparser_call_initializer(info, keynr))) + goto err; + + bzero(&wtree,sizeof(wtree)); + + init_tree(&aio.dtree,0,0,sizeof(FT_SUPERDOC),(qsort_cmp2)&FT_SUPERDOC_cmp,0, + NULL, NULL); + + maria_ft_parse_init(&wtree, aio.charset); + if (maria_ft_parse(&wtree, query, query_len, 0, parser, ftparser_param)) + goto err; + + if (tree_walk(&wtree, (tree_walk_action)&walk_and_match, &aio, + left_root_right)) + goto err; + + if (flags & FT_EXPAND && ft_query_expansion_limit) + { + QUEUE best; + init_queue(&best,ft_query_expansion_limit,0,0, (queue_compare) &FT_DOC_cmp, + 0); + tree_walk(&aio.dtree, (tree_walk_action) &walk_and_push, + &best, left_root_right); + while (best.elements) + { + my_off_t docid=((FT_DOC *)queue_remove(& best, 0))->dpos; + if (!(*info->read_record)(info,docid,record)) + { + info->update|= HA_STATE_AKTIV; + _ma_ft_parse(&wtree, info, keynr, record, 1, ftparser_param); + } + } + delete_queue(&best); + reset_tree(&aio.dtree); + if (tree_walk(&wtree, (tree_walk_action)&walk_and_match, &aio, + left_root_right)) + goto err; + + } + + /* + If ndocs == 0, this will not allocate RAM for FT_INFO.doc[], + so if ndocs == 0, FT_INFO.doc[] must not be accessed. + */ + dlist=(FT_INFO *)my_malloc(sizeof(FT_INFO)+ + sizeof(FT_DOC)* + (int)(aio.dtree.elements_in_tree-1), + MYF(0)); + if (!dlist) + goto err; + + dlist->please= (struct _ft_vft *) & _ma_ft_vft_nlq; + dlist->ndocs=aio.dtree.elements_in_tree; + dlist->curdoc=-1; + dlist->info=aio.info; + dptr=dlist->doc; + + tree_walk(&aio.dtree, (tree_walk_action) &walk_and_copy, + &dptr, left_root_right); + + if (flags & FT_SORTED) + qsort2(dlist->doc, dlist->ndocs, sizeof(FT_DOC), (qsort2_cmp)&FT_DOC_cmp, 0); + +err: + delete_tree(&aio.dtree); + delete_tree(&wtree); + info->lastpos=saved_lastpos; + DBUG_RETURN(dlist); +} + + +int maria_ft_nlq_read_next(FT_INFO *handler, char *record) +{ + MARIA_HA *info= (MARIA_HA *) handler->info; + + if (++handler->curdoc >= handler->ndocs) + { + --handler->curdoc; + return HA_ERR_END_OF_FILE; + } + + info->update&= (HA_STATE_CHANGED | HA_STATE_ROW_CHANGED); + + info->lastpos=handler->doc[handler->curdoc].dpos; + if (!(*info->read_record)(info,info->lastpos,record)) + { + info->update|= HA_STATE_AKTIV; /* Record is read */ + return 0; + } + return my_errno; +} + + +float maria_ft_nlq_find_relevance(FT_INFO *handler, + byte *record __attribute__((unused)), + uint length __attribute__((unused))) +{ + int a,b,c; + FT_DOC *docs=handler->doc; + my_off_t docid=handler->info->lastpos; + + if (docid == HA_POS_ERROR) + return -5.0; + + /* Assuming docs[] is sorted by dpos... */ + + for (a=0, b=handler->ndocs, c=(a+b)/2; b-a>1; c=(a+b)/2) + { + if (docs[c].dpos > docid) + b=c; + else + a=c; + } + /* bounds check to avoid accessing unallocated handler->doc */ + if (a < handler->ndocs && docs[a].dpos == docid) + return (float) docs[a].weight; + else + return 0.0; +} + + +void maria_ft_nlq_close_search(FT_INFO *handler) +{ + my_free((gptr)handler,MYF(0)); +} + + +float maria_ft_nlq_get_relevance(FT_INFO *handler) +{ + return (float) handler->doc[handler->curdoc].weight; +} + + +void maria_ft_nlq_reinit_search(FT_INFO *handler) +{ + handler->curdoc=-1; +} + diff --git a/storage/maria/ma_ft_parser.c b/storage/maria/ma_ft_parser.c new file mode 100644 index 00000000000..983bebf3562 --- /dev/null +++ b/storage/maria/ma_ft_parser.c @@ -0,0 +1,394 @@ +/* Copyright (C) 2006 MySQL AB & MySQL Finland AB & TCX DataKonsult AB + + 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; either version 2 of the License, or + (at your option) any later version. + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ + +/* Written by Sergei A. Golubchik, who has a shared copyright to this code */ + +#include "ma_ftdefs.h" + +typedef struct st_maria_ft_docstat { + FT_WORD *list; + uint uniq; + double sum; +} FT_DOCSTAT; + + +typedef struct st_my_maria_ft_parser_param +{ + TREE *wtree; + my_bool with_alloc; +} MY_FT_PARSER_PARAM; + + +static int FT_WORD_cmp(CHARSET_INFO* cs, FT_WORD *w1, FT_WORD *w2) +{ + return ha_compare_text(cs, (uchar*) w1->pos, w1->len, + (uchar*) w2->pos, w2->len, 0, 0); +} + +static int walk_and_copy(FT_WORD *word,uint32 count,FT_DOCSTAT *docstat) +{ + word->weight=LWS_IN_USE; + docstat->sum+=word->weight; + memcpy_fixed((docstat->list)++,word,sizeof(FT_WORD)); + return 0; +} + +/* transforms tree of words into the array, applying normalization */ + +FT_WORD * maria_ft_linearize(TREE *wtree) +{ + FT_WORD *wlist,*p; + FT_DOCSTAT docstat; + DBUG_ENTER("maria_ft_linearize"); + + if ((wlist=(FT_WORD *) my_malloc(sizeof(FT_WORD)* + (1+wtree->elements_in_tree),MYF(0)))) + { + docstat.list=wlist; + docstat.uniq=wtree->elements_in_tree; + docstat.sum=0; + tree_walk(wtree,(tree_walk_action)&walk_and_copy,&docstat,left_root_right); + } + delete_tree(wtree); + if (!wlist) + DBUG_RETURN(NULL); + + docstat.list->pos=NULL; + + for (p=wlist;p->pos;p++) + { + p->weight=PRENORM_IN_USE; + } + + for (p=wlist;p->pos;p++) + { + p->weight/=NORM_IN_USE; + } + + DBUG_RETURN(wlist); +} + +my_bool maria_ft_boolean_check_syntax_string(const byte *str) +{ + uint i, j; + + if (!str || + (strlen(str)+1 != sizeof(ft_boolean_syntax)) || + (str[0] != ' ' && str[1] != ' ')) + return 1; + for (i=0; i 127 || + my_isalnum(default_charset_info, str[i])) + return 1; + for (j=0; jyesno=(FTB_YES==' ') ? 1 : (param->quot != 0); + param->weight_adjust= param->wasign= 0; + param->type= FT_TOKEN_EOF; + + while (docquot) + { + param->quot=doc; + *start=doc+1; + param->type= FT_TOKEN_RIGHT_PAREN; + goto ret; + } + if (!param->quot) + { + if (*doc == FTB_LBR || *doc == FTB_RBR || *doc == FTB_LQUOT) + { + /* param->prev=' '; */ + *start=doc+1; + if (*doc == FTB_LQUOT) param->quot=*start; + param->type= (*doc == FTB_RBR ? FT_TOKEN_RIGHT_PAREN : FT_TOKEN_LEFT_PAREN); + goto ret; + } + if (param->prev == ' ') + { + if (*doc == FTB_YES ) { param->yesno=+1; continue; } else + if (*doc == FTB_EGAL) { param->yesno= 0; continue; } else + if (*doc == FTB_NO ) { param->yesno=-1; continue; } else + if (*doc == FTB_INC ) { param->weight_adjust++; continue; } else + if (*doc == FTB_DEC ) { param->weight_adjust--; continue; } else + if (*doc == FTB_NEG ) { param->wasign= !param->wasign; continue; } + } + } + param->prev=*doc; + param->yesno=(FTB_YES==' ') ? 1 : (param->quot != 0); + param->weight_adjust= param->wasign= 0; + } + + mwc=length=0; + for (word->pos=doc; docprev='A'; /* be sure *prev is true_word_char */ + word->len= (uint)(doc-word->pos) - mwc; + if ((param->trunc=(doc= ft_min_word_len && !is_stopword(word->pos, word->len)) + || param->trunc) && length < ft_max_word_len) + { + *start=doc; + param->type= FT_TOKEN_WORD; + goto ret; + } + else if (length) /* make sure length > 0 (if start contains spaces only) */ + { + *start= doc; + param->type= FT_TOKEN_STOPWORD; + goto ret; + } + } + if (param->quot) + { + param->quot=*start=doc; + param->type= 3; /* FT_RBR */ + goto ret; + } +ret: + return param->type; +} + +byte maria_ft_simple_get_word(CHARSET_INFO *cs, byte **start, const byte *end, + FT_WORD *word, my_bool skip_stopwords) +{ + byte *doc= *start; + uint mwc, length, mbl; + DBUG_ENTER("maria_ft_simple_get_word"); + + do + { + for (;; doc++) + { + if (doc >= end) DBUG_RETURN(0); + if (true_word_char(cs, *doc)) break; + } + + mwc= length= 0; + for (word->pos=doc; doclen= (uint)(doc-word->pos) - mwc; + + if (skip_stopwords == FALSE || + (length >= ft_min_word_len && length < ft_max_word_len && + !is_stopword(word->pos, word->len))) + { + *start= doc; + DBUG_RETURN(1); + } + } while (doc < end); + DBUG_RETURN(0); +} + +void maria_ft_parse_init(TREE *wtree, CHARSET_INFO *cs) +{ + DBUG_ENTER("maria_ft_parse_init"); + if (!is_tree_inited(wtree)) + init_tree(wtree,0,0,sizeof(FT_WORD),(qsort_cmp2)&FT_WORD_cmp,0,NULL, cs); + DBUG_VOID_RETURN; +} + + +static int maria_ft_add_word(void *param, byte *word, uint word_len, + MYSQL_FTPARSER_BOOLEAN_INFO *boolean_info __attribute__((unused))) +{ + TREE *wtree; + FT_WORD w; + DBUG_ENTER("maria_ft_add_word"); + wtree= ((MY_FT_PARSER_PARAM *)param)->wtree; + if (((MY_FT_PARSER_PARAM *)param)->with_alloc) + { + byte *ptr; + /* allocating the data in the tree - to avoid mallocs and frees */ + DBUG_ASSERT(wtree->with_delete == 0); + ptr= (byte *)alloc_root(&wtree->mem_root, word_len); + memcpy(ptr, word, word_len); + w.pos= ptr; + } + else + w.pos= word; + w.len= word_len; + if (!tree_insert(wtree, &w, 0, wtree->custom_arg)) + { + delete_tree(wtree); + DBUG_RETURN(1); + } + DBUG_RETURN(0); +} + + +static int maria_ft_parse_internal(void *param, byte *doc, uint doc_len) +{ + byte *end=doc+doc_len; + FT_WORD w; + TREE *wtree; + DBUG_ENTER("maria_ft_parse_internal"); + + wtree= ((MY_FT_PARSER_PARAM *)param)->wtree; + while (maria_ft_simple_get_word(wtree->custom_arg, &doc, end, &w, TRUE)) + if (maria_ft_add_word(param, w.pos, w.len, 0)) + DBUG_RETURN(1); + DBUG_RETURN(0); +} + + +int maria_ft_parse(TREE *wtree, byte *doc, int doclen, my_bool with_alloc, + struct st_mysql_ftparser *parser, + MYSQL_FTPARSER_PARAM *param) +{ + MY_FT_PARSER_PARAM my_param; + DBUG_ENTER("maria_ft_parse"); + DBUG_ASSERT(parser); + my_param.wtree= wtree; + my_param.with_alloc= with_alloc; + + param->mysql_parse= maria_ft_parse_internal; + param->mysql_add_word= maria_ft_add_word; + param->mysql_ftparam= &my_param; + param->cs= wtree->custom_arg; + param->doc= doc; + param->length= doclen; + param->mode= MYSQL_FTPARSER_SIMPLE_MODE; + DBUG_RETURN(parser->parse(param)); +} + + +MYSQL_FTPARSER_PARAM *maria_ftparser_call_initializer(MARIA_HA *info, uint keynr) +{ + uint32 ftparser_nr; + struct st_mysql_ftparser *parser; + if (! info->ftparser_param) + { + /* info->ftparser_param can not be zero after the initialization, + because it always includes built-in fulltext parser. And built-in + parser can be called even if the table has no fulltext indexes and + no varchar/text fields. */ + if (! info->s->ftparsers) + { + /* It's ok that modification to shared structure is done w/o mutex + locks, because all threads would set the same variables to the + same values. */ + uint i, j, keys= info->s->state.header.keys, ftparsers= 1; + for (i= 0; i < keys; i++) + { + MARIA_KEYDEF *keyinfo= &info->s->keyinfo[i]; + if (keyinfo->flag & HA_FULLTEXT) + { + for (j= 0;; j++) + { + if (j == i) + { + keyinfo->ftparser_nr= ftparsers++; + break; + } + if (info->s->keyinfo[j].flag & HA_FULLTEXT && + keyinfo->parser == info->s->keyinfo[j].parser) + { + keyinfo->ftparser_nr= info->s->keyinfo[j].ftparser_nr; + break; + } + } + } + } + info->s->ftparsers= ftparsers; + } + info->ftparser_param= (MYSQL_FTPARSER_PARAM *) + my_malloc(sizeof(MYSQL_FTPARSER_PARAM) * + info->s->ftparsers, MYF(MY_WME|MY_ZEROFILL)); + if (! info->ftparser_param) + return 0; + } + if (keynr == NO_SUCH_KEY) + { + ftparser_nr= 0; + parser= &ft_default_parser; + } + else + { + ftparser_nr= info->s->keyinfo[keynr].ftparser_nr; + parser= info->s->keyinfo[keynr].parser; + } + if (! info->ftparser_param[ftparser_nr].mysql_add_word) + { + /* Note, that mysql_add_word is used here as a flag: + mysql_add_word == 0 - parser is not initialized + mysql_add_word != 0 - parser is initialized, or no + initialization needed. */ + info->ftparser_param[ftparser_nr].mysql_add_word= (void *)1; + if (parser->init && parser->init(&info->ftparser_param[ftparser_nr])) + return 0; + } + return &info->ftparser_param[ftparser_nr]; +} + + +void maria_ftparser_call_deinitializer(MARIA_HA *info) +{ + uint i, keys= info->s->state.header.keys; + if (! info->ftparser_param) + return; + for (i= 0; i < keys; i++) + { + MARIA_KEYDEF *keyinfo= &info->s->keyinfo[i]; + MYSQL_FTPARSER_PARAM *ftparser_param= + &info->ftparser_param[keyinfo->ftparser_nr]; + if (keyinfo->flag & HA_FULLTEXT && ftparser_param->mysql_add_word) + { + if (keyinfo->parser->deinit) + keyinfo->parser->deinit(ftparser_param); + ftparser_param->mysql_add_word= 0; + } + } +} diff --git a/storage/maria/ma_ft_stem.c b/storage/maria/ma_ft_stem.c new file mode 100644 index 00000000000..7a2f8cfd7c5 --- /dev/null +++ b/storage/maria/ma_ft_stem.c @@ -0,0 +1,19 @@ +/* Copyright (C) 2006 MySQL AB & MySQL Finland AB & TCX DataKonsult AB + + 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; either version 2 of the License, or + (at your option) any later version. + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ + +/* Written by Sergei A. Golubchik, who has a shared copyright to this code */ + +/* mulitingual stem */ diff --git a/storage/maria/ma_ft_test1.c b/storage/maria/ma_ft_test1.c new file mode 100644 index 00000000000..2880f6bcdc1 --- /dev/null +++ b/storage/maria/ma_ft_test1.c @@ -0,0 +1,317 @@ +/* Copyright (C) 2006 MySQL AB & MySQL Finland AB & TCX DataKonsult AB + + 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; either version 2 of the License, or + (at your option) any later version. + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ + +/* Written by Sergei A. Golubchik, who has a shared copyright to this code + added support for long options (my_getopt) 22.5.2002 by Jani Tolonen */ + +#include "ma_ftdefs.h" +#include "maria_ft_test1.h" +#include + +static int key_field=FIELD_VARCHAR,extra_field=FIELD_SKIP_ENDSPACE; +static uint key_length=200,extra_length=50; +static int key_type=HA_KEYTYPE_TEXT; +static int verbose=0,silent=0,skip_update=0, + no_keys=0,no_stopwords=0,no_search=0,no_fulltext=0; +static int create_flag=0,error=0; + +#define MAX_REC_LENGTH 300 +static char record[MAX_REC_LENGTH],read_record[MAX_REC_LENGTH]; + +static int run_test(const char *filename); +static void get_options(int argc, char *argv[]); +static void create_record(char *, int); +static void usage(); + +static struct my_option my_long_options[] = +{ + {"", 'v', "", 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, + {"", '?', "", 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, + {"", 'h', "", 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, + {"", 'V', "", 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, + {"", 'v', "", 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, + {"", 's', "", 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, + {"", 'N', "", 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, + {"", 'S', "", 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, + {"", 'K', "", 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, + {"", 'F', "", 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, + {"", 'U', "", 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, + {"", '#', "", 0, 0, 0, GET_STR, OPT_ARG, 0, 0, 0, 0, 0, 0}, + { 0, 0, 0, 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0} +}; + +int main(int argc, char *argv[]) +{ + MY_INIT(argv[0]); + + get_options(argc,argv); + maria_init(); + + exit(run_test("FT1")); +} + +static MARIA_COLUMNDEF recinfo[3]; +static MARIA_KEYDEF keyinfo[2]; +static HA_KEYSEG keyseg[10]; + +static int run_test(const char *filename) +{ + MARIA_HA *file; + int i,j; + my_off_t pos; + + bzero((char*) recinfo,sizeof(recinfo)); + + /* First define 2 columns */ + recinfo[0].type=extra_field; + recinfo[0].length= (extra_field == FIELD_BLOB ? 4 + maria_portable_sizeof_char_ptr : + extra_length); + if (extra_field == FIELD_VARCHAR) + recinfo[0].length+= HA_VARCHAR_PACKLENGTH(extra_length); + recinfo[1].type=key_field; + recinfo[1].length= (key_field == FIELD_BLOB ? 4+maria_portable_sizeof_char_ptr : + key_length); + if (key_field == FIELD_VARCHAR) + recinfo[1].length+= HA_VARCHAR_PACKLENGTH(key_length); + + /* Define a key over the first column */ + keyinfo[0].seg=keyseg; + keyinfo[0].keysegs=1; + keyinfo[0].seg[0].type= key_type; + keyinfo[0].seg[0].flag= (key_field == FIELD_BLOB) ? HA_BLOB_PART: + (key_field == FIELD_VARCHAR) ? HA_VAR_LENGTH_PART:0; + keyinfo[0].seg[0].start=recinfo[0].length; + keyinfo[0].seg[0].length=key_length; + keyinfo[0].seg[0].null_bit= 0; + keyinfo[0].seg[0].null_pos=0; + keyinfo[0].seg[0].language= default_charset_info->number; + keyinfo[0].flag = (no_fulltext?HA_PACK_KEY:HA_FULLTEXT); + + if (!silent) + printf("- Creating isam-file\n"); + if (maria_create(filename,(no_keys?0:1),keyinfo,2,recinfo,0,NULL, + (MARIA_CREATE_INFO*) 0, create_flag)) + goto err; + if (!(file=maria_open(filename,2,0))) + goto err; + + if (!silent) + printf("- %s stopwords\n",no_stopwords?"Skipping":"Initializing"); + maria_ft_init_stopwords(no_stopwords?NULL:maria_ft_precompiled_stopwords); + + if (!silent) + printf("- Writing key:s\n"); + + my_errno=0; + for (i=NUPD ; indocs); + for (j=0;j<5;j++) + { + double w; int err; + err= maria_ft_nlq_read_next(result, read_record); + if (err==HA_ERR_END_OF_FILE) + { + printf("No more matches!\n"); + break; + } + else if (err) + { + printf("maria_ft_read_next %d failed with errno %3d\n",j,my_errno); + break; + } + w=maria_ft_nlq_get_relevance(result); + if (key_field == FIELD_VARCHAR) + { + uint l; + char *p; + p=recinfo[0].length+read_record; + l=uint2korr(p); + printf("%10.7f: %.*s\n",w,(int) l,p+2); + } + else + printf("%10.7f: %.*s\n",w,recinfo[1].length, + recinfo[0].length+read_record); + } + maria_ft_nlq_close_search(result); + } + + if (maria_close(file)) goto err; + maria_end(); + my_end(MY_CHECK_ERROR); + + return (0); +err: + printf("got error: %3d when using maria-database\n",my_errno); + return 1; /* skip warning */ +} + +static char blob_key[MAX_REC_LENGTH]; +/* static char blob_record[MAX_REC_LENGTH+20*20]; */ + +void create_record(char *pos, int n) +{ + bzero((char*) pos,MAX_REC_LENGTH); + if (recinfo[0].type == FIELD_BLOB) + { + uint tmp; + char *ptr; + strnmov(blob_key,data[n].f0,keyinfo[0].seg[0].length); + tmp=strlen(blob_key); + int4store(pos,tmp); + ptr=blob_key; + memcpy_fixed(pos+4,&ptr,sizeof(char*)); + pos+=recinfo[0].length; + } + else if (recinfo[0].type == FIELD_VARCHAR) + { + uint tmp; + /* -1 is here because pack_length is stored in seg->length */ + uint pack_length= HA_VARCHAR_PACKLENGTH(keyinfo[0].seg[0].length-1); + strnmov(pos+pack_length,data[n].f0,keyinfo[0].seg[0].length); + tmp=strlen(pos+pack_length); + if (pack_length == 1) + *pos= (char) tmp; + else + int2store(pos,tmp); + pos+=recinfo[0].length; + } + else + { + strnmov(pos,data[n].f0,keyinfo[0].seg[0].length); + pos+=recinfo[0].length; + } + if (recinfo[1].type == FIELD_BLOB) + { + uint tmp; + char *ptr; + strnmov(blob_key,data[n].f2,keyinfo[0].seg[0].length); + tmp=strlen(blob_key); + int4store(pos,tmp); + ptr=blob_key; + memcpy_fixed(pos+4,&ptr,sizeof(char*)); + pos+=recinfo[1].length; + } + else if (recinfo[1].type == FIELD_VARCHAR) + { + uint tmp; + /* -1 is here because pack_length is stored in seg->length */ + uint pack_length= HA_VARCHAR_PACKLENGTH(keyinfo[0].seg[0].length-1); + strnmov(pos+pack_length,data[n].f2,keyinfo[0].seg[0].length); + tmp=strlen(pos+1); + if (pack_length == 1) + *pos= (char) tmp; + else + int2store(pos,tmp); + pos+=recinfo[1].length; + } + else + { + strnmov(pos,data[n].f2,keyinfo[0].seg[0].length); + pos+=recinfo[1].length; + } +} + + +static my_bool +get_one_option(int optid, const struct my_option *opt __attribute__((unused)), + char *argument) +{ + switch(optid) { + case 'v': verbose=1; break; + case 's': silent=1; break; + case 'F': no_fulltext=1; no_search=1; + case 'U': skip_update=1; break; + case 'K': no_keys=no_search=1; break; + case 'N': no_search=1; break; + case 'S': no_stopwords=1; break; + case '#': + DBUG_PUSH (argument); + break; + case 'V': + case '?': + case 'h': + usage(); + exit(1); + } + return 0; +} + +/* Read options */ + +static void get_options(int argc,char *argv[]) +{ + int ho_error; + + if ((ho_error=handle_options(&argc, &argv, my_long_options, get_one_option))) + exit(ho_error); + return; +} /* get options */ + + +static void usage() +{ + printf("%s [options]\n", my_progname); + my_print_help(my_long_options); + my_print_variables(my_long_options); +} diff --git a/storage/maria/ma_ft_test1.h b/storage/maria/ma_ft_test1.h new file mode 100644 index 00000000000..9449f063125 --- /dev/null +++ b/storage/maria/ma_ft_test1.h @@ -0,0 +1,421 @@ +/* Copyright (C) 2006 MySQL AB & MySQL Finland AB & TCX DataKonsult AB + + 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; either version 2 of the License, or + (at your option) any later version. + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ + +/* Written by Sergei A. Golubchik, who has a shared copyright to this code */ + +#define NUPD 20 +#define NDATAS 389 +struct { const char *f0, *f2; } data[NDATAS] = { + {"1", "General Information about MySQL"}, + {"1.1", "What is MySQL?"}, + {"1.2", "About this manual"}, + {"1.3", "History of MySQL"}, + {"1.4", "The main features of MySQL"}, + {"1.5", "General SQL information and tutorials"}, + {"1.6", "Useful MySQL-related links"}, + {"1.7", "What are stored procedures and triggers and so on?"}, + {"2", "MySQL mailing lists and how to ask questions/give error (bug) reports"}, + {"2.1", "Subscribing to/un-subscribing from the MySQL mailing list"}, + {"2.2", "Asking questions or reporting bugs"}, + {"2.3", "I think I have found a bug. What information do you need to help me?"}, + {"2.3.1", "MySQL keeps crashing"}, + {"2.4", "Guidelines for answering questions on the mailing list"}, + {"3", "Licensing or When do I have/want to pay for MySQL?"}, + {"3.1", "How much does MySQL cost?"}, + {"3.2", "How do I get commercial support?"}, + {"3.2.1", "Types of commercial support"}, + {"3.2.1.1", "Basic email support"}, + {"3.2.1.2", "Extended email support"}, +/*------------------------------- NUPD=20 -------------------------------*/ + {"3.2.1.3", "Asking: Login support"}, + {"3.2.1.4", "Extended login support"}, + {"3.3", "How do I pay for licenses/support?"}, + {"3.4", "Who do I contact when I want more information about licensing/support?"}, + {"3.5", "What Copyright does MySQL use?"}, + {"3.6", "When may I distribute MySQL commercially without a fee?"}, + {"3.7", "I want to sell a product that can be configured to use MySQL"}, + {"3.8", "I am running a commercial web server using MySQL"}, + {"3.9", "Do I need a license to sell commercial Perl/tcl/PHP/Web+ etc applications?"}, + {"3.10", "Possible future changes in the licensing"}, + {"4", "Compiling and installing MySQL"}, + {"4.1", "How do I get MySQL?"}, + {"4.2", "Which MySQL version should I use?"}, + {"4.3", "How/when will you release updates?"}, + {"4.4", "What operating systems does MySQL support?"}, + {"4.5", "Compiling MySQL from source code"}, + {"4.5.1", "Quick installation overview"}, + {"4.5.2", "Usual configure switches"}, + {"4.5.3", "Applying a patch"}, + {"4.6", "Problems compiling?"}, + {"4.7", "General compilation notes"}, + {"4.8", "MIT-pthreads notes (FreeBSD)"}, + {"4.9", "Perl installation comments"}, + {"4.10", "Special things to consider for some machine/OS combinations"}, + {"4.10.1", "Solaris notes"}, + {"4.10.2", "SunOS 4 notes"}, + {"4.10.3", "Linux notes for all versions"}, + {"4.10.3.1", "Linux-x86 notes"}, + {"4.10.3.2", "RedHat 5.0"}, + {"4.10.3.3", "RedHat 5.1"}, + {"4.10.3.4", "Linux-Sparc notes"}, + {"4.10.3.5", "Linux-Alpha notes"}, + {"4.10.3.6", "MkLinux notes"}, + {"4.10.4", "Alpha-DEC-Unix notes"}, + {"4.10.5", "Alpha-DEC-OSF1 notes"}, + {"4.10.6", "SGI-IRIX notes"}, + {"4.10.7", "FreeBSD notes"}, + {"4.10.7.1", "FreeBSD-3.0 notes"}, + {"4.10.8", "BSD/OS 2.# notes"}, + {"4.10.8.1", "BSD/OS 3.# notes"}, + {"4.10.9", "SCO notes"}, + {"4.10.10", "SCO Unixware 7.0 notes"}, + {"4.10.11", "IBM-AIX notes"}, + {"4.10.12", "HP-UX notes"}, + {"4.11", "TcX binaries"}, + {"4.12", "Win32 notes"}, + {"4.13", "Installation instructions for MySQL binary releases"}, + {"4.13.1", "How to get MySQL Perl support working"}, + {"4.13.2", "Linux notes"}, + {"4.13.3", "HP-UX notes"}, + {"4.13.4", "Linking client libraries"}, + {"4.14", "Problems running mysql_install_db"}, + {"4.15", "Problems starting MySQL"}, + {"4.16", "Automatic start/stop of MySQL"}, + {"4.17", "Option files"}, + {"5", "How standards-compatible is MySQL?"}, + {"5.1", "What extensions has MySQL to ANSI SQL92?"}, + {"5.2", "What functionality is missing in MySQL?"}, + {"5.2.1", "Sub-selects"}, + {"5.2.2", "SELECT INTO TABLE"}, + {"5.2.3", "Transactions"}, + {"5.2.4", "Triggers"}, + {"5.2.5", "Foreign Keys"}, + {"5.2.5.1", "Some reasons NOT to use FOREIGN KEYS"}, + {"5.2.6", "Views"}, + {"5.2.7", "-- as start of a comment"}, + {"5.3", "What standards does MySQL follow?"}, + {"5.4", "What functions exist only for compatibility?"}, + {"5.5", "Limitations of BLOB and TEXT types"}, + {"5.6", "How to cope without COMMIT-ROLLBACK"}, + {"6", "The MySQL access privilege system"}, + {"6.1", "What the privilege system does"}, + {"6.2", "Connecting to the MySQL server"}, + {"6.2.1", "Keeping your password secure"}, + {"6.3", "Privileges provided by MySQL"}, + {"6.4", "How the privilege system works"}, + {"6.5", "The privilege tables"}, + {"6.6", "Setting up the initial MySQL privileges"}, + {"6.7", "Adding new user privileges to MySQL"}, + {"6.8", "An example permission setup"}, + {"6.9", "Causes of Access denied errors"}, + {"6.10", "How to make MySQL secure against crackers"}, + {"7", "MySQL language reference"}, + {"7.1", "Literals: how to write strings and numbers"}, + {"7.1.1", "Strings"}, + {"7.1.2", "Numbers"}, + {"7.1.3", "NULL values"}, + {"7.1.4", "Database, table, index, column and alias names"}, + {"7.1.4.1", "Case sensitivity in names"}, + {"7.2", "Column types"}, + {"7.2.1", "Column type storage requirements"}, + {"7.2.5", "Numeric types"}, + {"7.2.6", "Date and time types"}, + {"7.2.6.1", "The DATE type"}, + {"7.2.6.2", "The TIME type"}, + {"7.2.6.3", "The DATETIME type"}, + {"7.2.6.4", "The TIMESTAMP type"}, + {"7.2.6.5", "The YEAR type"}, + {"7.2.6.6", "Miscellaneous date and time properties"}, + {"7.2.7", "String types"}, + {"7.2.7.1", "The CHAR and VARCHAR types"}, + {"7.2.7.2", "The BLOB and TEXT types"}, + {"7.2.7.3", "The ENUM type"}, + {"7.2.7.4", "The SET type"}, + {"7.2.8", "Choosing the right type for a column"}, + {"7.2.9", "Column indexes"}, + {"7.2.10", "Multiple-column indexes"}, + {"7.2.11", "Using column types from other database engines"}, + {"7.3", "Functions for use in SELECT and WHERE clauses"}, + {"7.3.1", "Grouping functions"}, + {"7.3.2", "Normal arithmetic operations"}, + {"7.3.3", "Bit functions"}, + {"7.3.4", "Logical operations"}, + {"7.3.5", "Comparison operators"}, + {"7.3.6", "String comparison functions"}, + {"7.3.7", "Control flow functions"}, + {"7.3.8", "Mathematical functions"}, + {"7.3.9", "String functions"}, + {"7.3.10", "Date and time functions"}, + {"7.3.11", "Miscellaneous functions"}, + {"7.3.12", "Functions for use with GROUP BY clauses"}, + {"7.4", "CREATE DATABASE syntax"}, + {"7.5", "DROP DATABASE syntax"}, + {"7.6", "CREATE TABLE syntax"}, + {"7.7", "ALTER TABLE syntax"}, + {"7.8", "OPTIMIZE TABLE syntax"}, + {"7.9", "DROP TABLE syntax"}, + {"7.10", "DELETE syntax"}, + {"7.11", "SELECT syntax"}, + {"7.12", "JOIN syntax"}, + {"7.13", "INSERT syntax"}, + {"7.14", "REPLACE syntax"}, + {"7.15", "LOAD DATA INFILE syntax"}, + {"7.16", "UPDATE syntax"}, + {"7.17", "USE syntax"}, + {"7.18", "SHOW syntax (Get information about tables, columns...)"}, + {"7.19", "EXPLAIN syntax (Get information about a SELECT)"}, + {"7.20", "DESCRIBE syntax (Get information about columns)"}, + {"7.21", "LOCK TABLES/UNLOCK TABLES syntax"}, + {"7.22", "SET OPTION syntax"}, + {"7.23", "GRANT syntax (Compatibility function)"}, + {"7.24", "CREATE INDEX syntax (Compatibility function)"}, + {"7.25", "DROP INDEX syntax (Compatibility function)"}, + {"7.26", "Comment syntax"}, + {"7.27", "CREATE FUNCTION/DROP FUNCTION syntax"}, + {"7.28", "Is MySQL picky about reserved words?"}, + {"8", "Example SQL queries"}, + {"8.1", "Queries from twin project"}, + {"8.1.1", "Find all non-distributed twins"}, + {"8.1.2", "Show a table on twin pair status"}, + {"9", "How safe/stable is MySQL?"}, + {"9.1", "How stable is MySQL?"}, + {"9.2", "Why are there is so many releases of MySQL?"}, + {"9.3", "Checking a table for errors"}, + {"9.4", "How to repair tables"}, + {"9.5", "Is there anything special to do when upgrading/downgrading MySQL?"}, + {"9.5.1", "Upgrading from a 3.21 version to 3.22"}, + {"9.5.2", "Upgrading from a 3.20 version to 3.21"}, + {"9.5.3", "Upgrading to another architecture"}, + {"9.6", "Year 2000 compliance"}, + {"10", "MySQL Server functions"}, + {"10.1", "What languages are supported by MySQL?"}, + {"10.1.1", "Character set used for data & sorting"}, + {"10.2", "The update log"}, + {"10.3", "How big can MySQL tables be?"}, + {"11", "Getting maximum performance from MySQL"}, + {"11.1", "How does one change the size of MySQL buffers?"}, + {"11.2", "How compiling and linking affects the speed of MySQL"}, + {"11.3", "How does MySQL use memory?"}, + {"11.4", "How does MySQL use indexes?"}, + {"11.5", "What optimizations are done on WHERE clauses?"}, + {"11.6", "How does MySQL open & close tables?"}, + {"11.6.0.1", "What are the drawbacks of creating possibly thousands of tables in a database?"}, + {"11.7", "How does MySQL lock tables?"}, + {"11.8", "How should I arrange my table to be as fast/small as possible?"}, + {"11.9", "What affects the speed of INSERT statements?"}, + {"11.10", "What affects the speed DELETE statements?"}, + {"11.11", "How do I get MySQL to run at full speed?"}, + {"11.12", "What are the different row formats? Or, when should VARCHAR/CHAR be used?"}, + {"11.13", "Why so many open tables?"}, + {"12", "MySQL benchmark suite"}, + {"13", "MySQL Utilites"}, + {"13.1", "Overview of the different MySQL programs"}, + {"13.2", "The MySQL table check, optimize and repair program"}, + {"13.2.1", "isamchk memory use"}, + {"13.2.2", "Getting low-level table information"}, + {"13.3", "The MySQL compressed read-only table generator"}, + {"14", "Adding new functions to MySQL"}, + {"15", "MySQL ODBC Support"}, + {"15.1", "Operating systems supported by MyODBC"}, + {"15.2", "How to report problems with MyODBC"}, + {"15.3", "Programs known to work with MyODBC"}, + {"15.4", "How to fill in the various fields in the ODBC administrator program"}, + {"15.5", "How to get the value of an AUTO_INCREMENT column in ODBC"}, + {"16", "Problems and common errors"}, + {"16.1", "Some common errors when using MySQL"}, + {"16.1.1", "MySQL server has gone away error"}, + {"16.1.2", "Can't connect to local MySQL server error"}, + {"16.1.3", "Out of memory error"}, + {"16.1.4", "Packet too large error"}, + {"16.1.5", "The table is full error"}, + {"16.1.6", "Commands out of sync error in client"}, + {"16.1.7", "Removing user error"}, + {"16.2", "How MySQL handles a full disk"}, + {"16.3", "How to run SQL commands from a text file"}, + {"16.4", "Where MySQL stores temporary files"}, + {"16.5", "Access denied error"}, + {"16.6", "How to run MySQL as a normal user"}, + {"16.7", "Problems with file permissions"}, + {"16.8", "File not found"}, + {"16.9", "Problems using DATE columns"}, + {"16.10", "Case sensitivity in searches"}, + {"16.11", "Problems with NULL values"}, + {"17", "Solving some common problems with MySQL"}, + {"17.1", "Database replication"}, + {"17.2", "Database backups"}, + {"18", "MySQL client tools and API's"}, + {"18.1", "MySQL C API"}, + {"18.2", "C API datatypes"}, + {"18.3", "C API function overview"}, + {"18.4", "C API function descriptions"}, + {"18.4.1", "mysql_affected_rows()"}, + {"18.4.2", "mysql_close()"}, + {"18.4.3", "mysql_connect()"}, + {"18.4.4", "mysql_create_db()"}, + {"18.4.5", "mysql_data_seek()"}, + {"18.4.6", "mysql_debug()"}, + {"18.4.7", "mysql_drop_db()"}, + {"18.4.8", "mysql_dump_debug_info()"}, + {"18.4.9", "mysql_eof()"}, + {"18.4.10", "mysql_errno()"}, + {"18.4.11", "mysql_error()"}, + {"18.4.12", "mysql_escape_string()"}, + {"18.4.13", "mysql_fetch_field()"}, + {"18.4.14", "mysql_fetch_fields()"}, + {"18.4.15", "mysql_fetch_field_direct()"}, + {"18.4.16", "mysql_fetch_lengths()"}, + {"18.4.17", "mysql_fetch_row()"}, + {"18.4.18", "mysql_field_seek()"}, + {"18.4.19", "mysql_field_tell()"}, + {"18.4.20", "mysql_free_result()"}, + {"18.4.21", "mysql_get_client_info()"}, + {"18.4.22", "mysql_get_host_info()"}, + {"18.4.23", "mysql_get_proto_info()"}, + {"18.4.24", "mysql_get_server_info()"}, + {"18.4.25", "mysql_info()"}, + {"18.4.26", "mysql_init()"}, + {"18.4.27", "mysql_insert_id()"}, + {"18.4.28", "mysql_kill()"}, + {"18.4.29", "mysql_list_dbs()"}, + {"18.4.30", "mysql_list_fields()"}, + {"18.4.31", "mysql_list_processes()"}, + {"18.4.32", "mysql_list_tables()"}, + {"18.4.33", "mysql_num_fields()"}, + {"18.4.34", "mysql_num_rows()"}, + {"18.4.35", "mysql_query()"}, + {"18.4.36", "mysql_real_connect()"}, + {"18.4.37", "mysql_real_query()"}, + {"18.4.38", "mysql_reload()"}, + {"18.4.39", "mysql_row_tell()"}, + {"18.4.40", "mysql_select_db()"}, + {"18.4.41", "mysql_shutdown()"}, + {"18.4.42", "mysql_stat()"}, + {"18.4.43", "mysql_store_result()"}, + {"18.4.44", "mysql_thread_id()"}, + {"18.4.45", "mysql_use_result()"}, + {"18.4.46", "Why is it that after mysql_query() returns success, mysql_store_result() sometimes returns NULL?"}, + {"18.4.47", "What results can I get from a query?"}, + {"18.4.48", "How can I get the unique ID for the last inserted row?"}, + {"18.4.49", "Problems linking with the C API"}, + {"18.4.50", "How to make a thread-safe client"}, + {"18.5", "MySQL Perl API's"}, + {"18.5.1", "DBI with DBD::mysql"}, + {"18.5.1.1", "The DBI interface"}, + {"18.5.1.2", "More DBI/DBD information"}, + {"18.6", "MySQL Java connectivity (JDBC)"}, + {"18.7", "MySQL PHP API's"}, + {"18.8", "MySQL C++ API's"}, + {"18.9", "MySQL Python API's"}, + {"18.10", "MySQL TCL API's"}, + {"19", "How MySQL compares to other databases"}, + {"19.1", "How MySQL compares to mSQL"}, + {"19.1.1", "How to convert mSQL tools for MySQL"}, + {"19.1.2", "How mSQL and MySQL client/server communications protocols differ"}, + {"19.1.3", "How mSQL 2.0 SQL syntax differs from MySQL"}, + {"19.2", "How MySQL compares to PostgreSQL"}, + {"A", "Some users of MySQL"}, + {"B", "Contributed programs"}, + {"C", "Contributors to MySQL"}, + {"D", "MySQL change history"}, + {"19.3", "Changes in release 3.22.x (Alpha version)"}, + {"19.3.1", "Changes in release 3.22.7"}, + {"19.3.2", "Changes in release 3.22.6"}, + {"19.3.3", "Changes in release 3.22.5"}, + {"19.3.4", "Changes in release 3.22.4"}, + {"19.3.5", "Changes in release 3.22.3"}, + {"19.3.6", "Changes in release 3.22.2"}, + {"19.3.7", "Changes in release 3.22.1"}, + {"19.3.8", "Changes in release 3.22.0"}, + {"19.4", "Changes in release 3.21.x"}, + {"19.4.1", "Changes in release 3.21.33"}, + {"19.4.2", "Changes in release 3.21.32"}, + {"19.4.3", "Changes in release 3.21.31"}, + {"19.4.4", "Changes in release 3.21.30"}, + {"19.4.5", "Changes in release 3.21.29"}, + {"19.4.6", "Changes in release 3.21.28"}, + {"19.4.7", "Changes in release 3.21.27"}, + {"19.4.8", "Changes in release 3.21.26"}, + {"19.4.9", "Changes in release 3.21.25"}, + {"19.4.10", "Changes in release 3.21.24"}, + {"19.4.11", "Changes in release 3.21.23"}, + {"19.4.12", "Changes in release 3.21.22"}, + {"19.4.13", "Changes in release 3.21.21a"}, + {"19.4.14", "Changes in release 3.21.21"}, + {"19.4.15", "Changes in release 3.21.20"}, + {"19.4.16", "Changes in release 3.21.19"}, + {"19.4.17", "Changes in release 3.21.18"}, + {"19.4.18", "Changes in release 3.21.17"}, + {"19.4.19", "Changes in release 3.21.16"}, + {"19.4.20", "Changes in release 3.21.15"}, + {"19.4.21", "Changes in release 3.21.14b"}, + {"19.4.22", "Changes in release 3.21.14a"}, + {"19.4.23", "Changes in release 3.21.13"}, + {"19.4.24", "Changes in release 3.21.12"}, + {"19.4.25", "Changes in release 3.21.11"}, + {"19.4.26", "Changes in release 3.21.10"}, + {"19.4.27", "Changes in release 3.21.9"}, + {"19.4.28", "Changes in release 3.21.8"}, + {"19.4.29", "Changes in release 3.21.7"}, + {"19.4.30", "Changes in release 3.21.6"}, + {"19.4.31", "Changes in release 3.21.5"}, + {"19.4.32", "Changes in release 3.21.4"}, + {"19.4.33", "Changes in release 3.21.3"}, + {"19.4.34", "Changes in release 3.21.2"}, + {"19.4.35", "Changes in release 3.21.0"}, + {"19.5", "Changes in release 3.20.x"}, + {"19.5.1", "Changes in release 3.20.18"}, + {"19.5.2", "Changes in release 3.20.17"}, + {"19.5.3", "Changes in release 3.20.16"}, + {"19.5.4", "Changes in release 3.20.15"}, + {"19.5.5", "Changes in release 3.20.14"}, + {"19.5.6", "Changes in release 3.20.13"}, + {"19.5.7", "Changes in release 3.20.11"}, + {"19.5.8", "Changes in release 3.20.10"}, + {"19.5.9", "Changes in release 3.20.9"}, + {"19.5.10", "Changes in release 3.20.8"}, + {"19.5.11", "Changes in release 3.20.7"}, + {"19.5.12", "Changes in release 3.20.6"}, + {"19.5.13", "Changes in release 3.20.3"}, + {"19.5.14", "Changes in release 3.20.0"}, + {"19.6", "Changes in release 3.19.x"}, + {"19.6.1", "Changes in release 3.19.5"}, + {"19.6.2", "Changes in release 3.19.4"}, + {"19.6.3", "Changes in release 3.19.3"}, + {"E", "Known errors and design deficiencies in MySQL"}, + {"F", "List of things we want to add to MySQL in the future (The TODO)"}, + {"19.7", "Things that must done in the real near future"}, + {"19.8", "Things that have to be done sometime"}, + {"19.9", "Some things we don't have any plans to do"}, + {"G", "Comments on porting to other systems"}, + {"19.10", "Debugging MySQL"}, + {"19.11", "Comments about RTS threads"}, + {"19.12", "What is the difference between different thread packages?"}, + {"H", "Description of MySQL regular expression syntax"}, + {"I", "What is Unireg?"}, + {"J", "The MySQL server license"}, + {"K", "The MySQL license for Microsoft operating systems"}, + {"*", "SQL command, type and function index"}, + {"*", "Concept Index"} +}; + +#define NQUERIES 5 +const char *query[NQUERIES]={ + "mysql information and manual", + "upgrading from previous version", + "column indexes", + "against about after more right the with/without", /* stopwords test */ + "mysql license and copyright" +}; diff --git a/storage/maria/ma_ft_update.c b/storage/maria/ma_ft_update.c new file mode 100644 index 00000000000..c9e2112578c --- /dev/null +++ b/storage/maria/ma_ft_update.c @@ -0,0 +1,359 @@ +/* Copyright (C) 2006 MySQL AB & MySQL Finland AB & TCX DataKonsult AB + + 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; either version 2 of the License, or + (at your option) any later version. + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ + +/* Written by Sergei A. Golubchik, who has a shared copyright to this code */ + +/* functions to work with full-text indices */ + +#include "ma_ftdefs.h" +#include + +void _ma_ft_segiterator_init(MARIA_HA *info, uint keynr, const byte *record, + FT_SEG_ITERATOR *ftsi) +{ + DBUG_ENTER("_ma_ft_segiterator_init"); + + ftsi->num=info->s->keyinfo[keynr].keysegs; + ftsi->seg=info->s->keyinfo[keynr].seg; + ftsi->rec=record; + DBUG_VOID_RETURN; +} + +void _ma_ft_segiterator_dummy_init(const byte *record, uint len, + FT_SEG_ITERATOR *ftsi) +{ + DBUG_ENTER("_ma_ft_segiterator_dummy_init"); + + ftsi->num=1; + ftsi->seg=0; + ftsi->pos=record; + ftsi->len=len; + DBUG_VOID_RETURN; +} + +/* + This function breaks convention "return 0 in success" + but it's easier to use like this + + while(_ma_ft_segiterator()) + + so "1" means "OK", "0" means "EOF" +*/ + +uint _ma_ft_segiterator(register FT_SEG_ITERATOR *ftsi) +{ + DBUG_ENTER("_ma_ft_segiterator"); + + if (!ftsi->num) + DBUG_RETURN(0); + + ftsi->num--; + if (!ftsi->seg) + DBUG_RETURN(1); + + ftsi->seg--; + + if (ftsi->seg->null_bit && + (ftsi->rec[ftsi->seg->null_pos] & ftsi->seg->null_bit)) + { + ftsi->pos=0; + DBUG_RETURN(1); + } + ftsi->pos= ftsi->rec+ftsi->seg->start; + if (ftsi->seg->flag & HA_VAR_LENGTH_PART) + { + uint pack_length= (ftsi->seg->bit_start); + ftsi->len= (pack_length == 1 ? (uint) *(uchar*) ftsi->pos : + uint2korr(ftsi->pos)); + ftsi->pos+= pack_length; /* Skip VARCHAR length */ + DBUG_RETURN(1); + } + if (ftsi->seg->flag & HA_BLOB_PART) + { + ftsi->len= _ma_calc_blob_length(ftsi->seg->bit_start,ftsi->pos); + memcpy_fixed((char*) &ftsi->pos, ftsi->pos+ftsi->seg->bit_start, + sizeof(char*)); + DBUG_RETURN(1); + } + ftsi->len=ftsi->seg->length; + DBUG_RETURN(1); +} + + +/* parses a document i.e. calls maria_ft_parse for every keyseg */ + +uint _ma_ft_parse(TREE *parsed, MARIA_HA *info, uint keynr, + const byte *record, my_bool with_alloc, + MYSQL_FTPARSER_PARAM *param) +{ + FT_SEG_ITERATOR ftsi; + struct st_mysql_ftparser *parser; + DBUG_ENTER("_ma_ft_parse"); + + _ma_ft_segiterator_init(info, keynr, record, &ftsi); + + maria_ft_parse_init(parsed, info->s->keyinfo[keynr].seg->charset); + parser= info->s->keyinfo[keynr].parser; + while (_ma_ft_segiterator(&ftsi)) + { + if (ftsi.pos) + if (maria_ft_parse(parsed, (byte *)ftsi.pos, ftsi.len, with_alloc, parser, + param)) + DBUG_RETURN(1); + } + DBUG_RETURN(0); +} + +FT_WORD * _ma_ft_parserecord(MARIA_HA *info, uint keynr, const byte *record) +{ + TREE ptree; + MYSQL_FTPARSER_PARAM *param; + DBUG_ENTER("_ma_ft_parserecord"); + if (! (param= maria_ftparser_call_initializer(info, keynr))) + DBUG_RETURN(NULL); + bzero((char*) &ptree, sizeof(ptree)); + if (_ma_ft_parse(&ptree, info, keynr, record, 0, param)) + DBUG_RETURN(NULL); + + DBUG_RETURN(maria_ft_linearize(&ptree)); +} + +static int _ma_ft_store(MARIA_HA *info, uint keynr, byte *keybuf, + FT_WORD *wlist, my_off_t filepos) +{ + uint key_length; + DBUG_ENTER("_ma_ft_store"); + + for (; wlist->pos; wlist++) + { + key_length= _ma_ft_make_key(info,keynr,keybuf,wlist,filepos); + if (_ma_ck_write(info,keynr,(uchar*) keybuf,key_length)) + DBUG_RETURN(1); + } + DBUG_RETURN(0); +} + +static int _ma_ft_erase(MARIA_HA *info, uint keynr, byte *keybuf, + FT_WORD *wlist, my_off_t filepos) +{ + uint key_length, err=0; + DBUG_ENTER("_ma_ft_erase"); + + for (; wlist->pos; wlist++) + { + key_length= _ma_ft_make_key(info,keynr,keybuf,wlist,filepos); + if (_ma_ck_delete(info,keynr,(uchar*) keybuf,key_length)) + err=1; + } + DBUG_RETURN(err); +} + +/* + Compares an appropriate parts of two WORD_KEY keys directly out of records + returns 1 if they are different +*/ + +#define THOSE_TWO_DAMN_KEYS_ARE_REALLY_DIFFERENT 1 +#define GEE_THEY_ARE_ABSOLUTELY_IDENTICAL 0 + +int _ma_ft_cmp(MARIA_HA *info, uint keynr, const byte *rec1, const byte *rec2) +{ + FT_SEG_ITERATOR ftsi1, ftsi2; + CHARSET_INFO *cs=info->s->keyinfo[keynr].seg->charset; + DBUG_ENTER("_ma_ft_cmp"); +#ifndef MYSQL_HAS_TRUE_CTYPE_IMPLEMENTATION + if (cs->mbmaxlen > 1) + DBUG_RETURN(THOSE_TWO_DAMN_KEYS_ARE_REALLY_DIFFERENT); +#endif + + _ma_ft_segiterator_init(info, keynr, rec1, &ftsi1); + _ma_ft_segiterator_init(info, keynr, rec2, &ftsi2); + + while (_ma_ft_segiterator(&ftsi1) && _ma_ft_segiterator(&ftsi2)) + { + if ((ftsi1.pos != ftsi2.pos) && + (!ftsi1.pos || !ftsi2.pos || + ha_compare_text(cs, (uchar*) ftsi1.pos,ftsi1.len, + (uchar*) ftsi2.pos,ftsi2.len,0,0))) + DBUG_RETURN(THOSE_TWO_DAMN_KEYS_ARE_REALLY_DIFFERENT); + } + DBUG_RETURN(GEE_THEY_ARE_ABSOLUTELY_IDENTICAL); +} + + +/* update a document entry */ + +int _ma_ft_update(MARIA_HA *info, uint keynr, byte *keybuf, + const byte *oldrec, const byte *newrec, my_off_t pos) +{ + int error= -1; + FT_WORD *oldlist,*newlist, *old_word, *new_word; + CHARSET_INFO *cs=info->s->keyinfo[keynr].seg->charset; + uint key_length; + int cmp, cmp2; + DBUG_ENTER("_ma_ft_update"); + + if (!(old_word=oldlist= _ma_ft_parserecord(info, keynr, oldrec))) + goto err0; + if (!(new_word=newlist= _ma_ft_parserecord(info, keynr, newrec))) + goto err1; + + error=0; + while(old_word->pos && new_word->pos) + { + cmp= ha_compare_text(cs, (uchar*) old_word->pos,old_word->len, + (uchar*) new_word->pos,new_word->len,0,0); + cmp2= cmp ? 0 : (fabs(old_word->weight - new_word->weight) > 1.e-5); + + if (cmp < 0 || cmp2) + { + key_length= _ma_ft_make_key(info,keynr,keybuf,old_word,pos); + if ((error= _ma_ck_delete(info,keynr,(uchar*) keybuf,key_length))) + goto err2; + } + if (cmp > 0 || cmp2) + { + key_length= _ma_ft_make_key(info,keynr,keybuf,new_word,pos); + if ((error= _ma_ck_write(info,keynr,(uchar*) keybuf,key_length))) + goto err2; + } + if (cmp<=0) old_word++; + if (cmp>=0) new_word++; + } + if (old_word->pos) + error= _ma_ft_erase(info,keynr,keybuf,old_word,pos); + else if (new_word->pos) + error= _ma_ft_store(info,keynr,keybuf,new_word,pos); + +err2: + my_free((char*) newlist,MYF(0)); +err1: + my_free((char*) oldlist,MYF(0)); +err0: + DBUG_RETURN(error); +} + + +/* adds a document to the collection */ + +int _ma_ft_add(MARIA_HA *info, uint keynr, byte *keybuf, const byte *record, + my_off_t pos) +{ + int error= -1; + FT_WORD *wlist; + DBUG_ENTER("_ma_ft_add"); + + if ((wlist= _ma_ft_parserecord(info, keynr, record))) + { + error= _ma_ft_store(info,keynr,keybuf,wlist,pos); + my_free((char*) wlist,MYF(0)); + } + DBUG_RETURN(error); +} + + +/* removes a document from the collection */ + +int _ma_ft_del(MARIA_HA *info, uint keynr, byte *keybuf, const byte *record, + my_off_t pos) +{ + int error= -1; + FT_WORD *wlist; + DBUG_ENTER("_ma_ft_del"); + DBUG_PRINT("enter",("keynr: %d",keynr)); + + if ((wlist= _ma_ft_parserecord(info, keynr, record))) + { + error= _ma_ft_erase(info,keynr,keybuf,wlist,pos); + my_free((char*) wlist,MYF(0)); + } + DBUG_PRINT("exit",("Return: %d",error)); + DBUG_RETURN(error); +} + +uint _ma_ft_make_key(MARIA_HA *info, uint keynr, byte *keybuf, FT_WORD *wptr, + my_off_t filepos) +{ + byte buf[HA_FT_MAXBYTELEN+16]; + DBUG_ENTER("_ma_ft_make_key"); + +#if HA_FT_WTYPE == HA_KEYTYPE_FLOAT + { + float weight=(float) ((filepos==HA_OFFSET_ERROR) ? 0 : wptr->weight); + mi_float4store(buf,weight); + } +#else +#error +#endif + + int2store(buf+HA_FT_WLEN,wptr->len); + memcpy(buf+HA_FT_WLEN+2,wptr->pos,wptr->len); + DBUG_RETURN(_ma_make_key(info,keynr,(uchar*) keybuf,buf,filepos)); +} + + +/* + convert key value to ft2 +*/ + +uint _ma_ft_convert_to_ft2(MARIA_HA *info, uint keynr, uchar *key) +{ + my_off_t root; + DYNAMIC_ARRAY *da=info->ft1_to_ft2; + MARIA_KEYDEF *keyinfo=&info->s->ft2_keyinfo; + uchar *key_ptr= (uchar*) dynamic_array_ptr(da, 0), *end; + uint length, key_length; + DBUG_ENTER("_ma_ft_convert_to_ft2"); + + /* we'll generate one pageful at once, and insert the rest one-by-one */ + /* calculating the length of this page ...*/ + length=(keyinfo->block_length-2) / keyinfo->keylength; + set_if_smaller(length, da->elements); + length=length * keyinfo->keylength; + + get_key_full_length_rdonly(key_length, key); + while (_ma_ck_delete(info, keynr, key, key_length) == 0) + { + /* + nothing to do here. + _ma_ck_delete() will populate info->ft1_to_ft2 with deleted keys + */ + } + + /* creating pageful of keys */ + maria_putint(info->buff,length+2,0); + memcpy(info->buff+2, key_ptr, length); + info->buff_used=info->page_changed=1; /* info->buff is used */ + if ((root= _ma_new(info,keyinfo,DFLT_INIT_HITS)) == HA_OFFSET_ERROR || + _ma_write_keypage(info,keyinfo,root,DFLT_INIT_HITS,info->buff)) + DBUG_RETURN(-1); + + /* inserting the rest of key values */ + end= (uchar*) dynamic_array_ptr(da, da->elements); + for (key_ptr+=length; key_ptr < end; key_ptr+=keyinfo->keylength) + if(_ma_ck_real_write_btree(info, keyinfo, key_ptr, 0, &root, SEARCH_SAME)) + DBUG_RETURN(-1); + + /* now, writing the word key entry */ + ft_intXstore(key+key_length, - (int) da->elements); + _ma_dpointer(info, key+key_length+HA_FT_WLEN, root); + + DBUG_RETURN(_ma_ck_real_write_btree(info, + info->s->keyinfo+keynr, + key, 0, + &info->s->state.key_root[keynr], + SEARCH_SAME)); +} diff --git a/storage/maria/ma_ftdefs.h b/storage/maria/ma_ftdefs.h new file mode 100644 index 00000000000..41248d1bc9c --- /dev/null +++ b/storage/maria/ma_ftdefs.h @@ -0,0 +1,149 @@ +/* Copyright (C) 2006 MySQL AB & MySQL Finland AB & TCX DataKonsult AB + + 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; either version 2 of the License, or + (at your option) any later version. + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ + +/* Written by Sergei A. Golubchik, who has a shared copyright to this code */ + +/* some definitions for full-text indices */ + +#include "ma_fulltext.h" +#include +#include +#include +#include + +#define true_word_char(s,X) (my_isalnum(s,X) || (X)=='_') +#define misc_word_char(X) 0 +#define word_char(s,X) (true_word_char(s,X) || misc_word_char(X)) + +#define FT_MAX_WORD_LEN_FOR_SORT 31 + +#define COMPILE_STOPWORDS_IN + +/* Interested readers may consult SMART + (ftp://ftp.cs.cornell.edu/pub/smart/smart.11.0.tar.Z) + for an excellent implementation of vector space model we use. + It also demonstrate the usage of different weghting techniques. + This code, though, is completely original and is not based on the + SMART code but was in some cases inspired by it. + + NORM_PIVOT was taken from the article + A.Singhal, C.Buckley, M.Mitra, "Pivoted Document Length Normalization", + ACM SIGIR'96, 21-29, 1996 + */ + +#define LWS_FOR_QUERY LWS_TF +#define LWS_IN_USE LWS_LOG +#define PRENORM_IN_USE PRENORM_AVG +#define NORM_IN_USE NORM_PIVOT +#define GWS_IN_USE GWS_PROB +/*==============================================================*/ +#define LWS_TF (count) +#define LWS_BINARY (count>0) +#define LWS_SQUARE (count*count) +#define LWS_LOG (count?(log( (double) count)+1):0) +/*--------------------------------------------------------------*/ +#define PRENORM_NONE (p->weight) +#define PRENORM_MAX (p->weight/docstat.max) +#define PRENORM_AUG (0.4+0.6*p->weight/docstat.max) +#define PRENORM_AVG (p->weight/docstat.sum*docstat.uniq) +#define PRENORM_AVGLOG ((1+log(p->weight))/(1+log(docstat.sum/docstat.uniq))) +/*--------------------------------------------------------------*/ +#define NORM_NONE (1) +#define NORM_SUM (docstat.nsum) +#define NORM_COS (sqrt(docstat.nsum2)) + +#define PIVOT_VAL (0.0115) +#define NORM_PIVOT (1+PIVOT_VAL*docstat.uniq) +/*---------------------------------------------------------------*/ +#define GWS_NORM (1/sqrt(sum2)) +#define GWS_GFIDF (sum/doc_cnt) +/* Mysterious, but w/o (double) GWS_IDF performs better :-o */ +#define GWS_IDF log(aio->info->state->records/doc_cnt) +#define GWS_IDF1 log((double)aio->info->state->records/doc_cnt) +#define GWS_PROB ((aio->info->state->records > doc_cnt) ? log(((double)(aio->info->state->records-doc_cnt))/doc_cnt) : 0 ) +#define GWS_FREQ (1.0/doc_cnt) +#define GWS_SQUARED pow(log((double)aio->info->state->records/doc_cnt),2) +#define GWS_CUBIC pow(log((double)aio->info->state->records/doc_cnt),3) +#define GWS_ENTROPY (1-(suml/sum-log(sum))/log(aio->info->state->records)) +/*=================================================================*/ + +/* Boolean search operators */ +#define FTB_YES (ft_boolean_syntax[0]) +#define FTB_EGAL (ft_boolean_syntax[1]) +#define FTB_NO (ft_boolean_syntax[2]) +#define FTB_INC (ft_boolean_syntax[3]) +#define FTB_DEC (ft_boolean_syntax[4]) +#define FTB_LBR (ft_boolean_syntax[5]) +#define FTB_RBR (ft_boolean_syntax[6]) +#define FTB_NEG (ft_boolean_syntax[7]) +#define FTB_TRUNC (ft_boolean_syntax[8]) +#define FTB_LQUOT (ft_boolean_syntax[10]) +#define FTB_RQUOT (ft_boolean_syntax[11]) + +typedef struct st_maria_ft_word { + byte * pos; + uint len; + double weight; +} FT_WORD; + +int is_stopword(char *word, uint len); + +uint _ma_ft_make_key(MARIA_HA *, uint , byte *, FT_WORD *, my_off_t); + +byte maria_ft_get_word(CHARSET_INFO *, byte **, byte *, FT_WORD *, + MYSQL_FTPARSER_BOOLEAN_INFO *); +byte maria_ft_simple_get_word(CHARSET_INFO *, byte **, const byte *, + FT_WORD *, my_bool); + +typedef struct _st_maria_ft_seg_iterator { + uint num, len; + HA_KEYSEG *seg; + const byte *rec, *pos; +} FT_SEG_ITERATOR; + +void _ma_ft_segiterator_init(MARIA_HA *, uint, const byte *, FT_SEG_ITERATOR *); +void _ma_ft_segiterator_dummy_init(const byte *, uint, FT_SEG_ITERATOR *); +uint _ma_ft_segiterator(FT_SEG_ITERATOR *); + +void maria_ft_parse_init(TREE *, CHARSET_INFO *); +int maria_ft_parse(TREE *, byte *, int, my_bool, struct st_mysql_ftparser *parser, + MYSQL_FTPARSER_PARAM *param); +FT_WORD * maria_ft_linearize(TREE *); +FT_WORD * _ma_ft_parserecord(MARIA_HA *, uint, const byte *); +uint _ma_ft_parse(TREE *, MARIA_HA *, uint, const byte *, my_bool, + MYSQL_FTPARSER_PARAM *param); + +FT_INFO *maria_ft_init_nlq_search(MARIA_HA *, uint, byte *, uint, uint, byte *); +FT_INFO *maria_ft_init_boolean_search(MARIA_HA *, uint, byte *, uint, CHARSET_INFO *); + +extern const struct _ft_vft _ma_ft_vft_nlq; +int maria_ft_nlq_read_next(FT_INFO *, char *); +float maria_ft_nlq_find_relevance(FT_INFO *, byte *, uint); +void maria_ft_nlq_close_search(FT_INFO *); +float maria_ft_nlq_get_relevance(FT_INFO *); +my_off_t maria_ft_nlq_get_docid(FT_INFO *); +void maria_ft_nlq_reinit_search(FT_INFO *); + +extern const struct _ft_vft _ma_ft_vft_boolean; +int maria_ft_boolean_read_next(FT_INFO *, char *); +float maria_ft_boolean_find_relevance(FT_INFO *, byte *, uint); +void maria_ft_boolean_close_search(FT_INFO *); +float maria_ft_boolean_get_relevance(FT_INFO *); +my_off_t maria_ft_boolean_get_docid(FT_INFO *); +void maria_ft_boolean_reinit_search(FT_INFO *); +extern MYSQL_FTPARSER_PARAM *maria_ftparser_call_initializer(MARIA_HA *info, + uint keynr); +extern void maria_ftparser_call_deinitializer(MARIA_HA *info); diff --git a/storage/maria/ma_fulltext.h b/storage/maria/ma_fulltext.h new file mode 100644 index 00000000000..946a5628175 --- /dev/null +++ b/storage/maria/ma_fulltext.h @@ -0,0 +1,28 @@ +/* Copyright (C) 2006 MySQL AB & MySQL Finland AB & TCX DataKonsult AB + + 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; either version 2 of the License, or + (at your option) any later version. + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ + +/* Written by Sergei A. Golubchik, who has a shared copyright to this code */ + +/* some definitions for full-text indices */ + +#include "maria_def.h" +#include "ft_global.h" + +int _ma_ft_cmp(MARIA_HA *, uint, const byte *, const byte *); +int _ma_ft_add(MARIA_HA *, uint, byte *, const byte *, my_off_t); +int _ma_ft_del(MARIA_HA *, uint, byte *, const byte *, my_off_t); + +uint _ma_ft_convert_to_ft2(MARIA_HA *, uint, uchar *); diff --git a/storage/maria/ma_info.c b/storage/maria/ma_info.c new file mode 100644 index 00000000000..b22ffa41833 --- /dev/null +++ b/storage/maria/ma_info.c @@ -0,0 +1,133 @@ +/* Copyright (C) 2006 MySQL AB & MySQL Finland AB & TCX DataKonsult AB + + 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; either version 2 of the License, or + (at your option) any later version. + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ + +/* Return useful base information for an open table */ + +#include "maria_def.h" +#ifdef __WIN__ +#include +#endif + + /* Get position to last record */ + +my_off_t maria_position(MARIA_HA *info) +{ + return info->lastpos; +} + + +/* Get information about the table */ +/* if flag == 2 one get current info (no sync from database */ + +int maria_status(MARIA_HA *info, register MARIA_INFO *x, uint flag) +{ + MY_STAT state; + MARIA_SHARE *share=info->s; + DBUG_ENTER("maria_status"); + + x->recpos = info->lastpos; + if (flag == HA_STATUS_POS) + DBUG_RETURN(0); /* Compatible with ISAM */ + if (!(flag & HA_STATUS_NO_LOCK)) + { + pthread_mutex_lock(&share->intern_lock); + VOID(_ma_readinfo(info,F_RDLCK,0)); + fast_ma_writeinfo(info); + pthread_mutex_unlock(&share->intern_lock); + } + if (flag & HA_STATUS_VARIABLE) + { + x->records = info->state->records; + x->deleted = info->state->del; + x->delete_length = info->state->empty; + x->data_file_length =info->state->data_file_length; + x->index_file_length=info->state->key_file_length; + + x->keys = share->state.header.keys; + x->check_time = share->state.check_time; + x->mean_reclength = info->state->records ? + (ulong) ((info->state->data_file_length-info->state->empty)/ + info->state->records) : (ulong) share->min_pack_length; + } + if (flag & HA_STATUS_ERRKEY) + { + x->errkey = info->errkey; + x->dupp_key_pos= info->dupp_key_pos; + } + if (flag & HA_STATUS_CONST) + { + x->reclength = share->base.reclength; + x->max_data_file_length=share->base.max_data_file_length; + x->max_index_file_length=info->s->base.max_key_file_length; + x->filenr = info->dfile; + x->options = share->options; + x->create_time=share->state.create_time; + x->reflength= maria_get_pointer_length(share->base.max_data_file_length, + maria_data_pointer_size); + x->record_offset= ((share->options & + (HA_OPTION_PACK_RECORD | HA_OPTION_COMPRESS_RECORD)) ? + 0L : share->base.pack_reclength); + x->sortkey= -1; /* No clustering */ + x->rec_per_key = share->state.rec_per_key_part; + x->key_map = share->state.key_map; + x->data_file_name = share->data_file_name; + x->index_file_name = share->index_file_name; + } + if ((flag & HA_STATUS_TIME) && !my_fstat(info->dfile,&state,MYF(0))) + x->update_time=state.st_mtime; + else + x->update_time=0; + if (flag & HA_STATUS_AUTO) + { + x->auto_increment= share->state.auto_increment+1; + if (!x->auto_increment) /* This shouldn't happen */ + x->auto_increment= ~(ulonglong) 0; + } + DBUG_RETURN(0); +} + + +/* + Write a message to the error log. + + SYNOPSIS + _ma_report_error() + file_name Name of table file (e.g. index_file_name). + errcode Error number. + + DESCRIPTION + This function supplies my_error() with a table name. Most error + messages need one. Since string arguments in error messages are limited + to 64 characters by convention, we ensure that in case of truncation, + that the end of the index file path is in the message. This contains + the most valuable information (the table name and the database name). + + RETURN + void +*/ + +void _ma_report_error(int errcode, const char *file_name) +{ + size_t lgt; + DBUG_ENTER("_ma_report_error"); + DBUG_PRINT("enter",("errcode %d, table '%s'", errcode, file_name)); + + if ((lgt= strlen(file_name)) > 64) + file_name+= lgt - 64; + my_error(errcode, MYF(ME_NOREFRESH), file_name); + DBUG_VOID_RETURN; +} + diff --git a/storage/maria/ma_init.c b/storage/maria/ma_init.c new file mode 100644 index 00000000000..fc526c6ca3a --- /dev/null +++ b/storage/maria/ma_init.c @@ -0,0 +1,59 @@ +/* Copyright (C) 2006 MySQL AB & MySQL Finland AB & TCX DataKonsult AB + + 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; either version 2 of the License, or + (at your option) any later version. + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ + +/* Initialize an maria-database */ + +#include "maria_def.h" +#include + +static int maria_inited= 0; +pthread_mutex_t THR_LOCK_maria; + +/* + Initialize maria + + SYNOPSIS + maria_init() + + TODO + Open log files and do recovery if need + + RETURN + 0 ok + # error number +*/ + +int maria_init(void) +{ + if (!maria_inited) + { + maria_inited= 1; + pthread_mutex_init(&THR_LOCK_maria,MY_MUTEX_INIT_SLOW); + } + return 0; +} + + +void maria_end(void) +{ + if (maria_inited) + { + maria_inited= 0; + VOID(maria_logging(0)); /* Close log if neaded */ + ft_free_stopwords(); + pthread_mutex_destroy(&THR_LOCK_maria); + } +} diff --git a/storage/maria/ma_key.c b/storage/maria/ma_key.c new file mode 100644 index 00000000000..6a8c647aa7f --- /dev/null +++ b/storage/maria/ma_key.c @@ -0,0 +1,592 @@ +/* Copyright (C) 2006 MySQL AB & MySQL Finland AB & TCX DataKonsult AB + + 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; either version 2 of the License, or + (at your option) any later version. + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ + +/* Functions to handle keys */ + +#include "maria_def.h" +#include "m_ctype.h" +#include "ma_sp_defs.h" +#ifdef HAVE_IEEEFP_H +#include +#endif + +#define CHECK_KEYS /* Enable safety checks */ + +#define FIX_LENGTH(cs, pos, length, char_length) \ + do { \ + if (length > char_length) \ + char_length= my_charpos(cs, pos, pos+length, char_length); \ + set_if_smaller(char_length,length); \ + } while(0) + +static int _ma_put_key_in_record(MARIA_HA *info,uint keynr,byte *record); + +/* + Make a intern key from a record + + SYNOPSIS + _ma_make_key() + info MyiSAM handler + keynr key number + key Store created key here + record Record + filepos Position to record in the data file + + RETURN + Length of key +*/ + +uint _ma_make_key(register MARIA_HA *info, uint keynr, uchar *key, + const byte *record, my_off_t filepos) +{ + byte *pos,*end; + uchar *start; + reg1 HA_KEYSEG *keyseg; + my_bool is_ft= info->s->keyinfo[keynr].flag & HA_FULLTEXT; + DBUG_ENTER("_ma_make_key"); + + if (info->s->keyinfo[keynr].flag & HA_SPATIAL) + { + /* + TODO: nulls processing + */ +#ifdef HAVE_SPATIAL + DBUG_RETURN(sp_make_key(info,keynr,key,record,filepos)); +#else + DBUG_ASSERT(0); /* maria_open should check that this never happens*/ +#endif + } + + start=key; + for (keyseg=info->s->keyinfo[keynr].seg ; keyseg->type ;keyseg++) + { + enum ha_base_keytype type=(enum ha_base_keytype) keyseg->type; + uint length=keyseg->length; + uint char_length; + CHARSET_INFO *cs=keyseg->charset; + + if (keyseg->null_bit) + { + if (record[keyseg->null_pos] & keyseg->null_bit) + { + *key++= 0; /* NULL in key */ + continue; + } + *key++=1; /* Not NULL */ + } + + char_length= ((!is_ft && cs && cs->mbmaxlen > 1) ? length/cs->mbmaxlen : + length); + + pos= (byte*) record+keyseg->start; + if (type == HA_KEYTYPE_BIT) + { + if (keyseg->bit_length) + { + uchar bits= get_rec_bits((uchar*) record + keyseg->bit_pos, + keyseg->bit_start, keyseg->bit_length); + *key++= bits; + length--; + } + memcpy((byte*) key, pos, length); + key+= length; + continue; + } + if (keyseg->flag & HA_SPACE_PACK) + { + end= pos + length; + if (type != HA_KEYTYPE_NUM) + { + while (end > pos && end[-1] == ' ') + end--; + } + else + { + while (pos < end && pos[0] == ' ') + pos++; + } + length=(uint) (end-pos); + FIX_LENGTH(cs, pos, length, char_length); + store_key_length_inc(key,char_length); + memcpy((byte*) key,(byte*) pos,(size_t) char_length); + key+=char_length; + continue; + } + if (keyseg->flag & HA_VAR_LENGTH_PART) + { + uint pack_length= keyseg->bit_start; + uint tmp_length= (pack_length == 1 ? (uint) *(uchar*) pos : + uint2korr(pos)); + pos+= pack_length; /* Skip VARCHAR length */ + set_if_smaller(length,tmp_length); + FIX_LENGTH(cs, pos, length, char_length); + store_key_length_inc(key,char_length); + memcpy((byte*) key,(byte*) pos,(size_t) char_length); + key+= char_length; + continue; + } + else if (keyseg->flag & HA_BLOB_PART) + { + uint tmp_length= _ma_calc_blob_length(keyseg->bit_start,pos); + memcpy_fixed((byte*) &pos,pos+keyseg->bit_start,sizeof(char*)); + set_if_smaller(length,tmp_length); + FIX_LENGTH(cs, pos, length, char_length); + store_key_length_inc(key,char_length); + memcpy((byte*) key,(byte*) pos,(size_t) char_length); + key+= char_length; + continue; + } + else if (keyseg->flag & HA_SWAP_KEY) + { /* Numerical column */ +#ifdef HAVE_ISNAN + if (type == HA_KEYTYPE_FLOAT) + { + float nr; + float4get(nr,pos); + if (isnan(nr)) + { + /* Replace NAN with zero */ + bzero(key,length); + key+=length; + continue; + } + } + else if (type == HA_KEYTYPE_DOUBLE) + { + double nr; + float8get(nr,pos); + if (isnan(nr)) + { + bzero(key,length); + key+=length; + continue; + } + } +#endif + pos+=length; + while (length--) + { + *key++ = *--pos; + } + continue; + } + FIX_LENGTH(cs, pos, length, char_length); + memcpy((byte*) key, pos, char_length); + if (length > char_length) + cs->cset->fill(cs, (char*) key+char_length, length-char_length, ' '); + key+= length; + } + _ma_dpointer(info,key,filepos); + DBUG_PRINT("exit",("keynr: %d",keynr)); + DBUG_DUMP("key",(byte*) start,(uint) (key-start)+keyseg->length); + DBUG_EXECUTE("key", + _ma_print_key(DBUG_FILE,info->s->keyinfo[keynr].seg,start, + (uint) (key-start));); + DBUG_RETURN((uint) (key-start)); /* Return keylength */ +} /* _ma_make_key */ + + +/* + Pack a key to intern format from given format (c_rkey) + + SYNOPSIS + _ma_pack_key() + info MARIA handler + uint keynr key number + key Store packed key here + old Not packed key + k_length Length of 'old' to use + last_used_keyseg out parameter. May be NULL + + RETURN + length of packed key + + last_use_keyseg Store pointer to the keyseg after the last used one +*/ + +uint _ma_pack_key(register MARIA_HA *info, uint keynr, uchar *key, uchar *old, + uint k_length, HA_KEYSEG **last_used_keyseg) +{ + uchar *start_key=key; + HA_KEYSEG *keyseg; + my_bool is_ft= info->s->keyinfo[keynr].flag & HA_FULLTEXT; + DBUG_ENTER("_ma_pack_key"); + + for (keyseg=info->s->keyinfo[keynr].seg ; + keyseg->type && (int) k_length > 0; + old+=keyseg->length, keyseg++) + { + enum ha_base_keytype type=(enum ha_base_keytype) keyseg->type; + uint length=min((uint) keyseg->length,(uint) k_length); + uint char_length; + uchar *pos; + CHARSET_INFO *cs=keyseg->charset; + + if (keyseg->null_bit) + { + k_length--; + if (!(*key++= (char) 1-*old++)) /* Copy null marker */ + { + k_length-=length; + if (keyseg->flag & (HA_VAR_LENGTH_PART | HA_BLOB_PART)) + { + k_length-=2; /* Skip length */ + old+= 2; + } + continue; /* Found NULL */ + } + } + char_length= (!is_ft && cs && cs->mbmaxlen > 1) ? length/cs->mbmaxlen : length; + pos=old; + if (keyseg->flag & HA_SPACE_PACK) + { + uchar *end=pos+length; + if (type != HA_KEYTYPE_NUM) + { + while (end > pos && end[-1] == ' ') + end--; + } + else + { + while (pos < end && pos[0] == ' ') + pos++; + } + k_length-=length; + length=(uint) (end-pos); + FIX_LENGTH(cs, pos, length, char_length); + store_key_length_inc(key,char_length); + memcpy((byte*) key,pos,(size_t) char_length); + key+= char_length; + continue; + } + else if (keyseg->flag & (HA_VAR_LENGTH_PART | HA_BLOB_PART)) + { + /* Length of key-part used with maria_rkey() always 2 */ + uint tmp_length=uint2korr(pos); + k_length-= 2+length; + pos+=2; + set_if_smaller(length,tmp_length); /* Safety */ + FIX_LENGTH(cs, pos, length, char_length); + store_key_length_inc(key,char_length); + old+=2; /* Skip length */ + memcpy((byte*) key, pos,(size_t) char_length); + key+= char_length; + continue; + } + else if (keyseg->flag & HA_SWAP_KEY) + { /* Numerical column */ + pos+=length; + k_length-=length; + while (length--) + { + *key++ = *--pos; + } + continue; + } + FIX_LENGTH(cs, pos, length, char_length); + memcpy((byte*) key, pos, char_length); + if (length > char_length) + cs->cset->fill(cs, (char*) key+char_length, length-char_length, ' '); + key+= length; + k_length-=length; + } + if (last_used_keyseg) + *last_used_keyseg= keyseg; + +#ifdef NOT_USED + if (keyseg->type) + { + /* Part-key ; fill with ASCII 0 for easier searching */ + length= (uint) -k_length; /* unused part of last key */ + do + { + if (keyseg->flag & HA_NULL_PART) + length++; + if (keyseg->flag & HA_SPACE_PACK) + length+=2; + else + length+= keyseg->length; + keyseg++; + } while (keyseg->type); + bzero((byte*) key,length); + key+=length; + } +#endif + DBUG_RETURN((uint) (key-start_key)); +} /* _ma_pack_key */ + + + +/* + Store found key in record + + SYNOPSIS + _ma_put_key_in_record() + info MARIA handler + keynr Key number that was used + record Store key here + + Last read key is in info->lastkey + + NOTES + Used when only-keyread is wanted + + RETURN + 0 ok + 1 error +*/ + +static int _ma_put_key_in_record(register MARIA_HA *info, uint keynr, + byte *record) +{ + reg2 byte *key; + byte *pos,*key_end; + reg1 HA_KEYSEG *keyseg; + byte *blob_ptr; + DBUG_ENTER("_ma_put_key_in_record"); + + blob_ptr= (byte*) info->lastkey2; /* Place to put blob parts */ + key=(byte*) info->lastkey; /* KEy that was read */ + key_end=key+info->lastkey_length; + for (keyseg=info->s->keyinfo[keynr].seg ; keyseg->type ;keyseg++) + { + if (keyseg->null_bit) + { + if (!*key++) + { + record[keyseg->null_pos]|= keyseg->null_bit; + continue; + } + record[keyseg->null_pos]&= ~keyseg->null_bit; + } + if (keyseg->type == HA_KEYTYPE_BIT) + { + uint length= keyseg->length; + + if (keyseg->bit_length) + { + uchar bits= *key++; + set_rec_bits(bits, record + keyseg->bit_pos, keyseg->bit_start, + keyseg->bit_length); + length--; + } + else + { + clr_rec_bits(record + keyseg->bit_pos, keyseg->bit_start, + keyseg->bit_length); + } + memcpy(record + keyseg->start, (byte*) key, length); + key+= length; + continue; + } + if (keyseg->flag & HA_SPACE_PACK) + { + uint length; + get_key_length(length,key); +#ifdef CHECK_KEYS + if (length > keyseg->length || key+length > key_end) + goto err; +#endif + pos= record+keyseg->start; + if (keyseg->type != (int) HA_KEYTYPE_NUM) + { + memcpy(pos,key,(size_t) length); + bfill(pos+length,keyseg->length-length,' '); + } + else + { + bfill(pos,keyseg->length-length,' '); + memcpy(pos+keyseg->length-length,key,(size_t) length); + } + key+=length; + continue; + } + + if (keyseg->flag & HA_VAR_LENGTH_PART) + { + uint length; + get_key_length(length,key); +#ifdef CHECK_KEYS + if (length > keyseg->length || key+length > key_end) + goto err; +#endif + /* Store key length */ + if (keyseg->bit_start == 1) + *(uchar*) (record+keyseg->start)= (uchar) length; + else + int2store(record+keyseg->start, length); + /* And key data */ + memcpy(record+keyseg->start + keyseg->bit_start, (byte*) key, length); + key+= length; + } + else if (keyseg->flag & HA_BLOB_PART) + { + uint length; + get_key_length(length,key); +#ifdef CHECK_KEYS + if (length > keyseg->length || key+length > key_end) + goto err; +#endif + memcpy(record+keyseg->start+keyseg->bit_start, + (char*) &blob_ptr,sizeof(char*)); + memcpy(blob_ptr,key,length); + blob_ptr+=length; + + /* The above changed info->lastkey2. Inform maria_rnext_same(). */ + info->update&= ~HA_STATE_RNEXT_SAME; + + _ma_store_blob_length(record+keyseg->start, + (uint) keyseg->bit_start,length); + key+=length; + } + else if (keyseg->flag & HA_SWAP_KEY) + { + byte *to= record+keyseg->start+keyseg->length; + byte *end= key+keyseg->length; +#ifdef CHECK_KEYS + if (end > key_end) + goto err; +#endif + do + { + *--to= *key++; + } while (key != end); + continue; + } + else + { +#ifdef CHECK_KEYS + if (key+keyseg->length > key_end) + goto err; +#endif + memcpy(record+keyseg->start,(byte*) key, + (size_t) keyseg->length); + key+= keyseg->length; + } + } + DBUG_RETURN(0); + +err: + DBUG_RETURN(1); /* Crashed row */ +} /* _ma_put_key_in_record */ + + + /* Here when key reads are used */ + +int _ma_read_key_record(MARIA_HA *info, my_off_t filepos, byte *buf) +{ + fast_ma_writeinfo(info); + if (filepos != HA_OFFSET_ERROR) + { + if (info->lastinx >= 0) + { /* Read only key */ + if (_ma_put_key_in_record(info,(uint) info->lastinx,buf)) + { + maria_print_error(info->s, HA_ERR_CRASHED); + my_errno=HA_ERR_CRASHED; + return -1; + } + info->update|= HA_STATE_AKTIV; /* We should find a record */ + return 0; + } + my_errno=HA_ERR_WRONG_INDEX; + } + return(-1); /* Wrong data to read */ +} + + +/* + Update auto_increment info + + SYNOPSIS + _ma_update_auto_increment() + info MARIA handler + record Row to update + + IMPLEMENTATION + Only replace the auto_increment value if it is higher than the previous + one. For signed columns we don't update the auto increment value if it's + less than zero. +*/ + +void _ma_update_auto_increment(MARIA_HA *info,const byte *record) +{ + ulonglong value= 0; /* Store unsigned values here */ + longlong s_value= 0; /* Store signed values here */ + HA_KEYSEG *keyseg= info->s->keyinfo[info->s->base.auto_key-1].seg; + const uchar *key= (uchar*) record + keyseg->start; + + switch (keyseg->type) { + case HA_KEYTYPE_INT8: + s_value= (longlong) *(char*)key; + break; + case HA_KEYTYPE_BINARY: + value=(ulonglong) *(uchar*) key; + break; + case HA_KEYTYPE_SHORT_INT: + s_value= (longlong) sint2korr(key); + break; + case HA_KEYTYPE_USHORT_INT: + value=(ulonglong) uint2korr(key); + break; + case HA_KEYTYPE_LONG_INT: + s_value= (longlong) sint4korr(key); + break; + case HA_KEYTYPE_ULONG_INT: + value=(ulonglong) uint4korr(key); + break; + case HA_KEYTYPE_INT24: + s_value= (longlong) sint3korr(key); + break; + case HA_KEYTYPE_UINT24: + value=(ulonglong) uint3korr(key); + break; + case HA_KEYTYPE_FLOAT: /* This shouldn't be used */ + { + float f_1; + float4get(f_1,key); + /* Ignore negative values */ + value = (f_1 < (float) 0.0) ? 0 : (ulonglong) f_1; + break; + } + case HA_KEYTYPE_DOUBLE: /* This shouldn't be used */ + { + double f_1; + float8get(f_1,key); + /* Ignore negative values */ + value = (f_1 < 0.0) ? 0 : (ulonglong) f_1; + break; + } + case HA_KEYTYPE_LONGLONG: + s_value= sint8korr(key); + break; + case HA_KEYTYPE_ULONGLONG: + value= uint8korr(key); + break; + default: + DBUG_ASSERT(0); + value=0; /* Error */ + break; + } + + /* + The following code works becasue if s_value < 0 then value is 0 + and if s_value == 0 then value will contain either s_value or the + correct value. + */ + set_if_bigger(info->s->state.auto_increment, + (s_value > 0) ? (ulonglong) s_value : value); +} diff --git a/storage/maria/ma_keycache.c b/storage/maria/ma_keycache.c new file mode 100644 index 00000000000..837b0fbac66 --- /dev/null +++ b/storage/maria/ma_keycache.c @@ -0,0 +1,163 @@ +/* Copyright (C) 2006 MySQL AB + + 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; either version 2 of the License, or + (at your option) any later version. + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ + +/* + Key cache assignments +*/ + +#include "maria_def.h" + +/* + Assign pages of the index file for a table to a key cache + + SYNOPSIS + maria_assign_to_key_cache() + info open table + key_map map of indexes to assign to the key cache + key_cache_ptr pointer to the key cache handle + assign_lock Mutex to lock during assignment + + PREREQUESTS + One must have a READ lock or a WRITE lock on the table when calling + the function to ensure that there is no other writers to it. + + The caller must also ensure that one doesn't call this function from + two different threads with the same table. + + NOTES + At present pages for all indexes must be assigned to the same key cache. + In future only pages for indexes specified in the key_map parameter + of the table will be assigned to the specified key cache. + + RETURN VALUE + 0 If a success + # Error code +*/ + +int maria_assign_to_key_cache(MARIA_HA *info, + ulonglong key_map __attribute__((unused)), + KEY_CACHE *key_cache) +{ + int error= 0; + MARIA_SHARE* share= info->s; + DBUG_ENTER("maria_assign_to_key_cache"); + DBUG_PRINT("enter",("old_key_cache_handle: %lx new_key_cache_handle: %lx", + share->key_cache, key_cache)); + + /* + Skip operation if we didn't change key cache. This can happen if we + call this for all open instances of the same table + */ + if (share->key_cache == key_cache) + DBUG_RETURN(0); + + /* + First flush all blocks for the table in the old key cache. + This is to ensure that the disk is consistent with the data pages + in memory (which may not be the case if the table uses delayed_key_write) + + Note that some other read thread may still fill in the key cache with + new blocks during this call and after, but this doesn't matter as + all threads will start using the new key cache for their next call to + maria library and we know that there will not be any changed blocks + in the old key cache. + */ + + if (flush_key_blocks(share->key_cache, share->kfile, FLUSH_RELEASE)) + { + error= my_errno; + maria_print_error(info->s, HA_ERR_CRASHED); + maria_mark_crashed(info); /* Mark that table must be checked */ + } + + /* + Flush the new key cache for this file. This is needed to ensure + that there is no old blocks (with outdated data) left in the new key + cache from an earlier assign_to_keycache operation + + (This can never fail as there is never any not written data in the + new key cache) + */ + (void) flush_key_blocks(key_cache, share->kfile, FLUSH_RELEASE); + + /* + ensure that setting the key cache and changing the multi_key_cache + is done atomicly + */ + pthread_mutex_lock(&share->intern_lock); + /* + Tell all threads to use the new key cache + This should be seen at the lastes for the next call to an maria function. + */ + share->key_cache= key_cache; + + /* store the key cache in the global hash structure for future opens */ + if (multi_key_cache_set(share->unique_file_name, share->unique_name_length, + share->key_cache)) + error= my_errno; + pthread_mutex_unlock(&share->intern_lock); + DBUG_RETURN(error); +} + + +/* + Change all MARIA entries that uses one key cache to another key cache + + SYNOPSIS + maria_change_key_cache() + old_key_cache Old key cache + new_key_cache New key cache + + NOTES + This is used when we delete one key cache. + + To handle the case where some other threads tries to open an MARIA + table associated with the to-be-deleted key cache while this operation + is running, we have to call 'multi_key_cache_change()' from this + function while we have a lock on the MARIA table list structure. + + This is safe as long as it's only MARIA that is using this specific + key cache. +*/ + + +void maria_change_key_cache(KEY_CACHE *old_key_cache, + KEY_CACHE *new_key_cache) +{ + LIST *pos; + DBUG_ENTER("maria_change_key_cache"); + + /* + Lock list to ensure that no one can close the table while we manipulate it + */ + pthread_mutex_lock(&THR_LOCK_maria); + for (pos=maria_open_list ; pos ; pos=pos->next) + { + MARIA_HA *info= (MARIA_HA*) pos->data; + MARIA_SHARE *share= info->s; + if (share->key_cache == old_key_cache) + maria_assign_to_key_cache(info, (ulonglong) ~0, new_key_cache); + } + + /* + We have to do the following call while we have the lock on the + MARIA list structure to ensure that another thread is not trying to + open a new table that will be associted with the old key cache + */ + multi_key_cache_change(old_key_cache, new_key_cache); + pthread_mutex_unlock(&THR_LOCK_maria); + DBUG_VOID_RETURN; +} diff --git a/storage/maria/ma_locking.c b/storage/maria/ma_locking.c new file mode 100644 index 00000000000..697747021f8 --- /dev/null +++ b/storage/maria/ma_locking.c @@ -0,0 +1,554 @@ +/* Copyright (C) 2006 MySQL AB & MySQL Finland AB & TCX DataKonsult AB + + 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; either version 2 of the License, or + (at your option) any later version. + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ + +/* + locking of isam-tables. + reads info from a isam-table. Must be first request before doing any furter + calls to any isamfunktion. Is used to allow many process use the same + isamdatabase. +*/ + +#include "ma_ftdefs.h" + + /* lock table by F_UNLCK, F_RDLCK or F_WRLCK */ + +int maria_lock_database(MARIA_HA *info, int lock_type) +{ + int error; + uint count; + MARIA_SHARE *share=info->s; + uint flag; + DBUG_ENTER("maria_lock_database"); + DBUG_PRINT("enter",("lock_type: %d old lock %d r_locks: %u w_locks: %u " + "global_changed: %d open_count: %u name: '%s'", + lock_type, info->lock_type, share->r_locks, + share->w_locks, + share->global_changed, share->state.open_count, + share->index_file_name)); + if (share->options & HA_OPTION_READ_ONLY_DATA || + info->lock_type == lock_type) + DBUG_RETURN(0); + if (lock_type == F_EXTRA_LCK) /* Used by TMP tables */ + { + ++share->w_locks; + ++share->tot_locks; + info->lock_type= lock_type; + DBUG_RETURN(0); + } + + flag=error=0; + pthread_mutex_lock(&share->intern_lock); + if (share->kfile >= 0) /* May only be false on windows */ + { + switch (lock_type) { + case F_UNLCK: + maria_ftparser_call_deinitializer(info); + if (info->lock_type == F_RDLCK) + count= --share->r_locks; + else + count= --share->w_locks; + --share->tot_locks; + if (info->lock_type == F_WRLCK && !share->w_locks && + !share->delay_key_write && flush_key_blocks(share->key_cache, + share->kfile,FLUSH_KEEP)) + { + error=my_errno; + maria_print_error(info->s, HA_ERR_CRASHED); + maria_mark_crashed(info); /* Mark that table must be checked */ + } + if (info->opt_flag & (READ_CACHE_USED | WRITE_CACHE_USED)) + { + if (end_io_cache(&info->rec_cache)) + { + error=my_errno; + maria_print_error(info->s, HA_ERR_CRASHED); + maria_mark_crashed(info); + } + } + if (!count) + { + DBUG_PRINT("info",("changed: %u w_locks: %u", + (uint) share->changed, share->w_locks)); + if (share->changed && !share->w_locks) + { +#ifdef HAVE_MMAP + if ((info->s->mmaped_length != info->s->state.state.data_file_length) && + (info->s->nonmmaped_inserts > MAX_NONMAPPED_INSERTS)) + { + if (info->s->concurrent_insert) + rw_wrlock(&info->s->mmap_lock); + _ma_remap_file(info, info->s->state.state.data_file_length); + info->s->nonmmaped_inserts= 0; + if (info->s->concurrent_insert) + rw_unlock(&info->s->mmap_lock); + } +#endif + share->state.process= share->last_process=share->this_process; + share->state.unique= info->last_unique= info->this_unique; + share->state.update_count= info->last_loop= ++info->this_loop; + if (_ma_state_info_write(share->kfile, &share->state, 1)) + error=my_errno; + share->changed=0; + if (maria_flush) + { + if (my_sync(share->kfile, MYF(0))) + error= my_errno; + if (my_sync(info->dfile, MYF(0))) + error= my_errno; + } + else + share->not_flushed=1; + if (error) + { + maria_print_error(info->s, HA_ERR_CRASHED); + maria_mark_crashed(info); + } + } + if (info->lock_type != F_EXTRA_LCK) + { + if (share->r_locks) + { /* Only read locks left */ + flag=1; + if (my_lock(share->kfile,F_RDLCK,0L,F_TO_EOF, + MYF(MY_WME | MY_SEEK_NOT_DONE)) && !error) + error=my_errno; + } + else if (!share->w_locks) + { /* No more locks */ + flag=1; + if (my_lock(share->kfile,F_UNLCK,0L,F_TO_EOF, + MYF(MY_WME | MY_SEEK_NOT_DONE)) && !error) + error=my_errno; + } + } + } + info->opt_flag&= ~(READ_CACHE_USED | WRITE_CACHE_USED); + info->lock_type= F_UNLCK; + break; + case F_RDLCK: + if (info->lock_type == F_WRLCK) + { + /* + Change RW to READONLY + + mysqld does not turn write locks to read locks, + so we're never here in mysqld. + */ + if (share->w_locks == 1) + { + flag=1; + if (my_lock(share->kfile,lock_type,0L,F_TO_EOF, + MYF(MY_SEEK_NOT_DONE))) + { + error=my_errno; + break; + } + } + share->w_locks--; + share->r_locks++; + info->lock_type=lock_type; + break; + } + if (!share->r_locks && !share->w_locks) + { + flag=1; + if (my_lock(share->kfile,lock_type,0L,F_TO_EOF, + info->lock_wait | MY_SEEK_NOT_DONE)) + { + error=my_errno; + break; + } + if (_ma_state_info_read_dsk(share->kfile, &share->state, 1)) + { + error=my_errno; + VOID(my_lock(share->kfile,F_UNLCK,0L,F_TO_EOF,MYF(MY_SEEK_NOT_DONE))); + my_errno=error; + break; + } + } + VOID(_ma_test_if_changed(info)); + share->r_locks++; + share->tot_locks++; + info->lock_type=lock_type; + break; + case F_WRLCK: + if (info->lock_type == F_RDLCK) + { /* Change READONLY to RW */ + if (share->r_locks == 1) + { + flag=1; + if (my_lock(share->kfile,lock_type,0L,F_TO_EOF, + MYF(info->lock_wait | MY_SEEK_NOT_DONE))) + { + error=my_errno; + break; + } + share->r_locks--; + share->w_locks++; + info->lock_type=lock_type; + break; + } + } + if (!(share->options & HA_OPTION_READ_ONLY_DATA)) + { + if (!share->w_locks) + { + flag=1; + if (my_lock(share->kfile,lock_type,0L,F_TO_EOF, + info->lock_wait | MY_SEEK_NOT_DONE)) + { + error=my_errno; + break; + } + if (!share->r_locks) + { + if (_ma_state_info_read_dsk(share->kfile, &share->state, 1)) + { + error=my_errno; + VOID(my_lock(share->kfile,F_UNLCK,0L,F_TO_EOF, + info->lock_wait | MY_SEEK_NOT_DONE)); + my_errno=error; + break; + } + } + } + } + VOID(_ma_test_if_changed(info)); + + info->lock_type=lock_type; + info->invalidator=info->s->invalidator; + share->w_locks++; + share->tot_locks++; + break; + default: + break; /* Impossible */ + } + } + pthread_mutex_unlock(&share->intern_lock); +#if defined(FULL_LOG) || defined(_lint) + lock_type|=(int) (flag << 8); /* Set bit to set if real lock */ + maria_log_command(MARIA_LOG_LOCK,info,(byte*) &lock_type,sizeof(lock_type), + error); +#endif + DBUG_RETURN(error); +} /* maria_lock_database */ + + +/**************************************************************************** + The following functions are called by thr_lock() in threaded applications +****************************************************************************/ + +/* + Create a copy of the current status for the table + + SYNOPSIS + _ma_get_status() + param Pointer to Myisam handler + concurrent_insert Set to 1 if we are going to do concurrent inserts + (THR_WRITE_CONCURRENT_INSERT was used) +*/ + +void _ma_get_status(void* param, int concurrent_insert) +{ + MARIA_HA *info=(MARIA_HA*) param; + DBUG_ENTER("_ma_get_status"); + DBUG_PRINT("info",("key_file: %ld data_file: %ld concurrent_insert: %d", + (long) info->s->state.state.key_file_length, + (long) info->s->state.state.data_file_length, + concurrent_insert)); +#ifndef DBUG_OFF + if (info->state->key_file_length > info->s->state.state.key_file_length || + info->state->data_file_length > info->s->state.state.data_file_length) + DBUG_PRINT("warning",("old info: key_file: %ld data_file: %ld", + (long) info->state->key_file_length, + (long) info->state->data_file_length)); +#endif + info->save_state=info->s->state.state; + info->state= &info->save_state; + info->append_insert_at_end= concurrent_insert; + DBUG_VOID_RETURN; +} + + +void _ma_update_status(void* param) +{ + MARIA_HA *info=(MARIA_HA*) param; + /* + Because someone may have closed the table we point at, we only + update the state if its our own state. This isn't a problem as + we are always pointing at our own lock or at a read lock. + (This is enforced by thr_multi_lock.c) + */ + if (info->state == &info->save_state) + { +#ifndef DBUG_OFF + DBUG_PRINT("info",("updating status: key_file: %ld data_file: %ld", + (long) info->state->key_file_length, + (long) info->state->data_file_length)); + if (info->state->key_file_length < info->s->state.state.key_file_length || + info->state->data_file_length < info->s->state.state.data_file_length) + DBUG_PRINT("warning",("old info: key_file: %ld data_file: %ld", + (long) info->s->state.state.key_file_length, + (long) info->s->state.state.data_file_length)); +#endif + info->s->state.state= *info->state; + info->state= &info->s->state.state; + } + info->append_insert_at_end= 0; + + /* + We have to flush the write cache here as other threads may start + reading the table before maria_lock_database() is called + */ + if (info->opt_flag & WRITE_CACHE_USED) + { + if (end_io_cache(&info->rec_cache)) + { + maria_print_error(info->s, HA_ERR_CRASHED); + maria_mark_crashed(info); + } + info->opt_flag&= ~WRITE_CACHE_USED; + } +} + +void _ma_copy_status(void* to,void *from) +{ + ((MARIA_HA*) to)->state= &((MARIA_HA*) from)->save_state; +} + + +/* + Check if should allow concurrent inserts + + IMPLEMENTATION + Allow concurrent inserts if we don't have a hole in the table or + if there is no active write lock and there is active read locks and + maria_concurrent_insert == 2. In this last case the new + row('s) are inserted at end of file instead of filling up the hole. + + The last case is to allow one to inserts into a heavily read-used table + even if there is holes. + + NOTES + If there is a an rtree indexes in the table, concurrent inserts are + disabled in maria_open() + + RETURN + 0 ok to use concurrent inserts + 1 not ok +*/ + +my_bool _ma_check_status(void *param) +{ + MARIA_HA *info=(MARIA_HA*) param; + /* + The test for w_locks == 1 is here because this thread has already done an + external lock (in other words: w_locks == 1 means no other threads has + a write lock) + */ + DBUG_PRINT("info",("dellink: %ld r_locks: %u w_locks: %u", + (long) info->s->state.dellink, (uint) info->s->r_locks, + (uint) info->s->w_locks)); + return (my_bool) !(info->s->state.dellink == HA_OFFSET_ERROR || + (maria_concurrent_insert == 2 && info->s->r_locks && + info->s->w_locks == 1)); +} + + +/**************************************************************************** + ** functions to read / write the state +****************************************************************************/ + +int _ma_readinfo(register MARIA_HA *info, int lock_type, int check_keybuffer) +{ + DBUG_ENTER("_ma_readinfo"); + + if (info->lock_type == F_UNLCK) + { + MARIA_SHARE *share=info->s; + if (!share->tot_locks) + { + if (my_lock(share->kfile,lock_type,0L,F_TO_EOF, + info->lock_wait | MY_SEEK_NOT_DONE)) + DBUG_RETURN(1); + if (_ma_state_info_read_dsk(share->kfile, &share->state, 1)) + { + int error=my_errno ? my_errno : -1; + VOID(my_lock(share->kfile,F_UNLCK,0L,F_TO_EOF, + MYF(MY_SEEK_NOT_DONE))); + my_errno=error; + DBUG_RETURN(1); + } + } + if (check_keybuffer) + VOID(_ma_test_if_changed(info)); + info->invalidator=info->s->invalidator; + } + else if (lock_type == F_WRLCK && info->lock_type == F_RDLCK) + { + my_errno=EACCES; /* Not allowed to change */ + DBUG_RETURN(-1); /* when have read_lock() */ + } + DBUG_RETURN(0); +} /* _ma_readinfo */ + + +/* + Every isam-function that uppdates the isam-database MUST end with this + request +*/ + +int _ma_writeinfo(register MARIA_HA *info, uint operation) +{ + int error,olderror; + MARIA_SHARE *share=info->s; + DBUG_ENTER("_ma_writeinfo"); + DBUG_PRINT("info",("operation: %u tot_locks: %u", operation, + share->tot_locks)); + + error=0; + if (share->tot_locks == 0) + { + olderror=my_errno; /* Remember last error */ + if (operation) + { /* Two threads can't be here */ + share->state.process= share->last_process= share->this_process; + share->state.unique= info->last_unique= info->this_unique; + share->state.update_count= info->last_loop= ++info->this_loop; + if ((error=_ma_state_info_write(share->kfile, &share->state, 1))) + olderror=my_errno; +#ifdef __WIN__ + if (maria_flush) + { + _commit(share->kfile); + _commit(info->dfile); + } +#endif + } + if (!(operation & WRITEINFO_NO_UNLOCK) && + my_lock(share->kfile,F_UNLCK,0L,F_TO_EOF, + MYF(MY_WME | MY_SEEK_NOT_DONE)) && !error) + DBUG_RETURN(1); + my_errno=olderror; + } + else if (operation) + share->changed= 1; /* Mark keyfile changed */ + DBUG_RETURN(error); +} /* _ma_writeinfo */ + + + /* Test if someone has changed the database */ + /* (Should be called after readinfo) */ + +int _ma_test_if_changed(register MARIA_HA *info) +{ + MARIA_SHARE *share=info->s; + if (share->state.process != share->last_process || + share->state.unique != info->last_unique || + share->state.update_count != info->last_loop) + { /* Keyfile has changed */ + DBUG_PRINT("info",("index file changed")); + if (share->state.process != share->this_process) + VOID(flush_key_blocks(share->key_cache, share->kfile, FLUSH_RELEASE)); + share->last_process=share->state.process; + info->last_unique= share->state.unique; + info->last_loop= share->state.update_count; + info->update|= HA_STATE_WRITTEN; /* Must use file on next */ + info->data_changed= 1; /* For maria_is_changed */ + return 1; + } + return (!(info->update & HA_STATE_AKTIV) || + (info->update & (HA_STATE_WRITTEN | HA_STATE_DELETED | + HA_STATE_KEY_CHANGED))); +} /* _ma_test_if_changed */ + + +/* + Put a mark in the .MYI file that someone is updating the table + + + DOCUMENTATION + + state.open_count in the .MYI file is used the following way: + - For the first change of the .MYI file in this process open_count is + incremented by maria_mark_file_change(). (We have a write lock on the file + when this happens) + - In maria_close() it's decremented by _ma_decrement_open_count() if it + was incremented in the same process. + + This mean that if we are the only process using the file, the open_count + tells us if the MARIA file wasn't properly closed. (This is true if + my_disable_locking is set). +*/ + + +int _ma_mark_file_changed(MARIA_HA *info) +{ + char buff[3]; + register MARIA_SHARE *share=info->s; + DBUG_ENTER("_ma_mark_file_changed"); + + if (!(share->state.changed & STATE_CHANGED) || ! share->global_changed) + { + share->state.changed|=(STATE_CHANGED | STATE_NOT_ANALYZED | + STATE_NOT_OPTIMIZED_KEYS); + if (!share->global_changed) + { + share->global_changed=1; + share->state.open_count++; + } + if (!share->temporary) + { + mi_int2store(buff,share->state.open_count); + buff[2]=1; /* Mark that it's changed */ + DBUG_RETURN(my_pwrite(share->kfile,buff,sizeof(buff), + sizeof(share->state.header), + MYF(MY_NABP))); + } + } + DBUG_RETURN(0); +} + + +/* + This is only called by close or by extra(HA_FLUSH) if the OS has the pwrite() + call. In these context the following code should be safe! + */ + +int _ma_decrement_open_count(MARIA_HA *info) +{ + char buff[2]; + register MARIA_SHARE *share=info->s; + int lock_error=0,write_error=0; + if (share->global_changed) + { + uint old_lock=info->lock_type; + share->global_changed=0; + lock_error=maria_lock_database(info,F_WRLCK); + /* Its not fatal even if we couldn't get the lock ! */ + if (share->state.open_count > 0) + { + share->state.open_count--; + mi_int2store(buff,share->state.open_count); + write_error=my_pwrite(share->kfile,buff,sizeof(buff), + sizeof(share->state.header), + MYF(MY_NABP)); + } + if (!lock_error) + lock_error=maria_lock_database(info,old_lock); + } + return test(lock_error || write_error); +} diff --git a/storage/maria/ma_log.c b/storage/maria/ma_log.c new file mode 100644 index 00000000000..7c32c1068cb --- /dev/null +++ b/storage/maria/ma_log.c @@ -0,0 +1,164 @@ +/* Copyright (C) 2006 MySQL AB & MySQL Finland AB & TCX DataKonsult AB + + 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; either version 2 of the License, or + (at your option) any later version. + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ + +/* + Logging of MARIA commands and records on logfile for debugging + The log can be examined with help of the marialog command. +*/ + +#include "maria_def.h" +#if defined(MSDOS) || defined(__WIN__) +#include +#ifndef __WIN__ +#include +#endif +#endif +#ifdef VMS +#include +#endif + +#undef GETPID /* For HPUX */ +#ifdef THREAD +#define GETPID() (log_type == 1 ? (long) maria_pid : (long) my_thread_id()); +#else +#define GETPID() maria_pid +#endif + + /* Activate logging if flag is 1 and reset logging if flag is 0 */ + +static int log_type=0; +ulong maria_pid=0; + +int maria_logging(int activate_log) +{ + int error=0; + char buff[FN_REFLEN]; + DBUG_ENTER("maria_logging"); + + log_type=activate_log; + if (activate_log) + { + if (!maria_pid) + maria_pid=(ulong) getpid(); + if (maria_log_file < 0) + { + if ((maria_log_file = my_create(fn_format(buff,maria_log_filename, + "",".log",4), + 0,(O_RDWR | O_BINARY | O_APPEND),MYF(0))) + < 0) + DBUG_RETURN(my_errno); + } + } + else if (maria_log_file >= 0) + { + error=my_close(maria_log_file,MYF(0)) ? my_errno : 0 ; + maria_log_file= -1; + } + DBUG_RETURN(error); +} + + + /* Logging of records and commands on logfile */ + /* All logs starts with command(1) dfile(2) process(4) result(2) */ + +void _ma_log(enum maria_log_commands command, MARIA_HA *info, + const byte *buffert, uint length) +{ + char buff[11]; + int error,old_errno; + ulong pid=(ulong) GETPID(); + old_errno=my_errno; + bzero(buff,sizeof(buff)); + buff[0]=(char) command; + mi_int2store(buff+1,info->dfile); + mi_int4store(buff+3,pid); + mi_int2store(buff+9,length); + + pthread_mutex_lock(&THR_LOCK_maria); + error=my_lock(maria_log_file,F_WRLCK,0L,F_TO_EOF,MYF(MY_SEEK_NOT_DONE)); + VOID(my_write(maria_log_file,buff,sizeof(buff),MYF(0))); + VOID(my_write(maria_log_file,buffert,length,MYF(0))); + if (!error) + error=my_lock(maria_log_file,F_UNLCK,0L,F_TO_EOF,MYF(MY_SEEK_NOT_DONE)); + pthread_mutex_unlock(&THR_LOCK_maria); + my_errno=old_errno; +} + + +void _ma_log_command(enum maria_log_commands command, MARIA_HA *info, + const byte *buffert, uint length, int result) +{ + char buff[9]; + int error,old_errno; + ulong pid=(ulong) GETPID(); + + old_errno=my_errno; + buff[0]=(char) command; + mi_int2store(buff+1,info->dfile); + mi_int4store(buff+3,pid); + mi_int2store(buff+7,result); + pthread_mutex_lock(&THR_LOCK_maria); + error=my_lock(maria_log_file,F_WRLCK,0L,F_TO_EOF,MYF(MY_SEEK_NOT_DONE)); + VOID(my_write(maria_log_file,buff,sizeof(buff),MYF(0))); + if (buffert) + VOID(my_write(maria_log_file,buffert,length,MYF(0))); + if (!error) + error=my_lock(maria_log_file,F_UNLCK,0L,F_TO_EOF,MYF(MY_SEEK_NOT_DONE)); + pthread_mutex_unlock(&THR_LOCK_maria); + my_errno=old_errno; +} + + +void _ma_log_record(enum maria_log_commands command, MARIA_HA *info, + const byte *record, my_off_t filepos, int result) +{ + char buff[21],*pos; + int error,old_errno; + uint length; + ulong pid=(ulong) GETPID(); + + old_errno=my_errno; + if (!info->s->base.blobs) + length=info->s->base.reclength; + else + length=info->s->base.reclength+ _ma_calc_total_blob_length(info,record); + buff[0]=(char) command; + mi_int2store(buff+1,info->dfile); + mi_int4store(buff+3,pid); + mi_int2store(buff+7,result); + mi_sizestore(buff+9,filepos); + mi_int4store(buff+17,length); + pthread_mutex_lock(&THR_LOCK_maria); + error=my_lock(maria_log_file,F_WRLCK,0L,F_TO_EOF,MYF(MY_SEEK_NOT_DONE)); + VOID(my_write(maria_log_file,buff,sizeof(buff),MYF(0))); + VOID(my_write(maria_log_file,(byte*) record,info->s->base.reclength,MYF(0))); + if (info->s->base.blobs) + { + MARIA_BLOB *blob,*end; + + for (end=info->blobs+info->s->base.blobs, blob= info->blobs; + blob != end ; + blob++) + { + memcpy_fixed(&pos,record+blob->offset+blob->pack_length,sizeof(char*)); + VOID(my_write(maria_log_file,pos,blob->length,MYF(0))); + } + } + if (!error) + error=my_lock(maria_log_file,F_UNLCK,0L,F_TO_EOF,MYF(MY_SEEK_NOT_DONE)); + pthread_mutex_unlock(&THR_LOCK_maria); + my_errno=old_errno; +} diff --git a/storage/maria/ma_open.c b/storage/maria/ma_open.c new file mode 100644 index 00000000000..e1f1088c6d1 --- /dev/null +++ b/storage/maria/ma_open.c @@ -0,0 +1,1288 @@ +/* Copyright (C) 2006 MySQL AB & MySQL Finland AB & TCX DataKonsult AB + + 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; either version 2 of the License, or + (at your option) any later version. + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ + +/* open a isam-database */ + +#include "ma_fulltext.h" +#include "ma_sp_defs.h" +#include "ma_rt_index.h" +#include + +#if defined(MSDOS) || defined(__WIN__) +#ifdef __WIN__ +#include +#else +#include /* Prototype for getpid */ +#endif +#endif +#ifdef VMS +#include "static.c" +#endif + +static void setup_key_functions(MARIA_KEYDEF *keyinfo); +#define get_next_element(to,pos,size) { memcpy((char*) to,pos,(size_t) size); \ + pos+=size;} + + +#define disk_pos_assert(pos, end_pos) \ +if (pos > end_pos) \ +{ \ + my_errno=HA_ERR_CRASHED; \ + goto err; \ +} + + +/****************************************************************************** +** Return the shared struct if the table is already open. +** In MySQL the server will handle version issues. +******************************************************************************/ + +MARIA_HA *_ma_test_if_reopen(char *filename) +{ + LIST *pos; + + for (pos=maria_open_list ; pos ; pos=pos->next) + { + MARIA_HA *info=(MARIA_HA*) pos->data; + MARIA_SHARE *share=info->s; + if (!strcmp(share->unique_file_name,filename) && share->last_version) + return info; + } + return 0; +} + + +/****************************************************************************** + open a MARIA database. + See my_base.h for the handle_locking argument + if handle_locking and HA_OPEN_ABORT_IF_CRASHED then abort if the table + is marked crashed or if we are not using locking and the table doesn't + have an open count of 0. +******************************************************************************/ + +MARIA_HA *maria_open(const char *name, int mode, uint open_flags) +{ + int lock_error,kfile,open_mode,save_errno,have_rtree=0; + uint i,j,len,errpos,head_length,base_pos,offset,info_length,keys, + key_parts,unique_key_parts,fulltext_keys,uniques; + char name_buff[FN_REFLEN], org_name[FN_REFLEN], index_name[FN_REFLEN], + data_name[FN_REFLEN]; + char *disk_cache, *disk_pos, *end_pos; + MARIA_HA info,*m_info,*old_info; + MARIA_SHARE share_buff,*share; + ulong rec_per_key_part[HA_MAX_POSSIBLE_KEY*HA_MAX_KEY_SEG]; + my_off_t key_root[HA_MAX_POSSIBLE_KEY],key_del[MARIA_MAX_KEY_BLOCK_SIZE]; + ulonglong max_key_file_length, max_data_file_length; + DBUG_ENTER("maria_open"); + + LINT_INIT(m_info); + kfile= -1; + lock_error=1; + errpos=0; + head_length=sizeof(share_buff.state.header); + bzero((byte*) &info,sizeof(info)); + + my_realpath(name_buff, fn_format(org_name,name,"",MARIA_NAME_IEXT, + MY_UNPACK_FILENAME|MY_APPEND_EXT),MYF(0)); + pthread_mutex_lock(&THR_LOCK_maria); + if (!(old_info=_ma_test_if_reopen(name_buff))) + { + share= &share_buff; + bzero((gptr) &share_buff,sizeof(share_buff)); + share_buff.state.rec_per_key_part=rec_per_key_part; + share_buff.state.key_root=key_root; + share_buff.state.key_del=key_del; + share_buff.key_cache= multi_key_cache_search(name_buff, strlen(name_buff), + maria_key_cache); + + DBUG_EXECUTE_IF("maria_pretend_crashed_table_on_open", + if (strstr(name, "/t1")) + { + my_errno= HA_ERR_CRASHED; + goto err; + }); + if ((kfile=my_open(name_buff,(open_mode=O_RDWR) | O_SHARE,MYF(0))) < 0) + { + if ((errno != EROFS && errno != EACCES) || + mode != O_RDONLY || + (kfile=my_open(name_buff,(open_mode=O_RDONLY) | O_SHARE,MYF(0))) < 0) + goto err; + } + share->mode=open_mode; + errpos=1; + if (my_read(kfile,(char*) share->state.header.file_version,head_length, + MYF(MY_NABP))) + { + my_errno= HA_ERR_NOT_A_TABLE; + goto err; + } + if (memcmp((byte*) share->state.header.file_version, + (byte*) maria_file_magic, 4)) + { + DBUG_PRINT("error",("Wrong header in %s",name_buff)); + DBUG_DUMP("error_dump",(char*) share->state.header.file_version, + head_length); + my_errno=HA_ERR_NOT_A_TABLE; + goto err; + } + share->options= mi_uint2korr(share->state.header.options); + if (share->options & + ~(HA_OPTION_PACK_RECORD | HA_OPTION_PACK_KEYS | + HA_OPTION_COMPRESS_RECORD | HA_OPTION_READ_ONLY_DATA | + HA_OPTION_TEMP_COMPRESS_RECORD | HA_OPTION_CHECKSUM | + HA_OPTION_TMP_TABLE | HA_OPTION_DELAY_KEY_WRITE | + HA_OPTION_RELIES_ON_SQL_LAYER)) + { + DBUG_PRINT("error",("wrong options: 0x%lx", share->options)); + my_errno=HA_ERR_OLD_FILE; + goto err; + } + if ((share->options & HA_OPTION_RELIES_ON_SQL_LAYER) && + ! (open_flags & HA_OPEN_FROM_SQL_LAYER)) + { + DBUG_PRINT("error", ("table cannot be openned from non-sql layer")); + my_errno= HA_ERR_UNSUPPORTED; + goto err; + } + /* Don't call realpath() if the name can't be a link */ + if (!strcmp(name_buff, org_name) || + my_readlink(index_name, org_name, MYF(0)) == -1) + (void) strmov(index_name, org_name); + *strrchr(org_name, '.')= '\0'; + (void) fn_format(data_name,org_name,"",MARIA_NAME_DEXT, + MY_APPEND_EXT|MY_UNPACK_FILENAME|MY_RESOLVE_SYMLINKS); + + info_length=mi_uint2korr(share->state.header.header_length); + base_pos=mi_uint2korr(share->state.header.base_pos); + if (!(disk_cache=(char*) my_alloca(info_length+128))) + { + my_errno=ENOMEM; + goto err; + } + end_pos=disk_cache+info_length; + errpos=2; + + VOID(my_seek(kfile,0L,MY_SEEK_SET,MYF(0))); + if (!(open_flags & HA_OPEN_TMP_TABLE)) + { + if ((lock_error=my_lock(kfile,F_RDLCK,0L,F_TO_EOF, + MYF(open_flags & HA_OPEN_WAIT_IF_LOCKED ? + 0 : MY_DONT_WAIT))) && + !(open_flags & HA_OPEN_IGNORE_IF_LOCKED)) + goto err; + } + errpos=3; + if (my_read(kfile,disk_cache,info_length,MYF(MY_NABP))) + { + my_errno=HA_ERR_CRASHED; + goto err; + } + len=mi_uint2korr(share->state.header.state_info_length); + keys= (uint) share->state.header.keys; + uniques= (uint) share->state.header.uniques; + fulltext_keys= (uint) share->state.header.fulltext_keys; + key_parts= mi_uint2korr(share->state.header.key_parts); + unique_key_parts= mi_uint2korr(share->state.header.unique_key_parts); + if (len != MARIA_STATE_INFO_SIZE) + { + DBUG_PRINT("warning", + ("saved_state_info_length: %d state_info_length: %d", + len,MARIA_STATE_INFO_SIZE)); + } + share->state_diff_length=len-MARIA_STATE_INFO_SIZE; + + _ma_state_info_read((uchar*) disk_cache, &share->state); + len= mi_uint2korr(share->state.header.base_info_length); + if (len != MARIA_BASE_INFO_SIZE) + { + DBUG_PRINT("warning",("saved_base_info_length: %d base_info_length: %d", + len,MARIA_BASE_INFO_SIZE)); + } + disk_pos= (char*) + _ma_n_base_info_read((uchar*) disk_cache + base_pos, &share->base); + share->state.state_length=base_pos; + + if (!(open_flags & HA_OPEN_FOR_REPAIR) && + ((share->state.changed & STATE_CRASHED) || + ((open_flags & HA_OPEN_ABORT_IF_CRASHED) && + (my_disable_locking && share->state.open_count)))) + { + DBUG_PRINT("error",("Table is marked as crashed")); + my_errno=((share->state.changed & STATE_CRASHED_ON_REPAIR) ? + HA_ERR_CRASHED_ON_REPAIR : HA_ERR_CRASHED_ON_USAGE); + goto err; + } + + /* sanity check */ + if (share->base.keystart > 65535 || share->base.rec_reflength > 8) + { + my_errno=HA_ERR_CRASHED; + goto err; + } + + key_parts+=fulltext_keys*FT_SEGS; + if (share->base.max_key_length > HA_MAX_KEY_BUFF || keys > MARIA_MAX_KEY || + key_parts >= MARIA_MAX_KEY * HA_MAX_KEY_SEG) + { + DBUG_PRINT("error",("Wrong key info: Max_key_length: %d keys: %d key_parts: %d", share->base.max_key_length, keys, key_parts)); + my_errno=HA_ERR_UNSUPPORTED; + goto err; + } + + /* Correct max_file_length based on length of sizeof(off_t) */ + max_data_file_length= + (share->options & (HA_OPTION_PACK_RECORD | HA_OPTION_COMPRESS_RECORD)) ? + (((ulonglong) 1 << (share->base.rec_reflength*8))-1) : + (_ma_safe_mul(share->base.pack_reclength, + (ulonglong) 1 << (share->base.rec_reflength*8))-1); + max_key_file_length= + _ma_safe_mul(MARIA_MIN_KEY_BLOCK_LENGTH, + ((ulonglong) 1 << (share->base.key_reflength*8))-1); +#if SIZEOF_OFF_T == 4 + set_if_smaller(max_data_file_length, INT_MAX32); + set_if_smaller(max_key_file_length, INT_MAX32); +#endif +#if USE_RAID && SYSTEM_SIZEOF_OFF_T == 4 + set_if_smaller(max_key_file_length, INT_MAX32); + if (!share->base.raid_type) + { + set_if_smaller(max_data_file_length, INT_MAX32); + } + else + { + set_if_smaller(max_data_file_length, + (ulonglong) share->base.raid_chunks << 31); + } +#elif !defined(USE_RAID) + if (share->base.raid_type) + { + DBUG_PRINT("error",("Table uses RAID but we don't have RAID support")); + my_errno=HA_ERR_UNSUPPORTED; + goto err; + } +#endif + share->base.max_data_file_length=(my_off_t) max_data_file_length; + share->base.max_key_file_length=(my_off_t) max_key_file_length; + + if (share->options & HA_OPTION_COMPRESS_RECORD) + share->base.max_key_length+=2; /* For safety */ + + if (!my_multi_malloc(MY_WME, + &share,sizeof(*share), + &share->state.rec_per_key_part,sizeof(long)*key_parts, + &share->keyinfo,keys*sizeof(MARIA_KEYDEF), + &share->uniqueinfo,uniques*sizeof(MARIA_UNIQUEDEF), + &share->keyparts, + (key_parts+unique_key_parts+keys+uniques) * + sizeof(HA_KEYSEG), + &share->rec, + (share->base.fields+1)*sizeof(MARIA_COLUMNDEF), + &share->blobs,sizeof(MARIA_BLOB)*share->base.blobs, + &share->unique_file_name,strlen(name_buff)+1, + &share->index_file_name,strlen(index_name)+1, + &share->data_file_name,strlen(data_name)+1, + &share->state.key_root,keys*sizeof(my_off_t), + &share->state.key_del, + (share->state.header.max_block_size*sizeof(my_off_t)), +#ifdef THREAD + &share->key_root_lock,sizeof(rw_lock_t)*keys, +#endif + &share->mmap_lock,sizeof(rw_lock_t), + NullS)) + goto err; + errpos=4; + *share=share_buff; + memcpy((char*) share->state.rec_per_key_part, + (char*) rec_per_key_part, sizeof(long)*key_parts); + memcpy((char*) share->state.key_root, + (char*) key_root, sizeof(my_off_t)*keys); + memcpy((char*) share->state.key_del, + (char*) key_del, (sizeof(my_off_t) * + share->state.header.max_block_size)); + strmov(share->unique_file_name, name_buff); + share->unique_name_length= strlen(name_buff); + strmov(share->index_file_name, index_name); + strmov(share->data_file_name, data_name); + + share->blocksize=min(IO_SIZE,maria_block_size); + { + HA_KEYSEG *pos=share->keyparts; + for (i=0 ; i < keys ; i++) + { + share->keyinfo[i].share= share; + disk_pos=_ma_keydef_read(disk_pos, &share->keyinfo[i]); + disk_pos_assert(disk_pos + share->keyinfo[i].keysegs * HA_KEYSEG_SIZE, + end_pos); + if (share->keyinfo[i].key_alg == HA_KEY_ALG_RTREE) + have_rtree=1; + set_if_smaller(share->blocksize,share->keyinfo[i].block_length); + share->keyinfo[i].seg=pos; + for (j=0 ; j < share->keyinfo[i].keysegs; j++,pos++) + { + disk_pos=_ma_keyseg_read(disk_pos, pos); + + if (pos->type == HA_KEYTYPE_TEXT || + pos->type == HA_KEYTYPE_VARTEXT1 || + pos->type == HA_KEYTYPE_VARTEXT2) + { + if (!pos->language) + pos->charset=default_charset_info; + else if (!(pos->charset= get_charset(pos->language, MYF(MY_WME)))) + { + my_errno=HA_ERR_UNKNOWN_CHARSET; + goto err; + } + } + } + if (share->keyinfo[i].flag & HA_SPATIAL) + { +#ifdef HAVE_SPATIAL + uint sp_segs=SPDIMS*2; + share->keyinfo[i].seg=pos-sp_segs; + share->keyinfo[i].keysegs--; +#else + my_errno=HA_ERR_UNSUPPORTED; + goto err; +#endif + } + else if (share->keyinfo[i].flag & HA_FULLTEXT) + { + if (!fulltext_keys) + { /* 4.0 compatibility code, to be removed in 5.0 */ + share->keyinfo[i].seg=pos-FT_SEGS; + share->keyinfo[i].keysegs-=FT_SEGS; + } + else + { + uint j; + share->keyinfo[i].seg=pos; + for (j=0; j < FT_SEGS; j++) + { + *pos= ft_keysegs[j]; + pos[0].language= pos[-1].language; + if (!(pos[0].charset= pos[-1].charset)) + { + my_errno=HA_ERR_CRASHED; + goto err; + } + pos++; + } + } + if (!share->ft2_keyinfo.seg) + { + memcpy(& share->ft2_keyinfo, & share->keyinfo[i], sizeof(MARIA_KEYDEF)); + share->ft2_keyinfo.keysegs=1; + share->ft2_keyinfo.flag=0; + share->ft2_keyinfo.keylength= + share->ft2_keyinfo.minlength= + share->ft2_keyinfo.maxlength=HA_FT_WLEN+share->base.rec_reflength; + share->ft2_keyinfo.seg=pos-1; + share->ft2_keyinfo.end=pos; + setup_key_functions(& share->ft2_keyinfo); + } + } + setup_key_functions(share->keyinfo+i); + share->keyinfo[i].end=pos; + pos->type=HA_KEYTYPE_END; /* End */ + pos->length=share->base.rec_reflength; + pos->null_bit=0; + pos->flag=0; /* For purify */ + pos++; + } + for (i=0 ; i < uniques ; i++) + { + disk_pos=_ma_uniquedef_read(disk_pos, &share->uniqueinfo[i]); + disk_pos_assert(disk_pos + share->uniqueinfo[i].keysegs * + HA_KEYSEG_SIZE, end_pos); + share->uniqueinfo[i].seg=pos; + for (j=0 ; j < share->uniqueinfo[i].keysegs; j++,pos++) + { + disk_pos=_ma_keyseg_read(disk_pos, pos); + if (pos->type == HA_KEYTYPE_TEXT || + pos->type == HA_KEYTYPE_VARTEXT1 || + pos->type == HA_KEYTYPE_VARTEXT2) + { + if (!pos->language) + pos->charset=default_charset_info; + else if (!(pos->charset= get_charset(pos->language, MYF(MY_WME)))) + { + my_errno=HA_ERR_UNKNOWN_CHARSET; + goto err; + } + } + } + share->uniqueinfo[i].end=pos; + pos->type=HA_KEYTYPE_END; /* End */ + pos->null_bit=0; + pos->flag=0; + pos++; + } + share->ftparsers= 0; + } + + disk_pos_assert(disk_pos + share->base.fields *MARIA_COLUMNDEF_SIZE, end_pos); + for (i=j=offset=0 ; i < share->base.fields ; i++) + { + disk_pos=_ma_recinfo_read(disk_pos,&share->rec[i]); + share->rec[i].pack_type=0; + share->rec[i].huff_tree=0; + share->rec[i].offset=offset; + if (share->rec[i].type == (int) FIELD_BLOB) + { + share->blobs[j].pack_length= + share->rec[i].length-maria_portable_sizeof_char_ptr;; + share->blobs[j].offset=offset; + j++; + } + offset+=share->rec[i].length; + } + share->rec[i].type=(int) FIELD_LAST; /* End marker */ + + if (! lock_error) + { + VOID(my_lock(kfile,F_UNLCK,0L,F_TO_EOF,MYF(MY_SEEK_NOT_DONE))); + lock_error=1; /* Database unlocked */ + } + + if (_ma_open_datafile(&info, share, -1)) + goto err; + errpos=5; + + share->kfile=kfile; + share->this_process=(ulong) getpid(); + share->last_process= share->state.process; + share->base.key_parts=key_parts; + share->base.all_key_parts=key_parts+unique_key_parts; + if (!(share->last_version=share->state.version)) + share->last_version=1; /* Safety */ + share->rec_reflength=share->base.rec_reflength; /* May be changed */ + share->base.margin_key_file_length=(share->base.max_key_file_length - + (keys ? MARIA_INDEX_BLOCK_MARGIN * + share->blocksize * keys : 0)); + share->blocksize=min(IO_SIZE,maria_block_size); + share->data_file_type=STATIC_RECORD; + if (share->options & HA_OPTION_COMPRESS_RECORD) + { + share->data_file_type = COMPRESSED_RECORD; + share->options|= HA_OPTION_READ_ONLY_DATA; + info.s=share; + if (_ma_read_pack_info(&info, + (pbool) + test(!(share->options & + (HA_OPTION_PACK_RECORD | + HA_OPTION_TEMP_COMPRESS_RECORD))))) + goto err; + } + else if (share->options & HA_OPTION_PACK_RECORD) + share->data_file_type = DYNAMIC_RECORD; + my_afree((gptr) disk_cache); + _ma_setup_functions(share); +#ifdef THREAD + thr_lock_init(&share->lock); + VOID(pthread_mutex_init(&share->intern_lock,MY_MUTEX_INIT_FAST)); + for (i=0; ikey_root_lock[i], NULL)); + VOID(my_rwlock_init(&share->mmap_lock, NULL)); + if (!thr_lock_inited) + { + /* Probably a single threaded program; Don't use concurrent inserts */ + maria_concurrent_insert=0; + } + else if (maria_concurrent_insert) + { + share->concurrent_insert= + ((share->options & (HA_OPTION_READ_ONLY_DATA | HA_OPTION_TMP_TABLE | + HA_OPTION_COMPRESS_RECORD | + HA_OPTION_TEMP_COMPRESS_RECORD)) || + (open_flags & HA_OPEN_TMP_TABLE) || + have_rtree) ? 0 : 1; + if (share->concurrent_insert) + { + share->lock.get_status=_ma_get_status; + share->lock.copy_status=_ma_copy_status; + share->lock.update_status=_ma_update_status; + share->lock.check_status=_ma_check_status; + } + } +#endif + } + else + { + share= old_info->s; + if (mode == O_RDWR && share->mode == O_RDONLY) + { + my_errno=EACCES; /* Can't open in write mode */ + goto err; + } + if (_ma_open_datafile(&info, share, old_info->dfile)) + goto err; + errpos=5; + have_rtree= old_info->maria_rtree_recursion_state != NULL; + } + + /* alloc and set up private structure parts */ + if (!my_multi_malloc(MY_WME, + &m_info,sizeof(MARIA_HA), + &info.blobs,sizeof(MARIA_BLOB)*share->base.blobs, + &info.buff,(share->base.max_key_block_length*2+ + share->base.max_key_length), + &info.lastkey,share->base.max_key_length*3+1, + &info.first_mbr_key, share->base.max_key_length, + &info.filename,strlen(name)+1, + &info.maria_rtree_recursion_state,have_rtree ? 1024 : 0, + NullS)) + goto err; + errpos=6; + + if (!have_rtree) + info.maria_rtree_recursion_state= NULL; + + strmov(info.filename,name); + memcpy(info.blobs,share->blobs,sizeof(MARIA_BLOB)*share->base.blobs); + info.lastkey2=info.lastkey+share->base.max_key_length; + + info.s=share; + info.lastpos= HA_OFFSET_ERROR; + info.update= (short) (HA_STATE_NEXT_FOUND+HA_STATE_PREV_FOUND); + info.opt_flag=READ_CHECK_USED; + info.this_unique= (ulong) info.dfile; /* Uniq number in process */ + if (share->data_file_type == COMPRESSED_RECORD) + info.this_unique= share->state.unique; + info.this_loop=0; /* Update counter */ + info.last_unique= share->state.unique; + info.last_loop= share->state.update_count; + if (mode == O_RDONLY) + share->options|=HA_OPTION_READ_ONLY_DATA; + info.lock_type=F_UNLCK; + info.quick_mode=0; + info.bulk_insert=0; + info.ft1_to_ft2=0; + info.errkey= -1; + info.page_changed=1; + pthread_mutex_lock(&share->intern_lock); + info.read_record=share->read_record; + share->reopen++; + share->write_flag=MYF(MY_NABP | MY_WAIT_IF_FULL); + if (share->options & HA_OPTION_READ_ONLY_DATA) + { + info.lock_type=F_RDLCK; + share->r_locks++; + share->tot_locks++; + } + if ((open_flags & HA_OPEN_TMP_TABLE) || + (share->options & HA_OPTION_TMP_TABLE)) + { + share->temporary=share->delay_key_write=1; + share->write_flag=MYF(MY_NABP); + share->w_locks++; /* We don't have to update status */ + share->tot_locks++; + info.lock_type=F_WRLCK; + } + if (((open_flags & HA_OPEN_DELAY_KEY_WRITE) || + (share->options & HA_OPTION_DELAY_KEY_WRITE)) && + maria_delay_key_write) + share->delay_key_write=1; + info.state= &share->state.state; /* Change global values by default */ + pthread_mutex_unlock(&share->intern_lock); + + /* Allocate buffer for one record */ + + /* prerequisites: bzero(info) && info->s=share; are met. */ + if (!_ma_alloc_rec_buff(&info, -1, &info.rec_buff)) + goto err; + bzero(info.rec_buff, _ma_get_rec_buff_len(&info, info.rec_buff)); + + *m_info=info; +#ifdef THREAD + thr_lock_data_init(&share->lock,&m_info->lock,(void*) m_info); +#endif + m_info->open_list.data=(void*) m_info; + maria_open_list=list_add(maria_open_list,&m_info->open_list); + + pthread_mutex_unlock(&THR_LOCK_maria); + if (maria_log_file >= 0) + { + intern_filename(name_buff,share->index_file_name); + _ma_log(MARIA_LOG_OPEN,m_info,name_buff,(uint) strlen(name_buff)); + } + DBUG_RETURN(m_info); + +err: + save_errno=my_errno ? my_errno : HA_ERR_END_OF_FILE; + if ((save_errno == HA_ERR_CRASHED) || + (save_errno == HA_ERR_CRASHED_ON_USAGE) || + (save_errno == HA_ERR_CRASHED_ON_REPAIR)) + _ma_report_error(save_errno, name); + switch (errpos) { + case 6: + my_free((gptr) m_info,MYF(0)); + /* fall through */ + case 5: + VOID(my_close(info.dfile,MYF(0))); + if (old_info) + break; /* Don't remove open table */ + /* fall through */ + case 4: + my_free((gptr) share,MYF(0)); + /* fall through */ + case 3: + if (! lock_error) + VOID(my_lock(kfile, F_UNLCK, 0L, F_TO_EOF, MYF(MY_SEEK_NOT_DONE))); + /* fall through */ + case 2: + my_afree((gptr) disk_cache); + /* fall through */ + case 1: + VOID(my_close(kfile,MYF(0))); + /* fall through */ + case 0: + default: + break; + } + pthread_mutex_unlock(&THR_LOCK_maria); + my_errno=save_errno; + DBUG_RETURN (NULL); +} /* maria_open */ + + +byte *_ma_alloc_rec_buff(MARIA_HA *info, ulong length, byte **buf) +{ + uint extra; + uint32 old_length; + LINT_INIT(old_length); + + if (! *buf || length > (old_length=_ma_get_rec_buff_len(info, *buf))) + { + byte *newptr = *buf; + + /* to simplify initial init of info->rec_buf in maria_open and maria_extra */ + if (length == (ulong) -1) + { + length= max(info->s->base.pack_reclength, + info->s->base.max_key_length); + /* Avoid unnecessary realloc */ + if (newptr && length == old_length) + return newptr; + } + + extra= ((info->s->options & HA_OPTION_PACK_RECORD) ? + ALIGN_SIZE(MARIA_MAX_DYN_BLOCK_HEADER)+MARIA_SPLIT_LENGTH+ + MARIA_REC_BUFF_OFFSET : 0); + if (extra && newptr) + newptr-= MARIA_REC_BUFF_OFFSET; + if (!(newptr=(byte*) my_realloc((gptr)newptr, length+extra+8, + MYF(MY_ALLOW_ZERO_PTR)))) + return newptr; + *((uint32 *) newptr)= (uint32) length; + *buf= newptr+(extra ? MARIA_REC_BUFF_OFFSET : 0); + } + return *buf; +} + + +ulonglong _ma_safe_mul(ulonglong a, ulonglong b) +{ + ulonglong max_val= ~ (ulonglong) 0; /* my_off_t is unsigned */ + + if (!a || max_val / a < b) + return max_val; + return a*b; +} + + /* Set up functions in structs */ + +void _ma_setup_functions(register MARIA_SHARE *share) +{ + if (share->options & HA_OPTION_COMPRESS_RECORD) + { + share->read_record= _ma_read_pack_record; + share->read_rnd= _ma_read_rnd_pack_record; + if (!(share->options & HA_OPTION_TEMP_COMPRESS_RECORD)) + share->calc_checksum=0; /* No checksum */ + else if (share->options & HA_OPTION_PACK_RECORD) + share->calc_checksum= _ma_checksum; + else + share->calc_checksum= _ma_static_checksum; + } + else if (share->options & HA_OPTION_PACK_RECORD) + { + share->read_record= _ma_read_dynamic_record; + share->read_rnd= _ma_read_rnd_dynamic_record; + share->delete_record= _ma_delete_dynamic_record; + share->compare_record= _ma_cmp_dynamic_record; + share->compare_unique= _ma_cmp_dynamic_unique; + share->calc_checksum= _ma_checksum; + + /* add bits used to pack data to pack_reclength for faster allocation */ + share->base.pack_reclength+= share->base.pack_bits; + if (share->base.blobs) + { + share->update_record= _ma_update_blob_record; + share->write_record= _ma_write_blob_record; + } + else + { + share->write_record= _ma_write_dynamic_record; + share->update_record= _ma_update_dynamic_record; + } + } + else + { + share->read_record= _ma_read_static_record; + share->read_rnd= _ma_read_rnd_static_record; + share->delete_record= _ma_delete_static_record; + share->compare_record= _ma_cmp_static_record; + share->update_record= _ma_update_static_record; + share->write_record= _ma_write_static_record; + share->compare_unique= _ma_cmp_static_unique; + share->calc_checksum= _ma_static_checksum; + } + share->file_read= _ma_nommap_pread; + share->file_write= _ma_nommap_pwrite; + if (!(share->options & HA_OPTION_CHECKSUM)) + share->calc_checksum=0; + return; +} + + +static void setup_key_functions(register MARIA_KEYDEF *keyinfo) +{ + if (keyinfo->key_alg == HA_KEY_ALG_RTREE) + { +#ifdef HAVE_RTREE_KEYS + keyinfo->ck_insert = maria_rtree_insert; + keyinfo->ck_delete = maria_rtree_delete; +#else + DBUG_ASSERT(0); /* maria_open should check it never happens */ +#endif + } + else + { + keyinfo->ck_insert = _ma_ck_write; + keyinfo->ck_delete = _ma_ck_delete; + } + if (keyinfo->flag & HA_BINARY_PACK_KEY) + { /* Simple prefix compression */ + keyinfo->bin_search= _ma_seq_search; + keyinfo->get_key= _ma_get_binary_pack_key; + keyinfo->pack_key= _ma_calc_bin_pack_key_length; + keyinfo->store_key= _ma_store_bin_pack_key; + } + else if (keyinfo->flag & HA_VAR_LENGTH_KEY) + { + keyinfo->get_key= _ma_get_pack_key; + if (keyinfo->seg[0].flag & HA_PACK_KEY) + { /* Prefix compression */ + if (!keyinfo->seg->charset || use_strnxfrm(keyinfo->seg->charset) || + (keyinfo->seg->flag & HA_NULL_PART)) + keyinfo->bin_search= _ma_seq_search; + else + keyinfo->bin_search= _ma_prefix_search; + keyinfo->pack_key= _ma_calc_var_pack_key_length; + keyinfo->store_key= _ma_store_var_pack_key; + } + else + { + keyinfo->bin_search= _ma_seq_search; + keyinfo->pack_key= _ma_calc_var_key_length; /* Variable length key */ + keyinfo->store_key= _ma_store_static_key; + } + } + else + { + keyinfo->bin_search= _ma_bin_search; + keyinfo->get_key= _ma_get_static_key; + keyinfo->pack_key= _ma_calc_static_key_length; + keyinfo->store_key= _ma_store_static_key; + } + return; +} + + +/* + Function to save and store the header in the index file (.MYI) +*/ + +uint _ma_state_info_write(File file, MARIA_STATE_INFO *state, uint pWrite) +{ + uchar buff[MARIA_STATE_INFO_SIZE + MARIA_STATE_EXTRA_SIZE]; + uchar *ptr=buff; + uint i, keys= (uint) state->header.keys, + key_blocks=state->header.max_block_size; + DBUG_ENTER("_ma_state_info_write"); + + memcpy_fixed(ptr,&state->header,sizeof(state->header)); + ptr+=sizeof(state->header); + + /* open_count must be first because of _ma_mark_file_changed ! */ + mi_int2store(ptr,state->open_count); ptr +=2; + *ptr++= (uchar)state->changed; *ptr++= state->sortkey; + mi_rowstore(ptr,state->state.records); ptr +=8; + mi_rowstore(ptr,state->state.del); ptr +=8; + mi_rowstore(ptr,state->split); ptr +=8; + mi_sizestore(ptr,state->dellink); ptr +=8; + mi_sizestore(ptr,state->state.key_file_length); ptr +=8; + mi_sizestore(ptr,state->state.data_file_length); ptr +=8; + mi_sizestore(ptr,state->state.empty); ptr +=8; + mi_sizestore(ptr,state->state.key_empty); ptr +=8; + mi_int8store(ptr,state->auto_increment); ptr +=8; + mi_int8store(ptr,(ulonglong) state->state.checksum);ptr +=8; + mi_int4store(ptr,state->process); ptr +=4; + mi_int4store(ptr,state->unique); ptr +=4; + mi_int4store(ptr,state->status); ptr +=4; + mi_int4store(ptr,state->update_count); ptr +=4; + + ptr+=state->state_diff_length; + + for (i=0; i < keys; i++) + { + mi_sizestore(ptr,state->key_root[i]); ptr +=8; + } + for (i=0; i < key_blocks; i++) + { + mi_sizestore(ptr,state->key_del[i]); ptr +=8; + } + if (pWrite & 2) /* From isamchk */ + { + uint key_parts= mi_uint2korr(state->header.key_parts); + mi_int4store(ptr,state->sec_index_changed); ptr +=4; + mi_int4store(ptr,state->sec_index_used); ptr +=4; + mi_int4store(ptr,state->version); ptr +=4; + mi_int8store(ptr,state->key_map); ptr +=8; + mi_int8store(ptr,(ulonglong) state->create_time); ptr +=8; + mi_int8store(ptr,(ulonglong) state->recover_time); ptr +=8; + mi_int8store(ptr,(ulonglong) state->check_time); ptr +=8; + mi_sizestore(ptr,state->rec_per_key_rows); ptr+=8; + for (i=0 ; i < key_parts ; i++) + { + mi_int4store(ptr,state->rec_per_key_part[i]); ptr+=4; + } + } + + if (pWrite & 1) + DBUG_RETURN(my_pwrite(file,(char*) buff, (uint) (ptr-buff), 0L, + MYF(MY_NABP | MY_THREADSAFE))); + DBUG_RETURN(my_write(file, (char*) buff, (uint) (ptr-buff), + MYF(MY_NABP))); +} + + +uchar *_ma_state_info_read(uchar *ptr, MARIA_STATE_INFO *state) +{ + uint i,keys,key_parts,key_blocks; + memcpy_fixed(&state->header,ptr, sizeof(state->header)); + ptr +=sizeof(state->header); + keys=(uint) state->header.keys; + key_parts=mi_uint2korr(state->header.key_parts); + key_blocks=state->header.max_block_size; + + state->open_count = mi_uint2korr(ptr); ptr +=2; + state->changed= (bool) *ptr++; + state->sortkey = (uint) *ptr++; + state->state.records= mi_rowkorr(ptr); ptr +=8; + state->state.del = mi_rowkorr(ptr); ptr +=8; + state->split = mi_rowkorr(ptr); ptr +=8; + state->dellink= mi_sizekorr(ptr); ptr +=8; + state->state.key_file_length = mi_sizekorr(ptr); ptr +=8; + state->state.data_file_length= mi_sizekorr(ptr); ptr +=8; + state->state.empty = mi_sizekorr(ptr); ptr +=8; + state->state.key_empty= mi_sizekorr(ptr); ptr +=8; + state->auto_increment=mi_uint8korr(ptr); ptr +=8; + state->state.checksum=(ha_checksum) mi_uint8korr(ptr); ptr +=8; + state->process= mi_uint4korr(ptr); ptr +=4; + state->unique = mi_uint4korr(ptr); ptr +=4; + state->status = mi_uint4korr(ptr); ptr +=4; + state->update_count=mi_uint4korr(ptr); ptr +=4; + + ptr+= state->state_diff_length; + + for (i=0; i < keys; i++) + { + state->key_root[i]= mi_sizekorr(ptr); ptr +=8; + } + for (i=0; i < key_blocks; i++) + { + state->key_del[i] = mi_sizekorr(ptr); ptr +=8; + } + state->sec_index_changed = mi_uint4korr(ptr); ptr +=4; + state->sec_index_used = mi_uint4korr(ptr); ptr +=4; + state->version = mi_uint4korr(ptr); ptr +=4; + state->key_map = mi_uint8korr(ptr); ptr +=8; + state->create_time = (time_t) mi_sizekorr(ptr); ptr +=8; + state->recover_time =(time_t) mi_sizekorr(ptr); ptr +=8; + state->check_time = (time_t) mi_sizekorr(ptr); ptr +=8; + state->rec_per_key_rows=mi_sizekorr(ptr); ptr +=8; + for (i=0 ; i < key_parts ; i++) + { + state->rec_per_key_part[i]= mi_uint4korr(ptr); ptr+=4; + } + return ptr; +} + + +uint _ma_state_info_read_dsk(File file, MARIA_STATE_INFO *state, my_bool pRead) +{ + char buff[MARIA_STATE_INFO_SIZE + MARIA_STATE_EXTRA_SIZE]; + + if (!maria_single_user) + { + if (pRead) + { + if (my_pread(file, buff, state->state_length,0L, MYF(MY_NABP))) + return (MY_FILE_ERROR); + } + else if (my_read(file, buff, state->state_length,MYF(MY_NABP))) + return (MY_FILE_ERROR); + _ma_state_info_read((uchar*) buff, state); + } + return 0; +} + + +/**************************************************************************** +** store and read of MARIA_BASE_INFO +****************************************************************************/ + +uint _ma_base_info_write(File file, MARIA_BASE_INFO *base) +{ + uchar buff[MARIA_BASE_INFO_SIZE], *ptr=buff; + + mi_sizestore(ptr,base->keystart); ptr +=8; + mi_sizestore(ptr,base->max_data_file_length); ptr +=8; + mi_sizestore(ptr,base->max_key_file_length); ptr +=8; + mi_rowstore(ptr,base->records); ptr +=8; + mi_rowstore(ptr,base->reloc); ptr +=8; + mi_int4store(ptr,base->mean_row_length); ptr +=4; + mi_int4store(ptr,base->reclength); ptr +=4; + mi_int4store(ptr,base->pack_reclength); ptr +=4; + mi_int4store(ptr,base->min_pack_length); ptr +=4; + mi_int4store(ptr,base->max_pack_length); ptr +=4; + mi_int4store(ptr,base->min_block_length); ptr +=4; + mi_int4store(ptr,base->fields); ptr +=4; + mi_int4store(ptr,base->pack_fields); ptr +=4; + *ptr++=base->rec_reflength; + *ptr++=base->key_reflength; + *ptr++=base->keys; + *ptr++=base->auto_key; + mi_int2store(ptr,base->pack_bits); ptr +=2; + mi_int2store(ptr,base->blobs); ptr +=2; + mi_int2store(ptr,base->max_key_block_length); ptr +=2; + mi_int2store(ptr,base->max_key_length); ptr +=2; + mi_int2store(ptr,base->extra_alloc_bytes); ptr +=2; + *ptr++= base->extra_alloc_procent; + *ptr++= base->raid_type; + mi_int2store(ptr,base->raid_chunks); ptr +=2; + mi_int4store(ptr,base->raid_chunksize); ptr +=4; + bzero(ptr,6); ptr +=6; /* extra */ + return my_write(file,(char*) buff, (uint) (ptr-buff), MYF(MY_NABP)); +} + + +uchar *_ma_n_base_info_read(uchar *ptr, MARIA_BASE_INFO *base) +{ + base->keystart = mi_sizekorr(ptr); ptr +=8; + base->max_data_file_length = mi_sizekorr(ptr); ptr +=8; + base->max_key_file_length = mi_sizekorr(ptr); ptr +=8; + base->records = (ha_rows) mi_sizekorr(ptr); ptr +=8; + base->reloc = (ha_rows) mi_sizekorr(ptr); ptr +=8; + base->mean_row_length = mi_uint4korr(ptr); ptr +=4; + base->reclength = mi_uint4korr(ptr); ptr +=4; + base->pack_reclength = mi_uint4korr(ptr); ptr +=4; + base->min_pack_length = mi_uint4korr(ptr); ptr +=4; + base->max_pack_length = mi_uint4korr(ptr); ptr +=4; + base->min_block_length = mi_uint4korr(ptr); ptr +=4; + base->fields = mi_uint4korr(ptr); ptr +=4; + base->pack_fields = mi_uint4korr(ptr); ptr +=4; + + base->rec_reflength = *ptr++; + base->key_reflength = *ptr++; + base->keys= *ptr++; + base->auto_key= *ptr++; + base->pack_bits = mi_uint2korr(ptr); ptr +=2; + base->blobs = mi_uint2korr(ptr); ptr +=2; + base->max_key_block_length= mi_uint2korr(ptr); ptr +=2; + base->max_key_length = mi_uint2korr(ptr); ptr +=2; + base->extra_alloc_bytes = mi_uint2korr(ptr); ptr +=2; + base->extra_alloc_procent = *ptr++; + base->raid_type= *ptr++; + base->raid_chunks= mi_uint2korr(ptr); ptr +=2; + base->raid_chunksize= mi_uint4korr(ptr); ptr +=4; + /* TO BE REMOVED: Fix for old RAID files */ + if (base->raid_type == 0) + { + base->raid_chunks=0; + base->raid_chunksize=0; + } + + ptr+=6; + return ptr; +} + +/*-------------------------------------------------------------------------- + maria_keydef +---------------------------------------------------------------------------*/ + +uint _ma_keydef_write(File file, MARIA_KEYDEF *keydef) +{ + uchar buff[MARIA_KEYDEF_SIZE]; + uchar *ptr=buff; + + *ptr++ = (uchar) keydef->keysegs; + *ptr++ = keydef->key_alg; /* Rtree or Btree */ + mi_int2store(ptr,keydef->flag); ptr +=2; + mi_int2store(ptr,keydef->block_length); ptr +=2; + mi_int2store(ptr,keydef->keylength); ptr +=2; + mi_int2store(ptr,keydef->minlength); ptr +=2; + mi_int2store(ptr,keydef->maxlength); ptr +=2; + return my_write(file,(char*) buff, (uint) (ptr-buff), MYF(MY_NABP)); +} + +char *_ma_keydef_read(char *ptr, MARIA_KEYDEF *keydef) +{ + keydef->keysegs = (uint) *ptr++; + keydef->key_alg = *ptr++; /* Rtree or Btree */ + + keydef->flag = mi_uint2korr(ptr); ptr +=2; + keydef->block_length = mi_uint2korr(ptr); ptr +=2; + keydef->keylength = mi_uint2korr(ptr); ptr +=2; + keydef->minlength = mi_uint2korr(ptr); ptr +=2; + keydef->maxlength = mi_uint2korr(ptr); ptr +=2; + keydef->block_size = keydef->block_length/MARIA_MIN_KEY_BLOCK_LENGTH-1; + keydef->underflow_block_length=keydef->block_length/3; + keydef->version = 0; /* Not saved */ + keydef->parser = &ft_default_parser; + keydef->ftparser_nr = 0; + return ptr; +} + +/*************************************************************************** +** maria_keyseg +***************************************************************************/ + +int _ma_keyseg_write(File file, const HA_KEYSEG *keyseg) +{ + uchar buff[HA_KEYSEG_SIZE]; + uchar *ptr=buff; + ulong pos; + + *ptr++= keyseg->type; + *ptr++= keyseg->language; + *ptr++= keyseg->null_bit; + *ptr++= keyseg->bit_start; + *ptr++= keyseg->bit_end; + *ptr++= keyseg->bit_length; + mi_int2store(ptr,keyseg->flag); ptr+=2; + mi_int2store(ptr,keyseg->length); ptr+=2; + mi_int4store(ptr,keyseg->start); ptr+=4; + pos= keyseg->null_bit ? keyseg->null_pos : keyseg->bit_pos; + mi_int4store(ptr, pos); + ptr+=4; + + return my_write(file,(char*) buff, (uint) (ptr-buff), MYF(MY_NABP)); +} + + +char *_ma_keyseg_read(char *ptr, HA_KEYSEG *keyseg) +{ + keyseg->type = *ptr++; + keyseg->language = *ptr++; + keyseg->null_bit = *ptr++; + keyseg->bit_start = *ptr++; + keyseg->bit_end = *ptr++; + keyseg->bit_length = *ptr++; + keyseg->flag = mi_uint2korr(ptr); ptr +=2; + keyseg->length = mi_uint2korr(ptr); ptr +=2; + keyseg->start = mi_uint4korr(ptr); ptr +=4; + keyseg->null_pos = mi_uint4korr(ptr); ptr +=4; + keyseg->charset=0; /* Will be filled in later */ + if (keyseg->null_bit) + keyseg->bit_pos= (uint16)(keyseg->null_pos + (keyseg->null_bit == 7)); + else + { + keyseg->bit_pos= (uint16)keyseg->null_pos; + keyseg->null_pos= 0; + } + return ptr; +} + +/*-------------------------------------------------------------------------- + maria_uniquedef +---------------------------------------------------------------------------*/ + +uint _ma_uniquedef_write(File file, MARIA_UNIQUEDEF *def) +{ + uchar buff[MARIA_UNIQUEDEF_SIZE]; + uchar *ptr=buff; + + mi_int2store(ptr,def->keysegs); ptr+=2; + *ptr++= (uchar) def->key; + *ptr++ = (uchar) def->null_are_equal; + + return my_write(file,(char*) buff, (uint) (ptr-buff), MYF(MY_NABP)); +} + +char *_ma_uniquedef_read(char *ptr, MARIA_UNIQUEDEF *def) +{ + def->keysegs = mi_uint2korr(ptr); + def->key = ptr[2]; + def->null_are_equal=ptr[3]; + return ptr+4; /* 1 extra byte */ +} + +/*************************************************************************** +** MARIA_COLUMNDEF +***************************************************************************/ + +uint _ma_recinfo_write(File file, MARIA_COLUMNDEF *recinfo) +{ + uchar buff[MARIA_COLUMNDEF_SIZE]; + uchar *ptr=buff; + + mi_int2store(ptr,recinfo->type); ptr +=2; + mi_int2store(ptr,recinfo->length); ptr +=2; + *ptr++ = recinfo->null_bit; + mi_int2store(ptr,recinfo->null_pos); ptr+= 2; + return my_write(file,(char*) buff, (uint) (ptr-buff), MYF(MY_NABP)); +} + +char *_ma_recinfo_read(char *ptr, MARIA_COLUMNDEF *recinfo) +{ + recinfo->type= mi_sint2korr(ptr); ptr +=2; + recinfo->length=mi_uint2korr(ptr); ptr +=2; + recinfo->null_bit= (uint8) *ptr++; + recinfo->null_pos=mi_uint2korr(ptr); ptr +=2; + return ptr; +} + +/************************************************************************** +Open data file with or without RAID +We can't use dup() here as the data file descriptors need to have different +active seek-positions. + +The argument file_to_dup is here for the future if there would on some OS +exist a dup()-like call that would give us two different file descriptors. +*************************************************************************/ + +int _ma_open_datafile(MARIA_HA *info, MARIA_SHARE *share, File file_to_dup __attribute__((unused))) +{ +#ifdef USE_RAID + if (share->base.raid_type) + { + info->dfile=my_raid_open(share->data_file_name, + share->mode | O_SHARE, + share->base.raid_type, + share->base.raid_chunks, + share->base.raid_chunksize, + MYF(MY_WME | MY_RAID)); + } + else +#endif + info->dfile=my_open(share->data_file_name, share->mode | O_SHARE, + MYF(MY_WME)); + return info->dfile >= 0 ? 0 : 1; +} + + +int _ma_open_keyfile(MARIA_SHARE *share) +{ + if ((share->kfile=my_open(share->unique_file_name, share->mode | O_SHARE, + MYF(MY_WME))) < 0) + return 1; + return 0; +} + + +/* + Disable all indexes. + + SYNOPSIS + maria_disable_indexes() + info A pointer to the MARIA storage engine MARIA_HA struct. + + DESCRIPTION + Disable all indexes. + + RETURN + 0 ok +*/ + +int maria_disable_indexes(MARIA_HA *info) +{ + MARIA_SHARE *share= info->s; + + maria_clear_all_keys_active(share->state.key_map); + return 0; +} + + +/* + Enable all indexes + + SYNOPSIS + maria_enable_indexes() + info A pointer to the MARIA storage engine MARIA_HA struct. + + DESCRIPTION + Enable all indexes. The indexes might have been disabled + by maria_disable_index() before. + The function works only if both data and indexes are empty, + otherwise a repair is required. + To be sure, call handler::delete_all_rows() before. + + RETURN + 0 ok + HA_ERR_CRASHED data or index is non-empty. +*/ + +int maria_enable_indexes(MARIA_HA *info) +{ + int error= 0; + MARIA_SHARE *share= info->s; + + if (share->state.state.data_file_length || + (share->state.state.key_file_length != share->base.keystart)) + { + maria_print_error(info->s, HA_ERR_CRASHED); + error= HA_ERR_CRASHED; + } + else + maria_set_all_keys_active(share->state.key_map, share->base.keys); + return error; +} + + +/* + Test if indexes are disabled. + + SYNOPSIS + maria_indexes_are_disabled() + info A pointer to the MARIA storage engine MARIA_HA struct. + + DESCRIPTION + Test if indexes are disabled. + + RETURN + 0 indexes are not disabled + 1 all indexes are disabled + [2 non-unique indexes are disabled - NOT YET IMPLEMENTED] +*/ + +int maria_indexes_are_disabled(MARIA_HA *info) +{ + MARIA_SHARE *share= info->s; + + return (! maria_is_any_key_active(share->state.key_map) && share->base.keys); +} diff --git a/storage/maria/ma_packrec.c b/storage/maria/ma_packrec.c new file mode 100644 index 00000000000..eb99e299f9a --- /dev/null +++ b/storage/maria/ma_packrec.c @@ -0,0 +1,1346 @@ +/* Copyright (C) 2006 MySQL AB & MySQL Finland AB & TCX DataKonsult AB + + 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; either version 2 of the License, or + (at your option) any later version. + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ + + /* Functions to compressed records */ + +#include "maria_def.h" + +#define IS_CHAR ((uint) 32768) /* Bit if char (not offset) in tree */ + +#if INT_MAX > 65536L +#define BITS_SAVED 32 +#define MAX_QUICK_TABLE_BITS 9 /* Because we may shift in 24 bits */ +#else +#define BITS_SAVED 16 +#define MAX_QUICK_TABLE_BITS 6 +#endif + +#define get_bit(BU) ((BU)->bits ? \ + (BU)->current_byte & ((maria_bit_type) 1 << --(BU)->bits) :\ + (fill_buffer(BU), (BU)->bits= BITS_SAVED-1,\ + (BU)->current_byte & ((maria_bit_type) 1 << (BITS_SAVED-1)))) +#define skip_to_next_byte(BU) ((BU)->bits&=~7) +#define get_bits(BU,count) (((BU)->bits >= count) ? (((BU)->current_byte >> ((BU)->bits-=count)) & mask[count]) : fill_and_get_bits(BU,count)) + +#define decode_bytes_test_bit(bit) \ + if (low_byte & (1 << (7-bit))) \ + pos++; \ + if (*pos & IS_CHAR) \ + { bits-=(bit+1); break; } \ + pos+= *pos + +#define OFFSET_TABLE_SIZE 512 + +static uint read_huff_table(MARIA_BIT_BUFF *bit_buff,MARIA_DECODE_TREE *decode_tree, + uint16 **decode_table,byte **intervall_buff, + uint16 *tmp_buff); +static void make_quick_table(uint16 *to_table,uint16 *decode_table, + uint *next_free,uint value,uint bits, + uint max_bits); +static void fill_quick_table(uint16 *table,uint bits, uint max_bits, + uint value); +static uint copy_decode_table(uint16 *to_pos,uint offset, + uint16 *decode_table); +static uint find_longest_bitstream(uint16 *table, uint16 *end); +static void (*get_unpack_function(MARIA_COLUMNDEF *rec))(MARIA_COLUMNDEF *field, + MARIA_BIT_BUFF *buff, + uchar *to, + uchar *end); +static void uf_zerofill_skip_zero(MARIA_COLUMNDEF *rec,MARIA_BIT_BUFF *bit_buff, + uchar *to,uchar *end); +static void uf_skip_zero(MARIA_COLUMNDEF *rec,MARIA_BIT_BUFF *bit_buff, + uchar *to,uchar *end); +static void uf_space_normal(MARIA_COLUMNDEF *rec,MARIA_BIT_BUFF *bit_buff, + uchar *to,uchar *end); +static void uf_space_endspace_selected(MARIA_COLUMNDEF *rec,MARIA_BIT_BUFF *bit_buff, + uchar *to, uchar *end); +static void uf_endspace_selected(MARIA_COLUMNDEF *rec,MARIA_BIT_BUFF *bit_buff, + uchar *to,uchar *end); +static void uf_space_endspace(MARIA_COLUMNDEF *rec,MARIA_BIT_BUFF *bit_buff, + uchar *to,uchar *end); +static void uf_endspace(MARIA_COLUMNDEF *rec,MARIA_BIT_BUFF *bit_buff, + uchar *to,uchar *end); +static void uf_space_prespace_selected(MARIA_COLUMNDEF *rec,MARIA_BIT_BUFF *bit_buff, + uchar *to, uchar *end); +static void uf_prespace_selected(MARIA_COLUMNDEF *rec,MARIA_BIT_BUFF *bit_buff, + uchar *to,uchar *end); +static void uf_space_prespace(MARIA_COLUMNDEF *rec,MARIA_BIT_BUFF *bit_buff, + uchar *to,uchar *end); +static void uf_prespace(MARIA_COLUMNDEF *rec,MARIA_BIT_BUFF *bit_buff, + uchar *to,uchar *end); +static void uf_zerofill_normal(MARIA_COLUMNDEF *rec,MARIA_BIT_BUFF *bit_buff, + uchar *to,uchar *end); +static void uf_constant(MARIA_COLUMNDEF *rec,MARIA_BIT_BUFF *bit_buff, + uchar *to,uchar *end); +static void uf_intervall(MARIA_COLUMNDEF *rec,MARIA_BIT_BUFF *bit_buff, + uchar *to,uchar *end); +static void uf_zero(MARIA_COLUMNDEF *rec,MARIA_BIT_BUFF *bit_buff, + uchar *to,uchar *end); +static void uf_blob(MARIA_COLUMNDEF *rec, MARIA_BIT_BUFF *bit_buff, + uchar *to, uchar *end); +static void uf_varchar1(MARIA_COLUMNDEF *rec, MARIA_BIT_BUFF *bit_buff, + uchar *to, uchar *end); +static void uf_varchar2(MARIA_COLUMNDEF *rec, MARIA_BIT_BUFF *bit_buff, + uchar *to, uchar *end); +static void decode_bytes(MARIA_COLUMNDEF *rec,MARIA_BIT_BUFF *bit_buff, + uchar *to,uchar *end); +static uint decode_pos(MARIA_BIT_BUFF *bit_buff,MARIA_DECODE_TREE *decode_tree); +static void init_bit_buffer(MARIA_BIT_BUFF *bit_buff,uchar *buffer,uint length); +static uint fill_and_get_bits(MARIA_BIT_BUFF *bit_buff,uint count); +static void fill_buffer(MARIA_BIT_BUFF *bit_buff); +static uint max_bit(uint value); +static uint read_pack_length(uint version, const uchar *buf, ulong *length); +#ifdef HAVE_MMAP +static uchar *_ma_mempack_get_block_info(MARIA_HA *maria,MARIA_BLOCK_INFO *info, + uchar *header); +#endif + +static maria_bit_type mask[]= +{ + 0x00000000, + 0x00000001, 0x00000003, 0x00000007, 0x0000000f, + 0x0000001f, 0x0000003f, 0x0000007f, 0x000000ff, + 0x000001ff, 0x000003ff, 0x000007ff, 0x00000fff, + 0x00001fff, 0x00003fff, 0x00007fff, 0x0000ffff, +#if BITS_SAVED > 16 + 0x0001ffff, 0x0003ffff, 0x0007ffff, 0x000fffff, + 0x001fffff, 0x003fffff, 0x007fffff, 0x00ffffff, + 0x01ffffff, 0x03ffffff, 0x07ffffff, 0x0fffffff, + 0x1fffffff, 0x3fffffff, 0x7fffffff, 0xffffffff, +#endif + }; + + + /* Read all packed info, allocate memory and fix field structs */ + +my_bool _ma_read_pack_info(MARIA_HA *info, pbool fix_keys) +{ + File file; + int diff_length; + uint i,trees,huff_tree_bits,rec_reflength,length; + uint16 *decode_table,*tmp_buff; + ulong elements,intervall_length; + char *disk_cache,*intervall_buff; + uchar header[32]; + MARIA_SHARE *share=info->s; + MARIA_BIT_BUFF bit_buff; + DBUG_ENTER("_ma_read_pack_info"); + + if (maria_quick_table_bits < 4) + maria_quick_table_bits=4; + else if (maria_quick_table_bits > MAX_QUICK_TABLE_BITS) + maria_quick_table_bits=MAX_QUICK_TABLE_BITS; + + file=info->dfile; + my_errno=0; + if (my_read(file,(byte*) header,sizeof(header),MYF(MY_NABP))) + { + if (!my_errno) + my_errno=HA_ERR_END_OF_FILE; + goto err0; + } + if (memcmp((byte*) header, (byte*) maria_pack_file_magic, 3)) + { + my_errno=HA_ERR_WRONG_IN_RECORD; + goto err0; + } + share->pack.version= header[3]; + share->pack.header_length= uint4korr(header+4); + share->min_pack_length=(uint) uint4korr(header+8); + share->max_pack_length=(uint) uint4korr(header+12); + set_if_bigger(share->base.pack_reclength,share->max_pack_length); + elements=uint4korr(header+16); + intervall_length=uint4korr(header+20); + trees=uint2korr(header+24); + share->pack.ref_length=header[26]; + rec_reflength=header[27]; + diff_length=(int) rec_reflength - (int) share->base.rec_reflength; + if (fix_keys) + share->rec_reflength=rec_reflength; + share->base.min_block_length=share->min_pack_length+1; + if (share->min_pack_length > 254) + share->base.min_block_length+=2; + + if (!(share->decode_trees=(MARIA_DECODE_TREE*) + my_malloc((uint) (trees*sizeof(MARIA_DECODE_TREE)+ + intervall_length*sizeof(byte)), + MYF(MY_WME)))) + goto err0; + intervall_buff=(byte*) (share->decode_trees+trees); + + length=(uint) (elements*2+trees*(1 << maria_quick_table_bits)); + if (!(share->decode_tables=(uint16*) + my_malloc((length+OFFSET_TABLE_SIZE)*sizeof(uint16)+ + (uint) (share->pack.header_length+7), + MYF(MY_WME | MY_ZEROFILL)))) + goto err1; + tmp_buff=share->decode_tables+length; + disk_cache=(byte*) (tmp_buff+OFFSET_TABLE_SIZE); + + if (my_read(file,disk_cache, + (uint) (share->pack.header_length-sizeof(header)), + MYF(MY_NABP))) + goto err2; + + huff_tree_bits=max_bit(trees ? trees-1 : 0); + init_bit_buffer(&bit_buff, (uchar*) disk_cache, + (uint) (share->pack.header_length-sizeof(header))); + /* Read new info for each field */ + for (i=0 ; i < share->base.fields ; i++) + { + share->rec[i].base_type=(enum en_fieldtype) get_bits(&bit_buff,5); + share->rec[i].pack_type=(uint) get_bits(&bit_buff,6); + share->rec[i].space_length_bits=get_bits(&bit_buff,5); + share->rec[i].huff_tree=share->decode_trees+(uint) get_bits(&bit_buff, + huff_tree_bits); + share->rec[i].unpack=get_unpack_function(share->rec+i); + } + skip_to_next_byte(&bit_buff); + decode_table=share->decode_tables; + for (i=0 ; i < trees ; i++) + if (read_huff_table(&bit_buff,share->decode_trees+i,&decode_table, + &intervall_buff,tmp_buff)) + goto err3; + decode_table=(uint16*) + my_realloc((gptr) share->decode_tables, + (uint) ((byte*) decode_table - (byte*) share->decode_tables), + MYF(MY_HOLD_ON_ERROR)); + { + long diff=PTR_BYTE_DIFF(decode_table,share->decode_tables); + share->decode_tables=decode_table; + for (i=0 ; i < trees ; i++) + share->decode_trees[i].table=ADD_TO_PTR(share->decode_trees[i].table, + diff, uint16*); + } + + /* Fix record-ref-length for keys */ + if (fix_keys) + { + for (i=0 ; i < share->base.keys ; i++) + { + share->keyinfo[i].keylength+=(uint16) diff_length; + share->keyinfo[i].minlength+=(uint16) diff_length; + share->keyinfo[i].maxlength+=(uint16) diff_length; + share->keyinfo[i].seg[share->keyinfo[i].keysegs].length= + (uint16) rec_reflength; + } + } + + if (bit_buff.error || bit_buff.pos < bit_buff.end) + goto err3; + + DBUG_RETURN(0); + +err3: + my_errno=HA_ERR_WRONG_IN_RECORD; +err2: + my_free((gptr) share->decode_tables,MYF(0)); +err1: + my_free((gptr) share->decode_trees,MYF(0)); +err0: + DBUG_RETURN(1); +} + + + /* Read on huff-code-table from datafile */ + +static uint read_huff_table(MARIA_BIT_BUFF *bit_buff, MARIA_DECODE_TREE *decode_tree, + uint16 **decode_table, byte **intervall_buff, + uint16 *tmp_buff) +{ + uint min_chr,elements,char_bits,offset_bits,size,intervall_length,table_bits, + next_free_offset; + uint16 *ptr,*end; + + LINT_INIT(ptr); + if (!get_bits(bit_buff,1)) + { + min_chr=get_bits(bit_buff,8); + elements=get_bits(bit_buff,9); + char_bits=get_bits(bit_buff,5); + offset_bits=get_bits(bit_buff,5); + intervall_length=0; + ptr=tmp_buff; + } + else + { + min_chr=0; + elements=get_bits(bit_buff,15); + intervall_length=get_bits(bit_buff,16); + char_bits=get_bits(bit_buff,5); + offset_bits=get_bits(bit_buff,5); + decode_tree->quick_table_bits=0; + ptr= *decode_table; + } + size=elements*2-2; + + for (end=ptr+size ; ptr < end ; ptr++) + { + if (get_bit(bit_buff)) + *ptr= (uint16) get_bits(bit_buff,offset_bits); + else + *ptr= (uint16) (IS_CHAR + (get_bits(bit_buff,char_bits) + min_chr)); + } + skip_to_next_byte(bit_buff); + + decode_tree->table= *decode_table; + decode_tree->intervalls= *intervall_buff; + if (! intervall_length) + { + table_bits=find_longest_bitstream(tmp_buff, tmp_buff+OFFSET_TABLE_SIZE); + if (table_bits == (uint) ~0) + return 1; + if (table_bits > maria_quick_table_bits) + table_bits=maria_quick_table_bits; + next_free_offset= (1 << table_bits); + make_quick_table(*decode_table,tmp_buff,&next_free_offset,0,table_bits, + table_bits); + (*decode_table)+= next_free_offset; + decode_tree->quick_table_bits=table_bits; + } + else + { + (*decode_table)=end; + bit_buff->pos-= bit_buff->bits/8; + memcpy(*intervall_buff,bit_buff->pos,(size_t) intervall_length); + (*intervall_buff)+=intervall_length; + bit_buff->pos+=intervall_length; + bit_buff->bits=0; + } + return 0; +} + + +static void make_quick_table(uint16 *to_table, uint16 *decode_table, + uint *next_free_offset, uint value, uint bits, + uint max_bits) +{ + if (!bits--) + { + to_table[value]= (uint16) *next_free_offset; + *next_free_offset=copy_decode_table(to_table, *next_free_offset, + decode_table); + return; + } + if (!(*decode_table & IS_CHAR)) + { + make_quick_table(to_table,decode_table+ *decode_table, + next_free_offset,value,bits,max_bits); + } + else + fill_quick_table(to_table+value,bits,max_bits,(uint) *decode_table); + decode_table++; + value|= (1 << bits); + if (!(*decode_table & IS_CHAR)) + { + make_quick_table(to_table,decode_table+ *decode_table, + next_free_offset,value,bits,max_bits); + } + else + fill_quick_table(to_table+value,bits,max_bits,(uint) *decode_table); + return; +} + + +static void fill_quick_table(uint16 *table, uint bits, uint max_bits, + uint value) +{ + uint16 *end; + value|=(max_bits-bits) << 8; + for (end=table+ (1 << bits) ; + table < end ; + *table++ = (uint16) value | IS_CHAR) ; +} + + +static uint copy_decode_table(uint16 *to_pos, uint offset, + uint16 *decode_table) +{ + uint prev_offset; + prev_offset= offset; + + if (!(*decode_table & IS_CHAR)) + { + to_pos[offset]=2; + offset=copy_decode_table(to_pos,offset+2,decode_table+ *decode_table); + } + else + { + to_pos[offset]= *decode_table; + offset+=2; + } + decode_table++; + + if (!(*decode_table & IS_CHAR)) + { + to_pos[prev_offset+1]=(uint16) (offset-prev_offset-1); + offset=copy_decode_table(to_pos,offset,decode_table+ *decode_table); + } + else + to_pos[prev_offset+1]= *decode_table; + return offset; +} + + +static uint find_longest_bitstream(uint16 *table, uint16 *end) +{ + uint length=1,length2; + if (!(*table & IS_CHAR)) + { + uint16 *next= table + *table; + if (next > end || next == table) + return ~0; + length=find_longest_bitstream(next, end)+1; + } + table++; + if (!(*table & IS_CHAR)) + { + uint16 *next= table + *table; + if (next > end || next == table) + return ~0; + length2=find_longest_bitstream(table+ *table, end)+1; + length=max(length,length2); + } + return length; +} + + +/* + Read record from datafile. + + SYNOPSIS + _ma_read_pack_record() + info A pointer to MARIA_HA. + filepos File offset of the record. + buf RETURN The buffer to receive the record. + + RETURN + 0 on success + HA_ERR_WRONG_IN_RECORD or -1 on error +*/ + +int _ma_read_pack_record(MARIA_HA *info, my_off_t filepos, byte *buf) +{ + MARIA_BLOCK_INFO block_info; + File file; + DBUG_ENTER("maria_read_pack_record"); + + if (filepos == HA_OFFSET_ERROR) + DBUG_RETURN(-1); /* _search() didn't find record */ + + file=info->dfile; + if (_ma_pack_get_block_info(info, &block_info, file, filepos)) + goto err; + if (my_read(file,(byte*) info->rec_buff + block_info.offset , + block_info.rec_len - block_info.offset, MYF(MY_NABP))) + goto panic; + info->update|= HA_STATE_AKTIV; + DBUG_RETURN(_ma_pack_rec_unpack(info,buf,info->rec_buff,block_info.rec_len)); +panic: + my_errno=HA_ERR_WRONG_IN_RECORD; +err: + DBUG_RETURN(-1); +} + + + +int _ma_pack_rec_unpack(register MARIA_HA *info, register byte *to, byte *from, + ulong reclength) +{ + byte *end_field; + reg3 MARIA_COLUMNDEF *end; + MARIA_COLUMNDEF *current_field; + MARIA_SHARE *share=info->s; + DBUG_ENTER("_ma_pack_rec_unpack"); + + init_bit_buffer(&info->bit_buff, (uchar*) from,reclength); + + for (current_field=share->rec, end=current_field+share->base.fields ; + current_field < end ; + current_field++,to=end_field) + { + end_field=to+current_field->length; + (*current_field->unpack)(current_field,&info->bit_buff,(uchar*) to, + (uchar*) end_field); + } + if (! info->bit_buff.error && + info->bit_buff.pos - info->bit_buff.bits/8 == info->bit_buff.end) + DBUG_RETURN(0); + info->update&= ~HA_STATE_AKTIV; + DBUG_RETURN(my_errno=HA_ERR_WRONG_IN_RECORD); +} /* _ma_pack_rec_unpack */ + + + /* Return function to unpack field */ + +static void (*get_unpack_function(MARIA_COLUMNDEF *rec)) +(MARIA_COLUMNDEF *, MARIA_BIT_BUFF *, uchar *, uchar *) +{ + switch (rec->base_type) { + case FIELD_SKIP_ZERO: + if (rec->pack_type & PACK_TYPE_ZERO_FILL) + return &uf_zerofill_skip_zero; + return &uf_skip_zero; + case FIELD_NORMAL: + if (rec->pack_type & PACK_TYPE_SPACE_FIELDS) + return &uf_space_normal; + if (rec->pack_type & PACK_TYPE_ZERO_FILL) + return &uf_zerofill_normal; + return &decode_bytes; + case FIELD_SKIP_ENDSPACE: + if (rec->pack_type & PACK_TYPE_SPACE_FIELDS) + { + if (rec->pack_type & PACK_TYPE_SELECTED) + return &uf_space_endspace_selected; + return &uf_space_endspace; + } + if (rec->pack_type & PACK_TYPE_SELECTED) + return &uf_endspace_selected; + return &uf_endspace; + case FIELD_SKIP_PRESPACE: + if (rec->pack_type & PACK_TYPE_SPACE_FIELDS) + { + if (rec->pack_type & PACK_TYPE_SELECTED) + return &uf_space_prespace_selected; + return &uf_space_prespace; + } + if (rec->pack_type & PACK_TYPE_SELECTED) + return &uf_prespace_selected; + return &uf_prespace; + case FIELD_CONSTANT: + return &uf_constant; + case FIELD_INTERVALL: + return &uf_intervall; + case FIELD_ZERO: + case FIELD_CHECK: + return &uf_zero; + case FIELD_BLOB: + return &uf_blob; + case FIELD_VARCHAR: + if (rec->length <= 256) /* 255 + 1 byte length */ + return &uf_varchar1; + return &uf_varchar2; + case FIELD_LAST: + default: + return 0; /* This should never happend */ + } +} + + /* The different functions to unpack a field */ + +static void uf_zerofill_skip_zero(MARIA_COLUMNDEF *rec, MARIA_BIT_BUFF *bit_buff, + uchar *to, uchar *end) +{ + if (get_bit(bit_buff)) + bzero((char*) to,(uint) (end-to)); + else + { + end-=rec->space_length_bits; + decode_bytes(rec,bit_buff,to,end); + bzero((char*) end,rec->space_length_bits); + } +} + +static void uf_skip_zero(MARIA_COLUMNDEF *rec, MARIA_BIT_BUFF *bit_buff, uchar *to, + uchar *end) +{ + if (get_bit(bit_buff)) + bzero((char*) to,(uint) (end-to)); + else + decode_bytes(rec,bit_buff,to,end); +} + +static void uf_space_normal(MARIA_COLUMNDEF *rec, MARIA_BIT_BUFF *bit_buff, uchar *to, + uchar *end) +{ + if (get_bit(bit_buff)) + bfill((byte*) to,(end-to),' '); + else + decode_bytes(rec,bit_buff,to,end); +} + +static void uf_space_endspace_selected(MARIA_COLUMNDEF *rec, MARIA_BIT_BUFF *bit_buff, + uchar *to, uchar *end) +{ + uint spaces; + if (get_bit(bit_buff)) + bfill((byte*) to,(end-to),' '); + else + { + if (get_bit(bit_buff)) + { + if ((spaces=get_bits(bit_buff,rec->space_length_bits))+to > end) + { + bit_buff->error=1; + return; + } + if (to+spaces != end) + decode_bytes(rec,bit_buff,to,end-spaces); + bfill((byte*) end-spaces,spaces,' '); + } + else + decode_bytes(rec,bit_buff,to,end); + } +} + +static void uf_endspace_selected(MARIA_COLUMNDEF *rec, MARIA_BIT_BUFF *bit_buff, + uchar *to, uchar *end) +{ + uint spaces; + if (get_bit(bit_buff)) + { + if ((spaces=get_bits(bit_buff,rec->space_length_bits))+to > end) + { + bit_buff->error=1; + return; + } + if (to+spaces != end) + decode_bytes(rec,bit_buff,to,end-spaces); + bfill((byte*) end-spaces,spaces,' '); + } + else + decode_bytes(rec,bit_buff,to,end); +} + +static void uf_space_endspace(MARIA_COLUMNDEF *rec, MARIA_BIT_BUFF *bit_buff, uchar *to, + uchar *end) +{ + uint spaces; + if (get_bit(bit_buff)) + bfill((byte*) to,(end-to),' '); + else + { + if ((spaces=get_bits(bit_buff,rec->space_length_bits))+to > end) + { + bit_buff->error=1; + return; + } + if (to+spaces != end) + decode_bytes(rec,bit_buff,to,end-spaces); + bfill((byte*) end-spaces,spaces,' '); + } +} + +static void uf_endspace(MARIA_COLUMNDEF *rec, MARIA_BIT_BUFF *bit_buff, uchar *to, + uchar *end) +{ + uint spaces; + if ((spaces=get_bits(bit_buff,rec->space_length_bits))+to > end) + { + bit_buff->error=1; + return; + } + if (to+spaces != end) + decode_bytes(rec,bit_buff,to,end-spaces); + bfill((byte*) end-spaces,spaces,' '); +} + +static void uf_space_prespace_selected(MARIA_COLUMNDEF *rec, MARIA_BIT_BUFF *bit_buff, + uchar *to, uchar *end) +{ + uint spaces; + if (get_bit(bit_buff)) + bfill((byte*) to,(end-to),' '); + else + { + if (get_bit(bit_buff)) + { + if ((spaces=get_bits(bit_buff,rec->space_length_bits))+to > end) + { + bit_buff->error=1; + return; + } + bfill((byte*) to,spaces,' '); + if (to+spaces != end) + decode_bytes(rec,bit_buff,to+spaces,end); + } + else + decode_bytes(rec,bit_buff,to,end); + } +} + + +static void uf_prespace_selected(MARIA_COLUMNDEF *rec, MARIA_BIT_BUFF *bit_buff, + uchar *to, uchar *end) +{ + uint spaces; + if (get_bit(bit_buff)) + { + if ((spaces=get_bits(bit_buff,rec->space_length_bits))+to > end) + { + bit_buff->error=1; + return; + } + bfill((byte*) to,spaces,' '); + if (to+spaces != end) + decode_bytes(rec,bit_buff,to+spaces,end); + } + else + decode_bytes(rec,bit_buff,to,end); +} + + +static void uf_space_prespace(MARIA_COLUMNDEF *rec, MARIA_BIT_BUFF *bit_buff, uchar *to, + uchar *end) +{ + uint spaces; + if (get_bit(bit_buff)) + bfill((byte*) to,(end-to),' '); + else + { + if ((spaces=get_bits(bit_buff,rec->space_length_bits))+to > end) + { + bit_buff->error=1; + return; + } + bfill((byte*) to,spaces,' '); + if (to+spaces != end) + decode_bytes(rec,bit_buff,to+spaces,end); + } +} + +static void uf_prespace(MARIA_COLUMNDEF *rec, MARIA_BIT_BUFF *bit_buff, uchar *to, + uchar *end) +{ + uint spaces; + if ((spaces=get_bits(bit_buff,rec->space_length_bits))+to > end) + { + bit_buff->error=1; + return; + } + bfill((byte*) to,spaces,' '); + if (to+spaces != end) + decode_bytes(rec,bit_buff,to+spaces,end); +} + +static void uf_zerofill_normal(MARIA_COLUMNDEF *rec, MARIA_BIT_BUFF *bit_buff, uchar *to, + uchar *end) +{ + end-=rec->space_length_bits; + decode_bytes(rec,bit_buff,(uchar*) to,end); + bzero((char*) end,rec->space_length_bits); +} + +static void uf_constant(MARIA_COLUMNDEF *rec, + MARIA_BIT_BUFF *bit_buff __attribute__((unused)), + uchar *to, + uchar *end) +{ + memcpy(to,rec->huff_tree->intervalls,(size_t) (end-to)); +} + +static void uf_intervall(MARIA_COLUMNDEF *rec, MARIA_BIT_BUFF *bit_buff, uchar *to, + uchar *end) +{ + reg1 uint field_length=(uint) (end-to); + memcpy(to,rec->huff_tree->intervalls+field_length*decode_pos(bit_buff, + rec->huff_tree), + (size_t) field_length); +} + + +/*ARGSUSED*/ +static void uf_zero(MARIA_COLUMNDEF *rec __attribute__((unused)), + MARIA_BIT_BUFF *bit_buff __attribute__((unused)), + uchar *to, uchar *end) +{ + bzero((char*) to,(uint) (end-to)); +} + +static void uf_blob(MARIA_COLUMNDEF *rec, MARIA_BIT_BUFF *bit_buff, + uchar *to, uchar *end) +{ + if (get_bit(bit_buff)) + bzero((byte*) to,(end-to)); + else + { + ulong length=get_bits(bit_buff,rec->space_length_bits); + uint pack_length=(uint) (end-to)-maria_portable_sizeof_char_ptr; + if (bit_buff->blob_pos+length > bit_buff->blob_end) + { + bit_buff->error=1; + bzero((byte*) to,(end-to)); + return; + } + decode_bytes(rec,bit_buff,bit_buff->blob_pos,bit_buff->blob_pos+length); + _ma_store_blob_length((byte*) to,pack_length,length); + memcpy_fixed((char*) to+pack_length,(char*) &bit_buff->blob_pos, + sizeof(char*)); + bit_buff->blob_pos+=length; + } +} + + +static void uf_varchar1(MARIA_COLUMNDEF *rec, MARIA_BIT_BUFF *bit_buff, + uchar *to, uchar *end __attribute__((unused))) +{ + if (get_bit(bit_buff)) + to[0]= 0; /* Zero lengths */ + else + { + ulong length=get_bits(bit_buff,rec->space_length_bits); + *to= (uchar) length; + decode_bytes(rec,bit_buff,to+1,to+1+length); + } +} + + +static void uf_varchar2(MARIA_COLUMNDEF *rec, MARIA_BIT_BUFF *bit_buff, + uchar *to, uchar *end __attribute__((unused))) +{ + if (get_bit(bit_buff)) + to[0]=to[1]=0; /* Zero lengths */ + else + { + ulong length=get_bits(bit_buff,rec->space_length_bits); + int2store(to,length); + decode_bytes(rec,bit_buff,to+2,to+2+length); + } +} + + /* Functions to decode of buffer of bits */ + +#if BITS_SAVED == 64 + +static void decode_bytes(MARIA_COLUMNDEF *rec,MARIA_BIT_BUFF *bit_buff,uchar *to, + uchar *end) +{ + reg1 uint bits,low_byte; + reg3 uint16 *pos; + reg4 uint table_bits,table_and; + MARIA_DECODE_TREE *decode_tree; + + decode_tree=rec->decode_tree; + bits=bit_buff->bits; /* Save in reg for quicker access */ + table_bits=decode_tree->quick_table_bits; + table_and= (1 << table_bits)-1; + + do + { + if (bits <= 32) + { + if (bit_buff->pos > bit_buff->end+4) + { + bit_buff->error=1; + return; /* Can't be right */ + } + bit_buff->current_byte= (bit_buff->current_byte << 32) + + ((((uint) bit_buff->pos[3])) + + (((uint) bit_buff->pos[2]) << 8) + + (((uint) bit_buff->pos[1]) << 16) + + (((uint) bit_buff->pos[0]) << 24)); + bit_buff->pos+=4; + bits+=32; + } + /* First use info in quick_table */ + low_byte=(uint) (bit_buff->current_byte >> (bits - table_bits)) & table_and; + low_byte=decode_tree->table[low_byte]; + if (low_byte & IS_CHAR) + { + *to++ = (low_byte & 255); /* Found char in quick table */ + bits-= ((low_byte >> 8) & 31); /* Remove bits used */ + } + else + { /* Map through rest of decode-table */ + pos=decode_tree->table+low_byte; + bits-=table_bits; + for (;;) + { + low_byte=(uint) (bit_buff->current_byte >> (bits-8)); + decode_bytes_test_bit(0); + decode_bytes_test_bit(1); + decode_bytes_test_bit(2); + decode_bytes_test_bit(3); + decode_bytes_test_bit(4); + decode_bytes_test_bit(5); + decode_bytes_test_bit(6); + decode_bytes_test_bit(7); + bits-=8; + } + *to++ = *pos; + } + } while (to != end); + + bit_buff->bits=bits; + return; +} + +#else + +static void decode_bytes(MARIA_COLUMNDEF *rec, MARIA_BIT_BUFF *bit_buff, uchar *to, + uchar *end) +{ + reg1 uint bits,low_byte; + reg3 uint16 *pos; + reg4 uint table_bits,table_and; + MARIA_DECODE_TREE *decode_tree; + + decode_tree=rec->huff_tree; + bits=bit_buff->bits; /* Save in reg for quicker access */ + table_bits=decode_tree->quick_table_bits; + table_and= (1 << table_bits)-1; + + do + { + if (bits < table_bits) + { + if (bit_buff->pos > bit_buff->end+1) + { + bit_buff->error=1; + return; /* Can't be right */ + } +#if BITS_SAVED == 32 + bit_buff->current_byte= (bit_buff->current_byte << 24) + + (((uint) ((uchar) bit_buff->pos[2]))) + + (((uint) ((uchar) bit_buff->pos[1])) << 8) + + (((uint) ((uchar) bit_buff->pos[0])) << 16); + bit_buff->pos+=3; + bits+=24; +#else + if (bits) /* We must have at leasts 9 bits */ + { + bit_buff->current_byte= (bit_buff->current_byte << 8) + + (uint) ((uchar) bit_buff->pos[0]); + bit_buff->pos++; + bits+=8; + } + else + { + bit_buff->current_byte= ((uint) ((uchar) bit_buff->pos[0]) << 8) + + ((uint) ((uchar) bit_buff->pos[1])); + bit_buff->pos+=2; + bits+=16; + } +#endif + } + /* First use info in quick_table */ + low_byte=(bit_buff->current_byte >> (bits - table_bits)) & table_and; + low_byte=decode_tree->table[low_byte]; + if (low_byte & IS_CHAR) + { + *to++ = (low_byte & 255); /* Found char in quick table */ + bits-= ((low_byte >> 8) & 31); /* Remove bits used */ + } + else + { /* Map through rest of decode-table */ + pos=decode_tree->table+low_byte; + bits-=table_bits; + for (;;) + { + if (bits < 8) + { /* We don't need to check end */ +#if BITS_SAVED == 32 + bit_buff->current_byte= (bit_buff->current_byte << 24) + + (((uint) ((uchar) bit_buff->pos[2]))) + + (((uint) ((uchar) bit_buff->pos[1])) << 8) + + (((uint) ((uchar) bit_buff->pos[0])) << 16); + bit_buff->pos+=3; + bits+=24; +#else + bit_buff->current_byte= (bit_buff->current_byte << 8) + + (uint) ((uchar) bit_buff->pos[0]); + bit_buff->pos+=1; + bits+=8; +#endif + } + low_byte=(uint) (bit_buff->current_byte >> (bits-8)); + decode_bytes_test_bit(0); + decode_bytes_test_bit(1); + decode_bytes_test_bit(2); + decode_bytes_test_bit(3); + decode_bytes_test_bit(4); + decode_bytes_test_bit(5); + decode_bytes_test_bit(6); + decode_bytes_test_bit(7); + bits-=8; + } + *to++ = (uchar) *pos; + } + } while (to != end); + + bit_buff->bits=bits; + return; +} +#endif /* BIT_SAVED == 64 */ + + +static uint decode_pos(MARIA_BIT_BUFF *bit_buff, MARIA_DECODE_TREE *decode_tree) +{ + uint16 *pos=decode_tree->table; + for (;;) + { + if (get_bit(bit_buff)) + pos++; + if (*pos & IS_CHAR) + return (uint) (*pos & ~IS_CHAR); + pos+= *pos; + } +} + + +int _ma_read_rnd_pack_record(MARIA_HA *info, byte *buf, + register my_off_t filepos, + my_bool skip_deleted_blocks) +{ + uint b_type; + MARIA_BLOCK_INFO block_info; + MARIA_SHARE *share=info->s; + DBUG_ENTER("_ma_read_rnd_pack_record"); + + if (filepos >= info->state->data_file_length) + { + my_errno= HA_ERR_END_OF_FILE; + goto err; + } + + if (info->opt_flag & READ_CACHE_USED) + { + if (_ma_read_cache(&info->rec_cache,(byte*) block_info.header,filepos, + share->pack.ref_length, skip_deleted_blocks)) + goto err; + b_type= _ma_pack_get_block_info(info,&block_info,-1, filepos); + } + else + b_type= _ma_pack_get_block_info(info,&block_info,info->dfile,filepos); + if (b_type) + goto err; /* Error code is already set */ +#ifndef DBUG_OFF + if (block_info.rec_len > share->max_pack_length) + { + my_errno=HA_ERR_WRONG_IN_RECORD; + goto err; + } +#endif + + if (info->opt_flag & READ_CACHE_USED) + { + if (_ma_read_cache(&info->rec_cache,(byte*) info->rec_buff, + block_info.filepos, block_info.rec_len, + skip_deleted_blocks)) + goto err; + } + else + { + if (my_read(info->dfile,(byte*) info->rec_buff + block_info.offset, + block_info.rec_len-block_info.offset, + MYF(MY_NABP))) + goto err; + } + info->packed_length=block_info.rec_len; + info->lastpos=filepos; + info->nextpos=block_info.filepos+block_info.rec_len; + info->update|= HA_STATE_AKTIV | HA_STATE_KEY_CHANGED; + + DBUG_RETURN (_ma_pack_rec_unpack(info,buf,info->rec_buff, + block_info.rec_len)); + err: + DBUG_RETURN(my_errno); +} + + + /* Read and process header from a huff-record-file */ + +uint _ma_pack_get_block_info(MARIA_HA *maria, MARIA_BLOCK_INFO *info, File file, + my_off_t filepos) +{ + uchar *header=info->header; + uint head_length,ref_length; + LINT_INIT(ref_length); + + if (file >= 0) + { + ref_length=maria->s->pack.ref_length; + /* + We can't use my_pread() here because maria_read_rnd_pack_record assumes + position is ok + */ + VOID(my_seek(file,filepos,MY_SEEK_SET,MYF(0))); + if (my_read(file,(char*) header,ref_length,MYF(MY_NABP))) + return BLOCK_FATAL_ERROR; + DBUG_DUMP("header",(byte*) header,ref_length); + } + head_length= read_pack_length((uint) maria->s->pack.version, header, + &info->rec_len); + if (maria->s->base.blobs) + { + head_length+= read_pack_length((uint) maria->s->pack.version, + header + head_length, &info->blob_len); + if (!(_ma_alloc_rec_buff(maria,info->rec_len + info->blob_len, + &maria->rec_buff))) + return BLOCK_FATAL_ERROR; /* not enough memory */ + maria->bit_buff.blob_pos=(uchar*) maria->rec_buff+info->rec_len; + maria->bit_buff.blob_end= maria->bit_buff.blob_pos+info->blob_len; + maria->blob_length=info->blob_len; + } + info->filepos=filepos+head_length; + if (file > 0) + { + info->offset=min(info->rec_len, ref_length - head_length); + memcpy(maria->rec_buff, header+head_length, info->offset); + } + return 0; +} + + + /* rutines for bit buffer */ + /* Note buffer must be 6 byte bigger than longest row */ + +static void init_bit_buffer(MARIA_BIT_BUFF *bit_buff, uchar *buffer, uint length) +{ + bit_buff->pos=buffer; + bit_buff->end=buffer+length; + bit_buff->bits=bit_buff->error=0; + bit_buff->current_byte=0; /* Avoid purify errors */ +} + +static uint fill_and_get_bits(MARIA_BIT_BUFF *bit_buff, uint count) +{ + uint tmp; + count-=bit_buff->bits; + tmp=(bit_buff->current_byte & mask[bit_buff->bits]) << count; + fill_buffer(bit_buff); + bit_buff->bits=BITS_SAVED - count; + return tmp+(bit_buff->current_byte >> (BITS_SAVED - count)); +} + + /* Fill in empty bit_buff->current_byte from buffer */ + /* Sets bit_buff->error if buffer is exhausted */ + +static void fill_buffer(MARIA_BIT_BUFF *bit_buff) +{ + if (bit_buff->pos >= bit_buff->end) + { + bit_buff->error= 1; + bit_buff->current_byte=0; + return; + } +#if BITS_SAVED == 64 + bit_buff->current_byte= ((((uint) ((uchar) bit_buff->pos[7]))) + + (((uint) ((uchar) bit_buff->pos[6])) << 8) + + (((uint) ((uchar) bit_buff->pos[5])) << 16) + + (((uint) ((uchar) bit_buff->pos[4])) << 24) + + ((ulonglong) + ((((uint) ((uchar) bit_buff->pos[3]))) + + (((uint) ((uchar) bit_buff->pos[2])) << 8) + + (((uint) ((uchar) bit_buff->pos[1])) << 16) + + (((uint) ((uchar) bit_buff->pos[0])) << 24)) << 32)); + bit_buff->pos+=8; +#else +#if BITS_SAVED == 32 + bit_buff->current_byte= (((uint) ((uchar) bit_buff->pos[3])) + + (((uint) ((uchar) bit_buff->pos[2])) << 8) + + (((uint) ((uchar) bit_buff->pos[1])) << 16) + + (((uint) ((uchar) bit_buff->pos[0])) << 24)); + bit_buff->pos+=4; +#else + bit_buff->current_byte= (uint) (((uint) ((uchar) bit_buff->pos[1]))+ + (((uint) ((uchar) bit_buff->pos[0])) << 8)); + bit_buff->pos+=2; +#endif +#endif +} + + /* Get number of bits neaded to represent value */ + +static uint max_bit(register uint value) +{ + reg2 uint power=1; + + while ((value>>=1)) + power++; + return (power); +} + + +/***************************************************************************** + Some redefined functions to handle files when we are using memmap +*****************************************************************************/ +#ifdef HAVE_SYS_MMAN_H +#include +#endif + +#ifdef HAVE_MMAP + +static int _ma_read_mempack_record(MARIA_HA *info,my_off_t filepos,byte *buf); +static int _ma_read_rnd_mempack_record(MARIA_HA*, byte *,my_off_t, my_bool); + +my_bool _ma_memmap_file(MARIA_HA *info) +{ + MARIA_SHARE *share=info->s; + DBUG_ENTER("maria_memmap_file"); + + if (!info->s->file_map) + { + if (my_seek(info->dfile,0L,MY_SEEK_END,MYF(0)) < + share->state.state.data_file_length+MEMMAP_EXTRA_MARGIN) + { + DBUG_PRINT("warning",("File isn't extended for memmap")); + DBUG_RETURN(0); + } + if (_ma_dynmap_file(info, share->state.state.data_file_length)) + DBUG_RETURN(0); + } + info->opt_flag|= MEMMAP_USED; + info->read_record= share->read_record= _ma_read_mempack_record; + share->read_rnd= _ma_read_rnd_mempack_record; + DBUG_RETURN(1); +} + + +void _ma_unmap_file(MARIA_HA *info) +{ + VOID(my_munmap(info->s->file_map, + (size_t) info->s->mmaped_length + MEMMAP_EXTRA_MARGIN)); +} + + +static uchar *_ma_mempack_get_block_info(MARIA_HA *maria,MARIA_BLOCK_INFO *info, + uchar *header) +{ + header+= read_pack_length((uint) maria->s->pack.version, header, + &info->rec_len); + if (maria->s->base.blobs) + { + header+= read_pack_length((uint) maria->s->pack.version, header, + &info->blob_len); + /* _ma_alloc_rec_buff sets my_errno on error */ + if (!(_ma_alloc_rec_buff(maria, info->blob_len, + &maria->rec_buff))) + return 0; /* not enough memory */ + maria->bit_buff.blob_pos=(uchar*) maria->rec_buff; + maria->bit_buff.blob_end= (uchar*) maria->rec_buff + info->blob_len; + } + return header; +} + + +static int _ma_read_mempack_record(MARIA_HA *info, my_off_t filepos, byte *buf) +{ + MARIA_BLOCK_INFO block_info; + MARIA_SHARE *share=info->s; + byte *pos; + DBUG_ENTER("maria_read_mempack_record"); + + if (filepos == HA_OFFSET_ERROR) + DBUG_RETURN(-1); /* _search() didn't find record */ + + if (!(pos= (byte*) _ma_mempack_get_block_info(info,&block_info, + (uchar*) share->file_map+ + filepos))) + DBUG_RETURN(-1); + DBUG_RETURN(_ma_pack_rec_unpack(info, buf, pos, block_info.rec_len)); +} + + +/*ARGSUSED*/ +static int _ma_read_rnd_mempack_record(MARIA_HA *info, byte *buf, + register my_off_t filepos, + my_bool skip_deleted_blocks + __attribute__((unused))) +{ + MARIA_BLOCK_INFO block_info; + MARIA_SHARE *share=info->s; + byte *pos,*start; + DBUG_ENTER("_ma_read_rnd_mempack_record"); + + if (filepos >= share->state.state.data_file_length) + { + my_errno=HA_ERR_END_OF_FILE; + goto err; + } + if (!(pos= (byte*) _ma_mempack_get_block_info(info,&block_info, + (uchar*) + (start=share->file_map+ + filepos)))) + goto err; +#ifndef DBUG_OFF + if (block_info.rec_len > info->s->max_pack_length) + { + my_errno=HA_ERR_WRONG_IN_RECORD; + goto err; + } +#endif + info->packed_length=block_info.rec_len; + info->lastpos=filepos; + info->nextpos=filepos+(uint) (pos-start)+block_info.rec_len; + info->update|= HA_STATE_AKTIV | HA_STATE_KEY_CHANGED; + + DBUG_RETURN (_ma_pack_rec_unpack(info,buf,pos, block_info.rec_len)); + err: + DBUG_RETURN(my_errno); +} + +#endif /* HAVE_MMAP */ + + /* Save length of row */ + +uint _ma_save_pack_length(uint version, byte *block_buff, ulong length) +{ + if (length < 254) + { + *(uchar*) block_buff= (uchar) length; + return 1; + } + if (length <= 65535) + { + *(uchar*) block_buff=254; + int2store(block_buff+1,(uint) length); + return 3; + } + *(uchar*) block_buff=255; + if (version == 1) /* old format */ + { + DBUG_ASSERT(length <= 0xFFFFFF); + int3store(block_buff + 1, (ulong) length); + return 4; + } + else + { + int4store(block_buff + 1, (ulong) length); + return 5; + } +} + + +static uint read_pack_length(uint version, const uchar *buf, ulong *length) +{ + if (buf[0] < 254) + { + *length= buf[0]; + return 1; + } + else if (buf[0] == 254) + { + *length= uint2korr(buf + 1); + return 3; + } + if (version == 1) /* old format */ + { + *length= uint3korr(buf + 1); + return 4; + } + else + { + *length= uint4korr(buf + 1); + return 5; + } +} + + +uint _ma_calc_pack_length(uint version, ulong length) +{ + return (length < 254) ? 1 : (length < 65536) ? 3 : (version == 1) ? 4 : 5; +} diff --git a/storage/maria/ma_page.c b/storage/maria/ma_page.c new file mode 100644 index 00000000000..78864e2c9ac --- /dev/null +++ b/storage/maria/ma_page.c @@ -0,0 +1,160 @@ +/* Copyright (C) 2006 MySQL AB & MySQL Finland AB & TCX DataKonsult AB + + 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; either version 2 of the License, or + (at your option) any later version. + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ + +/* Read and write key blocks */ + +#include "maria_def.h" + + /* Fetch a key-page in memory */ + +uchar *_ma_fetch_keypage(register MARIA_HA *info, MARIA_KEYDEF *keyinfo, + my_off_t page, int level, + uchar *buff, int return_buffer) +{ + uchar *tmp; + uint page_size; + DBUG_ENTER("_ma_fetch_keypage"); + DBUG_PRINT("enter",("page: %ld",page)); + + tmp=(uchar*) key_cache_read(info->s->key_cache, + info->s->kfile, page, level, (byte*) buff, + (uint) keyinfo->block_length, + (uint) keyinfo->block_length, + return_buffer); + if (tmp == info->buff) + info->buff_used=1; + else if (!tmp) + { + DBUG_PRINT("error",("Got errno: %d from key_cache_read",my_errno)); + info->last_keypage=HA_OFFSET_ERROR; + maria_print_error(info->s, HA_ERR_CRASHED); + my_errno=HA_ERR_CRASHED; + DBUG_RETURN(0); + } + info->last_keypage=page; + page_size=maria_getint(tmp); + if (page_size < 4 || page_size > keyinfo->block_length) + { + DBUG_PRINT("error",("page %lu had wrong page length: %u", + (ulong) page, page_size)); + DBUG_DUMP("page", (char*) tmp, keyinfo->block_length); + info->last_keypage = HA_OFFSET_ERROR; + maria_print_error(info->s, HA_ERR_CRASHED); + my_errno = HA_ERR_CRASHED; + tmp = 0; + } + DBUG_RETURN(tmp); +} /* _ma_fetch_keypage */ + + + /* Write a key-page on disk */ + +int _ma_write_keypage(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, + my_off_t page, int level, uchar *buff) +{ + reg3 uint length; + DBUG_ENTER("_ma_write_keypage"); + +#ifndef FAST /* Safety check */ + if (page < info->s->base.keystart || + page+keyinfo->block_length > info->state->key_file_length || + (page & (MARIA_MIN_KEY_BLOCK_LENGTH-1))) + { + DBUG_PRINT("error",("Trying to write inside key status region: key_start: %lu length: %lu page: %lu", + (long) info->s->base.keystart, + (long) info->state->key_file_length, + (long) page)); + my_errno=EINVAL; + DBUG_RETURN((-1)); + } + DBUG_PRINT("page",("write page at: %lu",(long) page,buff)); + DBUG_DUMP("buff",(byte*) buff,maria_getint(buff)); +#endif + + if ((length=keyinfo->block_length) > IO_SIZE*2 && + info->state->key_file_length != page+length) + length= ((maria_getint(buff)+IO_SIZE-1) & (uint) ~(IO_SIZE-1)); +#ifdef HAVE_purify + { + length=maria_getint(buff); + bzero((byte*) buff+length,keyinfo->block_length-length); + length=keyinfo->block_length; + } +#endif + DBUG_RETURN((key_cache_write(info->s->key_cache, + info->s->kfile,page, level, (byte*) buff,length, + (uint) keyinfo->block_length, + (int) ((info->lock_type != F_UNLCK) || + info->s->delay_key_write)))); +} /* maria_write_keypage */ + + + /* Remove page from disk */ + +int _ma_dispose(register MARIA_HA *info, MARIA_KEYDEF *keyinfo, my_off_t pos, + int level) +{ + my_off_t old_link; + char buff[8]; + DBUG_ENTER("_ma_dispose"); + DBUG_PRINT("enter",("pos: %ld", (long) pos)); + + old_link=info->s->state.key_del[keyinfo->block_size]; + info->s->state.key_del[keyinfo->block_size]=pos; + mi_sizestore(buff,old_link); + info->s->state.changed|= STATE_NOT_SORTED_PAGES; + DBUG_RETURN(key_cache_write(info->s->key_cache, + info->s->kfile, pos , level, buff, + sizeof(buff), + (uint) keyinfo->block_length, + (int) (info->lock_type != F_UNLCK))); +} /* _ma_dispose */ + + + /* Make new page on disk */ + +my_off_t _ma_new(register MARIA_HA *info, MARIA_KEYDEF *keyinfo, int level) +{ + my_off_t pos; + char buff[8]; + DBUG_ENTER("_ma_new"); + + if ((pos=info->s->state.key_del[keyinfo->block_size]) == HA_OFFSET_ERROR) + { + if (info->state->key_file_length >= + info->s->base.max_key_file_length - keyinfo->block_length) + { + my_errno=HA_ERR_INDEX_FILE_FULL; + DBUG_RETURN(HA_OFFSET_ERROR); + } + pos=info->state->key_file_length; + info->state->key_file_length+= keyinfo->block_length; + } + else + { + if (!key_cache_read(info->s->key_cache, + info->s->kfile, pos, level, + buff, + (uint) sizeof(buff), + (uint) keyinfo->block_length,0)) + pos= HA_OFFSET_ERROR; + else + info->s->state.key_del[keyinfo->block_size]=mi_sizekorr(buff); + } + info->s->state.changed|= STATE_NOT_SORTED_PAGES; + DBUG_PRINT("exit",("Pos: %ld",(long) pos)); + DBUG_RETURN(pos); +} /* _ma_new */ diff --git a/storage/maria/ma_panic.c b/storage/maria/ma_panic.c new file mode 100644 index 00000000000..90239b3943b --- /dev/null +++ b/storage/maria/ma_panic.c @@ -0,0 +1,124 @@ +/* Copyright (C) 2006 MySQL AB & MySQL Finland AB & TCX DataKonsult AB + + 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; either version 2 of the License, or + (at your option) any later version. + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ + +#include "ma_fulltext.h" + +/* + Stop usage of Maria + + SYNOPSIS + maria_panic() + flag HA_PANIC_CLOSE: All maria files (tables and log) are closed. + maria_end() is called. + HA_PANIC_WRITE: All misam files are unlocked and + all changed data in single user maria is + written to file + HA_PANIC_READ All maria files that was locked when + maria_panic(HA_PANIC_WRITE) was done is + locked. A maria_readinfo() is done for + all single user files to get changes + in database + + RETURN + 0 ok + # error number in case of error +*/ + +int maria_panic(enum ha_panic_function flag) +{ + int error=0; + LIST *list_element,*next_open; + MARIA_HA *info; + DBUG_ENTER("maria_panic"); + + pthread_mutex_lock(&THR_LOCK_maria); + for (list_element=maria_open_list ; list_element ; list_element=next_open) + { + next_open=list_element->next; /* Save if close */ + info=(MARIA_HA*) list_element->data; + switch (flag) { + case HA_PANIC_CLOSE: + pthread_mutex_unlock(&THR_LOCK_maria); /* Not exactly right... */ + if (maria_close(info)) + error=my_errno; + pthread_mutex_lock(&THR_LOCK_maria); + break; + case HA_PANIC_WRITE: /* Do this to free databases */ +#ifdef CANT_OPEN_FILES_TWICE + if (info->s->options & HA_OPTION_READ_ONLY_DATA) + break; +#endif + if (flush_key_blocks(info->s->key_cache, info->s->kfile, FLUSH_RELEASE)) + error=my_errno; + if (info->opt_flag & WRITE_CACHE_USED) + if (flush_io_cache(&info->rec_cache)) + error=my_errno; + if (info->opt_flag & READ_CACHE_USED) + { + if (flush_io_cache(&info->rec_cache)) + error=my_errno; + reinit_io_cache(&info->rec_cache,READ_CACHE,0, + (pbool) (info->lock_type != F_UNLCK),1); + } + if (info->lock_type != F_UNLCK && ! info->was_locked) + { + info->was_locked=info->lock_type; + if (maria_lock_database(info,F_UNLCK)) + error=my_errno; + } +#ifdef CANT_OPEN_FILES_TWICE + if (info->s->kfile >= 0 && my_close(info->s->kfile,MYF(0))) + error = my_errno; + if (info->dfile >= 0 && my_close(info->dfile,MYF(0))) + error = my_errno; + info->s->kfile=info->dfile= -1; /* Files aren't open anymore */ + break; +#endif + case HA_PANIC_READ: /* Restore to before WRITE */ +#ifdef CANT_OPEN_FILES_TWICE + { /* Open closed files */ + char name_buff[FN_REFLEN]; + if (info->s->kfile < 0) + if ((info->s->kfile= my_open(fn_format(name_buff,info->filename,"", + N_NAME_IEXT,4),info->mode, + MYF(MY_WME))) < 0) + error = my_errno; + if (info->dfile < 0) + { + if ((info->dfile= my_open(fn_format(name_buff,info->filename,"", + N_NAME_DEXT,4),info->mode, + MYF(MY_WME))) < 0) + error = my_errno; + info->rec_cache.file=info->dfile; + } + } +#endif + if (info->was_locked) + { + if (maria_lock_database(info, info->was_locked)) + error=my_errno; + info->was_locked=0; + } + break; + } + } + pthread_mutex_unlock(&THR_LOCK_maria); + if (flag == HA_PANIC_CLOSE) + maria_end(); + if (!error) + DBUG_RETURN(0); + DBUG_RETURN(my_errno=error); +} /* maria_panic */ diff --git a/storage/maria/ma_preload.c b/storage/maria/ma_preload.c new file mode 100644 index 00000000000..f387f2b7de3 --- /dev/null +++ b/storage/maria/ma_preload.c @@ -0,0 +1,117 @@ +/* Copyright (C) 2006 MySQL AB & MySQL Finland AB & TCX DataKonsult AB + + 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; either version 2 of the License, or + (at your option) any later version. + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ + +/* + Preload indexes into key cache +*/ + +#include "maria_def.h" + + +/* + Preload pages of the index file for a table into the key cache + + SYNOPSIS + maria_preload() + info open table + map map of indexes to preload into key cache + ignore_leaves only non-leaves pages are to be preloaded + + RETURN VALUE + 0 if a success. error code - otherwise. + + NOTES. + At present pages for all indexes are preloaded. + In future only pages for indexes specified in the key_map parameter + of the table will be preloaded. +*/ + +int maria_preload(MARIA_HA *info, ulonglong key_map, my_bool ignore_leaves) +{ + uint i; + ulong length, block_length= 0; + uchar *buff= NULL; + MARIA_SHARE* share= info->s; + uint keys= share->state.header.keys; + MARIA_KEYDEF *keyinfo= share->keyinfo; + my_off_t key_file_length= share->state.state.key_file_length; + my_off_t pos= share->base.keystart; + DBUG_ENTER("maria_preload"); + + if (!keys || !maria_is_any_key_active(key_map) || key_file_length == pos) + DBUG_RETURN(0); + + block_length= keyinfo[0].block_length; + + /* Check whether all indexes use the same block size */ + for (i= 1 ; i < keys ; i++) + { + if (keyinfo[i].block_length != block_length) + DBUG_RETURN(my_errno= HA_ERR_NON_UNIQUE_BLOCK_SIZE); + } + + length= info->preload_buff_size/block_length * block_length; + set_if_bigger(length, block_length); + + if (!(buff= (uchar *) my_malloc(length, MYF(MY_WME)))) + DBUG_RETURN(my_errno= HA_ERR_OUT_OF_MEM); + + if (flush_key_blocks(share->key_cache,share->kfile, FLUSH_RELEASE)) + goto err; + + do + { + /* Read the next block of index file into the preload buffer */ + if ((my_off_t) length > (key_file_length-pos)) + length= (ulong) (key_file_length-pos); + if (my_pread(share->kfile, (byte*) buff, length, pos, MYF(MY_FAE|MY_FNABP))) + goto err; + + if (ignore_leaves) + { + uchar *end= buff+length; + do + { + if (_ma_test_if_nod(buff)) + { + if (key_cache_insert(share->key_cache, + share->kfile, pos, DFLT_INIT_HITS, + (byte*) buff, block_length)) + goto err; + } + pos+= block_length; + } + while ((buff+= block_length) != end); + buff= end-length; + } + else + { + if (key_cache_insert(share->key_cache, + share->kfile, pos, DFLT_INIT_HITS, + (byte*) buff, length)) + goto err; + pos+= length; + } + } + while (pos != key_file_length); + + my_free((char*) buff, MYF(0)); + DBUG_RETURN(0); + +err: + my_free((char*) buff, MYF(MY_ALLOW_ZERO_PTR)); + DBUG_RETURN(my_errno= errno); +} diff --git a/storage/maria/ma_range.c b/storage/maria/ma_range.c new file mode 100644 index 00000000000..0f6883f4c9d --- /dev/null +++ b/storage/maria/ma_range.c @@ -0,0 +1,244 @@ +/* Copyright (C) 2006 MySQL AB & MySQL Finland AB & TCX DataKonsult AB + + 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; either version 2 of the License, or + (at your option) any later version. + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ + +/* + Gives a approximated number of how many records there is between two keys. + Used when optimizing querries. + */ + +#include "maria_def.h" +#include "ma_rt_index.h" + +static ha_rows _ma_record_pos(MARIA_HA *info,const byte *key,uint key_len, + enum ha_rkey_function search_flag); +static double _ma_search_pos(MARIA_HA *info,MARIA_KEYDEF *keyinfo,uchar *key, + uint key_len,uint nextflag,my_off_t pos); +static uint _ma_keynr(MARIA_HA *info,MARIA_KEYDEF *keyinfo,uchar *page, + uchar *keypos,uint *ret_max_key); + + +/* + Estimate how many records there is in a given range + + SYNOPSIS + maria_records_in_range() + info MARIA handler + inx Index to use + min_key Min key. Is = 0 if no min range + max_key Max key. Is = 0 if no max range + + NOTES + We should ONLY return 0 if there is no rows in range + + RETURN + HA_POS_ERROR error (or we can't estimate number of rows) + number Estimated number of rows +*/ + + +ha_rows maria_records_in_range(MARIA_HA *info, int inx, key_range *min_key, + key_range *max_key) +{ + ha_rows start_pos,end_pos,res; + DBUG_ENTER("maria_records_in_range"); + + if ((inx = _ma_check_index(info,inx)) < 0) + DBUG_RETURN(HA_POS_ERROR); + + if (fast_ma_readinfo(info)) + DBUG_RETURN(HA_POS_ERROR); + info->update&= (HA_STATE_CHANGED+HA_STATE_ROW_CHANGED); + if (info->s->concurrent_insert) + rw_rdlock(&info->s->key_root_lock[inx]); + + switch(info->s->keyinfo[inx].key_alg){ +#ifdef HAVE_RTREE_KEYS + case HA_KEY_ALG_RTREE: + { + uchar * key_buff; + uint start_key_len; + + key_buff= info->lastkey+info->s->base.max_key_length; + start_key_len= _ma_pack_key(info,inx, key_buff, + (uchar*) min_key->key, min_key->length, + (HA_KEYSEG**) 0); + res= maria_rtree_estimate(info, inx, key_buff, start_key_len, + maria_read_vec[min_key->flag]); + res= res ? res : 1; /* Don't return 0 */ + break; + } +#endif + case HA_KEY_ALG_BTREE: + default: + start_pos= (min_key ? + _ma_record_pos(info, min_key->key, min_key->length, + min_key->flag) : + (ha_rows) 0); + end_pos= (max_key ? + _ma_record_pos(info, max_key->key, max_key->length, + max_key->flag) : + info->state->records+ (ha_rows) 1); + res= (end_pos < start_pos ? (ha_rows) 0 : + (end_pos == start_pos ? (ha_rows) 1 : end_pos-start_pos)); + if (start_pos == HA_POS_ERROR || end_pos == HA_POS_ERROR) + res=HA_POS_ERROR; + } + + if (info->s->concurrent_insert) + rw_unlock(&info->s->key_root_lock[inx]); + fast_ma_writeinfo(info); + + DBUG_PRINT("info",("records: %ld",(ulong) (res))); + DBUG_RETURN(res); +} + + + /* Find relative position (in records) for key in index-tree */ + +static ha_rows _ma_record_pos(MARIA_HA *info, const byte *key, uint key_len, + enum ha_rkey_function search_flag) +{ + uint inx=(uint) info->lastinx, nextflag; + MARIA_KEYDEF *keyinfo=info->s->keyinfo+inx; + uchar *key_buff; + double pos; + + DBUG_ENTER("_ma_record_pos"); + DBUG_PRINT("enter",("search_flag: %d",search_flag)); + + if (key_len == 0) + key_len=USE_WHOLE_KEY; + key_buff=info->lastkey+info->s->base.max_key_length; + key_len= _ma_pack_key(info,inx,key_buff,(uchar*) key,key_len, + (HA_KEYSEG**) 0); + DBUG_EXECUTE("key", _ma_print_key(DBUG_FILE,keyinfo->seg, + (uchar*) key_buff,key_len);); + nextflag=maria_read_vec[search_flag]; + if (!(nextflag & (SEARCH_FIND | SEARCH_NO_FIND | SEARCH_LAST))) + key_len=USE_WHOLE_KEY; + + pos= _ma_search_pos(info,keyinfo,key_buff,key_len, + nextflag | SEARCH_SAVE_BUFF, + info->s->state.key_root[inx]); + if (pos >= 0.0) + { + DBUG_PRINT("exit",("pos: %ld",(ulong) (pos*info->state->records))); + DBUG_RETURN((ulong) (pos*info->state->records+0.5)); + } + DBUG_RETURN(HA_POS_ERROR); +} + + + /* This is a modified version of _ma_search */ + /* Returns offset for key in indextable (decimal 0.0 <= x <= 1.0) */ + +static double _ma_search_pos(register MARIA_HA *info, + register MARIA_KEYDEF *keyinfo, + uchar *key, uint key_len, uint nextflag, + register my_off_t pos) +{ + int flag; + uint nod_flag,keynr,max_keynr; + my_bool after_key; + uchar *keypos,*buff; + double offset; + DBUG_ENTER("_ma_search_pos"); + + if (pos == HA_OFFSET_ERROR) + DBUG_RETURN(0.5); + + if (!(buff= _ma_fetch_keypage(info,keyinfo,pos,DFLT_INIT_HITS,info->buff,1))) + goto err; + flag=(*keyinfo->bin_search)(info,keyinfo,buff,key,key_len,nextflag, + &keypos,info->lastkey, &after_key); + nod_flag=_ma_test_if_nod(buff); + keynr= _ma_keynr(info,keyinfo,buff,keypos,&max_keynr); + + if (flag) + { + if (flag == MARIA_FOUND_WRONG_KEY) + DBUG_RETURN(-1); /* error */ + /* + Didn't found match. keypos points at next (bigger) key + Try to find a smaller, better matching key. + Matches keynr + [0-1] + */ + if (flag > 0 && ! nod_flag) + offset= 1.0; + else if ((offset= _ma_search_pos(info,keyinfo,key,key_len,nextflag, + _ma_kpos(nod_flag,keypos))) < 0) + DBUG_RETURN(offset); + } + else + { + /* + Found match. Keypos points at the start of the found key + Matches keynr+1 + */ + offset=1.0; /* Matches keynr+1 */ + if ((nextflag & SEARCH_FIND) && nod_flag && + ((keyinfo->flag & (HA_NOSAME | HA_NULL_PART)) != HA_NOSAME || + key_len != USE_WHOLE_KEY)) + { + /* + There may be identical keys in the tree. Try to match on of those. + Matches keynr + [0-1] + */ + if ((offset= _ma_search_pos(info,keyinfo,key,key_len,SEARCH_FIND, + _ma_kpos(nod_flag,keypos))) < 0) + DBUG_RETURN(offset); /* Read error */ + } + } + DBUG_PRINT("info",("keynr: %d offset: %g max_keynr: %d nod: %d flag: %d", + keynr,offset,max_keynr,nod_flag,flag)); + DBUG_RETURN((keynr+offset)/(max_keynr+1)); +err: + DBUG_PRINT("exit",("Error: %d",my_errno)); + DBUG_RETURN (-1.0); +} + + + /* Get keynummer of current key and max number of keys in nod */ + +static uint _ma_keynr(MARIA_HA *info, register MARIA_KEYDEF *keyinfo, uchar *page, + uchar *keypos, uint *ret_max_key) +{ + uint nod_flag,keynr,max_key; + uchar t_buff[HA_MAX_KEY_BUFF],*end; + + end= page+maria_getint(page); + nod_flag=_ma_test_if_nod(page); + page+=2+nod_flag; + + if (!(keyinfo->flag & (HA_VAR_LENGTH_KEY | HA_BINARY_PACK_KEY))) + { + *ret_max_key= (uint) (end-page)/(keyinfo->keylength+nod_flag); + return (uint) (keypos-page)/(keyinfo->keylength+nod_flag); + } + + max_key=keynr=0; + t_buff[0]=0; /* Safety */ + while (page < end) + { + if (!(*keyinfo->get_key)(keyinfo,nod_flag,&page,t_buff)) + return 0; /* Error */ + max_key++; + if (page == keypos) + keynr=max_key; + } + *ret_max_key=max_key; + return(keynr); +} diff --git a/storage/maria/ma_rename.c b/storage/maria/ma_rename.c new file mode 100644 index 00000000000..5f65cd2b213 --- /dev/null +++ b/storage/maria/ma_rename.c @@ -0,0 +1,61 @@ +/* Copyright (C) 2006 MySQL AB & MySQL Finland AB & TCX DataKonsult AB + + 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; either version 2 of the License, or + (at your option) any later version. + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ + +/* + Rename a table +*/ + +#include "ma_fulltext.h" + +int maria_rename(const char *old_name, const char *new_name) +{ + char from[FN_REFLEN],to[FN_REFLEN]; +#ifdef USE_RAID + uint raid_type=0,raid_chunks=0; +#endif + DBUG_ENTER("maria_rename"); + +#ifdef EXTRA_DEBUG + _ma_check_table_is_closed(old_name,"rename old_table"); + _ma_check_table_is_closed(new_name,"rename new table2"); +#endif +#ifdef USE_RAID + { + MARIA_HA *info; + if (!(info=maria_open(old_name, O_RDONLY, 0))) + DBUG_RETURN(my_errno); + raid_type = info->s->base.raid_type; + raid_chunks = info->s->base.raid_chunks; + maria_close(info); + } +#ifdef EXTRA_DEBUG + _ma_check_table_is_closed(old_name,"rename raidcheck"); +#endif +#endif /* USE_RAID */ + + fn_format(from,old_name,"",MARIA_NAME_IEXT,MY_UNPACK_FILENAME|MY_APPEND_EXT); + fn_format(to,new_name,"",MARIA_NAME_IEXT,MY_UNPACK_FILENAME|MY_APPEND_EXT); + if (my_rename_with_symlink(from, to, MYF(MY_WME))) + DBUG_RETURN(my_errno); + fn_format(from,old_name,"",MARIA_NAME_DEXT,MY_UNPACK_FILENAME|MY_APPEND_EXT); + fn_format(to,new_name,"",MARIA_NAME_DEXT,MY_UNPACK_FILENAME|MY_APPEND_EXT); +#ifdef USE_RAID + if (raid_type) + DBUG_RETURN(my_raid_rename(from, to, raid_chunks, MYF(MY_WME)) ? my_errno : + 0); +#endif + DBUG_RETURN(my_rename_with_symlink(from, to,MYF(MY_WME)) ? my_errno : 0); +} diff --git a/storage/maria/ma_rfirst.c b/storage/maria/ma_rfirst.c new file mode 100644 index 00000000000..503e8989936 --- /dev/null +++ b/storage/maria/ma_rfirst.c @@ -0,0 +1,27 @@ +/* Copyright (C) 2006 MySQL AB & MySQL Finland AB & TCX DataKonsult AB + + 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; either version 2 of the License, or + (at your option) any later version. + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ + +#include "maria_def.h" + + /* Read first row through a specfic key */ + +int maria_rfirst(MARIA_HA *info, byte *buf, int inx) +{ + DBUG_ENTER("maria_rfirst"); + info->lastpos= HA_OFFSET_ERROR; + info->update|= HA_STATE_PREV_FOUND; + DBUG_RETURN(maria_rnext(info,buf,inx)); +} /* maria_rfirst */ diff --git a/storage/maria/ma_rkey.c b/storage/maria/ma_rkey.c new file mode 100644 index 00000000000..abcce1a2582 --- /dev/null +++ b/storage/maria/ma_rkey.c @@ -0,0 +1,144 @@ +/* Copyright (C) 2006 MySQL AB & MySQL Finland AB & TCX DataKonsult AB + + 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; either version 2 of the License, or + (at your option) any later version. + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ + +/* Read record based on a key */ + +#include "maria_def.h" +#include "ma_rt_index.h" + + /* Read a record using key */ + /* Ordinary search_flag is 0 ; Give error if no record with key */ + +int maria_rkey(MARIA_HA *info, byte *buf, int inx, const byte *key, uint key_len, + enum ha_rkey_function search_flag) +{ + uchar *key_buff; + MARIA_SHARE *share=info->s; + MARIA_KEYDEF *keyinfo; + HA_KEYSEG *last_used_keyseg; + uint pack_key_length, use_key_length, nextflag; + DBUG_ENTER("maria_rkey"); + DBUG_PRINT("enter", ("base: %lx buf: %lx inx: %d search_flag: %d", + (long) info, (long) buf, inx, search_flag)); + + if ((inx = _ma_check_index(info,inx)) < 0) + DBUG_RETURN(my_errno); + + info->update&= (HA_STATE_CHANGED | HA_STATE_ROW_CHANGED); + info->last_key_func= search_flag; + keyinfo= share->keyinfo + inx; + + if (info->once_flags & USE_PACKED_KEYS) + { + info->once_flags&= ~USE_PACKED_KEYS; /* Reset flag */ + /* + key is already packed!; This happens when we are using a MERGE TABLE + */ + key_buff=info->lastkey+info->s->base.max_key_length; + pack_key_length= key_len; + bmove(key_buff,key,key_len); + last_used_keyseg= 0; + } + else + { + if (key_len == 0) + key_len=USE_WHOLE_KEY; + /* Save the packed key for later use in the second buffer of lastkey. */ + key_buff=info->lastkey+info->s->base.max_key_length; + pack_key_length= _ma_pack_key(info,(uint) inx, key_buff, (uchar*) key, + key_len, &last_used_keyseg); + /* Save packed_key_length for use by the MERGE engine. */ + info->pack_key_length= pack_key_length; + DBUG_EXECUTE("key", _ma_print_key(DBUG_FILE, keyinfo->seg, + key_buff, pack_key_length);); + } + + if (fast_ma_readinfo(info)) + goto err; + if (share->concurrent_insert) + rw_rdlock(&share->key_root_lock[inx]); + + nextflag=maria_read_vec[search_flag]; + use_key_length=pack_key_length; + if (!(nextflag & (SEARCH_FIND | SEARCH_NO_FIND | SEARCH_LAST))) + use_key_length=USE_WHOLE_KEY; + + switch (info->s->keyinfo[inx].key_alg) { +#ifdef HAVE_RTREE_KEYS + case HA_KEY_ALG_RTREE: + if (maria_rtree_find_first(info,inx,key_buff,use_key_length,nextflag) < 0) + { + maria_print_error(info->s, HA_ERR_CRASHED); + my_errno=HA_ERR_CRASHED; + goto err; + } + break; +#endif + case HA_KEY_ALG_BTREE: + default: + if (!_ma_search(info, keyinfo, key_buff, use_key_length, + maria_read_vec[search_flag], info->s->state.key_root[inx])) + { + while (info->lastpos >= info->state->data_file_length) + { + /* + Skip rows that are inserted by other threads since we got a lock + Note that this can only happen if we are not searching after an + exact key, because the keys are sorted according to position + */ + + if (_ma_search_next(info, keyinfo, info->lastkey, + info->lastkey_length, + maria_readnext_vec[search_flag], + info->s->state.key_root[inx])) + break; + } + } + } + if (share->concurrent_insert) + rw_unlock(&share->key_root_lock[inx]); + + /* Calculate length of the found key; Used by maria_rnext_same */ + if ((keyinfo->flag & HA_VAR_LENGTH_KEY) && last_used_keyseg && + info->lastpos != HA_OFFSET_ERROR) + info->last_rkey_length= _ma_keylength_part(keyinfo, info->lastkey, + last_used_keyseg); + else + info->last_rkey_length= pack_key_length; + + /* Check if we don't want to have record back, only error message */ + if (!buf) + DBUG_RETURN(info->lastpos == HA_OFFSET_ERROR ? my_errno : 0); + + if (!(*info->read_record)(info,info->lastpos,buf)) + { + info->update|= HA_STATE_AKTIV; /* Record is read */ + DBUG_RETURN(0); + } + + info->lastpos = HA_OFFSET_ERROR; /* Didn't find key */ + + /* Store last used key as a base for read next */ + memcpy(info->lastkey,key_buff,pack_key_length); + info->last_rkey_length= pack_key_length; + bzero((char*) info->lastkey+pack_key_length,info->s->base.rec_reflength); + info->lastkey_length=pack_key_length+info->s->base.rec_reflength; + + if (search_flag == HA_READ_AFTER_KEY) + info->update|=HA_STATE_NEXT_FOUND; /* Previous gives last row */ +err: + DBUG_RETURN(my_errno); +} /* _ma_rkey */ diff --git a/storage/maria/ma_rlast.c b/storage/maria/ma_rlast.c new file mode 100644 index 00000000000..8ce26afa78d --- /dev/null +++ b/storage/maria/ma_rlast.c @@ -0,0 +1,27 @@ +/* Copyright (C) 2006 MySQL AB & MySQL Finland AB & TCX DataKonsult AB + + 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; either version 2 of the License, or + (at your option) any later version. + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ + +#include "maria_def.h" + + /* Read last row with the same key as the previous read. */ + +int maria_rlast(MARIA_HA *info, byte *buf, int inx) +{ + DBUG_ENTER("maria_rlast"); + info->lastpos= HA_OFFSET_ERROR; + info->update|= HA_STATE_NEXT_FOUND; + DBUG_RETURN(maria_rprev(info,buf,inx)); +} /* maria_rlast */ diff --git a/storage/maria/ma_rnext.c b/storage/maria/ma_rnext.c new file mode 100644 index 00000000000..8f342c6a8d2 --- /dev/null +++ b/storage/maria/ma_rnext.c @@ -0,0 +1,122 @@ +/* Copyright (C) 2006 MySQL AB & MySQL Finland AB & TCX DataKonsult AB + + 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; either version 2 of the License, or + (at your option) any later version. + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ + +#include "maria_def.h" + +#include "ma_rt_index.h" + + /* + Read next row with the same key as previous read + One may have done a write, update or delete of the previous row. + NOTE! Even if one changes the previous row, the next read is done + based on the position of the last used key! + */ + +int maria_rnext(MARIA_HA *info, byte *buf, int inx) +{ + int error,changed; + uint flag; + DBUG_ENTER("maria_rnext"); + + if ((inx = _ma_check_index(info,inx)) < 0) + DBUG_RETURN(my_errno); + flag=SEARCH_BIGGER; /* Read next */ + if (info->lastpos == HA_OFFSET_ERROR && info->update & HA_STATE_PREV_FOUND) + flag=0; /* Read first */ + + if (fast_ma_readinfo(info)) + DBUG_RETURN(my_errno); + if (info->s->concurrent_insert) + rw_rdlock(&info->s->key_root_lock[inx]); + changed= _ma_test_if_changed(info); + if (!flag) + { + switch(info->s->keyinfo[inx].key_alg){ +#ifdef HAVE_RTREE_KEYS + case HA_KEY_ALG_RTREE: + error=maria_rtree_get_first(info,inx,info->lastkey_length); + break; +#endif + case HA_KEY_ALG_BTREE: + default: + error= _ma_search_first(info,info->s->keyinfo+inx, + info->s->state.key_root[inx]); + break; + } + } + else + { + switch (info->s->keyinfo[inx].key_alg) { +#ifdef HAVE_RTREE_KEYS + case HA_KEY_ALG_RTREE: + /* + Note that rtree doesn't support that the table + may be changed since last call, so we do need + to skip rows inserted by other threads like in btree + */ + error= maria_rtree_get_next(info,inx,info->lastkey_length); + break; +#endif + case HA_KEY_ALG_BTREE: + default: + if (!changed) + error= _ma_search_next(info,info->s->keyinfo+inx,info->lastkey, + info->lastkey_length,flag, + info->s->state.key_root[inx]); + else + error= _ma_search(info,info->s->keyinfo+inx,info->lastkey, + USE_WHOLE_KEY,flag, info->s->state.key_root[inx]); + } + } + + if (info->s->concurrent_insert) + { + if (!error) + { + while (info->lastpos >= info->state->data_file_length) + { + /* Skip rows inserted by other threads since we got a lock */ + if ((error= _ma_search_next(info,info->s->keyinfo+inx, + info->lastkey, + info->lastkey_length, + SEARCH_BIGGER, + info->s->state.key_root[inx]))) + break; + } + } + rw_unlock(&info->s->key_root_lock[inx]); + } + /* Don't clear if database-changed */ + info->update&= (HA_STATE_CHANGED | HA_STATE_ROW_CHANGED); + info->update|= HA_STATE_NEXT_FOUND; + + if (error) + { + if (my_errno == HA_ERR_KEY_NOT_FOUND) + my_errno=HA_ERR_END_OF_FILE; + } + else if (!buf) + { + DBUG_RETURN(info->lastpos==HA_OFFSET_ERROR ? my_errno : 0); + } + else if (!(*info->read_record)(info,info->lastpos,buf)) + { + info->update|= HA_STATE_AKTIV; /* Record is read */ + DBUG_RETURN(0); + } + DBUG_PRINT("error",("Got error: %d, errno: %d",error, my_errno)); + DBUG_RETURN(my_errno); +} /* maria_rnext */ diff --git a/storage/maria/ma_rnext_same.c b/storage/maria/ma_rnext_same.c new file mode 100644 index 00000000000..b53639073e3 --- /dev/null +++ b/storage/maria/ma_rnext_same.c @@ -0,0 +1,105 @@ +/* Copyright (C) 2006 MySQL AB & MySQL Finland AB & TCX DataKonsult AB + + 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; either version 2 of the License, or + (at your option) any later version. + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ + +#include "maria_def.h" +#include "ma_rt_index.h" + + /* + Read next row with the same key as previous read, but abort if + the key changes. + One may have done a write, update or delete of the previous row. + NOTE! Even if one changes the previous row, the next read is done + based on the position of the last used key! + */ + +int maria_rnext_same(MARIA_HA *info, byte *buf) +{ + int error; + uint inx,not_used[2]; + MARIA_KEYDEF *keyinfo; + DBUG_ENTER("maria_rnext_same"); + + if ((int) (inx=info->lastinx) < 0 || info->lastpos == HA_OFFSET_ERROR) + DBUG_RETURN(my_errno=HA_ERR_WRONG_INDEX); + keyinfo=info->s->keyinfo+inx; + if (fast_ma_readinfo(info)) + DBUG_RETURN(my_errno); + + if (info->s->concurrent_insert) + rw_rdlock(&info->s->key_root_lock[inx]); + + switch (keyinfo->key_alg) + { +#ifdef HAVE_RTREE_KEYS + case HA_KEY_ALG_RTREE: + if ((error=maria_rtree_find_next(info,inx, + maria_read_vec[info->last_key_func]))) + { + error=1; + my_errno=HA_ERR_END_OF_FILE; + info->lastpos= HA_OFFSET_ERROR; + break; + } + break; +#endif + case HA_KEY_ALG_BTREE: + default: + if (!(info->update & HA_STATE_RNEXT_SAME)) + { + /* First rnext_same; Store old key */ + memcpy(info->lastkey2,info->lastkey,info->last_rkey_length); + } + for (;;) + { + if ((error= _ma_search_next(info,keyinfo,info->lastkey, + info->lastkey_length,SEARCH_BIGGER, + info->s->state.key_root[inx]))) + break; + if (ha_key_cmp(keyinfo->seg, info->lastkey, info->lastkey2, + info->last_rkey_length, SEARCH_FIND, not_used)) + { + error=1; + my_errno=HA_ERR_END_OF_FILE; + info->lastpos= HA_OFFSET_ERROR; + break; + } + /* Skip rows that are inserted by other threads since we got a lock */ + if (info->lastpos < info->state->data_file_length) + break; + } + } + if (info->s->concurrent_insert) + rw_unlock(&info->s->key_root_lock[inx]); + /* Don't clear if database-changed */ + info->update&= (HA_STATE_CHANGED | HA_STATE_ROW_CHANGED); + info->update|= HA_STATE_NEXT_FOUND | HA_STATE_RNEXT_SAME; + + if (error) + { + if (my_errno == HA_ERR_KEY_NOT_FOUND) + my_errno=HA_ERR_END_OF_FILE; + } + else if (!buf) + { + DBUG_RETURN(info->lastpos==HA_OFFSET_ERROR ? my_errno : 0); + } + else if (!(*info->read_record)(info,info->lastpos,buf)) + { + info->update|= HA_STATE_AKTIV; /* Record is read */ + DBUG_RETURN(0); + } + DBUG_RETURN(my_errno); +} /* maria_rnext_same */ diff --git a/storage/maria/ma_rprev.c b/storage/maria/ma_rprev.c new file mode 100644 index 00000000000..8dd4498cf8b --- /dev/null +++ b/storage/maria/ma_rprev.c @@ -0,0 +1,88 @@ +/* Copyright (C) 2006 MySQL AB & MySQL Finland AB & TCX DataKonsult AB + + 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; either version 2 of the License, or + (at your option) any later version. + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ + +#include "maria_def.h" + + /* + Read previous row with the same key as previous read + One may have done a write, update or delete of the previous row. + NOTE! Even if one changes the previous row, the next read is done + based on the position of the last used key! + */ + +int maria_rprev(MARIA_HA *info, byte *buf, int inx) +{ + int error,changed; + register uint flag; + MARIA_SHARE *share=info->s; + DBUG_ENTER("maria_rprev"); + + if ((inx = _ma_check_index(info,inx)) < 0) + DBUG_RETURN(my_errno); + flag=SEARCH_SMALLER; /* Read previous */ + if (info->lastpos == HA_OFFSET_ERROR && info->update & HA_STATE_NEXT_FOUND) + flag=0; /* Read last */ + + if (fast_ma_readinfo(info)) + DBUG_RETURN(my_errno); + changed= _ma_test_if_changed(info); + if (share->concurrent_insert) + rw_rdlock(&share->key_root_lock[inx]); + if (!flag) + error= _ma_search_last(info, share->keyinfo+inx, + share->state.key_root[inx]); + else if (!changed) + error= _ma_search_next(info,share->keyinfo+inx,info->lastkey, + info->lastkey_length,flag, + share->state.key_root[inx]); + else + error= _ma_search(info,share->keyinfo+inx,info->lastkey, + USE_WHOLE_KEY, flag, share->state.key_root[inx]); + + if (share->concurrent_insert) + { + if (!error) + { + while (info->lastpos >= info->state->data_file_length) + { + /* Skip rows that are inserted by other threads since we got a lock */ + if ((error= _ma_search_next(info,share->keyinfo+inx,info->lastkey, + info->lastkey_length, + SEARCH_SMALLER, + share->state.key_root[inx]))) + break; + } + } + rw_unlock(&share->key_root_lock[inx]); + } + info->update&= (HA_STATE_CHANGED | HA_STATE_ROW_CHANGED); + info->update|= HA_STATE_PREV_FOUND; + if (error) + { + if (my_errno == HA_ERR_KEY_NOT_FOUND) + my_errno=HA_ERR_END_OF_FILE; + } + else if (!buf) + { + DBUG_RETURN(info->lastpos==HA_OFFSET_ERROR ? my_errno : 0); + } + else if (!(*info->read_record)(info,info->lastpos,buf)) + { + info->update|= HA_STATE_AKTIV; /* Record is read */ + DBUG_RETURN(0); + } + DBUG_RETURN(my_errno); +} /* maria_rprev */ diff --git a/storage/maria/ma_rrnd.c b/storage/maria/ma_rrnd.c new file mode 100644 index 00000000000..2f01c0e92c5 --- /dev/null +++ b/storage/maria/ma_rrnd.c @@ -0,0 +1,60 @@ +/* Copyright (C) 2006 MySQL AB & MySQL Finland AB & TCX DataKonsult AB + + 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; either version 2 of the License, or + (at your option) any later version. + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ + +/* Read a record with random-access. The position to the record must + get by MARIA_HA. The next record can be read with pos= MARIA_POS_ERROR */ + + +#include "maria_def.h" + +/* + Read a row based on position. + If filepos= HA_OFFSET_ERROR then read next row + Return values + Returns one of following values: + 0 = Ok. + HA_ERR_RECORD_DELETED = Record is deleted. + HA_ERR_END_OF_FILE = EOF. +*/ + +int maria_rrnd(MARIA_HA *info, byte *buf, register my_off_t filepos) +{ + my_bool skip_deleted_blocks; + DBUG_ENTER("maria_rrnd"); + + skip_deleted_blocks=0; + + if (filepos == HA_OFFSET_ERROR) + { + skip_deleted_blocks=1; + if (info->lastpos == HA_OFFSET_ERROR) /* First read ? */ + filepos= info->s->pack.header_length; /* Read first record */ + else + filepos= info->nextpos; + } + + if (info->once_flags & RRND_PRESERVE_LASTINX) + info->once_flags&= ~RRND_PRESERVE_LASTINX; + else + info->lastinx= -1; /* Can't forward or backward */ + /* Init all but update-flag */ + info->update&= (HA_STATE_CHANGED | HA_STATE_ROW_CHANGED); + + if (info->opt_flag & WRITE_CACHE_USED && flush_io_cache(&info->rec_cache)) + DBUG_RETURN(my_errno); + + DBUG_RETURN ((*info->s->read_rnd)(info,buf,filepos,skip_deleted_blocks)); +} diff --git a/storage/maria/ma_rsame.c b/storage/maria/ma_rsame.c new file mode 100644 index 00000000000..913ae3b4370 --- /dev/null +++ b/storage/maria/ma_rsame.c @@ -0,0 +1,66 @@ +/* Copyright (C) 2006 MySQL AB & MySQL Finland AB & TCX DataKonsult AB + + 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; either version 2 of the License, or + (at your option) any later version. + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ + +#include "maria_def.h" + + /* + ** Find current row with read on position or read on key + ** If inx >= 0 find record using key + ** Return values: + ** 0 = Ok. + ** HA_ERR_KEY_NOT_FOUND = Row is deleted + ** HA_ERR_END_OF_FILE = End of file + */ + + +int maria_rsame(MARIA_HA *info, byte *record, int inx) +{ + DBUG_ENTER("maria_rsame"); + + if (inx != -1 && ! maria_is_key_active(info->s->state.key_map, inx)) + { + DBUG_RETURN(my_errno=HA_ERR_WRONG_INDEX); + } + if (info->lastpos == HA_OFFSET_ERROR || info->update & HA_STATE_DELETED) + { + DBUG_RETURN(my_errno=HA_ERR_KEY_NOT_FOUND); /* No current record */ + } + info->update&= (HA_STATE_CHANGED | HA_STATE_ROW_CHANGED); + + /* Read row from data file */ + if (fast_ma_readinfo(info)) + DBUG_RETURN(my_errno); + + if (inx >= 0) + { + info->lastinx=inx; + info->lastkey_length= _ma_make_key(info,(uint) inx,info->lastkey,record, + info->lastpos); + if (info->s->concurrent_insert) + rw_rdlock(&info->s->key_root_lock[inx]); + VOID(_ma_search(info,info->s->keyinfo+inx,info->lastkey, USE_WHOLE_KEY, + SEARCH_SAME, + info->s->state.key_root[inx])); + if (info->s->concurrent_insert) + rw_unlock(&info->s->key_root_lock[inx]); + } + + if (!(*info->read_record)(info,info->lastpos,record)) + DBUG_RETURN(0); + if (my_errno == HA_ERR_RECORD_DELETED) + my_errno=HA_ERR_KEY_NOT_FOUND; + DBUG_RETURN(my_errno); +} /* maria_rsame */ diff --git a/storage/maria/ma_rsamepos.c b/storage/maria/ma_rsamepos.c new file mode 100644 index 00000000000..e30d41dd46c --- /dev/null +++ b/storage/maria/ma_rsamepos.c @@ -0,0 +1,56 @@ +/* Copyright (C) 2006 MySQL AB & MySQL Finland AB & TCX DataKonsult AB + + 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; either version 2 of the License, or + (at your option) any later version. + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ + +/* read record through position and fix key-position */ +/* As maria_rsame but supply a position */ + +#include "maria_def.h" + + + /* + ** If inx >= 0 update index pointer + ** Returns one of the following values: + ** 0 = Ok. + ** HA_ERR_KEY_NOT_FOUND = Row is deleted + ** HA_ERR_END_OF_FILE = End of file + */ + +int maria_rsame_with_pos(MARIA_HA *info, byte *record, int inx, my_off_t filepos) +{ + DBUG_ENTER("maria_rsame_with_pos"); + + if (inx < -1 || ! maria_is_key_active(info->s->state.key_map, inx)) + { + DBUG_RETURN(my_errno=HA_ERR_WRONG_INDEX); + } + + info->update&= (HA_STATE_CHANGED | HA_STATE_ROW_CHANGED); + if ((*info->s->read_rnd)(info,record,filepos,0)) + { + if (my_errno == HA_ERR_RECORD_DELETED) + my_errno=HA_ERR_KEY_NOT_FOUND; + DBUG_RETURN(my_errno); + } + info->lastpos=filepos; + info->lastinx=inx; + if (inx >= 0) + { + info->lastkey_length= _ma_make_key(info,(uint) inx,info->lastkey,record, + info->lastpos); + info->update|=HA_STATE_KEY_CHANGED; /* Don't use indexposition */ + } + DBUG_RETURN(0); +} /* maria_rsame_pos */ diff --git a/storage/maria/ma_rt_index.c b/storage/maria/ma_rt_index.c new file mode 100644 index 00000000000..ff10ae72027 --- /dev/null +++ b/storage/maria/ma_rt_index.c @@ -0,0 +1,1081 @@ +/* Copyright (C) 2006 MySQL AB & Ramil Kalimullin & MySQL Finland AB + & TCX DataKonsult AB + + 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; either version 2 of the License, or + (at your option) any later version. + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ + +#include "maria_def.h" + +#ifdef HAVE_RTREE_KEYS + +#include "ma_rt_index.h" +#include "ma_rt_key.h" +#include "ma_rt_mbr.h" + +#define REINSERT_BUFFER_INC 10 +#define PICK_BY_AREA +/*#define PICK_BY_PERIMETER*/ + +typedef struct st_page_level +{ + uint level; + my_off_t offs; +} stPageLevel; + +typedef struct st_page_list +{ + ulong n_pages; + ulong m_pages; + stPageLevel *pages; +} stPageList; + + +/* + Find next key in r-tree according to search_flag recursively + + NOTES + Used in maria_rtree_find_first() and maria_rtree_find_next() + + RETURN + -1 Error + 0 Found + 1 Not found +*/ + +static int maria_rtree_find_req(MARIA_HA *info, MARIA_KEYDEF *keyinfo, uint search_flag, + uint nod_cmp_flag, my_off_t page, int level) +{ + uchar *k; + uchar *last; + uint nod_flag; + int res; + uchar *page_buf; + int k_len; + uint *saved_key = (uint*) (info->maria_rtree_recursion_state) + level; + + if (!(page_buf = (uchar*)my_alloca((uint)keyinfo->block_length))) + { + my_errno = HA_ERR_OUT_OF_MEM; + return -1; + } + if (!_ma_fetch_keypage(info, keyinfo, page, DFLT_INIT_HITS, page_buf, 0)) + goto err1; + nod_flag = _ma_test_if_nod(page_buf); + + k_len = keyinfo->keylength - info->s->base.rec_reflength; + + if(info->maria_rtree_recursion_depth >= level) + { + k = page_buf + *saved_key; + } + else + { + k = rt_PAGE_FIRST_KEY(page_buf, nod_flag); + } + last = rt_PAGE_END(page_buf); + + for (; k < last; k = rt_PAGE_NEXT_KEY(k, k_len, nod_flag)) + { + if (nod_flag) + { + /* this is an internal node in the tree */ + if (!(res = maria_rtree_key_cmp(keyinfo->seg, info->first_mbr_key, k, + info->last_rkey_length, nod_cmp_flag))) + { + switch ((res = maria_rtree_find_req(info, keyinfo, search_flag, nod_cmp_flag, + _ma_kpos(nod_flag, k), level + 1))) + { + case 0: /* found - exit from recursion */ + *saved_key = k - page_buf; + goto ok; + case 1: /* not found - continue searching */ + info->maria_rtree_recursion_depth = level; + break; + default: /* error */ + case -1: + goto err1; + } + } + } + else + { + /* this is a leaf */ + if (!maria_rtree_key_cmp(keyinfo->seg, info->first_mbr_key, k, + info->last_rkey_length, search_flag)) + { + uchar *after_key = rt_PAGE_NEXT_KEY(k, k_len, nod_flag); + info->lastpos = _ma_dpos(info, 0, after_key); + info->lastkey_length = k_len + info->s->base.rec_reflength; + memcpy(info->lastkey, k, info->lastkey_length); + info->maria_rtree_recursion_depth = level; + *saved_key = last - page_buf; + + if (after_key < last) + { + info->int_keypos = info->buff; + info->int_maxpos = info->buff + (last - after_key); + memcpy(info->buff, after_key, last - after_key); + info->buff_used = 0; + } + else + { + info->buff_used = 1; + } + + res = 0; + goto ok; + } + } + } + info->lastpos = HA_OFFSET_ERROR; + my_errno = HA_ERR_KEY_NOT_FOUND; + res = 1; + +ok: + my_afree((byte*)page_buf); + return res; + +err1: + my_afree((byte*)page_buf); + info->lastpos = HA_OFFSET_ERROR; + return -1; +} + + +/* + Find first key in r-tree according to search_flag condition + + SYNOPSIS + maria_rtree_find_first() + info Handler to MARIA file + uint keynr Key number to use + key Key to search for + key_length Length of 'key' + search_flag Bitmap of flags how to do the search + + RETURN + -1 Error + 0 Found + 1 Not found +*/ + +int maria_rtree_find_first(MARIA_HA *info, uint keynr, uchar *key, uint key_length, + uint search_flag) +{ + my_off_t root; + uint nod_cmp_flag; + MARIA_KEYDEF *keyinfo = info->s->keyinfo + keynr; + + if ((root = info->s->state.key_root[keynr]) == HA_OFFSET_ERROR) + { + my_errno= HA_ERR_END_OF_FILE; + return -1; + } + + /* Save searched key */ + memcpy(info->first_mbr_key, key, keyinfo->keylength - + info->s->base.rec_reflength); + info->last_rkey_length = key_length; + + info->maria_rtree_recursion_depth = -1; + info->buff_used = 1; + + nod_cmp_flag = ((search_flag & (MBR_EQUAL | MBR_WITHIN)) ? + MBR_WITHIN : MBR_INTERSECT); + return maria_rtree_find_req(info, keyinfo, search_flag, nod_cmp_flag, root, 0); +} + + +/* + Find next key in r-tree according to search_flag condition + + SYNOPSIS + maria_rtree_find_next() + info Handler to MARIA file + uint keynr Key number to use + search_flag Bitmap of flags how to do the search + + RETURN + -1 Error + 0 Found + 1 Not found +*/ + +int maria_rtree_find_next(MARIA_HA *info, uint keynr, uint search_flag) +{ + my_off_t root; + uint nod_cmp_flag; + MARIA_KEYDEF *keyinfo = info->s->keyinfo + keynr; + + if (info->update & HA_STATE_DELETED) + return maria_rtree_find_first(info, keynr, info->lastkey, info->lastkey_length, + search_flag); + + if (!info->buff_used) + { + uchar *key= info->int_keypos; + + while (key < info->int_maxpos) + { + if (!maria_rtree_key_cmp(keyinfo->seg, info->first_mbr_key, key, + info->last_rkey_length, search_flag)) + { + uchar *after_key = key + keyinfo->keylength; + + info->lastpos= _ma_dpos(info, 0, after_key); + memcpy(info->lastkey, key, info->lastkey_length); + + if (after_key < info->int_maxpos) + info->int_keypos= after_key; + else + info->buff_used= 1; + return 0; + } + key+= keyinfo->keylength; + } + } + if ((root = info->s->state.key_root[keynr]) == HA_OFFSET_ERROR) + { + my_errno= HA_ERR_END_OF_FILE; + return -1; + } + + nod_cmp_flag = ((search_flag & (MBR_EQUAL | MBR_WITHIN)) ? + MBR_WITHIN : MBR_INTERSECT); + return maria_rtree_find_req(info, keyinfo, search_flag, nod_cmp_flag, root, 0); +} + + +/* + Get next key in r-tree recursively + + NOTES + Used in maria_rtree_get_first() and maria_rtree_get_next() + + RETURN + -1 Error + 0 Found + 1 Not found +*/ + +static int maria_rtree_get_req(MARIA_HA *info, MARIA_KEYDEF *keyinfo, uint key_length, + my_off_t page, int level) +{ + uchar *k; + uchar *last; + uint nod_flag; + int res; + uchar *page_buf; + uint k_len; + uint *saved_key = (uint*) (info->maria_rtree_recursion_state) + level; + + if (!(page_buf = (uchar*)my_alloca((uint)keyinfo->block_length))) + return -1; + if (!_ma_fetch_keypage(info, keyinfo, page, DFLT_INIT_HITS, page_buf, 0)) + goto err1; + nod_flag = _ma_test_if_nod(page_buf); + + k_len = keyinfo->keylength - info->s->base.rec_reflength; + + if(info->maria_rtree_recursion_depth >= level) + { + k = page_buf + *saved_key; + if (!nod_flag) + { + /* Only leaf pages contain data references. */ + /* Need to check next key with data reference. */ + k = rt_PAGE_NEXT_KEY(k, k_len, nod_flag); + } + } + else + { + k = rt_PAGE_FIRST_KEY(page_buf, nod_flag); + } + last = rt_PAGE_END(page_buf); + + for (; k < last; k = rt_PAGE_NEXT_KEY(k, k_len, nod_flag)) + { + if (nod_flag) + { + /* this is an internal node in the tree */ + switch ((res = maria_rtree_get_req(info, keyinfo, key_length, + _ma_kpos(nod_flag, k), level + 1))) + { + case 0: /* found - exit from recursion */ + *saved_key = k - page_buf; + goto ok; + case 1: /* not found - continue searching */ + info->maria_rtree_recursion_depth = level; + break; + default: + case -1: /* error */ + goto err1; + } + } + else + { + /* this is a leaf */ + uchar *after_key = rt_PAGE_NEXT_KEY(k, k_len, nod_flag); + info->lastpos = _ma_dpos(info, 0, after_key); + info->lastkey_length = k_len + info->s->base.rec_reflength; + memcpy(info->lastkey, k, info->lastkey_length); + + info->maria_rtree_recursion_depth = level; + *saved_key = k - page_buf; + + if (after_key < last) + { + info->int_keypos = (uchar*)saved_key; + memcpy(info->buff, page_buf, keyinfo->block_length); + info->int_maxpos = rt_PAGE_END(info->buff); + info->buff_used = 0; + } + else + { + info->buff_used = 1; + } + + res = 0; + goto ok; + } + } + info->lastpos = HA_OFFSET_ERROR; + my_errno = HA_ERR_KEY_NOT_FOUND; + res = 1; + +ok: + my_afree((byte*)page_buf); + return res; + +err1: + my_afree((byte*)page_buf); + info->lastpos = HA_OFFSET_ERROR; + return -1; +} + + +/* + Get first key in r-tree + + RETURN + -1 Error + 0 Found + 1 Not found +*/ + +int maria_rtree_get_first(MARIA_HA *info, uint keynr, uint key_length) +{ + my_off_t root; + MARIA_KEYDEF *keyinfo = info->s->keyinfo + keynr; + + if ((root = info->s->state.key_root[keynr]) == HA_OFFSET_ERROR) + { + my_errno= HA_ERR_END_OF_FILE; + return -1; + } + + info->maria_rtree_recursion_depth = -1; + info->buff_used = 1; + + return maria_rtree_get_req(info, &keyinfo[keynr], key_length, root, 0); +} + + +/* + Get next key in r-tree + + RETURN + -1 Error + 0 Found + 1 Not found +*/ + +int maria_rtree_get_next(MARIA_HA *info, uint keynr, uint key_length) +{ + my_off_t root; + MARIA_KEYDEF *keyinfo = info->s->keyinfo + keynr; + + if (!info->buff_used) + { + uint k_len = keyinfo->keylength - info->s->base.rec_reflength; + /* rt_PAGE_NEXT_KEY(info->int_keypos) */ + uchar *key = info->buff + *(int*)info->int_keypos + k_len + + info->s->base.rec_reflength; + /* rt_PAGE_NEXT_KEY(key) */ + uchar *after_key = key + k_len + info->s->base.rec_reflength; + + info->lastpos = _ma_dpos(info, 0, after_key); + info->lastkey_length = k_len + info->s->base.rec_reflength; + memcpy(info->lastkey, key, k_len + info->s->base.rec_reflength); + + *(int*)info->int_keypos = key - info->buff; + if (after_key >= info->int_maxpos) + { + info->buff_used = 1; + } + + return 0; + } + else + { + if ((root = info->s->state.key_root[keynr]) == HA_OFFSET_ERROR) + { + my_errno= HA_ERR_END_OF_FILE; + return -1; + } + + return maria_rtree_get_req(info, &keyinfo[keynr], key_length, root, 0); + } +} + + +/* + Choose non-leaf better key for insertion +*/ + +#ifdef PICK_BY_PERIMETER +static uchar *maria_rtree_pick_key(MARIA_HA *info, MARIA_KEYDEF *keyinfo, uchar *key, + uint key_length, uchar *page_buf, uint nod_flag) +{ + double increase; + double best_incr = DBL_MAX; + double perimeter; + double best_perimeter; + uchar *best_key; + uchar *k = rt_PAGE_FIRST_KEY(page_buf, nod_flag); + uchar *last = rt_PAGE_END(page_buf); + + LINT_INIT(best_perimeter); + LINT_INIT(best_key); + + for (; k < last; k = rt_PAGE_NEXT_KEY(k, key_length, nod_flag)) + { + if ((increase = maria_rtree_perimeter_increase(keyinfo->seg, k, key, key_length, + &perimeter)) == -1) + return NULL; + if ((increase < best_incr)|| + (increase == best_incr && perimeter < best_perimeter)) + { + best_key = k; + best_perimeter= perimeter; + best_incr = increase; + } + } + return best_key; +} + +#endif /*PICK_BY_PERIMETER*/ + +#ifdef PICK_BY_AREA +static uchar *maria_rtree_pick_key(MARIA_HA *info, MARIA_KEYDEF *keyinfo, uchar *key, + uint key_length, uchar *page_buf, uint nod_flag) +{ + double increase; + double best_incr = DBL_MAX; + double area; + double best_area; + uchar *best_key; + uchar *k = rt_PAGE_FIRST_KEY(page_buf, nod_flag); + uchar *last = rt_PAGE_END(page_buf); + + LINT_INIT(best_area); + LINT_INIT(best_key); + + for (; k < last; k = rt_PAGE_NEXT_KEY(k, key_length, nod_flag)) + { + /* The following is safe as -1.0 is an exact number */ + if ((increase = maria_rtree_area_increase(keyinfo->seg, k, key, key_length, + &area)) == -1.0) + return NULL; + /* The following should be safe, even if we compare doubles */ + if (increase < best_incr) + { + best_key = k; + best_area = area; + best_incr = increase; + } + else + { + /* The following should be safe, even if we compare doubles */ + if ((increase == best_incr) && (area < best_area)) + { + best_key = k; + best_area = area; + best_incr = increase; + } + } + } + return best_key; +} + +#endif /*PICK_BY_AREA*/ + +/* + Go down and insert key into tree + + RETURN + -1 Error + 0 Child was not split + 1 Child was split +*/ + +static int maria_rtree_insert_req(MARIA_HA *info, MARIA_KEYDEF *keyinfo, uchar *key, + uint key_length, my_off_t page, my_off_t *new_page, + int ins_level, int level) +{ + uchar *k; + uint nod_flag; + uchar *page_buf; + int res; + + if (!(page_buf = (uchar*)my_alloca((uint)keyinfo->block_length + + HA_MAX_KEY_BUFF))) + { + my_errno = HA_ERR_OUT_OF_MEM; + return -1; + } + if (!_ma_fetch_keypage(info, keyinfo, page, DFLT_INIT_HITS, page_buf, 0)) + goto err1; + nod_flag = _ma_test_if_nod(page_buf); + + if ((ins_level == -1 && nod_flag) || /* key: go down to leaf */ + (ins_level > -1 && ins_level > level)) /* branch: go down to ins_level */ + { + if ((k = maria_rtree_pick_key(info, keyinfo, key, key_length, page_buf, + nod_flag)) == NULL) + goto err1; + switch ((res = maria_rtree_insert_req(info, keyinfo, key, key_length, + _ma_kpos(nod_flag, k), new_page, ins_level, level + 1))) + { + case 0: /* child was not split */ + { + maria_rtree_combine_rect(keyinfo->seg, k, key, k, key_length); + if (_ma_write_keypage(info, keyinfo, page, DFLT_INIT_HITS, page_buf)) + goto err1; + goto ok; + } + case 1: /* child was split */ + { + uchar *new_key = page_buf + keyinfo->block_length + nod_flag; + /* set proper MBR for key */ + if (maria_rtree_set_key_mbr(info, keyinfo, k, key_length, + _ma_kpos(nod_flag, k))) + goto err1; + /* add new key for new page */ + _ma_kpointer(info, new_key - nod_flag, *new_page); + if (maria_rtree_set_key_mbr(info, keyinfo, new_key, key_length, *new_page)) + goto err1; + res = maria_rtree_add_key(info, keyinfo, new_key, key_length, + page_buf, new_page); + if (_ma_write_keypage(info, keyinfo, page, DFLT_INIT_HITS, page_buf)) + goto err1; + goto ok; + } + default: + case -1: /* error */ + { + goto err1; + } + } + } + else + { + res = maria_rtree_add_key(info, keyinfo, key, key_length, page_buf, new_page); + if (_ma_write_keypage(info, keyinfo, page, DFLT_INIT_HITS, page_buf)) + goto err1; + goto ok; + } + +ok: + my_afree((byte*)page_buf); + return res; + +err1: + my_afree((byte*)page_buf); + return -1; +} + + +/* + Insert key into the tree + + RETURN + -1 Error + 0 Root was not split + 1 Root was split +*/ + +static int maria_rtree_insert_level(MARIA_HA *info, uint keynr, uchar *key, + uint key_length, int ins_level) +{ + my_off_t old_root; + MARIA_KEYDEF *keyinfo = info->s->keyinfo + keynr; + int res; + my_off_t new_page; + + if ((old_root = info->s->state.key_root[keynr]) == HA_OFFSET_ERROR) + { + int res; + + if ((old_root = _ma_new(info, keyinfo, DFLT_INIT_HITS)) == HA_OFFSET_ERROR) + return -1; + info->buff_used = 1; + maria_putint(info->buff, 2, 0); + res = maria_rtree_add_key(info, keyinfo, key, key_length, info->buff, NULL); + if (_ma_write_keypage(info, keyinfo, old_root, DFLT_INIT_HITS, info->buff)) + return 1; + info->s->state.key_root[keynr] = old_root; + return res; + } + + switch ((res = maria_rtree_insert_req(info, keyinfo, key, key_length, + old_root, &new_page, ins_level, 0))) + { + case 0: /* root was not split */ + { + break; + } + case 1: /* root was split, grow a new root */ + { + uchar *new_root_buf; + my_off_t new_root; + uchar *new_key; + uint nod_flag = info->s->base.key_reflength; + + if (!(new_root_buf = (uchar*)my_alloca((uint)keyinfo->block_length + + HA_MAX_KEY_BUFF))) + { + my_errno = HA_ERR_OUT_OF_MEM; + return -1; + } + + maria_putint(new_root_buf, 2, nod_flag); + if ((new_root = _ma_new(info, keyinfo, DFLT_INIT_HITS)) == + HA_OFFSET_ERROR) + goto err1; + + new_key = new_root_buf + keyinfo->block_length + nod_flag; + + _ma_kpointer(info, new_key - nod_flag, old_root); + if (maria_rtree_set_key_mbr(info, keyinfo, new_key, key_length, old_root)) + goto err1; + if (maria_rtree_add_key(info, keyinfo, new_key, key_length, new_root_buf, NULL) + == -1) + goto err1; + _ma_kpointer(info, new_key - nod_flag, new_page); + if (maria_rtree_set_key_mbr(info, keyinfo, new_key, key_length, new_page)) + goto err1; + if (maria_rtree_add_key(info, keyinfo, new_key, key_length, new_root_buf, NULL) + == -1) + goto err1; + if (_ma_write_keypage(info, keyinfo, new_root, + DFLT_INIT_HITS, new_root_buf)) + goto err1; + info->s->state.key_root[keynr] = new_root; + + my_afree((byte*)new_root_buf); + break; +err1: + my_afree((byte*)new_root_buf); + return -1; + } + default: + case -1: /* error */ + { + break; + } + } + return res; +} + + +/* + Insert key into the tree - interface function + + RETURN + -1 Error + 0 OK +*/ + +int maria_rtree_insert(MARIA_HA *info, uint keynr, uchar *key, uint key_length) +{ + return (!key_length || + (maria_rtree_insert_level(info, keynr, key, key_length, -1) == -1)) ? -1 : 0; +} + + +/* + Fill reinsert page buffer + + RETURN + -1 Error + 0 OK +*/ + +static int maria_rtree_fill_reinsert_list(stPageList *ReinsertList, my_off_t page, + int level) +{ + if (ReinsertList->n_pages == ReinsertList->m_pages) + { + ReinsertList->m_pages += REINSERT_BUFFER_INC; + if (!(ReinsertList->pages = (stPageLevel*)my_realloc((gptr)ReinsertList->pages, + ReinsertList->m_pages * sizeof(stPageLevel), MYF(MY_ALLOW_ZERO_PTR)))) + goto err1; + } + /* save page to ReinsertList */ + ReinsertList->pages[ReinsertList->n_pages].offs = page; + ReinsertList->pages[ReinsertList->n_pages].level = level; + ReinsertList->n_pages++; + return 0; + +err1: + return -1; +} + + +/* + Go down and delete key from the tree + + RETURN + -1 Error + 0 Deleted + 1 Not found + 2 Empty leaf +*/ + +static int maria_rtree_delete_req(MARIA_HA *info, MARIA_KEYDEF *keyinfo, uchar *key, + uint key_length, my_off_t page, uint *page_size, + stPageList *ReinsertList, int level) +{ + uchar *k; + uchar *last; + ulong i; + uint nod_flag; + uchar *page_buf; + int res; + + if (!(page_buf = (uchar*)my_alloca((uint)keyinfo->block_length))) + { + my_errno = HA_ERR_OUT_OF_MEM; + return -1; + } + if (!_ma_fetch_keypage(info, keyinfo, page, DFLT_INIT_HITS, page_buf, 0)) + goto err1; + nod_flag = _ma_test_if_nod(page_buf); + + k = rt_PAGE_FIRST_KEY(page_buf, nod_flag); + last = rt_PAGE_END(page_buf); + + for (i = 0; k < last; k = rt_PAGE_NEXT_KEY(k, key_length, nod_flag), ++i) + { + if (nod_flag) + { + /* not leaf */ + if (!maria_rtree_key_cmp(keyinfo->seg, key, k, key_length, MBR_WITHIN)) + { + switch ((res = maria_rtree_delete_req(info, keyinfo, key, key_length, + _ma_kpos(nod_flag, k), page_size, ReinsertList, level + 1))) + { + case 0: /* deleted */ + { + /* test page filling */ + if (*page_size + key_length >= rt_PAGE_MIN_SIZE(keyinfo->block_length)) + { + /* OK */ + if (maria_rtree_set_key_mbr(info, keyinfo, k, key_length, + _ma_kpos(nod_flag, k))) + goto err1; + if (_ma_write_keypage(info, keyinfo, page, + DFLT_INIT_HITS, page_buf)) + goto err1; + } + else + { + /* too small: delete key & add it descendant to reinsert list */ + if (maria_rtree_fill_reinsert_list(ReinsertList, _ma_kpos(nod_flag, k), + level + 1)) + goto err1; + maria_rtree_delete_key(info, page_buf, k, key_length, nod_flag); + if (_ma_write_keypage(info, keyinfo, page, + DFLT_INIT_HITS, page_buf)) + goto err1; + *page_size = maria_getint(page_buf); + } + + goto ok; + } + case 1: /* not found - continue searching */ + { + break; + } + case 2: /* vacuous case: last key in the leaf */ + { + maria_rtree_delete_key(info, page_buf, k, key_length, nod_flag); + if (_ma_write_keypage(info, keyinfo, page, + DFLT_INIT_HITS, page_buf)) + goto err1; + *page_size = maria_getint(page_buf); + res = 0; + goto ok; + } + default: /* error */ + case -1: + { + goto err1; + } + } + } + } + else + { + /* leaf */ + if (!maria_rtree_key_cmp(keyinfo->seg, key, k, key_length, MBR_EQUAL | MBR_DATA)) + { + maria_rtree_delete_key(info, page_buf, k, key_length, nod_flag); + *page_size = maria_getint(page_buf); + if (*page_size == 2) + { + /* last key in the leaf */ + res = 2; + if (_ma_dispose(info, keyinfo, page, DFLT_INIT_HITS)) + goto err1; + } + else + { + res = 0; + if (_ma_write_keypage(info, keyinfo, page, DFLT_INIT_HITS, page_buf)) + goto err1; + } + goto ok; + } + } + } + res = 1; + +ok: + my_afree((byte*)page_buf); + return res; + +err1: + my_afree((byte*)page_buf); + return -1; +} + + +/* + Delete key - interface function + + RETURN + -1 Error + 0 Deleted +*/ + +int maria_rtree_delete(MARIA_HA *info, uint keynr, uchar *key, uint key_length) +{ + uint page_size; + stPageList ReinsertList; + my_off_t old_root; + MARIA_KEYDEF *keyinfo = info->s->keyinfo + keynr; + + if ((old_root = info->s->state.key_root[keynr]) == HA_OFFSET_ERROR) + { + my_errno= HA_ERR_END_OF_FILE; + return -1; + } + + ReinsertList.pages = NULL; + ReinsertList.n_pages = 0; + ReinsertList.m_pages = 0; + + switch (maria_rtree_delete_req(info, keyinfo, key, key_length, old_root, + &page_size, &ReinsertList, 0)) + { + case 2: + { + info->s->state.key_root[keynr] = HA_OFFSET_ERROR; + return 0; + } + case 0: + { + uint nod_flag; + ulong i; + for (i = 0; i < ReinsertList.n_pages; ++i) + { + uchar *page_buf; + uint nod_flag; + uchar *k; + uchar *last; + + if (!(page_buf = (uchar*)my_alloca((uint)keyinfo->block_length))) + { + my_errno = HA_ERR_OUT_OF_MEM; + goto err1; + } + if (!_ma_fetch_keypage(info, keyinfo, ReinsertList.pages[i].offs, + DFLT_INIT_HITS, page_buf, 0)) + goto err1; + nod_flag = _ma_test_if_nod(page_buf); + k = rt_PAGE_FIRST_KEY(page_buf, nod_flag); + last = rt_PAGE_END(page_buf); + for (; k < last; k = rt_PAGE_NEXT_KEY(k, key_length, nod_flag)) + { + if (maria_rtree_insert_level(info, keynr, k, key_length, + ReinsertList.pages[i].level) == -1) + { + my_afree((byte*)page_buf); + goto err1; + } + } + my_afree((byte*)page_buf); + if (_ma_dispose(info, keyinfo, ReinsertList.pages[i].offs, + DFLT_INIT_HITS)) + goto err1; + } + if (ReinsertList.pages) + my_free((byte*) ReinsertList.pages, MYF(0)); + + /* check for redundant root (not leaf, 1 child) and eliminate */ + if ((old_root = info->s->state.key_root[keynr]) == HA_OFFSET_ERROR) + goto err1; + if (!_ma_fetch_keypage(info, keyinfo, old_root, DFLT_INIT_HITS, + info->buff, 0)) + goto err1; + nod_flag = _ma_test_if_nod(info->buff); + page_size = maria_getint(info->buff); + if (nod_flag && (page_size == 2 + key_length + nod_flag)) + { + my_off_t new_root = _ma_kpos(nod_flag, + rt_PAGE_FIRST_KEY(info->buff, nod_flag)); + if (_ma_dispose(info, keyinfo, old_root, DFLT_INIT_HITS)) + goto err1; + info->s->state.key_root[keynr] = new_root; + } + info->update= HA_STATE_DELETED; + return 0; + +err1: + return -1; + } + case 1: /* not found */ + { + my_errno = HA_ERR_KEY_NOT_FOUND; + return -1; + } + default: + case -1: /* error */ + { + return -1; + } + } +} + + +/* + Estimate number of suitable keys in the tree + + RETURN + estimated value +*/ + +ha_rows maria_rtree_estimate(MARIA_HA *info, uint keynr, uchar *key, + uint key_length, uint flag) +{ + MARIA_KEYDEF *keyinfo = info->s->keyinfo + keynr; + my_off_t root; + uint i = 0; + uchar *k; + uchar *last; + uint nod_flag; + uchar *page_buf; + uint k_len; + double area = 0; + ha_rows res = 0; + + if (flag & MBR_DISJOINT) + return info->state->records; + + if ((root = info->s->state.key_root[keynr]) == HA_OFFSET_ERROR) + return HA_POS_ERROR; + if (!(page_buf = (uchar*)my_alloca((uint)keyinfo->block_length))) + return HA_POS_ERROR; + if (!_ma_fetch_keypage(info, keyinfo, root, DFLT_INIT_HITS, page_buf, 0)) + goto err1; + nod_flag = _ma_test_if_nod(page_buf); + + k_len = keyinfo->keylength - info->s->base.rec_reflength; + + k = rt_PAGE_FIRST_KEY(page_buf, nod_flag); + last = rt_PAGE_END(page_buf); + + for (; k < last; k = rt_PAGE_NEXT_KEY(k, k_len, nod_flag), ++i) + { + if (nod_flag) + { + double k_area = maria_rtree_rect_volume(keyinfo->seg, k, key_length); + + /* The following should be safe, even if we compare doubles */ + if (k_area == 0) + { + if (flag & (MBR_CONTAIN | MBR_INTERSECT)) + { + area += 1; + } + else if (flag & (MBR_WITHIN | MBR_EQUAL)) + { + if (!maria_rtree_key_cmp(keyinfo->seg, key, k, key_length, MBR_WITHIN)) + area += 1; + } + else + goto err1; + } + else + { + if (flag & (MBR_CONTAIN | MBR_INTERSECT)) + { + area += maria_rtree_overlapping_area(keyinfo->seg, key, k, key_length) / + k_area; + } + else if (flag & (MBR_WITHIN | MBR_EQUAL)) + { + if (!maria_rtree_key_cmp(keyinfo->seg, key, k, key_length, MBR_WITHIN)) + area += maria_rtree_rect_volume(keyinfo->seg, key, key_length) / + k_area; + } + else + goto err1; + } + } + else + { + if (!maria_rtree_key_cmp(keyinfo->seg, key, k, key_length, flag)) + ++res; + } + } + if (nod_flag) + { + if (i) + res = (ha_rows) (area / i * info->state->records); + else + res = HA_POS_ERROR; + } + + my_afree((byte*)page_buf); + return res; + +err1: + my_afree((byte*)page_buf); + return HA_POS_ERROR; +} + +#endif /*HAVE_RTREE_KEYS*/ diff --git a/storage/maria/ma_rt_index.h b/storage/maria/ma_rt_index.h new file mode 100644 index 00000000000..ff431d81372 --- /dev/null +++ b/storage/maria/ma_rt_index.h @@ -0,0 +1,47 @@ +/* Copyright (C) 2006 MySQL AB & Ramil Kalimullin & MySQL Finland AB + & TCX DataKonsult AB + + 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; either version 2 of the License, or + (at your option) any later version. + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ + +#ifndef _rt_index_h +#define _rt_index_h + +#ifdef HAVE_RTREE_KEYS + +#define rt_PAGE_FIRST_KEY(page, nod_flag) (page + 2 + nod_flag) +#define rt_PAGE_NEXT_KEY(key, key_length, nod_flag) (key + key_length + \ + (nod_flag ? nod_flag : info->s->base.rec_reflength)) +#define rt_PAGE_END(page) (page + maria_getint(page)) + +#define rt_PAGE_MIN_SIZE(block_length) ((uint)(block_length) / 3) + +int maria_rtree_insert(MARIA_HA *info, uint keynr, uchar *key, uint key_length); +int maria_rtree_delete(MARIA_HA *info, uint keynr, uchar *key, uint key_length); + +int maria_rtree_find_first(MARIA_HA *info, uint keynr, uchar *key, uint key_length, + uint search_flag); +int maria_rtree_find_next(MARIA_HA *info, uint keynr, uint search_flag); + +int maria_rtree_get_first(MARIA_HA *info, uint keynr, uint key_length); +int maria_rtree_get_next(MARIA_HA *info, uint keynr, uint key_length); + +ha_rows maria_rtree_estimate(MARIA_HA *info, uint keynr, uchar *key, + uint key_length, uint flag); + +int maria_rtree_split_page(MARIA_HA *info, MARIA_KEYDEF *keyinfo, uchar *page, + uchar *key, uint key_length, my_off_t *new_page_offs); + +#endif /*HAVE_RTREE_KEYS*/ +#endif /* _rt_index_h */ diff --git a/storage/maria/ma_rt_key.c b/storage/maria/ma_rt_key.c new file mode 100644 index 00000000000..2732fefffbe --- /dev/null +++ b/storage/maria/ma_rt_key.c @@ -0,0 +1,100 @@ +/* Copyright (C) 2006 MySQL AB & Ramil Kalimullin + + 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; either version 2 of the License, or + (at your option) any later version. + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ + +#include "maria_def.h" + +#ifdef HAVE_RTREE_KEYS +#include "ma_rt_index.h" +#include "ma_rt_key.h" +#include "ma_rt_mbr.h" + +/* + Add key to the page + + RESULT VALUES + -1 Error + 0 Not split + 1 Split +*/ + +int maria_rtree_add_key(MARIA_HA *info, MARIA_KEYDEF *keyinfo, uchar *key, + uint key_length, uchar *page_buf, my_off_t *new_page) +{ + uint page_size = maria_getint(page_buf); + uint nod_flag = _ma_test_if_nod(page_buf); + + if (page_size + key_length + info->s->base.rec_reflength <= + keyinfo->block_length) + { + /* split won't be necessary */ + if (nod_flag) + { + /* save key */ + memcpy(rt_PAGE_END(page_buf), key - nod_flag, key_length + nod_flag); + page_size += key_length + nod_flag; + } + else + { + /* save key */ + memcpy(rt_PAGE_END(page_buf), key, key_length + + info->s->base.rec_reflength); + page_size += key_length + info->s->base.rec_reflength; + } + maria_putint(page_buf, page_size, nod_flag); + return 0; + } + + return (maria_rtree_split_page(info, keyinfo, page_buf, key, key_length, + new_page) ? -1 : 1); +} + +/* + Delete key from the page +*/ +int maria_rtree_delete_key(MARIA_HA *info, uchar *page_buf, uchar *key, + uint key_length, uint nod_flag) +{ + uint16 page_size = maria_getint(page_buf); + uchar *key_start; + + key_start= key - nod_flag; + if (!nod_flag) + key_length += info->s->base.rec_reflength; + + memmove(key_start, key + key_length, page_size - key_length - + (key - page_buf)); + page_size-= key_length + nod_flag; + + maria_putint(page_buf, page_size, nod_flag); + return 0; +} + + +/* + Calculate and store key MBR +*/ + +int maria_rtree_set_key_mbr(MARIA_HA *info, MARIA_KEYDEF *keyinfo, uchar *key, + uint key_length, my_off_t child_page) +{ + if (!_ma_fetch_keypage(info, keyinfo, child_page, + DFLT_INIT_HITS, info->buff, 0)) + return -1; + + return maria_rtree_page_mbr(info, keyinfo->seg, info->buff, key, key_length); +} + +#endif /*HAVE_RTREE_KEYS*/ diff --git a/storage/maria/ma_rt_key.h b/storage/maria/ma_rt_key.h new file mode 100644 index 00000000000..448024ed8c5 --- /dev/null +++ b/storage/maria/ma_rt_key.h @@ -0,0 +1,33 @@ +/* Copyright (C) 2006 MySQL AB & Ramil Kalimullin & MySQL Finland AB + & TCX DataKonsult AB + + 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; either version 2 of the License, or + (at your option) any later version. + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ + +/* Written by Ramil Kalimullin, who has a shared copyright to this code */ + +#ifndef _rt_key_h +#define _rt_key_h + +#ifdef HAVE_RTREE_KEYS + +int maria_rtree_add_key(MARIA_HA *info, MARIA_KEYDEF *keyinfo, uchar *key, + uint key_length, uchar *page_buf, my_off_t *new_page); +int maria_rtree_delete_key(MARIA_HA *info, uchar *page, uchar *key, + uint key_length, uint nod_flag); +int maria_rtree_set_key_mbr(MARIA_HA *info, MARIA_KEYDEF *keyinfo, uchar *key, + uint key_length, my_off_t child_page); + +#endif /*HAVE_RTREE_KEYS*/ +#endif /* _rt_key_h */ diff --git a/storage/maria/ma_rt_mbr.c b/storage/maria/ma_rt_mbr.c new file mode 100644 index 00000000000..83d3a0a2f1c --- /dev/null +++ b/storage/maria/ma_rt_mbr.c @@ -0,0 +1,801 @@ +/* Copyright (C) 2006 MySQL AB & Ramil Kalimullin & MySQL Finland AB + & TCX DataKonsult AB + + 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; either version 2 of the License, or + (at your option) any later version. + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ + +#include "maria_def.h" + +#ifdef HAVE_RTREE_KEYS + +#include "ma_rt_index.h" +#include "ma_rt_mbr.h" + +#define INTERSECT_CMP(amin, amax, bmin, bmax) ((amin > bmax) || (bmin > amax)) +#define CONTAIN_CMP(amin, amax, bmin, bmax) ((bmin > amin) || (bmax < amax)) +#define WITHIN_CMP(amin, amax, bmin, bmax) ((amin > bmin) || (amax < bmax)) +#define DISJOINT_CMP(amin, amax, bmin, bmax) ((amin <= bmax) && (bmin <= amax)) +#define EQUAL_CMP(amin, amax, bmin, bmax) ((amin != bmin) || (amax != bmax)) + +#define FCMP(A, B) ((int)(A) - (int)(B)) +#define p_inc(A, B, X) {A += X; B += X;} + +#define RT_CMP(nextflag) \ + if (nextflag & MBR_INTERSECT) \ + { \ + if (INTERSECT_CMP(amin, amax, bmin, bmax)) \ + return 1; \ + } \ + else if (nextflag & MBR_CONTAIN) \ + { \ + if (CONTAIN_CMP(amin, amax, bmin, bmax)) \ + return 1; \ + } \ + else if (nextflag & MBR_WITHIN) \ + { \ + if (WITHIN_CMP(amin, amax, bmin, bmax)) \ + return 1; \ + } \ + else if (nextflag & MBR_EQUAL) \ + { \ + if (EQUAL_CMP(amin, amax, bmin, bmax)) \ + return 1; \ + } \ + else /* if (nextflag & MBR_DISJOINT) */ \ + { \ + if (DISJOINT_CMP(amin, amax, bmin, bmax)) \ + return 1; \ + } + +#define RT_CMP_KORR(type, korr_func, len, nextflag) \ +{ \ + type amin, amax, bmin, bmax; \ + amin = korr_func(a); \ + bmin = korr_func(b); \ + amax = korr_func(a+len); \ + bmax = korr_func(b+len); \ + RT_CMP(nextflag); \ +} + +#define RT_CMP_GET(type, get_func, len, nextflag) \ +{ \ + type amin, amax, bmin, bmax; \ + get_func(amin, a); \ + get_func(bmin, b); \ + get_func(amax, a+len); \ + get_func(bmax, b+len); \ + RT_CMP(nextflag); \ +} + +/* + Compares two keys a and b depending on nextflag + nextflag can contain these flags: + MBR_INTERSECT(a,b) a overlaps b + MBR_CONTAIN(a,b) a contains b + MBR_DISJOINT(a,b) a disjoint b + MBR_WITHIN(a,b) a within b + MBR_EQUAL(a,b) All coordinates of MBRs are equal + MBR_DATA(a,b) Data reference is the same + Returns 0 on success. +*/ +int maria_rtree_key_cmp(HA_KEYSEG *keyseg, uchar *b, uchar *a, uint key_length, + uint nextflag) +{ + for (; (int) key_length > 0; keyseg += 2 ) + { + uint32 keyseg_length; + switch ((enum ha_base_keytype) keyseg->type) { + case HA_KEYTYPE_INT8: + RT_CMP_KORR(int8, mi_sint1korr, 1, nextflag); + break; + case HA_KEYTYPE_BINARY: + RT_CMP_KORR(uint8, mi_uint1korr, 1, nextflag); + break; + case HA_KEYTYPE_SHORT_INT: + RT_CMP_KORR(int16, mi_sint2korr, 2, nextflag); + break; + case HA_KEYTYPE_USHORT_INT: + RT_CMP_KORR(uint16, mi_uint2korr, 2, nextflag); + break; + case HA_KEYTYPE_INT24: + RT_CMP_KORR(int32, mi_sint3korr, 3, nextflag); + break; + case HA_KEYTYPE_UINT24: + RT_CMP_KORR(uint32, mi_uint3korr, 3, nextflag); + break; + case HA_KEYTYPE_LONG_INT: + RT_CMP_KORR(int32, mi_sint4korr, 4, nextflag); + break; + case HA_KEYTYPE_ULONG_INT: + RT_CMP_KORR(uint32, mi_uint4korr, 4, nextflag); + break; +#ifdef HAVE_LONG_LONG + case HA_KEYTYPE_LONGLONG: + RT_CMP_KORR(longlong, mi_sint8korr, 8, nextflag) + break; + case HA_KEYTYPE_ULONGLONG: + RT_CMP_KORR(ulonglong, mi_uint8korr, 8, nextflag) + break; +#endif + case HA_KEYTYPE_FLOAT: + /* The following should be safe, even if we compare doubles */ + RT_CMP_GET(float, mi_float4get, 4, nextflag); + break; + case HA_KEYTYPE_DOUBLE: + RT_CMP_GET(double, mi_float8get, 8, nextflag); + break; + case HA_KEYTYPE_END: + goto end; + default: + return 1; + } + keyseg_length= keyseg->length * 2; + key_length-= keyseg_length; + a+= keyseg_length; + b+= keyseg_length; + } + +end: + if (nextflag & MBR_DATA) + { + uchar *end = a + keyseg->length; + do + { + if (*a++ != *b++) + return FCMP(a[-1], b[-1]); + } while (a != end); + } + return 0; +} + +#define RT_VOL_KORR(type, korr_func, len, cast) \ +{ \ + type amin, amax; \ + amin = korr_func(a); \ + amax = korr_func(a+len); \ + res *= (cast(amax) - cast(amin)); \ +} + +#define RT_VOL_GET(type, get_func, len, cast) \ +{ \ + type amin, amax; \ + get_func(amin, a); \ + get_func(amax, a+len); \ + res *= (cast(amax) - cast(amin)); \ +} + +/* + Calculates rectangle volume +*/ +double maria_rtree_rect_volume(HA_KEYSEG *keyseg, uchar *a, uint key_length) +{ + double res = 1; + for (; (int)key_length > 0; keyseg += 2) + { + uint32 keyseg_length; + switch ((enum ha_base_keytype) keyseg->type) { + case HA_KEYTYPE_INT8: + RT_VOL_KORR(int8, mi_sint1korr, 1, (double)); + break; + case HA_KEYTYPE_BINARY: + RT_VOL_KORR(uint8, mi_uint1korr, 1, (double)); + break; + case HA_KEYTYPE_SHORT_INT: + RT_VOL_KORR(int16, mi_sint2korr, 2, (double)); + break; + case HA_KEYTYPE_USHORT_INT: + RT_VOL_KORR(uint16, mi_uint2korr, 2, (double)); + break; + case HA_KEYTYPE_INT24: + RT_VOL_KORR(int32, mi_sint3korr, 3, (double)); + break; + case HA_KEYTYPE_UINT24: + RT_VOL_KORR(uint32, mi_uint3korr, 3, (double)); + break; + case HA_KEYTYPE_LONG_INT: + RT_VOL_KORR(int32, mi_sint4korr, 4, (double)); + break; + case HA_KEYTYPE_ULONG_INT: + RT_VOL_KORR(uint32, mi_uint4korr, 4, (double)); + break; +#ifdef HAVE_LONG_LONG + case HA_KEYTYPE_LONGLONG: + RT_VOL_KORR(longlong, mi_sint8korr, 8, (double)); + break; + case HA_KEYTYPE_ULONGLONG: + RT_VOL_KORR(longlong, mi_sint8korr, 8, ulonglong2double); + break; +#endif + case HA_KEYTYPE_FLOAT: + RT_VOL_GET(float, mi_float4get, 4, (double)); + break; + case HA_KEYTYPE_DOUBLE: + RT_VOL_GET(double, mi_float8get, 8, (double)); + break; + case HA_KEYTYPE_END: + key_length = 0; + break; + default: + return -1; + } + keyseg_length= keyseg->length * 2; + key_length-= keyseg_length; + a+= keyseg_length; + } + return res; +} + +#define RT_D_MBR_KORR(type, korr_func, len, cast) \ +{ \ + type amin, amax; \ + amin = korr_func(a); \ + amax = korr_func(a+len); \ + *res++ = cast(amin); \ + *res++ = cast(amax); \ +} + +#define RT_D_MBR_GET(type, get_func, len, cast) \ +{ \ + type amin, amax; \ + get_func(amin, a); \ + get_func(amax, a+len); \ + *res++ = cast(amin); \ + *res++ = cast(amax); \ +} + + +/* + Creates an MBR as an array of doubles. +*/ + +int maria_rtree_d_mbr(HA_KEYSEG *keyseg, uchar *a, uint key_length, double *res) +{ + for (; (int)key_length > 0; keyseg += 2) + { + uint32 keyseg_length; + switch ((enum ha_base_keytype) keyseg->type) { + case HA_KEYTYPE_INT8: + RT_D_MBR_KORR(int8, mi_sint1korr, 1, (double)); + break; + case HA_KEYTYPE_BINARY: + RT_D_MBR_KORR(uint8, mi_uint1korr, 1, (double)); + break; + case HA_KEYTYPE_SHORT_INT: + RT_D_MBR_KORR(int16, mi_sint2korr, 2, (double)); + break; + case HA_KEYTYPE_USHORT_INT: + RT_D_MBR_KORR(uint16, mi_uint2korr, 2, (double)); + break; + case HA_KEYTYPE_INT24: + RT_D_MBR_KORR(int32, mi_sint3korr, 3, (double)); + break; + case HA_KEYTYPE_UINT24: + RT_D_MBR_KORR(uint32, mi_uint3korr, 3, (double)); + break; + case HA_KEYTYPE_LONG_INT: + RT_D_MBR_KORR(int32, mi_sint4korr, 4, (double)); + break; + case HA_KEYTYPE_ULONG_INT: + RT_D_MBR_KORR(uint32, mi_uint4korr, 4, (double)); + break; +#ifdef HAVE_LONG_LONG + case HA_KEYTYPE_LONGLONG: + RT_D_MBR_KORR(longlong, mi_sint8korr, 8, (double)); + break; + case HA_KEYTYPE_ULONGLONG: + RT_D_MBR_KORR(longlong, mi_sint8korr, 8, ulonglong2double); + break; +#endif + case HA_KEYTYPE_FLOAT: + RT_D_MBR_GET(float, mi_float4get, 4, (double)); + break; + case HA_KEYTYPE_DOUBLE: + RT_D_MBR_GET(double, mi_float8get, 8, (double)); + break; + case HA_KEYTYPE_END: + key_length = 0; + break; + default: + return 1; + } + keyseg_length= keyseg->length * 2; + key_length-= keyseg_length; + a+= keyseg_length; + } + return 0; +} + +#define RT_COMB_KORR(type, korr_func, store_func, len) \ +{ \ + type amin, amax, bmin, bmax; \ + amin = korr_func(a); \ + bmin = korr_func(b); \ + amax = korr_func(a+len); \ + bmax = korr_func(b+len); \ + amin = min(amin, bmin); \ + amax = max(amax, bmax); \ + store_func(c, amin); \ + store_func(c+len, amax); \ +} + +#define RT_COMB_GET(type, get_func, store_func, len) \ +{ \ + type amin, amax, bmin, bmax; \ + get_func(amin, a); \ + get_func(bmin, b); \ + get_func(amax, a+len); \ + get_func(bmax, b+len); \ + amin = min(amin, bmin); \ + amax = max(amax, bmax); \ + store_func(c, amin); \ + store_func(c+len, amax); \ +} + +/* + Creates common minimal bounding rectungle + for two input rectagnles a and b + Result is written to c +*/ + +int maria_rtree_combine_rect(HA_KEYSEG *keyseg, uchar* a, uchar* b, uchar* c, + uint key_length) +{ + for ( ; (int) key_length > 0 ; keyseg += 2) + { + uint32 keyseg_length; + switch ((enum ha_base_keytype) keyseg->type) { + case HA_KEYTYPE_INT8: + RT_COMB_KORR(int8, mi_sint1korr, mi_int1store, 1); + break; + case HA_KEYTYPE_BINARY: + RT_COMB_KORR(uint8, mi_uint1korr, mi_int1store, 1); + break; + case HA_KEYTYPE_SHORT_INT: + RT_COMB_KORR(int16, mi_sint2korr, mi_int2store, 2); + break; + case HA_KEYTYPE_USHORT_INT: + RT_COMB_KORR(uint16, mi_uint2korr, mi_int2store, 2); + break; + case HA_KEYTYPE_INT24: + RT_COMB_KORR(int32, mi_sint3korr, mi_int3store, 3); + break; + case HA_KEYTYPE_UINT24: + RT_COMB_KORR(uint32, mi_uint3korr, mi_int3store, 3); + break; + case HA_KEYTYPE_LONG_INT: + RT_COMB_KORR(int32, mi_sint4korr, mi_int4store, 4); + break; + case HA_KEYTYPE_ULONG_INT: + RT_COMB_KORR(uint32, mi_uint4korr, mi_int4store, 4); + break; +#ifdef HAVE_LONG_LONG + case HA_KEYTYPE_LONGLONG: + RT_COMB_KORR(longlong, mi_sint8korr, mi_int8store, 8); + break; + case HA_KEYTYPE_ULONGLONG: + RT_COMB_KORR(ulonglong, mi_uint8korr, mi_int8store, 8); + break; +#endif + case HA_KEYTYPE_FLOAT: + RT_COMB_GET(float, mi_float4get, mi_float4store, 4); + break; + case HA_KEYTYPE_DOUBLE: + RT_COMB_GET(double, mi_float8get, mi_float8store, 8); + break; + case HA_KEYTYPE_END: + return 0; + default: + return 1; + } + keyseg_length= keyseg->length * 2; + key_length-= keyseg_length; + a+= keyseg_length; + b+= keyseg_length; + c+= keyseg_length; + } + return 0; +} + + +#define RT_OVL_AREA_KORR(type, korr_func, len) \ +{ \ + type amin, amax, bmin, bmax; \ + amin = korr_func(a); \ + bmin = korr_func(b); \ + amax = korr_func(a+len); \ + bmax = korr_func(b+len); \ + amin = max(amin, bmin); \ + amax = min(amax, bmax); \ + if (amin >= amax) \ + return 0; \ + res *= amax - amin; \ +} + +#define RT_OVL_AREA_GET(type, get_func, len) \ +{ \ + type amin, amax, bmin, bmax; \ + get_func(amin, a); \ + get_func(bmin, b); \ + get_func(amax, a+len); \ + get_func(bmax, b+len); \ + amin = max(amin, bmin); \ + amax = min(amax, bmax); \ + if (amin >= amax) \ + return 0; \ + res *= amax - amin; \ +} + +/* +Calculates overlapping area of two MBRs a & b +*/ +double maria_rtree_overlapping_area(HA_KEYSEG *keyseg, uchar* a, uchar* b, + uint key_length) +{ + double res = 1; + for (; (int) key_length > 0 ; keyseg += 2) + { + uint32 keyseg_length; + switch ((enum ha_base_keytype) keyseg->type) { + case HA_KEYTYPE_INT8: + RT_OVL_AREA_KORR(int8, mi_sint1korr, 1); + break; + case HA_KEYTYPE_BINARY: + RT_OVL_AREA_KORR(uint8, mi_uint1korr, 1); + break; + case HA_KEYTYPE_SHORT_INT: + RT_OVL_AREA_KORR(int16, mi_sint2korr, 2); + break; + case HA_KEYTYPE_USHORT_INT: + RT_OVL_AREA_KORR(uint16, mi_uint2korr, 2); + break; + case HA_KEYTYPE_INT24: + RT_OVL_AREA_KORR(int32, mi_sint3korr, 3); + break; + case HA_KEYTYPE_UINT24: + RT_OVL_AREA_KORR(uint32, mi_uint3korr, 3); + break; + case HA_KEYTYPE_LONG_INT: + RT_OVL_AREA_KORR(int32, mi_sint4korr, 4); + break; + case HA_KEYTYPE_ULONG_INT: + RT_OVL_AREA_KORR(uint32, mi_uint4korr, 4); + break; +#ifdef HAVE_LONG_LONG + case HA_KEYTYPE_LONGLONG: + RT_OVL_AREA_KORR(longlong, mi_sint8korr, 8); + break; + case HA_KEYTYPE_ULONGLONG: + RT_OVL_AREA_KORR(longlong, mi_sint8korr, 8); + break; +#endif + case HA_KEYTYPE_FLOAT: + RT_OVL_AREA_GET(float, mi_float4get, 4); + break; + case HA_KEYTYPE_DOUBLE: + RT_OVL_AREA_GET(double, mi_float8get, 8); + break; + case HA_KEYTYPE_END: + return res; + default: + return -1; + } + keyseg_length= keyseg->length * 2; + key_length-= keyseg_length; + a+= keyseg_length; + b+= keyseg_length; + } + return res; +} + +#define RT_AREA_INC_KORR(type, korr_func, len) \ +{ \ + type amin, amax, bmin, bmax; \ + amin = korr_func(a); \ + bmin = korr_func(b); \ + amax = korr_func(a+len); \ + bmax = korr_func(b+len); \ + a_area *= (((double)amax) - ((double)amin)); \ + loc_ab_area *= ((double)max(amax, bmax) - (double)min(amin, bmin)); \ +} + +#define RT_AREA_INC_GET(type, get_func, len)\ +{\ + type amin, amax, bmin, bmax; \ + get_func(amin, a); \ + get_func(bmin, b); \ + get_func(amax, a+len); \ + get_func(bmax, b+len); \ + a_area *= (((double)amax) - ((double)amin)); \ + loc_ab_area *= ((double)max(amax, bmax) - (double)min(amin, bmin)); \ +} + +/* +Calculates MBR_AREA(a+b) - MBR_AREA(a) +*/ +double maria_rtree_area_increase(HA_KEYSEG *keyseg, uchar* a, uchar* b, + uint key_length, double *ab_area) +{ + double a_area= 1.0; + double loc_ab_area= 1.0; + + *ab_area= 1.0; + for (; (int)key_length > 0; keyseg += 2) + { + uint32 keyseg_length; + + if (keyseg->null_bit) /* Handle NULL part */ + return -1; + + switch ((enum ha_base_keytype) keyseg->type) { + case HA_KEYTYPE_INT8: + RT_AREA_INC_KORR(int8, mi_sint1korr, 1); + break; + case HA_KEYTYPE_BINARY: + RT_AREA_INC_KORR(uint8, mi_uint1korr, 1); + break; + case HA_KEYTYPE_SHORT_INT: + RT_AREA_INC_KORR(int16, mi_sint2korr, 2); + break; + case HA_KEYTYPE_USHORT_INT: + RT_AREA_INC_KORR(uint16, mi_uint2korr, 2); + break; + case HA_KEYTYPE_INT24: + RT_AREA_INC_KORR(int32, mi_sint3korr, 3); + break; + case HA_KEYTYPE_UINT24: + RT_AREA_INC_KORR(int32, mi_uint3korr, 3); + break; + case HA_KEYTYPE_LONG_INT: + RT_AREA_INC_KORR(int32, mi_sint4korr, 4); + break; + case HA_KEYTYPE_ULONG_INT: + RT_AREA_INC_KORR(uint32, mi_uint4korr, 4); + break; +#ifdef HAVE_LONG_LONG + case HA_KEYTYPE_LONGLONG: + RT_AREA_INC_KORR(longlong, mi_sint8korr, 8); + break; + case HA_KEYTYPE_ULONGLONG: + RT_AREA_INC_KORR(longlong, mi_sint8korr, 8); + break; +#endif + case HA_KEYTYPE_FLOAT: + RT_AREA_INC_GET(float, mi_float4get, 4); + break; + case HA_KEYTYPE_DOUBLE: + RT_AREA_INC_GET(double, mi_float8get, 8); + break; + case HA_KEYTYPE_END: + goto safe_end; + default: + return -1; + } + keyseg_length= keyseg->length * 2; + key_length-= keyseg_length; + a+= keyseg_length; + b+= keyseg_length; + } +safe_end: + *ab_area= loc_ab_area; + return loc_ab_area - a_area; +} + +#define RT_PERIM_INC_KORR(type, korr_func, len) \ +{ \ + type amin, amax, bmin, bmax; \ + amin = korr_func(a); \ + bmin = korr_func(b); \ + amax = korr_func(a+len); \ + bmax = korr_func(b+len); \ + a_perim+= (((double)amax) - ((double)amin)); \ + *ab_perim+= ((double)max(amax, bmax) - (double)min(amin, bmin)); \ +} + +#define RT_PERIM_INC_GET(type, get_func, len)\ +{\ + type amin, amax, bmin, bmax; \ + get_func(amin, a); \ + get_func(bmin, b); \ + get_func(amax, a+len); \ + get_func(bmax, b+len); \ + a_perim+= (((double)amax) - ((double)amin)); \ + *ab_perim+= ((double)max(amax, bmax) - (double)min(amin, bmin)); \ +} + +/* +Calculates MBR_PERIMETER(a+b) - MBR_PERIMETER(a) +*/ +double maria_rtree_perimeter_increase(HA_KEYSEG *keyseg, uchar* a, uchar* b, + uint key_length, double *ab_perim) +{ + double a_perim = 0.0; + + *ab_perim= 0.0; + for (; (int)key_length > 0; keyseg += 2) + { + uint32 keyseg_length; + + if (keyseg->null_bit) /* Handle NULL part */ + return -1; + + switch ((enum ha_base_keytype) keyseg->type) { + case HA_KEYTYPE_INT8: + RT_PERIM_INC_KORR(int8, mi_sint1korr, 1); + break; + case HA_KEYTYPE_BINARY: + RT_PERIM_INC_KORR(uint8, mi_uint1korr, 1); + break; + case HA_KEYTYPE_SHORT_INT: + RT_PERIM_INC_KORR(int16, mi_sint2korr, 2); + break; + case HA_KEYTYPE_USHORT_INT: + RT_PERIM_INC_KORR(uint16, mi_uint2korr, 2); + break; + case HA_KEYTYPE_INT24: + RT_PERIM_INC_KORR(int32, mi_sint3korr, 3); + break; + case HA_KEYTYPE_UINT24: + RT_PERIM_INC_KORR(int32, mi_uint3korr, 3); + break; + case HA_KEYTYPE_LONG_INT: + RT_PERIM_INC_KORR(int32, mi_sint4korr, 4); + break; + case HA_KEYTYPE_ULONG_INT: + RT_PERIM_INC_KORR(uint32, mi_uint4korr, 4); + break; +#ifdef HAVE_LONG_LONG + case HA_KEYTYPE_LONGLONG: + RT_PERIM_INC_KORR(longlong, mi_sint8korr, 8); + break; + case HA_KEYTYPE_ULONGLONG: + RT_PERIM_INC_KORR(longlong, mi_sint8korr, 8); + break; +#endif + case HA_KEYTYPE_FLOAT: + RT_PERIM_INC_GET(float, mi_float4get, 4); + break; + case HA_KEYTYPE_DOUBLE: + RT_PERIM_INC_GET(double, mi_float8get, 8); + break; + case HA_KEYTYPE_END: + return *ab_perim - a_perim; + default: + return -1; + } + keyseg_length= keyseg->length * 2; + key_length-= keyseg_length; + a+= keyseg_length; + b+= keyseg_length; + } + return *ab_perim - a_perim; +} + + +#define RT_PAGE_MBR_KORR(type, korr_func, store_func, len) \ +{ \ + type amin, amax, bmin, bmax; \ + amin = korr_func(k + inc); \ + amax = korr_func(k + inc + len); \ + k = rt_PAGE_NEXT_KEY(k, k_len, nod_flag); \ + for (; k < last; k = rt_PAGE_NEXT_KEY(k, k_len, nod_flag)) \ +{ \ + bmin = korr_func(k + inc); \ + bmax = korr_func(k + inc + len); \ + if (amin > bmin) \ + amin = bmin; \ + if (amax < bmax) \ + amax = bmax; \ +} \ + store_func(c, amin); \ + c += len; \ + store_func(c, amax); \ + c += len; \ + inc += 2 * len; \ +} + +#define RT_PAGE_MBR_GET(type, get_func, store_func, len) \ +{ \ + type amin, amax, bmin, bmax; \ + get_func(amin, k + inc); \ + get_func(amax, k + inc + len); \ + k = rt_PAGE_NEXT_KEY(k, k_len, nod_flag); \ + for (; k < last; k = rt_PAGE_NEXT_KEY(k, k_len, nod_flag)) \ +{ \ + get_func(bmin, k + inc); \ + get_func(bmax, k + inc + len); \ + if (amin > bmin) \ + amin = bmin; \ + if (amax < bmax) \ + amax = bmax; \ +} \ + store_func(c, amin); \ + c += len; \ + store_func(c, amax); \ + c += len; \ + inc += 2 * len; \ +} + +/* +Calculates key page total MBR = MBR(key1) + MBR(key2) + ... +*/ +int maria_rtree_page_mbr(MARIA_HA *info, HA_KEYSEG *keyseg, uchar *page_buf, + uchar *c, uint key_length) +{ + uint inc = 0; + uint k_len = key_length; + uint nod_flag = _ma_test_if_nod(page_buf); + uchar *k; + uchar *last = rt_PAGE_END(page_buf); + + for (; (int)key_length > 0; keyseg += 2) + { + key_length -= keyseg->length * 2; + + /* Handle NULL part */ + if (keyseg->null_bit) + { + return 1; + } + + k = rt_PAGE_FIRST_KEY(page_buf, nod_flag); + + switch ((enum ha_base_keytype) keyseg->type) { + case HA_KEYTYPE_INT8: + RT_PAGE_MBR_KORR(int8, mi_sint1korr, mi_int1store, 1); + break; + case HA_KEYTYPE_BINARY: + RT_PAGE_MBR_KORR(uint8, mi_uint1korr, mi_int1store, 1); + break; + case HA_KEYTYPE_SHORT_INT: + RT_PAGE_MBR_KORR(int16, mi_sint2korr, mi_int2store, 2); + break; + case HA_KEYTYPE_USHORT_INT: + RT_PAGE_MBR_KORR(uint16, mi_uint2korr, mi_int2store, 2); + break; + case HA_KEYTYPE_INT24: + RT_PAGE_MBR_KORR(int32, mi_sint3korr, mi_int3store, 3); + break; + case HA_KEYTYPE_UINT24: + RT_PAGE_MBR_KORR(uint32, mi_uint3korr, mi_int3store, 3); + break; + case HA_KEYTYPE_LONG_INT: + RT_PAGE_MBR_KORR(int32, mi_sint4korr, mi_int4store, 4); + break; + case HA_KEYTYPE_ULONG_INT: + RT_PAGE_MBR_KORR(uint32, mi_uint4korr, mi_int4store, 4); + break; +#ifdef HAVE_LONG_LONG + case HA_KEYTYPE_LONGLONG: + RT_PAGE_MBR_KORR(longlong, mi_sint8korr, mi_int8store, 8); + break; + case HA_KEYTYPE_ULONGLONG: + RT_PAGE_MBR_KORR(ulonglong, mi_uint8korr, mi_int8store, 8); + break; +#endif + case HA_KEYTYPE_FLOAT: + RT_PAGE_MBR_GET(float, mi_float4get, mi_float4store, 4); + break; + case HA_KEYTYPE_DOUBLE: + RT_PAGE_MBR_GET(double, mi_float8get, mi_float8store, 8); + break; + case HA_KEYTYPE_END: + return 0; + default: + return 1; + } + } + return 0; +} + +#endif /*HAVE_RTREE_KEYS*/ diff --git a/storage/maria/ma_rt_mbr.h b/storage/maria/ma_rt_mbr.h new file mode 100644 index 00000000000..81e2a6851d4 --- /dev/null +++ b/storage/maria/ma_rt_mbr.h @@ -0,0 +1,38 @@ +/* Copyright (C) 2006 MySQL AB & Ramil Kalimullin & MySQL Finland AB + & TCX DataKonsult AB + + 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; either version 2 of the License, or + (at your option) any later version. + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ + +#ifndef _rt_mbr_h +#define _rt_mbr_h + +#ifdef HAVE_RTREE_KEYS + +int maria_rtree_key_cmp(HA_KEYSEG *keyseg, uchar *a, uchar *b, uint key_length, + uint nextflag); +int maria_rtree_combine_rect(HA_KEYSEG *keyseg,uchar *, uchar *, uchar*, + uint key_length); +double maria_rtree_rect_volume(HA_KEYSEG *keyseg, uchar*, uint key_length); +int maria_rtree_d_mbr(HA_KEYSEG *keyseg, uchar *a, uint key_length, double *res); +double maria_rtree_overlapping_area(HA_KEYSEG *keyseg, uchar *a, uchar *b, + uint key_length); +double maria_rtree_area_increase(HA_KEYSEG *keyseg, uchar *a, uchar *b, + uint key_length, double *ab_area); +double maria_rtree_perimeter_increase(HA_KEYSEG *keyseg, uchar* a, uchar* b, + uint key_length, double *ab_perim); +int maria_rtree_page_mbr(MARIA_HA *info, HA_KEYSEG *keyseg, uchar *page_buf, + uchar* c, uint key_length); +#endif /*HAVE_RTREE_KEYS*/ +#endif /* _rt_mbr_h */ diff --git a/storage/maria/ma_rt_split.c b/storage/maria/ma_rt_split.c new file mode 100644 index 00000000000..034799efd89 --- /dev/null +++ b/storage/maria/ma_rt_split.c @@ -0,0 +1,350 @@ +/* Copyright (C) 2006 MySQL AB & Alexey Botchkov & MySQL Finland AB + & TCX DataKonsult AB + + 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; either version 2 of the License, or + (at your option) any later version. + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ + +#include "maria_def.h" + +#ifdef HAVE_RTREE_KEYS + +#include "ma_rt_index.h" +#include "ma_rt_key.h" +#include "ma_rt_mbr.h" + +typedef struct +{ + double square; + int n_node; + uchar *key; + double *coords; +} SplitStruct; + +inline static double *reserve_coords(double **d_buffer, int n_dim) +{ + double *coords = *d_buffer; + (*d_buffer) += n_dim * 2; + return coords; +} + +static void mbr_join(double *a, const double *b, int n_dim) +{ + double *end = a + n_dim * 2; + do + { + if (a[0] > b[0]) + a[0] = b[0]; + + if (a[1] < b[1]) + a[1] = b[1]; + + a += 2; + b += 2; + }while (a != end); +} + +/* +Counts the square of mbr which is a join of a and b +*/ +static double mbr_join_square(const double *a, const double *b, int n_dim) +{ + const double *end = a + n_dim * 2; + double square = 1.0; + do + { + square *= + ((a[1] < b[1]) ? b[1] : a[1]) - ((a[0] > b[0]) ? b[0] : a[0]); + + a += 2; + b += 2; + }while (a != end); + + return square; +} + +static double count_square(const double *a, int n_dim) +{ + const double *end = a + n_dim * 2; + double square = 1.0; + do + { + square *= a[1] - a[0]; + a += 2; + }while (a != end); + return square; +} + +inline static void copy_coords(double *dst, const double *src, int n_dim) +{ + memcpy(dst, src, sizeof(double) * (n_dim * 2)); +} + +/* +Select two nodes to collect group upon +*/ +static void pick_seeds(SplitStruct *node, int n_entries, + SplitStruct **seed_a, SplitStruct **seed_b, int n_dim) +{ + SplitStruct *cur1; + SplitStruct *lim1 = node + (n_entries - 1); + SplitStruct *cur2; + SplitStruct *lim2 = node + n_entries; + + double max_d = -DBL_MAX; + double d; + + for (cur1 = node; cur1 < lim1; ++cur1) + { + for (cur2=cur1 + 1; cur2 < lim2; ++cur2) + { + + d = mbr_join_square(cur1->coords, cur2->coords, n_dim) - cur1->square - + cur2->square; + if (d > max_d) + { + max_d = d; + *seed_a = cur1; + *seed_b = cur2; + } + } + } +} + +/* +Select next node and group where to add +*/ +static void pick_next(SplitStruct *node, int n_entries, double *g1, double *g2, + SplitStruct **choice, int *n_group, int n_dim) +{ + SplitStruct *cur = node; + SplitStruct *end = node + n_entries; + + double max_diff = -DBL_MAX; + + for (; curn_node) + { + continue; + } + + diff = mbr_join_square(g1, cur->coords, n_dim) - + mbr_join_square(g2, cur->coords, n_dim); + + abs_diff = fabs(diff); + if (abs_diff > max_diff) + { + max_diff = abs_diff; + *n_group = 1 + (diff > 0); + *choice = cur; + } + } +} + +/* +Mark not-in-group entries as n_group +*/ +static void mark_all_entries(SplitStruct *node, int n_entries, int n_group) +{ + SplitStruct *cur = node; + SplitStruct *end = node + n_entries; + for (; curn_node) + { + continue; + } + cur->n_node = n_group; + } +} + +static int split_maria_rtree_node(SplitStruct *node, int n_entries, + int all_size, /* Total key's size */ + int key_size, + int min_size, /* Minimal group size */ + int size1, int size2 /* initial group sizes */, + double **d_buffer, int n_dim) +{ + SplitStruct *cur; + SplitStruct *a; + SplitStruct *b; + double *g1 = reserve_coords(d_buffer, n_dim); + double *g2 = reserve_coords(d_buffer, n_dim); + SplitStruct *next; + int next_node; + int i; + SplitStruct *end = node + n_entries; + + if (all_size < min_size * 2) + { + return 1; + } + + cur = node; + for (; cursquare = count_square(cur->coords, n_dim); + cur->n_node = 0; + } + + pick_seeds(node, n_entries, &a, &b, n_dim); + a->n_node = 1; + b->n_node = 2; + + + copy_coords(g1, a->coords, n_dim); + size1 += key_size; + copy_coords(g2, b->coords, n_dim); + size2 += key_size; + + + for (i=n_entries - 2; i>0; --i) + { + if (all_size - (size2 + key_size) < min_size) /* Can't write into group 2 */ + { + mark_all_entries(node, n_entries, 1); + break; + } + + if (all_size - (size1 + key_size) < min_size) /* Can't write into group 1 */ + { + mark_all_entries(node, n_entries, 2); + break; + } + + pick_next(node, n_entries, g1, g2, &next, &next_node, n_dim); + if (next_node == 1) + { + size1 += key_size; + mbr_join(g1, next->coords, n_dim); + } + else + { + size2 += key_size; + mbr_join(g2, next->coords, n_dim); + } + next->n_node = next_node; + } + + return 0; +} + +int maria_rtree_split_page(MARIA_HA *info, MARIA_KEYDEF *keyinfo, uchar *page, uchar *key, + uint key_length, my_off_t *new_page_offs) +{ + int n1, n2; /* Number of items in groups */ + + SplitStruct *task; + SplitStruct *cur; + SplitStruct *stop; + double *coord_buf; + double *next_coord; + double *old_coord; + int n_dim; + uchar *source_cur, *cur1, *cur2; + uchar *new_page; + int err_code= 0; + uint nod_flag= _ma_test_if_nod(page); + uint full_length= key_length + (nod_flag ? nod_flag : + info->s->base.rec_reflength); + int max_keys= (maria_getint(page)-2) / (full_length); + + n_dim = keyinfo->keysegs / 2; + + if (!(coord_buf= (double*) my_alloca(n_dim * 2 * sizeof(double) * + (max_keys + 1 + 4) + + sizeof(SplitStruct) * (max_keys + 1)))) + return -1; + + task= (SplitStruct *)(coord_buf + n_dim * 2 * (max_keys + 1 + 4)); + + next_coord = coord_buf; + + stop = task + max_keys; + source_cur = rt_PAGE_FIRST_KEY(page, nod_flag); + + for (cur = task; cur < stop; ++cur, source_cur = rt_PAGE_NEXT_KEY(source_cur, + key_length, nod_flag)) + { + cur->coords = reserve_coords(&next_coord, n_dim); + cur->key = source_cur; + maria_rtree_d_mbr(keyinfo->seg, source_cur, key_length, cur->coords); + } + + cur->coords = reserve_coords(&next_coord, n_dim); + maria_rtree_d_mbr(keyinfo->seg, key, key_length, cur->coords); + cur->key = key; + + old_coord = next_coord; + + if (split_maria_rtree_node(task, max_keys + 1, + maria_getint(page) + full_length + 2, full_length, + rt_PAGE_MIN_SIZE(keyinfo->block_length), + 2, 2, &next_coord, n_dim)) + { + err_code = 1; + goto split_err; + } + + if (!(new_page = (uchar*)my_alloca((uint)keyinfo->block_length))) + { + err_code= -1; + goto split_err; + } + + stop = task + (max_keys + 1); + cur1 = rt_PAGE_FIRST_KEY(page, nod_flag); + cur2 = rt_PAGE_FIRST_KEY(new_page, nod_flag); + + n1= n2 = 0; + for (cur = task; cur < stop; ++cur) + { + uchar *to; + if (cur->n_node == 1) + { + to = cur1; + cur1 = rt_PAGE_NEXT_KEY(cur1, key_length, nod_flag); + ++n1; + } + else + { + to = cur2; + cur2 = rt_PAGE_NEXT_KEY(cur2, key_length, nod_flag); + ++n2; + } + if (to != cur->key) + memcpy(to - nod_flag, cur->key - nod_flag, full_length); + } + + maria_putint(page, 2 + n1 * full_length, nod_flag); + maria_putint(new_page, 2 + n2 * full_length, nod_flag); + + if ((*new_page_offs= _ma_new(info, keyinfo, DFLT_INIT_HITS)) == + HA_OFFSET_ERROR) + err_code= -1; + else + err_code= _ma_write_keypage(info, keyinfo, *new_page_offs, + DFLT_INIT_HITS, new_page); + + my_afree((byte*)new_page); + +split_err: + my_afree((byte*) coord_buf); + return err_code; +} + +#endif /*HAVE_RTREE_KEYS*/ diff --git a/storage/maria/ma_rt_test.c b/storage/maria/ma_rt_test.c new file mode 100644 index 00000000000..ca4825c2ce2 --- /dev/null +++ b/storage/maria/ma_rt_test.c @@ -0,0 +1,473 @@ +/* Copyright (C) 2006 MySQL AB & MySQL Finland AB & TCX DataKonsult AB + + 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; either version 2 of the License, or + (at your option) any later version. + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ + +/* Testing of the basic functions of a MARIA rtree table */ +/* Written by Alex Barkov who has a shared copyright to this code */ + + +#include "maria.h" + +#ifdef HAVE_RTREE_KEYS + +#include "ma_rt_index.h" + +#define MAX_REC_LENGTH 1024 +#define ndims 2 +#define KEYALG HA_KEY_ALG_RTREE + +static int read_with_pos(MARIA_HA * file, int silent); +static void create_record(char *record,uint rownr); +static void create_record1(char *record,uint rownr); +static void print_record(char * record,my_off_t offs,const char * tail); +static int run_test(const char *filename); + +static double rt_data[]= +{ + /*1*/ 0,10,0,10, + /*2*/ 5,15,0,10, + /*3*/ 0,10,5,15, + /*4*/ 10,20,10,20, + /*5*/ 0,10,0,10, + /*6*/ 5,15,0,10, + /*7*/ 0,10,5,15, + /*8*/ 10,20,10,20, + /*9*/ 0,10,0,10, + /*10*/ 5,15,0,10, + /*11*/ 0,10,5,15, + /*12*/ 10,20,10,20, + /*13*/ 0,10,0,10, + /*14*/ 5,15,0,10, + /*15*/ 0,10,5,15, + /*16*/ 10,20,10,20, + /*17*/ 5,15,0,10, + /*18*/ 0,10,5,15, + /*19*/ 10,20,10,20, + /*20*/ 0,10,0,10, + + /*1*/ 100,110,0,10, + /*2*/ 105,115,0,10, + /*3*/ 100,110,5,15, + /*4*/ 110,120,10,20, + /*5*/ 100,110,0,10, + /*6*/ 105,115,0,10, + /*7*/ 100,110,5,15, + /*8*/ 110,120,10,20, + /*9*/ 100,110,0,10, + /*10*/ 105,115,0,10, + /*11*/ 100,110,5,15, + /*12*/ 110,120,10,20, + /*13*/ 100,110,0,10, + /*14*/ 105,115,0,10, + /*15*/ 100,110,5,15, + /*16*/ 110,120,10,20, + /*17*/ 105,115,0,10, + /*18*/ 100,110,5,15, + /*19*/ 110,120,10,20, + /*20*/ 100,110,0,10, + -1 +}; + +int main(int argc __attribute__((unused)),char *argv[] __attribute__((unused))) +{ + MY_INIT(argv[0]); + maria_init(); + exit(run_test("rt_test")); +} + + +static int run_test(const char *filename) +{ + MARIA_HA *file; + MARIA_UNIQUEDEF uniquedef; + MARIA_CREATE_INFO create_info; + MARIA_COLUMNDEF recinfo[20]; + MARIA_KEYDEF keyinfo[20]; + HA_KEYSEG keyseg[20]; + key_range range; + + int silent=0; + int opt_unique=0; + int create_flag=0; + int key_type=HA_KEYTYPE_DOUBLE; + int key_length=8; + int null_fields=0; + int nrecords=sizeof(rt_data)/(sizeof(double)*4);/* 3000;*/ + int rec_length=0; + int uniques=0; + int i; + int error; + int row_count=0; + char record[MAX_REC_LENGTH]; + char read_record[MAX_REC_LENGTH]; + int upd= 10; + ha_rows hrows; + + /* Define a column for NULLs and DEL markers*/ + + recinfo[0].type=FIELD_NORMAL; + recinfo[0].length=1; /* For NULL bits */ + rec_length=1; + + /* Define 2*ndims columns for coordinates*/ + + for (i=1; i<=2*ndims ;i++){ + recinfo[i].type=FIELD_NORMAL; + recinfo[i].length=key_length; + rec_length+=key_length; + } + + /* Define a key with 2*ndims segments */ + + keyinfo[0].seg=keyseg; + keyinfo[0].keysegs=2*ndims; + keyinfo[0].flag=0; + keyinfo[0].key_alg=KEYALG; + + for (i=0; i<2*ndims; i++){ + keyinfo[0].seg[i].type= key_type; + keyinfo[0].seg[i].flag=0; /* Things like HA_REVERSE_SORT */ + keyinfo[0].seg[i].start= (key_length*i)+1; + keyinfo[0].seg[i].length=key_length; + keyinfo[0].seg[i].null_bit= null_fields ? 2 : 0; + keyinfo[0].seg[i].null_pos=0; + keyinfo[0].seg[i].language=default_charset_info->number; + } + + if (!silent) + printf("- Creating isam-file\n"); + + bzero((char*) &create_info,sizeof(create_info)); + create_info.max_rows=10000000; + + if (maria_create(filename, + 1, /* keys */ + keyinfo, + 1+2*ndims+opt_unique, /* columns */ + recinfo,uniques,&uniquedef,&create_info,create_flag)) + goto err; + + if (!silent) + printf("- Open isam-file\n"); + + if (!(file=maria_open(filename,2,HA_OPEN_ABORT_IF_LOCKED))) + goto err; + + if (!silent) + printf("- Writing key:s\n"); + + for (i=0; i "); + print_record(record,maria_position(file),"\n"); + error=maria_update(file,read_record,record); + if (error) + { + printf("pos: %2d maria_update: %3d errno: %3d\n",i,error,my_errno); + goto err; + } + } + + if ((error=read_with_pos(file,silent))) + goto err; + + if (!silent) + printf("- Test maria_rkey then a sequence of maria_rnext_same\n"); + + create_record(record, nrecords*4/5); + print_record(record,0," search for\n"); + + if ((error=maria_rkey(file,read_record,0,record+1,0,HA_READ_MBR_INTERSECT))) + { + printf("maria_rkey: %3d errno: %3d\n",error,my_errno); + goto err; + } + print_record(read_record,maria_position(file)," maria_rkey\n"); + row_count=1; + + for (;;) + { + if ((error=maria_rnext_same(file,read_record))) + { + if (error==HA_ERR_END_OF_FILE) + break; + printf("maria_next: %3d errno: %3d\n",error,my_errno); + goto err; + } + print_record(read_record,maria_position(file)," maria_rnext_same\n"); + row_count++; + } + printf(" %d rows\n",row_count); + + if (!silent) + printf("- Test maria_rfirst then a sequence of maria_rnext\n"); + + error=maria_rfirst(file,read_record,0); + if (error) + { + printf("maria_rfirst: %3d errno: %3d\n",error,my_errno); + goto err; + } + row_count=1; + print_record(read_record,maria_position(file)," maria_frirst\n"); + + for (i=0;inextpos=info->s->pack.header_length; /* Read first record */ + info->lastinx= -1; /* Can't forward or backward */ + if (info->opt_flag & WRITE_CACHE_USED && flush_io_cache(&info->rec_cache)) + DBUG_RETURN(my_errno); + DBUG_RETURN(0); +} + +/* + Read a row based on position. + If filepos= HA_OFFSET_ERROR then read next row + Return values + Returns one of following values: + 0 = Ok. + HA_ERR_END_OF_FILE = EOF. +*/ + +int maria_scan(MARIA_HA *info, byte *buf) +{ + DBUG_ENTER("maria_scan"); + /* Init all but update-flag */ + info->update&= (HA_STATE_CHANGED | HA_STATE_ROW_CHANGED); + DBUG_RETURN ((*info->s->read_rnd)(info,buf,info->nextpos,1)); +} diff --git a/storage/maria/ma_search.c b/storage/maria/ma_search.c new file mode 100644 index 00000000000..3bb048ea239 --- /dev/null +++ b/storage/maria/ma_search.c @@ -0,0 +1,1894 @@ +/* Copyright (C) 2006 MySQL AB & MySQL Finland AB & TCX DataKonsult AB + + 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; either version 2 of the License, or + (at your option) any later version. + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ + +/* key handling functions */ + +#include "ma_fulltext.h" +#include "m_ctype.h" + +static my_bool _ma_get_prev_key(MARIA_HA *info, MARIA_KEYDEF *keyinfo, uchar *page, + uchar *key, uchar *keypos, + uint *return_key_length); + + /* Check index */ + +int _ma_check_index(MARIA_HA *info, int inx) +{ + if (inx == -1) /* Use last index */ + inx=info->lastinx; + if (inx < 0 || ! maria_is_key_active(info->s->state.key_map, inx)) + { + my_errno=HA_ERR_WRONG_INDEX; + return -1; + } + if (info->lastinx != inx) /* Index changed */ + { + info->lastinx = inx; + info->page_changed=1; + info->update= ((info->update & (HA_STATE_CHANGED | HA_STATE_ROW_CHANGED)) | + HA_STATE_NEXT_FOUND | HA_STATE_PREV_FOUND); + } + if (info->opt_flag & WRITE_CACHE_USED && flush_io_cache(&info->rec_cache)) + return(-1); + return(inx); +} /* _ma_check_index */ + + + /* + ** Search after row by a key + ** Position to row is stored in info->lastpos + ** Return: -1 if not found + ** 1 if one should continue search on higher level + */ + +int _ma_search(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, + uchar *key, uint key_len, uint nextflag, register my_off_t pos) +{ + my_bool last_key; + int error,flag; + uint nod_flag; + uchar *keypos,*maxpos; + uchar lastkey[HA_MAX_KEY_BUFF],*buff; + DBUG_ENTER("_ma_search"); + DBUG_PRINT("enter",("pos: %lu nextflag: %u lastpos: %lu", + (ulong) pos, nextflag, (ulong) info->lastpos)); + DBUG_EXECUTE("key", _ma_print_key(DBUG_FILE,keyinfo->seg,key,key_len);); + + if (pos == HA_OFFSET_ERROR) + { + my_errno=HA_ERR_KEY_NOT_FOUND; /* Didn't find key */ + info->lastpos= HA_OFFSET_ERROR; + if (!(nextflag & (SEARCH_SMALLER | SEARCH_BIGGER | SEARCH_LAST))) + DBUG_RETURN(-1); /* Not found ; return error */ + DBUG_RETURN(1); /* Search at upper levels */ + } + + if (!(buff= _ma_fetch_keypage(info,keyinfo,pos,DFLT_INIT_HITS,info->buff, + test(!(nextflag & SEARCH_SAVE_BUFF))))) + goto err; + DBUG_DUMP("page",(byte*) buff,maria_getint(buff)); + + flag=(*keyinfo->bin_search)(info,keyinfo,buff,key,key_len,nextflag, + &keypos,lastkey, &last_key); + if (flag == MARIA_FOUND_WRONG_KEY) + DBUG_RETURN(-1); + nod_flag=_ma_test_if_nod(buff); + maxpos=buff+maria_getint(buff)-1; + + if (flag) + { + if ((error= _ma_search(info,keyinfo,key,key_len,nextflag, + _ma_kpos(nod_flag,keypos))) <= 0) + DBUG_RETURN(error); + + if (flag >0) + { + if (nextflag & (SEARCH_SMALLER | SEARCH_LAST) && + keypos == buff+2+nod_flag) + DBUG_RETURN(1); /* Bigger than key */ + } + else if (nextflag & SEARCH_BIGGER && keypos >= maxpos) + DBUG_RETURN(1); /* Smaller than key */ + } + else + { + if ((nextflag & SEARCH_FIND) && nod_flag && + ((keyinfo->flag & (HA_NOSAME | HA_NULL_PART)) != HA_NOSAME || + key_len != USE_WHOLE_KEY)) + { + if ((error= _ma_search(info,keyinfo,key,key_len,SEARCH_FIND, + _ma_kpos(nod_flag,keypos))) >= 0 || + my_errno != HA_ERR_KEY_NOT_FOUND) + DBUG_RETURN(error); + info->last_keypage= HA_OFFSET_ERROR; /* Buffer not in mem */ + } + } + if (pos != info->last_keypage) + { + uchar *old_buff=buff; + if (!(buff= _ma_fetch_keypage(info,keyinfo,pos,DFLT_INIT_HITS,info->buff, + test(!(nextflag & SEARCH_SAVE_BUFF))))) + goto err; + keypos=buff+(keypos-old_buff); + maxpos=buff+(maxpos-old_buff); + } + + if ((nextflag & (SEARCH_SMALLER | SEARCH_LAST)) && flag != 0) + { + uint not_used[2]; + if (_ma_get_prev_key(info,keyinfo, buff, info->lastkey, keypos, + &info->lastkey_length)) + goto err; + if (!(nextflag & SEARCH_SMALLER) && + ha_key_cmp(keyinfo->seg, info->lastkey, key, key_len, SEARCH_FIND, + not_used)) + { + my_errno=HA_ERR_KEY_NOT_FOUND; /* Didn't find key */ + goto err; + } + } + else + { + info->lastkey_length=(*keyinfo->get_key)(keyinfo,nod_flag,&keypos,lastkey); + if (!info->lastkey_length) + goto err; + memcpy(info->lastkey,lastkey,info->lastkey_length); + } + info->lastpos= _ma_dpos(info,0,info->lastkey+info->lastkey_length); + /* Save position for a possible read next / previous */ + info->int_keypos=info->buff+ (keypos-buff); + info->int_maxpos=info->buff+ (maxpos-buff); + info->int_nod_flag=nod_flag; + info->int_keytree_version=keyinfo->version; + info->last_search_keypage=info->last_keypage; + info->page_changed=0; + info->buff_used= (info->buff != buff); /* If we have to reread buff */ + + DBUG_PRINT("exit",("found key at %lu",(ulong) info->lastpos)); + DBUG_RETURN(0); + +err: + DBUG_PRINT("exit",("Error: %d",my_errno)); + info->lastpos= HA_OFFSET_ERROR; + info->page_changed=1; + DBUG_RETURN (-1); +} /* _ma_search */ + + + /* Search after key in page-block */ + /* If packed key puts smaller or identical key in buff */ + /* ret_pos point to where find or bigger key starts */ + /* ARGSUSED */ + +int _ma_bin_search(MARIA_HA *info, register MARIA_KEYDEF *keyinfo, uchar *page, + uchar *key, uint key_len, uint comp_flag, uchar **ret_pos, + uchar *buff __attribute__((unused)), my_bool *last_key) +{ + reg4 int start,mid,end,save_end; + int flag; + uint totlength,nod_flag,not_used[2]; + DBUG_ENTER("_ma_bin_search"); + + LINT_INIT(flag); + totlength=keyinfo->keylength+(nod_flag=_ma_test_if_nod(page)); + start=0; mid=1; + save_end=end=(int) ((maria_getint(page)-2-nod_flag)/totlength-1); + DBUG_PRINT("test",("maria_getint: %d end: %d",maria_getint(page),end)); + page+=2+nod_flag; + + while (start != end) + { + mid= (start+end)/2; + if ((flag=ha_key_cmp(keyinfo->seg,page+(uint) mid*totlength,key,key_len, + comp_flag, not_used)) + >= 0) + end=mid; + else + start=mid+1; + } + if (mid != start) + flag=ha_key_cmp(keyinfo->seg,page+(uint) start*totlength,key,key_len, + comp_flag, not_used); + if (flag < 0) + start++; /* point at next, bigger key */ + *ret_pos=page+(uint) start*totlength; + *last_key= end == save_end; + DBUG_PRINT("exit",("flag: %d keypos: %d",flag,start)); + DBUG_RETURN(flag); +} /* _ma_bin_search */ + + +/* + Locate a packed key in a key page. + + SYNOPSIS + _ma_seq_search() + info Open table information. + keyinfo Key definition information. + page Key page (beginning). + key Search key. + key_len Length to use from search key or USE_WHOLE_KEY + comp_flag Search flags like SEARCH_SAME etc. + ret_pos RETURN Position in key page behind this key. + buff RETURN Copy of previous or identical unpacked key. + last_key RETURN If key is last in page. + + DESCRIPTION + Used instead of _ma_bin_search() when key is packed. + Puts smaller or identical key in buff. + Key is searched sequentially. + + RETURN + > 0 Key in 'buff' is smaller than search key. + 0 Key in 'buff' is identical to search key. + < 0 Not found. +*/ + +int _ma_seq_search(MARIA_HA *info, register MARIA_KEYDEF *keyinfo, uchar *page, + uchar *key, uint key_len, uint comp_flag, uchar **ret_pos, + uchar *buff, my_bool *last_key) +{ + int flag; + uint nod_flag,length,not_used[2]; + uchar t_buff[HA_MAX_KEY_BUFF],*end; + DBUG_ENTER("_ma_seq_search"); + + LINT_INIT(flag); LINT_INIT(length); + end= page+maria_getint(page); + nod_flag=_ma_test_if_nod(page); + page+=2+nod_flag; + *ret_pos=page; + t_buff[0]=0; /* Avoid bugs */ + while (page < end) + { + length=(*keyinfo->get_key)(keyinfo,nod_flag,&page,t_buff); + if (length == 0 || page > end) + { + maria_print_error(info->s, HA_ERR_CRASHED); + my_errno=HA_ERR_CRASHED; + DBUG_PRINT("error",("Found wrong key: length: %u page: %lx end: %lx", + length, (long) page, (long) end)); + DBUG_RETURN(MARIA_FOUND_WRONG_KEY); + } + if ((flag=ha_key_cmp(keyinfo->seg,t_buff,key,key_len,comp_flag, + not_used)) >= 0) + break; +#ifdef EXTRA_DEBUG + DBUG_PRINT("loop",("page: %lx key: '%s' flag: %d", (long) page, t_buff, + flag)); +#endif + memcpy(buff,t_buff,length); + *ret_pos=page; + } + if (flag == 0) + memcpy(buff,t_buff,length); /* Result is first key */ + *last_key= page == end; + DBUG_PRINT("exit",("flag: %d ret_pos: %lx", flag, (long) *ret_pos)); + DBUG_RETURN(flag); +} /* _ma_seq_search */ + + +int _ma_prefix_search(MARIA_HA *info, register MARIA_KEYDEF *keyinfo, uchar *page, + uchar *key, uint key_len, uint nextflag, uchar **ret_pos, + uchar *buff, my_bool *last_key) +{ + /* + my_flag is raw comparison result to be changed according to + SEARCH_NO_FIND,SEARCH_LAST and HA_REVERSE_SORT flags. + flag is the value returned by ha_key_cmp and as treated as final + */ + int flag=0, my_flag=-1; + uint nod_flag, length, len, matched, cmplen, kseg_len; + uint prefix_len,suffix_len; + int key_len_skip, seg_len_pack, key_len_left; + uchar *end, *kseg, *vseg; + uchar *sort_order=keyinfo->seg->charset->sort_order; + uchar tt_buff[HA_MAX_KEY_BUFF+2], *t_buff=tt_buff+2; + uchar *saved_from, *saved_to, *saved_vseg; + uint saved_length=0, saved_prefix_len=0; + uint length_pack; + DBUG_ENTER("_ma_prefix_search"); + + LINT_INIT(length); + LINT_INIT(prefix_len); + LINT_INIT(seg_len_pack); + LINT_INIT(saved_from); + LINT_INIT(saved_to); + LINT_INIT(saved_vseg); + + t_buff[0]=0; /* Avoid bugs */ + end= page+maria_getint(page); + nod_flag=_ma_test_if_nod(page); + page+=2+nod_flag; + *ret_pos=page; + kseg=key; + + get_key_pack_length(kseg_len,length_pack,kseg); + key_len_skip=length_pack+kseg_len; + key_len_left=(int) key_len- (int) key_len_skip; + /* If key_len is 0, then lenght_pack is 1, then key_len_left is -1. */ + cmplen=(key_len_left>=0) ? kseg_len : key_len-length_pack; + DBUG_PRINT("info",("key: '%.*s'",kseg_len,kseg)); + + /* + Keys are compressed the following way: + + If the max length of first key segment <= 127 bytes the prefix is + 1 byte else it's 2 byte + + (prefix) length The high bit is set if this is a prefix for the prev key. + [suffix length] Packed length of suffix if the previous was a prefix. + (suffix) data Key data bytes (past the common prefix or whole segment). + [next-key-seg] Next key segments (([packed length], data), ...) + pointer Reference to the data file (last_keyseg->length). + */ + + matched=0; /* how many char's from prefix were alredy matched */ + len=0; /* length of previous key unpacked */ + + while (page < end) + { + uint packed= *page & 128; + + vseg=page; + if (keyinfo->seg->length >= 127) + { + suffix_len=mi_uint2korr(vseg) & 32767; + vseg+=2; + } + else + suffix_len= *vseg++ & 127; + + if (packed) + { + if (suffix_len == 0) + { + /* == 0x80 or 0x8000, same key, prefix length == old key length. */ + prefix_len=len; + } + else + { + /* > 0x80 or 0x8000, this is prefix lgt, packed suffix lgt follows. */ + prefix_len=suffix_len; + get_key_length(suffix_len,vseg); + } + } + else + { + /* Not packed. No prefix used from last key. */ + prefix_len=0; + } + + len=prefix_len+suffix_len; + seg_len_pack=get_pack_length(len); + t_buff=tt_buff+3-seg_len_pack; + store_key_length(t_buff,len); + + if (prefix_len > saved_prefix_len) + memcpy(t_buff+seg_len_pack+saved_prefix_len,saved_vseg, + prefix_len-saved_prefix_len); + saved_vseg=vseg; + saved_prefix_len=prefix_len; + + DBUG_PRINT("loop",("page: '%.*s%.*s'",prefix_len,t_buff+seg_len_pack, + suffix_len,vseg)); + { + uchar *from=vseg+suffix_len; + HA_KEYSEG *keyseg; + uint l; + + for (keyseg=keyinfo->seg+1 ; keyseg->type ; keyseg++ ) + { + + if (keyseg->flag & HA_NULL_PART) + { + if (!(*from++)) + continue; + } + if (keyseg->flag & (HA_VAR_LENGTH_PART | HA_BLOB_PART | HA_SPACE_PACK)) + { + get_key_length(l,from); + } + else + l=keyseg->length; + + from+=l; + } + from+=keyseg->length; + page=from+nod_flag; + length=from-vseg; + } + + if (page > end) + { + maria_print_error(info->s, HA_ERR_CRASHED); + my_errno=HA_ERR_CRASHED; + DBUG_PRINT("error",("Found wrong key: length: %u page: %lx end: %lx", + length, (long) page, (long) end)); + DBUG_RETURN(MARIA_FOUND_WRONG_KEY); + } + + if (matched >= prefix_len) + { + /* We have to compare. But we can still skip part of the key */ + uint left; + uchar *k=kseg+prefix_len; + + /* + If prefix_len > cmplen then we are in the end-space comparison + phase. Do not try to acces the key any more ==> left= 0. + */ + left= ((len <= cmplen) ? suffix_len : + ((prefix_len < cmplen) ? cmplen - prefix_len : 0)); + + matched=prefix_len+left; + + if (sort_order) + { + for (my_flag=0;left;left--) + if ((my_flag= (int) sort_order[*vseg++] - (int) sort_order[*k++])) + break; + } + else + { + for (my_flag=0;left;left--) + if ((my_flag= (int) *vseg++ - (int) *k++)) + break; + } + + if (my_flag>0) /* mismatch */ + break; + if (my_flag==0) /* match */ + { + /* + ** len cmplen seg_left_len more_segs + ** < matched=len; continue search + ** > = prefix ? found : (matched=len; continue search) + ** > < - ok, found + ** = < - ok, found + ** = = - ok, found + ** = = + next seg + */ + if (len < cmplen) + { + if ((keyinfo->seg->type != HA_KEYTYPE_TEXT && + keyinfo->seg->type != HA_KEYTYPE_VARTEXT1 && + keyinfo->seg->type != HA_KEYTYPE_VARTEXT2)) + my_flag= -1; + else + { + /* We have to compare k and vseg as if they were space extended */ + uchar *end= k+ (cmplen - len); + for ( ; k < end && *k == ' '; k++) ; + if (k == end) + goto cmp_rest; /* should never happen */ + if (*k < (uchar) ' ') + { + my_flag= 1; /* Compared string is smaller */ + break; + } + my_flag= -1; /* Continue searching */ + } + } + else if (len > cmplen) + { + uchar *end; + if ((nextflag & SEARCH_PREFIX) && key_len_left == 0) + goto fix_flag; + + /* We have to compare k and vseg as if they were space extended */ + for (end=vseg + (len-cmplen) ; + vseg < end && *vseg == (uchar) ' '; + vseg++, matched++) ; + DBUG_ASSERT(vseg < end); + + if (*vseg > (uchar) ' ') + { + my_flag= 1; /* Compared string is smaller */ + break; + } + my_flag= -1; /* Continue searching */ + } + else + { + cmp_rest: + if (key_len_left>0) + { + uint not_used[2]; + if ((flag = ha_key_cmp(keyinfo->seg+1,vseg, + k, key_len_left, nextflag, not_used)) >= 0) + break; + } + else + { + /* + at this line flag==-1 if the following lines were already + visited and 0 otherwise, i.e. flag <=0 here always !!! + */ + fix_flag: + DBUG_ASSERT(flag <= 0); + if (nextflag & (SEARCH_NO_FIND | SEARCH_LAST)) + flag=(nextflag & (SEARCH_BIGGER | SEARCH_LAST)) ? -1 : 1; + if (flag>=0) + break; + } + } + } + matched-=left; + } + /* else (matched < prefix_len) ---> do nothing. */ + + memcpy(buff,t_buff,saved_length=seg_len_pack+prefix_len); + saved_to=buff+saved_length; + saved_from=saved_vseg; + saved_length=length; + *ret_pos=page; + } + if (my_flag) + flag=(keyinfo->seg->flag & HA_REVERSE_SORT) ? -my_flag : my_flag; + if (flag == 0) + { + memcpy(buff,t_buff,saved_length=seg_len_pack+prefix_len); + saved_to=buff+saved_length; + saved_from=saved_vseg; + saved_length=length; + } + if (saved_length) + memcpy(saved_to,saved_from,saved_length); + + *last_key= page == end; + + DBUG_PRINT("exit",("flag: %d ret_pos: %lx", flag, (long) *ret_pos)); + DBUG_RETURN(flag); +} /* _ma_prefix_search */ + + + /* Get pos to a key_block */ + +my_off_t _ma_kpos(uint nod_flag, uchar *after_key) +{ + after_key-=nod_flag; + switch (nod_flag) { +#if SIZEOF_OFF_T > 4 + case 7: + return mi_uint7korr(after_key)*MARIA_MIN_KEY_BLOCK_LENGTH; + case 6: + return mi_uint6korr(after_key)*MARIA_MIN_KEY_BLOCK_LENGTH; + case 5: + return mi_uint5korr(after_key)*MARIA_MIN_KEY_BLOCK_LENGTH; +#else + case 7: + after_key++; + case 6: + after_key++; + case 5: + after_key++; +#endif + case 4: + return ((my_off_t) mi_uint4korr(after_key))*MARIA_MIN_KEY_BLOCK_LENGTH; + case 3: + return ((my_off_t) mi_uint3korr(after_key))*MARIA_MIN_KEY_BLOCK_LENGTH; + case 2: + return (my_off_t) (mi_uint2korr(after_key)*MARIA_MIN_KEY_BLOCK_LENGTH); + case 1: + return (uint) (*after_key)*MARIA_MIN_KEY_BLOCK_LENGTH; + case 0: /* At leaf page */ + default: /* Impossible */ + return(HA_OFFSET_ERROR); + } +} /* _kpos */ + + + /* Save pos to a key_block */ + +void _ma_kpointer(register MARIA_HA *info, register uchar *buff, my_off_t pos) +{ + pos/=MARIA_MIN_KEY_BLOCK_LENGTH; + switch (info->s->base.key_reflength) { +#if SIZEOF_OFF_T > 4 + case 7: mi_int7store(buff,pos); break; + case 6: mi_int6store(buff,pos); break; + case 5: mi_int5store(buff,pos); break; +#else + case 7: *buff++=0; + /* fall trough */ + case 6: *buff++=0; + /* fall trough */ + case 5: *buff++=0; + /* fall trough */ +#endif + case 4: mi_int4store(buff,pos); break; + case 3: mi_int3store(buff,pos); break; + case 2: mi_int2store(buff,(uint) pos); break; + case 1: buff[0]= (uchar) pos; break; + default: abort(); /* impossible */ + } +} /* _ma_kpointer */ + + + /* Calc pos to a data-record from a key */ + + +my_off_t _ma_dpos(MARIA_HA *info, uint nod_flag, uchar *after_key) +{ + my_off_t pos; + after_key-=(nod_flag + info->s->rec_reflength); + switch (info->s->rec_reflength) { +#if SIZEOF_OFF_T > 4 + case 8: pos= (my_off_t) mi_uint8korr(after_key); break; + case 7: pos= (my_off_t) mi_uint7korr(after_key); break; + case 6: pos= (my_off_t) mi_uint6korr(after_key); break; + case 5: pos= (my_off_t) mi_uint5korr(after_key); break; +#else + case 8: pos= (my_off_t) mi_uint4korr(after_key+4); break; + case 7: pos= (my_off_t) mi_uint4korr(after_key+3); break; + case 6: pos= (my_off_t) mi_uint4korr(after_key+2); break; + case 5: pos= (my_off_t) mi_uint4korr(after_key+1); break; +#endif + case 4: pos= (my_off_t) mi_uint4korr(after_key); break; + case 3: pos= (my_off_t) mi_uint3korr(after_key); break; + case 2: pos= (my_off_t) mi_uint2korr(after_key); break; + default: + pos=0L; /* Shut compiler up */ + } + return (info->s->options & + (HA_OPTION_PACK_RECORD | HA_OPTION_COMPRESS_RECORD)) ? pos : + pos*info->s->base.pack_reclength; +} + + +/* Calc position from a record pointer ( in delete link chain ) */ + +my_off_t _ma_rec_pos(MARIA_SHARE *s, uchar *ptr) +{ + my_off_t pos; + switch (s->rec_reflength) { +#if SIZEOF_OFF_T > 4 + case 8: + pos= (my_off_t) mi_uint8korr(ptr); + if (pos == HA_OFFSET_ERROR) + return HA_OFFSET_ERROR; /* end of list */ + break; + case 7: + pos= (my_off_t) mi_uint7korr(ptr); + if (pos == (((my_off_t) 1) << 56) -1) + return HA_OFFSET_ERROR; /* end of list */ + break; + case 6: + pos= (my_off_t) mi_uint6korr(ptr); + if (pos == (((my_off_t) 1) << 48) -1) + return HA_OFFSET_ERROR; /* end of list */ + break; + case 5: + pos= (my_off_t) mi_uint5korr(ptr); + if (pos == (((my_off_t) 1) << 40) -1) + return HA_OFFSET_ERROR; /* end of list */ + break; +#else + case 8: + case 7: + case 6: + case 5: + ptr+= (s->rec_reflength-4); + /* fall through */ +#endif + case 4: + pos= (my_off_t) mi_uint4korr(ptr); + if (pos == (my_off_t) (uint32) ~0L) + return HA_OFFSET_ERROR; + break; + case 3: + pos= (my_off_t) mi_uint3korr(ptr); + if (pos == (my_off_t) (1 << 24) -1) + return HA_OFFSET_ERROR; + break; + case 2: + pos= (my_off_t) mi_uint2korr(ptr); + if (pos == (my_off_t) (1 << 16) -1) + return HA_OFFSET_ERROR; + break; + default: abort(); /* Impossible */ + } + return ((s->options & + (HA_OPTION_PACK_RECORD | HA_OPTION_COMPRESS_RECORD)) ? pos : + pos*s->base.pack_reclength); +} + + + /* save position to record */ + +void _ma_dpointer(MARIA_HA *info, uchar *buff, my_off_t pos) +{ + if (!(info->s->options & + (HA_OPTION_PACK_RECORD | HA_OPTION_COMPRESS_RECORD)) && + pos != HA_OFFSET_ERROR) + pos/=info->s->base.pack_reclength; + + switch (info->s->rec_reflength) { +#if SIZEOF_OFF_T > 4 + case 8: mi_int8store(buff,pos); break; + case 7: mi_int7store(buff,pos); break; + case 6: mi_int6store(buff,pos); break; + case 5: mi_int5store(buff,pos); break; +#else + case 8: *buff++=0; + /* fall trough */ + case 7: *buff++=0; + /* fall trough */ + case 6: *buff++=0; + /* fall trough */ + case 5: *buff++=0; + /* fall trough */ +#endif + case 4: mi_int4store(buff,pos); break; + case 3: mi_int3store(buff,pos); break; + case 2: mi_int2store(buff,(uint) pos); break; + default: abort(); /* Impossible */ + } +} /* _ma_dpointer */ + + + /* Get key from key-block */ + /* page points at previous key; its advanced to point at next key */ + /* key should contain previous key */ + /* Returns length of found key + pointers */ + /* nod_flag is a flag if we are on nod */ + + /* same as _ma_get_key but used with fixed length keys */ + +uint _ma_get_static_key(register MARIA_KEYDEF *keyinfo, uint nod_flag, + register uchar **page, register uchar *key) +{ + memcpy((byte*) key,(byte*) *page, + (size_t) (keyinfo->keylength+nod_flag)); + *page+=keyinfo->keylength+nod_flag; + return(keyinfo->keylength); +} /* _ma_get_static_key */ + + +/* + get key witch is packed against previous key or key with a NULL column. + + SYNOPSIS + _ma_get_pack_key() + keyinfo key definition information. + nod_flag If nod: Length of node pointer, else zero. + page_pos RETURN position in key page behind this key. + key IN/OUT in: prev key, out: unpacked key. + + RETURN + key_length + length of data pointer +*/ + +uint _ma_get_pack_key(register MARIA_KEYDEF *keyinfo, uint nod_flag, + register uchar **page_pos, register uchar *key) +{ + reg1 HA_KEYSEG *keyseg; + uchar *start_key,*page=*page_pos; + uint length; + + start_key=key; + for (keyseg=keyinfo->seg ; keyseg->type ;keyseg++) + { + if (keyseg->flag & HA_PACK_KEY) + { + /* key with length, packed to previous key */ + uchar *start=key; + uint packed= *page & 128,tot_length,rest_length; + if (keyseg->length >= 127) + { + length=mi_uint2korr(page) & 32767; + page+=2; + } + else + length= *page++ & 127; + + if (packed) + { + if (length > (uint) keyseg->length) + { + maria_print_error(keyinfo->share, HA_ERR_CRASHED); + my_errno=HA_ERR_CRASHED; + return 0; /* Error */ + } + if (length == 0) /* Same key */ + { + if (keyseg->flag & HA_NULL_PART) + *key++=1; /* Can't be NULL */ + get_key_length(length,key); + key+= length; /* Same diff_key as prev */ + if (length > keyseg->length) + { + DBUG_PRINT("error", + ("Found too long null packed key: %u of %u at %lx", + length, keyseg->length, (long) *page_pos)); + DBUG_DUMP("key",(char*) *page_pos,16); + maria_print_error(keyinfo->share, HA_ERR_CRASHED); + my_errno=HA_ERR_CRASHED; + return 0; + } + continue; + } + if (keyseg->flag & HA_NULL_PART) + { + key++; /* Skip null marker*/ + start++; + } + + get_key_length(rest_length,page); + tot_length=rest_length+length; + + /* If the stored length has changed, we must move the key */ + if (tot_length >= 255 && *start != 255) + { + /* length prefix changed from a length of one to a length of 3 */ + bmove_upp((char*) key+length+3,(char*) key+length+1,length); + *key=255; + mi_int2store(key+1,tot_length); + key+=3+length; + } + else if (tot_length < 255 && *start == 255) + { + bmove(key+1,key+3,length); + *key=tot_length; + key+=1+length; + } + else + { + store_key_length_inc(key,tot_length); + key+=length; + } + memcpy(key,page,rest_length); + page+=rest_length; + key+=rest_length; + continue; + } + else + { + if (keyseg->flag & HA_NULL_PART) + { + if (!length--) /* Null part */ + { + *key++=0; + continue; + } + *key++=1; /* Not null */ + } + } + if (length > (uint) keyseg->length) + { + DBUG_PRINT("error",("Found too long packed key: %u of %u at %lx", + length, keyseg->length, (long) *page_pos)); + DBUG_DUMP("key",(char*) *page_pos,16); + maria_print_error(keyinfo->share, HA_ERR_CRASHED); + my_errno=HA_ERR_CRASHED; + return 0; /* Error */ + } + store_key_length_inc(key,length); + } + else + { + if (keyseg->flag & HA_NULL_PART) + { + if (!(*key++ = *page++)) + continue; + } + if (keyseg->flag & + (HA_VAR_LENGTH_PART | HA_BLOB_PART | HA_SPACE_PACK)) + { + uchar *tmp=page; + get_key_length(length,tmp); + length+=(uint) (tmp-page); + } + else + length=keyseg->length; + } + memcpy((byte*) key,(byte*) page,(size_t) length); + key+=length; + page+=length; + } + length=keyseg->length+nod_flag; + bmove((byte*) key,(byte*) page,length); + *page_pos= page+length; + return ((uint) (key-start_key)+keyseg->length); +} /* _ma_get_pack_key */ + + + +/* key that is packed relatively to previous */ + +uint _ma_get_binary_pack_key(register MARIA_KEYDEF *keyinfo, uint nod_flag, + register uchar **page_pos, register uchar *key) +{ + reg1 HA_KEYSEG *keyseg; + uchar *start_key,*page,*page_end,*from,*from_end; + uint length,tmp; + DBUG_ENTER("_ma_get_binary_pack_key"); + + page= *page_pos; + page_end=page+HA_MAX_KEY_BUFF+1; + start_key=key; + + /* + Keys are compressed the following way: + + prefix length Packed length of prefix for the prev key. (1 or 3 bytes) + for each key segment: + [is null] Null indicator if can be null (1 byte, zero means null) + [length] Packed length if varlength (1 or 3 bytes) + pointer Reference to the data file (last_keyseg->length). + */ + get_key_length(length,page); + if (length) + { + if (length > keyinfo->maxlength) + { + DBUG_PRINT("error",("Found too long binary packed key: %u of %u at %lx", + length, keyinfo->maxlength, (long) *page_pos)); + DBUG_DUMP("key",(char*) *page_pos,16); + maria_print_error(keyinfo->share, HA_ERR_CRASHED); + my_errno=HA_ERR_CRASHED; + DBUG_RETURN(0); /* Wrong key */ + } + from=key; from_end=key+length; + } + else + { + from=page; from_end=page_end; /* Not packed key */ + } + + /* + The trouble is that key is split in two parts: + The first part is in from ...from_end-1. + The second part starts at page + */ + for (keyseg=keyinfo->seg ; keyseg->type ;keyseg++) + { + if (keyseg->flag & HA_NULL_PART) + { + if (from == from_end) { from=page; from_end=page_end; } + if (!(*key++ = *from++)) + continue; /* Null part */ + } + if (keyseg->flag & (HA_VAR_LENGTH_PART | HA_BLOB_PART | HA_SPACE_PACK)) + { + /* Get length of dynamic length key part */ + if (from == from_end) { from=page; from_end=page_end; } + if ((length= (*key++ = *from++)) == 255) + { + if (from == from_end) { from=page; from_end=page_end; } + length= (uint) ((*key++ = *from++)) << 8; + if (from == from_end) { from=page; from_end=page_end; } + length+= (uint) ((*key++ = *from++)); + } + } + else + length=keyseg->length; + + if ((tmp=(uint) (from_end-from)) <= length) + { + key+=tmp; /* Use old key */ + length-=tmp; + from=page; from_end=page_end; + } + DBUG_PRINT("info",("key: %lx from: %lx length: %u", + (long) key, (long) from, length)); + memmove((byte*) key, (byte*) from, (size_t) length); + key+=length; + from+=length; + } + length=keyseg->length+nod_flag; + if ((tmp=(uint) (from_end-from)) <= length) + { + memcpy(key+tmp,page,length-tmp); /* Get last part of key */ + *page_pos= page+length-tmp; + } + else + { + if (from_end != page_end) + { + DBUG_PRINT("error",("Error when unpacking key")); + maria_print_error(keyinfo->share, HA_ERR_CRASHED); + my_errno=HA_ERR_CRASHED; + DBUG_RETURN(0); /* Error */ + } + memcpy((byte*) key,(byte*) from,(size_t) length); + *page_pos= from+length; + } + DBUG_RETURN((uint) (key-start_key)+keyseg->length); +} + + + /* Get key at position without knowledge of previous key */ + /* Returns pointer to next key */ + +uchar *_ma_get_key(MARIA_HA *info, MARIA_KEYDEF *keyinfo, uchar *page, + uchar *key, uchar *keypos, uint *return_key_length) +{ + uint nod_flag; + DBUG_ENTER("_ma_get_key"); + + nod_flag=_ma_test_if_nod(page); + if (! (keyinfo->flag & (HA_VAR_LENGTH_KEY | HA_BINARY_PACK_KEY))) + { + bmove((byte*) key,(byte*) keypos,keyinfo->keylength+nod_flag); + DBUG_RETURN(keypos+keyinfo->keylength+nod_flag); + } + else + { + page+=2+nod_flag; + key[0]=0; /* safety */ + while (page <= keypos) + { + *return_key_length=(*keyinfo->get_key)(keyinfo,nod_flag,&page,key); + if (*return_key_length == 0) + { + maria_print_error(info->s, HA_ERR_CRASHED); + my_errno=HA_ERR_CRASHED; + DBUG_RETURN(0); + } + } + } + DBUG_PRINT("exit",("page: %lx length: %u", (long) page, + *return_key_length)); + DBUG_RETURN(page); +} /* _ma_get_key */ + + + /* Get key at position without knowledge of previous key */ + /* Returns 0 if ok */ + +static my_bool _ma_get_prev_key(MARIA_HA *info, MARIA_KEYDEF *keyinfo, uchar *page, + uchar *key, uchar *keypos, + uint *return_key_length) +{ + uint nod_flag; + DBUG_ENTER("_ma_get_prev_key"); + + nod_flag=_ma_test_if_nod(page); + if (! (keyinfo->flag & (HA_VAR_LENGTH_KEY | HA_BINARY_PACK_KEY))) + { + *return_key_length=keyinfo->keylength; + bmove((byte*) key,(byte*) keypos- *return_key_length-nod_flag, + *return_key_length); + DBUG_RETURN(0); + } + else + { + page+=2+nod_flag; + key[0]=0; /* safety */ + while (page < keypos) + { + *return_key_length=(*keyinfo->get_key)(keyinfo,nod_flag,&page,key); + if (*return_key_length == 0) + { + maria_print_error(info->s, HA_ERR_CRASHED); + my_errno=HA_ERR_CRASHED; + DBUG_RETURN(1); + } + } + } + DBUG_RETURN(0); +} /* _ma_get_key */ + + + + /* Get last key from key-page */ + /* Return pointer to where key starts */ + +uchar *_ma_get_last_key(MARIA_HA *info, MARIA_KEYDEF *keyinfo, uchar *page, + uchar *lastkey, uchar *endpos, uint *return_key_length) +{ + uint nod_flag; + uchar *lastpos; + DBUG_ENTER("_ma_get_last_key"); + DBUG_PRINT("enter",("page: %lx endpos: %lx", (long) page, (long) endpos)); + + nod_flag=_ma_test_if_nod(page); + if (! (keyinfo->flag & (HA_VAR_LENGTH_KEY | HA_BINARY_PACK_KEY))) + { + lastpos=endpos-keyinfo->keylength-nod_flag; + *return_key_length=keyinfo->keylength; + if (lastpos > page) + bmove((byte*) lastkey,(byte*) lastpos,keyinfo->keylength+nod_flag); + } + else + { + lastpos=(page+=2+nod_flag); + lastkey[0]=0; + while (page < endpos) + { + lastpos=page; + *return_key_length=(*keyinfo->get_key)(keyinfo,nod_flag,&page,lastkey); + if (*return_key_length == 0) + { + DBUG_PRINT("error",("Couldn't find last key: page: %lx", + (long) page)); + maria_print_error(info->s, HA_ERR_CRASHED); + my_errno=HA_ERR_CRASHED; + DBUG_RETURN(0); + } + } + } + DBUG_PRINT("exit",("lastpos: %lx length: %u", (long) lastpos, + *return_key_length)); + DBUG_RETURN(lastpos); +} /* _ma_get_last_key */ + + + /* Calculate length of key */ + +uint _ma_keylength(MARIA_KEYDEF *keyinfo, register uchar *key) +{ + reg1 HA_KEYSEG *keyseg; + uchar *start; + + if (! (keyinfo->flag & (HA_VAR_LENGTH_KEY | HA_BINARY_PACK_KEY))) + return (keyinfo->keylength); + + start=key; + for (keyseg=keyinfo->seg ; keyseg->type ; keyseg++) + { + if (keyseg->flag & HA_NULL_PART) + if (!*key++) + continue; + if (keyseg->flag & (HA_SPACE_PACK | HA_BLOB_PART | HA_VAR_LENGTH_PART)) + { + uint length; + get_key_length(length,key); + key+=length; + } + else + key+= keyseg->length; + } + return((uint) (key-start)+keyseg->length); +} /* _ma_keylength */ + + +/* + Calculate length of part key. + + Used in maria_rkey() to find the key found for the key-part that was used. + This is needed in case of multi-byte character sets where we may search + after '0xDF' but find 'ss' +*/ + +uint _ma_keylength_part(MARIA_KEYDEF *keyinfo, register uchar *key, + HA_KEYSEG *end) +{ + reg1 HA_KEYSEG *keyseg; + uchar *start= key; + + for (keyseg=keyinfo->seg ; keyseg != end ; keyseg++) + { + if (keyseg->flag & HA_NULL_PART) + if (!*key++) + continue; + if (keyseg->flag & (HA_SPACE_PACK | HA_BLOB_PART | HA_VAR_LENGTH_PART)) + { + uint length; + get_key_length(length,key); + key+=length; + } + else + key+= keyseg->length; + } + return (uint) (key-start); +} + + /* Move a key */ + +uchar *_ma_move_key(MARIA_KEYDEF *keyinfo, uchar *to, uchar *from) +{ + reg1 uint length; + memcpy((byte*) to, (byte*) from, + (size_t) (length= _ma_keylength(keyinfo,from))); + return to+length; +} + + /* Find next/previous record with same key */ + /* This can't be used when database is touched after last read */ + +int _ma_search_next(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, + uchar *key, uint key_length, uint nextflag, my_off_t pos) +{ + int error; + uint nod_flag; + uchar lastkey[HA_MAX_KEY_BUFF]; + DBUG_ENTER("_ma_search_next"); + DBUG_PRINT("enter",("nextflag: %u lastpos: %lu int_keypos: %lu", + nextflag, (ulong) info->lastpos, + (ulong) info->int_keypos)); + DBUG_EXECUTE("key", _ma_print_key(DBUG_FILE,keyinfo->seg,key,key_length);); + + /* Force full read if we are at last key or if we are not on a leaf + and the key tree has changed since we used it last time + Note that even if the key tree has changed since last read, we can use + the last read data from the leaf if we haven't used the buffer for + something else. + */ + + if (((nextflag & SEARCH_BIGGER) && info->int_keypos >= info->int_maxpos) || + info->page_changed || + (info->int_keytree_version != keyinfo->version && + (info->int_nod_flag || info->buff_used))) + DBUG_RETURN(_ma_search(info,keyinfo,key, USE_WHOLE_KEY, + nextflag | SEARCH_SAVE_BUFF, pos)); + + if (info->buff_used) + { + if (!_ma_fetch_keypage(info,keyinfo,info->last_search_keypage, + DFLT_INIT_HITS,info->buff,0)) + DBUG_RETURN(-1); + info->buff_used=0; + } + + /* Last used buffer is in info->buff */ + nod_flag=_ma_test_if_nod(info->buff); + + if (nextflag & SEARCH_BIGGER) /* Next key */ + { + my_off_t tmp_pos= _ma_kpos(nod_flag,info->int_keypos); + if (tmp_pos != HA_OFFSET_ERROR) + { + if ((error= _ma_search(info,keyinfo,key, USE_WHOLE_KEY, + nextflag | SEARCH_SAVE_BUFF, tmp_pos)) <=0) + DBUG_RETURN(error); + } + memcpy(lastkey,key,key_length); + if (!(info->lastkey_length=(*keyinfo->get_key)(keyinfo,nod_flag, + &info->int_keypos,lastkey))) + DBUG_RETURN(-1); + } + else /* Previous key */ + { + uint length; + /* Find start of previous key */ + info->int_keypos= _ma_get_last_key(info,keyinfo,info->buff,lastkey, + info->int_keypos, &length); + if (!info->int_keypos) + DBUG_RETURN(-1); + if (info->int_keypos == info->buff+2) + DBUG_RETURN(_ma_search(info,keyinfo,key, USE_WHOLE_KEY, + nextflag | SEARCH_SAVE_BUFF, pos)); + if ((error= _ma_search(info,keyinfo,key, USE_WHOLE_KEY, + nextflag | SEARCH_SAVE_BUFF, + _ma_kpos(nod_flag,info->int_keypos))) <= 0) + DBUG_RETURN(error); + + /* QQ: We should be able to optimize away the following call */ + if (! _ma_get_last_key(info,keyinfo,info->buff,lastkey, + info->int_keypos,&info->lastkey_length)) + DBUG_RETURN(-1); + } + memcpy(info->lastkey,lastkey,info->lastkey_length); + info->lastpos= _ma_dpos(info,0,info->lastkey+info->lastkey_length); + DBUG_PRINT("exit",("found key at %lu",(ulong) info->lastpos)); + DBUG_RETURN(0); +} /* _ma_search_next */ + + + /* Search after position for the first row in an index */ + /* This is stored in info->lastpos */ + +int _ma_search_first(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, + register my_off_t pos) +{ + uint nod_flag; + uchar *page; + DBUG_ENTER("_ma_search_first"); + + if (pos == HA_OFFSET_ERROR) + { + my_errno=HA_ERR_KEY_NOT_FOUND; + info->lastpos= HA_OFFSET_ERROR; + DBUG_RETURN(-1); + } + + do + { + if (!_ma_fetch_keypage(info,keyinfo,pos,DFLT_INIT_HITS,info->buff,0)) + { + info->lastpos= HA_OFFSET_ERROR; + DBUG_RETURN(-1); + } + nod_flag=_ma_test_if_nod(info->buff); + page=info->buff+2+nod_flag; + } while ((pos= _ma_kpos(nod_flag,page)) != HA_OFFSET_ERROR); + + if (!(info->lastkey_length=(*keyinfo->get_key)(keyinfo,nod_flag,&page, + info->lastkey))) + DBUG_RETURN(-1); /* Crashed */ + + info->int_keypos=page; info->int_maxpos=info->buff+maria_getint(info->buff)-1; + info->int_nod_flag=nod_flag; + info->int_keytree_version=keyinfo->version; + info->last_search_keypage=info->last_keypage; + info->page_changed=info->buff_used=0; + info->lastpos= _ma_dpos(info,0,info->lastkey+info->lastkey_length); + + DBUG_PRINT("exit",("found key at %lu", (ulong) info->lastpos)); + DBUG_RETURN(0); +} /* _ma_search_first */ + + + /* Search after position for the last row in an index */ + /* This is stored in info->lastpos */ + +int _ma_search_last(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, + register my_off_t pos) +{ + uint nod_flag; + uchar *buff,*page; + DBUG_ENTER("_ma_search_last"); + + if (pos == HA_OFFSET_ERROR) + { + my_errno=HA_ERR_KEY_NOT_FOUND; /* Didn't find key */ + info->lastpos= HA_OFFSET_ERROR; + DBUG_RETURN(-1); + } + + buff=info->buff; + do + { + if (!_ma_fetch_keypage(info,keyinfo,pos,DFLT_INIT_HITS,buff,0)) + { + info->lastpos= HA_OFFSET_ERROR; + DBUG_RETURN(-1); + } + page= buff+maria_getint(buff); + nod_flag=_ma_test_if_nod(buff); + } while ((pos= _ma_kpos(nod_flag,page)) != HA_OFFSET_ERROR); + + if (!_ma_get_last_key(info,keyinfo,buff,info->lastkey,page, + &info->lastkey_length)) + DBUG_RETURN(-1); + info->lastpos= _ma_dpos(info,0,info->lastkey+info->lastkey_length); + info->int_keypos=info->int_maxpos=page; + info->int_nod_flag=nod_flag; + info->int_keytree_version=keyinfo->version; + info->last_search_keypage=info->last_keypage; + info->page_changed=info->buff_used=0; + + DBUG_PRINT("exit",("found key at %lu",(ulong) info->lastpos)); + DBUG_RETURN(0); +} /* _ma_search_last */ + + + +/**************************************************************************** +** +** Functions to store and pack a key in a page +** +** maria_calc_xx_key_length takes the following arguments: +** nod_flag If nod: Length of nod-pointer +** next_key Position to pos after the new key in buffer +** org_key Key that was before the next key in buffer +** prev_key Last key before current key +** key Key that will be stored +** s_temp Information how next key will be packed +****************************************************************************/ + +/* Static length key */ + +int +_ma_calc_static_key_length(MARIA_KEYDEF *keyinfo,uint nod_flag, + uchar *next_pos __attribute__((unused)), + uchar *org_key __attribute__((unused)), + uchar *prev_key __attribute__((unused)), + uchar *key, MARIA_KEY_PARAM *s_temp) +{ + s_temp->key=key; + return (int) (s_temp->totlength=keyinfo->keylength+nod_flag); +} + +/* Variable length key */ + +int +_ma_calc_var_key_length(MARIA_KEYDEF *keyinfo,uint nod_flag, + uchar *next_pos __attribute__((unused)), + uchar *org_key __attribute__((unused)), + uchar *prev_key __attribute__((unused)), + uchar *key, MARIA_KEY_PARAM *s_temp) +{ + s_temp->key=key; + return (int) (s_temp->totlength= _ma_keylength(keyinfo,key)+nod_flag); +} + +/* + length of key with a variable length first segment which is prefix + compressed (mariachk reports 'packed + stripped') + + Keys are compressed the following way: + + If the max length of first key segment <= 127 bytes the prefix is + 1 byte else it's 2 byte + + prefix byte(s) The high bit is set if this is a prefix for the prev key + length Packed length if the previous was a prefix byte + [length] data bytes ('length' bytes) + next-key-seg Next key segments + + If the first segment can have NULL: + The length is 0 for NULLS and 1+length for not null columns. + +*/ + +int +_ma_calc_var_pack_key_length(MARIA_KEYDEF *keyinfo, uint nod_flag, + uchar *next_key, + uchar *org_key, uchar *prev_key, uchar *key, + MARIA_KEY_PARAM *s_temp) +{ + reg1 HA_KEYSEG *keyseg; + int length; + uint key_length,ref_length,org_key_length=0, + length_pack,new_key_length,diff_flag,pack_marker; + uchar *start,*end,*key_end,*sort_order; + bool same_length; + + length_pack=s_temp->ref_length=s_temp->n_ref_length=s_temp->n_length=0; + same_length=0; keyseg=keyinfo->seg; + key_length= _ma_keylength(keyinfo,key)+nod_flag; + + sort_order=0; + if ((keyinfo->flag & HA_FULLTEXT) && + ((keyseg->type == HA_KEYTYPE_TEXT) || + (keyseg->type == HA_KEYTYPE_VARTEXT1) || + (keyseg->type == HA_KEYTYPE_VARTEXT2)) && + !use_strnxfrm(keyseg->charset)) + sort_order=keyseg->charset->sort_order; + + /* diff flag contains how many bytes is needed to pack key */ + if (keyseg->length >= 127) + { + diff_flag=2; + pack_marker=32768; + } + else + { + diff_flag= 1; + pack_marker=128; + } + s_temp->pack_marker=pack_marker; + + /* Handle the case that the first part have NULL values */ + if (keyseg->flag & HA_NULL_PART) + { + if (!*key++) + { + s_temp->key=key; + s_temp->ref_length=s_temp->key_length=0; + s_temp->totlength=key_length-1+diff_flag; + s_temp->next_key_pos=0; /* No next key */ + return (s_temp->totlength); + } + s_temp->store_not_null=1; + key_length--; /* We don't store NULL */ + if (prev_key && !*prev_key++) + org_key=prev_key=0; /* Can't pack against prev */ + else if (org_key) + org_key++; /* Skip NULL */ + } + else + s_temp->store_not_null=0; + s_temp->prev_key=org_key; + + /* The key part will start with a packed length */ + + get_key_pack_length(new_key_length,length_pack,key); + end=key_end= key+ new_key_length; + start=key; + + /* Calc how many characters are identical between this and the prev. key */ + if (prev_key) + { + get_key_length(org_key_length,prev_key); + s_temp->prev_key=prev_key; /* Pointer at data */ + /* Don't use key-pack if length == 0 */ + if (new_key_length && new_key_length == org_key_length) + same_length=1; + else if (new_key_length > org_key_length) + end=key + org_key_length; + + if (sort_order) /* SerG */ + { + while (key < end && sort_order[*key] == sort_order[*prev_key]) + { + key++; prev_key++; + } + } + else + { + while (key < end && *key == *prev_key) + { + key++; prev_key++; + } + } + } + + s_temp->key=key; + s_temp->key_length= (uint) (key_end-key); + + if (same_length && key == key_end) + { + /* identical variable length key */ + s_temp->ref_length= pack_marker; + length=(int) key_length-(int) (key_end-start)-length_pack; + length+= diff_flag; + if (next_key) + { /* Can't combine with next */ + s_temp->n_length= *next_key; /* Needed by _ma_store_key */ + next_key=0; + } + } + else + { + if (start != key) + { /* Starts as prev key */ + ref_length= (uint) (key-start); + s_temp->ref_length= ref_length + pack_marker; + length= (int) (key_length - ref_length); + + length-= length_pack; + length+= diff_flag; + length+= ((new_key_length-ref_length) >= 255) ? 3 : 1;/* Rest_of_key */ + } + else + { + s_temp->key_length+=s_temp->store_not_null; /* If null */ + length= key_length - length_pack+ diff_flag; + } + } + s_temp->totlength=(uint) length; + s_temp->prev_length=0; + DBUG_PRINT("test",("tot_length: %u length: %d uniq_key_length: %u", + key_length, length, s_temp->key_length)); + + /* If something after that hasn't length=0, test if we can combine */ + if ((s_temp->next_key_pos=next_key)) + { + uint packed,n_length; + + packed = *next_key & 128; + if (diff_flag == 2) + { + n_length= mi_uint2korr(next_key) & 32767; /* Length of next key */ + next_key+=2; + } + else + n_length= *next_key++ & 127; + if (!packed) + n_length-= s_temp->store_not_null; + + if (n_length || packed) /* Don't pack 0 length keys */ + { + uint next_length_pack, new_ref_length=s_temp->ref_length; + + if (packed) + { + /* If first key and next key is packed (only on delete) */ + if (!prev_key && org_key) + { + get_key_length(org_key_length,org_key); + key=start; + if (sort_order) /* SerG */ + { + while (key < end && sort_order[*key] == sort_order[*org_key]) + { + key++; org_key++; + } + } + else + { + while (key < end && *key == *org_key) + { + key++; org_key++; + } + } + if ((new_ref_length= (uint) (key - start))) + new_ref_length+=pack_marker; + } + + if (!n_length) + { + /* + We put a different key between two identical variable length keys + Extend next key to have same prefix as this key + */ + if (new_ref_length) /* prefix of previus key */ + { /* make next key longer */ + s_temp->part_of_prev_key= new_ref_length; + s_temp->prev_length= org_key_length - + (new_ref_length-pack_marker); + s_temp->n_ref_length= s_temp->part_of_prev_key; + s_temp->n_length= s_temp->prev_length; + n_length= get_pack_length(s_temp->prev_length); + s_temp->prev_key+= (new_ref_length - pack_marker); + length+= s_temp->prev_length + n_length; + } + else + { /* Can't use prev key */ + s_temp->part_of_prev_key=0; + s_temp->prev_length= org_key_length; + s_temp->n_ref_length=s_temp->n_length= org_key_length; + length+= org_key_length; + /* +get_pack_length(org_key_length); */ + } + return (int) length; + } + + ref_length=n_length; + get_key_pack_length(n_length,next_length_pack,next_key); + + /* Test if new keys has fewer characters that match the previous key */ + if (!new_ref_length) + { /* Can't use prev key */ + s_temp->part_of_prev_key= 0; + s_temp->prev_length= ref_length; + s_temp->n_ref_length= s_temp->n_length= n_length+ref_length; + /* s_temp->prev_key+= get_pack_length(org_key_length); */ + return (int) length+ref_length-next_length_pack; + } + if (ref_length+pack_marker > new_ref_length) + { + uint new_pack_length=new_ref_length-pack_marker; + /* We must copy characters from the original key to the next key */ + s_temp->part_of_prev_key= new_ref_length; + s_temp->prev_length= ref_length - new_pack_length; + s_temp->n_ref_length=s_temp->n_length=n_length + s_temp->prev_length; + s_temp->prev_key+= new_pack_length; +/* +get_pack_length(org_key_length); */ + length= length-get_pack_length(ref_length)+ + get_pack_length(new_pack_length); + return (int) length + s_temp->prev_length; + } + } + else + { + /* Next key wasn't a prefix of previous key */ + ref_length=0; + next_length_pack=0; + } + DBUG_PRINT("test",("length: %d next_key: %lx", length, + (long) next_key)); + + { + uint tmp_length; + key=(start+=ref_length); + if (key+n_length < key_end) /* Normalize length based */ + key_end=key+n_length; + if (sort_order) /* SerG */ + { + while (key < key_end && sort_order[*key] == + sort_order[*next_key]) + { + key++; next_key++; + } + } + else + { + while (key < key_end && *key == *next_key) + { + key++; next_key++; + } + } + if (!(tmp_length=(uint) (key-start))) + { /* Key can't be re-packed */ + s_temp->next_key_pos=0; + return length; + } + ref_length+=tmp_length; + n_length-=tmp_length; + length-=tmp_length+next_length_pack; /* We gained these chars */ + } + if (n_length == 0 && ref_length == new_key_length) + { + s_temp->n_ref_length=pack_marker; /* Same as prev key */ + } + else + { + s_temp->n_ref_length=ref_length | pack_marker; + length+= get_pack_length(n_length); + s_temp->n_length=n_length; + } + } + } + return length; +} + + +/* Length of key which is prefix compressed */ + +int _ma_calc_bin_pack_key_length(MARIA_KEYDEF *keyinfo, uint nod_flag, + uchar *next_key, + uchar *org_key, uchar *prev_key, uchar *key, + MARIA_KEY_PARAM *s_temp) +{ + uint length,key_length,ref_length; + + s_temp->totlength=key_length= _ma_keylength(keyinfo,key)+nod_flag; +#ifdef HAVE_purify + s_temp->n_length= s_temp->n_ref_length=0; /* For valgrind */ +#endif + s_temp->key=key; + s_temp->prev_key=org_key; + if (prev_key) /* If not first key in block */ + { + /* pack key against previous key */ + /* + As keys may be identical when running a sort in mariachk, we + have to guard against the case where keys may be identical + */ + uchar *end; + end=key+key_length; + for ( ; *key == *prev_key && key < end; key++,prev_key++) ; + s_temp->ref_length= ref_length=(uint) (key-s_temp->key); + length=key_length - ref_length + get_pack_length(ref_length); + } + else + { + /* No previous key */ + s_temp->ref_length=ref_length=0; + length=key_length+1; + } + if ((s_temp->next_key_pos=next_key)) /* If another key after */ + { + /* pack key against next key */ + uint next_length,next_length_pack; + get_key_pack_length(next_length,next_length_pack,next_key); + + /* If first key and next key is packed (only on delete) */ + if (!prev_key && org_key && next_length) + { + uchar *end; + for (key= s_temp->key, end=key+next_length ; + *key == *org_key && key < end; + key++,org_key++) ; + ref_length= (uint) (key - s_temp->key); + } + + if (next_length > ref_length) + { + /* We put a key with different case between two keys with the same prefix + Extend next key to have same prefix as + this key */ + s_temp->n_ref_length= ref_length; + s_temp->prev_length= next_length-ref_length; + s_temp->prev_key+= ref_length; + return (int) (length+ s_temp->prev_length - next_length_pack + + get_pack_length(ref_length)); + } + /* Check how many characters are identical to next key */ + key= s_temp->key+next_length; + while (*key++ == *next_key++) ; + if ((ref_length= (uint) (key - s_temp->key)-1) == next_length) + { + s_temp->next_key_pos=0; + return length; /* can't pack next key */ + } + s_temp->prev_length=0; + s_temp->n_ref_length=ref_length; + return (int) (length-(ref_length - next_length) - next_length_pack + + get_pack_length(ref_length)); + } + return (int) length; +} + + +/* +** store a key packed with _ma_calc_xxx_key_length in page-buffert +*/ + +/* store key without compression */ + +void _ma_store_static_key(MARIA_KEYDEF *keyinfo __attribute__((unused)), + register uchar *key_pos, + register MARIA_KEY_PARAM *s_temp) +{ + memcpy((byte*) key_pos,(byte*) s_temp->key,(size_t) s_temp->totlength); +} + + +/* store variable length key with prefix compression */ + +#define store_pack_length(test,pos,length) { \ + if (test) { *((pos)++) = (uchar) (length); } else \ + { *((pos)++) = (uchar) ((length) >> 8); *((pos)++) = (uchar) (length); } } + + +void _ma_store_var_pack_key(MARIA_KEYDEF *keyinfo __attribute__((unused)), + register uchar *key_pos, + register MARIA_KEY_PARAM *s_temp) +{ + uint length; + uchar *start; + + start=key_pos; + + if (s_temp->ref_length) + { + /* Packed against previous key */ + store_pack_length(s_temp->pack_marker == 128,key_pos,s_temp->ref_length); + /* If not same key after */ + if (s_temp->ref_length != s_temp->pack_marker) + store_key_length_inc(key_pos,s_temp->key_length); + } + else + { + /* Not packed against previous key */ + store_pack_length(s_temp->pack_marker == 128,key_pos,s_temp->key_length); + } + bmove((byte*) key_pos,(byte*) s_temp->key, + (length=s_temp->totlength-(uint) (key_pos-start))); + + if (!s_temp->next_key_pos) /* No following key */ + return; + key_pos+=length; + + if (s_temp->prev_length) + { + /* Extend next key because new key didn't have same prefix as prev key */ + if (s_temp->part_of_prev_key) + { + store_pack_length(s_temp->pack_marker == 128,key_pos, + s_temp->part_of_prev_key); + store_key_length_inc(key_pos,s_temp->n_length); + } + else + { + s_temp->n_length+= s_temp->store_not_null; + store_pack_length(s_temp->pack_marker == 128,key_pos, + s_temp->n_length); + } + memcpy(key_pos, s_temp->prev_key, s_temp->prev_length); + } + else if (s_temp->n_ref_length) + { + store_pack_length(s_temp->pack_marker == 128,key_pos,s_temp->n_ref_length); + if (s_temp->n_ref_length == s_temp->pack_marker) + return; /* Identical key */ + store_key_length(key_pos,s_temp->n_length); + } + else + { + s_temp->n_length+= s_temp->store_not_null; + store_pack_length(s_temp->pack_marker == 128,key_pos,s_temp->n_length); + } +} + + +/* variable length key with prefix compression */ + +void _ma_store_bin_pack_key(MARIA_KEYDEF *keyinfo __attribute__((unused)), + register uchar *key_pos, + register MARIA_KEY_PARAM *s_temp) +{ + store_key_length_inc(key_pos,s_temp->ref_length); + memcpy((char*) key_pos,(char*) s_temp->key+s_temp->ref_length, + (size_t) s_temp->totlength-s_temp->ref_length); + + if (s_temp->next_key_pos) + { + key_pos+=(uint) (s_temp->totlength-s_temp->ref_length); + store_key_length_inc(key_pos,s_temp->n_ref_length); + if (s_temp->prev_length) /* If we must extend key */ + { + memcpy(key_pos,s_temp->prev_key,s_temp->prev_length); + } + } +} diff --git a/storage/maria/ma_sort.c b/storage/maria/ma_sort.c new file mode 100644 index 00000000000..df547792ffa --- /dev/null +++ b/storage/maria/ma_sort.c @@ -0,0 +1,1021 @@ +/* Copyright (C) 2006 MySQL AB & MySQL Finland AB & TCX DataKonsult AB + + 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; either version 2 of the License, or + (at your option) any later version. + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ + +/* + Creates a index for a database by reading keys, sorting them and outputing + them in sorted order through MARIA_SORT_INFO functions. +*/ + +#include "ma_fulltext.h" +#include + +/* static variables */ + +#undef MIN_SORT_MEMORY +#undef MYF_RW +#undef DISK_BUFFER_SIZE + +#define MERGEBUFF 15 +#define MERGEBUFF2 31 +#define MIN_SORT_MEMORY (4096-MALLOC_OVERHEAD) +#define MYF_RW MYF(MY_NABP | MY_WME | MY_WAIT_IF_FULL) +#define DISK_BUFFER_SIZE (IO_SIZE*16) + + +/* + Pointers of functions for store and read keys from temp file +*/ + +extern void print_error _VARARGS((const char *fmt,...)); + +/* Functions defined in this file */ + +static ha_rows NEAR_F find_all_keys(MARIA_SORT_PARAM *info,uint keys, + uchar **sort_keys, + DYNAMIC_ARRAY *buffpek,int *maxbuffer, + IO_CACHE *tempfile, + IO_CACHE *tempfile_for_exceptions); +static int NEAR_F write_keys(MARIA_SORT_PARAM *info,uchar **sort_keys, + uint count, BUFFPEK *buffpek,IO_CACHE *tempfile); +static int NEAR_F write_key(MARIA_SORT_PARAM *info, uchar *key, + IO_CACHE *tempfile); +static int NEAR_F write_index(MARIA_SORT_PARAM *info,uchar * *sort_keys, + uint count); +static int NEAR_F merge_many_buff(MARIA_SORT_PARAM *info,uint keys, + uchar * *sort_keys, + BUFFPEK *buffpek,int *maxbuffer, + IO_CACHE *t_file); +static uint NEAR_F read_to_buffer(IO_CACHE *fromfile,BUFFPEK *buffpek, + uint sort_length); +static int NEAR_F merge_buffers(MARIA_SORT_PARAM *info,uint keys, + IO_CACHE *from_file, IO_CACHE *to_file, + uchar * *sort_keys, BUFFPEK *lastbuff, + BUFFPEK *Fb, BUFFPEK *Tb); +static int NEAR_F merge_index(MARIA_SORT_PARAM *,uint,uchar **,BUFFPEK *, int, + IO_CACHE *); +static int flush_maria_ft_buf(MARIA_SORT_PARAM *info); + +static int NEAR_F write_keys_varlen(MARIA_SORT_PARAM *info,uchar **sort_keys, + uint count, BUFFPEK *buffpek, + IO_CACHE *tempfile); +static uint NEAR_F read_to_buffer_varlen(IO_CACHE *fromfile,BUFFPEK *buffpek, + uint sort_length); +static int NEAR_F write_merge_key(MARIA_SORT_PARAM *info, IO_CACHE *to_file, + char *key, uint sort_length, uint count); +static int NEAR_F write_merge_key_varlen(MARIA_SORT_PARAM *info, + IO_CACHE *to_file, + char* key, uint sort_length, + uint count); +static inline int +my_var_write(MARIA_SORT_PARAM *info, IO_CACHE *to_file, byte *bufs); + +/* + Creates a index of sorted keys + + SYNOPSIS + _ma_create_index_by_sort() + info Sort parameters + no_messages Set to 1 if no output + sortbuff_size Size if sortbuffer to allocate + + RESULT + 0 ok + <> 0 Error +*/ + +int _ma_create_index_by_sort(MARIA_SORT_PARAM *info,my_bool no_messages, + ulong sortbuff_size) +{ + int error,maxbuffer,skr; + uint memavl,old_memavl,keys,sort_length; + DYNAMIC_ARRAY buffpek; + ha_rows records; + uchar **sort_keys; + IO_CACHE tempfile, tempfile_for_exceptions; + DBUG_ENTER("_ma_create_index_by_sort"); + DBUG_PRINT("enter",("sort_length: %d", info->key_length)); + + if (info->keyinfo->flag & HA_VAR_LENGTH_KEY) + { + info->write_keys=write_keys_varlen; + info->read_to_buffer=read_to_buffer_varlen; + info->write_key=write_merge_key_varlen; + } + else + { + info->write_keys=write_keys; + info->read_to_buffer=read_to_buffer; + info->write_key=write_merge_key; + } + + my_b_clear(&tempfile); + my_b_clear(&tempfile_for_exceptions); + bzero((char*) &buffpek,sizeof(buffpek)); + sort_keys= (uchar **) NULL; error= 1; + maxbuffer=1; + + memavl=max(sortbuff_size,MIN_SORT_MEMORY); + records= info->sort_info->max_records; + sort_length= info->key_length; + LINT_INIT(keys); + + while (memavl >= MIN_SORT_MEMORY) + { + if ((my_off_t) (records+1)*(sort_length+sizeof(char*)) <= + (my_off_t) memavl) + keys= records+1; + else + do + { + skr=maxbuffer; + if (memavl < sizeof(BUFFPEK)*(uint) maxbuffer || + (keys=(memavl-sizeof(BUFFPEK)*(uint) maxbuffer)/ + (sort_length+sizeof(char*))) <= 1) + { + _ma_check_print_error(info->sort_info->param, + "sort_buffer_size is to small"); + goto err; + } + } + while ((maxbuffer= (int) (records/(keys-1)+1)) != skr); + + if ((sort_keys=(uchar **)my_malloc(keys*(sort_length+sizeof(char*))+ + HA_FT_MAXBYTELEN, MYF(0)))) + { + if (my_init_dynamic_array(&buffpek, sizeof(BUFFPEK), maxbuffer, + maxbuffer/2)) + { + my_free((gptr) sort_keys,MYF(0)); + sort_keys= 0; + } + else + break; + } + old_memavl=memavl; + if ((memavl=memavl/4*3) < MIN_SORT_MEMORY && old_memavl > MIN_SORT_MEMORY) + memavl=MIN_SORT_MEMORY; + } + if (memavl < MIN_SORT_MEMORY) + { + _ma_check_print_error(info->sort_info->param,"Sort buffer to small"); /* purecov: tested */ + goto err; /* purecov: tested */ + } + (*info->lock_in_memory)(info->sort_info->param);/* Everything is allocated */ + + if (!no_messages) + printf(" - Searching for keys, allocating buffer for %d keys\n",keys); + + if ((records=find_all_keys(info,keys,sort_keys,&buffpek,&maxbuffer, + &tempfile,&tempfile_for_exceptions)) + == HA_POS_ERROR) + goto err; /* purecov: tested */ + if (maxbuffer == 0) + { + if (!no_messages) + printf(" - Dumping %lu keys\n", (ulong) records); + if (write_index(info,sort_keys, (uint) records)) + goto err; /* purecov: inspected */ + } + else + { + keys=(keys*(sort_length+sizeof(char*)))/sort_length; + if (maxbuffer >= MERGEBUFF2) + { + if (!no_messages) + printf(" - Merging %lu keys\n", (ulong) records); /* purecov: tested */ + if (merge_many_buff(info,keys,sort_keys, + dynamic_element(&buffpek,0,BUFFPEK *),&maxbuffer,&tempfile)) + goto err; /* purecov: inspected */ + } + if (flush_io_cache(&tempfile) || + reinit_io_cache(&tempfile,READ_CACHE,0L,0,0)) + goto err; /* purecov: inspected */ + if (!no_messages) + printf(" - Last merge and dumping keys\n"); /* purecov: tested */ + if (merge_index(info,keys,sort_keys,dynamic_element(&buffpek,0,BUFFPEK *), + maxbuffer,&tempfile)) + goto err; /* purecov: inspected */ + } + + if (flush_maria_ft_buf(info) || _ma_flush_pending_blocks(info)) + goto err; + + if (my_b_inited(&tempfile_for_exceptions)) + { + MARIA_HA *index=info->sort_info->info; + uint keyno=info->key; + uint key_length, ref_length=index->s->rec_reflength; + + if (!no_messages) + printf(" - Adding exceptions\n"); /* purecov: tested */ + if (flush_io_cache(&tempfile_for_exceptions) || + reinit_io_cache(&tempfile_for_exceptions,READ_CACHE,0L,0,0)) + goto err; + + while (!my_b_read(&tempfile_for_exceptions,(byte*)&key_length, + sizeof(key_length)) + && !my_b_read(&tempfile_for_exceptions,(byte*)sort_keys, + (uint) key_length)) + { + if (_ma_ck_write(index,keyno,(uchar*) sort_keys,key_length-ref_length)) + goto err; + } + } + + error =0; + +err: + if (sort_keys) + my_free((gptr) sort_keys,MYF(0)); + delete_dynamic(&buffpek); + close_cached_file(&tempfile); + close_cached_file(&tempfile_for_exceptions); + + DBUG_RETURN(error ? -1 : 0); +} /* _ma_create_index_by_sort */ + + +/* Search after all keys and place them in a temp. file */ + +static ha_rows NEAR_F find_all_keys(MARIA_SORT_PARAM *info, uint keys, + uchar **sort_keys, DYNAMIC_ARRAY *buffpek, + int *maxbuffer, IO_CACHE *tempfile, + IO_CACHE *tempfile_for_exceptions) +{ + int error; + uint idx; + DBUG_ENTER("find_all_keys"); + + idx=error=0; + sort_keys[0]=(uchar*) (sort_keys+keys); + + while (!(error=(*info->key_read)(info,sort_keys[idx]))) + { + if (info->real_key_length > info->key_length) + { + if (write_key(info,sort_keys[idx],tempfile_for_exceptions)) + DBUG_RETURN(HA_POS_ERROR); /* purecov: inspected */ + continue; + } + + if (++idx == keys) + { + if (info->write_keys(info,sort_keys,idx-1,(BUFFPEK *)alloc_dynamic(buffpek), + tempfile)) + DBUG_RETURN(HA_POS_ERROR); /* purecov: inspected */ + + sort_keys[0]=(uchar*) (sort_keys+keys); + memcpy(sort_keys[0],sort_keys[idx-1],(size_t) info->key_length); + idx=1; + } + sort_keys[idx]=sort_keys[idx-1]+info->key_length; + } + if (error > 0) + DBUG_RETURN(HA_POS_ERROR); /* Aborted by get_key */ /* purecov: inspected */ + if (buffpek->elements) + { + if (info->write_keys(info,sort_keys,idx,(BUFFPEK *)alloc_dynamic(buffpek), + tempfile)) + DBUG_RETURN(HA_POS_ERROR); /* purecov: inspected */ + *maxbuffer=buffpek->elements-1; + } + else + *maxbuffer=0; + + DBUG_RETURN((*maxbuffer)*(keys-1)+idx); +} /* find_all_keys */ + + +#ifdef THREAD +/* Search after all keys and place them in a temp. file */ + +pthread_handler_t _ma_thr_find_all_keys(void *arg) +{ + MARIA_SORT_PARAM *info= (MARIA_SORT_PARAM*) arg; + int error; + uint memavl,old_memavl,keys,sort_length; + uint idx, maxbuffer; + uchar **sort_keys=0; + + LINT_INIT(keys); + + error=1; + + if (my_thread_init()) + goto err; + if (info->sort_info->got_error) + goto err; + + if (info->keyinfo->flag && HA_VAR_LENGTH_KEY) + { + info->write_keys=write_keys_varlen; + info->read_to_buffer=read_to_buffer_varlen; + info->write_key=write_merge_key_varlen; + } + else + { + info->write_keys=write_keys; + info->read_to_buffer=read_to_buffer; + info->write_key=write_merge_key; + } + + my_b_clear(&info->tempfile); + my_b_clear(&info->tempfile_for_exceptions); + bzero((char*) &info->buffpek,sizeof(info->buffpek)); + bzero((char*) &info->unique, sizeof(info->unique)); + sort_keys= (uchar **) NULL; + + memavl=max(info->sortbuff_size, MIN_SORT_MEMORY); + idx= info->sort_info->max_records; + sort_length= info->key_length; + maxbuffer= 1; + + while (memavl >= MIN_SORT_MEMORY) + { + if ((my_off_t) (idx+1)*(sort_length+sizeof(char*)) <= + (my_off_t) memavl) + keys= idx+1; + else + { + uint skr; + do + { + skr=maxbuffer; + if (memavl < sizeof(BUFFPEK)*maxbuffer || + (keys=(memavl-sizeof(BUFFPEK)*maxbuffer)/ + (sort_length+sizeof(char*))) <= 1) + { + _ma_check_print_error(info->sort_info->param, + "sort_buffer_size is to small"); + goto err; + } + } + while ((maxbuffer= (int) (idx/(keys-1)+1)) != skr); + } + if ((sort_keys=(uchar **)my_malloc(keys*(sort_length+sizeof(char*))+ + ((info->keyinfo->flag & HA_FULLTEXT) ? + HA_FT_MAXBYTELEN : 0), MYF(0)))) + { + if (my_init_dynamic_array(&info->buffpek, sizeof(BUFFPEK), + maxbuffer, maxbuffer/2)) + { + my_free((gptr) sort_keys,MYF(0)); + sort_keys= (uchar **) NULL; /* for err: label */ + } + else + break; + } + old_memavl=memavl; + if ((memavl=memavl/4*3) < MIN_SORT_MEMORY && old_memavl > MIN_SORT_MEMORY) + memavl=MIN_SORT_MEMORY; + } + if (memavl < MIN_SORT_MEMORY) + { + _ma_check_print_error(info->sort_info->param,"Sort buffer to small"); /* purecov: tested */ + goto err; /* purecov: tested */ + } + + if (info->sort_info->param->testflag & T_VERBOSE) + printf("Key %d - Allocating buffer for %d keys\n",info->key+1,keys); + info->sort_keys=sort_keys; + + idx=error=0; + sort_keys[0]=(uchar*) (sort_keys+keys); + + while (!(error=info->sort_info->got_error) && + !(error=(*info->key_read)(info,sort_keys[idx]))) + { + if (info->real_key_length > info->key_length) + { + if (write_key(info,sort_keys[idx], &info->tempfile_for_exceptions)) + goto err; + continue; + } + + if (++idx == keys) + { + if (info->write_keys(info,sort_keys,idx-1, + (BUFFPEK *)alloc_dynamic(&info->buffpek), + &info->tempfile)) + goto err; + sort_keys[0]=(uchar*) (sort_keys+keys); + memcpy(sort_keys[0],sort_keys[idx-1],(size_t) info->key_length); + idx=1; + } + sort_keys[idx]=sort_keys[idx-1]+info->key_length; + } + if (error > 0) + goto err; + if (info->buffpek.elements) + { + if (info->write_keys(info,sort_keys, idx, + (BUFFPEK *) alloc_dynamic(&info->buffpek), &info->tempfile)) + goto err; + info->keys=(info->buffpek.elements-1)*(keys-1)+idx; + } + else + info->keys=idx; + + info->sort_keys_length=keys; + goto ok; + +err: + info->sort_info->got_error=1; /* no need to protect this with a mutex */ + if (sort_keys) + my_free((gptr) sort_keys,MYF(0)); + info->sort_keys=0; + delete_dynamic(& info->buffpek); + close_cached_file(&info->tempfile); + close_cached_file(&info->tempfile_for_exceptions); + +ok: + remove_io_thread(&info->read_cache); + pthread_mutex_lock(&info->sort_info->mutex); + info->sort_info->threads_running--; + pthread_cond_signal(&info->sort_info->cond); + pthread_mutex_unlock(&info->sort_info->mutex); + my_thread_end(); + return NULL; +} + + +int _ma_thr_write_keys(MARIA_SORT_PARAM *sort_param) +{ + MARIA_SORT_INFO *sort_info=sort_param->sort_info; + HA_CHECK *param=sort_info->param; + ulong length, keys; + ulong *rec_per_key_part=param->rec_per_key_part; + int got_error=sort_info->got_error; + uint i; + MARIA_HA *info=sort_info->info; + MARIA_SHARE *share=info->s; + MARIA_SORT_PARAM *sinfo; + byte *mergebuf=0; + LINT_INIT(length); + + for (i= 0, sinfo= sort_param ; + i < sort_info->total_keys ; + i++, rec_per_key_part+=sinfo->keyinfo->keysegs, sinfo++) + { + if (!sinfo->sort_keys) + { + got_error=1; + continue; + } + if (!got_error) + { + maria_set_key_active(share->state.key_map, sinfo->key); + if (param->testflag & T_STATISTICS) + maria_update_key_parts(sinfo->keyinfo, rec_per_key_part, sinfo->unique, + param->stats_method == MI_STATS_METHOD_IGNORE_NULLS? + sinfo->notnull: NULL, + (ulonglong) info->state->records); + + + if (!sinfo->buffpek.elements) + { + if (param->testflag & T_VERBOSE) + { + printf("Key %d - Dumping %u keys\n",sinfo->key+1, sinfo->keys); + fflush(stdout); + } + if (write_index(sinfo, sinfo->sort_keys, sinfo->keys) || + flush_maria_ft_buf(sinfo) || _ma_flush_pending_blocks(sinfo)) + got_error=1; + } + } + my_free((gptr) sinfo->sort_keys,MYF(0)); + my_free(_ma_get_rec_buff_ptr(info, sinfo->rec_buff), + MYF(MY_ALLOW_ZERO_PTR)); + sinfo->sort_keys=0; + } + + for (i= 0, sinfo= sort_param ; + i < sort_info->total_keys ; + i++, + delete_dynamic(&sinfo->buffpek), + close_cached_file(&sinfo->tempfile), + close_cached_file(&sinfo->tempfile_for_exceptions), + sinfo++) + { + if (got_error) + continue; + if (sinfo->keyinfo->flag && HA_VAR_LENGTH_KEY) + { + sinfo->write_keys=write_keys_varlen; + sinfo->read_to_buffer=read_to_buffer_varlen; + sinfo->write_key=write_merge_key_varlen; + } + else + { + sinfo->write_keys=write_keys; + sinfo->read_to_buffer=read_to_buffer; + sinfo->write_key=write_merge_key; + } + if (sinfo->buffpek.elements) + { + uint maxbuffer=sinfo->buffpek.elements-1; + if (!mergebuf) + { + length=param->sort_buffer_length; + while (length >= MIN_SORT_MEMORY && !mergebuf) + { + mergebuf=my_malloc(length, MYF(0)); + length=length*3/4; + } + if (!mergebuf) + { + got_error=1; + continue; + } + } + keys=length/sinfo->key_length; + if (maxbuffer >= MERGEBUFF2) + { + if (param->testflag & T_VERBOSE) + printf("Key %d - Merging %u keys\n",sinfo->key+1, sinfo->keys); + if (merge_many_buff(sinfo, keys, (uchar **)mergebuf, + dynamic_element(&sinfo->buffpek, 0, BUFFPEK *), + (int*) &maxbuffer, &sinfo->tempfile)) + { + got_error=1; + continue; + } + } + if (flush_io_cache(&sinfo->tempfile) || + reinit_io_cache(&sinfo->tempfile,READ_CACHE,0L,0,0)) + { + got_error=1; + continue; + } + if (param->testflag & T_VERBOSE) + printf("Key %d - Last merge and dumping keys\n", sinfo->key+1); + if (merge_index(sinfo, keys, (uchar **)mergebuf, + dynamic_element(&sinfo->buffpek,0,BUFFPEK *), + maxbuffer,&sinfo->tempfile) || + flush_maria_ft_buf(sinfo) || + _ma_flush_pending_blocks(sinfo)) + { + got_error=1; + continue; + } + } + if (my_b_inited(&sinfo->tempfile_for_exceptions)) + { + uint key_length; + + if (param->testflag & T_VERBOSE) + printf("Key %d - Dumping 'long' keys\n", sinfo->key+1); + + if (flush_io_cache(&sinfo->tempfile_for_exceptions) || + reinit_io_cache(&sinfo->tempfile_for_exceptions,READ_CACHE,0L,0,0)) + { + got_error=1; + continue; + } + + while (!got_error && + !my_b_read(&sinfo->tempfile_for_exceptions,(byte*)&key_length, + sizeof(key_length))) + { + byte maria_ft_buf[HA_FT_MAXBYTELEN + HA_FT_WLEN + 10]; + if (key_length > sizeof(maria_ft_buf) || + my_b_read(&sinfo->tempfile_for_exceptions, (byte*)maria_ft_buf, + (uint)key_length) || + _ma_ck_write(info, sinfo->key, (uchar*)maria_ft_buf, + key_length - info->s->rec_reflength)) + got_error=1; + } + } + } + my_free((gptr) mergebuf,MYF(MY_ALLOW_ZERO_PTR)); + return got_error; +} +#endif /* THREAD */ + + /* Write all keys in memory to file for later merge */ + +static int NEAR_F write_keys(MARIA_SORT_PARAM *info, register uchar **sort_keys, + uint count, BUFFPEK *buffpek, IO_CACHE *tempfile) +{ + uchar **end; + uint sort_length=info->key_length; + DBUG_ENTER("write_keys"); + + qsort2((byte*) sort_keys,count,sizeof(byte*),(qsort2_cmp) info->key_cmp, + info); + if (!my_b_inited(tempfile) && + open_cached_file(tempfile, my_tmpdir(info->tmpdir), "ST", + DISK_BUFFER_SIZE, info->sort_info->param->myf_rw)) + DBUG_RETURN(1); /* purecov: inspected */ + + buffpek->file_pos=my_b_tell(tempfile); + buffpek->count=count; + + for (end=sort_keys+count ; sort_keys != end ; sort_keys++) + { + if (my_b_write(tempfile,(byte*) *sort_keys,(uint) sort_length)) + DBUG_RETURN(1); /* purecov: inspected */ + } + DBUG_RETURN(0); +} /* write_keys */ + + +static inline int +my_var_write(MARIA_SORT_PARAM *info, IO_CACHE *to_file, byte *bufs) +{ + int err; + uint16 len = _ma_keylength(info->keyinfo, (uchar*) bufs); + + /* The following is safe as this is a local file */ + if ((err= my_b_write(to_file, (byte*)&len, sizeof(len)))) + return (err); + if ((err= my_b_write(to_file,bufs, (uint) len))) + return (err); + return (0); +} + + +static int NEAR_F write_keys_varlen(MARIA_SORT_PARAM *info, + register uchar **sort_keys, + uint count, BUFFPEK *buffpek, + IO_CACHE *tempfile) +{ + uchar **end; + int err; + DBUG_ENTER("write_keys_varlen"); + + qsort2((byte*) sort_keys,count,sizeof(byte*),(qsort2_cmp) info->key_cmp, + info); + if (!my_b_inited(tempfile) && + open_cached_file(tempfile, my_tmpdir(info->tmpdir), "ST", + DISK_BUFFER_SIZE, info->sort_info->param->myf_rw)) + DBUG_RETURN(1); /* purecov: inspected */ + + buffpek->file_pos=my_b_tell(tempfile); + buffpek->count=count; + for (end=sort_keys+count ; sort_keys != end ; sort_keys++) + { + if ((err= my_var_write(info,tempfile, (byte*) *sort_keys))) + DBUG_RETURN(err); + } + DBUG_RETURN(0); +} /* write_keys_varlen */ + + +static int NEAR_F write_key(MARIA_SORT_PARAM *info, uchar *key, + IO_CACHE *tempfile) +{ + uint key_length=info->real_key_length; + DBUG_ENTER("write_key"); + + if (!my_b_inited(tempfile) && + open_cached_file(tempfile, my_tmpdir(info->tmpdir), "ST", + DISK_BUFFER_SIZE, info->sort_info->param->myf_rw)) + DBUG_RETURN(1); + + if (my_b_write(tempfile,(byte*)&key_length,sizeof(key_length)) || + my_b_write(tempfile,(byte*)key,(uint) key_length)) + DBUG_RETURN(1); + DBUG_RETURN(0); +} /* write_key */ + + +/* Write index */ + +static int NEAR_F write_index(MARIA_SORT_PARAM *info, register uchar **sort_keys, + register uint count) +{ + DBUG_ENTER("write_index"); + + qsort2((gptr) sort_keys,(size_t) count,sizeof(byte*), + (qsort2_cmp) info->key_cmp,info); + while (count--) + { + if ((*info->key_write)(info,*sort_keys++)) + DBUG_RETURN(-1); /* purecov: inspected */ + } + DBUG_RETURN(0); +} /* write_index */ + + + /* Merge buffers to make < MERGEBUFF2 buffers */ + +static int NEAR_F merge_many_buff(MARIA_SORT_PARAM *info, uint keys, + uchar **sort_keys, BUFFPEK *buffpek, + int *maxbuffer, IO_CACHE *t_file) +{ + register int i; + IO_CACHE t_file2, *from_file, *to_file, *temp; + BUFFPEK *lastbuff; + DBUG_ENTER("merge_many_buff"); + + if (*maxbuffer < MERGEBUFF2) + DBUG_RETURN(0); /* purecov: inspected */ + if (flush_io_cache(t_file) || + open_cached_file(&t_file2,my_tmpdir(info->tmpdir),"ST", + DISK_BUFFER_SIZE, info->sort_info->param->myf_rw)) + DBUG_RETURN(1); /* purecov: inspected */ + + from_file= t_file ; to_file= &t_file2; + while (*maxbuffer >= MERGEBUFF2) + { + reinit_io_cache(from_file,READ_CACHE,0L,0,0); + reinit_io_cache(to_file,WRITE_CACHE,0L,0,0); + lastbuff=buffpek; + for (i=0 ; i <= *maxbuffer-MERGEBUFF*3/2 ; i+=MERGEBUFF) + { + if (merge_buffers(info,keys,from_file,to_file,sort_keys,lastbuff++, + buffpek+i,buffpek+i+MERGEBUFF-1)) + break; /* purecov: inspected */ + } + if (merge_buffers(info,keys,from_file,to_file,sort_keys,lastbuff++, + buffpek+i,buffpek+ *maxbuffer)) + break; /* purecov: inspected */ + if (flush_io_cache(to_file)) + break; /* purecov: inspected */ + temp=from_file; from_file=to_file; to_file=temp; + *maxbuffer= (int) (lastbuff-buffpek)-1; + } + close_cached_file(to_file); /* This holds old result */ + if (to_file == t_file) + *t_file=t_file2; /* Copy result file */ + + DBUG_RETURN(*maxbuffer >= MERGEBUFF2); /* Return 1 if interrupted */ +} /* merge_many_buff */ + + +/* + Read data to buffer + + SYNOPSIS + read_to_buffer() + fromfile File to read from + buffpek Where to read from + sort_length max length to read + RESULT + > 0 Ammount of bytes read + -1 Error +*/ + +static uint NEAR_F read_to_buffer(IO_CACHE *fromfile, BUFFPEK *buffpek, + uint sort_length) +{ + register uint count; + uint length; + + if ((count=(uint) min((ha_rows) buffpek->max_keys,buffpek->count))) + { + if (my_pread(fromfile->file,(byte*) buffpek->base, + (length= sort_length*count),buffpek->file_pos,MYF_RW)) + return((uint) -1); /* purecov: inspected */ + buffpek->key=buffpek->base; + buffpek->file_pos+= length; /* New filepos */ + buffpek->count-= count; + buffpek->mem_count= count; + } + return (count*sort_length); +} /* read_to_buffer */ + +static uint NEAR_F read_to_buffer_varlen(IO_CACHE *fromfile, BUFFPEK *buffpek, + uint sort_length) +{ + register uint count; + uint16 length_of_key = 0; + uint idx; + uchar *buffp; + + if ((count=(uint) min((ha_rows) buffpek->max_keys,buffpek->count))) + { + buffp = buffpek->base; + + for (idx=1;idx<=count;idx++) + { + if (my_pread(fromfile->file,(byte*)&length_of_key,sizeof(length_of_key), + buffpek->file_pos,MYF_RW)) + return((uint) -1); + buffpek->file_pos+=sizeof(length_of_key); + if (my_pread(fromfile->file,(byte*) buffp,length_of_key, + buffpek->file_pos,MYF_RW)) + return((uint) -1); + buffpek->file_pos+=length_of_key; + buffp = buffp + sort_length; + } + buffpek->key=buffpek->base; + buffpek->count-= count; + buffpek->mem_count= count; + } + return (count*sort_length); +} /* read_to_buffer_varlen */ + + +static int NEAR_F write_merge_key_varlen(MARIA_SORT_PARAM *info, + IO_CACHE *to_file,char* key, + uint sort_length, uint count) +{ + uint idx; + + char *bufs = key; + for (idx=1;idx<=count;idx++) + { + int err; + if ((err= my_var_write(info,to_file, (byte*) bufs))) + return (err); + bufs=bufs+sort_length; + } + return(0); +} + + +static int NEAR_F write_merge_key(MARIA_SORT_PARAM *info __attribute__((unused)), + IO_CACHE *to_file, char* key, + uint sort_length, uint count) +{ + return my_b_write(to_file,(byte*) key,(uint) sort_length*count); +} + +/* + Merge buffers to one buffer + If to_file == 0 then use info->key_write +*/ + +static int NEAR_F +merge_buffers(MARIA_SORT_PARAM *info, uint keys, IO_CACHE *from_file, + IO_CACHE *to_file, uchar **sort_keys, BUFFPEK *lastbuff, + BUFFPEK *Fb, BUFFPEK *Tb) +{ + int error; + uint sort_length,maxcount; + ha_rows count; + my_off_t to_start_filepos; + uchar *strpos; + BUFFPEK *buffpek,**refpek; + QUEUE queue; + volatile int *killed= _ma_killed_ptr(info->sort_info->param); + + DBUG_ENTER("merge_buffers"); + + count=error=0; + maxcount=keys/((uint) (Tb-Fb) +1); + LINT_INIT(to_start_filepos); + if (to_file) + to_start_filepos=my_b_tell(to_file); + strpos=(uchar*) sort_keys; + sort_length=info->key_length; + + if (init_queue(&queue,(uint) (Tb-Fb)+1,offsetof(BUFFPEK,key),0, + (int (*)(void*, byte *,byte*)) info->key_cmp, + (void*) info)) + DBUG_RETURN(1); /* purecov: inspected */ + + for (buffpek= Fb ; buffpek <= Tb ; buffpek++) + { + count+= buffpek->count; + buffpek->base= strpos; + buffpek->max_keys=maxcount; + strpos+= (uint) (error=(int) info->read_to_buffer(from_file,buffpek, + sort_length)); + if (error == -1) + goto err; /* purecov: inspected */ + queue_insert(&queue,(char*) buffpek); + } + + while (queue.elements > 1) + { + for (;;) + { + if (*killed) + { + error=1; goto err; + } + buffpek=(BUFFPEK*) queue_top(&queue); + if (to_file) + { + if (info->write_key(info,to_file,(byte*) buffpek->key, + (uint) sort_length,1)) + { + error=1; goto err; /* purecov: inspected */ + } + } + else + { + if ((*info->key_write)(info,(void*) buffpek->key)) + { + error=1; goto err; /* purecov: inspected */ + } + } + buffpek->key+=sort_length; + if (! --buffpek->mem_count) + { + if (!(error=(int) info->read_to_buffer(from_file,buffpek,sort_length))) + { + uchar *base=buffpek->base; + uint max_keys=buffpek->max_keys; + + VOID(queue_remove(&queue,0)); + + /* Put room used by buffer to use in other buffer */ + for (refpek= (BUFFPEK**) &queue_top(&queue); + refpek <= (BUFFPEK**) &queue_end(&queue); + refpek++) + { + buffpek= *refpek; + if (buffpek->base+buffpek->max_keys*sort_length == base) + { + buffpek->max_keys+=max_keys; + break; + } + else if (base+max_keys*sort_length == buffpek->base) + { + buffpek->base=base; + buffpek->max_keys+=max_keys; + break; + } + } + break; /* One buffer have been removed */ + } + } + else if (error == -1) + goto err; /* purecov: inspected */ + queue_replaced(&queue); /* Top element has been replaced */ + } + } + buffpek=(BUFFPEK*) queue_top(&queue); + buffpek->base=(uchar *) sort_keys; + buffpek->max_keys=keys; + do + { + if (to_file) + { + if (info->write_key(info,to_file,(byte*) buffpek->key, + sort_length,buffpek->mem_count)) + { + error=1; goto err; /* purecov: inspected */ + } + } + else + { + register uchar *end; + strpos= buffpek->key; + for (end=strpos+buffpek->mem_count*sort_length; + strpos != end ; + strpos+=sort_length) + { + if ((*info->key_write)(info,(void*) strpos)) + { + error=1; goto err; /* purecov: inspected */ + } + } + } + } + while ((error=(int) info->read_to_buffer(from_file,buffpek,sort_length)) != -1 && + error != 0); + + lastbuff->count=count; + if (to_file) + lastbuff->file_pos=to_start_filepos; +err: + delete_queue(&queue); + DBUG_RETURN(error); +} /* merge_buffers */ + + + /* Do a merge to output-file (save only positions) */ + +static int NEAR_F +merge_index(MARIA_SORT_PARAM *info, uint keys, uchar **sort_keys, + BUFFPEK *buffpek, int maxbuffer, IO_CACHE *tempfile) +{ + DBUG_ENTER("merge_index"); + if (merge_buffers(info,keys,tempfile,(IO_CACHE*) 0,sort_keys,buffpek,buffpek, + buffpek+maxbuffer)) + DBUG_RETURN(1); /* purecov: inspected */ + DBUG_RETURN(0); +} /* merge_index */ + +static int +flush_maria_ft_buf(MARIA_SORT_PARAM *info) +{ + int err=0; + if (info->sort_info->ft_buf) + { + err=_ma_sort_ft_buf_flush(info); + my_free((gptr)info->sort_info->ft_buf, MYF(0)); + info->sort_info->ft_buf=0; + } + return err; +} + diff --git a/storage/maria/ma_sp_defs.h b/storage/maria/ma_sp_defs.h new file mode 100644 index 00000000000..a7e282f0ddc --- /dev/null +++ b/storage/maria/ma_sp_defs.h @@ -0,0 +1,48 @@ +/* Copyright (C) 2006 MySQL AB & Ramil Kalimullin & MySQL Finland AB + & TCX DataKonsult AB + + 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; either version 2 of the License, or + (at your option) any later version. + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ + +#ifndef _SP_DEFS_H +#define _SP_DEFS_H + +#define SPDIMS 2 +#define SPTYPE HA_KEYTYPE_DOUBLE +#define SPLEN 8 + +#ifdef HAVE_SPATIAL + +enum wkbType +{ + wkbPoint = 1, + wkbLineString = 2, + wkbPolygon = 3, + wkbMultiPoint = 4, + wkbMultiLineString = 5, + wkbMultiPolygon = 6, + wkbGeometryCollection = 7 +}; + +enum wkbByteOrder +{ + wkbXDR = 0, /* Big Endian */ + wkbNDR = 1 /* Little Endian */ +}; + +uint sp_make_key(register MARIA_HA *info, uint keynr, uchar *key, + const byte *record, my_off_t filepos); + +#endif /*HAVE_SPATIAL*/ +#endif /* _SP_DEFS_H */ diff --git a/storage/maria/ma_sp_key.c b/storage/maria/ma_sp_key.c new file mode 100644 index 00000000000..b9841fed1e7 --- /dev/null +++ b/storage/maria/ma_sp_key.c @@ -0,0 +1,300 @@ +/* Copyright (C) 2006 MySQL AB & Ramil Kalimullin + + 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; either version 2 of the License, or + (at your option) any later version. + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ + +#include "maria_def.h" + +#ifdef HAVE_SPATIAL + +#include "ma_sp_defs.h" + +static int sp_add_point_to_mbr(uchar *(*wkb), uchar *end, uint n_dims, + uchar byte_order, double *mbr); +static int sp_get_point_mbr(uchar *(*wkb), uchar *end, uint n_dims, + uchar byte_order, double *mbr); +static int sp_get_linestring_mbr(uchar *(*wkb), uchar *end, uint n_dims, + uchar byte_order, double *mbr); +static int sp_get_polygon_mbr(uchar *(*wkb), uchar *end, uint n_dims, + uchar byte_order, double *mbr); +static int sp_get_geometry_mbr(uchar *(*wkb), uchar *end, uint n_dims, + double *mbr, int top); +static int sp_mbr_from_wkb(uchar (*wkb), uint size, uint n_dims, double *mbr); + +static void get_double(double *d, const byte *pos) +{ + float8get(*d, pos); +} + +uint sp_make_key(register MARIA_HA *info, uint keynr, uchar *key, + const byte *record, my_off_t filepos) +{ + HA_KEYSEG *keyseg; + MARIA_KEYDEF *keyinfo = &info->s->keyinfo[keynr]; + uint len = 0; + byte *pos; + uint dlen; + uchar *dptr; + double mbr[SPDIMS * 2]; + uint i; + + keyseg = &keyinfo->seg[-1]; + pos = (byte*)record + keyseg->start; + + dlen = _ma_calc_blob_length(keyseg->bit_start, pos); + memcpy_fixed(&dptr, pos + keyseg->bit_start, sizeof(char*)); + if (!dptr) + { + my_errno= HA_ERR_NULL_IN_SPATIAL; + return 0; + } + sp_mbr_from_wkb(dptr + 4, dlen - 4, SPDIMS, mbr); /* SRID */ + + for (i = 0, keyseg = keyinfo->seg; keyseg->type; keyseg++, i++) + { + uint length = keyseg->length; + + pos = ((byte*)mbr) + keyseg->start; + if (keyseg->flag & HA_SWAP_KEY) + { +#ifdef HAVE_ISNAN + if (keyseg->type == HA_KEYTYPE_FLOAT) + { + float nr; + float4get(nr, pos); + if (isnan(nr)) + { + /* Replace NAN with zero */ + bzero(key, length); + key+= length; + continue; + } + } + else if (keyseg->type == HA_KEYTYPE_DOUBLE) + { + double nr; + get_double(&nr, pos); + if (isnan(nr)) + { + bzero(key, length); + key+= length; + continue; + } + } +#endif + pos += length; + while (length--) + { + *key++ = *--pos; + } + } + else + { + memcpy((byte*)key, pos, length); + key += keyseg->length; + } + len += keyseg->length; + } + _ma_dpointer(info, key, filepos); + return len; +} + +/* +Calculate minimal bounding rectangle (mbr) of the spatial object +stored in "well-known binary representation" (wkb) format. +*/ +static int sp_mbr_from_wkb(uchar *wkb, uint size, uint n_dims, double *mbr) +{ + uint i; + + for (i=0; i < n_dims; ++i) + { + mbr[i * 2] = DBL_MAX; + mbr[i * 2 + 1] = -DBL_MAX; + } + + return sp_get_geometry_mbr(&wkb, wkb + size, n_dims, mbr, 1); +} + +/* + Add one point stored in wkb to mbr +*/ + +static int sp_add_point_to_mbr(uchar *(*wkb), uchar *end, uint n_dims, + uchar byte_order __attribute__((unused)), + double *mbr) +{ + double ord; + double *mbr_end= mbr + n_dims * 2; + + while (mbr < mbr_end) + { + if ((*wkb) > end - 8) + return -1; + get_double(&ord, (const byte*) *wkb); + (*wkb)+= 8; + if (ord < *mbr) + float8store((char*) mbr, ord); + mbr++; + if (ord > *mbr) + float8store((char*) mbr, ord); + mbr++; + } + return 0; +} + + +static int sp_get_point_mbr(uchar *(*wkb), uchar *end, uint n_dims, + uchar byte_order, double *mbr) +{ + return sp_add_point_to_mbr(wkb, end, n_dims, byte_order, mbr); +} + + +static int sp_get_linestring_mbr(uchar *(*wkb), uchar *end, uint n_dims, + uchar byte_order, double *mbr) +{ + uint n_points; + + n_points = uint4korr(*wkb); + (*wkb) += 4; + for (; n_points > 0; --n_points) + { + /* Add next point to mbr */ + if (sp_add_point_to_mbr(wkb, end, n_dims, byte_order, mbr)) + return -1; + } + return 0; +} + + +static int sp_get_polygon_mbr(uchar *(*wkb), uchar *end, uint n_dims, + uchar byte_order, double *mbr) +{ + uint n_linear_rings; + uint n_points; + + n_linear_rings = uint4korr((*wkb)); + (*wkb) += 4; + + for (; n_linear_rings > 0; --n_linear_rings) + { + n_points = uint4korr((*wkb)); + (*wkb) += 4; + for (; n_points > 0; --n_points) + { + /* Add next point to mbr */ + if (sp_add_point_to_mbr(wkb, end, n_dims, byte_order, mbr)) + return -1; + } + } + return 0; +} + +static int sp_get_geometry_mbr(uchar *(*wkb), uchar *end, uint n_dims, + double *mbr, int top) +{ + int res; + uchar byte_order; + uint wkb_type; + + byte_order = *(*wkb); + ++(*wkb); + + wkb_type = uint4korr((*wkb)); + (*wkb) += 4; + + switch ((enum wkbType) wkb_type) + { + case wkbPoint: + res = sp_get_point_mbr(wkb, end, n_dims, byte_order, mbr); + break; + case wkbLineString: + res = sp_get_linestring_mbr(wkb, end, n_dims, byte_order, mbr); + break; + case wkbPolygon: + res = sp_get_polygon_mbr(wkb, end, n_dims, byte_order, mbr); + break; + case wkbMultiPoint: + { + uint n_items; + n_items = uint4korr((*wkb)); + (*wkb) += 4; + for (; n_items > 0; --n_items) + { + byte_order = *(*wkb); + ++(*wkb); + (*wkb) += 4; + if (sp_get_point_mbr(wkb, end, n_dims, byte_order, mbr)) + return -1; + } + res = 0; + break; + } + case wkbMultiLineString: + { + uint n_items; + n_items = uint4korr((*wkb)); + (*wkb) += 4; + for (; n_items > 0; --n_items) + { + byte_order = *(*wkb); + ++(*wkb); + (*wkb) += 4; + if (sp_get_linestring_mbr(wkb, end, n_dims, byte_order, mbr)) + return -1; + } + res = 0; + break; + } + case wkbMultiPolygon: + { + uint n_items; + n_items = uint4korr((*wkb)); + (*wkb) += 4; + for (; n_items > 0; --n_items) + { + byte_order = *(*wkb); + ++(*wkb); + (*wkb) += 4; + if (sp_get_polygon_mbr(wkb, end, n_dims, byte_order, mbr)) + return -1; + } + res = 0; + break; + } + case wkbGeometryCollection: + { + uint n_items; + + if (!top) + return -1; + + n_items = uint4korr((*wkb)); + (*wkb) += 4; + for (; n_items > 0; --n_items) + { + if (sp_get_geometry_mbr(wkb, end, n_dims, mbr, 0)) + return -1; + } + res = 0; + break; + } + default: + res = -1; + } + return res; +} + +#endif /*HAVE_SPATIAL*/ diff --git a/storage/maria/ma_sp_test.c b/storage/maria/ma_sp_test.c new file mode 100644 index 00000000000..ea812974c8c --- /dev/null +++ b/storage/maria/ma_sp_test.c @@ -0,0 +1,568 @@ +/* Copyright (C) 2006 MySQL AB & MySQL Finland AB & TCX DataKonsult AB + + 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; either version 2 of the License, or + (at your option) any later version. + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ + +/* Testing of the basic functions of a MARIA spatial table */ +/* Written by Alex Barkov, who has a shared copyright to this code */ + +#include "maria.h" + +#ifdef HAVE_SPATIAL +#include "ma_sp_defs.h" + +#define MAX_REC_LENGTH 1024 +#define KEYALG HA_KEY_ALG_RTREE + +static void create_linestring(char *record,uint rownr); +static void print_record(char * record,my_off_t offs,const char * tail); + +static void create_key(char *key,uint rownr); +static void print_key(const char *key,const char * tail); + +static int run_test(const char *filename); +static int read_with_pos(MARIA_HA * file, int silent); + +static int maria_rtree_CreateLineStringWKB(double *ords, uint n_dims, uint n_points, + uchar *wkb); +static void maria_rtree_PrintWKB(uchar *wkb, uint n_dims); + +static char blob_key[MAX_REC_LENGTH]; + + +int main(int argc __attribute__((unused)),char *argv[]) +{ + MY_INIT(argv[0]); + maria_init(); + exit(run_test("sp_test")); +} + + +int run_test(const char *filename) +{ + MARIA_HA *file; + MARIA_UNIQUEDEF uniquedef; + MARIA_CREATE_INFO create_info; + MARIA_COLUMNDEF recinfo[20]; + MARIA_KEYDEF keyinfo[20]; + HA_KEYSEG keyseg[20]; + key_range min_range, max_range; + int silent=0; + int create_flag=0; + int null_fields=0; + int nrecords=30; + int uniques=0; + int i; + int error; + int row_count=0; + char record[MAX_REC_LENGTH]; + char key[MAX_REC_LENGTH]; + char read_record[MAX_REC_LENGTH]; + int upd=10; + ha_rows hrows; + + /* Define a column for NULLs and DEL markers*/ + + recinfo[0].type=FIELD_NORMAL; + recinfo[0].length=1; /* For NULL bits */ + + + /* Define spatial column */ + + recinfo[1].type=FIELD_BLOB; + recinfo[1].length=4 + maria_portable_sizeof_char_ptr; + + + + /* Define a key with 1 spatial segment */ + + keyinfo[0].seg=keyseg; + keyinfo[0].keysegs=1; + keyinfo[0].flag=HA_SPATIAL; + keyinfo[0].key_alg=KEYALG; + + keyinfo[0].seg[0].type= HA_KEYTYPE_BINARY; + keyinfo[0].seg[0].flag=0; + keyinfo[0].seg[0].start= 1; + keyinfo[0].seg[0].length=1; /* Spatial ignores it anyway */ + keyinfo[0].seg[0].null_bit= null_fields ? 2 : 0; + keyinfo[0].seg[0].null_pos=0; + keyinfo[0].seg[0].language=default_charset_info->number; + keyinfo[0].seg[0].bit_start=4; /* Long BLOB */ + + + if (!silent) + printf("- Creating isam-file\n"); + + bzero((char*) &create_info,sizeof(create_info)); + create_info.max_rows=10000000; + + if (maria_create(filename, + 1, /* keys */ + keyinfo, + 2, /* columns */ + recinfo,uniques,&uniquedef,&create_info,create_flag)) + goto err; + + if (!silent) + printf("- Open isam-file\n"); + + if (!(file=maria_open(filename,2,HA_OPEN_ABORT_IF_LOCKED))) + goto err; + + if (!silent) + printf("- Writing key:s\n"); + + for (i=0; i "); + print_record(record,maria_position(file),"\n"); + error=maria_update(file,read_record,record); + if (error) + { + printf("pos: %2d maria_update: %3d errno: %3d\n",i,error,my_errno); + goto err; + } + } + + if ((error=read_with_pos(file,silent))) + goto err; + + if (!silent) + printf("- Test maria_rkey then a sequence of maria_rnext_same\n"); + + create_key(key, nrecords*4/5); + print_key(key," search for INTERSECT\n"); + + if ((error=maria_rkey(file,read_record,0,key,0,HA_READ_MBR_INTERSECT))) + { + printf("maria_rkey: %3d errno: %3d\n",error,my_errno); + goto err; + } + print_record(read_record,maria_position(file)," maria_rkey\n"); + row_count=1; + + for (;;) + { + if ((error=maria_rnext_same(file,read_record))) + { + if (error==HA_ERR_END_OF_FILE) + break; + printf("maria_next: %3d errno: %3d\n",error,my_errno); + goto err; + } + print_record(read_record,maria_position(file)," maria_rnext_same\n"); + row_count++; + } + printf(" %d rows\n",row_count); + + if (!silent) + printf("- Test maria_rfirst then a sequence of maria_rnext\n"); + + error=maria_rfirst(file,read_record,0); + if (error) + { + printf("maria_rfirst: %3d errno: %3d\n",error,my_errno); + goto err; + } + row_count=1; + print_record(read_record,maria_position(file)," maria_frirst\n"); + + for(i=0;i "); + printf(" offs=%ld ",(long int)offs); + printf("%s",tail); +} + + +#ifdef NOT_USED +static void create_point(char *record,uint rownr) +{ + uint tmp; + char *ptr; + char *pos=record; + double x[200]; + int i; + + for(i=0;i= , <= , > , < +*/ + +uint NEAR maria_read_vec[]= +{ + SEARCH_FIND, SEARCH_FIND | SEARCH_BIGGER, SEARCH_FIND | SEARCH_SMALLER, + SEARCH_NO_FIND | SEARCH_BIGGER, SEARCH_NO_FIND | SEARCH_SMALLER, + SEARCH_FIND | SEARCH_PREFIX, SEARCH_LAST, SEARCH_LAST | SEARCH_SMALLER, + MBR_CONTAIN, MBR_INTERSECT, MBR_WITHIN, MBR_DISJOINT, MBR_EQUAL +}; + +uint NEAR maria_readnext_vec[]= +{ + SEARCH_BIGGER, SEARCH_BIGGER, SEARCH_SMALLER, SEARCH_BIGGER, SEARCH_SMALLER, + SEARCH_BIGGER, SEARCH_SMALLER, SEARCH_SMALLER +}; diff --git a/storage/maria/ma_statrec.c b/storage/maria/ma_statrec.c new file mode 100644 index 00000000000..c9614de1c72 --- /dev/null +++ b/storage/maria/ma_statrec.c @@ -0,0 +1,301 @@ +/* Copyright (C) 2006 MySQL AB & MySQL Finland AB & TCX DataKonsult AB + + 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; either version 2 of the License, or + (at your option) any later version. + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ + + /* Functions to handle fixed-length-records */ + +#include "maria_def.h" + + +int _ma_write_static_record(MARIA_HA *info, const byte *record) +{ + uchar temp[8]; /* max pointer length */ + if (info->s->state.dellink != HA_OFFSET_ERROR && + !info->append_insert_at_end) + { + my_off_t filepos=info->s->state.dellink; + info->rec_cache.seek_not_done=1; /* We have done a seek */ + if (info->s->file_read(info,(char*) &temp[0],info->s->base.rec_reflength, + info->s->state.dellink+1, + MYF(MY_NABP))) + goto err; + info->s->state.dellink= _ma_rec_pos(info->s,temp); + info->state->del--; + info->state->empty-=info->s->base.pack_reclength; + if (info->s->file_write(info, (char*) record, info->s->base.reclength, + filepos, + MYF(MY_NABP))) + goto err; + } + else + { + if (info->state->data_file_length > info->s->base.max_data_file_length- + info->s->base.pack_reclength) + { + my_errno=HA_ERR_RECORD_FILE_FULL; + return(2); + } + if (info->opt_flag & WRITE_CACHE_USED) + { /* Cash in use */ + if (my_b_write(&info->rec_cache, (byte*) record, + info->s->base.reclength)) + goto err; + if (info->s->base.pack_reclength != info->s->base.reclength) + { + uint length=info->s->base.pack_reclength - info->s->base.reclength; + bzero((char*) temp,length); + if (my_b_write(&info->rec_cache, (byte*) temp,length)) + goto err; + } + } + else + { + info->rec_cache.seek_not_done=1; /* We have done a seek */ + if (info->s->file_write(info,(char*) record,info->s->base.reclength, + info->state->data_file_length, + info->s->write_flag)) + goto err; + if (info->s->base.pack_reclength != info->s->base.reclength) + { + uint length=info->s->base.pack_reclength - info->s->base.reclength; + bzero((char*) temp,length); + if (info->s->file_write(info, (byte*) temp,length, + info->state->data_file_length+ + info->s->base.reclength, + info->s->write_flag)) + goto err; + } + } + info->state->data_file_length+=info->s->base.pack_reclength; + info->s->state.split++; + } + return 0; + err: + return 1; +} + +int _ma_update_static_record(MARIA_HA *info, my_off_t pos, const byte *record) +{ + info->rec_cache.seek_not_done=1; /* We have done a seek */ + return (info->s->file_write(info, + (char*) record,info->s->base.reclength, + pos, + MYF(MY_NABP)) != 0); +} + + +int _ma_delete_static_record(MARIA_HA *info) +{ + uchar temp[9]; /* 1+sizeof(uint32) */ + + info->state->del++; + info->state->empty+=info->s->base.pack_reclength; + temp[0]= '\0'; /* Mark that record is deleted */ + _ma_dpointer(info,temp+1,info->s->state.dellink); + info->s->state.dellink = info->lastpos; + info->rec_cache.seek_not_done=1; + return (info->s->file_write(info,(byte*) temp, 1+info->s->rec_reflength, + info->lastpos, MYF(MY_NABP)) != 0); +} + + +int _ma_cmp_static_record(register MARIA_HA *info, register const byte *old) +{ + DBUG_ENTER("_ma_cmp_static_record"); + + /* We are going to do changes; dont let anybody disturb */ + dont_break(); /* Dont allow SIGHUP or SIGINT */ + + if (info->opt_flag & WRITE_CACHE_USED) + { + if (flush_io_cache(&info->rec_cache)) + { + DBUG_RETURN(-1); + } + info->rec_cache.seek_not_done=1; /* We have done a seek */ + } + + if ((info->opt_flag & READ_CHECK_USED)) + { /* If check isn't disabled */ + info->rec_cache.seek_not_done=1; /* We have done a seek */ + if (info->s->file_read(info, (char*) info->rec_buff, info->s->base.reclength, + info->lastpos, + MYF(MY_NABP))) + DBUG_RETURN(-1); + if (memcmp((byte*) info->rec_buff, (byte*) old, + (uint) info->s->base.reclength)) + { + DBUG_DUMP("read",old,info->s->base.reclength); + DBUG_DUMP("disk",info->rec_buff,info->s->base.reclength); + my_errno=HA_ERR_RECORD_CHANGED; /* Record have changed */ + DBUG_RETURN(1); + } + } + DBUG_RETURN(0); +} + + +int _ma_cmp_static_unique(MARIA_HA *info, MARIA_UNIQUEDEF *def, + const byte *record, my_off_t pos) +{ + DBUG_ENTER("_ma_cmp_static_unique"); + + info->rec_cache.seek_not_done=1; /* We have done a seek */ + if (info->s->file_read(info, (char*) info->rec_buff, info->s->base.reclength, + pos, MYF(MY_NABP))) + DBUG_RETURN(-1); + DBUG_RETURN(_ma_unique_comp(def, record, info->rec_buff, + def->null_are_equal)); +} + + + /* Read a fixed-length-record */ + /* Returns 0 if Ok. */ + /* 1 if record is deleted */ + /* MY_FILE_ERROR on read-error or locking-error */ + +int _ma_read_static_record(register MARIA_HA *info, register my_off_t pos, + register byte *record) +{ + int error; + + if (pos != HA_OFFSET_ERROR) + { + if (info->opt_flag & WRITE_CACHE_USED && + info->rec_cache.pos_in_file <= pos && + flush_io_cache(&info->rec_cache)) + return(-1); + info->rec_cache.seek_not_done=1; /* We have done a seek */ + + error=info->s->file_read(info,(char*) record,info->s->base.reclength, + pos,MYF(MY_NABP)) != 0; + fast_ma_writeinfo(info); + if (! error) + { + if (!*record) + { + my_errno=HA_ERR_RECORD_DELETED; + return(1); /* Record is deleted */ + } + info->update|= HA_STATE_AKTIV; /* Record is read */ + return(0); + } + return(-1); /* Error on read */ + } + fast_ma_writeinfo(info); /* No such record */ + return(-1); +} + + + +int _ma_read_rnd_static_record(MARIA_HA *info, byte *buf, + register my_off_t filepos, + my_bool skip_deleted_blocks) +{ + int locked,error,cache_read; + uint cache_length; + MARIA_SHARE *share=info->s; + DBUG_ENTER("_ma_read_rnd_static_record"); + + cache_read=0; + cache_length=0; + if (info->opt_flag & WRITE_CACHE_USED && + (info->rec_cache.pos_in_file <= filepos || skip_deleted_blocks) && + flush_io_cache(&info->rec_cache)) + DBUG_RETURN(my_errno); + if (info->opt_flag & READ_CACHE_USED) + { /* Cache in use */ + if (filepos == my_b_tell(&info->rec_cache) && + (skip_deleted_blocks || !filepos)) + { + cache_read=1; /* Read record using cache */ + cache_length=(uint) (info->rec_cache.read_end - info->rec_cache.read_pos); + } + else + info->rec_cache.seek_not_done=1; /* Filepos is changed */ + } + locked=0; + if (info->lock_type == F_UNLCK) + { + if (filepos >= info->state->data_file_length) + { /* Test if new records */ + if (_ma_readinfo(info,F_RDLCK,0)) + DBUG_RETURN(my_errno); + locked=1; + } + else + { /* We don't nead new info */ +#ifndef UNSAFE_LOCKING + if ((! cache_read || share->base.reclength > cache_length) && + share->tot_locks == 0) + { /* record not in cache */ + if (my_lock(share->kfile,F_RDLCK,0L,F_TO_EOF, + MYF(MY_SEEK_NOT_DONE) | info->lock_wait)) + DBUG_RETURN(my_errno); + locked=1; + } +#else + info->tmp_lock_type=F_RDLCK; +#endif + } + } + if (filepos >= info->state->data_file_length) + { + DBUG_PRINT("test",("filepos: %ld (%ld) records: %ld del: %ld", + filepos/share->base.reclength,filepos, + info->state->records, info->state->del)); + fast_ma_writeinfo(info); + DBUG_RETURN(my_errno=HA_ERR_END_OF_FILE); + } + info->lastpos= filepos; + info->nextpos= filepos+share->base.pack_reclength; + + if (! cache_read) /* No cacheing */ + { + if ((error= _ma_read_static_record(info,filepos,buf))) + { + if (error > 0) + error=my_errno=HA_ERR_RECORD_DELETED; + else + error=my_errno; + } + DBUG_RETURN(error); + } + + /* Read record with cacheing */ + error=my_b_read(&info->rec_cache,(byte*) buf,share->base.reclength); + if (info->s->base.pack_reclength != info->s->base.reclength && !error) + { + char tmp[8]; /* Skill fill bytes */ + error=my_b_read(&info->rec_cache,(byte*) tmp, + info->s->base.pack_reclength - info->s->base.reclength); + } + if (locked) + VOID(_ma_writeinfo(info,0)); /* Unlock keyfile */ + if (!error) + { + if (!buf[0]) + { /* Record is removed */ + DBUG_RETURN(my_errno=HA_ERR_RECORD_DELETED); + } + /* Found and may be updated */ + info->update|= HA_STATE_AKTIV | HA_STATE_KEY_CHANGED; + DBUG_RETURN(0); + } + /* my_errno should be set if rec_cache.error == -1 */ + if (info->rec_cache.error != -1 || my_errno == 0) + my_errno=HA_ERR_WRONG_IN_RECORD; + DBUG_RETURN(my_errno); /* Something wrong (EOF?) */ +} diff --git a/storage/maria/ma_test1.c b/storage/maria/ma_test1.c new file mode 100644 index 00000000000..621a6d90e65 --- /dev/null +++ b/storage/maria/ma_test1.c @@ -0,0 +1,681 @@ +/* Copyright (C) 2006 MySQL AB & MySQL Finland AB & TCX DataKonsult AB + + 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; either version 2 of the License, or + (at your option) any later version. + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ + +/* Testing of the basic functions of a MARIA table */ + +#include "maria.h" +#include +#include + +#define MAX_REC_LENGTH 1024 + +static void usage(); + +static int rec_pointer_size=0, flags[50]; +static int key_field=FIELD_SKIP_PRESPACE,extra_field=FIELD_SKIP_ENDSPACE; +static int key_type=HA_KEYTYPE_NUM; +static int create_flag=0; + +static uint insert_count, update_count, remove_count; +static uint pack_keys=0, pack_seg=0, key_length; +static uint unique_key=HA_NOSAME; +static my_bool key_cacheing, null_fields, silent, skip_update, opt_unique, + verbose; +static MARIA_COLUMNDEF recinfo[4]; +static MARIA_KEYDEF keyinfo[10]; +static HA_KEYSEG keyseg[10]; +static HA_KEYSEG uniqueseg[10]; + +static int run_test(const char *filename); +static void get_options(int argc, char *argv[]); +static void create_key(char *key,uint rownr); +static void create_record(char *record,uint rownr); +static void update_record(char *record); + +int main(int argc,char *argv[]) +{ + MY_INIT(argv[0]); + my_init(); + maria_init(); + if (key_cacheing) + init_key_cache(maria_key_cache,KEY_CACHE_BLOCK_SIZE,IO_SIZE*16,0,0); + get_options(argc,argv); + + exit(run_test("test1")); +} + + +static int run_test(const char *filename) +{ + MARIA_HA *file; + int i,j,error,deleted,rec_length,uniques=0; + ha_rows found,row_count; + my_off_t pos; + char record[MAX_REC_LENGTH],key[MAX_REC_LENGTH],read_record[MAX_REC_LENGTH]; + MARIA_UNIQUEDEF uniquedef; + MARIA_CREATE_INFO create_info; + + bzero((char*) recinfo,sizeof(recinfo)); + + /* First define 2 columns */ + recinfo[0].type=FIELD_NORMAL; recinfo[0].length=1; /* For NULL bits */ + recinfo[1].type=key_field; + recinfo[1].length= (key_field == FIELD_BLOB ? 4+maria_portable_sizeof_char_ptr : + key_length); + if (key_field == FIELD_VARCHAR) + recinfo[1].length+= HA_VARCHAR_PACKLENGTH(key_length);; + recinfo[2].type=extra_field; + recinfo[2].length= (extra_field == FIELD_BLOB ? 4 + maria_portable_sizeof_char_ptr : 24); + if (extra_field == FIELD_VARCHAR) + recinfo[2].length+= HA_VARCHAR_PACKLENGTH(recinfo[2].length); + if (opt_unique) + { + recinfo[3].type=FIELD_CHECK; + recinfo[3].length=MARIA_UNIQUE_HASH_LENGTH; + } + rec_length=recinfo[0].length+recinfo[1].length+recinfo[2].length+ + recinfo[3].length; + + if (key_type == HA_KEYTYPE_VARTEXT1 && + key_length > 255) + key_type= HA_KEYTYPE_VARTEXT2; + + /* Define a key over the first column */ + keyinfo[0].seg=keyseg; + keyinfo[0].keysegs=1; + keyinfo[0].key_alg=HA_KEY_ALG_BTREE; + keyinfo[0].seg[0].type= key_type; + keyinfo[0].seg[0].flag= pack_seg; + keyinfo[0].seg[0].start=1; + keyinfo[0].seg[0].length=key_length; + keyinfo[0].seg[0].null_bit= null_fields ? 2 : 0; + keyinfo[0].seg[0].null_pos=0; + keyinfo[0].seg[0].language= default_charset_info->number; + if (pack_seg & HA_BLOB_PART) + { + keyinfo[0].seg[0].bit_start=4; /* Length of blob length */ + } + keyinfo[0].flag = (uint8) (pack_keys | unique_key); + + bzero((byte*) flags,sizeof(flags)); + if (opt_unique) + { + uint start; + uniques=1; + bzero((char*) &uniquedef,sizeof(uniquedef)); + bzero((char*) uniqueseg,sizeof(uniqueseg)); + uniquedef.seg=uniqueseg; + uniquedef.keysegs=2; + + /* Make a unique over all columns (except first NULL fields) */ + for (i=0, start=1 ; i < 2 ; i++) + { + uniqueseg[i].start=start; + start+=recinfo[i+1].length; + uniqueseg[i].length=recinfo[i+1].length; + uniqueseg[i].language= default_charset_info->number; + } + uniqueseg[0].type= key_type; + uniqueseg[0].null_bit= null_fields ? 2 : 0; + uniqueseg[1].type= HA_KEYTYPE_TEXT; + if (extra_field == FIELD_BLOB) + { + uniqueseg[1].length=0; /* The whole blob */ + uniqueseg[1].bit_start=4; /* long blob */ + uniqueseg[1].flag|= HA_BLOB_PART; + } + else if (extra_field == FIELD_VARCHAR) + uniqueseg[1].flag|= HA_VAR_LENGTH_PART; + } + else + uniques=0; + + if (!silent) + printf("- Creating isam-file\n"); + bzero((char*) &create_info,sizeof(create_info)); + create_info.max_rows=(ulong) (rec_pointer_size ? + (1L << (rec_pointer_size*8))/40 : + 0); + if (maria_create(filename,1,keyinfo,3+opt_unique,recinfo, + uniques, &uniquedef, &create_info, + create_flag)) + goto err; + if (!(file=maria_open(filename,2,HA_OPEN_ABORT_IF_LOCKED))) + goto err; + if (!silent) + printf("- Writing key:s\n"); + + my_errno=0; + row_count=deleted=0; + for (i=49 ; i>=1 ; i-=2 ) + { + if (insert_count-- == 0) { VOID(maria_close(file)) ; exit(0) ; } + j=i%25 +1; + create_record(record,j); + error=maria_write(file,record); + if (!error) + row_count++; + flags[j]=1; + if (verbose || error) + printf("J= %2d maria_write: %d errno: %d\n", j,error,my_errno); + } + + /* Insert 2 rows with null values */ + if (null_fields) + { + create_record(record,0); + error=maria_write(file,record); + if (!error) + row_count++; + if (verbose || error) + printf("J= NULL maria_write: %d errno: %d\n", error,my_errno); + error=maria_write(file,record); + if (!error) + row_count++; + if (verbose || error) + printf("J= NULL maria_write: %d errno: %d\n", error,my_errno); + flags[0]=2; + } + + if (!skip_update) + { + if (opt_unique) + { + if (!silent) + printf("- Checking unique constraint\n"); + create_record(record,j); + if (!maria_write(file,record) || my_errno != HA_ERR_FOUND_DUPP_UNIQUE) + { + printf("unique check failed\n"); + } + } + if (!silent) + printf("- Updating rows\n"); + + /* Update first last row to force extend of file */ + if (maria_rsame(file,read_record,-1)) + { + printf("Can't find last row with maria_rsame\n"); + } + else + { + memcpy(record,read_record,rec_length); + update_record(record); + if (maria_update(file,read_record,record)) + { + printf("Can't update last row: %.*s\n", + keyinfo[0].seg[0].length,read_record+1); + } + } + + /* Read through all rows and update them */ + pos=(my_off_t) 0; + found=0; + while ((error=maria_rrnd(file,read_record,pos)) == 0) + { + if (update_count-- == 0) { VOID(maria_close(file)) ; exit(0) ; } + memcpy(record,read_record,rec_length); + update_record(record); + if (maria_update(file,read_record,record)) + { + printf("Can't update row: %.*s, error: %d\n", + keyinfo[0].seg[0].length,record+1,my_errno); + } + found++; + pos=HA_OFFSET_ERROR; + } + if (found != row_count) + printf("Found %ld of %ld rows\n", (ulong) found, (ulong) row_count); + } + + if (!silent) + printf("- Reopening file\n"); + if (maria_close(file)) goto err; + if (!(file=maria_open(filename,2,HA_OPEN_ABORT_IF_LOCKED))) goto err; + if (!skip_update) + { + if (!silent) + printf("- Removing keys\n"); + + for (i=0 ; i <= 10 ; i++) + { + /* testing */ + if (remove_count-- == 0) { VOID(maria_close(file)) ; exit(0) ; } + j=i*2; + if (!flags[j]) + continue; + create_key(key,j); + my_errno=0; + if ((error = maria_rkey(file,read_record,0,key,0,HA_READ_KEY_EXACT))) + { + if (verbose || (flags[j] >= 1 || + (error && my_errno != HA_ERR_KEY_NOT_FOUND))) + printf("key: '%.*s' maria_rkey: %3d errno: %3d\n", + (int) key_length,key+test(null_fields),error,my_errno); + } + else + { + error=maria_delete(file,read_record); + if (verbose || error) + printf("key: '%.*s' maria_delete: %3d errno: %3d\n", + (int) key_length, key+test(null_fields), error, my_errno); + if (! error) + { + deleted++; + flags[j]--; + } + } + } + } + if (!silent) + printf("- Reading rows with key\n"); + for (i=0 ; i <= 25 ; i++) + { + create_key(key,i); + my_errno=0; + error=maria_rkey(file,read_record,0,key,0,HA_READ_KEY_EXACT); + if (verbose || + (error == 0 && flags[i] == 0 && unique_key) || + (error && (flags[i] != 0 || my_errno != HA_ERR_KEY_NOT_FOUND))) + { + printf("key: '%.*s' maria_rkey: %3d errno: %3d record: %s\n", + (int) key_length,key+test(null_fields),error,my_errno,record+1); + } + } + + if (!silent) + printf("- Reading rows with position\n"); + for (i=1,found=0 ; i <= 30 ; i++) + { + my_errno=0; + if ((error=maria_rrnd(file,read_record,i == 1 ? 0L : HA_OFFSET_ERROR)) == -1) + { + if (found != row_count-deleted) + printf("Found only %ld of %ld rows\n", (ulong) found, + (ulong) (row_count - deleted)); + break; + } + if (!error) + found++; + if (verbose || (error != 0 && error != HA_ERR_RECORD_DELETED && + error != HA_ERR_END_OF_FILE)) + { + printf("pos: %2d maria_rrnd: %3d errno: %3d record: %s\n", + i-1,error,my_errno,read_record+1); + } + } + if (maria_close(file)) goto err; + maria_end(); + my_end(MY_CHECK_ERROR); + + return (0); +err: + printf("got error: %3d when using maria-database\n",my_errno); + return 1; /* skip warning */ +} + + +static void create_key_part(char *key,uint rownr) +{ + if (!unique_key) + rownr&=7; /* Some identical keys */ + if (keyinfo[0].seg[0].type == HA_KEYTYPE_NUM) + { + sprintf(key,"%*d",keyinfo[0].seg[0].length,rownr); + } + else if (keyinfo[0].seg[0].type == HA_KEYTYPE_VARTEXT1 || + keyinfo[0].seg[0].type == HA_KEYTYPE_VARTEXT2) + { /* Alpha record */ + /* Create a key that may be easily packed */ + bfill(key,keyinfo[0].seg[0].length,rownr < 10 ? 'A' : 'B'); + sprintf(key+keyinfo[0].seg[0].length-2,"%-2d",rownr); + if ((rownr & 7) == 0) + { + /* Change the key to force a unpack of the next key */ + bfill(key+3,keyinfo[0].seg[0].length-4,rownr < 10 ? 'a' : 'b'); + } + } + else + { /* Alpha record */ + if (keyinfo[0].seg[0].flag & HA_SPACE_PACK) + sprintf(key,"%-*d",keyinfo[0].seg[0].length,rownr); + else + { + /* Create a key that may be easily packed */ + bfill(key,keyinfo[0].seg[0].length,rownr < 10 ? 'A' : 'B'); + sprintf(key+keyinfo[0].seg[0].length-2,"%-2d",rownr); + if ((rownr & 7) == 0) + { + /* Change the key to force a unpack of the next key */ + key[1]= (rownr < 10 ? 'a' : 'b'); + } + } + } +} + + +static void create_key(char *key,uint rownr) +{ + if (keyinfo[0].seg[0].null_bit) + { + if (rownr == 0) + { + key[0]=1; /* null key */ + key[1]=0; /* Fore easy print of key */ + return; + } + *key++=0; + } + if (keyinfo[0].seg[0].flag & (HA_BLOB_PART | HA_VAR_LENGTH_PART)) + { + uint tmp; + create_key_part(key+2,rownr); + tmp=strlen(key+2); + int2store(key,tmp); + } + else + create_key_part(key,rownr); +} + + +static char blob_key[MAX_REC_LENGTH]; +static char blob_record[MAX_REC_LENGTH+20*20]; + + +static void create_record(char *record,uint rownr) +{ + char *pos; + bzero((char*) record,MAX_REC_LENGTH); + record[0]=1; /* delete marker */ + if (rownr == 0 && keyinfo[0].seg[0].null_bit) + record[0]|=keyinfo[0].seg[0].null_bit; /* Null key */ + + pos=record+1; + if (recinfo[1].type == FIELD_BLOB) + { + uint tmp; + char *ptr; + create_key_part(blob_key,rownr); + tmp=strlen(blob_key); + int4store(pos,tmp); + ptr=blob_key; + memcpy_fixed(pos+4,&ptr,sizeof(char*)); + pos+=recinfo[1].length; + } + else if (recinfo[1].type == FIELD_VARCHAR) + { + uint tmp, pack_length= HA_VARCHAR_PACKLENGTH(recinfo[1].length-1); + create_key_part(pos+pack_length,rownr); + tmp= strlen(pos+pack_length); + if (pack_length == 1) + *(uchar*) pos= (uchar) tmp; + else + int2store(pos,tmp); + pos+= recinfo[1].length; + } + else + { + create_key_part(pos,rownr); + pos+=recinfo[1].length; + } + if (recinfo[2].type == FIELD_BLOB) + { + uint tmp; + char *ptr;; + sprintf(blob_record,"... row: %d", rownr); + strappend(blob_record,max(MAX_REC_LENGTH-rownr,10),' '); + tmp=strlen(blob_record); + int4store(pos,tmp); + ptr=blob_record; + memcpy_fixed(pos+4,&ptr,sizeof(char*)); + } + else if (recinfo[2].type == FIELD_VARCHAR) + { + uint tmp, pack_length= HA_VARCHAR_PACKLENGTH(recinfo[1].length-1); + sprintf(pos+pack_length, "... row: %d", rownr); + tmp= strlen(pos+pack_length); + if (pack_length == 1) + *(uchar*) pos= (uchar) tmp; + else + int2store(pos,tmp); + } + else + { + sprintf(pos,"... row: %d", rownr); + strappend(pos,recinfo[2].length,' '); + } +} + +/* change row to test re-packing of rows and reallocation of keys */ + +static void update_record(char *record) +{ + char *pos=record+1; + if (recinfo[1].type == FIELD_BLOB) + { + char *column,*ptr; + int length; + length=uint4korr(pos); /* Long blob */ + memcpy_fixed(&column,pos+4,sizeof(char*)); + memcpy(blob_key,column,length); /* Move old key */ + ptr=blob_key; + memcpy_fixed(pos+4,&ptr,sizeof(char*)); /* Store pointer to new key */ + if (keyinfo[0].seg[0].type != HA_KEYTYPE_NUM) + default_charset_info->cset->casedn(default_charset_info, + blob_key, length, blob_key, length); + pos+=recinfo[1].length; + } + else if (recinfo[1].type == FIELD_VARCHAR) + { + uint pack_length= HA_VARCHAR_PACKLENGTH(recinfo[1].length-1); + uint length= pack_length == 1 ? (uint) *(uchar*) pos : uint2korr(pos); + default_charset_info->cset->casedn(default_charset_info, + pos + pack_length, length, + pos + pack_length, length); + pos+=recinfo[1].length; + } + else + { + if (keyinfo[0].seg[0].type != HA_KEYTYPE_NUM) + default_charset_info->cset->casedn(default_charset_info, + pos, keyinfo[0].seg[0].length, + pos, keyinfo[0].seg[0].length); + pos+=recinfo[1].length; + } + + if (recinfo[2].type == FIELD_BLOB) + { + char *column; + int length; + length=uint4korr(pos); + memcpy_fixed(&column,pos+4,sizeof(char*)); + memcpy(blob_record,column,length); + bfill(blob_record+length,20,'.'); /* Make it larger */ + length+=20; + int4store(pos,length); + column=blob_record; + memcpy_fixed(pos+4,&column,sizeof(char*)); + } + else if (recinfo[2].type == FIELD_VARCHAR) + { + /* Second field is longer than 10 characters */ + uint pack_length= HA_VARCHAR_PACKLENGTH(recinfo[1].length-1); + uint length= pack_length == 1 ? (uint) *(uchar*) pos : uint2korr(pos); + bfill(pos+pack_length+length,recinfo[2].length-length-pack_length,'.'); + length=recinfo[2].length-pack_length; + if (pack_length == 1) + *(uchar*) pos= (uchar) length; + else + int2store(pos,length); + } + else + { + bfill(pos+recinfo[2].length-10,10,'.'); + } +} + + +static struct my_option my_long_options[] = +{ + {"checksum", 'c', "Undocumented", + 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, +#ifndef DBUG_OFF + {"debug", '#', "Undocumented", + 0, 0, 0, GET_STR, REQUIRED_ARG, 0, 0, 0, 0, 0, 0}, +#endif + {"delete_rows", 'd', "Undocumented", (gptr*) &remove_count, + (gptr*) &remove_count, 0, GET_UINT, REQUIRED_ARG, 1000, 0, 0, 0, 0, 0}, + {"help", '?', "Display help and exit", + 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, + {"insert_rows", 'i', "Undocumented", (gptr*) &insert_count, + (gptr*) &insert_count, 0, GET_UINT, REQUIRED_ARG, 1000, 0, 0, 0, 0, 0}, + {"key_alpha", 'a', "Use a key of type HA_KEYTYPE_TEXT", + 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, + {"key_binary_pack", 'B', "Undocumented", + 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, + {"key_blob", 'b', "Undocumented", + 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, + {"key_cache", 'K', "Undocumented", (gptr*) &key_cacheing, + (gptr*) &key_cacheing, 0, GET_BOOL, NO_ARG, 0, 0, 0, 0, 0, 0}, + {"key_length", 'k', "Undocumented", (gptr*) &key_length, (gptr*) &key_length, + 0, GET_UINT, REQUIRED_ARG, 6, 0, 0, 0, 0, 0}, + {"key_multiple", 'm', "Undocumented", + 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, + {"key_prefix_pack", 'P', "Undocumented", + 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, + {"key_space_pack", 'p', "Undocumented", + 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, + {"key_varchar", 'w', "Test VARCHAR keys", + 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, + {"null_fields", 'N', "Define fields with NULL", + (gptr*) &null_fields, (gptr*) &null_fields, 0, GET_BOOL, NO_ARG, + 0, 0, 0, 0, 0, 0}, + {"row_fixed_size", 'S', "Undocumented", + 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, + {"row_pointer_size", 'R', "Undocumented", (gptr*) &rec_pointer_size, + (gptr*) &rec_pointer_size, 0, GET_INT, REQUIRED_ARG, 0, 0, 0, 0, 0, 0}, + {"silent", 's', "Undocumented", + (gptr*) &silent, (gptr*) &silent, 0, GET_BOOL, NO_ARG, 0, 0, 0, 0, 0, 0}, + {"skip_update", 'U', "Undocumented", (gptr*) &skip_update, + (gptr*) &skip_update, 0, GET_BOOL, NO_ARG, 0, 0, 0, 0, 0, 0}, + {"unique", 'C', "Undocumented", (gptr*) &opt_unique, (gptr*) &opt_unique, 0, + GET_BOOL, NO_ARG, 0, 0, 0, 0, 0, 0}, + {"update_rows", 'u', "Undocumented", (gptr*) &update_count, + (gptr*) &update_count, 0, GET_UINT, REQUIRED_ARG, 1000, 0, 0, 0, 0, 0}, + {"verbose", 'v', "Be more verbose", (gptr*) &verbose, (gptr*) &verbose, 0, + GET_BOOL, NO_ARG, 0, 0, 0, 0, 0, 0}, + {"version", 'V', "Print version number and exit", + 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, + { 0, 0, 0, 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0} +}; + + +static my_bool +get_one_option(int optid, const struct my_option *opt __attribute__((unused)), + char *argument) +{ + switch(optid) { + case 'a': + key_type= HA_KEYTYPE_TEXT; + break; + case 'c': + create_flag|= HA_CREATE_CHECKSUM; + break; + case 'R': /* Length of record pointer */ + if (rec_pointer_size > 3) + rec_pointer_size=0; + break; + case 'P': + pack_keys= HA_PACK_KEY; /* Use prefix compression */ + break; + case 'B': + pack_keys= HA_BINARY_PACK_KEY; /* Use binary compression */ + break; + case 'S': + if (key_field == FIELD_VARCHAR) + { + create_flag=0; /* Static sized varchar */ + } + else if (key_field != FIELD_BLOB) + { + key_field=FIELD_NORMAL; /* static-size record */ + extra_field=FIELD_NORMAL; + } + break; + case 'p': + pack_keys=HA_PACK_KEY; /* Use prefix + space packing */ + pack_seg=HA_SPACE_PACK; + key_type=HA_KEYTYPE_TEXT; + break; + case 'm': + unique_key=0; + break; + case 'b': + key_field=FIELD_BLOB; /* blob key */ + extra_field= FIELD_BLOB; + pack_seg|= HA_BLOB_PART; + key_type= HA_KEYTYPE_VARTEXT1; + break; + case 'k': + if (key_length < 4 || key_length > HA_MAX_KEY_LENGTH) + { + fprintf(stderr,"Wrong key length\n"); + exit(1); + } + break; + case 'w': + key_field=FIELD_VARCHAR; /* varchar keys */ + extra_field= FIELD_VARCHAR; + key_type= HA_KEYTYPE_VARTEXT1; + pack_seg|= HA_VAR_LENGTH_PART; + create_flag|= HA_PACK_RECORD; + break; + case 'K': /* Use key cacheing */ + key_cacheing=1; + break; + case 'V': + printf("test1 Ver 1.2 \n"); + exit(0); + case '#': + DBUG_PUSH (argument); + break; + case '?': + usage(); + exit(1); + } + return 0; +} + + +/* Read options */ + +static void get_options(int argc, char *argv[]) +{ + int ho_error; + + if ((ho_error=handle_options(&argc, &argv, my_long_options, get_one_option))) + exit(ho_error); + + return; +} /* get options */ + + +static void usage() +{ + printf("Usage: %s [options]\n\n", my_progname); + my_print_help(my_long_options); + my_print_variables(my_long_options); +} diff --git a/storage/maria/ma_test2.c b/storage/maria/ma_test2.c new file mode 100644 index 00000000000..06cfaf7cc5e --- /dev/null +++ b/storage/maria/ma_test2.c @@ -0,0 +1,1050 @@ +/* Copyright (C) 2006 MySQL AB & MySQL Finland AB & TCX DataKonsult AB + + 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; either version 2 of the License, or + (at your option) any later version. + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ + +/* Test av isam-databas: stor test */ + +#ifndef USE_MY_FUNC /* We want to be able to dbug this !! */ +#define USE_MY_FUNC +#endif +#ifdef DBUG_OFF +#undef DBUG_OFF +#endif +#ifndef SAFEMALLOC +#define SAFEMALLOC +#endif +#include "maria_def.h" +#include + +#define STANDARD_LENGTH 37 +#define MARIA_KEYS 6 +#define MAX_PARTS 4 +#if !defined(MSDOS) && !defined(labs) +#define labs(a) abs(a) +#endif + +static void get_options(int argc, char *argv[]); +static uint rnd(uint max_value); +static void fix_length(byte *record,uint length); +static void put_blob_in_record(char *blob_pos,char **blob_buffer); +static void copy_key(struct st_maria_info *info,uint inx, + uchar *record,uchar *key); + +static int verbose=0,testflag=0, + first_key=0,async_io=0,key_cacheing=0,write_cacheing=0,locking=0, + rec_pointer_size=0,pack_fields=1,use_log=0,silent=0, + opt_quick_mode=0; +static int pack_seg=HA_SPACE_PACK,pack_type=HA_PACK_KEY,remove_count=-1, + create_flag=0; +static ulong key_cache_size=IO_SIZE*16; +static uint key_cache_block_size= KEY_CACHE_BLOCK_SIZE; + +static uint keys=MARIA_KEYS,recant=1000; +static uint use_blob=0; +static uint16 key1[1001],key3[5000]; +static char record[300],record2[300],key[100],key2[100], + read_record[300],read_record2[300],read_record3[300]; +static HA_KEYSEG glob_keyseg[MARIA_KEYS][MAX_PARTS]; + + /* Test program */ + +int main(int argc, char *argv[]) +{ + uint i; + int j,n1,n2,n3,error,k; + uint write_count,update,dupp_keys,opt_delete,start,length,blob_pos, + reclength,ant,found_parts; + my_off_t lastpos; + ha_rows range_records,records; + MARIA_HA *file; + MARIA_KEYDEF keyinfo[10]; + MARIA_COLUMNDEF recinfo[10]; + MARIA_INFO info; + const char *filename; + char *blob_buffer; + MARIA_CREATE_INFO create_info; + MY_INIT(argv[0]); + + filename= "test2"; + get_options(argc,argv); + if (! async_io) + my_disable_async_io=1; + + maria_init(); + reclength=STANDARD_LENGTH+60+(use_blob ? 8 : 0); + blob_pos=STANDARD_LENGTH+60; + keyinfo[0].seg= &glob_keyseg[0][0]; + keyinfo[0].seg[0].start=0; + keyinfo[0].seg[0].length=6; + keyinfo[0].seg[0].type=HA_KEYTYPE_TEXT; + keyinfo[0].seg[0].language= default_charset_info->number; + keyinfo[0].seg[0].flag=(uint8) pack_seg; + keyinfo[0].seg[0].null_bit=0; + keyinfo[0].seg[0].null_pos=0; + keyinfo[0].key_alg=HA_KEY_ALG_BTREE; + keyinfo[0].keysegs=1; + keyinfo[0].flag = pack_type; + keyinfo[1].seg= &glob_keyseg[1][0]; + keyinfo[1].seg[0].start=7; + keyinfo[1].seg[0].length=6; + keyinfo[1].seg[0].type=HA_KEYTYPE_BINARY; + keyinfo[1].seg[0].flag=0; + keyinfo[1].seg[0].null_bit=0; + keyinfo[1].seg[0].null_pos=0; + keyinfo[1].seg[1].start=0; /* two part key */ + keyinfo[1].seg[1].length=6; + keyinfo[1].seg[1].type=HA_KEYTYPE_NUM; + keyinfo[1].seg[1].flag=HA_REVERSE_SORT; + keyinfo[1].seg[1].null_bit=0; + keyinfo[1].seg[1].null_pos=0; + keyinfo[1].key_alg=HA_KEY_ALG_BTREE; + keyinfo[1].keysegs=2; + keyinfo[1].flag =0; + keyinfo[2].seg= &glob_keyseg[2][0]; + keyinfo[2].seg[0].start=12; + keyinfo[2].seg[0].length=8; + keyinfo[2].seg[0].type=HA_KEYTYPE_BINARY; + keyinfo[2].seg[0].flag=HA_REVERSE_SORT; + keyinfo[2].seg[0].null_bit=0; + keyinfo[2].seg[0].null_pos=0; + keyinfo[2].key_alg=HA_KEY_ALG_BTREE; + keyinfo[2].keysegs=1; + keyinfo[2].flag =HA_NOSAME; + keyinfo[3].seg= &glob_keyseg[3][0]; + keyinfo[3].seg[0].start=0; + keyinfo[3].seg[0].length=reclength-(use_blob ? 8 : 0); + keyinfo[3].seg[0].type=HA_KEYTYPE_TEXT; + keyinfo[3].seg[0].language=default_charset_info->number; + keyinfo[3].seg[0].flag=(uint8) pack_seg; + keyinfo[3].seg[0].null_bit=0; + keyinfo[3].seg[0].null_pos=0; + keyinfo[3].key_alg=HA_KEY_ALG_BTREE; + keyinfo[3].keysegs=1; + keyinfo[3].flag = pack_type; + keyinfo[4].seg= &glob_keyseg[4][0]; + keyinfo[4].seg[0].start=0; + keyinfo[4].seg[0].length=5; + keyinfo[4].seg[0].type=HA_KEYTYPE_TEXT; + keyinfo[4].seg[0].language=default_charset_info->number; + keyinfo[4].seg[0].flag=0; + keyinfo[4].seg[0].null_bit=0; + keyinfo[4].seg[0].null_pos=0; + keyinfo[4].key_alg=HA_KEY_ALG_BTREE; + keyinfo[4].keysegs=1; + keyinfo[4].flag = pack_type; + keyinfo[5].seg= &glob_keyseg[5][0]; + keyinfo[5].seg[0].start=0; + keyinfo[5].seg[0].length=4; + keyinfo[5].seg[0].type=HA_KEYTYPE_TEXT; + keyinfo[5].seg[0].language=default_charset_info->number; + keyinfo[5].seg[0].flag=pack_seg; + keyinfo[5].seg[0].null_bit=0; + keyinfo[5].seg[0].null_pos=0; + keyinfo[5].key_alg=HA_KEY_ALG_BTREE; + keyinfo[5].keysegs=1; + keyinfo[5].flag = pack_type; + + recinfo[0].type=pack_fields ? FIELD_SKIP_PRESPACE : 0; + recinfo[0].length=7; + recinfo[0].null_bit=0; + recinfo[0].null_pos=0; + recinfo[1].type=pack_fields ? FIELD_SKIP_PRESPACE : 0; + recinfo[1].length=5; + recinfo[1].null_bit=0; + recinfo[1].null_pos=0; + recinfo[2].type=pack_fields ? FIELD_SKIP_PRESPACE : 0; + recinfo[2].length=9; + recinfo[2].null_bit=0; + recinfo[2].null_pos=0; + recinfo[3].type=FIELD_NORMAL; + recinfo[3].length=STANDARD_LENGTH-7-5-9-4; + recinfo[3].null_bit=0; + recinfo[3].null_pos=0; + recinfo[4].type=pack_fields ? FIELD_SKIP_ZERO : 0; + recinfo[4].length=4; + recinfo[4].null_bit=0; + recinfo[4].null_pos=0; + recinfo[5].type=pack_fields ? FIELD_SKIP_ENDSPACE : 0; + recinfo[5].length=60; + recinfo[5].null_bit=0; + recinfo[5].null_pos=0; + if (use_blob) + { + recinfo[6].type=FIELD_BLOB; + recinfo[6].length=4+maria_portable_sizeof_char_ptr; + recinfo[6].null_bit=0; + recinfo[6].null_pos=0; + } + + write_count=update=dupp_keys=opt_delete=0; + blob_buffer=0; + + for (i=1000 ; i>0 ; i--) key1[i]=0; + for (i=4999 ; i>0 ; i--) key3[i]=0; + + if (!silent) + printf("- Creating isam-file\n"); + /* DBUG_PUSH(""); */ + /* my_delete(filename,MYF(0)); */ /* Remove old locks under gdb */ + file= 0; + bzero((char*) &create_info,sizeof(create_info)); + create_info.max_rows=(ha_rows) (rec_pointer_size ? + (1L << (rec_pointer_size*8))/ + reclength : 0); + create_info.reloc_rows=(ha_rows) 100; + if (maria_create(filename,keys,&keyinfo[first_key], + use_blob ? 7 : 6, &recinfo[0], + 0,(MARIA_UNIQUEDEF*) 0, + &create_info,create_flag)) + goto err; + if (use_log) + maria_logging(1); + if (!(file=maria_open(filename,2,HA_OPEN_ABORT_IF_LOCKED))) + goto err; + if (!silent) + printf("- Writing key:s\n"); + if (key_cacheing) + init_key_cache(maria_key_cache,key_cache_block_size,key_cache_size,0,0); + if (locking) + maria_lock_database(file,F_WRLCK); + if (write_cacheing) + maria_extra(file,HA_EXTRA_WRITE_CACHE,0); + if (opt_quick_mode) + maria_extra(file,HA_EXTRA_QUICK,0); + + for (i=0 ; i < recant ; i++) + { + n1=rnd(1000); n2=rnd(100); n3=rnd(5000); + sprintf(record,"%6d:%4d:%8d:Pos: %4d ",n1,n2,n3,write_count); + int4store(record+STANDARD_LENGTH-4,(long) i); + fix_length(record,(uint) STANDARD_LENGTH+rnd(60)); + put_blob_in_record(record+blob_pos,&blob_buffer); + DBUG_PRINT("test",("record: %d",i)); + + if (maria_write(file,record)) + { + if (my_errno != HA_ERR_FOUND_DUPP_KEY || key3[n3] == 0) + { + printf("Error: %d in write at record: %d\n",my_errno,i); + goto err; + } + if (verbose) printf(" Double key: %d\n",n3); + } + else + { + if (key3[n3] == 1 && first_key <3 && first_key+keys >= 3) + { + printf("Error: Didn't get error when writing second key: '%8d'\n",n3); + goto err; + } + write_count++; key1[n1]++; key3[n3]=1; + } + + /* Check if we can find key without flushing database */ + if (i == recant/2) + { + for (j=rnd(1000)+1 ; j>0 && key1[j] == 0 ; j--) ; + if (!j) + for (j=999 ; j>0 && key1[j] == 0 ; j--) ; + sprintf(key,"%6d",j); + if (maria_rkey(file,read_record,0,key,0,HA_READ_KEY_EXACT)) + { + printf("Test in loop: Can't find key: \"%s\"\n",key); + goto err; + } + } + } + if (testflag==1) goto end; + + if (write_cacheing) + { + if (maria_extra(file,HA_EXTRA_NO_CACHE,0)) + { + puts("got error from maria_extra(HA_EXTRA_NO_CACHE)"); + goto end; + } + } + if (key_cacheing) + resize_key_cache(maria_key_cache,key_cache_block_size,key_cache_size*2,0,0); + + if (!silent) + printf("- Delete\n"); + for (i=0 ; i0 && key1[j] == 0 ; j--) ; + if (j != 0) + { + sprintf(key,"%6d",j); + if (maria_rkey(file,read_record,0,key,0,HA_READ_KEY_EXACT)) + { + printf("can't find key1: \"%s\"\n",key); + goto err; + } + if (opt_delete == (uint) remove_count) /* While testing */ + goto end; + if (maria_delete(file,read_record)) + { + printf("error: %d; can't delete record: \"%s\"\n", my_errno,read_record); + goto err; + } + opt_delete++; + key1[atoi(read_record+keyinfo[0].seg[0].start)]--; + key3[atoi(read_record+keyinfo[2].seg[0].start)]=0; + } + else + puts("Warning: Skipping delete test because no dupplicate keys"); + } + if (testflag==2) goto end; + + if (!silent) + printf("- Update\n"); + for (i=0 ; i0 && key1[j] == 0 ; j--) ; + if (j != 0) + { + sprintf(key,"%6d",j); + if (maria_rkey(file,read_record,0,key,0,HA_READ_KEY_EXACT)) + { + printf("can't find key1: \"%s\"\n",key); + goto err; + } + if (use_blob) + { + if (i & 1) + put_blob_in_record(record+blob_pos,&blob_buffer); + else + bmove(record+blob_pos,read_record+blob_pos,8); + } + if (maria_update(file,read_record,record2)) + { + if (my_errno != HA_ERR_FOUND_DUPP_KEY || key3[n3] == 0) + { + printf("error: %d; can't update:\nFrom: \"%s\"\nTo: \"%s\"\n", + my_errno,read_record,record2); + goto err; + } + if (verbose) + printf("Double key when tried to update:\nFrom: \"%s\"\nTo: \"%s\"\n",record,record2); + } + else + { + key1[atoi(read_record+keyinfo[0].seg[0].start)]--; + key3[atoi(read_record+keyinfo[2].seg[0].start)]=0; + key1[n1]++; key3[n3]=1; + update++; + } + } + } + if (testflag == 3) + goto end; + + for (i=999, dupp_keys=j=0 ; i>0 ; i--) + { + if (key1[i] > dupp_keys) + { + dupp_keys=key1[i]; j=i; + } + } + sprintf(key,"%6d",j); + start=keyinfo[0].seg[0].start; + length=keyinfo[0].seg[0].length; + if (dupp_keys) + { + if (!silent) + printf("- Same key: first - next -> last - prev -> first\n"); + DBUG_PRINT("progpos",("first - next -> last - prev -> first")); + if (verbose) printf(" Using key: \"%s\" Keys: %d\n",key,dupp_keys); + + if (maria_rkey(file,read_record,0,key,0,HA_READ_KEY_EXACT)) + goto err; + if (maria_rsame(file,read_record2,-1)) + goto err; + if (memcmp(read_record,read_record2,reclength) != 0) + { + printf("maria_rsame didn't find same record\n"); + goto end; + } + info.recpos=maria_position(file); + if (maria_rfirst(file,read_record2,0) || + maria_rsame_with_pos(file,read_record2,0,info.recpos) || + memcmp(read_record,read_record2,reclength) != 0) + { + printf("maria_rsame_with_pos didn't find same record\n"); + goto end; + } + { + int skr=maria_rnext(file,read_record2,0); + if ((skr && my_errno != HA_ERR_END_OF_FILE) || + maria_rprev(file,read_record2,-1) || + memcmp(read_record,read_record2,reclength) != 0) + { + printf("maria_rsame_with_pos lost position\n"); + goto end; + } + } + ant=1; + while (maria_rnext(file,read_record2,0) == 0 && + memcmp(read_record2+start,key,length) == 0) ant++; + if (ant != dupp_keys) + { + printf("next: Found: %d keys of %d\n",ant,dupp_keys); + goto end; + } + ant=0; + while (maria_rprev(file,read_record3,0) == 0 && + bcmp(read_record3+start,key,length) == 0) ant++; + if (ant != dupp_keys) + { + printf("prev: Found: %d records of %d\n",ant,dupp_keys); + goto end; + } + + /* Check of maria_rnext_same */ + if (maria_rkey(file,read_record,0,key,0,HA_READ_KEY_EXACT)) + goto err; + ant=1; + while (!maria_rnext_same(file,read_record3) && ant < dupp_keys+10) + ant++; + if (ant != dupp_keys || my_errno != HA_ERR_END_OF_FILE) + { + printf("maria_rnext_same: Found: %d records of %d\n",ant,dupp_keys); + goto end; + } + } + + if (!silent) + printf("- All keys: first - next -> last - prev -> first\n"); + DBUG_PRINT("progpos",("All keys: first - next -> last - prev -> first")); + ant=1; + if (maria_rfirst(file,read_record,0)) + { + printf("Can't find first record\n"); + goto end; + } + while ((error=maria_rnext(file,read_record3,0)) == 0 && ant < write_count+10) + ant++; + if (ant != write_count - opt_delete || error != HA_ERR_END_OF_FILE) + { + printf("next: I found: %d records of %d (error: %d)\n", + ant, write_count - opt_delete, error); + goto end; + } + if (maria_rlast(file,read_record2,0) || + bcmp(read_record2,read_record3,reclength)) + { + printf("Can't find last record\n"); + DBUG_DUMP("record2",(byte*) read_record2,reclength); + DBUG_DUMP("record3",(byte*) read_record3,reclength); + goto end; + } + ant=1; + while (maria_rprev(file,read_record3,0) == 0 && ant < write_count+10) + ant++; + if (ant != write_count - opt_delete) + { + printf("prev: I found: %d records of %d\n",ant,write_count); + goto end; + } + if (bcmp(read_record,read_record3,reclength)) + { + printf("Can't find first record\n"); + goto end; + } + + if (!silent) + printf("- Test if: Read first - next - prev - prev - next == first\n"); + DBUG_PRINT("progpos",("- Read first - next - prev - prev - next == first")); + if (maria_rfirst(file,read_record,0) || + maria_rnext(file,read_record3,0) || + maria_rprev(file,read_record3,0) || + maria_rprev(file,read_record3,0) == 0 || + maria_rnext(file,read_record3,0)) + goto err; + if (bcmp(read_record,read_record3,reclength) != 0) + printf("Can't find first record\n"); + + if (!silent) + printf("- Test if: Read last - prev - next - next - prev == last\n"); + DBUG_PRINT("progpos",("Read last - prev - next - next - prev == last")); + if (maria_rlast(file,read_record2,0) || + maria_rprev(file,read_record3,0) || + maria_rnext(file,read_record3,0) || + maria_rnext(file,read_record3,0) == 0 || + maria_rprev(file,read_record3,0)) + goto err; + if (bcmp(read_record2,read_record3,reclength)) + printf("Can't find last record\n"); + + if (!silent) + puts("- Test read key-part"); + strmov(key2,key); + for(i=strlen(key2) ; i-- > 1 ;) + { + key2[i]=0; + + /* The following row is just to catch some bugs in the key code */ + bzero((char*) file->lastkey,file->s->base.max_key_length*2); + if (maria_rkey(file,read_record,0,key2,(uint) i,HA_READ_PREFIX)) + goto err; + if (bcmp(read_record+start,key,(uint) i)) + { + puts("Didn't find right record"); + goto end; + } + } + if (dupp_keys > 2) + { + if (!silent) + printf("- Read key (first) - next - delete - next -> last\n"); + DBUG_PRINT("progpos",("first - next - delete - next -> last")); + if (maria_rkey(file,read_record,0,key,0,HA_READ_KEY_EXACT)) goto err; + if (maria_rnext(file,read_record3,0)) goto err; + if (maria_delete(file,read_record3)) goto err; + opt_delete++; + ant=1; + while (maria_rnext(file,read_record3,0) == 0 && + bcmp(read_record3+start,key,length) == 0) ant++; + if (ant != dupp_keys-1) + { + printf("next: I can only find: %d keys of %d\n",ant,dupp_keys-1); + goto end; + } + } + if (dupp_keys>4) + { + if (!silent) + printf("- Read last of key - prev - delete - prev -> first\n"); + DBUG_PRINT("progpos",("last - prev - delete - prev -> first")); + if (maria_rprev(file,read_record3,0)) goto err; + if (maria_rprev(file,read_record3,0)) goto err; + if (maria_delete(file,read_record3)) goto err; + opt_delete++; + ant=1; + while (maria_rprev(file,read_record3,0) == 0 && + bcmp(read_record3+start,key,length) == 0) ant++; + if (ant != dupp_keys-2) + { + printf("next: I can only find: %d keys of %d\n",ant,dupp_keys-2); + goto end; + } + } + if (dupp_keys > 6) + { + if (!silent) + printf("- Read first - delete - next -> last\n"); + DBUG_PRINT("progpos",("first - delete - next -> last")); + if (maria_rkey(file,read_record3,0,key,0,HA_READ_KEY_EXACT)) goto err; + if (maria_delete(file,read_record3)) goto err; + opt_delete++; + ant=1; + if (maria_rnext(file,read_record,0)) + goto err; /* Skall finnas poster */ + while (maria_rnext(file,read_record3,0) == 0 && + bcmp(read_record3+start,key,length) == 0) ant++; + if (ant != dupp_keys-3) + { + printf("next: I can only find: %d keys of %d\n",ant,dupp_keys-3); + goto end; + } + + if (!silent) + printf("- Read last - delete - prev -> first\n"); + DBUG_PRINT("progpos",("last - delete - prev -> first")); + if (maria_rprev(file,read_record3,0)) goto err; + if (maria_delete(file,read_record3)) goto err; + opt_delete++; + ant=0; + while (maria_rprev(file,read_record3,0) == 0 && + bcmp(read_record3+start,key,length) == 0) ant++; + if (ant != dupp_keys-4) + { + printf("next: I can only find: %d keys of %d\n",ant,dupp_keys-4); + goto end; + } + } + + if (!silent) + puts("- Test if: Read rrnd - same"); + DBUG_PRINT("progpos",("Read rrnd - same")); + for (i=0 ; i < write_count ; i++) + { + if (maria_rrnd(file,read_record,i == 0 ? 0L : HA_OFFSET_ERROR) == 0) + break; + } + if (i == write_count) + goto err; + + bmove(read_record2,read_record,reclength); + for (i=min(2,keys) ; i-- > 0 ;) + { + if (maria_rsame(file,read_record2,(int) i)) goto err; + if (bcmp(read_record,read_record2,reclength) != 0) + { + printf("is_rsame didn't find same record\n"); + goto end; + } + } + if (!silent) + puts("- Test maria_records_in_range"); + maria_status(file,&info,HA_STATUS_VARIABLE); + for (i=0 ; i < info.keys ; i++) + { + key_range min_key, max_key; + if (maria_rfirst(file,read_record,(int) i) || + maria_rlast(file,read_record2,(int) i)) + goto err; + copy_key(file,(uint) i,(uchar*) read_record,(uchar*) key); + copy_key(file,(uint) i,(uchar*) read_record2,(uchar*) key2); + min_key.key= key; + min_key.length= USE_WHOLE_KEY; + min_key.flag= HA_READ_KEY_EXACT; + max_key.key= key2; + max_key.length= USE_WHOLE_KEY; + max_key.flag= HA_READ_AFTER_KEY; + + range_records= maria_records_in_range(file,(int) i, &min_key, &max_key); + if (range_records < info.records*8/10 || + range_records > info.records*12/10) + { + printf("maria_records_range returned %ld; Should be about %ld\n", + (long) range_records,(long) info.records); + goto end; + } + if (verbose) + { + printf("maria_records_range returned %ld; Exact is %ld (diff: %4.2g %%)\n", + (long) range_records, (long) info.records, + labs((long) range_records - (long) info.records)*100.0/ + info.records); + } + } + for (i=0 ; i < 5 ; i++) + { + for (j=rnd(1000)+1 ; j>0 && key1[j] == 0 ; j--) ; + for (k=rnd(1000)+1 ; k>0 && key1[k] == 0 ; k--) ; + if (j != 0 && k != 0) + { + key_range min_key, max_key; + if (j > k) + swap_variables(int, j, k); + sprintf(key,"%6d",j); + sprintf(key2,"%6d",k); + + min_key.key= key; + min_key.length= USE_WHOLE_KEY; + min_key.flag= HA_READ_AFTER_KEY; + max_key.key= key2; + max_key.length= USE_WHOLE_KEY; + max_key.flag= HA_READ_BEFORE_KEY; + range_records= maria_records_in_range(file, 0, &min_key, &max_key); + records=0; + for (j++ ; j < k ; j++) + records+=key1[j]; + if ((long) range_records < (long) records*7/10-2 || + (long) range_records > (long) records*14/10+2) + { + printf("maria_records_range for key: %d returned %lu; Should be about %lu\n", + i, (ulong) range_records, (ulong) records); + goto end; + } + if (verbose && records) + { + printf("maria_records_range returned %lu; Exact is %lu (diff: %4.2g %%)\n", + (ulong) range_records, (ulong) records, + labs((long) range_records-(long) records)*100.0/records); + + } + } + } + + if (!silent) + printf("- maria_info\n"); + maria_status(file,&info,HA_STATUS_VARIABLE | HA_STATUS_CONST); + if (info.records != write_count-opt_delete || info.deleted > opt_delete + update + || info.keys != keys) + { + puts("Wrong info from maria_info"); + printf("Got: records: %lu delete: %lu i_keys: %d\n", + (ulong) info.records, (ulong) info.deleted, info.keys); + } + if (verbose) + { + char buff[80]; + get_date(buff,3,info.create_time); + printf("info: Created %s\n",buff); + get_date(buff,3,info.check_time); + printf("info: checked %s\n",buff); + get_date(buff,3,info.update_time); + printf("info: Modified %s\n",buff); + } + + maria_panic(HA_PANIC_WRITE); + maria_panic(HA_PANIC_READ); + if (maria_is_changed(file)) + puts("Warning: maria_is_changed reported that datafile was changed"); + + if (!silent) + printf("- maria_extra(CACHE) + maria_rrnd.... + maria_extra(NO_CACHE)\n"); + if (maria_extra(file,HA_EXTRA_RESET,0) || maria_extra(file,HA_EXTRA_CACHE,0)) + { + if (locking || (!use_blob && !pack_fields)) + { + puts("got error from maria_extra(HA_EXTRA_CACHE)"); + goto end; + } + } + ant=0; + while ((error=maria_rrnd(file,record,HA_OFFSET_ERROR)) != HA_ERR_END_OF_FILE && + ant < write_count + 10) + ant+= error ? 0 : 1; + if (ant != write_count-opt_delete) + { + printf("rrnd with cache: I can only find: %d records of %d\n", + ant,write_count-opt_delete); + goto end; + } + if (maria_extra(file,HA_EXTRA_NO_CACHE,0)) + { + puts("got error from maria_extra(HA_EXTRA_NO_CACHE)"); + goto end; + } + + ant=0; + maria_scan_init(file); + while ((error=maria_scan(file,record)) != HA_ERR_END_OF_FILE && + ant < write_count + 10) + ant+= error ? 0 : 1; + if (ant != write_count-opt_delete) + { + printf("scan with cache: I can only find: %d records of %d\n", + ant,write_count-opt_delete); + goto end; + } + + if (testflag == 4) goto end; + + if (!silent) + printf("- Removing keys\n"); + DBUG_PRINT("progpos",("Removing keys")); + lastpos = HA_OFFSET_ERROR; + /* DBUG_POP(); */ + maria_extra(file,HA_EXTRA_RESET,0); + found_parts=0; + while ((error=maria_rrnd(file,read_record,HA_OFFSET_ERROR)) != + HA_ERR_END_OF_FILE) + { + info.recpos=maria_position(file); + if (lastpos >= info.recpos && lastpos != HA_OFFSET_ERROR) + { + printf("maria_rrnd didn't advance filepointer; old: %ld, new: %ld\n", + (long) lastpos, (long) info.recpos); + goto err; + } + lastpos=info.recpos; + if (error == 0) + { + if (opt_delete == (uint) remove_count) /* While testing */ + goto end; + if (maria_rsame(file,read_record,-1)) + { + printf("can't find record %lx\n",(long) info.recpos); + goto err; + } + if (use_blob) + { + ulong blob_length,pos; + uchar *ptr; + longget(blob_length,read_record+blob_pos+4); + ptr=(uchar*) blob_length; + longget(blob_length,read_record+blob_pos); + for (pos=0 ; pos < blob_length ; pos++) + { + if (ptr[pos] != (uchar) (blob_length+pos)) + { + printf("found blob with wrong info at %ld\n",(long) lastpos); + use_blob=0; + break; + } + } + } + if (maria_delete(file,read_record)) + { + printf("can't delete record: %6.6s, delete_count: %d\n", + read_record, opt_delete); + goto err; + } + opt_delete++; + } + else + found_parts++; + } + if (my_errno != HA_ERR_END_OF_FILE && my_errno != HA_ERR_RECORD_DELETED) + printf("error: %d from maria_rrnd\n",my_errno); + if (write_count != opt_delete) + { + printf("Deleted only %d of %d records (%d parts)\n",opt_delete,write_count, + found_parts); + goto err; + } +end: + if (maria_close(file)) + goto err; + maria_panic(HA_PANIC_CLOSE); /* Should close log */ + if (!silent) + { + printf("\nFollowing test have been made:\n"); + printf("Write records: %d\nUpdate records: %d\nSame-key-read: %d\nDelete records: %d\n", write_count,update,dupp_keys,opt_delete); + if (rec_pointer_size) + printf("Record pointer size: %d\n",rec_pointer_size); + printf("maria_block_size: %lu\n", maria_block_size); + if (key_cacheing) + { + puts("Key cache used"); + printf("key_cache_block_size: %u\n", key_cache_block_size); + if (write_cacheing) + puts("Key cache resized"); + } + if (write_cacheing) + puts("Write cacheing used"); + if (write_cacheing) + puts("quick mode"); + if (async_io && locking) + puts("Asyncron io with locking used"); + else if (locking) + puts("Locking used"); + if (use_blob) + puts("blobs used"); + printf("key cache status: \n\ +blocks used:%10lu\n\ +not flushed:%10lu\n\ +w_requests: %10lu\n\ +writes: %10lu\n\ +r_requests: %10lu\n\ +reads: %10lu\n", + maria_key_cache->blocks_used, + maria_key_cache->global_blocks_changed, + (ulong) maria_key_cache->global_cache_w_requests, + (ulong) maria_key_cache->global_cache_write, + (ulong) maria_key_cache->global_cache_r_requests, + (ulong) maria_key_cache->global_cache_read); + } + end_key_cache(maria_key_cache,1); + if (blob_buffer) + my_free(blob_buffer,MYF(0)); + my_end(silent ? MY_CHECK_ERROR : MY_CHECK_ERROR | MY_GIVE_INFO); + return(0); +err: + printf("got error: %d when using MARIA-database\n",my_errno); + if (file) + VOID(maria_close(file)); + maria_end(); + return(1); +} /* main */ + + + /* l{ser optioner */ + /* OBS! intierar endast DEBUG - ingen debuggning h{r ! */ + +static void get_options(int argc, char **argv) +{ + char *pos,*progname; + + progname= argv[0]; + + while (--argc >0 && *(pos = *(++argv)) == '-' ) { + switch(*++pos) { + case 'B': + pack_type= HA_BINARY_PACK_KEY; + break; + case 'b': + use_blob=1; + break; + case 'K': /* Use key cacheing */ + key_cacheing=1; + if (*++pos) + key_cache_size=atol(pos); + break; + case 'W': /* Use write cacheing */ + write_cacheing=1; + if (*++pos) + my_default_record_cache_size=atoi(pos); + break; + case 'd': + remove_count= atoi(++pos); + break; + case 'i': + if (*++pos) + srand(atoi(pos)); + break; + case 'l': + use_log=1; + break; + case 'L': + locking=1; + break; + case 'A': /* use asyncron io */ + async_io=1; + if (*++pos) + my_default_record_cache_size=atoi(pos); + break; + case 'v': /* verbose */ + verbose=1; + break; + case 'm': /* records */ + if ((recant=atoi(++pos)) < 10) + { + fprintf(stderr,"record count must be >= 10\n"); + exit(1); + } + break; + case 'e': /* maria_block_length */ + if ((maria_block_size=atoi(++pos)) < MARIA_MIN_KEY_BLOCK_LENGTH || + maria_block_size > MARIA_MAX_KEY_BLOCK_LENGTH) + { + fprintf(stderr,"Wrong maria_block_length\n"); + exit(1); + } + maria_block_size=1 << my_bit_log2(maria_block_size); + break; + case 'E': /* maria_block_length */ + if ((key_cache_block_size=atoi(++pos)) < MARIA_MIN_KEY_BLOCK_LENGTH || + key_cache_block_size > MARIA_MAX_KEY_BLOCK_LENGTH) + { + fprintf(stderr,"Wrong key_cache_block_size\n"); + exit(1); + } + key_cache_block_size=1 << my_bit_log2(key_cache_block_size); + break; + case 'f': + if ((first_key=atoi(++pos)) < 0 || first_key >= MARIA_KEYS) + first_key=0; + break; + case 'k': + if ((keys=(uint) atoi(++pos)) < 1 || + keys > (uint) (MARIA_KEYS-first_key)) + keys=MARIA_KEYS-first_key; + break; + case 'P': + pack_type=0; /* Don't use DIFF_LENGTH */ + pack_seg=0; + break; + case 'R': /* Length of record pointer */ + rec_pointer_size=atoi(++pos); + if (rec_pointer_size > 7) + rec_pointer_size=0; + break; + case 'S': + pack_fields=0; /* Static-length-records */ + break; + case 's': + silent=1; + break; + case 't': + testflag=atoi(++pos); /* testmod */ + break; + case 'q': + opt_quick_mode=1; + break; + case 'c': + create_flag|= HA_CREATE_CHECKSUM; + break; + case 'D': + create_flag|=HA_CREATE_DELAY_KEY_WRITE; + break; + case '?': + case 'I': + case 'V': + printf("%s Ver 1.2 for %s at %s\n",progname,SYSTEM_TYPE,MACHINE_TYPE); + puts("By Monty, for your professional use\n"); + printf("Usage: %s [-?AbBcDIKLPRqSsVWltv] [-k#] [-f#] [-m#] [-e#] [-E#] [-t#]\n", + progname); + exit(0); + case '#': + DBUG_PUSH (++pos); + break; + default: + printf("Illegal option: '%c'\n",*pos); + break; + } + } + return; +} /* get options */ + + /* Get a random value 0 <= x <= n */ + +static uint rnd(uint max_value) +{ + return (uint) ((rand() & 32767)/32767.0*max_value); +} /* rnd */ + + + /* Create a variable length record */ + +static void fix_length(byte *rec, uint length) +{ + bmove(rec+STANDARD_LENGTH, + "0123456789012345678901234567890123456789012345678901234567890", + length-STANDARD_LENGTH); + strfill(rec+length,STANDARD_LENGTH+60-length,' '); +} /* fix_length */ + + + /* Put maybe a blob in record */ + +static void put_blob_in_record(char *blob_pos, char **blob_buffer) +{ + ulong i,length; + if (use_blob) + { + if (rnd(10) == 0) + { + if (! *blob_buffer && + !(*blob_buffer=my_malloc((uint) use_blob,MYF(MY_WME)))) + { + use_blob=0; + return; + } + length=rnd(use_blob); + for (i=0 ; i < length ; i++) + (*blob_buffer)[i]=(char) (length+i); + int4store(blob_pos,length); + memcpy_fixed(blob_pos+4,(char*) blob_buffer,sizeof(char*)); + } + else + { + int4store(blob_pos,0); + } + } + return; +} + + +static void copy_key(MARIA_HA *info,uint inx,uchar *rec,uchar *key_buff) +{ + HA_KEYSEG *keyseg; + + for (keyseg=info->s->keyinfo[inx].seg ; keyseg->type ; keyseg++) + { + memcpy(key_buff,rec+keyseg->start,(size_t) keyseg->length); + key_buff+=keyseg->length; + } + return; +} diff --git a/storage/maria/ma_test3.c b/storage/maria/ma_test3.c new file mode 100644 index 00000000000..bfb2c93a95f --- /dev/null +++ b/storage/maria/ma_test3.c @@ -0,0 +1,502 @@ +/* Copyright (C) 2006 MySQL AB & MySQL Finland AB & TCX DataKonsult AB + + 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; either version 2 of the License, or + (at your option) any later version. + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ + +/* Test av locking */ + +#ifndef __NETWARE__ + +#include "maria.h" +#include +#ifdef HAVE_SYS_WAIT_H +# include +#endif +#ifndef WEXITSTATUS +# define WEXITSTATUS(stat_val) ((unsigned)(stat_val) >> 8) +#endif +#ifndef WIFEXITED +# define WIFEXITED(stat_val) (((stat_val) & 255) == 0) +#endif + + +#if defined(HAVE_LRAND48) +#define rnd(X) (lrand48() % X) +#define rnd_init(X) srand48(X) +#else +#define rnd(X) (random() % X) +#define rnd_init(X) srandom(X) +#endif + + +const char *filename= "test3"; +uint tests=10,forks=10,key_cacheing=0,use_log=0; + +static void get_options(int argc, char *argv[]); +void start_test(int id); +int test_read(MARIA_HA *,int),test_write(MARIA_HA *,int,int), + test_update(MARIA_HA *,int,int),test_rrnd(MARIA_HA *,int); + +struct record { + char id[8]; + char nr[4]; + char text[10]; +} record; + + +int main(int argc,char **argv) +{ + int status,wait_ret; + uint i=0; + MARIA_KEYDEF keyinfo[10]; + MARIA_COLUMNDEF recinfo[10]; + HA_KEYSEG keyseg[10][2]; + MY_INIT(argv[0]); + get_options(argc,argv); + + maria_init(); + bzero((char*) keyinfo,sizeof(keyinfo)); + bzero((char*) recinfo,sizeof(recinfo)); + bzero((char*) keyseg,sizeof(keyseg)); + keyinfo[0].seg= &keyseg[0][0]; + keyinfo[0].seg[0].start=0; + keyinfo[0].seg[0].length=8; + keyinfo[0].seg[0].type=HA_KEYTYPE_TEXT; + keyinfo[0].seg[0].flag=HA_SPACE_PACK; + keyinfo[0].key_alg=HA_KEY_ALG_BTREE; + keyinfo[0].keysegs=1; + keyinfo[0].flag = (uint8) HA_PACK_KEY; + keyinfo[1].seg= &keyseg[1][0]; + keyinfo[1].seg[0].start=8; + keyinfo[1].seg[0].length=4; /* Long is always 4 in maria */ + keyinfo[1].seg[0].type=HA_KEYTYPE_LONG_INT; + keyinfo[1].seg[0].flag=0; + keyinfo[1].key_alg=HA_KEY_ALG_BTREE; + keyinfo[1].keysegs=1; + keyinfo[1].flag =HA_NOSAME; + + recinfo[0].type=0; + recinfo[0].length=sizeof(record.id); + recinfo[1].type=0; + recinfo[1].length=sizeof(record.nr); + recinfo[2].type=0; + recinfo[2].length=sizeof(record.text); + + puts("- Creating maria-file"); + my_delete(filename,MYF(0)); /* Remove old locks under gdb */ + if (maria_create(filename,2,&keyinfo[0],2,&recinfo[0],0,(MARIA_UNIQUEDEF*) 0, + (MARIA_CREATE_INFO*) 0,0)) + exit(1); + + rnd_init(0); + printf("- Starting %d processes\n",forks); fflush(stdout); + for (i=0 ; i < forks; i++) + { + if (!fork()) + { + start_test(i+1); + sleep(1); + return 0; + } + VOID(rnd(1)); + } + + for (i=0 ; i < forks ; i++) + while ((wait_ret=wait(&status)) && wait_ret == -1); + maria_end(); + return 0; +} + + +static void get_options(int argc, char **argv) +{ + char *pos,*progname; + + progname= argv[0]; + + while (--argc >0 && *(pos = *(++argv)) == '-' ) { + switch(*++pos) { + case 'l': + use_log=1; + break; + case 'f': + forks=atoi(++pos); + break; + case 't': + tests=atoi(++pos); + break; + case 'K': /* Use key cacheing */ + key_cacheing=1; + break; + case 'A': /* All flags */ + use_log=key_cacheing=1; + break; + case '?': + case 'I': + case 'V': + printf("%s Ver 1.0 for %s at %s\n",progname,SYSTEM_TYPE,MACHINE_TYPE); + puts("By Monty, for your professional use\n"); + puts("Test av locking with threads\n"); + printf("Usage: %s [-?lKA] [-f#] [-t#]\n",progname); + exit(0); + case '#': + DBUG_PUSH (++pos); + break; + default: + printf("Illegal option: '%c'\n",*pos); + break; + } + } + return; +} + + +void start_test(int id) +{ + uint i; + int error,lock_type; + MARIA_INFO isam_info; + MARIA_HA *file,*file1,*file2=0,*lock; + + if (use_log) + maria_logging(1); + if (!(file1=maria_open(filename,O_RDWR,HA_OPEN_WAIT_IF_LOCKED)) || + !(file2=maria_open(filename,O_RDWR,HA_OPEN_WAIT_IF_LOCKED))) + { + fprintf(stderr,"Can't open isam-file: %s\n",filename); + exit(1); + } + if (key_cacheing && rnd(2) == 0) + init_key_cache(maria_key_cache, KEY_CACHE_BLOCK_SIZE, 65536L, 0, 0); + printf("Process %d, pid: %d\n",id,getpid()); fflush(stdout); + + for (error=i=0 ; i < tests && !error; i++) + { + file= (rnd(2) == 1) ? file1 : file2; + lock=0 ; lock_type=0; + if (rnd(10) == 0) + { + if (maria_lock_database(lock=(rnd(2) ? file1 : file2), + lock_type=(rnd(2) == 0 ? F_RDLCK : F_WRLCK))) + { + fprintf(stderr,"%2d: start: Can't lock table %d\n",id,my_errno); + error=1; + break; + } + } + switch (rnd(4)) { + case 0: error=test_read(file,id); break; + case 1: error=test_rrnd(file,id); break; + case 2: error=test_write(file,id,lock_type); break; + case 3: error=test_update(file,id,lock_type); break; + } + if (lock) + maria_lock_database(lock,F_UNLCK); + } + if (!error) + { + maria_status(file1,&isam_info,HA_STATUS_VARIABLE); + printf("%2d: End of test. Records: %ld Deleted: %ld\n", + id,(long) isam_info.records, (long) isam_info.deleted); + fflush(stdout); + } + + maria_close(file1); + maria_close(file2); + if (use_log) + maria_logging(0); + if (error) + { + printf("%2d: Aborted\n",id); fflush(stdout); + exit(1); + } +} + + +int test_read(MARIA_HA *file,int id) +{ + uint i,lock,found,next,prev; + ulong find; + + lock=0; + if (rnd(2) == 0) + { + lock=1; + if (maria_lock_database(file,F_RDLCK)) + { + fprintf(stderr,"%2d: Can't lock table %d\n",id,my_errno); + return 1; + } + } + + found=next=prev=0; + for (i=0 ; i < 100 ; i++) + { + find=rnd(100000); + if (!maria_rkey(file,record.id,1,(byte*) &find, + sizeof(find),HA_READ_KEY_EXACT)) + found++; + else + { + if (my_errno != HA_ERR_KEY_NOT_FOUND) + { + fprintf(stderr,"%2d: Got error %d from read in read\n",id,my_errno); + return 1; + } + else if (!maria_rnext(file,record.id,1)) + next++; + else + { + if (my_errno != HA_ERR_END_OF_FILE) + { + fprintf(stderr,"%2d: Got error %d from rnext in read\n",id,my_errno); + return 1; + } + else if (!maria_rprev(file,record.id,1)) + prev++; + else + { + if (my_errno != HA_ERR_END_OF_FILE) + { + fprintf(stderr,"%2d: Got error %d from rnext in read\n", + id,my_errno); + return 1; + } + } + } + } + } + if (lock) + { + if (maria_lock_database(file,F_UNLCK)) + { + fprintf(stderr,"%2d: Can't unlock table\n",id); + return 1; + } + } + printf("%2d: read: found: %5d next: %5d prev: %5d\n", + id,found,next,prev); + fflush(stdout); + return 0; +} + + +int test_rrnd(MARIA_HA *file,int id) +{ + uint count,lock; + + lock=0; + if (rnd(2) == 0) + { + lock=1; + if (maria_lock_database(file,F_RDLCK)) + { + fprintf(stderr,"%2d: Can't lock table (%d)\n",id,my_errno); + maria_close(file); + return 1; + } + if (rnd(2) == 0) + maria_extra(file,HA_EXTRA_CACHE,0); + } + + count=0; + if (maria_rrnd(file,record.id,0L)) + { + if (my_errno == HA_ERR_END_OF_FILE) + goto end; + fprintf(stderr,"%2d: Can't read first record (%d)\n",id,my_errno); + return 1; + } + for (count=1 ; !maria_rrnd(file,record.id,HA_OFFSET_ERROR) ;count++) ; + if (my_errno != HA_ERR_END_OF_FILE) + { + fprintf(stderr,"%2d: Got error %d from rrnd\n",id,my_errno); + return 1; + } + +end: + if (lock) + { + maria_extra(file,HA_EXTRA_NO_CACHE,0); + if (maria_lock_database(file,F_UNLCK)) + { + fprintf(stderr,"%2d: Can't unlock table\n",id); + exit(0); + } + } + printf("%2d: rrnd: %5d\n",id,count); fflush(stdout); + return 0; +} + + +int test_write(MARIA_HA *file,int id,int lock_type) +{ + uint i,tries,count,lock; + + lock=0; + if (rnd(2) == 0 || lock_type == F_RDLCK) + { + lock=1; + if (maria_lock_database(file,F_WRLCK)) + { + if (lock_type == F_RDLCK && my_errno == EDEADLK) + { + printf("%2d: write: deadlock\n",id); fflush(stdout); + return 0; + } + fprintf(stderr,"%2d: Can't lock table (%d)\n",id,my_errno); + maria_close(file); + return 1; + } + if (rnd(2) == 0) + maria_extra(file,HA_EXTRA_WRITE_CACHE,0); + } + + sprintf(record.id,"%7d",getpid()); + strnmov(record.text,"Testing...", sizeof(record.text)); + + tries=(uint) rnd(100)+10; + for (i=count=0 ; i < tries ; i++) + { + uint32 tmp=rnd(80000)+20000; + int4store(record.nr,tmp); + if (!maria_write(file,record.id)) + count++; + else + { + if (my_errno != HA_ERR_FOUND_DUPP_KEY) + { + fprintf(stderr,"%2d: Got error %d (errno %d) from write\n",id,my_errno, + errno); + return 1; + } + } + } + if (lock) + { + maria_extra(file,HA_EXTRA_NO_CACHE,0); + if (maria_lock_database(file,F_UNLCK)) + { + fprintf(stderr,"%2d: Can't unlock table\n",id); + exit(0); + } + } + printf("%2d: write: %5d\n",id,count); fflush(stdout); + return 0; +} + + +int test_update(MARIA_HA *file,int id,int lock_type) +{ + uint i,lock,found,next,prev,update; + uint32 tmp; + char find[4]; + struct record new_record; + + lock=0; + if (rnd(2) == 0 || lock_type == F_RDLCK) + { + lock=1; + if (maria_lock_database(file,F_WRLCK)) + { + if (lock_type == F_RDLCK && my_errno == EDEADLK) + { + printf("%2d: write: deadlock\n",id); fflush(stdout); + return 0; + } + fprintf(stderr,"%2d: Can't lock table (%d)\n",id,my_errno); + return 1; + } + } + bzero((char*) &new_record,sizeof(new_record)); + strmov(new_record.text,"Updated"); + + found=next=prev=update=0; + for (i=0 ; i < 100 ; i++) + { + tmp=rnd(100000); + int4store(find,tmp); + if (!maria_rkey(file,record.id,1,(byte*) find, + sizeof(find),HA_READ_KEY_EXACT)) + found++; + else + { + if (my_errno != HA_ERR_KEY_NOT_FOUND) + { + fprintf(stderr,"%2d: Got error %d from read in update\n",id,my_errno); + return 1; + } + else if (!maria_rnext(file,record.id,1)) + next++; + else + { + if (my_errno != HA_ERR_END_OF_FILE) + { + fprintf(stderr,"%2d: Got error %d from rnext in update\n", + id,my_errno); + return 1; + } + else if (!maria_rprev(file,record.id,1)) + prev++; + else + { + if (my_errno != HA_ERR_END_OF_FILE) + { + fprintf(stderr,"%2d: Got error %d from rnext in update\n", + id,my_errno); + return 1; + } + continue; + } + } + } + memcpy_fixed(new_record.id,record.id,sizeof(record.id)); + tmp=rnd(20000)+40000; + int4store(new_record.nr,tmp); + if (!maria_update(file,record.id,new_record.id)) + update++; + else + { + if (my_errno != HA_ERR_RECORD_CHANGED && + my_errno != HA_ERR_RECORD_DELETED && + my_errno != HA_ERR_FOUND_DUPP_KEY) + { + fprintf(stderr,"%2d: Got error %d from update\n",id,my_errno); + return 1; + } + } + } + if (lock) + { + if (maria_lock_database(file,F_UNLCK)) + { + fprintf(stderr,"Can't unlock table,id, error%d\n",my_errno); + return 1; + } + } + printf("%2d: update: %5d\n",id,update); fflush(stdout); + return 0; +} + +#else /* __NETWARE__ */ + +#include + +main() +{ + fprintf(stderr,"this test has not been ported to NetWare\n"); + return 0; +} + +#endif /* __NETWARE__ */ diff --git a/storage/maria/ma_test_all.sh b/storage/maria/ma_test_all.sh new file mode 100755 index 00000000000..d848bc63b9a --- /dev/null +++ b/storage/maria/ma_test_all.sh @@ -0,0 +1,147 @@ +#!/bin/sh +# +# Execute some simple basic test on MyISAM libary to check if things +# works at all. + +valgrind="valgrind --alignment=8 --leak-check=yes" +silent="-s" + +if test -f ma_test1$MACH ; then suffix=$MACH else suffix=""; fi +ma_test1$suffix $silent +maria_chk$suffix -se test1 +ma_test1$suffix $silent -N -S +maria_chk$suffix -se test1 +ma_test1$suffix $silent -P --checksum +maria_chk$suffix -se test1 +ma_test1$suffix $silent -P -N -S +maria_chk$suffix -se test1 +ma_test1$suffix $silent -B -N -R2 +maria_chk$suffix -sm test1 +ma_test1$suffix $silent -a -k 480 --unique +maria_chk$suffix -sm test1 +ma_test1$suffix $silent -a -N -S -R1 +maria_chk$suffix -sm test1 +ma_test1$suffix $silent -p -S +maria_chk$suffix -sm test1 +ma_test1$suffix $silent -p -S -N --unique +maria_chk$suffix -sm test1 +ma_test1$suffix $silent -p -S -N --key_length=127 --checksum +maria_chk$suffix -sm test1 +ma_test1$suffix $silent -p -S -N --key_length=128 +maria_chk$suffix -sm test1 +ma_test1$suffix $silent -p -S --key_length=480 +maria_chk$suffix -sm test1 +ma_test1$suffix $silent -a -B +maria_chk$suffix -sm test1 +ma_test1$suffix $silent -a -B --key_length=64 --unique +maria_chk$suffix -sm test1 +ma_test1$suffix $silent -a -B -k 480 --checksum +maria_chk$suffix -sm test1 +ma_test1$suffix $silent -a -B -k 480 -N --unique --checksum +maria_chk$suffix -sm test1 +ma_test1$suffix $silent -a -m +maria_chk$suffix -sm test1 +ma_test1$suffix $silent -a -m -P --unique --checksum +maria_chk$suffix -sm test1 +ma_test1$suffix $silent -a -m -P --key_length=480 --key_cache +maria_chk$suffix -sm test1 +ma_test1$suffix $silent -m -p +maria_chk$suffix -sm test1 +ma_test1$suffix $silent -w -S --unique +maria_chk$suffix -sm test1 +ma_test1$suffix $silent -a -w --key_length=64 --checksum +maria_chk$suffix -sm test1 +ma_test1$suffix $silent -a -w -N --key_length=480 +maria_chk$suffix -sm test1 +ma_test1$suffix $silent -a -w -S --key_length=480 --checksum +maria_chk$suffix -sm test1 +ma_test1$suffix $silent -a -b -N +maria_chk$suffix -sm test1 +ma_test1$suffix $silent -a -b --key_length=480 +maria_chk$suffix -sm test1 +ma_test1$suffix $silent -p -B --key_length=480 +maria_chk$suffix -sm test1 + +ma_test1$suffix $silent --checksum +maria_chk$suffix -se test1 +maria_chk$suffix -rs test1 +maria_chk$suffix -se test1 +maria_chk$suffix -rqs test1 +maria_chk$suffix -se test1 +maria_chk$suffix -rs --correct-checksum test1 +maria_chk$suffix -se test1 +maria_chk$suffix -rqs --correct-checksum test1 +maria_chk$suffix -se test1 +maria_chk$suffix -ros --correct-checksum test1 +maria_chk$suffix -se test1 +maria_chk$suffix -rqos --correct-checksum test1 +maria_chk$suffix -se test1 + +# check of maria_pack / maria_chk +maria_pack$suffix --force -s test1 +maria_chk$suffix -es test1 +maria_chk$suffix -rqs test1 +maria_chk$suffix -es test1 +maria_chk$suffix -rs test1 +maria_chk$suffix -es test1 +maria_chk$suffix -rus test1 +maria_chk$suffix -es test1 + +ma_test1$suffix $silent --checksum -S +maria_chk$suffix -se test1 +maria_chk$suffix -ros test1 +maria_chk$suffix -rqs test1 +maria_chk$suffix -se test1 + +maria_pack$suffix --force -s test1 +maria_chk$suffix -rqs test1 +maria_chk$suffix -es test1 +maria_chk$suffix -rus test1 +maria_chk$suffix -es test1 + +ma_test1$suffix $silent --checksum --unique +maria_chk$suffix -se test1 +ma_test1$suffix $silent --unique -S +maria_chk$suffix -se test1 + + +ma_test1$suffix $silent --key_multiple -N -S +maria_chk$suffix -sm test1 +ma_test1$suffix $silent --key_multiple -a -p --key_length=480 +maria_chk$suffix -sm test1 +ma_test1$suffix $silent --key_multiple -a -B --key_length=480 +maria_chk$suffix -sm test1 +ma_test1$suffix $silent --key_multiple -P -S +maria_chk$suffix -sm test1 + +ma_test2$suffix $silent -L -K -W -P +maria_chk$suffix -sm test2 +ma_test2$suffix $silent -L -K -W -P -A +maria_chk$suffix -sm test2 +ma_test2$suffix $silent -L -K -W -P -S -R1 -m500 +echo "ma_test2$suffix $silent -L -K -R1 -m2000 ; Should give error 135" +maria_chk$suffix -sm test2 +ma_test2$suffix $silent -L -K -R1 -m2000 +maria_chk$suffix -sm test2 +ma_test2$suffix $silent -L -K -P -S -R3 -m50 -b1000000 +maria_chk$suffix -sm test2 +ma_test2$suffix $silent -L -B +maria_chk$suffix -sm test2 +ma_test2$suffix $silent -D -B -c +maria_chk$suffix -sm test2 +ma_test2$suffix $silent -m10000 -e8192 -K +maria_chk$suffix -sm test2 +ma_test2$suffix $silent -m10000 -e16384 -E16384 -K -L +maria_chk$suffix -sm test2 + +ma_test2$suffix $silent -L -K -W -P -m50 -l +maria_log$suffix +ma_test2$suffix $silent -L -K -W -P -m50 -l -b100 +maria_log$suffix +time ma_test2$suffix $silent +time ma_test2$suffix $silent -K -B +time ma_test2$suffix $silent -L -B +time ma_test2$suffix $silent -L -K -B +time ma_test2$suffix $silent -L -K -W -B +time ma_test2$suffix $silent -L -K -W -S -B +time ma_test2$suffix $silent -D -K -W -S -B diff --git a/storage/maria/ma_unique.c b/storage/maria/ma_unique.c new file mode 100644 index 00000000000..bc1aa71966b --- /dev/null +++ b/storage/maria/ma_unique.c @@ -0,0 +1,234 @@ +/* Copyright (C) 2006 MySQL AB & MySQL Finland AB & TCX DataKonsult AB + + 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; either version 2 of the License, or + (at your option) any later version. + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ + +/* Functions to check if a row is unique */ + +#include "maria_def.h" +#include + +my_bool _ma_check_unique(MARIA_HA *info, MARIA_UNIQUEDEF *def, byte *record, + ha_checksum unique_hash, my_off_t disk_pos) +{ + my_off_t lastpos=info->lastpos; + MARIA_KEYDEF *key= &info->s->keyinfo[def->key]; + uchar *key_buff=info->lastkey2; + DBUG_ENTER("_ma_check_unique"); + + maria_unique_store(record+key->seg->start, unique_hash); + _ma_make_key(info,def->key,key_buff,record,0); + + /* The above changed info->lastkey2. Inform maria_rnext_same(). */ + info->update&= ~HA_STATE_RNEXT_SAME; + + if (_ma_search(info,info->s->keyinfo+def->key,key_buff,MARIA_UNIQUE_HASH_LENGTH, + SEARCH_FIND,info->s->state.key_root[def->key])) + { + info->page_changed=1; /* Can't optimize read next */ + info->lastpos= lastpos; + DBUG_RETURN(0); /* No matching rows */ + } + + for (;;) + { + if (info->lastpos != disk_pos && + !(*info->s->compare_unique)(info,def,record,info->lastpos)) + { + my_errno=HA_ERR_FOUND_DUPP_UNIQUE; + info->errkey= (int) def->key; + info->dupp_key_pos= info->lastpos; + info->page_changed=1; /* Can't optimize read next */ + info->lastpos=lastpos; + DBUG_PRINT("info",("Found duplicate")); + DBUG_RETURN(1); /* Found identical */ + } + if (_ma_search_next(info,info->s->keyinfo+def->key, info->lastkey, + MARIA_UNIQUE_HASH_LENGTH, SEARCH_BIGGER, + info->s->state.key_root[def->key]) || + memcmp((char*) info->lastkey, (char*) key_buff, + MARIA_UNIQUE_HASH_LENGTH)) + { + info->page_changed=1; /* Can't optimize read next */ + info->lastpos=lastpos; + DBUG_RETURN(0); /* end of tree */ + } + } +} + + +/* + Calculate a hash for a row + + TODO + Add support for bit fields +*/ + +ha_checksum _ma_unique_hash(MARIA_UNIQUEDEF *def, const byte *record) +{ + const byte *pos, *end; + ha_checksum crc= 0; + ulong seed1=0, seed2= 4; + HA_KEYSEG *keyseg; + + for (keyseg=def->seg ; keyseg < def->end ; keyseg++) + { + enum ha_base_keytype type=(enum ha_base_keytype) keyseg->type; + uint length=keyseg->length; + + if (keyseg->null_bit) + { + if (record[keyseg->null_pos] & keyseg->null_bit) + { + /* + Change crc in a way different from an empty string or 0. + (This is an optimisation; The code will work even if this isn't + done) + */ + crc=((crc << 8) + 511+ + (crc >> (8*sizeof(ha_checksum)-8))); + continue; + } + } + pos= record+keyseg->start; + if (keyseg->flag & HA_VAR_LENGTH_PART) + { + uint pack_length= keyseg->bit_start; + uint tmp_length= (pack_length == 1 ? (uint) *(uchar*) pos : + uint2korr(pos)); + pos+= pack_length; /* Skip VARCHAR length */ + set_if_smaller(length,tmp_length); + } + else if (keyseg->flag & HA_BLOB_PART) + { + uint tmp_length= _ma_calc_blob_length(keyseg->bit_start,pos); + memcpy_fixed((byte*) &pos,pos+keyseg->bit_start,sizeof(char*)); + if (!length || length > tmp_length) + length=tmp_length; /* The whole blob */ + } + end= pos+length; + if (type == HA_KEYTYPE_TEXT || type == HA_KEYTYPE_VARTEXT1 || + type == HA_KEYTYPE_VARTEXT2) + { + keyseg->charset->coll->hash_sort(keyseg->charset, + (const uchar*) pos, length, &seed1, + &seed2); + crc^= seed1; + } + else + while (pos != end) + crc=((crc << 8) + + (((uchar) *(uchar*) pos++))) + + (crc >> (8*sizeof(ha_checksum)-8)); + } + return crc; +} + + +/* + compare unique key for two rows + + TODO + Add support for bit fields + + RETURN + 0 if both rows have equal unique value + # Rows are different +*/ + +int _ma_unique_comp(MARIA_UNIQUEDEF *def, const byte *a, const byte *b, + my_bool null_are_equal) +{ + const byte *pos_a, *pos_b, *end; + HA_KEYSEG *keyseg; + + for (keyseg=def->seg ; keyseg < def->end ; keyseg++) + { + enum ha_base_keytype type=(enum ha_base_keytype) keyseg->type; + uint a_length, b_length; + a_length= b_length= keyseg->length; + + /* If part is NULL it's regarded as different */ + if (keyseg->null_bit) + { + uint tmp; + if ((tmp=(a[keyseg->null_pos] & keyseg->null_bit)) != + (uint) (b[keyseg->null_pos] & keyseg->null_bit)) + return 1; + if (tmp) + { + if (!null_are_equal) + return 1; + continue; + } + } + pos_a= a+keyseg->start; + pos_b= b+keyseg->start; + if (keyseg->flag & HA_VAR_LENGTH_PART) + { + uint pack_length= keyseg->bit_start; + if (pack_length == 1) + { + a_length= (uint) *(uchar*) pos_a++; + b_length= (uint) *(uchar*) pos_b++; + } + else + { + a_length= uint2korr(pos_a); + b_length= uint2korr(pos_b); + pos_a+= 2; /* Skip VARCHAR length */ + pos_b+= 2; + } + set_if_smaller(a_length, keyseg->length); /* Safety */ + set_if_smaller(b_length, keyseg->length); /* safety */ + } + else if (keyseg->flag & HA_BLOB_PART) + { + /* Only compare 'length' characters if length != 0 */ + a_length= _ma_calc_blob_length(keyseg->bit_start,pos_a); + b_length= _ma_calc_blob_length(keyseg->bit_start,pos_b); + /* Check that a and b are of equal length */ + if (keyseg->length) + { + /* + This is used in some cases when we are not interested in comparing + the whole length of the blob. + */ + set_if_smaller(a_length, keyseg->length); + set_if_smaller(b_length, keyseg->length); + } + memcpy_fixed((byte*) &pos_a,pos_a+keyseg->bit_start,sizeof(char*)); + memcpy_fixed((byte*) &pos_b,pos_b+keyseg->bit_start,sizeof(char*)); + } + if (type == HA_KEYTYPE_TEXT || type == HA_KEYTYPE_VARTEXT1 || + type == HA_KEYTYPE_VARTEXT2) + { + if (ha_compare_text(keyseg->charset, (uchar *) pos_a, a_length, + (uchar *) pos_b, b_length, 0, 1)) + return 1; + } + else + { + if (a_length != b_length) + return 1; + end= pos_a+a_length; + while (pos_a != end) + { + if (*pos_a++ != *pos_b++) + return 1; + } + } + } + return 0; +} diff --git a/storage/maria/ma_update.c b/storage/maria/ma_update.c new file mode 100644 index 00000000000..d3c71cef9b4 --- /dev/null +++ b/storage/maria/ma_update.c @@ -0,0 +1,232 @@ +/* Copyright (C) 2006 MySQL AB & MySQL Finland AB & TCX DataKonsult AB + + 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; either version 2 of the License, or + (at your option) any later version. + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ + +/* Update an old row in a MARIA table */ + +#include "ma_fulltext.h" +#include "ma_rt_index.h" + +int maria_update(register MARIA_HA *info, const byte *oldrec, byte *newrec) +{ + int flag,key_changed,save_errno; + reg3 my_off_t pos; + uint i; + uchar old_key[HA_MAX_KEY_BUFF],*new_key; + bool auto_key_changed=0; + ulonglong changed; + MARIA_SHARE *share=info->s; + ha_checksum old_checksum; + DBUG_ENTER("maria_update"); + LINT_INIT(new_key); + LINT_INIT(changed); + LINT_INIT(old_checksum); + + DBUG_EXECUTE_IF("maria_pretend_crashed_table_on_usage", + maria_print_error(info->s, HA_ERR_CRASHED); + DBUG_RETURN(my_errno= HA_ERR_CRASHED);); + if (!(info->update & HA_STATE_AKTIV)) + { + DBUG_RETURN(my_errno=HA_ERR_KEY_NOT_FOUND); + } + if (share->options & HA_OPTION_READ_ONLY_DATA) + { + DBUG_RETURN(my_errno=EACCES); + } + if (info->state->key_file_length >= share->base.margin_key_file_length) + { + DBUG_RETURN(my_errno=HA_ERR_INDEX_FILE_FULL); + } + pos=info->lastpos; + if (_ma_readinfo(info,F_WRLCK,1)) + DBUG_RETURN(my_errno); + + if (share->calc_checksum) + old_checksum=info->checksum=(*share->calc_checksum)(info,oldrec); + if ((*share->compare_record)(info,oldrec)) + { + save_errno=my_errno; + goto err_end; /* Record has changed */ + } + + + /* Calculate and check all unique constraints */ + key_changed=0; + for (i=0 ; i < share->state.header.uniques ; i++) + { + MARIA_UNIQUEDEF *def=share->uniqueinfo+i; + if (_ma_unique_comp(def, newrec, oldrec,1) && + _ma_check_unique(info, def, newrec, _ma_unique_hash(def, newrec), + info->lastpos)) + { + save_errno=my_errno; + goto err_end; + } + } + if (_ma_mark_file_changed(info)) + { + save_errno=my_errno; + goto err_end; + } + + /* Check which keys changed from the original row */ + + new_key=info->lastkey2; + changed=0; + for (i=0 ; i < share->base.keys ; i++) + { + if (maria_is_key_active(share->state.key_map, i)) + { + if (share->keyinfo[i].flag & HA_FULLTEXT ) + { + if (_ma_ft_cmp(info,i,oldrec, newrec)) + { + if ((int) i == info->lastinx) + { + /* + We are changeing the index we are reading on. Mark that + the index data has changed and we need to do a full search + when doing read-next + */ + key_changed|=HA_STATE_WRITTEN; + } + changed|=((ulonglong) 1 << i); + if (_ma_ft_update(info,i,(char*) old_key,oldrec,newrec,pos)) + goto err; + } + } + else + { + uint new_length= _ma_make_key(info,i,new_key,newrec,pos); + uint old_length= _ma_make_key(info,i,old_key,oldrec,pos); + + /* The above changed info->lastkey2. Inform maria_rnext_same(). */ + info->update&= ~HA_STATE_RNEXT_SAME; + + if (new_length != old_length || + memcmp((byte*) old_key,(byte*) new_key,new_length)) + { + if ((int) i == info->lastinx) + key_changed|=HA_STATE_WRITTEN; /* Mark that keyfile changed */ + changed|=((ulonglong) 1 << i); + share->keyinfo[i].version++; + if (share->keyinfo[i].ck_delete(info,i,old_key,old_length)) goto err; + if (share->keyinfo[i].ck_insert(info,i,new_key,new_length)) goto err; + if (share->base.auto_key == i+1) + auto_key_changed=1; + } + } + } + } + /* + If we are running with external locking, we must update the index file + that something has changed. + */ + if (changed || !my_disable_locking) + key_changed|= HA_STATE_CHANGED; + + if (share->calc_checksum) + { + info->checksum=(*share->calc_checksum)(info,newrec); + /* Store new checksum in index file header */ + key_changed|= HA_STATE_CHANGED; + } + { + /* + Don't update index file if data file is not extended and no status + information changed + */ + MARIA_STATUS_INFO state; + ha_rows org_split; + my_off_t org_delete_link; + + memcpy((char*) &state, (char*) info->state, sizeof(state)); + org_split= share->state.split; + org_delete_link= share->state.dellink; + if ((*share->update_record)(info,pos,newrec)) + goto err; + if (!key_changed && + (memcmp((char*) &state, (char*) info->state, sizeof(state)) || + org_split != share->state.split || + org_delete_link != share->state.dellink)) + key_changed|= HA_STATE_CHANGED; /* Must update index file */ + } + if (auto_key_changed) + _ma_update_auto_increment(info,newrec); + if (share->calc_checksum) + info->state->checksum+=(info->checksum - old_checksum); + + info->update= (HA_STATE_CHANGED | HA_STATE_ROW_CHANGED | HA_STATE_AKTIV | + key_changed); + maria_log_record(MARIA_LOG_UPDATE,info,newrec,info->lastpos,0); + VOID(_ma_writeinfo(info,key_changed ? WRITEINFO_UPDATE_KEYFILE : 0)); + allow_break(); /* Allow SIGHUP & SIGINT */ + if (info->invalidator != 0) + { + DBUG_PRINT("info", ("invalidator... '%s' (update)", info->filename)); + (*info->invalidator)(info->filename); + info->invalidator=0; + } + DBUG_RETURN(0); + +err: + DBUG_PRINT("error",("key: %d errno: %d",i,my_errno)); + save_errno=my_errno; + if (changed) + key_changed|= HA_STATE_CHANGED; + if (my_errno == HA_ERR_FOUND_DUPP_KEY || my_errno == HA_ERR_RECORD_FILE_FULL) + { + info->errkey= (int) i; + flag=0; + do + { + if (((ulonglong) 1 << i) & changed) + { + if (share->keyinfo[i].flag & HA_FULLTEXT) + { + if ((flag++ && _ma_ft_del(info,i,(char*) new_key,newrec,pos)) || + _ma_ft_add(info,i,(char*) old_key,oldrec,pos)) + break; + } + else + { + uint new_length= _ma_make_key(info,i,new_key,newrec,pos); + uint old_length= _ma_make_key(info,i,old_key,oldrec,pos); + if ((flag++ && _ma_ck_delete(info,i,new_key,new_length)) || + _ma_ck_write(info,i,old_key,old_length)) + break; + } + } + } while (i-- != 0); + } + else + { + maria_print_error(info->s, HA_ERR_CRASHED); + maria_mark_crashed(info); + } + info->update= (HA_STATE_CHANGED | HA_STATE_AKTIV | HA_STATE_ROW_CHANGED | + key_changed); + + err_end: + maria_log_record(MARIA_LOG_UPDATE,info,newrec,info->lastpos,my_errno); + VOID(_ma_writeinfo(info,WRITEINFO_UPDATE_KEYFILE)); + allow_break(); /* Allow SIGHUP & SIGINT */ + if (save_errno == HA_ERR_KEY_NOT_FOUND) + { + maria_print_error(info->s, HA_ERR_CRASHED); + save_errno=HA_ERR_CRASHED; + } + DBUG_RETURN(my_errno=save_errno); +} /* maria_update */ diff --git a/storage/maria/ma_write.c b/storage/maria/ma_write.c new file mode 100644 index 00000000000..313a362f6d6 --- /dev/null +++ b/storage/maria/ma_write.c @@ -0,0 +1,1033 @@ +/* Copyright (C) 2006 MySQL AB & MySQL Finland AB & TCX DataKonsult AB + + 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; either version 2 of the License, or + (at your option) any later version. + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ + +/* Write a row to a MARIA table */ + +#include "ma_fulltext.h" +#include "ma_rt_index.h" + +#define MAX_POINTER_LENGTH 8 + + /* Functions declared in this file */ + +static int w_search(MARIA_HA *info,MARIA_KEYDEF *keyinfo, + uint comp_flag, uchar *key, + uint key_length, my_off_t pos, uchar *father_buff, + uchar *father_keypos, my_off_t father_page, + my_bool insert_last); +static int _ma_balance_page(MARIA_HA *info,MARIA_KEYDEF *keyinfo,uchar *key, + uchar *curr_buff,uchar *father_buff, + uchar *father_keypos,my_off_t father_page); +static uchar *_ma_find_last_pos(MARIA_KEYDEF *keyinfo, uchar *page, + uchar *key, uint *return_key_length, + uchar **after_key); +int _ma_ck_write_tree(register MARIA_HA *info, uint keynr,uchar *key, + uint key_length); +int _ma_ck_write_btree(register MARIA_HA *info, uint keynr,uchar *key, + uint key_length); + + /* Write new record to database */ + +int maria_write(MARIA_HA *info, byte *record) +{ + MARIA_SHARE *share=info->s; + uint i; + int save_errno; + my_off_t filepos; + uchar *buff; + my_bool lock_tree= share->concurrent_insert; + DBUG_ENTER("maria_write"); + DBUG_PRINT("enter",("isam: %d data: %d",info->s->kfile,info->dfile)); + + DBUG_EXECUTE_IF("maria_pretend_crashed_table_on_usage", + maria_print_error(info->s, HA_ERR_CRASHED); + DBUG_RETURN(my_errno= HA_ERR_CRASHED);); + if (share->options & HA_OPTION_READ_ONLY_DATA) + { + DBUG_RETURN(my_errno=EACCES); + } + if (_ma_readinfo(info,F_WRLCK,1)) + DBUG_RETURN(my_errno); + dont_break(); /* Dont allow SIGHUP or SIGINT */ +#if !defined(NO_LOCKING) && defined(USE_RECORD_LOCK) + if (!info->locked && my_lock(info->dfile,F_WRLCK,0L,F_TO_EOF, + MYF(MY_SEEK_NOT_DONE) | info->lock_wait)) + goto err; +#endif + filepos= ((share->state.dellink != HA_OFFSET_ERROR && + !info->append_insert_at_end) ? + share->state.dellink : + info->state->data_file_length); + + if (share->base.reloc == (ha_rows) 1 && + share->base.records == (ha_rows) 1 && + info->state->records == (ha_rows) 1) + { /* System file */ + my_errno=HA_ERR_RECORD_FILE_FULL; + goto err2; + } + if (info->state->key_file_length >= share->base.margin_key_file_length) + { + my_errno=HA_ERR_INDEX_FILE_FULL; + goto err2; + } + if (_ma_mark_file_changed(info)) + goto err2; + + /* Calculate and check all unique constraints */ + for (i=0 ; i < share->state.header.uniques ; i++) + { + if (_ma_check_unique(info,share->uniqueinfo+i,record, + _ma_unique_hash(share->uniqueinfo+i,record), + HA_OFFSET_ERROR)) + goto err2; + } + + /* Write all keys to indextree */ + + buff=info->lastkey2; + for (i=0 ; i < share->base.keys ; i++) + { + if (maria_is_key_active(share->state.key_map, i)) + { + bool local_lock_tree= (lock_tree && + !(info->bulk_insert && + is_tree_inited(&info->bulk_insert[i]))); + if (local_lock_tree) + { + rw_wrlock(&share->key_root_lock[i]); + share->keyinfo[i].version++; + } + if (share->keyinfo[i].flag & HA_FULLTEXT ) + { + if (_ma_ft_add(info,i,(char*) buff,record,filepos)) + { + if (local_lock_tree) + rw_unlock(&share->key_root_lock[i]); + DBUG_PRINT("error",("Got error: %d on write",my_errno)); + goto err; + } + } + else + { + if (share->keyinfo[i].ck_insert(info,i,buff, + _ma_make_key(info,i,buff,record, + filepos))) + { + if (local_lock_tree) + rw_unlock(&share->key_root_lock[i]); + DBUG_PRINT("error",("Got error: %d on write",my_errno)); + goto err; + } + } + + /* The above changed info->lastkey2. Inform maria_rnext_same(). */ + info->update&= ~HA_STATE_RNEXT_SAME; + + if (local_lock_tree) + rw_unlock(&share->key_root_lock[i]); + } + } + if (share->calc_checksum) + info->checksum=(*share->calc_checksum)(info,record); + if (!(info->opt_flag & OPT_NO_ROWS)) + { + if ((*share->write_record)(info,record)) + goto err; + info->state->checksum+=info->checksum; + } + if (share->base.auto_key) + _ma_update_auto_increment(info,record); + info->update= (HA_STATE_CHANGED | HA_STATE_AKTIV | HA_STATE_WRITTEN | + HA_STATE_ROW_CHANGED); + info->state->records++; + info->lastpos=filepos; + maria_log_record(MARIA_LOG_WRITE,info,record,filepos,0); + VOID(_ma_writeinfo(info, WRITEINFO_UPDATE_KEYFILE)); + if (info->invalidator != 0) + { + DBUG_PRINT("info", ("invalidator... '%s' (update)", info->filename)); + (*info->invalidator)(info->filename); + info->invalidator=0; + } + allow_break(); /* Allow SIGHUP & SIGINT */ + DBUG_RETURN(0); + +err: + save_errno=my_errno; + if (my_errno == HA_ERR_FOUND_DUPP_KEY || my_errno == HA_ERR_RECORD_FILE_FULL || + my_errno == HA_ERR_NULL_IN_SPATIAL) + { + if (info->bulk_insert) + { + uint j; + for (j=0 ; j < share->base.keys ; j++) + maria_flush_bulk_insert(info, j); + } + info->errkey= (int) i; + while ( i-- > 0) + { + if (maria_is_key_active(share->state.key_map, i)) + { + bool local_lock_tree= (lock_tree && + !(info->bulk_insert && + is_tree_inited(&info->bulk_insert[i]))); + if (local_lock_tree) + rw_wrlock(&share->key_root_lock[i]); + if (share->keyinfo[i].flag & HA_FULLTEXT) + { + if (_ma_ft_del(info,i,(char*) buff,record,filepos)) + { + if (local_lock_tree) + rw_unlock(&share->key_root_lock[i]); + break; + } + } + else + { + uint key_length= _ma_make_key(info,i,buff,record,filepos); + if (_ma_ck_delete(info,i,buff,key_length)) + { + if (local_lock_tree) + rw_unlock(&share->key_root_lock[i]); + break; + } + } + if (local_lock_tree) + rw_unlock(&share->key_root_lock[i]); + } + } + } + else + { + maria_print_error(info->s, HA_ERR_CRASHED); + maria_mark_crashed(info); + } + info->update= (HA_STATE_CHANGED | HA_STATE_WRITTEN | HA_STATE_ROW_CHANGED); + my_errno=save_errno; +err2: + save_errno=my_errno; + maria_log_record(MARIA_LOG_WRITE,info,record,filepos,my_errno); + VOID(_ma_writeinfo(info,WRITEINFO_UPDATE_KEYFILE)); + allow_break(); /* Allow SIGHUP & SIGINT */ + DBUG_RETURN(my_errno=save_errno); +} /* maria_write */ + + + /* Write one key to btree */ + +int _ma_ck_write(MARIA_HA *info, uint keynr, uchar *key, uint key_length) +{ + DBUG_ENTER("_ma_ck_write"); + + if (info->bulk_insert && is_tree_inited(&info->bulk_insert[keynr])) + { + DBUG_RETURN(_ma_ck_write_tree(info, keynr, key, key_length)); + } + else + { + DBUG_RETURN(_ma_ck_write_btree(info, keynr, key, key_length)); + } +} /* _ma_ck_write */ + + +/********************************************************************** + * Normal insert code * + **********************************************************************/ + +int _ma_ck_write_btree(register MARIA_HA *info, uint keynr, uchar *key, + uint key_length) +{ + int error; + uint comp_flag; + MARIA_KEYDEF *keyinfo=info->s->keyinfo+keynr; + my_off_t *root=&info->s->state.key_root[keynr]; + DBUG_ENTER("_ma_ck_write_btree"); + + if (keyinfo->flag & HA_SORT_ALLOWS_SAME) + comp_flag=SEARCH_BIGGER; /* Put after same key */ + else if (keyinfo->flag & (HA_NOSAME|HA_FULLTEXT)) + { + comp_flag=SEARCH_FIND | SEARCH_UPDATE; /* No duplicates */ + if (keyinfo->flag & HA_NULL_ARE_EQUAL) + comp_flag|= SEARCH_NULL_ARE_EQUAL; + } + else + comp_flag=SEARCH_SAME; /* Keys in rec-pos order */ + + error= _ma_ck_real_write_btree(info, keyinfo, key, key_length, + root, comp_flag); + if (info->ft1_to_ft2) + { + if (!error) + error= _ma_ft_convert_to_ft2(info, keynr, key); + delete_dynamic(info->ft1_to_ft2); + my_free((gptr)info->ft1_to_ft2, MYF(0)); + info->ft1_to_ft2=0; + } + DBUG_RETURN(error); +} /* _ma_ck_write_btree */ + +int _ma_ck_real_write_btree(MARIA_HA *info, MARIA_KEYDEF *keyinfo, + uchar *key, uint key_length, my_off_t *root, uint comp_flag) +{ + int error; + DBUG_ENTER("_ma_ck_real_write_btree"); + /* key_length parameter is used only if comp_flag is SEARCH_FIND */ + if (*root == HA_OFFSET_ERROR || + (error=w_search(info, keyinfo, comp_flag, key, key_length, + *root, (uchar *) 0, (uchar*) 0, + (my_off_t) 0, 1)) > 0) + error= _ma_enlarge_root(info,keyinfo,key,root); + DBUG_RETURN(error); +} /* _ma_ck_real_write_btree */ + + + /* Make a new root with key as only pointer */ + +int _ma_enlarge_root(MARIA_HA *info, MARIA_KEYDEF *keyinfo, uchar *key, + my_off_t *root) +{ + uint t_length,nod_flag; + MARIA_KEY_PARAM s_temp; + MARIA_SHARE *share=info->s; + DBUG_ENTER("_ma_enlarge_root"); + + nod_flag= (*root != HA_OFFSET_ERROR) ? share->base.key_reflength : 0; + _ma_kpointer(info,info->buff+2,*root); /* if nod */ + t_length=(*keyinfo->pack_key)(keyinfo,nod_flag,(uchar*) 0, + (uchar*) 0, (uchar*) 0, key,&s_temp); + maria_putint(info->buff,t_length+2+nod_flag,nod_flag); + (*keyinfo->store_key)(keyinfo,info->buff+2+nod_flag,&s_temp); + info->buff_used=info->page_changed=1; /* info->buff is used */ + if ((*root= _ma_new(info,keyinfo,DFLT_INIT_HITS)) == HA_OFFSET_ERROR || + _ma_write_keypage(info,keyinfo,*root,DFLT_INIT_HITS,info->buff)) + DBUG_RETURN(-1); + DBUG_RETURN(0); +} /* _ma_enlarge_root */ + + + /* + Search after a position for a key and store it there + Returns -1 = error + 0 = ok + 1 = key should be stored in higher tree + */ + +static int w_search(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, + uint comp_flag, uchar *key, uint key_length, my_off_t page, + uchar *father_buff, uchar *father_keypos, + my_off_t father_page, my_bool insert_last) +{ + int error,flag; + uint nod_flag, search_key_length; + uchar *temp_buff,*keypos; + uchar keybuff[HA_MAX_KEY_BUFF]; + my_bool was_last_key; + my_off_t next_page, dupp_key_pos; + DBUG_ENTER("w_search"); + DBUG_PRINT("enter",("page: %ld",page)); + + search_key_length= (comp_flag & SEARCH_FIND) ? key_length : USE_WHOLE_KEY; + if (!(temp_buff= (uchar*) my_alloca((uint) keyinfo->block_length+ + HA_MAX_KEY_BUFF*2))) + DBUG_RETURN(-1); + if (!_ma_fetch_keypage(info,keyinfo,page,DFLT_INIT_HITS,temp_buff,0)) + goto err; + + flag=(*keyinfo->bin_search)(info,keyinfo,temp_buff,key,search_key_length, + comp_flag, &keypos, keybuff, &was_last_key); + nod_flag= _ma_test_if_nod(temp_buff); + if (flag == 0) + { + uint tmp_key_length; + /* get position to record with duplicated key */ + tmp_key_length=(*keyinfo->get_key)(keyinfo,nod_flag,&keypos,keybuff); + if (tmp_key_length) + dupp_key_pos= _ma_dpos(info,0,keybuff+tmp_key_length); + else + dupp_key_pos= HA_OFFSET_ERROR; + + if (keyinfo->flag & HA_FULLTEXT) + { + uint off; + int subkeys; + + get_key_full_length_rdonly(off, keybuff); + subkeys=ft_sintXkorr(keybuff+off); + comp_flag=SEARCH_SAME; + if (subkeys >= 0) + { + /* normal word, one-level tree structure */ + flag=(*keyinfo->bin_search)(info, keyinfo, temp_buff, key, + USE_WHOLE_KEY, comp_flag, + &keypos, keybuff, &was_last_key); + } + else + { + /* popular word. two-level tree. going down */ + my_off_t root=dupp_key_pos; + keyinfo=&info->s->ft2_keyinfo; + get_key_full_length_rdonly(off, key); + key+=off; + keypos-=keyinfo->keylength+nod_flag; /* we'll modify key entry 'in vivo' */ + error= _ma_ck_real_write_btree(info, keyinfo, key, 0, + &root, comp_flag); + _ma_dpointer(info, keypos+HA_FT_WLEN, root); + subkeys--; /* should there be underflow protection ? */ + DBUG_ASSERT(subkeys < 0); + ft_intXstore(keypos, subkeys); + if (!error) + error= _ma_write_keypage(info,keyinfo,page,DFLT_INIT_HITS,temp_buff); + my_afree((byte*) temp_buff); + DBUG_RETURN(error); + } + } + else /* not HA_FULLTEXT, normal HA_NOSAME key */ + { + info->dupp_key_pos= dupp_key_pos; + my_afree((byte*) temp_buff); + my_errno=HA_ERR_FOUND_DUPP_KEY; + DBUG_RETURN(-1); + } + } + if (flag == MARIA_FOUND_WRONG_KEY) + DBUG_RETURN(-1); + if (!was_last_key) + insert_last=0; + next_page= _ma_kpos(nod_flag,keypos); + if (next_page == HA_OFFSET_ERROR || + (error=w_search(info, keyinfo, comp_flag, key, key_length, next_page, + temp_buff, keypos, page, insert_last)) >0) + { + error= _ma_insert(info,keyinfo,key,temp_buff,keypos,keybuff,father_buff, + father_keypos,father_page, insert_last); + if (_ma_write_keypage(info,keyinfo,page,DFLT_INIT_HITS,temp_buff)) + goto err; + } + my_afree((byte*) temp_buff); + DBUG_RETURN(error); +err: + my_afree((byte*) temp_buff); + DBUG_PRINT("exit",("Error: %d",my_errno)); + DBUG_RETURN (-1); +} /* w_search */ + + +/* + Insert new key. + + SYNOPSIS + _ma_insert() + info Open table information. + keyinfo Key definition information. + key New key. + anc_buff Key page (beginning). + key_pos Position in key page where to insert. + key_buff Copy of previous key. + father_buff parent key page for balancing. + father_key_pos position in parent key page for balancing. + father_page position of parent key page in file. + insert_last If to append at end of page. + + DESCRIPTION + Insert new key at right of key_pos. + + RETURN + 2 if key contains key to upper level. + 0 OK. + < 0 Error. +*/ + +int _ma_insert(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, + uchar *key, uchar *anc_buff, uchar *key_pos, uchar *key_buff, + uchar *father_buff, uchar *father_key_pos, my_off_t father_page, + my_bool insert_last) +{ + uint a_length,nod_flag; + int t_length; + uchar *endpos, *prev_key; + MARIA_KEY_PARAM s_temp; + DBUG_ENTER("_ma_insert"); + DBUG_PRINT("enter",("key_pos: %lx",key_pos)); + DBUG_EXECUTE("key", _ma_print_key(DBUG_FILE,keyinfo->seg,key, + USE_WHOLE_KEY);); + + nod_flag=_ma_test_if_nod(anc_buff); + a_length=maria_getint(anc_buff); + endpos= anc_buff+ a_length; + prev_key=(key_pos == anc_buff+2+nod_flag ? (uchar*) 0 : key_buff); + t_length=(*keyinfo->pack_key)(keyinfo,nod_flag, + (key_pos == endpos ? (uchar*) 0 : key_pos), + prev_key, prev_key, + key,&s_temp); +#ifndef DBUG_OFF + if (key_pos != anc_buff+2+nod_flag && (keyinfo->flag & + (HA_BINARY_PACK_KEY | HA_PACK_KEY))) + { + DBUG_DUMP("prev_key",(byte*) key_buff, _ma_keylength(keyinfo,key_buff)); + } + if (keyinfo->flag & HA_PACK_KEY) + { + DBUG_PRINT("test",("t_length: %d ref_len: %d", + t_length,s_temp.ref_length)); + DBUG_PRINT("test",("n_ref_len: %d n_length: %d key_pos: %lx", + s_temp.n_ref_length,s_temp.n_length,s_temp.key)); + } +#endif + if (t_length > 0) + { + if (t_length >= keyinfo->maxlength*2+MAX_POINTER_LENGTH) + { + maria_print_error(info->s, HA_ERR_CRASHED); + my_errno=HA_ERR_CRASHED; + DBUG_RETURN(-1); + } + bmove_upp((byte*) endpos+t_length,(byte*) endpos,(uint) (endpos-key_pos)); + } + else + { + if (-t_length >= keyinfo->maxlength*2+MAX_POINTER_LENGTH) + { + maria_print_error(info->s, HA_ERR_CRASHED); + my_errno=HA_ERR_CRASHED; + DBUG_RETURN(-1); + } + bmove(key_pos,key_pos-t_length,(uint) (endpos-key_pos)+t_length); + } + (*keyinfo->store_key)(keyinfo,key_pos,&s_temp); + a_length+=t_length; + maria_putint(anc_buff,a_length,nod_flag); + if (a_length <= keyinfo->block_length) + { + if (keyinfo->block_length - a_length < 32 && + keyinfo->flag & HA_FULLTEXT && key_pos == endpos && + info->s->base.key_reflength <= info->s->base.rec_reflength && + info->s->options & (HA_OPTION_PACK_RECORD | HA_OPTION_COMPRESS_RECORD)) + { + /* + Normal word. One-level tree. Page is almost full. + Let's consider converting. + We'll compare 'key' and the first key at anc_buff + */ + uchar *a=key, *b=anc_buff+2+nod_flag; + uint alen, blen, ft2len=info->s->ft2_keyinfo.keylength; + /* the very first key on the page is always unpacked */ + DBUG_ASSERT((*b & 128) == 0); +#if HA_FT_MAXLEN >= 127 + blen= mi_uint2korr(b); b+=2; +#else + blen= *b++; +#endif + get_key_length(alen,a); + DBUG_ASSERT(info->ft1_to_ft2==0); + if (alen == blen && + ha_compare_text(keyinfo->seg->charset, a, alen, b, blen, 0, 0)==0) + { + /* yup. converting */ + info->ft1_to_ft2=(DYNAMIC_ARRAY *) + my_malloc(sizeof(DYNAMIC_ARRAY), MYF(MY_WME)); + my_init_dynamic_array(info->ft1_to_ft2, ft2len, 300, 50); + + /* + now, adding all keys from the page to dynarray + if the page is a leaf (if not keys will be deleted later) + */ + if (!nod_flag) + { + /* let's leave the first key on the page, though, because + we cannot easily dispatch an empty page here */ + b+=blen+ft2len+2; + for (a=anc_buff+a_length ; b < a ; b+=ft2len+2) + insert_dynamic(info->ft1_to_ft2, (char*) b); + + /* fixing the page's length - it contains only one key now */ + maria_putint(anc_buff,2+blen+ft2len+2,0); + } + /* the rest will be done when we're back from recursion */ + } + } + DBUG_RETURN(0); /* There is room on page */ + } + /* Page is full */ + if (nod_flag) + insert_last=0; + if (!(keyinfo->flag & (HA_VAR_LENGTH_KEY | HA_BINARY_PACK_KEY)) && + father_buff && !insert_last) + DBUG_RETURN(_ma_balance_page(info,keyinfo,key,anc_buff,father_buff, + father_key_pos,father_page)); + DBUG_RETURN(_ma_split_page(info,keyinfo,key,anc_buff,key_buff, insert_last)); +} /* _ma_insert */ + + + /* split a full page in two and assign emerging item to key */ + +int _ma_split_page(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, + uchar *key, uchar *buff, uchar *key_buff, + my_bool insert_last_key) +{ + uint length,a_length,key_ref_length,t_length,nod_flag,key_length; + uchar *key_pos,*pos, *after_key; + my_off_t new_pos; + MARIA_KEY_PARAM s_temp; + DBUG_ENTER("maria_split_page"); + DBUG_DUMP("buff",(byte*) buff,maria_getint(buff)); + + if (info->s->keyinfo+info->lastinx == keyinfo) + info->page_changed=1; /* Info->buff is used */ + info->buff_used=1; + nod_flag=_ma_test_if_nod(buff); + key_ref_length=2+nod_flag; + if (insert_last_key) + key_pos= _ma_find_last_pos(keyinfo,buff,key_buff, &key_length, &after_key); + else + key_pos= _ma_find_half_pos(nod_flag,keyinfo,buff,key_buff, &key_length, + &after_key); + if (!key_pos) + DBUG_RETURN(-1); + + length=(uint) (key_pos-buff); + a_length=maria_getint(buff); + maria_putint(buff,length,nod_flag); + + key_pos=after_key; + if (nod_flag) + { + DBUG_PRINT("test",("Splitting nod")); + pos=key_pos-nod_flag; + memcpy((byte*) info->buff+2,(byte*) pos,(size_t) nod_flag); + } + + /* Move middle item to key and pointer to new page */ + if ((new_pos= _ma_new(info,keyinfo,DFLT_INIT_HITS)) == HA_OFFSET_ERROR) + DBUG_RETURN(-1); + _ma_kpointer(info, _ma_move_key(keyinfo,key,key_buff),new_pos); + + /* Store new page */ + if (!(*keyinfo->get_key)(keyinfo,nod_flag,&key_pos,key_buff)) + DBUG_RETURN(-1); + + t_length=(*keyinfo->pack_key)(keyinfo,nod_flag,(uchar *) 0, + (uchar*) 0, (uchar*) 0, + key_buff, &s_temp); + length=(uint) ((buff+a_length)-key_pos); + memcpy((byte*) info->buff+key_ref_length+t_length,(byte*) key_pos, + (size_t) length); + (*keyinfo->store_key)(keyinfo,info->buff+key_ref_length,&s_temp); + maria_putint(info->buff,length+t_length+key_ref_length,nod_flag); + + if (_ma_write_keypage(info,keyinfo,new_pos,DFLT_INIT_HITS,info->buff)) + DBUG_RETURN(-1); + DBUG_DUMP("key",(byte*) key, _ma_keylength(keyinfo,key)); + DBUG_RETURN(2); /* Middle key up */ +} /* _ma_split_page */ + + + /* + Calculate how to much to move to split a page in two + Returns pointer to start of key. + key will contain the key. + return_key_length will contain the length of key + after_key will contain the position to where the next key starts + */ + +uchar *_ma_find_half_pos(uint nod_flag, MARIA_KEYDEF *keyinfo, uchar *page, + uchar *key, uint *return_key_length, + uchar **after_key) +{ + uint keys,length,key_ref_length; + uchar *end,*lastpos; + DBUG_ENTER("_ma_find_half_pos"); + + key_ref_length=2+nod_flag; + length=maria_getint(page)-key_ref_length; + page+=key_ref_length; + if (!(keyinfo->flag & + (HA_PACK_KEY | HA_SPACE_PACK_USED | HA_VAR_LENGTH_KEY | + HA_BINARY_PACK_KEY))) + { + key_ref_length=keyinfo->keylength+nod_flag; + keys=length/(key_ref_length*2); + *return_key_length=keyinfo->keylength; + end=page+keys*key_ref_length; + *after_key=end+key_ref_length; + memcpy(key,end,key_ref_length); + DBUG_RETURN(end); + } + + end=page+length/2-key_ref_length; /* This is aprox. half */ + *key='\0'; + do + { + lastpos=page; + if (!(length=(*keyinfo->get_key)(keyinfo,nod_flag,&page,key))) + DBUG_RETURN(0); + } while (page < end); + *return_key_length=length; + *after_key=page; + DBUG_PRINT("exit",("returns: %lx page: %lx half: %lx",lastpos,page,end)); + DBUG_RETURN(lastpos); +} /* _ma_find_half_pos */ + + + /* + Split buffer at last key + Returns pointer to the start of the key before the last key + key will contain the last key + */ + +static uchar *_ma_find_last_pos(MARIA_KEYDEF *keyinfo, uchar *page, + uchar *key, uint *return_key_length, + uchar **after_key) +{ + uint keys,length,last_length,key_ref_length; + uchar *end,*lastpos,*prevpos; + uchar key_buff[HA_MAX_KEY_BUFF]; + DBUG_ENTER("_ma_find_last_pos"); + + key_ref_length=2; + length=maria_getint(page)-key_ref_length; + page+=key_ref_length; + if (!(keyinfo->flag & + (HA_PACK_KEY | HA_SPACE_PACK_USED | HA_VAR_LENGTH_KEY | + HA_BINARY_PACK_KEY))) + { + keys=length/keyinfo->keylength-2; + *return_key_length=length=keyinfo->keylength; + end=page+keys*length; + *after_key=end+length; + memcpy(key,end,length); + DBUG_RETURN(end); + } + + LINT_INIT(prevpos); + LINT_INIT(last_length); + end=page+length-key_ref_length; + *key='\0'; + length=0; + lastpos=page; + while (page < end) + { + prevpos=lastpos; lastpos=page; + last_length=length; + memcpy(key, key_buff, length); /* previous key */ + if (!(length=(*keyinfo->get_key)(keyinfo,0,&page,key_buff))) + { + maria_print_error(keyinfo->share, HA_ERR_CRASHED); + my_errno=HA_ERR_CRASHED; + DBUG_RETURN(0); + } + } + *return_key_length=last_length; + *after_key=lastpos; + DBUG_PRINT("exit",("returns: %lx page: %lx end: %lx",prevpos,page,end)); + DBUG_RETURN(prevpos); +} /* _ma_find_last_pos */ + + + /* Balance page with not packed keys with page on right/left */ + /* returns 0 if balance was done */ + +static int _ma_balance_page(register MARIA_HA *info, MARIA_KEYDEF *keyinfo, + uchar *key, uchar *curr_buff, uchar *father_buff, + uchar *father_key_pos, my_off_t father_page) +{ + my_bool right; + uint k_length,father_length,father_keylength,nod_flag,curr_keylength, + right_length,left_length,new_right_length,new_left_length,extra_length, + length,keys; + uchar *pos,*buff,*extra_buff; + my_off_t next_page,new_pos; + byte tmp_part_key[HA_MAX_KEY_BUFF]; + DBUG_ENTER("_ma_balance_page"); + + k_length=keyinfo->keylength; + father_length=maria_getint(father_buff); + father_keylength=k_length+info->s->base.key_reflength; + nod_flag=_ma_test_if_nod(curr_buff); + curr_keylength=k_length+nod_flag; + info->page_changed=1; + + if ((father_key_pos != father_buff+father_length && + (info->state->records & 1)) || + father_key_pos == father_buff+2+info->s->base.key_reflength) + { + right=1; + next_page= _ma_kpos(info->s->base.key_reflength, + father_key_pos+father_keylength); + buff=info->buff; + DBUG_PRINT("test",("use right page: %lu",next_page)); + } + else + { + right=0; + father_key_pos-=father_keylength; + next_page= _ma_kpos(info->s->base.key_reflength,father_key_pos); + /* Fix that curr_buff is to left */ + buff=curr_buff; curr_buff=info->buff; + DBUG_PRINT("test",("use left page: %lu",next_page)); + } /* father_key_pos ptr to parting key */ + + if (!_ma_fetch_keypage(info,keyinfo,next_page,DFLT_INIT_HITS,info->buff,0)) + goto err; + DBUG_DUMP("next",(byte*) info->buff,maria_getint(info->buff)); + + /* Test if there is room to share keys */ + + left_length=maria_getint(curr_buff); + right_length=maria_getint(buff); + keys=(left_length+right_length-4-nod_flag*2)/curr_keylength; + + if ((right ? right_length : left_length) + curr_keylength <= + keyinfo->block_length) + { /* Merge buffs */ + new_left_length=2+nod_flag+(keys/2)*curr_keylength; + new_right_length=2+nod_flag+((keys+1)/2)*curr_keylength; + maria_putint(curr_buff,new_left_length,nod_flag); + maria_putint(buff,new_right_length,nod_flag); + + if (left_length < new_left_length) + { /* Move keys buff -> leaf */ + pos=curr_buff+left_length; + memcpy((byte*) pos,(byte*) father_key_pos, (size_t) k_length); + memcpy((byte*) pos+k_length, (byte*) buff+2, + (size_t) (length=new_left_length - left_length - k_length)); + pos=buff+2+length; + memcpy((byte*) father_key_pos,(byte*) pos,(size_t) k_length); + bmove((byte*) buff+2,(byte*) pos+k_length,new_right_length); + } + else + { /* Move keys -> buff */ + + bmove_upp((byte*) buff+new_right_length,(byte*) buff+right_length, + right_length-2); + length=new_right_length-right_length-k_length; + memcpy((byte*) buff+2+length,father_key_pos,(size_t) k_length); + pos=curr_buff+new_left_length; + memcpy((byte*) father_key_pos,(byte*) pos,(size_t) k_length); + memcpy((byte*) buff+2,(byte*) pos+k_length,(size_t) length); + } + + if (_ma_write_keypage(info,keyinfo,next_page,DFLT_INIT_HITS,info->buff) || + _ma_write_keypage(info,keyinfo,father_page,DFLT_INIT_HITS,father_buff)) + goto err; + DBUG_RETURN(0); + } + + /* curr_buff[] and buff[] are full, lets split and make new nod */ + + extra_buff=info->buff+info->s->base.max_key_block_length; + new_left_length=new_right_length=2+nod_flag+(keys+1)/3*curr_keylength; + if (keys == 5) /* Too few keys to balance */ + new_left_length-=curr_keylength; + extra_length=nod_flag+left_length+right_length- + new_left_length-new_right_length-curr_keylength; + DBUG_PRINT("info",("left_length: %d right_length: %d new_left_length: %d new_right_length: %d extra_length: %d", + left_length, right_length, + new_left_length, new_right_length, + extra_length)); + maria_putint(curr_buff,new_left_length,nod_flag); + maria_putint(buff,new_right_length,nod_flag); + maria_putint(extra_buff,extra_length+2,nod_flag); + + /* move first largest keys to new page */ + pos=buff+right_length-extra_length; + memcpy((byte*) extra_buff+2,pos,(size_t) extra_length); + /* Save new parting key */ + memcpy(tmp_part_key, pos-k_length,k_length); + /* Make place for new keys */ + bmove_upp((byte*) buff+new_right_length,(byte*) pos-k_length, + right_length-extra_length-k_length-2); + /* Copy keys from left page */ + pos= curr_buff+new_left_length; + memcpy((byte*) buff+2,(byte*) pos+k_length, + (size_t) (length=left_length-new_left_length-k_length)); + /* Copy old parting key */ + memcpy((byte*) buff+2+length,father_key_pos,(size_t) k_length); + + /* Move new parting keys up to caller */ + memcpy((byte*) (right ? key : father_key_pos),pos,(size_t) k_length); + memcpy((byte*) (right ? father_key_pos : key),tmp_part_key, k_length); + + if ((new_pos= _ma_new(info,keyinfo,DFLT_INIT_HITS)) == HA_OFFSET_ERROR) + goto err; + _ma_kpointer(info,key+k_length,new_pos); + if (_ma_write_keypage(info,keyinfo,(right ? new_pos : next_page), + DFLT_INIT_HITS,info->buff) || + _ma_write_keypage(info,keyinfo,(right ? next_page : new_pos), + DFLT_INIT_HITS,extra_buff)) + goto err; + + DBUG_RETURN(1); /* Middle key up */ + +err: + DBUG_RETURN(-1); +} /* _ma_balance_page */ + +/********************************************************************** + * Bulk insert code * + **********************************************************************/ + +typedef struct { + MARIA_HA *info; + uint keynr; +} bulk_insert_param; + +int _ma_ck_write_tree(register MARIA_HA *info, uint keynr, uchar *key, + uint key_length) +{ + int error; + DBUG_ENTER("_ma_ck_write_tree"); + + error= tree_insert(&info->bulk_insert[keynr], key, + key_length + info->s->rec_reflength, + info->bulk_insert[keynr].custom_arg) ? 0 : HA_ERR_OUT_OF_MEM ; + + DBUG_RETURN(error); +} /* _ma_ck_write_tree */ + + +/* typeof(_ma_keys_compare)=qsort_cmp2 */ + +static int keys_compare(bulk_insert_param *param, uchar *key1, uchar *key2) +{ + uint not_used[2]; + return ha_key_cmp(param->info->s->keyinfo[param->keynr].seg, + key1, key2, USE_WHOLE_KEY, SEARCH_SAME, + not_used); +} + + +static int keys_free(uchar *key, TREE_FREE mode, bulk_insert_param *param) +{ + /* + Probably I can use info->lastkey here, but I'm not sure, + and to be safe I'd better use local lastkey. + */ + uchar lastkey[HA_MAX_KEY_BUFF]; + uint keylen; + MARIA_KEYDEF *keyinfo; + + switch (mode) { + case free_init: + if (param->info->s->concurrent_insert) + { + rw_wrlock(¶m->info->s->key_root_lock[param->keynr]); + param->info->s->keyinfo[param->keynr].version++; + } + return 0; + case free_free: + keyinfo=param->info->s->keyinfo+param->keynr; + keylen= _ma_keylength(keyinfo, key); + memcpy(lastkey, key, keylen); + return _ma_ck_write_btree(param->info,param->keynr,lastkey, + keylen - param->info->s->rec_reflength); + case free_end: + if (param->info->s->concurrent_insert) + rw_unlock(¶m->info->s->key_root_lock[param->keynr]); + return 0; + } + return -1; +} + + +int maria_init_bulk_insert(MARIA_HA *info, ulong cache_size, ha_rows rows) +{ + MARIA_SHARE *share=info->s; + MARIA_KEYDEF *key=share->keyinfo; + bulk_insert_param *params; + uint i, num_keys, total_keylength; + ulonglong key_map; + DBUG_ENTER("_ma_init_bulk_insert"); + DBUG_PRINT("enter",("cache_size: %lu", cache_size)); + + DBUG_ASSERT(!info->bulk_insert && + (!rows || rows >= MARIA_MIN_ROWS_TO_USE_BULK_INSERT)); + + maria_clear_all_keys_active(key_map); + for (i=total_keylength=num_keys=0 ; i < share->base.keys ; i++) + { + if (! (key[i].flag & HA_NOSAME) && (share->base.auto_key != i + 1) && + maria_is_key_active(share->state.key_map, i)) + { + num_keys++; + maria_set_key_active(key_map, i); + total_keylength+=key[i].maxlength+TREE_ELEMENT_EXTRA_SIZE; + } + } + + if (num_keys==0 || + num_keys * MARIA_MIN_SIZE_BULK_INSERT_TREE > cache_size) + DBUG_RETURN(0); + + if (rows && rows*total_keylength < cache_size) + cache_size=rows; + else + cache_size/=total_keylength*16; + + info->bulk_insert=(TREE *) + my_malloc((sizeof(TREE)*share->base.keys+ + sizeof(bulk_insert_param)*num_keys),MYF(0)); + + if (!info->bulk_insert) + DBUG_RETURN(HA_ERR_OUT_OF_MEM); + + params=(bulk_insert_param *)(info->bulk_insert+share->base.keys); + for (i=0 ; i < share->base.keys ; i++) + { + if (maria_is_key_active(key_map, i)) + { + params->info=info; + params->keynr=i; + /* Only allocate a 16'th of the buffer at a time */ + init_tree(&info->bulk_insert[i], + cache_size * key[i].maxlength, + cache_size * key[i].maxlength, 0, + (qsort_cmp2)keys_compare, 0, + (tree_element_free) keys_free, (void *)params++); + } + else + info->bulk_insert[i].root=0; + } + + DBUG_RETURN(0); +} + +void maria_flush_bulk_insert(MARIA_HA *info, uint inx) +{ + if (info->bulk_insert) + { + if (is_tree_inited(&info->bulk_insert[inx])) + reset_tree(&info->bulk_insert[inx]); + } +} + +void maria_end_bulk_insert(MARIA_HA *info) +{ + if (info->bulk_insert) + { + uint i; + for (i=0 ; i < info->s->base.keys ; i++) + { + if (is_tree_inited(& info->bulk_insert[i])) + { + delete_tree(& info->bulk_insert[i]); + } + } + my_free((void *)info->bulk_insert, MYF(0)); + info->bulk_insert=0; + } +} diff --git a/storage/maria/maria_chk.c b/storage/maria/maria_chk.c new file mode 100644 index 00000000000..de76344a800 --- /dev/null +++ b/storage/maria/maria_chk.c @@ -0,0 +1,1824 @@ +/* Copyright (C) 2006-2003 MySQL AB + + 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; either version 2 of the License, or + (at your option) any later version. + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ + +/* Describe, check and repair of MARIA tables */ + +#include "ma_fulltext.h" +#include +#include +#include +#include +#ifdef HAVE_SYS_VADVICE_H +#include +#endif +#ifdef HAVE_SYS_MMAN_H +#include +#endif +SET_STACK_SIZE(9000) /* Minimum stack size for program */ + +#ifndef USE_RAID +#define my_raid_create(A,B,C,D,E,F,G) my_create(A,B,C,G) +#define my_raid_delete(A,B,C) my_delete(A,B) +#endif + +#ifdef OS2 +#define _sanity(a,b) +#endif + +static uint decode_bits; +static char **default_argv; +static const char *load_default_groups[]= { "mariachk", 0 }; +static const char *set_collation_name, *opt_tmpdir; +static CHARSET_INFO *set_collation; +static long opt_maria_block_size; +static long opt_key_cache_block_size; +static const char *my_progname_short; +static int stopwords_inited= 0; +static MY_TMPDIR mariachk_tmpdir; + +static const char *type_names[]= +{ "impossible","char","binary", "short", "long", "float", + "double","number","unsigned short", + "unsigned long","longlong","ulonglong","int24", + "uint24","int8","varchar", "varbin","?", + "?"}; + +static const char *prefix_packed_txt="packed ", + *bin_packed_txt="prefix ", + *diff_txt="stripped ", + *null_txt="NULL", + *blob_txt="BLOB "; + +static const char *field_pack[]= +{"","no endspace", "no prespace", + "no zeros", "blob", "constant", "table-lockup", + "always zero","varchar","unique-hash","?","?"}; + +static const char *maria_stats_method_str="nulls_unequal"; + +static void get_options(int *argc,char * * *argv); +static void print_version(void); +static void usage(void); +static int mariachk(HA_CHECK *param, char *filename); +static void descript(HA_CHECK *param, register MARIA_HA *info, my_string name); +static int maria_sort_records(HA_CHECK *param, register MARIA_HA *info, + my_string name, uint sort_key, + my_bool write_info, my_bool update_index); +static int sort_record_index(MARIA_SORT_PARAM *sort_param, MARIA_HA *info, + MARIA_KEYDEF *keyinfo, + my_off_t page,uchar *buff,uint sortkey, + File new_file, my_bool update_index); + +HA_CHECK check_param; + + /* Main program */ + +int main(int argc, char **argv) +{ + int error; + MY_INIT(argv[0]); + my_progname_short= my_progname+dirname_length(my_progname); + +#ifdef __EMX__ + _wildcard (&argc, &argv); +#endif + + mariachk_init(&check_param); + check_param.opt_lock_memory= 1; /* Lock memory if possible */ + check_param.using_global_keycache = 0; + get_options(&argc,(char***) &argv); + maria_quick_table_bits=decode_bits; + error=0; + maria_init(); + + while (--argc >= 0) + { + int new_error=mariachk(&check_param, *(argv++)); + if ((check_param.testflag & T_REP_ANY) != T_REP) + check_param.testflag&= ~T_REP; + VOID(fflush(stdout)); + VOID(fflush(stderr)); + if ((check_param.error_printed | check_param.warning_printed) && + (check_param.testflag & T_FORCE_CREATE) && + (!(check_param.testflag & (T_REP | T_REP_BY_SORT | T_SORT_RECORDS | + T_SORT_INDEX)))) + { + uint old_testflag=check_param.testflag; + if (!(check_param.testflag & T_REP)) + check_param.testflag|= T_REP_BY_SORT; + check_param.testflag&= ~T_EXTEND; /* Don't needed */ + error|=mariachk(&check_param, argv[-1]); + check_param.testflag= old_testflag; + VOID(fflush(stdout)); + VOID(fflush(stderr)); + } + else + error|=new_error; + if (argc && (!(check_param.testflag & T_SILENT) || check_param.testflag & T_INFO)) + { + puts("\n---------\n"); + VOID(fflush(stdout)); + } + } + if (check_param.total_files > 1) + { /* Only if descript */ + char buff[22],buff2[22]; + if (!(check_param.testflag & T_SILENT) || check_param.testflag & T_INFO) + puts("\n---------\n"); + printf("\nTotal of all %d MARIA-files:\nData records: %9s Deleted blocks: %9s\n",check_param.total_files,llstr(check_param.total_records,buff), + llstr(check_param.total_deleted,buff2)); + } + free_defaults(default_argv); + free_tmpdir(&mariachk_tmpdir); + maria_end(); + my_end(check_param.testflag & T_INFO ? + MY_CHECK_ERROR | MY_GIVE_INFO : MY_CHECK_ERROR); + exit(error); +#ifndef _lint + return 0; /* No compiler warning */ +#endif +} /* main */ + +enum options_mc { + OPT_CHARSETS_DIR=256, OPT_SET_COLLATION,OPT_START_CHECK_POS, + OPT_CORRECT_CHECKSUM, OPT_KEY_BUFFER_SIZE, + OPT_KEY_CACHE_BLOCK_SIZE, OPT_MARIA_BLOCK_SIZE, + OPT_READ_BUFFER_SIZE, OPT_WRITE_BUFFER_SIZE, OPT_SORT_BUFFER_SIZE, + OPT_SORT_KEY_BLOCKS, OPT_DECODE_BITS, OPT_FT_MIN_WORD_LEN, + OPT_FT_MAX_WORD_LEN, OPT_FT_STOPWORD_FILE, + OPT_MAX_RECORD_LENGTH, OPT_AUTO_CLOSE, OPT_STATS_METHOD +}; + +static struct my_option my_long_options[] = +{ + {"analyze", 'a', + "Analyze distribution of keys. Will make some joins in MySQL faster. You can check the calculated distribution.", + 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, +#ifdef __NETWARE__ + {"autoclose", OPT_AUTO_CLOSE, "Auto close the screen on exit for Netware.", + 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, +#endif + {"block-search", 'b', + "No help available.", + 0, 0, 0, GET_ULONG, REQUIRED_ARG, 0, 0, 0, 0, 0, 0}, + {"backup", 'B', + "Make a backup of the .MYD file as 'filename-time.BAK'.", + 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, + {"character-sets-dir", OPT_CHARSETS_DIR, + "Directory where character sets are.", + (gptr*) &charsets_dir, 0, 0, GET_STR, REQUIRED_ARG, 0, 0, 0, 0, 0, 0}, + {"check", 'c', + "Check table for errors.", + 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, + {"check-only-changed", 'C', + "Check only tables that have changed since last check. It also applies to other requested actions (e.g. --analyze will be ignored if the table is already analyzed).", + 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, + {"correct-checksum", OPT_CORRECT_CHECKSUM, + "Correct checksum information for table.", + 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, +#ifndef DBUG_OFF + {"debug", '#', + "Output debug log. Often this is 'd:t:o,filename'.", + 0, 0, 0, GET_STR, OPT_ARG, 0, 0, 0, 0, 0, 0}, +#endif + {"description", 'd', + "Prints some information about table.", + 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, + {"data-file-length", 'D', + "Max length of data file (when recreating data-file when it's full).", + (gptr*) &check_param.max_data_file_length, + (gptr*) &check_param.max_data_file_length, + 0, GET_LL, REQUIRED_ARG, 0, 0, 0, 0, 0, 0}, + {"extend-check", 'e', + "If used when checking a table, ensure that the table is 100 percent consistent, which will take a long time. If used when repairing a table, try to recover every possible row from the data file. Normally this will also find a lot of garbage rows; Don't use this option with repair if you are not totally desperate.", + 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, + {"fast", 'F', + "Check only tables that haven't been closed properly. It also applies to other requested actions (e.g. --analyze will be ignored if the table is already analyzed).", + 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, + {"force", 'f', + "Restart with -r if there are any errors in the table. States will be updated as with --update-state.", + 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, + {"HELP", 'H', + "Display this help and exit.", + 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, + {"help", '?', + "Display this help and exit.", + 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, + {"information", 'i', + "Print statistics information about table that is checked.", + 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, + {"keys-used", 'k', + "Tell MARIA to update only some specific keys. # is a bit mask of which keys to use. This can be used to get faster inserts.", + (gptr*) &check_param.keys_in_use, + (gptr*) &check_param.keys_in_use, + 0, GET_ULL, REQUIRED_ARG, -1, 0, 0, 0, 0, 0}, + {"max-record-length", OPT_MAX_RECORD_LENGTH, + "Skip rows bigger than this if mariachk can't allocate memory to hold it", + (gptr*) &check_param.max_record_length, + (gptr*) &check_param.max_record_length, + 0, GET_ULL, REQUIRED_ARG, LONGLONG_MAX, 0, LONGLONG_MAX, 0, 0, 0}, + {"medium-check", 'm', + "Faster than extend-check, but only finds 99.99% of all errors. Should be good enough for most cases.", + 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, + {"quick", 'q', "Faster repair by not modifying the data file.", + 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, + {"read-only", 'T', + "Don't mark table as checked.", + 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, + {"recover", 'r', + "Can fix almost anything except unique keys that aren't unique.", + 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, + {"parallel-recover", 'p', + "Same as '-r' but creates all the keys in parallel.", + 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, + {"safe-recover", 'o', + "Uses old recovery method; Slower than '-r' but can handle a couple of cases where '-r' reports that it can't fix the data file.", + 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, + {"sort-recover", 'n', + "Force recovering with sorting even if the temporary file was very big.", + 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, +#ifdef DEBUG + {"start-check-pos", OPT_START_CHECK_POS, + "No help available.", + 0, 0, 0, GET_ULL, REQUIRED_ARG, 0, 0, 0, 0, 0, 0}, +#endif + {"set-auto-increment", 'A', + "Force auto_increment to start at this or higher value. If no value is given, then sets the next auto_increment value to the highest used value for the auto key + 1.", + (gptr*) &check_param.auto_increment_value, + (gptr*) &check_param.auto_increment_value, + 0, GET_ULL, OPT_ARG, 0, 0, 0, 0, 0, 0}, + {"set-collation", OPT_SET_COLLATION, + "Change the collation used by the index", + (gptr*) &set_collation_name, 0, 0, GET_STR, REQUIRED_ARG, 0, 0, 0, 0, 0, 0}, + {"set-variable", 'O', + "Change the value of a variable. Please note that this option is deprecated; you can set variables directly with --variable-name=value.", + 0, 0, 0, GET_STR, REQUIRED_ARG, 0, 0, 0, 0, 0, 0}, + {"silent", 's', + "Only print errors. One can use two -s to make mariachk very silent.", + 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, + {"sort-index", 'S', + "Sort index blocks. This speeds up 'read-next' in applications.", + 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, + {"sort-records", 'R', + "Sort records according to an index. This makes your data much more localized and may speed up things. (It may be VERY slow to do a sort the first time!)", + (gptr*) &check_param.opt_sort_key, + (gptr*) &check_param.opt_sort_key, + 0, GET_UINT, REQUIRED_ARG, 0, 0, 0, 0, 0, 0}, + {"tmpdir", 't', + "Path for temporary files.", + (gptr*) &opt_tmpdir, + 0, 0, GET_STR, REQUIRED_ARG, 0, 0, 0, 0, 0, 0}, + {"update-state", 'U', + "Mark tables as crashed if any errors were found.", + 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, + {"unpack", 'u', + "Unpack file packed with mariapack.", + 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, + {"verbose", 'v', + "Print more information. This can be used with --description and --check. Use many -v for more verbosity!", + 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, + {"version", 'V', + "Print version and exit.", + 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, + {"wait", 'w', + "Wait if table is locked.", + 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, + { "key_buffer_size", OPT_KEY_BUFFER_SIZE, "", + (gptr*) &check_param.use_buffers, (gptr*) &check_param.use_buffers, 0, + GET_ULONG, REQUIRED_ARG, (long) USE_BUFFER_INIT, (long) MALLOC_OVERHEAD, + (long) ~0L, (long) MALLOC_OVERHEAD, (long) IO_SIZE, 0}, + { "key_cache_block_size", OPT_KEY_CACHE_BLOCK_SIZE, "", + (gptr*) &opt_key_cache_block_size, + (gptr*) &opt_key_cache_block_size, 0, + GET_LONG, REQUIRED_ARG, MARIA_KEY_BLOCK_LENGTH, MARIA_MIN_KEY_BLOCK_LENGTH, + MARIA_MAX_KEY_BLOCK_LENGTH, 0, MARIA_MIN_KEY_BLOCK_LENGTH, 0}, + { "maria_block_size", OPT_MARIA_BLOCK_SIZE, "", + (gptr*) &opt_maria_block_size, (gptr*) &opt_maria_block_size, 0, + GET_LONG, REQUIRED_ARG, MARIA_KEY_BLOCK_LENGTH, MARIA_MIN_KEY_BLOCK_LENGTH, + MARIA_MAX_KEY_BLOCK_LENGTH, 0, MARIA_MIN_KEY_BLOCK_LENGTH, 0}, + { "read_buffer_size", OPT_READ_BUFFER_SIZE, "", + (gptr*) &check_param.read_buffer_length, + (gptr*) &check_param.read_buffer_length, 0, GET_ULONG, REQUIRED_ARG, + (long) READ_BUFFER_INIT, (long) MALLOC_OVERHEAD, + (long) ~0L, (long) MALLOC_OVERHEAD, (long) 1L, 0}, + { "write_buffer_size", OPT_WRITE_BUFFER_SIZE, "", + (gptr*) &check_param.write_buffer_length, + (gptr*) &check_param.write_buffer_length, 0, GET_ULONG, REQUIRED_ARG, + (long) READ_BUFFER_INIT, (long) MALLOC_OVERHEAD, + (long) ~0L, (long) MALLOC_OVERHEAD, (long) 1L, 0}, + { "sort_buffer_size", OPT_SORT_BUFFER_SIZE, "", + (gptr*) &check_param.sort_buffer_length, + (gptr*) &check_param.sort_buffer_length, 0, GET_ULONG, REQUIRED_ARG, + (long) SORT_BUFFER_INIT, (long) (MIN_SORT_BUFFER + MALLOC_OVERHEAD), + (long) ~0L, (long) MALLOC_OVERHEAD, (long) 1L, 0}, + { "sort_key_blocks", OPT_SORT_KEY_BLOCKS, "", + (gptr*) &check_param.sort_key_blocks, + (gptr*) &check_param.sort_key_blocks, 0, GET_ULONG, REQUIRED_ARG, + BUFFERS_WHEN_SORTING, 4L, 100L, 0L, 1L, 0}, + { "decode_bits", OPT_DECODE_BITS, "", (gptr*) &decode_bits, + (gptr*) &decode_bits, 0, GET_UINT, REQUIRED_ARG, 9L, 4L, 17L, 0L, 1L, 0}, + { "ft_min_word_len", OPT_FT_MIN_WORD_LEN, "", (gptr*) &ft_min_word_len, + (gptr*) &ft_min_word_len, 0, GET_ULONG, REQUIRED_ARG, 4, 1, HA_FT_MAXCHARLEN, + 0, 1, 0}, + { "ft_max_word_len", OPT_FT_MAX_WORD_LEN, "", (gptr*) &ft_max_word_len, + (gptr*) &ft_max_word_len, 0, GET_ULONG, REQUIRED_ARG, HA_FT_MAXCHARLEN, 10, + HA_FT_MAXCHARLEN, 0, 1, 0}, + { "maria_ft_stopword_file", OPT_FT_STOPWORD_FILE, + "Use stopwords from this file instead of built-in list.", + (gptr*) &ft_stopword_file, (gptr*) &ft_stopword_file, 0, GET_STR, + REQUIRED_ARG, 0, 0, 0, 0, 0, 0}, + {"stats_method", OPT_STATS_METHOD, + "Specifies how index statistics collection code should threat NULLs. " + "Possible values of name are \"nulls_unequal\" (default behavior for 4.1/5.0), " + "\"nulls_equal\" (emulate 4.0 behavior), and \"nulls_ignored\".", + (gptr*) &maria_stats_method_str, (gptr*) &maria_stats_method_str, 0, + GET_STR, REQUIRED_ARG, 0, 0, 0, 0, 0, 0}, + { 0, 0, 0, 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0} +}; + + +#include + +static void print_version(void) +{ + printf("%s Ver 2.7 for %s at %s\n", my_progname, SYSTEM_TYPE, + MACHINE_TYPE); + NETWARE_SET_SCREEN_MODE(1); +} + + +static void usage(void) +{ + print_version(); + puts("By Monty, for your professional use"); + puts("This software comes with NO WARRANTY: see the PUBLIC for details.\n"); + puts("Description, check and repair of MARIA tables."); + puts("Used without options all tables on the command will be checked for errors"); + printf("Usage: %s [OPTIONS] tables[.MYI]\n", my_progname_short); + printf("\nGlobal options:\n"); +#ifndef DBUG_OFF + printf("\ + -#, --debug=... Output debug log. Often this is 'd:t:o,filename'.\n"); +#endif + printf("\ + -?, --help Display this help and exit.\n\ + -O, --set-variable var=option.\n\ + Change the value of a variable. Please note that\n\ + this option is deprecated; you can set variables\n\ + directly with '--variable-name=value'.\n\ + -t, --tmpdir=path Path for temporary files. Multiple paths can be\n\ + specified, separated by "); +#if defined( __WIN__) || defined(OS2) || defined(__NETWARE__) + printf("semicolon (;)"); +#else + printf("colon (:)"); +#endif + printf(", they will be used\n\ + in a round-robin fashion.\n\ + -s, --silent Only print errors. One can use two -s to make\n\ + mariachk very silent.\n\ + -v, --verbose Print more information. This can be used with\n\ + --description and --check. Use many -v for more verbosity.\n\ + -V, --version Print version and exit.\n\ + -w, --wait Wait if table is locked.\n\n"); +#ifdef DEBUG + puts(" --start-check-pos=# Start reading file at given offset.\n"); +#endif + + puts("Check options (check is the default action for mariachk):\n\ + -c, --check Check table for errors.\n\ + -e, --extend-check Check the table VERY throughly. Only use this in\n\ + extreme cases as mariachk should normally be able to\n\ + find out if the table is ok even without this switch.\n\ + -F, --fast Check only tables that haven't been closed properly.\n\ + -C, --check-only-changed\n\ + Check only tables that have changed since last check.\n\ + -f, --force Restart with '-r' if there are any errors in the table.\n\ + States will be updated as with '--update-state'.\n\ + -i, --information Print statistics information about table that is checked.\n\ + -m, --medium-check Faster than extend-check, but only finds 99.99% of\n\ + all errors. Should be good enough for most cases.\n\ + -U --update-state Mark tables as crashed if you find any errors.\n\ + -T, --read-only Don't mark table as checked.\n"); + + puts("Repair options (When using '-r' or '-o'):\n\ + -B, --backup Make a backup of the .MYD file as 'filename-time.BAK'.\n\ + --correct-checksum Correct checksum information for table.\n\ + -D, --data-file-length=# Max length of data file (when recreating data\n\ + file when it's full).\n\ + -e, --extend-check Try to recover every possible row from the data file\n\ + Normally this will also find a lot of garbage rows;\n\ + Don't use this option if you are not totally desperate.\n\ + -f, --force Overwrite old temporary files.\n\ + -k, --keys-used=# Tell MARIA to update only some specific keys. # is a\n\ + bit mask of which keys to use. This can be used to\n\ + get faster inserts.\n\ + --max-record-length=#\n\ + Skip rows bigger than this if mariachk can't allocate\n\ + memory to hold it.\n\ + -r, --recover Can fix almost anything except unique keys that aren't\n\ + unique.\n\ + -n, --sort-recover Forces recovering with sorting even if the temporary\n\ + file would be very big.\n\ + -p, --parallel-recover\n\ + Uses the same technique as '-r' and '-n', but creates\n\ + all the keys in parallel, in different threads.\n\ + -o, --safe-recover Uses old recovery method; Slower than '-r' but can\n\ + handle a couple of cases where '-r' reports that it\n\ + can't fix the data file.\n\ + --character-sets-dir=...\n\ + Directory where character sets are.\n\ + --set-collation=name\n\ + Change the collation used by the index.\n\ + -q, --quick Faster repair by not modifying the data file.\n\ + One can give a second '-q' to force mariachk to\n\ + modify the original datafile in case of duplicate keys.\n\ + NOTE: Tables where the data file is currupted can't be\n\ + fixed with this option.\n\ + -u, --unpack Unpack file packed with mariapack.\n\ +"); + + puts("Other actions:\n\ + -a, --analyze Analyze distribution of keys. Will make some joins in\n\ + MySQL faster. You can check the calculated distribution\n\ + by using '--description --verbose table_name'.\n\ + --stats_method=name Specifies how index statistics collection code should\n\ + threat NULLs. Possible values of name are \"nulls_unequal\"\n\ + (default for 4.1/5.0), \"nulls_equal\" (emulate 4.0), and \n\ + \"nulls_ignored\".\n\ + -d, --description Prints some information about table.\n\ + -A, --set-auto-increment[=value]\n\ + Force auto_increment to start at this or higher value\n\ + If no value is given, then sets the next auto_increment\n\ + value to the highest used value for the auto key + 1.\n\ + -S, --sort-index Sort index blocks. This speeds up 'read-next' in\n\ + applications.\n\ + -R, --sort-records=#\n\ + Sort records according to an index. This makes your\n\ + data much more localized and may speed up things\n\ + (It may be VERY slow to do a sort the first time!).\n\ + -b, --block-search=#\n\ + Find a record, a block at given offset belongs to."); + + print_defaults("my", load_default_groups); + my_print_variables(my_long_options); +} + +#include + +const char *maria_stats_method_names[] = {"nulls_unequal", "nulls_equal", + "nulls_ignored", NullS}; +TYPELIB maria_stats_method_typelib= { + array_elements(maria_stats_method_names) - 1, "", + maria_stats_method_names, NULL}; + + /* Read options */ + +static my_bool +get_one_option(int optid, + const struct my_option *opt __attribute__((unused)), + char *argument) +{ + switch (optid) { +#ifdef __NETWARE__ + case OPT_AUTO_CLOSE: + setscreenmode(SCR_AUTOCLOSE_ON_EXIT); + break; +#endif + case 'a': + if (argument == disabled_my_option) + check_param.testflag&= ~T_STATISTICS; + else + check_param.testflag|= T_STATISTICS; + break; + case 'A': + if (argument) + check_param.auto_increment_value= strtoull(argument, NULL, 0); + else + check_param.auto_increment_value= 0; /* Set to max used value */ + check_param.testflag|= T_AUTO_INC; + break; + case 'b': + check_param.search_after_block= strtoul(argument, NULL, 10); + break; + case 'B': + if (argument == disabled_my_option) + check_param.testflag&= ~T_BACKUP_DATA; + else + check_param.testflag|= T_BACKUP_DATA; + break; + case 'c': + if (argument == disabled_my_option) + check_param.testflag&= ~T_CHECK; + else + check_param.testflag|= T_CHECK; + break; + case 'C': + if (argument == disabled_my_option) + check_param.testflag&= ~(T_CHECK | T_CHECK_ONLY_CHANGED); + else + check_param.testflag|= T_CHECK | T_CHECK_ONLY_CHANGED; + break; + case 'D': + check_param.max_data_file_length=strtoll(argument, NULL, 10); + break; + case 's': /* silent */ + if (argument == disabled_my_option) + check_param.testflag&= ~(T_SILENT | T_VERY_SILENT); + else + { + if (check_param.testflag & T_SILENT) + check_param.testflag|= T_VERY_SILENT; + check_param.testflag|= T_SILENT; + check_param.testflag&= ~T_WRITE_LOOP; + } + break; + case 'w': + if (argument == disabled_my_option) + check_param.testflag&= ~T_WAIT_FOREVER; + else + check_param.testflag|= T_WAIT_FOREVER; + break; + case 'd': /* description if isam-file */ + if (argument == disabled_my_option) + check_param.testflag&= ~T_DESCRIPT; + else + check_param.testflag|= T_DESCRIPT; + break; + case 'e': /* extend check */ + if (argument == disabled_my_option) + check_param.testflag&= ~T_EXTEND; + else + check_param.testflag|= T_EXTEND; + break; + case 'i': + if (argument == disabled_my_option) + check_param.testflag&= ~T_INFO; + else + check_param.testflag|= T_INFO; + break; + case 'f': + if (argument == disabled_my_option) + { + check_param.tmpfile_createflag= O_RDWR | O_TRUNC | O_EXCL; + check_param.testflag&= ~(T_FORCE_CREATE | T_UPDATE_STATE); + } + else + { + check_param.tmpfile_createflag= O_RDWR | O_TRUNC; + check_param.testflag|= T_FORCE_CREATE | T_UPDATE_STATE; + } + break; + case 'F': + if (argument == disabled_my_option) + check_param.testflag&= ~T_FAST; + else + check_param.testflag|= T_FAST; + break; + case 'k': + check_param.keys_in_use= (ulonglong) strtoll(argument, NULL, 10); + break; + case 'm': + if (argument == disabled_my_option) + check_param.testflag&= ~T_MEDIUM; + else + check_param.testflag|= T_MEDIUM; /* Medium check */ + break; + case 'r': /* Repair table */ + check_param.testflag&= ~T_REP_ANY; + if (argument != disabled_my_option) + check_param.testflag|= T_REP_BY_SORT; + break; + case 'p': + check_param.testflag&= ~T_REP_ANY; + if (argument != disabled_my_option) + check_param.testflag|= T_REP_PARALLEL; + break; + case 'o': + check_param.testflag&= ~T_REP_ANY; + check_param.force_sort= 0; + if (argument != disabled_my_option) + { + check_param.testflag|= T_REP; + my_disable_async_io= 1; /* More safety */ + } + break; + case 'n': + check_param.testflag&= ~T_REP_ANY; + if (argument == disabled_my_option) + check_param.force_sort= 0; + else + { + check_param.testflag|= T_REP_BY_SORT; + check_param.force_sort= 1; + } + break; + case 'q': + if (argument == disabled_my_option) + check_param.testflag&= ~(T_QUICK | T_FORCE_UNIQUENESS); + else + check_param.testflag|= + (check_param.testflag & T_QUICK) ? T_FORCE_UNIQUENESS : T_QUICK; + break; + case 'u': + if (argument == disabled_my_option) + check_param.testflag&= ~(T_UNPACK | T_REP_BY_SORT); + else + check_param.testflag|= T_UNPACK | T_REP_BY_SORT; + break; + case 'v': /* Verbose */ + if (argument == disabled_my_option) + { + check_param.testflag&= ~T_VERBOSE; + check_param.verbose=0; + } + else + { + check_param.testflag|= T_VERBOSE; + check_param.verbose++; + } + break; + case 'R': /* Sort records */ + if (argument == disabled_my_option) + check_param.testflag&= ~T_SORT_RECORDS; + else + { + check_param.testflag|= T_SORT_RECORDS; + check_param.opt_sort_key= (uint) atoi(argument) - 1; + if (check_param.opt_sort_key >= MARIA_MAX_KEY) + { + fprintf(stderr, + "The value of the sort key is bigger than max key: %d.\n", + MARIA_MAX_KEY); + exit(1); + } + } + break; + case 'S': /* Sort index */ + if (argument == disabled_my_option) + check_param.testflag&= ~T_SORT_INDEX; + else + check_param.testflag|= T_SORT_INDEX; + break; + case 'T': + if (argument == disabled_my_option) + check_param.testflag&= ~T_READONLY; + else + check_param.testflag|= T_READONLY; + break; + case 'U': + if (argument == disabled_my_option) + check_param.testflag&= ~T_UPDATE_STATE; + else + check_param.testflag|= T_UPDATE_STATE; + break; + case '#': + if (argument == disabled_my_option) + { + DBUG_POP(); + } + else + { + DBUG_PUSH(argument ? argument : "d:t:o,/tmp/mariachk.trace"); + } + break; + case 'V': + print_version(); + exit(0); + case OPT_CORRECT_CHECKSUM: + if (argument == disabled_my_option) + check_param.testflag&= ~T_CALC_CHECKSUM; + else + check_param.testflag|= T_CALC_CHECKSUM; + break; + case OPT_STATS_METHOD: + { + int method; + enum_handler_stats_method method_conv; + maria_stats_method_str= argument; + if ((method=find_type(argument, &maria_stats_method_typelib, 2)) <= 0) + { + fprintf(stderr, "Invalid value of stats_method: %s.\n", argument); + exit(1); + } + switch (method-1) { + case 0: + method_conv= MI_STATS_METHOD_NULLS_EQUAL; + break; + case 1: + method_conv= MI_STATS_METHOD_NULLS_NOT_EQUAL; + break; + case 2: + method_conv= MI_STATS_METHOD_IGNORE_NULLS; + break; + } + check_param.stats_method= method_conv; + break; + } +#ifdef DEBUG /* Only useful if debugging */ + case OPT_START_CHECK_POS: + check_param.start_check_pos= strtoull(argument, NULL, 0); + break; +#endif + case 'H': + my_print_help(my_long_options); + exit(0); + case '?': + usage(); + exit(0); + } + return 0; +} + + +static void get_options(register int *argc,register char ***argv) +{ + int ho_error; + + load_defaults("my", load_default_groups, argc, argv); + default_argv= *argv; + if (isatty(fileno(stdout))) + check_param.testflag|=T_WRITE_LOOP; + + if ((ho_error=handle_options(argc, argv, my_long_options, get_one_option))) + exit(ho_error); + + /* If using repair, then update checksum if one uses --update-state */ + if ((check_param.testflag & T_UPDATE_STATE) && + (check_param.testflag & T_REP_ANY)) + check_param.testflag|= T_CALC_CHECKSUM; + + if (*argc == 0) + { + usage(); + exit(-1); + } + + if ((check_param.testflag & T_UNPACK) && + (check_param.testflag & (T_QUICK | T_SORT_RECORDS))) + { + VOID(fprintf(stderr, + "%s: --unpack can't be used with --quick or --sort-records\n", + my_progname_short)); + exit(1); + } + if ((check_param.testflag & T_READONLY) && + (check_param.testflag & + (T_REP_ANY | T_STATISTICS | T_AUTO_INC | + T_SORT_RECORDS | T_SORT_INDEX | T_FORCE_CREATE))) + { + VOID(fprintf(stderr, + "%s: Can't use --readonly when repairing or sorting\n", + my_progname_short)); + exit(1); + } + + if (init_tmpdir(&mariachk_tmpdir, opt_tmpdir)) + exit(1); + + check_param.tmpdir=&mariachk_tmpdir; + check_param.key_cache_block_size= opt_key_cache_block_size; + + if (set_collation_name) + if (!(set_collation= get_charset_by_name(set_collation_name, + MYF(MY_WME)))) + exit(1); + + maria_block_size=(uint) 1 << my_bit_log2(opt_maria_block_size); + return; +} /* get options */ + + + /* Check table */ + +static int mariachk(HA_CHECK *param, my_string filename) +{ + int error,lock_type,recreate; + int rep_quick= param->testflag & (T_QUICK | T_FORCE_UNIQUENESS); + uint raid_chunks; + MARIA_HA *info; + File datafile; + char llbuff[22],llbuff2[22]; + my_bool state_updated=0; + MARIA_SHARE *share; + DBUG_ENTER("mariachk"); + + param->out_flag=error=param->warning_printed=param->error_printed= + recreate=0; + datafile=0; + param->isam_file_name=filename; /* For error messages */ + if (!(info=maria_open(filename, + (param->testflag & (T_DESCRIPT | T_READONLY)) ? + O_RDONLY : O_RDWR, + HA_OPEN_FOR_REPAIR | + ((param->testflag & T_WAIT_FOREVER) ? + HA_OPEN_WAIT_IF_LOCKED : + (param->testflag & T_DESCRIPT) ? + HA_OPEN_IGNORE_IF_LOCKED : HA_OPEN_ABORT_IF_LOCKED)))) + { + /* Avoid twice printing of isam file name */ + param->error_printed=1; + switch (my_errno) { + case HA_ERR_CRASHED: + _ma_check_print_error(param,"'%s' doesn't have a correct index definition. You need to recreate it before you can do a repair",filename); + break; + case HA_ERR_NOT_A_TABLE: + _ma_check_print_error(param,"'%s' is not a MARIA-table",filename); + break; + case HA_ERR_CRASHED_ON_USAGE: + _ma_check_print_error(param,"'%s' is marked as crashed",filename); + break; + case HA_ERR_CRASHED_ON_REPAIR: + _ma_check_print_error(param,"'%s' is marked as crashed after last repair",filename); + break; + case HA_ERR_OLD_FILE: + _ma_check_print_error(param,"'%s' is a old type of MARIA-table", filename); + break; + case HA_ERR_END_OF_FILE: + _ma_check_print_error(param,"Couldn't read complete header from '%s'", filename); + break; + case EAGAIN: + _ma_check_print_error(param,"'%s' is locked. Use -w to wait until unlocked",filename); + break; + case ENOENT: + _ma_check_print_error(param,"File '%s' doesn't exist",filename); + break; + case EACCES: + _ma_check_print_error(param,"You don't have permission to use '%s'",filename); + break; + default: + _ma_check_print_error(param,"%d when opening MARIA-table '%s'", + my_errno,filename); + break; + } + DBUG_RETURN(1); + } + share=info->s; + share->options&= ~HA_OPTION_READ_ONLY_DATA; /* We are modifing it */ + share->tot_locks-= share->r_locks; + share->r_locks=0; + raid_chunks=share->base.raid_chunks; + + /* + Skip the checking of the file if: + We are using --fast and the table is closed properly + We are using --check-only-changed-tables and the table hasn't changed + */ + if (param->testflag & (T_FAST | T_CHECK_ONLY_CHANGED)) + { + my_bool need_to_check= maria_is_crashed(info) || share->state.open_count != 0; + + if ((param->testflag & (T_REP_ANY | T_SORT_RECORDS)) && + ((share->state.changed & (STATE_CHANGED | STATE_CRASHED | + STATE_CRASHED_ON_REPAIR) || + !(param->testflag & T_CHECK_ONLY_CHANGED)))) + need_to_check=1; + + if (info->s->base.keys && info->state->records) + { + if ((param->testflag & T_STATISTICS) && + (share->state.changed & STATE_NOT_ANALYZED)) + need_to_check=1; + if ((param->testflag & T_SORT_INDEX) && + (share->state.changed & STATE_NOT_SORTED_PAGES)) + need_to_check=1; + if ((param->testflag & T_REP_BY_SORT) && + (share->state.changed & STATE_NOT_OPTIMIZED_KEYS)) + need_to_check=1; + } + if ((param->testflag & T_CHECK_ONLY_CHANGED) && + (share->state.changed & (STATE_CHANGED | STATE_CRASHED | + STATE_CRASHED_ON_REPAIR))) + need_to_check=1; + if (!need_to_check) + { + if (!(param->testflag & T_SILENT) || param->testflag & T_INFO) + printf("MARIA file: %s is already checked\n",filename); + if (maria_close(info)) + { + _ma_check_print_error(param,"%d when closing MARIA-table '%s'", + my_errno,filename); + DBUG_RETURN(1); + } + DBUG_RETURN(0); + } + } + if ((param->testflag & (T_REP_ANY | T_STATISTICS | + T_SORT_RECORDS | T_SORT_INDEX)) && + (((param->testflag & T_UNPACK) && + share->data_file_type == COMPRESSED_RECORD) || + mi_uint2korr(share->state.header.state_info_length) != + MARIA_STATE_INFO_SIZE || + mi_uint2korr(share->state.header.base_info_length) != + MARIA_BASE_INFO_SIZE || + maria_is_any_intersect_keys_active(param->keys_in_use, share->base.keys, + ~share->state.key_map) || + maria_test_if_almost_full(info) || + info->s->state.header.file_version[3] != maria_file_magic[3] || + (set_collation && + set_collation->number != share->state.header.language) || + maria_block_size != MARIA_KEY_BLOCK_LENGTH)) + { + if (set_collation) + param->language= set_collation->number; + if (maria_recreate_table(param, &info,filename)) + { + VOID(fprintf(stderr, + "MARIA-table '%s' is not fixed because of errors\n", + filename)); + return(-1); + } + recreate=1; + if (!(param->testflag & T_REP_ANY)) + { + param->testflag|=T_REP_BY_SORT; /* if only STATISTICS */ + if (!(param->testflag & T_SILENT)) + printf("- '%s' has old table-format. Recreating index\n",filename); + rep_quick|=T_QUICK; + } + share=info->s; + share->tot_locks-= share->r_locks; + share->r_locks=0; + } + + if (param->testflag & T_DESCRIPT) + { + param->total_files++; + param->total_records+=info->state->records; + param->total_deleted+=info->state->del; + descript(param, info, filename); + } + else + { + if (!stopwords_inited++) + ft_init_stopwords(); + + if (!(param->testflag & T_READONLY)) + lock_type = F_WRLCK; /* table is changed */ + else + lock_type= F_RDLCK; + if (info->lock_type == F_RDLCK) + info->lock_type=F_UNLCK; /* Read only table */ + if (_ma_readinfo(info,lock_type,0)) + { + _ma_check_print_error(param,"Can't lock indexfile of '%s', error: %d", + filename,my_errno); + param->error_printed=0; + goto end2; + } + /* + _ma_readinfo() has locked the table. + We mark the table as locked (without doing file locks) to be able to + use functions that only works on locked tables (like row caching). + */ + maria_lock_database(info, F_EXTRA_LCK); + datafile=info->dfile; + + if (param->testflag & (T_REP_ANY | T_SORT_RECORDS | T_SORT_INDEX)) + { + if (param->testflag & T_REP_ANY) + { + ulonglong tmp=share->state.key_map; + maria_copy_keys_active(share->state.key_map, share->base.keys, + param->keys_in_use); + if (tmp != share->state.key_map) + info->update|=HA_STATE_CHANGED; + } + if (rep_quick && maria_chk_del(param, info, param->testflag & ~T_VERBOSE)) + { + if (param->testflag & T_FORCE_CREATE) + { + rep_quick=0; + _ma_check_print_info(param,"Creating new data file\n"); + } + else + { + error=1; + _ma_check_print_error(param, + "Quick-recover aborted; Run recovery without switch 'q'"); + } + } + if (!error) + { + if ((param->testflag & (T_REP_BY_SORT | T_REP_PARALLEL)) && + (maria_is_any_key_active(share->state.key_map) || + (rep_quick && !param->keys_in_use && !recreate)) && + maria_test_if_sort_rep(info, info->state->records, + info->s->state.key_map, + param->force_sort)) + { + if (param->testflag & T_REP_BY_SORT) + error=maria_repair_by_sort(param,info,filename,rep_quick); + else + error=maria_repair_parallel(param,info,filename,rep_quick); + state_updated=1; + } + else if (param->testflag & T_REP_ANY) + error=maria_repair(param, info,filename,rep_quick); + } + if (!error && param->testflag & T_SORT_RECORDS) + { + /* + The data file is nowadays reopened in the repair code so we should + soon remove the following reopen-code + */ +#ifndef TO_BE_REMOVED + if (param->out_flag & O_NEW_DATA) + { /* Change temp file to org file */ + VOID(my_close(info->dfile,MYF(MY_WME))); /* Close new file */ + error|=maria_change_to_newfile(filename,MARIA_NAME_DEXT,DATA_TMP_EXT, + raid_chunks, + MYF(0)); + if (_ma_open_datafile(info,info->s, -1)) + error=1; + param->out_flag&= ~O_NEW_DATA; /* We are using new datafile */ + param->read_cache.file=info->dfile; + } +#endif + if (! error) + { + uint key; + /* + We can't update the index in maria_sort_records if we have a + prefix compressed or fulltext index + */ + my_bool update_index=1; + for (key=0 ; key < share->base.keys; key++) + if (share->keyinfo[key].flag & (HA_BINARY_PACK_KEY|HA_FULLTEXT)) + update_index=0; + + error=maria_sort_records(param,info,filename,param->opt_sort_key, + /* what is the following parameter for ? */ + (my_bool) !(param->testflag & T_REP), + update_index); + datafile=info->dfile; /* This is now locked */ + if (!error && !update_index) + { + if (param->verbose) + puts("Table had a compressed index; We must now recreate the index"); + error=maria_repair_by_sort(param,info,filename,1); + } + } + } + if (!error && param->testflag & T_SORT_INDEX) + error=maria_sort_index(param,info,filename); + if (!error) + share->state.changed&= ~(STATE_CHANGED | STATE_CRASHED | + STATE_CRASHED_ON_REPAIR); + else + maria_mark_crashed(info); + } + else if ((param->testflag & T_CHECK) || !(param->testflag & T_AUTO_INC)) + { + if (!(param->testflag & T_SILENT) || param->testflag & T_INFO) + printf("Checking MARIA file: %s\n",filename); + if (!(param->testflag & T_SILENT)) + printf("Data records: %7s Deleted blocks: %7s\n", + llstr(info->state->records,llbuff), + llstr(info->state->del,llbuff2)); + error =maria_chk_status(param,info); + maria_intersect_keys_active(share->state.key_map, param->keys_in_use); + error =maria_chk_size(param,info); + if (!error || !(param->testflag & (T_FAST | T_FORCE_CREATE))) + error|=maria_chk_del(param, info,param->testflag); + if ((!error || (!(param->testflag & (T_FAST | T_FORCE_CREATE)) && + !param->start_check_pos))) + { + error|=maria_chk_key(param, info); + if (!error && (param->testflag & (T_STATISTICS | T_AUTO_INC))) + error=maria_update_state_info(param, info, + ((param->testflag & T_STATISTICS) ? + UPDATE_STAT : 0) | + ((param->testflag & T_AUTO_INC) ? + UPDATE_AUTO_INC : 0)); + } + if ((!rep_quick && !error) || + !(param->testflag & (T_FAST | T_FORCE_CREATE))) + { + if (param->testflag & (T_EXTEND | T_MEDIUM)) + VOID(init_key_cache(maria_key_cache,opt_key_cache_block_size, + param->use_buffers, 0, 0)); + VOID(init_io_cache(¶m->read_cache,datafile, + (uint) param->read_buffer_length, + READ_CACHE, + (param->start_check_pos ? + param->start_check_pos : + share->pack.header_length), + 1, + MYF(MY_WME))); + maria_lock_memory(param); + if ((info->s->options & (HA_OPTION_PACK_RECORD | + HA_OPTION_COMPRESS_RECORD)) || + (param->testflag & (T_EXTEND | T_MEDIUM))) + error|=maria_chk_data_link(param, info, param->testflag & T_EXTEND); + error|=_ma_flush_blocks(param, share->key_cache, share->kfile); + VOID(end_io_cache(¶m->read_cache)); + } + if (!error) + { + if ((share->state.changed & STATE_CHANGED) && + (param->testflag & T_UPDATE_STATE)) + info->update|=HA_STATE_CHANGED | HA_STATE_ROW_CHANGED; + share->state.changed&= ~(STATE_CHANGED | STATE_CRASHED | + STATE_CRASHED_ON_REPAIR); + } + else if (!maria_is_crashed(info) && + (param->testflag & T_UPDATE_STATE)) + { /* Mark crashed */ + maria_mark_crashed(info); + info->update|=HA_STATE_CHANGED | HA_STATE_ROW_CHANGED; + } + } + } + if ((param->testflag & T_AUTO_INC) || + ((param->testflag & T_REP_ANY) && info->s->base.auto_key)) + _ma_update_auto_increment_key(param, info, + (my_bool) !test(param->testflag & T_AUTO_INC)); + + if (!(param->testflag & T_DESCRIPT)) + { + if (info->update & HA_STATE_CHANGED && ! (param->testflag & T_READONLY)) + error|=maria_update_state_info(param, info, + UPDATE_OPEN_COUNT | + (((param->testflag & T_REP_ANY) ? + UPDATE_TIME : 0) | + (state_updated ? UPDATE_STAT : 0) | + ((param->testflag & T_SORT_RECORDS) ? + UPDATE_SORT : 0))); + VOID(maria_lock_file(param, share->kfile,0L,F_UNLCK,"indexfile",filename)); + info->update&= ~HA_STATE_CHANGED; + } + maria_lock_database(info, F_UNLCK); +end2: + if (maria_close(info)) + { + _ma_check_print_error(param,"%d when closing MARIA-table '%s'",my_errno,filename); + DBUG_RETURN(1); + } + if (error == 0) + { + if (param->out_flag & O_NEW_DATA) + error|=maria_change_to_newfile(filename,MARIA_NAME_DEXT,DATA_TMP_EXT, + raid_chunks, + ((param->testflag & T_BACKUP_DATA) ? + MYF(MY_REDEL_MAKE_BACKUP) : MYF(0))); + if (param->out_flag & O_NEW_INDEX) + error|=maria_change_to_newfile(filename,MARIA_NAME_IEXT,INDEX_TMP_EXT,0, + MYF(0)); + } + VOID(fflush(stdout)); VOID(fflush(stderr)); + if (param->error_printed) + { + if (param->testflag & (T_REP_ANY | T_SORT_RECORDS | T_SORT_INDEX)) + { + VOID(fprintf(stderr, + "MARIA-table '%s' is not fixed because of errors\n", + filename)); + if (param->testflag & T_REP_ANY) + VOID(fprintf(stderr, + "Try fixing it by using the --safe-recover (-o), the --force (-f) option or by not using the --quick (-q) flag\n")); + } + else if (!(param->error_printed & 2) && + !(param->testflag & T_FORCE_CREATE)) + VOID(fprintf(stderr, + "MARIA-table '%s' is corrupted\nFix it using switch \"-r\" or \"-o\"\n", + filename)); + } + else if (param->warning_printed && + ! (param->testflag & (T_REP_ANY | T_SORT_RECORDS | T_SORT_INDEX | + T_FORCE_CREATE))) + VOID(fprintf(stderr, "MARIA-table '%s' is usable but should be fixed\n", + filename)); + VOID(fflush(stderr)); + DBUG_RETURN(error); +} /* mariachk */ + + + /* Write info about table */ + +static void descript(HA_CHECK *param, register MARIA_HA *info, my_string name) +{ + uint key,keyseg_nr,field,start; + reg3 MARIA_KEYDEF *keyinfo; + reg2 HA_KEYSEG *keyseg; + reg4 const char *text; + char buff[160],length[10],*pos,*end; + enum en_fieldtype type; + MARIA_SHARE *share=info->s; + char llbuff[22],llbuff2[22]; + DBUG_ENTER("describe"); + + printf("\nMARIA file: %s\n",name); + fputs("Record format: ",stdout); + if (share->options & HA_OPTION_COMPRESS_RECORD) + puts("Compressed"); + else if (share->options & HA_OPTION_PACK_RECORD) + puts("Packed"); + else + puts("Fixed length"); + printf("Character set: %s (%d)\n", + get_charset_name(share->state.header.language), + share->state.header.language); + + if (param->testflag & T_VERBOSE) + { + printf("File-version: %d\n", + (int) share->state.header.file_version[3]); + if (share->state.create_time) + { + get_date(buff,1,share->state.create_time); + printf("Creation time: %s\n",buff); + } + if (share->state.check_time) + { + get_date(buff,1,share->state.check_time); + printf("Recover time: %s\n",buff); + } + pos=buff; + if (share->state.changed & STATE_CRASHED) + strmov(buff,"crashed"); + else + { + if (share->state.open_count) + pos=strmov(pos,"open,"); + if (share->state.changed & STATE_CHANGED) + pos=strmov(pos,"changed,"); + else + pos=strmov(pos,"checked,"); + if (!(share->state.changed & STATE_NOT_ANALYZED)) + pos=strmov(pos,"analyzed,"); + if (!(share->state.changed & STATE_NOT_OPTIMIZED_KEYS)) + pos=strmov(pos,"optimized keys,"); + if (!(share->state.changed & STATE_NOT_SORTED_PAGES)) + pos=strmov(pos,"sorted index pages,"); + pos[-1]=0; /* Remove extra ',' */ + } + printf("Status: %s\n",buff); + if (share->base.auto_key) + { + printf("Auto increment key: %13d Last value: %13s\n", + share->base.auto_key, + llstr(share->state.auto_increment,llbuff)); + } + if (share->base.raid_type) + { + printf("RAID: Type: %u Chunks: %u Chunksize: %lu\n", + share->base.raid_type, + share->base.raid_chunks, + share->base.raid_chunksize); + } + if (share->options & (HA_OPTION_CHECKSUM | HA_OPTION_COMPRESS_RECORD)) + printf("Checksum: %23s\n",llstr(info->state->checksum,llbuff)); +; + if (share->options & HA_OPTION_DELAY_KEY_WRITE) + printf("Keys are only flushed at close\n"); + + } + printf("Data records: %13s Deleted blocks: %13s\n", + llstr(info->state->records,llbuff),llstr(info->state->del,llbuff2)); + if (param->testflag & T_SILENT) + DBUG_VOID_RETURN; /* This is enough */ + + if (param->testflag & T_VERBOSE) + { +#ifdef USE_RELOC + printf("Init-relocation: %13s\n",llstr(share->base.reloc,llbuff)); +#endif + printf("Datafile parts: %13s Deleted data: %13s\n", + llstr(share->state.split,llbuff), + llstr(info->state->empty,llbuff2)); + printf("Datafile pointer (bytes):%9d Keyfile pointer (bytes):%9d\n", + share->rec_reflength,share->base.key_reflength); + printf("Datafile length: %13s Keyfile length: %13s\n", + llstr(info->state->data_file_length,llbuff), + llstr(info->state->key_file_length,llbuff2)); + + if (info->s->base.reloc == 1L && info->s->base.records == 1L) + puts("This is a one-record table"); + else + { + if (share->base.max_data_file_length != HA_OFFSET_ERROR || + share->base.max_key_file_length != HA_OFFSET_ERROR) + printf("Max datafile length: %13s Max keyfile length: %13s\n", + llstr(share->base.max_data_file_length-1,llbuff), + llstr(share->base.max_key_file_length-1,llbuff2)); + } + } + + printf("Recordlength: %13d\n",(int) share->base.pack_reclength); + if (! maria_is_all_keys_active(share->state.key_map, share->base.keys)) + { + longlong2str(share->state.key_map,buff,2); + printf("Using only keys '%s' of %d possibly keys\n", + buff, share->base.keys); + } + puts("\ntable description:"); + printf("Key Start Len Index Type"); + if (param->testflag & T_VERBOSE) + printf(" Rec/key Root Blocksize"); + VOID(putchar('\n')); + + for (key=keyseg_nr=0, keyinfo= &share->keyinfo[0] ; + key < share->base.keys; + key++,keyinfo++) + { + keyseg=keyinfo->seg; + if (keyinfo->flag & HA_NOSAME) text="unique "; + else if (keyinfo->flag & HA_FULLTEXT) text="fulltext "; + else text="multip."; + + pos=buff; + if (keyseg->flag & HA_REVERSE_SORT) + *pos++ = '-'; + pos=strmov(pos,type_names[keyseg->type]); + *pos++ = ' '; + *pos=0; + if (keyinfo->flag & HA_PACK_KEY) + pos=strmov(pos,prefix_packed_txt); + if (keyinfo->flag & HA_BINARY_PACK_KEY) + pos=strmov(pos,bin_packed_txt); + if (keyseg->flag & HA_SPACE_PACK) + pos=strmov(pos,diff_txt); + if (keyseg->flag & HA_BLOB_PART) + pos=strmov(pos,blob_txt); + if (keyseg->flag & HA_NULL_PART) + pos=strmov(pos,null_txt); + *pos=0; + + printf("%-4d%-6ld%-3d %-8s%-21s", + key+1,(long) keyseg->start+1,keyseg->length,text,buff); + if (share->state.key_root[key] != HA_OFFSET_ERROR) + llstr(share->state.key_root[key],buff); + else + buff[0]=0; + if (param->testflag & T_VERBOSE) + printf("%11lu %12s %10d", + share->state.rec_per_key_part[keyseg_nr++], + buff,keyinfo->block_length); + VOID(putchar('\n')); + while ((++keyseg)->type != HA_KEYTYPE_END) + { + pos=buff; + if (keyseg->flag & HA_REVERSE_SORT) + *pos++ = '-'; + pos=strmov(pos,type_names[keyseg->type]); + *pos++= ' '; + if (keyseg->flag & HA_SPACE_PACK) + pos=strmov(pos,diff_txt); + if (keyseg->flag & HA_BLOB_PART) + pos=strmov(pos,blob_txt); + if (keyseg->flag & HA_NULL_PART) + pos=strmov(pos,null_txt); + *pos=0; + printf(" %-6ld%-3d %-21s", + (long) keyseg->start+1,keyseg->length,buff); + if (param->testflag & T_VERBOSE) + printf("%11lu", share->state.rec_per_key_part[keyseg_nr++]); + VOID(putchar('\n')); + } + keyseg++; + } + if (share->state.header.uniques) + { + MARIA_UNIQUEDEF *uniqueinfo; + puts("\nUnique Key Start Len Nullpos Nullbit Type"); + for (key=0,uniqueinfo= &share->uniqueinfo[0] ; + key < share->state.header.uniques; key++, uniqueinfo++) + { + my_bool new_row=0; + char null_bit[8],null_pos[8]; + printf("%-8d%-5d",key+1,uniqueinfo->key+1); + for (keyseg=uniqueinfo->seg ; keyseg->type != HA_KEYTYPE_END ; keyseg++) + { + if (new_row) + fputs(" ",stdout); + null_bit[0]=null_pos[0]=0; + if (keyseg->null_bit) + { + sprintf(null_bit,"%d",keyseg->null_bit); + sprintf(null_pos,"%ld",(long) keyseg->null_pos+1); + } + printf("%-7ld%-5d%-9s%-10s%-30s\n", + (long) keyseg->start+1,keyseg->length, + null_pos,null_bit, + type_names[keyseg->type]); + new_row=1; + } + } + } + if (param->verbose > 1) + { + char null_bit[8],null_pos[8]; + printf("\nField Start Length Nullpos Nullbit Type"); + if (share->options & HA_OPTION_COMPRESS_RECORD) + printf(" Huff tree Bits"); + VOID(putchar('\n')); + start=1; + for (field=0 ; field < share->base.fields ; field++) + { + if (share->options & HA_OPTION_COMPRESS_RECORD) + type=share->rec[field].base_type; + else + type=(enum en_fieldtype) share->rec[field].type; + end=strmov(buff,field_pack[type]); + if (share->options & HA_OPTION_COMPRESS_RECORD) + { + if (share->rec[field].pack_type & PACK_TYPE_SELECTED) + end=strmov(end,", not_always"); + if (share->rec[field].pack_type & PACK_TYPE_SPACE_FIELDS) + end=strmov(end,", no empty"); + if (share->rec[field].pack_type & PACK_TYPE_ZERO_FILL) + { + sprintf(end,", zerofill(%d)",share->rec[field].space_length_bits); + end=strend(end); + } + } + if (buff[0] == ',') + strmov(buff,buff+2); + int10_to_str((long) share->rec[field].length,length,10); + null_bit[0]=null_pos[0]=0; + if (share->rec[field].null_bit) + { + sprintf(null_bit,"%d",share->rec[field].null_bit); + sprintf(null_pos,"%d",share->rec[field].null_pos+1); + } + printf("%-6d%-6d%-7s%-8s%-8s%-35s",field+1,start,length, + null_pos, null_bit, buff); + if (share->options & HA_OPTION_COMPRESS_RECORD) + { + if (share->rec[field].huff_tree) + printf("%3d %2d", + (uint) (share->rec[field].huff_tree-share->decode_trees)+1, + share->rec[field].huff_tree->quick_table_bits); + } + VOID(putchar('\n')); + start+=share->rec[field].length; + } + } + DBUG_VOID_RETURN; +} /* describe */ + + + /* Sort records according to one key */ + +static int maria_sort_records(HA_CHECK *param, + register MARIA_HA *info, my_string name, + uint sort_key, + my_bool write_info, + my_bool update_index) +{ + int got_error; + uint key; + MARIA_KEYDEF *keyinfo; + File new_file; + uchar *temp_buff; + ha_rows old_record_count; + MARIA_SHARE *share=info->s; + char llbuff[22],llbuff2[22]; + MARIA_SORT_INFO sort_info; + MARIA_SORT_PARAM sort_param; + DBUG_ENTER("sort_records"); + + bzero((char*)&sort_info,sizeof(sort_info)); + bzero((char*)&sort_param,sizeof(sort_param)); + sort_param.sort_info=&sort_info; + sort_info.param=param; + keyinfo= &share->keyinfo[sort_key]; + got_error=1; + temp_buff=0; + new_file= -1; + + if (! maria_is_key_active(share->state.key_map, sort_key)) + { + _ma_check_print_warning(param, + "Can't sort table '%s' on key %d; No such key", + name,sort_key+1); + param->error_printed=0; + DBUG_RETURN(0); /* Nothing to do */ + } + if (keyinfo->flag & HA_FULLTEXT) + { + _ma_check_print_warning(param,"Can't sort table '%s' on FULLTEXT key %d", + name,sort_key+1); + param->error_printed=0; + DBUG_RETURN(0); /* Nothing to do */ + } + if (share->data_file_type == COMPRESSED_RECORD) + { + _ma_check_print_warning(param,"Can't sort read-only table '%s'", name); + param->error_printed=0; + DBUG_RETURN(0); /* Nothing to do */ + } + if (!(param->testflag & T_SILENT)) + { + printf("- Sorting records for MARIA-table '%s'\n",name); + if (write_info) + printf("Data records: %9s Deleted: %9s\n", + llstr(info->state->records,llbuff), + llstr(info->state->del,llbuff2)); + } + if (share->state.key_root[sort_key] == HA_OFFSET_ERROR) + DBUG_RETURN(0); /* Nothing to do */ + + init_key_cache(maria_key_cache, opt_key_cache_block_size, param->use_buffers, + 0, 0); + if (init_io_cache(&info->rec_cache,-1,(uint) param->write_buffer_length, + WRITE_CACHE,share->pack.header_length,1, + MYF(MY_WME | MY_WAIT_IF_FULL))) + goto err; + info->opt_flag|=WRITE_CACHE_USED; + + if (!(temp_buff=(uchar*) my_alloca((uint) keyinfo->block_length))) + { + _ma_check_print_error(param,"Not enough memory for key block"); + goto err; + } + if (!(sort_param.record=(byte*) my_malloc((uint) share->base.pack_reclength, + MYF(0)))) + { + _ma_check_print_error(param,"Not enough memory for record"); + goto err; + } + fn_format(param->temp_filename,name,"", MARIA_NAME_DEXT,2+4+32); + new_file=my_raid_create(fn_format(param->temp_filename, + param->temp_filename,"", + DATA_TMP_EXT,2+4), + 0,param->tmpfile_createflag, + share->base.raid_type, + share->base.raid_chunks, + share->base.raid_chunksize, + MYF(0)); + if (new_file < 0) + { + _ma_check_print_error(param,"Can't create new tempfile: '%s'", + param->temp_filename); + goto err; + } + if (share->pack.header_length) + if (maria_filecopy(param,new_file,info->dfile,0L,share->pack.header_length, + "datafile-header")) + goto err; + info->rec_cache.file=new_file; /* Use this file for cacheing*/ + + maria_lock_memory(param); + for (key=0 ; key < share->base.keys ; key++) + share->keyinfo[key].flag|= HA_SORT_ALLOWS_SAME; + + if (my_pread(share->kfile,(byte*) temp_buff, + (uint) keyinfo->block_length, + share->state.key_root[sort_key], + MYF(MY_NABP+MY_WME))) + { + _ma_check_print_error(param,"Can't read indexpage from filepos: %s", + (ulong) share->state.key_root[sort_key]); + goto err; + } + + /* Setup param for _ma_sort_write_record */ + sort_info.info=info; + sort_info.new_data_file_type=share->data_file_type; + sort_param.fix_datafile=1; + sort_param.master=1; + sort_param.filepos=share->pack.header_length; + old_record_count=info->state->records; + info->state->records=0; + if (sort_info.new_data_file_type != COMPRESSED_RECORD) + info->state->checksum=0; + + if (sort_record_index(&sort_param,info,keyinfo,share->state.key_root[sort_key], + temp_buff, sort_key,new_file,update_index) || + maria_write_data_suffix(&sort_info,1) || + flush_io_cache(&info->rec_cache)) + goto err; + + if (info->state->records != old_record_count) + { + _ma_check_print_error(param,"found %s of %s records", + llstr(info->state->records,llbuff), + llstr(old_record_count,llbuff2)); + goto err; + } + + VOID(my_close(info->dfile,MYF(MY_WME))); + param->out_flag|=O_NEW_DATA; /* Data in new file */ + info->dfile=new_file; /* Use new datafile */ + info->state->del=0; + info->state->empty=0; + share->state.dellink= HA_OFFSET_ERROR; + info->state->data_file_length=sort_param.filepos; + share->state.split=info->state->records; /* Only hole records */ + share->state.version=(ulong) time((time_t*) 0); + + info->update= (short) (HA_STATE_CHANGED | HA_STATE_ROW_CHANGED); + + if (param->testflag & T_WRITE_LOOP) + { + VOID(fputs(" \r",stdout)); VOID(fflush(stdout)); + } + got_error=0; + +err: + if (got_error && new_file >= 0) + { + VOID(end_io_cache(&info->rec_cache)); + (void) my_close(new_file,MYF(MY_WME)); + (void) my_raid_delete(param->temp_filename, share->base.raid_chunks, + MYF(MY_WME)); + } + if (temp_buff) + { + my_afree((gptr) temp_buff); + } + my_free(sort_param.record,MYF(MY_ALLOW_ZERO_PTR)); + info->opt_flag&= ~(READ_CACHE_USED | WRITE_CACHE_USED); + VOID(end_io_cache(&info->rec_cache)); + my_free(sort_info.buff,MYF(MY_ALLOW_ZERO_PTR)); + sort_info.buff=0; + share->state.sortkey=sort_key; + DBUG_RETURN(_ma_flush_blocks(param, share->key_cache, share->kfile) | + got_error); +} /* sort_records */ + + + /* Sort records recursive using one index */ + +static int sort_record_index(MARIA_SORT_PARAM *sort_param,MARIA_HA *info, + MARIA_KEYDEF *keyinfo, + my_off_t page, uchar *buff, uint sort_key, + File new_file,my_bool update_index) +{ + uint nod_flag,used_length,key_length; + uchar *temp_buff,*keypos,*endpos; + my_off_t next_page,rec_pos; + uchar lastkey[HA_MAX_KEY_BUFF]; + char llbuff[22]; + MARIA_SORT_INFO *sort_info= sort_param->sort_info; + HA_CHECK *param=sort_info->param; + DBUG_ENTER("sort_record_index"); + + nod_flag=_ma_test_if_nod(buff); + temp_buff=0; + + if (nod_flag) + { + if (!(temp_buff=(uchar*) my_alloca((uint) keyinfo->block_length))) + { + _ma_check_print_error(param,"Not Enough memory"); + DBUG_RETURN(-1); + } + } + used_length=maria_getint(buff); + keypos=buff+2+nod_flag; + endpos=buff+used_length; + for ( ;; ) + { + _sanity(__FILE__,__LINE__); + if (nod_flag) + { + next_page= _ma_kpos(nod_flag,keypos); + if (my_pread(info->s->kfile,(byte*) temp_buff, + (uint) keyinfo->block_length, next_page, + MYF(MY_NABP+MY_WME))) + { + _ma_check_print_error(param,"Can't read keys from filepos: %s", + llstr(next_page,llbuff)); + goto err; + } + if (sort_record_index(sort_param, info,keyinfo,next_page,temp_buff,sort_key, + new_file, update_index)) + goto err; + } + _sanity(__FILE__,__LINE__); + if (keypos >= endpos || + (key_length=(*keyinfo->get_key)(keyinfo,nod_flag,&keypos,lastkey)) + == 0) + break; + rec_pos= _ma_dpos(info,0,lastkey+key_length); + + if ((*info->s->read_rnd)(info,sort_param->record,rec_pos,0)) + { + _ma_check_print_error(param,"%d when reading datafile",my_errno); + goto err; + } + if (rec_pos != sort_param->filepos && update_index) + { + _ma_dpointer(info,keypos-nod_flag-info->s->rec_reflength, + sort_param->filepos); + if (maria_movepoint(info,sort_param->record,rec_pos,sort_param->filepos, + sort_key)) + { + _ma_check_print_error(param,"%d when updating key-pointers",my_errno); + goto err; + } + } + if (_ma_sort_write_record(sort_param)) + goto err; + } + /* Clear end of block to get better compression if the table is backuped */ + bzero((byte*) buff+used_length,keyinfo->block_length-used_length); + if (my_pwrite(info->s->kfile,(byte*) buff,(uint) keyinfo->block_length, + page,param->myf_rw)) + { + _ma_check_print_error(param,"%d when updating keyblock",my_errno); + goto err; + } + if (temp_buff) + my_afree((gptr) temp_buff); + DBUG_RETURN(0); +err: + if (temp_buff) + my_afree((gptr) temp_buff); + DBUG_RETURN(1); +} /* sort_record_index */ + + + +/* + Check if mariachk was killed by a signal + This is overloaded by other programs that want to be able to abort + sorting +*/ + +static int not_killed= 0; + +volatile int *_ma_killed_ptr(HA_CHECK *param __attribute__((unused))) +{ + return ¬_killed; /* always NULL */ +} + + /* print warnings and errors */ + /* VARARGS */ + +void _ma_check_print_info(HA_CHECK *param __attribute__((unused)), + const char *fmt,...) +{ + va_list args; + + va_start(args,fmt); + VOID(vfprintf(stdout, fmt, args)); + VOID(fputc('\n',stdout)); + va_end(args); +} + +/* VARARGS */ + +void _ma_check_print_warning(HA_CHECK *param, const char *fmt,...) +{ + va_list args; + DBUG_ENTER("_ma_check_print_warning"); + + fflush(stdout); + if (!param->warning_printed && !param->error_printed) + { + if (param->testflag & T_SILENT) + fprintf(stderr,"%s: MARIA file %s\n",my_progname_short, + param->isam_file_name); + param->out_flag|= O_DATA_LOST; + } + param->warning_printed=1; + va_start(args,fmt); + fprintf(stderr,"%s: warning: ",my_progname_short); + VOID(vfprintf(stderr, fmt, args)); + VOID(fputc('\n',stderr)); + fflush(stderr); + va_end(args); + DBUG_VOID_RETURN; +} + +/* VARARGS */ + +void _ma_check_print_error(HA_CHECK *param, const char *fmt,...) +{ + va_list args; + DBUG_ENTER("_ma_check_print_error"); + DBUG_PRINT("enter",("format: %s",fmt)); + + fflush(stdout); + if (!param->warning_printed && !param->error_printed) + { + if (param->testflag & T_SILENT) + fprintf(stderr,"%s: MARIA file %s\n",my_progname_short,param->isam_file_name); + param->out_flag|= O_DATA_LOST; + } + param->error_printed|=1; + va_start(args,fmt); + fprintf(stderr,"%s: error: ",my_progname_short); + VOID(vfprintf(stderr, fmt, args)); + VOID(fputc('\n',stderr)); + fflush(stderr); + va_end(args); + DBUG_VOID_RETURN; +} diff --git a/storage/maria/maria_def.h b/storage/maria/maria_def.h new file mode 100644 index 00000000000..e3f0219a663 --- /dev/null +++ b/storage/maria/maria_def.h @@ -0,0 +1,751 @@ +/* Copyright (C) 2006 MySQL AB & MySQL Finland AB & TCX DataKonsult AB + + 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; either version 2 of the License, or + (at your option) any later version. + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ + +/* This file is included by all internal maria files */ + +#include "maria.h" /* Structs & some defines */ +#include "myisampack.h" /* packing of keys */ +#include +#ifdef THREAD +#include +#include +#else +#include +#endif + +/* undef map from my_nosys; We need test-if-disk full */ +#undef my_write + +typedef struct st_maria_status_info +{ + ha_rows records; /* Rows in table */ + ha_rows del; /* Removed rows */ + my_off_t empty; /* lost space in datafile */ + my_off_t key_empty; /* lost space in indexfile */ + my_off_t key_file_length; + my_off_t data_file_length; + ha_checksum checksum; +} MARIA_STATUS_INFO; + +typedef struct st_maria_state_info +{ + struct + { /* Fileheader */ + uchar file_version[4]; + uchar options[2]; + uchar header_length[2]; + uchar state_info_length[2]; + uchar base_info_length[2]; + uchar base_pos[2]; + uchar key_parts[2]; /* Key parts */ + uchar unique_key_parts[2]; /* Key parts + unique parts */ + uchar keys; /* number of keys in file */ + uchar uniques; /* number of UNIQUE definitions */ + uchar language; /* Language for indexes */ + uchar max_block_size; /* max keyblock size */ + uchar fulltext_keys; + uchar not_used; /* To align to 8 */ + } header; + + MARIA_STATUS_INFO state; + ha_rows split; /* number of split blocks */ + my_off_t dellink; /* Link to next removed block */ + ulonglong auto_increment; + ulong process; /* process that updated table last */ + ulong unique; /* Unique number for this process */ + ulong update_count; /* Updated for each write lock */ + ulong status; + ulong *rec_per_key_part; + my_off_t *key_root; /* Start of key trees */ + my_off_t *key_del; /* delete links for trees */ + my_off_t rec_per_key_rows; /* Rows when calculating rec_per_key */ + + ulong sec_index_changed; /* Updated when new sec_index */ + ulong sec_index_used; /* which extra index are in use */ + ulonglong key_map; /* Which keys are in use */ + ulong version; /* timestamp of create */ + time_t create_time; /* Time when created database */ + time_t recover_time; /* Time for last recover */ + time_t check_time; /* Time for last check */ + uint sortkey; /* sorted by this key (not used) */ + uint open_count; + uint8 changed; /* Changed since mariachk */ + + /* the following isn't saved on disk */ + uint state_diff_length; /* Should be 0 */ + uint state_length; /* Length of state header in file */ + ulong *key_info; +} MARIA_STATE_INFO; + + +#define MARIA_STATE_INFO_SIZE (24+14*8+7*4+2*2+8) +#define MARIA_STATE_KEY_SIZE 8 +#define MARIA_STATE_KEYBLOCK_SIZE 8 +#define MARIA_STATE_KEYSEG_SIZE 4 +#define MARIA_STATE_EXTRA_SIZE ((MARIA_MAX_KEY+MARIA_MAX_KEY_BLOCK_SIZE)*MARIA_STATE_KEY_SIZE + MARIA_MAX_KEY*HA_MAX_KEY_SEG*MARIA_STATE_KEYSEG_SIZE) +#define MARIA_KEYDEF_SIZE (2+ 5*2) +#define MARIA_UNIQUEDEF_SIZE (2+1+1) +#define HA_KEYSEG_SIZE (6+ 2*2 + 4*2) +#define MARIA_COLUMNDEF_SIZE (2*3+1) +#define MARIA_BASE_INFO_SIZE (5*8 + 8*4 + 4 + 4*2 + 16) +#define MARIA_INDEX_BLOCK_MARGIN 16 /* Safety margin for .MYI tables */ + +typedef struct st__ma_base_info +{ + my_off_t keystart; /* Start of keys */ + my_off_t max_data_file_length; + my_off_t max_key_file_length; + my_off_t margin_key_file_length; + ha_rows records, reloc; /* Create information */ + ulong mean_row_length; /* Create information */ + ulong reclength; /* length of unpacked record */ + ulong pack_reclength; /* Length of full packed rec */ + ulong min_pack_length; + ulong max_pack_length; /* Max possibly length of + packed rec. */ + ulong min_block_length; + ulong fields, /* fields in table */ + pack_fields; /* packed fields in table */ + uint rec_reflength; /* = 2-8 */ + uint key_reflength; /* = 2-8 */ + uint keys; /* same as in state.header */ + uint auto_key; /* Which key-1 is a auto key */ + uint blobs; /* Number of blobs */ + uint pack_bits; /* Length of packed bits */ + uint max_key_block_length; /* Max block length */ + uint max_key_length; /* Max key length */ + /* Extra allocation when using dynamic record format */ + uint extra_alloc_bytes; + uint extra_alloc_procent; + /* Info about raid */ + uint raid_type, raid_chunks; + ulong raid_chunksize; + /* The following are from the header */ + uint key_parts, all_key_parts; +} MARIA_BASE_INFO; + + + /* Structs used intern in database */ + +typedef struct st_maria_blob /* Info of record */ +{ + ulong offset; /* Offset to blob in record */ + uint pack_length; /* Type of packed length */ + ulong length; /* Calc:ed for each record */ +} MARIA_BLOB; + + +typedef struct st_maria_pack +{ + ulong header_length; + uint ref_length; + uchar version; +} MARIA_PACK; + +#define MAX_NONMAPPED_INSERTS 1000 + +typedef struct st_maria_share +{ /* Shared between opens */ + MARIA_STATE_INFO state; + MARIA_BASE_INFO base; + MARIA_KEYDEF ft2_keyinfo; /* Second-level ft-key + definition */ + MARIA_KEYDEF *keyinfo; /* Key definitions */ + MARIA_UNIQUEDEF *uniqueinfo; /* unique definitions */ + HA_KEYSEG *keyparts; /* key part info */ + MARIA_COLUMNDEF *rec; /* Pointer to field information + */ + MARIA_PACK pack; /* Data about packed records */ + MARIA_BLOB *blobs; /* Pointer to blobs */ + char *unique_file_name; /* realpath() of index file */ + char *data_file_name, /* Resolved path names from + symlinks */ + *index_file_name; + byte *file_map; /* mem-map of file if possible */ + KEY_CACHE *key_cache; /* ref to the current key cache + */ + MARIA_DECODE_TREE *decode_trees; + uint16 *decode_tables; + int(*read_record) (struct st_maria_info *, my_off_t, byte *); + int(*write_record) (struct st_maria_info *, const byte *); + int(*update_record) (struct st_maria_info *, my_off_t, const byte *); + int(*delete_record) (struct st_maria_info *); + int(*read_rnd) (struct st_maria_info *, byte *, my_off_t, my_bool); + int(*compare_record) (struct st_maria_info *, const byte *); + ha_checksum(*calc_checksum) (struct st_maria_info *, const byte *); + int(*compare_unique) (struct st_maria_info *, MARIA_UNIQUEDEF *, + const byte *record, my_off_t pos); + uint(*file_read) (MARIA_HA *, byte *, uint, my_off_t, myf); + uint(*file_write) (MARIA_HA *, byte *, uint, my_off_t, myf); + invalidator_by_filename invalidator; /* query cache invalidator */ + ulong this_process; /* processid */ + ulong last_process; /* For table-change-check */ + ulong last_version; /* Version on start */ + ulong options; /* Options used */ + ulong min_pack_length; /* Theese are used by packed + data */ + ulong max_pack_length; + ulong state_diff_length; + uint rec_reflength; /* rec_reflength in use now */ + uint unique_name_length; + uint32 ftparsers; /* Number of distinct ftparsers + + 1 */ + File kfile; /* Shared keyfile */ + File data_file; /* Shared data file */ + int mode; /* mode of file on open */ + uint reopen; /* How many times reopened */ + uint w_locks, r_locks, tot_locks; /* Number of read/write locks */ + uint blocksize; /* blocksize of keyfile */ + myf write_flag; + enum data_file_type data_file_type; + my_bool changed, /* If changed since lock */ + global_changed, /* If changed since open */ + not_flushed, temporary, delay_key_write, concurrent_insert; +#ifdef THREAD + THR_LOCK lock; + pthread_mutex_t intern_lock; /* Locking for use with + _locking */ + rw_lock_t *key_root_lock; +#endif + my_off_t mmaped_length; + uint nonmmaped_inserts; /* counter of writing in + non-mmaped area */ + rw_lock_t mmap_lock; +} MARIA_SHARE; + + +typedef uint maria_bit_type; + +typedef struct st_maria_bit_buff +{ /* Used for packing of record */ + maria_bit_type current_byte; + uint bits; + uchar *pos, *end, *blob_pos, *blob_end; + uint error; +} MARIA_BIT_BUFF; + +struct st_maria_info +{ + MARIA_SHARE *s; /* Shared between open:s */ + MARIA_STATUS_INFO *state, save_state; + MARIA_BLOB *blobs; /* Pointer to blobs */ + MARIA_BIT_BUFF bit_buff; + /* accumulate indexfile changes between write's */ + TREE *bulk_insert; + DYNAMIC_ARRAY *ft1_to_ft2; /* used only in ft1->ft2 conversion */ + MYSQL_FTPARSER_PARAM *ftparser_param; /* share info between init/deinit */ + char *filename; /* parameter to open filename */ + uchar *buff, /* Temp area for key */ + *lastkey, *lastkey2; /* Last used search key */ + uchar *first_mbr_key; /* Searhed spatial key */ + byte *rec_buff; /* Tempbuff for recordpack */ + uchar *int_keypos, /* Save position for next/previous */ + *int_maxpos; /* -""- */ + uint int_nod_flag; /* -""- */ + uint32 int_keytree_version; /* -""- */ + int(*read_record) (struct st_maria_info *, my_off_t, byte *); + invalidator_by_filename invalidator; /* query cache invalidator */ + ulong this_unique; /* uniq filenumber or thread */ + ulong last_unique; /* last unique number */ + ulong this_loop; /* counter for this open */ + ulong last_loop; /* last used counter */ + my_off_t lastpos, /* Last record position */ + nextpos; /* Position to next record */ + my_off_t save_lastpos; + my_off_t pos; /* Intern variable */ + my_off_t last_keypage; /* Last key page read */ + my_off_t last_search_keypage; /* Last keypage when searching */ + my_off_t dupp_key_pos; + ha_checksum checksum; + /* + QQ: the folloing two xxx_length fields should be removed, + as they are not compatible with parallel repair + */ + ulong packed_length, blob_length; /* Length of found, packed record */ + int dfile; /* The datafile */ + uint opt_flag; /* Optim. for space/speed */ + uint update; /* If file changed since open */ + int lastinx; /* Last used index */ + uint lastkey_length; /* Length of key in lastkey */ + uint last_rkey_length; /* Last length in maria_rkey() */ + enum ha_rkey_function last_key_func; /* CONTAIN, OVERLAP, etc */ + uint save_lastkey_length; + uint pack_key_length; /* For MARIAMRG */ + int errkey; /* Got last error on this key */ + int lock_type; /* How database was locked */ + int tmp_lock_type; /* When locked by readinfo */ + uint data_changed; /* Somebody has changed data */ + uint save_update; /* When using KEY_READ */ + int save_lastinx; + LIST open_list; + IO_CACHE rec_cache; /* When cacheing records */ + uint preload_buff_size; /* When preloading indexes */ + myf lock_wait; /* is 0 or MY_DONT_WAIT */ + my_bool was_locked; /* Was locked in panic */ + my_bool append_insert_at_end; /* Set if concurrent insert */ + my_bool quick_mode; + /* If info->buff can't be used for rnext */ + my_bool page_changed; + /* If info->buff has to be reread for rnext */ + my_bool buff_used; + my_bool once_flags; /* For MARIAMRG */ +#ifdef THREAD + THR_LOCK_DATA lock; +#endif + uchar *maria_rtree_recursion_state; /* For RTREE */ + int maria_rtree_recursion_depth; +}; + +/* Some defines used by isam-funktions */ + +#define USE_WHOLE_KEY HA_MAX_KEY_BUFF*2 /* Use whole key in _search() */ +#define F_EXTRA_LCK -1 + +/* bits in opt_flag */ +#define MEMMAP_USED 32 +#define REMEMBER_OLD_POS 64 + +#define WRITEINFO_UPDATE_KEYFILE 1 +#define WRITEINFO_NO_UNLOCK 2 + +/* once_flags */ +#define USE_PACKED_KEYS 1 +#define RRND_PRESERVE_LASTINX 2 + +/* bits in state.changed */ + +#define STATE_CHANGED 1 +#define STATE_CRASHED 2 +#define STATE_CRASHED_ON_REPAIR 4 +#define STATE_NOT_ANALYZED 8 +#define STATE_NOT_OPTIMIZED_KEYS 16 +#define STATE_NOT_SORTED_PAGES 32 + +/* options to maria_read_cache */ + +#define READING_NEXT 1 +#define READING_HEADER 2 + +#define maria_getint(x) ((uint) mi_uint2korr(x) & 32767) +#define maria_putint(x,y,nod) { uint16 boh=(nod ? (uint16) 32768 : 0) + (uint16) (y);\ + mi_int2store(x,boh); } +#define _ma_test_if_nod(x) (x[0] & 128 ? info->s->base.key_reflength : 0) +#define maria_mark_crashed(x) (x)->s->state.changed|=STATE_CRASHED +#define maria_mark_crashed_on_repair(x) { (x)->s->state.changed|=STATE_CRASHED|STATE_CRASHED_ON_REPAIR ; (x)->update|= HA_STATE_CHANGED; } +#define maria_is_crashed(x) ((x)->s->state.changed & STATE_CRASHED) +#define maria_is_crashed_on_repair(x) ((x)->s->state.changed & STATE_CRASHED_ON_REPAIR) +#define maria_print_error(SHARE, ERRNO) \ + _ma_report_error((ERRNO), (SHARE)->index_file_name) + +/* Functions to store length of space packed keys, VARCHAR or BLOB keys */ + +#define store_key_length(key,length) \ +{ if ((length) < 255) \ + { *(key)=(length); } \ + else \ + { *(key)=255; mi_int2store((key)+1,(length)); } \ +} + +#define get_key_full_length(length,key) \ +{ if ((uchar) *(key) != 255) \ + length= ((uint) (uchar) *((key)++))+1; \ + else \ + { length=mi_uint2korr((key)+1)+3; (key)+=3; } \ +} + +#define get_key_full_length_rdonly(length,key) \ +{ if ((uchar) *(key) != 255) \ + length= ((uint) (uchar) *((key)))+1; \ + else \ + { length=mi_uint2korr((key)+1)+3; } \ +} + +#define get_pack_length(length) ((length) >= 255 ? 3 : 1) + +#define MARIA_MIN_BLOCK_LENGTH 20 /* Because of delete-link */ +/* Don't use to small record-blocks */ +#define MARIA_EXTEND_BLOCK_LENGTH 20 +#define MARIA_SPLIT_LENGTH ((MARIA_EXTEND_BLOCK_LENGTH+4)*2) + /* Max prefix of record-block */ +#define MARIA_MAX_DYN_BLOCK_HEADER 20 +#define MARIA_BLOCK_INFO_HEADER_LENGTH 20 +#define MARIA_DYN_DELETE_BLOCK_HEADER 20 /* length of delete-block-header */ +#define MARIA_DYN_MAX_BLOCK_LENGTH ((1L << 24)-4L) +#define MARIA_DYN_MAX_ROW_LENGTH (MARIA_DYN_MAX_BLOCK_LENGTH - MARIA_SPLIT_LENGTH) +#define MARIA_DYN_ALIGN_SIZE 4 /* Align blocks on this */ +#define MARIA_MAX_DYN_HEADER_BYTE 13 /* max header byte for dynamic rows */ +#define MARIA_MAX_BLOCK_LENGTH ((((ulong) 1 << 24)-1) & (~ (ulong) (MARIA_DYN_ALIGN_SIZE-1))) +#define MARIA_REC_BUFF_OFFSET ALIGN_SIZE(MARIA_DYN_DELETE_BLOCK_HEADER+sizeof(uint32)) + +#define MEMMAP_EXTRA_MARGIN 7 /* Write this as a suffix for file */ + +#define PACK_TYPE_SELECTED 1 /* Bits in field->pack_type */ +#define PACK_TYPE_SPACE_FIELDS 2 +#define PACK_TYPE_ZERO_FILL 4 +#define MARIA_FOUND_WRONG_KEY 32738 /* Impossible value from ha_key_cmp */ + +#define MARIA_MAX_KEY_BLOCK_SIZE (MARIA_MAX_KEY_BLOCK_LENGTH/MARIA_MIN_KEY_BLOCK_LENGTH) +#define MARIA_BLOCK_SIZE(key_length,data_pointer,key_pointer) (((((key_length)+(data_pointer)+(key_pointer))*4+(key_pointer)+2)/maria_block_size+1)*maria_block_size) +#define MARIA_MAX_KEYPTR_SIZE 5 /* For calculating block lengths */ +#define MARIA_MIN_KEYBLOCK_LENGTH 50 /* When to split delete blocks */ + +#define MARIA_MIN_SIZE_BULK_INSERT_TREE 16384 /* this is per key */ +#define MARIA_MIN_ROWS_TO_USE_BULK_INSERT 100 +#define MARIA_MIN_ROWS_TO_DISABLE_INDEXES 100 +#define MARIA_MIN_ROWS_TO_USE_WRITE_CACHE 10 + +/* The UNIQUE check is done with a hashed long key */ + +#define MARIA_UNIQUE_HASH_TYPE HA_KEYTYPE_ULONG_INT +#define maria_unique_store(A,B) mi_int4store((A),(B)) + +#ifdef THREAD +extern pthread_mutex_t THR_LOCK_maria; +#endif +#if !defined(THREAD) || defined(DONT_USE_RW_LOCKS) +#define rw_wrlock(A) {} +#define rw_rdlock(A) {} +#define rw_unlock(A) {} +#endif + + /* Some extern variables */ + +extern LIST *maria_open_list; +extern uchar NEAR maria_file_magic[], NEAR maria_pack_file_magic[]; +extern uint NEAR maria_read_vec[], NEAR maria_readnext_vec[]; +extern uint maria_quick_table_bits; +extern File maria_log_file; +extern ulong maria_pid; + + /* This is used by _ma_calc_xxx_key_length och _ma_store_key */ + +typedef struct st_maria_s_param +{ + uint ref_length, key_length, + n_ref_length, + n_length, totlength, part_of_prev_key, prev_length, pack_marker; + uchar *key, *prev_key, *next_key_pos; + bool store_not_null; +} MARIA_KEY_PARAM; + + /* Prototypes for intern functions */ + +extern int _ma_read_dynamic_record(MARIA_HA *info, my_off_t filepos, + byte *buf); +extern int _ma_write_dynamic_record(MARIA_HA *, const byte *); +extern int _ma_update_dynamic_record(MARIA_HA *, my_off_t, const byte *); +extern int _ma_delete_dynamic_record(MARIA_HA *info); +extern int _ma_cmp_dynamic_record(MARIA_HA *info, const byte *record); +extern int _ma_read_rnd_dynamic_record(MARIA_HA *, byte *, my_off_t, + my_bool); +extern int _ma_write_blob_record(MARIA_HA *, const byte *); +extern int _ma_update_blob_record(MARIA_HA *, my_off_t, const byte *); +extern int _ma_read_static_record(MARIA_HA *info, my_off_t filepos, + byte *buf); +extern int _ma_write_static_record(MARIA_HA *, const byte *); +extern int _ma_update_static_record(MARIA_HA *, my_off_t, const byte *); +extern int _ma_delete_static_record(MARIA_HA *info); +extern int _ma_cmp_static_record(MARIA_HA *info, const byte *record); +extern int _ma_read_rnd_static_record(MARIA_HA *, byte *, my_off_t, my_bool); +extern int _ma_ck_write(MARIA_HA *info, uint keynr, uchar *key, + uint length); +extern int _ma_ck_real_write_btree(MARIA_HA *info, MARIA_KEYDEF *keyinfo, + uchar *key, uint key_length, + my_off_t *root, uint comp_flag); +extern int _ma_enlarge_root(MARIA_HA *info, MARIA_KEYDEF *keyinfo, + uchar *key, my_off_t *root); +extern int _ma_insert(MARIA_HA *info, MARIA_KEYDEF *keyinfo, uchar *key, + uchar *anc_buff, uchar *key_pos, uchar *key_buff, + uchar *father_buff, uchar *father_keypos, + my_off_t father_page, my_bool insert_last); +extern int _ma_split_page(MARIA_HA *info, MARIA_KEYDEF *keyinfo, + uchar *key, uchar *buff, uchar *key_buff, + my_bool insert_last); +extern uchar *_ma_find_half_pos(uint nod_flag, MARIA_KEYDEF *keyinfo, + uchar *page, uchar *key, + uint *return_key_length, + uchar ** after_key); +extern int _ma_calc_static_key_length(MARIA_KEYDEF *keyinfo, uint nod_flag, + uchar *key_pos, uchar *org_key, + uchar *key_buff, uchar *key, + MARIA_KEY_PARAM *s_temp); +extern int _ma_calc_var_key_length(MARIA_KEYDEF *keyinfo, uint nod_flag, + uchar *key_pos, uchar *org_key, + uchar *key_buff, uchar *key, + MARIA_KEY_PARAM *s_temp); +extern int _ma_calc_var_pack_key_length(MARIA_KEYDEF *keyinfo, + uint nod_flag, uchar *key_pos, + uchar *org_key, uchar *prev_key, + uchar *key, + MARIA_KEY_PARAM *s_temp); +extern int _ma_calc_bin_pack_key_length(MARIA_KEYDEF *keyinfo, + uint nod_flag, uchar *key_pos, + uchar *org_key, uchar *prev_key, + uchar *key, + MARIA_KEY_PARAM *s_temp); +void _ma_store_static_key(MARIA_KEYDEF *keyinfo, uchar *key_pos, + MARIA_KEY_PARAM *s_temp); +void _ma_store_var_pack_key(MARIA_KEYDEF *keyinfo, uchar *key_pos, + MARIA_KEY_PARAM *s_temp); +#ifdef NOT_USED +void _ma_store_pack_key(MARIA_KEYDEF *keyinfo, uchar *key_pos, + MARIA_KEY_PARAM *s_temp); +#endif +void _ma_store_bin_pack_key(MARIA_KEYDEF *keyinfo, uchar *key_pos, + MARIA_KEY_PARAM *s_temp); + +extern int _ma_ck_delete(MARIA_HA *info, uint keynr, uchar *key, + uint key_length); +extern int _ma_readinfo(MARIA_HA *info, int lock_flag, int check_keybuffer); +extern int _ma_writeinfo(MARIA_HA *info, uint options); +extern int _ma_test_if_changed(MARIA_HA *info); +extern int _ma_mark_file_changed(MARIA_HA *info); +extern int _ma_decrement_open_count(MARIA_HA *info); +extern int _ma_check_index(MARIA_HA *info, int inx); +extern int _ma_search(MARIA_HA *info, MARIA_KEYDEF *keyinfo, uchar *key, + uint key_len, uint nextflag, my_off_t pos); +extern int _ma_bin_search(struct st_maria_info *info, MARIA_KEYDEF *keyinfo, + uchar *page, uchar *key, uint key_len, + uint comp_flag, uchar **ret_pos, uchar *buff, + my_bool *was_last_key); +extern int _ma_seq_search(MARIA_HA *info, MARIA_KEYDEF *keyinfo, + uchar *page, uchar *key, uint key_len, + uint comp_flag, uchar ** ret_pos, uchar *buff, + my_bool *was_last_key); +extern int _ma_prefix_search(MARIA_HA *info, MARIA_KEYDEF *keyinfo, + uchar *page, uchar *key, uint key_len, + uint comp_flag, uchar ** ret_pos, uchar *buff, + my_bool *was_last_key); +extern my_off_t _ma_kpos(uint nod_flag, uchar *after_key); +extern void _ma_kpointer(MARIA_HA *info, uchar *buff, my_off_t pos); +extern my_off_t _ma_dpos(MARIA_HA *info, uint nod_flag, uchar *after_key); +extern my_off_t _ma_rec_pos(MARIA_SHARE *info, uchar *ptr); +extern void _ma_dpointer(MARIA_HA *info, uchar *buff, my_off_t pos); +extern uint _ma_get_static_key(MARIA_KEYDEF *keyinfo, uint nod_flag, + uchar **page, uchar *key); +extern uint _ma_get_pack_key(MARIA_KEYDEF *keyinfo, uint nod_flag, + uchar **page, uchar *key); +extern uint _ma_get_binary_pack_key(MARIA_KEYDEF *keyinfo, uint nod_flag, + uchar ** page_pos, uchar *key); +extern uchar *_ma_get_last_key(MARIA_HA *info, MARIA_KEYDEF *keyinfo, + uchar *keypos, uchar *lastkey, + uchar *endpos, uint *return_key_length); +extern uchar *_ma_get_key(MARIA_HA *info, MARIA_KEYDEF *keyinfo, + uchar *page, uchar *key, uchar *keypos, + uint *return_key_length); +extern uint _ma_keylength(MARIA_KEYDEF *keyinfo, uchar *key); +extern uint _ma_keylength_part(MARIA_KEYDEF *keyinfo, register uchar *key, + HA_KEYSEG *end); +extern uchar *_ma_move_key(MARIA_KEYDEF *keyinfo, uchar *to, uchar *from); +extern int _ma_search_next(MARIA_HA *info, MARIA_KEYDEF *keyinfo, + uchar *key, uint key_length, uint nextflag, + my_off_t pos); +extern int _ma_search_first(MARIA_HA *info, MARIA_KEYDEF *keyinfo, + my_off_t pos); +extern int _ma_search_last(MARIA_HA *info, MARIA_KEYDEF *keyinfo, + my_off_t pos); +extern uchar *_ma_fetch_keypage(MARIA_HA *info, MARIA_KEYDEF *keyinfo, + my_off_t page, int level, uchar *buff, + int return_buffer); +extern int _ma_write_keypage(MARIA_HA *info, MARIA_KEYDEF *keyinfo, + my_off_t page, int level, uchar *buff); +extern int _ma_dispose(MARIA_HA *info, MARIA_KEYDEF *keyinfo, my_off_t pos, + int level); +extern my_off_t _ma_new(MARIA_HA *info, MARIA_KEYDEF *keyinfo, int level); +extern uint _ma_make_key(MARIA_HA *info, uint keynr, uchar *key, + const byte *record, my_off_t filepos); +extern uint _ma_pack_key(MARIA_HA *info, uint keynr, uchar *key, + uchar *old, uint key_length, + HA_KEYSEG ** last_used_keyseg); +extern int _ma_read_key_record(MARIA_HA *info, my_off_t filepos, + byte *buf); +extern int _ma_read_cache(IO_CACHE *info, byte *buff, my_off_t pos, + uint length, int re_read_if_possibly); +extern void _ma_update_auto_increment(MARIA_HA *info, const byte *record); + +extern byte *_ma_alloc_rec_buff(MARIA_HA *, ulong, byte **); +#define _ma_get_rec_buff_ptr(info,buf) \ + ((((info)->s->options & HA_OPTION_PACK_RECORD) && (buf)) ? \ + (buf) - MARIA_REC_BUFF_OFFSET : (buf)) +#define _ma_get_rec_buff_len(info,buf) \ + (*((uint32 *)(_ma_get_rec_buff_ptr(info,buf)))) + +extern ulong _ma_rec_unpack(MARIA_HA *info, byte *to, byte *from, + ulong reclength); +extern my_bool _ma_rec_check(MARIA_HA *info, const char *record, + byte *packpos, ulong packed_length, + my_bool with_checkum); +extern int _ma_write_part_record(MARIA_HA *info, my_off_t filepos, + ulong length, my_off_t next_filepos, + byte ** record, ulong *reclength, + int *flag); +extern void _ma_print_key(FILE *stream, HA_KEYSEG *keyseg, + const uchar *key, uint length); +extern my_bool _ma_read_pack_info(MARIA_HA *info, pbool fix_keys); +extern int _ma_read_pack_record(MARIA_HA *info, my_off_t filepos, + byte *buf); +extern int _ma_read_rnd_pack_record(MARIA_HA *, byte *, my_off_t, my_bool); +extern int _ma_pack_rec_unpack(MARIA_HA *info, byte *to, byte *from, + ulong reclength); +extern ulonglong _ma_safe_mul(ulonglong a, ulonglong b); +extern int _ma_ft_update(MARIA_HA *info, uint keynr, byte *keybuf, + const byte *oldrec, const byte *newrec, + my_off_t pos); + +/* Parameter to _ma_get_block_info */ + +typedef struct st_maria_block_info +{ + uchar header[MARIA_BLOCK_INFO_HEADER_LENGTH]; + ulong rec_len; + ulong data_len; + ulong block_len; + ulong blob_len; + my_off_t filepos; + my_off_t next_filepos; + my_off_t prev_filepos; + uint second_read; + uint offset; +} MARIA_BLOCK_INFO; + +/* bits in return from _ma_get_block_info */ + +#define BLOCK_FIRST 1 +#define BLOCK_LAST 2 +#define BLOCK_DELETED 4 +#define BLOCK_ERROR 8 /* Wrong data */ +#define BLOCK_SYNC_ERROR 16 /* Right data at wrong place */ +#define BLOCK_FATAL_ERROR 32 /* hardware-error */ + +#define NEED_MEM ((uint) 10*4*(IO_SIZE+32)+32) /* Nead for recursion */ +#define MAXERR 20 +#define BUFFERS_WHEN_SORTING 16 /* Alloc for sort-key-tree */ +#define WRITE_COUNT MY_HOW_OFTEN_TO_WRITE +#define INDEX_TMP_EXT ".TMM" +#define DATA_TMP_EXT ".TMD" + +#define UPDATE_TIME 1 +#define UPDATE_STAT 2 +#define UPDATE_SORT 4 +#define UPDATE_AUTO_INC 8 +#define UPDATE_OPEN_COUNT 16 + +#define USE_BUFFER_INIT (((1024L*512L-MALLOC_OVERHEAD)/IO_SIZE)*IO_SIZE) +#define READ_BUFFER_INIT (1024L*256L-MALLOC_OVERHEAD) +#define SORT_BUFFER_INIT (2048L*1024L-MALLOC_OVERHEAD) +#define MIN_SORT_BUFFER (4096-MALLOC_OVERHEAD) + +enum maria_log_commands +{ + MARIA_LOG_OPEN, MARIA_LOG_WRITE, MARIA_LOG_UPDATE, MARIA_LOG_DELETE, + MARIA_LOG_CLOSE, MARIA_LOG_EXTRA, MARIA_LOG_LOCK, MARIA_LOG_DELETE_ALL +}; + +#define maria_log(a,b,c,d) if (maria_log_file >= 0) _ma_log(a,b,c,d) +#define maria_log_command(a,b,c,d,e) if (maria_log_file >= 0) _ma_log_command(a,b,c,d,e) +#define maria_log_record(a,b,c,d,e) if (maria_log_file >= 0) _ma_log_record(a,b,c,d,e) + +#define fast_ma_writeinfo(INFO) if (!(INFO)->s->tot_locks) (void) _ma_writeinfo((INFO),0) +#define fast_ma_readinfo(INFO) ((INFO)->lock_type == F_UNLCK) && _ma_readinfo((INFO),F_RDLCK,1) + +extern uint _ma_get_block_info(MARIA_BLOCK_INFO *, File, my_off_t); +extern uint _ma_rec_pack(MARIA_HA *info, byte *to, const byte *from); +extern uint _ma_pack_get_block_info(MARIA_HA *, MARIA_BLOCK_INFO *, File, + my_off_t); +extern void _ma_store_blob_length(byte *pos, uint pack_length, uint length); +extern void _ma_log(enum maria_log_commands command, MARIA_HA *info, + const byte *buffert, uint length); +extern void _ma_log_command(enum maria_log_commands command, + MARIA_HA *info, const byte *buffert, + uint length, int result); +extern void _ma_log_record(enum maria_log_commands command, MARIA_HA *info, + const byte *record, my_off_t filepos, + int result); +extern void _ma_report_error(int errcode, const char *file_name); +extern my_bool _ma_memmap_file(MARIA_HA *info); +extern void _ma_unmap_file(MARIA_HA *info); +extern uint _ma_save_pack_length(uint version, byte * block_buff, + ulong length); +extern uint _ma_calc_pack_length(uint version, ulong length); +extern ulong _ma_calc_blob_length(uint length, const byte *pos); +extern uint _ma_mmap_pread(MARIA_HA *info, byte *Buffer, + uint Count, my_off_t offset, myf MyFlags); +extern uint _ma_mmap_pwrite(MARIA_HA *info, byte *Buffer, + uint Count, my_off_t offset, myf MyFlags); +extern uint _ma_nommap_pread(MARIA_HA *info, byte *Buffer, + uint Count, my_off_t offset, myf MyFlags); +extern uint _ma_nommap_pwrite(MARIA_HA *info, byte *Buffer, + uint Count, my_off_t offset, myf MyFlags); + +uint _ma_state_info_write(File file, MARIA_STATE_INFO *state, uint pWrite); +uchar *_ma_state_info_read(uchar *ptr, MARIA_STATE_INFO *state); +uint _ma_state_info_read_dsk(File file, MARIA_STATE_INFO *state, + my_bool pRead); +uint _ma_base_info_write(File file, MARIA_BASE_INFO *base); +uchar *_ma_n_base_info_read(uchar *ptr, MARIA_BASE_INFO *base); +int _ma_keyseg_write(File file, const HA_KEYSEG *keyseg); +char *_ma_keyseg_read(char *ptr, HA_KEYSEG *keyseg); +uint _ma_keydef_write(File file, MARIA_KEYDEF *keydef); +char *_ma_keydef_read(char *ptr, MARIA_KEYDEF *keydef); +uint _ma_uniquedef_write(File file, MARIA_UNIQUEDEF *keydef); +char *_ma_uniquedef_read(char *ptr, MARIA_UNIQUEDEF *keydef); +uint _ma_recinfo_write(File file, MARIA_COLUMNDEF *recinfo); +char *_ma_recinfo_read(char *ptr, MARIA_COLUMNDEF *recinfo); +ulong _ma_calc_total_blob_length(MARIA_HA *info, const byte *record); +ha_checksum _ma_checksum(MARIA_HA *info, const byte *buf); +ha_checksum _ma_static_checksum(MARIA_HA *info, const byte *buf); +my_bool _ma_check_unique(MARIA_HA *info, MARIA_UNIQUEDEF *def, + byte *record, ha_checksum unique_hash, + my_off_t pos); +ha_checksum _ma_unique_hash(MARIA_UNIQUEDEF *def, const byte *buf); +int _ma_cmp_static_unique(MARIA_HA *info, MARIA_UNIQUEDEF *def, + const byte *record, my_off_t pos); +int _ma_cmp_dynamic_unique(MARIA_HA *info, MARIA_UNIQUEDEF *def, + const byte *record, my_off_t pos); +int _ma_unique_comp(MARIA_UNIQUEDEF *def, const byte *a, const byte *b, + my_bool null_are_equal); +void _ma_get_status(void *param, int concurrent_insert); +void _ma_update_status(void *param); +void _ma_copy_status(void *to, void *from); +my_bool _ma_check_status(void *param); + +extern MARIA_HA *_ma_test_if_reopen(char *filename); +my_bool _ma_check_table_is_closed(const char *name, const char *where); +int _ma_open_datafile(MARIA_HA *info, MARIA_SHARE *share, File file_to_dup); +int _ma_open_keyfile(MARIA_SHARE *share); +void _ma_setup_functions(register MARIA_SHARE *share); +my_bool _ma_dynmap_file(MARIA_HA *info, my_off_t size); +void _ma_remap_file(MARIA_HA *info, my_off_t size); + +/* Functions needed by _ma_check (are overrided in MySQL) */ +C_MODE_START +volatile int *_ma_killed_ptr(HA_CHECK *param); +void _ma_check_print_error _VARARGS((HA_CHECK *param, const char *fmt, ...)); +void _ma_check_print_warning _VARARGS((HA_CHECK *param, const char *fmt, ...)); +void _ma_check_print_info _VARARGS((HA_CHECK *param, const char *fmt, ...)); +C_MODE_END + +int _ma_flush_pending_blocks(MARIA_SORT_PARAM *param); +int _ma_sort_ft_buf_flush(MARIA_SORT_PARAM *sort_param); +int _ma_thr_write_keys(MARIA_SORT_PARAM *sort_param); +#ifdef THREAD +pthread_handler_t _ma_thr_find_all_keys(void *arg); +#endif +int _ma_flush_blocks(HA_CHECK *param, KEY_CACHE *key_cache, File file); + +int _ma_sort_write_record(MARIA_SORT_PARAM *sort_param); +int _ma_create_index_by_sort(MARIA_SORT_PARAM *info, my_bool no_messages, + ulong); diff --git a/storage/maria/maria_ftdump.c b/storage/maria/maria_ftdump.c new file mode 100644 index 00000000000..eb5dec5aa3b --- /dev/null +++ b/storage/maria/maria_ftdump.c @@ -0,0 +1,279 @@ +/* Copyright (C) 2006 MySQL AB & MySQL Finland AB & TCX DataKonsult AB + + 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; either version 2 of the License, or + (at your option) any later version. + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ + +/* Written by Sergei A. Golubchik, who has a shared copyright to this code + added support for long options (my_getopt) 22.5.2002 by Jani Tolonen */ + +#include "ma_ftdefs.h" +#include + +static void usage(); +static void complain(int val); +static my_bool get_one_option(int, const struct my_option *, char *); + +static int count=0, stats=0, dump=0, lstats=0; +static my_bool verbose; +static char *query=NULL; +static uint lengths[256]; + +#define MAX_LEN (HA_FT_MAXBYTELEN+10) +#define HOW_OFTEN_TO_WRITE 10000 + +static struct my_option my_long_options[] = +{ + {"dump", 'd', "Dump index (incl. data offsets and word weights).", + 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, + {"stats", 's', "Report global stats.", + 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, + {"verbose", 'v', "Be verbose.", + (gptr*) &verbose, (gptr*) &verbose, 0, GET_BOOL, NO_ARG, 0, 0, 0, 0, 0, 0}, + {"count", 'c', "Calculate per-word stats (counts and global weights).", + 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, + {"length", 'l', "Report length distribution.", + 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, + {"help", 'h', "Display help and exit.", + 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, + {"help", '?', "Synonym for -h.", + 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, + { 0, 0, 0, 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0} +}; + + +int main(int argc,char *argv[]) +{ + int error=0, subkeys; + uint keylen, keylen2=0, inx, doc_cnt=0; + float weight= 1.0; + double gws, min_gws=0, avg_gws=0; + MARIA_HA *info; + char buf[MAX_LEN], buf2[MAX_LEN], buf_maxlen[MAX_LEN], buf_min_gws[MAX_LEN]; + ulong total=0, maxlen=0, uniq=0, max_doc_cnt=0; + struct { MARIA_HA *info; } aio0, *aio=&aio0; /* for GWS_IN_USE */ + + MY_INIT(argv[0]); + if ((error= handle_options(&argc, &argv, my_long_options, get_one_option))) + exit(error); + maria_init(); + if (count || dump) + verbose=0; + if (!count && !dump && !lstats && !query) + stats=1; + + if (verbose) + setbuf(stdout,NULL); + + if (argc < 2) + usage(); + + { + char *end; + inx= (uint) strtoll(argv[1], &end, 10); + if (*end) + usage(); + } + + init_key_cache(maria_key_cache,MARIA_KEY_BLOCK_LENGTH,USE_BUFFER_INIT, 0, 0); + + if (!(info=maria_open(argv[0],2,HA_OPEN_ABORT_IF_LOCKED|HA_OPEN_FROM_SQL_LAYER))) + { + error=my_errno; + goto err; + } + + *buf2=0; + aio->info=info; + + if ((inx >= info->s->base.keys) || + !(info->s->keyinfo[inx].flag & HA_FULLTEXT)) + { + printf("Key %d in table %s is not a FULLTEXT key\n", inx, info->filename); + goto err; + } + + maria_lock_database(info, F_EXTRA_LCK); + + info->lastpos= HA_OFFSET_ERROR; + info->update|= HA_STATE_PREV_FOUND; + + while (!(error=maria_rnext(info,NULL,inx))) + { + keylen=*(info->lastkey); + + subkeys=ft_sintXkorr(info->lastkey+keylen+1); + if (subkeys >= 0) + weight=*(float*)&subkeys; + +#ifdef HAVE_SNPRINTF + snprintf(buf,MAX_LEN,"%.*s",(int) keylen,info->lastkey+1); +#else + sprintf(buf,"%.*s",(int) keylen,info->lastkey+1); +#endif + my_casedn_str(default_charset_info,buf); + total++; + lengths[keylen]++; + + if (count || stats) + { + doc_cnt++; + if (strcmp(buf, buf2)) + { + if (*buf2) + { + uniq++; + avg_gws+=gws=GWS_IN_USE; + if (count) + printf("%9u %20.7f %s\n",doc_cnt,gws,buf2); + if (maxlen=0) + printf("%9lx %20.7f %s\n", (long) info->lastpos,weight,buf); + else + printf("%9lx => %17d %s\n",(long) info->lastpos,-subkeys,buf); + } + if (verbose && (total%HOW_OFTEN_TO_WRITE)==0) + printf("%10ld\r",total); + } + maria_lock_database(info, F_UNLCK); + + if (count || stats) + { + doc_cnt++; + if (*buf2) + { + uniq++; + avg_gws+=gws=GWS_IN_USE; + if (count) + printf("%9u %20.7f %s\n",doc_cnt,gws,buf2); + if (maxlen= total/2) + break; + } + printf("Total rows: %lu\nTotal words: %lu\n" + "Unique words: %lu\nLongest word: %lu chars (%s)\n" + "Median length: %u\n" + "Average global weight: %f\n" + "Most common word: %lu times, weight: %f (%s)\n", + (long) info->state->records, total, uniq, maxlen, buf_maxlen, + inx, avg_gws/uniq, max_doc_cnt, min_gws, buf_min_gws); + } + if (lstats) + { + count=0; + for (inx=0; inx<256; inx++) + { + count+=lengths[inx]; + if (count && lengths[inx]) + printf("%3u: %10lu %5.2f%% %20lu %4.1f%%\n", inx, + (ulong) lengths[inx],100.0*lengths[inx]/total,(ulong) count, + 100.0*count/total); + } + } + +err: + if (error && error != HA_ERR_END_OF_FILE) + printf("got error %d\n",my_errno); + if (info) + maria_close(info); + maria_end(); + return 0; +} + + +static my_bool +get_one_option(int optid, const struct my_option *opt __attribute__((unused)), + char *argument __attribute__((unused))) +{ + switch(optid) { + case 'd': + dump=1; + complain(count || query); + break; + case 's': + stats=1; + complain(query!=0); + break; + case 'c': + count= 1; + complain(dump || query); + break; + case 'l': + lstats=1; + complain(query!=0); + break; + case '?': + case 'h': + usage(); + } + return 0; +} + +#include + +static void usage() +{ + printf("Use: maria_ft_dump \n"); + my_print_help(my_long_options); + my_print_variables(my_long_options); + NETWARE_SET_SCREEN_MODE(1); + exit(1); +} + +#include + +static void complain(int val) /* Kinda assert :-) */ +{ + if (val) + { + printf("You cannot use these options together!\n"); + exit(1); + } +} diff --git a/storage/maria/maria_log.c b/storage/maria/maria_log.c new file mode 100644 index 00000000000..72a4d7e89d5 --- /dev/null +++ b/storage/maria/maria_log.c @@ -0,0 +1,848 @@ +/* Copyright (C) 2006 MySQL AB & MySQL Finland AB & TCX DataKonsult AB + + 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; either version 2 of the License, or + (at your option) any later version. + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ + +/* write whats in isam.log */ + +#ifndef USE_MY_FUNC +#define USE_MY_FUNC +#endif + +#include "maria_def.h" +#include +#include +#ifdef HAVE_GETRUSAGE +#include +#endif + +#define FILENAME(A) (A ? A->show_name : "Unknown") + +struct file_info { + long process; + int filenr,id; + uint rnd; + my_string name,show_name,record; + MARIA_HA *isam; + bool closed,used; + ulong accessed; +}; + +struct test_if_open_param { + my_string name; + int max_id; +}; + +struct st_access_param +{ + ulong min_accessed; + struct file_info *found; +}; + +#define NO_FILEPOS (ulong) ~0L + +extern int main(int argc,char * *argv); +static void get_options(int *argc,char ***argv); +static int examine_log(my_string file_name,char **table_names); +static int read_string(IO_CACHE *file,gptr *to,uint length); +static int file_info_compare(void *cmp_arg, void *a,void *b); +static int test_if_open(struct file_info *key,element_count count, + struct test_if_open_param *param); +static void fix_blob_pointers(MARIA_HA *isam,byte *record); +static int test_when_accessed(struct file_info *key,element_count count, + struct st_access_param *access_param); +static void file_info_free(struct file_info *info); +static int close_some_file(TREE *tree); +static int reopen_closed_file(TREE *tree,struct file_info *file_info); +static int find_record_with_key(struct file_info *file_info,byte *record); +static void printf_log(const char *str,...); +static bool cmp_filename(struct file_info *file_info,my_string name); + +static uint verbose=0,update=0,test_info=0,max_files=0,re_open_count=0, + recover=0,prefix_remove=0,opt_processes=0; +static my_string log_filename=0,filepath=0,write_filename=0,record_pos_file=0; +static ulong com_count[10][3],number_of_commands=(ulong) ~0L, + isamlog_process; +static my_off_t isamlog_filepos,start_offset=0,record_pos= HA_OFFSET_ERROR; +static const char *command_name[]= +{"open","write","update","delete","close","extra","lock","re-open", + "delete-all", NullS}; + + +int main(int argc, char **argv) +{ + int error,i,first; + ulong total_count,total_error,total_recover; + MY_INIT(argv[0]); + + log_filename=maria_log_filename; + get_options(&argc,&argv); + maria_init(); + + /* Number of MARIA files we can have open at one time */ + max_files= (my_set_max_open_files(min(max_files,8))-6)/2; + if (update) + printf("Trying to %s MARIA files according to log '%s'\n", + (recover ? "recover" : "update"),log_filename); + error= examine_log(log_filename,argv); + if (update && ! error) + puts("Tables updated successfully"); + total_count=total_error=total_recover=0; + for (i=first=0 ; command_name[i] ; i++) + { + if (com_count[i][0]) + { + if (!first++) + { + if (verbose || update) + puts(""); + puts("Commands Used count Errors Recover errors"); + } + printf("%-12s%9ld%10ld%17ld\n",command_name[i],com_count[i][0], + com_count[i][1],com_count[i][2]); + total_count+=com_count[i][0]; + total_error+=com_count[i][1]; + total_recover+=com_count[i][2]; + } + } + if (total_count) + printf("%-12s%9ld%10ld%17ld\n","Total",total_count,total_error, + total_recover); + if (re_open_count) + printf("Had to do %d re-open because of too few possibly open files\n", + re_open_count); + VOID(maria_panic(HA_PANIC_CLOSE)); + my_free_open_file_info(); + maria_end(); + my_end(test_info ? MY_CHECK_ERROR | MY_GIVE_INFO : MY_CHECK_ERROR); + exit(error); + return 0; /* No compiler warning */ +} /* main */ + + +static void get_options(register int *argc, register char ***argv) +{ + int help,version; + const char *pos,*usage; + char option; + + help=0; + usage="Usage: %s [-?iruvDIV] [-c #] [-f #] [-F filepath/] [-o #] [-R file recordpos] [-w write_file] [log-filename [table ...]] \n"; + pos=""; + + while (--*argc > 0 && *(pos = *(++*argv)) == '-' ) { + while (*++pos) + { + version=0; + switch((option=*pos)) { + case '#': + DBUG_PUSH (++pos); + pos=" "; /* Skip rest of arg */ + break; + case 'c': + if (! *++pos) + { + if (!--*argc) + goto err; + else + pos= *(++*argv); + } + number_of_commands=(ulong) atol(pos); + pos=" "; + break; + case 'u': + update=1; + break; + case 'f': + if (! *++pos) + { + if (!--*argc) + goto err; + else + pos= *(++*argv); + } + max_files=(uint) atoi(pos); + pos=" "; + break; + case 'i': + test_info=1; + break; + case 'o': + if (! *++pos) + { + if (!--*argc) + goto err; + else + pos= *(++*argv); + } + start_offset=(my_off_t) strtoll(pos,NULL,10); + pos=" "; + break; + case 'p': + if (! *++pos) + { + if (!--*argc) + goto err; + else + pos= *(++*argv); + } + prefix_remove=atoi(pos); + break; + case 'r': + update=1; + recover++; + break; + case 'P': + opt_processes=1; + break; + case 'R': + if (! *++pos) + { + if (!--*argc) + goto err; + else + pos= *(++*argv); + } + record_pos_file=(char*) pos; + if (!--*argc) + goto err; + record_pos=(my_off_t) strtoll(*(++*argv),NULL,10); + pos=" "; + break; + case 'v': + verbose++; + break; + case 'w': + if (! *++pos) + { + if (!--*argc) + goto err; + else + pos= *(++*argv); + } + write_filename=(char*) pos; + pos=" "; + break; + case 'F': + if (! *++pos) + { + if (!--*argc) + goto err; + else + pos= *(++*argv); + } + filepath= (char*) pos; + pos=" "; + break; + case 'V': + version=1; + /* Fall through */ + case 'I': + case '?': +#include + printf("%s Ver 1.4 for %s at %s\n",my_progname,SYSTEM_TYPE, + MACHINE_TYPE); + puts("By Monty, for your professional use\n"); + if (version) + break; + puts("Write info about whats in a MARIA log file."); + printf("If no file name is given %s is used\n",log_filename); + puts(""); + printf(usage,my_progname); + puts(""); + puts("Options: -? or -I \"Info\" -V \"version\" -c \"do only # commands\""); + puts(" -f \"max open files\" -F \"filepath\" -i \"extra info\""); + puts(" -o \"offset\" -p # \"remove # components from path\""); + puts(" -r \"recover\" -R \"file recordposition\""); + puts(" -u \"update\" -v \"verbose\" -w \"write file\""); + puts(" -D \"maria compiled with DBUG\" -P \"processes\""); + puts("\nOne can give a second and a third '-v' for more verbose."); + puts("Normaly one does a update (-u)."); + puts("If a recover is done all writes and all possibly updates and deletes is done\nand errors are only counted."); + puts("If one gives table names as arguments only these tables will be updated\n"); + help=1; +#include + break; + default: + printf("illegal option: \"-%c\"\n",*pos); + break; + } + } + } + if (! *argc) + { + if (help) + exit(0); + (*argv)++; + } + if (*argc >= 1) + { + log_filename=(char*) pos; + (*argc)--; + (*argv)++; + } + return; + err: + VOID(fprintf(stderr,"option \"%c\" used without or with wrong argument\n", + option)); + exit(1); +} + + +static int examine_log(my_string file_name, char **table_names) +{ + uint command,result,files_open; + ulong access_time,length; + my_off_t filepos; + int lock_command,maria_result; + char isam_file_name[FN_REFLEN],llbuff[21],llbuff2[21]; + uchar head[20]; + gptr buff; + struct test_if_open_param open_param; + IO_CACHE cache; + File file; + FILE *write_file; + enum ha_extra_function extra_command; + TREE tree; + struct file_info file_info,*curr_file_info; + DBUG_ENTER("examine_log"); + + if ((file=my_open(file_name,O_RDONLY,MYF(MY_WME))) < 0) + DBUG_RETURN(1); + write_file=0; + if (write_filename) + { + if (!(write_file=my_fopen(write_filename,O_WRONLY,MYF(MY_WME)))) + { + my_close(file,MYF(0)); + DBUG_RETURN(1); + } + } + + init_io_cache(&cache,file,0,READ_CACHE,start_offset,0,MYF(0)); + bzero((gptr) com_count,sizeof(com_count)); + init_tree(&tree,0,0,sizeof(file_info),(qsort_cmp2) file_info_compare,1, + (tree_element_free) file_info_free, NULL); + VOID(init_key_cache(maria_key_cache,KEY_CACHE_BLOCK_SIZE,KEY_CACHE_SIZE, + 0, 0)); + + files_open=0; access_time=0; + while (access_time++ != number_of_commands && + !my_b_read(&cache,(byte*) head,9)) + { + isamlog_filepos=my_b_tell(&cache)-9L; + file_info.filenr= mi_uint2korr(head+1); + isamlog_process=file_info.process=(long) mi_uint4korr(head+3); + if (!opt_processes) + file_info.process=0; + result= mi_uint2korr(head+7); + if ((curr_file_info=(struct file_info*) tree_search(&tree, &file_info, + tree.custom_arg))) + { + curr_file_info->accessed=access_time; + if (update && curr_file_info->used && curr_file_info->closed) + { + if (reopen_closed_file(&tree,curr_file_info)) + { + command=sizeof(com_count)/sizeof(com_count[0][0])/3; + result=0; + goto com_err; + } + } + } + command=(uint) head[0]; + if (command < sizeof(com_count)/sizeof(com_count[0][0])/3 && + (!table_names[0] || (curr_file_info && curr_file_info->used))) + { + com_count[command][0]++; + if (result) + com_count[command][1]++; + } + switch ((enum maria_log_commands) command) { + case MARIA_LOG_OPEN: + if (!table_names[0]) + { + com_count[command][0]--; /* Must be counted explicite */ + if (result) + com_count[command][1]--; + } + + if (curr_file_info) + printf("\nWarning: %s is opened with same process and filenumber\nMaybe you should use the -P option ?\n", + curr_file_info->show_name); + if (my_b_read(&cache,(byte*) head,2)) + goto err; + file_info.name=0; + file_info.show_name=0; + file_info.record=0; + if (read_string(&cache,(gptr*) &file_info.name, + (uint) mi_uint2korr(head))) + goto err; + { + uint i; + char *pos,*to; + + /* Fix if old DOS files to new format */ + for (pos=file_info.name; (pos=strchr(pos,'\\')) ; pos++) + *pos= '/'; + + pos=file_info.name; + for (i=0 ; i < prefix_remove ; i++) + { + char *next; + if (!(next=strchr(pos,'/'))) + break; + pos=next+1; + } + to=isam_file_name; + if (filepath) + to=convert_dirname(isam_file_name,filepath,NullS); + strmov(to,pos); + fn_ext(isam_file_name)[0]=0; /* Remove extension */ + } + open_param.name=file_info.name; + open_param.max_id=0; + VOID(tree_walk(&tree,(tree_walk_action) test_if_open,(void*) &open_param, + left_root_right)); + file_info.id=open_param.max_id+1; + /* + * In the line below +10 is added to accomodate '<' and '>' chars + * plus '\0' at the end, so that there is place for 7 digits. + * It is improbable that same table can have that many entries in + * the table cache. + * The additional space is needed for the sprintf commands two lines + * below. + */ + file_info.show_name=my_memdup(isam_file_name, + (uint) strlen(isam_file_name)+10, + MYF(MY_WME)); + if (file_info.id > 1) + sprintf(strend(file_info.show_name),"<%d>",file_info.id); + file_info.closed=1; + file_info.accessed=access_time; + file_info.used=1; + if (table_names[0]) + { + char **name; + file_info.used=0; + for (name=table_names ; *name ; name++) + { + if (!strcmp(*name,isam_file_name)) + file_info.used=1; /* Update/log only this */ + } + } + if (update && file_info.used) + { + if (files_open >= max_files) + { + if (close_some_file(&tree)) + goto com_err; + files_open--; + } + if (!(file_info.isam= maria_open(isam_file_name,O_RDWR, + HA_OPEN_WAIT_IF_LOCKED))) + goto com_err; + if (!(file_info.record=my_malloc(file_info.isam->s->base.reclength, + MYF(MY_WME)))) + goto end; + files_open++; + file_info.closed=0; + } + VOID(tree_insert(&tree, (gptr) &file_info, 0, tree.custom_arg)); + if (file_info.used) + { + if (verbose && !record_pos_file) + printf_log("%s: open -> %d",file_info.show_name, file_info.filenr); + com_count[command][0]++; + if (result) + com_count[command][1]++; + } + break; + case MARIA_LOG_CLOSE: + if (verbose && !record_pos_file && + (!table_names[0] || (curr_file_info && curr_file_info->used))) + printf_log("%s: %s -> %d",FILENAME(curr_file_info), + command_name[command],result); + if (curr_file_info) + { + if (!curr_file_info->closed) + files_open--; + VOID(tree_delete(&tree, (gptr) curr_file_info, tree.custom_arg)); + } + break; + case MARIA_LOG_EXTRA: + if (my_b_read(&cache,(byte*) head,1)) + goto err; + extra_command=(enum ha_extra_function) head[0]; + if (verbose && !record_pos_file && + (!table_names[0] || (curr_file_info && curr_file_info->used))) + printf_log("%s: %s(%d) -> %d",FILENAME(curr_file_info), + command_name[command], (int) extra_command,result); + if (update && curr_file_info && !curr_file_info->closed) + { + if (maria_extra(curr_file_info->isam, extra_command, 0) != (int) result) + { + fflush(stdout); + VOID(fprintf(stderr, + "Warning: error %d, expected %d on command %s at %s\n", + my_errno,result,command_name[command], + llstr(isamlog_filepos,llbuff))); + fflush(stderr); + } + } + break; + case MARIA_LOG_DELETE: + if (my_b_read(&cache,(byte*) head,8)) + goto err; + filepos=mi_sizekorr(head); + if (verbose && (!record_pos_file || + ((record_pos == filepos || record_pos == NO_FILEPOS) && + !cmp_filename(curr_file_info,record_pos_file))) && + (!table_names[0] || (curr_file_info && curr_file_info->used))) + printf_log("%s: %s at %ld -> %d",FILENAME(curr_file_info), + command_name[command],(long) filepos,result); + if (update && curr_file_info && !curr_file_info->closed) + { + if (maria_rrnd(curr_file_info->isam,curr_file_info->record,filepos)) + { + if (!recover) + goto com_err; + if (verbose) + printf_log("error: Didn't find row to delete with maria_rrnd"); + com_count[command][2]++; /* Mark error */ + } + maria_result=maria_delete(curr_file_info->isam,curr_file_info->record); + if ((maria_result == 0 && result) || + (maria_result && (uint) my_errno != result)) + { + if (!recover) + goto com_err; + if (maria_result) + com_count[command][2]++; /* Mark error */ + if (verbose) + printf_log("error: Got result %d from maria_delete instead of %d", + maria_result, result); + } + } + break; + case MARIA_LOG_WRITE: + case MARIA_LOG_UPDATE: + if (my_b_read(&cache,(byte*) head,12)) + goto err; + filepos=mi_sizekorr(head); + length=mi_uint4korr(head+8); + buff=0; + if (read_string(&cache,&buff,(uint) length)) + goto err; + if ((!record_pos_file || + ((record_pos == filepos || record_pos == NO_FILEPOS) && + !cmp_filename(curr_file_info,record_pos_file))) && + (!table_names[0] || (curr_file_info && curr_file_info->used))) + { + if (write_file && + (my_fwrite(write_file,buff,length,MYF(MY_WAIT_IF_FULL | MY_NABP)))) + goto end; + if (verbose) + printf_log("%s: %s at %ld, length=%ld -> %d", + FILENAME(curr_file_info), + command_name[command], filepos,length,result); + } + if (update && curr_file_info && !curr_file_info->closed) + { + if (curr_file_info->isam->s->base.blobs) + fix_blob_pointers(curr_file_info->isam,buff); + if ((enum maria_log_commands) command == MARIA_LOG_UPDATE) + { + if (maria_rrnd(curr_file_info->isam,curr_file_info->record,filepos)) + { + if (!recover) + { + result=0; + goto com_err; + } + if (verbose) + printf_log("error: Didn't find row to update with maria_rrnd"); + if (recover == 1 || result || + find_record_with_key(curr_file_info,buff)) + { + com_count[command][2]++; /* Mark error */ + break; + } + } + maria_result=maria_update(curr_file_info->isam,curr_file_info->record, + buff); + if ((maria_result == 0 && result) || + (maria_result && (uint) my_errno != result)) + { + if (!recover) + goto com_err; + if (verbose) + printf_log("error: Got result %d from maria_update instead of %d", + maria_result, result); + if (maria_result) + com_count[command][2]++; /* Mark error */ + } + } + else + { + maria_result=maria_write(curr_file_info->isam,buff); + if ((maria_result == 0 && result) || + (maria_result && (uint) my_errno != result)) + { + if (!recover) + goto com_err; + if (verbose) + printf_log("error: Got result %d from maria_write instead of %d", + maria_result, result); + if (maria_result) + com_count[command][2]++; /* Mark error */ + } + if (!recover && filepos != curr_file_info->isam->lastpos) + { + printf("error: Wrote at position: %s, should have been %s", + llstr(curr_file_info->isam->lastpos,llbuff), + llstr(filepos,llbuff2)); + goto end; + } + } + } + my_free(buff,MYF(0)); + break; + case MARIA_LOG_LOCK: + if (my_b_read(&cache,(byte*) head,sizeof(lock_command))) + goto err; + memcpy_fixed(&lock_command,head,sizeof(lock_command)); + if (verbose && !record_pos_file && + (!table_names[0] || (curr_file_info && curr_file_info->used))) + printf_log("%s: %s(%d) -> %d\n",FILENAME(curr_file_info), + command_name[command],lock_command,result); + if (update && curr_file_info && !curr_file_info->closed) + { + if (maria_lock_database(curr_file_info->isam,lock_command) != + (int) result) + goto com_err; + } + break; + case MARIA_LOG_DELETE_ALL: + if (verbose && !record_pos_file && + (!table_names[0] || (curr_file_info && curr_file_info->used))) + printf_log("%s: %s -> %d\n",FILENAME(curr_file_info), + command_name[command],result); + break; + default: + fflush(stdout); + VOID(fprintf(stderr, + "Error: found unknown command %d in logfile, aborted\n", + command)); + fflush(stderr); + goto end; + } + } + end_key_cache(maria_key_cache,1); + delete_tree(&tree); + VOID(end_io_cache(&cache)); + VOID(my_close(file,MYF(0))); + if (write_file && my_fclose(write_file,MYF(MY_WME))) + DBUG_RETURN(1); + DBUG_RETURN(0); + + err: + fflush(stdout); + VOID(fprintf(stderr,"Got error %d when reading from logfile\n",my_errno)); + fflush(stderr); + goto end; + com_err: + fflush(stdout); + VOID(fprintf(stderr,"Got error %d, expected %d on command %s at %s\n", + my_errno,result,command_name[command], + llstr(isamlog_filepos,llbuff))); + fflush(stderr); + end: + end_key_cache(maria_key_cache, 1); + delete_tree(&tree); + VOID(end_io_cache(&cache)); + VOID(my_close(file,MYF(0))); + if (write_file) + VOID(my_fclose(write_file,MYF(MY_WME))); + DBUG_RETURN(1); +} + + +static int read_string(IO_CACHE *file, register gptr *to, register uint length) +{ + DBUG_ENTER("read_string"); + + if (*to) + my_free((gptr) *to,MYF(0)); + if (!(*to= (gptr) my_malloc(length+1,MYF(MY_WME))) || + my_b_read(file,(byte*) *to,length)) + { + if (*to) + my_free(*to,MYF(0)); + *to= 0; + DBUG_RETURN(1); + } + *((char*) *to+length)= '\0'; + DBUG_RETURN (0); +} /* read_string */ + + +static int file_info_compare(void* cmp_arg __attribute__((unused)), + void *a, void *b) +{ + long lint; + + if ((lint=((struct file_info*) a)->process - + ((struct file_info*) b)->process)) + return lint < 0L ? -1 : 1; + return ((struct file_info*) a)->filenr - ((struct file_info*) b)->filenr; +} + + /* ARGSUSED */ + +static int test_if_open (struct file_info *key, + element_count count __attribute__((unused)), + struct test_if_open_param *param) +{ + if (!strcmp(key->name,param->name) && key->id > param->max_id) + param->max_id=key->id; + return 0; +} + + +static void fix_blob_pointers(MARIA_HA *info, byte *record) +{ + byte *pos; + MARIA_BLOB *blob,*end; + + pos=record+info->s->base.reclength; + for (end=info->blobs+info->s->base.blobs, blob= info->blobs; + blob != end ; + blob++) + { + memcpy_fixed(record+blob->offset+blob->pack_length,&pos,sizeof(char*)); + pos+= _ma_calc_blob_length(blob->pack_length,record+blob->offset); + } +} + + /* close the file with hasn't been accessed for the longest time */ + /* ARGSUSED */ + +static int test_when_accessed (struct file_info *key, + element_count count __attribute__((unused)), + struct st_access_param *access_param) +{ + if (key->accessed < access_param->min_accessed && ! key->closed) + { + access_param->min_accessed=key->accessed; + access_param->found=key; + } + return 0; +} + + +static void file_info_free(struct file_info *fileinfo) +{ + DBUG_ENTER("file_info_free"); + if (update) + { + if (!fileinfo->closed) + VOID(maria_close(fileinfo->isam)); + if (fileinfo->record) + my_free(fileinfo->record,MYF(0)); + } + my_free(fileinfo->name,MYF(0)); + my_free(fileinfo->show_name,MYF(0)); + DBUG_VOID_RETURN; +} + + + +static int close_some_file(TREE *tree) +{ + struct st_access_param access_param; + + access_param.min_accessed=LONG_MAX; + access_param.found=0; + + VOID(tree_walk(tree,(tree_walk_action) test_when_accessed, + (void*) &access_param,left_root_right)); + if (!access_param.found) + return 1; /* No open file that is possibly to close */ + if (maria_close(access_param.found->isam)) + return 1; + access_param.found->closed=1; + return 0; +} + + +static int reopen_closed_file(TREE *tree, struct file_info *fileinfo) +{ + char name[FN_REFLEN]; + if (close_some_file(tree)) + return 1; /* No file to close */ + strmov(name,fileinfo->show_name); + if (fileinfo->id > 1) + *strrchr(name,'<')='\0'; /* Remove "" */ + + if (!(fileinfo->isam= maria_open(name,O_RDWR,HA_OPEN_WAIT_IF_LOCKED))) + return 1; + fileinfo->closed=0; + re_open_count++; + return 0; +} + + /* Try to find record with uniq key */ + +static int find_record_with_key(struct file_info *file_info, byte *record) +{ + uint key; + MARIA_HA *info=file_info->isam; + uchar tmp_key[HA_MAX_KEY_BUFF]; + + for (key=0 ; key < info->s->base.keys ; key++) + { + if (maria_is_key_active(info->s->state.key_map, key) && + info->s->keyinfo[key].flag & HA_NOSAME) + { + VOID(_ma_make_key(info,key,tmp_key,record,0L)); + return maria_rkey(info,file_info->record,(int) key,(char*) tmp_key,0, + HA_READ_KEY_EXACT); + } + } + return 1; +} + + +static void printf_log(const char *format,...) +{ + char llbuff[21]; + va_list args; + va_start(args,format); + if (verbose > 2) + printf("%9s:",llstr(isamlog_filepos,llbuff)); + if (verbose > 1) + printf("%5ld ",isamlog_process); /* Write process number */ + (void) vprintf((char*) format,args); + putchar('\n'); + va_end(args); +} + + +static bool cmp_filename(struct file_info *file_info, my_string name) +{ + if (!file_info) + return 1; + return strcmp(file_info->name,name) ? 1 : 0; +} diff --git a/storage/maria/maria_pack.c b/storage/maria/maria_pack.c new file mode 100644 index 00000000000..2a83bbb0f3f --- /dev/null +++ b/storage/maria/maria_pack.c @@ -0,0 +1,3202 @@ +/* Copyright (C) 2006 MySQL AB & MySQL Finland AB & TCX DataKonsult AB + + 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; either version 2 of the License, or + (at your option) any later version. + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ + +/* Pack MARIA file */ + +#ifndef USE_MY_FUNC +#define USE_MY_FUNC /* We need at least my_malloc */ +#endif + +#include "maria_def.h" +#include +#include +#include "mysys_err.h" +#ifdef MSDOS +#include +#endif +#ifndef __GNU_LIBRARY__ +#define __GNU_LIBRARY__ /* Skip warnings in getopt.h */ +#endif +#include +#include + +#if SIZEOF_LONG_LONG > 4 +#define BITS_SAVED 64 +#else +#define BITS_SAVED 32 +#endif + +#define IS_OFFSET ((uint) 32768) /* Bit if offset or char in tree */ +#define HEAD_LENGTH 32 +#define ALLOWED_JOIN_DIFF 256 /* Diff allowed to join trees */ + +#define DATA_TMP_EXT ".TMD" +#define OLD_EXT ".OLD" +#define WRITE_COUNT MY_HOW_OFTEN_TO_WRITE + +struct st_file_buffer { + File file; + uchar *buffer,*pos,*end; + my_off_t pos_in_file; + int bits; + ulonglong bitbucket; +}; + +struct st_huff_tree; +struct st_huff_element; + +typedef struct st_huff_counts { + uint field_length,max_zero_fill; + uint pack_type; + uint max_end_space,max_pre_space,length_bits,min_space; + ulong max_length; + enum en_fieldtype field_type; + struct st_huff_tree *tree; /* Tree for field */ + my_off_t counts[256]; + my_off_t end_space[8]; + my_off_t pre_space[8]; + my_off_t tot_end_space,tot_pre_space,zero_fields,empty_fields,bytes_packed; + TREE int_tree; /* Tree for detecting distinct column values. */ + byte *tree_buff; /* Column values, 'field_length' each. */ + byte *tree_pos; /* Points to end of column values in 'tree_buff'. */ +} HUFF_COUNTS; + +typedef struct st_huff_element HUFF_ELEMENT; + +/* + WARNING: It is crucial for the optimizations in calc_packed_length() + that 'count' is the first element of 'HUFF_ELEMENT'. +*/ +struct st_huff_element { + my_off_t count; + union un_element { + struct st_nod { + HUFF_ELEMENT *left,*right; + } nod; + struct st_leaf { + HUFF_ELEMENT *null; + uint element_nr; /* Number of element */ + } leaf; + } a; +}; + + +typedef struct st_huff_tree { + HUFF_ELEMENT *root,*element_buffer; + HUFF_COUNTS *counts; + uint tree_number; + uint elements; + my_off_t bytes_packed; + uint tree_pack_length; + uint min_chr,max_chr,char_bits,offset_bits,max_offset,height; + ulonglong *code; + uchar *code_len; +} HUFF_TREE; + + +typedef struct st_isam_mrg { + MARIA_HA **file,**current,**end; + uint free_file; + uint count; + uint min_pack_length; /* Theese is used by packed data */ + uint max_pack_length; + uint ref_length; + uint max_blob_length; + my_off_t records; + /* true if at least one source file has at least one disabled index */ + my_bool src_file_has_indexes_disabled; +} PACK_MRG_INFO; + + +extern int main(int argc,char * *argv); +static void get_options(int *argc,char ***argv); +static MARIA_HA *open_isam_file(char *name,int mode); +static bool open_isam_files(PACK_MRG_INFO *mrg,char **names,uint count); +static int compress(PACK_MRG_INFO *file,char *join_name); +static HUFF_COUNTS *init_huff_count(MARIA_HA *info,my_off_t records); +static void free_counts_and_tree_and_queue(HUFF_TREE *huff_trees, + uint trees, + HUFF_COUNTS *huff_counts, + uint fields); +static int compare_tree(void* cmp_arg __attribute__((unused)), + const uchar *s,const uchar *t); +static int get_statistic(PACK_MRG_INFO *mrg,HUFF_COUNTS *huff_counts); +static void check_counts(HUFF_COUNTS *huff_counts,uint trees, + my_off_t records); +static int test_space_compress(HUFF_COUNTS *huff_counts,my_off_t records, + uint max_space_length,my_off_t *space_counts, + my_off_t tot_space_count, + enum en_fieldtype field_type); +static HUFF_TREE* make_huff_trees(HUFF_COUNTS *huff_counts,uint trees); +static int make_huff_tree(HUFF_TREE *tree,HUFF_COUNTS *huff_counts); +static int compare_huff_elements(void *not_used, byte *a,byte *b); +static int save_counts_in_queue(byte *key,element_count count, + HUFF_TREE *tree); +static my_off_t calc_packed_length(HUFF_COUNTS *huff_counts,uint flag); +static uint join_same_trees(HUFF_COUNTS *huff_counts,uint trees); +static int make_huff_decode_table(HUFF_TREE *huff_tree,uint trees); +static void make_traverse_code_tree(HUFF_TREE *huff_tree, + HUFF_ELEMENT *element,uint size, + ulonglong code); +static int write_header(PACK_MRG_INFO *isam_file, uint header_length,uint trees, + my_off_t tot_elements,my_off_t filelength); +static void write_field_info(HUFF_COUNTS *counts, uint fields,uint trees); +static my_off_t write_huff_tree(HUFF_TREE *huff_tree,uint trees); +static uint *make_offset_code_tree(HUFF_TREE *huff_tree, + HUFF_ELEMENT *element, + uint *offset); +static uint max_bit(uint value); +static int compress_isam_file(PACK_MRG_INFO *file,HUFF_COUNTS *huff_counts); +static char *make_new_name(char *new_name,char *old_name); +static char *make_old_name(char *new_name,char *old_name); +static void init_file_buffer(File file,pbool read_buffer); +static int flush_buffer(ulong neaded_length); +static void end_file_buffer(void); +static void write_bits(ulonglong value, uint bits); +static void flush_bits(void); +static int save_state(MARIA_HA *isam_file,PACK_MRG_INFO *mrg,my_off_t new_length, + ha_checksum crc); +static int save_state_mrg(File file,PACK_MRG_INFO *isam_file,my_off_t new_length, + ha_checksum crc); +static int mrg_close(PACK_MRG_INFO *mrg); +static int mrg_rrnd(PACK_MRG_INFO *info,byte *buf); +static void mrg_reset(PACK_MRG_INFO *mrg); +#if !defined(DBUG_OFF) +static void fakebigcodes(HUFF_COUNTS *huff_counts, HUFF_COUNTS *end_count); +static int fakecmp(my_off_t **count1, my_off_t **count2); +#endif + + +static int error_on_write=0,test_only=0,verbose=0,silent=0, + write_loop=0,force_pack=0, isamchk_neaded=0; +static int tmpfile_createflag=O_RDWR | O_TRUNC | O_EXCL; +static my_bool backup, opt_wait; +/* + tree_buff_length is somewhat arbitrary. The bigger it is the better + the chance to win in terms of compression factor. On the other hand, + this table becomes part of the compressed file header. And its length + is coded with 16 bits in the header. Hence the limit is 2**16 - 1. +*/ +static uint tree_buff_length= 65536 - MALLOC_OVERHEAD; +static char tmp_dir[FN_REFLEN]={0},*join_table; +static my_off_t intervall_length; +static ha_checksum glob_crc; +static struct st_file_buffer file_buffer; +static QUEUE queue; +static HUFF_COUNTS *global_count; +static char zero_string[]={0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0}; +static const char *load_default_groups[]= { "mariapack",0 }; + + /* The main program */ + +int main(int argc, char **argv) +{ + int error,ok; + PACK_MRG_INFO merge; + char **default_argv; + MY_INIT(argv[0]); + + load_defaults("my",load_default_groups,&argc,&argv); + default_argv= argv; + get_options(&argc,&argv); + maria_init(); + + error=ok=isamchk_neaded=0; + if (join_table) + { /* Join files into one */ + if (open_isam_files(&merge,argv,(uint) argc) || + compress(&merge,join_table)) + error=1; + } + else while (argc--) + { + MARIA_HA *isam_file; + if (!(isam_file=open_isam_file(*argv++,O_RDWR))) + error=1; + else + { + merge.file= &isam_file; + merge.current=0; + merge.free_file=0; + merge.count=1; + if (compress(&merge,0)) + error=1; + else + ok=1; + } + } + if (ok && isamchk_neaded && !silent) + puts("Remember to run mariachk -rq on compressed tables"); + VOID(fflush(stdout)); + VOID(fflush(stderr)); + free_defaults(default_argv); + maria_end(); + my_end(verbose ? MY_CHECK_ERROR | MY_GIVE_INFO : MY_CHECK_ERROR); + exit(error ? 2 : 0); +#ifndef _lint + return 0; /* No compiler warning */ +#endif +} + +enum options_mp {OPT_CHARSETS_DIR_MP=256, OPT_AUTO_CLOSE}; + +static struct my_option my_long_options[] = +{ +#ifdef __NETWARE__ + {"autoclose", OPT_AUTO_CLOSE, "Auto close the screen on exit for Netware.", + 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, +#endif + {"backup", 'b', "Make a backup of the table as table_name.OLD.", + (gptr*) &backup, (gptr*) &backup, 0, GET_BOOL, NO_ARG, 0, 0, 0, 0, 0, 0}, + {"character-sets-dir", OPT_CHARSETS_DIR_MP, + "Directory where character sets are.", (gptr*) &charsets_dir, + (gptr*) &charsets_dir, 0, GET_STR, REQUIRED_ARG, 0, 0, 0, 0, 0, 0}, + {"debug", '#', "Output debug log. Often this is 'd:t:o,filename'.", + 0, 0, 0, GET_STR, OPT_ARG, 0, 0, 0, 0, 0, 0}, + {"force", 'f', + "Force packing of table even if it gets bigger or if tempfile exists.", + 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, + {"join", 'j', + "Join all given tables into 'new_table_name'. All tables MUST have identical layouts.", + (gptr*) &join_table, (gptr*) &join_table, 0, GET_STR, REQUIRED_ARG, 0, 0, 0, + 0, 0, 0}, + {"help", '?', "Display this help and exit.", + 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, + {"silent", 's', "Be more silent.", + 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, + {"tmpdir", 'T', "Use temporary directory to store temporary table.", + 0, 0, 0, GET_STR, REQUIRED_ARG, 0, 0, 0, 0, 0, 0}, + {"test", 't', "Don't pack table, only test packing it.", + 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, + {"verbose", 'v', "Write info about progress and packing result. Use many -v for more verbosity!", + 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, + {"version", 'V', "Output version information and exit.", + 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, + {"wait", 'w', "Wait and retry if table is in use.", (gptr*) &opt_wait, + (gptr*) &opt_wait, 0, GET_BOOL, NO_ARG, 0, 0, 0, 0, 0, 0}, + { 0, 0, 0, 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0} +}; + +#include + +static void print_version(void) +{ + VOID(printf("%s Ver 1.23 for %s on %s\n", + my_progname, SYSTEM_TYPE, MACHINE_TYPE)); + NETWARE_SET_SCREEN_MODE(1); +} + + +static void usage(void) +{ + print_version(); + puts("Copyright (C) 2002 MySQL AB"); + puts("This software comes with ABSOLUTELY NO WARRANTY. This is free software,"); + puts("and you are welcome to modify and redistribute it under the GPL license\n"); + + puts("Pack a MARIA-table to take much less space."); + puts("Keys are not updated, you must run mariachk -rq on the datafile"); + puts("afterwards to update the keys."); + puts("You should give the .MYI file as the filename argument."); + + VOID(printf("\nUsage: %s [OPTIONS] filename...\n", my_progname)); + my_print_help(my_long_options); + print_defaults("my", load_default_groups); + my_print_variables(my_long_options); +} + +#include + +static my_bool +get_one_option(int optid, const struct my_option *opt __attribute__((unused)), + char *argument) +{ + uint length; + + switch(optid) { +#ifdef __NETWARE__ + case OPT_AUTO_CLOSE: + setscreenmode(SCR_AUTOCLOSE_ON_EXIT); + break; +#endif + case 'f': + force_pack= 1; + tmpfile_createflag= O_RDWR | O_TRUNC; + break; + case 's': + write_loop= verbose= 0; + silent= 1; + break; + case 't': + test_only= 1; + /* Avoid to reset 'verbose' if it was already set > 1. */ + if (! verbose) + verbose= 1; + break; + case 'T': + length= (uint) (strmov(tmp_dir, argument) - tmp_dir); + if (length != dirname_length(tmp_dir)) + { + tmp_dir[length]=FN_LIBCHAR; + tmp_dir[length+1]=0; + } + break; + case 'v': + verbose++; /* Allow for selecting the level of verbosity. */ + silent= 0; + break; + case '#': + DBUG_PUSH(argument ? argument : "d:t:o"); + break; + case 'V': + print_version(); + exit(0); + case 'I': + case '?': + usage(); + exit(0); + } + return 0; +} + + /* reads options */ + /* Initiates DEBUG - but no debugging here ! */ + +static void get_options(int *argc,char ***argv) +{ + int ho_error; + + my_progname= argv[0][0]; + if (isatty(fileno(stdout))) + write_loop=1; + + if ((ho_error=handle_options(argc, argv, my_long_options, get_one_option))) + exit(ho_error); + + if (!*argc) + { + usage(); + exit(1); + } + if (join_table) + { + backup=0; /* Not needed */ + tmp_dir[0]=0; + } + return; +} + + +static MARIA_HA *open_isam_file(char *name,int mode) +{ + MARIA_HA *isam_file; + MARIA_SHARE *share; + DBUG_ENTER("open_isam_file"); + + if (!(isam_file=maria_open(name,mode, + (opt_wait ? HA_OPEN_WAIT_IF_LOCKED : + HA_OPEN_ABORT_IF_LOCKED)))) + { + VOID(fprintf(stderr, "%s gave error %d on open\n", name, my_errno)); + DBUG_RETURN(0); + } + share=isam_file->s; + if (share->options & HA_OPTION_COMPRESS_RECORD && !join_table) + { + if (!force_pack) + { + VOID(fprintf(stderr, "%s is already compressed\n", name)); + VOID(maria_close(isam_file)); + DBUG_RETURN(0); + } + if (verbose) + puts("Recompressing already compressed table"); + share->options&= ~HA_OPTION_READ_ONLY_DATA; /* We are modifing it */ + } + if (! force_pack && share->state.state.records != 0 && + (share->state.state.records <= 1 || + share->state.state.data_file_length < 1024)) + { + VOID(fprintf(stderr, "%s is too small to compress\n", name)); + VOID(maria_close(isam_file)); + DBUG_RETURN(0); + } + VOID(maria_lock_database(isam_file,F_WRLCK)); + DBUG_RETURN(isam_file); +} + + +static bool open_isam_files(PACK_MRG_INFO *mrg,char **names,uint count) +{ + uint i,j; + mrg->count=0; + mrg->current=0; + mrg->file=(MARIA_HA**) my_malloc(sizeof(MARIA_HA*)*count,MYF(MY_FAE)); + mrg->free_file=1; + mrg->src_file_has_indexes_disabled= 0; + for (i=0; i < count ; i++) + { + if (!(mrg->file[i]=open_isam_file(names[i],O_RDONLY))) + goto error; + + mrg->src_file_has_indexes_disabled|= + ! maria_is_all_keys_active(mrg->file[i]->s->state.key_map, + mrg->file[i]->s->base.keys); + } + /* Check that files are identical */ + for (j=0 ; j < count-1 ; j++) + { + MARIA_COLUMNDEF *m1,*m2,*end; + if (mrg->file[j]->s->base.reclength != mrg->file[j+1]->s->base.reclength || + mrg->file[j]->s->base.fields != mrg->file[j+1]->s->base.fields) + goto diff_file; + m1=mrg->file[j]->s->rec; + end=m1+mrg->file[j]->s->base.fields; + m2=mrg->file[j+1]->s->rec; + for ( ; m1 != end ; m1++,m2++) + { + if (m1->type != m2->type || m1->length != m2->length) + goto diff_file; + } + } + mrg->count=count; + return 0; + + diff_file: + VOID(fprintf(stderr, "%s: Tables '%s' and '%s' are not identical\n", + my_progname, names[j], names[j+1])); + error: + while (i--) + maria_close(mrg->file[i]); + my_free((gptr) mrg->file,MYF(0)); + return 1; +} + + +static int compress(PACK_MRG_INFO *mrg,char *result_table) +{ + int error; + File new_file,join_isam_file; + MARIA_HA *isam_file; + MARIA_SHARE *share; + char org_name[FN_REFLEN],new_name[FN_REFLEN],temp_name[FN_REFLEN]; + uint i,header_length,fields,trees,used_trees; + my_off_t old_length,new_length,tot_elements; + HUFF_COUNTS *huff_counts; + HUFF_TREE *huff_trees; + DBUG_ENTER("compress"); + + isam_file=mrg->file[0]; /* Take this as an example */ + share=isam_file->s; + new_file=join_isam_file= -1; + trees=fields=0; + huff_trees=0; + huff_counts=0; + + /* Create temporary or join file */ + + if (backup) + VOID(fn_format(org_name,isam_file->filename,"",MARIA_NAME_DEXT,2)); + else + VOID(fn_format(org_name,isam_file->filename,"",MARIA_NAME_DEXT,2+4+16)); + if (!test_only && result_table) + { + /* Make a new indexfile based on first file in list */ + uint length; + char *buff; + strmov(org_name,result_table); /* Fix error messages */ + VOID(fn_format(new_name,result_table,"",MARIA_NAME_IEXT,2)); + if ((join_isam_file=my_create(new_name,0,tmpfile_createflag,MYF(MY_WME))) + < 0) + goto err; + length=(uint) share->base.keystart; + if (!(buff=my_malloc(length,MYF(MY_WME)))) + goto err; + if (my_pread(share->kfile,buff,length,0L,MYF(MY_WME | MY_NABP)) || + my_write(join_isam_file,buff,length, + MYF(MY_WME | MY_NABP | MY_WAIT_IF_FULL))) + { + my_free(buff,MYF(0)); + goto err; + } + my_free(buff,MYF(0)); + VOID(fn_format(new_name,result_table,"",MARIA_NAME_DEXT,2)); + } + else if (!tmp_dir[0]) + VOID(make_new_name(new_name,org_name)); + else + VOID(fn_format(new_name,org_name,tmp_dir,DATA_TMP_EXT,1+2+4)); + if (!test_only && + (new_file=my_create(new_name,0,tmpfile_createflag,MYF(MY_WME))) < 0) + goto err; + + /* Start calculating statistics */ + + mrg->records=0; + for (i=0 ; i < mrg->count ; i++) + mrg->records+=mrg->file[i]->s->state.state.records; + + DBUG_PRINT("info", ("Compressing %s: (%lu records)", + result_table ? new_name : org_name, + (ulong) mrg->records)); + if (write_loop || verbose) + { + VOID(printf("Compressing %s: (%lu records)\n", + result_table ? new_name : org_name, (ulong) mrg->records)); + } + trees=fields=share->base.fields; + huff_counts=init_huff_count(isam_file,mrg->records); + QUICK_SAFEMALLOC; + + /* + Read the whole data file(s) for statistics. + */ + DBUG_PRINT("info", ("- Calculating statistics")); + if (write_loop || verbose) + VOID(printf("- Calculating statistics\n")); + if (get_statistic(mrg,huff_counts)) + goto err; + NORMAL_SAFEMALLOC; + old_length=0; + for (i=0; i < mrg->count ; i++) + old_length+= (mrg->file[i]->s->state.state.data_file_length - + mrg->file[i]->s->state.state.empty); + + /* + Create a global priority queue in preparation for making + temporary Huffman trees. + */ + if (init_queue(&queue,256,0,0,compare_huff_elements,0)) + goto err; + + /* + Check each column if we should use pre-space-compress, end-space- + compress, empty-field-compress or zero-field-compress. + */ + check_counts(huff_counts,fields,mrg->records); + + /* + Build a Huffman tree for each column. + */ + huff_trees=make_huff_trees(huff_counts,trees); + + /* + If the packed lengths of combined columns is less then the sum of + the non-combined columns, then create common Huffman trees for them. + We do this only for byte compressed columns, not for distinct values + compressed columns. + */ + if ((int) (used_trees=join_same_trees(huff_counts,trees)) < 0) + goto err; + + /* + Assign codes to all byte or column values. + */ + if (make_huff_decode_table(huff_trees,fields)) + goto err; + + /* Prepare a file buffer. */ + init_file_buffer(new_file,0); + + /* + Reserve space in the target file for the fixed compressed file header. + */ + file_buffer.pos_in_file=HEAD_LENGTH; + if (! test_only) + VOID(my_seek(new_file,file_buffer.pos_in_file,MY_SEEK_SET,MYF(0))); + + /* + Write field infos: field type, pack type, length bits, tree number. + */ + write_field_info(huff_counts,fields,used_trees); + + /* + Write decode trees. + */ + if (!(tot_elements=write_huff_tree(huff_trees,trees))) + goto err; + + /* + Calculate the total length of the compression info header. + This includes the fixed compressed file header, the column compression + type descriptions, and the decode trees. + */ + header_length=(uint) file_buffer.pos_in_file+ + (uint) (file_buffer.pos-file_buffer.buffer); + + /* + Compress the source file into the target file. + */ + DBUG_PRINT("info", ("- Compressing file")); + if (write_loop || verbose) + VOID(printf("- Compressing file\n")); + error=compress_isam_file(mrg,huff_counts); + new_length=file_buffer.pos_in_file; + if (!error && !test_only) + { + char buff[MEMMAP_EXTRA_MARGIN]; /* End marginal for memmap */ + bzero(buff,sizeof(buff)); + error=my_write(file_buffer.file,buff,sizeof(buff), + MYF(MY_WME | MY_NABP | MY_WAIT_IF_FULL)) != 0; + } + + /* + Write the fixed compressed file header. + */ + if (!error) + error=write_header(mrg,header_length,used_trees,tot_elements, + new_length); + + /* Flush the file buffer. */ + end_file_buffer(); + + /* Display statistics. */ + DBUG_PRINT("info", ("Min record length: %6d Max length: %6d " + "Mean total length: %6ld\n", + mrg->min_pack_length, mrg->max_pack_length, + (ulong) (mrg->records ? (new_length/mrg->records) : 0))); + if (verbose && mrg->records) + VOID(printf("Min record length: %6d Max length: %6d " + "Mean total length: %6ld\n", mrg->min_pack_length, + mrg->max_pack_length, (ulong) (new_length/mrg->records))); + + /* Close source and target file. */ + if (!test_only) + { + error|=my_close(new_file,MYF(MY_WME)); + if (!result_table) + { + error|=my_close(isam_file->dfile,MYF(MY_WME)); + isam_file->dfile= -1; /* Tell maria_close file is closed */ + } + } + + /* Cleanup. */ + free_counts_and_tree_and_queue(huff_trees,trees,huff_counts,fields); + if (! test_only && ! error) + { + if (result_table) + { + error=save_state_mrg(join_isam_file,mrg,new_length,glob_crc); + } + else + { + if (backup) + { + if (my_rename(org_name,make_old_name(temp_name,isam_file->filename), + MYF(MY_WME))) + error=1; + else + { + if (tmp_dir[0]) + error=my_copy(new_name,org_name,MYF(MY_WME)); + else + error=my_rename(new_name,org_name,MYF(MY_WME)); + if (!error) + { + VOID(my_copystat(temp_name,org_name,MYF(MY_COPYTIME))); + if (tmp_dir[0]) + VOID(my_delete(new_name,MYF(MY_WME))); + } + } + } + else + { + if (tmp_dir[0]) + { + error=my_copy(new_name,org_name, + MYF(MY_WME | MY_HOLD_ORIGINAL_MODES | MY_COPYTIME)); + if (!error) + VOID(my_delete(new_name,MYF(MY_WME))); + } + else + error=my_redel(org_name,new_name,MYF(MY_WME | MY_COPYTIME)); + } + if (! error) + error=save_state(isam_file,mrg,new_length,glob_crc); + } + } + error|=mrg_close(mrg); + if (join_isam_file >= 0) + error|=my_close(join_isam_file,MYF(MY_WME)); + if (error) + { + VOID(fprintf(stderr, "Aborting: %s is not compressed\n", org_name)); + VOID(my_delete(new_name,MYF(MY_WME))); + DBUG_RETURN(-1); + } + if (write_loop || verbose) + { + if (old_length) + VOID(printf("%.4g%% \n", + (((longlong) (old_length - new_length)) * 100.0 / + (longlong) old_length))); + else + puts("Empty file saved in compressed format"); + } + DBUG_RETURN(0); + + err: + free_counts_and_tree_and_queue(huff_trees,trees,huff_counts,fields); + if (new_file >= 0) + VOID(my_close(new_file,MYF(0))); + if (join_isam_file >= 0) + VOID(my_close(join_isam_file,MYF(0))); + mrg_close(mrg); + VOID(fprintf(stderr, "Aborted: %s is not compressed\n", org_name)); + DBUG_RETURN(-1); +} + + /* Init a huff_count-struct for each field and init it */ + +static HUFF_COUNTS *init_huff_count(MARIA_HA *info,my_off_t records) +{ + reg2 uint i; + reg1 HUFF_COUNTS *count; + if ((count = (HUFF_COUNTS*) my_malloc(info->s->base.fields* + sizeof(HUFF_COUNTS), + MYF(MY_ZEROFILL | MY_WME)))) + { + for (i=0 ; i < info->s->base.fields ; i++) + { + enum en_fieldtype type; + count[i].field_length=info->s->rec[i].length; + type= count[i].field_type= (enum en_fieldtype) info->s->rec[i].type; + if (type == FIELD_INTERVALL || + type == FIELD_CONSTANT || + type == FIELD_ZERO) + type = FIELD_NORMAL; + if (count[i].field_length <= 8 && + (type == FIELD_NORMAL || + type == FIELD_SKIP_ZERO)) + count[i].max_zero_fill= count[i].field_length; + /* + For every column initialize a tree, which is used to detect distinct + column values. 'int_tree' works together with 'tree_buff' and + 'tree_pos'. It's keys are implemented by pointers into 'tree_buff'. + This is accomplished by '-1' as the element size. + */ + init_tree(&count[i].int_tree,0,0,-1,(qsort_cmp2) compare_tree,0, NULL, + NULL); + if (records && type != FIELD_BLOB && type != FIELD_VARCHAR) + count[i].tree_pos=count[i].tree_buff = + my_malloc(count[i].field_length > 1 ? tree_buff_length : 2, + MYF(MY_WME)); + } + } + return count; +} + + + /* Free memory used by counts and trees */ + +static void free_counts_and_tree_and_queue(HUFF_TREE *huff_trees, uint trees, + HUFF_COUNTS *huff_counts, + uint fields) +{ + register uint i; + + if (huff_trees) + { + for (i=0 ; i < trees ; i++) + { + if (huff_trees[i].element_buffer) + my_free((gptr) huff_trees[i].element_buffer,MYF(0)); + if (huff_trees[i].code) + my_free((gptr) huff_trees[i].code,MYF(0)); + } + my_free((gptr) huff_trees,MYF(0)); + } + if (huff_counts) + { + for (i=0 ; i < fields ; i++) + { + if (huff_counts[i].tree_buff) + { + my_free((gptr) huff_counts[i].tree_buff,MYF(0)); + delete_tree(&huff_counts[i].int_tree); + } + } + my_free((gptr) huff_counts,MYF(0)); + } + delete_queue(&queue); /* This is safe to free */ + return; +} + + /* Read through old file and gather some statistics */ + +static int get_statistic(PACK_MRG_INFO *mrg,HUFF_COUNTS *huff_counts) +{ + int error; + uint length; + ulong reclength,max_blob_length; + byte *record,*pos,*next_pos,*end_pos,*start_pos; + ha_rows record_count; + my_bool static_row_size; + HUFF_COUNTS *count,*end_count; + TREE_ELEMENT *element; + DBUG_ENTER("get_statistic"); + + reclength=mrg->file[0]->s->base.reclength; + record=(byte*) my_alloca(reclength); + end_count=huff_counts+mrg->file[0]->s->base.fields; + record_count=0; glob_crc=0; + max_blob_length=0; + + /* Check how to calculate checksum */ + static_row_size=1; + for (count=huff_counts ; count < end_count ; count++) + { + if (count->field_type == FIELD_BLOB || + count->field_type == FIELD_VARCHAR) + { + static_row_size=0; + break; + } + } + + mrg_reset(mrg); + while ((error=mrg_rrnd(mrg,record)) != HA_ERR_END_OF_FILE) + { + ulong tot_blob_length=0; + if (! error) + { + /* glob_crc is a checksum over all bytes of all records. */ + if (static_row_size) + glob_crc+=_ma_static_checksum(mrg->file[0],record); + else + glob_crc+=_ma_checksum(mrg->file[0],record); + + /* Count the incidence of values separately for every column. */ + for (pos=record,count=huff_counts ; + count < end_count ; + count++, + pos=next_pos) + { + next_pos=end_pos=(start_pos=pos)+count->field_length; + + /* + Put the whole column value in a tree if there is room for it. + 'int_tree' is used to quickly check for duplicate values. + 'tree_buff' collects as many distinct column values as + possible. If the field length is > 1, it is tree_buff_length, + else 2 bytes. Each value is 'field_length' bytes big. If there + are more distinct column values than fit into the buffer, we + give up with this tree. BLOBs and VARCHARs do not have a + tree_buff as it can only be used with fixed length columns. + For the special case of field length == 1, we handle only the + case that there is only one distinct value in the table(s). + Otherwise, we can have a maximum of 256 distinct values. This + is then handled by the normal Huffman tree build. + + Another limit for collecting distinct column values is the + number of values itself. Since we would need to build a + Huffman tree for the values, we are limited by the 'IS_OFFSET' + constant. This constant expresses a bit which is used to + determine if a tree element holds a final value or an offset + to a child element. Hence, all values and offsets need to be + smaller than 'IS_OFFSET'. A tree element is implemented with + two integer values, one for the left branch and one for the + right branch. For the extreme case that the first element + points to the last element, the number of integers in the tree + must be less or equal to IS_OFFSET. So the number of elements + must be less or equal to IS_OFFSET / 2. + + WARNING: At first, we insert a pointer into the record buffer + as the key for the tree. If we got a new distinct value, which + is really inserted into the tree, instead of being counted + only, we will copy the column value from the record buffer to + 'tree_buff' and adjust the key pointer of the tree accordingly. + */ + if (count->tree_buff) + { + global_count=count; + if (!(element=tree_insert(&count->int_tree,pos, 0, + count->int_tree.custom_arg)) || + (element->count == 1 && + (count->tree_buff + tree_buff_length < + count->tree_pos + count->field_length)) || + (count->int_tree.elements_in_tree > IS_OFFSET / 2) || + (count->field_length == 1 && + count->int_tree.elements_in_tree > 1)) + { + delete_tree(&count->int_tree); + my_free(count->tree_buff,MYF(0)); + count->tree_buff=0; + } + else + { + /* + If tree_insert() succeeds, it either creates a new element + or increments the counter of an existing element. + */ + if (element->count == 1) + { + /* Copy the new column value into 'tree_buff'. */ + memcpy(count->tree_pos,pos,(size_t) count->field_length); + /* Adjust the key pointer in the tree. */ + tree_set_pointer(element,count->tree_pos); + /* Point behind the last column value so far. */ + count->tree_pos+=count->field_length; + } + } + } + + /* Save character counters and space-counts and zero-field-counts */ + if (count->field_type == FIELD_NORMAL || + count->field_type == FIELD_SKIP_ENDSPACE) + { + /* Ignore trailing space. */ + for ( ; end_pos > pos ; end_pos--) + if (end_pos[-1] != ' ') + break; + /* Empty fields are just counted. Go to the next record. */ + if (end_pos == pos) + { + count->empty_fields++; + count->max_zero_fill=0; + continue; + } + /* + Count the total of all trailing spaces and the number of + short trailing spaces. Remember the longest trailing space. + */ + length= (uint) (next_pos-end_pos); + count->tot_end_space+=length; + if (length < 8) + count->end_space[length]++; + if (count->max_end_space < length) + count->max_end_space = length; + } + + if (count->field_type == FIELD_NORMAL || + count->field_type == FIELD_SKIP_PRESPACE) + { + /* Ignore leading space. */ + for (pos=start_pos; pos < end_pos ; pos++) + if (pos[0] != ' ') + break; + /* Empty fields are just counted. Go to the next record. */ + if (end_pos == pos) + { + count->empty_fields++; + count->max_zero_fill=0; + continue; + } + /* + Count the total of all leading spaces and the number of + short leading spaces. Remember the longest leading space. + */ + length= (uint) (pos-start_pos); + count->tot_pre_space+=length; + if (length < 8) + count->pre_space[length]++; + if (count->max_pre_space < length) + count->max_pre_space = length; + } + + /* Calculate pos, end_pos, and max_length for variable length fields. */ + if (count->field_type == FIELD_BLOB) + { + uint field_length=count->field_length -maria_portable_sizeof_char_ptr; + ulong blob_length= _ma_calc_blob_length(field_length, start_pos); + memcpy_fixed((char*) &pos, start_pos+field_length,sizeof(char*)); + end_pos=pos+blob_length; + tot_blob_length+=blob_length; + set_if_bigger(count->max_length,blob_length); + } + else if (count->field_type == FIELD_VARCHAR) + { + uint pack_length= HA_VARCHAR_PACKLENGTH(count->field_length-1); + length= (pack_length == 1 ? (uint) *(uchar*) start_pos : + uint2korr(start_pos)); + pos= start_pos+pack_length; + end_pos= pos+length; + set_if_bigger(count->max_length,length); + } + + /* Evaluate 'max_zero_fill' for short fields. */ + if (count->field_length <= 8 && + (count->field_type == FIELD_NORMAL || + count->field_type == FIELD_SKIP_ZERO)) + { + uint i; + /* Zero fields are just counted. Go to the next record. */ + if (!memcmp((byte*) start_pos,zero_string,count->field_length)) + { + count->zero_fields++; + continue; + } + /* + max_zero_fill starts with field_length. It is decreased every + time a shorter "zero trailer" is found. It is set to zero when + an empty field is found (see above). This suggests that the + variable should be called 'min_zero_fill'. + */ + for (i =0 ; i < count->max_zero_fill && ! end_pos[-1 - (int) i] ; + i++) ; + if (i < count->max_zero_fill) + count->max_zero_fill=i; + } + + /* Ignore zero fields and check fields. */ + if (count->field_type == FIELD_ZERO || + count->field_type == FIELD_CHECK) + continue; + + /* + Count the incidence of every byte value in the + significant field value. + */ + for ( ; pos < end_pos ; pos++) + count->counts[(uchar) *pos]++; + + /* Step to next field. */ + } + + if (tot_blob_length > max_blob_length) + max_blob_length=tot_blob_length; + record_count++; + if (write_loop && record_count % WRITE_COUNT == 0) + { + VOID(printf("%lu\r", (ulong) record_count)); + VOID(fflush(stdout)); + } + } + else if (error != HA_ERR_RECORD_DELETED) + { + VOID(fprintf(stderr, "Got error %d while reading rows", error)); + break; + } + + /* Step to next record. */ + } + if (write_loop) + { + VOID(printf(" \r")); + VOID(fflush(stdout)); + } + + /* + If --debug=d,fakebigcodes is set, fake the counts to get big Huffman + codes. + */ + DBUG_EXECUTE_IF("fakebigcodes", fakebigcodes(huff_counts, end_count);); + + DBUG_PRINT("info", ("Found the following number of incidents " + "of the byte codes:")); + if (verbose >= 2) + VOID(printf("Found the following number of incidents " + "of the byte codes:\n")); + for (count= huff_counts ; count < end_count; count++) + { + uint idx; + my_off_t total_count; + char llbuf[32]; + + DBUG_PRINT("info", ("column: %3u", count - huff_counts + 1)); + if (verbose >= 2) + VOID(printf("column: %3u\n", count - huff_counts + 1)); + if (count->tree_buff) + { + DBUG_PRINT("info", ("number of distinct values: %u", + (count->tree_pos - count->tree_buff) / + count->field_length)); + if (verbose >= 2) + VOID(printf("number of distinct values: %u\n", + (count->tree_pos - count->tree_buff) / + count->field_length)); + } + total_count= 0; + for (idx= 0; idx < 256; idx++) + { + if (count->counts[idx]) + { + total_count+= count->counts[idx]; + DBUG_PRINT("info", ("counts[0x%02x]: %12s", idx, + llstr((longlong) count->counts[idx], llbuf))); + if (verbose >= 2) + VOID(printf("counts[0x%02x]: %12s\n", idx, + llstr((longlong) count->counts[idx], llbuf))); + } + } + DBUG_PRINT("info", ("total: %12s", llstr((longlong) total_count, + llbuf))); + if ((verbose >= 2) && total_count) + { + VOID(printf("total: %12s\n", + llstr((longlong) total_count, llbuf))); + } + } + + mrg->records=record_count; + mrg->max_blob_length=max_blob_length; + my_afree((gptr) record); + DBUG_RETURN(error != HA_ERR_END_OF_FILE); +} + +static int compare_huff_elements(void *not_used __attribute__((unused)), + byte *a, byte *b) +{ + return *((my_off_t*) a) < *((my_off_t*) b) ? -1 : + (*((my_off_t*) a) == *((my_off_t*) b) ? 0 : 1); +} + + /* Check each tree if we should use pre-space-compress, end-space- + compress, empty-field-compress or zero-field-compress */ + +static void check_counts(HUFF_COUNTS *huff_counts, uint trees, + my_off_t records) +{ + uint space_fields,fill_zero_fields,field_count[(int) FIELD_enum_val_count]; + my_off_t old_length,new_length,length; + DBUG_ENTER("check_counts"); + + bzero((gptr) field_count,sizeof(field_count)); + space_fields=fill_zero_fields=0; + + for (; trees-- ; huff_counts++) + { + if (huff_counts->field_type == FIELD_BLOB) + { + huff_counts->length_bits=max_bit(huff_counts->max_length); + goto found_pack; + } + else if (huff_counts->field_type == FIELD_VARCHAR) + { + huff_counts->length_bits=max_bit(huff_counts->max_length); + goto found_pack; + } + else if (huff_counts->field_type == FIELD_CHECK) + { + huff_counts->bytes_packed=0; + huff_counts->counts[0]=0; + goto found_pack; + } + + huff_counts->field_type=FIELD_NORMAL; + huff_counts->pack_type=0; + + /* Check for zero-filled records (in this column), or zero records. */ + if (huff_counts->zero_fields || ! records) + { + my_off_t old_space_count; + /* + If there are only zero filled records (in this column), + or no records at all, we are done. + */ + if (huff_counts->zero_fields == records) + { + huff_counts->field_type= FIELD_ZERO; + huff_counts->bytes_packed=0; + huff_counts->counts[0]=0; + goto found_pack; + } + /* Remeber the number of significant spaces. */ + old_space_count=huff_counts->counts[' ']; + /* Add all leading and trailing spaces. */ + huff_counts->counts[' ']+= (huff_counts->tot_end_space + + huff_counts->tot_pre_space + + huff_counts->empty_fields * + huff_counts->field_length); + /* Check, what the compressed length of this would be. */ + old_length=calc_packed_length(huff_counts,0)+records/8; + /* Get the number of zero bytes. */ + length=huff_counts->zero_fields*huff_counts->field_length; + /* Add it to the counts. */ + huff_counts->counts[0]+=length; + /* Check, what the compressed length of this would be. */ + new_length=calc_packed_length(huff_counts,0); + /* If the compression without the zeroes would be shorter, we are done. */ + if (old_length < new_length && huff_counts->field_length > 1) + { + huff_counts->field_type=FIELD_SKIP_ZERO; + huff_counts->counts[0]-=length; + huff_counts->bytes_packed=old_length- records/8; + goto found_pack; + } + /* Remove the insignificant spaces, but keep the zeroes. */ + huff_counts->counts[' ']=old_space_count; + } + /* Check, what the compressed length of this column would be. */ + huff_counts->bytes_packed=calc_packed_length(huff_counts,0); + + /* + If there are enough empty records (in this column), + treating them specially may pay off. + */ + if (huff_counts->empty_fields) + { + if (huff_counts->field_length > 2 && + huff_counts->empty_fields + (records - huff_counts->empty_fields)* + (1+max_bit(max(huff_counts->max_pre_space, + huff_counts->max_end_space))) < + records * max_bit(huff_counts->field_length)) + { + huff_counts->pack_type |= PACK_TYPE_SPACE_FIELDS; + } + else + { + length=huff_counts->empty_fields*huff_counts->field_length; + if (huff_counts->tot_end_space || ! huff_counts->tot_pre_space) + { + huff_counts->tot_end_space+=length; + huff_counts->max_end_space=huff_counts->field_length; + if (huff_counts->field_length < 8) + huff_counts->end_space[huff_counts->field_length]+= + huff_counts->empty_fields; + } + if (huff_counts->tot_pre_space) + { + huff_counts->tot_pre_space+=length; + huff_counts->max_pre_space=huff_counts->field_length; + if (huff_counts->field_length < 8) + huff_counts->pre_space[huff_counts->field_length]+= + huff_counts->empty_fields; + } + } + } + + /* + If there are enough trailing spaces (in this column), + treating them specially may pay off. + */ + if (huff_counts->tot_end_space) + { + huff_counts->counts[' ']+=huff_counts->tot_pre_space; + if (test_space_compress(huff_counts,records,huff_counts->max_end_space, + huff_counts->end_space, + huff_counts->tot_end_space,FIELD_SKIP_ENDSPACE)) + goto found_pack; + huff_counts->counts[' ']-=huff_counts->tot_pre_space; + } + + /* + If there are enough leading spaces (in this column), + treating them specially may pay off. + */ + if (huff_counts->tot_pre_space) + { + if (test_space_compress(huff_counts,records,huff_counts->max_pre_space, + huff_counts->pre_space, + huff_counts->tot_pre_space,FIELD_SKIP_PRESPACE)) + goto found_pack; + } + + found_pack: /* Found field-packing */ + + /* Test if we can use zero-fill */ + + if (huff_counts->max_zero_fill && + (huff_counts->field_type == FIELD_NORMAL || + huff_counts->field_type == FIELD_SKIP_ZERO)) + { + huff_counts->counts[0]-=huff_counts->max_zero_fill* + (huff_counts->field_type == FIELD_SKIP_ZERO ? + records - huff_counts->zero_fields : records); + huff_counts->pack_type|=PACK_TYPE_ZERO_FILL; + huff_counts->bytes_packed=calc_packed_length(huff_counts,0); + } + + /* Test if intervall-field is better */ + + if (huff_counts->tree_buff) + { + HUFF_TREE tree; + + DBUG_EXECUTE_IF("forceintervall", + huff_counts->bytes_packed= ~ (my_off_t) 0;); + tree.element_buffer=0; + if (!make_huff_tree(&tree,huff_counts) && + tree.bytes_packed+tree.tree_pack_length < huff_counts->bytes_packed) + { + if (tree.elements == 1) + huff_counts->field_type=FIELD_CONSTANT; + else + huff_counts->field_type=FIELD_INTERVALL; + huff_counts->pack_type=0; + } + else + { + my_free((gptr) huff_counts->tree_buff,MYF(0)); + delete_tree(&huff_counts->int_tree); + huff_counts->tree_buff=0; + } + if (tree.element_buffer) + my_free((gptr) tree.element_buffer,MYF(0)); + } + if (huff_counts->pack_type & PACK_TYPE_SPACE_FIELDS) + space_fields++; + if (huff_counts->pack_type & PACK_TYPE_ZERO_FILL) + fill_zero_fields++; + field_count[huff_counts->field_type]++; + } + DBUG_PRINT("info", ("normal: %3d empty-space: %3d " + "empty-zero: %3d empty-fill: %3d", + field_count[FIELD_NORMAL],space_fields, + field_count[FIELD_SKIP_ZERO],fill_zero_fields)); + DBUG_PRINT("info", ("pre-space: %3d end-space: %3d " + "intervall-fields: %3d zero: %3d", + field_count[FIELD_SKIP_PRESPACE], + field_count[FIELD_SKIP_ENDSPACE], + field_count[FIELD_INTERVALL], + field_count[FIELD_ZERO])); + if (verbose) + VOID(printf("\nnormal: %3d empty-space: %3d " + "empty-zero: %3d empty-fill: %3d\n" + "pre-space: %3d end-space: %3d " + "intervall-fields: %3d zero: %3d\n", + field_count[FIELD_NORMAL],space_fields, + field_count[FIELD_SKIP_ZERO],fill_zero_fields, + field_count[FIELD_SKIP_PRESPACE], + field_count[FIELD_SKIP_ENDSPACE], + field_count[FIELD_INTERVALL], + field_count[FIELD_ZERO])); + DBUG_VOID_RETURN; +} + + /* Test if we can use space-compression and empty-field-compression */ + +static int +test_space_compress(HUFF_COUNTS *huff_counts, my_off_t records, + uint max_space_length, my_off_t *space_counts, + my_off_t tot_space_count, enum en_fieldtype field_type) +{ + int min_pos; + uint length_bits,i; + my_off_t space_count,min_space_count,min_pack,new_length,skip; + + length_bits=max_bit(max_space_length); + + /* Default no end_space-packing */ + space_count=huff_counts->counts[(uint) ' ']; + min_space_count= (huff_counts->counts[(uint) ' ']+= tot_space_count); + min_pack=calc_packed_length(huff_counts,0); + min_pos= -2; + huff_counts->counts[(uint) ' ']=space_count; + + /* Test with allways space-count */ + new_length=huff_counts->bytes_packed+length_bits*records/8; + if (new_length+1 < min_pack) + { + min_pos= -1; + min_pack=new_length; + min_space_count=space_count; + } + /* Test with length-flag */ + for (skip=0L, i=0 ; i < 8 ; i++) + { + if (space_counts[i]) + { + if (i) + huff_counts->counts[(uint) ' ']+=space_counts[i]; + skip+=huff_counts->pre_space[i]; + new_length=calc_packed_length(huff_counts,0)+ + (records+(records-skip)*(1+length_bits))/8; + if (new_length < min_pack) + { + min_pos=(int) i; + min_pack=new_length; + min_space_count=huff_counts->counts[(uint) ' ']; + } + } + } + + huff_counts->counts[(uint) ' ']=min_space_count; + huff_counts->bytes_packed=min_pack; + switch (min_pos) { + case -2: + return(0); /* No space-compress */ + case -1: /* Always space-count */ + huff_counts->field_type=field_type; + huff_counts->min_space=0; + huff_counts->length_bits=max_bit(max_space_length); + break; + default: + huff_counts->field_type=field_type; + huff_counts->min_space=(uint) min_pos; + huff_counts->pack_type|=PACK_TYPE_SELECTED; + huff_counts->length_bits=max_bit(max_space_length); + break; + } + return(1); /* Using space-compress */ +} + + + /* Make a huff_tree of each huff_count */ + +static HUFF_TREE* make_huff_trees(HUFF_COUNTS *huff_counts, uint trees) +{ + uint tree; + HUFF_TREE *huff_tree; + DBUG_ENTER("make_huff_trees"); + + if (!(huff_tree=(HUFF_TREE*) my_malloc(trees*sizeof(HUFF_TREE), + MYF(MY_WME | MY_ZEROFILL)))) + DBUG_RETURN(0); + + for (tree=0 ; tree < trees ; tree++) + { + if (make_huff_tree(huff_tree+tree,huff_counts+tree)) + { + while (tree--) + my_free((gptr) huff_tree[tree].element_buffer,MYF(0)); + my_free((gptr) huff_tree,MYF(0)); + DBUG_RETURN(0); + } + } + DBUG_RETURN(huff_tree); +} + +/* + Build a Huffman tree. + + SYNOPSIS + make_huff_tree() + huff_tree The Huffman tree. + huff_counts The counts. + + DESCRIPTION + Build a Huffman tree according to huff_counts->counts or + huff_counts->tree_buff. tree_buff, if non-NULL contains up to + tree_buff_length of distinct column values. In that case, whole + values can be Huffman encoded instead of single bytes. + + RETURN + 0 OK + != 0 Error +*/ + +static int make_huff_tree(HUFF_TREE *huff_tree, HUFF_COUNTS *huff_counts) +{ + uint i,found,bits_packed,first,last; + my_off_t bytes_packed; + HUFF_ELEMENT *a,*b,*new_huff_el; + + first=last=0; + if (huff_counts->tree_buff) + { + /* Calculate the number of distinct values in tree_buff. */ + found= (uint) (huff_counts->tree_pos - huff_counts->tree_buff) / + huff_counts->field_length; + first=0; last=found-1; + } + else + { + /* Count the number of byte codes found in the column. */ + for (i=found=0 ; i < 256 ; i++) + { + if (huff_counts->counts[i]) + { + if (! found++) + first=i; + last=i; + } + } + if (found < 2) + found=2; + } + + /* When using 'tree_buff' we can have more that 256 values. */ + if (queue.max_elements < found) + { + delete_queue(&queue); + if (init_queue(&queue,found,0,0,compare_huff_elements,0)) + return -1; + } + + /* Allocate or reallocate an element buffer for the Huffman tree. */ + if (!huff_tree->element_buffer) + { + if (!(huff_tree->element_buffer= + (HUFF_ELEMENT*) my_malloc(found*2*sizeof(HUFF_ELEMENT),MYF(MY_WME)))) + return 1; + } + else + { + HUFF_ELEMENT *temp; + if (!(temp= + (HUFF_ELEMENT*) my_realloc((gptr) huff_tree->element_buffer, + found*2*sizeof(HUFF_ELEMENT), + MYF(MY_WME)))) + return 1; + huff_tree->element_buffer=temp; + } + + huff_counts->tree=huff_tree; + huff_tree->counts=huff_counts; + huff_tree->min_chr=first; + huff_tree->max_chr=last; + huff_tree->char_bits=max_bit(last-first); + huff_tree->offset_bits=max_bit(found-1)+1; + + if (huff_counts->tree_buff) + { + huff_tree->elements=0; + huff_tree->tree_pack_length=(1+15+16+5+5+ + (huff_tree->char_bits+1)*found+ + (huff_tree->offset_bits+1)* + (found-2)+7)/8 + + (uint) (huff_tree->counts->tree_pos- + huff_tree->counts->tree_buff); + /* + Put a HUFF_ELEMENT into the queue for every distinct column value. + + tree_walk() calls save_counts_in_queue() for every element in + 'int_tree'. This takes elements from the target trees element + buffer and places references to them into the buffer of the + priority queue. We insert in column value order, but the order is + in fact irrelevant here. We will establish the correct order + later. + */ + tree_walk(&huff_counts->int_tree, + (int (*)(void*, element_count,void*)) save_counts_in_queue, + (gptr) huff_tree, left_root_right); + } + else + { + huff_tree->elements=found; + huff_tree->tree_pack_length=(9+9+5+5+ + (huff_tree->char_bits+1)*found+ + (huff_tree->offset_bits+1)* + (found-2)+7)/8; + /* + Put a HUFF_ELEMENT into the queue for every byte code found in the column. + + The elements are taken from the target trees element buffer. + Instead of using queue_insert(), we just place references to the + elements into the buffer of the priority queue. We insert in byte + value order, but the order is in fact irrelevant here. We will + establish the correct order later. + */ + for (i=first, found=0 ; i <= last ; i++) + { + if (huff_counts->counts[i]) + { + new_huff_el=huff_tree->element_buffer+(found++); + new_huff_el->count=huff_counts->counts[i]; + new_huff_el->a.leaf.null=0; + new_huff_el->a.leaf.element_nr=i; + queue.root[found]=(byte*) new_huff_el; + } + } + /* + If there is only a single byte value in this field in all records, + add a second element with zero incidence. This is required to enter + the loop, which builds the Huffman tree. + */ + while (found < 2) + { + new_huff_el=huff_tree->element_buffer+(found++); + new_huff_el->count=0; + new_huff_el->a.leaf.null=0; + if (last) + new_huff_el->a.leaf.element_nr=huff_tree->min_chr=last-1; + else + new_huff_el->a.leaf.element_nr=huff_tree->max_chr=last+1; + queue.root[found]=(byte*) new_huff_el; + } + } + + /* Make a queue from the queue buffer. */ + queue.elements=found; + + /* + Make a priority queue from the queue. Construct its index so that we + have a partially ordered tree. + */ + for (i=found/2 ; i > 0 ; i--) + _downheap(&queue,i); + + /* The Huffman algorithm. */ + bytes_packed=0; bits_packed=0; + for (i=1 ; i < found ; i++) + { + /* + Pop the top element from the queue (the one with the least incidence). + Popping from a priority queue includes a re-ordering of the queue, + to get the next least incidence element to the top. + */ + a=(HUFF_ELEMENT*) queue_remove(&queue,0); + /* + Copy the next least incidence element. The queue implementation + reserves root[0] for temporary purposes. root[1] is the top. + */ + b=(HUFF_ELEMENT*) queue.root[1]; + /* Get a new element from the element buffer. */ + new_huff_el=huff_tree->element_buffer+found+i; + /* The new element gets the sum of the two least incidence elements. */ + new_huff_el->count=a->count+b->count; + /* + The Huffman algorithm assigns another bit to the code for a byte + every time that bytes incidence is combined (directly or indirectly) + to a new element as one of the two least incidence elements. + This means that one more bit per incidence of that byte is required + in the resulting file. So we add the new combined incidence as the + number of bits by which the result grows. + */ + bits_packed+=(uint) (new_huff_el->count & 7); + bytes_packed+=new_huff_el->count/8; + /* The new element points to its children, lesser in left. */ + new_huff_el->a.nod.left=a; + new_huff_el->a.nod.right=b; + /* + Replace the copied top element by the new element and re-order the + queue. + */ + queue.root[1]=(byte*) new_huff_el; + queue_replaced(&queue); + } + huff_tree->root=(HUFF_ELEMENT*) queue.root[1]; + huff_tree->bytes_packed=bytes_packed+(bits_packed+7)/8; + return 0; +} + +static int compare_tree(void* cmp_arg __attribute__((unused)), + register const uchar *s, register const uchar *t) +{ + uint length; + for (length=global_count->field_length; length-- ;) + if (*s++ != *t++) + return (int) s[-1] - (int) t[-1]; + return 0; +} + +/* + Organize distinct column values and their incidences into a priority queue. + + SYNOPSIS + save_counts_in_queue() + key The column value. + count The incidence of this value. + tree The Huffman tree to be built later. + + DESCRIPTION + We use the element buffer of the targeted tree. The distinct column + values are organized in a priority queue first. The Huffman + algorithm will later organize the elements into a Huffman tree. For + the time being, we just place references to the elements into the + queue buffer. The buffer will later be organized into a priority + queue. + + RETURN + 0 + */ + +static int save_counts_in_queue(byte *key, element_count count, + HUFF_TREE *tree) +{ + HUFF_ELEMENT *new_huff_el; + + new_huff_el=tree->element_buffer+(tree->elements++); + new_huff_el->count=count; + new_huff_el->a.leaf.null=0; + new_huff_el->a.leaf.element_nr= (uint) (key- tree->counts->tree_buff) / + tree->counts->field_length; + queue.root[tree->elements]=(byte*) new_huff_el; + return 0; +} + + +/* + Calculate length of file if given counts should be used. + + SYNOPSIS + calc_packed_length() + huff_counts The counts for a column of the table(s). + add_tree_lenght If the decode tree length should be added. + + DESCRIPTION + We need to follow the Huffman algorithm until we know, how many bits + are required for each byte code. But we do not need the resulting + Huffman tree. Hence, we can leave out some steps which are essential + in make_huff_tree(). + + RETURN + Number of bytes required to compress this table column. +*/ + +static my_off_t calc_packed_length(HUFF_COUNTS *huff_counts, + uint add_tree_lenght) +{ + uint i,found,bits_packed,first,last; + my_off_t bytes_packed; + HUFF_ELEMENT element_buffer[256]; + DBUG_ENTER("calc_packed_length"); + + /* + WARNING: We use a small hack for efficiency: Instead of placing + references to HUFF_ELEMENTs into the queue, we just insert + references to the counts of the byte codes which appeared in this + table column. During the Huffman algorithm they are successively + replaced by references to HUFF_ELEMENTs. This works, because + HUFF_ELEMENTs have the incidence count at their beginning. + Regardless, wether the queue array contains references to counts of + type my_off_t or references to HUFF_ELEMENTs which have the count of + type my_off_t at their beginning, it always points to a count of the + same type. + + Instead of using queue_insert(), we just copy the references into + the buffer of the priority queue. We insert in byte value order, but + the order is in fact irrelevant here. We will establish the correct + order later. + */ + first=last=0; + for (i=found=0 ; i < 256 ; i++) + { + if (huff_counts->counts[i]) + { + if (! found++) + first=i; + last=i; + /* We start with root[1], which is the queues top element. */ + queue.root[found]=(byte*) &huff_counts->counts[i]; + } + } + if (!found) + DBUG_RETURN(0); /* Empty tree */ + /* + If there is only a single byte value in this field in all records, + add a second element with zero incidence. This is required to enter + the loop, which follows the Huffman algorithm. + */ + if (found < 2) + queue.root[++found]=(byte*) &huff_counts->counts[last ? 0 : 1]; + + /* Make a queue from the queue buffer. */ + queue.elements=found; + + bytes_packed=0; bits_packed=0; + /* Add the length of the coding table, which would become part of the file. */ + if (add_tree_lenght) + bytes_packed=(8+9+5+5+(max_bit(last-first)+1)*found+ + (max_bit(found-1)+1+1)*(found-2) +7)/8; + + /* + Make a priority queue from the queue. Construct its index so that we + have a partially ordered tree. + */ + for (i=(found+1)/2 ; i > 0 ; i--) + _downheap(&queue,i); + + /* The Huffman algorithm. */ + for (i=0 ; i < found-1 ; i++) + { + my_off_t *a; + my_off_t *b; + HUFF_ELEMENT *new_huff_el; + + /* + Pop the top element from the queue (the one with the least + incidence). Popping from a priority queue includes a re-ordering + of the queue, to get the next least incidence element to the top. + */ + a= (my_off_t*) queue_remove(&queue, 0); + /* + Copy the next least incidence element. The queue implementation + reserves root[0] for temporary purposes. root[1] is the top. + */ + b= (my_off_t*) queue.root[1]; + /* Create a new element in a local (automatic) buffer. */ + new_huff_el= element_buffer + i; + /* The new element gets the sum of the two least incidence elements. */ + new_huff_el->count= *a + *b; + /* + The Huffman algorithm assigns another bit to the code for a byte + every time that bytes incidence is combined (directly or indirectly) + to a new element as one of the two least incidence elements. + This means that one more bit per incidence of that byte is required + in the resulting file. So we add the new combined incidence as the + number of bits by which the result grows. + */ + bits_packed+=(uint) (new_huff_el->count & 7); + bytes_packed+=new_huff_el->count/8; + /* + Replace the copied top element by the new element and re-order the + queue. This successively replaces the references to counts by + references to HUFF_ELEMENTs. + */ + queue.root[1]=(byte*) new_huff_el; + queue_replaced(&queue); + } + DBUG_RETURN(bytes_packed+(bits_packed+7)/8); +} + + + /* Remove trees that don't give any compression */ + +static uint join_same_trees(HUFF_COUNTS *huff_counts, uint trees) +{ + uint k,tree_number; + HUFF_COUNTS count,*i,*j,*last_count; + + last_count=huff_counts+trees; + for (tree_number=0, i=huff_counts ; i < last_count ; i++) + { + if (!i->tree->tree_number) + { + i->tree->tree_number= ++tree_number; + if (i->tree_buff) + continue; /* Don't join intervall */ + for (j=i+1 ; j < last_count ; j++) + { + if (! j->tree->tree_number && ! j->tree_buff) + { + for (k=0 ; k < 256 ; k++) + count.counts[k]=i->counts[k]+j->counts[k]; + if (calc_packed_length(&count,1) <= + i->tree->bytes_packed + j->tree->bytes_packed+ + i->tree->tree_pack_length+j->tree->tree_pack_length+ + ALLOWED_JOIN_DIFF) + { + memcpy_fixed((byte*) i->counts,(byte*) count.counts, + sizeof(count.counts[0])*256); + my_free((gptr) j->tree->element_buffer,MYF(0)); + j->tree->element_buffer=0; + j->tree=i->tree; + bmove((byte*) i->counts,(byte*) count.counts, + sizeof(count.counts[0])*256); + if (make_huff_tree(i->tree,i)) + return (uint) -1; + } + } + } + } + } + DBUG_PRINT("info", ("Original trees: %d After join: %d", + trees, tree_number)); + if (verbose) + VOID(printf("Original trees: %d After join: %d\n", trees, tree_number)); + return tree_number; /* Return trees left */ +} + + +/* + Fill in huff_tree encode tables. + + SYNOPSIS + make_huff_decode_table() + huff_tree An array of HUFF_TREE which are to be encoded. + trees The number of HUFF_TREE in the array. + + RETURN + 0 success + != 0 error +*/ + +static int make_huff_decode_table(HUFF_TREE *huff_tree, uint trees) +{ + uint elements; + for ( ; trees-- ; huff_tree++) + { + if (huff_tree->tree_number > 0) + { + elements=huff_tree->counts->tree_buff ? huff_tree->elements : 256; + if (!(huff_tree->code = + (ulonglong*) my_malloc(elements* + (sizeof(ulonglong) + sizeof(uchar)), + MYF(MY_WME | MY_ZEROFILL)))) + return 1; + huff_tree->code_len=(uchar*) (huff_tree->code+elements); + make_traverse_code_tree(huff_tree, huff_tree->root, + 8 * sizeof(ulonglong), LL(0)); + } + } + return 0; +} + + +static void make_traverse_code_tree(HUFF_TREE *huff_tree, + HUFF_ELEMENT *element, + uint size, ulonglong code) +{ + uint chr; + if (!element->a.leaf.null) + { + chr=element->a.leaf.element_nr; + huff_tree->code_len[chr]= (uchar) (8 * sizeof(ulonglong) - size); + huff_tree->code[chr]= (code >> size); + if (huff_tree->height < 8 * sizeof(ulonglong) - size) + huff_tree->height= 8 * sizeof(ulonglong) - size; + } + else + { + size--; + make_traverse_code_tree(huff_tree,element->a.nod.left,size,code); + make_traverse_code_tree(huff_tree, element->a.nod.right, size, + code + (((ulonglong) 1) << size)); + } + return; +} + + +/* + Convert a value into binary digits. + + SYNOPSIS + bindigits() + value The value. + length The number of low order bits to convert. + + NOTE + The result string is in static storage. It is reused on every call. + So you cannot use it twice in one expression. + + RETURN + A pointer to a static NUL-terminated string. + */ + +static char *bindigits(ulonglong value, uint bits) +{ + static char digits[72]; + char *ptr= digits; + uint idx= bits; + + DBUG_ASSERT(idx < sizeof(digits)); + while (idx) + *(ptr++)= '0' + ((value >> (--idx)) & 1); + *ptr= '\0'; + return digits; +} + + +/* + Convert a value into hexadecimal digits. + + SYNOPSIS + hexdigits() + value The value. + + NOTE + The result string is in static storage. It is reused on every call. + So you cannot use it twice in one expression. + + RETURN + A pointer to a static NUL-terminated string. + */ + +static char *hexdigits(ulonglong value) +{ + static char digits[20]; + char *ptr= digits; + uint idx= 2 * sizeof(value); /* Two hex digits per byte. */ + + DBUG_ASSERT(idx < sizeof(digits)); + while (idx) + { + if ((*(ptr++)= '0' + ((value >> (4 * (--idx))) & 0xf)) > '9') + *(ptr - 1)+= 'a' - '9' - 1; + } + *ptr= '\0'; + return digits; +} + + + /* Write header to new packed data file */ + +static int write_header(PACK_MRG_INFO *mrg,uint head_length,uint trees, + my_off_t tot_elements,my_off_t filelength) +{ + byte *buff= (byte*) file_buffer.pos; + + bzero(buff,HEAD_LENGTH); + memcpy_fixed(buff,maria_pack_file_magic,4); + int4store(buff+4,head_length); + int4store(buff+8, mrg->min_pack_length); + int4store(buff+12,mrg->max_pack_length); + int4store(buff+16,tot_elements); + int4store(buff+20,intervall_length); + int2store(buff+24,trees); + buff[26]=(char) mrg->ref_length; + /* Save record pointer length */ + buff[27]= (uchar) maria_get_pointer_length((ulonglong) filelength,2); + if (test_only) + return 0; + VOID(my_seek(file_buffer.file,0L,MY_SEEK_SET,MYF(0))); + return my_write(file_buffer.file,(const byte *) file_buffer.pos,HEAD_LENGTH, + MYF(MY_WME | MY_NABP | MY_WAIT_IF_FULL)) != 0; +} + + /* Write fieldinfo to new packed file */ + +static void write_field_info(HUFF_COUNTS *counts, uint fields, uint trees) +{ + reg1 uint i; + uint huff_tree_bits; + huff_tree_bits=max_bit(trees ? trees-1 : 0); + + DBUG_PRINT("info", ("")); + DBUG_PRINT("info", ("column types:")); + DBUG_PRINT("info", ("FIELD_NORMAL 0")); + DBUG_PRINT("info", ("FIELD_SKIP_ENDSPACE 1")); + DBUG_PRINT("info", ("FIELD_SKIP_PRESPACE 2")); + DBUG_PRINT("info", ("FIELD_SKIP_ZERO 3")); + DBUG_PRINT("info", ("FIELD_BLOB 4")); + DBUG_PRINT("info", ("FIELD_CONSTANT 5")); + DBUG_PRINT("info", ("FIELD_INTERVALL 6")); + DBUG_PRINT("info", ("FIELD_ZERO 7")); + DBUG_PRINT("info", ("FIELD_VARCHAR 8")); + DBUG_PRINT("info", ("FIELD_CHECK 9")); + DBUG_PRINT("info", ("")); + DBUG_PRINT("info", ("pack type as a set of flags:")); + DBUG_PRINT("info", ("PACK_TYPE_SELECTED 1")); + DBUG_PRINT("info", ("PACK_TYPE_SPACE_FIELDS 2")); + DBUG_PRINT("info", ("PACK_TYPE_ZERO_FILL 4")); + DBUG_PRINT("info", ("")); + if (verbose >= 2) + { + VOID(printf("\n")); + VOID(printf("column types:\n")); + VOID(printf("FIELD_NORMAL 0\n")); + VOID(printf("FIELD_SKIP_ENDSPACE 1\n")); + VOID(printf("FIELD_SKIP_PRESPACE 2\n")); + VOID(printf("FIELD_SKIP_ZERO 3\n")); + VOID(printf("FIELD_BLOB 4\n")); + VOID(printf("FIELD_CONSTANT 5\n")); + VOID(printf("FIELD_INTERVALL 6\n")); + VOID(printf("FIELD_ZERO 7\n")); + VOID(printf("FIELD_VARCHAR 8\n")); + VOID(printf("FIELD_CHECK 9\n")); + VOID(printf("\n")); + VOID(printf("pack type as a set of flags:\n")); + VOID(printf("PACK_TYPE_SELECTED 1\n")); + VOID(printf("PACK_TYPE_SPACE_FIELDS 2\n")); + VOID(printf("PACK_TYPE_ZERO_FILL 4\n")); + VOID(printf("\n")); + } + for (i=0 ; i++ < fields ; counts++) + { + write_bits((ulonglong) (int) counts->field_type, 5); + write_bits(counts->pack_type,6); + if (counts->pack_type & PACK_TYPE_ZERO_FILL) + write_bits(counts->max_zero_fill,5); + else + write_bits(counts->length_bits,5); + write_bits((ulonglong) counts->tree->tree_number - 1, huff_tree_bits); + DBUG_PRINT("info", ("column: %3u type: %2u pack: %2u zero: %4u " + "lbits: %2u tree: %2u length: %4u", + i , counts->field_type, counts->pack_type, + counts->max_zero_fill, counts->length_bits, + counts->tree->tree_number, counts->field_length)); + if (verbose >= 2) + VOID(printf("column: %3u type: %2u pack: %2u zero: %4u lbits: %2u " + "tree: %2u length: %4u\n", i , counts->field_type, + counts->pack_type, counts->max_zero_fill, counts->length_bits, + counts->tree->tree_number, counts->field_length)); + } + flush_bits(); + return; +} + + /* Write all huff_trees to new datafile. Return tot count of + elements in all trees + Returns 0 on error */ + +static my_off_t write_huff_tree(HUFF_TREE *huff_tree, uint trees) +{ + uint i,int_length; + uint tree_no; + uint codes; + uint errors= 0; + uint *packed_tree,*offset,length; + my_off_t elements; + + /* Find the highest number of elements in the trees. */ + for (i=length=0 ; i < trees ; i++) + if (huff_tree[i].tree_number > 0 && huff_tree[i].elements > length) + length=huff_tree[i].elements; + /* + Allocate a buffer for packing a decode tree. Two numbers per element + (left child and right child). + */ + if (!(packed_tree=(uint*) my_alloca(sizeof(uint)*length*2))) + { + my_error(EE_OUTOFMEMORY,MYF(ME_BELL),sizeof(uint)*length*2); + return 0; + } + + DBUG_PRINT("info", ("")); + if (verbose >= 2) + VOID(printf("\n")); + tree_no= 0; + intervall_length=0; + for (elements=0; trees-- ; huff_tree++) + { + /* Skip columns that have been joined with other columns. */ + if (huff_tree->tree_number == 0) + continue; /* Deleted tree */ + tree_no++; + DBUG_PRINT("info", ("")); + if (verbose >= 3) + VOID(printf("\n")); + /* Count the total number of elements (byte codes or column values). */ + elements+=huff_tree->elements; + huff_tree->max_offset=2; + /* Build a tree of offsets and codes for decoding in 'packed_tree'. */ + if (huff_tree->elements <= 1) + offset=packed_tree; + else + offset=make_offset_code_tree(huff_tree,huff_tree->root,packed_tree); + + /* This should be the same as 'length' above. */ + huff_tree->offset_bits=max_bit(huff_tree->max_offset); + + /* + Since we check this during collecting the distinct column values, + this should never happen. + */ + if (huff_tree->max_offset >= IS_OFFSET) + { /* This should be impossible */ + VOID(fprintf(stderr, "Tree offset got too big: %d, aborted\n", + huff_tree->max_offset)); + my_afree((gptr) packed_tree); + return 0; + } + + DBUG_PRINT("info", ("pos: %lu elements: %u tree-elements: %lu " + "char_bits: %u\n", + (ulong) (file_buffer.pos - file_buffer.buffer), + huff_tree->elements, (ulong) (offset - packed_tree), + huff_tree->char_bits)); + if (!huff_tree->counts->tree_buff) + { + /* We do a byte compression on this column. Mark with bit 0. */ + write_bits(0,1); + write_bits(huff_tree->min_chr,8); + write_bits(huff_tree->elements,9); + write_bits(huff_tree->char_bits,5); + write_bits(huff_tree->offset_bits,5); + int_length=0; + } + else + { + int_length=(uint) (huff_tree->counts->tree_pos - + huff_tree->counts->tree_buff); + /* We have distinct column values for this column. Mark with bit 1. */ + write_bits(1,1); + write_bits(huff_tree->elements,15); + write_bits(int_length,16); + write_bits(huff_tree->char_bits,5); + write_bits(huff_tree->offset_bits,5); + intervall_length+=int_length; + } + DBUG_PRINT("info", ("tree: %2u elements: %4u char_bits: %2u " + "offset_bits: %2u %s: %5u codelen: %2u", + tree_no, huff_tree->elements, huff_tree->char_bits, + huff_tree->offset_bits, huff_tree->counts->tree_buff ? + "bufflen" : "min_chr", huff_tree->counts->tree_buff ? + int_length : huff_tree->min_chr, huff_tree->height)); + if (verbose >= 2) + VOID(printf("tree: %2u elements: %4u char_bits: %2u offset_bits: %2u " + "%s: %5u codelen: %2u\n", tree_no, huff_tree->elements, + huff_tree->char_bits, huff_tree->offset_bits, + huff_tree->counts->tree_buff ? "bufflen" : "min_chr", + huff_tree->counts->tree_buff ? int_length : + huff_tree->min_chr, huff_tree->height)); + + /* Check that the code tree length matches the element count. */ + length=(uint) (offset-packed_tree); + if (length != huff_tree->elements*2-2) + { + VOID(fprintf(stderr, "error: Huff-tree-length: %d != calc_length: %d\n", + length, huff_tree->elements * 2 - 2)); + errors++; + break; + } + + for (i=0 ; i < length ; i++) + { + if (packed_tree[i] & IS_OFFSET) + write_bits(packed_tree[i] - IS_OFFSET+ (1 << huff_tree->offset_bits), + huff_tree->offset_bits+1); + else + write_bits(packed_tree[i]-huff_tree->min_chr,huff_tree->char_bits+1); + DBUG_PRINT("info", ("tree[0x%04x]: %s0x%04x", + i, (packed_tree[i] & IS_OFFSET) ? + " -> " : "", (packed_tree[i] & IS_OFFSET) ? + packed_tree[i] - IS_OFFSET + i : packed_tree[i])); + if (verbose >= 3) + VOID(printf("tree[0x%04x]: %s0x%04x\n", + i, (packed_tree[i] & IS_OFFSET) ? " -> " : "", + (packed_tree[i] & IS_OFFSET) ? + packed_tree[i] - IS_OFFSET + i : packed_tree[i])); + } + flush_bits(); + + /* + Display coding tables and check their correctness. + */ + codes= huff_tree->counts->tree_buff ? huff_tree->elements : 256; + for (i= 0; i < codes; i++) + { + ulonglong code; + uint bits; + uint len; + uint idx; + + if (! (len= huff_tree->code_len[i])) + continue; + DBUG_PRINT("info", ("code[0x%04x]: 0x%s bits: %2u bin: %s", i, + hexdigits(huff_tree->code[i]), huff_tree->code_len[i], + bindigits(huff_tree->code[i], + huff_tree->code_len[i]))); + if (verbose >= 3) + VOID(printf("code[0x%04x]: 0x%s bits: %2u bin: %s\n", i, + hexdigits(huff_tree->code[i]), huff_tree->code_len[i], + bindigits(huff_tree->code[i], huff_tree->code_len[i]))); + + /* Check that the encode table decodes correctly. */ + code= 0; + bits= 0; + idx= 0; + DBUG_EXECUTE_IF("forcechkerr1", len--;); + DBUG_EXECUTE_IF("forcechkerr2", bits= 8 * sizeof(code);); + DBUG_EXECUTE_IF("forcechkerr3", idx= length;); + for (;;) + { + if (! len) + { + VOID(fflush(stdout)); + VOID(fprintf(stderr, "error: code 0x%s with %u bits not found\n", + hexdigits(huff_tree->code[i]), huff_tree->code_len[i])); + errors++; + break; + } + code<<= 1; + code|= (huff_tree->code[i] >> (--len)) & 1; + bits++; + if (bits > 8 * sizeof(code)) + { + VOID(fflush(stdout)); + VOID(fprintf(stderr, "error: Huffman code too long: %u/%u\n", + bits, 8 * sizeof(code))); + errors++; + break; + } + idx+= code & 1; + if (idx >= length) + { + VOID(fflush(stdout)); + VOID(fprintf(stderr, "error: illegal tree offset: %u/%u\n", + idx, length)); + errors++; + break; + } + if (packed_tree[idx] & IS_OFFSET) + idx+= packed_tree[idx] & ~IS_OFFSET; + else + break; /* Hit a leaf. This contains the result value. */ + } + if (errors) + break; + + DBUG_EXECUTE_IF("forcechkerr4", packed_tree[idx]++;); + if (packed_tree[idx] != i) + { + VOID(fflush(stdout)); + VOID(fprintf(stderr, "error: decoded value 0x%04x should be: 0x%04x\n", + packed_tree[idx], i)); + errors++; + break; + } + } /*end for (codes)*/ + if (errors) + break; + + /* Write column values in case of distinct column value compression. */ + if (huff_tree->counts->tree_buff) + { + for (i=0 ; i < int_length ; i++) + { + write_bits((ulonglong) (uchar) huff_tree->counts->tree_buff[i], 8); + DBUG_PRINT("info", ("column_values[0x%04x]: 0x%02x", + i, (uchar) huff_tree->counts->tree_buff[i])); + if (verbose >= 3) + VOID(printf("column_values[0x%04x]: 0x%02x\n", + i, (uchar) huff_tree->counts->tree_buff[i])); + } + } + flush_bits(); + } + DBUG_PRINT("info", ("")); + if (verbose >= 2) + VOID(printf("\n")); + my_afree((gptr) packed_tree); + if (errors) + { + VOID(fprintf(stderr, "Error: Generated decode trees are corrupt. Stop.\n")); + return 0; + } + return elements; +} + + +static uint *make_offset_code_tree(HUFF_TREE *huff_tree, HUFF_ELEMENT *element, + uint *offset) +{ + uint *prev_offset; + + prev_offset= offset; + /* + 'a.leaf.null' takes the same place as 'a.nod.left'. If this is null, + then there is no left child and, hence no right child either. This + is a property of a binary tree. An element is either a node with two + childs, or a leaf without childs. + + The current element is always a node with two childs. Go left first. + */ + if (!element->a.nod.left->a.leaf.null) + { + /* Store the byte code or the index of the column value. */ + prev_offset[0] =(uint) element->a.nod.left->a.leaf.element_nr; + offset+=2; + } + else + { + /* + Recursively traverse the tree to the left. Mark it as an offset to + another tree node (in contrast to a byte code or column value index). + */ + prev_offset[0]= IS_OFFSET+2; + offset=make_offset_code_tree(huff_tree,element->a.nod.left,offset+2); + } + + /* Now, check the right child. */ + if (!element->a.nod.right->a.leaf.null) + { + /* Store the byte code or the index of the column value. */ + prev_offset[1]=element->a.nod.right->a.leaf.element_nr; + return offset; + } + else + { + /* + Recursively traverse the tree to the right. Mark it as an offset to + another tree node (in contrast to a byte code or column value index). + */ + uint temp=(uint) (offset-prev_offset-1); + prev_offset[1]= IS_OFFSET+ temp; + if (huff_tree->max_offset < temp) + huff_tree->max_offset = temp; + return make_offset_code_tree(huff_tree,element->a.nod.right,offset); + } +} + + /* Get number of bits neaded to represent value */ + +static uint max_bit(register uint value) +{ + reg2 uint power=1; + + while ((value>>=1)) + power++; + return (power); +} + + +static int compress_isam_file(PACK_MRG_INFO *mrg, HUFF_COUNTS *huff_counts) +{ + int error; + uint i,max_calc_length,pack_ref_length,min_record_length,max_record_length, + intervall,field_length,max_pack_length,pack_blob_length; + my_off_t record_count; + char llbuf[32]; + ulong length,pack_length; + byte *record,*pos,*end_pos,*record_pos,*start_pos; + HUFF_COUNTS *count,*end_count; + HUFF_TREE *tree; + MARIA_HA *isam_file=mrg->file[0]; + uint pack_version= (uint) isam_file->s->pack.version; + DBUG_ENTER("compress_isam_file"); + + /* Allocate a buffer for the records (excluding blobs). */ + if (!(record=(byte*) my_alloca(isam_file->s->base.reclength))) + return -1; + + end_count=huff_counts+isam_file->s->base.fields; + min_record_length= (uint) ~0; + max_record_length=0; + + /* + Calculate the maximum number of bits required to pack the records. + Remember to understand 'max_zero_fill' as 'min_zero_fill'. + The tree height determines the maximum number of bits per value. + Some fields skip leading or trailing spaces or zeroes. The skipped + number of bytes is encoded by 'length_bits' bits. + Empty blobs and varchar are encoded with a single 1 bit. Other blobs + and varchar get a leading 0 bit. + */ + for (i=max_calc_length=0 ; i < isam_file->s->base.fields ; i++) + { + if (!(huff_counts[i].pack_type & PACK_TYPE_ZERO_FILL)) + huff_counts[i].max_zero_fill=0; + if (huff_counts[i].field_type == FIELD_CONSTANT || + huff_counts[i].field_type == FIELD_ZERO || + huff_counts[i].field_type == FIELD_CHECK) + continue; + if (huff_counts[i].field_type == FIELD_INTERVALL) + max_calc_length+=huff_counts[i].tree->height; + else if (huff_counts[i].field_type == FIELD_BLOB || + huff_counts[i].field_type == FIELD_VARCHAR) + max_calc_length+=huff_counts[i].tree->height*huff_counts[i].max_length + huff_counts[i].length_bits +1; + else + max_calc_length+= + (huff_counts[i].field_length - huff_counts[i].max_zero_fill)* + huff_counts[i].tree->height+huff_counts[i].length_bits; + } + max_calc_length= (max_calc_length + 7) / 8; + pack_ref_length= _ma_calc_pack_length(pack_version, max_calc_length); + record_count=0; + /* 'max_blob_length' is the max length of all blobs of a record. */ + pack_blob_length= isam_file->s->base.blobs ? + _ma_calc_pack_length(pack_version, mrg->max_blob_length) : 0; + max_pack_length=pack_ref_length+pack_blob_length; + + DBUG_PRINT("fields", ("===")); + mrg_reset(mrg); + while ((error=mrg_rrnd(mrg,record)) != HA_ERR_END_OF_FILE) + { + ulong tot_blob_length=0; + if (! error) + { + if (flush_buffer((ulong) max_calc_length + (ulong) max_pack_length)) + break; + record_pos= (byte*) file_buffer.pos; + file_buffer.pos+=max_pack_length; + for (start_pos=record, count= huff_counts; count < end_count ; count++) + { + end_pos=start_pos+(field_length=count->field_length); + tree=count->tree; + + DBUG_PRINT("fields", ("column: %3lu type: %2u pack: %2u zero: %4u " + "lbits: %2u tree: %2u length: %4u", + (ulong) (count - huff_counts + 1), + count->field_type, + count->pack_type, count->max_zero_fill, + count->length_bits, count->tree->tree_number, + count->field_length)); + + /* Check if the column contains spaces only. */ + if (count->pack_type & PACK_TYPE_SPACE_FIELDS) + { + for (pos=start_pos ; *pos == ' ' && pos < end_pos; pos++) ; + if (pos == end_pos) + { + DBUG_PRINT("fields", + ("PACK_TYPE_SPACE_FIELDS spaces only, bits: 1")); + DBUG_PRINT("fields", ("---")); + write_bits(1,1); + start_pos=end_pos; + continue; + } + DBUG_PRINT("fields", + ("PACK_TYPE_SPACE_FIELDS not only spaces, bits: 1")); + write_bits(0,1); + } + end_pos-=count->max_zero_fill; + field_length-=count->max_zero_fill; + + switch(count->field_type) { + case FIELD_SKIP_ZERO: + if (!memcmp((byte*) start_pos,zero_string,field_length)) + { + DBUG_PRINT("fields", ("FIELD_SKIP_ZERO zeroes only, bits: 1")); + write_bits(1,1); + start_pos=end_pos; + break; + } + DBUG_PRINT("fields", ("FIELD_SKIP_ZERO not only zeroes, bits: 1")); + write_bits(0,1); + /* Fall through */ + case FIELD_NORMAL: + DBUG_PRINT("fields", ("FIELD_NORMAL %lu bytes", + (ulong) (end_pos - start_pos))); + for ( ; start_pos < end_pos ; start_pos++) + { + DBUG_PRINT("fields", + ("value: 0x%02x code: 0x%s bits: %2u bin: %s", + (uchar) *start_pos, + hexdigits(tree->code[(uchar) *start_pos]), + (uint) tree->code_len[(uchar) *start_pos], + bindigits(tree->code[(uchar) *start_pos], + (uint) tree->code_len[(uchar) *start_pos]))); + write_bits(tree->code[(uchar) *start_pos], + (uint) tree->code_len[(uchar) *start_pos]); + } + break; + case FIELD_SKIP_ENDSPACE: + for (pos=end_pos ; pos > start_pos && pos[-1] == ' ' ; pos--) ; + length= (ulong) (end_pos - pos); + if (count->pack_type & PACK_TYPE_SELECTED) + { + if (length > count->min_space) + { + DBUG_PRINT("fields", + ("FIELD_SKIP_ENDSPACE more than min_space, bits: 1")); + DBUG_PRINT("fields", + ("FIELD_SKIP_ENDSPACE skip %lu/%u bytes, bits: %2u", + length, field_length, count->length_bits)); + write_bits(1,1); + write_bits(length,count->length_bits); + } + else + { + DBUG_PRINT("fields", + ("FIELD_SKIP_ENDSPACE not more than min_space, " + "bits: 1")); + write_bits(0,1); + pos=end_pos; + } + } + else + { + DBUG_PRINT("fields", + ("FIELD_SKIP_ENDSPACE skip %lu/%u bytes, bits: %2u", + length, field_length, count->length_bits)); + write_bits(length,count->length_bits); + } + /* Encode all significant bytes. */ + DBUG_PRINT("fields", ("FIELD_SKIP_ENDSPACE %lu bytes", + (ulong) (pos - start_pos))); + for ( ; start_pos < pos ; start_pos++) + { + DBUG_PRINT("fields", + ("value: 0x%02x code: 0x%s bits: %2u bin: %s", + (uchar) *start_pos, + hexdigits(tree->code[(uchar) *start_pos]), + (uint) tree->code_len[(uchar) *start_pos], + bindigits(tree->code[(uchar) *start_pos], + (uint) tree->code_len[(uchar) *start_pos]))); + write_bits(tree->code[(uchar) *start_pos], + (uint) tree->code_len[(uchar) *start_pos]); + } + start_pos=end_pos; + break; + case FIELD_SKIP_PRESPACE: + for (pos=start_pos ; pos < end_pos && pos[0] == ' ' ; pos++) ; + length= (ulong) (pos - start_pos); + if (count->pack_type & PACK_TYPE_SELECTED) + { + if (length > count->min_space) + { + DBUG_PRINT("fields", + ("FIELD_SKIP_PRESPACE more than min_space, bits: 1")); + DBUG_PRINT("fields", + ("FIELD_SKIP_PRESPACE skip %lu/%u bytes, bits: %2u", + length, field_length, count->length_bits)); + write_bits(1,1); + write_bits(length,count->length_bits); + } + else + { + DBUG_PRINT("fields", + ("FIELD_SKIP_PRESPACE not more than min_space, " + "bits: 1")); + pos=start_pos; + write_bits(0,1); + } + } + else + { + DBUG_PRINT("fields", + ("FIELD_SKIP_PRESPACE skip %lu/%u bytes, bits: %2u", + length, field_length, count->length_bits)); + write_bits(length,count->length_bits); + } + /* Encode all significant bytes. */ + DBUG_PRINT("fields", ("FIELD_SKIP_PRESPACE %lu bytes", + (ulong) (end_pos - start_pos))); + for (start_pos=pos ; start_pos < end_pos ; start_pos++) + { + DBUG_PRINT("fields", + ("value: 0x%02x code: 0x%s bits: %2u bin: %s", + (uchar) *start_pos, + hexdigits(tree->code[(uchar) *start_pos]), + (uint) tree->code_len[(uchar) *start_pos], + bindigits(tree->code[(uchar) *start_pos], + (uint) tree->code_len[(uchar) *start_pos]))); + write_bits(tree->code[(uchar) *start_pos], + (uint) tree->code_len[(uchar) *start_pos]); + } + break; + case FIELD_CONSTANT: + case FIELD_ZERO: + case FIELD_CHECK: + DBUG_PRINT("fields", ("FIELD_CONSTANT/ZERO/CHECK")); + start_pos=end_pos; + break; + case FIELD_INTERVALL: + global_count=count; + pos=(byte*) tree_search(&count->int_tree, start_pos, + count->int_tree.custom_arg); + intervall=(uint) (pos - count->tree_buff)/field_length; + DBUG_PRINT("fields", ("FIELD_INTERVALL")); + DBUG_PRINT("fields", ("index: %4u code: 0x%s bits: %2u", + intervall, hexdigits(tree->code[intervall]), + (uint) tree->code_len[intervall])); + write_bits(tree->code[intervall],(uint) tree->code_len[intervall]); + start_pos=end_pos; + break; + case FIELD_BLOB: + { + ulong blob_length= _ma_calc_blob_length(field_length- + maria_portable_sizeof_char_ptr, + start_pos); + /* Empty blobs are encoded with a single 1 bit. */ + if (!blob_length) + { + DBUG_PRINT("fields", ("FIELD_BLOB empty, bits: 1")); + write_bits(1,1); + } + else + { + byte *blob,*blob_end; + DBUG_PRINT("fields", ("FIELD_BLOB not empty, bits: 1")); + write_bits(0,1); + /* Write the blob length. */ + DBUG_PRINT("fields", ("FIELD_BLOB %lu bytes, bits: %2u", + blob_length, count->length_bits)); + write_bits(blob_length,count->length_bits); + memcpy_fixed(&blob,end_pos-maria_portable_sizeof_char_ptr, + sizeof(char*)); + blob_end=blob+blob_length; + /* Encode the blob bytes. */ + for ( ; blob < blob_end ; blob++) + { + DBUG_PRINT("fields", + ("value: 0x%02x code: 0x%s bits: %2u bin: %s", + (uchar) *blob, hexdigits(tree->code[(uchar) *blob]), + (uint) tree->code_len[(uchar) *blob], + bindigits(tree->code[(uchar) *start_pos], + (uint)tree->code_len[(uchar) *start_pos]))); + write_bits(tree->code[(uchar) *blob], + (uint) tree->code_len[(uchar) *blob]); + } + tot_blob_length+=blob_length; + } + start_pos= end_pos; + break; + } + case FIELD_VARCHAR: + { + uint pack_length= HA_VARCHAR_PACKLENGTH(count->field_length-1); + ulong col_length= (pack_length == 1 ? (uint) *(uchar*) start_pos : + uint2korr(start_pos)); + /* Empty varchar are encoded with a single 1 bit. */ + if (!col_length) + { + DBUG_PRINT("fields", ("FIELD_VARCHAR empty, bits: 1")); + write_bits(1,1); /* Empty varchar */ + } + else + { + byte *end=start_pos+pack_length+col_length; + DBUG_PRINT("fields", ("FIELD_VARCHAR not empty, bits: 1")); + write_bits(0,1); + /* Write the varchar length. */ + DBUG_PRINT("fields", ("FIELD_VARCHAR %lu bytes, bits: %2u", + col_length, count->length_bits)); + write_bits(col_length,count->length_bits); + /* Encode the varchar bytes. */ + for (start_pos+=pack_length ; start_pos < end ; start_pos++) + { + DBUG_PRINT("fields", + ("value: 0x%02x code: 0x%s bits: %2u bin: %s", + (uchar) *start_pos, + hexdigits(tree->code[(uchar) *start_pos]), + (uint) tree->code_len[(uchar) *start_pos], + bindigits(tree->code[(uchar) *start_pos], + (uint)tree->code_len[(uchar) *start_pos]))); + write_bits(tree->code[(uchar) *start_pos], + (uint) tree->code_len[(uchar) *start_pos]); + } + } + start_pos= end_pos; + break; + } + case FIELD_LAST: + case FIELD_enum_val_count: + abort(); /* Impossible */ + } + start_pos+=count->max_zero_fill; + DBUG_PRINT("fields", ("---")); + } + flush_bits(); + length=(ulong) ((byte*) file_buffer.pos - record_pos) - max_pack_length; + pack_length= _ma_save_pack_length(pack_version, record_pos, length); + if (pack_blob_length) + pack_length+= _ma_save_pack_length(pack_version, record_pos + pack_length, + tot_blob_length); + DBUG_PRINT("fields", ("record: %lu length: %lu blob-length: %lu " + "length-bytes: %lu", (ulong) record_count, length, + tot_blob_length, pack_length)); + DBUG_PRINT("fields", ("===")); + + /* Correct file buffer if the header was smaller */ + if (pack_length != max_pack_length) + { + bmove(record_pos+pack_length,record_pos+max_pack_length,length); + file_buffer.pos-= (max_pack_length-pack_length); + } + if (length < (ulong) min_record_length) + min_record_length=(uint) length; + if (length > (ulong) max_record_length) + max_record_length=(uint) length; + record_count++; + if (write_loop && record_count % WRITE_COUNT == 0) + { + VOID(printf("%lu\r", (ulong) record_count)); + VOID(fflush(stdout)); + } + } + else if (error != HA_ERR_RECORD_DELETED) + break; + } + if (error == HA_ERR_END_OF_FILE) + error=0; + else + { + VOID(fprintf(stderr, "%s: Got error %d reading records\n", + my_progname, error)); + } + if (verbose >= 2) + VOID(printf("wrote %s records.\n", llstr((longlong) record_count, llbuf))); + + my_afree((gptr) record); + mrg->ref_length=max_pack_length; + mrg->min_pack_length=max_record_length ? min_record_length : 0; + mrg->max_pack_length=max_record_length; + DBUG_RETURN(error || error_on_write || flush_buffer(~(ulong) 0)); +} + + +static char *make_new_name(char *new_name, char *old_name) +{ + return fn_format(new_name,old_name,"",DATA_TMP_EXT,2+4); +} + +static char *make_old_name(char *new_name, char *old_name) +{ + return fn_format(new_name,old_name,"",OLD_EXT,2+4); +} + + /* rutines for bit writing buffer */ + +static void init_file_buffer(File file, pbool read_buffer) +{ + file_buffer.file=file; + file_buffer.buffer= (uchar*) my_malloc(ALIGN_SIZE(RECORD_CACHE_SIZE), + MYF(MY_WME)); + file_buffer.end=file_buffer.buffer+ALIGN_SIZE(RECORD_CACHE_SIZE)-8; + file_buffer.pos_in_file=0; + error_on_write=0; + if (read_buffer) + { + + file_buffer.pos=file_buffer.end; + file_buffer.bits=0; + } + else + { + file_buffer.pos=file_buffer.buffer; + file_buffer.bits=BITS_SAVED; + } + file_buffer.bitbucket= 0; +} + + +static int flush_buffer(ulong neaded_length) +{ + ulong length; + + /* + file_buffer.end is 8 bytes lower than the real end of the buffer. + This is done so that the end-of-buffer condition does not need to be + checked for every byte (see write_bits()). Consequently, + file_buffer.pos can become greater than file_buffer.end. The + algorithms in the other functions ensure that there will never be + more than 8 bytes written to the buffer without an end-of-buffer + check. So the buffer cannot be overrun. But we need to check for the + near-to-buffer-end condition to avoid a negative result, which is + casted to unsigned and thus becomes giant. + */ + if ((file_buffer.pos < file_buffer.end) && + ((ulong) (file_buffer.end - file_buffer.pos) > neaded_length)) + return 0; + length=(ulong) (file_buffer.pos-file_buffer.buffer); + file_buffer.pos=file_buffer.buffer; + file_buffer.pos_in_file+=length; + if (test_only) + return 0; + if (error_on_write|| my_write(file_buffer.file, + (const byte*) file_buffer.buffer, + length, + MYF(MY_WME | MY_NABP | MY_WAIT_IF_FULL))) + { + error_on_write=1; + return 1; + } + + if (neaded_length != ~(ulong) 0 && + (ulong) (file_buffer.end-file_buffer.buffer) < neaded_length) + { + char *tmp; + neaded_length+=256; /* some margin */ + tmp= my_realloc((char*) file_buffer.buffer, neaded_length,MYF(MY_WME)); + if (!tmp) + return 1; + file_buffer.pos= ((uchar*) tmp + + (ulong) (file_buffer.pos - file_buffer.buffer)); + file_buffer.buffer= (uchar*) tmp; + file_buffer.end= (uchar*) (tmp+neaded_length-8); + } + return 0; +} + + +static void end_file_buffer(void) +{ + my_free((gptr) file_buffer.buffer,MYF(0)); +} + + /* output `bits` low bits of `value' */ + +static void write_bits(register ulonglong value, register uint bits) +{ + DBUG_ASSERT(((bits < 8 * sizeof(value)) && ! (value >> bits)) || + (bits == 8 * sizeof(value))); + + if ((file_buffer.bits-= (int) bits) >= 0) + { + file_buffer.bitbucket|= value << file_buffer.bits; + } + else + { + reg3 ulonglong bit_buffer; + bits= (uint) -file_buffer.bits; + bit_buffer= (file_buffer.bitbucket | + ((bits != 8 * sizeof(value)) ? (value >> bits) : 0)); +#if BITS_SAVED == 64 + *file_buffer.pos++= (uchar) (bit_buffer >> 56); + *file_buffer.pos++= (uchar) (bit_buffer >> 48); + *file_buffer.pos++= (uchar) (bit_buffer >> 40); + *file_buffer.pos++= (uchar) (bit_buffer >> 32); +#endif + *file_buffer.pos++= (uchar) (bit_buffer >> 24); + *file_buffer.pos++= (uchar) (bit_buffer >> 16); + *file_buffer.pos++= (uchar) (bit_buffer >> 8); + *file_buffer.pos++= (uchar) (bit_buffer); + + if (bits != 8 * sizeof(value)) + value&= (((ulonglong) 1) << bits) - 1; + if (file_buffer.pos >= file_buffer.end) + VOID(flush_buffer(~ (ulong) 0)); + file_buffer.bits=(int) (BITS_SAVED - bits); + file_buffer.bitbucket= value << (BITS_SAVED - bits); + } + return; +} + + /* Flush bits in bit_buffer to buffer */ + +static void flush_bits(void) +{ + int bits; + ulonglong bit_buffer; + + bits= file_buffer.bits & ~7; + bit_buffer= file_buffer.bitbucket >> bits; + bits= BITS_SAVED - bits; + while (bits > 0) + { + bits-= 8; + *file_buffer.pos++= (uchar) (bit_buffer >> bits); + } + file_buffer.bits= BITS_SAVED; + file_buffer.bitbucket= 0; +} + + +/**************************************************************************** +** functions to handle the joined files +****************************************************************************/ + +static int save_state(MARIA_HA *isam_file,PACK_MRG_INFO *mrg,my_off_t new_length, + ha_checksum crc) +{ + MARIA_SHARE *share=isam_file->s; + uint options=mi_uint2korr(share->state.header.options); + uint key; + DBUG_ENTER("save_state"); + + options|= HA_OPTION_COMPRESS_RECORD | HA_OPTION_READ_ONLY_DATA; + mi_int2store(share->state.header.options,options); + + share->state.state.data_file_length=new_length; + share->state.state.del=0; + share->state.state.empty=0; + share->state.dellink= HA_OFFSET_ERROR; + share->state.split=(ha_rows) mrg->records; + share->state.version=(ulong) time((time_t*) 0); + if (! maria_is_all_keys_active(share->state.key_map, share->base.keys)) + { + /* + Some indexes are disabled, cannot use current key_file_length value + as an estimate of upper bound of index file size. Use packed data file + size instead. + */ + share->state.state.key_file_length= new_length; + } + /* + If there are no disabled indexes, keep key_file_length value from + original file so "mariachk -rq" can use this value (this is necessary + because index size cannot be easily calculated for fulltext keys) + */ + maria_clear_all_keys_active(share->state.key_map); + for (key=0 ; key < share->base.keys ; key++) + share->state.key_root[key]= HA_OFFSET_ERROR; + for (key=0 ; key < share->state.header.max_block_size ; key++) + share->state.key_del[key]= HA_OFFSET_ERROR; + isam_file->state->checksum=crc; /* Save crc here */ + share->changed=1; /* Force write of header */ + share->state.open_count=0; + share->global_changed=0; + VOID(my_chsize(share->kfile, share->base.keystart, 0, MYF(0))); + if (share->base.keys) + isamchk_neaded=1; + DBUG_RETURN(_ma_state_info_write(share->kfile,&share->state,1+2)); +} + + +static int save_state_mrg(File file,PACK_MRG_INFO *mrg,my_off_t new_length, + ha_checksum crc) +{ + MARIA_STATE_INFO state; + MARIA_HA *isam_file=mrg->file[0]; + uint options; + DBUG_ENTER("save_state_mrg"); + + state= isam_file->s->state; + options= (mi_uint2korr(state.header.options) | HA_OPTION_COMPRESS_RECORD | + HA_OPTION_READ_ONLY_DATA); + mi_int2store(state.header.options,options); + state.state.data_file_length=new_length; + state.state.del=0; + state.state.empty=0; + state.state.records=state.split=(ha_rows) mrg->records; + /* See comment above in save_state about key_file_length handling. */ + if (mrg->src_file_has_indexes_disabled) + { + isam_file->s->state.state.key_file_length= + max(isam_file->s->state.state.key_file_length, new_length); + } + state.dellink= HA_OFFSET_ERROR; + state.version=(ulong) time((time_t*) 0); + maria_clear_all_keys_active(state.key_map); + state.state.checksum=crc; + if (isam_file->s->base.keys) + isamchk_neaded=1; + state.changed=STATE_CHANGED | STATE_NOT_ANALYZED; /* Force check of table */ + DBUG_RETURN (_ma_state_info_write(file,&state,1+2)); +} + + +/* reset for mrg_rrnd */ + +static void mrg_reset(PACK_MRG_INFO *mrg) +{ + if (mrg->current) + { + maria_extra(*mrg->current, HA_EXTRA_NO_CACHE, 0); + mrg->current=0; + } +} + +static int mrg_rrnd(PACK_MRG_INFO *info,byte *buf) +{ + int error; + MARIA_HA *isam_info; + my_off_t filepos; + + if (!info->current) + { + isam_info= *(info->current=info->file); + info->end=info->current+info->count; + maria_extra(isam_info, HA_EXTRA_RESET, 0); + maria_extra(isam_info, HA_EXTRA_CACHE, 0); + filepos=isam_info->s->pack.header_length; + } + else + { + isam_info= *info->current; + filepos= isam_info->nextpos; + } + + for (;;) + { + isam_info->update&= HA_STATE_CHANGED; + if (!(error=(*isam_info->s->read_rnd)(isam_info,(byte*) buf, + filepos, 1)) || + error != HA_ERR_END_OF_FILE) + return (error); + maria_extra(isam_info,HA_EXTRA_NO_CACHE, 0); + if (info->current+1 == info->end) + return(HA_ERR_END_OF_FILE); + info->current++; + isam_info= *info->current; + filepos=isam_info->s->pack.header_length; + maria_extra(isam_info,HA_EXTRA_RESET, 0); + maria_extra(isam_info,HA_EXTRA_CACHE, 0); + } +} + + +static int mrg_close(PACK_MRG_INFO *mrg) +{ + uint i; + int error=0; + for (i=0 ; i < mrg->count ; i++) + error|=maria_close(mrg->file[i]); + if (mrg->free_file) + my_free((gptr) mrg->file,MYF(0)); + return error; +} + + +#if !defined(DBUG_OFF) +/* + Fake the counts to get big Huffman codes. + + SYNOPSIS + fakebigcodes() + huff_counts A pointer to the counts array. + end_count A pointer past the counts array. + + DESCRIPTION + + Huffman coding works by removing the two least frequent values from + the list of values and add a new value with the sum of their + incidences in a loop until only one value is left. Every time a + value is reused for a new value, it gets one more bit for its + encoding. Hence, the least frequent values get the longest codes. + + To get a maximum code length for a value, two of the values must + have an incidence of 1. As their sum is 2, the next infrequent value + must have at least an incidence of 2, then 4, 8, 16 and so on. This + means that one needs 2**n bytes (values) for a code length of n + bits. However, using more distinct values forces the use of longer + codes, or reaching the code length with less total bytes (values). + + To get 64(32)-bit codes, I sort the counts by decreasing incidence. + I assign counts of 1 to the two most frequent values, a count of 2 + for the next one, then 4, 8, and so on until 2**64-1(2**30-1). All + the remaining values get 1. That way every possible byte has an + assigned code, though not all codes are used if not all byte values + are present in the column. + + This strategy would work with distinct column values too, but + requires that at least 64(32) values are present. To make things + easier here, I cancel all distinct column values and force byte + compression for all columns. + + RETURN + void +*/ + +static void fakebigcodes(HUFF_COUNTS *huff_counts, HUFF_COUNTS *end_count) +{ + HUFF_COUNTS *count; + my_off_t *cur_count_p; + my_off_t *end_count_p; + my_off_t **cur_sort_p; + my_off_t **end_sort_p; + my_off_t *sort_counts[256]; + my_off_t total; + DBUG_ENTER("fakebigcodes"); + + for (count= huff_counts; count < end_count; count++) + { + /* + Remove distinct column values. + */ + if (huff_counts->tree_buff) + { + my_free((gptr) huff_counts->tree_buff, MYF(0)); + delete_tree(&huff_counts->int_tree); + huff_counts->tree_buff= NULL; + DBUG_PRINT("fakebigcodes", ("freed distinct column values")); + } + + /* + Sort counts by decreasing incidence. + */ + cur_count_p= count->counts; + end_count_p= cur_count_p + 256; + cur_sort_p= sort_counts; + while (cur_count_p < end_count_p) + *(cur_sort_p++)= cur_count_p++; + (void) qsort(sort_counts, 256, sizeof(my_off_t*), (qsort_cmp) fakecmp); + + /* + Assign faked counts. + */ + cur_sort_p= sort_counts; +#if SIZEOF_LONG_LONG > 4 + end_sort_p= sort_counts + 8 * sizeof(ulonglong) - 1; +#else + end_sort_p= sort_counts + 8 * sizeof(ulonglong) - 2; +#endif + /* Most frequent value gets a faked count of 1. */ + **(cur_sort_p++)= 1; + total= 1; + while (cur_sort_p < end_sort_p) + { + **(cur_sort_p++)= total; + total<<= 1; + } + /* Set the last value. */ + **(cur_sort_p++)= --total; + /* + Set the remaining counts. + */ + end_sort_p= sort_counts + 256; + while (cur_sort_p < end_sort_p) + **(cur_sort_p++)= 1; + } + DBUG_VOID_RETURN; +} + + +/* + Compare two counts for reverse sorting. + + SYNOPSIS + fakecmp() + count1 One count. + count2 Another count. + + RETURN + 1 count1 < count2 + 0 count1 == count2 + -1 count1 > count2 +*/ + +static int fakecmp(my_off_t **count1, my_off_t **count2) +{ + return ((**count1 < **count2) ? 1 : + (**count1 > **count2) ? -1 : 0); +} +#endif diff --git a/storage/maria/maria_rename.sh b/storage/maria/maria_rename.sh new file mode 100755 index 00000000000..fb20e47e635 --- /dev/null +++ b/storage/maria/maria_rename.sh @@ -0,0 +1,17 @@ +#!/bin/sh + +replace myisam maria MYISAM MARIA MyISAM MARIA -- mysql-test/t/*maria*test mysql-test/r/*maria*result + +FILES=`echo sql/ha_maria.{cc,h} include/maria*h storage/maria/*.{c,h}` + +replace myisam maria MYISAM MARIA MyISAM MARIA myisam.h maria.h myisamdef.h maria_def.h mi_ maria_ ft_ maria_ft_ "Copyright (C) 2000" "Copyright (C) 2006" MI_ISAMINFO MARIA_INFO MI_CREATE_INFO MARIA_CREATE_INFO maria_isam_ maria_ MI_INFO MARIA_HA MI_ MARIA_ MARIACHK MARIA_CHK rt_index.h ma_rt_index.h rtree_ maria_rtree rt_key.h ma_rt_key.h rt_mbr.h ma_rt_mbr.h -- $FILES + +replace check_table_is_closed _ma_check_table_is_closed test_if_reopen _ma_test_if_reopen my_n_base_info_read maria_n_base_info_read update_auto_increment _ma_update_auto_increment save_pack_length _ma_save_packlength calc_pack_length _ma_calc_pack_length -- $FILES + +replace mi_ ma_ ft_ ma_ft_ rt_ ma_rt_ myisam maria myisamchk maria_chk myisampack maria_pack myisamlog maria_log -- storage/maria/Makefile.am + +# +# Restore wrong replaces +# + +replace maria_sint1korr mi_sint1korr maria_uint1korr mi_uint1korr maria_sint2korr mi_sint2korr maria_sint3korr mi_sint3korr maria_sint4korr mi_sint4korr maria_sint8korr mi_sint8korr maria_uint2korr mi_uint2korr maria_uint3korr mi_uint3korr maria_uint4korr mi_uint4korr maria_uint5korr mi_uint5korr maria_uint6korr mi_uint6korr maria_uint7korr mi_uint7korr maria_uint8korr mi_uint8korr maria_int1store mi_int1store maria_int2store mi_int2store maria_int3store mi_int3store maria_int4store mi_int4store maria_int5store mi_int5store maria_int6store mi_int6store maria_int7store mi_int7store maria_int8store mi_int8store maria_float4store mi_float4store maria_float4get mi_float4get maria_float8store mi_float8store maria_float8get mi_float8get maria_rowstore mi_rowstore maria_rowkorr mi_rowkorr maria_sizestore mi_sizestore maria_sizekorr mi_sizekorr _maria_maria_ _maria MARIA_MAX_POSSIBLE_KEY HA_MAX_POSSIBLE_KEY MARIA_MAX_KEY_BUFF HA_MAX_KEY_BUFF MARIA_MAX_KEY_SEG HA_MAX_KEY_SEG maria_ft_sintXkorr ft_sintXkorr maria_ft_intXstore ft_intXstore maria_ft_boolean_syntax ft_boolean_syntax maria_ft_min_word_len ft_min_word_len maria_ft_max_word_len ft_max_word_len -- $FILES diff --git a/storage/maria/test_pack b/storage/maria/test_pack new file mode 100755 index 00000000000..689645b1661 --- /dev/null +++ b/storage/maria/test_pack @@ -0,0 +1,10 @@ +silent="-s" +suffix="" + +ma_test1$suffix -s ; maria_pack$suffix --force -s test1 ; maria_chk$suffix -es test1 ; maria_chk$suffix -rqs test1 ; maria_chk$suffix -es test1 ; maria_chk$suffix -us test1 ; maria_chk$suffix -es test1 +ma_test1$suffix -s -S ; maria_pack$suffix --force -s test1 ; maria_chk$suffix -es test1 ; maria_chk$suffix -rqs test1 ; maria_chk$suffix -es test1 ;maria_chk$suffix -us test1 ; maria_chk$suffix -es test1 +ma_test1$suffix -s -b ; maria_pack$suffix --force -s test1 ; maria_chk$suffix -es test1 ; maria_chk$suffix -rqs test1 ; maria_chk$suffix -es test1 +ma_test1$suffix -s -w ; maria_pack$suffix --force -s test1 ; maria_chk$suffix -es test1 ; maria_chk$suffix -ros test1 ; maria_chk$suffix -es test1 + +ma_test2$suffix -s -t4 ; maria_pack$suffix --force -s test2 ; maria_chk$suffix -es test2 ; maria_chk$suffix -ros test2 ; maria_chk$suffix -es test2 ; maria_chk$suffix -s -u test2 ; maria_chk$suffix -sm test2 +ma_test2$suffix -s -t4 -b ; maria_pack$suffix --force -s test2 ; maria_chk$suffix -es test2 ; maria_chk$suffix -ros test2 ; maria_chk$suffix -es test2 ; maria_chk$suffix -s -u test2 ; maria_chk$suffix -sm test2 diff --git a/storage/myisam/Makefile.am b/storage/myisam/Makefile.am index 3c6a5c22234..c67db7c32fb 100644 --- a/storage/myisam/Makefile.am +++ b/storage/myisam/Makefile.am @@ -51,7 +51,8 @@ libmyisam_a_SOURCES = mi_open.c mi_extra.c mi_info.c mi_rkey.c \ mi_delete_table.c mi_rename.c mi_check.c \ mi_keycache.c mi_preload.c \ ft_parser.c ft_stopwords.c ft_static.c \ - ft_update.c ft_boolean_search.c ft_nlq_search.c sort.c \ + ft_update.c ft_boolean_search.c ft_nlq_search.c \ + ft_myisam.c sort.c \ rt_index.c rt_key.c rt_mbr.c rt_split.c sp_key.c CLEANFILES = test?.MY? FT?.MY? isam.log mi_test_all rt_test.MY? sp_test.MY? DEFS = -DMAP_TO_USE_RAID diff --git a/storage/myisam/ft_boolean_search.c b/storage/myisam/ft_boolean_search.c index 4204f211f2e..a4c39f06105 100644 --- a/storage/myisam/ft_boolean_search.c +++ b/storage/myisam/ft_boolean_search.c @@ -150,7 +150,7 @@ static int FTB_WORD_cmp(my_off_t *v, FTB_WORD *a, FTB_WORD *b) static int FTB_WORD_cmp_list(CHARSET_INFO *cs, FTB_WORD **a, FTB_WORD **b) { /* ORDER BY word DESC, ndepth DESC */ - int i= mi_compare_text(cs, (uchar*) (*b)->word+1,(*b)->len-1, + int i= ha_compare_text(cs, (uchar*) (*b)->word+1,(*b)->len-1, (uchar*) (*a)->word+1,(*a)->len-1,0,0); if (!i) i=CMP_NUM((*b)->ndepth,(*a)->ndepth); @@ -183,7 +183,7 @@ static int ftb_query_add_word(void *param, char *word, int word_len, case FT_TOKEN_WORD: ftbw= (FTB_WORD *)alloc_root(&ftb_param->ftb->mem_root, sizeof(FTB_WORD) + - (info->trunc ? MI_MAX_KEY_BUFF : + (info->trunc ? HA_MAX_KEY_BUFF : word_len * ftb_param->ftb->charset->mbmaxlen + HA_FT_WLEN + ftb_param->ftb->info->s->rec_reflength)); @@ -332,7 +332,6 @@ static int _ft2_search(FTB *ftb, FTB_WORD *ftbw, my_bool init_search) byte *lastkey_buf=ftbw->word+ftbw->off; LINT_INIT(off); - LINT_INIT(off); if (ftbw->flags & FTB_FLAG_TRUNC) lastkey_buf+=ftbw->len; @@ -376,7 +375,7 @@ static int _ft2_search(FTB *ftb, FTB_WORD *ftbw, my_bool init_search) if (!r && !ftbw->off) { - r= mi_compare_text(ftb->charset, + r= ha_compare_text(ftb->charset, info->lastkey+1, info->lastkey_length-extra-1, (uchar*) ftbw->word+1, @@ -599,8 +598,9 @@ static int ftb_phrase_add_word(void *param, char *word, int word_len, { FT_WORD *phrase_word= (FT_WORD *)phrase->data; FT_WORD *document_word= (FT_WORD *)document->data; - if (my_strnncoll(phrase_param->cs, phrase_word->pos, phrase_word->len, - document_word->pos, document_word->len)) + if (my_strnncoll(phrase_param->cs, (uchar*) phrase_word->pos, + phrase_word->len, + (uchar*) document_word->pos, document_word->len)) return 0; } phrase_param->match++; @@ -829,7 +829,7 @@ static int ftb_find_relevance_add_word(void *param, char *word, int len, for (a= 0, b= ftb->queue.elements, c= (a+b)/2; b-a>1; c= (a+b)/2) { ftbw= ftb->list[c]; - if (mi_compare_text(ftb->charset, (uchar*)word, len, + if (ha_compare_text(ftb->charset, (uchar*)word, len, (uchar*)ftbw->word+1, ftbw->len-1, (my_bool)(ftbw->flags&FTB_FLAG_TRUNC), 0) > 0) b= c; @@ -839,7 +839,7 @@ static int ftb_find_relevance_add_word(void *param, char *word, int len, for (; c >= 0; c--) { ftbw= ftb->list[c]; - if (mi_compare_text(ftb->charset, (uchar*)word, len, + if (ha_compare_text(ftb->charset, (uchar*)word, len, (uchar*)ftbw->word + 1,ftbw->len - 1, (my_bool)(ftbw->flags & FTB_FLAG_TRUNC), 0)) break; diff --git a/storage/myisam/ft_myisam.c b/storage/myisam/ft_myisam.c new file mode 100644 index 00000000000..76c04ba4c0b --- /dev/null +++ b/storage/myisam/ft_myisam.c @@ -0,0 +1,36 @@ +/* Copyright (C) 2000 MySQL AB & MySQL Finland AB & TCX DataKonsult AB + + 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; either version 2 of the License, or + (at your option) any later version. + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ + +/* Written by Sergei A. Golubchik, who has a shared copyright to this code */ + +/* + This function is for interface functions between fulltext and myisam +*/ + +#include "ftdefs.h" + +FT_INFO *ft_init_search(uint flags, void *info, uint keynr, + byte *query, uint query_len, CHARSET_INFO *cs, + byte *record) +{ + FT_INFO *res; + if (flags & FT_BOOL) + res= ft_init_boolean_search((MI_INFO *)info, keynr, query, query_len,cs); + else + res= ft_init_nlq_search((MI_INFO *)info, keynr, query, query_len, flags, + record); + return res; +} diff --git a/storage/myisam/ft_nlq_search.c b/storage/myisam/ft_nlq_search.c index b4468d8bd95..3e79366a81e 100644 --- a/storage/myisam/ft_nlq_search.c +++ b/storage/myisam/ft_nlq_search.c @@ -104,7 +104,7 @@ static int walk_and_match(FT_WORD *word, uint32 count, ALL_IN_ONE *aio) { if (keylen && - mi_compare_text(aio->charset,info->lastkey+1, + ha_compare_text(aio->charset,info->lastkey+1, info->lastkey_length-extra-1, keybuff+1,keylen-1,0,0)) break; diff --git a/storage/myisam/ft_parser.c b/storage/myisam/ft_parser.c index f333a661ea9..713ae8e8aab 100644 --- a/storage/myisam/ft_parser.c +++ b/storage/myisam/ft_parser.c @@ -34,7 +34,7 @@ typedef struct st_my_ft_parser_param static int FT_WORD_cmp(CHARSET_INFO* cs, FT_WORD *w1, FT_WORD *w2) { - return mi_compare_text(cs, (uchar*) w1->pos, w1->len, + return ha_compare_text(cs, (uchar*) w1->pos, w1->len, (uchar*) w2->pos, w2->len, 0, 0); } diff --git a/storage/myisam/ft_static.c b/storage/myisam/ft_static.c index 6cfb0d59e62..19f65b09ae8 100644 --- a/storage/myisam/ft_static.c +++ b/storage/myisam/ft_static.c @@ -56,19 +56,6 @@ const struct _ft_vft _ft_vft_boolean = { }; -FT_INFO *ft_init_search(uint flags, void *info, uint keynr, - byte *query, uint query_len, CHARSET_INFO *cs, - byte *record) -{ - FT_INFO *res; - if (flags & FT_BOOL) - res= ft_init_boolean_search((MI_INFO *)info, keynr, query, query_len,cs); - else - res= ft_init_nlq_search((MI_INFO *)info, keynr, query, query_len, flags, - record); - return res; -} - const char *ft_stopword_file = 0; const char *ft_precompiled_stopwords[] = { diff --git a/storage/myisam/ft_stopwords.c b/storage/myisam/ft_stopwords.c index 3b5a1752ff7..8da32474690 100644 --- a/storage/myisam/ft_stopwords.c +++ b/storage/myisam/ft_stopwords.c @@ -30,7 +30,7 @@ static TREE *stopwords3=NULL; static int FT_STOPWORD_cmp(void* cmp_arg __attribute__((unused)), FT_STOPWORD *w1, FT_STOPWORD *w2) { - return mi_compare_text(default_charset_info, + return ha_compare_text(default_charset_info, (uchar *)w1->pos,w1->len, (uchar *)w2->pos,w2->len,0,0); } diff --git a/storage/myisam/ft_update.c b/storage/myisam/ft_update.c index 1ec91b41218..548c5c13f36 100644 --- a/storage/myisam/ft_update.c +++ b/storage/myisam/ft_update.c @@ -186,7 +186,7 @@ int _mi_ft_cmp(MI_INFO *info, uint keynr, const byte *rec1, const byte *rec2) { if ((ftsi1.pos != ftsi2.pos) && (!ftsi1.pos || !ftsi2.pos || - mi_compare_text(cs, (uchar*) ftsi1.pos,ftsi1.len, + ha_compare_text(cs, (uchar*) ftsi1.pos,ftsi1.len, (uchar*) ftsi2.pos,ftsi2.len,0,0))) DBUG_RETURN(THOSE_TWO_DAMN_KEYS_ARE_REALLY_DIFFERENT); } @@ -214,7 +214,7 @@ int _mi_ft_update(MI_INFO *info, uint keynr, byte *keybuf, error=0; while(old_word->pos && new_word->pos) { - cmp= mi_compare_text(cs, (uchar*) old_word->pos,old_word->len, + cmp= ha_compare_text(cs, (uchar*) old_word->pos,old_word->len, (uchar*) new_word->pos,new_word->len,0,0); cmp2= cmp ? 0 : (fabs(old_word->weight - new_word->weight) > 1.e-5); diff --git a/storage/myisam/fulltext.h b/storage/myisam/fulltext.h index d8c74d4e94b..90f6ba3bb8b 100644 --- a/storage/myisam/fulltext.h +++ b/storage/myisam/fulltext.h @@ -21,18 +21,8 @@ #include "myisamdef.h" #include "ft_global.h" -#define HA_FT_WTYPE HA_KEYTYPE_FLOAT -#define HA_FT_WLEN 4 -#define FT_SEGS 2 - -#define ft_sintXkorr(A) mi_sint4korr(A) -#define ft_intXstore(T,A) mi_int4store(T,A) - -extern const HA_KEYSEG ft_keysegs[FT_SEGS]; - int _mi_ft_cmp(MI_INFO *, uint, const byte *, const byte *); int _mi_ft_add(MI_INFO *, uint, byte *, const byte *, my_off_t); int _mi_ft_del(MI_INFO *, uint, byte *, const byte *, my_off_t); uint _mi_ft_convert_to_ft2(MI_INFO *, uint, uchar *); - diff --git a/storage/myisam/mi_check.c b/storage/myisam/mi_check.c index eb2f42697ce..e23e220b2d5 100644 --- a/storage/myisam/mi_check.c +++ b/storage/myisam/mi_check.c @@ -35,15 +35,15 @@ /* Functions defined in this file */ -static int check_k_link(MI_CHECK *param, MI_INFO *info,uint nr); -static int chk_index(MI_CHECK *param, MI_INFO *info,MI_KEYDEF *keyinfo, +static int check_k_link(HA_CHECK *param, MI_INFO *info,uint nr); +static int chk_index(HA_CHECK *param, MI_INFO *info,MI_KEYDEF *keyinfo, my_off_t page, uchar *buff, ha_rows *keys, ha_checksum *key_checksum, uint level); static uint isam_key_length(MI_INFO *info,MI_KEYDEF *keyinfo); static ha_checksum calc_checksum(ha_rows count); -static int writekeys(MI_CHECK *param, MI_INFO *info,byte *buff, +static int writekeys(HA_CHECK *param, MI_INFO *info,byte *buff, my_off_t filepos); -static int sort_one_index(MI_CHECK *param, MI_INFO *info,MI_KEYDEF *keyinfo, +static int sort_one_index(HA_CHECK *param, MI_INFO *info,MI_KEYDEF *keyinfo, my_off_t pagepos, File new_file); static int sort_key_read(MI_SORT_PARAM *sort_param,void *key); static int sort_ft_key_read(MI_SORT_PARAM *sort_param,void *key); @@ -57,13 +57,13 @@ static int sort_insert_key(MI_SORT_PARAM *sort_param, reg1 SORT_KEY_BLOCKS *key_block, uchar *key, my_off_t prev_block); static int sort_delete_record(MI_SORT_PARAM *sort_param); -/*static int flush_pending_blocks(MI_CHECK *param);*/ -static SORT_KEY_BLOCKS *alloc_key_blocks(MI_CHECK *param, uint blocks, +/*static int flush_pending_blocks(HA_CHECK *param);*/ +static SORT_KEY_BLOCKS *alloc_key_blocks(HA_CHECK *param, uint blocks, uint buffer_length); static ha_checksum mi_byte_checksum(const byte *buf, uint length); -static void set_data_file_type(SORT_INFO *sort_info, MYISAM_SHARE *share); +static void set_data_file_type(MI_SORT_INFO *sort_info, MYISAM_SHARE *share); -void myisamchk_init(MI_CHECK *param) +void myisamchk_init(HA_CHECK *param) { bzero((gptr) param,sizeof(*param)); param->opt_follow_links=1; @@ -85,7 +85,7 @@ void myisamchk_init(MI_CHECK *param) /* Check the status flags for the table */ -int chk_status(MI_CHECK *param, register MI_INFO *info) +int chk_status(HA_CHECK *param, register MI_INFO *info) { MYISAM_SHARE *share=info->s; @@ -113,7 +113,7 @@ int chk_status(MI_CHECK *param, register MI_INFO *info) /* Check delete links */ -int chk_del(MI_CHECK *param, register MI_INFO *info, uint test_flag) +int chk_del(HA_CHECK *param, register MI_INFO *info, uint test_flag) { reg2 ha_rows i; uint delete_link_length; @@ -222,7 +222,7 @@ wrong: /* Check delete links in index file */ -static int check_k_link(MI_CHECK *param, register MI_INFO *info, uint nr) +static int check_k_link(HA_CHECK *param, register MI_INFO *info, uint nr) { my_off_t next_link; uint block_size=(nr+1)*MI_MIN_KEY_BLOCK_LENGTH; @@ -266,7 +266,7 @@ static int check_k_link(MI_CHECK *param, register MI_INFO *info, uint nr) /* Check sizes of files */ -int chk_size(MI_CHECK *param, register MI_INFO *info) +int chk_size(HA_CHECK *param, register MI_INFO *info) { int error=0; register my_off_t skr,size; @@ -342,7 +342,7 @@ int chk_size(MI_CHECK *param, register MI_INFO *info) /* Check keys */ -int chk_key(MI_CHECK *param, register MI_INFO *info) +int chk_key(HA_CHECK *param, register MI_INFO *info) { uint key,found_keys=0,full_text_keys=0,result=0; ha_rows keys; @@ -528,7 +528,7 @@ do_stat: } /* chk_key */ -static int chk_index_down(MI_CHECK *param, MI_INFO *info, MI_KEYDEF *keyinfo, +static int chk_index_down(HA_CHECK *param, MI_INFO *info, MI_KEYDEF *keyinfo, my_off_t page, uchar *buff, ha_rows *keys, ha_checksum *key_checksum, uint level) { @@ -651,13 +651,13 @@ int mi_collect_stats_nonulls_next(HA_KEYSEG *keyseg, ulonglong *notnull, /* Check if index is ok */ -static int chk_index(MI_CHECK *param, MI_INFO *info, MI_KEYDEF *keyinfo, +static int chk_index(HA_CHECK *param, MI_INFO *info, MI_KEYDEF *keyinfo, my_off_t page, uchar *buff, ha_rows *keys, ha_checksum *key_checksum, uint level) { int flag; uint used_length,comp_flag,nod_flag,key_length=0; - uchar key[MI_MAX_POSSIBLE_KEY_BUFF],*temp_buff,*keypos,*old_keypos,*endpos; + uchar key[HA_MAX_POSSIBLE_KEY_BUFF],*temp_buff,*keypos,*old_keypos,*endpos; my_off_t next_page,record; char llbuff[22]; uint diff_pos[2]; @@ -854,7 +854,7 @@ static uint isam_key_length(MI_INFO *info, register MI_KEYDEF *keyinfo) /* Check that record-link is ok */ -int chk_data_link(MI_CHECK *param, MI_INFO *info,int extend) +int chk_data_link(HA_CHECK *param, MI_INFO *info,int extend) { int error,got_error,flag; uint key,left_length,b_type,field; @@ -864,7 +864,7 @@ int chk_data_link(MI_CHECK *param, MI_INFO *info,int extend) byte *record,*to; char llbuff[22],llbuff2[22],llbuff3[22]; ha_checksum intern_record_checksum; - ha_checksum key_checksum[MI_MAX_POSSIBLE_KEY]; + ha_checksum key_checksum[HA_MAX_POSSIBLE_KEY]; my_bool static_row_size; MI_KEYDEF *keyinfo; MI_BLOCK_INFO block_info; @@ -1296,7 +1296,7 @@ int chk_data_link(MI_CHECK *param, MI_INFO *info,int extend) /* Recover old table by reading each record and writing all keys */ /* Save new datafile-name in temp_filename */ -int mi_repair(MI_CHECK *param, register MI_INFO *info, +int mi_repair(HA_CHECK *param, register MI_INFO *info, my_string name, int rep_quick) { int error,got_error; @@ -1306,7 +1306,7 @@ int mi_repair(MI_CHECK *param, register MI_INFO *info, File new_file; MYISAM_SHARE *share=info->s; char llbuff[22],llbuff2[22]; - SORT_INFO sort_info; + MI_SORT_INFO sort_info; MI_SORT_PARAM sort_param; DBUG_ENTER("mi_repair"); @@ -1573,7 +1573,7 @@ err: /* Uppate keyfile when doing repair */ -static int writekeys(MI_CHECK *param, register MI_INFO *info, byte *buff, +static int writekeys(HA_CHECK *param, register MI_INFO *info, byte *buff, my_off_t filepos) { register uint i; @@ -1686,7 +1686,7 @@ int movepoint(register MI_INFO *info, byte *record, my_off_t oldpos, /* Tell system that we want all memory for our cache */ -void lock_memory(MI_CHECK *param __attribute__((unused))) +void lock_memory(HA_CHECK *param __attribute__((unused))) { #ifdef SUN_OS /* Key-cacheing thrases on sun 4.1 */ if (param->opt_lock_memory) @@ -1702,7 +1702,7 @@ void lock_memory(MI_CHECK *param __attribute__((unused))) /* Flush all changed blocks to disk */ -int flush_blocks(MI_CHECK *param, KEY_CACHE *key_cache, File file) +int flush_blocks(HA_CHECK *param, KEY_CACHE *key_cache, File file) { if (flush_key_blocks(key_cache, file, FLUSH_RELEASE)) { @@ -1717,12 +1717,12 @@ int flush_blocks(MI_CHECK *param, KEY_CACHE *key_cache, File file) /* Sort index for more efficent reads */ -int mi_sort_index(MI_CHECK *param, register MI_INFO *info, my_string name) +int mi_sort_index(HA_CHECK *param, register MI_INFO *info, my_string name) { reg2 uint key; reg1 MI_KEYDEF *keyinfo; File new_file; - my_off_t index_pos[MI_MAX_POSSIBLE_KEY]; + my_off_t index_pos[HA_MAX_POSSIBLE_KEY]; uint r_locks,w_locks; int old_lock; MYISAM_SHARE *share=info->s; @@ -1811,12 +1811,12 @@ err2: /* Sort records recursive using one index */ -static int sort_one_index(MI_CHECK *param, MI_INFO *info, MI_KEYDEF *keyinfo, +static int sort_one_index(HA_CHECK *param, MI_INFO *info, MI_KEYDEF *keyinfo, my_off_t pagepos, File new_file) { uint length,nod_flag,used_length, key_length; uchar *buff,*keypos,*endpos; - uchar key[MI_MAX_POSSIBLE_KEY_BUFF]; + uchar key[HA_MAX_POSSIBLE_KEY_BUFF]; my_off_t new_page_pos,next_page; char llbuff[22]; DBUG_ENTER("sort_one_index"); @@ -1929,7 +1929,7 @@ int change_to_newfile(const char * filename, const char * old_ext, /* Locks a whole file */ /* Gives an error-message if file can't be locked */ -int lock_file(MI_CHECK *param, File file, my_off_t start, int lock_type, +int lock_file(HA_CHECK *param, File file, my_off_t start, int lock_type, const char *filetype, const char *filename) { if (my_lock(file,lock_type,start,F_TO_EOF, @@ -1946,7 +1946,7 @@ int lock_file(MI_CHECK *param, File file, my_off_t start, int lock_type, /* Copy a block between two files */ -int filecopy(MI_CHECK *param, File to,File from,my_off_t start, +int filecopy(HA_CHECK *param, File to,File from,my_off_t start, my_off_t length, const char *type) { char tmp_buff[IO_SIZE],*buff; @@ -1997,7 +1997,7 @@ err: <>0 Error */ -int mi_repair_by_sort(MI_CHECK *param, register MI_INFO *info, +int mi_repair_by_sort(HA_CHECK *param, register MI_INFO *info, const char * name, int rep_quick) { int got_error; @@ -2011,7 +2011,7 @@ int mi_repair_by_sort(MI_CHECK *param, register MI_INFO *info, HA_KEYSEG *keyseg; ulong *rec_per_key_part; char llbuff[22]; - SORT_INFO sort_info; + MI_SORT_INFO sort_info; ulonglong key_map=share->state.key_map; DBUG_ENTER("mi_repair_by_sort"); @@ -2367,7 +2367,7 @@ err: <>0 Error */ -int mi_repair_parallel(MI_CHECK *param, register MI_INFO *info, +int mi_repair_parallel(HA_CHECK *param, register MI_INFO *info, const char * name, int rep_quick) { #ifndef THREAD @@ -2385,7 +2385,7 @@ int mi_repair_parallel(MI_CHECK *param, register MI_INFO *info, HA_KEYSEG *keyseg; char llbuff[22]; IO_CACHE_SHARE io_share; - SORT_INFO sort_info; + MI_SORT_INFO sort_info; ulonglong key_map=share->state.key_map; pthread_attr_t thr_attr; DBUG_ENTER("mi_repair_parallel"); @@ -2774,7 +2774,7 @@ err: static int sort_key_read(MI_SORT_PARAM *sort_param, void *key) { int error; - SORT_INFO *sort_info=sort_param->sort_info; + MI_SORT_INFO *sort_info=sort_param->sort_info; MI_INFO *info=sort_info->info; DBUG_ENTER("sort_key_read"); @@ -2801,7 +2801,7 @@ static int sort_key_read(MI_SORT_PARAM *sort_param, void *key) static int sort_ft_key_read(MI_SORT_PARAM *sort_param, void *key) { int error; - SORT_INFO *sort_info=sort_param->sort_info; + MI_SORT_INFO *sort_info=sort_param->sort_info; MI_INFO *info=sort_info->info; FT_WORD *wptr=0; DBUG_ENTER("sort_ft_key_read"); @@ -2858,8 +2858,8 @@ static int sort_get_next_record(MI_SORT_PARAM *sort_param) my_off_t pos; byte *to; MI_BLOCK_INFO block_info; - SORT_INFO *sort_info=sort_param->sort_info; - MI_CHECK *param=sort_info->param; + MI_SORT_INFO *sort_info=sort_param->sort_info; + HA_CHECK *param=sort_info->param; MI_INFO *info=sort_info->info; MYISAM_SHARE *share=info->s; char llbuff[22],llbuff2[22]; @@ -3246,8 +3246,8 @@ int sort_write_record(MI_SORT_PARAM *sort_param) ulong block_length,reclength; byte *from; byte block_buff[8]; - SORT_INFO *sort_info=sort_param->sort_info; - MI_CHECK *param=sort_info->param; + MI_SORT_INFO *sort_info=sort_param->sort_info; + HA_CHECK *param=sort_info->param; MI_INFO *info=sort_info->info; MYISAM_SHARE *share=info->s; DBUG_ENTER("sort_write_record"); @@ -3273,7 +3273,7 @@ int sort_write_record(MI_SORT_PARAM *sort_param) { /* must be sure that local buffer is big enough */ reclength=info->s->base.pack_reclength+ - _my_calc_total_blob_length(info,sort_param->record)+ + _mi_calc_total_blob_length(info,sort_param->record)+ ALIGN_SIZE(MI_MAX_DYN_BLOCK_HEADER)+MI_SPLIT_LENGTH+ MI_DYN_DELETE_BLOCK_HEADER; if (sort_info->buff_length < reclength) @@ -3361,8 +3361,8 @@ static int sort_key_write(MI_SORT_PARAM *sort_param, const void *a) { uint diff_pos[2]; char llbuff[22],llbuff2[22]; - SORT_INFO *sort_info=sort_param->sort_info; - MI_CHECK *param= sort_info->param; + MI_SORT_INFO *sort_info=sort_param->sort_info; + HA_CHECK *param= sort_info->param; int cmp; if (sort_info->key_block->inited) @@ -3423,7 +3423,7 @@ static int sort_key_write(MI_SORT_PARAM *sort_param, const void *a) int sort_ft_buf_flush(MI_SORT_PARAM *sort_param) { - SORT_INFO *sort_info=sort_param->sort_info; + MI_SORT_INFO *sort_info=sort_param->sort_info; SORT_KEY_BLOCKS *key_block=sort_info->key_block; MYISAM_SHARE *share=sort_info->info->s; uint val_off, val_len; @@ -3470,7 +3470,7 @@ static int sort_ft_key_write(MI_SORT_PARAM *sort_param, const void *a) { uint a_len, val_off, val_len, error; uchar *p; - SORT_INFO *sort_info=sort_param->sort_info; + MI_SORT_INFO *sort_info=sort_param->sort_info; SORT_FT_BUF *ft_buf=sort_info->ft_buf; SORT_KEY_BLOCKS *key_block=sort_info->key_block; @@ -3500,7 +3500,7 @@ static int sort_ft_key_write(MI_SORT_PARAM *sort_param, const void *a) } get_key_full_length_rdonly(val_off, ft_buf->lastkey); - if (mi_compare_text(sort_param->seg->charset, + if (ha_compare_text(sort_param->seg->charset, ((uchar *)a)+1,a_len-1, ft_buf->lastkey+1,val_off-1, 0, 0)==0) { @@ -3575,8 +3575,8 @@ static int sort_insert_key(MI_SORT_PARAM *sort_param, MI_KEY_PARAM s_temp; MI_INFO *info; MI_KEYDEF *keyinfo=sort_param->keyinfo; - SORT_INFO *sort_info= sort_param->sort_info; - MI_CHECK *param=sort_info->param; + MI_SORT_INFO *sort_info= sort_param->sort_info; + HA_CHECK *param=sort_info->param; DBUG_ENTER("sort_insert_key"); anc_buff=key_block->buff; @@ -3654,8 +3654,8 @@ static int sort_delete_record(MI_SORT_PARAM *sort_param) uint i; int old_file,error; uchar *key; - SORT_INFO *sort_info=sort_param->sort_info; - MI_CHECK *param=sort_info->param; + MI_SORT_INFO *sort_info=sort_param->sort_info; + HA_CHECK *param=sort_info->param; MI_INFO *info=sort_info->info; DBUG_ENTER("sort_delete_record"); @@ -3711,7 +3711,7 @@ int flush_pending_blocks(MI_SORT_PARAM *sort_param) uint nod_flag,length; my_off_t filepos,key_file_length; SORT_KEY_BLOCKS *key_block; - SORT_INFO *sort_info= sort_param->sort_info; + MI_SORT_INFO *sort_info= sort_param->sort_info; myf myf_rw=sort_info->param->myf_rw; MI_INFO *info=sort_info->info; MI_KEYDEF *keyinfo=sort_param->keyinfo; @@ -3749,7 +3749,7 @@ int flush_pending_blocks(MI_SORT_PARAM *sort_param) /* alloc space and pointers for key_blocks */ -static SORT_KEY_BLOCKS *alloc_key_blocks(MI_CHECK *param, uint blocks, +static SORT_KEY_BLOCKS *alloc_key_blocks(HA_CHECK *param, uint blocks, uint buffer_length) { reg1 uint i; @@ -3786,7 +3786,7 @@ int test_if_almost_full(MI_INFO *info) /* Recreate table with bigger more alloced record-data */ -int recreate_table(MI_CHECK *param, MI_INFO **org_info, char *filename) +int recreate_table(HA_CHECK *param, MI_INFO **org_info, char *filename) { int error; MI_INFO info; @@ -3953,7 +3953,7 @@ end: /* write suffix to data file if neaded */ -int write_data_suffix(SORT_INFO *sort_info, my_bool fix_datafile) +int write_data_suffix(MI_SORT_INFO *sort_info, my_bool fix_datafile) { MI_INFO *info=sort_info->info; @@ -3974,7 +3974,7 @@ int write_data_suffix(SORT_INFO *sort_info, my_bool fix_datafile) /* Update state and myisamchk_time of indexfile */ -int update_state_info(MI_CHECK *param, MI_INFO *info,uint update) +int update_state_info(HA_CHECK *param, MI_INFO *info,uint update) { MYISAM_SHARE *share=info->s; @@ -4046,7 +4046,7 @@ err: param->auto_increment is bigger than the biggest key. */ -void update_auto_increment_key(MI_CHECK *param, MI_INFO *info, +void update_auto_increment_key(HA_CHECK *param, MI_INFO *info, my_bool repair_only) { byte *record; @@ -4278,7 +4278,7 @@ my_bool mi_test_if_sort_rep(MI_INFO *info, ha_rows rows, static void -set_data_file_type(SORT_INFO *sort_info, MYISAM_SHARE *share) +set_data_file_type(MI_SORT_INFO *sort_info, MYISAM_SHARE *share) { if ((sort_info->new_data_file_type=share->data_file_type) == COMPRESSED_RECORD && sort_info->param->testflag & T_UNPACK) diff --git a/storage/myisam/mi_create.c b/storage/myisam/mi_create.c index 3be998b2c17..d6ffba596f5 100644 --- a/storage/myisam/mi_create.c +++ b/storage/myisam/mi_create.c @@ -56,7 +56,7 @@ int mi_create(const char *name,uint keys,MI_KEYDEF *keydefs, HA_KEYSEG *keyseg,tmp_keyseg; MI_COLUMNDEF *rec; ulong *rec_per_key_part; - my_off_t key_root[MI_MAX_POSSIBLE_KEY],key_del[MI_MAX_KEY_BLOCK_SIZE]; + my_off_t key_root[HA_MAX_POSSIBLE_KEY],key_del[MI_MAX_KEY_BLOCK_SIZE]; MI_CREATE_INFO tmp_create_info; DBUG_ENTER("mi_create"); @@ -92,7 +92,7 @@ int mi_create(const char *name,uint keys,MI_KEYDEF *keydefs, ci->reloc_rows=ci->max_rows; /* Check if wrong parameter */ if (!(rec_per_key_part= - (ulong*) my_malloc((keys + uniques)*MI_MAX_KEY_SEG*sizeof(long), + (ulong*) my_malloc((keys + uniques)*HA_MAX_KEY_SEG*sizeof(long), MYF(MY_WME | MY_ZEROFILL)))) DBUG_RETURN(my_errno); @@ -414,7 +414,7 @@ int mi_create(const char *name,uint keys,MI_KEYDEF *keydefs, } } /* if HA_FULLTEXT */ key_segs+=keydef->keysegs; - if (keydef->keysegs > MI_MAX_KEY_SEG) + if (keydef->keysegs > HA_MAX_KEY_SEG) { my_errno=HA_WRONG_CREATE_OPTION; goto err; @@ -431,7 +431,7 @@ int mi_create(const char *name,uint keys,MI_KEYDEF *keydefs, keydef->block_length= MI_BLOCK_SIZE(length-real_length_diff, pointer,MI_MAX_KEYPTR_SIZE); if (keydef->block_length > MI_MAX_KEY_BLOCK_LENGTH || - length >= MI_MAX_KEY_BUFF) + length >= HA_MAX_KEY_BUFF) { my_errno=HA_WRONG_CREATE_OPTION; goto err; diff --git a/storage/myisam/mi_delete.c b/storage/myisam/mi_delete.c index 2bc99d65dd2..ae464936653 100644 --- a/storage/myisam/mi_delete.c +++ b/storage/myisam/mi_delete.c @@ -160,7 +160,7 @@ static int _mi_ck_real_delete(register MI_INFO *info, MI_KEYDEF *keyinfo, DBUG_RETURN(my_errno=HA_ERR_CRASHED); } if (!(root_buff= (uchar*) my_alloca((uint) keyinfo->block_length+ - MI_MAX_KEY_BUFF*2))) + HA_MAX_KEY_BUFF*2))) { DBUG_PRINT("error",("Couldn't allocate memory")); DBUG_RETURN(my_errno=ENOMEM); @@ -222,7 +222,7 @@ static int d_search(register MI_INFO *info, register MI_KEYDEF *keyinfo, my_bool last_key; uchar *leaf_buff,*keypos; my_off_t leaf_page,next_block; - uchar lastkey[MI_MAX_KEY_BUFF]; + uchar lastkey[HA_MAX_KEY_BUFF]; DBUG_ENTER("d_search"); DBUG_DUMP("page",(byte*) anc_buff,mi_getint(anc_buff)); @@ -307,7 +307,7 @@ static int d_search(register MI_INFO *info, register MI_KEYDEF *keyinfo, { leaf_page=_mi_kpos(nod_flag,keypos); if (!(leaf_buff= (uchar*) my_alloca((uint) keyinfo->block_length+ - MI_MAX_KEY_BUFF*2))) + HA_MAX_KEY_BUFF*2))) { DBUG_PRINT("error",("Couldn't allocate memory")); my_errno=ENOMEM; @@ -406,7 +406,7 @@ static int del(register MI_INFO *info, register MI_KEYDEF *keyinfo, uchar *key, int ret_value,length; uint a_length,nod_flag,tmp; my_off_t next_page; - uchar keybuff[MI_MAX_KEY_BUFF],*endpos,*next_buff,*key_start, *prev_key; + uchar keybuff[HA_MAX_KEY_BUFF],*endpos,*next_buff,*key_start, *prev_key; MYISAM_SHARE *share=info->s; MI_KEY_PARAM s_temp; DBUG_ENTER("del"); @@ -423,7 +423,7 @@ static int del(register MI_INFO *info, register MI_KEYDEF *keyinfo, uchar *key, { next_page= _mi_kpos(nod_flag,endpos); if (!(next_buff= (uchar*) my_alloca((uint) keyinfo->block_length+ - MI_MAX_KEY_BUFF*2))) + HA_MAX_KEY_BUFF*2))) DBUG_RETURN(-1); if (!_mi_fetch_keypage(info,keyinfo,next_page,DFLT_INIT_HITS,next_buff,0)) ret_value= -1; @@ -510,7 +510,7 @@ static int underflow(register MI_INFO *info, register MI_KEYDEF *keyinfo, uint length,anc_length,buff_length,leaf_length,p_length,s_length,nod_flag, key_reflength,key_length; my_off_t next_page; - uchar anc_key[MI_MAX_KEY_BUFF],leaf_key[MI_MAX_KEY_BUFF], + uchar anc_key[HA_MAX_KEY_BUFF],leaf_key[HA_MAX_KEY_BUFF], *buff,*endpos,*next_keypos,*anc_pos,*half_pos,*temp_pos,*prev_key, *after_key; MI_KEY_PARAM s_temp; diff --git a/storage/myisam/mi_delete_all.c b/storage/myisam/mi_delete_all.c index 51f1e44d6d2..1e2fcac4486 100644 --- a/storage/myisam/mi_delete_all.c +++ b/storage/myisam/mi_delete_all.c @@ -22,7 +22,6 @@ int mi_delete_all_rows(MI_INFO *info) { uint i; - char buf[22]; MYISAM_SHARE *share=info->s; MI_STATE_INFO *state=&share->state; DBUG_ENTER("mi_delete_all_rows"); diff --git a/storage/myisam/mi_dynrec.c b/storage/myisam/mi_dynrec.c index 36d88bd362a..44367fecc1c 100644 --- a/storage/myisam/mi_dynrec.c +++ b/storage/myisam/mi_dynrec.c @@ -240,7 +240,7 @@ int _mi_write_blob_record(MI_INFO *info, const byte *record) extra= (ALIGN_SIZE(MI_MAX_DYN_BLOCK_HEADER)+MI_SPLIT_LENGTH+ MI_DYN_DELETE_BLOCK_HEADER+1); reclength= (info->s->base.pack_reclength + - _my_calc_total_blob_length(info,record)+ extra); + _mi_calc_total_blob_length(info,record)+ extra); #ifdef NOT_USED /* We now support big rows */ if (reclength > MI_DYN_MAX_ROW_LENGTH) { @@ -274,7 +274,7 @@ int _mi_update_blob_record(MI_INFO *info, my_off_t pos, const byte *record) extra= (ALIGN_SIZE(MI_MAX_DYN_BLOCK_HEADER)+MI_SPLIT_LENGTH+ MI_DYN_DELETE_BLOCK_HEADER); reclength= (info->s->base.pack_reclength+ - _my_calc_total_blob_length(info,record)+ extra); + _mi_calc_total_blob_length(info,record)+ extra); #ifdef NOT_USED /* We now support big rows */ if (reclength > MI_DYN_MAX_ROW_LENGTH) { @@ -1244,7 +1244,7 @@ err: /* Calc length of blob. Update info in blobs->length */ -ulong _my_calc_total_blob_length(MI_INFO *info, const byte *record) +ulong _mi_calc_total_blob_length(MI_INFO *info, const byte *record) { ulong length; MI_BLOB *blob,*end; @@ -1278,7 +1278,7 @@ ulong _mi_calc_blob_length(uint length, const byte *pos) } -void _my_store_blob_length(byte *pos,uint pack_length,uint length) +void _mi_store_blob_length(byte *pos,uint pack_length,uint length) { switch (pack_length) { case 1: @@ -1431,7 +1431,7 @@ int _mi_cmp_dynamic_record(register MI_INFO *info, register const byte *record) if (info->s->base.blobs) { if (!(buffer=(byte*) my_alloca(info->s->base.pack_reclength+ - _my_calc_total_blob_length(info,record)))) + _mi_calc_total_blob_length(info,record)))) DBUG_RETURN(-1); } reclength=_mi_rec_pack(info,buffer,record); diff --git a/storage/myisam/mi_key.c b/storage/myisam/mi_key.c index f8463a0b6b0..859912dfdab 100644 --- a/storage/myisam/mi_key.c +++ b/storage/myisam/mi_key.c @@ -448,7 +448,7 @@ static int _mi_put_key_in_record(register MI_INFO *info, uint keynr, /* The above changed info->lastkey2. Inform mi_rnext_same(). */ info->update&= ~HA_STATE_RNEXT_SAME; - _my_store_blob_length(record+keyseg->start, + _mi_store_blob_length(record+keyseg->start, (uint) keyseg->bit_start,length); key+=length; } diff --git a/storage/myisam/mi_log.c b/storage/myisam/mi_log.c index 13842c56828..40566df10d8 100644 --- a/storage/myisam/mi_log.c +++ b/storage/myisam/mi_log.c @@ -134,7 +134,7 @@ void _myisam_log_record(enum myisam_log_commands command, MI_INFO *info, if (!info->s->base.blobs) length=info->s->base.reclength; else - length=info->s->base.reclength+ _my_calc_total_blob_length(info,record); + length=info->s->base.reclength+ _mi_calc_total_blob_length(info,record); buff[0]=(char) command; mi_int2store(buff+1,info->dfile); mi_int4store(buff+3,pid); diff --git a/storage/myisam/mi_open.c b/storage/myisam/mi_open.c index abf1d1ea9a7..a00bd6be008 100644 --- a/storage/myisam/mi_open.c +++ b/storage/myisam/mi_open.c @@ -83,8 +83,8 @@ MI_INFO *mi_open(const char *name, int mode, uint open_flags) char *disk_cache, *disk_pos, *end_pos; MI_INFO info,*m_info,*old_info; MYISAM_SHARE share_buff,*share; - ulong rec_per_key_part[MI_MAX_POSSIBLE_KEY*MI_MAX_KEY_SEG]; - my_off_t key_root[MI_MAX_POSSIBLE_KEY],key_del[MI_MAX_KEY_BLOCK_SIZE]; + ulong rec_per_key_part[HA_MAX_POSSIBLE_KEY*HA_MAX_KEY_SEG]; + my_off_t key_root[HA_MAX_POSSIBLE_KEY],key_del[MI_MAX_KEY_BLOCK_SIZE]; ulonglong max_key_file_length, max_data_file_length; DBUG_ENTER("mi_open"); @@ -105,7 +105,8 @@ MI_INFO *mi_open(const char *name, int mode, uint open_flags) share_buff.state.rec_per_key_part=rec_per_key_part; share_buff.state.key_root=key_root; share_buff.state.key_del=key_del; - share_buff.key_cache= multi_key_cache_search(name_buff, strlen(name_buff)); + share_buff.key_cache= multi_key_cache_search(name_buff, strlen(name_buff), + dflt_key_cache); DBUG_EXECUTE_IF("myisam_pretend_crashed_table_on_open", if (strstr(name, "/t1")) @@ -211,7 +212,7 @@ MI_INFO *mi_open(const char *name, int mode, uint open_flags) len,MI_BASE_INFO_SIZE)); } disk_pos= (char*) - my_n_base_info_read((uchar*) disk_cache + base_pos, &share->base); + mi_n_base_info_read((uchar*) disk_cache + base_pos, &share->base); share->state.state_length=base_pos; if (!(open_flags & HA_OPEN_FOR_REPAIR) && @@ -233,8 +234,8 @@ MI_INFO *mi_open(const char *name, int mode, uint open_flags) } key_parts+=fulltext_keys*FT_SEGS; - if (share->base.max_key_length > MI_MAX_KEY_BUFF || keys > MI_MAX_KEY || - key_parts >= MI_MAX_KEY * MI_MAX_KEY_SEG) + if (share->base.max_key_length > HA_MAX_KEY_BUFF || keys > MI_MAX_KEY || + key_parts >= MI_MAX_KEY * HA_MAX_KEY_SEG) { DBUG_PRINT("error",("Wrong key info: Max_key_length: %d keys: %d key_parts: %d", share->base.max_key_length, keys, key_parts)); my_errno=HA_ERR_UNSUPPORTED; @@ -990,7 +991,7 @@ uint mi_base_info_write(File file, MI_BASE_INFO *base) } -uchar *my_n_base_info_read(uchar *ptr, MI_BASE_INFO *base) +uchar *mi_n_base_info_read(uchar *ptr, MI_BASE_INFO *base) { base->keystart = mi_sizekorr(ptr); ptr +=8; base->max_data_file_length = mi_sizekorr(ptr); ptr +=8; diff --git a/storage/myisam/mi_packrec.c b/storage/myisam/mi_packrec.c index aa6ea016070..89b0833742e 100644 --- a/storage/myisam/mi_packrec.c +++ b/storage/myisam/mi_packrec.c @@ -102,6 +102,7 @@ static void init_bit_buffer(MI_BIT_BUFF *bit_buff,uchar *buffer,uint length); static uint fill_and_get_bits(MI_BIT_BUFF *bit_buff,uint count); static void fill_buffer(MI_BIT_BUFF *bit_buff); static uint max_bit(uint value); +static uint read_pack_length(uint version, const uchar *buf, ulong *length); #ifdef HAVE_MMAP static uchar *_mi_mempack_get_block_info(MI_INFO *myisam,MI_BLOCK_INFO *info, uchar *header); @@ -775,7 +776,7 @@ static void uf_blob(MI_COLUMNDEF *rec, MI_BIT_BUFF *bit_buff, return; } decode_bytes(rec,bit_buff,bit_buff->blob_pos,bit_buff->blob_pos+length); - _my_store_blob_length((byte*) to,pack_length,length); + _mi_store_blob_length((byte*) to,pack_length,length); memcpy_fixed((char*) to+pack_length,(char*) &bit_buff->blob_pos, sizeof(char*)); bit_buff->blob_pos+=length; @@ -1178,7 +1179,6 @@ static int _mi_read_rnd_mempack_record(MI_INFO*, byte *,my_off_t, my_bool); my_bool _mi_memmap_file(MI_INFO *info) { - byte *file_map; MYISAM_SHARE *share=info->s; DBUG_ENTER("mi_memmap_file"); @@ -1315,7 +1315,7 @@ uint save_pack_length(uint version, byte *block_buff, ulong length) } -uint read_pack_length(uint version, const uchar *buf, ulong *length) +static uint read_pack_length(uint version, const uchar *buf, ulong *length) { if (buf[0] < 254) { diff --git a/storage/myisam/mi_range.c b/storage/myisam/mi_range.c index e78f3b11625..1c41c96e95f 100644 --- a/storage/myisam/mi_range.c +++ b/storage/myisam/mi_range.c @@ -217,7 +217,7 @@ static uint _mi_keynr(MI_INFO *info, register MI_KEYDEF *keyinfo, uchar *page, uchar *keypos, uint *ret_max_key) { uint nod_flag,keynr,max_key; - uchar t_buff[MI_MAX_KEY_BUFF],*end; + uchar t_buff[HA_MAX_KEY_BUFF],*end; end= page+mi_getint(page); nod_flag=mi_test_if_nod(page); diff --git a/storage/myisam/mi_search.c b/storage/myisam/mi_search.c index a6c2cbd6082..b2c5cba45d9 100644 --- a/storage/myisam/mi_search.c +++ b/storage/myisam/mi_search.c @@ -61,7 +61,7 @@ int _mi_search(register MI_INFO *info, register MI_KEYDEF *keyinfo, int error,flag; uint nod_flag; uchar *keypos,*maxpos; - uchar lastkey[MI_MAX_KEY_BUFF],*buff; + uchar lastkey[HA_MAX_KEY_BUFF],*buff; DBUG_ENTER("_mi_search"); DBUG_PRINT("enter",("pos: %lu nextflag: %u lastpos: %lu", (ulong) pos, nextflag, (ulong) info->lastpos)); @@ -243,7 +243,7 @@ int _mi_seq_search(MI_INFO *info, register MI_KEYDEF *keyinfo, uchar *page, { int flag; uint nod_flag,length,not_used[2]; - uchar t_buff[MI_MAX_KEY_BUFF],*end; + uchar t_buff[HA_MAX_KEY_BUFF],*end; DBUG_ENTER("_mi_seq_search"); LINT_INIT(flag); LINT_INIT(length); @@ -296,7 +296,7 @@ int _mi_prefix_search(MI_INFO *info, register MI_KEYDEF *keyinfo, uchar *page, int key_len_skip, seg_len_pack, key_len_left; uchar *end, *kseg, *vseg; uchar *sort_order=keyinfo->seg->charset->sort_order; - uchar tt_buff[MI_MAX_KEY_BUFF+2], *t_buff=tt_buff+2; + uchar tt_buff[HA_MAX_KEY_BUFF+2], *t_buff=tt_buff+2; uchar *saved_from, *saved_to, *saved_vseg; uint saved_length=0, saved_prefix_len=0; uint length_pack; @@ -919,7 +919,7 @@ uint _mi_get_binary_pack_key(register MI_KEYDEF *keyinfo, uint nod_flag, DBUG_ENTER("_mi_get_binary_pack_key"); page= *page_pos; - page_end=page+MI_MAX_KEY_BUFF+1; + page_end=page+HA_MAX_KEY_BUFF+1; start_key=key; /* @@ -1207,7 +1207,7 @@ int _mi_search_next(register MI_INFO *info, register MI_KEYDEF *keyinfo, { int error; uint nod_flag; - uchar lastkey[MI_MAX_KEY_BUFF]; + uchar lastkey[HA_MAX_KEY_BUFF]; DBUG_ENTER("_mi_search_next"); DBUG_PRINT("enter",("nextflag: %u lastpos: %lu int_keypos: %lu", nextflag, (ulong) info->lastpos, diff --git a/storage/myisam/mi_test1.c b/storage/myisam/mi_test1.c index 0e62b074376..9f7c55444c1 100644 --- a/storage/myisam/mi_test1.c +++ b/storage/myisam/mi_test1.c @@ -628,7 +628,7 @@ get_one_option(int optid, const struct my_option *opt __attribute__((unused)), key_type= HA_KEYTYPE_VARTEXT1; break; case 'k': - if (key_length < 4 || key_length > MI_MAX_KEY_LENGTH) + if (key_length < 4 || key_length > HA_MAX_KEY_LENGTH) { fprintf(stderr,"Wrong key length\n"); exit(1); diff --git a/storage/myisam/mi_test2.c b/storage/myisam/mi_test2.c index e77a37d853f..fb118f8cad2 100644 --- a/storage/myisam/mi_test2.c +++ b/storage/myisam/mi_test2.c @@ -813,7 +813,7 @@ end: printf("Write records: %d\nUpdate records: %d\nSame-key-read: %d\nDelete records: %d\n", write_count,update,dupp_keys,opt_delete); if (rec_pointer_size) printf("Record pointer size: %d\n",rec_pointer_size); - printf("myisam_block_size: %u\n", myisam_block_size); + printf("myisam_block_size: %lu\n", myisam_block_size); if (key_cacheing) { puts("Key cache used"); diff --git a/storage/myisam/mi_unique.c b/storage/myisam/mi_unique.c index b698968127b..2779577e317 100644 --- a/storage/myisam/mi_unique.c +++ b/storage/myisam/mi_unique.c @@ -57,7 +57,7 @@ my_bool mi_check_unique(MI_INFO *info, MI_UNIQUEDEF *def, byte *record, if (_mi_search_next(info,info->s->keyinfo+def->key, info->lastkey, MI_UNIQUE_HASH_LENGTH, SEARCH_BIGGER, info->s->state.key_root[def->key]) || - bcmp(info->lastkey, key_buff, MI_UNIQUE_HASH_LENGTH)) + memcmp((char*) info->lastkey, (char*) key_buff, MI_UNIQUE_HASH_LENGTH)) { info->page_changed=1; /* Can't optimize read next */ info->lastpos=lastpos; @@ -213,7 +213,7 @@ int mi_unique_comp(MI_UNIQUEDEF *def, const byte *a, const byte *b, if (type == HA_KEYTYPE_TEXT || type == HA_KEYTYPE_VARTEXT1 || type == HA_KEYTYPE_VARTEXT2) { - if (mi_compare_text(keyseg->charset, (uchar *) pos_a, a_length, + if (ha_compare_text(keyseg->charset, (uchar *) pos_a, a_length, (uchar *) pos_b, b_length, 0, 1)) return 1; } diff --git a/storage/myisam/mi_update.c b/storage/myisam/mi_update.c index 937c9983b45..063c48bb7f0 100644 --- a/storage/myisam/mi_update.c +++ b/storage/myisam/mi_update.c @@ -24,7 +24,7 @@ int mi_update(register MI_INFO *info, const byte *oldrec, byte *newrec) int flag,key_changed,save_errno; reg3 my_off_t pos; uint i; - uchar old_key[MI_MAX_KEY_BUFF],*new_key; + uchar old_key[HA_MAX_KEY_BUFF],*new_key; bool auto_key_changed=0; ulonglong changed; MYISAM_SHARE *share=info->s; diff --git a/storage/myisam/mi_write.c b/storage/myisam/mi_write.c index 5e79b2937cc..6291d176612 100644 --- a/storage/myisam/mi_write.c +++ b/storage/myisam/mi_write.c @@ -334,7 +334,7 @@ static int w_search(register MI_INFO *info, register MI_KEYDEF *keyinfo, int error,flag; uint nod_flag, search_key_length; uchar *temp_buff,*keypos; - uchar keybuff[MI_MAX_KEY_BUFF]; + uchar keybuff[HA_MAX_KEY_BUFF]; my_bool was_last_key; my_off_t next_page, dupp_key_pos; DBUG_ENTER("w_search"); @@ -342,7 +342,7 @@ static int w_search(register MI_INFO *info, register MI_KEYDEF *keyinfo, search_key_length= (comp_flag & SEARCH_FIND) ? key_length : USE_WHOLE_KEY; if (!(temp_buff= (uchar*) my_alloca((uint) keyinfo->block_length+ - MI_MAX_KEY_BUFF*2))) + HA_MAX_KEY_BUFF*2))) DBUG_RETURN(-1); if (!_mi_fetch_keypage(info,keyinfo,page,DFLT_INIT_HITS,temp_buff,0)) goto err; @@ -533,7 +533,7 @@ int _mi_insert(register MI_INFO *info, register MI_KEYDEF *keyinfo, get_key_length(alen,a); DBUG_ASSERT(info->ft1_to_ft2==0); if (alen == blen && - mi_compare_text(keyinfo->seg->charset, a, alen, b, blen, 0, 0)==0) + ha_compare_text(keyinfo->seg->charset, a, alen, b, blen, 0, 0)==0) { /* yup. converting */ info->ft1_to_ft2=(DYNAMIC_ARRAY *) @@ -693,7 +693,7 @@ static uchar *_mi_find_last_pos(MI_KEYDEF *keyinfo, uchar *page, { uint keys,length,last_length,key_ref_length; uchar *end,*lastpos,*prevpos; - uchar key_buff[MI_MAX_KEY_BUFF]; + uchar key_buff[HA_MAX_KEY_BUFF]; DBUG_ENTER("_mi_find_last_pos"); key_ref_length=2; @@ -749,7 +749,7 @@ static int _mi_balance_page(register MI_INFO *info, MI_KEYDEF *keyinfo, length,keys; uchar *pos,*buff,*extra_buff; my_off_t next_page,new_pos; - byte tmp_part_key[MI_MAX_KEY_BUFF]; + byte tmp_part_key[HA_MAX_KEY_BUFF]; DBUG_ENTER("_mi_balance_page"); k_length=keyinfo->keylength; @@ -915,7 +915,7 @@ static int keys_free(uchar *key, TREE_FREE mode, bulk_insert_param *param) Probably I can use info->lastkey here, but I'm not sure, and to be safe I'd better use local lastkey. */ - uchar lastkey[MI_MAX_KEY_BUFF]; + uchar lastkey[HA_MAX_KEY_BUFF]; uint keylen; MI_KEYDEF *keyinfo; diff --git a/storage/myisam/myisamchk.c b/storage/myisam/myisamchk.c index e2c8b446322..192500e432e 100644 --- a/storage/myisam/myisamchk.c +++ b/storage/myisam/myisamchk.c @@ -17,7 +17,6 @@ /* Describe, check and repair of MyISAM tables */ #include "fulltext.h" - #include #include #include @@ -72,9 +71,9 @@ static const char *myisam_stats_method_str="nulls_unequal"; static void get_options(int *argc,char * * *argv); static void print_version(void); static void usage(void); -static int myisamchk(MI_CHECK *param, char *filename); -static void descript(MI_CHECK *param, register MI_INFO *info, my_string name); -static int mi_sort_records(MI_CHECK *param, register MI_INFO *info, +static int myisamchk(HA_CHECK *param, char *filename); +static void descript(HA_CHECK *param, register MI_INFO *info, my_string name); +static int mi_sort_records(HA_CHECK *param, register MI_INFO *info, my_string name, uint sort_key, my_bool write_info, my_bool update_index); static int sort_record_index(MI_SORT_PARAM *sort_param, MI_INFO *info, @@ -82,7 +81,7 @@ static int sort_record_index(MI_SORT_PARAM *sort_param, MI_INFO *info, my_off_t page,uchar *buff,uint sortkey, File new_file, my_bool update_index); -MI_CHECK check_param; +HA_CHECK check_param; /* Main program */ @@ -704,7 +703,7 @@ get_one_option(int optid, case OPT_STATS_METHOD: { int method; - enum_mi_stats_method method_conv; + enum_handler_stats_method method_conv; myisam_stats_method_str= argument; if ((method=find_type(argument, &myisam_stats_method_typelib, 2)) <= 0) { @@ -801,7 +800,7 @@ static void get_options(register int *argc,register char ***argv) /* Check table */ -static int myisamchk(MI_CHECK *param, my_string filename) +static int myisamchk(HA_CHECK *param, my_string filename) { int error,lock_type,recreate; int rep_quick= param->testflag & (T_QUICK | T_FORCE_UNIQUENESS); @@ -1206,7 +1205,7 @@ end2: /* Write info about table */ -static void descript(MI_CHECK *param, register MI_INFO *info, my_string name) +static void descript(HA_CHECK *param, register MI_INFO *info, my_string name) { uint key,keyseg_nr,field,start; reg3 MI_KEYDEF *keyinfo; @@ -1471,7 +1470,7 @@ static void descript(MI_CHECK *param, register MI_INFO *info, my_string name) /* Sort records according to one key */ -static int mi_sort_records(MI_CHECK *param, +static int mi_sort_records(HA_CHECK *param, register MI_INFO *info, my_string name, uint sort_key, my_bool write_info, @@ -1485,7 +1484,7 @@ static int mi_sort_records(MI_CHECK *param, ha_rows old_record_count; MYISAM_SHARE *share=info->s; char llbuff[22],llbuff2[22]; - SORT_INFO sort_info; + MI_SORT_INFO sort_info; MI_SORT_PARAM sort_param; DBUG_ENTER("sort_records"); @@ -1660,10 +1659,10 @@ static int sort_record_index(MI_SORT_PARAM *sort_param,MI_INFO *info, uint nod_flag,used_length,key_length; uchar *temp_buff,*keypos,*endpos; my_off_t next_page,rec_pos; - uchar lastkey[MI_MAX_KEY_BUFF]; + uchar lastkey[HA_MAX_KEY_BUFF]; char llbuff[22]; - SORT_INFO *sort_info= sort_param->sort_info; - MI_CHECK *param=sort_info->param; + MI_SORT_INFO *sort_info= sort_param->sort_info; + HA_CHECK *param=sort_info->param; DBUG_ENTER("sort_record_index"); nod_flag=mi_test_if_nod(buff); @@ -1751,7 +1750,7 @@ err: static int not_killed= 0; -volatile int *killed_ptr(MI_CHECK *param __attribute__((unused))) +volatile int *killed_ptr(HA_CHECK *param __attribute__((unused))) { return ¬_killed; /* always NULL */ } @@ -1759,7 +1758,7 @@ volatile int *killed_ptr(MI_CHECK *param __attribute__((unused))) /* print warnings and errors */ /* VARARGS */ -void mi_check_print_info(MI_CHECK *param __attribute__((unused)), +void mi_check_print_info(HA_CHECK *param __attribute__((unused)), const char *fmt,...) { va_list args; @@ -1772,7 +1771,7 @@ void mi_check_print_info(MI_CHECK *param __attribute__((unused)), /* VARARGS */ -void mi_check_print_warning(MI_CHECK *param, const char *fmt,...) +void mi_check_print_warning(HA_CHECK *param, const char *fmt,...) { va_list args; DBUG_ENTER("mi_check_print_warning"); @@ -1797,7 +1796,7 @@ void mi_check_print_warning(MI_CHECK *param, const char *fmt,...) /* VARARGS */ -void mi_check_print_error(MI_CHECK *param, const char *fmt,...) +void mi_check_print_error(HA_CHECK *param, const char *fmt,...) { va_list args; DBUG_ENTER("mi_check_print_error"); diff --git a/storage/myisam/myisamdef.h b/storage/myisam/myisamdef.h index caf6254c321..44b868fedb2 100644 --- a/storage/myisam/myisamdef.h +++ b/storage/myisam/myisamdef.h @@ -16,8 +16,8 @@ /* This file is included by all internal myisam files */ -#include "myisam.h" /* Structs & some defines */ -#include "myisampack.h" /* packing of keys */ +#include "myisam.h" /* Structs & some defines */ +#include "myisampack.h" /* packing of keys */ #include #ifdef THREAD #include @@ -27,15 +27,16 @@ #endif #if defined(my_write) && !defined(MAP_TO_USE_RAID) -#undef my_write /* undef map from my_nosys; We need test-if-disk full */ +/* undef map from my_nosys; We need test-if-disk full */ +#undef my_write #endif typedef struct st_mi_status_info { - ha_rows records; /* Rows in table */ - ha_rows del; /* Removed rows */ - my_off_t empty; /* lost space in datafile */ - my_off_t key_empty; /* lost space in indexfile */ + ha_rows records; /* Rows in table */ + ha_rows del; /* Removed rows */ + my_off_t empty; /* lost space in datafile */ + my_off_t key_empty; /* lost space in indexfile */ my_off_t key_file_length; my_off_t data_file_length; ha_checksum checksum; @@ -43,333 +44,293 @@ typedef struct st_mi_status_info typedef struct st_mi_state_info { - struct { /* Fileheader */ + struct + { /* Fileheader */ uchar file_version[4]; uchar options[2]; uchar header_length[2]; uchar state_info_length[2]; uchar base_info_length[2]; uchar base_pos[2]; - uchar key_parts[2]; /* Key parts */ - uchar unique_key_parts[2]; /* Key parts + unique parts */ - uchar keys; /* number of keys in file */ - uchar uniques; /* number of UNIQUE definitions */ - uchar language; /* Language for indexes */ - uchar max_block_size; /* max keyblock size */ + uchar key_parts[2]; /* Key parts */ + uchar unique_key_parts[2]; /* Key parts + unique parts */ + uchar keys; /* number of keys in file */ + uchar uniques; /* number of UNIQUE definitions */ + uchar language; /* Language for indexes */ + uchar max_block_size; /* max keyblock size */ uchar fulltext_keys; uchar not_used; /* To align to 8 */ } header; MI_STATUS_INFO state; - ha_rows split; /* number of split blocks */ - my_off_t dellink; /* Link to next removed block */ + ha_rows split; /* number of split blocks */ + my_off_t dellink; /* Link to next removed block */ ulonglong auto_increment; - ulong process; /* process that updated table last */ - ulong unique; /* Unique number for this process */ - ulong update_count; /* Updated for each write lock */ + ulong process; /* process that updated table last */ + ulong unique; /* Unique number for this process */ + ulong update_count; /* Updated for each write lock */ ulong status; ulong *rec_per_key_part; - my_off_t *key_root; /* Start of key trees */ - my_off_t *key_del; /* delete links for trees */ - my_off_t rec_per_key_rows; /* Rows when calculating rec_per_key */ - - ulong sec_index_changed; /* Updated when new sec_index */ - ulong sec_index_used; /* which extra index are in use */ - ulonglong key_map; /* Which keys are in use */ - ulong version; /* timestamp of create */ - time_t create_time; /* Time when created database */ - time_t recover_time; /* Time for last recover */ - time_t check_time; /* Time for last check */ - uint sortkey; /* sorted by this key (not used) */ + my_off_t *key_root; /* Start of key trees */ + my_off_t *key_del; /* delete links for trees */ + my_off_t rec_per_key_rows; /* Rows when calculating rec_per_key */ + + ulong sec_index_changed; /* Updated when new sec_index */ + ulong sec_index_used; /* which extra index are in use */ + ulonglong key_map; /* Which keys are in use */ + ulong version; /* timestamp of create */ + time_t create_time; /* Time when created database */ + time_t recover_time; /* Time for last recover */ + time_t check_time; /* Time for last check */ + uint sortkey; /* sorted by this key (not used) */ uint open_count; - uint8 changed; /* Changed since myisamchk */ + uint8 changed; /* Changed since myisamchk */ /* the following isn't saved on disk */ - uint state_diff_length; /* Should be 0 */ - uint state_length; /* Length of state header in file */ + uint state_diff_length; /* Should be 0 */ + uint state_length; /* Length of state header in file */ ulong *key_info; } MI_STATE_INFO; -#define MI_STATE_INFO_SIZE (24+14*8+7*4+2*2+8) -#define MI_STATE_KEY_SIZE 8 +#define MI_STATE_INFO_SIZE (24+14*8+7*4+2*2+8) +#define MI_STATE_KEY_SIZE 8 #define MI_STATE_KEYBLOCK_SIZE 8 -#define MI_STATE_KEYSEG_SIZE 4 -#define MI_STATE_EXTRA_SIZE ((MI_MAX_KEY+MI_MAX_KEY_BLOCK_SIZE)*MI_STATE_KEY_SIZE + MI_MAX_KEY*MI_MAX_KEY_SEG*MI_STATE_KEYSEG_SIZE) -#define MI_KEYDEF_SIZE (2+ 5*2) -#define MI_UNIQUEDEF_SIZE (2+1+1) -#define HA_KEYSEG_SIZE (6+ 2*2 + 4*2) -#define MI_COLUMNDEF_SIZE (2*3+1) -#define MI_BASE_INFO_SIZE (5*8 + 8*4 + 4 + 4*2 + 16) -#define MI_INDEX_BLOCK_MARGIN 16 /* Safety margin for .MYI tables */ +#define MI_STATE_KEYSEG_SIZE 4 +#define MI_STATE_EXTRA_SIZE ((MI_MAX_KEY+MI_MAX_KEY_BLOCK_SIZE)*MI_STATE_KEY_SIZE + MI_MAX_KEY*HA_MAX_KEY_SEG*MI_STATE_KEYSEG_SIZE) +#define MI_KEYDEF_SIZE (2+ 5*2) +#define MI_UNIQUEDEF_SIZE (2+1+1) +#define HA_KEYSEG_SIZE (6+ 2*2 + 4*2) +#define MI_COLUMNDEF_SIZE (2*3+1) +#define MI_BASE_INFO_SIZE (5*8 + 8*4 + 4 + 4*2 + 16) +#define MI_INDEX_BLOCK_MARGIN 16 /* Safety margin for .MYI tables */ typedef struct st_mi_base_info { - my_off_t keystart; /* Start of keys */ + my_off_t keystart; /* Start of keys */ my_off_t max_data_file_length; my_off_t max_key_file_length; my_off_t margin_key_file_length; - ha_rows records,reloc; /* Create information */ - ulong mean_row_length; /* Create information */ - ulong reclength; /* length of unpacked record */ - ulong pack_reclength; /* Length of full packed rec. */ + ha_rows records, reloc; /* Create information */ + ulong mean_row_length; /* Create information */ + ulong reclength; /* length of unpacked record */ + ulong pack_reclength; /* Length of full packed rec. */ ulong min_pack_length; - ulong max_pack_length; /* Max possibly length of packed rec.*/ + ulong max_pack_length; /* Max possibly length of packed rec.*/ ulong min_block_length; - ulong fields, /* fields in table */ - pack_fields; /* packed fields in table */ - uint rec_reflength; /* = 2-8 */ - uint key_reflength; /* = 2-8 */ - uint keys; /* same as in state.header */ - uint auto_key; /* Which key-1 is a auto key */ - uint blobs; /* Number of blobs */ - uint pack_bits; /* Length of packed bits */ - uint max_key_block_length; /* Max block length */ - uint max_key_length; /* Max key length */ + ulong fields, /* fields in table */ + pack_fields; /* packed fields in table */ + uint rec_reflength; /* = 2-8 */ + uint key_reflength; /* = 2-8 */ + uint keys; /* same as in state.header */ + uint auto_key; /* Which key-1 is a auto key */ + uint blobs; /* Number of blobs */ + uint pack_bits; /* Length of packed bits */ + uint max_key_block_length; /* Max block length */ + uint max_key_length; /* Max key length */ /* Extra allocation when using dynamic record format */ uint extra_alloc_bytes; uint extra_alloc_procent; /* Info about raid */ - uint raid_type,raid_chunks; + uint raid_type, raid_chunks; ulong raid_chunksize; /* The following are from the header */ - uint key_parts,all_key_parts; + uint key_parts, all_key_parts; } MI_BASE_INFO; - /* Structs used intern in database */ + /* Structs used intern in database */ -typedef struct st_mi_blob /* Info of record */ +typedef struct st_mi_blob /* Info of record */ { - ulong offset; /* Offset to blob in record */ - uint pack_length; /* Type of packed length */ - ulong length; /* Calc:ed for each record */ + ulong offset; /* Offset to blob in record */ + uint pack_length; /* Type of packed length */ + ulong length; /* Calc:ed for each record */ } MI_BLOB; -typedef struct st_mi_isam_pack { +typedef struct st_mi_isam_pack +{ ulong header_length; uint ref_length; uchar version; } MI_PACK; -#define MAX_NONMAPPED_INSERTS 1000 +#define MAX_NONMAPPED_INSERTS 1000 -typedef struct st_mi_isam_share { /* Shared between opens */ +typedef struct st_mi_isam_share +{ /* Shared between opens */ MI_STATE_INFO state; MI_BASE_INFO base; - MI_KEYDEF ft2_keyinfo; /* Second-level ft-key definition */ - MI_KEYDEF *keyinfo; /* Key definitions */ - MI_UNIQUEDEF *uniqueinfo; /* unique definitions */ - HA_KEYSEG *keyparts; /* key part info */ - MI_COLUMNDEF *rec; /* Pointer to field information */ - MI_PACK pack; /* Data about packed records */ - MI_BLOB *blobs; /* Pointer to blobs */ - char *unique_file_name; /* realpath() of index file */ - char *data_file_name, /* Resolved path names from symlinks */ - *index_file_name; - byte *file_map; /* mem-map of file if possible */ - KEY_CACHE *key_cache; /* ref to the current key cache */ + MI_KEYDEF ft2_keyinfo; /* Second-level ft-key definition */ + MI_KEYDEF *keyinfo; /* Key definitions */ + MI_UNIQUEDEF *uniqueinfo; /* unique definitions */ + HA_KEYSEG *keyparts; /* key part info */ + MI_COLUMNDEF *rec; /* Pointer to field information */ + MI_PACK pack; /* Data about packed records */ + MI_BLOB *blobs; /* Pointer to blobs */ + char *unique_file_name; /* realpath() of index file */ + char *data_file_name, /* Resolved path names from symlinks */ + *index_file_name; + byte *file_map; /* mem-map of file if possible */ + KEY_CACHE *key_cache; /* ref to the current key cache */ MI_DECODE_TREE *decode_trees; uint16 *decode_tables; - int (*read_record)(struct st_myisam_info*, my_off_t, byte*); - int (*write_record)(struct st_myisam_info*, const byte*); - int (*update_record)(struct st_myisam_info*, my_off_t, const byte*); - int (*delete_record)(struct st_myisam_info*); - int (*read_rnd)(struct st_myisam_info*, byte*, my_off_t, my_bool); - int (*compare_record)(struct st_myisam_info*, const byte *); - ha_checksum (*calc_checksum)(struct st_myisam_info*, const byte *); - int (*compare_unique)(struct st_myisam_info*, MI_UNIQUEDEF *, - const byte *record, my_off_t pos); - uint (*file_read)(MI_INFO *, byte *, uint, my_off_t, myf); - uint (*file_write)(MI_INFO *, byte *, uint, my_off_t, myf); + int(*read_record) (struct st_myisam_info *, my_off_t, byte *); + int(*write_record) (struct st_myisam_info *, const byte *); + int(*update_record) (struct st_myisam_info *, my_off_t, const byte *); + int(*delete_record) (struct st_myisam_info *); + int(*read_rnd) (struct st_myisam_info *, byte *, my_off_t, my_bool); + int(*compare_record) (struct st_myisam_info *, const byte *); + ha_checksum(*calc_checksum) (struct st_myisam_info *, const byte *); + int(*compare_unique) (struct st_myisam_info *, MI_UNIQUEDEF *, + const byte * record, my_off_t pos); + uint(*file_read) (MI_INFO *, byte *, uint, my_off_t, myf); + uint(*file_write) (MI_INFO *, byte *, uint, my_off_t, myf); invalidator_by_filename invalidator; /* query cache invalidator */ - ulong this_process; /* processid */ - ulong last_process; /* For table-change-check */ - ulong last_version; /* Version on start */ - ulong options; /* Options used */ - ulong min_pack_length; /* Theese are used by packed data */ + ulong this_process; /* processid */ + ulong last_process; /* For table-change-check */ + ulong last_version; /* Version on start */ + ulong options; /* Options used */ + ulong min_pack_length; /* Theese are used by packed data */ ulong max_pack_length; ulong state_diff_length; - uint rec_reflength; /* rec_reflength in use now */ - uint unique_name_length; + uint rec_reflength; /* rec_reflength in use now */ + uint unique_name_length; uint32 ftparsers; /* Number of distinct ftparsers + 1 */ - File kfile; /* Shared keyfile */ - File data_file; /* Shared data file */ - int mode; /* mode of file on open */ - uint reopen; /* How many times reopened */ - uint w_locks,r_locks,tot_locks; /* Number of read/write locks */ - uint blocksize; /* blocksize of keyfile */ + File kfile; /* Shared keyfile */ + File data_file; /* Shared data file */ + int mode; /* mode of file on open */ + uint reopen; /* How many times reopened */ + uint w_locks, r_locks, tot_locks; /* Number of read/write locks */ + uint blocksize; /* blocksize of keyfile */ myf write_flag; enum data_file_type data_file_type; - my_bool changed, /* If changed since lock */ - global_changed, /* If changed since open */ - not_flushed, - temporary,delay_key_write, - concurrent_insert; + my_bool changed, /* If changed since lock */ + global_changed, /* If changed since open */ + not_flushed, temporary, delay_key_write, concurrent_insert; #ifdef THREAD THR_LOCK lock; - pthread_mutex_t intern_lock; /* Locking for use with _locking */ + pthread_mutex_t intern_lock; /* Locking for use with _locking */ rw_lock_t *key_root_lock; #endif my_off_t mmaped_length; - uint nonmmaped_inserts; /* counter of writing in non-mmaped - area */ + /* counter of writing in non-mmaped area */ + uint nonmmaped_inserts; rw_lock_t mmap_lock; } MYISAM_SHARE; typedef uint mi_bit_type; -typedef struct st_mi_bit_buff { /* Used for packing of record */ +typedef struct st_mi_bit_buff +{ /* Used for packing of record */ mi_bit_type current_byte; uint bits; - uchar *pos,*end,*blob_pos,*blob_end; + uchar *pos, *end, *blob_pos, *blob_end; uint error; } MI_BIT_BUFF; -struct st_myisam_info { - MYISAM_SHARE *s; /* Shared between open:s */ - MI_STATUS_INFO *state,save_state; - MI_BLOB *blobs; /* Pointer to blobs */ - MI_BIT_BUFF bit_buff; +struct st_myisam_info +{ + MYISAM_SHARE *s; /* Shared between open:s */ + MI_STATUS_INFO *state, save_state; + MI_BLOB *blobs; /* Pointer to blobs */ + MI_BIT_BUFF bit_buff; /* accumulate indexfile changes between write's */ - TREE *bulk_insert; + TREE *bulk_insert; DYNAMIC_ARRAY *ft1_to_ft2; /* used only in ft1->ft2 conversion */ - MYSQL_FTPARSER_PARAM *ftparser_param; /* share info between init/deinit */ - char *filename; /* parameter to open filename */ - uchar *buff, /* Temp area for key */ - *lastkey,*lastkey2; /* Last used search key */ - uchar *first_mbr_key; /* Searhed spatial key */ - byte *rec_buff; /* Tempbuff for recordpack */ - uchar *int_keypos, /* Save position for next/previous */ - *int_maxpos; /* -""- */ - uint int_nod_flag; /* -""- */ - uint32 int_keytree_version; /* -""- */ - int (*read_record)(struct st_myisam_info*, my_off_t, byte*); + MYSQL_FTPARSER_PARAM *ftparser_param; /* share info between init/deinit */ + char *filename; /* parameter to open filename */ + uchar *buff, /* Temp area for key */ + *lastkey, *lastkey2; /* Last used search key */ + uchar *first_mbr_key; /* Searhed spatial key */ + byte *rec_buff; /* Tempbuff for recordpack */ + uchar *int_keypos, /* Save position for next/previous */ + *int_maxpos; /* -""- */ + uint int_nod_flag; /* -""- */ + uint32 int_keytree_version; /* -""- */ + int(*read_record) (struct st_myisam_info *, my_off_t, byte *); invalidator_by_filename invalidator; /* query cache invalidator */ - ulong this_unique; /* uniq filenumber or thread */ - ulong last_unique; /* last unique number */ - ulong this_loop; /* counter for this open */ - ulong last_loop; /* last used counter */ - my_off_t lastpos, /* Last record position */ - nextpos; /* Position to next record */ + ulong this_unique; /* uniq filenumber or thread */ + ulong last_unique; /* last unique number */ + ulong this_loop; /* counter for this open */ + ulong last_loop; /* last used counter */ + my_off_t lastpos, /* Last record position */ + nextpos; /* Position to next record */ my_off_t save_lastpos; - my_off_t pos; /* Intern variable */ - my_off_t last_keypage; /* Last key page read */ - my_off_t last_search_keypage; /* Last keypage when searching */ + my_off_t pos; /* Intern variable */ + my_off_t last_keypage; /* Last key page read */ + my_off_t last_search_keypage; /* Last keypage when searching */ my_off_t dupp_key_pos; ha_checksum checksum; - /* QQ: the folloing two xxx_length fields should be removed, - as they are not compatible with parallel repair */ - ulong packed_length,blob_length; /* Length of found, packed record */ - int dfile; /* The datafile */ - uint opt_flag; /* Optim. for space/speed */ - uint update; /* If file changed since open */ - int lastinx; /* Last used index */ - uint lastkey_length; /* Length of key in lastkey */ - uint last_rkey_length; /* Last length in mi_rkey() */ + /* + QQ: the folloing two xxx_length fields should be removed, + as they are not compatible with parallel repair + */ + ulong packed_length, blob_length; /* Length of found, packed record */ + int dfile; /* The datafile */ + uint opt_flag; /* Optim. for space/speed */ + uint update; /* If file changed since open */ + int lastinx; /* Last used index */ + uint lastkey_length; /* Length of key in lastkey */ + uint last_rkey_length; /* Last length in mi_rkey() */ enum ha_rkey_function last_key_func; /* CONTAIN, OVERLAP, etc */ - uint save_lastkey_length; - uint pack_key_length; /* For MYISAMMRG */ - int errkey; /* Got last error on this key */ - int lock_type; /* How database was locked */ - int tmp_lock_type; /* When locked by readinfo */ - uint data_changed; /* Somebody has changed data */ - uint save_update; /* When using KEY_READ */ - int save_lastinx; - LIST open_list; - IO_CACHE rec_cache; /* When cacheing records */ - uint preload_buff_size; /* When preloading indexes */ - myf lock_wait; /* is 0 or MY_DONT_WAIT */ - my_bool was_locked; /* Was locked in panic */ - my_bool append_insert_at_end; /* Set if concurrent insert */ + uint save_lastkey_length; + uint pack_key_length; /* For MYISAMMRG */ + int errkey; /* Got last error on this key */ + int lock_type; /* How database was locked */ + int tmp_lock_type; /* When locked by readinfo */ + uint data_changed; /* Somebody has changed data */ + uint save_update; /* When using KEY_READ */ + int save_lastinx; + LIST open_list; + IO_CACHE rec_cache; /* When cacheing records */ + uint preload_buff_size; /* When preloading indexes */ + myf lock_wait; /* is 0 or MY_DONT_WAIT */ + my_bool was_locked; /* Was locked in panic */ + my_bool append_insert_at_end; /* Set if concurrent insert */ my_bool quick_mode; - my_bool page_changed; /* If info->buff can't be used for rnext */ - my_bool buff_used; /* If info->buff has to be reread for rnext */ - my_bool once_flags; /* For MYISAMMRG */ + /* If info->buff can't be used for rnext */ + my_bool page_changed; + /* If info->buff has to be reread for rnext */ + my_bool buff_used; + my_bool once_flags; /* For MYISAMMRG */ #ifdef THREAD THR_LOCK_DATA lock; #endif - uchar *rtree_recursion_state; /* For RTREE */ - int rtree_recursion_depth; + uchar *rtree_recursion_state; /* For RTREE */ + int rtree_recursion_depth; }; -typedef struct st_buffpek { - my_off_t file_pos; /* Where we are in the sort file */ - uchar *base,*key; /* Key pointers */ - ha_rows count; /* Number of rows in table */ - ulong mem_count; /* numbers of keys in memory */ - ulong max_keys; /* Max keys in buffert */ -} BUFFPEK; +#define USE_WHOLE_KEY HA_MAX_KEY_BUFF*2 /* Use whole key in _mi_search() */ +#define F_EXTRA_LCK -1 -typedef struct st_mi_sort_param -{ - pthread_t thr; - IO_CACHE read_cache, tempfile, tempfile_for_exceptions; - DYNAMIC_ARRAY buffpek; - - /* - The next two are used to collect statistics, see update_key_parts for - description. - */ - ulonglong unique[MI_MAX_KEY_SEG+1]; - ulonglong notnull[MI_MAX_KEY_SEG+1]; - - my_off_t pos,max_pos,filepos,start_recpos; - uint key, key_length,real_key_length,sortbuff_size; - uint maxbuffers, keys, find_length, sort_keys_length; - my_bool fix_datafile, master; - MI_KEYDEF *keyinfo; - HA_KEYSEG *seg; - SORT_INFO *sort_info; - uchar **sort_keys; - byte *rec_buff; - void *wordlist, *wordptr; - char *record; - MY_TMPDIR *tmpdir; - int (*key_cmp)(struct st_mi_sort_param *, const void *, const void *); - int (*key_read)(struct st_mi_sort_param *,void *); - int (*key_write)(struct st_mi_sort_param *, const void *); - void (*lock_in_memory)(MI_CHECK *); - NEAR int (*write_keys)(struct st_mi_sort_param *, register uchar **, - uint , struct st_buffpek *, IO_CACHE *); - NEAR uint (*read_to_buffer)(IO_CACHE *,struct st_buffpek *, uint); - NEAR int (*write_key)(struct st_mi_sort_param *, IO_CACHE *,char *, - uint, uint); -} MI_SORT_PARAM; - /* Some defines used by isam-funktions */ - -#define USE_WHOLE_KEY MI_MAX_KEY_BUFF*2 /* Use whole key in _mi_search() */ -#define F_EXTRA_LCK -1 - - /* bits in opt_flag */ -#define MEMMAP_USED 32 +/* bits in opt_flag */ +#define MEMMAP_USED 32 #define REMEMBER_OLD_POS 64 -#define WRITEINFO_UPDATE_KEYFILE 1 -#define WRITEINFO_NO_UNLOCK 2 +#define WRITEINFO_UPDATE_KEYFILE 1 +#define WRITEINFO_NO_UNLOCK 2 - /* once_flags */ +/* once_flags */ #define USE_PACKED_KEYS 1 #define RRND_PRESERVE_LASTINX 2 - /* bits in state.changed */ - -#define STATE_CHANGED 1 -#define STATE_CRASHED 2 +/* bits in state.changed */ +#define STATE_CHANGED 1 +#define STATE_CRASHED 2 #define STATE_CRASHED_ON_REPAIR 4 -#define STATE_NOT_ANALYZED 8 +#define STATE_NOT_ANALYZED 8 #define STATE_NOT_OPTIMIZED_KEYS 16 -#define STATE_NOT_SORTED_PAGES 32 +#define STATE_NOT_SORTED_PAGES 32 - /* options to mi_read_cache */ +/* options to mi_read_cache */ +#define READING_NEXT 1 +#define READING_HEADER 2 -#define READING_NEXT 1 -#define READING_HEADER 2 - -#define mi_getint(x) ((uint) mi_uint2korr(x) & 32767) +#define mi_getint(x) ((uint) mi_uint2korr(x) & 32767) #define mi_putint(x,y,nod) { uint16 boh=(nod ? (uint16) 32768 : 0) + (uint16) (y);\ - mi_int2store(x,boh); } + mi_int2store(x,boh); } #define mi_test_if_nod(x) (x[0] & 128 ? info->s->base.key_reflength : 0) #define mi_mark_crashed(x) (x)->s->state.changed|=STATE_CRASHED #define mi_mark_crashed_on_repair(x) { (x)->s->state.changed|=STATE_CRASHED|STATE_CRASHED_ON_REPAIR ; (x)->update|= HA_STATE_CHANGED; } @@ -380,13 +341,6 @@ typedef struct st_mi_sort_param /* Functions to store length of space packed keys, VARCHAR or BLOB keys */ -#define store_key_length_inc(key,length) \ -{ if ((length) < 255) \ - { *(key)++=(length); } \ - else \ - { *(key)=255; mi_int2store((key)+1,(length)); (key)+=3; } \ -} - #define store_key_length(key,length) \ { if ((length) < 255) \ { *(key)=(length); } \ @@ -410,39 +364,39 @@ typedef struct st_mi_sort_param #define get_pack_length(length) ((length) >= 255 ? 3 : 1) -#define MI_MIN_BLOCK_LENGTH 20 /* Because of delete-link */ -#define MI_EXTEND_BLOCK_LENGTH 20 /* Don't use to small record-blocks */ -#define MI_SPLIT_LENGTH ((MI_EXTEND_BLOCK_LENGTH+4)*2) -#define MI_MAX_DYN_BLOCK_HEADER 20 /* Max prefix of record-block */ +#define MI_MIN_BLOCK_LENGTH 20 /* Because of delete-link */ +#define MI_EXTEND_BLOCK_LENGTH 20 /* Don't use to small record-blocks */ +#define MI_SPLIT_LENGTH ((MI_EXTEND_BLOCK_LENGTH+4)*2) +#define MI_MAX_DYN_BLOCK_HEADER 20 /* Max prefix of record-block */ #define MI_BLOCK_INFO_HEADER_LENGTH 20 -#define MI_DYN_DELETE_BLOCK_HEADER 20 /* length of delete-block-header */ -#define MI_DYN_MAX_BLOCK_LENGTH ((1L << 24)-4L) -#define MI_DYN_MAX_ROW_LENGTH (MI_DYN_MAX_BLOCK_LENGTH - MI_SPLIT_LENGTH) -#define MI_DYN_ALIGN_SIZE 4 /* Align blocks on this */ -#define MI_MAX_DYN_HEADER_BYTE 13 /* max header byte for dynamic rows */ -#define MI_MAX_BLOCK_LENGTH ((((ulong) 1 << 24)-1) & (~ (ulong) (MI_DYN_ALIGN_SIZE-1))) +#define MI_DYN_DELETE_BLOCK_HEADER 20 /* length of delete-block-header */ +#define MI_DYN_MAX_BLOCK_LENGTH ((1L << 24)-4L) +#define MI_DYN_MAX_ROW_LENGTH (MI_DYN_MAX_BLOCK_LENGTH - MI_SPLIT_LENGTH) +#define MI_DYN_ALIGN_SIZE 4 /* Align blocks on this */ +#define MI_MAX_DYN_HEADER_BYTE 13 /* max header byte for dynamic rows */ +#define MI_MAX_BLOCK_LENGTH ((((ulong) 1 << 24)-1) & (~ (ulong) (MI_DYN_ALIGN_SIZE-1))) #define MI_REC_BUFF_OFFSET ALIGN_SIZE(MI_DYN_DELETE_BLOCK_HEADER+sizeof(uint32)) -#define MEMMAP_EXTRA_MARGIN 7 /* Write this as a suffix for file */ +#define MEMMAP_EXTRA_MARGIN 7 /* Write this as a suffix for file */ -#define PACK_TYPE_SELECTED 1 /* Bits in field->pack_type */ -#define PACK_TYPE_SPACE_FIELDS 2 -#define PACK_TYPE_ZERO_FILL 4 -#define MI_FOUND_WRONG_KEY 32738 /* Impossible value from ha_key_cmp */ +#define PACK_TYPE_SELECTED 1 /* Bits in field->pack_type */ +#define PACK_TYPE_SPACE_FIELDS 2 +#define PACK_TYPE_ZERO_FILL 4 +#define MI_FOUND_WRONG_KEY 32738 /* Impossible value from ha_key_cmp */ -#define MI_MAX_KEY_BLOCK_SIZE (MI_MAX_KEY_BLOCK_LENGTH/MI_MIN_KEY_BLOCK_LENGTH) +#define MI_MAX_KEY_BLOCK_SIZE (MI_MAX_KEY_BLOCK_LENGTH/MI_MIN_KEY_BLOCK_LENGTH) #define MI_BLOCK_SIZE(key_length,data_pointer,key_pointer) (((((key_length)+(data_pointer)+(key_pointer))*4+(key_pointer)+2)/myisam_block_size+1)*myisam_block_size) -#define MI_MAX_KEYPTR_SIZE 5 /* For calculating block lengths */ -#define MI_MIN_KEYBLOCK_LENGTH 50 /* When to split delete blocks */ +#define MI_MAX_KEYPTR_SIZE 5 /* For calculating block lengths */ +#define MI_MIN_KEYBLOCK_LENGTH 50 /* When to split delete blocks */ -#define MI_MIN_SIZE_BULK_INSERT_TREE 16384 /* this is per key */ +#define MI_MIN_SIZE_BULK_INSERT_TREE 16384 /* this is per key */ #define MI_MIN_ROWS_TO_USE_BULK_INSERT 100 #define MI_MIN_ROWS_TO_DISABLE_INDEXES 100 #define MI_MIN_ROWS_TO_USE_WRITE_CACHE 10 /* The UNIQUE check is done with a hashed long key */ -#define MI_UNIQUE_HASH_TYPE HA_KEYTYPE_ULONG_INT +#define MI_UNIQUE_HASH_TYPE HA_KEYTYPE_ULONG_INT #define mi_unique_store(A,B) mi_int4store((A),(B)) #ifdef THREAD @@ -454,174 +408,181 @@ extern pthread_mutex_t THR_LOCK_myisam; #define rw_unlock(A) {} #endif - /* Some extern variables */ +/* Some extern variables */ extern LIST *myisam_open_list; -extern uchar NEAR myisam_file_magic[],NEAR myisam_pack_file_magic[]; -extern uint NEAR myisam_read_vec[],NEAR myisam_readnext_vec[]; +extern uchar NEAR myisam_file_magic[], NEAR myisam_pack_file_magic[]; +extern uint NEAR myisam_read_vec[], NEAR myisam_readnext_vec[]; extern uint myisam_quick_table_bits; extern File myisam_log_file; extern ulong myisam_pid; - /* This is used by _mi_calc_xxx_key_length och _mi_store_key */ +/* This is used by _mi_calc_xxx_key_length och _mi_store_key */ typedef struct st_mi_s_param { - uint ref_length,key_length, - n_ref_length, - n_length, - totlength, - part_of_prev_key,prev_length,pack_marker; - uchar *key, *prev_key,*next_key_pos; - bool store_not_null; + uint ref_length, key_length, + n_ref_length, + n_length, totlength, part_of_prev_key, prev_length, pack_marker; + uchar *key, *prev_key, *next_key_pos; + bool store_not_null; } MI_KEY_PARAM; - /* Prototypes for intern functions */ +/* Prototypes for intern functions */ -extern int _mi_read_dynamic_record(MI_INFO *info,my_off_t filepos,byte *buf); -extern int _mi_write_dynamic_record(MI_INFO*, const byte*); -extern int _mi_update_dynamic_record(MI_INFO*, my_off_t, const byte*); +extern int _mi_read_dynamic_record(MI_INFO *info, my_off_t filepos, byte *buf); +extern int _mi_write_dynamic_record(MI_INFO *, const byte *); +extern int _mi_update_dynamic_record(MI_INFO *, my_off_t, const byte *); extern int _mi_delete_dynamic_record(MI_INFO *info); -extern int _mi_cmp_dynamic_record(MI_INFO *info,const byte *record); -extern int _mi_read_rnd_dynamic_record(MI_INFO *, byte *,my_off_t, my_bool); -extern int _mi_write_blob_record(MI_INFO*, const byte*); -extern int _mi_update_blob_record(MI_INFO*, my_off_t, const byte*); -extern int _mi_read_static_record(MI_INFO *info, my_off_t filepos,byte *buf); -extern int _mi_write_static_record(MI_INFO*, const byte*); -extern int _mi_update_static_record(MI_INFO*, my_off_t, const byte*); +extern int _mi_cmp_dynamic_record(MI_INFO *info, const byte *record); +extern int _mi_read_rnd_dynamic_record(MI_INFO *, byte *, my_off_t, my_bool); +extern int _mi_write_blob_record(MI_INFO *, const byte *); +extern int _mi_update_blob_record(MI_INFO *, my_off_t, const byte *); +extern int _mi_read_static_record(MI_INFO *info, my_off_t filepos, byte *buf); +extern int _mi_write_static_record(MI_INFO *, const byte *); +extern int _mi_update_static_record(MI_INFO *, my_off_t, const byte *); extern int _mi_delete_static_record(MI_INFO *info); -extern int _mi_cmp_static_record(MI_INFO *info,const byte *record); -extern int _mi_read_rnd_static_record(MI_INFO*, byte *,my_off_t, my_bool); -extern int _mi_ck_write(MI_INFO *info,uint keynr,uchar *key,uint length); +extern int _mi_cmp_static_record(MI_INFO *info, const byte *record); +extern int _mi_read_rnd_static_record(MI_INFO *, byte *, my_off_t, my_bool); +extern int _mi_ck_write(MI_INFO *info, uint keynr, uchar *key, uint length); extern int _mi_ck_real_write_btree(MI_INFO *info, MI_KEYDEF *keyinfo, uchar *key, uint key_length, my_off_t *root, uint comp_flag); -extern int _mi_enlarge_root(MI_INFO *info,MI_KEYDEF *keyinfo,uchar *key, my_off_t *root); -extern int _mi_insert(MI_INFO *info,MI_KEYDEF *keyinfo,uchar *key, - uchar *anc_buff,uchar *key_pos,uchar *key_buff, - uchar *father_buff, uchar *father_keypos, - my_off_t father_page, my_bool insert_last); -extern int _mi_split_page(MI_INFO *info,MI_KEYDEF *keyinfo,uchar *key, - uchar *buff,uchar *key_buff, my_bool insert_last); -extern uchar *_mi_find_half_pos(uint nod_flag,MI_KEYDEF *keyinfo,uchar *page, - uchar *key,uint *return_key_length, - uchar **after_key); -extern int _mi_calc_static_key_length(MI_KEYDEF *keyinfo,uint nod_flag, - uchar *key_pos, uchar *org_key, - uchar *key_buff, - uchar *key, MI_KEY_PARAM *s_temp); -extern int _mi_calc_var_key_length(MI_KEYDEF *keyinfo,uint nod_flag, - uchar *key_pos, uchar *org_key, - uchar *key_buff, - uchar *key, MI_KEY_PARAM *s_temp); -extern int _mi_calc_var_pack_key_length(MI_KEYDEF *keyinfo,uint nod_flag, - uchar *key_pos, uchar *org_key, - uchar *prev_key, - uchar *key, MI_KEY_PARAM *s_temp); -extern int _mi_calc_bin_pack_key_length(MI_KEYDEF *keyinfo,uint nod_flag, - uchar *key_pos,uchar *org_key, - uchar *prev_key, - uchar *key, MI_KEY_PARAM *s_temp); -void _mi_store_static_key(MI_KEYDEF *keyinfo, uchar *key_pos, - MI_KEY_PARAM *s_temp); -void _mi_store_var_pack_key(MI_KEYDEF *keyinfo, uchar *key_pos, - MI_KEY_PARAM *s_temp); +extern int _mi_enlarge_root(MI_INFO *info, MI_KEYDEF *keyinfo, uchar *key, + my_off_t *root); +extern int _mi_insert(MI_INFO *info, MI_KEYDEF *keyinfo, uchar *key, + uchar *anc_buff, uchar *key_pos, uchar *key_buff, + uchar *father_buff, uchar *father_keypos, + my_off_t father_page, my_bool insert_last); +extern int _mi_split_page(MI_INFO *info, MI_KEYDEF *keyinfo, uchar *key, + uchar *buff, uchar *key_buff, my_bool insert_last); +extern uchar *_mi_find_half_pos(uint nod_flag, MI_KEYDEF *keyinfo, + uchar *page, uchar *key, + uint *return_key_length, uchar ** after_key); +extern int _mi_calc_static_key_length(MI_KEYDEF *keyinfo, uint nod_flag, + uchar *key_pos, uchar *org_key, + uchar *key_buff, uchar *key, + MI_KEY_PARAM *s_temp); +extern int _mi_calc_var_key_length(MI_KEYDEF *keyinfo, uint nod_flag, + uchar *key_pos, uchar *org_key, + uchar *key_buff, uchar *key, + MI_KEY_PARAM *s_temp); +extern int _mi_calc_var_pack_key_length(MI_KEYDEF *keyinfo, uint nod_flag, + uchar *key_pos, uchar *org_key, + uchar *prev_key, uchar *key, + MI_KEY_PARAM *s_temp); +extern int _mi_calc_bin_pack_key_length(MI_KEYDEF *keyinfo, uint nod_flag, + uchar *key_pos, uchar *org_key, + uchar *prev_key, uchar *key, + MI_KEY_PARAM *s_temp); +void _mi_store_static_key(MI_KEYDEF *keyinfo, uchar *key_pos, + MI_KEY_PARAM *s_temp); +void _mi_store_var_pack_key(MI_KEYDEF *keyinfo, uchar *key_pos, + MI_KEY_PARAM *s_temp); #ifdef NOT_USED -void _mi_store_pack_key(MI_KEYDEF *keyinfo, uchar *key_pos, - MI_KEY_PARAM *s_temp); +void _mi_store_pack_key(MI_KEYDEF *keyinfo, uchar *key_pos, + MI_KEY_PARAM *s_temp); #endif -void _mi_store_bin_pack_key(MI_KEYDEF *keyinfo, uchar *key_pos, - MI_KEY_PARAM *s_temp); +void _mi_store_bin_pack_key(MI_KEYDEF *keyinfo, uchar *key_pos, + MI_KEY_PARAM *s_temp); -extern int _mi_ck_delete(MI_INFO *info,uint keynr,uchar *key,uint key_length); -extern int _mi_readinfo(MI_INFO *info,int lock_flag,int check_keybuffer); -extern int _mi_writeinfo(MI_INFO *info,uint options); +extern int _mi_ck_delete(MI_INFO *info, uint keynr, uchar *key, + uint key_length); +extern int _mi_readinfo(MI_INFO *info, int lock_flag, int check_keybuffer); +extern int _mi_writeinfo(MI_INFO *info, uint options); extern int _mi_test_if_changed(MI_INFO *info); extern int _mi_mark_file_changed(MI_INFO *info); extern int _mi_decrement_open_count(MI_INFO *info); -extern int _mi_check_index(MI_INFO *info,int inx); -extern int _mi_search(MI_INFO *info,MI_KEYDEF *keyinfo,uchar *key,uint key_len, - uint nextflag,my_off_t pos); -extern int _mi_bin_search(struct st_myisam_info *info,MI_KEYDEF *keyinfo, - uchar *page,uchar *key,uint key_len,uint comp_flag, - uchar * *ret_pos,uchar *buff, my_bool *was_last_key); -extern int _mi_seq_search(MI_INFO *info,MI_KEYDEF *keyinfo,uchar *page, - uchar *key,uint key_len,uint comp_flag, - uchar **ret_pos,uchar *buff, my_bool *was_last_key); -extern int _mi_prefix_search(MI_INFO *info,MI_KEYDEF *keyinfo,uchar *page, - uchar *key,uint key_len,uint comp_flag, - uchar **ret_pos,uchar *buff, my_bool *was_last_key); -extern my_off_t _mi_kpos(uint nod_flag,uchar *after_key); -extern void _mi_kpointer(MI_INFO *info,uchar *buff,my_off_t pos); -extern my_off_t _mi_dpos(MI_INFO *info, uint nod_flag,uchar *after_key); +extern int _mi_check_index(MI_INFO *info, int inx); +extern int _mi_search(MI_INFO *info, MI_KEYDEF *keyinfo, uchar *key, + uint key_len, uint nextflag, my_off_t pos); +extern int _mi_bin_search(struct st_myisam_info *info, MI_KEYDEF *keyinfo, + uchar *page, uchar *key, uint key_len, + uint comp_flag, uchar **ret_pos, uchar *buff, + my_bool *was_last_key); +extern int _mi_seq_search(MI_INFO *info, MI_KEYDEF *keyinfo, uchar *page, + uchar *key, uint key_len, uint comp_flag, + uchar ** ret_pos, uchar *buff, + my_bool *was_last_key); +extern int _mi_prefix_search(MI_INFO *info, MI_KEYDEF *keyinfo, uchar *page, + uchar *key, uint key_len, uint comp_flag, + uchar ** ret_pos, uchar *buff, + my_bool *was_last_key); +extern my_off_t _mi_kpos(uint nod_flag, uchar *after_key); +extern void _mi_kpointer(MI_INFO *info, uchar *buff, my_off_t pos); +extern my_off_t _mi_dpos(MI_INFO *info, uint nod_flag, uchar *after_key); extern my_off_t _mi_rec_pos(MYISAM_SHARE *info, uchar *ptr); -extern void _mi_dpointer(MI_INFO *info, uchar *buff,my_off_t pos); -extern int ha_key_cmp(HA_KEYSEG *keyseg, uchar *a,uchar *b, - uint key_length,uint nextflag,uint *diff_length); -extern uint _mi_get_static_key(MI_KEYDEF *keyinfo,uint nod_flag,uchar * *page, - uchar *key); -extern uint _mi_get_pack_key(MI_KEYDEF *keyinfo,uint nod_flag,uchar * *page, - uchar *key); +extern void _mi_dpointer(MI_INFO *info, uchar *buff, my_off_t pos); +extern int ha_key_cmp(HA_KEYSEG *keyseg, uchar *a, uchar *b, + uint key_length, uint nextflag, uint *diff_length); +extern uint _mi_get_static_key(MI_KEYDEF *keyinfo, uint nod_flag, + uchar **page, uchar *key); +extern uint _mi_get_pack_key(MI_KEYDEF *keyinfo, uint nod_flag, uchar **page, + uchar *key); extern uint _mi_get_binary_pack_key(MI_KEYDEF *keyinfo, uint nod_flag, - uchar **page_pos, uchar *key); -extern uchar *_mi_get_last_key(MI_INFO *info,MI_KEYDEF *keyinfo,uchar *keypos, - uchar *lastkey,uchar *endpos, - uint *return_key_length); + uchar ** page_pos, uchar *key); +extern uchar *_mi_get_last_key(MI_INFO *info, MI_KEYDEF *keyinfo, + uchar *keypos, uchar *lastkey, uchar *endpos, + uint *return_key_length); extern uchar *_mi_get_key(MI_INFO *info, MI_KEYDEF *keyinfo, uchar *page, - uchar *key, uchar *keypos, uint *return_key_length); -extern uint _mi_keylength(MI_KEYDEF *keyinfo,uchar *key); + uchar *key, uchar *keypos, + uint *return_key_length); +extern uint _mi_keylength(MI_KEYDEF *keyinfo, uchar *key); extern uint _mi_keylength_part(MI_KEYDEF *keyinfo, register uchar *key, - HA_KEYSEG *end); -extern uchar *_mi_move_key(MI_KEYDEF *keyinfo,uchar *to,uchar *from); -extern int _mi_search_next(MI_INFO *info,MI_KEYDEF *keyinfo,uchar *key, - uint key_length,uint nextflag,my_off_t pos); -extern int _mi_search_first(MI_INFO *info,MI_KEYDEF *keyinfo,my_off_t pos); -extern int _mi_search_last(MI_INFO *info,MI_KEYDEF *keyinfo,my_off_t pos); -extern uchar *_mi_fetch_keypage(MI_INFO *info,MI_KEYDEF *keyinfo,my_off_t page, - int level,uchar *buff,int return_buffer); -extern int _mi_write_keypage(MI_INFO *info,MI_KEYDEF *keyinfo,my_off_t page, - int level, uchar *buff); -extern int _mi_dispose(MI_INFO *info,MI_KEYDEF *keyinfo,my_off_t pos, - int level); -extern my_off_t _mi_new(MI_INFO *info,MI_KEYDEF *keyinfo,int level); -extern uint _mi_make_key(MI_INFO *info,uint keynr,uchar *key, - const byte *record,my_off_t filepos); -extern uint _mi_pack_key(MI_INFO *info,uint keynr,uchar *key,uchar *old, - uint key_length, HA_KEYSEG **last_used_keyseg); -extern int _mi_read_key_record(MI_INFO *info,my_off_t filepos,byte *buf); -extern int _mi_read_cache(IO_CACHE *info,byte *buff,my_off_t pos, - uint length,int re_read_if_possibly); -extern void update_auto_increment(MI_INFO *info,const byte *record); - -extern byte *mi_alloc_rec_buff(MI_INFO *,ulong, byte**); + HA_KEYSEG *end); +extern uchar *_mi_move_key(MI_KEYDEF *keyinfo, uchar *to, uchar *from); +extern int _mi_search_next(MI_INFO *info, MI_KEYDEF *keyinfo, uchar *key, + uint key_length, uint nextflag, my_off_t pos); +extern int _mi_search_first(MI_INFO *info, MI_KEYDEF *keyinfo, my_off_t pos); +extern int _mi_search_last(MI_INFO *info, MI_KEYDEF *keyinfo, my_off_t pos); +extern uchar *_mi_fetch_keypage(MI_INFO *info, MI_KEYDEF *keyinfo, + my_off_t page, int level, uchar *buff, + int return_buffer); +extern int _mi_write_keypage(MI_INFO *info, MI_KEYDEF *keyinfo, my_off_t page, + int level, uchar *buff); +extern int _mi_dispose(MI_INFO *info, MI_KEYDEF *keyinfo, my_off_t pos, + int level); +extern my_off_t _mi_new(MI_INFO *info, MI_KEYDEF *keyinfo, int level); +extern uint _mi_make_key(MI_INFO *info, uint keynr, uchar *key, + const byte *record, my_off_t filepos); +extern uint _mi_pack_key(MI_INFO *info, uint keynr, uchar *key, uchar *old, + uint key_length, HA_KEYSEG ** last_used_keyseg); +extern int _mi_read_key_record(MI_INFO *info, my_off_t filepos, byte *buf); +extern int _mi_read_cache(IO_CACHE *info, byte *buff, my_off_t pos, + uint length, int re_read_if_possibly); +extern void update_auto_increment(MI_INFO *info, const byte *record); + +extern byte *mi_alloc_rec_buff(MI_INFO *, ulong, byte **); #define mi_get_rec_buff_ptr(info,buf) \ ((((info)->s->options & HA_OPTION_PACK_RECORD) && (buf)) ? \ (buf) - MI_REC_BUFF_OFFSET : (buf)) #define mi_get_rec_buff_len(info,buf) \ (*((uint32 *)(mi_get_rec_buff_ptr(info,buf)))) -extern ulong _mi_rec_unpack(MI_INFO *info,byte *to,byte *from, - ulong reclength); -extern my_bool _mi_rec_check(MI_INFO *info,const char *record, byte *packpos, +extern ulong _mi_rec_unpack(MI_INFO *info, byte *to, byte *from, + ulong reclength); +extern my_bool _mi_rec_check(MI_INFO *info, const char *record, byte *packpos, ulong packed_length, my_bool with_checkum); -extern int _mi_write_part_record(MI_INFO *info,my_off_t filepos,ulong length, - my_off_t next_filepos,byte **record, - ulong *reclength,int *flag); -extern void _mi_print_key(FILE *stream,HA_KEYSEG *keyseg,const uchar *key, - uint length); -extern my_bool _mi_read_pack_info(MI_INFO *info,pbool fix_keys); -extern int _mi_read_pack_record(MI_INFO *info,my_off_t filepos,byte *buf); -extern int _mi_read_rnd_pack_record(MI_INFO*, byte *,my_off_t, my_bool); -extern int _mi_pack_rec_unpack(MI_INFO *info,byte *to,byte *from, - ulong reclength); -extern ulonglong mi_safe_mul(ulonglong a,ulonglong b); +extern int _mi_write_part_record(MI_INFO *info, my_off_t filepos, ulong length, + my_off_t next_filepos, byte ** record, + ulong *reclength, int *flag); +extern void _mi_print_key(FILE *stream, HA_KEYSEG *keyseg, const uchar *key, + uint length); +extern my_bool _mi_read_pack_info(MI_INFO *info, pbool fix_keys); +extern int _mi_read_pack_record(MI_INFO *info, my_off_t filepos, byte *buf); +extern int _mi_read_rnd_pack_record(MI_INFO *, byte *, my_off_t, my_bool); +extern int _mi_pack_rec_unpack(MI_INFO *info, byte *to, byte *from, + ulong reclength); +extern ulonglong mi_safe_mul(ulonglong a, ulonglong b); extern int _mi_ft_update(MI_INFO *info, uint keynr, byte *keybuf, - const byte *oldrec, const byte *newrec, my_off_t pos); + const byte *oldrec, const byte *newrec, + my_off_t pos); struct st_sort_info; -typedef struct st_mi_block_info { /* Parameter to _mi_get_block_info */ +typedef struct st_mi_block_info /* Parameter to _mi_get_block_info */ +{ uchar header[MI_BLOCK_INFO_HEADER_LENGTH]; ulong rec_len; ulong data_len; @@ -634,35 +595,37 @@ typedef struct st_mi_block_info { /* Parameter to _mi_get_block_info */ uint offset; } MI_BLOCK_INFO; - /* bits in return from _mi_get_block_info */ - -#define BLOCK_FIRST 1 -#define BLOCK_LAST 2 -#define BLOCK_DELETED 4 -#define BLOCK_ERROR 8 /* Wrong data */ -#define BLOCK_SYNC_ERROR 16 /* Right data at wrong place */ -#define BLOCK_FATAL_ERROR 32 /* hardware-error */ - -#define NEED_MEM ((uint) 10*4*(IO_SIZE+32)+32) /* Nead for recursion */ -#define MAXERR 20 -#define BUFFERS_WHEN_SORTING 16 /* Alloc for sort-key-tree */ -#define WRITE_COUNT MY_HOW_OFTEN_TO_WRITE -#define INDEX_TMP_EXT ".TMM" -#define DATA_TMP_EXT ".TMD" - -#define UPDATE_TIME 1 -#define UPDATE_STAT 2 -#define UPDATE_SORT 4 -#define UPDATE_AUTO_INC 8 -#define UPDATE_OPEN_COUNT 16 - -#define USE_BUFFER_INIT (((1024L*512L-MALLOC_OVERHEAD)/IO_SIZE)*IO_SIZE) -#define READ_BUFFER_INIT (1024L*256L-MALLOC_OVERHEAD) -#define SORT_BUFFER_INIT (2048L*1024L-MALLOC_OVERHEAD) -#define MIN_SORT_BUFFER (4096-MALLOC_OVERHEAD) - -enum myisam_log_commands { - MI_LOG_OPEN,MI_LOG_WRITE,MI_LOG_UPDATE,MI_LOG_DELETE,MI_LOG_CLOSE,MI_LOG_EXTRA,MI_LOG_LOCK,MI_LOG_DELETE_ALL + /* bits in return from _mi_get_block_info */ + +#define BLOCK_FIRST 1 +#define BLOCK_LAST 2 +#define BLOCK_DELETED 4 +#define BLOCK_ERROR 8 /* Wrong data */ +#define BLOCK_SYNC_ERROR 16 /* Right data at wrong place */ +#define BLOCK_FATAL_ERROR 32 /* hardware-error */ + +#define NEED_MEM ((uint) 10*4*(IO_SIZE+32)+32) /* Nead for recursion */ +#define MAXERR 20 +#define BUFFERS_WHEN_SORTING 16 /* Alloc for sort-key-tree */ +#define WRITE_COUNT MY_HOW_OFTEN_TO_WRITE +#define INDEX_TMP_EXT ".TMM" +#define DATA_TMP_EXT ".TMD" + +#define UPDATE_TIME 1 +#define UPDATE_STAT 2 +#define UPDATE_SORT 4 +#define UPDATE_AUTO_INC 8 +#define UPDATE_OPEN_COUNT 16 + +#define USE_BUFFER_INIT (((1024L*512L-MALLOC_OVERHEAD)/IO_SIZE)*IO_SIZE) +#define READ_BUFFER_INIT (1024L*256L-MALLOC_OVERHEAD) +#define SORT_BUFFER_INIT (2048L*1024L-MALLOC_OVERHEAD) +#define MIN_SORT_BUFFER (4096-MALLOC_OVERHEAD) + +enum myisam_log_commands +{ + MI_LOG_OPEN, MI_LOG_WRITE, MI_LOG_UPDATE, MI_LOG_DELETE, MI_LOG_CLOSE, + MI_LOG_EXTRA, MI_LOG_LOCK, MI_LOG_DELETE_ALL }; #define myisam_log(a,b,c,d) if (myisam_log_file >= 0) _myisam_log(a,b,c,d) @@ -672,34 +635,32 @@ enum myisam_log_commands { #define fast_mi_writeinfo(INFO) if (!(INFO)->s->tot_locks) (void) _mi_writeinfo((INFO),0) #define fast_mi_readinfo(INFO) ((INFO)->lock_type == F_UNLCK) && _mi_readinfo((INFO),F_RDLCK,1) -#ifdef __cplusplus +#ifdef __cplusplus extern "C" { #endif - -extern uint _mi_get_block_info(MI_BLOCK_INFO *,File, my_off_t); -extern uint _mi_rec_pack(MI_INFO *info,byte *to,const byte *from); + extern uint _mi_get_block_info(MI_BLOCK_INFO *, File, my_off_t); +extern uint _mi_rec_pack(MI_INFO *info, byte *to, const byte *from); extern uint _mi_pack_get_block_info(MI_INFO *, MI_BLOCK_INFO *, File, my_off_t); -extern void _my_store_blob_length(byte *pos,uint pack_length,uint length); -extern void _myisam_log(enum myisam_log_commands command,MI_INFO *info, - const byte *buffert,uint length); +extern void _mi_store_blob_length(byte *pos, uint pack_length, uint length); +extern void _myisam_log(enum myisam_log_commands command, MI_INFO *info, + const byte *buffert, uint length); extern void _myisam_log_command(enum myisam_log_commands command, - MI_INFO *info, const byte *buffert, - uint length, int result); -extern void _myisam_log_record(enum myisam_log_commands command,MI_INFO *info, - const byte *record,my_off_t filepos, - int result); + MI_INFO *info, const byte *buffert, + uint length, int result); +extern void _myisam_log_record(enum myisam_log_commands command, MI_INFO *info, + const byte *record, my_off_t filepos, + int result); extern void mi_report_error(int errcode, const char *file_name); extern my_bool _mi_memmap_file(MI_INFO *info); extern void _mi_unmap_file(MI_INFO *info); extern uint save_pack_length(uint version, byte *block_buff, ulong length); -extern uint read_pack_length(uint version, const uchar *buf, ulong *length); extern uint calc_pack_length(uint version, ulong length); extern uint mi_mmap_pread(MI_INFO *info, byte *Buffer, - uint Count, my_off_t offset, myf MyFlags); + uint Count, my_off_t offset, myf MyFlags); extern uint mi_mmap_pwrite(MI_INFO *info, byte *Buffer, - uint Count, my_off_t offset, myf MyFlags); + uint Count, my_off_t offset, myf MyFlags); extern uint mi_nommap_pread(MI_INFO *info, byte *Buffer, - uint Count, my_off_t offset, myf MyFlags); + uint Count, my_off_t offset, myf MyFlags); extern uint mi_nommap_pwrite(MI_INFO *info, byte *Buffer, uint Count, my_off_t offset, myf MyFlags); @@ -707,7 +668,7 @@ uint mi_state_info_write(File file, MI_STATE_INFO *state, uint pWrite); uchar *mi_state_info_read(uchar *ptr, MI_STATE_INFO *state); uint mi_state_info_read_dsk(File file, MI_STATE_INFO *state, my_bool pRead); uint mi_base_info_write(File file, MI_BASE_INFO *base); -uchar *my_n_base_info_read(uchar *ptr, MI_BASE_INFO *base); +uchar *mi_n_base_info_read(uchar *ptr, MI_BASE_INFO *base); int mi_keyseg_write(File file, const HA_KEYSEG *keyseg); char *mi_keyseg_read(char *ptr, HA_KEYSEG *keyseg); uint mi_keydef_write(File file, MI_KEYDEF *keydef); @@ -719,22 +680,22 @@ char *mi_recinfo_read(char *ptr, MI_COLUMNDEF *recinfo); extern int mi_disable_indexes(MI_INFO *info); extern int mi_enable_indexes(MI_INFO *info); extern int mi_indexes_are_disabled(MI_INFO *info); -ulong _my_calc_total_blob_length(MI_INFO *info, const byte *record); +ulong _mi_calc_total_blob_length(MI_INFO *info, const byte *record); ha_checksum mi_checksum(MI_INFO *info, const byte *buf); ha_checksum mi_static_checksum(MI_INFO *info, const byte *buf); my_bool mi_check_unique(MI_INFO *info, MI_UNIQUEDEF *def, byte *record, - ha_checksum unique_hash, my_off_t pos); + ha_checksum unique_hash, my_off_t pos); ha_checksum mi_unique_hash(MI_UNIQUEDEF *def, const byte *buf); int _mi_cmp_static_unique(MI_INFO *info, MI_UNIQUEDEF *def, - const byte *record, my_off_t pos); + const byte *record, my_off_t pos); int _mi_cmp_dynamic_unique(MI_INFO *info, MI_UNIQUEDEF *def, - const byte *record, my_off_t pos); + const byte *record, my_off_t pos); int mi_unique_comp(MI_UNIQUEDEF *def, const byte *a, const byte *b, - my_bool null_are_equal); -void mi_get_status(void* param, int concurrent_insert); -void mi_update_status(void* param); -void mi_copy_status(void* to,void *from); -my_bool mi_check_status(void* param); + my_bool null_are_equal); +void mi_get_status(void *param, int concurrent_insert); +void mi_update_status(void *param); +void mi_copy_status(void *to, void *from); +my_bool mi_check_status(void *param); void mi_disable_non_unique_index(MI_INFO *info, ha_rows rows); extern MI_INFO *test_if_reopen(char *filename); @@ -742,24 +703,18 @@ my_bool check_table_is_closed(const char *name, const char *where); int mi_open_datafile(MI_INFO *info, MYISAM_SHARE *share, File file_to_dup); int mi_open_keyfile(MYISAM_SHARE *share); void mi_setup_functions(register MYISAM_SHARE *share); +my_bool mi_dynmap_file(MI_INFO *info, my_off_t size); +void mi_remap_file(MI_INFO *info, my_off_t size); /* Functions needed by mi_check */ -volatile int *killed_ptr(MI_CHECK *param); -void mi_check_print_error _VARARGS((MI_CHECK *param, const char *fmt,...)); -void mi_check_print_warning _VARARGS((MI_CHECK *param, const char *fmt,...)); -void mi_check_print_info _VARARGS((MI_CHECK *param, const char *fmt,...)); -int flush_pending_blocks(MI_SORT_PARAM *param); -int sort_ft_buf_flush(MI_SORT_PARAM *sort_param); -int thr_write_keys(MI_SORT_PARAM *sort_param); +volatile int *killed_ptr(HA_CHECK *param); +void mi_check_print_error _VARARGS((HA_CHECK *param, const char *fmt, ...)); +void mi_check_print_warning _VARARGS((HA_CHECK *param, const char *fmt, ...)); +void mi_check_print_info _VARARGS((HA_CHECK *param, const char *fmt, ...)); #ifdef THREAD pthread_handler_t thr_find_all_keys(void *arg); #endif -int flush_blocks(MI_CHECK *param, KEY_CACHE *key_cache, File file); - -int sort_write_record(MI_SORT_PARAM *sort_param); -int _create_index_by_sort(MI_SORT_PARAM *info,my_bool no_messages, ulong); - +int flush_blocks(HA_CHECK *param, KEY_CACHE *key_cache, File file); #ifdef __cplusplus } #endif - diff --git a/storage/myisam/myisamlog.c b/storage/myisam/myisamlog.c index de55b86252c..2b36e6438c1 100644 --- a/storage/myisam/myisamlog.c +++ b/storage/myisam/myisamlog.c @@ -806,7 +806,7 @@ static int find_record_with_key(struct file_info *file_info, byte *record) { uint key; MI_INFO *info=file_info->isam; - uchar tmp_key[MI_MAX_KEY_BUFF]; + uchar tmp_key[HA_MAX_KEY_BUFF]; for (key=0 ; key < info->s->base.keys ; key++) { diff --git a/storage/myisam/myisampack.c b/storage/myisam/myisampack.c index e80a3ffacd9..18313603394 100644 --- a/storage/myisam/myisampack.c +++ b/storage/myisam/myisampack.c @@ -2726,6 +2726,7 @@ static int compress_isam_file(PACK_MRG_INFO *mrg, HUFF_COUNTS *huff_counts) break; } case FIELD_LAST: + case FIELD_enum_val_count: abort(); /* Impossible */ } start_pos+=count->max_zero_fill; diff --git a/storage/myisam/rt_index.c b/storage/myisam/rt_index.c index 97554dca4e6..9803b4e110b 100644 --- a/storage/myisam/rt_index.c +++ b/storage/myisam/rt_index.c @@ -540,7 +540,7 @@ static int rtree_insert_req(MI_INFO *info, MI_KEYDEF *keyinfo, uchar *key, int res; if (!(page_buf = (uchar*)my_alloca((uint)keyinfo->block_length + - MI_MAX_KEY_BUFF))) + HA_MAX_KEY_BUFF))) { my_errno = HA_ERR_OUT_OF_MEM; return -1; @@ -654,7 +654,7 @@ static int rtree_insert_level(MI_INFO *info, uint keynr, uchar *key, uint nod_flag = info->s->base.key_reflength; if (!(new_root_buf = (uchar*)my_alloca((uint)keyinfo->block_length + - MI_MAX_KEY_BUFF))) + HA_MAX_KEY_BUFF))) { my_errno = HA_ERR_OUT_OF_MEM; return -1; diff --git a/storage/myisam/sort.c b/storage/myisam/sort.c index c9562461f56..443d2b11afe 100644 --- a/storage/myisam/sort.c +++ b/storage/myisam/sort.c @@ -16,7 +16,7 @@ /* Creates a index for a database by reading keys, sorting them and outputing - them in sorted order through SORT_INFO functions. + them in sorted order through MI_SORT_INFO functions. */ #include "fulltext.h" @@ -459,8 +459,8 @@ ok: int thr_write_keys(MI_SORT_PARAM *sort_param) { - SORT_INFO *sort_info=sort_param->sort_info; - MI_CHECK *param=sort_info->param; + MI_SORT_INFO *sort_info=sort_param->sort_info; + HA_CHECK *param=sort_info->param; ulong length, keys; ulong *rec_per_key_part=param->rec_per_key_part; int got_error=sort_info->got_error; @@ -870,7 +870,6 @@ merge_buffers(MI_SORT_PARAM *info, uint keys, IO_CACHE *from_file, BUFFPEK *buffpek,**refpek; QUEUE queue; volatile int *killed= killed_ptr(info->sort_info->param); - DBUG_ENTER("merge_buffers"); count=error=0; -- cgit v1.2.1 From a63446fe798735505020e96eecf5b8f1c478292f Mon Sep 17 00:00:00 2001 From: unknown Date: Tue, 11 Apr 2006 20:00:57 +0300 Subject: Add ma_test_all.res storage/maria/ma_test_all.res: New BitKeeper file ``storage/maria/ma_test_all.res'' --- storage/maria/ma_test_all.res | 53 +++++++++++++++++++++++++++++++++++++++++++ 1 file changed, 53 insertions(+) create mode 100644 storage/maria/ma_test_all.res diff --git a/storage/maria/ma_test_all.res b/storage/maria/ma_test_all.res new file mode 100644 index 00000000000..7ffd3378b51 --- /dev/null +++ b/storage/maria/ma_test_all.res @@ -0,0 +1,53 @@ +maria_chk: MARIA file test1 +maria_chk: warning: Size of indexfile is: 8192 Should be: 16384 +MARIA-table 'test1' is usable but should be fixed +ma_test2 -s -L -K -R1 -m2000 ; Should give error 135 +Error: 135 in write at record: 1105 +got error: 135 when using MARIA-database +maria_chk: MARIA file test2 +maria_chk: warning: Datafile is almost full, 65532 of 65534 used +MARIA-table 'test2' is usable but should be fixed +Commands Used count Errors Recover errors +open 1 0 0 +write 50 0 0 +update 5 0 0 +delete 50 0 0 +close 1 0 0 +extra 6 0 0 +Total 113 0 0 +Commands Used count Errors Recover errors +open 2 0 0 +write 100 0 0 +update 10 0 0 +delete 100 0 0 +close 2 0 0 +extra 12 0 0 +Total 226 0 0 + +real 0m0.994s +user 0m0.432s +sys 0m0.184s + +real 0m2.153s +user 0m1.196s +sys 0m0.228s + +real 0m1.483s +user 0m0.772s +sys 0m0.180s + +real 0m1.992s +user 0m1.180s +sys 0m0.188s + +real 0m2.028s +user 0m1.184s +sys 0m0.152s + +real 0m1.878s +user 0m1.028s +sys 0m0.136s + +real 0m1.980s +user 0m1.116s +sys 0m0.192s -- cgit v1.2.1 From 379b95780824f3e7d915f6d8dd7a976e924e5723 Mon Sep 17 00:00:00 2001 From: unknown Date: Wed, 12 Apr 2006 15:36:53 +0200 Subject: easy fixes to ma_test_all (to make shell happy) storage/maria/ma_test_all.sh: ; is needed before "else". Some ./ for executables to be found. --- storage/maria/ma_test_all.sh | 256 +++++++++++++++++++++---------------------- 1 file changed, 128 insertions(+), 128 deletions(-) diff --git a/storage/maria/ma_test_all.sh b/storage/maria/ma_test_all.sh index d848bc63b9a..f857127dca9 100755 --- a/storage/maria/ma_test_all.sh +++ b/storage/maria/ma_test_all.sh @@ -6,142 +6,142 @@ valgrind="valgrind --alignment=8 --leak-check=yes" silent="-s" -if test -f ma_test1$MACH ; then suffix=$MACH else suffix=""; fi -ma_test1$suffix $silent -maria_chk$suffix -se test1 -ma_test1$suffix $silent -N -S -maria_chk$suffix -se test1 -ma_test1$suffix $silent -P --checksum -maria_chk$suffix -se test1 -ma_test1$suffix $silent -P -N -S -maria_chk$suffix -se test1 -ma_test1$suffix $silent -B -N -R2 -maria_chk$suffix -sm test1 -ma_test1$suffix $silent -a -k 480 --unique -maria_chk$suffix -sm test1 -ma_test1$suffix $silent -a -N -S -R1 -maria_chk$suffix -sm test1 -ma_test1$suffix $silent -p -S -maria_chk$suffix -sm test1 -ma_test1$suffix $silent -p -S -N --unique -maria_chk$suffix -sm test1 -ma_test1$suffix $silent -p -S -N --key_length=127 --checksum -maria_chk$suffix -sm test1 -ma_test1$suffix $silent -p -S -N --key_length=128 -maria_chk$suffix -sm test1 -ma_test1$suffix $silent -p -S --key_length=480 -maria_chk$suffix -sm test1 -ma_test1$suffix $silent -a -B -maria_chk$suffix -sm test1 -ma_test1$suffix $silent -a -B --key_length=64 --unique -maria_chk$suffix -sm test1 -ma_test1$suffix $silent -a -B -k 480 --checksum -maria_chk$suffix -sm test1 -ma_test1$suffix $silent -a -B -k 480 -N --unique --checksum -maria_chk$suffix -sm test1 -ma_test1$suffix $silent -a -m -maria_chk$suffix -sm test1 -ma_test1$suffix $silent -a -m -P --unique --checksum -maria_chk$suffix -sm test1 -ma_test1$suffix $silent -a -m -P --key_length=480 --key_cache -maria_chk$suffix -sm test1 -ma_test1$suffix $silent -m -p -maria_chk$suffix -sm test1 -ma_test1$suffix $silent -w -S --unique -maria_chk$suffix -sm test1 -ma_test1$suffix $silent -a -w --key_length=64 --checksum -maria_chk$suffix -sm test1 -ma_test1$suffix $silent -a -w -N --key_length=480 -maria_chk$suffix -sm test1 -ma_test1$suffix $silent -a -w -S --key_length=480 --checksum -maria_chk$suffix -sm test1 -ma_test1$suffix $silent -a -b -N -maria_chk$suffix -sm test1 -ma_test1$suffix $silent -a -b --key_length=480 -maria_chk$suffix -sm test1 -ma_test1$suffix $silent -p -B --key_length=480 -maria_chk$suffix -sm test1 +if test -f ma_test1$MACH ; then suffix=$MACH ; else suffix=""; fi +./ma_test1$suffix $silent +./maria_chk$suffix -se test1 +./ma_test1$suffix $silent -N -S +./maria_chk$suffix -se test1 +./ma_test1$suffix $silent -P --checksum +./maria_chk$suffix -se test1 +./ma_test1$suffix $silent -P -N -S +./maria_chk$suffix -se test1 +./ma_test1$suffix $silent -B -N -R2 +./maria_chk$suffix -sm test1 +./ma_test1$suffix $silent -a -k 480 --unique +./maria_chk$suffix -sm test1 +./ma_test1$suffix $silent -a -N -S -R1 +./maria_chk$suffix -sm test1 +./ma_test1$suffix $silent -p -S +./maria_chk$suffix -sm test1 +./ma_test1$suffix $silent -p -S -N --unique +./maria_chk$suffix -sm test1 +./ma_test1$suffix $silent -p -S -N --key_length=127 --checksum +./maria_chk$suffix -sm test1 +./ma_test1$suffix $silent -p -S -N --key_length=128 +./maria_chk$suffix -sm test1 +./ma_test1$suffix $silent -p -S --key_length=480 +./maria_chk$suffix -sm test1 +./ma_test1$suffix $silent -a -B +./maria_chk$suffix -sm test1 +./ma_test1$suffix $silent -a -B --key_length=64 --unique +./maria_chk$suffix -sm test1 +./ma_test1$suffix $silent -a -B -k 480 --checksum +./maria_chk$suffix -sm test1 +./ma_test1$suffix $silent -a -B -k 480 -N --unique --checksum +./maria_chk$suffix -sm test1 +./ma_test1$suffix $silent -a -m +./maria_chk$suffix -sm test1 +./ma_test1$suffix $silent -a -m -P --unique --checksum +./maria_chk$suffix -sm test1 +./ma_test1$suffix $silent -a -m -P --key_length=480 --key_cache +./maria_chk$suffix -sm test1 +./ma_test1$suffix $silent -m -p +./maria_chk$suffix -sm test1 +./ma_test1$suffix $silent -w -S --unique +./maria_chk$suffix -sm test1 +./ma_test1$suffix $silent -a -w --key_length=64 --checksum +./maria_chk$suffix -sm test1 +./ma_test1$suffix $silent -a -w -N --key_length=480 +./maria_chk$suffix -sm test1 +./ma_test1$suffix $silent -a -w -S --key_length=480 --checksum +./maria_chk$suffix -sm test1 +./ma_test1$suffix $silent -a -b -N +./maria_chk$suffix -sm test1 +./ma_test1$suffix $silent -a -b --key_length=480 +./maria_chk$suffix -sm test1 +./ma_test1$suffix $silent -p -B --key_length=480 +./maria_chk$suffix -sm test1 -ma_test1$suffix $silent --checksum -maria_chk$suffix -se test1 -maria_chk$suffix -rs test1 -maria_chk$suffix -se test1 -maria_chk$suffix -rqs test1 -maria_chk$suffix -se test1 -maria_chk$suffix -rs --correct-checksum test1 -maria_chk$suffix -se test1 -maria_chk$suffix -rqs --correct-checksum test1 -maria_chk$suffix -se test1 -maria_chk$suffix -ros --correct-checksum test1 -maria_chk$suffix -se test1 -maria_chk$suffix -rqos --correct-checksum test1 -maria_chk$suffix -se test1 +./ma_test1$suffix $silent --checksum +./maria_chk$suffix -se test1 +./maria_chk$suffix -rs test1 +./maria_chk$suffix -se test1 +./maria_chk$suffix -rqs test1 +./maria_chk$suffix -se test1 +./maria_chk$suffix -rs --correct-checksum test1 +./maria_chk$suffix -se test1 +./maria_chk$suffix -rqs --correct-checksum test1 +./maria_chk$suffix -se test1 +./maria_chk$suffix -ros --correct-checksum test1 +./maria_chk$suffix -se test1 +./maria_chk$suffix -rqos --correct-checksum test1 +./maria_chk$suffix -se test1 # check of maria_pack / maria_chk -maria_pack$suffix --force -s test1 -maria_chk$suffix -es test1 -maria_chk$suffix -rqs test1 -maria_chk$suffix -es test1 -maria_chk$suffix -rs test1 -maria_chk$suffix -es test1 -maria_chk$suffix -rus test1 -maria_chk$suffix -es test1 +./maria_pack$suffix --force -s test1 +./maria_chk$suffix -es test1 +./maria_chk$suffix -rqs test1 +./maria_chk$suffix -es test1 +./maria_chk$suffix -rs test1 +./maria_chk$suffix -es test1 +./maria_chk$suffix -rus test1 +./maria_chk$suffix -es test1 -ma_test1$suffix $silent --checksum -S -maria_chk$suffix -se test1 -maria_chk$suffix -ros test1 -maria_chk$suffix -rqs test1 -maria_chk$suffix -se test1 +./ma_test1$suffix $silent --checksum -S +./maria_chk$suffix -se test1 +./maria_chk$suffix -ros test1 +./maria_chk$suffix -rqs test1 +./maria_chk$suffix -se test1 -maria_pack$suffix --force -s test1 -maria_chk$suffix -rqs test1 -maria_chk$suffix -es test1 -maria_chk$suffix -rus test1 -maria_chk$suffix -es test1 +./maria_pack$suffix --force -s test1 +./maria_chk$suffix -rqs test1 +./maria_chk$suffix -es test1 +./maria_chk$suffix -rus test1 +./maria_chk$suffix -es test1 -ma_test1$suffix $silent --checksum --unique -maria_chk$suffix -se test1 -ma_test1$suffix $silent --unique -S -maria_chk$suffix -se test1 +./ma_test1$suffix $silent --checksum --unique +./maria_chk$suffix -se test1 +./ma_test1$suffix $silent --unique -S +./maria_chk$suffix -se test1 -ma_test1$suffix $silent --key_multiple -N -S -maria_chk$suffix -sm test1 -ma_test1$suffix $silent --key_multiple -a -p --key_length=480 -maria_chk$suffix -sm test1 -ma_test1$suffix $silent --key_multiple -a -B --key_length=480 -maria_chk$suffix -sm test1 -ma_test1$suffix $silent --key_multiple -P -S -maria_chk$suffix -sm test1 +./ma_test1$suffix $silent --key_multiple -N -S +./maria_chk$suffix -sm test1 +./ma_test1$suffix $silent --key_multiple -a -p --key_length=480 +./maria_chk$suffix -sm test1 +./ma_test1$suffix $silent --key_multiple -a -B --key_length=480 +./maria_chk$suffix -sm test1 +./ma_test1$suffix $silent --key_multiple -P -S +./maria_chk$suffix -sm test1 -ma_test2$suffix $silent -L -K -W -P -maria_chk$suffix -sm test2 -ma_test2$suffix $silent -L -K -W -P -A -maria_chk$suffix -sm test2 -ma_test2$suffix $silent -L -K -W -P -S -R1 -m500 +./ma_test2$suffix $silent -L -K -W -P +./maria_chk$suffix -sm test2 +./ma_test2$suffix $silent -L -K -W -P -A +./maria_chk$suffix -sm test2 +./ma_test2$suffix $silent -L -K -W -P -S -R1 -m500 echo "ma_test2$suffix $silent -L -K -R1 -m2000 ; Should give error 135" -maria_chk$suffix -sm test2 -ma_test2$suffix $silent -L -K -R1 -m2000 -maria_chk$suffix -sm test2 -ma_test2$suffix $silent -L -K -P -S -R3 -m50 -b1000000 -maria_chk$suffix -sm test2 -ma_test2$suffix $silent -L -B -maria_chk$suffix -sm test2 -ma_test2$suffix $silent -D -B -c -maria_chk$suffix -sm test2 -ma_test2$suffix $silent -m10000 -e8192 -K -maria_chk$suffix -sm test2 -ma_test2$suffix $silent -m10000 -e16384 -E16384 -K -L -maria_chk$suffix -sm test2 +./maria_chk$suffix -sm test2 +./ma_test2$suffix $silent -L -K -R1 -m2000 +./maria_chk$suffix -sm test2 +./ma_test2$suffix $silent -L -K -P -S -R3 -m50 -b1000000 +./maria_chk$suffix -sm test2 +./ma_test2$suffix $silent -L -B +./maria_chk$suffix -sm test2 +./ma_test2$suffix $silent -D -B -c +./maria_chk$suffix -sm test2 +./ma_test2$suffix $silent -m10000 -e8192 -K +./maria_chk$suffix -sm test2 +./ma_test2$suffix $silent -m10000 -e16384 -E16384 -K -L +./maria_chk$suffix -sm test2 -ma_test2$suffix $silent -L -K -W -P -m50 -l -maria_log$suffix -ma_test2$suffix $silent -L -K -W -P -m50 -l -b100 -maria_log$suffix -time ma_test2$suffix $silent -time ma_test2$suffix $silent -K -B -time ma_test2$suffix $silent -L -B -time ma_test2$suffix $silent -L -K -B -time ma_test2$suffix $silent -L -K -W -B -time ma_test2$suffix $silent -L -K -W -S -B -time ma_test2$suffix $silent -D -K -W -S -B +./ma_test2$suffix $silent -L -K -W -P -m50 -l +./maria_log$suffix +./ma_test2$suffix $silent -L -K -W -P -m50 -l -b100 +./maria_log$suffix +time ./ma_test2$suffix $silent +time ./ma_test2$suffix $silent -K -B +time ./ma_test2$suffix $silent -L -B +time ./ma_test2$suffix $silent -L -K -B +time ./ma_test2$suffix $silent -L -K -W -B +time ./ma_test2$suffix $silent -L -K -W -S -B +time ./ma_test2$suffix $silent -D -K -W -S -B -- cgit v1.2.1 From 99a86a34bbff7cdc01d1ee47d461227304fc8d51 Mon Sep 17 00:00:00 2001 From: unknown Date: Wed, 12 Apr 2006 15:40:50 +0200 Subject: Exact same change as the one I just pushed in 5.1-new: 10 MB is enough for most cases. mysql-test/mysql-test-run.pl: 10 instead of 128 --- mysql-test/mysql-test-run.pl | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/mysql-test/mysql-test-run.pl b/mysql-test/mysql-test-run.pl index 6cec64e7120..918c7457677 100755 --- a/mysql-test/mysql-test-run.pl +++ b/mysql-test/mysql-test-run.pl @@ -2566,7 +2566,7 @@ sub mysqld_arguments ($$$$$$) { mtr_add_arg($args, "%s--server-id=%d", $prefix, $id); mtr_add_arg($args, "%s--socket=%s", $prefix, $master->[$idx]->{'path_mysock'}); - mtr_add_arg($args, "%s--innodb_data_file_path=ibdata1:128M:autoextend", $prefix); + mtr_add_arg($args, "%s--innodb_data_file_path=ibdata1:10M:autoextend", $prefix); mtr_add_arg($args, "%s--local-infile", $prefix); mtr_add_arg($args, "%s--datadir=%s", $prefix, $master->[$idx]->{'path_myddir'}); -- cgit v1.2.1 From 06f7675b957fa68c69ae37c31b457032921f838e Mon Sep 17 00:00:00 2001 From: unknown Date: Thu, 27 Apr 2006 16:06:46 +0200 Subject: Maria: first version of checkpoint (WL#3071), least-recently-dirtied page flushing (WL#3261), recovery (WL#3072), control file (WL#3234), to serve as a detailed LLD. It looks like C code, but does not compile (no point in making it compile, as other modules on which I depend are not yet fully speficied or written); some pieces are not coded and just marked in comments. Files' organization (names, directories of C files) does not matter at this point. I don't think I had to commit so early, but it feels good to publish something, gives me the impression of moving forward :) storage/maria/checkpoint.c: WL#3071 Maria checkpoint, implementation storage/maria/checkpoint.h: WL#3071 Maria checkpoint, interface storage/maria/control_file.c: WL#3234 Maria control file, implementation storage/maria/control_file.h: WL#3234 Maria control file, interface storage/maria/least_recently_dirtied.c: WL#3261 Maria background flushing of least-recently-dirtied pages, implementation storage/maria/least_recently_dirtied.h: WL#3261 Maria background flushing of least-recently-dirtied pages, interface storage/maria/recovery.c: WL#3072 Maria recovery, implementation storage/maria/recovery.h: WL#3072 Maria recovery, interface --- storage/maria/checkpoint.c | 394 +++++++++++++++++++++++++++++++++ storage/maria/checkpoint.h | 23 ++ storage/maria/control_file.c | 77 +++++++ storage/maria/control_file.h | 24 ++ storage/maria/least_recently_dirtied.c | 175 +++++++++++++++ storage/maria/least_recently_dirtied.h | 10 + storage/maria/recovery.c | 224 +++++++++++++++++++ storage/maria/recovery.h | 10 + 8 files changed, 937 insertions(+) create mode 100644 storage/maria/checkpoint.c create mode 100644 storage/maria/checkpoint.h create mode 100644 storage/maria/control_file.c create mode 100644 storage/maria/control_file.h create mode 100644 storage/maria/least_recently_dirtied.c create mode 100644 storage/maria/least_recently_dirtied.h create mode 100644 storage/maria/recovery.c create mode 100644 storage/maria/recovery.h diff --git a/storage/maria/checkpoint.c b/storage/maria/checkpoint.c new file mode 100644 index 00000000000..af37377455e --- /dev/null +++ b/storage/maria/checkpoint.c @@ -0,0 +1,394 @@ +/* + WL#3071 Maria checkpoint + First version written by Guilhem Bichot on 2006-04-27. + Does not compile yet. +*/ + +/* Here is the implementation of this module */ + +#include "page_cache.h" +#include "least_recently_dirtied.h" +#include "transaction.h" +#include "share.h" +#include "log.h" + +/* + this transaction is used for any system work (purge, checkpoint writing + etc), that is, background threads. It will not be declared/initialized here + in the final version. +*/ +st_transaction system_trans= {0 /* long trans id */, 0 /* short trans id */,0,...}; + +/* + The maximum rec_lsn in the LRD when last checkpoint was run, serves for the + MEDIUM checkpoint. +*/ +LSN max_rec_lsn_at_last_checkpoint= 0; + +/* Picks a checkpoint request and executes it */ +my_bool checkpoint() +{ + CHECKPOINT_LEVEL level; + DBUG_ENTER("checkpoint"); + + level= checkpoint_running= checkpoint_request; + unlock(log_mutex); + + DBUG_ASSERT(level != NONE); + + switch (level) + { + case FULL: + /* flush all pages up to the current end of the LRD */ + flush_all_LRD_to_lsn(MAX_LSN); /* MAX_LSN==ULONGLONG_MAX */ + /* this will go full speed (normal scheduling, no sleep) */ + break; + case MEDIUM: + /* + flush all pages which were already dirty at last checkpoint: + ensures that recovery will never start from before the next-to-last + checkpoint (two-checkpoint rule). + It is max, not min as the WL says (TODO update WL). + */ + flush_all_LRD_to_lsn(max_rec_lsn_at_last_checkpoint); + /* this will go full speed (normal scheduling, no sleep) */ + break; + } + + error= checkpoint_indirect(); + + lock(log_mutex); + /* + this portion cannot be done as a hook in write_log_record() for the + LOGREC_CHECKPOINT type because: + - at that moment we still have not written to the control file so cannot + mark the request as done; this could be solved by writing to the control + file in the hook but that would be an I/O under the log's mutex, bad. + - it would not be nice organisation of code (I tried it :). + */ + mark_checkpoint_done(error); + unlock(log_mutex); + DBUG_RETURN(error); +} + + +my_bool checkpoint_indirect() +{ + DBUG_ENTER("checkpoint_indirect"); + + int error= 0; + /* checkpoint record data: */ + LSN checkpoint_start_lsn; + LEX_STRING string1={0,0}, string2={0,0}, string3={0,0}; + LEX_STRING *string_array[4]; + char *ptr; + LSN checkpoint_lsn; + LSN candidate_max_rec_lsn_at_last_checkpoint= 0; + list_element *el; /* to scan lists */ + + + DBUG_ASSERT(sizeof(byte *) <= 8); + DBUG_ASSERT(sizeof(LSN) <= 8); + + lock(log_mutex); /* will probably be in log_read_end_lsn() already */ + checkpoint_start_lsn= log_read_end_lsn(); + unlock(log_mutex); + + DBUG_PRINT("info",("checkpoint_start_lsn %lu", checkpoint_start_lsn)); + + lock(global_LRD_mutex); + string1.length= 8+8+(8+8)*LRD->count; + if (NULL == (string1.str= my_malloc(string1.length))) + goto err; + ptr= string1.str; + int8store(ptr, checkpoint_start_lsn); + ptr+= 8; + int8store(ptr, LRD->count); + ptr+= 8; + if (LRD->count) + { + candidate_max_rec_lsn_at_last_checkpoint= LRD->last->rec_lsn; + for (el= LRD->first; el; el= el->next) + { + int8store(ptr, el->page_id); + ptr+= 8; + int8store(ptr, el->rec_lsn); + ptr+= 8; + } + } + unlock(global_LRD_mutex); + + /* + If trx are in more than one list (e.g. three: + running transactions, committed transactions, purge queue), we can either + take mutexes of all three together or do crabbing. + But if an element can move from list 1 to list 3 without passing through + list 2, crabbing is dangerous. + Hopefully it's ok to take 3 mutexes together... + Otherwise I'll have to make sure I miss no important trx and I handle dups. + */ + lock(global_transactions_list_mutex); /* or 3 mutexes if there are 3 */ + string2.length= 8+(8+8)*trx_list->count; + if (NULL == (string2.str= my_malloc(string2.length))) + goto err; + ptr= string2.str; + int8store(ptr, trx_list->count); + ptr+= 8; + for (el= trx_list->first; el; el= el->next) + { + /* possibly latch el.rwlock */ + *ptr= el->state; + ptr++; + int7store(ptr, el->long_trans_id); + ptr+= 7; + int2store(ptr, el->short_trans_id); + ptr+= 2; + int8store(ptr, el->undo_lsn); + ptr+= 8; + int8store(ptr, el->undo_purge_lsn); + ptr+= 8; + /* + if no latch, use double variable of type ULONGLONG_CONSISTENT in + st_transaction, or even no need if Intel >=486 + */ + int8store(ptr, el->first_purge_lsn); + ptr+= 8; + /* possibly unlatch el.rwlock */ + } + unlock(global_transactions_list_mutex); + + lock(global_share_list_mutex); + string3.length= 8+(8+8)*share_list->count; + if (NULL == (string3.str= my_malloc(string3.length))) + goto err; + ptr= string3.str; + /* possibly latch each MARIA_SHARE */ + make_copy_of_global_share_list_to_array; + unlock(global_share_list_mutex); + + /* work on copy */ + int8store(ptr, elements_in_array); + ptr+= 8; + for (scan_array) + { + int8store(ptr, array[...].file_id); + ptr+= 8; + memcpy(ptr, array[...].file_name, ...); + ptr+= ...; + /* + these two are long ops (involving disk I/O) that's why we copied the + list: + */ + flush_bitmap_pages(el); + /* + fsyncs the fd, that's the loooong operation (e.g. max 150 fsync per + second, so if you have touched 1000 files it's 7 seconds). + */ + force_file(el); + } + + /* now write the record */ + string_array[0]= string1; + string_array[1]= string2; + string_array[2]= string3; + string_array[3]= NULL; + + checkpoint_lsn= log_write_record(LOGREC_CHECKPOINT, + &system_trans, string_array); + + if (0 == checkpoint_lsn) /* maybe 0 is impossible LSN to indicate error ? */ + goto err; + + if (0 != control_file_write_and_force(checkpoint_lsn, NULL)) + goto err; + + maximum_rec_lsn_last_checkpoint= candidate_max_rec_lsn_at_last_checkpoint; + + DBUG_RETURN(0); + +err: + + print_error_to_error_log(the_error_message); + my_free(buffer1.str, MYF(MY_ALLOW_ZERO_PTR)); + my_free(buffer2.str, MYF(MY_ALLOW_ZERO_PTR)); + my_free(buffer3.str, MYF(MY_ALLOW_ZERO_PTR)); + + DBUG_RETURN(1); +} + + + +/* + Here's what should be put in log_write_record() in the log handler: +*/ +log_write_record(...) +{ + ...; + lock(log_mutex); + ...; + write_to_log(length); + written_since_last_checkpoint+= length; + if (written_since_last_checkpoint > + MAX_LOG_BYTES_WRITTEN_BETWEEN_CHECKPOINTS) + { + /* + ask one system thread (the "LRD background flusher and checkpointer + thread" WL#3261) to do a checkpoint + */ + request_checkpoint(INDIRECT, 0 /*wait_for_completion*/); + } + ...; + unlock(log_mutex); + ...; +} + +/* + Call this when you want to request a checkpoint. + In real life it will be called by log_write_record() and by client thread + which explicitely wants to do checkpoint (ALTER ENGINE CHECKPOINT + checkpoint_level). +*/ +int request_checkpoint(CHECKPOINT_LEVEL level, my_bool wait_for_completion) +{ + int error= 0; + /* + If caller wants to wait for completion we'll have to release the log mutex + to wait on condition, if caller had log mutex he may not be happy that we + release it, so we check that caller didn't have log mutex. + */ + if (wait_for_completion) + { + lock(log_mutex); + } + else + safemutex_assert_owner(log_mutex); + + DBUG_ASSERT(checkpoint_request >= checkpoint_running); + DBUG_ASSERT(level > NONE); + if (checkpoint_request < level) + { + /* no equal or stronger running or to run, we post request */ + /* + note that thousands of requests for checkpoints are going to come all + at the same time (when the log bound is passed), so it may not be a good + idea for each of them to broadcast a cond. We just don't broacast a + cond, the checkpoint thread will wake up in max one second. + */ + checkpoint_request= level; /* post request */ + } + + if (wait_for_completion) + { + uint checkpoints_done_copy= checkpoints_done; + uint checkpoint_errors_copy= checkpoint_errors; + /* + note that the "==done" works when the uint counter wraps too, so counter + can even be smaller than uint if we wanted (however it should be big + enough so that max_the_int_type checkpoints cannot happen between two + wakeups of our thread below). uint sounds fine. + Wait for our checkpoint to be done: + */ + + if (checkpoint_running != NONE) /* not ours, let it pass */ + { + while (1) + { + if (checkpoints_done != checkpoints_done_copy) + { + if (checkpoints_done == (checkpoints_done_copy+1)) + { + /* not our checkpoint, forget about it */ + checkpoints_done_copy= checkpoints_done; + } + break; /* maybe even ours has been done at this stage! */ + } + cond_wait(checkpoint_done_cond, log_mutex); + } + } + + /* now we come to waiting for our checkpoint */ + while (1) + { + if (checkpoints_done != checkpoints_done_copy) + { + /* our checkpoint has been done */ + break; + } + if (checkpoint_errors != checkpoint_errors_copy) + { + /* + the one which was running a few milliseconds ago (if there was one), + and/or ours, had an error, just assume it was ours. So there + is a possibility that we return error though we succeeded, in which + case user will have to retry; but two simultanate checkpoints have + high changes to fail together (as the error probably comes from + malloc or disk write problem), so chance of false alarm is low. + Reporting the error only to the one which caused the error would + require having a (not fixed size) list of all requests, not worth it. + */ + error= 1; + break; + } + cond_wait(checkpoint_done_cond, log_mutex); + } + unlock(log_mutex); + } /* ... if (wait_for_completion) */ + + /* + If wait_for_completion was false, and there was an error, only an error + message to the error log will say it; normal, for a checkpoint triggered + by a log write, we probably don't want the client's log write to throw an + error, as the log write succeeded and a checkpoint failure is not + critical: the failure in this case is more for the DBA to know than for + the end user. + */ + return error; +} + +void mark_checkpoint_done(int error) +{ + safemutex_assert_owner(log_mutex); + if (error) + checkpoint_errors++; + /* a checkpoint is said done even if it had an error */ + checkpoints_done++; + if (checkpoint_request == checkpoint_running) + { + /* + No new request has been posted, so we satisfied all requests, forget + about them. + */ + checkpoint_request= NONE; + } + checkpoint_running= NONE; + written_since_last_checkpoint= 0; + broadcast(checkpoint_done_cond); +} + +/* + Alternative (not to be done, too disturbing): + do the autocheckpoint in the thread which passed the bound first (and do the + checkpoint in the client thread which requested it). + It will give a delay to that client thread which passed the bound (time to + fsync() for example 1000 files is 16 s on my laptop). Here is code for + explicit and implicit checkpoints, where client thread does the job: +*/ +#if 0 +{ + lock(log_mutex); /* explicit takes it here, implicit already has it */ + while (checkpoint_running != NONE) + { + if (checkpoint_running >= my_level) /* always true for auto checkpoints */ + goto end; /* we skip checkpoint */ + /* a less strong is running, I'll go next */ + wait_on_checkpoint_done_cond(); + } + checkpoint_running= my_level; + checkpoint(my_level); // can gather checkpoint_start_lsn before unlock + lock(log_mutex); + checkpoint_running= NONE; + written_since_last_checkpoint= 0; +end: + unlock(log_mutex); +} +#endif diff --git a/storage/maria/checkpoint.h b/storage/maria/checkpoint.h new file mode 100644 index 00000000000..ce8066de93d --- /dev/null +++ b/storage/maria/checkpoint.h @@ -0,0 +1,23 @@ +/* + WL#3071 Maria checkpoint + First version written by Guilhem Bichot on 2006-04-27. + Does not compile yet. +*/ + +/* This is the interface of this module. */ + +typedef enum enum_checkpoint_level { + NONE=-1, + INDIRECT, /* just write dirty_pages, transactions table and sync files */ + MEDIUM, /* also flush all dirty pages which were already dirty at prev checkpoint*/ + FULL /* also flush all dirty pages */ +} CHECKPOINT_LEVEL; + +/* + Call this when you want to request a checkpoint. + In real life it will be called by log_write_record() and by client thread + which explicitely wants to do checkpoint (ALTER ENGINE CHECKPOINT + checkpoint_level). +*/ +int request_checkpoint(CHECKPOINT_LEVEL level, my_bool wait_for_completion); +/* that's all that's needed in the interface */ diff --git a/storage/maria/control_file.c b/storage/maria/control_file.c new file mode 100644 index 00000000000..897e0b0f0ee --- /dev/null +++ b/storage/maria/control_file.c @@ -0,0 +1,77 @@ +/* + WL#3234 Maria control file + First version written by Guilhem Bichot on 2006-04-27. + Does not compile yet. +*/ + +/* Here is the implementation of this module */ + +/* Control file is 512 bytes (a disk sector), to be as atomic as possible */ + +int control_file_fd; + +/* + Looks for the control file. If absent, it's a fresh start, create file. + If present, read it to find out last checkpoint's LSN and last log. + Called at engine's start. +*/ +int control_file_create_or_open() +{ + char buffer[4]; + /* name is concatenation of Maria's home dir and "control" */ + if ((control_file_fd= my_open(name, O_RDWR)) < 0) + { + /* failure, try to create it */ + if ((control_file_fd= my_create(name, O_RDWR)) < 0) + return 1; + /* + So this is a start from scratch, to be safer we should make sure that + there are no logs or data/index files around (indeed it could be that + the control file alone was deleted or not restored, and we should not + go on with life at this point. + For now we trust (this is alpha version), but for beta if would be great + to verify. + + We could have a tool which can rebuild the control file, by reading the + directory of logs, finding the newest log, reading it to find last + checkpoint... Slow but can save your db. + */ + last_checkpoint_lsn_at_startup= 0; + last_log_name_at_startup= NULL; + return 0; + } + /* Already existing file, read it */ + if (my_read(control_file_fd, buffer, 8, MYF(MY_FNABP))) + return 1; + last_checkpoint_lsn_at_startup= uint8korr(buffer); + if (last_log_name_at_startup= my_malloc(512-8+1)) + return 1; + if (my_read(control_file_fd, last_log_name_at_startup, 512-8), MYF(MY_FNABP)) + return 1; + last_log_name[512-8]= 0; /* end zero to be nice */ + return 0; +} + +/* + Write information durably to the control file. + Called when we have created a new log (after syncing this log's creation) + and when we have written a checkpoint (after syncing this log record). +*/ +int control_file_write_and_force(LSN lsn, char *log_name) +{ + char buffer[512]; + uint start=8,end=8; + if (lsn != 0) /* LSN was specified */ + { + start= 0; + int8store(buffer, lsn); + } + if (log_name != NULL) /* log name was specified */ + { + end= 512; + memcpy(buffer+8, log_name, 512-8); + } + DBUG_ASSERT(start != end); + return (my_pwrite(control_file_fd, buffer, end-start, start, MYF(MY_FNABP)) || + my_sync(control_file_fd)) +} diff --git a/storage/maria/control_file.h b/storage/maria/control_file.h new file mode 100644 index 00000000000..d148ab7d88a --- /dev/null +++ b/storage/maria/control_file.h @@ -0,0 +1,24 @@ +/* + WL#3234 Maria control file + First version written by Guilhem Bichot on 2006-04-27. + Does not compile yet. +*/ + +/* Here is the interface of this module */ + +LSN last_checkpoint_lsn_at_startup; +char *last_log_name_at_startup; + +/* + Looks for the control file. If absent, it's a fresh start, create file. + If present, read it to find out last checkpoint's LSN and last log. + Called at engine's start. +*/ +int control_file_create_or_open(); + +/* + Write information durably to the control file. + Called when we have created a new log (after syncing this log's creation) + and when we have written a checkpoint (after syncing this log record). +*/ +int control_file_write_and_force(LSN lsn, char *log_name); diff --git a/storage/maria/least_recently_dirtied.c b/storage/maria/least_recently_dirtied.c new file mode 100644 index 00000000000..e251e3c582c --- /dev/null +++ b/storage/maria/least_recently_dirtied.c @@ -0,0 +1,175 @@ +/* + WL#3261 Maria - background flushing of the least-recently-dirtied pages + First version written by Guilhem Bichot on 2006-04-27. + Does not compile yet. +*/ + +/* + To be part of the page cache. + The pseudocode below is dependent on the page cache + which is being designed WL#3134. It is not clear if I need to do page + copies, as the page cache already keeps page copies. + So, this code will move to the page cache and take inspiration from its + methods. Below is just to give the idea of what could be done. + And I should compare my imaginations to WL#3134. +*/ + +/* Here is the implementation of this module */ + +#include "page_cache.h" +#include "least_recently_dirtied.h" + +/* + When we flush a page, we should pin page. + This "pin" is to protect against that: + I make copy, + you modify in memory and flush to disk and remove from LRD and from cache, + I write copy to disk, + checkpoint happens. + result: old page is on disk, page is absent from LRD, your REDO will be + wrongly ignored. + + Pin: there can be multiple pins, flushing imposes that there are zero pins. + For example, pin could be a uint counter protected by the page's latch. + + Maybe it's ok if when there is a page replacement, the replacer does not + remove page from the LRD (it would save global mutex); for that, background + flusher should be prepared to see pages in the LRD which are not in the page + cache (then just ignore them). However checkpoint will contain superfluous + entries and so do more work. +*/ + +#define PAGE_SIZE (16*1024) /* just as an example */ +/* + Optimization: + LRD flusher should not flush pages one by one: to be fast, it flushes a + group of pages in sequential disk order if possible; a group of pages is just + FLUSH_GROUP_SIZE pages. + Key cache has groupping already somehow Monty said (investigate that). +*/ +#define FLUSH_GROUP_SIZE 512 /* 8 MB */ + +/* + This thread does background flush of pieces of the LRD, and all checkpoints. + Just launch it when engine starts. +*/ +pthread_handler_decl background_flush_and_checkpoint_thread() +{ + char *flush_group_buffer= my_malloc(PAGE_SIZE*FLUSH_GROUP_SIZE); + while (this_thread_not_killed) + { + lock(log_mutex); + if (checkpoint_request) + checkpoint(); /* will unlock mutex */ + else + { + unlock(log_mutex); + lock(global_LRD_mutex); + flush_one_group_from_LRD(); + safemutex_assert_not_owner(global_LRD_mutex); + } + my_sleep(1000000); /* one second ? */ + } + my_free(flush_group_buffer); +} + +/* + flushes only the first FLUSH_GROUP_SIZE pages of the LRD. +*/ +flush_one_group_from_LRD() +{ + char *ptr; + safe_mutex_assert_owner(global_LRD_mutex); + + for (page= 0; pagedata, PAGE_SIZE); + pin_page; + page_cache_unlatch(page_id, KEEP_PINNED); /* but keep pinned */ + } + for (scan_the_array) + { + /* + As an optimization, we try to identify contiguous-in-the-file segments (to + issue one big write()). + In non-optimized version, contiguous segment is always only one page. + */ + if ((next_page.page_id - this_page.page_id) == 1) + { + /* + this page and next page are in same file and are contiguous in the + file: add page to contiguous segment... + */ + continue; /* defer write() to next pages */ + } + /* contiguous segment ends */ + my_pwrite(file, contiguous_segment_start_offset, contiguous_segment_size); + + /* + note that if we had doublewrite, doublewrite buffer may prevent us from + doing this write() grouping (if doublewrite space is shorter). + */ + } + /* + Now remove pages from LRD. As we have pinned them, all pages that we + managed to pin are still in the LRD, in the same order, we can just cut + the LRD at the last element of "array". This is more efficient that + removing element by element (which would take LRD mutex many times) in the + loop above. + */ + lock(global_LRD_mutex); + /* cut LRD by bending LRD->first, free cut portion... */ + unlock(global_LRD_mutex); + for (scan_array) + { + /* + if the page has a property "modified since last flush" (i.e. which is + redundant with the presence of the page in the LRD, this property can + just be a pointer to the LRD element) we should reset it + (note that then the property would live slightly longer than + the presence in LRD). + */ + page_cache_unpin(page_id); + /* + order between unpin and removal from LRD is not clear, depends on what + pin actually is. + */ + } + free(array); +} + +/* flushes all page from LRD up to approximately rec_lsn>=max_lsn */ +int flush_all_LRD_to_lsn(LSN max_lsn) +{ + lock(global_LRD_mutex); + if (max_lsn == MAX_LSN) /* don't want to flush forever, so make it fixed: */ + max_lsn= LRD->first->prev->rec_lsn; + while (LRD->first->rec_lsn < max_lsn) + { + if (flush_one_group_from_LRD()) /* will unlock mutex */ + return 1; + /* scheduler may preempt us here so that we don't take full CPU */ + lock(global_LRD_mutex); + } + unlock(global_LRD_mutex); + return 0; +} diff --git a/storage/maria/least_recently_dirtied.h b/storage/maria/least_recently_dirtied.h new file mode 100644 index 00000000000..6a30db4b5f0 --- /dev/null +++ b/storage/maria/least_recently_dirtied.h @@ -0,0 +1,10 @@ +/* + WL#3261 Maria - background flushing of the least-recently-dirtied pages + First version written by Guilhem Bichot on 2006-04-27. + Does not compile yet. +*/ + +/* This is the interface of this module. */ + +/* flushes all page from LRD up to approximately rec_lsn>=max_lsn */ +int flush_all_LRD_to_lsn(LSN max_lsn); diff --git a/storage/maria/recovery.c b/storage/maria/recovery.c new file mode 100644 index 00000000000..aee8f1de749 --- /dev/null +++ b/storage/maria/recovery.c @@ -0,0 +1,224 @@ +/* + WL#3072 Maria recovery + First version written by Guilhem Bichot on 2006-04-27. + Does not compile yet. +*/ + +/* Here is the implementation of this module */ + +#include "page_cache.h" +#include "least_recently_dirtied.h" +#include "transaction.h" +#include "share.h" +#include "log.h" + +typedef struct st_record_type_properties { + /* used for debug error messages or "maria_read_log" command-line tool: */ + char *name, + my_bool record_ends_group; + int (*record_execute)(RECORD *); /* param will be record header instead later */ +} RECORD_TYPE_PROPERTIES; + +RECORD_TYPE_PROPERTIES all_record_type_properties[]= +{ + /* listed here in the order of the "log records type" enumeration */ + {"REDO_INSERT_HEAD", 0, redo_insert_head_execute}, + ..., + {"UNDO_INSERT" , 1, undo_insert_execute }, + {"COMMIT", , 1, commit_execute }, + ... +}; + +int redo_insert_head_execute(RECORD *record) +{ + /* write the data to the proper page */ +} + +int undo_insert_execute(RECORD *record) +{ + trans_table[short_trans_id].undo_lsn= record.lsn; + /* restore the old version of the row */ +} + +int commit_execute(RECORD *record) +{ + trans_table[short_trans_id].state= COMMITTED; + /* + and that's all: the delete/update handler should not be woken up! as there + may be REDO for purge further in the log. + */ +} + +#define record_ends_group(R) \ + all_record_type_properties[(R)->type].record_ends_group) + +#define execute_log_record(R) \ + all_record_type_properties[(R).type].record_execute(R) + + +int recovery() +{ + control_file_create_or_open(); + /* + init log handler: tell it that we are going to do large reads of the + log, sequential and backward. Log handler could decide to alloc a big + read-only IO_CACHE for this, or use its usual page cache. + */ + + /* read checkpoint log record from log handler */ + RECORD *checkpoint_record= log_read_record(last_checkpoint_lsn_at_start); + + /* parse this record, build structs (dirty_pages, transactions table, file_map) */ + /* + read log records (note: sometimes only the header is needed, for ex during + REDO phase only the header of UNDO is needed, not the 4G blob in the + variable-length part, so I could use that; however for PREPARE (which is a + variable-length record) I'll need to read the full record in the REDO + phase): + */ + + record= log_read_record(min(rec_lsn, ...)); + /* + if log handler knows the end LSN of the log, we could print here how many + MB of log we have to read (to give an idea of the time), and print + progress notes. + */ + + while (record != NULL) + { + /* + A complete group is a set of log records with an "end mark" record + (e.g. a set of REDOs for an operation, terminated by an UNDO for this + operation); if there is no "end mark" record the group is incomplete + and won't be executed. + */ + if (record_ends_group(record) + { + /* + such end events can always be executed immediately (they don't touch + the disk). + */ + execute_log_record(record); + if (trans_table[record.short_trans_id].group_start_lsn != 0) + { + /* + There is a complete group for this transaction. + We're going to read recently read log records: + for this log_read_record() to be efficient (not touch the disk), + log handler could cache recently read pages + (can just use an IO_CACHE of 10 MB to read the log, or the normal + log handler page cache). + Without it only OS file cache will help. + */ + record2= log_read_record(trans_table[record.short_trans_id].group_start_lsn); + while (record2.lsn < record.lsn) + { + if (record2.short_trans_id == record.short_trans_id) + execute_log_record(record2); /* it's in our group */ + record2= log_read_next_record(); + } + trans_table[record.short_trans_id].group_start_lsn= 0; /* group finished */ + /* we're now at the UNDO, re-read it to advance log pointer */ + record2= log_read_next_record(); /* and throw it away */ + } + } + else /* record does not end group */ + { + /* just record the fact, can't know if can execute yet */ + if (trans_table[short_trans_id].group_start_lsn == 0) /* group not yet started */ + trans_table[short_trans_id].group_start_lsn= record.lsn; + } + + /* + Later we can optimize: instead of "execute_log_record(record2)", do + copy_record_into_exec_buffer(record2): + this will just copy record into a multi-record (10 MB?) memory buffer, + and when buffer is full, will do sorting of REDOs per + page id and execute them. + This sorting will enable us to do more sequential reads of the + data/index pages. + Note that updating bitmap pages (when we have executed a REDO for a page + we update its bitmap page) may break the sequential read of pages, + so maybe we should read and cache bitmap pages in the beginning. + Or ok the sequence will be broken, but quickly all bitmap pages will be + in memory and so the sequence will not be broken anymore. + Sorting could even determine, based on physical device of files + ("st_dev" in stat()), that some files should be should be taken by + different threads, if we want to do parallism. + */ + /* + Here's how to read a complete variable-length record if needed: + read the header, allocate buffer of record length, read whole + record. + */ + record= log_read_next_record(); + } + + /* + Earlier or here, create true transactions in TM. + If done earlier, note that TM should not wake up the delete/update handler + when it receives a commit info, as existing REDO for purge may exist in + the log, and so the delete/update handler may do changes which conflict + with these REDOs. + Even if done here, better to not wake it up now as we're going to free the + page cache: + */ + + /* + We want to have two steps: + engine->recover_with_max_memory(); + next_engine->recover_with_max_memory(); + engine->init_with_normal_memory(); + next_engine->init_with_normal_memory(); + So: in recover_with_max_memory() allocate a giant page cache, do REDO + phase, then all page cache is flushed and emptied and freed (only retain + small structures like TM): take full checkpoint, which is useful if + next engine crashes in its recovery the next second. + Destroy all shares (maria_close()), then at init_with_normal_memory() we + do this: + */ + + print_information_to_error_log(nb of trans to roll back, nb of prepared trans); + + /* + Launch one or more threads to do the background rollback. Don't wait for + them to complete their rollback (background rollback; for debugging, we + can have an option which waits). + + Note that InnoDB's rollback-in-background works as long as InnoDB is the + last engine to recover, otherwise MySQL will refuse new connections until + the last engine has recovered so it's not "background" from the user's + point of view. InnoDB is near top of sys_table_types so all others + (e.g. BDB) recover after it... So it's really "online rollback" only if + InnoDB is the only engine. + */ + + /* wake up delete/update handler */ + /* tell the TM that it can now accept new transactions */ + + /* + mark that checkpoint requests are now allowed. + */ + /* + when all rollback threads have terminated, somebody should print "rollback + finished" to the error log. + */ +} + +pthread_handler_decl rollback_background_thread() +{ + /* + execute the normal runtime-rollback code for a bunch of transactions. + */ + while (trans in list_of_trans_to_rollback_by_this_thread) + { + while (trans->undo_lsn != 0) + { + /* this is the normal runtime-rollback code: */ + record= log_read_record(trans->undo_lsn); + execute_log_record(record); + trans->undo_lsn= record.prev_undo_lsn; + } + /* remove trans from list */ + } +} diff --git a/storage/maria/recovery.h b/storage/maria/recovery.h new file mode 100644 index 00000000000..b85ffdeef59 --- /dev/null +++ b/storage/maria/recovery.h @@ -0,0 +1,10 @@ +/* + WL#3072 Maria recovery + First version written by Guilhem Bichot on 2006-04-27. + Does not compile yet. +*/ + +/* This is the interface of this module. */ + +/* Performs recovery of the engine at start */ +int recovery(); -- cgit v1.2.1 From 34dba480548e7cc9647d8c943beb92d161ec0281 Mon Sep 17 00:00:00 2001 From: unknown Date: Thu, 4 May 2006 10:59:19 +0200 Subject: WL#3072 Maria recovery, small fixes to pseudocode, discussion why no checkpoint at end of UNDO phase. storage/maria/control_file.h: small fix storage/maria/recovery.c: small fixes to the pseudocode and discussion of why no checkpoint at end of UNDO phase. --- storage/maria/control_file.h | 4 ++-- storage/maria/recovery.c | 19 ++++++++++++++++++- 2 files changed, 20 insertions(+), 3 deletions(-) diff --git a/storage/maria/control_file.h b/storage/maria/control_file.h index d148ab7d88a..522e7565341 100644 --- a/storage/maria/control_file.h +++ b/storage/maria/control_file.h @@ -6,8 +6,8 @@ /* Here is the interface of this module */ -LSN last_checkpoint_lsn_at_startup; -char *last_log_name_at_startup; +extern LSN last_checkpoint_lsn_at_startup; +extern char *last_log_name_at_startup; /* Looks for the control file. If absent, it's a fresh start, create file. diff --git a/storage/maria/recovery.c b/storage/maria/recovery.c index aee8f1de749..5af7019e5fb 100644 --- a/storage/maria/recovery.c +++ b/storage/maria/recovery.c @@ -183,7 +183,8 @@ int recovery() /* Launch one or more threads to do the background rollback. Don't wait for them to complete their rollback (background rollback; for debugging, we - can have an option which waits). + can have an option which waits). Set a counter (total_of_rollback_threads) + to the number of threads to lauch. Note that InnoDB's rollback-in-background works as long as InnoDB is the last engine to recover, otherwise MySQL will refuse new connections until @@ -221,4 +222,20 @@ pthread_handler_decl rollback_background_thread() } /* remove trans from list */ } + lock_mutex(rollback_threads); /* or atomic counter */ + if (--total_of_rollback_threads == 0) + { + /* + All rollback threads are done. Print "rollback finished" to the error + log. The UNDO phase has the reputation of being a slow operation + (slower than the REDO phase), so taking a checkpoint at the end of it is + intelligent, but as this UNDO phase generates REDOs and CLR_ENDs, if it + did a lot of work then the "automatic checkpoint when much has been + written to the log" will do it; and if the UNDO phase didn't do a lot of + work, no need for a checkpoint. If we change our mind and want to force + a checkpoint at the end of the UNDO phase, simply call it here. + */ + } + unlock_mutex(rollback_threads); + pthread_exit(); } -- cgit v1.2.1 From 157002b12f4134eb6ca2b84ba95558e3b6638929 Mon Sep 17 00:00:00 2001 From: unknown Date: Fri, 5 May 2006 20:32:02 +0200 Subject: WL#3270 "Maria - cleanups of inherited MyISAM functionality" Removing the "external lock" functionality from Maria (as two separate processes wanting to share a table should not only my_lock() the data and index files but also the log files, and share memory (as the latest data is in the page cache), it sounds useless to feature this). Removing the MyISAM logging from Maria (as REDO logging will be done differently). BitKeeper/deleted/.del-maria_log.c~1fb295a18c3f5d4c: Delete: storage/maria/maria_log.c BitKeeper/deleted/.del-ma_log.c~4a44ec11d547772f: Delete: storage/maria/ma_log.c include/maria.h: unneeded storage/maria/Makefile.am: log removed storage/maria/ma_check.c: external locking removed storage/maria/ma_close.c: log removed storage/maria/ma_delete.c: log removed storage/maria/ma_delete_all.c: log removed. Unused var. storage/maria/ma_dynrec.c: external locking removed storage/maria/ma_extra.c: log removed storage/maria/ma_init.c: log removed storage/maria/ma_locking.c: external locking removed, log removed storage/maria/ma_open.c: external locking removed, log removed storage/maria/ma_static.c: log removed storage/maria/ma_statrec.c: external locking removed storage/maria/ma_test2.c: log removed storage/maria/ma_test3.c: log removed storage/maria/ma_update.c: log removed storage/maria/ma_write.c: external locking removed, log removed storage/maria/maria_chk.c: external locking removed storage/maria/maria_def.h: log removed, maria_pid unused. storage/maria/maria_pack.c: fixes for warnings (where pointers are like ulong and so %u is not enough). --- include/maria.h | 2 - storage/maria/Makefile.am | 5 +- storage/maria/ma_check.c | 20 +- storage/maria/ma_close.c | 1 - storage/maria/ma_delete.c | 2 - storage/maria/ma_delete_all.c | 2 - storage/maria/ma_dynrec.c | 6 - storage/maria/ma_extra.c | 1 - storage/maria/ma_init.c | 1 - storage/maria/ma_locking.c | 70 +--- storage/maria/ma_log.c | 164 -------- storage/maria/ma_open.c | 24 +- storage/maria/ma_static.c | 2 - storage/maria/ma_statrec.c | 3 - storage/maria/ma_test2.c | 7 +- storage/maria/ma_test3.c | 11 +- storage/maria/ma_update.c | 2 - storage/maria/ma_write.c | 7 - storage/maria/maria_chk.c | 1 - storage/maria/maria_def.h | 20 - storage/maria/maria_log.c | 848 ------------------------------------------ storage/maria/maria_pack.c | 10 +- 22 files changed, 13 insertions(+), 1196 deletions(-) delete mode 100644 storage/maria/ma_log.c delete mode 100644 storage/maria/maria_log.c diff --git a/include/maria.h b/include/maria.h index c7586143fe7..72fb1cf2dff 100644 --- a/include/maria.h +++ b/include/maria.h @@ -372,8 +372,6 @@ int maria_repair_parallel(HA_CHECK *param, register MARIA_HA *info, const char *name, int rep_quick); int maria_change_to_newfile(const char *filename, const char *old_ext, const char *new_ext, uint raid_chunks, myf myflags); -int maria_lock_file(HA_CHECK *param, File file, my_off_t start, int lock_type, - const char *filetype, const char *filename); void maria_lock_memory(HA_CHECK *param); int maria_update_state_info(HA_CHECK *param, MARIA_HA *info, uint update); void maria_update_key_parts(MARIA_KEYDEF *keyinfo, ulong *rec_per_key_part, diff --git a/storage/maria/Makefile.am b/storage/maria/Makefile.am index bf22428c18f..4132a4208a2 100644 --- a/storage/maria/Makefile.am +++ b/storage/maria/Makefile.am @@ -24,9 +24,8 @@ LDADD = @CLIENT_EXTRA_LDFLAGS@ libmaria.a \ $(top_builddir)/dbug/libdbug.a \ $(top_builddir)/strings/libmystrings.a @ZLIB_LIBS@ pkglib_LIBRARIES = libmaria.a -bin_PROGRAMS = maria_chk maria_log maria_pack maria_ftdump +bin_PROGRAMS = maria_chk maria_pack maria_ftdump maria_chk_DEPENDENCIES= $(LIBRARIES) -maria_log_DEPENDENCIES= $(LIBRARIES) maria_pack_DEPENDENCIES=$(LIBRARIES) noinst_PROGRAMS = ma_test1 ma_test2 ma_test3 ma_rt_test ma_sp_test noinst_HEADERS = maria_def.h ma_rt_index.h ma_rt_key.h ma_rt_mbr.h ma_sp_defs.h ma_fulltext.h ma_ftdefs.h ma_ft_test1.h ma_ft_eval.h @@ -47,7 +46,7 @@ libmaria_a_SOURCES = ma_init.c ma_open.c ma_extra.c ma_info.c ma_rkey.c \ ma_delete.c \ ma_rprev.c ma_rfirst.c ma_rlast.c ma_rsame.c \ ma_rsamepos.c ma_panic.c ma_close.c ma_create.c\ - ma_range.c ma_dbug.c ma_checksum.c ma_log.c \ + ma_range.c ma_dbug.c ma_checksum.c \ ma_changed.c ma_static.c ma_delete_all.c \ ma_delete_table.c ma_rename.c ma_check.c \ ma_keycache.c ma_preload.c ma_ft_parser.c \ diff --git a/storage/maria/ma_check.c b/storage/maria/ma_check.c index 4680caed2da..4af0f955b8b 100644 --- a/storage/maria/ma_check.c +++ b/storage/maria/ma_check.c @@ -1928,25 +1928,7 @@ int maria_change_to_newfile(const char * filename, const char * old_ext, } /* maria_change_to_newfile */ - /* Locks a whole file */ - /* Gives an error-message if file can't be locked */ - -int maria_lock_file(HA_CHECK *param, File file, my_off_t start, int lock_type, - const char *filetype, const char *filename) -{ - if (my_lock(file,lock_type,start,F_TO_EOF, - param->testflag & T_WAIT_FOREVER ? MYF(MY_SEEK_NOT_DONE) : - MYF(MY_SEEK_NOT_DONE | MY_DONT_WAIT))) - { - _ma_check_print_error(param," %d when locking %s '%s'",my_errno,filetype,filename); - param->error_printed=2; /* Don't give that data is crashed */ - return 1; - } - return 0; -} /* maria_lock_file */ - - - /* Copy a block between two files */ +/* Copy a block between two files */ int maria_filecopy(HA_CHECK *param, File to,File from,my_off_t start, my_off_t length, const char *type) diff --git a/storage/maria/ma_close.c b/storage/maria/ma_close.c index f3a1b2ba261..5b940eaf4c3 100644 --- a/storage/maria/ma_close.c +++ b/storage/maria/ma_close.c @@ -113,7 +113,6 @@ int maria_close(register MARIA_HA *info) if (info->dfile >= 0 && my_close(info->dfile,MYF(0))) error = my_errno; - maria_log_command(MARIA_LOG_CLOSE,info,NULL,0,error); my_free((gptr) info,MYF(0)); if (error) diff --git a/storage/maria/ma_delete.c b/storage/maria/ma_delete.c index 9e06b633171..d1ad9edbed5 100644 --- a/storage/maria/ma_delete.c +++ b/storage/maria/ma_delete.c @@ -101,7 +101,6 @@ int maria_delete(MARIA_HA *info,const byte *record) info->state->records--; mi_sizestore(lastpos,info->lastpos); - maria_log_command(MARIA_LOG_DELETE,info,(byte*) lastpos,sizeof(lastpos),0); VOID(_ma_writeinfo(info,WRITEINFO_UPDATE_KEYFILE)); allow_break(); /* Allow SIGHUP & SIGINT */ if (info->invalidator != 0) @@ -115,7 +114,6 @@ int maria_delete(MARIA_HA *info,const byte *record) err: save_errno=my_errno; mi_sizestore(lastpos,info->lastpos); - maria_log_command(MARIA_LOG_DELETE,info,(byte*) lastpos, sizeof(lastpos),0); if (save_errno != HA_ERR_RECORD_CHANGED) { maria_print_error(info->s, HA_ERR_CRASHED); diff --git a/storage/maria/ma_delete_all.c b/storage/maria/ma_delete_all.c index d71e4d7dce7..b16d82ed9f7 100644 --- a/storage/maria/ma_delete_all.c +++ b/storage/maria/ma_delete_all.c @@ -22,7 +22,6 @@ int maria_delete_all_rows(MARIA_HA *info) { uint i; - char buf[22]; MARIA_SHARE *share=info->s; MARIA_STATE_INFO *state=&share->state; DBUG_ENTER("maria_delete_all_rows"); @@ -49,7 +48,6 @@ int maria_delete_all_rows(MARIA_HA *info) for (i=0 ; i < share->base.keys ; i++) state->key_root[i]= HA_OFFSET_ERROR; - maria_log_command(MARIA_LOG_DELETE_ALL,info,(byte*) 0,0,0); /* If we are using delayed keys or if the user has done changes to the tables since it was locked then there may be key blocks in the key cache diff --git a/storage/maria/ma_dynrec.c b/storage/maria/ma_dynrec.c index 0fbf28b949a..18efc0adbd0 100644 --- a/storage/maria/ma_dynrec.c +++ b/storage/maria/ma_dynrec.c @@ -1529,12 +1529,6 @@ int _ma_read_rnd_dynamic_record(MARIA_HA *info, byte *buf, if (info->lock_type == F_UNLCK) { #ifndef UNSAFE_LOCKING - if (share->tot_locks == 0) - { - if (my_lock(share->kfile,F_RDLCK,0L,F_TO_EOF, - MYF(MY_SEEK_NOT_DONE) | info->lock_wait)) - DBUG_RETURN(my_errno); - } #else info->tmp_lock_type=F_RDLCK; #endif diff --git a/storage/maria/ma_extra.c b/storage/maria/ma_extra.c index bd5d4280c9d..06a36cba238 100644 --- a/storage/maria/ma_extra.c +++ b/storage/maria/ma_extra.c @@ -397,7 +397,6 @@ int maria_extra(MARIA_HA *info, enum ha_extra_function function, void *extra_arg { char tmp[1]; tmp[0]=function; - maria_log_command(MARIA_LOG_EXTRA,info,(byte*) tmp,1,error); } DBUG_RETURN(error); } /* maria_extra */ diff --git a/storage/maria/ma_init.c b/storage/maria/ma_init.c index fc526c6ca3a..318bbe341e4 100644 --- a/storage/maria/ma_init.c +++ b/storage/maria/ma_init.c @@ -52,7 +52,6 @@ void maria_end(void) if (maria_inited) { maria_inited= 0; - VOID(maria_logging(0)); /* Close log if neaded */ ft_free_stopwords(); pthread_mutex_destroy(&THR_LOCK_maria); } diff --git a/storage/maria/ma_locking.c b/storage/maria/ma_locking.c index 697747021f8..adb4b03bebe 100644 --- a/storage/maria/ma_locking.c +++ b/storage/maria/ma_locking.c @@ -30,7 +30,6 @@ int maria_lock_database(MARIA_HA *info, int lock_type) int error; uint count; MARIA_SHARE *share=info->s; - uint flag; DBUG_ENTER("maria_lock_database"); DBUG_PRINT("enter",("lock_type: %d old lock %d r_locks: %u w_locks: %u " "global_changed: %d open_count: %u name: '%s'", @@ -49,7 +48,7 @@ int maria_lock_database(MARIA_HA *info, int lock_type) DBUG_RETURN(0); } - flag=error=0; + error=0; pthread_mutex_lock(&share->intern_lock); if (share->kfile >= 0) /* May only be false on windows */ { @@ -117,23 +116,6 @@ int maria_lock_database(MARIA_HA *info, int lock_type) maria_mark_crashed(info); } } - if (info->lock_type != F_EXTRA_LCK) - { - if (share->r_locks) - { /* Only read locks left */ - flag=1; - if (my_lock(share->kfile,F_RDLCK,0L,F_TO_EOF, - MYF(MY_WME | MY_SEEK_NOT_DONE)) && !error) - error=my_errno; - } - else if (!share->w_locks) - { /* No more locks */ - flag=1; - if (my_lock(share->kfile,F_UNLCK,0L,F_TO_EOF, - MYF(MY_WME | MY_SEEK_NOT_DONE)) && !error) - error=my_errno; - } - } } info->opt_flag&= ~(READ_CACHE_USED | WRITE_CACHE_USED); info->lock_type= F_UNLCK; @@ -147,16 +129,6 @@ int maria_lock_database(MARIA_HA *info, int lock_type) mysqld does not turn write locks to read locks, so we're never here in mysqld. */ - if (share->w_locks == 1) - { - flag=1; - if (my_lock(share->kfile,lock_type,0L,F_TO_EOF, - MYF(MY_SEEK_NOT_DONE))) - { - error=my_errno; - break; - } - } share->w_locks--; share->r_locks++; info->lock_type=lock_type; @@ -164,18 +136,9 @@ int maria_lock_database(MARIA_HA *info, int lock_type) } if (!share->r_locks && !share->w_locks) { - flag=1; - if (my_lock(share->kfile,lock_type,0L,F_TO_EOF, - info->lock_wait | MY_SEEK_NOT_DONE)) - { - error=my_errno; - break; - } if (_ma_state_info_read_dsk(share->kfile, &share->state, 1)) { error=my_errno; - VOID(my_lock(share->kfile,F_UNLCK,0L,F_TO_EOF,MYF(MY_SEEK_NOT_DONE))); - my_errno=error; break; } } @@ -189,13 +152,6 @@ int maria_lock_database(MARIA_HA *info, int lock_type) { /* Change READONLY to RW */ if (share->r_locks == 1) { - flag=1; - if (my_lock(share->kfile,lock_type,0L,F_TO_EOF, - MYF(info->lock_wait | MY_SEEK_NOT_DONE))) - { - error=my_errno; - break; - } share->r_locks--; share->w_locks++; info->lock_type=lock_type; @@ -206,21 +162,11 @@ int maria_lock_database(MARIA_HA *info, int lock_type) { if (!share->w_locks) { - flag=1; - if (my_lock(share->kfile,lock_type,0L,F_TO_EOF, - info->lock_wait | MY_SEEK_NOT_DONE)) - { - error=my_errno; - break; - } if (!share->r_locks) { if (_ma_state_info_read_dsk(share->kfile, &share->state, 1)) { error=my_errno; - VOID(my_lock(share->kfile,F_UNLCK,0L,F_TO_EOF, - info->lock_wait | MY_SEEK_NOT_DONE)); - my_errno=error; break; } } @@ -238,11 +184,6 @@ int maria_lock_database(MARIA_HA *info, int lock_type) } } pthread_mutex_unlock(&share->intern_lock); -#if defined(FULL_LOG) || defined(_lint) - lock_type|=(int) (flag << 8); /* Set bit to set if real lock */ - maria_log_command(MARIA_LOG_LOCK,info,(byte*) &lock_type,sizeof(lock_type), - error); -#endif DBUG_RETURN(error); } /* maria_lock_database */ @@ -381,14 +322,9 @@ int _ma_readinfo(register MARIA_HA *info, int lock_type, int check_keybuffer) MARIA_SHARE *share=info->s; if (!share->tot_locks) { - if (my_lock(share->kfile,lock_type,0L,F_TO_EOF, - info->lock_wait | MY_SEEK_NOT_DONE)) - DBUG_RETURN(1); if (_ma_state_info_read_dsk(share->kfile, &share->state, 1)) { int error=my_errno ? my_errno : -1; - VOID(my_lock(share->kfile,F_UNLCK,0L,F_TO_EOF, - MYF(MY_SEEK_NOT_DONE))); my_errno=error; DBUG_RETURN(1); } @@ -438,10 +374,6 @@ int _ma_writeinfo(register MARIA_HA *info, uint operation) } #endif } - if (!(operation & WRITEINFO_NO_UNLOCK) && - my_lock(share->kfile,F_UNLCK,0L,F_TO_EOF, - MYF(MY_WME | MY_SEEK_NOT_DONE)) && !error) - DBUG_RETURN(1); my_errno=olderror; } else if (operation) diff --git a/storage/maria/ma_log.c b/storage/maria/ma_log.c deleted file mode 100644 index 7c32c1068cb..00000000000 --- a/storage/maria/ma_log.c +++ /dev/null @@ -1,164 +0,0 @@ -/* Copyright (C) 2006 MySQL AB & MySQL Finland AB & TCX DataKonsult AB - - 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; either version 2 of the License, or - (at your option) any later version. - - 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ - -/* - Logging of MARIA commands and records on logfile for debugging - The log can be examined with help of the marialog command. -*/ - -#include "maria_def.h" -#if defined(MSDOS) || defined(__WIN__) -#include -#ifndef __WIN__ -#include -#endif -#endif -#ifdef VMS -#include -#endif - -#undef GETPID /* For HPUX */ -#ifdef THREAD -#define GETPID() (log_type == 1 ? (long) maria_pid : (long) my_thread_id()); -#else -#define GETPID() maria_pid -#endif - - /* Activate logging if flag is 1 and reset logging if flag is 0 */ - -static int log_type=0; -ulong maria_pid=0; - -int maria_logging(int activate_log) -{ - int error=0; - char buff[FN_REFLEN]; - DBUG_ENTER("maria_logging"); - - log_type=activate_log; - if (activate_log) - { - if (!maria_pid) - maria_pid=(ulong) getpid(); - if (maria_log_file < 0) - { - if ((maria_log_file = my_create(fn_format(buff,maria_log_filename, - "",".log",4), - 0,(O_RDWR | O_BINARY | O_APPEND),MYF(0))) - < 0) - DBUG_RETURN(my_errno); - } - } - else if (maria_log_file >= 0) - { - error=my_close(maria_log_file,MYF(0)) ? my_errno : 0 ; - maria_log_file= -1; - } - DBUG_RETURN(error); -} - - - /* Logging of records and commands on logfile */ - /* All logs starts with command(1) dfile(2) process(4) result(2) */ - -void _ma_log(enum maria_log_commands command, MARIA_HA *info, - const byte *buffert, uint length) -{ - char buff[11]; - int error,old_errno; - ulong pid=(ulong) GETPID(); - old_errno=my_errno; - bzero(buff,sizeof(buff)); - buff[0]=(char) command; - mi_int2store(buff+1,info->dfile); - mi_int4store(buff+3,pid); - mi_int2store(buff+9,length); - - pthread_mutex_lock(&THR_LOCK_maria); - error=my_lock(maria_log_file,F_WRLCK,0L,F_TO_EOF,MYF(MY_SEEK_NOT_DONE)); - VOID(my_write(maria_log_file,buff,sizeof(buff),MYF(0))); - VOID(my_write(maria_log_file,buffert,length,MYF(0))); - if (!error) - error=my_lock(maria_log_file,F_UNLCK,0L,F_TO_EOF,MYF(MY_SEEK_NOT_DONE)); - pthread_mutex_unlock(&THR_LOCK_maria); - my_errno=old_errno; -} - - -void _ma_log_command(enum maria_log_commands command, MARIA_HA *info, - const byte *buffert, uint length, int result) -{ - char buff[9]; - int error,old_errno; - ulong pid=(ulong) GETPID(); - - old_errno=my_errno; - buff[0]=(char) command; - mi_int2store(buff+1,info->dfile); - mi_int4store(buff+3,pid); - mi_int2store(buff+7,result); - pthread_mutex_lock(&THR_LOCK_maria); - error=my_lock(maria_log_file,F_WRLCK,0L,F_TO_EOF,MYF(MY_SEEK_NOT_DONE)); - VOID(my_write(maria_log_file,buff,sizeof(buff),MYF(0))); - if (buffert) - VOID(my_write(maria_log_file,buffert,length,MYF(0))); - if (!error) - error=my_lock(maria_log_file,F_UNLCK,0L,F_TO_EOF,MYF(MY_SEEK_NOT_DONE)); - pthread_mutex_unlock(&THR_LOCK_maria); - my_errno=old_errno; -} - - -void _ma_log_record(enum maria_log_commands command, MARIA_HA *info, - const byte *record, my_off_t filepos, int result) -{ - char buff[21],*pos; - int error,old_errno; - uint length; - ulong pid=(ulong) GETPID(); - - old_errno=my_errno; - if (!info->s->base.blobs) - length=info->s->base.reclength; - else - length=info->s->base.reclength+ _ma_calc_total_blob_length(info,record); - buff[0]=(char) command; - mi_int2store(buff+1,info->dfile); - mi_int4store(buff+3,pid); - mi_int2store(buff+7,result); - mi_sizestore(buff+9,filepos); - mi_int4store(buff+17,length); - pthread_mutex_lock(&THR_LOCK_maria); - error=my_lock(maria_log_file,F_WRLCK,0L,F_TO_EOF,MYF(MY_SEEK_NOT_DONE)); - VOID(my_write(maria_log_file,buff,sizeof(buff),MYF(0))); - VOID(my_write(maria_log_file,(byte*) record,info->s->base.reclength,MYF(0))); - if (info->s->base.blobs) - { - MARIA_BLOB *blob,*end; - - for (end=info->blobs+info->s->base.blobs, blob= info->blobs; - blob != end ; - blob++) - { - memcpy_fixed(&pos,record+blob->offset+blob->pack_length,sizeof(char*)); - VOID(my_write(maria_log_file,pos,blob->length,MYF(0))); - } - } - if (!error) - error=my_lock(maria_log_file,F_UNLCK,0L,F_TO_EOF,MYF(MY_SEEK_NOT_DONE)); - pthread_mutex_unlock(&THR_LOCK_maria); - my_errno=old_errno; -} diff --git a/storage/maria/ma_open.c b/storage/maria/ma_open.c index e1f1088c6d1..c695cbb1a44 100644 --- a/storage/maria/ma_open.c +++ b/storage/maria/ma_open.c @@ -75,7 +75,7 @@ MARIA_HA *_ma_test_if_reopen(char *filename) MARIA_HA *maria_open(const char *name, int mode, uint open_flags) { - int lock_error,kfile,open_mode,save_errno,have_rtree=0; + int kfile,open_mode,save_errno,have_rtree=0; uint i,j,len,errpos,head_length,base_pos,offset,info_length,keys, key_parts,unique_key_parts,fulltext_keys,uniques; char name_buff[FN_REFLEN], org_name[FN_REFLEN], index_name[FN_REFLEN], @@ -90,7 +90,6 @@ MARIA_HA *maria_open(const char *name, int mode, uint open_flags) LINT_INIT(m_info); kfile= -1; - lock_error=1; errpos=0; head_length=sizeof(share_buff.state.header); bzero((byte*) &info,sizeof(info)); @@ -176,14 +175,6 @@ MARIA_HA *maria_open(const char *name, int mode, uint open_flags) errpos=2; VOID(my_seek(kfile,0L,MY_SEEK_SET,MYF(0))); - if (!(open_flags & HA_OPEN_TMP_TABLE)) - { - if ((lock_error=my_lock(kfile,F_RDLCK,0L,F_TO_EOF, - MYF(open_flags & HA_OPEN_WAIT_IF_LOCKED ? - 0 : MY_DONT_WAIT))) && - !(open_flags & HA_OPEN_IGNORE_IF_LOCKED)) - goto err; - } errpos=3; if (my_read(kfile,disk_cache,info_length,MYF(MY_NABP))) { @@ -451,12 +442,6 @@ MARIA_HA *maria_open(const char *name, int mode, uint open_flags) } share->rec[i].type=(int) FIELD_LAST; /* End marker */ - if (! lock_error) - { - VOID(my_lock(kfile,F_UNLCK,0L,F_TO_EOF,MYF(MY_SEEK_NOT_DONE))); - lock_error=1; /* Database unlocked */ - } - if (_ma_open_datafile(&info, share, -1)) goto err; errpos=5; @@ -613,11 +598,6 @@ MARIA_HA *maria_open(const char *name, int mode, uint open_flags) maria_open_list=list_add(maria_open_list,&m_info->open_list); pthread_mutex_unlock(&THR_LOCK_maria); - if (maria_log_file >= 0) - { - intern_filename(name_buff,share->index_file_name); - _ma_log(MARIA_LOG_OPEN,m_info,name_buff,(uint) strlen(name_buff)); - } DBUG_RETURN(m_info); err: @@ -639,8 +619,6 @@ err: my_free((gptr) share,MYF(0)); /* fall through */ case 3: - if (! lock_error) - VOID(my_lock(kfile, F_UNLCK, 0L, F_TO_EOF, MYF(MY_SEEK_NOT_DONE))); /* fall through */ case 2: my_afree((gptr) disk_cache); diff --git a/storage/maria/ma_static.c b/storage/maria/ma_static.c index 7e972ecd850..511c5507aaf 100644 --- a/storage/maria/ma_static.c +++ b/storage/maria/ma_static.c @@ -28,8 +28,6 @@ uchar NEAR maria_file_magic[]= { (uchar) 254, (uchar) 254,'\007', '\001', }; uchar NEAR maria_pack_file_magic[]= { (uchar) 254, (uchar) 254,'\010', '\002', }; -my_string maria_log_filename=(char*) "maria.log"; -File maria_log_file= -1; uint maria_quick_table_bits=9; ulong maria_block_size= MARIA_KEY_BLOCK_LENGTH; my_bool maria_flush=0, maria_delay_key_write=0, maria_single_user=0; diff --git a/storage/maria/ma_statrec.c b/storage/maria/ma_statrec.c index c9614de1c72..0aef24f40a9 100644 --- a/storage/maria/ma_statrec.c +++ b/storage/maria/ma_statrec.c @@ -241,9 +241,6 @@ int _ma_read_rnd_static_record(MARIA_HA *info, byte *buf, if ((! cache_read || share->base.reclength > cache_length) && share->tot_locks == 0) { /* record not in cache */ - if (my_lock(share->kfile,F_RDLCK,0L,F_TO_EOF, - MYF(MY_SEEK_NOT_DONE) | info->lock_wait)) - DBUG_RETURN(my_errno); locked=1; } #else diff --git a/storage/maria/ma_test2.c b/storage/maria/ma_test2.c index 06cfaf7cc5e..7dd1ffeb7fd 100644 --- a/storage/maria/ma_test2.c +++ b/storage/maria/ma_test2.c @@ -44,7 +44,7 @@ static void copy_key(struct st_maria_info *info,uint inx, static int verbose=0,testflag=0, first_key=0,async_io=0,key_cacheing=0,write_cacheing=0,locking=0, - rec_pointer_size=0,pack_fields=1,use_log=0,silent=0, + rec_pointer_size=0,pack_fields=1,silent=0, opt_quick_mode=0; static int pack_seg=HA_SPACE_PACK,pack_type=HA_PACK_KEY,remove_count=-1, create_flag=0; @@ -209,8 +209,6 @@ int main(int argc, char *argv[]) 0,(MARIA_UNIQUEDEF*) 0, &create_info,create_flag)) goto err; - if (use_log) - maria_logging(1); if (!(file=maria_open(filename,2,HA_OPEN_ABORT_IF_LOCKED))) goto err; if (!silent) @@ -894,9 +892,6 @@ static void get_options(int argc, char **argv) if (*++pos) srand(atoi(pos)); break; - case 'l': - use_log=1; - break; case 'L': locking=1; break; diff --git a/storage/maria/ma_test3.c b/storage/maria/ma_test3.c index bfb2c93a95f..d2fe8d90cf6 100644 --- a/storage/maria/ma_test3.c +++ b/storage/maria/ma_test3.c @@ -41,7 +41,7 @@ const char *filename= "test3"; -uint tests=10,forks=10,key_cacheing=0,use_log=0; +uint tests=10,forks=10,key_cacheing=0; static void get_options(int argc, char *argv[]); void start_test(int id); @@ -127,9 +127,6 @@ static void get_options(int argc, char **argv) while (--argc >0 && *(pos = *(++argv)) == '-' ) { switch(*++pos) { - case 'l': - use_log=1; - break; case 'f': forks=atoi(++pos); break; @@ -140,7 +137,7 @@ static void get_options(int argc, char **argv) key_cacheing=1; break; case 'A': /* All flags */ - use_log=key_cacheing=1; + key_cacheing=1; break; case '?': case 'I': @@ -169,8 +166,6 @@ void start_test(int id) MARIA_INFO isam_info; MARIA_HA *file,*file1,*file2=0,*lock; - if (use_log) - maria_logging(1); if (!(file1=maria_open(filename,O_RDWR,HA_OPEN_WAIT_IF_LOCKED)) || !(file2=maria_open(filename,O_RDWR,HA_OPEN_WAIT_IF_LOCKED))) { @@ -214,8 +209,6 @@ void start_test(int id) maria_close(file1); maria_close(file2); - if (use_log) - maria_logging(0); if (error) { printf("%2d: Aborted\n",id); fflush(stdout); diff --git a/storage/maria/ma_update.c b/storage/maria/ma_update.c index d3c71cef9b4..3f1ba4b1fac 100644 --- a/storage/maria/ma_update.c +++ b/storage/maria/ma_update.c @@ -170,7 +170,6 @@ int maria_update(register MARIA_HA *info, const byte *oldrec, byte *newrec) info->update= (HA_STATE_CHANGED | HA_STATE_ROW_CHANGED | HA_STATE_AKTIV | key_changed); - maria_log_record(MARIA_LOG_UPDATE,info,newrec,info->lastpos,0); VOID(_ma_writeinfo(info,key_changed ? WRITEINFO_UPDATE_KEYFILE : 0)); allow_break(); /* Allow SIGHUP & SIGINT */ if (info->invalidator != 0) @@ -220,7 +219,6 @@ err: key_changed); err_end: - maria_log_record(MARIA_LOG_UPDATE,info,newrec,info->lastpos,my_errno); VOID(_ma_writeinfo(info,WRITEINFO_UPDATE_KEYFILE)); allow_break(); /* Allow SIGHUP & SIGINT */ if (save_errno == HA_ERR_KEY_NOT_FOUND) diff --git a/storage/maria/ma_write.c b/storage/maria/ma_write.c index 313a362f6d6..5a1a540b88b 100644 --- a/storage/maria/ma_write.c +++ b/storage/maria/ma_write.c @@ -62,11 +62,6 @@ int maria_write(MARIA_HA *info, byte *record) if (_ma_readinfo(info,F_WRLCK,1)) DBUG_RETURN(my_errno); dont_break(); /* Dont allow SIGHUP or SIGINT */ -#if !defined(NO_LOCKING) && defined(USE_RECORD_LOCK) - if (!info->locked && my_lock(info->dfile,F_WRLCK,0L,F_TO_EOF, - MYF(MY_SEEK_NOT_DONE) | info->lock_wait)) - goto err; -#endif filepos= ((share->state.dellink != HA_OFFSET_ERROR && !info->append_insert_at_end) ? share->state.dellink : @@ -155,7 +150,6 @@ int maria_write(MARIA_HA *info, byte *record) HA_STATE_ROW_CHANGED); info->state->records++; info->lastpos=filepos; - maria_log_record(MARIA_LOG_WRITE,info,record,filepos,0); VOID(_ma_writeinfo(info, WRITEINFO_UPDATE_KEYFILE)); if (info->invalidator != 0) { @@ -220,7 +214,6 @@ err: my_errno=save_errno; err2: save_errno=my_errno; - maria_log_record(MARIA_LOG_WRITE,info,record,filepos,my_errno); VOID(_ma_writeinfo(info,WRITEINFO_UPDATE_KEYFILE)); allow_break(); /* Allow SIGHUP & SIGINT */ DBUG_RETURN(my_errno=save_errno); diff --git a/storage/maria/maria_chk.c b/storage/maria/maria_chk.c index de76344a800..7f48a0805da 100644 --- a/storage/maria/maria_chk.c +++ b/storage/maria/maria_chk.c @@ -1158,7 +1158,6 @@ static int mariachk(HA_CHECK *param, my_string filename) (state_updated ? UPDATE_STAT : 0) | ((param->testflag & T_SORT_RECORDS) ? UPDATE_SORT : 0))); - VOID(maria_lock_file(param, share->kfile,0L,F_UNLCK,"indexfile",filename)); info->update&= ~HA_STATE_CHANGED; } maria_lock_database(info, F_UNLCK); diff --git a/storage/maria/maria_def.h b/storage/maria/maria_def.h index e3f0219a663..9bab65126c7 100644 --- a/storage/maria/maria_def.h +++ b/storage/maria/maria_def.h @@ -427,8 +427,6 @@ extern LIST *maria_open_list; extern uchar NEAR maria_file_magic[], NEAR maria_pack_file_magic[]; extern uint NEAR maria_read_vec[], NEAR maria_readnext_vec[]; extern uint maria_quick_table_bits; -extern File maria_log_file; -extern ulong maria_pid; /* This is used by _ma_calc_xxx_key_length och _ma_store_key */ @@ -648,16 +646,6 @@ typedef struct st_maria_block_info #define SORT_BUFFER_INIT (2048L*1024L-MALLOC_OVERHEAD) #define MIN_SORT_BUFFER (4096-MALLOC_OVERHEAD) -enum maria_log_commands -{ - MARIA_LOG_OPEN, MARIA_LOG_WRITE, MARIA_LOG_UPDATE, MARIA_LOG_DELETE, - MARIA_LOG_CLOSE, MARIA_LOG_EXTRA, MARIA_LOG_LOCK, MARIA_LOG_DELETE_ALL -}; - -#define maria_log(a,b,c,d) if (maria_log_file >= 0) _ma_log(a,b,c,d) -#define maria_log_command(a,b,c,d,e) if (maria_log_file >= 0) _ma_log_command(a,b,c,d,e) -#define maria_log_record(a,b,c,d,e) if (maria_log_file >= 0) _ma_log_record(a,b,c,d,e) - #define fast_ma_writeinfo(INFO) if (!(INFO)->s->tot_locks) (void) _ma_writeinfo((INFO),0) #define fast_ma_readinfo(INFO) ((INFO)->lock_type == F_UNLCK) && _ma_readinfo((INFO),F_RDLCK,1) @@ -666,14 +654,6 @@ extern uint _ma_rec_pack(MARIA_HA *info, byte *to, const byte *from); extern uint _ma_pack_get_block_info(MARIA_HA *, MARIA_BLOCK_INFO *, File, my_off_t); extern void _ma_store_blob_length(byte *pos, uint pack_length, uint length); -extern void _ma_log(enum maria_log_commands command, MARIA_HA *info, - const byte *buffert, uint length); -extern void _ma_log_command(enum maria_log_commands command, - MARIA_HA *info, const byte *buffert, - uint length, int result); -extern void _ma_log_record(enum maria_log_commands command, MARIA_HA *info, - const byte *record, my_off_t filepos, - int result); extern void _ma_report_error(int errcode, const char *file_name); extern my_bool _ma_memmap_file(MARIA_HA *info); extern void _ma_unmap_file(MARIA_HA *info); diff --git a/storage/maria/maria_log.c b/storage/maria/maria_log.c deleted file mode 100644 index 72a4d7e89d5..00000000000 --- a/storage/maria/maria_log.c +++ /dev/null @@ -1,848 +0,0 @@ -/* Copyright (C) 2006 MySQL AB & MySQL Finland AB & TCX DataKonsult AB - - 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; either version 2 of the License, or - (at your option) any later version. - - 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ - -/* write whats in isam.log */ - -#ifndef USE_MY_FUNC -#define USE_MY_FUNC -#endif - -#include "maria_def.h" -#include -#include -#ifdef HAVE_GETRUSAGE -#include -#endif - -#define FILENAME(A) (A ? A->show_name : "Unknown") - -struct file_info { - long process; - int filenr,id; - uint rnd; - my_string name,show_name,record; - MARIA_HA *isam; - bool closed,used; - ulong accessed; -}; - -struct test_if_open_param { - my_string name; - int max_id; -}; - -struct st_access_param -{ - ulong min_accessed; - struct file_info *found; -}; - -#define NO_FILEPOS (ulong) ~0L - -extern int main(int argc,char * *argv); -static void get_options(int *argc,char ***argv); -static int examine_log(my_string file_name,char **table_names); -static int read_string(IO_CACHE *file,gptr *to,uint length); -static int file_info_compare(void *cmp_arg, void *a,void *b); -static int test_if_open(struct file_info *key,element_count count, - struct test_if_open_param *param); -static void fix_blob_pointers(MARIA_HA *isam,byte *record); -static int test_when_accessed(struct file_info *key,element_count count, - struct st_access_param *access_param); -static void file_info_free(struct file_info *info); -static int close_some_file(TREE *tree); -static int reopen_closed_file(TREE *tree,struct file_info *file_info); -static int find_record_with_key(struct file_info *file_info,byte *record); -static void printf_log(const char *str,...); -static bool cmp_filename(struct file_info *file_info,my_string name); - -static uint verbose=0,update=0,test_info=0,max_files=0,re_open_count=0, - recover=0,prefix_remove=0,opt_processes=0; -static my_string log_filename=0,filepath=0,write_filename=0,record_pos_file=0; -static ulong com_count[10][3],number_of_commands=(ulong) ~0L, - isamlog_process; -static my_off_t isamlog_filepos,start_offset=0,record_pos= HA_OFFSET_ERROR; -static const char *command_name[]= -{"open","write","update","delete","close","extra","lock","re-open", - "delete-all", NullS}; - - -int main(int argc, char **argv) -{ - int error,i,first; - ulong total_count,total_error,total_recover; - MY_INIT(argv[0]); - - log_filename=maria_log_filename; - get_options(&argc,&argv); - maria_init(); - - /* Number of MARIA files we can have open at one time */ - max_files= (my_set_max_open_files(min(max_files,8))-6)/2; - if (update) - printf("Trying to %s MARIA files according to log '%s'\n", - (recover ? "recover" : "update"),log_filename); - error= examine_log(log_filename,argv); - if (update && ! error) - puts("Tables updated successfully"); - total_count=total_error=total_recover=0; - for (i=first=0 ; command_name[i] ; i++) - { - if (com_count[i][0]) - { - if (!first++) - { - if (verbose || update) - puts(""); - puts("Commands Used count Errors Recover errors"); - } - printf("%-12s%9ld%10ld%17ld\n",command_name[i],com_count[i][0], - com_count[i][1],com_count[i][2]); - total_count+=com_count[i][0]; - total_error+=com_count[i][1]; - total_recover+=com_count[i][2]; - } - } - if (total_count) - printf("%-12s%9ld%10ld%17ld\n","Total",total_count,total_error, - total_recover); - if (re_open_count) - printf("Had to do %d re-open because of too few possibly open files\n", - re_open_count); - VOID(maria_panic(HA_PANIC_CLOSE)); - my_free_open_file_info(); - maria_end(); - my_end(test_info ? MY_CHECK_ERROR | MY_GIVE_INFO : MY_CHECK_ERROR); - exit(error); - return 0; /* No compiler warning */ -} /* main */ - - -static void get_options(register int *argc, register char ***argv) -{ - int help,version; - const char *pos,*usage; - char option; - - help=0; - usage="Usage: %s [-?iruvDIV] [-c #] [-f #] [-F filepath/] [-o #] [-R file recordpos] [-w write_file] [log-filename [table ...]] \n"; - pos=""; - - while (--*argc > 0 && *(pos = *(++*argv)) == '-' ) { - while (*++pos) - { - version=0; - switch((option=*pos)) { - case '#': - DBUG_PUSH (++pos); - pos=" "; /* Skip rest of arg */ - break; - case 'c': - if (! *++pos) - { - if (!--*argc) - goto err; - else - pos= *(++*argv); - } - number_of_commands=(ulong) atol(pos); - pos=" "; - break; - case 'u': - update=1; - break; - case 'f': - if (! *++pos) - { - if (!--*argc) - goto err; - else - pos= *(++*argv); - } - max_files=(uint) atoi(pos); - pos=" "; - break; - case 'i': - test_info=1; - break; - case 'o': - if (! *++pos) - { - if (!--*argc) - goto err; - else - pos= *(++*argv); - } - start_offset=(my_off_t) strtoll(pos,NULL,10); - pos=" "; - break; - case 'p': - if (! *++pos) - { - if (!--*argc) - goto err; - else - pos= *(++*argv); - } - prefix_remove=atoi(pos); - break; - case 'r': - update=1; - recover++; - break; - case 'P': - opt_processes=1; - break; - case 'R': - if (! *++pos) - { - if (!--*argc) - goto err; - else - pos= *(++*argv); - } - record_pos_file=(char*) pos; - if (!--*argc) - goto err; - record_pos=(my_off_t) strtoll(*(++*argv),NULL,10); - pos=" "; - break; - case 'v': - verbose++; - break; - case 'w': - if (! *++pos) - { - if (!--*argc) - goto err; - else - pos= *(++*argv); - } - write_filename=(char*) pos; - pos=" "; - break; - case 'F': - if (! *++pos) - { - if (!--*argc) - goto err; - else - pos= *(++*argv); - } - filepath= (char*) pos; - pos=" "; - break; - case 'V': - version=1; - /* Fall through */ - case 'I': - case '?': -#include - printf("%s Ver 1.4 for %s at %s\n",my_progname,SYSTEM_TYPE, - MACHINE_TYPE); - puts("By Monty, for your professional use\n"); - if (version) - break; - puts("Write info about whats in a MARIA log file."); - printf("If no file name is given %s is used\n",log_filename); - puts(""); - printf(usage,my_progname); - puts(""); - puts("Options: -? or -I \"Info\" -V \"version\" -c \"do only # commands\""); - puts(" -f \"max open files\" -F \"filepath\" -i \"extra info\""); - puts(" -o \"offset\" -p # \"remove # components from path\""); - puts(" -r \"recover\" -R \"file recordposition\""); - puts(" -u \"update\" -v \"verbose\" -w \"write file\""); - puts(" -D \"maria compiled with DBUG\" -P \"processes\""); - puts("\nOne can give a second and a third '-v' for more verbose."); - puts("Normaly one does a update (-u)."); - puts("If a recover is done all writes and all possibly updates and deletes is done\nand errors are only counted."); - puts("If one gives table names as arguments only these tables will be updated\n"); - help=1; -#include - break; - default: - printf("illegal option: \"-%c\"\n",*pos); - break; - } - } - } - if (! *argc) - { - if (help) - exit(0); - (*argv)++; - } - if (*argc >= 1) - { - log_filename=(char*) pos; - (*argc)--; - (*argv)++; - } - return; - err: - VOID(fprintf(stderr,"option \"%c\" used without or with wrong argument\n", - option)); - exit(1); -} - - -static int examine_log(my_string file_name, char **table_names) -{ - uint command,result,files_open; - ulong access_time,length; - my_off_t filepos; - int lock_command,maria_result; - char isam_file_name[FN_REFLEN],llbuff[21],llbuff2[21]; - uchar head[20]; - gptr buff; - struct test_if_open_param open_param; - IO_CACHE cache; - File file; - FILE *write_file; - enum ha_extra_function extra_command; - TREE tree; - struct file_info file_info,*curr_file_info; - DBUG_ENTER("examine_log"); - - if ((file=my_open(file_name,O_RDONLY,MYF(MY_WME))) < 0) - DBUG_RETURN(1); - write_file=0; - if (write_filename) - { - if (!(write_file=my_fopen(write_filename,O_WRONLY,MYF(MY_WME)))) - { - my_close(file,MYF(0)); - DBUG_RETURN(1); - } - } - - init_io_cache(&cache,file,0,READ_CACHE,start_offset,0,MYF(0)); - bzero((gptr) com_count,sizeof(com_count)); - init_tree(&tree,0,0,sizeof(file_info),(qsort_cmp2) file_info_compare,1, - (tree_element_free) file_info_free, NULL); - VOID(init_key_cache(maria_key_cache,KEY_CACHE_BLOCK_SIZE,KEY_CACHE_SIZE, - 0, 0)); - - files_open=0; access_time=0; - while (access_time++ != number_of_commands && - !my_b_read(&cache,(byte*) head,9)) - { - isamlog_filepos=my_b_tell(&cache)-9L; - file_info.filenr= mi_uint2korr(head+1); - isamlog_process=file_info.process=(long) mi_uint4korr(head+3); - if (!opt_processes) - file_info.process=0; - result= mi_uint2korr(head+7); - if ((curr_file_info=(struct file_info*) tree_search(&tree, &file_info, - tree.custom_arg))) - { - curr_file_info->accessed=access_time; - if (update && curr_file_info->used && curr_file_info->closed) - { - if (reopen_closed_file(&tree,curr_file_info)) - { - command=sizeof(com_count)/sizeof(com_count[0][0])/3; - result=0; - goto com_err; - } - } - } - command=(uint) head[0]; - if (command < sizeof(com_count)/sizeof(com_count[0][0])/3 && - (!table_names[0] || (curr_file_info && curr_file_info->used))) - { - com_count[command][0]++; - if (result) - com_count[command][1]++; - } - switch ((enum maria_log_commands) command) { - case MARIA_LOG_OPEN: - if (!table_names[0]) - { - com_count[command][0]--; /* Must be counted explicite */ - if (result) - com_count[command][1]--; - } - - if (curr_file_info) - printf("\nWarning: %s is opened with same process and filenumber\nMaybe you should use the -P option ?\n", - curr_file_info->show_name); - if (my_b_read(&cache,(byte*) head,2)) - goto err; - file_info.name=0; - file_info.show_name=0; - file_info.record=0; - if (read_string(&cache,(gptr*) &file_info.name, - (uint) mi_uint2korr(head))) - goto err; - { - uint i; - char *pos,*to; - - /* Fix if old DOS files to new format */ - for (pos=file_info.name; (pos=strchr(pos,'\\')) ; pos++) - *pos= '/'; - - pos=file_info.name; - for (i=0 ; i < prefix_remove ; i++) - { - char *next; - if (!(next=strchr(pos,'/'))) - break; - pos=next+1; - } - to=isam_file_name; - if (filepath) - to=convert_dirname(isam_file_name,filepath,NullS); - strmov(to,pos); - fn_ext(isam_file_name)[0]=0; /* Remove extension */ - } - open_param.name=file_info.name; - open_param.max_id=0; - VOID(tree_walk(&tree,(tree_walk_action) test_if_open,(void*) &open_param, - left_root_right)); - file_info.id=open_param.max_id+1; - /* - * In the line below +10 is added to accomodate '<' and '>' chars - * plus '\0' at the end, so that there is place for 7 digits. - * It is improbable that same table can have that many entries in - * the table cache. - * The additional space is needed for the sprintf commands two lines - * below. - */ - file_info.show_name=my_memdup(isam_file_name, - (uint) strlen(isam_file_name)+10, - MYF(MY_WME)); - if (file_info.id > 1) - sprintf(strend(file_info.show_name),"<%d>",file_info.id); - file_info.closed=1; - file_info.accessed=access_time; - file_info.used=1; - if (table_names[0]) - { - char **name; - file_info.used=0; - for (name=table_names ; *name ; name++) - { - if (!strcmp(*name,isam_file_name)) - file_info.used=1; /* Update/log only this */ - } - } - if (update && file_info.used) - { - if (files_open >= max_files) - { - if (close_some_file(&tree)) - goto com_err; - files_open--; - } - if (!(file_info.isam= maria_open(isam_file_name,O_RDWR, - HA_OPEN_WAIT_IF_LOCKED))) - goto com_err; - if (!(file_info.record=my_malloc(file_info.isam->s->base.reclength, - MYF(MY_WME)))) - goto end; - files_open++; - file_info.closed=0; - } - VOID(tree_insert(&tree, (gptr) &file_info, 0, tree.custom_arg)); - if (file_info.used) - { - if (verbose && !record_pos_file) - printf_log("%s: open -> %d",file_info.show_name, file_info.filenr); - com_count[command][0]++; - if (result) - com_count[command][1]++; - } - break; - case MARIA_LOG_CLOSE: - if (verbose && !record_pos_file && - (!table_names[0] || (curr_file_info && curr_file_info->used))) - printf_log("%s: %s -> %d",FILENAME(curr_file_info), - command_name[command],result); - if (curr_file_info) - { - if (!curr_file_info->closed) - files_open--; - VOID(tree_delete(&tree, (gptr) curr_file_info, tree.custom_arg)); - } - break; - case MARIA_LOG_EXTRA: - if (my_b_read(&cache,(byte*) head,1)) - goto err; - extra_command=(enum ha_extra_function) head[0]; - if (verbose && !record_pos_file && - (!table_names[0] || (curr_file_info && curr_file_info->used))) - printf_log("%s: %s(%d) -> %d",FILENAME(curr_file_info), - command_name[command], (int) extra_command,result); - if (update && curr_file_info && !curr_file_info->closed) - { - if (maria_extra(curr_file_info->isam, extra_command, 0) != (int) result) - { - fflush(stdout); - VOID(fprintf(stderr, - "Warning: error %d, expected %d on command %s at %s\n", - my_errno,result,command_name[command], - llstr(isamlog_filepos,llbuff))); - fflush(stderr); - } - } - break; - case MARIA_LOG_DELETE: - if (my_b_read(&cache,(byte*) head,8)) - goto err; - filepos=mi_sizekorr(head); - if (verbose && (!record_pos_file || - ((record_pos == filepos || record_pos == NO_FILEPOS) && - !cmp_filename(curr_file_info,record_pos_file))) && - (!table_names[0] || (curr_file_info && curr_file_info->used))) - printf_log("%s: %s at %ld -> %d",FILENAME(curr_file_info), - command_name[command],(long) filepos,result); - if (update && curr_file_info && !curr_file_info->closed) - { - if (maria_rrnd(curr_file_info->isam,curr_file_info->record,filepos)) - { - if (!recover) - goto com_err; - if (verbose) - printf_log("error: Didn't find row to delete with maria_rrnd"); - com_count[command][2]++; /* Mark error */ - } - maria_result=maria_delete(curr_file_info->isam,curr_file_info->record); - if ((maria_result == 0 && result) || - (maria_result && (uint) my_errno != result)) - { - if (!recover) - goto com_err; - if (maria_result) - com_count[command][2]++; /* Mark error */ - if (verbose) - printf_log("error: Got result %d from maria_delete instead of %d", - maria_result, result); - } - } - break; - case MARIA_LOG_WRITE: - case MARIA_LOG_UPDATE: - if (my_b_read(&cache,(byte*) head,12)) - goto err; - filepos=mi_sizekorr(head); - length=mi_uint4korr(head+8); - buff=0; - if (read_string(&cache,&buff,(uint) length)) - goto err; - if ((!record_pos_file || - ((record_pos == filepos || record_pos == NO_FILEPOS) && - !cmp_filename(curr_file_info,record_pos_file))) && - (!table_names[0] || (curr_file_info && curr_file_info->used))) - { - if (write_file && - (my_fwrite(write_file,buff,length,MYF(MY_WAIT_IF_FULL | MY_NABP)))) - goto end; - if (verbose) - printf_log("%s: %s at %ld, length=%ld -> %d", - FILENAME(curr_file_info), - command_name[command], filepos,length,result); - } - if (update && curr_file_info && !curr_file_info->closed) - { - if (curr_file_info->isam->s->base.blobs) - fix_blob_pointers(curr_file_info->isam,buff); - if ((enum maria_log_commands) command == MARIA_LOG_UPDATE) - { - if (maria_rrnd(curr_file_info->isam,curr_file_info->record,filepos)) - { - if (!recover) - { - result=0; - goto com_err; - } - if (verbose) - printf_log("error: Didn't find row to update with maria_rrnd"); - if (recover == 1 || result || - find_record_with_key(curr_file_info,buff)) - { - com_count[command][2]++; /* Mark error */ - break; - } - } - maria_result=maria_update(curr_file_info->isam,curr_file_info->record, - buff); - if ((maria_result == 0 && result) || - (maria_result && (uint) my_errno != result)) - { - if (!recover) - goto com_err; - if (verbose) - printf_log("error: Got result %d from maria_update instead of %d", - maria_result, result); - if (maria_result) - com_count[command][2]++; /* Mark error */ - } - } - else - { - maria_result=maria_write(curr_file_info->isam,buff); - if ((maria_result == 0 && result) || - (maria_result && (uint) my_errno != result)) - { - if (!recover) - goto com_err; - if (verbose) - printf_log("error: Got result %d from maria_write instead of %d", - maria_result, result); - if (maria_result) - com_count[command][2]++; /* Mark error */ - } - if (!recover && filepos != curr_file_info->isam->lastpos) - { - printf("error: Wrote at position: %s, should have been %s", - llstr(curr_file_info->isam->lastpos,llbuff), - llstr(filepos,llbuff2)); - goto end; - } - } - } - my_free(buff,MYF(0)); - break; - case MARIA_LOG_LOCK: - if (my_b_read(&cache,(byte*) head,sizeof(lock_command))) - goto err; - memcpy_fixed(&lock_command,head,sizeof(lock_command)); - if (verbose && !record_pos_file && - (!table_names[0] || (curr_file_info && curr_file_info->used))) - printf_log("%s: %s(%d) -> %d\n",FILENAME(curr_file_info), - command_name[command],lock_command,result); - if (update && curr_file_info && !curr_file_info->closed) - { - if (maria_lock_database(curr_file_info->isam,lock_command) != - (int) result) - goto com_err; - } - break; - case MARIA_LOG_DELETE_ALL: - if (verbose && !record_pos_file && - (!table_names[0] || (curr_file_info && curr_file_info->used))) - printf_log("%s: %s -> %d\n",FILENAME(curr_file_info), - command_name[command],result); - break; - default: - fflush(stdout); - VOID(fprintf(stderr, - "Error: found unknown command %d in logfile, aborted\n", - command)); - fflush(stderr); - goto end; - } - } - end_key_cache(maria_key_cache,1); - delete_tree(&tree); - VOID(end_io_cache(&cache)); - VOID(my_close(file,MYF(0))); - if (write_file && my_fclose(write_file,MYF(MY_WME))) - DBUG_RETURN(1); - DBUG_RETURN(0); - - err: - fflush(stdout); - VOID(fprintf(stderr,"Got error %d when reading from logfile\n",my_errno)); - fflush(stderr); - goto end; - com_err: - fflush(stdout); - VOID(fprintf(stderr,"Got error %d, expected %d on command %s at %s\n", - my_errno,result,command_name[command], - llstr(isamlog_filepos,llbuff))); - fflush(stderr); - end: - end_key_cache(maria_key_cache, 1); - delete_tree(&tree); - VOID(end_io_cache(&cache)); - VOID(my_close(file,MYF(0))); - if (write_file) - VOID(my_fclose(write_file,MYF(MY_WME))); - DBUG_RETURN(1); -} - - -static int read_string(IO_CACHE *file, register gptr *to, register uint length) -{ - DBUG_ENTER("read_string"); - - if (*to) - my_free((gptr) *to,MYF(0)); - if (!(*to= (gptr) my_malloc(length+1,MYF(MY_WME))) || - my_b_read(file,(byte*) *to,length)) - { - if (*to) - my_free(*to,MYF(0)); - *to= 0; - DBUG_RETURN(1); - } - *((char*) *to+length)= '\0'; - DBUG_RETURN (0); -} /* read_string */ - - -static int file_info_compare(void* cmp_arg __attribute__((unused)), - void *a, void *b) -{ - long lint; - - if ((lint=((struct file_info*) a)->process - - ((struct file_info*) b)->process)) - return lint < 0L ? -1 : 1; - return ((struct file_info*) a)->filenr - ((struct file_info*) b)->filenr; -} - - /* ARGSUSED */ - -static int test_if_open (struct file_info *key, - element_count count __attribute__((unused)), - struct test_if_open_param *param) -{ - if (!strcmp(key->name,param->name) && key->id > param->max_id) - param->max_id=key->id; - return 0; -} - - -static void fix_blob_pointers(MARIA_HA *info, byte *record) -{ - byte *pos; - MARIA_BLOB *blob,*end; - - pos=record+info->s->base.reclength; - for (end=info->blobs+info->s->base.blobs, blob= info->blobs; - blob != end ; - blob++) - { - memcpy_fixed(record+blob->offset+blob->pack_length,&pos,sizeof(char*)); - pos+= _ma_calc_blob_length(blob->pack_length,record+blob->offset); - } -} - - /* close the file with hasn't been accessed for the longest time */ - /* ARGSUSED */ - -static int test_when_accessed (struct file_info *key, - element_count count __attribute__((unused)), - struct st_access_param *access_param) -{ - if (key->accessed < access_param->min_accessed && ! key->closed) - { - access_param->min_accessed=key->accessed; - access_param->found=key; - } - return 0; -} - - -static void file_info_free(struct file_info *fileinfo) -{ - DBUG_ENTER("file_info_free"); - if (update) - { - if (!fileinfo->closed) - VOID(maria_close(fileinfo->isam)); - if (fileinfo->record) - my_free(fileinfo->record,MYF(0)); - } - my_free(fileinfo->name,MYF(0)); - my_free(fileinfo->show_name,MYF(0)); - DBUG_VOID_RETURN; -} - - - -static int close_some_file(TREE *tree) -{ - struct st_access_param access_param; - - access_param.min_accessed=LONG_MAX; - access_param.found=0; - - VOID(tree_walk(tree,(tree_walk_action) test_when_accessed, - (void*) &access_param,left_root_right)); - if (!access_param.found) - return 1; /* No open file that is possibly to close */ - if (maria_close(access_param.found->isam)) - return 1; - access_param.found->closed=1; - return 0; -} - - -static int reopen_closed_file(TREE *tree, struct file_info *fileinfo) -{ - char name[FN_REFLEN]; - if (close_some_file(tree)) - return 1; /* No file to close */ - strmov(name,fileinfo->show_name); - if (fileinfo->id > 1) - *strrchr(name,'<')='\0'; /* Remove "" */ - - if (!(fileinfo->isam= maria_open(name,O_RDWR,HA_OPEN_WAIT_IF_LOCKED))) - return 1; - fileinfo->closed=0; - re_open_count++; - return 0; -} - - /* Try to find record with uniq key */ - -static int find_record_with_key(struct file_info *file_info, byte *record) -{ - uint key; - MARIA_HA *info=file_info->isam; - uchar tmp_key[HA_MAX_KEY_BUFF]; - - for (key=0 ; key < info->s->base.keys ; key++) - { - if (maria_is_key_active(info->s->state.key_map, key) && - info->s->keyinfo[key].flag & HA_NOSAME) - { - VOID(_ma_make_key(info,key,tmp_key,record,0L)); - return maria_rkey(info,file_info->record,(int) key,(char*) tmp_key,0, - HA_READ_KEY_EXACT); - } - } - return 1; -} - - -static void printf_log(const char *format,...) -{ - char llbuff[21]; - va_list args; - va_start(args,format); - if (verbose > 2) - printf("%9s:",llstr(isamlog_filepos,llbuff)); - if (verbose > 1) - printf("%5ld ",isamlog_process); /* Write process number */ - (void) vprintf((char*) format,args); - putchar('\n'); - va_end(args); -} - - -static bool cmp_filename(struct file_info *file_info, my_string name) -{ - if (!file_info) - return 1; - return strcmp(file_info->name,name) ? 1 : 0; -} diff --git a/storage/maria/maria_pack.c b/storage/maria/maria_pack.c index 2a83bbb0f3f..514d8ba0bf5 100644 --- a/storage/maria/maria_pack.c +++ b/storage/maria/maria_pack.c @@ -1107,16 +1107,16 @@ static int get_statistic(PACK_MRG_INFO *mrg,HUFF_COUNTS *huff_counts) my_off_t total_count; char llbuf[32]; - DBUG_PRINT("info", ("column: %3u", count - huff_counts + 1)); + DBUG_PRINT("info", ("column: %3lu", count - huff_counts + 1)); if (verbose >= 2) - VOID(printf("column: %3u\n", count - huff_counts + 1)); + VOID(printf("column: %3lu\n", count - huff_counts + 1)); if (count->tree_buff) { - DBUG_PRINT("info", ("number of distinct values: %u", + DBUG_PRINT("info", ("number of distinct values: %lu", (count->tree_pos - count->tree_buff) / count->field_length)); if (verbose >= 2) - VOID(printf("number of distinct values: %u\n", + VOID(printf("number of distinct values: %lu\n", (count->tree_pos - count->tree_buff) / count->field_length)); } @@ -2281,7 +2281,7 @@ static my_off_t write_huff_tree(HUFF_TREE *huff_tree, uint trees) if (bits > 8 * sizeof(code)) { VOID(fflush(stdout)); - VOID(fprintf(stderr, "error: Huffman code too long: %u/%u\n", + VOID(fprintf(stderr, "error: Huffman code too long: %u/%lu\n", bits, 8 * sizeof(code))); errors++; break; -- cgit v1.2.1 From c262b880a5f74bc938d8925f4dda8ada57bc272b Mon Sep 17 00:00:00 2001 From: unknown Date: Wed, 24 May 2006 16:44:26 +0300 Subject: postmerge fix BUILD/SETUP.sh: postmerge changes configure.in: postmerge changes mysql-test/r/ps_maria.result: changed result according last MyISAM test changes --- BUILD/SETUP.sh | 1 - configure.in | 9 +++++---- mysql-test/r/ps_maria.result | 4 ++-- sql/Makefile.am | 4 ++-- sql/ha_maria.cc | 21 +++++++++++++++++++-- 5 files changed, 28 insertions(+), 11 deletions(-) diff --git a/BUILD/SETUP.sh b/BUILD/SETUP.sh index 5efa42e4619..754883bb20b 100755 --- a/BUILD/SETUP.sh +++ b/BUILD/SETUP.sh @@ -141,7 +141,6 @@ base_configs="--prefix=$prefix --enable-assembler " base_configs="$base_configs --with-extra-charsets=complex " base_configs="$base_configs --enable-thread-safe-client --with-readline " base_configs="$base_configs --with-big-tables" -base_configs="$base_configs --with-maria-storage-engine" # we need local-infile in all binaries for rpl000001 # if you need to disable local-infile in the client, write a build script diff --git a/configure.in b/configure.in index 90ca6c69587..d4905ff7c44 100644 --- a/configure.in +++ b/configure.in @@ -2252,13 +2252,14 @@ MYSQL_PLUGIN_ACTIONS(ndbcluster,[MYSQL_SETUP_NDBCLUSTER]) MYSQL_STORAGE_ENGINE(partition, partition, [Partition Support], [MySQL Partitioning Support], [max,max-no-ndb]) +MYSQL_STORAGE_ENGINE(maria,maria, [Maria Storage Engine], + [Traditional transactional MySQL tables]) +MYSQL_PLUGIN_DIRECTORY(maria, [storage/maria]) +MYSQL_PLUGIN_STATIC(maria, [libmaria.a]) +MYSQL_PLUGIN_MANDATORY(maria) dnl -- ndbcluster requires partition to be enabled MYSQL_PLUGIN_DEPENDS(ndbcluster, [partition]) -MYSQL_STORAGE_ENGINE(maria,,,,,,storage/maria,, - \$(top_builddir)/storage/maria/libmaria.a, [ - AC_CONFIG_FILES(storage/maria/Makefile) -]) MYSQL_CONFIGURE_PLUGINS([none]) diff --git a/mysql-test/r/ps_maria.result b/mysql-test/r/ps_maria.result index a5cb3050dcb..00e32cf8fce 100644 --- a/mysql-test/r/ps_maria.result +++ b/mysql-test/r/ps_maria.result @@ -1777,7 +1777,7 @@ Table Create Table t5 CREATE TABLE `t5` ( `const01` bigint(1) NOT NULL DEFAULT '0', `param01` bigint(20) DEFAULT NULL, - `const02` decimal(2,1) unsigned NOT NULL DEFAULT '0.0', + `const02` decimal(2,1) NOT NULL DEFAULT '0.0', `param02` decimal(65,30) DEFAULT NULL, `const03` double NOT NULL DEFAULT '0', `param03` double DEFAULT NULL, @@ -1807,7 +1807,7 @@ select * from t5 ; Catalog Database Table Table_alias Column Column_alias Type Length Max length Is_null Flags Decimals Charsetnr def test t5 t5 const01 const01 8 1 1 N 32769 0 63 def test t5 t5 param01 param01 8 20 1 Y 32768 0 63 -def test t5 t5 const02 const02 246 3 3 N 33 1 63 +def test t5 t5 const02 const02 246 4 3 N 1 1 63 def test t5 t5 param02 param02 246 67 32 Y 0 30 63 def test t5 t5 const03 const03 5 17 1 N 32769 31 63 def test t5 t5 param03 param03 5 23 1 Y 32768 31 63 diff --git a/sql/Makefile.am b/sql/Makefile.am index eec7209bf50..1c4a2ec5402 100644 --- a/sql/Makefile.am +++ b/sql/Makefile.am @@ -52,7 +52,7 @@ noinst_HEADERS = item.h item_func.h item_sum.h item_cmpfunc.h \ ha_heap.h ha_myisam.h ha_myisammrg.h ha_partition.h \ ha_innodb.h ha_berkeley.h ha_federated.h \ ha_ndbcluster.h ha_ndbcluster_binlog.h \ - ha_ndbcluster_tables.h \ + ha_ndbcluster_tables.h ha_maria.h\ opt_range.h protocol.h rpl_tblmap.h \ log.h sql_show.h rpl_rli.h \ sql_select.h structs.h table.h sql_udf.h hash_filo.h \ @@ -88,7 +88,7 @@ mysqld_SOURCES = sql_lex.cc sql_handler.cc sql_partition.cc \ records.cc filesort.cc handler.cc \ ha_heap.cc ha_myisam.cc ha_myisammrg.cc \ ha_partition.cc ha_innodb.cc ha_berkeley.cc \ - ha_federated.cc \ + ha_federated.cc ha_maria.cc\ ha_ndbcluster.cc ha_ndbcluster_binlog.cc \ sql_db.cc sql_table.cc sql_rename.cc sql_crypt.cc \ sql_load.cc mf_iocache.cc field_conv.cc sql_show.cc \ diff --git a/sql/ha_maria.cc b/sql/ha_maria.cc index 46324d0b188..740dfa85a23 100644 --- a/sql/ha_maria.cc +++ b/sql/ha_maria.cc @@ -63,13 +63,16 @@ TYPELIB maria_stats_method_typelib= static handler *maria_create_handler(TABLE_SHARE * table); /* MARIA handlerton */ +static const char maria_hton_name[]= "MARIA"; +static const char maria_hton_comment[]= + "Transactional storage engine, optimized for long running transactions"; handlerton maria_hton= { MYSQL_HANDLERTON_INTERFACE_VERSION, - "MARIA", + maria_hton_name, SHOW_OPTION_YES, - "Transactional storage engine, optimized for long running transactions", + maria_hton_comment, DB_TYPE_MARIA, ha_maria_init, 0, /* slot */ @@ -1834,3 +1837,17 @@ bool ha_maria::check_if_incompatible_data(HA_CREATE_INFO *info, return COMPATIBLE_DATA_NO; return COMPATIBLE_DATA_YES; } + +mysql_declare_plugin(maria) +{ + MYSQL_STORAGE_ENGINE_PLUGIN, + &maria_hton, + maria_hton_name, + "MySQL AB", + maria_hton_comment, + NULL, /* Plugin Init */ + NULL, /* Plugin Deinit */ + 0x0100, /* 1.0 */ + 0 +} +mysql_declare_plugin_end; -- cgit v1.2.1 From fbe22b6020a5d41e5cc0819192d437f3963fd298 Mon Sep 17 00:00:00 2001 From: unknown Date: Thu, 6 Jul 2006 11:10:34 +0300 Subject: The pagae cache added. mysys/Makefile.am: the page cache and test proms for it added include/pagecache.h: New BitKeeper file ``include/pagecache.h'' mysys/mf_pagecache.c: New BitKeeper file ``mysys/mf_pagecache.c'' mysys/test_file.c: New BitKeeper file ``mysys/test_file.c'' mysys/test_file.h: New BitKeeper file ``mysys/test_file.h'' mysys/test_pagecache_consist.c: New BitKeeper file ``mysys/test_pagecache_consist.c'' mysys/test_pagecache_single.c: New BitKeeper file ``mysys/test_pagecache_single.c'' --- include/pagecache.h | 226 +++ mysys/Makefile.am | 58 +- mysys/mf_pagecache.c | 3920 ++++++++++++++++++++++++++++++++++++++++ mysys/test_file.c | 70 + mysys/test_file.h | 14 + mysys/test_pagecache_consist.c | 447 +++++ mysys/test_pagecache_single.c | 589 ++++++ 7 files changed, 5323 insertions(+), 1 deletion(-) create mode 100644 include/pagecache.h create mode 100755 mysys/mf_pagecache.c create mode 100644 mysys/test_file.c create mode 100644 mysys/test_file.h create mode 100755 mysys/test_pagecache_consist.c create mode 100644 mysys/test_pagecache_single.c diff --git a/include/pagecache.h b/include/pagecache.h new file mode 100644 index 00000000000..018e695fa27 --- /dev/null +++ b/include/pagecache.h @@ -0,0 +1,226 @@ +/* Copyright (C) 2006 MySQL AB + + 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; either version 2 of the License, or + (at your option) any later version. + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ + +/* Page cache variable structures */ + +#ifndef _pagecache_h +#define _pagecache_h +C_MODE_START + +/* Type of the page */ +enum pagecache_page_type +{ +#ifndef DBUG_OFF + /* used only for control page type chenging during debugging */ + PAGECACHE_EMPTY_PAGE, +#endif + /* the page does not contain LSN */ + PAGECACHE_PLAIN_PAGE, + /* the page contain LSN (maria tablespace page) */ + PAGECACHE_LSN_PAGE +}; + +/* + This enum describe lock status changing. every typr of page cache will + interpret WRITE/READ lock as it need. +*/ +enum pagecache_page_lock +{ + PAGECACHE_LOCK_LEFT_UNLOCKED, /* free -> free */ + PAGECACHE_LOCK_LEFT_READLOCKED, /* read -> read */ + PAGECACHE_LOCK_LEFT_WRITELOCKED, /* write -> write */ + PAGECACHE_LOCK_READ, /* free -> read */ + PAGECACHE_LOCK_WRITE, /* free -> write */ + PAGECACHE_LOCK_READ_UNLOCK, /* read -> free */ + PAGECACHE_LOCK_WRITE_UNLOCK, /* write -> free */ + PAGECACHE_LOCK_WRITE_TO_READ /* write -> read */ +}; +/* + This enum describe pin status changing +*/ +enum pagecache_page_pin +{ + PAGECACHE_PIN_LEFT_PINNED, /* pinned -> pinned */ + PAGECACHE_PIN_LEFT_UNPINNED, /* unpinned -> unpinned */ + PAGECACHE_PIN, /* unpinned -> pinned */ + PAGECACHE_UNPIN /* pinned -> unpinned */ +}; +/* How to write the page */ +enum pagecache_write_mode +{ + /* do not write immediately, i.e. it will be dirty page */ + PAGECACHE_WRITE_DELAY, + /* write page to the file and put it to the cache */ + PAGECACHE_WRITE_NOW, + /* page already is in the file. (key cache insert analogue) */ + PAGECACHE_WRITE_DONE +}; + +typedef void *PAGECACHE_PAGE_LINK; + +/* TODO: move to loghandler emulator */ +typedef void LOG_HANDLER; +typedef void *LSN; + +/* file descriptor for Maria */ +typedef struct st_pagecache_file +{ + int file; /* it is for debugging purposes then it will be uint32 file_no */ +} PAGECACHE_FILE; + +/* page number for maria */ +typedef uint32 maria_page_no_t; + +/* declare structures that is used by st_pagecache */ + +struct st_pagecache_block_link; +typedef struct st_pagecache_block_link PAGECACHE_BLOCK_LINK; +struct st_pagecache_page; +typedef struct st_pagecache_page PAGECACHE_PAGE; +struct st_pagecache_hash_link; +typedef struct st_pagecache_hash_link PAGECACHE_HASH_LINK; + +/* info about requests in a waiting queue */ +typedef struct st_pagecache_wqueue +{ + struct st_my_thread_var *last_thread; /* circular list of waiting threads */ +} PAGECACHE_WQUEUE; + +#define PAGECACHE_CHANGED_BLOCKS_HASH 128 /* must be power of 2 */ + +/* + The page cache structure + It also contains read-only statistics parameters. +*/ + +typedef struct st_pagecache +{ + my_bool inited; + my_bool resize_in_flush; /* true during flush of resize operation */ + my_bool can_be_used; /* usage of cache for read/write is allowed */ + uint shift; /* block size = 2 ^ shift */ + my_size_t mem_size; /* specified size of the cache memory */ + uint32 block_size; /* size of the page buffer of a cache block */ + ulong min_warm_blocks; /* min number of warm blocks; */ + ulong age_threshold; /* age threshold for hot blocks */ + ulonglong time; /* total number of block link operations */ + uint hash_entries; /* max number of entries in the hash table */ + int hash_links; /* max number of hash links */ + int hash_links_used; /* number of hash links currently used */ + int disk_blocks; /* max number of blocks in the cache */ + ulong blocks_used; /* maximum number of concurrently used blocks */ + ulong blocks_unused; /* number of currently unused blocks */ + ulong blocks_changed; /* number of currently dirty blocks */ + ulong warm_blocks; /* number of blocks in warm sub-chain */ + ulong cnt_for_resize_op; /* counter to block resize operation */ + long blocks_available; /* number of blocks available in the LRU chain */ + PAGECACHE_HASH_LINK **hash_root;/* arr. of entries into hash table buckets */ + PAGECACHE_HASH_LINK *hash_link_root;/* memory for hash table links */ + PAGECACHE_HASH_LINK *free_hash_list;/* list of free hash links */ + PAGECACHE_BLOCK_LINK *free_block_list;/* list of free blocks */ + PAGECACHE_BLOCK_LINK *block_root;/* memory for block links */ + byte HUGE_PTR *block_mem; /* memory for block buffers */ + PAGECACHE_BLOCK_LINK *used_last;/* ptr to the last block of the LRU chain */ + PAGECACHE_BLOCK_LINK *used_ins;/* ptr to the insertion block in LRU chain */ + pthread_mutex_t cache_lock; /* to lock access to the cache structure */ + PAGECACHE_WQUEUE resize_queue; /* threads waiting during resize operation */ + PAGECACHE_WQUEUE waiting_for_hash_link;/* waiting for a free hash link */ + PAGECACHE_WQUEUE waiting_for_block; /* requests waiting for a free block */ + /* hash for dirty file bl.*/ + PAGECACHE_BLOCK_LINK *changed_blocks[PAGECACHE_CHANGED_BLOCKS_HASH]; + /* hash for other file bl.*/ + PAGECACHE_BLOCK_LINK *file_blocks[PAGECACHE_CHANGED_BLOCKS_HASH]; + + LOG_HANDLER *loghandler; /* loghandler structure */ + + /* + The following variables are and variables used to hold parameters for + initializing the key cache. + */ + + ulonglong param_buff_size; /* size the memory allocated for the cache */ + ulong param_block_size; /* size of the blocks in the key cache */ + ulong param_division_limit; /* min. percentage of warm blocks */ + ulong param_age_threshold; /* determines when hot block is downgraded */ + + /* Statistics variables. These are reset in reset_key_cache_counters(). */ + ulong global_blocks_changed; /* number of currently dirty blocks */ + ulonglong global_cache_w_requests;/* number of write requests (write hits) */ + ulonglong global_cache_write; /* number of writes from cache to files */ + ulonglong global_cache_r_requests;/* number of read requests (read hits) */ + ulonglong global_cache_read; /* number of reads from files to cache */ + + int blocks; /* max number of blocks in the cache */ + my_bool in_init; /* Set to 1 in MySQL during init/resize */ +} PAGECACHE; + +extern int init_pagecache(PAGECACHE *pagecache, my_size_t use_mem, + uint division_limit, uint age_threshold, + uint block_size, + LOG_HANDLER *loghandler); +extern int resize_pagecache(PAGECACHE *pagecache, + my_size_t use_mem, uint division_limit, + uint age_threshold); +extern void change_pagecache_param(PAGECACHE *pagecache, uint division_limit, + uint age_threshold); +extern byte *pagecache_read(PAGECACHE *pagecache, + PAGECACHE_FILE *file, + maria_page_no_t pageno, + uint level, + byte *buff, + enum pagecache_page_type type, + enum pagecache_page_lock lock, + PAGECACHE_PAGE_LINK *link); +extern my_bool pagecache_write(PAGECACHE *pagecache, + PAGECACHE_FILE *file, + maria_page_no_t pageno, + uint level, + byte *buff, + enum pagecache_page_type type, + enum pagecache_page_lock lock, + enum pagecache_page_pin pin, + enum pagecache_write_mode write_mode, + PAGECACHE_PAGE_LINK *link); +void pagecache_unlock_page(PAGECACHE *pagecache, + PAGECACHE_FILE *file, + maria_page_no_t pageno, + enum pagecache_page_lock lock, + enum pagecache_page_pin pin, + my_bool stamp_this_page, + LSN first_REDO_LSN_for_page); +void pagecache_unlock(PAGECACHE *pagecache, + PAGECACHE_PAGE_LINK *link, + enum pagecache_page_lock lock, + enum pagecache_page_pin pin, + my_bool stamp_this_page, + LSN first_REDO_LSN_for_page); +void pagecache_unpin_page(PAGECACHE *pagecache, + PAGECACHE_FILE *file, + maria_page_no_t pageno); +void pagecache_unpin(PAGECACHE *pagecache, + PAGECACHE_PAGE_LINK *link); +extern int flush_pagecache_blocks(PAGECACHE *keycache, + PAGECACHE_FILE *file, + enum flush_type type); +my_bool pagecache_delete_page(PAGECACHE *pagecache, + PAGECACHE_FILE *file, + maria_page_no_t pageno, + enum pagecache_page_lock lock, + my_bool flush); +extern void end_pagecache(PAGECACHE *keycache, my_bool cleanup); + +C_MODE_END +#endif /* _keycache_h */ diff --git a/mysys/Makefile.am b/mysys/Makefile.am index 1241e8cdded..485887a085e 100644 --- a/mysys/Makefile.am +++ b/mysys/Makefile.am @@ -54,7 +54,7 @@ libmysys_a_SOURCES = my_init.c my_getwd.c mf_getdate.c my_mmap.c \ my_gethostbyname.c rijndael.c my_aes.c sha1.c \ my_handler.c my_netware.c my_largepage.c \ my_memmem.c \ - my_windac.c my_access.c base64.c + my_windac.c my_access.c base64.c mf_pagecache.c EXTRA_DIST = thr_alarm.c thr_lock.c my_pthread.c my_thr_init.c \ thr_mutex.c thr_rwlock.c \ CMakeLists.txt mf_soundex.c \ @@ -129,5 +129,61 @@ test_base64$(EXEEXT): base64.c $(LIBRARIES) $(LINK) $(FLAGS) -DMAIN ./test_base64.c $(LDADD) $(LIBS) $(RM) -f ./test_base64.c +test_mf_pagecache.o: mf_pagecache.c ../include/pagecache.h $(LIBRARIES) + $(CP) $(srcdir)/mf_pagecache.c test_mf_pagecache.c + $(COMPILE) $(FLAGS) -DPAGECACHE_DEBUG -DEXTRA_DEBUG -c test_mf_pagecache.c + +test_file.o: test_file.c test_file.h + $(COMPILE) $(FLAGS) -DPAGECACHE_DEBUG -DEXTRA_DEBUG -c test_file.c + +test_pagecache_single1k$(EXEEXT): test_pagecache_single.c test_mf_pagecache.o ../unittest/mytap/tap.o test_file.o $(LIBRARIES) + $(LINK) $(FLAGS) -DMAIN -DPAGECACHE_DEBUG -DPAGE_SIZE=1024 -DEXTRA_DEBUG $(srcdir)/test_pagecache_single.c test_mf_pagecache.o ../unittest/mytap/tap.o test_file.o $(LDADD) $(LIBS) + +test_pagecache_single8k$(EXEEXT): test_pagecache_single.c test_mf_pagecache.o ../unittest/mytap/tap.o test_file.o $(LIBRARIES) + $(LINK) $(FLAGS) -DMAIN -DPAGECACHE_DEBUG -DPAGE_SIZE=8192 -DEXTRA_DEBUG $(srcdir)/test_pagecache_single.c test_mf_pagecache.o ../unittest/mytap/tap.o test_file.o $(LDADD) $(LIBS) + +test_pagecache_single64k$(EXEEXT): test_pagecache_single.c test_mf_pagecache.o ../unittest/mytap/tap.o test_file.o $(LIBRARIES) + $(LINK) $(FLAGS) -DMAIN -DPAGECACHE_DEBUG -DPAGE_SIZE=65536 -DEXTRA_DEBUG $(srcdir)/test_pagecache_single.c test_mf_pagecache.o ../unittest/mytap/tap.o test_file.o $(LDADD) $(LIBS) + +test_pagecache_single: test_pagecache_single1k$(EXEEXT) test_pagecache_single8k$(EXEEXT) test_pagecache_single64k$(EXEEXT) + ./test_pagecache_single64k$(EXEEXT) + ./test_pagecache_single8k$(EXEEXT) + ./test_pagecache_single1k$(EXEEXT) + +test_pagecache_consist1k$(EXEEXT): test_pagecache_consist.c test_mf_pagecache.o ../unittest/mytap/tap.o $(LIBRARIES) + $(LINK) $(FLAGS) -DMAIN -DPAGECACHE_DEBUG -DPAGE_SIZE=1024 -DEXTRA_DEBUG $(srcdir)/test_pagecache_consist.c test_mf_pagecache.o ../unittest/mytap/tap.o $(LDADD) $(LIBS) + +test_pagecache_consist64k$(EXEEXT): test_pagecache_consist.c test_mf_pagecache.o ../unittest/mytap/tap.o $(LIBRARIES) + $(LINK) $(FLAGS) -DMAIN -DPAGECACHE_DEBUG -DPAGE_SIZE=65536 -DEXTRA_DEBUG $(srcdir)/test_pagecache_consist.c test_mf_pagecache.o ../unittest/mytap/tap.o $(LDADD) $(LIBS) + +test_pagecache_consist1kHC$(EXEEXT): test_pagecache_consist.c test_mf_pagecache.o ../unittest/mytap/tap.o $(LIBRARIES) + $(LINK) $(FLAGS) -DMAIN -DPAGECACHE_DEBUG -DTEST_HIGH_CONCURENCY -DPAGE_SIZE=1024 -DEXTRA_DEBUG $(srcdir)/test_pagecache_consist.c test_mf_pagecache.o ../unittest/mytap/tap.o $(LDADD) $(LIBS) + +test_pagecache_consist64kHC$(EXEEXT): test_pagecache_consist.c test_mf_pagecache.o ../unittest/mytap/tap.o $(LIBRARIES) + $(LINK) $(FLAGS) -DMAIN -DPAGECACHE_DEBUG -DTEST_HIGH_CONCURENCY -DPAGE_SIZE=65536 -DEXTRA_DEBUG $(srcdir)/test_pagecache_consist.c test_mf_pagecache.o ../unittest/mytap/tap.o $(LDADD) $(LIBS) + +test_pagecache_consist1kRD$(EXEEXT): test_pagecache_consist.c test_mf_pagecache.o ../unittest/mytap/tap.o $(LIBRARIES) + $(LINK) $(FLAGS) -DMAIN -DPAGECACHE_DEBUG -DTEST_READERS -DPAGE_SIZE=1024 -DEXTRA_DEBUG $(srcdir)/test_pagecache_consist.c test_mf_pagecache.o ../unittest/mytap/tap.o $(LDADD) $(LIBS) + +test_pagecache_consist64kRD$(EXEEXT): test_pagecache_consist.c test_mf_pagecache.o ../unittest/mytap/tap.o $(LIBRARIES) + $(LINK) $(FLAGS) -DMAIN -DPAGECACHE_DEBUG -DTEST_READERS -DPAGE_SIZE=65536 -DEXTRA_DEBUG $(srcdir)/test_pagecache_consist.c test_mf_pagecache.o ../unittest/mytap/tap.o $(LDADD) $(LIBS) + +test_pagecache_consist1kWR$(EXEEXT): test_pagecache_consist.c test_mf_pagecache.o ../unittest/mytap/tap.o $(LIBRARIES) + $(LINK) $(FLAGS) -DMAIN -DPAGECACHE_DEBUG -DTEST_WRITERS -DPAGE_SIZE=1024 -DEXTRA_DEBUG $(srcdir)/test_pagecache_consist.c test_mf_pagecache.o ../unittest/mytap/tap.o $(LDADD) $(LIBS) + +test_pagecache_consist64kWR$(EXEEXT): test_pagecache_consist.c test_mf_pagecache.o ../unittest/mytap/tap.o $(LIBRARIES) + $(LINK) $(FLAGS) -DMAIN -DPAGECACHE_DEBUG -DTEST_WRITERS -DPAGE_SIZE=65536 -DEXTRA_DEBUG $(srcdir)/test_pagecache_consist.c test_mf_pagecache.o ../unittest/mytap/tap.o $(LDADD) $(LIBS) + +test_pagecache_consist: test_pagecache_consist1k$(EXEEXT) test_pagecache_consist64k$(EXEEXT) test_pagecache_consist1kHC$(EXEEXT) test_pagecache_consist64kHC$(EXEEXT) test_pagecache_consist1kRD$(EXEEXT) test_pagecache_consist64kRD$(EXEEXT) test_pagecache_consist1kWR$(EXEEXT) test_pagecache_consist64kWR$(EXEEXT) + ./test_pagecache_consist1k$(EXEEXT) + ./test_pagecache_consist64k$(EXEEXT) + ./test_pagecache_consist1kHC$(EXEEXT) + ./test_pagecache_consist64kHC$(EXEEXT) + ./test_pagecache_consist1kRD$(EXEEXT) + ./test_pagecache_consist64kRD$(EXEEXT) + ./test_pagecache_consist1kWR$(EXEEXT) + ./test_pagecache_consist64kWR$(EXEEXT) + + # Don't update the files from bitkeeper %::SCCS/s.% diff --git a/mysys/mf_pagecache.c b/mysys/mf_pagecache.c new file mode 100755 index 00000000000..4693995f922 --- /dev/null +++ b/mysys/mf_pagecache.c @@ -0,0 +1,3920 @@ +/* Copyright (C) 2000 MySQL AB + + 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; either version 2 of the License, or + (at your option) any later version. + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ + +/* + These functions handle page cacheing for Maria tables. + + One cache can handle many files. + It must contain buffers of the same blocksize. + init_pagecache() should be used to init cache handler. + + The free list (free_block_list) is a stack like structure. + When a block is freed by free_block(), it is pushed onto the stack. + When a new block is required it is first tried to pop one from the stack. + If the stack is empty, it is tried to get a never-used block from the pool. + If this is empty too, then a block is taken from the LRU ring, flushing it + to disk, if neccessary. This is handled in find_key_block(). + With the new free list, the blocks can have three temperatures: + hot, warm and cold (which is free). This is remembered in the block header + by the enum BLOCK_TEMPERATURE temperature variable. Remembering the + temperature is neccessary to correctly count the number of warm blocks, + which is required to decide when blocks are allowed to become hot. Whenever + a block is inserted to another (sub-)chain, we take the old and new + temperature into account to decide if we got one more or less warm block. + blocks_unused is the sum of never used blocks in the pool and of currently + free blocks. blocks_used is the number of blocks fetched from the pool and + as such gives the maximum number of in-use blocks at any time. +*/ + +#include "mysys_priv.h" +#include +#include "my_static.h" +#include +#include +#include + +/* + Some compilation flags have been added specifically for this module + to control the following: + - not to let a thread to yield the control when reading directly + from page cache, which might improve performance in many cases; + to enable this add: + #define SERIALIZED_READ_FROM_CACHE + - to set an upper bound for number of threads simultaneously + using the page cache; this setting helps to determine an optimal + size for hash table and improve performance when the number of + blocks in the page cache much less than the number of threads + accessing it; + to set this number equal to add + #define MAX_THREADS + - to substitute calls of pthread_cond_wait for calls of + pthread_cond_timedwait (wait with timeout set up); + this setting should be used only when you want to trap a deadlock + situation, which theoretically should not happen; + to set timeout equal to seconds add + #define PAGECACHE_TIMEOUT + - to enable the module traps and to send debug information from + page cache module to a special debug log add: + #define PAGECACHE_DEBUG + the name of this debug log file can be set through: + #define PAGECACHE_DEBUG_LOG + if the name is not defined, it's set by default; + if the PAGECACHE_DEBUG flag is not set up and we are in a debug + mode, i.e. when ! defined(DBUG_OFF), the debug information from the + module is sent to the regular debug log. + + Example of the settings: + #define SERIALIZED_READ_FROM_CACHE + #define MAX_THREADS 100 + #define PAGECACHE_TIMEOUT 1 + #define PAGECACHE_DEBUG + #define PAGECACHE_DEBUG_LOG "my_pagecache_debug.log" +*/ + +#define PAGECACHE_DEBUG_LOG "my_pagecache_debug.log" + +/* + In key cache we have external raw locking here we use + SERIALIZED_READ_FROM_CACHE to avoid problem of reading + not consistent data from te page +*/ +#define SERIALIZED_READ_FROM_CACHE yes + +#define BLOCK_INFO(B) DBUG_PRINT("info", \ + ("block 0x%lx, file %lu, page %lu, s %0x", \ + (ulong)(B), \ + (ulong)((B)->hash_link ? \ + (B)->hash_link->file.file : \ + 0), \ + (ulong)((B)->hash_link ? \ + (B)->hash_link->pageno : \ + 0), \ + (B)->status)) + +/* TODO: put it to my_static.c */ +my_bool my_disable_flush_pagecache_blocks= 0; + +#if defined(MSDOS) && !defined(M_IC80386) +/* we nead much memory */ +#undef my_malloc_lock +#undef my_free_lock +#define my_malloc_lock(A,B) halloc((long) (A/IO_SIZE),IO_SIZE) +#define my_free_lock(A,B) hfree(A) +#endif /* defined(MSDOS) && !defined(M_IC80386) */ + +#define STRUCT_PTR(TYPE, MEMBER, a) \ + (TYPE *) ((char *) (a) - offsetof(TYPE, MEMBER)) + +/* types of condition variables */ +#define COND_FOR_REQUESTED 0 +#define COND_FOR_SAVED 1 +#define COND_FOR_WRLOCK 2 +#define COND_FOR_COPY 3 +#define COND_SIZE 4 + +typedef pthread_cond_t KEYCACHE_CONDVAR; + +/* descriptor of the page in the page cache block buffer */ +struct st_pagecache_page +{ + PAGECACHE_FILE file; /* file to which the page belongs to */ + maria_page_no_t pageno; /* number of the page in the file */ +}; + +/* element in the chain of a hash table bucket */ +struct st_pagecache_hash_link +{ + struct st_pagecache_hash_link + *next, **prev; /* to connect links in the same bucket */ + struct st_pagecache_block_link + *block; /* reference to the block for the page: */ + PAGECACHE_FILE file; /* from such a file */ + maria_page_no_t pageno; /* this page */ + uint requests; /* number of requests for the page */ +}; + +/* simple states of a block */ +#define BLOCK_ERROR 1 /* an error occured when performing disk i/o */ +#define BLOCK_READ 2 /* the is page in the block buffer */ +#define BLOCK_IN_SWITCH 4 /* block is preparing to read new page */ +#define BLOCK_REASSIGNED 8 /* block does not accept requests for old page */ +#define BLOCK_IN_FLUSH 16 /* block is in flush operation */ +#define BLOCK_CHANGED 32 /* block buffer contains a dirty page */ +#define BLOCK_WRLOCK 64 /* write locked block */ +#define BLOCK_CPYWRT 128 /* block buffer is in copy&write (see also cpyrd) */ + +/* page status, returned by find_key_block */ +#define PAGE_READ 0 +#define PAGE_TO_BE_READ 1 +#define PAGE_WAIT_TO_BE_READ 2 + +/* block temperature determines in which (sub-)chain the block currently is */ +enum BLOCK_TEMPERATURE { BLOCK_COLD /*free*/ , BLOCK_WARM , BLOCK_HOT }; + +/* debug info */ +#ifndef DBUG_OFF +static char *page_cache_page_type_str[]= +{ + (char*)"PLAIN", + (char*)"LSN" +}; +static char *page_cache_page_write_mode_str[]= +{ + (char*)"DELAY", + (char*)"NOW", + (char*)"DONE" +}; +static char *page_cache_page_lock_str[]= +{ + (char*)"free -> free ", + (char*)"read -> read ", + (char*)"write -> write", + (char*)"free -> read ", + (char*)"free -> write", + (char*)"read -> free ", + (char*)"write -> free ", + (char*)"write -> read " +}; +static char *page_cache_page_pin_str[]= +{ + (char*)"pinned -> pinned ", + (char*)"unpinned -> unpinned", + (char*)"unpinned -> pinned ", + (char*)"pinned -> unpinned" +}; +#endif +#ifdef PAGECACHE_DEBUG +typedef struct st_pagecache_pin_info +{ + struct st_pagecache_pin_info *next, **prev; + struct st_my_thread_var *thread; +} PAGECACHE_PIN_INFO; +/* + st_pagecache_lock_info structure should be kept in next, prev, thread part + compatible with st_pagecache_pin_info to be compatible in functions. +*/ +typedef struct st_pagecache_lock_info +{ + struct st_pagecache_lock_info *next, **prev; + struct st_my_thread_var *thread; + my_bool write_lock; +} PAGECACHE_LOCK_INFO; +/* service functions */ +void info_link(PAGECACHE_PIN_INFO **list, PAGECACHE_PIN_INFO *node) +{ + if ((node->next= *list)) + node->next->prev= &(node->next); + *list= node; + node->prev= list; +} +void info_unlink(PAGECACHE_PIN_INFO *node) +{ + if ((*node->prev= node->next)) + node->next->prev= node->prev; +} +PAGECACHE_PIN_INFO *info_find(PAGECACHE_PIN_INFO *list, + struct st_my_thread_var *thread) +{ + register PAGECACHE_PIN_INFO *i= list; + for(; i != 0; i= i->next) + if (i->thread == thread) + return i; + return 0; +} +#endif + +/* page cache block */ +struct st_pagecache_block_link +{ + struct st_pagecache_block_link + *next_used, **prev_used; /* to connect links in the LRU chain (ring) */ + struct st_pagecache_block_link + *next_changed, **prev_changed; /* for lists of file dirty/clean blocks */ + struct st_pagecache_hash_link + *hash_link; /* backward ptr to referring hash_link */ + PAGECACHE_WQUEUE + wqueue[COND_SIZE]; /* queues on waiting requests for new/old pages */ + uint requests; /* number of requests for the block */ + byte *buffer; /* buffer for the block page */ + volatile uint status; /* state of the block */ + volatile uint pins; /* pin counter */ +#ifdef PAGECACHE_DEBUG + PAGECACHE_PIN_INFO *pin_list; + PAGECACHE_LOCK_INFO *lock_list; +#endif + enum BLOCK_TEMPERATURE temperature; /* block temperature: cold, warm, hot */ + enum pagecache_page_type type; /* type of the block */ + uint hits_left; /* number of hits left until promotion */ + ulonglong last_hit_time; /* timestamp of the last hit */ + KEYCACHE_CONDVAR *condvar; /* condition variable for 'no readers' event */ +}; + +#ifdef PAGECACHE_DEBUG +/* debug checks */ +bool info_check_pin(PAGECACHE_BLOCK_LINK *block, + enum pagecache_page_pin mode) +{ + struct st_my_thread_var *thread= my_thread_var; + DBUG_ENTER("info_check_pin"); + PAGECACHE_PIN_INFO *info= info_find(block->pin_list, thread); + if (info) + { + if (mode == PAGECACHE_PIN_LEFT_UNPINNED) + { + DBUG_PRINT("info", + ("info_check_pin: thread: 0x%lx block 0x%lx: LEFT_UNPINNED!!!", + (ulong)thread, (ulong)block)); + DBUG_RETURN(1); + } + else if (mode == PAGECACHE_PIN) + { + DBUG_PRINT("info", + ("info_check_pin: thread: 0x%lx block 0x%lx: PIN!!!", + (ulong)thread, (ulong)block)); + DBUG_RETURN(1); + } + } + else + { + if (mode == PAGECACHE_PIN_LEFT_PINNED) + { + DBUG_PRINT("info", + ("info_check_pin: thread: 0x%lx block 0x%lx: LEFT_PINNED!!!", + (ulong)thread, (ulong)block)); + DBUG_RETURN(1); + } + else if (mode == PAGECACHE_UNPIN) + { + DBUG_PRINT("info", + ("info_check_pin: thread: 0x%lx block 0x%lx: UNPIN!!!", + (ulong)thread, (ulong)block)); + DBUG_RETURN(1); + } + } + DBUG_RETURN(0); +} +bool info_check_lock(PAGECACHE_BLOCK_LINK *block, + enum pagecache_page_lock lock, + enum pagecache_page_pin pin) +{ + struct st_my_thread_var *thread= my_thread_var; + DBUG_ENTER("info_check_lock"); + PAGECACHE_LOCK_INFO *info= + (PAGECACHE_LOCK_INFO *) info_find((PAGECACHE_PIN_INFO *) block->lock_list, + thread); + switch(lock) + { + case PAGECACHE_LOCK_LEFT_UNLOCKED: + DBUG_ASSERT(pin == PAGECACHE_PIN_LEFT_UNPINNED); + if (info) + { + DBUG_PRINT("info", + ("info_check_lock: thread: 0x%lx block 0x%lx: %c : U->U", + (ulong)thread, (ulong)block, (info->write_lock?'W':'R'))); + DBUG_RETURN(1); + } + break; + case PAGECACHE_LOCK_LEFT_READLOCKED: + DBUG_ASSERT(pin == PAGECACHE_PIN_LEFT_UNPINNED || + pin == PAGECACHE_PIN_LEFT_UNPINNED); + if (info == 0 || info->write_lock) + { + DBUG_PRINT("info", + ("info_check_lock: thread: 0x%lx block 0x%lx: %c : R->R", + (ulong)thread, (ulong)block, (info?'W':'U'))); + DBUG_RETURN(1); + } + break; + case PAGECACHE_LOCK_LEFT_WRITELOCKED: + DBUG_ASSERT(pin == PAGECACHE_PIN_LEFT_PINNED); + if (info == 0 || !info->write_lock) + { + DBUG_PRINT("info", + ("info_check_lock: thread: 0x%lx block 0x%lx: %c : W->W", + (ulong)thread, (ulong)block, (info?'R':'U'))); + DBUG_RETURN(1); + } + break; + case PAGECACHE_LOCK_READ: + DBUG_ASSERT(pin == PAGECACHE_PIN_LEFT_UNPINNED || + pin == PAGECACHE_PIN); + if (info != 0) + { + DBUG_PRINT("info", + ("info_check_lock: thread: 0x%lx block 0x%lx: %c : U->R", + (ulong)thread, (ulong)block, (info->write_lock?'W':'R'))); + DBUG_RETURN(1); + } + break; + case PAGECACHE_LOCK_WRITE: + DBUG_ASSERT(pin == PAGECACHE_PIN); + if (info != 0) + { + DBUG_PRINT("info", + ("info_check_lock: thread: 0x%lx block 0x%lx: %c : U->W", + (ulong)thread, (ulong)block, (info->write_lock?'W':'R'))); + DBUG_RETURN(1); + } + break; + + case PAGECACHE_LOCK_READ_UNLOCK: + DBUG_ASSERT(pin == PAGECACHE_PIN_LEFT_UNPINNED || + pin == PAGECACHE_UNPIN); + if (info == 0 || info->write_lock) + { + DBUG_PRINT("info", + ("info_check_lock: thread: 0x%lx block 0x%lx: %c : R->U", + (ulong)thread, (ulong)block, (info?'W':'U'))); + DBUG_RETURN(1); + } + break; + case PAGECACHE_LOCK_WRITE_UNLOCK: + DBUG_ASSERT(pin == PAGECACHE_UNPIN); + if (info == 0 || !info->write_lock) + { + DBUG_PRINT("info", + ("info_check_lock: thread: 0x%lx block 0x%lx: %c : W->U", + (ulong)thread, (ulong)block, (info?'R':'U'))); + DBUG_RETURN(1); + } + break; + case PAGECACHE_LOCK_WRITE_TO_READ: + DBUG_ASSERT(pin == PAGECACHE_PIN_LEFT_PINNED || + pin == PAGECACHE_UNPIN); + if (info == 0 || !info->write_lock) + { + DBUG_PRINT("info", + ("info_check_lock: thread: 0x%lx block 0x%lx: %c : W->U", + (ulong)thread, (ulong)block, (info?'R':'U'))); + DBUG_RETURN(1); + } + break; + } + DBUG_RETURN(0); +} +#endif + +#define FLUSH_CACHE 2000 /* sort this many blocks at once */ + +static int flush_all_key_blocks(PAGECACHE *pagecache); +#ifdef THREAD +static void link_into_queue(PAGECACHE_WQUEUE *wqueue, + struct st_my_thread_var *thread); +static void unlink_from_queue(PAGECACHE_WQUEUE *wqueue, + struct st_my_thread_var *thread); +#endif +static void free_block(PAGECACHE *pagecache, PAGECACHE_BLOCK_LINK *block); +static void test_key_cache(PAGECACHE *pagecache, + const char *where, my_bool lock); + +#define PAGECACHE_HASH(p, f, pos) (((ulong) (pos) + \ + (ulong) (f).file) & (p->hash_entries-1)) +#define FILE_HASH(f) ((uint) (f).file & (PAGECACHE_CHANGED_BLOCKS_HASH - 1)) + +#define DEFAULT_PAGECACHE_DEBUG_LOG "pagecache_debug.log" + +#if defined(PAGECACHE_DEBUG) && ! defined(PAGECACHE_DEBUG_LOG) +#define PAGECACHE_DEBUG_LOG DEFAULT_PAGECACHE_DEBUG_LOG +#endif + +#if defined(PAGECACHE_DEBUG_LOG) +static FILE *pagecache_debug_log= NULL; +static void pagecache_debug_print _VARARGS((const char *fmt, ...)); +#define PAGECACHE_DEBUG_OPEN \ + if (!pagecache_debug_log) \ + { \ + pagecache_debug_log= fopen(PAGECACHE_DEBUG_LOG, "w"); \ + (void) setvbuf(pagecache_debug_log, NULL, _IOLBF, BUFSIZ); \ + } + +#define PAGECACHE_DEBUG_CLOSE \ + if (pagecache_debug_log) \ + { \ + fclose(pagecache_debug_log); \ + pagecache_debug_log= 0; \ + } +#else +#define PAGECACHE_DEBUG_OPEN +#define PAGECACHE_DEBUG_CLOSE +#endif /* defined(PAGECACHE_DEBUG_LOG) */ + +#if defined(PAGECACHE_DEBUG_LOG) && defined(PAGECACHE_DEBUG) +#define KEYCACHE_DBUG_PRINT(l, m) \ + { if (pagecache_debug_log) \ + fprintf(pagecache_debug_log, "%s: ", l); \ + pagecache_debug_print m; } + +#define KEYCACHE_DBUG_ASSERT(a) \ + { if (! (a) && pagecache_debug_log) \ + fclose(pagecache_debug_log); \ + assert(a); } +#else +#define KEYCACHE_DBUG_PRINT(l, m) DBUG_PRINT(l, m) +#define KEYCACHE_DBUG_ASSERT(a) DBUG_ASSERT(a) +#endif /* defined(PAGECACHE_DEBUG_LOG) && defined(PAGECACHE_DEBUG) */ + +#if defined(PAGECACHE_DEBUG) || !defined(DBUG_OFF) +#ifdef THREAD +static long pagecache_thread_id; +#define KEYCACHE_THREAD_TRACE(l) \ + KEYCACHE_DBUG_PRINT(l,("|thread %ld",pagecache_thread_id)) + +#define KEYCACHE_THREAD_TRACE_BEGIN(l) \ + { struct st_my_thread_var *thread_var= my_thread_var; \ + pagecache_thread_id= thread_var->id; \ + KEYCACHE_DBUG_PRINT(l,("[thread %ld",pagecache_thread_id)) } + +#define KEYCACHE_THREAD_TRACE_END(l) \ + KEYCACHE_DBUG_PRINT(l,("]thread %ld",pagecache_thread_id)) +#else /* THREAD */ +#define KEYCACHE_THREAD_TRACE(l) KEYCACHE_DBUG_PRINT(l,("")) +#define KEYCACHE_THREAD_TRACE_BEGIN(l) KEYCACHE_DBUG_PRINT(l,("")) +#define KEYCACHE_THREAD_TRACE_END(l) KEYCACHE_DBUG_PRINT(l,("")) +#endif /* THREAD */ +#else +#define KEYCACHE_THREAD_TRACE_BEGIN(l) +#define KEYCACHE_THREAD_TRACE_END(l) +#define KEYCACHE_THREAD_TRACE(l) +#endif /* defined(PAGECACHE_DEBUG) || !defined(DBUG_OFF) */ + +#define BLOCK_NUMBER(p, b) \ + ((uint) (((char*)(b)-(char *) p->block_root)/sizeof(PAGECACHE_BLOCK_LINK))) +#define PAGECACHE_HASH_LINK_NUMBER(p, h) \ + ((uint) (((char*)(h)-(char *) p->hash_link_root)/ \ + sizeof(PAGECACHE_HASH_LINK))) + +#if (defined(PAGECACHE_TIMEOUT) && !defined(__WIN__)) || defined(PAGECACHE_DEBUG) +static int pagecache_pthread_cond_wait(pthread_cond_t *cond, + pthread_mutex_t *mutex); +#else +#define pagecache_pthread_cond_wait pthread_cond_wait +#endif + +#if defined(PAGECACHE_DEBUG) +static int ___pagecache_pthread_mutex_lock(pthread_mutex_t *mutex); +static void ___pagecache_pthread_mutex_unlock(pthread_mutex_t *mutex); +static int ___pagecache_pthread_cond_signal(pthread_cond_t *cond); +#define pagecache_pthread_mutex_lock(M) \ +{ DBUG_PRINT("lock", ("mutex lock 0x%lx %u", (ulong)(M), __LINE__)); \ + ___pagecache_pthread_mutex_lock(M);} +#define pagecache_pthread_mutex_unlock(M) \ +{ DBUG_PRINT("lock", ("mutex unlock 0x%lx %u", (ulong)(M), __LINE__)); \ + ___pagecache_pthread_mutex_unlock(M);} +#define pagecache_pthread_cond_signal(M) \ +{ DBUG_PRINT("lock", ("signal 0x%lx %u", (ulong)(M), __LINE__)); \ + ___pagecache_pthread_cond_signal(M);} +#else +#define pagecache_pthread_mutex_lock pthread_mutex_lock +#define pagecache_pthread_mutex_unlock pthread_mutex_unlock +#define pagecache_pthread_cond_signal pthread_cond_signal +#endif /* defined(PAGECACHE_DEBUG) */ + + +/* + Read page from the disk + + SYNOPSIS + pagecache_fwrite() + pagecache - page cache pointer + filedesc - pagecache file descriptor structure + buffer - buffer in which we will read + type - page type (plain or with LSN) + flags - MYF() flags + + RETURN + 0 - OK + !=0 - Error +*/ +uint pagecache_fwrite(PAGECACHE *pagecache, + PAGECACHE_FILE *filedesc, + byte *buffer, + maria_page_no_t pageno, + enum pagecache_page_type type, + myf flags) +{ + DBUG_ENTER("pagecache_fwrite"); + if (type == PAGECACHE_LSN_PAGE) + { + DBUG_PRINT("info", ("Log handler call")); + /* TODO: put here loghandler call */ + } + DBUG_RETURN(my_pwrite(filedesc->file, buffer, pagecache->block_size, + (pageno)<<(pagecache->shift), flags)); +} + + +/* + Read page from the disk + + SYNOPSIS + pagecache_fread() + pagecache - page cache pointer + filedesc - pagecache file descriptor structure + buffer - buffer in which we will read + pageno - page number + flags - MYF() flags +*/ +#define pagecache_fread(pagecache, filedesc, buffer, pageno, flags) \ + my_pread((filedesc)->file, buffer, pagecache->block_size, \ + (pageno)<<(pagecache->shift), flags) + + +static uint next_power(uint value) +{ + uint old_value= 1; + while (value) + { + old_value= value; + value&= value-1; + } + return (old_value << 1); +} + + +/* + Initialize a page cache + + SYNOPSIS + init_pagecache() + pagecache pointer to a page cache data structure + key_cache_block_size size of blocks to keep cached data + use_mem total memory to use for the key cache + division_limit division limit (may be zero) + age_threshold age threshold (may be zero) + block_size size of block (should be power of 2) + loghandler logfandler pointer to call it in case of + pages with LSN + + RETURN VALUE + number of blocks in the key cache, if successful, + 0 - otherwise. + + NOTES. + if pagecache->inited != 0 we assume that the key cache + is already initialized. This is for now used by myisamchk, but shouldn't + be something that a program should rely on! + + It's assumed that no two threads call this function simultaneously + referring to the same key cache handle. + +*/ + +int init_pagecache(PAGECACHE *pagecache, my_size_t use_mem, + uint division_limit, uint age_threshold, + uint block_size, + LOG_HANDLER *loghandler) +{ + int blocks, hash_links, length; + int error; + DBUG_ENTER("init_key_cache"); + DBUG_ASSERT(block_size >= 512); + + PAGECACHE_DEBUG_OPEN; + if (pagecache->inited && pagecache->disk_blocks > 0) + { + DBUG_PRINT("warning",("key cache already in use")); + DBUG_RETURN(0); + } + + pagecache->loghandler= loghandler; + + pagecache->global_cache_w_requests= pagecache->global_cache_r_requests= 0; + pagecache->global_cache_read= pagecache->global_cache_write= 0; + pagecache->disk_blocks= -1; + if (! pagecache->inited) + { + pagecache->inited= 1; + pagecache->in_init= 0; + pthread_mutex_init(&pagecache->cache_lock, MY_MUTEX_INIT_FAST); + pagecache->resize_queue.last_thread= NULL; + } + + pagecache->mem_size= use_mem; + pagecache->block_size= block_size; + pagecache->shift= my_bit_log2(block_size); + DBUG_PRINT("info", ("block_size: %u", + block_size)); + + blocks= (int) (use_mem / (sizeof(PAGECACHE_BLOCK_LINK) + + 2 * sizeof(PAGECACHE_HASH_LINK) + + sizeof(PAGECACHE_HASH_LINK*) * + 5/4 + block_size)); + /* It doesn't make sense to have too few blocks (less than 8) */ + if (blocks >= 8 && pagecache->disk_blocks < 0) + { + for ( ; ; ) + { + /* Set my_hash_entries to the next bigger 2 power */ + if ((pagecache->hash_entries= next_power((uint)blocks)) < + ((uint)blocks) * 5/4) + pagecache->hash_entries<<= 1; + hash_links= 2 * blocks; +#if defined(MAX_THREADS) + if (hash_links < MAX_THREADS + blocks - 1) + hash_links= MAX_THREADS + blocks - 1; +#endif + while ((length= (ALIGN_SIZE(blocks * sizeof(PAGECACHE_BLOCK_LINK)) + + ALIGN_SIZE(hash_links * sizeof(PAGECACHE_HASH_LINK)) + + ALIGN_SIZE(sizeof(PAGECACHE_HASH_LINK*) * + pagecache->hash_entries))) + + ((ulong) blocks << pagecache->shift) > use_mem) + blocks--; + /* Allocate memory for cache page buffers */ + if ((pagecache->block_mem= + my_large_malloc((ulong) blocks * pagecache->block_size, + MYF(MY_WME)))) + { + /* + Allocate memory for blocks, hash_links and hash entries; + For each block 2 hash links are allocated + */ + if ((pagecache->block_root= + (PAGECACHE_BLOCK_LINK*) my_malloc((uint) length, + MYF(0)))) + break; + my_large_free(pagecache->block_mem, MYF(0)); + pagecache->block_mem= 0; + } + if (blocks < 8) + { + my_errno= ENOMEM; + goto err; + } + blocks= blocks / 4*3; + } + pagecache->blocks_unused= (ulong) blocks; + pagecache->disk_blocks= (int) blocks; + pagecache->hash_links= hash_links; + pagecache->hash_root= + (PAGECACHE_HASH_LINK**) ((char*) pagecache->block_root + + ALIGN_SIZE(blocks*sizeof(PAGECACHE_BLOCK_LINK))); + pagecache->hash_link_root= + (PAGECACHE_HASH_LINK*) ((char*) pagecache->hash_root + + ALIGN_SIZE((sizeof(PAGECACHE_HASH_LINK*) * + pagecache->hash_entries))); + bzero((byte*) pagecache->block_root, + pagecache->disk_blocks * sizeof(PAGECACHE_BLOCK_LINK)); + bzero((byte*) pagecache->hash_root, + pagecache->hash_entries * sizeof(PAGECACHE_HASH_LINK*)); + bzero((byte*) pagecache->hash_link_root, + pagecache->hash_links * sizeof(PAGECACHE_HASH_LINK)); + pagecache->hash_links_used= 0; + pagecache->free_hash_list= NULL; + pagecache->blocks_used= pagecache->blocks_changed= 0; + + pagecache->global_blocks_changed= 0; + pagecache->blocks_available=0; /* For debugging */ + + /* The LRU chain is empty after initialization */ + pagecache->used_last= NULL; + pagecache->used_ins= NULL; + pagecache->free_block_list= NULL; + pagecache->time= 0; + pagecache->warm_blocks= 0; + pagecache->min_warm_blocks= (division_limit ? + blocks * division_limit / 100 + 1 : + (ulong)blocks); + pagecache->age_threshold= (age_threshold ? + blocks * age_threshold / 100 : + (ulong)blocks); + + pagecache->cnt_for_resize_op= 0; + pagecache->resize_in_flush= 0; + pagecache->can_be_used= 1; + + pagecache->waiting_for_hash_link.last_thread= NULL; + pagecache->waiting_for_block.last_thread= NULL; + DBUG_PRINT("exit", + ("disk_blocks: %d block_root: 0x%lx hash_entries: %d\ + hash_root: 0x%lx hash_links: %d hash_link_root: 0x%lx", + pagecache->disk_blocks, pagecache->block_root, + pagecache->hash_entries, pagecache->hash_root, + pagecache->hash_links, pagecache->hash_link_root)); + bzero((gptr) pagecache->changed_blocks, + sizeof(pagecache->changed_blocks[0]) * + PAGECACHE_CHANGED_BLOCKS_HASH); + bzero((gptr) pagecache->file_blocks, + sizeof(pagecache->file_blocks[0]) * + PAGECACHE_CHANGED_BLOCKS_HASH); + } + + pagecache->blocks= pagecache->disk_blocks > 0 ? pagecache->disk_blocks : 0; + DBUG_RETURN((uint) pagecache->disk_blocks); + +err: + error= my_errno; + pagecache->disk_blocks= 0; + pagecache->blocks= 0; + if (pagecache->block_mem) + { + my_large_free((gptr) pagecache->block_mem, MYF(0)); + pagecache->block_mem= NULL; + } + if (pagecache->block_root) + { + my_free((gptr) pagecache->block_root, MYF(0)); + pagecache->block_root= NULL; + } + my_errno= error; + pagecache->can_be_used= 0; + DBUG_RETURN(0); +} + + +/* + Resize a key cache + + SYNOPSIS + resize_pagecache() + pagecache pointer to a page cache data structure + use_mem total memory to use for the new key cache + division_limit new division limit (if not zero) + age_threshold new age threshold (if not zero) + + RETURN VALUE + number of blocks in the key cache, if successful, + 0 - otherwise. + + NOTES. + The function first compares the memory size parameter + with the key cache value. + + If they differ the function free the the memory allocated for the + old key cache blocks by calling the end_pagecache function and + then rebuilds the key cache with new blocks by calling + init_key_cache. + + The function starts the operation only when all other threads + performing operations with the key cache let her to proceed + (when cnt_for_resize=0). +*/ + +int resize_pagecache(PAGECACHE *pagecache, + my_size_t use_mem, uint division_limit, + uint age_threshold) +{ + int blocks; + struct st_my_thread_var *thread; + PAGECACHE_WQUEUE *wqueue; + DBUG_ENTER("resize_pagecache"); + + if (!pagecache->inited) + DBUG_RETURN(pagecache->disk_blocks); + + if(use_mem == pagecache->mem_size) + { + change_pagecache_param(pagecache, division_limit, age_threshold); + DBUG_RETURN(pagecache->disk_blocks); + } + + pagecache_pthread_mutex_lock(&pagecache->cache_lock); + +#ifdef THREAD + wqueue= &pagecache->resize_queue; + thread= my_thread_var; + link_into_queue(wqueue, thread); + + while (wqueue->last_thread->next != thread) + { + pagecache_pthread_cond_wait(&thread->suspend, &pagecache->cache_lock); + } +#endif + + pagecache->resize_in_flush= 1; + if (flush_all_key_blocks(pagecache)) + { + /* TODO: if this happens, we should write a warning in the log file ! */ + pagecache->resize_in_flush= 0; + blocks= 0; + pagecache->can_be_used= 0; + goto finish; + } + pagecache->resize_in_flush= 0; + pagecache->can_be_used= 0; +#ifdef THREAD + while (pagecache->cnt_for_resize_op) + { + KEYCACHE_DBUG_PRINT("resize_pagecache: wait", + ("suspend thread %ld", thread->id)); + pagecache_pthread_cond_wait(&thread->suspend, &pagecache->cache_lock); + } +#else + KEYCACHE_DBUG_ASSERT(pagecache->cnt_for_resize_op == 0); +#endif + + end_pagecache(pagecache, 0); /* Don't free mutex */ + /* The following will work even if use_mem is 0 */ + blocks= init_pagecache(pagecache, pagecache->block_size, use_mem, + division_limit, age_threshold, + pagecache->loghandler); + +finish: +#ifdef THREAD + unlink_from_queue(wqueue, thread); + /* Signal for the next resize request to proceeed if any */ + if (wqueue->last_thread) + { + KEYCACHE_DBUG_PRINT("resize_pagecache: signal", + ("thread %ld", wqueue->last_thread->next->id)); + pagecache_pthread_cond_signal(&wqueue->last_thread->next->suspend); + } +#endif + pagecache_pthread_mutex_unlock(&pagecache->cache_lock); + DBUG_RETURN(blocks); +} + + +/* + Increment counter blocking resize key cache operation +*/ +static inline void inc_counter_for_resize_op(PAGECACHE *pagecache) +{ + pagecache->cnt_for_resize_op++; +} + + +/* + Decrement counter blocking resize key cache operation; + Signal the operation to proceed when counter becomes equal zero +*/ +static inline void dec_counter_for_resize_op(PAGECACHE *pagecache) +{ +#ifdef THREAD + struct st_my_thread_var *last_thread; + if (!--pagecache->cnt_for_resize_op && + (last_thread= pagecache->resize_queue.last_thread)) + { + KEYCACHE_DBUG_PRINT("dec_counter_for_resize_op: signal", + ("thread %ld", last_thread->next->id)); + pagecache_pthread_cond_signal(&last_thread->next->suspend); + } +#else + pagecache->cnt_for_resize_op--; +#endif +} + +/* + Change the page cache parameters + + SYNOPSIS + change_pagecache_param() + pagecache pointer to a page cache data structure + division_limit new division limit (if not zero) + age_threshold new age threshold (if not zero) + + RETURN VALUE + none + + NOTES. + Presently the function resets the key cache parameters + concerning midpoint insertion strategy - division_limit and + age_threshold. +*/ + +void change_pagecache_param(PAGECACHE *pagecache, uint division_limit, + uint age_threshold) +{ + DBUG_ENTER("change_pagecache_param"); + + pagecache_pthread_mutex_lock(&pagecache->cache_lock); + if (division_limit) + pagecache->min_warm_blocks= (pagecache->disk_blocks * + division_limit / 100 + 1); + if (age_threshold) + pagecache->age_threshold= (pagecache->disk_blocks * + age_threshold / 100); + pagecache_pthread_mutex_unlock(&pagecache->cache_lock); + DBUG_VOID_RETURN; +} + + +/* + Remove page cache from memory + + SYNOPSIS + end_pagecache() + pagecache page cache handle + cleanup Complete free (Free also mutex for key cache) + + RETURN VALUE + none +*/ + +void end_pagecache(PAGECACHE *pagecache, my_bool cleanup) +{ + DBUG_ENTER("end_pagecache"); + DBUG_PRINT("enter", ("key_cache: 0x%lx", pagecache)); + + if (!pagecache->inited) + DBUG_VOID_RETURN; + + if (pagecache->disk_blocks > 0) + { + if (pagecache->block_mem) + { + my_large_free((gptr) pagecache->block_mem, MYF(0)); + pagecache->block_mem= NULL; + my_free((gptr) pagecache->block_root, MYF(0)); + pagecache->block_root= NULL; + } + pagecache->disk_blocks= -1; + /* Reset blocks_changed to be safe if flush_all_key_blocks is called */ + pagecache->blocks_changed= 0; + } + + DBUG_PRINT("status", ("used: %d changed: %d w_requests: %lu " + "writes: %lu r_requests: %lu reads: %lu", + pagecache->blocks_used, pagecache->global_blocks_changed, + (ulong) pagecache->global_cache_w_requests, + (ulong) pagecache->global_cache_write, + (ulong) pagecache->global_cache_r_requests, + (ulong) pagecache->global_cache_read)); + + if (cleanup) + { + pthread_mutex_destroy(&pagecache->cache_lock); + pagecache->inited= pagecache->can_be_used= 0; + PAGECACHE_DEBUG_CLOSE; + } + DBUG_VOID_RETURN; +} /* end_pagecache */ + + +#ifdef THREAD +/* + Link a thread into double-linked queue of waiting threads. + + SYNOPSIS + link_into_queue() + wqueue pointer to the queue structure + thread pointer to the thread to be added to the queue + + RETURN VALUE + none + + NOTES. + Queue is represented by a circular list of the thread structures + The list is double-linked of the type (**prev,*next), accessed by + a pointer to the last element. +*/ + +static void link_into_queue(PAGECACHE_WQUEUE *wqueue, + struct st_my_thread_var *thread) +{ + struct st_my_thread_var *last; + if (! (last= wqueue->last_thread)) + { + /* Queue is empty */ + thread->next= thread; + thread->prev= &thread->next; + } + else + { + thread->prev= last->next->prev; + last->next->prev= &thread->next; + thread->next= last->next; + last->next= thread; + } + wqueue->last_thread= thread; +} + +/* + Unlink a thread from double-linked queue of waiting threads + + SYNOPSIS + unlink_from_queue() + wqueue pointer to the queue structure + thread pointer to the thread to be removed from the queue + + RETURN VALUE + none + + NOTES. + See NOTES for link_into_queue +*/ + +static void unlink_from_queue(PAGECACHE_WQUEUE *wqueue, + struct st_my_thread_var *thread) +{ + KEYCACHE_DBUG_PRINT("unlink_from_queue", ("thread %ld", thread->id)); + if (thread->next == thread) + /* The queue contains only one member */ + wqueue->last_thread= NULL; + else + { + thread->next->prev= thread->prev; + *thread->prev=thread->next; + if (wqueue->last_thread == thread) + wqueue->last_thread= STRUCT_PTR(struct st_my_thread_var, next, + thread->prev); + } + thread->next= NULL; +} + + +/* + Add a thread to single-linked queue of waiting threads + + SYNOPSIS + add_to_queue() + wqueue pointer to the queue structure + thread pointer to the thread to be added to the queue + + RETURN VALUE + none + + NOTES. + Queue is represented by a circular list of the thread structures + The list is single-linked of the type (*next), accessed by a pointer + to the last element. +*/ + +static inline void add_to_queue(PAGECACHE_WQUEUE *wqueue, + struct st_my_thread_var *thread) +{ + struct st_my_thread_var *last; + if (! (last= wqueue->last_thread)) + thread->next= thread; + else + { + thread->next= last->next; + last->next= thread; + } + wqueue->last_thread= thread; +} + + +/* + Remove all threads from queue signaling them to proceed + + SYNOPSIS + realease_queue() + wqueue pointer to the queue structure + thread pointer to the thread to be added to the queue + + RETURN VALUE + none + + NOTES. + See notes for add_to_queue + When removed from the queue each thread is signaled via condition + variable thread->suspend. +*/ + +static void release_queue(PAGECACHE_WQUEUE *wqueue) +{ + struct st_my_thread_var *last= wqueue->last_thread; + struct st_my_thread_var *next= last->next; + struct st_my_thread_var *thread; + do + { + thread=next; + KEYCACHE_DBUG_PRINT("release_queue: signal", ("thread %ld", thread->id)); + pagecache_pthread_cond_signal(&thread->suspend); + next=thread->next; + thread->next= NULL; + } + while (thread != last); + wqueue->last_thread= NULL; +} +#endif + + +/* + Unlink a block from the chain of dirty/clean blocks +*/ + +static inline void unlink_changed(PAGECACHE_BLOCK_LINK *block) +{ + if (block->next_changed) + block->next_changed->prev_changed= block->prev_changed; + *block->prev_changed= block->next_changed; +} + + +/* + Link a block into the chain of dirty/clean blocks +*/ + +static inline void link_changed(PAGECACHE_BLOCK_LINK *block, + PAGECACHE_BLOCK_LINK **phead) +{ + block->prev_changed= phead; + if ((block->next_changed= *phead)) + (*phead)->prev_changed= &block->next_changed; + *phead= block; +} + + +/* + Unlink a block from the chain of dirty/clean blocks, if it's asked for, + and link it to the chain of clean blocks for the specified file +*/ + +static void link_to_file_list(PAGECACHE *pagecache, + PAGECACHE_BLOCK_LINK *block, + PAGECACHE_FILE *file, my_bool unlink) +{ + if (unlink) + unlink_changed(block); + link_changed(block, &pagecache->file_blocks[FILE_HASH(*file)]); + if (block->status & BLOCK_CHANGED) + { + block->status&= ~BLOCK_CHANGED; + pagecache->blocks_changed--; + pagecache->global_blocks_changed--; + } +} + + +/* + Unlink a block from the chain of clean blocks for the specified + file and link it to the chain of dirty blocks for this file +*/ + +static inline void link_to_changed_list(PAGECACHE *pagecache, + PAGECACHE_BLOCK_LINK *block) +{ + unlink_changed(block); + link_changed(block, + &pagecache->changed_blocks[FILE_HASH(block->hash_link->file)]); + block->status|=BLOCK_CHANGED; + pagecache->blocks_changed++; + pagecache->global_blocks_changed++; +} + + +/* + Link a block to the LRU chain at the beginning or at the end of + one of two parts. + + SYNOPSIS + link_block() + pagecache pointer to a page cache data structure + block pointer to the block to link to the LRU chain + hot <-> to link the block into the hot subchain + at_end <-> to link the block at the end of the subchain + + RETURN VALUE + none + + NOTES. + The LRU chain is represented by a curcular list of block structures. + The list is double-linked of the type (**prev,*next) type. + The LRU chain is divided into two parts - hot and warm. + There are two pointers to access the last blocks of these two + parts. The beginning of the warm part follows right after the + end of the hot part. + Only blocks of the warm part can be used for replacement. + The first block from the beginning of this subchain is always + taken for eviction (pagecache->last_used->next) + + LRU chain: +------+ H O T +------+ + +----| end |----...<----| beg |----+ + | +------+last +------+ | + v<-link in latest hot (new end) | + | link in latest warm (new end)->^ + | +------+ W A R M +------+ | + +----| beg |---->...----| end |----+ + +------+ +------+ins + first for eviction +*/ + +static void link_block(PAGECACHE *pagecache, PAGECACHE_BLOCK_LINK *block, + my_bool hot, my_bool at_end) +{ + PAGECACHE_BLOCK_LINK *ins; + PAGECACHE_BLOCK_LINK **pins; + + KEYCACHE_DBUG_ASSERT(! (block->hash_link && block->hash_link->requests)); +#ifdef THREAD + if (!hot && pagecache->waiting_for_block.last_thread) + { + /* Signal that in the LRU warm sub-chain an available block has appeared */ + struct st_my_thread_var *last_thread= + pagecache->waiting_for_block.last_thread; + struct st_my_thread_var *first_thread= last_thread->next; + struct st_my_thread_var *next_thread= first_thread; + PAGECACHE_HASH_LINK *hash_link= + (PAGECACHE_HASH_LINK *) first_thread->opt_info; + struct st_my_thread_var *thread; + do + { + thread= next_thread; + next_thread= thread->next; + /* + We notify about the event all threads that ask + for the same page as the first thread in the queue + */ + if ((PAGECACHE_HASH_LINK *) thread->opt_info == hash_link) + { + KEYCACHE_DBUG_PRINT("link_block: signal", ("thread %ld", thread->id)); + pagecache_pthread_cond_signal(&thread->suspend); + unlink_from_queue(&pagecache->waiting_for_block, thread); + block->requests++; + } + } + while (thread != last_thread); + hash_link->block= block; + KEYCACHE_THREAD_TRACE("link_block: after signaling"); +#if defined(PAGECACHE_DEBUG) + KEYCACHE_DBUG_PRINT("link_block", + ("linked,unlinked block %u status=%x #requests=%u #available=%u", + BLOCK_NUMBER(pagecache, block), block->status, + block->requests, pagecache->blocks_available)); +#endif + return; + } +#else /* THREAD */ + KEYCACHE_DBUG_ASSERT(! (!hot && pagecache->waiting_for_block.last_thread)); + /* Condition not transformed using DeMorgan, to keep the text identical */ +#endif /* THREAD */ + pins= hot ? &pagecache->used_ins : &pagecache->used_last; + ins= *pins; + if (ins) + { + ins->next_used->prev_used= &block->next_used; + block->next_used= ins->next_used; + block->prev_used= &ins->next_used; + ins->next_used= block; + if (at_end) + *pins= block; + } + else + { + /* The LRU chain is empty */ + pagecache->used_last= pagecache->used_ins= block->next_used= block; + block->prev_used= &block->next_used; + } + KEYCACHE_THREAD_TRACE("link_block"); +#if defined(PAGECACHE_DEBUG) + pagecache->blocks_available++; + KEYCACHE_DBUG_PRINT("link_block", + ("linked block %u:%1u status=%x #requests=%u #available=%u", + BLOCK_NUMBER(pagecache, block), at_end, block->status, + block->requests, pagecache->blocks_available)); + KEYCACHE_DBUG_ASSERT((ulong) pagecache->blocks_available <= + pagecache->blocks_used); +#endif +} + + +/* + Unlink a block from the LRU chain + + SYNOPSIS + unlink_block() + pagecache pointer to a page cache data structure + block pointer to the block to unlink from the LRU chain + + RETURN VALUE + none + + NOTES. + See NOTES for link_block +*/ + +static void unlink_block(PAGECACHE *pagecache, PAGECACHE_BLOCK_LINK *block) +{ + if (block->next_used == block) + /* The list contains only one member */ + pagecache->used_last= pagecache->used_ins= NULL; + else + { + block->next_used->prev_used= block->prev_used; + *block->prev_used= block->next_used; + if (pagecache->used_last == block) + pagecache->used_last= STRUCT_PTR(PAGECACHE_BLOCK_LINK, + next_used, block->prev_used); + if (pagecache->used_ins == block) + pagecache->used_ins= STRUCT_PTR(PAGECACHE_BLOCK_LINK, + next_used, block->prev_used); + } + block->next_used= NULL; + + KEYCACHE_THREAD_TRACE("unlink_block"); +#if defined(PAGECACHE_DEBUG) + pagecache->blocks_available--; + KEYCACHE_DBUG_PRINT("unlink_block", + ("unlinked block 0x%lx (%u) status=%x #requests=%u #available=%u", + (ulong)block, BLOCK_NUMBER(pagecache, block), block->status, + block->requests, pagecache->blocks_available)); + BLOCK_INFO(block); + KEYCACHE_DBUG_ASSERT(pagecache->blocks_available >= 0); +#endif +} + + +/* + Register requests for a block +*/ +static void reg_requests(PAGECACHE *pagecache, PAGECACHE_BLOCK_LINK *block, + int count) +{ + DBUG_ENTER("reg_requests"); + DBUG_PRINT("enter", ("block 0x%lx (%u) status=%x, reqs: %u", + (ulong)block, BLOCK_NUMBER(pagecache, block), + block->status, block->requests)); + BLOCK_INFO(block); + if (! block->requests) + /* First request for the block unlinks it */ + unlink_block(pagecache, block); + block->requests+=count; + DBUG_VOID_RETURN; +} + + +/* + Unregister request for a block + linking it to the LRU chain if it's the last request + + SYNOPSIS + unreg_request() + pagecache pointer to a page cache data structure + block pointer to the block to link to the LRU chain + at_end <-> to link the block at the end of the LRU chain + + RETURN VALUE + none + + NOTES. + Every linking to the LRU chain decrements by one a special block + counter (if it's positive). If the at_end parameter is TRUE the block is + added either at the end of warm sub-chain or at the end of hot sub-chain. + It is added to the hot subchain if its counter is zero and number of + blocks in warm sub-chain is not less than some low limit (determined by + the division_limit parameter). Otherwise the block is added to the warm + sub-chain. If the at_end parameter is FALSE the block is always added + at beginning of the warm sub-chain. + Thus a warm block can be promoted to the hot sub-chain when its counter + becomes zero for the first time. + At the same time the block at the very beginning of the hot subchain + might be moved to the beginning of the warm subchain if it stays untouched + for a too long time (this time is determined by parameter age_threshold). +*/ + +static void unreg_request(PAGECACHE *pagecache, + PAGECACHE_BLOCK_LINK *block, int at_end) +{ + DBUG_ENTER("unreg_request"); + DBUG_PRINT("enter", ("block 0x%lx (%u) status=%x, reqs: %u", + (ulong)block, BLOCK_NUMBER(pagecache, block), + block->status, block->requests)); + BLOCK_INFO(block); + if (! --block->requests) + { + my_bool hot; + if (block->hits_left) + block->hits_left--; + hot= !block->hits_left && at_end && + pagecache->warm_blocks > pagecache->min_warm_blocks; + if (hot) + { + if (block->temperature == BLOCK_WARM) + pagecache->warm_blocks--; + block->temperature= BLOCK_HOT; + KEYCACHE_DBUG_PRINT("unreg_request", ("#warm_blocks=%u", + pagecache->warm_blocks)); + } + link_block(pagecache, block, hot, (my_bool)at_end); + block->last_hit_time= pagecache->time; + pagecache->time++; + + block= pagecache->used_ins; + /* Check if we should link a hot block to the warm block */ + if (block && pagecache->time - block->last_hit_time > + pagecache->age_threshold) + { + unlink_block(pagecache, block); + link_block(pagecache, block, 0, 0); + if (block->temperature != BLOCK_WARM) + { + pagecache->warm_blocks++; + block->temperature= BLOCK_WARM; + } + KEYCACHE_DBUG_PRINT("unreg_request", ("#warm_blocks=%u", + pagecache->warm_blocks)); + } + } + DBUG_VOID_RETURN; +} + +/* + Remove a reader of the page in block +*/ + +static inline void remove_reader(PAGECACHE_BLOCK_LINK *block) +{ + if (! --block->hash_link->requests && block->condvar) + pagecache_pthread_cond_signal(block->condvar); +} + + +/* + Wait until the last reader of the page in block + signals on its termination +*/ + +static inline void wait_for_readers(PAGECACHE *pagecache, + PAGECACHE_BLOCK_LINK *block) +{ +#ifdef THREAD + struct st_my_thread_var *thread= my_thread_var; + while (block->hash_link->requests) + { + KEYCACHE_DBUG_PRINT("wait_for_readers: wait", + ("suspend thread %ld block %u", + thread->id, BLOCK_NUMBER(pagecache, block))); + block->condvar= &thread->suspend; + pagecache_pthread_cond_wait(&thread->suspend, &pagecache->cache_lock); + block->condvar= NULL; + } +#else + KEYCACHE_DBUG_ASSERT(block->hash_link->requests == 0); +#endif +} + + +/* + Add a hash link to a bucket in the hash_table +*/ + +static inline void link_hash(PAGECACHE_HASH_LINK **start, + PAGECACHE_HASH_LINK *hash_link) +{ + if (*start) + (*start)->prev= &hash_link->next; + hash_link->next= *start; + hash_link->prev= start; + *start= hash_link; +} + + +/* + Remove a hash link from the hash table +*/ + +static void unlink_hash(PAGECACHE *pagecache, PAGECACHE_HASH_LINK *hash_link) +{ + KEYCACHE_DBUG_PRINT("unlink_hash", ("fd: %u pos_ %lu #requests=%u", + (uint) hash_link->file.file, (ulong) hash_link->pageno, + hash_link->requests)); + KEYCACHE_DBUG_ASSERT(hash_link->requests == 0); + if ((*hash_link->prev= hash_link->next)) + hash_link->next->prev= hash_link->prev; + hash_link->block= NULL; +#ifdef THREAD + if (pagecache->waiting_for_hash_link.last_thread) + { + /* Signal that a free hash link has appeared */ + struct st_my_thread_var *last_thread= + pagecache->waiting_for_hash_link.last_thread; + struct st_my_thread_var *first_thread= last_thread->next; + struct st_my_thread_var *next_thread= first_thread; + PAGECACHE_PAGE *first_page= (PAGECACHE_PAGE *) (first_thread->opt_info); + struct st_my_thread_var *thread; + + hash_link->file= first_page->file; + hash_link->pageno= first_page->pageno; + do + { + PAGECACHE_PAGE *page; + thread= next_thread; + page= (PAGECACHE_PAGE *) thread->opt_info; + next_thread= thread->next; + /* + We notify about the event all threads that ask + for the same page as the first thread in the queue + */ + if (page->file.file == hash_link->file.file && + page->pageno == hash_link->pageno) + { + KEYCACHE_DBUG_PRINT("unlink_hash: signal", ("thread %ld", thread->id)); + pagecache_pthread_cond_signal(&thread->suspend); + unlink_from_queue(&pagecache->waiting_for_hash_link, thread); + } + } + while (thread != last_thread); + link_hash(&pagecache->hash_root[PAGECACHE_HASH(pagecache, + hash_link->file, + hash_link->pageno)], + hash_link); + return; + } +#else /* THREAD */ + KEYCACHE_DBUG_ASSERT(! (pagecache->waiting_for_hash_link.last_thread)); +#endif /* THREAD */ + hash_link->next= pagecache->free_hash_list; + pagecache->free_hash_list= hash_link; +} +/* + Get the hash link for the page if it is inthe cache +*/ + +static PAGECACHE_HASH_LINK *get_present_hash_link(PAGECACHE *pagecache, + PAGECACHE_FILE *file, + maria_page_no_t pageno, + PAGECACHE_HASH_LINK ***start) +{ + reg1 PAGECACHE_HASH_LINK *hash_link; +#if defined(PAGECACHE_DEBUG) + int cnt; +#endif + DBUG_ENTER("get_present_hash_link"); + + KEYCACHE_DBUG_PRINT("get_present_hash_link", ("fd: %u pos: %lu", + (uint) file->file, (ulong) pageno)); + + /* + Find the bucket in the hash table for the pair (file, pageno); + start contains the head of the bucket list, + hash_link points to the first member of the list + */ + hash_link= *(*start= &pagecache->hash_root[PAGECACHE_HASH(pagecache, + *file, pageno)]); +#if defined(PAGECACHE_DEBUG) + cnt= 0; +#endif + /* Look for an element for the pair (file, pageno) in the bucket chain */ + while (hash_link && + (hash_link->pageno != pageno || + hash_link->file.file != file->file)) + { + hash_link= hash_link->next; +#if defined(PAGECACHE_DEBUG) + cnt++; + if (! (cnt <= pagecache->hash_links_used)) + { + int i; + for (i=0, hash_link= **start ; + i < cnt ; i++, hash_link= hash_link->next) + { + KEYCACHE_DBUG_PRINT("get_present_hash_link", ("fd: %u pos: %lu", + (uint) hash_link->file.file, (ulong) hash_link->pageno)); + } + } + KEYCACHE_DBUG_ASSERT(cnt <= pagecache->hash_links_used); +#endif + } + DBUG_RETURN(hash_link); +} + + +/* + Get the hash link for a page +*/ + +static PAGECACHE_HASH_LINK *get_hash_link(PAGECACHE *pagecache, + PAGECACHE_FILE *file, + maria_page_no_t pageno) +{ + reg1 PAGECACHE_HASH_LINK *hash_link; + PAGECACHE_HASH_LINK **start; + PAGECACHE_PAGE page; + + KEYCACHE_DBUG_PRINT("get_hash_link", ("fd: %u pos: %lu", + (uint) file->file, (ulong) pageno)); + +restart: + /* try to find the page in the cache */ + hash_link= get_present_hash_link(pagecache, file, pageno, + &start); + if (! hash_link) + { + /* There is no hash link in the hash table for the pair (file, pageno) */ + if (pagecache->free_hash_list) + { + hash_link= pagecache->free_hash_list; + pagecache->free_hash_list= hash_link->next; + } + else if (pagecache->hash_links_used < pagecache->hash_links) + { + hash_link= &pagecache->hash_link_root[pagecache->hash_links_used++]; + } + else + { +#ifdef THREAD + /* Wait for a free hash link */ + struct st_my_thread_var *thread= my_thread_var; + KEYCACHE_DBUG_PRINT("get_hash_link", ("waiting")); + page.file= *file; + page.pageno= pageno; + thread->opt_info= (void *) &page; + link_into_queue(&pagecache->waiting_for_hash_link, thread); + KEYCACHE_DBUG_PRINT("get_hash_link: wait", + ("suspend thread %ld", thread->id)); + pagecache_pthread_cond_wait(&thread->suspend, + &pagecache->cache_lock); + thread->opt_info= NULL; +#else + KEYCACHE_DBUG_ASSERT(0); +#endif + goto restart; + } + hash_link->file= *file; + hash_link->pageno= pageno; + link_hash(start, hash_link); + } + /* Register the request for the page */ + hash_link->requests++; + + return hash_link; +} + + +/* + Get a block for the file page requested by a pagecache read/write operation; + If the page is not in the cache return a free block, if there is none + return the lru block after saving its buffer if the page is dirty. + + SYNOPSIS + + find_key_block() + pagecache pointer to a page cache data structure + file handler for the file to read page from + pageno number of the page in the file + init_hits_left how initialize the block counter for the page + wrmode <-> get for writing + reg_req Register request to thye page + page_st out {PAGE_READ,PAGE_TO_BE_READ,PAGE_WAIT_TO_BE_READ} + + RETURN VALUE + Pointer to the found block if successful, 0 - otherwise + + NOTES. + For the page from file positioned at pageno the function checks whether + the page is in the key cache specified by the first parameter. + If this is the case it immediately returns the block. + If not, the function first chooses a block for this page. If there is + no not used blocks in the key cache yet, the function takes the block + at the very beginning of the warm sub-chain. It saves the page in that + block if it's dirty before returning the pointer to it. + The function returns in the page_st parameter the following values: + PAGE_READ - if page already in the block, + PAGE_TO_BE_READ - if it is to be read yet by the current thread + WAIT_TO_BE_READ - if it is to be read by another thread + If an error occurs THE BLOCK_ERROR bit is set in the block status. + It might happen that there are no blocks in LRU chain (in warm part) - + all blocks are unlinked for some read/write operations. Then the function + waits until first of this operations links any block back. +*/ + +static PAGECACHE_BLOCK_LINK *find_key_block(PAGECACHE *pagecache, + PAGECACHE_FILE *file, + maria_page_no_t pageno, + int init_hits_left, + my_bool wrmode, + my_bool reg_req, + int *page_st) +{ + PAGECACHE_HASH_LINK *hash_link; + PAGECACHE_BLOCK_LINK *block; + int error= 0; + int page_status; + + DBUG_ENTER("find_key_block"); + KEYCACHE_THREAD_TRACE("find_key_block:begin"); + DBUG_PRINT("enter", ("fd: %u pos %lu wrmode: %lu", + (uint) file->file, (ulong) pageno, (uint) wrmode)); + KEYCACHE_DBUG_PRINT("find_key_block", ("fd: %u pos: %lu wrmode: %lu", + (uint) file->file, (ulong) pageno, + (uint) wrmode)); +#if !defined(DBUG_OFF) && defined(EXTRA_DEBUG) + DBUG_EXECUTE("check_pagecache", + test_key_cache(pagecache, "start of find_key_block", 0);); +#endif + +restart: + /* Find the hash link for the requested page (file, pageno) */ + hash_link= get_hash_link(pagecache, file, pageno); + + page_status= -1; + if ((block= hash_link->block) && + block->hash_link == hash_link && (block->status & BLOCK_READ)) + page_status= PAGE_READ; + + if (wrmode && pagecache->resize_in_flush) + { + /* This is a write request during the flush phase of a resize operation */ + + if (page_status != PAGE_READ) + { + /* We don't need the page in the cache: we are going to write on disk */ + hash_link->requests--; + unlink_hash(pagecache, hash_link); + return 0; + } + if (!(block->status & BLOCK_IN_FLUSH)) + { + hash_link->requests--; + /* + Remove block to invalidate the page in the block buffer + as we are going to write directly on disk. + Although we have an exlusive lock for the updated key part + the control can be yieded by the current thread as we might + have unfinished readers of other key parts in the block + buffer. Still we are guaranteed not to have any readers + of the key part we are writing into until the block is + removed from the cache as we set the BLOCK_REASSIGNED + flag (see the code below that handles reading requests). + */ + free_block(pagecache, block); + return 0; + } + /* Wait intil the page is flushed on disk */ + hash_link->requests--; + { +#ifdef THREAD + struct st_my_thread_var *thread= my_thread_var; + add_to_queue(&block->wqueue[COND_FOR_SAVED], thread); + do + { + KEYCACHE_DBUG_PRINT("find_key_block: wait", + ("suspend thread %ld", thread->id)); + pagecache_pthread_cond_wait(&thread->suspend, + &pagecache->cache_lock); + } + while(thread->next); +#else + KEYCACHE_DBUG_ASSERT(0); + /* + Given the use of "resize_in_flush", it seems impossible + that this whole branch is ever entered in single-threaded case + because "(wrmode && pagecache->resize_in_flush)" cannot be true. + TODO: Check this, and then put the whole branch into the + "#ifdef THREAD" guard. + */ +#endif + } + /* Invalidate page in the block if it has not been done yet */ + if (block->status) + free_block(pagecache, block); + return 0; + } + + if (page_status == PAGE_READ && + (block->status & (BLOCK_IN_SWITCH | BLOCK_REASSIGNED))) + { + /* This is a request for a page to be removed from cache */ + + KEYCACHE_DBUG_PRINT("find_key_block", + ("request for old page in block %u " + "wrmode: %d block->status: %d", + BLOCK_NUMBER(pagecache, block), wrmode, + block->status)); + /* + Only reading requests can proceed until the old dirty page is flushed, + all others are to be suspended, then resubmitted + */ + if (!wrmode && !(block->status & BLOCK_REASSIGNED)) + { + if (reg_req) + reg_requests(pagecache, block, 1); + } + else + { + hash_link->requests--; + KEYCACHE_DBUG_PRINT("find_key_block", + ("request waiting for old page to be saved")); + { +#ifdef THREAD + struct st_my_thread_var *thread= my_thread_var; + /* Put the request into the queue of those waiting for the old page */ + add_to_queue(&block->wqueue[COND_FOR_SAVED], thread); + /* Wait until the request can be resubmitted */ + do + { + KEYCACHE_DBUG_PRINT("find_key_block: wait", + ("suspend thread %ld", thread->id)); + pagecache_pthread_cond_wait(&thread->suspend, + &pagecache->cache_lock); + } + while(thread->next); +#else + KEYCACHE_DBUG_ASSERT(0); + /* No parallel requests in single-threaded case */ +#endif + } + KEYCACHE_DBUG_PRINT("find_key_block", + ("request for old page resubmitted")); + /* Resubmit the request */ + goto restart; + } + } + else + { + /* This is a request for a new page or for a page not to be removed */ + if (! block) + { + /* No block is assigned for the page yet */ + if (pagecache->blocks_unused) + { + if (pagecache->free_block_list) + { + /* There is a block in the free list. */ + block= pagecache->free_block_list; + pagecache->free_block_list= block->next_used; + block->next_used= NULL; + } + else + { + /* There are some never used blocks, take first of them */ + block= &pagecache->block_root[pagecache->blocks_used]; + block->buffer= ADD_TO_PTR(pagecache->block_mem, + ((ulong) pagecache->blocks_used* + pagecache->block_size), + byte*); + pagecache->blocks_used++; + } + pagecache->blocks_unused--; + DBUG_ASSERT((block->status & BLOCK_WRLOCK) == 0); + block->status= 0; +#ifndef DBUG_OFF + block->type= PAGECACHE_EMPTY_PAGE; +#endif + block->requests= 1; + block->temperature= BLOCK_COLD; + block->hits_left= init_hits_left; + block->last_hit_time= 0; + link_to_file_list(pagecache, block, file, 0); + block->hash_link= hash_link; + hash_link->block= block; + page_status= PAGE_TO_BE_READ; + KEYCACHE_DBUG_PRINT("find_key_block", + ("got free or never used block %u", + BLOCK_NUMBER(pagecache, block))); + } + else + { + /* There are no never used blocks, use a block from the LRU chain */ + + /* + Wait until a new block is added to the LRU chain; + several threads might wait here for the same page, + all of them must get the same block + */ + +#ifdef THREAD + if (! pagecache->used_last) + { + struct st_my_thread_var *thread= my_thread_var; + thread->opt_info= (void *) hash_link; + link_into_queue(&pagecache->waiting_for_block, thread); + do + { + KEYCACHE_DBUG_PRINT("find_key_block: wait", + ("suspend thread %ld", thread->id)); + pagecache_pthread_cond_wait(&thread->suspend, + &pagecache->cache_lock); + } + while (thread->next); + thread->opt_info= NULL; + } +#else + KEYCACHE_DBUG_ASSERT(pagecache->used_last); +#endif + block= hash_link->block; + if (! block) + { + /* + Take the first block from the LRU chain + unlinking it from the chain + */ + block= pagecache->used_last->next_used; + block->hits_left= init_hits_left; + block->last_hit_time= 0; + if (reg_req) + reg_requests(pagecache, block,1); + hash_link->block= block; + } + + if (block->hash_link != hash_link && + ! (block->status & BLOCK_IN_SWITCH) ) + { + /* this is a primary request for a new page */ + block->status|= BLOCK_IN_SWITCH; + + KEYCACHE_DBUG_PRINT("find_key_block", + ("got block %u for new page", + BLOCK_NUMBER(pagecache, block))); + + if (block->status & BLOCK_CHANGED) + { + /* The block contains a dirty page - push it out of the cache */ + + KEYCACHE_DBUG_PRINT("find_key_block", ("block is dirty")); + + pagecache_pthread_mutex_unlock(&pagecache->cache_lock); + /* + The call is thread safe because only the current + thread might change the block->hash_link value + */ + DBUG_ASSERT(block->pins == 0); + error= pagecache_fwrite(pagecache, + &block->hash_link->file, + block->buffer, + block->hash_link->pageno, + block->type, + MYF(MY_NABP | MY_WAIT_IF_FULL)); + pagecache_pthread_mutex_lock(&pagecache->cache_lock); + pagecache->global_cache_write++; + } + + block->status|= BLOCK_REASSIGNED; + if (block->hash_link) + { + /* + Wait until all pending read requests + for this page are executed + (we could have avoided this waiting, if we had read + a page in the cache in a sweep, without yielding control) + */ + wait_for_readers(pagecache, block); + + /* Remove the hash link for this page from the hash table */ + unlink_hash(pagecache, block->hash_link); + /* All pending requests for this page must be resubmitted */ + if (block->wqueue[COND_FOR_SAVED].last_thread) + release_queue(&block->wqueue[COND_FOR_SAVED]); + } + link_to_file_list(pagecache, block, file, + (my_bool)(block->hash_link ? 1 : 0)); + DBUG_ASSERT((block->status & BLOCK_WRLOCK) == 0); + block->status= error? BLOCK_ERROR : 0; +#ifndef DBUG_OFF + block->type= PAGECACHE_EMPTY_PAGE; +#endif + block->hash_link= hash_link; + page_status= PAGE_TO_BE_READ; + + KEYCACHE_DBUG_ASSERT(block->hash_link->block == block); + KEYCACHE_DBUG_ASSERT(hash_link->block->hash_link == hash_link); + } + else + { + /* This is for secondary requests for a new page only */ + KEYCACHE_DBUG_PRINT("find_key_block", + ("block->hash_link: %p hash_link: %p " + "block->status: %u", block->hash_link, + hash_link, block->status )); + page_status= (((block->hash_link == hash_link) && + (block->status & BLOCK_READ)) ? + PAGE_READ : PAGE_WAIT_TO_BE_READ); + } + } + pagecache->global_cache_read++; + } + else + { + if (reg_req) + reg_requests(pagecache, block, 1); + KEYCACHE_DBUG_PRINT("find_key_block", + ("block->hash_link: %p hash_link: %p " + "block->status: %u", block->hash_link, + hash_link, block->status )); + page_status= (((block->hash_link == hash_link) && + (block->status & BLOCK_READ)) ? + PAGE_READ : PAGE_WAIT_TO_BE_READ); + } + } + + KEYCACHE_DBUG_ASSERT(page_status != -1); + *page_st=page_status; + DBUG_PRINT("info", + ("block: 0x%lx fd: %u pos %lu block->status %u page_status %lu", + (ulong) block, (uint) file->file, + (ulong) pageno, block->status, (uint) page_status)); + KEYCACHE_DBUG_PRINT("find_key_block", + ("block: 0x%lx fd: %u pos %lu block->status %u page_status %lu", + (ulong) block, + (uint) file->file, (ulong) pageno, block->status, + (uint) page_status)); + +#if !defined(DBUG_OFF) && defined(EXTRA_DEBUG) + DBUG_EXECUTE("check_pagecache", + test_key_cache(pagecache, "end of find_key_block",0);); +#endif + KEYCACHE_THREAD_TRACE("find_key_block:end"); + DBUG_RETURN(block); +} + + +void pagecache_add_pin(PAGECACHE_BLOCK_LINK *block) +{ + DBUG_ENTER("pagecache_add_pin"); + DBUG_PRINT("enter", ("block 0x%lx pins: %u", + (ulong) block, + block->pins)); + BLOCK_INFO(block); + block->pins++; +#ifdef PAGECACHE_DEBUG + { + PAGECACHE_PIN_INFO *info= + (PAGECACHE_PIN_INFO *)my_malloc(sizeof(PAGECACHE_PIN_INFO), MYF(0)); + info->thread= my_thread_var; + info_link(&block->pin_list, info); + } +#endif + DBUG_VOID_RETURN; +} + +void pagecache_remove_pin(PAGECACHE_BLOCK_LINK *block) +{ + DBUG_ENTER("pagecache_remove_pin"); + DBUG_PRINT("enter", ("block 0x%lx pins: %u", + (ulong) block, + block->pins)); + BLOCK_INFO(block); + DBUG_ASSERT(block->pins > 0); + block->pins--; +#ifdef PAGECACHE_DEBUG + { + PAGECACHE_PIN_INFO *info= info_find(block->pin_list, my_thread_var); + DBUG_ASSERT(info != 0); + info_unlink(info); + my_free((gptr) info, MYF(0)); + } +#endif + DBUG_VOID_RETURN; +} +#ifdef PAGECACHE_DEBUG +void pagecache_add_lock(PAGECACHE_BLOCK_LINK *block, bool wl) +{ + PAGECACHE_LOCK_INFO *info= + (PAGECACHE_LOCK_INFO *)my_malloc(sizeof(PAGECACHE_LOCK_INFO), MYF(0)); + info->thread= my_thread_var; + info->write_lock= wl; + info_link((PAGECACHE_PIN_INFO **)&block->lock_list, + (PAGECACHE_PIN_INFO *)info); +} +void pagecache_remove_lock(PAGECACHE_BLOCK_LINK *block) +{ + PAGECACHE_LOCK_INFO *info= + (PAGECACHE_LOCK_INFO *)info_find((PAGECACHE_PIN_INFO *)block->lock_list, + my_thread_var); + DBUG_ASSERT(info != 0); + info_unlink((PAGECACHE_PIN_INFO *)info); + my_free((gptr)info, MYF(0)); +} +void pagecache_change_lock(PAGECACHE_BLOCK_LINK *block, bool wl) +{ + PAGECACHE_LOCK_INFO *info= + (PAGECACHE_LOCK_INFO *)info_find((PAGECACHE_PIN_INFO *)block->lock_list, + my_thread_var); + DBUG_ASSERT(info != 0 && info->write_lock != wl); + info->write_lock= wl; +} +#else +#define pagecache_add_lock(B,W) +#define pagecache_remove_lock(B) +#define pagecache_change_lock(B,W) +#endif + +/* + Put on the block "update" type lock + + SYNOPSIS + pagecache_lock_block() + pagecache pointer to a page cache data structure + block the block to work with + + RETURN + 0 - OK + 1 - Try to lock the block failed +*/ + +my_bool pagecache_lock_block(PAGECACHE *pagecache, + PAGECACHE_BLOCK_LINK *block) +{ + DBUG_ENTER("pagecache_lock_block"); + BLOCK_INFO(block); + if (block->status & BLOCK_WRLOCK) + { + DBUG_PRINT("info", ("fail to lock, waiting...")); + /* Lock failed we will wait */ +#ifdef THREAD + struct st_my_thread_var *thread= my_thread_var; + add_to_queue(&block->wqueue[COND_FOR_WRLOCK], thread); + dec_counter_for_resize_op(pagecache); + do + { + KEYCACHE_DBUG_PRINT("pagecache_lock_block: wait", + ("suspend thread %ld", thread->id)); + pagecache_pthread_cond_wait(&thread->suspend, + &pagecache->cache_lock); + } + while(thread->next); +#else + DBUG_ASSERT(0); +#endif + BLOCK_INFO(block); + DBUG_RETURN(1); + } + /* we are doing it by global cache mutex protectio, so it is OK */ + block->status|= BLOCK_WRLOCK; + DBUG_RETURN(0); +} + +void pagecache_ulock_block(PAGECACHE_BLOCK_LINK *block) +{ + DBUG_ENTER("pagecache_ulock_block"); + BLOCK_INFO(block); + DBUG_ASSERT(block->status & BLOCK_WRLOCK); + block->status&= ~BLOCK_WRLOCK; +#ifdef THREAD + /* release all threads waiting for write lock */ + if (block->wqueue[COND_FOR_WRLOCK].last_thread) + release_queue(&block->wqueue[COND_FOR_WRLOCK]); +#endif + BLOCK_INFO(block); + DBUG_VOID_RETURN; +} + +/* + Try to lock/uplock and pin/unpin the block + + SYNOPSIS + pagecache_make_lock_and_pin() + pagecache pointer to a page cache data structure + block the block to work with + lock lock change mode + pin pinchange mode + + RETURN + 0 - OK + 1 - Try to lock the block failed +*/ + +my_bool pagecache_make_lock_and_pin(PAGECACHE *pagecache, + PAGECACHE_BLOCK_LINK *block, + enum pagecache_page_lock lock, + enum pagecache_page_pin pin) +{ + DBUG_ENTER("pagecache_make_lock_and_pin"); + DBUG_PRINT("enter", ("block: 0x%lx (%u), wrlock: %c pins: %u, lock %s, pin: %s", + (ulong)block, BLOCK_NUMBER(pagecache, block), + ((block->status & BLOCK_WRLOCK)?'Y':'N'), + block->pins, + page_cache_page_lock_str[lock], + page_cache_page_pin_str[pin])); + BLOCK_INFO(block); +#ifdef PAGECACHE_DEBUG + DBUG_ASSERT(info_check_pin(block, pin) == 0 && + info_check_lock(block, lock, pin) == 0); +#endif + switch (lock) + { + case PAGECACHE_LOCK_WRITE: /* free -> write */ + /* Writelock and pin the buffer */ + if (pagecache_lock_block(pagecache, block)) + { + DBUG_PRINT("info", ("restart")); + /* in case of fail pagecache_lock_block unlock cache */ + DBUG_RETURN(1); + } + /* The cache is locked so nothing afraid off */ + pagecache_add_pin(block); + pagecache_add_lock(block, 1); + break; + case PAGECACHE_LOCK_WRITE_TO_READ: /* write -> read */ + case PAGECACHE_LOCK_WRITE_UNLOCK: /* write -> free */ + /* + Removes writelog and puts read lock (which is nothing in our + implementation) + */ + pagecache_ulock_block(block); + case PAGECACHE_LOCK_READ_UNLOCK: /* read -> free */ + case PAGECACHE_LOCK_LEFT_READLOCKED: /* read -> read */ +#ifndef DBUG_OFF + if (pin == PAGECACHE_UNPIN) + { + pagecache_remove_pin(block); + } +#endif +#ifdef PAGECACHE_DEBUG + if (lock == PAGECACHE_LOCK_WRITE_TO_READ) + { + pagecache_change_lock(block, 0); + } else if (lock == PAGECACHE_LOCK_WRITE_UNLOCK || + lock == PAGECACHE_LOCK_READ_UNLOCK) + { + pagecache_remove_lock(block); + } +#endif + break; + case PAGECACHE_LOCK_READ: /* free -> read */ +#ifndef DBUG_OFF + if (pin == PAGECACHE_PIN) + { + /* The cache is locked so nothing afraid off */ + pagecache_add_pin(block); + } + pagecache_add_lock(block, 0); + break; +#endif + case PAGECACHE_LOCK_LEFT_UNLOCKED: /* free -> free */ + case PAGECACHE_LOCK_LEFT_WRITELOCKED: /* write -> write */ + break; /* do nothing */ + default: + DBUG_ASSERT(0); /* Never should happened */ + } + + BLOCK_INFO(block); + DBUG_RETURN(0); +} + + +/* + Read into a key cache block buffer from disk. + + SYNOPSIS + + read_block() + pagecache pointer to a page cache data structure + block block to which buffer the data is to be read + primary <-> the current thread will read the data + + RETURN VALUE + None + + NOTES. + The function either reads a page data from file to the block buffer, + or waits until another thread reads it. What page to read is determined + by a block parameter - reference to a hash link for this page. + If an error occurs THE BLOCK_ERROR bit is set in the block status. +*/ + +static void read_block(PAGECACHE *pagecache, + PAGECACHE_BLOCK_LINK *block, + my_bool primary) +{ + uint got_length; + + /* On entry cache_lock is locked */ + + KEYCACHE_THREAD_TRACE("read_block"); + if (primary) + { + /* + This code is executed only by threads + that submitted primary requests + */ + + KEYCACHE_DBUG_PRINT("read_block", + ("page to be read by primary request")); + + /* Page is not in buffer yet, is to be read from disk */ + pagecache_pthread_mutex_unlock(&pagecache->cache_lock); + /* + Here other threads may step in and register as secondary readers. + They will register in block->wqueue[COND_FOR_REQUESTED]. + */ + got_length= pagecache_fread(pagecache, &block->hash_link->file, + block->buffer, + block->hash_link->pageno, MYF(0)); + pagecache_pthread_mutex_lock(&pagecache->cache_lock); + if (got_length < pagecache->block_size) + block->status|= BLOCK_ERROR; + else + block->status= (BLOCK_READ | (block->status & BLOCK_WRLOCK)); + + KEYCACHE_DBUG_PRINT("read_block", + ("primary request: new page in cache")); + /* Signal that all pending requests for this page now can be processed */ + if (block->wqueue[COND_FOR_REQUESTED].last_thread) + release_queue(&block->wqueue[COND_FOR_REQUESTED]); + } + else + { + /* + This code is executed only by threads + that submitted secondary requests + */ + KEYCACHE_DBUG_PRINT("read_block", + ("secondary request waiting for new page to be read")); + { +#ifdef THREAD + struct st_my_thread_var *thread= my_thread_var; + /* Put the request into a queue and wait until it can be processed */ + add_to_queue(&block->wqueue[COND_FOR_REQUESTED], thread); + do + { + KEYCACHE_DBUG_PRINT("read_block: wait", + ("suspend thread %ld", thread->id)); + pagecache_pthread_cond_wait(&thread->suspend, + &pagecache->cache_lock); + } + while (thread->next); +#else + KEYCACHE_DBUG_ASSERT(0); + /* No parallel requests in single-threaded case */ +#endif + } + KEYCACHE_DBUG_PRINT("read_block", + ("secondary request: new page in cache")); + } +} + + +/* + Unlock/unpin page and put LSN stamp if it need + + SYNOPSIS + pagecache_unlock_page() + pagecache pointer to a page cache data structure + file handler for the file for the block of data to be read + pageno number of the block of data in the file + lock lock change + pin pin page + stamp_this_page put LSN stamp on the page + first_REDO_LSN_for_page +*/ + +void pagecache_unlock_page(PAGECACHE *pagecache, + PAGECACHE_FILE *file, + maria_page_no_t pageno, + enum pagecache_page_lock lock, + enum pagecache_page_pin pin, + my_bool stamp_this_page, + LSN first_REDO_LSN_for_page) +{ + PAGECACHE_BLOCK_LINK *block; + int page_st; + DBUG_ENTER("pagecache_unlock_page"); + DBUG_PRINT("enter", ("fd: %u page: %lu l%s p%s", + (uint) file->file, (ulong) pageno, + page_cache_page_lock_str[lock], + page_cache_page_pin_str[pin])); + /* we do not allow any lock/pin increasing here */ + DBUG_ASSERT(pin != PAGECACHE_PIN && + lock != PAGECACHE_LOCK_READ && + lock != PAGECACHE_LOCK_WRITE); + if (pin == PAGECACHE_PIN_LEFT_UNPINNED && + lock == PAGECACHE_LOCK_READ_UNLOCK) + { +#ifndef DBUG_OFF + if ( +#endif + /* block do not need here so we do not provide it */ + pagecache_make_lock_and_pin(pagecache, 0, lock, pin) +#ifndef DBUG_OFF + ) + { + DBUG_ASSERT(0); /* should not happend */ + } +#else + ; +#endif + DBUG_VOID_RETURN; + } + + pagecache_pthread_mutex_lock(&pagecache->cache_lock); + /* + As soon as we keep lock cache can be used, and we have lock bacause want + aunlock. + */ + DBUG_ASSERT(pagecache->can_be_used); + + inc_counter_for_resize_op(pagecache); + block= find_key_block(pagecache, file, pageno, 0, 0, 0, &page_st); + DBUG_ASSERT(block != 0 && page_st == PAGE_READ); + if (stamp_this_page) + { + DBUG_ASSERT(lock == PAGECACHE_LOCK_WRITE_UNLOCK && + pin == PAGECACHE_UNPIN); + /* TODO: insert LSN writing code */ + } + +#ifndef DBUG_OFF + if ( +#endif + pagecache_make_lock_and_pin(pagecache, block, lock, pin) +#ifndef DBUG_OFF + ) + { + DBUG_ASSERT(0); /* should not happend */ + } +#else + ; +#endif + + remove_reader(block); + /* + Link the block into the LRU chain if it's the last submitted request + for the block and block will not be pinned + */ + if (pin != PAGECACHE_PIN_LEFT_PINNED) + unreg_request(pagecache, block, 1); + + dec_counter_for_resize_op(pagecache); + + pagecache_pthread_mutex_unlock(&pagecache->cache_lock); + + DBUG_VOID_RETURN; +} + + +/* + Unpin page + + SYNOPSIS + pagecache_unpin_page() + pagecache pointer to a page cache data structure + file handler for the file for the block of data to be read + pageno number of the block of data in the file +*/ + +void pagecache_unpin_page(PAGECACHE *pagecache, + PAGECACHE_FILE *file, + maria_page_no_t pageno) +{ + PAGECACHE_BLOCK_LINK *block; + int page_st; + DBUG_ENTER("pagecache_unpin_page"); + DBUG_PRINT("enter", ("fd: %u page: %lu", + (uint) file->file, (ulong) pageno)); + + pagecache_pthread_mutex_lock(&pagecache->cache_lock); + /* + As soon as we keep lock cache can be used, and we have lock bacause want + aunlock. + */ + DBUG_ASSERT(pagecache->can_be_used); + + inc_counter_for_resize_op(pagecache); + block= find_key_block(pagecache, file, pageno, 0, 0, 0, &page_st); + DBUG_ASSERT(block != 0 && page_st == PAGE_READ); + +#ifndef DBUG_OFF + if ( +#endif + /* + we can just unpin only with keeping read lock because: + a) we can't pin without any lock + b) we can't unpin keeping write lock + */ + pagecache_make_lock_and_pin(pagecache, block, + PAGECACHE_LOCK_LEFT_READLOCKED, + PAGECACHE_UNPIN) +#ifndef DBUG_OFF + ) + { + DBUG_ASSERT(0); /* should not happend */ + } +#else + ; +#endif + + remove_reader(block); + /* + Link the block into the LRU chain if it's the last submitted request + for the block and block will not be pinned + */ + unreg_request(pagecache, block, 1); + + dec_counter_for_resize_op(pagecache); + + pagecache_pthread_mutex_unlock(&pagecache->cache_lock); + + DBUG_VOID_RETURN; +} + + +/* + Unlock/unpin page and put LSN stamp if it need + (uses direct block/page pointer) + + SYNOPSIS + pagecache_unlock() + pagecache pointer to a page cache data structure + link direct link to page (returned by read or write) + lock lock change + pin pin page + stamp_this_page put LSN stamp on the page + first_REDO_LSN_for_page +*/ + +void pagecache_unlock(PAGECACHE *pagecache, + PAGECACHE_PAGE_LINK *link, + enum pagecache_page_lock lock, + enum pagecache_page_pin pin, + my_bool stamp_this_page, + LSN first_REDO_LSN_for_page) +{ + PAGECACHE_BLOCK_LINK *block= (PAGECACHE_BLOCK_LINK *)link; + DBUG_ENTER("pagecache_unlock"); + DBUG_PRINT("enter", ("block: 0x%lx fd: %u page: %lu l%s p%s", + (ulong) block, + (uint) block->hash_link->file.file, + (ulong) block->hash_link->pageno, + page_cache_page_lock_str[lock], + page_cache_page_pin_str[pin])); + /* we do not allow any lock/pin increasing here */ + DBUG_ASSERT(pin != PAGECACHE_PIN && + lock != PAGECACHE_LOCK_READ && + lock != PAGECACHE_LOCK_WRITE); + if (pin == PAGECACHE_PIN_LEFT_UNPINNED && + lock == PAGECACHE_LOCK_READ_UNLOCK) + { +#ifndef DBUG_OFF + if ( +#endif + /* block do not need here so we do not provide it */ + pagecache_make_lock_and_pin(pagecache, 0, lock, pin) +#ifndef DBUG_OFF + ) + { + DBUG_ASSERT(0); /* should not happend */ + } +#else + ; +#endif + DBUG_VOID_RETURN; + } + + pagecache_pthread_mutex_lock(&pagecache->cache_lock); + /* + As soon as we keep lock cache can be used, and we have lock bacause want + aunlock. + */ + DBUG_ASSERT(pagecache->can_be_used); + + inc_counter_for_resize_op(pagecache); + if (stamp_this_page) + { + DBUG_ASSERT(lock == PAGECACHE_LOCK_WRITE_UNLOCK && + pin == PAGECACHE_UNPIN); + /* TODO: insert LSN writing code */ + } + +#ifndef DBUG_OFF + if ( +#endif + pagecache_make_lock_and_pin(pagecache, block, lock, pin) +#ifndef DBUG_OFF + ) + { + DBUG_ASSERT(0); /* should not happend */ + } +#else + ; +#endif + + remove_reader(block); + /* + Link the block into the LRU chain if it's the last submitted request + for the block and block will not be pinned + */ + if (pin != PAGECACHE_PIN_LEFT_PINNED) + unreg_request(pagecache, block, 1); + + dec_counter_for_resize_op(pagecache); + + pagecache_pthread_mutex_unlock(&pagecache->cache_lock); + + DBUG_VOID_RETURN; +} + + +/* + Unpin page + (uses direct block/page pointer) + + SYNOPSIS + pagecache_unpin_page() + pagecache pointer to a page cache data structure + link direct link to page (returned by read or write) +*/ + +void pagecache_unpin(PAGECACHE *pagecache, + PAGECACHE_PAGE_LINK *link) +{ + PAGECACHE_BLOCK_LINK *block= (PAGECACHE_BLOCK_LINK *)link; + DBUG_ENTER("pagecache_unpin"); + DBUG_PRINT("enter", ("block: 0x%lx fd: %u page: %lu", + (ulong) block, + (uint) block->hash_link->file.file, + (ulong) block->hash_link->pageno)); + + pagecache_pthread_mutex_lock(&pagecache->cache_lock); + /* + As soon as we keep lock cache can be used, and we have lock bacause want + aunlock. + */ + DBUG_ASSERT(pagecache->can_be_used); + + inc_counter_for_resize_op(pagecache); + +#ifndef DBUG_OFF + if ( +#endif + /* + we can just unpin only with keeping read lock because: + a) we can't pin without any lock + b) we can't unpin keeping write lock + */ + pagecache_make_lock_and_pin(pagecache, block, + PAGECACHE_LOCK_LEFT_READLOCKED, + PAGECACHE_UNPIN) +#ifndef DBUG_OFF + ) + { + DBUG_ASSERT(0); /* should not happend */ + } +#else + ; +#endif + + remove_reader(block); + /* + Link the block into the LRU chain if it's the last submitted request + for the block and block will not be pinned + */ + unreg_request(pagecache, block, 1); + + dec_counter_for_resize_op(pagecache); + + pagecache_pthread_mutex_unlock(&pagecache->cache_lock); + + DBUG_VOID_RETURN; +} + + +/* + Read a block of data from a cached file into a buffer; + + SYNOPSIS + pagecache_read() + pagecache pointer to a page cache data structure + file handler for the file for the block of data to be read + pageno number of the block of data in the file + level determines the weight of the data + buff buffer to where the data must be placed + type type of the page + lock lock change + link link to the page if we pin it + + RETURN VALUE + Returns address from where the data is placed if sucessful, 0 - otherwise. + + NOTES. + + The function ensures that a block of data of size length from file + positioned at pageno is in the buffers for some key cache blocks. + Then the function copies the data into the buffer buff. + + Pin will be choosen according to lock parameter (see lock_to_pin) +*/ +static enum pagecache_page_pin lock_to_pin[]= +{ + PAGECACHE_PIN_LEFT_UNPINNED /*PAGECACHE_LOCK_LEFT_UNLOCKED*/, + PAGECACHE_PIN_LEFT_UNPINNED /*PAGECACHE_LOCK_LEFT_READLOCKED*/, + PAGECACHE_PIN_LEFT_PINNED /*PAGECACHE_LOCK_LEFT_WRITELOCKED*/, + PAGECACHE_PIN_LEFT_UNPINNED /*PAGECACHE_LOCK_READ*/, + PAGECACHE_PIN /*PAGECACHE_LOCK_WRITE*/, + PAGECACHE_PIN_LEFT_UNPINNED /*PAGECACHE_LOCK_READ_UNLOCK*/, + PAGECACHE_UNPIN /*PAGECACHE_LOCK_WRITE_UNLOCK*/, + PAGECACHE_UNPIN /*PAGECACHE_LOCK_WRITE_TO_READ*/ +}; + +byte *pagecache_read(PAGECACHE *pagecache, + PAGECACHE_FILE *file, + maria_page_no_t pageno, + uint level, + byte *buff, + enum pagecache_page_type type, + enum pagecache_page_lock lock, + PAGECACHE_PAGE_LINK *link) +{ + int error= 0; + enum pagecache_page_pin pin= lock_to_pin[lock]; + PAGECACHE_PAGE_LINK fake_link; + DBUG_ENTER("page_cache_read"); + DBUG_PRINT("enter", ("fd: %u page: %lu level: %u t:%s l%s p%s", + (uint) file->file, (ulong) pageno, level, + page_cache_page_type_str[type], + page_cache_page_lock_str[lock], + page_cache_page_pin_str[pin])); + + if (!link) + link= &fake_link; + else + *link= 0; + +restart: + + if (pagecache->can_be_used) + { + /* Key cache is used */ + reg1 PAGECACHE_BLOCK_LINK *block; + uint status; + int page_st; + + pagecache_pthread_mutex_lock(&pagecache->cache_lock); + if (!pagecache->can_be_used) + { + pagecache_pthread_mutex_unlock(&pagecache->cache_lock); + goto no_key_cache; + } + + inc_counter_for_resize_op(pagecache); + pagecache->global_cache_r_requests++; + block= find_key_block(pagecache, file, pageno, level, 0, + (((pin == PAGECACHE_PIN_LEFT_PINNED) || + (pin == PAGECACHE_UNPIN)) ? 0 : 1), + &page_st); + DBUG_ASSERT(block->type == PAGECACHE_EMPTY_PAGE || + block->type == type); + block->type= type; + if (pagecache_make_lock_and_pin(pagecache, block, lock, pin)) + { + /* + We failed to writelock the block, cache is unlocked, and last write + lock is released, we will try to get the block again. + */ + pagecache_pthread_mutex_unlock(&pagecache->cache_lock); + goto restart; + } + if (block->status != BLOCK_ERROR && page_st != PAGE_READ) + { + /* The requested page is to be read into the block buffer */ + read_block(pagecache, block, + (my_bool)(page_st == PAGE_TO_BE_READ)); + } + + if (! ((status= block->status) & BLOCK_ERROR)) + { +#if !defined(SERIALIZED_READ_FROM_CACHE) + pagecache_pthread_mutex_unlock(&pagecache->cache_lock); +#endif + + DBUG_ASSERT((pagecache->block_size & 511) == 0); + /* Copy data from the cache buffer */ + bmove512(buff, block->buffer, pagecache->block_size); + +#if !defined(SERIALIZED_READ_FROM_CACHE) + pagecache_pthread_mutex_lock(&pagecache->cache_lock); +#endif + } + + remove_reader(block); + /* + Link the block into the LRU chain if it's the last submitted request + for the block and block will not be pinned + */ + if (pin != PAGECACHE_PIN_LEFT_PINNED && pin != PAGECACHE_PIN) + unreg_request(pagecache, block, 1); + else + *link= (PAGECACHE_PAGE_LINK)block; + + dec_counter_for_resize_op(pagecache); + + pagecache_pthread_mutex_unlock(&pagecache->cache_lock); + + if (status & BLOCK_ERROR) + DBUG_RETURN((byte *) 0); + + DBUG_RETURN(buff); + } + +no_key_cache: /* Key cache is not used */ + + /* We can't use mutex here as the key cache may not be initialized */ + pagecache->global_cache_r_requests++; + pagecache->global_cache_read++; + if (pagecache_fread(pagecache, file, (byte*) buff, pageno, MYF(MY_NABP))) + error= 1; + DBUG_RETURN(error ? (byte*) 0 : buff); +} + + +/* + Delete page from the buffer + + SYNOPSIS + pagecache_delete_page() + pagecache pointer to a page cache data structure + file handler for the file for the block of data to be read + pageno number of the block of data in the file + lock lock change + flush flush page if it is dirty + + RETURN VALUE + 0 - deleted or was not present at all + 1 - error + + NOTES. + lock can be only PAGECACHE_LOCK_LEFT_WRITELOCKED (page was write locked + before) or PAGECACHE_LOCK_WRITE (delete will write lock page before delete) +*/ +my_bool pagecache_delete_page(PAGECACHE *pagecache, + PAGECACHE_FILE *file, + maria_page_no_t pageno, + enum pagecache_page_lock lock, + my_bool flush) +{ + int error= 0; + enum pagecache_page_pin pin= lock_to_pin[lock]; + DBUG_ENTER("pagecache_delete_page"); + DBUG_PRINT("enter", ("fd: %u page: %lu l%s p%s", + (uint) file->file, (ulong) pageno, + page_cache_page_lock_str[lock], + page_cache_page_pin_str[pin])); + DBUG_ASSERT(lock == PAGECACHE_LOCK_WRITE || + lock == PAGECACHE_LOCK_LEFT_WRITELOCKED); + +restart: + + if (pagecache->can_be_used) + { + /* Key cache is used */ + reg1 PAGECACHE_BLOCK_LINK *block; + PAGECACHE_HASH_LINK **unused_start, *link; + + pagecache_pthread_mutex_lock(&pagecache->cache_lock); + if (!pagecache->can_be_used) + goto end; + + inc_counter_for_resize_op(pagecache); + link= get_present_hash_link(pagecache, file, pageno, &unused_start); + if (!link) + { + DBUG_PRINT("info", ("There is no fuch page in the cache")); + DBUG_RETURN(0); + } + block= link->block; + DBUG_ASSERT(block != 0); + if (pagecache_make_lock_and_pin(pagecache, block, lock, pin)) + { + /* + We failed to writelock the block, cache is unlocked, and last write + lock is released, we will try to get the block again. + */ + pagecache_pthread_mutex_unlock(&pagecache->cache_lock); + goto restart; + } + + if (block->status & BLOCK_CHANGED && flush) + { + if (flush) + { + /* The block contains a dirty page - push it out of the cache */ + + KEYCACHE_DBUG_PRINT("find_key_block", ("block is dirty")); + + pagecache_pthread_mutex_unlock(&pagecache->cache_lock); + /* + The call is thread safe because only the current + thread might change the block->hash_link value + */ + DBUG_ASSERT(block->pins == 1); + error= pagecache_fwrite(pagecache, + &block->hash_link->file, + block->buffer, + block->hash_link->pageno, + block->type, + MYF(MY_NABP | MY_WAIT_IF_FULL)); + pagecache_pthread_mutex_lock(&pagecache->cache_lock); + pagecache->global_cache_write++; + if (error) + { + block->status|= BLOCK_ERROR; + goto err; + } + } + pagecache->blocks_changed--; + pagecache->global_blocks_changed--; + + } + /* Cache is locked, so we can relese page before freeing it */ + pagecache_make_lock_and_pin(pagecache, block, + PAGECACHE_LOCK_WRITE_UNLOCK, + PAGECACHE_UNPIN); + if (pin == PAGECACHE_PIN_LEFT_PINNED) + unreg_request(pagecache, block, 1); + free_block(pagecache, block); + +err: + dec_counter_for_resize_op(pagecache); +end: + pagecache_pthread_mutex_unlock(&pagecache->cache_lock); + } + + DBUG_RETURN(error); +} + + +/* + Write a buffer into a cached file. + + SYNOPSIS + + pagecache_write() + pagecache pointer to a page cache data structure + file handler for the file to write data to + pageno number of the block of data in the file + level determines the weight of the data + buff buffer to where the data must be placed + type type of the page + lock lock change + pin pin page + write_mode how to write page + link link to the page if we pin it + + RETURN VALUE + 0 if a success, 1 - otherwise. +*/ + +struct write_lock_change +{ + int need_lock_change; + enum pagecache_page_lock new_lock; + enum pagecache_page_lock unlock_lock; +}; + +static struct write_lock_change write_lock_change_table[]= +{ + {1, + PAGECACHE_LOCK_WRITE, + PAGECACHE_LOCK_WRITE_UNLOCK} /*PAGECACHE_LOCK_LEFT_UNLOCKED*/, + {0, /*unsupported*/ + PAGECACHE_LOCK_LEFT_UNLOCKED, + PAGECACHE_LOCK_LEFT_UNLOCKED} /*PAGECACHE_LOCK_LEFT_READLOCKED*/, + {0, PAGECACHE_LOCK_LEFT_WRITELOCKED, 0} /*PAGECACHE_LOCK_LEFT_WRITELOCKED*/, + {1, + PAGECACHE_LOCK_WRITE, + PAGECACHE_LOCK_WRITE_TO_READ} /*PAGECACHE_LOCK_READ*/, + {0, PAGECACHE_LOCK_WRITE, 0} /*PAGECACHE_LOCK_WRITE*/, + {0, /*unsupported*/ + PAGECACHE_LOCK_LEFT_UNLOCKED, + PAGECACHE_LOCK_LEFT_UNLOCKED} /*PAGECACHE_LOCK_READ_UNLOCK*/, + {1, + PAGECACHE_LOCK_LEFT_WRITELOCKED, + PAGECACHE_LOCK_WRITE_UNLOCK } /*PAGECACHE_LOCK_WRITE_UNLOCK*/, + {1, + PAGECACHE_LOCK_LEFT_WRITELOCKED, + PAGECACHE_LOCK_WRITE_TO_READ}/*PAGECACHE_LOCK_WRITE_TO_READ*/ +}; + +struct write_pin_change +{ + enum pagecache_page_pin new_pin; + enum pagecache_page_pin unlock_pin; +}; + +static struct write_pin_change write_pin_change_table[]= +{ + {PAGECACHE_PIN_LEFT_PINNED, + PAGECACHE_PIN_LEFT_PINNED} /*PAGECACHE_PIN_LEFT_PINNED*/, + {PAGECACHE_PIN, + PAGECACHE_UNPIN} /*PAGECACHE_PIN_LEFT_UNPINNED*/, + {PAGECACHE_PIN, + PAGECACHE_PIN_LEFT_PINNED} /*PAGECACHE_PIN*/, + {PAGECACHE_PIN_LEFT_PINNED, + PAGECACHE_UNPIN} /*PAGECACHE_UNPIN*/ +}; + +my_bool pagecache_write(PAGECACHE *pagecache, + PAGECACHE_FILE *file, + maria_page_no_t pageno, + uint level, + byte *buff, + enum pagecache_page_type type, + enum pagecache_page_lock lock, + enum pagecache_page_pin pin, + enum pagecache_write_mode write_mode, + PAGECACHE_PAGE_LINK *link) +{ + reg1 PAGECACHE_BLOCK_LINK *block; + PAGECACHE_PAGE_LINK fake_link; + int error= 0; + int need_lock_change= write_lock_change_table[lock].need_lock_change; + DBUG_ENTER("pagecache_write"); + DBUG_PRINT("enter", ("fd: %u page: %lu level: %u t:%s l%s p%s m%s", + (uint) file->file, (ulong) pageno, level, + page_cache_page_type_str[type], + page_cache_page_lock_str[lock], + page_cache_page_pin_str[pin], + page_cache_page_write_mode_str[write_mode])); + DBUG_ASSERT(lock != PAGECACHE_LOCK_LEFT_READLOCKED && + lock != PAGECACHE_LOCK_READ_UNLOCK); + if (!link) + link= &fake_link; + else + *link= 0; + + if (write_mode == PAGECACHE_WRITE_NOW) + { + /* we allow direct write if wwe do not use long term lockings */ + DBUG_ASSERT(lock == PAGECACHE_LOCK_LEFT_UNLOCKED); + /* Force writing from buff into disk */ + pagecache->global_cache_write++; + if (pagecache_fwrite(pagecache, file, buff, pageno, type, + MYF(MY_NABP | MY_WAIT_IF_FULL))) + DBUG_RETURN(1); + } +restart: + +#if !defined(DBUG_OFF) && defined(EXTRA_DEBUG) + DBUG_EXECUTE("check_pagecache", + test_key_cache(pagecache, "start of key_cache_write", 1);); +#endif + + if (pagecache->can_be_used) + { + /* Key cache is used */ + int page_st; + + pagecache_pthread_mutex_lock(&pagecache->cache_lock); + if (!pagecache->can_be_used) + { + pagecache_pthread_mutex_unlock(&pagecache->cache_lock); + goto no_key_cache; + } + + inc_counter_for_resize_op(pagecache); + pagecache->global_cache_w_requests++; + block= find_key_block(pagecache, file, pageno, level, + (write_mode == PAGECACHE_WRITE_DONE ? 0 : 1), + (((pin == PAGECACHE_PIN_LEFT_PINNED) || + (pin == PAGECACHE_UNPIN)) ? 0 : 1), + &page_st); + if (!block) + { + DBUG_ASSERT(write_mode != PAGECACHE_WRITE_DONE); + /* It happens only for requests submitted during resize operation */ + dec_counter_for_resize_op(pagecache); + pagecache_pthread_mutex_unlock(&pagecache->cache_lock); + /* Write to the disk key cache is in resize at the moment*/ + goto no_key_cache; + } + + DBUG_ASSERT(block->type == PAGECACHE_EMPTY_PAGE || + block->type == type); + block->type= type; + + if (pagecache_make_lock_and_pin(pagecache, block, + write_lock_change_table[lock].new_lock, + (need_lock_change ? + write_pin_change_table[pin].new_pin : + pin))) + { + /* + We failed to writelock the block, cache is unlocked, and last write + lock is released, we will try to get the block again. + */ + pagecache_pthread_mutex_unlock(&pagecache->cache_lock); + goto restart; + } + + + if (write_mode == PAGECACHE_WRITE_DONE) + { + if (block->status != BLOCK_ERROR && page_st != PAGE_READ) + { + /* Copy data from buff */ + bmove512(block->buffer, buff, pagecache->block_size); + block->status= (BLOCK_READ | (block->status & BLOCK_WRLOCK)); + KEYCACHE_DBUG_PRINT("key_cache_insert", + ("primary request: new page in cache")); + /* Signal that all pending requests for this now can be processed. */ + if (block->wqueue[COND_FOR_REQUESTED].last_thread) + release_queue(&block->wqueue[COND_FOR_REQUESTED]); + } + } + else + { + if (write_mode == PAGECACHE_WRITE_NOW) + { + /* buff has been written to disk at start */ + if (block->status & BLOCK_CHANGED) + link_to_file_list(pagecache, block, &block->hash_link->file, 1); + } + else + { + if (! (block->status & BLOCK_CHANGED)) + link_to_changed_list(pagecache, block); + } + if (! (block->status & BLOCK_ERROR)) + { + bmove512(block->buffer, buff, pagecache->block_size); + } + block->status|= BLOCK_READ; + } + + + if (need_lock_change) + { +#ifndef DBUG_OFF + int rc= +#endif + pagecache_make_lock_and_pin(pagecache, block, + write_lock_change_table[lock].unlock_lock, + write_pin_change_table[pin].unlock_pin); +#ifndef DBUG_OFF + DBUG_ASSERT(rc == 0); +#endif + } + + /* Unregister the request */ + block->hash_link->requests--; + if (pin != PAGECACHE_PIN_LEFT_PINNED && pin != PAGECACHE_PIN) + unreg_request(pagecache, block, 1); + else + *link= (PAGECACHE_PAGE_LINK)block; + + + if (block->status & BLOCK_ERROR) + error= 1; + + dec_counter_for_resize_op(pagecache); + + pagecache_pthread_mutex_unlock(&pagecache->cache_lock); + + goto end; + } + +no_key_cache: + /* Key cache is not used */ + if (write_mode == PAGECACHE_WRITE_DELAY) + { + pagecache->global_cache_w_requests++; + pagecache->global_cache_write++; + if (pagecache_fwrite(pagecache, file, (byte*) buff, pageno, type, + MYF(MY_NABP | MY_WAIT_IF_FULL))) + error=1; + } + +end: +#if !defined(DBUG_OFF) && defined(EXTRA_DEBUG) + DBUG_EXECUTE("exec", + test_key_cache(pagecache, "end of key_cache_write", 1);); +#endif + DBUG_RETURN(error); +} + + +/* + Free block: remove reference to it from hash table, + remove it from the chain file of dirty/clean blocks + and add it to the free list. +*/ + +static void free_block(PAGECACHE *pagecache, PAGECACHE_BLOCK_LINK *block) +{ + KEYCACHE_THREAD_TRACE("free block"); + KEYCACHE_DBUG_PRINT("free_block", + ("block %u to be freed, hash_link %p", + BLOCK_NUMBER(pagecache, block), block->hash_link)); + if (block->hash_link) + { + /* + While waiting for readers to finish, new readers might request the + block. But since we set block->status|= BLOCK_REASSIGNED, they + will wait on block->wqueue[COND_FOR_SAVED]. They must be signalled + later. + */ + block->status|= BLOCK_REASSIGNED; + wait_for_readers(pagecache, block); + unlink_hash(pagecache, block->hash_link); + } + + unlink_changed(block); + DBUG_ASSERT((block->status & BLOCK_WRLOCK) == 0); + block->status= 0; +#ifndef DBUG_OFF + block->type= PAGECACHE_EMPTY_PAGE; +#endif + KEYCACHE_THREAD_TRACE("free block"); + KEYCACHE_DBUG_PRINT("free_block", + ("block is freed")); + unreg_request(pagecache, block, 0); + block->hash_link= NULL; + + /* Remove the free block from the LRU ring. */ + unlink_block(pagecache, block); + if (block->temperature == BLOCK_WARM) + pagecache->warm_blocks--; + block->temperature= BLOCK_COLD; + /* Insert the free block in the free list. */ + block->next_used= pagecache->free_block_list; + pagecache->free_block_list= block; + /* Keep track of the number of currently unused blocks. */ + pagecache->blocks_unused++; + + /* All pending requests for this page must be resubmitted. */ + if (block->wqueue[COND_FOR_SAVED].last_thread) + release_queue(&block->wqueue[COND_FOR_SAVED]); +} + + +static int cmp_sec_link(PAGECACHE_BLOCK_LINK **a, PAGECACHE_BLOCK_LINK **b) +{ + return (((*a)->hash_link->pageno < (*b)->hash_link->pageno) ? -1 : + ((*a)->hash_link->pageno > (*b)->hash_link->pageno) ? 1 : 0); +} + + +/* + Flush a portion of changed blocks to disk, + free used blocks if requested +*/ + +static int flush_cached_blocks(PAGECACHE *pagecache, + PAGECACHE_FILE *file, + PAGECACHE_BLOCK_LINK **cache, + PAGECACHE_BLOCK_LINK **end, + enum flush_type type) +{ + int error; + int last_errno= 0; + uint count= (uint) (end-cache); + DBUG_ENTER("flush_cached_blocks"); + + /* Don't lock the cache during the flush */ + pagecache_pthread_mutex_unlock(&pagecache->cache_lock); + /* + As all blocks referred in 'cache' are marked by BLOCK_IN_FLUSH + we are guarunteed no thread will change them + */ + qsort((byte*) cache, count, sizeof(*cache), (qsort_cmp) cmp_sec_link); + + pagecache_pthread_mutex_lock(&pagecache->cache_lock); + for (; cache != end; cache++) + { + PAGECACHE_BLOCK_LINK *block= *cache; + + if (block->pins) + { + KEYCACHE_DBUG_PRINT("flush_cached_blocks", + ("block %u (0x%lx) pinned", + BLOCK_NUMBER(pagecache, block), (ulong)block)); + DBUG_PRINT("info", ("block %u (0x%lx) pinned", + BLOCK_NUMBER(pagecache, block), (ulong)block)); + BLOCK_INFO(block); + last_errno= -1; + unreg_request(pagecache, block, 1); + continue; + } + /* if the block is not pinned then it is not write locked */ + DBUG_ASSERT((block->status & BLOCK_WRLOCK) == 0); +#ifndef DBUG_OFF + { + int rc= +#endif + pagecache_make_lock_and_pin(pagecache, block, + PAGECACHE_LOCK_WRITE, PAGECACHE_PIN); +#ifndef DBUG_OFF + DBUG_ASSERT(rc == 0); + } +#endif + + KEYCACHE_DBUG_PRINT("flush_cached_blocks", + ("block %u (0x%lx) to be flushed", + BLOCK_NUMBER(pagecache, block), (ulong)block)); + DBUG_PRINT("info", ("block %u (0x%lx) to be flushed", + BLOCK_NUMBER(pagecache, block), (ulong)block)); + BLOCK_INFO(block); + pagecache_pthread_mutex_unlock(&pagecache->cache_lock); + DBUG_PRINT("info", ("block %u (0x%lx) pins: %u", + BLOCK_NUMBER(pagecache, block), (ulong)block, + block->pins)); + DBUG_ASSERT(block->pins == 1); + error= pagecache_fwrite(pagecache, file, + block->buffer, + block->hash_link->pageno, + block->type, + MYF(MY_NABP | MY_WAIT_IF_FULL)); + pagecache_pthread_mutex_lock(&pagecache->cache_lock); + + pagecache_make_lock_and_pin(pagecache, block, + PAGECACHE_LOCK_WRITE_UNLOCK, + PAGECACHE_UNPIN); + + pagecache->global_cache_write++; + if (error) + { + block->status|= BLOCK_ERROR; + if (!last_errno) + last_errno= errno ? errno : -1; + } + /* + Let to proceed for possible waiting requests to write to the block page. + It might happen only during an operation to resize the key cache. + */ + if (block->wqueue[COND_FOR_SAVED].last_thread) + release_queue(&block->wqueue[COND_FOR_SAVED]); + /* type will never be FLUSH_IGNORE_CHANGED here */ + if (! (type == FLUSH_KEEP || type == FLUSH_FORCE_WRITE)) + { + pagecache->blocks_changed--; + pagecache->global_blocks_changed--; + free_block(pagecache, block); + } + else + { + block->status&= ~BLOCK_IN_FLUSH; + link_to_file_list(pagecache, block, file, 1); + unreg_request(pagecache, block, 1); + } + } + DBUG_RETURN(last_errno); +} + + +/* + flush all key blocks for a file to disk, but don't do any mutex locks + + flush_pagecache_blocks_int() + pagecache pointer to a key cache data structure + file handler for the file to flush to + flush_type type of the flush + + NOTES + This function doesn't do any mutex locks because it needs to be called + both from flush_pagecache_blocks and flush_all_key_blocks (the later one + does the mutex lock in the resize_pagecache() function). + + RETURN + 0 ok + 1 error +*/ + +static int flush_pagecache_blocks_int(PAGECACHE *pagecache, + PAGECACHE_FILE *file, + enum flush_type type) +{ + PAGECACHE_BLOCK_LINK *cache_buff[FLUSH_CACHE],**cache; + int last_errno= 0; + DBUG_ENTER("flush_pagecache_blocks_int"); + DBUG_PRINT("enter",("file: %d blocks_used: %d blocks_changed: %d", + file->file, pagecache->blocks_used, pagecache->blocks_changed)); + +#if !defined(DBUG_OFF) && defined(EXTRA_DEBUG) + DBUG_EXECUTE("check_pagecache", + test_key_cache(pagecache, + "start of flush_pagecache_blocks", 0);); +#endif + + cache= cache_buff; + if (pagecache->disk_blocks > 0 && + (!my_disable_flush_pagecache_blocks || type != FLUSH_KEEP)) + { + /* Key cache exists and flush is not disabled */ + int error= 0; + uint count= 0; + PAGECACHE_BLOCK_LINK **pos, **end; + PAGECACHE_BLOCK_LINK *first_in_switch= NULL; + PAGECACHE_BLOCK_LINK *block, *next; +#if defined(PAGECACHE_DEBUG) + uint cnt= 0; +#endif + + if (type != FLUSH_IGNORE_CHANGED) + { + /* + Count how many key blocks we have to cache to be able + to flush all dirty pages with minimum seek moves + */ + for (block= pagecache->changed_blocks[FILE_HASH(*file)] ; + block; + block= block->next_changed) + { + if (block->hash_link->file.file == file->file) + { + count++; + KEYCACHE_DBUG_ASSERT(count<= pagecache->blocks_used); + } + } + /* Allocate a new buffer only if its bigger than the one we have */ + if (count > FLUSH_CACHE && + !(cache= + (PAGECACHE_BLOCK_LINK**) + my_malloc(sizeof(PAGECACHE_BLOCK_LINK*)*count, MYF(0)))) + { + cache= cache_buff; + count= FLUSH_CACHE; + } + } + + /* Retrieve the blocks and write them to a buffer to be flushed */ +restart: + end= (pos= cache)+count; + for (block= pagecache->changed_blocks[FILE_HASH(*file)] ; + block; + block= next) + { +#if defined(PAGECACHE_DEBUG) + cnt++; + KEYCACHE_DBUG_ASSERT(cnt <= pagecache->blocks_used); +#endif + next= block->next_changed; + if (block->hash_link->file.file == file->file) + { + /* + Mark the block with BLOCK_IN_FLUSH in order not to let + other threads to use it for new pages and interfere with + our sequence ot flushing dirty file pages + */ + block->status|= BLOCK_IN_FLUSH; + + if (! (block->status & BLOCK_IN_SWITCH)) + { + /* + We care only for the blocks for which flushing was not + initiated by other threads as a result of page swapping + */ + reg_requests(pagecache, block, 1); + if (type != FLUSH_IGNORE_CHANGED) + { + /* It's not a temporary file */ + if (pos == end) + { + /* + This happens only if there is not enough + memory for the big block + */ + if ((error= flush_cached_blocks(pagecache, file, cache, + end,type))) + last_errno=error; + /* + Restart the scan as some other thread might have changed + the changed blocks chain: the blocks that were in switch + state before the flush started have to be excluded + */ + goto restart; + } + *pos++= block; + } + else + { + /* It's a temporary file */ + pagecache->blocks_changed--; + pagecache->global_blocks_changed--; + free_block(pagecache, block); + } + } + else + { + /* Link the block into a list of blocks 'in switch' */ + unlink_changed(block); + link_changed(block, &first_in_switch); + } + } + } + if (pos != cache) + { + if ((error= flush_cached_blocks(pagecache, file, cache, pos, type))) + last_errno= error; + } + /* Wait until list of blocks in switch is empty */ + while (first_in_switch) + { +#if defined(PAGECACHE_DEBUG) + cnt= 0; +#endif + block= first_in_switch; + { +#ifdef THREAD + struct st_my_thread_var *thread= my_thread_var; + add_to_queue(&block->wqueue[COND_FOR_SAVED], thread); + do + { + KEYCACHE_DBUG_PRINT("flush_pagecache_blocks_int: wait", + ("suspend thread %ld", thread->id)); + pagecache_pthread_cond_wait(&thread->suspend, + &pagecache->cache_lock); + } + while (thread->next); +#else + KEYCACHE_DBUG_ASSERT(0); + /* No parallel requests in single-threaded case */ +#endif + } +#if defined(PAGECACHE_DEBUG) + cnt++; + KEYCACHE_DBUG_ASSERT(cnt <= pagecache->blocks_used); +#endif + } + /* The following happens very seldom */ + if (! (type == FLUSH_KEEP || type == FLUSH_FORCE_WRITE)) + { +#if defined(PAGECACHE_DEBUG) + cnt=0; +#endif + for (block= pagecache->file_blocks[FILE_HASH(*file)] ; + block; + block= next) + { +#if defined(PAGECACHE_DEBUG) + cnt++; + KEYCACHE_DBUG_ASSERT(cnt <= pagecache->blocks_used); +#endif + next= block->next_changed; + if (block->hash_link->file.file == file->file && + (! (block->status & BLOCK_CHANGED) + || type == FLUSH_IGNORE_CHANGED)) + { + reg_requests(pagecache, block, 1); + free_block(pagecache, block); + } + } + } + } + +#ifndef DBUG_OFF + DBUG_EXECUTE("check_pagecache", + test_key_cache(pagecache, "end of flush_pagecache_blocks", 0);); +#endif + if (cache != cache_buff) + my_free((gptr) cache, MYF(0)); + if (last_errno) + errno=last_errno; /* Return first error */ + DBUG_RETURN(last_errno != 0); +} + + +/* + Flush all blocks for a file to disk + + SYNOPSIS + + flush_pagecache_blocks() + pagecache pointer to a page cache data structure + file handler for the file to flush to + flush_type type of the flush + + RETURN + 0 ok + 1 error +*/ + +int flush_pagecache_blocks(PAGECACHE *pagecache, + PAGECACHE_FILE *file, enum flush_type type) +{ + int res; + DBUG_ENTER("flush_pagecache_blocks"); + DBUG_PRINT("enter", ("pagecache: 0x%lx", pagecache)); + + if (pagecache->disk_blocks <= 0) + DBUG_RETURN(0); + pagecache_pthread_mutex_lock(&pagecache->cache_lock); + inc_counter_for_resize_op(pagecache); + res= flush_pagecache_blocks_int(pagecache, file, type); + dec_counter_for_resize_op(pagecache); + pagecache_pthread_mutex_unlock(&pagecache->cache_lock); + DBUG_RETURN(res); +} + + +/* + Flush all blocks in the key cache to disk +*/ + +static int flush_all_key_blocks(PAGECACHE *pagecache) +{ +#if defined(PAGECACHE_DEBUG) + uint cnt=0; +#endif + while (pagecache->blocks_changed > 0) + { + PAGECACHE_BLOCK_LINK *block; + for (block= pagecache->used_last->next_used ; ; block=block->next_used) + { + if (block->hash_link) + { +#if defined(PAGECACHE_DEBUG) + cnt++; + KEYCACHE_DBUG_ASSERT(cnt <= pagecache->blocks_used); +#endif + if (flush_pagecache_blocks_int(pagecache, &block->hash_link->file, + FLUSH_RELEASE)) + return 1; + break; + } + if (block == pagecache->used_last) + break; + } + } + return 0; +} + + +/* + Reset the counters of a key cache. + + SYNOPSIS + reset_key_cache_counters() + name the name of a key cache + key_cache pointer to the key kache to be reset + + DESCRIPTION + This procedure is used by process_key_caches() to reset the counters of all + currently used key caches, both the default one and the named ones. + + RETURN + 0 on success (always because it can't fail) +*/ + +int reset_key_cache_counters(const char *name, PAGECACHE *key_cache) +{ + DBUG_ENTER("reset_key_cache_counters"); + if (!key_cache->inited) + { + DBUG_PRINT("info", ("Key cache %s not initialized.", name)); + DBUG_RETURN(0); + } + DBUG_PRINT("info", ("Resetting counters for key cache %s.", name)); + + key_cache->global_blocks_changed= 0; /* Key_blocks_not_flushed */ + key_cache->global_cache_r_requests= 0; /* Key_read_requests */ + key_cache->global_cache_read= 0; /* Key_reads */ + key_cache->global_cache_w_requests= 0; /* Key_write_requests */ + key_cache->global_cache_write= 0; /* Key_writes */ + DBUG_RETURN(0); +} + + +#ifndef DBUG_OFF +/* + Test if disk-cache is ok +*/ +static void test_key_cache(PAGECACHE *pagecache __attribute__((unused)), + const char *where __attribute__((unused)), + my_bool lock __attribute__((unused))) +{ + /* TODO */ +} +#endif + +#if defined(PAGECACHE_TIMEOUT) + +#define KEYCACHE_DUMP_FILE "pagecache_dump.txt" +#define MAX_QUEUE_LEN 100 + + +static void pagecache_dump(PAGECACHE *pagecache) +{ + FILE *pagecache_dump_file=fopen(KEYCACHE_DUMP_FILE, "w"); + struct st_my_thread_var *last; + struct st_my_thread_var *thread; + PAGECACHE_BLOCK_LINK *block; + PAGECACHE_HASH_LINK *hash_link; + PAGECACHE_PAGE *page; + uint i; + + fprintf(pagecache_dump_file, "thread:%u\n", thread->id); + + i=0; + thread=last=waiting_for_hash_link.last_thread; + fprintf(pagecache_dump_file, "queue of threads waiting for hash link\n"); + if (thread) + do + { + thread= thread->next; + page= (PAGECACHE_PAGE *) thread->opt_info; + fprintf(pagecache_dump_file, + "thread:%u, (file,pageno)=(%u,%lu)\n", + thread->id,(uint) page->file.file,(ulong) page->pageno); + if (++i == MAX_QUEUE_LEN) + break; + } + while (thread != last); + + i=0; + thread=last=waiting_for_block.last_thread; + fprintf(pagecache_dump_file, "queue of threads waiting for block\n"); + if (thread) + do + { + thread=thread->next; + hash_link= (PAGECACHE_HASH_LINK *) thread->opt_info; + fprintf(pagecache_dump_file, + "thread:%u hash_link:%u (file,pageno)=(%u,%lu)\n", + thread->id, (uint) PAGECACHE_HASH_LINK_NUMBER(pagecache, hash_link), + (uint) hash_link->file.file,(ulong) hash_link->pageno); + if (++i == MAX_QUEUE_LEN) + break; + } + while (thread != last); + + for (i=0 ; i < pagecache->blocks_used ; i++) + { + int j; + block= &pagecache->block_root[i]; + hash_link= block->hash_link; + fprintf(pagecache_dump_file, + "block:%u hash_link:%d status:%x #requests=%u waiting_for_readers:%d\n", + i, (int) (hash_link ? + PAGECACHE_HASH_LINK_NUMBER(pagecache, hash_link) : + -1), + block->status, block->requests, block->condvar ? 1 : 0); + for (j=0 ; j < COND_SIZE; j++) + { + PAGECACHE_WQUEUE *wqueue=&block->wqueue[j]; + thread= last= wqueue->last_thread; + fprintf(pagecache_dump_file, "queue #%d\n", j); + if (thread) + { + do + { + thread=thread->next; + fprintf(pagecache_dump_file, + "thread:%u\n", thread->id); + if (++i == MAX_QUEUE_LEN) + break; + } + while (thread != last); + } + } + } + fprintf(pagecache_dump_file, "LRU chain:"); + block= pagecache= used_last; + if (block) + { + do + { + block= block->next_used; + fprintf(pagecache_dump_file, + "block:%u, ", BLOCK_NUMBER(pagecache, block)); + } + while (block != pagecache->used_last); + } + fprintf(pagecache_dump_file, "\n"); + + fclose(pagecache_dump_file); +} + +#endif /* defined(PAGECACHE_TIMEOUT) */ + +#if defined(PAGECACHE_TIMEOUT) && !defined(__WIN__) + + +static int pagecache_pthread_cond_wait(pthread_cond_t *cond, + pthread_mutex_t *mutex) +{ + int rc; + struct timeval now; /* time when we started waiting */ + struct timespec timeout; /* timeout value for the wait function */ + struct timezone tz; +#if defined(PAGECACHE_DEBUG) + int cnt=0; +#endif + + /* Get current time */ + gettimeofday(&now, &tz); + /* Prepare timeout value */ + timeout.tv_sec= now.tv_sec + PAGECACHE_TIMEOUT; + /* + timeval uses microseconds. + timespec uses nanoseconds. + 1 nanosecond = 1000 micro seconds + */ + timeout.tv_nsec= now.tv_usec * 1000; + KEYCACHE_THREAD_TRACE_END("started waiting"); +#if defined(PAGECACHE_DEBUG) + cnt++; + if (cnt % 100 == 0) + fprintf(pagecache_debug_log, "waiting...\n"); + fflush(pagecache_debug_log); +#endif + rc= pthread_cond_timedwait(cond, mutex, &timeout); + KEYCACHE_THREAD_TRACE_BEGIN("finished waiting"); + if (rc == ETIMEDOUT || rc == ETIME) + { +#if defined(PAGECACHE_DEBUG) + fprintf(pagecache_debug_log,"aborted by pagecache timeout\n"); + fclose(pagecache_debug_log); + abort(); +#endif + pagecache_dump(); + } + +#if defined(PAGECACHE_DEBUG) + KEYCACHE_DBUG_ASSERT(rc != ETIMEDOUT); +#else + assert(rc != ETIMEDOUT); +#endif + return rc; +} +#else +#if defined(PAGECACHE_DEBUG) +static int pagecache_pthread_cond_wait(pthread_cond_t *cond, + pthread_mutex_t *mutex) +{ + int rc; + KEYCACHE_THREAD_TRACE_END("started waiting"); + rc= pthread_cond_wait(cond, mutex); + KEYCACHE_THREAD_TRACE_BEGIN("finished waiting"); + return rc; +} +#endif +#endif /* defined(PAGECACHE_TIMEOUT) && !defined(__WIN__) */ + +#if defined(PAGECACHE_DEBUG) +static int ___pagecache_pthread_mutex_lock(pthread_mutex_t *mutex) +{ + int rc; + rc= pthread_mutex_lock(mutex); + KEYCACHE_THREAD_TRACE_BEGIN(""); + return rc; +} + + +static void ___pagecache_pthread_mutex_unlock(pthread_mutex_t *mutex) +{ + KEYCACHE_THREAD_TRACE_END(""); + pthread_mutex_unlock(mutex); + return; +} + + +static int ___pagecache_pthread_cond_signal(pthread_cond_t *cond) +{ + int rc; + KEYCACHE_THREAD_TRACE("signal"); + rc= pthread_cond_signal(cond); + return rc; +} + + +#if defined(PAGECACHE_DEBUG_LOG) + + +static void pagecache_debug_print(const char * fmt, ...) +{ + va_list args; + va_start(args,fmt); + if (pagecache_debug_log) + { + VOID(vfprintf(pagecache_debug_log, fmt, args)); + VOID(fputc('\n',pagecache_debug_log)); + } + va_end(args); +} +#endif /* defined(PAGECACHE_DEBUG_LOG) */ + +#if defined(PAGECACHE_DEBUG_LOG) + + +void pagecache_debug_log_close(void) +{ + if (pagecache_debug_log) + fclose(pagecache_debug_log); +} +#endif /* defined(PAGECACHE_DEBUG_LOG) */ + +#endif /* defined(PAGECACHE_DEBUG) */ diff --git a/mysys/test_file.c b/mysys/test_file.c new file mode 100644 index 00000000000..7ffca48023d --- /dev/null +++ b/mysys/test_file.c @@ -0,0 +1,70 @@ +#include "mysys_priv.h" +#include "my_dir.h" +#include +#include +#include +#include "test_file.h" + + +/* + Check that file contance correspond to descriptor + + SYNOPSIS + test_file() + file File to test + file_name Path (and name) of file which is tested + size size of file + buff_size size of buffer which is enought to check the file + desc file descriptor to check with + + RETURN + 1 file if OK + 0 error +*/ + +int test_file(PAGECACHE_FILE file, char *file_name, + off_t size, size_t buff_size, struct file_desc *desc) +{ + MY_STAT stat_buff, *stat; + unsigned char *buffr= malloc(buff_size); + off_t pos= 0; + size_t byte; + int step= 0; + + if ((stat= my_stat(file_name, &stat_buff, MYF(0))) == NULL) + { + diag("Can't stat() %s (errno: %d)\n", file_name, errno); + return 0; + } + if (stat->st_size != size) + { + diag("file %s size is %lu (should be %lu)\n", + file_name, (ulong) stat->st_size, (ulong) size); + return 0; + } + /* check content */ + my_seek(file.file, 0, SEEK_SET, MYF(0)); + while (desc[step].length != 0) + { + if (my_read(file.file, (char*)buffr, desc[step].length, MYF(0)) != + desc[step].length) + { + diag("Can't read %u bytes from %s (errno: %d)\n", + (uint)desc[step].length, file_name, errno); + return 0; + } + for (byte= 0; byte < desc[step].length; byte++) + { + if (buffr[byte] != desc[step].content) + { + diag("content of %s mismatch 0x%x in position %lu instead of 0x%x\n", + file_name, (uint) buffr[byte], (ulong) (pos + byte), + desc[step].content); + return 0; + } + } + pos+= desc[step].length; + step++; + } + return 1; +} diff --git a/mysys/test_file.h b/mysys/test_file.h new file mode 100644 index 00000000000..ea787c123ed --- /dev/null +++ b/mysys/test_file.h @@ -0,0 +1,14 @@ + +#include + +/* + File content descriptor +*/ +struct file_desc +{ + unsigned int length; + unsigned char content; +}; + +int test_file(PAGECACHE_FILE file, char *file_name, + off_t size, size_t buff_size, struct file_desc *desc); diff --git a/mysys/test_pagecache_consist.c b/mysys/test_pagecache_consist.c new file mode 100755 index 00000000000..1cc54af2460 --- /dev/null +++ b/mysys/test_pagecache_consist.c @@ -0,0 +1,447 @@ +#include "mysys_priv.h" +#include "../include/my_pthread.h" +#include "../include/pagecache.h" +#include +#include "my_dir.h" +#include +#include +#include +#include +#include "../unittest/mytap/tap.h" +#include + +/*#define PAGE_SIZE 65536*/ +#define PCACHE_SIZE (PAGE_SIZE*1024*20) + +#ifndef DBUG_OFF +static const char* default_dbug_option; +#endif + +static char *file1_name= (char*)"page_cache_test_file_1"; +static PAGECACHE_FILE file1; +static pthread_cond_t COND_thread_count; +static pthread_mutex_t LOCK_thread_count; +static uint thread_count; +static PAGECACHE pagecache; + +#ifdef TEST_HIGH_CONCURENCY +static uint number_of_readers= 10; +static uint number_of_writers= 20; +static uint number_of_tests= 30000; +static uint record_length_limit= PAGE_SIZE/200; +static uint number_of_pages= 20; +static uint flush_divider= 1000; +#else /*TEST_HIGH_CONCURENCY*/ +#ifdef TEST_READERS +static uint number_of_readers= 10; +static uint number_of_writers= 1; +static uint number_of_tests= 30000; +static uint record_length_limit= PAGE_SIZE/200; +static uint number_of_pages= 20; +static uint flush_divider= 1000; +#else /*TEST_READERS*/ +#ifdef TEST_WRITERS +static uint number_of_readers= 0; +static uint number_of_writers= 10; +static uint number_of_tests= 30000; +static uint record_length_limit= PAGE_SIZE/200; +static uint number_of_pages= 20; +static uint flush_divider= 1000; +#else /*TEST_WRITERS*/ +static uint number_of_readers= 10; +static uint number_of_writers= 10; +static uint number_of_tests= 50000; +static uint record_length_limit= PAGE_SIZE/200; +static uint number_of_pages= 20000; +static uint flush_divider= 1000; +#endif /*TEST_WRITERS*/ +#endif /*TEST_READERS*/ +#endif /*TEST_HIGH_CONCURENCY*/ + + +/* check page consistemcy */ +uint check_page(uchar *buff, ulong offset, int page_locked, int page_no, + int tag) +{ + uint end= sizeof(uint); + uint num= *((uint *)buff); + uint i; + DBUG_ENTER("check_page"); + + for (i= 0; i < num; i++) + { + uint len= *((uint *)(buff + end)); + uint j; + end+= sizeof(uint)+ sizeof(uint); + if (len + end > PAGE_SIZE) + { + diag("incorrect field header #%u by offset %lu\n", i, offset + end + j); + goto err; + } + for(j= 0; j < len; j++) + { + if (buff[end + j] != (uchar)((i+1) % 256)) + { + diag("incorrect %lu byte\n", offset + end + j); + goto err; + } + } + end+= len; + } + for(i= end; i < PAGE_SIZE; i++) + { + if (buff[i] != 0) + { + int h; + DBUG_PRINT("err", + ("byte %lu (%lu + %u), page %u (%s, end: %u, recs: %u, tag: %d) should be 0\n", + offset + i, offset, i, page_no, + (page_locked ? "locked" : "unlocked"), + end, num, tag)); + diag("byte %lu (%lu + %u), page %u (%s, end: %u, recs: %u, tag: %d) should be 0\n", + offset + i, offset, i, page_no, + (page_locked ? "locked" : "unlocked"), + end, num, tag); + h= my_open("wrong_page", O_CREAT | O_TRUNC | O_RDWR, MYF(0)); + my_pwrite(h, buff, PAGE_SIZE, 0, MYF(0)); + my_close(h, MYF(0)); + goto err; + } + } + DBUG_RETURN(end); +err: + DBUG_PRINT("err", ("try to flush")); + if (page_locked) + { + pagecache_delete_page(&pagecache, &file1, page_no, + PAGECACHE_LOCK_LEFT_WRITELOCKED, 1); + } + else + { + flush_pagecache_blocks(&pagecache, &file1, FLUSH_RELEASE); + } + exit(1); +} + +void put_rec(uchar *buff, uint end, uint len, uint tag) +{ + uint i; + uint num= *((uint *)buff); + if (!len) + len= 1; + if (end + sizeof(uint)*2 + len > PAGE_SIZE) + return; + *((uint *)(buff + end))= len; + end+= sizeof(uint); + *((uint *)(buff + end))= tag; + end+= sizeof(uint); + num++; + *((uint *)buff)= num; + *((uint*)(buff + end))= len; + for (i= end; i < (len + end); i++) + { + buff[i]= (uchar) num % 256; + } +} + +/* + Recreate and reopen a file for test + + SYNOPSIS + reset_file() + file File to reset + file_name Path (and name) of file which should be reset +*/ + +void reset_file(PAGECACHE_FILE file, char *file_name) +{ + flush_pagecache_blocks(&pagecache, &file1, FLUSH_RELEASE); + if (my_close(file1.file, MYF(0)) != 0) + { + diag("Got error during %s closing from close() (errno: %d)\n", + file_name, errno); + exit(1); + } + my_delete(file_name, MYF(0)); + if ((file.file= my_open(file_name, + O_CREAT | O_TRUNC | O_RDWR, MYF(0))) == -1) + { + diag("Got error during %s creation from open() (errno: %d)\n", + file_name, errno); + exit(1); + } +} + + +void reader(int num) +{ + unsigned char *buffr= malloc(PAGE_SIZE); + uint i; + + for (i= 0; i < number_of_tests; i++) + { + uint page= rand()/(RAND_MAX/number_of_pages); + pagecache_read(&pagecache, &file1, page, 3, (char*)buffr, + PAGECACHE_PLAIN_PAGE, + PAGECACHE_LOCK_LEFT_UNLOCKED, + 0); + check_page(buffr, page * PAGE_SIZE, 0, page, -num); + if (i % 500 == 0) + printf("reader%d: %d\n", num, i); + + } + printf("reader%d: done\n", num); + free(buffr); +} + + +void writer(int num) +{ + unsigned char *buffr= malloc(PAGE_SIZE); + uint i; + + for (i= 0; i < number_of_tests; i++) + { + uint end; + uint page= rand()/(RAND_MAX/number_of_pages); + pagecache_read(&pagecache, &file1, page, 3, (char*)buffr, + PAGECACHE_PLAIN_PAGE, + PAGECACHE_LOCK_WRITE, + 0); + end= check_page(buffr, page * PAGE_SIZE, 1, page, num); + put_rec(buffr, end, rand()/(RAND_MAX/record_length_limit), num); + pagecache_write(&pagecache, &file1, page, 3, (char*)buffr, + PAGECACHE_PLAIN_PAGE, + PAGECACHE_LOCK_WRITE_UNLOCK, + PAGECACHE_UNPIN, + PAGECACHE_WRITE_DELAY, + 0); + + if (i % flush_divider == 0) + flush_pagecache_blocks(&pagecache, &file1, FLUSH_FORCE_WRITE); + if (i % 500 == 0) + printf("writer%d: %d\n", num, i); + } + printf("writer%d: done\n", num); + free(buffr); +} + + +static void *test_thread_reader(void *arg) +{ + int param=*((int*) arg); + + my_thread_init(); + DBUG_ENTER("test_reader"); + DBUG_PRINT("enter", ("param: %d", param)); + + reader(param); + + DBUG_PRINT("info", ("Thread %s ended\n", my_thread_name())); + pthread_mutex_lock(&LOCK_thread_count); + thread_count--; + VOID(pthread_cond_signal(&COND_thread_count)); /* Tell main we are ready */ + pthread_mutex_unlock(&LOCK_thread_count); + free((gptr) arg); + my_thread_end(); + DBUG_RETURN(0); +} + +static void *test_thread_writer(void *arg) +{ + int param=*((int*) arg); + + my_thread_init(); + DBUG_ENTER("test_writer"); + DBUG_PRINT("enter", ("param: %d", param)); + + writer(param); + + DBUG_PRINT("info", ("Thread %s ended\n", my_thread_name())); + pthread_mutex_lock(&LOCK_thread_count); + thread_count--; + VOID(pthread_cond_signal(&COND_thread_count)); /* Tell main we are ready */ + pthread_mutex_unlock(&LOCK_thread_count); + free((gptr) arg); + my_thread_end(); + DBUG_RETURN(0); +} + +int main(int argc, char **argv __attribute__((unused))) +{ + pthread_t tid; + pthread_attr_t thr_attr; + int *param, error, pagen; + + MY_INIT(argv[0]); + +#ifndef DBUG_OFF +#if defined(__WIN__) + default_dbug_option= "d:t:i:O,\\test_pagecache_consist.trace"; +#else + default_dbug_option= "d:t:i:o,/tmp/test_pagecache_consist.trace"; +#endif + if (argc > 1) + { + DBUG_SET(default_dbug_option); + DBUG_SET_INITIAL(default_dbug_option); + } +#endif + + + DBUG_ENTER("main"); + DBUG_PRINT("info", ("Main thread: %s\n", my_thread_name())); + if ((file1.file= my_open(file1_name, + O_CREAT | O_TRUNC | O_RDWR, MYF(0))) == -1) + { + fprintf(stderr, "Got error during file1 creation from open() (errno: %d)\n", + errno); + exit(1); + } + DBUG_PRINT("info", ("file1: %d", file1.file)); + if (chmod(file1_name, S_IRWXU | S_IRWXG | S_IRWXO) != 0) + { + fprintf(stderr, "Got error during file1 chmod() (errno: %d)\n", + errno); + exit(1); + } + my_pwrite(file1.file, "test file", 9, 0, MYF(0)); + + if ((error= pthread_cond_init(&COND_thread_count, NULL))) + { + fprintf(stderr, "COND_thread_count: %d from pthread_cond_init (errno: %d)\n", + error, errno); + exit(1); + } + if ((error= pthread_mutex_init(&LOCK_thread_count, MY_MUTEX_INIT_FAST))) + { + fprintf(stderr, "LOCK_thread_count: %d from pthread_cond_init (errno: %d)\n", + error, errno); + exit(1); + } + + if ((error= pthread_attr_init(&thr_attr))) + { + fprintf(stderr,"Got error: %d from pthread_attr_init (errno: %d)\n", + error,errno); + exit(1); + } + if ((error= pthread_attr_setdetachstate(&thr_attr, PTHREAD_CREATE_DETACHED))) + { + fprintf(stderr, + "Got error: %d from pthread_attr_setdetachstate (errno: %d)\n", + error,errno); + exit(1); + } + +#ifndef pthread_attr_setstacksize /* void return value */ + if ((error= pthread_attr_setstacksize(&thr_attr, 65536L))) + { + fprintf(stderr,"Got error: %d from pthread_attr_setstacksize (errno: %d)\n", + error,errno); + exit(1); + } +#endif +#ifdef HAVE_THR_SETCONCURRENCY + VOID(thr_setconcurrency(2)); +#endif + + my_thread_global_init(); + + + if ((pagen= init_pagecache(&pagecache, PCACHE_SIZE, 0, 0, + PAGE_SIZE, 0)) == 0) + { + fprintf(stderr,"Got error: init_pagecache() (errno: %d)\n", + errno); + exit(1); + } + DBUG_PRINT("info", ("Page cache %d pages", pagen)); + { + unsigned char *buffr= malloc(PAGE_SIZE); + uint i; + memset(buffr, '\0', PAGE_SIZE); + for (i= 0; i < number_of_pages; i++) + { + pagecache_write(&pagecache, &file1, i, 3, (char*)buffr, + PAGECACHE_PLAIN_PAGE, + PAGECACHE_LOCK_LEFT_UNLOCKED, + PAGECACHE_PIN_LEFT_UNPINNED, + PAGECACHE_WRITE_DELAY, + 0); + } + flush_pagecache_blocks(&pagecache, &file1, FLUSH_FORCE_WRITE); + free(buffr); + } + if ((error= pthread_mutex_lock(&LOCK_thread_count))) + { + fprintf(stderr,"LOCK_thread_count: %d from pthread_mutex_lock (errno: %d)\n", + error,errno); + exit(1); + } + while (number_of_readers != 0 || number_of_writers != 0) + { + if (number_of_readers != 0) + { + param=(int*) malloc(sizeof(int)); + *param= number_of_readers; + if ((error= pthread_create(&tid, &thr_attr, test_thread_reader, + (void*) param))) + { + fprintf(stderr,"Got error: %d from pthread_create (errno: %d)\n", + error,errno); + exit(1); + } + thread_count++; + number_of_readers--; + } + if (number_of_writers != 0) + { + param=(int*) malloc(sizeof(int)); + *param= number_of_writers; + if ((error= pthread_create(&tid, &thr_attr, test_thread_writer, + (void*) param))) + { + fprintf(stderr,"Got error: %d from pthread_create (errno: %d)\n", + error,errno); + exit(1); + } + thread_count++; + number_of_writers--; + } + } + DBUG_PRINT("info", ("Thread started")); + pthread_mutex_unlock(&LOCK_thread_count); + + pthread_attr_destroy(&thr_attr); + + /* wait finishing */ + if ((error= pthread_mutex_lock(&LOCK_thread_count))) + fprintf(stderr,"LOCK_thread_count: %d from pthread_mutex_lock\n",error); + while (thread_count) + { + if ((error= pthread_cond_wait(&COND_thread_count,&LOCK_thread_count))) + fprintf(stderr,"COND_thread_count: %d from pthread_cond_wait\n",error); + } + if ((error= pthread_mutex_unlock(&LOCK_thread_count))) + fprintf(stderr,"LOCK_thread_count: %d from pthread_mutex_unlock\n",error); + DBUG_PRINT("info", ("thread ended")); + + end_pagecache(&pagecache, 1); + DBUG_PRINT("info", ("Page cache ended")); + + if (my_close(file1.file, MYF(0)) != 0) + { + fprintf(stderr, "Got error during file1 closing from close() (errno: %d)\n", + errno); + exit(1); + } + /*my_delete(file1_name, MYF(0));*/ + my_thread_global_end(); + + DBUG_PRINT("info", ("file1 (%d) closed", file1.file)); + + DBUG_PRINT("info", ("Program end")); + + DBUG_RETURN(0); +} diff --git a/mysys/test_pagecache_single.c b/mysys/test_pagecache_single.c new file mode 100644 index 00000000000..9df7844cfa5 --- /dev/null +++ b/mysys/test_pagecache_single.c @@ -0,0 +1,589 @@ +#include "mysys_priv.h" +#include "../include/my_pthread.h" +#include "../include/pagecache.h" +#include "my_dir.h" +#include +#include +#include +#include +#include +#include "../unittest/mytap/tap.h" +#include "test_file.h" + +/* #define PAGE_SIZE 1024 */ +#define PCACHE_SIZE (PAGE_SIZE*1024*10) + +#ifndef DBUG_OFF +static const char* default_dbug_option; +#endif + +static char *file1_name= (char*)"page_cache_test_file_1"; +static PAGECACHE_FILE file1; +static pthread_cond_t COND_thread_count; +static pthread_mutex_t LOCK_thread_count; +static uint thread_count; +static PAGECACHE pagecache; + +/* + File contance descriptors +*/ +static struct file_desc simple_read_write_test_file[]= +{ + {PAGE_SIZE, '\1'}, + { 0, 0} +}; +static struct file_desc simple_read_change_write_read_test_file[]= +{ + {PAGE_SIZE/2, '\65'}, + {PAGE_SIZE/2, '\1'}, + { 0, 0} +}; +static struct file_desc simple_pin_test_file1[]= +{ + {PAGE_SIZE*2, '\1'}, + { 0, 0} +}; +static struct file_desc simple_pin_test_file2[]= +{ + {PAGE_SIZE/2, '\1'}, + {PAGE_SIZE/2, (unsigned char)129}, + {PAGE_SIZE, '\1'}, + { 0, 0} +}; +static struct file_desc simple_delete_forget_test_file[]= +{ + {PAGE_SIZE, '\1'}, + { 0, 0} +}; +static struct file_desc simple_delete_flush_test_file[]= +{ + {PAGE_SIZE, '\2'}, + { 0, 0} +}; + + +/* + Recreate and reopen a file for test + + SYNOPSIS + reset_file() + file File to reset + file_name Path (and name) of file which should be reset +*/ + +void reset_file(PAGECACHE_FILE file, char *file_name) +{ + flush_pagecache_blocks(&pagecache, &file1, FLUSH_RELEASE); + if (my_close(file1.file, MYF(0)) != 0) + { + diag("Got error during %s closing from close() (errno: %d)\n", + file_name, errno); + exit(1); + } + my_delete(file_name, MYF(0)); + if ((file.file= my_open(file_name, + O_CREAT | O_TRUNC | O_RDWR, MYF(0))) == -1) + { + diag("Got error during %s creation from open() (errno: %d)\n", + file_name, errno); + exit(1); + } +} + +/* + Write then read page, check file on disk +*/ + +int simple_read_write_test() +{ + unsigned char *buffw= malloc(PAGE_SIZE); + unsigned char *buffr= malloc(PAGE_SIZE); + int res; + DBUG_ENTER("simple_read_write_test"); + memset(buffw, '\1', PAGE_SIZE); + pagecache_write(&pagecache, &file1, 0, 3, (char*)buffw, + PAGECACHE_PLAIN_PAGE, + PAGECACHE_LOCK_LEFT_UNLOCKED, + PAGECACHE_PIN_LEFT_UNPINNED, + PAGECACHE_WRITE_DELAY, + 0); + pagecache_read(&pagecache, &file1, 0, 3, (char*)buffr, + PAGECACHE_PLAIN_PAGE, + PAGECACHE_LOCK_LEFT_UNLOCKED, + 0); + ok((res= test(memcmp(buffr, buffw, PAGE_SIZE) == 0)), + "Simple write-read page "); + flush_pagecache_blocks(&pagecache, &file1, FLUSH_FORCE_WRITE); + ok((res&= test(test_file(file1, file1_name, PAGE_SIZE, PAGE_SIZE, + simple_read_write_test_file))), + "Simple write-read page file"); + if (res) + reset_file(file1, file1_name); + free(buffw); + free(buffr); + DBUG_RETURN(res); +} + + +/* + Prepare page, then read (and lock), change (write new value and unlock), + then check the page in the cache and on the disk +*/ +int simple_read_change_write_read_test() +{ + unsigned char *buffw= malloc(PAGE_SIZE); + unsigned char *buffr= malloc(PAGE_SIZE); + int res; + DBUG_ENTER("simple_read_change_write_read_test"); + /* prepare the file */ + memset(buffw, '\1', PAGE_SIZE); + pagecache_write(&pagecache, &file1, 0, 3, (char*)buffw, + PAGECACHE_PLAIN_PAGE, + PAGECACHE_LOCK_LEFT_UNLOCKED, + PAGECACHE_PIN_LEFT_UNPINNED, + PAGECACHE_WRITE_DELAY, + 0); + flush_pagecache_blocks(&pagecache, &file1, FLUSH_FORCE_WRITE); + /* test */ + pagecache_read(&pagecache, &file1, 0, 3, (char*)buffw, + PAGECACHE_PLAIN_PAGE, + PAGECACHE_LOCK_WRITE, + 0); + memset(buffw, '\65', PAGE_SIZE/2); + pagecache_write(&pagecache, &file1, 0, 3, (char*)buffw, + PAGECACHE_PLAIN_PAGE, + PAGECACHE_LOCK_WRITE_UNLOCK, + PAGECACHE_UNPIN, + PAGECACHE_WRITE_DELAY, + 0); + + pagecache_read(&pagecache, &file1, 0, 3, (char*)buffr, + PAGECACHE_PLAIN_PAGE, + PAGECACHE_LOCK_LEFT_UNLOCKED, + 0); + ok((res= test(memcmp(buffr, buffw, PAGE_SIZE) == 0)), + "Simple read-change-write-read page "); + flush_pagecache_blocks(&pagecache, &file1, FLUSH_FORCE_WRITE); + ok((res&= test(test_file(file1, file1_name, PAGE_SIZE, PAGE_SIZE, + simple_read_change_write_read_test_file))), + "Simple read-change-write-read page file"); + if (res) + reset_file(file1, file1_name); + free(buffw); + free(buffr); + DBUG_RETURN(res); +} + + +/* + Prepare page, read page 0 (and pin) then write page 1 and page 0. + Flush the file (shold flush only page 1 and return 1 (page 0 is + still pinned). + Check file on the disk. + Unpin and flush. + Check file on the disk. +*/ +int simple_pin_test() +{ + unsigned char *buffw= malloc(PAGE_SIZE); + unsigned char *buffr= malloc(PAGE_SIZE); + int res; + DBUG_ENTER("simple_pin_test"); + /* prepare the file */ + memset(buffw, '\1', PAGE_SIZE); + pagecache_write(&pagecache, &file1, 0, 3, (char*)buffw, + PAGECACHE_PLAIN_PAGE, + PAGECACHE_LOCK_LEFT_UNLOCKED, + PAGECACHE_PIN_LEFT_UNPINNED, + PAGECACHE_WRITE_DELAY, + 0); + /* test */ + if (flush_pagecache_blocks(&pagecache, &file1, FLUSH_FORCE_WRITE)) + { + diag("error in flush_pagecache_blocks\n"); + exit(1); + } + pagecache_read(&pagecache, &file1, 0, 3, (char*)buffw, + PAGECACHE_PLAIN_PAGE, + PAGECACHE_LOCK_WRITE, + 0); + pagecache_write(&pagecache, &file1, 1, 3, (char*)buffw, + PAGECACHE_PLAIN_PAGE, + PAGECACHE_LOCK_LEFT_UNLOCKED, + PAGECACHE_PIN_LEFT_UNPINNED, + PAGECACHE_WRITE_DELAY, + 0); + memset(buffw + PAGE_SIZE/2, ((unsigned char) 129), PAGE_SIZE/2); + pagecache_write(&pagecache, &file1, 0, 3, (char*)buffw, + PAGECACHE_PLAIN_PAGE, + PAGECACHE_LOCK_WRITE_TO_READ, + PAGECACHE_PIN_LEFT_PINNED, + PAGECACHE_WRITE_DELAY, + 0); + /* + We have to get error because one page of the file is pinned, + other page should be flushed + */ + if (!flush_pagecache_blocks(&pagecache, &file1, FLUSH_FORCE_WRITE)) + { + diag("Did not get error in flush_pagecache_blocks\n"); + res= 0; + goto err; + } + ok((res= test(test_file(file1, file1_name, PAGE_SIZE*2, PAGE_SIZE*2, + simple_pin_test_file1))), + "Simple pin page file with pin"); + pagecache_unlock_page(&pagecache, + &file1, + 0, + PAGECACHE_LOCK_READ_UNLOCK, + PAGECACHE_UNPIN, + 0, 0); + if (flush_pagecache_blocks(&pagecache, &file1, FLUSH_FORCE_WRITE)) + { + diag("Got error in flush_pagecache_blocks\n"); + res= 0; + goto err; + } + ok((res&= test(test_file(file1, file1_name, PAGE_SIZE*2, PAGE_SIZE, + simple_pin_test_file2))), + "Simple pin page result file"); + if (res) + reset_file(file1, file1_name); +err: + free(buffw); + free(buffr); + DBUG_RETURN(res); +} + +/* + Prepare page, write new value, then delete page from cache without flush, + on the disk should be page with old content written during preparation +*/ + +int simple_delete_forget_test() +{ + unsigned char *buffw= malloc(PAGE_SIZE); + unsigned char *buffr= malloc(PAGE_SIZE); + int res; + DBUG_ENTER("simple_delete_forget_test"); + /* prepare the file */ + memset(buffw, '\1', PAGE_SIZE); + pagecache_write(&pagecache, &file1, 0, 3, (char*)buffw, + PAGECACHE_PLAIN_PAGE, + PAGECACHE_LOCK_LEFT_UNLOCKED, + PAGECACHE_PIN_LEFT_UNPINNED, + PAGECACHE_WRITE_DELAY, + 0); + flush_pagecache_blocks(&pagecache, &file1, FLUSH_FORCE_WRITE); + /* test */ + memset(buffw, '\2', PAGE_SIZE); + pagecache_write(&pagecache, &file1, 0, 3, (char*)buffw, + PAGECACHE_PLAIN_PAGE, + PAGECACHE_LOCK_LEFT_UNLOCKED, + PAGECACHE_PIN_LEFT_UNPINNED, + PAGECACHE_WRITE_DELAY, + 0); + pagecache_delete_page(&pagecache, &file1, 0, + PAGECACHE_LOCK_WRITE, 0); + flush_pagecache_blocks(&pagecache, &file1, FLUSH_FORCE_WRITE); + ok((res= test(test_file(file1, file1_name, PAGE_SIZE, PAGE_SIZE, + simple_delete_forget_test_file))), + "Simple delete-forget page file"); + if (res) + reset_file(file1, file1_name); + free(buffw); + free(buffr); + DBUG_RETURN(res); +} + +/* + Prepare page with locking, write new content to the page, + delete page with flush and on existing lock, + check that page on disk contain new value. +*/ + +int simple_delete_flush_test() +{ + unsigned char *buffw= malloc(PAGE_SIZE); + unsigned char *buffr= malloc(PAGE_SIZE); + int res; + DBUG_ENTER("simple_delete_flush_test"); + /* prepare the file */ + memset(buffw, '\1', PAGE_SIZE); + pagecache_write(&pagecache, &file1, 0, 3, (char*)buffw, + PAGECACHE_PLAIN_PAGE, + PAGECACHE_LOCK_WRITE, + PAGECACHE_PIN, + PAGECACHE_WRITE_DELAY, + 0); + flush_pagecache_blocks(&pagecache, &file1, FLUSH_FORCE_WRITE); + /* test */ + memset(buffw, '\2', PAGE_SIZE); + pagecache_write(&pagecache, &file1, 0, 3, (char*)buffw, + PAGECACHE_PLAIN_PAGE, + PAGECACHE_LOCK_LEFT_WRITELOCKED, + PAGECACHE_PIN_LEFT_PINNED, + PAGECACHE_WRITE_DELAY, + 0); + pagecache_delete_page(&pagecache, &file1, 0, + PAGECACHE_LOCK_LEFT_WRITELOCKED, 1); + flush_pagecache_blocks(&pagecache, &file1, FLUSH_FORCE_WRITE); + ok((res= test(test_file(file1, file1_name, PAGE_SIZE, PAGE_SIZE, + simple_delete_flush_test_file))), + "Simple delete-forget page file"); + if (res) + reset_file(file1, file1_name); + free(buffw); + free(buffr); + DBUG_RETURN(res); +} + + +/* + write then read file bigger then cache +*/ + +int simple_big_test() +{ + unsigned char *buffw= (unsigned char *)malloc(PAGE_SIZE); + unsigned char *buffr= (unsigned char *)malloc(PAGE_SIZE); + struct file_desc *desc= + (struct file_desc *)malloc((PCACHE_SIZE/(PAGE_SIZE/2)) * + sizeof(struct file_desc)); + int res, i; + DBUG_ENTER("simple_big_test"); + /* prepare the file twice larger then cache */ + for (i= 0; i < PCACHE_SIZE/(PAGE_SIZE/2); i++) + { + memset(buffw, (unsigned char) (i & 0xff), PAGE_SIZE); + desc[i].length= PAGE_SIZE; + desc[i].content= (i & 0xff); + pagecache_write(&pagecache, &file1, i, 3, (char*)buffw, + PAGECACHE_PLAIN_PAGE, + PAGECACHE_LOCK_LEFT_UNLOCKED, + PAGECACHE_PIN_LEFT_UNPINNED, + PAGECACHE_WRITE_DELAY, + 0); + } + ok(1, "Simple big file write"); + /* check written pages sequentally read */ + for (i= 0; i < PCACHE_SIZE/(PAGE_SIZE/2); i++) + { + int j; + pagecache_read(&pagecache, &file1, i, 3, (char*)buffr, + PAGECACHE_PLAIN_PAGE, + PAGECACHE_LOCK_LEFT_UNLOCKED, + 0); + for(j= 0; j < PAGE_SIZE; j++) + { + if (buffr[j] != (i & 0xff)) + { + diag("simple_big_test seq: page %u byte %u mismatch\n", i, j); + return 0; + } + } + } + ok(1, "simple big file sequentally read"); + /* chack random reads */ + for (i= 0; i < PCACHE_SIZE/(PAGE_SIZE); i++) + { + int j, page; + page= rand() % (PCACHE_SIZE/(PAGE_SIZE/2)); + pagecache_read(&pagecache, &file1, page, 3, (char*)buffr, + PAGECACHE_PLAIN_PAGE, + PAGECACHE_LOCK_LEFT_UNLOCKED, + 0); + for(j= 0; j < PAGE_SIZE; j++) + { + if (buffr[j] != (page & 0xff)) + { + diag("simple_big_test rnd: page %u byte %u mismatch\n", page, j); + return 0; + } + } + } + ok(1, "simple big file random read"); + flush_pagecache_blocks(&pagecache, &file1, FLUSH_FORCE_WRITE); + + ok((res= test(test_file(file1, file1_name, PCACHE_SIZE*2, PAGE_SIZE, + desc))), + "Simple big file"); + if (res) + reset_file(file1, file1_name); + free(buffw); + free(buffr); + DBUG_RETURN(res); +} +/* + Thread function +*/ + +static void *test_thread(void *arg) +{ + int param=*((int*) arg); + + my_thread_init(); + DBUG_ENTER("test_thread"); + + DBUG_PRINT("enter", ("param: %d", param)); + + if (!simple_read_write_test() || + !simple_read_change_write_read_test() || + !simple_pin_test() || + !simple_delete_forget_test() || + !simple_delete_flush_test() || + !simple_big_test()) + exit(1); + + DBUG_PRINT("info", ("Thread %s ended\n", my_thread_name())); + pthread_mutex_lock(&LOCK_thread_count); + thread_count--; + VOID(pthread_cond_signal(&COND_thread_count)); /* Tell main we are ready */ + pthread_mutex_unlock(&LOCK_thread_count); + free((gptr) arg); + my_thread_end(); + DBUG_RETURN(0); +} + + +int main(int argc, char **argv __attribute__((unused))) +{ + pthread_t tid; + pthread_attr_t thr_attr; + int *param, error, pagen; + + MY_INIT(argv[0]); + +#ifndef DBUG_OFF +#if defined(__WIN__) + default_dbug_option= "d:t:i:O,\\test_pagecache_single.trace"; +#else + default_dbug_option= "d:t:i:o,/tmp/test_pagecache_single.trace"; +#endif + if (argc > 1) + { + DBUG_SET(default_dbug_option); + DBUG_SET_INITIAL(default_dbug_option); + } +#endif + + + DBUG_ENTER("main"); + DBUG_PRINT("info", ("Main thread: %s\n", my_thread_name())); + if ((file1.file= my_open(file1_name, + O_CREAT | O_TRUNC | O_RDWR, MYF(0))) == -1) + { + fprintf(stderr, "Got error during file1 creation from open() (errno: %d)\n", + errno); + exit(1); + } + DBUG_PRINT("info", ("file1: %d", file1.file)); + if (chmod(file1_name, S_IRWXU | S_IRWXG | S_IRWXO) != 0) + { + fprintf(stderr, "Got error during file1 chmod() (errno: %d)\n", + errno); + exit(1); + } + my_pwrite(file1.file, "test file", 9, 0, MYF(0)); + + if ((error= pthread_cond_init(&COND_thread_count, NULL))) + { + fprintf(stderr, "Got error: %d from pthread_cond_init (errno: %d)\n", + error, errno); + exit(1); + } + if ((error= pthread_mutex_init(&LOCK_thread_count, MY_MUTEX_INIT_FAST))) + { + fprintf(stderr, "Got error: %d from pthread_cond_init (errno: %d)\n", + error, errno); + exit(1); + } + + if ((error= pthread_attr_init(&thr_attr))) + { + fprintf(stderr,"Got error: %d from pthread_attr_init (errno: %d)\n", + error,errno); + exit(1); + } + if ((error= pthread_attr_setdetachstate(&thr_attr, PTHREAD_CREATE_DETACHED))) + { + fprintf(stderr, + "Got error: %d from pthread_attr_setdetachstate (errno: %d)\n", + error,errno); + exit(1); + } + +#ifndef pthread_attr_setstacksize /* void return value */ + if ((error= pthread_attr_setstacksize(&thr_attr, 65536L))) + { + fprintf(stderr,"Got error: %d from pthread_attr_setstacksize (errno: %d)\n", + error,errno); + exit(1); + } +#endif +#ifdef HAVE_THR_SETCONCURRENCY + VOID(thr_setconcurrency(2)); +#endif + + my_thread_global_init(); + + + if ((pagen= init_pagecache(&pagecache, PCACHE_SIZE, 0, 0, + PAGE_SIZE, 0)) == 0) + { + fprintf(stderr,"Got error: init_pagecache() (errno: %d)\n", + errno); + exit(1); + } + DBUG_PRINT("info", ("Page cache %d pages", pagen)); + + if ((error=pthread_mutex_lock(&LOCK_thread_count))) + { + fprintf(stderr,"Got error: %d from pthread_mutex_lock (errno: %d)\n", + error,errno); + exit(1); + } + param=(int*) malloc(sizeof(int)); + *param= 1; + if ((error= pthread_create(&tid, &thr_attr, test_thread, (void*) param))) + { + fprintf(stderr,"Got error: %d from pthread_create (errno: %d)\n", + error,errno); + exit(1); + } + thread_count++; + DBUG_PRINT("info", ("Thread started")); + pthread_mutex_unlock(&LOCK_thread_count); + + pthread_attr_destroy(&thr_attr); + + if ((error= pthread_mutex_lock(&LOCK_thread_count))) + fprintf(stderr,"Got error: %d from pthread_mutex_lock\n",error); + while (thread_count) + { + if ((error= pthread_cond_wait(&COND_thread_count,&LOCK_thread_count))) + fprintf(stderr,"Got error: %d from pthread_cond_wait\n",error); + } + if ((error= pthread_mutex_unlock(&LOCK_thread_count))) + fprintf(stderr,"Got error: %d from pthread_mutex_unlock\n",error); + DBUG_PRINT("info", ("thread ended")); + + end_pagecache(&pagecache, 1); + DBUG_PRINT("info", ("Page cache ended")); + + if (my_close(file1.file, MYF(0)) != 0) + { + fprintf(stderr, "Got error during file1 closing from close() (errno: %d)\n", + errno); + exit(1); + } + /*my_delete(file1_name, MYF(0));*/ + my_thread_global_end(); + + DBUG_PRINT("info", ("file1 (%d) closed", file1.file)); + + DBUG_PRINT("info", ("Program end")); + + DBUG_RETURN(0); +} -- cgit v1.2.1 From 30e5a9bd1b853c60e067fe71d61c5f55bb448d85 Mon Sep 17 00:00:00 2001 From: unknown Date: Mon, 24 Jul 2006 21:50:18 +0200 Subject: Post-vacation-musing fixes to WL#3071 "Maria checkpoint": changes to how synchronous checkpoint requests are executed. changes to how the background LRD flushing thread refrains from using all resources. See more comments for each file. storage/maria/checkpoint.c: I was not happy that checkpoint requests which want to know the success/error of their executed request, get inaccurate information in case of error (no error string etc). Instead of implementing a more complete communication protocol between requestor and executor, I make the requestor do the execution itself. I call this a synchronous checkpoint. For asynchronous checkpoints (requestor does not want to know success/error, does not want to wait for completion), no change, checkpoint is executed by the background thread. Comments, constants, mutex usage fixes. storage/maria/checkpoint.h: new prototype of "API" (the calls exposed by the checkpoint module) storage/maria/least_recently_dirtied.c: A better solution than sleeping one second after flushing a piece of the LRD: instead we pthread_yield(). Hopefully this will slow down the background thread (avoiding it using all the disk's bandwidth) if there are other threads competing, and will not slow it down if this thread is alone (where we do want it to run fast and not do useless sleeps). This thread will probe for asynchronous checkpoint requests every few seconds. --- storage/maria/checkpoint.c | 299 +++++++++++++++------------------ storage/maria/checkpoint.h | 10 +- storage/maria/least_recently_dirtied.c | 39 +++-- 3 files changed, 169 insertions(+), 179 deletions(-) diff --git a/storage/maria/checkpoint.c b/storage/maria/checkpoint.c index af37377455e..ed18e95c360 100644 --- a/storage/maria/checkpoint.c +++ b/storage/maria/checkpoint.c @@ -6,12 +6,28 @@ /* Here is the implementation of this module */ +/* + Summary: + - there are asynchronous checkpoints (a writer to the log notices that it's + been a long time since we last checkpoint-ed, so posts a request for a + background thread to do a checkpoint; does not care about the success of the + checkpoint). Then the checkpoint is done by the checkpoint thread, at an + unspecified moment ("later") (==soon, of course). + - there are synchronous checkpoints: a thread requests a checkpoint to + happen now and wants to know when it finishes and if it succeeded; then the + checkpoint is done by that same thread. +*/ + #include "page_cache.h" #include "least_recently_dirtied.h" #include "transaction.h" #include "share.h" #include "log.h" +/* could also be called LSN_ERROR */ +#define LSN_IMPOSSIBLE ((LSN)0) +#define LSN_MAX ((LSN)ULONGLONG_MAX) + /* this transaction is used for any system work (purge, checkpoint writing etc), that is, background threads. It will not be declared/initialized here @@ -19,43 +35,112 @@ */ st_transaction system_trans= {0 /* long trans id */, 0 /* short trans id */,0,...}; +/* those three are protected by the log's mutex */ /* The maximum rec_lsn in the LRD when last checkpoint was run, serves for the MEDIUM checkpoint. */ LSN max_rec_lsn_at_last_checkpoint= 0; +CHECKPOINT_LEVEL next_asynchronous_checkpoint_to_do= NONE; +CHECKPOINT_LEVEL synchronous_checkpoint_in_progress= NONE; + +/* + Used by MySQL client threads requesting a checkpoint (like "ALTER MARIA + ENGINE DO CHECKPOINT"), and probably by maria_panic(). +*/ +my_bool execute_synchronous_checkpoint(CHECKPOINT_LEVEL level) +{ + DBUG_ENTER("execute_synchronous_checkpoint"); + DBUG_ASSERT(level > NONE); + + lock(log_mutex); + while ((synchronous_checkpoint_in_progress != NONE) || + (next_asynchronous_checkpoint_to_do != NONE)) + wait_on_checkpoint_done_cond(); + + synchronous_checkpoint_in_progress= level; + execute_checkpoint(level); + safemutex_assert_owner(log_mutex); + synchronous_checkpoint_in_progress= NONE; + unlock(log_mutex); + broadcast(checkpoint_done_cond); +} -/* Picks a checkpoint request and executes it */ -my_bool checkpoint() +/* Picks a checkpoint request, if there is one, and executes it */ +my_bool execute_asynchronous_checkpoint_if_any() { CHECKPOINT_LEVEL level; - DBUG_ENTER("checkpoint"); + DBUG_ENTER("execute_asynchronous_checkpoint"); + + lock(log_mutex); + if (likely(next_asynchronous_checkpoint_to_do == NONE)) + { + unlock(log_mutex); + DBUG_RETURN(FALSE); + } - level= checkpoint_running= checkpoint_request; + while (synchronous_checkpoint_in_progress) + wait_on_checkpoint_done_cond(); + +do_checkpoint: + level= next_asynchronous_checkpoint_to_do; + DBUG_ASSERT(level > NONE); + execute_checkpoint(level); + safemutex_assert_owner(log_mutex); + if (next_asynchronous_checkpoint_to_do > level) + goto do_checkpoint; /* one more request was posted */ + else + { + DBUG_ASSERT(next_asynchronous_checkpoint_to_do == level); + next_asynchronous_checkpoint_to_do= NONE; /* all work done */ + } unlock(log_mutex); + broadcast(checkpoint_done_cond); +} - DBUG_ASSERT(level != NONE); - switch (level) +/* + Does the actual checkpointing. Called by + execute_synchronous_checkpoint() and + execute_asynchronous_checkpoint_if_any(). +*/ +my_bool execute_checkpoint(CHECKPOINT_LEVEL level) +{ + LSN candidate_max_rec_lsn_at_last_checkpoint; + /* to avoid { lock + no-op + unlock } in the common (==indirect) case */ + my_bool need_log_mutex; + + DBUG_ENTER("execute_checkpoint"); + + safemutex_assert_owner(log_mutex); + copy_of_max_rec_lsn_at_last_checkpoint= max_rec_lsn_at_last_checkpoint; + + if (unlikely(need_log_mutex= (level > INDIRECT))) { - case FULL: - /* flush all pages up to the current end of the LRD */ - flush_all_LRD_to_lsn(MAX_LSN); /* MAX_LSN==ULONGLONG_MAX */ - /* this will go full speed (normal scheduling, no sleep) */ - break; - case MEDIUM: - /* - flush all pages which were already dirty at last checkpoint: - ensures that recovery will never start from before the next-to-last - checkpoint (two-checkpoint rule). - It is max, not min as the WL says (TODO update WL). - */ - flush_all_LRD_to_lsn(max_rec_lsn_at_last_checkpoint); - /* this will go full speed (normal scheduling, no sleep) */ - break; + /* much I/O work to do, release log mutex */ + unlock(log_mutex); + + switch (level) + { + case FULL: + /* flush all pages up to the current end of the LRD */ + flush_all_LRD_to_lsn(LSN_MAX); + /* this will go full speed (normal scheduling, no sleep) */ + break; + case MEDIUM: + /* + flush all pages which were already dirty at last checkpoint: + ensures that recovery will never start from before the next-to-last + checkpoint (two-checkpoint rule). + It is max, not min as the WL says (TODO update WL). + */ + flush_all_LRD_to_lsn(copy_of_max_rec_lsn_at_last_checkpoint); + /* this will go full speed (normal scheduling, no sleep) */ + break; + } } - - error= checkpoint_indirect(); + + candidate_max_rec_lsn_at_last_checkpoint= checkpoint_indirect(need_log_mutex); lock(log_mutex); /* @@ -66,13 +151,22 @@ my_bool checkpoint() file in the hook but that would be an I/O under the log's mutex, bad. - it would not be nice organisation of code (I tried it :). */ - mark_checkpoint_done(error); - unlock(log_mutex); - DBUG_RETURN(error); + if (candidate_max_rec_lsn_at_last_checkpoint != LSN_IMPOSSIBLE) + { + /* checkpoint succeeded */ + maximum_rec_lsn_last_checkpoint= candidate_max_rec_lsn_at_last_checkpoint; + written_since_last_checkpoint= (my_off_t)0; + DBUG_RETURN(FALSE); + } + /* + keep mutex locked because callers will want to clear mutex-protected + status variables + */ + DBUG_RETURN(TRUE); } -my_bool checkpoint_indirect() +LSN checkpoint_indirect(my_bool need_log_mutex) { DBUG_ENTER("checkpoint_indirect"); @@ -90,7 +184,8 @@ my_bool checkpoint_indirect() DBUG_ASSERT(sizeof(byte *) <= 8); DBUG_ASSERT(sizeof(LSN) <= 8); - lock(log_mutex); /* will probably be in log_read_end_lsn() already */ + if (need_log_mutex) + lock(log_mutex); /* maybe this will clash with log_read_end_lsn() */ checkpoint_start_lsn= log_read_end_lsn(); unlock(log_mutex); @@ -196,15 +291,13 @@ my_bool checkpoint_indirect() checkpoint_lsn= log_write_record(LOGREC_CHECKPOINT, &system_trans, string_array); - if (0 == checkpoint_lsn) /* maybe 0 is impossible LSN to indicate error ? */ + if (LSN_IMPOSSIBLE == checkpoint_lsn) goto err; if (0 != control_file_write_and_force(checkpoint_lsn, NULL)) goto err; - maximum_rec_lsn_last_checkpoint= candidate_max_rec_lsn_at_last_checkpoint; - - DBUG_RETURN(0); + DBUG_RETURN(candidate_max_rec_lsn_at_last_checkpoint); err: @@ -213,7 +306,7 @@ err: my_free(buffer2.str, MYF(MY_ALLOW_ZERO_PTR)); my_free(buffer3.str, MYF(MY_ALLOW_ZERO_PTR)); - DBUG_RETURN(1); + DBUG_RETURN(LSN_IMPOSSIBLE); } @@ -235,7 +328,7 @@ log_write_record(...) ask one system thread (the "LRD background flusher and checkpointer thread" WL#3261) to do a checkpoint */ - request_checkpoint(INDIRECT, 0 /*wait_for_completion*/); + request_asynchronous_checkpoint(INDIRECT); } ...; unlock(log_mutex); @@ -243,152 +336,38 @@ log_write_record(...) } /* - Call this when you want to request a checkpoint. - In real life it will be called by log_write_record() and by client thread + Requests a checkpoint from the background thread, *asynchronously* + (requestor does not wait for completion, and does not even later check the + result). + In real life it will be called by log_write_record(). which explicitely wants to do checkpoint (ALTER ENGINE CHECKPOINT checkpoint_level). */ -int request_checkpoint(CHECKPOINT_LEVEL level, my_bool wait_for_completion) +void request_asynchronous_checkpoint(CHECKPOINT_LEVEL level); { - int error= 0; - /* - If caller wants to wait for completion we'll have to release the log mutex - to wait on condition, if caller had log mutex he may not be happy that we - release it, so we check that caller didn't have log mutex. - */ - if (wait_for_completion) - { - lock(log_mutex); - } - else - safemutex_assert_owner(log_mutex); + safemutex_assert_owner(log_mutex); - DBUG_ASSERT(checkpoint_request >= checkpoint_running); DBUG_ASSERT(level > NONE); if (checkpoint_request < level) { /* no equal or stronger running or to run, we post request */ /* note that thousands of requests for checkpoints are going to come all - at the same time (when the log bound is passed), so it may not be a good - idea for each of them to broadcast a cond. We just don't broacast a - cond, the checkpoint thread will wake up in max one second. + at the same time (when the log bound + MAX_LOG_BYTES_WRITTEN_BETWEEN_CHECKPOINTS is passed), so it may not be a + good idea for each of them to broadcast a cond to wake up the background + checkpoint thread. We just don't broacast a cond, the checkpoint thread + will notice our request in max a few seconds. */ checkpoint_request= level; /* post request */ } - - if (wait_for_completion) - { - uint checkpoints_done_copy= checkpoints_done; - uint checkpoint_errors_copy= checkpoint_errors; - /* - note that the "==done" works when the uint counter wraps too, so counter - can even be smaller than uint if we wanted (however it should be big - enough so that max_the_int_type checkpoints cannot happen between two - wakeups of our thread below). uint sounds fine. - Wait for our checkpoint to be done: - */ - - if (checkpoint_running != NONE) /* not ours, let it pass */ - { - while (1) - { - if (checkpoints_done != checkpoints_done_copy) - { - if (checkpoints_done == (checkpoints_done_copy+1)) - { - /* not our checkpoint, forget about it */ - checkpoints_done_copy= checkpoints_done; - } - break; /* maybe even ours has been done at this stage! */ - } - cond_wait(checkpoint_done_cond, log_mutex); - } - } - - /* now we come to waiting for our checkpoint */ - while (1) - { - if (checkpoints_done != checkpoints_done_copy) - { - /* our checkpoint has been done */ - break; - } - if (checkpoint_errors != checkpoint_errors_copy) - { - /* - the one which was running a few milliseconds ago (if there was one), - and/or ours, had an error, just assume it was ours. So there - is a possibility that we return error though we succeeded, in which - case user will have to retry; but two simultanate checkpoints have - high changes to fail together (as the error probably comes from - malloc or disk write problem), so chance of false alarm is low. - Reporting the error only to the one which caused the error would - require having a (not fixed size) list of all requests, not worth it. - */ - error= 1; - break; - } - cond_wait(checkpoint_done_cond, log_mutex); - } - unlock(log_mutex); - } /* ... if (wait_for_completion) */ /* - If wait_for_completion was false, and there was an error, only an error + If there was an error, only an error message to the error log will say it; normal, for a checkpoint triggered by a log write, we probably don't want the client's log write to throw an error, as the log write succeeded and a checkpoint failure is not critical: the failure in this case is more for the DBA to know than for the end user. */ - return error; -} - -void mark_checkpoint_done(int error) -{ - safemutex_assert_owner(log_mutex); - if (error) - checkpoint_errors++; - /* a checkpoint is said done even if it had an error */ - checkpoints_done++; - if (checkpoint_request == checkpoint_running) - { - /* - No new request has been posted, so we satisfied all requests, forget - about them. - */ - checkpoint_request= NONE; - } - checkpoint_running= NONE; - written_since_last_checkpoint= 0; - broadcast(checkpoint_done_cond); -} - -/* - Alternative (not to be done, too disturbing): - do the autocheckpoint in the thread which passed the bound first (and do the - checkpoint in the client thread which requested it). - It will give a delay to that client thread which passed the bound (time to - fsync() for example 1000 files is 16 s on my laptop). Here is code for - explicit and implicit checkpoints, where client thread does the job: -*/ -#if 0 -{ - lock(log_mutex); /* explicit takes it here, implicit already has it */ - while (checkpoint_running != NONE) - { - if (checkpoint_running >= my_level) /* always true for auto checkpoints */ - goto end; /* we skip checkpoint */ - /* a less strong is running, I'll go next */ - wait_on_checkpoint_done_cond(); - } - checkpoint_running= my_level; - checkpoint(my_level); // can gather checkpoint_start_lsn before unlock - lock(log_mutex); - checkpoint_running= NONE; - written_since_last_checkpoint= 0; -end: - unlock(log_mutex); } -#endif diff --git a/storage/maria/checkpoint.h b/storage/maria/checkpoint.h index ce8066de93d..a9de18c695f 100644 --- a/storage/maria/checkpoint.h +++ b/storage/maria/checkpoint.h @@ -13,11 +13,7 @@ typedef enum enum_checkpoint_level { FULL /* also flush all dirty pages */ } CHECKPOINT_LEVEL; -/* - Call this when you want to request a checkpoint. - In real life it will be called by log_write_record() and by client thread - which explicitely wants to do checkpoint (ALTER ENGINE CHECKPOINT - checkpoint_level). -*/ -int request_checkpoint(CHECKPOINT_LEVEL level, my_bool wait_for_completion); +void request_asynchronous_checkpoint(CHECKPOINT_LEVEL level); +my_bool execute_synchronous_checkpoint(CHECKPOINT_LEVEL level); +my_bool execute_asynchronous_checkpoint_if_any(); /* that's all that's needed in the interface */ diff --git a/storage/maria/least_recently_dirtied.c b/storage/maria/least_recently_dirtied.c index e251e3c582c..a99f0c372d5 100644 --- a/storage/maria/least_recently_dirtied.c +++ b/storage/maria/least_recently_dirtied.c @@ -48,6 +48,15 @@ Key cache has groupping already somehow Monty said (investigate that). */ #define FLUSH_GROUP_SIZE 512 /* 8 MB */ +/* + We don't want to probe for checkpoint requests all the time (it takes + the log mutex). + If FLUSH_GROUP_SIZE is 8MB, assuming a local disk which can write 30MB/s + (1.8GB/min), probing every 16th call to flush_one_group_from_LRD() is every + 16*8=128MB which is every 128/30=4.2second. + Using a power of 2 gives a fast modulo operation. +*/ +#define CHECKPOINT_PROBING_PERIOD_LOG2 4 /* This thread does background flush of pieces of the LRD, and all checkpoints. @@ -56,19 +65,19 @@ pthread_handler_decl background_flush_and_checkpoint_thread() { char *flush_group_buffer= my_malloc(PAGE_SIZE*FLUSH_GROUP_SIZE); + uint flush_calls= 0; while (this_thread_not_killed) { - lock(log_mutex); - if (checkpoint_request) - checkpoint(); /* will unlock mutex */ - else - { - unlock(log_mutex); - lock(global_LRD_mutex); - flush_one_group_from_LRD(); - safemutex_assert_not_owner(global_LRD_mutex); - } - my_sleep(1000000); /* one second ? */ + if ((flush_calls++) & ((2<=max_lsn */ @@ -165,7 +180,7 @@ int flush_all_LRD_to_lsn(LSN max_lsn) max_lsn= LRD->first->prev->rec_lsn; while (LRD->first->rec_lsn < max_lsn) { - if (flush_one_group_from_LRD()) /* will unlock mutex */ + if (flush_one_group_from_LRD()) /* will unlock LRD mutex */ return 1; /* scheduler may preempt us here so that we don't take full CPU */ lock(global_LRD_mutex); -- cgit v1.2.1 From 832776fd2d4d11825eaeb945e57d6eeab816cefc Mon Sep 17 00:00:00 2001 From: unknown Date: Wed, 26 Jul 2006 09:47:44 +0200 Subject: two include files were missing in the source tarball, causing pushbuild build failures. include/Makefile.am: two include files missing in the source tarball --- include/Makefile.am | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/include/Makefile.am b/include/Makefile.am index e1ddadb933a..457e49c338b 100644 --- a/include/Makefile.am +++ b/include/Makefile.am @@ -24,7 +24,7 @@ pkginclude_HEADERS = my_dbug.h m_string.h my_sys.h my_list.h my_xml.h \ sslopt-vars.h sslopt-case.h sql_common.h keycache.h \ mysql_time.h mysql/plugin.h $(BUILT_SOURCES) noinst_HEADERS = config-win.h config-netware.h \ - heap.h my_bitmap.h my_uctype.h \ + heap.h maria.h myisamchk.h my_bitmap.h my_uctype.h \ myisam.h myisampack.h myisammrg.h ft_global.h\ mysys_err.h my_base.h help_start.h help_end.h \ my_nosys.h my_alarm.h queues.h rijndael.h sha1.h \ -- cgit v1.2.1 From e6efa27c0b720702f3841a4d28e795c49432bfc5 Mon Sep 17 00:00:00 2001 From: unknown Date: Wed, 26 Jul 2006 10:36:34 +0200 Subject: Writing down MikaelR's comments made in May in Helsinki so that they are not forgotten. Minor fixes. storage/maria/checkpoint.c: minor fixes storage/maria/least_recently_dirtied.c: writing down MikaelR's comments. storage/maria/recovery.c: writing down MikaelR's comments. Some small fixes. --- storage/maria/checkpoint.c | 10 +++-- storage/maria/least_recently_dirtied.c | 9 +++++ storage/maria/recovery.c | 67 ++++++++++++++++++++++------------ 3 files changed, 59 insertions(+), 27 deletions(-) diff --git a/storage/maria/checkpoint.c b/storage/maria/checkpoint.c index ed18e95c360..151efe6ad4f 100644 --- a/storage/maria/checkpoint.c +++ b/storage/maria/checkpoint.c @@ -246,7 +246,7 @@ LSN checkpoint_indirect(my_bool need_log_mutex) if no latch, use double variable of type ULONGLONG_CONSISTENT in st_transaction, or even no need if Intel >=486 */ - int8store(ptr, el->first_purge_lsn); + int8store(ptr, el->first_undo_lsn); ptr+= 8; /* possibly unlatch el.rwlock */ } @@ -297,16 +297,18 @@ LSN checkpoint_indirect(my_bool need_log_mutex) if (0 != control_file_write_and_force(checkpoint_lsn, NULL)) goto err; - DBUG_RETURN(candidate_max_rec_lsn_at_last_checkpoint); + goto end; err: - print_error_to_error_log(the_error_message); + candidate_max_rec_lsn_at_last_checkpoint= LSN_IMPOSSIBLE; + +end: my_free(buffer1.str, MYF(MY_ALLOW_ZERO_PTR)); my_free(buffer2.str, MYF(MY_ALLOW_ZERO_PTR)); my_free(buffer3.str, MYF(MY_ALLOW_ZERO_PTR)); - DBUG_RETURN(LSN_IMPOSSIBLE); + DBUG_RETURN(candidate_max_rec_lsn_at_last_checkpoint); } diff --git a/storage/maria/least_recently_dirtied.c b/storage/maria/least_recently_dirtied.c index a99f0c372d5..bca13ca6f1f 100644 --- a/storage/maria/least_recently_dirtied.c +++ b/storage/maria/least_recently_dirtied.c @@ -19,6 +19,13 @@ #include "page_cache.h" #include "least_recently_dirtied.h" +/* + MikaelR suggested removing this global_LRD_mutex (I have a paper note of + comments), however at least for the first version we'll start with this + mutex (which will be a LOCK-based atomic_rwlock). +*/ +pthread_mutex_t global_LRD_mutex; + /* When we flush a page, we should pin page. This "pin" is to protect against that: @@ -61,6 +68,8 @@ /* This thread does background flush of pieces of the LRD, and all checkpoints. Just launch it when engine starts. + MikaelR questioned why the same thread does two different jobs, the risk + could be that while a checkpoint happens no LRD flushing happens. */ pthread_handler_decl background_flush_and_checkpoint_thread() { diff --git a/storage/maria/recovery.c b/storage/maria/recovery.c index 5af7019e5fb..589e0971686 100644 --- a/storage/maria/recovery.c +++ b/storage/maria/recovery.c @@ -16,31 +16,42 @@ typedef struct st_record_type_properties { /* used for debug error messages or "maria_read_log" command-line tool: */ char *name, my_bool record_ends_group; - int (*record_execute)(RECORD *); /* param will be record header instead later */ + /* a function to execute when we see the record during the REDO phase */ + int (*record_execute_in_redo_phase)(RECORD *); /* param will be record header instead later */ + /* a function to execute when we see the record during the UNDO phase */ + int (*record_execute_in_undo_phase)(RECORD *); /* param will be record header instead later */ } RECORD_TYPE_PROPERTIES; +int no_op(RECORD *) {return 0}; + RECORD_TYPE_PROPERTIES all_record_type_properties[]= { /* listed here in the order of the "log records type" enumeration */ - {"REDO_INSERT_HEAD", 0, redo_insert_head_execute}, + {"REDO_INSERT_HEAD", FALSE, redo_insert_head_execute_in_redo_phase, no_op}, ..., - {"UNDO_INSERT" , 1, undo_insert_execute }, - {"COMMIT", , 1, commit_execute }, + {"UNDO_INSERT" , TRUE , undo_insert_execute_in_redo_phase, undo_insert_execute_in_undo_phase}, + {"COMMIT", , TRUE , commit_execute_in_redo_phase, no_op}, ... }; -int redo_insert_head_execute(RECORD *record) +int redo_insert_head_execute_in_redo_phase(RECORD *record) { /* write the data to the proper page */ } -int undo_insert_execute(RECORD *record) +int undo_insert_execute_in_redo_phase(RECORD *record) { trans_table[short_trans_id].undo_lsn= record.lsn; + /* don't restore the old version of the row */ +} + +int undo_insert_execute_in_undo_phase(RECORD *record) +{ /* restore the old version of the row */ + trans_table[short_trans_id].undo_lsn= record.prev_undo_lsn; } -int commit_execute(RECORD *record) +int commit_execute_in_redo_phase(RECORD *record) { trans_table[short_trans_id].state= COMMITTED; /* @@ -52,8 +63,8 @@ int commit_execute(RECORD *record) #define record_ends_group(R) \ all_record_type_properties[(R)->type].record_ends_group) -#define execute_log_record(R) \ - all_record_type_properties[(R).type].record_execute(R) +#define execute_log_record_in_redo_phase(R) \ + all_record_type_properties[(R).type].record_execute_in_redo_phase(R) int recovery() @@ -77,7 +88,10 @@ int recovery() phase): */ - record= log_read_record(min(rec_lsn, ...)); + /**** REDO PHASE *****/ + + record= log_read_record(min(rec_lsn, ...)); /* later, read only header */ + /* if log handler knows the end LSN of the log, we could print here how many MB of log we have to read (to give an idea of the time), and print @@ -94,15 +108,11 @@ int recovery() */ if (record_ends_group(record) { - /* - such end events can always be executed immediately (they don't touch - the disk). - */ - execute_log_record(record); if (trans_table[record.short_trans_id].group_start_lsn != 0) { /* - There is a complete group for this transaction. + There is a complete group for this transaction, containing more than + this event. We're going to read recently read log records: for this log_read_record() to be efficient (not touch the disk), log handler could cache recently read pages @@ -110,17 +120,19 @@ int recovery() log handler page cache). Without it only OS file cache will help. */ - record2= log_read_record(trans_table[record.short_trans_id].group_start_lsn); - while (record2.lsn < record.lsn) + record2= + log_read_record(trans_table[record.short_trans_id].group_start_lsn); + + do { if (record2.short_trans_id == record.short_trans_id) - execute_log_record(record2); /* it's in our group */ + execute_log_record_in_redo_phase(record2); /* it's in our group */ record2= log_read_next_record(); } + while (record2.lsn < record.lsn); trans_table[record.short_trans_id].group_start_lsn= 0; /* group finished */ - /* we're now at the UNDO, re-read it to advance log pointer */ - record2= log_read_next_record(); /* and throw it away */ } + execute_log_record_in_redo_phase(record); } else /* record does not end group */ { @@ -161,7 +173,14 @@ int recovery() the log, and so the delete/update handler may do changes which conflict with these REDOs. Even if done here, better to not wake it up now as we're going to free the - page cache: + page cache. + + MikaelR suggests: support checkpoints during REDO phase too: do checkpoint + after a certain amount of log records have been executed. This helps + against repeated crashes. Those checkpoints could not be user-requested + (as engine is not communicating during the REDO phase), so they would be + automatic: this changes the original assumption that we don't write to the + log while in the REDO phase, but why not. How often should we checkpoint? */ /* @@ -178,6 +197,8 @@ int recovery() do this: */ + /**** UNDO PHASE *****/ + print_information_to_error_log(nb of trans to roll back, nb of prepared trans); /* @@ -217,7 +238,7 @@ pthread_handler_decl rollback_background_thread() { /* this is the normal runtime-rollback code: */ record= log_read_record(trans->undo_lsn); - execute_log_record(record); + execute_log_record_in_undo_phase(record); trans->undo_lsn= record.prev_undo_lsn; } /* remove trans from list */ -- cgit v1.2.1 From 99c431db92f8904bf50f6944e1488a0172c4ebd8 Mon Sep 17 00:00:00 2001 From: unknown Date: Thu, 10 Aug 2006 16:36:54 +0200 Subject: Completion of merge of mysql-5.1 into mysql-maria. Manually imported changes done to MyISAM (include/myisam.h, storage/myisam/*, sql/ha_myisam.*, mysql-test/t/myisam.test, mysql-test/t/ps_2myisam.test) the last months into Maria (tedious, should do it more frequently in the future), including those not done at the previous 5.1->Maria merge (please in the future don't forget to apply MyISAM changes to Maria when you merge 5.1 into Maria). Note: I didn't try to import anything which could be MyISAM-related in other tests of mysql-test (I didn't want to dig in all csets), but as QA is working to make most tests re-usable for other engines (Falcon), it is likely that we'll benefit from this and just have to set engine=Maria somewhere to run those tests on Maria. func_group and partition tests fail but they already do in main 5.1 on my machine. No Valgrind error in t/*maria*.test. Monty: please see the commit comment of maria.result and check. BitKeeper/deleted/.del-ha_maria.m4: Delete: config/ac-macros/ha_maria.m4 configure.in: fix for the new way of enabling engines include/maria.h: importing changes done to MyISAM the last months into Maria include/my_handler.h: importing changes done to MyISAM the last months into Maria include/myisam.h: importing changes done to MyISAM the last months into Maria mysql-test/r/maria.result: identical to myisam.result, except the engine name in some places AND in the line testing key_block_size=1000000000000000000: Maria gives a key block size of 8192 while MyISAM gives 4096; is it explainable by the difference between MARIA_KEY_BLOCK_LENGTH and the same constant in MyISAM? Monty? mysql-test/r/ps_maria.result: identical to ps_2myisam.result (except the engine name in some places) mysql-test/t/maria.test: instead of engine=maria everywhere, I use @@storage_engine (reduces the diff with myisam.test). importing changes done to MyISAM the last months into Maria mysys/my_handler.c: importing changes done to MyISAM the last months into Maria sql/ha_maria.cc: importing changes done to MyISAM the last months into Maria sql/ha_maria.h: importing changes done to MyISAM the last months into Maria sql/mysqld.cc: unneeded storage/maria/Makefile.am: importing changes done to MyISAM the last months into Maria storage/maria/ma_check.c: importing changes done to MyISAM the last months into Maria storage/maria/ma_create.c: importing changes done to MyISAM the last months into Maria storage/maria/ma_delete_table.c: importing changes done to MyISAM the last months into Maria storage/maria/ma_dynrec.c: importing changes done to MyISAM the last months into Maria storage/maria/ma_extra.c: importing changes done to MyISAM the last months into Maria storage/maria/ma_ft_boolean_search.c: importing changes done to MyISAM the last months into Maria storage/maria/ma_ft_eval.c: importing changes done to MyISAM the last months into Maria storage/maria/ma_ft_nlq_search.c: importing changes done to MyISAM the last months into Maria storage/maria/ma_ft_parser.c: importing changes done to MyISAM the last months into Maria storage/maria/ma_ft_test1.c: importing changes done to MyISAM the last months into Maria storage/maria/ma_ft_update.c: importing changes done to MyISAM the last months into Maria storage/maria/ma_ftdefs.h: importing changes done to MyISAM the last months into Maria storage/maria/ma_key.c: importing changes done to MyISAM the last months into Maria storage/maria/ma_open.c: importing changes done to MyISAM the last months into Maria storage/maria/ma_page.c: importing changes done to MyISAM the last months into Maria storage/maria/ma_rkey.c: importing changes done to MyISAM the last months into Maria storage/maria/ma_rsamepos.c: importing changes done to MyISAM the last months into Maria storage/maria/ma_rt_index.c: importing changes done to MyISAM the last months into Maria storage/maria/ma_rt_mbr.c: importing changes done to MyISAM the last months into Maria storage/maria/ma_search.c: importing changes done to MyISAM the last months into Maria storage/maria/ma_sort.c: importing changes done to MyISAM the last months into Maria storage/maria/ma_test1.c: importing changes done to MyISAM the last months into Maria storage/maria/ma_test2.c: importing changes done to MyISAM the last months into Maria storage/maria/ma_test3.c: importing changes done to MyISAM the last months into Maria storage/maria/ma_update.c: importing changes done to MyISAM the last months into Maria storage/maria/ma_write.c: importing changes done to MyISAM the last months into Maria storage/maria/maria_chk.c: importing changes done to MyISAM the last months into Maria storage/maria/maria_def.h: importing changes done to MyISAM the last months into Maria storage/maria/maria_ftdump.c: importing changes done to MyISAM the last months into Maria storage/maria/maria_pack.c: importing changes done to MyISAM the last months into Maria --- config/ac-macros/ha_maria.m4 | 29 ---- configure.in | 9 +- include/maria.h | 6 +- include/my_handler.h | 2 - include/myisam.h | 1 + mysql-test/r/maria.result | 294 +++++++++++++++++++++++++++-------- mysql-test/r/ps_maria.result | 4 +- mysql-test/t/maria.test | 242 ++++++++++++++++++++-------- mysys/my_handler.c | 1 - sql/ha_maria.cc | 161 +++++++++---------- sql/ha_maria.h | 8 +- sql/mysqld.cc | 1 - storage/maria/Makefile.am | 2 +- storage/maria/ma_check.c | 105 ++++++++----- storage/maria/ma_create.c | 82 ++++++++-- storage/maria/ma_delete_table.c | 24 ++- storage/maria/ma_dynrec.c | 8 + storage/maria/ma_extra.c | 57 ++++--- storage/maria/ma_ft_boolean_search.c | 62 ++++---- storage/maria/ma_ft_eval.c | 1 + storage/maria/ma_ft_nlq_search.c | 10 +- storage/maria/ma_ft_parser.c | 101 +++++++----- storage/maria/ma_ft_test1.c | 1 + storage/maria/ma_ft_update.c | 57 +++---- storage/maria/ma_ftdefs.h | 22 +-- storage/maria/ma_key.c | 16 +- storage/maria/ma_open.c | 12 +- storage/maria/ma_page.c | 9 +- storage/maria/ma_rkey.c | 19 ++- storage/maria/ma_rsamepos.c | 3 +- storage/maria/ma_rt_index.c | 8 +- storage/maria/ma_rt_mbr.c | 6 +- storage/maria/ma_search.c | 47 +++--- storage/maria/ma_sort.c | 1 + storage/maria/ma_test1.c | 1 + storage/maria/ma_test2.c | 16 +- storage/maria/ma_test3.c | 2 + storage/maria/ma_update.c | 3 +- storage/maria/ma_write.c | 3 +- storage/maria/maria_chk.c | 10 +- storage/maria/maria_def.h | 7 +- storage/maria/maria_ftdump.c | 17 +- storage/maria/maria_pack.c | 6 +- 43 files changed, 950 insertions(+), 526 deletions(-) delete mode 100644 config/ac-macros/ha_maria.m4 diff --git a/config/ac-macros/ha_maria.m4 b/config/ac-macros/ha_maria.m4 deleted file mode 100644 index 56ec8720a09..00000000000 --- a/config/ac-macros/ha_maria.m4 +++ /dev/null @@ -1,29 +0,0 @@ -dnl --------------------------------------------------------------------------- -dnl Macro: MYSQL_CHECK_MARIA -dnl Sets HAVE_MARIA_DB if --with-maria-storage-engine is used -dnl --------------------------------------------------------------------------- -AC_DEFUN([MYSQL_CHECK_MARIA], [ - AC_ARG_WITH([maria-storage-engine], - [ - --with-maria-storage-engine - Enable the Maria Storage Engine], - [mariadb="$withval"], - [mariadb=no]) - AC_MSG_CHECKING([for Maria storage engine]) - - case "$mariadb" in - yes ) - AC_DEFINE([HAVE_MARIA_DB], [1], [Builds Maria Storage Engine]) - AC_MSG_RESULT([yes]) - [mariadb=yes] - ;; - * ) - AC_MSG_RESULT([no]) - [mariadb=no] - ;; - esac - -]) -dnl --------------------------------------------------------------------------- -dnl END OF MYSQL_CHECK_MARIA SECTION -dnl --------------------------------------------------------------------------- diff --git a/configure.in b/configure.in index 989c3d21906..c5e6575ac5f 100644 --- a/configure.in +++ b/configure.in @@ -2200,15 +2200,16 @@ MYSQL_PLUGIN_ACTIONS(ndbcluster,[MYSQL_SETUP_NDBCLUSTER]) MYSQL_STORAGE_ENGINE(partition, partition, [Partition Support], [MySQL Partitioning Support], [max,max-no-ndb]) + +dnl -- ndbcluster requires partition to be enabled +MYSQL_PLUGIN_DEPENDS(ndbcluster, [partition]) + MYSQL_STORAGE_ENGINE(maria,maria, [Maria Storage Engine], - [Traditional transactional MySQL tables]) + [Traditional transactional MySQL tables], [max,max-no-ndb]) MYSQL_PLUGIN_DIRECTORY(maria, [storage/maria]) MYSQL_PLUGIN_STATIC(maria, [libmaria.a]) MYSQL_PLUGIN_MANDATORY(maria) -dnl -- ndbcluster requires partition to be enabled -MYSQL_PLUGIN_DEPENDS(ndbcluster, [partition]) - MYSQL_CONFIGURE_PLUGINS([none]) # Only build client code? diff --git a/include/maria.h b/include/maria.h index 72fb1cf2dff..705ba7c91c7 100644 --- a/include/maria.h +++ b/include/maria.h @@ -168,7 +168,7 @@ typedef struct st_maria_keydef /* Key definition with open & info */ uint16 keylength; /* Tot length of keyparts (auto) */ uint16 minlength; /* min length of (packed) key (auto) */ uint16 maxlength; /* max length of (packed) key (auto) */ - uint16 block_size; /* block_size (auto) */ + uint16 block_size_index; /* block_size (auto) */ uint32 version; /* For concurrent read/write */ uint32 ftparser_nr; /* distinct ftparser number */ @@ -235,7 +235,6 @@ typedef struct st_maria_columndef /* column information */ } MARIA_COLUMNDEF; -extern my_string maria_log_filename; /* Name of logfile */ extern ulong maria_block_size; extern ulong maria_concurrent_insert; extern my_bool maria_flush, maria_delay_key_write, maria_single_user; @@ -281,9 +280,9 @@ extern int maria_delete_table(const char *name); extern int maria_rename(const char *from, const char *to); extern int maria_extra(struct st_maria_info *file, enum ha_extra_function function, void *extra_arg); +extern int maria_reset(struct st_maria_info *file); extern ha_rows maria_records_in_range(struct st_maria_info *info, int inx, key_range *min_key, key_range *max_key); -extern int maria_logging(int activate_log); extern int maria_is_changed(struct st_maria_info *info); extern int maria_delete_all_rows(struct st_maria_info *info); extern uint maria_get_pointer_length(ulonglong file_length, uint def); @@ -328,6 +327,7 @@ typedef struct st_maria_sort_param uchar **sort_keys; byte *rec_buff; void *wordlist, *wordptr; + MEM_ROOT wordroot; char *record; MY_TMPDIR *tmpdir; diff --git a/include/my_handler.h b/include/my_handler.h index ea027b18725..9b086036389 100644 --- a/include/my_handler.h +++ b/include/my_handler.h @@ -18,8 +18,6 @@ #ifndef _my_handler_h #define _my_handler_h -#include "my_base.h" -#include "m_ctype.h" #include "myisampack.h" /* diff --git a/include/myisam.h b/include/myisam.h index 07e8346f981..99c8c8d73ae 100644 --- a/include/myisam.h +++ b/include/myisam.h @@ -334,6 +334,7 @@ typedef struct st_mi_sort_param uchar **sort_keys; byte *rec_buff; void *wordlist, *wordptr; + MEM_ROOT wordroot; char *record; MY_TMPDIR *tmpdir; diff --git a/mysql-test/r/maria.result b/mysql-test/r/maria.result index 3b360873c36..b6924e4395b 100644 --- a/mysql-test/r/maria.result +++ b/mysql-test/r/maria.result @@ -1,9 +1,11 @@ +set global storage_engine=maria; +set session storage_engine=maria; drop table if exists t1,t2; SET SQL_WARNINGS=1; CREATE TABLE t1 ( STRING_DATA char(255) default NULL, KEY string_data (STRING_DATA) -) ENGINE=MARIA; +); INSERT INTO t1 VALUES ('AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA'); INSERT INTO t1 VALUES ('DDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDD'); INSERT INTO t1 VALUES ('FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF'); @@ -14,7 +16,7 @@ CHECK TABLE t1; Table Op Msg_type Msg_text test.t1 check status OK drop table t1; -create table t1 (a tinyint not null auto_increment, b blob not null, primary key (a)) engine=maria; +create table t1 (a tinyint not null auto_increment, b blob not null, primary key (a)); check table t1; Table Op Msg_type Msg_text test.t1 check status OK @@ -32,7 +34,7 @@ check table t1; Table Op Msg_type Msg_text test.t1 check status OK drop table t1; -create table t1 (a int not null auto_increment, b int not null, primary key (a), index(b)) engine=maria; +create table t1 (a int not null auto_increment, b int not null, primary key (a), index(b)); insert into t1 (b) values (1),(2),(2),(2),(2); optimize table t1; Table Op Msg_type Msg_text @@ -49,7 +51,7 @@ Table Non_unique Key_name Seq_in_index Column_name Collation Cardinality Sub_par t1 0 PRIMARY 1 a A 5 NULL NULL BTREE t1 1 b 1 b A 1 NULL NULL BTREE drop table t1; -create table t1 (a int not null, b int not null, c int not null, primary key (a),key(b)) engine=maria; +create table t1 (a int not null, b int not null, c int not null, primary key (a),key(b)); insert into t1 values (3,3,3),(1,1,1),(2,2,2),(4,4,4); explain select * from t1 order by a; id select_type table type possible_keys key key_len ref rows Extra @@ -76,7 +78,7 @@ explain select a,b,c from t1; id select_type table type possible_keys key key_len ref rows Extra 1 SIMPLE t1 ALL NULL NULL NULL NULL 4 drop table t1; -CREATE TABLE t1 (a INT) engine=maria; +CREATE TABLE t1 (a INT); INSERT INTO t1 VALUES (1), (2), (3); LOCK TABLES t1 WRITE; INSERT INTO t1 VALUES (1), (2), (3); @@ -84,7 +86,7 @@ OPTIMIZE TABLE t1; Table Op Msg_type Msg_text test.t1 optimize status OK DROP TABLE t1; -create table t1 ( t1 char(255), key(t1(250))) engine=maria; +create table t1 ( t1 char(255), key(t1(250))); insert t1 values ('137513751375137513751375137513751375137569516951695169516951695169516951695169'); insert t1 values ('178417841784178417841784178417841784178403420342034203420342034203420342034203'); insert t1 values ('213872387238723872387238723872387238723867376737673767376737673767376737673767'); @@ -245,7 +247,7 @@ int, i967 int, i968 int, i969 int, i970 int, i971 int, i972 int, i973 int, i974 int, i975 int, i976 int, i977 int, i978 int, i979 int, i980 int, i981 int, i982 int, i983 int, i984 int, i985 int, i986 int, i987 int, i988 int, i989 int, i990 int, i991 int, i992 int, i993 int, i994 int, i995 int, i996 int, i997 int, i998 -int, i999 int, i1000 int, b blob) row_format=dynamic engine=maria; +int, i999 int, i1000 int, b blob) row_format=dynamic; insert into t1 values (1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, @@ -313,7 +315,7 @@ KEY `ip` (`ip`), KEY `poster_login` (`poster_login`), KEY `topic_id` (`topic_id`), FULLTEXT KEY `post_text` (`post_text`) -) ENGINE=MARIA; +); INSERT INTO t1 (post_text) VALUES ('ceci est un test'),('ceci est un test'),('ceci est un test'),('ceci est un test'),('ceci est un test'); REPAIR TABLE t1; Table Op Msg_type Msg_text @@ -322,15 +324,15 @@ CHECK TABLE t1; Table Op Msg_type Msg_text test.t1 check status OK drop table t1; -CREATE TABLE t1 (a varchar(255), b varchar(255), c varchar(255), d varchar(255), e varchar(255), KEY t1 (a, b, c, d, e)) engine=maria; +CREATE TABLE t1 (a varchar(255), b varchar(255), c varchar(255), d varchar(255), e varchar(255), KEY t1 (a, b, c, d, e)); ERROR 42000: Specified key was too long; max key length is 1000 bytes -CREATE TABLE t1 (a varchar(255), b varchar(255), c varchar(255), d varchar(255), e varchar(255)) engine=maria; +CREATE TABLE t1 (a varchar(255), b varchar(255), c varchar(255), d varchar(255), e varchar(255)); ALTER TABLE t1 ADD INDEX t1 (a, b, c, d, e); ERROR 42000: Specified key was too long; max key length is 1000 bytes DROP TABLE t1; -CREATE TABLE t1 (a int not null, b int, c int, key(b), key(c), key(a,b), key(c,a)) engine=maria; +CREATE TABLE t1 (a int not null, b int, c int, key(b), key(c), key(a,b), key(c,a)); INSERT into t1 values (0, null, 0), (0, null, 1), (0, null, 2), (0, null,3), (1,1,4); -create table t2 (a int not null, b int, c int, key(b), key(c), key(a)) engine=maria; +create table t2 (a int not null, b int, c int, key(b), key(c), key(a)); INSERT into t2 values (1,1,1), (2,2,2); optimize table t1; Table Op Msg_type Msg_text @@ -346,11 +348,11 @@ t1 1 c_2 2 a A 5 NULL NULL BTREE explain select * from t1,t2 where t1.a=t2.a; id select_type table type possible_keys key key_len ref rows Extra 1 SIMPLE t2 ALL a NULL NULL NULL 2 -1 SIMPLE t1 ALL a NULL NULL NULL 4 Using where +1 SIMPLE t1 ALL a NULL NULL NULL 5 Using where explain select * from t1,t2 force index(a) where t1.a=t2.a; id select_type table type possible_keys key key_len ref rows Extra 1 SIMPLE t2 ALL a NULL NULL NULL 2 -1 SIMPLE t1 ALL a NULL NULL NULL 4 Using where +1 SIMPLE t1 ALL a NULL NULL NULL 5 Using where explain select * from t1 force index(a),t2 force index(a) where t1.a=t2.a; id select_type table type possible_keys key key_len ref rows Extra 1 SIMPLE t2 ALL a NULL NULL NULL 2 @@ -362,7 +364,7 @@ id select_type table type possible_keys key key_len ref rows Extra explain select * from t1,t2 force index(c) where t1.a=t2.a; id select_type table type possible_keys key key_len ref rows Extra 1 SIMPLE t2 ALL NULL NULL NULL NULL 2 -1 SIMPLE t1 ALL a NULL NULL NULL 4 Using where +1 SIMPLE t1 ALL a NULL NULL NULL 5 Using where explain select * from t1 where a=0 or a=2; id select_type table type possible_keys key key_len ref rows Extra 1 SIMPLE t1 ALL a NULL NULL NULL 5 Using where @@ -376,7 +378,7 @@ explain select * from t1 use index() where c=1; id select_type table type possible_keys key key_len ref rows Extra 1 SIMPLE t1 ALL NULL NULL NULL NULL 5 Using where drop table t1,t2; -create table t1 (a int not null auto_increment primary key, b varchar(255)) engine=maria; +create table t1 (a int not null auto_increment primary key, b varchar(255)); insert into t1 (b) values (repeat('a',100)),(repeat('b',100)),(repeat('c',100)); update t1 set b=repeat(left(b,1),200) where a=1; delete from t1 where (a & 1)= 0; @@ -398,7 +400,7 @@ check table t1; Table Op Msg_type Msg_text test.t1 check status OK drop table t1; -create table t1 ( a text not null, key a (a(20))) engine=maria; +create table t1 ( a text not null, key a (a(20))); insert into t1 values ('aaa '),('aaa'),('aa'); check table t1; Table Op Msg_type Msg_text @@ -420,7 +422,7 @@ bbb. bbb. aa. drop table t1; -create table t1(a text not null, b text not null, c text not null, index (a(10),b(10),c(10))) engine=maria; +create table t1(a text not null, b text not null, c text not null, index (a(10),b(10),c(10))); insert into t1 values('807780', '477', '165'); insert into t1 values('807780', '477', '162'); insert into t1 values('807780', '472', '162'); @@ -431,7 +433,7 @@ drop table t1; DROP TABLE IF EXISTS t1; Warnings: Note 1051 Unknown table 't1' -CREATE TABLE t1 (a varchar(150) NOT NULL, KEY (a)) engine=maria; +CREATE TABLE t1 (a varchar(150) NOT NULL, KEY (a)); INSERT t1 VALUES ("can \tcan"); INSERT t1 VALUES ("can can"); INSERT t1 VALUES ("can"); @@ -444,7 +446,7 @@ CHECK TABLE t1; Table Op Msg_type Msg_text test.t1 check status OK DROP TABLE t1; -create table t1 (a blob) engine=maria; +create table t1 (a blob); insert into t1 values('a '),('a'); select concat(a,'.') from t1 where a='a'; concat(a,'.') @@ -460,7 +462,7 @@ select concat(a,'.') from t1 where a='a '; concat(a,'.') a . drop table t1; -create table t1 (a int not null auto_increment primary key, b text not null, unique b (b(20))) engine=maria; +create table t1 (a int not null auto_increment primary key, b text not null, unique b (b(20))); insert into t1 (b) values ('a'),('b'),('c'); select concat(b,'.') from t1; concat(b,'.') @@ -483,8 +485,8 @@ a concat(b,'.') 1 a. 3 c. drop table t1; -create table t1 (a int not null) engine=maria; -create table t2 (a int not null, primary key (a)) engine=maria; +create table t1 (a int not null); +create table t2 (a int not null, primary key (a)); insert into t1 values (1); insert into t2 values (1),(2); select sql_big_result distinct t1.a from t1,t2 order by t2.a; @@ -508,17 +510,17 @@ drop table t1,t2; create table t1 ( c1 varchar(32), key (c1) -) engine=maria; +); alter table t1 disable keys; insert into t1 values ('a'), ('b'); select c1 from t1 order by c1 limit 1; c1 a drop table t1; -CREATE TABLE t1 (`a` int(11) NOT NULL default '0', `b` int(11) NOT NULL default '0', UNIQUE KEY `a` USING RTREE (`a`,`b`)) ENGINE=MARIA; +CREATE TABLE t1 (`a` int(11) NOT NULL default '0', `b` int(11) NOT NULL default '0', UNIQUE KEY `a` USING RTREE (`a`,`b`)); Got one of the listed errors -create table t1 (a int, b varchar(200), c text not null) checksum=1 engine=maria; -create table t2 (a int, b varchar(200), c text not null) checksum=0 engine=maria; +create table t1 (a int, b varchar(200), c text not null) checksum=1; +create table t2 (a int, b varchar(200), c text not null) checksum=0; insert t1 values (1, "aaa", "bbb"), (NULL, "", "ccccc"), (0, NULL, ""); insert t2 select * from t1; checksum table t1, t2, t3 quick; @@ -543,7 +545,7 @@ test.t3 NULL Warnings: Error 1146 Table 'test.t3' doesn't exist drop table t1,t2; -create table t1 (a int, key (a)) engine=maria; +create table t1 (a int, key (a)); show keys from t1; Table Non_unique Key_name Seq_in_index Column_name Collation Cardinality Sub_part Packed Null Index_type Comment t1 1 a 1 a A NULL NULL NULL YES BTREE @@ -551,7 +553,7 @@ alter table t1 disable keys; show keys from t1; Table Non_unique Key_name Seq_in_index Column_name Collation Cardinality Sub_part Packed Null Index_type Comment t1 1 a 1 a A NULL NULL NULL YES BTREE disabled -create table t2 (a int) engine=maria; +create table t2 (a int); set @@rand_seed1=31415926,@@rand_seed2=2718281828; insert t1 select * from t2; show keys from t1; @@ -569,7 +571,7 @@ show keys from t1; Table Non_unique Key_name Seq_in_index Column_name Collation Cardinality Sub_part Packed Null Index_type Comment t1 1 a 1 a NULL 500 NULL NULL YES HASH drop table t1,t2; -create table t1 ( a tinytext, b char(1), index idx (a(1),b)) engine=maria; +create table t1 ( a tinytext, b char(1), index idx (a(1),b) ); insert into t1 values (null,''), (null,''); explain select count(*) from t1 where a is null; id select_type table type possible_keys key key_len ref rows Extra @@ -579,20 +581,20 @@ count(*) 2 drop table t1; create table t1 (c1 int, c2 varchar(4) not null default '', -key(c2(3))) default charset=utf8 engine=maria; +key(c2(3))) default charset=utf8; insert into t1 values (1,'A'), (2, 'B'), (3, 'A'); update t1 set c2='A B' where c1=2; check table t1; Table Op Msg_type Msg_text test.t1 check status OK drop table t1; -create table t1 (c1 int) engine=maria; +create table t1 (c1 int); insert into t1 values (1),(2),(3),(4); checksum table t1; Table Checksum test.t1 149057747 delete from t1 where c1 = 1; -create table t2 engine=maria as select * from t1; +create table t2 as select * from t1; checksum table t1; Table Checksum test.t1 984116287 @@ -603,7 +605,7 @@ drop table t1, t2; show variables like 'maria_stats_method'; Variable_name Value maria_stats_method nulls_unequal -create table t1 (a int, key(a)) engine=maria; +create table t1 (a int, key(a)); insert into t1 values (0),(1),(2),(3),(4); insert into t1 select NULL from t1; analyze table t1; @@ -668,7 +670,7 @@ maria_stats_method nulls_ignored create table t1 ( a char(3), b char(4), c char(5), d char(6), key(a,b,c,d) -) engine=maria; +); insert into t1 values ('bcd','def1', NULL, 'zz'); insert into t1 values ('bcd','def2', NULL, 'zz'); insert into t1 values ('bce','def1', 'yuu', NULL); @@ -699,7 +701,7 @@ cip INT NOT NULL, time TIME NOT NULL, score INT NOT NULL DEFAULT 0, bob TINYBLOB -) engine=maria; +); insert into t1 (cip, time) VALUES (1, '00:01'), (2, '00:02'), (3,'00:03'); insert into t1 (cip, bob, time) VALUES (4, 'a', '00:04'), (5, 'b', '00:05'), (6, 'c', '00:06'); @@ -729,7 +731,27 @@ select count(id1) from t1 where id2 = 10; count(id1) 5 drop table t1; -set storage_engine=MARIA; +CREATE TABLE t1(a TINYINT, KEY(a)); +INSERT INTO t1 VALUES(1); +SELECT MAX(a) FROM t1 IGNORE INDEX(a); +MAX(a) +1 +ALTER TABLE t1 DISABLE KEYS; +SELECT MAX(a) FROM t1; +MAX(a) +1 +SELECT MAX(a) FROM t1 IGNORE INDEX(a); +MAX(a) +1 +DROP TABLE t1; +CREATE TABLE t1(a CHAR(9), b VARCHAR(7)); +INSERT INTO t1(a) VALUES('xxxxxxxxx'),('xxxxxxxxx'); +UPDATE t1 AS ta1,t1 AS ta2 SET ta1.b='aaaaaa',ta2.b='bbbbbb'; +SELECT * FROM t1; +a b +xxxxxxxxx bbbbbb +xxxxxxxxx bbbbbb +DROP TABLE t1; drop table if exists t1,t2,t3; --- Testing varchar --- --- Testing varchar --- @@ -1301,11 +1323,11 @@ select * from t1 where a=20 and b is null; a b 20 NULL drop table t1; -create table t1 (v varchar(65530), key(v)) engine=maria; +create table t1 (v varchar(65530), key(v)); Warnings: Warning 1071 Specified key was too long; max key length is 1000 bytes drop table if exists t1; -create table t1 (v varchar(65536)) engine=maria; +create table t1 (v varchar(65536)); Warnings: Note 1246 Converting column 'v' from VARCHAR to TEXT show create table t1; @@ -1314,7 +1336,7 @@ t1 CREATE TABLE `t1` ( `v` mediumtext ) ENGINE=MARIA DEFAULT CHARSET=latin1 drop table t1; -create table t1 (v varchar(65530) character set utf8) engine=maria; +create table t1 (v varchar(65530) character set utf8); Warnings: Note 1246 Converting column 'v' from VARCHAR to TEXT show create table t1; @@ -1323,24 +1345,11 @@ t1 CREATE TABLE `t1` ( `v` mediumtext CHARACTER SET utf8 ) ENGINE=MARIA DEFAULT CHARSET=latin1 drop table t1; -create table t1 (v varchar(65535)) engine=maria; +create table t1 (v varchar(65535)); ERROR 42000: Row size too large. The maximum row size for the used table type, not counting BLOBs, is 65535. You have to change some columns to TEXT or BLOBs -set storage_engine=MyISAM; -create table t1 (a int) engine=maria; -drop table if exists t1; -Warnings: -Error 2 Can't find file: 't1' (errno: 2) -create table t1 (a int) engine=maria; -drop table t1; -Got one of the listed errors -create table t1 (a int) engine=maria; -drop table t1; -Got one of the listed errors -drop table t1; -ERROR 42S02: Unknown table 't1' set @save_concurrent_insert=@@concurrent_insert; set global concurrent_insert=1; -create table t1 (a int) engine=maria; +create table t1 (a int); insert into t1 values (1),(2),(3),(4),(5); lock table t1 read local; insert into t1 values(6),(7); @@ -1367,7 +1376,7 @@ check table t1; Table Op Msg_type Msg_text test.t1 check status OK drop table t1; -create table t1 (a int, b varchar(30) default "hello") engine=maria; +create table t1 (a int, b varchar(30) default "hello"); insert into t1 (a) values (1),(2),(3),(4),(5); lock table t1 read local; insert into t1 (a) values(6),(7); @@ -1395,7 +1404,7 @@ Table Op Msg_type Msg_text test.t1 check status OK drop table t1; set global concurrent_insert=@save_concurrent_insert; -create table t1 (a int, key(a)) engine=maria; +create table t1 (a int, key(a)); insert into t1 values (1),(2),(3),(4),(NULL),(NULL),(NULL),(NULL); analyze table t1; Table Op Msg_type Msg_text @@ -1409,9 +1418,172 @@ show keys from t1; Table Non_unique Key_name Seq_in_index Column_name Collation Cardinality Sub_part Packed Null Index_type Comment t1 1 a 1 a A 8 NULL NULL YES BTREE drop table t1; -create table t1 (c1 int) engine=maria pack_keys=0; -create table t2 (c1 int) engine=maria pack_keys=1; -create table t3 (c1 int) engine=maria pack_keys=default; -create table t4 (c1 int) engine=maria pack_keys=2; +show create table t1; +show create table t1; +create table t1 (a int) select 42 a; +select * from t1; +a +9 +select * from t1; +a +99 +select * from t1; +a +42 +drop table t1; +End of 4.1 tests +create table t1 (c1 int) pack_keys=0; +create table t2 (c1 int) pack_keys=1; +create table t3 (c1 int) pack_keys=default; +create table t4 (c1 int) pack_keys=2; ERROR 42000: You have an error in your SQL syntax; check the manual that corresponds to your MySQL server version for the right syntax to use near '2' at line 1 drop table t1, t2, t3; +End of 5.0 tests +create table t1 (a int not null, key `a` (a) key_block_size=1024); +show create table t1; +Table Create Table +t1 CREATE TABLE `t1` ( + `a` int(11) NOT NULL, + KEY `a` (`a`) KEY_BLOCK_SIZE=1024 +) ENGINE=MARIA DEFAULT CHARSET=latin1 +drop table t1; +create table t1 (a int not null, key `a` (a) key_block_size=2048); +show create table t1; +Table Create Table +t1 CREATE TABLE `t1` ( + `a` int(11) NOT NULL, + KEY `a` (`a`) KEY_BLOCK_SIZE=2048 +) ENGINE=MARIA DEFAULT CHARSET=latin1 +drop table t1; +create table t1 (a varchar(2048), key `a` (a)); +Warnings: +Warning 1071 Specified key was too long; max key length is 1000 bytes +show create table t1; +Table Create Table +t1 CREATE TABLE `t1` ( + `a` varchar(2048) DEFAULT NULL, + KEY `a` (`a`(1000)) +) ENGINE=MARIA DEFAULT CHARSET=latin1 +drop table t1; +create table t1 (a varchar(2048), key `a` (a) key_block_size=1024); +Warnings: +Warning 1071 Specified key was too long; max key length is 1000 bytes +show create table t1; +Table Create Table +t1 CREATE TABLE `t1` ( + `a` varchar(2048) DEFAULT NULL, + KEY `a` (`a`(1000)) KEY_BLOCK_SIZE=4096 +) ENGINE=MARIA DEFAULT CHARSET=latin1 +drop table t1; +create table t1 (a int not null, b varchar(2048), key (a), key(b)) key_block_size=1024; +Warnings: +Warning 1071 Specified key was too long; max key length is 1000 bytes +show create table t1; +Table Create Table +t1 CREATE TABLE `t1` ( + `a` int(11) NOT NULL, + `b` varchar(2048) DEFAULT NULL, + KEY `a` (`a`), + KEY `b` (`b`(1000)) KEY_BLOCK_SIZE=4096 +) ENGINE=MARIA DEFAULT CHARSET=latin1 KEY_BLOCK_SIZE=1024 +alter table t1 key_block_size=2048; +show create table t1; +Table Create Table +t1 CREATE TABLE `t1` ( + `a` int(11) NOT NULL, + `b` varchar(2048) DEFAULT NULL, + KEY `a` (`a`) KEY_BLOCK_SIZE=1024, + KEY `b` (`b`(1000)) KEY_BLOCK_SIZE=4096 +) ENGINE=MARIA DEFAULT CHARSET=latin1 KEY_BLOCK_SIZE=2048 +alter table t1 add c int, add key (c); +show create table t1; +Table Create Table +t1 CREATE TABLE `t1` ( + `a` int(11) NOT NULL, + `b` varchar(2048) DEFAULT NULL, + `c` int(11) DEFAULT NULL, + KEY `a` (`a`) KEY_BLOCK_SIZE=1024, + KEY `b` (`b`(1000)) KEY_BLOCK_SIZE=4096, + KEY `c` (`c`) +) ENGINE=MARIA DEFAULT CHARSET=latin1 KEY_BLOCK_SIZE=2048 +alter table t1 key_block_size=0; +alter table t1 add d int, add key (d); +show create table t1; +Table Create Table +t1 CREATE TABLE `t1` ( + `a` int(11) NOT NULL, + `b` varchar(2048) DEFAULT NULL, + `c` int(11) DEFAULT NULL, + `d` int(11) DEFAULT NULL, + KEY `a` (`a`) KEY_BLOCK_SIZE=1024, + KEY `b` (`b`(1000)) KEY_BLOCK_SIZE=4096, + KEY `c` (`c`) KEY_BLOCK_SIZE=2048, + KEY `d` (`d`) +) ENGINE=MARIA DEFAULT CHARSET=latin1 +drop table t1; +create table t1 (a int not null, b varchar(2048), key (a), key(b)) key_block_size=8192; +Warnings: +Warning 1071 Specified key was too long; max key length is 1000 bytes +show create table t1; +Table Create Table +t1 CREATE TABLE `t1` ( + `a` int(11) NOT NULL, + `b` varchar(2048) DEFAULT NULL, + KEY `a` (`a`), + KEY `b` (`b`(1000)) +) ENGINE=MARIA DEFAULT CHARSET=latin1 KEY_BLOCK_SIZE=8192 +drop table t1; +create table t1 (a int not null, b varchar(2048), key (a) key_block_size=1024, key(b)) key_block_size=8192; +Warnings: +Warning 1071 Specified key was too long; max key length is 1000 bytes +show create table t1; +Table Create Table +t1 CREATE TABLE `t1` ( + `a` int(11) NOT NULL, + `b` varchar(2048) DEFAULT NULL, + KEY `a` (`a`) KEY_BLOCK_SIZE=1024, + KEY `b` (`b`(1000)) +) ENGINE=MARIA DEFAULT CHARSET=latin1 KEY_BLOCK_SIZE=8192 +drop table t1; +create table t1 (a int not null, b int, key (a) key_block_size=1024, key(b) key_block_size=8192) key_block_size=16384; +show create table t1; +Table Create Table +t1 CREATE TABLE `t1` ( + `a` int(11) NOT NULL, + `b` int(11) DEFAULT NULL, + KEY `a` (`a`) KEY_BLOCK_SIZE=1024, + KEY `b` (`b`) KEY_BLOCK_SIZE=8192 +) ENGINE=MARIA DEFAULT CHARSET=latin1 KEY_BLOCK_SIZE=16384 +drop table t1; +create table t1 (a int not null, key `a` (a) key_block_size=512); +show create table t1; +Table Create Table +t1 CREATE TABLE `t1` ( + `a` int(11) NOT NULL, + KEY `a` (`a`) KEY_BLOCK_SIZE=1024 +) ENGINE=MARIA DEFAULT CHARSET=latin1 +drop table t1; +create table t1 (a varchar(2048), key `a` (a) key_block_size=1000000000000000000); +Warnings: +Warning 1071 Specified key was too long; max key length is 1000 bytes +show create table t1; +Table Create Table +t1 CREATE TABLE `t1` ( + `a` varchar(2048) DEFAULT NULL, + KEY `a` (`a`(1000)) KEY_BLOCK_SIZE=8192 +) ENGINE=MARIA DEFAULT CHARSET=latin1 +drop table t1; +create table t1 (a int not null, key `a` (a) key_block_size=1025); +show create table t1; +Table Create Table +t1 CREATE TABLE `t1` ( + `a` int(11) NOT NULL, + KEY `a` (`a`) KEY_BLOCK_SIZE=2048 +) ENGINE=MARIA DEFAULT CHARSET=latin1 +drop table t1; +create table t1 (a int not null, key key_block_size=1024 (a)); +ERROR 42000: You have an error in your SQL syntax; check the manual that corresponds to your MySQL server version for the right syntax to use near '=1024 (a))' at line 1 +create table t1 (a int not null, key `a` key_block_size=1024 (a)); +ERROR 42000: You have an error in your SQL syntax; check the manual that corresponds to your MySQL server version for the right syntax to use near 'key_block_size=1024 (a))' at line 1 +End of 5.1 tests +set global storage_engine=MyISAM; diff --git a/mysql-test/r/ps_maria.result b/mysql-test/r/ps_maria.result index 00e32cf8fce..f47c72187e2 100644 --- a/mysql-test/r/ps_maria.result +++ b/mysql-test/r/ps_maria.result @@ -1775,7 +1775,7 @@ NULL as const12, @arg12 as param12, show create table t5 ; Table Create Table t5 CREATE TABLE `t5` ( - `const01` bigint(1) NOT NULL DEFAULT '0', + `const01` int(1) NOT NULL DEFAULT '0', `param01` bigint(20) DEFAULT NULL, `const02` decimal(2,1) NOT NULL DEFAULT '0.0', `param02` decimal(65,30) DEFAULT NULL, @@ -1805,7 +1805,7 @@ t5 CREATE TABLE `t5` ( ) ENGINE=MyISAM DEFAULT CHARSET=latin1 select * from t5 ; Catalog Database Table Table_alias Column Column_alias Type Length Max length Is_null Flags Decimals Charsetnr -def test t5 t5 const01 const01 8 1 1 N 32769 0 63 +def test t5 t5 const01 const01 3 1 1 N 32769 0 63 def test t5 t5 param01 param01 8 20 1 Y 32768 0 63 def test t5 t5 const02 const02 246 4 3 N 1 1 63 def test t5 t5 param02 param02 246 67 32 Y 0 30 63 diff --git a/mysql-test/t/maria.test b/mysql-test/t/maria.test index 59d36206039..1a44775277b 100644 --- a/mysql-test/t/maria.test +++ b/mysql-test/t/maria.test @@ -3,6 +3,10 @@ # -- source include/have_maria.inc +let $default=`select @@global.storage_engine`; +set global storage_engine=maria; +set session storage_engine=maria; + # Initialise --disable_warnings drop table if exists t1,t2; @@ -16,7 +20,7 @@ SET SQL_WARNINGS=1; CREATE TABLE t1 ( STRING_DATA char(255) default NULL, KEY string_data (STRING_DATA) -) ENGINE=MARIA; +); INSERT INTO t1 VALUES ('AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA'); INSERT INTO t1 VALUES ('DDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDD'); @@ -31,7 +35,7 @@ drop table t1; # Test problem with rows that are 65517-65520 bytes long # -create table t1 (a tinyint not null auto_increment, b blob not null, primary key (a)) engine=maria; +create table t1 (a tinyint not null auto_increment, b blob not null, primary key (a)); let $1=100; disable_query_log; @@ -57,7 +61,7 @@ drop table t1; # Test bug: Two optimize in a row reset index cardinality # -create table t1 (a int not null auto_increment, b int not null, primary key (a), index(b)) engine=maria; +create table t1 (a int not null auto_increment, b int not null, primary key (a), index(b)); insert into t1 (b) values (1),(2),(2),(2),(2); optimize table t1; show index from t1; @@ -69,7 +73,7 @@ drop table t1; # Test of how ORDER BY works when doing it on the whole table # -create table t1 (a int not null, b int not null, c int not null, primary key (a),key(b)) engine=maria; +create table t1 (a int not null, b int not null, c int not null, primary key (a),key(b)); insert into t1 values (3,3,3),(1,1,1),(2,2,2),(4,4,4); explain select * from t1 order by a; explain select * from t1 order by b; @@ -84,7 +88,7 @@ drop table t1; # # Test of OPTIMIZE of locked and modified tables # -CREATE TABLE t1 (a INT) engine=maria; +CREATE TABLE t1 (a INT); INSERT INTO t1 VALUES (1), (2), (3); LOCK TABLES t1 WRITE; INSERT INTO t1 VALUES (1), (2), (3); @@ -96,7 +100,7 @@ DROP TABLE t1; # in ha_maria::repair, and index size is changed (decreased). # -create table t1 ( t1 char(255), key(t1(250))) engine=maria; +create table t1 ( t1 char(255), key(t1(250))); insert t1 values ('137513751375137513751375137513751375137569516951695169516951695169516951695169'); insert t1 values ('178417841784178417841784178417841784178403420342034203420342034203420342034203'); insert t1 values ('213872387238723872387238723872387238723867376737673767376737673767376737673767'); @@ -258,7 +262,7 @@ int, i967 int, i968 int, i969 int, i970 int, i971 int, i972 int, i973 int, i974 int, i975 int, i976 int, i977 int, i978 int, i979 int, i980 int, i981 int, i982 int, i983 int, i984 int, i985 int, i986 int, i987 int, i988 int, i989 int, i990 int, i991 int, i992 int, i993 int, i994 int, i995 int, i996 int, i997 int, i998 -int, i999 int, i1000 int, b blob) row_format=dynamic engine=maria; +int, i999 int, i1000 int, b blob) row_format=dynamic; insert into t1 values (1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, @@ -325,7 +329,7 @@ CREATE TABLE `t1` ( KEY `poster_login` (`poster_login`), KEY `topic_id` (`topic_id`), FULLTEXT KEY `post_text` (`post_text`) -) ENGINE=MARIA; +); INSERT INTO t1 (post_text) VALUES ('ceci est un test'),('ceci est un test'),('ceci est un test'),('ceci est un test'),('ceci est un test'); @@ -338,8 +342,8 @@ drop table t1; # --error 1071 -CREATE TABLE t1 (a varchar(255), b varchar(255), c varchar(255), d varchar(255), e varchar(255), KEY t1 (a, b, c, d, e)) engine=maria; -CREATE TABLE t1 (a varchar(255), b varchar(255), c varchar(255), d varchar(255), e varchar(255)) engine=maria; +CREATE TABLE t1 (a varchar(255), b varchar(255), c varchar(255), d varchar(255), e varchar(255), KEY t1 (a, b, c, d, e)); +CREATE TABLE t1 (a varchar(255), b varchar(255), c varchar(255), d varchar(255), e varchar(255)); --error 1071 ALTER TABLE t1 ADD INDEX t1 (a, b, c, d, e); DROP TABLE t1; @@ -348,9 +352,9 @@ DROP TABLE t1; # Test of cardinality of keys with NULL # -CREATE TABLE t1 (a int not null, b int, c int, key(b), key(c), key(a,b), key(c,a)) engine=maria; +CREATE TABLE t1 (a int not null, b int, c int, key(b), key(c), key(a,b), key(c,a)); INSERT into t1 values (0, null, 0), (0, null, 1), (0, null, 2), (0, null,3), (1,1,4); -create table t2 (a int not null, b int, c int, key(b), key(c), key(a)) engine=maria; +create table t2 (a int not null, b int, c int, key(b), key(c), key(a)); INSERT into t2 values (1,1,1), (2,2,2); optimize table t1; show index from t1; @@ -369,7 +373,7 @@ drop table t1,t2; # Test bug when updating a split dynamic row where keys are not changed # -create table t1 (a int not null auto_increment primary key, b varchar(255)) engine=maria; +create table t1 (a int not null auto_increment primary key, b varchar(255)); insert into t1 (b) values (repeat('a',100)),(repeat('b',100)),(repeat('c',100)); update t1 set b=repeat(left(b,1),200) where a=1; delete from t1 where (a & 1)= 0; @@ -402,7 +406,7 @@ drop table t1; # # two bugs in maria-space-stripping feature # -create table t1 ( a text not null, key a (a(20))) engine=maria; +create table t1 ( a text not null, key a (a(20))); insert into t1 values ('aaa '),('aaa'),('aa'); check table t1; repair table t1; @@ -416,7 +420,7 @@ drop table t1; # Third bug in the same code (BUG#2295) # -create table t1(a text not null, b text not null, c text not null, index (a(10),b(10),c(10))) engine=maria; +create table t1(a text not null, b text not null, c text not null, index (a(10),b(10),c(10))); insert into t1 values('807780', '477', '165'); insert into t1 values('807780', '477', '162'); insert into t1 values('807780', '472', '162'); @@ -427,7 +431,7 @@ drop table t1; # space-stripping in _mi_prefix_search: BUG#5284 # DROP TABLE IF EXISTS t1; -CREATE TABLE t1 (a varchar(150) NOT NULL, KEY (a)) engine=maria; +CREATE TABLE t1 (a varchar(150) NOT NULL, KEY (a)); INSERT t1 VALUES ("can \tcan"); INSERT t1 VALUES ("can can"); INSERT t1 VALUES ("can"); @@ -438,7 +442,7 @@ DROP TABLE t1; # # Verify blob handling # -create table t1 (a blob) engine=maria; +create table t1 (a blob); insert into t1 values('a '),('a'); select concat(a,'.') from t1 where a='a'; select concat(a,'.') from t1 where a='a '; @@ -450,7 +454,7 @@ drop table t1; # # Test text and unique # -create table t1 (a int not null auto_increment primary key, b text not null, unique b (b(20))) engine=maria; +create table t1 (a int not null auto_increment primary key, b text not null, unique b (b(20))); insert into t1 (b) values ('a'),('b'),('c'); select concat(b,'.') from t1; update t1 set b='b ' where a=2; @@ -466,8 +470,8 @@ drop table t1; # # Test keys with 0 segments. (Bug #3203) # -create table t1 (a int not null) engine=maria; -create table t2 (a int not null, primary key (a)) engine=maria; +create table t1 (a int not null); +create table t2 (a int not null, primary key (a)); insert into t1 values (1); insert into t2 values (1),(2); select sql_big_result distinct t1.a from t1,t2 order by t2.a; @@ -483,7 +487,7 @@ drop table t1,t2; create table t1 ( c1 varchar(32), key (c1) -) engine=maria; +); alter table t1 disable keys; insert into t1 values ('a'), ('b'); select c1 from t1 order by c1 limit 1; @@ -493,13 +497,13 @@ drop table t1; # Test RTREE index # --error 1235, 1289 -CREATE TABLE t1 (`a` int(11) NOT NULL default '0', `b` int(11) NOT NULL default '0', UNIQUE KEY `a` USING RTREE (`a`,`b`)) ENGINE=MARIA; +CREATE TABLE t1 (`a` int(11) NOT NULL default '0', `b` int(11) NOT NULL default '0', UNIQUE KEY `a` USING RTREE (`a`,`b`)); # INSERT INTO t1 VALUES (1,1),(1,1); # DELETE FROM rt WHERE a<1; # DROP TABLE IF EXISTS t1; -create table t1 (a int, b varchar(200), c text not null) checksum=1 engine=maria; -create table t2 (a int, b varchar(200), c text not null) checksum=0 engine=maria; +create table t1 (a int, b varchar(200), c text not null) checksum=1; +create table t2 (a int, b varchar(200), c text not null) checksum=0; insert t1 values (1, "aaa", "bbb"), (NULL, "", "ccccc"), (0, NULL, ""); insert t2 select * from t1; checksum table t1, t2, t3 quick; @@ -508,11 +512,11 @@ checksum table t1, t2, t3 extended; #show table status; drop table t1,t2; -create table t1 (a int, key (a)) engine=maria; +create table t1 (a int, key (a)); show keys from t1; alter table t1 disable keys; show keys from t1; -create table t2 (a int) engine=maria; +create table t2 (a int); let $i=1000; set @@rand_seed1=31415926,@@rand_seed2=2718281828; --disable_query_log @@ -534,7 +538,7 @@ drop table t1,t2; # # index search for NULL in blob. Bug #4816 # -create table t1 ( a tinytext, b char(1), index idx (a(1),b)) engine=maria; +create table t1 ( a tinytext, b char(1), index idx (a(1),b) ); insert into t1 values (null,''), (null,''); explain select count(*) from t1 where a is null; select count(*) from t1 where a is null; @@ -544,7 +548,7 @@ drop table t1; # bug9188 - Corruption Can't open file: 'table.MYI' (errno: 145) # create table t1 (c1 int, c2 varchar(4) not null default '', - key(c2(3))) default charset=utf8 engine=maria; + key(c2(3))) default charset=utf8; insert into t1 values (1,'A'), (2, 'B'), (3, 'A'); update t1 set c2='A B' where c1=2; check table t1; @@ -555,11 +559,11 @@ drop table t1; # Bug#12296 - CHECKSUM TABLE reports 0 for the table # This happened if the first record was marked as deleted. # -create table t1 (c1 int) engine=maria; +create table t1 (c1 int); insert into t1 values (1),(2),(3),(4); checksum table t1; delete from t1 where c1 = 1; -create table t2 engine=maria as select * from t1; +create table t2 as select * from t1; # The following returns 0 with the bug in place. checksum table t1; # The above should give the same number as the following. @@ -572,7 +576,7 @@ drop table t1, t2; show variables like 'maria_stats_method'; -create table t1 (a int, key(a)) engine=maria; +create table t1 (a int, key(a)); insert into t1 values (0),(1),(2),(3),(4); insert into t1 select NULL from t1; @@ -623,7 +627,7 @@ show variables like 'maria_stats_method'; create table t1 ( a char(3), b char(4), c char(5), d char(6), key(a,b,c,d) -) engine=maria; +); insert into t1 values ('bcd','def1', NULL, 'zz'); insert into t1 values ('bcd','def2', NULL, 'zz'); insert into t1 values ('bce','def1', 'yuu', NULL); @@ -644,7 +648,7 @@ create table t1( time TIME NOT NULL, score INT NOT NULL DEFAULT 0, bob TINYBLOB -) engine=maria; +); insert into t1 (cip, time) VALUES (1, '00:01'), (2, '00:02'), (3,'00:03'); insert into t1 (cip, bob, time) VALUES (4, 'a', '00:04'), (5, 'b', '00:05'), @@ -663,7 +667,7 @@ create table t1 ( t text not null, primary key (id1), key x (id2, t(32)) -) engine=maria; +) engine=maria; # engine clause is redundant but it's to test its parsing insert into t1 (id2, t) values (10, 'abc'), (10, 'abc'), (10, 'abc'), (20, 'abc'), (20, 'abc'), (20, 'def'), @@ -672,51 +676,49 @@ select count(*) from t1 where id2 = 10; select count(id1) from t1 where id2 = 10; drop table t1; -# End of 4.1 tests +# +# BUG##20357 - Got error 124 from storage engine using MIN and MAX functions +# in queries +# +CREATE TABLE t1(a TINYINT, KEY(a)); +INSERT INTO t1 VALUES(1); +SELECT MAX(a) FROM t1 IGNORE INDEX(a); +ALTER TABLE t1 DISABLE KEYS; +SELECT MAX(a) FROM t1; +SELECT MAX(a) FROM t1 IGNORE INDEX(a); +DROP TABLE t1; + +# +# BUG#18036 - update of table joined to self reports table as crashed +# +CREATE TABLE t1(a CHAR(9), b VARCHAR(7)); +INSERT INTO t1(a) VALUES('xxxxxxxxx'),('xxxxxxxxx'); +UPDATE t1 AS ta1,t1 AS ta2 SET ta1.b='aaaaaa',ta2.b='bbbbbb'; +SELECT * FROM t1; +DROP TABLE t1; # # Test varchar # -let $default=`select @@storage_engine`; -set storage_engine=MARIA; source include/varchar.inc; # # Some errors/warnings on create # -create table t1 (v varchar(65530), key(v)) engine=maria; +create table t1 (v varchar(65530), key(v)); drop table if exists t1; -create table t1 (v varchar(65536)) engine=maria; +create table t1 (v varchar(65536)); show create table t1; drop table t1; -create table t1 (v varchar(65530) character set utf8) engine=maria; +create table t1 (v varchar(65530) character set utf8); show create table t1; drop table t1; # MARIA specific varchar tests --error 1118 -create table t1 (v varchar(65535)) engine=maria; - -eval set storage_engine=$default; - -# -# Test how DROP TABLE works if the index or data file doesn't exists - -create table t1 (a int) engine=maria; -system rm $MYSQLTEST_VARDIR/master-data/test/t1.MAI ; -drop table if exists t1; -create table t1 (a int) engine=maria; -system rm $MYSQLTEST_VARDIR/master-data/test/t1.MAI ; ---error 1051,6 -drop table t1; -create table t1 (a int) engine=maria; -system rm $MYSQLTEST_VARDIR/master-data/test/t1.MAD ; ---error 1105,6,29 -drop table t1; ---error 1051 -drop table t1; +create table t1 (v varchar(65535)); # # Test concurrent insert @@ -724,7 +726,7 @@ drop table t1; # set @save_concurrent_insert=@@concurrent_insert; set global concurrent_insert=1; -create table t1 (a int) engine=maria; +create table t1 (a int); insert into t1 values (1),(2),(3),(4),(5); lock table t1 read local; connect (con1,localhost,root,,); @@ -749,7 +751,7 @@ drop table t1; disconnect con1; # Same test with dynamic record length -create table t1 (a int, b varchar(30) default "hello") engine=maria; +create table t1 (a int, b varchar(30) default "hello"); insert into t1 (a) values (1),(2),(3),(4),(5); lock table t1 read local; connect (con1,localhost,root,,); @@ -777,7 +779,7 @@ set global concurrent_insert=@save_concurrent_insert; # BUG#9622 - ANALYZE TABLE and ALTER TABLE .. ENABLE INDEX produce # different statistics on the same table with NULL values. -create table t1 (a int, key(a)) engine=maria; +create table t1 (a int, key(a)); insert into t1 values (1),(2),(3),(4),(NULL),(NULL),(NULL),(NULL); analyze table t1; @@ -789,14 +791,120 @@ show keys from t1; drop table t1; +# +# Bug#8706 - temporary table with data directory option fails +# +connect (session1,localhost,root,,); +connect (session2,localhost,root,,); + +connection session1; +disable_query_log; +eval create temporary table t1 (a int) data directory="$MYSQLTEST_VARDIR/tmp" select 9 a; +enable_query_log; +disable_result_log; +show create table t1; +enable_result_log; + +connection session2; +disable_query_log; +eval create temporary table t1 (a int) data directory="$MYSQLTEST_VARDIR/tmp" select 99 a; +enable_query_log; +disable_result_log; +show create table t1; +enable_result_log; + +connection default; +create table t1 (a int) select 42 a; + +connection session1; +select * from t1; +disconnect session1; +connection session2; +select * from t1; +disconnect session2; +connection default; +select * from t1; +drop table t1; + +--echo End of 4.1 tests + # # Bug#10056 - PACK_KEYS option take values greater than 1 while creating table # -create table t1 (c1 int) engine=maria pack_keys=0; -create table t2 (c1 int) engine=maria pack_keys=1; -create table t3 (c1 int) engine=maria pack_keys=default; +create table t1 (c1 int) pack_keys=0; +create table t2 (c1 int) pack_keys=1; +create table t3 (c1 int) pack_keys=default; --error 1064 -create table t4 (c1 int) engine=maria pack_keys=2; +create table t4 (c1 int) pack_keys=2; drop table t1, t2, t3; +--echo End of 5.0 tests + +# +# Test of key_block_size +# + +create table t1 (a int not null, key `a` (a) key_block_size=1024); +show create table t1; +drop table t1; + +create table t1 (a int not null, key `a` (a) key_block_size=2048); +show create table t1; +drop table t1; + +create table t1 (a varchar(2048), key `a` (a)); +show create table t1; +drop table t1; + +create table t1 (a varchar(2048), key `a` (a) key_block_size=1024); +show create table t1; +drop table t1; + +create table t1 (a int not null, b varchar(2048), key (a), key(b)) key_block_size=1024; +show create table t1; +alter table t1 key_block_size=2048; +show create table t1; +alter table t1 add c int, add key (c); +show create table t1; +alter table t1 key_block_size=0; +alter table t1 add d int, add key (d); +show create table t1; +drop table t1; + +create table t1 (a int not null, b varchar(2048), key (a), key(b)) key_block_size=8192; +show create table t1; +drop table t1; + +create table t1 (a int not null, b varchar(2048), key (a) key_block_size=1024, key(b)) key_block_size=8192; +show create table t1; +drop table t1; + +create table t1 (a int not null, b int, key (a) key_block_size=1024, key(b) key_block_size=8192) key_block_size=16384; +show create table t1; +drop table t1; + + +# Test limits and errors of key_block_size + +create table t1 (a int not null, key `a` (a) key_block_size=512); +show create table t1; +drop table t1; + +create table t1 (a varchar(2048), key `a` (a) key_block_size=1000000000000000000); +show create table t1; +drop table t1; + +create table t1 (a int not null, key `a` (a) key_block_size=1025); +show create table t1; +drop table t1; + +--error 1064 +create table t1 (a int not null, key key_block_size=1024 (a)); +--error 1064 +create table t1 (a int not null, key `a` key_block_size=1024 (a)); + +--echo End of 5.1 tests + +eval set global storage_engine=$default; + # End of 5.2 tests diff --git a/mysys/my_handler.c b/mysys/my_handler.c index 256f67f6125..3230a669641 100644 --- a/mysys/my_handler.c +++ b/mysys/my_handler.c @@ -524,7 +524,6 @@ HA_KEYSEG *ha_find_null(HA_KEYSEG *keyseg, uchar *a) case HA_KEYTYPE_VARTEXT1: case HA_KEYTYPE_VARTEXT2: case HA_KEYTYPE_VARBINARY1: - case HA_KEYTYPE_VARTEXT2: case HA_KEYTYPE_VARBINARY2: { int a_length; diff --git a/sql/ha_maria.cc b/sql/ha_maria.cc index 740dfa85a23..aa244450bb5 100644 --- a/sql/ha_maria.cc +++ b/sql/ha_maria.cc @@ -30,8 +30,9 @@ #include "../storage/maria/ma_rt_index.h" #endif +#include + ulong maria_recover_options= HA_RECOVER_NONE; -static bool ha_maria_init(); /* bits in maria_recover_options */ const char *maria_recover_names[]= @@ -60,65 +61,9 @@ TYPELIB maria_stats_method_typelib= ** MARIA tables *****************************************************************************/ -static handler *maria_create_handler(TABLE_SHARE * table); - -/* MARIA handlerton */ -static const char maria_hton_name[]= "MARIA"; -static const char maria_hton_comment[]= - "Transactional storage engine, optimized for long running transactions"; - -handlerton maria_hton= -{ - MYSQL_HANDLERTON_INTERFACE_VERSION, - maria_hton_name, - SHOW_OPTION_YES, - maria_hton_comment, - DB_TYPE_MARIA, - ha_maria_init, - 0, /* slot */ - 0, /* savepoint size. */ - NULL, /* close_connection */ - NULL, /* savepoint */ - NULL, /* rollback to savepoint */ - NULL, /* release savepoint */ - NULL, /* commit */ - NULL, /* rollback */ - NULL, /* prepare */ - NULL, /* recover */ - NULL, /* commit_by_xid */ - NULL, /* rollback_by_xid */ - NULL, /* create_cursor_read_view */ - NULL, /* set_cursor_read_view */ - NULL, /* close_cursor_read_view */ - /* - MARIA doesn't support transactions yet and doesn't have - transaction-dependent context: cursors can survive a commit. - */ - maria_create_handler, /* Create a new handler */ - NULL, /* Drop a database */ - maria_panic, /* Panic call */ - NULL, /* Start Consistent Snapshot */ - NULL, /* Flush logs */ - NULL, /* Show status */ - NULL, /* Partition flags */ - NULL, /* Alter table flags */ - NULL, /* Alter Tablespace */ - NULL, /* Fill Files Table */ - HTON_CAN_RECREATE, - NULL, /* binlog_func */ - NULL, /* binlog_log_query */ - NULL /* release_temporary_latches */ -}; - - -static bool ha_maria_init() +static handler *maria_create_handler(TABLE_SHARE * table, MEM_ROOT *mem_root) { - return test(maria_init()); -} - -static handler *maria_create_handler(TABLE_SHARE *table) -{ - return new ha_maria(table); + return new (mem_root) ha_maria(table); } @@ -208,9 +153,10 @@ void _ma_check_print_warning(HA_CHECK *param, const char *fmt, ...) ha_maria::ha_maria(TABLE_SHARE *table_arg): handler(&maria_hton, table_arg), file(0), int_table_flags(HA_NULL_IN_KEY | HA_CAN_FULLTEXT | HA_CAN_SQL_HANDLER | - HA_DUPP_POS | HA_CAN_INDEX_BLOBS | HA_AUTO_PART_KEY | - HA_FILE_BASED | HA_CAN_GEOMETRY | HA_READ_RND_SAME | - HA_CAN_INSERT_DELAYED | HA_CAN_BIT_FIELD), + HA_DUPLICATE_POS | HA_CAN_INDEX_BLOBS | HA_AUTO_PART_KEY | + HA_FILE_BASED | HA_CAN_GEOMETRY | HA_NO_TRANSACTIONS | + HA_CAN_INSERT_DELAYED | HA_CAN_BIT_FIELD | + HA_HAS_RECORDS | HA_STATS_RECORDS_IS_EXACT), can_enable_indexes(1) {} @@ -376,6 +322,7 @@ int ha_maria::open(const char *name, int mode, uint test_if_locked) if (table->key_info[i].flags & HA_USES_PARSER) file->s->keyinfo[i].parser= (struct st_mysql_ftparser *) parser->plugin->info; + table->key_info[i].block_size= file->s->keyinfo[i].block_length; } return (0); } @@ -1284,13 +1231,14 @@ int ha_maria::index_read_idx(byte * buf, uint index, const byte * key, int ha_maria::index_read_last(byte * buf, const byte * key, uint key_len) { + DBUG_ENTER("ha_maria::index_read_last"); DBUG_ASSERT(inited == INDEX); statistic_increment(table->in_use->status_var.ha_read_key_count, &LOCK_status); int error= maria_rkey(file, buf, active_index, key, key_len, HA_READ_PREFIX_LAST); table->status= error ? STATUS_NOT_FOUND : 0; - return error; + DBUG_RETURN(error); } @@ -1355,7 +1303,7 @@ int ha_maria::rnd_init(bool scan) { if (scan) return maria_scan_init(file); - return maria_extra(file, HA_EXTRA_RESET, 0); + return maria_reset(file); // Free buffers } @@ -1400,24 +1348,23 @@ void ha_maria::info(uint flag) (void) maria_status(file, &info, flag); if (flag & HA_STATUS_VARIABLE) { - records= info.records; - deleted= info.deleted; - data_file_length= info.data_file_length; - index_file_length= info.index_file_length; - delete_length= info.delete_length; - check_time= info.check_time; - mean_rec_length= info.mean_reclength; + stats.records= info.records; + stats.deleted= info.deleted; + stats.data_file_length= info.data_file_length; + stats.index_file_length= info.index_file_length; + stats.delete_length= info.delete_length; + stats.check_time= info.check_time; + stats.mean_rec_length= info.mean_reclength; } if (flag & HA_STATUS_CONST) { TABLE_SHARE *share= table->s; - max_data_file_length= info.max_data_file_length; - max_index_file_length= info.max_index_file_length; - create_time= info.create_time; - sortkey= info.sortkey; + stats.max_data_file_length= info.max_data_file_length; + stats.max_index_file_length= info.max_index_file_length; + stats.create_time= info.create_time; ref_length= info.reflength; share->db_options_in_use= info.options; - block_size= maria_block_size; + stats.block_size= maria_block_size; /* Update share */ if (share->tmp_table == NO_TMP_TABLE) @@ -1448,11 +1395,11 @@ void ha_maria::info(uint flag) if (flag & HA_STATUS_ERRKEY) { errkey= info.errkey; - my_store_ptr(dupp_ref, ref_length, info.dupp_key_pos); + my_store_ptr(dup_ref, ref_length, info.dupp_key_pos); } /* Faster to always update, than to do it based on flag */ - update_time= info.update_time; - auto_increment_value= info.auto_increment; + stats.update_time= info.update_time; + stats.auto_increment_value= info.auto_increment; } @@ -1463,6 +1410,10 @@ int ha_maria::extra(enum ha_extra_function operation) return maria_extra(file, operation, 0); } +int ha_maria::reset(void) +{ + return maria_reset(file); +} /* To be used with WRITE_CACHE and EXTRA_CACHE */ @@ -1510,7 +1461,7 @@ void ha_maria::update_create_info(HA_CREATE_INFO *create_info) ha_maria::info(HA_STATUS_AUTO | HA_STATUS_CONST); if (!(create_info->used_fields & HA_CREATE_USED_AUTO)) { - create_info->auto_increment_value= auto_increment_value; + create_info->auto_increment_value= stats.auto_increment_value; } create_info->data_file_name= data_file_name; create_info->index_file_name= index_file_name; @@ -1552,6 +1503,8 @@ int ha_maria::create(const char *name, register TABLE *table_arg, keydef[i].key_alg= pos->algorithm == HA_KEY_ALG_UNDEF ? (pos->flags & HA_SPATIAL ? HA_KEY_ALG_RTREE : HA_KEY_ALG_BTREE) : pos->algorithm; + keydef[i].block_length= pos->block_size; + keydef[i].seg= keyseg; keydef[i].keysegs= pos->key_parts; for (j= 0; j < pos->key_parts; j++) @@ -1731,7 +1684,10 @@ int ha_maria::rename_table(const char *from, const char *to) } -ulonglong ha_maria::get_auto_increment() +void ha_maria::get_auto_increment(ulonglong offset, ulonglong increment, + ulonglong nb_desired_values, + ulonglong *first_value, + ulonglong *nb_reserved_values) { ulonglong nr; int error; @@ -1740,7 +1696,10 @@ ulonglong ha_maria::get_auto_increment() if (!table->s->next_number_key_offset) { // Autoincrement at key-start ha_maria::info(HA_STATUS_AUTO); - return auto_increment_value; + *first_value= stats.auto_increment_value; + /* Maria has only table-level lock for now, so reserves to +inf */ + *nb_reserved_values= ULONGLONG_MAX; + return; } /* it's safe to call the following if bulk_insert isn't on */ @@ -1761,7 +1720,14 @@ ulonglong ha_maria::get_auto_increment() val_int_offset(table->s->rec_buff_length) + 1); } extra(HA_EXTRA_NO_KEYREAD); - return nr; + *first_value= nr; + /* + MySQL needs to call us for next row: assume we are inserting ("a",null) + here, we return 3, and next this statement will want to insert ("b",null): + there is no reason why ("b",3+1) would be the good row to insert: maybe it + already exists, maybe 3+1 is too large... + */ + *nb_reserved_values= 1; } @@ -1825,9 +1791,11 @@ bool ha_maria::check_if_incompatible_data(HA_CREATE_INFO *info, { uint options= table->s->db_options_in_use; - if (info->auto_increment_value != auto_increment_value || + if (info->auto_increment_value != stats.auto_increment_value || info->data_file_name != data_file_name || - info->index_file_name != index_file_name || table_changes == IS_EQUAL_NO) + info->index_file_name != index_file_name || + table_changes == IS_EQUAL_NO || + table_changes & IS_EQUAL_PACK_LENGTH) // Not implemented yet return COMPATIBLE_DATA_NO; if ((options & (HA_OPTION_PACK_RECORD | HA_OPTION_CHECKSUM | @@ -1838,14 +1806,29 @@ bool ha_maria::check_if_incompatible_data(HA_CREATE_INFO *info, return COMPATIBLE_DATA_YES; } +handlerton maria_hton; + +static int ha_maria_init() +{ + maria_hton.state=SHOW_OPTION_YES; + maria_hton.db_type=DB_TYPE_MARIA; + maria_hton.create=maria_create_handler; + maria_hton.panic=maria_panic; + maria_hton.flags=HTON_CAN_RECREATE; + return test(maria_init()); +} + +struct st_mysql_storage_engine maria_storage_engine= +{ MYSQL_HANDLERTON_INTERFACE_VERSION, &maria_hton }; + mysql_declare_plugin(maria) { MYSQL_STORAGE_ENGINE_PLUGIN, - &maria_hton, - maria_hton_name, + &maria_storage_engine, + "Maria", "MySQL AB", - maria_hton_comment, - NULL, /* Plugin Init */ + "Traditional transactional MySQL tables", + ha_maria_init, /* Plugin Init */ NULL, /* Plugin Deinit */ 0x0100, /* 1.0 */ 0 diff --git a/sql/ha_maria.h b/sql/ha_maria.h index 5387820edc6..cdd2305f6a1 100644 --- a/sql/ha_maria.h +++ b/sql/ha_maria.h @@ -49,7 +49,7 @@ public: { return "MARIA"; } const char *index_type(uint key_number); const char **bas_ext() const; - ulong table_flags() const + ulonglong table_flags() const { return int_table_flags; } ulong index_flags(uint inx, uint part, bool all_parts) const { @@ -106,6 +106,7 @@ public: void info(uint); int extra(enum ha_extra_function operation); int extra_opt(enum ha_extra_function operation, ulong cache_size); + int reset(void); int external_lock(THD * thd, int lock_type); int delete_all_rows(void); int disable_indexes(uint mode); @@ -118,7 +119,10 @@ public: int create(const char *name, TABLE * form, HA_CREATE_INFO * create_info); THR_LOCK_DATA **store_lock(THD * thd, THR_LOCK_DATA ** to, enum thr_lock_type lock_type); - ulonglong get_auto_increment(); + virtual void get_auto_increment(ulonglong offset, ulonglong increment, + ulonglong nb_desired_values, + ulonglong *first_value, + ulonglong *nb_reserved_values); int rename_table(const char *from, const char *to); int delete_table(const char *name); int check(THD * thd, HA_CHECK_OPT * check_opt); diff --git a/sql/mysqld.cc b/sql/mysqld.cc index b1ff5bdc54d..a666bec32d7 100644 --- a/sql/mysqld.cc +++ b/sql/mysqld.cc @@ -6021,7 +6021,6 @@ log and this option does nothing anymore.", (gptr*) &maria_stats_method_str, (gptr*) &maria_stats_method_str, 0, GET_STR, REQUIRED_ARG, 0, 0, 0, 0, 0, 0}, #endif - {"max_allowed_packet", OPT_MAX_ALLOWED_PACKET, "Max packetlength to send/receive from to server.", (gptr*) &global_system_variables.max_allowed_packet, diff --git a/storage/maria/Makefile.am b/storage/maria/Makefile.am index 4132a4208a2..d4315b4d446 100644 --- a/storage/maria/Makefile.am +++ b/storage/maria/Makefile.am @@ -14,7 +14,7 @@ # along with this program; if not, write to the Free Software # Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA -EXTRA_DIST = ma_test_all.sh ma_test_all.res ma_ft_stem.c cmakelists.txt +EXTRA_DIST = ma_test_all.sh ma_test_all.res ma_ft_stem.c CMakeLists.txt pkgdata_DATA = ma_test_all ma_test_all.res INCLUDES = -I$(top_builddir)/include -I$(top_srcdir)/include diff --git a/storage/maria/ma_check.c b/storage/maria/ma_check.c index 4af0f955b8b..69d863e6366 100644 --- a/storage/maria/ma_check.c +++ b/storage/maria/ma_check.c @@ -359,7 +359,7 @@ int maria_chk_key(HA_CHECK *param, register MARIA_HA *info) puts("- check key delete-chain"); param->key_file_blocks=info->s->base.keystart; - for (key=0 ; key < info->s->state.header.max_block_size ; key++) + for (key=0 ; key < info->s->state.header.max_block_size_index ; key++) if (check_k_link(param,info,key)) { if (param->testflag & T_VERBOSE) puts(""); @@ -454,25 +454,24 @@ int maria_chk_key(HA_CHECK *param, register MARIA_HA *info) if ((uint) share->base.auto_key -1 == key) { /* Check that auto_increment key is bigger than max key value */ - ulonglong save_auto_value=info->s->state.auto_increment; - info->s->state.auto_increment=0; + ulonglong auto_increment; info->lastinx=key; _ma_read_key_record(info, 0L, info->rec_buff); - _ma_update_auto_increment(info, info->rec_buff); - if (info->s->state.auto_increment > save_auto_value) + auto_increment= ma_retrieve_auto_increment(info, info->rec_buff); + if (auto_increment > info->s->state.auto_increment) { _ma_check_print_warning(param, "Auto-increment value: %s is smaller than max used value: %s", - llstr(save_auto_value,buff2), - llstr(info->s->state.auto_increment, buff)); + llstr(info->s->state.auto_increment,buff2), + llstr(auto_increment, buff)); } if (param->testflag & T_AUTO_INC) { - set_if_bigger(info->s->state.auto_increment, - param->auto_increment_value); + set_if_bigger(info->s->state.auto_increment, + auto_increment); + set_if_bigger(info->s->state.auto_increment, + param->auto_increment_value); } - else - info->s->state.auto_increment=save_auto_value; /* Check that there isn't a row with auto_increment = 0 in the table */ maria_extra(info,HA_EXTRA_KEYREAD,0); @@ -1160,7 +1159,7 @@ int maria_chk_data_link(HA_CHECK *param, MARIA_HA *info,int extend) #ifdef HAVE_RTREE_KEYS (keyinfo->flag & HA_SPATIAL) ? maria_rtree_find_first(info, key, info->lastkey, key_length, - SEARCH_SAME) : + MBR_EQUAL | MBR_DATA) : #endif _ma_search(info,keyinfo,info->lastkey,key_length, SEARCH_SAME, info->s->state.key_root[key]); @@ -1412,7 +1411,7 @@ int maria_repair(HA_CHECK *param, register MARIA_HA *info, share->state.key_root[i]= HA_OFFSET_ERROR; /* Drop the delete chain. */ - for (i=0 ; i < share->state.header.max_block_size ; i++) + for (i=0 ; i < share->state.header.max_block_size_index ; i++) share->state.key_del[i]= HA_OFFSET_ERROR; /* @@ -1796,7 +1795,7 @@ int maria_sort_index(HA_CHECK *param, register MARIA_HA *info, my_string name) info->update= (short) (HA_STATE_CHANGED | HA_STATE_ROW_CHANGED); for (key=0 ; key < info->s->base.keys ; key++) info->s->state.key_root[key]=index_pos[key]; - for (key=0 ; key < info->s->state.header.max_block_size ; key++) + for (key=0 ; key < info->s->state.header.max_block_size_index ; key++) info->s->state.key_del[key]= HA_OFFSET_ERROR; info->s->state.changed&= ~STATE_NOT_SORTED_PAGES; @@ -2079,7 +2078,7 @@ int maria_repair_by_sort(HA_CHECK *param, register MARIA_HA *info, /* Clear the pointers to the given rows */ for (i=0 ; i < share->base.keys ; i++) share->state.key_root[i]= HA_OFFSET_ERROR; - for (i=0 ; i < share->state.header.max_block_size ; i++) + for (i=0 ; i < share->state.header.max_block_size_index ; i++) share->state.key_del[i]= HA_OFFSET_ERROR; info->state->key_file_length=share->base.keystart; } @@ -2101,6 +2100,7 @@ int maria_repair_by_sort(HA_CHECK *param, register MARIA_HA *info, my_seek(param->read_cache.file,0L,MY_SEEK_END,MYF(0)); sort_param.wordlist=NULL; + init_alloc_root(&sort_param.wordroot, FTPARSER_MEMROOT_ALLOC_SIZE, 0); if (share->data_file_type == DYNAMIC_RECORD) length=max(share->base.min_pack_length+1,share->base.min_block_length); @@ -2163,12 +2163,36 @@ int maria_repair_by_sort(HA_CHECK *param, register MARIA_HA *info, { uint ft_max_word_len_for_sort=FT_MAX_WORD_LEN_FOR_SORT* sort_param.keyinfo->seg->charset->mbmaxlen; - sort_info.max_records= - (ha_rows) (sort_info.filelength/ft_min_word_len+1); + sort_param.key_length+=ft_max_word_len_for_sort-HA_FT_MAXBYTELEN; + /* + fulltext indexes may have much more entries than the + number of rows in the table. We estimate the number here. + + Note, built-in parser is always nr. 0 - see ftparser_call_initializer() + */ + if (sort_param.keyinfo->ftparser_nr == 0) + { + /* + for built-in parser the number of generated index entries + cannot be larger than the size of the data file divided + by the minimal word's length + */ + sort_info.max_records= + (ha_rows) (sort_info.filelength/ft_min_word_len+1); + } + else + { + /* + for external plugin parser we cannot tell anything at all :( + so, we'll use all the sort memory and start from ~10 buffpeks. + (see _create_index_by_sort) + */ + sort_info.max_records= + 10*param->sort_buffer_length/sort_param.key_length; + } sort_param.key_read=sort_maria_ft_key_read; sort_param.key_write=sort_maria_ft_key_write; - sort_param.key_length+=ft_max_word_len_for_sort-HA_FT_MAXBYTELEN; } else { @@ -2184,6 +2208,7 @@ int maria_repair_by_sort(HA_CHECK *param, register MARIA_HA *info, goto err; } param->calc_checksum=0; /* No need to calc glob_crc */ + free_root(&sort_param.wordroot, MYF(0)); /* Set for next loop */ sort_info.max_records= (ha_rows) info->state->records; @@ -2447,7 +2472,7 @@ int maria_repair_parallel(HA_CHECK *param, register MARIA_HA *info, /* Clear the pointers to the given rows */ for (i=0 ; i < share->base.keys ; i++) share->state.key_root[i]= HA_OFFSET_ERROR; - for (i=0 ; i < share->state.header.max_block_size ; i++) + for (i=0 ; i < share->state.header.max_block_size_index ; i++) share->state.key_del[i]= HA_OFFSET_ERROR; info->state->key_file_length=share->base.keystart; } @@ -2573,6 +2598,7 @@ int maria_repair_parallel(HA_CHECK *param, register MARIA_HA *info, uint ft_max_word_len_for_sort=FT_MAX_WORD_LEN_FOR_SORT* sort_param[i].keyinfo->seg->charset->mbmaxlen; sort_param[i].key_length+=ft_max_word_len_for_sort-HA_FT_MAXBYTELEN; + init_alloc_root(&sort_param[i].wordroot, FTPARSER_MEMROOT_ALLOC_SIZE, 0); } } sort_info.total_keys=i; @@ -2794,10 +2820,12 @@ static int sort_maria_ft_key_read(MARIA_SORT_PARAM *sort_param, void *key) { for (;;) { - my_free((char*) wptr, MYF(MY_ALLOW_ZERO_PTR)); + free_root(&sort_param->wordroot, MYF(MY_MARK_BLOCKS_FREE)); if ((error=sort_get_next_record(sort_param))) DBUG_RETURN(error); - if (!(wptr= _ma_ft_parserecord(info,sort_param->key,sort_param->record))) + if (!(wptr= _ma_ft_parserecord(info,sort_param->key,sort_param->record, + &sort_param->wordroot))) + DBUG_RETURN(1); if (wptr->pos) break; @@ -2821,7 +2849,7 @@ static int sort_maria_ft_key_read(MARIA_SORT_PARAM *sort_param, void *key) #endif if (!wptr->pos) { - my_free((char*) sort_param->wordlist, MYF(0)); + free_root(&sort_param->wordroot, MYF(MY_MARK_BLOCKS_FREE)); sort_param->wordlist=0; error=_ma_sort_write_record(sort_param); } @@ -3784,6 +3812,7 @@ int maria_recreate_table(HA_CHECK *param, MARIA_HA **org_info, char *filename) ha_rows max_records; ulonglong file_length,tmp_length; MARIA_CREATE_INFO create_info; + DBUG_ENTER("maria_recreate_table"); error=1; /* Default error */ info= **org_info; @@ -3793,7 +3822,7 @@ int maria_recreate_table(HA_CHECK *param, MARIA_HA **org_info, char *filename) unpack= (share.options & HA_OPTION_COMPRESS_RECORD) && (param->testflag & T_UNPACK); if (!(keyinfo=(MARIA_KEYDEF*) my_alloca(sizeof(MARIA_KEYDEF)*share.base.keys))) - return 0; + DBUG_RETURN(0); memcpy((byte*) keyinfo,(byte*) share.keyinfo, (size_t) (sizeof(MARIA_KEYDEF)*share.base.keys)); @@ -3802,14 +3831,14 @@ int maria_recreate_table(HA_CHECK *param, MARIA_HA **org_info, char *filename) (key_parts+share.base.keys)))) { my_afree((gptr) keyinfo); - return 1; + DBUG_RETURN(1); } if (!(recdef=(MARIA_COLUMNDEF*) my_alloca(sizeof(MARIA_COLUMNDEF)*(share.base.fields+1)))) { my_afree((gptr) keyinfo); my_afree((gptr) keysegs); - return 1; + DBUG_RETURN(1); } if (!(uniquedef=(MARIA_UNIQUEDEF*) my_alloca(sizeof(MARIA_UNIQUEDEF)*(share.state.header.uniques+1)))) @@ -3817,7 +3846,7 @@ int maria_recreate_table(HA_CHECK *param, MARIA_HA **org_info, char *filename) my_afree((gptr) recdef); my_afree((gptr) keyinfo); my_afree((gptr) keysegs); - return 1; + DBUG_RETURN(1); } /* Copy the column definitions */ @@ -3887,6 +3916,11 @@ int maria_recreate_table(HA_CHECK *param, MARIA_HA **org_info, char *filename) create_info.language = (param->language ? param->language : share.state.header.language); create_info.key_file_length= status_info.key_file_length; + /* + Allow for creating an auto_increment key. This has an effect only if + an auto_increment key exists in the original table. + */ + create_info.with_auto_increment= TRUE; /* We don't have to handle symlinks here because we are using HA_DONT_TOUCH_DATA */ if (maria_create(filename, @@ -3931,7 +3965,7 @@ end: my_afree((gptr) keyinfo); my_afree((gptr) recdef); my_afree((gptr) keysegs); - return error; + DBUG_RETURN(error); } @@ -4034,6 +4068,8 @@ void _ma_update_auto_increment_key(HA_CHECK *param, MARIA_HA *info, my_bool repair_only) { byte *record; + DBUG_ENTER("update_auto_increment_key"); + if (!info->s->base.auto_key || ! maria_is_key_active(info->s->state.key_map, info->s->base.auto_key - 1)) { @@ -4041,7 +4077,7 @@ void _ma_update_auto_increment_key(HA_CHECK *param, MARIA_HA *info, _ma_check_print_info(param, "Table: %s doesn't have an auto increment key\n", param->isam_file_name); - return; + DBUG_VOID_RETURN; } if (!(param->testflag & T_SILENT) && !(param->testflag & T_REP)) @@ -4054,7 +4090,7 @@ void _ma_update_auto_increment_key(HA_CHECK *param, MARIA_HA *info, MYF(0)))) { _ma_check_print_error(param,"Not enough memory for extra record"); - return; + DBUG_VOID_RETURN; } maria_extra(info,HA_EXTRA_KEYREAD,0); @@ -4065,23 +4101,22 @@ void _ma_update_auto_increment_key(HA_CHECK *param, MARIA_HA *info, maria_extra(info,HA_EXTRA_NO_KEYREAD,0); my_free((char*) record, MYF(0)); _ma_check_print_error(param,"%d when reading last record",my_errno); - return; + DBUG_VOID_RETURN; } if (!repair_only) info->s->state.auto_increment=param->auto_increment_value; } else { - ulonglong auto_increment= (repair_only ? info->s->state.auto_increment : - param->auto_increment_value); - info->s->state.auto_increment=0; - _ma_update_auto_increment(info, record); + ulonglong auto_increment= ma_retrieve_auto_increment(info, record); set_if_bigger(info->s->state.auto_increment,auto_increment); + if (!repair_only) + set_if_bigger(info->s->state.auto_increment, param->auto_increment_value); } maria_extra(info,HA_EXTRA_NO_KEYREAD,0); my_free((char*) record, MYF(0)); maria_update_state_info(param, info, UPDATE_AUTO_INC); - return; + DBUG_VOID_RETURN; } diff --git a/storage/maria/ma_create.c b/storage/maria/ma_create.c index b15b5b0ae02..b9fb4eb0d5b 100644 --- a/storage/maria/ma_create.c +++ b/storage/maria/ma_create.c @@ -28,9 +28,9 @@ #endif #include - /* - ** Old options is used when recreating database, from isamchk - */ +/* + Old options is used when recreating database, from maria_chk +*/ int maria_create(const char *name,uint keys,MARIA_KEYDEF *keydefs, uint columns, MARIA_COLUMNDEF *recinfo, @@ -45,6 +45,7 @@ int maria_create(const char *name,uint keys,MARIA_KEYDEF *keydefs, key_length,info_length,key_segs,options,min_key_length_skip, base_pos,long_varchar_count,varchar_length, max_key_block_length,unique_key_parts,fulltext_keys,offset; + uint aligned_key_start, block_length; ulong reclength, real_reclength,min_pack_length; char filename[FN_REFLEN],linkname[FN_REFLEN], *linkname_ptr; ulong pack_reclength; @@ -59,6 +60,8 @@ int maria_create(const char *name,uint keys,MARIA_KEYDEF *keydefs, my_off_t key_root[HA_MAX_POSSIBLE_KEY],key_del[MARIA_MAX_KEY_BLOCK_SIZE]; MARIA_CREATE_INFO tmp_create_info; DBUG_ENTER("maria_create"); + DBUG_PRINT("enter", ("keys: %u columns: %u uniques: %u flags: %u", + keys, columns, uniques, flags)); if (!ci) { @@ -428,8 +431,16 @@ int maria_create(const char *name,uint keys,MARIA_KEYDEF *keydefs, key_segs) share.state.rec_per_key_part[key_segs-1]=1L; length+=key_length; + /* Get block length for key, if defined by user */ + block_length= (keydef->block_length ? + my_round_up_to_next_power(keydef->block_length) : + maria_block_size); + block_length= max(block_length, MARIA_MIN_KEY_BLOCK_LENGTH); + block_length= min(block_length, MARIA_MAX_KEY_BLOCK_LENGTH); + keydef->block_length= MARIA_BLOCK_SIZE(length-real_length_diff, - pointer,MARIA_MAX_KEYPTR_SIZE); + pointer,MARIA_MAX_KEYPTR_SIZE, + block_length); if (keydef->block_length > MARIA_MAX_KEY_BLOCK_LENGTH || length >= HA_MAX_KEY_BUFF) { @@ -474,6 +485,17 @@ int maria_create(const char *name,uint keys,MARIA_KEYDEF *keydefs, (key_segs + unique_key_parts)*HA_KEYSEG_SIZE+ columns*MARIA_COLUMNDEF_SIZE); + DBUG_PRINT("info", ("info_length: %u", info_length)); + /* There are only 16 bits for the total header length. */ + if (info_length > 65535) + { + my_printf_error(0, "Maria table '%s' has too many columns and/or " + "indexes and/or unique constraints.", + MYF(0), name + dirname_length(name)); + my_errno= HA_WRONG_CREATE_OPTION; + goto err; + } + bmove(share.state.header.file_version,(byte*) maria_file_magic,4); ci->old_options=options| (ci->old_options & HA_OPTION_TEMP_COMPRESS_RECORD ? HA_OPTION_COMPRESS_RECORD | @@ -485,7 +507,7 @@ int maria_create(const char *name,uint keys,MARIA_KEYDEF *keydefs, mi_int2store(share.state.header.base_pos,base_pos); share.state.header.language= (ci->language ? ci->language : default_charset_info->number); - share.state.header.max_block_size=max_key_block_length/MARIA_MIN_KEY_BLOCK_LENGTH; + share.state.header.max_block_size_index= max_key_block_length/MARIA_MIN_KEY_BLOCK_LENGTH; share.state.dellink = HA_OFFSET_ERROR; share.state.process= (ulong) getpid(); @@ -512,8 +534,12 @@ int maria_create(const char *name,uint keys,MARIA_KEYDEF *keydefs, mi_int2store(share.state.header.unique_key_parts,unique_key_parts); maria_set_all_keys_active(share.state.key_map, keys); + aligned_key_start= my_round_up_to_next_power(max_key_block_length ? + max_key_block_length : + maria_block_size); + share.base.keystart = share.state.state.key_file_length= - MY_ALIGN(info_length, maria_block_size); + MY_ALIGN(info_length, aligned_key_start); share.base.max_key_block_length=max_key_block_length; share.base.max_key_length=ALIGN_SIZE(max_key_length+4); share.base.records=ci->max_rows; @@ -549,9 +575,21 @@ int maria_create(const char *name,uint keys,MARIA_KEYDEF *keydefs, { char *iext= strrchr(ci->index_file_name, '.'); int have_iext= iext && !strcmp(iext, MARIA_NAME_IEXT); - - fn_format(filename, ci->index_file_name, "", MARIA_NAME_IEXT, - MY_UNPACK_FILENAME| (have_iext ? MY_REPLACE_EXT :MY_APPEND_EXT)); + if (options & HA_OPTION_TMP_TABLE) + { + char *path; + /* chop off the table name, tempory tables use generated name */ + if ((path= strrchr(ci->index_file_name, FN_LIBCHAR))) + *path= '\0'; + fn_format(filename, name, ci->index_file_name, MARIA_NAME_IEXT, + MY_REPLACE_DIR | MY_UNPACK_FILENAME | MY_APPEND_EXT); + } + else + { + fn_format(filename, ci->index_file_name, "", MARIA_NAME_IEXT, + MY_UNPACK_FILENAME | (have_iext ? MY_REPLACE_EXT : + MY_APPEND_EXT)); + } fn_format(linkname, name, "", MARIA_NAME_IEXT, MY_UNPACK_FILENAME|MY_APPEND_EXT); linkname_ptr=linkname; @@ -614,9 +652,21 @@ int maria_create(const char *name,uint keys,MARIA_KEYDEF *keydefs, char *dext= strrchr(ci->data_file_name, '.'); int have_dext= dext && !strcmp(dext, MARIA_NAME_DEXT); - fn_format(filename, ci->data_file_name, "", MARIA_NAME_DEXT, - MY_UNPACK_FILENAME | - (have_dext ? MY_REPLACE_EXT : MY_APPEND_EXT)); + if (options & HA_OPTION_TMP_TABLE) + { + char *path; + /* chop off the table name, tempory tables use generated name */ + if ((path= strrchr(ci->data_file_name, FN_LIBCHAR))) + *path= '\0'; + fn_format(filename, name, ci->data_file_name, MARIA_NAME_DEXT, + MY_REPLACE_DIR | MY_UNPACK_FILENAME | MY_APPEND_EXT); + } + else + { + fn_format(filename, ci->data_file_name, "", MARIA_NAME_DEXT, + MY_UNPACK_FILENAME | + (have_dext ? MY_REPLACE_EXT : MY_APPEND_EXT)); + } fn_format(linkname, name, "",MARIA_NAME_DEXT, MY_UNPACK_FILENAME | MY_APPEND_EXT); linkname_ptr=linkname; @@ -636,7 +686,7 @@ int maria_create(const char *name,uint keys,MARIA_KEYDEF *keydefs, } errpos=3; } - + DBUG_PRINT("info", ("write state info and base info")); if (_ma_state_info_write(file, &share.state, 2) || _ma_base_info_write(file, &share.base)) goto err; @@ -650,6 +700,7 @@ int maria_create(const char *name,uint keys,MARIA_KEYDEF *keydefs, #endif /* Write key and keyseg definitions */ + DBUG_PRINT("info", ("write key and keyseg definitions")); for (i=0 ; i < share.base.keys - uniques; i++) { uint sp_segs=(keydefs[i].flag & HA_SPATIAL) ? 2*SPDIMS : 0; @@ -700,6 +751,7 @@ int maria_create(const char *name,uint keys,MARIA_KEYDEF *keydefs, } /* Save unique definition */ + DBUG_PRINT("info", ("write unique definitions")); for (i=0 ; i < share.state.header.uniques ; i++) { HA_KEYSEG *keyseg_end; @@ -730,6 +782,7 @@ int maria_create(const char *name,uint keys,MARIA_KEYDEF *keydefs, goto err; } } + DBUG_PRINT("info", ("write field definitions")); for (i=0 ; i < share.base.fields ; i++) if (_ma_recinfo_write(file, &recinfo[i])) goto err; @@ -744,6 +797,7 @@ int maria_create(const char *name,uint keys,MARIA_KEYDEF *keydefs, #endif /* Enlarge files */ + DBUG_PRINT("info", ("enlarge to keystart: %lu", (ulong) share.base.keystart)); if (my_chsize(file,(ulong) share.base.keystart,0,MYF(0))) goto err; @@ -772,7 +826,7 @@ err: VOID(my_close(dfile,MYF(0))); /* fall through */ case 2: - /* QQ: Tõnu should add a call to my_raid_delete() here */ + /* QQ: Tõnu should add a call to my_raid_delete() here */ if (! (flags & HA_DONT_TOUCH_DATA)) my_delete_with_symlink(fn_format(filename,name,"",MARIA_NAME_DEXT, MY_UNPACK_FILENAME | MY_APPEND_EXT), diff --git a/storage/maria/ma_delete_table.c b/storage/maria/ma_delete_table.c index a9af9a62c99..dd781a93fc4 100644 --- a/storage/maria/ma_delete_table.c +++ b/storage/maria/ma_delete_table.c @@ -34,12 +34,24 @@ int maria_delete_table(const char *name) #ifdef USE_RAID { MARIA_HA *info; - /* we use 'open_for_repair' to be able to delete a crashed table */ - if (!(info=maria_open(name, O_RDONLY, HA_OPEN_FOR_REPAIR))) - DBUG_RETURN(my_errno); - raid_type = info->s->base.raid_type; - raid_chunks = info->s->base.raid_chunks; - maria_close(info); + /* + When built with RAID support, we need to determine if this table + makes use of the raid feature. If yes, we need to remove all raid + chunks. This is done with my_raid_delete(). Unfortunately it is + necessary to open the table just to check this. We use + 'open_for_repair' to be able to open even a crashed table. If even + this open fails, we assume no raid configuration for this table + and try to remove the normal data file only. This may however + leave the raid chunks behind. + */ + if (!(info= maria_open(name, O_RDONLY, HA_OPEN_FOR_REPAIR))) + raid_type= 0; + else + { + raid_type= info->s->base.raid_type; + raid_chunks= info->s->base.raid_chunks; + maria_close(info); + } } #ifdef EXTRA_DEBUG _ma_check_table_is_closed(name,"delete"); diff --git a/storage/maria/ma_dynrec.c b/storage/maria/ma_dynrec.c index 18efc0adbd0..047826408c3 100644 --- a/storage/maria/ma_dynrec.c +++ b/storage/maria/ma_dynrec.c @@ -67,6 +67,11 @@ static int _ma_cmp_buffer(File file, const byte *buff, my_off_t filepos, my_bool _ma_dynmap_file(MARIA_HA *info, my_off_t size) { DBUG_ENTER("_ma_dynmap_file"); + if (size > (my_off_t) (~((size_t) 0)) - MEMMAP_EXTRA_MARGIN) + { + DBUG_PRINT("warning", ("File is too large for mmap")); + DBUG_RETURN(1); + } info->s->file_map= (byte*) my_mmap(0, (size_t)(size + MEMMAP_EXTRA_MARGIN), info->s->mode==O_RDONLY ? PROT_READ : @@ -1324,6 +1329,9 @@ int _ma_read_dynamic_record(MARIA_HA *info, my_off_t filepos, byte *buf) info->rec_cache.pos_in_file <= block_info.next_filepos && flush_io_cache(&info->rec_cache)) goto err; + /* A corrupted table can have wrong pointers. (Bug# 19835) */ + if (block_info.next_filepos == HA_OFFSET_ERROR) + goto panic; info->rec_cache.seek_not_done=1; if ((b_type= _ma_get_block_info(&block_info,file, block_info.next_filepos)) diff --git a/storage/maria/ma_extra.c b/storage/maria/ma_extra.c index 06a36cba238..d600fedb99b 100644 --- a/storage/maria/ma_extra.c +++ b/storage/maria/ma_extra.c @@ -47,29 +47,6 @@ int maria_extra(MARIA_HA *info, enum ha_extra_function function, void *extra_arg DBUG_PRINT("enter",("function: %d",(int) function)); switch (function) { - case HA_EXTRA_RESET: - /* - Free buffers and reset the following flags: - EXTRA_CACHE, EXTRA_WRITE_CACHE, EXTRA_KEYREAD, EXTRA_QUICK - - If the row buffer cache is large (for dynamic tables), reduce it - to save memory. - */ - if (info->opt_flag & (READ_CACHE_USED | WRITE_CACHE_USED)) - { - info->opt_flag&= ~(READ_CACHE_USED | WRITE_CACHE_USED); - error=end_io_cache(&info->rec_cache); - } - if (share->base.blobs) - _ma_alloc_rec_buff(info, -1, &info->rec_buff); -#if defined(HAVE_MMAP) && defined(HAVE_MADVISE) - if (info->opt_flag & MEMMAP_USED) - madvise(share->file_map,share->state.state.data_file_length,MADV_RANDOM); -#endif - info->opt_flag&= ~(KEY_READ_USED | REMEMBER_OLD_POS); - info->quick_mode=0; - /* Fall through */ - case HA_EXTRA_RESET_STATE: /* Reset state (don't free buffers) */ info->lastinx= 0; /* Use first index as def */ info->last_search_keypage=info->lastpos= HA_OFFSET_ERROR; @@ -423,3 +400,37 @@ static void maria_extra_keyflag(MARIA_HA *info, enum ha_extra_function function) } } } + + +int maria_reset(MARIA_HA *info) +{ + int error= 0; + MARIA_SHARE *share=info->s; + DBUG_ENTER("maria_reset"); + /* + Free buffers and reset the following flags: + EXTRA_CACHE, EXTRA_WRITE_CACHE, EXTRA_KEYREAD, EXTRA_QUICK + + If the row buffer cache is large (for dynamic tables), reduce it + to save memory. + */ + if (info->opt_flag & (READ_CACHE_USED | WRITE_CACHE_USED)) + { + info->opt_flag&= ~(READ_CACHE_USED | WRITE_CACHE_USED); + error= end_io_cache(&info->rec_cache); + } + if (share->base.blobs) + _ma_alloc_rec_buff(info, -1, &info->rec_buff); +#if defined(HAVE_MMAP) && defined(HAVE_MADVISE) + if (info->opt_flag & MEMMAP_USED) + madvise(share->file_map,share->state.state.data_file_length,MADV_RANDOM); +#endif + info->opt_flag&= ~(KEY_READ_USED | REMEMBER_OLD_POS); + info->quick_mode=0; + info->lastinx= 0; /* Use first index as def */ + info->last_search_keypage= info->lastpos= HA_OFFSET_ERROR; + info->page_changed= 1; + info->update= ((info->update & HA_STATE_CHANGED) | HA_STATE_NEXT_FOUND | + HA_STATE_PREV_FOUND); + DBUG_RETURN(error); +} diff --git a/storage/maria/ma_ft_boolean_search.c b/storage/maria/ma_ft_boolean_search.c index 2b8e0d8b97a..83901cb5e47 100644 --- a/storage/maria/ma_ft_boolean_search.c +++ b/storage/maria/ma_ft_boolean_search.c @@ -167,10 +167,11 @@ typedef struct st_my_ftb_param } MY_FTB_PARAM; -static int ftb_query_add_word(void *param, char *word, int word_len, +static int ftb_query_add_word(MYSQL_FTPARSER_PARAM *param, + char *word, int word_len, MYSQL_FTPARSER_BOOLEAN_INFO *info) { - MY_FTB_PARAM *ftb_param= (MY_FTB_PARAM *)param; + MY_FTB_PARAM *ftb_param= param->mysql_ftparam; FTB_WORD *ftbw; FTB_EXPR *ftbe, *tmp_expr; FT_WORD *phrase_word; @@ -268,9 +269,10 @@ static int ftb_query_add_word(void *param, char *word, int word_len, } -static int ftb_parse_query_internal(void *param, char *query, int len) +static int ftb_parse_query_internal(MYSQL_FTPARSER_PARAM *param, + char *query, int len) { - MY_FTB_PARAM *ftb_param= (MY_FTB_PARAM *)param; + MY_FTB_PARAM *ftb_param= param->mysql_ftparam; MYSQL_FTPARSER_BOOLEAN_INFO info; CHARSET_INFO *cs= ftb_param->ftb->charset; char **start= &query; @@ -280,7 +282,7 @@ static int ftb_parse_query_internal(void *param, char *query, int len) info.prev= ' '; info.quot= 0; while (maria_ft_get_word(cs, start, end, &w, &info)) - ftb_query_add_word(param, w.pos, w.len, &info); + param->mysql_add_word(param, w.pos, w.len, &info); return(0); } @@ -295,20 +297,21 @@ static void _ftb_parse_query(FTB *ftb, byte *query, uint len, if (ftb->state != UNINITIALIZED) DBUG_VOID_RETURN; + if (! (param= maria_ftparser_call_initializer(ftb->info, ftb->keynr, 0))) + DBUG_VOID_RETURN; ftb_param.ftb= ftb; ftb_param.depth= 0; ftb_param.ftbe= ftb->root; ftb_param.up_quot= 0; - if (! (param= maria_ftparser_call_initializer(ftb->info, ftb->keynr))) - DBUG_VOID_RETURN; param->mysql_parse= ftb_parse_query_internal; param->mysql_add_word= ftb_query_add_word; param->mysql_ftparam= (void *)&ftb_param; param->cs= ftb->charset; param->doc= query; param->length= len; + param->flags= 0; param->mode= MYSQL_FTPARSER_FULL_BOOLEAN_INFO; parser->parse(param); DBUG_VOID_RETURN; @@ -577,10 +580,11 @@ typedef struct st_my_ftb_phrase_param } MY_FTB_PHRASE_PARAM; -static int ftb_phrase_add_word(void *param, char *word, int word_len, +static int ftb_phrase_add_word(MYSQL_FTPARSER_PARAM *param, + char *word, int word_len, MYSQL_FTPARSER_BOOLEAN_INFO *boolean_info __attribute__((unused))) { - MY_FTB_PHRASE_PARAM *phrase_param= (MY_FTB_PHRASE_PARAM *)param; + MY_FTB_PHRASE_PARAM *phrase_param= param->mysql_ftparam; FT_WORD *w= (FT_WORD *)phrase_param->document->data; LIST *phrase, *document; w->pos= word; @@ -608,14 +612,15 @@ static int ftb_phrase_add_word(void *param, char *word, int word_len, } -static int ftb_check_phrase_internal(void *param, char *document, int len) +static int ftb_check_phrase_internal(MYSQL_FTPARSER_PARAM *param, + char *document, int len) { FT_WORD word; - MY_FTB_PHRASE_PARAM *phrase_param= (MY_FTB_PHRASE_PARAM *)param; + MY_FTB_PHRASE_PARAM *phrase_param= param->mysql_ftparam; const char *docend= document + len; while (maria_ft_simple_get_word(phrase_param->cs, &document, docend, &word, FALSE)) { - ftb_phrase_add_word(param, word.pos, word.len, 0); + param->mysql_add_word(param, word.pos, word.len, 0); if (phrase_param->match) return 1; } @@ -644,7 +649,8 @@ static int _ftb_check_phrase(FTB *ftb, const byte *document, uint len, MYSQL_FTPARSER_PARAM *param; DBUG_ENTER("_ftb_check_phrase"); DBUG_ASSERT(parser); - if (! (param= maria_ftparser_call_initializer(ftb->info, ftb->keynr))) + + if (! (param= maria_ftparser_call_initializer(ftb->info, ftb->keynr, 1))) DBUG_RETURN(0); ftb_param.phrase= ftbe->phrase; ftb_param.document= ftbe->document; @@ -659,6 +665,7 @@ static int _ftb_check_phrase(FTB *ftb, const byte *document, uint len, param->cs= ftb->charset; param->doc= (byte *)document; param->length= len; + param->flags= 0; param->mode= MYSQL_FTPARSER_WITH_STOPWORDS; parser->parse(param); DBUG_RETURN(ftb_param.match ? 1 : 0); @@ -819,10 +826,11 @@ typedef struct st_my_ftb_find_param } MY_FTB_FIND_PARAM; -static int ftb_find_relevance_add_word(void *param, char *word, int len, +static int ftb_find_relevance_add_word(MYSQL_FTPARSER_PARAM *param, + char *word, int len, MYSQL_FTPARSER_BOOLEAN_INFO *boolean_info __attribute__((unused))) { - MY_FTB_FIND_PARAM *ftb_param= (MY_FTB_FIND_PARAM *)param; + MY_FTB_FIND_PARAM *ftb_param= param->mysql_ftparam; FT_INFO *ftb= ftb_param->ftb; FTB_WORD *ftbw; int a, b, c; @@ -852,13 +860,15 @@ static int ftb_find_relevance_add_word(void *param, char *word, int len, } -static int ftb_find_relevance_parse(void *param, char *doc, int len) +static int ftb_find_relevance_parse(MYSQL_FTPARSER_PARAM *param, + char *doc, int len) { - FT_INFO *ftb= ((MY_FTB_FIND_PARAM *)param)->ftb; + MY_FTB_FIND_PARAM *ftb_param= param->mysql_ftparam; + FT_INFO *ftb= ftb_param->ftb; char *end= doc + len; FT_WORD w; while (maria_ft_simple_get_word(ftb->charset, &doc, end, &w, TRUE)) - ftb_find_relevance_add_word(param, w.pos, w.len, 0); + param->mysql_add_word(param, w.pos, w.len, 0); return(0); } @@ -878,7 +888,7 @@ float maria_ft_boolean_find_relevance(FT_INFO *ftb, byte *record, uint length) return -2.0; if (!ftb->queue.elements) return 0; - if (! (param= maria_ftparser_call_initializer(ftb->info, ftb->keynr))) + if (! (param= maria_ftparser_call_initializer(ftb->info, ftb->keynr, 0))) return 0; if (ftb->state != INDEX_SEARCH && docid <= ftb->lastpos) @@ -904,17 +914,17 @@ float maria_ft_boolean_find_relevance(FT_INFO *ftb, byte *record, uint length) ftb_param.ftb= ftb; ftb_param.ftsi= &ftsi2; + param->mysql_parse= ftb_find_relevance_parse; + param->mysql_add_word= ftb_find_relevance_add_word; + param->mysql_ftparam= (void *)&ftb_param; + param->flags= 0; + param->cs= ftb->charset; + param->mode= MYSQL_FTPARSER_SIMPLE_MODE; + while (_ma_ft_segiterator(&ftsi)) { if (!ftsi.pos) continue; - /* Since subsequent call to _ftb_check_phrase overwrites param elements, - it must be reinitialized at each iteration _inside_ the loop. */ - param->mysql_parse= ftb_find_relevance_parse; - param->mysql_add_word= ftb_find_relevance_add_word; - param->mysql_ftparam= (void *)&ftb_param; - param->cs= ftb->charset; - param->mode= MYSQL_FTPARSER_SIMPLE_MODE; param->doc= (byte *)ftsi.pos; param->length= ftsi.len; parser->parse(param); diff --git a/storage/maria/ma_ft_eval.c b/storage/maria/ma_ft_eval.c index b9b496fc268..fe4900aeb64 100644 --- a/storage/maria/ma_ft_eval.c +++ b/storage/maria/ma_ft_eval.c @@ -55,6 +55,7 @@ int main(int argc, char *argv[]) /* Define a key over the first column */ keyinfo[0].seg=keyseg; keyinfo[0].keysegs=1; + keyinfo[0].block_length= 0; /* Default block length */ keyinfo[0].seg[0].type= HA_KEYTYPE_TEXT; keyinfo[0].seg[0].flag= HA_BLOB_PART; keyinfo[0].seg[0].start=recinfo[0].length; diff --git a/storage/maria/ma_ft_nlq_search.c b/storage/maria/ma_ft_nlq_search.c index a9741787fc9..993857aecbb 100644 --- a/storage/maria/ma_ft_nlq_search.c +++ b/storage/maria/ma_ft_nlq_search.c @@ -226,7 +226,7 @@ FT_INFO *maria_ft_init_nlq_search(MARIA_HA *info, uint keynr, byte *query, aio.charset=info->s->keyinfo[keynr].seg->charset; aio.keybuff=info->lastkey+info->s->base.max_key_length; parser= info->s->keyinfo[keynr].parser; - if (! (ftparser_param= maria_ftparser_call_initializer(info, keynr))) + if (! (ftparser_param= maria_ftparser_call_initializer(info, keynr, 0))) goto err; bzero(&wtree,sizeof(wtree)); @@ -235,7 +235,9 @@ FT_INFO *maria_ft_init_nlq_search(MARIA_HA *info, uint keynr, byte *query, NULL, NULL); maria_ft_parse_init(&wtree, aio.charset); - if (maria_ft_parse(&wtree, query, query_len, 0, parser, ftparser_param)) + ftparser_param->flags= 0; + if (maria_ft_parse(&wtree, query, query_len, parser, ftparser_param, + &wtree.mem_root)) goto err; if (tree_walk(&wtree, (tree_walk_action)&walk_and_match, &aio, @@ -255,7 +257,9 @@ FT_INFO *maria_ft_init_nlq_search(MARIA_HA *info, uint keynr, byte *query, if (!(*info->read_record)(info,docid,record)) { info->update|= HA_STATE_AKTIV; - _ma_ft_parse(&wtree, info, keynr, record, 1, ftparser_param); + ftparser_param->flags= MYSQL_FTFLAGS_NEED_COPY; + _ma_ft_parse(&wtree, info, keynr, record, ftparser_param, + &wtree.mem_root); } } delete_queue(&best); diff --git a/storage/maria/ma_ft_parser.c b/storage/maria/ma_ft_parser.c index 983bebf3562..1c6e0267d53 100644 --- a/storage/maria/ma_ft_parser.c +++ b/storage/maria/ma_ft_parser.c @@ -28,7 +28,7 @@ typedef struct st_maria_ft_docstat { typedef struct st_my_maria_ft_parser_param { TREE *wtree; - my_bool with_alloc; + MEM_ROOT *mem_root; } MY_FT_PARSER_PARAM; @@ -48,14 +48,14 @@ static int walk_and_copy(FT_WORD *word,uint32 count,FT_DOCSTAT *docstat) /* transforms tree of words into the array, applying normalization */ -FT_WORD * maria_ft_linearize(TREE *wtree) +FT_WORD * maria_ft_linearize(TREE *wtree, MEM_ROOT *mem_root) { FT_WORD *wlist,*p; FT_DOCSTAT docstat; DBUG_ENTER("maria_ft_linearize"); - if ((wlist=(FT_WORD *) my_malloc(sizeof(FT_WORD)* - (1+wtree->elements_in_tree),MYF(0)))) + if ((wlist=(FT_WORD *) alloc_root(mem_root, sizeof(FT_WORD)* + (1+wtree->elements_in_tree)))) { docstat.list=wlist; docstat.uniq=wtree->elements_in_tree; @@ -114,6 +114,7 @@ byte maria_ft_get_word(CHARSET_INFO *cs, byte **start, byte *end, FT_WORD *word, MYSQL_FTPARSER_BOOLEAN_INFO *param) { byte *doc=*start; + int ctype; uint mwc, length, mbl; param->yesno=(FTB_YES==' ') ? 1 : (param->quot != 0); @@ -122,9 +123,11 @@ byte maria_ft_get_word(CHARSET_INFO *cs, byte **start, byte *end, while (doc 0 ? mbl : 1)) { - if (true_word_char(cs,*doc)) break; + mbl= cs->cset->ctype(cs, &ctype, (uchar*)doc, (uchar*)end); + if (true_word_char(ctype, *doc)) + break; if (*doc == FTB_RQUOT && param->quot) { param->quot=doc; @@ -158,14 +161,16 @@ byte maria_ft_get_word(CHARSET_INFO *cs, byte **start, byte *end, } mwc=length=0; - for (word->pos=doc; docpos= doc; doc < end; length++, doc+= (mbl > 0 ? mbl : 1)) + { + mbl= cs->cset->ctype(cs, &ctype, (uchar*)doc, (uchar*)end); + if (true_word_char(ctype, *doc)) mwc=0; else if (!misc_word_char(*doc) || mwc) break; else mwc++; - + } param->prev='A'; /* be sure *prev is true_word_char */ word->len= (uint)(doc-word->pos) - mwc; if ((param->trunc=(doc 0 ? mbl : 1)) { - if (doc >= end) DBUG_RETURN(0); - if (true_word_char(cs, *doc)) break; + if (doc >= end) + DBUG_RETURN(0); + mbl= cs->cset->ctype(cs, &ctype, (uchar*)doc, (uchar*)end); + if (true_word_char(ctype, *doc)) + break; } mwc= length= 0; - for (word->pos=doc; docpos= doc; doc < end; length++, doc+= (mbl > 0 ? mbl : 1)) + { + mbl= cs->cset->ctype(cs, &ctype, (uchar*)doc, (uchar*)end); + if (true_word_char(ctype, *doc)) mwc= 0; else if (!misc_word_char(*doc) || mwc) break; else mwc++; + } word->len= (uint)(doc-word->pos) - mwc; @@ -241,19 +253,20 @@ void maria_ft_parse_init(TREE *wtree, CHARSET_INFO *cs) } -static int maria_ft_add_word(void *param, byte *word, uint word_len, +static int maria_ft_add_word(MYSQL_FTPARSER_PARAM *param, + char *word, int word_len, MYSQL_FTPARSER_BOOLEAN_INFO *boolean_info __attribute__((unused))) { TREE *wtree; FT_WORD w; + MY_FT_PARSER_PARAM *ft_param=param->mysql_ftparam; DBUG_ENTER("maria_ft_add_word"); - wtree= ((MY_FT_PARSER_PARAM *)param)->wtree; - if (((MY_FT_PARSER_PARAM *)param)->with_alloc) + wtree= ft_param->wtree; + if (param->flags & MYSQL_FTFLAGS_NEED_COPY) { byte *ptr; - /* allocating the data in the tree - to avoid mallocs and frees */ DBUG_ASSERT(wtree->with_delete == 0); - ptr= (byte *)alloc_root(&wtree->mem_root, word_len); + ptr= (byte *)alloc_root(ft_param->mem_root, word_len); memcpy(ptr, word, word_len); w.pos= ptr; } @@ -269,30 +282,31 @@ static int maria_ft_add_word(void *param, byte *word, uint word_len, } -static int maria_ft_parse_internal(void *param, byte *doc, uint doc_len) +static int maria_ft_parse_internal(MYSQL_FTPARSER_PARAM *param, + byte *doc, int doc_len) { byte *end=doc+doc_len; + MY_FT_PARSER_PARAM *ft_param=param->mysql_ftparam; + TREE *wtree= ft_param->wtree; FT_WORD w; - TREE *wtree; DBUG_ENTER("maria_ft_parse_internal"); - wtree= ((MY_FT_PARSER_PARAM *)param)->wtree; while (maria_ft_simple_get_word(wtree->custom_arg, &doc, end, &w, TRUE)) - if (maria_ft_add_word(param, w.pos, w.len, 0)) + if (param->mysql_add_word(param, w.pos, w.len, 0)) DBUG_RETURN(1); DBUG_RETURN(0); } -int maria_ft_parse(TREE *wtree, byte *doc, int doclen, my_bool with_alloc, +int maria_ft_parse(TREE *wtree, byte *doc, int doclen, struct st_mysql_ftparser *parser, - MYSQL_FTPARSER_PARAM *param) + MYSQL_FTPARSER_PARAM *param, MEM_ROOT *mem_root) { MY_FT_PARSER_PARAM my_param; DBUG_ENTER("maria_ft_parse"); DBUG_ASSERT(parser); my_param.wtree= wtree; - my_param.with_alloc= with_alloc; + my_param.mem_root= mem_root; param->mysql_parse= maria_ft_parse_internal; param->mysql_add_word= maria_ft_add_word; @@ -305,7 +319,9 @@ int maria_ft_parse(TREE *wtree, byte *doc, int doclen, my_bool with_alloc, } -MYSQL_FTPARSER_PARAM *maria_ftparser_call_initializer(MARIA_HA *info, uint keynr) +#define MAX_PARAM_NR 2 +MYSQL_FTPARSER_PARAM *maria_ftparser_call_initializer(MARIA_HA *info, + uint keynr, uint paramnr) { uint32 ftparser_nr; struct st_mysql_ftparser *parser; @@ -344,9 +360,16 @@ MYSQL_FTPARSER_PARAM *maria_ftparser_call_initializer(MARIA_HA *info, uint keynr } info->s->ftparsers= ftparsers; } + /* + We have to allocate two MYSQL_FTPARSER_PARAM structures per plugin + because in a boolean search a parser is called recursively + ftb_find_relevance* calls ftb_check_phrase* + (MAX_PARAM_NR=2) + */ info->ftparser_param= (MYSQL_FTPARSER_PARAM *) - my_malloc(sizeof(MYSQL_FTPARSER_PARAM) * + my_malloc(MAX_PARAM_NR * sizeof(MYSQL_FTPARSER_PARAM) * info->s->ftparsers, MYF(MY_WME|MY_ZEROFILL)); + init_alloc_root(&info->ft_memroot, FTPARSER_MEMROOT_ALLOC_SIZE, 0); if (! info->ftparser_param) return 0; } @@ -360,6 +383,8 @@ MYSQL_FTPARSER_PARAM *maria_ftparser_call_initializer(MARIA_HA *info, uint keynr ftparser_nr= info->s->keyinfo[keynr].ftparser_nr; parser= info->s->keyinfo[keynr].parser; } + DBUG_ASSERT(paramnr < MAX_PARAM_NR); + ftparser_nr= ftparser_nr*MAX_PARAM_NR + paramnr; if (! info->ftparser_param[ftparser_nr].mysql_add_word) { /* Note, that mysql_add_word is used here as a flag: @@ -376,19 +401,25 @@ MYSQL_FTPARSER_PARAM *maria_ftparser_call_initializer(MARIA_HA *info, uint keynr void maria_ftparser_call_deinitializer(MARIA_HA *info) { - uint i, keys= info->s->state.header.keys; + uint i, j, keys= info->s->state.header.keys; + free_root(&info->ft_memroot, MYF(0)); if (! info->ftparser_param) return; for (i= 0; i < keys; i++) { MARIA_KEYDEF *keyinfo= &info->s->keyinfo[i]; - MYSQL_FTPARSER_PARAM *ftparser_param= - &info->ftparser_param[keyinfo->ftparser_nr]; - if (keyinfo->flag & HA_FULLTEXT && ftparser_param->mysql_add_word) + for (j=0; j < MAX_PARAM_NR; j++) { - if (keyinfo->parser->deinit) - keyinfo->parser->deinit(ftparser_param); - ftparser_param->mysql_add_word= 0; + MYSQL_FTPARSER_PARAM *ftparser_param= + &info->ftparser_param[keyinfo->ftparser_nr*MAX_PARAM_NR + j]; + if (keyinfo->flag & HA_FULLTEXT && ftparser_param->mysql_add_word) + { + if (keyinfo->parser->deinit) + keyinfo->parser->deinit(ftparser_param); + ftparser_param->mysql_add_word= 0; + } + else + break; } } } diff --git a/storage/maria/ma_ft_test1.c b/storage/maria/ma_ft_test1.c index 2880f6bcdc1..595c31e774c 100644 --- a/storage/maria/ma_ft_test1.c +++ b/storage/maria/ma_ft_test1.c @@ -90,6 +90,7 @@ static int run_test(const char *filename) /* Define a key over the first column */ keyinfo[0].seg=keyseg; keyinfo[0].keysegs=1; + keyinfo[0].block_length= 0; /* Default block length */ keyinfo[0].seg[0].type= key_type; keyinfo[0].seg[0].flag= (key_field == FIELD_BLOB) ? HA_BLOB_PART: (key_field == FIELD_VARCHAR) ? HA_VAR_LENGTH_PART:0; diff --git a/storage/maria/ma_ft_update.c b/storage/maria/ma_ft_update.c index c9e2112578c..965f9afc91d 100644 --- a/storage/maria/ma_ft_update.c +++ b/storage/maria/ma_ft_update.c @@ -95,9 +95,8 @@ uint _ma_ft_segiterator(register FT_SEG_ITERATOR *ftsi) /* parses a document i.e. calls maria_ft_parse for every keyseg */ -uint _ma_ft_parse(TREE *parsed, MARIA_HA *info, uint keynr, - const byte *record, my_bool with_alloc, - MYSQL_FTPARSER_PARAM *param) +uint _ma_ft_parse(TREE *parsed, MARIA_HA *info, uint keynr, const byte *record, + MYSQL_FTPARSER_PARAM *param, MEM_ROOT *mem_root) { FT_SEG_ITERATOR ftsi; struct st_mysql_ftparser *parser; @@ -110,25 +109,27 @@ uint _ma_ft_parse(TREE *parsed, MARIA_HA *info, uint keynr, while (_ma_ft_segiterator(&ftsi)) { if (ftsi.pos) - if (maria_ft_parse(parsed, (byte *)ftsi.pos, ftsi.len, with_alloc, parser, - param)) + if (maria_ft_parse(parsed, (byte *)ftsi.pos, ftsi.len, parser, param, + mem_root)) DBUG_RETURN(1); } DBUG_RETURN(0); } -FT_WORD * _ma_ft_parserecord(MARIA_HA *info, uint keynr, const byte *record) +FT_WORD * _ma_ft_parserecord(MARIA_HA *info, uint keynr, const byte *record, + MEM_ROOT *mem_root) { TREE ptree; MYSQL_FTPARSER_PARAM *param; DBUG_ENTER("_ma_ft_parserecord"); - if (! (param= maria_ftparser_call_initializer(info, keynr))) + if (! (param= maria_ftparser_call_initializer(info, keynr, 0))) DBUG_RETURN(NULL); bzero((char*) &ptree, sizeof(ptree)); - if (_ma_ft_parse(&ptree, info, keynr, record, 0, param)) + param->flags= 0; + if (_ma_ft_parse(&ptree, info, keynr, record, param, mem_root)) DBUG_RETURN(NULL); - DBUG_RETURN(maria_ft_linearize(&ptree)); + DBUG_RETURN(maria_ft_linearize(&ptree, mem_root)); } static int _ma_ft_store(MARIA_HA *info, uint keynr, byte *keybuf, @@ -174,10 +175,6 @@ int _ma_ft_cmp(MARIA_HA *info, uint keynr, const byte *rec1, const byte *rec2) FT_SEG_ITERATOR ftsi1, ftsi2; CHARSET_INFO *cs=info->s->keyinfo[keynr].seg->charset; DBUG_ENTER("_ma_ft_cmp"); -#ifndef MYSQL_HAS_TRUE_CTYPE_IMPLEMENTATION - if (cs->mbmaxlen > 1) - DBUG_RETURN(THOSE_TWO_DAMN_KEYS_ARE_REALLY_DIFFERENT); -#endif _ma_ft_segiterator_init(info, keynr, rec1, &ftsi1); _ma_ft_segiterator_init(info, keynr, rec2, &ftsi2); @@ -206,10 +203,11 @@ int _ma_ft_update(MARIA_HA *info, uint keynr, byte *keybuf, int cmp, cmp2; DBUG_ENTER("_ma_ft_update"); - if (!(old_word=oldlist= _ma_ft_parserecord(info, keynr, oldrec))) - goto err0; - if (!(new_word=newlist= _ma_ft_parserecord(info, keynr, newrec))) - goto err1; + if (!(old_word=oldlist=_ma_ft_parserecord(info, keynr, oldrec, + &info->ft_memroot)) || + !(new_word=newlist=_ma_ft_parserecord(info, keynr, newrec, + &info->ft_memroot))) + goto err; error=0; while(old_word->pos && new_word->pos) @@ -222,13 +220,13 @@ int _ma_ft_update(MARIA_HA *info, uint keynr, byte *keybuf, { key_length= _ma_ft_make_key(info,keynr,keybuf,old_word,pos); if ((error= _ma_ck_delete(info,keynr,(uchar*) keybuf,key_length))) - goto err2; + goto err; } if (cmp > 0 || cmp2) { key_length= _ma_ft_make_key(info,keynr,keybuf,new_word,pos); if ((error= _ma_ck_write(info,keynr,(uchar*) keybuf,key_length))) - goto err2; + goto err; } if (cmp<=0) old_word++; if (cmp>=0) new_word++; @@ -238,11 +236,8 @@ int _ma_ft_update(MARIA_HA *info, uint keynr, byte *keybuf, else if (new_word->pos) error= _ma_ft_store(info,keynr,keybuf,new_word,pos); -err2: - my_free((char*) newlist,MYF(0)); -err1: - my_free((char*) oldlist,MYF(0)); -err0: +err: + free_root(&info->ft_memroot, MYF(MY_MARK_BLOCKS_FREE)); DBUG_RETURN(error); } @@ -255,12 +250,12 @@ int _ma_ft_add(MARIA_HA *info, uint keynr, byte *keybuf, const byte *record, int error= -1; FT_WORD *wlist; DBUG_ENTER("_ma_ft_add"); + DBUG_PRINT("enter",("keynr: %d",keynr)); - if ((wlist= _ma_ft_parserecord(info, keynr, record))) - { + if ((wlist= _ma_ft_parserecord(info, keynr, record, &info->ft_memroot))) error= _ma_ft_store(info,keynr,keybuf,wlist,pos); - my_free((char*) wlist,MYF(0)); - } + free_root(&info->ft_memroot, MYF(MY_MARK_BLOCKS_FREE)); + DBUG_PRINT("exit",("Return: %d",error)); DBUG_RETURN(error); } @@ -275,11 +270,9 @@ int _ma_ft_del(MARIA_HA *info, uint keynr, byte *keybuf, const byte *record, DBUG_ENTER("_ma_ft_del"); DBUG_PRINT("enter",("keynr: %d",keynr)); - if ((wlist= _ma_ft_parserecord(info, keynr, record))) - { + if ((wlist= _ma_ft_parserecord(info, keynr, record, &info->ft_memroot))) error= _ma_ft_erase(info,keynr,keybuf,wlist,pos); - my_free((char*) wlist,MYF(0)); - } + free_root(&info->ft_memroot, MYF(MY_MARK_BLOCKS_FREE)); DBUG_PRINT("exit",("Return: %d",error)); DBUG_RETURN(error); } diff --git a/storage/maria/ma_ftdefs.h b/storage/maria/ma_ftdefs.h index 41248d1bc9c..def7e92e6e0 100644 --- a/storage/maria/ma_ftdefs.h +++ b/storage/maria/ma_ftdefs.h @@ -24,12 +24,15 @@ #include #include -#define true_word_char(s,X) (my_isalnum(s,X) || (X)=='_') +#define true_word_char(ctype, character) \ + ((ctype) & (_MY_U | _MY_L | _MY_NMR) || \ + (character) == '_') #define misc_word_char(X) 0 -#define word_char(s,X) (true_word_char(s,X) || misc_word_char(X)) #define FT_MAX_WORD_LEN_FOR_SORT 31 +#define FTPARSER_MEMROOT_ALLOC_SIZE 65536 + #define COMPILE_STOPWORDS_IN /* Interested readers may consult SMART @@ -119,12 +122,12 @@ void _ma_ft_segiterator_dummy_init(const byte *, uint, FT_SEG_ITERATOR *); uint _ma_ft_segiterator(FT_SEG_ITERATOR *); void maria_ft_parse_init(TREE *, CHARSET_INFO *); -int maria_ft_parse(TREE *, byte *, int, my_bool, struct st_mysql_ftparser *parser, - MYSQL_FTPARSER_PARAM *param); -FT_WORD * maria_ft_linearize(TREE *); -FT_WORD * _ma_ft_parserecord(MARIA_HA *, uint, const byte *); -uint _ma_ft_parse(TREE *, MARIA_HA *, uint, const byte *, my_bool, - MYSQL_FTPARSER_PARAM *param); +int maria_ft_parse(TREE *, byte *, int, struct st_mysql_ftparser *parser, + MYSQL_FTPARSER_PARAM *, MEM_ROOT *); +FT_WORD * maria_ft_linearize(TREE *, MEM_ROOT *); +FT_WORD * _ma_ft_parserecord(MARIA_HA *, uint, const byte *, MEM_ROOT *); +uint _ma_ft_parse(TREE *, MARIA_HA *, uint, const byte *, + MYSQL_FTPARSER_PARAM *, MEM_ROOT *); FT_INFO *maria_ft_init_nlq_search(MARIA_HA *, uint, byte *, uint, uint, byte *); FT_INFO *maria_ft_init_boolean_search(MARIA_HA *, uint, byte *, uint, CHARSET_INFO *); @@ -145,5 +148,6 @@ float maria_ft_boolean_get_relevance(FT_INFO *); my_off_t maria_ft_boolean_get_docid(FT_INFO *); void maria_ft_boolean_reinit_search(FT_INFO *); extern MYSQL_FTPARSER_PARAM *maria_ftparser_call_initializer(MARIA_HA *info, - uint keynr); + uint keynr, + uint paramnr); extern void maria_ftparser_call_deinitializer(MARIA_HA *info); diff --git a/storage/maria/ma_key.c b/storage/maria/ma_key.c index 6a8c647aa7f..ecd51f5dc92 100644 --- a/storage/maria/ma_key.c +++ b/storage/maria/ma_key.c @@ -127,7 +127,7 @@ uint _ma_make_key(register MARIA_HA *info, uint keynr, uchar *key, } if (keyseg->flag & HA_VAR_LENGTH_PART) { - uint pack_length= keyseg->bit_start; + uint pack_length= (keyseg->bit_start == 1 ? 1 : 2); uint tmp_length= (pack_length == 1 ? (uint) *(uchar*) pos : uint2korr(pos)); pos+= pack_length; /* Skip VARCHAR length */ @@ -509,20 +509,19 @@ int _ma_read_key_record(MARIA_HA *info, my_off_t filepos, byte *buf) /* - Update auto_increment info + Retrieve auto_increment info SYNOPSIS - _ma_update_auto_increment() - info MARIA handler + retrieve_auto_increment() + info Maria handler record Row to update IMPLEMENTATION - Only replace the auto_increment value if it is higher than the previous - one. For signed columns we don't update the auto increment value if it's + For signed columns we don't retrieve the auto increment value if it's less than zero. */ -void _ma_update_auto_increment(MARIA_HA *info,const byte *record) +ulonglong ma_retrieve_auto_increment(MARIA_HA *info,const byte *record) { ulonglong value= 0; /* Store unsigned values here */ longlong s_value= 0; /* Store signed values here */ @@ -587,6 +586,5 @@ void _ma_update_auto_increment(MARIA_HA *info,const byte *record) and if s_value == 0 then value will contain either s_value or the correct value. */ - set_if_bigger(info->s->state.auto_increment, - (s_value > 0) ? (ulonglong) s_value : value); + return (s_value > 0) ? (ulonglong) s_value : value; } diff --git a/storage/maria/ma_open.c b/storage/maria/ma_open.c index c695cbb1a44..38e71a44f8b 100644 --- a/storage/maria/ma_open.c +++ b/storage/maria/ma_open.c @@ -95,7 +95,7 @@ MARIA_HA *maria_open(const char *name, int mode, uint open_flags) bzero((byte*) &info,sizeof(info)); my_realpath(name_buff, fn_format(org_name,name,"",MARIA_NAME_IEXT, - MY_UNPACK_FILENAME|MY_APPEND_EXT),MYF(0)); + MY_UNPACK_FILENAME),MYF(0)); pthread_mutex_lock(&THR_LOCK_maria); if (!(old_info=_ma_test_if_reopen(name_buff))) { @@ -287,7 +287,7 @@ MARIA_HA *maria_open(const char *name, int mode, uint open_flags) &share->data_file_name,strlen(data_name)+1, &share->state.key_root,keys*sizeof(my_off_t), &share->state.key_del, - (share->state.header.max_block_size*sizeof(my_off_t)), + (share->state.header.max_block_size_index*sizeof(my_off_t)), #ifdef THREAD &share->key_root_lock,sizeof(rw_lock_t)*keys, #endif @@ -302,7 +302,7 @@ MARIA_HA *maria_open(const char *name, int mode, uint open_flags) (char*) key_root, sizeof(my_off_t)*keys); memcpy((char*) share->state.key_del, (char*) key_del, (sizeof(my_off_t) * - share->state.header.max_block_size)); + share->state.header.max_block_size_index)); strmov(share->unique_file_name, name_buff); share->unique_name_length= strlen(name_buff); strmov(share->index_file_name, index_name); @@ -799,7 +799,7 @@ uint _ma_state_info_write(File file, MARIA_STATE_INFO *state, uint pWrite) uchar buff[MARIA_STATE_INFO_SIZE + MARIA_STATE_EXTRA_SIZE]; uchar *ptr=buff; uint i, keys= (uint) state->header.keys, - key_blocks=state->header.max_block_size; + key_blocks=state->header.max_block_size_index; DBUG_ENTER("_ma_state_info_write"); memcpy_fixed(ptr,&state->header,sizeof(state->header)); @@ -865,7 +865,7 @@ uchar *_ma_state_info_read(uchar *ptr, MARIA_STATE_INFO *state) ptr +=sizeof(state->header); keys=(uint) state->header.keys; key_parts=mi_uint2korr(state->header.key_parts); - key_blocks=state->header.max_block_size; + key_blocks=state->header.max_block_size_index; state->open_count = mi_uint2korr(ptr); ptr +=2; state->changed= (bool) *ptr++; @@ -1038,7 +1038,7 @@ char *_ma_keydef_read(char *ptr, MARIA_KEYDEF *keydef) keydef->keylength = mi_uint2korr(ptr); ptr +=2; keydef->minlength = mi_uint2korr(ptr); ptr +=2; keydef->maxlength = mi_uint2korr(ptr); ptr +=2; - keydef->block_size = keydef->block_length/MARIA_MIN_KEY_BLOCK_LENGTH-1; + keydef->block_size_index = keydef->block_length/MARIA_MIN_KEY_BLOCK_LENGTH-1; keydef->underflow_block_length=keydef->block_length/3; keydef->version = 0; /* Not saved */ keydef->parser = &ft_default_parser; diff --git a/storage/maria/ma_page.c b/storage/maria/ma_page.c index 78864e2c9ac..054b8e16468 100644 --- a/storage/maria/ma_page.c +++ b/storage/maria/ma_page.c @@ -112,8 +112,8 @@ int _ma_dispose(register MARIA_HA *info, MARIA_KEYDEF *keyinfo, my_off_t pos, DBUG_ENTER("_ma_dispose"); DBUG_PRINT("enter",("pos: %ld", (long) pos)); - old_link=info->s->state.key_del[keyinfo->block_size]; - info->s->state.key_del[keyinfo->block_size]=pos; + old_link= info->s->state.key_del[keyinfo->block_size_index]; + info->s->state.key_del[keyinfo->block_size_index]= pos; mi_sizestore(buff,old_link); info->s->state.changed|= STATE_NOT_SORTED_PAGES; DBUG_RETURN(key_cache_write(info->s->key_cache, @@ -132,7 +132,8 @@ my_off_t _ma_new(register MARIA_HA *info, MARIA_KEYDEF *keyinfo, int level) char buff[8]; DBUG_ENTER("_ma_new"); - if ((pos=info->s->state.key_del[keyinfo->block_size]) == HA_OFFSET_ERROR) + if ((pos= info->s->state.key_del[keyinfo->block_size_index]) == + HA_OFFSET_ERROR) { if (info->state->key_file_length >= info->s->base.max_key_file_length - keyinfo->block_length) @@ -152,7 +153,7 @@ my_off_t _ma_new(register MARIA_HA *info, MARIA_KEYDEF *keyinfo, int level) (uint) keyinfo->block_length,0)) pos= HA_OFFSET_ERROR; else - info->s->state.key_del[keyinfo->block_size]=mi_sizekorr(buff); + info->s->state.key_del[keyinfo->block_size_index]= mi_sizekorr(buff); } info->s->state.changed|= STATE_NOT_SORTED_PAGES; DBUG_PRINT("exit",("Pos: %ld",(long) pos)); diff --git a/storage/maria/ma_rkey.c b/storage/maria/ma_rkey.c index abcce1a2582..2cb54a73b15 100644 --- a/storage/maria/ma_rkey.c +++ b/storage/maria/ma_rkey.c @@ -92,7 +92,18 @@ int maria_rkey(MARIA_HA *info, byte *buf, int inx, const byte *key, uint key_len if (!_ma_search(info, keyinfo, key_buff, use_key_length, maria_read_vec[search_flag], info->s->state.key_root[inx])) { - while (info->lastpos >= info->state->data_file_length) + /* + If we are searching for an exact key (including the data pointer) + and this was added by an concurrent insert, + then the result is "key not found". + */ + if ((search_flag == HA_READ_KEY_EXACT) && + (info->lastpos >= info->state->data_file_length)) + { + my_errno= HA_ERR_KEY_NOT_FOUND; + info->lastpos= HA_OFFSET_ERROR; + } + else while (info->lastpos >= info->state->data_file_length) { /* Skip rows that are inserted by other threads since we got a lock @@ -101,9 +112,9 @@ int maria_rkey(MARIA_HA *info, byte *buf, int inx, const byte *key, uint key_len */ if (_ma_search_next(info, keyinfo, info->lastkey, - info->lastkey_length, - maria_readnext_vec[search_flag], - info->s->state.key_root[inx])) + info->lastkey_length, + maria_readnext_vec[search_flag], + info->s->state.key_root[inx])) break; } } diff --git a/storage/maria/ma_rsamepos.c b/storage/maria/ma_rsamepos.c index e30d41dd46c..09861c03c32 100644 --- a/storage/maria/ma_rsamepos.c +++ b/storage/maria/ma_rsamepos.c @@ -31,8 +31,9 @@ int maria_rsame_with_pos(MARIA_HA *info, byte *record, int inx, my_off_t filepos) { DBUG_ENTER("maria_rsame_with_pos"); + DBUG_PRINT("enter",("index: %d filepos: %ld", inx, (long) filepos)); - if (inx < -1 || ! maria_is_key_active(info->s->state.key_map, inx)) + if (inx < -1 || (inx >= 0 && !maria_is_key_active(info->s->state.key_map, inx))) { DBUG_RETURN(my_errno=HA_ERR_WRONG_INDEX); } diff --git a/storage/maria/ma_rt_index.c b/storage/maria/ma_rt_index.c index ff10ae72027..83ced5b8167 100644 --- a/storage/maria/ma_rt_index.c +++ b/storage/maria/ma_rt_index.c @@ -183,9 +183,11 @@ int maria_rtree_find_first(MARIA_HA *info, uint keynr, uchar *key, uint key_leng return -1; } - /* Save searched key */ - memcpy(info->first_mbr_key, key, keyinfo->keylength - - info->s->base.rec_reflength); + /* + Save searched key, include data pointer. + The data pointer is required if the search_flag contains MBR_DATA. + */ + memcpy(info->first_mbr_key, key, keyinfo->keylength); info->last_rkey_length = key_length; info->maria_rtree_recursion_depth = -1; diff --git a/storage/maria/ma_rt_mbr.c b/storage/maria/ma_rt_mbr.c index 83d3a0a2f1c..67b1d59f505 100644 --- a/storage/maria/ma_rt_mbr.c +++ b/storage/maria/ma_rt_mbr.c @@ -52,10 +52,14 @@ if (EQUAL_CMP(amin, amax, bmin, bmax)) \ return 1; \ } \ - else /* if (nextflag & MBR_DISJOINT) */ \ + else if (nextflag & MBR_DISJOINT) \ { \ if (DISJOINT_CMP(amin, amax, bmin, bmax)) \ return 1; \ + }\ + else /* if unknown comparison operator */ \ + { \ + DBUG_ASSERT(0); \ } #define RT_CMP_KORR(type, korr_func, len, nextflag) \ diff --git a/storage/maria/ma_search.c b/storage/maria/ma_search.c index 3bb048ea239..af25be06a09 100644 --- a/storage/maria/ma_search.c +++ b/storage/maria/ma_search.c @@ -259,15 +259,16 @@ int _ma_seq_search(MARIA_HA *info, register MARIA_KEYDEF *keyinfo, uchar *page, { maria_print_error(info->s, HA_ERR_CRASHED); my_errno=HA_ERR_CRASHED; - DBUG_PRINT("error",("Found wrong key: length: %u page: %lx end: %lx", - length, (long) page, (long) end)); + DBUG_PRINT("error", + ("Found wrong key: length: %u page: 0x%lx end: 0x%lx", + length, (long) page, (long) end)); DBUG_RETURN(MARIA_FOUND_WRONG_KEY); } if ((flag=ha_key_cmp(keyinfo->seg,t_buff,key,key_len,comp_flag, not_used)) >= 0) break; #ifdef EXTRA_DEBUG - DBUG_PRINT("loop",("page: %lx key: '%s' flag: %d", (long) page, t_buff, + DBUG_PRINT("loop",("page: 0x%lx key: '%s' flag: %d", (long) page, t_buff, flag)); #endif memcpy(buff,t_buff,length); @@ -276,7 +277,7 @@ int _ma_seq_search(MARIA_HA *info, register MARIA_KEYDEF *keyinfo, uchar *page, if (flag == 0) memcpy(buff,t_buff,length); /* Result is first key */ *last_key= page == end; - DBUG_PRINT("exit",("flag: %d ret_pos: %lx", flag, (long) *ret_pos)); + DBUG_PRINT("exit",("flag: %d ret_pos: 0x%lx", flag, (long) *ret_pos)); DBUG_RETURN(flag); } /* _ma_seq_search */ @@ -416,8 +417,9 @@ int _ma_prefix_search(MARIA_HA *info, register MARIA_KEYDEF *keyinfo, uchar *pag { maria_print_error(info->s, HA_ERR_CRASHED); my_errno=HA_ERR_CRASHED; - DBUG_PRINT("error",("Found wrong key: length: %u page: %lx end: %lx", - length, (long) page, (long) end)); + DBUG_PRINT("error", + ("Found wrong key: length: %u page: 0x%lx end: %lx", + length, (long) page, (long) end)); DBUG_RETURN(MARIA_FOUND_WRONG_KEY); } @@ -551,7 +553,7 @@ int _ma_prefix_search(MARIA_HA *info, register MARIA_KEYDEF *keyinfo, uchar *pag *last_key= page == end; - DBUG_PRINT("exit",("flag: %d ret_pos: %lx", flag, (long) *ret_pos)); + DBUG_PRINT("exit",("flag: %d ret_pos: 0x%lx", flag, (long) *ret_pos)); DBUG_RETURN(flag); } /* _ma_prefix_search */ @@ -813,7 +815,7 @@ uint _ma_get_pack_key(register MARIA_KEYDEF *keyinfo, uint nod_flag, if (length > keyseg->length) { DBUG_PRINT("error", - ("Found too long null packed key: %u of %u at %lx", + ("Found too long null packed key: %u of %u at 0x%lx", length, keyseg->length, (long) *page_pos)); DBUG_DUMP("key",(char*) *page_pos,16); maria_print_error(keyinfo->share, HA_ERR_CRASHED); @@ -870,7 +872,7 @@ uint _ma_get_pack_key(register MARIA_KEYDEF *keyinfo, uint nod_flag, } if (length > (uint) keyseg->length) { - DBUG_PRINT("error",("Found too long packed key: %u of %u at %lx", + DBUG_PRINT("error",("Found too long packed key: %u of %u at 0x%lx", length, keyseg->length, (long) *page_pos)); DBUG_DUMP("key",(char*) *page_pos,16); maria_print_error(keyinfo->share, HA_ERR_CRASHED); @@ -936,8 +938,9 @@ uint _ma_get_binary_pack_key(register MARIA_KEYDEF *keyinfo, uint nod_flag, { if (length > keyinfo->maxlength) { - DBUG_PRINT("error",("Found too long binary packed key: %u of %u at %lx", - length, keyinfo->maxlength, (long) *page_pos)); + DBUG_PRINT("error", + ("Found too long binary packed key: %u of %u at 0x%lx", + length, keyinfo->maxlength, (long) *page_pos)); DBUG_DUMP("key",(char*) *page_pos,16); maria_print_error(keyinfo->share, HA_ERR_CRASHED); my_errno=HA_ERR_CRASHED; @@ -984,7 +987,7 @@ uint _ma_get_binary_pack_key(register MARIA_KEYDEF *keyinfo, uint nod_flag, length-=tmp; from=page; from_end=page_end; } - DBUG_PRINT("info",("key: %lx from: %lx length: %u", + DBUG_PRINT("info",("key: 0x%lx from: 0x%lx length: %u", (long) key, (long) from, length)); memmove((byte*) key, (byte*) from, (size_t) length); key+=length; @@ -1042,7 +1045,7 @@ uchar *_ma_get_key(MARIA_HA *info, MARIA_KEYDEF *keyinfo, uchar *page, } } } - DBUG_PRINT("exit",("page: %lx length: %u", (long) page, + DBUG_PRINT("exit",("page: 0x%lx length: %u", (long) page, *return_key_length)); DBUG_RETURN(page); } /* _ma_get_key */ @@ -1095,7 +1098,8 @@ uchar *_ma_get_last_key(MARIA_HA *info, MARIA_KEYDEF *keyinfo, uchar *page, uint nod_flag; uchar *lastpos; DBUG_ENTER("_ma_get_last_key"); - DBUG_PRINT("enter",("page: %lx endpos: %lx", (long) page, (long) endpos)); + DBUG_PRINT("enter",("page: 0x%lx endpos: 0x%lx", (long) page, + (long) endpos)); nod_flag=_ma_test_if_nod(page); if (! (keyinfo->flag & (HA_VAR_LENGTH_KEY | HA_BINARY_PACK_KEY))) @@ -1115,7 +1119,7 @@ uchar *_ma_get_last_key(MARIA_HA *info, MARIA_KEYDEF *keyinfo, uchar *page, *return_key_length=(*keyinfo->get_key)(keyinfo,nod_flag,&page,lastkey); if (*return_key_length == 0) { - DBUG_PRINT("error",("Couldn't find last key: page: %lx", + DBUG_PRINT("error",("Couldn't find last key: page: 0x%lx", (long) page)); maria_print_error(info->s, HA_ERR_CRASHED); my_errno=HA_ERR_CRASHED; @@ -1123,7 +1127,7 @@ uchar *_ma_get_last_key(MARIA_HA *info, MARIA_KEYDEF *keyinfo, uchar *page, } } } - DBUG_PRINT("exit",("lastpos: %lx length: %u", (long) lastpos, + DBUG_PRINT("exit",("lastpos: 0x%lx length: %u", (long) lastpos, *return_key_length)); DBUG_RETURN(lastpos); } /* _ma_get_last_key */ @@ -1472,7 +1476,7 @@ _ma_calc_var_pack_key_length(MARIA_KEYDEF *keyinfo, uint nod_flag, if (!*key++) { s_temp->key=key; - s_temp->ref_length=s_temp->key_length=0; + s_temp->key_length=0; s_temp->totlength=key_length-1+diff_flag; s_temp->next_key_pos=0; /* No next key */ return (s_temp->totlength); @@ -1627,12 +1631,12 @@ _ma_calc_var_pack_key_length(MARIA_KEYDEF *keyinfo, uint nod_flag, s_temp->prev_length= org_key_length; s_temp->n_ref_length=s_temp->n_length= org_key_length; length+= org_key_length; - /* +get_pack_length(org_key_length); */ } return (int) length; } ref_length=n_length; + /* Get information about not packed key suffix */ get_key_pack_length(n_length,next_length_pack,next_key); /* Test if new keys has fewer characters that match the previous key */ @@ -1641,7 +1645,6 @@ _ma_calc_var_pack_key_length(MARIA_KEYDEF *keyinfo, uint nod_flag, s_temp->part_of_prev_key= 0; s_temp->prev_length= ref_length; s_temp->n_ref_length= s_temp->n_length= n_length+ref_length; - /* s_temp->prev_key+= get_pack_length(org_key_length); */ return (int) length+ref_length-next_length_pack; } if (ref_length+pack_marker > new_ref_length) @@ -1652,9 +1655,7 @@ _ma_calc_var_pack_key_length(MARIA_KEYDEF *keyinfo, uint nod_flag, s_temp->prev_length= ref_length - new_pack_length; s_temp->n_ref_length=s_temp->n_length=n_length + s_temp->prev_length; s_temp->prev_key+= new_pack_length; -/* +get_pack_length(org_key_length); */ - length= length-get_pack_length(ref_length)+ - get_pack_length(new_pack_length); + length-= (next_length_pack - get_pack_length(s_temp->n_length)); return (int) length + s_temp->prev_length; } } @@ -1664,7 +1665,7 @@ _ma_calc_var_pack_key_length(MARIA_KEYDEF *keyinfo, uint nod_flag, ref_length=0; next_length_pack=0; } - DBUG_PRINT("test",("length: %d next_key: %lx", length, + DBUG_PRINT("test",("length: %d next_key: 0x%lx", length, (long) next_key)); { diff --git a/storage/maria/ma_sort.c b/storage/maria/ma_sort.c index df547792ffa..5ae23c37261 100644 --- a/storage/maria/ma_sort.c +++ b/storage/maria/ma_sort.c @@ -442,6 +442,7 @@ err: close_cached_file(&info->tempfile_for_exceptions); ok: + free_root(&info->wordroot, MYF(0)); remove_io_thread(&info->read_cache); pthread_mutex_lock(&info->sort_info->mutex); info->sort_info->threads_running--; diff --git a/storage/maria/ma_test1.c b/storage/maria/ma_test1.c index 621a6d90e65..69d432a5d95 100644 --- a/storage/maria/ma_test1.c +++ b/storage/maria/ma_test1.c @@ -96,6 +96,7 @@ static int run_test(const char *filename) /* Define a key over the first column */ keyinfo[0].seg=keyseg; keyinfo[0].keysegs=1; + keyinfo[0].block_length= 0; /* Default block length */ keyinfo[0].key_alg=HA_KEY_ALG_BTREE; keyinfo[0].seg[0].type= key_type; keyinfo[0].seg[0].flag= pack_seg; diff --git a/storage/maria/ma_test2.c b/storage/maria/ma_test2.c index 7dd1ffeb7fd..840ecb2eeb7 100644 --- a/storage/maria/ma_test2.c +++ b/storage/maria/ma_test2.c @@ -96,6 +96,7 @@ int main(int argc, char *argv[]) keyinfo[0].key_alg=HA_KEY_ALG_BTREE; keyinfo[0].keysegs=1; keyinfo[0].flag = pack_type; + keyinfo[0].block_length= 0; /* Default block length */ keyinfo[1].seg= &glob_keyseg[1][0]; keyinfo[1].seg[0].start=7; keyinfo[1].seg[0].length=6; @@ -112,6 +113,7 @@ int main(int argc, char *argv[]) keyinfo[1].key_alg=HA_KEY_ALG_BTREE; keyinfo[1].keysegs=2; keyinfo[1].flag =0; + keyinfo[1].block_length= MARIA_MIN_KEY_BLOCK_LENGTH; /* Diff blocklength */ keyinfo[2].seg= &glob_keyseg[2][0]; keyinfo[2].seg[0].start=12; keyinfo[2].seg[0].length=8; @@ -122,6 +124,7 @@ int main(int argc, char *argv[]) keyinfo[2].key_alg=HA_KEY_ALG_BTREE; keyinfo[2].keysegs=1; keyinfo[2].flag =HA_NOSAME; + keyinfo[2].block_length= 0; /* Default block length */ keyinfo[3].seg= &glob_keyseg[3][0]; keyinfo[3].seg[0].start=0; keyinfo[3].seg[0].length=reclength-(use_blob ? 8 : 0); @@ -133,6 +136,7 @@ int main(int argc, char *argv[]) keyinfo[3].key_alg=HA_KEY_ALG_BTREE; keyinfo[3].keysegs=1; keyinfo[3].flag = pack_type; + keyinfo[3].block_length= 0; /* Default block length */ keyinfo[4].seg= &glob_keyseg[4][0]; keyinfo[4].seg[0].start=0; keyinfo[4].seg[0].length=5; @@ -144,6 +148,7 @@ int main(int argc, char *argv[]) keyinfo[4].key_alg=HA_KEY_ALG_BTREE; keyinfo[4].keysegs=1; keyinfo[4].flag = pack_type; + keyinfo[4].block_length= 0; /* Default block length */ keyinfo[5].seg= &glob_keyseg[5][0]; keyinfo[5].seg[0].start=0; keyinfo[5].seg[0].length=4; @@ -155,6 +160,7 @@ int main(int argc, char *argv[]) keyinfo[5].key_alg=HA_KEY_ALG_BTREE; keyinfo[5].keysegs=1; keyinfo[5].flag = pack_type; + keyinfo[5].block_length= 0; /* Default block length */ recinfo[0].type=pack_fields ? FIELD_SKIP_PRESPACE : 0; recinfo[0].length=7; @@ -701,7 +707,7 @@ int main(int argc, char *argv[]) if (!silent) printf("- maria_extra(CACHE) + maria_rrnd.... + maria_extra(NO_CACHE)\n"); - if (maria_extra(file,HA_EXTRA_RESET,0) || maria_extra(file,HA_EXTRA_CACHE,0)) + if (maria_reset(file) || maria_extra(file,HA_EXTRA_CACHE,0)) { if (locking || (!use_blob && !pack_fields)) { @@ -744,7 +750,7 @@ int main(int argc, char *argv[]) DBUG_PRINT("progpos",("Removing keys")); lastpos = HA_OFFSET_ERROR; /* DBUG_POP(); */ - maria_extra(file,HA_EXTRA_RESET,0); + maria_reset(file); found_parts=0; while ((error=maria_rrnd(file,read_record,HA_OFFSET_ERROR)) != HA_ERR_END_OF_FILE) @@ -911,13 +917,13 @@ static void get_options(int argc, char **argv) } break; case 'e': /* maria_block_length */ - if ((maria_block_size=atoi(++pos)) < MARIA_MIN_KEY_BLOCK_LENGTH || + if ((maria_block_size= atoi(++pos)) < MARIA_MIN_KEY_BLOCK_LENGTH || maria_block_size > MARIA_MAX_KEY_BLOCK_LENGTH) { fprintf(stderr,"Wrong maria_block_length\n"); exit(1); } - maria_block_size=1 << my_bit_log2(maria_block_size); + maria_block_size= my_round_up_to_next_power(maria_block_size); break; case 'E': /* maria_block_length */ if ((key_cache_block_size=atoi(++pos)) < MARIA_MIN_KEY_BLOCK_LENGTH || @@ -926,7 +932,7 @@ static void get_options(int argc, char **argv) fprintf(stderr,"Wrong key_cache_block_size\n"); exit(1); } - key_cache_block_size=1 << my_bit_log2(key_cache_block_size); + key_cache_block_size= my_round_up_to_next_power(key_cache_block_size); break; case 'f': if ((first_key=atoi(++pos)) < 0 || first_key >= MARIA_KEYS) diff --git a/storage/maria/ma_test3.c b/storage/maria/ma_test3.c index d2fe8d90cf6..96b896b03c6 100644 --- a/storage/maria/ma_test3.c +++ b/storage/maria/ma_test3.c @@ -77,6 +77,7 @@ int main(int argc,char **argv) keyinfo[0].key_alg=HA_KEY_ALG_BTREE; keyinfo[0].keysegs=1; keyinfo[0].flag = (uint8) HA_PACK_KEY; + keyinfo[0].block_length= 0; /* Default block length */ keyinfo[1].seg= &keyseg[1][0]; keyinfo[1].seg[0].start=8; keyinfo[1].seg[0].length=4; /* Long is always 4 in maria */ @@ -85,6 +86,7 @@ int main(int argc,char **argv) keyinfo[1].key_alg=HA_KEY_ALG_BTREE; keyinfo[1].keysegs=1; keyinfo[1].flag =HA_NOSAME; + keyinfo[1].block_length= 0; /* Default block length */ recinfo[0].type=0; recinfo[0].length=sizeof(record.id); diff --git a/storage/maria/ma_update.c b/storage/maria/ma_update.c index 3f1ba4b1fac..248b17ce2c9 100644 --- a/storage/maria/ma_update.c +++ b/storage/maria/ma_update.c @@ -164,7 +164,8 @@ int maria_update(register MARIA_HA *info, const byte *oldrec, byte *newrec) key_changed|= HA_STATE_CHANGED; /* Must update index file */ } if (auto_key_changed) - _ma_update_auto_increment(info,newrec); + set_if_bigger(info->s->state.auto_increment, + ma_retrieve_auto_increment(info, newrec)); if (share->calc_checksum) info->state->checksum+=(info->checksum - old_checksum); diff --git a/storage/maria/ma_write.c b/storage/maria/ma_write.c index 5a1a540b88b..24768b36c89 100644 --- a/storage/maria/ma_write.c +++ b/storage/maria/ma_write.c @@ -145,7 +145,8 @@ int maria_write(MARIA_HA *info, byte *record) info->state->checksum+=info->checksum; } if (share->base.auto_key) - _ma_update_auto_increment(info,record); + set_if_bigger(info->s->state.auto_increment, + ma_retrieve_auto_increment(info, record)); info->update= (HA_STATE_CHANGED | HA_STATE_AKTIV | HA_STATE_WRITTEN | HA_STATE_ROW_CHANGED); info->state->records++; diff --git a/storage/maria/maria_chk.c b/storage/maria/maria_chk.c index 7f48a0805da..e423a3f5c36 100644 --- a/storage/maria/maria_chk.c +++ b/storage/maria/maria_chk.c @@ -34,10 +34,6 @@ SET_STACK_SIZE(9000) /* Minimum stack size for program */ #define my_raid_delete(A,B,C) my_delete(A,B) #endif -#ifdef OS2 -#define _sanity(a,b) -#endif - static uint decode_bits; static char **default_argv; static const char *load_default_groups[]= { "mariachk", 0 }; @@ -92,10 +88,6 @@ int main(int argc, char **argv) MY_INIT(argv[0]); my_progname_short= my_progname+dirname_length(my_progname); -#ifdef __EMX__ - _wildcard (&argc, &argv); -#endif - mariachk_init(&check_param); check_param.opt_lock_memory= 1; /* Lock memory if possible */ check_param.using_global_keycache = 0; @@ -381,7 +373,7 @@ static void usage(void) directly with '--variable-name=value'.\n\ -t, --tmpdir=path Path for temporary files. Multiple paths can be\n\ specified, separated by "); -#if defined( __WIN__) || defined(OS2) || defined(__NETWARE__) +#if defined( __WIN__) || defined(__NETWARE__) printf("semicolon (;)"); #else printf("colon (:)"); diff --git a/storage/maria/maria_def.h b/storage/maria/maria_def.h index 9bab65126c7..ecd93807a06 100644 --- a/storage/maria/maria_def.h +++ b/storage/maria/maria_def.h @@ -55,7 +55,7 @@ typedef struct st_maria_state_info uchar keys; /* number of keys in file */ uchar uniques; /* number of UNIQUE definitions */ uchar language; /* Language for indexes */ - uchar max_block_size; /* max keyblock size */ + uchar max_block_size_index; /* max keyblock size */ uchar fulltext_keys; uchar not_used; /* To align to 8 */ } header; @@ -246,6 +246,7 @@ struct st_maria_info /* accumulate indexfile changes between write's */ TREE *bulk_insert; DYNAMIC_ARRAY *ft1_to_ft2; /* used only in ft1->ft2 conversion */ + MEM_ROOT ft_memroot; /* used by the parser */ MYSQL_FTPARSER_PARAM *ftparser_param; /* share info between init/deinit */ char *filename; /* parameter to open filename */ uchar *buff, /* Temp area for key */ @@ -398,7 +399,7 @@ struct st_maria_info #define MARIA_FOUND_WRONG_KEY 32738 /* Impossible value from ha_key_cmp */ #define MARIA_MAX_KEY_BLOCK_SIZE (MARIA_MAX_KEY_BLOCK_LENGTH/MARIA_MIN_KEY_BLOCK_LENGTH) -#define MARIA_BLOCK_SIZE(key_length,data_pointer,key_pointer) (((((key_length)+(data_pointer)+(key_pointer))*4+(key_pointer)+2)/maria_block_size+1)*maria_block_size) +#define MARIA_BLOCK_SIZE(key_length,data_pointer,key_pointer,block_size) (((((key_length)+(data_pointer)+(key_pointer))*4+(key_pointer)+2)/(block_size)+1)*(block_size)) #define MARIA_MAX_KEYPTR_SIZE 5 /* For calculating block lengths */ #define MARIA_MIN_KEYBLOCK_LENGTH 50 /* When to split delete blocks */ @@ -572,7 +573,7 @@ extern int _ma_read_key_record(MARIA_HA *info, my_off_t filepos, byte *buf); extern int _ma_read_cache(IO_CACHE *info, byte *buff, my_off_t pos, uint length, int re_read_if_possibly); -extern void _ma_update_auto_increment(MARIA_HA *info, const byte *record); +extern ulonglong ma_retrieve_auto_increment(MARIA_HA *info, const byte *record); extern byte *_ma_alloc_rec_buff(MARIA_HA *, ulong, byte **); #define _ma_get_rec_buff_ptr(info,buf) \ diff --git a/storage/maria/maria_ftdump.c b/storage/maria/maria_ftdump.c index eb5dec5aa3b..b840072aed0 100644 --- a/storage/maria/maria_ftdump.c +++ b/storage/maria/maria_ftdump.c @@ -34,20 +34,20 @@ static uint lengths[256]; static struct my_option my_long_options[] = { - {"dump", 'd', "Dump index (incl. data offsets and word weights).", + {"help", 'h', "Display help and exit.", 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, - {"stats", 's', "Report global stats.", + {"help", '?', "Synonym for -h.", 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, - {"verbose", 'v', "Be verbose.", - (gptr*) &verbose, (gptr*) &verbose, 0, GET_BOOL, NO_ARG, 0, 0, 0, 0, 0, 0}, {"count", 'c', "Calculate per-word stats (counts and global weights).", 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, - {"length", 'l', "Report length distribution.", + {"dump", 'd', "Dump index (incl. data offsets and word weights).", 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, - {"help", 'h', "Display help and exit.", + {"length", 'l', "Report length distribution.", 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, - {"help", '?', "Synonym for -h.", + {"stats", 's', "Report global stats.", 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, + {"verbose", 'v', "Be verbose.", + (gptr*) &verbose, (gptr*) &verbose, 0, GET_BOOL, NO_ARG, 0, 0, 0, 0, 0, 0}, { 0, 0, 0, 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0} }; @@ -87,7 +87,8 @@ int main(int argc,char *argv[]) init_key_cache(maria_key_cache,MARIA_KEY_BLOCK_LENGTH,USE_BUFFER_INIT, 0, 0); - if (!(info=maria_open(argv[0],2,HA_OPEN_ABORT_IF_LOCKED|HA_OPEN_FROM_SQL_LAYER))) + if (!(info=maria_open(argv[0], O_RDONLY, + HA_OPEN_ABORT_IF_LOCKED|HA_OPEN_FROM_SQL_LAYER))) { error=my_errno; goto err; diff --git a/storage/maria/maria_pack.c b/storage/maria/maria_pack.c index 514d8ba0bf5..c5a53b1ffac 100644 --- a/storage/maria/maria_pack.c +++ b/storage/maria/maria_pack.c @@ -2968,7 +2968,7 @@ static int save_state(MARIA_HA *isam_file,PACK_MRG_INFO *mrg,my_off_t new_length maria_clear_all_keys_active(share->state.key_map); for (key=0 ; key < share->base.keys ; key++) share->state.key_root[key]= HA_OFFSET_ERROR; - for (key=0 ; key < share->state.header.max_block_size ; key++) + for (key=0 ; key < share->state.header.max_block_size_index ; key++) share->state.key_del[key]= HA_OFFSET_ERROR; isam_file->state->checksum=crc; /* Save crc here */ share->changed=1; /* Force write of header */ @@ -3035,7 +3035,7 @@ static int mrg_rrnd(PACK_MRG_INFO *info,byte *buf) { isam_info= *(info->current=info->file); info->end=info->current+info->count; - maria_extra(isam_info, HA_EXTRA_RESET, 0); + maria_reset(isam_info); maria_extra(isam_info, HA_EXTRA_CACHE, 0); filepos=isam_info->s->pack.header_length; } @@ -3058,7 +3058,7 @@ static int mrg_rrnd(PACK_MRG_INFO *info,byte *buf) info->current++; isam_info= *info->current; filepos=isam_info->s->pack.header_length; - maria_extra(isam_info,HA_EXTRA_RESET, 0); + maria_reset(isam_info); maria_extra(isam_info,HA_EXTRA_CACHE, 0); } } -- cgit v1.2.1 From e8524352dc79e9231fd2c2d835bf0e479ba6e77a Mon Sep 17 00:00:00 2001 From: unknown Date: Thu, 10 Aug 2006 17:24:50 +0200 Subject: adding handler flag missing from previous merge, for Maria sql/ha_maria.cc: flag missing from previous merge --- sql/ha_maria.cc | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/sql/ha_maria.cc b/sql/ha_maria.cc index aa244450bb5..29718f1493e 100644 --- a/sql/ha_maria.cc +++ b/sql/ha_maria.cc @@ -155,7 +155,7 @@ handler(&maria_hton, table_arg), file(0), int_table_flags(HA_NULL_IN_KEY | HA_CAN_FULLTEXT | HA_CAN_SQL_HANDLER | HA_DUPLICATE_POS | HA_CAN_INDEX_BLOBS | HA_AUTO_PART_KEY | HA_FILE_BASED | HA_CAN_GEOMETRY | HA_NO_TRANSACTIONS | - HA_CAN_INSERT_DELAYED | HA_CAN_BIT_FIELD | + HA_CAN_INSERT_DELAYED | HA_CAN_BIT_FIELD | HA_CAN_RTREEKEYS | HA_HAS_RECORDS | HA_STATS_RECORDS_IS_EXACT), can_enable_indexes(1) {} -- cgit v1.2.1 From cd876fb11883f68f93027a70b5f3f99ad9234f27 Mon Sep 17 00:00:00 2001 From: unknown Date: Thu, 10 Aug 2006 19:19:47 +0200 Subject: amd64 atomic ops lock-free alloc (WL#3229), lock-free hash (WL#3230) bit functions made inline include/Makefile.am: lf.h added mysys/Makefile.am: lf_hash.c lf_dynarray.c lf_alloc-pin.c include/atomic/nolock.h: amd64 atomic ops include/atomic/rwlock.h: s/rw_lock/mutex/g include/atomic/x86-gcc.h: amd64 atomic ops try PAUSE include/my_global.h: STATIC_INLINE mysys/mf_keycache.c: make bit functions inline mysys/my_atomic.c: STATIC_INLINE mysys/my_bitmap.c: make bit functions inline sql/ha_myisam.cc: make bit functions inline sql/item_func.cc: make bit functions inline include/my_atomic.h: STATIC_INLINE mysys/my_bit.c: make bit functions inline sql/sql_select.cc: make bit functions inline storage/myisam/mi_create.c: make bit functions inline storage/myisam/mi_test2.c: make bit functions inline storage/myisam/myisamchk.c: make bit functions inline mysys/my_init.c: thread_size moved to mysys sql/mysql_priv.h: thread_size moved to mysys sql/set_var.cc: thread_size moved to mysys include/my_sys.h: thread_size moved to mysys sql/mysqld.cc: thread_size moved to mysys sql/sql_parse.cc: thread_size moved to mysys sql/sql_test.cc: thread_size moved to mysys include/lf.h: dylf_dynarray refactored to remove 65536 elements limit mysys/lf_alloc-pin.c: dylf_dynarray refactored to remove 65536 elements limit mysys/lf_dynarray.c: dylf_dynarray refactored to remove 65536 elements limit mysys/lf_hash.c: dylf_dynarray refactored to remove 65536 elements limit unittest/mysys/my_atomic-t.c: fix to commit (remove debug code) --- include/Makefile.am | 2 +- include/atomic/nolock.h | 27 ++-- include/atomic/rwlock.h | 21 ++- include/atomic/x86-gcc.h | 19 ++- include/lf.h | 238 ++++++++++++++++++++++++++++ include/my_atomic.h | 14 +- include/my_bit.h | 107 +++++++++++++ include/my_global.h | 2 + include/my_sys.h | 7 +- mysys/Makefile.am | 3 +- mysys/lf_alloc-pin.c | 319 +++++++++++++++++++++++++++++++++++++ mysys/lf_dynarray.c | 186 ++++++++++++++++++++++ mysys/lf_hash.c | 370 +++++++++++++++++++++++++++++++++++++++++++ mysys/mf_keycache.c | 1 + mysys/my_atomic.c | 7 +- mysys/my_bit.c | 100 ++++-------- mysys/my_bitmap.c | 1 + mysys/my_init.c | 1 + sql/ha_myisam.cc | 1 + sql/item_func.cc | 1 + sql/mysql_priv.h | 2 +- sql/mysqld.cc | 25 +-- sql/set_var.cc | 4 +- sql/sql_parse.cc | 4 +- sql/sql_select.cc | 1 + sql/sql_test.cc | 4 +- storage/myisam/mi_create.c | 3 +- storage/myisam/mi_test2.c | 1 + storage/myisam/myisamchk.c | 1 + unittest/mysys/my_atomic-t.c | 190 +++++++++++++++++++--- 30 files changed, 1508 insertions(+), 154 deletions(-) create mode 100644 include/lf.h create mode 100644 include/my_bit.h create mode 100644 mysys/lf_alloc-pin.c create mode 100644 mysys/lf_dynarray.c create mode 100644 mysys/lf_hash.c diff --git a/include/Makefile.am b/include/Makefile.am index 26161b36ab8..fe0ac1be31a 100644 --- a/include/Makefile.am +++ b/include/Makefile.am @@ -24,7 +24,7 @@ pkginclude_HEADERS = my_dbug.h m_string.h my_sys.h my_list.h my_xml.h \ sslopt-vars.h sslopt-case.h sql_common.h keycache.h \ mysql_time.h mysql/plugin.h $(BUILT_SOURCES) noinst_HEADERS = config-win.h config-netware.h \ - heap.h my_bitmap.h my_uctype.h \ + heap.h my_bitmap.h my_uctype.h lf.h \ myisam.h myisampack.h myisammrg.h ft_global.h\ mysys_err.h my_base.h help_start.h help_end.h \ my_nosys.h my_alarm.h queues.h rijndael.h sha1.h \ diff --git a/include/atomic/nolock.h b/include/atomic/nolock.h index 1151a334b06..21f41484ac9 100644 --- a/include/atomic/nolock.h +++ b/include/atomic/nolock.h @@ -14,19 +14,20 @@ along with this program; if not, write to the Free Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ -#if defined(__i386__) || defined(_M_IX86) - -#ifdef MY_ATOMIC_MODE_DUMMY -# define LOCK "" -#else -# define LOCK "lock" -#endif - -#ifdef __GNUC__ -#include "x86-gcc.h" -#elif defined(_MSC_VER) -#include "x86-msvc.h" -#endif +#if defined(__i386__) || defined(_M_IX86) || defined(__x86_64__) + +# ifdef MY_ATOMIC_MODE_DUMMY +# define LOCK "" +# else +# define LOCK "lock" +# endif + +# ifdef __GNUC__ +# include "x86-gcc.h" +# elif defined(_MSC_VER) +# error Broken! +# include "x86-msvc.h" +# endif #endif #ifdef make_atomic_cas_body diff --git a/include/atomic/rwlock.h b/include/atomic/rwlock.h index 12d0dd3f069..3d8edb9e27e 100644 --- a/include/atomic/rwlock.h +++ b/include/atomic/rwlock.h @@ -14,7 +14,7 @@ along with this program; if not, write to the Free Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ -typedef struct {pthread_rwlock_t rw;} my_atomic_rwlock_t; +typedef struct {pthread_mutex_t rw;} my_atomic_rwlock_t; #ifdef MY_ATOMIC_MODE_DUMMY /* @@ -32,13 +32,18 @@ typedef struct {pthread_rwlock_t rw;} my_atomic_rwlock_t; #define my_atomic_rwlock_wrunlock(name) #define MY_ATOMIC_MODE "dummy (non-atomic)" #else -#define my_atomic_rwlock_destroy(name) pthread_rwlock_destroy(& (name)->rw) -#define my_atomic_rwlock_init(name) pthread_rwlock_init(& (name)->rw, 0) -#define my_atomic_rwlock_rdlock(name) pthread_rwlock_rdlock(& (name)->rw) -#define my_atomic_rwlock_wrlock(name) pthread_rwlock_wrlock(& (name)->rw) -#define my_atomic_rwlock_rdunlock(name) pthread_rwlock_unlock(& (name)->rw) -#define my_atomic_rwlock_wrunlock(name) pthread_rwlock_unlock(& (name)->rw) -#define MY_ATOMIC_MODE "rwlocks" +/* + we're using read-write lock macros but map them to mutex locks, and they're + faster. Still, having semantically rich API we can change the + underlying implementation, if necessary. +*/ +#define my_atomic_rwlock_destroy(name) pthread_mutex_destroy(& (name)->rw) +#define my_atomic_rwlock_init(name) pthread_mutex_init(& (name)->rw, 0) +#define my_atomic_rwlock_rdlock(name) pthread_mutex_lock(& (name)->rw) +#define my_atomic_rwlock_wrlock(name) pthread_mutex_lock(& (name)->rw) +#define my_atomic_rwlock_rdunlock(name) pthread_mutex_unlock(& (name)->rw) +#define my_atomic_rwlock_wrunlock(name) pthread_mutex_unlock(& (name)->rw) +#define MY_ATOMIC_MODE "mutex" #endif #define make_atomic_add_body(S) int ## S sav; sav= *a; *a+= v; v=sav; diff --git a/include/atomic/x86-gcc.h b/include/atomic/x86-gcc.h index 5a72f318a61..3f0a82a9400 100644 --- a/include/atomic/x86-gcc.h +++ b/include/atomic/x86-gcc.h @@ -20,10 +20,18 @@ architectures support double-word (128-bit) cas. */ -#ifdef MY_ATOMIC_NO_XADD -#define MY_ATOMIC_MODE "gcc-x86" LOCK "-no-xadd" +#ifdef __x86_64__ +# ifdef MY_ATOMIC_NO_XADD +# define MY_ATOMIC_MODE "gcc-amd64" LOCK "-no-xadd" +# else +# define MY_ATOMIC_MODE "gcc-amd64" LOCK +# endif #else -#define MY_ATOMIC_MODE "gcc-x86" LOCK +# ifdef MY_ATOMIC_NO_XADD +# define MY_ATOMIC_MODE "gcc-x86" LOCK "-no-xadd" +# else +# define MY_ATOMIC_MODE "gcc-x86" LOCK +# endif #endif /* fix -ansi errors while maintaining readability */ @@ -54,6 +62,9 @@ asm volatile (LOCK "; cmpxchg %2, %0" \ : "+m" (*a), "+a" (ret): "r" (ret)) #define make_atomic_store_body(S) \ - asm volatile ("; xchg %0, %1;" : "+m" (*a) : "r" (v)) + asm volatile ("; xchg %0, %1;" : "+m" (*a), "+r" (v)) #endif +/* TODO test on intel whether the below helps. on AMD it makes no difference */ +//#define LF_BACKOFF ({asm volatile ("rep; nop"); 1; }) + diff --git a/include/lf.h b/include/lf.h new file mode 100644 index 00000000000..299b4e07aa5 --- /dev/null +++ b/include/lf.h @@ -0,0 +1,238 @@ + +/* + TODO + 1. copyright + 6. reduce the number of memory barriers +*/ + +#ifndef _lf_h +#define _lf_h + +#include + +/* + Generic helpers +*/ + +#define lock_wrap(f,t,proto_args, args, lock) \ +t _ ## f proto_args; \ +static inline t f proto_args \ +{ \ + t ret; \ + my_atomic_rwlock_wrlock(lock); \ + ret= _ ## f args; \ + my_atomic_rwlock_wrunlock(lock); \ + return ret; \ +} + +#define lock_wrap_void(f,proto_args, args, lock) \ +void _ ## f proto_args; \ +static inline void f proto_args \ +{ \ + my_atomic_rwlock_wrlock(lock); \ + _ ## f args; \ + my_atomic_rwlock_wrunlock(lock); \ +} + +#define nolock_wrap(f,t,proto_args, args) \ +t _ ## f proto_args; \ +static inline t f proto_args \ +{ \ + return _ ## f args; \ +} + +#define nolock_wrap_void(f,proto_args, args) \ +void _ ## f proto_args; \ +static inline void f proto_args \ +{ \ + _ ## f args; \ +} + +/* + dynamic array + + 4 levels of 256 elements each mean 4311810304 elements in an array - it + should be enough for a while +*/ +#define LF_DYNARRAY_LEVEL_LENGTH 256 +#define LF_DYNARRAY_LEVELS 4 + +typedef struct { + void * volatile level[LF_DYNARRAY_LEVELS]; + uint size_of_element; + my_atomic_rwlock_t lock; +} LF_DYNARRAY; + +typedef int (*lf_dynarray_func)(void *, void *); + +void lf_dynarray_init(LF_DYNARRAY *array, uint element_size); +void lf_dynarray_end(LF_DYNARRAY *array); + +nolock_wrap(lf_dynarray_nr, int, + (LF_DYNARRAY *array, void *el), + (array,el)); + +nolock_wrap(lf_dynarray_value, void *, + (LF_DYNARRAY *array, uint idx), + (array,idx)); + +lock_wrap(lf_dynarray_lvalue, void *, + (LF_DYNARRAY *array, uint idx), + (array,idx), + &array->lock); +nolock_wrap(lf_dynarray_iterate, int, + (LF_DYNARRAY *array, lf_dynarray_func func, void *arg), + (array,func,arg)); + +/* + pin manager for memory allocator +*/ + +#define LF_PINBOX_PINS 3 +#define LF_PURGATORY_SIZE 11 + +typedef void lf_pinbox_free_func(void *, void *); + +typedef struct { + LF_DYNARRAY pinstack; + lf_pinbox_free_func *free_func; + void * free_func_arg; + uint32 volatile pinstack_top_ver; /* this is a versioned pointer */ + uint32 volatile pins_in_stack; /* number of elements in array */ +} LF_PINBOX; + +/* we want sizeof(LF_PINS) to be close to 128 to avoid false sharing */ +typedef struct { + void * volatile pin[LF_PINBOX_PINS]; + void * purgatory[LF_PURGATORY_SIZE]; + LF_PINBOX *pinbox; + uint32 purgatory_count; + uint32 volatile link; + char pad[128-sizeof(uint32)*2 + -sizeof(void *)*(LF_PINBOX_PINS+LF_PURGATORY_SIZE+1)]; +} LF_PINS; + +#define lf_lock_by_pins(PINS) \ + my_atomic_rwlock_wrlock(&(PINS)->pinbox->pinstack.lock) +#define lf_unlock_by_pins(PINS) \ + my_atomic_rwlock_wrunlock(&(PINS)->pinbox->pinstack.lock) + +/* + compile-time assert, to require "no less than N" pins + it's enough if it'll fail on at least one compiler, so + we'll enable it on GCC only, which supports zero-length arrays. +*/ +#if defined(__GNUC__) && defined(MY_LF_EXTRA_DEBUG) +#define LF_REQUIRE_PINS(N) \ + static const char require_pins[LF_PINBOX_PINS-N]; \ + static const int LF_NUM_PINS_IN_THIS_FILE=N; +#define _lf_pin(PINS, PIN, ADDR) \ + ( \ + my_atomic_storeptr(&(PINS)->pin[PIN], (ADDR)), \ + assert(PIN < LF_NUM_PINS_IN_THIS_FILE) \ + ) +#else +#define LF_REQUIRE_PINS(N) +#define _lf_pin(PINS, PIN, ADDR) my_atomic_storeptr(&(PINS)->pin[PIN], (ADDR)) +#endif + +#define _lf_unpin(PINS, PIN) _lf_pin(PINS, PIN, NULL) +#define lf_pin(PINS, PIN, ADDR) \ + do { \ + lf_lock_pins(PINS); \ + _lf_pin(PINS, PIN, ADDR); \ + lf_unlock_pins(PINS); \ + } while (0) +#define lf_unpin(PINS, PIN) lf_pin(PINS, PIN, NULL) + +void lf_pinbox_init(LF_PINBOX *pinbox, lf_pinbox_free_func *free_func, + void * free_func_arg); +void lf_pinbox_end(LF_PINBOX *pinbox); + +lock_wrap(lf_pinbox_get_pins, LF_PINS *, + (LF_PINBOX *pinbox), + (pinbox), + &pinbox->pinstack.lock); +lock_wrap_void(lf_pinbox_put_pins, + (LF_PINS *pins), + (pins), + &pins->pinbox->pinstack.lock); +#if 0 +lock_wrap_void(lf_pinbox_real_free, + (LF_PINS *pins), + (pins), + &pins->pinbox->pinstack.lock); +#endif +lock_wrap_void(lf_pinbox_free, + (LF_PINS *pins, void *addr), + (pins,addr), + &pins->pinbox->pinstack.lock); + +/* + memory allocator +*/ + +typedef struct st_lf_allocator { + LF_PINBOX pinbox; + void * volatile top; + uint element_size; + uint32 volatile mallocs; +} LF_ALLOCATOR; + +void lf_alloc_init(LF_ALLOCATOR *allocator, uint size); +void lf_alloc_end(LF_ALLOCATOR *allocator); +uint lf_alloc_in_pool(LF_ALLOCATOR *allocator); +#define _lf_alloc_free(PINS, PTR) _lf_pinbox_free((PINS), (PTR)) +#define lf_alloc_free(PINS, PTR) lf_pinbox_free((PINS), (PTR)) +#define _lf_alloc_get_pins(ALLOC) _lf_pinbox_get_pins(&(ALLOC)->pinbox) +#define lf_alloc_get_pins(ALLOC) lf_pinbox_get_pins(&(ALLOC)->pinbox) +#define _lf_alloc_put_pins(PINS) _lf_pinbox_put_pins(PINS) +#define lf_alloc_put_pins(PINS) lf_pinbox_put_pins(PINS) +#define lf_alloc_real_free(ALLOC,ADDR) my_free((gptr)(ADDR), MYF(0)) + +lock_wrap(lf_alloc_new, void *, + (LF_PINS *pins), + (pins), + &pins->pinbox->pinstack.lock); + +/* + extendible hash +*/ +#include + +#define LF_HASH_UNIQUE 1 + +typedef struct { + LF_DYNARRAY array; /* hash itself */ + LF_ALLOCATOR alloc; /* allocator for elements */ + hash_get_key get_key; /* see HASH */ + CHARSET_INFO *charset; /* see HASH */ + uint key_offset, key_length; /* see HASH */ + uint element_size, flags; /* LF_HASH_UNIQUE, etc */ + int32 volatile size; /* size of array */ + int32 volatile count; /* number of elements in the hash */ +} LF_HASH; + +void lf_hash_init(LF_HASH *hash, uint element_size, uint flags, + uint key_offset, uint key_length, hash_get_key get_key, + CHARSET_INFO *charset); +void lf_hash_end(LF_HASH *hash); +int lf_hash_insert(LF_HASH *hash, LF_PINS *pins, const void *data); +int lf_hash_search(LF_HASH *hash, LF_PINS *pins, const uchar *key, uint keylen); +int lf_hash_delete(LF_HASH *hash, LF_PINS *pins, const uchar *key, uint keylen); +#define _lf_hash_get_pins(HASH) _lf_alloc_get_pins(&(HASH)->alloc) +#define lf_hash_get_pins(HASH) lf_alloc_get_pins(&(HASH)->alloc) +#define _lf_hash_put_pins(PINS) _lf_pinbox_put_pins(PINS) +#define lf_hash_put_pins(PINS) lf_pinbox_put_pins(PINS) + +/* + cleanup +*/ + +#undef lock_wrap_void +#undef lock_wrap +#undef nolock_wrap_void +#undef nolock_wrap + +#endif + diff --git a/include/my_atomic.h b/include/my_atomic.h index 9a319f84451..d3e4e0055d3 100644 --- a/include/my_atomic.h +++ b/include/my_atomic.h @@ -36,7 +36,7 @@ #ifdef HAVE_INLINE #define make_atomic_add(S) \ -static inline int ## S my_atomic_add ## S( \ +STATIC_INLINE int ## S my_atomic_add ## S( \ int ## S volatile *a, int ## S v) \ { \ make_atomic_add_body(S); \ @@ -44,7 +44,7 @@ static inline int ## S my_atomic_add ## S( \ } #define make_atomic_swap(S) \ -static inline int ## S my_atomic_swap ## S( \ +STATIC_INLINE int ## S my_atomic_swap ## S( \ int ## S volatile *a, int ## S v) \ { \ make_atomic_swap_body(S); \ @@ -52,7 +52,7 @@ static inline int ## S my_atomic_swap ## S( \ } #define make_atomic_cas(S) \ -static inline int my_atomic_cas ## S(int ## S volatile *a, \ +STATIC_INLINE int my_atomic_cas ## S(int ## S volatile *a, \ int ## S *cmp, int ## S set) \ { \ int8 ret; \ @@ -61,7 +61,7 @@ static inline int my_atomic_cas ## S(int ## S volatile *a, \ } #define make_atomic_load(S) \ -static inline int ## S my_atomic_load ## S(int ## S volatile *a) \ +STATIC_INLINE int ## S my_atomic_load ## S(int ## S volatile *a) \ { \ int ## S ret; \ make_atomic_load_body(S); \ @@ -69,7 +69,7 @@ static inline int ## S my_atomic_load ## S(int ## S volatile *a) \ } #define make_atomic_store(S) \ -static inline void my_atomic_store ## S( \ +STATIC_INLINE void my_atomic_store ## S( \ int ## S volatile *a, int ## S v) \ { \ make_atomic_store_body(S); \ @@ -135,6 +135,10 @@ make_atomic_swap(ptr) #undef _atomic_h_cleanup_ #endif +#ifndef LF_BACKOFF +#define LF_BACKOFF (1) +#endif + #if SIZEOF_CHARP == SIZEOF_INT typedef int intptr; #elif SIZEOF_CHARP == SIZEOF_LONG diff --git a/include/my_bit.h b/include/my_bit.h new file mode 100644 index 00000000000..71bbe2d4ba3 --- /dev/null +++ b/include/my_bit.h @@ -0,0 +1,107 @@ +/* + Some useful bit functions +*/ + +#ifdef HAVE_INLINE + +extern const char _my_bits_nbits[256]; +extern const uchar _my_bits_reverse_table[256]; + +/* + Find smallest X in 2^X >= value + This can be used to divide a number with value by doing a shift instead +*/ + +STATIC_INLINE uint my_bit_log2(ulong value) +{ + uint bit; + for (bit=0 ; value > 1 ; value>>=1, bit++) ; + return bit; +} + +STATIC_INLINE uint my_count_bits(ulonglong v) +{ +#if SIZEOF_LONG_LONG > 4 + /* The following code is a bit faster on 16 bit machines than if we would + only shift v */ + ulong v2=(ulong) (v >> 32); + return (uint) (uchar) (_my_bits_nbits[(uchar) v] + + _my_bits_nbits[(uchar) (v >> 8)] + + _my_bits_nbits[(uchar) (v >> 16)] + + _my_bits_nbits[(uchar) (v >> 24)] + + _my_bits_nbits[(uchar) (v2)] + + _my_bits_nbits[(uchar) (v2 >> 8)] + + _my_bits_nbits[(uchar) (v2 >> 16)] + + _my_bits_nbits[(uchar) (v2 >> 24)]); +#else + return (uint) (uchar) (_my_bits_nbits[(uchar) v] + + _my_bits_nbits[(uchar) (v >> 8)] + + _my_bits_nbits[(uchar) (v >> 16)] + + _my_bits_nbits[(uchar) (v >> 24)]); +#endif +} + +STATIC_INLINE uint my_count_bits_ushort(ushort v) +{ + return _my_bits_nbits[v]; +} + + +/* + Next highest power of two + + SYNOPSIS + my_round_up_to_next_power() + v Value to check + + RETURN + Next or equal power of 2 + Note: 0 will return 0 + + NOTES + Algorithm by Sean Anderson, according to: + http://graphics.stanford.edu/~seander/bithacks.html + (Orignal code public domain) + + Comments shows how this works with 01100000000000000000000000001011 +*/ + +STATIC_INLINE uint32 my_round_up_to_next_power(uint32 v) +{ + v--; /* 01100000000000000000000000001010 */ + v|= v >> 1; /* 01110000000000000000000000001111 */ + v|= v >> 2; /* 01111100000000000000000000001111 */ + v|= v >> 4; /* 01111111110000000000000000001111 */ + v|= v >> 8; /* 01111111111111111100000000001111 */ + v|= v >> 16; /* 01111111111111111111111111111111 */ + return v+1; /* 10000000000000000000000000000000 */ +} + +STATIC_INLINE uint32 my_clear_highest_bit(uint32 v) +{ + uint32 w=v >> 1; + w|= w >> 1; + w|= w >> 2; + w|= w >> 4; + w|= w >> 8; + w|= w >> 16; + return v & w; +} + +STATIC_INLINE uint32 my_reverse_bits(uint key) +{ + return + (_my_bits_reverse_table[ key & 255] << 24) | + (_my_bits_reverse_table[(key>> 8) & 255] << 16) | + (_my_bits_reverse_table[(key>>16) & 255] << 8) | + _my_bits_reverse_table[(key>>24) ]; +} + +#else +extern uint my_bit_log2(ulong value); +extern uint32 my_round_up_to_next_power(uint32 v); +uint32 my_clear_highest_bit(uint32 v); +uint32 my_reverse_bits(uint key); +extern uint my_count_bits(ulonglong v); +extern uint my_count_bits_ushort(ushort v); +#endif diff --git a/include/my_global.h b/include/my_global.h index 21b2251af1d..563ef379252 100644 --- a/include/my_global.h +++ b/include/my_global.h @@ -192,6 +192,8 @@ #endif #undef inline_test_2 #undef inline_test_1 +/* helper macro for "instantiating" inline functions */ +#define STATIC_INLINE static inline /* The following macros are used to control inlining a bit more than diff --git a/include/my_sys.h b/include/my_sys.h index 2dc4053f70d..81e5380be96 100644 --- a/include/my_sys.h +++ b/include/my_sys.h @@ -216,6 +216,7 @@ extern int (*error_handler_hook)(uint my_err, const char *str,myf MyFlags); extern int (*fatal_error_handler_hook)(uint my_err, const char *str, myf MyFlags); extern uint my_file_limit; +extern ulong my_thread_stack_size; #ifdef HAVE_LARGE_PAGES extern my_bool my_use_large_pages; @@ -821,10 +822,6 @@ extern int packfrm(const void *, uint, const void **, uint *); extern int unpackfrm(const void **, uint *, const void *); extern ha_checksum my_checksum(ha_checksum crc, const byte *mem, uint count); -extern uint my_bit_log2(ulong value); -extern uint32 my_round_up_to_next_power(uint32 v); -extern uint my_count_bits(ulonglong v); -extern uint my_count_bits_ushort(ushort v); extern void my_sleep(ulong m_seconds); extern ulong crc32(ulong crc, const uchar *buf, uint len); extern uint my_set_max_open_files(uint files); @@ -840,7 +837,7 @@ extern int my_getncpus(); #ifndef MAP_NOSYNC #define MAP_NOSYNC 0 #endif -#ifndef MAP_NORESERVE +#ifndef MAP_NORESERVE #define MAP_NORESERVE 0 /* For irix and AIX */ #endif diff --git a/mysys/Makefile.am b/mysys/Makefile.am index b209d64e78f..d870437573a 100644 --- a/mysys/Makefile.am +++ b/mysys/Makefile.am @@ -31,7 +31,8 @@ libmysys_a_SOURCES = my_init.c my_getwd.c mf_getdate.c my_mmap.c \ mf_tempdir.c my_lock.c mf_brkhant.c my_alarm.c \ my_malloc.c my_realloc.c my_once.c mulalloc.c \ my_alloc.c safemalloc.c my_new.cc \ - my_vle.c my_atomic.c \ + my_vle.c my_atomic.c lf_hash.c \ + lf_dynarray.c lf_alloc-pin.c \ my_fopen.c my_fstream.c my_getsystime.c \ my_error.c errors.c my_div.c my_messnc.c \ mf_format.c mf_same.c mf_dirname.c mf_fn_ext.c \ diff --git a/mysys/lf_alloc-pin.c b/mysys/lf_alloc-pin.c new file mode 100644 index 00000000000..a9ea1802c03 --- /dev/null +++ b/mysys/lf_alloc-pin.c @@ -0,0 +1,319 @@ +/* Copyright (C) 2000 MySQL AB + + 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; either version 2 of the License, or + (at your option) any later version. + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ + +/* + concurrent allocator based on pinning addresses + + strictly speaking it's not lock-free, as it can be blocked + if a thread's purgatory is full and all addresses from there + are pinned. + + But until the above happens, it's wait-free. + + It can be made strictly wait-free by increasing purgatory size. + If it's larger than pins_in_stack*LF_PINBOX_PINS, then apocalyptical + condition above will never happen. But than the memory requirements + will be O(pins_in_stack^2). + + Note, that for large purgatory sizes it makes sense to remove + purgatory array, and link objects in a list using embedded pointer. + + TODO test with more than 256 threads + TODO test w/o alloca +*/ + +#include +#include +#include + +#define LF_PINBOX_MAX_PINS 65536 + +static void _lf_pinbox_real_free(LF_PINS *pins); + +void lf_pinbox_init(LF_PINBOX *pinbox, lf_pinbox_free_func *free_func, + void *free_func_arg) +{ + DBUG_ASSERT(sizeof(LF_PINS) == 128); + lf_dynarray_init(&pinbox->pinstack, sizeof(LF_PINS)); + pinbox->pinstack_top_ver=0; + pinbox->pins_in_stack=0; + pinbox->free_func=free_func; + pinbox->free_func_arg=free_func_arg; +} + +void lf_pinbox_end(LF_PINBOX *pinbox) +{ + lf_dynarray_end(&pinbox->pinstack); +} + +LF_PINS *_lf_pinbox_get_pins(LF_PINBOX *pinbox) +{ + uint32 pins, next, top_ver; + LF_PINS *el; + + top_ver=pinbox->pinstack_top_ver; + do + { + if (!(pins=top_ver % LF_PINBOX_MAX_PINS)) + { + pins=my_atomic_add32(&pinbox->pins_in_stack, 1)+1; + el=(LF_PINS *)_lf_dynarray_lvalue(&pinbox->pinstack, pins); + break; + } + el=(LF_PINS *)_lf_dynarray_value(&pinbox->pinstack, pins); + next=el->link; + } while (!my_atomic_cas32(&pinbox->pinstack_top_ver, &top_ver, + top_ver-pins+next+LF_PINBOX_MAX_PINS)); + el->link=pins; + el->purgatory_count=0; + el->pinbox=pinbox; + return el; +} + +void _lf_pinbox_put_pins(LF_PINS *pins) +{ + LF_PINBOX *pinbox=pins->pinbox; + uint32 top_ver, nr; + nr=pins->link; +#ifdef MY_LF_EXTRA_DEBUG + { + int i; + for (i=0; i < LF_PINBOX_PINS; i++) + assert(pins->pin[i] == 0); + } +#endif + while (pins->purgatory_count) + { + _lf_pinbox_real_free(pins); + if (pins->purgatory_count && my_getncpus() == 1) + { + my_atomic_rwlock_wrunlock(&pins->pinbox->pinstack.lock); + pthread_yield(); + my_atomic_rwlock_wrlock(&pins->pinbox->pinstack.lock); + } + } + top_ver=pinbox->pinstack_top_ver; + if (nr == pinbox->pins_in_stack) + { + int32 tmp=nr; + if (my_atomic_cas32(&pinbox->pins_in_stack, &tmp, tmp-1)) + goto ret; + } + + do + { + pins->link=top_ver % LF_PINBOX_MAX_PINS; + } while (!my_atomic_cas32(&pinbox->pinstack_top_ver, &top_ver, + top_ver-pins->link+nr+LF_PINBOX_MAX_PINS)); +ret: + return; +} + +static int ptr_cmp(void **a, void **b) +{ + return *a < *b ? -1 : *a == *b ? 0 : 1; +} + +void _lf_pinbox_free(LF_PINS *pins, void *addr) +{ + while (pins->purgatory_count == LF_PURGATORY_SIZE) + { + _lf_pinbox_real_free(pins); + if (pins->purgatory_count == LF_PURGATORY_SIZE && my_getncpus() == 1) + { + my_atomic_rwlock_wrunlock(&pins->pinbox->pinstack.lock); + pthread_yield(); + my_atomic_rwlock_wrlock(&pins->pinbox->pinstack.lock); + } + } + pins->purgatory[pins->purgatory_count++]=addr; +} + +struct st_harvester { + void **granary; + int npins; +}; + +static int harvest_pins(LF_PINS *el, struct st_harvester *hv) +{ + int i; + LF_PINS *el_end= el+min(hv->npins, LF_DYNARRAY_LEVEL_LENGTH); + for (; el < el_end; el++) + { + for (i= 0; i < LF_PINBOX_PINS; i++) + { + void *p= el->pin[i]; + if (p) + *hv->granary++= p; + } + } + hv->npins-= LF_DYNARRAY_LEVEL_LENGTH; + return 0; +} + +static int match_pins(LF_PINS *el, void *addr) +{ + int i; + LF_PINS *el_end= el+LF_DYNARRAY_LEVEL_LENGTH; + for (; el < el_end; el++) + for (i= 0; i < LF_PINBOX_PINS; i++) + if (el->pin[i] == addr) + return 1; + return 0; +} + +static void _lf_pinbox_real_free(LF_PINS *pins) +{ + int npins; + void **addr=0; + void **start, **cur, **end=pins->purgatory+pins->purgatory_count; + LF_PINBOX *pinbox=pins->pinbox; + + npins=pinbox->pins_in_stack+1; + +#ifdef HAVE_ALLOCA + /* create a sorted list of pinned addresses, to speed up searches */ + if (sizeof(void *)*LF_PINBOX_PINS*npins < my_thread_stack_size) + { + struct st_harvester hv; + addr= (void **) alloca(sizeof(void *)*LF_PINBOX_PINS*npins); + hv.granary=addr; + hv.npins=npins; + _lf_dynarray_iterate(&pinbox->pinstack, + (lf_dynarray_func)harvest_pins, &hv); + + npins=hv.granary-addr; + if (npins) + qsort(addr, npins, sizeof(void *), (qsort_cmp)ptr_cmp); + } +#endif + + start= cur= pins->purgatory; + end= start+pins->purgatory_count; + for (; cur < end; cur++) + { + if (npins) + { + if (addr) + { + void **a,**b,**c; + for (a=addr, b=addr+npins-1, c=a+(b-a)/2; b-a>1; c=a+(b-a)/2) + if (*cur == *c) + a=b=c; + else if (*cur > *c) + a=c; + else + b=c; + if (*cur == *a || *cur == *b) + goto found; + } + else + { + if (_lf_dynarray_iterate(&pinbox->pinstack, + (lf_dynarray_func)match_pins, *cur)) + goto found; + } + } + /* not pinned - freeing */ + pinbox->free_func(*cur, pinbox->free_func_arg); + continue; +found: + /* pinned - keeping */ + *start++=*cur; + } + pins->purgatory_count=start-pins->purgatory; +#ifdef MY_LF_EXTRA_DEBUG + while (start < pins->purgatory + LF_PURGATORY_SIZE) + *start++=0; +#endif +} + +static void alloc_free(void *node, LF_ALLOCATOR *allocator) +{ + void *tmp; + tmp=allocator->top; + do + { + (*(void **)node)=tmp; + } while (!my_atomic_casptr((void **)&allocator->top, (void **)&tmp, node) && + LF_BACKOFF); +} + +LF_REQUIRE_PINS(1); +void *_lf_alloc_new(LF_PINS *pins) +{ + LF_ALLOCATOR *allocator=(LF_ALLOCATOR *)(pins->pinbox->free_func_arg); + void *node; + for (;;) + { + do + { + node=allocator->top; + _lf_pin(pins, 0, node); + } while (node !=allocator->top && LF_BACKOFF); + if (!node) + { + if (!(node=my_malloc(allocator->element_size, MYF(MY_WME|MY_ZEROFILL)))) + goto ret; +#ifdef MY_LF_EXTRA_DEBUG + my_atomic_add32(&allocator->mallocs, 1); +#endif + goto ret; + } + if (my_atomic_casptr((void **)&allocator->top, + (void *)&node, *(void **)node)) + goto ret; + } +ret: + _lf_unpin(pins, 0); + return node; +} + +void lf_alloc_init(LF_ALLOCATOR *allocator, uint size) +{ + lf_pinbox_init(&allocator->pinbox, + (lf_pinbox_free_func *)alloc_free, allocator); + allocator->top=0; + allocator->mallocs=0; + allocator->element_size=size; + DBUG_ASSERT(size >= (int)sizeof(void *)); +} + +void lf_alloc_end(LF_ALLOCATOR *allocator) +{ + void *el=allocator->top; + while (el) + { + void *tmp=*(void **)el; + my_free(el, MYF(0)); + el=tmp; + } + lf_pinbox_end(&allocator->pinbox); + allocator->top=0; +} + +/* + NOTE + this is NOT thread-safe !!! +*/ +uint lf_alloc_in_pool(LF_ALLOCATOR *allocator) +{ + uint i; + void *node; + for (node=allocator->top, i=0; node; node=*(void **)node, i++) /* no op */; + return i; +} + diff --git a/mysys/lf_dynarray.c b/mysys/lf_dynarray.c new file mode 100644 index 00000000000..dcf99163ca1 --- /dev/null +++ b/mysys/lf_dynarray.c @@ -0,0 +1,186 @@ +/* Copyright (C) 2000 MySQL AB + + 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; either version 2 of the License, or + (at your option) any later version. + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ + +/* + Analog of DYNAMIC_ARRAY that never reallocs + (so no pointer into the array may ever become invalid). + + Memory is allocated in non-contiguous chunks. + This data structure is not space efficient for sparce arrays. + + The number of elements is limited to 2^16 + + Every element is aligned to sizeof(element) boundary + (to avoid false sharing if element is big enough). + + Actually, it's wait-free, not lock-free ;-) +*/ + +#undef DBUG_OFF +#include +#include +#include +#include + +void lf_dynarray_init(LF_DYNARRAY *array, uint element_size) +{ + bzero(array, sizeof(*array)); + array->size_of_element=element_size; + my_atomic_rwlock_init(&array->lock); +} + +static void recursive_free(void **alloc, int level) +{ + if (!alloc) return; + + if (level) + { + int i; + for (i=0; i < LF_DYNARRAY_LEVEL_LENGTH; i++) + recursive_free(alloc[i], level-1); + my_free((void *)alloc, MYF(0)); + } + else + my_free(alloc[-1], MYF(0)); +} + +void lf_dynarray_end(LF_DYNARRAY *array) +{ + int i; + for (i=0; i < LF_DYNARRAY_LEVELS; i++) + recursive_free(array->level[i], i); + my_atomic_rwlock_destroy(&array->lock); + bzero(array, sizeof(*array)); +} + +static const int dynarray_idxes_in_level[LF_DYNARRAY_LEVELS]= +{ + 0, /* +1 here to to avoid -1's below */ + LF_DYNARRAY_LEVEL_LENGTH, + LF_DYNARRAY_LEVEL_LENGTH * LF_DYNARRAY_LEVEL_LENGTH, + LF_DYNARRAY_LEVEL_LENGTH * LF_DYNARRAY_LEVEL_LENGTH * + LF_DYNARRAY_LEVEL_LENGTH +}; + +void *_lf_dynarray_lvalue(LF_DYNARRAY *array, uint idx) +{ + void * ptr, * volatile * ptr_ptr=0; + int i; + + for (i=3; i > 0; i--) + { + if (ptr_ptr || idx >= dynarray_idxes_in_level[i]) + { + if (!ptr_ptr) + { + ptr_ptr=&array->level[i]; + idx-= dynarray_idxes_in_level[i]; + } + ptr=*ptr_ptr; + if (!ptr) + { + void *alloc=my_malloc(LF_DYNARRAY_LEVEL_LENGTH * sizeof(void *), + MYF(MY_WME|MY_ZEROFILL)); + if (!alloc) + return(NULL); + if (my_atomic_casptr(ptr_ptr, &ptr, alloc)) + ptr= alloc; + else + my_free(alloc, MYF(0)); + } + ptr_ptr=((void **)ptr) + idx / dynarray_idxes_in_level[i]; + idx%= dynarray_idxes_in_level[i]; + } + } + if (!ptr_ptr) + ptr_ptr=&array->level[0]; + ptr=*ptr_ptr; + if (!ptr) + { + void *alloc, *data; + alloc=my_malloc(LF_DYNARRAY_LEVEL_LENGTH * array->size_of_element + + max(array->size_of_element, sizeof(void *)), + MYF(MY_WME|MY_ZEROFILL)); + if (!alloc) + return(NULL); + /* reserve the space for free() address */ + data= alloc + sizeof(void *); + { /* alignment */ + intptr mod= ((intptr)data) % array->size_of_element; + if (mod) + data+= array->size_of_element - mod; + } + ((void **)data)[-1]=alloc; /* free() will need the original pointer */ + if (my_atomic_casptr(ptr_ptr, &ptr, data)) + ptr= data; + else + my_free(alloc, MYF(0)); + } + return ptr + array->size_of_element * idx; +} + +void *_lf_dynarray_value(LF_DYNARRAY *array, uint idx) +{ + void * ptr, * volatile * ptr_ptr=0; + int i; + + for (i=3; i > 0; i--) + { + if (ptr_ptr || idx >= dynarray_idxes_in_level[i]) + { + if (!ptr_ptr) + { + ptr_ptr=&array->level[i]; + idx-= dynarray_idxes_in_level[i]; + } + ptr=*ptr_ptr; + if (!ptr) + return(NULL); + ptr_ptr=((void **)ptr) + idx / dynarray_idxes_in_level[i]; + idx %= dynarray_idxes_in_level[i]; + } + } + if (!ptr_ptr) + ptr_ptr=&array->level[0]; + ptr=*ptr_ptr; + if (!ptr) + return(NULL); + return ptr + array->size_of_element * idx; +} + +static int recursive_iterate(LF_DYNARRAY *array, void *ptr, int level, + lf_dynarray_func func, void *arg) +{ + int res, i; + if (!ptr) + return 0; + if (!level) + return func(ptr, arg); + for (i=0; i < LF_DYNARRAY_LEVEL_LENGTH; i++) + if ((res=recursive_iterate(array, ((void **)ptr)[i], level-1, func, arg))) + return res; + return 0; +} + +int _lf_dynarray_iterate(LF_DYNARRAY *array, lf_dynarray_func func, void *arg) +{ + int i, res; + for (i=0; i < LF_DYNARRAY_LEVELS; i++) + if ((res=recursive_iterate(array, array->level[i], i, func, arg))) + return res; + return 0; +} + diff --git a/mysys/lf_hash.c b/mysys/lf_hash.c new file mode 100644 index 00000000000..6d3d30ebc3f --- /dev/null +++ b/mysys/lf_hash.c @@ -0,0 +1,370 @@ +/* Copyright (C) 2000 MySQL AB + + 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; either version 2 of the License, or + (at your option) any later version. + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ + +/* + extensible hash + + TODO + dummy nodes use element_size=0 + try to get rid of dummy nodes ? +*/ +#include +#include +#include +#include + +LF_REQUIRE_PINS(3); + +typedef struct { + intptr volatile link; + uint32 hashnr; + const uchar *key; + uint keylen; +} LF_SLIST; + +typedef struct { + intptr volatile *prev; + LF_SLIST *curr, *next; +} CURSOR; + +#define PTR(V) (LF_SLIST *)((V) & (~(intptr)1)) +#define DELETED(V) ((V) & 1) + +/* + RETURN + 0 - not found + 1 - found + + NOTE + cursor is positioned in either case + pins[0..2] are used, they are NOT removed on return +*/ +static int lfind(intptr volatile *head, uint32 hashnr, + const uchar *key, uint keylen, CURSOR *cursor, LF_PINS *pins) +{ + uint32 cur_hashnr; + const uchar *cur_key; + uint cur_keylen; + intptr link; + +retry: + cursor->prev=head; + do { + cursor->curr=PTR(*cursor->prev); + _lf_pin(pins,1,cursor->curr); + } while(*cursor->prev != (intptr)cursor->curr && LF_BACKOFF); + for (;;) + { + if (!cursor->curr) + return 0; + do { // XXX or goto retry ? + link=cursor->curr->link; + cursor->next=PTR(link); + _lf_pin(pins, 0, cursor->next); + } while(link != cursor->curr->link && LF_BACKOFF); + cur_hashnr=cursor->curr->hashnr; + cur_key=cursor->curr->key; + cur_keylen=cursor->curr->keylen; + if (*cursor->prev != (intptr)cursor->curr) + { + LF_BACKOFF; + goto retry; + } + if (!DELETED(link)) + { + if (cur_hashnr >= hashnr) + { + int r=1; + if (cur_hashnr > hashnr || (r=memcmp(cur_key, key, keylen)) >= 0) + return !r; + } + cursor->prev=&(cursor->curr->link); + _lf_pin(pins, 2, cursor->curr); + } + else + { + if (my_atomic_casptr((void **)cursor->prev, + (void **)&cursor->curr, cursor->next)) + _lf_alloc_free(pins, cursor->curr); + else + { + LF_BACKOFF; + goto retry; + } + } + cursor->curr=cursor->next; + _lf_pin(pins, 1, cursor->curr); + } +} + +/* + RETURN + 0 - inserted + not 0 - a pointer to a conflict + + NOTE + it uses pins[0..2], on return all pins are removed. +*/ +static LF_SLIST *linsert(LF_SLIST * volatile *head, LF_SLIST *node, + LF_PINS *pins, uint flags) +{ + CURSOR cursor; + int res=-1; + + do + { + if (lfind((intptr*)head, node->hashnr, node->key, node->keylen, + &cursor, pins) && + (flags & LF_HASH_UNIQUE)) + res=0; + else + { + node->link=(intptr)cursor.curr; + assert(node->link != (intptr)node); + assert(cursor.prev != &node->link); + if (my_atomic_casptr((void **)cursor.prev, (void **)&cursor.curr, node)) + res=1; + } + } while (res == -1); + _lf_unpin(pins, 0); + _lf_unpin(pins, 1); + _lf_unpin(pins, 2); + return res ? 0 : cursor.curr; +} + +/* + RETURN + 0 - ok + 1 - not found + NOTE + it uses pins[0..2], on return all pins are removed. +*/ +static int ldelete(LF_SLIST * volatile *head, uint32 hashnr, + const uchar *key, uint keylen, LF_PINS *pins) +{ + CURSOR cursor; + int res=-1; + + do + { + if (!lfind((intptr *)head, hashnr, key, keylen, &cursor, pins)) + res= 1; + else + if (my_atomic_casptr((void **)&(cursor.curr->link), + (void **)&cursor.next, 1+(char *)cursor.next)) + { + if (my_atomic_casptr((void **)cursor.prev, + (void **)&cursor.curr, cursor.next)) + _lf_alloc_free(pins, cursor.curr); + else + lfind((intptr *)head, hashnr, key, keylen, &cursor, pins); + res= 0; + } + } while (res == -1); + _lf_unpin(pins, 0); + _lf_unpin(pins, 1); + _lf_unpin(pins, 2); + return res; +} + +/* + RETURN + 0 - not found + node - found + NOTE + it uses pins[0..2], on return the pin[2] keeps the node found + all other pins are removed. +*/ +static LF_SLIST *lsearch(LF_SLIST * volatile *head, uint32 hashnr, + const uchar *key, uint keylen, LF_PINS *pins) +{ + CURSOR cursor; + int res=lfind((intptr *)head, hashnr, key, keylen, &cursor, pins); + if (res) _lf_pin(pins, 2, cursor.curr); + _lf_unpin(pins, 0); + _lf_unpin(pins, 1); + return res ? cursor.curr : 0; +} + +static inline const uchar* hash_key(const LF_HASH *hash, + const uchar *record, uint *length) +{ + if (hash->get_key) + return (*hash->get_key)(record,length,0); + *length=hash->key_length; + return record + hash->key_offset; +} + +static inline uint calc_hash(LF_HASH *hash, const uchar *key, uint keylen) +{ + ulong nr1=1, nr2=4; + hash->charset->coll->hash_sort(hash->charset,key,keylen,&nr1,&nr2); + return nr1 & INT_MAX32; +} + +#define MAX_LOAD 1 +static void initialize_bucket(LF_HASH *, LF_SLIST * volatile*, uint, LF_PINS *); + +void lf_hash_init(LF_HASH *hash, uint element_size, uint flags, + uint key_offset, uint key_length, hash_get_key get_key, + CHARSET_INFO *charset) +{ + lf_alloc_init(&hash->alloc,sizeof(LF_SLIST)+element_size); + lf_dynarray_init(&hash->array, sizeof(LF_SLIST **)); + hash->size=1; + hash->count=0; + hash->element_size=element_size; + hash->flags=flags; + hash->charset=charset; + hash->key_offset=key_offset; + hash->key_length=key_length; + hash->get_key=get_key; + DBUG_ASSERT(get_key ? !key_offset && !key_length : key_length); +} + +void lf_hash_end(LF_HASH *hash) +{ + LF_SLIST *el=*(LF_SLIST **)_lf_dynarray_lvalue(&hash->array, 0); + while (el) + { + intptr next=el->link; + lf_alloc_real_free(&hash->alloc, el); + el=(LF_SLIST *)next; + } + lf_alloc_end(&hash->alloc); + lf_dynarray_end(&hash->array); +} + +/* + NOTE + see linsert() for pin usage +*/ +int lf_hash_insert(LF_HASH *hash, LF_PINS *pins, const void *data) +{ + uint csize, bucket, hashnr, keylen; + LF_SLIST *node, * volatile *el; + const uchar *key; + + key= hash_key(hash, data, &keylen); + hashnr= calc_hash(hash, key, keylen); + bucket= hashnr % hash->size; + lf_lock_by_pins(pins); + node=(LF_SLIST *)_lf_alloc_new(pins); + memcpy(node+1, data, hash->element_size); + el=_lf_dynarray_lvalue(&hash->array, bucket); + if (*el == NULL) + initialize_bucket(hash, el, bucket, pins); + node->hashnr=my_reverse_bits(hashnr) | 1; + node->key=((char *)(node+1))+(key-(uchar *)data); + node->keylen=keylen; + if (linsert(el, node, pins, hash->flags)) + { + _lf_alloc_free(pins, node); + lf_unlock_by_pins(pins); + return 0; + } + csize= hash->size; + if ((my_atomic_add32(&hash->count, 1)+1.0) / csize > MAX_LOAD) + my_atomic_cas32(&hash->size, &csize, csize*2); +#if 0 + node=*(LF_SLIST **)_lf_dynarray_lvalue(&hash->array, 0); + hashnr=0; + while (node) + { + assert (node->hashnr >= hashnr); + hashnr=node->hashnr; + node=(LF_SLIST *)node->link; + } +#endif + lf_unlock_by_pins(pins); + return 1; +} + +/* + NOTE + see ldelete() for pin usage +*/ +int lf_hash_delete(LF_HASH *hash, LF_PINS *pins, const uchar *key, uint keylen) +{ + LF_SLIST * volatile *el; + uint bucket, hashnr=calc_hash(hash, key, keylen); + + bucket= hashnr % hash->size; + lf_lock_by_pins(pins); + el=_lf_dynarray_lvalue(&hash->array, bucket); + if (*el == NULL) + initialize_bucket(hash, el, bucket, pins); + if (ldelete(el, my_reverse_bits(hashnr) | 1, key, keylen, pins)) + { + lf_unlock_by_pins(pins); + return 0; + } + my_atomic_add32(&hash->count, -1); +#if 0 + { + LF_SLIST *node=*(LF_SLIST **)_lf_dynarray_lvalue(&hash->array, 0); + hashnr=0; + while (node) + { + assert (node->hashnr >= hashnr); + hashnr=node->hashnr; + node=(LF_SLIST *)node->link; + } + } +#endif + lf_unlock_by_pins(pins); + return 1; +} + +/* + NOTE + see lsearch() for pin usage +*/ +int lf_hash_search(LF_HASH *hash, LF_PINS *pins, const uchar *key, uint keylen) +{ + int res; + LF_SLIST * volatile *el; + uint bucket, hashnr=calc_hash(hash, key, keylen); + + bucket= hashnr % hash->size; + lf_lock_by_pins(pins); + el=_lf_dynarray_lvalue(&hash->array, bucket); + if (*el == NULL) + initialize_bucket(hash, el, bucket, pins); + res=NULL != lsearch(el, my_reverse_bits(hashnr) | 1, key, keylen, pins); + lf_unlock_by_pins(pins); + return res; +} + +static void initialize_bucket(LF_HASH *hash, LF_SLIST * volatile *node, + uint bucket, LF_PINS *pins) +{ + uint parent= my_clear_highest_bit(bucket); + LF_SLIST *dummy=_lf_alloc_new(pins), **tmp=0, *cur; + LF_SLIST * volatile *el=_lf_dynarray_lvalue(&hash->array, parent); + if (*el == NULL && bucket) + initialize_bucket(hash, el, parent, pins); + dummy->hashnr=my_reverse_bits(bucket); + LINT_INIT(dummy->key); + LINT_INIT(dummy->keylen); + if ((cur= linsert(el, dummy, pins, 0))) + { + _lf_alloc_free(pins, dummy); + dummy= cur; + } + my_atomic_casptr((void **)node, (void **)&tmp, dummy); +} + diff --git a/mysys/mf_keycache.c b/mysys/mf_keycache.c index e6f4348968f..9a99a278bc5 100644 --- a/mysys/mf_keycache.c +++ b/mysys/mf_keycache.c @@ -43,6 +43,7 @@ #include #include "my_static.h" #include +#include #include #include diff --git a/mysys/my_atomic.c b/mysys/my_atomic.c index 6f56d76e6b8..2908a44961a 100644 --- a/mysys/my_atomic.c +++ b/mysys/my_atomic.c @@ -18,11 +18,10 @@ #include #ifndef HAVE_INLINE -/* - the following will cause all inline functions to be instantiated -*/ +/* the following will cause all inline functions to be instantiated */ #define HAVE_INLINE -#define static extern +#undef STATIC_INLINE +#define STATIC_INLINE extern #endif #include diff --git a/mysys/my_bit.c b/mysys/my_bit.c index 6ef0e171695..11d98f5f6ae 100644 --- a/mysys/my_bit.c +++ b/mysys/my_bit.c @@ -14,23 +14,18 @@ along with this program; if not, write to the Free Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ -/* Some useful bit functions */ +#include -#include "mysys_priv.h" - -/* - Find smallest X in 2^X >= value - This can be used to divide a number with value by doing a shift instead -*/ +#ifndef HAVE_INLINE +/* the following will cause all inline functions to be instantiated */ +#define HAVE_INLINE +#undef STATIC_INLINE +#define STATIC_INLINE extern +#endif -uint my_bit_log2(ulong value) -{ - uint bit; - for (bit=0 ; value > 1 ; value>>=1, bit++) ; - return bit; -} +#include -static char nbits[256] = { +const char _my_bits_nbits[256] = { 0, 1, 1, 2, 1, 2, 2, 3, 1, 2, 2, 3, 2, 3, 3, 4, 1, 2, 2, 3, 2, 3, 3, 4, 2, 3, 3, 4, 3, 4, 4, 5, 1, 2, 2, 3, 2, 3, 3, 4, 2, 3, 3, 4, 3, 4, 4, 5, @@ -49,60 +44,29 @@ static char nbits[256] = { 4, 5, 5, 6, 5, 6, 6, 7, 5, 6, 6, 7, 6, 7, 7, 8, }; -uint my_count_bits(ulonglong v) -{ -#if SIZEOF_LONG_LONG > 4 - /* The following code is a bit faster on 16 bit machines than if we would - only shift v */ - ulong v2=(ulong) (v >> 32); - return (uint) (uchar) (nbits[(uchar) v] + - nbits[(uchar) (v >> 8)] + - nbits[(uchar) (v >> 16)] + - nbits[(uchar) (v >> 24)] + - nbits[(uchar) (v2)] + - nbits[(uchar) (v2 >> 8)] + - nbits[(uchar) (v2 >> 16)] + - nbits[(uchar) (v2 >> 24)]); -#else - return (uint) (uchar) (nbits[(uchar) v] + - nbits[(uchar) (v >> 8)] + - nbits[(uchar) (v >> 16)] + - nbits[(uchar) (v >> 24)]); -#endif -} - -uint my_count_bits_ushort(ushort v) -{ - return nbits[v]; -} - - /* - Next highest power of two - - SYNOPSIS - my_round_up_to_next_power() - v Value to check - - RETURN - Next or equal power of 2 - Note: 0 will return 0 - - NOTES - Algorithm by Sean Anderson, according to: - http://graphics.stanford.edu/~seander/bithacks.html - (Orignal code public domain) - - Comments shows how this works with 01100000000000000000000000001011 + perl -e 'print map{", 0x".unpack H2,pack B8,unpack b8,chr$_}(0..255)' */ +const uchar _my_bits_reverse_table[256]={ +0x00, 0x80, 0x40, 0xC0, 0x20, 0xA0, 0x60, 0xE0, 0x10, 0x90, 0x50, 0xD0, 0x30, +0xB0, 0x70, 0xF0, 0x08, 0x88, 0x48, 0xC8, 0x28, 0xA8, 0x68, 0xE8, 0x18, 0x98, +0x58, 0xD8, 0x38, 0xB8, 0x78, 0xF8, 0x04, 0x84, 0x44, 0xC4, 0x24, 0xA4, 0x64, +0xE4, 0x14, 0x94, 0x54, 0xD4, 0x34, 0xB4, 0x74, 0xF4, 0x0C, 0x8C, 0x4C, 0xCC, +0x2C, 0xAC, 0x6C, 0xEC, 0x1C, 0x9C, 0x5C, 0xDC, 0x3C, 0xBC, 0x7C, 0xFC, 0x02, +0x82, 0x42, 0xC2, 0x22, 0xA2, 0x62, 0xE2, 0x12, 0x92, 0x52, 0xD2, 0x32, 0xB2, +0x72, 0xF2, 0x0A, 0x8A, 0x4A, 0xCA, 0x2A, 0xAA, 0x6A, 0xEA, 0x1A, 0x9A, 0x5A, +0xDA, 0x3A, 0xBA, 0x7A, 0xFA, 0x06, 0x86, 0x46, 0xC6, 0x26, 0xA6, 0x66, 0xE6, +0x16, 0x96, 0x56, 0xD6, 0x36, 0xB6, 0x76, 0xF6, 0x0E, 0x8E, 0x4E, 0xCE, 0x2E, +0xAE, 0x6E, 0xEE, 0x1E, 0x9E, 0x5E, 0xDE, 0x3E, 0xBE, 0x7E, 0xFE, 0x01, 0x81, +0x41, 0xC1, 0x21, 0xA1, 0x61, 0xE1, 0x11, 0x91, 0x51, 0xD1, 0x31, 0xB1, 0x71, +0xF1, 0x09, 0x89, 0x49, 0xC9, 0x29, 0xA9, 0x69, 0xE9, 0x19, 0x99, 0x59, 0xD9, +0x39, 0xB9, 0x79, 0xF9, 0x05, 0x85, 0x45, 0xC5, 0x25, 0xA5, 0x65, 0xE5, 0x15, +0x95, 0x55, 0xD5, 0x35, 0xB5, 0x75, 0xF5, 0x0D, 0x8D, 0x4D, 0xCD, 0x2D, 0xAD, +0x6D, 0xED, 0x1D, 0x9D, 0x5D, 0xDD, 0x3D, 0xBD, 0x7D, 0xFD, 0x03, 0x83, 0x43, +0xC3, 0x23, 0xA3, 0x63, 0xE3, 0x13, 0x93, 0x53, 0xD3, 0x33, 0xB3, 0x73, 0xF3, +0x0B, 0x8B, 0x4B, 0xCB, 0x2B, 0xAB, 0x6B, 0xEB, 0x1B, 0x9B, 0x5B, 0xDB, 0x3B, +0xBB, 0x7B, 0xFB, 0x07, 0x87, 0x47, 0xC7, 0x27, 0xA7, 0x67, 0xE7, 0x17, 0x97, +0x57, 0xD7, 0x37, 0xB7, 0x77, 0xF7, 0x0F, 0x8F, 0x4F, 0xCF, 0x2F, 0xAF, 0x6F, +0xEF, 0x1F, 0x9F, 0x5F, 0xDF, 0x3F, 0xBF, 0x7F, 0xFF +}; -uint32 my_round_up_to_next_power(uint32 v) -{ - v--; /* 01100000000000000000000000001010 */ - v|= v >> 1; /* 01110000000000000000000000001111 */ - v|= v >> 2; /* 01111100000000000000000000001111 */ - v|= v >> 4; /* 01111111110000000000000000001111 */ - v|= v >> 8; /* 01111111111111111100000000001111 */ - v|= v >> 16; /* 01111111111111111111111111111111 */ - return v+1; /* 10000000000000000000000000000000 */ -} diff --git a/mysys/my_bitmap.c b/mysys/my_bitmap.c index 2c85ce0bf04..8cf8ac0e97c 100644 --- a/mysys/my_bitmap.c +++ b/mysys/my_bitmap.c @@ -39,6 +39,7 @@ #include "mysys_priv.h" #include #include +#include void create_last_word_mask(MY_BITMAP *map) { diff --git a/mysys/my_init.c b/mysys/my_init.c index dca68637161..a1dca635054 100644 --- a/mysys/my_init.c +++ b/mysys/my_init.c @@ -44,6 +44,7 @@ static void netware_init(); my_bool my_init_done= 0; uint mysys_usage_id= 0; /* Incremented for each my_init() */ +ulong my_thread_stack_size= 65536; static ulong atoi_octal(const char *str) { diff --git a/sql/ha_myisam.cc b/sql/ha_myisam.cc index 2d097c34f97..983ebdc2ab8 100644 --- a/sql/ha_myisam.cc +++ b/sql/ha_myisam.cc @@ -21,6 +21,7 @@ #include "mysql_priv.h" #include +#include #include #include "ha_myisam.h" #include diff --git a/sql/item_func.cc b/sql/item_func.cc index 8139ba81777..190d4091beb 100644 --- a/sql/item_func.cc +++ b/sql/item_func.cc @@ -27,6 +27,7 @@ #include #include #include +#include #include "sp_head.h" #include "sp_rcontext.h" diff --git a/sql/mysql_priv.h b/sql/mysql_priv.h index c0b453b7d69..527c8fb513f 100644 --- a/sql/mysql_priv.h +++ b/sql/mysql_priv.h @@ -1497,7 +1497,7 @@ extern ulong max_connections,max_connect_errors, connect_timeout; extern ulong slave_net_timeout, slave_trans_retries; extern uint max_user_connections; extern ulong what_to_log,flush_time; -extern ulong query_buff_size, thread_stack; +extern ulong query_buff_size; extern ulong max_prepared_stmt_count, prepared_stmt_count; extern ulong binlog_cache_size, max_binlog_cache_size, open_files_limit; extern ulong max_binlog_size, max_relay_log_size; diff --git a/sql/mysqld.cc b/sql/mysqld.cc index 6e57993a61a..279ddc2c5d5 100644 --- a/sql/mysqld.cc +++ b/sql/mysqld.cc @@ -17,6 +17,7 @@ #include "mysql_priv.h" #include #include +#include #include "slave.h" #include "sql_repl.h" #include "rpl_filter.h" @@ -480,7 +481,7 @@ uint volatile thread_count, thread_running; ulonglong thd_startup_options; ulong back_log, connect_timeout, concurrency, server_id; ulong table_cache_size, table_def_size; -ulong thread_stack, what_to_log; +ulong what_to_log; ulong query_buff_size, slow_launch_time, slave_open_temp_tables; ulong open_files_limit, max_binlog_size, max_relay_log_size; ulong slave_net_timeout, slave_trans_retries; @@ -2142,7 +2143,7 @@ the thread stack. Please read http://www.mysql.com/doc/en/Linux.html\n\n", { fprintf(stderr,"thd=%p\n",thd); print_stacktrace(thd ? (gptr) thd->thread_stack : (gptr) 0, - thread_stack); + my_thread_stack_size); } if (thd) { @@ -2274,9 +2275,9 @@ static void start_signal_handler(void) Peculiar things with ia64 platforms - it seems we only have half the stack size in reality, so we have to double it here */ - pthread_attr_setstacksize(&thr_attr,thread_stack*2); + pthread_attr_setstacksize(&thr_attr,my_thread_stack_size*2); #else - pthread_attr_setstacksize(&thr_attr,thread_stack); + pthread_attr_setstacksize(&thr_attr,my_thread_stack_size); #endif #endif @@ -3493,9 +3494,9 @@ int main(int argc, char **argv) Peculiar things with ia64 platforms - it seems we only have half the stack size in reality, so we have to double it here */ - pthread_attr_setstacksize(&connection_attrib,thread_stack*2); + pthread_attr_setstacksize(&connection_attrib,my_thread_stack_size*2); #else - pthread_attr_setstacksize(&connection_attrib,thread_stack); + pthread_attr_setstacksize(&connection_attrib,my_thread_stack_size); #endif #ifdef HAVE_PTHREAD_ATTR_GETSTACKSIZE { @@ -3506,15 +3507,15 @@ int main(int argc, char **argv) stack_size/= 2; #endif /* We must check if stack_size = 0 as Solaris 2.9 can return 0 here */ - if (stack_size && stack_size < thread_stack) + if (stack_size && stack_size < my_thread_stack_size) { if (global_system_variables.log_warnings) sql_print_warning("Asked for %ld thread stack, but got %ld", - thread_stack, stack_size); + my_thread_stack_size, stack_size); #if defined(__ia64__) || defined(__ia64) - thread_stack= stack_size*2; + my_thread_stack_size= stack_size*2; #else - thread_stack= stack_size; + my_thread_stack_size= stack_size; #endif } } @@ -6276,8 +6277,8 @@ The minimum value for this variable is 4096.", (gptr*) &concurrency, (gptr*) &concurrency, 0, GET_ULONG, REQUIRED_ARG, DEFAULT_CONCURRENCY, 1, 512, 0, 1, 0}, {"thread_stack", OPT_THREAD_STACK, - "The stack size for each thread.", (gptr*) &thread_stack, - (gptr*) &thread_stack, 0, GET_ULONG, REQUIRED_ARG,DEFAULT_THREAD_STACK, + "The stack size for each thread.", (gptr*) &my_thread_stack_size, + (gptr*) &my_thread_stack_size, 0, GET_ULONG, REQUIRED_ARG,DEFAULT_THREAD_STACK, 1024L*128L, ~0L, 0, 1024, 0}, { "time_format", OPT_TIME_FORMAT, "The TIME format (for future).", diff --git a/sql/set_var.cc b/sql/set_var.cc index b0ecc7eccef..6c859d6bda9 100644 --- a/sql/set_var.cc +++ b/sql/set_var.cc @@ -1037,10 +1037,10 @@ SHOW_VAR init_vars[]= { #ifdef HAVE_THR_SETCONCURRENCY {"thread_concurrency", (char*) &concurrency, SHOW_LONG}, #endif - {"thread_stack", (char*) &thread_stack, SHOW_LONG}, + {"thread_stack", (char*) &my_thread_stack_size, SHOW_LONG}, {sys_time_format.name, (char*) &sys_time_format, SHOW_SYS}, {"time_zone", (char*) &sys_time_zone, SHOW_SYS}, - {sys_timed_mutexes.name, (char*) &sys_timed_mutexes, SHOW_SYS}, + {sys_timed_mutexes.name, (char*) &sys_timed_mutexes, SHOW_SYS}, {sys_tmp_table_size.name, (char*) &sys_tmp_table_size, SHOW_SYS}, {sys_tmpdir.name, (char*) &sys_tmpdir, SHOW_SYS}, {sys_trans_alloc_block_size.name, (char*) &sys_trans_alloc_block_size, diff --git a/sql/sql_parse.cc b/sql/sql_parse.cc index 232df095816..9015470f11c 100644 --- a/sql/sql_parse.cc +++ b/sql/sql_parse.cc @@ -5740,10 +5740,10 @@ bool check_stack_overrun(THD *thd, long margin, long stack_used; DBUG_ASSERT(thd == current_thd); if ((stack_used=used_stack(thd->thread_stack,(char*) &stack_used)) >= - (long) (thread_stack - margin)) + (long) (my_thread_stack_size - margin)) { sprintf(errbuff[0],ER(ER_STACK_OVERRUN_NEED_MORE), - stack_used,thread_stack,margin); + stack_used,my_thread_stack_size,margin); my_message(ER_STACK_OVERRUN_NEED_MORE,errbuff[0],MYF(0)); thd->fatal_error(); return 1; diff --git a/sql/sql_select.cc b/sql/sql_select.cc index d34ff070eb1..f32a0d1fba3 100644 --- a/sql/sql_select.cc +++ b/sql/sql_select.cc @@ -26,6 +26,7 @@ #include "sql_cursor.h" #include +#include #include #include diff --git a/sql/sql_test.cc b/sql/sql_test.cc index b28aa4a1ce5..77bbee0bf69 100644 --- a/sql/sql_test.cc +++ b/sql/sql_test.cc @@ -455,7 +455,7 @@ void mysql_print_status() VOID(my_getwd(current_dir, sizeof(current_dir),MYF(0))); printf("Current dir: %s\n", current_dir); printf("Running threads: %d Stack size: %ld\n", thread_count, - (long) thread_stack); + (long) my_thread_stack_size); thr_print_locks(); // Write some debug info #ifndef DBUG_OFF print_cached_tables(); @@ -530,7 +530,7 @@ Estimated memory (with thread stack): %ld\n", (int) info.uordblks, (int) info.fordblks, (int) info.keepcost, - (long) (thread_count * thread_stack + info.hblkhd + info.arena)); + (long) (thread_count * my_thread_stack_size + info.hblkhd + info.arena)); #endif puts(""); } diff --git a/storage/myisam/mi_create.c b/storage/myisam/mi_create.c index 22cbde278be..1f45b2ce070 100644 --- a/storage/myisam/mi_create.c +++ b/storage/myisam/mi_create.c @@ -18,6 +18,7 @@ #include "ftdefs.h" #include "sp_defs.h" +#include #if defined(MSDOS) || defined(__WIN__) #ifdef __WIN__ @@ -430,7 +431,7 @@ int mi_create(const char *name,uint keys,MI_KEYDEF *keydefs, share.state.rec_per_key_part[key_segs-1]=1L; length+=key_length; /* Get block length for key, if defined by user */ - block_length= (keydef->block_length ? + block_length= (keydef->block_length ? my_round_up_to_next_power(keydef->block_length) : myisam_block_size); block_length= max(block_length, MI_MIN_KEY_BLOCK_LENGTH); diff --git a/storage/myisam/mi_test2.c b/storage/myisam/mi_test2.c index 357128b7a40..498b433f846 100644 --- a/storage/myisam/mi_test2.c +++ b/storage/myisam/mi_test2.c @@ -27,6 +27,7 @@ #endif #include "myisamdef.h" #include +#include #define STANDARD_LENGTH 37 #define MYISAM_KEYS 6 diff --git a/storage/myisam/myisamchk.c b/storage/myisam/myisamchk.c index 7fcfb8fc65a..3a688613a4d 100644 --- a/storage/myisam/myisamchk.c +++ b/storage/myisam/myisamchk.c @@ -21,6 +21,7 @@ #include #include #include +#include #ifdef HAVE_SYS_VADVICE_H #include #endif diff --git a/unittest/mysys/my_atomic-t.c b/unittest/mysys/my_atomic-t.c index 71408ce957f..1e7206441eb 100644 --- a/unittest/mysys/my_atomic-t.c +++ b/unittest/mysys/my_atomic-t.c @@ -14,13 +14,20 @@ along with this program; if not, write to the Free Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ -#include +//#define MY_ATOMIC_MODE_RWLOCKS +//#define MY_ATOMIC_MODE_DUMMY + #include + +#include #include #include +#include -int32 a32,b32,c32; +volatile uint32 a32,b32,c32; my_atomic_rwlock_t rwl; +LF_ALLOCATOR lf_allocator; +LF_HASH lf_hash; pthread_attr_t thr_attr; pthread_mutex_t mutex; @@ -30,9 +37,9 @@ int N; /* add and sub a random number in a loop. Must get 0 at the end */ pthread_handler_t test_atomic_add_handler(void *arg) { - int m=*(int *)arg; + int m=(*(int *)arg)/2; int32 x; - for (x=((int)(&m)); m ; m--) + for (x=((int)(intptr)(&m)); m ; m--) { x=x*m+0x87654321; my_atomic_rwlock_wrlock(&rwl); @@ -61,15 +68,9 @@ pthread_handler_t test_atomic_add_handler(void *arg) pthread_handler_t test_atomic_swap_handler(void *arg) { int m=*(int *)arg; - int32 x; - - my_atomic_rwlock_wrlock(&rwl); - x=my_atomic_add32(&b32, 1); - my_atomic_rwlock_wrunlock(&rwl); + uint32 x=my_atomic_add32(&b32, 1); - my_atomic_rwlock_wrlock(&rwl); my_atomic_add32(&a32, x); - my_atomic_rwlock_wrunlock(&rwl); for (; m ; m--) { @@ -98,29 +99,29 @@ pthread_handler_t test_atomic_swap_handler(void *arg) /* same as test_atomic_add_handler, but my_atomic_add32 is emulated with - (slower) my_atomic_cas32 + my_atomic_cas32 - notice that the slowdown is proportional to the + number of CPUs */ pthread_handler_t test_atomic_cas_handler(void *arg) { - int m=*(int *)arg, ok; - int32 x,y; - for (x=((int)(&m)); m ; m--) + int m=(*(int *)arg)/2, ok=0; + int32 x, y; + for (x=((int)(intptr)(&m)); m ; m--) { my_atomic_rwlock_wrlock(&rwl); y=my_atomic_load32(&a32); my_atomic_rwlock_wrunlock(&rwl); - x=x*m+0x87654321; do { my_atomic_rwlock_wrlock(&rwl); ok=my_atomic_cas32(&a32, &y, y+x); my_atomic_rwlock_wrunlock(&rwl); - } while (!ok); + } while (!ok) ; do { my_atomic_rwlock_wrlock(&rwl); ok=my_atomic_cas32(&a32, &y, y-x); my_atomic_rwlock_wrunlock(&rwl); - } while (!ok); + } while (!ok) ; } pthread_mutex_lock(&mutex); N--; @@ -129,6 +130,128 @@ pthread_handler_t test_atomic_cas_handler(void *arg) return 0; } +/* + pin allocator - alloc and release an element in a loop +*/ +pthread_handler_t test_lf_pinbox(void *arg) +{ + int m=*(int *)arg; + int32 x=0; + LF_PINS *pins; + + pins=lf_pinbox_get_pins(&lf_allocator.pinbox); + + for (x=((int)(intptr)(&m)); m ; m--) + { + lf_pinbox_put_pins(pins); + pins=lf_pinbox_get_pins(&lf_allocator.pinbox); + } + lf_pinbox_put_pins(pins); + pthread_mutex_lock(&mutex); + N--; + if (!N) + pthread_cond_signal(&cond); + pthread_mutex_unlock(&mutex); + return 0; +} + +typedef union { + int32 data; + void *not_used; /* to guarantee sizeof(TLA) >= sizeof(void *) */ +} TLA; + +pthread_handler_t test_lf_alloc(void *arg) +{ + int m=(*(int *)arg)/2; + int32 x,y=0; + LF_PINS *pins; + + pins=lf_alloc_get_pins(&lf_allocator); + + for (x=((int)(intptr)(&m)); m ; m--) + { + TLA *node1, *node2; + x=x*m+0x87654321; + node1=(TLA *)lf_alloc_new(pins); + node1->data=x; + y+=node1->data; + node1->data=0; + node2=(TLA *)lf_alloc_new(pins); + node2->data=x; + y-=node2->data; + node2->data=0; + lf_alloc_free(pins, node1); + lf_alloc_free(pins, node2); + } + lf_alloc_put_pins(pins); + my_atomic_rwlock_wrlock(&rwl); + my_atomic_add32(&a32, y); + my_atomic_rwlock_wrunlock(&rwl); + pthread_mutex_lock(&mutex); + N--; + if (!N) + { + diag("%d mallocs, %d pins in stack", + lf_allocator.mallocs, lf_allocator.pinbox.pins_in_stack); +#ifdef MY_LF_EXTRA_DEBUG + a32|=lf_allocator.mallocs - lf_alloc_in_pool(&lf_allocator); +#endif + pthread_cond_signal(&cond); + } + pthread_mutex_unlock(&mutex); + return 0; +} + +#define N_TLH 1000 +pthread_handler_t test_lf_hash(void *arg) +{ + int m=(*(int *)arg)/(2*N_TLH); + int32 x,y,z,sum=0, ins=0; + LF_PINS *pins; + + pins=lf_hash_get_pins(&lf_hash); + + for (x=((int)(intptr)(&m)); m ; m--) + { + int i; + y=x; + for (i=0; i < N_TLH; i++) + { + x=x*(m+i)+0x87654321; + z=(x<0) ? -x : x; + if (lf_hash_insert(&lf_hash, pins, &z)) + { + sum+=z; + ins++; + } + } + for (i=0; i < N_TLH; i++) + { + y=y*(m+i)+0x87654321; + z=(y<0) ? -y : y; + if (lf_hash_delete(&lf_hash, pins, (uchar *)&z, sizeof(z))) + sum-=z; + } + } + lf_hash_put_pins(pins); + my_atomic_rwlock_wrlock(&rwl); + my_atomic_add32(&a32, sum); + my_atomic_add32(&b32, ins); + my_atomic_rwlock_wrunlock(&rwl); + pthread_mutex_lock(&mutex); + N--; + if (!N) + { + diag("%d mallocs, %d pins in stack, %d hash size, %d inserts", + lf_hash.alloc.mallocs, lf_hash.alloc.pinbox.pins_in_stack, + lf_hash.size, b32); + a32|=lf_hash.count; + pthread_cond_signal(&cond); + } + pthread_mutex_unlock(&mutex); + return 0; +} + void test_atomic(const char *test, pthread_handler handler, int n, int m) { pthread_t t; @@ -141,23 +264,24 @@ void test_atomic(const char *test, pthread_handler handler, int n, int m) diag("Testing %s with %d threads, %d iterations... ", test, n, m); for (N=n ; n ; n--) pthread_create(&t, &thr_attr, handler, &m); - pthread_mutex_lock(&mutex); while (N) pthread_cond_wait(&cond, &mutex); pthread_mutex_unlock(&mutex); now=my_getsystime()-now; - ok(a32 == 0, "tested %s in %g secs", test, ((double)now)/1e7); + ok(a32 == 0, "tested %s in %g secs (%d)", test, ((double)now)/1e7, a32); } int main() { int err; - diag("N CPUs: %d", my_getncpus()); + my_init(); + + diag("N CPUs: %d, atomic ops: %s", my_getncpus(), MY_ATOMIC_MODE); err= my_atomic_initialize(); - plan(4); + plan(7); ok(err == 0, "my_atomic_initialize() returned %d", err); pthread_attr_init(&thr_attr); @@ -165,15 +289,31 @@ int main() pthread_mutex_init(&mutex, 0); pthread_cond_init(&cond, 0); my_atomic_rwlock_init(&rwl); + lf_alloc_init(&lf_allocator, sizeof(TLA)); + lf_hash_init(&lf_hash, sizeof(int), LF_HASH_UNIQUE, 0, sizeof(int), 0, + &my_charset_bin); + +#ifdef MY_ATOMIC_MODE_RWLOCKS +#define CYCLES 10000 +#else +#define CYCLES 1000000 +#endif +#define THREADS 100 - test_atomic("my_atomic_add32", test_atomic_add_handler, 100,1000000); - test_atomic("my_atomic_swap32", test_atomic_swap_handler, 100,1000000); - test_atomic("my_atomic_cas32", test_atomic_cas_handler, 100,1000000); + test_atomic("my_atomic_add32", test_atomic_add_handler, THREADS,CYCLES); + test_atomic("my_atomic_swap32", test_atomic_swap_handler, THREADS,CYCLES); + test_atomic("my_atomic_cas32", test_atomic_cas_handler, THREADS,CYCLES); + test_atomic("lf_pinbox", test_lf_pinbox, THREADS,CYCLES); + test_atomic("lf_alloc", test_lf_alloc, THREADS,CYCLES); + test_atomic("lf_hash", test_lf_hash, THREADS,CYCLES); + lf_hash_end(&lf_hash); + lf_alloc_end(&lf_allocator); pthread_mutex_destroy(&mutex); pthread_cond_destroy(&cond); pthread_attr_destroy(&thr_attr); my_atomic_rwlock_destroy(&rwl); + my_end(0); return exit_status(); } -- cgit v1.2.1 From d1a8a2c7345847fc750c2d36908913dedb052c9c Mon Sep 17 00:00:00 2001 From: unknown Date: Fri, 11 Aug 2006 01:16:51 +0200 Subject: fixes for inline bit functions --- sql/ha_maria.cc | 1 + storage/maria/ma_create.c | 1 + storage/maria/ma_test2.c | 1 + storage/maria/maria_chk.c | 1 + 4 files changed, 4 insertions(+) diff --git a/sql/ha_maria.cc b/sql/ha_maria.cc index 29718f1493e..b883d56e3f7 100644 --- a/sql/ha_maria.cc +++ b/sql/ha_maria.cc @@ -22,6 +22,7 @@ #include "mysql_priv.h" #include #include +#include #include "ha_maria.h" #ifndef MASTER #include "../srclib/maria/maria_def.h" diff --git a/storage/maria/ma_create.c b/storage/maria/ma_create.c index b9fb4eb0d5b..5926bba9406 100644 --- a/storage/maria/ma_create.c +++ b/storage/maria/ma_create.c @@ -18,6 +18,7 @@ #include "ma_ftdefs.h" #include "ma_sp_defs.h" +#include #if defined(MSDOS) || defined(__WIN__) #ifdef __WIN__ diff --git a/storage/maria/ma_test2.c b/storage/maria/ma_test2.c index 840ecb2eeb7..40f7d2aaffb 100644 --- a/storage/maria/ma_test2.c +++ b/storage/maria/ma_test2.c @@ -27,6 +27,7 @@ #endif #include "maria_def.h" #include +#include #define STANDARD_LENGTH 37 #define MARIA_KEYS 6 diff --git a/storage/maria/maria_chk.c b/storage/maria/maria_chk.c index e423a3f5c36..89858ee2d07 100644 --- a/storage/maria/maria_chk.c +++ b/storage/maria/maria_chk.c @@ -18,6 +18,7 @@ #include "ma_fulltext.h" #include +#include #include #include #include -- cgit v1.2.1 From 74d050d000ff9db79e36931988386fe7988f8dd2 Mon Sep 17 00:00:00 2001 From: unknown Date: Thu, 17 Aug 2006 15:20:58 +0200 Subject: maria transaction manager with unit tests include/lf.h: few lf API changes mysys/lf_alloc-pin.c: few lf API changes mysys/lf_dynarray.c: few lf API changes mysys/lf_hash.c: few lf API changes storage/maria/Makefile.am: transaction manager unittest/Makefile.am: maria transaction manager unittest/mysys/my_atomic-t.c: ensure that values are positive storage/maria/trxman.h: New BitKeeper file ``storage/maria/trxman.h'' unittest/maria/Makefile.am: New BitKeeper file ``unittest/maria/Makefile.am'' unittest/maria/trxman-t.c: New BitKeeper file ``unittest/maria/trxman-t.c'' storage/maria/trxman.c: comment clarified --- configure.in | 2 +- include/lf.h | 16 +-- mysys/lf_alloc-pin.c | 8 +- mysys/lf_dynarray.c | 2 +- mysys/lf_hash.c | 78 +++++-------- storage/maria/Makefile.am | 6 +- storage/maria/trxman.c | 261 +++++++++++++++++++++++++++++++++++++++++++ storage/maria/trxman.h | 28 +++++ unittest/Makefile.am | 4 +- unittest/maria/Makefile.am | 12 ++ unittest/maria/trxman-t.c | 138 +++++++++++++++++++++++ unittest/mysys/my_atomic-t.c | 17 ++- 12 files changed, 495 insertions(+), 77 deletions(-) create mode 100644 storage/maria/trxman.c create mode 100644 storage/maria/trxman.h create mode 100644 unittest/maria/Makefile.am create mode 100644 unittest/maria/trxman-t.c diff --git a/configure.in b/configure.in index e984f1524da..d518760ca74 100644 --- a/configure.in +++ b/configure.in @@ -2537,7 +2537,7 @@ AC_SUBST(MAKE_BINARY_DISTRIBUTION_OPTIONS) # Output results AC_CONFIG_FILES(Makefile extra/Makefile mysys/Makefile dnl unittest/Makefile unittest/mytap/Makefile unittest/mytap/t/Makefile dnl - unittest/mysys/Makefile unittest/examples/Makefile dnl + unittest/mysys/Makefile unittest/examples/Makefile unittest/maria/Makefile dnl strings/Makefile regex/Makefile storage/Makefile dnl man/Makefile BUILD/Makefile vio/Makefile dnl libmysql/Makefile client/Makefile dnl diff --git a/include/lf.h b/include/lf.h index 299b4e07aa5..6a5047f6052 100644 --- a/include/lf.h +++ b/include/lf.h @@ -66,7 +66,7 @@ typedef struct { typedef int (*lf_dynarray_func)(void *, void *); void lf_dynarray_init(LF_DYNARRAY *array, uint element_size); -void lf_dynarray_end(LF_DYNARRAY *array); +void lf_dynarray_destroy(LF_DYNARRAY *array); nolock_wrap(lf_dynarray_nr, int, (LF_DYNARRAY *array, void *el), @@ -139,15 +139,15 @@ typedef struct { #define _lf_unpin(PINS, PIN) _lf_pin(PINS, PIN, NULL) #define lf_pin(PINS, PIN, ADDR) \ do { \ - lf_lock_pins(PINS); \ + lf_lock_by_pins(PINS); \ _lf_pin(PINS, PIN, ADDR); \ - lf_unlock_pins(PINS); \ + lf_unlock_by_pins(PINS); \ } while (0) #define lf_unpin(PINS, PIN) lf_pin(PINS, PIN, NULL) void lf_pinbox_init(LF_PINBOX *pinbox, lf_pinbox_free_func *free_func, void * free_func_arg); -void lf_pinbox_end(LF_PINBOX *pinbox); +void lf_pinbox_destroy(LF_PINBOX *pinbox); lock_wrap(lf_pinbox_get_pins, LF_PINS *, (LF_PINBOX *pinbox), @@ -180,7 +180,7 @@ typedef struct st_lf_allocator { } LF_ALLOCATOR; void lf_alloc_init(LF_ALLOCATOR *allocator, uint size); -void lf_alloc_end(LF_ALLOCATOR *allocator); +void lf_alloc_destroy(LF_ALLOCATOR *allocator); uint lf_alloc_in_pool(LF_ALLOCATOR *allocator); #define _lf_alloc_free(PINS, PTR) _lf_pinbox_free((PINS), (PTR)) #define lf_alloc_free(PINS, PTR) lf_pinbox_free((PINS), (PTR)) @@ -216,10 +216,10 @@ typedef struct { void lf_hash_init(LF_HASH *hash, uint element_size, uint flags, uint key_offset, uint key_length, hash_get_key get_key, CHARSET_INFO *charset); -void lf_hash_end(LF_HASH *hash); +void lf_hash_destroy(LF_HASH *hash); int lf_hash_insert(LF_HASH *hash, LF_PINS *pins, const void *data); -int lf_hash_search(LF_HASH *hash, LF_PINS *pins, const uchar *key, uint keylen); -int lf_hash_delete(LF_HASH *hash, LF_PINS *pins, const uchar *key, uint keylen); +void *lf_hash_search(LF_HASH *hash, LF_PINS *pins, const void *key, uint keylen); +int lf_hash_delete(LF_HASH *hash, LF_PINS *pins, const void *key, uint keylen); #define _lf_hash_get_pins(HASH) _lf_alloc_get_pins(&(HASH)->alloc) #define lf_hash_get_pins(HASH) lf_alloc_get_pins(&(HASH)->alloc) #define _lf_hash_put_pins(PINS) _lf_pinbox_put_pins(PINS) diff --git a/mysys/lf_alloc-pin.c b/mysys/lf_alloc-pin.c index a9ea1802c03..cf1612b73d1 100644 --- a/mysys/lf_alloc-pin.c +++ b/mysys/lf_alloc-pin.c @@ -54,9 +54,9 @@ void lf_pinbox_init(LF_PINBOX *pinbox, lf_pinbox_free_func *free_func, pinbox->free_func_arg=free_func_arg; } -void lf_pinbox_end(LF_PINBOX *pinbox) +void lf_pinbox_destroy(LF_PINBOX *pinbox) { - lf_dynarray_end(&pinbox->pinstack); + lf_dynarray_destroy(&pinbox->pinstack); } LF_PINS *_lf_pinbox_get_pins(LF_PINBOX *pinbox) @@ -292,7 +292,7 @@ void lf_alloc_init(LF_ALLOCATOR *allocator, uint size) DBUG_ASSERT(size >= (int)sizeof(void *)); } -void lf_alloc_end(LF_ALLOCATOR *allocator) +void lf_alloc_destroy(LF_ALLOCATOR *allocator) { void *el=allocator->top; while (el) @@ -301,7 +301,7 @@ void lf_alloc_end(LF_ALLOCATOR *allocator) my_free(el, MYF(0)); el=tmp; } - lf_pinbox_end(&allocator->pinbox); + lf_pinbox_destroy(&allocator->pinbox); allocator->top=0; } diff --git a/mysys/lf_dynarray.c b/mysys/lf_dynarray.c index dcf99163ca1..0fa04ab095c 100644 --- a/mysys/lf_dynarray.c +++ b/mysys/lf_dynarray.c @@ -57,7 +57,7 @@ static void recursive_free(void **alloc, int level) my_free(alloc[-1], MYF(0)); } -void lf_dynarray_end(LF_DYNARRAY *array) +void lf_dynarray_destroy(LF_DYNARRAY *array) { int i; for (i=0; i < LF_DYNARRAY_LEVELS; i++) diff --git a/mysys/lf_hash.c b/mysys/lf_hash.c index 6d3d30ebc3f..57936ea7b1f 100644 --- a/mysys/lf_hash.c +++ b/mysys/lf_hash.c @@ -228,14 +228,14 @@ void lf_hash_init(LF_HASH *hash, uint element_size, uint flags, hash->count=0; hash->element_size=element_size; hash->flags=flags; - hash->charset=charset; + hash->charset=charset ? charset : &my_charset_bin; hash->key_offset=key_offset; hash->key_length=key_length; hash->get_key=get_key; DBUG_ASSERT(get_key ? !key_offset && !key_length : key_length); } -void lf_hash_end(LF_HASH *hash) +void lf_hash_destroy(LF_HASH *hash) { LF_SLIST *el=*(LF_SLIST **)_lf_dynarray_lvalue(&hash->array, 0); while (el) @@ -244,109 +244,89 @@ void lf_hash_end(LF_HASH *hash) lf_alloc_real_free(&hash->alloc, el); el=(LF_SLIST *)next; } - lf_alloc_end(&hash->alloc); - lf_dynarray_end(&hash->array); + lf_alloc_destroy(&hash->alloc); + lf_dynarray_destroy(&hash->array); } /* + RETURN + 0 - inserted + 1 - didn't (unique key conflict) NOTE - see linsert() for pin usage + see linsert() for pin usage notes */ int lf_hash_insert(LF_HASH *hash, LF_PINS *pins, const void *data) { - uint csize, bucket, hashnr, keylen; + uint csize, bucket, hashnr; LF_SLIST *node, * volatile *el; - const uchar *key; - key= hash_key(hash, data, &keylen); - hashnr= calc_hash(hash, key, keylen); - bucket= hashnr % hash->size; lf_lock_by_pins(pins); node=(LF_SLIST *)_lf_alloc_new(pins); memcpy(node+1, data, hash->element_size); + node->key= hash_key(hash, node+1, &node->keylen); + hashnr= calc_hash(hash, node->key, node->keylen); + bucket= hashnr % hash->size; el=_lf_dynarray_lvalue(&hash->array, bucket); if (*el == NULL) initialize_bucket(hash, el, bucket, pins); node->hashnr=my_reverse_bits(hashnr) | 1; - node->key=((char *)(node+1))+(key-(uchar *)data); - node->keylen=keylen; if (linsert(el, node, pins, hash->flags)) { _lf_alloc_free(pins, node); lf_unlock_by_pins(pins); - return 0; + return 1; } csize= hash->size; if ((my_atomic_add32(&hash->count, 1)+1.0) / csize > MAX_LOAD) my_atomic_cas32(&hash->size, &csize, csize*2); -#if 0 - node=*(LF_SLIST **)_lf_dynarray_lvalue(&hash->array, 0); - hashnr=0; - while (node) - { - assert (node->hashnr >= hashnr); - hashnr=node->hashnr; - node=(LF_SLIST *)node->link; - } -#endif lf_unlock_by_pins(pins); - return 1; + return 0; } /* + RETURN + 0 - deleted + 1 - didn't (not found) NOTE - see ldelete() for pin usage + see ldelete() for pin usage notes */ -int lf_hash_delete(LF_HASH *hash, LF_PINS *pins, const uchar *key, uint keylen) +int lf_hash_delete(LF_HASH *hash, LF_PINS *pins, const void *key, uint keylen) { LF_SLIST * volatile *el; - uint bucket, hashnr=calc_hash(hash, key, keylen); + uint bucket, hashnr=calc_hash(hash, (uchar *)key, keylen); bucket= hashnr % hash->size; lf_lock_by_pins(pins); el=_lf_dynarray_lvalue(&hash->array, bucket); if (*el == NULL) initialize_bucket(hash, el, bucket, pins); - if (ldelete(el, my_reverse_bits(hashnr) | 1, key, keylen, pins)) + if (ldelete(el, my_reverse_bits(hashnr) | 1, (uchar *)key, keylen, pins)) { lf_unlock_by_pins(pins); - return 0; + return 1; } my_atomic_add32(&hash->count, -1); -#if 0 - { - LF_SLIST *node=*(LF_SLIST **)_lf_dynarray_lvalue(&hash->array, 0); - hashnr=0; - while (node) - { - assert (node->hashnr >= hashnr); - hashnr=node->hashnr; - node=(LF_SLIST *)node->link; - } - } -#endif lf_unlock_by_pins(pins); - return 1; + return 0; } /* NOTE - see lsearch() for pin usage + see lsearch() for pin usage notes */ -int lf_hash_search(LF_HASH *hash, LF_PINS *pins, const uchar *key, uint keylen) +void *lf_hash_search(LF_HASH *hash, LF_PINS *pins, const void *key, uint keylen) { - int res; - LF_SLIST * volatile *el; - uint bucket, hashnr=calc_hash(hash, key, keylen); + LF_SLIST * volatile *el, *found; + uint bucket, hashnr=calc_hash(hash, (uchar *)key, keylen); bucket= hashnr % hash->size; lf_lock_by_pins(pins); el=_lf_dynarray_lvalue(&hash->array, bucket); if (*el == NULL) initialize_bucket(hash, el, bucket, pins); - res=NULL != lsearch(el, my_reverse_bits(hashnr) | 1, key, keylen, pins); + found= lsearch(el, my_reverse_bits(hashnr) | 1, (uchar *)key, keylen, pins); lf_unlock_by_pins(pins); - return res; + return found+1; } static void initialize_bucket(LF_HASH *hash, LF_SLIST * volatile *node, diff --git a/storage/maria/Makefile.am b/storage/maria/Makefile.am index d4315b4d446..cacfa5dbcdc 100644 --- a/storage/maria/Makefile.am +++ b/storage/maria/Makefile.am @@ -28,7 +28,9 @@ bin_PROGRAMS = maria_chk maria_pack maria_ftdump maria_chk_DEPENDENCIES= $(LIBRARIES) maria_pack_DEPENDENCIES=$(LIBRARIES) noinst_PROGRAMS = ma_test1 ma_test2 ma_test3 ma_rt_test ma_sp_test -noinst_HEADERS = maria_def.h ma_rt_index.h ma_rt_key.h ma_rt_mbr.h ma_sp_defs.h ma_fulltext.h ma_ftdefs.h ma_ft_test1.h ma_ft_eval.h +noinst_HEADERS = maria_def.h ma_rt_index.h ma_rt_key.h ma_rt_mbr.h \ + ma_sp_defs.h ma_fulltext.h ma_ftdefs.h ma_ft_test1.h \ + ma_ft_eval.h trxman.h ma_test1_DEPENDENCIES= $(LIBRARIES) ma_test2_DEPENDENCIES= $(LIBRARIES) ma_test3_DEPENDENCIES= $(LIBRARIES) @@ -53,7 +55,7 @@ libmaria_a_SOURCES = ma_init.c ma_open.c ma_extra.c ma_info.c ma_rkey.c \ ma_ft_update.c ma_ft_boolean_search.c \ ma_ft_nlq_search.c ft_maria.c ma_sort.c \ ma_rt_index.c ma_rt_key.c ma_rt_mbr.c ma_rt_split.c \ - ma_sp_key.c + ma_sp_key.c trxman.c CLEANFILES = test?.MA? FT?.MA? isam.log ma_test_all ma_rt_test.MA? sp_test.MA? DEFS = diff --git a/storage/maria/trxman.c b/storage/maria/trxman.c new file mode 100644 index 00000000000..3c64d93183a --- /dev/null +++ b/storage/maria/trxman.c @@ -0,0 +1,261 @@ + +#include +#include +#include +#include "trxman.h" + +TRX active_list_min, active_list_max, + committed_list_min, committed_list_max, *pool; + +pthread_mutex_t LOCK_trx_list; +uint active_transactions; +TrID global_trid_generator; + +TRX **short_id_to_trx; +my_atomic_rwlock_t LOCK_short_id_to_trx; + +LF_HASH trid_to_trx; + +static byte *trx_get_hash_key(const byte *trx,uint* len, my_bool unused) +{ + *len= sizeof(TrID); + return (byte *) & ((*((TRX **)trx))->trid); +} + +int trxman_init() +{ + pthread_mutex_init(&LOCK_trx_list, MY_MUTEX_INIT_FAST); + active_list_max.trid= active_list_min.trid= 0; + active_list_max.min_read_from=~0; + active_list_max.next= active_list_min.prev= 0; + active_list_max.prev= &active_list_min; + active_list_min.next= &active_list_max; + active_transactions= 0; + + committed_list_max.commit_trid= ~0; + committed_list_max.next= committed_list_min.prev= 0; + committed_list_max.prev= &committed_list_min; + committed_list_min.next= &committed_list_max; + + pool=0; + global_trid_generator=0; /* set later by recovery code */ + lf_hash_init(&trid_to_trx, sizeof(TRX*), LF_HASH_UNIQUE, + 0, 0, trx_get_hash_key, 0); + my_atomic_rwlock_init(&LOCK_short_id_to_trx); + short_id_to_trx=(TRX **)my_malloc(SHORT_ID_MAX*sizeof(TRX*), + MYF(MY_WME|MY_ZEROFILL)); + if (!short_id_to_trx) + return 1; + short_id_to_trx--; /* min short_id is 1 */ + + return 0; +} + +int trxman_destroy() +{ + DBUG_ASSERT(trid_to_trx.count == 0); + DBUG_ASSERT(active_transactions == 0); + DBUG_ASSERT(active_list_max.prev == &active_list_min); + DBUG_ASSERT(active_list_min.next == &active_list_max); + DBUG_ASSERT(committed_list_max.prev == &committed_list_min); + DBUG_ASSERT(committed_list_min.next == &committed_list_max); + while (pool) + { + TRX *tmp=pool->next; + my_free(pool, MYF(0)); + pool=tmp; + } + lf_hash_destroy(&trid_to_trx); + pthread_mutex_destroy(&LOCK_trx_list); + my_atomic_rwlock_destroy(&LOCK_short_id_to_trx); + my_free((void *)(short_id_to_trx+1), MYF(0)); +} + +static TrID new_trid() +{ + DBUG_ASSERT(global_trid_generator < 0xffffffffffffLL); + safe_mutex_assert_owner(&LOCK_trx_list); + return ++global_trid_generator; +} + +static void set_short_id(TRX *trx) +{ + int i= (global_trid_generator + (intptr)trx) * 312089 % SHORT_ID_MAX; + my_atomic_rwlock_wrlock(&LOCK_short_id_to_trx); + for ( ; ; i= i % SHORT_ID_MAX + 1) /* the range is [1..SHORT_ID_MAX] */ + { + void *tmp=NULL; + if (short_id_to_trx[i] == NULL && + my_atomic_casptr((void **)&short_id_to_trx[i], &tmp, trx)) + break; + } + my_atomic_rwlock_wrunlock(&LOCK_short_id_to_trx); + trx->short_id= i; +} + +extern int global_malloc; +TRX *trxman_new_trx() +{ + TRX *trx; + + my_atomic_add32(&active_transactions, 1); + + /* + we need a mutex here to ensure that + transactions in the active list are ordered by the trid. + So, incrementing global_trid_generator and + adding to the list must be atomic. + + and as we have a mutex, we can as well do everything + under it - allocating a TRX, incrementing active_transactions, + setting trx->min_read_from. + + Note that all the above is fast. generating short_id may be slow, + as it involves scanning a big array - so it's still done + outside of the mutex. + */ + + pthread_mutex_lock(&LOCK_trx_list); + trx=pool; + while (trx && !my_atomic_casptr((void **)&pool, (void **)&trx, trx->next)) + /* no-op */; + + if (!trx) + { + trx=(TRX *)my_malloc(sizeof(TRX), MYF(MY_WME)); + global_malloc++; + } + if (!trx) + return 0; + + trx->min_read_from= active_list_min.next->trid; + + trx->trid= new_trid(); + trx->short_id= 0; + + trx->next= &active_list_max; + trx->prev= active_list_max.prev; + active_list_max.prev= trx->prev->next= trx; + pthread_mutex_unlock(&LOCK_trx_list); + + trx->pins=lf_hash_get_pins(&trid_to_trx); + + if (!trx->min_read_from) + trx->min_read_from= trx->trid; + + trx->commit_trid=0; + + set_short_id(trx); /* this must be the last! */ + + + return trx; +} + +/* + remove a trx from the active list, + move to committed list, + set commit_trid + + TODO + integrate with lock manager, log manager. That means: + a common "commit" mutex - forcing the log and setting commit_trid + must be done atomically (QQ how the heck it could be done with + group commit ???) + + trid_to_trx, active_list_*, and committed_list_* can be + updated asyncronously. +*/ +void trxman_end_trx(TRX *trx, my_bool commit) +{ + int res; + TRX *free_me= 0; + LF_PINS *pins= trx->pins; + + pthread_mutex_lock(&LOCK_trx_list); + trx->next->prev= trx->prev; + trx->prev->next= trx->next; + + if (trx->prev == &active_list_min) + { + TRX *t; + for (t= committed_list_min.next; + t->commit_trid < active_list_min.next->min_read_from; + t= t->next) /* no-op */; + + if (t != committed_list_min.next) + { + free_me= committed_list_min.next; + committed_list_min.next= t; + t->prev->next=0; + t->prev= &committed_list_min; + } + } + + my_atomic_rwlock_wrlock(&LOCK_short_id_to_trx); + my_atomic_storeptr((void **)&short_id_to_trx[trx->short_id], 0); + my_atomic_rwlock_wrunlock(&LOCK_short_id_to_trx); + + if (commit && active_list_min.next != &active_list_max) + { + trx->commit_trid= global_trid_generator; + + trx->next= &committed_list_max; + trx->prev= committed_list_max.prev; + committed_list_max.prev= trx->prev->next= trx; + + res= lf_hash_insert(&trid_to_trx, pins, &trx); + DBUG_ASSERT(res == 0); + } + else + { + trx->next=free_me; + free_me=trx; + } + pthread_mutex_unlock(&LOCK_trx_list); + + my_atomic_add32(&active_transactions, -1); + + while (free_me) + { + int res; + TRX *t= free_me; + free_me= free_me->next; + + res= lf_hash_delete(&trid_to_trx, pins, &t->trid, sizeof(TrID)); + + trxman_free_trx(t); + } + + lf_hash_put_pins(pins); +} + +/* free a trx (add to the pool, that is */ +void trxman_free_trx(TRX *trx) +{ + TRX *tmp=pool; + + do + { + trx->next=tmp; + } while (!my_atomic_casptr((void **)&pool, (void **)&tmp, trx)); +} + +my_bool trx_can_read_from(TRX *trx, TrID trid) +{ + TRX *found; + my_bool can; + + if (trid < trx->min_read_from) + return TRUE; + if (trid > trx->trid) + return FALSE; + + found= lf_hash_search(&trid_to_trx, trx->pins, &trid, sizeof(trid)); + if (!found) + return FALSE; /* not in the hash = cannot read */ + + can= found->commit_trid < trx->trid; + lf_unpin(trx->pins, 2); + return can; +} + diff --git a/storage/maria/trxman.h b/storage/maria/trxman.h new file mode 100644 index 00000000000..ae794470a47 --- /dev/null +++ b/storage/maria/trxman.h @@ -0,0 +1,28 @@ + +typedef uint64 TrID; /* our TrID is 6 bytes */ + +typedef struct st_transaction +{ + TrID trid, min_read_from, commit_trid; + struct st_transaction *next, *prev; + /* Note! if short_id is 0, trx is NOT initialized */ + uint16 short_id; + LF_PINS *pins; +} TRX; + +#define SHORT_ID_MAX 65535 + +extern uint active_transactions; + +extern TRX **short_id_to_trx; +extern my_atomic_rwlock_t LOCK_short_id_to_trx; + +int trxman_init(); +int trxman_end(); +TRX *trxman_new_trx(); +void trxman_end_trx(TRX *trx, my_bool commit); +#define trxman_commit_trx(T) trxman_end_trx(T, TRUE) +#define trxman_abort_trx(T) trxman_end_trx(T, FALSE) +void trxman_free_trx(TRX *trx); +my_bool trx_can_read_from(TRX *trx, TrID trid); + diff --git a/unittest/Makefile.am b/unittest/Makefile.am index ca3291efde0..295ecd7186f 100644 --- a/unittest/Makefile.am +++ b/unittest/Makefile.am @@ -1,10 +1,10 @@ -SUBDIRS = mytap . mysys examples +SUBDIRS = mytap mysys maria examples noinst_SCRIPTS = unit EXTRA_DIST = unit.pl CLEANFILES = unit -unittests = mytap mysys +unittests = mytap mysys maria test: unit ./unit run $(unittests) diff --git a/unittest/maria/Makefile.am b/unittest/maria/Makefile.am new file mode 100644 index 00000000000..667d1e09a07 --- /dev/null +++ b/unittest/maria/Makefile.am @@ -0,0 +1,12 @@ + +AM_CPPFLAGS = @ZLIB_INCLUDES@ -I$(top_builddir)/include +AM_CPPFLAGS += -I$(top_srcdir)/include -I$(top_srcdir)/unittest/mytap + +LDADD = $(top_builddir)/unittest/mytap/libmytap.a \ + $(top_builddir)/storage/maria/libmaria.a \ + $(top_builddir)/mysys/libmysys.a \ + $(top_builddir)/dbug/libdbug.a \ + $(top_builddir)/strings/libmystrings.a + +noinst_PROGRAMS = trxman-t + diff --git a/unittest/maria/trxman-t.c b/unittest/maria/trxman-t.c new file mode 100644 index 00000000000..a29bf5abc8e --- /dev/null +++ b/unittest/maria/trxman-t.c @@ -0,0 +1,138 @@ +/* Copyright (C) 2006 MySQL AB + + 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; either version 2 of the License, or + (at your option) any later version. + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ + +#include + +#include +#include +#include +#include +#include "../../storage/maria/trxman.h" + +pthread_attr_t rt_attr; +pthread_mutex_t rt_mutex; +pthread_cond_t rt_cond; +int rt_num_threads; + +int litmus; + +/* template for a test: the goal is to have litmus==0 if the test passed + +#define ITER nnn +pthread_handler_t test_XXXXXXXX(void *arg) +{ + int m=(*(int *)arg)/ITER, x; + + for (x=((int)(intptr)(&m)); m ; m--) + { + // do something with litmus + } + // do something more with litmus + + pthread_mutex_lock(&rt_mutex); + rt_num_threads--; + if (!rt_num_threads) + { + diag("whatever diagnostics we want", blabla, foobar); + pthread_cond_signal(&rt_cond); + } + pthread_mutex_unlock(&rt_mutex); + return 0; +} +#undef ITER + +*/ + +/* + create and end (commit or rollback) transactions randomly +*/ +#define MAX_ITER 100 +pthread_handler_t test_trxman(void *arg) +{ + int m=(*(int *)arg); + uint x, y, i, j, n; + TRX *trx[MAX_ITER]; + + for (x=((int)(intptr)(&m)); m > 0; ) + { + y= x= (x*3628273133 + 1500450271) % 9576890767; /* three prime numbers */ + m-= n= x % MAX_ITER; + for (i=0; i < n; i++) + trx[i]=trxman_new_trx(); + for (i=0; i < n; i++) + { + y=(y*19 + 7) % 31; + trxman_end_trx(trx[i], y & 1); + } + } + + pthread_mutex_lock(&rt_mutex); + rt_num_threads--; + if (!rt_num_threads) + pthread_cond_signal(&rt_cond); + pthread_mutex_unlock(&rt_mutex); + return 0; +} +#undef MAX_ITER + +void run_test(const char *test, pthread_handler handler, int n, int m) +{ + pthread_t t; + ulonglong now=my_getsystime(); + + litmus= 0; + + diag("Testing %s with %d threads, %d iterations... ", test, n, m); + for (rt_num_threads=n ; n ; n--) + pthread_create(&t, &rt_attr, handler, &m); + pthread_mutex_lock(&rt_mutex); + while (rt_num_threads) + pthread_cond_wait(&rt_cond, &rt_mutex); + pthread_mutex_unlock(&rt_mutex); + now=my_getsystime()-now; + ok(litmus == 0, "tested %s in %g secs (%d)", test, ((double)now)/1e7, litmus); +} + +int global_malloc=0; +int main() +{ + plan(1); + + if (my_atomic_initialize()) + return exit_status(); + + my_init(); + + pthread_attr_init(&rt_attr); + pthread_attr_setdetachstate(&rt_attr,PTHREAD_CREATE_DETACHED); + pthread_mutex_init(&rt_mutex, 0); + pthread_cond_init(&rt_cond, 0); + +#define CYCLES 10000 +#define THREADS 10 + + trxman_init(); + run_test("trxman", test_trxman, THREADS,CYCLES); + trxman_destroy(); + diag("mallocs: %d\n", global_malloc); + + pthread_mutex_destroy(&rt_mutex); + pthread_cond_destroy(&rt_cond); + pthread_attr_destroy(&rt_attr); + my_end(0); + return exit_status(); +} + diff --git a/unittest/mysys/my_atomic-t.c b/unittest/mysys/my_atomic-t.c index 1e7206441eb..8962131d45a 100644 --- a/unittest/mysys/my_atomic-t.c +++ b/unittest/mysys/my_atomic-t.c @@ -14,9 +14,6 @@ along with this program; if not, write to the Free Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ -//#define MY_ATOMIC_MODE_RWLOCKS -//#define MY_ATOMIC_MODE_DUMMY - #include #include @@ -41,7 +38,7 @@ pthread_handler_t test_atomic_add_handler(void *arg) int32 x; for (x=((int)(intptr)(&m)); m ; m--) { - x=x*m+0x87654321; + x=(x*m+0x87654321) & INT_MAX32; my_atomic_rwlock_wrlock(&rwl); my_atomic_add32(&a32, x); my_atomic_rwlock_wrunlock(&rwl); @@ -111,7 +108,7 @@ pthread_handler_t test_atomic_cas_handler(void *arg) my_atomic_rwlock_wrlock(&rwl); y=my_atomic_load32(&a32); my_atomic_rwlock_wrunlock(&rwl); - x=x*m+0x87654321; + x=(x*m+0x87654321) & INT_MAX32; do { my_atomic_rwlock_wrlock(&rwl); ok=my_atomic_cas32(&a32, &y, y+x); @@ -171,7 +168,7 @@ pthread_handler_t test_lf_alloc(void *arg) for (x=((int)(intptr)(&m)); m ; m--) { TLA *node1, *node2; - x=x*m+0x87654321; + x=(x*m+0x87654321) & INT_MAX32; node1=(TLA *)lf_alloc_new(pins); node1->data=x; y+=node1->data; @@ -217,7 +214,7 @@ pthread_handler_t test_lf_hash(void *arg) y=x; for (i=0; i < N_TLH; i++) { - x=x*(m+i)+0x87654321; + x=(x*(m+i)+0x87654321) & INT_MAX32; z=(x<0) ? -x : x; if (lf_hash_insert(&lf_hash, pins, &z)) { @@ -227,7 +224,7 @@ pthread_handler_t test_lf_hash(void *arg) } for (i=0; i < N_TLH; i++) { - y=y*(m+i)+0x87654321; + y=(y*(m+i)+0x87654321) & INT_MAX32; z=(y<0) ? -y : y; if (lf_hash_delete(&lf_hash, pins, (uchar *)&z, sizeof(z))) sum-=z; @@ -307,8 +304,8 @@ int main() test_atomic("lf_alloc", test_lf_alloc, THREADS,CYCLES); test_atomic("lf_hash", test_lf_hash, THREADS,CYCLES); - lf_hash_end(&lf_hash); - lf_alloc_end(&lf_allocator); + lf_hash_destroy(&lf_hash); + lf_alloc_destroy(&lf_allocator); pthread_mutex_destroy(&mutex); pthread_cond_destroy(&cond); pthread_attr_destroy(&thr_attr); -- cgit v1.2.1 From cd40855e9d8ee3a70f53aa061a9e303eec38a52d Mon Sep 17 00:00:00 2001 From: unknown Date: Thu, 17 Aug 2006 15:22:54 +0200 Subject: lf_hash: only data nodes use lf_alloc now, dummy nodes are malloc'ed directly mysys/lf_hash.c: only data nodes use lf_alloc now, dummy nodes are malloc'ed directly --- mysys/lf_hash.c | 17 +++++++++++------ 1 file changed, 11 insertions(+), 6 deletions(-) diff --git a/mysys/lf_hash.c b/mysys/lf_hash.c index 57936ea7b1f..736c3ea4887 100644 --- a/mysys/lf_hash.c +++ b/mysys/lf_hash.c @@ -18,7 +18,6 @@ extensible hash TODO - dummy nodes use element_size=0 try to get rid of dummy nodes ? */ #include @@ -241,7 +240,10 @@ void lf_hash_destroy(LF_HASH *hash) while (el) { intptr next=el->link; - lf_alloc_real_free(&hash->alloc, el); + if (el->hashnr & 1) + lf_alloc_real_free(&hash->alloc, el); + else + my_free((void *)el, MYF(0)); el=(LF_SLIST *)next; } lf_alloc_destroy(&hash->alloc); @@ -263,7 +265,7 @@ int lf_hash_insert(LF_HASH *hash, LF_PINS *pins, const void *data) lf_lock_by_pins(pins); node=(LF_SLIST *)_lf_alloc_new(pins); memcpy(node+1, data, hash->element_size); - node->key= hash_key(hash, node+1, &node->keylen); + node->key= hash_key(hash, (uchar *)(node+1), &node->keylen); hashnr= calc_hash(hash, node->key, node->keylen); bucket= hashnr % hash->size; el=_lf_dynarray_lvalue(&hash->array, bucket); @@ -329,17 +331,20 @@ void *lf_hash_search(LF_HASH *hash, LF_PINS *pins, const void *key, uint keylen) return found+1; } +static char *dummy_key=""; + static void initialize_bucket(LF_HASH *hash, LF_SLIST * volatile *node, uint bucket, LF_PINS *pins) { uint parent= my_clear_highest_bit(bucket); - LF_SLIST *dummy=_lf_alloc_new(pins), **tmp=0, *cur; + LF_SLIST *dummy=(LF_SLIST *)my_malloc(sizeof(LF_SLIST), MYF(MY_WME)); + LF_SLIST **tmp=0, *cur; LF_SLIST * volatile *el=_lf_dynarray_lvalue(&hash->array, parent); if (*el == NULL && bucket) initialize_bucket(hash, el, parent, pins); dummy->hashnr=my_reverse_bits(bucket); - LINT_INIT(dummy->key); - LINT_INIT(dummy->keylen); + dummy->key=dummy_key; + dummy->keylen=0; if ((cur= linsert(el, dummy, pins, 0))) { _lf_alloc_free(pins, dummy); -- cgit v1.2.1 From d83bc171b6775281d556604b47d6d13a8bf5cecf Mon Sep 17 00:00:00 2001 From: unknown Date: Wed, 23 Aug 2006 17:27:21 +0200 Subject: renamed global variables --- storage/maria/trxman.c | 23 ++++++++++------------- storage/maria/trxman.h | 2 +- unittest/maria/trxman-t.c | 3 +-- 3 files changed, 12 insertions(+), 16 deletions(-) diff --git a/storage/maria/trxman.c b/storage/maria/trxman.c index 3c64d93183a..a3e746af9ca 100644 --- a/storage/maria/trxman.c +++ b/storage/maria/trxman.c @@ -8,7 +8,7 @@ TRX active_list_min, active_list_max, committed_list_min, committed_list_max, *pool; pthread_mutex_t LOCK_trx_list; -uint active_transactions; +uint trxman_active_transactions, trxman_allocated_transactions; TrID global_trid_generator; TRX **short_id_to_trx; @@ -30,7 +30,8 @@ int trxman_init() active_list_max.next= active_list_min.prev= 0; active_list_max.prev= &active_list_min; active_list_min.next= &active_list_max; - active_transactions= 0; + trxman_active_transactions= 0; + trxman_allocated_transactions= 0; committed_list_max.commit_trid= ~0; committed_list_max.next= committed_list_min.prev= 0; @@ -54,7 +55,7 @@ int trxman_init() int trxman_destroy() { DBUG_ASSERT(trid_to_trx.count == 0); - DBUG_ASSERT(active_transactions == 0); + DBUG_ASSERT(trxman_active_transactions == 0); DBUG_ASSERT(active_list_max.prev == &active_list_min); DBUG_ASSERT(active_list_min.next == &active_list_max); DBUG_ASSERT(committed_list_max.prev == &committed_list_min); @@ -62,7 +63,7 @@ int trxman_destroy() while (pool) { TRX *tmp=pool->next; - my_free(pool, MYF(0)); + my_free((void *)pool, MYF(0)); pool=tmp; } lf_hash_destroy(&trid_to_trx); @@ -93,21 +94,17 @@ static void set_short_id(TRX *trx) trx->short_id= i; } -extern int global_malloc; TRX *trxman_new_trx() { TRX *trx; - my_atomic_add32(&active_transactions, 1); + my_atomic_add32(&trxman_active_transactions, 1); /* - we need a mutex here to ensure that - transactions in the active list are ordered by the trid. - So, incrementing global_trid_generator and - adding to the list must be atomic. + see trxman_end_trx to see why we need a mutex here and as we have a mutex, we can as well do everything - under it - allocating a TRX, incrementing active_transactions, + under it - allocating a TRX, incrementing trxman_active_transactions, setting trx->min_read_from. Note that all the above is fast. generating short_id may be slow, @@ -123,7 +120,7 @@ TRX *trxman_new_trx() if (!trx) { trx=(TRX *)my_malloc(sizeof(TRX), MYF(MY_WME)); - global_malloc++; + trxman_allocated_transactions++; } if (!trx) return 0; @@ -213,7 +210,7 @@ void trxman_end_trx(TRX *trx, my_bool commit) } pthread_mutex_unlock(&LOCK_trx_list); - my_atomic_add32(&active_transactions, -1); + my_atomic_add32(&trxman_active_transactions, -1); while (free_me) { diff --git a/storage/maria/trxman.h b/storage/maria/trxman.h index ae794470a47..5ac989d03a4 100644 --- a/storage/maria/trxman.h +++ b/storage/maria/trxman.h @@ -12,7 +12,7 @@ typedef struct st_transaction #define SHORT_ID_MAX 65535 -extern uint active_transactions; +extern uint trxman_active_transactions, trxman_allocated_transactions; extern TRX **short_id_to_trx; extern my_atomic_rwlock_t LOCK_short_id_to_trx; diff --git a/unittest/maria/trxman-t.c b/unittest/maria/trxman-t.c index a29bf5abc8e..7536e5534bf 100644 --- a/unittest/maria/trxman-t.c +++ b/unittest/maria/trxman-t.c @@ -106,7 +106,6 @@ void run_test(const char *test, pthread_handler handler, int n, int m) ok(litmus == 0, "tested %s in %g secs (%d)", test, ((double)now)/1e7, litmus); } -int global_malloc=0; int main() { plan(1); @@ -127,7 +126,7 @@ int main() trxman_init(); run_test("trxman", test_trxman, THREADS,CYCLES); trxman_destroy(); - diag("mallocs: %d\n", global_malloc); + diag("mallocs: %d\n", trxman_allocated_transactions); pthread_mutex_destroy(&rt_mutex); pthread_cond_destroy(&rt_cond); -- cgit v1.2.1 From 87aa4ae2d35d0291bbfc47ece3f19cfa8746a573 Mon Sep 17 00:00:00 2001 From: unknown Date: Tue, 29 Aug 2006 11:30:35 +0200 Subject: just comment changes and line formatting changes. storage/maria/checkpoint.c: comments storage/maria/least_recently_dirtied.c: comments storage/maria/ma_check.c: line formatting changes neglected from my last merge storage/maria/ma_ft_parser.c: it reduces the diff of MyISAM vs Maria :) storage/maria/recovery.c: comments --- storage/maria/checkpoint.c | 8 ++++---- storage/maria/least_recently_dirtied.c | 14 ++++++++++++-- storage/maria/ma_check.c | 17 +++++++++-------- storage/maria/ma_ft_parser.c | 2 +- storage/maria/recovery.c | 12 +----------- 5 files changed, 27 insertions(+), 26 deletions(-) diff --git a/storage/maria/checkpoint.c b/storage/maria/checkpoint.c index 151efe6ad4f..22e7b93d2f4 100644 --- a/storage/maria/checkpoint.c +++ b/storage/maria/checkpoint.c @@ -46,7 +46,8 @@ CHECKPOINT_LEVEL synchronous_checkpoint_in_progress= NONE; /* Used by MySQL client threads requesting a checkpoint (like "ALTER MARIA - ENGINE DO CHECKPOINT"), and probably by maria_panic(). + ENGINE DO CHECKPOINT"), and probably by maria_panic(), and at the end of the + UNDO recovery phase. */ my_bool execute_synchronous_checkpoint(CHECKPOINT_LEVEL level) { @@ -342,8 +343,6 @@ log_write_record(...) (requestor does not wait for completion, and does not even later check the result). In real life it will be called by log_write_record(). - which explicitely wants to do checkpoint (ALTER ENGINE CHECKPOINT - checkpoint_level). */ void request_asynchronous_checkpoint(CHECKPOINT_LEVEL level); { @@ -359,7 +358,8 @@ void request_asynchronous_checkpoint(CHECKPOINT_LEVEL level); MAX_LOG_BYTES_WRITTEN_BETWEEN_CHECKPOINTS is passed), so it may not be a good idea for each of them to broadcast a cond to wake up the background checkpoint thread. We just don't broacast a cond, the checkpoint thread - will notice our request in max a few seconds. + (see least_recently_dirtied.c) will notice our request in max a few + seconds. */ checkpoint_request= level; /* post request */ } diff --git a/storage/maria/least_recently_dirtied.c b/storage/maria/least_recently_dirtied.c index bca13ca6f1f..c6285fe47cd 100644 --- a/storage/maria/least_recently_dirtied.c +++ b/storage/maria/least_recently_dirtied.c @@ -181,7 +181,15 @@ flush_one_group_from_LRD() */ } -/* flushes all page from LRD up to approximately rec_lsn>=max_lsn */ +/* + Flushes all page from LRD up to approximately rec_lsn>=max_lsn. + This is approximate because we flush groups, and because the LRD list may + not be exactly sorted by rec_lsn (because for a big row, all pages of the + row are inserted into the LRD with rec_lsn being the LSN of the REDO for the + first page, so if there are concurrent insertions, the last page of the big + row may have a smaller rec_lsn than the previous pages inserted by + concurrent inserters). +*/ int flush_all_LRD_to_lsn(LSN max_lsn) { lock(global_LRD_mutex); @@ -191,7 +199,9 @@ int flush_all_LRD_to_lsn(LSN max_lsn) { if (flush_one_group_from_LRD()) /* will unlock LRD mutex */ return 1; - /* scheduler may preempt us here so that we don't take full CPU */ + /* + The scheduler may preempt us here as we released the mutex; this is good. + */ lock(global_LRD_mutex); } unlock(global_LRD_mutex); diff --git a/storage/maria/ma_check.c b/storage/maria/ma_check.c index 69d863e6366..624df8a7881 100644 --- a/storage/maria/ma_check.c +++ b/storage/maria/ma_check.c @@ -460,10 +460,10 @@ int maria_chk_key(HA_CHECK *param, register MARIA_HA *info) auto_increment= ma_retrieve_auto_increment(info, info->rec_buff); if (auto_increment > info->s->state.auto_increment) { - _ma_check_print_warning(param, - "Auto-increment value: %s is smaller than max used value: %s", - llstr(info->s->state.auto_increment,buff2), - llstr(auto_increment, buff)); + _ma_check_print_warning(param, "Auto-increment value: %s is smaller " + "than max used value: %s", + llstr(info->s->state.auto_increment,buff2), + llstr(auto_increment, buff)); } if (param->testflag & T_AUTO_INC) { @@ -481,8 +481,8 @@ int maria_chk_key(HA_CHECK *param, register MARIA_HA *info) { /* Don't count this as a real warning, as mariachk can't correct it */ uint save=param->warning_printed; - _ma_check_print_warning(param, - "Found row where the auto_increment column has the value 0"); + _ma_check_print_warning(param, "Found row where the auto_increment " + "column has the value 0"); param->warning_printed=save; } maria_extra(info,HA_EXTRA_NO_KEYREAD,0); @@ -1165,8 +1165,9 @@ int maria_chk_data_link(HA_CHECK *param, MARIA_HA *info,int extend) SEARCH_SAME, info->s->state.key_root[key]); if (search_result) { - _ma_check_print_error(param,"Record at: %10s Can't find key for index: %2d", - llstr(start_recpos,llbuff),key+1); + _ma_check_print_error(param,"Record at: %10s " + "Can't find key for index: %2d", + llstr(start_recpos,llbuff),key+1); if (error++ > MAXERR || !(param->testflag & T_VERBOSE)) goto err2; } diff --git a/storage/maria/ma_ft_parser.c b/storage/maria/ma_ft_parser.c index 1c6e0267d53..e5ec97b090d 100644 --- a/storage/maria/ma_ft_parser.c +++ b/storage/maria/ma_ft_parser.c @@ -204,8 +204,8 @@ byte maria_ft_simple_get_word(CHARSET_INFO *cs, byte **start, const byte *end, FT_WORD *word, my_bool skip_stopwords) { byte *doc= *start; - int ctype; uint mwc, length, mbl; + int ctype; DBUG_ENTER("maria_ft_simple_get_word"); do diff --git a/storage/maria/recovery.c b/storage/maria/recovery.c index 589e0971686..babf7507ef1 100644 --- a/storage/maria/recovery.c +++ b/storage/maria/recovery.c @@ -221,10 +221,6 @@ int recovery() /* mark that checkpoint requests are now allowed. */ - /* - when all rollback threads have terminated, somebody should print "rollback - finished" to the error log. - */ } pthread_handler_decl rollback_background_thread() @@ -248,13 +244,7 @@ pthread_handler_decl rollback_background_thread() { /* All rollback threads are done. Print "rollback finished" to the error - log. The UNDO phase has the reputation of being a slow operation - (slower than the REDO phase), so taking a checkpoint at the end of it is - intelligent, but as this UNDO phase generates REDOs and CLR_ENDs, if it - did a lot of work then the "automatic checkpoint when much has been - written to the log" will do it; and if the UNDO phase didn't do a lot of - work, no need for a checkpoint. If we change our mind and want to force - a checkpoint at the end of the UNDO phase, simply call it here. + log and take a full checkpoint. */ } unlock_mutex(rollback_threads); -- cgit v1.2.1 From 52191ea4d8dda55173a7c721371faf15bf0ea39c Mon Sep 17 00:00:00 2001 From: unknown Date: Tue, 29 Aug 2006 22:10:06 +0200 Subject: importing Sanja's changes to the control file, with my changes on them. mysys/my_pread.c: print errno in case of error storage/maria/control_file.c: importing Sanja's changes, with my minor changes on them :) storage/maria/control_file.h: importing Sanja's changes, with my minor changes on them :) --- mysys/my_pread.c | 3 +- storage/maria/control_file.c | 154 +++++++++++++++++++++++++++++++++---------- storage/maria/control_file.h | 26 +++++++- 3 files changed, 147 insertions(+), 36 deletions(-) diff --git a/mysys/my_pread.c b/mysys/my_pread.c index ac52895efe9..072c18ae7f6 100644 --- a/mysys/my_pread.c +++ b/mysys/my_pread.c @@ -155,7 +155,8 @@ uint my_pwrite(int Filedes, const byte *Buffer, uint Count, my_off_t offset, Count-=writenbytes; offset+=writenbytes; } - DBUG_PRINT("error",("Write only %d bytes",writenbytes)); + DBUG_PRINT("error",("Write only %d bytes, error: %d", + writenbytes, my_errno)); #ifndef NO_BACKGROUND #ifdef THREAD if (my_thread_var->abort) diff --git a/storage/maria/control_file.c b/storage/maria/control_file.c index 897e0b0f0ee..22ea860456a 100644 --- a/storage/maria/control_file.c +++ b/storage/maria/control_file.c @@ -4,74 +4,162 @@ Does not compile yet. */ +#include "maria_def.h" + + /* Here is the implementation of this module */ -/* Control file is 512 bytes (a disk sector), to be as atomic as possible */ +/* should be sector size for atomic write operation */ +#define STAT_FILE_FILENO_SIZE 4 +#define STAT_FILE_FILEOFFSET_SIZE 4 +#define STAT_FILE_LSN_SIZE (STAT_FILE_FILENO_SIZE + STAT_FILE_FILEOFFSET_SIZE) +#define STAT_FILE_MAX_SIZE (STAT_FILE_LSN_SIZE + STAT_FILE_FILENO_SIZE) + + +LSN last_checkpoint_lsn_at_startup; +uint32 last_logno_at_startup; + + +/* + Control file is less then 512 bytes (a disk sector), + to be as atomic as possible +*/ +static int control_file_fd; -int control_file_fd; /* + Initialize control file subsystem + + SYNOPSIS + control_file_create_or_open() + Looks for the control file. If absent, it's a fresh start, create file. If present, read it to find out last checkpoint's LSN and last log. Called at engine's start. + + RETURN + 0 - OK + 1 - Error */ int control_file_create_or_open() { - char buffer[4]; + char buffer[STAT_FILE_MAX_SIZE]; + char name[FN_REFLEN]; + MY_STAT stat_buff; + /* name is concatenation of Maria's home dir and "control" */ - if ((control_file_fd= my_open(name, O_RDWR)) < 0) + if (fn_format(name, "control", maria_data_root, "", MYF(MY_WME)) == NullS) + return 1; + + if ((control_file_fd= my_open(name, + O_CREAT | O_BINARY | /*O_DIRECT |*/ O_RDWR, + MYF(MY_WME))) < 0) + return 1; + + /* + TODO: from "man fsync" on Linux: + "fsync does not necessarily ensure that the entry in the direc- tory + containing the file has also reached disk. For that an explicit + fsync on the file descriptor of the directory is also needed." + So if we just created the file we should sync the directory. + Maybe there should be a flag of my_create() to do this. + */ + + if (my_stat(name, &stat_buff, MYF(MY_WME)) == NULL) + return 1; + + if (stat_buff.st_size < STAT_FILE_MAX_SIZE) { - /* failure, try to create it */ - if ((control_file_fd= my_create(name, O_RDWR)) < 0) - return 1; /* - So this is a start from scratch, to be safer we should make sure that - there are no logs or data/index files around (indeed it could be that - the control file alone was deleted or not restored, and we should not - go on with life at this point. - For now we trust (this is alpha version), but for beta if would be great - to verify. + File shorter than expected (either we just created it, or a previous run + crashed between creation and first write); do first write. + */ + char buffer[STAT_FILE_MAX_SIZE]; + /* + To be safer we should make sure that there are no logs or data/index + files around (indeed it could be that the control file alone was deleted + or not restored, and we should not go on with life at this point). + + TODO: For now we trust (this is alpha version), but for beta if would + be great to verify. We could have a tool which can rebuild the control file, by reading the directory of logs, finding the newest log, reading it to find last checkpoint... Slow but can save your db. */ - last_checkpoint_lsn_at_startup= 0; - last_log_name_at_startup= NULL; - return 0; + last_checkpoint_lsn_at_startup.file_no= CONTROL_FILE_IMPOSSIBLE_LOGNO; + last_checkpoint_lsn_at_startup.rec_offset= 0; + last_logno_at_startup= CONTROL_FILE_IMPOSSIBLE_LOGNO; + + /* init the file with these "undefined" values */ + return control_file_write_and_force(last_checkpoint_lsn_at_startup, + last_logno_at_startup); } /* Already existing file, read it */ - if (my_read(control_file_fd, buffer, 8, MYF(MY_FNABP))) - return 1; - last_checkpoint_lsn_at_startup= uint8korr(buffer); - if (last_log_name_at_startup= my_malloc(512-8+1)) - return 1; - if (my_read(control_file_fd, last_log_name_at_startup, 512-8), MYF(MY_FNABP)) + if (my_read(control_file_fd, buffer, STAT_FILE_MAX_SIZE, + MYF(MY_FNABP | MY_WME))) return 1; - last_log_name[512-8]= 0; /* end zero to be nice */ + last_checkpoint_lsn_at_startup.file_no= uint4korr(buffer); + last_checkpoint_lsn_at_startup.rec_offset= uint4korr(buffer + + STAT_FILE_FILENO_SIZE); + last_logno_at_startup= uint4korr(buffer + STAT_FILE_LSN_SIZE); return 0; } + /* Write information durably to the control file. + + SYNOPSIS + control_file_write_and_force() + checkpoint_lsn LSN of checkpoint + log_no last log file number + args_to_write bitmap of 1 (write the LSN) and 2 (write the LOGNO) + Called when we have created a new log (after syncing this log's creation) and when we have written a checkpoint (after syncing this log record). + + RETURN + 0 - OK + 1 - Error */ -int control_file_write_and_force(LSN lsn, char *log_name) + +int control_file_write_and_force(LSN *checkpoint_lsn, uint32 log_no, + uint args_to_write) { - char buffer[512]; - uint start=8,end=8; - if (lsn != 0) /* LSN was specified */ + char buffer[STAT_FILE_MAX_SIZE]; + uint start= STAT_FILE_LSN_SIZE, end= STAT_FILE_LSN_SIZE; + /* + If LSN was specified... + + rec_offset can't be 0 in real LSN, because all files have header page + */ + if ((args_to_write & 1) && checkpoint_lsn) /* write checkpoint LSN */ { start= 0; - int8store(buffer, lsn); + int4store(buffer, checkpoint_lsn->file_no); + int4store(buffer + STAT_FILE_FILENO_SIZE, checkpoint_lsn->rec_offset); } - if (log_name != NULL) /* log name was specified */ + if (args_to_write & 2) /* write logno */ { - end= 512; - memcpy(buffer+8, log_name, 512-8); + end= STAT_FILE_MAX_SIZE; + int4store(buffer + STAT_FILE_LSN_SIZE, log_no); } DBUG_ASSERT(start != end); - return (my_pwrite(control_file_fd, buffer, end-start, start, MYF(MY_FNABP)) || - my_sync(control_file_fd)) + return (my_pwrite(control_file_fd, buffer + start, end - start, start, + MYF(MY_FNABP | MY_WME)) || + my_sync(control_file_fd, MYF(MY_WME))); +} + + +/* + Free resources taken by control file subsystem + + SYNOPSIS + control_file_end() +*/ + +void control_file_end() +{ + my_close(control_file_fd, MYF(MY_WME)); } diff --git a/storage/maria/control_file.h b/storage/maria/control_file.h index 522e7565341..add466e6cbd 100644 --- a/storage/maria/control_file.h +++ b/storage/maria/control_file.h @@ -4,10 +4,25 @@ Does not compile yet. */ +#ifndef _control_file_h +#define _control_file_h + +/* indicate absence of the log file number */ +#define CONTROL_FILE_IMPOSSIBLE_LOGNO 0xFFFFFFFF + /* Here is the interface of this module */ +/* + LSN of the last checkoint + (if last_checkpoint_lsn_at_startup.file_no == CONTROL_FILE_IMPOSSIBLE_LOGNO + then there was never a checkpoint) +*/ extern LSN last_checkpoint_lsn_at_startup; -extern char *last_log_name_at_startup; +/* + Last log number at startup time (if last_logno_at_startup == + CONTROL_FILE_IMPOSSIBLE_LOGNO then there is no log file yet) +*/ +extern uint32 last_logno_at_startup; /* Looks for the control file. If absent, it's a fresh start, create file. @@ -21,4 +36,11 @@ int control_file_create_or_open(); Called when we have created a new log (after syncing this log's creation) and when we have written a checkpoint (after syncing this log record). */ -int control_file_write_and_force(LSN lsn, char *log_name); +int control_file_write_and_force(LSN *checkpoint_lsn, uint32 log_no, + uint args_to_write); + + +/* Free resources taken by control file subsystem */ +void control_file_end(); + +#endif -- cgit v1.2.1 From 8a2901a7584a87a326eee67b00ccb9eb8519ea03 Mon Sep 17 00:00:00 2001 From: unknown Date: Wed, 30 Aug 2006 10:55:27 +0200 Subject: write a magic string "MACF" (MAria Control File) at the start of the control file. lsn8store, lsn8korr. storage/maria/control_file.h: parameter changes name --- storage/maria/control_file.c | 118 ++++++++++++++++++++++++++++++------------- storage/maria/control_file.h | 2 +- 2 files changed, 83 insertions(+), 37 deletions(-) diff --git a/storage/maria/control_file.c b/storage/maria/control_file.c index 22ea860456a..70eb62c645b 100644 --- a/storage/maria/control_file.c +++ b/storage/maria/control_file.c @@ -9,11 +9,20 @@ /* Here is the implementation of this module */ -/* should be sector size for atomic write operation */ -#define STAT_FILE_FILENO_SIZE 4 -#define STAT_FILE_FILEOFFSET_SIZE 4 -#define STAT_FILE_LSN_SIZE (STAT_FILE_FILENO_SIZE + STAT_FILE_FILEOFFSET_SIZE) -#define STAT_FILE_MAX_SIZE (STAT_FILE_LSN_SIZE + STAT_FILE_FILENO_SIZE) +/* + a control file contains 3 objects: magic string, LSN of last checkpoint, + number of last log. +*/ + +/* total size should be < sector size for atomic write operation */ +#define CONTROL_FILE_MAGIC_STRING "MACF" +#define CONTROL_FILE_MAGIC_STRING_OFFSET 0 +#define CONTROL_FILE_MAGIC_STRING_SIZE sizeof(CONTROL_FILE_MAGIC_STRING) +#define CONTROL_FILE_LSN_OFFSET (CONTROL_FILE_MAGIC_STRING_OFFSET + CONTROL_FILE_MAGIC_STRING_SIZE) +#define CONTROL_FILE_LSN_SIZE (4+4) +#define CONTROL_FILE_FILENO_OFFSET (CONTROL_FILE_LSN_OFFSET + CONTROL_FILE_LSN_SIZE) +#define CONTROL_FILE_FILENO_SIZE 4 +#define CONTROL_FILE_MAX_SIZE (CONTROL_FILE_FILENO_OFFSET + CONTROL_FILE_FILENO_SIZE) LSN last_checkpoint_lsn_at_startup; @@ -26,6 +35,19 @@ uint32 last_logno_at_startup; */ static int control_file_fd; +static void lsn8store(char *buffer, LSN *lsn) +{ + int4store(buffer, lsn->file_no); + int4store(buffer + CONTROL_FILE_FILENO_SIZE, lsn->rec_offset); +} + +static LSN lsn8korr(char *buffer) +{ + LSN tmp; + tmp.file_no= uint4korr(buffer); + tmp.rec_offset= uint4korr(buffer + CONTROL_FILE_FILENO_SIZE); + return tmp; +} /* Initialize control file subsystem @@ -43,10 +65,18 @@ static int control_file_fd; */ int control_file_create_or_open() { - char buffer[STAT_FILE_MAX_SIZE]; + char buffer[CONTROL_FILE_MAX_SIZE]; char name[FN_REFLEN]; MY_STAT stat_buff; + /* + If you change sizes in the #defines, you at least have to change the + "*store" and "*korr" calls in this file, and can even create backward + compatibility problems. Beware! + */ + DBUG_ASSERT(CONTROL_FILE_LSN_SIZE == (4+4)); + DBUG_ASSERT(CONTROL_FILE_FILENO_SIZE == 4); + /* name is concatenation of Maria's home dir and "control" */ if (fn_format(name, "control", maria_data_root, "", MYF(MY_WME)) == NullS) return 1; @@ -68,13 +98,13 @@ int control_file_create_or_open() if (my_stat(name, &stat_buff, MYF(MY_WME)) == NULL) return 1; - if (stat_buff.st_size < STAT_FILE_MAX_SIZE) + if (stat_buff.st_size < CONTROL_FILE_MAX_SIZE) { /* File shorter than expected (either we just created it, or a previous run crashed between creation and first write); do first write. */ - char buffer[STAT_FILE_MAX_SIZE]; + char buffer[CONTROL_FILE_MAX_SIZE]; /* To be safer we should make sure that there are no logs or data/index files around (indeed it could be that the control file alone was deleted @@ -87,67 +117,83 @@ int control_file_create_or_open() directory of logs, finding the newest log, reading it to find last checkpoint... Slow but can save your db. */ - last_checkpoint_lsn_at_startup.file_no= CONTROL_FILE_IMPOSSIBLE_LOGNO; + last_checkpoint_lsn_at_startup.file_no= CONTROL_FILE_IMPOSSIBLE_FILENO; last_checkpoint_lsn_at_startup.rec_offset= 0; - last_logno_at_startup= CONTROL_FILE_IMPOSSIBLE_LOGNO; + last_logno_at_startup= CONTROL_FILE_IMPOSSIBLE_FILENO; /* init the file with these "undefined" values */ return control_file_write_and_force(last_checkpoint_lsn_at_startup, - last_logno_at_startup); + last_logno_at_startup, + CONTROL_FILE_WRITE_ALL); } /* Already existing file, read it */ - if (my_read(control_file_fd, buffer, STAT_FILE_MAX_SIZE, + if (my_read(control_file_fd, buffer, CONTROL_FILE_MAX_SIZE, MYF(MY_FNABP | MY_WME))) return 1; - last_checkpoint_lsn_at_startup.file_no= uint4korr(buffer); - last_checkpoint_lsn_at_startup.rec_offset= uint4korr(buffer + - STAT_FILE_FILENO_SIZE); - last_logno_at_startup= uint4korr(buffer + STAT_FILE_LSN_SIZE); + if (memcmp(buffer + CONTROL_FILE_MAGIC_STRING_OFFSET, + CONTROL_FILE_MAGIC_STRING, CONTROL_FILE_MAGIC_STRING_SIZE)) + return 1; + last_checkpoint_lsn_at_startup= lsn8korr(buffer + CONTROL_FILE_LSN_OFFSET); + last_logno_at_startup= uint4korr(buffer + CONTROL_FILE_FILENO_OFFSET); return 0; } +#define CONTROL_FILE_WRITE_ALL 0 /* write all 3 objects */ +#define CONTROL_FILE_WRITE_ONLY_LSN 1 +#define CONTROL_FILE_WRITE_ONLY_LOGNO 2 /* Write information durably to the control file. SYNOPSIS control_file_write_and_force() - checkpoint_lsn LSN of checkpoint + checkpoint_lsn LSN of last checkpoint log_no last log file number - args_to_write bitmap of 1 (write the LSN) and 2 (write the LOGNO) + objs_to_write what we should write Called when we have created a new log (after syncing this log's creation) and when we have written a checkpoint (after syncing this log record). + NOTE + We always want to do one single my_pwrite() here to be as atomic as + possible. + RETURN 0 - OK 1 - Error */ int control_file_write_and_force(LSN *checkpoint_lsn, uint32 log_no, - uint args_to_write) + uint objs_to_write) { - char buffer[STAT_FILE_MAX_SIZE]; - uint start= STAT_FILE_LSN_SIZE, end= STAT_FILE_LSN_SIZE; - /* - If LSN was specified... - - rec_offset can't be 0 in real LSN, because all files have header page - */ - if ((args_to_write & 1) && checkpoint_lsn) /* write checkpoint LSN */ + char buffer[CONTROL_FILE_MAX_SIZE]; + uint start, size; + memcpy(buffer + CONTROL_FILE_MAGIC_STRING_OFFSET, + CONTROL_FILE_MAGIC_STRING, CONTROL_FILE_MAGIC_STRING_SIZE); + /* write checkpoint LSN */ + if (checkpoint_lsn) + lsn8store(buffer + CONTROL_FILE_LSN_OFFSET, checkpoint_lsn); + /* write logno */ + int4store(buffer + CONTROL_FILE_FILENO_OFFSET, log_no); + if (objs_to_write == CONTROL_FILE_WRITE_ALL) + { + start= CONTROL_FILE_MAGIC_STRING_OFFSET; + size= CONTROL_FILE_MAX_SIZE; + } + else if (objs_to_write == CONTROL_FILE_WRITE_ONLY_LSN) { - start= 0; - int4store(buffer, checkpoint_lsn->file_no); - int4store(buffer + STAT_FILE_FILENO_SIZE, checkpoint_lsn->rec_offset); + start= CONTROL_FILE_LSN_OFFSET; + size= CONTROL_FILE_LSN_SIZE; } - if (args_to_write & 2) /* write logno */ + else if (objs_to_write == CONTROL_FILE_WRITE_ONLY_LOGNO) { - end= STAT_FILE_MAX_SIZE; - int4store(buffer + STAT_FILE_LSN_SIZE, log_no); + start= CONTROL_FILE_FILENO_OFFSET; + size= CONTROL_FILE_FILENO_SIZE; } - DBUG_ASSERT(start != end); - return (my_pwrite(control_file_fd, buffer + start, end - start, start, - MYF(MY_FNABP | MY_WME)) || + else /* incorrect value of objs_to_write */ + DBUG_ASSERT(0); + return (my_pwrite(control_file_fd, buffer + start, size, + start, MYF(MY_FNABP | MY_WME)) || my_sync(control_file_fd, MYF(MY_WME))); } diff --git a/storage/maria/control_file.h b/storage/maria/control_file.h index add466e6cbd..66a1f225cd8 100644 --- a/storage/maria/control_file.h +++ b/storage/maria/control_file.h @@ -37,7 +37,7 @@ int control_file_create_or_open(); and when we have written a checkpoint (after syncing this log record). */ int control_file_write_and_force(LSN *checkpoint_lsn, uint32 log_no, - uint args_to_write); + uint objs_to_write); /* Free resources taken by control file subsystem */ -- cgit v1.2.1 From a1f25544d531b8909e58d36734ee6f65c1e189d5 Mon Sep 17 00:00:00 2001 From: unknown Date: Fri, 1 Sep 2006 17:53:10 +0200 Subject: WL#3234 "Maria - control file manager" - fixes to the control file module - unit test for it - renames of all Maria files I created to start with ma_ storage/maria/ma_checkpoint.c: Rename: storage/maria/checkpoint.c -> storage/maria/ma_checkpoint.c storage/maria/ma_checkpoint.h: Rename: storage/maria/checkpoint.h -> storage/maria/ma_checkpoint.h storage/maria/ma_least_recently_dirtied.c: Rename: storage/maria/least_recently_dirtied.c -> storage/maria/ma_least_recently_dirtied.c storage/maria/ma_least_recently_dirtied.h: Rename: storage/maria/least_recently_dirtied.h -> storage/maria/ma_least_recently_dirtied.h storage/maria/ma_recovery.c: Rename: storage/maria/recovery.c -> storage/maria/ma_recovery.c storage/maria/ma_recovery.h: Rename: storage/maria/recovery.h -> storage/maria/ma_recovery.h storage/maria/Makefile.am: control file module and its unit test program storage/maria/ma_control_file.c: DBUG_ tags. Fix for gcc warnings. log_no -> logno (I felt "_no" sounded like a standalone "No" word). ma_ prefix for some functions. last_checkpoint_lsn_at_startup -> last_checkpoint_lsn (no need to make special vars for the values at startup). Same for last_logno. ma_control_file_write_and_force() now updates last_checkpoint_lsn and last_logno, the idea being that they belong to the module, others should not update them. And thus when the module shuts down, it zeroes those vars. storage/maria/ma_control_file.h: importing structs from Sanja to get the control file module to compile; we'll remove that when Sanja pushes the log handler. CONTROL_FILE_IMPOSSIBLE_LOGNO is 0, not FFFFFFFF. storage/maria/ma_control_file_test.c: Unit test program for the Maria control file module. Modelled after other ma_test* files in this directory (so, does not follow the unit test framework recently introduced with libtap; TODO as a task on all ma_test* programs). We test that writing to the control file works, and re-reading from it too, we check (by reading the file by ourselves) that its content on disk is correct, and check that a corrupted control file is detected. --- storage/maria/Makefile.am | 7 +- storage/maria/checkpoint.c | 375 ------------------------------ storage/maria/checkpoint.h | 19 -- storage/maria/control_file.c | 211 ----------------- storage/maria/control_file.h | 46 ---- storage/maria/least_recently_dirtied.c | 209 ----------------- storage/maria/least_recently_dirtied.h | 10 - storage/maria/ma_checkpoint.c | 375 ++++++++++++++++++++++++++++++ storage/maria/ma_checkpoint.h | 19 ++ storage/maria/ma_control_file.c | 234 +++++++++++++++++++ storage/maria/ma_control_file.h | 75 ++++++ storage/maria/ma_control_file_test.c | 290 +++++++++++++++++++++++ storage/maria/ma_least_recently_dirtied.c | 209 +++++++++++++++++ storage/maria/ma_least_recently_dirtied.h | 10 + storage/maria/ma_recovery.c | 252 ++++++++++++++++++++ storage/maria/ma_recovery.h | 10 + storage/maria/recovery.c | 252 -------------------- storage/maria/recovery.h | 10 - 18 files changed, 1478 insertions(+), 1135 deletions(-) delete mode 100644 storage/maria/checkpoint.c delete mode 100644 storage/maria/checkpoint.h delete mode 100644 storage/maria/control_file.c delete mode 100644 storage/maria/control_file.h delete mode 100644 storage/maria/least_recently_dirtied.c delete mode 100644 storage/maria/least_recently_dirtied.h create mode 100644 storage/maria/ma_checkpoint.c create mode 100644 storage/maria/ma_checkpoint.h create mode 100644 storage/maria/ma_control_file.c create mode 100644 storage/maria/ma_control_file.h create mode 100644 storage/maria/ma_control_file_test.c create mode 100644 storage/maria/ma_least_recently_dirtied.c create mode 100644 storage/maria/ma_least_recently_dirtied.h create mode 100644 storage/maria/ma_recovery.c create mode 100644 storage/maria/ma_recovery.h delete mode 100644 storage/maria/recovery.c delete mode 100644 storage/maria/recovery.h diff --git a/storage/maria/Makefile.am b/storage/maria/Makefile.am index d4315b4d446..e2689698d62 100644 --- a/storage/maria/Makefile.am +++ b/storage/maria/Makefile.am @@ -27,7 +27,7 @@ pkglib_LIBRARIES = libmaria.a bin_PROGRAMS = maria_chk maria_pack maria_ftdump maria_chk_DEPENDENCIES= $(LIBRARIES) maria_pack_DEPENDENCIES=$(LIBRARIES) -noinst_PROGRAMS = ma_test1 ma_test2 ma_test3 ma_rt_test ma_sp_test +noinst_PROGRAMS = ma_test1 ma_test2 ma_test3 ma_rt_test ma_sp_test ma_control_file_test noinst_HEADERS = maria_def.h ma_rt_index.h ma_rt_key.h ma_rt_mbr.h ma_sp_defs.h ma_fulltext.h ma_ftdefs.h ma_ft_test1.h ma_ft_eval.h ma_test1_DEPENDENCIES= $(LIBRARIES) ma_test2_DEPENDENCIES= $(LIBRARIES) @@ -53,8 +53,9 @@ libmaria_a_SOURCES = ma_init.c ma_open.c ma_extra.c ma_info.c ma_rkey.c \ ma_ft_update.c ma_ft_boolean_search.c \ ma_ft_nlq_search.c ft_maria.c ma_sort.c \ ma_rt_index.c ma_rt_key.c ma_rt_mbr.c ma_rt_split.c \ - ma_sp_key.c -CLEANFILES = test?.MA? FT?.MA? isam.log ma_test_all ma_rt_test.MA? sp_test.MA? + ma_sp_key.c \ + ma_control_file.c +CLEANFILES = test?.MA? FT?.MA? isam.log ma_test_all ma_rt_test.MA? sp_test.MA? maria_control DEFS = SUFFIXES = .sh diff --git a/storage/maria/checkpoint.c b/storage/maria/checkpoint.c deleted file mode 100644 index 22e7b93d2f4..00000000000 --- a/storage/maria/checkpoint.c +++ /dev/null @@ -1,375 +0,0 @@ -/* - WL#3071 Maria checkpoint - First version written by Guilhem Bichot on 2006-04-27. - Does not compile yet. -*/ - -/* Here is the implementation of this module */ - -/* - Summary: - - there are asynchronous checkpoints (a writer to the log notices that it's - been a long time since we last checkpoint-ed, so posts a request for a - background thread to do a checkpoint; does not care about the success of the - checkpoint). Then the checkpoint is done by the checkpoint thread, at an - unspecified moment ("later") (==soon, of course). - - there are synchronous checkpoints: a thread requests a checkpoint to - happen now and wants to know when it finishes and if it succeeded; then the - checkpoint is done by that same thread. -*/ - -#include "page_cache.h" -#include "least_recently_dirtied.h" -#include "transaction.h" -#include "share.h" -#include "log.h" - -/* could also be called LSN_ERROR */ -#define LSN_IMPOSSIBLE ((LSN)0) -#define LSN_MAX ((LSN)ULONGLONG_MAX) - -/* - this transaction is used for any system work (purge, checkpoint writing - etc), that is, background threads. It will not be declared/initialized here - in the final version. -*/ -st_transaction system_trans= {0 /* long trans id */, 0 /* short trans id */,0,...}; - -/* those three are protected by the log's mutex */ -/* - The maximum rec_lsn in the LRD when last checkpoint was run, serves for the - MEDIUM checkpoint. -*/ -LSN max_rec_lsn_at_last_checkpoint= 0; -CHECKPOINT_LEVEL next_asynchronous_checkpoint_to_do= NONE; -CHECKPOINT_LEVEL synchronous_checkpoint_in_progress= NONE; - -/* - Used by MySQL client threads requesting a checkpoint (like "ALTER MARIA - ENGINE DO CHECKPOINT"), and probably by maria_panic(), and at the end of the - UNDO recovery phase. -*/ -my_bool execute_synchronous_checkpoint(CHECKPOINT_LEVEL level) -{ - DBUG_ENTER("execute_synchronous_checkpoint"); - DBUG_ASSERT(level > NONE); - - lock(log_mutex); - while ((synchronous_checkpoint_in_progress != NONE) || - (next_asynchronous_checkpoint_to_do != NONE)) - wait_on_checkpoint_done_cond(); - - synchronous_checkpoint_in_progress= level; - execute_checkpoint(level); - safemutex_assert_owner(log_mutex); - synchronous_checkpoint_in_progress= NONE; - unlock(log_mutex); - broadcast(checkpoint_done_cond); -} - -/* Picks a checkpoint request, if there is one, and executes it */ -my_bool execute_asynchronous_checkpoint_if_any() -{ - CHECKPOINT_LEVEL level; - DBUG_ENTER("execute_asynchronous_checkpoint"); - - lock(log_mutex); - if (likely(next_asynchronous_checkpoint_to_do == NONE)) - { - unlock(log_mutex); - DBUG_RETURN(FALSE); - } - - while (synchronous_checkpoint_in_progress) - wait_on_checkpoint_done_cond(); - -do_checkpoint: - level= next_asynchronous_checkpoint_to_do; - DBUG_ASSERT(level > NONE); - execute_checkpoint(level); - safemutex_assert_owner(log_mutex); - if (next_asynchronous_checkpoint_to_do > level) - goto do_checkpoint; /* one more request was posted */ - else - { - DBUG_ASSERT(next_asynchronous_checkpoint_to_do == level); - next_asynchronous_checkpoint_to_do= NONE; /* all work done */ - } - unlock(log_mutex); - broadcast(checkpoint_done_cond); -} - - -/* - Does the actual checkpointing. Called by - execute_synchronous_checkpoint() and - execute_asynchronous_checkpoint_if_any(). -*/ -my_bool execute_checkpoint(CHECKPOINT_LEVEL level) -{ - LSN candidate_max_rec_lsn_at_last_checkpoint; - /* to avoid { lock + no-op + unlock } in the common (==indirect) case */ - my_bool need_log_mutex; - - DBUG_ENTER("execute_checkpoint"); - - safemutex_assert_owner(log_mutex); - copy_of_max_rec_lsn_at_last_checkpoint= max_rec_lsn_at_last_checkpoint; - - if (unlikely(need_log_mutex= (level > INDIRECT))) - { - /* much I/O work to do, release log mutex */ - unlock(log_mutex); - - switch (level) - { - case FULL: - /* flush all pages up to the current end of the LRD */ - flush_all_LRD_to_lsn(LSN_MAX); - /* this will go full speed (normal scheduling, no sleep) */ - break; - case MEDIUM: - /* - flush all pages which were already dirty at last checkpoint: - ensures that recovery will never start from before the next-to-last - checkpoint (two-checkpoint rule). - It is max, not min as the WL says (TODO update WL). - */ - flush_all_LRD_to_lsn(copy_of_max_rec_lsn_at_last_checkpoint); - /* this will go full speed (normal scheduling, no sleep) */ - break; - } - } - - candidate_max_rec_lsn_at_last_checkpoint= checkpoint_indirect(need_log_mutex); - - lock(log_mutex); - /* - this portion cannot be done as a hook in write_log_record() for the - LOGREC_CHECKPOINT type because: - - at that moment we still have not written to the control file so cannot - mark the request as done; this could be solved by writing to the control - file in the hook but that would be an I/O under the log's mutex, bad. - - it would not be nice organisation of code (I tried it :). - */ - if (candidate_max_rec_lsn_at_last_checkpoint != LSN_IMPOSSIBLE) - { - /* checkpoint succeeded */ - maximum_rec_lsn_last_checkpoint= candidate_max_rec_lsn_at_last_checkpoint; - written_since_last_checkpoint= (my_off_t)0; - DBUG_RETURN(FALSE); - } - /* - keep mutex locked because callers will want to clear mutex-protected - status variables - */ - DBUG_RETURN(TRUE); -} - - -LSN checkpoint_indirect(my_bool need_log_mutex) -{ - DBUG_ENTER("checkpoint_indirect"); - - int error= 0; - /* checkpoint record data: */ - LSN checkpoint_start_lsn; - LEX_STRING string1={0,0}, string2={0,0}, string3={0,0}; - LEX_STRING *string_array[4]; - char *ptr; - LSN checkpoint_lsn; - LSN candidate_max_rec_lsn_at_last_checkpoint= 0; - list_element *el; /* to scan lists */ - - - DBUG_ASSERT(sizeof(byte *) <= 8); - DBUG_ASSERT(sizeof(LSN) <= 8); - - if (need_log_mutex) - lock(log_mutex); /* maybe this will clash with log_read_end_lsn() */ - checkpoint_start_lsn= log_read_end_lsn(); - unlock(log_mutex); - - DBUG_PRINT("info",("checkpoint_start_lsn %lu", checkpoint_start_lsn)); - - lock(global_LRD_mutex); - string1.length= 8+8+(8+8)*LRD->count; - if (NULL == (string1.str= my_malloc(string1.length))) - goto err; - ptr= string1.str; - int8store(ptr, checkpoint_start_lsn); - ptr+= 8; - int8store(ptr, LRD->count); - ptr+= 8; - if (LRD->count) - { - candidate_max_rec_lsn_at_last_checkpoint= LRD->last->rec_lsn; - for (el= LRD->first; el; el= el->next) - { - int8store(ptr, el->page_id); - ptr+= 8; - int8store(ptr, el->rec_lsn); - ptr+= 8; - } - } - unlock(global_LRD_mutex); - - /* - If trx are in more than one list (e.g. three: - running transactions, committed transactions, purge queue), we can either - take mutexes of all three together or do crabbing. - But if an element can move from list 1 to list 3 without passing through - list 2, crabbing is dangerous. - Hopefully it's ok to take 3 mutexes together... - Otherwise I'll have to make sure I miss no important trx and I handle dups. - */ - lock(global_transactions_list_mutex); /* or 3 mutexes if there are 3 */ - string2.length= 8+(8+8)*trx_list->count; - if (NULL == (string2.str= my_malloc(string2.length))) - goto err; - ptr= string2.str; - int8store(ptr, trx_list->count); - ptr+= 8; - for (el= trx_list->first; el; el= el->next) - { - /* possibly latch el.rwlock */ - *ptr= el->state; - ptr++; - int7store(ptr, el->long_trans_id); - ptr+= 7; - int2store(ptr, el->short_trans_id); - ptr+= 2; - int8store(ptr, el->undo_lsn); - ptr+= 8; - int8store(ptr, el->undo_purge_lsn); - ptr+= 8; - /* - if no latch, use double variable of type ULONGLONG_CONSISTENT in - st_transaction, or even no need if Intel >=486 - */ - int8store(ptr, el->first_undo_lsn); - ptr+= 8; - /* possibly unlatch el.rwlock */ - } - unlock(global_transactions_list_mutex); - - lock(global_share_list_mutex); - string3.length= 8+(8+8)*share_list->count; - if (NULL == (string3.str= my_malloc(string3.length))) - goto err; - ptr= string3.str; - /* possibly latch each MARIA_SHARE */ - make_copy_of_global_share_list_to_array; - unlock(global_share_list_mutex); - - /* work on copy */ - int8store(ptr, elements_in_array); - ptr+= 8; - for (scan_array) - { - int8store(ptr, array[...].file_id); - ptr+= 8; - memcpy(ptr, array[...].file_name, ...); - ptr+= ...; - /* - these two are long ops (involving disk I/O) that's why we copied the - list: - */ - flush_bitmap_pages(el); - /* - fsyncs the fd, that's the loooong operation (e.g. max 150 fsync per - second, so if you have touched 1000 files it's 7 seconds). - */ - force_file(el); - } - - /* now write the record */ - string_array[0]= string1; - string_array[1]= string2; - string_array[2]= string3; - string_array[3]= NULL; - - checkpoint_lsn= log_write_record(LOGREC_CHECKPOINT, - &system_trans, string_array); - - if (LSN_IMPOSSIBLE == checkpoint_lsn) - goto err; - - if (0 != control_file_write_and_force(checkpoint_lsn, NULL)) - goto err; - - goto end; - -err: - print_error_to_error_log(the_error_message); - candidate_max_rec_lsn_at_last_checkpoint= LSN_IMPOSSIBLE; - -end: - my_free(buffer1.str, MYF(MY_ALLOW_ZERO_PTR)); - my_free(buffer2.str, MYF(MY_ALLOW_ZERO_PTR)); - my_free(buffer3.str, MYF(MY_ALLOW_ZERO_PTR)); - - DBUG_RETURN(candidate_max_rec_lsn_at_last_checkpoint); -} - - - -/* - Here's what should be put in log_write_record() in the log handler: -*/ -log_write_record(...) -{ - ...; - lock(log_mutex); - ...; - write_to_log(length); - written_since_last_checkpoint+= length; - if (written_since_last_checkpoint > - MAX_LOG_BYTES_WRITTEN_BETWEEN_CHECKPOINTS) - { - /* - ask one system thread (the "LRD background flusher and checkpointer - thread" WL#3261) to do a checkpoint - */ - request_asynchronous_checkpoint(INDIRECT); - } - ...; - unlock(log_mutex); - ...; -} - -/* - Requests a checkpoint from the background thread, *asynchronously* - (requestor does not wait for completion, and does not even later check the - result). - In real life it will be called by log_write_record(). -*/ -void request_asynchronous_checkpoint(CHECKPOINT_LEVEL level); -{ - safemutex_assert_owner(log_mutex); - - DBUG_ASSERT(level > NONE); - if (checkpoint_request < level) - { - /* no equal or stronger running or to run, we post request */ - /* - note that thousands of requests for checkpoints are going to come all - at the same time (when the log bound - MAX_LOG_BYTES_WRITTEN_BETWEEN_CHECKPOINTS is passed), so it may not be a - good idea for each of them to broadcast a cond to wake up the background - checkpoint thread. We just don't broacast a cond, the checkpoint thread - (see least_recently_dirtied.c) will notice our request in max a few - seconds. - */ - checkpoint_request= level; /* post request */ - } - - /* - If there was an error, only an error - message to the error log will say it; normal, for a checkpoint triggered - by a log write, we probably don't want the client's log write to throw an - error, as the log write succeeded and a checkpoint failure is not - critical: the failure in this case is more for the DBA to know than for - the end user. - */ -} diff --git a/storage/maria/checkpoint.h b/storage/maria/checkpoint.h deleted file mode 100644 index a9de18c695f..00000000000 --- a/storage/maria/checkpoint.h +++ /dev/null @@ -1,19 +0,0 @@ -/* - WL#3071 Maria checkpoint - First version written by Guilhem Bichot on 2006-04-27. - Does not compile yet. -*/ - -/* This is the interface of this module. */ - -typedef enum enum_checkpoint_level { - NONE=-1, - INDIRECT, /* just write dirty_pages, transactions table and sync files */ - MEDIUM, /* also flush all dirty pages which were already dirty at prev checkpoint*/ - FULL /* also flush all dirty pages */ -} CHECKPOINT_LEVEL; - -void request_asynchronous_checkpoint(CHECKPOINT_LEVEL level); -my_bool execute_synchronous_checkpoint(CHECKPOINT_LEVEL level); -my_bool execute_asynchronous_checkpoint_if_any(); -/* that's all that's needed in the interface */ diff --git a/storage/maria/control_file.c b/storage/maria/control_file.c deleted file mode 100644 index 70eb62c645b..00000000000 --- a/storage/maria/control_file.c +++ /dev/null @@ -1,211 +0,0 @@ -/* - WL#3234 Maria control file - First version written by Guilhem Bichot on 2006-04-27. - Does not compile yet. -*/ - -#include "maria_def.h" - - -/* Here is the implementation of this module */ - -/* - a control file contains 3 objects: magic string, LSN of last checkpoint, - number of last log. -*/ - -/* total size should be < sector size for atomic write operation */ -#define CONTROL_FILE_MAGIC_STRING "MACF" -#define CONTROL_FILE_MAGIC_STRING_OFFSET 0 -#define CONTROL_FILE_MAGIC_STRING_SIZE sizeof(CONTROL_FILE_MAGIC_STRING) -#define CONTROL_FILE_LSN_OFFSET (CONTROL_FILE_MAGIC_STRING_OFFSET + CONTROL_FILE_MAGIC_STRING_SIZE) -#define CONTROL_FILE_LSN_SIZE (4+4) -#define CONTROL_FILE_FILENO_OFFSET (CONTROL_FILE_LSN_OFFSET + CONTROL_FILE_LSN_SIZE) -#define CONTROL_FILE_FILENO_SIZE 4 -#define CONTROL_FILE_MAX_SIZE (CONTROL_FILE_FILENO_OFFSET + CONTROL_FILE_FILENO_SIZE) - - -LSN last_checkpoint_lsn_at_startup; -uint32 last_logno_at_startup; - - -/* - Control file is less then 512 bytes (a disk sector), - to be as atomic as possible -*/ -static int control_file_fd; - -static void lsn8store(char *buffer, LSN *lsn) -{ - int4store(buffer, lsn->file_no); - int4store(buffer + CONTROL_FILE_FILENO_SIZE, lsn->rec_offset); -} - -static LSN lsn8korr(char *buffer) -{ - LSN tmp; - tmp.file_no= uint4korr(buffer); - tmp.rec_offset= uint4korr(buffer + CONTROL_FILE_FILENO_SIZE); - return tmp; -} - -/* - Initialize control file subsystem - - SYNOPSIS - control_file_create_or_open() - - Looks for the control file. If absent, it's a fresh start, create file. - If present, read it to find out last checkpoint's LSN and last log. - Called at engine's start. - - RETURN - 0 - OK - 1 - Error -*/ -int control_file_create_or_open() -{ - char buffer[CONTROL_FILE_MAX_SIZE]; - char name[FN_REFLEN]; - MY_STAT stat_buff; - - /* - If you change sizes in the #defines, you at least have to change the - "*store" and "*korr" calls in this file, and can even create backward - compatibility problems. Beware! - */ - DBUG_ASSERT(CONTROL_FILE_LSN_SIZE == (4+4)); - DBUG_ASSERT(CONTROL_FILE_FILENO_SIZE == 4); - - /* name is concatenation of Maria's home dir and "control" */ - if (fn_format(name, "control", maria_data_root, "", MYF(MY_WME)) == NullS) - return 1; - - if ((control_file_fd= my_open(name, - O_CREAT | O_BINARY | /*O_DIRECT |*/ O_RDWR, - MYF(MY_WME))) < 0) - return 1; - - /* - TODO: from "man fsync" on Linux: - "fsync does not necessarily ensure that the entry in the direc- tory - containing the file has also reached disk. For that an explicit - fsync on the file descriptor of the directory is also needed." - So if we just created the file we should sync the directory. - Maybe there should be a flag of my_create() to do this. - */ - - if (my_stat(name, &stat_buff, MYF(MY_WME)) == NULL) - return 1; - - if (stat_buff.st_size < CONTROL_FILE_MAX_SIZE) - { - /* - File shorter than expected (either we just created it, or a previous run - crashed between creation and first write); do first write. - */ - char buffer[CONTROL_FILE_MAX_SIZE]; - /* - To be safer we should make sure that there are no logs or data/index - files around (indeed it could be that the control file alone was deleted - or not restored, and we should not go on with life at this point). - - TODO: For now we trust (this is alpha version), but for beta if would - be great to verify. - - We could have a tool which can rebuild the control file, by reading the - directory of logs, finding the newest log, reading it to find last - checkpoint... Slow but can save your db. - */ - last_checkpoint_lsn_at_startup.file_no= CONTROL_FILE_IMPOSSIBLE_FILENO; - last_checkpoint_lsn_at_startup.rec_offset= 0; - last_logno_at_startup= CONTROL_FILE_IMPOSSIBLE_FILENO; - - /* init the file with these "undefined" values */ - return control_file_write_and_force(last_checkpoint_lsn_at_startup, - last_logno_at_startup, - CONTROL_FILE_WRITE_ALL); - } - /* Already existing file, read it */ - if (my_read(control_file_fd, buffer, CONTROL_FILE_MAX_SIZE, - MYF(MY_FNABP | MY_WME))) - return 1; - if (memcmp(buffer + CONTROL_FILE_MAGIC_STRING_OFFSET, - CONTROL_FILE_MAGIC_STRING, CONTROL_FILE_MAGIC_STRING_SIZE)) - return 1; - last_checkpoint_lsn_at_startup= lsn8korr(buffer + CONTROL_FILE_LSN_OFFSET); - last_logno_at_startup= uint4korr(buffer + CONTROL_FILE_FILENO_OFFSET); - return 0; -} - - -#define CONTROL_FILE_WRITE_ALL 0 /* write all 3 objects */ -#define CONTROL_FILE_WRITE_ONLY_LSN 1 -#define CONTROL_FILE_WRITE_ONLY_LOGNO 2 -/* - Write information durably to the control file. - - SYNOPSIS - control_file_write_and_force() - checkpoint_lsn LSN of last checkpoint - log_no last log file number - objs_to_write what we should write - - Called when we have created a new log (after syncing this log's creation) - and when we have written a checkpoint (after syncing this log record). - - NOTE - We always want to do one single my_pwrite() here to be as atomic as - possible. - - RETURN - 0 - OK - 1 - Error -*/ - -int control_file_write_and_force(LSN *checkpoint_lsn, uint32 log_no, - uint objs_to_write) -{ - char buffer[CONTROL_FILE_MAX_SIZE]; - uint start, size; - memcpy(buffer + CONTROL_FILE_MAGIC_STRING_OFFSET, - CONTROL_FILE_MAGIC_STRING, CONTROL_FILE_MAGIC_STRING_SIZE); - /* write checkpoint LSN */ - if (checkpoint_lsn) - lsn8store(buffer + CONTROL_FILE_LSN_OFFSET, checkpoint_lsn); - /* write logno */ - int4store(buffer + CONTROL_FILE_FILENO_OFFSET, log_no); - if (objs_to_write == CONTROL_FILE_WRITE_ALL) - { - start= CONTROL_FILE_MAGIC_STRING_OFFSET; - size= CONTROL_FILE_MAX_SIZE; - } - else if (objs_to_write == CONTROL_FILE_WRITE_ONLY_LSN) - { - start= CONTROL_FILE_LSN_OFFSET; - size= CONTROL_FILE_LSN_SIZE; - } - else if (objs_to_write == CONTROL_FILE_WRITE_ONLY_LOGNO) - { - start= CONTROL_FILE_FILENO_OFFSET; - size= CONTROL_FILE_FILENO_SIZE; - } - else /* incorrect value of objs_to_write */ - DBUG_ASSERT(0); - return (my_pwrite(control_file_fd, buffer + start, size, - start, MYF(MY_FNABP | MY_WME)) || - my_sync(control_file_fd, MYF(MY_WME))); -} - - -/* - Free resources taken by control file subsystem - - SYNOPSIS - control_file_end() -*/ - -void control_file_end() -{ - my_close(control_file_fd, MYF(MY_WME)); -} diff --git a/storage/maria/control_file.h b/storage/maria/control_file.h deleted file mode 100644 index 66a1f225cd8..00000000000 --- a/storage/maria/control_file.h +++ /dev/null @@ -1,46 +0,0 @@ -/* - WL#3234 Maria control file - First version written by Guilhem Bichot on 2006-04-27. - Does not compile yet. -*/ - -#ifndef _control_file_h -#define _control_file_h - -/* indicate absence of the log file number */ -#define CONTROL_FILE_IMPOSSIBLE_LOGNO 0xFFFFFFFF - -/* Here is the interface of this module */ - -/* - LSN of the last checkoint - (if last_checkpoint_lsn_at_startup.file_no == CONTROL_FILE_IMPOSSIBLE_LOGNO - then there was never a checkpoint) -*/ -extern LSN last_checkpoint_lsn_at_startup; -/* - Last log number at startup time (if last_logno_at_startup == - CONTROL_FILE_IMPOSSIBLE_LOGNO then there is no log file yet) -*/ -extern uint32 last_logno_at_startup; - -/* - Looks for the control file. If absent, it's a fresh start, create file. - If present, read it to find out last checkpoint's LSN and last log. - Called at engine's start. -*/ -int control_file_create_or_open(); - -/* - Write information durably to the control file. - Called when we have created a new log (after syncing this log's creation) - and when we have written a checkpoint (after syncing this log record). -*/ -int control_file_write_and_force(LSN *checkpoint_lsn, uint32 log_no, - uint objs_to_write); - - -/* Free resources taken by control file subsystem */ -void control_file_end(); - -#endif diff --git a/storage/maria/least_recently_dirtied.c b/storage/maria/least_recently_dirtied.c deleted file mode 100644 index c6285fe47cd..00000000000 --- a/storage/maria/least_recently_dirtied.c +++ /dev/null @@ -1,209 +0,0 @@ -/* - WL#3261 Maria - background flushing of the least-recently-dirtied pages - First version written by Guilhem Bichot on 2006-04-27. - Does not compile yet. -*/ - -/* - To be part of the page cache. - The pseudocode below is dependent on the page cache - which is being designed WL#3134. It is not clear if I need to do page - copies, as the page cache already keeps page copies. - So, this code will move to the page cache and take inspiration from its - methods. Below is just to give the idea of what could be done. - And I should compare my imaginations to WL#3134. -*/ - -/* Here is the implementation of this module */ - -#include "page_cache.h" -#include "least_recently_dirtied.h" - -/* - MikaelR suggested removing this global_LRD_mutex (I have a paper note of - comments), however at least for the first version we'll start with this - mutex (which will be a LOCK-based atomic_rwlock). -*/ -pthread_mutex_t global_LRD_mutex; - -/* - When we flush a page, we should pin page. - This "pin" is to protect against that: - I make copy, - you modify in memory and flush to disk and remove from LRD and from cache, - I write copy to disk, - checkpoint happens. - result: old page is on disk, page is absent from LRD, your REDO will be - wrongly ignored. - - Pin: there can be multiple pins, flushing imposes that there are zero pins. - For example, pin could be a uint counter protected by the page's latch. - - Maybe it's ok if when there is a page replacement, the replacer does not - remove page from the LRD (it would save global mutex); for that, background - flusher should be prepared to see pages in the LRD which are not in the page - cache (then just ignore them). However checkpoint will contain superfluous - entries and so do more work. -*/ - -#define PAGE_SIZE (16*1024) /* just as an example */ -/* - Optimization: - LRD flusher should not flush pages one by one: to be fast, it flushes a - group of pages in sequential disk order if possible; a group of pages is just - FLUSH_GROUP_SIZE pages. - Key cache has groupping already somehow Monty said (investigate that). -*/ -#define FLUSH_GROUP_SIZE 512 /* 8 MB */ -/* - We don't want to probe for checkpoint requests all the time (it takes - the log mutex). - If FLUSH_GROUP_SIZE is 8MB, assuming a local disk which can write 30MB/s - (1.8GB/min), probing every 16th call to flush_one_group_from_LRD() is every - 16*8=128MB which is every 128/30=4.2second. - Using a power of 2 gives a fast modulo operation. -*/ -#define CHECKPOINT_PROBING_PERIOD_LOG2 4 - -/* - This thread does background flush of pieces of the LRD, and all checkpoints. - Just launch it when engine starts. - MikaelR questioned why the same thread does two different jobs, the risk - could be that while a checkpoint happens no LRD flushing happens. -*/ -pthread_handler_decl background_flush_and_checkpoint_thread() -{ - char *flush_group_buffer= my_malloc(PAGE_SIZE*FLUSH_GROUP_SIZE); - uint flush_calls= 0; - while (this_thread_not_killed) - { - if ((flush_calls++) & ((2<data, PAGE_SIZE); - pin_page; - page_cache_unlatch(page_id, KEEP_PINNED); /* but keep pinned */ - } - for (scan_the_array) - { - /* - As an optimization, we try to identify contiguous-in-the-file segments (to - issue one big write()). - In non-optimized version, contiguous segment is always only one page. - */ - if ((next_page.page_id - this_page.page_id) == 1) - { - /* - this page and next page are in same file and are contiguous in the - file: add page to contiguous segment... - */ - continue; /* defer write() to next pages */ - } - /* contiguous segment ends */ - my_pwrite(file, contiguous_segment_start_offset, contiguous_segment_size); - - /* - note that if we had doublewrite, doublewrite buffer may prevent us from - doing this write() grouping (if doublewrite space is shorter). - */ - } - /* - Now remove pages from LRD. As we have pinned them, all pages that we - managed to pin are still in the LRD, in the same order, we can just cut - the LRD at the last element of "array". This is more efficient that - removing element by element (which would take LRD mutex many times) in the - loop above. - */ - lock(global_LRD_mutex); - /* cut LRD by bending LRD->first, free cut portion... */ - unlock(global_LRD_mutex); - for (scan_array) - { - /* - if the page has a property "modified since last flush" (i.e. which is - redundant with the presence of the page in the LRD, this property can - just be a pointer to the LRD element) we should reset it - (note that then the property would live slightly longer than - the presence in LRD). - */ - page_cache_unpin(page_id); - /* - order between unpin and removal from LRD is not clear, depends on what - pin actually is. - */ - } - free(array); - /* - MikaelR noted that he observed that Linux's file cache may never fsync to - disk until this cache is full, at which point it decides to empty the - cache, making the machine very slow. A solution was to fsync after writing - 2 MB. - */ -} - -/* - Flushes all page from LRD up to approximately rec_lsn>=max_lsn. - This is approximate because we flush groups, and because the LRD list may - not be exactly sorted by rec_lsn (because for a big row, all pages of the - row are inserted into the LRD with rec_lsn being the LSN of the REDO for the - first page, so if there are concurrent insertions, the last page of the big - row may have a smaller rec_lsn than the previous pages inserted by - concurrent inserters). -*/ -int flush_all_LRD_to_lsn(LSN max_lsn) -{ - lock(global_LRD_mutex); - if (max_lsn == MAX_LSN) /* don't want to flush forever, so make it fixed: */ - max_lsn= LRD->first->prev->rec_lsn; - while (LRD->first->rec_lsn < max_lsn) - { - if (flush_one_group_from_LRD()) /* will unlock LRD mutex */ - return 1; - /* - The scheduler may preempt us here as we released the mutex; this is good. - */ - lock(global_LRD_mutex); - } - unlock(global_LRD_mutex); - return 0; -} diff --git a/storage/maria/least_recently_dirtied.h b/storage/maria/least_recently_dirtied.h deleted file mode 100644 index 6a30db4b5f0..00000000000 --- a/storage/maria/least_recently_dirtied.h +++ /dev/null @@ -1,10 +0,0 @@ -/* - WL#3261 Maria - background flushing of the least-recently-dirtied pages - First version written by Guilhem Bichot on 2006-04-27. - Does not compile yet. -*/ - -/* This is the interface of this module. */ - -/* flushes all page from LRD up to approximately rec_lsn>=max_lsn */ -int flush_all_LRD_to_lsn(LSN max_lsn); diff --git a/storage/maria/ma_checkpoint.c b/storage/maria/ma_checkpoint.c new file mode 100644 index 00000000000..22e7b93d2f4 --- /dev/null +++ b/storage/maria/ma_checkpoint.c @@ -0,0 +1,375 @@ +/* + WL#3071 Maria checkpoint + First version written by Guilhem Bichot on 2006-04-27. + Does not compile yet. +*/ + +/* Here is the implementation of this module */ + +/* + Summary: + - there are asynchronous checkpoints (a writer to the log notices that it's + been a long time since we last checkpoint-ed, so posts a request for a + background thread to do a checkpoint; does not care about the success of the + checkpoint). Then the checkpoint is done by the checkpoint thread, at an + unspecified moment ("later") (==soon, of course). + - there are synchronous checkpoints: a thread requests a checkpoint to + happen now and wants to know when it finishes and if it succeeded; then the + checkpoint is done by that same thread. +*/ + +#include "page_cache.h" +#include "least_recently_dirtied.h" +#include "transaction.h" +#include "share.h" +#include "log.h" + +/* could also be called LSN_ERROR */ +#define LSN_IMPOSSIBLE ((LSN)0) +#define LSN_MAX ((LSN)ULONGLONG_MAX) + +/* + this transaction is used for any system work (purge, checkpoint writing + etc), that is, background threads. It will not be declared/initialized here + in the final version. +*/ +st_transaction system_trans= {0 /* long trans id */, 0 /* short trans id */,0,...}; + +/* those three are protected by the log's mutex */ +/* + The maximum rec_lsn in the LRD when last checkpoint was run, serves for the + MEDIUM checkpoint. +*/ +LSN max_rec_lsn_at_last_checkpoint= 0; +CHECKPOINT_LEVEL next_asynchronous_checkpoint_to_do= NONE; +CHECKPOINT_LEVEL synchronous_checkpoint_in_progress= NONE; + +/* + Used by MySQL client threads requesting a checkpoint (like "ALTER MARIA + ENGINE DO CHECKPOINT"), and probably by maria_panic(), and at the end of the + UNDO recovery phase. +*/ +my_bool execute_synchronous_checkpoint(CHECKPOINT_LEVEL level) +{ + DBUG_ENTER("execute_synchronous_checkpoint"); + DBUG_ASSERT(level > NONE); + + lock(log_mutex); + while ((synchronous_checkpoint_in_progress != NONE) || + (next_asynchronous_checkpoint_to_do != NONE)) + wait_on_checkpoint_done_cond(); + + synchronous_checkpoint_in_progress= level; + execute_checkpoint(level); + safemutex_assert_owner(log_mutex); + synchronous_checkpoint_in_progress= NONE; + unlock(log_mutex); + broadcast(checkpoint_done_cond); +} + +/* Picks a checkpoint request, if there is one, and executes it */ +my_bool execute_asynchronous_checkpoint_if_any() +{ + CHECKPOINT_LEVEL level; + DBUG_ENTER("execute_asynchronous_checkpoint"); + + lock(log_mutex); + if (likely(next_asynchronous_checkpoint_to_do == NONE)) + { + unlock(log_mutex); + DBUG_RETURN(FALSE); + } + + while (synchronous_checkpoint_in_progress) + wait_on_checkpoint_done_cond(); + +do_checkpoint: + level= next_asynchronous_checkpoint_to_do; + DBUG_ASSERT(level > NONE); + execute_checkpoint(level); + safemutex_assert_owner(log_mutex); + if (next_asynchronous_checkpoint_to_do > level) + goto do_checkpoint; /* one more request was posted */ + else + { + DBUG_ASSERT(next_asynchronous_checkpoint_to_do == level); + next_asynchronous_checkpoint_to_do= NONE; /* all work done */ + } + unlock(log_mutex); + broadcast(checkpoint_done_cond); +} + + +/* + Does the actual checkpointing. Called by + execute_synchronous_checkpoint() and + execute_asynchronous_checkpoint_if_any(). +*/ +my_bool execute_checkpoint(CHECKPOINT_LEVEL level) +{ + LSN candidate_max_rec_lsn_at_last_checkpoint; + /* to avoid { lock + no-op + unlock } in the common (==indirect) case */ + my_bool need_log_mutex; + + DBUG_ENTER("execute_checkpoint"); + + safemutex_assert_owner(log_mutex); + copy_of_max_rec_lsn_at_last_checkpoint= max_rec_lsn_at_last_checkpoint; + + if (unlikely(need_log_mutex= (level > INDIRECT))) + { + /* much I/O work to do, release log mutex */ + unlock(log_mutex); + + switch (level) + { + case FULL: + /* flush all pages up to the current end of the LRD */ + flush_all_LRD_to_lsn(LSN_MAX); + /* this will go full speed (normal scheduling, no sleep) */ + break; + case MEDIUM: + /* + flush all pages which were already dirty at last checkpoint: + ensures that recovery will never start from before the next-to-last + checkpoint (two-checkpoint rule). + It is max, not min as the WL says (TODO update WL). + */ + flush_all_LRD_to_lsn(copy_of_max_rec_lsn_at_last_checkpoint); + /* this will go full speed (normal scheduling, no sleep) */ + break; + } + } + + candidate_max_rec_lsn_at_last_checkpoint= checkpoint_indirect(need_log_mutex); + + lock(log_mutex); + /* + this portion cannot be done as a hook in write_log_record() for the + LOGREC_CHECKPOINT type because: + - at that moment we still have not written to the control file so cannot + mark the request as done; this could be solved by writing to the control + file in the hook but that would be an I/O under the log's mutex, bad. + - it would not be nice organisation of code (I tried it :). + */ + if (candidate_max_rec_lsn_at_last_checkpoint != LSN_IMPOSSIBLE) + { + /* checkpoint succeeded */ + maximum_rec_lsn_last_checkpoint= candidate_max_rec_lsn_at_last_checkpoint; + written_since_last_checkpoint= (my_off_t)0; + DBUG_RETURN(FALSE); + } + /* + keep mutex locked because callers will want to clear mutex-protected + status variables + */ + DBUG_RETURN(TRUE); +} + + +LSN checkpoint_indirect(my_bool need_log_mutex) +{ + DBUG_ENTER("checkpoint_indirect"); + + int error= 0; + /* checkpoint record data: */ + LSN checkpoint_start_lsn; + LEX_STRING string1={0,0}, string2={0,0}, string3={0,0}; + LEX_STRING *string_array[4]; + char *ptr; + LSN checkpoint_lsn; + LSN candidate_max_rec_lsn_at_last_checkpoint= 0; + list_element *el; /* to scan lists */ + + + DBUG_ASSERT(sizeof(byte *) <= 8); + DBUG_ASSERT(sizeof(LSN) <= 8); + + if (need_log_mutex) + lock(log_mutex); /* maybe this will clash with log_read_end_lsn() */ + checkpoint_start_lsn= log_read_end_lsn(); + unlock(log_mutex); + + DBUG_PRINT("info",("checkpoint_start_lsn %lu", checkpoint_start_lsn)); + + lock(global_LRD_mutex); + string1.length= 8+8+(8+8)*LRD->count; + if (NULL == (string1.str= my_malloc(string1.length))) + goto err; + ptr= string1.str; + int8store(ptr, checkpoint_start_lsn); + ptr+= 8; + int8store(ptr, LRD->count); + ptr+= 8; + if (LRD->count) + { + candidate_max_rec_lsn_at_last_checkpoint= LRD->last->rec_lsn; + for (el= LRD->first; el; el= el->next) + { + int8store(ptr, el->page_id); + ptr+= 8; + int8store(ptr, el->rec_lsn); + ptr+= 8; + } + } + unlock(global_LRD_mutex); + + /* + If trx are in more than one list (e.g. three: + running transactions, committed transactions, purge queue), we can either + take mutexes of all three together or do crabbing. + But if an element can move from list 1 to list 3 without passing through + list 2, crabbing is dangerous. + Hopefully it's ok to take 3 mutexes together... + Otherwise I'll have to make sure I miss no important trx and I handle dups. + */ + lock(global_transactions_list_mutex); /* or 3 mutexes if there are 3 */ + string2.length= 8+(8+8)*trx_list->count; + if (NULL == (string2.str= my_malloc(string2.length))) + goto err; + ptr= string2.str; + int8store(ptr, trx_list->count); + ptr+= 8; + for (el= trx_list->first; el; el= el->next) + { + /* possibly latch el.rwlock */ + *ptr= el->state; + ptr++; + int7store(ptr, el->long_trans_id); + ptr+= 7; + int2store(ptr, el->short_trans_id); + ptr+= 2; + int8store(ptr, el->undo_lsn); + ptr+= 8; + int8store(ptr, el->undo_purge_lsn); + ptr+= 8; + /* + if no latch, use double variable of type ULONGLONG_CONSISTENT in + st_transaction, or even no need if Intel >=486 + */ + int8store(ptr, el->first_undo_lsn); + ptr+= 8; + /* possibly unlatch el.rwlock */ + } + unlock(global_transactions_list_mutex); + + lock(global_share_list_mutex); + string3.length= 8+(8+8)*share_list->count; + if (NULL == (string3.str= my_malloc(string3.length))) + goto err; + ptr= string3.str; + /* possibly latch each MARIA_SHARE */ + make_copy_of_global_share_list_to_array; + unlock(global_share_list_mutex); + + /* work on copy */ + int8store(ptr, elements_in_array); + ptr+= 8; + for (scan_array) + { + int8store(ptr, array[...].file_id); + ptr+= 8; + memcpy(ptr, array[...].file_name, ...); + ptr+= ...; + /* + these two are long ops (involving disk I/O) that's why we copied the + list: + */ + flush_bitmap_pages(el); + /* + fsyncs the fd, that's the loooong operation (e.g. max 150 fsync per + second, so if you have touched 1000 files it's 7 seconds). + */ + force_file(el); + } + + /* now write the record */ + string_array[0]= string1; + string_array[1]= string2; + string_array[2]= string3; + string_array[3]= NULL; + + checkpoint_lsn= log_write_record(LOGREC_CHECKPOINT, + &system_trans, string_array); + + if (LSN_IMPOSSIBLE == checkpoint_lsn) + goto err; + + if (0 != control_file_write_and_force(checkpoint_lsn, NULL)) + goto err; + + goto end; + +err: + print_error_to_error_log(the_error_message); + candidate_max_rec_lsn_at_last_checkpoint= LSN_IMPOSSIBLE; + +end: + my_free(buffer1.str, MYF(MY_ALLOW_ZERO_PTR)); + my_free(buffer2.str, MYF(MY_ALLOW_ZERO_PTR)); + my_free(buffer3.str, MYF(MY_ALLOW_ZERO_PTR)); + + DBUG_RETURN(candidate_max_rec_lsn_at_last_checkpoint); +} + + + +/* + Here's what should be put in log_write_record() in the log handler: +*/ +log_write_record(...) +{ + ...; + lock(log_mutex); + ...; + write_to_log(length); + written_since_last_checkpoint+= length; + if (written_since_last_checkpoint > + MAX_LOG_BYTES_WRITTEN_BETWEEN_CHECKPOINTS) + { + /* + ask one system thread (the "LRD background flusher and checkpointer + thread" WL#3261) to do a checkpoint + */ + request_asynchronous_checkpoint(INDIRECT); + } + ...; + unlock(log_mutex); + ...; +} + +/* + Requests a checkpoint from the background thread, *asynchronously* + (requestor does not wait for completion, and does not even later check the + result). + In real life it will be called by log_write_record(). +*/ +void request_asynchronous_checkpoint(CHECKPOINT_LEVEL level); +{ + safemutex_assert_owner(log_mutex); + + DBUG_ASSERT(level > NONE); + if (checkpoint_request < level) + { + /* no equal or stronger running or to run, we post request */ + /* + note that thousands of requests for checkpoints are going to come all + at the same time (when the log bound + MAX_LOG_BYTES_WRITTEN_BETWEEN_CHECKPOINTS is passed), so it may not be a + good idea for each of them to broadcast a cond to wake up the background + checkpoint thread. We just don't broacast a cond, the checkpoint thread + (see least_recently_dirtied.c) will notice our request in max a few + seconds. + */ + checkpoint_request= level; /* post request */ + } + + /* + If there was an error, only an error + message to the error log will say it; normal, for a checkpoint triggered + by a log write, we probably don't want the client's log write to throw an + error, as the log write succeeded and a checkpoint failure is not + critical: the failure in this case is more for the DBA to know than for + the end user. + */ +} diff --git a/storage/maria/ma_checkpoint.h b/storage/maria/ma_checkpoint.h new file mode 100644 index 00000000000..a9de18c695f --- /dev/null +++ b/storage/maria/ma_checkpoint.h @@ -0,0 +1,19 @@ +/* + WL#3071 Maria checkpoint + First version written by Guilhem Bichot on 2006-04-27. + Does not compile yet. +*/ + +/* This is the interface of this module. */ + +typedef enum enum_checkpoint_level { + NONE=-1, + INDIRECT, /* just write dirty_pages, transactions table and sync files */ + MEDIUM, /* also flush all dirty pages which were already dirty at prev checkpoint*/ + FULL /* also flush all dirty pages */ +} CHECKPOINT_LEVEL; + +void request_asynchronous_checkpoint(CHECKPOINT_LEVEL level); +my_bool execute_synchronous_checkpoint(CHECKPOINT_LEVEL level); +my_bool execute_asynchronous_checkpoint_if_any(); +/* that's all that's needed in the interface */ diff --git a/storage/maria/ma_control_file.c b/storage/maria/ma_control_file.c new file mode 100644 index 00000000000..d36e1c04c0c --- /dev/null +++ b/storage/maria/ma_control_file.c @@ -0,0 +1,234 @@ +/* + WL#3234 Maria control file + First version written by Guilhem Bichot on 2006-04-27. + Does not compile yet. +*/ + +#include "maria_def.h" +#include "ma_control_file.h" + +/* Here is the implementation of this module */ + +/* + a control file contains 3 objects: magic string, LSN of last checkpoint, + number of last log. +*/ + +/* total size should be < sector size for atomic write operation */ +#define CONTROL_FILE_MAGIC_STRING "MACF" +#define CONTROL_FILE_MAGIC_STRING_OFFSET 0 +#define CONTROL_FILE_MAGIC_STRING_SIZE 4 +#define CONTROL_FILE_LSN_OFFSET (CONTROL_FILE_MAGIC_STRING_OFFSET + CONTROL_FILE_MAGIC_STRING_SIZE) +#define CONTROL_FILE_LSN_SIZE (4+4) +#define CONTROL_FILE_FILENO_OFFSET (CONTROL_FILE_LSN_OFFSET + CONTROL_FILE_LSN_SIZE) +#define CONTROL_FILE_FILENO_SIZE 4 +#define CONTROL_FILE_MAX_SIZE (CONTROL_FILE_FILENO_OFFSET + CONTROL_FILE_FILENO_SIZE) + +/* + This module owns these two vars. + uint32 is always atomically updated, but LSN is 8 bytes, we will need + provisions to ensure that it's updated atomically in + ma_control_file_write_and_force(). Probably the log mutex could be + used. TODO. +*/ +LSN last_checkpoint_lsn; +uint32 last_logno; + + +/* + Control file is less then 512 bytes (a disk sector), + to be as atomic as possible +*/ +static int control_file_fd; + +static void lsn8store(char *buffer, const LSN *lsn) +{ + int4store(buffer, lsn->file_no); + int4store(buffer + CONTROL_FILE_FILENO_SIZE, lsn->rec_offset); +} + +static LSN lsn8korr(char *buffer) +{ + LSN tmp; + tmp.file_no= uint4korr(buffer); + tmp.rec_offset= uint4korr(buffer + CONTROL_FILE_FILENO_SIZE); + return tmp; +} + +/* + Initialize control file subsystem + + SYNOPSIS + ma_control_file_create_or_open() + + Looks for the control file. If absent, it's a fresh start, creates file. + If present, reads it to find out last checkpoint's LSN and last log, updates + the last_checkpoint_lsn and last_logno global variables. + Called at engine's start. + + RETURN + 0 - OK + 1 - Error +*/ +int ma_control_file_create_or_open() +{ + char buffer[CONTROL_FILE_MAX_SIZE]; + char name[FN_REFLEN]; + MY_STAT stat_buff; + DBUG_ENTER("ma_control_file_create_or_open"); + + /* + If you change sizes in the #defines, you at least have to change the + "*store" and "*korr" calls in this file, and can even create backward + compatibility problems. Beware! + */ + DBUG_ASSERT(CONTROL_FILE_LSN_SIZE == (4+4)); + DBUG_ASSERT(CONTROL_FILE_FILENO_SIZE == 4); + + /* name is concatenation of Maria's home dir and "control" */ + if (fn_format(name, "control", maria_data_root, "", MYF(MY_WME)) == NullS) + DBUG_RETURN(1); + + if ((control_file_fd= my_open(name, + O_CREAT | O_BINARY | /*O_DIRECT |*/ O_RDWR, + MYF(MY_WME))) < 0) + DBUG_RETURN(1); + + /* + TODO: from "man fsync" on Linux: + "fsync does not necessarily ensure that the entry in the direc- tory + containing the file has also reached disk. For that an explicit + fsync on the file descriptor of the directory is also needed." + So if we just created the file we should sync the directory. + Maybe there should be a flag of my_create() to do this. + */ + + if (my_stat(name, &stat_buff, MYF(MY_WME)) == NULL) + DBUG_RETURN(1); + + if ((uint)stat_buff.st_size < CONTROL_FILE_MAX_SIZE) + { + /* + File shorter than expected (either we just created it, or a previous run + crashed between creation and first write); do first write. + + To be safer we should make sure that there are no logs or data/index + files around (indeed it could be that the control file alone was deleted + or not restored, and we should not go on with life at this point). + + TODO: For now we trust (this is alpha version), but for beta if would + be great to verify. + + We could have a tool which can rebuild the control file, by reading the + directory of logs, finding the newest log, reading it to find last + checkpoint... Slow but can save your db. + */ + LSN imposs_lsn= CONTROL_FILE_IMPOSSIBLE_LSN; + uint32 imposs_logno= CONTROL_FILE_IMPOSSIBLE_FILENO; + + /* init the file with these "undefined" values */ + DBUG_RETURN(ma_control_file_write_and_force(&imposs_lsn, imposs_logno, + CONTROL_FILE_WRITE_ALL)); + } + /* Already existing file, read it */ + if (my_read(control_file_fd, buffer, CONTROL_FILE_MAX_SIZE, + MYF(MY_FNABP | MY_WME))) + DBUG_RETURN(1); + if (memcmp(buffer + CONTROL_FILE_MAGIC_STRING_OFFSET, + CONTROL_FILE_MAGIC_STRING, CONTROL_FILE_MAGIC_STRING_SIZE)) + { + /* + TODO: what is the good way to report the error? Knowing that this + happens at startup, probably stderr. + */ + DBUG_PRINT("error", ("bad magic string")); + DBUG_RETURN(1); + } + last_checkpoint_lsn= lsn8korr(buffer + CONTROL_FILE_LSN_OFFSET); + last_logno= uint4korr(buffer + CONTROL_FILE_FILENO_OFFSET); + DBUG_RETURN(0); +} + + +/* + Write information durably to the control file; stores this information into + the last_checkpoint_lsn and last_logno global variables. + + SYNOPSIS + ma_control_file_write_and_force() + checkpoint_lsn LSN of last checkpoint + logno last log file number + objs_to_write what we should write + + Called when we have created a new log (after syncing this log's creation) + and when we have written a checkpoint (after syncing this log record). + + NOTE + We always want to do one single my_pwrite() here to be as atomic as + possible. + + RETURN + 0 - OK + 1 - Error +*/ + +int ma_control_file_write_and_force(const LSN *checkpoint_lsn, uint32 logno, + uint objs_to_write) +{ + char buffer[CONTROL_FILE_MAX_SIZE]; + uint start, size; + DBUG_ENTER("ma_control_file_write_and_force"); + + memcpy(buffer + CONTROL_FILE_MAGIC_STRING_OFFSET, + CONTROL_FILE_MAGIC_STRING, CONTROL_FILE_MAGIC_STRING_SIZE); + /* write checkpoint LSN */ + if (checkpoint_lsn) + lsn8store(buffer + CONTROL_FILE_LSN_OFFSET, checkpoint_lsn); + /* write logno */ + int4store(buffer + CONTROL_FILE_FILENO_OFFSET, logno); + if (objs_to_write == CONTROL_FILE_WRITE_ALL) + { + start= CONTROL_FILE_MAGIC_STRING_OFFSET; + size= CONTROL_FILE_MAX_SIZE; + last_checkpoint_lsn= *checkpoint_lsn; + last_logno= logno; + } + else if (objs_to_write == CONTROL_FILE_WRITE_ONLY_LSN) + { + start= CONTROL_FILE_LSN_OFFSET; + size= CONTROL_FILE_LSN_SIZE; + last_checkpoint_lsn= *checkpoint_lsn; + } + else if (objs_to_write == CONTROL_FILE_WRITE_ONLY_LOGNO) + { + start= CONTROL_FILE_FILENO_OFFSET; + size= CONTROL_FILE_FILENO_SIZE; + last_logno= logno; + } + else /* incorrect value of objs_to_write */ + DBUG_ASSERT(0); + DBUG_RETURN(my_pwrite(control_file_fd, buffer + start, size, + start, MYF(MY_FNABP | MY_WME)) || + my_sync(control_file_fd, MYF(MY_WME))); +} + + +/* + Free resources taken by control file subsystem + + SYNOPSIS + ma_control_file_end() +*/ + +void ma_control_file_end() +{ + DBUG_ENTER("ma_control_file_end"); + my_close(control_file_fd, MYF(MY_WME)); + /* + As this module owns these variables, closing the module forbids access to + them (just a safety): + */ + last_checkpoint_lsn= CONTROL_FILE_IMPOSSIBLE_LSN; + last_logno= CONTROL_FILE_IMPOSSIBLE_FILENO; + DBUG_VOID_RETURN; +} diff --git a/storage/maria/ma_control_file.h b/storage/maria/ma_control_file.h new file mode 100644 index 00000000000..d081718b919 --- /dev/null +++ b/storage/maria/ma_control_file.h @@ -0,0 +1,75 @@ +/* + WL#3234 Maria control file + First version written by Guilhem Bichot on 2006-04-27. + Does not compile yet. +*/ + +#ifndef _control_file_h +#define _control_file_h + +/* + Not everybody needs to call the control file that's why control_file.h is + not in maria_def.h. However, policy or habit may want to change this. +*/ + +#ifndef REMOVE_WHEN_SANJA_PUSHES_LOG_HANDLER +/* + this is to get the control file to compile, until Sanja pushes the log + handler which will supersede those definitions. +*/ +typedef struct st_lsn { + uint32 file_no; + uint32 rec_offset; +} LSN; +#define maria_data_root "." +#endif + +/* + indicate absence of the log file number; first log is always number 1, 0 is + impossible. +*/ +#define CONTROL_FILE_IMPOSSIBLE_FILENO 0 +/* logs always have a header */ +#define CONTROL_FILE_IMPOSSIBLE_LOG_OFFSET 0 +/* + indicate absence of LSN. +*/ +#define CONTROL_FILE_IMPOSSIBLE_LSN ((LSN){CONTROL_FILE_IMPOSSIBLE_FILENO,CONTROL_FILE_IMPOSSIBLE_LOG_OFFSET}) + +/* Here is the interface of this module */ + +/* + LSN of the last checkoint + (if last_checkpoint_lsn.file_no == CONTROL_FILE_IMPOSSIBLE_FILENO + then there was never a checkpoint) +*/ +extern LSN last_checkpoint_lsn; +/* + Last log number (if last_logno == + CONTROL_FILE_IMPOSSIBLE_FILENO then there is no log file yet) +*/ +extern uint32 last_logno; + +/* + Looks for the control file. If absent, it's a fresh start, create file. + If present, read it to find out last checkpoint's LSN and last log. + Called at engine's start. +*/ +int ma_control_file_create_or_open(); + +/* + Write information durably to the control file. + Called when we have created a new log (after syncing this log's creation) + and when we have written a checkpoint (after syncing this log record). +*/ +#define CONTROL_FILE_WRITE_ALL 0 /* write all 3 objects */ +#define CONTROL_FILE_WRITE_ONLY_LSN 1 +#define CONTROL_FILE_WRITE_ONLY_LOGNO 2 +int ma_control_file_write_and_force(const LSN *checkpoint_lsn, uint32 logno, + uint objs_to_write); + + +/* Free resources taken by control file subsystem */ +void ma_control_file_end(); + +#endif diff --git a/storage/maria/ma_control_file_test.c b/storage/maria/ma_control_file_test.c new file mode 100644 index 00000000000..b3ba27c8e4b --- /dev/null +++ b/storage/maria/ma_control_file_test.c @@ -0,0 +1,290 @@ +/* Copyright (C) 2006 MySQL AB & MySQL Finland AB & TCX DataKonsult AB + + 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; either version 2 of the License, or + (at your option) any later version. + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ + +/* Unit test of the control file module of the Maria engine */ + +/* TODO: make it fit the mytap framework */ + +/* + Note that it is not possible to test the durability of the write (can't + pull the plug programmatically :) +*/ + +#include "maria.h" +#include "ma_control_file.h" +#include + +char file_name[FN_REFLEN]; +int fd= -1; + +static void clean_files(); +static void run_test_normal(); +static void run_test_abnormal(); +static void usage(); +static void get_options(int argc, char *argv[]); + +int main(int argc,char *argv[]) +{ + MY_INIT(argv[0]); + + get_options(argc,argv); + + clean_files(); + run_test_normal(); + run_test_abnormal(); + + exit(0); /* all ok, if some test failed, we will have aborted */ +} + +/* + Abort unless given expression is non-zero. + + SYNOPSIS + DIE_UNLESS(expr) + + DESCRIPTION + We can't use any kind of system assert as we need to + preserve tested invariants in release builds as well. + + NOTE + This is infamous copy-paste from mysql_client_test.c; + we should instead put it in some include in one single place. +*/ + +#define DIE_UNLESS(expr) \ + ((void) ((expr) ? 0 : (die(__FILE__, __LINE__, #expr), 0))) +#define DIE_IF(expr) \ + ((void) (!(expr) ? 0 : (die(__FILE__, __LINE__, #expr), 0))) +#define DIE(expr) \ + die(__FILE__, __LINE__, #expr) + +void die(const char *file, int line, const char *expr) +{ + fprintf(stderr, "%s:%d: check failed: '%s'\n", file, line, expr); + abort(); +} + + +static void clean_files() +{ + DIE_IF(fn_format(file_name, "control", maria_data_root, "", MYF(MY_WME)) == + NullS); + my_delete(file_name, MYF(0)); /* maybe file does not exist, ignore error */ +} + + +static void run_test_normal() +{ + LSN checkpoint_lsn; + uint32 logno; + uint objs_to_write; + uint i; + char buffer[4]; + + /* TEST0: Instance starts from scratch (control file does not exist) */ + DIE_UNLESS(ma_control_file_create_or_open() == 0); + /* Check that the module reports no information */ + DIE_UNLESS(last_logno == CONTROL_FILE_IMPOSSIBLE_FILENO); + DIE_UNLESS(last_checkpoint_lsn.file_no == CONTROL_FILE_IMPOSSIBLE_FILENO); + DIE_UNLESS(last_checkpoint_lsn.rec_offset == CONTROL_FILE_IMPOSSIBLE_LOG_OFFSET); + + /* TEST1: Simulate creation of one log */ + + objs_to_write= CONTROL_FILE_WRITE_ONLY_LOGNO; + logno= 123; + DIE_UNLESS(ma_control_file_write_and_force(NULL, logno, + objs_to_write) == 0); + /* Check that last_logno was updated */ + DIE_UNLESS(last_logno == logno); + /* Simulate shutdown */ + ma_control_file_end(); + /* Verify amnesia */ + DIE_UNLESS(last_logno == CONTROL_FILE_IMPOSSIBLE_FILENO); + DIE_UNLESS(last_checkpoint_lsn.file_no == CONTROL_FILE_IMPOSSIBLE_FILENO); + DIE_UNLESS(last_checkpoint_lsn.rec_offset == CONTROL_FILE_IMPOSSIBLE_LOG_OFFSET); + /* And restart */ + DIE_UNLESS(ma_control_file_create_or_open() == 0); + DIE_UNLESS(last_logno == logno); + + /* TEST2: Simulate creation of 5 logs */ + + objs_to_write= CONTROL_FILE_WRITE_ONLY_LOGNO; + logno= 100; + for (i= 0; i<5; i++) + { + logno*= 3; + DIE_UNLESS(ma_control_file_write_and_force(NULL, logno, + objs_to_write) == 0); + } + ma_control_file_end(); + DIE_UNLESS(last_logno == CONTROL_FILE_IMPOSSIBLE_FILENO); + DIE_UNLESS(last_checkpoint_lsn.file_no == CONTROL_FILE_IMPOSSIBLE_FILENO); + DIE_UNLESS(last_checkpoint_lsn.rec_offset == CONTROL_FILE_IMPOSSIBLE_LOG_OFFSET); + DIE_UNLESS(ma_control_file_create_or_open() == 0); + DIE_UNLESS(last_logno == logno); + + /* + TEST3: Simulate one checkpoint, one log creation, two checkpoints, one + log creation. + */ + + objs_to_write= CONTROL_FILE_WRITE_ONLY_LSN; + checkpoint_lsn= (LSN){5, 10000}; + logno= 10; + DIE_UNLESS(ma_control_file_write_and_force(&checkpoint_lsn, logno, + objs_to_write) == 0); + /* check that last_logno was not updated */ + DIE_UNLESS(last_logno != logno); + /* Check that last_checkpoint_lsn was updated */ + DIE_UNLESS(last_checkpoint_lsn.file_no == checkpoint_lsn.file_no); + DIE_UNLESS(last_checkpoint_lsn.rec_offset == checkpoint_lsn.rec_offset); + + objs_to_write= CONTROL_FILE_WRITE_ONLY_LOGNO; + checkpoint_lsn= (LSN){5, 20000}; + logno= 17; + DIE_UNLESS(ma_control_file_write_and_force(&checkpoint_lsn, logno, + objs_to_write) == 0); + /* Check that checkpoint LSN was not updated */ + DIE_UNLESS(last_checkpoint_lsn.rec_offset != checkpoint_lsn.rec_offset); + objs_to_write= CONTROL_FILE_WRITE_ONLY_LSN; + checkpoint_lsn= (LSN){17, 20000}; + DIE_UNLESS(ma_control_file_write_and_force(&checkpoint_lsn, logno, + objs_to_write) == 0); + objs_to_write= CONTROL_FILE_WRITE_ONLY_LSN; + checkpoint_lsn= (LSN){17, 45000}; + DIE_UNLESS(ma_control_file_write_and_force(&checkpoint_lsn, logno, + objs_to_write) == 0); + objs_to_write= CONTROL_FILE_WRITE_ONLY_LOGNO; + logno= 19; + DIE_UNLESS(ma_control_file_write_and_force(&checkpoint_lsn, logno, + objs_to_write) == 0); + + ma_control_file_end(); + DIE_UNLESS(last_logno == CONTROL_FILE_IMPOSSIBLE_FILENO); + DIE_UNLESS(last_checkpoint_lsn.file_no == CONTROL_FILE_IMPOSSIBLE_FILENO); + DIE_UNLESS(last_checkpoint_lsn.rec_offset == CONTROL_FILE_IMPOSSIBLE_LOG_OFFSET); + DIE_UNLESS(ma_control_file_create_or_open() == 0); + DIE_UNLESS(last_logno == logno); + DIE_UNLESS(last_checkpoint_lsn.file_no == checkpoint_lsn.file_no); + DIE_UNLESS(last_checkpoint_lsn.rec_offset == checkpoint_lsn.rec_offset); + + /* + TEST4: actually check by ourselves the content of the file. + Note that constants (offsets) are hard-coded here, precisely to prevent + someone from changing them in the control file module and breaking + backward-compatibility. + */ + + DIE_IF((fd= my_open(file_name, + O_BINARY | O_RDWR, + MYF(MY_WME))) < 0); + DIE_IF(my_read(fd, buffer, 16, MYF(MY_FNABP | MY_WME)) != 0); + DIE_IF(my_close(fd, MYF(MY_WME)) != 0); + i= uint4korr(buffer+4); + DIE_UNLESS(i == last_checkpoint_lsn.file_no); + i= uint4korr(buffer+8); + DIE_UNLESS(i == last_checkpoint_lsn.rec_offset); + i= uint4korr(buffer+12); + DIE_UNLESS(i == last_logno); + + + /* TEST5: Simulate stop/start/nothing/stop/start */ + + ma_control_file_end(); + DIE_UNLESS(last_logno == CONTROL_FILE_IMPOSSIBLE_FILENO); + DIE_UNLESS(ma_control_file_create_or_open() == 0); + ma_control_file_end(); + DIE_UNLESS(last_logno == CONTROL_FILE_IMPOSSIBLE_FILENO); + DIE_UNLESS(ma_control_file_create_or_open() == 0); + DIE_UNLESS(last_logno == logno); + DIE_UNLESS(last_checkpoint_lsn.file_no == checkpoint_lsn.file_no); + DIE_UNLESS(last_checkpoint_lsn.rec_offset == checkpoint_lsn.rec_offset); + +} + +static void run_test_abnormal() +{ + /* Corrupt the control file */ + DIE_IF((fd= my_open(file_name, + O_BINARY | O_RDWR, + MYF(MY_WME))) < 0); + DIE_IF(my_write(fd, "papa", 4, MYF(MY_FNABP | MY_WME)) != 0); + DIE_IF(my_close(fd, MYF(MY_WME)) != 0); + + /* Check that control file module sees the problem */ + DIE_IF(ma_control_file_create_or_open() == 0); +} + + +static struct my_option my_long_options[] = +{ +#ifndef DBUG_OFF + {"debug", '#', "Debug log.", + 0, 0, 0, GET_STR, REQUIRED_ARG, 0, 0, 0, 0, 0, 0}, +#endif + {"help", '?', "Display help and exit", + 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, + {"version", 'V', "Print version number and exit", + 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, + { 0, 0, 0, 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0} +}; + + +static void version() +{ + printf("ma_control_file_test: unit test for the control file " + "module of the Maria storage engine. Ver 1.0 \n"); +} + +static my_bool +get_one_option(int optid, const struct my_option *opt __attribute__((unused)), + char *argument) +{ + switch(optid) { + case 'V': + version(); + exit(0); + case '#': + DBUG_PUSH (argument); + break; + case '?': + version(); + usage(); + exit(0); + } + return 0; +} + + +/* Read options */ + +static void get_options(int argc, char *argv[]) +{ + int ho_error; + + if ((ho_error=handle_options(&argc, &argv, my_long_options, get_one_option))) + exit(ho_error); + + return; +} /* get options */ + + +static void usage() +{ + printf("Usage: %s [options]\n\n", my_progname); + my_print_help(my_long_options); + my_print_variables(my_long_options); +} diff --git a/storage/maria/ma_least_recently_dirtied.c b/storage/maria/ma_least_recently_dirtied.c new file mode 100644 index 00000000000..c6285fe47cd --- /dev/null +++ b/storage/maria/ma_least_recently_dirtied.c @@ -0,0 +1,209 @@ +/* + WL#3261 Maria - background flushing of the least-recently-dirtied pages + First version written by Guilhem Bichot on 2006-04-27. + Does not compile yet. +*/ + +/* + To be part of the page cache. + The pseudocode below is dependent on the page cache + which is being designed WL#3134. It is not clear if I need to do page + copies, as the page cache already keeps page copies. + So, this code will move to the page cache and take inspiration from its + methods. Below is just to give the idea of what could be done. + And I should compare my imaginations to WL#3134. +*/ + +/* Here is the implementation of this module */ + +#include "page_cache.h" +#include "least_recently_dirtied.h" + +/* + MikaelR suggested removing this global_LRD_mutex (I have a paper note of + comments), however at least for the first version we'll start with this + mutex (which will be a LOCK-based atomic_rwlock). +*/ +pthread_mutex_t global_LRD_mutex; + +/* + When we flush a page, we should pin page. + This "pin" is to protect against that: + I make copy, + you modify in memory and flush to disk and remove from LRD and from cache, + I write copy to disk, + checkpoint happens. + result: old page is on disk, page is absent from LRD, your REDO will be + wrongly ignored. + + Pin: there can be multiple pins, flushing imposes that there are zero pins. + For example, pin could be a uint counter protected by the page's latch. + + Maybe it's ok if when there is a page replacement, the replacer does not + remove page from the LRD (it would save global mutex); for that, background + flusher should be prepared to see pages in the LRD which are not in the page + cache (then just ignore them). However checkpoint will contain superfluous + entries and so do more work. +*/ + +#define PAGE_SIZE (16*1024) /* just as an example */ +/* + Optimization: + LRD flusher should not flush pages one by one: to be fast, it flushes a + group of pages in sequential disk order if possible; a group of pages is just + FLUSH_GROUP_SIZE pages. + Key cache has groupping already somehow Monty said (investigate that). +*/ +#define FLUSH_GROUP_SIZE 512 /* 8 MB */ +/* + We don't want to probe for checkpoint requests all the time (it takes + the log mutex). + If FLUSH_GROUP_SIZE is 8MB, assuming a local disk which can write 30MB/s + (1.8GB/min), probing every 16th call to flush_one_group_from_LRD() is every + 16*8=128MB which is every 128/30=4.2second. + Using a power of 2 gives a fast modulo operation. +*/ +#define CHECKPOINT_PROBING_PERIOD_LOG2 4 + +/* + This thread does background flush of pieces of the LRD, and all checkpoints. + Just launch it when engine starts. + MikaelR questioned why the same thread does two different jobs, the risk + could be that while a checkpoint happens no LRD flushing happens. +*/ +pthread_handler_decl background_flush_and_checkpoint_thread() +{ + char *flush_group_buffer= my_malloc(PAGE_SIZE*FLUSH_GROUP_SIZE); + uint flush_calls= 0; + while (this_thread_not_killed) + { + if ((flush_calls++) & ((2<data, PAGE_SIZE); + pin_page; + page_cache_unlatch(page_id, KEEP_PINNED); /* but keep pinned */ + } + for (scan_the_array) + { + /* + As an optimization, we try to identify contiguous-in-the-file segments (to + issue one big write()). + In non-optimized version, contiguous segment is always only one page. + */ + if ((next_page.page_id - this_page.page_id) == 1) + { + /* + this page and next page are in same file and are contiguous in the + file: add page to contiguous segment... + */ + continue; /* defer write() to next pages */ + } + /* contiguous segment ends */ + my_pwrite(file, contiguous_segment_start_offset, contiguous_segment_size); + + /* + note that if we had doublewrite, doublewrite buffer may prevent us from + doing this write() grouping (if doublewrite space is shorter). + */ + } + /* + Now remove pages from LRD. As we have pinned them, all pages that we + managed to pin are still in the LRD, in the same order, we can just cut + the LRD at the last element of "array". This is more efficient that + removing element by element (which would take LRD mutex many times) in the + loop above. + */ + lock(global_LRD_mutex); + /* cut LRD by bending LRD->first, free cut portion... */ + unlock(global_LRD_mutex); + for (scan_array) + { + /* + if the page has a property "modified since last flush" (i.e. which is + redundant with the presence of the page in the LRD, this property can + just be a pointer to the LRD element) we should reset it + (note that then the property would live slightly longer than + the presence in LRD). + */ + page_cache_unpin(page_id); + /* + order between unpin and removal from LRD is not clear, depends on what + pin actually is. + */ + } + free(array); + /* + MikaelR noted that he observed that Linux's file cache may never fsync to + disk until this cache is full, at which point it decides to empty the + cache, making the machine very slow. A solution was to fsync after writing + 2 MB. + */ +} + +/* + Flushes all page from LRD up to approximately rec_lsn>=max_lsn. + This is approximate because we flush groups, and because the LRD list may + not be exactly sorted by rec_lsn (because for a big row, all pages of the + row are inserted into the LRD with rec_lsn being the LSN of the REDO for the + first page, so if there are concurrent insertions, the last page of the big + row may have a smaller rec_lsn than the previous pages inserted by + concurrent inserters). +*/ +int flush_all_LRD_to_lsn(LSN max_lsn) +{ + lock(global_LRD_mutex); + if (max_lsn == MAX_LSN) /* don't want to flush forever, so make it fixed: */ + max_lsn= LRD->first->prev->rec_lsn; + while (LRD->first->rec_lsn < max_lsn) + { + if (flush_one_group_from_LRD()) /* will unlock LRD mutex */ + return 1; + /* + The scheduler may preempt us here as we released the mutex; this is good. + */ + lock(global_LRD_mutex); + } + unlock(global_LRD_mutex); + return 0; +} diff --git a/storage/maria/ma_least_recently_dirtied.h b/storage/maria/ma_least_recently_dirtied.h new file mode 100644 index 00000000000..6a30db4b5f0 --- /dev/null +++ b/storage/maria/ma_least_recently_dirtied.h @@ -0,0 +1,10 @@ +/* + WL#3261 Maria - background flushing of the least-recently-dirtied pages + First version written by Guilhem Bichot on 2006-04-27. + Does not compile yet. +*/ + +/* This is the interface of this module. */ + +/* flushes all page from LRD up to approximately rec_lsn>=max_lsn */ +int flush_all_LRD_to_lsn(LSN max_lsn); diff --git a/storage/maria/ma_recovery.c b/storage/maria/ma_recovery.c new file mode 100644 index 00000000000..babf7507ef1 --- /dev/null +++ b/storage/maria/ma_recovery.c @@ -0,0 +1,252 @@ +/* + WL#3072 Maria recovery + First version written by Guilhem Bichot on 2006-04-27. + Does not compile yet. +*/ + +/* Here is the implementation of this module */ + +#include "page_cache.h" +#include "least_recently_dirtied.h" +#include "transaction.h" +#include "share.h" +#include "log.h" + +typedef struct st_record_type_properties { + /* used for debug error messages or "maria_read_log" command-line tool: */ + char *name, + my_bool record_ends_group; + /* a function to execute when we see the record during the REDO phase */ + int (*record_execute_in_redo_phase)(RECORD *); /* param will be record header instead later */ + /* a function to execute when we see the record during the UNDO phase */ + int (*record_execute_in_undo_phase)(RECORD *); /* param will be record header instead later */ +} RECORD_TYPE_PROPERTIES; + +int no_op(RECORD *) {return 0}; + +RECORD_TYPE_PROPERTIES all_record_type_properties[]= +{ + /* listed here in the order of the "log records type" enumeration */ + {"REDO_INSERT_HEAD", FALSE, redo_insert_head_execute_in_redo_phase, no_op}, + ..., + {"UNDO_INSERT" , TRUE , undo_insert_execute_in_redo_phase, undo_insert_execute_in_undo_phase}, + {"COMMIT", , TRUE , commit_execute_in_redo_phase, no_op}, + ... +}; + +int redo_insert_head_execute_in_redo_phase(RECORD *record) +{ + /* write the data to the proper page */ +} + +int undo_insert_execute_in_redo_phase(RECORD *record) +{ + trans_table[short_trans_id].undo_lsn= record.lsn; + /* don't restore the old version of the row */ +} + +int undo_insert_execute_in_undo_phase(RECORD *record) +{ + /* restore the old version of the row */ + trans_table[short_trans_id].undo_lsn= record.prev_undo_lsn; +} + +int commit_execute_in_redo_phase(RECORD *record) +{ + trans_table[short_trans_id].state= COMMITTED; + /* + and that's all: the delete/update handler should not be woken up! as there + may be REDO for purge further in the log. + */ +} + +#define record_ends_group(R) \ + all_record_type_properties[(R)->type].record_ends_group) + +#define execute_log_record_in_redo_phase(R) \ + all_record_type_properties[(R).type].record_execute_in_redo_phase(R) + + +int recovery() +{ + control_file_create_or_open(); + /* + init log handler: tell it that we are going to do large reads of the + log, sequential and backward. Log handler could decide to alloc a big + read-only IO_CACHE for this, or use its usual page cache. + */ + + /* read checkpoint log record from log handler */ + RECORD *checkpoint_record= log_read_record(last_checkpoint_lsn_at_start); + + /* parse this record, build structs (dirty_pages, transactions table, file_map) */ + /* + read log records (note: sometimes only the header is needed, for ex during + REDO phase only the header of UNDO is needed, not the 4G blob in the + variable-length part, so I could use that; however for PREPARE (which is a + variable-length record) I'll need to read the full record in the REDO + phase): + */ + + /**** REDO PHASE *****/ + + record= log_read_record(min(rec_lsn, ...)); /* later, read only header */ + + /* + if log handler knows the end LSN of the log, we could print here how many + MB of log we have to read (to give an idea of the time), and print + progress notes. + */ + + while (record != NULL) + { + /* + A complete group is a set of log records with an "end mark" record + (e.g. a set of REDOs for an operation, terminated by an UNDO for this + operation); if there is no "end mark" record the group is incomplete + and won't be executed. + */ + if (record_ends_group(record) + { + if (trans_table[record.short_trans_id].group_start_lsn != 0) + { + /* + There is a complete group for this transaction, containing more than + this event. + We're going to read recently read log records: + for this log_read_record() to be efficient (not touch the disk), + log handler could cache recently read pages + (can just use an IO_CACHE of 10 MB to read the log, or the normal + log handler page cache). + Without it only OS file cache will help. + */ + record2= + log_read_record(trans_table[record.short_trans_id].group_start_lsn); + + do + { + if (record2.short_trans_id == record.short_trans_id) + execute_log_record_in_redo_phase(record2); /* it's in our group */ + record2= log_read_next_record(); + } + while (record2.lsn < record.lsn); + trans_table[record.short_trans_id].group_start_lsn= 0; /* group finished */ + } + execute_log_record_in_redo_phase(record); + } + else /* record does not end group */ + { + /* just record the fact, can't know if can execute yet */ + if (trans_table[short_trans_id].group_start_lsn == 0) /* group not yet started */ + trans_table[short_trans_id].group_start_lsn= record.lsn; + } + + /* + Later we can optimize: instead of "execute_log_record(record2)", do + copy_record_into_exec_buffer(record2): + this will just copy record into a multi-record (10 MB?) memory buffer, + and when buffer is full, will do sorting of REDOs per + page id and execute them. + This sorting will enable us to do more sequential reads of the + data/index pages. + Note that updating bitmap pages (when we have executed a REDO for a page + we update its bitmap page) may break the sequential read of pages, + so maybe we should read and cache bitmap pages in the beginning. + Or ok the sequence will be broken, but quickly all bitmap pages will be + in memory and so the sequence will not be broken anymore. + Sorting could even determine, based on physical device of files + ("st_dev" in stat()), that some files should be should be taken by + different threads, if we want to do parallism. + */ + /* + Here's how to read a complete variable-length record if needed: + read the header, allocate buffer of record length, read whole + record. + */ + record= log_read_next_record(); + } + + /* + Earlier or here, create true transactions in TM. + If done earlier, note that TM should not wake up the delete/update handler + when it receives a commit info, as existing REDO for purge may exist in + the log, and so the delete/update handler may do changes which conflict + with these REDOs. + Even if done here, better to not wake it up now as we're going to free the + page cache. + + MikaelR suggests: support checkpoints during REDO phase too: do checkpoint + after a certain amount of log records have been executed. This helps + against repeated crashes. Those checkpoints could not be user-requested + (as engine is not communicating during the REDO phase), so they would be + automatic: this changes the original assumption that we don't write to the + log while in the REDO phase, but why not. How often should we checkpoint? + */ + + /* + We want to have two steps: + engine->recover_with_max_memory(); + next_engine->recover_with_max_memory(); + engine->init_with_normal_memory(); + next_engine->init_with_normal_memory(); + So: in recover_with_max_memory() allocate a giant page cache, do REDO + phase, then all page cache is flushed and emptied and freed (only retain + small structures like TM): take full checkpoint, which is useful if + next engine crashes in its recovery the next second. + Destroy all shares (maria_close()), then at init_with_normal_memory() we + do this: + */ + + /**** UNDO PHASE *****/ + + print_information_to_error_log(nb of trans to roll back, nb of prepared trans); + + /* + Launch one or more threads to do the background rollback. Don't wait for + them to complete their rollback (background rollback; for debugging, we + can have an option which waits). Set a counter (total_of_rollback_threads) + to the number of threads to lauch. + + Note that InnoDB's rollback-in-background works as long as InnoDB is the + last engine to recover, otherwise MySQL will refuse new connections until + the last engine has recovered so it's not "background" from the user's + point of view. InnoDB is near top of sys_table_types so all others + (e.g. BDB) recover after it... So it's really "online rollback" only if + InnoDB is the only engine. + */ + + /* wake up delete/update handler */ + /* tell the TM that it can now accept new transactions */ + + /* + mark that checkpoint requests are now allowed. + */ +} + +pthread_handler_decl rollback_background_thread() +{ + /* + execute the normal runtime-rollback code for a bunch of transactions. + */ + while (trans in list_of_trans_to_rollback_by_this_thread) + { + while (trans->undo_lsn != 0) + { + /* this is the normal runtime-rollback code: */ + record= log_read_record(trans->undo_lsn); + execute_log_record_in_undo_phase(record); + trans->undo_lsn= record.prev_undo_lsn; + } + /* remove trans from list */ + } + lock_mutex(rollback_threads); /* or atomic counter */ + if (--total_of_rollback_threads == 0) + { + /* + All rollback threads are done. Print "rollback finished" to the error + log and take a full checkpoint. + */ + } + unlock_mutex(rollback_threads); + pthread_exit(); +} diff --git a/storage/maria/ma_recovery.h b/storage/maria/ma_recovery.h new file mode 100644 index 00000000000..b85ffdeef59 --- /dev/null +++ b/storage/maria/ma_recovery.h @@ -0,0 +1,10 @@ +/* + WL#3072 Maria recovery + First version written by Guilhem Bichot on 2006-04-27. + Does not compile yet. +*/ + +/* This is the interface of this module. */ + +/* Performs recovery of the engine at start */ +int recovery(); diff --git a/storage/maria/recovery.c b/storage/maria/recovery.c deleted file mode 100644 index babf7507ef1..00000000000 --- a/storage/maria/recovery.c +++ /dev/null @@ -1,252 +0,0 @@ -/* - WL#3072 Maria recovery - First version written by Guilhem Bichot on 2006-04-27. - Does not compile yet. -*/ - -/* Here is the implementation of this module */ - -#include "page_cache.h" -#include "least_recently_dirtied.h" -#include "transaction.h" -#include "share.h" -#include "log.h" - -typedef struct st_record_type_properties { - /* used for debug error messages or "maria_read_log" command-line tool: */ - char *name, - my_bool record_ends_group; - /* a function to execute when we see the record during the REDO phase */ - int (*record_execute_in_redo_phase)(RECORD *); /* param will be record header instead later */ - /* a function to execute when we see the record during the UNDO phase */ - int (*record_execute_in_undo_phase)(RECORD *); /* param will be record header instead later */ -} RECORD_TYPE_PROPERTIES; - -int no_op(RECORD *) {return 0}; - -RECORD_TYPE_PROPERTIES all_record_type_properties[]= -{ - /* listed here in the order of the "log records type" enumeration */ - {"REDO_INSERT_HEAD", FALSE, redo_insert_head_execute_in_redo_phase, no_op}, - ..., - {"UNDO_INSERT" , TRUE , undo_insert_execute_in_redo_phase, undo_insert_execute_in_undo_phase}, - {"COMMIT", , TRUE , commit_execute_in_redo_phase, no_op}, - ... -}; - -int redo_insert_head_execute_in_redo_phase(RECORD *record) -{ - /* write the data to the proper page */ -} - -int undo_insert_execute_in_redo_phase(RECORD *record) -{ - trans_table[short_trans_id].undo_lsn= record.lsn; - /* don't restore the old version of the row */ -} - -int undo_insert_execute_in_undo_phase(RECORD *record) -{ - /* restore the old version of the row */ - trans_table[short_trans_id].undo_lsn= record.prev_undo_lsn; -} - -int commit_execute_in_redo_phase(RECORD *record) -{ - trans_table[short_trans_id].state= COMMITTED; - /* - and that's all: the delete/update handler should not be woken up! as there - may be REDO for purge further in the log. - */ -} - -#define record_ends_group(R) \ - all_record_type_properties[(R)->type].record_ends_group) - -#define execute_log_record_in_redo_phase(R) \ - all_record_type_properties[(R).type].record_execute_in_redo_phase(R) - - -int recovery() -{ - control_file_create_or_open(); - /* - init log handler: tell it that we are going to do large reads of the - log, sequential and backward. Log handler could decide to alloc a big - read-only IO_CACHE for this, or use its usual page cache. - */ - - /* read checkpoint log record from log handler */ - RECORD *checkpoint_record= log_read_record(last_checkpoint_lsn_at_start); - - /* parse this record, build structs (dirty_pages, transactions table, file_map) */ - /* - read log records (note: sometimes only the header is needed, for ex during - REDO phase only the header of UNDO is needed, not the 4G blob in the - variable-length part, so I could use that; however for PREPARE (which is a - variable-length record) I'll need to read the full record in the REDO - phase): - */ - - /**** REDO PHASE *****/ - - record= log_read_record(min(rec_lsn, ...)); /* later, read only header */ - - /* - if log handler knows the end LSN of the log, we could print here how many - MB of log we have to read (to give an idea of the time), and print - progress notes. - */ - - while (record != NULL) - { - /* - A complete group is a set of log records with an "end mark" record - (e.g. a set of REDOs for an operation, terminated by an UNDO for this - operation); if there is no "end mark" record the group is incomplete - and won't be executed. - */ - if (record_ends_group(record) - { - if (trans_table[record.short_trans_id].group_start_lsn != 0) - { - /* - There is a complete group for this transaction, containing more than - this event. - We're going to read recently read log records: - for this log_read_record() to be efficient (not touch the disk), - log handler could cache recently read pages - (can just use an IO_CACHE of 10 MB to read the log, or the normal - log handler page cache). - Without it only OS file cache will help. - */ - record2= - log_read_record(trans_table[record.short_trans_id].group_start_lsn); - - do - { - if (record2.short_trans_id == record.short_trans_id) - execute_log_record_in_redo_phase(record2); /* it's in our group */ - record2= log_read_next_record(); - } - while (record2.lsn < record.lsn); - trans_table[record.short_trans_id].group_start_lsn= 0; /* group finished */ - } - execute_log_record_in_redo_phase(record); - } - else /* record does not end group */ - { - /* just record the fact, can't know if can execute yet */ - if (trans_table[short_trans_id].group_start_lsn == 0) /* group not yet started */ - trans_table[short_trans_id].group_start_lsn= record.lsn; - } - - /* - Later we can optimize: instead of "execute_log_record(record2)", do - copy_record_into_exec_buffer(record2): - this will just copy record into a multi-record (10 MB?) memory buffer, - and when buffer is full, will do sorting of REDOs per - page id and execute them. - This sorting will enable us to do more sequential reads of the - data/index pages. - Note that updating bitmap pages (when we have executed a REDO for a page - we update its bitmap page) may break the sequential read of pages, - so maybe we should read and cache bitmap pages in the beginning. - Or ok the sequence will be broken, but quickly all bitmap pages will be - in memory and so the sequence will not be broken anymore. - Sorting could even determine, based on physical device of files - ("st_dev" in stat()), that some files should be should be taken by - different threads, if we want to do parallism. - */ - /* - Here's how to read a complete variable-length record if needed: - read the header, allocate buffer of record length, read whole - record. - */ - record= log_read_next_record(); - } - - /* - Earlier or here, create true transactions in TM. - If done earlier, note that TM should not wake up the delete/update handler - when it receives a commit info, as existing REDO for purge may exist in - the log, and so the delete/update handler may do changes which conflict - with these REDOs. - Even if done here, better to not wake it up now as we're going to free the - page cache. - - MikaelR suggests: support checkpoints during REDO phase too: do checkpoint - after a certain amount of log records have been executed. This helps - against repeated crashes. Those checkpoints could not be user-requested - (as engine is not communicating during the REDO phase), so they would be - automatic: this changes the original assumption that we don't write to the - log while in the REDO phase, but why not. How often should we checkpoint? - */ - - /* - We want to have two steps: - engine->recover_with_max_memory(); - next_engine->recover_with_max_memory(); - engine->init_with_normal_memory(); - next_engine->init_with_normal_memory(); - So: in recover_with_max_memory() allocate a giant page cache, do REDO - phase, then all page cache is flushed and emptied and freed (only retain - small structures like TM): take full checkpoint, which is useful if - next engine crashes in its recovery the next second. - Destroy all shares (maria_close()), then at init_with_normal_memory() we - do this: - */ - - /**** UNDO PHASE *****/ - - print_information_to_error_log(nb of trans to roll back, nb of prepared trans); - - /* - Launch one or more threads to do the background rollback. Don't wait for - them to complete their rollback (background rollback; for debugging, we - can have an option which waits). Set a counter (total_of_rollback_threads) - to the number of threads to lauch. - - Note that InnoDB's rollback-in-background works as long as InnoDB is the - last engine to recover, otherwise MySQL will refuse new connections until - the last engine has recovered so it's not "background" from the user's - point of view. InnoDB is near top of sys_table_types so all others - (e.g. BDB) recover after it... So it's really "online rollback" only if - InnoDB is the only engine. - */ - - /* wake up delete/update handler */ - /* tell the TM that it can now accept new transactions */ - - /* - mark that checkpoint requests are now allowed. - */ -} - -pthread_handler_decl rollback_background_thread() -{ - /* - execute the normal runtime-rollback code for a bunch of transactions. - */ - while (trans in list_of_trans_to_rollback_by_this_thread) - { - while (trans->undo_lsn != 0) - { - /* this is the normal runtime-rollback code: */ - record= log_read_record(trans->undo_lsn); - execute_log_record_in_undo_phase(record); - trans->undo_lsn= record.prev_undo_lsn; - } - /* remove trans from list */ - } - lock_mutex(rollback_threads); /* or atomic counter */ - if (--total_of_rollback_threads == 0) - { - /* - All rollback threads are done. Print "rollback finished" to the error - log and take a full checkpoint. - */ - } - unlock_mutex(rollback_threads); - pthread_exit(); -} diff --git a/storage/maria/recovery.h b/storage/maria/recovery.h deleted file mode 100644 index b85ffdeef59..00000000000 --- a/storage/maria/recovery.h +++ /dev/null @@ -1,10 +0,0 @@ -/* - WL#3072 Maria recovery - First version written by Guilhem Bichot on 2006-04-27. - Does not compile yet. -*/ - -/* This is the interface of this module. */ - -/* Performs recovery of the engine at start */ -int recovery(); -- cgit v1.2.1 From fa793bca310a208242d315b763bb512c252cfa49 Mon Sep 17 00:00:00 2001 From: unknown Date: Mon, 4 Sep 2006 16:53:09 +0200 Subject: WL#3234 "Maria - control file manager": added checksum of the file. Now we have size + magic string + checksum to detect that all is ok. Plus misc fixes for "make dist" to work and the resulting tarball to build include/Makefile.am: adding pagecache.h to help the tarball build. The model of pagecache.h, keycache.h, is in pkginclude_HEADERS, wonder why. Adding pagecache.h to noinst_HEADERS for now. storage/maria/Makefile.am: adding ma_control_file.h to help the tarball build storage/maria/ma_control_file.c: adding a simple checksum to the control file. We protect against corruption of this file like this: - test size - test magic string at start - test checksum I also add some simple my_message() errors (to be changed to a better reporting later). storage/maria/ma_control_file.h: comments storage/maria/ma_control_file_test.c: test of wrong checksum in control file storage/maria/CMakeLists.txt: just to make "make dist" happy for now. --- include/Makefile.am | 2 +- storage/maria/CMakeLists.txt | 1 + storage/maria/Makefile.am | 4 +- storage/maria/ma_control_file.c | 189 +++++++++++++++++++++++------------ storage/maria/ma_control_file.h | 6 +- storage/maria/ma_control_file_test.c | 52 +++++++--- 6 files changed, 168 insertions(+), 86 deletions(-) create mode 100644 storage/maria/CMakeLists.txt diff --git a/include/Makefile.am b/include/Makefile.am index e81fa3a6520..bf7dcd265be 100644 --- a/include/Makefile.am +++ b/include/Makefile.am @@ -33,7 +33,7 @@ noinst_HEADERS = config-win.h config-netware.h \ mysql_version.h.in my_handler.h my_time.h decimal.h \ my_vle.h my_user.h my_atomic.h atomic/nolock.h \ atomic/rwlock.h atomic/x86-gcc.h atomic/x86-msvc.h \ - my_libwrap.h + my_libwrap.h pagecache.h # mysql_version.h are generated CLEANFILES = mysql_version.h my_config.h readline openssl diff --git a/storage/maria/CMakeLists.txt b/storage/maria/CMakeLists.txt new file mode 100644 index 00000000000..cfe23054e2f --- /dev/null +++ b/storage/maria/CMakeLists.txt @@ -0,0 +1 @@ +# empty for the moment; will fill it when we build under Windows diff --git a/storage/maria/Makefile.am b/storage/maria/Makefile.am index e2689698d62..15a5771b5d2 100644 --- a/storage/maria/Makefile.am +++ b/storage/maria/Makefile.am @@ -28,7 +28,9 @@ bin_PROGRAMS = maria_chk maria_pack maria_ftdump maria_chk_DEPENDENCIES= $(LIBRARIES) maria_pack_DEPENDENCIES=$(LIBRARIES) noinst_PROGRAMS = ma_test1 ma_test2 ma_test3 ma_rt_test ma_sp_test ma_control_file_test -noinst_HEADERS = maria_def.h ma_rt_index.h ma_rt_key.h ma_rt_mbr.h ma_sp_defs.h ma_fulltext.h ma_ftdefs.h ma_ft_test1.h ma_ft_eval.h +noinst_HEADERS = maria_def.h ma_rt_index.h ma_rt_key.h ma_rt_mbr.h \ + ma_sp_defs.h ma_fulltext.h ma_ftdefs.h ma_ft_test1.h ma_ft_eval.h \ + ma_control_file.h ma_test1_DEPENDENCIES= $(LIBRARIES) ma_test2_DEPENDENCIES= $(LIBRARIES) ma_test3_DEPENDENCIES= $(LIBRARIES) diff --git a/storage/maria/ma_control_file.c b/storage/maria/ma_control_file.c index d36e1c04c0c..5861246baf9 100644 --- a/storage/maria/ma_control_file.c +++ b/storage/maria/ma_control_file.c @@ -17,12 +17,14 @@ /* total size should be < sector size for atomic write operation */ #define CONTROL_FILE_MAGIC_STRING "MACF" #define CONTROL_FILE_MAGIC_STRING_OFFSET 0 -#define CONTROL_FILE_MAGIC_STRING_SIZE 4 -#define CONTROL_FILE_LSN_OFFSET (CONTROL_FILE_MAGIC_STRING_OFFSET + CONTROL_FILE_MAGIC_STRING_SIZE) +#define CONTROL_FILE_MAGIC_STRING_SIZE (sizeof(CONTROL_FILE_MAGIC_STRING)-1) +#define CONTROL_FILE_CHECKSUM_OFFSET (CONTROL_FILE_MAGIC_STRING_OFFSET + CONTROL_FILE_MAGIC_STRING_SIZE) +#define CONTROL_FILE_CHECKSUM_SIZE 1 +#define CONTROL_FILE_LSN_OFFSET (CONTROL_FILE_CHECKSUM_OFFSET + CONTROL_FILE_CHECKSUM_SIZE) #define CONTROL_FILE_LSN_SIZE (4+4) #define CONTROL_FILE_FILENO_OFFSET (CONTROL_FILE_LSN_OFFSET + CONTROL_FILE_LSN_SIZE) #define CONTROL_FILE_FILENO_SIZE 4 -#define CONTROL_FILE_MAX_SIZE (CONTROL_FILE_FILENO_OFFSET + CONTROL_FILE_FILENO_SIZE) +#define CONTROL_FILE_SIZE (CONTROL_FILE_FILENO_OFFSET + CONTROL_FILE_FILENO_SIZE) /* This module owns these two vars. @@ -55,6 +57,16 @@ static LSN lsn8korr(char *buffer) return tmp; } +static char simple_checksum(char *buffer, uint size) +{ + /* TODO: improve this sum if we want */ + char s= 0; + uint i; + for (i= 0; i Date: Thu, 7 Sep 2006 11:12:37 +0200 Subject: Merge of Myisam changes into Maria. First step: ha_maria moves to storage/maria. BitKeeper/deleted/.del-ft_maria.c: Delete: storage/maria/ft_maria.c storage/maria/ha_maria.h: Rename: sql/ha_maria.h -> storage/maria/ha_maria.h storage/maria/ha_maria.cc: Rename: sql/ha_maria.cc -> storage/maria/ha_maria.cc libmysqld/Makefile.am: ha_maria moves to other dir (like myisam has) sql/Makefile.am: ha_maria moves to other dir (like myisam has) sql/mysqld.cc: ha_maria moves to other dir (like myisam has) storage/maria/Makefile.am: I delete ft_maria.c like ft_myisam.c has storage/maria/ma_test_all.sh: -l option is removed (no MyISAM log in Maria), maria_log is removed too. --- libmysqld/Makefile.am | 2 +- sql/Makefile.am | 4 +- sql/ha_maria.cc | 1836 ------------------------------------------ sql/ha_maria.h | 145 ---- sql/mysqld.cc | 2 +- storage/maria/Makefile.am | 2 +- storage/maria/ft_maria.c | 49 -- storage/maria/ha_maria.cc | 1836 ++++++++++++++++++++++++++++++++++++++++++ storage/maria/ha_maria.h | 145 ++++ storage/maria/ma_test_all.sh | 6 +- 10 files changed, 1988 insertions(+), 2039 deletions(-) delete mode 100644 sql/ha_maria.cc delete mode 100644 sql/ha_maria.h delete mode 100644 storage/maria/ft_maria.c create mode 100644 storage/maria/ha_maria.cc create mode 100644 storage/maria/ha_maria.h diff --git a/libmysqld/Makefile.am b/libmysqld/Makefile.am index 07d8a98362d..befbd3fad3a 100644 --- a/libmysqld/Makefile.am +++ b/libmysqld/Makefile.am @@ -46,7 +46,7 @@ noinst_HEADERS = embedded_priv.h emb_qcache.h sqlsources = derror.cc field.cc field_conv.cc strfunc.cc filesort.cc \ ha_heap.cc ha_myisam.cc ha_myisammrg.cc \ ha_innodb.cc ha_berkeley.cc ha_federated.cc ha_ndbcluster.cc \ - ha_ndbcluster_binlog.cc ha_partition.cc ha_maria.cc\ + ha_ndbcluster_binlog.cc ha_partition.cc \ handler.cc sql_handler.cc \ hostname.cc init.cc password.c \ item.cc item_buff.cc item_cmpfunc.cc item_create.cc \ diff --git a/sql/Makefile.am b/sql/Makefile.am index 0e4621c688e..e453bd6010f 100644 --- a/sql/Makefile.am +++ b/sql/Makefile.am @@ -52,7 +52,7 @@ noinst_HEADERS = item.h item_func.h item_sum.h item_cmpfunc.h \ ha_heap.h ha_myisam.h ha_myisammrg.h ha_partition.h \ ha_innodb.h ha_berkeley.h ha_federated.h \ ha_ndbcluster.h ha_ndbcluster_binlog.h \ - ha_ndbcluster_tables.h ha_maria.h\ + ha_ndbcluster_tables.h \ opt_range.h protocol.h rpl_tblmap.h \ log.h sql_show.h rpl_rli.h \ sql_select.h structs.h table.h sql_udf.h hash_filo.h \ @@ -89,7 +89,7 @@ mysqld_SOURCES = sql_lex.cc sql_handler.cc sql_partition.cc \ records.cc filesort.cc handler.cc \ ha_heap.cc ha_myisam.cc ha_myisammrg.cc \ ha_partition.cc ha_innodb.cc ha_berkeley.cc \ - ha_federated.cc ha_maria.cc\ + ha_federated.cc \ ha_ndbcluster.cc ha_ndbcluster_binlog.cc \ sql_db.cc sql_table.cc sql_rename.cc sql_crypt.cc \ sql_load.cc mf_iocache.cc field_conv.cc sql_show.cc \ diff --git a/sql/ha_maria.cc b/sql/ha_maria.cc deleted file mode 100644 index 29718f1493e..00000000000 --- a/sql/ha_maria.cc +++ /dev/null @@ -1,1836 +0,0 @@ -/* Copyright (C) 2006,2004 MySQL AB & MySQL Finland AB & TCX DataKonsult AB - - 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; either version 2 of the License, or - (at your option) any later version. - - 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ - - -#ifdef USE_PRAGMA_IMPLEMENTATION -#pragma implementation // gcc: Class implementation -#endif - -#include "mysql_priv.h" -#include -#include -#include "ha_maria.h" -#ifndef MASTER -#include "../srclib/maria/maria_def.h" -#else -#include "../storage/maria/maria_def.h" -#include "../storage/maria/ma_rt_index.h" -#endif - -#include - -ulong maria_recover_options= HA_RECOVER_NONE; - -/* bits in maria_recover_options */ -const char *maria_recover_names[]= -{ - "DEFAULT", "BACKUP", "FORCE", "QUICK", NullS -}; -TYPELIB maria_recover_typelib= -{ - array_elements(maria_recover_names) - 1, "", - maria_recover_names, NULL -}; - -const char *maria_stats_method_names[]= -{ - "nulls_unequal", "nulls_equal", - "nulls_ignored", NullS -}; -TYPELIB maria_stats_method_typelib= -{ - array_elements(maria_stats_method_names) - 1, "", - maria_stats_method_names, NULL -}; - - -/***************************************************************************** -** MARIA tables -*****************************************************************************/ - -static handler *maria_create_handler(TABLE_SHARE * table, MEM_ROOT *mem_root) -{ - return new (mem_root) ha_maria(table); -} - - -// collect errors printed by maria_check routines - -static void _ma_check_print_msg(HA_CHECK *param, const char *msg_type, - const char *fmt, va_list args) -{ - THD *thd= (THD *) param->thd; - Protocol *protocol= thd->protocol; - uint length, msg_length; - char msgbuf[MARIA_MAX_MSG_BUF]; - char name[NAME_LEN * 2 + 2]; - - msg_length= my_vsnprintf(msgbuf, sizeof(msgbuf), fmt, args); - msgbuf[sizeof(msgbuf) - 1]= 0; // healthy paranoia - - DBUG_PRINT(msg_type, ("message: %s", msgbuf)); - - if (!thd->vio_ok()) - { - sql_print_error(msgbuf); - return; - } - - if (param->testflag & - (T_CREATE_MISSING_KEYS | T_SAFE_REPAIR | T_AUTO_REPAIR)) - { - my_message(ER_NOT_KEYFILE, msgbuf, MYF(MY_WME)); - return; - } - length= (uint) (strxmov(name, param->db_name, ".", param->table_name, - NullS) - name); - protocol->prepare_for_resend(); - protocol->store(name, length, system_charset_info); - protocol->store(param->op_name, system_charset_info); - protocol->store(msg_type, system_charset_info); - protocol->store(msgbuf, msg_length, system_charset_info); - if (protocol->write()) - sql_print_error("Failed on my_net_write, writing to stderr instead: %s\n", - msgbuf); - return; -} - -extern "C" { - -volatile int *_ma_killed_ptr(HA_CHECK *param) -{ - /* In theory Unsafe conversion, but should be ok for now */ - return (int*) &(((THD *) (param->thd))->killed); -} - - -void _ma_check_print_error(HA_CHECK *param, const char *fmt, ...) -{ - param->error_printed |= 1; - param->out_flag |= O_DATA_LOST; - va_list args; - va_start(args, fmt); - _ma_check_print_msg(param, "error", fmt, args); - va_end(args); -} - - -void _ma_check_print_info(HA_CHECK *param, const char *fmt, ...) -{ - va_list args; - va_start(args, fmt); - _ma_check_print_msg(param, "info", fmt, args); - va_end(args); -} - - -void _ma_check_print_warning(HA_CHECK *param, const char *fmt, ...) -{ - param->warning_printed= 1; - param->out_flag |= O_DATA_LOST; - va_list args; - va_start(args, fmt); - _ma_check_print_msg(param, "warning", fmt, args); - va_end(args); -} - -} - - -ha_maria::ha_maria(TABLE_SHARE *table_arg): -handler(&maria_hton, table_arg), file(0), -int_table_flags(HA_NULL_IN_KEY | HA_CAN_FULLTEXT | HA_CAN_SQL_HANDLER | - HA_DUPLICATE_POS | HA_CAN_INDEX_BLOBS | HA_AUTO_PART_KEY | - HA_FILE_BASED | HA_CAN_GEOMETRY | HA_NO_TRANSACTIONS | - HA_CAN_INSERT_DELAYED | HA_CAN_BIT_FIELD | HA_CAN_RTREEKEYS | - HA_HAS_RECORDS | HA_STATS_RECORDS_IS_EXACT), -can_enable_indexes(1) -{} - - -static const char *ha_maria_exts[]= -{ - MARIA_NAME_IEXT, - MARIA_NAME_DEXT, - NullS -}; - - -const char **ha_maria::bas_ext() const -{ - return ha_maria_exts; -} - - -const char *ha_maria::index_type(uint key_number) -{ - return ((table->key_info[key_number].flags & HA_FULLTEXT) ? - "FULLTEXT" : - (table->key_info[key_number].flags & HA_SPATIAL) ? - "SPATIAL" : - (table->key_info[key_number].algorithm == HA_KEY_ALG_RTREE) ? - "RTREE" : "BTREE"); -} - - -#ifdef HAVE_REPLICATION -int ha_maria::net_read_dump(NET * net) -{ - int data_fd= file->dfile; - int error= 0; - - my_seek(data_fd, 0L, MY_SEEK_SET, MYF(MY_WME)); - for (;;) - { - ulong packet_len= my_net_read(net); - if (!packet_len) - break; // end of file - if (packet_len == packet_error) - { - sql_print_error("ha_maria::net_read_dump - read error "); - error= -1; - goto err; - } - if (my_write(data_fd, (byte *) net->read_pos, (uint) packet_len, - MYF(MY_WME | MY_FNABP))) - { - error= errno; - goto err; - } - } -err: - return error; -} - - -int ha_maria::dump(THD * thd, int fd) -{ - MARIA_SHARE *share= file->s; - NET *net= &thd->net; - uint blocksize= share->blocksize; - my_off_t bytes_to_read= share->state.state.data_file_length; - int data_fd= file->dfile; - byte *buf= (byte *) my_malloc(blocksize, MYF(MY_WME)); - if (!buf) - return ENOMEM; - - int error= 0; - my_seek(data_fd, 0L, MY_SEEK_SET, MYF(MY_WME)); - for (; bytes_to_read > 0;) - { - uint bytes= my_read(data_fd, buf, blocksize, MYF(MY_WME)); - if (bytes == MY_FILE_ERROR) - { - error= errno; - goto err; - } - - if (fd >= 0) - { - if (my_write(fd, buf, bytes, MYF(MY_WME | MY_FNABP))) - { - error= errno ? errno : EPIPE; - goto err; - } - } - else - { - if (my_net_write(net, (char*) buf, bytes)) - { - error= errno ? errno : EPIPE; - goto err; - } - } - bytes_to_read -= bytes; - } - - if (fd < 0) - { - if (my_net_write(net, "", 0)) - error= errno ? errno : EPIPE; - net_flush(net); - } - -err: - my_free((gptr) buf, MYF(0)); - return error; -} -#endif /* HAVE_REPLICATION */ - - -bool ha_maria::check_if_locking_is_allowed(uint sql_command, - ulong type, TABLE *table, - uint count, - bool called_by_logger_thread) -{ - /* - To be able to open and lock for reading system tables like 'mysql.proc', - when we already have some tables opened and locked, and avoid deadlocks - we have to disallow write-locking of these tables with any other tables. - */ - if (table->s->system_table && - table->reginfo.lock_type >= TL_WRITE_ALLOW_WRITE && count != 1) - { - my_error(ER_WRONG_LOCK_OF_SYSTEM_TABLE, MYF(0), table->s->db.str, - table->s->table_name.str); - return FALSE; - } - return TRUE; -} - - - /* Name is here without an extension */ - -int ha_maria::open(const char *name, int mode, uint test_if_locked) -{ - uint i; - if (!(file= maria_open(name, mode, test_if_locked | HA_OPEN_FROM_SQL_LAYER))) - return (my_errno ? my_errno : -1); - - if (test_if_locked & (HA_OPEN_IGNORE_IF_LOCKED | HA_OPEN_TMP_TABLE)) - VOID(maria_extra(file, HA_EXTRA_NO_WAIT_LOCK, 0)); - -#ifdef NOT_USED - if (!(test_if_locked & HA_OPEN_TMP_TABLE) && opt_maria_use_mmap) - VOID(maria_extra(file, HA_EXTRA_MMAP, 0)); -#endif - - info(HA_STATUS_NO_LOCK | HA_STATUS_VARIABLE | HA_STATUS_CONST); - if (!(test_if_locked & HA_OPEN_WAIT_IF_LOCKED)) - VOID(maria_extra(file, HA_EXTRA_WAIT_LOCK, 0)); - if (!table->s->db_record_offset) - int_table_flags |= HA_REC_NOT_IN_SEQ; - if (file->s->options & (HA_OPTION_CHECKSUM | HA_OPTION_COMPRESS_RECORD)) - int_table_flags |= HA_HAS_CHECKSUM; - - for (i= 0; i < table->s->keys; i++) - { - struct st_plugin_int *parser= table->key_info[i].parser; - if (table->key_info[i].flags & HA_USES_PARSER) - file->s->keyinfo[i].parser= - (struct st_mysql_ftparser *) parser->plugin->info; - table->key_info[i].block_size= file->s->keyinfo[i].block_length; - } - return (0); -} - - -int ha_maria::close(void) -{ - MARIA_HA *tmp= file; - file= 0; - return maria_close(tmp); -} - - -int ha_maria::write_row(byte * buf) -{ - statistic_increment(table->in_use->status_var.ha_write_count, &LOCK_status); - - /* If we have a timestamp column, update it to the current time */ - if (table->timestamp_field_type & TIMESTAMP_AUTO_SET_ON_INSERT) - table->timestamp_field->set_time(); - - /* - If we have an auto_increment column and we are writing a changed row - or a new row, then update the auto_increment value in the record. - */ - if (table->next_number_field && buf == table->record[0]) - update_auto_increment(); - return maria_write(file, buf); -} - - -int ha_maria::check(THD * thd, HA_CHECK_OPT * check_opt) -{ - if (!file) - return HA_ADMIN_INTERNAL_ERROR; - int error; - HA_CHECK param; - MARIA_SHARE *share= file->s; - const char *old_proc_info= thd->proc_info; - - thd->proc_info= "Checking table"; - mariachk_init(¶m); - param.thd= thd; - param.op_name= "check"; - param.db_name= table->s->db.str; - param.table_name= table->alias; - param.testflag= check_opt->flags | T_CHECK | T_SILENT; - param.stats_method= (enum_handler_stats_method) thd->variables. - maria_stats_method; - - if (!(table->db_stat & HA_READ_ONLY)) - param.testflag |= T_STATISTICS; - param.using_global_keycache= 1; - - if (!maria_is_crashed(file) && - (((param.testflag & T_CHECK_ONLY_CHANGED) && - !(share->state.changed & (STATE_CHANGED | STATE_CRASHED | - STATE_CRASHED_ON_REPAIR)) && - share->state.open_count == 0) || - ((param.testflag & T_FAST) && (share->state.open_count == - (uint) (share->global_changed ? 1 : - 0))))) - return HA_ADMIN_ALREADY_DONE; - - error= maria_chk_status(¶m, file); // Not fatal - error= maria_chk_size(¶m, file); - if (!error) - error |= maria_chk_del(¶m, file, param.testflag); - if (!error) - error= maria_chk_key(¶m, file); - if (!error) - { - if ((!(param.testflag & T_QUICK) && - ((share->options & - (HA_OPTION_PACK_RECORD | HA_OPTION_COMPRESS_RECORD)) || - (param.testflag & (T_EXTEND | T_MEDIUM)))) || maria_is_crashed(file)) - { - uint old_testflag= param.testflag; - param.testflag |= T_MEDIUM; - if (!(error= init_io_cache(¶m.read_cache, file->dfile, - my_default_record_cache_size, READ_CACHE, - share->pack.header_length, 1, MYF(MY_WME)))) - { - error= maria_chk_data_link(¶m, file, param.testflag & T_EXTEND); - end_io_cache(&(param.read_cache)); - } - param.testflag= old_testflag; - } - } - if (!error) - { - if ((share->state.changed & (STATE_CHANGED | - STATE_CRASHED_ON_REPAIR | - STATE_CRASHED | STATE_NOT_ANALYZED)) || - (param.testflag & T_STATISTICS) || maria_is_crashed(file)) - { - file->update |= HA_STATE_CHANGED | HA_STATE_ROW_CHANGED; - pthread_mutex_lock(&share->intern_lock); - share->state.changed &= ~(STATE_CHANGED | STATE_CRASHED | - STATE_CRASHED_ON_REPAIR); - if (!(table->db_stat & HA_READ_ONLY)) - error= maria_update_state_info(¶m, file, UPDATE_TIME | UPDATE_OPEN_COUNT | - UPDATE_STAT); - pthread_mutex_unlock(&share->intern_lock); - info(HA_STATUS_NO_LOCK | HA_STATUS_TIME | HA_STATUS_VARIABLE | - HA_STATUS_CONST); - } - } - else if (!maria_is_crashed(file) && !thd->killed) - { - maria_mark_crashed(file); - file->update |= HA_STATE_CHANGED | HA_STATE_ROW_CHANGED; - } - - thd->proc_info= old_proc_info; - return error ? HA_ADMIN_CORRUPT : HA_ADMIN_OK; -} - - -/* - Analyze the key distribution in the table - As the table may be only locked for read, we have to take into account that - two threads may do an analyze at the same time! -*/ - -int ha_maria::analyze(THD *thd, HA_CHECK_OPT * check_opt) -{ - int error= 0; - HA_CHECK param; - MARIA_SHARE *share= file->s; - - mariachk_init(¶m); - param.thd= thd; - param.op_name= "analyze"; - param.db_name= table->s->db.str; - param.table_name= table->alias; - param.testflag= (T_FAST | T_CHECK | T_SILENT | T_STATISTICS | - T_DONT_CHECK_CHECKSUM); - param.using_global_keycache= 1; - param.stats_method= (enum_handler_stats_method) thd->variables. - maria_stats_method; - - if (!(share->state.changed & STATE_NOT_ANALYZED)) - return HA_ADMIN_ALREADY_DONE; - - error= maria_chk_key(¶m, file); - if (!error) - { - pthread_mutex_lock(&share->intern_lock); - error= maria_update_state_info(¶m, file, UPDATE_STAT); - pthread_mutex_unlock(&share->intern_lock); - } - else if (!maria_is_crashed(file) && !thd->killed) - maria_mark_crashed(file); - return error ? HA_ADMIN_CORRUPT : HA_ADMIN_OK; -} - - -int ha_maria::restore(THD * thd, HA_CHECK_OPT *check_opt) -{ - HA_CHECK_OPT tmp_check_opt; - char *backup_dir= thd->lex->backup_dir; - char src_path[FN_REFLEN], dst_path[FN_REFLEN]; - const char *table_name= table->s->table_name.str; - int error; - const char *errmsg; - DBUG_ENTER("restore"); - - if (fn_format_relative_to_data_home(src_path, table_name, backup_dir, - MARIA_NAME_DEXT)) - DBUG_RETURN(HA_ADMIN_INVALID); - - strxmov(dst_path, table->s->normalized_path.str, MARIA_NAME_DEXT, NullS); - if (my_copy(src_path, dst_path, MYF(MY_WME))) - { - error= HA_ADMIN_FAILED; - errmsg= "Failed in my_copy (Error %d)"; - goto err; - } - - tmp_check_opt.init(); - tmp_check_opt.flags |= T_VERY_SILENT | T_CALC_CHECKSUM | T_QUICK; - DBUG_RETURN(repair(thd, &tmp_check_opt)); - -err: - { - HA_CHECK param; - mariachk_init(¶m); - param.thd= thd; - param.op_name= "restore"; - param.db_name= table->s->db.str; - param.table_name= table->s->table_name.str; - param.testflag= 0; - _ma_check_print_error(¶m, errmsg, my_errno); - DBUG_RETURN(error); - } -} - - -int ha_maria::backup(THD * thd, HA_CHECK_OPT *check_opt) -{ - char *backup_dir= thd->lex->backup_dir; - char src_path[FN_REFLEN], dst_path[FN_REFLEN]; - const char *table_name= table->s->table_name.str; - int error; - const char *errmsg; - DBUG_ENTER("ha_maria::backup"); - - if (fn_format_relative_to_data_home(dst_path, table_name, backup_dir, - reg_ext)) - { - errmsg= "Failed in fn_format() for .frm file (errno: %d)"; - error= HA_ADMIN_INVALID; - goto err; - } - - strxmov(src_path, table->s->normalized_path.str, reg_ext, NullS); - if (my_copy(src_path, dst_path, - MYF(MY_WME | MY_HOLD_ORIGINAL_MODES | MY_DONT_OVERWRITE_FILE))) - { - error= HA_ADMIN_FAILED; - errmsg= "Failed copying .frm file (errno: %d)"; - goto err; - } - - /* Change extension */ - if (fn_format_relative_to_data_home(dst_path, table_name, backup_dir, - MARIA_NAME_DEXT)) - { - errmsg= "Failed in fn_format() for .MYD file (errno: %d)"; - error= HA_ADMIN_INVALID; - goto err; - } - - strxmov(src_path, table->s->normalized_path.str, MARIA_NAME_DEXT, NullS); - if (my_copy(src_path, dst_path, - MYF(MY_WME | MY_HOLD_ORIGINAL_MODES | MY_DONT_OVERWRITE_FILE))) - { - errmsg= "Failed copying .MYD file (errno: %d)"; - error= HA_ADMIN_FAILED; - goto err; - } - DBUG_RETURN(HA_ADMIN_OK); - -err: - { - HA_CHECK param; - mariachk_init(¶m); - param.thd= thd; - param.op_name= "backup"; - param.db_name= table->s->db.str; - param.table_name= table->s->table_name.str; - param.testflag= 0; - _ma_check_print_error(¶m, errmsg, my_errno); - DBUG_RETURN(error); - } -} - - -int ha_maria::repair(THD * thd, HA_CHECK_OPT *check_opt) -{ - int error; - HA_CHECK param; - ha_rows start_records; - - if (!file) - return HA_ADMIN_INTERNAL_ERROR; - - mariachk_init(¶m); - param.thd= thd; - param.op_name= "repair"; - param.testflag= ((check_opt->flags & ~(T_EXTEND)) | - T_SILENT | T_FORCE_CREATE | T_CALC_CHECKSUM | - (check_opt->flags & T_EXTEND ? T_REP : T_REP_BY_SORT)); - param.sort_buffer_length= check_opt->sort_buffer_size; - start_records= file->state->records; - while ((error= repair(thd, param, 0)) && param.retry_repair) - { - param.retry_repair= 0; - if (test_all_bits(param.testflag, - (uint) (T_RETRY_WITHOUT_QUICK | T_QUICK))) - { - param.testflag &= ~T_RETRY_WITHOUT_QUICK; - sql_print_information("Retrying repair of: '%s' without quick", - table->s->path); - continue; - } - param.testflag &= ~T_QUICK; - if ((param.testflag & T_REP_BY_SORT)) - { - param.testflag= (param.testflag & ~T_REP_BY_SORT) | T_REP; - sql_print_information("Retrying repair of: '%s' with keycache", - table->s->path); - continue; - } - break; - } - if (!error && start_records != file->state->records && - !(check_opt->flags & T_VERY_SILENT)) - { - char llbuff[22], llbuff2[22]; - sql_print_information("Found %s of %s rows when repairing '%s'", - llstr(file->state->records, llbuff), - llstr(start_records, llbuff2), table->s->path); - } - return error; -} - -int ha_maria::optimize(THD * thd, HA_CHECK_OPT *check_opt) -{ - int error; - if (!file) - return HA_ADMIN_INTERNAL_ERROR; - HA_CHECK param; - - mariachk_init(¶m); - param.thd= thd; - param.op_name= "optimize"; - param.testflag= (check_opt->flags | T_SILENT | T_FORCE_CREATE | - T_REP_BY_SORT | T_STATISTICS | T_SORT_INDEX); - param.sort_buffer_length= check_opt->sort_buffer_size; - if ((error= repair(thd, param, 1)) && param.retry_repair) - { - sql_print_warning("Warning: Optimize table got errno %d, retrying", - my_errno); - param.testflag &= ~T_REP_BY_SORT; - error= repair(thd, param, 1); - } - return error; -} - - -int ha_maria::repair(THD *thd, HA_CHECK ¶m, bool optimize) -{ - int error= 0; - uint local_testflag= param.testflag; - bool optimize_done= !optimize, statistics_done= 0; - const char *old_proc_info= thd->proc_info; - char fixed_name[FN_REFLEN]; - MARIA_SHARE *share= file->s; - ha_rows rows= file->state->records; - DBUG_ENTER("ha_maria::repair"); - - param.db_name= table->s->db.str; - param.table_name= table->alias; - param.tmpfile_createflag= O_RDWR | O_TRUNC; - param.using_global_keycache= 1; - param.thd= thd; - param.tmpdir= &mysql_tmpdir_list; - param.out_flag= 0; - strmov(fixed_name, file->filename); - - // Don't lock tables if we have used LOCK TABLE - if (!thd->locked_tables && - maria_lock_database(file, table->s->tmp_table ? F_EXTRA_LCK : F_WRLCK)) - { - _ma_check_print_error(¶m, ER(ER_CANT_LOCK), my_errno); - DBUG_RETURN(HA_ADMIN_FAILED); - } - - if (!optimize || - ((file->state->del || share->state.split != file->state->records) && - (!(param.testflag & T_QUICK) || - !(share->state.changed & STATE_NOT_OPTIMIZED_KEYS)))) - { - ulonglong key_map= ((local_testflag & T_CREATE_MISSING_KEYS) ? - maria_get_mask_all_keys_active(share->base.keys) : - share->state.key_map); - uint testflag= param.testflag; - if (maria_test_if_sort_rep(file, file->state->records, key_map, 0) && - (local_testflag & T_REP_BY_SORT)) - { - local_testflag |= T_STATISTICS; - param.testflag |= T_STATISTICS; // We get this for free - statistics_done= 1; - if (thd->variables.maria_repair_threads > 1) - { - char buf[40]; - /* TODO: respect maria_repair_threads variable */ - my_snprintf(buf, 40, "Repair with %d threads", my_count_bits(key_map)); - thd->proc_info= buf; - error= maria_repair_parallel(¶m, file, fixed_name, - param.testflag & T_QUICK); - thd->proc_info= "Repair done"; // to reset proc_info, as - // it was pointing to local buffer - } - else - { - thd->proc_info= "Repair by sorting"; - error= maria_repair_by_sort(¶m, file, fixed_name, - param.testflag & T_QUICK); - } - } - else - { - thd->proc_info= "Repair with keycache"; - param.testflag &= ~T_REP_BY_SORT; - error= maria_repair(¶m, file, fixed_name, param.testflag & T_QUICK); - } - param.testflag= testflag; - optimize_done= 1; - } - if (!error) - { - if ((local_testflag & T_SORT_INDEX) && - (share->state.changed & STATE_NOT_SORTED_PAGES)) - { - optimize_done= 1; - thd->proc_info= "Sorting index"; - error= maria_sort_index(¶m, file, fixed_name); - } - if (!statistics_done && (local_testflag & T_STATISTICS)) - { - if (share->state.changed & STATE_NOT_ANALYZED) - { - optimize_done= 1; - thd->proc_info= "Analyzing"; - error= maria_chk_key(¶m, file); - } - else - local_testflag &= ~T_STATISTICS; // Don't update statistics - } - } - thd->proc_info= "Saving state"; - if (!error) - { - if ((share->state.changed & STATE_CHANGED) || maria_is_crashed(file)) - { - share->state.changed &= ~(STATE_CHANGED | STATE_CRASHED | - STATE_CRASHED_ON_REPAIR); - file->update |= HA_STATE_CHANGED | HA_STATE_ROW_CHANGED; - } - /* - the following 'if', thought conceptually wrong, - is a useful optimization nevertheless. - */ - if (file->state != &file->s->state.state) - file->s->state.state= *file->state; - if (file->s->base.auto_key) - _ma_update_auto_increment_key(¶m, file, 1); - if (optimize_done) - error= maria_update_state_info(¶m, file, - UPDATE_TIME | UPDATE_OPEN_COUNT | - (local_testflag & - T_STATISTICS ? UPDATE_STAT : 0)); - info(HA_STATUS_NO_LOCK | HA_STATUS_TIME | HA_STATUS_VARIABLE | - HA_STATUS_CONST); - if (rows != file->state->records && !(param.testflag & T_VERY_SILENT)) - { - char llbuff[22], llbuff2[22]; - _ma_check_print_warning(¶m, "Number of rows changed from %s to %s", - llstr(rows, llbuff), - llstr(file->state->records, llbuff2)); - } - } - else - { - maria_mark_crashed_on_repair(file); - file->update |= HA_STATE_CHANGED | HA_STATE_ROW_CHANGED; - maria_update_state_info(¶m, file, 0); - } - thd->proc_info= old_proc_info; - if (!thd->locked_tables) - maria_lock_database(file, F_UNLCK); - DBUG_RETURN(error ? HA_ADMIN_FAILED : - !optimize_done ? HA_ADMIN_ALREADY_DONE : HA_ADMIN_OK); -} - - -/* - Assign table indexes to a specific key cache. -*/ - -int ha_maria::assign_to_keycache(THD * thd, HA_CHECK_OPT *check_opt) -{ - KEY_CACHE *new_key_cache= check_opt->key_cache; - const char *errmsg= 0; - int error= HA_ADMIN_OK; - ulonglong map= ~(ulonglong) 0; - TABLE_LIST *table_list= table->pos_in_table_list; - DBUG_ENTER("ha_maria::assign_to_keycache"); - - /* Check validity of the index references */ - if (table_list->use_index) - { - /* We only come here when the user did specify an index map */ - key_map kmap; - if (get_key_map_from_key_list(&kmap, table, table_list->use_index)) - { - errmsg= thd->net.last_error; - error= HA_ADMIN_FAILED; - goto err; - } - map= kmap.to_ulonglong(); - } - - if ((error= maria_assign_to_key_cache(file, map, new_key_cache))) - { - char buf[STRING_BUFFER_USUAL_SIZE]; - my_snprintf(buf, sizeof(buf), - "Failed to flush to index file (errno: %d)", error); - errmsg= buf; - error= HA_ADMIN_CORRUPT; - } - -err: - if (error != HA_ADMIN_OK) - { - /* Send error to user */ - HA_CHECK param; - mariachk_init(¶m); - param.thd= thd; - param.op_name= "assign_to_keycache"; - param.db_name= table->s->db.str; - param.table_name= table->s->table_name.str; - param.testflag= 0; - _ma_check_print_error(¶m, errmsg); - } - DBUG_RETURN(error); -} - - -/* - Preload pages of the index file for a table into the key cache. -*/ - -int ha_maria::preload_keys(THD * thd, HA_CHECK_OPT *check_opt) -{ - int error; - const char *errmsg; - ulonglong map= ~(ulonglong) 0; - TABLE_LIST *table_list= table->pos_in_table_list; - my_bool ignore_leaves= table_list->ignore_leaves; - - DBUG_ENTER("ha_maria::preload_keys"); - - /* Check validity of the index references */ - if (table_list->use_index) - { - key_map kmap; - get_key_map_from_key_list(&kmap, table, table_list->use_index); - if (kmap.is_set_all()) - { - errmsg= thd->net.last_error; - error= HA_ADMIN_FAILED; - goto err; - } - if (!kmap.is_clear_all()) - map= kmap.to_ulonglong(); - } - - maria_extra(file, HA_EXTRA_PRELOAD_BUFFER_SIZE, - (void*) &thd->variables.preload_buff_size); - - if ((error= maria_preload(file, map, ignore_leaves))) - { - switch (error) { - case HA_ERR_NON_UNIQUE_BLOCK_SIZE: - errmsg= "Indexes use different block sizes"; - break; - case HA_ERR_OUT_OF_MEM: - errmsg= "Failed to allocate buffer"; - break; - default: - char buf[ERRMSGSIZE + 20]; - my_snprintf(buf, ERRMSGSIZE, - "Failed to read from index file (errno: %d)", my_errno); - errmsg= buf; - } - error= HA_ADMIN_FAILED; - goto err; - } - - DBUG_RETURN(HA_ADMIN_OK); - -err: - { - HA_CHECK param; - mariachk_init(¶m); - param.thd= thd; - param.op_name= "preload_keys"; - param.db_name= table->s->db.str; - param.table_name= table->s->table_name.str; - param.testflag= 0; - _ma_check_print_error(¶m, errmsg); - DBUG_RETURN(error); - } -} - - -/* - Disable indexes, making it persistent if requested. - - SYNOPSIS - disable_indexes() - mode mode of operation: - HA_KEY_SWITCH_NONUNIQ disable all non-unique keys - HA_KEY_SWITCH_ALL disable all keys - HA_KEY_SWITCH_NONUNIQ_SAVE dis. non-uni. and make persistent - HA_KEY_SWITCH_ALL_SAVE dis. all keys and make persistent - - IMPLEMENTATION - HA_KEY_SWITCH_NONUNIQ is not implemented. - HA_KEY_SWITCH_ALL_SAVE is not implemented. - - RETURN - 0 ok - HA_ERR_WRONG_COMMAND mode not implemented. -*/ - -int ha_maria::disable_indexes(uint mode) -{ - int error; - - if (mode == HA_KEY_SWITCH_ALL) - { - /* call a storage engine function to switch the key map */ - error= maria_disable_indexes(file); - } - else if (mode == HA_KEY_SWITCH_NONUNIQ_SAVE) - { - maria_extra(file, HA_EXTRA_NO_KEYS, 0); - info(HA_STATUS_CONST); // Read new key info - error= 0; - } - else - { - /* mode not implemented */ - error= HA_ERR_WRONG_COMMAND; - } - return error; -} - - -/* - Enable indexes, making it persistent if requested. - - SYNOPSIS - enable_indexes() - mode mode of operation: - HA_KEY_SWITCH_NONUNIQ enable all non-unique keys - HA_KEY_SWITCH_ALL enable all keys - HA_KEY_SWITCH_NONUNIQ_SAVE en. non-uni. and make persistent - HA_KEY_SWITCH_ALL_SAVE en. all keys and make persistent - - DESCRIPTION - Enable indexes, which might have been disabled by disable_index() before. - The modes without _SAVE work only if both data and indexes are empty, - since the MARIA repair would enable them persistently. - To be sure in these cases, call handler::delete_all_rows() before. - - IMPLEMENTATION - HA_KEY_SWITCH_NONUNIQ is not implemented. - HA_KEY_SWITCH_ALL_SAVE is not implemented. - - RETURN - 0 ok - !=0 Error, among others: - HA_ERR_CRASHED data or index is non-empty. Delete all rows and retry. - HA_ERR_WRONG_COMMAND mode not implemented. -*/ - -int ha_maria::enable_indexes(uint mode) -{ - int error; - - if (maria_is_all_keys_active(file->s->state.key_map, file->s->base.keys)) - { - /* All indexes are enabled already. */ - return 0; - } - - if (mode == HA_KEY_SWITCH_ALL) - { - error= maria_enable_indexes(file); - /* - Do not try to repair on error, - as this could make the enabled state persistent, - but mode==HA_KEY_SWITCH_ALL forbids it. - */ - } - else if (mode == HA_KEY_SWITCH_NONUNIQ_SAVE) - { - THD *thd= current_thd; - HA_CHECK param; - const char *save_proc_info= thd->proc_info; - thd->proc_info= "Creating index"; - mariachk_init(¶m); - param.op_name= "recreating_index"; - param.testflag= (T_SILENT | T_REP_BY_SORT | T_QUICK | - T_CREATE_MISSING_KEYS); - param.myf_rw &= ~MY_WAIT_IF_FULL; - param.sort_buffer_length= thd->variables.maria_sort_buff_size; - param.stats_method= (enum_handler_stats_method) thd->variables. - maria_stats_method; - param.tmpdir= &mysql_tmpdir_list; - if ((error= (repair(thd, param, 0) != HA_ADMIN_OK)) && param.retry_repair) - { - sql_print_warning("Warning: Enabling keys got errno %d, retrying", - my_errno); - /* Repairing by sort failed. Now try standard repair method. */ - param.testflag &= ~(T_REP_BY_SORT | T_QUICK); - error= (repair(thd, param, 0) != HA_ADMIN_OK); - /* - If the standard repair succeeded, clear all error messages which - might have been set by the first repair. They can still be seen - with SHOW WARNINGS then. - */ - if (!error) - thd->clear_error(); - } - info(HA_STATUS_CONST); - thd->proc_info= save_proc_info; - } - else - { - /* mode not implemented */ - error= HA_ERR_WRONG_COMMAND; - } - return error; -} - - -/* - Test if indexes are disabled. - - - SYNOPSIS - indexes_are_disabled() - no parameters - - - RETURN - 0 indexes are not disabled - 1 all indexes are disabled - [2 non-unique indexes are disabled - NOT YET IMPLEMENTED] -*/ - -int ha_maria::indexes_are_disabled(void) -{ - return maria_indexes_are_disabled(file); -} - - -/* - prepare for a many-rows insert operation - e.g. - disable indexes (if they can be recreated fast) or - activate special bulk-insert optimizations - - SYNOPSIS - start_bulk_insert(rows) - rows Rows to be inserted - 0 if we don't know - - NOTICE - Do not forget to call end_bulk_insert() later! -*/ - -void ha_maria::start_bulk_insert(ha_rows rows) -{ - DBUG_ENTER("ha_maria::start_bulk_insert"); - THD *thd= current_thd; - ulong size= min(thd->variables.read_buff_size, - table->s->avg_row_length * rows); - DBUG_PRINT("info", ("start_bulk_insert: rows %lu size %lu", - (ulong) rows, size)); - - /* don't enable row cache if too few rows */ - if (!rows || (rows > MARIA_MIN_ROWS_TO_USE_WRITE_CACHE)) - maria_extra(file, HA_EXTRA_WRITE_CACHE, (void*) &size); - - can_enable_indexes= maria_is_all_keys_active(file->s->state.key_map, - file->s->base.keys); - - if (!(specialflag & SPECIAL_SAFE_MODE)) - { - /* - Only disable old index if the table was empty and we are inserting - a lot of rows. - We should not do this for only a few rows as this is slower and - we don't want to update the key statistics based of only a few rows. - */ - if (file->state->records == 0 && can_enable_indexes && - (!rows || rows >= MARIA_MIN_ROWS_TO_DISABLE_INDEXES)) - maria_disable_non_unique_index(file, rows); - else - if (!file->bulk_insert && - (!rows || rows >= MARIA_MIN_ROWS_TO_USE_BULK_INSERT)) - { - maria_init_bulk_insert(file, thd->variables.bulk_insert_buff_size, rows); - } - } - DBUG_VOID_RETURN; -} - - -/* - end special bulk-insert optimizations, - which have been activated by start_bulk_insert(). - - SYNOPSIS - end_bulk_insert() - no arguments - - RETURN - 0 OK - != 0 Error -*/ - -int ha_maria::end_bulk_insert() -{ - maria_end_bulk_insert(file); - int err= maria_extra(file, HA_EXTRA_NO_CACHE, 0); - return err ? err : can_enable_indexes ? - enable_indexes(HA_KEY_SWITCH_NONUNIQ_SAVE) : 0; -} - - -bool ha_maria::check_and_repair(THD *thd) -{ - int error= 0; - int marked_crashed; - char *old_query; - uint old_query_length; - HA_CHECK_OPT check_opt; - DBUG_ENTER("ha_maria::check_and_repair"); - - check_opt.init(); - check_opt.flags= T_MEDIUM | T_AUTO_REPAIR; - // Don't use quick if deleted rows - if (!file->state->del && (maria_recover_options & HA_RECOVER_QUICK)) - check_opt.flags |= T_QUICK; - sql_print_warning("Checking table: '%s'", table->s->path); - - old_query= thd->query; - old_query_length= thd->query_length; - pthread_mutex_lock(&LOCK_thread_count); - thd->query= table->s->table_name.str; - thd->query_length= table->s->table_name.length; - pthread_mutex_unlock(&LOCK_thread_count); - - if ((marked_crashed= maria_is_crashed(file)) || check(thd, &check_opt)) - { - sql_print_warning("Recovering table: '%s'", table->s->path); - check_opt.flags= - ((maria_recover_options & HA_RECOVER_BACKUP ? T_BACKUP_DATA : 0) | - (marked_crashed ? 0 : T_QUICK) | - (maria_recover_options & HA_RECOVER_FORCE ? 0 : T_SAFE_REPAIR) | - T_AUTO_REPAIR); - if (repair(thd, &check_opt)) - error= 1; - } - pthread_mutex_lock(&LOCK_thread_count); - thd->query= old_query; - thd->query_length= old_query_length; - pthread_mutex_unlock(&LOCK_thread_count); - DBUG_RETURN(error); -} - - -bool ha_maria::is_crashed() const -{ - return (file->s->state.changed & STATE_CRASHED || - (my_disable_locking && file->s->state.open_count)); -} - - -int ha_maria::update_row(const byte * old_data, byte * new_data) -{ - statistic_increment(table->in_use->status_var.ha_update_count, &LOCK_status); - if (table->timestamp_field_type & TIMESTAMP_AUTO_SET_ON_UPDATE) - table->timestamp_field->set_time(); - return maria_update(file, old_data, new_data); -} - - -int ha_maria::delete_row(const byte * buf) -{ - statistic_increment(table->in_use->status_var.ha_delete_count, &LOCK_status); - return maria_delete(file, buf); -} - - -int ha_maria::index_read(byte * buf, const byte * key, - uint key_len, enum ha_rkey_function find_flag) -{ - DBUG_ASSERT(inited == INDEX); - statistic_increment(table->in_use->status_var.ha_read_key_count, - &LOCK_status); - int error= maria_rkey(file, buf, active_index, key, key_len, find_flag); - table->status= error ? STATUS_NOT_FOUND : 0; - return error; -} - - -int ha_maria::index_read_idx(byte * buf, uint index, const byte * key, - uint key_len, enum ha_rkey_function find_flag) -{ - statistic_increment(table->in_use->status_var.ha_read_key_count, - &LOCK_status); - int error= maria_rkey(file, buf, index, key, key_len, find_flag); - table->status= error ? STATUS_NOT_FOUND : 0; - return error; -} - - -int ha_maria::index_read_last(byte * buf, const byte * key, uint key_len) -{ - DBUG_ENTER("ha_maria::index_read_last"); - DBUG_ASSERT(inited == INDEX); - statistic_increment(table->in_use->status_var.ha_read_key_count, - &LOCK_status); - int error= maria_rkey(file, buf, active_index, key, key_len, - HA_READ_PREFIX_LAST); - table->status= error ? STATUS_NOT_FOUND : 0; - DBUG_RETURN(error); -} - - -int ha_maria::index_next(byte * buf) -{ - DBUG_ASSERT(inited == INDEX); - statistic_increment(table->in_use->status_var.ha_read_next_count, - &LOCK_status); - int error= maria_rnext(file, buf, active_index); - table->status= error ? STATUS_NOT_FOUND : 0; - return error; -} - - -int ha_maria::index_prev(byte * buf) -{ - DBUG_ASSERT(inited == INDEX); - statistic_increment(table->in_use->status_var.ha_read_prev_count, - &LOCK_status); - int error= maria_rprev(file, buf, active_index); - table->status= error ? STATUS_NOT_FOUND : 0; - return error; -} - - -int ha_maria::index_first(byte * buf) -{ - DBUG_ASSERT(inited == INDEX); - statistic_increment(table->in_use->status_var.ha_read_first_count, - &LOCK_status); - int error= maria_rfirst(file, buf, active_index); - table->status= error ? STATUS_NOT_FOUND : 0; - return error; -} - - -int ha_maria::index_last(byte * buf) -{ - DBUG_ASSERT(inited == INDEX); - statistic_increment(table->in_use->status_var.ha_read_last_count, - &LOCK_status); - int error= maria_rlast(file, buf, active_index); - table->status= error ? STATUS_NOT_FOUND : 0; - return error; -} - - -int ha_maria::index_next_same(byte * buf, - const byte *key __attribute__ ((unused)), - uint length __attribute__ ((unused))) -{ - DBUG_ASSERT(inited == INDEX); - statistic_increment(table->in_use->status_var.ha_read_next_count, - &LOCK_status); - int error= maria_rnext_same(file, buf); - table->status= error ? STATUS_NOT_FOUND : 0; - return error; -} - - -int ha_maria::rnd_init(bool scan) -{ - if (scan) - return maria_scan_init(file); - return maria_reset(file); // Free buffers -} - - -int ha_maria::rnd_next(byte *buf) -{ - statistic_increment(table->in_use->status_var.ha_read_rnd_next_count, - &LOCK_status); - int error= maria_scan(file, buf); - table->status= error ? STATUS_NOT_FOUND : 0; - return error; -} - - -int ha_maria::restart_rnd_next(byte *buf, byte *pos) -{ - return rnd_pos(buf, pos); -} - - -int ha_maria::rnd_pos(byte * buf, byte *pos) -{ - statistic_increment(table->in_use->status_var.ha_read_rnd_count, - &LOCK_status); - int error= maria_rrnd(file, buf, my_get_ptr(pos, ref_length)); - table->status= error ? STATUS_NOT_FOUND : 0; - return error; -} - - -void ha_maria::position(const byte * record) -{ - my_off_t position= maria_position(file); - my_store_ptr(ref, ref_length, position); -} - - -void ha_maria::info(uint flag) -{ - MARIA_INFO info; - char name_buff[FN_REFLEN]; - - (void) maria_status(file, &info, flag); - if (flag & HA_STATUS_VARIABLE) - { - stats.records= info.records; - stats.deleted= info.deleted; - stats.data_file_length= info.data_file_length; - stats.index_file_length= info.index_file_length; - stats.delete_length= info.delete_length; - stats.check_time= info.check_time; - stats.mean_rec_length= info.mean_reclength; - } - if (flag & HA_STATUS_CONST) - { - TABLE_SHARE *share= table->s; - stats.max_data_file_length= info.max_data_file_length; - stats.max_index_file_length= info.max_index_file_length; - stats.create_time= info.create_time; - ref_length= info.reflength; - share->db_options_in_use= info.options; - stats.block_size= maria_block_size; - - /* Update share */ - if (share->tmp_table == NO_TMP_TABLE) - pthread_mutex_lock(&share->mutex); - share->keys_in_use.set_prefix(share->keys); - share->keys_in_use.intersect_extended(info.key_map); - share->keys_for_keyread.intersect(share->keys_in_use); - share->db_record_offset= info.record_offset; - if (share->key_parts) - memcpy((char*) table->key_info[0].rec_per_key, - (char*) info.rec_per_key, - sizeof(table->key_info[0].rec_per_key) * share->key_parts); - if (share->tmp_table == NO_TMP_TABLE) - pthread_mutex_unlock(&share->mutex); - - /* - Set data_file_name and index_file_name to point at the symlink value - if table is symlinked (Ie; Real name is not same as generated name) - */ - data_file_name= index_file_name= 0; - fn_format(name_buff, file->filename, "", MARIA_NAME_DEXT, MY_APPEND_EXT); - if (strcmp(name_buff, info.data_file_name)) - data_file_name= info.data_file_name; - fn_format(name_buff, file->filename, "", MARIA_NAME_IEXT, MY_APPEND_EXT); - if (strcmp(name_buff, info.index_file_name)) - index_file_name= info.index_file_name; - } - if (flag & HA_STATUS_ERRKEY) - { - errkey= info.errkey; - my_store_ptr(dup_ref, ref_length, info.dupp_key_pos); - } - /* Faster to always update, than to do it based on flag */ - stats.update_time= info.update_time; - stats.auto_increment_value= info.auto_increment; -} - - -int ha_maria::extra(enum ha_extra_function operation) -{ - if ((specialflag & SPECIAL_SAFE_MODE) && operation == HA_EXTRA_KEYREAD) - return 0; - return maria_extra(file, operation, 0); -} - -int ha_maria::reset(void) -{ - return maria_reset(file); -} - -/* To be used with WRITE_CACHE and EXTRA_CACHE */ - -int ha_maria::extra_opt(enum ha_extra_function operation, ulong cache_size) -{ - if ((specialflag & SPECIAL_SAFE_MODE) && operation == HA_EXTRA_WRITE_CACHE) - return 0; - return maria_extra(file, operation, (void*) &cache_size); -} - - -int ha_maria::delete_all_rows() -{ - return maria_delete_all_rows(file); -} - - -int ha_maria::delete_table(const char *name) -{ - return maria_delete_table(name); -} - - -int ha_maria::external_lock(THD *thd, int lock_type) -{ - return maria_lock_database(file, !table->s->tmp_table ? - lock_type : ((lock_type == F_UNLCK) ? - F_UNLCK : F_EXTRA_LCK)); -} - - -THR_LOCK_DATA **ha_maria::store_lock(THD *thd, - THR_LOCK_DATA **to, - enum thr_lock_type lock_type) -{ - if (lock_type != TL_IGNORE && file->lock.type == TL_UNLOCK) - file->lock.type= lock_type; - *to++= &file->lock; - return to; -} - - -void ha_maria::update_create_info(HA_CREATE_INFO *create_info) -{ - ha_maria::info(HA_STATUS_AUTO | HA_STATUS_CONST); - if (!(create_info->used_fields & HA_CREATE_USED_AUTO)) - { - create_info->auto_increment_value= stats.auto_increment_value; - } - create_info->data_file_name= data_file_name; - create_info->index_file_name= index_file_name; -} - - -int ha_maria::create(const char *name, register TABLE *table_arg, - HA_CREATE_INFO *info) -{ - int error; - uint i, j, recpos, minpos, fieldpos, temp_length, length, create_flags= 0; - bool found_real_auto_increment= 0; - enum ha_base_keytype type; - char buff[FN_REFLEN]; - KEY *pos; - MARIA_KEYDEF *keydef; - MARIA_COLUMNDEF *recinfo, *recinfo_pos; - HA_KEYSEG *keyseg; - TABLE_SHARE *share= table_arg->s; - uint options= share->db_options_in_use; - DBUG_ENTER("ha_maria::create"); - - type= HA_KEYTYPE_BINARY; // Keep compiler happy - if (!(my_multi_malloc(MYF(MY_WME), - &recinfo, (share->fields * 2 + 2) * - sizeof(MARIA_COLUMNDEF), - &keydef, share->keys * sizeof(MARIA_KEYDEF), - &keyseg, - ((share->key_parts + share->keys) * - sizeof(HA_KEYSEG)), NullS))) - DBUG_RETURN(HA_ERR_OUT_OF_MEM); - - pos= table_arg->key_info; - for (i= 0; i < share->keys; i++, pos++) - { - if (pos->flags & HA_USES_PARSER) - create_flags |= HA_CREATE_RELIES_ON_SQL_LAYER; - keydef[i].flag= (pos->flags & (HA_NOSAME | HA_FULLTEXT | HA_SPATIAL)); - keydef[i].key_alg= pos->algorithm == HA_KEY_ALG_UNDEF ? - (pos->flags & HA_SPATIAL ? HA_KEY_ALG_RTREE : HA_KEY_ALG_BTREE) : - pos->algorithm; - keydef[i].block_length= pos->block_size; - - keydef[i].seg= keyseg; - keydef[i].keysegs= pos->key_parts; - for (j= 0; j < pos->key_parts; j++) - { - Field *field= pos->key_part[j].field; - type= field->key_type(); - keydef[i].seg[j].flag= pos->key_part[j].key_part_flag; - - if (options & HA_OPTION_PACK_KEYS || - (pos->flags & (HA_PACK_KEY | HA_BINARY_PACK_KEY | - HA_SPACE_PACK_USED))) - { - if (pos->key_part[j].length > 8 && - (type == HA_KEYTYPE_TEXT || - type == HA_KEYTYPE_NUM || - (type == HA_KEYTYPE_BINARY && !field->zero_pack()))) - { - /* No blobs here */ - if (j == 0) - keydef[i].flag |= HA_PACK_KEY; - if (!(field->flags & ZEROFILL_FLAG) && - (field->type() == MYSQL_TYPE_STRING || - field->type() == MYSQL_TYPE_VAR_STRING || - ((int) (pos->key_part[j].length - field->decimals())) >= 4)) - keydef[i].seg[j].flag |= HA_SPACE_PACK; - } - else if (j == 0 && (!(pos->flags & HA_NOSAME) || pos->key_length > 16)) - keydef[i].flag |= HA_BINARY_PACK_KEY; - } - keydef[i].seg[j].type= (int) type; - keydef[i].seg[j].start= pos->key_part[j].offset; - keydef[i].seg[j].length= pos->key_part[j].length; - keydef[i].seg[j].bit_start= keydef[i].seg[j].bit_end= - keydef[i].seg[j].bit_length= 0; - keydef[i].seg[j].bit_pos= 0; - keydef[i].seg[j].language= field->charset()->number; - - if (field->null_ptr) - { - keydef[i].seg[j].null_bit= field->null_bit; - keydef[i].seg[j].null_pos= (uint) (field->null_ptr - - (uchar *) table_arg->record[0]); - } - else - { - keydef[i].seg[j].null_bit= 0; - keydef[i].seg[j].null_pos= 0; - } - if (field->type() == FIELD_TYPE_BLOB || - field->type() == FIELD_TYPE_GEOMETRY) - { - keydef[i].seg[j].flag |= HA_BLOB_PART; - /* save number of bytes used to pack length */ - keydef[i].seg[j].bit_start= (uint) (field->pack_length() - - share->blob_ptr_size); - } - else if (field->type() == FIELD_TYPE_BIT) - { - keydef[i].seg[j].bit_length= ((Field_bit *) field)->bit_len; - keydef[i].seg[j].bit_start= ((Field_bit *) field)->bit_ofs; - keydef[i].seg[j].bit_pos= (uint) (((Field_bit *) field)->bit_ptr - - (uchar *) table_arg->record[0]); - } - } - keyseg += pos->key_parts; - } - - if (table_arg->found_next_number_field) - { - keydef[share->next_number_index].flag |= HA_AUTO_KEY; - found_real_auto_increment= share->next_number_key_offset == 0; - } - - recpos= 0; - recinfo_pos= recinfo; - while (recpos < (uint) share->reclength) - { - Field **field, *found= 0; - minpos= share->reclength; - length= 0; - - for (field= table_arg->field; *field; field++) - { - if ((fieldpos= (*field)->offset()) >= recpos && fieldpos <= minpos) - { - /* skip null fields */ - if (!(temp_length= (*field)->pack_length_in_rec())) - continue; /* Skip null-fields */ - if (!found || fieldpos < minpos || - (fieldpos == minpos && temp_length < length)) - { - minpos= fieldpos; - found= *field; - length= temp_length; - } - } - } - DBUG_PRINT("loop", ("found: 0x%lx recpos: %d minpos: %d length: %d", - found, recpos, minpos, length)); - if (recpos != minpos) - { // Reserved space (Null bits?) - bzero((char*) recinfo_pos, sizeof(*recinfo_pos)); - recinfo_pos->type= (int) FIELD_NORMAL; - recinfo_pos++->length= (uint16) (minpos - recpos); - } - if (!found) - break; - - if (found->flags & BLOB_FLAG) - recinfo_pos->type= (int) FIELD_BLOB; - else if (found->type() == MYSQL_TYPE_VARCHAR) - recinfo_pos->type= FIELD_VARCHAR; - else if (!(options & HA_OPTION_PACK_RECORD)) - recinfo_pos->type= (int) FIELD_NORMAL; - else if (found->zero_pack()) - recinfo_pos->type= (int) FIELD_SKIP_ZERO; - else - recinfo_pos->type= (int) ((length <= 3 || - (found->flags & ZEROFILL_FLAG)) ? - FIELD_NORMAL : - found->type() == MYSQL_TYPE_STRING || - found->type() == MYSQL_TYPE_VAR_STRING ? - FIELD_SKIP_ENDSPACE : FIELD_SKIP_PRESPACE); - if (found->null_ptr) - { - recinfo_pos->null_bit= found->null_bit; - recinfo_pos->null_pos= (uint) (found->null_ptr - - (uchar *) table_arg->record[0]); - } - else - { - recinfo_pos->null_bit= 0; - recinfo_pos->null_pos= 0; - } - (recinfo_pos++)->length= (uint16) length; - recpos= minpos + length; - DBUG_PRINT("loop", ("length: %d type: %d", - recinfo_pos[-1].length, recinfo_pos[-1].type)); - - } - MARIA_CREATE_INFO create_info; - bzero((char*) &create_info, sizeof(create_info)); - create_info.max_rows= share->max_rows; - create_info.reloc_rows= share->min_rows; - create_info.with_auto_increment= found_real_auto_increment; - create_info.auto_increment= (info->auto_increment_value ? - info->auto_increment_value - 1 : (ulonglong) 0); - create_info.data_file_length= ((ulonglong) share->max_rows * - share->avg_row_length); - create_info.data_file_name= info->data_file_name; - create_info.index_file_name= info->index_file_name; - - if (info->options & HA_LEX_CREATE_TMP_TABLE) - create_flags |= HA_CREATE_TMP_TABLE; - if (options & HA_OPTION_PACK_RECORD) - create_flags |= HA_PACK_RECORD; - if (options & HA_OPTION_CHECKSUM) - create_flags |= HA_CREATE_CHECKSUM; - if (options & HA_OPTION_DELAY_KEY_WRITE) - create_flags |= HA_CREATE_DELAY_KEY_WRITE; - - /* TODO: Check that the following fn_format is really needed */ - error= - maria_create(fn_format - (buff, name, "", "", MY_UNPACK_FILENAME | MY_APPEND_EXT), - share->keys, keydef, (uint) (recinfo_pos - recinfo), recinfo, - 0, (MARIA_UNIQUEDEF *) 0, &create_info, create_flags); - - my_free((gptr) recinfo, MYF(0)); - DBUG_RETURN(error); -} - - -int ha_maria::rename_table(const char *from, const char *to) -{ - return maria_rename(from, to); -} - - -void ha_maria::get_auto_increment(ulonglong offset, ulonglong increment, - ulonglong nb_desired_values, - ulonglong *first_value, - ulonglong *nb_reserved_values) -{ - ulonglong nr; - int error; - byte key[HA_MAX_KEY_LENGTH]; - - if (!table->s->next_number_key_offset) - { // Autoincrement at key-start - ha_maria::info(HA_STATUS_AUTO); - *first_value= stats.auto_increment_value; - /* Maria has only table-level lock for now, so reserves to +inf */ - *nb_reserved_values= ULONGLONG_MAX; - return; - } - - /* it's safe to call the following if bulk_insert isn't on */ - maria_flush_bulk_insert(file, table->s->next_number_index); - - (void) extra(HA_EXTRA_KEYREAD); - key_copy(key, table->record[0], - table->key_info + table->s->next_number_index, - table->s->next_number_key_offset); - error= maria_rkey(file, table->record[1], (int) table->s->next_number_index, - key, table->s->next_number_key_offset, HA_READ_PREFIX_LAST); - if (error) - nr= 1; - else - { - /* Get data from record[1] */ - nr= ((ulonglong) table->next_number_field-> - val_int_offset(table->s->rec_buff_length) + 1); - } - extra(HA_EXTRA_NO_KEYREAD); - *first_value= nr; - /* - MySQL needs to call us for next row: assume we are inserting ("a",null) - here, we return 3, and next this statement will want to insert ("b",null): - there is no reason why ("b",3+1) would be the good row to insert: maybe it - already exists, maybe 3+1 is too large... - */ - *nb_reserved_values= 1; -} - - -/* - Find out how many rows there is in the given range - - SYNOPSIS - records_in_range() - inx Index to use - min_key Start of range. Null pointer if from first key - max_key End of range. Null pointer if to last key - - NOTES - min_key.flag can have one of the following values: - HA_READ_KEY_EXACT Include the key in the range - HA_READ_AFTER_KEY Don't include key in range - - max_key.flag can have one of the following values: - HA_READ_BEFORE_KEY Don't include key in range - HA_READ_AFTER_KEY Include all 'end_key' values in the range - - RETURN - HA_POS_ERROR Something is wrong with the index tree. - 0 There is no matching keys in the given range - number > 0 There is approximately 'number' matching rows in - the range. -*/ - -ha_rows ha_maria::records_in_range(uint inx, key_range *min_key, - key_range *max_key) -{ - return (ha_rows) maria_records_in_range(file, (int) inx, min_key, max_key); -} - - -int ha_maria::ft_read(byte * buf) -{ - int error; - - if (!ft_handler) - return -1; - - thread_safe_increment(table->in_use->status_var.ha_read_next_count, - &LOCK_status); // why ? - - error= ft_handler->please->read_next(ft_handler, (char*) buf); - - table->status= error ? STATUS_NOT_FOUND : 0; - return error; -} - - -uint ha_maria::checksum() const -{ - return (uint) file->state->checksum; -} - - -bool ha_maria::check_if_incompatible_data(HA_CREATE_INFO *info, - uint table_changes) -{ - uint options= table->s->db_options_in_use; - - if (info->auto_increment_value != stats.auto_increment_value || - info->data_file_name != data_file_name || - info->index_file_name != index_file_name || - table_changes == IS_EQUAL_NO || - table_changes & IS_EQUAL_PACK_LENGTH) // Not implemented yet - return COMPATIBLE_DATA_NO; - - if ((options & (HA_OPTION_PACK_RECORD | HA_OPTION_CHECKSUM | - HA_OPTION_DELAY_KEY_WRITE)) != - (info->table_options & (HA_OPTION_PACK_RECORD | HA_OPTION_CHECKSUM | - HA_OPTION_DELAY_KEY_WRITE))) - return COMPATIBLE_DATA_NO; - return COMPATIBLE_DATA_YES; -} - -handlerton maria_hton; - -static int ha_maria_init() -{ - maria_hton.state=SHOW_OPTION_YES; - maria_hton.db_type=DB_TYPE_MARIA; - maria_hton.create=maria_create_handler; - maria_hton.panic=maria_panic; - maria_hton.flags=HTON_CAN_RECREATE; - return test(maria_init()); -} - -struct st_mysql_storage_engine maria_storage_engine= -{ MYSQL_HANDLERTON_INTERFACE_VERSION, &maria_hton }; - -mysql_declare_plugin(maria) -{ - MYSQL_STORAGE_ENGINE_PLUGIN, - &maria_storage_engine, - "Maria", - "MySQL AB", - "Traditional transactional MySQL tables", - ha_maria_init, /* Plugin Init */ - NULL, /* Plugin Deinit */ - 0x0100, /* 1.0 */ - 0 -} -mysql_declare_plugin_end; diff --git a/sql/ha_maria.h b/sql/ha_maria.h deleted file mode 100644 index cdd2305f6a1..00000000000 --- a/sql/ha_maria.h +++ /dev/null @@ -1,145 +0,0 @@ -/* Copyright (C) 2006,2004 MySQL AB & MySQL Finland AB & TCX DataKonsult AB - - 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; either version 2 of the License, or - (at your option) any later version. - - 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ - - -#ifdef USE_PRAGMA_INTERFACE -#pragma interface /* gcc class implementation */ -#endif - -/* class for the the maria handler */ - -#include - -#define HA_RECOVER_NONE 0 /* No automatic recover */ -#define HA_RECOVER_DEFAULT 1 /* Automatic recover active */ -#define HA_RECOVER_BACKUP 2 /* Make a backupfile on recover */ -#define HA_RECOVER_FORCE 4 /* Recover even if we loose rows */ -#define HA_RECOVER_QUICK 8 /* Don't check rows in data file */ - -extern ulong maria_sort_buffer_size; -extern TYPELIB maria_recover_typelib; -extern ulong maria_recover_options; - -class ha_maria :public handler -{ - MARIA_HA *file; - ulong int_table_flags; - char *data_file_name, *index_file_name; - bool can_enable_indexes; - int repair(THD * thd, HA_CHECK ¶m, bool optimize); - -public: - ha_maria(TABLE_SHARE * table_arg); - ~ha_maria() - {} - const char *table_type() const - { return "MARIA"; } - const char *index_type(uint key_number); - const char **bas_ext() const; - ulonglong table_flags() const - { return int_table_flags; } - ulong index_flags(uint inx, uint part, bool all_parts) const - { - return ((table_share->key_info[inx].algorithm == HA_KEY_ALG_FULLTEXT) ? - 0 : HA_READ_NEXT | HA_READ_PREV | HA_READ_RANGE | - HA_READ_ORDER | HA_KEYREAD_ONLY); - } - uint max_supported_keys() const - { return MARIA_MAX_KEY; } - uint max_supported_key_length() const - { return HA_MAX_KEY_LENGTH; } - uint max_supported_key_part_length() const - { return HA_MAX_KEY_LENGTH; } - uint checksum() const; - - virtual bool check_if_locking_is_allowed(uint sql_command, - ulong type, TABLE * table, - uint count, - bool called_by_logger_thread); - int open(const char *name, int mode, uint test_if_locked); - int close(void); - int write_row(byte * buf); - int update_row(const byte * old_data, byte * new_data); - int delete_row(const byte * buf); - int index_read(byte * buf, const byte * key, - uint key_len, enum ha_rkey_function find_flag); - int index_read_idx(byte * buf, uint idx, const byte * key, - uint key_len, enum ha_rkey_function find_flag); - int index_read_last(byte * buf, const byte * key, uint key_len); - int index_next(byte * buf); - int index_prev(byte * buf); - int index_first(byte * buf); - int index_last(byte * buf); - int index_next_same(byte * buf, const byte * key, uint keylen); - int ft_init() - { - if (!ft_handler) - return 1; - ft_handler->please->reinit_search(ft_handler); - return 0; - } - FT_INFO *ft_init_ext(uint flags, uint inx, String * key) - { - return maria_ft_init_search(flags, file, inx, - (byte *) key->ptr(), key->length(), - key->charset(), table->record[0]); - } - int ft_read(byte * buf); - int rnd_init(bool scan); - int rnd_next(byte * buf); - int rnd_pos(byte * buf, byte * pos); - int restart_rnd_next(byte * buf, byte * pos); - void position(const byte * record); - void info(uint); - int extra(enum ha_extra_function operation); - int extra_opt(enum ha_extra_function operation, ulong cache_size); - int reset(void); - int external_lock(THD * thd, int lock_type); - int delete_all_rows(void); - int disable_indexes(uint mode); - int enable_indexes(uint mode); - int indexes_are_disabled(void); - void start_bulk_insert(ha_rows rows); - int end_bulk_insert(); - ha_rows records_in_range(uint inx, key_range * min_key, key_range * max_key); - void update_create_info(HA_CREATE_INFO * create_info); - int create(const char *name, TABLE * form, HA_CREATE_INFO * create_info); - THR_LOCK_DATA **store_lock(THD * thd, THR_LOCK_DATA ** to, - enum thr_lock_type lock_type); - virtual void get_auto_increment(ulonglong offset, ulonglong increment, - ulonglong nb_desired_values, - ulonglong *first_value, - ulonglong *nb_reserved_values); - int rename_table(const char *from, const char *to); - int delete_table(const char *name); - int check(THD * thd, HA_CHECK_OPT * check_opt); - int analyze(THD * thd, HA_CHECK_OPT * check_opt); - int repair(THD * thd, HA_CHECK_OPT * check_opt); - bool check_and_repair(THD * thd); - bool is_crashed() const; - bool auto_repair() const - { return maria_recover_options != 0; } - int optimize(THD * thd, HA_CHECK_OPT * check_opt); - int restore(THD * thd, HA_CHECK_OPT * check_opt); - int backup(THD * thd, HA_CHECK_OPT * check_opt); - int assign_to_keycache(THD * thd, HA_CHECK_OPT * check_opt); - int preload_keys(THD * thd, HA_CHECK_OPT * check_opt); - bool check_if_incompatible_data(HA_CREATE_INFO * info, uint table_changes); -#ifdef HAVE_REPLICATION - int dump(THD * thd, int fd); - int net_read_dump(NET * net); -#endif -}; diff --git a/sql/mysqld.cc b/sql/mysqld.cc index a666bec32d7..6465759af77 100644 --- a/sql/mysqld.cc +++ b/sql/mysqld.cc @@ -28,7 +28,7 @@ #include "ha_myisam.h" #ifdef WITH_MARIA_STORAGE_ENGINE -#include "ha_maria.h" +#include "../storage/maria/ha_maria.h" #endif #ifdef HAVE_ROW_BASED_REPLICATION diff --git a/storage/maria/Makefile.am b/storage/maria/Makefile.am index 15a5771b5d2..0271e29461e 100644 --- a/storage/maria/Makefile.am +++ b/storage/maria/Makefile.am @@ -53,7 +53,7 @@ libmaria_a_SOURCES = ma_init.c ma_open.c ma_extra.c ma_info.c ma_rkey.c \ ma_delete_table.c ma_rename.c ma_check.c \ ma_keycache.c ma_preload.c ma_ft_parser.c \ ma_ft_update.c ma_ft_boolean_search.c \ - ma_ft_nlq_search.c ft_maria.c ma_sort.c \ + ma_ft_nlq_search.c ma_sort.c \ ma_rt_index.c ma_rt_key.c ma_rt_mbr.c ma_rt_split.c \ ma_sp_key.c \ ma_control_file.c diff --git a/storage/maria/ft_maria.c b/storage/maria/ft_maria.c deleted file mode 100644 index 7104c6704ba..00000000000 --- a/storage/maria/ft_maria.c +++ /dev/null @@ -1,49 +0,0 @@ -/* Copyright (C) 2006 MySQL AB & MySQL Finland AB & TCX DataKonsult AB - - 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; either version 2 of the License, or - (at your option) any later version. - - 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ - -/* Written by Sergei A. Golubchik, who has a shared copyright to this code */ - -/* - This function is for interface functions between fulltext and maria -*/ - -#include "ma_ftdefs.h" - -FT_INFO *maria_ft_init_search(uint flags, void *info, uint keynr, - byte *query, uint query_len, CHARSET_INFO *cs, - byte *record) -{ - FT_INFO *res; - if (flags & FT_BOOL) - res= maria_ft_init_boolean_search((MARIA_HA *) info, keynr, query, - query_len, cs); - else - res= maria_ft_init_nlq_search((MARIA_HA *) info, keynr, query, query_len, - flags, record); - return res; -} - -const struct _ft_vft _ma_ft_vft_nlq = { - maria_ft_nlq_read_next, maria_ft_nlq_find_relevance, - maria_ft_nlq_close_search, maria_ft_nlq_get_relevance, - maria_ft_nlq_reinit_search -}; -const struct _ft_vft _ma_ft_vft_boolean = { - maria_ft_boolean_read_next, maria_ft_boolean_find_relevance, - maria_ft_boolean_close_search, maria_ft_boolean_get_relevance, - maria_ft_boolean_reinit_search -}; - diff --git a/storage/maria/ha_maria.cc b/storage/maria/ha_maria.cc new file mode 100644 index 00000000000..29718f1493e --- /dev/null +++ b/storage/maria/ha_maria.cc @@ -0,0 +1,1836 @@ +/* Copyright (C) 2006,2004 MySQL AB & MySQL Finland AB & TCX DataKonsult AB + + 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; either version 2 of the License, or + (at your option) any later version. + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ + + +#ifdef USE_PRAGMA_IMPLEMENTATION +#pragma implementation // gcc: Class implementation +#endif + +#include "mysql_priv.h" +#include +#include +#include "ha_maria.h" +#ifndef MASTER +#include "../srclib/maria/maria_def.h" +#else +#include "../storage/maria/maria_def.h" +#include "../storage/maria/ma_rt_index.h" +#endif + +#include + +ulong maria_recover_options= HA_RECOVER_NONE; + +/* bits in maria_recover_options */ +const char *maria_recover_names[]= +{ + "DEFAULT", "BACKUP", "FORCE", "QUICK", NullS +}; +TYPELIB maria_recover_typelib= +{ + array_elements(maria_recover_names) - 1, "", + maria_recover_names, NULL +}; + +const char *maria_stats_method_names[]= +{ + "nulls_unequal", "nulls_equal", + "nulls_ignored", NullS +}; +TYPELIB maria_stats_method_typelib= +{ + array_elements(maria_stats_method_names) - 1, "", + maria_stats_method_names, NULL +}; + + +/***************************************************************************** +** MARIA tables +*****************************************************************************/ + +static handler *maria_create_handler(TABLE_SHARE * table, MEM_ROOT *mem_root) +{ + return new (mem_root) ha_maria(table); +} + + +// collect errors printed by maria_check routines + +static void _ma_check_print_msg(HA_CHECK *param, const char *msg_type, + const char *fmt, va_list args) +{ + THD *thd= (THD *) param->thd; + Protocol *protocol= thd->protocol; + uint length, msg_length; + char msgbuf[MARIA_MAX_MSG_BUF]; + char name[NAME_LEN * 2 + 2]; + + msg_length= my_vsnprintf(msgbuf, sizeof(msgbuf), fmt, args); + msgbuf[sizeof(msgbuf) - 1]= 0; // healthy paranoia + + DBUG_PRINT(msg_type, ("message: %s", msgbuf)); + + if (!thd->vio_ok()) + { + sql_print_error(msgbuf); + return; + } + + if (param->testflag & + (T_CREATE_MISSING_KEYS | T_SAFE_REPAIR | T_AUTO_REPAIR)) + { + my_message(ER_NOT_KEYFILE, msgbuf, MYF(MY_WME)); + return; + } + length= (uint) (strxmov(name, param->db_name, ".", param->table_name, + NullS) - name); + protocol->prepare_for_resend(); + protocol->store(name, length, system_charset_info); + protocol->store(param->op_name, system_charset_info); + protocol->store(msg_type, system_charset_info); + protocol->store(msgbuf, msg_length, system_charset_info); + if (protocol->write()) + sql_print_error("Failed on my_net_write, writing to stderr instead: %s\n", + msgbuf); + return; +} + +extern "C" { + +volatile int *_ma_killed_ptr(HA_CHECK *param) +{ + /* In theory Unsafe conversion, but should be ok for now */ + return (int*) &(((THD *) (param->thd))->killed); +} + + +void _ma_check_print_error(HA_CHECK *param, const char *fmt, ...) +{ + param->error_printed |= 1; + param->out_flag |= O_DATA_LOST; + va_list args; + va_start(args, fmt); + _ma_check_print_msg(param, "error", fmt, args); + va_end(args); +} + + +void _ma_check_print_info(HA_CHECK *param, const char *fmt, ...) +{ + va_list args; + va_start(args, fmt); + _ma_check_print_msg(param, "info", fmt, args); + va_end(args); +} + + +void _ma_check_print_warning(HA_CHECK *param, const char *fmt, ...) +{ + param->warning_printed= 1; + param->out_flag |= O_DATA_LOST; + va_list args; + va_start(args, fmt); + _ma_check_print_msg(param, "warning", fmt, args); + va_end(args); +} + +} + + +ha_maria::ha_maria(TABLE_SHARE *table_arg): +handler(&maria_hton, table_arg), file(0), +int_table_flags(HA_NULL_IN_KEY | HA_CAN_FULLTEXT | HA_CAN_SQL_HANDLER | + HA_DUPLICATE_POS | HA_CAN_INDEX_BLOBS | HA_AUTO_PART_KEY | + HA_FILE_BASED | HA_CAN_GEOMETRY | HA_NO_TRANSACTIONS | + HA_CAN_INSERT_DELAYED | HA_CAN_BIT_FIELD | HA_CAN_RTREEKEYS | + HA_HAS_RECORDS | HA_STATS_RECORDS_IS_EXACT), +can_enable_indexes(1) +{} + + +static const char *ha_maria_exts[]= +{ + MARIA_NAME_IEXT, + MARIA_NAME_DEXT, + NullS +}; + + +const char **ha_maria::bas_ext() const +{ + return ha_maria_exts; +} + + +const char *ha_maria::index_type(uint key_number) +{ + return ((table->key_info[key_number].flags & HA_FULLTEXT) ? + "FULLTEXT" : + (table->key_info[key_number].flags & HA_SPATIAL) ? + "SPATIAL" : + (table->key_info[key_number].algorithm == HA_KEY_ALG_RTREE) ? + "RTREE" : "BTREE"); +} + + +#ifdef HAVE_REPLICATION +int ha_maria::net_read_dump(NET * net) +{ + int data_fd= file->dfile; + int error= 0; + + my_seek(data_fd, 0L, MY_SEEK_SET, MYF(MY_WME)); + for (;;) + { + ulong packet_len= my_net_read(net); + if (!packet_len) + break; // end of file + if (packet_len == packet_error) + { + sql_print_error("ha_maria::net_read_dump - read error "); + error= -1; + goto err; + } + if (my_write(data_fd, (byte *) net->read_pos, (uint) packet_len, + MYF(MY_WME | MY_FNABP))) + { + error= errno; + goto err; + } + } +err: + return error; +} + + +int ha_maria::dump(THD * thd, int fd) +{ + MARIA_SHARE *share= file->s; + NET *net= &thd->net; + uint blocksize= share->blocksize; + my_off_t bytes_to_read= share->state.state.data_file_length; + int data_fd= file->dfile; + byte *buf= (byte *) my_malloc(blocksize, MYF(MY_WME)); + if (!buf) + return ENOMEM; + + int error= 0; + my_seek(data_fd, 0L, MY_SEEK_SET, MYF(MY_WME)); + for (; bytes_to_read > 0;) + { + uint bytes= my_read(data_fd, buf, blocksize, MYF(MY_WME)); + if (bytes == MY_FILE_ERROR) + { + error= errno; + goto err; + } + + if (fd >= 0) + { + if (my_write(fd, buf, bytes, MYF(MY_WME | MY_FNABP))) + { + error= errno ? errno : EPIPE; + goto err; + } + } + else + { + if (my_net_write(net, (char*) buf, bytes)) + { + error= errno ? errno : EPIPE; + goto err; + } + } + bytes_to_read -= bytes; + } + + if (fd < 0) + { + if (my_net_write(net, "", 0)) + error= errno ? errno : EPIPE; + net_flush(net); + } + +err: + my_free((gptr) buf, MYF(0)); + return error; +} +#endif /* HAVE_REPLICATION */ + + +bool ha_maria::check_if_locking_is_allowed(uint sql_command, + ulong type, TABLE *table, + uint count, + bool called_by_logger_thread) +{ + /* + To be able to open and lock for reading system tables like 'mysql.proc', + when we already have some tables opened and locked, and avoid deadlocks + we have to disallow write-locking of these tables with any other tables. + */ + if (table->s->system_table && + table->reginfo.lock_type >= TL_WRITE_ALLOW_WRITE && count != 1) + { + my_error(ER_WRONG_LOCK_OF_SYSTEM_TABLE, MYF(0), table->s->db.str, + table->s->table_name.str); + return FALSE; + } + return TRUE; +} + + + /* Name is here without an extension */ + +int ha_maria::open(const char *name, int mode, uint test_if_locked) +{ + uint i; + if (!(file= maria_open(name, mode, test_if_locked | HA_OPEN_FROM_SQL_LAYER))) + return (my_errno ? my_errno : -1); + + if (test_if_locked & (HA_OPEN_IGNORE_IF_LOCKED | HA_OPEN_TMP_TABLE)) + VOID(maria_extra(file, HA_EXTRA_NO_WAIT_LOCK, 0)); + +#ifdef NOT_USED + if (!(test_if_locked & HA_OPEN_TMP_TABLE) && opt_maria_use_mmap) + VOID(maria_extra(file, HA_EXTRA_MMAP, 0)); +#endif + + info(HA_STATUS_NO_LOCK | HA_STATUS_VARIABLE | HA_STATUS_CONST); + if (!(test_if_locked & HA_OPEN_WAIT_IF_LOCKED)) + VOID(maria_extra(file, HA_EXTRA_WAIT_LOCK, 0)); + if (!table->s->db_record_offset) + int_table_flags |= HA_REC_NOT_IN_SEQ; + if (file->s->options & (HA_OPTION_CHECKSUM | HA_OPTION_COMPRESS_RECORD)) + int_table_flags |= HA_HAS_CHECKSUM; + + for (i= 0; i < table->s->keys; i++) + { + struct st_plugin_int *parser= table->key_info[i].parser; + if (table->key_info[i].flags & HA_USES_PARSER) + file->s->keyinfo[i].parser= + (struct st_mysql_ftparser *) parser->plugin->info; + table->key_info[i].block_size= file->s->keyinfo[i].block_length; + } + return (0); +} + + +int ha_maria::close(void) +{ + MARIA_HA *tmp= file; + file= 0; + return maria_close(tmp); +} + + +int ha_maria::write_row(byte * buf) +{ + statistic_increment(table->in_use->status_var.ha_write_count, &LOCK_status); + + /* If we have a timestamp column, update it to the current time */ + if (table->timestamp_field_type & TIMESTAMP_AUTO_SET_ON_INSERT) + table->timestamp_field->set_time(); + + /* + If we have an auto_increment column and we are writing a changed row + or a new row, then update the auto_increment value in the record. + */ + if (table->next_number_field && buf == table->record[0]) + update_auto_increment(); + return maria_write(file, buf); +} + + +int ha_maria::check(THD * thd, HA_CHECK_OPT * check_opt) +{ + if (!file) + return HA_ADMIN_INTERNAL_ERROR; + int error; + HA_CHECK param; + MARIA_SHARE *share= file->s; + const char *old_proc_info= thd->proc_info; + + thd->proc_info= "Checking table"; + mariachk_init(¶m); + param.thd= thd; + param.op_name= "check"; + param.db_name= table->s->db.str; + param.table_name= table->alias; + param.testflag= check_opt->flags | T_CHECK | T_SILENT; + param.stats_method= (enum_handler_stats_method) thd->variables. + maria_stats_method; + + if (!(table->db_stat & HA_READ_ONLY)) + param.testflag |= T_STATISTICS; + param.using_global_keycache= 1; + + if (!maria_is_crashed(file) && + (((param.testflag & T_CHECK_ONLY_CHANGED) && + !(share->state.changed & (STATE_CHANGED | STATE_CRASHED | + STATE_CRASHED_ON_REPAIR)) && + share->state.open_count == 0) || + ((param.testflag & T_FAST) && (share->state.open_count == + (uint) (share->global_changed ? 1 : + 0))))) + return HA_ADMIN_ALREADY_DONE; + + error= maria_chk_status(¶m, file); // Not fatal + error= maria_chk_size(¶m, file); + if (!error) + error |= maria_chk_del(¶m, file, param.testflag); + if (!error) + error= maria_chk_key(¶m, file); + if (!error) + { + if ((!(param.testflag & T_QUICK) && + ((share->options & + (HA_OPTION_PACK_RECORD | HA_OPTION_COMPRESS_RECORD)) || + (param.testflag & (T_EXTEND | T_MEDIUM)))) || maria_is_crashed(file)) + { + uint old_testflag= param.testflag; + param.testflag |= T_MEDIUM; + if (!(error= init_io_cache(¶m.read_cache, file->dfile, + my_default_record_cache_size, READ_CACHE, + share->pack.header_length, 1, MYF(MY_WME)))) + { + error= maria_chk_data_link(¶m, file, param.testflag & T_EXTEND); + end_io_cache(&(param.read_cache)); + } + param.testflag= old_testflag; + } + } + if (!error) + { + if ((share->state.changed & (STATE_CHANGED | + STATE_CRASHED_ON_REPAIR | + STATE_CRASHED | STATE_NOT_ANALYZED)) || + (param.testflag & T_STATISTICS) || maria_is_crashed(file)) + { + file->update |= HA_STATE_CHANGED | HA_STATE_ROW_CHANGED; + pthread_mutex_lock(&share->intern_lock); + share->state.changed &= ~(STATE_CHANGED | STATE_CRASHED | + STATE_CRASHED_ON_REPAIR); + if (!(table->db_stat & HA_READ_ONLY)) + error= maria_update_state_info(¶m, file, UPDATE_TIME | UPDATE_OPEN_COUNT | + UPDATE_STAT); + pthread_mutex_unlock(&share->intern_lock); + info(HA_STATUS_NO_LOCK | HA_STATUS_TIME | HA_STATUS_VARIABLE | + HA_STATUS_CONST); + } + } + else if (!maria_is_crashed(file) && !thd->killed) + { + maria_mark_crashed(file); + file->update |= HA_STATE_CHANGED | HA_STATE_ROW_CHANGED; + } + + thd->proc_info= old_proc_info; + return error ? HA_ADMIN_CORRUPT : HA_ADMIN_OK; +} + + +/* + Analyze the key distribution in the table + As the table may be only locked for read, we have to take into account that + two threads may do an analyze at the same time! +*/ + +int ha_maria::analyze(THD *thd, HA_CHECK_OPT * check_opt) +{ + int error= 0; + HA_CHECK param; + MARIA_SHARE *share= file->s; + + mariachk_init(¶m); + param.thd= thd; + param.op_name= "analyze"; + param.db_name= table->s->db.str; + param.table_name= table->alias; + param.testflag= (T_FAST | T_CHECK | T_SILENT | T_STATISTICS | + T_DONT_CHECK_CHECKSUM); + param.using_global_keycache= 1; + param.stats_method= (enum_handler_stats_method) thd->variables. + maria_stats_method; + + if (!(share->state.changed & STATE_NOT_ANALYZED)) + return HA_ADMIN_ALREADY_DONE; + + error= maria_chk_key(¶m, file); + if (!error) + { + pthread_mutex_lock(&share->intern_lock); + error= maria_update_state_info(¶m, file, UPDATE_STAT); + pthread_mutex_unlock(&share->intern_lock); + } + else if (!maria_is_crashed(file) && !thd->killed) + maria_mark_crashed(file); + return error ? HA_ADMIN_CORRUPT : HA_ADMIN_OK; +} + + +int ha_maria::restore(THD * thd, HA_CHECK_OPT *check_opt) +{ + HA_CHECK_OPT tmp_check_opt; + char *backup_dir= thd->lex->backup_dir; + char src_path[FN_REFLEN], dst_path[FN_REFLEN]; + const char *table_name= table->s->table_name.str; + int error; + const char *errmsg; + DBUG_ENTER("restore"); + + if (fn_format_relative_to_data_home(src_path, table_name, backup_dir, + MARIA_NAME_DEXT)) + DBUG_RETURN(HA_ADMIN_INVALID); + + strxmov(dst_path, table->s->normalized_path.str, MARIA_NAME_DEXT, NullS); + if (my_copy(src_path, dst_path, MYF(MY_WME))) + { + error= HA_ADMIN_FAILED; + errmsg= "Failed in my_copy (Error %d)"; + goto err; + } + + tmp_check_opt.init(); + tmp_check_opt.flags |= T_VERY_SILENT | T_CALC_CHECKSUM | T_QUICK; + DBUG_RETURN(repair(thd, &tmp_check_opt)); + +err: + { + HA_CHECK param; + mariachk_init(¶m); + param.thd= thd; + param.op_name= "restore"; + param.db_name= table->s->db.str; + param.table_name= table->s->table_name.str; + param.testflag= 0; + _ma_check_print_error(¶m, errmsg, my_errno); + DBUG_RETURN(error); + } +} + + +int ha_maria::backup(THD * thd, HA_CHECK_OPT *check_opt) +{ + char *backup_dir= thd->lex->backup_dir; + char src_path[FN_REFLEN], dst_path[FN_REFLEN]; + const char *table_name= table->s->table_name.str; + int error; + const char *errmsg; + DBUG_ENTER("ha_maria::backup"); + + if (fn_format_relative_to_data_home(dst_path, table_name, backup_dir, + reg_ext)) + { + errmsg= "Failed in fn_format() for .frm file (errno: %d)"; + error= HA_ADMIN_INVALID; + goto err; + } + + strxmov(src_path, table->s->normalized_path.str, reg_ext, NullS); + if (my_copy(src_path, dst_path, + MYF(MY_WME | MY_HOLD_ORIGINAL_MODES | MY_DONT_OVERWRITE_FILE))) + { + error= HA_ADMIN_FAILED; + errmsg= "Failed copying .frm file (errno: %d)"; + goto err; + } + + /* Change extension */ + if (fn_format_relative_to_data_home(dst_path, table_name, backup_dir, + MARIA_NAME_DEXT)) + { + errmsg= "Failed in fn_format() for .MYD file (errno: %d)"; + error= HA_ADMIN_INVALID; + goto err; + } + + strxmov(src_path, table->s->normalized_path.str, MARIA_NAME_DEXT, NullS); + if (my_copy(src_path, dst_path, + MYF(MY_WME | MY_HOLD_ORIGINAL_MODES | MY_DONT_OVERWRITE_FILE))) + { + errmsg= "Failed copying .MYD file (errno: %d)"; + error= HA_ADMIN_FAILED; + goto err; + } + DBUG_RETURN(HA_ADMIN_OK); + +err: + { + HA_CHECK param; + mariachk_init(¶m); + param.thd= thd; + param.op_name= "backup"; + param.db_name= table->s->db.str; + param.table_name= table->s->table_name.str; + param.testflag= 0; + _ma_check_print_error(¶m, errmsg, my_errno); + DBUG_RETURN(error); + } +} + + +int ha_maria::repair(THD * thd, HA_CHECK_OPT *check_opt) +{ + int error; + HA_CHECK param; + ha_rows start_records; + + if (!file) + return HA_ADMIN_INTERNAL_ERROR; + + mariachk_init(¶m); + param.thd= thd; + param.op_name= "repair"; + param.testflag= ((check_opt->flags & ~(T_EXTEND)) | + T_SILENT | T_FORCE_CREATE | T_CALC_CHECKSUM | + (check_opt->flags & T_EXTEND ? T_REP : T_REP_BY_SORT)); + param.sort_buffer_length= check_opt->sort_buffer_size; + start_records= file->state->records; + while ((error= repair(thd, param, 0)) && param.retry_repair) + { + param.retry_repair= 0; + if (test_all_bits(param.testflag, + (uint) (T_RETRY_WITHOUT_QUICK | T_QUICK))) + { + param.testflag &= ~T_RETRY_WITHOUT_QUICK; + sql_print_information("Retrying repair of: '%s' without quick", + table->s->path); + continue; + } + param.testflag &= ~T_QUICK; + if ((param.testflag & T_REP_BY_SORT)) + { + param.testflag= (param.testflag & ~T_REP_BY_SORT) | T_REP; + sql_print_information("Retrying repair of: '%s' with keycache", + table->s->path); + continue; + } + break; + } + if (!error && start_records != file->state->records && + !(check_opt->flags & T_VERY_SILENT)) + { + char llbuff[22], llbuff2[22]; + sql_print_information("Found %s of %s rows when repairing '%s'", + llstr(file->state->records, llbuff), + llstr(start_records, llbuff2), table->s->path); + } + return error; +} + +int ha_maria::optimize(THD * thd, HA_CHECK_OPT *check_opt) +{ + int error; + if (!file) + return HA_ADMIN_INTERNAL_ERROR; + HA_CHECK param; + + mariachk_init(¶m); + param.thd= thd; + param.op_name= "optimize"; + param.testflag= (check_opt->flags | T_SILENT | T_FORCE_CREATE | + T_REP_BY_SORT | T_STATISTICS | T_SORT_INDEX); + param.sort_buffer_length= check_opt->sort_buffer_size; + if ((error= repair(thd, param, 1)) && param.retry_repair) + { + sql_print_warning("Warning: Optimize table got errno %d, retrying", + my_errno); + param.testflag &= ~T_REP_BY_SORT; + error= repair(thd, param, 1); + } + return error; +} + + +int ha_maria::repair(THD *thd, HA_CHECK ¶m, bool optimize) +{ + int error= 0; + uint local_testflag= param.testflag; + bool optimize_done= !optimize, statistics_done= 0; + const char *old_proc_info= thd->proc_info; + char fixed_name[FN_REFLEN]; + MARIA_SHARE *share= file->s; + ha_rows rows= file->state->records; + DBUG_ENTER("ha_maria::repair"); + + param.db_name= table->s->db.str; + param.table_name= table->alias; + param.tmpfile_createflag= O_RDWR | O_TRUNC; + param.using_global_keycache= 1; + param.thd= thd; + param.tmpdir= &mysql_tmpdir_list; + param.out_flag= 0; + strmov(fixed_name, file->filename); + + // Don't lock tables if we have used LOCK TABLE + if (!thd->locked_tables && + maria_lock_database(file, table->s->tmp_table ? F_EXTRA_LCK : F_WRLCK)) + { + _ma_check_print_error(¶m, ER(ER_CANT_LOCK), my_errno); + DBUG_RETURN(HA_ADMIN_FAILED); + } + + if (!optimize || + ((file->state->del || share->state.split != file->state->records) && + (!(param.testflag & T_QUICK) || + !(share->state.changed & STATE_NOT_OPTIMIZED_KEYS)))) + { + ulonglong key_map= ((local_testflag & T_CREATE_MISSING_KEYS) ? + maria_get_mask_all_keys_active(share->base.keys) : + share->state.key_map); + uint testflag= param.testflag; + if (maria_test_if_sort_rep(file, file->state->records, key_map, 0) && + (local_testflag & T_REP_BY_SORT)) + { + local_testflag |= T_STATISTICS; + param.testflag |= T_STATISTICS; // We get this for free + statistics_done= 1; + if (thd->variables.maria_repair_threads > 1) + { + char buf[40]; + /* TODO: respect maria_repair_threads variable */ + my_snprintf(buf, 40, "Repair with %d threads", my_count_bits(key_map)); + thd->proc_info= buf; + error= maria_repair_parallel(¶m, file, fixed_name, + param.testflag & T_QUICK); + thd->proc_info= "Repair done"; // to reset proc_info, as + // it was pointing to local buffer + } + else + { + thd->proc_info= "Repair by sorting"; + error= maria_repair_by_sort(¶m, file, fixed_name, + param.testflag & T_QUICK); + } + } + else + { + thd->proc_info= "Repair with keycache"; + param.testflag &= ~T_REP_BY_SORT; + error= maria_repair(¶m, file, fixed_name, param.testflag & T_QUICK); + } + param.testflag= testflag; + optimize_done= 1; + } + if (!error) + { + if ((local_testflag & T_SORT_INDEX) && + (share->state.changed & STATE_NOT_SORTED_PAGES)) + { + optimize_done= 1; + thd->proc_info= "Sorting index"; + error= maria_sort_index(¶m, file, fixed_name); + } + if (!statistics_done && (local_testflag & T_STATISTICS)) + { + if (share->state.changed & STATE_NOT_ANALYZED) + { + optimize_done= 1; + thd->proc_info= "Analyzing"; + error= maria_chk_key(¶m, file); + } + else + local_testflag &= ~T_STATISTICS; // Don't update statistics + } + } + thd->proc_info= "Saving state"; + if (!error) + { + if ((share->state.changed & STATE_CHANGED) || maria_is_crashed(file)) + { + share->state.changed &= ~(STATE_CHANGED | STATE_CRASHED | + STATE_CRASHED_ON_REPAIR); + file->update |= HA_STATE_CHANGED | HA_STATE_ROW_CHANGED; + } + /* + the following 'if', thought conceptually wrong, + is a useful optimization nevertheless. + */ + if (file->state != &file->s->state.state) + file->s->state.state= *file->state; + if (file->s->base.auto_key) + _ma_update_auto_increment_key(¶m, file, 1); + if (optimize_done) + error= maria_update_state_info(¶m, file, + UPDATE_TIME | UPDATE_OPEN_COUNT | + (local_testflag & + T_STATISTICS ? UPDATE_STAT : 0)); + info(HA_STATUS_NO_LOCK | HA_STATUS_TIME | HA_STATUS_VARIABLE | + HA_STATUS_CONST); + if (rows != file->state->records && !(param.testflag & T_VERY_SILENT)) + { + char llbuff[22], llbuff2[22]; + _ma_check_print_warning(¶m, "Number of rows changed from %s to %s", + llstr(rows, llbuff), + llstr(file->state->records, llbuff2)); + } + } + else + { + maria_mark_crashed_on_repair(file); + file->update |= HA_STATE_CHANGED | HA_STATE_ROW_CHANGED; + maria_update_state_info(¶m, file, 0); + } + thd->proc_info= old_proc_info; + if (!thd->locked_tables) + maria_lock_database(file, F_UNLCK); + DBUG_RETURN(error ? HA_ADMIN_FAILED : + !optimize_done ? HA_ADMIN_ALREADY_DONE : HA_ADMIN_OK); +} + + +/* + Assign table indexes to a specific key cache. +*/ + +int ha_maria::assign_to_keycache(THD * thd, HA_CHECK_OPT *check_opt) +{ + KEY_CACHE *new_key_cache= check_opt->key_cache; + const char *errmsg= 0; + int error= HA_ADMIN_OK; + ulonglong map= ~(ulonglong) 0; + TABLE_LIST *table_list= table->pos_in_table_list; + DBUG_ENTER("ha_maria::assign_to_keycache"); + + /* Check validity of the index references */ + if (table_list->use_index) + { + /* We only come here when the user did specify an index map */ + key_map kmap; + if (get_key_map_from_key_list(&kmap, table, table_list->use_index)) + { + errmsg= thd->net.last_error; + error= HA_ADMIN_FAILED; + goto err; + } + map= kmap.to_ulonglong(); + } + + if ((error= maria_assign_to_key_cache(file, map, new_key_cache))) + { + char buf[STRING_BUFFER_USUAL_SIZE]; + my_snprintf(buf, sizeof(buf), + "Failed to flush to index file (errno: %d)", error); + errmsg= buf; + error= HA_ADMIN_CORRUPT; + } + +err: + if (error != HA_ADMIN_OK) + { + /* Send error to user */ + HA_CHECK param; + mariachk_init(¶m); + param.thd= thd; + param.op_name= "assign_to_keycache"; + param.db_name= table->s->db.str; + param.table_name= table->s->table_name.str; + param.testflag= 0; + _ma_check_print_error(¶m, errmsg); + } + DBUG_RETURN(error); +} + + +/* + Preload pages of the index file for a table into the key cache. +*/ + +int ha_maria::preload_keys(THD * thd, HA_CHECK_OPT *check_opt) +{ + int error; + const char *errmsg; + ulonglong map= ~(ulonglong) 0; + TABLE_LIST *table_list= table->pos_in_table_list; + my_bool ignore_leaves= table_list->ignore_leaves; + + DBUG_ENTER("ha_maria::preload_keys"); + + /* Check validity of the index references */ + if (table_list->use_index) + { + key_map kmap; + get_key_map_from_key_list(&kmap, table, table_list->use_index); + if (kmap.is_set_all()) + { + errmsg= thd->net.last_error; + error= HA_ADMIN_FAILED; + goto err; + } + if (!kmap.is_clear_all()) + map= kmap.to_ulonglong(); + } + + maria_extra(file, HA_EXTRA_PRELOAD_BUFFER_SIZE, + (void*) &thd->variables.preload_buff_size); + + if ((error= maria_preload(file, map, ignore_leaves))) + { + switch (error) { + case HA_ERR_NON_UNIQUE_BLOCK_SIZE: + errmsg= "Indexes use different block sizes"; + break; + case HA_ERR_OUT_OF_MEM: + errmsg= "Failed to allocate buffer"; + break; + default: + char buf[ERRMSGSIZE + 20]; + my_snprintf(buf, ERRMSGSIZE, + "Failed to read from index file (errno: %d)", my_errno); + errmsg= buf; + } + error= HA_ADMIN_FAILED; + goto err; + } + + DBUG_RETURN(HA_ADMIN_OK); + +err: + { + HA_CHECK param; + mariachk_init(¶m); + param.thd= thd; + param.op_name= "preload_keys"; + param.db_name= table->s->db.str; + param.table_name= table->s->table_name.str; + param.testflag= 0; + _ma_check_print_error(¶m, errmsg); + DBUG_RETURN(error); + } +} + + +/* + Disable indexes, making it persistent if requested. + + SYNOPSIS + disable_indexes() + mode mode of operation: + HA_KEY_SWITCH_NONUNIQ disable all non-unique keys + HA_KEY_SWITCH_ALL disable all keys + HA_KEY_SWITCH_NONUNIQ_SAVE dis. non-uni. and make persistent + HA_KEY_SWITCH_ALL_SAVE dis. all keys and make persistent + + IMPLEMENTATION + HA_KEY_SWITCH_NONUNIQ is not implemented. + HA_KEY_SWITCH_ALL_SAVE is not implemented. + + RETURN + 0 ok + HA_ERR_WRONG_COMMAND mode not implemented. +*/ + +int ha_maria::disable_indexes(uint mode) +{ + int error; + + if (mode == HA_KEY_SWITCH_ALL) + { + /* call a storage engine function to switch the key map */ + error= maria_disable_indexes(file); + } + else if (mode == HA_KEY_SWITCH_NONUNIQ_SAVE) + { + maria_extra(file, HA_EXTRA_NO_KEYS, 0); + info(HA_STATUS_CONST); // Read new key info + error= 0; + } + else + { + /* mode not implemented */ + error= HA_ERR_WRONG_COMMAND; + } + return error; +} + + +/* + Enable indexes, making it persistent if requested. + + SYNOPSIS + enable_indexes() + mode mode of operation: + HA_KEY_SWITCH_NONUNIQ enable all non-unique keys + HA_KEY_SWITCH_ALL enable all keys + HA_KEY_SWITCH_NONUNIQ_SAVE en. non-uni. and make persistent + HA_KEY_SWITCH_ALL_SAVE en. all keys and make persistent + + DESCRIPTION + Enable indexes, which might have been disabled by disable_index() before. + The modes without _SAVE work only if both data and indexes are empty, + since the MARIA repair would enable them persistently. + To be sure in these cases, call handler::delete_all_rows() before. + + IMPLEMENTATION + HA_KEY_SWITCH_NONUNIQ is not implemented. + HA_KEY_SWITCH_ALL_SAVE is not implemented. + + RETURN + 0 ok + !=0 Error, among others: + HA_ERR_CRASHED data or index is non-empty. Delete all rows and retry. + HA_ERR_WRONG_COMMAND mode not implemented. +*/ + +int ha_maria::enable_indexes(uint mode) +{ + int error; + + if (maria_is_all_keys_active(file->s->state.key_map, file->s->base.keys)) + { + /* All indexes are enabled already. */ + return 0; + } + + if (mode == HA_KEY_SWITCH_ALL) + { + error= maria_enable_indexes(file); + /* + Do not try to repair on error, + as this could make the enabled state persistent, + but mode==HA_KEY_SWITCH_ALL forbids it. + */ + } + else if (mode == HA_KEY_SWITCH_NONUNIQ_SAVE) + { + THD *thd= current_thd; + HA_CHECK param; + const char *save_proc_info= thd->proc_info; + thd->proc_info= "Creating index"; + mariachk_init(¶m); + param.op_name= "recreating_index"; + param.testflag= (T_SILENT | T_REP_BY_SORT | T_QUICK | + T_CREATE_MISSING_KEYS); + param.myf_rw &= ~MY_WAIT_IF_FULL; + param.sort_buffer_length= thd->variables.maria_sort_buff_size; + param.stats_method= (enum_handler_stats_method) thd->variables. + maria_stats_method; + param.tmpdir= &mysql_tmpdir_list; + if ((error= (repair(thd, param, 0) != HA_ADMIN_OK)) && param.retry_repair) + { + sql_print_warning("Warning: Enabling keys got errno %d, retrying", + my_errno); + /* Repairing by sort failed. Now try standard repair method. */ + param.testflag &= ~(T_REP_BY_SORT | T_QUICK); + error= (repair(thd, param, 0) != HA_ADMIN_OK); + /* + If the standard repair succeeded, clear all error messages which + might have been set by the first repair. They can still be seen + with SHOW WARNINGS then. + */ + if (!error) + thd->clear_error(); + } + info(HA_STATUS_CONST); + thd->proc_info= save_proc_info; + } + else + { + /* mode not implemented */ + error= HA_ERR_WRONG_COMMAND; + } + return error; +} + + +/* + Test if indexes are disabled. + + + SYNOPSIS + indexes_are_disabled() + no parameters + + + RETURN + 0 indexes are not disabled + 1 all indexes are disabled + [2 non-unique indexes are disabled - NOT YET IMPLEMENTED] +*/ + +int ha_maria::indexes_are_disabled(void) +{ + return maria_indexes_are_disabled(file); +} + + +/* + prepare for a many-rows insert operation + e.g. - disable indexes (if they can be recreated fast) or + activate special bulk-insert optimizations + + SYNOPSIS + start_bulk_insert(rows) + rows Rows to be inserted + 0 if we don't know + + NOTICE + Do not forget to call end_bulk_insert() later! +*/ + +void ha_maria::start_bulk_insert(ha_rows rows) +{ + DBUG_ENTER("ha_maria::start_bulk_insert"); + THD *thd= current_thd; + ulong size= min(thd->variables.read_buff_size, + table->s->avg_row_length * rows); + DBUG_PRINT("info", ("start_bulk_insert: rows %lu size %lu", + (ulong) rows, size)); + + /* don't enable row cache if too few rows */ + if (!rows || (rows > MARIA_MIN_ROWS_TO_USE_WRITE_CACHE)) + maria_extra(file, HA_EXTRA_WRITE_CACHE, (void*) &size); + + can_enable_indexes= maria_is_all_keys_active(file->s->state.key_map, + file->s->base.keys); + + if (!(specialflag & SPECIAL_SAFE_MODE)) + { + /* + Only disable old index if the table was empty and we are inserting + a lot of rows. + We should not do this for only a few rows as this is slower and + we don't want to update the key statistics based of only a few rows. + */ + if (file->state->records == 0 && can_enable_indexes && + (!rows || rows >= MARIA_MIN_ROWS_TO_DISABLE_INDEXES)) + maria_disable_non_unique_index(file, rows); + else + if (!file->bulk_insert && + (!rows || rows >= MARIA_MIN_ROWS_TO_USE_BULK_INSERT)) + { + maria_init_bulk_insert(file, thd->variables.bulk_insert_buff_size, rows); + } + } + DBUG_VOID_RETURN; +} + + +/* + end special bulk-insert optimizations, + which have been activated by start_bulk_insert(). + + SYNOPSIS + end_bulk_insert() + no arguments + + RETURN + 0 OK + != 0 Error +*/ + +int ha_maria::end_bulk_insert() +{ + maria_end_bulk_insert(file); + int err= maria_extra(file, HA_EXTRA_NO_CACHE, 0); + return err ? err : can_enable_indexes ? + enable_indexes(HA_KEY_SWITCH_NONUNIQ_SAVE) : 0; +} + + +bool ha_maria::check_and_repair(THD *thd) +{ + int error= 0; + int marked_crashed; + char *old_query; + uint old_query_length; + HA_CHECK_OPT check_opt; + DBUG_ENTER("ha_maria::check_and_repair"); + + check_opt.init(); + check_opt.flags= T_MEDIUM | T_AUTO_REPAIR; + // Don't use quick if deleted rows + if (!file->state->del && (maria_recover_options & HA_RECOVER_QUICK)) + check_opt.flags |= T_QUICK; + sql_print_warning("Checking table: '%s'", table->s->path); + + old_query= thd->query; + old_query_length= thd->query_length; + pthread_mutex_lock(&LOCK_thread_count); + thd->query= table->s->table_name.str; + thd->query_length= table->s->table_name.length; + pthread_mutex_unlock(&LOCK_thread_count); + + if ((marked_crashed= maria_is_crashed(file)) || check(thd, &check_opt)) + { + sql_print_warning("Recovering table: '%s'", table->s->path); + check_opt.flags= + ((maria_recover_options & HA_RECOVER_BACKUP ? T_BACKUP_DATA : 0) | + (marked_crashed ? 0 : T_QUICK) | + (maria_recover_options & HA_RECOVER_FORCE ? 0 : T_SAFE_REPAIR) | + T_AUTO_REPAIR); + if (repair(thd, &check_opt)) + error= 1; + } + pthread_mutex_lock(&LOCK_thread_count); + thd->query= old_query; + thd->query_length= old_query_length; + pthread_mutex_unlock(&LOCK_thread_count); + DBUG_RETURN(error); +} + + +bool ha_maria::is_crashed() const +{ + return (file->s->state.changed & STATE_CRASHED || + (my_disable_locking && file->s->state.open_count)); +} + + +int ha_maria::update_row(const byte * old_data, byte * new_data) +{ + statistic_increment(table->in_use->status_var.ha_update_count, &LOCK_status); + if (table->timestamp_field_type & TIMESTAMP_AUTO_SET_ON_UPDATE) + table->timestamp_field->set_time(); + return maria_update(file, old_data, new_data); +} + + +int ha_maria::delete_row(const byte * buf) +{ + statistic_increment(table->in_use->status_var.ha_delete_count, &LOCK_status); + return maria_delete(file, buf); +} + + +int ha_maria::index_read(byte * buf, const byte * key, + uint key_len, enum ha_rkey_function find_flag) +{ + DBUG_ASSERT(inited == INDEX); + statistic_increment(table->in_use->status_var.ha_read_key_count, + &LOCK_status); + int error= maria_rkey(file, buf, active_index, key, key_len, find_flag); + table->status= error ? STATUS_NOT_FOUND : 0; + return error; +} + + +int ha_maria::index_read_idx(byte * buf, uint index, const byte * key, + uint key_len, enum ha_rkey_function find_flag) +{ + statistic_increment(table->in_use->status_var.ha_read_key_count, + &LOCK_status); + int error= maria_rkey(file, buf, index, key, key_len, find_flag); + table->status= error ? STATUS_NOT_FOUND : 0; + return error; +} + + +int ha_maria::index_read_last(byte * buf, const byte * key, uint key_len) +{ + DBUG_ENTER("ha_maria::index_read_last"); + DBUG_ASSERT(inited == INDEX); + statistic_increment(table->in_use->status_var.ha_read_key_count, + &LOCK_status); + int error= maria_rkey(file, buf, active_index, key, key_len, + HA_READ_PREFIX_LAST); + table->status= error ? STATUS_NOT_FOUND : 0; + DBUG_RETURN(error); +} + + +int ha_maria::index_next(byte * buf) +{ + DBUG_ASSERT(inited == INDEX); + statistic_increment(table->in_use->status_var.ha_read_next_count, + &LOCK_status); + int error= maria_rnext(file, buf, active_index); + table->status= error ? STATUS_NOT_FOUND : 0; + return error; +} + + +int ha_maria::index_prev(byte * buf) +{ + DBUG_ASSERT(inited == INDEX); + statistic_increment(table->in_use->status_var.ha_read_prev_count, + &LOCK_status); + int error= maria_rprev(file, buf, active_index); + table->status= error ? STATUS_NOT_FOUND : 0; + return error; +} + + +int ha_maria::index_first(byte * buf) +{ + DBUG_ASSERT(inited == INDEX); + statistic_increment(table->in_use->status_var.ha_read_first_count, + &LOCK_status); + int error= maria_rfirst(file, buf, active_index); + table->status= error ? STATUS_NOT_FOUND : 0; + return error; +} + + +int ha_maria::index_last(byte * buf) +{ + DBUG_ASSERT(inited == INDEX); + statistic_increment(table->in_use->status_var.ha_read_last_count, + &LOCK_status); + int error= maria_rlast(file, buf, active_index); + table->status= error ? STATUS_NOT_FOUND : 0; + return error; +} + + +int ha_maria::index_next_same(byte * buf, + const byte *key __attribute__ ((unused)), + uint length __attribute__ ((unused))) +{ + DBUG_ASSERT(inited == INDEX); + statistic_increment(table->in_use->status_var.ha_read_next_count, + &LOCK_status); + int error= maria_rnext_same(file, buf); + table->status= error ? STATUS_NOT_FOUND : 0; + return error; +} + + +int ha_maria::rnd_init(bool scan) +{ + if (scan) + return maria_scan_init(file); + return maria_reset(file); // Free buffers +} + + +int ha_maria::rnd_next(byte *buf) +{ + statistic_increment(table->in_use->status_var.ha_read_rnd_next_count, + &LOCK_status); + int error= maria_scan(file, buf); + table->status= error ? STATUS_NOT_FOUND : 0; + return error; +} + + +int ha_maria::restart_rnd_next(byte *buf, byte *pos) +{ + return rnd_pos(buf, pos); +} + + +int ha_maria::rnd_pos(byte * buf, byte *pos) +{ + statistic_increment(table->in_use->status_var.ha_read_rnd_count, + &LOCK_status); + int error= maria_rrnd(file, buf, my_get_ptr(pos, ref_length)); + table->status= error ? STATUS_NOT_FOUND : 0; + return error; +} + + +void ha_maria::position(const byte * record) +{ + my_off_t position= maria_position(file); + my_store_ptr(ref, ref_length, position); +} + + +void ha_maria::info(uint flag) +{ + MARIA_INFO info; + char name_buff[FN_REFLEN]; + + (void) maria_status(file, &info, flag); + if (flag & HA_STATUS_VARIABLE) + { + stats.records= info.records; + stats.deleted= info.deleted; + stats.data_file_length= info.data_file_length; + stats.index_file_length= info.index_file_length; + stats.delete_length= info.delete_length; + stats.check_time= info.check_time; + stats.mean_rec_length= info.mean_reclength; + } + if (flag & HA_STATUS_CONST) + { + TABLE_SHARE *share= table->s; + stats.max_data_file_length= info.max_data_file_length; + stats.max_index_file_length= info.max_index_file_length; + stats.create_time= info.create_time; + ref_length= info.reflength; + share->db_options_in_use= info.options; + stats.block_size= maria_block_size; + + /* Update share */ + if (share->tmp_table == NO_TMP_TABLE) + pthread_mutex_lock(&share->mutex); + share->keys_in_use.set_prefix(share->keys); + share->keys_in_use.intersect_extended(info.key_map); + share->keys_for_keyread.intersect(share->keys_in_use); + share->db_record_offset= info.record_offset; + if (share->key_parts) + memcpy((char*) table->key_info[0].rec_per_key, + (char*) info.rec_per_key, + sizeof(table->key_info[0].rec_per_key) * share->key_parts); + if (share->tmp_table == NO_TMP_TABLE) + pthread_mutex_unlock(&share->mutex); + + /* + Set data_file_name and index_file_name to point at the symlink value + if table is symlinked (Ie; Real name is not same as generated name) + */ + data_file_name= index_file_name= 0; + fn_format(name_buff, file->filename, "", MARIA_NAME_DEXT, MY_APPEND_EXT); + if (strcmp(name_buff, info.data_file_name)) + data_file_name= info.data_file_name; + fn_format(name_buff, file->filename, "", MARIA_NAME_IEXT, MY_APPEND_EXT); + if (strcmp(name_buff, info.index_file_name)) + index_file_name= info.index_file_name; + } + if (flag & HA_STATUS_ERRKEY) + { + errkey= info.errkey; + my_store_ptr(dup_ref, ref_length, info.dupp_key_pos); + } + /* Faster to always update, than to do it based on flag */ + stats.update_time= info.update_time; + stats.auto_increment_value= info.auto_increment; +} + + +int ha_maria::extra(enum ha_extra_function operation) +{ + if ((specialflag & SPECIAL_SAFE_MODE) && operation == HA_EXTRA_KEYREAD) + return 0; + return maria_extra(file, operation, 0); +} + +int ha_maria::reset(void) +{ + return maria_reset(file); +} + +/* To be used with WRITE_CACHE and EXTRA_CACHE */ + +int ha_maria::extra_opt(enum ha_extra_function operation, ulong cache_size) +{ + if ((specialflag & SPECIAL_SAFE_MODE) && operation == HA_EXTRA_WRITE_CACHE) + return 0; + return maria_extra(file, operation, (void*) &cache_size); +} + + +int ha_maria::delete_all_rows() +{ + return maria_delete_all_rows(file); +} + + +int ha_maria::delete_table(const char *name) +{ + return maria_delete_table(name); +} + + +int ha_maria::external_lock(THD *thd, int lock_type) +{ + return maria_lock_database(file, !table->s->tmp_table ? + lock_type : ((lock_type == F_UNLCK) ? + F_UNLCK : F_EXTRA_LCK)); +} + + +THR_LOCK_DATA **ha_maria::store_lock(THD *thd, + THR_LOCK_DATA **to, + enum thr_lock_type lock_type) +{ + if (lock_type != TL_IGNORE && file->lock.type == TL_UNLOCK) + file->lock.type= lock_type; + *to++= &file->lock; + return to; +} + + +void ha_maria::update_create_info(HA_CREATE_INFO *create_info) +{ + ha_maria::info(HA_STATUS_AUTO | HA_STATUS_CONST); + if (!(create_info->used_fields & HA_CREATE_USED_AUTO)) + { + create_info->auto_increment_value= stats.auto_increment_value; + } + create_info->data_file_name= data_file_name; + create_info->index_file_name= index_file_name; +} + + +int ha_maria::create(const char *name, register TABLE *table_arg, + HA_CREATE_INFO *info) +{ + int error; + uint i, j, recpos, minpos, fieldpos, temp_length, length, create_flags= 0; + bool found_real_auto_increment= 0; + enum ha_base_keytype type; + char buff[FN_REFLEN]; + KEY *pos; + MARIA_KEYDEF *keydef; + MARIA_COLUMNDEF *recinfo, *recinfo_pos; + HA_KEYSEG *keyseg; + TABLE_SHARE *share= table_arg->s; + uint options= share->db_options_in_use; + DBUG_ENTER("ha_maria::create"); + + type= HA_KEYTYPE_BINARY; // Keep compiler happy + if (!(my_multi_malloc(MYF(MY_WME), + &recinfo, (share->fields * 2 + 2) * + sizeof(MARIA_COLUMNDEF), + &keydef, share->keys * sizeof(MARIA_KEYDEF), + &keyseg, + ((share->key_parts + share->keys) * + sizeof(HA_KEYSEG)), NullS))) + DBUG_RETURN(HA_ERR_OUT_OF_MEM); + + pos= table_arg->key_info; + for (i= 0; i < share->keys; i++, pos++) + { + if (pos->flags & HA_USES_PARSER) + create_flags |= HA_CREATE_RELIES_ON_SQL_LAYER; + keydef[i].flag= (pos->flags & (HA_NOSAME | HA_FULLTEXT | HA_SPATIAL)); + keydef[i].key_alg= pos->algorithm == HA_KEY_ALG_UNDEF ? + (pos->flags & HA_SPATIAL ? HA_KEY_ALG_RTREE : HA_KEY_ALG_BTREE) : + pos->algorithm; + keydef[i].block_length= pos->block_size; + + keydef[i].seg= keyseg; + keydef[i].keysegs= pos->key_parts; + for (j= 0; j < pos->key_parts; j++) + { + Field *field= pos->key_part[j].field; + type= field->key_type(); + keydef[i].seg[j].flag= pos->key_part[j].key_part_flag; + + if (options & HA_OPTION_PACK_KEYS || + (pos->flags & (HA_PACK_KEY | HA_BINARY_PACK_KEY | + HA_SPACE_PACK_USED))) + { + if (pos->key_part[j].length > 8 && + (type == HA_KEYTYPE_TEXT || + type == HA_KEYTYPE_NUM || + (type == HA_KEYTYPE_BINARY && !field->zero_pack()))) + { + /* No blobs here */ + if (j == 0) + keydef[i].flag |= HA_PACK_KEY; + if (!(field->flags & ZEROFILL_FLAG) && + (field->type() == MYSQL_TYPE_STRING || + field->type() == MYSQL_TYPE_VAR_STRING || + ((int) (pos->key_part[j].length - field->decimals())) >= 4)) + keydef[i].seg[j].flag |= HA_SPACE_PACK; + } + else if (j == 0 && (!(pos->flags & HA_NOSAME) || pos->key_length > 16)) + keydef[i].flag |= HA_BINARY_PACK_KEY; + } + keydef[i].seg[j].type= (int) type; + keydef[i].seg[j].start= pos->key_part[j].offset; + keydef[i].seg[j].length= pos->key_part[j].length; + keydef[i].seg[j].bit_start= keydef[i].seg[j].bit_end= + keydef[i].seg[j].bit_length= 0; + keydef[i].seg[j].bit_pos= 0; + keydef[i].seg[j].language= field->charset()->number; + + if (field->null_ptr) + { + keydef[i].seg[j].null_bit= field->null_bit; + keydef[i].seg[j].null_pos= (uint) (field->null_ptr - + (uchar *) table_arg->record[0]); + } + else + { + keydef[i].seg[j].null_bit= 0; + keydef[i].seg[j].null_pos= 0; + } + if (field->type() == FIELD_TYPE_BLOB || + field->type() == FIELD_TYPE_GEOMETRY) + { + keydef[i].seg[j].flag |= HA_BLOB_PART; + /* save number of bytes used to pack length */ + keydef[i].seg[j].bit_start= (uint) (field->pack_length() - + share->blob_ptr_size); + } + else if (field->type() == FIELD_TYPE_BIT) + { + keydef[i].seg[j].bit_length= ((Field_bit *) field)->bit_len; + keydef[i].seg[j].bit_start= ((Field_bit *) field)->bit_ofs; + keydef[i].seg[j].bit_pos= (uint) (((Field_bit *) field)->bit_ptr - + (uchar *) table_arg->record[0]); + } + } + keyseg += pos->key_parts; + } + + if (table_arg->found_next_number_field) + { + keydef[share->next_number_index].flag |= HA_AUTO_KEY; + found_real_auto_increment= share->next_number_key_offset == 0; + } + + recpos= 0; + recinfo_pos= recinfo; + while (recpos < (uint) share->reclength) + { + Field **field, *found= 0; + minpos= share->reclength; + length= 0; + + for (field= table_arg->field; *field; field++) + { + if ((fieldpos= (*field)->offset()) >= recpos && fieldpos <= minpos) + { + /* skip null fields */ + if (!(temp_length= (*field)->pack_length_in_rec())) + continue; /* Skip null-fields */ + if (!found || fieldpos < minpos || + (fieldpos == minpos && temp_length < length)) + { + minpos= fieldpos; + found= *field; + length= temp_length; + } + } + } + DBUG_PRINT("loop", ("found: 0x%lx recpos: %d minpos: %d length: %d", + found, recpos, minpos, length)); + if (recpos != minpos) + { // Reserved space (Null bits?) + bzero((char*) recinfo_pos, sizeof(*recinfo_pos)); + recinfo_pos->type= (int) FIELD_NORMAL; + recinfo_pos++->length= (uint16) (minpos - recpos); + } + if (!found) + break; + + if (found->flags & BLOB_FLAG) + recinfo_pos->type= (int) FIELD_BLOB; + else if (found->type() == MYSQL_TYPE_VARCHAR) + recinfo_pos->type= FIELD_VARCHAR; + else if (!(options & HA_OPTION_PACK_RECORD)) + recinfo_pos->type= (int) FIELD_NORMAL; + else if (found->zero_pack()) + recinfo_pos->type= (int) FIELD_SKIP_ZERO; + else + recinfo_pos->type= (int) ((length <= 3 || + (found->flags & ZEROFILL_FLAG)) ? + FIELD_NORMAL : + found->type() == MYSQL_TYPE_STRING || + found->type() == MYSQL_TYPE_VAR_STRING ? + FIELD_SKIP_ENDSPACE : FIELD_SKIP_PRESPACE); + if (found->null_ptr) + { + recinfo_pos->null_bit= found->null_bit; + recinfo_pos->null_pos= (uint) (found->null_ptr - + (uchar *) table_arg->record[0]); + } + else + { + recinfo_pos->null_bit= 0; + recinfo_pos->null_pos= 0; + } + (recinfo_pos++)->length= (uint16) length; + recpos= minpos + length; + DBUG_PRINT("loop", ("length: %d type: %d", + recinfo_pos[-1].length, recinfo_pos[-1].type)); + + } + MARIA_CREATE_INFO create_info; + bzero((char*) &create_info, sizeof(create_info)); + create_info.max_rows= share->max_rows; + create_info.reloc_rows= share->min_rows; + create_info.with_auto_increment= found_real_auto_increment; + create_info.auto_increment= (info->auto_increment_value ? + info->auto_increment_value - 1 : (ulonglong) 0); + create_info.data_file_length= ((ulonglong) share->max_rows * + share->avg_row_length); + create_info.data_file_name= info->data_file_name; + create_info.index_file_name= info->index_file_name; + + if (info->options & HA_LEX_CREATE_TMP_TABLE) + create_flags |= HA_CREATE_TMP_TABLE; + if (options & HA_OPTION_PACK_RECORD) + create_flags |= HA_PACK_RECORD; + if (options & HA_OPTION_CHECKSUM) + create_flags |= HA_CREATE_CHECKSUM; + if (options & HA_OPTION_DELAY_KEY_WRITE) + create_flags |= HA_CREATE_DELAY_KEY_WRITE; + + /* TODO: Check that the following fn_format is really needed */ + error= + maria_create(fn_format + (buff, name, "", "", MY_UNPACK_FILENAME | MY_APPEND_EXT), + share->keys, keydef, (uint) (recinfo_pos - recinfo), recinfo, + 0, (MARIA_UNIQUEDEF *) 0, &create_info, create_flags); + + my_free((gptr) recinfo, MYF(0)); + DBUG_RETURN(error); +} + + +int ha_maria::rename_table(const char *from, const char *to) +{ + return maria_rename(from, to); +} + + +void ha_maria::get_auto_increment(ulonglong offset, ulonglong increment, + ulonglong nb_desired_values, + ulonglong *first_value, + ulonglong *nb_reserved_values) +{ + ulonglong nr; + int error; + byte key[HA_MAX_KEY_LENGTH]; + + if (!table->s->next_number_key_offset) + { // Autoincrement at key-start + ha_maria::info(HA_STATUS_AUTO); + *first_value= stats.auto_increment_value; + /* Maria has only table-level lock for now, so reserves to +inf */ + *nb_reserved_values= ULONGLONG_MAX; + return; + } + + /* it's safe to call the following if bulk_insert isn't on */ + maria_flush_bulk_insert(file, table->s->next_number_index); + + (void) extra(HA_EXTRA_KEYREAD); + key_copy(key, table->record[0], + table->key_info + table->s->next_number_index, + table->s->next_number_key_offset); + error= maria_rkey(file, table->record[1], (int) table->s->next_number_index, + key, table->s->next_number_key_offset, HA_READ_PREFIX_LAST); + if (error) + nr= 1; + else + { + /* Get data from record[1] */ + nr= ((ulonglong) table->next_number_field-> + val_int_offset(table->s->rec_buff_length) + 1); + } + extra(HA_EXTRA_NO_KEYREAD); + *first_value= nr; + /* + MySQL needs to call us for next row: assume we are inserting ("a",null) + here, we return 3, and next this statement will want to insert ("b",null): + there is no reason why ("b",3+1) would be the good row to insert: maybe it + already exists, maybe 3+1 is too large... + */ + *nb_reserved_values= 1; +} + + +/* + Find out how many rows there is in the given range + + SYNOPSIS + records_in_range() + inx Index to use + min_key Start of range. Null pointer if from first key + max_key End of range. Null pointer if to last key + + NOTES + min_key.flag can have one of the following values: + HA_READ_KEY_EXACT Include the key in the range + HA_READ_AFTER_KEY Don't include key in range + + max_key.flag can have one of the following values: + HA_READ_BEFORE_KEY Don't include key in range + HA_READ_AFTER_KEY Include all 'end_key' values in the range + + RETURN + HA_POS_ERROR Something is wrong with the index tree. + 0 There is no matching keys in the given range + number > 0 There is approximately 'number' matching rows in + the range. +*/ + +ha_rows ha_maria::records_in_range(uint inx, key_range *min_key, + key_range *max_key) +{ + return (ha_rows) maria_records_in_range(file, (int) inx, min_key, max_key); +} + + +int ha_maria::ft_read(byte * buf) +{ + int error; + + if (!ft_handler) + return -1; + + thread_safe_increment(table->in_use->status_var.ha_read_next_count, + &LOCK_status); // why ? + + error= ft_handler->please->read_next(ft_handler, (char*) buf); + + table->status= error ? STATUS_NOT_FOUND : 0; + return error; +} + + +uint ha_maria::checksum() const +{ + return (uint) file->state->checksum; +} + + +bool ha_maria::check_if_incompatible_data(HA_CREATE_INFO *info, + uint table_changes) +{ + uint options= table->s->db_options_in_use; + + if (info->auto_increment_value != stats.auto_increment_value || + info->data_file_name != data_file_name || + info->index_file_name != index_file_name || + table_changes == IS_EQUAL_NO || + table_changes & IS_EQUAL_PACK_LENGTH) // Not implemented yet + return COMPATIBLE_DATA_NO; + + if ((options & (HA_OPTION_PACK_RECORD | HA_OPTION_CHECKSUM | + HA_OPTION_DELAY_KEY_WRITE)) != + (info->table_options & (HA_OPTION_PACK_RECORD | HA_OPTION_CHECKSUM | + HA_OPTION_DELAY_KEY_WRITE))) + return COMPATIBLE_DATA_NO; + return COMPATIBLE_DATA_YES; +} + +handlerton maria_hton; + +static int ha_maria_init() +{ + maria_hton.state=SHOW_OPTION_YES; + maria_hton.db_type=DB_TYPE_MARIA; + maria_hton.create=maria_create_handler; + maria_hton.panic=maria_panic; + maria_hton.flags=HTON_CAN_RECREATE; + return test(maria_init()); +} + +struct st_mysql_storage_engine maria_storage_engine= +{ MYSQL_HANDLERTON_INTERFACE_VERSION, &maria_hton }; + +mysql_declare_plugin(maria) +{ + MYSQL_STORAGE_ENGINE_PLUGIN, + &maria_storage_engine, + "Maria", + "MySQL AB", + "Traditional transactional MySQL tables", + ha_maria_init, /* Plugin Init */ + NULL, /* Plugin Deinit */ + 0x0100, /* 1.0 */ + 0 +} +mysql_declare_plugin_end; diff --git a/storage/maria/ha_maria.h b/storage/maria/ha_maria.h new file mode 100644 index 00000000000..cdd2305f6a1 --- /dev/null +++ b/storage/maria/ha_maria.h @@ -0,0 +1,145 @@ +/* Copyright (C) 2006,2004 MySQL AB & MySQL Finland AB & TCX DataKonsult AB + + 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; either version 2 of the License, or + (at your option) any later version. + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ + + +#ifdef USE_PRAGMA_INTERFACE +#pragma interface /* gcc class implementation */ +#endif + +/* class for the the maria handler */ + +#include + +#define HA_RECOVER_NONE 0 /* No automatic recover */ +#define HA_RECOVER_DEFAULT 1 /* Automatic recover active */ +#define HA_RECOVER_BACKUP 2 /* Make a backupfile on recover */ +#define HA_RECOVER_FORCE 4 /* Recover even if we loose rows */ +#define HA_RECOVER_QUICK 8 /* Don't check rows in data file */ + +extern ulong maria_sort_buffer_size; +extern TYPELIB maria_recover_typelib; +extern ulong maria_recover_options; + +class ha_maria :public handler +{ + MARIA_HA *file; + ulong int_table_flags; + char *data_file_name, *index_file_name; + bool can_enable_indexes; + int repair(THD * thd, HA_CHECK ¶m, bool optimize); + +public: + ha_maria(TABLE_SHARE * table_arg); + ~ha_maria() + {} + const char *table_type() const + { return "MARIA"; } + const char *index_type(uint key_number); + const char **bas_ext() const; + ulonglong table_flags() const + { return int_table_flags; } + ulong index_flags(uint inx, uint part, bool all_parts) const + { + return ((table_share->key_info[inx].algorithm == HA_KEY_ALG_FULLTEXT) ? + 0 : HA_READ_NEXT | HA_READ_PREV | HA_READ_RANGE | + HA_READ_ORDER | HA_KEYREAD_ONLY); + } + uint max_supported_keys() const + { return MARIA_MAX_KEY; } + uint max_supported_key_length() const + { return HA_MAX_KEY_LENGTH; } + uint max_supported_key_part_length() const + { return HA_MAX_KEY_LENGTH; } + uint checksum() const; + + virtual bool check_if_locking_is_allowed(uint sql_command, + ulong type, TABLE * table, + uint count, + bool called_by_logger_thread); + int open(const char *name, int mode, uint test_if_locked); + int close(void); + int write_row(byte * buf); + int update_row(const byte * old_data, byte * new_data); + int delete_row(const byte * buf); + int index_read(byte * buf, const byte * key, + uint key_len, enum ha_rkey_function find_flag); + int index_read_idx(byte * buf, uint idx, const byte * key, + uint key_len, enum ha_rkey_function find_flag); + int index_read_last(byte * buf, const byte * key, uint key_len); + int index_next(byte * buf); + int index_prev(byte * buf); + int index_first(byte * buf); + int index_last(byte * buf); + int index_next_same(byte * buf, const byte * key, uint keylen); + int ft_init() + { + if (!ft_handler) + return 1; + ft_handler->please->reinit_search(ft_handler); + return 0; + } + FT_INFO *ft_init_ext(uint flags, uint inx, String * key) + { + return maria_ft_init_search(flags, file, inx, + (byte *) key->ptr(), key->length(), + key->charset(), table->record[0]); + } + int ft_read(byte * buf); + int rnd_init(bool scan); + int rnd_next(byte * buf); + int rnd_pos(byte * buf, byte * pos); + int restart_rnd_next(byte * buf, byte * pos); + void position(const byte * record); + void info(uint); + int extra(enum ha_extra_function operation); + int extra_opt(enum ha_extra_function operation, ulong cache_size); + int reset(void); + int external_lock(THD * thd, int lock_type); + int delete_all_rows(void); + int disable_indexes(uint mode); + int enable_indexes(uint mode); + int indexes_are_disabled(void); + void start_bulk_insert(ha_rows rows); + int end_bulk_insert(); + ha_rows records_in_range(uint inx, key_range * min_key, key_range * max_key); + void update_create_info(HA_CREATE_INFO * create_info); + int create(const char *name, TABLE * form, HA_CREATE_INFO * create_info); + THR_LOCK_DATA **store_lock(THD * thd, THR_LOCK_DATA ** to, + enum thr_lock_type lock_type); + virtual void get_auto_increment(ulonglong offset, ulonglong increment, + ulonglong nb_desired_values, + ulonglong *first_value, + ulonglong *nb_reserved_values); + int rename_table(const char *from, const char *to); + int delete_table(const char *name); + int check(THD * thd, HA_CHECK_OPT * check_opt); + int analyze(THD * thd, HA_CHECK_OPT * check_opt); + int repair(THD * thd, HA_CHECK_OPT * check_opt); + bool check_and_repair(THD * thd); + bool is_crashed() const; + bool auto_repair() const + { return maria_recover_options != 0; } + int optimize(THD * thd, HA_CHECK_OPT * check_opt); + int restore(THD * thd, HA_CHECK_OPT * check_opt); + int backup(THD * thd, HA_CHECK_OPT * check_opt); + int assign_to_keycache(THD * thd, HA_CHECK_OPT * check_opt); + int preload_keys(THD * thd, HA_CHECK_OPT * check_opt); + bool check_if_incompatible_data(HA_CREATE_INFO * info, uint table_changes); +#ifdef HAVE_REPLICATION + int dump(THD * thd, int fd); + int net_read_dump(NET * net); +#endif +}; diff --git a/storage/maria/ma_test_all.sh b/storage/maria/ma_test_all.sh index f857127dca9..ec4d1778db9 100755 --- a/storage/maria/ma_test_all.sh +++ b/storage/maria/ma_test_all.sh @@ -134,10 +134,8 @@ echo "ma_test2$suffix $silent -L -K -R1 -m2000 ; Should give error 135" ./ma_test2$suffix $silent -m10000 -e16384 -E16384 -K -L ./maria_chk$suffix -sm test2 -./ma_test2$suffix $silent -L -K -W -P -m50 -l -./maria_log$suffix -./ma_test2$suffix $silent -L -K -W -P -m50 -l -b100 -./maria_log$suffix +./ma_test2$suffix $silent -L -K -W -P -m50 +./ma_test2$suffix $silent -L -K -W -P -m50 -b100 time ./ma_test2$suffix $silent time ./ma_test2$suffix $silent -K -B time ./ma_test2$suffix $silent -L -B -- cgit v1.2.1 From 4c6971b40225446e3f142d485ce6e93766122136 Mon Sep 17 00:00:00 2001 From: unknown Date: Thu, 7 Sep 2006 17:07:17 +0200 Subject: Manually merging changes made to MyISAM into Maria. End of merge. storage/maria/ft_maria.c: Rename: BitKeeper/deleted/.del-ft_maria.c -> storage/maria/ft_maria.c configure.in: maria moves to its plug.in storage/maria/Makefile.am: merging changes made to MyISAM into Maria. ft_maria.c is still needed. storage/maria/ha_maria.cc: merging changes made to MyISAM into Maria storage/maria/ma_dynrec.c: merging changes made to MyISAM into Maria storage/maria/ma_extra.c: merging changes made to MyISAM into Maria storage/maria/ma_ft_parser.c: merging changes made to MyISAM into Maria storage/maria/ma_open.c: merging changes made to MyISAM into Maria storage/maria/ma_sort.c: merging changes made to MyISAM into Maria storage/maria/ma_update.c: merging changes made to MyISAM into Maria storage/maria/ma_write.c: merging changes made to MyISAM into Maria storage/maria/maria_def.h: merging changes made to MyISAM into Maria storage/myisam/Makefile.am: merging changes made to MyISAM into Maria storage/maria/plug.in: merging changes made to MyISAM into Maria --- configure.in | 7 -- storage/maria/Makefile.am | 79 +++++++++++++++++---- storage/maria/ft_maria.c | 49 +++++++++++++ storage/maria/ha_maria.cc | 30 +++++--- storage/maria/ma_dynrec.c | 163 +++++++++++++++++++++++++++++++++++-------- storage/maria/ma_extra.c | 5 ++ storage/maria/ma_ft_parser.c | 2 +- storage/maria/ma_open.c | 1 + storage/maria/ma_sort.c | 11 ++- storage/maria/ma_update.c | 13 +++- storage/maria/ma_write.c | 12 ++++ storage/maria/maria_def.h | 3 + storage/maria/plug.in | 7 ++ storage/myisam/Makefile.am | 2 +- 14 files changed, 319 insertions(+), 65 deletions(-) create mode 100644 storage/maria/ft_maria.c create mode 100644 storage/maria/plug.in diff --git a/configure.in b/configure.in index bf9a8286d01..f0cdcde1abb 100644 --- a/configure.in +++ b/configure.in @@ -2149,13 +2149,6 @@ MYSQL_STORAGE_ENGINE(partition, partition, [Partition Support], [MySQL Partitioning Support], [max,max-no-ndb]) dnl -- ndbcluster requires partition to be enabled -MYSQL_PLUGIN_DEPENDS(ndbcluster, [partition]) - -MYSQL_STORAGE_ENGINE(maria,maria, [Maria Storage Engine], - [Traditional transactional MySQL tables], [max,max-no-ndb]) -MYSQL_PLUGIN_DIRECTORY(maria, [storage/maria]) -MYSQL_PLUGIN_STATIC(maria, [libmaria.a]) -MYSQL_PLUGIN_MANDATORY(maria) MYSQL_CONFIGURE_PLUGINS([none]) diff --git a/storage/maria/Makefile.am b/storage/maria/Makefile.am index 0271e29461e..f5678a55266 100644 --- a/storage/maria/Makefile.am +++ b/storage/maria/Makefile.am @@ -14,31 +14,87 @@ # along with this program; if not, write to the Free Software # Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA +MYSQLDATAdir = $(localstatedir) +MYSQLSHAREdir = $(pkgdatadir) +MYSQLBASEdir= $(prefix) +MYSQLLIBdir= $(pkglibdir) +INCLUDES = -I$(top_srcdir)/include -I$(top_builddir)/include \ + -I$(top_srcdir)/regex \ + -I$(top_srcdir)/sql \ + -I$(srcdir) +WRAPLIBS= + +LDADD = + +DEFS = @DEFS@ + EXTRA_DIST = ma_test_all.sh ma_test_all.res ma_ft_stem.c CMakeLists.txt pkgdata_DATA = ma_test_all ma_test_all.res - -INCLUDES = -I$(top_builddir)/include -I$(top_srcdir)/include -LDADD = @CLIENT_EXTRA_LDFLAGS@ libmaria.a \ - $(top_builddir)/storage/myisam/libmyisam.a \ - $(top_builddir)/mysys/libmysys.a \ - $(top_builddir)/dbug/libdbug.a \ - $(top_builddir)/strings/libmystrings.a @ZLIB_LIBS@ pkglib_LIBRARIES = libmaria.a bin_PROGRAMS = maria_chk maria_pack maria_ftdump maria_chk_DEPENDENCIES= $(LIBRARIES) +# Only reason to link with libmyisam.a here is that it's where some fulltext +# pieces are (but soon we'll remove fulltext dependencies from Maria). +# For now, it imposes that storage/myisam be built before storage/maria. +maria_chk_LDADD= @CLIENT_EXTRA_LDFLAGS@ libmaria.a \ + $(top_builddir)/storage/myisam/libmyisam.a \ + $(top_builddir)/mysys/libmysys.a \ + $(top_builddir)/dbug/libdbug.a \ + $(top_builddir)/strings/libmystrings.a @ZLIB_LIBS@ maria_pack_DEPENDENCIES=$(LIBRARIES) +maria_pack_LDADD= @CLIENT_EXTRA_LDFLAGS@ libmaria.a \ + $(top_builddir)/storage/myisam/libmyisam.a \ + $(top_builddir)/mysys/libmysys.a \ + $(top_builddir)/dbug/libdbug.a \ + $(top_builddir)/strings/libmystrings.a @ZLIB_LIBS@ noinst_PROGRAMS = ma_test1 ma_test2 ma_test3 ma_rt_test ma_sp_test ma_control_file_test noinst_HEADERS = maria_def.h ma_rt_index.h ma_rt_key.h ma_rt_mbr.h \ ma_sp_defs.h ma_fulltext.h ma_ftdefs.h ma_ft_test1.h ma_ft_eval.h \ - ma_control_file.h + ma_control_file.h ha_maria.h ma_test1_DEPENDENCIES= $(LIBRARIES) +ma_test1_LDADD= @CLIENT_EXTRA_LDFLAGS@ libmaria.a \ + $(top_builddir)/storage/myisam/libmyisam.a \ + $(top_builddir)/mysys/libmysys.a \ + $(top_builddir)/dbug/libdbug.a \ + $(top_builddir)/strings/libmystrings.a @ZLIB_LIBS@ ma_test2_DEPENDENCIES= $(LIBRARIES) +ma_test2_LDADD= @CLIENT_EXTRA_LDFLAGS@ libmaria.a \ + $(top_builddir)/storage/myisam/libmyisam.a \ + $(top_builddir)/mysys/libmysys.a \ + $(top_builddir)/dbug/libdbug.a \ + $(top_builddir)/strings/libmystrings.a @ZLIB_LIBS@ ma_test3_DEPENDENCIES= $(LIBRARIES) +ma_test3_LDADD= @CLIENT_EXTRA_LDFLAGS@ libmaria.a \ + $(top_builddir)/storage/myisam/libmyisam.a \ + $(top_builddir)/mysys/libmysys.a \ + $(top_builddir)/dbug/libdbug.a \ + $(top_builddir)/strings/libmystrings.a @ZLIB_LIBS@ #ma_ft_test1_DEPENDENCIES= $(LIBRARIES) #ma_ft_eval_DEPENDENCIES= $(LIBRARIES) maria_ftdump_DEPENDENCIES= $(LIBRARIES) +maria_ftdump_LDADD= @CLIENT_EXTRA_LDFLAGS@ libmaria.a \ + $(top_builddir)/storage/myisam/libmyisam.a \ + $(top_builddir)/mysys/libmysys.a \ + $(top_builddir)/dbug/libdbug.a \ + $(top_builddir)/strings/libmystrings.a @ZLIB_LIBS@ ma_rt_test_DEPENDENCIES= $(LIBRARIES) +ma_rt_test_LDADD= @CLIENT_EXTRA_LDFLAGS@ libmaria.a \ + $(top_builddir)/storage/myisam/libmyisam.a \ + $(top_builddir)/mysys/libmysys.a \ + $(top_builddir)/dbug/libdbug.a \ + $(top_builddir)/strings/libmystrings.a @ZLIB_LIBS@ ma_sp_test_DEPENDENCIES= $(LIBRARIES) +ma_sp_test_LDADD= @CLIENT_EXTRA_LDFLAGS@ libmaria.a \ + $(top_builddir)/storage/myisam/libmyisam.a \ + $(top_builddir)/mysys/libmysys.a \ + $(top_builddir)/dbug/libdbug.a \ + $(top_builddir)/strings/libmystrings.a @ZLIB_LIBS@ +ma_control_file_test_DEPENDENCIES= $(LIBRARIES) +ma_control_file_test_LDADD= @CLIENT_EXTRA_LDFLAGS@ libmaria.a \ + $(top_builddir)/storage/myisam/libmyisam.a \ + $(top_builddir)/mysys/libmysys.a \ + $(top_builddir)/dbug/libdbug.a \ + $(top_builddir)/strings/libmystrings.a @ZLIB_LIBS@ libmaria_a_SOURCES = ma_init.c ma_open.c ma_extra.c ma_info.c ma_rkey.c \ ma_rnext.c ma_rnext_same.c \ ma_search.c ma_page.c ma_key.c ma_locking.c \ @@ -53,12 +109,11 @@ libmaria_a_SOURCES = ma_init.c ma_open.c ma_extra.c ma_info.c ma_rkey.c \ ma_delete_table.c ma_rename.c ma_check.c \ ma_keycache.c ma_preload.c ma_ft_parser.c \ ma_ft_update.c ma_ft_boolean_search.c \ - ma_ft_nlq_search.c ma_sort.c \ + ma_ft_nlq_search.c ft_maria.c ma_sort.c \ + ha_maria.cc \ ma_rt_index.c ma_rt_key.c ma_rt_mbr.c ma_rt_split.c \ - ma_sp_key.c \ - ma_control_file.c + ma_sp_key.c ma_control_file.c CLEANFILES = test?.MA? FT?.MA? isam.log ma_test_all ma_rt_test.MA? sp_test.MA? maria_control -DEFS = SUFFIXES = .sh diff --git a/storage/maria/ft_maria.c b/storage/maria/ft_maria.c new file mode 100644 index 00000000000..7104c6704ba --- /dev/null +++ b/storage/maria/ft_maria.c @@ -0,0 +1,49 @@ +/* Copyright (C) 2006 MySQL AB & MySQL Finland AB & TCX DataKonsult AB + + 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; either version 2 of the License, or + (at your option) any later version. + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ + +/* Written by Sergei A. Golubchik, who has a shared copyright to this code */ + +/* + This function is for interface functions between fulltext and maria +*/ + +#include "ma_ftdefs.h" + +FT_INFO *maria_ft_init_search(uint flags, void *info, uint keynr, + byte *query, uint query_len, CHARSET_INFO *cs, + byte *record) +{ + FT_INFO *res; + if (flags & FT_BOOL) + res= maria_ft_init_boolean_search((MARIA_HA *) info, keynr, query, + query_len, cs); + else + res= maria_ft_init_nlq_search((MARIA_HA *) info, keynr, query, query_len, + flags, record); + return res; +} + +const struct _ft_vft _ma_ft_vft_nlq = { + maria_ft_nlq_read_next, maria_ft_nlq_find_relevance, + maria_ft_nlq_close_search, maria_ft_nlq_get_relevance, + maria_ft_nlq_reinit_search +}; +const struct _ft_vft _ma_ft_vft_boolean = { + maria_ft_boolean_read_next, maria_ft_boolean_find_relevance, + maria_ft_boolean_close_search, maria_ft_boolean_get_relevance, + maria_ft_boolean_reinit_search +}; + diff --git a/storage/maria/ha_maria.cc b/storage/maria/ha_maria.cc index 29718f1493e..a0084ef5e9c 100644 --- a/storage/maria/ha_maria.cc +++ b/storage/maria/ha_maria.cc @@ -19,18 +19,15 @@ #pragma implementation // gcc: Class implementation #endif +#define MYSQL_SERVER 1 #include "mysql_priv.h" +#include #include #include #include "ha_maria.h" -#ifndef MASTER -#include "../srclib/maria/maria_def.h" -#else -#include "../storage/maria/maria_def.h" -#include "../storage/maria/ma_rt_index.h" -#endif -#include +#include "maria_def.h" +#include "ma_rt_index.h" ulong maria_recover_options= HA_RECOVER_NONE; @@ -288,6 +285,15 @@ bool ha_maria::check_if_locking_is_allowed(uint sql_command, table->s->table_name.str); return FALSE; } + + /* + Deny locking of the log tables, which is incompatible with + concurrent insert. Unless called from a logger THD: + general_log_thd or slow_log_thd. + */ + if (!called_by_logger_thread) + return check_if_log_table_locking_is_allowed(sql_command, type, table); + return TRUE; } @@ -486,11 +492,14 @@ int ha_maria::restore(THD * thd, HA_CHECK_OPT *check_opt) HA_CHECK_OPT tmp_check_opt; char *backup_dir= thd->lex->backup_dir; char src_path[FN_REFLEN], dst_path[FN_REFLEN]; - const char *table_name= table->s->table_name.str; + char table_name[FN_REFLEN]; int error; const char *errmsg; DBUG_ENTER("restore"); + VOID(tablename_to_filename(table->s->table_name.str, table_name, + sizeof(table_name))); + if (fn_format_relative_to_data_home(src_path, table_name, backup_dir, MARIA_NAME_DEXT)) DBUG_RETURN(HA_ADMIN_INVALID); @@ -526,11 +535,14 @@ int ha_maria::backup(THD * thd, HA_CHECK_OPT *check_opt) { char *backup_dir= thd->lex->backup_dir; char src_path[FN_REFLEN], dst_path[FN_REFLEN]; - const char *table_name= table->s->table_name.str; + char table_name[FN_REFLEN]; int error; const char *errmsg; DBUG_ENTER("ha_maria::backup"); + VOID(tablename_to_filename(table->s->table_name.str, table_name, + sizeof(table_name))); + if (fn_format_relative_to_data_home(dst_path, table_name, backup_dir, reg_ext)) { diff --git a/storage/maria/ma_dynrec.c b/storage/maria/ma_dynrec.c index 047826408c3..253a538861a 100644 --- a/storage/maria/ma_dynrec.c +++ b/storage/maria/ma_dynrec.c @@ -1304,12 +1304,41 @@ void _ma_store_blob_length(byte *pos,uint pack_length,uint length) } - /* Read record from datafile */ - /* Returns 0 if ok, -1 if error */ +/* + Read record from datafile. + + SYNOPSIS + _ma_read_dynamic_record() + info MARIA_HA pointer to table. + filepos From where to read the record. + buf Destination for record. + + NOTE + + If a write buffer is active, it needs to be flushed if its contents + intersects with the record to read. We always check if the position + of the first byte of the write buffer is lower than the position + past the last byte to read. In theory this is also true if the write + buffer is completely below the read segment. That is, if there is no + intersection. But this case is unusual. We flush anyway. Only if the + first byte in the write buffer is above the last byte to read, we do + not flush. + + A dynamic record may need several reads. So this check must be done + before every read. Reading a dynamic record starts with reading the + block header. If the record does not fit into the free space of the + header, the block may be longer than the header. In this case a + second read is necessary. These one or two reads repeat for every + part of the record. + + RETURN + 0 OK + -1 Error +*/ int _ma_read_dynamic_record(MARIA_HA *info, my_off_t filepos, byte *buf) { - int flag; + int block_of_record; uint b_type,left_length; byte *to; MARIA_BLOCK_INFO block_info; @@ -1321,20 +1350,19 @@ int _ma_read_dynamic_record(MARIA_HA *info, my_off_t filepos, byte *buf) LINT_INIT(to); LINT_INIT(left_length); file=info->dfile; - block_info.next_filepos=filepos; /* for easyer loop */ - flag=block_info.second_read=0; + block_of_record= 0; /* First block of record is numbered as zero. */ + block_info.second_read= 0; do - { + { + /* A corrupted table can have wrong pointers. (Bug# 19835) */ + if (filepos == HA_OFFSET_ERROR) + goto panic; if (info->opt_flag & WRITE_CACHE_USED && - info->rec_cache.pos_in_file <= block_info.next_filepos && + info->rec_cache.pos_in_file < filepos + MARIA_BLOCK_INFO_HEADER_LENGTH && flush_io_cache(&info->rec_cache)) goto err; - /* A corrupted table can have wrong pointers. (Bug# 19835) */ - if (block_info.next_filepos == HA_OFFSET_ERROR) - goto panic; info->rec_cache.seek_not_done=1; - if ((b_type= _ma_get_block_info(&block_info,file, - block_info.next_filepos)) + if ((b_type= _ma_get_block_info(&block_info, file, filepos)) & (BLOCK_DELETED | BLOCK_ERROR | BLOCK_SYNC_ERROR | BLOCK_FATAL_ERROR)) { @@ -1342,15 +1370,14 @@ int _ma_read_dynamic_record(MARIA_HA *info, my_off_t filepos, byte *buf) my_errno=HA_ERR_RECORD_DELETED; goto err; } - if (flag == 0) /* First block */ + if (block_of_record++ == 0) /* First block */ { - flag=1; if (block_info.rec_len > (uint) info->s->base.max_pack_length) goto panic; if (info->s->base.blobs) { if (!(to=_ma_alloc_rec_buff(info, block_info.rec_len, - &info->rec_buff))) + &info->rec_buff))) goto err; } else @@ -1359,11 +1386,41 @@ int _ma_read_dynamic_record(MARIA_HA *info, my_off_t filepos, byte *buf) } if (left_length < block_info.data_len || ! block_info.data_len) goto panic; /* Wrong linked record */ - if (info->s->file_read(info,(byte*) to,block_info.data_len,block_info.filepos, - MYF(MY_NABP))) - goto panic; - left_length-=block_info.data_len; - to+=block_info.data_len; + /* copy information that is already read */ + { + uint offset= (uint) (block_info.filepos - filepos); + uint prefetch_len= (sizeof(block_info.header) - offset); + filepos+= sizeof(block_info.header); + + if (prefetch_len > block_info.data_len) + prefetch_len= block_info.data_len; + if (prefetch_len) + { + memcpy((byte*) to, block_info.header + offset, prefetch_len); + block_info.data_len-= prefetch_len; + left_length-= prefetch_len; + to+= prefetch_len; + } + } + /* read rest of record from file */ + if (block_info.data_len) + { + if (info->opt_flag & WRITE_CACHE_USED && + info->rec_cache.pos_in_file < filepos + block_info.data_len && + flush_io_cache(&info->rec_cache)) + goto err; + /* + What a pity that this method is not called 'file_pread' and that + there is no equivalent without seeking. We are at the right + position already. :( + */ + if (info->s->file_read(info, (byte*) to, block_info.data_len, + filepos, MYF(MY_NABP))) + goto panic; + left_length-=block_info.data_len; + to+=block_info.data_len; + } + filepos= block_info.next_filepos; } while (left_length); info->update|= HA_STATE_AKTIV; /* We have a aktive record */ @@ -1520,11 +1577,45 @@ err: } +/* + Read record from datafile. + + SYNOPSIS + _ma_read_rnd_dynamic_record() + info MARIA_HA pointer to table. + buf Destination for record. + filepos From where to read the record. + skip_deleted_blocks If to repeat reading until a non-deleted + record is found. + + NOTE + + If a write buffer is active, it needs to be flushed if its contents + intersects with the record to read. We always check if the position + of the first byte of the write buffer is lower than the position + past the last byte to read. In theory this is also true if the write + buffer is completely below the read segment. That is, if there is no + intersection. But this case is unusual. We flush anyway. Only if the + first byte in the write buffer is above the last byte to read, we do + not flush. + + A dynamic record may need several reads. So this check must be done + before every read. Reading a dynamic record starts with reading the + block header. If the record does not fit into the free space of the + header, the block may be longer than the header. In this case a + second read is necessary. These one or two reads repeat for every + part of the record. + + RETURN + 0 OK + != 0 Error +*/ + int _ma_read_rnd_dynamic_record(MARIA_HA *info, byte *buf, register my_off_t filepos, my_bool skip_deleted_blocks) { - int flag,info_read,save_errno; + int block_of_record, info_read, save_errno; uint left_len,b_type; byte *to; MARIA_BLOCK_INFO block_info; @@ -1544,7 +1635,8 @@ int _ma_read_rnd_dynamic_record(MARIA_HA *info, byte *buf, else info_read=1; /* memory-keyinfoblock is ok */ - flag=block_info.second_read=0; + block_of_record= 0; /* First block of record is numbered as zero. */ + block_info.second_read= 0; left_len=1; do { @@ -1567,15 +1659,15 @@ int _ma_read_rnd_dynamic_record(MARIA_HA *info, byte *buf, { if (_ma_read_cache(&info->rec_cache,(byte*) block_info.header,filepos, sizeof(block_info.header), - (!flag && skip_deleted_blocks ? READING_NEXT : 0) | - READING_HEADER)) + (!block_of_record && skip_deleted_blocks ? + READING_NEXT : 0) | READING_HEADER)) goto panic; b_type= _ma_get_block_info(&block_info,-1,filepos); } else { if (info->opt_flag & WRITE_CACHE_USED && - info->rec_cache.pos_in_file <= filepos && + info->rec_cache.pos_in_file < filepos + MARIA_BLOCK_INFO_HEADER_LENGTH && flush_io_cache(&info->rec_cache)) DBUG_RETURN(my_errno); info->rec_cache.seek_not_done=1; @@ -1600,7 +1692,7 @@ int _ma_read_rnd_dynamic_record(MARIA_HA *info, byte *buf, } goto err; } - if (flag == 0) /* First block */ + if (block_of_record == 0) /* First block */ { if (block_info.rec_len > (uint) share->base.max_pack_length) goto panic; @@ -1642,11 +1734,17 @@ int _ma_read_rnd_dynamic_record(MARIA_HA *info, byte *buf, { if (_ma_read_cache(&info->rec_cache,(byte*) to,filepos, block_info.data_len, - (!flag && skip_deleted_blocks) ? READING_NEXT :0)) + (!block_of_record && skip_deleted_blocks) ? + READING_NEXT : 0)) goto panic; } else { + if (info->opt_flag & WRITE_CACHE_USED && + info->rec_cache.pos_in_file < + block_info.filepos + block_info.data_len && + flush_io_cache(&info->rec_cache)) + goto err; /* VOID(my_seek(info->dfile,filepos,MY_SEEK_SET,MYF(0))); */ if (my_read(info->dfile,(byte*) to,block_info.data_len,MYF(MY_NABP))) { @@ -1656,7 +1754,11 @@ int _ma_read_rnd_dynamic_record(MARIA_HA *info, byte *buf, } } } - if (flag++ == 0) + /* + Increment block-of-record counter. If it was the first block, + remember the position behind the block for the next call. + */ + if (block_of_record++ == 0) { info->nextpos=block_info.filepos+block_info.block_len; skip_deleted_blocks=0; @@ -1691,6 +1793,11 @@ uint _ma_get_block_info(MARIA_BLOCK_INFO *info, File file, my_off_t filepos) if (file >= 0) { + /* + We do not use my_pread() here because we want to have the file + pointer set to the end of the header after this function. + my_pread() may leave the file pointer untouched. + */ VOID(my_seek(file,filepos,MY_SEEK_SET,MYF(0))); if (my_read(file,(char*) header,sizeof(info->header),MYF(0)) != sizeof(info->header)) diff --git a/storage/maria/ma_extra.c b/storage/maria/ma_extra.c index d600fedb99b..57e540242b9 100644 --- a/storage/maria/ma_extra.c +++ b/storage/maria/ma_extra.c @@ -366,6 +366,11 @@ int maria_extra(MARIA_HA *info, enum ha_extra_function function, void *extra_arg pthread_mutex_unlock(&share->intern_lock); #endif break; + case HA_EXTRA_MARK_AS_LOG_TABLE: + pthread_mutex_lock(&share->intern_lock); + share->is_log_table= TRUE; + pthread_mutex_unlock(&share->intern_lock); + break; case HA_EXTRA_KEY_CACHE: case HA_EXTRA_NO_KEY_CACHE: default: diff --git a/storage/maria/ma_ft_parser.c b/storage/maria/ma_ft_parser.c index e5ec97b090d..24713c1344f 100644 --- a/storage/maria/ma_ft_parser.c +++ b/storage/maria/ma_ft_parser.c @@ -283,7 +283,7 @@ static int maria_ft_add_word(MYSQL_FTPARSER_PARAM *param, static int maria_ft_parse_internal(MYSQL_FTPARSER_PARAM *param, - byte *doc, int doc_len) + char *doc, int doc_len) { byte *end=doc+doc_len; MY_FT_PARSER_PARAM *ft_param=param->mysql_ftparam; diff --git a/storage/maria/ma_open.c b/storage/maria/ma_open.c index 38e71a44f8b..cf21ccb09e5 100644 --- a/storage/maria/ma_open.c +++ b/storage/maria/ma_open.c @@ -475,6 +475,7 @@ MARIA_HA *maria_open(const char *name, int mode, uint open_flags) share->data_file_type = DYNAMIC_RECORD; my_afree((gptr) disk_cache); _ma_setup_functions(share); + share->is_log_table= FALSE; #ifdef THREAD thr_lock_init(&share->lock); VOID(pthread_mutex_init(&share->intern_lock,MY_MUTEX_INIT_FAST)); diff --git a/storage/maria/ma_sort.c b/storage/maria/ma_sort.c index 5ae23c37261..795bfdb7fda 100644 --- a/storage/maria/ma_sort.c +++ b/storage/maria/ma_sort.c @@ -479,12 +479,6 @@ int _ma_thr_write_keys(MARIA_SORT_PARAM *sort_param) if (!got_error) { maria_set_key_active(share->state.key_map, sinfo->key); - if (param->testflag & T_STATISTICS) - maria_update_key_parts(sinfo->keyinfo, rec_per_key_part, sinfo->unique, - param->stats_method == MI_STATS_METHOD_IGNORE_NULLS? - sinfo->notnull: NULL, - (ulonglong) info->state->records); - if (!sinfo->buffpek.elements) { @@ -497,6 +491,11 @@ int _ma_thr_write_keys(MARIA_SORT_PARAM *sort_param) flush_maria_ft_buf(sinfo) || _ma_flush_pending_blocks(sinfo)) got_error=1; } + if (!got_error && param->testflag & T_STATISTICS) + maria_update_key_parts(sinfo->keyinfo, rec_per_key_part, sinfo->unique, + param->stats_method == MI_STATS_METHOD_IGNORE_NULLS? + sinfo->notnull: NULL, + (ulonglong) info->state->records); } my_free((gptr) sinfo->sort_keys,MYF(0)); my_free(_ma_get_rec_buff_ptr(info, sinfo->rec_buff), diff --git a/storage/maria/ma_update.c b/storage/maria/ma_update.c index 248b17ce2c9..b916b211159 100644 --- a/storage/maria/ma_update.c +++ b/storage/maria/ma_update.c @@ -171,7 +171,18 @@ int maria_update(register MARIA_HA *info, const byte *oldrec, byte *newrec) info->update= (HA_STATE_CHANGED | HA_STATE_ROW_CHANGED | HA_STATE_AKTIV | key_changed); - VOID(_ma_writeinfo(info,key_changed ? WRITEINFO_UPDATE_KEYFILE : 0)); + + /* + Every Maria function that updates Maria table must end with + call to _ma_writeinfo(). If operation (second param of + _ma_writeinfo()) is not 0 it sets share->changed to 1, that is + flags that data has changed. If operation is 0, this function + equals to no-op in this case. + + ma_update() must always pass !0 value as operation, since even if + there is no index change there could be data change. + */ + VOID(_ma_writeinfo(info, WRITEINFO_UPDATE_KEYFILE)); allow_break(); /* Allow SIGHUP & SIGINT */ if (info->invalidator != 0) { diff --git a/storage/maria/ma_write.c b/storage/maria/ma_write.c index 24768b36c89..c04a4a51eca 100644 --- a/storage/maria/ma_write.c +++ b/storage/maria/ma_write.c @@ -158,6 +158,18 @@ int maria_write(MARIA_HA *info, byte *record) (*info->invalidator)(info->filename); info->invalidator=0; } + + /* + Update status of the table. We need to do so after each row write + for the log tables, as we want the new row to become visible to + other threads as soon as possible. We lock mutex here to follow + pthread memory visibility rules. + */ + pthread_mutex_lock(&share->intern_lock); + if (share->is_log_table) + _ma_update_status((void*) info); + pthread_mutex_unlock(&share->intern_lock); + allow_break(); /* Allow SIGHUP & SIGINT */ DBUG_RETURN(0); diff --git a/storage/maria/maria_def.h b/storage/maria/maria_def.h index ecd93807a06..e0ba4bdb406 100644 --- a/storage/maria/maria_def.h +++ b/storage/maria/maria_def.h @@ -211,6 +211,9 @@ typedef struct st_maria_share uint blocksize; /* blocksize of keyfile */ myf write_flag; enum data_file_type data_file_type; + /* Below flag is needed to make log tables work with concurrent insert */ + my_bool is_log_table; + my_bool changed, /* If changed since lock */ global_changed, /* If changed since open */ not_flushed, temporary, delay_key_write, concurrent_insert; diff --git a/storage/maria/plug.in b/storage/maria/plug.in new file mode 100644 index 00000000000..de74293bd96 --- /dev/null +++ b/storage/maria/plug.in @@ -0,0 +1,7 @@ +MYSQL_STORAGE_ENGINE(maria, no, [Maria Storage Engine], + [Traditional transactional MySQL tables], [max,max-no-ndb]) +MYSQL_PLUGIN_DIRECTORY(maria, [storage/maria]) +MYSQL_PLUGIN_STATIC(maria, [libmaria.a]) +# Maria will probably go first into max builds, not all builds, +# so we don't declare it mandatory. + diff --git a/storage/myisam/Makefile.am b/storage/myisam/Makefile.am index e4766bfa05b..fb73635463d 100644 --- a/storage/myisam/Makefile.am +++ b/storage/myisam/Makefile.am @@ -99,7 +99,7 @@ libmyisam_a_SOURCES = mi_open.c mi_extra.c mi_info.c mi_rkey.c \ mi_keycache.c mi_preload.c \ ft_parser.c ft_stopwords.c ft_static.c \ ft_update.c ft_boolean_search.c ft_nlq_search.c \ - sort.c ha_myisam.cc \ + sort.c ha_myisam.cc ft_myisam.c \ rt_index.c rt_key.c rt_mbr.c rt_split.c sp_key.c CLEANFILES = test?.MY? FT?.MY? isam.log mi_test_all rt_test.MY? sp_test.MY? -- cgit v1.2.1 From 46b0384dc9dcacf567e094f590703102f27a9fcf Mon Sep 17 00:00:00 2001 From: unknown Date: Thu, 7 Sep 2006 18:40:10 +0200 Subject: a file to record known Maria bugs until we use a bugs db. BUGS_IN_MARIA.txt: a file to record our bugs until we use a bugs db --- BUGS_IN_MARIA.txt | 49 +++++++++++++++++++++++++++++++++++++++++++++++++ 1 file changed, 49 insertions(+) create mode 100644 BUGS_IN_MARIA.txt diff --git a/BUGS_IN_MARIA.txt b/BUGS_IN_MARIA.txt new file mode 100644 index 00000000000..e3a2894d399 --- /dev/null +++ b/BUGS_IN_MARIA.txt @@ -0,0 +1,49 @@ +This is our bugs database for now, until we switch to something +better. + +MARIABUG#1 +reported by Guilhem on 2006-09-07 +status: VERIFIED +========================================================== + +ma_test_all shows table corruption messages like this one: + +"error: key delete-link-chain corrupted" in table test2. + +For a simpler testcase extracted from ma_test_all, just run: + +rm -f test2.M*; ./ma_test2; ./maria_chk test2 + +While the similar MyISAM commands don't show any problem. + +You can make MyISAM show this same problem with just this patch: +===== myisam.h 1.80 vs edited ===== +*** /tmp/bk_myisam.h-1.80_O9fSEo 2006-09-07 18:30:28 +02:00 +--- edited/myisam.h 2006-09-07 18:25:54 +02:00 +*************** +*** 52,58 **** + #define MI_MAX_TEMP_LENGTH 2*1024L*1024L*1024L + + /* Possible values for myisam_block_size (must be power of 2) */ +! #define MI_KEY_BLOCK_LENGTH 1024 /* default key block length */ + #define MI_MIN_KEY_BLOCK_LENGTH 1024 /* Min key block length */ + #define MI_MAX_KEY_BLOCK_LENGTH 16384 + +--- 52,58 ---- + #define MI_MAX_TEMP_LENGTH 2*1024L*1024L*1024L + + /* Possible values for myisam_block_size (must be power of 2) */ +! #define MI_KEY_BLOCK_LENGTH 8192 /* default key block length */ + #define MI_MIN_KEY_BLOCK_LENGTH 1024 /* Min key block length */ + #define MI_MAX_KEY_BLOCK_LENGTH 16384 + +(Maria has 8192, and also 32768 for the max_key_block_length define). + +So it's not Maria-specific. + +I noticed that the corruption appears only for "recant>=64" (where +recant is a variable defined in mi_test2.c). + +For MyISAM we don't care as it's using 1024, but for Maria we have a +problem... +===================================================================== -- cgit v1.2.1 From e321f8eb29e8c3baf67cf0f745261dababf36877 Mon Sep 17 00:00:00 2001 From: unknown Date: Fri, 8 Sep 2006 15:37:02 +0200 Subject: There is a Maria category now in the bugs db (but only devs can see this category), so no need for this file. BitKeeper/deleted/.del-BUGS_IN_MARIA.txt: Delete: BUGS_IN_MARIA.txt --- BUGS_IN_MARIA.txt | 49 ------------------------------------------------- 1 file changed, 49 deletions(-) delete mode 100644 BUGS_IN_MARIA.txt diff --git a/BUGS_IN_MARIA.txt b/BUGS_IN_MARIA.txt deleted file mode 100644 index e3a2894d399..00000000000 --- a/BUGS_IN_MARIA.txt +++ /dev/null @@ -1,49 +0,0 @@ -This is our bugs database for now, until we switch to something -better. - -MARIABUG#1 -reported by Guilhem on 2006-09-07 -status: VERIFIED -========================================================== - -ma_test_all shows table corruption messages like this one: - -"error: key delete-link-chain corrupted" in table test2. - -For a simpler testcase extracted from ma_test_all, just run: - -rm -f test2.M*; ./ma_test2; ./maria_chk test2 - -While the similar MyISAM commands don't show any problem. - -You can make MyISAM show this same problem with just this patch: -===== myisam.h 1.80 vs edited ===== -*** /tmp/bk_myisam.h-1.80_O9fSEo 2006-09-07 18:30:28 +02:00 ---- edited/myisam.h 2006-09-07 18:25:54 +02:00 -*************** -*** 52,58 **** - #define MI_MAX_TEMP_LENGTH 2*1024L*1024L*1024L - - /* Possible values for myisam_block_size (must be power of 2) */ -! #define MI_KEY_BLOCK_LENGTH 1024 /* default key block length */ - #define MI_MIN_KEY_BLOCK_LENGTH 1024 /* Min key block length */ - #define MI_MAX_KEY_BLOCK_LENGTH 16384 - ---- 52,58 ---- - #define MI_MAX_TEMP_LENGTH 2*1024L*1024L*1024L - - /* Possible values for myisam_block_size (must be power of 2) */ -! #define MI_KEY_BLOCK_LENGTH 8192 /* default key block length */ - #define MI_MIN_KEY_BLOCK_LENGTH 1024 /* Min key block length */ - #define MI_MAX_KEY_BLOCK_LENGTH 16384 - -(Maria has 8192, and also 32768 for the max_key_block_length define). - -So it's not Maria-specific. - -I noticed that the corruption appears only for "recant>=64" (where -recant is a variable defined in mi_test2.c). - -For MyISAM we don't care as it's using 1024, but for Maria we have a -problem... -===================================================================== -- cgit v1.2.1 From 26fb36067a97b21e12fc2704fd08f9ddfd231ded Mon Sep 17 00:00:00 2001 From: unknown Date: Mon, 11 Sep 2006 16:12:31 +0200 Subject: WL#3234 Maria control file manager. Fitting ma_control_file_test into the mytap unittest framework: new directories: - unittest/storage/ for unit tests of any storage engine - unittest/storage/maria for ... Maria, containing ma_control_file-t. Later, older tests like ma_test*, ma_test_all (but which is Unix dependent in its current form) could move here too. The plugins macro enable building of unittest/storage/X for any enabled engine X which has such a directory. If Falcon wants to have unit tests there too, I may have to merge this patch into 5.x one day. config/ac-macros/plugins.m4: If a storage engine has a directory in unittest/storage, build this directory. configure.in: build storage engines' unit tests. storage/maria/Makefile.am: ma_control_file_test moves to unittest/storage/maria storage/maria/ma_control_file.c: more error codes when opening the control file fails. ma_control_file_end() may now return an error if my_close() failed. storage/maria/ma_control_file.h: more error codes when opening the control file fails. unittest/Makefile.am: adding unit tests for storage engines. Note that unit.pl simply recurses into "storage", so if a unit test for storage engine X has been built previously, and now you re-configure (without making clean) to disable this engine, then the unit test of X will not be rebuilt but will still be present in storage/X, so will be run. unittest/storage/maria/ma_control_file-t.c: Making the test fit the mytap framework (return all the way up the stack instead of assert(); use the mytap functions plan(), ok() etc). Adding test of file too short/long. unittest/storage/maria/Makefile.am: a_control_file-t is added to the Maria unit tests. Later, older tests (ma_test1 etc) could also move here. unittest/storage/Makefile.am: New BitKeeper file ``unittest/storage/Makefile.am'' --- config/ac-macros/plugins.m4 | 20 +- configure.in | 1 + storage/maria/Makefile.am | 10 +- storage/maria/ma_control_file.c | 77 +++-- storage/maria/ma_control_file.h | 15 +- storage/maria/ma_control_file_test.c | 312 -------------------- unittest/Makefile.am | 4 +- unittest/storage/Makefile.am | 27 ++ unittest/storage/maria/Makefile.am | 29 ++ unittest/storage/maria/ma_control_file-t.c | 448 +++++++++++++++++++++++++++++ 10 files changed, 592 insertions(+), 351 deletions(-) delete mode 100644 storage/maria/ma_control_file_test.c create mode 100644 unittest/storage/Makefile.am create mode 100644 unittest/storage/maria/Makefile.am create mode 100644 unittest/storage/maria/ma_control_file-t.c diff --git a/config/ac-macros/plugins.m4 b/config/ac-macros/plugins.m4 index cfc5f8dbcbe..791405f8fbd 100644 --- a/config/ac-macros/plugins.m4 +++ b/config/ac-macros/plugins.m4 @@ -280,6 +280,7 @@ AC_DEFUN([MYSQL_CONFIGURE_PLUGINS],[ _MYSQL_EMIT_PLUGIN_ACTIONS(m4_bpatsubst(__mysql_plugin_list__, :, [,])) AC_SUBST([mysql_se_dirs]) AC_SUBST([mysql_pg_dirs]) + AC_SUBST([mysql_se_unittest_dirs]) ]) ]) ]) @@ -315,6 +316,7 @@ AC_DEFUN([__MYSQL_EMIT_CHECK_PLUGIN],[ ]) AC_MSG_CHECKING([whether to use ]$3) mysql_use_plugin_dir="" + mysql_use_plugin_unittest_dir="" m4_ifdef([$10],[ if test "X[$mysql_plugin_]$2" = Xyes -a \ "X[$with_plugin_]$2" != Xno -o \ @@ -407,10 +409,24 @@ dnl Although this is "pretty", it breaks libmysqld build m4_syscmd(test -f "$6/configure") ifelse(m4_sysval, 0, [AC_CONFIG_SUBDIRS($6)], - [AC_CONFIG_FILES($6/Makefile)] + [ + AC_CONFIG_FILES($6/Makefile) + m4_syscmd(test -d "unittest/$6") + ifelse(m4_sysval, 0, + [ + mysql_use_plugin_unittest_dir="$6" + AC_CONFIG_FILES(unittest/$6/Makefile) + ], []) + ] ) ifelse(m4_substr($6, 0, 8), [storage/], - [mysql_se_dirs="$mysql_se_dirs ]m4_substr($6, 8)", + [ + [mysql_se_name="]m4_substr($6, 8)" + mysql_se_dirs="$mysql_se_dirs $mysql_se_name" + if test -n "$mysql_use_plugin_unittest_dir" ; then + mysql_se_unittest_dirs="$mysql_se_unitest_dirs $mysql_se_name" + fi + ], m4_substr($6, 0, 7), [plugin/], [mysql_pg_dirs="$mysql_pg_dirs ]m4_substr($6, 7)", [AC_FATAL([don't know how to handle plugin dir ]$6)]) diff --git a/configure.in b/configure.in index f0cdcde1abb..61c2deb5b5e 100644 --- a/configure.in +++ b/configure.in @@ -2482,6 +2482,7 @@ AC_SUBST(MAKE_BINARY_DISTRIBUTION_OPTIONS) AC_CONFIG_FILES(Makefile extra/Makefile mysys/Makefile dnl unittest/Makefile unittest/mytap/Makefile unittest/mytap/t/Makefile dnl unittest/mysys/Makefile unittest/examples/Makefile dnl + unittest/storage/Makefile dnl strings/Makefile regex/Makefile storage/Makefile dnl man/Makefile BUILD/Makefile vio/Makefile dnl libmysql/Makefile client/Makefile dnl diff --git a/storage/maria/Makefile.am b/storage/maria/Makefile.am index f5678a55266..10204226742 100644 --- a/storage/maria/Makefile.am +++ b/storage/maria/Makefile.am @@ -47,7 +47,7 @@ maria_pack_LDADD= @CLIENT_EXTRA_LDFLAGS@ libmaria.a \ $(top_builddir)/mysys/libmysys.a \ $(top_builddir)/dbug/libdbug.a \ $(top_builddir)/strings/libmystrings.a @ZLIB_LIBS@ -noinst_PROGRAMS = ma_test1 ma_test2 ma_test3 ma_rt_test ma_sp_test ma_control_file_test +noinst_PROGRAMS = ma_test1 ma_test2 ma_test3 ma_rt_test ma_sp_test noinst_HEADERS = maria_def.h ma_rt_index.h ma_rt_key.h ma_rt_mbr.h \ ma_sp_defs.h ma_fulltext.h ma_ftdefs.h ma_ft_test1.h ma_ft_eval.h \ ma_control_file.h ha_maria.h @@ -89,12 +89,6 @@ ma_sp_test_LDADD= @CLIENT_EXTRA_LDFLAGS@ libmaria.a \ $(top_builddir)/mysys/libmysys.a \ $(top_builddir)/dbug/libdbug.a \ $(top_builddir)/strings/libmystrings.a @ZLIB_LIBS@ -ma_control_file_test_DEPENDENCIES= $(LIBRARIES) -ma_control_file_test_LDADD= @CLIENT_EXTRA_LDFLAGS@ libmaria.a \ - $(top_builddir)/storage/myisam/libmyisam.a \ - $(top_builddir)/mysys/libmysys.a \ - $(top_builddir)/dbug/libdbug.a \ - $(top_builddir)/strings/libmystrings.a @ZLIB_LIBS@ libmaria_a_SOURCES = ma_init.c ma_open.c ma_extra.c ma_info.c ma_rkey.c \ ma_rnext.c ma_rnext_same.c \ ma_search.c ma_page.c ma_key.c ma_locking.c \ @@ -113,7 +107,7 @@ libmaria_a_SOURCES = ma_init.c ma_open.c ma_extra.c ma_info.c ma_rkey.c \ ha_maria.cc \ ma_rt_index.c ma_rt_key.c ma_rt_mbr.c ma_rt_split.c \ ma_sp_key.c ma_control_file.c -CLEANFILES = test?.MA? FT?.MA? isam.log ma_test_all ma_rt_test.MA? sp_test.MA? maria_control +CLEANFILES = test?.MA? FT?.MA? isam.log ma_test_all ma_rt_test.MA? sp_test.MA? SUFFIXES = .sh diff --git a/storage/maria/ma_control_file.c b/storage/maria/ma_control_file.c index 5861246baf9..5fbb0a084df 100644 --- a/storage/maria/ma_control_file.c +++ b/storage/maria/ma_control_file.c @@ -41,7 +41,7 @@ uint32 last_logno; Control file is less then 512 bytes (a disk sector), to be as atomic as possible */ -static int control_file_fd; +static int control_file_fd= -1; static void lsn8store(char *buffer, const LSN *lsn) { @@ -87,15 +87,16 @@ static char simple_checksum(char *buffer, uint size) RETURN 0 - OK - 1 - Error + 1 - Error (in which case the file is left closed) */ -int ma_control_file_create_or_open() +CONTROL_FILE_ERROR ma_control_file_create_or_open() { char buffer[CONTROL_FILE_SIZE]; char name[FN_REFLEN]; MY_STAT stat_buff; my_bool create_file; int open_flags= O_BINARY | /*O_DIRECT |*/ O_RDWR; + int error= CONTROL_FILE_UNKNOWN_ERROR; DBUG_ENTER("ma_control_file_create_or_open"); /* @@ -106,16 +107,19 @@ int ma_control_file_create_or_open() DBUG_ASSERT(CONTROL_FILE_LSN_SIZE == (4+4)); DBUG_ASSERT(CONTROL_FILE_FILENO_SIZE == 4); - /* name is concatenation of Maria's home dir and "control" */ - if (fn_format(name, "control", maria_data_root, "", MYF(MY_WME)) == NullS) - DBUG_RETURN(1); + if (control_file_fd >= 0) /* already open */ + DBUG_RETURN(0); + + if (fn_format(name, CONTROL_FILE_BASE_NAME, + maria_data_root, "", MYF(MY_WME)) == NullS) + DBUG_RETURN(CONTROL_FILE_UNKNOWN_ERROR); create_file= test(my_access(name,F_OK)); if (create_file) { if ((control_file_fd= my_create(name, 0, open_flags, MYF(0))) < 0) - DBUG_RETURN(1); + DBUG_RETURN(CONTROL_FILE_UNKNOWN_ERROR); /* TODO: from "man fsync" on Linux: "fsync does not necessarily ensure that the entry in the directory @@ -127,10 +131,10 @@ int ma_control_file_create_or_open() To be safer we should make sure that there are no logs or data/index files around (indeed it could be that the control file alone was deleted or not restored, and we should not go on with life at this point). - + TODO: For now we trust (this is alpha version), but for beta if would be great to verify. - + We could have a tool which can rebuild the control file, by reading the directory of logs, finding the newest log, reading it to find last checkpoint... Slow but can save your db. @@ -138,7 +142,7 @@ int ma_control_file_create_or_open() LSN imposs_lsn= CONTROL_FILE_IMPOSSIBLE_LSN; uint32 imposs_logno= CONTROL_FILE_IMPOSSIBLE_FILENO; - + /* init the file with these "undefined" values */ DBUG_RETURN(ma_control_file_write_and_force(&imposs_lsn, imposs_logno, CONTROL_FILE_UPDATE_ALL)); @@ -147,12 +151,12 @@ int ma_control_file_create_or_open() /* Otherwise, file exists */ if ((control_file_fd= my_open(name, open_flags, MYF(MY_WME))) < 0) - DBUG_RETURN(1); - + goto err; + if (my_stat(name, &stat_buff, MYF(MY_WME)) == NULL) - DBUG_RETURN(1); + goto err; - if ((uint)stat_buff.st_size != CONTROL_FILE_SIZE) + if ((uint)stat_buff.st_size < CONTROL_FILE_SIZE) { /* Given that normally we write only a sector and it's atomic, the only @@ -165,31 +169,43 @@ int ma_control_file_create_or_open() disk/filesystem has a problem. So let's be rigid. */ - my_message(0, "wrong file size", MYF(0)); /* TODO: improve errors */ - my_error(HA_ERR_CRASHED, MYF(0), name); - DBUG_RETURN(1); + my_message(0, "too small file", MYF(0)); /* TODO: improve errors */ + error= CONTROL_FILE_TOO_SMALL; + goto err; + } + + if ((uint)stat_buff.st_size > CONTROL_FILE_SIZE) + { + my_message(0, "too big file", MYF(0)); /* TODO: improve errors */ + error= CONTROL_FILE_TOO_BIG; + goto err; } if (my_read(control_file_fd, buffer, CONTROL_FILE_SIZE, MYF(MY_FNABP | MY_WME))) - DBUG_RETURN(1); + goto err; if (memcmp(buffer + CONTROL_FILE_MAGIC_STRING_OFFSET, CONTROL_FILE_MAGIC_STRING, CONTROL_FILE_MAGIC_STRING_SIZE)) { my_message(0, "bad magic string", MYF(0)); - DBUG_RETURN(1); + error= CONTROL_FILE_BAD_MAGIC_STRING; + goto err; } if (simple_checksum(buffer + CONTROL_FILE_LSN_OFFSET, CONTROL_FILE_SIZE - CONTROL_FILE_LSN_OFFSET) != buffer[CONTROL_FILE_CHECKSUM_OFFSET]) { my_message(0, "checksum mismatch", MYF(0)); - DBUG_RETURN(1); + error= CONTROL_FILE_BAD_CHECKSUM; + goto err; } last_checkpoint_lsn= lsn8korr(buffer + CONTROL_FILE_LSN_OFFSET); last_logno= uint4korr(buffer + CONTROL_FILE_FILENO_OFFSET); DBUG_RETURN(0); +err: + ma_control_file_end(); + DBUG_RETURN(error); } @@ -227,6 +243,8 @@ int ma_control_file_write_and_force(const LSN *checkpoint_lsn, uint32 logno, my_bool update_checkpoint_lsn= FALSE, update_logno= FALSE; DBUG_ENTER("ma_control_file_write_and_force"); + DBUG_ASSERT(control_file_fd >= 0); /* must be open */ + memcpy(buffer + CONTROL_FILE_MAGIC_STRING_OFFSET, CONTROL_FILE_MAGIC_STRING, CONTROL_FILE_MAGIC_STRING_SIZE); @@ -259,7 +277,7 @@ int ma_control_file_write_and_force(const LSN *checkpoint_lsn, uint32 logno, 0, MYF(MY_FNABP | MY_WME)) || my_sync(control_file_fd, MYF(MY_WME))) DBUG_RETURN(1); - + /* TODO: you need some protection to be able to write last_* global vars */ if (update_checkpoint_lsn) last_checkpoint_lsn= *checkpoint_lsn; @@ -277,15 +295,26 @@ int ma_control_file_write_and_force(const LSN *checkpoint_lsn, uint32 logno, ma_control_file_end() */ -void ma_control_file_end() +int ma_control_file_end() { + int close_error; DBUG_ENTER("ma_control_file_end"); - my_close(control_file_fd, MYF(MY_WME)); + + if (control_file_fd < 0) /* already closed */ + DBUG_RETURN(0); + + close_error= my_close(control_file_fd, MYF(MY_WME)); + /* + As my_close() frees structures even if close() fails, we do the same, + i.e. we mark the file as closed in all cases. + */ + control_file_fd= -1; /* As this module owns these variables, closing the module forbids access to them (just a safety): */ last_checkpoint_lsn= CONTROL_FILE_IMPOSSIBLE_LSN; last_logno= CONTROL_FILE_IMPOSSIBLE_FILENO; - DBUG_VOID_RETURN; + + DBUG_RETURN(close_error); } diff --git a/storage/maria/ma_control_file.h b/storage/maria/ma_control_file.h index 5f5581137b7..5ac6f158183 100644 --- a/storage/maria/ma_control_file.h +++ b/storage/maria/ma_control_file.h @@ -24,6 +24,7 @@ typedef struct st_lsn { #define maria_data_root "." #endif +#define CONTROL_FILE_BASE_NAME "maria_control" /* indicate absence of the log file number; first log is always number 1, 0 is impossible. @@ -55,7 +56,15 @@ extern uint32 last_logno; If present, read it to find out last checkpoint's LSN and last log. Called at engine's start. */ -int ma_control_file_create_or_open(); +typedef enum enum_control_file_error { + CONTROL_FILE_OK= 0, + CONTROL_FILE_TOO_SMALL, + CONTROL_FILE_TOO_BIG, + CONTROL_FILE_BAD_MAGIC_STRING, + CONTROL_FILE_BAD_CHECKSUM, + CONTROL_FILE_UNKNOWN_ERROR /* any other error */ +} CONTROL_FILE_ERROR; +CONTROL_FILE_ERROR ma_control_file_create_or_open(); /* Write information durably to the control file. @@ -66,10 +75,10 @@ int ma_control_file_create_or_open(); #define CONTROL_FILE_UPDATE_ONLY_LSN 1 #define CONTROL_FILE_UPDATE_ONLY_LOGNO 2 int ma_control_file_write_and_force(const LSN *checkpoint_lsn, uint32 logno, - uint objs_to_write); + uint objs_to_write); /* Free resources taken by control file subsystem */ -void ma_control_file_end(); +int ma_control_file_end(); #endif diff --git a/storage/maria/ma_control_file_test.c b/storage/maria/ma_control_file_test.c deleted file mode 100644 index b99c61da4fb..00000000000 --- a/storage/maria/ma_control_file_test.c +++ /dev/null @@ -1,312 +0,0 @@ -/* Copyright (C) 2006 MySQL AB & MySQL Finland AB & TCX DataKonsult AB - - 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; either version 2 of the License, or - (at your option) any later version. - - 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ - -/* Unit test of the control file module of the Maria engine */ - -/* TODO: make it fit the mytap framework */ - -/* - Note that it is not possible to test the durability of the write (can't - pull the plug programmatically :) -*/ - -#include "maria.h" -#include "ma_control_file.h" -#include - -char file_name[FN_REFLEN]; -int fd= -1; - -static void clean_files(); -static void run_test_normal(); -static void run_test_abnormal(); -static void usage(); -static void get_options(int argc, char *argv[]); - -int main(int argc,char *argv[]) -{ - MY_INIT(argv[0]); - - get_options(argc,argv); - - clean_files(); - run_test_normal(); - run_test_abnormal(); - - fprintf(stderr, "All tests succeeded\n"); - exit(0); /* all ok, if some test failed, we will have aborted */ -} - -/* - Abort unless given expression is non-zero. - - SYNOPSIS - DIE_UNLESS(expr) - - DESCRIPTION - We can't use any kind of system assert as we need to - preserve tested invariants in release builds as well. - - NOTE - This is infamous copy-paste from mysql_client_test.c; - we should instead put it in some include in one single place. -*/ - -#define DIE_UNLESS(expr) \ - ((void) ((expr) ? 0 : (die(__FILE__, __LINE__, #expr), 0))) -#define DIE_IF(expr) \ - ((void) (!(expr) ? 0 : (die(__FILE__, __LINE__, #expr), 0))) -#define DIE(expr) \ - die(__FILE__, __LINE__, #expr) - -void die(const char *file, int line, const char *expr) -{ - fprintf(stderr, "%s:%d: check failed: '%s'\n", file, line, expr); - abort(); -} - - -static void clean_files() -{ - DIE_IF(fn_format(file_name, "control", maria_data_root, "", MYF(MY_WME)) == - NullS); - my_delete(file_name, MYF(0)); /* maybe file does not exist, ignore error */ -} - - -static void run_test_normal() -{ - LSN checkpoint_lsn; - uint32 logno; - uint objs_to_write; - uint i; - char buffer[17]; - - /* TEST0: Instance starts from scratch (control file does not exist) */ - DIE_UNLESS(ma_control_file_create_or_open() == 0); - /* Check that the module reports no information */ - DIE_UNLESS(last_logno == CONTROL_FILE_IMPOSSIBLE_FILENO); - DIE_UNLESS(last_checkpoint_lsn.file_no == CONTROL_FILE_IMPOSSIBLE_FILENO); - DIE_UNLESS(last_checkpoint_lsn.rec_offset == CONTROL_FILE_IMPOSSIBLE_LOG_OFFSET); - - /* TEST1: Simulate creation of one log */ - - objs_to_write= CONTROL_FILE_UPDATE_ONLY_LOGNO; - logno= 123; - DIE_UNLESS(ma_control_file_write_and_force(NULL, logno, - objs_to_write) == 0); - /* Check that last_logno was updated */ - DIE_UNLESS(last_logno == logno); - /* Simulate shutdown */ - ma_control_file_end(); - /* Verify amnesia */ - DIE_UNLESS(last_logno == CONTROL_FILE_IMPOSSIBLE_FILENO); - DIE_UNLESS(last_checkpoint_lsn.file_no == CONTROL_FILE_IMPOSSIBLE_FILENO); - DIE_UNLESS(last_checkpoint_lsn.rec_offset == CONTROL_FILE_IMPOSSIBLE_LOG_OFFSET); - /* And restart */ - DIE_UNLESS(ma_control_file_create_or_open() == 0); - DIE_UNLESS(last_logno == logno); - - /* TEST2: Simulate creation of 5 logs */ - - objs_to_write= CONTROL_FILE_UPDATE_ONLY_LOGNO; - logno= 100; - for (i= 0; i<5; i++) - { - logno*= 3; - DIE_UNLESS(ma_control_file_write_and_force(NULL, logno, - objs_to_write) == 0); - } - ma_control_file_end(); - DIE_UNLESS(last_logno == CONTROL_FILE_IMPOSSIBLE_FILENO); - DIE_UNLESS(last_checkpoint_lsn.file_no == CONTROL_FILE_IMPOSSIBLE_FILENO); - DIE_UNLESS(last_checkpoint_lsn.rec_offset == CONTROL_FILE_IMPOSSIBLE_LOG_OFFSET); - DIE_UNLESS(ma_control_file_create_or_open() == 0); - DIE_UNLESS(last_logno == logno); - - /* - TEST3: Simulate one checkpoint, one log creation, two checkpoints, one - log creation. - */ - - objs_to_write= CONTROL_FILE_UPDATE_ONLY_LSN; - checkpoint_lsn= (LSN){5, 10000}; - logno= 10; - DIE_UNLESS(ma_control_file_write_and_force(&checkpoint_lsn, logno, - objs_to_write) == 0); - /* check that last_logno was not updated */ - DIE_UNLESS(last_logno != logno); - /* Check that last_checkpoint_lsn was updated */ - DIE_UNLESS(last_checkpoint_lsn.file_no == checkpoint_lsn.file_no); - DIE_UNLESS(last_checkpoint_lsn.rec_offset == checkpoint_lsn.rec_offset); - - objs_to_write= CONTROL_FILE_UPDATE_ONLY_LOGNO; - checkpoint_lsn= (LSN){5, 20000}; - logno= 17; - DIE_UNLESS(ma_control_file_write_and_force(&checkpoint_lsn, logno, - objs_to_write) == 0); - /* Check that checkpoint LSN was not updated */ - DIE_UNLESS(last_checkpoint_lsn.rec_offset != checkpoint_lsn.rec_offset); - objs_to_write= CONTROL_FILE_UPDATE_ONLY_LSN; - checkpoint_lsn= (LSN){17, 20000}; - DIE_UNLESS(ma_control_file_write_and_force(&checkpoint_lsn, logno, - objs_to_write) == 0); - objs_to_write= CONTROL_FILE_UPDATE_ONLY_LSN; - checkpoint_lsn= (LSN){17, 45000}; - DIE_UNLESS(ma_control_file_write_and_force(&checkpoint_lsn, logno, - objs_to_write) == 0); - objs_to_write= CONTROL_FILE_UPDATE_ONLY_LOGNO; - logno= 19; - DIE_UNLESS(ma_control_file_write_and_force(&checkpoint_lsn, logno, - objs_to_write) == 0); - - ma_control_file_end(); - DIE_UNLESS(last_logno == CONTROL_FILE_IMPOSSIBLE_FILENO); - DIE_UNLESS(last_checkpoint_lsn.file_no == CONTROL_FILE_IMPOSSIBLE_FILENO); - DIE_UNLESS(last_checkpoint_lsn.rec_offset == CONTROL_FILE_IMPOSSIBLE_LOG_OFFSET); - DIE_UNLESS(ma_control_file_create_or_open() == 0); - DIE_UNLESS(last_logno == logno); - DIE_UNLESS(last_checkpoint_lsn.file_no == checkpoint_lsn.file_no); - DIE_UNLESS(last_checkpoint_lsn.rec_offset == checkpoint_lsn.rec_offset); - - /* - TEST4: actually check by ourselves the content of the file. - Note that constants (offsets) are hard-coded here, precisely to prevent - someone from changing them in the control file module and breaking - backward-compatibility. - TODO: when we reach the format-freeze state, we may even just do a - comparison with a raw binary string, to not depend on any uint4korr - future change/breakage. - */ - - DIE_IF((fd= my_open(file_name, - O_BINARY | O_RDWR, - MYF(MY_WME))) < 0); - DIE_IF(my_read(fd, buffer, 17, MYF(MY_FNABP | MY_WME)) != 0); - DIE_IF(my_close(fd, MYF(MY_WME)) != 0); - i= uint4korr(buffer+5); - DIE_UNLESS(i == last_checkpoint_lsn.file_no); - i= uint4korr(buffer+9); - DIE_UNLESS(i == last_checkpoint_lsn.rec_offset); - i= uint4korr(buffer+13); - DIE_UNLESS(i == last_logno); - - - /* TEST5: Simulate stop/start/nothing/stop/start */ - - ma_control_file_end(); - DIE_UNLESS(last_logno == CONTROL_FILE_IMPOSSIBLE_FILENO); - DIE_UNLESS(ma_control_file_create_or_open() == 0); - ma_control_file_end(); - DIE_UNLESS(last_logno == CONTROL_FILE_IMPOSSIBLE_FILENO); - DIE_UNLESS(ma_control_file_create_or_open() == 0); - DIE_UNLESS(last_logno == logno); - DIE_UNLESS(last_checkpoint_lsn.file_no == checkpoint_lsn.file_no); - DIE_UNLESS(last_checkpoint_lsn.rec_offset == checkpoint_lsn.rec_offset); - -} - -static void run_test_abnormal() -{ - char buffer[4]; - /* Corrupt the control file */ - DIE_IF((fd= my_open(file_name, - O_BINARY | O_RDWR, - MYF(MY_WME))) < 0); - DIE_IF(my_pread(fd, buffer, 4, 0, MYF(MY_FNABP | MY_WME)) != 0); - DIE_IF(my_pwrite(fd, "papa", 4, 0, MYF(MY_FNABP | MY_WME)) != 0); - DIE_IF(my_close(fd, MYF(MY_WME)) != 0); - - /* Check that control file module sees the problem */ - DIE_IF(ma_control_file_create_or_open() == 0); - - /* Restore it and corrupt it differently */ - DIE_IF((fd= my_open(file_name, - O_BINARY | O_RDWR, - MYF(MY_WME))) < 0); - /* Restore magic string */ - DIE_IF(my_pwrite(fd, buffer, 4, 0, MYF(MY_FNABP | MY_WME)) != 0); - DIE_IF(my_pread(fd, buffer, 1, 4, MYF(MY_FNABP | MY_WME)) != 0); - buffer[1]= buffer[0]+3; /* mangle checksum */ - DIE_IF(my_pwrite(fd, buffer+1, 1, 4, MYF(MY_FNABP | MY_WME)) != 0); - DIE_IF(my_close(fd, MYF(MY_WME)) != 0); - - /* Check that control file module sees the problem */ - DIE_IF(ma_control_file_create_or_open() == 0); - - /* Note that control file is left corrupted at this point */ -} - - -static struct my_option my_long_options[] = -{ -#ifndef DBUG_OFF - {"debug", '#', "Debug log.", - 0, 0, 0, GET_STR, REQUIRED_ARG, 0, 0, 0, 0, 0, 0}, -#endif - {"help", '?', "Display help and exit", - 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, - {"version", 'V', "Print version number and exit", - 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, - { 0, 0, 0, 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0} -}; - - -static void version() -{ - printf("ma_control_file_test: unit test for the control file " - "module of the Maria storage engine. Ver 1.0 \n"); -} - -static my_bool -get_one_option(int optid, const struct my_option *opt __attribute__((unused)), - char *argument) -{ - switch(optid) { - case 'V': - version(); - exit(0); - case '#': - DBUG_PUSH (argument); - break; - case '?': - version(); - usage(); - exit(0); - } - return 0; -} - - -/* Read options */ - -static void get_options(int argc, char *argv[]) -{ - int ho_error; - - if ((ho_error=handle_options(&argc, &argv, my_long_options, get_one_option))) - exit(ho_error); - - return; -} /* get options */ - - -static void usage() -{ - printf("Usage: %s [options]\n\n", my_progname); - my_print_help(my_long_options); - my_print_variables(my_long_options); -} diff --git a/unittest/Makefile.am b/unittest/Makefile.am index ca3291efde0..176ac020bea 100644 --- a/unittest/Makefile.am +++ b/unittest/Makefile.am @@ -1,10 +1,10 @@ -SUBDIRS = mytap . mysys examples +SUBDIRS = mytap . mysys storage examples noinst_SCRIPTS = unit EXTRA_DIST = unit.pl CLEANFILES = unit -unittests = mytap mysys +unittests = mytap mysys storage test: unit ./unit run $(unittests) diff --git a/unittest/storage/Makefile.am b/unittest/storage/Makefile.am new file mode 100644 index 00000000000..21de2de75fc --- /dev/null +++ b/unittest/storage/Makefile.am @@ -0,0 +1,27 @@ +# Copyright (C) 2000 MySQL AB & MySQL Finland AB & TCX DataKonsult AB +# +# 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; either version 2 of the License, or +# (at your option) any later version. +# +# 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA + +# Process this file with automake to create Makefile.in + +AUTOMAKE_OPTIONS = foreign + +# These are built from source in the Docs directory +EXTRA_DIST = +# Cannot use @mysql_se_dirs@ as not all engines have unit tests here +SUBDIRS = @mysql_se_unittest_dirs@ + +# Don't update the files from bitkeeper +%::SCCS/s.% diff --git a/unittest/storage/maria/Makefile.am b/unittest/storage/maria/Makefile.am new file mode 100644 index 00000000000..eae2990aea9 --- /dev/null +++ b/unittest/storage/maria/Makefile.am @@ -0,0 +1,29 @@ +# Copyright (C) 2000 MySQL AB & MySQL Finland AB & TCX DataKonsult AB +# +# 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; either version 2 of the License, or +# (at your option) any later version. +# +# 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA + +AM_CPPFLAGS = @ZLIB_INCLUDES@ -I$(top_builddir)/include +AM_CPPFLAGS += -I$(top_srcdir)/include -I$(top_srcdir)/unittest/mytap + +# Only reason to link with libmyisam.a here is that it's where some fulltext +# pieces are (but soon we'll remove fulltext dependencies from Maria). +LDADD= $(top_builddir)/unittest/mytap/libmytap.a \ + $(top_builddir)/storage/maria/libmaria.a \ + $(top_builddir)/storage/myisam/libmyisam.a \ + $(top_builddir)/mysys/libmysys.a \ + $(top_builddir)/dbug/libdbug.a \ + $(top_builddir)/strings/libmystrings.a @ZLIB_LIBS@ +noinst_PROGRAMS = ma_control_file-t +CLEANFILES = maria_control diff --git a/unittest/storage/maria/ma_control_file-t.c b/unittest/storage/maria/ma_control_file-t.c new file mode 100644 index 00000000000..3ea6932c754 --- /dev/null +++ b/unittest/storage/maria/ma_control_file-t.c @@ -0,0 +1,448 @@ +/* Copyright (C) 2006 MySQL AB & MySQL Finland AB & TCX DataKonsult AB + + 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; either version 2 of the License, or + (at your option) any later version. + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ + +/* Unit test of the control file module of the Maria engine WL#3234 */ + +/* + Note that it is not possible to test the durability of the write (can't + pull the plug programmatically :) +*/ + +#include +#include +#include + +#ifndef WITH_MARIA_STORAGE_ENGINE +/* + If Maria is not compiled in, normally we don't come to building this test. +*/ +#error "Maria engine is not compiled in, test cannot be built" +#endif + +#include "maria.h" +#include "../../../storage/maria/ma_control_file.h" +#include + +char file_name[FN_REFLEN]; + +/* The values we'll set and expect the control file module to return */ +LSN expect_checkpoint_lsn; +uint32 expect_logno; + +static int delete_file(); +/* + Those are test-specific wrappers around the module's API functions: after + calling the module's API functions they perform checks on the result. +*/ +static int close_file(); /* wraps ma_control_file_end */ +static int create_or_open_file(); /* wraps ma_control_file_open_or_create */ +static int write_file(); /* wraps ma_control_file_write_and_force */ + +/* Tests */ +static int test_one_log(); +static int test_five_logs(); +static int test_3_checkpoints_and_2_logs(); +static int test_binary_content(); +static int test_start_stop(); +static int test_2_open_and_2_close(); +static int test_bad_magic_string(); +static int test_bad_checksum(); +static int test_bad_size(); + +/* Utility */ +static int verify_module_values_match_expected(); +static int verify_module_values_are_impossible(); +static void usage(); +static void get_options(int argc, char *argv[]); + +/* + If "expr" is FALSE, this macro will make the function print a diagnostic + message and immediately return 1. + This is inspired from assert() but does not crash the binary (sometimes we + may want to see how other tests go even if one fails). + RET_ERR means "return error". +*/ + +#define RET_ERR_UNLESS(expr) \ + {if (!(expr)) {diag("line %d: failure: '%s'", __LINE__, #expr); return 1;}} + + +int main(int argc,char *argv[]) +{ + MY_INIT(argv[0]); + + plan(9); + + diag("Unit tests for control file"); + + get_options(argc,argv); + + diag("Deleting control file at startup, if there is an old one"); + RET_ERR_UNLESS(0 == delete_file()); /* if fails, can't continue */ + + diag("Tests of normal conditions"); + ok(0 == test_one_log(), "test of creating one log"); + ok(0 == test_five_logs(), "test of creating five logs"); + ok(0 == test_3_checkpoints_and_2_logs(), + "test of creating three checkpoints and two logs"); + ok(0 == test_binary_content(), "test of the binary content of the file"); + ok(0 == test_start_stop(), "test of multiple starts and stops"); + diag("Tests of abnormal conditions"); + ok(0 == test_2_open_and_2_close(), + "test of two open and two close (strange call sequence)"); + ok(0 == test_bad_magic_string(), "test of bad magic string"); + ok(0 == test_bad_checksum(), "test of bad checksum"); + ok(0 == test_bad_size(), "test of too small/big file"); + + return exit_status(); +} + + +static int delete_file() +{ + RET_ERR_UNLESS(fn_format(file_name, CONTROL_FILE_BASE_NAME, + maria_data_root, "", MYF(MY_WME)) != NullS); + /* + Maybe file does not exist, ignore error. + The error will however be printed on stderr. + */ + my_delete(file_name, MYF(MY_WME)); + expect_checkpoint_lsn= CONTROL_FILE_IMPOSSIBLE_LSN; + expect_logno= CONTROL_FILE_IMPOSSIBLE_FILENO; + + return 0; +} + +/* + Verifies that global values last_checkpoint_lsn and last_logno (belonging + to the module) match what we expect. +*/ +static int verify_module_values_match_expected() +{ + RET_ERR_UNLESS(last_logno == expect_logno); + RET_ERR_UNLESS(last_checkpoint_lsn.file_no == + expect_checkpoint_lsn.file_no); + RET_ERR_UNLESS(last_checkpoint_lsn.rec_offset == + expect_checkpoint_lsn.rec_offset); + return 0; +} + + +/* + Verifies that global values last_checkpoint_lsn and last_logno (belonging + to the module) are impossible (this is used when the file has been closed). +*/ +static int verify_module_values_are_impossible() +{ + RET_ERR_UNLESS(last_logno == CONTROL_FILE_IMPOSSIBLE_FILENO); + RET_ERR_UNLESS(last_checkpoint_lsn.file_no == + CONTROL_FILE_IMPOSSIBLE_FILENO); + RET_ERR_UNLESS(last_checkpoint_lsn.rec_offset == + CONTROL_FILE_IMPOSSIBLE_LOG_OFFSET); + return 0; +} + + +static int close_file() +{ + /* Simulate shutdown */ + ma_control_file_end(); + /* Verify amnesia */ + RET_ERR_UNLESS(verify_module_values_are_impossible() == 0); + return 0; +} + +static int create_or_open_file() +{ + RET_ERR_UNLESS(ma_control_file_create_or_open() == CONTROL_FILE_OK); + /* Check that the module reports expected information */ + RET_ERR_UNLESS(verify_module_values_match_expected() == 0); + return 0; +} + +static int write_file(const LSN *checkpoint_lsn, + uint32 logno, + uint objs_to_write) +{ + RET_ERR_UNLESS(ma_control_file_write_and_force(checkpoint_lsn, logno, + objs_to_write) == 0); + /* Check that the module reports expected information */ + RET_ERR_UNLESS(verify_module_values_match_expected() == 0); + return 0; +} + +static int test_one_log() +{ + uint objs_to_write; + + RET_ERR_UNLESS(create_or_open_file() == CONTROL_FILE_OK); + objs_to_write= CONTROL_FILE_UPDATE_ONLY_LOGNO; + expect_logno= 123; + RET_ERR_UNLESS(write_file(NULL, expect_logno, + objs_to_write) == 0); + RET_ERR_UNLESS(close_file() == 0); + return 0; +} + +static int test_five_logs() +{ + uint objs_to_write; + uint i; + + RET_ERR_UNLESS(create_or_open_file() == CONTROL_FILE_OK); + objs_to_write= CONTROL_FILE_UPDATE_ONLY_LOGNO; + expect_logno= 100; + for (i= 0; i<5; i++) + { + expect_logno*= 3; + RET_ERR_UNLESS(write_file(NULL, expect_logno, + objs_to_write) == 0); + } + RET_ERR_UNLESS(close_file() == 0); + return 0; +} + +static int test_3_checkpoints_and_2_logs() +{ + uint objs_to_write; + /* + Simulate one checkpoint, one log creation, two checkpoints, one + log creation. + */ + RET_ERR_UNLESS(create_or_open_file() == CONTROL_FILE_OK); + objs_to_write= CONTROL_FILE_UPDATE_ONLY_LSN; + expect_checkpoint_lsn= (LSN){5, 10000}; + RET_ERR_UNLESS(write_file(&expect_checkpoint_lsn, + expect_logno, objs_to_write) == 0); + + objs_to_write= CONTROL_FILE_UPDATE_ONLY_LOGNO; + expect_logno= 17; + RET_ERR_UNLESS(write_file(&expect_checkpoint_lsn, + expect_logno, objs_to_write) == 0); + + objs_to_write= CONTROL_FILE_UPDATE_ONLY_LSN; + expect_checkpoint_lsn= (LSN){17, 20000}; + RET_ERR_UNLESS(write_file(&expect_checkpoint_lsn, + expect_logno, objs_to_write) == 0); + + objs_to_write= CONTROL_FILE_UPDATE_ONLY_LSN; + expect_checkpoint_lsn= (LSN){17, 45000}; + RET_ERR_UNLESS(write_file(&expect_checkpoint_lsn, + expect_logno, objs_to_write) == 0); + + objs_to_write= CONTROL_FILE_UPDATE_ONLY_LOGNO; + expect_logno= 19; + RET_ERR_UNLESS(write_file(&expect_checkpoint_lsn, + expect_logno, objs_to_write) == 0); + RET_ERR_UNLESS(close_file() == 0); + return 0; +} + +static int test_binary_content() +{ + uint i; + int fd; + + /* + TEST4: actually check by ourselves the content of the file. + Note that constants (offsets) are hard-coded here, precisely to prevent + someone from changing them in the control file module and breaking + backward-compatibility. + TODO: when we reach the format-freeze state, we may even just do a + comparison with a raw binary string, to not depend on any uint4korr + future change/breakage. + */ + + char buffer[17]; + RET_ERR_UNLESS((fd= my_open(file_name, + O_BINARY | O_RDWR, + MYF(MY_WME))) >= 0); + RET_ERR_UNLESS(my_read(fd, buffer, 17, MYF(MY_FNABP | MY_WME)) == 0); + RET_ERR_UNLESS(my_close(fd, MYF(MY_WME)) == 0); + RET_ERR_UNLESS(create_or_open_file() == CONTROL_FILE_OK); + i= uint4korr(buffer+5); + RET_ERR_UNLESS(i == last_checkpoint_lsn.file_no); + i= uint4korr(buffer+9); + RET_ERR_UNLESS(i == last_checkpoint_lsn.rec_offset); + i= uint4korr(buffer+13); + RET_ERR_UNLESS(i == last_logno); + RET_ERR_UNLESS(close_file() == 0); + return 0; +} + +static int test_start_stop() +{ + /* TEST5: Simulate start/nothing/stop/start/nothing/stop/start */ + + RET_ERR_UNLESS(create_or_open_file() == CONTROL_FILE_OK); + RET_ERR_UNLESS(close_file() == 0); + RET_ERR_UNLESS(create_or_open_file() == CONTROL_FILE_OK); + RET_ERR_UNLESS(close_file() == 0); + RET_ERR_UNLESS(create_or_open_file() == CONTROL_FILE_OK); + RET_ERR_UNLESS(close_file() == 0); + return 0; +} + +static int test_2_open_and_2_close() +{ + RET_ERR_UNLESS(create_or_open_file() == CONTROL_FILE_OK); + RET_ERR_UNLESS(create_or_open_file() == CONTROL_FILE_OK); + RET_ERR_UNLESS(close_file() == 0); + RET_ERR_UNLESS(close_file() == 0); + return 0; +} + + +static int test_bad_magic_string() +{ + char buffer[4]; + int fd; + + RET_ERR_UNLESS(create_or_open_file() == CONTROL_FILE_OK); + RET_ERR_UNLESS(close_file() == 0); + + /* Corrupt magic string */ + RET_ERR_UNLESS((fd= my_open(file_name, + O_BINARY | O_RDWR, + MYF(MY_WME))) >= 0); + RET_ERR_UNLESS(my_pread(fd, buffer, 4, 0, MYF(MY_FNABP | MY_WME)) == 0); + RET_ERR_UNLESS(my_pwrite(fd, "papa", 4, 0, MYF(MY_FNABP | MY_WME)) == 0); + + /* Check that control file module sees the problem */ + RET_ERR_UNLESS(ma_control_file_create_or_open() == + CONTROL_FILE_BAD_MAGIC_STRING); + /* Restore magic string */ + RET_ERR_UNLESS(my_pwrite(fd, buffer, 4, 0, MYF(MY_FNABP | MY_WME)) == 0); + RET_ERR_UNLESS(my_close(fd, MYF(MY_WME)) == 0); + RET_ERR_UNLESS(create_or_open_file() == CONTROL_FILE_OK); + RET_ERR_UNLESS(close_file() == 0); + return 0; +} + +static int test_bad_checksum() +{ + char buffer[4]; + int fd; + + RET_ERR_UNLESS(create_or_open_file() == CONTROL_FILE_OK); + RET_ERR_UNLESS(close_file() == 0); + + /* Corrupt checksum */ + RET_ERR_UNLESS((fd= my_open(file_name, + O_BINARY | O_RDWR, + MYF(MY_WME))) >= 0); + RET_ERR_UNLESS(my_pread(fd, buffer, 1, 4, MYF(MY_FNABP | MY_WME)) == 0); + buffer[0]+= 3; /* mangle checksum */ + RET_ERR_UNLESS(my_pwrite(fd, buffer+1, 1, 4, MYF(MY_FNABP | MY_WME)) == 0); + /* Check that control file module sees the problem */ + RET_ERR_UNLESS(ma_control_file_create_or_open() == + CONTROL_FILE_BAD_CHECKSUM); + /* Restore checksum */ + buffer[0]-= 3; + RET_ERR_UNLESS(my_pwrite(fd, buffer+1, 1, 4, MYF(MY_FNABP | MY_WME)) == 0); + RET_ERR_UNLESS(my_close(fd, MYF(MY_WME)) == 0); + + return 0; +} + + +static int test_bad_size() +{ + char buffer[]="aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa"; + int fd; + + /* A too short file */ + RET_ERR_UNLESS(delete_file() == 0); + RET_ERR_UNLESS((fd= my_open(file_name, + O_BINARY | O_RDWR | O_CREAT, + MYF(MY_WME))) >= 0); + RET_ERR_UNLESS(my_write(fd, buffer, 10, MYF(MY_FNABP | MY_WME)) == 0); + /* Check that control file module sees the problem */ + RET_ERR_UNLESS(ma_control_file_create_or_open() == CONTROL_FILE_TOO_SMALL); + RET_ERR_UNLESS(my_write(fd, buffer, 30, MYF(MY_FNABP | MY_WME)) == 0); + /* Check that control file module sees the problem */ + RET_ERR_UNLESS(ma_control_file_create_or_open() == CONTROL_FILE_TOO_BIG); + RET_ERR_UNLESS(my_close(fd, MYF(MY_WME)) == 0); + + /* Leave a correct control file */ + RET_ERR_UNLESS(delete_file() == 0); + RET_ERR_UNLESS(create_or_open_file() == CONTROL_FILE_OK); + RET_ERR_UNLESS(close_file() == 0); + + return 0; +} + + +static struct my_option my_long_options[] = +{ +#ifndef DBUG_OFF + {"debug", '#', "Debug log.", + 0, 0, 0, GET_STR, REQUIRED_ARG, 0, 0, 0, 0, 0, 0}, +#endif + {"help", '?', "Display help and exit", + 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, + {"version", 'V', "Print version number and exit", + 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, + { 0, 0, 0, 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0} +}; + + +static void version() +{ + printf("ma_control_file_test: unit test for the control file " + "module of the Maria storage engine. Ver 1.0 \n"); +} + +static my_bool +get_one_option(int optid, const struct my_option *opt __attribute__((unused)), + char *argument) +{ + switch(optid) { + case 'V': + version(); + exit(0); + case '#': + DBUG_PUSH (argument); + break; + case '?': + version(); + usage(); + exit(0); + } + return 0; +} + + +/* Read options */ + +static void get_options(int argc, char *argv[]) +{ + int ho_error; + + if ((ho_error=handle_options(&argc, &argv, my_long_options, + get_one_option))) + exit(ho_error); + + return; +} /* get options */ + + +static void usage() +{ + printf("Usage: %s [options]\n\n", my_progname); + my_print_help(my_long_options); + my_print_variables(my_long_options); +} -- cgit v1.2.1 From 15b9ce2201453ce7ecc346b053910023e7d51b83 Mon Sep 17 00:00:00 2001 From: unknown Date: Tue, 12 Sep 2006 15:52:01 +0200 Subject: WL#3234 Maria control file Fixes after Brian's and Serg's comments: storage engine's unit tests are now in storage//unittest instead of unittest/storage/ BitKeeper/deleted/.del-Makefile.am~78ea2e42d44d121f: Delete: unittest/storage/maria/Makefile.am BitKeeper/deleted/.del-Makefile.am~80497dbf1a80bdf3: Delete: unittest/storage/Makefile.am storage/maria/unittest/ma_control_file-t.c: Rename: unittest/storage/maria/ma_control_file-t.c -> storage/maria/unittest/ma_control_file-t.c config/ac-macros/plugins.m4: we change from unittest/storage// to storage//unittest: if the engine is enabled has such a directory, build this directory, and add it to the list of unit tests to run. configure.in: this dir does not exist anymore storage/maria/Makefile.am: to build Maria unittests, libmaria must be built. unittest/Makefile.am: unittest/storage is removed. target "unitests" is defined at "configure" time based on enabled engines. storage/maria/unittest/Makefile.am: simple Makefile.am to build ma_control_file-t --- config/ac-macros/plugins.m4 | 8 +- configure.in | 1 - storage/maria/Makefile.am | 3 + storage/maria/unittest/Makefile.am | 29 ++ storage/maria/unittest/ma_control_file-t.c | 448 +++++++++++++++++++++++++++++ unittest/Makefile.am | 4 +- unittest/storage/Makefile.am | 27 -- unittest/storage/maria/Makefile.am | 29 -- unittest/storage/maria/ma_control_file-t.c | 448 ----------------------------- 9 files changed, 486 insertions(+), 511 deletions(-) create mode 100644 storage/maria/unittest/Makefile.am create mode 100644 storage/maria/unittest/ma_control_file-t.c delete mode 100644 unittest/storage/Makefile.am delete mode 100644 unittest/storage/maria/Makefile.am delete mode 100644 unittest/storage/maria/ma_control_file-t.c diff --git a/config/ac-macros/plugins.m4 b/config/ac-macros/plugins.m4 index 791405f8fbd..ba7b3fcb087 100644 --- a/config/ac-macros/plugins.m4 +++ b/config/ac-macros/plugins.m4 @@ -411,11 +411,11 @@ dnl Although this is "pretty", it breaks libmysqld build [AC_CONFIG_SUBDIRS($6)], [ AC_CONFIG_FILES($6/Makefile) - m4_syscmd(test -d "unittest/$6") + m4_syscmd(test -d "$6/unittest") ifelse(m4_sysval, 0, [ - mysql_use_plugin_unittest_dir="$6" - AC_CONFIG_FILES(unittest/$6/Makefile) + mysql_use_plugin_unittest_dir="$6/unittest" + AC_CONFIG_FILES($6/unittest/Makefile) ], []) ] ) @@ -424,7 +424,7 @@ dnl Although this is "pretty", it breaks libmysqld build [mysql_se_name="]m4_substr($6, 8)" mysql_se_dirs="$mysql_se_dirs $mysql_se_name" if test -n "$mysql_use_plugin_unittest_dir" ; then - mysql_se_unittest_dirs="$mysql_se_unitest_dirs $mysql_se_name" + mysql_se_unittest_dirs="$mysql_se_unitest_dirs ../$mysql_use_plugin_unittest_dir" fi ], m4_substr($6, 0, 7), [plugin/], diff --git a/configure.in b/configure.in index 61c2deb5b5e..f0cdcde1abb 100644 --- a/configure.in +++ b/configure.in @@ -2482,7 +2482,6 @@ AC_SUBST(MAKE_BINARY_DISTRIBUTION_OPTIONS) AC_CONFIG_FILES(Makefile extra/Makefile mysys/Makefile dnl unittest/Makefile unittest/mytap/Makefile unittest/mytap/t/Makefile dnl unittest/mysys/Makefile unittest/examples/Makefile dnl - unittest/storage/Makefile dnl strings/Makefile regex/Makefile storage/Makefile dnl man/Makefile BUILD/Makefile vio/Makefile dnl libmysql/Makefile client/Makefile dnl diff --git a/storage/maria/Makefile.am b/storage/maria/Makefile.am index 10204226742..100000fa6cd 100644 --- a/storage/maria/Makefile.am +++ b/storage/maria/Makefile.am @@ -28,6 +28,9 @@ LDADD = DEFS = @DEFS@ +# "." is needed first because tests in unittest need libmaria +SUBDIRS = . unittest + EXTRA_DIST = ma_test_all.sh ma_test_all.res ma_ft_stem.c CMakeLists.txt pkgdata_DATA = ma_test_all ma_test_all.res pkglib_LIBRARIES = libmaria.a diff --git a/storage/maria/unittest/Makefile.am b/storage/maria/unittest/Makefile.am new file mode 100644 index 00000000000..eae2990aea9 --- /dev/null +++ b/storage/maria/unittest/Makefile.am @@ -0,0 +1,29 @@ +# Copyright (C) 2000 MySQL AB & MySQL Finland AB & TCX DataKonsult AB +# +# 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; either version 2 of the License, or +# (at your option) any later version. +# +# 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA + +AM_CPPFLAGS = @ZLIB_INCLUDES@ -I$(top_builddir)/include +AM_CPPFLAGS += -I$(top_srcdir)/include -I$(top_srcdir)/unittest/mytap + +# Only reason to link with libmyisam.a here is that it's where some fulltext +# pieces are (but soon we'll remove fulltext dependencies from Maria). +LDADD= $(top_builddir)/unittest/mytap/libmytap.a \ + $(top_builddir)/storage/maria/libmaria.a \ + $(top_builddir)/storage/myisam/libmyisam.a \ + $(top_builddir)/mysys/libmysys.a \ + $(top_builddir)/dbug/libdbug.a \ + $(top_builddir)/strings/libmystrings.a @ZLIB_LIBS@ +noinst_PROGRAMS = ma_control_file-t +CLEANFILES = maria_control diff --git a/storage/maria/unittest/ma_control_file-t.c b/storage/maria/unittest/ma_control_file-t.c new file mode 100644 index 00000000000..3ea6932c754 --- /dev/null +++ b/storage/maria/unittest/ma_control_file-t.c @@ -0,0 +1,448 @@ +/* Copyright (C) 2006 MySQL AB & MySQL Finland AB & TCX DataKonsult AB + + 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; either version 2 of the License, or + (at your option) any later version. + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ + +/* Unit test of the control file module of the Maria engine WL#3234 */ + +/* + Note that it is not possible to test the durability of the write (can't + pull the plug programmatically :) +*/ + +#include +#include +#include + +#ifndef WITH_MARIA_STORAGE_ENGINE +/* + If Maria is not compiled in, normally we don't come to building this test. +*/ +#error "Maria engine is not compiled in, test cannot be built" +#endif + +#include "maria.h" +#include "../../../storage/maria/ma_control_file.h" +#include + +char file_name[FN_REFLEN]; + +/* The values we'll set and expect the control file module to return */ +LSN expect_checkpoint_lsn; +uint32 expect_logno; + +static int delete_file(); +/* + Those are test-specific wrappers around the module's API functions: after + calling the module's API functions they perform checks on the result. +*/ +static int close_file(); /* wraps ma_control_file_end */ +static int create_or_open_file(); /* wraps ma_control_file_open_or_create */ +static int write_file(); /* wraps ma_control_file_write_and_force */ + +/* Tests */ +static int test_one_log(); +static int test_five_logs(); +static int test_3_checkpoints_and_2_logs(); +static int test_binary_content(); +static int test_start_stop(); +static int test_2_open_and_2_close(); +static int test_bad_magic_string(); +static int test_bad_checksum(); +static int test_bad_size(); + +/* Utility */ +static int verify_module_values_match_expected(); +static int verify_module_values_are_impossible(); +static void usage(); +static void get_options(int argc, char *argv[]); + +/* + If "expr" is FALSE, this macro will make the function print a diagnostic + message and immediately return 1. + This is inspired from assert() but does not crash the binary (sometimes we + may want to see how other tests go even if one fails). + RET_ERR means "return error". +*/ + +#define RET_ERR_UNLESS(expr) \ + {if (!(expr)) {diag("line %d: failure: '%s'", __LINE__, #expr); return 1;}} + + +int main(int argc,char *argv[]) +{ + MY_INIT(argv[0]); + + plan(9); + + diag("Unit tests for control file"); + + get_options(argc,argv); + + diag("Deleting control file at startup, if there is an old one"); + RET_ERR_UNLESS(0 == delete_file()); /* if fails, can't continue */ + + diag("Tests of normal conditions"); + ok(0 == test_one_log(), "test of creating one log"); + ok(0 == test_five_logs(), "test of creating five logs"); + ok(0 == test_3_checkpoints_and_2_logs(), + "test of creating three checkpoints and two logs"); + ok(0 == test_binary_content(), "test of the binary content of the file"); + ok(0 == test_start_stop(), "test of multiple starts and stops"); + diag("Tests of abnormal conditions"); + ok(0 == test_2_open_and_2_close(), + "test of two open and two close (strange call sequence)"); + ok(0 == test_bad_magic_string(), "test of bad magic string"); + ok(0 == test_bad_checksum(), "test of bad checksum"); + ok(0 == test_bad_size(), "test of too small/big file"); + + return exit_status(); +} + + +static int delete_file() +{ + RET_ERR_UNLESS(fn_format(file_name, CONTROL_FILE_BASE_NAME, + maria_data_root, "", MYF(MY_WME)) != NullS); + /* + Maybe file does not exist, ignore error. + The error will however be printed on stderr. + */ + my_delete(file_name, MYF(MY_WME)); + expect_checkpoint_lsn= CONTROL_FILE_IMPOSSIBLE_LSN; + expect_logno= CONTROL_FILE_IMPOSSIBLE_FILENO; + + return 0; +} + +/* + Verifies that global values last_checkpoint_lsn and last_logno (belonging + to the module) match what we expect. +*/ +static int verify_module_values_match_expected() +{ + RET_ERR_UNLESS(last_logno == expect_logno); + RET_ERR_UNLESS(last_checkpoint_lsn.file_no == + expect_checkpoint_lsn.file_no); + RET_ERR_UNLESS(last_checkpoint_lsn.rec_offset == + expect_checkpoint_lsn.rec_offset); + return 0; +} + + +/* + Verifies that global values last_checkpoint_lsn and last_logno (belonging + to the module) are impossible (this is used when the file has been closed). +*/ +static int verify_module_values_are_impossible() +{ + RET_ERR_UNLESS(last_logno == CONTROL_FILE_IMPOSSIBLE_FILENO); + RET_ERR_UNLESS(last_checkpoint_lsn.file_no == + CONTROL_FILE_IMPOSSIBLE_FILENO); + RET_ERR_UNLESS(last_checkpoint_lsn.rec_offset == + CONTROL_FILE_IMPOSSIBLE_LOG_OFFSET); + return 0; +} + + +static int close_file() +{ + /* Simulate shutdown */ + ma_control_file_end(); + /* Verify amnesia */ + RET_ERR_UNLESS(verify_module_values_are_impossible() == 0); + return 0; +} + +static int create_or_open_file() +{ + RET_ERR_UNLESS(ma_control_file_create_or_open() == CONTROL_FILE_OK); + /* Check that the module reports expected information */ + RET_ERR_UNLESS(verify_module_values_match_expected() == 0); + return 0; +} + +static int write_file(const LSN *checkpoint_lsn, + uint32 logno, + uint objs_to_write) +{ + RET_ERR_UNLESS(ma_control_file_write_and_force(checkpoint_lsn, logno, + objs_to_write) == 0); + /* Check that the module reports expected information */ + RET_ERR_UNLESS(verify_module_values_match_expected() == 0); + return 0; +} + +static int test_one_log() +{ + uint objs_to_write; + + RET_ERR_UNLESS(create_or_open_file() == CONTROL_FILE_OK); + objs_to_write= CONTROL_FILE_UPDATE_ONLY_LOGNO; + expect_logno= 123; + RET_ERR_UNLESS(write_file(NULL, expect_logno, + objs_to_write) == 0); + RET_ERR_UNLESS(close_file() == 0); + return 0; +} + +static int test_five_logs() +{ + uint objs_to_write; + uint i; + + RET_ERR_UNLESS(create_or_open_file() == CONTROL_FILE_OK); + objs_to_write= CONTROL_FILE_UPDATE_ONLY_LOGNO; + expect_logno= 100; + for (i= 0; i<5; i++) + { + expect_logno*= 3; + RET_ERR_UNLESS(write_file(NULL, expect_logno, + objs_to_write) == 0); + } + RET_ERR_UNLESS(close_file() == 0); + return 0; +} + +static int test_3_checkpoints_and_2_logs() +{ + uint objs_to_write; + /* + Simulate one checkpoint, one log creation, two checkpoints, one + log creation. + */ + RET_ERR_UNLESS(create_or_open_file() == CONTROL_FILE_OK); + objs_to_write= CONTROL_FILE_UPDATE_ONLY_LSN; + expect_checkpoint_lsn= (LSN){5, 10000}; + RET_ERR_UNLESS(write_file(&expect_checkpoint_lsn, + expect_logno, objs_to_write) == 0); + + objs_to_write= CONTROL_FILE_UPDATE_ONLY_LOGNO; + expect_logno= 17; + RET_ERR_UNLESS(write_file(&expect_checkpoint_lsn, + expect_logno, objs_to_write) == 0); + + objs_to_write= CONTROL_FILE_UPDATE_ONLY_LSN; + expect_checkpoint_lsn= (LSN){17, 20000}; + RET_ERR_UNLESS(write_file(&expect_checkpoint_lsn, + expect_logno, objs_to_write) == 0); + + objs_to_write= CONTROL_FILE_UPDATE_ONLY_LSN; + expect_checkpoint_lsn= (LSN){17, 45000}; + RET_ERR_UNLESS(write_file(&expect_checkpoint_lsn, + expect_logno, objs_to_write) == 0); + + objs_to_write= CONTROL_FILE_UPDATE_ONLY_LOGNO; + expect_logno= 19; + RET_ERR_UNLESS(write_file(&expect_checkpoint_lsn, + expect_logno, objs_to_write) == 0); + RET_ERR_UNLESS(close_file() == 0); + return 0; +} + +static int test_binary_content() +{ + uint i; + int fd; + + /* + TEST4: actually check by ourselves the content of the file. + Note that constants (offsets) are hard-coded here, precisely to prevent + someone from changing them in the control file module and breaking + backward-compatibility. + TODO: when we reach the format-freeze state, we may even just do a + comparison with a raw binary string, to not depend on any uint4korr + future change/breakage. + */ + + char buffer[17]; + RET_ERR_UNLESS((fd= my_open(file_name, + O_BINARY | O_RDWR, + MYF(MY_WME))) >= 0); + RET_ERR_UNLESS(my_read(fd, buffer, 17, MYF(MY_FNABP | MY_WME)) == 0); + RET_ERR_UNLESS(my_close(fd, MYF(MY_WME)) == 0); + RET_ERR_UNLESS(create_or_open_file() == CONTROL_FILE_OK); + i= uint4korr(buffer+5); + RET_ERR_UNLESS(i == last_checkpoint_lsn.file_no); + i= uint4korr(buffer+9); + RET_ERR_UNLESS(i == last_checkpoint_lsn.rec_offset); + i= uint4korr(buffer+13); + RET_ERR_UNLESS(i == last_logno); + RET_ERR_UNLESS(close_file() == 0); + return 0; +} + +static int test_start_stop() +{ + /* TEST5: Simulate start/nothing/stop/start/nothing/stop/start */ + + RET_ERR_UNLESS(create_or_open_file() == CONTROL_FILE_OK); + RET_ERR_UNLESS(close_file() == 0); + RET_ERR_UNLESS(create_or_open_file() == CONTROL_FILE_OK); + RET_ERR_UNLESS(close_file() == 0); + RET_ERR_UNLESS(create_or_open_file() == CONTROL_FILE_OK); + RET_ERR_UNLESS(close_file() == 0); + return 0; +} + +static int test_2_open_and_2_close() +{ + RET_ERR_UNLESS(create_or_open_file() == CONTROL_FILE_OK); + RET_ERR_UNLESS(create_or_open_file() == CONTROL_FILE_OK); + RET_ERR_UNLESS(close_file() == 0); + RET_ERR_UNLESS(close_file() == 0); + return 0; +} + + +static int test_bad_magic_string() +{ + char buffer[4]; + int fd; + + RET_ERR_UNLESS(create_or_open_file() == CONTROL_FILE_OK); + RET_ERR_UNLESS(close_file() == 0); + + /* Corrupt magic string */ + RET_ERR_UNLESS((fd= my_open(file_name, + O_BINARY | O_RDWR, + MYF(MY_WME))) >= 0); + RET_ERR_UNLESS(my_pread(fd, buffer, 4, 0, MYF(MY_FNABP | MY_WME)) == 0); + RET_ERR_UNLESS(my_pwrite(fd, "papa", 4, 0, MYF(MY_FNABP | MY_WME)) == 0); + + /* Check that control file module sees the problem */ + RET_ERR_UNLESS(ma_control_file_create_or_open() == + CONTROL_FILE_BAD_MAGIC_STRING); + /* Restore magic string */ + RET_ERR_UNLESS(my_pwrite(fd, buffer, 4, 0, MYF(MY_FNABP | MY_WME)) == 0); + RET_ERR_UNLESS(my_close(fd, MYF(MY_WME)) == 0); + RET_ERR_UNLESS(create_or_open_file() == CONTROL_FILE_OK); + RET_ERR_UNLESS(close_file() == 0); + return 0; +} + +static int test_bad_checksum() +{ + char buffer[4]; + int fd; + + RET_ERR_UNLESS(create_or_open_file() == CONTROL_FILE_OK); + RET_ERR_UNLESS(close_file() == 0); + + /* Corrupt checksum */ + RET_ERR_UNLESS((fd= my_open(file_name, + O_BINARY | O_RDWR, + MYF(MY_WME))) >= 0); + RET_ERR_UNLESS(my_pread(fd, buffer, 1, 4, MYF(MY_FNABP | MY_WME)) == 0); + buffer[0]+= 3; /* mangle checksum */ + RET_ERR_UNLESS(my_pwrite(fd, buffer+1, 1, 4, MYF(MY_FNABP | MY_WME)) == 0); + /* Check that control file module sees the problem */ + RET_ERR_UNLESS(ma_control_file_create_or_open() == + CONTROL_FILE_BAD_CHECKSUM); + /* Restore checksum */ + buffer[0]-= 3; + RET_ERR_UNLESS(my_pwrite(fd, buffer+1, 1, 4, MYF(MY_FNABP | MY_WME)) == 0); + RET_ERR_UNLESS(my_close(fd, MYF(MY_WME)) == 0); + + return 0; +} + + +static int test_bad_size() +{ + char buffer[]="aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa"; + int fd; + + /* A too short file */ + RET_ERR_UNLESS(delete_file() == 0); + RET_ERR_UNLESS((fd= my_open(file_name, + O_BINARY | O_RDWR | O_CREAT, + MYF(MY_WME))) >= 0); + RET_ERR_UNLESS(my_write(fd, buffer, 10, MYF(MY_FNABP | MY_WME)) == 0); + /* Check that control file module sees the problem */ + RET_ERR_UNLESS(ma_control_file_create_or_open() == CONTROL_FILE_TOO_SMALL); + RET_ERR_UNLESS(my_write(fd, buffer, 30, MYF(MY_FNABP | MY_WME)) == 0); + /* Check that control file module sees the problem */ + RET_ERR_UNLESS(ma_control_file_create_or_open() == CONTROL_FILE_TOO_BIG); + RET_ERR_UNLESS(my_close(fd, MYF(MY_WME)) == 0); + + /* Leave a correct control file */ + RET_ERR_UNLESS(delete_file() == 0); + RET_ERR_UNLESS(create_or_open_file() == CONTROL_FILE_OK); + RET_ERR_UNLESS(close_file() == 0); + + return 0; +} + + +static struct my_option my_long_options[] = +{ +#ifndef DBUG_OFF + {"debug", '#', "Debug log.", + 0, 0, 0, GET_STR, REQUIRED_ARG, 0, 0, 0, 0, 0, 0}, +#endif + {"help", '?', "Display help and exit", + 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, + {"version", 'V', "Print version number and exit", + 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, + { 0, 0, 0, 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0} +}; + + +static void version() +{ + printf("ma_control_file_test: unit test for the control file " + "module of the Maria storage engine. Ver 1.0 \n"); +} + +static my_bool +get_one_option(int optid, const struct my_option *opt __attribute__((unused)), + char *argument) +{ + switch(optid) { + case 'V': + version(); + exit(0); + case '#': + DBUG_PUSH (argument); + break; + case '?': + version(); + usage(); + exit(0); + } + return 0; +} + + +/* Read options */ + +static void get_options(int argc, char *argv[]) +{ + int ho_error; + + if ((ho_error=handle_options(&argc, &argv, my_long_options, + get_one_option))) + exit(ho_error); + + return; +} /* get options */ + + +static void usage() +{ + printf("Usage: %s [options]\n\n", my_progname); + my_print_help(my_long_options); + my_print_variables(my_long_options); +} diff --git a/unittest/Makefile.am b/unittest/Makefile.am index 176ac020bea..76004e67fae 100644 --- a/unittest/Makefile.am +++ b/unittest/Makefile.am @@ -1,10 +1,10 @@ -SUBDIRS = mytap . mysys storage examples +SUBDIRS = mytap . mysys examples noinst_SCRIPTS = unit EXTRA_DIST = unit.pl CLEANFILES = unit -unittests = mytap mysys storage +unittests = mytap mysys @mysql_se_unittest_dirs@ test: unit ./unit run $(unittests) diff --git a/unittest/storage/Makefile.am b/unittest/storage/Makefile.am deleted file mode 100644 index 21de2de75fc..00000000000 --- a/unittest/storage/Makefile.am +++ /dev/null @@ -1,27 +0,0 @@ -# Copyright (C) 2000 MySQL AB & MySQL Finland AB & TCX DataKonsult AB -# -# 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; either version 2 of the License, or -# (at your option) any later version. -# -# 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA - -# Process this file with automake to create Makefile.in - -AUTOMAKE_OPTIONS = foreign - -# These are built from source in the Docs directory -EXTRA_DIST = -# Cannot use @mysql_se_dirs@ as not all engines have unit tests here -SUBDIRS = @mysql_se_unittest_dirs@ - -# Don't update the files from bitkeeper -%::SCCS/s.% diff --git a/unittest/storage/maria/Makefile.am b/unittest/storage/maria/Makefile.am deleted file mode 100644 index eae2990aea9..00000000000 --- a/unittest/storage/maria/Makefile.am +++ /dev/null @@ -1,29 +0,0 @@ -# Copyright (C) 2000 MySQL AB & MySQL Finland AB & TCX DataKonsult AB -# -# 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; either version 2 of the License, or -# (at your option) any later version. -# -# 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA - -AM_CPPFLAGS = @ZLIB_INCLUDES@ -I$(top_builddir)/include -AM_CPPFLAGS += -I$(top_srcdir)/include -I$(top_srcdir)/unittest/mytap - -# Only reason to link with libmyisam.a here is that it's where some fulltext -# pieces are (but soon we'll remove fulltext dependencies from Maria). -LDADD= $(top_builddir)/unittest/mytap/libmytap.a \ - $(top_builddir)/storage/maria/libmaria.a \ - $(top_builddir)/storage/myisam/libmyisam.a \ - $(top_builddir)/mysys/libmysys.a \ - $(top_builddir)/dbug/libdbug.a \ - $(top_builddir)/strings/libmystrings.a @ZLIB_LIBS@ -noinst_PROGRAMS = ma_control_file-t -CLEANFILES = maria_control diff --git a/unittest/storage/maria/ma_control_file-t.c b/unittest/storage/maria/ma_control_file-t.c deleted file mode 100644 index 3ea6932c754..00000000000 --- a/unittest/storage/maria/ma_control_file-t.c +++ /dev/null @@ -1,448 +0,0 @@ -/* Copyright (C) 2006 MySQL AB & MySQL Finland AB & TCX DataKonsult AB - - 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; either version 2 of the License, or - (at your option) any later version. - - 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ - -/* Unit test of the control file module of the Maria engine WL#3234 */ - -/* - Note that it is not possible to test the durability of the write (can't - pull the plug programmatically :) -*/ - -#include -#include -#include - -#ifndef WITH_MARIA_STORAGE_ENGINE -/* - If Maria is not compiled in, normally we don't come to building this test. -*/ -#error "Maria engine is not compiled in, test cannot be built" -#endif - -#include "maria.h" -#include "../../../storage/maria/ma_control_file.h" -#include - -char file_name[FN_REFLEN]; - -/* The values we'll set and expect the control file module to return */ -LSN expect_checkpoint_lsn; -uint32 expect_logno; - -static int delete_file(); -/* - Those are test-specific wrappers around the module's API functions: after - calling the module's API functions they perform checks on the result. -*/ -static int close_file(); /* wraps ma_control_file_end */ -static int create_or_open_file(); /* wraps ma_control_file_open_or_create */ -static int write_file(); /* wraps ma_control_file_write_and_force */ - -/* Tests */ -static int test_one_log(); -static int test_five_logs(); -static int test_3_checkpoints_and_2_logs(); -static int test_binary_content(); -static int test_start_stop(); -static int test_2_open_and_2_close(); -static int test_bad_magic_string(); -static int test_bad_checksum(); -static int test_bad_size(); - -/* Utility */ -static int verify_module_values_match_expected(); -static int verify_module_values_are_impossible(); -static void usage(); -static void get_options(int argc, char *argv[]); - -/* - If "expr" is FALSE, this macro will make the function print a diagnostic - message and immediately return 1. - This is inspired from assert() but does not crash the binary (sometimes we - may want to see how other tests go even if one fails). - RET_ERR means "return error". -*/ - -#define RET_ERR_UNLESS(expr) \ - {if (!(expr)) {diag("line %d: failure: '%s'", __LINE__, #expr); return 1;}} - - -int main(int argc,char *argv[]) -{ - MY_INIT(argv[0]); - - plan(9); - - diag("Unit tests for control file"); - - get_options(argc,argv); - - diag("Deleting control file at startup, if there is an old one"); - RET_ERR_UNLESS(0 == delete_file()); /* if fails, can't continue */ - - diag("Tests of normal conditions"); - ok(0 == test_one_log(), "test of creating one log"); - ok(0 == test_five_logs(), "test of creating five logs"); - ok(0 == test_3_checkpoints_and_2_logs(), - "test of creating three checkpoints and two logs"); - ok(0 == test_binary_content(), "test of the binary content of the file"); - ok(0 == test_start_stop(), "test of multiple starts and stops"); - diag("Tests of abnormal conditions"); - ok(0 == test_2_open_and_2_close(), - "test of two open and two close (strange call sequence)"); - ok(0 == test_bad_magic_string(), "test of bad magic string"); - ok(0 == test_bad_checksum(), "test of bad checksum"); - ok(0 == test_bad_size(), "test of too small/big file"); - - return exit_status(); -} - - -static int delete_file() -{ - RET_ERR_UNLESS(fn_format(file_name, CONTROL_FILE_BASE_NAME, - maria_data_root, "", MYF(MY_WME)) != NullS); - /* - Maybe file does not exist, ignore error. - The error will however be printed on stderr. - */ - my_delete(file_name, MYF(MY_WME)); - expect_checkpoint_lsn= CONTROL_FILE_IMPOSSIBLE_LSN; - expect_logno= CONTROL_FILE_IMPOSSIBLE_FILENO; - - return 0; -} - -/* - Verifies that global values last_checkpoint_lsn and last_logno (belonging - to the module) match what we expect. -*/ -static int verify_module_values_match_expected() -{ - RET_ERR_UNLESS(last_logno == expect_logno); - RET_ERR_UNLESS(last_checkpoint_lsn.file_no == - expect_checkpoint_lsn.file_no); - RET_ERR_UNLESS(last_checkpoint_lsn.rec_offset == - expect_checkpoint_lsn.rec_offset); - return 0; -} - - -/* - Verifies that global values last_checkpoint_lsn and last_logno (belonging - to the module) are impossible (this is used when the file has been closed). -*/ -static int verify_module_values_are_impossible() -{ - RET_ERR_UNLESS(last_logno == CONTROL_FILE_IMPOSSIBLE_FILENO); - RET_ERR_UNLESS(last_checkpoint_lsn.file_no == - CONTROL_FILE_IMPOSSIBLE_FILENO); - RET_ERR_UNLESS(last_checkpoint_lsn.rec_offset == - CONTROL_FILE_IMPOSSIBLE_LOG_OFFSET); - return 0; -} - - -static int close_file() -{ - /* Simulate shutdown */ - ma_control_file_end(); - /* Verify amnesia */ - RET_ERR_UNLESS(verify_module_values_are_impossible() == 0); - return 0; -} - -static int create_or_open_file() -{ - RET_ERR_UNLESS(ma_control_file_create_or_open() == CONTROL_FILE_OK); - /* Check that the module reports expected information */ - RET_ERR_UNLESS(verify_module_values_match_expected() == 0); - return 0; -} - -static int write_file(const LSN *checkpoint_lsn, - uint32 logno, - uint objs_to_write) -{ - RET_ERR_UNLESS(ma_control_file_write_and_force(checkpoint_lsn, logno, - objs_to_write) == 0); - /* Check that the module reports expected information */ - RET_ERR_UNLESS(verify_module_values_match_expected() == 0); - return 0; -} - -static int test_one_log() -{ - uint objs_to_write; - - RET_ERR_UNLESS(create_or_open_file() == CONTROL_FILE_OK); - objs_to_write= CONTROL_FILE_UPDATE_ONLY_LOGNO; - expect_logno= 123; - RET_ERR_UNLESS(write_file(NULL, expect_logno, - objs_to_write) == 0); - RET_ERR_UNLESS(close_file() == 0); - return 0; -} - -static int test_five_logs() -{ - uint objs_to_write; - uint i; - - RET_ERR_UNLESS(create_or_open_file() == CONTROL_FILE_OK); - objs_to_write= CONTROL_FILE_UPDATE_ONLY_LOGNO; - expect_logno= 100; - for (i= 0; i<5; i++) - { - expect_logno*= 3; - RET_ERR_UNLESS(write_file(NULL, expect_logno, - objs_to_write) == 0); - } - RET_ERR_UNLESS(close_file() == 0); - return 0; -} - -static int test_3_checkpoints_and_2_logs() -{ - uint objs_to_write; - /* - Simulate one checkpoint, one log creation, two checkpoints, one - log creation. - */ - RET_ERR_UNLESS(create_or_open_file() == CONTROL_FILE_OK); - objs_to_write= CONTROL_FILE_UPDATE_ONLY_LSN; - expect_checkpoint_lsn= (LSN){5, 10000}; - RET_ERR_UNLESS(write_file(&expect_checkpoint_lsn, - expect_logno, objs_to_write) == 0); - - objs_to_write= CONTROL_FILE_UPDATE_ONLY_LOGNO; - expect_logno= 17; - RET_ERR_UNLESS(write_file(&expect_checkpoint_lsn, - expect_logno, objs_to_write) == 0); - - objs_to_write= CONTROL_FILE_UPDATE_ONLY_LSN; - expect_checkpoint_lsn= (LSN){17, 20000}; - RET_ERR_UNLESS(write_file(&expect_checkpoint_lsn, - expect_logno, objs_to_write) == 0); - - objs_to_write= CONTROL_FILE_UPDATE_ONLY_LSN; - expect_checkpoint_lsn= (LSN){17, 45000}; - RET_ERR_UNLESS(write_file(&expect_checkpoint_lsn, - expect_logno, objs_to_write) == 0); - - objs_to_write= CONTROL_FILE_UPDATE_ONLY_LOGNO; - expect_logno= 19; - RET_ERR_UNLESS(write_file(&expect_checkpoint_lsn, - expect_logno, objs_to_write) == 0); - RET_ERR_UNLESS(close_file() == 0); - return 0; -} - -static int test_binary_content() -{ - uint i; - int fd; - - /* - TEST4: actually check by ourselves the content of the file. - Note that constants (offsets) are hard-coded here, precisely to prevent - someone from changing them in the control file module and breaking - backward-compatibility. - TODO: when we reach the format-freeze state, we may even just do a - comparison with a raw binary string, to not depend on any uint4korr - future change/breakage. - */ - - char buffer[17]; - RET_ERR_UNLESS((fd= my_open(file_name, - O_BINARY | O_RDWR, - MYF(MY_WME))) >= 0); - RET_ERR_UNLESS(my_read(fd, buffer, 17, MYF(MY_FNABP | MY_WME)) == 0); - RET_ERR_UNLESS(my_close(fd, MYF(MY_WME)) == 0); - RET_ERR_UNLESS(create_or_open_file() == CONTROL_FILE_OK); - i= uint4korr(buffer+5); - RET_ERR_UNLESS(i == last_checkpoint_lsn.file_no); - i= uint4korr(buffer+9); - RET_ERR_UNLESS(i == last_checkpoint_lsn.rec_offset); - i= uint4korr(buffer+13); - RET_ERR_UNLESS(i == last_logno); - RET_ERR_UNLESS(close_file() == 0); - return 0; -} - -static int test_start_stop() -{ - /* TEST5: Simulate start/nothing/stop/start/nothing/stop/start */ - - RET_ERR_UNLESS(create_or_open_file() == CONTROL_FILE_OK); - RET_ERR_UNLESS(close_file() == 0); - RET_ERR_UNLESS(create_or_open_file() == CONTROL_FILE_OK); - RET_ERR_UNLESS(close_file() == 0); - RET_ERR_UNLESS(create_or_open_file() == CONTROL_FILE_OK); - RET_ERR_UNLESS(close_file() == 0); - return 0; -} - -static int test_2_open_and_2_close() -{ - RET_ERR_UNLESS(create_or_open_file() == CONTROL_FILE_OK); - RET_ERR_UNLESS(create_or_open_file() == CONTROL_FILE_OK); - RET_ERR_UNLESS(close_file() == 0); - RET_ERR_UNLESS(close_file() == 0); - return 0; -} - - -static int test_bad_magic_string() -{ - char buffer[4]; - int fd; - - RET_ERR_UNLESS(create_or_open_file() == CONTROL_FILE_OK); - RET_ERR_UNLESS(close_file() == 0); - - /* Corrupt magic string */ - RET_ERR_UNLESS((fd= my_open(file_name, - O_BINARY | O_RDWR, - MYF(MY_WME))) >= 0); - RET_ERR_UNLESS(my_pread(fd, buffer, 4, 0, MYF(MY_FNABP | MY_WME)) == 0); - RET_ERR_UNLESS(my_pwrite(fd, "papa", 4, 0, MYF(MY_FNABP | MY_WME)) == 0); - - /* Check that control file module sees the problem */ - RET_ERR_UNLESS(ma_control_file_create_or_open() == - CONTROL_FILE_BAD_MAGIC_STRING); - /* Restore magic string */ - RET_ERR_UNLESS(my_pwrite(fd, buffer, 4, 0, MYF(MY_FNABP | MY_WME)) == 0); - RET_ERR_UNLESS(my_close(fd, MYF(MY_WME)) == 0); - RET_ERR_UNLESS(create_or_open_file() == CONTROL_FILE_OK); - RET_ERR_UNLESS(close_file() == 0); - return 0; -} - -static int test_bad_checksum() -{ - char buffer[4]; - int fd; - - RET_ERR_UNLESS(create_or_open_file() == CONTROL_FILE_OK); - RET_ERR_UNLESS(close_file() == 0); - - /* Corrupt checksum */ - RET_ERR_UNLESS((fd= my_open(file_name, - O_BINARY | O_RDWR, - MYF(MY_WME))) >= 0); - RET_ERR_UNLESS(my_pread(fd, buffer, 1, 4, MYF(MY_FNABP | MY_WME)) == 0); - buffer[0]+= 3; /* mangle checksum */ - RET_ERR_UNLESS(my_pwrite(fd, buffer+1, 1, 4, MYF(MY_FNABP | MY_WME)) == 0); - /* Check that control file module sees the problem */ - RET_ERR_UNLESS(ma_control_file_create_or_open() == - CONTROL_FILE_BAD_CHECKSUM); - /* Restore checksum */ - buffer[0]-= 3; - RET_ERR_UNLESS(my_pwrite(fd, buffer+1, 1, 4, MYF(MY_FNABP | MY_WME)) == 0); - RET_ERR_UNLESS(my_close(fd, MYF(MY_WME)) == 0); - - return 0; -} - - -static int test_bad_size() -{ - char buffer[]="aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa"; - int fd; - - /* A too short file */ - RET_ERR_UNLESS(delete_file() == 0); - RET_ERR_UNLESS((fd= my_open(file_name, - O_BINARY | O_RDWR | O_CREAT, - MYF(MY_WME))) >= 0); - RET_ERR_UNLESS(my_write(fd, buffer, 10, MYF(MY_FNABP | MY_WME)) == 0); - /* Check that control file module sees the problem */ - RET_ERR_UNLESS(ma_control_file_create_or_open() == CONTROL_FILE_TOO_SMALL); - RET_ERR_UNLESS(my_write(fd, buffer, 30, MYF(MY_FNABP | MY_WME)) == 0); - /* Check that control file module sees the problem */ - RET_ERR_UNLESS(ma_control_file_create_or_open() == CONTROL_FILE_TOO_BIG); - RET_ERR_UNLESS(my_close(fd, MYF(MY_WME)) == 0); - - /* Leave a correct control file */ - RET_ERR_UNLESS(delete_file() == 0); - RET_ERR_UNLESS(create_or_open_file() == CONTROL_FILE_OK); - RET_ERR_UNLESS(close_file() == 0); - - return 0; -} - - -static struct my_option my_long_options[] = -{ -#ifndef DBUG_OFF - {"debug", '#', "Debug log.", - 0, 0, 0, GET_STR, REQUIRED_ARG, 0, 0, 0, 0, 0, 0}, -#endif - {"help", '?', "Display help and exit", - 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, - {"version", 'V', "Print version number and exit", - 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, - { 0, 0, 0, 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0} -}; - - -static void version() -{ - printf("ma_control_file_test: unit test for the control file " - "module of the Maria storage engine. Ver 1.0 \n"); -} - -static my_bool -get_one_option(int optid, const struct my_option *opt __attribute__((unused)), - char *argument) -{ - switch(optid) { - case 'V': - version(); - exit(0); - case '#': - DBUG_PUSH (argument); - break; - case '?': - version(); - usage(); - exit(0); - } - return 0; -} - - -/* Read options */ - -static void get_options(int argc, char *argv[]) -{ - int ho_error; - - if ((ho_error=handle_options(&argc, &argv, my_long_options, - get_one_option))) - exit(ho_error); - - return; -} /* get options */ - - -static void usage() -{ - printf("Usage: %s [options]\n\n", my_progname); - my_print_help(my_long_options); - my_print_variables(my_long_options); -} -- cgit v1.2.1 From cdf831cf94fe9aabde6ffb5b19557893416061d6 Mon Sep 17 00:00:00 2001 From: unknown Date: Thu, 14 Sep 2006 19:06:51 +0200 Subject: WL#3071 Maria checkpoint: changing pseudocode to use the structures of the Maria pagecache ("pagecache->changed_blocks" etc) and other Maria structures inherited from MyISAM (THR_LOCK_maria etc). mysys/mf_pagecache.c: comment storage/maria/ma_checkpoint.c: changing pseudocode to use the structures of the Maria pagecache ("pagecache->changed_blocks" etc) and other Maria structures inherited from MyISAM (THR_LOCK_maria etc). storage/maria/ma_checkpoint.h: copyright storage/maria/ma_control_file.c: copyright storage/maria/ma_control_file.h: copyright storage/maria/ma_least_recently_dirtied.c: copyright storage/maria/ma_least_recently_dirtied.h: copyright storage/maria/ma_recovery.c: copyright storage/maria/ma_recovery.h: copyright storage/maria/unittest/Makefile.am: copyright --- mysys/mf_pagecache.c | 4 + storage/maria/ma_checkpoint.c | 124 +++++++++++++++++++++++++----- storage/maria/ma_checkpoint.h | 16 ++++ storage/maria/ma_control_file.c | 21 ++++- storage/maria/ma_control_file.h | 16 ++++ storage/maria/ma_least_recently_dirtied.c | 16 ++++ storage/maria/ma_least_recently_dirtied.h | 16 ++++ storage/maria/ma_recovery.c | 16 ++++ storage/maria/ma_recovery.h | 16 ++++ storage/maria/unittest/Makefile.am | 2 +- 10 files changed, 225 insertions(+), 22 deletions(-) diff --git a/mysys/mf_pagecache.c b/mysys/mf_pagecache.c index 4693995f922..3a054077809 100755 --- a/mysys/mf_pagecache.c +++ b/mysys/mf_pagecache.c @@ -2937,6 +2937,10 @@ restart: } pagecache->blocks_changed--; pagecache->global_blocks_changed--; + /* + free_block() will change the status of the block so no need to change + it here. + */ } /* Cache is locked, so we can relese page before freeing it */ diff --git a/storage/maria/ma_checkpoint.c b/storage/maria/ma_checkpoint.c index 22e7b93d2f4..83312ce37b8 100644 --- a/storage/maria/ma_checkpoint.c +++ b/storage/maria/ma_checkpoint.c @@ -1,3 +1,19 @@ +/* Copyright (C) 2006 MySQL AB & MySQL Finland AB & TCX DataKonsult AB + + 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; either version 2 of the License, or + (at your option) any later version. + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ + /* WL#3071 Maria checkpoint First version written by Guilhem Bichot on 2006-04-27. @@ -110,7 +126,7 @@ my_bool execute_checkpoint(CHECKPOINT_LEVEL level) LSN candidate_max_rec_lsn_at_last_checkpoint; /* to avoid { lock + no-op + unlock } in the common (==indirect) case */ my_bool need_log_mutex; - + DBUG_ENTER("execute_checkpoint"); safemutex_assert_owner(log_mutex); @@ -120,7 +136,7 @@ my_bool execute_checkpoint(CHECKPOINT_LEVEL level) { /* much I/O work to do, release log mutex */ unlock(log_mutex); - + switch (level) { case FULL: @@ -167,6 +183,13 @@ my_bool execute_checkpoint(CHECKPOINT_LEVEL level) } +/* + Does an indirect checpoint (collects data from data structures, writes into + a checkpoint log record). + Returns the largest LSN of the LRD when the checkpoint happened (this is a + fuzzy definition), or LSN_IMPOSSIBLE on error. That LSN is used for the + "two-checkpoint rule" (MEDIUM checkpoints). +*/ LSN checkpoint_indirect(my_bool need_log_mutex) { DBUG_ENTER("checkpoint_indirect"); @@ -180,6 +203,7 @@ LSN checkpoint_indirect(my_bool need_log_mutex) LSN checkpoint_lsn; LSN candidate_max_rec_lsn_at_last_checkpoint= 0; list_element *el; /* to scan lists */ + ulong stored_LRD_size= 0; DBUG_ASSERT(sizeof(byte *) <= 8); @@ -192,27 +216,70 @@ LSN checkpoint_indirect(my_bool need_log_mutex) DBUG_PRINT("info",("checkpoint_start_lsn %lu", checkpoint_start_lsn)); - lock(global_LRD_mutex); - string1.length= 8+8+(8+8)*LRD->count; + /* STEP 1: fetch information about dirty pages */ + + /* + We lock the entire cache but will be quick, just reading/writing a few MBs + of memory at most. + */ + pagecache_pthread_mutex_lock(&pagecache->cache_lock); + + /* + This is an over-estimation, as in theory blocks_changed may contain + non-PAGECACHE_LSN_PAGE pages, which we don't want to store into the + checkpoint record; the true number of page-LRD-info we'll store into the + record is stored_LRD_size. + */ + string1.length= 8+8+(8+8)*pagecache->blocks_changed; if (NULL == (string1.str= my_malloc(string1.length))) goto err; ptr= string1.str; int8store(ptr, checkpoint_start_lsn); - ptr+= 8; - int8store(ptr, LRD->count); - ptr+= 8; - if (LRD->count) + ptr+= 8+8; /* don't store stored_LRD_size now, wait */ + if (pagecache->blocks_changed > 0) { - candidate_max_rec_lsn_at_last_checkpoint= LRD->last->rec_lsn; - for (el= LRD->first; el; el= el->next) + /* + There are different ways to scan the dirty blocks; + flush_all_key_blocks() uses a loop over pagecache->used_last->next_used, + and for each element of the loop, loops into + pagecache->changed_blocks[FILE_HASH(file of the element)]. + This has the drawback that used_last includes non-dirty blocks, and it's + two loops over many elements. Here we try something simpler. + If there are no blocks in changed_blocks[file_hash], we should hit + zeroes and skip them. + */ + uint file_hash; + for (file_hash= 0; file_hash < PAGECACHE_CHANGED_BLOCKS_HASH; file_hash++) { - int8store(ptr, el->page_id); - ptr+= 8; - int8store(ptr, el->rec_lsn); - ptr+= 8; + PAGECACHE_BLOCK_LINK *block; + for (block= pagecache->changed_blocks[file_hash] ; + block; + block= block->next_changed) + { + DBUG_ASSERT(block->hash_link != NULL); + DBUG_ASSERT(block->status & BLOCK_CHANGED); + if (block->type != PAGECACHE_LSN_PAGE) + { + /* no need to store it in the checkpoint record */ + continue; + } + /* Q: two "block"s cannot have the same "hash_link", right? */ + int8store(ptr, block->hash_link->pageno); + ptr+= 8; + /* I assume rec_lsn will be member of "block", not of "hash_link" */ + int8store(ptr, block->rec_lsn); + ptr+= 8; + stored_LRD_size++; + set_if_bigger(candidate_max_rec_lsn_at_last_checkpoint, + block->rec_lsn); + } } - } - unlock(global_LRD_mutex); + pagecache_pthread_mutex_unlock(&pagecache->cache_lock); + DBUG_ASSERT(stored_LRD_size <= pagecache->blocks_changed); + int8store(string1.str+8, stored_LRD_size); + string1.length= 8+8+(8+8)*stored_LRD_size; + + /* STEP 2: fetch information about transactions */ /* If trx are in more than one list (e.g. three: @@ -253,19 +320,28 @@ LSN checkpoint_indirect(my_bool need_log_mutex) } unlock(global_transactions_list_mutex); + /* STEP 3: fetch information about table files */ + + /* This global mutex is in fact THR_LOCK_maria (see ma_open()) */ lock(global_share_list_mutex); string3.length= 8+(8+8)*share_list->count; if (NULL == (string3.str= my_malloc(string3.length))) goto err; ptr= string3.str; - /* possibly latch each MARIA_SHARE */ + /* possibly latch each MARIA_SHARE, one by one, like this: */ + pthread_mutex_lock(&share->intern_lock); + /* + We'll copy the file id (a bit like share->kfile), the file name + (like share->unique_file_name[_length]). + */ make_copy_of_global_share_list_to_array; + pthread_mutex_unlock(&share->intern_lock); unlock(global_share_list_mutex); /* work on copy */ int8store(ptr, elements_in_array); ptr+= 8; - for (scan_array) + for (el in array) { int8store(ptr, array[...].file_id); ptr+= 8; @@ -273,9 +349,11 @@ LSN checkpoint_indirect(my_bool need_log_mutex) ptr+= ...; /* these two are long ops (involving disk I/O) that's why we copied the - list: + list, to not keep the list locked for long: */ flush_bitmap_pages(el); + /* TODO: and also autoinc counter, logical file end, free page list */ + /* fsyncs the fd, that's the loooong operation (e.g. max 150 fsync per second, so if you have touched 1000 files it's 7 seconds). @@ -283,7 +361,8 @@ LSN checkpoint_indirect(my_bool need_log_mutex) force_file(el); } - /* now write the record */ + /* LAST STEP: now write the checkpoint log record */ + string_array[0]= string1; string_array[1]= string2; string_array[2]= string3; @@ -292,6 +371,11 @@ LSN checkpoint_indirect(my_bool need_log_mutex) checkpoint_lsn= log_write_record(LOGREC_CHECKPOINT, &system_trans, string_array); + /* + Do nothing between the log write and the control file write, for the + "repair control file" tool to be possible one day. + */ + if (LSN_IMPOSSIBLE == checkpoint_lsn) goto err; diff --git a/storage/maria/ma_checkpoint.h b/storage/maria/ma_checkpoint.h index a9de18c695f..1b8064fa755 100644 --- a/storage/maria/ma_checkpoint.h +++ b/storage/maria/ma_checkpoint.h @@ -1,3 +1,19 @@ +/* Copyright (C) 2006 MySQL AB & MySQL Finland AB & TCX DataKonsult AB + + 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; either version 2 of the License, or + (at your option) any later version. + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ + /* WL#3071 Maria checkpoint First version written by Guilhem Bichot on 2006-04-27. diff --git a/storage/maria/ma_control_file.c b/storage/maria/ma_control_file.c index 5fbb0a084df..5b66577938f 100644 --- a/storage/maria/ma_control_file.c +++ b/storage/maria/ma_control_file.c @@ -1,3 +1,19 @@ +/* Copyright (C) 2006 MySQL AB & MySQL Finland AB & TCX DataKonsult AB + + 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; either version 2 of the License, or + (at your option) any later version. + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ + /* WL#3234 Maria control file First version written by Guilhem Bichot on 2006-04-27. @@ -137,7 +153,10 @@ CONTROL_FILE_ERROR ma_control_file_create_or_open() We could have a tool which can rebuild the control file, by reading the directory of logs, finding the newest log, reading it to find last - checkpoint... Slow but can save your db. + checkpoint... Slow but can save your db. For this to be possible, we + must always write to the control file right after writing the checkpoint + log record, and do nothing in between (i.e. the checkpoint must be + usable as soon as it has been written to the log). */ LSN imposs_lsn= CONTROL_FILE_IMPOSSIBLE_LSN; diff --git a/storage/maria/ma_control_file.h b/storage/maria/ma_control_file.h index 5ac6f158183..9a99a721469 100644 --- a/storage/maria/ma_control_file.h +++ b/storage/maria/ma_control_file.h @@ -1,3 +1,19 @@ +/* Copyright (C) 2006 MySQL AB & MySQL Finland AB & TCX DataKonsult AB + + 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; either version 2 of the License, or + (at your option) any later version. + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ + /* WL#3234 Maria control file First version written by Guilhem Bichot on 2006-04-27. diff --git a/storage/maria/ma_least_recently_dirtied.c b/storage/maria/ma_least_recently_dirtied.c index c6285fe47cd..b0b7fb1ef10 100644 --- a/storage/maria/ma_least_recently_dirtied.c +++ b/storage/maria/ma_least_recently_dirtied.c @@ -1,3 +1,19 @@ +/* Copyright (C) 2006 MySQL AB & MySQL Finland AB & TCX DataKonsult AB + + 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; either version 2 of the License, or + (at your option) any later version. + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ + /* WL#3261 Maria - background flushing of the least-recently-dirtied pages First version written by Guilhem Bichot on 2006-04-27. diff --git a/storage/maria/ma_least_recently_dirtied.h b/storage/maria/ma_least_recently_dirtied.h index 6a30db4b5f0..f6d7420febc 100644 --- a/storage/maria/ma_least_recently_dirtied.h +++ b/storage/maria/ma_least_recently_dirtied.h @@ -1,3 +1,19 @@ +/* Copyright (C) 2006 MySQL AB & MySQL Finland AB & TCX DataKonsult AB + + 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; either version 2 of the License, or + (at your option) any later version. + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ + /* WL#3261 Maria - background flushing of the least-recently-dirtied pages First version written by Guilhem Bichot on 2006-04-27. diff --git a/storage/maria/ma_recovery.c b/storage/maria/ma_recovery.c index babf7507ef1..b6739b86874 100644 --- a/storage/maria/ma_recovery.c +++ b/storage/maria/ma_recovery.c @@ -1,3 +1,19 @@ +/* Copyright (C) 2006 MySQL AB & MySQL Finland AB & TCX DataKonsult AB + + 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; either version 2 of the License, or + (at your option) any later version. + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ + /* WL#3072 Maria recovery First version written by Guilhem Bichot on 2006-04-27. diff --git a/storage/maria/ma_recovery.h b/storage/maria/ma_recovery.h index b85ffdeef59..05026f4b52a 100644 --- a/storage/maria/ma_recovery.h +++ b/storage/maria/ma_recovery.h @@ -1,3 +1,19 @@ +/* Copyright (C) 2006 MySQL AB & MySQL Finland AB & TCX DataKonsult AB + + 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; either version 2 of the License, or + (at your option) any later version. + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ + /* WL#3072 Maria recovery First version written by Guilhem Bichot on 2006-04-27. diff --git a/storage/maria/unittest/Makefile.am b/storage/maria/unittest/Makefile.am index eae2990aea9..8a5ca3d669f 100644 --- a/storage/maria/unittest/Makefile.am +++ b/storage/maria/unittest/Makefile.am @@ -1,4 +1,4 @@ -# Copyright (C) 2000 MySQL AB & MySQL Finland AB & TCX DataKonsult AB +# Copyright (C) 2006 MySQL AB & MySQL Finland AB & TCX DataKonsult AB # # 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 -- cgit v1.2.1 From 09a7f30973304620e00b934597b72f7dd7eeb966 Mon Sep 17 00:00:00 2001 From: unknown Date: Fri, 15 Sep 2006 11:05:35 +0200 Subject: WL#3234 Maria Control file manager last round of fixes to the storage engines' and plugins' unit tests structure. Will extract a total patch and push it in 5.1 as has been approved. Makefile.am: unittest must be before storage and plugin, because engine and plugin may have unit tests which link with libtap which is found in unitttest. config/ac-macros/plugins.m4: When enabling an engine/plugin, add its directory to the list of directories where unit tests should be searched. That is, its directory will be recursively searched by our unit test framework which will execute any executable *-t file. storage/maria/ma_control_file.c: those my_message pollute the output of unit tests. storage/maria/plug.in: When Maria is enabled, add its unittest Makefile. unittest/Makefile.am: plugins too --- Makefile.am | 4 ++-- config/ac-macros/plugins.m4 | 24 ++++++++---------------- storage/maria/ma_control_file.c | 11 +++++++---- storage/maria/plug.in | 2 +- unittest/Makefile.am | 2 +- 5 files changed, 19 insertions(+), 24 deletions(-) diff --git a/Makefile.am b/Makefile.am index 4adc06acad7..4f0faa59845 100644 --- a/Makefile.am +++ b/Makefile.am @@ -24,11 +24,11 @@ EXTRA_DIST = INSTALL-SOURCE INSTALL-WIN-SOURCE \ SUBDIRS = . include @docs_dirs@ @zlib_dir@ \ @readline_topdir@ sql-common \ @thread_dirs@ pstack \ - @sql_union_dirs@ storage plugin \ + @sql_union_dirs@ unittest storage plugin \ @sql_server@ scripts @man_dirs@ tests \ netware @libmysqld_dirs@ \ mysql-test support-files @tools_dirs@ \ - unittest win + win DIST_SUBDIRS = $(SUBDIRS) BUILD diff --git a/config/ac-macros/plugins.m4 b/config/ac-macros/plugins.m4 index ba7b3fcb087..87f057e696a 100644 --- a/config/ac-macros/plugins.m4 +++ b/config/ac-macros/plugins.m4 @@ -281,6 +281,7 @@ AC_DEFUN([MYSQL_CONFIGURE_PLUGINS],[ AC_SUBST([mysql_se_dirs]) AC_SUBST([mysql_pg_dirs]) AC_SUBST([mysql_se_unittest_dirs]) + AC_SUBST([mysql_pg_unittest_dirs]) ]) ]) ]) @@ -316,7 +317,6 @@ AC_DEFUN([__MYSQL_EMIT_CHECK_PLUGIN],[ ]) AC_MSG_CHECKING([whether to use ]$3) mysql_use_plugin_dir="" - mysql_use_plugin_unittest_dir="" m4_ifdef([$10],[ if test "X[$mysql_plugin_]$2" = Xyes -a \ "X[$with_plugin_]$2" != Xno -o \ @@ -409,26 +409,18 @@ dnl Although this is "pretty", it breaks libmysqld build m4_syscmd(test -f "$6/configure") ifelse(m4_sysval, 0, [AC_CONFIG_SUBDIRS($6)], - [ - AC_CONFIG_FILES($6/Makefile) - m4_syscmd(test -d "$6/unittest") - ifelse(m4_sysval, 0, - [ - mysql_use_plugin_unittest_dir="$6/unittest" - AC_CONFIG_FILES($6/unittest/Makefile) - ], []) - ] + [AC_CONFIG_FILES($6/Makefile)] ) ifelse(m4_substr($6, 0, 8), [storage/], [ - [mysql_se_name="]m4_substr($6, 8)" - mysql_se_dirs="$mysql_se_dirs $mysql_se_name" - if test -n "$mysql_use_plugin_unittest_dir" ; then - mysql_se_unittest_dirs="$mysql_se_unitest_dirs ../$mysql_use_plugin_unittest_dir" - fi + [mysql_se_dirs="$mysql_se_dirs ]m4_substr($6, 8)" + mysql_se_unittest_dirs="$mysql_se_unittest_dirs ../$6" ], m4_substr($6, 0, 7), [plugin/], - [mysql_pg_dirs="$mysql_pg_dirs ]m4_substr($6, 7)", + [ + [mysql_pg_dirs="$mysql_pg_dirs ]m4_substr($6, 7)" + mysql_pg_unittest_dirs="$mysql_pg_unittest_dirs ../$6" + ], [AC_FATAL([don't know how to handle plugin dir ]$6)]) fi ]) diff --git a/storage/maria/ma_control_file.c b/storage/maria/ma_control_file.c index 5b66577938f..5090fac4182 100644 --- a/storage/maria/ma_control_file.c +++ b/storage/maria/ma_control_file.c @@ -188,14 +188,17 @@ CONTROL_FILE_ERROR ma_control_file_create_or_open() disk/filesystem has a problem. So let's be rigid. */ - my_message(0, "too small file", MYF(0)); /* TODO: improve errors */ + /* + TODO: store a message "too small file" somewhere, so that it goes to + MySQL's error log at startup. + */ error= CONTROL_FILE_TOO_SMALL; goto err; } if ((uint)stat_buff.st_size > CONTROL_FILE_SIZE) { - my_message(0, "too big file", MYF(0)); /* TODO: improve errors */ + /* TODO: store "too big file" message */ error= CONTROL_FILE_TOO_BIG; goto err; } @@ -206,7 +209,7 @@ CONTROL_FILE_ERROR ma_control_file_create_or_open() if (memcmp(buffer + CONTROL_FILE_MAGIC_STRING_OFFSET, CONTROL_FILE_MAGIC_STRING, CONTROL_FILE_MAGIC_STRING_SIZE)) { - my_message(0, "bad magic string", MYF(0)); + /* TODO: store message "bad magic string" somewhere */ error= CONTROL_FILE_BAD_MAGIC_STRING; goto err; } @@ -214,7 +217,7 @@ CONTROL_FILE_ERROR ma_control_file_create_or_open() CONTROL_FILE_SIZE - CONTROL_FILE_LSN_OFFSET) != buffer[CONTROL_FILE_CHECKSUM_OFFSET]) { - my_message(0, "checksum mismatch", MYF(0)); + /* TODO: store message "checksum mismatch" somewhere */ error= CONTROL_FILE_BAD_CHECKSUM; goto err; } diff --git a/storage/maria/plug.in b/storage/maria/plug.in index de74293bd96..a9b35aefbfb 100644 --- a/storage/maria/plug.in +++ b/storage/maria/plug.in @@ -1,7 +1,7 @@ MYSQL_STORAGE_ENGINE(maria, no, [Maria Storage Engine], [Traditional transactional MySQL tables], [max,max-no-ndb]) MYSQL_PLUGIN_DIRECTORY(maria, [storage/maria]) +MYSQL_PLUGIN_ACTIONS(maria, [AC_CONFIG_FILES(storage/maria/unittest/Makefile)]) MYSQL_PLUGIN_STATIC(maria, [libmaria.a]) # Maria will probably go first into max builds, not all builds, # so we don't declare it mandatory. - diff --git a/unittest/Makefile.am b/unittest/Makefile.am index 76004e67fae..7323ef5fb77 100644 --- a/unittest/Makefile.am +++ b/unittest/Makefile.am @@ -4,7 +4,7 @@ noinst_SCRIPTS = unit EXTRA_DIST = unit.pl CLEANFILES = unit -unittests = mytap mysys @mysql_se_unittest_dirs@ +unittests = mytap mysys @mysql_se_unittest_dirs@ @mysql_pg_unittest_dirs@ test: unit ./unit run $(unittests) -- cgit v1.2.1 From b860539e37bb585794e144d5785906af13e0e1b9 Mon Sep 17 00:00:00 2001 From: unknown Date: Wed, 20 Sep 2006 10:42:15 +0200 Subject: Maria, very minor changes. mysys/Makefile.am: test_pagecache* programs need to link with dbug (I used BUILD/compile-pentium64-valgrind-max) mysys/mf_pagecache.c: unused constant --- mysys/Makefile.am | 3 ++- mysys/mf_pagecache.c | 1 - 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/mysys/Makefile.am b/mysys/Makefile.am index 6317d6c9285..edafd128eb0 100644 --- a/mysys/Makefile.am +++ b/mysys/Makefile.am @@ -20,7 +20,8 @@ MYSQLBASEdir= $(prefix) INCLUDES = @ZLIB_INCLUDES@ -I$(top_builddir)/include \ -I$(top_srcdir)/include -I$(srcdir) pkglib_LIBRARIES = libmysys.a -LDADD = libmysys.a $(top_builddir)/strings/libmystrings.a +LDADD = libmysys.a $(top_builddir)/strings/libmystrings.a \ + $(top_builddir)/dbug/libdbug.a noinst_HEADERS = mysys_priv.h my_static.h libmysys_a_SOURCES = my_init.c my_getwd.c mf_getdate.c my_mmap.c \ mf_path.c mf_loadpath.c my_file.c \ diff --git a/mysys/mf_pagecache.c b/mysys/mf_pagecache.c index 3a054077809..9c6f586be8a 100755 --- a/mysys/mf_pagecache.c +++ b/mysys/mf_pagecache.c @@ -154,7 +154,6 @@ struct st_pagecache_hash_link #define BLOCK_IN_FLUSH 16 /* block is in flush operation */ #define BLOCK_CHANGED 32 /* block buffer contains a dirty page */ #define BLOCK_WRLOCK 64 /* write locked block */ -#define BLOCK_CPYWRT 128 /* block buffer is in copy&write (see also cpyrd) */ /* page status, returned by find_key_block */ #define PAGE_READ 0 -- cgit v1.2.1 From 8e04cdb2dd1b5102e578c9c98b220a07857bfcbb Mon Sep 17 00:00:00 2001 From: unknown Date: Thu, 21 Sep 2006 23:12:56 +0200 Subject: Maria: fixes for build failures in pushbuild. Comments, fixing of function names. mysys/mf_pagecache.c: comments fixing. More comments. pagecache_ulock_block->page_unlock_block sql/mysqld.cc: MyISAM is always enabled so Maria needs have_maria which MyISAM does not need. This should fix a link failure in pushbuild storage/Makefile.am: force myisam to be built before maria (will not be needed when Maria does not depend on MyISAM anymore) --- mysys/mf_pagecache.c | 25 +++++++++++++++---------- sql/mysqld.cc | 2 ++ storage/Makefile.am | 7 ++++++- 3 files changed, 23 insertions(+), 11 deletions(-) diff --git a/mysys/mf_pagecache.c b/mysys/mf_pagecache.c index 9c6f586be8a..4fa814d8188 100755 --- a/mysys/mf_pagecache.c +++ b/mysys/mf_pagecache.c @@ -523,13 +523,13 @@ static int ___pagecache_pthread_cond_signal(pthread_cond_t *cond); /* - Read page from the disk + Write page to the disk SYNOPSIS pagecache_fwrite() pagecache - page cache pointer filedesc - pagecache file descriptor structure - buffer - buffer in which we will read + buffer - buffer which we will write type - page type (plain or with LSN) flags - MYF() flags @@ -571,6 +571,10 @@ uint pagecache_fwrite(PAGECACHE *pagecache, (pageno)<<(pagecache->shift), flags) +/* + next_power(value) is 2 at the power of (1+floor(log2(value))); + e.g. next_power(2)=4, next_power(3)=4. +*/ static uint next_power(uint value) { uint old_value= 1; @@ -2167,9 +2171,9 @@ my_bool pagecache_lock_block(PAGECACHE *pagecache, DBUG_RETURN(0); } -void pagecache_ulock_block(PAGECACHE_BLOCK_LINK *block) +void pagecache_unlock_block(PAGECACHE_BLOCK_LINK *block) { - DBUG_ENTER("pagecache_ulock_block"); + DBUG_ENTER("pagecache_unlock_block"); BLOCK_INFO(block); DBUG_ASSERT(block->status & BLOCK_WRLOCK); block->status&= ~BLOCK_WRLOCK; @@ -2234,7 +2238,7 @@ my_bool pagecache_make_lock_and_pin(PAGECACHE *pagecache, Removes writelog and puts read lock (which is nothing in our implementation) */ - pagecache_ulock_block(block); + pagecache_unlock_block(block); case PAGECACHE_LOCK_READ_UNLOCK: /* read -> free */ case PAGECACHE_LOCK_LEFT_READLOCKED: /* read -> read */ #ifndef DBUG_OFF @@ -2247,8 +2251,9 @@ my_bool pagecache_make_lock_and_pin(PAGECACHE *pagecache, if (lock == PAGECACHE_LOCK_WRITE_TO_READ) { pagecache_change_lock(block, 0); - } else if (lock == PAGECACHE_LOCK_WRITE_UNLOCK || - lock == PAGECACHE_LOCK_READ_UNLOCK) + } + else if (lock == PAGECACHE_LOCK_WRITE_UNLOCK || + lock == PAGECACHE_LOCK_READ_UNLOCK) { pagecache_remove_lock(block); } @@ -2422,8 +2427,8 @@ void pagecache_unlock_page(PAGECACHE *pagecache, pagecache_pthread_mutex_lock(&pagecache->cache_lock); /* - As soon as we keep lock cache can be used, and we have lock bacause want - aunlock. + As soon as we keep lock cache can be used, and we have lock because want + to unlock. */ DBUG_ASSERT(pagecache->can_be_used); @@ -2891,7 +2896,7 @@ restart: link= get_present_hash_link(pagecache, file, pageno, &unused_start); if (!link) { - DBUG_PRINT("info", ("There is no fuch page in the cache")); + DBUG_PRINT("info", ("There is no such page in the cache")); DBUG_RETURN(0); } block= link->block; diff --git a/sql/mysqld.cc b/sql/mysqld.cc index f1d550d0ff1..684b327df6d 100644 --- a/sql/mysqld.cc +++ b/sql/mysqld.cc @@ -8108,6 +8108,7 @@ void refresh_status(THD *thd) #undef have_federated_db #undef have_partition_db #undef have_blackhole_db +#undef have_maria SHOW_COMP_OPTION have_innodb= SHOW_OPTION_NO; SHOW_COMP_OPTION have_ndbcluster= SHOW_OPTION_NO; @@ -8117,6 +8118,7 @@ SHOW_COMP_OPTION have_csv_db= SHOW_OPTION_NO; SHOW_COMP_OPTION have_federated_db= SHOW_OPTION_NO; SHOW_COMP_OPTION have_partition_db= SHOW_OPTION_NO; SHOW_COMP_OPTION have_blackhole_db= SHOW_OPTION_NO; +SHOW_COMP_OPTION have_maria= SHOW_OPTION_NO; #ifndef WITH_INNOBASE_STORAGE_ENGINE uint innobase_flush_log_at_trx_commit; diff --git a/storage/Makefile.am b/storage/Makefile.am index 900e486c6ac..e5d7ac2778f 100644 --- a/storage/Makefile.am +++ b/storage/Makefile.am @@ -20,7 +20,12 @@ AUTOMAKE_OPTIONS = foreign # These are built from source in the Docs directory EXTRA_DIST = -SUBDIRS = @mysql_se_dirs@ +# Until we remove fulltext-related references from Maria to MyISAM +# MyISAM must be built before Maria, which is not the case by default +# because of alphabetical order +# So we put myisam first; this is very ugly regarding plugins' logic +# but it works, and we'll remove it soon. +SUBDIRS = myisam @mysql_se_dirs@ # Don't update the files from bitkeeper %::SCCS/s.% -- cgit v1.2.1 From 0a1dc8af5b665647a08c51f75b5d5c2e1a112d99 Mon Sep 17 00:00:00 2001 From: unknown Date: Wed, 11 Oct 2006 19:30:16 +0300 Subject: Merge of recent MyISAM changes into Maria. Testsuite passes as much as in the main 5.1 (partition and ndb_alter_table fail). mysql-test/r/maria.result: merge from MyISAM mysql-test/r/ps_maria.result: merge from MyISAM mysql-test/t/maria.test: merge from MyISAM sql/mysql_priv.h: fix after wrong merge sql/mysqld.cc: fix after wrong merge sql/set_var.cc: adding _db like other engines have storage/maria/Makefile.am: merge from MyISAM storage/maria/ha_maria.cc: merge from MyISAM storage/maria/ha_maria.h: merge from MyISAM storage/maria/ma_check.c: merge from MyISAM storage/maria/ma_delete.c: merge from MyISAM storage/maria/ma_init.c: maria_inited should rather be my_bool storage/maria/ma_locking.c: merge from MyISAM storage/maria/ma_packrec.c: merge from MyISAM storage/maria/ma_panic.c: maria_panic() should not take mutex if engine has not been inited. storage/maria/ma_rkey.c: merge from MyISAM storage/maria/ma_write.c: merge from MyISAM storage/maria/maria_def.h: merge from MyISAM. maria_inited is needed for maria_panic(). storage/maria/maria_ftdump.c: merge from MyISAM --- mysql-test/r/maria.result | 28 ++++ mysql-test/r/ps_maria.result | 320 +++++++++++++++++++++---------------------- mysql-test/t/maria.test | 36 +++++ sql/mysql_priv.h | 3 + sql/mysqld.cc | 5 + sql/set_var.cc | 4 +- storage/maria/Makefile.am | 2 +- storage/maria/ha_maria.cc | 54 ++++++-- storage/maria/ha_maria.h | 6 +- storage/maria/ma_check.c | 15 +- storage/maria/ma_delete.c | 2 +- storage/maria/ma_init.c | 6 +- storage/maria/ma_locking.c | 15 ++ storage/maria/ma_packrec.c | 18 ++- storage/maria/ma_panic.c | 2 + storage/maria/ma_rkey.c | 55 +++++--- storage/maria/ma_write.c | 8 +- storage/maria/maria_def.h | 8 ++ storage/maria/maria_ftdump.c | 3 +- 19 files changed, 368 insertions(+), 222 deletions(-) diff --git a/mysql-test/r/maria.result b/mysql-test/r/maria.result index b6924e4395b..396b5da4bc1 100644 --- a/mysql-test/r/maria.result +++ b/mysql-test/r/maria.result @@ -517,6 +517,34 @@ select c1 from t1 order by c1 limit 1; c1 a drop table t1; +create table t1 (a int not null, primary key(a)); +create table t2 (a int not null, b int not null, primary key(a,b)); +insert into t1 values (1),(2),(3),(4),(5),(6); +insert into t2 values (1,1),(2,1); +lock tables t1 read local, t2 read local; +select straight_join * from t1,t2 force index (primary) where t1.a=t2.a; +a a b +1 1 1 +2 2 1 +insert into t2 values(2,0); +select straight_join * from t1,t2 force index (primary) where t1.a=t2.a; +a a b +1 1 1 +2 2 1 +drop table t1,t2; +CREATE TABLE t1 (c1 varchar(250) NOT NULL); +CREATE TABLE t2 (c1 varchar(250) NOT NULL, PRIMARY KEY (c1)); +INSERT INTO t1 VALUES ('test000001'), ('test000002'), ('test000003'); +INSERT INTO t2 VALUES ('test000002'), ('test000003'), ('test000004'); +LOCK TABLES t1 READ LOCAL, t2 READ LOCAL; +SELECT t1.c1 AS t1c1, t2.c1 AS t2c1 FROM t1, t2 +WHERE t1.c1 = t2.c1 HAVING t1c1 != t2c1; +t1c1 t2c1 +INSERT INTO t2 VALUES ('test000001'), ('test000005'); +SELECT t1.c1 AS t1c1, t2.c1 AS t2c1 FROM t1, t2 +WHERE t1.c1 = t2.c1 HAVING t1c1 != t2c1; +t1c1 t2c1 +DROP TABLE t1,t2; CREATE TABLE t1 (`a` int(11) NOT NULL default '0', `b` int(11) NOT NULL default '0', UNIQUE KEY `a` USING RTREE (`a`,`b`)); Got one of the listed errors create table t1 (a int, b varchar(200), c text not null) checksum=1; diff --git a/mysql-test/r/ps_maria.result b/mysql-test/r/ps_maria.result index f47c72187e2..9268c44eecd 100644 --- a/mysql-test/r/ps_maria.result +++ b/mysql-test/r/ps_maria.result @@ -1929,26 +1929,26 @@ def @arg09 253 23 1 Y 128 31 63 def @arg10 253 23 1 Y 128 31 63 def @arg11 253 67 6 Y 128 30 63 def @arg12 253 67 6 Y 128 30 63 -def @arg13 253 8192 10 Y 128 31 63 -def @arg14 253 8192 19 Y 128 31 63 -def @arg15 253 8192 19 Y 128 31 63 -def @arg16 253 8192 8 Y 128 31 63 +def @arg13 253 16777216 10 Y 128 31 63 +def @arg14 253 16777216 19 Y 128 31 63 +def @arg15 253 16777216 19 Y 128 31 63 +def @arg16 253 16777216 8 Y 128 31 63 def @arg17 253 20 4 Y 128 0 63 def @arg18 253 20 1 Y 128 0 63 def @arg19 253 20 1 Y 128 0 63 -def @arg20 253 8192 1 Y 0 31 8 -def @arg21 253 8192 10 Y 0 31 8 -def @arg22 253 8192 30 Y 0 31 8 -def @arg23 253 8192 8 Y 128 31 63 -def @arg24 253 8192 8 Y 0 31 8 -def @arg25 253 8192 4 Y 128 31 63 -def @arg26 253 8192 4 Y 0 31 8 -def @arg27 253 8192 10 Y 128 31 63 -def @arg28 253 8192 10 Y 0 31 8 -def @arg29 253 8192 8 Y 128 31 63 -def @arg30 253 8192 8 Y 0 31 8 -def @arg31 253 8192 3 Y 0 31 8 -def @arg32 253 8192 6 Y 0 31 8 +def @arg20 253 16777216 1 Y 0 31 8 +def @arg21 253 16777216 10 Y 0 31 8 +def @arg22 253 16777216 30 Y 0 31 8 +def @arg23 253 16777216 8 Y 128 31 63 +def @arg24 253 16777216 8 Y 0 31 8 +def @arg25 253 16777216 4 Y 128 31 63 +def @arg26 253 16777216 4 Y 0 31 8 +def @arg27 253 16777216 10 Y 128 31 63 +def @arg28 253 16777216 10 Y 0 31 8 +def @arg29 253 16777216 8 Y 128 31 63 +def @arg30 253 16777216 8 Y 0 31 8 +def @arg31 253 16777216 3 Y 0 31 8 +def @arg32 253 16777216 6 Y 0 31 8 @arg01 @arg02 @arg03 @arg04 @arg05 @arg06 @arg07 @arg08 @arg09 @arg10 @arg11 @arg12 @arg13 @arg14 @arg15 @arg16 @arg17 @arg18 @arg19 @arg20 @arg21 @arg22 @arg23 @arg24 @arg25 @arg26 @arg27 @arg28 @arg29 @arg30 @arg31 @arg32 1 1 1 1 1 1 1 1 1 1 1.0000 1.0000 2004-02-29 2004-02-29 11:11:11 2004-02-29 11:11:11 11:11:11 2004 1 1 a 123456789a 123456789a123456789b123456789c tinyblob tinytext blob text mediumblob mediumtext longblob longtext one monday select @arg01:= c1, @arg02:= c2, @arg03:= c3, @arg04:= c4, @@ -1976,26 +1976,26 @@ def @arg09 253 23 0 Y 128 31 63 def @arg10 253 23 0 Y 128 31 63 def @arg11 253 67 0 Y 128 30 63 def @arg12 253 67 0 Y 128 30 63 -def @arg13 253 8192 0 Y 128 31 63 -def @arg14 253 8192 0 Y 128 31 63 -def @arg15 253 8192 19 Y 128 31 63 -def @arg16 253 8192 0 Y 128 31 63 +def @arg13 253 16777216 0 Y 128 31 63 +def @arg14 253 16777216 0 Y 128 31 63 +def @arg15 253 16777216 19 Y 128 31 63 +def @arg16 253 16777216 0 Y 128 31 63 def @arg17 253 20 0 Y 128 0 63 def @arg18 253 20 0 Y 128 0 63 def @arg19 253 20 0 Y 128 0 63 -def @arg20 253 8192 0 Y 0 31 8 -def @arg21 253 8192 0 Y 0 31 8 -def @arg22 253 8192 0 Y 0 31 8 -def @arg23 253 8192 0 Y 128 31 63 -def @arg24 253 8192 0 Y 0 31 8 -def @arg25 253 8192 0 Y 128 31 63 -def @arg26 253 8192 0 Y 0 31 8 -def @arg27 253 8192 0 Y 128 31 63 -def @arg28 253 8192 0 Y 0 31 8 -def @arg29 253 8192 0 Y 128 31 63 -def @arg30 253 8192 0 Y 0 31 8 -def @arg31 253 8192 0 Y 0 31 8 -def @arg32 253 8192 0 Y 0 31 8 +def @arg20 253 16777216 0 Y 0 31 8 +def @arg21 253 16777216 0 Y 0 31 8 +def @arg22 253 16777216 0 Y 0 31 8 +def @arg23 253 16777216 0 Y 128 31 63 +def @arg24 253 16777216 0 Y 0 31 8 +def @arg25 253 16777216 0 Y 128 31 63 +def @arg26 253 16777216 0 Y 0 31 8 +def @arg27 253 16777216 0 Y 128 31 63 +def @arg28 253 16777216 0 Y 0 31 8 +def @arg29 253 16777216 0 Y 128 31 63 +def @arg30 253 16777216 0 Y 0 31 8 +def @arg31 253 16777216 0 Y 0 31 8 +def @arg32 253 16777216 0 Y 0 31 8 @arg01 @arg02 @arg03 @arg04 @arg05 @arg06 @arg07 @arg08 @arg09 @arg10 @arg11 @arg12 @arg13 @arg14 @arg15 @arg16 @arg17 @arg18 @arg19 @arg20 @arg21 @arg22 @arg23 @arg24 @arg25 @arg26 @arg27 @arg28 @arg29 @arg30 @arg31 @arg32 0 NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL 1991-01-01 01:01:01 NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL prepare stmt1 from "select @@ -2026,26 +2026,26 @@ def @arg09 253 23 1 Y 128 31 63 def @arg10 253 23 1 Y 128 31 63 def @arg11 253 67 6 Y 128 30 63 def @arg12 253 67 6 Y 128 30 63 -def @arg13 253 8192 10 Y 128 31 63 -def @arg14 253 8192 19 Y 128 31 63 -def @arg15 253 8192 19 Y 128 31 63 -def @arg16 253 8192 8 Y 128 31 63 +def @arg13 253 16777216 10 Y 128 31 63 +def @arg14 253 16777216 19 Y 128 31 63 +def @arg15 253 16777216 19 Y 128 31 63 +def @arg16 253 16777216 8 Y 128 31 63 def @arg17 253 20 4 Y 128 0 63 def @arg18 253 20 1 Y 128 0 63 def @arg19 253 20 1 Y 128 0 63 -def @arg20 253 8192 1 Y 0 31 8 -def @arg21 253 8192 10 Y 0 31 8 -def @arg22 253 8192 30 Y 0 31 8 -def @arg23 253 8192 8 Y 128 31 63 -def @arg24 253 8192 8 Y 0 31 8 -def @arg25 253 8192 4 Y 128 31 63 -def @arg26 253 8192 4 Y 0 31 8 -def @arg27 253 8192 10 Y 128 31 63 -def @arg28 253 8192 10 Y 0 31 8 -def @arg29 253 8192 8 Y 128 31 63 -def @arg30 253 8192 8 Y 0 31 8 -def @arg31 253 8192 3 Y 0 31 8 -def @arg32 253 8192 6 Y 0 31 8 +def @arg20 253 16777216 1 Y 0 31 8 +def @arg21 253 16777216 10 Y 0 31 8 +def @arg22 253 16777216 30 Y 0 31 8 +def @arg23 253 16777216 8 Y 128 31 63 +def @arg24 253 16777216 8 Y 0 31 8 +def @arg25 253 16777216 4 Y 128 31 63 +def @arg26 253 16777216 4 Y 0 31 8 +def @arg27 253 16777216 10 Y 128 31 63 +def @arg28 253 16777216 10 Y 0 31 8 +def @arg29 253 16777216 8 Y 128 31 63 +def @arg30 253 16777216 8 Y 0 31 8 +def @arg31 253 16777216 3 Y 0 31 8 +def @arg32 253 16777216 6 Y 0 31 8 @arg01 @arg02 @arg03 @arg04 @arg05 @arg06 @arg07 @arg08 @arg09 @arg10 @arg11 @arg12 @arg13 @arg14 @arg15 @arg16 @arg17 @arg18 @arg19 @arg20 @arg21 @arg22 @arg23 @arg24 @arg25 @arg26 @arg27 @arg28 @arg29 @arg30 @arg31 @arg32 1 1 1 1 1 1 1 1 1 1 1.0000 1.0000 2004-02-29 2004-02-29 11:11:11 2004-02-29 11:11:11 11:11:11 2004 1 1 a 123456789a 123456789a123456789b123456789c tinyblob tinytext blob text mediumblob mediumtext longblob longtext one monday set @my_key= 0 ; @@ -2066,26 +2066,26 @@ def @arg09 253 23 0 Y 128 31 63 def @arg10 253 23 0 Y 128 31 63 def @arg11 253 67 0 Y 128 30 63 def @arg12 253 67 0 Y 128 30 63 -def @arg13 253 8192 0 Y 128 31 63 -def @arg14 253 8192 0 Y 128 31 63 -def @arg15 253 8192 19 Y 128 31 63 -def @arg16 253 8192 0 Y 128 31 63 +def @arg13 253 16777216 0 Y 128 31 63 +def @arg14 253 16777216 0 Y 128 31 63 +def @arg15 253 16777216 19 Y 128 31 63 +def @arg16 253 16777216 0 Y 128 31 63 def @arg17 253 20 0 Y 128 0 63 def @arg18 253 20 0 Y 128 0 63 def @arg19 253 20 0 Y 128 0 63 -def @arg20 253 8192 0 Y 0 31 8 -def @arg21 253 8192 0 Y 0 31 8 -def @arg22 253 8192 0 Y 0 31 8 -def @arg23 253 8192 0 Y 128 31 63 -def @arg24 253 8192 0 Y 0 31 8 -def @arg25 253 8192 0 Y 128 31 63 -def @arg26 253 8192 0 Y 0 31 8 -def @arg27 253 8192 0 Y 128 31 63 -def @arg28 253 8192 0 Y 0 31 8 -def @arg29 253 8192 0 Y 128 31 63 -def @arg30 253 8192 0 Y 0 31 8 -def @arg31 253 8192 0 Y 0 31 8 -def @arg32 253 8192 0 Y 0 31 8 +def @arg20 253 16777216 0 Y 0 31 8 +def @arg21 253 16777216 0 Y 0 31 8 +def @arg22 253 16777216 0 Y 0 31 8 +def @arg23 253 16777216 0 Y 128 31 63 +def @arg24 253 16777216 0 Y 0 31 8 +def @arg25 253 16777216 0 Y 128 31 63 +def @arg26 253 16777216 0 Y 0 31 8 +def @arg27 253 16777216 0 Y 128 31 63 +def @arg28 253 16777216 0 Y 0 31 8 +def @arg29 253 16777216 0 Y 128 31 63 +def @arg30 253 16777216 0 Y 0 31 8 +def @arg31 253 16777216 0 Y 0 31 8 +def @arg32 253 16777216 0 Y 0 31 8 @arg01 @arg02 @arg03 @arg04 @arg05 @arg06 @arg07 @arg08 @arg09 @arg10 @arg11 @arg12 @arg13 @arg14 @arg15 @arg16 @arg17 @arg18 @arg19 @arg20 @arg21 @arg22 @arg23 @arg24 @arg25 @arg26 @arg27 @arg28 @arg29 @arg30 @arg31 @arg32 0 NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL 1991-01-01 01:01:01 NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL prepare stmt1 from "select ? := c1 from t9 where c1= 1" ; @@ -2114,26 +2114,26 @@ def @arg09 253 23 1 Y 128 31 63 def @arg10 253 23 1 Y 128 31 63 def @arg11 253 67 6 Y 128 30 63 def @arg12 253 67 6 Y 128 30 63 -def @arg13 253 8192 10 Y 128 31 63 -def @arg14 253 8192 19 Y 128 31 63 -def @arg15 253 8192 19 Y 128 31 63 -def @arg16 253 8192 8 Y 128 31 63 +def @arg13 253 16777216 10 Y 128 31 63 +def @arg14 253 16777216 19 Y 128 31 63 +def @arg15 253 16777216 19 Y 128 31 63 +def @arg16 253 16777216 8 Y 128 31 63 def @arg17 253 20 4 Y 128 0 63 def @arg18 253 20 1 Y 128 0 63 def @arg19 253 20 1 Y 128 0 63 -def @arg20 253 8192 1 Y 0 31 8 -def @arg21 253 8192 10 Y 0 31 8 -def @arg22 253 8192 30 Y 0 31 8 -def @arg23 253 8192 8 Y 128 31 63 -def @arg24 253 8192 8 Y 0 31 8 -def @arg25 253 8192 4 Y 128 31 63 -def @arg26 253 8192 4 Y 0 31 8 -def @arg27 253 8192 10 Y 128 31 63 -def @arg28 253 8192 10 Y 0 31 8 -def @arg29 253 8192 8 Y 128 31 63 -def @arg30 253 8192 8 Y 0 31 8 -def @arg31 253 8192 3 Y 0 31 8 -def @arg32 253 8192 6 Y 0 31 8 +def @arg20 253 16777216 1 Y 0 31 8 +def @arg21 253 16777216 10 Y 0 31 8 +def @arg22 253 16777216 30 Y 0 31 8 +def @arg23 253 16777216 8 Y 128 31 63 +def @arg24 253 16777216 8 Y 0 31 8 +def @arg25 253 16777216 4 Y 128 31 63 +def @arg26 253 16777216 4 Y 0 31 8 +def @arg27 253 16777216 10 Y 128 31 63 +def @arg28 253 16777216 10 Y 0 31 8 +def @arg29 253 16777216 8 Y 128 31 63 +def @arg30 253 16777216 8 Y 0 31 8 +def @arg31 253 16777216 3 Y 0 31 8 +def @arg32 253 16777216 6 Y 0 31 8 @arg01 @arg02 @arg03 @arg04 @arg05 @arg06 @arg07 @arg08 @arg09 @arg10 @arg11 @arg12 @arg13 @arg14 @arg15 @arg16 @arg17 @arg18 @arg19 @arg20 @arg21 @arg22 @arg23 @arg24 @arg25 @arg26 @arg27 @arg28 @arg29 @arg30 @arg31 @arg32 1 1 1 1 1 1 1 1 1 1 1.0000 1.0000 2004-02-29 2004-02-29 11:11:11 2004-02-29 11:11:11 11:11:11 2004 1 1 a 123456789a 123456789a123456789b123456789c tinyblob tinytext blob text mediumblob mediumtext longblob longtext one monday select c1, c2, c3, c4, c5, c6, c7, c8, c9, c10, c11, c12, @@ -2158,26 +2158,26 @@ def @arg09 253 23 0 Y 128 31 63 def @arg10 253 23 0 Y 128 31 63 def @arg11 253 67 0 Y 128 30 63 def @arg12 253 67 0 Y 128 30 63 -def @arg13 253 8192 0 Y 128 31 63 -def @arg14 253 8192 0 Y 128 31 63 -def @arg15 253 8192 19 Y 128 31 63 -def @arg16 253 8192 0 Y 128 31 63 +def @arg13 253 16777216 0 Y 128 31 63 +def @arg14 253 16777216 0 Y 128 31 63 +def @arg15 253 16777216 19 Y 128 31 63 +def @arg16 253 16777216 0 Y 128 31 63 def @arg17 253 20 0 Y 128 0 63 def @arg18 253 20 0 Y 128 0 63 def @arg19 253 20 0 Y 128 0 63 -def @arg20 253 8192 0 Y 0 31 8 -def @arg21 253 8192 0 Y 0 31 8 -def @arg22 253 8192 0 Y 0 31 8 -def @arg23 253 8192 0 Y 128 31 63 -def @arg24 253 8192 0 Y 0 31 8 -def @arg25 253 8192 0 Y 128 31 63 -def @arg26 253 8192 0 Y 0 31 8 -def @arg27 253 8192 0 Y 128 31 63 -def @arg28 253 8192 0 Y 0 31 8 -def @arg29 253 8192 0 Y 128 31 63 -def @arg30 253 8192 0 Y 0 31 8 -def @arg31 253 8192 0 Y 0 31 8 -def @arg32 253 8192 0 Y 0 31 8 +def @arg20 253 16777216 0 Y 0 31 8 +def @arg21 253 16777216 0 Y 0 31 8 +def @arg22 253 16777216 0 Y 0 31 8 +def @arg23 253 16777216 0 Y 128 31 63 +def @arg24 253 16777216 0 Y 0 31 8 +def @arg25 253 16777216 0 Y 128 31 63 +def @arg26 253 16777216 0 Y 0 31 8 +def @arg27 253 16777216 0 Y 128 31 63 +def @arg28 253 16777216 0 Y 0 31 8 +def @arg29 253 16777216 0 Y 128 31 63 +def @arg30 253 16777216 0 Y 0 31 8 +def @arg31 253 16777216 0 Y 0 31 8 +def @arg32 253 16777216 0 Y 0 31 8 @arg01 @arg02 @arg03 @arg04 @arg05 @arg06 @arg07 @arg08 @arg09 @arg10 @arg11 @arg12 @arg13 @arg14 @arg15 @arg16 @arg17 @arg18 @arg19 @arg20 @arg21 @arg22 @arg23 @arg24 @arg25 @arg26 @arg27 @arg28 @arg29 @arg30 @arg31 @arg32 0 NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL 1991-01-01 01:01:01 NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL prepare stmt1 from "select c1, c2, c3, c4, c5, c6, c7, c8, c9, c10, c11, c12, @@ -2204,26 +2204,26 @@ def @arg09 253 23 1 Y 128 31 63 def @arg10 253 23 1 Y 128 31 63 def @arg11 253 67 6 Y 128 30 63 def @arg12 253 67 6 Y 128 30 63 -def @arg13 253 8192 10 Y 128 31 63 -def @arg14 253 8192 19 Y 128 31 63 -def @arg15 253 8192 19 Y 128 31 63 -def @arg16 253 8192 8 Y 128 31 63 +def @arg13 253 16777216 10 Y 128 31 63 +def @arg14 253 16777216 19 Y 128 31 63 +def @arg15 253 16777216 19 Y 128 31 63 +def @arg16 253 16777216 8 Y 128 31 63 def @arg17 253 20 4 Y 128 0 63 def @arg18 253 20 1 Y 128 0 63 def @arg19 253 20 1 Y 128 0 63 -def @arg20 253 8192 1 Y 0 31 8 -def @arg21 253 8192 10 Y 0 31 8 -def @arg22 253 8192 30 Y 0 31 8 -def @arg23 253 8192 8 Y 128 31 63 -def @arg24 253 8192 8 Y 0 31 8 -def @arg25 253 8192 4 Y 128 31 63 -def @arg26 253 8192 4 Y 0 31 8 -def @arg27 253 8192 10 Y 128 31 63 -def @arg28 253 8192 10 Y 0 31 8 -def @arg29 253 8192 8 Y 128 31 63 -def @arg30 253 8192 8 Y 0 31 8 -def @arg31 253 8192 3 Y 0 31 8 -def @arg32 253 8192 6 Y 0 31 8 +def @arg20 253 16777216 1 Y 0 31 8 +def @arg21 253 16777216 10 Y 0 31 8 +def @arg22 253 16777216 30 Y 0 31 8 +def @arg23 253 16777216 8 Y 128 31 63 +def @arg24 253 16777216 8 Y 0 31 8 +def @arg25 253 16777216 4 Y 128 31 63 +def @arg26 253 16777216 4 Y 0 31 8 +def @arg27 253 16777216 10 Y 128 31 63 +def @arg28 253 16777216 10 Y 0 31 8 +def @arg29 253 16777216 8 Y 128 31 63 +def @arg30 253 16777216 8 Y 0 31 8 +def @arg31 253 16777216 3 Y 0 31 8 +def @arg32 253 16777216 6 Y 0 31 8 @arg01 @arg02 @arg03 @arg04 @arg05 @arg06 @arg07 @arg08 @arg09 @arg10 @arg11 @arg12 @arg13 @arg14 @arg15 @arg16 @arg17 @arg18 @arg19 @arg20 @arg21 @arg22 @arg23 @arg24 @arg25 @arg26 @arg27 @arg28 @arg29 @arg30 @arg31 @arg32 1 1 1 1 1 1 1 1 1 1 1.0000 1.0000 2004-02-29 2004-02-29 11:11:11 2004-02-29 11:11:11 11:11:11 2004 1 1 a 123456789a 123456789a123456789b123456789c tinyblob tinytext blob text mediumblob mediumtext longblob longtext one monday set @my_key= 0 ; @@ -2242,26 +2242,26 @@ def @arg09 253 23 0 Y 128 31 63 def @arg10 253 23 0 Y 128 31 63 def @arg11 253 67 0 Y 128 30 63 def @arg12 253 67 0 Y 128 30 63 -def @arg13 253 8192 0 Y 128 31 63 -def @arg14 253 8192 0 Y 128 31 63 -def @arg15 253 8192 19 Y 128 31 63 -def @arg16 253 8192 0 Y 128 31 63 +def @arg13 253 16777216 0 Y 128 31 63 +def @arg14 253 16777216 0 Y 128 31 63 +def @arg15 253 16777216 19 Y 128 31 63 +def @arg16 253 16777216 0 Y 128 31 63 def @arg17 253 20 0 Y 128 0 63 def @arg18 253 20 0 Y 128 0 63 def @arg19 253 20 0 Y 128 0 63 -def @arg20 253 8192 0 Y 0 31 8 -def @arg21 253 8192 0 Y 0 31 8 -def @arg22 253 8192 0 Y 0 31 8 -def @arg23 253 8192 0 Y 128 31 63 -def @arg24 253 8192 0 Y 0 31 8 -def @arg25 253 8192 0 Y 128 31 63 -def @arg26 253 8192 0 Y 0 31 8 -def @arg27 253 8192 0 Y 128 31 63 -def @arg28 253 8192 0 Y 0 31 8 -def @arg29 253 8192 0 Y 128 31 63 -def @arg30 253 8192 0 Y 0 31 8 -def @arg31 253 8192 0 Y 0 31 8 -def @arg32 253 8192 0 Y 0 31 8 +def @arg20 253 16777216 0 Y 0 31 8 +def @arg21 253 16777216 0 Y 0 31 8 +def @arg22 253 16777216 0 Y 0 31 8 +def @arg23 253 16777216 0 Y 128 31 63 +def @arg24 253 16777216 0 Y 0 31 8 +def @arg25 253 16777216 0 Y 128 31 63 +def @arg26 253 16777216 0 Y 0 31 8 +def @arg27 253 16777216 0 Y 128 31 63 +def @arg28 253 16777216 0 Y 0 31 8 +def @arg29 253 16777216 0 Y 128 31 63 +def @arg30 253 16777216 0 Y 0 31 8 +def @arg31 253 16777216 0 Y 0 31 8 +def @arg32 253 16777216 0 Y 0 31 8 @arg01 @arg02 @arg03 @arg04 @arg05 @arg06 @arg07 @arg08 @arg09 @arg10 @arg11 @arg12 @arg13 @arg14 @arg15 @arg16 @arg17 @arg18 @arg19 @arg20 @arg21 @arg22 @arg23 @arg24 @arg25 @arg26 @arg27 @arg28 @arg29 @arg30 @arg31 @arg32 0 NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL 1991-01-01 01:01:01 NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL prepare stmt1 from "select c1 into ? from t9 where c1= 1" ; @@ -2689,21 +2689,21 @@ set @arg00= '1.11111111111111111111e+50' ; execute my_insert using @arg00, @arg00, @arg00, @arg00, @arg00, @arg00, @arg00, @arg00, @arg00, @arg00, @arg00 ; Warnings: -Warning 1265 Data truncated for column 'c1' at row 1 -Warning 1265 Data truncated for column 'c2' at row 1 -Warning 1265 Data truncated for column 'c3' at row 1 -Warning 1265 Data truncated for column 'c4' at row 1 -Warning 1265 Data truncated for column 'c5' at row 1 -Warning 1265 Data truncated for column 'c6' at row 1 +Warning 1264 Out of range value for column 'c1' at row 1 +Warning 1264 Out of range value for column 'c2' at row 1 +Warning 1264 Out of range value for column 'c3' at row 1 +Warning 1264 Out of range value for column 'c4' at row 1 +Warning 1264 Out of range value for column 'c5' at row 1 +Warning 1264 Out of range value for column 'c6' at row 1 Warning 1264 Out of range value for column 'c7' at row 1 Warning 1264 Out of range value for column 'c12' at row 1 execute my_select ; -c1 1 -c2 1 -c3 1 -c4 1 -c5 1 -c6 1 +c1 127 +c2 32767 +c3 8388607 +c4 2147483647 +c5 2147483647 +c6 9223372036854775807 c7 3.40282e+38 c8 1.11111111111111e+50 c9 1.11111111111111e+50 @@ -2739,21 +2739,21 @@ set @arg00= '-1.11111111111111111111e+50' ; execute my_insert using @arg00, @arg00, @arg00, @arg00, @arg00, @arg00, @arg00, @arg00, @arg00, @arg00, @arg00 ; Warnings: -Warning 1265 Data truncated for column 'c1' at row 1 -Warning 1265 Data truncated for column 'c2' at row 1 -Warning 1265 Data truncated for column 'c3' at row 1 -Warning 1265 Data truncated for column 'c4' at row 1 -Warning 1265 Data truncated for column 'c5' at row 1 -Warning 1265 Data truncated for column 'c6' at row 1 +Warning 1264 Out of range value for column 'c1' at row 1 +Warning 1264 Out of range value for column 'c2' at row 1 +Warning 1264 Out of range value for column 'c3' at row 1 +Warning 1264 Out of range value for column 'c4' at row 1 +Warning 1264 Out of range value for column 'c5' at row 1 +Warning 1264 Out of range value for column 'c6' at row 1 Warning 1264 Out of range value for column 'c7' at row 1 Warning 1264 Out of range value for column 'c12' at row 1 execute my_select ; -c1 -1 -c2 -1 -c3 -1 -c4 -1 -c5 -1 -c6 -1 +c1 -128 +c2 -32768 +c3 -8388608 +c4 -2147483648 +c5 -2147483648 +c6 -9223372036854775808 c7 -3.40282e+38 c8 -1.11111111111111e+50 c9 -1.11111111111111e+50 diff --git a/mysql-test/t/maria.test b/mysql-test/t/maria.test index 1a44775277b..fa9983690cb 100644 --- a/mysql-test/t/maria.test +++ b/mysql-test/t/maria.test @@ -493,6 +493,42 @@ insert into t1 values ('a'), ('b'); select c1 from t1 order by c1 limit 1; drop table t1; +# +# Bug #14400 Join could miss concurrently inserted row +# +# Partial key. +create table t1 (a int not null, primary key(a)); +create table t2 (a int not null, b int not null, primary key(a,b)); +insert into t1 values (1),(2),(3),(4),(5),(6); +insert into t2 values (1,1),(2,1); +lock tables t1 read local, t2 read local; +select straight_join * from t1,t2 force index (primary) where t1.a=t2.a; +connect (root,localhost,root,,test,$MASTER_MYPORT,master.sock); +insert into t2 values(2,0); +disconnect root; +connection default; +select straight_join * from t1,t2 force index (primary) where t1.a=t2.a; +drop table t1,t2; +# +# Full key. +CREATE TABLE t1 (c1 varchar(250) NOT NULL); +CREATE TABLE t2 (c1 varchar(250) NOT NULL, PRIMARY KEY (c1)); +INSERT INTO t1 VALUES ('test000001'), ('test000002'), ('test000003'); +INSERT INTO t2 VALUES ('test000002'), ('test000003'), ('test000004'); +LOCK TABLES t1 READ LOCAL, t2 READ LOCAL; +SELECT t1.c1 AS t1c1, t2.c1 AS t2c1 FROM t1, t2 + WHERE t1.c1 = t2.c1 HAVING t1c1 != t2c1; +connect (con1,localhost,root,,); +connection con1; +INSERT INTO t2 VALUES ('test000001'), ('test000005'); +disconnect con1; +connection default; +SELECT t1.c1 AS t1c1, t2.c1 AS t2c1 FROM t1, t2 + WHERE t1.c1 = t2.c1 HAVING t1c1 != t2c1; +DROP TABLE t1,t2; + +# End of 4.0 tests + # # Test RTREE index # diff --git a/sql/mysql_priv.h b/sql/mysql_priv.h index 9ac880333f6..4545ce99159 100644 --- a/sql/mysql_priv.h +++ b/sql/mysql_priv.h @@ -1651,6 +1651,9 @@ extern SHOW_COMP_OPTION have_ndbcluster; extern SHOW_COMP_OPTION have_partition_db; extern SHOW_COMP_OPTION have_merge_db; extern SHOW_COMP_OPTION have_maria_db; +extern handlerton *partition_hton; +extern handlerton *myisam_hton; +extern handlerton *heap_hton; extern SHOW_COMP_OPTION have_row_based_replication; extern SHOW_COMP_OPTION have_openssl, have_symlink, have_dlopen; diff --git a/sql/mysqld.cc b/sql/mysqld.cc index 61d0ddbee2f..9dfc725773a 100644 --- a/sql/mysqld.cc +++ b/sql/mysqld.cc @@ -7095,6 +7095,11 @@ static void mysql_init_variables(void) #else have_csv_db= SHOW_OPTION_NO; #endif +#ifdef WITH_MARIA_STORAGE_ENGINE + have_maria_db= SHOW_OPTION_YES; +#else + have_maria_db= SHOW_OPTION_NO; +#endif #ifdef WITH_NDBCLUSTER_STORAGE_ENGINE have_ndbcluster= SHOW_OPTION_DISABLED; #else diff --git a/sql/set_var.cc b/sql/set_var.cc index 44223ca16f9..c41f3afae60 100644 --- a/sql/set_var.cc +++ b/sql/set_var.cc @@ -686,7 +686,7 @@ sys_var_have_variable sys_have_federated_db("have_federated_engine", &have_federated_db); sys_var_have_variable sys_have_geometry("have_geometry", &have_geometry); sys_var_have_variable sys_have_innodb("have_innodb", &have_innodb); -sys_var_have_variable sys_have_maria("have_maria", &have_maria); +sys_var_have_variable sys_have_maria_db("have_maria", &have_maria_db); sys_var_have_variable sys_have_merge_db("have_merge", &have_merge_db); sys_var_have_variable sys_have_ndbcluster("have_ndbcluster", &have_ndbcluster); sys_var_have_variable sys_have_openssl("have_openssl", &have_openssl); @@ -818,7 +818,7 @@ SHOW_VAR init_vars[]= { {sys_have_federated_db.name,(char*) &have_federated_db, SHOW_HAVE}, {sys_have_geometry.name, (char*) &have_geometry, SHOW_HAVE}, {sys_have_innodb.name, (char*) &have_innodb, SHOW_HAVE}, - {sys_have_maria.name, (char*) &have_maria, SHOW_HAVE}, + {sys_have_maria_db.name, (char*) &have_maria_db, SHOW_HAVE}, {sys_have_merge_db.name, (char*) &have_merge_db, SHOW_HAVE}, {sys_have_ndbcluster.name, (char*) &have_ndbcluster, SHOW_HAVE}, {sys_have_openssl.name, (char*) &have_openssl, SHOW_HAVE}, diff --git a/storage/maria/Makefile.am b/storage/maria/Makefile.am index 100000fa6cd..eb8cdfa3aba 100644 --- a/storage/maria/Makefile.am +++ b/storage/maria/Makefile.am @@ -31,7 +31,7 @@ DEFS = @DEFS@ # "." is needed first because tests in unittest need libmaria SUBDIRS = . unittest -EXTRA_DIST = ma_test_all.sh ma_test_all.res ma_ft_stem.c CMakeLists.txt +EXTRA_DIST = ma_test_all.sh ma_test_all.res ma_ft_stem.c CMakeLists.txt plug.in pkgdata_DATA = ma_test_all ma_test_all.res pkglib_LIBRARIES = libmaria.a bin_PROGRAMS = maria_chk maria_pack maria_ftdump diff --git a/storage/maria/ha_maria.cc b/storage/maria/ha_maria.cc index a0084ef5e9c..4cdf0c4c086 100644 --- a/storage/maria/ha_maria.cc +++ b/storage/maria/ha_maria.cc @@ -58,9 +58,11 @@ TYPELIB maria_stats_method_typelib= ** MARIA tables *****************************************************************************/ -static handler *maria_create_handler(TABLE_SHARE * table, MEM_ROOT *mem_root) +static handler *maria_create_handler(handlerton *hton, + TABLE_SHARE * table, + MEM_ROOT *mem_root) { - return new (mem_root) ha_maria(table); + return new (mem_root) ha_maria(hton, table); } @@ -147,8 +149,8 @@ void _ma_check_print_warning(HA_CHECK *param, const char *fmt, ...) } -ha_maria::ha_maria(TABLE_SHARE *table_arg): -handler(&maria_hton, table_arg), file(0), +ha_maria::ha_maria(handlerton *hton, TABLE_SHARE *table_arg): +handler(hton, table_arg), file(0), int_table_flags(HA_NULL_IN_KEY | HA_CAN_FULLTEXT | HA_CAN_SQL_HANDLER | HA_DUPLICATE_POS | HA_CAN_INDEX_BLOBS | HA_AUTO_PART_KEY | HA_FILE_BASED | HA_CAN_GEOMETRY | HA_NO_TRANSACTIONS | @@ -158,6 +160,15 @@ can_enable_indexes(1) {} +handler *ha_maria::clone(MEM_ROOT *mem_root) +{ + ha_maria *new_handler= static_cast (handler::clone(mem_root)); + if (new_handler) + new_handler->file->state= file->state; + return new_handler; +} + + static const char *ha_maria_exts[]= { MARIA_NAME_IEXT, @@ -355,7 +366,11 @@ int ha_maria::write_row(byte * buf) or a new row, then update the auto_increment value in the record. */ if (table->next_number_field && buf == table->record[0]) - update_auto_increment(); + { + int error; + if ((error= update_auto_increment())) + return error; + } return maria_write(file, buf); } @@ -1818,20 +1833,28 @@ bool ha_maria::check_if_incompatible_data(HA_CREATE_INFO *info, return COMPATIBLE_DATA_YES; } -handlerton maria_hton; +extern int maria_panic(enum ha_panic_function flag); +int maria_panic(handlerton *hton, ha_panic_function flag) +{ + return maria_panic(flag); +} -static int ha_maria_init() +static int ha_maria_init(void *p) { - maria_hton.state=SHOW_OPTION_YES; - maria_hton.db_type=DB_TYPE_MARIA; - maria_hton.create=maria_create_handler; - maria_hton.panic=maria_panic; - maria_hton.flags=HTON_CAN_RECREATE; + handlerton *maria_hton; + + maria_hton= (handlerton *)p; + maria_hton->state= SHOW_OPTION_YES; + maria_hton->db_type= DB_TYPE_MARIA; + maria_hton->create= maria_create_handler; + maria_hton->panic= maria_panic; + /* TODO: decide if we support Maria being used for log tables */ + maria_hton->flags= HTON_CAN_RECREATE | HTON_SUPPORT_LOG_TABLES; return test(maria_init()); } struct st_mysql_storage_engine maria_storage_engine= -{ MYSQL_HANDLERTON_INTERFACE_VERSION, &maria_hton }; +{ MYSQL_HANDLERTON_INTERFACE_VERSION }; mysql_declare_plugin(maria) { @@ -1840,9 +1863,12 @@ mysql_declare_plugin(maria) "Maria", "MySQL AB", "Traditional transactional MySQL tables", + PLUGIN_LICENSE_GPL, ha_maria_init, /* Plugin Init */ NULL, /* Plugin Deinit */ 0x0100, /* 1.0 */ - 0 + NULL, /* status variables */ + NULL, /* system variables */ + NULL /* config options */ } mysql_declare_plugin_end; diff --git a/storage/maria/ha_maria.h b/storage/maria/ha_maria.h index cdd2305f6a1..e4edcc80982 100644 --- a/storage/maria/ha_maria.h +++ b/storage/maria/ha_maria.h @@ -42,9 +42,9 @@ class ha_maria :public handler int repair(THD * thd, HA_CHECK ¶m, bool optimize); public: - ha_maria(TABLE_SHARE * table_arg); - ~ha_maria() - {} + ha_maria(handlerton *hton, TABLE_SHARE * table_arg); + ~ha_maria() {} + handler *clone(MEM_ROOT *mem_root); const char *table_type() const { return "MARIA"; } const char *index_type(uint key_number); diff --git a/storage/maria/ma_check.c b/storage/maria/ma_check.c index 624df8a7881..03802ba6989 100644 --- a/storage/maria/ma_check.c +++ b/storage/maria/ma_check.c @@ -1371,8 +1371,9 @@ int maria_repair(HA_CHECK *param, register MARIA_HA *info, param->temp_filename); goto err; } - if (maria_filecopy(param,new_file,info->dfile,0L,new_header_length, - "datafile-header")) + if (new_header_length && + maria_filecopy(param,new_file,info->dfile,0L,new_header_length, + "datafile-header")) goto err; info->s->state.dellink= HA_OFFSET_ERROR; info->rec_cache.file=new_file; @@ -2056,8 +2057,9 @@ int maria_repair_by_sort(HA_CHECK *param, register MARIA_HA *info, param->temp_filename); goto err; } - if (maria_filecopy(param, new_file,info->dfile,0L,new_header_length, - "datafile-header")) + if (new_header_length && + maria_filecopy(param, new_file,info->dfile,0L,new_header_length, + "datafile-header")) goto err; if (param->testflag & T_UNPACK) { @@ -2450,8 +2452,9 @@ int maria_repair_parallel(HA_CHECK *param, register MARIA_HA *info, param->temp_filename); goto err; } - if (maria_filecopy(param, new_file,info->dfile,0L,new_header_length, - "datafile-header")) + if (new_header_length && + maria_filecopy(param, new_file,info->dfile,0L,new_header_length, + "datafile-header")) goto err; if (param->testflag & T_UNPACK) { diff --git a/storage/maria/ma_delete.c b/storage/maria/ma_delete.c index d1ad9edbed5..8849c89e30c 100644 --- a/storage/maria/ma_delete.c +++ b/storage/maria/ma_delete.c @@ -442,7 +442,7 @@ static int del(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, uchar *k else { DBUG_PRINT("test",("Inserting of key when deleting")); - if (_ma_get_last_key(info,keyinfo,leaf_buff,keybuff,endpos, + if (!_ma_get_last_key(info,keyinfo,leaf_buff,keybuff,endpos, &tmp)) goto err; ret_value= _ma_insert(info,keyinfo,key,leaf_buff,endpos,keybuff, diff --git a/storage/maria/ma_init.c b/storage/maria/ma_init.c index 318bbe341e4..476e8f694fc 100644 --- a/storage/maria/ma_init.c +++ b/storage/maria/ma_init.c @@ -19,7 +19,7 @@ #include "maria_def.h" #include -static int maria_inited= 0; +my_bool maria_inited= FALSE; pthread_mutex_t THR_LOCK_maria; /* @@ -40,7 +40,7 @@ int maria_init(void) { if (!maria_inited) { - maria_inited= 1; + maria_inited= TRUE; pthread_mutex_init(&THR_LOCK_maria,MY_MUTEX_INIT_SLOW); } return 0; @@ -51,7 +51,7 @@ void maria_end(void) { if (maria_inited) { - maria_inited= 0; + maria_inited= FALSE; ft_free_stopwords(); pthread_mutex_destroy(&THR_LOCK_maria); } diff --git a/storage/maria/ma_locking.c b/storage/maria/ma_locking.c index adb4b03bebe..5689d57f2a5 100644 --- a/storage/maria/ma_locking.c +++ b/storage/maria/ma_locking.c @@ -183,6 +183,21 @@ int maria_lock_database(MARIA_HA *info, int lock_type) break; /* Impossible */ } } +#ifdef __WIN__ + else + { + /* + Check for bad file descriptors if this table is part + of a merge union. Failing to capture this may cause + a crash on windows if the table is renamed and + later on referenced by the merge table. + */ + if( info->owned_by_merge && (info->s)->kfile < 0 ) + { + error = HA_ERR_NO_SUCH_TABLE; + } + } +#endif pthread_mutex_unlock(&share->intern_lock); DBUG_RETURN(error); } /* maria_lock_database */ diff --git a/storage/maria/ma_packrec.c b/storage/maria/ma_packrec.c index eb99e299f9a..3be893f39f8 100644 --- a/storage/maria/ma_packrec.c +++ b/storage/maria/ma_packrec.c @@ -231,11 +231,19 @@ my_bool _ma_read_pack_info(MARIA_HA *info, pbool fix_keys) { for (i=0 ; i < share->base.keys ; i++) { - share->keyinfo[i].keylength+=(uint16) diff_length; - share->keyinfo[i].minlength+=(uint16) diff_length; - share->keyinfo[i].maxlength+=(uint16) diff_length; - share->keyinfo[i].seg[share->keyinfo[i].keysegs].length= - (uint16) rec_reflength; + MARIA_KEYDEF *keyinfo= &share->keyinfo[i]; + keyinfo->keylength+= (uint16) diff_length; + keyinfo->minlength+= (uint16) diff_length; + keyinfo->maxlength+= (uint16) diff_length; + keyinfo->seg[keyinfo->flag & HA_FULLTEXT ? + FT_SEGS : keyinfo->keysegs].length= (uint16) rec_reflength; + } + if (share->ft2_keyinfo.seg) + { + MARIA_KEYDEF *ft2_keyinfo= &share->ft2_keyinfo; + ft2_keyinfo->keylength+= (uint16) diff_length; + ft2_keyinfo->minlength+= (uint16) diff_length; + ft2_keyinfo->maxlength+= (uint16) diff_length; } } diff --git a/storage/maria/ma_panic.c b/storage/maria/ma_panic.c index 90239b3943b..c1312cb1e77 100644 --- a/storage/maria/ma_panic.c +++ b/storage/maria/ma_panic.c @@ -44,6 +44,8 @@ int maria_panic(enum ha_panic_function flag) MARIA_HA *info; DBUG_ENTER("maria_panic"); + if (!maria_inited) + DBUG_RETURN(0); pthread_mutex_lock(&THR_LOCK_maria); for (list_element=maria_open_list ; list_element ; list_element=next_open) { diff --git a/storage/maria/ma_rkey.c b/storage/maria/ma_rkey.c index 2cb54a73b15..7338c96482f 100644 --- a/storage/maria/ma_rkey.c +++ b/storage/maria/ma_rkey.c @@ -93,29 +93,42 @@ int maria_rkey(MARIA_HA *info, byte *buf, int inx, const byte *key, uint key_len maria_read_vec[search_flag], info->s->state.key_root[inx])) { /* - If we are searching for an exact key (including the data pointer) - and this was added by an concurrent insert, - then the result is "key not found". + If we searching for a partial key (or using >, >=, < or <=) and + the data is outside of the data file, we need to continue searching + for the first key inside the data file */ - if ((search_flag == HA_READ_KEY_EXACT) && - (info->lastpos >= info->state->data_file_length)) + if (info->lastpos >= info->state->data_file_length && + (search_flag != HA_READ_KEY_EXACT || + last_used_keyseg != keyinfo->seg + keyinfo->keysegs)) { - my_errno= HA_ERR_KEY_NOT_FOUND; - info->lastpos= HA_OFFSET_ERROR; - } - else while (info->lastpos >= info->state->data_file_length) - { - /* - Skip rows that are inserted by other threads since we got a lock - Note that this can only happen if we are not searching after an - exact key, because the keys are sorted according to position - */ - - if (_ma_search_next(info, keyinfo, info->lastkey, - info->lastkey_length, - maria_readnext_vec[search_flag], - info->s->state.key_root[inx])) - break; + do + { + uint not_used[2]; + /* + Skip rows that are inserted by other threads since we got a lock + Note that this can only happen if we are not searching after an + full length exact key, because the keys are sorted + according to position + */ + if (_ma_search_next(info, keyinfo, info->lastkey, + info->lastkey_length, + maria_readnext_vec[search_flag], + info->s->state.key_root[inx])) + break; + /* + Check that the found key does still match the search. + _ma_search_next() delivers the next key regardless of its + value. + */ + if (search_flag == HA_READ_KEY_EXACT && + ha_key_cmp(keyinfo->seg, key_buff, info->lastkey, use_key_length, + SEARCH_FIND, not_used)) + { + my_errno= HA_ERR_KEY_NOT_FOUND; + info->lastpos= HA_OFFSET_ERROR; + break; + } + } while (info->lastpos >= info->state->data_file_length); } } } diff --git a/storage/maria/ma_write.c b/storage/maria/ma_write.c index c04a4a51eca..c56a26fefff 100644 --- a/storage/maria/ma_write.c +++ b/storage/maria/ma_write.c @@ -162,13 +162,13 @@ int maria_write(MARIA_HA *info, byte *record) /* Update status of the table. We need to do so after each row write for the log tables, as we want the new row to become visible to - other threads as soon as possible. We lock mutex here to follow - pthread memory visibility rules. + other threads as soon as possible. We don't lock mutex here + (as it is required by pthread memory visibility rules) as (1) it's + not critical to use outdated share->is_log_table value (2) locking + mutex here for every write is too expensive. */ - pthread_mutex_lock(&share->intern_lock); if (share->is_log_table) _ma_update_status((void*) info); - pthread_mutex_unlock(&share->intern_lock); allow_break(); /* Allow SIGHUP & SIGINT */ DBUG_RETURN(0); diff --git a/storage/maria/maria_def.h b/storage/maria/maria_def.h index e0ba4bdb406..506bdbc71ca 100644 --- a/storage/maria/maria_def.h +++ b/storage/maria/maria_def.h @@ -305,7 +305,14 @@ struct st_maria_info my_bool page_changed; /* If info->buff has to be reread for rnext */ my_bool buff_used; + /* + TODO: decide if we will have Maria-MERGE tables, and if no, + remove some members here. + */ my_bool once_flags; /* For MARIAMRG */ +#ifdef __WIN__ + my_bool owned_by_merge; /* This Maria table is part of a merge union */ +#endif #ifdef THREAD THR_LOCK_DATA lock; #endif @@ -431,6 +438,7 @@ extern LIST *maria_open_list; extern uchar NEAR maria_file_magic[], NEAR maria_pack_file_magic[]; extern uint NEAR maria_read_vec[], NEAR maria_readnext_vec[]; extern uint maria_quick_table_bits; +extern my_bool maria_inited; /* This is used by _ma_calc_xxx_key_length och _ma_store_key */ diff --git a/storage/maria/maria_ftdump.c b/storage/maria/maria_ftdump.c index b840072aed0..4e0b4bf5ce3 100644 --- a/storage/maria/maria_ftdump.c +++ b/storage/maria/maria_ftdump.c @@ -128,7 +128,6 @@ int main(int argc,char *argv[]) if (count || stats) { - doc_cnt++; if (strcmp(buf, buf2)) { if (*buf2) @@ -153,6 +152,7 @@ int main(int argc,char *argv[]) keylen2=keylen; doc_cnt=0; } + doc_cnt+= (subkeys >= 0 ? 1 : -subkeys); } if (dump) { @@ -168,7 +168,6 @@ int main(int argc,char *argv[]) if (count || stats) { - doc_cnt++; if (*buf2) { uniq++; -- cgit v1.2.1 From c2872bafde6d6ec2444c293f7a8aa397eb1dbb59 Mon Sep 17 00:00:00 2001 From: unknown Date: Fri, 13 Oct 2006 11:37:27 +0200 Subject: push for trnman review (lockmanager still fails unit tests) BitKeeper/deleted/.del-Makefile.am~4375ae3d4de2bdf0: Delete: unittest/maria/Makefile.am configure.in: silence up configure warnings, don't generate unittest/maria/Makefile include/atomic/nolock.h: s/LOCK/LOCK_prefix/ include/atomic/x86-gcc.h: s/LOCK/LOCK_prefix/ include/atomic/x86-msvc.h: s/LOCK/LOCK_prefix/ include/lf.h: pin asserts, renames include/my_atomic.h: move cleanup include/my_bit.h: s/uint/uint32/ mysys/lf_dynarray.c: style fixes, split for() in two, remove if()s mysys/lf_hash.c: renames, minor fixes mysys/my_atomic.c: run-time assert -> compile-time assert storage/maria/Makefile.am: lockman here storage/maria/unittest/Makefile.am: new unit tests storage/maria/unittest/trnman-t.c: lots of changes storage/maria/lockman.c: many changes: second meaning of "blocker" portability: s/gettimeofday/my_getsystime/ move mutex/cond out of LOCK_OWNER - it creates a race condition that will be fixed in a separate changeset increment lm->count for every element, not only for distinct ones - because we cannot decrease it for distinct elements only :( storage/maria/lockman.h: move mutex/cond out of LOCK_OWNER storage/maria/trnman.c: move mutex/cond out of LOCK_OWNER atomic-ops to access short_trid_to_trn[] storage/maria/trnman.h: move mutex/cond out of LOCK_OWNER storage/maria/unittest/lockman-t.c: unit stress test --- configure.in | 6 +- include/atomic/nolock.h | 4 +- include/atomic/x86-gcc.h | 18 +- include/atomic/x86-msvc.h | 10 +- include/lf.h | 14 +- include/my_atomic.h | 10 +- include/my_bit.h | 4 +- mysys/lf_dynarray.c | 92 ++--- mysys/lf_hash.c | 41 +-- mysys/my_atomic.c | 2 +- storage/maria/Makefile.am | 4 +- storage/maria/lockman.c | 681 +++++++++++++++++++++++++++++++++++++ storage/maria/lockman.h | 73 ++++ storage/maria/trnman.c | 332 ++++++++++++++++++ storage/maria/trnman.h | 48 +++ storage/maria/trxman.c | 258 -------------- storage/maria/trxman.h | 28 -- storage/maria/unittest/Makefile.am | 2 +- storage/maria/unittest/lockman-t.c | 246 ++++++++++++++ storage/maria/unittest/trnman-t.c | 177 ++++++++++ unittest/maria/Makefile.am | 12 - unittest/maria/trxman-t.c | 137 -------- 22 files changed, 1653 insertions(+), 546 deletions(-) create mode 100644 storage/maria/lockman.c create mode 100644 storage/maria/lockman.h create mode 100644 storage/maria/trnman.c create mode 100644 storage/maria/trnman.h delete mode 100644 storage/maria/trxman.c delete mode 100644 storage/maria/trxman.h create mode 100644 storage/maria/unittest/lockman-t.c create mode 100644 storage/maria/unittest/trnman-t.c delete mode 100644 unittest/maria/Makefile.am delete mode 100644 unittest/maria/trxman-t.c diff --git a/configure.in b/configure.in index 61ae233be06..7e997f1054c 100644 --- a/configure.in +++ b/configure.in @@ -2260,9 +2260,9 @@ AC_ARG_WITH(man, if test "$with_man" = "yes" then man_dirs="man" - man1_files=`ls -1 $srcdir/man/*.1 | sed -e 's;^.*man/;;'` + man1_files=`ls -1 $srcdir/man/*.1 2>/dev/null| sed -e 's;^.*man/;;'` man1_files=`echo $man1_files` - man8_files=`ls -1 $srcdir/man/*.8 | sed -e 's;^.*man/;;'` + man8_files=`ls -1 $srcdir/man/*.8 2>/dev/null| sed -e 's;^.*man/;;'` man8_files=`echo $man8_files` else man_dirs="" @@ -2481,7 +2481,7 @@ AC_SUBST(MAKE_BINARY_DISTRIBUTION_OPTIONS) # Output results AC_CONFIG_FILES(Makefile extra/Makefile mysys/Makefile dnl unittest/Makefile unittest/mytap/Makefile unittest/mytap/t/Makefile dnl - unittest/mysys/Makefile unittest/examples/Makefile unittest/maria/Makefile dnl + unittest/mysys/Makefile unittest/examples/Makefile dnl strings/Makefile regex/Makefile storage/Makefile dnl man/Makefile BUILD/Makefile vio/Makefile dnl libmysql/Makefile client/Makefile dnl diff --git a/include/atomic/nolock.h b/include/atomic/nolock.h index 21f41484ac9..a696e008f03 100644 --- a/include/atomic/nolock.h +++ b/include/atomic/nolock.h @@ -17,9 +17,9 @@ #if defined(__i386__) || defined(_M_IX86) || defined(__x86_64__) # ifdef MY_ATOMIC_MODE_DUMMY -# define LOCK "" +# define LOCK_prefix "" # else -# define LOCK "lock" +# define LOCK_prefix "lock" # endif # ifdef __GNUC__ diff --git a/include/atomic/x86-gcc.h b/include/atomic/x86-gcc.h index 3f0a82a9400..0be8fdf9244 100644 --- a/include/atomic/x86-gcc.h +++ b/include/atomic/x86-gcc.h @@ -22,15 +22,15 @@ #ifdef __x86_64__ # ifdef MY_ATOMIC_NO_XADD -# define MY_ATOMIC_MODE "gcc-amd64" LOCK "-no-xadd" +# define MY_ATOMIC_MODE "gcc-amd64" LOCK_prefix "-no-xadd" # else -# define MY_ATOMIC_MODE "gcc-amd64" LOCK +# define MY_ATOMIC_MODE "gcc-amd64" LOCK_prefix # endif #else # ifdef MY_ATOMIC_NO_XADD -# define MY_ATOMIC_MODE "gcc-x86" LOCK "-no-xadd" +# define MY_ATOMIC_MODE "gcc-x86" LOCK_prefix "-no-xadd" # else -# define MY_ATOMIC_MODE "gcc-x86" LOCK +# define MY_ATOMIC_MODE "gcc-x86" LOCK_prefix # endif #endif @@ -41,12 +41,12 @@ #ifndef MY_ATOMIC_NO_XADD #define make_atomic_add_body(S) \ - asm volatile (LOCK "; xadd %0, %1;" : "+r" (v) , "+m" (*a)) + asm volatile (LOCK_prefix "; xadd %0, %1;" : "+r" (v) , "+m" (*a)) #endif #define make_atomic_swap_body(S) \ - asm volatile ("; xchg %0, %1;" : "+r" (v) , "+m" (*a)) + asm volatile ("xchg %0, %1;" : "+r" (v) , "+m" (*a)) #define make_atomic_cas_body(S) \ - asm volatile (LOCK "; cmpxchg %3, %0; setz %2;" \ + asm volatile (LOCK_prefix "; cmpxchg %3, %0; setz %2;" \ : "+m" (*a), "+a" (*cmp), "=q" (ret): "r" (set)) #ifdef MY_ATOMIC_MODE_DUMMY @@ -55,11 +55,11 @@ #else /* Actually 32-bit reads/writes are always atomic on x86 - But we add LOCK here anyway to force memory barriers + But we add LOCK_prefix here anyway to force memory barriers */ #define make_atomic_load_body(S) \ ret=0; \ - asm volatile (LOCK "; cmpxchg %2, %0" \ + asm volatile (LOCK_prefix "; cmpxchg %2, %0" \ : "+m" (*a), "+a" (ret): "r" (ret)) #define make_atomic_store_body(S) \ asm volatile ("; xchg %0, %1;" : "+m" (*a), "+r" (v)) diff --git a/include/atomic/x86-msvc.h b/include/atomic/x86-msvc.h index d4024a854fb..8f3e55aaed7 100644 --- a/include/atomic/x86-msvc.h +++ b/include/atomic/x86-msvc.h @@ -26,19 +26,19 @@ #ifndef _atomic_h_cleanup_ #define _atomic_h_cleanup_ "atomic/x86-msvc.h" -#define MY_ATOMIC_MODE "msvc-x86" LOCK +#define MY_ATOMIC_MODE "msvc-x86" LOCK_prefix #define make_atomic_add_body(S) \ _asm { \ _asm mov reg_ ## S, v \ - _asm LOCK xadd *a, reg_ ## S \ + _asm LOCK_prefix xadd *a, reg_ ## S \ _asm movzx v, reg_ ## S \ } #define make_atomic_cas_body(S) \ _asm { \ _asm mov areg_ ## S, *cmp \ _asm mov reg2_ ## S, set \ - _asm LOCK cmpxchg *a, reg2_ ## S \ + _asm LOCK_prefix cmpxchg *a, reg2_ ## S \ _asm mov *cmp, areg_ ## S \ _asm setz al \ _asm movzx ret, al \ @@ -56,13 +56,13 @@ #else /* Actually 32-bit reads/writes are always atomic on x86 - But we add LOCK here anyway to force memory barriers + But we add LOCK_prefix here anyway to force memory barriers */ #define make_atomic_load_body(S) \ _asm { \ _asm mov areg_ ## S, 0 \ _asm mov reg2_ ## S, areg_ ## S \ - _asm LOCK cmpxchg *a, reg2_ ## S \ + _asm LOCK_prefix cmpxchg *a, reg2_ ## S \ _asm mov ret, areg_ ## S \ } #define make_atomic_store_body(S) \ diff --git a/include/lf.h b/include/lf.h index 6a5047f6052..4c6765b2d40 100644 --- a/include/lf.h +++ b/include/lf.h @@ -88,8 +88,8 @@ nolock_wrap(lf_dynarray_iterate, int, pin manager for memory allocator */ -#define LF_PINBOX_PINS 3 -#define LF_PURGATORY_SIZE 11 +#define LF_PINBOX_PINS 4 +#define LF_PURGATORY_SIZE 10 typedef void lf_pinbox_free_func(void *, void *); @@ -112,9 +112,9 @@ typedef struct { -sizeof(void *)*(LF_PINBOX_PINS+LF_PURGATORY_SIZE+1)]; } LF_PINS; -#define lf_lock_by_pins(PINS) \ +#define lf_rwlock_by_pins(PINS) \ my_atomic_rwlock_wrlock(&(PINS)->pinbox->pinstack.lock) -#define lf_unlock_by_pins(PINS) \ +#define lf_rwunlock_by_pins(PINS) \ my_atomic_rwlock_wrunlock(&(PINS)->pinbox->pinstack.lock) /* @@ -139,11 +139,13 @@ typedef struct { #define _lf_unpin(PINS, PIN) _lf_pin(PINS, PIN, NULL) #define lf_pin(PINS, PIN, ADDR) \ do { \ - lf_lock_by_pins(PINS); \ + lf_rwlock_by_pins(PINS); \ _lf_pin(PINS, PIN, ADDR); \ - lf_unlock_by_pins(PINS); \ + lf_rwunlock_by_pins(PINS); \ } while (0) #define lf_unpin(PINS, PIN) lf_pin(PINS, PIN, NULL) +#define _lf_assert_pin(PINS, PIN) assert((PINS)->pin[PIN] != 0) +#define _lf_assert_unpin(PINS, PIN) assert((PINS)->pin[PIN]==0) void lf_pinbox_init(LF_PINBOX *pinbox, lf_pinbox_free_func *free_func, void * free_func_arg); diff --git a/include/my_atomic.h b/include/my_atomic.h index d3e4e0055d3..921b55e68a2 100644 --- a/include/my_atomic.h +++ b/include/my_atomic.h @@ -118,6 +118,11 @@ make_atomic_swap(16) make_atomic_swap(32) make_atomic_swap(ptr) +#ifdef _atomic_h_cleanup_ +#include _atomic_h_cleanup_ +#undef _atomic_h_cleanup_ +#endif + #undef make_atomic_add #undef make_atomic_cas #undef make_atomic_load @@ -130,11 +135,6 @@ make_atomic_swap(ptr) #undef make_atomic_swap_body #undef intptr -#ifdef _atomic_h_cleanup_ -#include _atomic_h_cleanup_ -#undef _atomic_h_cleanup_ -#endif - #ifndef LF_BACKOFF #define LF_BACKOFF (1) #endif diff --git a/include/my_bit.h b/include/my_bit.h index 71bbe2d4ba3..58e8bb39683 100644 --- a/include/my_bit.h +++ b/include/my_bit.h @@ -88,7 +88,7 @@ STATIC_INLINE uint32 my_clear_highest_bit(uint32 v) return v & w; } -STATIC_INLINE uint32 my_reverse_bits(uint key) +STATIC_INLINE uint32 my_reverse_bits(uint32 key) { return (_my_bits_reverse_table[ key & 255] << 24) | @@ -101,7 +101,7 @@ STATIC_INLINE uint32 my_reverse_bits(uint key) extern uint my_bit_log2(ulong value); extern uint32 my_round_up_to_next_power(uint32 v); uint32 my_clear_highest_bit(uint32 v); -uint32 my_reverse_bits(uint key); +uint32 my_reverse_bits(uint32 key); extern uint my_count_bits(ulonglong v); extern uint my_count_bits_ushort(ushort v); #endif diff --git a/mysys/lf_dynarray.c b/mysys/lf_dynarray.c index 0fa04ab095c..ade1c28d51c 100644 --- a/mysys/lf_dynarray.c +++ b/mysys/lf_dynarray.c @@ -38,7 +38,7 @@ void lf_dynarray_init(LF_DYNARRAY *array, uint element_size) { bzero(array, sizeof(*array)); - array->size_of_element=element_size; + array->size_of_element= element_size; my_atomic_rwlock_init(&array->lock); } @@ -49,7 +49,7 @@ static void recursive_free(void **alloc, int level) if (level) { int i; - for (i=0; i < LF_DYNARRAY_LEVEL_LENGTH; i++) + for (i= 0; i < LF_DYNARRAY_LEVEL_LENGTH; i++) recursive_free(alloc[i], level-1); my_free((void *)alloc, MYF(0)); } @@ -60,13 +60,13 @@ static void recursive_free(void **alloc, int level) void lf_dynarray_destroy(LF_DYNARRAY *array) { int i; - for (i=0; i < LF_DYNARRAY_LEVELS; i++) + for (i= 0; i < LF_DYNARRAY_LEVELS; i++) recursive_free(array->level[i], i); my_atomic_rwlock_destroy(&array->lock); bzero(array, sizeof(*array)); } -static const int dynarray_idxes_in_level[LF_DYNARRAY_LEVELS]= +static const int dynarray_idxes_in_prev_level[LF_DYNARRAY_LEVELS]= { 0, /* +1 here to to avoid -1's below */ LF_DYNARRAY_LEVEL_LENGTH, @@ -77,41 +77,32 @@ static const int dynarray_idxes_in_level[LF_DYNARRAY_LEVELS]= void *_lf_dynarray_lvalue(LF_DYNARRAY *array, uint idx) { - void * ptr, * volatile * ptr_ptr=0; + void * ptr, * volatile * ptr_ptr= 0; int i; - for (i=3; i > 0; i--) + for (i= 3; idx < dynarray_idxes_in_prev_level[i]; i--) /* no-op */; + ptr_ptr= &array->level[i]; + idx-= dynarray_idxes_in_prev_level[i]; + for (; i > 0; i--) { - if (ptr_ptr || idx >= dynarray_idxes_in_level[i]) + if (!(ptr= *ptr_ptr)) { - if (!ptr_ptr) - { - ptr_ptr=&array->level[i]; - idx-= dynarray_idxes_in_level[i]; - } - ptr=*ptr_ptr; - if (!ptr) - { - void *alloc=my_malloc(LF_DYNARRAY_LEVEL_LENGTH * sizeof(void *), - MYF(MY_WME|MY_ZEROFILL)); - if (!alloc) - return(NULL); - if (my_atomic_casptr(ptr_ptr, &ptr, alloc)) - ptr= alloc; - else - my_free(alloc, MYF(0)); - } - ptr_ptr=((void **)ptr) + idx / dynarray_idxes_in_level[i]; - idx%= dynarray_idxes_in_level[i]; + void *alloc= my_malloc(LF_DYNARRAY_LEVEL_LENGTH * sizeof(void *), + MYF(MY_WME|MY_ZEROFILL)); + if (!alloc) + return(NULL); + if (my_atomic_casptr(ptr_ptr, &ptr, alloc)) + ptr= alloc; + else + my_free(alloc, MYF(0)); } + ptr_ptr= ((void **)ptr) + idx / dynarray_idxes_in_prev_level[i]; + idx%= dynarray_idxes_in_prev_level[i]; } - if (!ptr_ptr) - ptr_ptr=&array->level[0]; - ptr=*ptr_ptr; - if (!ptr) + if (!(ptr= *ptr_ptr)) { void *alloc, *data; - alloc=my_malloc(LF_DYNARRAY_LEVEL_LENGTH * array->size_of_element + + alloc= my_malloc(LF_DYNARRAY_LEVEL_LENGTH * array->size_of_element + max(array->size_of_element, sizeof(void *)), MYF(MY_WME|MY_ZEROFILL)); if (!alloc) @@ -123,7 +114,7 @@ void *_lf_dynarray_lvalue(LF_DYNARRAY *array, uint idx) if (mod) data+= array->size_of_element - mod; } - ((void **)data)[-1]=alloc; /* free() will need the original pointer */ + ((void **)data)[-1]= alloc; /* free() will need the original pointer */ if (my_atomic_casptr(ptr_ptr, &ptr, data)) ptr= data; else @@ -134,29 +125,20 @@ void *_lf_dynarray_lvalue(LF_DYNARRAY *array, uint idx) void *_lf_dynarray_value(LF_DYNARRAY *array, uint idx) { - void * ptr, * volatile * ptr_ptr=0; + void * ptr, * volatile * ptr_ptr= 0; int i; - for (i=3; i > 0; i--) + for (i= 3; idx < dynarray_idxes_in_prev_level[i]; i--) /* no-op */; + ptr_ptr= &array->level[i]; + idx-= dynarray_idxes_in_prev_level[i]; + for (; i > 0; i--) { - if (ptr_ptr || idx >= dynarray_idxes_in_level[i]) - { - if (!ptr_ptr) - { - ptr_ptr=&array->level[i]; - idx-= dynarray_idxes_in_level[i]; - } - ptr=*ptr_ptr; - if (!ptr) - return(NULL); - ptr_ptr=((void **)ptr) + idx / dynarray_idxes_in_level[i]; - idx %= dynarray_idxes_in_level[i]; - } + if (!(ptr= *ptr_ptr)) + return(NULL); + ptr_ptr= ((void **)ptr) + idx / dynarray_idxes_in_prev_level[i]; + idx %= dynarray_idxes_in_prev_level[i]; } - if (!ptr_ptr) - ptr_ptr=&array->level[0]; - ptr=*ptr_ptr; - if (!ptr) + if (!(ptr= *ptr_ptr)) return(NULL); return ptr + array->size_of_element * idx; } @@ -169,8 +151,8 @@ static int recursive_iterate(LF_DYNARRAY *array, void *ptr, int level, return 0; if (!level) return func(ptr, arg); - for (i=0; i < LF_DYNARRAY_LEVEL_LENGTH; i++) - if ((res=recursive_iterate(array, ((void **)ptr)[i], level-1, func, arg))) + for (i= 0; i < LF_DYNARRAY_LEVEL_LENGTH; i++) + if ((res= recursive_iterate(array, ((void **)ptr)[i], level-1, func, arg))) return res; return 0; } @@ -178,8 +160,8 @@ static int recursive_iterate(LF_DYNARRAY *array, void *ptr, int level, int _lf_dynarray_iterate(LF_DYNARRAY *array, lf_dynarray_func func, void *arg) { int i, res; - for (i=0; i < LF_DYNARRAY_LEVELS; i++) - if ((res=recursive_iterate(array, array->level[i], i, func, arg))) + for (i= 0; i < LF_DYNARRAY_LEVELS; i++) + if ((res= recursive_iterate(array, array->level[i], i, func, arg))) return res; return 0; } diff --git a/mysys/lf_hash.c b/mysys/lf_hash.c index 736c3ea4887..a0425e89556 100644 --- a/mysys/lf_hash.c +++ b/mysys/lf_hash.c @@ -19,6 +19,7 @@ TODO try to get rid of dummy nodes ? + for non-unique hash, count only _distinct_ values */ #include #include @@ -51,7 +52,7 @@ typedef struct { cursor is positioned in either case pins[0..2] are used, they are NOT removed on return */ -static int lfind(intptr volatile *head, uint32 hashnr, +static int lfind(LF_SLIST * volatile *head, uint32 hashnr, const uchar *key, uint keylen, CURSOR *cursor, LF_PINS *pins) { uint32 cur_hashnr; @@ -60,7 +61,7 @@ static int lfind(intptr volatile *head, uint32 hashnr, intptr link; retry: - cursor->prev=head; + cursor->prev=(intptr *)head; do { cursor->curr=PTR(*cursor->prev); _lf_pin(pins,1,cursor->curr); @@ -112,7 +113,7 @@ retry: /* RETURN 0 - inserted - not 0 - a pointer to a conflict + not 0 - a pointer to a conflict (not pinned and thus unusable) NOTE it uses pins[0..2], on return all pins are removed. @@ -125,17 +126,17 @@ static LF_SLIST *linsert(LF_SLIST * volatile *head, LF_SLIST *node, do { - if (lfind((intptr*)head, node->hashnr, node->key, node->keylen, + if (lfind(head, node->hashnr, node->key, node->keylen, &cursor, pins) && (flags & LF_HASH_UNIQUE)) - res=0; + res=0; /* duplicate found */ else { node->link=(intptr)cursor.curr; assert(node->link != (intptr)node); assert(cursor.prev != &node->link); if (my_atomic_casptr((void **)cursor.prev, (void **)&cursor.curr, node)) - res=1; + res=1; /* inserted ok */ } } while (res == -1); _lf_unpin(pins, 0); @@ -159,7 +160,7 @@ static int ldelete(LF_SLIST * volatile *head, uint32 hashnr, do { - if (!lfind((intptr *)head, hashnr, key, keylen, &cursor, pins)) + if (!lfind(head, hashnr, key, keylen, &cursor, pins)) res= 1; else if (my_atomic_casptr((void **)&(cursor.curr->link), @@ -169,7 +170,7 @@ static int ldelete(LF_SLIST * volatile *head, uint32 hashnr, (void **)&cursor.curr, cursor.next)) _lf_alloc_free(pins, cursor.curr); else - lfind((intptr *)head, hashnr, key, keylen, &cursor, pins); + lfind(head, hashnr, key, keylen, &cursor, pins); res= 0; } } while (res == -1); @@ -191,7 +192,7 @@ static LF_SLIST *lsearch(LF_SLIST * volatile *head, uint32 hashnr, const uchar *key, uint keylen, LF_PINS *pins) { CURSOR cursor; - int res=lfind((intptr *)head, hashnr, key, keylen, &cursor, pins); + int res=lfind(head, hashnr, key, keylen, &cursor, pins); if (res) _lf_pin(pins, 2, cursor.curr); _lf_unpin(pins, 0); _lf_unpin(pins, 1); @@ -214,7 +215,7 @@ static inline uint calc_hash(LF_HASH *hash, const uchar *key, uint keylen) return nr1 & INT_MAX32; } -#define MAX_LOAD 1 +#define MAX_LOAD 1.0 static void initialize_bucket(LF_HASH *, LF_SLIST * volatile*, uint, LF_PINS *); void lf_hash_init(LF_HASH *hash, uint element_size, uint flags, @@ -262,7 +263,7 @@ int lf_hash_insert(LF_HASH *hash, LF_PINS *pins, const void *data) uint csize, bucket, hashnr; LF_SLIST *node, * volatile *el; - lf_lock_by_pins(pins); + lf_rwlock_by_pins(pins); node=(LF_SLIST *)_lf_alloc_new(pins); memcpy(node+1, data, hash->element_size); node->key= hash_key(hash, (uchar *)(node+1), &node->keylen); @@ -275,13 +276,13 @@ int lf_hash_insert(LF_HASH *hash, LF_PINS *pins, const void *data) if (linsert(el, node, pins, hash->flags)) { _lf_alloc_free(pins, node); - lf_unlock_by_pins(pins); + lf_rwunlock_by_pins(pins); return 1; } csize= hash->size; if ((my_atomic_add32(&hash->count, 1)+1.0) / csize > MAX_LOAD) my_atomic_cas32(&hash->size, &csize, csize*2); - lf_unlock_by_pins(pins); + lf_rwunlock_by_pins(pins); return 0; } @@ -298,17 +299,17 @@ int lf_hash_delete(LF_HASH *hash, LF_PINS *pins, const void *key, uint keylen) uint bucket, hashnr=calc_hash(hash, (uchar *)key, keylen); bucket= hashnr % hash->size; - lf_lock_by_pins(pins); + lf_rwlock_by_pins(pins); el=_lf_dynarray_lvalue(&hash->array, bucket); if (*el == NULL) initialize_bucket(hash, el, bucket, pins); if (ldelete(el, my_reverse_bits(hashnr) | 1, (uchar *)key, keylen, pins)) { - lf_unlock_by_pins(pins); + lf_rwunlock_by_pins(pins); return 1; } my_atomic_add32(&hash->count, -1); - lf_unlock_by_pins(pins); + lf_rwunlock_by_pins(pins); return 0; } @@ -322,13 +323,13 @@ void *lf_hash_search(LF_HASH *hash, LF_PINS *pins, const void *key, uint keylen) uint bucket, hashnr=calc_hash(hash, (uchar *)key, keylen); bucket= hashnr % hash->size; - lf_lock_by_pins(pins); + lf_rwlock_by_pins(pins); el=_lf_dynarray_lvalue(&hash->array, bucket); if (*el == NULL) initialize_bucket(hash, el, bucket, pins); found= lsearch(el, my_reverse_bits(hashnr) | 1, (uchar *)key, keylen, pins); - lf_unlock_by_pins(pins); - return found+1; + lf_rwunlock_by_pins(pins); + return found ? found+1 : 0; } static char *dummy_key=""; @@ -347,7 +348,7 @@ static void initialize_bucket(LF_HASH *hash, LF_SLIST * volatile *node, dummy->keylen=0; if ((cur= linsert(el, dummy, pins, 0))) { - _lf_alloc_free(pins, dummy); + my_free((void *)dummy, MYF(0)); dummy= cur; } my_atomic_casptr((void **)node, (void **)&tmp, dummy); diff --git a/mysys/my_atomic.c b/mysys/my_atomic.c index 2908a44961a..fbeb3d63bef 100644 --- a/mysys/my_atomic.c +++ b/mysys/my_atomic.c @@ -35,7 +35,7 @@ */ int my_atomic_initialize() { - DBUG_ASSERT(sizeof(intptr) == sizeof(void *)); + char assert_the_size[sizeof(intptr) == sizeof(void *) ? 1 : -1]; /* currently the only thing worth checking is SMP/UP issue */ #ifdef MY_ATOMIC_MODE_DUMMY return my_getncpus() == 1 ? MY_ATOMIC_OK : MY_ATOMIC_NOT_1CPU; diff --git a/storage/maria/Makefile.am b/storage/maria/Makefile.am index 54fd70d7ae5..4f348cd2894 100644 --- a/storage/maria/Makefile.am +++ b/storage/maria/Makefile.am @@ -53,7 +53,7 @@ maria_pack_LDADD= @CLIENT_EXTRA_LDFLAGS@ libmaria.a \ noinst_PROGRAMS = ma_test1 ma_test2 ma_test3 ma_rt_test ma_sp_test noinst_HEADERS = maria_def.h ma_rt_index.h ma_rt_key.h ma_rt_mbr.h \ ma_sp_defs.h ma_fulltext.h ma_ftdefs.h ma_ft_test1.h \ - ma_ft_eval.h trxman.h \ + ma_ft_eval.h trnman.h lockman.h \ ma_control_file.h ha_maria.h ma_test1_DEPENDENCIES= $(LIBRARIES) ma_test1_LDADD= @CLIENT_EXTRA_LDFLAGS@ libmaria.a \ @@ -108,7 +108,7 @@ libmaria_a_SOURCES = ma_init.c ma_open.c ma_extra.c ma_info.c ma_rkey.c \ ma_keycache.c ma_preload.c ma_ft_parser.c \ ma_ft_update.c ma_ft_boolean_search.c \ ma_ft_nlq_search.c ft_maria.c ma_sort.c \ - ha_maria.cc trxman.c \ + ha_maria.cc trnman.c lockman.c \ ma_rt_index.c ma_rt_key.c ma_rt_mbr.c ma_rt_split.c \ ma_sp_key.c ma_control_file.c CLEANFILES = test?.MA? FT?.MA? isam.log ma_test_all ma_rt_test.MA? sp_test.MA? diff --git a/storage/maria/lockman.c b/storage/maria/lockman.c new file mode 100644 index 00000000000..e8ddbd1a25a --- /dev/null +++ b/storage/maria/lockman.c @@ -0,0 +1,681 @@ +// TODO lock escalation, instant duration locks +// automatically place S instead of LS if possible +/* + TODO optimization: table locks - they have completely + different characteristics. long lists, few distinct resources - + slow to scan, [possibly] high retry rate +*/ +/* Copyright (C) 2000 MySQL AB + + 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; either version 2 of the License, or + (at your option) any later version. + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ + +#include +#include +#include +#include +#include "lockman.h" + +/* + Lock compatibility matrix. + + It's asymmetric. Read it as "Somebody has the lock , can I set the lock ?" + + ') Though you can take LS lock while somebody has S lock, it makes no + sense - it's simpler to take S lock too. + + ") Strictly speaking you can take LX lock while somebody has S lock. + But in this case you lock no rows, because all rows are locked by this + somebody. So we prefer to define that LX cannot be taken when S + exists. Same about LX and X. + + 1 - compatible + 0 - incompatible + -1 - "impossible", so that we can assert the impossibility. +*/ +static int lock_compatibility_matrix[10][10]= +{ /* N S X IS IX SIX LS LX SLX LSIX */ + { -1, 1, 1, 1, 1, 1, 1, 1, 1, 1 }, /* N */ + { -1, 1, 0, 1, 0, 0, 1, 0, 0, 0 }, /* S */ + { -1, 0, 0, 0, 0, 0, 0, 0, 0, 0 }, /* X */ + { -1, 1, 0, 1, 1, 1, 1, 1, 1, 1 }, /* IS */ + { -1, 0, 0, 1, 1, 0, 1, 1, 0, 1 }, /* IX */ + { -1, 0, 0, 1, 0, 0, 1, 0, 0, 0 }, /* SIX */ + { -1, 1, 0, 1, 0, 0, 1, 0, 0, 0 }, /* LS */ + { -1, 0, 0, 0, 0, 0, 0, 0, 0, 0 }, /* LX */ + { -1, 0, 0, 0, 0, 0, 0, 0, 0, 0 }, /* SLX */ + { -1, 0, 0, 1, 0, 0, 1, 0, 0, 0 } /* LSIX */ +}; + +/* + Lock combining matrix. + + It's symmetric. Read it as "what lock level L is identical to the + set of two locks A and B" + + One should never get N from it, we assert the impossibility +*/ +static enum lock_type lock_combining_matrix[10][10]= +{/* N S X IS IX SIX LS LX SLX LSIX */ + { N, S, X, IS, IX, SIX, S, SLX, SLX, SIX}, /* N */ + { S, S, X, S, SIX, SIX, S, SLX, SLX, SIX}, /* S */ + { X, X, X, X, X, X, X, X, X, X}, /* X */ + { IS, S, X, IS, IX, SIX, LS, LX, SLX, LSIX}, /* IS */ + { IX, SIX, X, IX, IX, SIX, LSIX, LX, SLX, LSIX}, /* IX */ + { SIX, SIX, X, SIX, SIX, SIX, SIX, SLX, SLX, SIX}, /* SIX */ + { LS, S, X, LS, LSIX, SIX, LS, LX, SLX, LSIX}, /* LS */ + { LX, SLX, X, LX, LX, SLX, LX, LX, SLX, LX}, /* LX */ + { SLX, SLX, X, SLX, SLX, SLX, SLX, SLX, SLX, SLX}, /* SLX */ + { LSIX, SIX, X, LSIX, LSIX, SIX, LSIX, LX, SLX, LSIX} /* LSIX */ +}; + +#define REPEAT_ONCE_MORE 0 +#define OK_TO_PLACE_THE_LOCK 1 +#define OK_TO_PLACE_THE_REQUEST 2 +#define ALREADY_HAVE_THE_LOCK 4 +#define ALREADY_HAVE_THE_REQUEST 8 +#define PLACE_NEW_DISABLE_OLD 16 +#define REQUEST_NEW_DISABLE_OLD 32 +#define RESOURCE_WAS_UNLOCKED 64 + +#define NEED_TO_WAIT (OK_TO_PLACE_THE_REQUEST | ALREADY_HAVE_THE_REQUEST |\ + REQUEST_NEW_DISABLE_OLD) +#define ALREADY_HAVE (ALREADY_HAVE_THE_LOCK | ALREADY_HAVE_THE_REQUEST) +#define LOCK_UPGRADE (PLACE_NEW_DISABLE_OLD | REQUEST_NEW_DISABLE_OLD) + + +/* + the return codes for lockman_getlock + + It's asymmetric. Read it as "I have the lock , + what value should be returned for ?" + + 0 means impossible combination (assert!) + + Defines below help to preserve the table structure. + I/L/A values are self explanatory + x means the combination is possible (assert should not crash) + but cannot happen in row locks, only in table locks (S,X), or + lock escalations (LS,LX) +*/ +#define I GOT_THE_LOCK_NEED_TO_LOCK_A_SUBRESOURCE +#define L GOT_THE_LOCK_NEED_TO_INSTANT_LOCK_A_SUBRESOURCE +#define A GOT_THE_LOCK +#define x GOT_THE_LOCK +static enum lockman_getlock_result getlock_result[10][10]= +{/* N S X IS IX SIX LS LX SLX LSIX */ + { 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}, /* N */ + { 0, x, 0, A, 0, 0, x, 0, 0, 0}, /* S */ + { 0, x, x, A, A, 0, x, x, 0, 0}, /* X */ + { 0, 0, 0, I, 0, 0, 0, 0, 0, 0}, /* IS */ + { 0, 0, 0, I, I, 0, 0, 0, 0, 0}, /* IX */ + { 0, x, 0, A, I, 0, x, 0, 0, 0}, /* SIX */ + { 0, 0, 0, L, 0, 0, x, 0, 0, 0}, /* LS */ + { 0, 0, 0, L, L, 0, x, x, 0, 0}, /* LX */ + { 0, x, 0, A, L, 0, x, x, 0, 0}, /* SLX */ + { 0, 0, 0, L, I, 0, x, 0, 0, 0} /* LSIX */ +}; +#undef I +#undef L +#undef A +#undef x + +LF_REQUIRE_PINS(4); + +typedef struct lockman_lock { + uint64 resource; + struct lockman_lock *lonext; + intptr volatile link; + uint32 hashnr; +//#warning TODO - remove hashnr from LOCK + uint16 loid; + uchar lock; /* sizeof(uchar) <= sizeof(enum) */ + uchar flags; +} LOCK; + +#define IGNORE_ME 1 +#define UPGRADED 2 + +typedef struct { + intptr volatile *prev; + LOCK *curr, *next; + LOCK *blocker, *upgrade_from; +} CURSOR; + +#define PTR(V) (LOCK *)((V) & (~(intptr)1)) +#define DELETED(V) ((V) & 1) + +/* + NOTE + cursor is positioned in either case + pins[0..3] are used, they are NOT removed on return +*/ +static int lockfind(LOCK * volatile *head, LOCK *node, + CURSOR *cursor, LF_PINS *pins) +{ + uint32 hashnr, cur_hashnr; + uint64 resource, cur_resource; + intptr link; + my_bool cur_active, compatible, upgrading, prev_active; + enum lock_type lock, prev_lock, cur_lock; + uint16 loid, cur_loid; + int upgraded_pairs, cur_flags, flags; + + hashnr= node->hashnr; + resource= node->resource; + lock= node->lock; + loid= node->loid; + flags= node->flags; + +retry: + cursor->prev= (intptr *)head; + prev_lock= N; + cur_active= TRUE; + compatible= TRUE; + upgrading= FALSE; + cursor->blocker= cursor->upgrade_from= 0; + _lf_unpin(pins, 3); + upgraded_pairs= 0; + do { + cursor->curr= PTR(*cursor->prev); + _lf_pin(pins,1,cursor->curr); + } while(*cursor->prev != (intptr)cursor->curr && LF_BACKOFF); + for (;;) + { + if (!cursor->curr) + break; + do { + link= cursor->curr->link; + cursor->next= PTR(link); + _lf_pin(pins, 0, cursor->next); + } while(link != cursor->curr->link && LF_BACKOFF); + cur_hashnr= cursor->curr->hashnr; + cur_resource= cursor->curr->resource; + cur_lock= cursor->curr->lock; + cur_loid= cursor->curr->loid; + cur_flags= cursor->curr->flags; + if (*cursor->prev != (intptr)cursor->curr) + { + LF_BACKOFF; + goto retry; + } + if (!DELETED(link)) + { + if (cur_hashnr > hashnr || + (cur_hashnr == hashnr && cur_resource >= resource)) + { + if (cur_hashnr > hashnr || cur_resource > resource) + { + if (upgraded_pairs != 0) + goto retry; + break; + } + /* ok, we have a lock for this resource */ + DBUG_ASSERT(lock_compatibility_matrix[prev_lock][cur_lock] >= 0); + DBUG_ASSERT(lock_compatibility_matrix[cur_lock][lock] >= 0); + if (cur_flags & UPGRADED) + upgraded_pairs++; + if ((cur_flags & IGNORE_ME) && ! (flags & IGNORE_ME)) + { + DBUG_ASSERT(cur_active); + upgraded_pairs--; + if (cur_loid == loid) + cursor->upgrade_from= cursor->curr; + } + else + { + prev_active= cur_active; + cur_active&= lock_compatibility_matrix[prev_lock][cur_lock]; + if (upgrading && !cur_active && upgraded_pairs == 0) + break; + if (prev_active && !cur_active) + { + cursor->blocker= cursor->curr; + _lf_pin(pins, 3, cursor->curr); + } + if (cur_loid == loid) + { + /* we already have a lock on this resource */ + DBUG_ASSERT(lock_combining_matrix[cur_lock][lock] != N); + DBUG_ASSERT(!upgrading); /* can happen only once */ + if (lock_combining_matrix[cur_lock][lock] == cur_lock) + { + /* new lock is compatible */ + return cur_active ? ALREADY_HAVE_THE_LOCK + : ALREADY_HAVE_THE_REQUEST; + } + /* not compatible, upgrading */ + upgrading= TRUE; + cursor->upgrade_from= cursor->curr; + } + else + { + if (!lock_compatibility_matrix[cur_lock][lock]) + { + compatible= FALSE; + cursor->blocker= cursor->curr; + _lf_pin(pins, 3, cursor->curr); + } + prev_lock= lock_combining_matrix[prev_lock][cur_lock]; + DBUG_ASSERT(prev_lock != N); + } + } + } + cursor->prev= &(cursor->curr->link); + _lf_pin(pins, 2, cursor->curr); + } + else + { + if (my_atomic_casptr((void **)cursor->prev, + (void **)&cursor->curr, cursor->next)) + _lf_alloc_free(pins, cursor->curr); + else + { + LF_BACKOFF; + goto retry; + } + } + cursor->curr= cursor->next; + _lf_pin(pins, 1, cursor->curr); + } + /* + either the end of lock list - no more locks for this resource, + or upgrading and the end of active lock list + */ + if (upgrading) + { + if (compatible) + return PLACE_NEW_DISABLE_OLD; + else + return REQUEST_NEW_DISABLE_OLD; + } + if (cur_active && compatible) + { + /* + either no locks for this resource or all are compatible. + ok to place the lock in any case. + */ + return prev_lock == N ? RESOURCE_WAS_UNLOCKED + : OK_TO_PLACE_THE_LOCK; + } + /* we have a lock conflict. ok to place a lock request. And wait */ + return OK_TO_PLACE_THE_REQUEST; +} + +/* + NOTE + it uses pins[0..3], on return pins 0..2 are removed, pin 3 (blocker) stays +*/ +static int lockinsert(LOCK * volatile *head, LOCK *node, LF_PINS *pins, + LOCK **blocker) +{ + CURSOR cursor; + int res; + + do + { + res= lockfind(head, node, &cursor, pins); + DBUG_ASSERT(res != ALREADY_HAVE_THE_REQUEST); + if (!(res & ALREADY_HAVE)) + { + if (res & LOCK_UPGRADE) + { + node->flags|= UPGRADED; + node->lock= lock_combining_matrix[cursor.upgrade_from->lock][node->lock]; + } + node->link= (intptr)cursor.curr; + DBUG_ASSERT(node->link != (intptr)node); + DBUG_ASSERT(cursor.prev != &node->link); + if (!my_atomic_casptr((void **)cursor.prev, (void **)&cursor.curr, node)) + res= REPEAT_ONCE_MORE; + if (res & LOCK_UPGRADE) + cursor.upgrade_from->flags|= IGNORE_ME; + } + + } while (res == REPEAT_ONCE_MORE); + _lf_unpin(pins, 0); + _lf_unpin(pins, 1); + _lf_unpin(pins, 2); + /* + note that cursor.curr is NOT pinned on return. + this is ok as it's either a dummy node for initialize_bucket + and dummy nodes don't need pinning, + or it's a lock of the same transaction for lockman_getlock, + and it cannot be removed by another thread + */ + *blocker= cursor.blocker ? cursor.blocker : cursor.curr; + return res; +} + +/* + NOTE + it uses pins[0..3], on return pins 0..2 are removed, pin 3 (blocker) stays +*/ +static int lockpeek(LOCK * volatile *head, LOCK *node, LF_PINS *pins, + LOCK **blocker) +{ + CURSOR cursor; + int res; + + res= lockfind(head, node, &cursor, pins); + + _lf_unpin(pins, 0); + _lf_unpin(pins, 1); + _lf_unpin(pins, 2); + if (blocker) + *blocker= cursor.blocker; + return res; +} + +/* + NOTE + it uses pins[0..3], on return all pins are removed. + + One _must_ have the lock (or request) to call this +*/ +static int lockdelete(LOCK * volatile *head, LOCK *node, LF_PINS *pins) +{ + CURSOR cursor; + int res; + + do + { + res= lockfind(head, node, &cursor, pins); + DBUG_ASSERT(res & ALREADY_HAVE); + + if (cursor.upgrade_from) + cursor.upgrade_from->flags&= ~IGNORE_ME; + + if (my_atomic_casptr((void **)&(cursor.curr->link), + (void **)&cursor.next, 1+(char *)cursor.next)) + { + if (my_atomic_casptr((void **)cursor.prev, + (void **)&cursor.curr, cursor.next)) + _lf_alloc_free(pins, cursor.curr); + else + lockfind(head, node, &cursor, pins); + } + else + res= REPEAT_ONCE_MORE; + } while (res == REPEAT_ONCE_MORE); + _lf_unpin(pins, 0); + _lf_unpin(pins, 1); + _lf_unpin(pins, 2); + _lf_unpin(pins, 3); + return res; +} + +void lockman_init(LOCKMAN *lm, loid_to_lo_func *func, uint timeout) +{ + lf_alloc_init(&lm->alloc,sizeof(LOCK)); + lf_dynarray_init(&lm->array, sizeof(LOCK **)); + lm->size= 1; + lm->count= 0; + lm->loid_to_lo= func; + lm->lock_timeout= timeout; +} + +void lockman_destroy(LOCKMAN *lm) +{ + LOCK *el= *(LOCK **)_lf_dynarray_lvalue(&lm->array, 0); + while (el) + { + intptr next= el->link; + if (el->hashnr & 1) + lf_alloc_real_free(&lm->alloc, el); + else + my_free((void *)el, MYF(0)); + el= (LOCK *)next; + } + lf_alloc_destroy(&lm->alloc); + lf_dynarray_destroy(&lm->array); +} + +/* TODO: optimize it */ +#define MAX_LOAD 1 + +static void initialize_bucket(LOCKMAN *lm, LOCK * volatile *node, + uint bucket, LF_PINS *pins) +{ + int res; + uint parent= my_clear_highest_bit(bucket); + LOCK *dummy= (LOCK *)my_malloc(sizeof(LOCK), MYF(MY_WME)); + LOCK **tmp= 0, *cur; + LOCK * volatile *el= _lf_dynarray_lvalue(&lm->array, parent); + + if (*el == NULL && bucket) + initialize_bucket(lm, el, parent, pins); + dummy->hashnr= my_reverse_bits(bucket); + dummy->loid= 0; + dummy->lock= X; /* doesn't matter, in fact */ + dummy->resource= 0; + dummy->flags= 0; + res= lockinsert(el, dummy, pins, &cur); + DBUG_ASSERT(res & (ALREADY_HAVE_THE_LOCK | RESOURCE_WAS_UNLOCKED)); + if (res & ALREADY_HAVE_THE_LOCK) + { + my_free((void *)dummy, MYF(0)); + dummy= cur; + } + my_atomic_casptr((void **)node, (void **)&tmp, dummy); +} + +static inline uint calc_hash(uint64 resource) +{ + const uchar *pos= (uchar *)&resource; + ulong nr1= 1, nr2= 4, i; + for (i= 0; i < sizeof(resource) ; i++, pos++) + { + nr1^= (ulong) ((((uint) nr1 & 63)+nr2) * ((uint)*pos)) + (nr1 << 8); + nr2+= 3; + } + return nr1 & INT_MAX32; +} + +/* + RETURN + see enum lockman_getlock_result + NOTE + uses pins[0..3], they're removed on return +*/ +enum lockman_getlock_result lockman_getlock(LOCKMAN *lm, LOCK_OWNER *lo, + uint64 resource, + enum lock_type lock) +{ + int res; + uint csize, bucket, hashnr; + LOCK *node, * volatile *el, *blocker; + LF_PINS *pins= lo->pins; + enum lock_type old_lock; + + DBUG_ASSERT(lo->loid); + lf_rwlock_by_pins(pins); + node= (LOCK *)_lf_alloc_new(pins); + node->flags= 0; + node->lock= lock; + node->loid= lo->loid; + node->resource= resource; + hashnr= calc_hash(resource); + bucket= hashnr % lm->size; + el= _lf_dynarray_lvalue(&lm->array, bucket); + if (*el == NULL) + initialize_bucket(lm, el, bucket, pins); + node->hashnr= my_reverse_bits(hashnr) | 1; + res= lockinsert(el, node, pins, &blocker); + if (res & ALREADY_HAVE) + { + old_lock= blocker->lock; + _lf_assert_unpin(pins, 3); /* unpin should not be needed */ + _lf_alloc_free(pins, node); + lf_rwunlock_by_pins(pins); + res= getlock_result[old_lock][lock]; + DBUG_ASSERT(res); + return res; + } + /* a new value was added to the hash */ + csize= lm->size; + if ((my_atomic_add32(&lm->count, 1)+1.0) / csize > MAX_LOAD) + my_atomic_cas32(&lm->size, &csize, csize*2); + node->lonext= lo->all_locks; + lo->all_locks= node; + for ( ; res & NEED_TO_WAIT; res= lockpeek(el, node, pins, &blocker)) + { + LOCK_OWNER *wait_for_lo; + ulonglong deadline; + struct timespec timeout; + + _lf_assert_pin(pins, 3); /* blocker must be pinned here */ + lf_rwunlock_by_pins(pins); + + wait_for_lo= lm->loid_to_lo(blocker->loid); + /* + now, this is tricky. blocker is not necessarily a LOCK + we're waiting for. If it's compatible with what we want, + then we're waiting for a lock that blocker is waiting for + (see two places where blocker is set in lockfind) + In the latter case, let's "dereference" it + */ + if (lock_compatibility_matrix[blocker->lock][lock]) + { + blocker= wait_for_lo->all_locks; + lf_pin(pins, 3, blocker); + if (blocker != wait_for_lo->all_locks) + { + lf_rwlock_by_pins(pins); + continue; + } + wait_for_lo= wait_for_lo->waiting_for; + } + + /* + note that the blocker transaction may have ended by now, + its LOCK_OWNER and short id were reused, so 'wait_for_lo' may point + to an unrelated - albeit valid - LOCK_OWNER + */ + if (!wait_for_lo) + { + /* blocker transaction has ended, short id was released */ + lf_rwlock_by_pins(pins); + continue; + } + /* + We lock a mutex - it may belong to a wrong LOCK_OWNER, but it must + belong to _some_ LOCK_OWNER. It means, we can never free() a LOCK_OWNER, + if there're other active LOCK_OWNERs. + */ +#warning race condition here + pthread_mutex_lock(wait_for_lo->mutex); + if (DELETED(blocker->link)) + { + /* + blocker transaction was ended, or a savepoint that owned + the lock was rolled back. Either way - the lock was removed + */ + pthread_mutex_unlock(wait_for_lo->mutex); + lf_rwlock_by_pins(pins); + continue; + } + /* yuck. waiting */ + lo->waiting_for= wait_for_lo; + + deadline= my_getsystime() + lm->lock_timeout * 10000; + timeout.tv_sec= deadline/10000000; + timeout.tv_nsec= (deadline % 10000000) * 100; + do + { + pthread_cond_timedwait(wait_for_lo->cond, wait_for_lo->mutex, &timeout); + } while (!DELETED(blocker->link) && my_getsystime() < deadline); + pthread_mutex_unlock(wait_for_lo->mutex); + lf_rwlock_by_pins(pins); + if (!DELETED(blocker->link)) + { + /* + timeout. + note that we _don't_ release the lock request here. + Instead we're relying on the caller to abort the transaction, + and release all locks at once - see lockman_release_locks() + */ + lf_rwunlock_by_pins(pins); + return DIDNT_GET_THE_LOCK; + } + lo->waiting_for= 0; + } + _lf_assert_unpin(pins, 3); /* unpin should not be needed */ + lf_rwunlock_by_pins(pins); + return getlock_result[lock][lock]; +} + +/* + RETURN + 0 - deleted + 1 - didn't (not found) + NOTE + see lockdelete() for pin usage notes +*/ +int lockman_release_locks(LOCKMAN *lm, LOCK_OWNER *lo) +{ + LOCK * volatile *el, *node; + uint bucket; + LF_PINS *pins= lo->pins; + + pthread_mutex_lock(lo->mutex); + lf_rwlock_by_pins(pins); + for (node= lo->all_locks; node; node= node->lonext) + { + bucket= calc_hash(node->resource) % lm->size; + el= _lf_dynarray_lvalue(&lm->array, bucket); + if (*el == NULL) + initialize_bucket(lm, el, bucket, pins); + lockdelete(el, node, pins); + my_atomic_add32(&lm->count, -1); + } + lf_rwunlock_by_pins(pins); + lo->all_locks= 0; + /* now signal all waiters */ + pthread_cond_broadcast(lo->cond); + pthread_mutex_unlock(lo->mutex); + return 0; +} + +#ifdef MY_LF_EXTRA_DEBUG +/* + NOTE + the function below is NOT thread-safe !!! +*/ +static char *lock2str[]= +{ "N", "S", "X", "IS", "IX", "SIX", "LS", "LX", "SLX", "LSIX" }; +void print_lockhash(LOCKMAN *lm) +{ + LOCK *el= *(LOCK **)_lf_dynarray_lvalue(&lm->array, 0); + printf("hash: size=%u count=%u\n", lm->size, lm->count); + while (el) + { + intptr next= el->link; + if (el->hashnr & 1) + printf("0x%08x { resource %llu, loid %u, lock %s", + el->hashnr, el->resource, el->loid, lock2str[el->lock]); + else + { + printf("0x%08x { dummy ", el->hashnr); + DBUG_ASSERT(el->resource == 0 && el->loid == 0 && el->lock == X); + } + if (el->flags & IGNORE_ME) printf(" IGNORE_ME"); + if (el->flags & UPGRADED) printf(" UPGRADED"); + printf("}\n"); + el= (LOCK *)next; + } +} +#endif + diff --git a/storage/maria/lockman.h b/storage/maria/lockman.h new file mode 100644 index 00000000000..a3c96786935 --- /dev/null +++ b/storage/maria/lockman.h @@ -0,0 +1,73 @@ +/* Copyright (C) 2000 MySQL AB + + 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; either version 2 of the License, or + (at your option) any later version. + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ + +#ifndef _lockman_h +#define _lockman_h + +/* + N - "no lock", not a lock, used sometimes to simplify the code + S - Shared + X - eXclusive + IS - Intention Shared + IX - Intention eXclusive + SIX - Shared + Intention eXclusive + LS - Loose Shared + LX - Loose eXclusive + SLX - Shared + Loose eXclusive + LSIX - Loose Shared + Intention eXclusive +*/ +enum lock_type { N, S, X, IS, IX, SIX, LS, LX, SLX, LSIX }; + +struct lockman_lock; + +typedef struct st_lock_owner LOCK_OWNER; +struct st_lock_owner { + LF_PINS *pins; + struct lockman_lock *all_locks; + LOCK_OWNER *waiting_for; + pthread_cond_t *cond; /* transactions waiting for this, wait on 'cond' */ + pthread_mutex_t *mutex; /* mutex is required to use 'cond' */ + uint16 loid; +}; + +typedef LOCK_OWNER *loid_to_lo_func(uint16); +typedef struct { + LF_DYNARRAY array; /* hash itself */ + LF_ALLOCATOR alloc; /* allocator for elements */ + int32 volatile size; /* size of array */ + int32 volatile count; /* number of elements in the hash */ + uint lock_timeout; + loid_to_lo_func *loid_to_lo; +} LOCKMAN; + +enum lockman_getlock_result { + DIDNT_GET_THE_LOCK=0, GOT_THE_LOCK, + GOT_THE_LOCK_NEED_TO_LOCK_A_SUBRESOURCE, + GOT_THE_LOCK_NEED_TO_INSTANT_LOCK_A_SUBRESOURCE +}; + +void lockman_init(LOCKMAN *, loid_to_lo_func *, uint); +void lockman_destroy(LOCKMAN *); +enum lockman_getlock_result lockman_getlock(LOCKMAN *lm, LOCK_OWNER *lo, + uint64 resource, + enum lock_type lock); +int lockman_release_locks(LOCKMAN *, LOCK_OWNER *); + +#ifdef EXTRA_DEBUG +void print_lockhash(LOCKMAN *lm); +#endif + +#endif diff --git a/storage/maria/trnman.c b/storage/maria/trnman.c new file mode 100644 index 00000000000..49f49a3e26b --- /dev/null +++ b/storage/maria/trnman.c @@ -0,0 +1,332 @@ +/* Copyright (C) 2000 MySQL AB + + 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; either version 2 of the License, or + (at your option) any later version. + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ + + +#include +#include +#include +#include "trnman.h" + +uint trnman_active_transactions, trnman_allocated_transactions; + +static TRN active_list_min, active_list_max, + committed_list_min, committed_list_max, *pool; + +static pthread_mutex_t LOCK_trn_list; +static TrID global_trid_generator; + +static LF_HASH trid_to_trn; +static LOCKMAN maria_lockman; + +static TRN **short_trid_to_trn; +static my_atomic_rwlock_t LOCK_short_trid_to_trn, LOCK_pool; + +static byte *trn_get_hash_key(const byte *trn,uint* len, my_bool unused) +{ + *len= sizeof(TrID); + return (byte *) & ((*((TRN **)trn))->trid); +} + +static LOCK_OWNER *trnman_short_trid_to_TRN(uint16 short_trid) +{ + TRN *trn; + my_atomic_rwlock_rdlock(&LOCK_short_trid_to_trn); + trn= my_atomic_loadptr((void **)&short_trid_to_trn[short_trid]); + my_atomic_rwlock_rdunlock(&LOCK_short_trid_to_trn); + return (LOCK_OWNER *)trn; +} + +int trnman_init() +{ + pthread_mutex_init(&LOCK_trn_list, MY_MUTEX_INIT_FAST); + active_list_max.trid= active_list_min.trid= 0; + active_list_max.min_read_from= ~0; + active_list_max.next= active_list_min.prev= 0; + active_list_max.prev= &active_list_min; + active_list_min.next= &active_list_max; + trnman_active_transactions= 0; + trnman_allocated_transactions= 0; + + committed_list_max.commit_trid= ~0; + committed_list_max.next= committed_list_min.prev= 0; + committed_list_max.prev= &committed_list_min; + committed_list_min.next= &committed_list_max; + + pool= 0; + global_trid_generator= 0; /* set later by recovery code */ + lf_hash_init(&trid_to_trn, sizeof(TRN*), LF_HASH_UNIQUE, + 0, 0, trn_get_hash_key, 0); + my_atomic_rwlock_init(&LOCK_short_trid_to_trn); + my_atomic_rwlock_init(&LOCK_pool); + short_trid_to_trn= (TRN **)my_malloc(SHORT_TRID_MAX*sizeof(TRN*), + MYF(MY_WME|MY_ZEROFILL)); + if (!short_trid_to_trn) + return 1; + short_trid_to_trn--; /* min short_trid is 1 */ + + lockman_init(&maria_lockman, &trnman_short_trid_to_TRN, 10000); + + return 0; +} + +int trnman_destroy() +{ + DBUG_ASSERT(trid_to_trn.count == 0); + DBUG_ASSERT(trnman_active_transactions == 0); + DBUG_ASSERT(active_list_max.prev == &active_list_min); + DBUG_ASSERT(active_list_min.next == &active_list_max); + DBUG_ASSERT(committed_list_max.prev == &committed_list_min); + DBUG_ASSERT(committed_list_min.next == &committed_list_max); + while (pool) + { + TRN *trn= pool; + pool= pool->next; + DBUG_ASSERT(trn->locks.mutex == 0); + DBUG_ASSERT(trn->locks.cond == 0); + my_free((void *)trn, MYF(0)); + } + lf_hash_destroy(&trid_to_trn); + pthread_mutex_destroy(&LOCK_trn_list); + my_atomic_rwlock_destroy(&LOCK_short_trid_to_trn); + my_atomic_rwlock_destroy(&LOCK_pool); + my_free((void *)(short_trid_to_trn+1), MYF(0)); + lockman_destroy(&maria_lockman); +} + +static TrID new_trid() +{ + DBUG_ASSERT(global_trid_generator < 0xffffffffffffLL); + safe_mutex_assert_owner(&LOCK_trn_list); + return ++global_trid_generator; +} + +static void set_short_trid(TRN *trn) +{ + int i= (global_trid_generator + (intptr)trn) * 312089 % SHORT_TRID_MAX; + my_atomic_rwlock_wrlock(&LOCK_short_trid_to_trn); + for ( ; ; i= i % SHORT_TRID_MAX + 1) /* the range is [1..SHORT_TRID_MAX] */ + { + void *tmp= NULL; + if (short_trid_to_trn[i] == NULL && + my_atomic_casptr((void **)&short_trid_to_trn[i], &tmp, trn)) + break; + } + my_atomic_rwlock_wrunlock(&LOCK_short_trid_to_trn); + trn->locks.loid= i; +} + +TRN *trnman_new_trn(pthread_mutex_t *mutex, pthread_cond_t *cond) +{ + TRN *trn; + + /* + see trnman_end_trn to see why we need a mutex here + + and as we have a mutex, we can as well do everything + under it - allocating a TRN, incrementing trnman_active_transactions, + setting trn->min_read_from. + + Note that all the above is fast. generating short_trid may be slow, + as it involves scanning a big array - so it's still done + outside of the mutex. + */ + + pthread_mutex_lock(&LOCK_trn_list); + trnman_active_transactions++; + + trn= pool; + my_atomic_rwlock_wrlock(&LOCK_pool); + while (trn && !my_atomic_casptr((void **)&pool, (void **)&trn, + (void *)trn->next)) + /* no-op */; + my_atomic_rwlock_wrunlock(&LOCK_pool); + + if (!trn) + { + trn= (TRN *)my_malloc(sizeof(TRN), MYF(MY_WME)); + if (!trn) + { + pthread_mutex_unlock(&LOCK_trn_list); + return 0; + } + trnman_allocated_transactions++; + } + + trn->min_read_from= active_list_min.next->trid; + + trn->trid= new_trid(); + trn->locks.loid= 0; + + trn->next= &active_list_max; + trn->prev= active_list_max.prev; + active_list_max.prev= trn->prev->next= trn; + pthread_mutex_unlock(&LOCK_trn_list); + + trn->pins= lf_hash_get_pins(&trid_to_trn); + + if (!trn->min_read_from) + trn->min_read_from= trn->trid; + + trn->locks.mutex= mutex; + trn->locks.cond= cond; + trn->commit_trid= 0; + trn->locks.waiting_for= 0; + trn->locks.all_locks= 0; + trn->locks.pins= lf_alloc_get_pins(&maria_lockman.alloc); + + set_short_trid(trn); /* this must be the last! */ + + return trn; +} + +/* + remove a trn from the active list, + move to committed list, + set commit_trid + + TODO + integrate with log manager. That means: + a common "commit" mutex - forcing the log and setting commit_trid + must be done atomically (QQ how the heck it could be done with + group commit ???) XXX - why did I think it must be done atomically ? + + trid_to_trn, active_list_*, and committed_list_* can be + updated asyncronously. +*/ +void trnman_end_trn(TRN *trn, my_bool commit) +{ + int res; + TRN *free_me= 0; + LF_PINS *pins= trn->pins; + + pthread_mutex_lock(&LOCK_trn_list); + trn->next->prev= trn->prev; + trn->prev->next= trn->next; + + if (trn->prev == &active_list_min) + { + TRN *t; + for (t= committed_list_min.next; + t->commit_trid < active_list_min.next->min_read_from; + t= t->next) /* no-op */; + + if (t != committed_list_min.next) + { + free_me= committed_list_min.next; + committed_list_min.next= t; + t->prev->next= 0; + t->prev= &committed_list_min; + } + } + + if (commit && active_list_min.next != &active_list_max) + { + trn->commit_trid= global_trid_generator; + + trn->next= &committed_list_max; + trn->prev= committed_list_max.prev; + committed_list_max.prev= trn->prev->next= trn; + + res= lf_hash_insert(&trid_to_trn, pins, &trn); + DBUG_ASSERT(res == 0); + } + else + { + trn->next= free_me; + free_me= trn; + } + trnman_active_transactions--; + pthread_mutex_unlock(&LOCK_trn_list); + + lockman_release_locks(&maria_lockman, &trn->locks); + trn->locks.mutex= 0; + trn->locks.cond= 0; + my_atomic_rwlock_rdlock(&LOCK_short_trid_to_trn); + my_atomic_storeptr((void **)&short_trid_to_trn[trn->locks.loid], 0); + my_atomic_rwlock_rdunlock(&LOCK_short_trid_to_trn); + + + while (free_me) // XXX send them to the purge thread + { + int res; + TRN *t= free_me; + free_me= free_me->next; + + res= lf_hash_delete(&trid_to_trn, pins, &t->trid, sizeof(TrID)); + + trnman_free_trn(t); + } + + lf_hash_put_pins(pins); + lf_pinbox_put_pins(trn->locks.pins); +} + +/* + free a trn (add to the pool, that is) + note - we can never really free() a TRN if there's at least one + other running transaction - see, e.g., how lock waits are implemented + in lockman.c +*/ +void trnman_free_trn(TRN *trn) +{ + TRN *tmp= pool; + + my_atomic_rwlock_wrlock(&LOCK_pool); + do + { + /* + without volatile cast gcc-3.4.4 moved the assignment + down after the loop at -O2 + */ + *(TRN * volatile *)&(trn->next)= tmp; + } while (!my_atomic_casptr((void **)&pool, (void **)&tmp, trn)); + my_atomic_rwlock_wrunlock(&LOCK_pool); +} + +/* + NOTE + here we access the hash in a lock-free manner. + It's safe, a 'found' TRN can never be freed/reused before we access it. + In fact, it cannot be freed before 'trn' ends, because a 'found' TRN + can only be removed from the hash when: + found->commit_trid < ALL (trn->min_read_from) + that is, at least + found->commit_trid < trn->min_read_from + but + found->trid >= trn->min_read_from + and + found->commit_trid > found->trid +*/ +my_bool trnman_can_read_from(TRN *trn, TrID trid) +{ + TRN **found; + my_bool can; + LF_REQUIRE_PINS(3); + + if (trid < trn->min_read_from) + return TRUE; + if (trid > trn->trid) + return FALSE; + + found= lf_hash_search(&trid_to_trn, trn->pins, &trid, sizeof(trid)); + if (!found) + return FALSE; /* not in the hash of committed transactions = cannot read*/ + + can= (*found)->commit_trid < trn->trid; + lf_unpin(trn->pins, 2); + return can; +} + diff --git a/storage/maria/trnman.h b/storage/maria/trnman.h new file mode 100644 index 00000000000..9470678f3b2 --- /dev/null +++ b/storage/maria/trnman.h @@ -0,0 +1,48 @@ +/* Copyright (C) 2000 MySQL AB + + 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; either version 2 of the License, or + (at your option) any later version. + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ + +#ifndef _trnman_h +#define _trnman_h + +#include "lockman.h" + +typedef uint64 TrID; /* our TrID is 6 bytes */ +typedef struct st_transaction TRN; + +struct st_transaction +{ + LOCK_OWNER locks; + LF_PINS *pins; + TrID trid, min_read_from, commit_trid; + TRN *next, *prev; + /* Note! if locks.loid is 0, trn is NOT initialized */ +}; + +#define SHORT_TRID_MAX 65535 + +extern uint trnman_active_transactions, trnman_allocated_transactions; + +int trnman_init(void); +int trnman_destroy(void); +TRN *trnman_new_trn(pthread_mutex_t *mutex, pthread_cond_t *cond); +void trnman_end_trn(TRN *trn, my_bool commit); +#define trnman_commit_trn(T) trnman_end_trn(T, TRUE) +#define trnman_abort_trn(T) trnman_end_trn(T, FALSE) +void trnman_free_trn(TRN *trn); +my_bool trnman_can_read_from(TRN *trn, TrID trid); + +#endif + diff --git a/storage/maria/trxman.c b/storage/maria/trxman.c deleted file mode 100644 index a3e746af9ca..00000000000 --- a/storage/maria/trxman.c +++ /dev/null @@ -1,258 +0,0 @@ - -#include -#include -#include -#include "trxman.h" - -TRX active_list_min, active_list_max, - committed_list_min, committed_list_max, *pool; - -pthread_mutex_t LOCK_trx_list; -uint trxman_active_transactions, trxman_allocated_transactions; -TrID global_trid_generator; - -TRX **short_id_to_trx; -my_atomic_rwlock_t LOCK_short_id_to_trx; - -LF_HASH trid_to_trx; - -static byte *trx_get_hash_key(const byte *trx,uint* len, my_bool unused) -{ - *len= sizeof(TrID); - return (byte *) & ((*((TRX **)trx))->trid); -} - -int trxman_init() -{ - pthread_mutex_init(&LOCK_trx_list, MY_MUTEX_INIT_FAST); - active_list_max.trid= active_list_min.trid= 0; - active_list_max.min_read_from=~0; - active_list_max.next= active_list_min.prev= 0; - active_list_max.prev= &active_list_min; - active_list_min.next= &active_list_max; - trxman_active_transactions= 0; - trxman_allocated_transactions= 0; - - committed_list_max.commit_trid= ~0; - committed_list_max.next= committed_list_min.prev= 0; - committed_list_max.prev= &committed_list_min; - committed_list_min.next= &committed_list_max; - - pool=0; - global_trid_generator=0; /* set later by recovery code */ - lf_hash_init(&trid_to_trx, sizeof(TRX*), LF_HASH_UNIQUE, - 0, 0, trx_get_hash_key, 0); - my_atomic_rwlock_init(&LOCK_short_id_to_trx); - short_id_to_trx=(TRX **)my_malloc(SHORT_ID_MAX*sizeof(TRX*), - MYF(MY_WME|MY_ZEROFILL)); - if (!short_id_to_trx) - return 1; - short_id_to_trx--; /* min short_id is 1 */ - - return 0; -} - -int trxman_destroy() -{ - DBUG_ASSERT(trid_to_trx.count == 0); - DBUG_ASSERT(trxman_active_transactions == 0); - DBUG_ASSERT(active_list_max.prev == &active_list_min); - DBUG_ASSERT(active_list_min.next == &active_list_max); - DBUG_ASSERT(committed_list_max.prev == &committed_list_min); - DBUG_ASSERT(committed_list_min.next == &committed_list_max); - while (pool) - { - TRX *tmp=pool->next; - my_free((void *)pool, MYF(0)); - pool=tmp; - } - lf_hash_destroy(&trid_to_trx); - pthread_mutex_destroy(&LOCK_trx_list); - my_atomic_rwlock_destroy(&LOCK_short_id_to_trx); - my_free((void *)(short_id_to_trx+1), MYF(0)); -} - -static TrID new_trid() -{ - DBUG_ASSERT(global_trid_generator < 0xffffffffffffLL); - safe_mutex_assert_owner(&LOCK_trx_list); - return ++global_trid_generator; -} - -static void set_short_id(TRX *trx) -{ - int i= (global_trid_generator + (intptr)trx) * 312089 % SHORT_ID_MAX; - my_atomic_rwlock_wrlock(&LOCK_short_id_to_trx); - for ( ; ; i= i % SHORT_ID_MAX + 1) /* the range is [1..SHORT_ID_MAX] */ - { - void *tmp=NULL; - if (short_id_to_trx[i] == NULL && - my_atomic_casptr((void **)&short_id_to_trx[i], &tmp, trx)) - break; - } - my_atomic_rwlock_wrunlock(&LOCK_short_id_to_trx); - trx->short_id= i; -} - -TRX *trxman_new_trx() -{ - TRX *trx; - - my_atomic_add32(&trxman_active_transactions, 1); - - /* - see trxman_end_trx to see why we need a mutex here - - and as we have a mutex, we can as well do everything - under it - allocating a TRX, incrementing trxman_active_transactions, - setting trx->min_read_from. - - Note that all the above is fast. generating short_id may be slow, - as it involves scanning a big array - so it's still done - outside of the mutex. - */ - - pthread_mutex_lock(&LOCK_trx_list); - trx=pool; - while (trx && !my_atomic_casptr((void **)&pool, (void **)&trx, trx->next)) - /* no-op */; - - if (!trx) - { - trx=(TRX *)my_malloc(sizeof(TRX), MYF(MY_WME)); - trxman_allocated_transactions++; - } - if (!trx) - return 0; - - trx->min_read_from= active_list_min.next->trid; - - trx->trid= new_trid(); - trx->short_id= 0; - - trx->next= &active_list_max; - trx->prev= active_list_max.prev; - active_list_max.prev= trx->prev->next= trx; - pthread_mutex_unlock(&LOCK_trx_list); - - trx->pins=lf_hash_get_pins(&trid_to_trx); - - if (!trx->min_read_from) - trx->min_read_from= trx->trid; - - trx->commit_trid=0; - - set_short_id(trx); /* this must be the last! */ - - - return trx; -} - -/* - remove a trx from the active list, - move to committed list, - set commit_trid - - TODO - integrate with lock manager, log manager. That means: - a common "commit" mutex - forcing the log and setting commit_trid - must be done atomically (QQ how the heck it could be done with - group commit ???) - - trid_to_trx, active_list_*, and committed_list_* can be - updated asyncronously. -*/ -void trxman_end_trx(TRX *trx, my_bool commit) -{ - int res; - TRX *free_me= 0; - LF_PINS *pins= trx->pins; - - pthread_mutex_lock(&LOCK_trx_list); - trx->next->prev= trx->prev; - trx->prev->next= trx->next; - - if (trx->prev == &active_list_min) - { - TRX *t; - for (t= committed_list_min.next; - t->commit_trid < active_list_min.next->min_read_from; - t= t->next) /* no-op */; - - if (t != committed_list_min.next) - { - free_me= committed_list_min.next; - committed_list_min.next= t; - t->prev->next=0; - t->prev= &committed_list_min; - } - } - - my_atomic_rwlock_wrlock(&LOCK_short_id_to_trx); - my_atomic_storeptr((void **)&short_id_to_trx[trx->short_id], 0); - my_atomic_rwlock_wrunlock(&LOCK_short_id_to_trx); - - if (commit && active_list_min.next != &active_list_max) - { - trx->commit_trid= global_trid_generator; - - trx->next= &committed_list_max; - trx->prev= committed_list_max.prev; - committed_list_max.prev= trx->prev->next= trx; - - res= lf_hash_insert(&trid_to_trx, pins, &trx); - DBUG_ASSERT(res == 0); - } - else - { - trx->next=free_me; - free_me=trx; - } - pthread_mutex_unlock(&LOCK_trx_list); - - my_atomic_add32(&trxman_active_transactions, -1); - - while (free_me) - { - int res; - TRX *t= free_me; - free_me= free_me->next; - - res= lf_hash_delete(&trid_to_trx, pins, &t->trid, sizeof(TrID)); - - trxman_free_trx(t); - } - - lf_hash_put_pins(pins); -} - -/* free a trx (add to the pool, that is */ -void trxman_free_trx(TRX *trx) -{ - TRX *tmp=pool; - - do - { - trx->next=tmp; - } while (!my_atomic_casptr((void **)&pool, (void **)&tmp, trx)); -} - -my_bool trx_can_read_from(TRX *trx, TrID trid) -{ - TRX *found; - my_bool can; - - if (trid < trx->min_read_from) - return TRUE; - if (trid > trx->trid) - return FALSE; - - found= lf_hash_search(&trid_to_trx, trx->pins, &trid, sizeof(trid)); - if (!found) - return FALSE; /* not in the hash = cannot read */ - - can= found->commit_trid < trx->trid; - lf_unpin(trx->pins, 2); - return can; -} - diff --git a/storage/maria/trxman.h b/storage/maria/trxman.h deleted file mode 100644 index 5ac989d03a4..00000000000 --- a/storage/maria/trxman.h +++ /dev/null @@ -1,28 +0,0 @@ - -typedef uint64 TrID; /* our TrID is 6 bytes */ - -typedef struct st_transaction -{ - TrID trid, min_read_from, commit_trid; - struct st_transaction *next, *prev; - /* Note! if short_id is 0, trx is NOT initialized */ - uint16 short_id; - LF_PINS *pins; -} TRX; - -#define SHORT_ID_MAX 65535 - -extern uint trxman_active_transactions, trxman_allocated_transactions; - -extern TRX **short_id_to_trx; -extern my_atomic_rwlock_t LOCK_short_id_to_trx; - -int trxman_init(); -int trxman_end(); -TRX *trxman_new_trx(); -void trxman_end_trx(TRX *trx, my_bool commit); -#define trxman_commit_trx(T) trxman_end_trx(T, TRUE) -#define trxman_abort_trx(T) trxman_end_trx(T, FALSE) -void trxman_free_trx(TRX *trx); -my_bool trx_can_read_from(TRX *trx, TrID trid); - diff --git a/storage/maria/unittest/Makefile.am b/storage/maria/unittest/Makefile.am index 8a5ca3d669f..e29bc7f86cb 100644 --- a/storage/maria/unittest/Makefile.am +++ b/storage/maria/unittest/Makefile.am @@ -25,5 +25,5 @@ LDADD= $(top_builddir)/unittest/mytap/libmytap.a \ $(top_builddir)/mysys/libmysys.a \ $(top_builddir)/dbug/libdbug.a \ $(top_builddir)/strings/libmystrings.a @ZLIB_LIBS@ -noinst_PROGRAMS = ma_control_file-t +noinst_PROGRAMS = ma_control_file-t trnman-t lockman-t CLEANFILES = maria_control diff --git a/storage/maria/unittest/lockman-t.c b/storage/maria/unittest/lockman-t.c new file mode 100644 index 00000000000..8b62ccfe094 --- /dev/null +++ b/storage/maria/unittest/lockman-t.c @@ -0,0 +1,246 @@ +/* Copyright (C) 2006 MySQL AB + + 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; either version 2 of the License, or + (at your option) any later version. + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ + +//#define EXTRA_VERBOSE + +#include + +#include +#include +#include +#include +#include "../lockman.h" + +#define Nlos 10 +LOCK_OWNER loarray[Nlos]; +pthread_mutex_t mutexes[Nlos]; +pthread_cond_t conds[Nlos]; +LOCKMAN lockman; + +#ifndef EXTRA_VERBOSE +#define print_lockhash(X) /* no-op */ +#define DIAG(X) /* no-op */ +#else +#define DIAG(X) diag X +#endif + +LOCK_OWNER *loid2lo(uint16 loid) +{ + return loarray+loid-1; +} + +#define unlock_all(O) diag("lo" #O "> release all locks"); \ + lockman_release_locks(&lockman, loid2lo(O));print_lockhash(&lockman) +#define test_lock(O, R, L, S, RES) \ + ok(lockman_getlock(&lockman, loid2lo(O), R, L) == RES, \ + "lo" #O "> " S " lock resource " #R " with " #L "-lock"); \ + print_lockhash(&lockman) +#define lock_ok_a(O,R,L) test_lock(O,R,L,"",GOT_THE_LOCK) +#define lock_ok_i(O,R,L) test_lock(O,R,L,"",GOT_THE_LOCK_NEED_TO_LOCK_A_SUBRESOURCE) +#define lock_ok_l(O,R,L) test_lock(O,R,L,"",GOT_THE_LOCK_NEED_TO_INSTANT_LOCK_A_SUBRESOURCE) +#define lock_conflict(O,R,L) test_lock(O,R,L,"cannot ",DIDNT_GET_THE_LOCK); \ + unlock_all(O) + +void test_lockman_simple() +{ + /* simple */ + lock_ok_a(1, 1, S); + lock_ok_i(2, 2, IS); + lock_ok_i(1, 2, IX); + /* lock escalation */ + lock_ok_a(1, 1, X); + lock_ok_i(2, 2, IX); + /* failures */ + lock_conflict(2,1,X); /* this removes all locks of lo2 */ + lock_ok_a(1,2,S); + lock_ok_a(1,2,IS); + lock_ok_a(1,2,LS); + lock_ok_i(1,3,IX); + lock_ok_a(2,3,LS); + lock_ok_i(1,3,IX); + lock_ok_l(2,3,IS); + lockman_release_locks(&lockman, loid2lo(1)); + lockman_release_locks(&lockman, loid2lo(2)); + +} + +pthread_attr_t rt_attr; +pthread_mutex_t rt_mutex; +pthread_cond_t rt_cond; +int rt_num_threads; +int litmus; +void run_test(const char *test, pthread_handler handler, int n, int m) +{ + pthread_t t; + ulonglong now= my_getsystime(); + + litmus= 0; + + diag("Testing %s with %d threads, %d iterations... ", test, n, m); + for (rt_num_threads= n ; n ; n--) + pthread_create(&t, &rt_attr, handler, &m); + pthread_mutex_lock(&rt_mutex); + while (rt_num_threads) + pthread_cond_wait(&rt_cond, &rt_mutex); + pthread_mutex_unlock(&rt_mutex); + now= my_getsystime()-now; + ok(litmus == 0, "tested %s in %g secs (%d)", test, ((double)now)/1e7, litmus); +} + +int thread_number= 0, timeouts=0; +#define Nrows 1000 +#define Ntables 10 +#define TABLE_LOCK_RATIO 10 +enum lock_type lock_array[6]={S,X,LS,LX,IS,IX}; +char *lock2str[6]={"S","X","LS","LX","IS","IX"}; +char *res2str[6]={ + "DIDN'T GET THE LOCK", + "GOT THE LOCK", + "GOT THE LOCK NEED TO LOCK A SUBRESOURCE", + "GOT THE LOCK NEED TO INSTANT LOCK A SUBRESOURCE"}; +pthread_handler_t test_lockman(void *arg) +{ + int m= (*(int *)arg); + uint x, loid, row, table, res, locklevel, timeout= 0; + LOCK_OWNER *lo; + + pthread_mutex_lock(&rt_mutex); + loid= ++thread_number; + pthread_mutex_unlock(&rt_mutex); + lo= loid2lo(loid); + + for (x= ((int)(intptr)(&m)); m > 0; m--) + { + x= (x*3628273133 + 1500450271) % 9576890767; /* three prime numbers */ + row= x % Nrows + Ntables; + table= row % Ntables; + locklevel= (x/Nrows) & 3; + if ((x/Nrows/4) % TABLE_LOCK_RATIO == 0) + { /* table lock */ + res= lockman_getlock(&lockman, lo, table, lock_array[locklevel]); + DIAG(("loid=%2d, table %d lock %s, res=%s", loid, table, lock2str[locklevel], res2str[res])); + if (res == DIDNT_GET_THE_LOCK) + { + lockman_release_locks(&lockman, lo); + DIAG(("loid=%2d, release all locks", loid)); + timeout++; + continue; + } + DBUG_ASSERT(res == GOT_THE_LOCK); + } + else + { /* row lock */ + locklevel&= 1; + res= lockman_getlock(&lockman, lo, table, lock_array[locklevel + 4]); + DIAG(("loid=%2d, row %d lock %s, res=%s", loid, row, lock2str[locklevel+4], res2str[res])); + switch (res) + { + case DIDNT_GET_THE_LOCK: + lockman_release_locks(&lockman, lo); + DIAG(("loid=%2d, release all locks", loid)); + timeout++; + continue; + case GOT_THE_LOCK: + continue; + case GOT_THE_LOCK_NEED_TO_INSTANT_LOCK_A_SUBRESOURCE: + /* not implemented, so take a regular lock */ + case GOT_THE_LOCK_NEED_TO_LOCK_A_SUBRESOURCE: + res= lockman_getlock(&lockman, lo, row, lock_array[locklevel]); + DIAG(("loid=%2d, ROW %d lock %s, res=%s", loid, row, lock2str[locklevel], res2str[res])); + if (res == DIDNT_GET_THE_LOCK) + { + lockman_release_locks(&lockman, lo); + DIAG(("loid=%2d, release all locks", loid)); + timeout++; + continue; + } + DBUG_ASSERT(res == GOT_THE_LOCK); + continue; + default: + DBUG_ASSERT(0); + } + } + } + + lockman_release_locks(&lockman, lo); + + pthread_mutex_lock(&rt_mutex); + rt_num_threads--; + timeouts+= timeout; + if (!rt_num_threads) + { + pthread_cond_signal(&rt_cond); + diag("number of timeouts: %d", timeouts); + } + pthread_mutex_unlock(&rt_mutex); + + return 0; +} + +int main() +{ + int i; + + my_init(); + + plan(14); + + if (my_atomic_initialize()) + return exit_status(); + + pthread_attr_init(&rt_attr); + pthread_attr_setdetachstate(&rt_attr,PTHREAD_CREATE_DETACHED); + pthread_mutex_init(&rt_mutex, 0); + pthread_cond_init(&rt_cond, 0); + + lockman_init(&lockman, &loid2lo, 50); + + for (i= 0; i < Nlos; i++) + { + loarray[i].pins= lf_alloc_get_pins(&lockman.alloc); + loarray[i].all_locks= 0; + loarray[i].waiting_for= 0; + pthread_mutex_init(&mutexes[i], MY_MUTEX_INIT_FAST); + pthread_cond_init (&conds[i], 0); + loarray[i].mutex= &mutexes[i]; + loarray[i].cond= &conds[i]; + loarray[i].loid= i+1; + } + + test_lockman_simple(); + +#define CYCLES 100 +#define THREADS Nlos /* don't change this line */ + + run_test("lockman", test_lockman, THREADS,CYCLES); + + for (i= 0; i < Nlos; i++) + { + lockman_release_locks(&lockman, &loarray[i]); + pthread_mutex_destroy(loarray[i].mutex); + pthread_cond_destroy(loarray[i].cond); + lf_pinbox_put_pins(loarray[i].pins); + } + + lockman_destroy(&lockman); + + pthread_mutex_destroy(&rt_mutex); + pthread_cond_destroy(&rt_cond); + pthread_attr_destroy(&rt_attr); + my_end(0); + return exit_status(); +} + diff --git a/storage/maria/unittest/trnman-t.c b/storage/maria/unittest/trnman-t.c new file mode 100644 index 00000000000..6d4b48c6d3d --- /dev/null +++ b/storage/maria/unittest/trnman-t.c @@ -0,0 +1,177 @@ +/* Copyright (C) 2006 MySQL AB + + 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; either version 2 of the License, or + (at your option) any later version. + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ + +#include + +#include +#include +#include +#include +#include "../trnman.h" + +pthread_attr_t rt_attr; +pthread_mutex_t rt_mutex; +pthread_cond_t rt_cond; +int rt_num_threads; + +int litmus; + +/* + create and end (commit or rollback) transactions randomly +*/ +#define MAX_ITER 100 +pthread_handler_t test_trnman(void *arg) +{ + int m= (*(int *)arg); + uint x, y, i, j, n; + TRN *trn[MAX_ITER]; + pthread_mutex_t mutexes[MAX_ITER]; + pthread_cond_t conds[MAX_ITER]; + + for (i=0; i < MAX_ITER; i++) + { + pthread_mutex_init(&mutexes[i], MY_MUTEX_INIT_FAST); + pthread_cond_init(&conds[i], 0); + } + + for (x= ((int)(intptr)(&m)); m > 0; ) + { + y= x= (x*3628273133 + 1500450271) % 9576890767; /* three prime numbers */ + m-= n= x % MAX_ITER; + for (i= 0; i < n; i++) + trn[i]= trnman_new_trn(&mutexes[i], &conds[i]); + for (i= 0; i < n; i++) + { + y= (y*19 + 7) % 31; + trnman_end_trn(trn[i], y & 1); + } + } + + for (i=0; i < MAX_ITER; i++) + { + pthread_mutex_destroy(&mutexes[i]); + pthread_cond_destroy(&conds[i]); + } + pthread_mutex_lock(&rt_mutex); + rt_num_threads--; + if (!rt_num_threads) + pthread_cond_signal(&rt_cond); + pthread_mutex_unlock(&rt_mutex); + + return 0; +} +#undef MAX_ITER + +void run_test(const char *test, pthread_handler handler, int n, int m) +{ + pthread_t t; + ulonglong now= my_getsystime(); + + litmus= 0; + + diag("Testing %s with %d threads, %d iterations... ", test, n, m); + for (rt_num_threads= n ; n ; n--) + pthread_create(&t, &rt_attr, handler, &m); + pthread_mutex_lock(&rt_mutex); + while (rt_num_threads) + pthread_cond_wait(&rt_cond, &rt_mutex); + pthread_mutex_unlock(&rt_mutex); + now= my_getsystime()-now; + ok(litmus == 0, "tested %s in %g secs (%d)", test, ((double)now)/1e7, litmus); +} + +#define ok_read_from(T1, T2, RES) \ + i=trnman_can_read_from(trn[T1], trid[T2]); \ + ok(i == RES, "trn" #T1 " %s read from trn" #T2, i ? "can" : "cannot") +#define start_transaction(T) \ + trn[T]= trnman_new_trn(&mutexes[T], &conds[T]); \ + trid[T]= trn[T]->trid +#define commit(T) trnman_commit_trn(trn[T]) +#define abort(T) trnman_abort_trn(trn[T]) + +#define Ntrns 4 +void test_trnman_read_from() +{ + TRN *trn[Ntrns]; + TrID trid[Ntrns]; + pthread_mutex_t mutexes[Ntrns]; + pthread_cond_t conds[Ntrns]; + int i; + + for (i=0; i < Ntrns; i++) + { + pthread_mutex_init(&mutexes[i], MY_MUTEX_INIT_FAST); + pthread_cond_init(&conds[i], 0); + } + + start_transaction(0); /* start trn1 */ + start_transaction(1); /* start trn2 */ + ok_read_from(1,0,0); + commit(0); /* commit trn1 */ + start_transaction(2); /* start trn4 */ + abort(2); /* abort trn4 */ + start_transaction(3); /* start trn5 */ + ok_read_from(3,0,1); + ok_read_from(3,1,0); + ok_read_from(3,2,0); + commit(1); /* commit trn2 */ + ok_read_from(3,1,0); + commit(3); /* commit trn5 */ + + for (i=0; i < Ntrns; i++) + { + pthread_mutex_destroy(&mutexes[i]); + pthread_cond_destroy(&conds[i]); + } +} + +int main() +{ + my_init(); + + plan(6); + + if (my_atomic_initialize()) + return exit_status(); + + pthread_attr_init(&rt_attr); + pthread_attr_setdetachstate(&rt_attr,PTHREAD_CREATE_DETACHED); + pthread_mutex_init(&rt_mutex, 0); + pthread_cond_init(&rt_cond, 0); + +#define CYCLES 10000 +#define THREADS 10 + + trnman_init(); + + test_trnman_read_from(); + run_test("trnman", test_trnman, THREADS,CYCLES); + + diag("mallocs: %d", trnman_allocated_transactions); + { + ulonglong now= my_getsystime(); + trnman_destroy(); + now= my_getsystime()-now; + diag("trnman_destroy: %g", ((double)now)/1e7); + } + + pthread_mutex_destroy(&rt_mutex); + pthread_cond_destroy(&rt_cond); + pthread_attr_destroy(&rt_attr); + my_end(0); + return exit_status(); +} + diff --git a/unittest/maria/Makefile.am b/unittest/maria/Makefile.am deleted file mode 100644 index 667d1e09a07..00000000000 --- a/unittest/maria/Makefile.am +++ /dev/null @@ -1,12 +0,0 @@ - -AM_CPPFLAGS = @ZLIB_INCLUDES@ -I$(top_builddir)/include -AM_CPPFLAGS += -I$(top_srcdir)/include -I$(top_srcdir)/unittest/mytap - -LDADD = $(top_builddir)/unittest/mytap/libmytap.a \ - $(top_builddir)/storage/maria/libmaria.a \ - $(top_builddir)/mysys/libmysys.a \ - $(top_builddir)/dbug/libdbug.a \ - $(top_builddir)/strings/libmystrings.a - -noinst_PROGRAMS = trxman-t - diff --git a/unittest/maria/trxman-t.c b/unittest/maria/trxman-t.c deleted file mode 100644 index 7536e5534bf..00000000000 --- a/unittest/maria/trxman-t.c +++ /dev/null @@ -1,137 +0,0 @@ -/* Copyright (C) 2006 MySQL AB - - 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; either version 2 of the License, or - (at your option) any later version. - - 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ - -#include - -#include -#include -#include -#include -#include "../../storage/maria/trxman.h" - -pthread_attr_t rt_attr; -pthread_mutex_t rt_mutex; -pthread_cond_t rt_cond; -int rt_num_threads; - -int litmus; - -/* template for a test: the goal is to have litmus==0 if the test passed - -#define ITER nnn -pthread_handler_t test_XXXXXXXX(void *arg) -{ - int m=(*(int *)arg)/ITER, x; - - for (x=((int)(intptr)(&m)); m ; m--) - { - // do something with litmus - } - // do something more with litmus - - pthread_mutex_lock(&rt_mutex); - rt_num_threads--; - if (!rt_num_threads) - { - diag("whatever diagnostics we want", blabla, foobar); - pthread_cond_signal(&rt_cond); - } - pthread_mutex_unlock(&rt_mutex); - return 0; -} -#undef ITER - -*/ - -/* - create and end (commit or rollback) transactions randomly -*/ -#define MAX_ITER 100 -pthread_handler_t test_trxman(void *arg) -{ - int m=(*(int *)arg); - uint x, y, i, j, n; - TRX *trx[MAX_ITER]; - - for (x=((int)(intptr)(&m)); m > 0; ) - { - y= x= (x*3628273133 + 1500450271) % 9576890767; /* three prime numbers */ - m-= n= x % MAX_ITER; - for (i=0; i < n; i++) - trx[i]=trxman_new_trx(); - for (i=0; i < n; i++) - { - y=(y*19 + 7) % 31; - trxman_end_trx(trx[i], y & 1); - } - } - - pthread_mutex_lock(&rt_mutex); - rt_num_threads--; - if (!rt_num_threads) - pthread_cond_signal(&rt_cond); - pthread_mutex_unlock(&rt_mutex); - return 0; -} -#undef MAX_ITER - -void run_test(const char *test, pthread_handler handler, int n, int m) -{ - pthread_t t; - ulonglong now=my_getsystime(); - - litmus= 0; - - diag("Testing %s with %d threads, %d iterations... ", test, n, m); - for (rt_num_threads=n ; n ; n--) - pthread_create(&t, &rt_attr, handler, &m); - pthread_mutex_lock(&rt_mutex); - while (rt_num_threads) - pthread_cond_wait(&rt_cond, &rt_mutex); - pthread_mutex_unlock(&rt_mutex); - now=my_getsystime()-now; - ok(litmus == 0, "tested %s in %g secs (%d)", test, ((double)now)/1e7, litmus); -} - -int main() -{ - plan(1); - - if (my_atomic_initialize()) - return exit_status(); - - my_init(); - - pthread_attr_init(&rt_attr); - pthread_attr_setdetachstate(&rt_attr,PTHREAD_CREATE_DETACHED); - pthread_mutex_init(&rt_mutex, 0); - pthread_cond_init(&rt_cond, 0); - -#define CYCLES 10000 -#define THREADS 10 - - trxman_init(); - run_test("trxman", test_trxman, THREADS,CYCLES); - trxman_destroy(); - diag("mallocs: %d\n", trxman_allocated_transactions); - - pthread_mutex_destroy(&rt_mutex); - pthread_cond_destroy(&rt_cond); - pthread_attr_destroy(&rt_attr); - my_end(0); - return exit_status(); -} - -- cgit v1.2.1 From 0cb46440105a7b31b2959f4c8fa22be5e333e87a Mon Sep 17 00:00:00 2001 From: unknown Date: Sun, 15 Oct 2006 22:26:42 +0300 Subject: This is not supposed to be pushed, it is just to check of source of mac os problems storage/maria/_MakeLists.txt: Rename: storage/maria/CMakeLists.txt -> storage/maria/_MakeLists.txt --- storage/maria/CMakeLists.txt | 1 - storage/maria/_MakeLists.txt | 1 + 2 files changed, 1 insertion(+), 1 deletion(-) delete mode 100644 storage/maria/CMakeLists.txt create mode 100644 storage/maria/_MakeLists.txt diff --git a/storage/maria/CMakeLists.txt b/storage/maria/CMakeLists.txt deleted file mode 100644 index cfe23054e2f..00000000000 --- a/storage/maria/CMakeLists.txt +++ /dev/null @@ -1 +0,0 @@ -# empty for the moment; will fill it when we build under Windows diff --git a/storage/maria/_MakeLists.txt b/storage/maria/_MakeLists.txt new file mode 100644 index 00000000000..cfe23054e2f --- /dev/null +++ b/storage/maria/_MakeLists.txt @@ -0,0 +1 @@ +# empty for the moment; will fill it when we build under Windows -- cgit v1.2.1 From 12a55aeabc353fdc1c3829ddd8baacb142160c80 Mon Sep 17 00:00:00 2001 From: unknown Date: Wed, 18 Oct 2006 17:24:07 +0200 Subject: lock manager passed unit tests storage/maria/trnman.c: comments include/my_dbug.h: make DBUG_ASSERT always a statement storage/maria/lockman.h: comments include/lf.h: lf_pinbox - don't use a fixed-size purgatory. mysys/lf_alloc-pin.c: lf_pinbox - don't use a fixed-size purgatory. mysys/lf_hash.c: lf_pinbox - don't use a fixed-size purgatory. storage/maria/lockman.c: removed IGNORE_ME/UPGDARED matching - it was wrong in the first place. updated for "lf_pinbox - don't use a fixed-size purgatory" storage/maria/unittest/lockman-t.c: IGNORE_ME/UPGRADED pair counting bugtest. more tests unittest/mysys/my_atomic-t.c: lf_pinbox - don't use a fixed-size purgatory. --- include/lf.h | 15 +-- include/my_dbug.h | 2 +- mysys/lf_alloc-pin.c | 172 ++++++++++++++++---------------- mysys/lf_hash.c | 4 +- storage/maria/lockman.c | 199 +++++++++++++++++++++++++++---------- storage/maria/lockman.h | 9 +- storage/maria/trnman.c | 22 ++-- storage/maria/unittest/lockman-t.c | 85 ++++++++++++---- unittest/mysys/my_atomic-t.c | 4 +- 9 files changed, 330 insertions(+), 182 deletions(-) diff --git a/include/lf.h b/include/lf.h index 4c6765b2d40..45c5f109e1c 100644 --- a/include/lf.h +++ b/include/lf.h @@ -96,20 +96,21 @@ typedef void lf_pinbox_free_func(void *, void *); typedef struct { LF_DYNARRAY pinstack; lf_pinbox_free_func *free_func; - void * free_func_arg; + void *free_func_arg; + uint free_ptr_offset; uint32 volatile pinstack_top_ver; /* this is a versioned pointer */ uint32 volatile pins_in_stack; /* number of elements in array */ } LF_PINBOX; -/* we want sizeof(LF_PINS) to be close to 128 to avoid false sharing */ +/* we want sizeof(LF_PINS) to be 128 to avoid false sharing */ typedef struct { void * volatile pin[LF_PINBOX_PINS]; - void * purgatory[LF_PURGATORY_SIZE]; LF_PINBOX *pinbox; + void *purgatory; uint32 purgatory_count; uint32 volatile link; char pad[128-sizeof(uint32)*2 - -sizeof(void *)*(LF_PINBOX_PINS+LF_PURGATORY_SIZE+1)]; + -sizeof(void *)*(LF_PINBOX_PINS+2)]; } LF_PINS; #define lf_rwlock_by_pins(PINS) \ @@ -147,8 +148,8 @@ typedef struct { #define _lf_assert_pin(PINS, PIN) assert((PINS)->pin[PIN] != 0) #define _lf_assert_unpin(PINS, PIN) assert((PINS)->pin[PIN]==0) -void lf_pinbox_init(LF_PINBOX *pinbox, lf_pinbox_free_func *free_func, - void * free_func_arg); +void lf_pinbox_init(LF_PINBOX *pinbox, uint free_ptr_offset, + lf_pinbox_free_func *free_func, void * free_func_arg); void lf_pinbox_destroy(LF_PINBOX *pinbox); lock_wrap(lf_pinbox_get_pins, LF_PINS *, @@ -181,7 +182,7 @@ typedef struct st_lf_allocator { uint32 volatile mallocs; } LF_ALLOCATOR; -void lf_alloc_init(LF_ALLOCATOR *allocator, uint size); +void lf_alloc_init(LF_ALLOCATOR *allocator, uint size, uint free_ptr_offset); void lf_alloc_destroy(LF_ALLOCATOR *allocator); uint lf_alloc_in_pool(LF_ALLOCATOR *allocator); #define _lf_alloc_free(PINS, PTR) _lf_pinbox_free((PINS), (PTR)) diff --git a/include/my_dbug.h b/include/my_dbug.h index a397861c1af..e3fdf9d6461 100644 --- a/include/my_dbug.h +++ b/include/my_dbug.h @@ -100,7 +100,7 @@ extern FILE *_db_fp_(void); #define DBUG_LONGJMP(a1) longjmp(a1) #define DBUG_DUMP(keyword,a1,a2) #define DBUG_END() -#define DBUG_ASSERT(A) +#define DBUG_ASSERT(A) do { } while(0) #define DBUG_LOCK_FILE #define DBUG_FILE (stderr) #define DBUG_UNLOCK_FILE diff --git a/mysys/lf_alloc-pin.c b/mysys/lf_alloc-pin.c index cf1612b73d1..842c9690463 100644 --- a/mysys/lf_alloc-pin.c +++ b/mysys/lf_alloc-pin.c @@ -15,21 +15,7 @@ Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ /* - concurrent allocator based on pinning addresses - - strictly speaking it's not lock-free, as it can be blocked - if a thread's purgatory is full and all addresses from there - are pinned. - - But until the above happens, it's wait-free. - - It can be made strictly wait-free by increasing purgatory size. - If it's larger than pins_in_stack*LF_PINBOX_PINS, then apocalyptical - condition above will never happen. But than the memory requirements - will be O(pins_in_stack^2). - - Note, that for large purgatory sizes it makes sense to remove - purgatory array, and link objects in a list using embedded pointer. + wait-free concurrent allocator based on pinning addresses TODO test with more than 256 threads TODO test w/o alloca @@ -43,15 +29,17 @@ static void _lf_pinbox_real_free(LF_PINS *pins); -void lf_pinbox_init(LF_PINBOX *pinbox, lf_pinbox_free_func *free_func, - void *free_func_arg) +void lf_pinbox_init(LF_PINBOX *pinbox, uint free_ptr_offset, + lf_pinbox_free_func *free_func,void *free_func_arg) { DBUG_ASSERT(sizeof(LF_PINS) == 128); + DBUG_ASSERT(free_ptr_offset % sizeof(void *) == 0); lf_dynarray_init(&pinbox->pinstack, sizeof(LF_PINS)); - pinbox->pinstack_top_ver=0; - pinbox->pins_in_stack=0; - pinbox->free_func=free_func; - pinbox->free_func_arg=free_func_arg; + pinbox->pinstack_top_ver= 0; + pinbox->pins_in_stack= 0; + pinbox->free_ptr_offset= free_ptr_offset; + pinbox->free_func= free_func; + pinbox->free_func_arg= free_func_arg; } void lf_pinbox_destroy(LF_PINBOX *pinbox) @@ -64,58 +52,64 @@ LF_PINS *_lf_pinbox_get_pins(LF_PINBOX *pinbox) uint32 pins, next, top_ver; LF_PINS *el; - top_ver=pinbox->pinstack_top_ver; + top_ver= pinbox->pinstack_top_ver; do { - if (!(pins=top_ver % LF_PINBOX_MAX_PINS)) + if (!(pins= top_ver % LF_PINBOX_MAX_PINS)) { - pins=my_atomic_add32(&pinbox->pins_in_stack, 1)+1; - el=(LF_PINS *)_lf_dynarray_lvalue(&pinbox->pinstack, pins); + pins= my_atomic_add32(&pinbox->pins_in_stack, 1)+1; + el= (LF_PINS *)_lf_dynarray_lvalue(&pinbox->pinstack, pins); break; } - el=(LF_PINS *)_lf_dynarray_value(&pinbox->pinstack, pins); - next=el->link; + el= (LF_PINS *)_lf_dynarray_value(&pinbox->pinstack, pins); + next= el->link; } while (!my_atomic_cas32(&pinbox->pinstack_top_ver, &top_ver, top_ver-pins+next+LF_PINBOX_MAX_PINS)); - el->link=pins; - el->purgatory_count=0; - el->pinbox=pinbox; + el->link= pins; + el->purgatory_count= 0; + el->pinbox= pinbox; return el; } void _lf_pinbox_put_pins(LF_PINS *pins) { - LF_PINBOX *pinbox=pins->pinbox; + LF_PINBOX *pinbox= pins->pinbox; uint32 top_ver, nr; - nr=pins->link; + nr= pins->link; #ifdef MY_LF_EXTRA_DEBUG { int i; - for (i=0; i < LF_PINBOX_PINS; i++) + for (i= 0; i < LF_PINBOX_PINS; i++) assert(pins->pin[i] == 0); } #endif + /* + Note - this will deadlock if other threads will wait for + the caller to do something after _lf_pinbox_put_pins(), + and they would have pinned addresses that the caller wants to free. + Thus: only free pins when all work is done and nobody can wait for you!!! + */ while (pins->purgatory_count) { _lf_pinbox_real_free(pins); - if (pins->purgatory_count && my_getncpus() == 1) + if (pins->purgatory_count) { my_atomic_rwlock_wrunlock(&pins->pinbox->pinstack.lock); pthread_yield(); my_atomic_rwlock_wrlock(&pins->pinbox->pinstack.lock); } } - top_ver=pinbox->pinstack_top_ver; + top_ver= pinbox->pinstack_top_ver; if (nr == pinbox->pins_in_stack) { - int32 tmp=nr; + int32 tmp= nr; if (my_atomic_cas32(&pinbox->pins_in_stack, &tmp, tmp-1)) goto ret; } do { - pins->link=top_ver % LF_PINBOX_MAX_PINS; + pins->link= top_ver % LF_PINBOX_MAX_PINS; } while (!my_atomic_cas32(&pinbox->pinstack_top_ver, &top_ver, top_ver-pins->link+nr+LF_PINBOX_MAX_PINS)); ret: @@ -127,19 +121,20 @@ static int ptr_cmp(void **a, void **b) return *a < *b ? -1 : *a == *b ? 0 : 1; } +#define add_to_purgatory(PINS, ADDR) \ + do \ + { \ + *(void **)((char *)(ADDR)+(PINS)->pinbox->free_ptr_offset)= \ + (PINS)->purgatory; \ + (PINS)->purgatory= (ADDR); \ + (PINS)->purgatory_count++; \ + } while (0) + void _lf_pinbox_free(LF_PINS *pins, void *addr) { - while (pins->purgatory_count == LF_PURGATORY_SIZE) - { + if (pins->purgatory_count % LF_PURGATORY_SIZE) _lf_pinbox_real_free(pins); - if (pins->purgatory_count == LF_PURGATORY_SIZE && my_getncpus() == 1) - { - my_atomic_rwlock_wrunlock(&pins->pinbox->pinstack.lock); - pthread_yield(); - my_atomic_rwlock_wrlock(&pins->pinbox->pinstack.lock); - } - } - pins->purgatory[pins->purgatory_count++]=addr; + add_to_purgatory(pins, addr); } struct st_harvester { @@ -178,11 +173,11 @@ static int match_pins(LF_PINS *el, void *addr) static void _lf_pinbox_real_free(LF_PINS *pins) { int npins; - void **addr=0; - void **start, **cur, **end=pins->purgatory+pins->purgatory_count; - LF_PINBOX *pinbox=pins->pinbox; + void *list; + void **addr; + LF_PINBOX *pinbox= pins->pinbox; - npins=pinbox->pins_in_stack+1; + npins= pinbox->pins_in_stack+1; #ifdef HAVE_ALLOCA /* create a sorted list of pinned addresses, to speed up searches */ @@ -190,64 +185,64 @@ static void _lf_pinbox_real_free(LF_PINS *pins) { struct st_harvester hv; addr= (void **) alloca(sizeof(void *)*LF_PINBOX_PINS*npins); - hv.granary=addr; - hv.npins=npins; + hv.granary= addr; + hv.npins= npins; _lf_dynarray_iterate(&pinbox->pinstack, (lf_dynarray_func)harvest_pins, &hv); - npins=hv.granary-addr; + npins= hv.granary-addr; if (npins) qsort(addr, npins, sizeof(void *), (qsort_cmp)ptr_cmp); } + else #endif + addr= 0; - start= cur= pins->purgatory; - end= start+pins->purgatory_count; - for (; cur < end; cur++) + list= pins->purgatory; + pins->purgatory= 0; + pins->purgatory_count= 0; + while (list) { + void *cur= list; + list= *(void **)((char *)cur+pinbox->free_ptr_offset); if (npins) { if (addr) { void **a,**b,**c; - for (a=addr, b=addr+npins-1, c=a+(b-a)/2; b-a>1; c=a+(b-a)/2) - if (*cur == *c) - a=b=c; - else if (*cur > *c) - a=c; + for (a= addr, b= addr+npins-1, c= a+(b-a)/2; b-a>1; c= a+(b-a)/2) + if (cur == *c) + a= b= c; + else if (cur > *c) + a= c; else - b=c; - if (*cur == *a || *cur == *b) + b= c; + if (cur == *a || cur == *b) goto found; } else { if (_lf_dynarray_iterate(&pinbox->pinstack, - (lf_dynarray_func)match_pins, *cur)) + (lf_dynarray_func)match_pins, cur)) goto found; } } /* not pinned - freeing */ - pinbox->free_func(*cur, pinbox->free_func_arg); + pinbox->free_func(cur, pinbox->free_func_arg); continue; found: /* pinned - keeping */ - *start++=*cur; + add_to_purgatory(pins, cur); } - pins->purgatory_count=start-pins->purgatory; -#ifdef MY_LF_EXTRA_DEBUG - while (start < pins->purgatory + LF_PURGATORY_SIZE) - *start++=0; -#endif } static void alloc_free(void *node, LF_ALLOCATOR *allocator) { void *tmp; - tmp=allocator->top; + tmp= allocator->top; do { - (*(void **)node)=tmp; + (*(void **)node)= tmp; } while (!my_atomic_casptr((void **)&allocator->top, (void **)&tmp, node) && LF_BACKOFF); } @@ -255,18 +250,18 @@ static void alloc_free(void *node, LF_ALLOCATOR *allocator) LF_REQUIRE_PINS(1); void *_lf_alloc_new(LF_PINS *pins) { - LF_ALLOCATOR *allocator=(LF_ALLOCATOR *)(pins->pinbox->free_func_arg); + LF_ALLOCATOR *allocator= (LF_ALLOCATOR *)(pins->pinbox->free_func_arg); void *node; for (;;) { do { - node=allocator->top; + node= allocator->top; _lf_pin(pins, 0, node); - } while (node !=allocator->top && LF_BACKOFF); + } while (node != allocator->top && LF_BACKOFF); if (!node) { - if (!(node=my_malloc(allocator->element_size, MYF(MY_WME|MY_ZEROFILL)))) + if (!(node= my_malloc(allocator->element_size, MYF(MY_WME|MY_ZEROFILL)))) goto ret; #ifdef MY_LF_EXTRA_DEBUG my_atomic_add32(&allocator->mallocs, 1); @@ -282,27 +277,27 @@ ret: return node; } -void lf_alloc_init(LF_ALLOCATOR *allocator, uint size) +void lf_alloc_init(LF_ALLOCATOR *allocator, uint size, uint free_ptr_offset) { - lf_pinbox_init(&allocator->pinbox, + lf_pinbox_init(&allocator->pinbox, free_ptr_offset, (lf_pinbox_free_func *)alloc_free, allocator); - allocator->top=0; - allocator->mallocs=0; - allocator->element_size=size; + allocator->top= 0; + allocator->mallocs= 0; + allocator->element_size= size; DBUG_ASSERT(size >= (int)sizeof(void *)); } void lf_alloc_destroy(LF_ALLOCATOR *allocator) { - void *el=allocator->top; + void *el= allocator->top; while (el) { - void *tmp=*(void **)el; + void *tmp= *(void **)el; my_free(el, MYF(0)); - el=tmp; + el= tmp; } lf_pinbox_destroy(&allocator->pinbox); - allocator->top=0; + allocator->top= 0; } /* @@ -313,7 +308,8 @@ uint lf_alloc_in_pool(LF_ALLOCATOR *allocator) { uint i; void *node; - for (node=allocator->top, i=0; node; node=*(void **)node, i++) /* no op */; + for (node= allocator->top, i= 0; node; node= *(void **)node, i++) + /* no op */; return i; } diff --git a/mysys/lf_hash.c b/mysys/lf_hash.c index a0425e89556..45b45f7531e 100644 --- a/mysys/lf_hash.c +++ b/mysys/lf_hash.c @@ -20,6 +20,7 @@ TODO try to get rid of dummy nodes ? for non-unique hash, count only _distinct_ values + (but how to do it in lf_hash_delete ?) */ #include #include @@ -222,7 +223,8 @@ void lf_hash_init(LF_HASH *hash, uint element_size, uint flags, uint key_offset, uint key_length, hash_get_key get_key, CHARSET_INFO *charset) { - lf_alloc_init(&hash->alloc,sizeof(LF_SLIST)+element_size); + lf_alloc_init(&hash->alloc, sizeof(LF_SLIST)+element_size, + offsetof(LF_SLIST, key)); lf_dynarray_init(&hash->array, sizeof(LF_SLIST **)); hash->size=1; hash->count=0; diff --git a/storage/maria/lockman.c b/storage/maria/lockman.c index e8ddbd1a25a..1712f6f2221 100644 --- a/storage/maria/lockman.c +++ b/storage/maria/lockman.c @@ -1,4 +1,4 @@ -// TODO lock escalation, instant duration locks +// TODO instant duration locks // automatically place S instead of LS if possible /* TODO optimization: table locks - they have completely @@ -21,6 +21,94 @@ along with this program; if not, write to the Free Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ +/* + Generic Lock Manager + + Lock manager handles locks on "resources", a resource must be uniquely + identified by a 64-bit number. Lock manager itself does not imply + anything about the nature of a resource - it can be a row, a table, a + database, or just anything. + + Locks belong to "lock owners". A Lock owner is uniquely identified by a + 16-bit number. A function loid2lo must be provided by the application + that takes such a number as an argument and returns a LOCK_OWNER + structure. + + Lock levels are completely defined by three tables. Lock compatibility + matrix specifies which locks can be held at the same time on a resource. + Lock combining matrix specifies what lock level has the same behaviour as + a pair of two locks of given levels. getlock_result matrix simplifies + intention locking and lock escalation for an application, basically it + defines which locks are intention locks and which locks are "loose" + locks. It is only used to provide better diagnostics for the + application, lock manager itself does not differentiate between normal, + intention, and loose locks. + + Internally lock manager is based on a lock-free hash, see lf_hash.c for + details. All locks are stored in a hash, with a resource id as a search + key, so all locks for the same resource will be considered collisions and + will be put in a one (lock-free) linked list. The main lock-handling + logic is in the inner loop that searches for a lock in such a linked + list - lockfind(). + + This works as follows. Locks generally are added to the end of the list + (with one exception, see below). When scanning the list it is always + possible to determine what locks are granted (active) and what locks are + waiting - first lock is obviously active, the second is active if it's + compatible with the first, and so on, a lock is active if it's compatible + with all previous locks and all locks before it are also active. + To calculate the "compatible with all previous locks" all locks are + accumulated in prev_lock variable using lock_combining_matrix. + + Lock upgrades: when a thread that has a lock on a given resource, + requests a new lock on the same resource and the old lock is not enough + to satisfy new lock requirements (which is defined by + lock_combining_matrix[old_lock][new_lock] != old_lock), a new lock is + placed in the list. Depending on other locks it is immediately active or + it will wait for other locks. Here's an exception to "locks are added + to the end" rule - upgraded locks are added after the last active lock + but before all waiting locks. Old lock (the one we upgraded from) is + not removed from the list, indeed we may need to return to it later if + the new lock was in a savepoint that gets rolled back. So old lock is + marked as "ignored" (IGNORE_ME flag). New lock gets an UPGRADED flag. + + Loose locks add an important exception to the above. Loose locks do not + always commute with other locks. In the list IX-LS both locks are active, + while in the LS-IX list only the first lock is active. This creates a + problem in lock upgrades. If the list was IX-LS and the owner of the + first lock wants to place LS lock (which can be immediately granted), the + IX lock is upgraded to LSIX and the list becomes IX-LS-LSIX, which, + according to the lock compatibility matrix means that the last lock is + waiting - of course it all happened because IX and LS were swapped and + they don't commute. To work around this there's ACTIVE flag which is set + in every lock that never waited (was placed active), and this flag + overrides "compatible with all previous locks" rule. + + When a lock is placed to the end of the list it's either compatible with + all locks and all locks are active - new lock becomes active at once, or + it conflicts with some of the locks, in this case in the 'blocker' + variable a conflicting lock is returned and the calling thread waits on a + pthread condition in the LOCK_OWNER structure of the owner of the + conflicting lock. Or a new lock is compatible with all locks, but some + existing locks are not compatible with previous locks (example: request IS, + when the list is S-IX) - that is not all locks are active. In this case a + first waiting lock is returned in the 'blocker' variable, + lockman_getlock() notices that a "blocker" does not conflict with the + requested lock, and "dereferences" it, to find the lock that it's waiting + on. The calling thread than begins to wait on the same lock. + + To better support table-row relations where one needs to lock the table + with an intention lock before locking the row, extended diagnostics is + provided. When an intention lock (presumably on a table) is granted, + lockman_getlock() returns one of GOT_THE_LOCK (no need to lock the row, + perhaps the thread already has a normal lock on this table), + GOT_THE_LOCK_NEED_TO_LOCK_A_SUBRESOURCE (need to lock the row, as usual), + GOT_THE_LOCK_NEED_TO_INSTANT_LOCK_A_SUBRESOURCE (only need to check + whether it's possible to lock the row, but no need to lock it - perhaps + the thread has a loose lock on this table). This is defined by + getlock_result[] table. +*/ + #include #include #include @@ -36,11 +124,6 @@ ') Though you can take LS lock while somebody has S lock, it makes no sense - it's simpler to take S lock too. - ") Strictly speaking you can take LX lock while somebody has S lock. - But in this case you lock no rows, because all rows are locked by this - somebody. So we prefer to define that LX cannot be taken when S - exists. Same about LX and X. - 1 - compatible 0 - incompatible -1 - "impossible", so that we can assert the impossibility. @@ -107,8 +190,8 @@ static enum lock_type lock_combining_matrix[10][10]= Defines below help to preserve the table structure. I/L/A values are self explanatory x means the combination is possible (assert should not crash) - but cannot happen in row locks, only in table locks (S,X), or - lock escalations (LS,LX) + but it cannot happen in row locks, only in table locks (S,X), + or lock escalations (LS,LX) */ #define I GOT_THE_LOCK_NEED_TO_LOCK_A_SUBRESOURCE #define L GOT_THE_LOCK_NEED_TO_INSTANT_LOCK_A_SUBRESOURCE @@ -138,8 +221,7 @@ typedef struct lockman_lock { uint64 resource; struct lockman_lock *lonext; intptr volatile link; - uint32 hashnr; -//#warning TODO - remove hashnr from LOCK + uint32 hashnr; // TODO - remove hashnr from LOCK uint16 loid; uchar lock; /* sizeof(uchar) <= sizeof(enum) */ uchar flags; @@ -147,6 +229,7 @@ typedef struct lockman_lock { #define IGNORE_ME 1 #define UPGRADED 2 +#define ACTIVE 4 typedef struct { intptr volatile *prev; @@ -171,7 +254,7 @@ static int lockfind(LOCK * volatile *head, LOCK *node, my_bool cur_active, compatible, upgrading, prev_active; enum lock_type lock, prev_lock, cur_lock; uint16 loid, cur_loid; - int upgraded_pairs, cur_flags, flags; + int cur_flags, flags; hashnr= node->hashnr; resource= node->resource; @@ -187,7 +270,6 @@ retry: upgrading= FALSE; cursor->blocker= cursor->upgrade_from= 0; _lf_unpin(pins, 3); - upgraded_pairs= 0; do { cursor->curr= PTR(*cursor->prev); _lf_pin(pins,1,cursor->curr); @@ -217,28 +299,24 @@ retry: (cur_hashnr == hashnr && cur_resource >= resource)) { if (cur_hashnr > hashnr || cur_resource > resource) - { - if (upgraded_pairs != 0) - goto retry; break; - } /* ok, we have a lock for this resource */ DBUG_ASSERT(lock_compatibility_matrix[prev_lock][cur_lock] >= 0); DBUG_ASSERT(lock_compatibility_matrix[cur_lock][lock] >= 0); - if (cur_flags & UPGRADED) - upgraded_pairs++; if ((cur_flags & IGNORE_ME) && ! (flags & IGNORE_ME)) { DBUG_ASSERT(cur_active); - upgraded_pairs--; if (cur_loid == loid) cursor->upgrade_from= cursor->curr; } else { prev_active= cur_active; - cur_active&= lock_compatibility_matrix[prev_lock][cur_lock]; - if (upgrading && !cur_active && upgraded_pairs == 0) + if (cur_flags & ACTIVE) + DBUG_ASSERT(prev_active == TRUE); + else + cur_active&= lock_compatibility_matrix[prev_lock][cur_lock]; + if (upgrading && !cur_active) break; if (prev_active && !cur_active) { @@ -253,8 +331,14 @@ retry: if (lock_combining_matrix[cur_lock][lock] == cur_lock) { /* new lock is compatible */ - return cur_active ? ALREADY_HAVE_THE_LOCK - : ALREADY_HAVE_THE_REQUEST; + if (cur_active) + { + cursor->blocker= cursor->curr; /* loose-locks! */ + _lf_unpin(pins, 3); /* loose-locks! */ + return ALREADY_HAVE_THE_LOCK; + } + else + return ALREADY_HAVE_THE_REQUEST; } /* not compatible, upgrading */ upgrading= TRUE; @@ -268,9 +352,9 @@ retry: cursor->blocker= cursor->curr; _lf_pin(pins, 3, cursor->curr); } - prev_lock= lock_combining_matrix[prev_lock][cur_lock]; - DBUG_ASSERT(prev_lock != N); } + prev_lock= lock_combining_matrix[prev_lock][cur_lock]; + DBUG_ASSERT(prev_lock != N); } } cursor->prev= &(cursor->curr->link); @@ -335,6 +419,10 @@ static int lockinsert(LOCK * volatile *head, LOCK *node, LF_PINS *pins, node->flags|= UPGRADED; node->lock= lock_combining_matrix[cursor.upgrade_from->lock][node->lock]; } + if (!(res & NEED_TO_WAIT)) + node->flags|= ACTIVE; + else + node->flags&= ~ACTIVE; /* if we're retrying on REPEAT_ONCE_MORE */ node->link= (intptr)cursor.curr; DBUG_ASSERT(node->link != (intptr)node); DBUG_ASSERT(cursor.prev != &node->link); @@ -349,13 +437,13 @@ static int lockinsert(LOCK * volatile *head, LOCK *node, LF_PINS *pins, _lf_unpin(pins, 1); _lf_unpin(pins, 2); /* - note that cursor.curr is NOT pinned on return. - this is ok as it's either a dummy node for initialize_bucket + note that blocker is not necessarily pinned here (when it's == curr). + this is ok as it's either a dummy node then for initialize_bucket and dummy nodes don't need pinning, or it's a lock of the same transaction for lockman_getlock, and it cannot be removed by another thread */ - *blocker= cursor.blocker ? cursor.blocker : cursor.curr; + *blocker= cursor.blocker; return res; } @@ -419,7 +507,7 @@ static int lockdelete(LOCK * volatile *head, LOCK *node, LF_PINS *pins) void lockman_init(LOCKMAN *lm, loid_to_lo_func *func, uint timeout) { - lf_alloc_init(&lm->alloc,sizeof(LOCK)); + lf_alloc_init(&lm->alloc,sizeof(LOCK), offsetof(LOCK,lonext)); lf_dynarray_init(&lm->array, sizeof(LOCK **)); lm->size= 1; lm->count= 0; @@ -516,13 +604,13 @@ enum lockman_getlock_result lockman_getlock(LOCKMAN *lm, LOCK_OWNER *lo, res= lockinsert(el, node, pins, &blocker); if (res & ALREADY_HAVE) { + int r; old_lock= blocker->lock; - _lf_assert_unpin(pins, 3); /* unpin should not be needed */ _lf_alloc_free(pins, node); lf_rwunlock_by_pins(pins); - res= getlock_result[old_lock][lock]; - DBUG_ASSERT(res); - return res; + r= getlock_result[old_lock][lock]; + DBUG_ASSERT(r); + return r; } /* a new value was added to the hash */ csize= lm->size; @@ -537,9 +625,8 @@ enum lockman_getlock_result lockman_getlock(LOCKMAN *lm, LOCK_OWNER *lo, struct timespec timeout; _lf_assert_pin(pins, 3); /* blocker must be pinned here */ - lf_rwunlock_by_pins(pins); - wait_for_lo= lm->loid_to_lo(blocker->loid); + /* now, this is tricky. blocker is not necessarily a LOCK we're waiting for. If it's compatible with what we want, @@ -550,12 +637,9 @@ enum lockman_getlock_result lockman_getlock(LOCKMAN *lm, LOCK_OWNER *lo, if (lock_compatibility_matrix[blocker->lock][lock]) { blocker= wait_for_lo->all_locks; - lf_pin(pins, 3, blocker); + _lf_pin(pins, 3, blocker); if (blocker != wait_for_lo->all_locks) - { - lf_rwlock_by_pins(pins); continue; - } wait_for_lo= wait_for_lo->waiting_for; } @@ -565,11 +649,11 @@ enum lockman_getlock_result lockman_getlock(LOCKMAN *lm, LOCK_OWNER *lo, to an unrelated - albeit valid - LOCK_OWNER */ if (!wait_for_lo) - { - /* blocker transaction has ended, short id was released */ - lf_rwlock_by_pins(pins); continue; - } + + lo->waiting_for= wait_for_lo; + lf_rwunlock_by_pins(pins); + /* We lock a mutex - it may belong to a wrong LOCK_OWNER, but it must belong to _some_ LOCK_OWNER. It means, we can never free() a LOCK_OWNER, @@ -587,9 +671,8 @@ enum lockman_getlock_result lockman_getlock(LOCKMAN *lm, LOCK_OWNER *lo, lf_rwlock_by_pins(pins); continue; } - /* yuck. waiting */ - lo->waiting_for= wait_for_lo; + /* yuck. waiting */ deadline= my_getsystime() + lm->lock_timeout * 10000; timeout.tv_sec= deadline/10000000; timeout.tv_nsec= (deadline % 10000000) * 100; @@ -607,11 +690,12 @@ enum lockman_getlock_result lockman_getlock(LOCKMAN *lm, LOCK_OWNER *lo, Instead we're relying on the caller to abort the transaction, and release all locks at once - see lockman_release_locks() */ + _lf_unpin(pins, 3); lf_rwunlock_by_pins(pins); return DIDNT_GET_THE_LOCK; } - lo->waiting_for= 0; } + lo->waiting_for= 0; _lf_assert_unpin(pins, 3); /* unpin should not be needed */ lf_rwunlock_by_pins(pins); return getlock_result[lock][lock]; @@ -626,14 +710,15 @@ enum lockman_getlock_result lockman_getlock(LOCKMAN *lm, LOCK_OWNER *lo, */ int lockman_release_locks(LOCKMAN *lm, LOCK_OWNER *lo) { - LOCK * volatile *el, *node; + LOCK * volatile *el, *node, *next; uint bucket; LF_PINS *pins= lo->pins; pthread_mutex_lock(lo->mutex); lf_rwlock_by_pins(pins); - for (node= lo->all_locks; node; node= node->lonext) + for (node= lo->all_locks; node; node= next) { + next= node->lonext; bucket= calc_hash(node->resource) % lm->size; el= _lf_dynarray_lvalue(&lm->array, bucket); if (*el == NULL) @@ -650,12 +735,12 @@ int lockman_release_locks(LOCKMAN *lm, LOCK_OWNER *lo) } #ifdef MY_LF_EXTRA_DEBUG +static char *lock2str[]= +{ "N", "S", "X", "IS", "IX", "SIX", "LS", "LX", "SLX", "LSIX" }; /* NOTE the function below is NOT thread-safe !!! */ -static char *lock2str[]= -{ "N", "S", "X", "IS", "IX", "SIX", "LS", "LX", "SLX", "LSIX" }; void print_lockhash(LOCKMAN *lm) { LOCK *el= *(LOCK **)_lf_dynarray_lvalue(&lm->array, 0); @@ -664,17 +749,21 @@ void print_lockhash(LOCKMAN *lm) { intptr next= el->link; if (el->hashnr & 1) + { printf("0x%08x { resource %llu, loid %u, lock %s", el->hashnr, el->resource, el->loid, lock2str[el->lock]); + if (el->flags & IGNORE_ME) printf(" IGNORE_ME"); + if (el->flags & UPGRADED) printf(" UPGRADED"); + if (el->flags & ACTIVE) printf(" ACTIVE"); + if (DELETED(next)) printf(" ***DELETED***"); + printf("}\n"); + } else { - printf("0x%08x { dummy ", el->hashnr); + /*printf("0x%08x { dummy }\n", el->hashnr);*/ DBUG_ASSERT(el->resource == 0 && el->loid == 0 && el->lock == X); } - if (el->flags & IGNORE_ME) printf(" IGNORE_ME"); - if (el->flags & UPGRADED) printf(" UPGRADED"); - printf("}\n"); - el= (LOCK *)next; + el= PTR(next); } } #endif diff --git a/storage/maria/lockman.h b/storage/maria/lockman.h index a3c96786935..9edd79eb7f1 100644 --- a/storage/maria/lockman.h +++ b/storage/maria/lockman.h @@ -18,7 +18,10 @@ #define _lockman_h /* - N - "no lock", not a lock, used sometimes to simplify the code + Lock levels: + ^^^^^^^^^^^ + + N - "no lock", not a lock, used sometimes internally to simplify the code S - Shared X - eXclusive IS - Intention Shared @@ -35,8 +38,8 @@ struct lockman_lock; typedef struct st_lock_owner LOCK_OWNER; struct st_lock_owner { - LF_PINS *pins; - struct lockman_lock *all_locks; + LF_PINS *pins; /* must be allocated from lockman's pinbox */ + struct lockman_lock *all_locks; /* a LIFO */ LOCK_OWNER *waiting_for; pthread_cond_t *cond; /* transactions waiting for this, wait on 'cond' */ pthread_mutex_t *mutex; /* mutex is required to use 'cond' */ diff --git a/storage/maria/trnman.c b/storage/maria/trnman.c index 49f49a3e26b..ecabf100cb8 100644 --- a/storage/maria/trnman.c +++ b/storage/maria/trnman.c @@ -128,6 +128,11 @@ static void set_short_trid(TRN *trn) trn->locks.loid= i; } +/* + DESCRIPTION + start a new transaction, allocate and initialize transaction object + mutex and cond will be used for lock waits +*/ TRN *trnman_new_trn(pthread_mutex_t *mutex, pthread_cond_t *cond) { TRN *trn; @@ -148,6 +153,7 @@ TRN *trnman_new_trn(pthread_mutex_t *mutex, pthread_cond_t *cond) trnman_active_transactions++; trn= pool; + /* popping an element from a stack */ my_atomic_rwlock_wrlock(&LOCK_pool); while (trn && !my_atomic_casptr((void **)&pool, (void **)&trn, (void *)trn->next)) @@ -213,9 +219,12 @@ void trnman_end_trn(TRN *trn, my_bool commit) LF_PINS *pins= trn->pins; pthread_mutex_lock(&LOCK_trn_list); + + /* remove from active list */ trn->next->prev= trn->prev; trn->prev->next= trn->next; + /* if this transaction was the oldest - clean up committed list */ if (trn->prev == &active_list_min) { TRN *t; @@ -232,6 +241,7 @@ void trnman_end_trn(TRN *trn, my_bool commit) } } + /* add transaction to the committed list (for read-from relations) */ if (commit && active_list_min.next != &active_list_max) { trn->commit_trid= global_trid_generator; @@ -243,7 +253,7 @@ void trnman_end_trn(TRN *trn, my_bool commit) res= lf_hash_insert(&trid_to_trn, pins, &trn); DBUG_ASSERT(res == 0); } - else + else /* or free it right away */ { trn->next= free_me; free_me= trn; @@ -251,6 +261,7 @@ void trnman_end_trn(TRN *trn, my_bool commit) trnman_active_transactions--; pthread_mutex_unlock(&LOCK_trn_list); + /* the rest is done outside of a critical section */ lockman_release_locks(&maria_lockman, &trn->locks); trn->locks.mutex= 0; trn->locks.cond= 0; @@ -258,7 +269,6 @@ void trnman_end_trn(TRN *trn, my_bool commit) my_atomic_storeptr((void **)&short_trid_to_trn[trn->locks.loid], 0); my_atomic_rwlock_rdunlock(&LOCK_short_trid_to_trn); - while (free_me) // XXX send them to the purge thread { int res; @@ -288,7 +298,7 @@ void trnman_free_trn(TRN *trn) do { /* - without volatile cast gcc-3.4.4 moved the assignment + without this volatile cast gcc-3.4.4 moved the assignment down after the loop at -O2 */ *(TRN * volatile *)&(trn->next)= tmp; @@ -317,13 +327,13 @@ my_bool trnman_can_read_from(TRN *trn, TrID trid) LF_REQUIRE_PINS(3); if (trid < trn->min_read_from) - return TRUE; + return TRUE; /* can read */ if (trid > trn->trid) - return FALSE; + return FALSE; /* cannot read */ found= lf_hash_search(&trid_to_trn, trn->pins, &trid, sizeof(trid)); if (!found) - return FALSE; /* not in the hash of committed transactions = cannot read*/ + return FALSE; /* not in the hash of committed transactions = cannot read */ can= (*found)->commit_trid < trn->trid; lf_unpin(trn->pins, 2); diff --git a/storage/maria/unittest/lockman-t.c b/storage/maria/unittest/lockman-t.c index 8b62ccfe094..638078fea65 100644 --- a/storage/maria/unittest/lockman-t.c +++ b/storage/maria/unittest/lockman-t.c @@ -14,7 +14,7 @@ along with this program; if not, write to the Free Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ -//#define EXTRA_VERBOSE +#undef EXTRA_VERBOSE #include @@ -24,7 +24,7 @@ #include #include "../lockman.h" -#define Nlos 10 +#define Nlos 100 LOCK_OWNER loarray[Nlos]; pthread_mutex_t mutexes[Nlos]; pthread_cond_t conds[Nlos]; @@ -51,8 +51,7 @@ LOCK_OWNER *loid2lo(uint16 loid) #define lock_ok_a(O,R,L) test_lock(O,R,L,"",GOT_THE_LOCK) #define lock_ok_i(O,R,L) test_lock(O,R,L,"",GOT_THE_LOCK_NEED_TO_LOCK_A_SUBRESOURCE) #define lock_ok_l(O,R,L) test_lock(O,R,L,"",GOT_THE_LOCK_NEED_TO_INSTANT_LOCK_A_SUBRESOURCE) -#define lock_conflict(O,R,L) test_lock(O,R,L,"cannot ",DIDNT_GET_THE_LOCK); \ - unlock_all(O) +#define lock_conflict(O,R,L) test_lock(O,R,L,"cannot ",DIDNT_GET_THE_LOCK); void test_lockman_simple() { @@ -64,7 +63,8 @@ void test_lockman_simple() lock_ok_a(1, 1, X); lock_ok_i(2, 2, IX); /* failures */ - lock_conflict(2,1,X); /* this removes all locks of lo2 */ + lock_conflict(2,1,X); + unlock_all(2); lock_ok_a(1,2,S); lock_ok_a(1,2,IS); lock_ok_a(1,2,LS); @@ -72,8 +72,36 @@ void test_lockman_simple() lock_ok_a(2,3,LS); lock_ok_i(1,3,IX); lock_ok_l(2,3,IS); - lockman_release_locks(&lockman, loid2lo(1)); - lockman_release_locks(&lockman, loid2lo(2)); + unlock_all(1); + unlock_all(2); + + lock_ok_i(1,1,IX); + lock_conflict(2,1,S); + lock_ok_a(1,1,LS); + unlock_all(1); + unlock_all(2); + + lock_ok_i(1,1,IX); + lock_ok_a(2,1,LS); + lock_ok_a(1,1,LS); + lock_ok_i(1,1,IX); + lock_ok_i(3,1,IS); + unlock_all(1); + unlock_all(2); + unlock_all(3); + + lock_ok_i(1,4,IS); + lock_ok_i(2,4,IS); + lock_ok_i(3,4,IS); + lock_ok_a(3,4,LS); + lock_ok_i(4,4,IS); + lock_conflict(4,4,IX); + lock_conflict(2,4,IX); + lock_ok_a(1,4,LS); + unlock_all(1); + unlock_all(2); + unlock_all(3); + unlock_all(4); } @@ -82,11 +110,13 @@ pthread_mutex_t rt_mutex; pthread_cond_t rt_cond; int rt_num_threads; int litmus; +int thread_number= 0, timeouts=0; void run_test(const char *test, pthread_handler handler, int n, int m) { pthread_t t; ulonglong now= my_getsystime(); + thread_number= timeouts= 0; litmus= 0; diag("Testing %s with %d threads, %d iterations... ", test, n, m); @@ -100,13 +130,12 @@ void run_test(const char *test, pthread_handler handler, int n, int m) ok(litmus == 0, "tested %s in %g secs (%d)", test, ((double)now)/1e7, litmus); } -int thread_number= 0, timeouts=0; -#define Nrows 1000 -#define Ntables 10 -#define TABLE_LOCK_RATIO 10 +int Nrows= 100; +int Ntables= 10; +int table_lock_ratio= 10; enum lock_type lock_array[6]={S,X,LS,LX,IS,IX}; char *lock2str[6]={"S","X","LS","LX","IS","IX"}; -char *res2str[6]={ +char *res2str[4]={ "DIDN'T GET THE LOCK", "GOT THE LOCK", "GOT THE LOCK NEED TO LOCK A SUBRESOURCE", @@ -128,10 +157,11 @@ pthread_handler_t test_lockman(void *arg) row= x % Nrows + Ntables; table= row % Ntables; locklevel= (x/Nrows) & 3; - if ((x/Nrows/4) % TABLE_LOCK_RATIO == 0) + if (table_lock_ratio && (x/Nrows/4) % table_lock_ratio == 0) { /* table lock */ res= lockman_getlock(&lockman, lo, table, lock_array[locklevel]); - DIAG(("loid=%2d, table %d lock %s, res=%s", loid, table, lock2str[locklevel], res2str[res])); + DIAG(("loid=%2d, table %d lock %s, res=%s", loid, table, + lock2str[locklevel], res2str[res])); if (res == DIDNT_GET_THE_LOCK) { lockman_release_locks(&lockman, lo); @@ -145,7 +175,8 @@ pthread_handler_t test_lockman(void *arg) { /* row lock */ locklevel&= 1; res= lockman_getlock(&lockman, lo, table, lock_array[locklevel + 4]); - DIAG(("loid=%2d, row %d lock %s, res=%s", loid, row, lock2str[locklevel+4], res2str[res])); + DIAG(("loid=%2d, row %d lock %s, res=%s", loid, row, + lock2str[locklevel+4], res2str[res])); switch (res) { case DIDNT_GET_THE_LOCK: @@ -159,7 +190,8 @@ pthread_handler_t test_lockman(void *arg) /* not implemented, so take a regular lock */ case GOT_THE_LOCK_NEED_TO_LOCK_A_SUBRESOURCE: res= lockman_getlock(&lockman, lo, row, lock_array[locklevel]); - DIAG(("loid=%2d, ROW %d lock %s, res=%s", loid, row, lock2str[locklevel], res2str[res])); + DIAG(("loid=%2d, ROW %d lock %s, res=%s", loid, row, + lock2str[locklevel], res2str[res])); if (res == DIDNT_GET_THE_LOCK) { lockman_release_locks(&lockman, lo); @@ -196,7 +228,7 @@ int main() my_init(); - plan(14); + plan(31); if (my_atomic_initialize()) return exit_status(); @@ -222,11 +254,21 @@ int main() test_lockman_simple(); -#define CYCLES 100 +#define CYCLES 1000 #define THREADS Nlos /* don't change this line */ + /* mixed load, stress-test with random locks */ + Nrows= 100; + Ntables= 10; + table_lock_ratio= 10; run_test("lockman", test_lockman, THREADS,CYCLES); + /* "real-life" simulation - many rows, no table locks */ + Nrows= 1000000; + Ntables= 10; + table_lock_ratio= 0; + run_test("lockman", test_lockman, THREADS,10000); + for (i= 0; i < Nlos; i++) { lockman_release_locks(&lockman, &loarray[i]); @@ -235,7 +277,12 @@ int main() lf_pinbox_put_pins(loarray[i].pins); } - lockman_destroy(&lockman); + { + ulonglong now= my_getsystime(); + lockman_destroy(&lockman); + now= my_getsystime()-now; + diag("lockman_destroy: %g", ((double)now)/1e7); + } pthread_mutex_destroy(&rt_mutex); pthread_cond_destroy(&rt_cond); diff --git a/unittest/mysys/my_atomic-t.c b/unittest/mysys/my_atomic-t.c index 881f80a66f8..e2177b94720 100644 --- a/unittest/mysys/my_atomic-t.c +++ b/unittest/mysys/my_atomic-t.c @@ -154,7 +154,7 @@ pthread_handler_t test_lf_pinbox(void *arg) typedef union { int32 data; - void *not_used; /* to guarantee sizeof(TLA) >= sizeof(void *) */ + void *not_used; } TLA; pthread_handler_t test_lf_alloc(void *arg) @@ -294,7 +294,7 @@ int main() pthread_mutex_init(&mutex, 0); pthread_cond_init(&cond, 0); my_atomic_rwlock_init(&rwl); - lf_alloc_init(&lf_allocator, sizeof(TLA)); + lf_alloc_init(&lf_allocator, sizeof(TLA), offsetof(TLA, not_used)); lf_hash_init(&lf_hash, sizeof(int), LF_HASH_UNIQUE, 0, sizeof(int), 0, &my_charset_bin); -- cgit v1.2.1 From aea73116c14da8e946422de8e49c93acc85817d0 Mon Sep 17 00:00:00 2001 From: unknown Date: Thu, 19 Oct 2006 12:21:30 +0200 Subject: post-review fixes (style) include/lf.h: comments --- include/lf.h | 43 ++++++----- storage/maria/lockman.c | 8 +- storage/maria/lockman.h | 2 +- storage/maria/trnman.c | 147 +++++++++++++++++++++++-------------- storage/maria/trnman.h | 2 +- storage/maria/unittest/lockman-t.c | 97 ++++++++++++------------ storage/maria/unittest/trnman-t.c | 31 ++++---- 7 files changed, 186 insertions(+), 144 deletions(-) diff --git a/include/lf.h b/include/lf.h index 45c5f109e1c..3a3b5b28ade 100644 --- a/include/lf.h +++ b/include/lf.h @@ -1,9 +1,18 @@ +/* Copyright (C) 2006 MySQL AB -/* - TODO - 1. copyright - 6. reduce the number of memory barriers -*/ + 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; either version 2 of the License, or + (at your option) any later version. + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ #ifndef _lf_h #define _lf_h @@ -11,7 +20,8 @@ #include /* - Generic helpers + Helpers to define both func() and _func(), where + func() is a _func() protected by my_atomic_rwlock_wrlock() */ #define lock_wrap(f,t,proto_args, args, lock) \ @@ -49,7 +59,7 @@ static inline void f proto_args \ } /* - dynamic array + wait-free dynamic array, see lf_dynarray.c 4 levels of 256 elements each mean 4311810304 elements in an array - it should be enough for a while @@ -68,14 +78,9 @@ typedef int (*lf_dynarray_func)(void *, void *); void lf_dynarray_init(LF_DYNARRAY *array, uint element_size); void lf_dynarray_destroy(LF_DYNARRAY *array); -nolock_wrap(lf_dynarray_nr, int, - (LF_DYNARRAY *array, void *el), - (array,el)); - nolock_wrap(lf_dynarray_value, void *, (LF_DYNARRAY *array, uint idx), (array,idx)); - lock_wrap(lf_dynarray_lvalue, void *, (LF_DYNARRAY *array, uint idx), (array,idx), @@ -85,7 +90,7 @@ nolock_wrap(lf_dynarray_iterate, int, (array,func,arg)); /* - pin manager for memory allocator + pin manager for memory allocator, lf_alloc-pin.c */ #define LF_PINBOX_PINS 4 @@ -102,13 +107,13 @@ typedef struct { uint32 volatile pins_in_stack; /* number of elements in array */ } LF_PINBOX; -/* we want sizeof(LF_PINS) to be 128 to avoid false sharing */ typedef struct { void * volatile pin[LF_PINBOX_PINS]; LF_PINBOX *pinbox; void *purgatory; uint32 purgatory_count; uint32 volatile link; +/* we want sizeof(LF_PINS) to be 128 to avoid false sharing */ char pad[128-sizeof(uint32)*2 -sizeof(void *)*(LF_PINBOX_PINS+2)]; } LF_PINS; @@ -160,19 +165,13 @@ lock_wrap_void(lf_pinbox_put_pins, (LF_PINS *pins), (pins), &pins->pinbox->pinstack.lock); -#if 0 -lock_wrap_void(lf_pinbox_real_free, - (LF_PINS *pins), - (pins), - &pins->pinbox->pinstack.lock); -#endif lock_wrap_void(lf_pinbox_free, (LF_PINS *pins, void *addr), (pins,addr), &pins->pinbox->pinstack.lock); /* - memory allocator + memory allocator, lf_alloc-pin.c */ typedef struct st_lf_allocator { @@ -199,7 +198,7 @@ lock_wrap(lf_alloc_new, void *, &pins->pinbox->pinstack.lock); /* - extendible hash + extendible hash, lf_hash.c */ #include diff --git a/storage/maria/lockman.c b/storage/maria/lockman.c index 1712f6f2221..b37c31fb52c 100644 --- a/storage/maria/lockman.c +++ b/storage/maria/lockman.c @@ -5,7 +5,7 @@ different characteristics. long lists, few distinct resources - slow to scan, [possibly] high retry rate */ -/* Copyright (C) 2000 MySQL AB +/* Copyright (C) 2006 MySQL AB 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 @@ -272,7 +272,7 @@ retry: _lf_unpin(pins, 3); do { cursor->curr= PTR(*cursor->prev); - _lf_pin(pins,1,cursor->curr); + _lf_pin(pins, 1, cursor->curr); } while(*cursor->prev != (intptr)cursor->curr && LF_BACKOFF); for (;;) { @@ -507,7 +507,7 @@ static int lockdelete(LOCK * volatile *head, LOCK *node, LF_PINS *pins) void lockman_init(LOCKMAN *lm, loid_to_lo_func *func, uint timeout) { - lf_alloc_init(&lm->alloc,sizeof(LOCK), offsetof(LOCK,lonext)); + lf_alloc_init(&lm->alloc, sizeof(LOCK), offsetof(LOCK, lonext)); lf_dynarray_init(&lm->array, sizeof(LOCK **)); lm->size= 1; lm->count= 0; @@ -744,7 +744,7 @@ static char *lock2str[]= void print_lockhash(LOCKMAN *lm) { LOCK *el= *(LOCK **)_lf_dynarray_lvalue(&lm->array, 0); - printf("hash: size=%u count=%u\n", lm->size, lm->count); + printf("hash: size:%u count:%u\n", lm->size, lm->count); while (el) { intptr next= el->link; diff --git a/storage/maria/lockman.h b/storage/maria/lockman.h index 9edd79eb7f1..6577a5e80fc 100644 --- a/storage/maria/lockman.h +++ b/storage/maria/lockman.h @@ -1,4 +1,4 @@ -/* Copyright (C) 2000 MySQL AB +/* Copyright (C) 2006 MySQL AB 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 diff --git a/storage/maria/trnman.c b/storage/maria/trnman.c index ecabf100cb8..95e5be4284a 100644 --- a/storage/maria/trnman.c +++ b/storage/maria/trnman.c @@ -1,4 +1,4 @@ -/* Copyright (C) 2000 MySQL AB +/* Copyright (C) 2006 MySQL AB 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 @@ -20,21 +20,35 @@ #include #include "trnman.h" +/* status variables */ uint trnman_active_transactions, trnman_allocated_transactions; -static TRN active_list_min, active_list_max, - committed_list_min, committed_list_max, *pool; +/* list of active transactions in the trid order */ +static TRN active_list_min, active_list_max; +/* list of committed transactions in the trid order */ +static TRN committed_list_min, committed_list_max; -static pthread_mutex_t LOCK_trn_list; +/* a counter, used to generate transaction ids */ static TrID global_trid_generator; -static LF_HASH trid_to_trn; -static LOCKMAN maria_lockman; +/* the mutex for everything above */ +static pthread_mutex_t LOCK_trn_list; + +/* LIFO pool of unused TRN structured for reuse */ +static TRN *pool; + +/* a hash for committed transactions that maps trid to a TRN structure */ +static LF_HASH trid_to_committed_trn; -static TRN **short_trid_to_trn; +/* an array that maps short_trid of an active transaction to a TRN structure */ +static TRN **short_trid_to_active_trn; + +/* locks for short_trid_to_active_trn and pool */ static my_atomic_rwlock_t LOCK_short_trid_to_trn, LOCK_pool; -static byte *trn_get_hash_key(const byte *trn,uint* len, my_bool unused) +static LOCKMAN maria_lockman; + +static byte *trn_get_hash_key(const byte *trn, uint* len, my_bool unused) { *len= sizeof(TrID); return (byte *) & ((*((TRN **)trn))->trid); @@ -44,7 +58,7 @@ static LOCK_OWNER *trnman_short_trid_to_TRN(uint16 short_trid) { TRN *trn; my_atomic_rwlock_rdlock(&LOCK_short_trid_to_trn); - trn= my_atomic_loadptr((void **)&short_trid_to_trn[short_trid]); + trn= my_atomic_loadptr((void **)&short_trid_to_active_trn[short_trid]); my_atomic_rwlock_rdunlock(&LOCK_short_trid_to_trn); return (LOCK_OWNER *)trn; } @@ -52,39 +66,56 @@ static LOCK_OWNER *trnman_short_trid_to_TRN(uint16 short_trid) int trnman_init() { pthread_mutex_init(&LOCK_trn_list, MY_MUTEX_INIT_FAST); + + /* + Initialize lists. + active_list_max.min_read_from must be larger than any trid, + so that when an active list is empty we would could free + all committed list. + And committed_list_max itself can not be freed so + committed_list_max.commit_trid must not be smaller that + active_list_max.min_read_from + */ + active_list_max.trid= active_list_min.trid= 0; active_list_max.min_read_from= ~0; active_list_max.next= active_list_min.prev= 0; active_list_max.prev= &active_list_min; active_list_min.next= &active_list_max; - trnman_active_transactions= 0; - trnman_allocated_transactions= 0; committed_list_max.commit_trid= ~0; committed_list_max.next= committed_list_min.prev= 0; committed_list_max.prev= &committed_list_min; committed_list_min.next= &committed_list_max; + trnman_active_transactions= 0; + trnman_allocated_transactions= 0; + pool= 0; - global_trid_generator= 0; /* set later by recovery code */ - lf_hash_init(&trid_to_trn, sizeof(TRN*), LF_HASH_UNIQUE, + global_trid_generator= 0; /* set later by the recovery code */ + lf_hash_init(&trid_to_committed_trn, sizeof(TRN*), LF_HASH_UNIQUE, 0, 0, trn_get_hash_key, 0); my_atomic_rwlock_init(&LOCK_short_trid_to_trn); my_atomic_rwlock_init(&LOCK_pool); - short_trid_to_trn= (TRN **)my_malloc(SHORT_TRID_MAX*sizeof(TRN*), + short_trid_to_active_trn= (TRN **)my_malloc(SHORT_TRID_MAX*sizeof(TRN*), MYF(MY_WME|MY_ZEROFILL)); - if (!short_trid_to_trn) + if (!short_trid_to_active_trn) return 1; - short_trid_to_trn--; /* min short_trid is 1 */ + short_trid_to_active_trn--; /* min short_trid is 1 */ lockman_init(&maria_lockman, &trnman_short_trid_to_TRN, 10000); return 0; } +/* + NOTE + this could only be called in the "idle" state - no transaction can be + running. See asserts below. +*/ int trnman_destroy() { - DBUG_ASSERT(trid_to_trn.count == 0); + DBUG_ASSERT(trid_to_committed_trn.count == 0); DBUG_ASSERT(trnman_active_transactions == 0); DBUG_ASSERT(active_list_max.prev == &active_list_min); DBUG_ASSERT(active_list_min.next == &active_list_max); @@ -98,14 +129,20 @@ int trnman_destroy() DBUG_ASSERT(trn->locks.cond == 0); my_free((void *)trn, MYF(0)); } - lf_hash_destroy(&trid_to_trn); + lf_hash_destroy(&trid_to_committed_trn); pthread_mutex_destroy(&LOCK_trn_list); my_atomic_rwlock_destroy(&LOCK_short_trid_to_trn); my_atomic_rwlock_destroy(&LOCK_pool); - my_free((void *)(short_trid_to_trn+1), MYF(0)); + my_free((void *)(short_trid_to_active_trn+1), MYF(0)); lockman_destroy(&maria_lockman); } +/* + NOTE + TrID is limited to 6 bytes. Initial value of the generator + is set by the recovery code - being read from the last checkpoint + (or 1 on a first run). +*/ static TrID new_trid() { DBUG_ASSERT(global_trid_generator < 0xffffffffffffLL); @@ -120,8 +157,8 @@ static void set_short_trid(TRN *trn) for ( ; ; i= i % SHORT_TRID_MAX + 1) /* the range is [1..SHORT_TRID_MAX] */ { void *tmp= NULL; - if (short_trid_to_trn[i] == NULL && - my_atomic_casptr((void **)&short_trid_to_trn[i], &tmp, trn)) + if (short_trid_to_active_trn[i] == NULL && + my_atomic_casptr((void **)&short_trid_to_active_trn[i], &tmp, trn)) break; } my_atomic_rwlock_wrunlock(&LOCK_short_trid_to_trn); @@ -138,38 +175,37 @@ TRN *trnman_new_trn(pthread_mutex_t *mutex, pthread_cond_t *cond) TRN *trn; /* - see trnman_end_trn to see why we need a mutex here - - and as we have a mutex, we can as well do everything - under it - allocating a TRN, incrementing trnman_active_transactions, - setting trn->min_read_from. + we have a mutex, to do simple things under it - allocate a TRN, + increment trnman_active_transactions, set trn->min_read_from. Note that all the above is fast. generating short_trid may be slow, - as it involves scanning a big array - so it's still done - outside of the mutex. + as it involves scanning a large array - so it's done outside of the + mutex. */ pthread_mutex_lock(&LOCK_trn_list); - trnman_active_transactions++; + /* Allocating a new TRN structure */ trn= pool; - /* popping an element from a stack */ + /* Popping an unused TRN from the pool */ my_atomic_rwlock_wrlock(&LOCK_pool); while (trn && !my_atomic_casptr((void **)&pool, (void **)&trn, (void *)trn->next)) /* no-op */; my_atomic_rwlock_wrunlock(&LOCK_pool); + /* Nothing in the pool ? Allocate a new one */ if (!trn) { trn= (TRN *)my_malloc(sizeof(TRN), MYF(MY_WME)); - if (!trn) + if (unlikely(!trn)) { pthread_mutex_unlock(&LOCK_trn_list); return 0; } trnman_allocated_transactions++; } + trnman_active_transactions++; trn->min_read_from= active_list_min.next->trid; @@ -181,36 +217,31 @@ TRN *trnman_new_trn(pthread_mutex_t *mutex, pthread_cond_t *cond) active_list_max.prev= trn->prev->next= trn; pthread_mutex_unlock(&LOCK_trn_list); - trn->pins= lf_hash_get_pins(&trid_to_trn); + trn->pins= lf_hash_get_pins(&trid_to_committed_trn); - if (!trn->min_read_from) + if (unlikely(!trn->min_read_from)) trn->min_read_from= trn->trid; + trn->commit_trid= 0; + trn->locks.mutex= mutex; trn->locks.cond= cond; - trn->commit_trid= 0; trn->locks.waiting_for= 0; trn->locks.all_locks= 0; trn->locks.pins= lf_alloc_get_pins(&maria_lockman.alloc); - set_short_trid(trn); /* this must be the last! */ + /* + only after the following function TRN is considered initialized, + so it must be done the last + */ + set_short_trid(trn); return trn; } /* - remove a trn from the active list, - move to committed list, - set commit_trid - - TODO - integrate with log manager. That means: - a common "commit" mutex - forcing the log and setting commit_trid - must be done atomically (QQ how the heck it could be done with - group commit ???) XXX - why did I think it must be done atomically ? - - trid_to_trn, active_list_*, and committed_list_* can be - updated asyncronously. + remove a trn from the active list. + if necessary - move to committed list and set commit_trid */ void trnman_end_trn(TRN *trn, my_bool commit) { @@ -224,7 +255,11 @@ void trnman_end_trn(TRN *trn, my_bool commit) trn->next->prev= trn->prev; trn->prev->next= trn->next; - /* if this transaction was the oldest - clean up committed list */ + /* + if trn was the oldest active transaction, now that it goes away there + may be committed transactions in the list which no active transaction + needs to bother about - clean up the committed list + */ if (trn->prev == &active_list_min) { TRN *t; @@ -232,6 +267,7 @@ void trnman_end_trn(TRN *trn, my_bool commit) t->commit_trid < active_list_min.next->min_read_from; t= t->next) /* no-op */; + /* found transactions committed before the oldest active one */ if (t != committed_list_min.next) { free_me= committed_list_min.next; @@ -241,7 +277,10 @@ void trnman_end_trn(TRN *trn, my_bool commit) } } - /* add transaction to the committed list (for read-from relations) */ + /* + if transaction is committed and it was not the only active transaction - + add it to the committed list (which is used for read-from relation) + */ if (commit && active_list_min.next != &active_list_max) { trn->commit_trid= global_trid_generator; @@ -250,10 +289,10 @@ void trnman_end_trn(TRN *trn, my_bool commit) trn->prev= committed_list_max.prev; committed_list_max.prev= trn->prev->next= trn; - res= lf_hash_insert(&trid_to_trn, pins, &trn); + res= lf_hash_insert(&trid_to_committed_trn, pins, &trn); DBUG_ASSERT(res == 0); } - else /* or free it right away */ + else /* otherwise free it right away */ { trn->next= free_me; free_me= trn; @@ -266,7 +305,7 @@ void trnman_end_trn(TRN *trn, my_bool commit) trn->locks.mutex= 0; trn->locks.cond= 0; my_atomic_rwlock_rdlock(&LOCK_short_trid_to_trn); - my_atomic_storeptr((void **)&short_trid_to_trn[trn->locks.loid], 0); + my_atomic_storeptr((void **)&short_trid_to_active_trn[trn->locks.loid], 0); my_atomic_rwlock_rdunlock(&LOCK_short_trid_to_trn); while (free_me) // XXX send them to the purge thread @@ -275,7 +314,7 @@ void trnman_end_trn(TRN *trn, my_bool commit) TRN *t= free_me; free_me= free_me->next; - res= lf_hash_delete(&trid_to_trn, pins, &t->trid, sizeof(TrID)); + res= lf_hash_delete(&trid_to_committed_trn, pins, &t->trid, sizeof(TrID)); trnman_free_trn(t); } @@ -331,7 +370,7 @@ my_bool trnman_can_read_from(TRN *trn, TrID trid) if (trid > trn->trid) return FALSE; /* cannot read */ - found= lf_hash_search(&trid_to_trn, trn->pins, &trid, sizeof(trid)); + found= lf_hash_search(&trid_to_committed_trn, trn->pins, &trid, sizeof(trid)); if (!found) return FALSE; /* not in the hash of committed transactions = cannot read */ diff --git a/storage/maria/trnman.h b/storage/maria/trnman.h index 9470678f3b2..c059947f35c 100644 --- a/storage/maria/trnman.h +++ b/storage/maria/trnman.h @@ -1,4 +1,4 @@ -/* Copyright (C) 2000 MySQL AB +/* Copyright (C) 2006 MySQL AB 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 diff --git a/storage/maria/unittest/lockman-t.c b/storage/maria/unittest/lockman-t.c index 638078fea65..ad382a360fb 100644 --- a/storage/maria/unittest/lockman-t.c +++ b/storage/maria/unittest/lockman-t.c @@ -42,16 +42,20 @@ LOCK_OWNER *loid2lo(uint16 loid) return loarray+loid-1; } -#define unlock_all(O) diag("lo" #O "> release all locks"); \ +#define unlock_all(O) diag("lo" #O "> release all locks"); \ lockman_release_locks(&lockman, loid2lo(O));print_lockhash(&lockman) -#define test_lock(O, R, L, S, RES) \ - ok(lockman_getlock(&lockman, loid2lo(O), R, L) == RES, \ - "lo" #O "> " S " lock resource " #R " with " #L "-lock"); \ +#define test_lock(O, R, L, S, RES) \ + ok(lockman_getlock(&lockman, loid2lo(O), R, L) == RES, \ + "lo" #O "> " S " lock resource " #R " with " #L "-lock"); \ print_lockhash(&lockman) -#define lock_ok_a(O,R,L) test_lock(O,R,L,"",GOT_THE_LOCK) -#define lock_ok_i(O,R,L) test_lock(O,R,L,"",GOT_THE_LOCK_NEED_TO_LOCK_A_SUBRESOURCE) -#define lock_ok_l(O,R,L) test_lock(O,R,L,"",GOT_THE_LOCK_NEED_TO_INSTANT_LOCK_A_SUBRESOURCE) -#define lock_conflict(O,R,L) test_lock(O,R,L,"cannot ",DIDNT_GET_THE_LOCK); +#define lock_ok_a(O, R, L) \ + test_lock(O, R, L, "", GOT_THE_LOCK) +#define lock_ok_i(O, R, L) \ + test_lock(O, R, L, "", GOT_THE_LOCK_NEED_TO_LOCK_A_SUBRESOURCE) +#define lock_ok_l(O, R, L) \ + test_lock(O, R, L, "", GOT_THE_LOCK_NEED_TO_INSTANT_LOCK_A_SUBRESOURCE) +#define lock_conflict(O, R, L) \ + test_lock(O, R, L, "cannot ", DIDNT_GET_THE_LOCK); void test_lockman_simple() { @@ -63,41 +67,41 @@ void test_lockman_simple() lock_ok_a(1, 1, X); lock_ok_i(2, 2, IX); /* failures */ - lock_conflict(2,1,X); + lock_conflict(2, 1, X); unlock_all(2); - lock_ok_a(1,2,S); - lock_ok_a(1,2,IS); - lock_ok_a(1,2,LS); - lock_ok_i(1,3,IX); - lock_ok_a(2,3,LS); - lock_ok_i(1,3,IX); - lock_ok_l(2,3,IS); + lock_ok_a(1, 2, S); + lock_ok_a(1, 2, IS); + lock_ok_a(1, 2, LS); + lock_ok_i(1, 3, IX); + lock_ok_a(2, 3, LS); + lock_ok_i(1, 3, IX); + lock_ok_l(2, 3, IS); unlock_all(1); unlock_all(2); - lock_ok_i(1,1,IX); - lock_conflict(2,1,S); - lock_ok_a(1,1,LS); + lock_ok_i(1, 1, IX); + lock_conflict(2, 1, S); + lock_ok_a(1, 1, LS); unlock_all(1); unlock_all(2); - lock_ok_i(1,1,IX); - lock_ok_a(2,1,LS); - lock_ok_a(1,1,LS); - lock_ok_i(1,1,IX); - lock_ok_i(3,1,IS); + lock_ok_i(1, 1, IX); + lock_ok_a(2, 1, LS); + lock_ok_a(1, 1, LS); + lock_ok_i(1, 1, IX); + lock_ok_i(3, 1, IS); unlock_all(1); unlock_all(2); unlock_all(3); - lock_ok_i(1,4,IS); - lock_ok_i(2,4,IS); - lock_ok_i(3,4,IS); - lock_ok_a(3,4,LS); - lock_ok_i(4,4,IS); - lock_conflict(4,4,IX); - lock_conflict(2,4,IX); - lock_ok_a(1,4,LS); + lock_ok_i(1, 4, IS); + lock_ok_i(2, 4, IS); + lock_ok_i(3, 4, IS); + lock_ok_a(3, 4, LS); + lock_ok_i(4, 4, IS); + lock_conflict(4, 4, IX); + lock_conflict(2, 4, IX); + lock_ok_a(1, 4, LS); unlock_all(1); unlock_all(2); unlock_all(3); @@ -110,7 +114,7 @@ pthread_mutex_t rt_mutex; pthread_cond_t rt_cond; int rt_num_threads; int litmus; -int thread_number= 0, timeouts=0; +int thread_number= 0, timeouts= 0; void run_test(const char *test, pthread_handler handler, int n, int m) { pthread_t t; @@ -121,7 +125,8 @@ void run_test(const char *test, pthread_handler handler, int n, int m) diag("Testing %s with %d threads, %d iterations... ", test, n, m); for (rt_num_threads= n ; n ; n--) - pthread_create(&t, &rt_attr, handler, &m); + if (pthread_create(&t, &rt_attr, handler, &m)) + abort(); pthread_mutex_lock(&rt_mutex); while (rt_num_threads) pthread_cond_wait(&rt_cond, &rt_mutex); @@ -133,9 +138,9 @@ void run_test(const char *test, pthread_handler handler, int n, int m) int Nrows= 100; int Ntables= 10; int table_lock_ratio= 10; -enum lock_type lock_array[6]={S,X,LS,LX,IS,IX}; -char *lock2str[6]={"S","X","LS","LX","IS","IX"}; -char *res2str[4]={ +enum lock_type lock_array[6]= {S, X, LS, LX, IS, IX}; +char *lock2str[6]= {"S", "X", "LS", "LX", "IS", "IX"}; +char *res2str[4]= { "DIDN'T GET THE LOCK", "GOT THE LOCK", "GOT THE LOCK NEED TO LOCK A SUBRESOURCE", @@ -160,12 +165,12 @@ pthread_handler_t test_lockman(void *arg) if (table_lock_ratio && (x/Nrows/4) % table_lock_ratio == 0) { /* table lock */ res= lockman_getlock(&lockman, lo, table, lock_array[locklevel]); - DIAG(("loid=%2d, table %d lock %s, res=%s", loid, table, + DIAG(("loid %2d, table %d, lock %s, res %s", loid, table, lock2str[locklevel], res2str[res])); if (res == DIDNT_GET_THE_LOCK) { lockman_release_locks(&lockman, lo); - DIAG(("loid=%2d, release all locks", loid)); + DIAG(("loid %2d, release all locks", loid)); timeout++; continue; } @@ -175,13 +180,13 @@ pthread_handler_t test_lockman(void *arg) { /* row lock */ locklevel&= 1; res= lockman_getlock(&lockman, lo, table, lock_array[locklevel + 4]); - DIAG(("loid=%2d, row %d lock %s, res=%s", loid, row, + DIAG(("loid %2d, row %d, lock %s, res %s", loid, row, lock2str[locklevel+4], res2str[res])); switch (res) { case DIDNT_GET_THE_LOCK: lockman_release_locks(&lockman, lo); - DIAG(("loid=%2d, release all locks", loid)); + DIAG(("loid %2d, release all locks", loid)); timeout++; continue; case GOT_THE_LOCK: @@ -190,12 +195,12 @@ pthread_handler_t test_lockman(void *arg) /* not implemented, so take a regular lock */ case GOT_THE_LOCK_NEED_TO_LOCK_A_SUBRESOURCE: res= lockman_getlock(&lockman, lo, row, lock_array[locklevel]); - DIAG(("loid=%2d, ROW %d lock %s, res=%s", loid, row, + DIAG(("loid %2d, ROW %d, lock %s, res %s", loid, row, lock2str[locklevel], res2str[res])); if (res == DIDNT_GET_THE_LOCK) { lockman_release_locks(&lockman, lo); - DIAG(("loid=%2d, release all locks", loid)); + DIAG(("loid %2d, release all locks", loid)); timeout++; continue; } @@ -234,7 +239,7 @@ int main() return exit_status(); pthread_attr_init(&rt_attr); - pthread_attr_setdetachstate(&rt_attr,PTHREAD_CREATE_DETACHED); + pthread_attr_setdetachstate(&rt_attr, PTHREAD_CREATE_DETACHED); pthread_mutex_init(&rt_mutex, 0); pthread_cond_init(&rt_cond, 0); @@ -261,13 +266,13 @@ int main() Nrows= 100; Ntables= 10; table_lock_ratio= 10; - run_test("lockman", test_lockman, THREADS,CYCLES); + run_test("lockman", test_lockman, THREADS, CYCLES); /* "real-life" simulation - many rows, no table locks */ Nrows= 1000000; Ntables= 10; table_lock_ratio= 0; - run_test("lockman", test_lockman, THREADS,10000); + run_test("lockman", test_lockman, THREADS, 10000); for (i= 0; i < Nlos; i++) { diff --git a/storage/maria/unittest/trnman-t.c b/storage/maria/unittest/trnman-t.c index 6d4b48c6d3d..7bde9f5f720 100644 --- a/storage/maria/unittest/trnman-t.c +++ b/storage/maria/unittest/trnman-t.c @@ -41,7 +41,7 @@ pthread_handler_t test_trnman(void *arg) pthread_mutex_t mutexes[MAX_ITER]; pthread_cond_t conds[MAX_ITER]; - for (i=0; i < MAX_ITER; i++) + for (i= 0; i < MAX_ITER; i++) { pthread_mutex_init(&mutexes[i], MY_MUTEX_INIT_FAST); pthread_cond_init(&conds[i], 0); @@ -60,7 +60,7 @@ pthread_handler_t test_trnman(void *arg) } } - for (i=0; i < MAX_ITER; i++) + for (i= 0; i < MAX_ITER; i++) { pthread_mutex_destroy(&mutexes[i]); pthread_cond_destroy(&conds[i]); @@ -84,7 +84,8 @@ void run_test(const char *test, pthread_handler handler, int n, int m) diag("Testing %s with %d threads, %d iterations... ", test, n, m); for (rt_num_threads= n ; n ; n--) - pthread_create(&t, &rt_attr, handler, &m); + if (pthread_create(&t, &rt_attr, handler, &m)) + abort(); pthread_mutex_lock(&rt_mutex); while (rt_num_threads) pthread_cond_wait(&rt_cond, &rt_mutex); @@ -94,11 +95,10 @@ void run_test(const char *test, pthread_handler handler, int n, int m) } #define ok_read_from(T1, T2, RES) \ - i=trnman_can_read_from(trn[T1], trid[T2]); \ + i= trnman_can_read_from(trn[T1], trn[T2]->trid); \ ok(i == RES, "trn" #T1 " %s read from trn" #T2, i ? "can" : "cannot") #define start_transaction(T) \ - trn[T]= trnman_new_trn(&mutexes[T], &conds[T]); \ - trid[T]= trn[T]->trid + trn[T]= trnman_new_trn(&mutexes[T], &conds[T]) #define commit(T) trnman_commit_trn(trn[T]) #define abort(T) trnman_abort_trn(trn[T]) @@ -106,12 +106,11 @@ void run_test(const char *test, pthread_handler handler, int n, int m) void test_trnman_read_from() { TRN *trn[Ntrns]; - TrID trid[Ntrns]; pthread_mutex_t mutexes[Ntrns]; pthread_cond_t conds[Ntrns]; int i; - for (i=0; i < Ntrns; i++) + for (i= 0; i < Ntrns; i++) { pthread_mutex_init(&mutexes[i], MY_MUTEX_INIT_FAST); pthread_cond_init(&conds[i], 0); @@ -119,19 +118,19 @@ void test_trnman_read_from() start_transaction(0); /* start trn1 */ start_transaction(1); /* start trn2 */ - ok_read_from(1,0,0); + ok_read_from(1, 0, 0); commit(0); /* commit trn1 */ start_transaction(2); /* start trn4 */ abort(2); /* abort trn4 */ start_transaction(3); /* start trn5 */ - ok_read_from(3,0,1); - ok_read_from(3,1,0); - ok_read_from(3,2,0); + ok_read_from(3, 0, 1); + ok_read_from(3, 1, 0); + ok_read_from(3, 2, 0); commit(1); /* commit trn2 */ - ok_read_from(3,1,0); + ok_read_from(3, 1, 0); commit(3); /* commit trn5 */ - for (i=0; i < Ntrns; i++) + for (i= 0; i < Ntrns; i++) { pthread_mutex_destroy(&mutexes[i]); pthread_cond_destroy(&conds[i]); @@ -148,7 +147,7 @@ int main() return exit_status(); pthread_attr_init(&rt_attr); - pthread_attr_setdetachstate(&rt_attr,PTHREAD_CREATE_DETACHED); + pthread_attr_setdetachstate(&rt_attr, PTHREAD_CREATE_DETACHED); pthread_mutex_init(&rt_mutex, 0); pthread_cond_init(&rt_cond, 0); @@ -158,7 +157,7 @@ int main() trnman_init(); test_trnman_read_from(); - run_test("trnman", test_trnman, THREADS,CYCLES); + run_test("trnman", test_trnman, THREADS, CYCLES); diag("mallocs: %d", trnman_allocated_transactions); { -- cgit v1.2.1 From a79868ae99567e974373d29d631db552f998ccc7 Mon Sep 17 00:00:00 2001 From: unknown Date: Thu, 19 Oct 2006 13:33:49 +0200 Subject: comments --- mysys/lf_alloc-pin.c | 193 ++++++++++++++++++++++++++++++++++++++++++--------- 1 file changed, 159 insertions(+), 34 deletions(-) diff --git a/mysys/lf_alloc-pin.c b/mysys/lf_alloc-pin.c index 842c9690463..ac55185864a 100644 --- a/mysys/lf_alloc-pin.c +++ b/mysys/lf_alloc-pin.c @@ -17,8 +17,63 @@ /* wait-free concurrent allocator based on pinning addresses - TODO test with more than 256 threads - TODO test w/o alloca + It works as follows: every thread (strictly speaking - every CPU, but it's + too difficult to do) has a small array of pointers. They're called "pins". + Before using an object its address must be stored in this array (pinned). + When an object is no longer necessary its address must be removed from + this array (unpinned). When a thread wants to free() an object it + scans all pins of all threads to see if somebody has this object pinned. + If yes - the object is not freed (but stored in a purgatory). + To reduce the cost of a single free() pins are not scanned on every free() + but only added to (thread-local) purgatory. On every LF_PURGATORY_SIZE + free() purgatory is scanned and all unpinned objects are freed. + + Pins are used to solve ABA problem. To use pins one must obey + a pinning protocol: + 1. Let's assume that PTR is a shared pointer to an object. Shared means + that any thread may modify it anytime to point to a different object and + free the old object. Later the freed object may be potentially allocated + by another thread. If we're unlucky that another thread may set PTR to + point to this object again. This is ABA problem. + 2. Create a local pointer LOCAL_PTR. + 3. Pin the PTR in a loop: + do + { + LOCAL_PTR= PTR; + pin(PTR, PIN_NUMBER); + } while (LOCAL_PTR != PTR) + 4. It is guaranteed that after the loop is ended, LOCAL_PTR + points to an object (or NULL, if PTR may be NULL), that + will never be freed. It is not guaranteed though + that LOCAL_PTR == PTR + 5. When done working with the object, remove the pin: + unpin(PIN_NUMBER) + 6. When copying pins (as in the list: + while () + { + pin(CUR, 0); + do + { + NEXT=CUR->next; + pin(NEXT, 1); + } while (NEXT != CUR->next); + ... + ... + pin(CUR, 1); + CUR=NEXT; + } + which keeps CUR address constantly pinned), note than pins may be copied + only upwards (!!!), that is pin N to pin M > N. + 7. Don't keep the object pinned longer than necessary - the number of pins + you have is limited (and small), keeping an object pinned prevents its + reuse and cause unnecessary mallocs. + + Implementation details: + Pins are given away from a "pinbox". Pinbox is stack-based allocator. + It used dynarray for storing pins, new elements are allocated by dynarray + as necessary, old are pushed in the stack for reuse. ABA is solved by + versioning a pointer - because we use an array, a pointer to pins is 32 bit, + upper 32 bits are used for a version. */ #include @@ -29,6 +84,10 @@ static void _lf_pinbox_real_free(LF_PINS *pins); +/* + Initialize a pinbox. Must be usually called from lf_alloc_init. + See the latter for details. +*/ void lf_pinbox_init(LF_PINBOX *pinbox, uint free_ptr_offset, lf_pinbox_free_func *free_func,void *free_func_arg) { @@ -47,6 +106,14 @@ void lf_pinbox_destroy(LF_PINBOX *pinbox) lf_dynarray_destroy(&pinbox->pinstack); } +/* + Get pins from a pinbox. Usually called via lf_alloc_get_pins() or + lf_hash_get_pins(). + + DESCRIPTION + get a new LF_PINS structure from a stack of unused pins, + or allocate a new one out of dynarray. +*/ LF_PINS *_lf_pinbox_get_pins(LF_PINBOX *pinbox) { uint32 pins, next, top_ver; @@ -71,6 +138,14 @@ LF_PINS *_lf_pinbox_get_pins(LF_PINBOX *pinbox) return el; } +/* + Put pins back to a pinbox. Usually called via lf_alloc_put_pins() or + lf_hash_put_pins(). + + DESCRIPTION + empty the purgatory (XXX deadlock warning below!), + push LF_PINS structure to a stack +*/ void _lf_pinbox_put_pins(LF_PINS *pins) { LF_PINBOX *pinbox= pins->pinbox; @@ -80,11 +155,11 @@ void _lf_pinbox_put_pins(LF_PINS *pins) { int i; for (i= 0; i < LF_PINBOX_PINS; i++) - assert(pins->pin[i] == 0); + DBUG_ASSERT(pins->pin[i] == 0); } #endif /* - Note - this will deadlock if other threads will wait for + XXX this will deadlock if other threads will wait for the caller to do something after _lf_pinbox_put_pins(), and they would have pinned addresses that the caller wants to free. Thus: only free pins when all work is done and nobody can wait for you!!! @@ -130,6 +205,13 @@ static int ptr_cmp(void **a, void **b) (PINS)->purgatory_count++; \ } while (0) +/* + Free an object allocated via pinbox allocator + + DESCRIPTION + add an object to purgatory. if necessary, call _lf_pinbox_real_free() + to actually free something. +*/ void _lf_pinbox_free(LF_PINS *pins, void *addr) { if (pins->purgatory_count % LF_PURGATORY_SIZE) @@ -142,6 +224,10 @@ struct st_harvester { int npins; }; +/* + callback for _lf_dynarray_iterate: + scan all pins or all threads and accumulate all pins +*/ static int harvest_pins(LF_PINS *el, struct st_harvester *hv) { int i; @@ -159,6 +245,10 @@ static int harvest_pins(LF_PINS *el, struct st_harvester *hv) return 0; } +/* + callback for _lf_dynarray_iterate: + scan all pins or all threads and see if addr is present there +*/ static int match_pins(LF_PINS *el, void *addr) { int i; @@ -170,6 +260,9 @@ static int match_pins(LF_PINS *el, void *addr) return 0; } +/* + Scan the purgatory as free everything that can be freed +*/ static void _lf_pinbox_real_free(LF_PINS *pins) { int npins; @@ -187,10 +280,12 @@ static void _lf_pinbox_real_free(LF_PINS *pins) addr= (void **) alloca(sizeof(void *)*LF_PINBOX_PINS*npins); hv.granary= addr; hv.npins= npins; + /* scan the dynarray and accumulate all pinned addresses */ _lf_dynarray_iterate(&pinbox->pinstack, (lf_dynarray_func)harvest_pins, &hv); npins= hv.granary-addr; + /* and sort them */ if (npins) qsort(addr, npins, sizeof(void *), (qsort_cmp)ptr_cmp); } @@ -207,7 +302,7 @@ static void _lf_pinbox_real_free(LF_PINS *pins) list= *(void **)((char *)cur+pinbox->free_ptr_offset); if (npins) { - if (addr) + if (addr) /* use binary search */ { void **a,**b,**c; for (a= addr, b= addr+npins-1, c= a+(b-a)/2; b-a>1; c= a+(b-a)/2) @@ -220,7 +315,7 @@ static void _lf_pinbox_real_free(LF_PINS *pins) if (cur == *a || cur == *b) goto found; } - else + else /* no alloca - no cookie. linear search here */ { if (_lf_dynarray_iterate(&pinbox->pinstack, (lf_dynarray_func)match_pins, cur)) @@ -236,6 +331,10 @@ found: } } +/* + callback for _lf_pinbox_real_free to free an unpinned object - + add it back to the allocator stack +*/ static void alloc_free(void *node, LF_ALLOCATOR *allocator) { void *tmp; @@ -247,7 +346,55 @@ static void alloc_free(void *node, LF_ALLOCATOR *allocator) LF_BACKOFF); } +/* lock-free memory allocator for fixed-size objects */ + LF_REQUIRE_PINS(1); + +/* + initialize lock-free allocatod. + + SYNOPSYS + allocator - + size a size of an object to allocate + free_ptr_offset an offset inside the object to a sizeof(void *) + memory that is guaranteed to be unused after + the object is put in the purgatory. Unused by ANY + thread, not only the purgatory owner. +*/ +void lf_alloc_init(LF_ALLOCATOR *allocator, uint size, uint free_ptr_offset) +{ + lf_pinbox_init(&allocator->pinbox, free_ptr_offset, + (lf_pinbox_free_func *)alloc_free, allocator); + allocator->top= 0; + allocator->mallocs= 0; + allocator->element_size= size; + DBUG_ASSERT(size >= (int)sizeof(void *)); + DBUG_ASSERT(free_ptr_offset < size); +} + +/* + destroy the allocator, free everything that's in it +*/ +void lf_alloc_destroy(LF_ALLOCATOR *allocator) +{ + void *el= allocator->top; + while (el) + { + void *tmp= *(void **)el; + my_free(el, MYF(0)); + el= tmp; + } + lf_pinbox_destroy(&allocator->pinbox); + allocator->top= 0; +} + +/* + Allocate and return an new object. + + DESCRIPTION + Pop an unused object from the stack or malloc it is the stack is empty. + pin[0] is used, it's removed on return. +*/ void *_lf_alloc_new(LF_PINS *pins) { LF_ALLOCATOR *allocator= (LF_ALLOCATOR *)(pins->pinbox->free_func_arg); @@ -262,47 +409,25 @@ void *_lf_alloc_new(LF_PINS *pins) if (!node) { if (!(node= my_malloc(allocator->element_size, MYF(MY_WME|MY_ZEROFILL)))) - goto ret; + break; #ifdef MY_LF_EXTRA_DEBUG my_atomic_add32(&allocator->mallocs, 1); #endif - goto ret; + break; } if (my_atomic_casptr((void **)&allocator->top, (void *)&node, *(void **)node)) - goto ret; + break; } -ret: _lf_unpin(pins, 0); return node; } -void lf_alloc_init(LF_ALLOCATOR *allocator, uint size, uint free_ptr_offset) -{ - lf_pinbox_init(&allocator->pinbox, free_ptr_offset, - (lf_pinbox_free_func *)alloc_free, allocator); - allocator->top= 0; - allocator->mallocs= 0; - allocator->element_size= size; - DBUG_ASSERT(size >= (int)sizeof(void *)); -} - -void lf_alloc_destroy(LF_ALLOCATOR *allocator) -{ - void *el= allocator->top; - while (el) - { - void *tmp= *(void **)el; - my_free(el, MYF(0)); - el= tmp; - } - lf_pinbox_destroy(&allocator->pinbox); - allocator->top= 0; -} - /* + count the number of objects in a pool. + NOTE - this is NOT thread-safe !!! + This is NOT thread-safe !!! */ uint lf_alloc_in_pool(LF_ALLOCATOR *allocator) { -- cgit v1.2.1 From fb818dd7b0be3b4facd159de14bc3d9afcbcf16e Mon Sep 17 00:00:00 2001 From: unknown Date: Fri, 20 Oct 2006 14:02:18 +0200 Subject: more post-review fixes - comments, renames, error checks in unit tests concurrency bug in lock manager include/my_global.h: compile-time assert macro mysys/my_atomic.c: use compile_time_assert() macro storage/maria/lockman.c: bug in concurrent lockdelete (with retries) storage/maria/trnman.c: more post-review fixes - comments, renames storage/maria/trnman.h: more post-review fixes - comments storage/maria/unittest/lockman-t.c: friendlier error checks storage/maria/unittest/trnman-t.c: friendlier error checks --- include/my_global.h | 7 +++++ mysys/my_atomic.c | 2 +- storage/maria/lockman.c | 11 ++++++-- storage/maria/trnman.c | 56 +++++++++++++++++++++++++++----------- storage/maria/trnman.h | 10 ++++++- storage/maria/unittest/lockman-t.c | 14 ++++++---- storage/maria/unittest/trnman-t.c | 15 ++++++++-- 7 files changed, 87 insertions(+), 28 deletions(-) diff --git a/include/my_global.h b/include/my_global.h index 7f57b43f89f..7f7b502e1a7 100644 --- a/include/my_global.h +++ b/include/my_global.h @@ -428,6 +428,13 @@ C_MODE_END */ #include +/* an assert that works at compile-time. only for constant expression */ +#define compile_time_assert(X) \ + do \ + { \ + char compile_time_assert[(X) ? 1 : -1]; \ + } while(0) + /* Go around some bugs in different OS and compilers */ #if defined (HPUX11) && defined(_LARGEFILE_SOURCE) #define _LARGEFILE64_SOURCE diff --git a/mysys/my_atomic.c b/mysys/my_atomic.c index fbeb3d63bef..e1af7e0328d 100644 --- a/mysys/my_atomic.c +++ b/mysys/my_atomic.c @@ -35,7 +35,7 @@ */ int my_atomic_initialize() { - char assert_the_size[sizeof(intptr) == sizeof(void *) ? 1 : -1]; + compile_time_assert(sizeof(intptr) == sizeof(void *)); /* currently the only thing worth checking is SMP/UP issue */ #ifdef MY_ATOMIC_MODE_DUMMY return my_getncpus() == 1 ? MY_ATOMIC_OK : MY_ATOMIC_NOT_1CPU; diff --git a/storage/maria/lockman.c b/storage/maria/lockman.c index b37c31fb52c..937c61021cc 100644 --- a/storage/maria/lockman.c +++ b/storage/maria/lockman.c @@ -421,13 +421,14 @@ static int lockinsert(LOCK * volatile *head, LOCK *node, LF_PINS *pins, } if (!(res & NEED_TO_WAIT)) node->flags|= ACTIVE; - else - node->flags&= ~ACTIVE; /* if we're retrying on REPEAT_ONCE_MORE */ node->link= (intptr)cursor.curr; DBUG_ASSERT(node->link != (intptr)node); DBUG_ASSERT(cursor.prev != &node->link); if (!my_atomic_casptr((void **)cursor.prev, (void **)&cursor.curr, node)) + { res= REPEAT_ONCE_MORE; + node->flags&= ~ACTIVE; + } if (res & LOCK_UPGRADE) cursor.upgrade_from->flags|= IGNORE_ME; } @@ -496,7 +497,11 @@ static int lockdelete(LOCK * volatile *head, LOCK *node, LF_PINS *pins) lockfind(head, node, &cursor, pins); } else + { res= REPEAT_ONCE_MORE; + if (cursor.upgrade_from) /* to satisfy the assert in lockfind */ + cursor.upgrade_from->flags|= IGNORE_ME; + } } while (res == REPEAT_ONCE_MORE); _lf_unpin(pins, 0); _lf_unpin(pins, 1); @@ -744,7 +749,7 @@ static char *lock2str[]= void print_lockhash(LOCKMAN *lm) { LOCK *el= *(LOCK **)_lf_dynarray_lvalue(&lm->array, 0); - printf("hash: size:%u count:%u\n", lm->size, lm->count); + printf("hash: size %u count %u\n", lm->size, lm->count); while (el) { intptr next= el->link; diff --git a/storage/maria/trnman.c b/storage/maria/trnman.c index 95e5be4284a..a227df16395 100644 --- a/storage/maria/trnman.c +++ b/storage/maria/trnman.c @@ -48,25 +48,35 @@ static my_atomic_rwlock_t LOCK_short_trid_to_trn, LOCK_pool; static LOCKMAN maria_lockman; -static byte *trn_get_hash_key(const byte *trn, uint* len, my_bool unused) -{ - *len= sizeof(TrID); - return (byte *) & ((*((TRN **)trn))->trid); -} +/* + short transaction id is at the same time its identifier + for a lock manager - its lock owner identifier (loid) +*/ +#define short_id locks.loid -static LOCK_OWNER *trnman_short_trid_to_TRN(uint16 short_trid) +/* + NOTE + Just as short_id doubles as loid, this function doubles as + short_trid_to_LOCK_OWNER. See the compile-time assert below. +*/ +static TRN *short_trid_to_TRN(uint16 short_trid) { TRN *trn; + compile_time_assert(offsetof(TRN, locks) == 0); my_atomic_rwlock_rdlock(&LOCK_short_trid_to_trn); trn= my_atomic_loadptr((void **)&short_trid_to_active_trn[short_trid]); my_atomic_rwlock_rdunlock(&LOCK_short_trid_to_trn); - return (LOCK_OWNER *)trn; + return (TRN *)trn; } -int trnman_init() +static byte *trn_get_hash_key(const byte *trn, uint* len, my_bool unused) { - pthread_mutex_init(&LOCK_trn_list, MY_MUTEX_INIT_FAST); + *len= sizeof(TrID); + return (byte *) & ((*((TRN **)trn))->trid); +} +int trnman_init() +{ /* Initialize lists. active_list_max.min_read_from must be larger than any trid, @@ -95,6 +105,7 @@ int trnman_init() global_trid_generator= 0; /* set later by the recovery code */ lf_hash_init(&trid_to_committed_trn, sizeof(TRN*), LF_HASH_UNIQUE, 0, 0, trn_get_hash_key, 0); + pthread_mutex_init(&LOCK_trn_list, MY_MUTEX_INIT_FAST); my_atomic_rwlock_init(&LOCK_short_trid_to_trn); my_atomic_rwlock_init(&LOCK_pool); short_trid_to_active_trn= (TRN **)my_malloc(SHORT_TRID_MAX*sizeof(TRN*), @@ -103,7 +114,7 @@ int trnman_init() return 1; short_trid_to_active_trn--; /* min short_trid is 1 */ - lockman_init(&maria_lockman, &trnman_short_trid_to_TRN, 10000); + lockman_init(&maria_lockman, (loid_to_lo_func *)&short_trid_to_TRN, 10000); return 0; } @@ -162,7 +173,7 @@ static void set_short_trid(TRN *trn) break; } my_atomic_rwlock_wrunlock(&LOCK_short_trid_to_trn); - trn->locks.loid= i; + trn->short_id= i; } /* @@ -210,7 +221,7 @@ TRN *trnman_new_trn(pthread_mutex_t *mutex, pthread_cond_t *cond) trn->min_read_from= active_list_min.next->trid; trn->trid= new_trid(); - trn->locks.loid= 0; + trn->short_id= 0; trn->next= &active_list_max; trn->prev= active_list_max.prev; @@ -242,6 +253,15 @@ TRN *trnman_new_trn(pthread_mutex_t *mutex, pthread_cond_t *cond) /* remove a trn from the active list. if necessary - move to committed list and set commit_trid + + NOTE + Locks are released at the end. In particular, after placing the + transaction in commit list, and after setting commit_trid. It's + important, as commit_trid affects visibility. Locks don't affect + anything they simply delay execution of other threads - they could be + released arbitrarily late. In other words, when locks are released it + serves as a start banner for other threads, they start to run. So + everything they may need must be ready at that point. */ void trnman_end_trn(TRN *trn, my_bool commit) { @@ -305,7 +325,7 @@ void trnman_end_trn(TRN *trn, my_bool commit) trn->locks.mutex= 0; trn->locks.cond= 0; my_atomic_rwlock_rdlock(&LOCK_short_trid_to_trn); - my_atomic_storeptr((void **)&short_trid_to_active_trn[trn->locks.loid], 0); + my_atomic_storeptr((void **)&short_trid_to_active_trn[trn->short_id], 0); my_atomic_rwlock_rdunlock(&LOCK_short_trid_to_trn); while (free_me) // XXX send them to the purge thread @@ -325,9 +345,13 @@ void trnman_end_trn(TRN *trn, my_bool commit) /* free a trn (add to the pool, that is) - note - we can never really free() a TRN if there's at least one - other running transaction - see, e.g., how lock waits are implemented - in lockman.c + note - we can never really free() a TRN if there's at least one other + running transaction - see, e.g., how lock waits are implemented in + lockman.c + The same is true for other lock-free data structures too. We may need some + kind of FLUSH command to reset them all - ensuring that no transactions are + running. It may even be called automatically on checkpoints if no + transactions are running. */ void trnman_free_trn(TRN *trn) { diff --git a/storage/maria/trnman.h b/storage/maria/trnman.h index c059947f35c..eeab253ae71 100644 --- a/storage/maria/trnman.h +++ b/storage/maria/trnman.h @@ -22,9 +22,17 @@ typedef uint64 TrID; /* our TrID is 6 bytes */ typedef struct st_transaction TRN; +/* + trid - 6 byte transaction identifier. Assigned when a transaction + is created. Transaction can always be identified by its trid, + even after transaction has ended. + + short_trid - 2-byte transaction identifier, identifies a running + transaction, is reassigned when transaction ends. +*/ struct st_transaction { - LOCK_OWNER locks; + LOCK_OWNER locks; /* must be the first! see short_trid_to_TRN() */ LF_PINS *pins; TrID trid, min_read_from, commit_trid; TRN *next, *prev; diff --git a/storage/maria/unittest/lockman-t.c b/storage/maria/unittest/lockman-t.c index ad382a360fb..6b5e5912fdf 100644 --- a/storage/maria/unittest/lockman-t.c +++ b/storage/maria/unittest/lockman-t.c @@ -123,16 +123,20 @@ void run_test(const char *test, pthread_handler handler, int n, int m) thread_number= timeouts= 0; litmus= 0; - diag("Testing %s with %d threads, %d iterations... ", test, n, m); + diag("Running %s with %d threads, %d iterations... ", test, n, m); for (rt_num_threads= n ; n ; n--) if (pthread_create(&t, &rt_attr, handler, &m)) - abort(); + { + diag("Could not create thread"); + litmus++; + rt_num_threads--; + } pthread_mutex_lock(&rt_mutex); while (rt_num_threads) pthread_cond_wait(&rt_cond, &rt_mutex); pthread_mutex_unlock(&rt_mutex); now= my_getsystime()-now; - ok(litmus == 0, "tested %s in %g secs (%d)", test, ((double)now)/1e7, litmus); + ok(litmus == 0, "Finished %s in %g secs (%d)", test, ((double)now)/1e7, litmus); } int Nrows= 100; @@ -266,13 +270,13 @@ int main() Nrows= 100; Ntables= 10; table_lock_ratio= 10; - run_test("lockman", test_lockman, THREADS, CYCLES); + run_test("\"random lock\" stress test", test_lockman, THREADS, CYCLES); /* "real-life" simulation - many rows, no table locks */ Nrows= 1000000; Ntables= 10; table_lock_ratio= 0; - run_test("lockman", test_lockman, THREADS, 10000); + run_test("\"real-life\" simulation test", test_lockman, THREADS, CYCLES*10); for (i= 0; i < Nlos; i++) { diff --git a/storage/maria/unittest/trnman-t.c b/storage/maria/unittest/trnman-t.c index 7bde9f5f720..822f5cd755b 100644 --- a/storage/maria/unittest/trnman-t.c +++ b/storage/maria/unittest/trnman-t.c @@ -52,14 +52,21 @@ pthread_handler_t test_trnman(void *arg) y= x= (x*3628273133 + 1500450271) % 9576890767; /* three prime numbers */ m-= n= x % MAX_ITER; for (i= 0; i < n; i++) + { trn[i]= trnman_new_trn(&mutexes[i], &conds[i]); + if (!trn[i]) + { + diag("trnman_new_trn() failed"); + litmus++; + } + } for (i= 0; i < n; i++) { y= (y*19 + 7) % 31; trnman_end_trn(trn[i], y & 1); } } - +end: for (i= 0; i < MAX_ITER; i++) { pthread_mutex_destroy(&mutexes[i]); @@ -85,7 +92,11 @@ void run_test(const char *test, pthread_handler handler, int n, int m) diag("Testing %s with %d threads, %d iterations... ", test, n, m); for (rt_num_threads= n ; n ; n--) if (pthread_create(&t, &rt_attr, handler, &m)) - abort(); + { + diag("Could not create thread"); + litmus++; + rt_num_threads--; + } pthread_mutex_lock(&rt_mutex); while (rt_num_threads) pthread_cond_wait(&rt_cond, &rt_mutex); -- cgit v1.2.1 From 79ff2b036c39f82a81d9890c9f4ab5539fd6755f Mon Sep 17 00:00:00 2001 From: unknown Date: Sat, 21 Oct 2006 15:53:42 +0200 Subject: malloc() failure is unlikely(). storage/maria/trnman.c: malloc() failure is unlikely --- storage/maria/trnman.c | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/storage/maria/trnman.c b/storage/maria/trnman.c index a227df16395..8262c57fa85 100644 --- a/storage/maria/trnman.c +++ b/storage/maria/trnman.c @@ -110,7 +110,7 @@ int trnman_init() my_atomic_rwlock_init(&LOCK_pool); short_trid_to_active_trn= (TRN **)my_malloc(SHORT_TRID_MAX*sizeof(TRN*), MYF(MY_WME|MY_ZEROFILL)); - if (!short_trid_to_active_trn) + if (unlikely(!short_trid_to_active_trn)) return 1; short_trid_to_active_trn--; /* min short_trid is 1 */ -- cgit v1.2.1 From d1852da743be3a636bba35114d1342b4b11039f0 Mon Sep 17 00:00:00 2001 From: unknown Date: Sat, 21 Oct 2006 20:06:56 +0200 Subject: less confusing and more correct name for my_atomic_swap --- include/atomic/rwlock.h | 2 +- include/atomic/x86-gcc.h | 2 +- include/atomic/x86-msvc.h | 2 +- include/my_atomic.h | 22 +++++++++++----------- unittest/mysys/my_atomic-t.c | 8 ++++---- 5 files changed, 18 insertions(+), 18 deletions(-) diff --git a/include/atomic/rwlock.h b/include/atomic/rwlock.h index 3d8edb9e27e..5a3ca5a434e 100644 --- a/include/atomic/rwlock.h +++ b/include/atomic/rwlock.h @@ -47,7 +47,7 @@ typedef struct {pthread_mutex_t rw;} my_atomic_rwlock_t; #endif #define make_atomic_add_body(S) int ## S sav; sav= *a; *a+= v; v=sav; -#define make_atomic_swap_body(S) int ## S sav; sav= *a; *a= v; v=sav; +#define make_atomic_fas_body(S) int ## S sav; sav= *a; *a= v; v=sav; #define make_atomic_cas_body(S) if ((ret= (*a == *cmp))) *a= set; else *cmp=*a; #define make_atomic_load_body(S) ret= *a; #define make_atomic_store_body(S) *a= v; diff --git a/include/atomic/x86-gcc.h b/include/atomic/x86-gcc.h index 0be8fdf9244..cd84faeee31 100644 --- a/include/atomic/x86-gcc.h +++ b/include/atomic/x86-gcc.h @@ -43,7 +43,7 @@ #define make_atomic_add_body(S) \ asm volatile (LOCK_prefix "; xadd %0, %1;" : "+r" (v) , "+m" (*a)) #endif -#define make_atomic_swap_body(S) \ +#define make_atomic_fas_body(S) \ asm volatile ("xchg %0, %1;" : "+r" (v) , "+m" (*a)) #define make_atomic_cas_body(S) \ asm volatile (LOCK_prefix "; cmpxchg %3, %0; setz %2;" \ diff --git a/include/atomic/x86-msvc.h b/include/atomic/x86-msvc.h index 8f3e55aaed7..bd5a7fdd96c 100644 --- a/include/atomic/x86-msvc.h +++ b/include/atomic/x86-msvc.h @@ -43,7 +43,7 @@ _asm setz al \ _asm movzx ret, al \ } -#define make_atomic_swap_body(S) \ +#define make_atomic_fas_body(S) \ _asm { \ _asm mov reg_ ## S, v \ _asm xchg *a, reg_ ## S \ diff --git a/include/my_atomic.h b/include/my_atomic.h index 921b55e68a2..aad69295998 100644 --- a/include/my_atomic.h +++ b/include/my_atomic.h @@ -43,11 +43,11 @@ STATIC_INLINE int ## S my_atomic_add ## S( \ return v; \ } -#define make_atomic_swap(S) \ -STATIC_INLINE int ## S my_atomic_swap ## S( \ +#define make_atomic_fas(S) \ +STATIC_INLINE int ## S my_atomic_fas ## S( \ int ## S volatile *a, int ## S v) \ { \ - make_atomic_swap_body(S); \ + make_atomic_fas_body(S); \ return v; \ } @@ -80,8 +80,8 @@ STATIC_INLINE void my_atomic_store ## S( \ #define make_atomic_add(S) \ extern int ## S my_atomic_add ## S(int ## S volatile *a, int ## S v); -#define make_atomic_swap(S) \ -extern int ## S my_atomic_swap ## S(int ## S volatile *a, int ## S v); +#define make_atomic_fas(S) \ +extern int ## S my_atomic_fas ## S(int ## S volatile *a, int ## S v); #define make_atomic_cas(S) \ extern int my_atomic_cas ## S(int ## S volatile *a, int ## S *cmp, int ## S set); @@ -113,10 +113,10 @@ make_atomic_store(16) make_atomic_store(32) make_atomic_store(ptr) -make_atomic_swap( 8) -make_atomic_swap(16) -make_atomic_swap(32) -make_atomic_swap(ptr) +make_atomic_fas( 8) +make_atomic_fas(16) +make_atomic_fas(32) +make_atomic_fas(ptr) #ifdef _atomic_h_cleanup_ #include _atomic_h_cleanup_ @@ -127,12 +127,12 @@ make_atomic_swap(ptr) #undef make_atomic_cas #undef make_atomic_load #undef make_atomic_store -#undef make_atomic_swap +#undef make_atomic_fas #undef make_atomic_add_body #undef make_atomic_cas_body #undef make_atomic_load_body #undef make_atomic_store_body -#undef make_atomic_swap_body +#undef make_atomic_fas_body #undef intptr #ifndef LF_BACKOFF diff --git a/unittest/mysys/my_atomic-t.c b/unittest/mysys/my_atomic-t.c index e2177b94720..a3b2241f5ed 100644 --- a/unittest/mysys/my_atomic-t.c +++ b/unittest/mysys/my_atomic-t.c @@ -62,7 +62,7 @@ pthread_handler_t test_atomic_add_handler(void *arg) 5. subtract result from a32 must get 0 in a32 at the end */ -pthread_handler_t test_atomic_swap_handler(void *arg) +pthread_handler_t test_atomic_fas_handler(void *arg) { int m=*(int *)arg; uint32 x=my_atomic_add32(&b32, 1); @@ -72,14 +72,14 @@ pthread_handler_t test_atomic_swap_handler(void *arg) for (; m ; m--) { my_atomic_rwlock_wrlock(&rwl); - x=my_atomic_swap32(&c32, x); + x=my_atomic_fas32(&c32, x); my_atomic_rwlock_wrunlock(&rwl); } if (!x) { my_atomic_rwlock_wrlock(&rwl); - x=my_atomic_swap32(&c32, x); + x=my_atomic_fas32(&c32, x); my_atomic_rwlock_wrunlock(&rwl); } @@ -306,7 +306,7 @@ int main() #define THREADS 100 test_atomic("my_atomic_add32", test_atomic_add_handler, THREADS,CYCLES); - test_atomic("my_atomic_swap32", test_atomic_swap_handler, THREADS,CYCLES); + test_atomic("my_atomic_fas32", test_atomic_fas_handler, THREADS,CYCLES); test_atomic("my_atomic_cas32", test_atomic_cas_handler, THREADS,CYCLES); test_atomic("lf_pinbox", test_lf_pinbox, THREADS,CYCLES); test_atomic("lf_alloc", test_lf_alloc, THREADS,CYCLES); -- cgit v1.2.1 From 32739cbbcac9b8af23b4708db21bf2854b5e27a8 Mon Sep 17 00:00:00 2001 From: unknown Date: Sun, 22 Oct 2006 16:05:18 +0200 Subject: make sure all test threads really exit before main() - workaround for NPTL bug in some linux kernels (Bug#22320) --- storage/maria/unittest/lockman-t.c | 39 ++++----- storage/maria/unittest/trnman-t.c | 36 ++++---- unittest/mysys/my_atomic-t.c | 171 ++++++++++++++++--------------------- 3 files changed, 107 insertions(+), 139 deletions(-) diff --git a/storage/maria/unittest/lockman-t.c b/storage/maria/unittest/lockman-t.c index 6b5e5912fdf..de824f05671 100644 --- a/storage/maria/unittest/lockman-t.c +++ b/storage/maria/unittest/lockman-t.c @@ -109,36 +109,41 @@ void test_lockman_simple() } -pthread_attr_t rt_attr; -pthread_mutex_t rt_mutex; -pthread_cond_t rt_cond; int rt_num_threads; int litmus; int thread_number= 0, timeouts= 0; void run_test(const char *test, pthread_handler handler, int n, int m) { - pthread_t t; + pthread_t *threads; ulonglong now= my_getsystime(); + int i; thread_number= timeouts= 0; litmus= 0; + threads= (pthread_t *)my_malloc(sizeof(void *)*n, MYF(0)); + if (!threads) + { + diag("Out of memory"); + abort(); + } + diag("Running %s with %d threads, %d iterations... ", test, n, m); - for (rt_num_threads= n ; n ; n--) - if (pthread_create(&t, &rt_attr, handler, &m)) + rt_num_threads= n; + for (i= 0; i < n ; i++) + if (pthread_create(threads+i, 0, handler, &m)) { diag("Could not create thread"); - litmus++; - rt_num_threads--; + abort(); } - pthread_mutex_lock(&rt_mutex); - while (rt_num_threads) - pthread_cond_wait(&rt_cond, &rt_mutex); - pthread_mutex_unlock(&rt_mutex); + for (i= 0 ; i < n ; i++) + pthread_join(threads[i], 0); now= my_getsystime()-now; ok(litmus == 0, "Finished %s in %g secs (%d)", test, ((double)now)/1e7, litmus); + my_free((void*)threads, MYF(0)); } +pthread_mutex_t rt_mutex; int Nrows= 100; int Ntables= 10; int table_lock_ratio= 10; @@ -222,10 +227,7 @@ pthread_handler_t test_lockman(void *arg) rt_num_threads--; timeouts+= timeout; if (!rt_num_threads) - { - pthread_cond_signal(&rt_cond); diag("number of timeouts: %d", timeouts); - } pthread_mutex_unlock(&rt_mutex); return 0; @@ -236,16 +238,13 @@ int main() int i; my_init(); + pthread_mutex_init(&rt_mutex, 0); plan(31); if (my_atomic_initialize()) return exit_status(); - pthread_attr_init(&rt_attr); - pthread_attr_setdetachstate(&rt_attr, PTHREAD_CREATE_DETACHED); - pthread_mutex_init(&rt_mutex, 0); - pthread_cond_init(&rt_cond, 0); lockman_init(&lockman, &loid2lo, 50); @@ -294,8 +293,6 @@ int main() } pthread_mutex_destroy(&rt_mutex); - pthread_cond_destroy(&rt_cond); - pthread_attr_destroy(&rt_attr); my_end(0); return exit_status(); } diff --git a/storage/maria/unittest/trnman-t.c b/storage/maria/unittest/trnman-t.c index 822f5cd755b..7da8202b881 100644 --- a/storage/maria/unittest/trnman-t.c +++ b/storage/maria/unittest/trnman-t.c @@ -22,9 +22,7 @@ #include #include "../trnman.h" -pthread_attr_t rt_attr; pthread_mutex_t rt_mutex; -pthread_cond_t rt_cond; int rt_num_threads; int litmus; @@ -74,8 +72,6 @@ end: } pthread_mutex_lock(&rt_mutex); rt_num_threads--; - if (!rt_num_threads) - pthread_cond_signal(&rt_cond); pthread_mutex_unlock(&rt_mutex); return 0; @@ -84,25 +80,32 @@ end: void run_test(const char *test, pthread_handler handler, int n, int m) { - pthread_t t; + pthread_t *threads; ulonglong now= my_getsystime(); + int i; litmus= 0; + threads= (pthread_t *)my_malloc(sizeof(void *)*n, MYF(0)); + if (!threads) + { + diag("Out of memory"); + abort(); + } + diag("Testing %s with %d threads, %d iterations... ", test, n, m); - for (rt_num_threads= n ; n ; n--) - if (pthread_create(&t, &rt_attr, handler, &m)) + rt_num_threads= n; + for (i= 0; i < n ; i++) + if (pthread_create(threads+i, 0, handler, &m)) { diag("Could not create thread"); - litmus++; - rt_num_threads--; + abort(); } - pthread_mutex_lock(&rt_mutex); - while (rt_num_threads) - pthread_cond_wait(&rt_cond, &rt_mutex); - pthread_mutex_unlock(&rt_mutex); + for (i= 0 ; i < n ; i++) + pthread_join(threads[i], 0); now= my_getsystime()-now; - ok(litmus == 0, "tested %s in %g secs (%d)", test, ((double)now)/1e7, litmus); + ok(litmus == 0, "Tested %s in %g secs (%d)", test, ((double)now)/1e7, litmus); + my_free((void*)threads, MYF(0)); } #define ok_read_from(T1, T2, RES) \ @@ -157,10 +160,7 @@ int main() if (my_atomic_initialize()) return exit_status(); - pthread_attr_init(&rt_attr); - pthread_attr_setdetachstate(&rt_attr, PTHREAD_CREATE_DETACHED); pthread_mutex_init(&rt_mutex, 0); - pthread_cond_init(&rt_cond, 0); #define CYCLES 10000 #define THREADS 10 @@ -179,8 +179,6 @@ int main() } pthread_mutex_destroy(&rt_mutex); - pthread_cond_destroy(&rt_cond); - pthread_attr_destroy(&rt_attr); my_end(0); return exit_status(); } diff --git a/unittest/mysys/my_atomic-t.c b/unittest/mysys/my_atomic-t.c index a3b2241f5ed..88fa2b03242 100644 --- a/unittest/mysys/my_atomic-t.c +++ b/unittest/mysys/my_atomic-t.c @@ -21,24 +21,19 @@ #include #include -volatile uint32 a32,b32,c32; +volatile uint32 a32,b32,c32, N; my_atomic_rwlock_t rwl; LF_ALLOCATOR lf_allocator; LF_HASH lf_hash; -pthread_attr_t thr_attr; -pthread_mutex_t mutex; -pthread_cond_t cond; -int N; - /* add and sub a random number in a loop. Must get 0 at the end */ pthread_handler_t test_atomic_add_handler(void *arg) { - int m=(*(int *)arg)/2; + int m= (*(int *)arg)/2; int32 x; - for (x=((int)(intptr)(&m)); m ; m--) + for (x= ((int)(intptr)(&m)); m ; m--) { - x=(x*m+0x87654321) & INT_MAX32; + x= (x*m+0x87654321) & INT_MAX32; my_atomic_rwlock_wrlock(&rwl); my_atomic_add32(&a32, x); my_atomic_rwlock_wrunlock(&rwl); @@ -47,10 +42,6 @@ pthread_handler_t test_atomic_add_handler(void *arg) my_atomic_add32(&a32, -x); my_atomic_rwlock_wrunlock(&rwl); } - pthread_mutex_lock(&mutex); - N--; - if (!N) pthread_cond_signal(&cond); - pthread_mutex_unlock(&mutex); return 0; } @@ -64,22 +55,22 @@ pthread_handler_t test_atomic_add_handler(void *arg) */ pthread_handler_t test_atomic_fas_handler(void *arg) { - int m=*(int *)arg; - uint32 x=my_atomic_add32(&b32, 1); + int m= *(int *)arg; + uint32 x= my_atomic_add32(&b32, 1); my_atomic_add32(&a32, x); for (; m ; m--) { my_atomic_rwlock_wrlock(&rwl); - x=my_atomic_fas32(&c32, x); + x= my_atomic_fas32(&c32, x); my_atomic_rwlock_wrunlock(&rwl); } if (!x) { my_atomic_rwlock_wrlock(&rwl); - x=my_atomic_fas32(&c32, x); + x= my_atomic_fas32(&c32, x); my_atomic_rwlock_wrunlock(&rwl); } @@ -87,10 +78,6 @@ pthread_handler_t test_atomic_fas_handler(void *arg) my_atomic_add32(&a32, -x); my_atomic_rwlock_wrunlock(&rwl); - pthread_mutex_lock(&mutex); - N--; - if (!N) pthread_cond_signal(&cond); - pthread_mutex_unlock(&mutex); return 0; } @@ -101,29 +88,25 @@ pthread_handler_t test_atomic_fas_handler(void *arg) */ pthread_handler_t test_atomic_cas_handler(void *arg) { - int m=(*(int *)arg)/2, ok=0; + int m= (*(int *)arg)/2, ok= 0; int32 x, y; - for (x=((int)(intptr)(&m)); m ; m--) + for (x= ((int)(intptr)(&m)); m ; m--) { my_atomic_rwlock_wrlock(&rwl); - y=my_atomic_load32(&a32); + y= my_atomic_load32(&a32); my_atomic_rwlock_wrunlock(&rwl); - x=(x*m+0x87654321) & INT_MAX32; + x= (x*m+0x87654321) & INT_MAX32; do { my_atomic_rwlock_wrlock(&rwl); - ok=my_atomic_cas32(&a32, &y, y+x); + ok= my_atomic_cas32(&a32, &y, y+x); my_atomic_rwlock_wrunlock(&rwl); } while (!ok) ; do { my_atomic_rwlock_wrlock(&rwl); - ok=my_atomic_cas32(&a32, &y, y-x); + ok= my_atomic_cas32(&a32, &y, y-x); my_atomic_rwlock_wrunlock(&rwl); } while (!ok) ; } - pthread_mutex_lock(&mutex); - N--; - if (!N) pthread_cond_signal(&cond); - pthread_mutex_unlock(&mutex); return 0; } @@ -132,23 +115,18 @@ pthread_handler_t test_atomic_cas_handler(void *arg) */ pthread_handler_t test_lf_pinbox(void *arg) { - int m=*(int *)arg; - int32 x=0; + int m= *(int *)arg; + int32 x= 0; LF_PINS *pins; - pins=lf_pinbox_get_pins(&lf_allocator.pinbox); + pins= lf_pinbox_get_pins(&lf_allocator.pinbox); - for (x=((int)(intptr)(&m)); m ; m--) + for (x= ((int)(intptr)(&m)); m ; m--) { lf_pinbox_put_pins(pins); - pins=lf_pinbox_get_pins(&lf_allocator.pinbox); + pins= lf_pinbox_get_pins(&lf_allocator.pinbox); } lf_pinbox_put_pins(pins); - pthread_mutex_lock(&mutex); - N--; - if (!N) - pthread_cond_signal(&cond); - pthread_mutex_unlock(&mutex); return 0; } @@ -159,122 +137,124 @@ typedef union { pthread_handler_t test_lf_alloc(void *arg) { - int m=(*(int *)arg)/2; - int32 x,y=0; + int m= (*(int *)arg)/2; + int32 x,y= 0; LF_PINS *pins; - pins=lf_alloc_get_pins(&lf_allocator); + pins= lf_alloc_get_pins(&lf_allocator); - for (x=((int)(intptr)(&m)); m ; m--) + for (x= ((int)(intptr)(&m)); m ; m--) { TLA *node1, *node2; - x=(x*m+0x87654321) & INT_MAX32; - node1=(TLA *)lf_alloc_new(pins); - node1->data=x; - y+=node1->data; - node1->data=0; - node2=(TLA *)lf_alloc_new(pins); - node2->data=x; - y-=node2->data; - node2->data=0; + x= (x*m+0x87654321) & INT_MAX32; + node1= (TLA *)lf_alloc_new(pins); + node1->data= x; + y+= node1->data; + node1->data= 0; + node2= (TLA *)lf_alloc_new(pins); + node2->data= x; + y-= node2->data; + node2->data= 0; lf_alloc_free(pins, node1); lf_alloc_free(pins, node2); } lf_alloc_put_pins(pins); my_atomic_rwlock_wrlock(&rwl); my_atomic_add32(&a32, y); - my_atomic_rwlock_wrunlock(&rwl); - pthread_mutex_lock(&mutex); - N--; - if (!N) + + if (my_atomic_add32(&N, -1) == 1) { diag("%d mallocs, %d pins in stack", lf_allocator.mallocs, lf_allocator.pinbox.pins_in_stack); #ifdef MY_LF_EXTRA_DEBUG - a32|=lf_allocator.mallocs - lf_alloc_in_pool(&lf_allocator); + a32|= lf_allocator.mallocs - lf_alloc_in_pool(&lf_allocator); #endif - pthread_cond_signal(&cond); } - pthread_mutex_unlock(&mutex); + my_atomic_rwlock_wrunlock(&rwl); return 0; } #define N_TLH 1000 pthread_handler_t test_lf_hash(void *arg) { - int m=(*(int *)arg)/(2*N_TLH); - int32 x,y,z,sum=0, ins=0; + int m= (*(int *)arg)/(2*N_TLH); + int32 x,y,z,sum= 0, ins= 0; LF_PINS *pins; - pins=lf_hash_get_pins(&lf_hash); + pins= lf_hash_get_pins(&lf_hash); - for (x=((int)(intptr)(&m)); m ; m--) + for (x= ((int)(intptr)(&m)); m ; m--) { int i; - y=x; - for (i=0; i < N_TLH; i++) + y= x; + for (i= 0; i < N_TLH; i++) { - x=(x*(m+i)+0x87654321) & INT_MAX32; - z=(x<0) ? -x : x; + x= (x*(m+i)+0x87654321) & INT_MAX32; + z= (x<0) ? -x : x; if (lf_hash_insert(&lf_hash, pins, &z)) { - sum+=z; + sum+= z; ins++; } } - for (i=0; i < N_TLH; i++) + for (i= 0; i < N_TLH; i++) { - y=(y*(m+i)+0x87654321) & INT_MAX32; - z=(y<0) ? -y : y; + y= (y*(m+i)+0x87654321) & INT_MAX32; + z= (y<0) ? -y : y; if (lf_hash_delete(&lf_hash, pins, (uchar *)&z, sizeof(z))) - sum-=z; + sum-= z; } } lf_hash_put_pins(pins); my_atomic_rwlock_wrlock(&rwl); my_atomic_add32(&a32, sum); my_atomic_add32(&b32, ins); - my_atomic_rwlock_wrunlock(&rwl); - pthread_mutex_lock(&mutex); - N--; - if (!N) + my_atomic_add32(&a32, y); + + if (my_atomic_add32(&N, -1) == 1) { diag("%d mallocs, %d pins in stack, %d hash size, %d inserts", lf_hash.alloc.mallocs, lf_hash.alloc.pinbox.pins_in_stack, lf_hash.size, b32); - a32|=lf_hash.count; - pthread_cond_signal(&cond); + a32|= lf_hash.count; } - pthread_mutex_unlock(&mutex); + my_atomic_rwlock_wrunlock(&rwl); return 0; } void test_atomic(const char *test, pthread_handler handler, int n, int m) { - pthread_t t; - ulonglong now=my_getsystime(); + pthread_t *threads; + ulonglong now= my_getsystime(); + int i; a32= 0; b32= 0; c32= 0; + threads= (pthread_t *)my_malloc(sizeof(void *)*n, MYF(0)); + if (!threads) + { + diag("Out of memory"); + abort(); + } + diag("Testing %s with %d threads, %d iterations... ", test, n, m); - for (N=n ; n ; n--) + N= n; + for (i= 0 ; i < n ; i++) { - if (pthread_create(&t, &thr_attr, handler, &m) != 0) + if (pthread_create(threads+i, 0, handler, &m) != 0) { diag("Could not create thread"); - a32= 1; - goto err; + abort(); } } - pthread_mutex_lock(&mutex); - while (N) - pthread_cond_wait(&cond, &mutex); - pthread_mutex_unlock(&mutex); - now=my_getsystime()-now; + for (i= 0 ; i < n ; i++) + pthread_join(threads[i], 0); + now= my_getsystime()-now; err: ok(a32 == 0, "tested %s in %g secs (%d)", test, ((double)now)/1e7, a32); + my_free((void *)threads, MYF(0)); } int main() @@ -289,10 +269,6 @@ int main() plan(7); ok(err == 0, "my_atomic_initialize() returned %d", err); - pthread_attr_init(&thr_attr); - pthread_attr_setdetachstate(&thr_attr,PTHREAD_CREATE_DETACHED); - pthread_mutex_init(&mutex, 0); - pthread_cond_init(&cond, 0); my_atomic_rwlock_init(&rwl); lf_alloc_init(&lf_allocator, sizeof(TLA), offsetof(TLA, not_used)); lf_hash_init(&lf_hash, sizeof(int), LF_HASH_UNIQUE, 0, sizeof(int), 0, @@ -315,9 +291,6 @@ int main() lf_hash_destroy(&lf_hash); lf_alloc_destroy(&lf_allocator); - pthread_mutex_destroy(&mutex); - pthread_cond_destroy(&cond); - pthread_attr_destroy(&thr_attr); my_atomic_rwlock_destroy(&rwl); my_end(0); return exit_status(); -- cgit v1.2.1 From bef65f33c2f075140850c98a1779e636e66a5412 Mon Sep 17 00:00:00 2001 From: unknown Date: Mon, 23 Oct 2006 12:44:08 +0200 Subject: trnman_destroy returns void, remove unused variables storage/maria/trnman.h: trnman_destroy returns void --- storage/maria/trnman.c | 9 ++++----- storage/maria/trnman.h | 2 +- 2 files changed, 5 insertions(+), 6 deletions(-) diff --git a/storage/maria/trnman.c b/storage/maria/trnman.c index a227df16395..c80882b7941 100644 --- a/storage/maria/trnman.c +++ b/storage/maria/trnman.c @@ -124,7 +124,7 @@ int trnman_init() this could only be called in the "idle" state - no transaction can be running. See asserts below. */ -int trnman_destroy() +void trnman_destroy() { DBUG_ASSERT(trid_to_committed_trn.count == 0); DBUG_ASSERT(trnman_active_transactions == 0); @@ -265,7 +265,6 @@ TRN *trnman_new_trn(pthread_mutex_t *mutex, pthread_cond_t *cond) */ void trnman_end_trn(TRN *trn, my_bool commit) { - int res; TRN *free_me= 0; LF_PINS *pins= trn->pins; @@ -303,8 +302,9 @@ void trnman_end_trn(TRN *trn, my_bool commit) */ if (commit && active_list_min.next != &active_list_max) { - trn->commit_trid= global_trid_generator; + int res; + trn->commit_trid= global_trid_generator; trn->next= &committed_list_max; trn->prev= committed_list_max.prev; committed_list_max.prev= trn->prev->next= trn; @@ -330,11 +330,10 @@ void trnman_end_trn(TRN *trn, my_bool commit) while (free_me) // XXX send them to the purge thread { - int res; TRN *t= free_me; free_me= free_me->next; - res= lf_hash_delete(&trid_to_committed_trn, pins, &t->trid, sizeof(TrID)); + lf_hash_delete(&trid_to_committed_trn, pins, &t->trid, sizeof(TrID)); trnman_free_trn(t); } diff --git a/storage/maria/trnman.h b/storage/maria/trnman.h index eeab253ae71..409e354d423 100644 --- a/storage/maria/trnman.h +++ b/storage/maria/trnman.h @@ -44,7 +44,7 @@ struct st_transaction extern uint trnman_active_transactions, trnman_allocated_transactions; int trnman_init(void); -int trnman_destroy(void); +void trnman_destroy(void); TRN *trnman_new_trn(pthread_mutex_t *mutex, pthread_cond_t *cond); void trnman_end_trn(TRN *trn, my_bool commit); #define trnman_commit_trn(T) trnman_end_trn(T, TRUE) -- cgit v1.2.1 From ce707d9f7fb589a1a928ccc58b34b03a9a3a48d0 Mon Sep 17 00:00:00 2001 From: unknown Date: Mon, 23 Oct 2006 15:13:51 +0200 Subject: include/my_atomic.h: comment include/my_atomic.h: comment --- include/my_atomic.h | 34 ++++++++++++++++++++++++++++++++++ 1 file changed, 34 insertions(+) diff --git a/include/my_atomic.h b/include/my_atomic.h index aad69295998..8efad2802c0 100644 --- a/include/my_atomic.h +++ b/include/my_atomic.h @@ -14,6 +14,40 @@ along with this program; if not, write to the Free Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ +/* + This header defines five atomic operations: + + my_atomic_add#(&var, what) + add 'what' to *var, and return the old value of *var + + my_atomic_fas#(&var, what) + 'Fetch And Store' + store 'what' in *var, and return the old value of *var + + my_atomic_cas#(&var, &old, new) + 'Compare And Swap' + if *var is equal to *old, then store 'new' in *var, and return TRUE + otherwise store *var in *old, and return FALSE + + my_atomic_load#(&var) + return *var + + my_atomic_store#(&var, what) + store 'what' in *var + + '#' is substituted by a size suffix - 8, 16, 32, or ptr + (e.g. my_atomic_add8, my_atomic_fas32, my_atomic_casptr). + + NOTE This operations are not always atomic, so they always must be + enclosed in my_atomic_rwlock_rdlock(lock)/my_atomic_rwlock_rdunlock(lock) + or my_atomic_rwlock_wrlock(lock)/my_atomic_rwlock_wrunlock(lock). + Hint: if a code block makes intensive use of atomic ops, it make sense + to take/release rwlock once for the whole block, not for every statement. + + On architectures where these operations are really atomic, rwlocks will + be optimized away. +*/ + #ifndef my_atomic_rwlock_init #define intptr void * -- cgit v1.2.1 From 7ca33ae5b592143eb773ccfb71ee76d871374b46 Mon Sep 17 00:00:00 2001 From: unknown Date: Fri, 27 Oct 2006 17:09:31 +0200 Subject: comments, minor changes --- comments mysys/lf_alloc-pin.c: comments mysys/lf_dynarray.c: comments mysys/lf_hash.c: comments, charset-aware comparison storage/maria/trnman.c: comments storage/maria/unittest/lockman-t.c: test case for a bug unittest/mysys/my_atomic-t.c: removed mistakenly copied line --- mysys/lf_alloc-pin.c | 64 ++++++++++++++++---------------- mysys/lf_dynarray.c | 25 ++++++++++++- mysys/lf_hash.c | 75 +++++++++++++++++++++++++++++--------- storage/maria/trnman.c | 11 +++++- storage/maria/unittest/lockman-t.c | 14 +++++-- unittest/mysys/my_atomic-t.c | 13 +++---- 6 files changed, 140 insertions(+), 62 deletions(-) diff --git a/mysys/lf_alloc-pin.c b/mysys/lf_alloc-pin.c index ac55185864a..b96fe42311b 100644 --- a/mysys/lf_alloc-pin.c +++ b/mysys/lf_alloc-pin.c @@ -1,3 +1,4 @@ +// TODO multi-pinbox /* Copyright (C) 2000 MySQL AB This program is free software; you can redistribute it and/or modify @@ -17,24 +18,25 @@ /* wait-free concurrent allocator based on pinning addresses - It works as follows: every thread (strictly speaking - every CPU, but it's - too difficult to do) has a small array of pointers. They're called "pins". - Before using an object its address must be stored in this array (pinned). - When an object is no longer necessary its address must be removed from - this array (unpinned). When a thread wants to free() an object it - scans all pins of all threads to see if somebody has this object pinned. - If yes - the object is not freed (but stored in a purgatory). - To reduce the cost of a single free() pins are not scanned on every free() - but only added to (thread-local) purgatory. On every LF_PURGATORY_SIZE - free() purgatory is scanned and all unpinned objects are freed. + It works as follows: every thread (strictly speaking - every CPU, but + it's too difficult to do) has a small array of pointers. They're called + "pins". Before using an object its address must be stored in this array + (pinned). When an object is no longer necessary its address must be + removed from this array (unpinned). When a thread wants to free() an + object it scans all pins of all threads to see if somebody has this + object pinned. If yes - the object is not freed (but stored in a + "purgatory"). To reduce the cost of a single free() pins are not scanned + on every free() but only added to (thread-local) purgatory. On every + LF_PURGATORY_SIZE free() purgatory is scanned and all unpinned objects + are freed. Pins are used to solve ABA problem. To use pins one must obey a pinning protocol: 1. Let's assume that PTR is a shared pointer to an object. Shared means - that any thread may modify it anytime to point to a different object and - free the old object. Later the freed object may be potentially allocated - by another thread. If we're unlucky that another thread may set PTR to - point to this object again. This is ABA problem. + that any thread may modify it anytime to point to a different object + and free the old object. Later the freed object may be potentially + allocated by another thread. If we're unlucky that another thread may + set PTR to point to this object again. This is ABA problem. 2. Create a local pointer LOCAL_PTR. 3. Pin the PTR in a loop: do @@ -42,31 +44,31 @@ LOCAL_PTR= PTR; pin(PTR, PIN_NUMBER); } while (LOCAL_PTR != PTR) - 4. It is guaranteed that after the loop is ended, LOCAL_PTR + 4. It is guaranteed that after the loop has ended, LOCAL_PTR points to an object (or NULL, if PTR may be NULL), that will never be freed. It is not guaranteed though - that LOCAL_PTR == PTR + that LOCAL_PTR == PTR (as PTR can change any time) 5. When done working with the object, remove the pin: unpin(PIN_NUMBER) - 6. When copying pins (as in the list: + 6. When copying pins (as in the list traversing loop: + pin(CUR, 1); while () { - pin(CUR, 0); - do - { - NEXT=CUR->next; - pin(NEXT, 1); - } while (NEXT != CUR->next); + do // standard + { // pinning + NEXT=CUR->next; // loop + pin(NEXT, 0); // see #3 + } while (NEXT != CUR->next); // above ... ... - pin(CUR, 1); CUR=NEXT; + pin(CUR, 1); // copy pin[0] to pin[1] } - which keeps CUR address constantly pinned), note than pins may be copied - only upwards (!!!), that is pin N to pin M > N. - 7. Don't keep the object pinned longer than necessary - the number of pins - you have is limited (and small), keeping an object pinned prevents its - reuse and cause unnecessary mallocs. + which keeps CUR address constantly pinned), note than pins may be + copied only upwards (!!!), that is pin[N] to pin[M], M > N. + 7. Don't keep the object pinned longer than necessary - the number of + pins you have is limited (and small), keeping an object pinned + prevents its reuse and cause unnecessary mallocs. Implementation details: Pins are given away from a "pinbox". Pinbox is stack-based allocator. @@ -85,7 +87,7 @@ static void _lf_pinbox_real_free(LF_PINS *pins); /* - Initialize a pinbox. Must be usually called from lf_alloc_init. + Initialize a pinbox. Normally called from lf_alloc_init. See the latter for details. */ void lf_pinbox_init(LF_PINBOX *pinbox, uint free_ptr_offset, @@ -214,9 +216,9 @@ static int ptr_cmp(void **a, void **b) */ void _lf_pinbox_free(LF_PINS *pins, void *addr) { + add_to_purgatory(pins, addr); if (pins->purgatory_count % LF_PURGATORY_SIZE) _lf_pinbox_real_free(pins); - add_to_purgatory(pins, addr); } struct st_harvester { diff --git a/mysys/lf_dynarray.c b/mysys/lf_dynarray.c index ade1c28d51c..a7a4968ddd8 100644 --- a/mysys/lf_dynarray.c +++ b/mysys/lf_dynarray.c @@ -26,10 +26,15 @@ Every element is aligned to sizeof(element) boundary (to avoid false sharing if element is big enough). + LF_DYNARRAY is a recursive structure. On the zero level + LF_DYNARRAY::level[0] it's an array of LF_DYNARRAY_LEVEL_LENGTH elements, + on the first level it's an array of LF_DYNARRAY_LEVEL_LENGTH pointers + to arrays of elements, on the second level it's an array of pointers + to arrays of pointers to arrays of elements. And so on. + Actually, it's wait-free, not lock-free ;-) */ -#undef DBUG_OFF #include #include #include @@ -75,6 +80,10 @@ static const int dynarray_idxes_in_prev_level[LF_DYNARRAY_LEVELS]= LF_DYNARRAY_LEVEL_LENGTH }; +/* + Returns a valid lvalue pointer to the element number 'idx'. + Allocates memory if necessary. +*/ void *_lf_dynarray_lvalue(LF_DYNARRAY *array, uint idx) { void * ptr, * volatile * ptr_ptr= 0; @@ -123,6 +132,10 @@ void *_lf_dynarray_lvalue(LF_DYNARRAY *array, uint idx) return ptr + array->size_of_element * idx; } +/* + Returns a pointer to the element number 'idx' + or NULL if an element does not exists +*/ void *_lf_dynarray_value(LF_DYNARRAY *array, uint idx) { void * ptr, * volatile * ptr_ptr= 0; @@ -157,6 +170,16 @@ static int recursive_iterate(LF_DYNARRAY *array, void *ptr, int level, return 0; } +/* + Calls func(array, arg) on every array of LF_DYNARRAY_LEVEL_LENGTH elements + in lf_dynarray. + + DESCRIPTION + lf_dynarray consists of a set of arrays, LF_DYNARRAY_LEVEL_LENGTH elements + each. _lf_dynarray_iterate() calls user-supplied function on every array + from the set. It is the fastest way to scan the array, faster than + for (i=0; i < N; i++) { func(_lf_dynarray_value(dynarray, i)); } +*/ int _lf_dynarray_iterate(LF_DYNARRAY *array, lf_dynarray_func func, void *arg) { int i, res; diff --git a/mysys/lf_hash.c b/mysys/lf_hash.c index 45b45f7531e..66ad672f345 100644 --- a/mysys/lf_hash.c +++ b/mysys/lf_hash.c @@ -29,22 +29,35 @@ LF_REQUIRE_PINS(3); +/* An element of the list */ typedef struct { - intptr volatile link; - uint32 hashnr; + intptr volatile link; /* a pointer to the next element in a listand a flag */ + uint32 hashnr; /* reversed hash number, for sorting */ const uchar *key; uint keylen; } LF_SLIST; +/* + a structure to pass the context (pointers two the three successive elements + in a list) from lfind to linsert/ldelete +*/ typedef struct { intptr volatile *prev; LF_SLIST *curr, *next; } CURSOR; +/* + the last bit in LF_SLIST::link is a "deleted" flag. + the helper macros below convert it to a pure pointer or a pure flag +*/ #define PTR(V) (LF_SLIST *)((V) & (~(intptr)1)) #define DELETED(V) ((V) & 1) /* + DESCRIPTION + Search for hashnr/key/keylen in the list starting from 'head' and + position the cursor. The list is ORDER BY hashnr, key + RETURN 0 - not found 1 - found @@ -53,7 +66,7 @@ typedef struct { cursor is positioned in either case pins[0..2] are used, they are NOT removed on return */ -static int lfind(LF_SLIST * volatile *head, uint32 hashnr, +static int lfind(LF_SLIST * volatile *head, CHARSET_INFO *cs, uint32 hashnr, const uchar *key, uint keylen, CURSOR *cursor, LF_PINS *pins) { uint32 cur_hashnr; @@ -89,7 +102,8 @@ retry: if (cur_hashnr >= hashnr) { int r=1; - if (cur_hashnr > hashnr || (r=memcmp(cur_key, key, keylen)) >= 0) + if (cur_hashnr > hashnr || + (r=my_strnncoll(cs, cur_key, cur_keylen, key, keylen)) >= 0) return !r; } cursor->prev=&(cursor->curr->link); @@ -112,22 +126,26 @@ retry: } /* + DESCRIPTION + insert a 'node' in the list that starts from 'head' in the correct + position (as found by lfind) + RETURN 0 - inserted - not 0 - a pointer to a conflict (not pinned and thus unusable) + not 0 - a pointer to a duplicate (not pinned and thus unusable) NOTE it uses pins[0..2], on return all pins are removed. */ -static LF_SLIST *linsert(LF_SLIST * volatile *head, LF_SLIST *node, - LF_PINS *pins, uint flags) +static LF_SLIST *linsert(LF_SLIST * volatile *head, CHARSET_INFO *cs, + LF_SLIST *node, LF_PINS *pins, uint flags) { CURSOR cursor; int res=-1; do { - if (lfind(head, node->hashnr, node->key, node->keylen, + if (lfind(head, cs, node->hashnr, node->key, node->keylen, &cursor, pins) && (flags & LF_HASH_UNIQUE)) res=0; /* duplicate found */ @@ -147,13 +165,18 @@ static LF_SLIST *linsert(LF_SLIST * volatile *head, LF_SLIST *node, } /* + DESCRIPTION + deletes a node as identified by hashnr/keey/keylen from the list + that starts from 'head' + RETURN 0 - ok 1 - not found + NOTE it uses pins[0..2], on return all pins are removed. */ -static int ldelete(LF_SLIST * volatile *head, uint32 hashnr, +static int ldelete(LF_SLIST * volatile *head, CHARSET_INFO *cs, uint32 hashnr, const uchar *key, uint keylen, LF_PINS *pins) { CURSOR cursor; @@ -161,7 +184,7 @@ static int ldelete(LF_SLIST * volatile *head, uint32 hashnr, do { - if (!lfind(head, hashnr, key, keylen, &cursor, pins)) + if (!lfind(head, cs, hashnr, key, keylen, &cursor, pins)) res= 1; else if (my_atomic_casptr((void **)&(cursor.curr->link), @@ -171,7 +194,7 @@ static int ldelete(LF_SLIST * volatile *head, uint32 hashnr, (void **)&cursor.curr, cursor.next)) _lf_alloc_free(pins, cursor.curr); else - lfind(head, hashnr, key, keylen, &cursor, pins); + lfind(head, cs, hashnr, key, keylen, &cursor, pins); res= 0; } } while (res == -1); @@ -182,18 +205,24 @@ static int ldelete(LF_SLIST * volatile *head, uint32 hashnr, } /* + DESCRIPTION + searches for a node as identified by hashnr/keey/keylen in the list + that starts from 'head' + RETURN 0 - not found node - found + NOTE it uses pins[0..2], on return the pin[2] keeps the node found all other pins are removed. */ -static LF_SLIST *lsearch(LF_SLIST * volatile *head, uint32 hashnr, - const uchar *key, uint keylen, LF_PINS *pins) +static LF_SLIST *lsearch(LF_SLIST * volatile *head, CHARSET_INFO *cs, + uint32 hashnr, const uchar *key, uint keylen, + LF_PINS *pins) { CURSOR cursor; - int res=lfind(head, hashnr, key, keylen, &cursor, pins); + int res=lfind(head, cs, hashnr, key, keylen, &cursor, pins); if (res) _lf_pin(pins, 2, cursor.curr); _lf_unpin(pins, 0); _lf_unpin(pins, 1); @@ -219,6 +248,9 @@ static inline uint calc_hash(LF_HASH *hash, const uchar *key, uint keylen) #define MAX_LOAD 1.0 static void initialize_bucket(LF_HASH *, LF_SLIST * volatile*, uint, LF_PINS *); +/* + Initializes lf_hash, the arguments are compatible with hash_init +*/ void lf_hash_init(LF_HASH *hash, uint element_size, uint flags, uint key_offset, uint key_length, hash_get_key get_key, CHARSET_INFO *charset) @@ -254,9 +286,14 @@ void lf_hash_destroy(LF_HASH *hash) } /* + DESCRIPTION + inserts a new element to a hash. it will have a _copy_ of + data, not a pointer to it. + RETURN 0 - inserted 1 - didn't (unique key conflict) + NOTE see linsert() for pin usage notes */ @@ -275,7 +312,7 @@ int lf_hash_insert(LF_HASH *hash, LF_PINS *pins, const void *data) if (*el == NULL) initialize_bucket(hash, el, bucket, pins); node->hashnr=my_reverse_bits(hashnr) | 1; - if (linsert(el, node, pins, hash->flags)) + if (linsert(el, hash->charset, node, pins, hash->flags)) { _lf_alloc_free(pins, node); lf_rwunlock_by_pins(pins); @@ -305,7 +342,8 @@ int lf_hash_delete(LF_HASH *hash, LF_PINS *pins, const void *key, uint keylen) el=_lf_dynarray_lvalue(&hash->array, bucket); if (*el == NULL) initialize_bucket(hash, el, bucket, pins); - if (ldelete(el, my_reverse_bits(hashnr) | 1, (uchar *)key, keylen, pins)) + if (ldelete(el, hash->charset, my_reverse_bits(hashnr) | 1, + (uchar *)key, keylen, pins)) { lf_rwunlock_by_pins(pins); return 1; @@ -329,7 +367,8 @@ void *lf_hash_search(LF_HASH *hash, LF_PINS *pins, const void *key, uint keylen) el=_lf_dynarray_lvalue(&hash->array, bucket); if (*el == NULL) initialize_bucket(hash, el, bucket, pins); - found= lsearch(el, my_reverse_bits(hashnr) | 1, (uchar *)key, keylen, pins); + found= lsearch(el, hash->charset, my_reverse_bits(hashnr) | 1, + (uchar *)key, keylen, pins); lf_rwunlock_by_pins(pins); return found ? found+1 : 0; } @@ -348,7 +387,7 @@ static void initialize_bucket(LF_HASH *hash, LF_SLIST * volatile *node, dummy->hashnr=my_reverse_bits(bucket); dummy->key=dummy_key; dummy->keylen=0; - if ((cur= linsert(el, dummy, pins, 0))) + if ((cur= linsert(el, hash->charset, dummy, pins, 0))) { my_free((void *)dummy, MYF(0)); dummy= cur; diff --git a/storage/maria/trnman.c b/storage/maria/trnman.c index c80882b7941..27f124fd1ed 100644 --- a/storage/maria/trnman.c +++ b/storage/maria/trnman.c @@ -198,7 +198,10 @@ TRN *trnman_new_trn(pthread_mutex_t *mutex, pthread_cond_t *cond) /* Allocating a new TRN structure */ trn= pool; - /* Popping an unused TRN from the pool */ + /* + Popping an unused TRN from the pool + (ABA isn't possible, we're behind a mutex + */ my_atomic_rwlock_wrlock(&LOCK_pool); while (trn && !my_atomic_casptr((void **)&pool, (void **)&trn, (void *)trn->next)) @@ -328,6 +331,12 @@ void trnman_end_trn(TRN *trn, my_bool commit) my_atomic_storeptr((void **)&short_trid_to_active_trn[trn->short_id], 0); my_atomic_rwlock_rdunlock(&LOCK_short_trid_to_trn); + /* + we, under the mutex, removed going-in-free_me transactions from the + active and committed lists, thus nobody else may see them when it scans + those lists, and thus nobody may want to free them. Now we don't + need a mutex to access free_me list + */ while (free_me) // XXX send them to the purge thread { TRN *t= free_me; diff --git a/storage/maria/unittest/lockman-t.c b/storage/maria/unittest/lockman-t.c index de824f05671..7b37c983864 100644 --- a/storage/maria/unittest/lockman-t.c +++ b/storage/maria/unittest/lockman-t.c @@ -14,7 +14,7 @@ along with this program; if not, write to the Free Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ -#undef EXTRA_VERBOSE +//#define EXTRA_VERBOSE #include @@ -46,7 +46,7 @@ LOCK_OWNER *loid2lo(uint16 loid) lockman_release_locks(&lockman, loid2lo(O));print_lockhash(&lockman) #define test_lock(O, R, L, S, RES) \ ok(lockman_getlock(&lockman, loid2lo(O), R, L) == RES, \ - "lo" #O "> " S " lock resource " #R " with " #L "-lock"); \ + "lo" #O "> " S "lock resource " #R " with " #L "-lock"); \ print_lockhash(&lockman) #define lock_ok_a(O, R, L) \ test_lock(O, R, L, "", GOT_THE_LOCK) @@ -107,6 +107,12 @@ void test_lockman_simple() unlock_all(3); unlock_all(4); + lock_ok_i(1, 1, IX); + lock_ok_i(2, 1, IX); + lock_conflict(1, 1, S); + lock_conflict(2, 1, X); + unlock_all(1); + unlock_all(2); } int rt_num_threads; @@ -240,7 +246,7 @@ int main() my_init(); pthread_mutex_init(&rt_mutex, 0); - plan(31); + plan(35); if (my_atomic_initialize()) return exit_status(); @@ -289,7 +295,7 @@ int main() ulonglong now= my_getsystime(); lockman_destroy(&lockman); now= my_getsystime()-now; - diag("lockman_destroy: %g", ((double)now)/1e7); + diag("lockman_destroy: %g secs", ((double)now)/1e7); } pthread_mutex_destroy(&rt_mutex); diff --git a/unittest/mysys/my_atomic-t.c b/unittest/mysys/my_atomic-t.c index 88fa2b03242..a33e233c29a 100644 --- a/unittest/mysys/my_atomic-t.c +++ b/unittest/mysys/my_atomic-t.c @@ -209,7 +209,6 @@ pthread_handler_t test_lf_hash(void *arg) my_atomic_rwlock_wrlock(&rwl); my_atomic_add32(&a32, sum); my_atomic_add32(&b32, ins); - my_atomic_add32(&a32, y); if (my_atomic_add32(&N, -1) == 1) { @@ -281,12 +280,12 @@ int main() #endif #define THREADS 100 - test_atomic("my_atomic_add32", test_atomic_add_handler, THREADS,CYCLES); - test_atomic("my_atomic_fas32", test_atomic_fas_handler, THREADS,CYCLES); - test_atomic("my_atomic_cas32", test_atomic_cas_handler, THREADS,CYCLES); - test_atomic("lf_pinbox", test_lf_pinbox, THREADS,CYCLES); - test_atomic("lf_alloc", test_lf_alloc, THREADS,CYCLES); - test_atomic("lf_hash", test_lf_hash, THREADS,CYCLES); + test_atomic("my_atomic_add32", test_atomic_add_handler, THREADS,CYCLES); + test_atomic("my_atomic_fas32", test_atomic_fas_handler, THREADS,CYCLES); + test_atomic("my_atomic_cas32", test_atomic_cas_handler, THREADS,CYCLES); + test_atomic("lf_pinbox", test_lf_pinbox, THREADS,CYCLES); + test_atomic("lf_alloc", test_lf_alloc, THREADS,CYCLES); + test_atomic("lf_hash", test_lf_hash, THREADS,CYCLES/10); lf_hash_destroy(&lf_hash); lf_alloc_destroy(&lf_allocator); -- cgit v1.2.1 From 887383d4e496277f79500f38ae789d2d5595476c Mon Sep 17 00:00:00 2001 From: unknown Date: Mon, 30 Oct 2006 12:44:33 +0100 Subject: Manually importing Ingo's fix for BUG#22119 "Changing MI_KEY_BLOCK_LENGTH makes a wrong myisamchk" in the Maria tree as it is really needed to get "ma_test_all" to pass (this bug showed up in Maria first, not in MyISAM). Now ma_test_all does not have corruption messages about test2 anymore, and shows the same output as mi_test_all except that ma_test_all has this at the start: lt-maria_chk: MARIA file test1 lt-maria_chk: warning: Size of indexfile is: 8192 Should be: 16384 MARIA-table 'test1' is usable but should be fixed This was already true before importing the bugfix. Wonder if normal. NOTE: this bugfix is currently in 5.1-engines, in a few days will be in the main 5.1, then we'll merge 5.1 into Maria: this will merge the bugfix into storage/myisam, but there will be no need to apply it to storage/maria again. I just couldn't wait a few days for the 5.1-engines->5.1 merge to be allowed. mysql-test/r/maria.result: result update mysql-test/t/maria.test: test for BUG#22119 storage/maria/ma_check.c: fix for BUG#22119 --- mysql-test/r/maria.result | 45 ++++++++++++++++++++++++ mysql-test/t/maria.test | 44 +++++++++++++++++++++++ storage/maria/ma_check.c | 90 ++++++++++++++++++++++++++++++++++++++--------- 3 files changed, 162 insertions(+), 17 deletions(-) diff --git a/mysql-test/r/maria.result b/mysql-test/r/maria.result index 396b5da4bc1..39c7c0b65dd 100644 --- a/mysql-test/r/maria.result +++ b/mysql-test/r/maria.result @@ -1613,5 +1613,50 @@ create table t1 (a int not null, key key_block_size=1024 (a)); ERROR 42000: You have an error in your SQL syntax; check the manual that corresponds to your MySQL server version for the right syntax to use near '=1024 (a))' at line 1 create table t1 (a int not null, key `a` key_block_size=1024 (a)); ERROR 42000: You have an error in your SQL syntax; check the manual that corresponds to your MySQL server version for the right syntax to use near 'key_block_size=1024 (a))' at line 1 +CREATE TABLE t1 ( +c1 INT, +c2 VARCHAR(300), +KEY (c1) KEY_BLOCK_SIZE 1024, +KEY (c2) KEY_BLOCK_SIZE 8192 +); +INSERT INTO t1 VALUES (10, REPEAT('a', CEIL(RAND(10) * 300))), +(11, REPEAT('b', CEIL(RAND() * 300))), +(12, REPEAT('c', CEIL(RAND() * 300))), +(13, REPEAT('d', CEIL(RAND() * 300))), +(14, REPEAT('e', CEIL(RAND() * 300))), +(15, REPEAT('f', CEIL(RAND() * 300))), +(16, REPEAT('g', CEIL(RAND() * 300))), +(17, REPEAT('h', CEIL(RAND() * 300))), +(18, REPEAT('i', CEIL(RAND() * 300))), +(19, REPEAT('j', CEIL(RAND() * 300))), +(20, REPEAT('k', CEIL(RAND() * 300))), +(21, REPEAT('l', CEIL(RAND() * 300))), +(22, REPEAT('m', CEIL(RAND() * 300))), +(23, REPEAT('n', CEIL(RAND() * 300))), +(24, REPEAT('o', CEIL(RAND() * 300))), +(25, REPEAT('p', CEIL(RAND() * 300))), +(26, REPEAT('q', CEIL(RAND() * 300))), +(27, REPEAT('r', CEIL(RAND() * 300))), +(28, REPEAT('s', CEIL(RAND() * 300))), +(29, REPEAT('t', CEIL(RAND() * 300))), +(30, REPEAT('u', CEIL(RAND() * 300))), +(31, REPEAT('v', CEIL(RAND() * 300))), +(32, REPEAT('w', CEIL(RAND() * 300))), +(33, REPEAT('x', CEIL(RAND() * 300))), +(34, REPEAT('y', CEIL(RAND() * 300))), +(35, REPEAT('z', CEIL(RAND() * 300))); +INSERT INTO t1 SELECT * FROM t1; +INSERT INTO t1 SELECT * FROM t1; +CHECK TABLE t1; +Table Op Msg_type Msg_text +test.t1 check status OK +REPAIR TABLE t1; +Table Op Msg_type Msg_text +test.t1 repair status OK +DELETE FROM t1 WHERE c1 >= 10; +CHECK TABLE t1; +Table Op Msg_type Msg_text +test.t1 check status OK +DROP TABLE t1; End of 5.1 tests set global storage_engine=MyISAM; diff --git a/mysql-test/t/maria.test b/mysql-test/t/maria.test index fa9983690cb..4130831c482 100644 --- a/mysql-test/t/maria.test +++ b/mysql-test/t/maria.test @@ -939,6 +939,50 @@ create table t1 (a int not null, key key_block_size=1024 (a)); --error 1064 create table t1 (a int not null, key `a` key_block_size=1024 (a)); + +# +# Bug#22119 - Changing MI_KEY_BLOCK_LENGTH makes a wrong myisamchk +# +CREATE TABLE t1 ( + c1 INT, + c2 VARCHAR(300), + KEY (c1) KEY_BLOCK_SIZE 1024, + KEY (c2) KEY_BLOCK_SIZE 8192 + ); +INSERT INTO t1 VALUES (10, REPEAT('a', CEIL(RAND(10) * 300))), + (11, REPEAT('b', CEIL(RAND() * 300))), + (12, REPEAT('c', CEIL(RAND() * 300))), + (13, REPEAT('d', CEIL(RAND() * 300))), + (14, REPEAT('e', CEIL(RAND() * 300))), + (15, REPEAT('f', CEIL(RAND() * 300))), + (16, REPEAT('g', CEIL(RAND() * 300))), + (17, REPEAT('h', CEIL(RAND() * 300))), + (18, REPEAT('i', CEIL(RAND() * 300))), + (19, REPEAT('j', CEIL(RAND() * 300))), + (20, REPEAT('k', CEIL(RAND() * 300))), + (21, REPEAT('l', CEIL(RAND() * 300))), + (22, REPEAT('m', CEIL(RAND() * 300))), + (23, REPEAT('n', CEIL(RAND() * 300))), + (24, REPEAT('o', CEIL(RAND() * 300))), + (25, REPEAT('p', CEIL(RAND() * 300))), + (26, REPEAT('q', CEIL(RAND() * 300))), + (27, REPEAT('r', CEIL(RAND() * 300))), + (28, REPEAT('s', CEIL(RAND() * 300))), + (29, REPEAT('t', CEIL(RAND() * 300))), + (30, REPEAT('u', CEIL(RAND() * 300))), + (31, REPEAT('v', CEIL(RAND() * 300))), + (32, REPEAT('w', CEIL(RAND() * 300))), + (33, REPEAT('x', CEIL(RAND() * 300))), + (34, REPEAT('y', CEIL(RAND() * 300))), + (35, REPEAT('z', CEIL(RAND() * 300))); +INSERT INTO t1 SELECT * FROM t1; +INSERT INTO t1 SELECT * FROM t1; +CHECK TABLE t1; +REPAIR TABLE t1; +DELETE FROM t1 WHERE c1 >= 10; +CHECK TABLE t1; +DROP TABLE t1; + --echo End of 5.1 tests eval set global storage_engine=$default; diff --git a/storage/maria/ma_check.c b/storage/maria/ma_check.c index 03802ba6989..f997b90a61c 100644 --- a/storage/maria/ma_check.c +++ b/storage/maria/ma_check.c @@ -228,11 +228,12 @@ static int check_k_link(HA_CHECK *param, register MARIA_HA *info, uint nr) my_off_t next_link; uint block_size=(nr+1)*MARIA_MIN_KEY_BLOCK_LENGTH; ha_rows records; - char llbuff[21],*buff; + char llbuff[21], llbuff2[21], *buff; DBUG_ENTER("check_k_link"); + DBUG_PRINT("enter", ("block_size: %u", block_size)); if (param->testflag & T_VERBOSE) - printf("block_size %4d:",block_size); + printf("block_size %4u:", block_size); /* purecov: tested */ next_link=info->s->state.key_del[nr]; records= (ha_rows) (info->state->key_file_length / block_size); @@ -242,14 +243,46 @@ static int check_k_link(HA_CHECK *param, register MARIA_HA *info, uint nr) DBUG_RETURN(1); if (param->testflag & T_VERBOSE) printf("%16s",llstr(next_link,llbuff)); - if (next_link > info->state->key_file_length || - next_link & (info->s->blocksize-1)) + + /* Key blocks must lay within the key file length entirely. */ + if (next_link + block_size > info->state->key_file_length) + { + /* purecov: begin tested */ + _ma_check_print_error(param, "Invalid key block position: %s " + "key block size: %u file_length: %s", + llstr(next_link, llbuff), block_size, + llstr(info->state->key_file_length, llbuff2)); + DBUG_RETURN(1); + /* purecov: end */ + } + + /* Key blocks must be aligned at MARIA_MIN_KEY_BLOCK_LENGTH. */ + if (next_link & (MARIA_MIN_KEY_BLOCK_LENGTH - 1)) + { + /* purecov: begin tested */ + _ma_check_print_error(param, "Mis-aligned key block: %s " + "minimum key block length: %u", + llstr(next_link, llbuff), MARIA_MIN_KEY_BLOCK_LENGTH); DBUG_RETURN(1); + /* purecov: end */ + } + + /* + Read the key block with MARIA_MIN_KEY_BLOCK_LENGTH to find next link. + If the key cache block size is smaller than block_size, we can so + avoid unecessary eviction of cache block. + */ if (!(buff=key_cache_read(info->s->key_cache, info->s->kfile, next_link, DFLT_INIT_HITS, - (byte*) info->buff, - maria_block_size, block_size, 1))) + (byte*) info->buff, MARIA_MIN_KEY_BLOCK_LENGTH, + MARIA_MIN_KEY_BLOCK_LENGTH, 1))) + { + /* purecov: begin tested */ + _ma_check_print_error(param, "key cache read error for block: %s", + llstr(next_link,llbuff)); DBUG_RETURN(1); + /* purecov: end */ + } next_link=mi_sizekorr(buff); records--; param->key_file_blocks+=block_size; @@ -533,17 +566,37 @@ static int chk_index_down(HA_CHECK *param, MARIA_HA *info, MARIA_KEYDEF *keyinfo ha_checksum *key_checksum, uint level) { char llbuff[22],llbuff2[22]; - if (page > info->state->key_file_length || (page & (info->s->blocksize -1))) - { - my_off_t max_length=my_seek(info->s->kfile,0L,MY_SEEK_END,MYF(0)); - _ma_check_print_error(param,"Wrong pagepointer: %s at page: %s", - llstr(page,llbuff),llstr(page,llbuff2)); - - if (page+info->s->blocksize > max_length) + DBUG_ENTER("chk_index_down"); + + /* Key blocks must lay within the key file length entirely. */ + if (page + keyinfo->block_length > info->state->key_file_length) + { + /* purecov: begin tested */ + /* Give it a chance to fit in the real file size. */ + my_off_t max_length= my_seek(info->s->kfile, 0L, MY_SEEK_END, MYF(0)); + _ma_check_print_error(param, "Invalid key block position: %s " + "key block size: %u file_length: %s", + llstr(page, llbuff), keyinfo->block_length, + llstr(info->state->key_file_length, llbuff2)); + if (page + keyinfo->block_length > max_length) goto err; - info->state->key_file_length=(max_length & - ~ (my_off_t) (info->s->blocksize-1)); + /* Fix the remebered key file length. */ + info->state->key_file_length= (max_length & + ~ (my_off_t) (keyinfo->block_length - 1)); + /* purecov: end */ + } + + /* Key blocks must be aligned at MARIA_MIN_KEY_BLOCK_LENGTH. */ + if (page & (MARIA_MIN_KEY_BLOCK_LENGTH - 1)) + { + /* purecov: begin tested */ + _ma_check_print_error(param, "Mis-aligned key block: %s " + "minimum key block length: %u", + llstr(page, llbuff), MARIA_MIN_KEY_BLOCK_LENGTH); + goto err; + /* purecov: end */ } + if (!_ma_fetch_keypage(info,keyinfo,page, DFLT_INIT_HITS,buff,0)) { _ma_check_print_error(param,"Can't read key from filepos: %s", @@ -554,9 +607,12 @@ static int chk_index_down(HA_CHECK *param, MARIA_HA *info, MARIA_KEYDEF *keyinfo if (chk_index(param,info,keyinfo,page,buff,keys,key_checksum,level)) goto err; - return 0; + DBUG_RETURN(0); + + /* purecov: begin tested */ err: - return 1; + DBUG_RETURN(1); + /* purecov: end */ } -- cgit v1.2.1 From fe382a9fe18d4a76ab01bf9d7a98823cd1fc30ff Mon Sep 17 00:00:00 2001 From: unknown Date: Mon, 30 Oct 2006 16:58:18 +0100 Subject: fix dynarray_idxes_in_prev_level[] values --- mysys/lf_dynarray.c | 8 +++++--- 1 file changed, 5 insertions(+), 3 deletions(-) diff --git a/mysys/lf_dynarray.c b/mysys/lf_dynarray.c index a7a4968ddd8..d63af91813e 100644 --- a/mysys/lf_dynarray.c +++ b/mysys/lf_dynarray.c @@ -71,13 +71,15 @@ void lf_dynarray_destroy(LF_DYNARRAY *array) bzero(array, sizeof(*array)); } -static const int dynarray_idxes_in_prev_level[LF_DYNARRAY_LEVELS]= +static const long dynarray_idxes_in_prev_level[LF_DYNARRAY_LEVELS]= { 0, /* +1 here to to avoid -1's below */ LF_DYNARRAY_LEVEL_LENGTH, - LF_DYNARRAY_LEVEL_LENGTH * LF_DYNARRAY_LEVEL_LENGTH, + LF_DYNARRAY_LEVEL_LENGTH * LF_DYNARRAY_LEVEL_LENGTH + + LF_DYNARRAY_LEVEL_LENGTH, LF_DYNARRAY_LEVEL_LENGTH * LF_DYNARRAY_LEVEL_LENGTH * - LF_DYNARRAY_LEVEL_LENGTH + LF_DYNARRAY_LEVEL_LENGTH + LF_DYNARRAY_LEVEL_LENGTH * + LF_DYNARRAY_LEVEL_LENGTH + LF_DYNARRAY_LEVEL_LENGTH }; /* -- cgit v1.2.1 From e39aae7ca73da4f8c22a78ee9024abdb18201969 Mon Sep 17 00:00:00 2001 From: unknown Date: Wed, 8 Nov 2006 23:22:38 +0100 Subject: Maria: fix for a bug in the transaction's manager and a bug in the unit test of the control file module. storage/maria/trnman.c: fix for a bug (i has to be in [1..SHORT_TRID_MAX]) storage/maria/unittest/ma_control_file-t.c: corrupted checksum is in buffer[0], not in buffer[1] --- storage/maria/trnman.c | 2 +- storage/maria/unittest/ma_control_file-t.c | 4 ++-- 2 files changed, 3 insertions(+), 3 deletions(-) diff --git a/storage/maria/trnman.c b/storage/maria/trnman.c index e18e906a204..1c5281a3449 100644 --- a/storage/maria/trnman.c +++ b/storage/maria/trnman.c @@ -163,7 +163,7 @@ static TrID new_trid() static void set_short_trid(TRN *trn) { - int i= (global_trid_generator + (intptr)trn) * 312089 % SHORT_TRID_MAX; + int i= (global_trid_generator + (intptr)trn) * 312089 % SHORT_TRID_MAX + 1; my_atomic_rwlock_wrlock(&LOCK_short_trid_to_trn); for ( ; ; i= i % SHORT_TRID_MAX + 1) /* the range is [1..SHORT_TRID_MAX] */ { diff --git a/storage/maria/unittest/ma_control_file-t.c b/storage/maria/unittest/ma_control_file-t.c index 3ea6932c754..beb86843dd3 100644 --- a/storage/maria/unittest/ma_control_file-t.c +++ b/storage/maria/unittest/ma_control_file-t.c @@ -346,13 +346,13 @@ static int test_bad_checksum() MYF(MY_WME))) >= 0); RET_ERR_UNLESS(my_pread(fd, buffer, 1, 4, MYF(MY_FNABP | MY_WME)) == 0); buffer[0]+= 3; /* mangle checksum */ - RET_ERR_UNLESS(my_pwrite(fd, buffer+1, 1, 4, MYF(MY_FNABP | MY_WME)) == 0); + RET_ERR_UNLESS(my_pwrite(fd, buffer, 1, 4, MYF(MY_FNABP | MY_WME)) == 0); /* Check that control file module sees the problem */ RET_ERR_UNLESS(ma_control_file_create_or_open() == CONTROL_FILE_BAD_CHECKSUM); /* Restore checksum */ buffer[0]-= 3; - RET_ERR_UNLESS(my_pwrite(fd, buffer+1, 1, 4, MYF(MY_FNABP | MY_WME)) == 0); + RET_ERR_UNLESS(my_pwrite(fd, buffer, 1, 4, MYF(MY_FNABP | MY_WME)) == 0); RET_ERR_UNLESS(my_close(fd, MYF(MY_WME)) == 0); return 0; -- cgit v1.2.1 From 96d3604d99f366903c0e3e543858957211071d5a Mon Sep 17 00:00:00 2001 From: unknown Date: Thu, 9 Nov 2006 16:20:40 +0100 Subject: lock manager optimized for table locks storage/maria/unittest/lockman1-t.c: New BitKeeper file ``storage/maria/unittest/lockman1-t.c'' storage/maria/tablockman.c: New BitKeeper file ``storage/maria/tablockman.c'' storage/maria/tablockman.h: New BitKeeper file ``storage/maria/tablockman.h'' storage/maria/unittest/lockman2-t.c: New BitKeeper file ``storage/maria/unittest/lockman2-t.c'' --- storage/maria/Makefile.am | 4 +- storage/maria/lockman.c | 55 ++--- storage/maria/tablockman.c | 399 ++++++++++++++++++++++++++++++++++++ storage/maria/tablockman.h | 84 ++++++++ storage/maria/unittest/Makefile.am | 2 +- storage/maria/unittest/lockman-t.c | 2 +- storage/maria/unittest/lockman1-t.c | 330 +++++++++++++++++++++++++++++ storage/maria/unittest/lockman2-t.c | 318 ++++++++++++++++++++++++++++ 8 files changed, 1165 insertions(+), 29 deletions(-) create mode 100644 storage/maria/tablockman.c create mode 100644 storage/maria/tablockman.h create mode 100644 storage/maria/unittest/lockman1-t.c create mode 100644 storage/maria/unittest/lockman2-t.c diff --git a/storage/maria/Makefile.am b/storage/maria/Makefile.am index ee5f6238b46..24636f139ab 100644 --- a/storage/maria/Makefile.am +++ b/storage/maria/Makefile.am @@ -53,7 +53,7 @@ maria_pack_LDADD= @CLIENT_EXTRA_LDFLAGS@ libmaria.a \ noinst_PROGRAMS = ma_test1 ma_test2 ma_test3 ma_rt_test ma_sp_test noinst_HEADERS = maria_def.h ma_rt_index.h ma_rt_key.h ma_rt_mbr.h \ ma_sp_defs.h ma_fulltext.h ma_ftdefs.h ma_ft_test1.h \ - ma_ft_eval.h trnman.h lockman.h \ + ma_ft_eval.h trnman.h lockman.h tablockman.h \ ma_control_file.h ha_maria.h ma_test1_DEPENDENCIES= $(LIBRARIES) ma_test1_LDADD= @CLIENT_EXTRA_LDFLAGS@ libmaria.a \ @@ -108,7 +108,7 @@ libmaria_a_SOURCES = ma_init.c ma_open.c ma_extra.c ma_info.c ma_rkey.c \ ma_keycache.c ma_preload.c ma_ft_parser.c \ ma_ft_update.c ma_ft_boolean_search.c \ ma_ft_nlq_search.c ft_maria.c ma_sort.c \ - ha_maria.cc trnman.c lockman.c \ + ha_maria.cc trnman.c lockman.c tablockman.c \ ma_rt_index.c ma_rt_key.c ma_rt_mbr.c ma_rt_split.c \ ma_sp_key.c ma_control_file.c CLEANFILES = test?.MA? FT?.MA? isam.log ma_test_all ma_rt_test.MA? sp_test.MA? diff --git a/storage/maria/lockman.c b/storage/maria/lockman.c index 937c61021cc..31867d3903d 100644 --- a/storage/maria/lockman.c +++ b/storage/maria/lockman.c @@ -1,10 +1,6 @@ +// TODO - allocate everything from dynarrays !!! (benchmark) // TODO instant duration locks // automatically place S instead of LS if possible -/* - TODO optimization: table locks - they have completely - different characteristics. long lists, few distinct resources - - slow to scan, [possibly] high retry rate -*/ /* Copyright (C) 2006 MySQL AB This program is free software; you can redistribute it and/or modify @@ -68,9 +64,9 @@ it will wait for other locks. Here's an exception to "locks are added to the end" rule - upgraded locks are added after the last active lock but before all waiting locks. Old lock (the one we upgraded from) is - not removed from the list, indeed we may need to return to it later if - the new lock was in a savepoint that gets rolled back. So old lock is - marked as "ignored" (IGNORE_ME flag). New lock gets an UPGRADED flag. + not removed from the list, indeed it may be needed if the new lock was + in a savepoint that gets rolled back. So old lock is marked as "ignored" + (IGNORE_ME flag). New lock gets an UPGRADED flag. Loose locks add an important exception to the above. Loose locks do not always commute with other locks. In the list IX-LS both locks are active, @@ -90,12 +86,12 @@ variable a conflicting lock is returned and the calling thread waits on a pthread condition in the LOCK_OWNER structure of the owner of the conflicting lock. Or a new lock is compatible with all locks, but some - existing locks are not compatible with previous locks (example: request IS, + existing locks are not compatible with each other (example: request IS, when the list is S-IX) - that is not all locks are active. In this case a - first waiting lock is returned in the 'blocker' variable, - lockman_getlock() notices that a "blocker" does not conflict with the - requested lock, and "dereferences" it, to find the lock that it's waiting - on. The calling thread than begins to wait on the same lock. + first waiting lock is returned in the 'blocker' variable, lockman_getlock() + notices that a "blocker" does not conflict with the requested lock, and + "dereferences" it, to find the lock that it's waiting on. The calling + thread than begins to wait on the same lock. To better support table-row relations where one needs to lock the table with an intention lock before locking the row, extended diagnostics is @@ -107,6 +103,10 @@ whether it's possible to lock the row, but no need to lock it - perhaps the thread has a loose lock on this table). This is defined by getlock_result[] table. + + TODO optimization: table locks - they have completely + different characteristics. long lists, few distinct resources - + slow to scan, [possibly] high retry rate */ #include @@ -316,7 +316,7 @@ retry: DBUG_ASSERT(prev_active == TRUE); else cur_active&= lock_compatibility_matrix[prev_lock][cur_lock]; - if (upgrading && !cur_active) + if (upgrading && !cur_active /*&& !(cur_flags & UPGRADED)*/) break; if (prev_active && !cur_active) { @@ -327,7 +327,7 @@ retry: { /* we already have a lock on this resource */ DBUG_ASSERT(lock_combining_matrix[cur_lock][lock] != N); - DBUG_ASSERT(!upgrading); /* can happen only once */ + DBUG_ASSERT(!upgrading || (flags & IGNORE_ME)); if (lock_combining_matrix[cur_lock][lock] == cur_lock) { /* new lock is compatible */ @@ -380,7 +380,7 @@ retry: */ if (upgrading) { - if (compatible) + if (compatible /*&& prev_active*/) return PLACE_NEW_DISABLE_OLD; else return REQUEST_NEW_DISABLE_OLD; @@ -431,6 +431,9 @@ static int lockinsert(LOCK * volatile *head, LOCK *node, LF_PINS *pins, } if (res & LOCK_UPGRADE) cursor.upgrade_from->flags|= IGNORE_ME; +#warning is this OK ? if a reader has already read upgrade_from, \ + it may find it conflicting with node :( +//#error another bug - see the last test from test_lockman_simple() } } while (res == REPEAT_ONCE_MORE); @@ -439,8 +442,8 @@ static int lockinsert(LOCK * volatile *head, LOCK *node, LF_PINS *pins, _lf_unpin(pins, 2); /* note that blocker is not necessarily pinned here (when it's == curr). - this is ok as it's either a dummy node then for initialize_bucket - and dummy nodes don't need pinning, + this is ok as in such a case it's either a dummy node for + initialize_bucket() and dummy nodes don't need pinning, or it's a lock of the same transaction for lockman_getlock, and it cannot be removed by another thread */ @@ -484,9 +487,15 @@ static int lockdelete(LOCK * volatile *head, LOCK *node, LF_PINS *pins) res= lockfind(head, node, &cursor, pins); DBUG_ASSERT(res & ALREADY_HAVE); - if (cursor.upgrade_from) - cursor.upgrade_from->flags&= ~IGNORE_ME; - + /* + XXX this does not work with savepoints, as old lock is left ignored. + It cannot be unignored, as would basically mean moving the lock back + in the lock chain (from upgraded). And the latter is not allowed - + because it breaks list scanning. So old ignored lock must be deleted, + new - same - lock must be installed right after the lock we're deleting, + then we can delete. Good news is - this is only required when rolling + back a savepoint. + */ if (my_atomic_casptr((void **)&(cursor.curr->link), (void **)&cursor.next, 1+(char *)cursor.next)) { @@ -497,11 +506,7 @@ static int lockdelete(LOCK * volatile *head, LOCK *node, LF_PINS *pins) lockfind(head, node, &cursor, pins); } else - { res= REPEAT_ONCE_MORE; - if (cursor.upgrade_from) /* to satisfy the assert in lockfind */ - cursor.upgrade_from->flags|= IGNORE_ME; - } } while (res == REPEAT_ONCE_MORE); _lf_unpin(pins, 0); _lf_unpin(pins, 1); diff --git a/storage/maria/tablockman.c b/storage/maria/tablockman.c new file mode 100644 index 00000000000..d04bbbab0e4 --- /dev/null +++ b/storage/maria/tablockman.c @@ -0,0 +1,399 @@ +// TODO - allocate everything from dynarrays !!! (benchmark) +// TODO instant duration locks +// automatically place S instead of LS if possible +/* Copyright (C) 2006 MySQL AB + + 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; either version 2 of the License, or + (at your option) any later version. + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ + +#include +#include +#include +#include +#include "tablockman.h" + +/* + Lock compatibility matrix. + + It's asymmetric. Read it as "Somebody has the lock , can I set the lock ?" + + ') Though you can take LS lock while somebody has S lock, it makes no + sense - it's simpler to take S lock too. + + 1 - compatible + 0 - incompatible + -1 - "impossible", so that we can assert the impossibility. +*/ +static int lock_compatibility_matrix[10][10]= +{ /* N S X IS IX SIX LS LX SLX LSIX */ + { -1, 1, 1, 1, 1, 1, 1, 1, 1, 1 }, /* N */ + { -1, 1, 0, 1, 0, 0, 1, 0, 0, 0 }, /* S */ + { -1, 0, 0, 0, 0, 0, 0, 0, 0, 0 }, /* X */ + { -1, 1, 0, 1, 1, 1, 1, 1, 1, 1 }, /* IS */ + { -1, 0, 0, 1, 1, 0, 1, 1, 0, 1 }, /* IX */ + { -1, 0, 0, 1, 0, 0, 1, 0, 0, 0 }, /* SIX */ + { -1, 1, 0, 1, 0, 0, 1, 0, 0, 0 }, /* LS */ + { -1, 0, 0, 0, 0, 0, 0, 0, 0, 0 }, /* LX */ + { -1, 0, 0, 0, 0, 0, 0, 0, 0, 0 }, /* SLX */ + { -1, 0, 0, 1, 0, 0, 1, 0, 0, 0 } /* LSIX */ +}; + +/* + Lock combining matrix. + + It's symmetric. Read it as "what lock level L is identical to the + set of two locks A and B" + + One should never get N from it, we assert the impossibility +*/ +static enum lock_type lock_combining_matrix[10][10]= +{/* N S X IS IX SIX LS LX SLX LSIX */ + { N, S, X, IS, IX, SIX, S, SLX, SLX, SIX}, /* N */ + { S, S, X, S, SIX, SIX, S, SLX, SLX, SIX}, /* S */ + { X, X, X, X, X, X, X, X, X, X}, /* X */ + { IS, S, X, IS, IX, SIX, LS, LX, SLX, LSIX}, /* IS */ + { IX, SIX, X, IX, IX, SIX, LSIX, LX, SLX, LSIX}, /* IX */ + { SIX, SIX, X, SIX, SIX, SIX, SIX, SLX, SLX, SIX}, /* SIX */ + { LS, S, X, LS, LSIX, SIX, LS, LX, SLX, LSIX}, /* LS */ + { LX, SLX, X, LX, LX, SLX, LX, LX, SLX, LX}, /* LX */ + { SLX, SLX, X, SLX, SLX, SLX, SLX, SLX, SLX, SLX}, /* SLX */ + { LSIX, SIX, X, LSIX, LSIX, SIX, LSIX, LX, SLX, LSIX} /* LSIX */ +}; + +/* + the return codes for lockman_getlock + + It's asymmetric. Read it as "I have the lock , + what value should be returned for ?" + + 0 means impossible combination (assert!) + + Defines below help to preserve the table structure. + I/L/A values are self explanatory + x means the combination is possible (assert should not crash) + but it cannot happen in row locks, only in table locks (S,X), + or lock escalations (LS,LX) +*/ +#define I GOT_THE_LOCK_NEED_TO_LOCK_A_SUBRESOURCE +#define L GOT_THE_LOCK_NEED_TO_INSTANT_LOCK_A_SUBRESOURCE +#define A GOT_THE_LOCK +#define x GOT_THE_LOCK +static enum lockman_getlock_result getlock_result[10][10]= +{/* N S X IS IX SIX LS LX SLX LSIX */ + { 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}, /* N */ + { 0, x, 0, A, 0, 0, x, 0, 0, 0}, /* S */ + { 0, x, x, A, A, 0, x, x, 0, 0}, /* X */ + { 0, 0, 0, I, 0, 0, 0, 0, 0, 0}, /* IS */ + { 0, 0, 0, I, I, 0, 0, 0, 0, 0}, /* IX */ + { 0, x, 0, A, I, 0, x, 0, 0, 0}, /* SIX */ + { 0, 0, 0, L, 0, 0, x, 0, 0, 0}, /* LS */ + { 0, 0, 0, L, L, 0, x, x, 0, 0}, /* LX */ + { 0, x, 0, A, L, 0, x, x, 0, 0}, /* SLX */ + { 0, 0, 0, L, I, 0, x, 0, 0, 0} /* LSIX */ +}; +#undef I +#undef L +#undef A +#undef x + +/* + this structure is optimized for a case when there're many locks + on the same resource - e.g. a table +*/ + +struct st_table_lock { + struct st_table_lock *next_in_lo, *upgraded_from, *next, *prev; + struct st_locked_table *table; + uint16 loid; + char lock_type; +}; + +#define hash_insert my_hash_insert /* for consistency :) */ + +enum lockman_getlock_result +tablockman_getlock(TABLOCKMAN *lm, TABLE_LOCK_OWNER *lo, LOCKED_TABLE *table, enum lock_type lock) +{ + TABLE_LOCK *old, *new, *blocker; + TABLE_LOCK_OWNER *wait_for; + int i; + ulonglong deadline; + struct timespec timeout; + enum lock_type new_lock; + + pthread_mutex_lock(& table->mutex); + old= (TABLE_LOCK *)hash_search(& table->active, (byte *)&lo->loid, + sizeof(lo->loid)); + + /* perhaps we have the lock already ? */ + if (old && lock_combining_matrix[old->lock_type][lock] == old->lock_type) + { + pthread_mutex_unlock(& table->mutex); + return getlock_result[old->lock_type][lock]; + } + + pthread_mutex_lock(& lm->pool_mutex); + new= lm->pool; + if (new) + { + lm->pool= new->next; + pthread_mutex_unlock(& lm->pool_mutex); + } + else + { + pthread_mutex_unlock(& lm->pool_mutex); + new= (TABLE_LOCK *)my_malloc(sizeof(*new), MYF(MY_WME)); + if (!new) + { + pthread_mutex_unlock(& table->mutex); + return DIDNT_GET_THE_LOCK; + } + } + + /* calculate required upgraded lock type */ + new_lock= old ? lock_combining_matrix[old->lock_type][lock] : lock; + + new->loid= lo->loid; + new->lock_type= new_lock; + new->table= table; + + for (new->next= table->wait_queue_in ; ; ) + { + if (!old && new->next) + { + /* need to wait */ + DBUG_ASSERT(table->wait_queue_out); + DBUG_ASSERT(table->wait_queue_in); + blocker= new->next; + if (lock_compatibility_matrix[blocker->lock_type][lock]) + wait_for= lm->loid_to_lo(blocker->loid)->waiting_for; + else + wait_for= lm->loid_to_lo(blocker->loid); + } + else + { + /* checking for compatibility with existing locks */ + for (blocker= 0, i= 0; i < LOCK_TYPES; i++) + { + if (table->active_locks[i] && !lock_compatibility_matrix[i+1][lock]) + { + for (blocker= table->active_locks[i]; + blocker && blocker->loid == lo->loid; + blocker= blocker->next); + if (blocker) + break; + } + } + if (!blocker) + break; + wait_for= lm->loid_to_lo(blocker->loid); + } + + lo->waiting_for= wait_for; + if (!lo->waiting_lock) /* first iteration */ + { + /* lock upgrade or new lock request ? */ + if (old) + { + new->prev= 0; + if ((new->next= table->wait_queue_out)) + new->next->prev= new; + table->wait_queue_out= new; + if (!table->wait_queue_in) + table->wait_queue_in=table->wait_queue_out; + } + else + { + new->next= 0; + if ((new->prev= table->wait_queue_in)) + new->prev->next= new; + table->wait_queue_in= new; + if (!table->wait_queue_out) + table->wait_queue_out=table->wait_queue_in; + } + + lo->waiting_lock= new; + + deadline= my_getsystime() + lm->lock_timeout * 10000; + timeout.tv_sec= deadline/10000000; + timeout.tv_nsec= (deadline % 10000000) * 100; + } + else + { + if (my_getsystime() > deadline) + { + pthread_mutex_unlock(& table->mutex); + return DIDNT_GET_THE_LOCK; + } + } + + pthread_mutex_lock(wait_for->mutex); + pthread_mutex_unlock(& table->mutex); + + pthread_cond_timedwait(wait_for->cond, wait_for->mutex, &timeout); + + pthread_mutex_unlock(wait_for->mutex); + pthread_mutex_lock(& table->mutex); + } + + if (lo->waiting_lock) + { + if (new->prev) + new->prev->next= new->next; + if (new->next) + new->next->prev= new->prev; + if (table->wait_queue_in == new) + table->wait_queue_in= new->prev; + if (table->wait_queue_out == new) + table->wait_queue_out= new->next; + + lo->waiting_lock= 0; + } + + new->next_in_lo= lo->active_locks; + lo->active_locks= new; + + new->prev= 0; + if ((new->next= table->active_locks[new_lock-1])) + new->next->prev= new; + table->active_locks[new_lock-1]= new; + + /* placing the lock */ + hash_insert(& table->active, (byte *)new); + if (old) + { + new->upgraded_from= old; + hash_delete(& table->active, (byte *)old); + } + else + new->upgraded_from= 0; + + pthread_mutex_unlock(& table->mutex); + return getlock_result[lock][lock]; +} + +void tablockman_release_locks(TABLOCKMAN *lm, TABLE_LOCK_OWNER *lo) +{ + TABLE_LOCK *lock, *tmp, *local_pool= 0, *local_pool_end; + + local_pool_end= lo->waiting_lock ? lo->waiting_lock : lo->active_locks; + if (!local_pool_end) + return; + + if ((lock= lo->waiting_lock)) + { + pthread_mutex_lock(& lock->table->mutex); + + if (lock->prev) + lock->prev->next= lock->next; + if (lock->next) + lock->next->prev= lock->prev; + if (lock->table->wait_queue_in == lock) + lock->table->wait_queue_in= lock->prev; + if (lock->table->wait_queue_out == lock) + lock->table->wait_queue_out= lock->next; + pthread_mutex_unlock(& lock->table->mutex); + + lock->next= local_pool; + local_pool= lock; + DBUG_ASSERT(lock->loid == lo->loid); + } + + lock= lo->active_locks; + while (lock) + { + TABLE_LOCK *cur= lock; + pthread_mutex_t *mutex= & lock->table->mutex; + + lock= lock->next_in_lo; + + pthread_mutex_lock(mutex); + hash_delete(& cur->table->active, (byte *)cur); + + if (cur->prev) + cur->prev->next= cur->next; + if (cur->next) + cur->next->prev= cur->prev; + if (cur->table->active_locks[cur->lock_type-1] == cur) + cur->table->active_locks[cur->lock_type-1]= cur->next; + + cur->next= local_pool; + local_pool= cur; + DBUG_ASSERT(cur->loid == lo->loid); + + pthread_mutex_unlock(mutex); + } + + lo->waiting_lock= lo->active_locks= 0; + + pthread_mutex_lock(lo->mutex); + pthread_cond_broadcast(lo->cond); + pthread_mutex_unlock(lo->mutex); + + pthread_mutex_lock(& lm->pool_mutex); + local_pool_end->next= lm->pool; + lm->pool= local_pool; + pthread_mutex_unlock(& lm->pool_mutex); +} + +void tablockman_init(TABLOCKMAN *lm, loid_to_tlo_func *func, uint timeout) +{ + lm->pool= 0; + lm->loid_to_lo= func; + lm->lock_timeout= timeout; + pthread_mutex_init(&lm->pool_mutex, MY_MUTEX_INIT_FAST); +} + +void tablockman_destroy(TABLOCKMAN *lm) +{ + while (lm->pool) + { + TABLE_LOCK *tmp= lm->pool; + lm->pool= tmp->next; + my_free((void *)tmp, MYF(0)); + } + pthread_mutex_destroy(&lm->pool_mutex); +} + +void tablockman_init_locked_table(LOCKED_TABLE *lt) +{ + TABLE_LOCK *unused; + bzero(lt, sizeof(*lt)); + pthread_mutex_init(& lt->mutex, MY_MUTEX_INIT_FAST); + hash_init(& lt->active, &my_charset_bin, 10/*FIXME*/, + offsetof(TABLE_LOCK, loid), sizeof(unused->loid), 0, 0, 0); +} + +void tablockman_destroy_locked_table(LOCKED_TABLE *lt) +{ + hash_free(& lt->active); + pthread_mutex_destroy(& lt->mutex); +} + +static char *lock2str[LOCK_TYPES+1]= {"N", "S", "X", "IS", "IX", "SIX", + "LS", "LX", "SLX", "LSIX"}; + +void print_tlo(TABLE_LOCK_OWNER *lo) +{ + TABLE_LOCK *lock; + printf("lo%d>", lo->loid); + if ((lock= lo->waiting_lock)) + printf(" (%s.%p)", lock2str[lock->lock_type], lock->table); + for (lock= lo->active_locks; lock && lock != lock->next_in_lo; lock= lock->next_in_lo) + printf(" %s.%p", lock2str[lock->lock_type], lock->table); + if (lock && lock == lock->next_in_lo) + printf("!"); + printf("\n"); +} + diff --git a/storage/maria/tablockman.h b/storage/maria/tablockman.h new file mode 100644 index 00000000000..4b2d165af54 --- /dev/null +++ b/storage/maria/tablockman.h @@ -0,0 +1,84 @@ +/* Copyright (C) 2006 MySQL AB + + 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; either version 2 of the License, or + (at your option) any later version. + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ + +#ifndef _tablockman_h +#define _tablockman_h + +/* + Lock levels: + ^^^^^^^^^^^ + + N - "no lock", not a lock, used sometimes internally to simplify the code + S - Shared + X - eXclusive + IS - Intention Shared + IX - Intention eXclusive + SIX - Shared + Intention eXclusive + LS - Loose Shared + LX - Loose eXclusive + SLX - Shared + Loose eXclusive + LSIX - Loose Shared + Intention eXclusive +*/ +#ifndef _lockman_h +enum lock_type { N, S, X, IS, IX, SIX, LS, LX, SLX, LSIX }; +enum lockman_getlock_result { + DIDNT_GET_THE_LOCK=0, GOT_THE_LOCK, + GOT_THE_LOCK_NEED_TO_LOCK_A_SUBRESOURCE, + GOT_THE_LOCK_NEED_TO_INSTANT_LOCK_A_SUBRESOURCE +}; + +#endif + +#define LOCK_TYPES LSIX + +typedef struct st_table_lock_owner TABLE_LOCK_OWNER; +typedef struct st_table_lock TABLE_LOCK; +typedef struct st_locked_table LOCKED_TABLE; +typedef TABLE_LOCK_OWNER *loid_to_tlo_func(uint16); + +typedef struct { + pthread_mutex_t pool_mutex; + TABLE_LOCK *pool; + uint lock_timeout; + loid_to_tlo_func *loid_to_lo; +} TABLOCKMAN; + +struct st_table_lock_owner { + TABLE_LOCK *active_locks, *waiting_lock; + TABLE_LOCK_OWNER *waiting_for; + pthread_cond_t *cond; /* transactions waiting for this, wait on 'cond' */ + pthread_mutex_t *mutex; /* mutex is required to use 'cond' */ + uint16 loid; +}; + +struct st_locked_table { + pthread_mutex_t mutex; + HASH active; // fast to remove + TABLE_LOCK *active_locks[LOCK_TYPES]; // fast to see a conflict + TABLE_LOCK *wait_queue_in, *wait_queue_out; +}; + +void tablockman_init(TABLOCKMAN *, loid_to_tlo_func *, uint); +void tablockman_destroy(TABLOCKMAN *); +enum lockman_getlock_result tablockman_getlock(TABLOCKMAN *, TABLE_LOCK_OWNER *, + LOCKED_TABLE *, + enum lock_type lock); +void tablockman_release_locks(TABLOCKMAN *, TABLE_LOCK_OWNER *); +void tablockman_init_locked_table(LOCKED_TABLE *); +void print_tlo(TABLE_LOCK_OWNER *); + +#endif + diff --git a/storage/maria/unittest/Makefile.am b/storage/maria/unittest/Makefile.am index e29bc7f86cb..d0b247d65e1 100644 --- a/storage/maria/unittest/Makefile.am +++ b/storage/maria/unittest/Makefile.am @@ -25,5 +25,5 @@ LDADD= $(top_builddir)/unittest/mytap/libmytap.a \ $(top_builddir)/mysys/libmysys.a \ $(top_builddir)/dbug/libdbug.a \ $(top_builddir)/strings/libmystrings.a @ZLIB_LIBS@ -noinst_PROGRAMS = ma_control_file-t trnman-t lockman-t +noinst_PROGRAMS = ma_control_file-t trnman-t lockman-t lockman1-t lockman2-t CLEANFILES = maria_control diff --git a/storage/maria/unittest/lockman-t.c b/storage/maria/unittest/lockman-t.c index 7b37c983864..94b034bdaad 100644 --- a/storage/maria/unittest/lockman-t.c +++ b/storage/maria/unittest/lockman-t.c @@ -268,7 +268,7 @@ int main() test_lockman_simple(); -#define CYCLES 1000 +#define CYCLES 10000 #define THREADS Nlos /* don't change this line */ /* mixed load, stress-test with random locks */ diff --git a/storage/maria/unittest/lockman1-t.c b/storage/maria/unittest/lockman1-t.c new file mode 100644 index 00000000000..c9a4ff98f2a --- /dev/null +++ b/storage/maria/unittest/lockman1-t.c @@ -0,0 +1,330 @@ +/* Copyright (C) 2006 MySQL AB + + 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; either version 2 of the License, or + (at your option) any later version. + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ + +//#define EXTRA_VERBOSE + +#include + +#include +#include +#include +#include +#include "../lockman.h" +#include "../tablockman.h" + +#define Nlos 100 +#define Ntbls 10 +LOCK_OWNER loarray[Nlos]; +TABLE_LOCK_OWNER loarray1[Nlos]; +pthread_mutex_t mutexes[Nlos]; +pthread_cond_t conds[Nlos]; +LOCKED_TABLE ltarray[Ntbls]; +LOCKMAN lockman; +TABLOCKMAN tablockman; + +#ifndef EXTRA_VERBOSE +#define print_lo1(X) /* no-op */ +#define DIAG(X) /* no-op */ +#else +#define DIAG(X) diag X +#endif + +LOCK_OWNER *loid2lo(uint16 loid) +{ + return loarray+loid-1; +} +TABLE_LOCK_OWNER *loid2lo1(uint16 loid) +{ + return loarray1+loid-1; +} + +#define unlock_all(O) diag("lo" #O "> release all locks"); \ + tablockman_release_locks(&tablockman, loid2lo1(O)); +#define test_lock(O, R, L, S, RES) \ + ok(tablockman_getlock(&tablockman, loid2lo1(O), <array[R], L) == RES, \ + "lo" #O "> " S "lock resource " #R " with " #L "-lock"); \ + print_lo1(loid2lo1(O)); +#define lock_ok_a(O, R, L) \ + test_lock(O, R, L, "", GOT_THE_LOCK) +#define lock_ok_i(O, R, L) \ + test_lock(O, R, L, "", GOT_THE_LOCK_NEED_TO_LOCK_A_SUBRESOURCE) +#define lock_ok_l(O, R, L) \ + test_lock(O, R, L, "", GOT_THE_LOCK_NEED_TO_INSTANT_LOCK_A_SUBRESOURCE) +#define lock_conflict(O, R, L) \ + test_lock(O, R, L, "cannot ", DIDNT_GET_THE_LOCK); + +void test_tablockman_simple() +{ + /* simple */ + lock_ok_a(1, 1, S); + lock_ok_i(2, 2, IS); + lock_ok_i(1, 2, IX); + /* lock escalation */ + lock_ok_a(1, 1, X); + lock_ok_i(2, 2, IX); + /* failures */ + lock_conflict(2, 1, X); + unlock_all(2); + lock_ok_a(1, 2, S); + lock_ok_a(1, 2, IS); + lock_ok_a(1, 2, LS); + lock_ok_i(1, 3, IX); + lock_ok_a(2, 3, LS); + lock_ok_i(1, 3, IX); + lock_ok_l(2, 3, IS); + unlock_all(1); + unlock_all(2); + + lock_ok_i(1, 1, IX); + lock_conflict(2, 1, S); + lock_ok_a(1, 1, LS); + unlock_all(1); + unlock_all(2); + + lock_ok_i(1, 1, IX); + lock_ok_a(2, 1, LS); + lock_ok_a(1, 1, LS); + lock_ok_i(1, 1, IX); + lock_ok_i(3, 1, IS); + unlock_all(1); + unlock_all(2); + unlock_all(3); + + lock_ok_i(1, 4, IS); + lock_ok_i(2, 4, IS); + lock_ok_i(3, 4, IS); + lock_ok_a(3, 4, LS); + lock_ok_i(4, 4, IS); + lock_conflict(4, 4, IX); + lock_conflict(2, 4, IX); + lock_ok_a(1, 4, LS); + unlock_all(1); + unlock_all(2); + unlock_all(3); + unlock_all(4); + + lock_ok_i(1, 1, IX); + lock_ok_i(2, 1, IX); + lock_conflict(1, 1, S); + lock_conflict(2, 1, X); + unlock_all(1); + unlock_all(2); +} + +int rt_num_threads; +int litmus; +int thread_number= 0, timeouts= 0; +void run_test(const char *test, pthread_handler handler, int n, int m) +{ + pthread_t *threads; + ulonglong now= my_getsystime(); + int i; + + thread_number= timeouts= 0; + litmus= 0; + + threads= (pthread_t *)my_malloc(sizeof(void *)*n, MYF(0)); + if (!threads) + { + diag("Out of memory"); + abort(); + } + + diag("Running %s with %d threads, %d iterations... ", test, n, m); + rt_num_threads= n; + for (i= 0; i < n ; i++) + if (pthread_create(threads+i, 0, handler, &m)) + { + diag("Could not create thread"); + abort(); + } + for (i= 0 ; i < n ; i++) + pthread_join(threads[i], 0); + now= my_getsystime()-now; + ok(litmus == 0, "Finished %s in %g secs (%d)", test, ((double)now)/1e7, litmus); + my_free((void*)threads, MYF(0)); +} + +pthread_mutex_t rt_mutex; +int Nrows= 100; +int Ntables= 10; +int table_lock_ratio= 10; +enum lock_type lock_array[6]= {S, X, LS, LX, IS, IX}; +char *lock2str[6]= {"S", "X", "LS", "LX", "IS", "IX"}; +char *res2str[4]= { + "DIDN'T GET THE LOCK", + "GOT THE LOCK", + "GOT THE LOCK NEED TO LOCK A SUBRESOURCE", + "GOT THE LOCK NEED TO INSTANT LOCK A SUBRESOURCE"}; +pthread_handler_t test_lockman(void *arg) +{ + int m= (*(int *)arg); + uint x, loid, row, table, res, locklevel, timeout= 0; + LOCK_OWNER *lo; TABLE_LOCK_OWNER *lo1; DBUG_ASSERT(Ntables <= Ntbls); + + pthread_mutex_lock(&rt_mutex); + loid= ++thread_number; + pthread_mutex_unlock(&rt_mutex); + lo= loid2lo(loid); lo1= loid2lo1(loid); + + for (x= ((int)(intptr)(&m)); m > 0; m--) + { + x= (x*3628273133 + 1500450271) % 9576890767; /* three prime numbers */ + row= x % Nrows + Ntables; + table= row % Ntables; + locklevel= (x/Nrows) & 3; + if (table_lock_ratio && (x/Nrows/4) % table_lock_ratio == 0) + { /* table lock */ + res= tablockman_getlock(&tablockman, lo1, ltarray+table, lock_array[locklevel]); + DIAG(("loid %2d, table %d, lock %s, res %s", loid, table, + lock2str[locklevel], res2str[res])); + if (res == DIDNT_GET_THE_LOCK) + { + lockman_release_locks(&lockman, lo); tablockman_release_locks(&tablockman, lo1); + DIAG(("loid %2d, release all locks", loid)); + timeout++; + continue; + } + DBUG_ASSERT(res == GOT_THE_LOCK); + } + else + { /* row lock */ + locklevel&= 1; + res= tablockman_getlock(&tablockman, lo1, ltarray+table, lock_array[locklevel + 4]); + DIAG(("loid %2d, row %d, lock %s, res %s", loid, row, + lock2str[locklevel+4], res2str[res])); + switch (res) + { + case DIDNT_GET_THE_LOCK: + lockman_release_locks(&lockman, lo); tablockman_release_locks(&tablockman, lo1); + DIAG(("loid %2d, release all locks", loid)); + timeout++; + continue; + case GOT_THE_LOCK: + continue; + case GOT_THE_LOCK_NEED_TO_INSTANT_LOCK_A_SUBRESOURCE: + /* not implemented, so take a regular lock */ + case GOT_THE_LOCK_NEED_TO_LOCK_A_SUBRESOURCE: + res= lockman_getlock(&lockman, lo, row, lock_array[locklevel]); + DIAG(("loid %2d, ROW %d, lock %s, res %s", loid, row, + lock2str[locklevel], res2str[res])); + if (res == DIDNT_GET_THE_LOCK) + { + lockman_release_locks(&lockman, lo); + tablockman_release_locks(&tablockman, lo1); + DIAG(("loid %2d, release all locks", loid)); + timeout++; + continue; + } + DBUG_ASSERT(res == GOT_THE_LOCK); + continue; + default: + DBUG_ASSERT(0); + } + } + } + + lockman_release_locks(&lockman, lo); + tablockman_release_locks(&tablockman, lo1); + + pthread_mutex_lock(&rt_mutex); + rt_num_threads--; + timeouts+= timeout; + if (!rt_num_threads) + diag("number of timeouts: %d", timeouts); + pthread_mutex_unlock(&rt_mutex); + + return 0; +} + +int main() +{ + int i; + + my_init(); + pthread_mutex_init(&rt_mutex, 0); + + plan(35); + + if (my_atomic_initialize()) + return exit_status(); + + + lockman_init(&lockman, &loid2lo, 50); + tablockman_init(&tablockman, &loid2lo1, 50); + + for (i= 0; i < Nlos; i++) + { + pthread_mutex_init(&mutexes[i], MY_MUTEX_INIT_FAST); + pthread_cond_init (&conds[i], 0); + + loarray[i].pins= lf_alloc_get_pins(&lockman.alloc); + loarray[i].all_locks= 0; + loarray[i].waiting_for= 0; + loarray[i].mutex= &mutexes[i]; + loarray[i].cond= &conds[i]; + loarray[i].loid= i+1; + + loarray1[i].active_locks= 0; + loarray1[i].waiting_lock= 0; + loarray1[i].waiting_for= 0; + loarray1[i].mutex= &mutexes[i]; + loarray1[i].cond= &conds[i]; + loarray1[i].loid= i+1; + } + + for (i= 0; i < Ntbls; i++) + { + tablockman_init_locked_table(ltarray+i); + } + + //test_tablockman_simple(); + +#define CYCLES 10000 +#define THREADS Nlos /* don't change this line */ + + /* mixed load, stress-test with random locks */ + Nrows= 100; + Ntables= 10; + table_lock_ratio= 10; + run_test("\"random lock\" stress test", test_lockman, THREADS, CYCLES); + + /* "real-life" simulation - many rows, no table locks */ + Nrows= 1000000; + Ntables= 10; + table_lock_ratio= 0; + run_test("\"real-life\" simulation test", test_lockman, THREADS, CYCLES*10); + + for (i= 0; i < Nlos; i++) + { + lockman_release_locks(&lockman, &loarray[i]); + pthread_mutex_destroy(loarray[i].mutex); + pthread_cond_destroy(loarray[i].cond); + lf_pinbox_put_pins(loarray[i].pins); + } + + { + ulonglong now= my_getsystime(); + lockman_destroy(&lockman); + now= my_getsystime()-now; + diag("lockman_destroy: %g secs", ((double)now)/1e7); + } + + pthread_mutex_destroy(&rt_mutex); + my_end(0); + return exit_status(); +} + diff --git a/storage/maria/unittest/lockman2-t.c b/storage/maria/unittest/lockman2-t.c new file mode 100644 index 00000000000..72b3993e4b4 --- /dev/null +++ b/storage/maria/unittest/lockman2-t.c @@ -0,0 +1,318 @@ +/* Copyright (C) 2006 MySQL AB + + 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; either version 2 of the License, or + (at your option) any later version. + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ + +//#define EXTRA_VERBOSE + +#include + +#include +#include +#include +#include +#include "../tablockman.h" + +#define Nlos 100 +#define Ntbls 110 +TABLE_LOCK_OWNER loarray1[Nlos]; +pthread_mutex_t mutexes[Nlos]; +pthread_cond_t conds[Nlos]; +LOCKED_TABLE ltarray[Ntbls]; +TABLOCKMAN tablockman; + +#ifndef EXTRA_VERBOSE +#define print_lo1(X) /* no-op */ +#define DIAG(X) /* no-op */ +#else +#define DIAG(X) diag X +#endif + +TABLE_LOCK_OWNER *loid2lo1(uint16 loid) +{ + return loarray1+loid-1; +} + +#define unlock_all(O) diag("lo" #O "> release all locks"); \ + tablockman_release_locks(&tablockman, loid2lo1(O)); +#define test_lock(O, R, L, S, RES) \ + ok(tablockman_getlock(&tablockman, loid2lo1(O), <array[R], L) == RES, \ + "lo" #O "> " S "lock resource " #R " with " #L "-lock"); \ + print_lo1(loid2lo1(O)); +#define lock_ok_a(O, R, L) \ + test_lock(O, R, L, "", GOT_THE_LOCK) +#define lock_ok_i(O, R, L) \ + test_lock(O, R, L, "", GOT_THE_LOCK_NEED_TO_LOCK_A_SUBRESOURCE) +#define lock_ok_l(O, R, L) \ + test_lock(O, R, L, "", GOT_THE_LOCK_NEED_TO_INSTANT_LOCK_A_SUBRESOURCE) +#define lock_conflict(O, R, L) \ + test_lock(O, R, L, "cannot ", DIDNT_GET_THE_LOCK); + +void test_tablockman_simple() +{ + /* simple */ + lock_ok_a(1, 1, S); + lock_ok_i(2, 2, IS); + lock_ok_i(1, 2, IX); + /* lock escalation */ + lock_ok_a(1, 1, X); + lock_ok_i(2, 2, IX); + /* failures */ + lock_conflict(2, 1, X); + unlock_all(2); + lock_ok_a(1, 2, S); + lock_ok_a(1, 2, IS); + lock_ok_a(1, 2, LS); + lock_ok_i(1, 3, IX); + lock_ok_a(2, 3, LS); + lock_ok_i(1, 3, IX); + lock_ok_l(2, 3, IS); + unlock_all(1); + unlock_all(2); + + lock_ok_i(1, 1, IX); + lock_conflict(2, 1, S); + lock_ok_a(1, 1, LS); + unlock_all(1); + unlock_all(2); + + lock_ok_i(1, 1, IX); + lock_ok_a(2, 1, LS); + lock_ok_a(1, 1, LS); + lock_ok_i(1, 1, IX); + lock_ok_i(3, 1, IS); + unlock_all(1); + unlock_all(2); + unlock_all(3); + + lock_ok_i(1, 4, IS); + lock_ok_i(2, 4, IS); + lock_ok_i(3, 4, IS); + lock_ok_a(3, 4, LS); + lock_ok_i(4, 4, IS); + lock_conflict(4, 4, IX); + lock_conflict(2, 4, IX); + lock_ok_a(1, 4, LS); + unlock_all(1); + unlock_all(2); + unlock_all(3); + unlock_all(4); + + lock_ok_i(1, 1, IX); + lock_ok_i(2, 1, IX); + lock_conflict(1, 1, S); + lock_conflict(2, 1, X); + unlock_all(1); + unlock_all(2); +} + +int rt_num_threads; +int litmus; +int thread_number= 0, timeouts= 0; +void run_test(const char *test, pthread_handler handler, int n, int m) +{ + pthread_t *threads; + ulonglong now= my_getsystime(); + int i; + + thread_number= timeouts= 0; + litmus= 0; + + threads= (pthread_t *)my_malloc(sizeof(void *)*n, MYF(0)); + if (!threads) + { + diag("Out of memory"); + abort(); + } + + diag("Running %s with %d threads, %d iterations... ", test, n, m); + rt_num_threads= n; + for (i= 0; i < n ; i++) + if (pthread_create(threads+i, 0, handler, &m)) + { + diag("Could not create thread"); + abort(); + } + for (i= 0 ; i < n ; i++) + pthread_join(threads[i], 0); + now= my_getsystime()-now; + ok(litmus == 0, "Finished %s in %g secs (%d)", test, ((double)now)/1e7, litmus); + my_free((void*)threads, MYF(0)); +} + +pthread_mutex_t rt_mutex; +int Nrows= 100; +int Ntables= 10; +int table_lock_ratio= 10; +enum lock_type lock_array[6]= {S, X, LS, LX, IS, IX}; +char *lock2str[6]= {"S", "X", "LS", "LX", "IS", "IX"}; +char *res2str[4]= { + "DIDN'T GET THE LOCK", + "GOT THE LOCK", + "GOT THE LOCK NEED TO LOCK A SUBRESOURCE", + "GOT THE LOCK NEED TO INSTANT LOCK A SUBRESOURCE"}; +pthread_handler_t test_lockman(void *arg) +{ + int m= (*(int *)arg); + uint x, loid, row, table, res, locklevel, timeout= 0; + TABLE_LOCK_OWNER *lo1; + DBUG_ASSERT(Ntables <= Ntbls); + DBUG_ASSERT(Nrows + Ntables <= Ntbls); + + pthread_mutex_lock(&rt_mutex); + loid= ++thread_number; + pthread_mutex_unlock(&rt_mutex); + lo1= loid2lo1(loid); + + for (x= ((int)(intptr)(&m)); m > 0; m--) + { + x= (x*3628273133 + 1500450271) % 9576890767; /* three prime numbers */ + row= x % Nrows + Ntables; + table= row % Ntables; + locklevel= (x/Nrows) & 3; + if (table_lock_ratio && (x/Nrows/4) % table_lock_ratio == 0) + { /* table lock */ + res= tablockman_getlock(&tablockman, lo1, ltarray+table, lock_array[locklevel]); + DIAG(("loid %2d, table %d, lock %s, res %s", loid, table, + lock2str[locklevel], res2str[res])); + if (res == DIDNT_GET_THE_LOCK) + { + tablockman_release_locks(&tablockman, lo1); + DIAG(("loid %2d, release all locks", loid)); + timeout++; + continue; + } + DBUG_ASSERT(res == GOT_THE_LOCK); + } + else + { /* row lock */ + locklevel&= 1; + res= tablockman_getlock(&tablockman, lo1, ltarray+table, lock_array[locklevel + 4]); + DIAG(("loid %2d, row %d, lock %s, res %s", loid, row, + lock2str[locklevel+4], res2str[res])); + switch (res) + { + case DIDNT_GET_THE_LOCK: + tablockman_release_locks(&tablockman, lo1); + DIAG(("loid %2d, release all locks", loid)); + timeout++; + continue; + case GOT_THE_LOCK: + continue; + case GOT_THE_LOCK_NEED_TO_INSTANT_LOCK_A_SUBRESOURCE: + /* not implemented, so take a regular lock */ + case GOT_THE_LOCK_NEED_TO_LOCK_A_SUBRESOURCE: + res= tablockman_getlock(&tablockman, lo1, ltarray+row, lock_array[locklevel]); + DIAG(("loid %2d, ROW %d, lock %s, res %s", loid, row, + lock2str[locklevel], res2str[res])); + if (res == DIDNT_GET_THE_LOCK) + { + tablockman_release_locks(&tablockman, lo1); + DIAG(("loid %2d, release all locks", loid)); + timeout++; + continue; + } + DBUG_ASSERT(res == GOT_THE_LOCK); + continue; + default: + DBUG_ASSERT(0); + } + } + } + + tablockman_release_locks(&tablockman, lo1); + + pthread_mutex_lock(&rt_mutex); + rt_num_threads--; + timeouts+= timeout; + if (!rt_num_threads) + diag("number of timeouts: %d", timeouts); + pthread_mutex_unlock(&rt_mutex); + + return 0; +} + +int main() +{ + int i; + + my_init(); + pthread_mutex_init(&rt_mutex, 0); + + plan(35); + + if (my_atomic_initialize()) + return exit_status(); + + + tablockman_init(&tablockman, &loid2lo1, 50); + + for (i= 0; i < Nlos; i++) + { + pthread_mutex_init(&mutexes[i], MY_MUTEX_INIT_FAST); + pthread_cond_init (&conds[i], 0); + + loarray1[i].active_locks= 0; + loarray1[i].waiting_lock= 0; + loarray1[i].waiting_for= 0; + loarray1[i].mutex= &mutexes[i]; + loarray1[i].cond= &conds[i]; + loarray1[i].loid= i+1; + } + + for (i= 0; i < Ntbls; i++) + { + tablockman_init_locked_table(ltarray+i); + } + + test_tablockman_simple(); + +#define CYCLES 10000 +#define THREADS Nlos /* don't change this line */ + + /* mixed load, stress-test with random locks */ + Nrows= 100; + Ntables= 10; + table_lock_ratio= 10; + run_test("\"random lock\" stress test", test_lockman, THREADS, CYCLES); +#if 0 + /* "real-life" simulation - many rows, no table locks */ + Nrows= 1000000; + Ntables= 10; + table_lock_ratio= 0; + run_test("\"real-life\" simulation test", test_lockman, THREADS, CYCLES*10); +#endif + for (i= 0; i < Nlos; i++) + { + tablockman_release_locks(&tablockman, &loarray1[i]); + pthread_mutex_destroy(loarray1[i].mutex); + pthread_cond_destroy(loarray1[i].cond); + } + + { + ulonglong now= my_getsystime(); + for (i= 0; i < Ntbls; i++) + { + tablockman_destroy_locked_table(ltarray+i); + } + tablockman_destroy(&tablockman); + now= my_getsystime()-now; + diag("lockman_destroy: %g secs", ((double)now)/1e7); + } + + pthread_mutex_destroy(&rt_mutex); + my_end(0); + return exit_status(); +} + -- cgit v1.2.1 From aad040fde0bfdbfbf2876dba8d7034333493fbf2 Mon Sep 17 00:00:00 2001 From: unknown Date: Fri, 10 Nov 2006 10:56:05 +0100 Subject: comments storage/maria/tablockman.c: comments. bugfix - a special case in release_locks storage/maria/unittest/lockman1-t.c: updated storage/maria/unittest/lockman2-t.c: new tests --- storage/maria/tablockman.c | 238 +++++++++++++++++++++++++++++------- storage/maria/tablockman.h | 32 ++--- storage/maria/unittest/lockman1-t.c | 4 +- storage/maria/unittest/lockman2-t.c | 18 ++- 4 files changed, 233 insertions(+), 59 deletions(-) diff --git a/storage/maria/tablockman.c b/storage/maria/tablockman.c index d04bbbab0e4..608f852cb7b 100644 --- a/storage/maria/tablockman.c +++ b/storage/maria/tablockman.c @@ -1,5 +1,4 @@ // TODO - allocate everything from dynarrays !!! (benchmark) -// TODO instant duration locks // automatically place S instead of LS if possible /* Copyright (C) 2006 MySQL AB @@ -23,6 +22,93 @@ #include #include "tablockman.h" +/* + Lock Manager for Table Locks + + The code below handles locks on resources - but it is optimized for a + case when a number of resources is not very large, and there are many of + locks per resource - that is a resource is likely to be a table or a + database, but hardly a row in a table. + + Locks belong to "lock owners". A Lock Owner is uniquely identified by a + 16-bit number - loid (lock owner identifier). A function loid_to_tlo must + be provided by the application that takes such a number as an argument + and returns a TABLE_LOCK_OWNER structure. + + Lock levels are completely defined by three tables. Lock compatibility + matrix specifies which locks can be held at the same time on a resource. + Lock combining matrix specifies what lock level has the same behaviour as + a pair of two locks of given levels. getlock_result matrix simplifies + intention locking and lock escalation for an application, basically it + defines which locks are intention locks and which locks are "loose" + locks. It is only used to provide better diagnostics for the + application, lock manager itself does not differentiate between normal, + intention, and loose locks. + + The assumptions are: few distinct resources, many locks are held at the + same time on one resource. Thus: a lock structure _per resource_ can be + rather large; a lock structure _per lock_ does not need to be very small + either; we need to optimize for _speed_. Operations we need are: place a + lock, check if a particular transaction already has a lock on this + resource, check if a conflicting lock exists, if yes - find who owns it. + + Solution: every resource has a structure with + 1. Hash of "active" (see below for the description of "active") granted + locks with loid as a key. Thus, checking if a given transaction has a + lock on this resource is O(1) operation. + 2. Doubly-linked lists of all granted locks - one list for every lock + type. Thus, checking if a conflicting lock exists is a check whether + an appropriate list head pointer is not null, also O(1). + 3. Every lock has a loid of the owner, thus checking who owns a + conflicting lock is also O(1). + 4. Deque of waiting locks. It's a deque not a fifo, because for lock + upgrades requests are added to the queue head, not tail. There's never + a need to scan the queue. + + Result: adding or removing a lock is always a O(1) operation, it does not + depend on the number of locks on the resource, or number of transactions, + or number of resources. It _does_ depend on the number of different lock + levels - O(number_of_lock_levels) - but it's a constant. + + Waiting: if there is a conflicting lock or if wait queue is not empty, a + requested lock cannot be granted at once. It is added to the end of the + wait queue. If there is a conflicting lock - the "blocker" transaction is + the owner of this lock. If there's no conflict but a queue was not empty, + than the "blocker" is the transaction that the owner of the lock at the + end of the queue is waiting for (in other words, our lock is added to the + end of the wait queue, and our blocker is the same as of the lock right + before us). + + Lock upgrades: when a thread that has a lock on a given resource, + requests a new lock on the same resource and the old lock is not enough + to satisfy new lock requirements (which is defined by + lock_combining_matrix[old_lock][new_lock] != old_lock), a new lock + (defineded by lock_combining_matrix as above) is placed. Depending on + other granted locks it is immediately active or it has to wait. Here the + lock is added to the start of the waiting queue, not to the end. Old + lock, is removed from the hash, but not from the doubly-linked lists. + (indeed, a transaction checks "do I have a lock on this resource ?" by + looking in a hash, and it should find a latest lock, so old locks must be + removed; but a transaction checks "are the conflicting locks ?" by + checking doubly-linked lists, it doesn't matter if it will find an old + lock - if it would be removed, a new lock would be also a conflict). + + To better support table-row relations where one needs to lock the table + with an intention lock before locking the row, extended diagnostics is + provided. When an intention lock (presumably on a table) is granted, + lockman_getlock() returns one of GOT_THE_LOCK (no need to lock the row, + perhaps the thread already has a normal lock on this table), + GOT_THE_LOCK_NEED_TO_LOCK_A_SUBRESOURCE (need to lock the row, as usual), + GOT_THE_LOCK_NEED_TO_INSTANT_LOCK_A_SUBRESOURCE (only need to check + whether it's possible to lock the row, but no need to lock it - perhaps + the thread has a loose lock on this table). This is defined by + getlock_result[] table. + + Instant duration locks are not supported. Though they're trivial to add, + they are normally only used on rows, not on tables. So, presumably, + they are not needed here. +*/ + /* Lock compatibility matrix. @@ -121,28 +207,63 @@ struct st_table_lock { }; #define hash_insert my_hash_insert /* for consistency :) */ +#define remove_from_wait_queue(LOCK, TABLE) \ + do \ + { \ + if ((LOCK)->prev) \ + { \ + DBUG_ASSERT((TABLE)->wait_queue_out != (LOCK)); \ + (LOCK)->prev->next= (LOCK)->next; \ + } \ + else \ + { \ + DBUG_ASSERT((TABLE)->wait_queue_out == (LOCK)); \ + (TABLE)->wait_queue_out= (LOCK)->next; \ + } \ + if ((LOCK)->next) \ + { \ + DBUG_ASSERT((TABLE)->wait_queue_in != (LOCK)); \ + (LOCK)->next->prev= (LOCK)->prev; \ + } \ + else \ + { \ + DBUG_ASSERT((TABLE)->wait_queue_in == (LOCK)); \ + (TABLE)->wait_queue_in= (LOCK)->prev; \ + } \ + } while (0) +/* + DESCRIPTION + tries to lock a resource 'table' with a lock level 'lock'. + + RETURN + see enum lockman_getlock_result +*/ enum lockman_getlock_result -tablockman_getlock(TABLOCKMAN *lm, TABLE_LOCK_OWNER *lo, LOCKED_TABLE *table, enum lock_type lock) +tablockman_getlock(TABLOCKMAN *lm, TABLE_LOCK_OWNER *lo, + LOCKED_TABLE *table, enum lock_type lock) { TABLE_LOCK *old, *new, *blocker; TABLE_LOCK_OWNER *wait_for; - int i; ulonglong deadline; struct timespec timeout; enum lock_type new_lock; + int i; pthread_mutex_lock(& table->mutex); + /* do we alreasy have a lock on this resource ? */ old= (TABLE_LOCK *)hash_search(& table->active, (byte *)&lo->loid, sizeof(lo->loid)); - /* perhaps we have the lock already ? */ + /* and if yes, is it enough to satisfy the new request */ if (old && lock_combining_matrix[old->lock_type][lock] == old->lock_type) { + /* yes */ pthread_mutex_unlock(& table->mutex); return getlock_result[old->lock_type][lock]; } + /* no, placing a new lock. first - take a free lock structure from the pool */ pthread_mutex_lock(& lm->pool_mutex); new= lm->pool; if (new) @@ -161,25 +282,28 @@ tablockman_getlock(TABLOCKMAN *lm, TABLE_LOCK_OWNER *lo, LOCKED_TABLE *table, en } } - /* calculate required upgraded lock type */ + /* calculate the level of the upgraded lock */ new_lock= old ? lock_combining_matrix[old->lock_type][lock] : lock; new->loid= lo->loid; new->lock_type= new_lock; new->table= table; - for (new->next= table->wait_queue_in ; ; ) + /* and try to place it */ + for (new->prev= table->wait_queue_in ; ; ) { - if (!old && new->next) + /* waiting queue is not empty and we're not upgrading */ + if (!old && new->prev) { /* need to wait */ DBUG_ASSERT(table->wait_queue_out); DBUG_ASSERT(table->wait_queue_in); - blocker= new->next; + blocker= new->prev; + /* wait for a previous lock in the queue or for a lock it's waiting for */ if (lock_compatibility_matrix[blocker->lock_type][lock]) - wait_for= lm->loid_to_lo(blocker->loid)->waiting_for; + wait_for= lm->loid_to_tlo(blocker->loid)->waiting_for; else - wait_for= lm->loid_to_lo(blocker->loid); + wait_for= lm->loid_to_tlo(blocker->loid); } else { @@ -188,24 +312,27 @@ tablockman_getlock(TABLOCKMAN *lm, TABLE_LOCK_OWNER *lo, LOCKED_TABLE *table, en { if (table->active_locks[i] && !lock_compatibility_matrix[i+1][lock]) { + /* the first lock in the list may be our own - skip it */ for (blocker= table->active_locks[i]; blocker && blocker->loid == lo->loid; - blocker= blocker->next); + blocker= blocker->next) /* no-op */; if (blocker) break; } } - if (!blocker) + if (!blocker) /* free to go */ break; - wait_for= lm->loid_to_lo(blocker->loid); + wait_for= lm->loid_to_tlo(blocker->loid); } + /* ok, we're here - the wait is inevitable */ lo->waiting_for= wait_for; - if (!lo->waiting_lock) /* first iteration */ + if (!lo->waiting_lock) /* first iteration of the for() loop */ { /* lock upgrade or new lock request ? */ if (old) { + /* upgrade - add the lock to the _start_ of the wait queue */ new->prev= 0; if ((new->next= table->wait_queue_out)) new->next->prev= new; @@ -215,6 +342,7 @@ tablockman_getlock(TABLOCKMAN *lm, TABLE_LOCK_OWNER *lo, LOCKED_TABLE *table, en } else { + /* new lock - add the lock to the _end_ of the wait queue */ new->next= 0; if ((new->prev= table->wait_queue_in)) new->prev->next= new; @@ -222,7 +350,6 @@ tablockman_getlock(TABLOCKMAN *lm, TABLE_LOCK_OWNER *lo, LOCKED_TABLE *table, en if (!table->wait_queue_out) table->wait_queue_out=table->wait_queue_in; } - lo->waiting_lock= new; deadline= my_getsystime() + lm->lock_timeout * 10000; @@ -238,6 +365,7 @@ tablockman_getlock(TABLOCKMAN *lm, TABLE_LOCK_OWNER *lo, LOCKED_TABLE *table, en } } + /* now really wait */ pthread_mutex_lock(wait_for->mutex); pthread_mutex_unlock(& table->mutex); @@ -245,32 +373,30 @@ tablockman_getlock(TABLOCKMAN *lm, TABLE_LOCK_OWNER *lo, LOCKED_TABLE *table, en pthread_mutex_unlock(wait_for->mutex); pthread_mutex_lock(& table->mutex); + + /* ... and repeat from the beginning */ } + /* yeah! we can place the lock now */ + /* remove the lock from the wait queue, if it was there */ if (lo->waiting_lock) { - if (new->prev) - new->prev->next= new->next; - if (new->next) - new->next->prev= new->prev; - if (table->wait_queue_in == new) - table->wait_queue_in= new->prev; - if (table->wait_queue_out == new) - table->wait_queue_out= new->next; - + remove_from_wait_queue(new, table); lo->waiting_lock= 0; + lo->waiting_for= 0; } + /* add it to the list of all locks of this lock owner */ new->next_in_lo= lo->active_locks; lo->active_locks= new; + /* and to the list of active locks of this lock type */ new->prev= 0; if ((new->next= table->active_locks[new_lock-1])) new->next->prev= new; table->active_locks[new_lock-1]= new; - /* placing the lock */ - hash_insert(& table->active, (byte *)new); + /* remove the old lock from the hash, if upgrading */ if (old) { new->upgraded_from= old; @@ -279,45 +405,69 @@ tablockman_getlock(TABLOCKMAN *lm, TABLE_LOCK_OWNER *lo, LOCKED_TABLE *table, en else new->upgraded_from= 0; + /* and add a new lock to the hash, voila */ + hash_insert(& table->active, (byte *)new); + pthread_mutex_unlock(& table->mutex); return getlock_result[lock][lock]; } +/* + DESCRIPTION + release all locks belonging to a transaction. + signal waiters to continue +*/ void tablockman_release_locks(TABLOCKMAN *lm, TABLE_LOCK_OWNER *lo) { - TABLE_LOCK *lock, *tmp, *local_pool= 0, *local_pool_end; + TABLE_LOCK *lock, *local_pool= 0, *local_pool_end; + /* + instead of adding released locks to a pool one by one, we'll link + them in a list and add to a pool in one short action (under a mutex) + */ local_pool_end= lo->waiting_lock ? lo->waiting_lock : lo->active_locks; if (!local_pool_end) return; + /* release a waiting lock, if any */ if ((lock= lo->waiting_lock)) { + DBUG_ASSERT(lock->loid == lo->loid); pthread_mutex_lock(& lock->table->mutex); - - if (lock->prev) - lock->prev->next= lock->next; - if (lock->next) - lock->next->prev= lock->prev; - if (lock->table->wait_queue_in == lock) - lock->table->wait_queue_in= lock->prev; - if (lock->table->wait_queue_out == lock) - lock->table->wait_queue_out= lock->next; + remove_from_wait_queue(lock, lock->table); + + /* + a special case: if this lock was not the last in the wait queue + and it's compatible with the next lock, than the next lock + is waiting for our blocker though really it waits for us, indirectly. + Signal our blocker to release this next lock (after we removed our + lock from the wait queue, of course). + */ + if (lock->prev && + lock_compatibility_matrix[lock->prev->lock_type][lock->lock_type]) + { + pthread_mutex_lock(lo->waiting_for->mutex); + pthread_cond_broadcast(lo->waiting_for->cond); + pthread_mutex_unlock(lo->waiting_for->mutex); + } + lo->waiting_for= 0; pthread_mutex_unlock(& lock->table->mutex); lock->next= local_pool; local_pool= lock; - DBUG_ASSERT(lock->loid == lo->loid); } + /* now release granted locks */ lock= lo->active_locks; while (lock) { TABLE_LOCK *cur= lock; pthread_mutex_t *mutex= & lock->table->mutex; + DBUG_ASSERT(cur->loid == lo->loid); lock= lock->next_in_lo; + /* TODO ? group locks by table to reduce the number of mutex locks */ pthread_mutex_lock(mutex); hash_delete(& cur->table->active, (byte *)cur); @@ -330,17 +480,21 @@ void tablockman_release_locks(TABLOCKMAN *lm, TABLE_LOCK_OWNER *lo) cur->next= local_pool; local_pool= cur; - DBUG_ASSERT(cur->loid == lo->loid); pthread_mutex_unlock(mutex); } lo->waiting_lock= lo->active_locks= 0; + /* + okay, all locks released. now signal that we're leaving, + in case somebody's waiting for it + */ pthread_mutex_lock(lo->mutex); pthread_cond_broadcast(lo->cond); pthread_mutex_unlock(lo->mutex); + /* and push all freed locks to the lockman's pool */ pthread_mutex_lock(& lm->pool_mutex); local_pool_end->next= lm->pool; lm->pool= local_pool; @@ -350,7 +504,7 @@ void tablockman_release_locks(TABLOCKMAN *lm, TABLE_LOCK_OWNER *lo) void tablockman_init(TABLOCKMAN *lm, loid_to_tlo_func *func, uint timeout) { lm->pool= 0; - lm->loid_to_lo= func; + lm->loid_to_tlo= func; lm->lock_timeout= timeout; pthread_mutex_init(&lm->pool_mutex, MY_MUTEX_INIT_FAST); } @@ -366,12 +520,12 @@ void tablockman_destroy(TABLOCKMAN *lm) pthread_mutex_destroy(&lm->pool_mutex); } -void tablockman_init_locked_table(LOCKED_TABLE *lt) +void tablockman_init_locked_table(LOCKED_TABLE *lt, int initial_hash_size) { TABLE_LOCK *unused; bzero(lt, sizeof(*lt)); pthread_mutex_init(& lt->mutex, MY_MUTEX_INIT_FAST); - hash_init(& lt->active, &my_charset_bin, 10/*FIXME*/, + hash_init(& lt->active, &my_charset_bin, initial_hash_size, offsetof(TABLE_LOCK, loid), sizeof(unused->loid), 0, 0, 0); } @@ -381,6 +535,7 @@ void tablockman_destroy_locked_table(LOCKED_TABLE *lt) pthread_mutex_destroy(& lt->mutex); } +#ifdef EXTRA_DEBUG static char *lock2str[LOCK_TYPES+1]= {"N", "S", "X", "IS", "IX", "SIX", "LS", "LX", "SLX", "LSIX"}; @@ -396,4 +551,5 @@ void print_tlo(TABLE_LOCK_OWNER *lo) printf("!"); printf("\n"); } +#endif diff --git a/storage/maria/tablockman.h b/storage/maria/tablockman.h index 4b2d165af54..d307b2c844f 100644 --- a/storage/maria/tablockman.h +++ b/storage/maria/tablockman.h @@ -51,34 +51,38 @@ typedef TABLE_LOCK_OWNER *loid_to_tlo_func(uint16); typedef struct { pthread_mutex_t pool_mutex; - TABLE_LOCK *pool; + TABLE_LOCK *pool; /* lifo pool of free locks */ uint lock_timeout; - loid_to_tlo_func *loid_to_lo; + loid_to_tlo_func *loid_to_tlo; /* for mapping loid to TABLE_LOCK_OWNER */ } TABLOCKMAN; struct st_table_lock_owner { - TABLE_LOCK *active_locks, *waiting_lock; - TABLE_LOCK_OWNER *waiting_for; - pthread_cond_t *cond; /* transactions waiting for this, wait on 'cond' */ - pthread_mutex_t *mutex; /* mutex is required to use 'cond' */ - uint16 loid; + TABLE_LOCK *active_locks; /* list of active locks */ + TABLE_LOCK *waiting_lock; /* waiting lock (one lock only) */ + TABLE_LOCK_OWNER *waiting_for; /* transaction we're wating for */ + pthread_cond_t *cond; /* transactions waiting for us, wait on 'cond' */ + pthread_mutex_t *mutex; /* mutex is required to use 'cond' */ + uint16 loid; /* Lock Owner IDentifier */ }; struct st_locked_table { - pthread_mutex_t mutex; - HASH active; // fast to remove - TABLE_LOCK *active_locks[LOCK_TYPES]; // fast to see a conflict - TABLE_LOCK *wait_queue_in, *wait_queue_out; + pthread_mutex_t mutex; /* mutex for everything below */ + HASH active; /* active locks ina hash */ + TABLE_LOCK *active_locks[LOCK_TYPES]; /* dl-list of locks per type */ + TABLE_LOCK *wait_queue_in, *wait_queue_out; /* wait deque */ }; void tablockman_init(TABLOCKMAN *, loid_to_tlo_func *, uint); void tablockman_destroy(TABLOCKMAN *); enum lockman_getlock_result tablockman_getlock(TABLOCKMAN *, TABLE_LOCK_OWNER *, - LOCKED_TABLE *, - enum lock_type lock); + LOCKED_TABLE *, enum lock_type); void tablockman_release_locks(TABLOCKMAN *, TABLE_LOCK_OWNER *); -void tablockman_init_locked_table(LOCKED_TABLE *); +void tablockman_init_locked_table(LOCKED_TABLE *, int); +void tablockman_destroy_locked_table(LOCKED_TABLE *); + +#ifdef EXTRA_DEBUG void print_tlo(TABLE_LOCK_OWNER *); +#endif #endif diff --git a/storage/maria/unittest/lockman1-t.c b/storage/maria/unittest/lockman1-t.c index c9a4ff98f2a..ab4b7beba9f 100644 --- a/storage/maria/unittest/lockman1-t.c +++ b/storage/maria/unittest/lockman1-t.c @@ -288,10 +288,10 @@ int main() for (i= 0; i < Ntbls; i++) { - tablockman_init_locked_table(ltarray+i); + tablockman_init_locked_table(ltarray+i, Nlos); } - //test_tablockman_simple(); + test_tablockman_simple(); #define CYCLES 10000 #define THREADS Nlos /* don't change this line */ diff --git a/storage/maria/unittest/lockman2-t.c b/storage/maria/unittest/lockman2-t.c index 72b3993e4b4..bde0766c885 100644 --- a/storage/maria/unittest/lockman2-t.c +++ b/storage/maria/unittest/lockman2-t.c @@ -115,6 +115,20 @@ void test_tablockman_simple() lock_conflict(2, 1, X); unlock_all(1); unlock_all(2); + + lock_ok_i(1, 1, IS); + lock_conflict(2, 1, X); + lock_conflict(3, 1, IS); + unlock_all(1); + unlock_all(2); + unlock_all(3); + + lock_ok_a(1, 1, S); + lock_conflict(2, 1, IX); + lock_conflict(3, 1, IS); + unlock_all(1); + unlock_all(2); + unlock_all(3); } int rt_num_threads; @@ -273,7 +287,7 @@ int main() for (i= 0; i < Ntbls; i++) { - tablockman_init_locked_table(ltarray+i); + tablockman_init_locked_table(ltarray+i, Nlos); } test_tablockman_simple(); @@ -285,7 +299,7 @@ int main() Nrows= 100; Ntables= 10; table_lock_ratio= 10; - run_test("\"random lock\" stress test", test_lockman, THREADS, CYCLES); + //run_test("\"random lock\" stress test", test_lockman, THREADS, CYCLES); #if 0 /* "real-life" simulation - many rows, no table locks */ Nrows= 1000000; -- cgit v1.2.1 From f6edbbc85f0566cc6f79240529d42b2a9d9f7add Mon Sep 17 00:00:00 2001 From: unknown Date: Fri, 10 Nov 2006 16:18:10 +0100 Subject: Maria: importing change made to MyISAM's mi_test_all, into Maria's ma_test_all This makes an expected warning message about the index file's size, go away, as intended. storage/maria/ma_test_all.sh: importing change made to MyISAM's mi_test_all, into Maria's ma_test_all This makes an expected warning message about the index file's size, go away, as intended. --- storage/maria/ma_test_all.sh | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/storage/maria/ma_test_all.sh b/storage/maria/ma_test_all.sh index ec4d1778db9..73234d93ae4 100755 --- a/storage/maria/ma_test_all.sh +++ b/storage/maria/ma_test_all.sh @@ -79,7 +79,8 @@ if test -f ma_test1$MACH ; then suffix=$MACH ; else suffix=""; fi # check of maria_pack / maria_chk ./maria_pack$suffix --force -s test1 -./maria_chk$suffix -es test1 +# Ignore error for index file +./maria_chk$suffix -es test1 2>&1 >& /dev/null ./maria_chk$suffix -rqs test1 ./maria_chk$suffix -es test1 ./maria_chk$suffix -rs test1 -- cgit v1.2.1 From 41f890bc98fe26128ab18737e4acc846db09b74f Mon Sep 17 00:00:00 2001 From: unknown Date: Sun, 12 Nov 2006 14:44:12 +0100 Subject: minor unittest fixes storage/maria/lockman.c: restore removed lines storage/maria/unittest/lockman2-t.c: fix the plan --- storage/maria/lockman.c | 11 +++++++---- storage/maria/unittest/lockman2-t.c | 2 +- 2 files changed, 8 insertions(+), 5 deletions(-) diff --git a/storage/maria/lockman.c b/storage/maria/lockman.c index 31867d3903d..7a6b97b3d51 100644 --- a/storage/maria/lockman.c +++ b/storage/maria/lockman.c @@ -103,10 +103,6 @@ whether it's possible to lock the row, but no need to lock it - perhaps the thread has a loose lock on this table). This is defined by getlock_result[] table. - - TODO optimization: table locks - they have completely - different characteristics. long lists, few distinct resources - - slow to scan, [possibly] high retry rate */ #include @@ -487,6 +483,9 @@ static int lockdelete(LOCK * volatile *head, LOCK *node, LF_PINS *pins) res= lockfind(head, node, &cursor, pins); DBUG_ASSERT(res & ALREADY_HAVE); + if (cursor.upgrade_from) + cursor.upgrade_from->flags&= ~IGNORE_ME; + /* XXX this does not work with savepoints, as old lock is left ignored. It cannot be unignored, as would basically mean moving the lock back @@ -506,7 +505,11 @@ static int lockdelete(LOCK * volatile *head, LOCK *node, LF_PINS *pins) lockfind(head, node, &cursor, pins); } else + { res= REPEAT_ONCE_MORE; + if (cursor.upgrade_from) + cursor.upgrade_from->flags|= IGNORE_ME; + } } while (res == REPEAT_ONCE_MORE); _lf_unpin(pins, 0); _lf_unpin(pins, 1); diff --git a/storage/maria/unittest/lockman2-t.c b/storage/maria/unittest/lockman2-t.c index bde0766c885..c515fc7ecd7 100644 --- a/storage/maria/unittest/lockman2-t.c +++ b/storage/maria/unittest/lockman2-t.c @@ -264,7 +264,7 @@ int main() my_init(); pthread_mutex_init(&rt_mutex, 0); - plan(35); + plan(39); if (my_atomic_initialize()) return exit_status(); -- cgit v1.2.1 From e5858e1fb429617bf16f94031b660a3694e2acdd Mon Sep 17 00:00:00 2001 From: unknown Date: Mon, 13 Nov 2006 00:26:29 +0200 Subject: postreview fixes fixed bug in the pagecache which lead to assertion in multithread test include/pagecache.h: post review fixes: - comments fixed - types fixed - keyword 'extern' added to all interface functions mysys/mf_pagecache.c: postreview fixes: - comments fixed and added - types fixed - typo fixed Added write locking flag set when we took a block from LRU and going to free it (to prevent locking/using it for write) mysys/test_pagecache_consist.c: pagecache size reduced (to be able reproduce problems found by Guilhem) typo in the comment fixed --- include/pagecache.h | 50 +++++------ mysys/mf_pagecache.c | 190 ++++++++++++++++++++++++++++++----------- mysys/test_pagecache_consist.c | 4 +- 3 files changed, 169 insertions(+), 75 deletions(-) diff --git a/include/pagecache.h b/include/pagecache.h index 018e695fa27..4d64070ad62 100644 --- a/include/pagecache.h +++ b/include/pagecache.h @@ -119,14 +119,14 @@ typedef struct st_pagecache ulonglong time; /* total number of block link operations */ uint hash_entries; /* max number of entries in the hash table */ int hash_links; /* max number of hash links */ - int hash_links_used; /* number of hash links currently used */ + int hash_links_used; /* number of hash links taken from free links pool */ int disk_blocks; /* max number of blocks in the cache */ ulong blocks_used; /* maximum number of concurrently used blocks */ ulong blocks_unused; /* number of currently unused blocks */ ulong blocks_changed; /* number of currently dirty blocks */ ulong warm_blocks; /* number of blocks in warm sub-chain */ ulong cnt_for_resize_op; /* counter to block resize operation */ - long blocks_available; /* number of blocks available in the LRU chain */ + ulong blocks_available; /* number of blocks available in the LRU chain */ PAGECACHE_HASH_LINK **hash_root;/* arr. of entries into hash table buckets */ PAGECACHE_HASH_LINK *hash_link_root;/* memory for hash table links */ PAGECACHE_HASH_LINK *free_hash_list;/* list of free hash links */ @@ -194,32 +194,32 @@ extern my_bool pagecache_write(PAGECACHE *pagecache, enum pagecache_page_pin pin, enum pagecache_write_mode write_mode, PAGECACHE_PAGE_LINK *link); -void pagecache_unlock_page(PAGECACHE *pagecache, - PAGECACHE_FILE *file, - maria_page_no_t pageno, - enum pagecache_page_lock lock, - enum pagecache_page_pin pin, - my_bool stamp_this_page, - LSN first_REDO_LSN_for_page); -void pagecache_unlock(PAGECACHE *pagecache, - PAGECACHE_PAGE_LINK *link, - enum pagecache_page_lock lock, - enum pagecache_page_pin pin, - my_bool stamp_this_page, - LSN first_REDO_LSN_for_page); -void pagecache_unpin_page(PAGECACHE *pagecache, - PAGECACHE_FILE *file, - maria_page_no_t pageno); -void pagecache_unpin(PAGECACHE *pagecache, - PAGECACHE_PAGE_LINK *link); +extern void pagecache_unlock_page(PAGECACHE *pagecache, + PAGECACHE_FILE *file, + maria_page_no_t pageno, + enum pagecache_page_lock lock, + enum pagecache_page_pin pin, + my_bool stamp_this_page, + LSN first_REDO_LSN_for_page); +extern void pagecache_unlock(PAGECACHE *pagecache, + PAGECACHE_PAGE_LINK *link, + enum pagecache_page_lock lock, + enum pagecache_page_pin pin, + my_bool stamp_this_page, + LSN first_REDO_LSN_for_page); +extern void pagecache_unpin_page(PAGECACHE *pagecache, + PAGECACHE_FILE *file, + maria_page_no_t pageno); +extern void pagecache_unpin(PAGECACHE *pagecache, + PAGECACHE_PAGE_LINK *link); extern int flush_pagecache_blocks(PAGECACHE *keycache, PAGECACHE_FILE *file, enum flush_type type); -my_bool pagecache_delete_page(PAGECACHE *pagecache, - PAGECACHE_FILE *file, - maria_page_no_t pageno, - enum pagecache_page_lock lock, - my_bool flush); +extern my_bool pagecache_delete_page(PAGECACHE *pagecache, + PAGECACHE_FILE *file, + maria_page_no_t pageno, + enum pagecache_page_lock lock, + my_bool flush); extern void end_pagecache(PAGECACHE *keycache, my_bool cleanup); C_MODE_END diff --git a/mysys/mf_pagecache.c b/mysys/mf_pagecache.c index 4fa814d8188..e1ccab9e06f 100755 --- a/mysys/mf_pagecache.c +++ b/mysys/mf_pagecache.c @@ -1,4 +1,4 @@ -/* Copyright (C) 2000 MySQL AB +/* Copyright (C) 2000-2006 MySQL AB This program is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by @@ -26,11 +26,11 @@ When a new block is required it is first tried to pop one from the stack. If the stack is empty, it is tried to get a never-used block from the pool. If this is empty too, then a block is taken from the LRU ring, flushing it - to disk, if neccessary. This is handled in find_key_block(). + to disk, if necessary. This is handled in find_key_block(). With the new free list, the blocks can have three temperatures: hot, warm and cold (which is free). This is remembered in the block header by the enum BLOCK_TEMPERATURE temperature variable. Remembering the - temperature is neccessary to correctly count the number of warm blocks, + temperature is necessary to correctly count the number of warm blocks, which is required to decide when blocks are allowed to become hot. Whenever a block is inserted to another (sub-)chain, we take the old and new temperature into account to decide if we got one more or less warm block. @@ -93,16 +93,22 @@ */ #define SERIALIZED_READ_FROM_CACHE yes -#define BLOCK_INFO(B) DBUG_PRINT("info", \ - ("block 0x%lx, file %lu, page %lu, s %0x", \ - (ulong)(B), \ - (ulong)((B)->hash_link ? \ - (B)->hash_link->file.file : \ - 0), \ - (ulong)((B)->hash_link ? \ - (B)->hash_link->pageno : \ - 0), \ - (B)->status)) +#define BLOCK_INFO(B) \ + DBUG_PRINT("info", \ + ("block 0x%lx, file %lu, page %lu, s %0x, hshL 0x%lx, req %u/%u", \ + (ulong)(B), \ + (ulong)((B)->hash_link ? \ + (B)->hash_link->file.file : \ + 0), \ + (ulong)((B)->hash_link ? \ + (B)->hash_link->pageno : \ + 0), \ + (B)->status, \ + (ulong)(B)->hash_link, \ + (uint) (B)->requests, \ + (uint)((B)->hash_link ? \ + (B)->hash_link->requests : \ + 0))) /* TODO: put it to my_static.c */ my_bool my_disable_flush_pagecache_blocks= 0; @@ -147,7 +153,7 @@ struct st_pagecache_hash_link }; /* simple states of a block */ -#define BLOCK_ERROR 1 /* an error occured when performing disk i/o */ +#define BLOCK_ERROR 1 /* an error occurred when performing disk i/o */ #define BLOCK_READ 2 /* the is page in the block buffer */ #define BLOCK_IN_SWITCH 4 /* block is preparing to read new page */ #define BLOCK_REASSIGNED 8 /* block does not accept requests for old page */ @@ -211,7 +217,20 @@ typedef struct st_pagecache_lock_info struct st_my_thread_var *thread; my_bool write_lock; } PAGECACHE_LOCK_INFO; -/* service functions */ + + +/* service functions maintain debugging info about pin & lock */ + + +/* + Links information about thread pinned/locked the block to the list + + SYNOPSIS + info_link() + list the list to link in + node the node which should be linked +*/ + void info_link(PAGECACHE_PIN_INFO **list, PAGECACHE_PIN_INFO *node) { if ((node->next= *list)) @@ -219,11 +238,38 @@ void info_link(PAGECACHE_PIN_INFO **list, PAGECACHE_PIN_INFO *node) *list= node; node->prev= list; } + + +/* + Unlinks information about thread pinned/locked the block from the list + + SYNOPSIS + info_unlink() + node the node which should be unlinked +*/ + void info_unlink(PAGECACHE_PIN_INFO *node) { if ((*node->prev= node->next)) node->next->prev= node->prev; } + + +/* + Finds information about given thread in the list of threads which + pinned/locked this block. + + SYNOPSIS + info_find() + list the list where to find the thread + thread thread ID (reference to the st_my_thread_var + of the thread) + + RETURN + 0 - the thread was not found + pointer to the information node of the thread in the list +*/ + PAGECACHE_PIN_INFO *info_find(PAGECACHE_PIN_INFO *list, struct st_my_thread_var *thread) { @@ -263,8 +309,8 @@ struct st_pagecache_block_link #ifdef PAGECACHE_DEBUG /* debug checks */ -bool info_check_pin(PAGECACHE_BLOCK_LINK *block, - enum pagecache_page_pin mode) +my_bool info_check_pin(PAGECACHE_BLOCK_LINK *block, + enum pagecache_page_pin mode) { struct st_my_thread_var *thread= my_thread_var; DBUG_ENTER("info_check_pin"); @@ -305,9 +351,10 @@ bool info_check_pin(PAGECACHE_BLOCK_LINK *block, } DBUG_RETURN(0); } -bool info_check_lock(PAGECACHE_BLOCK_LINK *block, - enum pagecache_page_lock lock, - enum pagecache_page_pin pin) + +my_bool info_check_lock(PAGECACHE_BLOCK_LINK *block, + enum pagecache_page_lock lock, + enum pagecache_page_pin pin) { struct st_my_thread_var *thread= my_thread_var; DBUG_ENTER("info_check_lock"); @@ -328,7 +375,7 @@ bool info_check_lock(PAGECACHE_BLOCK_LINK *block, break; case PAGECACHE_LOCK_LEFT_READLOCKED: DBUG_ASSERT(pin == PAGECACHE_PIN_LEFT_UNPINNED || - pin == PAGECACHE_PIN_LEFT_UNPINNED); + pin == PAGECACHE_PIN_LEFT_PINNED); if (info == 0 || info->write_lock) { DBUG_PRINT("info", @@ -650,6 +697,7 @@ int init_pagecache(PAGECACHE *pagecache, my_size_t use_mem, pagecache->shift= my_bit_log2(block_size); DBUG_PRINT("info", ("block_size: %u", block_size)); + DBUG_ASSERT((1 << pagecache->shift) == block_size); blocks= (int) (use_mem / (sizeof(PAGECACHE_BLOCK_LINK) + 2 * sizeof(PAGECACHE_HASH_LINK) + @@ -755,7 +803,7 @@ int init_pagecache(PAGECACHE *pagecache, my_size_t use_mem, } pagecache->blocks= pagecache->disk_blocks > 0 ? pagecache->disk_blocks : 0; - DBUG_RETURN((uint) pagecache->disk_blocks); + DBUG_RETURN((uint) pagecache->blocks); err: error= my_errno; @@ -945,7 +993,7 @@ void change_pagecache_param(PAGECACHE *pagecache, uint division_limit, /* - Remove page cache from memory + Flushes and removes page cache from memory SYNOPSIS end_pagecache() @@ -1240,7 +1288,7 @@ static void link_block(PAGECACHE *pagecache, PAGECACHE_BLOCK_LINK *block, my_bool hot, my_bool at_end) { PAGECACHE_BLOCK_LINK *ins; - PAGECACHE_BLOCK_LINK **pins; + PAGECACHE_BLOCK_LINK **ptr_ins; KEYCACHE_DBUG_ASSERT(! (block->hash_link && block->hash_link->requests)); #ifdef THREAD @@ -1285,8 +1333,8 @@ static void link_block(PAGECACHE *pagecache, PAGECACHE_BLOCK_LINK *block, KEYCACHE_DBUG_ASSERT(! (!hot && pagecache->waiting_for_block.last_thread)); /* Condition not transformed using DeMorgan, to keep the text identical */ #endif /* THREAD */ - pins= hot ? &pagecache->used_ins : &pagecache->used_last; - ins= *pins; + ptr_ins= hot ? &pagecache->used_ins : &pagecache->used_last; + ins= *ptr_ins; if (ins) { ins->next_used->prev_used= &block->next_used; @@ -1294,7 +1342,7 @@ static void link_block(PAGECACHE *pagecache, PAGECACHE_BLOCK_LINK *block, block->prev_used= &ins->next_used; ins->next_used= block; if (at_end) - *pins= block; + *ptr_ins= block; } else { @@ -1363,6 +1411,16 @@ static void unlink_block(PAGECACHE *pagecache, PAGECACHE_BLOCK_LINK *block) /* Register requests for a block + + SYNOPSIS + reg_requests() + pagecache this page cache reference + block the block we request reference + count how many requests we register (it is 1 everywhere) + + NOTE + Registration of request means we are going to use this block so we exclude + it from the LRU if it is first request */ static void reg_requests(PAGECACHE *pagecache, PAGECACHE_BLOCK_LINK *block, int count) @@ -1375,7 +1433,7 @@ static void reg_requests(PAGECACHE *pagecache, PAGECACHE_BLOCK_LINK *block, if (! block->requests) /* First request for the block unlinks it */ unlink_block(pagecache, block); - block->requests+=count; + block->requests+= count; DBUG_VOID_RETURN; } @@ -1461,8 +1519,11 @@ static void unreg_request(PAGECACHE *pagecache, static inline void remove_reader(PAGECACHE_BLOCK_LINK *block) { + DBUG_ENTER("remove_reader"); + BLOCK_INFO(block); if (! --block->hash_link->requests && block->condvar) pagecache_pthread_cond_signal(block->condvar); + DBUG_VOID_RETURN; } @@ -1563,8 +1624,21 @@ static void unlink_hash(PAGECACHE *pagecache, PAGECACHE_HASH_LINK *hash_link) hash_link->next= pagecache->free_hash_list; pagecache->free_hash_list= hash_link; } + + /* - Get the hash link for the page if it is inthe cache + Get the hash link for the page if it is in the cache + + SYNOPSIS + get_present_hash_link() + pagecache Pagecache reference + file file ID + pageno page number in the file + start where to put pointer to found hash link (for + direct referring it) + + RETURN + found hashlink pointer */ static PAGECACHE_HASH_LINK *get_present_hash_link(PAGECACHE *pagecache, @@ -1766,8 +1840,8 @@ restart: /* Remove block to invalidate the page in the block buffer as we are going to write directly on disk. - Although we have an exlusive lock for the updated key part - the control can be yieded by the current thread as we might + Although we have an exclusive lock for the updated key part + the control can be yielded by the current thread as we might have unfinished readers of other key parts in the block buffer. Still we are guaranteed not to have any readers of the key part we are writing into until the block is @@ -1777,7 +1851,7 @@ restart: free_block(pagecache, block); return 0; } - /* Wait intil the page is flushed on disk */ + /* Wait until the page is flushed on disk */ hash_link->requests--; { #ifdef THREAD @@ -1943,12 +2017,17 @@ restart: reg_requests(pagecache, block,1); hash_link->block= block; } + else + { + DBUG_ASSERT((block->status & BLOCK_WRLOCK) == 0); + } if (block->hash_link != hash_link && ! (block->status & BLOCK_IN_SWITCH) ) { /* this is a primary request for a new page */ - block->status|= BLOCK_IN_SWITCH; + DBUG_ASSERT((block->status & BLOCK_WRLOCK) == 0); + block->status|= (BLOCK_IN_SWITCH | BLOCK_WRLOCK); KEYCACHE_DBUG_PRINT("find_key_block", ("got block %u for new page", @@ -1995,7 +2074,7 @@ restart: } link_to_file_list(pagecache, block, file, (my_bool)(block->hash_link ? 1 : 0)); - DBUG_ASSERT((block->status & BLOCK_WRLOCK) == 0); + BLOCK_INFO(block); block->status= error? BLOCK_ERROR : 0; #ifndef DBUG_OFF block->type= PAGECACHE_EMPTY_PAGE; @@ -2094,7 +2173,7 @@ void pagecache_remove_pin(PAGECACHE_BLOCK_LINK *block) DBUG_VOID_RETURN; } #ifdef PAGECACHE_DEBUG -void pagecache_add_lock(PAGECACHE_BLOCK_LINK *block, bool wl) +void pagecache_add_lock(PAGECACHE_BLOCK_LINK *block, my_bool wl) { PAGECACHE_LOCK_INFO *info= (PAGECACHE_LOCK_INFO *)my_malloc(sizeof(PAGECACHE_LOCK_INFO), MYF(0)); @@ -2112,7 +2191,7 @@ void pagecache_remove_lock(PAGECACHE_BLOCK_LINK *block) info_unlink((PAGECACHE_PIN_INFO *)info); my_free((gptr)info, MYF(0)); } -void pagecache_change_lock(PAGECACHE_BLOCK_LINK *block, bool wl) +void pagecache_change_lock(PAGECACHE_BLOCK_LINK *block, my_bool wl) { PAGECACHE_LOCK_INFO *info= (PAGECACHE_LOCK_INFO *)info_find((PAGECACHE_PIN_INFO *)block->lock_list, @@ -2144,7 +2223,7 @@ my_bool pagecache_lock_block(PAGECACHE *pagecache, { DBUG_ENTER("pagecache_lock_block"); BLOCK_INFO(block); - if (block->status & BLOCK_WRLOCK) + while (block->status & BLOCK_WRLOCK) { DBUG_PRINT("info", ("fail to lock, waiting...")); /* Lock failed we will wait */ @@ -2168,6 +2247,7 @@ my_bool pagecache_lock_block(PAGECACHE *pagecache, } /* we are doing it by global cache mutex protectio, so it is OK */ block->status|= BLOCK_WRLOCK; + DBUG_PRINT("info", ("WR lock set, block 0x%lx", (ulong)block)); DBUG_RETURN(0); } @@ -2177,6 +2257,7 @@ void pagecache_unlock_block(PAGECACHE_BLOCK_LINK *block) BLOCK_INFO(block); DBUG_ASSERT(block->status & BLOCK_WRLOCK); block->status&= ~BLOCK_WRLOCK; + DBUG_PRINT("info", ("WR lock reset, block 0x%lx", (ulong)block)); #ifdef THREAD /* release all threads waiting for write lock */ if (block->wqueue[COND_FOR_WRLOCK].last_thread) @@ -2434,6 +2515,7 @@ void pagecache_unlock_page(PAGECACHE *pagecache, inc_counter_for_resize_op(pagecache); block= find_key_block(pagecache, file, pageno, 0, 0, 0, &page_st); + BLOCK_INFO(block); DBUG_ASSERT(block != 0 && page_st == PAGE_READ); if (stamp_this_page) { @@ -2776,17 +2858,18 @@ restart: inc_counter_for_resize_op(pagecache); pagecache->global_cache_r_requests++; - block= find_key_block(pagecache, file, pageno, level, 0, - (((pin == PAGECACHE_PIN_LEFT_PINNED) || - (pin == PAGECACHE_UNPIN)) ? 0 : 1), - &page_st); + block= find_key_block(pagecache, file, pageno, level, + ((lock == PAGECACHE_LOCK_WRITE) ? 1 : 0), + (((pin == PAGECACHE_PIN_LEFT_PINNED) || + (pin == PAGECACHE_UNPIN)) ? 0 : 1), + &page_st); DBUG_ASSERT(block->type == PAGECACHE_EMPTY_PAGE || block->type == type); block->type= type; if (pagecache_make_lock_and_pin(pagecache, block, lock, pin)) { /* - We failed to writelock the block, cache is unlocked, and last write + We failed to write lock the block, cache is unlocked, and last write lock is released, we will try to get the block again. */ pagecache_pthread_mutex_unlock(&pagecache->cache_lock); @@ -3095,11 +3178,16 @@ restart: inc_counter_for_resize_op(pagecache); pagecache->global_cache_w_requests++; - block= find_key_block(pagecache, file, pageno, level, - (write_mode == PAGECACHE_WRITE_DONE ? 0 : 1), - (((pin == PAGECACHE_PIN_LEFT_PINNED) || - (pin == PAGECACHE_UNPIN)) ? 0 : 1), - &page_st); + { + int need_wrlock= (write_mode != PAGECACHE_WRITE_DONE && + lock != PAGECACHE_LOCK_LEFT_WRITELOCKED && + lock != PAGECACHE_LOCK_WRITE_UNLOCK && + lock != PAGECACHE_LOCK_WRITE_TO_READ); + block= find_key_block(pagecache, file, pageno, level, + (need_wrlock ? 1 : 0), + (need_wrlock ? 1 : 0), + &page_st); + } if (!block) { DBUG_ASSERT(write_mode != PAGECACHE_WRITE_DONE); @@ -3178,9 +3266,15 @@ restart: } /* Unregister the request */ + block->hash_link->requests--; if (pin != PAGECACHE_PIN_LEFT_PINNED && pin != PAGECACHE_PIN) - unreg_request(pagecache, block, 1); + { + if (write_mode != PAGECACHE_WRITE_DONE) + { + unreg_request(pagecache, block, 1); + } + } else *link= (PAGECACHE_PAGE_LINK)block; diff --git a/mysys/test_pagecache_consist.c b/mysys/test_pagecache_consist.c index 1cc54af2460..86698fbb860 100755 --- a/mysys/test_pagecache_consist.c +++ b/mysys/test_pagecache_consist.c @@ -11,7 +11,7 @@ #include /*#define PAGE_SIZE 65536*/ -#define PCACHE_SIZE (PAGE_SIZE*1024*20) +#define PCACHE_SIZE (PAGE_SIZE*1024*8) #ifndef DBUG_OFF static const char* default_dbug_option; @@ -59,7 +59,7 @@ static uint flush_divider= 1000; #endif /*TEST_HIGH_CONCURENCY*/ -/* check page consistemcy */ +/* check page consistency */ uint check_page(uchar *buff, ulong offset, int page_locked, int page_no, int tag) { -- cgit v1.2.1 From e2cddea9b5ca3ad796f3b6b77d33c1591fc30a98 Mon Sep 17 00:00:00 2001 From: unknown Date: Tue, 14 Nov 2006 20:27:38 +0200 Subject: make it possible to use the tree on caseinsensitive file systems BitKeeper/deleted/.del-cmakelists_.txt: Delete: storage/maria/cmakelists_.txt --- storage/maria/cmakelists.txt | 26 -------------------------- 1 file changed, 26 deletions(-) delete mode 100644 storage/maria/cmakelists.txt diff --git a/storage/maria/cmakelists.txt b/storage/maria/cmakelists.txt deleted file mode 100644 index 3ba7aba4555..00000000000 --- a/storage/maria/cmakelists.txt +++ /dev/null @@ -1,26 +0,0 @@ -SET(CMAKE_CXX_FLAGS_DEBUG "${CMAKE_CXX_FLAGS_DEBUG} -DSAFEMALLOC -DSAFE_MUTEX") -SET(CMAKE_C_FLAGS_DEBUG "${CMAKE_C_FLAGS_DEBUG} -DSAFEMALLOC -DSAFE_MUTEX") - -INCLUDE_DIRECTORIES(${CMAKE_SOURCE_DIR}/include) -ADD_LIBRARY(myisam ft_boolean_search.c ft_nlq_search.c ft_parser.c ft_static.c ft_stem.c - ft_stopwords.c ft_update.c mi_cache.c mi_changed.c mi_check.c - mi_checksum.c mi_close.c mi_create.c mi_dbug.c mi_delete.c - mi_delete_all.c mi_delete_table.c mi_dynrec.c mi_extra.c mi_info.c - mi_key.c mi_keycache.c mi_locking.c mi_log.c mi_open.c - mi_packrec.c mi_page.c mi_panic.c mi_preload.c mi_range.c mi_rename.c - mi_rfirst.c mi_rlast.c mi_rnext.c mi_rnext_same.c mi_rprev.c mi_rrnd.c - mi_rsame.c mi_rsamepos.c mi_scan.c mi_search.c mi_static.c mi_statrec.c - mi_unique.c mi_update.c mi_write.c rt_index.c rt_key.c rt_mbr.c - rt_split.c sort.c sp_key.c ft_eval.h myisamdef.h rt_index.h mi_rkey.c) - -ADD_EXECUTABLE(myisam_ftdump myisam_ftdump.c) -TARGET_LINK_LIBRARIES(myisam_ftdump myisam mysys dbug strings zlib wsock32) - -ADD_EXECUTABLE(myisamchk myisamchk.c) -TARGET_LINK_LIBRARIES(myisamchk myisam mysys dbug strings zlib wsock32) - -ADD_EXECUTABLE(myisamlog myisamlog.c) -TARGET_LINK_LIBRARIES(myisamlog myisam mysys dbug strings zlib wsock32) - -ADD_EXECUTABLE(myisampack myisampack.c) -TARGET_LINK_LIBRARIES(myisampack myisam mysys dbug strings zlib wsock32) -- cgit v1.2.1 From 9a06c5023676d3750e54c5ef91e5b7ca4feb0fea Mon Sep 17 00:00:00 2001 From: unknown Date: Wed, 15 Nov 2006 12:58:37 +0200 Subject: fixed typo storage/maria/CMakeLists.txt: Rename: storage/maria/_MakeLists.txt -> storage/maria/CMakeLists.txt --- storage/maria/CMakeLists.txt | 1 + storage/maria/_MakeLists.txt | 1 - 2 files changed, 1 insertion(+), 1 deletion(-) create mode 100644 storage/maria/CMakeLists.txt delete mode 100644 storage/maria/_MakeLists.txt diff --git a/storage/maria/CMakeLists.txt b/storage/maria/CMakeLists.txt new file mode 100644 index 00000000000..cfe23054e2f --- /dev/null +++ b/storage/maria/CMakeLists.txt @@ -0,0 +1 @@ +# empty for the moment; will fill it when we build under Windows diff --git a/storage/maria/_MakeLists.txt b/storage/maria/_MakeLists.txt deleted file mode 100644 index cfe23054e2f..00000000000 --- a/storage/maria/_MakeLists.txt +++ /dev/null @@ -1 +0,0 @@ -# empty for the moment; will fill it when we build under Windows -- cgit v1.2.1 From 915cebdd53fe5071dc9443a236a798764f504c22 Mon Sep 17 00:00:00 2001 From: unknown Date: Thu, 16 Nov 2006 15:40:08 +0100 Subject: post-review fixes. tablockman: fixed a bug in finding a blocker lock mysys/my_getsystime.c: this is no longer true storage/maria/lockman.h: post-review fixes storage/maria/tablockman.h: post-review fixes storage/maria/unittest/lockman-t.c: post-review fixes storage/maria/unittest/lockman1-t.c: post-review fixes storage/maria/unittest/lockman2-t.c: post-review fixes include/my_atomic.h: moved intptr definition to my_global.h storage/maria/tablockman.c: post-review fixes BUILD/SETUP.sh: add -DMY_LF_EXTRA_DEBUG to debug builds include/atomic/nolock.h: suppress warning include/my_global.h: suppress warning mysys/lf_alloc-pin.c: post-review fixes mysys/lf_dynarray.c: post-review fixes mysys/lf_hash.c: post-review fixes storage/maria/trnman.c: suppress warning include/lf.h: post-review fix --- BUILD/SETUP.sh | 3 +- include/atomic/nolock.h | 2 +- include/lf.h | 47 +++-- include/my_atomic.h | 8 - include/my_global.h | 11 +- mysys/lf_alloc-pin.c | 29 +-- mysys/lf_dynarray.c | 35 ++-- mysys/lf_hash.c | 119 ++++++------ mysys/my_getsystime.c | 4 - storage/maria/lockman.h | 7 +- storage/maria/tablockman.c | 371 +++++++++++++++++++++++------------- storage/maria/tablockman.h | 52 ++--- storage/maria/trnman.c | 3 +- storage/maria/unittest/lockman-t.c | 4 + storage/maria/unittest/lockman1-t.c | 23 ++- storage/maria/unittest/lockman2-t.c | 54 ++++-- 16 files changed, 471 insertions(+), 301 deletions(-) diff --git a/BUILD/SETUP.sh b/BUILD/SETUP.sh index f5fc16d9c9e..4677087bc3b 100755 --- a/BUILD/SETUP.sh +++ b/BUILD/SETUP.sh @@ -117,8 +117,9 @@ valgrind_flags="-USAFEMALLOC -UFORCE_INIT_OF_VARS -DHAVE_purify " valgrind_flags="$valgrind_flags -DMYSQL_SERVER_SUFFIX=-valgrind-max" # # Used in -debug builds -debug_cflags="-DUNIV_MUST_NOT_INLINE -DEXTRA_DEBUG -DFORCE_INIT_OF_VARS " +debug_cflags="-DUNIV_MUST_NOT_INLINE -DEXTRA_DEBUG -DFORCE_INIT_OF_VARS" debug_cflags="$debug_cflags -DSAFEMALLOC -DPEDANTIC_SAFEMALLOC -DSAFE_MUTEX" +debug_cflags="$debug_cflags -DMY_LF_EXTRA_DEBUG" error_inject="--with-error-inject " # # Base C++ flags for all builds diff --git a/include/atomic/nolock.h b/include/atomic/nolock.h index a696e008f03..9d367657a69 100644 --- a/include/atomic/nolock.h +++ b/include/atomic/nolock.h @@ -32,7 +32,7 @@ #ifdef make_atomic_cas_body -typedef struct { } my_atomic_rwlock_t; +typedef struct { } my_atomic_rwlock_t __attribute__ ((unused)); #define my_atomic_rwlock_destroy(name) #define my_atomic_rwlock_init(name) #define my_atomic_rwlock_rdlock(name) diff --git a/include/lf.h b/include/lf.h index 3a3b5b28ade..c7198ab75b6 100644 --- a/include/lf.h +++ b/include/lf.h @@ -24,7 +24,7 @@ func() is a _func() protected by my_atomic_rwlock_wrlock() */ -#define lock_wrap(f,t,proto_args, args, lock) \ +#define lock_wrap(f, t, proto_args, args, lock) \ t _ ## f proto_args; \ static inline t f proto_args \ { \ @@ -35,7 +35,7 @@ static inline t f proto_args \ return ret; \ } -#define lock_wrap_void(f,proto_args, args, lock) \ +#define lock_wrap_void(f, proto_args, args, lock) \ void _ ## f proto_args; \ static inline void f proto_args \ { \ @@ -44,14 +44,14 @@ static inline void f proto_args \ my_atomic_rwlock_wrunlock(lock); \ } -#define nolock_wrap(f,t,proto_args, args) \ +#define nolock_wrap(f, t, proto_args, args) \ t _ ## f proto_args; \ static inline t f proto_args \ { \ return _ ## f args; \ } -#define nolock_wrap_void(f,proto_args, args) \ +#define nolock_wrap_void(f, proto_args, args) \ void _ ## f proto_args; \ static inline void f proto_args \ { \ @@ -80,14 +80,14 @@ void lf_dynarray_destroy(LF_DYNARRAY *array); nolock_wrap(lf_dynarray_value, void *, (LF_DYNARRAY *array, uint idx), - (array,idx)); + (array, idx)); lock_wrap(lf_dynarray_lvalue, void *, (LF_DYNARRAY *array, uint idx), - (array,idx), + (array, idx), &array->lock); nolock_wrap(lf_dynarray_iterate, int, (LF_DYNARRAY *array, lf_dynarray_func func, void *arg), - (array,func,arg)); + (array, func, arg)); /* pin manager for memory allocator, lf_alloc-pin.c @@ -115,9 +115,14 @@ typedef struct { uint32 volatile link; /* we want sizeof(LF_PINS) to be 128 to avoid false sharing */ char pad[128-sizeof(uint32)*2 - -sizeof(void *)*(LF_PINBOX_PINS+2)]; + -sizeof(LF_PINBOX *) + -sizeof(void *)*(LF_PINBOX_PINS+1)]; } LF_PINS; +/* + shortcut macros to do an atomic_wrlock on a structure that uses pins + (e.g. lf_hash). +*/ #define lf_rwlock_by_pins(PINS) \ my_atomic_rwlock_wrlock(&(PINS)->pinbox->pinstack.lock) #define lf_rwunlock_by_pins(PINS) \ @@ -131,11 +136,11 @@ typedef struct { #if defined(__GNUC__) && defined(MY_LF_EXTRA_DEBUG) #define LF_REQUIRE_PINS(N) \ static const char require_pins[LF_PINBOX_PINS-N]; \ - static const int LF_NUM_PINS_IN_THIS_FILE=N; + static const int LF_NUM_PINS_IN_THIS_FILE= N; #define _lf_pin(PINS, PIN, ADDR) \ ( \ - my_atomic_storeptr(&(PINS)->pin[PIN], (ADDR)), \ - assert(PIN < LF_NUM_PINS_IN_THIS_FILE) \ + assert(PIN < LF_NUM_PINS_IN_THIS_FILE), \ + my_atomic_storeptr(&(PINS)->pin[PIN], (ADDR)) \ ) #else #define LF_REQUIRE_PINS(N) @@ -151,7 +156,7 @@ typedef struct { } while (0) #define lf_unpin(PINS, PIN) lf_pin(PINS, PIN, NULL) #define _lf_assert_pin(PINS, PIN) assert((PINS)->pin[PIN] != 0) -#define _lf_assert_unpin(PINS, PIN) assert((PINS)->pin[PIN]==0) +#define _lf_assert_unpin(PINS, PIN) assert((PINS)->pin[PIN] == 0) void lf_pinbox_init(LF_PINBOX *pinbox, uint free_ptr_offset, lf_pinbox_free_func *free_func, void * free_func_arg); @@ -167,16 +172,20 @@ lock_wrap_void(lf_pinbox_put_pins, &pins->pinbox->pinstack.lock); lock_wrap_void(lf_pinbox_free, (LF_PINS *pins, void *addr), - (pins,addr), + (pins, addr), &pins->pinbox->pinstack.lock); /* memory allocator, lf_alloc-pin.c */ +struct st_lf_alloc_node { + struct st_lf_alloc_node *next; +}; + typedef struct st_lf_allocator { LF_PINBOX pinbox; - void * volatile top; + struct st_lf_alloc_node * volatile top; uint element_size; uint32 volatile mallocs; } LF_ALLOCATOR; @@ -184,13 +193,17 @@ typedef struct st_lf_allocator { void lf_alloc_init(LF_ALLOCATOR *allocator, uint size, uint free_ptr_offset); void lf_alloc_destroy(LF_ALLOCATOR *allocator); uint lf_alloc_in_pool(LF_ALLOCATOR *allocator); +/* + shortcut macros to access underlying pinbox functions from an LF_ALLOCATOR + see _lf_pinbox_get_pins() and _lf_pinbox_put_pins() +*/ #define _lf_alloc_free(PINS, PTR) _lf_pinbox_free((PINS), (PTR)) #define lf_alloc_free(PINS, PTR) lf_pinbox_free((PINS), (PTR)) #define _lf_alloc_get_pins(ALLOC) _lf_pinbox_get_pins(&(ALLOC)->pinbox) #define lf_alloc_get_pins(ALLOC) lf_pinbox_get_pins(&(ALLOC)->pinbox) #define _lf_alloc_put_pins(PINS) _lf_pinbox_put_pins(PINS) #define lf_alloc_put_pins(PINS) lf_pinbox_put_pins(PINS) -#define lf_alloc_real_free(ALLOC,ADDR) my_free((gptr)(ADDR), MYF(0)) +#define lf_alloc_real_free(ALLOC, ADDR) my_free((gptr)(ADDR), MYF(0)) lock_wrap(lf_alloc_new, void *, (LF_PINS *pins), @@ -222,6 +235,10 @@ void lf_hash_destroy(LF_HASH *hash); int lf_hash_insert(LF_HASH *hash, LF_PINS *pins, const void *data); void *lf_hash_search(LF_HASH *hash, LF_PINS *pins, const void *key, uint keylen); int lf_hash_delete(LF_HASH *hash, LF_PINS *pins, const void *key, uint keylen); +/* + shortcut macros to access underlying pinbox functions from an LF_HASH + see _lf_pinbox_get_pins() and _lf_pinbox_put_pins() +*/ #define _lf_hash_get_pins(HASH) _lf_alloc_get_pins(&(HASH)->alloc) #define lf_hash_get_pins(HASH) lf_alloc_get_pins(&(HASH)->alloc) #define _lf_hash_put_pins(PINS) _lf_pinbox_put_pins(PINS) diff --git a/include/my_atomic.h b/include/my_atomic.h index 8efad2802c0..78121c59c40 100644 --- a/include/my_atomic.h +++ b/include/my_atomic.h @@ -173,14 +173,6 @@ make_atomic_fas(ptr) #define LF_BACKOFF (1) #endif -#if SIZEOF_CHARP == SIZEOF_INT -typedef int intptr; -#elif SIZEOF_CHARP == SIZEOF_LONG -typedef long intptr; -#else -#error -#endif - #define MY_ATOMIC_OK 0 #define MY_ATOMIC_NOT_1CPU 1 extern int my_atomic_initialize(); diff --git a/include/my_global.h b/include/my_global.h index 7f7b502e1a7..6831a76c945 100644 --- a/include/my_global.h +++ b/include/my_global.h @@ -432,7 +432,8 @@ C_MODE_END #define compile_time_assert(X) \ do \ { \ - char compile_time_assert[(X) ? 1 : -1]; \ + char compile_time_assert[(X) ? 1 : -1] \ + __attribute__ ((unused)); \ } while(0) /* Go around some bugs in different OS and compilers */ @@ -964,6 +965,14 @@ typedef unsigned __int64 my_ulonglong; typedef unsigned long long my_ulonglong; #endif +#if SIZEOF_CHARP == SIZEOF_INT +typedef int intptr; +#elif SIZEOF_CHARP == SIZEOF_LONG +typedef long intptr; +#else +#error +#endif + #ifdef USE_RAID /* The following is done with a if to not get problems with pre-processors diff --git a/mysys/lf_alloc-pin.c b/mysys/lf_alloc-pin.c index b96fe42311b..43055766c3e 100644 --- a/mysys/lf_alloc-pin.c +++ b/mysys/lf_alloc-pin.c @@ -91,7 +91,7 @@ static void _lf_pinbox_real_free(LF_PINS *pins); See the latter for details. */ void lf_pinbox_init(LF_PINBOX *pinbox, uint free_ptr_offset, - lf_pinbox_free_func *free_func,void *free_func_arg) + lf_pinbox_free_func *free_func, void *free_func_arg) { DBUG_ASSERT(sizeof(LF_PINS) == 128); DBUG_ASSERT(free_ptr_offset % sizeof(void *) == 0); @@ -306,7 +306,7 @@ static void _lf_pinbox_real_free(LF_PINS *pins) { if (addr) /* use binary search */ { - void **a,**b,**c; + void **a, **b, **c; for (a= addr, b= addr+npins-1, c= a+(b-a)/2; b-a>1; c= a+(b-a)/2) if (cur == *c) a= b= c; @@ -337,13 +337,13 @@ found: callback for _lf_pinbox_real_free to free an unpinned object - add it back to the allocator stack */ -static void alloc_free(void *node, LF_ALLOCATOR *allocator) +static void alloc_free(struct st_lf_alloc_node *node, LF_ALLOCATOR *allocator) { - void *tmp; + struct st_lf_alloc_node *tmp; tmp= allocator->top; do { - (*(void **)node)= tmp; + node->next= tmp; } while (!my_atomic_casptr((void **)&allocator->top, (void **)&tmp, node) && LF_BACKOFF); } @@ -379,12 +379,12 @@ void lf_alloc_init(LF_ALLOCATOR *allocator, uint size, uint free_ptr_offset) */ void lf_alloc_destroy(LF_ALLOCATOR *allocator) { - void *el= allocator->top; - while (el) + struct st_lf_alloc_node *node= allocator->top; + while (node) { - void *tmp= *(void **)el; - my_free(el, MYF(0)); - el= tmp; + struct st_lf_alloc_node *tmp= node->next; + my_free((void *)node, MYF(0)); + node= tmp; } lf_pinbox_destroy(&allocator->pinbox); allocator->top= 0; @@ -400,7 +400,7 @@ void lf_alloc_destroy(LF_ALLOCATOR *allocator) void *_lf_alloc_new(LF_PINS *pins) { LF_ALLOCATOR *allocator= (LF_ALLOCATOR *)(pins->pinbox->free_func_arg); - void *node; + struct st_lf_alloc_node *node; for (;;) { do @@ -410,7 +410,8 @@ void *_lf_alloc_new(LF_PINS *pins) } while (node != allocator->top && LF_BACKOFF); if (!node) { - if (!(node= my_malloc(allocator->element_size, MYF(MY_WME|MY_ZEROFILL)))) + if (!(node= (void *)my_malloc(allocator->element_size, + MYF(MY_WME|MY_ZEROFILL)))) break; #ifdef MY_LF_EXTRA_DEBUG my_atomic_add32(&allocator->mallocs, 1); @@ -434,8 +435,8 @@ void *_lf_alloc_new(LF_PINS *pins) uint lf_alloc_in_pool(LF_ALLOCATOR *allocator) { uint i; - void *node; - for (node= allocator->top, i= 0; node; node= *(void **)node, i++) + struct st_lf_alloc_node *node; + for (node= allocator->top, i= 0; node; node= node->next, i++) /* no op */; return i; } diff --git a/mysys/lf_dynarray.c b/mysys/lf_dynarray.c index d63af91813e..c6dd654bf03 100644 --- a/mysys/lf_dynarray.c +++ b/mysys/lf_dynarray.c @@ -19,9 +19,9 @@ (so no pointer into the array may ever become invalid). Memory is allocated in non-contiguous chunks. - This data structure is not space efficient for sparce arrays. + This data structure is not space efficient for sparse arrays. - The number of elements is limited to 2^16 + The number of elements is limited to 4311810304 Every element is aligned to sizeof(element) boundary (to avoid false sharing if element is big enough). @@ -49,7 +49,8 @@ void lf_dynarray_init(LF_DYNARRAY *array, uint element_size) static void recursive_free(void **alloc, int level) { - if (!alloc) return; + if (!alloc) + return; if (level) { @@ -68,10 +69,9 @@ void lf_dynarray_destroy(LF_DYNARRAY *array) for (i= 0; i < LF_DYNARRAY_LEVELS; i++) recursive_free(array->level[i], i); my_atomic_rwlock_destroy(&array->lock); - bzero(array, sizeof(*array)); } -static const long dynarray_idxes_in_prev_level[LF_DYNARRAY_LEVELS]= +static const ulong dynarray_idxes_in_prev_levels[LF_DYNARRAY_LEVELS]= { 0, /* +1 here to to avoid -1's below */ LF_DYNARRAY_LEVEL_LENGTH, @@ -82,6 +82,15 @@ static const long dynarray_idxes_in_prev_level[LF_DYNARRAY_LEVELS]= LF_DYNARRAY_LEVEL_LENGTH + LF_DYNARRAY_LEVEL_LENGTH }; +static const ulong dynarray_idxes_in_prev_level[LF_DYNARRAY_LEVELS]= +{ + 0, /* +1 here to to avoid -1's below */ + LF_DYNARRAY_LEVEL_LENGTH, + LF_DYNARRAY_LEVEL_LENGTH * LF_DYNARRAY_LEVEL_LENGTH, + LF_DYNARRAY_LEVEL_LENGTH * LF_DYNARRAY_LEVEL_LENGTH * + LF_DYNARRAY_LEVEL_LENGTH, +}; + /* Returns a valid lvalue pointer to the element number 'idx'. Allocates memory if necessary. @@ -91,16 +100,17 @@ void *_lf_dynarray_lvalue(LF_DYNARRAY *array, uint idx) void * ptr, * volatile * ptr_ptr= 0; int i; - for (i= 3; idx < dynarray_idxes_in_prev_level[i]; i--) /* no-op */; + for (i= LF_DYNARRAY_LEVELS-1; idx < dynarray_idxes_in_prev_levels[i]; i--) + /* no-op */; ptr_ptr= &array->level[i]; - idx-= dynarray_idxes_in_prev_level[i]; + idx-= dynarray_idxes_in_prev_levels[i]; for (; i > 0; i--) { if (!(ptr= *ptr_ptr)) { void *alloc= my_malloc(LF_DYNARRAY_LEVEL_LENGTH * sizeof(void *), - MYF(MY_WME|MY_ZEROFILL)); - if (!alloc) + MYF(MY_WME|MY_ZEROFILL)); + if (unlikely(!alloc)) return(NULL); if (my_atomic_casptr(ptr_ptr, &ptr, alloc)) ptr= alloc; @@ -116,7 +126,7 @@ void *_lf_dynarray_lvalue(LF_DYNARRAY *array, uint idx) alloc= my_malloc(LF_DYNARRAY_LEVEL_LENGTH * array->size_of_element + max(array->size_of_element, sizeof(void *)), MYF(MY_WME|MY_ZEROFILL)); - if (!alloc) + if (unlikely(!alloc)) return(NULL); /* reserve the space for free() address */ data= alloc + sizeof(void *); @@ -143,9 +153,10 @@ void *_lf_dynarray_value(LF_DYNARRAY *array, uint idx) void * ptr, * volatile * ptr_ptr= 0; int i; - for (i= 3; idx < dynarray_idxes_in_prev_level[i]; i--) /* no-op */; + for (i= LF_DYNARRAY_LEVELS-1; idx < dynarray_idxes_in_prev_levels[i]; i--) + /* no-op */; ptr_ptr= &array->level[i]; - idx-= dynarray_idxes_in_prev_level[i]; + idx-= dynarray_idxes_in_prev_levels[i]; for (; i > 0; i--) { if (!(ptr= *ptr_ptr)) diff --git a/mysys/lf_hash.c b/mysys/lf_hash.c index 66ad672f345..ff0eb8326d5 100644 --- a/mysys/lf_hash.c +++ b/mysys/lf_hash.c @@ -23,6 +23,7 @@ (but how to do it in lf_hash_delete ?) */ #include +#include #include #include #include @@ -33,7 +34,7 @@ LF_REQUIRE_PINS(3); typedef struct { intptr volatile link; /* a pointer to the next element in a listand a flag */ uint32 hashnr; /* reversed hash number, for sorting */ - const uchar *key; + const byte *key; uint keylen; } LF_SLIST; @@ -67,31 +68,31 @@ typedef struct { pins[0..2] are used, they are NOT removed on return */ static int lfind(LF_SLIST * volatile *head, CHARSET_INFO *cs, uint32 hashnr, - const uchar *key, uint keylen, CURSOR *cursor, LF_PINS *pins) + const byte *key, uint keylen, CURSOR *cursor, LF_PINS *pins) { uint32 cur_hashnr; - const uchar *cur_key; + const byte *cur_key; uint cur_keylen; intptr link; retry: - cursor->prev=(intptr *)head; + cursor->prev= (intptr *)head; do { - cursor->curr=PTR(*cursor->prev); - _lf_pin(pins,1,cursor->curr); + cursor->curr= PTR(*cursor->prev); + _lf_pin(pins, 1, cursor->curr); } while(*cursor->prev != (intptr)cursor->curr && LF_BACKOFF); for (;;) { if (!cursor->curr) return 0; do { // XXX or goto retry ? - link=cursor->curr->link; - cursor->next=PTR(link); + link= cursor->curr->link; + cursor->next= PTR(link); _lf_pin(pins, 0, cursor->next); } while(link != cursor->curr->link && LF_BACKOFF); - cur_hashnr=cursor->curr->hashnr; - cur_key=cursor->curr->key; - cur_keylen=cursor->curr->keylen; + cur_hashnr= cursor->curr->hashnr; + cur_key= cursor->curr->key; + cur_keylen= cursor->curr->keylen; if (*cursor->prev != (intptr)cursor->curr) { LF_BACKOFF; @@ -101,12 +102,12 @@ retry: { if (cur_hashnr >= hashnr) { - int r=1; + int r= 1; if (cur_hashnr > hashnr || - (r=my_strnncoll(cs, cur_key, cur_keylen, key, keylen)) >= 0) + (r= my_strnncoll(cs, cur_key, cur_keylen, key, keylen)) >= 0) return !r; } - cursor->prev=&(cursor->curr->link); + cursor->prev= &(cursor->curr->link); _lf_pin(pins, 2, cursor->curr); } else @@ -120,7 +121,7 @@ retry: goto retry; } } - cursor->curr=cursor->next; + cursor->curr= cursor->next; _lf_pin(pins, 1, cursor->curr); } } @@ -141,21 +142,21 @@ static LF_SLIST *linsert(LF_SLIST * volatile *head, CHARSET_INFO *cs, LF_SLIST *node, LF_PINS *pins, uint flags) { CURSOR cursor; - int res=-1; + int res= -1; do { if (lfind(head, cs, node->hashnr, node->key, node->keylen, &cursor, pins) && (flags & LF_HASH_UNIQUE)) - res=0; /* duplicate found */ + res= 0; /* duplicate found */ else { - node->link=(intptr)cursor.curr; + node->link= (intptr)cursor.curr; assert(node->link != (intptr)node); assert(cursor.prev != &node->link); if (my_atomic_casptr((void **)cursor.prev, (void **)&cursor.curr, node)) - res=1; /* inserted ok */ + res= 1; /* inserted ok */ } } while (res == -1); _lf_unpin(pins, 0); @@ -177,10 +178,10 @@ static LF_SLIST *linsert(LF_SLIST * volatile *head, CHARSET_INFO *cs, it uses pins[0..2], on return all pins are removed. */ static int ldelete(LF_SLIST * volatile *head, CHARSET_INFO *cs, uint32 hashnr, - const uchar *key, uint keylen, LF_PINS *pins) + const byte *key, uint keylen, LF_PINS *pins) { CURSOR cursor; - int res=-1; + int res= -1; do { @@ -218,30 +219,30 @@ static int ldelete(LF_SLIST * volatile *head, CHARSET_INFO *cs, uint32 hashnr, all other pins are removed. */ static LF_SLIST *lsearch(LF_SLIST * volatile *head, CHARSET_INFO *cs, - uint32 hashnr, const uchar *key, uint keylen, + uint32 hashnr, const byte *key, uint keylen, LF_PINS *pins) { CURSOR cursor; - int res=lfind(head, cs, hashnr, key, keylen, &cursor, pins); + int res= lfind(head, cs, hashnr, key, keylen, &cursor, pins); if (res) _lf_pin(pins, 2, cursor.curr); _lf_unpin(pins, 0); _lf_unpin(pins, 1); return res ? cursor.curr : 0; } -static inline const uchar* hash_key(const LF_HASH *hash, - const uchar *record, uint *length) +static inline const byte* hash_key(const LF_HASH *hash, + const byte *record, uint *length) { if (hash->get_key) - return (*hash->get_key)(record,length,0); - *length=hash->key_length; + return (*hash->get_key)(record, length, 0); + *length= hash->key_length; return record + hash->key_offset; } -static inline uint calc_hash(LF_HASH *hash, const uchar *key, uint keylen) +static inline uint calc_hash(LF_HASH *hash, const byte *key, uint keylen) { - ulong nr1=1, nr2=4; - hash->charset->coll->hash_sort(hash->charset,key,keylen,&nr1,&nr2); + ulong nr1= 1, nr2= 4; + hash->charset->coll->hash_sort(hash->charset, key, keylen, &nr1, &nr2); return nr1 & INT_MAX32; } @@ -258,28 +259,28 @@ void lf_hash_init(LF_HASH *hash, uint element_size, uint flags, lf_alloc_init(&hash->alloc, sizeof(LF_SLIST)+element_size, offsetof(LF_SLIST, key)); lf_dynarray_init(&hash->array, sizeof(LF_SLIST **)); - hash->size=1; - hash->count=0; - hash->element_size=element_size; - hash->flags=flags; - hash->charset=charset ? charset : &my_charset_bin; - hash->key_offset=key_offset; - hash->key_length=key_length; - hash->get_key=get_key; + hash->size= 1; + hash->count= 0; + hash->element_size= element_size; + hash->flags= flags; + hash->charset= charset ? charset : &my_charset_bin; + hash->key_offset= key_offset; + hash->key_length= key_length; + hash->get_key= get_key; DBUG_ASSERT(get_key ? !key_offset && !key_length : key_length); } void lf_hash_destroy(LF_HASH *hash) { - LF_SLIST *el=*(LF_SLIST **)_lf_dynarray_lvalue(&hash->array, 0); + LF_SLIST *el= *(LF_SLIST **)_lf_dynarray_lvalue(&hash->array, 0); while (el) { - intptr next=el->link; + intptr next= el->link; if (el->hashnr & 1) lf_alloc_real_free(&hash->alloc, el); else my_free((void *)el, MYF(0)); - el=(LF_SLIST *)next; + el= (LF_SLIST *)next; } lf_alloc_destroy(&hash->alloc); lf_dynarray_destroy(&hash->array); @@ -299,19 +300,19 @@ void lf_hash_destroy(LF_HASH *hash) */ int lf_hash_insert(LF_HASH *hash, LF_PINS *pins, const void *data) { - uint csize, bucket, hashnr; + int csize, bucket, hashnr; LF_SLIST *node, * volatile *el; lf_rwlock_by_pins(pins); - node=(LF_SLIST *)_lf_alloc_new(pins); + node= (LF_SLIST *)_lf_alloc_new(pins); memcpy(node+1, data, hash->element_size); - node->key= hash_key(hash, (uchar *)(node+1), &node->keylen); + node->key= hash_key(hash, (byte *)(node+1), &node->keylen); hashnr= calc_hash(hash, node->key, node->keylen); bucket= hashnr % hash->size; - el=_lf_dynarray_lvalue(&hash->array, bucket); + el= _lf_dynarray_lvalue(&hash->array, bucket); if (*el == NULL) initialize_bucket(hash, el, bucket, pins); - node->hashnr=my_reverse_bits(hashnr) | 1; + node->hashnr= my_reverse_bits(hashnr) | 1; if (linsert(el, hash->charset, node, pins, hash->flags)) { _lf_alloc_free(pins, node); @@ -335,15 +336,15 @@ int lf_hash_insert(LF_HASH *hash, LF_PINS *pins, const void *data) int lf_hash_delete(LF_HASH *hash, LF_PINS *pins, const void *key, uint keylen) { LF_SLIST * volatile *el; - uint bucket, hashnr=calc_hash(hash, (uchar *)key, keylen); + uint bucket, hashnr= calc_hash(hash, (byte *)key, keylen); bucket= hashnr % hash->size; lf_rwlock_by_pins(pins); - el=_lf_dynarray_lvalue(&hash->array, bucket); + el= _lf_dynarray_lvalue(&hash->array, bucket); if (*el == NULL) initialize_bucket(hash, el, bucket, pins); if (ldelete(el, hash->charset, my_reverse_bits(hashnr) | 1, - (uchar *)key, keylen, pins)) + (byte *)key, keylen, pins)) { lf_rwunlock_by_pins(pins); return 1; @@ -360,33 +361,33 @@ int lf_hash_delete(LF_HASH *hash, LF_PINS *pins, const void *key, uint keylen) void *lf_hash_search(LF_HASH *hash, LF_PINS *pins, const void *key, uint keylen) { LF_SLIST * volatile *el, *found; - uint bucket, hashnr=calc_hash(hash, (uchar *)key, keylen); + uint bucket, hashnr= calc_hash(hash, (byte *)key, keylen); bucket= hashnr % hash->size; lf_rwlock_by_pins(pins); - el=_lf_dynarray_lvalue(&hash->array, bucket); + el= _lf_dynarray_lvalue(&hash->array, bucket); if (*el == NULL) initialize_bucket(hash, el, bucket, pins); found= lsearch(el, hash->charset, my_reverse_bits(hashnr) | 1, - (uchar *)key, keylen, pins); + (byte *)key, keylen, pins); lf_rwunlock_by_pins(pins); return found ? found+1 : 0; } -static char *dummy_key=""; +static char *dummy_key= ""; static void initialize_bucket(LF_HASH *hash, LF_SLIST * volatile *node, uint bucket, LF_PINS *pins) { uint parent= my_clear_highest_bit(bucket); - LF_SLIST *dummy=(LF_SLIST *)my_malloc(sizeof(LF_SLIST), MYF(MY_WME)); - LF_SLIST **tmp=0, *cur; - LF_SLIST * volatile *el=_lf_dynarray_lvalue(&hash->array, parent); + LF_SLIST *dummy= (LF_SLIST *)my_malloc(sizeof(LF_SLIST), MYF(MY_WME)); + LF_SLIST **tmp= 0, *cur; + LF_SLIST * volatile *el= _lf_dynarray_lvalue(&hash->array, parent); if (*el == NULL && bucket) initialize_bucket(hash, el, parent, pins); - dummy->hashnr=my_reverse_bits(bucket); - dummy->key=dummy_key; - dummy->keylen=0; + dummy->hashnr= my_reverse_bits(bucket); + dummy->key= dummy_key; + dummy->keylen= 0; if ((cur= linsert(el, hash->charset, dummy, pins, 0))) { my_free((void *)dummy, MYF(0)); diff --git a/mysys/my_getsystime.c b/mysys/my_getsystime.c index 91c977f0b5a..d1ed4f2ec92 100644 --- a/mysys/my_getsystime.c +++ b/mysys/my_getsystime.c @@ -35,10 +35,6 @@ ulonglong my_getsystime() LARGE_INTEGER t_cnt; if (!offset) { - /* strictly speaking there should be a mutex to protect - initialization section. But my_getsystime() is called from - UUID() code, and UUID() calls are serialized with a mutex anyway - */ LARGE_INTEGER li; FILETIME ft; GetSystemTimeAsFileTime(&ft); diff --git a/storage/maria/lockman.h b/storage/maria/lockman.h index 6577a5e80fc..fd96f8930d5 100644 --- a/storage/maria/lockman.h +++ b/storage/maria/lockman.h @@ -32,7 +32,7 @@ SLX - Shared + Loose eXclusive LSIX - Loose Shared + Intention eXclusive */ -enum lock_type { N, S, X, IS, IX, SIX, LS, LX, SLX, LSIX }; +enum lock_type { N, S, X, IS, IX, SIX, LS, LX, SLX, LSIX, LOCK_TYPE_LAST }; struct lockman_lock; @@ -55,9 +55,10 @@ typedef struct { uint lock_timeout; loid_to_lo_func *loid_to_lo; } LOCKMAN; - +#define DIDNT_GET_THE_LOCK 0 enum lockman_getlock_result { - DIDNT_GET_THE_LOCK=0, GOT_THE_LOCK, + NO_MEMORY_FOR_LOCK=1, DEADLOCK, LOCK_TIMEOUT, + GOT_THE_LOCK, GOT_THE_LOCK_NEED_TO_LOCK_A_SUBRESOURCE, GOT_THE_LOCK_NEED_TO_INSTANT_LOCK_A_SUBRESOURCE }; diff --git a/storage/maria/tablockman.c b/storage/maria/tablockman.c index 608f852cb7b..b4cf77401bb 100644 --- a/storage/maria/tablockman.c +++ b/storage/maria/tablockman.c @@ -1,5 +1,5 @@ -// TODO - allocate everything from dynarrays !!! (benchmark) -// automatically place S instead of LS if possible +#warning TODO - allocate everything from dynarrays !!! (benchmark) +#warning automatically place S instead of LS if possible /* Copyright (C) 2006 MySQL AB This program is free software; you can redistribute it and/or modify @@ -16,10 +16,8 @@ along with this program; if not, write to the Free Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ -#include -#include -#include -#include +#include +#include #include "tablockman.h" /* @@ -53,45 +51,54 @@ resource, check if a conflicting lock exists, if yes - find who owns it. Solution: every resource has a structure with - 1. Hash of "active" (see below for the description of "active") granted - locks with loid as a key. Thus, checking if a given transaction has a - lock on this resource is O(1) operation. + 1. Hash of latest (see the lock upgrade section below) granted locks with + loid as a key. Thus, checking if a given transaction has a lock on + this resource is O(1) operation. 2. Doubly-linked lists of all granted locks - one list for every lock type. Thus, checking if a conflicting lock exists is a check whether an appropriate list head pointer is not null, also O(1). 3. Every lock has a loid of the owner, thus checking who owns a conflicting lock is also O(1). - 4. Deque of waiting locks. It's a deque not a fifo, because for lock - upgrades requests are added to the queue head, not tail. There's never - a need to scan the queue. - - Result: adding or removing a lock is always a O(1) operation, it does not - depend on the number of locks on the resource, or number of transactions, - or number of resources. It _does_ depend on the number of different lock - levels - O(number_of_lock_levels) - but it's a constant. + 4. Deque of waiting locks. It's a deque (double-ended queue) not a fifo, + because for lock upgrades requests are added to the queue head, not + tail. This is a single place where there it gets O(N) on number + of locks - when a transaction wakes up from waiting on a condition, + it may need to scan the queue backward to the beginning to find + a conflicting lock. It is guaranteed though that "all transactions + before it" received the same - or earlier - signal. In other words a + transaction needs to scan all transactions before it that received the + signal but didn't have a chance to resume the execution yet, so + practically OS scheduler won't let the scan to be O(N). Waiting: if there is a conflicting lock or if wait queue is not empty, a requested lock cannot be granted at once. It is added to the end of the - wait queue. If there is a conflicting lock - the "blocker" transaction is - the owner of this lock. If there's no conflict but a queue was not empty, - than the "blocker" is the transaction that the owner of the lock at the - end of the queue is waiting for (in other words, our lock is added to the - end of the wait queue, and our blocker is the same as of the lock right - before us). + wait queue. If a queue was empty and there is a conflicting lock - the + "blocker" transaction is the owner of this lock. If a queue is not empty, + an owner of the previous lock in the queue is the "blocker". But if the + previous lock is compatible with the request, then the "blocker" is the + transaction that the owner of the lock at the end of the queue is waiting + for (in other words, our lock is added to the end of the wait queue, and + our blocker is the same as of the lock right before us). Lock upgrades: when a thread that has a lock on a given resource, requests a new lock on the same resource and the old lock is not enough to satisfy new lock requirements (which is defined by lock_combining_matrix[old_lock][new_lock] != old_lock), a new lock - (defineded by lock_combining_matrix as above) is placed. Depending on - other granted locks it is immediately active or it has to wait. Here the + (defined by lock_combining_matrix as above) is placed. Depending on + other granted locks it is immediately granted or it has to wait. Here the lock is added to the start of the waiting queue, not to the end. Old lock, is removed from the hash, but not from the doubly-linked lists. (indeed, a transaction checks "do I have a lock on this resource ?" by looking in a hash, and it should find a latest lock, so old locks must be - removed; but a transaction checks "are the conflicting locks ?" by + removed; but a transaction checks "are there conflicting locks ?" by checking doubly-linked lists, it doesn't matter if it will find an old lock - if it would be removed, a new lock would be also a conflict). + So, a hash contains only "latest" locks - there can be only one latest + lock per resource per transaction. But doubly-linked lists contain all + locks, even "obsolete" ones, because it doesnt't hurt. Note that old + locks can not be freed early, in particular they stay in the + 'active_locks' list of a lock owner, because they may be "re-enabled" + on a savepoint rollback. To better support table-row relations where one needs to lock the table with an intention lock before locking the row, extended diagnostics is @@ -107,6 +114,18 @@ Instant duration locks are not supported. Though they're trivial to add, they are normally only used on rows, not on tables. So, presumably, they are not needed here. + + Mutexes: there're table mutexes (LOCKED_TABLE::mutex), lock owner mutexes + (TABLE_LOCK_OWNER::mutex), and a pool mutex (TABLOCKMAN::pool_mutex). + table mutex protects operations on the table lock structures, and lock + owner pointers waiting_for and waiting_for_loid. + lock owner mutex is only used to wait on lock owner condition + (TABLE_LOCK_OWNER::cond), there's no need to protect owner's lock + structures, and only lock owner itself may access them. + The pool mutex protects a pool of unused locks. Note the locking order: + first the table mutex, then the owner mutex or a pool mutex. + Table mutex lock cannot be attempted when owner or pool mutex are locked. + No mutex lock can be attempted if owner or pool mutex are locked. */ /* @@ -122,9 +141,9 @@ 0 - incompatible -1 - "impossible", so that we can assert the impossibility. */ -static int lock_compatibility_matrix[10][10]= +static const int lock_compatibility_matrix[10][10]= { /* N S X IS IX SIX LS LX SLX LSIX */ - { -1, 1, 1, 1, 1, 1, 1, 1, 1, 1 }, /* N */ + { -1, -1, -1, -1, -1, -1, -1, -1, -1, -1 }, /* N */ { -1, 1, 0, 1, 0, 0, 1, 0, 0, 0 }, /* S */ { -1, 0, 0, 0, 0, 0, 0, 0, 0, 0 }, /* X */ { -1, 1, 0, 1, 1, 1, 1, 1, 1, 1 }, /* IS */ @@ -144,18 +163,18 @@ static int lock_compatibility_matrix[10][10]= One should never get N from it, we assert the impossibility */ -static enum lock_type lock_combining_matrix[10][10]= +static const enum lock_type lock_combining_matrix[10][10]= {/* N S X IS IX SIX LS LX SLX LSIX */ - { N, S, X, IS, IX, SIX, S, SLX, SLX, SIX}, /* N */ - { S, S, X, S, SIX, SIX, S, SLX, SLX, SIX}, /* S */ - { X, X, X, X, X, X, X, X, X, X}, /* X */ - { IS, S, X, IS, IX, SIX, LS, LX, SLX, LSIX}, /* IS */ - { IX, SIX, X, IX, IX, SIX, LSIX, LX, SLX, LSIX}, /* IX */ - { SIX, SIX, X, SIX, SIX, SIX, SIX, SLX, SLX, SIX}, /* SIX */ - { LS, S, X, LS, LSIX, SIX, LS, LX, SLX, LSIX}, /* LS */ - { LX, SLX, X, LX, LX, SLX, LX, LX, SLX, LX}, /* LX */ - { SLX, SLX, X, SLX, SLX, SLX, SLX, SLX, SLX, SLX}, /* SLX */ - { LSIX, SIX, X, LSIX, LSIX, SIX, LSIX, LX, SLX, LSIX} /* LSIX */ + { N, N, N, N, N, N, N, N, N, N}, /* N */ + { N, S, X, S, SIX, SIX, S, SLX, SLX, SIX}, /* S */ + { N, X, X, X, X, X, X, X, X, X}, /* X */ + { N, S, X, IS, IX, SIX, LS, LX, SLX, LSIX}, /* IS */ + { N, SIX, X, IX, IX, SIX, LSIX, LX, SLX, LSIX}, /* IX */ + { N, SIX, X, SIX, SIX, SIX, SIX, SLX, SLX, SIX}, /* SIX */ + { N, S, X, LS, LSIX, SIX, LS, LX, SLX, LSIX}, /* LS */ + { N, SLX, X, LX, LX, SLX, LX, LX, SLX, LX}, /* LX */ + { N, SLX, X, SLX, SLX, SLX, SLX, SLX, SLX, SLX}, /* SLX */ + { N, SIX, X, LSIX, LSIX, SIX, LSIX, LX, SLX, LSIX} /* LSIX */ }; /* @@ -176,7 +195,7 @@ static enum lock_type lock_combining_matrix[10][10]= #define L GOT_THE_LOCK_NEED_TO_INSTANT_LOCK_A_SUBRESOURCE #define A GOT_THE_LOCK #define x GOT_THE_LOCK -static enum lockman_getlock_result getlock_result[10][10]= +static const enum lockman_getlock_result getlock_result[10][10]= {/* N S X IS IX SIX LS LX SLX LSIX */ { 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}, /* N */ { 0, x, 0, A, 0, 0, x, 0, 0, 0}, /* S */ @@ -200,37 +219,47 @@ static enum lockman_getlock_result getlock_result[10][10]= */ struct st_table_lock { +#warning do we need upgraded_from ? struct st_table_lock *next_in_lo, *upgraded_from, *next, *prev; struct st_locked_table *table; uint16 loid; - char lock_type; + uchar lock_type; }; #define hash_insert my_hash_insert /* for consistency :) */ -#define remove_from_wait_queue(LOCK, TABLE) \ - do \ - { \ - if ((LOCK)->prev) \ - { \ - DBUG_ASSERT((TABLE)->wait_queue_out != (LOCK)); \ - (LOCK)->prev->next= (LOCK)->next; \ - } \ - else \ - { \ - DBUG_ASSERT((TABLE)->wait_queue_out == (LOCK)); \ - (TABLE)->wait_queue_out= (LOCK)->next; \ - } \ - if ((LOCK)->next) \ - { \ - DBUG_ASSERT((TABLE)->wait_queue_in != (LOCK)); \ - (LOCK)->next->prev= (LOCK)->prev; \ - } \ - else \ - { \ - DBUG_ASSERT((TABLE)->wait_queue_in == (LOCK)); \ - (TABLE)->wait_queue_in= (LOCK)->prev; \ - } \ - } while (0) + +static inline +TABLE_LOCK *find_loid(LOCKED_TABLE *table, uint16 loid) +{ + return (TABLE_LOCK *)hash_search(& table->latest_locks, + (byte *)& loid, sizeof(loid)); +} + +static inline +void remove_from_wait_queue(TABLE_LOCK *lock, LOCKED_TABLE *table) +{ + DBUG_ASSERT(table == lock->table); + if (lock->prev) + { + DBUG_ASSERT(table->wait_queue_out != lock); + lock->prev->next= lock->next; + } + else + { + DBUG_ASSERT(table->wait_queue_out == lock); + table->wait_queue_out= lock->next; + } + if (lock->next) + { + DBUG_ASSERT(table->wait_queue_in != lock); + lock->next->prev= lock->prev; + } + else + { + DBUG_ASSERT(table->wait_queue_in == lock); + table->wait_queue_in= lock->prev; + } +} /* DESCRIPTION @@ -243,24 +272,31 @@ enum lockman_getlock_result tablockman_getlock(TABLOCKMAN *lm, TABLE_LOCK_OWNER *lo, LOCKED_TABLE *table, enum lock_type lock) { - TABLE_LOCK *old, *new, *blocker; + TABLE_LOCK *old, *new, *blocker, *blocker2; TABLE_LOCK_OWNER *wait_for; ulonglong deadline; struct timespec timeout; enum lock_type new_lock; + enum lockman_getlock_result res; int i; + DBUG_ASSERT(lo->waiting_lock == 0); + DBUG_ASSERT(lo->waiting_for == 0); + DBUG_ASSERT(lo->waiting_for_loid == 0); + pthread_mutex_lock(& table->mutex); - /* do we alreasy have a lock on this resource ? */ - old= (TABLE_LOCK *)hash_search(& table->active, (byte *)&lo->loid, - sizeof(lo->loid)); + /* do we already have a lock on this resource ? */ + old= find_loid(table, lo->loid); - /* and if yes, is it enough to satisfy the new request */ - if (old && lock_combining_matrix[old->lock_type][lock] == old->lock_type) + /* calculate the level of the upgraded lock, if yes */ + new_lock= old ? lock_combining_matrix[old->lock_type][lock] : lock; + + /* and check if old lock is enough to satisfy the new request */ + if (old && new_lock == old->lock_type) { /* yes */ - pthread_mutex_unlock(& table->mutex); - return getlock_result[old->lock_type][lock]; + res= getlock_result[old->lock_type][lock]; + goto ret; } /* no, placing a new lock. first - take a free lock structure from the pool */ @@ -275,48 +311,81 @@ tablockman_getlock(TABLOCKMAN *lm, TABLE_LOCK_OWNER *lo, { pthread_mutex_unlock(& lm->pool_mutex); new= (TABLE_LOCK *)my_malloc(sizeof(*new), MYF(MY_WME)); - if (!new) + if (unlikely(!new)) { - pthread_mutex_unlock(& table->mutex); - return DIDNT_GET_THE_LOCK; + res= NO_MEMORY_FOR_LOCK; + goto ret; } } - /* calculate the level of the upgraded lock */ - new_lock= old ? lock_combining_matrix[old->lock_type][lock] : lock; - new->loid= lo->loid; new->lock_type= new_lock; new->table= table; /* and try to place it */ - for (new->prev= table->wait_queue_in ; ; ) + for (new->prev= table->wait_queue_in;;) { - /* waiting queue is not empty and we're not upgrading */ - if (!old && new->prev) + wait_for= 0; + if (!old) { - /* need to wait */ - DBUG_ASSERT(table->wait_queue_out); - DBUG_ASSERT(table->wait_queue_in); - blocker= new->prev; - /* wait for a previous lock in the queue or for a lock it's waiting for */ - if (lock_compatibility_matrix[blocker->lock_type][lock]) - wait_for= lm->loid_to_tlo(blocker->loid)->waiting_for; - else - wait_for= lm->loid_to_tlo(blocker->loid); + /* not upgrading - a lock must be added to the _end_ of the wait queue */ + for (blocker= new->prev; blocker && !wait_for; blocker= blocker->prev) + { + TABLE_LOCK_OWNER *tmp= lm->loid_to_tlo(blocker->loid); + + /* find a blocking lock */ + DBUG_ASSERT(table->wait_queue_out); + DBUG_ASSERT(table->wait_queue_in); + if (!lock_compatibility_matrix[blocker->lock_type][lock]) + { + /* found! */ + wait_for= tmp; + } + else + { + /* + hmm, the lock before doesn't block us, let's look one step further. + the condition below means: + + if we never waited on a condition yet + OR + the lock before ours (blocker) waits on a lock (blocker2) that is + present in the hash AND and conflicts with 'blocker' + + the condition after OR may fail if 'blocker2' was removed from + the hash, its signal woke us up, but 'blocker' itself didn't see + the signal yet. + */ + if (!lo->waiting_lock || + ((blocker2= find_loid(table, tmp->waiting_for_loid)) && + !lock_compatibility_matrix[blocker2->lock_type] + [blocker->lock_type])) + { + /* but it's waiting for a real lock. we'll wait for the same lock */ + wait_for= tmp->waiting_for; + } + /* + otherwise - a lock it's waiting for doesn't exist. + We've no choice but to scan the wait queue backwards, looking + for a conflicting lock or a lock waiting for a real lock. + QQ is there a way to avoid this scanning ? + */ + } + } } - else + + if (wait_for == 0) { /* checking for compatibility with existing locks */ for (blocker= 0, i= 0; i < LOCK_TYPES; i++) { if (table->active_locks[i] && !lock_compatibility_matrix[i+1][lock]) { - /* the first lock in the list may be our own - skip it */ - for (blocker= table->active_locks[i]; - blocker && blocker->loid == lo->loid; - blocker= blocker->next) /* no-op */; - if (blocker) + blocker= table->active_locks[i]; + /* if the first lock in the list is our own - skip it */ + if (blocker->loid == lo->loid) + blocker= blocker->next; + if (blocker) /* found a conflicting lock, need to wait */ break; } } @@ -327,6 +396,7 @@ tablockman_getlock(TABLOCKMAN *lm, TABLE_LOCK_OWNER *lo, /* ok, we're here - the wait is inevitable */ lo->waiting_for= wait_for; + lo->waiting_for_loid= wait_for->loid; if (!lo->waiting_lock) /* first iteration of the for() loop */ { /* lock upgrade or new lock request ? */ @@ -338,7 +408,7 @@ tablockman_getlock(TABLOCKMAN *lm, TABLE_LOCK_OWNER *lo, new->next->prev= new; table->wait_queue_out= new; if (!table->wait_queue_in) - table->wait_queue_in=table->wait_queue_out; + table->wait_queue_in= table->wait_queue_out; } else { @@ -348,7 +418,7 @@ tablockman_getlock(TABLOCKMAN *lm, TABLE_LOCK_OWNER *lo, new->prev->next= new; table->wait_queue_in= new; if (!table->wait_queue_out) - table->wait_queue_out=table->wait_queue_in; + table->wait_queue_out= table->wait_queue_in; } lo->waiting_lock= new; @@ -356,22 +426,28 @@ tablockman_getlock(TABLOCKMAN *lm, TABLE_LOCK_OWNER *lo, timeout.tv_sec= deadline/10000000; timeout.tv_nsec= (deadline % 10000000) * 100; } - else - { - if (my_getsystime() > deadline) - { - pthread_mutex_unlock(& table->mutex); - return DIDNT_GET_THE_LOCK; - } - } - /* now really wait */ + /* + prepare to wait. + we must lock blocker's mutex to wait on blocker's cond. + and we must release table's mutex. + note that blocker's mutex is locked _before_ table's mutex is released + */ pthread_mutex_lock(wait_for->mutex); pthread_mutex_unlock(& table->mutex); - pthread_cond_timedwait(wait_for->cond, wait_for->mutex, &timeout); + /* now really wait */ + i= pthread_cond_timedwait(wait_for->cond, wait_for->mutex, & timeout); pthread_mutex_unlock(wait_for->mutex); + + if (i == ETIMEDOUT || i == ETIME) + { + /* we rely on the caller to rollback and release all locks */ + res= LOCK_TIMEOUT; + goto ret2; + } + pthread_mutex_lock(& table->mutex); /* ... and repeat from the beginning */ @@ -384,6 +460,7 @@ tablockman_getlock(TABLOCKMAN *lm, TABLE_LOCK_OWNER *lo, remove_from_wait_queue(new, table); lo->waiting_lock= 0; lo->waiting_for= 0; + lo->waiting_for_loid= 0; } /* add it to the list of all locks of this lock owner */ @@ -396,20 +473,20 @@ tablockman_getlock(TABLOCKMAN *lm, TABLE_LOCK_OWNER *lo, new->next->prev= new; table->active_locks[new_lock-1]= new; - /* remove the old lock from the hash, if upgrading */ + /* update the latest_locks hash */ if (old) - { - new->upgraded_from= old; - hash_delete(& table->active, (byte *)old); - } - else - new->upgraded_from= 0; + hash_delete(& table->latest_locks, (byte *)old); + hash_insert(& table->latest_locks, (byte *)new); - /* and add a new lock to the hash, voila */ - hash_insert(& table->active, (byte *)new); + new->upgraded_from= old; + res= getlock_result[lock][lock]; + +ret: pthread_mutex_unlock(& table->mutex); - return getlock_result[lock][lock]; +ret2: + DBUG_ASSERT(res); + return res; } /* @@ -443,6 +520,17 @@ void tablockman_release_locks(TABLOCKMAN *lm, TABLE_LOCK_OWNER *lo) Signal our blocker to release this next lock (after we removed our lock from the wait queue, of course). */ + /* + An example to clarify the above: + trn1> S-lock the table. Granted. + trn2> IX-lock the table. Added to the wait queue. trn2 waits on trn1 + trn3> IS-lock the table. The queue is not empty, so IS-lock is added + to the queue. It's compatible with the waiting IX-lock, so trn3 + waits for trn2->waiting_for, that is trn1. + if trn1 releases the lock it signals trn1->cond and both waiting + transactions are awaken. But if trn2 times out, trn3 must be notified + too (as IS and S locks are compatible). So trn2 must signal trn1->cond. + */ if (lock->prev && lock_compatibility_matrix[lock->prev->lock_type][lock->lock_type]) { @@ -451,6 +539,7 @@ void tablockman_release_locks(TABLOCKMAN *lm, TABLE_LOCK_OWNER *lo) pthread_mutex_unlock(lo->waiting_for->mutex); } lo->waiting_for= 0; + lo->waiting_for_loid= 0; pthread_mutex_unlock(& lock->table->mutex); lock->next= local_pool; @@ -465,11 +554,12 @@ void tablockman_release_locks(TABLOCKMAN *lm, TABLE_LOCK_OWNER *lo) pthread_mutex_t *mutex= & lock->table->mutex; DBUG_ASSERT(cur->loid == lo->loid); + DBUG_ASSERT(lock != lock->next_in_lo); lock= lock->next_in_lo; /* TODO ? group locks by table to reduce the number of mutex locks */ pthread_mutex_lock(mutex); - hash_delete(& cur->table->active, (byte *)cur); + hash_delete(& cur->table->latest_locks, (byte *)cur); if (cur->prev) cur->prev->next= cur->next; @@ -506,7 +596,8 @@ void tablockman_init(TABLOCKMAN *lm, loid_to_tlo_func *func, uint timeout) lm->pool= 0; lm->loid_to_tlo= func; lm->lock_timeout= timeout; - pthread_mutex_init(&lm->pool_mutex, MY_MUTEX_INIT_FAST); + pthread_mutex_init(& lm->pool_mutex, MY_MUTEX_INIT_FAST); + my_getsystime(); /* ensure that my_getsystime() is initialized */ } void tablockman_destroy(TABLOCKMAN *lm) @@ -517,36 +608,54 @@ void tablockman_destroy(TABLOCKMAN *lm) lm->pool= tmp->next; my_free((void *)tmp, MYF(0)); } - pthread_mutex_destroy(&lm->pool_mutex); + pthread_mutex_destroy(& lm->pool_mutex); } +/* + initialize a LOCKED_TABLE structure + + SYNOPSYS + lt a LOCKED_TABLE to initialize + initial_hash_size initial size for 'latest_locks' hash +*/ void tablockman_init_locked_table(LOCKED_TABLE *lt, int initial_hash_size) { - TABLE_LOCK *unused; bzero(lt, sizeof(*lt)); pthread_mutex_init(& lt->mutex, MY_MUTEX_INIT_FAST); - hash_init(& lt->active, &my_charset_bin, initial_hash_size, - offsetof(TABLE_LOCK, loid), sizeof(unused->loid), 0, 0, 0); + hash_init(& lt->latest_locks, & my_charset_bin, initial_hash_size, + offsetof(TABLE_LOCK, loid), + sizeof(((TABLE_LOCK*)0)->loid), 0, 0, 0); } void tablockman_destroy_locked_table(LOCKED_TABLE *lt) { - hash_free(& lt->active); + int i; + + DBUG_ASSERT(lt->wait_queue_out == 0); + DBUG_ASSERT(lt->wait_queue_in == 0); + DBUG_ASSERT(lt->latest_locks.records == 0); + for (i= 0; iactive_locks[i] == 0); + + hash_free(& lt->latest_locks); pthread_mutex_destroy(& lt->mutex); } #ifdef EXTRA_DEBUG -static char *lock2str[LOCK_TYPES+1]= {"N", "S", "X", "IS", "IX", "SIX", +static const char *lock2str[LOCK_TYPES+1]= {"N", "S", "X", "IS", "IX", "SIX", "LS", "LX", "SLX", "LSIX"}; -void print_tlo(TABLE_LOCK_OWNER *lo) +void tablockman_print_tlo(TABLE_LOCK_OWNER *lo) { TABLE_LOCK *lock; + printf("lo%d>", lo->loid); if ((lock= lo->waiting_lock)) - printf(" (%s.%p)", lock2str[lock->lock_type], lock->table); - for (lock= lo->active_locks; lock && lock != lock->next_in_lo; lock= lock->next_in_lo) - printf(" %s.%p", lock2str[lock->lock_type], lock->table); + printf(" (%s.0x%lx)", lock2str[lock->lock_type], (intptr)lock->table); + for (lock= lo->active_locks; + lock && lock != lock->next_in_lo; + lock= lock->next_in_lo) + printf(" %s.0x%lx", lock2str[lock->lock_type], (intptr)lock->table); if (lock && lock == lock->next_in_lo) printf("!"); printf("\n"); diff --git a/storage/maria/tablockman.h b/storage/maria/tablockman.h index d307b2c844f..4498e7027b4 100644 --- a/storage/maria/tablockman.h +++ b/storage/maria/tablockman.h @@ -33,45 +33,45 @@ LSIX - Loose Shared + Intention eXclusive */ #ifndef _lockman_h -enum lock_type { N, S, X, IS, IX, SIX, LS, LX, SLX, LSIX }; +#warning TODO remove N-locks +enum lock_type { N, S, X, IS, IX, SIX, LS, LX, SLX, LSIX, LOCK_TYPE_LAST }; enum lockman_getlock_result { - DIDNT_GET_THE_LOCK=0, GOT_THE_LOCK, + NO_MEMORY_FOR_LOCK=1, DEADLOCK, LOCK_TIMEOUT, + GOT_THE_LOCK, GOT_THE_LOCK_NEED_TO_LOCK_A_SUBRESOURCE, GOT_THE_LOCK_NEED_TO_INSTANT_LOCK_A_SUBRESOURCE }; - #endif -#define LOCK_TYPES LSIX +#define LOCK_TYPES (LOCK_TYPE_LAST-1) -typedef struct st_table_lock_owner TABLE_LOCK_OWNER; typedef struct st_table_lock TABLE_LOCK; -typedef struct st_locked_table LOCKED_TABLE; + +typedef struct st_table_lock_owner { + TABLE_LOCK *active_locks; /* list of active locks */ + TABLE_LOCK *waiting_lock; /* waiting lock (one lock only) */ + struct st_table_lock_owner *waiting_for; /* transaction we're waiting for */ + pthread_cond_t *cond; /* transactions waiting for us, wait on 'cond' */ + pthread_mutex_t *mutex; /* mutex is required to use 'cond' */ + uint16 loid, waiting_for_loid; /* Lock Owner IDentifier */ +} TABLE_LOCK_OWNER; + +typedef struct st_locked_table { + pthread_mutex_t mutex; /* mutex for everything below */ + HASH latest_locks; /* latest locks in a hash */ + TABLE_LOCK *active_locks[LOCK_TYPES]; /* dl-list of locks per type */ + TABLE_LOCK *wait_queue_in, *wait_queue_out; /* wait deque (double-end queue)*/ +} LOCKED_TABLE; + typedef TABLE_LOCK_OWNER *loid_to_tlo_func(uint16); typedef struct { pthread_mutex_t pool_mutex; - TABLE_LOCK *pool; /* lifo pool of free locks */ - uint lock_timeout; - loid_to_tlo_func *loid_to_tlo; /* for mapping loid to TABLE_LOCK_OWNER */ + TABLE_LOCK *pool; /* lifo pool of free locks */ + uint lock_timeout; /* lock timeout in milliseconds */ + loid_to_tlo_func *loid_to_tlo; /* for mapping loid to TABLE_LOCK_OWNER */ } TABLOCKMAN; -struct st_table_lock_owner { - TABLE_LOCK *active_locks; /* list of active locks */ - TABLE_LOCK *waiting_lock; /* waiting lock (one lock only) */ - TABLE_LOCK_OWNER *waiting_for; /* transaction we're wating for */ - pthread_cond_t *cond; /* transactions waiting for us, wait on 'cond' */ - pthread_mutex_t *mutex; /* mutex is required to use 'cond' */ - uint16 loid; /* Lock Owner IDentifier */ -}; - -struct st_locked_table { - pthread_mutex_t mutex; /* mutex for everything below */ - HASH active; /* active locks ina hash */ - TABLE_LOCK *active_locks[LOCK_TYPES]; /* dl-list of locks per type */ - TABLE_LOCK *wait_queue_in, *wait_queue_out; /* wait deque */ -}; - void tablockman_init(TABLOCKMAN *, loid_to_tlo_func *, uint); void tablockman_destroy(TABLOCKMAN *); enum lockman_getlock_result tablockman_getlock(TABLOCKMAN *, TABLE_LOCK_OWNER *, @@ -81,7 +81,7 @@ void tablockman_init_locked_table(LOCKED_TABLE *, int); void tablockman_destroy_locked_table(LOCKED_TABLE *); #ifdef EXTRA_DEBUG -void print_tlo(TABLE_LOCK_OWNER *); +void tablockman_print_tlo(TABLE_LOCK_OWNER *); #endif #endif diff --git a/storage/maria/trnman.c b/storage/maria/trnman.c index 1c5281a3449..1df4c67b4aa 100644 --- a/storage/maria/trnman.c +++ b/storage/maria/trnman.c @@ -69,7 +69,8 @@ static TRN *short_trid_to_TRN(uint16 short_trid) return (TRN *)trn; } -static byte *trn_get_hash_key(const byte *trn, uint* len, my_bool unused) +static byte *trn_get_hash_key(const byte *trn, uint* len, + my_bool unused __attribute__ ((unused))) { *len= sizeof(TrID); return (byte *) & ((*((TRN **)trn))->trid); diff --git a/storage/maria/unittest/lockman-t.c b/storage/maria/unittest/lockman-t.c index 94b034bdaad..df8be054ba3 100644 --- a/storage/maria/unittest/lockman-t.c +++ b/storage/maria/unittest/lockman-t.c @@ -14,6 +14,10 @@ along with this program; if not, write to the Free Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ +/* + lockman for row and table locks +*/ + //#define EXTRA_VERBOSE #include diff --git a/storage/maria/unittest/lockman1-t.c b/storage/maria/unittest/lockman1-t.c index ab4b7beba9f..cf4791067dc 100644 --- a/storage/maria/unittest/lockman1-t.c +++ b/storage/maria/unittest/lockman1-t.c @@ -14,6 +14,10 @@ along with this program; if not, write to the Free Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ +/* + lockman for row locks, tablockman for table locks +*/ + //#define EXTRA_VERBOSE #include @@ -64,7 +68,7 @@ TABLE_LOCK_OWNER *loid2lo1(uint16 loid) #define lock_ok_l(O, R, L) \ test_lock(O, R, L, "", GOT_THE_LOCK_NEED_TO_INSTANT_LOCK_A_SUBRESOURCE) #define lock_conflict(O, R, L) \ - test_lock(O, R, L, "cannot ", DIDNT_GET_THE_LOCK); + test_lock(O, R, L, "cannot ", LOCK_TIMEOUT); void test_tablockman_simple() { @@ -164,8 +168,11 @@ int Ntables= 10; int table_lock_ratio= 10; enum lock_type lock_array[6]= {S, X, LS, LX, IS, IX}; char *lock2str[6]= {"S", "X", "LS", "LX", "IS", "IX"}; -char *res2str[4]= { +char *res2str[]= { "DIDN'T GET THE LOCK", + "OUT OF MEMORY", + "DEADLOCK", + "LOCK TIMEOUT", "GOT THE LOCK", "GOT THE LOCK NEED TO LOCK A SUBRESOURCE", "GOT THE LOCK NEED TO INSTANT LOCK A SUBRESOURCE"}; @@ -191,7 +198,7 @@ pthread_handler_t test_lockman(void *arg) res= tablockman_getlock(&tablockman, lo1, ltarray+table, lock_array[locklevel]); DIAG(("loid %2d, table %d, lock %s, res %s", loid, table, lock2str[locklevel], res2str[res])); - if (res == DIDNT_GET_THE_LOCK) + if (res < GOT_THE_LOCK) { lockman_release_locks(&lockman, lo); tablockman_release_locks(&tablockman, lo1); DIAG(("loid %2d, release all locks", loid)); @@ -208,11 +215,6 @@ pthread_handler_t test_lockman(void *arg) lock2str[locklevel+4], res2str[res])); switch (res) { - case DIDNT_GET_THE_LOCK: - lockman_release_locks(&lockman, lo); tablockman_release_locks(&tablockman, lo1); - DIAG(("loid %2d, release all locks", loid)); - timeout++; - continue; case GOT_THE_LOCK: continue; case GOT_THE_LOCK_NEED_TO_INSTANT_LOCK_A_SUBRESOURCE: @@ -232,7 +234,10 @@ pthread_handler_t test_lockman(void *arg) DBUG_ASSERT(res == GOT_THE_LOCK); continue; default: - DBUG_ASSERT(0); + lockman_release_locks(&lockman, lo); tablockman_release_locks(&tablockman, lo1); + DIAG(("loid %2d, release all locks", loid)); + timeout++; + continue; } } } diff --git a/storage/maria/unittest/lockman2-t.c b/storage/maria/unittest/lockman2-t.c index c515fc7ecd7..18c3072b241 100644 --- a/storage/maria/unittest/lockman2-t.c +++ b/storage/maria/unittest/lockman2-t.c @@ -14,6 +14,10 @@ along with this program; if not, write to the Free Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ +/* + tablockman for row and table locks +*/ + //#define EXTRA_VERBOSE #include @@ -57,7 +61,7 @@ TABLE_LOCK_OWNER *loid2lo1(uint16 loid) #define lock_ok_l(O, R, L) \ test_lock(O, R, L, "", GOT_THE_LOCK_NEED_TO_INSTANT_LOCK_A_SUBRESOURCE) #define lock_conflict(O, R, L) \ - test_lock(O, R, L, "cannot ", DIDNT_GET_THE_LOCK); + test_lock(O, R, L, "cannot ", LOCK_TIMEOUT); void test_tablockman_simple() { @@ -165,14 +169,34 @@ void run_test(const char *test, pthread_handler handler, int n, int m) my_free((void*)threads, MYF(0)); } +static void reinit_tlo(TABLOCKMAN *lm, TABLE_LOCK_OWNER *lo) +{ + TABLE_LOCK_OWNER backup= *lo; + + tablockman_release_locks(lm, lo); + /* + pthread_mutex_destroy(lo->mutex); + pthread_cond_destroy(lo->cond); + bzero(lo, sizeof(*lo)); + + lo->mutex= backup.mutex; + lo->cond= backup.cond; + lo->loid= backup.loid; + pthread_mutex_init(lo->mutex, MY_MUTEX_INIT_FAST); + pthread_cond_init(lo->cond, 0);*/ +} + pthread_mutex_t rt_mutex; int Nrows= 100; int Ntables= 10; int table_lock_ratio= 10; enum lock_type lock_array[6]= {S, X, LS, LX, IS, IX}; char *lock2str[6]= {"S", "X", "LS", "LX", "IS", "IX"}; -char *res2str[4]= { - "DIDN'T GET THE LOCK", +char *res2str[]= { + 0, + "OUT OF MEMORY", + "DEADLOCK", + "LOCK TIMEOUT", "GOT THE LOCK", "GOT THE LOCK NEED TO LOCK A SUBRESOURCE", "GOT THE LOCK NEED TO INSTANT LOCK A SUBRESOURCE"}; @@ -200,9 +224,9 @@ pthread_handler_t test_lockman(void *arg) res= tablockman_getlock(&tablockman, lo1, ltarray+table, lock_array[locklevel]); DIAG(("loid %2d, table %d, lock %s, res %s", loid, table, lock2str[locklevel], res2str[res])); - if (res == DIDNT_GET_THE_LOCK) + if (res < GOT_THE_LOCK) { - tablockman_release_locks(&tablockman, lo1); + reinit_tlo(&tablockman, lo1); DIAG(("loid %2d, release all locks", loid)); timeout++; continue; @@ -217,11 +241,6 @@ pthread_handler_t test_lockman(void *arg) lock2str[locklevel+4], res2str[res])); switch (res) { - case DIDNT_GET_THE_LOCK: - tablockman_release_locks(&tablockman, lo1); - DIAG(("loid %2d, release all locks", loid)); - timeout++; - continue; case GOT_THE_LOCK: continue; case GOT_THE_LOCK_NEED_TO_INSTANT_LOCK_A_SUBRESOURCE: @@ -230,9 +249,9 @@ pthread_handler_t test_lockman(void *arg) res= tablockman_getlock(&tablockman, lo1, ltarray+row, lock_array[locklevel]); DIAG(("loid %2d, ROW %d, lock %s, res %s", loid, row, lock2str[locklevel], res2str[res])); - if (res == DIDNT_GET_THE_LOCK) + if (res < GOT_THE_LOCK) { - tablockman_release_locks(&tablockman, lo1); + reinit_tlo(&tablockman, lo1); DIAG(("loid %2d, release all locks", loid)); timeout++; continue; @@ -240,12 +259,15 @@ pthread_handler_t test_lockman(void *arg) DBUG_ASSERT(res == GOT_THE_LOCK); continue; default: - DBUG_ASSERT(0); + reinit_tlo(&tablockman, lo1); + DIAG(("loid %2d, release all locks", loid)); + timeout++; + continue; } } } - tablockman_release_locks(&tablockman, lo1); + reinit_tlo(&tablockman, lo1); pthread_mutex_lock(&rt_mutex); rt_num_threads--; @@ -264,7 +286,7 @@ int main() my_init(); pthread_mutex_init(&rt_mutex, 0); - plan(39); + plan(40); if (my_atomic_initialize()) return exit_status(); @@ -299,7 +321,7 @@ int main() Nrows= 100; Ntables= 10; table_lock_ratio= 10; - //run_test("\"random lock\" stress test", test_lockman, THREADS, CYCLES); + run_test("\"random lock\" stress test", test_lockman, THREADS, CYCLES); #if 0 /* "real-life" simulation - many rows, no table locks */ Nrows= 1000000; -- cgit v1.2.1 From 6691ce30ecd82baf9877242c078a5e1d12a78369 Mon Sep 17 00:00:00 2001 From: unknown Date: Thu, 16 Nov 2006 22:53:53 +0100 Subject: Maria - WL#3134 Key cache to Page cache conversion Moving the test_pagecache_ tests from mysys to unittest/mysys. Means fixing includes to work with the new directory, some Makefile.am editing, replacing memset() with bfill(). test_page_cache_*.c renamed to mf_pagecache_*-t.c (-t is the standard suffix for tests in the mytap protocol). Also added plan() and exit_status() calls to tests. Sanja, I put some TODOs for you at the start of mf_pagecache_*.c unittest/mysys/test_file.h: Rename: mysys/test_file.h -> unittest/mysys/test_file.h mysys/Makefile.am: pagecache test files move to top/unittest/mysys mysys/mf_pagecache.c: my_bit.h needed to compile. unittest/mysys/Makefile.am: INCLUDES is a better place for includes than AM_CPPFLAGS (the latter get overriden by prog_CPPFLAGS, which is not desirable here). Adding pagecache's test programs (moved from mysys); test_pagecache_* has been renamed to mf_pagecache*-t (-t is the required suffix for test executables in the mytap framework). unittest/mysys/mf_pagecache_consist.c: fixing includes to work with the new directory. The test must return an exit code informing if any part failed. TODOs for Sanja. unittest/mysys/mf_pagecache_single.c: fixing includes to work with new directory. adding a plan() to account for the number of tests. Adding exit_status() to tell how many tests failed. memset() was giving a compilation warning (implicit declaration etc), properly due to me removing stdio.h etc, so I replaced it with bfill(). TODOs for Sanja. unittest/mysys/test_file.c: moved from mysys (piece of the page cache tests) and includes fixed. --- mysys/Makefile.am | 55 ---- mysys/mf_pagecache.c | 2 + mysys/test_file.c | 70 ---- mysys/test_file.h | 14 - mysys/test_pagecache_consist.c | 447 -------------------------- mysys/test_pagecache_single.c | 589 ---------------------------------- unittest/mysys/Makefile.am | 46 ++- unittest/mysys/mf_pagecache_consist.c | 445 +++++++++++++++++++++++++ unittest/mysys/mf_pagecache_single.c | 587 +++++++++++++++++++++++++++++++++ unittest/mysys/test_file.c | 68 ++++ unittest/mysys/test_file.h | 14 + 11 files changed, 1159 insertions(+), 1178 deletions(-) delete mode 100644 mysys/test_file.c delete mode 100644 mysys/test_file.h delete mode 100755 mysys/test_pagecache_consist.c delete mode 100644 mysys/test_pagecache_single.c create mode 100755 unittest/mysys/mf_pagecache_consist.c create mode 100644 unittest/mysys/mf_pagecache_single.c create mode 100644 unittest/mysys/test_file.c create mode 100644 unittest/mysys/test_file.h diff --git a/mysys/Makefile.am b/mysys/Makefile.am index 2f8b34dc301..4d9570febbd 100644 --- a/mysys/Makefile.am +++ b/mysys/Makefile.am @@ -130,61 +130,6 @@ test_base64$(EXEEXT): base64.c $(LIBRARIES) $(LINK) $(FLAGS) -DMAIN ./test_base64.c $(LDADD) $(LIBS) $(RM) -f ./test_base64.c -test_mf_pagecache.o: mf_pagecache.c ../include/pagecache.h $(LIBRARIES) - $(CP) $(srcdir)/mf_pagecache.c test_mf_pagecache.c - $(COMPILE) $(FLAGS) -DPAGECACHE_DEBUG -DEXTRA_DEBUG -c test_mf_pagecache.c - -test_file.o: test_file.c test_file.h - $(COMPILE) $(FLAGS) -DPAGECACHE_DEBUG -DEXTRA_DEBUG -c test_file.c - -test_pagecache_single1k$(EXEEXT): test_pagecache_single.c test_mf_pagecache.o ../unittest/mytap/tap.o test_file.o $(LIBRARIES) - $(LINK) $(FLAGS) -DMAIN -DPAGECACHE_DEBUG -DPAGE_SIZE=1024 -DEXTRA_DEBUG $(srcdir)/test_pagecache_single.c test_mf_pagecache.o ../unittest/mytap/tap.o test_file.o $(LDADD) $(LIBS) - -test_pagecache_single8k$(EXEEXT): test_pagecache_single.c test_mf_pagecache.o ../unittest/mytap/tap.o test_file.o $(LIBRARIES) - $(LINK) $(FLAGS) -DMAIN -DPAGECACHE_DEBUG -DPAGE_SIZE=8192 -DEXTRA_DEBUG $(srcdir)/test_pagecache_single.c test_mf_pagecache.o ../unittest/mytap/tap.o test_file.o $(LDADD) $(LIBS) - -test_pagecache_single64k$(EXEEXT): test_pagecache_single.c test_mf_pagecache.o ../unittest/mytap/tap.o test_file.o $(LIBRARIES) - $(LINK) $(FLAGS) -DMAIN -DPAGECACHE_DEBUG -DPAGE_SIZE=65536 -DEXTRA_DEBUG $(srcdir)/test_pagecache_single.c test_mf_pagecache.o ../unittest/mytap/tap.o test_file.o $(LDADD) $(LIBS) - -test_pagecache_single: test_pagecache_single1k$(EXEEXT) test_pagecache_single8k$(EXEEXT) test_pagecache_single64k$(EXEEXT) - ./test_pagecache_single64k$(EXEEXT) - ./test_pagecache_single8k$(EXEEXT) - ./test_pagecache_single1k$(EXEEXT) - -test_pagecache_consist1k$(EXEEXT): test_pagecache_consist.c test_mf_pagecache.o ../unittest/mytap/tap.o $(LIBRARIES) - $(LINK) $(FLAGS) -DMAIN -DPAGECACHE_DEBUG -DPAGE_SIZE=1024 -DEXTRA_DEBUG $(srcdir)/test_pagecache_consist.c test_mf_pagecache.o ../unittest/mytap/tap.o $(LDADD) $(LIBS) - -test_pagecache_consist64k$(EXEEXT): test_pagecache_consist.c test_mf_pagecache.o ../unittest/mytap/tap.o $(LIBRARIES) - $(LINK) $(FLAGS) -DMAIN -DPAGECACHE_DEBUG -DPAGE_SIZE=65536 -DEXTRA_DEBUG $(srcdir)/test_pagecache_consist.c test_mf_pagecache.o ../unittest/mytap/tap.o $(LDADD) $(LIBS) - -test_pagecache_consist1kHC$(EXEEXT): test_pagecache_consist.c test_mf_pagecache.o ../unittest/mytap/tap.o $(LIBRARIES) - $(LINK) $(FLAGS) -DMAIN -DPAGECACHE_DEBUG -DTEST_HIGH_CONCURENCY -DPAGE_SIZE=1024 -DEXTRA_DEBUG $(srcdir)/test_pagecache_consist.c test_mf_pagecache.o ../unittest/mytap/tap.o $(LDADD) $(LIBS) - -test_pagecache_consist64kHC$(EXEEXT): test_pagecache_consist.c test_mf_pagecache.o ../unittest/mytap/tap.o $(LIBRARIES) - $(LINK) $(FLAGS) -DMAIN -DPAGECACHE_DEBUG -DTEST_HIGH_CONCURENCY -DPAGE_SIZE=65536 -DEXTRA_DEBUG $(srcdir)/test_pagecache_consist.c test_mf_pagecache.o ../unittest/mytap/tap.o $(LDADD) $(LIBS) - -test_pagecache_consist1kRD$(EXEEXT): test_pagecache_consist.c test_mf_pagecache.o ../unittest/mytap/tap.o $(LIBRARIES) - $(LINK) $(FLAGS) -DMAIN -DPAGECACHE_DEBUG -DTEST_READERS -DPAGE_SIZE=1024 -DEXTRA_DEBUG $(srcdir)/test_pagecache_consist.c test_mf_pagecache.o ../unittest/mytap/tap.o $(LDADD) $(LIBS) - -test_pagecache_consist64kRD$(EXEEXT): test_pagecache_consist.c test_mf_pagecache.o ../unittest/mytap/tap.o $(LIBRARIES) - $(LINK) $(FLAGS) -DMAIN -DPAGECACHE_DEBUG -DTEST_READERS -DPAGE_SIZE=65536 -DEXTRA_DEBUG $(srcdir)/test_pagecache_consist.c test_mf_pagecache.o ../unittest/mytap/tap.o $(LDADD) $(LIBS) - -test_pagecache_consist1kWR$(EXEEXT): test_pagecache_consist.c test_mf_pagecache.o ../unittest/mytap/tap.o $(LIBRARIES) - $(LINK) $(FLAGS) -DMAIN -DPAGECACHE_DEBUG -DTEST_WRITERS -DPAGE_SIZE=1024 -DEXTRA_DEBUG $(srcdir)/test_pagecache_consist.c test_mf_pagecache.o ../unittest/mytap/tap.o $(LDADD) $(LIBS) - -test_pagecache_consist64kWR$(EXEEXT): test_pagecache_consist.c test_mf_pagecache.o ../unittest/mytap/tap.o $(LIBRARIES) - $(LINK) $(FLAGS) -DMAIN -DPAGECACHE_DEBUG -DTEST_WRITERS -DPAGE_SIZE=65536 -DEXTRA_DEBUG $(srcdir)/test_pagecache_consist.c test_mf_pagecache.o ../unittest/mytap/tap.o $(LDADD) $(LIBS) - -test_pagecache_consist: test_pagecache_consist1k$(EXEEXT) test_pagecache_consist64k$(EXEEXT) test_pagecache_consist1kHC$(EXEEXT) test_pagecache_consist64kHC$(EXEEXT) test_pagecache_consist1kRD$(EXEEXT) test_pagecache_consist64kRD$(EXEEXT) test_pagecache_consist1kWR$(EXEEXT) test_pagecache_consist64kWR$(EXEEXT) - ./test_pagecache_consist1k$(EXEEXT) - ./test_pagecache_consist64k$(EXEEXT) - ./test_pagecache_consist1kHC$(EXEEXT) - ./test_pagecache_consist64kHC$(EXEEXT) - ./test_pagecache_consist1kRD$(EXEEXT) - ./test_pagecache_consist64kRD$(EXEEXT) - ./test_pagecache_consist1kWR$(EXEEXT) - ./test_pagecache_consist64kWR$(EXEEXT) - # Don't update the files from bitkeeper %::SCCS/s.% diff --git a/mysys/mf_pagecache.c b/mysys/mf_pagecache.c index e1ccab9e06f..d8e235bc7f4 100755 --- a/mysys/mf_pagecache.c +++ b/mysys/mf_pagecache.c @@ -45,6 +45,8 @@ #include #include #include +#include + /* Some compilation flags have been added specifically for this module diff --git a/mysys/test_file.c b/mysys/test_file.c deleted file mode 100644 index 7ffca48023d..00000000000 --- a/mysys/test_file.c +++ /dev/null @@ -1,70 +0,0 @@ -#include "mysys_priv.h" -#include "my_dir.h" -#include -#include -#include -#include "test_file.h" - - -/* - Check that file contance correspond to descriptor - - SYNOPSIS - test_file() - file File to test - file_name Path (and name) of file which is tested - size size of file - buff_size size of buffer which is enought to check the file - desc file descriptor to check with - - RETURN - 1 file if OK - 0 error -*/ - -int test_file(PAGECACHE_FILE file, char *file_name, - off_t size, size_t buff_size, struct file_desc *desc) -{ - MY_STAT stat_buff, *stat; - unsigned char *buffr= malloc(buff_size); - off_t pos= 0; - size_t byte; - int step= 0; - - if ((stat= my_stat(file_name, &stat_buff, MYF(0))) == NULL) - { - diag("Can't stat() %s (errno: %d)\n", file_name, errno); - return 0; - } - if (stat->st_size != size) - { - diag("file %s size is %lu (should be %lu)\n", - file_name, (ulong) stat->st_size, (ulong) size); - return 0; - } - /* check content */ - my_seek(file.file, 0, SEEK_SET, MYF(0)); - while (desc[step].length != 0) - { - if (my_read(file.file, (char*)buffr, desc[step].length, MYF(0)) != - desc[step].length) - { - diag("Can't read %u bytes from %s (errno: %d)\n", - (uint)desc[step].length, file_name, errno); - return 0; - } - for (byte= 0; byte < desc[step].length; byte++) - { - if (buffr[byte] != desc[step].content) - { - diag("content of %s mismatch 0x%x in position %lu instead of 0x%x\n", - file_name, (uint) buffr[byte], (ulong) (pos + byte), - desc[step].content); - return 0; - } - } - pos+= desc[step].length; - step++; - } - return 1; -} diff --git a/mysys/test_file.h b/mysys/test_file.h deleted file mode 100644 index ea787c123ed..00000000000 --- a/mysys/test_file.h +++ /dev/null @@ -1,14 +0,0 @@ - -#include - -/* - File content descriptor -*/ -struct file_desc -{ - unsigned int length; - unsigned char content; -}; - -int test_file(PAGECACHE_FILE file, char *file_name, - off_t size, size_t buff_size, struct file_desc *desc); diff --git a/mysys/test_pagecache_consist.c b/mysys/test_pagecache_consist.c deleted file mode 100755 index 86698fbb860..00000000000 --- a/mysys/test_pagecache_consist.c +++ /dev/null @@ -1,447 +0,0 @@ -#include "mysys_priv.h" -#include "../include/my_pthread.h" -#include "../include/pagecache.h" -#include -#include "my_dir.h" -#include -#include -#include -#include -#include "../unittest/mytap/tap.h" -#include - -/*#define PAGE_SIZE 65536*/ -#define PCACHE_SIZE (PAGE_SIZE*1024*8) - -#ifndef DBUG_OFF -static const char* default_dbug_option; -#endif - -static char *file1_name= (char*)"page_cache_test_file_1"; -static PAGECACHE_FILE file1; -static pthread_cond_t COND_thread_count; -static pthread_mutex_t LOCK_thread_count; -static uint thread_count; -static PAGECACHE pagecache; - -#ifdef TEST_HIGH_CONCURENCY -static uint number_of_readers= 10; -static uint number_of_writers= 20; -static uint number_of_tests= 30000; -static uint record_length_limit= PAGE_SIZE/200; -static uint number_of_pages= 20; -static uint flush_divider= 1000; -#else /*TEST_HIGH_CONCURENCY*/ -#ifdef TEST_READERS -static uint number_of_readers= 10; -static uint number_of_writers= 1; -static uint number_of_tests= 30000; -static uint record_length_limit= PAGE_SIZE/200; -static uint number_of_pages= 20; -static uint flush_divider= 1000; -#else /*TEST_READERS*/ -#ifdef TEST_WRITERS -static uint number_of_readers= 0; -static uint number_of_writers= 10; -static uint number_of_tests= 30000; -static uint record_length_limit= PAGE_SIZE/200; -static uint number_of_pages= 20; -static uint flush_divider= 1000; -#else /*TEST_WRITERS*/ -static uint number_of_readers= 10; -static uint number_of_writers= 10; -static uint number_of_tests= 50000; -static uint record_length_limit= PAGE_SIZE/200; -static uint number_of_pages= 20000; -static uint flush_divider= 1000; -#endif /*TEST_WRITERS*/ -#endif /*TEST_READERS*/ -#endif /*TEST_HIGH_CONCURENCY*/ - - -/* check page consistency */ -uint check_page(uchar *buff, ulong offset, int page_locked, int page_no, - int tag) -{ - uint end= sizeof(uint); - uint num= *((uint *)buff); - uint i; - DBUG_ENTER("check_page"); - - for (i= 0; i < num; i++) - { - uint len= *((uint *)(buff + end)); - uint j; - end+= sizeof(uint)+ sizeof(uint); - if (len + end > PAGE_SIZE) - { - diag("incorrect field header #%u by offset %lu\n", i, offset + end + j); - goto err; - } - for(j= 0; j < len; j++) - { - if (buff[end + j] != (uchar)((i+1) % 256)) - { - diag("incorrect %lu byte\n", offset + end + j); - goto err; - } - } - end+= len; - } - for(i= end; i < PAGE_SIZE; i++) - { - if (buff[i] != 0) - { - int h; - DBUG_PRINT("err", - ("byte %lu (%lu + %u), page %u (%s, end: %u, recs: %u, tag: %d) should be 0\n", - offset + i, offset, i, page_no, - (page_locked ? "locked" : "unlocked"), - end, num, tag)); - diag("byte %lu (%lu + %u), page %u (%s, end: %u, recs: %u, tag: %d) should be 0\n", - offset + i, offset, i, page_no, - (page_locked ? "locked" : "unlocked"), - end, num, tag); - h= my_open("wrong_page", O_CREAT | O_TRUNC | O_RDWR, MYF(0)); - my_pwrite(h, buff, PAGE_SIZE, 0, MYF(0)); - my_close(h, MYF(0)); - goto err; - } - } - DBUG_RETURN(end); -err: - DBUG_PRINT("err", ("try to flush")); - if (page_locked) - { - pagecache_delete_page(&pagecache, &file1, page_no, - PAGECACHE_LOCK_LEFT_WRITELOCKED, 1); - } - else - { - flush_pagecache_blocks(&pagecache, &file1, FLUSH_RELEASE); - } - exit(1); -} - -void put_rec(uchar *buff, uint end, uint len, uint tag) -{ - uint i; - uint num= *((uint *)buff); - if (!len) - len= 1; - if (end + sizeof(uint)*2 + len > PAGE_SIZE) - return; - *((uint *)(buff + end))= len; - end+= sizeof(uint); - *((uint *)(buff + end))= tag; - end+= sizeof(uint); - num++; - *((uint *)buff)= num; - *((uint*)(buff + end))= len; - for (i= end; i < (len + end); i++) - { - buff[i]= (uchar) num % 256; - } -} - -/* - Recreate and reopen a file for test - - SYNOPSIS - reset_file() - file File to reset - file_name Path (and name) of file which should be reset -*/ - -void reset_file(PAGECACHE_FILE file, char *file_name) -{ - flush_pagecache_blocks(&pagecache, &file1, FLUSH_RELEASE); - if (my_close(file1.file, MYF(0)) != 0) - { - diag("Got error during %s closing from close() (errno: %d)\n", - file_name, errno); - exit(1); - } - my_delete(file_name, MYF(0)); - if ((file.file= my_open(file_name, - O_CREAT | O_TRUNC | O_RDWR, MYF(0))) == -1) - { - diag("Got error during %s creation from open() (errno: %d)\n", - file_name, errno); - exit(1); - } -} - - -void reader(int num) -{ - unsigned char *buffr= malloc(PAGE_SIZE); - uint i; - - for (i= 0; i < number_of_tests; i++) - { - uint page= rand()/(RAND_MAX/number_of_pages); - pagecache_read(&pagecache, &file1, page, 3, (char*)buffr, - PAGECACHE_PLAIN_PAGE, - PAGECACHE_LOCK_LEFT_UNLOCKED, - 0); - check_page(buffr, page * PAGE_SIZE, 0, page, -num); - if (i % 500 == 0) - printf("reader%d: %d\n", num, i); - - } - printf("reader%d: done\n", num); - free(buffr); -} - - -void writer(int num) -{ - unsigned char *buffr= malloc(PAGE_SIZE); - uint i; - - for (i= 0; i < number_of_tests; i++) - { - uint end; - uint page= rand()/(RAND_MAX/number_of_pages); - pagecache_read(&pagecache, &file1, page, 3, (char*)buffr, - PAGECACHE_PLAIN_PAGE, - PAGECACHE_LOCK_WRITE, - 0); - end= check_page(buffr, page * PAGE_SIZE, 1, page, num); - put_rec(buffr, end, rand()/(RAND_MAX/record_length_limit), num); - pagecache_write(&pagecache, &file1, page, 3, (char*)buffr, - PAGECACHE_PLAIN_PAGE, - PAGECACHE_LOCK_WRITE_UNLOCK, - PAGECACHE_UNPIN, - PAGECACHE_WRITE_DELAY, - 0); - - if (i % flush_divider == 0) - flush_pagecache_blocks(&pagecache, &file1, FLUSH_FORCE_WRITE); - if (i % 500 == 0) - printf("writer%d: %d\n", num, i); - } - printf("writer%d: done\n", num); - free(buffr); -} - - -static void *test_thread_reader(void *arg) -{ - int param=*((int*) arg); - - my_thread_init(); - DBUG_ENTER("test_reader"); - DBUG_PRINT("enter", ("param: %d", param)); - - reader(param); - - DBUG_PRINT("info", ("Thread %s ended\n", my_thread_name())); - pthread_mutex_lock(&LOCK_thread_count); - thread_count--; - VOID(pthread_cond_signal(&COND_thread_count)); /* Tell main we are ready */ - pthread_mutex_unlock(&LOCK_thread_count); - free((gptr) arg); - my_thread_end(); - DBUG_RETURN(0); -} - -static void *test_thread_writer(void *arg) -{ - int param=*((int*) arg); - - my_thread_init(); - DBUG_ENTER("test_writer"); - DBUG_PRINT("enter", ("param: %d", param)); - - writer(param); - - DBUG_PRINT("info", ("Thread %s ended\n", my_thread_name())); - pthread_mutex_lock(&LOCK_thread_count); - thread_count--; - VOID(pthread_cond_signal(&COND_thread_count)); /* Tell main we are ready */ - pthread_mutex_unlock(&LOCK_thread_count); - free((gptr) arg); - my_thread_end(); - DBUG_RETURN(0); -} - -int main(int argc, char **argv __attribute__((unused))) -{ - pthread_t tid; - pthread_attr_t thr_attr; - int *param, error, pagen; - - MY_INIT(argv[0]); - -#ifndef DBUG_OFF -#if defined(__WIN__) - default_dbug_option= "d:t:i:O,\\test_pagecache_consist.trace"; -#else - default_dbug_option= "d:t:i:o,/tmp/test_pagecache_consist.trace"; -#endif - if (argc > 1) - { - DBUG_SET(default_dbug_option); - DBUG_SET_INITIAL(default_dbug_option); - } -#endif - - - DBUG_ENTER("main"); - DBUG_PRINT("info", ("Main thread: %s\n", my_thread_name())); - if ((file1.file= my_open(file1_name, - O_CREAT | O_TRUNC | O_RDWR, MYF(0))) == -1) - { - fprintf(stderr, "Got error during file1 creation from open() (errno: %d)\n", - errno); - exit(1); - } - DBUG_PRINT("info", ("file1: %d", file1.file)); - if (chmod(file1_name, S_IRWXU | S_IRWXG | S_IRWXO) != 0) - { - fprintf(stderr, "Got error during file1 chmod() (errno: %d)\n", - errno); - exit(1); - } - my_pwrite(file1.file, "test file", 9, 0, MYF(0)); - - if ((error= pthread_cond_init(&COND_thread_count, NULL))) - { - fprintf(stderr, "COND_thread_count: %d from pthread_cond_init (errno: %d)\n", - error, errno); - exit(1); - } - if ((error= pthread_mutex_init(&LOCK_thread_count, MY_MUTEX_INIT_FAST))) - { - fprintf(stderr, "LOCK_thread_count: %d from pthread_cond_init (errno: %d)\n", - error, errno); - exit(1); - } - - if ((error= pthread_attr_init(&thr_attr))) - { - fprintf(stderr,"Got error: %d from pthread_attr_init (errno: %d)\n", - error,errno); - exit(1); - } - if ((error= pthread_attr_setdetachstate(&thr_attr, PTHREAD_CREATE_DETACHED))) - { - fprintf(stderr, - "Got error: %d from pthread_attr_setdetachstate (errno: %d)\n", - error,errno); - exit(1); - } - -#ifndef pthread_attr_setstacksize /* void return value */ - if ((error= pthread_attr_setstacksize(&thr_attr, 65536L))) - { - fprintf(stderr,"Got error: %d from pthread_attr_setstacksize (errno: %d)\n", - error,errno); - exit(1); - } -#endif -#ifdef HAVE_THR_SETCONCURRENCY - VOID(thr_setconcurrency(2)); -#endif - - my_thread_global_init(); - - - if ((pagen= init_pagecache(&pagecache, PCACHE_SIZE, 0, 0, - PAGE_SIZE, 0)) == 0) - { - fprintf(stderr,"Got error: init_pagecache() (errno: %d)\n", - errno); - exit(1); - } - DBUG_PRINT("info", ("Page cache %d pages", pagen)); - { - unsigned char *buffr= malloc(PAGE_SIZE); - uint i; - memset(buffr, '\0', PAGE_SIZE); - for (i= 0; i < number_of_pages; i++) - { - pagecache_write(&pagecache, &file1, i, 3, (char*)buffr, - PAGECACHE_PLAIN_PAGE, - PAGECACHE_LOCK_LEFT_UNLOCKED, - PAGECACHE_PIN_LEFT_UNPINNED, - PAGECACHE_WRITE_DELAY, - 0); - } - flush_pagecache_blocks(&pagecache, &file1, FLUSH_FORCE_WRITE); - free(buffr); - } - if ((error= pthread_mutex_lock(&LOCK_thread_count))) - { - fprintf(stderr,"LOCK_thread_count: %d from pthread_mutex_lock (errno: %d)\n", - error,errno); - exit(1); - } - while (number_of_readers != 0 || number_of_writers != 0) - { - if (number_of_readers != 0) - { - param=(int*) malloc(sizeof(int)); - *param= number_of_readers; - if ((error= pthread_create(&tid, &thr_attr, test_thread_reader, - (void*) param))) - { - fprintf(stderr,"Got error: %d from pthread_create (errno: %d)\n", - error,errno); - exit(1); - } - thread_count++; - number_of_readers--; - } - if (number_of_writers != 0) - { - param=(int*) malloc(sizeof(int)); - *param= number_of_writers; - if ((error= pthread_create(&tid, &thr_attr, test_thread_writer, - (void*) param))) - { - fprintf(stderr,"Got error: %d from pthread_create (errno: %d)\n", - error,errno); - exit(1); - } - thread_count++; - number_of_writers--; - } - } - DBUG_PRINT("info", ("Thread started")); - pthread_mutex_unlock(&LOCK_thread_count); - - pthread_attr_destroy(&thr_attr); - - /* wait finishing */ - if ((error= pthread_mutex_lock(&LOCK_thread_count))) - fprintf(stderr,"LOCK_thread_count: %d from pthread_mutex_lock\n",error); - while (thread_count) - { - if ((error= pthread_cond_wait(&COND_thread_count,&LOCK_thread_count))) - fprintf(stderr,"COND_thread_count: %d from pthread_cond_wait\n",error); - } - if ((error= pthread_mutex_unlock(&LOCK_thread_count))) - fprintf(stderr,"LOCK_thread_count: %d from pthread_mutex_unlock\n",error); - DBUG_PRINT("info", ("thread ended")); - - end_pagecache(&pagecache, 1); - DBUG_PRINT("info", ("Page cache ended")); - - if (my_close(file1.file, MYF(0)) != 0) - { - fprintf(stderr, "Got error during file1 closing from close() (errno: %d)\n", - errno); - exit(1); - } - /*my_delete(file1_name, MYF(0));*/ - my_thread_global_end(); - - DBUG_PRINT("info", ("file1 (%d) closed", file1.file)); - - DBUG_PRINT("info", ("Program end")); - - DBUG_RETURN(0); -} diff --git a/mysys/test_pagecache_single.c b/mysys/test_pagecache_single.c deleted file mode 100644 index 9df7844cfa5..00000000000 --- a/mysys/test_pagecache_single.c +++ /dev/null @@ -1,589 +0,0 @@ -#include "mysys_priv.h" -#include "../include/my_pthread.h" -#include "../include/pagecache.h" -#include "my_dir.h" -#include -#include -#include -#include -#include -#include "../unittest/mytap/tap.h" -#include "test_file.h" - -/* #define PAGE_SIZE 1024 */ -#define PCACHE_SIZE (PAGE_SIZE*1024*10) - -#ifndef DBUG_OFF -static const char* default_dbug_option; -#endif - -static char *file1_name= (char*)"page_cache_test_file_1"; -static PAGECACHE_FILE file1; -static pthread_cond_t COND_thread_count; -static pthread_mutex_t LOCK_thread_count; -static uint thread_count; -static PAGECACHE pagecache; - -/* - File contance descriptors -*/ -static struct file_desc simple_read_write_test_file[]= -{ - {PAGE_SIZE, '\1'}, - { 0, 0} -}; -static struct file_desc simple_read_change_write_read_test_file[]= -{ - {PAGE_SIZE/2, '\65'}, - {PAGE_SIZE/2, '\1'}, - { 0, 0} -}; -static struct file_desc simple_pin_test_file1[]= -{ - {PAGE_SIZE*2, '\1'}, - { 0, 0} -}; -static struct file_desc simple_pin_test_file2[]= -{ - {PAGE_SIZE/2, '\1'}, - {PAGE_SIZE/2, (unsigned char)129}, - {PAGE_SIZE, '\1'}, - { 0, 0} -}; -static struct file_desc simple_delete_forget_test_file[]= -{ - {PAGE_SIZE, '\1'}, - { 0, 0} -}; -static struct file_desc simple_delete_flush_test_file[]= -{ - {PAGE_SIZE, '\2'}, - { 0, 0} -}; - - -/* - Recreate and reopen a file for test - - SYNOPSIS - reset_file() - file File to reset - file_name Path (and name) of file which should be reset -*/ - -void reset_file(PAGECACHE_FILE file, char *file_name) -{ - flush_pagecache_blocks(&pagecache, &file1, FLUSH_RELEASE); - if (my_close(file1.file, MYF(0)) != 0) - { - diag("Got error during %s closing from close() (errno: %d)\n", - file_name, errno); - exit(1); - } - my_delete(file_name, MYF(0)); - if ((file.file= my_open(file_name, - O_CREAT | O_TRUNC | O_RDWR, MYF(0))) == -1) - { - diag("Got error during %s creation from open() (errno: %d)\n", - file_name, errno); - exit(1); - } -} - -/* - Write then read page, check file on disk -*/ - -int simple_read_write_test() -{ - unsigned char *buffw= malloc(PAGE_SIZE); - unsigned char *buffr= malloc(PAGE_SIZE); - int res; - DBUG_ENTER("simple_read_write_test"); - memset(buffw, '\1', PAGE_SIZE); - pagecache_write(&pagecache, &file1, 0, 3, (char*)buffw, - PAGECACHE_PLAIN_PAGE, - PAGECACHE_LOCK_LEFT_UNLOCKED, - PAGECACHE_PIN_LEFT_UNPINNED, - PAGECACHE_WRITE_DELAY, - 0); - pagecache_read(&pagecache, &file1, 0, 3, (char*)buffr, - PAGECACHE_PLAIN_PAGE, - PAGECACHE_LOCK_LEFT_UNLOCKED, - 0); - ok((res= test(memcmp(buffr, buffw, PAGE_SIZE) == 0)), - "Simple write-read page "); - flush_pagecache_blocks(&pagecache, &file1, FLUSH_FORCE_WRITE); - ok((res&= test(test_file(file1, file1_name, PAGE_SIZE, PAGE_SIZE, - simple_read_write_test_file))), - "Simple write-read page file"); - if (res) - reset_file(file1, file1_name); - free(buffw); - free(buffr); - DBUG_RETURN(res); -} - - -/* - Prepare page, then read (and lock), change (write new value and unlock), - then check the page in the cache and on the disk -*/ -int simple_read_change_write_read_test() -{ - unsigned char *buffw= malloc(PAGE_SIZE); - unsigned char *buffr= malloc(PAGE_SIZE); - int res; - DBUG_ENTER("simple_read_change_write_read_test"); - /* prepare the file */ - memset(buffw, '\1', PAGE_SIZE); - pagecache_write(&pagecache, &file1, 0, 3, (char*)buffw, - PAGECACHE_PLAIN_PAGE, - PAGECACHE_LOCK_LEFT_UNLOCKED, - PAGECACHE_PIN_LEFT_UNPINNED, - PAGECACHE_WRITE_DELAY, - 0); - flush_pagecache_blocks(&pagecache, &file1, FLUSH_FORCE_WRITE); - /* test */ - pagecache_read(&pagecache, &file1, 0, 3, (char*)buffw, - PAGECACHE_PLAIN_PAGE, - PAGECACHE_LOCK_WRITE, - 0); - memset(buffw, '\65', PAGE_SIZE/2); - pagecache_write(&pagecache, &file1, 0, 3, (char*)buffw, - PAGECACHE_PLAIN_PAGE, - PAGECACHE_LOCK_WRITE_UNLOCK, - PAGECACHE_UNPIN, - PAGECACHE_WRITE_DELAY, - 0); - - pagecache_read(&pagecache, &file1, 0, 3, (char*)buffr, - PAGECACHE_PLAIN_PAGE, - PAGECACHE_LOCK_LEFT_UNLOCKED, - 0); - ok((res= test(memcmp(buffr, buffw, PAGE_SIZE) == 0)), - "Simple read-change-write-read page "); - flush_pagecache_blocks(&pagecache, &file1, FLUSH_FORCE_WRITE); - ok((res&= test(test_file(file1, file1_name, PAGE_SIZE, PAGE_SIZE, - simple_read_change_write_read_test_file))), - "Simple read-change-write-read page file"); - if (res) - reset_file(file1, file1_name); - free(buffw); - free(buffr); - DBUG_RETURN(res); -} - - -/* - Prepare page, read page 0 (and pin) then write page 1 and page 0. - Flush the file (shold flush only page 1 and return 1 (page 0 is - still pinned). - Check file on the disk. - Unpin and flush. - Check file on the disk. -*/ -int simple_pin_test() -{ - unsigned char *buffw= malloc(PAGE_SIZE); - unsigned char *buffr= malloc(PAGE_SIZE); - int res; - DBUG_ENTER("simple_pin_test"); - /* prepare the file */ - memset(buffw, '\1', PAGE_SIZE); - pagecache_write(&pagecache, &file1, 0, 3, (char*)buffw, - PAGECACHE_PLAIN_PAGE, - PAGECACHE_LOCK_LEFT_UNLOCKED, - PAGECACHE_PIN_LEFT_UNPINNED, - PAGECACHE_WRITE_DELAY, - 0); - /* test */ - if (flush_pagecache_blocks(&pagecache, &file1, FLUSH_FORCE_WRITE)) - { - diag("error in flush_pagecache_blocks\n"); - exit(1); - } - pagecache_read(&pagecache, &file1, 0, 3, (char*)buffw, - PAGECACHE_PLAIN_PAGE, - PAGECACHE_LOCK_WRITE, - 0); - pagecache_write(&pagecache, &file1, 1, 3, (char*)buffw, - PAGECACHE_PLAIN_PAGE, - PAGECACHE_LOCK_LEFT_UNLOCKED, - PAGECACHE_PIN_LEFT_UNPINNED, - PAGECACHE_WRITE_DELAY, - 0); - memset(buffw + PAGE_SIZE/2, ((unsigned char) 129), PAGE_SIZE/2); - pagecache_write(&pagecache, &file1, 0, 3, (char*)buffw, - PAGECACHE_PLAIN_PAGE, - PAGECACHE_LOCK_WRITE_TO_READ, - PAGECACHE_PIN_LEFT_PINNED, - PAGECACHE_WRITE_DELAY, - 0); - /* - We have to get error because one page of the file is pinned, - other page should be flushed - */ - if (!flush_pagecache_blocks(&pagecache, &file1, FLUSH_FORCE_WRITE)) - { - diag("Did not get error in flush_pagecache_blocks\n"); - res= 0; - goto err; - } - ok((res= test(test_file(file1, file1_name, PAGE_SIZE*2, PAGE_SIZE*2, - simple_pin_test_file1))), - "Simple pin page file with pin"); - pagecache_unlock_page(&pagecache, - &file1, - 0, - PAGECACHE_LOCK_READ_UNLOCK, - PAGECACHE_UNPIN, - 0, 0); - if (flush_pagecache_blocks(&pagecache, &file1, FLUSH_FORCE_WRITE)) - { - diag("Got error in flush_pagecache_blocks\n"); - res= 0; - goto err; - } - ok((res&= test(test_file(file1, file1_name, PAGE_SIZE*2, PAGE_SIZE, - simple_pin_test_file2))), - "Simple pin page result file"); - if (res) - reset_file(file1, file1_name); -err: - free(buffw); - free(buffr); - DBUG_RETURN(res); -} - -/* - Prepare page, write new value, then delete page from cache without flush, - on the disk should be page with old content written during preparation -*/ - -int simple_delete_forget_test() -{ - unsigned char *buffw= malloc(PAGE_SIZE); - unsigned char *buffr= malloc(PAGE_SIZE); - int res; - DBUG_ENTER("simple_delete_forget_test"); - /* prepare the file */ - memset(buffw, '\1', PAGE_SIZE); - pagecache_write(&pagecache, &file1, 0, 3, (char*)buffw, - PAGECACHE_PLAIN_PAGE, - PAGECACHE_LOCK_LEFT_UNLOCKED, - PAGECACHE_PIN_LEFT_UNPINNED, - PAGECACHE_WRITE_DELAY, - 0); - flush_pagecache_blocks(&pagecache, &file1, FLUSH_FORCE_WRITE); - /* test */ - memset(buffw, '\2', PAGE_SIZE); - pagecache_write(&pagecache, &file1, 0, 3, (char*)buffw, - PAGECACHE_PLAIN_PAGE, - PAGECACHE_LOCK_LEFT_UNLOCKED, - PAGECACHE_PIN_LEFT_UNPINNED, - PAGECACHE_WRITE_DELAY, - 0); - pagecache_delete_page(&pagecache, &file1, 0, - PAGECACHE_LOCK_WRITE, 0); - flush_pagecache_blocks(&pagecache, &file1, FLUSH_FORCE_WRITE); - ok((res= test(test_file(file1, file1_name, PAGE_SIZE, PAGE_SIZE, - simple_delete_forget_test_file))), - "Simple delete-forget page file"); - if (res) - reset_file(file1, file1_name); - free(buffw); - free(buffr); - DBUG_RETURN(res); -} - -/* - Prepare page with locking, write new content to the page, - delete page with flush and on existing lock, - check that page on disk contain new value. -*/ - -int simple_delete_flush_test() -{ - unsigned char *buffw= malloc(PAGE_SIZE); - unsigned char *buffr= malloc(PAGE_SIZE); - int res; - DBUG_ENTER("simple_delete_flush_test"); - /* prepare the file */ - memset(buffw, '\1', PAGE_SIZE); - pagecache_write(&pagecache, &file1, 0, 3, (char*)buffw, - PAGECACHE_PLAIN_PAGE, - PAGECACHE_LOCK_WRITE, - PAGECACHE_PIN, - PAGECACHE_WRITE_DELAY, - 0); - flush_pagecache_blocks(&pagecache, &file1, FLUSH_FORCE_WRITE); - /* test */ - memset(buffw, '\2', PAGE_SIZE); - pagecache_write(&pagecache, &file1, 0, 3, (char*)buffw, - PAGECACHE_PLAIN_PAGE, - PAGECACHE_LOCK_LEFT_WRITELOCKED, - PAGECACHE_PIN_LEFT_PINNED, - PAGECACHE_WRITE_DELAY, - 0); - pagecache_delete_page(&pagecache, &file1, 0, - PAGECACHE_LOCK_LEFT_WRITELOCKED, 1); - flush_pagecache_blocks(&pagecache, &file1, FLUSH_FORCE_WRITE); - ok((res= test(test_file(file1, file1_name, PAGE_SIZE, PAGE_SIZE, - simple_delete_flush_test_file))), - "Simple delete-forget page file"); - if (res) - reset_file(file1, file1_name); - free(buffw); - free(buffr); - DBUG_RETURN(res); -} - - -/* - write then read file bigger then cache -*/ - -int simple_big_test() -{ - unsigned char *buffw= (unsigned char *)malloc(PAGE_SIZE); - unsigned char *buffr= (unsigned char *)malloc(PAGE_SIZE); - struct file_desc *desc= - (struct file_desc *)malloc((PCACHE_SIZE/(PAGE_SIZE/2)) * - sizeof(struct file_desc)); - int res, i; - DBUG_ENTER("simple_big_test"); - /* prepare the file twice larger then cache */ - for (i= 0; i < PCACHE_SIZE/(PAGE_SIZE/2); i++) - { - memset(buffw, (unsigned char) (i & 0xff), PAGE_SIZE); - desc[i].length= PAGE_SIZE; - desc[i].content= (i & 0xff); - pagecache_write(&pagecache, &file1, i, 3, (char*)buffw, - PAGECACHE_PLAIN_PAGE, - PAGECACHE_LOCK_LEFT_UNLOCKED, - PAGECACHE_PIN_LEFT_UNPINNED, - PAGECACHE_WRITE_DELAY, - 0); - } - ok(1, "Simple big file write"); - /* check written pages sequentally read */ - for (i= 0; i < PCACHE_SIZE/(PAGE_SIZE/2); i++) - { - int j; - pagecache_read(&pagecache, &file1, i, 3, (char*)buffr, - PAGECACHE_PLAIN_PAGE, - PAGECACHE_LOCK_LEFT_UNLOCKED, - 0); - for(j= 0; j < PAGE_SIZE; j++) - { - if (buffr[j] != (i & 0xff)) - { - diag("simple_big_test seq: page %u byte %u mismatch\n", i, j); - return 0; - } - } - } - ok(1, "simple big file sequentally read"); - /* chack random reads */ - for (i= 0; i < PCACHE_SIZE/(PAGE_SIZE); i++) - { - int j, page; - page= rand() % (PCACHE_SIZE/(PAGE_SIZE/2)); - pagecache_read(&pagecache, &file1, page, 3, (char*)buffr, - PAGECACHE_PLAIN_PAGE, - PAGECACHE_LOCK_LEFT_UNLOCKED, - 0); - for(j= 0; j < PAGE_SIZE; j++) - { - if (buffr[j] != (page & 0xff)) - { - diag("simple_big_test rnd: page %u byte %u mismatch\n", page, j); - return 0; - } - } - } - ok(1, "simple big file random read"); - flush_pagecache_blocks(&pagecache, &file1, FLUSH_FORCE_WRITE); - - ok((res= test(test_file(file1, file1_name, PCACHE_SIZE*2, PAGE_SIZE, - desc))), - "Simple big file"); - if (res) - reset_file(file1, file1_name); - free(buffw); - free(buffr); - DBUG_RETURN(res); -} -/* - Thread function -*/ - -static void *test_thread(void *arg) -{ - int param=*((int*) arg); - - my_thread_init(); - DBUG_ENTER("test_thread"); - - DBUG_PRINT("enter", ("param: %d", param)); - - if (!simple_read_write_test() || - !simple_read_change_write_read_test() || - !simple_pin_test() || - !simple_delete_forget_test() || - !simple_delete_flush_test() || - !simple_big_test()) - exit(1); - - DBUG_PRINT("info", ("Thread %s ended\n", my_thread_name())); - pthread_mutex_lock(&LOCK_thread_count); - thread_count--; - VOID(pthread_cond_signal(&COND_thread_count)); /* Tell main we are ready */ - pthread_mutex_unlock(&LOCK_thread_count); - free((gptr) arg); - my_thread_end(); - DBUG_RETURN(0); -} - - -int main(int argc, char **argv __attribute__((unused))) -{ - pthread_t tid; - pthread_attr_t thr_attr; - int *param, error, pagen; - - MY_INIT(argv[0]); - -#ifndef DBUG_OFF -#if defined(__WIN__) - default_dbug_option= "d:t:i:O,\\test_pagecache_single.trace"; -#else - default_dbug_option= "d:t:i:o,/tmp/test_pagecache_single.trace"; -#endif - if (argc > 1) - { - DBUG_SET(default_dbug_option); - DBUG_SET_INITIAL(default_dbug_option); - } -#endif - - - DBUG_ENTER("main"); - DBUG_PRINT("info", ("Main thread: %s\n", my_thread_name())); - if ((file1.file= my_open(file1_name, - O_CREAT | O_TRUNC | O_RDWR, MYF(0))) == -1) - { - fprintf(stderr, "Got error during file1 creation from open() (errno: %d)\n", - errno); - exit(1); - } - DBUG_PRINT("info", ("file1: %d", file1.file)); - if (chmod(file1_name, S_IRWXU | S_IRWXG | S_IRWXO) != 0) - { - fprintf(stderr, "Got error during file1 chmod() (errno: %d)\n", - errno); - exit(1); - } - my_pwrite(file1.file, "test file", 9, 0, MYF(0)); - - if ((error= pthread_cond_init(&COND_thread_count, NULL))) - { - fprintf(stderr, "Got error: %d from pthread_cond_init (errno: %d)\n", - error, errno); - exit(1); - } - if ((error= pthread_mutex_init(&LOCK_thread_count, MY_MUTEX_INIT_FAST))) - { - fprintf(stderr, "Got error: %d from pthread_cond_init (errno: %d)\n", - error, errno); - exit(1); - } - - if ((error= pthread_attr_init(&thr_attr))) - { - fprintf(stderr,"Got error: %d from pthread_attr_init (errno: %d)\n", - error,errno); - exit(1); - } - if ((error= pthread_attr_setdetachstate(&thr_attr, PTHREAD_CREATE_DETACHED))) - { - fprintf(stderr, - "Got error: %d from pthread_attr_setdetachstate (errno: %d)\n", - error,errno); - exit(1); - } - -#ifndef pthread_attr_setstacksize /* void return value */ - if ((error= pthread_attr_setstacksize(&thr_attr, 65536L))) - { - fprintf(stderr,"Got error: %d from pthread_attr_setstacksize (errno: %d)\n", - error,errno); - exit(1); - } -#endif -#ifdef HAVE_THR_SETCONCURRENCY - VOID(thr_setconcurrency(2)); -#endif - - my_thread_global_init(); - - - if ((pagen= init_pagecache(&pagecache, PCACHE_SIZE, 0, 0, - PAGE_SIZE, 0)) == 0) - { - fprintf(stderr,"Got error: init_pagecache() (errno: %d)\n", - errno); - exit(1); - } - DBUG_PRINT("info", ("Page cache %d pages", pagen)); - - if ((error=pthread_mutex_lock(&LOCK_thread_count))) - { - fprintf(stderr,"Got error: %d from pthread_mutex_lock (errno: %d)\n", - error,errno); - exit(1); - } - param=(int*) malloc(sizeof(int)); - *param= 1; - if ((error= pthread_create(&tid, &thr_attr, test_thread, (void*) param))) - { - fprintf(stderr,"Got error: %d from pthread_create (errno: %d)\n", - error,errno); - exit(1); - } - thread_count++; - DBUG_PRINT("info", ("Thread started")); - pthread_mutex_unlock(&LOCK_thread_count); - - pthread_attr_destroy(&thr_attr); - - if ((error= pthread_mutex_lock(&LOCK_thread_count))) - fprintf(stderr,"Got error: %d from pthread_mutex_lock\n",error); - while (thread_count) - { - if ((error= pthread_cond_wait(&COND_thread_count,&LOCK_thread_count))) - fprintf(stderr,"Got error: %d from pthread_cond_wait\n",error); - } - if ((error= pthread_mutex_unlock(&LOCK_thread_count))) - fprintf(stderr,"Got error: %d from pthread_mutex_unlock\n",error); - DBUG_PRINT("info", ("thread ended")); - - end_pagecache(&pagecache, 1); - DBUG_PRINT("info", ("Page cache ended")); - - if (my_close(file1.file, MYF(0)) != 0) - { - fprintf(stderr, "Got error during file1 closing from close() (errno: %d)\n", - errno); - exit(1); - } - /*my_delete(file1_name, MYF(0));*/ - my_thread_global_end(); - - DBUG_PRINT("info", ("file1 (%d) closed", file1.file)); - - DBUG_PRINT("info", ("Program end")); - - DBUG_RETURN(0); -} diff --git a/unittest/mysys/Makefile.am b/unittest/mysys/Makefile.am index 8e6255b9e68..9b230272329 100644 --- a/unittest/mysys/Makefile.am +++ b/unittest/mysys/Makefile.am @@ -1,11 +1,51 @@ +INCLUDES = @ZLIB_INCLUDES@ -I$(top_builddir)/include \ + -I$(top_srcdir)/include -I$(top_srcdir)/unittest/mytap -AM_CPPFLAGS = @ZLIB_INCLUDES@ -I$(top_builddir)/include -AM_CPPFLAGS += -I$(top_srcdir)/include -I$(top_srcdir)/unittest/mytap LDADD = $(top_builddir)/unittest/mytap/libmytap.a \ $(top_builddir)/mysys/libmysys.a \ $(top_builddir)/dbug/libdbug.a \ $(top_builddir)/strings/libmystrings.a -noinst_PROGRAMS = bitmap-t base64-t my_atomic-t +noinst_PROGRAMS = bitmap-t base64-t my_atomic-t \ + mf_pagecache_single_1k-t mf_pagecache_single_8k-t \ + mf_pagecache_single_64k-t \ + mf_pagecache_consist_1k-t mf_pagecache_consist_64k-t \ + mf_pagecache_consist_1kHC-t mf_pagecache_consist_64kHC-t \ + mf_pagecache_consist_1kRD-t mf_pagecache_consist_64kRD-t \ + mf_pagecache_consist_1kWR-t mf_pagecache_consist_64kWR-t +# tests for mysys/mf_pagecache.c + +mf_pagecache_single_src = mf_pagecache_single.c $(top_srcdir)/mysys/mf_pagecache.c test_file.c +mf_pagecache_consist_src = mf_pagecache_consist.c $(top_srcdir)/mysys/mf_pagecache.c test_file.c +mf_pagecache_common_cppflags = -DEXTRA_DEBUG -DPAGECACHE_DEBUG -DMAIN + +mf_pagecache_single_1k_t_SOURCES = $(mf_pagecache_single_src) +mf_pagecache_single_8k_t_SOURCES = $(mf_pagecache_single_src) +mf_pagecache_single_64k_t_SOURCES = $(mf_pagecache_single_src) +mf_pagecache_single_1k_t_CPPFLAGS = $(mf_pagecache_common_cppflags) -DPAGE_SIZE=1024 +mf_pagecache_single_8k_t_CPPFLAGS = $(mf_pagecache_common_cppflags) -DPAGE_SIZE=8192 +mf_pagecache_single_64k_t_CPPFLAGS = $(mf_pagecache_common_cppflags) -DPAGE_SIZE=65536 + +mf_pagecache_consist_1k_t_SOURCES = $(mf_pagecache_consist_src) +mf_pagecache_consist_1k_t_CPPFLAGS = $(mf_pagecache_common_cppflags) -DPAGE_SIZE=1024 +mf_pagecache_consist_64k_t_SOURCES = $(mf_pagecache_consist_src) +mf_pagecache_consist_64k_t_CPPFLAGS = $(mf_pagecache_common_cppflags) -DPAGE_SIZE=65536 + +mf_pagecache_consist_1kHC_t_SOURCES = $(mf_pagecache_consist_src) +mf_pagecache_consist_1kHC_t_CPPFLAGS = $(mf_pagecache_common_cppflags) -DPAGE_SIZE=1024 -DTEST_HIGH_CONCURENCY +mf_pagecache_consist_64kHC_t_SOURCES = $(mf_pagecache_consist_src) +mf_pagecache_consist_64kHC_t_CPPFLAGS = $(mf_pagecache_common_cppflags) -DPAGE_SIZE=65536 -DTEST_HIGH_CONCURENCY + +mf_pagecache_consist_1kRD_t_SOURCES = $(mf_pagecache_consist_src) +mf_pagecache_consist_1kRD_t_CPPFLAGS = $(mf_pagecache_common_cppflags) -DPAGE_SIZE=1024 -DTEST_READERS +mf_pagecache_consist_64kRD_t_SOURCES = $(mf_pagecache_consist_src) +mf_pagecache_consist_64kRD_t_CPPFLAGS = $(mf_pagecache_common_cppflags) -DPAGE_SIZE=65536 -DTEST_READERS + +mf_pagecache_consist_1kWR_t_SOURCES = $(mf_pagecache_consist_src) +mf_pagecache_consist_1kWR_t_CPPFLAGS = $(mf_pagecache_common_cppflags) -DPAGE_SIZE=1024 -DTEST_WRITERS +mf_pagecache_consist_64kWR_t_SOURCES = $(mf_pagecache_consist_src) +mf_pagecache_consist_64kWR_t_CPPFLAGS = $(mf_pagecache_common_cppflags) -DPAGE_SIZE=65536 -DTEST_WRITERS + +CLEANFILES = my_pagecache_debug.log page_cache_test_file_1 diff --git a/unittest/mysys/mf_pagecache_consist.c b/unittest/mysys/mf_pagecache_consist.c new file mode 100755 index 00000000000..f7724222a09 --- /dev/null +++ b/unittest/mysys/mf_pagecache_consist.c @@ -0,0 +1,445 @@ +/* + TODO: use pthread_join instead of wait_for_thread_count_to_be_zero, like in + my_atomic-t.c (see BUG#22320). + Use diag() instead of fprintf(stderr). Use ok() and plan(). +*/ + +#include +#include +#include +#include "test_file.h" + +#define PCACHE_SIZE (PAGE_SIZE*1024*8) + +#ifndef DBUG_OFF +static const char* default_dbug_option; +#endif + +static char *file1_name= (char*)"page_cache_test_file_1"; +static PAGECACHE_FILE file1; +static pthread_cond_t COND_thread_count; +static pthread_mutex_t LOCK_thread_count; +static uint thread_count; +static PAGECACHE pagecache; + +#ifdef TEST_HIGH_CONCURENCY +static uint number_of_readers= 10; +static uint number_of_writers= 20; +static uint number_of_tests= 30000; +static uint record_length_limit= PAGE_SIZE/200; +static uint number_of_pages= 20; +static uint flush_divider= 1000; +#else /*TEST_HIGH_CONCURENCY*/ +#ifdef TEST_READERS +static uint number_of_readers= 10; +static uint number_of_writers= 1; +static uint number_of_tests= 30000; +static uint record_length_limit= PAGE_SIZE/200; +static uint number_of_pages= 20; +static uint flush_divider= 1000; +#else /*TEST_READERS*/ +#ifdef TEST_WRITERS +static uint number_of_readers= 0; +static uint number_of_writers= 10; +static uint number_of_tests= 30000; +static uint record_length_limit= PAGE_SIZE/200; +static uint number_of_pages= 20; +static uint flush_divider= 1000; +#else /*TEST_WRITERS*/ +static uint number_of_readers= 10; +static uint number_of_writers= 10; +static uint number_of_tests= 50000; +static uint record_length_limit= PAGE_SIZE/200; +static uint number_of_pages= 20000; +static uint flush_divider= 1000; +#endif /*TEST_WRITERS*/ +#endif /*TEST_READERS*/ +#endif /*TEST_HIGH_CONCURENCY*/ + + +/* check page consistency */ +uint check_page(uchar *buff, ulong offset, int page_locked, int page_no, + int tag) +{ + uint end= sizeof(uint); + uint num= *((uint *)buff); + uint i; + DBUG_ENTER("check_page"); + + for (i= 0; i < num; i++) + { + uint len= *((uint *)(buff + end)); + uint j; + end+= sizeof(uint)+ sizeof(uint); + if (len + end > PAGE_SIZE) + { + diag("incorrect field header #%u by offset %lu\n", i, offset + end + j); + goto err; + } + for(j= 0; j < len; j++) + { + if (buff[end + j] != (uchar)((i+1) % 256)) + { + diag("incorrect %lu byte\n", offset + end + j); + goto err; + } + } + end+= len; + } + for(i= end; i < PAGE_SIZE; i++) + { + if (buff[i] != 0) + { + int h; + DBUG_PRINT("err", + ("byte %lu (%lu + %u), page %u (%s, end: %u, recs: %u, tag: %d) should be 0\n", + offset + i, offset, i, page_no, + (page_locked ? "locked" : "unlocked"), + end, num, tag)); + diag("byte %lu (%lu + %u), page %u (%s, end: %u, recs: %u, tag: %d) should be 0\n", + offset + i, offset, i, page_no, + (page_locked ? "locked" : "unlocked"), + end, num, tag); + h= my_open("wrong_page", O_CREAT | O_TRUNC | O_RDWR, MYF(0)); + my_pwrite(h, buff, PAGE_SIZE, 0, MYF(0)); + my_close(h, MYF(0)); + goto err; + } + } + DBUG_RETURN(end); +err: + DBUG_PRINT("err", ("try to flush")); + if (page_locked) + { + pagecache_delete_page(&pagecache, &file1, page_no, + PAGECACHE_LOCK_LEFT_WRITELOCKED, 1); + } + else + { + flush_pagecache_blocks(&pagecache, &file1, FLUSH_RELEASE); + } + exit(1); +} + +void put_rec(uchar *buff, uint end, uint len, uint tag) +{ + uint i; + uint num= *((uint *)buff); + if (!len) + len= 1; + if (end + sizeof(uint)*2 + len > PAGE_SIZE) + return; + *((uint *)(buff + end))= len; + end+= sizeof(uint); + *((uint *)(buff + end))= tag; + end+= sizeof(uint); + num++; + *((uint *)buff)= num; + *((uint*)(buff + end))= len; + for (i= end; i < (len + end); i++) + { + buff[i]= (uchar) num % 256; + } +} + +/* + Recreate and reopen a file for test + + SYNOPSIS + reset_file() + file File to reset + file_name Path (and name) of file which should be reset +*/ + +void reset_file(PAGECACHE_FILE file, char *file_name) +{ + flush_pagecache_blocks(&pagecache, &file1, FLUSH_RELEASE); + if (my_close(file1.file, MYF(0)) != 0) + { + diag("Got error during %s closing from close() (errno: %d)\n", + file_name, errno); + exit(1); + } + my_delete(file_name, MYF(0)); + if ((file.file= my_open(file_name, + O_CREAT | O_TRUNC | O_RDWR, MYF(0))) == -1) + { + diag("Got error during %s creation from open() (errno: %d)\n", + file_name, errno); + exit(1); + } +} + + +void reader(int num) +{ + unsigned char *buffr= malloc(PAGE_SIZE); + uint i; + + for (i= 0; i < number_of_tests; i++) + { + uint page= rand()/(RAND_MAX/number_of_pages); + pagecache_read(&pagecache, &file1, page, 3, (char*)buffr, + PAGECACHE_PLAIN_PAGE, + PAGECACHE_LOCK_LEFT_UNLOCKED, + 0); + check_page(buffr, page * PAGE_SIZE, 0, page, -num); + if (i % 500 == 0) + printf("reader%d: %d\n", num, i); + + } + printf("reader%d: done\n", num); + free(buffr); +} + + +void writer(int num) +{ + unsigned char *buffr= malloc(PAGE_SIZE); + uint i; + + for (i= 0; i < number_of_tests; i++) + { + uint end; + uint page= rand()/(RAND_MAX/number_of_pages); + pagecache_read(&pagecache, &file1, page, 3, (char*)buffr, + PAGECACHE_PLAIN_PAGE, + PAGECACHE_LOCK_WRITE, + 0); + end= check_page(buffr, page * PAGE_SIZE, 1, page, num); + put_rec(buffr, end, rand()/(RAND_MAX/record_length_limit), num); + pagecache_write(&pagecache, &file1, page, 3, (char*)buffr, + PAGECACHE_PLAIN_PAGE, + PAGECACHE_LOCK_WRITE_UNLOCK, + PAGECACHE_UNPIN, + PAGECACHE_WRITE_DELAY, + 0); + + if (i % flush_divider == 0) + flush_pagecache_blocks(&pagecache, &file1, FLUSH_FORCE_WRITE); + if (i % 500 == 0) + printf("writer%d: %d\n", num, i); + } + printf("writer%d: done\n", num); + free(buffr); +} + + +static void *test_thread_reader(void *arg) +{ + int param=*((int*) arg); + + my_thread_init(); + DBUG_ENTER("test_reader"); + DBUG_PRINT("enter", ("param: %d", param)); + + reader(param); + + DBUG_PRINT("info", ("Thread %s ended\n", my_thread_name())); + pthread_mutex_lock(&LOCK_thread_count); + thread_count--; + VOID(pthread_cond_signal(&COND_thread_count)); /* Tell main we are ready */ + pthread_mutex_unlock(&LOCK_thread_count); + free((gptr) arg); + my_thread_end(); + DBUG_RETURN(0); +} + +static void *test_thread_writer(void *arg) +{ + int param=*((int*) arg); + + my_thread_init(); + DBUG_ENTER("test_writer"); + DBUG_PRINT("enter", ("param: %d", param)); + + writer(param); + + DBUG_PRINT("info", ("Thread %s ended\n", my_thread_name())); + pthread_mutex_lock(&LOCK_thread_count); + thread_count--; + VOID(pthread_cond_signal(&COND_thread_count)); /* Tell main we are ready */ + pthread_mutex_unlock(&LOCK_thread_count); + free((gptr) arg); + my_thread_end(); + DBUG_RETURN(0); +} + +int main(int argc, char **argv __attribute__((unused))) +{ + pthread_t tid; + pthread_attr_t thr_attr; + int *param, error, pagen; + + MY_INIT(argv[0]); + +#ifndef DBUG_OFF +#if defined(__WIN__) + default_dbug_option= "d:t:i:O,\\test_pagecache_consist.trace"; +#else + default_dbug_option= "d:t:i:o,/tmp/test_pagecache_consist.trace"; +#endif + if (argc > 1) + { + DBUG_SET(default_dbug_option); + DBUG_SET_INITIAL(default_dbug_option); + } +#endif + + + DBUG_ENTER("main"); + DBUG_PRINT("info", ("Main thread: %s\n", my_thread_name())); + if ((file1.file= my_open(file1_name, + O_CREAT | O_TRUNC | O_RDWR, MYF(0))) == -1) + { + fprintf(stderr, "Got error during file1 creation from open() (errno: %d)\n", + errno); + exit(1); + } + DBUG_PRINT("info", ("file1: %d", file1.file)); + if (chmod(file1_name, S_IRWXU | S_IRWXG | S_IRWXO) != 0) + { + fprintf(stderr, "Got error during file1 chmod() (errno: %d)\n", + errno); + exit(1); + } + my_pwrite(file1.file, "test file", 9, 0, MYF(0)); + + if ((error= pthread_cond_init(&COND_thread_count, NULL))) + { + fprintf(stderr, "COND_thread_count: %d from pthread_cond_init (errno: %d)\n", + error, errno); + exit(1); + } + if ((error= pthread_mutex_init(&LOCK_thread_count, MY_MUTEX_INIT_FAST))) + { + fprintf(stderr, "LOCK_thread_count: %d from pthread_cond_init (errno: %d)\n", + error, errno); + exit(1); + } + + if ((error= pthread_attr_init(&thr_attr))) + { + fprintf(stderr,"Got error: %d from pthread_attr_init (errno: %d)\n", + error,errno); + exit(1); + } + if ((error= pthread_attr_setdetachstate(&thr_attr, PTHREAD_CREATE_DETACHED))) + { + fprintf(stderr, + "Got error: %d from pthread_attr_setdetachstate (errno: %d)\n", + error,errno); + exit(1); + } + +#ifndef pthread_attr_setstacksize /* void return value */ + if ((error= pthread_attr_setstacksize(&thr_attr, 65536L))) + { + fprintf(stderr,"Got error: %d from pthread_attr_setstacksize (errno: %d)\n", + error,errno); + exit(1); + } +#endif +#ifdef HAVE_THR_SETCONCURRENCY + VOID(thr_setconcurrency(2)); +#endif + + my_thread_global_init(); + + + if ((pagen= init_pagecache(&pagecache, PCACHE_SIZE, 0, 0, + PAGE_SIZE, 0)) == 0) + { + fprintf(stderr,"Got error: init_pagecache() (errno: %d)\n", + errno); + exit(1); + } + DBUG_PRINT("info", ("Page cache %d pages", pagen)); + { + unsigned char *buffr= malloc(PAGE_SIZE); + uint i; + memset(buffr, '\0', PAGE_SIZE); + for (i= 0; i < number_of_pages; i++) + { + pagecache_write(&pagecache, &file1, i, 3, (char*)buffr, + PAGECACHE_PLAIN_PAGE, + PAGECACHE_LOCK_LEFT_UNLOCKED, + PAGECACHE_PIN_LEFT_UNPINNED, + PAGECACHE_WRITE_DELAY, + 0); + } + flush_pagecache_blocks(&pagecache, &file1, FLUSH_FORCE_WRITE); + free(buffr); + } + if ((error= pthread_mutex_lock(&LOCK_thread_count))) + { + fprintf(stderr,"LOCK_thread_count: %d from pthread_mutex_lock (errno: %d)\n", + error,errno); + exit(1); + } + while (number_of_readers != 0 || number_of_writers != 0) + { + if (number_of_readers != 0) + { + param=(int*) malloc(sizeof(int)); + *param= number_of_readers; + if ((error= pthread_create(&tid, &thr_attr, test_thread_reader, + (void*) param))) + { + fprintf(stderr,"Got error: %d from pthread_create (errno: %d)\n", + error,errno); + exit(1); + } + thread_count++; + number_of_readers--; + } + if (number_of_writers != 0) + { + param=(int*) malloc(sizeof(int)); + *param= number_of_writers; + if ((error= pthread_create(&tid, &thr_attr, test_thread_writer, + (void*) param))) + { + fprintf(stderr,"Got error: %d from pthread_create (errno: %d)\n", + error,errno); + exit(1); + } + thread_count++; + number_of_writers--; + } + } + DBUG_PRINT("info", ("Thread started")); + pthread_mutex_unlock(&LOCK_thread_count); + + pthread_attr_destroy(&thr_attr); + + /* wait finishing */ + if ((error= pthread_mutex_lock(&LOCK_thread_count))) + fprintf(stderr,"LOCK_thread_count: %d from pthread_mutex_lock\n",error); + while (thread_count) + { + if ((error= pthread_cond_wait(&COND_thread_count,&LOCK_thread_count))) + fprintf(stderr,"COND_thread_count: %d from pthread_cond_wait\n",error); + } + if ((error= pthread_mutex_unlock(&LOCK_thread_count))) + fprintf(stderr,"LOCK_thread_count: %d from pthread_mutex_unlock\n",error); + DBUG_PRINT("info", ("thread ended")); + + end_pagecache(&pagecache, 1); + DBUG_PRINT("info", ("Page cache ended")); + + if (my_close(file1.file, MYF(0)) != 0) + { + fprintf(stderr, "Got error during file1 closing from close() (errno: %d)\n", + errno); + exit(1); + } + /*my_delete(file1_name, MYF(0));*/ + my_thread_global_end(); + + DBUG_PRINT("info", ("file1 (%d) closed", file1.file)); + + DBUG_PRINT("info", ("Program end")); + + DBUG_RETURN(exit_status()); +} diff --git a/unittest/mysys/mf_pagecache_single.c b/unittest/mysys/mf_pagecache_single.c new file mode 100644 index 00000000000..49ecd2986ab --- /dev/null +++ b/unittest/mysys/mf_pagecache_single.c @@ -0,0 +1,587 @@ +/* + TODO: use pthread_join instead of wait_for_thread_count_to_be_zero, like in + my_atomic-t.c (see BUG#22320). + Use diag() instead of fprintf(stderr). +*/ +#include +#include +#include +#include "test_file.h" + +#define PCACHE_SIZE (PAGE_SIZE*1024*10) + +#ifndef DBUG_OFF +static const char* default_dbug_option; +#endif + +static char *file1_name= (char*)"page_cache_test_file_1"; +static PAGECACHE_FILE file1; +static pthread_cond_t COND_thread_count; +static pthread_mutex_t LOCK_thread_count; +static uint thread_count; +static PAGECACHE pagecache; + +/* + File contance descriptors +*/ +static struct file_desc simple_read_write_test_file[]= +{ + {PAGE_SIZE, '\1'}, + { 0, 0} +}; +static struct file_desc simple_read_change_write_read_test_file[]= +{ + {PAGE_SIZE/2, '\65'}, + {PAGE_SIZE/2, '\1'}, + { 0, 0} +}; +static struct file_desc simple_pin_test_file1[]= +{ + {PAGE_SIZE*2, '\1'}, + { 0, 0} +}; +static struct file_desc simple_pin_test_file2[]= +{ + {PAGE_SIZE/2, '\1'}, + {PAGE_SIZE/2, (unsigned char)129}, + {PAGE_SIZE, '\1'}, + { 0, 0} +}; +static struct file_desc simple_delete_forget_test_file[]= +{ + {PAGE_SIZE, '\1'}, + { 0, 0} +}; +static struct file_desc simple_delete_flush_test_file[]= +{ + {PAGE_SIZE, '\2'}, + { 0, 0} +}; + + +/* + Recreate and reopen a file for test + + SYNOPSIS + reset_file() + file File to reset + file_name Path (and name) of file which should be reset +*/ + +void reset_file(PAGECACHE_FILE file, char *file_name) +{ + flush_pagecache_blocks(&pagecache, &file1, FLUSH_RELEASE); + if (my_close(file1.file, MYF(0)) != 0) + { + diag("Got error during %s closing from close() (errno: %d)\n", + file_name, errno); + exit(1); + } + my_delete(file_name, MYF(0)); + if ((file.file= my_open(file_name, + O_CREAT | O_TRUNC | O_RDWR, MYF(0))) == -1) + { + diag("Got error during %s creation from open() (errno: %d)\n", + file_name, errno); + exit(1); + } +} + +/* + Write then read page, check file on disk +*/ + +int simple_read_write_test() +{ + unsigned char *buffw= malloc(PAGE_SIZE); + unsigned char *buffr= malloc(PAGE_SIZE); + int res; + DBUG_ENTER("simple_read_write_test"); + bfill(buffw, PAGE_SIZE, '\1'); + pagecache_write(&pagecache, &file1, 0, 3, (char*)buffw, + PAGECACHE_PLAIN_PAGE, + PAGECACHE_LOCK_LEFT_UNLOCKED, + PAGECACHE_PIN_LEFT_UNPINNED, + PAGECACHE_WRITE_DELAY, + 0); + pagecache_read(&pagecache, &file1, 0, 3, (char*)buffr, + PAGECACHE_PLAIN_PAGE, + PAGECACHE_LOCK_LEFT_UNLOCKED, + 0); + ok((res= test(memcmp(buffr, buffw, PAGE_SIZE) == 0)), + "Simple write-read page "); + flush_pagecache_blocks(&pagecache, &file1, FLUSH_FORCE_WRITE); + ok((res&= test(test_file(file1, file1_name, PAGE_SIZE, PAGE_SIZE, + simple_read_write_test_file))), + "Simple write-read page file"); + if (res) + reset_file(file1, file1_name); + free(buffw); + free(buffr); + DBUG_RETURN(res); +} + + +/* + Prepare page, then read (and lock), change (write new value and unlock), + then check the page in the cache and on the disk +*/ +int simple_read_change_write_read_test() +{ + unsigned char *buffw= malloc(PAGE_SIZE); + unsigned char *buffr= malloc(PAGE_SIZE); + int res; + DBUG_ENTER("simple_read_change_write_read_test"); + /* prepare the file */ + bfill(buffw, PAGE_SIZE, '\1'); + pagecache_write(&pagecache, &file1, 0, 3, (char*)buffw, + PAGECACHE_PLAIN_PAGE, + PAGECACHE_LOCK_LEFT_UNLOCKED, + PAGECACHE_PIN_LEFT_UNPINNED, + PAGECACHE_WRITE_DELAY, + 0); + flush_pagecache_blocks(&pagecache, &file1, FLUSH_FORCE_WRITE); + /* test */ + pagecache_read(&pagecache, &file1, 0, 3, (char*)buffw, + PAGECACHE_PLAIN_PAGE, + PAGECACHE_LOCK_WRITE, + 0); + bfill(buffw, PAGE_SIZE/2, '\65'); + pagecache_write(&pagecache, &file1, 0, 3, (char*)buffw, + PAGECACHE_PLAIN_PAGE, + PAGECACHE_LOCK_WRITE_UNLOCK, + PAGECACHE_UNPIN, + PAGECACHE_WRITE_DELAY, + 0); + + pagecache_read(&pagecache, &file1, 0, 3, (char*)buffr, + PAGECACHE_PLAIN_PAGE, + PAGECACHE_LOCK_LEFT_UNLOCKED, + 0); + ok((res= test(memcmp(buffr, buffw, PAGE_SIZE) == 0)), + "Simple read-change-write-read page "); + flush_pagecache_blocks(&pagecache, &file1, FLUSH_FORCE_WRITE); + ok((res&= test(test_file(file1, file1_name, PAGE_SIZE, PAGE_SIZE, + simple_read_change_write_read_test_file))), + "Simple read-change-write-read page file"); + if (res) + reset_file(file1, file1_name); + free(buffw); + free(buffr); + DBUG_RETURN(res); +} + + +/* + Prepare page, read page 0 (and pin) then write page 1 and page 0. + Flush the file (shold flush only page 1 and return 1 (page 0 is + still pinned). + Check file on the disk. + Unpin and flush. + Check file on the disk. +*/ +int simple_pin_test() +{ + unsigned char *buffw= malloc(PAGE_SIZE); + unsigned char *buffr= malloc(PAGE_SIZE); + int res; + DBUG_ENTER("simple_pin_test"); + /* prepare the file */ + bfill(buffw, PAGE_SIZE, '\1'); + pagecache_write(&pagecache, &file1, 0, 3, (char*)buffw, + PAGECACHE_PLAIN_PAGE, + PAGECACHE_LOCK_LEFT_UNLOCKED, + PAGECACHE_PIN_LEFT_UNPINNED, + PAGECACHE_WRITE_DELAY, + 0); + /* test */ + if (flush_pagecache_blocks(&pagecache, &file1, FLUSH_FORCE_WRITE)) + { + diag("error in flush_pagecache_blocks\n"); + exit(1); + } + pagecache_read(&pagecache, &file1, 0, 3, (char*)buffw, + PAGECACHE_PLAIN_PAGE, + PAGECACHE_LOCK_WRITE, + 0); + pagecache_write(&pagecache, &file1, 1, 3, (char*)buffw, + PAGECACHE_PLAIN_PAGE, + PAGECACHE_LOCK_LEFT_UNLOCKED, + PAGECACHE_PIN_LEFT_UNPINNED, + PAGECACHE_WRITE_DELAY, + 0); + bfill(buffw + PAGE_SIZE/2, PAGE_SIZE/2, ((unsigned char) 129)); + pagecache_write(&pagecache, &file1, 0, 3, (char*)buffw, + PAGECACHE_PLAIN_PAGE, + PAGECACHE_LOCK_WRITE_TO_READ, + PAGECACHE_PIN_LEFT_PINNED, + PAGECACHE_WRITE_DELAY, + 0); + /* + We have to get error because one page of the file is pinned, + other page should be flushed + */ + if (!flush_pagecache_blocks(&pagecache, &file1, FLUSH_FORCE_WRITE)) + { + diag("Did not get error in flush_pagecache_blocks\n"); + res= 0; + goto err; + } + ok((res= test(test_file(file1, file1_name, PAGE_SIZE*2, PAGE_SIZE*2, + simple_pin_test_file1))), + "Simple pin page file with pin"); + pagecache_unlock_page(&pagecache, + &file1, + 0, + PAGECACHE_LOCK_READ_UNLOCK, + PAGECACHE_UNPIN, + 0, 0); + if (flush_pagecache_blocks(&pagecache, &file1, FLUSH_FORCE_WRITE)) + { + diag("Got error in flush_pagecache_blocks\n"); + res= 0; + goto err; + } + ok((res&= test(test_file(file1, file1_name, PAGE_SIZE*2, PAGE_SIZE, + simple_pin_test_file2))), + "Simple pin page result file"); + if (res) + reset_file(file1, file1_name); +err: + free(buffw); + free(buffr); + DBUG_RETURN(res); +} + +/* + Prepare page, write new value, then delete page from cache without flush, + on the disk should be page with old content written during preparation +*/ + +int simple_delete_forget_test() +{ + unsigned char *buffw= malloc(PAGE_SIZE); + unsigned char *buffr= malloc(PAGE_SIZE); + int res; + DBUG_ENTER("simple_delete_forget_test"); + /* prepare the file */ + bfill(buffw, PAGE_SIZE, '\1'); + pagecache_write(&pagecache, &file1, 0, 3, (char*)buffw, + PAGECACHE_PLAIN_PAGE, + PAGECACHE_LOCK_LEFT_UNLOCKED, + PAGECACHE_PIN_LEFT_UNPINNED, + PAGECACHE_WRITE_DELAY, + 0); + flush_pagecache_blocks(&pagecache, &file1, FLUSH_FORCE_WRITE); + /* test */ + bfill(buffw, PAGE_SIZE, '\2'); + pagecache_write(&pagecache, &file1, 0, 3, (char*)buffw, + PAGECACHE_PLAIN_PAGE, + PAGECACHE_LOCK_LEFT_UNLOCKED, + PAGECACHE_PIN_LEFT_UNPINNED, + PAGECACHE_WRITE_DELAY, + 0); + pagecache_delete_page(&pagecache, &file1, 0, + PAGECACHE_LOCK_WRITE, 0); + flush_pagecache_blocks(&pagecache, &file1, FLUSH_FORCE_WRITE); + ok((res= test(test_file(file1, file1_name, PAGE_SIZE, PAGE_SIZE, + simple_delete_forget_test_file))), + "Simple delete-forget page file"); + if (res) + reset_file(file1, file1_name); + free(buffw); + free(buffr); + DBUG_RETURN(res); +} + +/* + Prepare page with locking, write new content to the page, + delete page with flush and on existing lock, + check that page on disk contain new value. +*/ + +int simple_delete_flush_test() +{ + unsigned char *buffw= malloc(PAGE_SIZE); + unsigned char *buffr= malloc(PAGE_SIZE); + int res; + DBUG_ENTER("simple_delete_flush_test"); + /* prepare the file */ + bfill(buffw, PAGE_SIZE, '\1'); + pagecache_write(&pagecache, &file1, 0, 3, (char*)buffw, + PAGECACHE_PLAIN_PAGE, + PAGECACHE_LOCK_WRITE, + PAGECACHE_PIN, + PAGECACHE_WRITE_DELAY, + 0); + flush_pagecache_blocks(&pagecache, &file1, FLUSH_FORCE_WRITE); + /* test */ + bfill(buffw, PAGE_SIZE, '\2'); + pagecache_write(&pagecache, &file1, 0, 3, (char*)buffw, + PAGECACHE_PLAIN_PAGE, + PAGECACHE_LOCK_LEFT_WRITELOCKED, + PAGECACHE_PIN_LEFT_PINNED, + PAGECACHE_WRITE_DELAY, + 0); + pagecache_delete_page(&pagecache, &file1, 0, + PAGECACHE_LOCK_LEFT_WRITELOCKED, 1); + flush_pagecache_blocks(&pagecache, &file1, FLUSH_FORCE_WRITE); + ok((res= test(test_file(file1, file1_name, PAGE_SIZE, PAGE_SIZE, + simple_delete_flush_test_file))), + "Simple delete-forget page file"); + if (res) + reset_file(file1, file1_name); + free(buffw); + free(buffr); + DBUG_RETURN(res); +} + + +/* + write then read file bigger then cache +*/ + +int simple_big_test() +{ + unsigned char *buffw= (unsigned char *)malloc(PAGE_SIZE); + unsigned char *buffr= (unsigned char *)malloc(PAGE_SIZE); + struct file_desc *desc= + (struct file_desc *)malloc((PCACHE_SIZE/(PAGE_SIZE/2)) * + sizeof(struct file_desc)); + int res, i; + DBUG_ENTER("simple_big_test"); + /* prepare the file twice larger then cache */ + for (i= 0; i < PCACHE_SIZE/(PAGE_SIZE/2); i++) + { + bfill(buffw, PAGE_SIZE, (unsigned char) (i & 0xff)); + desc[i].length= PAGE_SIZE; + desc[i].content= (i & 0xff); + pagecache_write(&pagecache, &file1, i, 3, (char*)buffw, + PAGECACHE_PLAIN_PAGE, + PAGECACHE_LOCK_LEFT_UNLOCKED, + PAGECACHE_PIN_LEFT_UNPINNED, + PAGECACHE_WRITE_DELAY, + 0); + } + ok(1, "Simple big file write"); + /* check written pages sequentally read */ + for (i= 0; i < PCACHE_SIZE/(PAGE_SIZE/2); i++) + { + int j; + pagecache_read(&pagecache, &file1, i, 3, (char*)buffr, + PAGECACHE_PLAIN_PAGE, + PAGECACHE_LOCK_LEFT_UNLOCKED, + 0); + for(j= 0; j < PAGE_SIZE; j++) + { + if (buffr[j] != (i & 0xff)) + { + diag("simple_big_test seq: page %u byte %u mismatch\n", i, j); + return 0; + } + } + } + ok(1, "simple big file sequentally read"); + /* chack random reads */ + for (i= 0; i < PCACHE_SIZE/(PAGE_SIZE); i++) + { + int j, page; + page= rand() % (PCACHE_SIZE/(PAGE_SIZE/2)); + pagecache_read(&pagecache, &file1, page, 3, (char*)buffr, + PAGECACHE_PLAIN_PAGE, + PAGECACHE_LOCK_LEFT_UNLOCKED, + 0); + for(j= 0; j < PAGE_SIZE; j++) + { + if (buffr[j] != (page & 0xff)) + { + diag("simple_big_test rnd: page %u byte %u mismatch\n", page, j); + return 0; + } + } + } + ok(1, "simple big file random read"); + flush_pagecache_blocks(&pagecache, &file1, FLUSH_FORCE_WRITE); + + ok((res= test(test_file(file1, file1_name, PCACHE_SIZE*2, PAGE_SIZE, + desc))), + "Simple big file"); + if (res) + reset_file(file1, file1_name); + free(buffw); + free(buffr); + DBUG_RETURN(res); +} +/* + Thread function +*/ + +static void *test_thread(void *arg) +{ + int param=*((int*) arg); + + my_thread_init(); + DBUG_ENTER("test_thread"); + + DBUG_PRINT("enter", ("param: %d", param)); + + if (!simple_read_write_test() || + !simple_read_change_write_read_test() || + !simple_pin_test() || + !simple_delete_forget_test() || + !simple_delete_flush_test() || + !simple_big_test()) + exit(1); + + DBUG_PRINT("info", ("Thread %s ended\n", my_thread_name())); + pthread_mutex_lock(&LOCK_thread_count); + thread_count--; + VOID(pthread_cond_signal(&COND_thread_count)); /* Tell main we are ready */ + pthread_mutex_unlock(&LOCK_thread_count); + free((gptr) arg); + my_thread_end(); + DBUG_RETURN(0); +} + + +int main(int argc, char **argv __attribute__((unused))) +{ + pthread_t tid; + pthread_attr_t thr_attr; + int *param, error, pagen; + + MY_INIT(argv[0]); + +#ifndef DBUG_OFF +#if defined(__WIN__) + default_dbug_option= "d:t:i:O,\\test_pagecache_single.trace"; +#else + default_dbug_option= "d:t:i:o,/tmp/test_pagecache_single.trace"; +#endif + if (argc > 1) + { + DBUG_SET(default_dbug_option); + DBUG_SET_INITIAL(default_dbug_option); + } +#endif + + + DBUG_ENTER("main"); + DBUG_PRINT("info", ("Main thread: %s\n", my_thread_name())); + if ((file1.file= my_open(file1_name, + O_CREAT | O_TRUNC | O_RDWR, MYF(0))) == -1) + { + fprintf(stderr, "Got error during file1 creation from open() (errno: %d)\n", + errno); + exit(1); + } + DBUG_PRINT("info", ("file1: %d", file1.file)); + if (chmod(file1_name, S_IRWXU | S_IRWXG | S_IRWXO) != 0) + { + fprintf(stderr, "Got error during file1 chmod() (errno: %d)\n", + errno); + exit(1); + } + my_pwrite(file1.file, "test file", 9, 0, MYF(0)); + + if ((error= pthread_cond_init(&COND_thread_count, NULL))) + { + fprintf(stderr, "Got error: %d from pthread_cond_init (errno: %d)\n", + error, errno); + exit(1); + } + if ((error= pthread_mutex_init(&LOCK_thread_count, MY_MUTEX_INIT_FAST))) + { + fprintf(stderr, "Got error: %d from pthread_cond_init (errno: %d)\n", + error, errno); + exit(1); + } + + if ((error= pthread_attr_init(&thr_attr))) + { + fprintf(stderr,"Got error: %d from pthread_attr_init (errno: %d)\n", + error,errno); + exit(1); + } + if ((error= pthread_attr_setdetachstate(&thr_attr, PTHREAD_CREATE_DETACHED))) + { + fprintf(stderr, + "Got error: %d from pthread_attr_setdetachstate (errno: %d)\n", + error,errno); + exit(1); + } + +#ifndef pthread_attr_setstacksize /* void return value */ + if ((error= pthread_attr_setstacksize(&thr_attr, 65536L))) + { + fprintf(stderr,"Got error: %d from pthread_attr_setstacksize (errno: %d)\n", + error,errno); + exit(1); + } +#endif +#ifdef HAVE_THR_SETCONCURRENCY + VOID(thr_setconcurrency(2)); +#endif + + my_thread_global_init(); + + plan(12); + + if ((pagen= init_pagecache(&pagecache, PCACHE_SIZE, 0, 0, + PAGE_SIZE, 0)) == 0) + { + fprintf(stderr,"Got error: init_pagecache() (errno: %d)\n", + errno); + exit(1); + } + DBUG_PRINT("info", ("Page cache %d pages", pagen)); + + if ((error=pthread_mutex_lock(&LOCK_thread_count))) + { + fprintf(stderr,"Got error: %d from pthread_mutex_lock (errno: %d)\n", + error,errno); + exit(1); + } + param=(int*) malloc(sizeof(int)); + *param= 1; + if ((error= pthread_create(&tid, &thr_attr, test_thread, (void*) param))) + { + fprintf(stderr,"Got error: %d from pthread_create (errno: %d)\n", + error,errno); + exit(1); + } + thread_count++; + DBUG_PRINT("info", ("Thread started")); + pthread_mutex_unlock(&LOCK_thread_count); + + pthread_attr_destroy(&thr_attr); + + if ((error= pthread_mutex_lock(&LOCK_thread_count))) + fprintf(stderr,"Got error: %d from pthread_mutex_lock\n",error); + while (thread_count) + { + if ((error= pthread_cond_wait(&COND_thread_count,&LOCK_thread_count))) + fprintf(stderr,"Got error: %d from pthread_cond_wait\n",error); + } + if ((error= pthread_mutex_unlock(&LOCK_thread_count))) + fprintf(stderr,"Got error: %d from pthread_mutex_unlock\n",error); + DBUG_PRINT("info", ("thread ended")); + + end_pagecache(&pagecache, 1); + DBUG_PRINT("info", ("Page cache ended")); + + if (my_close(file1.file, MYF(0)) != 0) + { + fprintf(stderr, "Got error during file1 closing from close() (errno: %d)\n", + errno); + exit(1); + } + /*my_delete(file1_name, MYF(0));*/ + my_thread_global_end(); + + DBUG_PRINT("info", ("file1 (%d) closed", file1.file)); + + DBUG_PRINT("info", ("Program end")); + + DBUG_RETURN(exit_status()); +} diff --git a/unittest/mysys/test_file.c b/unittest/mysys/test_file.c new file mode 100644 index 00000000000..758d0bfa81b --- /dev/null +++ b/unittest/mysys/test_file.c @@ -0,0 +1,68 @@ +#include +#include +#include +#include "test_file.h" + + +/* + Check that file contance correspond to descriptor + + SYNOPSIS + test_file() + file File to test + file_name Path (and name) of file which is tested + size size of file + buff_size size of buffer which is enought to check the file + desc file descriptor to check with + + RETURN + 1 file if OK + 0 error +*/ + +int test_file(PAGECACHE_FILE file, char *file_name, + off_t size, size_t buff_size, struct file_desc *desc) +{ + MY_STAT stat_buff, *stat; + unsigned char *buffr= malloc(buff_size); + off_t pos= 0; + size_t byte; + int step= 0; + + if ((stat= my_stat(file_name, &stat_buff, MYF(0))) == NULL) + { + diag("Can't stat() %s (errno: %d)\n", file_name, errno); + return 0; + } + if (stat->st_size != size) + { + diag("file %s size is %lu (should be %lu)\n", + file_name, (ulong) stat->st_size, (ulong) size); + return 0; + } + /* check content */ + my_seek(file.file, 0, SEEK_SET, MYF(0)); + while (desc[step].length != 0) + { + if (my_read(file.file, (char*)buffr, desc[step].length, MYF(0)) != + desc[step].length) + { + diag("Can't read %u bytes from %s (errno: %d)\n", + (uint)desc[step].length, file_name, errno); + return 0; + } + for (byte= 0; byte < desc[step].length; byte++) + { + if (buffr[byte] != desc[step].content) + { + diag("content of %s mismatch 0x%x in position %lu instead of 0x%x\n", + file_name, (uint) buffr[byte], (ulong) (pos + byte), + desc[step].content); + return 0; + } + } + pos+= desc[step].length; + step++; + } + return 1; +} diff --git a/unittest/mysys/test_file.h b/unittest/mysys/test_file.h new file mode 100644 index 00000000000..ea787c123ed --- /dev/null +++ b/unittest/mysys/test_file.h @@ -0,0 +1,14 @@ + +#include + +/* + File content descriptor +*/ +struct file_desc +{ + unsigned int length; + unsigned char content; +}; + +int test_file(PAGECACHE_FILE file, char *file_name, + off_t size, size_t buff_size, struct file_desc *desc); -- cgit v1.2.1 From 3becab22e9dd774d58983e553b6fbbfb9960f852 Mon Sep 17 00:00:00 2001 From: unknown Date: Thu, 16 Nov 2006 23:04:53 +0100 Subject: Maria: fix for compiler warning (signed-ness). #warning to warn (!) about current issues in the pagecache hindering checkpoint (we will have to fix them!). mysys/mf_pagecache.c: fix for compiler warning (signed-ness). #warning to warn (!) about current issues in the pagecache hindering checkpoint (we will have to fix them!). --- mysys/mf_pagecache.c | 8 +++++++- 1 file changed, 7 insertions(+), 1 deletion(-) diff --git a/mysys/mf_pagecache.c b/mysys/mf_pagecache.c index d8e235bc7f4..ab27e056c37 100755 --- a/mysys/mf_pagecache.c +++ b/mysys/mf_pagecache.c @@ -699,7 +699,7 @@ int init_pagecache(PAGECACHE *pagecache, my_size_t use_mem, pagecache->shift= my_bit_log2(block_size); DBUG_PRINT("info", ("block_size: %u", block_size)); - DBUG_ASSERT((1 << pagecache->shift) == block_size); + DBUG_ASSERT(((uint)(1 << pagecache->shift)) == block_size); blocks= (int) (use_mem / (sizeof(PAGECACHE_BLOCK_LINK) + 2 * sizeof(PAGECACHE_HASH_LINK) + @@ -3259,6 +3259,7 @@ restart: #ifndef DBUG_OFF int rc= #endif +#warning we are doing an unlock here, so need to give the page its rec_lsn! pagecache_make_lock_and_pin(pagecache, block, write_lock_change_table[lock].unlock_lock, write_pin_change_table[pin].unlock_pin); @@ -3614,6 +3615,11 @@ restart: else { /* Link the block into a list of blocks 'in switch' */ +#warning this unlink_changed() is a serious problem for Maria's Checkpoint: it \ +removes a page from the list of dirty pages, while it's still dirty. A \ + solution is to abandon first_in_switch, just wait for this page to be \ + flushed by somebody else, and loop. TODO: check all places where we remove a \ + page from the list of dirty pages unlink_changed(block); link_changed(block, &first_in_switch); } -- cgit v1.2.1 From 6ea0c5c7d6dcddbdade4e7f41273c09284b739b1 Mon Sep 17 00:00:00 2001 From: unknown Date: Fri, 17 Nov 2006 14:41:33 +0200 Subject: Post review changes. mysys/mf_pagecache.c: Unneed volitile definitions removed. Comments added. info_check_lock() code simplified. pagecache_delete_page() fixed. --- mysys/mf_pagecache.c | 142 ++++++++++++++++++++++----------------------------- 1 file changed, 62 insertions(+), 80 deletions(-) diff --git a/mysys/mf_pagecache.c b/mysys/mf_pagecache.c index ab27e056c37..4b92f68d9bf 100755 --- a/mysys/mf_pagecache.c +++ b/mysys/mf_pagecache.c @@ -127,11 +127,10 @@ my_bool my_disable_flush_pagecache_blocks= 0; (TYPE *) ((char *) (a) - offsetof(TYPE, MEMBER)) /* types of condition variables */ -#define COND_FOR_REQUESTED 0 -#define COND_FOR_SAVED 1 -#define COND_FOR_WRLOCK 2 -#define COND_FOR_COPY 3 -#define COND_SIZE 4 +#define COND_FOR_REQUESTED 0 /* queue of thread waiting for read operation */ +#define COND_FOR_SAVED 1 /* queue of thread waiting for flush */ +#define COND_FOR_WRLOCK 2 /* queue of write lock */ +#define COND_SIZE 3 /* number of COND_* queues */ typedef pthread_cond_t KEYCACHE_CONDVAR; @@ -296,8 +295,8 @@ struct st_pagecache_block_link wqueue[COND_SIZE]; /* queues on waiting requests for new/old pages */ uint requests; /* number of requests for the block */ byte *buffer; /* buffer for the block page */ - volatile uint status; /* state of the block */ - volatile uint pins; /* pin counter */ + uint status; /* state of the block */ + uint pins; /* pin counter */ #ifdef PAGECACHE_DEBUG PAGECACHE_PIN_INFO *pin_list; PAGECACHE_LOCK_INFO *lock_list; @@ -354,6 +353,20 @@ my_bool info_check_pin(PAGECACHE_BLOCK_LINK *block, DBUG_RETURN(0); } + +/* + Debug function which checks current lock/pin state and requested changes + + SYNOPSIS + info_check_lock() + lock requested lock changes + pin requested pin changes + + RETURN + 0 - OK + 1 - Error +*/ + my_bool info_check_lock(PAGECACHE_BLOCK_LINK *block, enum pagecache_page_lock lock, enum pagecache_page_pin pin) @@ -368,90 +381,58 @@ my_bool info_check_lock(PAGECACHE_BLOCK_LINK *block, case PAGECACHE_LOCK_LEFT_UNLOCKED: DBUG_ASSERT(pin == PAGECACHE_PIN_LEFT_UNPINNED); if (info) - { - DBUG_PRINT("info", - ("info_check_lock: thread: 0x%lx block 0x%lx: %c : U->U", - (ulong)thread, (ulong)block, (info->write_lock?'W':'R'))); - DBUG_RETURN(1); - } + goto error; break; case PAGECACHE_LOCK_LEFT_READLOCKED: DBUG_ASSERT(pin == PAGECACHE_PIN_LEFT_UNPINNED || pin == PAGECACHE_PIN_LEFT_PINNED); if (info == 0 || info->write_lock) - { - DBUG_PRINT("info", - ("info_check_lock: thread: 0x%lx block 0x%lx: %c : R->R", - (ulong)thread, (ulong)block, (info?'W':'U'))); - DBUG_RETURN(1); - } + goto error; break; case PAGECACHE_LOCK_LEFT_WRITELOCKED: DBUG_ASSERT(pin == PAGECACHE_PIN_LEFT_PINNED); if (info == 0 || !info->write_lock) - { - DBUG_PRINT("info", - ("info_check_lock: thread: 0x%lx block 0x%lx: %c : W->W", - (ulong)thread, (ulong)block, (info?'R':'U'))); - DBUG_RETURN(1); - } + goto error; break; case PAGECACHE_LOCK_READ: DBUG_ASSERT(pin == PAGECACHE_PIN_LEFT_UNPINNED || pin == PAGECACHE_PIN); if (info != 0) - { - DBUG_PRINT("info", - ("info_check_lock: thread: 0x%lx block 0x%lx: %c : U->R", - (ulong)thread, (ulong)block, (info->write_lock?'W':'R'))); - DBUG_RETURN(1); - } + goto error; break; case PAGECACHE_LOCK_WRITE: DBUG_ASSERT(pin == PAGECACHE_PIN); if (info != 0) - { - DBUG_PRINT("info", - ("info_check_lock: thread: 0x%lx block 0x%lx: %c : U->W", - (ulong)thread, (ulong)block, (info->write_lock?'W':'R'))); - DBUG_RETURN(1); - } + goto error; break; - case PAGECACHE_LOCK_READ_UNLOCK: DBUG_ASSERT(pin == PAGECACHE_PIN_LEFT_UNPINNED || pin == PAGECACHE_UNPIN); if (info == 0 || info->write_lock) - { - DBUG_PRINT("info", - ("info_check_lock: thread: 0x%lx block 0x%lx: %c : R->U", - (ulong)thread, (ulong)block, (info?'W':'U'))); - DBUG_RETURN(1); - } + goto error; break; case PAGECACHE_LOCK_WRITE_UNLOCK: DBUG_ASSERT(pin == PAGECACHE_UNPIN); if (info == 0 || !info->write_lock) - { - DBUG_PRINT("info", - ("info_check_lock: thread: 0x%lx block 0x%lx: %c : W->U", - (ulong)thread, (ulong)block, (info?'R':'U'))); - DBUG_RETURN(1); - } + goto error; break; case PAGECACHE_LOCK_WRITE_TO_READ: DBUG_ASSERT(pin == PAGECACHE_PIN_LEFT_PINNED || pin == PAGECACHE_UNPIN); if (info == 0 || !info->write_lock) - { - DBUG_PRINT("info", - ("info_check_lock: thread: 0x%lx block 0x%lx: %c : W->U", - (ulong)thread, (ulong)block, (info?'R':'U'))); - DBUG_RETURN(1); - } + goto error; break; } DBUG_RETURN(0); +error: + DBUG_PRINT("info", + ("info_check_lock: thread: 0x%lx block 0x%lx: info: %d wrt: %d," + "to lock: %s, to pin: %s", + (ulong)thread, (ulong)block, test(info), + (info ? info->write_lock : 0), + page_cache_page_lock_str[lock], + page_cache_page_pin_str[pin])); + DBUG_RETURN(1); } #endif @@ -1629,7 +1610,8 @@ static void unlink_hash(PAGECACHE *pagecache, PAGECACHE_HASH_LINK *hash_link) /* - Get the hash link for the page if it is in the cache + Get the hash link for the page if it is in the cache (do not put the + page in the cache if it is absent there) SYNOPSIS get_present_hash_link() @@ -2982,6 +2964,7 @@ restart: if (!link) { DBUG_PRINT("info", ("There is no such page in the cache")); + pagecache_pthread_mutex_unlock(&pagecache->cache_lock); DBUG_RETURN(0); } block= link->block; @@ -2998,32 +2981,31 @@ restart: if (block->status & BLOCK_CHANGED && flush) { - if (flush) - { - /* The block contains a dirty page - push it out of the cache */ + /* The block contains a dirty page - push it out of the cache */ - KEYCACHE_DBUG_PRINT("find_key_block", ("block is dirty")); + KEYCACHE_DBUG_PRINT("find_key_block", ("block is dirty")); - pagecache_pthread_mutex_unlock(&pagecache->cache_lock); - /* - The call is thread safe because only the current - thread might change the block->hash_link value - */ - DBUG_ASSERT(block->pins == 1); - error= pagecache_fwrite(pagecache, - &block->hash_link->file, - block->buffer, - block->hash_link->pageno, - block->type, - MYF(MY_NABP | MY_WAIT_IF_FULL)); - pagecache_pthread_mutex_lock(&pagecache->cache_lock); - pagecache->global_cache_write++; - if (error) - { - block->status|= BLOCK_ERROR; - goto err; - } + pagecache_pthread_mutex_unlock(&pagecache->cache_lock); + /* + The call is thread safe because only the current + thread might change the block->hash_link value + */ + DBUG_ASSERT(block->pins == 1); + error= pagecache_fwrite(pagecache, + &block->hash_link->file, + block->buffer, + block->hash_link->pageno, + block->type, + MYF(MY_NABP | MY_WAIT_IF_FULL)); + pagecache_pthread_mutex_lock(&pagecache->cache_lock); + pagecache->global_cache_write++; + + if (error) + { + block->status|= BLOCK_ERROR; + goto err; } + pagecache->blocks_changed--; pagecache->global_blocks_changed--; /* -- cgit v1.2.1 From 4b10971a3a887c176e3e71d092dd4cf4b4f68f3b Mon Sep 17 00:00:00 2001 From: unknown Date: Sun, 19 Nov 2006 21:32:16 +0100 Subject: post-review fixes --- storage/maria/tablockman.c | 73 +++++++++++++++++++++++++++------------------- 1 file changed, 43 insertions(+), 30 deletions(-) diff --git a/storage/maria/tablockman.c b/storage/maria/tablockman.c index b4cf77401bb..d8dffa09a5e 100644 --- a/storage/maria/tablockman.c +++ b/storage/maria/tablockman.c @@ -229,7 +229,7 @@ struct st_table_lock { #define hash_insert my_hash_insert /* for consistency :) */ static inline -TABLE_LOCK *find_loid(LOCKED_TABLE *table, uint16 loid) +TABLE_LOCK *find_by_loid(LOCKED_TABLE *table, uint16 loid) { return (TABLE_LOCK *)hash_search(& table->latest_locks, (byte *)& loid, sizeof(loid)); @@ -286,7 +286,7 @@ tablockman_getlock(TABLOCKMAN *lm, TABLE_LOCK_OWNER *lo, pthread_mutex_lock(& table->mutex); /* do we already have a lock on this resource ? */ - old= find_loid(table, lo->loid); + old= find_by_loid(table, lo->loid); /* calculate the level of the upgraded lock, if yes */ new_lock= old ? lock_combining_matrix[old->lock_type][lock] : lock; @@ -340,37 +340,50 @@ tablockman_getlock(TABLOCKMAN *lm, TABLE_LOCK_OWNER *lo, { /* found! */ wait_for= tmp; + break; } - else - { - /* - hmm, the lock before doesn't block us, let's look one step further. - the condition below means: - - if we never waited on a condition yet - OR - the lock before ours (blocker) waits on a lock (blocker2) that is - present in the hash AND and conflicts with 'blocker' - the condition after OR may fail if 'blocker2' was removed from - the hash, its signal woke us up, but 'blocker' itself didn't see - the signal yet. - */ - if (!lo->waiting_lock || - ((blocker2= find_loid(table, tmp->waiting_for_loid)) && - !lock_compatibility_matrix[blocker2->lock_type] - [blocker->lock_type])) - { - /* but it's waiting for a real lock. we'll wait for the same lock */ - wait_for= tmp->waiting_for; - } + /* + hmm, the lock before doesn't block us, let's look one step further. + the condition below means: + + if we never waited on a condition yet + OR + the lock before ours (blocker) waits on a lock (blocker2) that is + present in the hash AND and conflicts with 'blocker' + + the condition after OR may fail if 'blocker2' was removed from + the hash, its signal woke us up, but 'blocker' itself didn't see + the signal yet. + */ + if (!lo->waiting_lock || + ((blocker2= find_by_loid(table, tmp->waiting_for_loid)) && + !lock_compatibility_matrix[blocker2->lock_type] + [blocker->lock_type])) + { + /* but it's waiting for a real lock. we'll wait for the same lock */ + wait_for= tmp->waiting_for; /* - otherwise - a lock it's waiting for doesn't exist. - We've no choice but to scan the wait queue backwards, looking - for a conflicting lock or a lock waiting for a real lock. - QQ is there a way to avoid this scanning ? + We don't really need tmp->waiting_for, as tmp->waiting_for_loid + is enough. waiting_for is just a local cache to avoid calling + loid_to_tlo(). + But it's essensial that tmp->waiting_for pointer can ONLY + be dereferenced if find_by_loid() above returns a non-null + pointer, because a TABLE_LOCK_OWNER object that it points to + may've been freed when we come here after a signal. + In particular tmp->waiting_for_loid cannot be replaced + with tmp->waiting_for->loid. */ + DBUG_ASSERT(wait_for == lm->loid_to_tlo(tmp->waiting_for_loid)); + break; } + + /* + otherwise - a lock it's waiting for doesn't exist. + We've no choice but to scan the wait queue backwards, looking + for a conflicting lock or a lock waiting for a real lock. + QQ is there a way to avoid this scanning ? + */ } } @@ -531,8 +544,8 @@ void tablockman_release_locks(TABLOCKMAN *lm, TABLE_LOCK_OWNER *lo) transactions are awaken. But if trn2 times out, trn3 must be notified too (as IS and S locks are compatible). So trn2 must signal trn1->cond. */ - if (lock->prev && - lock_compatibility_matrix[lock->prev->lock_type][lock->lock_type]) + if (lock->next && + lock_compatibility_matrix[lock->next->lock_type][lock->lock_type]) { pthread_mutex_lock(lo->waiting_for->mutex); pthread_cond_broadcast(lo->waiting_for->cond); -- cgit v1.2.1 From a41ac15b960aee306e3464835b05a835fd98771d Mon Sep 17 00:00:00 2001 From: unknown Date: Tue, 21 Nov 2006 22:22:59 +0100 Subject: Maria - various fixes around durability of files: 1) on Mac OS X >=10.3, fcntl() is recommended over fsync (from the man page: "[With fsync()] the disk drive may also re-order the data so that later writes may be present while earlier writes are not. Applications such as databases that require a strict ordering of writes should use F_FULLFSYNC to ensure their data is written in the order they expect"). I have seen two other pieces of software changing from fsync to F_FULLFSYNC on Mac OS X. 2) to make a file creation/deletion/renaming durable on Linux (at least ext2 as I have tested) (see "man fsync"), a fsync() on the directory is needed: new functions to do that, and a flag MY_SYNC_DIR to do it in my_create/my_delete/my_rename. 3) now using this directory syncing when creating he frm if opt_sync_frm, and for Maria's control file when it is created. include/my_sys.h: new flag to my_create/my_delete/my_rename, which asks to sync the directory after the operation is done (currently does nothing except on Linux) libmysql/CMakeLists.txt: my_create() now depends on my_sync() so my_sync is needed for libmysql libmysql/Makefile.shared: my_create() now depends on my_sync() so my_sync is needed for libmysql mysys/my_create.c: my_create() can now sync the directory if asked for mysys/my_delete.c: my_delete() can now sync the directory if asked for mysys/my_open.c: it was a bug that my_close() is done on fd but a positive fd would still be returned, by my_register_filename(). mysys/my_rename.c: my_rename() can now sync the two directories (the one of "from" and the one of "to") if asked for. mysys/my_sync.c: On recent Mac OS X, fcntl(F_FULLFSYNC) is recommended over fsync() (see "man fsync" on Mac OS X 10.3). my_sync_dir(): to sync a directory after a file creation/deletion/ renaming; can be called directly or via MY_SYNC_DIR in my_create/ my_delete/my_rename(). No-op except on Linux (see "man fsync" on Linux). my_sync_dir_from_file(): same as above, just more practical when the caller has a file name but no directory name ready. Should the #warning even be a #error? I mean do we want to release binaries which don't guarantee any durability? sql/log.cc: a TODO for the future. sql/unireg.cc: If we sync the frm it makes sense to also sync its creation in the directory. storage/maria/ma_control_file.c: control file is vital, try to make it to disk --- include/my_sys.h | 3 ++ libmysql/CMakeLists.txt | 1 + libmysql/Makefile.shared | 2 +- mysys/my_create.c | 3 ++ mysys/my_delete.c | 2 ++ mysys/my_open.c | 1 + mysys/my_rename.c | 5 +++ mysys/my_sync.c | 74 +++++++++++++++++++++++++++++++++++++++++ sql/log.cc | 5 +++ sql/unireg.cc | 9 +++-- storage/maria/ma_control_file.c | 11 ++---- 11 files changed, 104 insertions(+), 12 deletions(-) diff --git a/include/my_sys.h b/include/my_sys.h index fc83c583201..3c5b5e6cbf4 100644 --- a/include/my_sys.h +++ b/include/my_sys.h @@ -55,6 +55,7 @@ extern int NEAR my_errno; /* Last error in mysys */ #define MY_WME 16 /* Write message on error */ #define MY_WAIT_IF_FULL 32 /* Wait and try again if disk full error */ #define MY_IGNORE_BADFD 32 /* my_sync: ignore 'bad descriptor' errors */ +#define MY_SYNC_DIR 1024 /* my_create/delete/rename: sync directory */ #define MY_RAID 64 /* Support for RAID */ #define MY_FULL_IO 512 /* For my_read - loop intil I/O is complete */ #define MY_DONT_CHECK_FILESIZE 128 /* Option to init_io_cache() */ @@ -622,6 +623,8 @@ extern FILE *my_fdopen(File Filedes,const char *name, int Flags,myf MyFlags); extern int my_fclose(FILE *fd,myf MyFlags); extern int my_chsize(File fd,my_off_t newlength, int filler, myf MyFlags); extern int my_sync(File fd, myf my_flags); +extern void my_sync_dir(const char *dir_name, myf my_flags); +extern void my_sync_dir_by_file(const char *file_name, myf my_flags); extern int my_error _VARARGS((int nr,myf MyFlags, ...)); extern int my_printf_error _VARARGS((uint my_err, const char *format, myf MyFlags, ...)) diff --git a/libmysql/CMakeLists.txt b/libmysql/CMakeLists.txt index d12b6ca6c10..3c9ed53a866 100644 --- a/libmysql/CMakeLists.txt +++ b/libmysql/CMakeLists.txt @@ -37,6 +37,7 @@ ADD_LIBRARY(libmysql SHARED dll.c libmysql.def ../mysys/my_open.c ../mysys/my_pread.c ../mysys/my_pthread.c ../mysys/my_read.c ../mysys/my_realloc.c ../mysys/my_rename.c ../mysys/my_seek.c ../mysys/my_static.c ../strings/my_strtoll10.c ../mysys/my_symlink.c + ../mysys/my_sync.c ../mysys/my_symlink2.c ../mysys/my_thr_init.c ../sql-common/my_time.c ../strings/my_vsnprintf.c ../mysys/my_wincond.c ../mysys/my_winthread.c ../mysys/my_write.c ../sql/net_serv.cc ../sql-common/pack.c ../sql/password.c diff --git a/libmysql/Makefile.shared b/libmysql/Makefile.shared index c2d98a81042..6326e255559 100644 --- a/libmysql/Makefile.shared +++ b/libmysql/Makefile.shared @@ -68,7 +68,7 @@ mysysobjects1 = my_init.lo my_static.lo my_malloc.lo my_realloc.lo \ mf_iocache2.lo my_seek.lo my_sleep.lo \ my_pread.lo mf_cache.lo md5.lo sha1.lo \ my_getopt.lo my_gethostbyname.lo my_port.lo \ - my_rename.lo my_chsize.lo + my_rename.lo my_chsize.lo my_sync.lo sqlobjects = net.lo sql_cmn_objects = pack.lo client.lo my_time.lo diff --git a/mysys/my_create.c b/mysys/my_create.c index e1e32b50842..bb3801691a5 100644 --- a/mysys/my_create.c +++ b/mysys/my_create.c @@ -53,6 +53,9 @@ File my_create(const char *FileName, int CreateFlags, int access_flags, fd = open(FileName, access_flags); #endif + if ((MyFlags & MY_SYNC_DIR) && (fd >=0)) + my_sync_dir_by_file(FileName, MyFlags); + DBUG_RETURN(my_register_filename(fd, FileName, FILE_BY_CREATE, EE_CANTCREATEFILE, MyFlags)); } /* my_create */ diff --git a/mysys/my_delete.c b/mysys/my_delete.c index de2a9814a56..6d90caa48ed 100644 --- a/mysys/my_delete.c +++ b/mysys/my_delete.c @@ -30,6 +30,8 @@ int my_delete(const char *name, myf MyFlags) my_error(EE_DELETE,MYF(ME_BELL+ME_WAITTANG+(MyFlags & ME_NOINPUT)), name,errno); } + else if (MyFlags & MY_SYNC_DIR) + my_sync_dir_by_file(name, MyFlags); DBUG_RETURN(err); } /* my_delete */ diff --git a/mysys/my_open.c b/mysys/my_open.c index ab2f7c9ff27..344e9c0a43b 100644 --- a/mysys/my_open.c +++ b/mysys/my_open.c @@ -162,6 +162,7 @@ File my_register_filename(File fd, const char *FileName, enum file_type } pthread_mutex_unlock(&THR_LOCK_open); (void) my_close(fd, MyFlags); + fd= -1; my_errno=ENOMEM; } else diff --git a/mysys/my_rename.c b/mysys/my_rename.c index 9c27238cc72..2c9ace6223a 100644 --- a/mysys/my_rename.c +++ b/mysys/my_rename.c @@ -61,5 +61,10 @@ int my_rename(const char *from, const char *to, myf MyFlags) if (MyFlags & (MY_FAE+MY_WME)) my_error(EE_LINK, MYF(ME_BELL+ME_WAITTANG),from,to,my_errno); } + else if (MyFlags & MY_SYNC_DIR) + { + my_sync_dir_by_file(from, MyFlags); + my_sync_dir_by_file(to, MyFlags); + } DBUG_RETURN(error); } /* my_rename */ diff --git a/mysys/my_sync.c b/mysys/my_sync.c index c557324b52c..eaa26ef07a7 100644 --- a/mysys/my_sync.c +++ b/mysys/my_sync.c @@ -49,6 +49,12 @@ int my_sync(File fd, myf my_flags) do { +#if defined(F_FULLFSYNC) + /* Recent Mac OS X versions insist this call is safer than fsync() */ + if (!(res= fcntl(fd, F_FULLFSYNC, 0))) + break; /* ok */ + /* Some fs don't support F_FULLFSYNC and fail above, fallback: */ +#endif #if defined(HAVE_FDATASYNC) res= fdatasync(fd); #elif defined(HAVE_FSYNC) @@ -56,6 +62,7 @@ int my_sync(File fd, myf my_flags) #elif defined(__WIN__) res= _commit(fd); #else +#warning Cannot find a way to sync a file, durability in danger res= 0; /* No sync (strange OS) */ #endif } while (res == -1 && errno == EINTR); @@ -74,3 +81,70 @@ int my_sync(File fd, myf my_flags) DBUG_RETURN(res); } /* my_sync */ + +/* + Force directory information to disk. Only Linux is known to need this to + make sure a file creation/deletion/renaming in(from,to) this directory + durable. + + SYNOPSIS + my_sync_dir() + dir_name the name of the directory + my_flags unused + + RETURN + nothing (the sync may fail sometimes). +*/ +void my_sync_dir(const char *dir_name, myf my_flags __attribute__((unused))) +{ +#ifdef TARGET_OS_LINUX + DBUG_ENTER("my_sync_dir"); + DBUG_PRINT("my",("Dir: '%s' my_flags: %d", dir_name, my_flags)); + File dir_fd; + int error= 0; + /* + Syncing a dir does not work on all filesystems (e.g. tmpfs->EINVAL) : + ignore errors. But print them to the debug log. + */ + if (((dir_fd= my_open(dir_name, O_RDONLY, MYF(0))) >= 0)) + { + if (my_sync(dir_fd, MYF(0))) + { + error= errno; + DBUG_PRINT("info",("my_sync failed errno: %d", error)); + } + my_close(dir_fd, MYF(0)); + } + else + { + error= errno; + DBUG_PRINT("info",("my_open failed errno: %d", error)); + } + DBUG_VOID_RETURN; +#endif +} + + +/* + Force directory information to disk. Only Linux is known to need this to + make sure a file creation/deletion/renaming in(from,to) this directory + durable. + + SYNOPSIS + my_sync_dir_by_file() + file_name the name of a file in the directory + my_flags unused + + RETURN + nothing (the sync may fail sometimes). +*/ +void my_sync_dir_by_file(const char *file_name, + myf my_flags __attribute__((unused))) +{ +#ifdef TARGET_OS_LINUX + char dir_name[FN_REFLEN]; + dirname_part(dir_name, file_name); + return my_sync_dir(dir_name, my_flags); +#endif +} + diff --git a/sql/log.cc b/sql/log.cc index b63ec563baf..cbba6ec1ddf 100644 --- a/sql/log.cc +++ b/sql/log.cc @@ -2102,6 +2102,11 @@ bool MYSQL_BIN_LOG::open_index_file(const char *index_file_name_arg, my_seek(index_file_nr,0L,MY_SEEK_END,MYF(0)), 0, MYF(MY_WME | MY_WAIT_IF_FULL))) { + /* + TODO: all operations creating/deleting the index file or a log, should + call my_sync_dir() or my_sync_dir_by_file() to be durable. + TODO: file creation should be done with my_create() not my_open(). + */ if (index_file_nr >= 0) my_close(index_file_nr,MYF(0)); return TRUE; diff --git a/sql/unireg.cc b/sql/unireg.cc index 2ea572c782c..5c2997c1483 100644 --- a/sql/unireg.cc +++ b/sql/unireg.cc @@ -285,9 +285,12 @@ bool mysql_create_frm(THD *thd, const char *file_name, my_free((gptr) screen_buff,MYF(0)); my_free((gptr) keybuff, MYF(0)); - if (opt_sync_frm && !(create_info->options & HA_LEX_CREATE_TMP_TABLE) && - my_sync(file, MYF(MY_WME))) - goto err2; + if (opt_sync_frm && !(create_info->options & HA_LEX_CREATE_TMP_TABLE)) + { + if (my_sync(file, MYF(MY_WME))) + goto err2; + my_sync_dir_by_file(file_name, MYF(0)); + } if (my_close(file,MYF(MY_WME))) goto err3; diff --git a/storage/maria/ma_control_file.c b/storage/maria/ma_control_file.c index 5090fac4182..47583466cd7 100644 --- a/storage/maria/ma_control_file.c +++ b/storage/maria/ma_control_file.c @@ -134,16 +134,11 @@ CONTROL_FILE_ERROR ma_control_file_create_or_open() if (create_file) { - if ((control_file_fd= my_create(name, 0, open_flags, MYF(0))) < 0) + if ((control_file_fd= my_create(name, 0, + open_flags, MYF(MY_SYNC_DIR))) < 0) DBUG_RETURN(CONTROL_FILE_UNKNOWN_ERROR); - /* - TODO: from "man fsync" on Linux: - "fsync does not necessarily ensure that the entry in the directory - containing the file has also reached disk. For that an explicit - fsync on the file descriptor of the directory is also needed." - So if we just created the file we should sync the directory. - Maybe there should be a flag of my_create() to do this. + /* To be safer we should make sure that there are no logs or data/index files around (indeed it could be that the control file alone was deleted or not restored, and we should not go on with life at this point). -- cgit v1.2.1 From 354d4a7e06074beb0a08afab06b0f5dfd63d7d1d Mon Sep 17 00:00:00 2001 From: unknown Date: Wed, 22 Nov 2006 15:53:26 +0100 Subject: shut up signedness warnings in atomic ops in gcc unittest/mysys/my_atomic-t.c: try both signed and unsigned arguments for atomic ops --- include/my_atomic.h | 139 ++++++++++++++++++++++++++++++------------- unittest/mysys/my_atomic-t.c | 5 +- 2 files changed, 101 insertions(+), 43 deletions(-) diff --git a/include/my_atomic.h b/include/my_atomic.h index 78121c59c40..7b1b56815e0 100644 --- a/include/my_atomic.h +++ b/include/my_atomic.h @@ -61,70 +61,115 @@ #endif #ifndef make_atomic_add_body -#define make_atomic_add_body(S) \ +#define make_atomic_add_body(S) \ int ## S tmp=*a; \ while (!my_atomic_cas ## S(a, &tmp, tmp+v)); \ v=tmp; #endif +#ifdef __GNUC__ +/* + we want to be able to use my_atomic_xxx functions with + both signed and unsigned integers. But gcc will issue a warning + "passing arg N of `my_atomic_XXX' as [un]signed due to prototype" + if the signedness of the argument doesn't match the prototype, or + "pointer targets in passing argument N of my_atomic_XXX differ in signedness" + if int* is used where uint* is expected (or vice versa). + Let's shut these warnings up +*/ +#define make_transparent_unions(S) \ + typedef union { \ + int ## S i; \ + uint ## S u; \ + } U_ ## S __attribute__ ((transparent_union)); \ + typedef union { \ + int ## S volatile *i; \ + uint ## S volatile *u; \ + } Uv_ ## S __attribute__ ((transparent_union)); +#define uintptr intptr +make_transparent_unions(8) +make_transparent_unions(16) +make_transparent_unions(32) +make_transparent_unions(ptr) +#undef uintptr +#undef make_transparent_unions +#define a U_a.i +#define cmp U_cmp.i +#define v U_v.i +#define set U_set.i +#else +#define U_8 int8 +#define U_16 int16 +#define U_32 int32 +#define U_ptr intptr +#define Uv_8 int8 +#define Uv_16 int16 +#define Uv_32 int32 +#define Uv_ptr intptr +#define U_a volatile *a +#define U_cmp *cmp +#define U_v v +#define U_set set +#endif /* __GCC__ transparent_union magic */ + #ifdef HAVE_INLINE -#define make_atomic_add(S) \ -STATIC_INLINE int ## S my_atomic_add ## S( \ - int ## S volatile *a, int ## S v) \ -{ \ - make_atomic_add_body(S); \ - return v; \ +#define make_atomic_add(S) \ +STATIC_INLINE int ## S my_atomic_add ## S( \ + Uv_ ## S U_a, U_ ## S U_v) \ +{ \ + make_atomic_add_body(S); \ + return v; \ } -#define make_atomic_fas(S) \ -STATIC_INLINE int ## S my_atomic_fas ## S( \ - int ## S volatile *a, int ## S v) \ -{ \ - make_atomic_fas_body(S); \ - return v; \ +#define make_atomic_fas(S) \ +STATIC_INLINE int ## S my_atomic_fas ## S( \ + Uv_ ## S U_a, U_ ## S U_v) \ +{ \ + make_atomic_fas_body(S); \ + return v; \ } -#define make_atomic_cas(S) \ -STATIC_INLINE int my_atomic_cas ## S(int ## S volatile *a, \ - int ## S *cmp, int ## S set) \ -{ \ - int8 ret; \ - make_atomic_cas_body(S); \ - return ret; \ +#define make_atomic_cas(S) \ +STATIC_INLINE int my_atomic_cas ## S(Uv_ ## S U_a, \ + Uv_ ## S U_cmp, U_ ## S U_set) \ +{ \ + int8 ret; \ + make_atomic_cas_body(S); \ + return ret; \ } -#define make_atomic_load(S) \ -STATIC_INLINE int ## S my_atomic_load ## S(int ## S volatile *a) \ -{ \ - int ## S ret; \ - make_atomic_load_body(S); \ - return ret; \ +#define make_atomic_load(S) \ +STATIC_INLINE int ## S my_atomic_load ## S(Uv_ ## S U_a) \ +{ \ + int ## S ret; \ + make_atomic_load_body(S); \ + return ret; \ } -#define make_atomic_store(S) \ -STATIC_INLINE void my_atomic_store ## S( \ - int ## S volatile *a, int ## S v) \ -{ \ - make_atomic_store_body(S); \ +#define make_atomic_store(S) \ +STATIC_INLINE void my_atomic_store ## S( \ + Uv_ ## S U_a, U_ ## S U_v) \ +{ \ + make_atomic_store_body(S); \ } #else /* no inline functions */ -#define make_atomic_add(S) \ -extern int ## S my_atomic_add ## S(int ## S volatile *a, int ## S v); +#define make_atomic_add(S) \ +extern int ## S my_atomic_add ## S(Uv_ ## S, U_ ## S); -#define make_atomic_fas(S) \ -extern int ## S my_atomic_fas ## S(int ## S volatile *a, int ## S v); +#define make_atomic_fas(S) \ +extern int ## S my_atomic_fas ## S(Uv_ ## S, U_ ## S); -#define make_atomic_cas(S) \ -extern int my_atomic_cas ## S(int ## S volatile *a, int ## S *cmp, int ## S set); +#define make_atomic_cas(S) \ +extern int my_atomic_cas ## S(Uv_ ## S, Uv_ ## S, U_ ## S); -#define make_atomic_load(S) \ -extern int ## S my_atomic_load ## S(int ## S volatile *a); +#define make_atomic_load(S) \ +extern int ## S my_atomic_load ## S(Uv_ ## S); -#define make_atomic_store(S) \ -extern void my_atomic_store ## S(int ## S volatile *a, int ## S v); +#define make_atomic_store(S) \ +extern void my_atomic_store ## S(Uv_ ## S, U_ ## S); #endif @@ -157,6 +202,18 @@ make_atomic_fas(ptr) #undef _atomic_h_cleanup_ #endif +#undef U_8 +#undef U_16 +#undef U_32 +#undef U_ptr +#undef a +#undef cmp +#undef v +#undef set +#undef U_a +#undef U_cmp +#undef U_v +#undef U_set #undef make_atomic_add #undef make_atomic_cas #undef make_atomic_load diff --git a/unittest/mysys/my_atomic-t.c b/unittest/mysys/my_atomic-t.c index a33e233c29a..314335944d3 100644 --- a/unittest/mysys/my_atomic-t.c +++ b/unittest/mysys/my_atomic-t.c @@ -21,7 +21,8 @@ #include #include -volatile uint32 a32,b32,c32, N; +volatile uint32 a32,b32; +volatile int32 c32, N; my_atomic_rwlock_t rwl; LF_ALLOCATOR lf_allocator; LF_HASH lf_hash; @@ -98,7 +99,7 @@ pthread_handler_t test_atomic_cas_handler(void *arg) x= (x*m+0x87654321) & INT_MAX32; do { my_atomic_rwlock_wrlock(&rwl); - ok= my_atomic_cas32(&a32, &y, y+x); + ok= my_atomic_cas32(&a32, &y, (uint32)y+x); my_atomic_rwlock_wrunlock(&rwl); } while (!ok) ; do { -- cgit v1.2.1 From adfba203ffd1bd89d74a63ff09de9b9a40fb64d7 Mon Sep 17 00:00:00 2001 From: unknown Date: Wed, 22 Nov 2006 23:38:10 +0100 Subject: Maria - post-review fixes about my_sync_dir(): make it return an error (except if certain errno), test this error in callers. Do a single my_sync_dir() in my_rename() if possible. include/my_global.h: better have a symbol name talking about the feature, use it in the code of the feature, and define the symbol once depending on the platform, rather than have the platform "tested" in the code of the feature several times. include/my_sys.h: my_sync_dir() now can return error mysys/my_create.c: my_sync_dir() can now return an error mysys/my_delete.c: my_sync_dir() can now return an error mysys/my_rename.c: my_sync_dir() can now return an error. Do a single sync if "from" and "to" are the same directory. #ifdef here to not even compile dirname_part() if useless. mysys/my_sync.c: more comments. A compilation error if no way to make my_sync() work (I guess we don't want to ship a binary which cannot do any sync at all; users of strange OSes compile from source and can remove the #error). my_sync_dir() now returns an error (except for certain errno values considered ok; EIO "input/output error" is not ok). sql/unireg.cc: my_sync_dir() now returns an error which must be tested --- include/my_global.h | 8 +++++++ include/my_sys.h | 4 ++-- mysys/my_create.c | 8 +++++-- mysys/my_delete.c | 5 ++-- mysys/my_rename.c | 12 ++++++++-- mysys/my_sync.c | 67 +++++++++++++++++++++++++++-------------------------- sql/unireg.cc | 9 ++++--- 7 files changed, 67 insertions(+), 46 deletions(-) diff --git a/include/my_global.h b/include/my_global.h index 6831a76c945..f337c079c82 100644 --- a/include/my_global.h +++ b/include/my_global.h @@ -1468,4 +1468,12 @@ do { doubleget_union _tmp; \ #define dlerror() "" #endif +/* + Only Linux is known to need an explicit sync of the directory to make sure a + file creation/deletion/renaming in(from,to) this directory durable. +*/ +#ifdef TARGET_OS_LINUX +#define NEED_EXPLICIT_SYNC_DIR 1 +#endif + #endif /* my_global_h */ diff --git a/include/my_sys.h b/include/my_sys.h index 3c5b5e6cbf4..f235c4d3fed 100644 --- a/include/my_sys.h +++ b/include/my_sys.h @@ -623,8 +623,8 @@ extern FILE *my_fdopen(File Filedes,const char *name, int Flags,myf MyFlags); extern int my_fclose(FILE *fd,myf MyFlags); extern int my_chsize(File fd,my_off_t newlength, int filler, myf MyFlags); extern int my_sync(File fd, myf my_flags); -extern void my_sync_dir(const char *dir_name, myf my_flags); -extern void my_sync_dir_by_file(const char *file_name, myf my_flags); +extern int my_sync_dir(const char *dir_name, myf my_flags); +extern int my_sync_dir_by_file(const char *file_name, myf my_flags); extern int my_error _VARARGS((int nr,myf MyFlags, ...)); extern int my_printf_error _VARARGS((uint my_err, const char *format, myf MyFlags, ...)) diff --git a/mysys/my_create.c b/mysys/my_create.c index bb3801691a5..0b1bfa12c18 100644 --- a/mysys/my_create.c +++ b/mysys/my_create.c @@ -53,8 +53,12 @@ File my_create(const char *FileName, int CreateFlags, int access_flags, fd = open(FileName, access_flags); #endif - if ((MyFlags & MY_SYNC_DIR) && (fd >=0)) - my_sync_dir_by_file(FileName, MyFlags); + if ((MyFlags & MY_SYNC_DIR) && (fd >=0) && + my_sync_dir_by_file(FileName, MyFlags)) + { + my_close(fd, MyFlags); + fd= -1; + } DBUG_RETURN(my_register_filename(fd, FileName, FILE_BY_CREATE, EE_CANTCREATEFILE, MyFlags)); diff --git a/mysys/my_delete.c b/mysys/my_delete.c index 6d90caa48ed..d56507c36c0 100644 --- a/mysys/my_delete.c +++ b/mysys/my_delete.c @@ -30,8 +30,9 @@ int my_delete(const char *name, myf MyFlags) my_error(EE_DELETE,MYF(ME_BELL+ME_WAITTANG+(MyFlags & ME_NOINPUT)), name,errno); } - else if (MyFlags & MY_SYNC_DIR) - my_sync_dir_by_file(name, MyFlags); + else if ((MyFlags & MY_SYNC_DIR) && + my_sync_dir_by_file(name, MyFlags)) + err= -1; DBUG_RETURN(err); } /* my_delete */ diff --git a/mysys/my_rename.c b/mysys/my_rename.c index 2c9ace6223a..8c2a354324b 100644 --- a/mysys/my_rename.c +++ b/mysys/my_rename.c @@ -63,8 +63,16 @@ int my_rename(const char *from, const char *to, myf MyFlags) } else if (MyFlags & MY_SYNC_DIR) { - my_sync_dir_by_file(from, MyFlags); - my_sync_dir_by_file(to, MyFlags); +#ifdef NEED_EXPLICIT_SYNC_DIR + /* do only the needed amount of syncs: */ + char dir_from[FN_REFLEN], dir_to[FN_REFLEN]; + dirname_part(dir_from, from); + dirname_part(dir_to, to); + if (my_sync_dir(dir_from, MyFlags) || + (strcmp(dir_from, dir_to) && + my_sync_dir(dir_to, MyFlags))) + error= -1; +#endif } DBUG_RETURN(error); } /* my_rename */ diff --git a/mysys/my_sync.c b/mysys/my_sync.c index eaa26ef07a7..ada2ea84414 100644 --- a/mysys/my_sync.c +++ b/mysys/my_sync.c @@ -50,10 +50,14 @@ int my_sync(File fd, myf my_flags) do { #if defined(F_FULLFSYNC) - /* Recent Mac OS X versions insist this call is safer than fsync() */ + /* + In Mac OS X >= 10.3 this call is safer than fsync() (it forces the + disk's cache). + */ if (!(res= fcntl(fd, F_FULLFSYNC, 0))) break; /* ok */ - /* Some fs don't support F_FULLFSYNC and fail above, fallback: */ + /* Some file systems don't support F_FULLFSYNC and fail above: */ + DBUG_PRINT("info",("fcntl(F_FULLFSYNC) failed, falling back")); #endif #if defined(HAVE_FDATASYNC) res= fdatasync(fd); @@ -62,7 +66,7 @@ int my_sync(File fd, myf my_flags) #elif defined(__WIN__) res= _commit(fd); #else -#warning Cannot find a way to sync a file, durability in danger +#error Cannot find a way to sync a file, durability in danger res= 0; /* No sync (strange OS) */ #endif } while (res == -1 && errno == EINTR); @@ -74,7 +78,10 @@ int my_sync(File fd, myf my_flags) my_errno= -1; /* Unknown error */ if ((my_flags & MY_IGNORE_BADFD) && (er == EBADF || er == EINVAL || er == EROFS)) + { + DBUG_PRINT("info", ("ignoring errno %d", er)); res= 0; + } else if (my_flags & MY_WME) my_error(EE_SYNC, MYF(ME_BELL+ME_WAITTANG), my_filename(fd), my_errno); } @@ -83,68 +90,62 @@ int my_sync(File fd, myf my_flags) /* - Force directory information to disk. Only Linux is known to need this to - make sure a file creation/deletion/renaming in(from,to) this directory - durable. + Force directory information to disk. SYNOPSIS my_sync_dir() dir_name the name of the directory - my_flags unused + my_flags flags (MY_WME etc) RETURN - nothing (the sync may fail sometimes). + 0 if ok, !=0 if error */ -void my_sync_dir(const char *dir_name, myf my_flags __attribute__((unused))) +int my_sync_dir(const char *dir_name, myf my_flags) { -#ifdef TARGET_OS_LINUX +#ifdef NEED_EXPLICIT_SYNC_DIR DBUG_ENTER("my_sync_dir"); DBUG_PRINT("my",("Dir: '%s' my_flags: %d", dir_name, my_flags)); File dir_fd; - int error= 0; + int res= 0; /* - Syncing a dir does not work on all filesystems (e.g. tmpfs->EINVAL) : - ignore errors. But print them to the debug log. + Syncing a dir may give EINVAL on tmpfs on Linux, which is ok. + EIO on the other hand is very important. Hence MY_IGNORE_BADFD. */ - if (((dir_fd= my_open(dir_name, O_RDONLY, MYF(0))) >= 0)) + if ((dir_fd= my_open(dir_name, O_RDONLY, MYF(my_flags))) >= 0) { - if (my_sync(dir_fd, MYF(0))) - { - error= errno; - DBUG_PRINT("info",("my_sync failed errno: %d", error)); - } - my_close(dir_fd, MYF(0)); + if (my_sync(dir_fd, MYF(my_flags | MY_IGNORE_BADFD))) + res= 2; + if (my_close(dir_fd, MYF(my_flags))) + res= 3; } else - { - error= errno; - DBUG_PRINT("info",("my_open failed errno: %d", error)); - } - DBUG_VOID_RETURN; + res= 1; + DBUG_RETURN(res); +#else + return 0; #endif } /* - Force directory information to disk. Only Linux is known to need this to - make sure a file creation/deletion/renaming in(from,to) this directory - durable. + Force directory information to disk. SYNOPSIS my_sync_dir_by_file() file_name the name of a file in the directory - my_flags unused + my_flags flags (MY_WME etc) RETURN - nothing (the sync may fail sometimes). + 0 if ok, !=0 if error */ -void my_sync_dir_by_file(const char *file_name, - myf my_flags __attribute__((unused))) +int my_sync_dir_by_file(const char *file_name, myf my_flags) { -#ifdef TARGET_OS_LINUX +#ifdef NEED_EXPLICIT_SYNC_DIR char dir_name[FN_REFLEN]; dirname_part(dir_name, file_name); return my_sync_dir(dir_name, my_flags); +#else + return 0; #endif } diff --git a/sql/unireg.cc b/sql/unireg.cc index 5c2997c1483..b1d0e75723e 100644 --- a/sql/unireg.cc +++ b/sql/unireg.cc @@ -285,12 +285,11 @@ bool mysql_create_frm(THD *thd, const char *file_name, my_free((gptr) screen_buff,MYF(0)); my_free((gptr) keybuff, MYF(0)); - if (opt_sync_frm && !(create_info->options & HA_LEX_CREATE_TMP_TABLE)) - { - if (my_sync(file, MYF(MY_WME))) + if (opt_sync_frm && !(create_info->options & HA_LEX_CREATE_TMP_TABLE) && + (my_sync(file, MYF(MY_WME)) || + my_sync_dir_by_file(file_name, MYF(MY_WME)))) goto err2; - my_sync_dir_by_file(file_name, MYF(0)); - } + if (my_close(file,MYF(MY_WME))) goto err3; -- cgit v1.2.1 From de6f550ec7015fccd044a54c7628cdf8cdc2ed8c Mon Sep 17 00:00:00 2001 From: unknown Date: Mon, 27 Nov 2006 22:01:29 +0100 Subject: WL#3072 Maria Recovery. Making DDLs durable in Maria: Sync table files after CREATE (of non-temp table), DROP, RENAME, TRUNCATE, sync directories and symlinks (for the 3 first commands). Comments for future log records. In ma_rename(), if rename of index works and then rename of data fails, try to undo the rename of the index to leave a consistent state. mysys/my_symlink.c: sync directory after creation of a symbolic link in it, if asked mysys/my_sync.c: comment. Fix for when the file's name has no directory in it. storage/maria/ma_create.c: sync files and links and dirs when creating a non-temporary table. Optimizations of the above to reduce syncs in the common cases: * if index file and data file have the exact same paths (regular and link), sync the directories (of regular and link) only once after creating the last file (the data file). * don't sync the data file if we didn't write to it (always true in our builds). storage/maria/ma_delete_all.c: sync files after truncating a table storage/maria/ma_delete_table.c: sync files and symbolic links and dirs after dropping a table storage/maria/ma_extra.c: a function which wraps the sync of the index file and the sync of the data file. storage/maria/ma_locking.c: using a wrapper function storage/maria/ma_rename.c: sync files and symbolic links and dirs after renaming a table. If rename of index works and then rename of data fails, try to undo the rename of the index to leave a consistent state. That is just a try, it may fail... storage/maria/ma_test3.c: warning to not pay attention to this test. storage/maria/maria_def.h: declaration for the function added to ma_extra.c --- mysys/my_symlink.c | 2 ++ mysys/my_sync.c | 8 +++++-- storage/maria/ma_create.c | 49 +++++++++++++++++++++++++++++++++-------- storage/maria/ma_delete_all.c | 26 ++++++++++++++++++++++ storage/maria/ma_delete_table.c | 17 +++++++++++--- storage/maria/ma_extra.c | 11 ++++++--- storage/maria/ma_locking.c | 4 +--- storage/maria/ma_rename.c | 36 ++++++++++++++++++++++++++---- storage/maria/ma_test3.c | 4 ++++ storage/maria/maria_def.h | 1 + 10 files changed, 134 insertions(+), 24 deletions(-) diff --git a/mysys/my_symlink.c b/mysys/my_symlink.c index 7be3fcd36f0..b3d68992578 100644 --- a/mysys/my_symlink.c +++ b/mysys/my_symlink.c @@ -85,6 +85,8 @@ int my_symlink(const char *content, const char *linkname, myf MyFlags) if (MyFlags & MY_WME) my_error(EE_CANT_SYMLINK, MYF(0), linkname, content, errno); } + else if ((MyFlags & MY_SYNC_DIR) && my_sync_dir_by_file(linkname, MyFlags)) + result= -1; DBUG_RETURN(result); #endif /* HAVE_READLINK */ } diff --git a/mysys/my_sync.c b/mysys/my_sync.c index ada2ea84414..26bee5a293f 100644 --- a/mysys/my_sync.c +++ b/mysys/my_sync.c @@ -52,7 +52,7 @@ int my_sync(File fd, myf my_flags) #if defined(F_FULLFSYNC) /* In Mac OS X >= 10.3 this call is safer than fsync() (it forces the - disk's cache). + disk's cache and guarantees ordered writes). */ if (!(res= fcntl(fd, F_FULLFSYNC, 0))) break; /* ok */ @@ -89,6 +89,7 @@ int my_sync(File fd, myf my_flags) } /* my_sync */ +static const char cur_dir_name[]= {FN_CURLIB, 0}; /* Force directory information to disk. @@ -107,11 +108,14 @@ int my_sync_dir(const char *dir_name, myf my_flags) DBUG_PRINT("my",("Dir: '%s' my_flags: %d", dir_name, my_flags)); File dir_fd; int res= 0; + const char *correct_dir_name; + /* Sometimes the path does not contain an explicit directory */ + correct_dir_name= (dir_name[0] == 0) ? cur_dir_name : dir_name; /* Syncing a dir may give EINVAL on tmpfs on Linux, which is ok. EIO on the other hand is very important. Hence MY_IGNORE_BADFD. */ - if ((dir_fd= my_open(dir_name, O_RDONLY, MYF(my_flags))) >= 0) + if ((dir_fd= my_open(correct_dir_name, O_RDONLY, MYF(my_flags))) >= 0) { if (my_sync(dir_fd, MYF(my_flags | MY_IGNORE_BADFD))) res= 2; diff --git a/storage/maria/ma_create.c b/storage/maria/ma_create.c index 5926bba9406..76942e3d5e8 100644 --- a/storage/maria/ma_create.c +++ b/storage/maria/ma_create.c @@ -60,6 +60,8 @@ int maria_create(const char *name,uint keys,MARIA_KEYDEF *keydefs, ulong *rec_per_key_part; my_off_t key_root[HA_MAX_POSSIBLE_KEY],key_del[MARIA_MAX_KEY_BLOCK_SIZE]; MARIA_CREATE_INFO tmp_create_info; + my_bool tmp_table= FALSE; /* cache for presence of HA_OPTION_TMP_TABLE */ + myf sync_dir= MY_SYNC_DIR; DBUG_ENTER("maria_create"); DBUG_PRINT("enter", ("keys: %u columns: %u uniques: %u flags: %u", keys, columns, uniques, flags)); @@ -560,7 +562,11 @@ int maria_create(const char *name,uint keys,MARIA_KEYDEF *keydefs, /* max_data_file_length and max_key_file_length are recalculated on open */ if (options & HA_OPTION_TMP_TABLE) + { + tmp_table= TRUE; + sync_dir= 0; share.base.max_data_file_length=(my_off_t) ci->data_file_length; + } share.base.min_block_length= (share.base.pack_reclength+3 < MARIA_EXTEND_BLOCK_LENGTH && @@ -576,7 +582,7 @@ int maria_create(const char *name,uint keys,MARIA_KEYDEF *keydefs, { char *iext= strrchr(ci->index_file_name, '.'); int have_iext= iext && !strcmp(iext, MARIA_NAME_IEXT); - if (options & HA_OPTION_TMP_TABLE) + if (tmp_table) { char *path; /* chop off the table name, tempory tables use generated name */ @@ -597,8 +603,11 @@ int maria_create(const char *name,uint keys,MARIA_KEYDEF *keydefs, /* Don't create the table if the link or file exists to ensure that one doesn't accidently destroy another table. + Don't sync dir now if the data file has the same path. */ - create_flag=0; + create_flag= + (ci->data_file_name && + !strcmp(ci->index_file_name, ci->data_file_name)) ? 0 : sync_dir; } else { @@ -607,8 +616,11 @@ int maria_create(const char *name,uint keys,MARIA_KEYDEF *keydefs, (flags & HA_DONT_TOUCH_DATA) ? MY_RETURN_REAL_PATH : 0) | MY_APPEND_EXT); linkname_ptr=0; - /* Replace the current file */ - create_flag=MY_DELETE_OLD; + /* + Replace the current file. + Don't sync dir now if the data file has the same path. + */ + create_flag= MY_DELETE_OLD | (!ci->data_file_name ? 0 : sync_dir); } /* @@ -627,7 +639,7 @@ int maria_create(const char *name,uint keys,MARIA_KEYDEF *keydefs, } if ((file= my_create_with_symlink(linkname_ptr, filename, 0, create_mode, - MYF(MY_WME | create_flag))) < 0) + MYF(MY_WME|create_flag))) < 0) goto err; errpos=1; @@ -653,7 +665,7 @@ int maria_create(const char *name,uint keys,MARIA_KEYDEF *keydefs, char *dext= strrchr(ci->data_file_name, '.'); int have_dext= dext && !strcmp(dext, MARIA_NAME_DEXT); - if (options & HA_OPTION_TMP_TABLE) + if (tmp_table) { char *path; /* chop off the table name, tempory tables use generated name */ @@ -682,7 +694,7 @@ int maria_create(const char *name,uint keys,MARIA_KEYDEF *keydefs, } if ((dfile= my_create_with_symlink(linkname_ptr, filename, 0, create_mode, - MYF(MY_WME | create_flag))) < 0) + MYF(MY_WME | create_flag | sync_dir))) < 0) goto err; } errpos=3; @@ -802,12 +814,18 @@ int maria_create(const char *name,uint keys,MARIA_KEYDEF *keydefs, if (my_chsize(file,(ulong) share.base.keystart,0,MYF(0))) goto err; + if (!tmp_table && my_sync(file, MYF(0))) + goto err; + if (! (flags & HA_DONT_TOUCH_DATA)) { #ifdef USE_RELOC if (my_chsize(dfile,share.base.min_pack_length*ci->reloc_rows,0,MYF(0))) goto err; + if (!tmp_table && my_sync(dfile, MYF(0))) + goto err; #endif + /* if !USE_RELOC, there was no write to the file, no need to sync it */ errpos=2; if (my_close(dfile,MYF(0))) goto err; @@ -816,6 +834,19 @@ int maria_create(const char *name,uint keys,MARIA_KEYDEF *keydefs, pthread_mutex_unlock(&THR_LOCK_maria); if (my_close(file,MYF(0))) goto err; + /* + RECOVERYTODO + Write a log record describing the CREATE operation (just the file + names, link names, and the full header's content). + For this record to be of any use for Recovery, we need the upper + MySQL layer to be crash-safe, which it is not now (that would require work + using the ddl_log of sql/sql_table.cc); when is is, we should reconsider + the moment of writing this log record (before or after op, under + THR_LOCK_maria or not...), how to use it in Recovery, and force the log. + For now this record is just informative. + If operation failed earlier, we clean up in "err:" and the MySQL layer + will clean up the frm, so we needn't write anything to the log. + */ my_free((char*) rec_per_key_part,MYF(0)); DBUG_RETURN(0); @@ -831,14 +862,14 @@ err: if (! (flags & HA_DONT_TOUCH_DATA)) my_delete_with_symlink(fn_format(filename,name,"",MARIA_NAME_DEXT, MY_UNPACK_FILENAME | MY_APPEND_EXT), - MYF(0)); + MYF(sync_dir)); /* fall through */ case 1: VOID(my_close(file,MYF(0))); if (! (flags & HA_DONT_TOUCH_DATA)) my_delete_with_symlink(fn_format(filename,name,"",MARIA_NAME_IEXT, MY_UNPACK_FILENAME | MY_APPEND_EXT), - MYF(0)); + MYF(sync_dir)); } my_free((char*) rec_per_key_part, MYF(0)); DBUG_RETURN(my_errno=save_errno); /* return the fatal errno */ diff --git a/storage/maria/ma_delete_all.c b/storage/maria/ma_delete_all.c index b16d82ed9f7..fccd29b15f1 100644 --- a/storage/maria/ma_delete_all.c +++ b/storage/maria/ma_delete_all.c @@ -30,6 +30,7 @@ int maria_delete_all_rows(MARIA_HA *info) { DBUG_RETURN(my_errno=EACCES); } + /* LOCKTODO take X-lock on table here */ if (_ma_readinfo(info,F_WRLCK,1)) DBUG_RETURN(my_errno); if (_ma_mark_file_changed(info)) @@ -53,9 +54,23 @@ int maria_delete_all_rows(MARIA_HA *info) since it was locked then there may be key blocks in the key cache */ flush_key_blocks(share->key_cache, share->kfile, FLUSH_IGNORE_CHANGED); + /* + RECOVERYTODO Log the two chsize and header modifications and force the + log. So that if crash between the two chsize, we finish the work at + Recovery. For this scenario: + "TRUNCATE TABLE t1; DROP TABLE t1; RENAME TABLE t2 to t1; crash;" + Recovery mustn't truncate the new t1, so the log records of TRUNCATE + should be applied only if t1 exists and its ZeroDirtyPagesLSN is smaller + than the records'. See more comments below. + */ if (my_chsize(info->dfile, 0, 0, MYF(MY_WME)) || my_chsize(share->kfile, share->base.keystart, 0, MYF(MY_WME)) ) goto err; + /* + RECOVERYTODO Consider updating ZeroDirtyPagesLSN here. It is + not a necessity (it is one only in RENAME commands) but an optional + optimization which will allow some REDO skipping at Recovery. + */ VOID(_ma_writeinfo(info,WRITEINFO_UPDATE_KEYFILE)); #ifdef HAVE_MMAP /* Resize mmaped area */ @@ -63,14 +78,25 @@ int maria_delete_all_rows(MARIA_HA *info) _ma_remap_file(info, (my_off_t)0); rw_unlock(&info->s->mmap_lock); #endif + /* + RECOVERYTODO Until we have the TRUNCATE log record and take it into + account for log-low-water-mark calculation and use it in Recovery, we need + to sync. + */ + if (_ma_sync_table_files(info)) + goto err; allow_break(); /* Allow SIGHUP & SIGINT */ DBUG_RETURN(0); err: { int save_errno=my_errno; + /* RECOVERYTODO log the header modifications */ VOID(_ma_writeinfo(info,WRITEINFO_UPDATE_KEYFILE)); info->update|=HA_STATE_WRITTEN; /* Buffer changed */ + /* RECOVERYTODO until we log above we have to sync */ + if (_ma_sync_table_files(info) && !save_errno) + save_errno= my_errno; allow_break(); /* Allow SIGHUP & SIGINT */ DBUG_RETURN(my_errno=save_errno); } diff --git a/storage/maria/ma_delete_table.c b/storage/maria/ma_delete_table.c index dd781a93fc4..5c7b4337b20 100644 --- a/storage/maria/ma_delete_table.c +++ b/storage/maria/ma_delete_table.c @@ -31,6 +31,7 @@ int maria_delete_table(const char *name) #ifdef EXTRA_DEBUG _ma_check_table_is_closed(name,"delete"); #endif + /* LOCKTODO take X-lock on table here */ #ifdef USE_RAID { MARIA_HA *info; @@ -59,12 +60,22 @@ int maria_delete_table(const char *name) #endif /* USE_RAID */ fn_format(from,name,"",MARIA_NAME_IEXT,MY_UNPACK_FILENAME|MY_APPEND_EXT); - if (my_delete_with_symlink(from, MYF(MY_WME))) + /* + RECOVERYTODO log the two deletes below. + Then do the file deletions. + For this log record to be of any use for Recovery, we need the upper MySQL + layer to be crash-safe in DDLs; when it is we should reconsider the moment + of writing this log record, how to use it in Recovery, and force the log. + For now this record is only informative. + */ + if (my_delete_with_symlink(from, MYF(MY_WME | MY_SYNC_DIR))) DBUG_RETURN(my_errno); fn_format(from,name,"",MARIA_NAME_DEXT,MY_UNPACK_FILENAME|MY_APPEND_EXT); #ifdef USE_RAID if (raid_type) - DBUG_RETURN(my_raid_delete(from, raid_chunks, MYF(MY_WME)) ? my_errno : 0); + DBUG_RETURN(my_raid_delete(from, raid_chunks, MYF(MY_WME | MY_SYNC_DIR)) ? + my_errno : 0); #endif - DBUG_RETURN(my_delete_with_symlink(from, MYF(MY_WME)) ? my_errno : 0); + DBUG_RETURN(my_delete_with_symlink(from, MYF(MY_WME | MY_SYNC_DIR)) ? + my_errno : 0); } diff --git a/storage/maria/ma_extra.c b/storage/maria/ma_extra.c index 57e540242b9..1f649a00753 100644 --- a/storage/maria/ma_extra.c +++ b/storage/maria/ma_extra.c @@ -316,9 +316,7 @@ int maria_extra(MARIA_HA *info, enum ha_extra_function function, void *extra_arg if (share->not_flushed) { share->not_flushed=0; - if (my_sync(share->kfile, MYF(0))) - error= my_errno; - if (my_sync(info->dfile, MYF(0))) + if (_ma_sync_table_files(info)) error= my_errno; if (error) { @@ -439,3 +437,10 @@ int maria_reset(MARIA_HA *info) HA_STATE_PREV_FOUND); DBUG_RETURN(error); } + + +int _ma_sync_table_files(const MARIA_HA *info) +{ + return (my_sync(info->dfile, MYF(0)) || + my_sync(info->s->kfile, MYF(0))); +} diff --git a/storage/maria/ma_locking.c b/storage/maria/ma_locking.c index 5689d57f2a5..848fb7e9682 100644 --- a/storage/maria/ma_locking.c +++ b/storage/maria/ma_locking.c @@ -103,9 +103,7 @@ int maria_lock_database(MARIA_HA *info, int lock_type) share->changed=0; if (maria_flush) { - if (my_sync(share->kfile, MYF(0))) - error= my_errno; - if (my_sync(info->dfile, MYF(0))) + if (_ma_sync_table_files(info)) error= my_errno; } else diff --git a/storage/maria/ma_rename.c b/storage/maria/ma_rename.c index 5f65cd2b213..5d89cc063d7 100644 --- a/storage/maria/ma_rename.c +++ b/storage/maria/ma_rename.c @@ -23,6 +23,7 @@ int maria_rename(const char *old_name, const char *new_name) { char from[FN_REFLEN],to[FN_REFLEN]; + int data_file_rename_error; #ifdef USE_RAID uint raid_type=0,raid_chunks=0; #endif @@ -32,6 +33,7 @@ int maria_rename(const char *old_name, const char *new_name) _ma_check_table_is_closed(old_name,"rename old_table"); _ma_check_table_is_closed(new_name,"rename new table2"); #endif + /* LOCKTODO take X-lock on table here */ #ifdef USE_RAID { MARIA_HA *info; @@ -48,14 +50,40 @@ int maria_rename(const char *old_name, const char *new_name) fn_format(from,old_name,"",MARIA_NAME_IEXT,MY_UNPACK_FILENAME|MY_APPEND_EXT); fn_format(to,new_name,"",MARIA_NAME_IEXT,MY_UNPACK_FILENAME|MY_APPEND_EXT); - if (my_rename_with_symlink(from, to, MYF(MY_WME))) + /* + RECOVERYTODO log the two renames below. Update + ZeroDirtyPagesLSN of the table on disk (=> sync the files), this is + needed so that Recovery does not pick a wrong table. + Then do the file renames. + For this log record to be of any use for Recovery, we need the upper MySQL + layer to be crash-safe in DDLs; when it is we should reconsider the moment + of writing this log record, how to use it in Recovery, and force the log. + For now this record is only informative. But ZeroDirtyPagesLSN is + critically needed! + */ + if (my_rename_with_symlink(from, to, MYF(MY_WME | MY_SYNC_DIR))) DBUG_RETURN(my_errno); fn_format(from,old_name,"",MARIA_NAME_DEXT,MY_UNPACK_FILENAME|MY_APPEND_EXT); fn_format(to,new_name,"",MARIA_NAME_DEXT,MY_UNPACK_FILENAME|MY_APPEND_EXT); #ifdef USE_RAID if (raid_type) - DBUG_RETURN(my_raid_rename(from, to, raid_chunks, MYF(MY_WME)) ? my_errno : - 0); + data_file_rename_error= my_raid_rename(from, to, raid_chunks, + MYF(MY_WME | MY_SYNC_DIR)); + else #endif - DBUG_RETURN(my_rename_with_symlink(from, to,MYF(MY_WME)) ? my_errno : 0); + data_file_rename_error= + my_rename_with_symlink(from, to, MYF(MY_WME | MY_SYNC_DIR)); + if (data_file_rename_error) + { + /* + now we have a renamed index file and a non-renamed data file, try to + undo the rename of the index file. + */ + data_file_rename_error= my_errno; + fn_format(from, old_name, "", MARIA_NAME_IEXT, MYF(MY_UNPACK_FILENAME|MY_APPEND_EXT)); + fn_format(to, new_name, "", MARIA_NAME_IEXT, MYF(MY_UNPACK_FILENAME|MY_APPEND_EXT)); + my_rename_with_symlink(to, from, MYF(MY_WME | MY_SYNC_DIR)); + } + DBUG_RETURN(data_file_rename_error); + } diff --git a/storage/maria/ma_test3.c b/storage/maria/ma_test3.c index 96b896b03c6..2f205c33b12 100644 --- a/storage/maria/ma_test3.c +++ b/storage/maria/ma_test3.c @@ -65,6 +65,10 @@ int main(int argc,char **argv) MY_INIT(argv[0]); get_options(argc,argv); + fprintf(stderr, "WARNING! this program is to test 'external locking'" + " (when several processes share a table through file locking)" + " which is not supported by Maria at all; expect errors." + " We may soon remove this program.\n"); maria_init(); bzero((char*) keyinfo,sizeof(keyinfo)); bzero((char*) recinfo,sizeof(recinfo)); diff --git a/storage/maria/maria_def.h b/storage/maria/maria_def.h index 506bdbc71ca..62c50187888 100644 --- a/storage/maria/maria_def.h +++ b/storage/maria/maria_def.h @@ -741,3 +741,4 @@ int _ma_flush_blocks(HA_CHECK *param, KEY_CACHE *key_cache, File file); int _ma_sort_write_record(MARIA_SORT_PARAM *sort_param); int _ma_create_index_by_sort(MARIA_SORT_PARAM *info, my_bool no_messages, ulong); +int _ma_sync_table_files(const MARIA_HA *info); -- cgit v1.2.1 From 7a5d0c2e6efbffd95998b723e52a22de51c21dae Mon Sep 17 00:00:00 2001 From: unknown Date: Mon, 27 Nov 2006 22:09:06 +0100 Subject: Maria - Fixes to help some pushbuild hosts compile. include/Makefile.am: put my_bit.h in the tarball (need it for mf_keycache.c) mysys/lf_alloc-pin.c: // in .c file mysys/lf_hash.c: // in .c file --- include/Makefile.am | 2 +- mysys/lf_alloc-pin.c | 2 +- mysys/lf_hash.c | 3 ++- 3 files changed, 4 insertions(+), 3 deletions(-) diff --git a/include/Makefile.am b/include/Makefile.am index d5e75d31593..dd7c25e91f4 100644 --- a/include/Makefile.am +++ b/include/Makefile.am @@ -23,7 +23,7 @@ pkginclude_HEADERS = my_dbug.h m_string.h my_sys.h my_list.h my_xml.h \ my_getopt.h sslopt-longopts.h my_dir.h typelib.h \ sslopt-vars.h sslopt-case.h sql_common.h keycache.h \ mysql_time.h mysql/plugin.h $(BUILT_SOURCES) -noinst_HEADERS = config-win.h config-netware.h lf.h \ +noinst_HEADERS = config-win.h config-netware.h lf.h my_bit.h \ heap.h maria.h myisamchk.h my_bitmap.h my_uctype.h \ myisam.h myisampack.h myisammrg.h ft_global.h\ mysys_err.h my_base.h help_start.h help_end.h \ diff --git a/mysys/lf_alloc-pin.c b/mysys/lf_alloc-pin.c index 43055766c3e..d0fa29ddaaf 100644 --- a/mysys/lf_alloc-pin.c +++ b/mysys/lf_alloc-pin.c @@ -1,4 +1,4 @@ -// TODO multi-pinbox +#warning TODO multi-pinbox /* Copyright (C) 2000 MySQL AB This program is free software; you can redistribute it and/or modify diff --git a/mysys/lf_hash.c b/mysys/lf_hash.c index ff0eb8326d5..7e61ef690c6 100644 --- a/mysys/lf_hash.c +++ b/mysys/lf_hash.c @@ -85,7 +85,8 @@ retry: { if (!cursor->curr) return 0; - do { // XXX or goto retry ? + do { +#warning XXX or goto retry ? link= cursor->curr->link; cursor->next= PTR(link); _lf_pin(pins, 0, cursor->next); -- cgit v1.2.1 -- cgit v1.2.1 From 3345ed80a569bbd96fb9e0f3d9a0d5e1c0e1b3b7 Mon Sep 17 00:00:00 2001 From: unknown Date: Thu, 30 Nov 2006 12:14:36 +0100 Subject: Maria - pagecache unit tests: add test_file.h to the tarball unittest/mysys/Makefile.am: add test_file.h in the tarball --- unittest/mysys/Makefile.am | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/unittest/mysys/Makefile.am b/unittest/mysys/Makefile.am index 9b230272329..3ef1c3af587 100644 --- a/unittest/mysys/Makefile.am +++ b/unittest/mysys/Makefile.am @@ -17,8 +17,8 @@ noinst_PROGRAMS = bitmap-t base64-t my_atomic-t \ # tests for mysys/mf_pagecache.c -mf_pagecache_single_src = mf_pagecache_single.c $(top_srcdir)/mysys/mf_pagecache.c test_file.c -mf_pagecache_consist_src = mf_pagecache_consist.c $(top_srcdir)/mysys/mf_pagecache.c test_file.c +mf_pagecache_single_src = mf_pagecache_single.c $(top_srcdir)/mysys/mf_pagecache.c test_file.c test_file.h +mf_pagecache_consist_src = mf_pagecache_consist.c $(top_srcdir)/mysys/mf_pagecache.c test_file.c test_file.h mf_pagecache_common_cppflags = -DEXTRA_DEBUG -DPAGECACHE_DEBUG -DMAIN mf_pagecache_single_1k_t_SOURCES = $(mf_pagecache_single_src) -- cgit v1.2.1 From 9fcc34b4ae1bdcf8aeb4dad0295b7d3534491a17 Mon Sep 17 00:00:00 2001 From: unknown Date: Thu, 30 Nov 2006 15:12:28 +0100 Subject: Maria - misc fixes: don't run tests depending on the generic lock manager which will be removed; don't run page cache unit tests by default (too intensive). storage/maria/unittest/Makefile.am: don't run tests depending on the generic lock manager (it will be removed in the end), one of them crashes. unittest/mysys/Makefile.am: page cache tests put a too high load, causes problems on shared machines; so we still build them but give them a suffix so that they are not run by the default "test-unit" Makefile target. --- storage/maria/unittest/Makefile.am | 4 ++- unittest/mysys/Makefile.am | 68 ++++++++++++++++++++------------------ 2 files changed, 39 insertions(+), 33 deletions(-) diff --git a/storage/maria/unittest/Makefile.am b/storage/maria/unittest/Makefile.am index d0b247d65e1..969529db9c9 100644 --- a/storage/maria/unittest/Makefile.am +++ b/storage/maria/unittest/Makefile.am @@ -25,5 +25,7 @@ LDADD= $(top_builddir)/unittest/mytap/libmytap.a \ $(top_builddir)/mysys/libmysys.a \ $(top_builddir)/dbug/libdbug.a \ $(top_builddir)/strings/libmystrings.a @ZLIB_LIBS@ -noinst_PROGRAMS = ma_control_file-t trnman-t lockman-t lockman1-t lockman2-t +noinst_PROGRAMS = ma_control_file-t trnman-t lockman2-t +# the generic lock manager may not be used in the end and lockman1-t crashes, +# so we don't build lockman-t and lockman1-t CLEANFILES = maria_control diff --git a/unittest/mysys/Makefile.am b/unittest/mysys/Makefile.am index 3ef1c3af587..8dd0416dbac 100644 --- a/unittest/mysys/Makefile.am +++ b/unittest/mysys/Makefile.am @@ -7,13 +7,17 @@ LDADD = $(top_builddir)/unittest/mytap/libmytap.a \ $(top_builddir)/dbug/libdbug.a \ $(top_builddir)/strings/libmystrings.a +# tests with the "big" suffix are too big to run for every push, +# cause load problems on shared machines. So we build them, +# but don't run them by default (for this, a non "-t" suffix is used). +# You can run them by hand (how: see "test" target in upper dir). noinst_PROGRAMS = bitmap-t base64-t my_atomic-t \ - mf_pagecache_single_1k-t mf_pagecache_single_8k-t \ - mf_pagecache_single_64k-t \ - mf_pagecache_consist_1k-t mf_pagecache_consist_64k-t \ - mf_pagecache_consist_1kHC-t mf_pagecache_consist_64kHC-t \ - mf_pagecache_consist_1kRD-t mf_pagecache_consist_64kRD-t \ - mf_pagecache_consist_1kWR-t mf_pagecache_consist_64kWR-t + mf_pagecache_single_1k-t-big mf_pagecache_single_8k-t-big \ + mf_pagecache_single_64k-t-big \ + mf_pagecache_consist_1k-t-big mf_pagecache_consist_64k-t-big \ + mf_pagecache_consist_1kHC-t-big mf_pagecache_consist_64kHC-t-big \ + mf_pagecache_consist_1kRD-t-big mf_pagecache_consist_64kRD-t-big \ + mf_pagecache_consist_1kWR-t-big mf_pagecache_consist_64kWR-t-big # tests for mysys/mf_pagecache.c @@ -21,31 +25,31 @@ mf_pagecache_single_src = mf_pagecache_single.c $(top_srcdir)/mysys/mf_pagecache mf_pagecache_consist_src = mf_pagecache_consist.c $(top_srcdir)/mysys/mf_pagecache.c test_file.c test_file.h mf_pagecache_common_cppflags = -DEXTRA_DEBUG -DPAGECACHE_DEBUG -DMAIN -mf_pagecache_single_1k_t_SOURCES = $(mf_pagecache_single_src) -mf_pagecache_single_8k_t_SOURCES = $(mf_pagecache_single_src) -mf_pagecache_single_64k_t_SOURCES = $(mf_pagecache_single_src) -mf_pagecache_single_1k_t_CPPFLAGS = $(mf_pagecache_common_cppflags) -DPAGE_SIZE=1024 -mf_pagecache_single_8k_t_CPPFLAGS = $(mf_pagecache_common_cppflags) -DPAGE_SIZE=8192 -mf_pagecache_single_64k_t_CPPFLAGS = $(mf_pagecache_common_cppflags) -DPAGE_SIZE=65536 - -mf_pagecache_consist_1k_t_SOURCES = $(mf_pagecache_consist_src) -mf_pagecache_consist_1k_t_CPPFLAGS = $(mf_pagecache_common_cppflags) -DPAGE_SIZE=1024 -mf_pagecache_consist_64k_t_SOURCES = $(mf_pagecache_consist_src) -mf_pagecache_consist_64k_t_CPPFLAGS = $(mf_pagecache_common_cppflags) -DPAGE_SIZE=65536 - -mf_pagecache_consist_1kHC_t_SOURCES = $(mf_pagecache_consist_src) -mf_pagecache_consist_1kHC_t_CPPFLAGS = $(mf_pagecache_common_cppflags) -DPAGE_SIZE=1024 -DTEST_HIGH_CONCURENCY -mf_pagecache_consist_64kHC_t_SOURCES = $(mf_pagecache_consist_src) -mf_pagecache_consist_64kHC_t_CPPFLAGS = $(mf_pagecache_common_cppflags) -DPAGE_SIZE=65536 -DTEST_HIGH_CONCURENCY - -mf_pagecache_consist_1kRD_t_SOURCES = $(mf_pagecache_consist_src) -mf_pagecache_consist_1kRD_t_CPPFLAGS = $(mf_pagecache_common_cppflags) -DPAGE_SIZE=1024 -DTEST_READERS -mf_pagecache_consist_64kRD_t_SOURCES = $(mf_pagecache_consist_src) -mf_pagecache_consist_64kRD_t_CPPFLAGS = $(mf_pagecache_common_cppflags) -DPAGE_SIZE=65536 -DTEST_READERS - -mf_pagecache_consist_1kWR_t_SOURCES = $(mf_pagecache_consist_src) -mf_pagecache_consist_1kWR_t_CPPFLAGS = $(mf_pagecache_common_cppflags) -DPAGE_SIZE=1024 -DTEST_WRITERS -mf_pagecache_consist_64kWR_t_SOURCES = $(mf_pagecache_consist_src) -mf_pagecache_consist_64kWR_t_CPPFLAGS = $(mf_pagecache_common_cppflags) -DPAGE_SIZE=65536 -DTEST_WRITERS +mf_pagecache_single_1k_t_big_SOURCES = $(mf_pagecache_single_src) +mf_pagecache_single_8k_t_big_SOURCES = $(mf_pagecache_single_src) +mf_pagecache_single_64k_t_big_SOURCES = $(mf_pagecache_single_src) +mf_pagecache_single_1k_t_big_CPPFLAGS = $(mf_pagecache_common_cppflags) -DPAGE_SIZE=1024 +mf_pagecache_single_8k_t_big_CPPFLAGS = $(mf_pagecache_common_cppflags) -DPAGE_SIZE=8192 +mf_pagecache_single_64k_t_big_CPPFLAGS = $(mf_pagecache_common_cppflags) -DPAGE_SIZE=65536 + +mf_pagecache_consist_1k_t_big_SOURCES = $(mf_pagecache_consist_src) +mf_pagecache_consist_1k_t_big_CPPFLAGS = $(mf_pagecache_common_cppflags) -DPAGE_SIZE=1024 +mf_pagecache_consist_64k_t_big_SOURCES = $(mf_pagecache_consist_src) +mf_pagecache_consist_64k_t_big_CPPFLAGS = $(mf_pagecache_common_cppflags) -DPAGE_SIZE=65536 + +mf_pagecache_consist_1kHC_t_big_SOURCES = $(mf_pagecache_consist_src) +mf_pagecache_consist_1kHC_t_big_CPPFLAGS = $(mf_pagecache_common_cppflags) -DPAGE_SIZE=1024 -DTEST_HIGH_CONCURENCY +mf_pagecache_consist_64kHC_t_big_SOURCES = $(mf_pagecache_consist_src) +mf_pagecache_consist_64kHC_t_big_CPPFLAGS = $(mf_pagecache_common_cppflags) -DPAGE_SIZE=65536 -DTEST_HIGH_CONCURENCY + +mf_pagecache_consist_1kRD_t_big_SOURCES = $(mf_pagecache_consist_src) +mf_pagecache_consist_1kRD_t_big_CPPFLAGS = $(mf_pagecache_common_cppflags) -DPAGE_SIZE=1024 -DTEST_READERS +mf_pagecache_consist_64kRD_t_big_SOURCES = $(mf_pagecache_consist_src) +mf_pagecache_consist_64kRD_t_big_CPPFLAGS = $(mf_pagecache_common_cppflags) -DPAGE_SIZE=65536 -DTEST_READERS + +mf_pagecache_consist_1kWR_t_big_SOURCES = $(mf_pagecache_consist_src) +mf_pagecache_consist_1kWR_t_big_CPPFLAGS = $(mf_pagecache_common_cppflags) -DPAGE_SIZE=1024 -DTEST_WRITERS +mf_pagecache_consist_64kWR_t_big_SOURCES = $(mf_pagecache_consist_src) +mf_pagecache_consist_64kWR_t_big_CPPFLAGS = $(mf_pagecache_common_cppflags) -DPAGE_SIZE=65536 -DTEST_WRITERS CLEANFILES = my_pagecache_debug.log page_cache_test_file_1 -- cgit v1.2.1 From c86be6303bbc9d85c564876c1f9507ab634b442b Mon Sep 17 00:00:00 2001 From: unknown Date: Thu, 30 Nov 2006 15:43:15 +0100 Subject: Maria - fixes for gcc -ansi (no //) storage/maria/lockman.c: no // storage/maria/trnman.c: no // --- storage/maria/lockman.c | 11 ++++++----- storage/maria/trnman.c | 3 ++- 2 files changed, 8 insertions(+), 6 deletions(-) diff --git a/storage/maria/lockman.c b/storage/maria/lockman.c index 7a6b97b3d51..54ea95c6b61 100644 --- a/storage/maria/lockman.c +++ b/storage/maria/lockman.c @@ -1,6 +1,6 @@ -// TODO - allocate everything from dynarrays !!! (benchmark) -// TODO instant duration locks -// automatically place S instead of LS if possible +#warning TODO - allocate everything from dynarrays !!! (benchmark) +#warning TODO instant duration locks +#warning automatically place S instead of LS if possible /* Copyright (C) 2006 MySQL AB This program is free software; you can redistribute it and/or modify @@ -217,7 +217,8 @@ typedef struct lockman_lock { uint64 resource; struct lockman_lock *lonext; intptr volatile link; - uint32 hashnr; // TODO - remove hashnr from LOCK + uint32 hashnr; +#warning TODO - remove hashnr from LOCK uint16 loid; uchar lock; /* sizeof(uchar) <= sizeof(enum) */ uchar flags; @@ -429,7 +430,7 @@ static int lockinsert(LOCK * volatile *head, LOCK *node, LF_PINS *pins, cursor.upgrade_from->flags|= IGNORE_ME; #warning is this OK ? if a reader has already read upgrade_from, \ it may find it conflicting with node :( -//#error another bug - see the last test from test_lockman_simple() +#warning another bug - see the last test from test_lockman_simple() } } while (res == REPEAT_ONCE_MORE); diff --git a/storage/maria/trnman.c b/storage/maria/trnman.c index 1df4c67b4aa..4399f0e1208 100644 --- a/storage/maria/trnman.c +++ b/storage/maria/trnman.c @@ -338,7 +338,8 @@ void trnman_end_trn(TRN *trn, my_bool commit) those lists, and thus nobody may want to free them. Now we don't need a mutex to access free_me list */ - while (free_me) // XXX send them to the purge thread + while (free_me) +#warning XXX send them to the purge thread { TRN *t= free_me; free_me= free_me->next; -- cgit v1.2.1 From 8993dab52dbacff5cc220b80e1679cd99dc2235c Mon Sep 17 00:00:00 2001 From: unknown Date: Thu, 30 Nov 2006 21:12:17 +0100 Subject: Maria - fixes for gcc -ansi storage/maria/unittest/lockman-t.c: no // storage/maria/unittest/lockman1-t.c: no // storage/maria/unittest/lockman2-t.c: no // --- storage/maria/unittest/lockman-t.c | 2 +- storage/maria/unittest/lockman1-t.c | 2 +- storage/maria/unittest/lockman2-t.c | 2 +- 3 files changed, 3 insertions(+), 3 deletions(-) diff --git a/storage/maria/unittest/lockman-t.c b/storage/maria/unittest/lockman-t.c index df8be054ba3..8c0f71175e7 100644 --- a/storage/maria/unittest/lockman-t.c +++ b/storage/maria/unittest/lockman-t.c @@ -18,7 +18,7 @@ lockman for row and table locks */ -//#define EXTRA_VERBOSE +/* #define EXTRA_VERBOSE */ #include diff --git a/storage/maria/unittest/lockman1-t.c b/storage/maria/unittest/lockman1-t.c index cf4791067dc..41a1f0fd2f4 100644 --- a/storage/maria/unittest/lockman1-t.c +++ b/storage/maria/unittest/lockman1-t.c @@ -18,7 +18,7 @@ lockman for row locks, tablockman for table locks */ -//#define EXTRA_VERBOSE +/* #define EXTRA_VERBOSE */ #include diff --git a/storage/maria/unittest/lockman2-t.c b/storage/maria/unittest/lockman2-t.c index 18c3072b241..2a8090ab9ac 100644 --- a/storage/maria/unittest/lockman2-t.c +++ b/storage/maria/unittest/lockman2-t.c @@ -18,7 +18,7 @@ tablockman for row and table locks */ -//#define EXTRA_VERBOSE +/* #define EXTRA_VERBOSE */ #include -- cgit v1.2.1 From b72903fe54cf83c3dcbec7e3532e23bf946249ba Mon Sep 17 00:00:00 2001 From: unknown Date: Sun, 3 Dec 2006 17:06:27 +0100 Subject: Maria: * merging changes done to the key cache since May 2006 into Maria * enabling two small enough page cache's unit tests by default * fix to have non-buffered output in unit tests (to not have a false timeout killing in pushbuild) (patch given by Serg) * removing some warnings of gcc -ansi include/lf.h: getting rid of "warning: ISO C does not allow extra `;' outside of a function" (gcc -ansi) mysys/lf_hash.c: getting rid of "warning: ISO C does not allow extra `;' outside of a function" (gcc -ansi) mysys/mf_pagecache.c: Cosmetic changes to minimize the diff with the key cache. #define PAGECACHE_DEBUG_LOG is not needed (just define PAGECACHE_DEBUG if you want) (this change removes "warning: 'pagecache_debug_print' declared `static' but never defined"). Importing changes made to mf_keycache.c since May 2006, into the page cache. Disabling online resizing in the page cache. Fix for "warning: ISO C90 forbids mixed declarations and code". unittest/mysys/Makefile.am: Of the page cache's unit tests, two are small enough to run on pushbuild, renaming them to a -t suffix. unittest/mytap/tap.c: pushbuild kills a test after seeing no output from it for 10 minutes; so we set the mytap framework to not buffer output (patch given by Serg) so that output is seen more frequently and not "all at the end of the test". --- include/lf.h | 14 +++++----- mysys/lf_hash.c | 2 +- mysys/mf_pagecache.c | 66 +++++++++++++++++++++++++--------------------- unittest/mysys/Makefile.am | 15 ++++++----- unittest/mytap/tap.c | 2 ++ 5 files changed, 55 insertions(+), 44 deletions(-) diff --git a/include/lf.h b/include/lf.h index c7198ab75b6..ba29b6e9ca5 100644 --- a/include/lf.h +++ b/include/lf.h @@ -80,14 +80,14 @@ void lf_dynarray_destroy(LF_DYNARRAY *array); nolock_wrap(lf_dynarray_value, void *, (LF_DYNARRAY *array, uint idx), - (array, idx)); + (array, idx)) lock_wrap(lf_dynarray_lvalue, void *, (LF_DYNARRAY *array, uint idx), (array, idx), - &array->lock); + &array->lock) nolock_wrap(lf_dynarray_iterate, int, (LF_DYNARRAY *array, lf_dynarray_func func, void *arg), - (array, func, arg)); + (array, func, arg)) /* pin manager for memory allocator, lf_alloc-pin.c @@ -165,15 +165,15 @@ void lf_pinbox_destroy(LF_PINBOX *pinbox); lock_wrap(lf_pinbox_get_pins, LF_PINS *, (LF_PINBOX *pinbox), (pinbox), - &pinbox->pinstack.lock); + &pinbox->pinstack.lock) lock_wrap_void(lf_pinbox_put_pins, (LF_PINS *pins), (pins), - &pins->pinbox->pinstack.lock); + &pins->pinbox->pinstack.lock) lock_wrap_void(lf_pinbox_free, (LF_PINS *pins, void *addr), (pins, addr), - &pins->pinbox->pinstack.lock); + &pins->pinbox->pinstack.lock) /* memory allocator, lf_alloc-pin.c @@ -208,7 +208,7 @@ uint lf_alloc_in_pool(LF_ALLOCATOR *allocator); lock_wrap(lf_alloc_new, void *, (LF_PINS *pins), (pins), - &pins->pinbox->pinstack.lock); + &pins->pinbox->pinstack.lock) /* extendible hash, lf_hash.c diff --git a/mysys/lf_hash.c b/mysys/lf_hash.c index 7e61ef690c6..e13c5f64f54 100644 --- a/mysys/lf_hash.c +++ b/mysys/lf_hash.c @@ -28,7 +28,7 @@ #include #include -LF_REQUIRE_PINS(3); +LF_REQUIRE_PINS(3) /* An element of the list */ typedef struct { diff --git a/mysys/mf_pagecache.c b/mysys/mf_pagecache.c index 4b92f68d9bf..fe2d827ab50 100755 --- a/mysys/mf_pagecache.c +++ b/mysys/mf_pagecache.c @@ -43,9 +43,9 @@ #include #include "my_static.h" #include +#include #include #include -#include /* @@ -86,8 +86,6 @@ #define PAGECACHE_DEBUG_LOG "my_pagecache_debug.log" */ -#define PAGECACHE_DEBUG_LOG "my_pagecache_debug.log" - /* In key cache we have external raw locking here we use SERIALIZED_READ_FROM_CACHE to avoid problem of reading @@ -115,14 +113,6 @@ /* TODO: put it to my_static.c */ my_bool my_disable_flush_pagecache_blocks= 0; -#if defined(MSDOS) && !defined(M_IC80386) -/* we nead much memory */ -#undef my_malloc_lock -#undef my_free_lock -#define my_malloc_lock(A,B) halloc((long) (A/IO_SIZE),IO_SIZE) -#define my_free_lock(A,B) hfree(A) -#endif /* defined(MSDOS) && !defined(M_IC80386) */ - #define STRUCT_PTR(TYPE, MEMBER, a) \ (TYPE *) ((char *) (a) - offsetof(TYPE, MEMBER)) @@ -314,8 +304,8 @@ my_bool info_check_pin(PAGECACHE_BLOCK_LINK *block, enum pagecache_page_pin mode) { struct st_my_thread_var *thread= my_thread_var; - DBUG_ENTER("info_check_pin"); PAGECACHE_PIN_INFO *info= info_find(block->pin_list, thread); + DBUG_ENTER("info_check_pin"); if (info) { if (mode == PAGECACHE_PIN_LEFT_UNPINNED) @@ -372,10 +362,10 @@ my_bool info_check_lock(PAGECACHE_BLOCK_LINK *block, enum pagecache_page_pin pin) { struct st_my_thread_var *thread= my_thread_var; - DBUG_ENTER("info_check_lock"); PAGECACHE_LOCK_INFO *info= (PAGECACHE_LOCK_INFO *) info_find((PAGECACHE_PIN_INFO *) block->lock_list, thread); + DBUG_ENTER("info_check_lock"); switch(lock) { case PAGECACHE_LOCK_LEFT_UNLOCKED: @@ -605,15 +595,9 @@ uint pagecache_fwrite(PAGECACHE *pagecache, next_power(value) is 2 at the power of (1+floor(log2(value))); e.g. next_power(2)=4, next_power(3)=4. */ -static uint next_power(uint value) +static inline uint next_power(uint value) { - uint old_value= 1; - while (value) - { - old_value= value; - value&= value-1; - } - return (old_value << 1); + return (uint) my_round_up_to_next_power((uint32) value) << 1; } @@ -834,15 +818,24 @@ err: The function starts the operation only when all other threads performing operations with the key cache let her to proceed (when cnt_for_resize=0). -*/ + Before being usable, this function needs: + - to receive fixes for BUG#17332 "changing key_buffer_size on a running + server can crash under load" similar to those done to the key cache + - to have us (Sanja) look at the additional constraints placed on + resizing, due to the page locking specific to this page cache. + So we disable it for now. +*/ +#if 0 /* keep disabled until code is fixed see above !! */ int resize_pagecache(PAGECACHE *pagecache, my_size_t use_mem, uint division_limit, uint age_threshold) { int blocks; +#ifdef THREAD struct st_my_thread_var *thread; PAGECACHE_WQUEUE *wqueue; +#endif DBUG_ENTER("resize_pagecache"); if (!pagecache->inited) @@ -909,6 +902,7 @@ finish: pagecache_pthread_mutex_unlock(&pagecache->cache_lock); DBUG_RETURN(blocks); } +#endif /* 0 */ /* @@ -1504,8 +1498,12 @@ static inline void remove_reader(PAGECACHE_BLOCK_LINK *block) { DBUG_ENTER("remove_reader"); BLOCK_INFO(block); +#ifdef THREAD if (! --block->hash_link->requests && block->condvar) pagecache_pthread_cond_signal(block->condvar); +#else + --block->hash_link->requests; +#endif DBUG_VOID_RETURN; } @@ -1515,7 +1513,8 @@ static inline void remove_reader(PAGECACHE_BLOCK_LINK *block) signals on its termination */ -static inline void wait_for_readers(PAGECACHE *pagecache, +static inline void wait_for_readers(PAGECACHE *pagecache + __attribute__((unused)), PAGECACHE_BLOCK_LINK *block) { #ifdef THREAD @@ -1684,7 +1683,6 @@ static PAGECACHE_HASH_LINK *get_hash_link(PAGECACHE *pagecache, { reg1 PAGECACHE_HASH_LINK *hash_link; PAGECACHE_HASH_LINK **start; - PAGECACHE_PAGE page; KEYCACHE_DBUG_PRINT("get_hash_link", ("fd: %u pos: %lu", (uint) file->file, (ulong) pageno)); @@ -1710,6 +1708,7 @@ restart: #ifdef THREAD /* Wait for a free hash link */ struct st_my_thread_var *thread= my_thread_var; + PAGECACHE_PAGE page; KEYCACHE_DBUG_PRINT("get_hash_link", ("waiting")); page.file= *file; page.pageno= pageno; @@ -2053,8 +2052,10 @@ restart: /* Remove the hash link for this page from the hash table */ unlink_hash(pagecache, block->hash_link); /* All pending requests for this page must be resubmitted */ +#ifdef THREAD if (block->wqueue[COND_FOR_SAVED].last_thread) release_queue(&block->wqueue[COND_FOR_SAVED]); +#endif } link_to_file_list(pagecache, block, file, (my_bool)(block->hash_link ? 1 : 0)); @@ -2209,10 +2210,10 @@ my_bool pagecache_lock_block(PAGECACHE *pagecache, BLOCK_INFO(block); while (block->status & BLOCK_WRLOCK) { - DBUG_PRINT("info", ("fail to lock, waiting...")); /* Lock failed we will wait */ #ifdef THREAD struct st_my_thread_var *thread= my_thread_var; + DBUG_PRINT("info", ("fail to lock, waiting...")); add_to_queue(&block->wqueue[COND_FOR_WRLOCK], thread); dec_counter_for_resize_op(pagecache); do @@ -2403,8 +2404,10 @@ static void read_block(PAGECACHE *pagecache, KEYCACHE_DBUG_PRINT("read_block", ("primary request: new page in cache")); /* Signal that all pending requests for this page now can be processed */ +#ifdef THREAD if (block->wqueue[COND_FOR_REQUESTED].last_thread) release_queue(&block->wqueue[COND_FOR_REQUESTED]); +#endif } else { @@ -3210,9 +3213,11 @@ restart: block->status= (BLOCK_READ | (block->status & BLOCK_WRLOCK)); KEYCACHE_DBUG_PRINT("key_cache_insert", ("primary request: new page in cache")); +#ifdef THREAD /* Signal that all pending requests for this now can be processed. */ if (block->wqueue[COND_FOR_REQUESTED].last_thread) release_queue(&block->wqueue[COND_FOR_REQUESTED]); +#endif } } else @@ -3223,11 +3228,9 @@ restart: if (block->status & BLOCK_CHANGED) link_to_file_list(pagecache, block, &block->hash_link->file, 1); } - else - { - if (! (block->status & BLOCK_CHANGED)) + else if (! (block->status & BLOCK_CHANGED)) link_to_changed_list(pagecache, block); - } + if (! (block->status & BLOCK_ERROR)) { bmove512(block->buffer, buff, pagecache->block_size); @@ -3342,9 +3345,11 @@ static void free_block(PAGECACHE *pagecache, PAGECACHE_BLOCK_LINK *block) /* Keep track of the number of currently unused blocks. */ pagecache->blocks_unused++; +#ifdef THREAD /* All pending requests for this page must be resubmitted. */ if (block->wqueue[COND_FOR_SAVED].last_thread) release_queue(&block->wqueue[COND_FOR_SAVED]); +#endif } @@ -3438,12 +3443,14 @@ static int flush_cached_blocks(PAGECACHE *pagecache, if (!last_errno) last_errno= errno ? errno : -1; } +#ifdef THREAD /* Let to proceed for possible waiting requests to write to the block page. It might happen only during an operation to resize the key cache. */ if (block->wqueue[COND_FOR_SAVED].last_thread) release_queue(&block->wqueue[COND_FOR_SAVED]); +#endif /* type will never be FLUSH_IGNORE_CHANGED here */ if (! (type == FLUSH_KEEP || type == FLUSH_FORCE_WRITE)) { @@ -3970,7 +3977,6 @@ static void ___pagecache_pthread_mutex_unlock(pthread_mutex_t *mutex) { KEYCACHE_THREAD_TRACE_END(""); pthread_mutex_unlock(mutex); - return; } diff --git a/unittest/mysys/Makefile.am b/unittest/mysys/Makefile.am index 8dd0416dbac..f7b72255a72 100644 --- a/unittest/mysys/Makefile.am +++ b/unittest/mysys/Makefile.am @@ -12,7 +12,7 @@ LDADD = $(top_builddir)/unittest/mytap/libmytap.a \ # but don't run them by default (for this, a non "-t" suffix is used). # You can run them by hand (how: see "test" target in upper dir). noinst_PROGRAMS = bitmap-t base64-t my_atomic-t \ - mf_pagecache_single_1k-t-big mf_pagecache_single_8k-t-big \ + mf_pagecache_single_1k-t mf_pagecache_single_8k-t \ mf_pagecache_single_64k-t-big \ mf_pagecache_consist_1k-t-big mf_pagecache_consist_64k-t-big \ mf_pagecache_consist_1kHC-t-big mf_pagecache_consist_64kHC-t-big \ @@ -25,11 +25,14 @@ mf_pagecache_single_src = mf_pagecache_single.c $(top_srcdir)/mysys/mf_pagecache mf_pagecache_consist_src = mf_pagecache_consist.c $(top_srcdir)/mysys/mf_pagecache.c test_file.c test_file.h mf_pagecache_common_cppflags = -DEXTRA_DEBUG -DPAGECACHE_DEBUG -DMAIN -mf_pagecache_single_1k_t_big_SOURCES = $(mf_pagecache_single_src) -mf_pagecache_single_8k_t_big_SOURCES = $(mf_pagecache_single_src) +# Those two take reasonable resources (<100 MB) for < 1 minute +mf_pagecache_single_1k_t_SOURCES = $(mf_pagecache_single_src) +mf_pagecache_single_8k_t_SOURCES = $(mf_pagecache_single_src) +mf_pagecache_single_1k_t_CPPFLAGS = $(mf_pagecache_common_cppflags) -DPAGE_SIZE=1024 +mf_pagecache_single_8k_t_CPPFLAGS = $(mf_pagecache_common_cppflags) -DPAGE_SIZE=8192 + +# All others below are big mf_pagecache_single_64k_t_big_SOURCES = $(mf_pagecache_single_src) -mf_pagecache_single_1k_t_big_CPPFLAGS = $(mf_pagecache_common_cppflags) -DPAGE_SIZE=1024 -mf_pagecache_single_8k_t_big_CPPFLAGS = $(mf_pagecache_common_cppflags) -DPAGE_SIZE=8192 mf_pagecache_single_64k_t_big_CPPFLAGS = $(mf_pagecache_common_cppflags) -DPAGE_SIZE=65536 mf_pagecache_consist_1k_t_big_SOURCES = $(mf_pagecache_consist_src) @@ -52,4 +55,4 @@ mf_pagecache_consist_1kWR_t_big_CPPFLAGS = $(mf_pagecache_common_cppflags) -DPA mf_pagecache_consist_64kWR_t_big_SOURCES = $(mf_pagecache_consist_src) mf_pagecache_consist_64kWR_t_big_CPPFLAGS = $(mf_pagecache_common_cppflags) -DPAGE_SIZE=65536 -DTEST_WRITERS -CLEANFILES = my_pagecache_debug.log page_cache_test_file_1 +CLEANFILES = pagecache_debug.log page_cache_test_file_1 diff --git a/unittest/mytap/tap.c b/unittest/mytap/tap.c index 17ec51863d9..51330ad820b 100644 --- a/unittest/mytap/tap.c +++ b/unittest/mytap/tap.c @@ -150,6 +150,8 @@ static signal_entry install_signal[]= { void plan(int const count) { + + setvbuf(tapout, 0, _IONBF, 0); /* provide output at once */ /* Install signal handler */ -- cgit v1.2.1 From 5750daa4cf73eba0a6f1d9909844bf106228ae80 Mon Sep 17 00:00:00 2001 From: unknown Date: Mon, 4 Dec 2006 15:31:04 +0100 Subject: Maria - post-review change of "fixes for gcc -ansi". And reducing the time taken by my_atomic-t. include/lf.h: fix to be able to add a ";" at the end of the macro's invokation (removing it here, removes a warning from "gcc -ansi" about a standalone ";"). mysys/lf_hash.c: ";" is ok now after LF_REQUIRE_PINS mysys/mf_pagecache.c: comment fix unittest/mysys/my_atomic-t.c: decreasing number of iterations to make test take less time. --- include/lf.h | 2 +- mysys/lf_hash.c | 2 +- mysys/mf_pagecache.c | 2 +- unittest/mysys/my_atomic-t.c | 4 ++-- 4 files changed, 5 insertions(+), 5 deletions(-) diff --git a/include/lf.h b/include/lf.h index ba29b6e9ca5..c161bd5044d 100644 --- a/include/lf.h +++ b/include/lf.h @@ -136,7 +136,7 @@ typedef struct { #if defined(__GNUC__) && defined(MY_LF_EXTRA_DEBUG) #define LF_REQUIRE_PINS(N) \ static const char require_pins[LF_PINBOX_PINS-N]; \ - static const int LF_NUM_PINS_IN_THIS_FILE= N; + static const int LF_NUM_PINS_IN_THIS_FILE= N #define _lf_pin(PINS, PIN, ADDR) \ ( \ assert(PIN < LF_NUM_PINS_IN_THIS_FILE), \ diff --git a/mysys/lf_hash.c b/mysys/lf_hash.c index e13c5f64f54..7e61ef690c6 100644 --- a/mysys/lf_hash.c +++ b/mysys/lf_hash.c @@ -28,7 +28,7 @@ #include #include -LF_REQUIRE_PINS(3) +LF_REQUIRE_PINS(3); /* An element of the list */ typedef struct { diff --git a/mysys/mf_pagecache.c b/mysys/mf_pagecache.c index fe2d827ab50..3e3484d5efb 100755 --- a/mysys/mf_pagecache.c +++ b/mysys/mf_pagecache.c @@ -970,7 +970,7 @@ void change_pagecache_param(PAGECACHE *pagecache, uint division_limit, /* - Flushes and removes page cache from memory + Removes page cache from memory. Does NOT flush pages to disk. SYNOPSIS end_pagecache() diff --git a/unittest/mysys/my_atomic-t.c b/unittest/mysys/my_atomic-t.c index 314335944d3..02e07acd67b 100644 --- a/unittest/mysys/my_atomic-t.c +++ b/unittest/mysys/my_atomic-t.c @@ -275,9 +275,9 @@ int main() &my_charset_bin); #ifdef MY_ATOMIC_MODE_RWLOCKS -#define CYCLES 10000 +#define CYCLES 3000 #else -#define CYCLES 1000000 +#define CYCLES 300000 #endif #define THREADS 100 -- cgit v1.2.1 From ad29d5520b1ba379a75adc447f301851ff4588a4 Mon Sep 17 00:00:00 2001 From: unknown Date: Thu, 7 Dec 2006 15:23:50 +0100 Subject: Maria - fix for "statement with no effect" warning mysys/lf_hash.c: fix for "statement with no effect" warning storage/maria/lockman.c: fix for "statement with no effect" warning --- mysys/lf_hash.c | 4 ++-- storage/maria/lockman.c | 4 ++-- 2 files changed, 4 insertions(+), 4 deletions(-) diff --git a/mysys/lf_hash.c b/mysys/lf_hash.c index 7e61ef690c6..b2ad7a93ace 100644 --- a/mysys/lf_hash.c +++ b/mysys/lf_hash.c @@ -96,7 +96,7 @@ retry: cur_keylen= cursor->curr->keylen; if (*cursor->prev != (intptr)cursor->curr) { - LF_BACKOFF; + (void)LF_BACKOFF; goto retry; } if (!DELETED(link)) @@ -118,7 +118,7 @@ retry: _lf_alloc_free(pins, cursor->curr); else { - LF_BACKOFF; + (void)LF_BACKOFF; goto retry; } } diff --git a/storage/maria/lockman.c b/storage/maria/lockman.c index 54ea95c6b61..7672fadb068 100644 --- a/storage/maria/lockman.c +++ b/storage/maria/lockman.c @@ -287,7 +287,7 @@ retry: cur_flags= cursor->curr->flags; if (*cursor->prev != (intptr)cursor->curr) { - LF_BACKOFF; + (void)LF_BACKOFF; goto retry; } if (!DELETED(link)) @@ -364,7 +364,7 @@ retry: _lf_alloc_free(pins, cursor->curr); else { - LF_BACKOFF; + (void)LF_BACKOFF; goto retry; } } -- cgit v1.2.1 From fa05e9c9f426a19f016897ec57c047c277bf52c7 Mon Sep 17 00:00:00 2001 From: unknown Date: Sat, 16 Dec 2006 18:10:47 +0100 Subject: WL#3071 - Maria checkpoint Adding rec_lsn to Maria's page cache. Misc fixes to Checkpoint. mysys/mf_pagecache.c: adding rec_lsn, the LSN when a page first became dirty. It is set when unlocking a page (TODO: should also be set when the unlocking is an implicit part of pagecache_write()). It is reset in link_to_file_list() and free_block() (one of which is used every time we flush a block). It is a ulonglong and not LSN, because its destination is comparisons for which ulonglong is better than a struct. storage/maria/ma_checkpoint.c: misc fixes to Checkpoint (updates now that the transaction manager and the page cache are more known) storage/maria/ma_close.c: an important note for the future. storage/maria/ma_least_recently_dirtied.c: comment --- mysys/mf_pagecache.c | 12 +- storage/maria/ma_checkpoint.c | 433 ++++++++++++++++++------------ storage/maria/ma_close.c | 6 + storage/maria/ma_least_recently_dirtied.c | 5 +- 4 files changed, 277 insertions(+), 179 deletions(-) diff --git a/mysys/mf_pagecache.c b/mysys/mf_pagecache.c index 3e3484d5efb..807a3ea520a 100755 --- a/mysys/mf_pagecache.c +++ b/mysys/mf_pagecache.c @@ -295,6 +295,7 @@ struct st_pagecache_block_link enum pagecache_page_type type; /* type of the block */ uint hits_left; /* number of hits left until promotion */ ulonglong last_hit_time; /* timestamp of the last hit */ + ulonglong rec_lsn; /* LSN when first became dirty */ KEYCACHE_CONDVAR *condvar; /* condition variable for 'no readers' event */ }; @@ -1202,6 +1203,7 @@ static void link_to_file_list(PAGECACHE *pagecache, if (block->status & BLOCK_CHANGED) { block->status&= ~BLOCK_CHANGED; + block->rec_lsn= 0; pagecache->blocks_changed--; pagecache->global_blocks_changed--; } @@ -2509,6 +2511,8 @@ void pagecache_unlock_page(PAGECACHE *pagecache, DBUG_ASSERT(lock == PAGECACHE_LOCK_WRITE_UNLOCK && pin == PAGECACHE_UNPIN); /* TODO: insert LSN writing code */ + DBUG_ASSERT(first_REDO_LSN_for_page > 0); + set_if_bigger(block->rec_lsn, first_REDO_LSN_for_page); } #ifndef DBUG_OFF @@ -2671,6 +2675,8 @@ void pagecache_unlock(PAGECACHE *pagecache, DBUG_ASSERT(lock == PAGECACHE_LOCK_WRITE_UNLOCK && pin == PAGECACHE_UNPIN); /* TODO: insert LSN writing code */ + DBUG_ASSERT(first_REDO_LSN_for_page > 0); + set_if_bigger(block->rec_lsn, first_REDO_LSN_for_page); } #ifndef DBUG_OFF @@ -3012,10 +3018,9 @@ restart: pagecache->blocks_changed--; pagecache->global_blocks_changed--; /* - free_block() will change the status of the block so no need to change - it here. + free_block() will change the status and rec_lsn of the block so no + need to change them here. */ - } /* Cache is locked, so we can relese page before freeing it */ pagecache_make_lock_and_pin(pagecache, block, @@ -3328,6 +3333,7 @@ static void free_block(PAGECACHE *pagecache, PAGECACHE_BLOCK_LINK *block) #ifndef DBUG_OFF block->type= PAGECACHE_EMPTY_PAGE; #endif + block->rec_lsn= 0; KEYCACHE_THREAD_TRACE("free block"); KEYCACHE_DBUG_PRINT("free_block", ("block is freed")); diff --git a/storage/maria/ma_checkpoint.c b/storage/maria/ma_checkpoint.c index 83312ce37b8..717b6202559 100644 --- a/storage/maria/ma_checkpoint.c +++ b/storage/maria/ma_checkpoint.c @@ -40,8 +40,7 @@ #include "share.h" #include "log.h" -/* could also be called LSN_ERROR */ -#define LSN_IMPOSSIBLE ((LSN)0) +#define LSN_IMPOSSIBLE ((LSN)0) /* could also be called LSN_ERROR */ #define LSN_MAX ((LSN)ULONGLONG_MAX) /* @@ -57,9 +56,12 @@ st_transaction system_trans= {0 /* long trans id */, 0 /* short trans id */,0,.. MEDIUM checkpoint. */ LSN max_rec_lsn_at_last_checkpoint= 0; +/* last submitted checkpoint request; cleared only when executed */ CHECKPOINT_LEVEL next_asynchronous_checkpoint_to_do= NONE; CHECKPOINT_LEVEL synchronous_checkpoint_in_progress= NONE; +static inline ulonglong read_non_atomic(ulonglong volatile *x); + /* Used by MySQL client threads requesting a checkpoint (like "ALTER MARIA ENGINE DO CHECKPOINT"), and probably by maria_panic(), and at the end of the @@ -67,6 +69,7 @@ CHECKPOINT_LEVEL synchronous_checkpoint_in_progress= NONE; */ my_bool execute_synchronous_checkpoint(CHECKPOINT_LEVEL level) { + my_bool result; DBUG_ENTER("execute_synchronous_checkpoint"); DBUG_ASSERT(level > NONE); @@ -76,43 +79,52 @@ my_bool execute_synchronous_checkpoint(CHECKPOINT_LEVEL level) wait_on_checkpoint_done_cond(); synchronous_checkpoint_in_progress= level; - execute_checkpoint(level); + result= execute_checkpoint(level); safemutex_assert_owner(log_mutex); synchronous_checkpoint_in_progress= NONE; unlock(log_mutex); broadcast(checkpoint_done_cond); + DBUG_RETURN(result); } -/* Picks a checkpoint request, if there is one, and executes it */ +/* + If no checkpoint is running, and there is a pending asynchronous checkpoint + request, executes it. + Is safe if multiple threads call it, though in first version only one will. + It's intended to be used by a thread which regularly calls this function; + this is why, if there is a request,it does not wait in a loop for + synchronous checkpoints to be finished, but just exits (because the thread + may want to do something useful meanwhile (flushing dirty pages for example) + instead of waiting). +*/ my_bool execute_asynchronous_checkpoint_if_any() { + my_bool result; CHECKPOINT_LEVEL level; DBUG_ENTER("execute_asynchronous_checkpoint"); lock(log_mutex); - if (likely(next_asynchronous_checkpoint_to_do == NONE)) + if (likely((next_asynchronous_checkpoint_to_do == NONE) || + (synchronous_checkpoint_in_progress != NONE))) { unlock(log_mutex); DBUG_RETURN(FALSE); } - while (synchronous_checkpoint_in_progress) - wait_on_checkpoint_done_cond(); - -do_checkpoint: level= next_asynchronous_checkpoint_to_do; DBUG_ASSERT(level > NONE); - execute_checkpoint(level); + result= execute_checkpoint(level); safemutex_assert_owner(log_mutex); - if (next_asynchronous_checkpoint_to_do > level) - goto do_checkpoint; /* one more request was posted */ - else + /* If only one thread calls this function, "<" can never happen below */ + if (next_asynchronous_checkpoint_to_do <= level) { - DBUG_ASSERT(next_asynchronous_checkpoint_to_do == level); - next_asynchronous_checkpoint_to_do= NONE; /* all work done */ + /* it's our request or weaker/equal ones, all work is done */ + next_asynchronous_checkpoint_to_do= NONE; } + /* otherwise if it is a stronger request, we'll deal with it at next call */ unlock(log_mutex); broadcast(checkpoint_done_cond); + DBUG_RETURN(result); } @@ -123,17 +135,14 @@ do_checkpoint: */ my_bool execute_checkpoint(CHECKPOINT_LEVEL level) { - LSN candidate_max_rec_lsn_at_last_checkpoint; - /* to avoid { lock + no-op + unlock } in the common (==indirect) case */ - my_bool need_log_mutex; - DBUG_ENTER("execute_checkpoint"); safemutex_assert_owner(log_mutex); - copy_of_max_rec_lsn_at_last_checkpoint= max_rec_lsn_at_last_checkpoint; - if (unlikely(need_log_mutex= (level > INDIRECT))) + if (unlikely(level > INDIRECT)) { + LSN copy_of_max_rec_lsn_at_last_checkpoint= + max_rec_lsn_at_last_checkpoint; /* much I/O work to do, release log mutex */ unlock(log_mutex); @@ -149,51 +158,29 @@ my_bool execute_checkpoint(CHECKPOINT_LEVEL level) flush all pages which were already dirty at last checkpoint: ensures that recovery will never start from before the next-to-last checkpoint (two-checkpoint rule). - It is max, not min as the WL says (TODO update WL). */ flush_all_LRD_to_lsn(copy_of_max_rec_lsn_at_last_checkpoint); /* this will go full speed (normal scheduling, no sleep) */ break; } + lock(log_mutex); } - candidate_max_rec_lsn_at_last_checkpoint= checkpoint_indirect(need_log_mutex); - - lock(log_mutex); - /* - this portion cannot be done as a hook in write_log_record() for the - LOGREC_CHECKPOINT type because: - - at that moment we still have not written to the control file so cannot - mark the request as done; this could be solved by writing to the control - file in the hook but that would be an I/O under the log's mutex, bad. - - it would not be nice organisation of code (I tried it :). - */ - if (candidate_max_rec_lsn_at_last_checkpoint != LSN_IMPOSSIBLE) - { - /* checkpoint succeeded */ - maximum_rec_lsn_last_checkpoint= candidate_max_rec_lsn_at_last_checkpoint; - written_since_last_checkpoint= (my_off_t)0; - DBUG_RETURN(FALSE); - } /* - keep mutex locked because callers will want to clear mutex-protected - status variables + keep mutex locked upon exit because callers will want to clear + mutex-protected status variables */ - DBUG_RETURN(TRUE); + DBUG_RETURN(execute_checkpoint_indirect()); } /* Does an indirect checpoint (collects data from data structures, writes into a checkpoint log record). - Returns the largest LSN of the LRD when the checkpoint happened (this is a - fuzzy definition), or LSN_IMPOSSIBLE on error. That LSN is used for the - "two-checkpoint rule" (MEDIUM checkpoints). + Starts and ends while having log's mutex (released in the middle). */ -LSN checkpoint_indirect(my_bool need_log_mutex) +my_bool execute_checkpoint_indirect() { - DBUG_ENTER("checkpoint_indirect"); - int error= 0; /* checkpoint record data: */ LSN checkpoint_start_lsn; @@ -202,163 +189,198 @@ LSN checkpoint_indirect(my_bool need_log_mutex) char *ptr; LSN checkpoint_lsn; LSN candidate_max_rec_lsn_at_last_checkpoint= 0; - list_element *el; /* to scan lists */ - ulong stored_LRD_size= 0; - + DBUG_ENTER("execute_checkpoint_indirect"); DBUG_ASSERT(sizeof(byte *) <= 8); DBUG_ASSERT(sizeof(LSN) <= 8); - if (need_log_mutex) - lock(log_mutex); /* maybe this will clash with log_read_end_lsn() */ + safemutex_assert_owner(log_mutex); checkpoint_start_lsn= log_read_end_lsn(); + if (LSN_IMPOSSIBLE == checkpoint_start_lsn) /* error */ + DBUG_RETURN(TRUE); unlock(log_mutex); DBUG_PRINT("info",("checkpoint_start_lsn %lu", checkpoint_start_lsn)); /* STEP 1: fetch information about dirty pages */ - - /* - We lock the entire cache but will be quick, just reading/writing a few MBs - of memory at most. - */ - pagecache_pthread_mutex_lock(&pagecache->cache_lock); - - /* - This is an over-estimation, as in theory blocks_changed may contain - non-PAGECACHE_LSN_PAGE pages, which we don't want to store into the - checkpoint record; the true number of page-LRD-info we'll store into the - record is stored_LRD_size. - */ - string1.length= 8+8+(8+8)*pagecache->blocks_changed; - if (NULL == (string1.str= my_malloc(string1.length))) - goto err; - ptr= string1.str; - int8store(ptr, checkpoint_start_lsn); - ptr+= 8+8; /* don't store stored_LRD_size now, wait */ - if (pagecache->blocks_changed > 0) + /* note: this piece will move into mysys/mf_pagecache.c */ { + ulong stored_LRD_size= 0; + /* + We lock the entire cache but will be quick, just reading/writing a few MBs + of memory at most. + When we enter here, we must be sure that no "first_in_switch" situation + is happening or will happen (either we have to get rid of + first_in_switch in the code or, first_in_switch has to increment a + "danger" counter for Checkpoint to know it has to wait. + */ + pagecache_pthread_mutex_lock(&pagecache->cache_lock); + + /* + This is an over-estimation, as in theory blocks_changed may contain + non-PAGECACHE_LSN_PAGE pages, which we don't want to store into the + checkpoint record; the true number of page-LRD-info we'll store into the + record is stored_LRD_size. + */ /* - There are different ways to scan the dirty blocks; - flush_all_key_blocks() uses a loop over pagecache->used_last->next_used, - and for each element of the loop, loops into - pagecache->changed_blocks[FILE_HASH(file of the element)]. - This has the drawback that used_last includes non-dirty blocks, and it's - two loops over many elements. Here we try something simpler. - If there are no blocks in changed_blocks[file_hash], we should hit - zeroes and skip them. + TODO: Ingo says blocks_changed is not a reliable number (see his + document); ask him. */ - uint file_hash; - for (file_hash= 0; file_hash < PAGECACHE_CHANGED_BLOCKS_HASH; file_hash++) + string1.length= 8+8+(8+8+8)*pagecache->blocks_changed; + if (NULL == (string1.str= my_malloc(string1.length))) + goto err; + ptr= string1.str; + int8store(ptr, checkpoint_start_lsn); + ptr+= 8+8; /* don't store stored_LRD_size now, wait */ + if (pagecache->blocks_changed > 0) { - PAGECACHE_BLOCK_LINK *block; - for (block= pagecache->changed_blocks[file_hash] ; - block; - block= block->next_changed) + uint file_hash; + for (file_hash= 0; file_hash < PAGECACHE_CHANGED_BLOCKS_HASH; file_hash++) { - DBUG_ASSERT(block->hash_link != NULL); - DBUG_ASSERT(block->status & BLOCK_CHANGED); - if (block->type != PAGECACHE_LSN_PAGE) + PAGECACHE_BLOCK_LINK *block; + for (block= pagecache->changed_blocks[file_hash] ; + block; + block= block->next_changed) { - /* no need to store it in the checkpoint record */ - continue; + DBUG_ASSERT(block->hash_link != NULL); + DBUG_ASSERT(block->status & BLOCK_CHANGED); + if (block->type != PAGECACHE_LSN_PAGE) + { + continue; /* no need to store it in the checkpoint record */ + } + /* + In the current pagecache, rec_lsn is not set correctly: + 1) it is set on pagecache_unlock(), too late (a page is dirty + (BLOCK_CHANGED) since the first pagecache_write()). It may however + be not too late, because until unlock(), the page's update is not + committed, so it's ok that REDOs for it be skipped at Recovery + (which is what happens with an unset rec_lsn). Note that this + relies on the assumption that a transaction never commits while + holding locks on pages. + 2) sometimes the unlocking can be an implicit action of + pagecache_write(), without any call to pagecache_unlock(), then + rec_lsn is not set. That one is a critical problem. + TODO: fix this when Monty has explained how he writes BLOB pages. + */ + if (0 == block->rec_lsn) + abort(); /* always fail in all builds, in case it's problem 2) */ + + int8store(ptr, block->hash_link->file.file); + ptr+= 8; + int8store(ptr, block->hash_link->pageno); + ptr+= 8; + int8store(ptr, block->rec_lsn); + ptr+= 8; + stored_LRD_size++; + DBUG_ASSERT(stored_LRD_size <= pagecache->blocks_changed); + set_if_bigger(candidate_max_rec_lsn_at_last_checkpoint, + block->rec_lsn); } - /* Q: two "block"s cannot have the same "hash_link", right? */ - int8store(ptr, block->hash_link->pageno); - ptr+= 8; - /* I assume rec_lsn will be member of "block", not of "hash_link" */ - int8store(ptr, block->rec_lsn); - ptr+= 8; - stored_LRD_size++; - set_if_bigger(candidate_max_rec_lsn_at_last_checkpoint, - block->rec_lsn); } - } - pagecache_pthread_mutex_unlock(&pagecache->cache_lock); - DBUG_ASSERT(stored_LRD_size <= pagecache->blocks_changed); - int8store(string1.str+8, stored_LRD_size); - string1.length= 8+8+(8+8)*stored_LRD_size; + pagecache_pthread_mutex_unlock(&pagecache->cache_lock); + int8store(string1.str+8, stored_LRD_size); + string1.length= 8+8+(8+8+8)*stored_LRD_size; + } /* STEP 2: fetch information about transactions */ - + /* note: this piece will move into trnman.c */ /* - If trx are in more than one list (e.g. three: - running transactions, committed transactions, purge queue), we can either - take mutexes of all three together or do crabbing. - But if an element can move from list 1 to list 3 without passing through - list 2, crabbing is dangerous. - Hopefully it's ok to take 3 mutexes together... - Otherwise I'll have to make sure I miss no important trx and I handle dups. + Transactions are in the "active list" (protected by a mutex) and in a + lock-free hash of "committed" (insertion protected by the same mutex, + deletion lock-free). */ - lock(global_transactions_list_mutex); /* or 3 mutexes if there are 3 */ - string2.length= 8+(8+8)*trx_list->count; - if (NULL == (string2.str= my_malloc(string2.length))) - goto err; - ptr= string2.str; - int8store(ptr, trx_list->count); - ptr+= 8; - for (el= trx_list->first; el; el= el->next) { - /* possibly latch el.rwlock */ - *ptr= el->state; - ptr++; - int7store(ptr, el->long_trans_id); - ptr+= 7; - int2store(ptr, el->short_trans_id); - ptr+= 2; - int8store(ptr, el->undo_lsn); - ptr+= 8; - int8store(ptr, el->undo_purge_lsn); + TRN *trn; + ulong stored_trn_size= 0; + /* First, the active transactions */ + pthread_mutex_lock(LOCK_trn_list); + string2.length= 8+(7+2+8+8+8)*trnman_active_transactions; + if (NULL == (string2.str= my_malloc(string2.length))) + goto err; + ptr= string2.str; ptr+= 8; + for (trn= active_list_min.next; trn != &active_list_max; trn= trn->next) + { + /* we would latch trn.rwlock if it existed */ + if (0 == trn->short_trid) /* trn is not inited, skip */ + continue; + /* state is not needed for now (only when we have prepared trx) */ + /* int7store does not exist but mi_int7store does */ + int7store(ptr, trn->trid); + ptr+= 7; + int2store(ptr, trn->short_trid); + ptr+= 2; + int8store(ptr, trn->undo_lsn); /* is an LSN 7 or 8 bytes really? */ + ptr+= 8; + int8store(ptr, trn->undo_purge_lsn); + ptr+= 8; + int8store(ptr, read_non_atomic(&trn->first_undo_lsn)); + ptr+= 8; + /* possibly unlatch el.rwlock */ + stored_trn_size++; + } + pthread_mutex_unlock(LOCK_trn_list); /* - if no latch, use double variable of type ULONGLONG_CONSISTENT in - st_transaction, or even no need if Intel >=486 + Now the committed ones. + We need a function which scans the hash's list of elements in a + lock-free manner (a bit like lfind(), starting from bucket 0), and for + each node (committed transaction) stores the transaction's + information (trid, undo_purge_lsn, first_undo_lsn) into a buffer. + This big buffer is malloc'ed at the start, so the number of elements (or + an upper bound of it) found in the hash needs to be known in advance + (one solution is to keep LOCK_trn_list locked, ensuring that nodes are + only deleted). */ - int8store(ptr, el->first_undo_lsn); - ptr+= 8; - /* possibly unlatch el.rwlock */ + /* + TODO: if we see there exists no transaction (active and committed) we can + tell the lock-free structures to do some freeing (my_free()). + */ + int8store(string1.str, stored_trn_size); + string2.length= 8+(7+2+8+8+8)*stored_trn_size; } - unlock(global_transactions_list_mutex); /* STEP 3: fetch information about table files */ - /* This global mutex is in fact THR_LOCK_maria (see ma_open()) */ - lock(global_share_list_mutex); - string3.length= 8+(8+8)*share_list->count; - if (NULL == (string3.str= my_malloc(string3.length))) - goto err; - ptr= string3.str; - /* possibly latch each MARIA_SHARE, one by one, like this: */ - pthread_mutex_lock(&share->intern_lock); - /* - We'll copy the file id (a bit like share->kfile), the file name - (like share->unique_file_name[_length]). - */ - make_copy_of_global_share_list_to_array; - pthread_mutex_unlock(&share->intern_lock); - unlock(global_share_list_mutex); - - /* work on copy */ - int8store(ptr, elements_in_array); - ptr+= 8; - for (el in array) { - int8store(ptr, array[...].file_id); - ptr+= 8; - memcpy(ptr, array[...].file_name, ...); - ptr+= ...; + /* This global mutex is in fact THR_LOCK_maria (see ma_open()) */ + lock(global_share_list_mutex); + string3.length= 8+(8+8)*share_list->count; + if (NULL == (string3.str= my_malloc(string3.length))) + goto err; + ptr= string3.str; + /* possibly latch each MARIA_SHARE, one by one, like this: */ + pthread_mutex_lock(&share->intern_lock); /* - these two are long ops (involving disk I/O) that's why we copied the - list, to not keep the list locked for long: + We'll copy the file id (a bit like share->kfile), the file name + (like share->unique_file_name[_length]). */ - flush_bitmap_pages(el); - /* TODO: and also autoinc counter, logical file end, free page list */ + make_copy_of_global_share_list_to_array; + pthread_mutex_unlock(&share->intern_lock); + unlock(global_share_list_mutex); - /* - fsyncs the fd, that's the loooong operation (e.g. max 150 fsync per - second, so if you have touched 1000 files it's 7 seconds). - */ - force_file(el); + /* work on copy */ + int8store(ptr, elements_in_array); + ptr+= 8; + for (el in array) + { + int8store(ptr, array[...].file_id); + ptr+= 8; + memcpy(ptr, array[...].file_name, ...); + ptr+= ...; + /* + these two are long ops (involving disk I/O) that's why we copied the + list, to not keep the list locked for long: + */ + /* TODO: what if the table pointer is gone/reused now? */ + flush_bitmap_pages(el); + /* TODO: and also autoinc counter, logical file end, free page list */ + + /* + fsyncs the fd, that's the loooong operation (e.g. max 150 fsync per + second, so if you have touched 1000 files it's 7 seconds). + */ + force_file(el); + } } /* LAST STEP: now write the checkpoint log record */ @@ -389,11 +411,38 @@ err: candidate_max_rec_lsn_at_last_checkpoint= LSN_IMPOSSIBLE; end: + my_free(buffer1.str, MYF(MY_ALLOW_ZERO_PTR)); my_free(buffer2.str, MYF(MY_ALLOW_ZERO_PTR)); my_free(buffer3.str, MYF(MY_ALLOW_ZERO_PTR)); - DBUG_RETURN(candidate_max_rec_lsn_at_last_checkpoint); + /* + this portion cannot be done as a hook in write_log_record() for the + LOGREC_CHECKPOINT type because: + - at that moment we still have not written to the control file so cannot + mark the request as done; this could be solved by writing to the control + file in the hook but that would be an I/O under the log's mutex, bad. + - it would not be nice organisation of code (I tried it :). + */ + if (candidate_max_rec_lsn_at_last_checkpoint != LSN_IMPOSSIBLE) + { + /* checkpoint succeeded */ + /* + TODO: compute log's low water mark (how to do that with our fuzzy + ARIES-like reads of data structures? TODO think about it :). + */ + lock(log_mutex); + /* That LSN is used for the "two-checkpoint rule" (MEDIUM checkpoints) */ + maximum_rec_lsn_last_checkpoint= candidate_max_rec_lsn_at_last_checkpoint; + written_since_last_checkpoint= (my_off_t)0; + DBUG_RETURN(FALSE); + } + lock(log_mutex); + DBUG_RETURN(TRUE); + /* + keep mutex locked upon exit because callers will want to clear + mutex-protected status variables + */ } @@ -433,7 +482,7 @@ void request_asynchronous_checkpoint(CHECKPOINT_LEVEL level); safemutex_assert_owner(log_mutex); DBUG_ASSERT(level > NONE); - if (checkpoint_request < level) + if (next_asynchronous_checkpoint_to_do < level) { /* no equal or stronger running or to run, we post request */ /* @@ -445,7 +494,7 @@ void request_asynchronous_checkpoint(CHECKPOINT_LEVEL level); (see least_recently_dirtied.c) will notice our request in max a few seconds. */ - checkpoint_request= level; /* post request */ + next_asynchronous_checkpoint_to_do= level; /* post request */ } /* @@ -457,3 +506,37 @@ void request_asynchronous_checkpoint(CHECKPOINT_LEVEL level); the end user. */ } + + +/* + If a 64-bit variable transitions from both halves being zero to both halves + being non-zero, and never changes after that (like the transaction's + first_undo_lsn), this function can be used to do a read of it (without + mutex, without atomic load) which always produces a correct (though maybe + slightly old) value (even on 32-bit CPUs). +*/ +static inline ulonglong read_non_atomic(ulonglong volatile *x) +{ +#if ( SIZEOF_CHARP >= 8 ) + /* 64-bit CPU (right?), 64-bit reads are atomic */ + return *x; +#else + /* + 32-bit CPU, 64-bit reads may give a mixed of old half and new half (old + low bits and new high bits, or the contrary). + As the variable we read transitions from both halves being zero to both + halves being non-zero, and never changes then, we can detect atomicity + problems: + */ + ulonglong y; + for (;;) /* loop until no atomicity problems */ + { + y= *x; + if (likely(((0 == y) || + ((0 != (y >> 32)) && (0 != (y << 32))))) + return y; + /* Worth seeing it! */ + DBUG_PRINT("info",("atomicity problem")); + } +#endif +} diff --git a/storage/maria/ma_close.c b/storage/maria/ma_close.c index 5b940eaf4c3..73764cf444a 100644 --- a/storage/maria/ma_close.c +++ b/storage/maria/ma_close.c @@ -57,6 +57,12 @@ int maria_close(register MARIA_HA *info) info->opt_flag&= ~(READ_CACHE_USED | WRITE_CACHE_USED); } flag= !--share->reopen; + /* + RECOVERYTODO: + Below we are going to make the table unknown to future checkpoints, so it + needs to have fsync'ed itself entirely (bitmap, pages, etc) at this + point. + */ maria_open_list=list_delete(maria_open_list,&info->open_list); pthread_mutex_unlock(&share->intern_lock); diff --git a/storage/maria/ma_least_recently_dirtied.c b/storage/maria/ma_least_recently_dirtied.c index b0b7fb1ef10..809442b4e97 100644 --- a/storage/maria/ma_least_recently_dirtied.c +++ b/storage/maria/ma_least_recently_dirtied.c @@ -94,7 +94,10 @@ pthread_handler_decl background_flush_and_checkpoint_thread() while (this_thread_not_killed) { if ((flush_calls++) & ((2< Date: Sat, 16 Dec 2006 18:10:48 +0100 Subject: WL#3071 - Maria checkpoint. Correcting comment about a bad problem. storage/maria/ma_checkpoint.c: I was too optimistic; problem 1) is really a bad problem. --- storage/maria/ma_checkpoint.c | 24 +++++++++++++++--------- 1 file changed, 15 insertions(+), 9 deletions(-) diff --git a/storage/maria/ma_checkpoint.c b/storage/maria/ma_checkpoint.c index 717b6202559..608a6fb9fcd 100644 --- a/storage/maria/ma_checkpoint.c +++ b/storage/maria/ma_checkpoint.c @@ -212,7 +212,7 @@ my_bool execute_checkpoint_indirect() When we enter here, we must be sure that no "first_in_switch" situation is happening or will happen (either we have to get rid of first_in_switch in the code or, first_in_switch has to increment a - "danger" counter for Checkpoint to know it has to wait. + "danger" counter for Checkpoint to know it has to wait. TODO. */ pagecache_pthread_mutex_lock(&pagecache->cache_lock); @@ -251,19 +251,25 @@ my_bool execute_checkpoint_indirect() /* In the current pagecache, rec_lsn is not set correctly: 1) it is set on pagecache_unlock(), too late (a page is dirty - (BLOCK_CHANGED) since the first pagecache_write()). It may however - be not too late, because until unlock(), the page's update is not - committed, so it's ok that REDOs for it be skipped at Recovery - (which is what happens with an unset rec_lsn). Note that this - relies on the assumption that a transaction never commits while - holding locks on pages. + (BLOCK_CHANGED) since the first pagecache_write()). So in this + scenario: + thread1: thread2: + write_REDO + pagecache_write() + checkpoint : reclsn not known + pagecache_unlock(sets rec_lsn) + commit + crash, + at recovery we will wrongly skip the REDO. It also affects the + low-water mark's computation. 2) sometimes the unlocking can be an implicit action of pagecache_write(), without any call to pagecache_unlock(), then - rec_lsn is not set. That one is a critical problem. + rec_lsn is not set. + 1) and 2) are critical problems. TODO: fix this when Monty has explained how he writes BLOB pages. */ if (0 == block->rec_lsn) - abort(); /* always fail in all builds, in case it's problem 2) */ + abort(); /* always fail in all builds */ int8store(ptr, block->hash_link->file.file); ptr+= 8; -- cgit v1.2.1 From 7199c905590391f64802913369aab7d288eff4c8 Mon Sep 17 00:00:00 2001 From: unknown Date: Mon, 18 Dec 2006 17:24:02 +0100 Subject: WL#3071 Maria checkpoint - cleanups, simplifications - moving the construction of the "dirty pages table" into the pagecache where it belongs (because it's the pagecache which knows dirty pages). TODO: do the same soon for the "transactions table". - fix for a small bug in the pagecache (decrementation of "changed_blocks") include/pagecache.h: prototype mysys/mf_pagecache.c: m_string.h moves up for LEX_STRING to be known for pagecache.h. In pagecache_delete_page(), we must decrement "blocks_changed" even if we just delete the page without flushing it. A new function pagecache_collect_changed_blocks_with_LSN() (used by the Checkpoint module), which stores information about the changed blocks (a.k.a. "the dirty pages table") into a LEX_STRING. This function is not tested now, it will be when there is a Checkpoint. storage/maria/ma_checkpoint.c: refining the checkpoint code: factoring functions, moving the construction of the "dirty pages table" into mf_pagecache.c (I'll do the same with the construction of the "transactions table" once Serg tells me what's the best way to do it). storage/maria/ma_least_recently_dirtied.c: Simplifying the thread which does background flushing of least-recently-dirtied pages: - in first version that thread will not flush, just do checkpoints - in 2nd version, flushing should re-use existing page cache functions like flush_pagecache_blocks(). unittest/mysys/test_file.h: m_string.h moves up for LEX_STRING to be known in pagecache.h --- include/pagecache.h | 3 + mysys/mf_pagecache.c | 180 ++++++++++++++++++++++++----- storage/maria/ma_checkpoint.c | 179 +++++++---------------------- storage/maria/ma_least_recently_dirtied.c | 182 +++++------------------------- unittest/mysys/test_file.h | 2 +- 5 files changed, 231 insertions(+), 315 deletions(-) diff --git a/include/pagecache.h b/include/pagecache.h index 4d64070ad62..9f215325ae5 100644 --- a/include/pagecache.h +++ b/include/pagecache.h @@ -221,6 +221,9 @@ extern my_bool pagecache_delete_page(PAGECACHE *pagecache, enum pagecache_page_lock lock, my_bool flush); extern void end_pagecache(PAGECACHE *keycache, my_bool cleanup); +extern my_bool pagecache_collect_changed_blocks_with_LSN(PAGECACHE *pagecache, + LEX_STRING *str, + LSN *max_lsn); C_MODE_END #endif /* _keycache_h */ diff --git a/mysys/mf_pagecache.c b/mysys/mf_pagecache.c index 807a3ea520a..96c855fda0a 100755 --- a/mysys/mf_pagecache.c +++ b/mysys/mf_pagecache.c @@ -40,9 +40,9 @@ */ #include "mysys_priv.h" +#include #include #include "my_static.h" -#include #include #include #include @@ -295,7 +295,7 @@ struct st_pagecache_block_link enum pagecache_page_type type; /* type of the block */ uint hits_left; /* number of hits left until promotion */ ulonglong last_hit_time; /* timestamp of the last hit */ - ulonglong rec_lsn; /* LSN when first became dirty */ + LSN rec_lsn; /* LSN when first became dirty */ KEYCACHE_CONDVAR *condvar; /* condition variable for 'no readers' event */ }; @@ -2988,33 +2988,35 @@ restart: goto restart; } - if (block->status & BLOCK_CHANGED && flush) + if (block->status & BLOCK_CHANGED) { - /* The block contains a dirty page - push it out of the cache */ - - KEYCACHE_DBUG_PRINT("find_key_block", ("block is dirty")); - - pagecache_pthread_mutex_unlock(&pagecache->cache_lock); - /* - The call is thread safe because only the current - thread might change the block->hash_link value - */ - DBUG_ASSERT(block->pins == 1); - error= pagecache_fwrite(pagecache, - &block->hash_link->file, - block->buffer, - block->hash_link->pageno, - block->type, - MYF(MY_NABP | MY_WAIT_IF_FULL)); - pagecache_pthread_mutex_lock(&pagecache->cache_lock); - pagecache->global_cache_write++; - - if (error) + if (flush) { - block->status|= BLOCK_ERROR; - goto err; + /* The block contains a dirty page - push it out of the cache */ + + KEYCACHE_DBUG_PRINT("find_key_block", ("block is dirty")); + + pagecache_pthread_mutex_unlock(&pagecache->cache_lock); + /* + The call is thread safe because only the current + thread might change the block->hash_link value + */ + DBUG_ASSERT(block->pins == 1); + error= pagecache_fwrite(pagecache, + &block->hash_link->file, + block->buffer, + block->hash_link->pageno, + block->type, + MYF(MY_NABP | MY_WAIT_IF_FULL)); + pagecache_pthread_mutex_lock(&pagecache->cache_lock); + pagecache->global_cache_write++; + + if (error) + { + block->status|= BLOCK_ERROR; + goto err; + } } - pagecache->blocks_changed--; pagecache->global_blocks_changed--; /* @@ -3793,6 +3795,132 @@ int reset_key_cache_counters(const char *name, PAGECACHE *key_cache) } +/* + Allocates a buffer and stores in it some information about all dirty pages + of type PAGECACHE_LSN_PAGE. + + SYNOPSIS + pagecache_collect_changed_blocks_with_LSN() + pagecache pointer to the page cache + str (OUT) pointer to a LEX_STRING where the allocated buffer, and + its size, will be put + max_lsn (OUT) pointer to a LSN where the maximum rec_lsn of all + relevant dirty pages will be put + + DESCRIPTION + Does the allocation because the caller cannot know the size itself. + Memory freeing is done by the caller. + Ignores all pages of another type than PAGECACHE_LSN_PAGE, because they + are not interesting for a checkpoint record. + The caller has the intention of doing checkpoints. + + RETURN + 0 on success + 1 on error +*/ +my_bool pagecache_collect_changed_blocks_with_LSN(PAGECACHE *pagecache, + LEX_STRING *str, + LSN *max_lsn) +{ + my_bool error; + ulong stored_LRD_size= 0; + uint file_hash; + char *ptr; + DBUG_ENTER("pagecache_collect_changed_blocks_with_LSN"); + + *max_lsn= 0; + /* + We lock the entire cache but will be quick, just reading/writing a few MBs + of memory at most. + When we enter here, we must be sure that no "first_in_switch" situation + is happening or will happen (either we have to get rid of + first_in_switch in the code or, first_in_switch has to increment a + "danger" counter for this function to know it has to wait). TODO. + */ + pagecache_pthread_mutex_lock(&pagecache->cache_lock); + + /* Count how many dirty pages are interesting */ + for (file_hash= 0; file_hash < PAGECACHE_CHANGED_BLOCKS_HASH; file_hash++) + { + PAGECACHE_BLOCK_LINK *block; + for (block= pagecache->changed_blocks[file_hash] ; + block; + block= block->next_changed) + { + /* + Q: is there somthing subtle with block->hash_link: can it be NULL? + does it have to be == hash_link->block... ? + */ + DBUG_ASSERT(block->hash_link != NULL); + DBUG_ASSERT(block->status & BLOCK_CHANGED); + if (block->type != PAGECACHE_LSN_PAGE) + continue; /* no need to store it */ + /* + In the current pagecache, rec_lsn is not set correctly: + 1) it is set on pagecache_unlock(), too late (a page is dirty + (BLOCK_CHANGED) since the first pagecache_write()). So in this + scenario: + thread1: thread2: + write_REDO + pagecache_write() checkpoint : reclsn not known + pagecache_unlock(sets rec_lsn) + commit + crash, + at recovery we will wrongly skip the REDO. It also affects the + low-water mark's computation. + 2) sometimes the unlocking can be an implicit action of + pagecache_write(), without any call to pagecache_unlock(), then + rec_lsn is not set. + 1) and 2) are critical problems. + TODO: fix this when Monty has explained how he writes BLOB pages. + */ + if (0 == block->rec_lsn) + { + DBUG_ASSERT(0); + goto err; + } + stored_LRD_size++; + } + } + + str->length= 8+(4+4+8)*stored_LRD_size; + if (NULL == (str->str= my_malloc(str->length, MYF(MY_WME)))) + goto err; + ptr= str->str; + int8store(ptr, stored_LRD_size); + ptr+= 8; + if (0 == stored_LRD_size) + goto end; + for (file_hash= 0; file_hash < PAGECACHE_CHANGED_BLOCKS_HASH; file_hash++) + { + PAGECACHE_BLOCK_LINK *block; + for (block= pagecache->changed_blocks[file_hash] ; + block; + block= block->next_changed) + { + if (block->type != PAGECACHE_LSN_PAGE) + continue; /* no need to store it in the checkpoint record */ + DBUG_ASSERT((4 == sizeof(block->hash_link->file.file)) && + (4 == sizeof(block->hash_link->pageno))); + int4store(ptr, block->hash_link->file.file); + ptr+= 4; + int4store(ptr, block->hash_link->pageno); + ptr+= 4; + int8store(ptr, (ulonglong)block->rec_lsn); + ptr+= 8; + set_if_bigger(*max_lsn, block->rec_lsn); + } + } + error= 0; + goto end; +err: + error= 1; +end: + pagecache_pthread_mutex_unlock(&pagecache->cache_lock); + DBUG_RETURN(error); +} + + #ifndef DBUG_OFF /* Test if disk-cache is ok diff --git a/storage/maria/ma_checkpoint.c b/storage/maria/ma_checkpoint.c index 608a6fb9fcd..a1d094d7da1 100644 --- a/storage/maria/ma_checkpoint.c +++ b/storage/maria/ma_checkpoint.c @@ -56,9 +56,9 @@ st_transaction system_trans= {0 /* long trans id */, 0 /* short trans id */,0,.. MEDIUM checkpoint. */ LSN max_rec_lsn_at_last_checkpoint= 0; -/* last submitted checkpoint request; cleared only when executed */ +/* last submitted checkpoint request; cleared when starts */ CHECKPOINT_LEVEL next_asynchronous_checkpoint_to_do= NONE; -CHECKPOINT_LEVEL synchronous_checkpoint_in_progress= NONE; +CHECKPOINT_LEVEL checkpoint_in_progress= NONE; static inline ulonglong read_non_atomic(ulonglong volatile *x); @@ -74,16 +74,10 @@ my_bool execute_synchronous_checkpoint(CHECKPOINT_LEVEL level) DBUG_ASSERT(level > NONE); lock(log_mutex); - while ((synchronous_checkpoint_in_progress != NONE) || - (next_asynchronous_checkpoint_to_do != NONE)) + while (checkpoint_in_progress != NONE) wait_on_checkpoint_done_cond(); - synchronous_checkpoint_in_progress= level; result= execute_checkpoint(level); - safemutex_assert_owner(log_mutex); - synchronous_checkpoint_in_progress= NONE; - unlock(log_mutex); - broadcast(checkpoint_done_cond); DBUG_RETURN(result); } @@ -92,7 +86,7 @@ my_bool execute_synchronous_checkpoint(CHECKPOINT_LEVEL level) request, executes it. Is safe if multiple threads call it, though in first version only one will. It's intended to be used by a thread which regularly calls this function; - this is why, if there is a request,it does not wait in a loop for + this is why, if there is a request, it does not wait in a loop for synchronous checkpoints to be finished, but just exits (because the thread may want to do something useful meanwhile (flushing dirty pages for example) instead of waiting). @@ -103,27 +97,20 @@ my_bool execute_asynchronous_checkpoint_if_any() CHECKPOINT_LEVEL level; DBUG_ENTER("execute_asynchronous_checkpoint"); + /* first check without mutex, ok to see old data */ + if (likely((next_asynchronous_checkpoint_to_do == NONE) || + (checkpoint_in_progress != NONE))) + DBUG_RETURN(FALSE); + lock(log_mutex); if (likely((next_asynchronous_checkpoint_to_do == NONE) || - (synchronous_checkpoint_in_progress != NONE))) + (checkpoint_in_progress != NONE))) { unlock(log_mutex); DBUG_RETURN(FALSE); } - level= next_asynchronous_checkpoint_to_do; - DBUG_ASSERT(level > NONE); - result= execute_checkpoint(level); - safemutex_assert_owner(log_mutex); - /* If only one thread calls this function, "<" can never happen below */ - if (next_asynchronous_checkpoint_to_do <= level) - { - /* it's our request or weaker/equal ones, all work is done */ - next_asynchronous_checkpoint_to_do= NONE; - } - /* otherwise if it is a stronger request, we'll deal with it at next call */ - unlock(log_mutex); - broadcast(checkpoint_done_cond); + result= execute_checkpoint(next_asynchronous_checkpoint_to_do); DBUG_RETURN(result); } @@ -135,9 +122,13 @@ my_bool execute_asynchronous_checkpoint_if_any() */ my_bool execute_checkpoint(CHECKPOINT_LEVEL level) { + my_bool result; DBUG_ENTER("execute_checkpoint"); safemutex_assert_owner(log_mutex); + if (next_asynchronous_checkpoint_to_do <= level) + next_asynchronous_checkpoint_to_do= NONE; + checkpoint_in_progress= level; if (unlikely(level > INDIRECT)) { @@ -166,11 +157,11 @@ my_bool execute_checkpoint(CHECKPOINT_LEVEL level) lock(log_mutex); } - /* - keep mutex locked upon exit because callers will want to clear - mutex-protected status variables - */ - DBUG_RETURN(execute_checkpoint_indirect()); + result= execute_checkpoint_indirect(); + checkpoint_in_progress= NONE; + unlock(log_mutex); + broadcast(checkpoint_done_cond); + DBUG_RETURN(result); } @@ -181,114 +172,37 @@ my_bool execute_checkpoint(CHECKPOINT_LEVEL level) */ my_bool execute_checkpoint_indirect() { - int error= 0; + int error= 0, i; /* checkpoint record data: */ LSN checkpoint_start_lsn; - LEX_STRING string1={0,0}, string2={0,0}, string3={0,0}; - LEX_STRING *string_array[4]; + char checkpoint_start_lsn_char[8]; + LEX_STRING strings[5]={ {&checkpoint_start_lsn_str, 8}, {0,0}, {0,0}, {0,0}, {0,0} }; char *ptr; LSN checkpoint_lsn; - LSN candidate_max_rec_lsn_at_last_checkpoint= 0; + LSN candidate_max_rec_lsn_at_last_checkpoint; DBUG_ENTER("execute_checkpoint_indirect"); DBUG_ASSERT(sizeof(byte *) <= 8); DBUG_ASSERT(sizeof(LSN) <= 8); safemutex_assert_owner(log_mutex); + + /* STEP 1: record current end-of-log LSN */ checkpoint_start_lsn= log_read_end_lsn(); if (LSN_IMPOSSIBLE == checkpoint_start_lsn) /* error */ DBUG_RETURN(TRUE); unlock(log_mutex); DBUG_PRINT("info",("checkpoint_start_lsn %lu", checkpoint_start_lsn)); + int8store(strings[0].str, checkpoint_start_lsn); - /* STEP 1: fetch information about dirty pages */ - /* note: this piece will move into mysys/mf_pagecache.c */ - { - ulong stored_LRD_size= 0; - /* - We lock the entire cache but will be quick, just reading/writing a few MBs - of memory at most. - When we enter here, we must be sure that no "first_in_switch" situation - is happening or will happen (either we have to get rid of - first_in_switch in the code or, first_in_switch has to increment a - "danger" counter for Checkpoint to know it has to wait. TODO. - */ - pagecache_pthread_mutex_lock(&pagecache->cache_lock); + /* STEP 2: fetch information about dirty pages */ - /* - This is an over-estimation, as in theory blocks_changed may contain - non-PAGECACHE_LSN_PAGE pages, which we don't want to store into the - checkpoint record; the true number of page-LRD-info we'll store into the - record is stored_LRD_size. - */ - /* - TODO: Ingo says blocks_changed is not a reliable number (see his - document); ask him. - */ - string1.length= 8+8+(8+8+8)*pagecache->blocks_changed; - if (NULL == (string1.str= my_malloc(string1.length))) - goto err; - ptr= string1.str; - int8store(ptr, checkpoint_start_lsn); - ptr+= 8+8; /* don't store stored_LRD_size now, wait */ - if (pagecache->blocks_changed > 0) - { - uint file_hash; - for (file_hash= 0; file_hash < PAGECACHE_CHANGED_BLOCKS_HASH; file_hash++) - { - PAGECACHE_BLOCK_LINK *block; - for (block= pagecache->changed_blocks[file_hash] ; - block; - block= block->next_changed) - { - DBUG_ASSERT(block->hash_link != NULL); - DBUG_ASSERT(block->status & BLOCK_CHANGED); - if (block->type != PAGECACHE_LSN_PAGE) - { - continue; /* no need to store it in the checkpoint record */ - } - /* - In the current pagecache, rec_lsn is not set correctly: - 1) it is set on pagecache_unlock(), too late (a page is dirty - (BLOCK_CHANGED) since the first pagecache_write()). So in this - scenario: - thread1: thread2: - write_REDO - pagecache_write() - checkpoint : reclsn not known - pagecache_unlock(sets rec_lsn) - commit - crash, - at recovery we will wrongly skip the REDO. It also affects the - low-water mark's computation. - 2) sometimes the unlocking can be an implicit action of - pagecache_write(), without any call to pagecache_unlock(), then - rec_lsn is not set. - 1) and 2) are critical problems. - TODO: fix this when Monty has explained how he writes BLOB pages. - */ - if (0 == block->rec_lsn) - abort(); /* always fail in all builds */ - - int8store(ptr, block->hash_link->file.file); - ptr+= 8; - int8store(ptr, block->hash_link->pageno); - ptr+= 8; - int8store(ptr, block->rec_lsn); - ptr+= 8; - stored_LRD_size++; - DBUG_ASSERT(stored_LRD_size <= pagecache->blocks_changed); - set_if_bigger(candidate_max_rec_lsn_at_last_checkpoint, - block->rec_lsn); - } - } - pagecache_pthread_mutex_unlock(&pagecache->cache_lock); - int8store(string1.str+8, stored_LRD_size); - string1.length= 8+8+(8+8+8)*stored_LRD_size; - } + if (pagecache_collect_changed_blocks_with_LSN(pagecache, &strings[1], + &candidate_max_rec_lsn_at_last_checkpoint)) + goto err; - /* STEP 2: fetch information about transactions */ + /* STEP 3: fetch information about transactions */ /* note: this piece will move into trnman.c */ /* Transactions are in the "active list" (protected by a mutex) and in a @@ -345,7 +259,7 @@ my_bool execute_checkpoint_indirect() string2.length= 8+(7+2+8+8+8)*stored_trn_size; } - /* STEP 3: fetch information about table files */ + /* STEP 4: fetch information about table files */ { /* This global mutex is in fact THR_LOCK_maria (see ma_open()) */ @@ -391,13 +305,8 @@ my_bool execute_checkpoint_indirect() /* LAST STEP: now write the checkpoint log record */ - string_array[0]= string1; - string_array[1]= string2; - string_array[2]= string3; - string_array[3]= NULL; - checkpoint_lsn= log_write_record(LOGREC_CHECKPOINT, - &system_trans, string_array); + &system_trans, strings); /* Do nothing between the log write and the control file write, for the @@ -418,9 +327,8 @@ err: end: - my_free(buffer1.str, MYF(MY_ALLOW_ZERO_PTR)); - my_free(buffer2.str, MYF(MY_ALLOW_ZERO_PTR)); - my_free(buffer3.str, MYF(MY_ALLOW_ZERO_PTR)); + for (i= 1; i<5; i++) + my_free(strings[i], MYF(MY_ALLOW_ZERO_PTR)); /* this portion cannot be done as a hook in write_log_record() for the @@ -440,7 +348,6 @@ end: lock(log_mutex); /* That LSN is used for the "two-checkpoint rule" (MEDIUM checkpoints) */ maximum_rec_lsn_last_checkpoint= candidate_max_rec_lsn_at_last_checkpoint; - written_since_last_checkpoint= (my_off_t)0; DBUG_RETURN(FALSE); } lock(log_mutex); @@ -471,6 +378,8 @@ log_write_record(...) thread" WL#3261) to do a checkpoint */ request_asynchronous_checkpoint(INDIRECT); + /* prevent similar redundant requests */ + written_since_last_checkpoint= (my_off_t)0; } ...; unlock(log_mutex); @@ -488,16 +397,13 @@ void request_asynchronous_checkpoint(CHECKPOINT_LEVEL level); safemutex_assert_owner(log_mutex); DBUG_ASSERT(level > NONE); - if (next_asynchronous_checkpoint_to_do < level) + if ((next_asynchronous_checkpoint_to_do < level) && + (checkpoint_in_progress < level)) { /* no equal or stronger running or to run, we post request */ /* - note that thousands of requests for checkpoints are going to come all - at the same time (when the log bound - MAX_LOG_BYTES_WRITTEN_BETWEEN_CHECKPOINTS is passed), so it may not be a - good idea for each of them to broadcast a cond to wake up the background - checkpoint thread. We just don't broacast a cond, the checkpoint thread - (see least_recently_dirtied.c) will notice our request in max a few + We just don't broacast a cond, the checkpoint thread + (see ma_least_recently_dirtied.c) will notice our request in max a few seconds. */ next_asynchronous_checkpoint_to_do= level; /* post request */ @@ -520,6 +426,7 @@ void request_asynchronous_checkpoint(CHECKPOINT_LEVEL level); first_undo_lsn), this function can be used to do a read of it (without mutex, without atomic load) which always produces a correct (though maybe slightly old) value (even on 32-bit CPUs). + The prototype will change with Sanja's new LSN type. */ static inline ulonglong read_non_atomic(ulonglong volatile *x) { diff --git a/storage/maria/ma_least_recently_dirtied.c b/storage/maria/ma_least_recently_dirtied.c index 809442b4e97..170e59a601a 100644 --- a/storage/maria/ma_least_recently_dirtied.c +++ b/storage/maria/ma_least_recently_dirtied.c @@ -36,162 +36,57 @@ #include "least_recently_dirtied.h" /* - MikaelR suggested removing this global_LRD_mutex (I have a paper note of - comments), however at least for the first version we'll start with this - mutex (which will be a LOCK-based atomic_rwlock). -*/ -pthread_mutex_t global_LRD_mutex; - -/* - When we flush a page, we should pin page. - This "pin" is to protect against that: - I make copy, - you modify in memory and flush to disk and remove from LRD and from cache, - I write copy to disk, - checkpoint happens. - result: old page is on disk, page is absent from LRD, your REDO will be - wrongly ignored. - - Pin: there can be multiple pins, flushing imposes that there are zero pins. - For example, pin could be a uint counter protected by the page's latch. - - Maybe it's ok if when there is a page replacement, the replacer does not - remove page from the LRD (it would save global mutex); for that, background - flusher should be prepared to see pages in the LRD which are not in the page - cache (then just ignore them). However checkpoint will contain superfluous - entries and so do more work. -*/ - -#define PAGE_SIZE (16*1024) /* just as an example */ -/* - Optimization: - LRD flusher should not flush pages one by one: to be fast, it flushes a - group of pages in sequential disk order if possible; a group of pages is just - FLUSH_GROUP_SIZE pages. - Key cache has groupping already somehow Monty said (investigate that). -*/ -#define FLUSH_GROUP_SIZE 512 /* 8 MB */ -/* - We don't want to probe for checkpoint requests all the time (it takes - the log mutex). - If FLUSH_GROUP_SIZE is 8MB, assuming a local disk which can write 30MB/s - (1.8GB/min), probing every 16th call to flush_one_group_from_LRD() is every - 16*8=128MB which is every 128/30=4.2second. - Using a power of 2 gives a fast modulo operation. -*/ -#define CHECKPOINT_PROBING_PERIOD_LOG2 4 - -/* - This thread does background flush of pieces of the LRD, and all checkpoints. + This thread does background flush of pieces of the LRD, and serves + requests for asynchronous checkpoints. Just launch it when engine starts. MikaelR questioned why the same thread does two different jobs, the risk could be that while a checkpoint happens no LRD flushing happens. + For now, we only do checkpoints - no LRD flushing (to be done when the + second version of the page cache is ready WL#3077). + Reasons to delay: + - Recovery will work (just slower) + - new page cache may be different, why do then re-do + - current pagecache probably has issues with flushing when somebody is + writing to the table being flushed - better avoid that. */ pthread_handler_decl background_flush_and_checkpoint_thread() { - char *flush_group_buffer= my_malloc(PAGE_SIZE*FLUSH_GROUP_SIZE); - uint flush_calls= 0; while (this_thread_not_killed) { - if ((flush_calls++) & ((2<data, PAGE_SIZE); - pin_page; - page_cache_unlatch(page_id, KEEP_PINNED); /* but keep pinned */ - } - for (scan_the_array) - { - /* - As an optimization, we try to identify contiguous-in-the-file segments (to - issue one big write()). - In non-optimized version, contiguous segment is always only one page. - */ - if ((next_page.page_id - this_page.page_id) == 1) - { - /* - this page and next page are in same file and are contiguous in the - file: add page to contiguous segment... - */ - continue; /* defer write() to next pages */ - } - /* contiguous segment ends */ - my_pwrite(file, contiguous_segment_start_offset, contiguous_segment_size); - /* - note that if we had doublewrite, doublewrite buffer may prevent us from - doing this write() grouping (if doublewrite space is shorter). - */ - } /* - Now remove pages from LRD. As we have pinned them, all pages that we - managed to pin are still in the LRD, in the same order, we can just cut - the LRD at the last element of "array". This is more efficient that - removing element by element (which would take LRD mutex many times) in the - loop above. + Build a list of pages to flush: + changed_blocks[i] is roughly sorted by descending rec_lsn, + so we could do a merge sort of changed_blocks[] lists, stopping after we + have the max_this_number first elements or after we have found a page with + rec_lsn > max_this_lsn. + Then do like pagecache_flush_blocks_int() does (beware! this time we are + not alone on the file! there may be dangers! TODO: sort this out). */ - lock(global_LRD_mutex); - /* cut LRD by bending LRD->first, free cut portion... */ - unlock(global_LRD_mutex); - for (scan_array) - { - /* - if the page has a property "modified since last flush" (i.e. which is - redundant with the presence of the page in the LRD, this property can - just be a pointer to the LRD element) we should reset it - (note that then the property would live slightly longer than - the presence in LRD). - */ - page_cache_unpin(page_id); - /* - order between unpin and removal from LRD is not clear, depends on what - pin actually is. - */ - } - free(array); + /* MikaelR noted that he observed that Linux's file cache may never fsync to disk until this cache is full, at which point it decides to empty the @@ -201,28 +96,11 @@ flush_one_group_from_LRD() } /* - Flushes all page from LRD up to approximately rec_lsn>=max_lsn. - This is approximate because we flush groups, and because the LRD list may + Note that when we flush all page from LRD up to rec_lsn>=max_lsn, + this is approximate because the LRD list may not be exactly sorted by rec_lsn (because for a big row, all pages of the row are inserted into the LRD with rec_lsn being the LSN of the REDO for the first page, so if there are concurrent insertions, the last page of the big row may have a smaller rec_lsn than the previous pages inserted by concurrent inserters). */ -int flush_all_LRD_to_lsn(LSN max_lsn) -{ - lock(global_LRD_mutex); - if (max_lsn == MAX_LSN) /* don't want to flush forever, so make it fixed: */ - max_lsn= LRD->first->prev->rec_lsn; - while (LRD->first->rec_lsn < max_lsn) - { - if (flush_one_group_from_LRD()) /* will unlock LRD mutex */ - return 1; - /* - The scheduler may preempt us here as we released the mutex; this is good. - */ - lock(global_LRD_mutex); - } - unlock(global_LRD_mutex); - return 0; -} diff --git a/unittest/mysys/test_file.h b/unittest/mysys/test_file.h index ea787c123ed..bfc660b13d0 100644 --- a/unittest/mysys/test_file.h +++ b/unittest/mysys/test_file.h @@ -1,4 +1,4 @@ - +#include #include /* -- cgit v1.2.1 From c2f2a41ed32522fa89f5c79fec42ab60e0b4aa62 Mon Sep 17 00:00:00 2001 From: unknown Date: Tue, 19 Dec 2006 19:15:53 +0100 Subject: Maria - merging recent changes done to MyISAM into Maria. Plus compiler warnings, and a fix to the pagecache unit tests for IA64 include/maria.h: merging MyISAM into Maria include/myisam.h: post-merge fixes mysql-test/r/maria.result: merging MyISAM into Maria mysql-test/t/maria.test: merging MyISAM into Maria sql/mysqld.cc: post-merge fixes storage/maria/ha_maria.cc: merging MyISAM into Maria storage/maria/ha_maria.h: merging MyISAM into Maria storage/maria/ma_check.c: merging MyISAM into Maria storage/maria/ma_open.c: merging MyISAM into Maria storage/maria/ma_packrec.c: merging MyISAM into Maria storage/maria/ma_range.c: merging MyISAM into Maria storage/maria/ma_sort.c: merging MyISAM into Maria storage/maria/maria_def.h: merging MyISAM into Maria storage/maria/maria_pack.c: merging MyISAM into Maria storage/maria/plug.in: merging MyISAM into Maria storage/myisam/myisamdef.h: merging MyISAM into Maria storage/myisam/myisampack.c: fix for compiler warnings unittest/mysys/mf_pagecache_consist.c: this sets the stack size lower than the minimum on IA64, we remove it (it made the test fail) unittest/mysys/mf_pagecache_single.c: this sets the stack size lower than the minimum on IA64, we remove it (it made the test fail) --- include/maria.h | 15 +- include/myisam.h | 15 ++ mysql-test/r/maria.result | 142 +++++++++++++++- mysql-test/t/maria.test | 92 ++++++++++ sql/mysqld.cc | 2 +- storage/maria/ha_maria.cc | 12 +- storage/maria/ha_maria.h | 2 +- storage/maria/ma_check.c | 305 ++++++++++++++++++++++++++++------ storage/maria/ma_open.c | 5 +- storage/maria/ma_packrec.c | 76 +++++---- storage/maria/ma_range.c | 15 ++ storage/maria/ma_sort.c | 256 +++++++++++++++------------- storage/maria/maria_def.h | 34 ++-- storage/maria/maria_pack.c | 6 +- storage/maria/plug.in | 1 + storage/myisam/myisamdef.h | 12 +- storage/myisam/myisampack.c | 8 +- unittest/mysys/mf_pagecache_consist.c | 8 - unittest/mysys/mf_pagecache_single.c | 8 - 19 files changed, 754 insertions(+), 260 deletions(-) diff --git a/include/maria.h b/include/maria.h index 705ba7c91c7..da9cbd49688 100644 --- a/include/maria.h +++ b/include/maria.h @@ -293,6 +293,17 @@ extern uint maria_get_pointer_length(ulonglong file_length, uint def); #define MARIA_CHK_REPAIR 1 /* equivalent to mariachk -r */ #define MARIA_CHK_VERIFY 2 /* Verify, run repair if failure */ +typedef uint maria_bit_type; + +typedef struct st_maria_bit_buff +{ /* Used for packing of record */ + maria_bit_type current_byte; + uint bits; + uchar *pos, *end, *blob_pos, *blob_end; + uint error; +} MARIA_BIT_BUFF; + + typedef struct st_maria_sort_info { #ifdef THREAD @@ -305,7 +316,6 @@ typedef struct st_maria_sort_info char *buff; SORT_KEY_BLOCKS *key_block, *key_block_end; SORT_FT_BUF *ft_buf; - my_off_t filelength, dupp, buff_length; ha_rows max_records; uint current_key, total_keys; @@ -314,12 +324,12 @@ typedef struct st_maria_sort_info enum data_file_type new_data_file_type; } MARIA_SORT_INFO; - typedef struct st_maria_sort_param { pthread_t thr; IO_CACHE read_cache, tempfile, tempfile_for_exceptions; DYNAMIC_ARRAY buffpek; + MARIA_BIT_BUFF bit_buff; /* For parallel repair of packrec. */ MARIA_KEYDEF *keyinfo; MARIA_SORT_INFO *sort_info; @@ -342,6 +352,7 @@ typedef struct st_maria_sort_param uint key, key_length,real_key_length,sortbuff_size; uint maxbuffers, keys, find_length, sort_keys_length; my_bool fix_datafile, master; + my_bool calc_checksum; /* calculate table checksum */ int (*key_cmp)(struct st_maria_sort_param *, const void *, const void *); int (*key_read)(struct st_maria_sort_param *,void *); diff --git a/include/myisam.h b/include/myisam.h index fc7468a382d..9171fec0b95 100644 --- a/include/myisam.h +++ b/include/myisam.h @@ -300,6 +300,17 @@ extern uint mi_get_pointer_length(ulonglong file_length, uint def); #define MYISAMCHK_REPAIR 1 /* equivalent to myisamchk -r */ #define MYISAMCHK_VERIFY 2 /* Verify, run repair if failure */ +typedef uint mi_bit_type; + +typedef struct st_mi_bit_buff +{ /* Used for packing of record */ + mi_bit_type current_byte; + uint bits; + uchar *pos, *end, *blob_pos, *blob_end; + uint error; +} MI_BIT_BUFF; + + typedef struct st_sort_info { #ifdef THREAD @@ -319,10 +330,13 @@ typedef struct st_sort_info myf myf_rw; enum data_file_type new_data_file_type; } MI_SORT_INFO; + typedef struct st_mi_sort_param +{ pthread_t thr; IO_CACHE read_cache, tempfile, tempfile_for_exceptions; DYNAMIC_ARRAY buffpek; + MI_BIT_BUFF bit_buff; /* For parallel repair of packrec. */ MI_KEYDEF *keyinfo; MI_SORT_INFO *sort_info; @@ -345,6 +359,7 @@ typedef struct st_mi_sort_param uint key, key_length,real_key_length,sortbuff_size; uint maxbuffers, keys, find_length, sort_keys_length; my_bool fix_datafile, master; + my_bool calc_checksum; /* calculate table checksum */ int (*key_cmp)(struct st_mi_sort_param *, const void *, const void *); int (*key_read)(struct st_mi_sort_param *,void *); diff --git a/mysql-test/r/maria.result b/mysql-test/r/maria.result index 39c7c0b65dd..7812f04336a 100644 --- a/mysql-test/r/maria.result +++ b/mysql-test/r/maria.result @@ -780,6 +780,132 @@ a b xxxxxxxxx bbbbbb xxxxxxxxx bbbbbb DROP TABLE t1; +SET @@maria_repair_threads=2; +SHOW VARIABLES LIKE 'maria_repair%'; +Variable_name Value +maria_repair_threads 2 +CREATE TABLE t1 ( +`_id` int(11) NOT NULL default '0', +`url` text, +`email` text, +`description` text, +`loverlap` int(11) default NULL, +`roverlap` int(11) default NULL, +`lneighbor_id` int(11) default NULL, +`rneighbor_id` int(11) default NULL, +`length_` int(11) default NULL, +`sequence` mediumtext, +`name` text, +`_obj_class` text NOT NULL, +PRIMARY KEY (`_id`), +UNIQUE KEY `sequence_name_index` (`name`(50)), +KEY (`length_`) +) ENGINE=maria DEFAULT CHARSET=latin1; +INSERT INTO t1 VALUES +(1,NULL,NULL,NULL,NULL,NULL,NULL,NULL,NULL,NULL,'sample1',''), +(2,NULL,NULL,NULL,NULL,NULL,NULL,NULL,NULL,NULL,'sample2',''), +(3,NULL,NULL,NULL,NULL,NULL,NULL,NULL,NULL,NULL,'sample3',''), +(4,NULL,NULL,NULL,NULL,NULL,NULL,NULL,NULL,NULL,'sample4',''), +(5,NULL,NULL,NULL,NULL,NULL,NULL,NULL,NULL,NULL,'sample5',''), +(6,NULL,NULL,NULL,NULL,NULL,NULL,NULL,NULL,NULL,'sample6',''), +(7,NULL,NULL,NULL,NULL,NULL,NULL,NULL,NULL,NULL,'sample7',''), +(8,NULL,NULL,NULL,NULL,NULL,NULL,NULL,NULL,NULL,'sample8',''), +(9,NULL,NULL,NULL,NULL,NULL,NULL,NULL,NULL,NULL,'sample9',''); +SELECT _id FROM t1; +_id +1 +2 +3 +4 +5 +6 +7 +8 +9 +DELETE FROM t1 WHERE _id < 8; +SHOW TABLE STATUS LIKE 't1'; +Name Engine Version Row_format Rows Avg_row_length Data_length Max_data_length Index_length Data_free Auto_increment Create_time Update_time Check_time Collation Checksum Create_options Comment +t1 MARIA 10 Dynamic 2 # # # # 140 # # # # # # +CHECK TABLE t1 EXTENDED; +Table Op Msg_type Msg_text +test.t1 check status OK +OPTIMIZE TABLE t1; +Table Op Msg_type Msg_text +test.t1 optimize status OK +CHECK TABLE t1 EXTENDED; +Table Op Msg_type Msg_text +test.t1 check status OK +SHOW TABLE STATUS LIKE 't1'; +Name Engine Version Row_format Rows Avg_row_length Data_length Max_data_length Index_length Data_free Auto_increment Create_time Update_time Check_time Collation Checksum Create_options Comment +t1 MARIA 10 Dynamic 2 # # # # 0 # # # # # # +SELECT _id FROM t1; +_id +8 +9 +DROP TABLE t1; +CREATE TABLE t1 ( +`_id` int(11) NOT NULL default '0', +`url` text, +`email` text, +`description` text, +`loverlap` int(11) default NULL, +`roverlap` int(11) default NULL, +`lneighbor_id` int(11) default NULL, +`rneighbor_id` int(11) default NULL, +`length_` int(11) default NULL, +`sequence` mediumtext, +`name` text, +`_obj_class` text NOT NULL, +PRIMARY KEY (`_id`), +UNIQUE KEY `sequence_name_index` (`name`(50)), +KEY (`length_`) +) ENGINE=maria DEFAULT CHARSET=latin1; +INSERT INTO t1 VALUES +(1,NULL,NULL,NULL,NULL,NULL,NULL,NULL,NULL,NULL,'sample1',''), +(2,NULL,NULL,NULL,NULL,NULL,NULL,NULL,NULL,NULL,'sample2',''), +(3,NULL,NULL,NULL,NULL,NULL,NULL,NULL,NULL,NULL,'sample3',''), +(4,NULL,NULL,NULL,NULL,NULL,NULL,NULL,NULL,NULL,'sample4',''), +(5,NULL,NULL,NULL,NULL,NULL,NULL,NULL,NULL,NULL,'sample5',''), +(6,NULL,NULL,NULL,NULL,NULL,NULL,NULL,NULL,NULL,'sample6',''), +(7,NULL,NULL,NULL,NULL,NULL,NULL,NULL,NULL,NULL,'sample7',''), +(8,NULL,NULL,NULL,NULL,NULL,NULL,NULL,NULL,NULL,'sample8',''), +(9,NULL,NULL,NULL,NULL,NULL,NULL,NULL,NULL,NULL,'sample9',''); +SELECT _id FROM t1; +_id +1 +2 +3 +4 +5 +6 +7 +8 +9 +DELETE FROM t1 WHERE _id < 8; +SHOW TABLE STATUS LIKE 't1'; +Name Engine Version Row_format Rows Avg_row_length Data_length Max_data_length Index_length Data_free Auto_increment Create_time Update_time Check_time Collation Checksum Create_options Comment +t1 MARIA 10 Dynamic 2 # # # # 140 # # # # # # +CHECK TABLE t1 EXTENDED; +Table Op Msg_type Msg_text +test.t1 check status OK +REPAIR TABLE t1 QUICK; +Table Op Msg_type Msg_text +test.t1 repair status OK +CHECK TABLE t1 EXTENDED; +Table Op Msg_type Msg_text +test.t1 check status OK +SHOW TABLE STATUS LIKE 't1'; +Name Engine Version Row_format Rows Avg_row_length Data_length Max_data_length Index_length Data_free Auto_increment Create_time Update_time Check_time Collation Checksum Create_options Comment +t1 MARIA 10 Dynamic 2 # # # # 140 # # # # # # +SELECT _id FROM t1; +_id +8 +9 +DROP TABLE t1; +SET @@maria_repair_threads=1; +SHOW VARIABLES LIKE 'maria_repair%'; +Variable_name Value +maria_repair_threads 1 drop table if exists t1,t2,t3; --- Testing varchar --- --- Testing varchar --- @@ -985,15 +1111,15 @@ i 10 select sql_big_result v,count(c) from t1 group by v limit 10; v count(c) a 1 -a 10 -b 10 -c 10 -d 10 -e 10 -f 10 -g 10 +a 10 +b 10 +c 10 +d 10 +e 10 +f 10 +g 10 h 10 -i 10 +i 10 select c,count(*) from t1 group by c limit 10; c count(*) a 1 diff --git a/mysql-test/t/maria.test b/mysql-test/t/maria.test index 4130831c482..87a38ff60b0 100644 --- a/mysql-test/t/maria.test +++ b/mysql-test/t/maria.test @@ -733,6 +733,98 @@ UPDATE t1 AS ta1,t1 AS ta2 SET ta1.b='aaaaaa',ta2.b='bbbbbb'; SELECT * FROM t1; DROP TABLE t1; +# +# Bug#8283 - OPTIMIZE TABLE causes data loss +# +SET @@maria_repair_threads=2; +SHOW VARIABLES LIKE 'maria_repair%'; +# +# Test OPTIMIZE. This creates a new data file. +CREATE TABLE t1 ( + `_id` int(11) NOT NULL default '0', + `url` text, + `email` text, + `description` text, + `loverlap` int(11) default NULL, + `roverlap` int(11) default NULL, + `lneighbor_id` int(11) default NULL, + `rneighbor_id` int(11) default NULL, + `length_` int(11) default NULL, + `sequence` mediumtext, + `name` text, + `_obj_class` text NOT NULL, + PRIMARY KEY (`_id`), + UNIQUE KEY `sequence_name_index` (`name`(50)), + KEY (`length_`) +) ENGINE=maria DEFAULT CHARSET=latin1; +# +INSERT INTO t1 VALUES + (1,NULL,NULL,NULL,NULL,NULL,NULL,NULL,NULL,NULL,'sample1',''), + (2,NULL,NULL,NULL,NULL,NULL,NULL,NULL,NULL,NULL,'sample2',''), + (3,NULL,NULL,NULL,NULL,NULL,NULL,NULL,NULL,NULL,'sample3',''), + (4,NULL,NULL,NULL,NULL,NULL,NULL,NULL,NULL,NULL,'sample4',''), + (5,NULL,NULL,NULL,NULL,NULL,NULL,NULL,NULL,NULL,'sample5',''), + (6,NULL,NULL,NULL,NULL,NULL,NULL,NULL,NULL,NULL,'sample6',''), + (7,NULL,NULL,NULL,NULL,NULL,NULL,NULL,NULL,NULL,'sample7',''), + (8,NULL,NULL,NULL,NULL,NULL,NULL,NULL,NULL,NULL,'sample8',''), + (9,NULL,NULL,NULL,NULL,NULL,NULL,NULL,NULL,NULL,'sample9',''); +# +SELECT _id FROM t1; +DELETE FROM t1 WHERE _id < 8; +--replace_column 6 # 7 # 8 # 9 # 11 # 12 # 13 # 14 # 15 # 16 # +SHOW TABLE STATUS LIKE 't1'; +CHECK TABLE t1 EXTENDED; +OPTIMIZE TABLE t1; +CHECK TABLE t1 EXTENDED; +--replace_column 6 # 7 # 8 # 9 # 11 # 12 # 13 # 14 # 15 # 16 # +SHOW TABLE STATUS LIKE 't1'; +SELECT _id FROM t1; +DROP TABLE t1; +# +# Test REPAIR QUICK. This retains the old data file. +CREATE TABLE t1 ( + `_id` int(11) NOT NULL default '0', + `url` text, + `email` text, + `description` text, + `loverlap` int(11) default NULL, + `roverlap` int(11) default NULL, + `lneighbor_id` int(11) default NULL, + `rneighbor_id` int(11) default NULL, + `length_` int(11) default NULL, + `sequence` mediumtext, + `name` text, + `_obj_class` text NOT NULL, + PRIMARY KEY (`_id`), + UNIQUE KEY `sequence_name_index` (`name`(50)), + KEY (`length_`) +) ENGINE=maria DEFAULT CHARSET=latin1; +# +INSERT INTO t1 VALUES + (1,NULL,NULL,NULL,NULL,NULL,NULL,NULL,NULL,NULL,'sample1',''), + (2,NULL,NULL,NULL,NULL,NULL,NULL,NULL,NULL,NULL,'sample2',''), + (3,NULL,NULL,NULL,NULL,NULL,NULL,NULL,NULL,NULL,'sample3',''), + (4,NULL,NULL,NULL,NULL,NULL,NULL,NULL,NULL,NULL,'sample4',''), + (5,NULL,NULL,NULL,NULL,NULL,NULL,NULL,NULL,NULL,'sample5',''), + (6,NULL,NULL,NULL,NULL,NULL,NULL,NULL,NULL,NULL,'sample6',''), + (7,NULL,NULL,NULL,NULL,NULL,NULL,NULL,NULL,NULL,'sample7',''), + (8,NULL,NULL,NULL,NULL,NULL,NULL,NULL,NULL,NULL,'sample8',''), + (9,NULL,NULL,NULL,NULL,NULL,NULL,NULL,NULL,NULL,'sample9',''); +# +SELECT _id FROM t1; +DELETE FROM t1 WHERE _id < 8; +--replace_column 6 # 7 # 8 # 9 # 11 # 12 # 13 # 14 # 15 # 16 # +SHOW TABLE STATUS LIKE 't1'; +CHECK TABLE t1 EXTENDED; +REPAIR TABLE t1 QUICK; +CHECK TABLE t1 EXTENDED; +--replace_column 6 # 7 # 8 # 9 # 11 # 12 # 13 # 14 # 15 # 16 # +SHOW TABLE STATUS LIKE 't1'; +SELECT _id FROM t1; +DROP TABLE t1; +# +SET @@maria_repair_threads=1; +SHOW VARIABLES LIKE 'maria_repair%'; # # Test varchar # diff --git a/sql/mysqld.cc b/sql/mysqld.cc index ade7539251b..188f3ae9a69 100644 --- a/sql/mysqld.cc +++ b/sql/mysqld.cc @@ -3537,7 +3537,7 @@ int main(int argc, char **argv) { if (global_system_variables.log_warnings) sql_print_warning("Asked for %ld thread stack, but got %ld", - my_thread_stack, (long) stack_size); + my_thread_stack_size, (long) stack_size); #if defined(__ia64__) || defined(__ia64) my_thread_stack_size= stack_size*2; #else diff --git a/storage/maria/ha_maria.cc b/storage/maria/ha_maria.cc index d9e42467351..56f33693528 100644 --- a/storage/maria/ha_maria.cc +++ b/storage/maria/ha_maria.cc @@ -283,7 +283,7 @@ err: bool ha_maria::check_if_locking_is_allowed(uint sql_command, ulong type, TABLE *table, uint count, - bool called_by_logger_thread) + bool called_by_privileged_thread) { /* To be able to open and lock for reading system tables like 'mysql.proc', @@ -300,10 +300,10 @@ bool ha_maria::check_if_locking_is_allowed(uint sql_command, /* Deny locking of the log tables, which is incompatible with - concurrent insert. Unless called from a logger THD: - general_log_thd or slow_log_thd. + concurrent insert. Unless called from a logger THD (general_log_thd + or slow_log_thd) or by a privileged thread. */ - if (!called_by_logger_thread) + if (!called_by_privileged_thread) return check_if_log_table_locking_is_allowed(sql_command, type, table); return TRUE; @@ -1368,7 +1368,7 @@ void ha_maria::position(const byte * record) } -void ha_maria::info(uint flag) +int ha_maria::info(uint flag) { MARIA_INFO info; char name_buff[FN_REFLEN]; @@ -1428,6 +1428,8 @@ void ha_maria::info(uint flag) /* Faster to always update, than to do it based on flag */ stats.update_time= info.update_time; stats.auto_increment_value= info.auto_increment; + + return 0; } diff --git a/storage/maria/ha_maria.h b/storage/maria/ha_maria.h index e4edcc80982..52f289a7428 100644 --- a/storage/maria/ha_maria.h +++ b/storage/maria/ha_maria.h @@ -103,7 +103,7 @@ public: int rnd_pos(byte * buf, byte * pos); int restart_rnd_next(byte * buf, byte * pos); void position(const byte * record); - void info(uint); + int info(uint); int extra(enum ha_extra_function operation); int extra_opt(enum ha_extra_function operation, ulong cache_size); int reset(void); diff --git a/storage/maria/ma_check.c b/storage/maria/ma_check.c index f997b90a61c..2a800ae41ad 100644 --- a/storage/maria/ma_check.c +++ b/storage/maria/ma_check.c @@ -16,6 +16,31 @@ /* Describe, check and repair of MARIA tables */ +/* + About checksum calculation. + + There are two types of checksums. Table checksum and row checksum. + + Row checksum is an additional byte at the end of dynamic length + records. It must be calculated if the table is configured for them. + Otherwise they must not be used. The variable + MYISAM_SHARE::calc_checksum determines if row checksums are used. + MI_INFO::checksum is used as temporary storage during row handling. + For parallel repair we must assure that only one thread can use this + variable. There is no problem on the write side as this is done by one + thread only. But when checking a record after read this could go + wrong. But since all threads read through a common read buffer, it is + sufficient if only one thread checks it. + + Table checksum is an eight byte value in the header of the index file. + It can be calculated even if row checksums are not used. The variable + MI_CHECK::glob_crc is calculated over all records. + MI_SORT_PARAM::calc_checksum determines if this should be done. This + variable is not part of MI_CHECK because it must be set per thread for + parallel repair. The global glob_crc must be changed by one thread + only. And it is sufficient to calculate the checksum once only. +*/ + #include "ma_ftdefs.h" #include #include @@ -42,8 +67,7 @@ static int chk_index(HA_CHECK *param, MARIA_HA *info,MARIA_KEYDEF *keyinfo, ha_checksum *key_checksum, uint level); static uint isam_key_length(MARIA_HA *info,MARIA_KEYDEF *keyinfo); static ha_checksum calc_checksum(ha_rows count); -static int writekeys(HA_CHECK *param, MARIA_HA *info,byte *buff, - my_off_t filepos); +static int writekeys(MARIA_SORT_PARAM *sort_param); static int sort_one_index(HA_CHECK *param, MARIA_HA *info,MARIA_KEYDEF *keyinfo, my_off_t pagepos, File new_file); static int sort_key_read(MARIA_SORT_PARAM *sort_param,void *key); @@ -1159,7 +1183,8 @@ int maria_chk_data_link(HA_CHECK *param, MARIA_HA *info,int extend) goto err; start_recpos=pos; splits++; - VOID(_ma_pack_get_block_info(info,&block_info, -1, start_recpos)); + VOID(_ma_pack_get_block_info(info, &info->bit_buff, &block_info, + &info->rec_buff, -1, start_recpos)); pos=block_info.filepos+block_info.rec_len; if (block_info.rec_len < (uint) info->s->min_pack_length || block_info.rec_len > (uint) info->s->max_pack_length) @@ -1173,7 +1198,8 @@ int maria_chk_data_link(HA_CHECK *param, MARIA_HA *info,int extend) if (_ma_read_cache(¶m->read_cache,(byte*) info->rec_buff, block_info.filepos, block_info.rec_len, READING_NEXT)) goto err; - if (_ma_pack_rec_unpack(info,record,info->rec_buff,block_info.rec_len)) + if (_ma_pack_rec_unpack(info, &info->bit_buff, record, + info->rec_buff, block_info.rec_len)) { _ma_check_print_error(param,"Found wrong record at %s", llstr(start_recpos,llbuff)); @@ -1457,7 +1483,7 @@ int maria_repair(HA_CHECK *param, register MARIA_HA *info, info->state->empty=0; param->glob_crc=0; if (param->testflag & T_CALC_CHECKSUM) - param->calc_checksum=1; + sort_param.calc_checksum= 1; info->update= (short) (HA_STATE_CHANGED | HA_STATE_ROW_CHANGED); @@ -1486,7 +1512,7 @@ int maria_repair(HA_CHECK *param, register MARIA_HA *info, /* Re-create all keys, which are set in key_map. */ while (!(error=sort_get_next_record(&sort_param))) { - if (writekeys(param,info,(byte*)sort_param.record,sort_param.filepos)) + if (writekeys(&sort_param)) { if (my_errno != HA_ERR_FOUND_DUPP_KEY) goto err; @@ -1631,11 +1657,13 @@ err: /* Uppate keyfile when doing repair */ -static int writekeys(HA_CHECK *param, register MARIA_HA *info, byte *buff, - my_off_t filepos) +static int writekeys(MARIA_SORT_PARAM *sort_param) { register uint i; - uchar *key; + uchar *key; + MARIA_HA *info= sort_param->sort_info->info; + byte *buff= sort_param->record; + my_off_t filepos= sort_param->filepos; DBUG_ENTER("writekeys"); key=info->lastkey+info->s->base.max_key_length; @@ -1689,8 +1717,8 @@ static int writekeys(HA_CHECK *param, register MARIA_HA *info, byte *buff, } } /* Remove checksum that was added to glob_crc in sort_get_next_record */ - if (param->calc_checksum) - param->glob_crc-= info->checksum; + if (sort_param->calc_checksum) + sort_param->sort_info->param->glob_crc-= info->checksum; DBUG_PRINT("error",("errno: %d",my_errno)); DBUG_RETURN(-1); } /* writekeys */ @@ -2180,7 +2208,7 @@ int maria_repair_by_sort(HA_CHECK *param, register MARIA_HA *info, del=info->state->del; param->glob_crc=0; if (param->testflag & T_CALC_CHECKSUM) - param->calc_checksum=1; + sort_param.calc_checksum= 1; rec_per_key_part= param->rec_per_key_part; for (sort_param.key=0 ; sort_param.key < share->base.keys ; @@ -2266,7 +2294,8 @@ int maria_repair_by_sort(HA_CHECK *param, register MARIA_HA *info, param->retry_repair=1; goto err; } - param->calc_checksum=0; /* No need to calc glob_crc */ + /* No need to calculate checksum again. */ + sort_param.calc_checksum= 0; free_root(&sort_param.wordroot, MYF(0)); /* Set for next loop */ @@ -2430,6 +2459,28 @@ err: Each key is handled by a separate thread. TODO: make a number of threads a parameter + In parallel repair we use one thread per index. There are two modes: + + Quick + + Only the indexes are rebuilt. All threads share a read buffer. + Every thread that needs fresh data in the buffer enters the shared + cache lock. The last thread joining the lock reads the buffer from + the data file and wakes all other threads. + + Non-quick + + The data file is rebuilt and all indexes are rebuilt to point to + the new record positions. One thread is the master thread. It + reads from the old data file and writes to the new data file. It + also creates one of the indexes. The other threads read from a + buffer which is filled by the master. If they need fresh data, + they enter the shared cache lock. If the masters write buffer is + full, it flushes it to the new data file and enters the shared + cache lock too. When all threads joined in the lock, the master + copies its write buffer to the read buffer for the other threads + and wakes them. + RESULT 0 ok <>0 Error @@ -2452,6 +2503,7 @@ int maria_repair_parallel(HA_CHECK *param, register MARIA_HA *info, ulong *rec_per_key_part; HA_KEYSEG *keyseg; char llbuff[22]; + IO_CACHE new_data_cache; /* For non-quick repair. */ IO_CACHE_SHARE io_share; MARIA_SORT_INFO sort_info; ulonglong key_map=share->state.key_map; @@ -2473,19 +2525,55 @@ int maria_repair_parallel(HA_CHECK *param, register MARIA_HA *info, if (info->s->options & (HA_OPTION_CHECKSUM | HA_OPTION_COMPRESS_RECORD)) param->testflag|=T_CALC_CHECKSUM; + /* + Quick repair (not touching data file, rebuilding indexes): + { + Read cache is (MI_CHECK *param)->read_cache using info->dfile. + } + + Non-quick repair (rebuilding data file and indexes): + { + Master thread: + + Read cache is (MI_CHECK *param)->read_cache using info->dfile. + Write cache is (MI_INFO *info)->rec_cache using new_file. + + Slave threads: + + Read cache is new_data_cache synced to master rec_cache. + + The final assignment of the filedescriptor for rec_cache is done + after the cache creation. + + Don't check file size on new_data_cache, as the resulting file size + is not known yet. + + As rec_cache and new_data_cache are synced, write_buffer_length is + used for the read cache 'new_data_cache'. Both start at the same + position 'new_header_length'. + } + */ + DBUG_PRINT("info", ("is quick repair: %d", rep_quick)); bzero((char*)&sort_info,sizeof(sort_info)); + /* Initialize pthread structures before goto err. */ + pthread_mutex_init(&sort_info.mutex, MY_MUTEX_INIT_FAST); + pthread_cond_init(&sort_info.cond, 0); + if (!(sort_info.key_block= - alloc_key_blocks(param, - (uint) param->sort_key_blocks, - share->base.max_key_block_length)) - || init_io_cache(¶m->read_cache,info->dfile, - (uint) param->read_buffer_length, - READ_CACHE,share->pack.header_length,1,MYF(MY_WME)) || - (! rep_quick && - init_io_cache(&info->rec_cache,info->dfile, - (uint) param->write_buffer_length, - WRITE_CACHE,new_header_length,1, - MYF(MY_WME | MY_WAIT_IF_FULL) & param->myf_rw))) + alloc_key_blocks(param, (uint) param->sort_key_blocks, + share->base.max_key_block_length)) || + init_io_cache(¶m->read_cache, info->dfile, + (uint) param->read_buffer_length, + READ_CACHE, share->pack.header_length, 1, MYF(MY_WME)) || + (!rep_quick && + (init_io_cache(&info->rec_cache, info->dfile, + (uint) param->write_buffer_length, + WRITE_CACHE, new_header_length, 1, + MYF(MY_WME | MY_WAIT_IF_FULL) & param->myf_rw) || + init_io_cache(&new_data_cache, -1, + (uint) param->write_buffer_length, + READ_CACHE, new_header_length, 1, + MYF(MY_WME | MY_DONT_CHECK_FILESIZE))))) goto err; sort_info.key_block_end=sort_info.key_block+param->sort_key_blocks; info->opt_flag|=WRITE_CACHE_USED; @@ -2576,8 +2664,6 @@ int maria_repair_parallel(HA_CHECK *param, register MARIA_HA *info, del=info->state->del; param->glob_crc=0; - if (param->testflag & T_CALC_CHECKSUM) - param->calc_checksum=1; if (!(sort_param=(MARIA_SORT_PARAM *) my_malloc((uint) share->base.keys * @@ -2627,6 +2713,7 @@ int maria_repair_parallel(HA_CHECK *param, register MARIA_HA *info, sort_param[i].sort_info=&sort_info; sort_param[i].master=0; sort_param[i].fix_datafile=0; + sort_param[i].calc_checksum= 0; sort_param[i].filepos=new_header_length; sort_param[i].max_pos=sort_param[i].pos=share->pack.header_length; @@ -2664,19 +2751,45 @@ int maria_repair_parallel(HA_CHECK *param, register MARIA_HA *info, sort_info.total_keys=i; sort_param[0].master= 1; sort_param[0].fix_datafile= (my_bool)(! rep_quick); + sort_param[0].calc_checksum= test(param->testflag & T_CALC_CHECKSUM); sort_info.got_error=0; - pthread_mutex_init(&sort_info.mutex, MY_MUTEX_INIT_FAST); - pthread_cond_init(&sort_info.cond, 0); pthread_mutex_lock(&sort_info.mutex); - init_io_cache_share(¶m->read_cache, &io_share, i); + /* + Initialize the I/O cache share for use with the read caches and, in + case of non-quick repair, the write cache. When all threads join on + the cache lock, the writer copies the write cache contents to the + read caches. + */ + if (i > 1) + { + if (rep_quick) + init_io_cache_share(¶m->read_cache, &io_share, NULL, i); + else + init_io_cache_share(&new_data_cache, &io_share, &info->rec_cache, i); + } + else + io_share.total_threads= 0; /* share not used */ + (void) pthread_attr_init(&thr_attr); (void) pthread_attr_setdetachstate(&thr_attr,PTHREAD_CREATE_DETACHED); for (i=0 ; i < sort_info.total_keys ; i++) { - sort_param[i].read_cache=param->read_cache; + /* + Copy the properly initialized IO_CACHE structure so that every + thread has its own copy. In quick mode param->read_cache is shared + for use by all threads. In non-quick mode all threads but the + first copy the shared new_data_cache, which is synchronized to the + write cache of the first thread. The first thread copies + param->read_cache, which is not shared. + */ + sort_param[i].read_cache= ((rep_quick || !i) ? param->read_cache : + new_data_cache); + DBUG_PRINT("io_cache_share", ("thread: %u read_cache: 0x%lx", + i, (long) &sort_param[i].read_cache)); + /* two approaches: the same amount of memory for each thread or the memory for the same number of keys for each thread... @@ -2694,7 +2807,10 @@ int maria_repair_parallel(HA_CHECK *param, register MARIA_HA *info, (void *) (sort_param+i))) { _ma_check_print_error(param,"Cannot start a repair thread"); - remove_io_thread(¶m->read_cache); + /* Cleanup: Detach from the share. Avoid others to be blocked. */ + if (io_share.total_threads) + remove_io_thread(&sort_param[i].read_cache); + DBUG_PRINT("error", ("Cannot start a repair thread")); sort_info.got_error=1; } else @@ -2716,6 +2832,11 @@ int maria_repair_parallel(HA_CHECK *param, register MARIA_HA *info, if (sort_param[0].fix_datafile) { + /* + Append some nuls to the end of a memory mapped file. Destroy the + write cache. The master thread did already detach from the share + by remove_io_thread() in sort.c:thr_find_all_keys(). + */ if (maria_write_data_suffix(&sort_info,1) || end_io_cache(&info->rec_cache)) goto err; if (param->testflag & T_SAFE_REPAIR) @@ -2731,8 +2852,13 @@ int maria_repair_parallel(HA_CHECK *param, register MARIA_HA *info, sort_param->filepos; /* Only whole records */ share->state.version=(ulong) time((time_t*) 0); + /* + Exchange the data file descriptor of the table, so that we use the + new file from now on. + */ my_close(info->dfile,MYF(0)); info->dfile=new_file; + share->data_file_type=sort_info.new_data_file_type; share->pack.header_length=(ulong) new_header_length; } @@ -2787,7 +2913,20 @@ int maria_repair_parallel(HA_CHECK *param, register MARIA_HA *info, err: got_error|= _ma_flush_blocks(param, share->key_cache, share->kfile); + /* + Destroy the write cache. The master thread did already detach from + the share by remove_io_thread() or it was not yet started (if the + error happend before creating the thread). + */ VOID(end_io_cache(&info->rec_cache)); + /* + Destroy the new data cache in case of non-quick repair. All slave + threads did either detach from the share by remove_io_thread() + already or they were not yet started (if the error happend before + creating the threads). + */ + if (!rep_quick) + VOID(end_io_cache(&new_data_cache)); if (!got_error) { /* Replace the actual file with the temporary file */ @@ -2920,12 +3059,41 @@ static int sort_maria_ft_key_read(MARIA_SORT_PARAM *sort_param, void *key) } /* sort_maria_ft_key_read */ - /* Read next record from file using parameters in sort_info */ - /* Return -1 if end of file, 0 if ok and > 0 if error */ +/* + Read next record from file using parameters in sort_info. + + SYNOPSIS + sort_get_next_record() + sort_param Information about and for the sort process + + NOTE + + Dynamic Records With Non-Quick Parallel Repair + + For non-quick parallel repair we use a synchronized read/write + cache. This means that one thread is the master who fixes the data + file by reading each record from the old data file and writing it + to the new data file. By doing this the records in the new data + file are written contiguously. Whenever the write buffer is full, + it is copied to the read buffer. The slaves read from the read + buffer, which is not associated with a file. Thus read_cache.file + is -1. When using _mi_read_cache(), the slaves must always set + flag to READING_NEXT so that the function never tries to read from + file. This is safe because the records are contiguous. There is no + need to read outside the cache. This condition is evaluated in the + variable 'parallel_flag' for quick reference. read_cache.file must + be >= 0 in every other case. + + RETURN + -1 end of file + 0 ok + > 0 error +*/ static int sort_get_next_record(MARIA_SORT_PARAM *sort_param) { int searching; + int parallel_flag; uint found_record,b_type,left_length; my_off_t pos; byte *to; @@ -2963,7 +3131,7 @@ static int sort_get_next_record(MARIA_SORT_PARAM *sort_param) sort_param->max_pos=(sort_param->pos+=share->base.pack_reclength); if (*sort_param->record) { - if (param->calc_checksum) + if (sort_param->calc_checksum) param->glob_crc+= (info->checksum= _ma_static_checksum(info,sort_param->record)); DBUG_RETURN(0); @@ -2978,6 +3146,7 @@ static int sort_get_next_record(MARIA_SORT_PARAM *sort_param) LINT_INIT(to); pos=sort_param->pos; searching=(sort_param->fix_datafile && (param->testflag & T_EXTEND)); + parallel_flag= (sort_param->read_cache.file < 0) ? READING_NEXT : 0; for (;;) { found_record=block_info.second_read= 0; @@ -3008,7 +3177,7 @@ static int sort_get_next_record(MARIA_SORT_PARAM *sort_param) (byte*) block_info.header,pos, MARIA_BLOCK_INFO_HEADER_LENGTH, (! found_record ? READING_NEXT : 0) | - READING_HEADER)) + parallel_flag | READING_HEADER)) { if (found_record) { @@ -3185,9 +3354,31 @@ static int sort_get_next_record(MARIA_SORT_PARAM *sort_param) llstr(sort_param->start_recpos,llbuff)); goto try_next; } - if (_ma_read_cache(&sort_param->read_cache,to,block_info.filepos, - block_info.data_len, - (found_record == 1 ? READING_NEXT : 0))) + /* + Copy information that is already read. Avoid accessing data + below the cache start. This could happen if the header + streched over the end of the previous buffer contents. + */ + { + uint header_len= (uint) (block_info.filepos - pos); + uint prefetch_len= (MARIA_BLOCK_INFO_HEADER_LENGTH - header_len); + + if (prefetch_len > block_info.data_len) + prefetch_len= block_info.data_len; + if (prefetch_len) + { + memcpy(to, block_info.header + header_len, prefetch_len); + block_info.filepos+= prefetch_len; + block_info.data_len-= prefetch_len; + left_length-= prefetch_len; + to+= prefetch_len; + } + } + if (block_info.data_len && + _ma_read_cache(&sort_param->read_cache,to,block_info.filepos, + block_info.data_len, + (found_record == 1 ? READING_NEXT : 0) | + parallel_flag)) { _ma_check_print_info(param, "Read error for block at: %s (error: %d); Skipped", @@ -3217,13 +3408,14 @@ static int sort_get_next_record(MARIA_SORT_PARAM *sort_param) { if (sort_param->read_cache.error < 0) DBUG_RETURN(1); - if (info->s->calc_checksum) - info->checksum=_ma_checksum(info,sort_param->record); + if (sort_param->calc_checksum) + info->checksum= _ma_checksum(info, sort_param->record); if ((param->testflag & (T_EXTEND | T_REP)) || searching) { if (_ma_rec_check(info, sort_param->record, sort_param->rec_buff, sort_param->find_length, (param->testflag & T_QUICK) && + sort_param->calc_checksum && test(info->s->calc_checksum))) { _ma_check_print_info(param,"Found wrong packed record at %s", @@ -3231,7 +3423,7 @@ static int sort_get_next_record(MARIA_SORT_PARAM *sort_param) goto try_next; } } - if (param->calc_checksum) + if (sort_param->calc_checksum) param->glob_crc+= info->checksum; DBUG_RETURN(0); } @@ -3258,7 +3450,8 @@ static int sort_get_next_record(MARIA_SORT_PARAM *sort_param) DBUG_RETURN(1); /* Something wrong with data */ } sort_param->start_recpos=sort_param->pos; - if (_ma_pack_get_block_info(info,&block_info,-1,sort_param->pos)) + if (_ma_pack_get_block_info(info, &sort_param->bit_buff, &block_info, + &sort_param->rec_buff, -1, sort_param->pos)) DBUG_RETURN(-1); if (!block_info.rec_len && sort_param->pos + MEMMAP_EXTRA_MARGIN == @@ -3282,15 +3475,14 @@ static int sort_get_next_record(MARIA_SORT_PARAM *sort_param) llstr(sort_param->pos,llbuff)); continue; } - if (_ma_pack_rec_unpack(info,sort_param->record,sort_param->rec_buff, - block_info.rec_len)) + if (_ma_pack_rec_unpack(info, &sort_param->bit_buff, sort_param->record, + sort_param->rec_buff, block_info.rec_len)) { if (! searching) _ma_check_print_info(param,"Found wrong record at %s", llstr(sort_param->pos,llbuff)); continue; } - info->checksum=_ma_checksum(info,sort_param->record); if (!sort_param->fix_datafile) { sort_param->filepos=sort_param->pos; @@ -3300,8 +3492,9 @@ static int sort_get_next_record(MARIA_SORT_PARAM *sort_param) sort_param->max_pos=(sort_param->pos=block_info.filepos+ block_info.rec_len); info->packed_length=block_info.rec_len; - if (param->calc_checksum) - param->glob_crc+= info->checksum; + if (sort_param->calc_checksum) + param->glob_crc+= (info->checksum= + _ma_checksum(info, sort_param->record)); DBUG_RETURN(0); } } @@ -3309,7 +3502,20 @@ static int sort_get_next_record(MARIA_SORT_PARAM *sort_param) } - /* Write record to new file */ +/* + Write record to new file. + + SYNOPSIS + _ma_sort_write_record() + sort_param Sort parameters. + + NOTE + This is only called by a master thread if parallel repair is used. + + RETURN + 0 OK + 1 Error +*/ int _ma_sort_write_record(MARIA_SORT_PARAM *sort_param) { @@ -3358,6 +3564,7 @@ int _ma_sort_write_record(MARIA_SORT_PARAM *sort_param) } from=sort_info->buff+ALIGN_SIZE(MARIA_MAX_DYN_BLOCK_HEADER); } + /* We can use info->checksum here as only one thread calls this. */ info->checksum=_ma_checksum(info,sort_param->record); reclength= _ma_rec_pack(info,from,sort_param->record); flag=0; @@ -3767,7 +3974,7 @@ static int sort_delete_record(MARIA_SORT_PARAM *sort_param) DBUG_RETURN(1); } } - if (param->calc_checksum) + if (sort_param->calc_checksum) param->glob_crc-=(*info->s->calc_checksum)(info, sort_param->record); } error=flush_io_cache(&info->rec_cache) || (*info->s->delete_record)(info); diff --git a/storage/maria/ma_open.c b/storage/maria/ma_open.c index cf21ccb09e5..4041c101bde 100644 --- a/storage/maria/ma_open.c +++ b/storage/maria/ma_open.c @@ -211,7 +211,10 @@ MARIA_HA *maria_open(const char *name, int mode, uint open_flags) ((open_flags & HA_OPEN_ABORT_IF_CRASHED) && (my_disable_locking && share->state.open_count)))) { - DBUG_PRINT("error",("Table is marked as crashed")); + DBUG_PRINT("error",("Table is marked as crashed. open_flags: %u " + "changed: %u open_count: %u !locking: %d", + open_flags, share->state.changed, + share->state.open_count, my_disable_locking)); my_errno=((share->state.changed & STATE_CRASHED_ON_REPAIR) ? HA_ERR_CRASHED_ON_REPAIR : HA_ERR_CRASHED_ON_USAGE); goto err; diff --git a/storage/maria/ma_packrec.c b/storage/maria/ma_packrec.c index 3be893f39f8..c29cf1a672a 100644 --- a/storage/maria/ma_packrec.c +++ b/storage/maria/ma_packrec.c @@ -104,7 +104,8 @@ static void fill_buffer(MARIA_BIT_BUFF *bit_buff); static uint max_bit(uint value); static uint read_pack_length(uint version, const uchar *buf, ulong *length); #ifdef HAVE_MMAP -static uchar *_ma_mempack_get_block_info(MARIA_HA *maria,MARIA_BLOCK_INFO *info, +static uchar *_ma_mempack_get_block_info(MARIA_HA *maria, MARIA_BIT_BUFF *bit_buff, + MARIA_BLOCK_INFO *info, byte **rec_buff_p, uchar *header); #endif @@ -450,13 +451,15 @@ int _ma_read_pack_record(MARIA_HA *info, my_off_t filepos, byte *buf) DBUG_RETURN(-1); /* _search() didn't find record */ file=info->dfile; - if (_ma_pack_get_block_info(info, &block_info, file, filepos)) + if (_ma_pack_get_block_info(info, &info->bit_buff, &block_info, + &info->rec_buff, file, filepos)) goto err; if (my_read(file,(byte*) info->rec_buff + block_info.offset , block_info.rec_len - block_info.offset, MYF(MY_NABP))) goto panic; info->update|= HA_STATE_AKTIV; - DBUG_RETURN(_ma_pack_rec_unpack(info,buf,info->rec_buff,block_info.rec_len)); + DBUG_RETURN(_ma_pack_rec_unpack(info,&info->bit_buff, buf, + info->rec_buff, block_info.rec_len)); panic: my_errno=HA_ERR_WRONG_IN_RECORD; err: @@ -465,8 +468,8 @@ err: -int _ma_pack_rec_unpack(register MARIA_HA *info, register byte *to, byte *from, - ulong reclength) +int _ma_pack_rec_unpack(register MARIA_HA *info, MARIA_BIT_BUFF *bit_buff, + register byte *to, byte *from, ulong reclength) { byte *end_field; reg3 MARIA_COLUMNDEF *end; @@ -474,18 +477,18 @@ int _ma_pack_rec_unpack(register MARIA_HA *info, register byte *to, byte *from, MARIA_SHARE *share=info->s; DBUG_ENTER("_ma_pack_rec_unpack"); - init_bit_buffer(&info->bit_buff, (uchar*) from,reclength); + init_bit_buffer(bit_buff, (uchar*) from, reclength); for (current_field=share->rec, end=current_field+share->base.fields ; current_field < end ; current_field++,to=end_field) { end_field=to+current_field->length; - (*current_field->unpack)(current_field,&info->bit_buff,(uchar*) to, + (*current_field->unpack)(current_field, bit_buff, (uchar*) to, (uchar*) end_field); } - if (! info->bit_buff.error && - info->bit_buff.pos - info->bit_buff.bits/8 == info->bit_buff.end) + if (!bit_buff->error && + bit_buff->pos - bit_buff->bits / 8 == bit_buff->end) DBUG_RETURN(0); info->update&= ~HA_STATE_AKTIV; DBUG_RETURN(my_errno=HA_ERR_WRONG_IN_RECORD); @@ -1016,13 +1019,16 @@ int _ma_read_rnd_pack_record(MARIA_HA *info, byte *buf, if (info->opt_flag & READ_CACHE_USED) { - if (_ma_read_cache(&info->rec_cache,(byte*) block_info.header,filepos, - share->pack.ref_length, skip_deleted_blocks)) + if (_ma_read_cache(&info->rec_cache, (byte*) block_info.header, + filepos, share->pack.ref_length, + skip_deleted_blocks ? READING_NEXT : 0)) goto err; - b_type= _ma_pack_get_block_info(info,&block_info,-1, filepos); + b_type= _ma_pack_get_block_info(info, &info->bit_buff, &block_info, + &info->rec_buff, -1, filepos); } else - b_type= _ma_pack_get_block_info(info,&block_info,info->dfile,filepos); + b_type= _ma_pack_get_block_info(info, &info->bit_buff, &block_info, + &info->rec_buff, info->dfile, filepos); if (b_type) goto err; /* Error code is already set */ #ifndef DBUG_OFF @@ -1035,9 +1041,9 @@ int _ma_read_rnd_pack_record(MARIA_HA *info, byte *buf, if (info->opt_flag & READ_CACHE_USED) { - if (_ma_read_cache(&info->rec_cache,(byte*) info->rec_buff, - block_info.filepos, block_info.rec_len, - skip_deleted_blocks)) + if (_ma_read_cache(&info->rec_cache, (byte*) info->rec_buff, + block_info.filepos, block_info.rec_len, + skip_deleted_blocks ? READING_NEXT : 0)) goto err; } else @@ -1052,8 +1058,8 @@ int _ma_read_rnd_pack_record(MARIA_HA *info, byte *buf, info->nextpos=block_info.filepos+block_info.rec_len; info->update|= HA_STATE_AKTIV | HA_STATE_KEY_CHANGED; - DBUG_RETURN (_ma_pack_rec_unpack(info,buf,info->rec_buff, - block_info.rec_len)); + DBUG_RETURN (_ma_pack_rec_unpack(info, &info->bit_buff, buf, + info->rec_buff, block_info.rec_len)); err: DBUG_RETURN(my_errno); } @@ -1061,8 +1067,9 @@ int _ma_read_rnd_pack_record(MARIA_HA *info, byte *buf, /* Read and process header from a huff-record-file */ -uint _ma_pack_get_block_info(MARIA_HA *maria, MARIA_BLOCK_INFO *info, File file, - my_off_t filepos) +uint _ma_pack_get_block_info(MARIA_HA *maria, MARIA_BIT_BUFF *bit_buff, + MARIA_BLOCK_INFO *info, byte **rec_buff_p, + File file, my_off_t filepos) { uchar *header=info->header; uint head_length,ref_length; @@ -1087,17 +1094,17 @@ uint _ma_pack_get_block_info(MARIA_HA *maria, MARIA_BLOCK_INFO *info, File file, head_length+= read_pack_length((uint) maria->s->pack.version, header + head_length, &info->blob_len); if (!(_ma_alloc_rec_buff(maria,info->rec_len + info->blob_len, - &maria->rec_buff))) + rec_buff_p))) return BLOCK_FATAL_ERROR; /* not enough memory */ - maria->bit_buff.blob_pos=(uchar*) maria->rec_buff+info->rec_len; - maria->bit_buff.blob_end= maria->bit_buff.blob_pos+info->blob_len; + bit_buff->blob_pos= (uchar*) *rec_buff_p + info->rec_len; + bit_buff->blob_end= bit_buff->blob_pos + info->blob_len; maria->blob_length=info->blob_len; } info->filepos=filepos+head_length; if (file > 0) { info->offset=min(info->rec_len, ref_length - head_length); - memcpy(maria->rec_buff, header+head_length, info->offset); + memcpy(*rec_buff_p, header + head_length, info->offset); } return 0; } @@ -1215,7 +1222,8 @@ void _ma_unmap_file(MARIA_HA *info) } -static uchar *_ma_mempack_get_block_info(MARIA_HA *maria,MARIA_BLOCK_INFO *info, +static uchar *_ma_mempack_get_block_info(MARIA_HA *maria, MARIA_BIT_BUFF *bit_buff, + MARIA_BLOCK_INFO *info, byte **rec_buff_p, uchar *header) { header+= read_pack_length((uint) maria->s->pack.version, header, @@ -1226,10 +1234,10 @@ static uchar *_ma_mempack_get_block_info(MARIA_HA *maria,MARIA_BLOCK_INFO *info, &info->blob_len); /* _ma_alloc_rec_buff sets my_errno on error */ if (!(_ma_alloc_rec_buff(maria, info->blob_len, - &maria->rec_buff))) + rec_buff_p))) return 0; /* not enough memory */ - maria->bit_buff.blob_pos=(uchar*) maria->rec_buff; - maria->bit_buff.blob_end= (uchar*) maria->rec_buff + info->blob_len; + bit_buff->blob_pos= (uchar*) *rec_buff_p; + bit_buff->blob_end= (uchar*) *rec_buff_p + info->blob_len; } return header; } @@ -1245,11 +1253,13 @@ static int _ma_read_mempack_record(MARIA_HA *info, my_off_t filepos, byte *buf) if (filepos == HA_OFFSET_ERROR) DBUG_RETURN(-1); /* _search() didn't find record */ - if (!(pos= (byte*) _ma_mempack_get_block_info(info,&block_info, + if (!(pos= (byte*) _ma_mempack_get_block_info(info, &info->bit_buff, + &block_info, &info->rec_buff, (uchar*) share->file_map+ filepos))) DBUG_RETURN(-1); - DBUG_RETURN(_ma_pack_rec_unpack(info, buf, pos, block_info.rec_len)); + DBUG_RETURN(_ma_pack_rec_unpack(info, &info->bit_buff, buf, + pos, block_info.rec_len)); } @@ -1269,7 +1279,8 @@ static int _ma_read_rnd_mempack_record(MARIA_HA *info, byte *buf, my_errno=HA_ERR_END_OF_FILE; goto err; } - if (!(pos= (byte*) _ma_mempack_get_block_info(info,&block_info, + if (!(pos= (byte*) _ma_mempack_get_block_info(info, &info->bit_buff, + &block_info, &info->rec_buff, (uchar*) (start=share->file_map+ filepos)))) @@ -1286,7 +1297,8 @@ static int _ma_read_rnd_mempack_record(MARIA_HA *info, byte *buf, info->nextpos=filepos+(uint) (pos-start)+block_info.rec_len; info->update|= HA_STATE_AKTIV | HA_STATE_KEY_CHANGED; - DBUG_RETURN (_ma_pack_rec_unpack(info,buf,pos, block_info.rec_len)); + DBUG_RETURN (_ma_pack_rec_unpack(info, &info->bit_buff, buf, + pos, block_info.rec_len)); err: DBUG_RETURN(my_errno); } diff --git a/storage/maria/ma_range.c b/storage/maria/ma_range.c index 0f6883f4c9d..c837fe5e6e3 100644 --- a/storage/maria/ma_range.c +++ b/storage/maria/ma_range.c @@ -71,6 +71,21 @@ ha_rows maria_records_in_range(MARIA_HA *info, int inx, key_range *min_key, uchar * key_buff; uint start_key_len; + /* + The problem is that the optimizer doesn't support + RTree keys properly at the moment. + Hope this will be fixed some day. + But now NULL in the min_key means that we + didn't make the task for the RTree key + and expect BTree functionality from it. + As it's not able to handle such request + we return the error. + */ + if (!min_key) + { + res= HA_POS_ERROR; + break; + } key_buff= info->lastkey+info->s->base.max_key_length; start_key_len= _ma_pack_key(info,inx, key_buff, (uchar*) min_key->key, min_key->length, diff --git a/storage/maria/ma_sort.c b/storage/maria/ma_sort.c index 795bfdb7fda..c9e3a369851 100644 --- a/storage/maria/ma_sort.c +++ b/storage/maria/ma_sort.c @@ -20,6 +20,11 @@ */ #include "ma_fulltext.h" +#if defined(MSDOS) || defined(__WIN__) +#include +#else +#include +#endif #include /* static variables */ @@ -143,7 +148,8 @@ int _ma_create_index_by_sort(MARIA_SORT_PARAM *info,my_bool no_messages, skr=maxbuffer; if (memavl < sizeof(BUFFPEK)*(uint) maxbuffer || (keys=(memavl-sizeof(BUFFPEK)*(uint) maxbuffer)/ - (sort_length+sizeof(char*))) <= 1) + (sort_length+sizeof(char*))) <= 1 || + keys < (uint) maxbuffer) { _ma_check_print_error(info->sort_info->param, "sort_buffer_size is to small"); @@ -304,7 +310,7 @@ static ha_rows NEAR_F find_all_keys(MARIA_SORT_PARAM *info, uint keys, pthread_handler_t _ma_thr_find_all_keys(void *arg) { - MARIA_SORT_PARAM *info= (MARIA_SORT_PARAM*) arg; + MARIA_SORT_PARAM *sort_param= (MARIA_SORT_PARAM*) arg; int error; uint memavl,old_memavl,keys,sort_length; uint idx, maxbuffer; @@ -316,138 +322,163 @@ pthread_handler_t _ma_thr_find_all_keys(void *arg) if (my_thread_init()) goto err; - if (info->sort_info->got_error) - goto err; - if (info->keyinfo->flag && HA_VAR_LENGTH_KEY) - { - info->write_keys=write_keys_varlen; - info->read_to_buffer=read_to_buffer_varlen; - info->write_key=write_merge_key_varlen; - } - else - { - info->write_keys=write_keys; - info->read_to_buffer=read_to_buffer; - info->write_key=write_merge_key; - } + { /* Add extra block since DBUG_ENTER declare variables */ + DBUG_ENTER("_ma_thr_find_all_keys"); + DBUG_PRINT("enter", ("master: %d", sort_param->master)); + if (sort_param->sort_info->got_error) + goto err; - my_b_clear(&info->tempfile); - my_b_clear(&info->tempfile_for_exceptions); - bzero((char*) &info->buffpek,sizeof(info->buffpek)); - bzero((char*) &info->unique, sizeof(info->unique)); - sort_keys= (uchar **) NULL; + if (sort_param->keyinfo->flag & HA_VAR_LENGTH_KEY) + { + sort_param->write_keys= write_keys_varlen; + sort_param->read_to_buffer= read_to_buffer_varlen; + sort_param->write_key= write_merge_key_varlen; + } + else + { + sort_param->write_keys= write_keys; + sort_param->read_to_buffer= read_to_buffer; + sort_param->write_key= write_merge_key; + } - memavl=max(info->sortbuff_size, MIN_SORT_MEMORY); - idx= info->sort_info->max_records; - sort_length= info->key_length; - maxbuffer= 1; + my_b_clear(&sort_param->tempfile); + my_b_clear(&sort_param->tempfile_for_exceptions); + bzero((char*) &sort_param->buffpek,sizeof(sort_param->buffpek)); + bzero((char*) &sort_param->unique, sizeof(sort_param->unique)); + sort_keys= (uchar **) NULL; - while (memavl >= MIN_SORT_MEMORY) - { - if ((my_off_t) (idx+1)*(sort_length+sizeof(char*)) <= - (my_off_t) memavl) - keys= idx+1; - else + memavl= max(sort_param->sortbuff_size, MIN_SORT_MEMORY); + idx= sort_param->sort_info->max_records; + sort_length= sort_param->key_length; + maxbuffer= 1; + + while (memavl >= MIN_SORT_MEMORY) { - uint skr; - do + if ((my_off_t) (idx+1)*(sort_length+sizeof(char*)) <= + (my_off_t) memavl) + keys= idx+1; + else { - skr=maxbuffer; - if (memavl < sizeof(BUFFPEK)*maxbuffer || - (keys=(memavl-sizeof(BUFFPEK)*maxbuffer)/ - (sort_length+sizeof(char*))) <= 1) + uint skr; + do { - _ma_check_print_error(info->sort_info->param, - "sort_buffer_size is to small"); - goto err; + skr= maxbuffer; + if (memavl < sizeof(BUFFPEK)*maxbuffer || + (keys=(memavl-sizeof(BUFFPEK)*maxbuffer)/ + (sort_length+sizeof(char*))) <= 1 || + keys < (uint) maxbuffer) + { + _ma_check_print_error(sort_param->sort_info->param, + "sort_buffer_size is to small"); + goto err; + } } + while ((maxbuffer= (int) (idx/(keys-1)+1)) != skr); } - while ((maxbuffer= (int) (idx/(keys-1)+1)) != skr); - } - if ((sort_keys=(uchar **)my_malloc(keys*(sort_length+sizeof(char*))+ - ((info->keyinfo->flag & HA_FULLTEXT) ? - HA_FT_MAXBYTELEN : 0), MYF(0)))) - { - if (my_init_dynamic_array(&info->buffpek, sizeof(BUFFPEK), - maxbuffer, maxbuffer/2)) + if ((sort_keys= (uchar **) + my_malloc(keys*(sort_length+sizeof(char*))+ + ((sort_param->keyinfo->flag & HA_FULLTEXT) ? + HA_FT_MAXBYTELEN : 0), MYF(0)))) { - my_free((gptr) sort_keys,MYF(0)); - sort_keys= (uchar **) NULL; /* for err: label */ + if (my_init_dynamic_array(&sort_param->buffpek, sizeof(BUFFPEK), + maxbuffer, maxbuffer/2)) + { + my_free((gptr) sort_keys,MYF(0)); + sort_keys= (uchar **) NULL; /* for err: label */ + } + else + break; } - else - break; + old_memavl= memavl; + if ((memavl= memavl/4*3) < MIN_SORT_MEMORY && + old_memavl > MIN_SORT_MEMORY) + memavl= MIN_SORT_MEMORY; + } + if (memavl < MIN_SORT_MEMORY) + { + _ma_check_print_error(sort_param->sort_info->param, + "Sort buffer too small"); + goto err; /* purecov: tested */ } - old_memavl=memavl; - if ((memavl=memavl/4*3) < MIN_SORT_MEMORY && old_memavl > MIN_SORT_MEMORY) - memavl=MIN_SORT_MEMORY; - } - if (memavl < MIN_SORT_MEMORY) - { - _ma_check_print_error(info->sort_info->param,"Sort buffer to small"); /* purecov: tested */ - goto err; /* purecov: tested */ - } - if (info->sort_info->param->testflag & T_VERBOSE) - printf("Key %d - Allocating buffer for %d keys\n",info->key+1,keys); - info->sort_keys=sort_keys; + if (sort_param->sort_info->param->testflag & T_VERBOSE) + printf("Key %d - Allocating buffer for %d keys\n", + sort_param->key+1, keys); + sort_param->sort_keys= sort_keys; - idx=error=0; - sort_keys[0]=(uchar*) (sort_keys+keys); + idx= error= 0; + sort_keys[0]= (uchar*) (sort_keys+keys); - while (!(error=info->sort_info->got_error) && - !(error=(*info->key_read)(info,sort_keys[idx]))) - { - if (info->real_key_length > info->key_length) + DBUG_PRINT("info", ("reading keys")); + while (!(error= sort_param->sort_info->got_error) && + !(error= (*sort_param->key_read)(sort_param, sort_keys[idx]))) { - if (write_key(info,sort_keys[idx], &info->tempfile_for_exceptions)) - goto err; - continue; - } + if (sort_param->real_key_length > sort_param->key_length) + { + if (write_key(sort_param,sort_keys[idx], + &sort_param->tempfile_for_exceptions)) + goto err; + continue; + } - if (++idx == keys) + if (++idx == keys) + { + if (sort_param->write_keys(sort_param, sort_keys, idx - 1, + (BUFFPEK *)alloc_dynamic(&sort_param->buffpek), + &sort_param->tempfile)) + goto err; + sort_keys[0]= (uchar*) (sort_keys+keys); + memcpy(sort_keys[0], sort_keys[idx - 1], (size_t) sort_param->key_length); + idx= 1; + } + sort_keys[idx]=sort_keys[idx - 1] + sort_param->key_length; + } + if (error > 0) + goto err; + if (sort_param->buffpek.elements) { - if (info->write_keys(info,sort_keys,idx-1, - (BUFFPEK *)alloc_dynamic(&info->buffpek), - &info->tempfile)) + if (sort_param->write_keys(sort_param,sort_keys, idx, + (BUFFPEK *) alloc_dynamic(&sort_param->buffpek), &sort_param->tempfile)) goto err; - sort_keys[0]=(uchar*) (sort_keys+keys); - memcpy(sort_keys[0],sort_keys[idx-1],(size_t) info->key_length); - idx=1; + sort_param->keys= (sort_param->buffpek.elements - 1) * (keys - 1) + idx; } - sort_keys[idx]=sort_keys[idx-1]+info->key_length; - } - if (error > 0) - goto err; - if (info->buffpek.elements) - { - if (info->write_keys(info,sort_keys, idx, - (BUFFPEK *) alloc_dynamic(&info->buffpek), &info->tempfile)) - goto err; - info->keys=(info->buffpek.elements-1)*(keys-1)+idx; - } - else - info->keys=idx; + else + sort_param->keys= idx; - info->sort_keys_length=keys; - goto ok; + sort_param->sort_keys_length= keys; + goto ok; err: - info->sort_info->got_error=1; /* no need to protect this with a mutex */ - if (sort_keys) - my_free((gptr) sort_keys,MYF(0)); - info->sort_keys=0; - delete_dynamic(& info->buffpek); - close_cached_file(&info->tempfile); - close_cached_file(&info->tempfile_for_exceptions); + DBUG_PRINT("error", ("got some error")); + sort_param->sort_info->got_error= 1; /* no need to protect with a mutex */ + if (sort_keys) + my_free((gptr) sort_keys,MYF(0)); + sort_param->sort_keys=0; + delete_dynamic(& sort_param->buffpek); + close_cached_file(&sort_param->tempfile); + close_cached_file(&sort_param->tempfile_for_exceptions); ok: - free_root(&info->wordroot, MYF(0)); - remove_io_thread(&info->read_cache); - pthread_mutex_lock(&info->sort_info->mutex); - info->sort_info->threads_running--; - pthread_cond_signal(&info->sort_info->cond); - pthread_mutex_unlock(&info->sort_info->mutex); + free_root(&sort_param->wordroot, MYF(0)); + /* + Detach from the share if the writer is involved. Avoid others to + be blocked. This includes a flush of the write buffer. This will + also indicate EOF to the readers. + */ + if (sort_param->sort_info->info->rec_cache.share) + remove_io_thread(&sort_param->sort_info->info->rec_cache); + + /* Readers detach from the share if any. Avoid others to be blocked. */ + if (sort_param->read_cache.share) + remove_io_thread(&sort_param->read_cache); + + pthread_mutex_lock(&sort_param->sort_info->mutex); + if (!--sort_param->sort_info->threads_running) + pthread_cond_signal(&sort_param->sort_info->cond); + pthread_mutex_unlock(&sort_param->sort_info->mutex); + DBUG_PRINT("exit", ("======== ending thread ========")); + } my_thread_end(); return NULL; } @@ -465,6 +496,7 @@ int _ma_thr_write_keys(MARIA_SORT_PARAM *sort_param) MARIA_SHARE *share=info->s; MARIA_SORT_PARAM *sinfo; byte *mergebuf=0; + DBUG_ENTER("_ma_thr_write_keys"); LINT_INIT(length); for (i= 0, sinfo= sort_param ; @@ -474,6 +506,8 @@ int _ma_thr_write_keys(MARIA_SORT_PARAM *sort_param) if (!sinfo->sort_keys) { got_error=1; + my_free(_ma_get_rec_buff_ptr(info, sinfo->rec_buff), + MYF(MY_ALLOW_ZERO_PTR)); continue; } if (!got_error) @@ -513,7 +547,7 @@ int _ma_thr_write_keys(MARIA_SORT_PARAM *sort_param) { if (got_error) continue; - if (sinfo->keyinfo->flag && HA_VAR_LENGTH_KEY) + if (sinfo->keyinfo->flag & HA_VAR_LENGTH_KEY) { sinfo->write_keys=write_keys_varlen; sinfo->read_to_buffer=read_to_buffer_varlen; @@ -602,7 +636,7 @@ int _ma_thr_write_keys(MARIA_SORT_PARAM *sort_param) } } my_free((gptr) mergebuf,MYF(MY_ALLOW_ZERO_PTR)); - return got_error; + DBUG_RETURN(got_error); } #endif /* THREAD */ diff --git a/storage/maria/maria_def.h b/storage/maria/maria_def.h index 62c50187888..5fb25512d4c 100644 --- a/storage/maria/maria_def.h +++ b/storage/maria/maria_def.h @@ -69,6 +69,7 @@ typedef struct st_maria_state_info ulong update_count; /* Updated for each write lock */ ulong status; ulong *rec_per_key_part; + ha_checksum checksum; /* Table checksum */ my_off_t *key_root; /* Start of key trees */ my_off_t *key_del; /* delete links for trees */ my_off_t rec_per_key_rows; /* Rows when calculating rec_per_key */ @@ -179,6 +180,7 @@ typedef struct st_maria_share */ MARIA_DECODE_TREE *decode_trees; uint16 *decode_tables; + /* Function to use for a row checksum. */ int(*read_record) (struct st_maria_info *, my_off_t, byte *); int(*write_record) (struct st_maria_info *, const byte *); int(*update_record) (struct st_maria_info *, my_off_t, const byte *); @@ -230,16 +232,6 @@ typedef struct st_maria_share } MARIA_SHARE; -typedef uint maria_bit_type; - -typedef struct st_maria_bit_buff -{ /* Used for packing of record */ - maria_bit_type current_byte; - uint bits; - uchar *pos, *end, *blob_pos, *blob_end; - uint error; -} MARIA_BIT_BUFF; - struct st_maria_info { MARIA_SHARE *s; /* Shared between open:s */ @@ -273,7 +265,7 @@ struct st_maria_info my_off_t last_keypage; /* Last key page read */ my_off_t last_search_keypage; /* Last keypage when searching */ my_off_t dupp_key_pos; - ha_checksum checksum; + ha_checksum checksum; /* Temp storage for row checksum */ /* QQ: the folloing two xxx_length fields should be removed, as they are not compatible with parallel repair @@ -354,8 +346,15 @@ struct st_maria_info #define maria_putint(x,y,nod) { uint16 boh=(nod ? (uint16) 32768 : 0) + (uint16) (y);\ mi_int2store(x,boh); } #define _ma_test_if_nod(x) (x[0] & 128 ? info->s->base.key_reflength : 0) -#define maria_mark_crashed(x) (x)->s->state.changed|=STATE_CRASHED -#define maria_mark_crashed_on_repair(x) { (x)->s->state.changed|=STATE_CRASHED|STATE_CRASHED_ON_REPAIR ; (x)->update|= HA_STATE_CHANGED; } +#define maria_mark_crashed(x) do{(x)->s->state.changed|= STATE_CRASHED; \ + DBUG_PRINT("error", ("Marked table crashed")); \ + }while(0) +#define maria_mark_crashed_on_repair(x) do{(x)->s->state.changed|= \ + STATE_CRASHED|STATE_CRASHED_ON_REPAIR; \ + (x)->update|= HA_STATE_CHANGED; \ + DBUG_PRINT("error", \ + ("Marked table crashed")); \ + }while(0) #define maria_is_crashed(x) ((x)->s->state.changed & STATE_CRASHED) #define maria_is_crashed_on_repair(x) ((x)->s->state.changed & STATE_CRASHED_ON_REPAIR) #define maria_print_error(SHARE, ERRNO) \ @@ -608,8 +607,8 @@ extern my_bool _ma_read_pack_info(MARIA_HA *info, pbool fix_keys); extern int _ma_read_pack_record(MARIA_HA *info, my_off_t filepos, byte *buf); extern int _ma_read_rnd_pack_record(MARIA_HA *, byte *, my_off_t, my_bool); -extern int _ma_pack_rec_unpack(MARIA_HA *info, byte *to, byte *from, - ulong reclength); +extern int _ma_pack_rec_unpack(MARIA_HA *info, MARIA_BIT_BUFF *bit_buff, + byte *to, byte *from, ulong reclength); extern ulonglong _ma_safe_mul(ulonglong a, ulonglong b); extern int _ma_ft_update(MARIA_HA *info, uint keynr, byte *keybuf, const byte *oldrec, const byte *newrec, @@ -663,8 +662,9 @@ typedef struct st_maria_block_info extern uint _ma_get_block_info(MARIA_BLOCK_INFO *, File, my_off_t); extern uint _ma_rec_pack(MARIA_HA *info, byte *to, const byte *from); -extern uint _ma_pack_get_block_info(MARIA_HA *, MARIA_BLOCK_INFO *, File, - my_off_t); +extern uint _ma_pack_get_block_info(MARIA_HA *mari, MARIA_BIT_BUFF *bit_buff, + MARIA_BLOCK_INFO *info, byte **rec_buff_p, + File file, my_off_t filepos); extern void _ma_store_blob_length(byte *pos, uint pack_length, uint length); extern void _ma_report_error(int errcode, const char *file_name); extern my_bool _ma_memmap_file(MARIA_HA *info); diff --git a/storage/maria/maria_pack.c b/storage/maria/maria_pack.c index c5a53b1ffac..5a5aaf037f0 100644 --- a/storage/maria/maria_pack.c +++ b/storage/maria/maria_pack.c @@ -1967,7 +1967,7 @@ static char *bindigits(ulonglong value, uint bits) DBUG_ASSERT(idx < sizeof(digits)); while (idx) - *(ptr++)= '0' + ((value >> (--idx)) & 1); + *(ptr++)= '0' + ((char) (value >> (--idx)) & (char) 1); *ptr= '\0'; return digits; } @@ -1997,7 +1997,7 @@ static char *hexdigits(ulonglong value) DBUG_ASSERT(idx < sizeof(digits)); while (idx) { - if ((*(ptr++)= '0' + ((value >> (4 * (--idx))) & 0xf)) > '9') + if ((*(ptr++)= '0' + ((char) (value >> (4 * (--idx))) & (char) 0xf)) > '9') *(ptr - 1)+= 'a' - '9' - 1; } *ptr= '\0'; @@ -2286,7 +2286,7 @@ static my_off_t write_huff_tree(HUFF_TREE *huff_tree, uint trees) errors++; break; } - idx+= code & 1; + idx+= (uint) code & 1; if (idx >= length) { VOID(fflush(stdout)); diff --git a/storage/maria/plug.in b/storage/maria/plug.in index a9b35aefbfb..198f5d8c289 100644 --- a/storage/maria/plug.in +++ b/storage/maria/plug.in @@ -5,3 +5,4 @@ MYSQL_PLUGIN_ACTIONS(maria, [AC_CONFIG_FILES(storage/maria/unittest/Makefile)]) MYSQL_PLUGIN_STATIC(maria, [libmaria.a]) # Maria will probably go first into max builds, not all builds, # so we don't declare it mandatory. +MYSQL_PLUGIN_DEPENDS_ON_MYSQL_INTERNALS(maria, [ha_maria.cc]) diff --git a/storage/myisam/myisamdef.h b/storage/myisam/myisamdef.h index d579213d7f9..be27c13dbe5 100644 --- a/storage/myisam/myisamdef.h +++ b/storage/myisam/myisamdef.h @@ -84,6 +84,7 @@ typedef struct st_mi_state_info time_t recover_time; /* Time for last recover */ time_t check_time; /* Time for last check */ uint sortkey; /* sorted by this key (not used) */ + uint open_count; uint8 changed; /* Changed since myisamchk */ /* the following isn't saved on disk */ @@ -224,16 +225,6 @@ typedef struct st_mi_isam_share } MYISAM_SHARE; -typedef uint mi_bit_type; - -typedef struct st_mi_bit_buff -{ /* Used for packing of record */ - mi_bit_type current_byte; - uint bits; - uchar *pos, *end, *blob_pos, *blob_end; - uint error; -} MI_BIT_BUFF; - struct st_myisam_info { MYISAM_SHARE *s; /* Shared between open:s */ @@ -309,6 +300,7 @@ struct st_myisam_info uchar *rtree_recursion_state; /* For RTREE */ int rtree_recursion_depth; }; + #define USE_WHOLE_KEY HA_MAX_KEY_BUFF*2 /* Use whole key in _mi_search() */ #define F_EXTRA_LCK -1 /* bits in opt_flag */ diff --git a/storage/myisam/myisampack.c b/storage/myisam/myisampack.c index be68ffbdc5a..d67fa8fa918 100644 --- a/storage/myisam/myisampack.c +++ b/storage/myisam/myisampack.c @@ -1105,16 +1105,16 @@ static int get_statistic(PACK_MRG_INFO *mrg,HUFF_COUNTS *huff_counts) my_off_t total_count; char llbuf[32]; - DBUG_PRINT("info", ("column: %3u", count - huff_counts + 1)); + DBUG_PRINT("info", ("column: %3lu", count - huff_counts + 1)); if (verbose >= 2) - VOID(printf("column: %3u\n", count - huff_counts + 1)); + VOID(printf("column: %3lu\n", count - huff_counts + 1)); if (count->tree_buff) { - DBUG_PRINT("info", ("number of distinct values: %u", + DBUG_PRINT("info", ("number of distinct values: %lu", (count->tree_pos - count->tree_buff) / count->field_length)); if (verbose >= 2) - VOID(printf("number of distinct values: %u\n", + VOID(printf("number of distinct values: %lu\n", (count->tree_pos - count->tree_buff) / count->field_length)); } diff --git a/unittest/mysys/mf_pagecache_consist.c b/unittest/mysys/mf_pagecache_consist.c index f7724222a09..95c2135edad 100755 --- a/unittest/mysys/mf_pagecache_consist.c +++ b/unittest/mysys/mf_pagecache_consist.c @@ -332,14 +332,6 @@ int main(int argc, char **argv __attribute__((unused))) exit(1); } -#ifndef pthread_attr_setstacksize /* void return value */ - if ((error= pthread_attr_setstacksize(&thr_attr, 65536L))) - { - fprintf(stderr,"Got error: %d from pthread_attr_setstacksize (errno: %d)\n", - error,errno); - exit(1); - } -#endif #ifdef HAVE_THR_SETCONCURRENCY VOID(thr_setconcurrency(2)); #endif diff --git a/unittest/mysys/mf_pagecache_single.c b/unittest/mysys/mf_pagecache_single.c index 49ecd2986ab..9c467343156 100644 --- a/unittest/mysys/mf_pagecache_single.c +++ b/unittest/mysys/mf_pagecache_single.c @@ -511,14 +511,6 @@ int main(int argc, char **argv __attribute__((unused))) exit(1); } -#ifndef pthread_attr_setstacksize /* void return value */ - if ((error= pthread_attr_setstacksize(&thr_attr, 65536L))) - { - fprintf(stderr,"Got error: %d from pthread_attr_setstacksize (errno: %d)\n", - error,errno); - exit(1); - } -#endif #ifdef HAVE_THR_SETCONCURRENCY VOID(thr_setconcurrency(2)); #endif -- cgit v1.2.1 From 714f3b73e513f2d12fb45e8256fa6299e60cd5a2 Mon Sep 17 00:00:00 2001 From: unknown Date: Wed, 20 Dec 2006 18:58:35 +0100 Subject: merge of recent MyISAM changes into Maria. Only failure is ndb_restore - could have been worse!! include/pagecache.h: LSN->lsn mysys/mf_keycache.c: page_status is int mysys/mf_pagecache.c: merge of recent key cache changes sql/mysqld.cc: post-merge fixes sql/set_var.cc: post-merge fixes storage/maria/ha_maria.cc: merge of recent MyISAM changes into Maria storage/maria/ha_maria.h: merge of recent MyISAM changes into Maria storage/maria/ma_close.c: merge of recent MyISAM changes into Maria storage/maria/ma_create.c: merge of recent MyISAM changes into Maria storage/maria/ma_delete.c: merge of recent MyISAM changes into Maria storage/maria/ma_dynrec.c: merge of recent MyISAM changes into Maria storage/maria/ma_ft_boolean_search.c: merge of recent MyISAM changes into Maria storage/maria/ma_key.c: merge of recent MyISAM changes into Maria storage/maria/ma_keycache.c: merge of recent MyISAM changes into Maria storage/maria/ma_open.c: merge of recent MyISAM changes into Maria storage/maria/ma_page.c: merge of recent MyISAM changes into Maria storage/maria/ma_rsamepos.c: merge of recent MyISAM changes into Maria storage/maria/ma_statrec.c: merge of recent MyISAM changes into Maria storage/maria/ma_unique.c: merge of recent MyISAM changes into Maria storage/maria/maria_chk.c: merge of recent MyISAM changes into Maria storage/maria/maria_pack.c: merge of recent MyISAM changes into Maria storage/myisam/myisampack.c: compiler warning --- include/pagecache.h | 2 +- mysys/mf_keycache.c | 4 +-- mysys/mf_pagecache.c | 54 +++++++++++++++++++----------------- sql/mysqld.cc | 2 ++ sql/set_var.cc | 2 ++ storage/maria/ha_maria.cc | 18 +++++++----- storage/maria/ha_maria.h | 2 +- storage/maria/ma_close.c | 5 ++-- storage/maria/ma_create.c | 6 ++-- storage/maria/ma_delete.c | 9 +++--- storage/maria/ma_dynrec.c | 4 +-- storage/maria/ma_ft_boolean_search.c | 1 - storage/maria/ma_key.c | 15 +++++----- storage/maria/ma_keycache.c | 4 +-- storage/maria/ma_open.c | 23 +++++++++++++-- storage/maria/ma_page.c | 4 +-- storage/maria/ma_rsamepos.c | 3 +- storage/maria/ma_statrec.c | 4 +-- storage/maria/ma_unique.c | 4 +-- storage/maria/maria_chk.c | 1 + storage/maria/maria_pack.c | 36 ++++++++++++------------ storage/myisam/myisampack.c | 4 +-- 22 files changed, 121 insertions(+), 86 deletions(-) diff --git a/include/pagecache.h b/include/pagecache.h index 9f215325ae5..418e0203f64 100644 --- a/include/pagecache.h +++ b/include/pagecache.h @@ -221,7 +221,7 @@ extern my_bool pagecache_delete_page(PAGECACHE *pagecache, enum pagecache_page_lock lock, my_bool flush); extern void end_pagecache(PAGECACHE *keycache, my_bool cleanup); -extern my_bool pagecache_collect_changed_blocks_with_LSN(PAGECACHE *pagecache, +extern my_bool pagecache_collect_changed_blocks_with_lsn(PAGECACHE *pagecache, LEX_STRING *str, LSN *max_lsn); diff --git a/mysys/mf_keycache.c b/mysys/mf_keycache.c index 111a28d01a4..12be01184b6 100644 --- a/mysys/mf_keycache.c +++ b/mysys/mf_keycache.c @@ -1642,9 +1642,9 @@ restart: KEYCACHE_DBUG_ASSERT(page_status != -1); *page_st=page_status; KEYCACHE_DBUG_PRINT("find_key_block", - ("fd: %d pos: %lu block->status: %u page_status: %u", + ("fd: %d pos: %lu block->status: %u page_status: %d", file, (ulong) filepos, block->status, - (uint) page_status)); + page_status)); #if !defined(DBUG_OFF) && defined(EXTRA_DEBUG) DBUG_EXECUTE("check_keycache2", diff --git a/mysys/mf_pagecache.c b/mysys/mf_pagecache.c index 96c855fda0a..5f4e8d1a97d 100755 --- a/mysys/mf_pagecache.c +++ b/mysys/mf_pagecache.c @@ -759,9 +759,9 @@ int init_pagecache(PAGECACHE *pagecache, my_size_t use_mem, DBUG_PRINT("exit", ("disk_blocks: %d block_root: 0x%lx hash_entries: %d\ hash_root: 0x%lx hash_links: %d hash_link_root: 0x%lx", - pagecache->disk_blocks, pagecache->block_root, - pagecache->hash_entries, pagecache->hash_root, - pagecache->hash_links, pagecache->hash_link_root)); + pagecache->disk_blocks, (long) pagecache->block_root, + pagecache->hash_entries, (long) pagecache->hash_root, + pagecache->hash_links, (long) pagecache->hash_link_root)); bzero((gptr) pagecache->changed_blocks, sizeof(pagecache->changed_blocks[0]) * PAGECACHE_CHANGED_BLOCKS_HASH); @@ -985,7 +985,7 @@ void change_pagecache_param(PAGECACHE *pagecache, uint division_limit, void end_pagecache(PAGECACHE *pagecache, my_bool cleanup) { DBUG_ENTER("end_pagecache"); - DBUG_PRINT("enter", ("key_cache: 0x%lx", pagecache)); + DBUG_PRINT("enter", ("key_cache: 0x%lx", (long) pagecache)); if (!pagecache->inited) DBUG_VOID_RETURN; @@ -1004,7 +1004,7 @@ void end_pagecache(PAGECACHE *pagecache, my_bool cleanup) pagecache->blocks_changed= 0; } - DBUG_PRINT("status", ("used: %d changed: %d w_requests: %lu " + DBUG_PRINT("status", ("used: %lu changed: %lu w_requests: %lu " "writes: %lu r_requests: %lu reads: %lu", pagecache->blocks_used, pagecache->global_blocks_changed, (ulong) pagecache->global_cache_w_requests, @@ -1466,7 +1466,7 @@ static void unreg_request(PAGECACHE *pagecache, if (block->temperature == BLOCK_WARM) pagecache->warm_blocks--; block->temperature= BLOCK_HOT; - KEYCACHE_DBUG_PRINT("unreg_request", ("#warm_blocks=%u", + KEYCACHE_DBUG_PRINT("unreg_request", ("#warm_blocks: %lu", pagecache->warm_blocks)); } link_block(pagecache, block, hot, (my_bool)at_end); @@ -1485,7 +1485,7 @@ static void unreg_request(PAGECACHE *pagecache, pagecache->warm_blocks++; block->temperature= BLOCK_WARM; } - KEYCACHE_DBUG_PRINT("unreg_request", ("#warm_blocks=%u", + KEYCACHE_DBUG_PRINT("unreg_request", ("#warm_blocks: %lu", pagecache->warm_blocks)); } } @@ -1789,11 +1789,11 @@ static PAGECACHE_BLOCK_LINK *find_key_block(PAGECACHE *pagecache, DBUG_ENTER("find_key_block"); KEYCACHE_THREAD_TRACE("find_key_block:begin"); - DBUG_PRINT("enter", ("fd: %u pos %lu wrmode: %lu", - (uint) file->file, (ulong) pageno, (uint) wrmode)); - KEYCACHE_DBUG_PRINT("find_key_block", ("fd: %u pos: %lu wrmode: %lu", - (uint) file->file, (ulong) pageno, - (uint) wrmode)); + DBUG_PRINT("enter", ("fd: %d pos: %lu wrmode: %d", + file->file, (ulong) pageno, wrmode)); + KEYCACHE_DBUG_PRINT("find_key_block", ("fd: %d pos: %lu wrmode: %d", + file->file, (ulong) pageno, + wrmode)); #if !defined(DBUG_OFF) && defined(EXTRA_DEBUG) DBUG_EXECUTE("check_pagecache", test_key_cache(pagecache, "start of find_key_block", 0);); @@ -2103,14 +2103,14 @@ restart: KEYCACHE_DBUG_ASSERT(page_status != -1); *page_st=page_status; DBUG_PRINT("info", - ("block: 0x%lx fd: %u pos %lu block->status %u page_status %lu", + ("block: 0x%lx fd: %u pos %lu block->status %u page_status %u", (ulong) block, (uint) file->file, (ulong) pageno, block->status, (uint) page_status)); KEYCACHE_DBUG_PRINT("find_key_block", - ("block: 0x%lx fd: %u pos %lu block->status %u page_status %lu", + ("block: 0x%lx fd: %d pos: %lu block->status: %u page_status: %d", (ulong) block, - (uint) file->file, (ulong) pageno, block->status, - (uint) page_status)); + file->file, (ulong) pageno, block->status, + page_status)); #if !defined(DBUG_OFF) && defined(EXTRA_DEBUG) DBUG_EXECUTE("check_pagecache", @@ -3502,7 +3502,7 @@ static int flush_pagecache_blocks_int(PAGECACHE *pagecache, PAGECACHE_BLOCK_LINK *cache_buff[FLUSH_CACHE],**cache; int last_errno= 0; DBUG_ENTER("flush_pagecache_blocks_int"); - DBUG_PRINT("enter",("file: %d blocks_used: %d blocks_changed: %d", + DBUG_PRINT("enter",("file: %d blocks_used: %lu blocks_changed: %lu", file->file, pagecache->blocks_used, pagecache->blocks_changed)); #if !defined(DBUG_OFF) && defined(EXTRA_DEBUG) @@ -3714,7 +3714,7 @@ int flush_pagecache_blocks(PAGECACHE *pagecache, { int res; DBUG_ENTER("flush_pagecache_blocks"); - DBUG_PRINT("enter", ("pagecache: 0x%lx", pagecache)); + DBUG_PRINT("enter", ("pagecache: 0x%lx", (long) pagecache)); if (pagecache->disk_blocks <= 0) DBUG_RETURN(0); @@ -3800,7 +3800,7 @@ int reset_key_cache_counters(const char *name, PAGECACHE *key_cache) of type PAGECACHE_LSN_PAGE. SYNOPSIS - pagecache_collect_changed_blocks_with_LSN() + pagecache_collect_changed_blocks_with_lsn() pagecache pointer to the page cache str (OUT) pointer to a LEX_STRING where the allocated buffer, and its size, will be put @@ -3809,7 +3809,8 @@ int reset_key_cache_counters(const char *name, PAGECACHE *key_cache) DESCRIPTION Does the allocation because the caller cannot know the size itself. - Memory freeing is done by the caller. + Memory freeing is to be done by the caller (if the "str" member of the + LEX_STRING is not NULL). Ignores all pages of another type than PAGECACHE_LSN_PAGE, because they are not interesting for a checkpoint record. The caller has the intention of doing checkpoints. @@ -3818,17 +3819,18 @@ int reset_key_cache_counters(const char *name, PAGECACHE *key_cache) 0 on success 1 on error */ -my_bool pagecache_collect_changed_blocks_with_LSN(PAGECACHE *pagecache, +my_bool pagecache_collect_changed_blocks_with_lsn(PAGECACHE *pagecache, LEX_STRING *str, LSN *max_lsn) { my_bool error; - ulong stored_LRD_size= 0; + ulong stored_list_size= 0; uint file_hash; char *ptr; DBUG_ENTER("pagecache_collect_changed_blocks_with_LSN"); *max_lsn= 0; + DBUG_ASSERT(NULL == str->str); /* We lock the entire cache but will be quick, just reading/writing a few MBs of memory at most. @@ -3879,17 +3881,17 @@ my_bool pagecache_collect_changed_blocks_with_LSN(PAGECACHE *pagecache, DBUG_ASSERT(0); goto err; } - stored_LRD_size++; + stored_list_size++; } } - str->length= 8+(4+4+8)*stored_LRD_size; + str->length= 8+(4+4+8)*stored_list_size; if (NULL == (str->str= my_malloc(str->length, MYF(MY_WME)))) goto err; ptr= str->str; - int8store(ptr, stored_LRD_size); + int8store(ptr, stored_list_size); ptr+= 8; - if (0 == stored_LRD_size) + if (0 == stored_list_size) goto end; for (file_hash= 0; file_hash < PAGECACHE_CHANGED_BLOCKS_HASH; file_hash++) { diff --git a/sql/mysqld.cc b/sql/mysqld.cc index 8b54efb1199..22c1b6937fb 100644 --- a/sql/mysqld.cc +++ b/sql/mysqld.cc @@ -8187,11 +8187,13 @@ void refresh_status(THD *thd) #undef have_innodb #undef have_ndbcluster #undef have_csv_db +#undef have_maria_db SHOW_COMP_OPTION have_innodb= SHOW_OPTION_NO; SHOW_COMP_OPTION have_ndbcluster= SHOW_OPTION_NO; SHOW_COMP_OPTION have_csv_db= SHOW_OPTION_NO; SHOW_COMP_OPTION have_partition_db= SHOW_OPTION_NO; +SHOW_COMP_OPTION have_maria_db= SHOW_OPTION_NO; #ifndef WITH_INNOBASE_STORAGE_ENGINE uint innobase_flush_log_at_trx_commit; diff --git a/sql/set_var.cc b/sql/set_var.cc index c72fe6fd6fe..bd68863ff97 100644 --- a/sql/set_var.cc +++ b/sql/set_var.cc @@ -674,6 +674,7 @@ sys_var_have_variable sys_have_csv_db("have_csv", &have_csv_db); sys_var_have_variable sys_have_dlopen("have_dynamic_loading", &have_dlopen); sys_var_have_variable sys_have_geometry("have_geometry", &have_geometry); sys_var_have_variable sys_have_innodb("have_innodb", &have_innodb); +sys_var_have_variable sys_have_maria_db("have_maria", &have_maria_db); sys_var_have_variable sys_have_ndbcluster("have_ndbcluster", &have_ndbcluster); sys_var_have_variable sys_have_openssl("have_openssl", &have_openssl); sys_var_have_variable sys_have_partition_db("have_partitioning", @@ -800,6 +801,7 @@ SHOW_VAR init_vars[]= { {sys_have_dlopen.name, (char*) &have_dlopen, SHOW_HAVE}, {sys_have_geometry.name, (char*) &have_geometry, SHOW_HAVE}, {sys_have_innodb.name, (char*) &have_innodb, SHOW_HAVE}, + {sys_have_maria_db.name, (char*) &have_maria_db, SHOW_HAVE}, {sys_have_ndbcluster.name, (char*) &have_ndbcluster, SHOW_HAVE}, {sys_have_openssl.name, (char*) &have_openssl, SHOW_HAVE}, {sys_have_partition_db.name,(char*) &have_partition_db, SHOW_HAVE}, diff --git a/storage/maria/ha_maria.cc b/storage/maria/ha_maria.cc index 56f33693528..13f468b41aa 100644 --- a/storage/maria/ha_maria.cc +++ b/storage/maria/ha_maria.cc @@ -635,7 +635,7 @@ int ha_maria::repair(THD * thd, HA_CHECK_OPT *check_opt) { param.testflag &= ~T_RETRY_WITHOUT_QUICK; sql_print_information("Retrying repair of: '%s' without quick", - table->s->path); + table->s->path.str); continue; } param.testflag &= ~T_QUICK; @@ -643,7 +643,7 @@ int ha_maria::repair(THD * thd, HA_CHECK_OPT *check_opt) { param.testflag= (param.testflag & ~T_REP_BY_SORT) | T_REP; sql_print_information("Retrying repair of: '%s' with keycache", - table->s->path); + table->s->path.str); continue; } break; @@ -654,7 +654,8 @@ int ha_maria::repair(THD * thd, HA_CHECK_OPT *check_opt) char llbuff[22], llbuff2[22]; sql_print_information("Found %s of %s rows when repairing '%s'", llstr(file->state->records, llbuff), - llstr(start_records, llbuff2), table->s->path); + llstr(start_records, llbuff2), + table->s->path.str); } return error; } @@ -1183,7 +1184,7 @@ bool ha_maria::check_and_repair(THD *thd) // Don't use quick if deleted rows if (!file->state->del && (maria_recover_options & HA_RECOVER_QUICK)) check_opt.flags |= T_QUICK; - sql_print_warning("Checking table: '%s'", table->s->path); + sql_print_warning("Checking table: '%s'", table->s->path.str); old_query= thd->query; old_query_length= thd->query_length; @@ -1194,7 +1195,7 @@ bool ha_maria::check_and_repair(THD *thd) if ((marked_crashed= maria_is_crashed(file)) || check(thd, &check_opt)) { - sql_print_warning("Recovering table: '%s'", table->s->path); + sql_print_warning("Recovering table: '%s'", table->s->path.str); check_opt.flags= ((maria_recover_options & HA_RECOVER_BACKUP ? T_BACKUP_DATA : 0) | (marked_crashed ? 0 : T_QUICK) | @@ -1506,6 +1507,7 @@ int ha_maria::create(const char *name, register TABLE *table_arg, bool found_real_auto_increment= 0; enum ha_base_keytype type; char buff[FN_REFLEN]; + byte *record; KEY *pos; MARIA_KEYDEF *keydef; MARIA_COLUMNDEF *recinfo, *recinfo_pos; @@ -1608,6 +1610,7 @@ int ha_maria::create(const char *name, register TABLE *table_arg, found_real_auto_increment= share->next_number_key_offset == 0; } + record= table_arg->record[0]; recpos= 0; recinfo_pos= recinfo; while (recpos < (uint) share->reclength) @@ -1618,7 +1621,8 @@ int ha_maria::create(const char *name, register TABLE *table_arg, for (field= table_arg->field; *field; field++) { - if ((fieldpos= (*field)->offset()) >= recpos && fieldpos <= minpos) + if ((fieldpos=(*field)->offset(record)) >= recpos && + fieldpos <= minpos) { /* skip null fields */ if (!(temp_length= (*field)->pack_length_in_rec())) @@ -1633,7 +1637,7 @@ int ha_maria::create(const char *name, register TABLE *table_arg, } } DBUG_PRINT("loop", ("found: 0x%lx recpos: %d minpos: %d length: %d", - found, recpos, minpos, length)); + (long) found, recpos, minpos, length)); if (recpos != minpos) { // Reserved space (Null bits?) bzero((char*) recinfo_pos, sizeof(*recinfo_pos)); diff --git a/storage/maria/ha_maria.h b/storage/maria/ha_maria.h index 52f289a7428..1f243f9ec59 100644 --- a/storage/maria/ha_maria.h +++ b/storage/maria/ha_maria.h @@ -36,7 +36,7 @@ extern ulong maria_recover_options; class ha_maria :public handler { MARIA_HA *file; - ulong int_table_flags; + ulonglong int_table_flags; char *data_file_name, *index_file_name; bool can_enable_indexes; int repair(THD * thd, HA_CHECK ¶m, bool optimize); diff --git a/storage/maria/ma_close.c b/storage/maria/ma_close.c index 73764cf444a..5d7b24cc314 100644 --- a/storage/maria/ma_close.c +++ b/storage/maria/ma_close.c @@ -28,8 +28,9 @@ int maria_close(register MARIA_HA *info) int error=0,flag; MARIA_SHARE *share=info->s; DBUG_ENTER("maria_close"); - DBUG_PRINT("enter",("base: %lx reopen: %u locks: %u", - info,(uint) share->reopen, (uint) share->tot_locks)); + DBUG_PRINT("enter",("base: 0x%lx reopen: %u locks: %u", + (long) info, (uint) share->reopen, + (uint) share->tot_locks)); pthread_mutex_lock(&THR_LOCK_maria); if (info->lock_type == F_EXTRA_LCK) diff --git a/storage/maria/ma_create.c b/storage/maria/ma_create.c index 76942e3d5e8..d99c4fcc26b 100644 --- a/storage/maria/ma_create.c +++ b/storage/maria/ma_create.c @@ -441,9 +441,9 @@ int maria_create(const char *name,uint keys,MARIA_KEYDEF *keydefs, block_length= max(block_length, MARIA_MIN_KEY_BLOCK_LENGTH); block_length= min(block_length, MARIA_MAX_KEY_BLOCK_LENGTH); - keydef->block_length= MARIA_BLOCK_SIZE(length-real_length_diff, - pointer,MARIA_MAX_KEYPTR_SIZE, - block_length); + keydef->block_length= (uint16) MARIA_BLOCK_SIZE(length-real_length_diff, + pointer,MARIA_MAX_KEYPTR_SIZE, + block_length); if (keydef->block_length > MARIA_MAX_KEY_BLOCK_LENGTH || length >= HA_MAX_KEY_BUFF) { diff --git a/storage/maria/ma_delete.c b/storage/maria/ma_delete.c index 8849c89e30c..f0295a3c413 100644 --- a/storage/maria/ma_delete.c +++ b/storage/maria/ma_delete.c @@ -163,7 +163,7 @@ static int _ma_ck_real_delete(register MARIA_HA *info, MARIA_KEYDEF *keyinfo, DBUG_PRINT("error",("Couldn't allocate memory")); DBUG_RETURN(my_errno=ENOMEM); } - DBUG_PRINT("info",("root_page: %ld",old_root)); + DBUG_PRINT("info",("root_page: %ld", (long) old_root)); if (!_ma_fetch_keypage(info,keyinfo,old_root,DFLT_INIT_HITS,root_buff,0)) { error= -1; @@ -406,7 +406,7 @@ static int del(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, uchar *k MARIA_SHARE *share=info->s; MARIA_KEY_PARAM s_temp; DBUG_ENTER("del"); - DBUG_PRINT("enter",("leaf_page: %ld keypos: 0x%lx", leaf_page, + DBUG_PRINT("enter",("leaf_page: %ld keypos: 0x%lx", (long) leaf_page, (ulong) keypos)); DBUG_DUMP("leaf_buff",(byte*) leaf_buff,maria_getint(leaf_buff)); @@ -593,7 +593,8 @@ static int underflow(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, else { /* Page is full */ endpos=anc_buff+anc_length; - DBUG_PRINT("test",("anc_buff: %lx endpos: %lx",anc_buff,endpos)); + DBUG_PRINT("test",("anc_buff: 0x%lx endpos: 0x%lx", + (long) anc_buff, (long) endpos)); if (keypos != anc_buff+2+key_reflength && !_ma_get_last_key(info,keyinfo,anc_buff,anc_key,keypos,&length)) goto err; @@ -771,7 +772,7 @@ static uint remove_key(MARIA_KEYDEF *keyinfo, uint nod_flag, int s_length; uchar *start; DBUG_ENTER("remove_key"); - DBUG_PRINT("enter",("keypos: %lx page_end: %lx",keypos,page_end)); + DBUG_PRINT("enter",("keypos: 0x%lx page_end: 0x%lx",(long) keypos, (long) page_end)); start=keypos; if (!(keyinfo->flag & diff --git a/storage/maria/ma_dynrec.c b/storage/maria/ma_dynrec.c index 253a538861a..e0f87addb43 100644 --- a/storage/maria/ma_dynrec.c +++ b/storage/maria/ma_dynrec.c @@ -1240,8 +1240,8 @@ ulong _ma_rec_unpack(register MARIA_HA *info, register byte *to, byte *from, err: my_errno= HA_ERR_WRONG_IN_RECORD; - DBUG_PRINT("error",("to_end: %lx -> %lx from_end: %lx -> %lx", - to,to_end,from,from_end)); + DBUG_PRINT("error",("to_end: 0x%lx -> 0x%lx from_end: 0x%lx -> 0x%lx", + (long) to, (long) to_end, (long) from, (long) from_end)); DBUG_DUMP("from",(byte*) info->rec_buff,info->s->base.min_pack_length); DBUG_RETURN(MY_FILE_ERROR); } /* _ma_rec_unpack */ diff --git a/storage/maria/ma_ft_boolean_search.c b/storage/maria/ma_ft_boolean_search.c index 83901cb5e47..24b51ef469c 100644 --- a/storage/maria/ma_ft_boolean_search.c +++ b/storage/maria/ma_ft_boolean_search.c @@ -333,7 +333,6 @@ static int _ft2_search(FTB *ftb, FTB_WORD *ftbw, my_bool init_search) MARIA_HA *info=ftb->info; uint off, extra=HA_FT_WLEN+info->s->base.rec_reflength; byte *lastkey_buf=ftbw->word+ftbw->off; - LINT_INIT(off); if (ftbw->flags & FTB_FLAG_TRUNC) lastkey_buf+=ftbw->len; diff --git a/storage/maria/ma_key.c b/storage/maria/ma_key.c index ecd51f5dc92..78465ca729f 100644 --- a/storage/maria/ma_key.c +++ b/storage/maria/ma_key.c @@ -52,7 +52,7 @@ static int _ma_put_key_in_record(MARIA_HA *info,uint keynr,byte *record); uint _ma_make_key(register MARIA_HA *info, uint keynr, uchar *key, const byte *record, my_off_t filepos) { - byte *pos,*end; + byte *pos; uchar *start; reg1 HA_KEYSEG *keyseg; my_bool is_ft= info->s->keyinfo[keynr].flag & HA_FULLTEXT; @@ -107,18 +107,17 @@ uint _ma_make_key(register MARIA_HA *info, uint keynr, uchar *key, } if (keyseg->flag & HA_SPACE_PACK) { - end= pos + length; if (type != HA_KEYTYPE_NUM) { - while (end > pos && end[-1] == ' ') - end--; + length= cs->cset->lengthsp(cs, pos, length); } else { + byte *end= pos + length; while (pos < end && pos[0] == ' ') pos++; + length=(uint) (end-pos); } - length=(uint) (end-pos); FIX_LENGTH(cs, pos, length, char_length); store_key_length_inc(key,char_length); memcpy((byte*) key,(byte*) pos,(size_t) char_length); @@ -403,8 +402,10 @@ static int _ma_put_key_in_record(register MARIA_HA *info, uint keynr, pos= record+keyseg->start; if (keyseg->type != (int) HA_KEYTYPE_NUM) { - memcpy(pos,key,(size_t) length); - bfill(pos+length,keyseg->length-length,' '); + memcpy(pos,key,(size_t) length); + keyseg->charset->cset->fill(keyseg->charset, + pos + length, keyseg->length - length, + ' '); } else { diff --git a/storage/maria/ma_keycache.c b/storage/maria/ma_keycache.c index 837b0fbac66..4f150d905ed 100644 --- a/storage/maria/ma_keycache.c +++ b/storage/maria/ma_keycache.c @@ -54,8 +54,8 @@ int maria_assign_to_key_cache(MARIA_HA *info, int error= 0; MARIA_SHARE* share= info->s; DBUG_ENTER("maria_assign_to_key_cache"); - DBUG_PRINT("enter",("old_key_cache_handle: %lx new_key_cache_handle: %lx", - share->key_cache, key_cache)); + DBUG_PRINT("enter",("old_key_cache_handle: 0x%lx new_key_cache_handle: 0x%lx", + (long) share->key_cache, (long) key_cache)); /* Skip operation if we didn't change key cache. This can happen if we diff --git a/storage/maria/ma_open.c b/storage/maria/ma_open.c index 4041c101bde..ab38d2baea1 100644 --- a/storage/maria/ma_open.c +++ b/storage/maria/ma_open.c @@ -340,6 +340,8 @@ MARIA_HA *maria_open(const char *name, int mode, uint open_flags) goto err; } } + else if (pos->type == HA_KEYTYPE_BINARY) + pos->charset= &my_charset_bin; } if (share->keyinfo[i].flag & HA_SPATIAL) { @@ -1259,12 +1261,29 @@ int maria_enable_indexes(MARIA_HA *info) RETURN 0 indexes are not disabled 1 all indexes are disabled - [2 non-unique indexes are disabled - NOT YET IMPLEMENTED] + 2 non-unique indexes are disabled */ int maria_indexes_are_disabled(MARIA_HA *info) { MARIA_SHARE *share= info->s; - return (! maria_is_any_key_active(share->state.key_map) && share->base.keys); + /* + No keys or all are enabled. keys is the number of keys. Left shifted + gives us only one bit set. When decreased by one, gives us all all bits + up to this one set and it gets unset. + */ + if (!share->base.keys || + (maria_is_all_keys_active(share->state.key_map, share->base.keys))) + return 0; + + /* All are disabled */ + if (maria_is_any_key_active(share->state.key_map)) + return 1; + + /* + We have keys. Some enabled, some disabled. + Don't check for any non-unique disabled but return directly 2 + */ + return 2; } diff --git a/storage/maria/ma_page.c b/storage/maria/ma_page.c index 054b8e16468..e5f9f47eaa5 100644 --- a/storage/maria/ma_page.c +++ b/storage/maria/ma_page.c @@ -27,7 +27,7 @@ uchar *_ma_fetch_keypage(register MARIA_HA *info, MARIA_KEYDEF *keyinfo, uchar *tmp; uint page_size; DBUG_ENTER("_ma_fetch_keypage"); - DBUG_PRINT("enter",("page: %ld",page)); + DBUG_PRINT("enter",("page: %ld", (long) page)); tmp=(uchar*) key_cache_read(info->s->key_cache, info->s->kfile, page, level, (byte*) buff, @@ -80,7 +80,7 @@ int _ma_write_keypage(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, my_errno=EINVAL; DBUG_RETURN((-1)); } - DBUG_PRINT("page",("write page at: %lu",(long) page,buff)); + DBUG_PRINT("page",("write page at: %lu",(long) page)); DBUG_DUMP("buff",(byte*) buff,maria_getint(buff)); #endif diff --git a/storage/maria/ma_rsamepos.c b/storage/maria/ma_rsamepos.c index 09861c03c32..b49ef8d294a 100644 --- a/storage/maria/ma_rsamepos.c +++ b/storage/maria/ma_rsamepos.c @@ -33,7 +33,8 @@ int maria_rsame_with_pos(MARIA_HA *info, byte *record, int inx, my_off_t filepos DBUG_ENTER("maria_rsame_with_pos"); DBUG_PRINT("enter",("index: %d filepos: %ld", inx, (long) filepos)); - if (inx < -1 || (inx >= 0 && !maria_is_key_active(info->s->state.key_map, inx))) + if (inx < -1 || + (inx >= 0 && ! maria_is_key_active(info->s->state.key_map, inx))) { DBUG_RETURN(my_errno=HA_ERR_WRONG_INDEX); } diff --git a/storage/maria/ma_statrec.c b/storage/maria/ma_statrec.c index 0aef24f40a9..82e31115ec7 100644 --- a/storage/maria/ma_statrec.c +++ b/storage/maria/ma_statrec.c @@ -251,8 +251,8 @@ int _ma_read_rnd_static_record(MARIA_HA *info, byte *buf, if (filepos >= info->state->data_file_length) { DBUG_PRINT("test",("filepos: %ld (%ld) records: %ld del: %ld", - filepos/share->base.reclength,filepos, - info->state->records, info->state->del)); + (long) filepos/share->base.reclength, (long) filepos, + (long) info->state->records, (long) info->state->del)); fast_ma_writeinfo(info); DBUG_RETURN(my_errno=HA_ERR_END_OF_FILE); } diff --git a/storage/maria/ma_unique.c b/storage/maria/ma_unique.c index bc1aa71966b..e2c7ca3c80c 100644 --- a/storage/maria/ma_unique.c +++ b/storage/maria/ma_unique.c @@ -57,8 +57,8 @@ my_bool _ma_check_unique(MARIA_HA *info, MARIA_UNIQUEDEF *def, byte *record, if (_ma_search_next(info,info->s->keyinfo+def->key, info->lastkey, MARIA_UNIQUE_HASH_LENGTH, SEARCH_BIGGER, info->s->state.key_root[def->key]) || - memcmp((char*) info->lastkey, (char*) key_buff, - MARIA_UNIQUE_HASH_LENGTH)) + bcmp((char*) info->lastkey, (char*) key_buff, + MARIA_UNIQUE_HASH_LENGTH)) { info->page_changed=1; /* Can't optimize read next */ info->lastpos=lastpos; diff --git a/storage/maria/maria_chk.c b/storage/maria/maria_chk.c index 89858ee2d07..6ba5200918e 100644 --- a/storage/maria/maria_chk.c +++ b/storage/maria/maria_chk.c @@ -701,6 +701,7 @@ get_one_option(int optid, { int method; enum_handler_stats_method method_conv; + LINT_INIT(method_conv); maria_stats_method_str= argument; if ((method=find_type(argument, &maria_stats_method_typelib, 2)) <= 0) { diff --git a/storage/maria/maria_pack.c b/storage/maria/maria_pack.c index 5a5aaf037f0..24de45a89bc 100644 --- a/storage/maria/maria_pack.c +++ b/storage/maria/maria_pack.c @@ -1107,18 +1107,18 @@ static int get_statistic(PACK_MRG_INFO *mrg,HUFF_COUNTS *huff_counts) my_off_t total_count; char llbuf[32]; - DBUG_PRINT("info", ("column: %3lu", count - huff_counts + 1)); + DBUG_PRINT("info", ("column: %3u", (uint) (count - huff_counts + 1))); if (verbose >= 2) - VOID(printf("column: %3lu\n", count - huff_counts + 1)); + VOID(printf("column: %3u\n", (uint) (count - huff_counts + 1))); if (count->tree_buff) { - DBUG_PRINT("info", ("number of distinct values: %lu", - (count->tree_pos - count->tree_buff) / - count->field_length)); + DBUG_PRINT("info", ("number of distinct values: %u", + (uint) ((count->tree_pos - count->tree_buff) / + count->field_length))); if (verbose >= 2) - VOID(printf("number of distinct values: %lu\n", - (count->tree_pos - count->tree_buff) / - count->field_length)); + VOID(printf("number of distinct values: %u\n", + (uint) ((count->tree_pos - count->tree_buff) / + count->field_length))); } total_count= 0; for (idx= 0; idx < 256; idx++) @@ -2038,7 +2038,7 @@ static void write_field_info(HUFF_COUNTS *counts, uint fields, uint trees) uint huff_tree_bits; huff_tree_bits=max_bit(trees ? trees-1 : 0); - DBUG_PRINT("info", ("")); + DBUG_PRINT("info", (" ")); DBUG_PRINT("info", ("column types:")); DBUG_PRINT("info", ("FIELD_NORMAL 0")); DBUG_PRINT("info", ("FIELD_SKIP_ENDSPACE 1")); @@ -2050,12 +2050,12 @@ static void write_field_info(HUFF_COUNTS *counts, uint fields, uint trees) DBUG_PRINT("info", ("FIELD_ZERO 7")); DBUG_PRINT("info", ("FIELD_VARCHAR 8")); DBUG_PRINT("info", ("FIELD_CHECK 9")); - DBUG_PRINT("info", ("")); + DBUG_PRINT("info", (" ")); DBUG_PRINT("info", ("pack type as a set of flags:")); DBUG_PRINT("info", ("PACK_TYPE_SELECTED 1")); DBUG_PRINT("info", ("PACK_TYPE_SPACE_FIELDS 2")); DBUG_PRINT("info", ("PACK_TYPE_ZERO_FILL 4")); - DBUG_PRINT("info", ("")); + DBUG_PRINT("info", (" ")); if (verbose >= 2) { VOID(printf("\n")); @@ -2128,7 +2128,7 @@ static my_off_t write_huff_tree(HUFF_TREE *huff_tree, uint trees) return 0; } - DBUG_PRINT("info", ("")); + DBUG_PRINT("info", (" ")); if (verbose >= 2) VOID(printf("\n")); tree_no= 0; @@ -2139,7 +2139,7 @@ static my_off_t write_huff_tree(HUFF_TREE *huff_tree, uint trees) if (huff_tree->tree_number == 0) continue; /* Deleted tree */ tree_no++; - DBUG_PRINT("info", ("")); + DBUG_PRINT("info", (" ")); if (verbose >= 3) VOID(printf("\n")); /* Count the total number of elements (byte codes or column values). */ @@ -2281,8 +2281,8 @@ static my_off_t write_huff_tree(HUFF_TREE *huff_tree, uint trees) if (bits > 8 * sizeof(code)) { VOID(fflush(stdout)); - VOID(fprintf(stderr, "error: Huffman code too long: %u/%lu\n", - bits, 8 * sizeof(code))); + VOID(fprintf(stderr, "error: Huffman code too long: %u/%u\n", + bits, (uint) (8 * sizeof(code)))); errors++; break; } @@ -2331,7 +2331,7 @@ static my_off_t write_huff_tree(HUFF_TREE *huff_tree, uint trees) } flush_bits(); } - DBUG_PRINT("info", ("")); + DBUG_PRINT("info", (" ")); if (verbose >= 2) VOID(printf("\n")); my_afree((gptr) packed_tree); @@ -2509,7 +2509,7 @@ static int compress_isam_file(PACK_MRG_INFO *mrg, HUFF_COUNTS *huff_counts) end_pos-=count->max_zero_fill; field_length-=count->max_zero_fill; - switch(count->field_type) { + switch (count->field_type) { case FIELD_SKIP_ZERO: if (!memcmp((byte*) start_pos,zero_string,field_length)) { @@ -2925,6 +2925,8 @@ static void flush_bits(void) bits-= 8; *file_buffer.pos++= (uchar) (bit_buffer >> bits); } + if (file_buffer.pos >= file_buffer.end) + VOID(flush_buffer(~ (ulong) 0)); file_buffer.bits= BITS_SAVED; file_buffer.bitbucket= 0; } diff --git a/storage/myisam/myisampack.c b/storage/myisam/myisampack.c index 623d65f7ce3..2753ed5f7b2 100644 --- a/storage/myisam/myisampack.c +++ b/storage/myisam/myisampack.c @@ -1110,11 +1110,11 @@ static int get_statistic(PACK_MRG_INFO *mrg,HUFF_COUNTS *huff_counts) VOID(printf("column: %3u\n", (uint) (count - huff_counts + 1))); if (count->tree_buff) { - DBUG_PRINT("info", ("number of distinct values: %lu", + DBUG_PRINT("info", ("number of distinct values: %u", (uint) ((count->tree_pos - count->tree_buff) / count->field_length))); if (verbose >= 2) - VOID(printf("number of distinct values: %lu\n", + VOID(printf("number of distinct values: %u\n", (uint) ((count->tree_pos - count->tree_buff) / count->field_length))); } -- cgit v1.2.1 From 649b3b46055594e77cadabe5768b762b8aa486da Mon Sep 17 00:00:00 2001 From: unknown Date: Wed, 20 Dec 2006 19:01:07 +0100 Subject: WL#3071 - Maria checkpoint: a function to store information about transactions into buffers, is added to the transaction manager, and called by the Checkpoint module. storage/maria/ma_checkpoint.c: "collecting info about transactions" moves to trnman.c storage/maria/trnman.c: a function to store information about the active transactions list and committed transactions list, into buffers, for use by the Checkpoint module. This function needs to know how many trns there are in the committed list, so we introduce a counter, trnman_committed_transactions. m_string.h is needed for LEX_STRING. storage/maria/trnman.h: A function to store information about the active transactions list and committed transactions list, into buffers, for use by the Checkpoint module. storage/maria/unittest/trnman-t.c: trnman.h needs LEX_STRING so m_string.h --- storage/maria/ma_checkpoint.c | 68 +++-------------------- storage/maria/trnman.c | 112 ++++++++++++++++++++++++++++++++++++-- storage/maria/trnman.h | 1 + storage/maria/unittest/trnman-t.c | 1 + 4 files changed, 118 insertions(+), 64 deletions(-) diff --git a/storage/maria/ma_checkpoint.c b/storage/maria/ma_checkpoint.c index a1d094d7da1..4294acb5627 100644 --- a/storage/maria/ma_checkpoint.c +++ b/storage/maria/ma_checkpoint.c @@ -176,7 +176,8 @@ my_bool execute_checkpoint_indirect() /* checkpoint record data: */ LSN checkpoint_start_lsn; char checkpoint_start_lsn_char[8]; - LEX_STRING strings[5]={ {&checkpoint_start_lsn_str, 8}, {0,0}, {0,0}, {0,0}, {0,0} }; + LEX_STRING strings[6]= + {checkpoint_start_lsn_char, 8}, {0,0}, {0,0}, {0,0}, {0,0}, {0,0} }; char *ptr; LSN checkpoint_lsn; LSN candidate_max_rec_lsn_at_last_checkpoint; @@ -203,69 +204,16 @@ my_bool execute_checkpoint_indirect() goto err; /* STEP 3: fetch information about transactions */ - /* note: this piece will move into trnman.c */ - /* - Transactions are in the "active list" (protected by a mutex) and in a - lock-free hash of "committed" (insertion protected by the same mutex, - deletion lock-free). - */ - { - TRN *trn; - ulong stored_trn_size= 0; - /* First, the active transactions */ - pthread_mutex_lock(LOCK_trn_list); - string2.length= 8+(7+2+8+8+8)*trnman_active_transactions; - if (NULL == (string2.str= my_malloc(string2.length))) - goto err; - ptr= string2.str; - ptr+= 8; - for (trn= active_list_min.next; trn != &active_list_max; trn= trn->next) - { - /* we would latch trn.rwlock if it existed */ - if (0 == trn->short_trid) /* trn is not inited, skip */ - continue; - /* state is not needed for now (only when we have prepared trx) */ - /* int7store does not exist but mi_int7store does */ - int7store(ptr, trn->trid); - ptr+= 7; - int2store(ptr, trn->short_trid); - ptr+= 2; - int8store(ptr, trn->undo_lsn); /* is an LSN 7 or 8 bytes really? */ - ptr+= 8; - int8store(ptr, trn->undo_purge_lsn); - ptr+= 8; - int8store(ptr, read_non_atomic(&trn->first_undo_lsn)); - ptr+= 8; - /* possibly unlatch el.rwlock */ - stored_trn_size++; - } - pthread_mutex_unlock(LOCK_trn_list); - /* - Now the committed ones. - We need a function which scans the hash's list of elements in a - lock-free manner (a bit like lfind(), starting from bucket 0), and for - each node (committed transaction) stores the transaction's - information (trid, undo_purge_lsn, first_undo_lsn) into a buffer. - This big buffer is malloc'ed at the start, so the number of elements (or - an upper bound of it) found in the hash needs to be known in advance - (one solution is to keep LOCK_trn_list locked, ensuring that nodes are - only deleted). - */ - /* - TODO: if we see there exists no transaction (active and committed) we can - tell the lock-free structures to do some freeing (my_free()). - */ - int8store(string1.str, stored_trn_size); - string2.length= 8+(7+2+8+8+8)*stored_trn_size; - } + if (trnman_collect_transactions(&strings[2], &strings[3])) + goto err; /* STEP 4: fetch information about table files */ { /* This global mutex is in fact THR_LOCK_maria (see ma_open()) */ lock(global_share_list_mutex); - string3.length= 8+(8+8)*share_list->count; - if (NULL == (string3.str= my_malloc(string3.length))) + strings[4].length= 8+(8+8)*share_list->count; + if (NULL == (strings[4].str= my_malloc(strings[4].length))) goto err; ptr= string3.str; /* possibly latch each MARIA_SHARE, one by one, like this: */ @@ -327,8 +275,8 @@ err: end: - for (i= 1; i<5; i++) - my_free(strings[i], MYF(MY_ALLOW_ZERO_PTR)); + for (i= 1; i<6; i++) + my_free(strings[i].str, MYF(MY_ALLOW_ZERO_PTR)); /* this portion cannot be done as a hook in write_log_record() for the diff --git a/storage/maria/trnman.c b/storage/maria/trnman.c index 4399f0e1208..f289f6fcc5b 100644 --- a/storage/maria/trnman.c +++ b/storage/maria/trnman.c @@ -18,10 +18,16 @@ #include #include #include +#include #include "trnman.h" -/* status variables */ -uint trnman_active_transactions, trnman_allocated_transactions; +/* + status variables: + how many trns in the active list currently, + in the committed list currently, allocated since startup. +*/ +uint trnman_active_transactions, trnman_committed_transactions, + trnman_allocated_transactions; /* list of active transactions in the trid order */ static TRN active_list_min, active_list_max; @@ -100,6 +106,7 @@ int trnman_init() committed_list_min.next= &committed_list_max; trnman_active_transactions= 0; + trnman_committed_transactions= 0; trnman_allocated_transactions= 0; pool= 0; @@ -129,6 +136,7 @@ void trnman_destroy() { DBUG_ASSERT(trid_to_committed_trn.count == 0); DBUG_ASSERT(trnman_active_transactions == 0); + DBUG_ASSERT(trnman_committed_transactions == 0); DBUG_ASSERT(active_list_max.prev == &active_list_min); DBUG_ASSERT(active_list_min.next == &active_list_max); DBUG_ASSERT(committed_list_max.prev == &committed_list_min); @@ -285,11 +293,14 @@ void trnman_end_trn(TRN *trn, my_bool commit) */ if (trn->prev == &active_list_min) { + uint free_me_count; TRN *t; - for (t= committed_list_min.next; + for (t= committed_list_min.next, free_me_count= 0; t->commit_trid < active_list_min.next->min_read_from; - t= t->next) /* no-op */; + t= t->next, free_me_count++) /* no-op */; + DBUG_ASSERT((t != committed_list_min.next && free_me_count > 0) || + (t == committed_list_min.next && free_me_count == 0)); /* found transactions committed before the oldest active one */ if (t != committed_list_min.next) { @@ -297,6 +308,7 @@ void trnman_end_trn(TRN *trn, my_bool commit) committed_list_min.next= t; t->prev->next= 0; t->prev= &committed_list_min; + trnman_committed_transactions-= free_me_count; } } @@ -312,6 +324,7 @@ void trnman_end_trn(TRN *trn, my_bool commit) trn->next= &committed_list_max; trn->prev= committed_list_max.prev; committed_list_max.prev= trn->prev->next= trn; + trnman_committed_transactions++; res= lf_hash_insert(&trid_to_committed_trn, pins, &trn); DBUG_ASSERT(res == 0); @@ -413,3 +426,94 @@ my_bool trnman_can_read_from(TRN *trn, TrID trid) return can; } + +/* + Allocates two buffers and stores in them some information about transactions + of the active list (into the first buffer) and of the committed list (into + the second buffer). + + SYNOPSIS + trnman_collect_transactions() + str_act (OUT) pointer to a LEX_STRING where the allocated buffer, and + its size, will be put + str_com (OUT) pointer to a LEX_STRING where the allocated buffer, and + its size, will be put + + + DESCRIPTION + Does the allocation because the caller cannot know the size itself. + Memory freeing is to be done by the caller (if the "str" member of the + LEX_STRING is not NULL). + The caller has the intention of doing checkpoints. + + RETURN + 0 on success + 1 on error +*/ +my_bool trnman_collect_transactions(LEX_STRING *str_act, LEX_STRING *str_com) +{ + my_bool error; + TRN *trn; + char *ptr; + DBUG_ENTER("trnman_collect_transactions"); + + DBUG_ASSERT((NULL == str_act->str) && (NULL == str_com->str)); + + pthread_mutex_lock(&LOCK_trn_list); + str_act->length= 8+(6+2+7+7+7)*trnman_active_transactions; + str_com->length= 8+(6+7+7)*trnman_committed_transactions; + if ((NULL == (str_act->str= my_malloc(str_act->length, MYF(MY_WME)))) || + (NULL == (str_com->str= my_malloc(str_com->length, MYF(MY_WME))))) + goto err; + /* First, the active transactions */ + ptr= str_act->str; + int8store(ptr, (ulonglong)trnman_active_transactions); + ptr+= 8; + for (trn= active_list_min.next; trn != &active_list_max; trn= trn->next) + { + /* + trns with a short trid of 0 are not initialized; Recovery will recognize + this and ignore them. + State is not needed for now (only when we supported prepared trns). + For LSNs, Sanja will soon push lsn7store. + */ + int6store(ptr, trn->trid); + ptr+= 6; + int2store(ptr, trn->short_id); + ptr+= 2; + /* needed for rollback */ + /* lsn7store(ptr, trn->undo_lsn); */ + ptr+= 7; + /* needed for purge */ + /* lsn7store(ptr, trn->undo_purge_lsn); */ + ptr+= 7; + /* needed for low-water mark calculation */ + /* lsn7store(ptr, read_non_atomic(&trn->first_undo_lsn)); */ + ptr+= 7; + } + /* do the same for committed ones */ + ptr= str_com->str; + int8store(ptr, (ulonglong)trnman_committed_transactions); + ptr+= 8; + for (trn= committed_list_min.next; trn != &committed_list_max; + trn= trn->next) + { + int6store(ptr, trn->trid); + ptr+= 6; + /* mi_int7store(ptr, trn->undo_purge_lsn); */ + ptr+= 7; + /* mi_int7store(ptr, read_non_atomic(&trn->first_undo_lsn)); */ + ptr+= 7; + } + /* + TODO: if we see there exists no transaction (active and committed) we can + tell the lock-free structures to do some freeing (my_free()). + */ + error= 0; + goto end; +err: + error= 1; +end: + pthread_mutex_unlock(&LOCK_trn_list); + DBUG_RETURN(error); +} diff --git a/storage/maria/trnman.h b/storage/maria/trnman.h index 409e354d423..267e3cabd7a 100644 --- a/storage/maria/trnman.h +++ b/storage/maria/trnman.h @@ -51,6 +51,7 @@ void trnman_end_trn(TRN *trn, my_bool commit); #define trnman_abort_trn(T) trnman_end_trn(T, FALSE) void trnman_free_trn(TRN *trn); my_bool trnman_can_read_from(TRN *trn, TrID trid); +my_bool trnman_collect_transactions(LEX_STRING *str_act, LEX_STRING *str_com); #endif diff --git a/storage/maria/unittest/trnman-t.c b/storage/maria/unittest/trnman-t.c index 7da8202b881..3c70d10c440 100644 --- a/storage/maria/unittest/trnman-t.c +++ b/storage/maria/unittest/trnman-t.c @@ -20,6 +20,7 @@ #include #include #include +#include #include "../trnman.h" pthread_mutex_t rt_mutex; -- cgit v1.2.1 From ba7a2a9ab788748b2dffa8526fd4aa1d9396a26e Mon Sep 17 00:00:00 2001 From: unknown Date: Wed, 20 Dec 2006 22:30:00 +0100 Subject: Maria - fixes to the pagecache unit tests (which started to print "x threads didn't exit" today after my merge); the main() thread was calling my_init() and my_thread_global_init() which is redudant; it was not calling my_end() though it's needed. unittest/mysys/mf_pagecache_consist.c: program already calls my_init(), which calls my_thread_global_init(), so no need for my_thread_global_init() (double call caused the main() thread to wait for itself thus timing out and printing "x threads didn't exit". my_thread_global_end() is not enough, as we call my_init() we must call my_end() (which calls my_thread_global_end()). unittest/mysys/mf_pagecache_single.c: program already calls my_init(), which calls my_thread_global_init(), so no need for my_thread_global_init() (double call caused the main() thread to wait for itself thus timing out and printing "x threads didn't exit". my_thread_global_end() is not enough, as we call my_init() we must call my_end() (which calls my_thread_global_end()). --- unittest/mysys/mf_pagecache_consist.c | 5 +---- unittest/mysys/mf_pagecache_single.c | 4 +--- 2 files changed, 2 insertions(+), 7 deletions(-) diff --git a/unittest/mysys/mf_pagecache_consist.c b/unittest/mysys/mf_pagecache_consist.c index 95c2135edad..d31c19cc746 100755 --- a/unittest/mysys/mf_pagecache_consist.c +++ b/unittest/mysys/mf_pagecache_consist.c @@ -336,9 +336,6 @@ int main(int argc, char **argv __attribute__((unused))) VOID(thr_setconcurrency(2)); #endif - my_thread_global_init(); - - if ((pagen= init_pagecache(&pagecache, PCACHE_SIZE, 0, 0, PAGE_SIZE, 0)) == 0) { @@ -427,7 +424,7 @@ int main(int argc, char **argv __attribute__((unused))) exit(1); } /*my_delete(file1_name, MYF(0));*/ - my_thread_global_end(); + my_end(0); DBUG_PRINT("info", ("file1 (%d) closed", file1.file)); diff --git a/unittest/mysys/mf_pagecache_single.c b/unittest/mysys/mf_pagecache_single.c index 9c467343156..4fadf2a554f 100644 --- a/unittest/mysys/mf_pagecache_single.c +++ b/unittest/mysys/mf_pagecache_single.c @@ -515,8 +515,6 @@ int main(int argc, char **argv __attribute__((unused))) VOID(thr_setconcurrency(2)); #endif - my_thread_global_init(); - plan(12); if ((pagen= init_pagecache(&pagecache, PCACHE_SIZE, 0, 0, @@ -569,7 +567,7 @@ int main(int argc, char **argv __attribute__((unused))) exit(1); } /*my_delete(file1_name, MYF(0));*/ - my_thread_global_end(); + my_end(0); DBUG_PRINT("info", ("file1 (%d) closed", file1.file)); -- cgit v1.2.1 From b635df555adec7ebdc4cd40e102d51006c802639 Mon Sep 17 00:00:00 2001 From: unknown Date: Wed, 3 Jan 2007 12:41:51 +0100 Subject: very minor comments and merges from MyISAM into Maria. storage/maria/ma_checkpoint.c: comments storage/maria/ma_close.c: comments storage/maria/ma_write.c: merge from myisam storage/maria/maria_def.h: typo storage/myisam/mi_delete.c: unneeded {}, making it identical to Maria --- storage/maria/ma_checkpoint.c | 44 ++++++++++++++++++++++++++++++++++--------- storage/maria/ma_close.c | 8 +++++--- storage/maria/ma_write.c | 18 ++++++++++-------- storage/maria/maria_def.h | 2 +- storage/myisam/mi_delete.c | 2 -- 5 files changed, 51 insertions(+), 23 deletions(-) diff --git a/storage/maria/ma_checkpoint.c b/storage/maria/ma_checkpoint.c index 4294acb5627..02d887f758a 100644 --- a/storage/maria/ma_checkpoint.c +++ b/storage/maria/ma_checkpoint.c @@ -216,14 +216,27 @@ my_bool execute_checkpoint_indirect() if (NULL == (strings[4].str= my_malloc(strings[4].length))) goto err; ptr= string3.str; - /* possibly latch each MARIA_SHARE, one by one, like this: */ - pthread_mutex_lock(&share->intern_lock); /* - We'll copy the file id (a bit like share->kfile), the file name - (like share->unique_file_name[_length]). + Note that maria_open_list is a list of MARIA_HA*, while we would prefer + a list of MARIA_SHARE* here (we are interested in the short id, + unique file name, members of MARIA_SHARE*, and in file descriptors, + which will in the end be in MARIA_SHARE*). */ - make_copy_of_global_share_list_to_array; - pthread_mutex_unlock(&share->intern_lock); + for (iterate on the maria_open_list) + { + /* latch each MARIA_SHARE, one by one, like this: */ + pthread_mutex_lock(&share->intern_lock); + /* + TODO: + we need to prevent the share from going away while we later flush and + force it without holding THR_LOCK_maria. For example if the share is + free()d by maria_close() we'll have a problem. Or if the share's file + descriptor is closed by maria_close() we will not be able to my_sync() + it. + */ + pthread_mutex_unlock(&share->intern_lock); + store the share pointer into a private array; + } unlock(global_share_list_mutex); /* work on copy */ @@ -231,15 +244,15 @@ my_bool execute_checkpoint_indirect() ptr+= 8; for (el in array) { - int8store(ptr, array[...].file_id); + int8store(ptr, array[...].short_id); ptr+= 8; - memcpy(ptr, array[...].file_name, ...); + memcpy(ptr, array[...].unique_file_name[_length], ...); ptr+= ...; + /* maybe we need to lock share->intern_lock here */ /* these two are long ops (involving disk I/O) that's why we copied the list, to not keep the list locked for long: */ - /* TODO: what if the table pointer is gone/reused now? */ flush_bitmap_pages(el); /* TODO: and also autoinc counter, logical file end, free page list */ @@ -267,6 +280,19 @@ my_bool execute_checkpoint_indirect() if (0 != control_file_write_and_force(checkpoint_lsn, NULL)) goto err; + /* + Note that we should not alter memory structures until we have successfully + written the checkpoint record and control file. + Btw, a log write failure is serious: + - if we know how many bytes we managed to write, we should try to write + more, keeping the log's mutex (MY_FULL_IO) + - if we don't know, this log record is corrupted and we have no way to + "de-corrupt" it, so it will stay corrupted, and as the log is sequential, + any log record written after it will not be reachable (for example if we + would write UNDOs and crash, we would not be able to read the log and so + not be able to rollback), so we should stop the engine now (holding the + log's mutex) and do a recovery. + */ goto end; err: diff --git a/storage/maria/ma_close.c b/storage/maria/ma_close.c index 5d7b24cc314..e1a4bb08301 100644 --- a/storage/maria/ma_close.c +++ b/storage/maria/ma_close.c @@ -60,9 +60,11 @@ int maria_close(register MARIA_HA *info) flag= !--share->reopen; /* RECOVERYTODO: - Below we are going to make the table unknown to future checkpoints, so it - needs to have fsync'ed itself entirely (bitmap, pages, etc) at this - point. + If "flag" is TRUE, in the line below we are going to make the table + unknown to future checkpoints, so it needs to have fsync'ed itself + entirely (bitmap, pages, etc) at this point. + The flushing is currently done a few lines further (which is ok, as we + still hold THR_LOCK_maria), but syncing is missing. */ maria_open_list=list_delete(maria_open_list,&info->open_list); pthread_mutex_unlock(&share->intern_lock); diff --git a/storage/maria/ma_write.c b/storage/maria/ma_write.c index c56a26fefff..55c0bf27259 100644 --- a/storage/maria/ma_write.c +++ b/storage/maria/ma_write.c @@ -345,7 +345,7 @@ static int w_search(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, my_bool was_last_key; my_off_t next_page, dupp_key_pos; DBUG_ENTER("w_search"); - DBUG_PRINT("enter",("page: %ld",page)); + DBUG_PRINT("enter",("page: %ld", (long) page)); search_key_length= (comp_flag & SEARCH_FIND) ? key_length : USE_WHOLE_KEY; if (!(temp_buff= (uchar*) my_alloca((uint) keyinfo->block_length+ @@ -468,7 +468,7 @@ int _ma_insert(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, uchar *endpos, *prev_key; MARIA_KEY_PARAM s_temp; DBUG_ENTER("_ma_insert"); - DBUG_PRINT("enter",("key_pos: %lx",key_pos)); + DBUG_PRINT("enter",("key_pos: %lx", (long) key_pos)); DBUG_EXECUTE("key", _ma_print_key(DBUG_FILE,keyinfo->seg,key, USE_WHOLE_KEY);); @@ -490,8 +490,8 @@ int _ma_insert(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, { DBUG_PRINT("test",("t_length: %d ref_len: %d", t_length,s_temp.ref_length)); - DBUG_PRINT("test",("n_ref_len: %d n_length: %d key_pos: %lx", - s_temp.n_ref_length,s_temp.n_length,s_temp.key)); + DBUG_PRINT("test",("n_ref_len: %d n_length: %d key_pos: 0x%lx", + s_temp.n_ref_length,s_temp.n_length, (long) s_temp.key)); } #endif if (t_length > 0) @@ -684,7 +684,8 @@ uchar *_ma_find_half_pos(uint nod_flag, MARIA_KEYDEF *keyinfo, uchar *page, } while (page < end); *return_key_length=length; *after_key=page; - DBUG_PRINT("exit",("returns: %lx page: %lx half: %lx",lastpos,page,end)); + DBUG_PRINT("exit",("returns: 0x%lx page: 0x%lx half: 0x%lx", + (long) lastpos, (long) page, (long) end)); DBUG_RETURN(lastpos); } /* _ma_find_half_pos */ @@ -739,7 +740,8 @@ static uchar *_ma_find_last_pos(MARIA_KEYDEF *keyinfo, uchar *page, } *return_key_length=last_length; *after_key=lastpos; - DBUG_PRINT("exit",("returns: %lx page: %lx end: %lx",prevpos,page,end)); + DBUG_PRINT("exit",("returns: 0x%lx page: 0x%lx end: 0x%lx", + (long) prevpos,(long) page,(long) end)); DBUG_RETURN(prevpos); } /* _ma_find_last_pos */ @@ -775,7 +777,7 @@ static int _ma_balance_page(register MARIA_HA *info, MARIA_KEYDEF *keyinfo, next_page= _ma_kpos(info->s->base.key_reflength, father_key_pos+father_keylength); buff=info->buff; - DBUG_PRINT("test",("use right page: %lu",next_page)); + DBUG_PRINT("test",("use right page: %lu", (ulong) next_page)); } else { @@ -784,7 +786,7 @@ static int _ma_balance_page(register MARIA_HA *info, MARIA_KEYDEF *keyinfo, next_page= _ma_kpos(info->s->base.key_reflength,father_key_pos); /* Fix that curr_buff is to left */ buff=curr_buff; curr_buff=info->buff; - DBUG_PRINT("test",("use left page: %lu",next_page)); + DBUG_PRINT("test",("use left page: %lu", (ulong) next_page)); } /* father_key_pos ptr to parting key */ if (!_ma_fetch_keypage(info,keyinfo,next_page,DFLT_INIT_HITS,info->buff,0)) diff --git a/storage/maria/maria_def.h b/storage/maria/maria_def.h index 5fb25512d4c..9585ecad292 100644 --- a/storage/maria/maria_def.h +++ b/storage/maria/maria_def.h @@ -662,7 +662,7 @@ typedef struct st_maria_block_info extern uint _ma_get_block_info(MARIA_BLOCK_INFO *, File, my_off_t); extern uint _ma_rec_pack(MARIA_HA *info, byte *to, const byte *from); -extern uint _ma_pack_get_block_info(MARIA_HA *mari, MARIA_BIT_BUFF *bit_buff, +extern uint _ma_pack_get_block_info(MARIA_HA *maria, MARIA_BIT_BUFF *bit_buff, MARIA_BLOCK_INFO *info, byte **rec_buff_p, File file, my_off_t filepos); extern void _ma_store_blob_length(byte *pos, uint pack_length, uint length); diff --git a/storage/myisam/mi_delete.c b/storage/myisam/mi_delete.c index 7972ef614a0..1a683e2ac30 100644 --- a/storage/myisam/mi_delete.c +++ b/storage/myisam/mi_delete.c @@ -366,9 +366,7 @@ static int d_search(register MI_INFO *info, register MI_KEYDEF *keyinfo, { /* This happens only with packed keys */ DBUG_PRINT("test",("Enlarging of key when deleting")); if (!_mi_get_last_key(info,keyinfo,anc_buff,lastkey,keypos,&length)) - { goto err; - } ret_value=_mi_insert(info,keyinfo,key,anc_buff,keypos,lastkey, (uchar*) 0,(uchar*) 0,(my_off_t) 0,(my_bool) 0); } -- cgit v1.2.1 From 345959c660d7401c9dc991a2c572ba145d6e199c Mon Sep 17 00:00:00 2001 From: unknown Date: Thu, 18 Jan 2007 21:38:14 +0200 Subject: Implementation of rows-in-block - Fixes some things missed in myisam->maria port - Moved variables that holds the state for the current row to 'cur_row' - Changed most uchar * to byte * to be able to remove a lot of casts - Removed RAID support - Added CHECK for rows-in-block - Added allocate_dynamic() for easier usage of dynamic rows when we know how many entries we will need - Reorder columns after CREATE for more optimal row storage (for rows-in-block) - Removed flag 'RRND_PRESERVER_LASTINX' (not needed) - Extended ma_test_all.sh to test more completely all row formats - New structs and variables to hold rows-in-block and bitmap information - Added org_data_file_type in header to allow easy restore of old record format when doing maria_pack / maria_chk -u - More virtual functions to handle different row types - Pointer to row is now MARIA_RECORD_POS instead of my_off_t - New header signature for MARIA index files - Fixed bugs in ma_test1.c and ma_test2.c - All key and row blocks are now of same size - We now only have one link chain for deleted key blocks include/m_string.h: Define bzero_if_purify include/maria.h: Implementation of rows-in-block include/my_base.h: Implementation of rows-in-block include/my_handler.h: Cleanup macros Added size_to_store_key_length() include/my_sys.h: Added 'allocate_dynamic()' include/myisamchk.h: Implementation of rows-in-block mysys/array.c: Added allocate_dynamic() mysys/mf_keycache.c: Moved DBUG_ENTER to it's right position mysys/my_pread.c: Ensure my_errno is always set sql/filesort.cc: Fixed some compiler warnings sql/gen_lex_hash.cc: Removed not needed 'inline' sql/ha_maria.cc: Implementation of rows-in-block Fixed compiler warnings sql/mysqld.cc: Fixed setting of wrong variable sql/uniques.cc: Fixed compiler warnings storage/maria/Makefile.am: Implementation of rows-in-block storage/maria/ma_check.c: Removed RAID functions Added support for CHECK of rows-in-blocks rows storage/maria/ma_checksum.c: Implementation of rows-in-block storage/maria/ma_close.c: Implementation of rows-in-block storage/maria/ma_create.c: Implementation of rows-in-block: - Reorder columns - All key blocks are now of same size - Removed old RAID support storage/maria/ma_dbug.c: Implementation of rows-in-block storage/maria/ma_delete.c: Implementation of rows-in-block storage/maria/ma_delete_all.c: Implementation of rows-in-block storage/maria/ma_dynrec.c: info->rec_buff is now allocated through _ma_alloc_buffer() Use new info->cur_row structure storage/maria/ma_extra.c: Implementation of rows-in-block storage/maria/ma_ft_boolean_search.c: Removed compiler warnings Indentation fixes storage/maria/ma_ft_nlq_search.c: Removed compiler warnings Indentation fixes storage/maria/ma_ft_update.c: Removed some casts storage/maria/ma_fulltext.h: Changed pointer type storage/maria/ma_info.c: Implementation of rows-in-block More general _ma_report_error() storage/maria/ma_init.c: Implementation of rows-in-block storage/maria/ma_key.c: Implementation of rows-in-block Removed some casts storage/maria/ma_keycache.c: Fixed DBUG entry storage/maria/ma_locking.c: Implementation of rows-in-block storage/maria/ma_open.c: Implementation of rows-in-block storage/maria/ma_packrec.c: Indentation fixes Changed uchar * to byte * to make it possible to remove some casts storage/maria/ma_page.c: Implementation of rows-in-block storage/maria/ma_range.c: Implementation of rows-in-block storage/maria/ma_rfirst.c: Implementation of rows-in-block storage/maria/ma_rkey.c: Implementation of rows-in-block Indentation fixes storage/maria/ma_rlast.c: Implementation of rows-in-block storage/maria/ma_rnext.c: Implementation of rows-in-block storage/maria/ma_rnext_same.c: Implementation of rows-in-block storage/maria/ma_rprev.c: Implementation of rows-in-block storage/maria/ma_rrnd.c: Implementation of rows-in-block Removed flag 'RRND_PRESERVER_LASTINX', by not resetting lastinx (This is reset by maria_scan_init()) storage/maria/ma_rsame.c: Implementation of rows-in-block storage/maria/ma_rsamepos.c: Implementation of rows-in-block storage/maria/ma_rt_index.c: Implementation of rows-in-block storage/maria/ma_rt_index.h: Implementation of rows-in-block storage/maria/ma_rt_key.c: Implementation of rows-in-block storage/maria/ma_rt_key.h: Implementation of rows-in-block storage/maria/ma_rt_mbr.c: Implementation of rows-in-block storage/maria/ma_rt_mbr.h: Implementation of rows-in-block storage/maria/ma_rt_split.c: Implementation of rows-in-block storage/maria/ma_rt_test.c: Indentation fix storage/maria/ma_scan.c: Implementation of rows-in-block Added 'maria_scan_end()' storage/maria/ma_search.c: Implementation of rows-in-block storage/maria/ma_sort.c: Indentation fixes uchar -> byte to be able to remove some casts storage/maria/ma_sp_defs.h: uchar * -> byte * storage/maria/ma_sp_key.c: uchar * -> byte * storage/maria/ma_sp_test.c: Indentation fixes storage/maria/ma_static.c: New header signature for MARIA storage/maria/ma_statrec.c: int -> my_bool functions my_off_t -> MARIA_RECORD_POS Fixed argument order for _ma_read_static_record() storage/maria/ma_test1.c: Implementation of rows-in-block Fixed some bugs in VARCHAR and BLOB testing storage/maria/ma_test2.c: Implementation of rows-in-block Fixed bug in BLOB testing storage/maria/ma_test3.c: Implementation of rows-in-block storage/maria/ma_test_all.sh: Run all tests with dynamic, static and block row formats (For the moment we skip REPAIR test of rows-in-block as this is not yet implemented) storage/maria/ma_unique.c: Implementation of rows-in-block storage/maria/ma_update.c: Implementation of rows-in-block storage/maria/ma_write.c: Implementation of rows-in-block Write of row is split into two parts, as rows-in-block format require us to do write of row before keys (to get row position) in contrast to all other row formats storage/maria/maria_chk.c: Implementation of rows-in-block storage/maria/maria_def.h: Implementation of rows-in-block - New structs and variables to hold rows-in-block and bitmap information - Added org_data_file_type in header to allow easy restore of old record format when doing maria_pack / maria_chk -u - More virtual functions to handle different row types - Pointer to row is now MARIA_RECORD_POS instead of my_off_t - uchar -> byte for many parameters to avoid casts storage/maria/maria_ftdump.c: Implementation of rows-in-block storage/maria/maria_pack.c: Implementation of rows-in-block storage/myisam/mi_check.c: Added new row types into switch to avoid compiler warnings Added some casts to avoid warnings after changing type of lastkey and buff storage/myisam/mi_create.c: Fix that 'pack_fields' is calculated correctly storage/myisam/mi_rsamepos.c: Implementation of rows-in-block storage/myisam/mi_test2.c: Fixed wrong printf storage/myisam/sort.c: uchar * -> byte * support-files/magic: Added support for Maria files Fided wrong entry's for MyISAM files storage/maria/ma_bitmap.c: New BitKeeper file ``storage/maria/ma_bitmap.c'' storage/maria/ma_blockrec.c: New BitKeeper file ``storage/maria/ma_blockrec.c'' storage/maria/ma_blockrec.h: New BitKeeper file ``storage/maria/ma_blockrec.h'' --- include/m_string.h | 7 +- include/maria.h | 96 +- include/my_base.h | 4 +- include/my_handler.h | 22 +- include/my_sys.h | 1 + include/myisamchk.h | 20 +- mysys/array.c | 56 +- mysys/mf_keycache.c | 17 +- mysys/my_pread.c | 6 +- sql/filesort.cc | 12 +- sql/gen_lex_hash.cc | 4 +- sql/ha_maria.cc | 69 +- sql/mysqld.cc | 1 + sql/uniques.cc | 2 +- storage/maria/Makefile.am | 3 +- storage/maria/ma_bitmap.c | 1704 +++++++++++++++++++++ storage/maria/ma_blockrec.c | 2742 ++++++++++++++++++++++++++++++++++ storage/maria/ma_blockrec.h | 160 ++ storage/maria/ma_check.c | 1716 +++++++++++++-------- storage/maria/ma_checksum.c | 25 +- storage/maria/ma_close.c | 29 +- storage/maria/ma_create.c | 454 ++++-- storage/maria/ma_dbug.c | 6 +- storage/maria/ma_delete.c | 204 +-- storage/maria/ma_delete_all.c | 3 +- storage/maria/ma_dynrec.c | 261 ++-- storage/maria/ma_extra.c | 34 +- storage/maria/ma_ft_boolean_search.c | 34 +- storage/maria/ma_ft_nlq_search.c | 35 +- storage/maria/ma_ft_update.c | 23 +- storage/maria/ma_fulltext.h | 2 +- storage/maria/ma_info.c | 22 +- storage/maria/ma_init.c | 2 + storage/maria/ma_key.c | 68 +- storage/maria/ma_keycache.c | 5 +- storage/maria/ma_locking.c | 53 +- storage/maria/ma_open.c | 653 ++++---- storage/maria/ma_packrec.c | 271 ++-- storage/maria/ma_page.c | 35 +- storage/maria/ma_range.c | 37 +- storage/maria/ma_rfirst.c | 2 +- storage/maria/ma_rkey.c | 76 +- storage/maria/ma_rlast.c | 2 +- storage/maria/ma_rnext.c | 9 +- storage/maria/ma_rnext_same.c | 33 +- storage/maria/ma_rprev.c | 9 +- storage/maria/ma_rrnd.c | 34 +- storage/maria/ma_rsame.c | 26 +- storage/maria/ma_rsamepos.c | 14 +- storage/maria/ma_rt_index.c | 266 ++-- storage/maria/ma_rt_index.h | 17 +- storage/maria/ma_rt_key.c | 12 +- storage/maria/ma_rt_key.h | 12 +- storage/maria/ma_rt_mbr.c | 34 +- storage/maria/ma_rt_mbr.h | 29 +- storage/maria/ma_rt_split.c | 15 +- storage/maria/ma_rt_test.c | 9 +- storage/maria/ma_scan.c | 33 +- storage/maria/ma_search.c | 351 ++--- storage/maria/ma_sort.c | 132 +- storage/maria/ma_sp_defs.h | 2 +- storage/maria/ma_sp_key.c | 2 +- storage/maria/ma_sp_test.c | 9 +- storage/maria/ma_static.c | 12 +- storage/maria/ma_statrec.c | 74 +- storage/maria/ma_test1.c | 162 +- storage/maria/ma_test2.c | 122 +- storage/maria/ma_test3.c | 4 +- storage/maria/ma_test_all.sh | 289 ++-- storage/maria/ma_unique.c | 30 +- storage/maria/ma_update.c | 37 +- storage/maria/ma_write.c | 216 +-- storage/maria/maria_chk.c | 195 +-- storage/maria/maria_def.h | 463 +++--- storage/maria/maria_ftdump.c | 6 +- storage/maria/maria_pack.c | 100 +- storage/myisam/mi_check.c | 63 +- storage/myisam/mi_create.c | 10 +- storage/myisam/mi_rsamepos.c | 3 +- storage/myisam/mi_test2.c | 2 +- storage/myisam/sort.c | 10 +- support-files/magic | 8 +- 82 files changed, 8852 insertions(+), 2980 deletions(-) create mode 100644 storage/maria/ma_bitmap.c create mode 100644 storage/maria/ma_blockrec.c create mode 100644 storage/maria/ma_blockrec.h diff --git a/include/m_string.h b/include/m_string.h index e009447c192..6ba1f5f8914 100644 --- a/include/m_string.h +++ b/include/m_string.h @@ -68,7 +68,7 @@ # define bcopy(s, d, n) memcpy((d), (s), (n)) # define bcmp(A,B,C) memcmp((A),(B),(C)) # define bzero(A,B) memset((A),0,(B)) -# define bmove_align(A,B,C) memcpy((A),(B),(C)) +# define bmove_align(A,B,C) memcpy((A),(B),(C)) #endif #if defined(__cplusplus) @@ -127,7 +127,10 @@ extern int bcmp(const char *s1,const char *s2,uint len); extern int my_bcmp(const char *s1,const char *s2,uint len); #undef bcmp #define bcmp(A,B,C) my_bcmp((A),(B),(C)) -#endif +#define bzero_if_purify(A,B) bzero(A,B) +#else +#define bzero_if_purify(A,B) +#endif /* HAVE_purify */ #ifndef bmove512 extern void bmove512(gptr dst,const gptr src,uint len); diff --git a/include/maria.h b/include/maria.h index 705ba7c91c7..b917efe1f6c 100644 --- a/include/maria.h +++ b/include/maria.h @@ -55,6 +55,7 @@ extern "C" { #define MARIA_MIN_KEY_BLOCK_LENGTH 1024 /* Min key block length */ #define MARIA_MAX_KEY_BLOCK_LENGTH 32768 #define maria_portable_sizeof_char_ptr 8 +#define MARIA_MAX_KEY_LENGTH 1000 /* Max length in bytes */ /* In the following macros '_keyno_' is 0 .. keys-1. @@ -109,13 +110,15 @@ extern "C" { /* Param to/from maria_info */ +typedef ulonglong MARIA_RECORD_POS; + typedef struct st_maria_isaminfo /* Struct from h_info */ { ha_rows records; /* Records in database */ ha_rows deleted; /* Deleted records in database */ - my_off_t recpos; /* Pos for last used record */ - my_off_t newrecpos; /* Pos if we write new record */ - my_off_t dupp_key_pos; /* Position to record with dup key */ + MARIA_RECORD_POS recpos; /* Pos for last used record */ + MARIA_RECORD_POS newrecpos; /* Pos if we write new record */ + MARIA_RECORD_POS dup_key_pos; /* Position to record with dup key */ my_off_t data_file_length; /* Length of data file */ my_off_t max_data_file_length, index_file_length; my_off_t max_index_file_length, delete_length; @@ -146,9 +149,11 @@ typedef struct st_maria_create_info ulonglong auto_increment; ulonglong data_file_length; ulonglong key_file_length; + uint null_bytes; uint old_options; + enum data_file_type org_data_file_type; uint8 language; - my_bool with_auto_increment; + my_bool with_auto_increment, transactional; } MARIA_CREATE_INFO; struct st_maria_info; /* For referense */ @@ -168,25 +173,24 @@ typedef struct st_maria_keydef /* Key definition with open & info */ uint16 keylength; /* Tot length of keyparts (auto) */ uint16 minlength; /* min length of (packed) key (auto) */ uint16 maxlength; /* max length of (packed) key (auto) */ - uint16 block_size_index; /* block_size (auto) */ uint32 version; /* For concurrent read/write */ uint32 ftparser_nr; /* distinct ftparser number */ HA_KEYSEG *seg, *end; struct st_mysql_ftparser *parser; /* Fulltext [pre]parser */ - int(*bin_search) (struct st_maria_info *info, - struct st_maria_keydef *keyinfo, uchar *page, uchar *key, - uint key_len, uint comp_flag, uchar **ret_pos, - uchar *buff, my_bool *was_last_key); - uint(*get_key) (struct st_maria_keydef *keyinfo, uint nod_flag, - uchar **page, uchar *key); - int(*pack_key) (struct st_maria_keydef *keyinfo, uint nod_flag, - uchar *next_key, uchar *org_key, uchar *prev_key, - uchar *key, struct st_maria_s_param *s_temp); - void(*store_key) (struct st_maria_keydef *keyinfo, uchar *key_pos, + int (*bin_search)(struct st_maria_info *info, + struct st_maria_keydef *keyinfo, byte *page, byte *key, + uint key_len, uint comp_flag, byte **ret_pos, + byte *buff, my_bool *was_last_key); + uint(*get_key)(struct st_maria_keydef *keyinfo, uint nod_flag, + byte **page, byte *key); + int (*pack_key)(struct st_maria_keydef *keyinfo, uint nod_flag, + byte *next_key, byte *org_key, byte *prev_key, + const byte *key, struct st_maria_s_param *s_temp); + void (*store_key)(struct st_maria_keydef *keyinfo, byte *key_pos, struct st_maria_s_param *s_temp); - int(*ck_insert) (struct st_maria_info *inf, uint k_nr, uchar *k, uint klen); - int(*ck_delete) (struct st_maria_info *inf, uint k_nr, uchar *k, uint klen); + int (*ck_insert)(struct st_maria_info *inf, uint k_nr, byte *k, uint klen); + int (*ck_delete)(struct st_maria_info *inf, uint k_nr, byte *k, uint klen); } MARIA_KEYDEF; @@ -195,7 +199,7 @@ typedef struct st_maria_keydef /* Key definition with open & info */ typedef struct st_maria_unique_def /* Segment definition of unique */ { uint16 keysegs; /* Number of key-segment */ - uchar key; /* Mapped to which key */ + uint8 key; /* Mapped to which key */ uint8 null_are_equal; HA_KEYSEG *seg, *end; } MARIA_UNIQUEDEF; @@ -218,16 +222,19 @@ struct st_maria_bit_buff; typedef struct st_maria_columndef /* column information */ { - int16 type; /* en_fieldtype */ + uint64 offset; /* Offset to position in row */ + enum en_fieldtype type; uint16 length; /* length of field */ - uint32 offset; /* Offset to position in row */ - uint8 null_bit; /* If column may be 0 */ - uint16 null_pos; /* position for null marker */ + uint16 fill_length; + uint16 null_pos; /* Position for null marker */ + uint16 empty_pos; /* Position for empty marker */ + uint8 null_bit; /* If column may be NULL */ + uint8 empty_bit; /* If column may be empty */ #ifndef NOT_PACKED_DATABASES - void(*unpack) (struct st_maria_columndef *rec, - struct st_maria_bit_buff *buff, - uchar *start, uchar *end); + void(*unpack)(struct st_maria_columndef *rec, + struct st_maria_bit_buff *buff, + byte *start, byte *end); enum en_fieldtype base_type; uint space_length_bits, pack_type; MARIA_DECODE_TREE *huff_tree; @@ -237,7 +244,8 @@ typedef struct st_maria_columndef /* column information */ extern ulong maria_block_size; extern ulong maria_concurrent_insert; -extern my_bool maria_flush, maria_delay_key_write, maria_single_user; +extern my_bool maria_flush, maria_single_user; +extern my_bool maria_delay_key_write, maria_delay_rec_write; extern my_off_t maria_max_temp_length; extern ulong maria_bulk_insert_tree_size, maria_data_pointer_size; extern KEY_CACHE maria_key_cache_var, *maria_key_cache; @@ -260,19 +268,22 @@ extern int maria_rlast(struct st_maria_info *file, byte *buf, int inx); extern int maria_rnext(struct st_maria_info *file, byte *buf, int inx); extern int maria_rnext_same(struct st_maria_info *info, byte *buf); extern int maria_rprev(struct st_maria_info *file, byte *buf, int inx); -extern int maria_rrnd(struct st_maria_info *file, byte *buf, my_off_t pos); +extern int maria_rrnd(struct st_maria_info *file, byte *buf, + MARIA_RECORD_POS pos); extern int maria_scan_init(struct st_maria_info *file); extern int maria_scan(struct st_maria_info *file, byte *buf); +extern void maria_scan_end(struct st_maria_info *file); extern int maria_rsame(struct st_maria_info *file, byte *record, int inx); extern int maria_rsame_with_pos(struct st_maria_info *file, byte *record, - int inx, my_off_t pos); + int inx, MARIA_RECORD_POS pos); extern int maria_update(struct st_maria_info *file, const byte *old, byte *new_record); extern int maria_write(struct st_maria_info *file, byte *buff); -extern my_off_t maria_position(struct st_maria_info *file); +extern MARIA_RECORD_POS maria_position(struct st_maria_info *file); extern int maria_status(struct st_maria_info *info, MARIA_INFO *x, uint flag); extern int maria_lock_database(struct st_maria_info *file, int lock_type); -extern int maria_create(const char *name, uint keys, MARIA_KEYDEF *keydef, +extern int maria_create(const char *name, enum data_file_type record_type, + uint keys, MARIA_KEYDEF *keydef, uint columns, MARIA_COLUMNDEF *columndef, uint uniques, MARIA_UNIQUEDEF *uniquedef, MARIA_CREATE_INFO *create_info, uint flags); @@ -324,7 +335,7 @@ typedef struct st_maria_sort_param MARIA_KEYDEF *keyinfo; MARIA_SORT_INFO *sort_info; HA_KEYSEG *seg; - uchar **sort_keys; + byte **sort_keys; byte *rec_buff; void *wordlist, *wordptr; MEM_ROOT wordroot; @@ -338,16 +349,17 @@ typedef struct st_maria_sort_param ulonglong unique[HA_MAX_KEY_SEG+1]; ulonglong notnull[HA_MAX_KEY_SEG+1]; - my_off_t pos,max_pos,filepos,start_recpos; + MARIA_RECORD_POS pos,max_pos,filepos,start_recpos; uint key, key_length,real_key_length,sortbuff_size; uint maxbuffers, keys, find_length, sort_keys_length; my_bool fix_datafile, master; + my_size_t rec_buff_size; int (*key_cmp)(struct st_maria_sort_param *, const void *, const void *); - int (*key_read)(struct st_maria_sort_param *,void *); - int (*key_write)(struct st_maria_sort_param *, const void *); + int (*key_read)(struct st_maria_sort_param *, byte *); + int (*key_write)(struct st_maria_sort_param *, const byte *); void (*lock_in_memory)(HA_CHECK *); - NEAR int (*write_keys)(struct st_maria_sort_param *, register uchar **, + NEAR int (*write_keys)(struct st_maria_sort_param *, register byte **, uint , struct st_buffpek *, IO_CACHE *); NEAR uint (*read_to_buffer)(IO_CACHE *,struct st_buffpek *, uint); NEAR int (*write_key)(struct st_maria_sort_param *, IO_CACHE *,char *, @@ -356,7 +368,7 @@ typedef struct st_maria_sort_param /* functions in maria_check */ -void mariachk_init(HA_CHECK *param); +void maria_chk_init(HA_CHECK *param); int maria_chk_status(HA_CHECK *param, MARIA_HA *info); int maria_chk_del(HA_CHECK *param, register MARIA_HA *info, uint test_flag); int maria_chk_size(HA_CHECK *param, MARIA_HA *info); @@ -371,19 +383,19 @@ int maria_repair_by_sort(HA_CHECK *param, register MARIA_HA *info, int maria_repair_parallel(HA_CHECK *param, register MARIA_HA *info, const char *name, int rep_quick); int maria_change_to_newfile(const char *filename, const char *old_ext, - const char *new_ext, uint raid_chunks, myf myflags); + const char *new_ext, myf myflags); void maria_lock_memory(HA_CHECK *param); int maria_update_state_info(HA_CHECK *param, MARIA_HA *info, uint update); void maria_update_key_parts(MARIA_KEYDEF *keyinfo, ulong *rec_per_key_part, - ulonglong *unique, ulonglong *notnull, - ulonglong records); + ulonglong *unique, ulonglong *notnull, + ulonglong records); int maria_filecopy(HA_CHECK *param, File to, File from, my_off_t start, - my_off_t length, const char *type); + my_off_t length, const char *type); int maria_movepoint(MARIA_HA *info, byte *record, my_off_t oldpos, - my_off_t newpos, uint prot_key); + my_off_t newpos, uint prot_key); int maria_write_data_suffix(MARIA_SORT_INFO *sort_info, my_bool fix_datafile); int maria_test_if_almost_full(MARIA_HA *info); -int maria_recreate_table(HA_CHECK *param, MARIA_HA ** org_info, char *filename); +int maria_recreate_table(HA_CHECK *param, MARIA_HA **org_info, char *filename); int maria_disable_indexes(MARIA_HA *info); int maria_enable_indexes(MARIA_HA *info); int maria_indexes_are_disabled(MARIA_HA *info); diff --git a/include/my_base.h b/include/my_base.h index 5ad7ac57fe1..76a67e3f3e7 100644 --- a/include/my_base.h +++ b/include/my_base.h @@ -48,7 +48,7 @@ #define HA_OPEN_FOR_REPAIR 32 /* open even if crashed */ #define HA_OPEN_FROM_SQL_LAYER 64 - /* The following is parameter to ha_rkey() how to use key */ +/* The following is parameter to ha_rkey() how to use key */ /* We define a complete-field prefix of a key value as a prefix where @@ -433,7 +433,7 @@ enum en_fieldtype { }; enum data_file_type { - STATIC_RECORD,DYNAMIC_RECORD,COMPRESSED_RECORD + STATIC_RECORD, DYNAMIC_RECORD, COMPRESSED_RECORD, BLOCK_RECORD }; /* For key ranges */ diff --git a/include/my_handler.h b/include/my_handler.h index 9b086036389..fd1bafe8d5c 100644 --- a/include/my_handler.h +++ b/include/my_handler.h @@ -58,33 +58,35 @@ typedef struct st_HA_KEYSEG /* Key-portion */ } HA_KEYSEG; #define get_key_length(length,key) \ -{ if ((uchar) *(key) != 255) \ - length= (uint) (uchar) *((key)++); \ +{ if (*(uchar*) (key) != 255) \ + length= (uint) *(uchar*) ((key)++); \ else \ - { length=mi_uint2korr((key)+1); (key)+=3; } \ + { length= mi_uint2korr((key)+1); (key)+=3; } \ } #define get_key_length_rdonly(length,key) \ -{ if ((uchar) *(key) != 255) \ - length= ((uint) (uchar) *((key))); \ +{ if (*(uchar*) (key) != 255) \ + length= ((uint) *(uchar*) ((key))); \ else \ - { length=mi_uint2korr((key)+1); } \ + { length= mi_uint2korr((key)+1); } \ } #define get_key_pack_length(length,length_pack,key) \ -{ if ((uchar) *(key) != 255) \ - { length= (uint) (uchar) *((key)++); length_pack=1; }\ +{ if (*(uchar*) (key) != 255) \ + { length= (uint) *(uchar*) ((key)++); length_pack= 1; }\ else \ - { length=mi_uint2korr((key)+1); (key)+=3; length_pack=3; } \ + { length=mi_uint2korr((key)+1); (key)+= 3; length_pack= 3; } \ } #define store_key_length_inc(key,length) \ { if ((length) < 255) \ - { *(key)++=(length); } \ + { *(key)++= (length); } \ else \ { *(key)=255; mi_int2store((key)+1,(length)); (key)+=3; } \ } +#define size_to_store_key_length(length) ((length) < 255 ? 1 : 3) + #define get_rec_bits(bit_ptr, bit_ofs, bit_len) \ (((((uint16) (bit_ptr)[1] << 8) | (uint16) (bit_ptr)[0]) >> (bit_ofs)) & \ ((1 << (bit_len)) - 1)) diff --git a/include/my_sys.h b/include/my_sys.h index 4b31f6bcd2b..3a491baf47f 100644 --- a/include/my_sys.h +++ b/include/my_sys.h @@ -760,6 +760,7 @@ extern my_bool insert_dynamic(DYNAMIC_ARRAY *array,gptr element); extern byte *alloc_dynamic(DYNAMIC_ARRAY *array); extern byte *pop_dynamic(DYNAMIC_ARRAY*); extern my_bool set_dynamic(DYNAMIC_ARRAY *array,gptr element,uint array_index); +extern my_bool allocate_dynamic(DYNAMIC_ARRAY *array, uint max_elements); extern void get_dynamic(DYNAMIC_ARRAY *array,gptr element,uint array_index); extern void delete_dynamic(DYNAMIC_ARRAY *array); extern void delete_dynamic_element(DYNAMIC_ARRAY *array, uint array_index); diff --git a/include/myisamchk.h b/include/myisamchk.h index 442fe5f7ece..66d0a77b62f 100644 --- a/include/myisamchk.h +++ b/include/myisamchk.h @@ -76,8 +76,8 @@ typedef struct st_sort_key_blocks /* Used when sorting */ { - uchar *buff, *end_pos; - uchar lastkey[HA_MAX_POSSIBLE_KEY_BUFF]; + byte *buff, *end_pos; + byte lastkey[HA_MAX_POSSIBLE_KEY_BUFF]; uint last_length; int inited; } SORT_KEY_BLOCKS; @@ -121,20 +121,24 @@ typedef struct st_handler_check_param my_off_t search_after_block; my_off_t new_file_pos, key_file_blocks; my_off_t keydata, totaldata, key_blocks, start_check_pos; - ha_rows total_records, total_deleted; + my_off_t used, empty, splits, del_length, link_used; + ha_rows total_records, total_deleted, records,del_blocks; + ha_rows full_page_count, tail_count; ha_checksum record_checksum, glob_crc; ha_checksum key_crc[HA_MAX_POSSIBLE_KEY]; + ha_checksum tmp_key_crc[HA_MAX_POSSIBLE_KEY]; + ha_checksum tmp_record_checksum; ulong use_buffers, read_buffer_length, write_buffer_length; ulong sort_buffer_length, sort_key_blocks; ulong rec_per_key_part[HA_MAX_KEY_SEG * HA_MAX_POSSIBLE_KEY]; uint out_flag, warning_printed, error_printed, verbose; uint opt_sort_key, total_files, max_level; uint testflag, key_cache_block_size; - int tmpfile_createflag; + int tmpfile_createflag, err_count; myf myf_rw; uint8 language; my_bool using_global_keycache, opt_lock_memory, opt_follow_links; - my_bool retry_repair, force_sort, calc_checksum; + my_bool retry_repair, force_sort, calc_checksum, static_row_size; char temp_filename[FN_REFLEN]; IO_CACHE read_cache; enum_handler_stats_method stats_method; @@ -143,15 +147,15 @@ typedef struct st_handler_check_param typedef struct st_sort_ftbuf { - uchar *buf, *end; + byte *buf, *end; int count; - uchar lastkey[HA_MAX_KEY_BUFF]; + byte lastkey[HA_MAX_KEY_BUFF]; } SORT_FT_BUF; typedef struct st_buffpek { my_off_t file_pos; /* Where we are in the sort file */ - uchar *base,*key; /* Key pointers */ + byte *base, *key; /* Key pointers */ ha_rows count; /* Number of rows in table */ ulong mem_count; /* numbers of keys in memory */ ulong max_keys; /* Max keys in buffert */ diff --git a/mysys/array.c b/mysys/array.c index a50d8b78178..2017bba5b61 100644 --- a/mysys/array.c +++ b/mysys/array.c @@ -61,7 +61,8 @@ my_bool init_dynamic_array(DYNAMIC_ARRAY *array, uint element_size, array->max_element=init_alloc; array->alloc_increment=alloc_increment; array->size_of_element=element_size; - if (!(array->buffer=(char*) my_malloc_ci(element_size*init_alloc,MYF(MY_WME)))) + if (!(array->buffer=(char*) my_malloc_ci(element_size*init_alloc, + MYF(MY_WME)))) { array->max_element=0; DBUG_RETURN(TRUE); @@ -154,7 +155,7 @@ byte *pop_dynamic(DYNAMIC_ARRAY *array) } /* - Replace elemnent in array with given element and index + Replace element in array with given element and index SYNOPSIS set_dynamic() @@ -175,19 +176,8 @@ my_bool set_dynamic(DYNAMIC_ARRAY *array, gptr element, uint idx) { if (idx >= array->elements) { - if (idx >= array->max_element) - { - uint size; - char *new_ptr; - size=(idx+array->alloc_increment)/array->alloc_increment; - size*= array->alloc_increment; - if (!(new_ptr=(char*) my_realloc(array->buffer,size* - array->size_of_element, - MYF(MY_WME | MY_ALLOW_ZERO_PTR)))) - return TRUE; - array->buffer=new_ptr; - array->max_element=size; - } + if (idx >= array->max_element && allocate_dynamic(array, idx)) + return TRUE; bzero((gptr) (array->buffer+array->elements*array->size_of_element), (idx - array->elements)*array->size_of_element); array->elements=idx+1; @@ -197,6 +187,42 @@ my_bool set_dynamic(DYNAMIC_ARRAY *array, gptr element, uint idx) return FALSE; } + +/* + Ensure that dynamic array has enough elements + + SYNOPSIS + allocate_dynamic() + array + max_elements Numbers of elements that is needed + + NOTES + Any new allocated element are NOT initialized + + RETURN VALUE + FALSE Ok + TRUE Allocation of new memory failed +*/ + +my_bool allocate_dynamic(DYNAMIC_ARRAY *array, uint max_elements) +{ + if (max_elements >= array->max_element) + { + uint size; + char *new_ptr; + size= (max_elements + array->alloc_increment)/array->alloc_increment; + size*= array->alloc_increment; + if (!(new_ptr= (char*) my_realloc(array->buffer,size* + array->size_of_element, + MYF(MY_WME | MY_ALLOW_ZERO_PTR)))) + return TRUE; + array->buffer= new_ptr; + array->max_element= size; + } + return FALSE; +} + + /* Get an element from array by given index diff --git a/mysys/mf_keycache.c b/mysys/mf_keycache.c index e6f4348968f..10a3e85eb4d 100644 --- a/mysys/mf_keycache.c +++ b/mysys/mf_keycache.c @@ -1791,8 +1791,6 @@ byte *key_cache_read(KEY_CACHE *keycache, uint offset= 0; byte *start= buff; DBUG_ENTER("key_cache_read"); - DBUG_PRINT("enter", ("fd: %u pos: %lu length: %u", - (uint) file, (ulong) filepos, length)); if (keycache->can_be_used) { @@ -1802,6 +1800,11 @@ byte *key_cache_read(KEY_CACHE *keycache, uint status; int page_st; + DBUG_PRINT("enter", ("fd: %u pos: %lu page: %lu length: %u", + (uint) file, (ulong) filepos, + (ulong) (filepos / keycache->key_cache_block_size), + length)); + offset= (uint) (filepos & (keycache->key_cache_block_size-1)); /* Read data in key_cache_block_size increments */ do @@ -2053,10 +2056,6 @@ int key_cache_write(KEY_CACHE *keycache, reg1 BLOCK_LINK *block; int error=0; DBUG_ENTER("key_cache_write"); - DBUG_PRINT("enter", - ("fd: %u pos: %lu length: %u block_length: %u key_block_length: %u", - (uint) file, (ulong) filepos, length, block_length, - keycache ? keycache->key_cache_block_size : 0)); if (!dont_write) { @@ -2078,6 +2077,12 @@ int key_cache_write(KEY_CACHE *keycache, int page_st; uint offset; + DBUG_PRINT("enter", + ("fd: %u pos: %lu page: %lu length: %u block_length: %u", + (uint) file, (ulong) filepos, + (ulong) (filepos / keycache->key_cache_block_size), + length, block_length)); + offset= (uint) (filepos & (keycache->key_cache_block_size-1)); do { diff --git a/mysys/my_pread.c b/mysys/my_pread.c index ac52895efe9..dd174c8a4cc 100644 --- a/mysys/my_pread.c +++ b/mysys/my_pread.c @@ -52,7 +52,7 @@ uint my_pread(File Filedes, byte *Buffer, uint Count, my_off_t offset, if (!error) /* Seek was successful */ { if ((readbytes = (uint) read(Filedes, Buffer, Count)) == -1L) - my_errno= errno; + my_errno= errno ? errno : -1; /* We should seek back, even if read failed. If this fails, @@ -68,7 +68,7 @@ uint my_pread(File Filedes, byte *Buffer, uint Count, my_off_t offset, #else if ((error= ((readbytes = (uint) pread(Filedes, Buffer, Count, offset)) != Count))) - my_errno= errno; + my_errno= errno ? errno : -1; #endif if (error || readbytes != Count) { @@ -84,8 +84,10 @@ uint my_pread(File Filedes, byte *Buffer, uint Count, my_off_t offset, my_error(EE_READ, MYF(ME_BELL+ME_WAITTANG), my_filename(Filedes),my_errno); else if (MyFlags & (MY_NABP | MY_FNABP)) + { my_error(EE_EOFERR, MYF(ME_BELL+ME_WAITTANG), my_filename(Filedes),my_errno); + } } if ((int) readbytes == -1 || (MyFlags & (MY_FNABP | MY_NABP))) DBUG_RETURN(MY_FILE_ERROR); /* Return with error */ diff --git a/sql/filesort.cc b/sql/filesort.cc index f41d72ac07a..3fb8517c624 100644 --- a/sql/filesort.cc +++ b/sql/filesort.cc @@ -1032,7 +1032,7 @@ uint read_to_buffer(IO_CACHE *fromfile, BUFFPEK *buffpek, void reuse_freed_buff(QUEUE *queue, BUFFPEK *reuse, uint key_length) { - uchar *reuse_end= reuse->base + reuse->max_keys * key_length; + byte *reuse_end= reuse->base + reuse->max_keys * key_length; for (uint i= 0; i < queue->elements; ++i) { BUFFPEK *bp= (BUFFPEK *) queue_element(queue, i); @@ -1103,7 +1103,7 @@ int merge_buffers(SORTPARAM *param, IO_CACHE *from_file, offset= rec_length-res_length; maxcount= (ulong) (param->keys/((uint) (Tb-Fb) +1)); to_start_filepos= my_b_tell(to_file); - strpos= (uchar*) sort_buffer; + strpos= sort_buffer; org_max_rows=max_rows= param->max_rows; /* The following will fire if there is not enough space in sort_buffer */ @@ -1115,10 +1115,10 @@ int merge_buffers(SORTPARAM *param, IO_CACHE *from_file, DBUG_RETURN(1); /* purecov: inspected */ for (buffpek= Fb ; buffpek <= Tb ; buffpek++) { - buffpek->base= strpos; + buffpek->base= (byte*) strpos; buffpek->max_keys= maxcount; strpos+= (uint) (error= (int) read_to_buffer(from_file, buffpek, - rec_length)); + rec_length)); if (error == -1) goto err; /* purecov: inspected */ buffpek->max_keys= buffpek->mem_count; // If less data in buffers than expected @@ -1207,7 +1207,7 @@ int merge_buffers(SORTPARAM *param, IO_CACHE *from_file, } } buffpek= (BUFFPEK*) queue_top(&queue); - buffpek->base= sort_buffer; + buffpek->base= (byte*) sort_buffer; buffpek->max_keys= param->keys; /* @@ -1242,7 +1242,7 @@ int merge_buffers(SORTPARAM *param, IO_CACHE *from_file, else { register uchar *end; - strpos= buffpek->key+offset; + strpos= (uchar*) buffpek->key+offset; for (end= strpos+buffpek->mem_count*rec_length ; strpos != end ; strpos+= rec_length) diff --git a/sql/gen_lex_hash.cc b/sql/gen_lex_hash.cc index 7e0b178f7af..4b319d0a87c 100644 --- a/sql/gen_lex_hash.cc +++ b/sql/gen_lex_hash.cc @@ -463,8 +463,8 @@ int main(int argc,char **argv) printf("\nstatic unsigned int symbols_max_len=%d;\n\n", max_len2); printf("\ -static inline SYMBOL *get_hash_symbol(const char *s,\n\ - unsigned int len,bool function)\n\ +static SYMBOL *get_hash_symbol(const char *s,\n\ + unsigned int len,bool function)\n\ {\n\ register uchar *hash_map;\n\ register const char *cur_str= s;\n\ diff --git a/sql/ha_maria.cc b/sql/ha_maria.cc index 29718f1493e..aa37cc96452 100644 --- a/sql/ha_maria.cc +++ b/sql/ha_maria.cc @@ -220,10 +220,10 @@ int ha_maria::dump(THD * thd, int fd) { MARIA_SHARE *share= file->s; NET *net= &thd->net; - uint blocksize= share->blocksize; + uint block_size= share->block_size; my_off_t bytes_to_read= share->state.state.data_file_length; int data_fd= file->dfile; - byte *buf= (byte *) my_malloc(blocksize, MYF(MY_WME)); + byte *buf= (byte *) my_malloc(block_size, MYF(MY_WME)); if (!buf) return ENOMEM; @@ -231,7 +231,7 @@ int ha_maria::dump(THD * thd, int fd) my_seek(data_fd, 0L, MY_SEEK_SET, MYF(MY_WME)); for (; bytes_to_read > 0;) { - uint bytes= my_read(data_fd, buf, blocksize, MYF(MY_WME)); + uint bytes= my_read(data_fd, buf, block_size, MYF(MY_WME)); if (bytes == MY_FILE_ERROR) { error= errno; @@ -364,7 +364,7 @@ int ha_maria::check(THD * thd, HA_CHECK_OPT * check_opt) const char *old_proc_info= thd->proc_info; thd->proc_info= "Checking table"; - mariachk_init(¶m); + maria_chk_init(¶m); param.thd= thd; param.op_name= "check"; param.db_name= table->s->db.str; @@ -454,7 +454,7 @@ int ha_maria::analyze(THD *thd, HA_CHECK_OPT * check_opt) HA_CHECK param; MARIA_SHARE *share= file->s; - mariachk_init(¶m); + maria_chk_init(¶m); param.thd= thd; param.op_name= "analyze"; param.db_name= table->s->db.str; @@ -510,7 +510,7 @@ int ha_maria::restore(THD * thd, HA_CHECK_OPT *check_opt) err: { HA_CHECK param; - mariachk_init(¶m); + maria_chk_init(¶m); param.thd= thd; param.op_name= "restore"; param.db_name= table->s->db.str; @@ -570,7 +570,7 @@ int ha_maria::backup(THD * thd, HA_CHECK_OPT *check_opt) err: { HA_CHECK param; - mariachk_init(¶m); + maria_chk_init(¶m); param.thd= thd; param.op_name= "backup"; param.db_name= table->s->db.str; @@ -591,7 +591,7 @@ int ha_maria::repair(THD * thd, HA_CHECK_OPT *check_opt) if (!file) return HA_ADMIN_INTERNAL_ERROR; - mariachk_init(¶m); + maria_chk_init(¶m); param.thd= thd; param.op_name= "repair"; param.testflag= ((check_opt->flags & ~(T_EXTEND)) | @@ -638,7 +638,7 @@ int ha_maria::optimize(THD * thd, HA_CHECK_OPT *check_opt) return HA_ADMIN_INTERNAL_ERROR; HA_CHECK param; - mariachk_init(¶m); + maria_chk_init(¶m); param.thd= thd; param.op_name= "optimize"; param.testflag= (check_opt->flags | T_SILENT | T_FORCE_CREATE | @@ -833,7 +833,7 @@ err: { /* Send error to user */ HA_CHECK param; - mariachk_init(¶m); + maria_chk_init(¶m); param.thd= thd; param.op_name= "assign_to_keycache"; param.db_name= table->s->db.str; @@ -901,7 +901,7 @@ int ha_maria::preload_keys(THD * thd, HA_CHECK_OPT *check_opt) err: { HA_CHECK param; - mariachk_init(¶m); + maria_chk_init(¶m); param.thd= thd; param.op_name= "preload_keys"; param.db_name= table->s->db.str; @@ -1010,7 +1010,7 @@ int ha_maria::enable_indexes(uint mode) HA_CHECK param; const char *save_proc_info= thd->proc_info; thd->proc_info= "Creating index"; - mariachk_init(¶m); + maria_chk_init(¶m); param.op_name= "recreating_index"; param.testflag= (T_SILENT | T_REP_BY_SORT | T_QUICK | T_CREATE_MISSING_KEYS); @@ -1395,7 +1395,7 @@ void ha_maria::info(uint flag) if (flag & HA_STATUS_ERRKEY) { errkey= info.errkey; - my_store_ptr(dup_ref, ref_length, info.dupp_key_pos); + my_store_ptr(dup_ref, ref_length, info.dup_key_pos); } /* Faster to always update, than to do it based on flag */ stats.update_time= info.update_time; @@ -1482,6 +1482,7 @@ int ha_maria::create(const char *name, register TABLE *table_arg, HA_KEYSEG *keyseg; TABLE_SHARE *share= table_arg->s; uint options= share->db_options_in_use; + enum data_file_type row_type; DBUG_ENTER("ha_maria::create"); type= HA_KEYTYPE_BINARY; // Keep compiler happy @@ -1607,27 +1608,27 @@ int ha_maria::create(const char *name, register TABLE *table_arg, if (recpos != minpos) { // Reserved space (Null bits?) bzero((char*) recinfo_pos, sizeof(*recinfo_pos)); - recinfo_pos->type= (int) FIELD_NORMAL; + recinfo_pos->type= FIELD_NORMAL; recinfo_pos++->length= (uint16) (minpos - recpos); } if (!found) break; if (found->flags & BLOB_FLAG) - recinfo_pos->type= (int) FIELD_BLOB; + recinfo_pos->type= FIELD_BLOB; else if (found->type() == MYSQL_TYPE_VARCHAR) recinfo_pos->type= FIELD_VARCHAR; else if (!(options & HA_OPTION_PACK_RECORD)) - recinfo_pos->type= (int) FIELD_NORMAL; + recinfo_pos->type= FIELD_NORMAL; else if (found->zero_pack()) - recinfo_pos->type= (int) FIELD_SKIP_ZERO; + recinfo_pos->type= FIELD_SKIP_ZERO; else - recinfo_pos->type= (int) ((length <= 3 || - (found->flags & ZEROFILL_FLAG)) ? - FIELD_NORMAL : - found->type() == MYSQL_TYPE_STRING || - found->type() == MYSQL_TYPE_VAR_STRING ? - FIELD_SKIP_ENDSPACE : FIELD_SKIP_PRESPACE); + recinfo_pos->type= ((length <= 3 || + (found->flags & ZEROFILL_FLAG)) ? + FIELD_NORMAL : + found->type() == MYSQL_TYPE_STRING || + found->type() == MYSQL_TYPE_VAR_STRING ? + FIELD_SKIP_ENDSPACE : FIELD_SKIP_PRESPACE); if (found->null_ptr) { recinfo_pos->null_bit= found->null_bit; @@ -1666,12 +1667,26 @@ int ha_maria::create(const char *name, register TABLE *table_arg, if (options & HA_OPTION_DELAY_KEY_WRITE) create_flags |= HA_CREATE_DELAY_KEY_WRITE; + switch (info->row_type) { + case ROW_TYPE_FIXED: + row_type= STATIC_RECORD; + break; + case ROW_TYPE_DYNAMIC: + row_type= DYNAMIC_RECORD; + break; + default: + case ROW_TYPE_PAGES: + row_type= BLOCK_RECORD; + break; + } + /* TODO: Check that the following fn_format is really needed */ error= - maria_create(fn_format - (buff, name, "", "", MY_UNPACK_FILENAME | MY_APPEND_EXT), - share->keys, keydef, (uint) (recinfo_pos - recinfo), recinfo, - 0, (MARIA_UNIQUEDEF *) 0, &create_info, create_flags); + maria_create(fn_format(buff, name, "", "", + MY_UNPACK_FILENAME | MY_APPEND_EXT), + row_type, share->keys, keydef, (uint) (recinfo_pos - recinfo), + recinfo, 0, (MARIA_UNIQUEDEF *) 0, &create_info, + create_flags); my_free((gptr) recinfo, MYF(0)); DBUG_RETURN(error); diff --git a/sql/mysqld.cc b/sql/mysqld.cc index a666bec32d7..45adaa66a4c 100644 --- a/sql/mysqld.cc +++ b/sql/mysqld.cc @@ -7133,6 +7133,7 @@ static void mysql_init_variables(void) when collecting index statistics for MyISAM tables. */ global_system_variables.myisam_stats_method= MI_STATS_METHOD_NULLS_NOT_EQUAL; + global_system_variables.maria_stats_method= MI_STATS_METHOD_NULLS_NOT_EQUAL; /* Variables that depends on compile options */ #ifndef DBUG_OFF diff --git a/sql/uniques.cc b/sql/uniques.cc index ad074f8b2b0..933283bf305 100644 --- a/sql/uniques.cc +++ b/sql/uniques.cc @@ -446,7 +446,7 @@ static bool merge_walk(uchar *merge_buffer, uint merge_buffer_size, */ for (top= begin; top != end; ++top) { - top->base= merge_buffer + (top - begin) * piece_size; + top->base= (byte*) (merge_buffer + (top - begin) * piece_size); top->max_keys= max_key_count_per_piece; bytes_read= read_to_buffer(file, top, key_length); if (bytes_read == (uint) (-1)) diff --git a/storage/maria/Makefile.am b/storage/maria/Makefile.am index d4315b4d446..b2936143c36 100644 --- a/storage/maria/Makefile.am +++ b/storage/maria/Makefile.am @@ -28,7 +28,7 @@ bin_PROGRAMS = maria_chk maria_pack maria_ftdump maria_chk_DEPENDENCIES= $(LIBRARIES) maria_pack_DEPENDENCIES=$(LIBRARIES) noinst_PROGRAMS = ma_test1 ma_test2 ma_test3 ma_rt_test ma_sp_test -noinst_HEADERS = maria_def.h ma_rt_index.h ma_rt_key.h ma_rt_mbr.h ma_sp_defs.h ma_fulltext.h ma_ftdefs.h ma_ft_test1.h ma_ft_eval.h +noinst_HEADERS = maria_def.h ma_blockrec.h ma_rt_index.h ma_rt_key.h ma_rt_mbr.h ma_sp_defs.h ma_fulltext.h ma_ftdefs.h ma_ft_test1.h ma_ft_eval.h ma_test1_DEPENDENCIES= $(LIBRARIES) ma_test2_DEPENDENCIES= $(LIBRARIES) ma_test3_DEPENDENCIES= $(LIBRARIES) @@ -42,6 +42,7 @@ libmaria_a_SOURCES = ma_init.c ma_open.c ma_extra.c ma_info.c ma_rkey.c \ ma_search.c ma_page.c ma_key.c ma_locking.c \ ma_rrnd.c ma_scan.c ma_cache.c \ ma_statrec.c ma_packrec.c ma_dynrec.c \ + ma_blockrec.c ma_bitmap.c \ ma_update.c ma_write.c ma_unique.c \ ma_delete.c \ ma_rprev.c ma_rfirst.c ma_rlast.c ma_rsame.c \ diff --git a/storage/maria/ma_bitmap.c b/storage/maria/ma_bitmap.c new file mode 100644 index 00000000000..5ed5a776658 --- /dev/null +++ b/storage/maria/ma_bitmap.c @@ -0,0 +1,1704 @@ +/* Copyright (C) 2007 Michael Widenius + + 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; either version 2 of the License, or + (at your option) any later version. + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ + +/* + Bitmap handling (for records in block) + + The data file starts with a bitmap page, followed by as many data + pages as the bitmap can cover. After this there is a new bitmap page + and more data pages etc. + + The bitmap code assumes there is always an active bitmap page and thus + that there is at least one bitmap page in the file + + Structure of bitmap page: + + Fixed size records (to be implemented later): + + 2 bits are used to indicate: + + 0 Empty + 1 50-75 % full (at least room for 2 records) + 2 75-100 % full (at least room for one record) + 3 100 % full (no more room for records) + + Assuming 8K pages, this will allow us to map: + 8192 (bytes per page) * 4 (pages mapped per byte) * 8192 (page size)= 256M + + (For Maria this will be 7*4 * 8192 = 224K smaller because of LSN) + + Note that for fixed size rows, we can't add more columns without doing + a full reorganization of the table. The user can always force a dynamic + size row format by specifying ROW_FORMAT=dynamic. + + + Dynamic size records: + + 3 bits are used to indicate + + 0 Empty page + 1 0-30 % full (at least room for 3 records) + 2 30-60 % full (at least room for 2 records) + 3 60-90 % full (at least room for one record) + 4 100 % full (no more room for records) + 5 Tail page, 0-40 % full + 6 Tail page, 40-80 % full + 7 Full tail page or full blob page + + Assuming 8K pages, this will allow us to map: + 8192 (bytes per page) * 8 bits/byte / 3 bits/page * 8192 (page size)= 170.7M + + Note that values 1-3 may be adjust for each individual table based on + 'min record length'. Tail pages are for overflow data which can be of + any size and thus doesn't have to be adjusted for different tables. + If we add more columns to the table, some of the originally calculated + 'cut off' points may not be optimal, but they shouldn't be 'drasticly + wrong'. + + When allocating data from the bitmap, we are trying to do it in a + 'best fit' manner. Blobs and varchar blocks are given out in large + continuous extents to allow fast access to these. Before allowing a + row to 'flow over' to other blocks, we will compact the page and use + all space on it. If there is many rows in the page, we will ensure + there is *LEFT_TO_GROW_ON_SPLIT* bytes left on the page to allow other + rows to grow. + + The bitmap format allows us to extend the row file in big chunks, if needed. + + When calculating the size for a packed row, we will calculate the following + things separately: + - Row header + null_bits + empty_bits fixed size segments etc. + - Size of all char/varchar fields + - Size of each blob field + + The bitmap handler will get all the above information and return + either one page or a set of pages to put the different parts. + + Bitmaps are read on demand in response to insert/delete/update operations. + The following bitmap pointers will be cached and stored on disk on close: + - Current insert_bitmap; When inserting new data we will first try to + fill this one. + - First bitmap which is not completely full. This is updated when we + free data with an update or delete. + + While flushing out bitmaps, we will cache the status of the bitmap in memory + to avoid having to read a bitmap for insert of new data that will not + be of any use + - Total empty space + - Largest number of continuous pages + + Bitmap ONLY goes to disk in the following scenarios + - The file is closed (and we flush all changes to disk) + - On checkpoint + (Ie: When we do a checkpoint, we have to ensure that all bitmaps are + put on disk even if they are not in the page cache). + - When explicitely requested (for example on backup or after recvoery, + to simplify things) + +*/ + +#include "maria_def.h" +#include "ma_blockrec.h" + +/* Number of pages to store blob parts */ +#define BLOB_SEGMENT_MIN_SIZE 128 + +#define FULL_HEAD_PAGE 4 +#define FULL_TAIL_PAGE 7 + +static inline my_bool write_changed_bitmap(MARIA_SHARE *share, + MARIA_FILE_BITMAP *bitmap) +{ + return (key_cache_write(share->key_cache, + bitmap->file, bitmap->page * bitmap->block_size, 0, + (byte*) bitmap->map, + bitmap->block_size, bitmap->block_size, 1)); +} + +/* + Initialize bitmap. This is called the first time a file is opened +*/ + +my_bool _ma_bitmap_init(MARIA_SHARE *share, File file) +{ + uint aligned_bit_blocks; + uint max_page_size; + MARIA_FILE_BITMAP *bitmap= &share->bitmap; + uint size= share->block_size; +#ifndef DBUG_OFF + /* We want to have a copy of the bitmap to be able to print differences */ + size*= 2; +#endif + + if (!(bitmap->map= (uchar*) my_malloc(size, MYF(MY_WME)))) + return 1; + + bitmap->file= file; + bitmap->changed= 0; + bitmap->block_size= share->block_size; + /* Size needs to be alligned on 6 */ + aligned_bit_blocks= share->block_size / 6; + bitmap->total_size= aligned_bit_blocks * 6; + /* + In each 6 bytes, we have 6*8/3 = 16 pages covered + The +1 is to add the bitmap page, as this doesn't have to be covered + */ + bitmap->pages_covered= aligned_bit_blocks * 16 + 1; + + /* Update size for bits */ + /* TODO; Make this dependent of the row size */ + max_page_size= share->block_size - PAGE_OVERHEAD_SIZE; + bitmap->sizes[0]= max_page_size; /* Empty page */ + bitmap->sizes[1]= max_page_size - max_page_size * 30 / 100; + bitmap->sizes[2]= max_page_size - max_page_size * 60 / 100; + bitmap->sizes[3]= max_page_size - max_page_size * 90 / 100; + bitmap->sizes[4]= 0; /* Full page */ + bitmap->sizes[5]= max_page_size - max_page_size * 40 / 100; + bitmap->sizes[6]= max_page_size - max_page_size * 80 / 100; + bitmap->sizes[7]= 0; + + pthread_mutex_init(&share->bitmap.bitmap_lock, MY_MUTEX_INIT_SLOW); + + /* + Start by reading first page (assume table scan) + Later code is simpler if it can assume we always have an active bitmap. + */ + if (_ma_read_bitmap_page(share, bitmap, (ulonglong) 0)) + return(1); + return 0; +} + + +/* + Free data allocated by _ma_bitmap_init +*/ + +my_bool _ma_bitmap_end(MARIA_SHARE *share) +{ + my_bool res= 0; + _ma_flush_bitmap(share); + pthread_mutex_destroy(&share->bitmap.bitmap_lock); + my_free((byte*) share->bitmap.map, MYF(MY_ALLOW_ZERO_PTR)); + return res; +} + + +/* + Flush bitmap to disk +*/ + +my_bool _ma_flush_bitmap(MARIA_SHARE *share) +{ + my_bool res= 0; + if (share->bitmap.changed) + { + pthread_mutex_lock(&share->bitmap.bitmap_lock); + if (share->bitmap.changed) + { + res= write_changed_bitmap(share, &share->bitmap); + share->bitmap.changed= 0; + } + pthread_mutex_unlock(&share->bitmap.bitmap_lock); + } + return res; +} + + +/* + Return bitmap pattern for the smallest head block that can hold 'size' + + SYNOPSIS + size_to_head_pattern() + bitmap Bitmap + size Requested size + + RETURN + 0-3 For a description of the bitmap sizes, see the header +*/ + +static uint size_to_head_pattern(MARIA_FILE_BITMAP *bitmap, uint size) +{ + if (size <= bitmap->sizes[3]) + return 3; + if (size <= bitmap->sizes[2]) + return 2; + if (size <= bitmap->sizes[1]) + return 1; + DBUG_ASSERT(size <= bitmap->sizes[0]); + return 0; +} + + +/* + Return bitmap pattern for block where there is size bytes free +*/ + +uint _ma_free_size_to_head_pattern(MARIA_FILE_BITMAP *bitmap, uint size) +{ + if (size < bitmap->sizes[3]) + return 4; + if (size < bitmap->sizes[2]) + return 3; + if (size < bitmap->sizes[1]) + return 2; + return (size < bitmap->sizes[0]) ? 1 : 0; +} + + +/* + Return bitmap pattern for the smallest tail block that can hold 'size' + + SYNOPSIS + size_to_tail_pattern() + bitmap Bitmap + size Requested size + + RETURN + 0, 5 or 6 For a description of the bitmap sizes, see the header +*/ + +static uint size_to_tail_pattern(MARIA_FILE_BITMAP *bitmap, uint size) +{ + if (size <= bitmap->sizes[6]) + return 6; + if (size <= bitmap->sizes[5]) + return 5; + DBUG_ASSERT(size <= bitmap->sizes[0]); + return 0; +} + + +static uint free_size_to_tail_pattern(MARIA_FILE_BITMAP *bitmap, uint size) +{ + if (size >= bitmap->sizes[0]) + return 0; /* Revert to empty page */ + if (size < bitmap->sizes[6]) + return 7; + if (size < bitmap->sizes[5]) + return 6; + return 5; +} + + +/* + Return size guranteed to be available on a page + + SYNOPSIS + pattern_to_head_size + bitmap Bitmap + pattern Pattern (0-7) + + RETURN + 0 - block_size +*/ + +static inline uint pattern_to_size(MARIA_FILE_BITMAP *bitmap, uint pattern) +{ + DBUG_ASSERT(pattern <= 7); + return bitmap->sizes[pattern]; +} + + +/* + Print bitmap for debugging +*/ + +#ifndef DBUG_OFF + +const char *bits_to_txt[]= +{ + "empty", "00-30% full", "30-60% full", "60-90% full", "full", + "tail 00-40 % full", "tail 40-80 % full", "tail/blob full" +}; + +static void _ma_print_bitmap(MARIA_FILE_BITMAP *bitmap) +{ + uchar *pos, *end, *org_pos; + ulong page; + + end= bitmap->map+ bitmap->used_size; + DBUG_LOCK_FILE; + fprintf(DBUG_FILE,"\nBitmap page changes at page %lu\n", + (ulong) bitmap->page); + + page= (ulong) bitmap->page+1; + for (pos= bitmap->map, org_pos= bitmap->map+bitmap->block_size ; pos < end ; + pos+= 6, org_pos+= 6) + { + ulonglong bits= uint6korr(pos); /* 6 bytes = 6*8/3= 16 patterns */ + ulonglong org_bits= uint6korr(org_pos); + uint i; + if (bits != org_bits) + { + for (i= 0; i < 16 ; i++, bits>>= 3, org_bits>>= 3) + { + if ((bits & 7) != (org_bits & 7)) + fprintf(DBUG_FILE, "Page: %8lu %s -> %s\n", page+i, + bits_to_txt[org_bits & 7], bits_to_txt[bits & 7]); + } + } + page+= 16; + } + fputc('\n', DBUG_FILE); + DBUG_UNLOCK_FILE; + memcpy(bitmap->map+ bitmap->block_size, bitmap->map, bitmap->block_size); +} + +#endif /* DBUG_OFF */ + + +/*************************************************************************** + Reading & writing bitmap pages +***************************************************************************/ + +/* + Read a given bitmap page + + SYNOPSIS + read_bitmap_page() + info Maria handler + bitmap Bitmap handler + page Page to read + + TODO + Update 'bitmap->used_size' to real size of used bitmap + + RETURN + 0 ok + 1 error (Error writing old bitmap or reading bitmap page) +*/ + +my_bool _ma_read_bitmap_page(MARIA_SHARE *share, MARIA_FILE_BITMAP *bitmap, + ulonglong page) +{ + my_off_t position= page * bitmap->block_size; + my_bool res; + DBUG_ENTER("_ma_read_bitmap_page"); + DBUG_ASSERT(page % bitmap->pages_covered == 0); + + bitmap->page= page; + if (position >= share->state.state.data_file_length) + { + share->state.state.data_file_length= position + bitmap->block_size; + bzero(bitmap->map, bitmap->block_size); + bitmap->used_size= 0; + DBUG_RETURN(0); + } + bitmap->used_size= bitmap->total_size; + res= key_cache_read(share->key_cache, + bitmap->file, position, 0, + (byte*) bitmap->map, + bitmap->block_size, bitmap->block_size, 0) == 0; +#ifndef DBUG_OFF + if (!res) + memcpy(bitmap->map+ bitmap->block_size, bitmap->map, bitmap->block_size); +#endif + DBUG_RETURN(res); +} + + +/* + Change to another bitmap page + + SYNOPSIS + _ma_change_bitmap_page() + info Maria handler + bitmap Bitmap handler + page Bitmap page to read + + NOTES + If old bitmap was changed, write it out before reading new one + We return empty bitmap if page is outside of file size + + RETURN + 0 ok + 1 error (Error writing old bitmap or reading bitmap page) +*/ + +static my_bool _ma_change_bitmap_page(MARIA_HA *info, + MARIA_FILE_BITMAP *bitmap, + ulonglong page) +{ + DBUG_ENTER("_ma_change_bitmap_page"); + DBUG_ASSERT(page % bitmap->pages_covered == 0); + + if (bitmap->changed) + { + if (write_changed_bitmap(info->s, bitmap)) + DBUG_RETURN(1); + bitmap->changed= 0; + } + DBUG_RETURN(_ma_read_bitmap_page(info->s, bitmap, page)); +} + + +/* + Read next suitable bitmap + + SYNOPSIS + move_to_next_bitmap() + bitmap Bitmap handle + + TODO + Add cache of bitmaps to not read something that is not usable + + RETURN + 0 ok + 1 error (either couldn't save old bitmap or read new one +*/ + +static my_bool move_to_next_bitmap(MARIA_HA *info, MARIA_FILE_BITMAP *bitmap) +{ + ulonglong page= bitmap->page; + MARIA_STATE_INFO *state= &info->s->state; + DBUG_ENTER("move_to_next_bitmap"); + + if (state->first_bitmap_with_space != ~(ulonglong) 0 && + state->first_bitmap_with_space != page) + { + page= state->first_bitmap_with_space; + state->first_bitmap_with_space= ~(ulonglong) 0; + } + else + page+= bitmap->pages_covered; + DBUG_RETURN(_ma_change_bitmap_page(info, bitmap, page)); +} + + +/**************************************************************************** + Allocate data in bitmaps +****************************************************************************/ + +/* + Store data in 'block' and mark the place used in the bitmap + + SYNOPSIS + fill_block() + bitmap Bitmap handle + block Store data about what we found + best_data Pointer to best 6 byte aligned area in bitmap->map + best_pos Which bit in *best_data the area starts + 0 = first bit pattern, 1 second bit pattern etc + fill_pattern Bitmap pattern to store in best_data[best_pos] +*/ + +static void fill_block(MARIA_FILE_BITMAP *bitmap, + MARIA_BITMAP_BLOCK *block, + uchar *best_data, uint best_pos, uint best_bits, + uint fill_pattern) +{ + uint page, offset, tmp; + uchar *data; + + /* For each 6 bytes we have 6*8/3= 16 patterns */ + page= (best_data - bitmap->map) / 6 * 16 + best_pos; + block->page= bitmap->page + 1 + page; + block->page_count= 1 + TAIL_BIT; + block->empty_space= pattern_to_size(bitmap, best_bits); + block->sub_blocks= 1; + block->org_bitmap_value= best_bits; + block->used= BLOCKUSED_TAIL; + + /* + Mark place used by reading/writing 2 bytes at a time to handle + bitmaps in overlapping bytes + */ + best_pos*= 3; + data= best_data+ best_pos / 8; + offset= best_pos & 7; + tmp= uint2korr(data); + tmp= (tmp & ~(7 << offset)) | (fill_pattern << offset); + int2store(data, tmp); + bitmap->changed= 1; + DBUG_EXECUTE("bitmap", _ma_print_bitmap(bitmap);); +} + + +/* + Allocate data for head block + + SYNOPSIS + allocate_head() + bitmap bitmap + size Size of block we need to find + block Store found information here + + RETURN + 0 ok (block is updated) + 1 error (no space in bitmap; block is not touched) +*/ + + +static my_bool allocate_head(MARIA_FILE_BITMAP *bitmap, uint size, + MARIA_BITMAP_BLOCK *block) +{ + uint min_bits= size_to_head_pattern(bitmap, size); + uchar *data= bitmap->map, *end= data + bitmap->used_size; + uchar *best_data= 0; + uint best_bits= (uint) -1, best_pos; + DBUG_ENTER("allocate_head"); + + LINT_INIT(best_pos); + DBUG_ASSERT(size <= FULL_PAGE_SIZE(bitmap->block_size)); + + for (; data < end; data += 6) + { + ulonglong bits= uint6korr(data); /* 6 bytes = 6*8/3= 16 patterns */ + uint i; + + /* + Skip common patterns + We can skip empty pages (if we already found a match) or + anything matching the following pattern as this will be either + a full page or a tail page + */ + if ((!bits && best_data) || + ((bits & LL(04444444444444444)) == LL(04444444444444444))) + continue; + for (i= 0; i < 16 ; i++, bits >>= 3) + { + uint pattern= bits & 7; + if (pattern <= min_bits) + { + if (pattern == min_bits) + { + /* Found perfect match */ + best_bits= min_bits; + best_data= data; + best_pos= i; + goto found; + } + if ((int) pattern > (int) best_bits) + { + best_bits= pattern; + best_data= data; + best_pos= i; + } + } + } + } + if (!best_data) + { + if (bitmap->used_size == bitmap->total_size) + DBUG_RETURN(1); + /* Allocate data at end of bitmap */ + bitmap->used_size+= 6; + best_data= data; + best_pos= best_bits= 0; + } + +found: + fill_block(bitmap, block, best_data, best_pos, best_bits, FULL_HEAD_PAGE); + DBUG_RETURN(0); +} + + +/* + Allocate data for tail block + + SYNOPSIS + allocate_tail() + bitmap bitmap + size Size of block we need to find + block Store found information here + + RETURN + 0 ok (block is updated) + 1 error (no space in bitmap; block is not touched) +*/ + + +static my_bool allocate_tail(MARIA_FILE_BITMAP *bitmap, uint size, + MARIA_BITMAP_BLOCK *block) +{ + uint min_bits= size_to_tail_pattern(bitmap, size); + uchar *data= bitmap->map, *end= data + bitmap->used_size; + uchar *best_data= 0; + uint best_bits= (uint) -1, best_pos; + DBUG_ENTER("allocate_tail"); + DBUG_PRINT("enter", ("size: %u", size)); + + LINT_INIT(best_pos); + DBUG_ASSERT(size <= FULL_PAGE_SIZE(bitmap->block_size)); + + for (; data < end; data += 6) + { + ulonglong bits= uint6korr(data); /* 6 bytes = 6*8/3= 16 patterns */ + uint i; + + /* + Skip common patterns + We can skip empty pages (if we already found a match) or + the following patterns: 1-4 or 7 + */ + + if ((!bits && best_data) || bits == LL(0xffffffffffff)) + continue; + for (i= 0; i < 16; i++, bits >>= 3) + { + uint pattern= bits & 7; + if (pattern <= min_bits && (!pattern || pattern >= 5)) + { + if (pattern == min_bits) + { + best_bits= min_bits; + best_data= data; + best_pos= i; + goto found; + } + if ((int) pattern > (int) best_bits) + { + best_bits= pattern; + best_data= data; + best_pos= i; + } + } + } + } + if (!best_data) + { + if (bitmap->used_size == bitmap->total_size) + DBUG_RETURN(1); + /* Allocate data at end of bitmap */ + bitmap->used_size+= 6; + best_pos= best_bits= 0; + } + +found: + fill_block(bitmap, block, best_data, best_pos, best_bits, FULL_TAIL_PAGE); + DBUG_RETURN(0); +} + + +/* + Allocate data for full blocks + + SYNOPSIS + allocate_full_pages() + bitmap bitmap + pages_needed Total size in pages (bitmap->total_size) we would like to have + block Store found information here + full_page 1 if we are not allowed to split extent + + IMPLEMENTATION + We will return the smallest area >= size. If there is no such + block, we will return the biggest area that satisfies + area_size >= min(BLOB_SEGMENT_MIN_SIZE*full_page_size, size) + + To speed up searches, we will only consider areas that has at least 16 free + pages starting on an even boundary. When finding such an area, we will + extend it with all previous and following free pages. This will ensure + we don't get holes between areas + + RETURN + # Blocks used + 0 error (no space in bitmap; block is not touched) +*/ + +static ulong allocate_full_pages(MARIA_FILE_BITMAP *bitmap, + ulong pages_needed, + MARIA_BITMAP_BLOCK *block, my_bool full_page) +{ + uchar *data= bitmap->map, *data_end= data + bitmap->used_size; + uchar *page_end= data + bitmap->total_size; + uchar *best_data= 0; + uint min_size; + uint best_area_size, best_prefix_area_size, best_suffix_area_size; + uint page, size; + ulonglong best_prefix_bits; + DBUG_ENTER("allocate_full_pages"); + DBUG_PRINT("enter", ("pages_needed: %lu", pages_needed)); + + /* Following variables are only used if best_data is set */ + LINT_INIT(best_prefix_bits); + LINT_INIT(best_prefix_area_size); + LINT_INIT(best_suffix_area_size); + + min_size= pages_needed; + if (!full_page && min_size > BLOB_SEGMENT_MIN_SIZE) + min_size= BLOB_SEGMENT_MIN_SIZE; + best_area_size= ~(uint) 0; + + for (; data < page_end; data+= 6) + { + ulonglong bits= uint6korr(data); /* 6 bytes = 6*8/3= 16 patterns */ + uchar *data_start; + ulonglong prefix_bits= 0; + uint area_size, prefix_area_size, suffix_area_size; + + /* Find area with at least 16 free pages */ + if (bits) + continue; + data_start= data; + /* Find size of area */ + for (data+=6 ; data < data_end ; data+= 6) + { + if ((bits= uint6korr(data))) + break; + } + area_size= (data - data_start) / 6 * 16; + if (area_size >= best_area_size) + continue; + prefix_area_size= suffix_area_size= 0; + if (!bits) + { + /* + End of page; All the rest of the bits on page are part of area + This is needed because bitmap->used_size only covers the set bits + in the bitmap. + */ + area_size+= (page_end - data) / 6 * 16; + if (area_size >= best_area_size) + break; + data= page_end; + } + else + { + /* Add bits at end of page */ + for (; !(bits & 7); bits >>= 3) + suffix_area_size++; + area_size+= suffix_area_size; + } + if (data_start != bitmap->map) + { + /* Add bits before page */ + bits= prefix_bits= uint6korr(data_start - 6); + DBUG_ASSERT(bits != 0); + /* 111 000 000 000 000 000 000 000 000 000 000 000 000 000 000 000 */ + if (!(bits & LL(07000000000000000))) + { + data_start-= 6; + do + { + prefix_area_size++; + bits<<= 3; + } while (!(bits & LL(07000000000000000))); + area_size+= prefix_area_size; + /* Calculate offset to page from data_start */ + prefix_area_size= 16 - prefix_area_size; + } + } + if (area_size >= min_size && area_size <= best_area_size) + { + best_data= data_start; + best_area_size= area_size; + best_prefix_bits= prefix_bits; + best_prefix_area_size= prefix_area_size; + best_suffix_area_size= suffix_area_size; + + /* Prefer to put data in biggest possible area */ + if (area_size <= pages_needed) + min_size= area_size; + else + min_size= pages_needed; + } + } + if (!best_data) + DBUG_RETURN(0); /* No room on page */ + + /* + Now allocate min(pages_needed, area_size), starting from + best_start + best_prefix_area_size + */ + if (best_area_size > pages_needed) + best_area_size= pages_needed; + + /* For each 6 bytes we have 6*8/3= 16 patterns */ + page= ((best_data - bitmap->map) * 8) / 3 + best_prefix_area_size; + block->page= bitmap->page + 1 + page; + block->page_count= best_area_size; + block->empty_space= 0; + block->sub_blocks= 1; + block->org_bitmap_value= 0; + block->used= 0; + DBUG_PRINT("info", ("page: %lu page_count: %u", + (ulong) block->page, block->page_count)); + + if (best_prefix_area_size) + { + ulonglong tmp; + /* Convert offset back to bits */ + best_prefix_area_size= 16 - best_prefix_area_size; + if (best_area_size < best_prefix_area_size) + { + tmp= (LL(1) << best_area_size*3) - 1; + best_area_size= best_prefix_area_size; /* for easy end test */ + } + else + tmp= (LL(1) << best_prefix_area_size*3) - 1; + tmp<<= (16 - best_prefix_area_size) * 3; + DBUG_ASSERT((best_prefix_bits & tmp) == 0); + best_prefix_bits|= tmp; + int6store(best_data, best_prefix_bits); + if (!(best_area_size-= best_prefix_area_size)) + { + DBUG_EXECUTE("bitmap", _ma_print_bitmap(bitmap);); + DBUG_RETURN(block->page_count); + } + best_data+= 6; + } + best_area_size*= 3; /* Bits to set */ + size= best_area_size/8; /* Bytes to set */ + bfill(best_data, size, 255); + best_data+= size; + if ((best_area_size-= size * 8)) + { + /* fill last byte */ + *best_data|= (uchar) ((1 << best_area_size) -1); + best_data++; + } + if (data_end < best_data) + bitmap->used_size= (uint) (best_data - bitmap->map); + bitmap->changed= 1; + DBUG_EXECUTE("bitmap", _ma_print_bitmap(bitmap);); + DBUG_RETURN(block->page_count); +} + + +/**************************************************************************** + Find right bitmaps where to store data +****************************************************************************/ + +/* + Find right bitmap and position for head block + + RETURN + 0 ok + 1 error +*/ + +static my_bool find_head(MARIA_HA *info, uint length, uint position) +{ + MARIA_FILE_BITMAP *bitmap= &info->s->bitmap; + MARIA_BITMAP_BLOCK *block; + /* There is always place for head blocks in bitmap_blocks */ + block= dynamic_element(&info->bitmap_blocks, position, MARIA_BITMAP_BLOCK *); + + while (allocate_head(bitmap, length, block)) + if (move_to_next_bitmap(info, bitmap)) + return 1; + return 0; +} + + +/* + Find right bitmap and position for tail + + RETURN + 0 ok + 1 error +*/ + +static my_bool find_tail(MARIA_HA *info, uint length, uint position) +{ + MARIA_FILE_BITMAP *bitmap= &info->s->bitmap; + MARIA_BITMAP_BLOCK *block; + DBUG_ENTER("find_tail"); + + /* Needed, as there is no error checking in dynamic_element */ + if (allocate_dynamic(&info->bitmap_blocks, position)) + DBUG_RETURN(1); + block= dynamic_element(&info->bitmap_blocks, position, MARIA_BITMAP_BLOCK *); + + while (allocate_tail(bitmap, length, block)) + if (move_to_next_bitmap(info, bitmap)) + DBUG_RETURN(1); + DBUG_RETURN(0); +} + + +/* + Find right bitmap and position for full blocks in one extent + + NOTES + This is used to allocate the main extent after the 'head' block + + RETURN + 0 ok + 1 error +*/ + +static my_bool find_mid(MARIA_HA *info, ulong pages, uint position) +{ + MARIA_FILE_BITMAP *bitmap= &info->s->bitmap; + MARIA_BITMAP_BLOCK *block; + block= dynamic_element(&info->bitmap_blocks, position, MARIA_BITMAP_BLOCK *); + + while (allocate_full_pages(bitmap, pages, block, 1)) + { + if (move_to_next_bitmap(info, bitmap)) + return 1; + } + return 0; +} + + +/* + Find right bitmap and position for putting a blob + + NOTES + The extents are stored last in info->bitmap_blocks + + IMPLEMENTATION + Allocate all full pages for the block + optionally one tail + + RETURN + 0 ok + 1 error +*/ + +static my_bool find_blob(MARIA_HA *info, ulong length) +{ + MARIA_FILE_BITMAP *bitmap= &info->s->bitmap; + uint full_page_size= FULL_PAGE_SIZE(info->s->block_size); + ulong pages; + uint rest_length, used; + uint first_block_pos; + MARIA_BITMAP_BLOCK *first_block= 0; + DBUG_ENTER("find_blob"); + DBUG_PRINT("enter", ("length: %lu", length)); + + pages= length / full_page_size; + rest_length= (uint) (length - pages * full_page_size); + if (rest_length >= MAX_TAIL_SIZE(info->s->block_size)) + { + pages++; + rest_length= 0; + } + + if (pages) + { + MARIA_BITMAP_BLOCK *block; + if (allocate_dynamic(&info->bitmap_blocks, + info->bitmap_blocks.elements + + pages / BLOB_SEGMENT_MIN_SIZE + 2)) + DBUG_RETURN(1); + first_block_pos= info->bitmap_blocks.elements; + block= dynamic_element(&info->bitmap_blocks, info->bitmap_blocks.elements, + MARIA_BITMAP_BLOCK*); + first_block= block; + do + { + used= allocate_full_pages(bitmap, + (pages >= 65535 ? 65535 : (uint) pages), block, + 0); + if (!used && move_to_next_bitmap(info, bitmap)) + DBUG_RETURN(1); + info->bitmap_blocks.elements++; + block++; + } while ((pages-= used) != 0); + } + if (rest_length && find_tail(info, rest_length, + info->bitmap_blocks.elements++)) + DBUG_RETURN(1); + if (first_block) + first_block->sub_blocks= info->bitmap_blocks.elements - first_block_pos; + DBUG_RETURN(0); +} + + +static my_bool allocate_blobs(MARIA_HA *info, MARIA_ROW *row) +{ + ulong *length, *end; + uint elements; + /* + Reserve size for: + head block + one extent + tail block + */ + elements= info->bitmap_blocks.elements; + for (length= row->blob_lengths, end= length + info->s->base.blobs; + length < end; length++) + { + if (*length && find_blob(info, *length)) + return 1; + } + row->extents_count= (info->bitmap_blocks.elements - elements); + return 0; +} + + +static void use_head(MARIA_HA *info, ulonglong page, uint size, + uint block_position) +{ + MARIA_FILE_BITMAP *bitmap= &info->s->bitmap; + MARIA_BITMAP_BLOCK *block; + uchar *data; + uint offset, tmp, offset_page; + + block= dynamic_element(&info->bitmap_blocks, block_position, + MARIA_BITMAP_BLOCK*); + block->page= page; + block->page_count= 1 + TAIL_BIT; + block->empty_space= size; + block->sub_blocks= 1; + block->used= BLOCKUSED_TAIL; + + /* + Mark place used by reading/writing 2 bytes at a time to handle + bitmaps in overlapping bytes + */ + offset_page= (uint) (page - bitmap->page - 1) * 3; + offset= offset_page & 7; + data= bitmap->map + offset_page / 8; + tmp= uint2korr(data); + block->org_bitmap_value= (tmp >> offset) & 7; + tmp= (tmp & ~(7 << offset)) | (FULL_HEAD_PAGE << offset); + int2store(data, tmp); + bitmap->changed= 1; + DBUG_EXECUTE("bitmap", _ma_print_bitmap(bitmap);); +} + + +/* + Find out where to split the row; +*/ + +static uint find_where_to_split_row(MARIA_SHARE *share, MARIA_ROW *row, + uint extents_length, uint split_size) +{ + uint row_length= row->base_length; + uint *lengths, *lengths_end; + + DBUG_ASSERT(row_length < split_size); + /* + Store first in all_field_lengths the different parts that are written + to the row. This needs to be in same order as in + ma_block_rec.c::write_block_record() + */ + row->null_field_lengths[-3]= extents_length; + row->null_field_lengths[-2]= share->base.fixed_not_null_fields_length; + row->null_field_lengths[-1]= row->field_lengths_length; + for (lengths= row->null_field_lengths - EXTRA_LENGTH_FIELDS, + lengths_end= (lengths + share->base.pack_fields - share->base.blobs + + EXTRA_LENGTH_FIELDS); lengths < lengths_end; lengths++) + { + if (row_length + *lengths > split_size) + break; + row_length+= *lengths; + } + return row_length; +} + + +static my_bool write_rest_of_head(MARIA_HA *info, uint position, + ulong rest_length) +{ + MARIA_SHARE *share= info->s; + uint full_page_size= FULL_PAGE_SIZE(share->block_size); + MARIA_BITMAP_BLOCK *block; + + if (position == 0) + { + /* Write out full pages */ + uint pages= rest_length / full_page_size; + + rest_length%= full_page_size; + if (rest_length >= MAX_TAIL_SIZE(share->block_size)) + { + /* Put tail on a full page */ + pages++; + rest_length= 0; + } + if (find_mid(info, rest_length / full_page_size, 1)) + return 1; + /* + Insert empty block after full pages, to allow write_block_record() to + split segment into used + free page + */ + block= dynamic_element(&info->bitmap_blocks, 2, MARIA_BITMAP_BLOCK*); + block->page_count= 0; + block->used= 0; + } + if (rest_length) + { + if (find_tail(info, rest_length, ELEMENTS_RESERVED_FOR_MAIN_PART - 1)) + return 1; + } + else + { + /* Empty tail block */ + block= dynamic_element(&info->bitmap_blocks, + ELEMENTS_RESERVED_FOR_MAIN_PART - 1, + MARIA_BITMAP_BLOCK *); + block->page_count= 0; + block->used= 0; + } + return 0; +} + + +/* + Find where to store one row + + SYNPOSIS + _ma_bitmap_find_place() + info Maria handler + row Information about row to write + blocks Store data about allocated places here + + RETURN + 0 ok + 1 error +*/ + +my_bool _ma_bitmap_find_place(MARIA_HA *info, MARIA_ROW *row, + MARIA_BITMAP_BLOCKS *blocks) +{ + MARIA_SHARE *share= info->s; + my_bool res= 1; + uint full_page_size, position, max_page_size; + uint head_length, row_length, rest_length, extents_length; + DBUG_ENTER("_ma_bitmap_find_place"); + + blocks->count= 0; + blocks->tail_page_skipped= blocks->page_skipped= 0; + row->extents_count= 0; + /* + Reserver place for the following blocks: + - Head block + - Full page block + - Marker block to allow write_block_record() to split full page blocks + into full and free part + - Tail block + */ + + info->bitmap_blocks.elements= ELEMENTS_RESERVED_FOR_MAIN_PART; + max_page_size= (share->block_size - PAGE_OVERHEAD_SIZE); + + pthread_mutex_lock(&share->bitmap.bitmap_lock); + + if (row->total_length <= max_page_size) + { + /* Row fits in one page */ + position= ELEMENTS_RESERVED_FOR_MAIN_PART - 1; + if (find_head(info, (uint) row->total_length, position)) + goto abort; + goto end; + } + + /* + First allocate all blobs (so that we can find out the needed size for + the main block. + */ + if (row->blob_length && allocate_blobs(info, row)) + goto abort; + + extents_length= row->extents_count * ROW_EXTENT_SIZE; + if ((head_length= (row->head_length + extents_length)) <= max_page_size) + { + /* Main row part fits into one page */ + position= ELEMENTS_RESERVED_FOR_MAIN_PART - 1; + if (find_head(info, head_length, position)) + goto abort; + goto end; + } + + /* Allocate enough space */ + head_length+= ELEMENTS_RESERVED_FOR_MAIN_PART * ROW_EXTENT_SIZE; + + /* The first segment size is stored in 'row_length' */ + row_length= find_where_to_split_row(share, row, extents_length, + max_page_size); + + full_page_size= FULL_PAGE_SIZE(share->block_size); + position= 0; + if (head_length - row_length <= full_page_size) + position= ELEMENTS_RESERVED_FOR_MAIN_PART -2; /* Only head and tail */ + if (find_head(info, row_length, position)) + goto abort; + rest_length= head_length - row_length; + if (write_rest_of_head(info, position, rest_length)) + goto abort; + +end: + blocks->block= dynamic_element(&info->bitmap_blocks, position, + MARIA_BITMAP_BLOCK*); + blocks->block->sub_blocks= ELEMENTS_RESERVED_FOR_MAIN_PART - position; + /* First block's page_count is for all blocks */ + blocks->count= info->bitmap_blocks.elements - position; + res= 0; + +abort: + pthread_mutex_unlock(&share->bitmap.bitmap_lock); + DBUG_RETURN(res); +} + + +/* + Find where to put row on update (when head page is already defined) + + SYNPOSIS + _ma_bitmap_find_new_place() + info Maria handler + row Information about row to write + page On which page original row was stored + free_size Free size on head page + blocks Store data about allocated places here + + NOTES + This function is only called when the new row can't fit in the space of + the old row in the head page. + + This is essently same as _ma_bitmap_find_place() except that + we don't call find_head() to search in bitmaps where to put the page. + + RETURN + 0 ok + 1 error +*/ + +my_bool _ma_bitmap_find_new_place(MARIA_HA *info, MARIA_ROW *row, + ulonglong page, uint free_size, + MARIA_BITMAP_BLOCKS *blocks) +{ + MARIA_SHARE *share= info->s; + my_bool res= 1; + uint full_page_size, position; + uint head_length, row_length, rest_length, extents_length; + DBUG_ENTER("_ma_bitmap_find_new_place"); + + blocks->count= 0; + blocks->tail_page_skipped= blocks->page_skipped= 0; + row->extents_count= 0; + info->bitmap_blocks.elements= ELEMENTS_RESERVED_FOR_MAIN_PART; + + pthread_mutex_lock(&share->bitmap.bitmap_lock); + if (share->bitmap.page != page / share->bitmap.pages_covered && + _ma_change_bitmap_page(info, &share->bitmap, + page / share->bitmap.pages_covered)) + goto abort; + + /* + First allocate all blobs (so that we can find out the needed size for + the main block. + */ + if (row->blob_length && allocate_blobs(info, row)) + goto abort; + + extents_length= row->extents_count * ROW_EXTENT_SIZE; + if ((head_length= (row->head_length + extents_length)) <= free_size) + { + /* Main row part fits into one page */ + position= ELEMENTS_RESERVED_FOR_MAIN_PART - 1; + use_head(info, page, head_length, position); + goto end; + } + + /* Allocate enough space */ + head_length+= ELEMENTS_RESERVED_FOR_MAIN_PART * ROW_EXTENT_SIZE; + + /* The first segment size is stored in 'row_length' */ + row_length= find_where_to_split_row(share, row, extents_length, free_size); + + full_page_size= FULL_PAGE_SIZE(share->block_size); + position= 0; + if (head_length - row_length <= full_page_size) + position= ELEMENTS_RESERVED_FOR_MAIN_PART -2; /* Only head and tail */ + use_head(info, page, row_length, position); + rest_length= head_length - row_length; + + if (write_rest_of_head(info, position, rest_length)) + goto abort; + +end: + blocks->block= dynamic_element(&info->bitmap_blocks, position, + MARIA_BITMAP_BLOCK*); + blocks->block->sub_blocks= ELEMENTS_RESERVED_FOR_MAIN_PART - position; + /* First block's page_count is for all blocks */ + blocks->count= info->bitmap_blocks.elements - position; + res= 0; + +abort: + pthread_mutex_unlock(&share->bitmap.bitmap_lock); + DBUG_RETURN(res); +} + + +/**************************************************************************** + Clear and reset bits +****************************************************************************/ + +static my_bool set_page_bits(MARIA_HA *info, MARIA_FILE_BITMAP *bitmap, + ulonglong page, uint fill_pattern) +{ + ulonglong bitmap_page; + uint offset_page, offset, tmp, org_tmp; + uchar *data; + DBUG_ENTER("set_page_bits"); + + bitmap_page= page / bitmap->pages_covered; + if (bitmap_page != bitmap->page && + _ma_change_bitmap_page(info, bitmap, bitmap_page)) + DBUG_RETURN(1); + + /* Find page number from start of bitmap */ + offset_page= page - bitmap->page - 1; + /* + Mark place used by reading/writing 2 bytes at a time to handle + bitmaps in overlapping bytes + */ + offset_page*= 3; + offset= offset_page & 7; + data= bitmap->map + offset_page / 8; + org_tmp= tmp= uint2korr(data); + tmp= (tmp & ~(7 << offset)) | (fill_pattern << offset); + if (tmp == org_tmp) + DBUG_RETURN(0); /* No changes */ + int2store(data, tmp); + + bitmap->changed= 1; + DBUG_EXECUTE("bitmap", _ma_print_bitmap(bitmap);); + if (fill_pattern != 3 && fill_pattern != 7 && + page < info->s->state.first_bitmap_with_space) + info->s->state.first_bitmap_with_space= page; + DBUG_RETURN(0); +} + + +/* + Get bitmap pattern for a given page + + SYNOPSIS + + get_page_bits() + info Maria handler + bitmap Bitmap handler + page Page number + + RETURN + 0-7 Bitmap pattern + ~0 Error (couldn't read page) +*/ + +static uint get_page_bits(MARIA_HA *info, MARIA_FILE_BITMAP *bitmap, + ulonglong page) +{ + ulonglong bitmap_page; + uint offset_page, offset, tmp; + uchar *data; + DBUG_ENTER("get_page_bits"); + + bitmap_page= page / bitmap->pages_covered; + if (bitmap_page != bitmap->page && + _ma_change_bitmap_page(info, bitmap, bitmap_page)) + DBUG_RETURN(~ (uint) 0); + + /* Find page number from start of bitmap */ + offset_page= page - bitmap->page - 1; + /* + Mark place used by reading/writing 2 bytes at a time to handle + bitmaps in overlapping bytes + */ + offset_page*= 3; + offset= offset_page & 7; + data= bitmap->map + offset_page / 8; + tmp= uint2korr(data); + DBUG_RETURN((tmp >> offset) & 7); +} + + +/* + Mark all pages in a region as free + + SYNOPSIS + reset_full_page_bits() + info Maria handler + bitmap Bitmap handler + page Start page + page_count Number of pages + + NOTES + We assume that all pages in region is covered by same bitmap + One must have a lock on info->s->bitmap.bitmap_lock + + RETURN + 0 ok + 1 Error (when reading bitmap) +*/ + +my_bool _ma_reset_full_page_bits(MARIA_HA *info, MARIA_FILE_BITMAP *bitmap, + ulonglong page, uint page_count) +{ + ulonglong bitmap_page; + uint offset, bit_start, bit_count, tmp; + uchar *data; + DBUG_ENTER("_ma_reset_full_page_bits"); + DBUG_PRINT("enter", ("page: %lu page_count: %u", (ulong) page, page_count)); + safe_mutex_assert_owner(&info->s->bitmap.bitmap_lock); + + bitmap_page= page / bitmap->pages_covered; + if (bitmap_page != bitmap->page && + _ma_change_bitmap_page(info, bitmap, bitmap_page)) + DBUG_RETURN(1); + + /* Find page number from start of bitmap */ + page= page - bitmap->page - 1; + + /* Clear bits from 'page * 3' -> '(page + page_count) * 3' */ + bit_start= page * 3; + bit_count= page_count * 3; + + data= bitmap->map + bit_start / 8; + offset= bit_start & 7; + + tmp= (255 << offset); /* Bits to keep */ + if (bit_count + offset < 8) + { + /* Only clear bits between 'offset' and 'offset+bit_count-1' */ + tmp^= (255 << (offset + bit_count)); + } + *data&= ~tmp; + + if ((int) (bit_count-= (8 - offset)) > 0) + { + uint fill; + data++; + /* + -1 is here to avoid one 'if' statement and to let the following code + handle the last byte + */ + if ((fill= (bit_count - 1) / 8)) + { + bzero(data, fill); + data+= fill; + } + bit_count-= fill * 8; /* Bits left to clear */ + tmp= (1 << bit_count) - 1; + *data&= ~tmp; + } + if (bitmap->page < (ulonglong) info->s->state.first_bitmap_with_space) + info->s->state.first_bitmap_with_space= bitmap->page; + bitmap->changed= 1; + DBUG_EXECUTE("bitmap", _ma_print_bitmap(bitmap);); + DBUG_RETURN(0); +} + + +/* + Correct bitmap pages to reflect the true allocation + + SYNOPSIS + _ma_bitmap_release_unused() + info Maria handle + blocks Bitmap blocks + + IMPLEMENTATION + If block->used & BLOCKUSED_TAIL is set: + If block->used & BLOCKUSED_USED is set, then the bits for the + corresponding page is set according to block->empty_space + If block->used & BLOCKUSED_USED is not set, then the bits for + the corresponding page is set to org_bitmap_value; + + If block->used & BLOCKUSED_TAIL is not set: + if block->used is not set, the bits for the corresponding page are + cleared + + For the first block (head block) the logic is same as for a tail block + + RETURN + 0 ok + 1 error (Couldn't write or read bitmap page) +*/ + +my_bool _ma_bitmap_release_unused(MARIA_HA *info, MARIA_BITMAP_BLOCKS *blocks) +{ + MARIA_BITMAP_BLOCK *block= blocks->block, *end= block + blocks->count; + MARIA_FILE_BITMAP *bitmap= &info->s->bitmap; + uint bits, current_bitmap_value; + DBUG_ENTER("_ma_bitmap_release_unused"); + + /* + We can skip FULL_HEAD_PAGE (4) as the page was marked as 'full' + when we allocated space in the page + */ + current_bitmap_value= FULL_HEAD_PAGE; + + pthread_mutex_lock(&info->s->bitmap.bitmap_lock); + + /* First handle head block */ + if (block->used & BLOCKUSED_USED) + { + DBUG_PRINT("info", ("head empty_space: %u", block->empty_space)); + bits= _ma_free_size_to_head_pattern(bitmap, block->empty_space); + if (block->used & BLOCKUSED_USE_ORG_BITMAP) + current_bitmap_value= block->org_bitmap_value; + } + else + bits= block->org_bitmap_value; + if (bits != current_bitmap_value && + set_page_bits(info, bitmap, block->page, bits)) + goto err; + + + /* Handle all full pages and tail pages (for head page and blob) */ + for (block++; block < end; block++) + { + if (block->used & BLOCKUSED_TAIL) + { + if (block->used & BLOCKUSED_USED) + { + DBUG_PRINT("info", ("tail empty_space: %u", block->empty_space)); + bits= free_size_to_tail_pattern(bitmap, block->empty_space); + } + else + bits= block->org_bitmap_value; + if (bits != FULL_TAIL_PAGE && + set_page_bits(info, bitmap, block->page, bits)) + goto err; + } + if (!(block->used & BLOCKUSED_USED) && + _ma_reset_full_page_bits(info, bitmap, + block->page, block->page_count)) + goto err; + } + pthread_mutex_unlock(&info->s->bitmap.bitmap_lock); + DBUG_RETURN(0); + +err: + pthread_mutex_unlock(&info->s->bitmap.bitmap_lock); + DBUG_RETURN(1); +} + + +/* + Free full pages from bitmap + + SYNOPSIS + _ma_bitmap_free_full_pages() + info Maria handle + extents Extents (as stored on disk) + count Number of extents + + IMPLEMENTATION + Mark all full pages (not tails) from extents as free + + RETURN + 0 ok + 1 error (Couldn't write or read bitmap page) +*/ + +my_bool _ma_bitmap_free_full_pages(MARIA_HA *info, const byte *extents, + uint count) +{ + DBUG_ENTER("_ma_bitmap_free_full_pages"); + + pthread_mutex_lock(&info->s->bitmap.bitmap_lock); + for (; count--; extents += ROW_EXTENT_SIZE) + { + ulonglong page= uint5korr(extents); + uint page_count= uint2korr(extents + ROW_EXTENT_PAGE_SIZE); + if (!(page_count & TAIL_BIT)) + { + if (_ma_reset_full_page_bits(info, &info->s->bitmap, page, page_count)) + { + pthread_mutex_unlock(&info->s->bitmap.bitmap_lock); + DBUG_RETURN(1); + } + } + } + pthread_mutex_unlock(&info->s->bitmap.bitmap_lock); + DBUG_RETURN(0); +} + + + +my_bool _ma_bitmap_set(MARIA_HA *info, ulonglong pos, my_bool head, + uint empty_space) +{ + MARIA_FILE_BITMAP *bitmap= &info->s->bitmap; + uint bits; + my_bool res; + DBUG_ENTER("_ma_bitmap_set"); + + pthread_mutex_lock(&info->s->bitmap.bitmap_lock); + bits= (head ? + _ma_free_size_to_head_pattern(bitmap, empty_space) : + free_size_to_tail_pattern(bitmap, empty_space)); + res= set_page_bits(info, bitmap, pos, bits); + pthread_mutex_unlock(&info->s->bitmap.bitmap_lock); + DBUG_RETURN(res); +} + + +/* + Check that bitmap pattern is correct for a page + + NOTES + Used in maria_chk + + RETURN + 0 ok + 1 error +*/ + +my_bool _ma_check_bitmap_data(MARIA_HA *info, + enum en_page_type page_type, ulonglong page, + uint empty_space, uint *bitmap_pattern) +{ + uint bits; + switch (page_type) { + case UNALLOCATED_PAGE: + case MAX_PAGE_TYPE: + bits= 0; + break; + case HEAD_PAGE: + bits= _ma_free_size_to_head_pattern(&info->s->bitmap, empty_space); + break; + case TAIL_PAGE: + bits= free_size_to_tail_pattern(&info->s->bitmap, empty_space); + break; + case BLOB_PAGE: + bits= FULL_TAIL_PAGE; + break; + } + return (*bitmap_pattern= get_page_bits(info, &info->s->bitmap, page)) != + bits; +} + + +/* + Check that bitmap pattern is correct for a page + + NOTES + Used in maria_chk + + RETURN + 0 ok + 1 error +*/ + +my_bool _ma_check_if_right_bitmap_type(MARIA_HA *info, + enum en_page_type page_type, + ulonglong page, + uint *bitmap_pattern) +{ + if ((*bitmap_pattern= get_page_bits(info, &info->s->bitmap, page)) > 7) + return 1; /* Couldn't read page */ + switch (page_type) { + case HEAD_PAGE: + return *bitmap_pattern < 1 || *bitmap_pattern > 4; + case TAIL_PAGE: + return *bitmap_pattern < 5; + case BLOB_PAGE: + return *bitmap_pattern != 7; + default: + break; + } + DBUG_ASSERT(0); + return 1; +} diff --git a/storage/maria/ma_blockrec.c b/storage/maria/ma_blockrec.c new file mode 100644 index 00000000000..6eac151d76e --- /dev/null +++ b/storage/maria/ma_blockrec.c @@ -0,0 +1,2742 @@ +/* Copyright (C) 2007 Michael Widenius + + 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; either version 2 of the License, or + (at your option) any later version. + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ + +/* + Storage of records in block + + Maria will have a LSN at start of each page (including the bitmap page) + Maria will for each row have the additional information: + + TRANSID Transaction ID that last updated row (6 bytes) + VER_PTR Version pointer that points on the UNDO entry that + contains last version of the row versions (7 bytes) + + The different page types that are in a data file are: + + Bitmap pages Map of free pages in the next extent (8129 page size + gives us 256M of mapped pages / bitmap) + Head page Start of rows are stored on this page. + A rowid always points to a head page + Blob page This page is totally filled with data from one blob or by + a set of long VARCHAR/CHAR fields + Tail page This contains the last part from different rows, blobs + or varchar fields. + + The data file starts with a bitmap page, followed by as many data + pages as the bitmap can cover. After this there is a new bitmap page + and more data pages etc. + + For information about the bitmap page, see ma_bitmap.c + + Structure of data and tail page: + + The page has a row directory at end of page to allow us to do deletes + without having to reorganize the page. It also allows us to store some + extra bytes after each row to allow them to grow without having to move + around other rows + + Page header: + + LSN 7 bytes Log position for last page change + PAGE_TYPE 1 byte 1 for head / 2 for tail / 3 for blob + NO 1 byte Number of row/tail entries on page + empty space 2 bytes Empty space on page + + The upmost bit in PAGE_TYPE is set to 1 if the data on the page + can be compacted to get more space. (PAGE_CAN_BE_COMPACTED) + + Row data + + Row directory of NO entires, that consist of the following for each row + (in reverse order; ie, first record is stored last): + + Position 2 bytes Position of row on page + Length 2 bytes Length of entry + + For Position and Length, the 1 upmost bit of the position and the 1 + upmost bit of the length could be used for some states of the row (in + other words, we should try to keep these reserved) + + eof flag 1 byte Reserved for full page read testing + + ---------------- + + Structure of blob pages: + + LSN 7 bytes Log position for last page change + PAGE_TYPE 1 byte 3 + + data + + ----------------- + + Row data structure: + + Flag 1 byte Marker of which header field exists + TRANSID 6 bytes TRANSID of changing transaction + (optional, added on insert and first + update/delete) + VER_PTR 7 bytes Pointer to older version in log + (undo record) + (optional, added after first + update/delete) + DELETE_TRANSID 6 bytes (optional). TRANSID of original row. + Added on delete. + Nulls_extended 1 byte To allow us to add new DEFAULT NULL + fields (optional, added after first + change of row after alter table) + Number of ROW_EXTENT's 1-3 byte Length encoded, optional + This is the number of extents the + row is split into + First row_extent 7 byte Pointer to first row extent (optional) + + Total length of length array 1-3 byte Only used if we have + char/varchar/blob fields. + Row checksum 1 byte Only if table created with checksums + Null_bits .. One bit for each NULL field + Empty_bits .. One bit for each NOT NULL field. This bit is + 0 if the value is 0 or empty string. + + field_offsets 2 byte/offset + For each 32 field, there is one offset that + points to where the field information starts + in the block. This is to provide fast access + to later field in the row when we only need + to return a small set of fields. + + Things marked above as 'optional' will only be present if the corresponding + bit is set in 'Flag' field. + + Data in the following order: + (Field order is precalculated when table is created) + + Critical fixed length, not null, fields. (Note, these can't be dropped) + Fixed length, null fields + + Length array, 1-4 byte per field for all CHAR/VARCHAR/BLOB fields. + Number of bytes used in length array per entry is depending on max length + for field. + + ROW_EXTENT's + CHAR data (space stripped) + VARCHAR data + BLOB data + + Fields marked in null_bits or empty_bits are not stored in data part or + length array. + + If row doesn't fit into the given block, then the first EXTENT will be + stored last on the row. This is done so that we don't break any field + data in the middle. + + We first try to store the full row into one block. If that's not possible + we move out each big blob into their own extents. If this is not enough we + move out a concatenation of all varchars to their own extent. + + Each blob and the concatenated char/varchar fields are stored the following + way: + - Store the parts in as many full-contiguous pages as possible. + - The last part, that doesn't fill a full page, is stored in tail page. + + When doing an insert of a new row, we don't have to have + VER_PTR in the row. This will make rows that are not changed stored + efficiently. On update and delete we would add TRANSID (if it was an old + committed row) and VER_PTR to + the row. On row page compaction we can easily detect rows where + TRANSID was committed before the the longest running transaction + started and we can then delete TRANSID and VER_PTR from the row to + gain more space. + + If a row is deleted in Maria, we change TRANSID to current transid and + change VER_PTR to point to the undo record for the delete. The undo + record must contain the original TRANSID, so that another transaction + can use this to check if they should use the found row or go to the + previous row pointed to by the VER_PTR in the undo row. + + Description of the different parts: + + Flag is coded as: + + Description bit + TRANS_ID_exists 0 + VER_PTR_exists 1 + Row is deleted 2 (Means that DELETE_TRANSID exists) + Nulls_extended_exists 3 + Row is split 7 This means that 'Number_of_row_extents' exists + + + This would be a way to get more space on a page when doing page + compaction as we don't need to store TRANSID that have committed + before the smallest running transaction we have in memory. + + Nulls_extended is the number of new DEFAULT NULL fields in the row + compared to the number of DEFAULT NULL fields when the first version + of the table was created. If Nulls_extended doesn't exist in the row, + we know it's 0 as this must be one of the original rows from when the + table was created first time. This coding allows us to add 255*8 = + 2048 new fields without requiring a full alter table. + + Empty_bits is used to allow us to store 0, 0.0, empty string, empty + varstring and empty blob efficiently. (This is very good for data + warehousing where NULL's are often regarded as evil). Having this + bitmap also allows us to drop information of a field during a future + delete if field was deleted with ALTER TABLE DROP COLUMN. To be able + to handle DROP COLUMN, we must store in the index header the fields + that has been dropped. When unpacking a row we will ignore dropped + fields. When storing a row, we will mark a dropped field either with a + null in the null bit map or in the empty_bits and not store any data + for it. + + One ROW_EXTENT is coded as: + + START_PAGE 5 bytes + PAGE_COUNT 2 bytes. High bit is used to indicate tail page/ + end of blob + With 8K pages, we can cover 256M in one extent. This coding gives us a + maximum file size of 2^40*8192 = 8192 tera + + As an example of ROW_EXTENT handling, assume a row with one integer + field (value 5), two big VARCHAR fields (size 250 and 8192*3), and 2 + big BLOB fields that we have updated. + + The record format for storing this into an empty file would be: + + Page 1: + + 00 00 00 00 00 00 00 LSN + 01 Only one row in page + xx xx Empty space on page + + 10 Flag: row split, VER_PTR exists + 01 00 00 00 00 00 TRANSID 1 + 00 00 00 00 00 01 00 VER_PTR to first block in LOG file 1 + 5 Number of row extents + 02 00 00 00 00 03 00 VARCHAR's are stored in full pages 2,3,4 + 0 No null fields + 0 No empty fields + 05 00 00 00 00 00 80 Tail page for VARCHAR, rowid 0 + 06 00 00 00 00 80 00 First blob, stored at page 6-133 + 05 00 00 00 00 01 80 Tail of first blob (896 bytes) at page 5 + 86 00 00 00 00 80 00 Second blob, stored at page 134-262 + 05 00 00 00 00 02 80 Tail of second blob (896 bytes) at page 5 + 05 00 5 integer + FA Length of first varchar field (size 250) + 00 60 Length of second varchar field (size 8192*3) + 00 60 10 First medium BLOB, 1M + 01 00 10 00 Second BLOB, 1M + xx xx xx xx xx xx Varchars are stored here until end of page + + ..... until end of page + + 09 00 F4 1F 00 (Start position 9, length 8180, end byte) +*/ + +#define SANITY_CHECKS + +#include "maria_def.h" +#include "ma_blockrec.h" + +typedef struct st_maria_extent_cursor +{ + byte *extent; + byte *data_start; /* For error checking */ + MARIA_RECORD_POS *tail_positions; + my_off_t page; + uint extent_count, page_count; + uint tail; /* <> 0 if current extent is a tail page */ + my_bool first_extent; +} MARIA_EXTENT_CURSOR; + + +static my_bool delete_tails(MARIA_HA *info, MARIA_RECORD_POS *tails); +static my_bool delete_head_or_tail(MARIA_HA *info, + ulonglong page, uint record_number, + my_bool head); +static void _ma_print_directory(byte *buff, uint block_size); + +/**************************************************************************** + Initialization +****************************************************************************/ + +/* + Initialize data needed for block structures +*/ + + +/* Size of the different header elements for a row */ + +static uchar header_sizes[]= +{ + TRANSID_SIZE, + VERPTR_SIZE, + TRANSID_SIZE, /* Delete transid */ + 1 /* Null extends */ +}; + +/* + Calculate array of all used headers + + Used to speed up: + + size= 1; + if (flag & 1) + size+= TRANSID_SIZE; + if (flag & 2) + size+= VERPTR_SIZE; + if (flag & 4) + size+= TRANSID_SIZE + if (flag & 8) + size+= 1; + + NOTES + This is called only once at startup of Maria +*/ + +static uchar total_header_size[1 << array_elements(header_sizes)]; +#define PRECALC_HEADER_BITMASK (array_elements(total_header_size) -1) + +void _ma_init_block_record_data(void) +{ + uint i; + bzero(total_header_size, sizeof(total_header_size)); + total_header_size[0]= FLAG_SIZE; /* Flag byte */ + for (i= 1; i < array_elements(total_header_size); i++) + { + uint size= FLAG_SIZE, j, bit; + for (j= 0; (bit= (1 << j)) <= i; j++) + { + if (i & bit) + size+= header_sizes[j]; + } + total_header_size[i]= size; + } +} + + +my_bool _ma_once_init_block_row(MARIA_SHARE *share, File data_file) +{ + + share->base.max_data_file_length= + (((ulonglong) 1 << ((share->base.rec_reflength-1)*8))-1) * + share->block_size; +#if SIZEOF_OFF_T == 4 + set_if_smaller(max_data_file_length, INT_MAX32); +#endif + return _ma_bitmap_init(share, data_file); +} + + +my_bool _ma_once_end_block_row(MARIA_SHARE *share) +{ + int res= _ma_bitmap_end(share); + if (flush_key_blocks(share->key_cache, share->bitmap.file, + share->temporary ? FLUSH_IGNORE_CHANGED : + FLUSH_RELEASE)) + res= 1; + if (share->bitmap.file >= 0 && my_close(share->bitmap.file, MYF(MY_WME))) + res= 1; + return res; +} + + +/* Init info->cur_row structure */ + +my_bool _ma_init_block_row(MARIA_HA *info) +{ + MARIA_ROW *row= &info->cur_row, *new_row= &info->new_row; + DBUG_ENTER("_ma_init_block_row"); + + if (!my_multi_malloc(MY_WME, + &row->empty_bits_buffer, info->s->base.pack_bytes, + &row->field_lengths, info->s->base.max_field_lengths, + &row->blob_lengths, sizeof(ulong) * info->s->base.blobs, + &row->null_field_lengths, (sizeof(uint) * + (info->s->base.fields - + info->s->base.blobs)), + &row->tail_positions, (sizeof(MARIA_RECORD_POS) * + (info->s->base.blobs + 2)), + &new_row->empty_bits_buffer, info->s->base.pack_bytes, + &new_row->field_lengths, + info->s->base.max_field_lengths, + &new_row->blob_lengths, + sizeof(ulong) * info->s->base.blobs, + &new_row->null_field_lengths, (sizeof(uint) * + (info->s->base.fields - + info->s->base.blobs)), + NullS, 0)) + DBUG_RETURN(1); + if (my_init_dynamic_array(&info->bitmap_blocks, + sizeof(MARIA_BITMAP_BLOCK), + ELEMENTS_RESERVED_FOR_MAIN_PART, 16)) + my_free((char*) &info->bitmap_blocks, MYF(0)); + row->base_length= new_row->base_length= info->s->base_length; + DBUG_RETURN(0); +} + +void _ma_end_block_row(MARIA_HA *info) +{ + DBUG_ENTER("_ma_end_block_row"); + my_free((gptr) info->cur_row.empty_bits_buffer, MYF(MY_ALLOW_ZERO_PTR)); + delete_dynamic(&info->bitmap_blocks); + my_free((gptr) info->cur_row.extents, MYF(MY_ALLOW_ZERO_PTR)); + DBUG_VOID_RETURN; +} + + +/**************************************************************************** + Helper functions +****************************************************************************/ + +static inline uint empty_pos_after_row(byte *dir) +{ + byte *prev; + /* + Find previous used entry. (There is always a previous entry as + the directory never starts with a deleted entry) + */ + for (prev= dir - DIR_ENTRY_SIZE ; + prev[0] == 0 && prev[1] == 0 ; + prev-= DIR_ENTRY_SIZE) + {} + return (uint) uint2korr(prev); +} + + +static my_bool check_if_zero(byte *pos, uint length) +{ + byte *end; + for (end= pos+ length; pos != end ; pos++) + if (pos[0] != 0) + return 1; + return 0; +} + + +/* + Find free postion in directory + + SYNOPSIS + find_free_position() + buff Page + block_size Size of page + res_rownr Store index to free position here + res_length Store length of found segment here + empty_space Store length of empty space on disk here. This is + all empty space, including the found block. + + NOTES + If there is a free directory entry (entry with postion == 0), + then use it and change it to be the size of the empty block + after the previous entry. This guarantees that all row entries + are stored on disk in inverse directory order, which makes life easier for + 'compact_page()' and to know if there is free space after any block. + + If there is no free entry (entry with postion == 0), then we create + a new one. + + We will update the offset and the length of the found dir entry to + match the position and empty space found. + + buff[EMPTY_SPACE_OFFSET] is NOT updated but left up to the caller + + RETURN + 0 Error (directory full) + # Pointer to directory entry on page +*/ + +static byte *find_free_position(byte *buff, uint block_size, uint *res_rownr, + uint *res_length, uint *empty_space) +{ + uint max_entry= (uint) ((uchar*) buff)[DIR_ENTRY_OFFSET]; + uint entry, length, first_pos; + byte *dir, *end; + + dir= (buff + block_size - DIR_ENTRY_SIZE * max_entry - PAGE_SUFFIX_SIZE); + end= buff + block_size - PAGE_SUFFIX_SIZE - DIR_ENTRY_SIZE; + + first_pos= PAGE_HEADER_SIZE; + *empty_space= uint2korr(buff + EMPTY_SPACE_OFFSET); + + /* Search after first empty position */ + for (entry= 0 ; dir <= end ; end-= DIR_ENTRY_SIZE, entry--) + { + if (end[0] == 0 && end[1] == 0) /* Found not used entry */ + { + length= empty_pos_after_row(end) - first_pos; + int2store(end, first_pos); /* Update dir entry */ + int2store(end + 2, length); + *res_rownr= entry; + *res_length= length; + return end; + } + first_pos= uint2korr(end) + uint2korr(end + 2); + } + /* No empty places in dir; create a new one */ + if (max_entry == MAX_ROWS_PER_PAGE) + return 0; + buff[DIR_ENTRY_OFFSET]= (byte) (uchar) max_entry+1; + dir-= DIR_ENTRY_SIZE; + length= (uint) (dir - buff - first_pos); + DBUG_ASSERT(length <= *empty_space - DIR_ENTRY_SIZE); + int2store(dir, first_pos); + int2store(dir+2, length); /* Current max length */ + *res_rownr= max_entry; + *res_length= length; + + /* Reduce directory entry size from free space size */ + (*empty_space)-= DIR_ENTRY_SIZE; + return dir; + +} + +/**************************************************************************** + Updating records +****************************************************************************/ + +/* + Calculate length of all the different field parts +*/ + +static void calc_record_size(MARIA_HA *info, const byte *record, + MARIA_ROW *row) +{ + MARIA_SHARE *share= info->s; + byte *field_length_data; + MARIA_COLUMNDEF *rec, *end_field; + uint blob_count= 0, *null_field_lengths= row->null_field_lengths; + + row->normal_length= row->char_length= row->varchar_length= + row->blob_length= row->extents_count= 0; + + /* Create empty bitmap and calculate length of each varlength/char field */ + bzero(row->empty_bits_buffer, share->base.pack_bytes); + row->empty_bits= row->empty_bits_buffer; + field_length_data= row->field_lengths; + for (rec= share->rec + share->base.fixed_not_null_fields, + end_field= share->rec + share->base.fields; + rec < end_field; rec++, null_field_lengths++) + { + if ((record[rec->null_pos] & rec->null_bit)) + { + if (rec->type != FIELD_BLOB) + *null_field_lengths= 0; + continue; + } + switch ((enum en_fieldtype) rec->type) { + case FIELD_CHECK: + case FIELD_NORMAL: /* Fixed length field */ + case FIELD_ZERO: + DBUG_ASSERT(rec->empty_bit == 0); + /* fall through */ + case FIELD_SKIP_PRESPACE: /* Not packed */ + row->normal_length+= rec->length; + *null_field_lengths= rec->length; + break; + case FIELD_SKIP_ZERO: /* Fixed length field */ + if (memcmp(record+ rec->null_pos, maria_zero_string, + rec->length) == 0) + { + row->empty_bits[rec->empty_pos] |= rec->empty_bit; + *null_field_lengths= 0; + } + else + { + row->normal_length+= rec->length; + *null_field_lengths= rec->length; + } + break; + case FIELD_SKIP_ENDSPACE: /* CHAR */ + { + const char *pos, *end; + for (pos= record + rec->offset, end= pos + rec->length; + end > pos && end[-1] == ' '; end--) + ; + if (pos == end) /* If empty string */ + { + row->empty_bits[rec->empty_pos]|= rec->empty_bit; + *null_field_lengths= 0; + } + else + { + uint length= (end - pos); + if (rec->length <= 255) + *field_length_data++= (byte) (uchar) length; + else + { + int2store(field_length_data, length); + field_length_data+= 2; + } + row->char_length+= length; + *null_field_lengths= length; + } + break; + } + case FIELD_VARCHAR: + { + uint length; + const byte *field_pos= record + rec->offset; + /* 256 is correct as this includes the length byte */ + if (rec->length <= 256) + { + if (!(length= (uint) (uchar) *field_pos)) + { + row->empty_bits[rec->empty_pos]|= rec->empty_bit; + *null_field_lengths= 0; + break; + } + *field_length_data++= *field_pos; + } + else + { + if (!(length= uint2korr(field_pos))) + { + row->empty_bits[rec->empty_pos]|= rec->empty_bit; + break; + } + field_length_data[0]= field_pos[0]; + field_length_data[1]= field_pos[1]; + field_length_data+= 2; + } + row->varchar_length+= length; + *null_field_lengths= length; + break; + } + case FIELD_BLOB: + { + const byte *field_pos= record + rec->offset; + uint size_length= rec->length - maria_portable_sizeof_char_ptr; + ulong blob_length= _ma_calc_blob_length(size_length, field_pos); + if (!blob_length) + { + row->empty_bits[rec->empty_pos]|= rec->empty_bit; + row->blob_lengths[blob_count++]= 0; + break; + } + row->blob_length+= blob_length; + row->blob_lengths[blob_count++]= blob_length; + memcpy(field_length_data, field_pos, size_length); + field_length_data+= size_length; + break; + } + default: + DBUG_ASSERT(0); + } + } + row->field_lengths_length= (uint) (field_length_data - row->field_lengths); + row->head_length= (row->base_length + + share->base.fixed_not_null_fields_length + + row->field_lengths_length + + size_to_store_key_length(row->field_lengths_length) + + row->normal_length + + row->char_length + row->varchar_length); + row->total_length= (row->head_length + row->blob_length); + if (row->total_length < share->base.min_row_length) + row->total_length= share->base.min_row_length; +} + + +/* + Compact page by removing all space between rows + + IMPLEMENTATION + Move up all rows to start of page. + Move blocks that are directly after each other with one memmove. + + TODO LATER + Remove TRANSID from rows that are visible to all transactions + + SYNOPSIS + compact_page() + buff Page to compact + block_size Size of page + recnr Put empty data after this row +*/ + + +void compact_page(byte *buff, uint block_size, uint rownr) +{ + uint max_entry= (uint) ((uchar *) buff)[DIR_ENTRY_OFFSET]; + uint page_pos, next_free_pos, start_of_found_block, diff, end_of_found_block; + byte *dir, *end; + DBUG_ENTER("compact_page"); + DBUG_PRINT("enter", ("rownr: %u", rownr)); + DBUG_ASSERT(max_entry > 0 && + max_entry < (block_size - PAGE_HEADER_SIZE - + PAGE_SUFFIX_SIZE) / DIR_ENTRY_SIZE); + + /* Move all entries before and including rownr up to start of page */ + dir= buff + block_size - DIR_ENTRY_SIZE * (rownr+1) - PAGE_SUFFIX_SIZE; + end= buff + block_size - DIR_ENTRY_SIZE - PAGE_SUFFIX_SIZE; + page_pos= next_free_pos= start_of_found_block= PAGE_HEADER_SIZE; + diff= 0; + for (; dir <= end ; end-= DIR_ENTRY_SIZE) + { + uint offset= uint2korr(end); + + if (offset) + { + uint row_length= uint2korr(end + 2); + DBUG_ASSERT(offset >= page_pos); + DBUG_ASSERT(buff + offset + row_length <= dir); + + if (offset != next_free_pos) + { + uint length= (next_free_pos - start_of_found_block); + /* + There was empty space before this and prev block + Check if we have to move prevous block up to page start + */ + if (page_pos != start_of_found_block) + { + /* move up previous block */ + memmove(buff + page_pos, buff + start_of_found_block, length); + } + page_pos+= length; + /* next continous block starts here */ + start_of_found_block= offset; + diff= offset - page_pos; + } + int2store(end, offset - diff); /* correct current pos */ + next_free_pos= offset + row_length; + } + } + if (page_pos != start_of_found_block) + { + uint length= (next_free_pos - start_of_found_block); + memmove(buff + page_pos, buff + start_of_found_block, length); + } + start_of_found_block= uint2korr(dir); + + if (rownr != max_entry - 1) + { + /* Move all entries after rownr to end of page */ + uint rownr_length; + next_free_pos= end_of_found_block= page_pos= + block_size - DIR_ENTRY_SIZE * max_entry - PAGE_SUFFIX_SIZE; + diff= 0; + /* End points to entry before 'rownr' */ + for (dir= buff + end_of_found_block ; dir <= end ; dir+= DIR_ENTRY_SIZE) + { + uint offset= uint2korr(dir); + uint row_length= uint2korr(dir + 2); + uint row_end= offset + row_length; + if (!offset) + continue; + DBUG_ASSERT(offset >= start_of_found_block && row_end <= next_free_pos); + + if (row_end != next_free_pos) + { + uint length= (end_of_found_block - next_free_pos); + if (page_pos != end_of_found_block) + { + /* move next block down */ + memmove(buff + page_pos - length, buff + next_free_pos, length); + } + page_pos-= length; + /* next continous block starts here */ + end_of_found_block= row_end; + diff= page_pos - row_end; + } + int2store(dir, offset + diff); /* correct current pos */ + next_free_pos= offset; + } + if (page_pos != end_of_found_block) + { + uint length= (end_of_found_block - next_free_pos); + memmove(buff + page_pos - length, buff + next_free_pos, length); + next_free_pos= page_pos- length; + } + /* Extend rownr block to cover hole */ + rownr_length= next_free_pos - start_of_found_block; + int2store(dir+2, rownr_length); + } + else + { + /* Extend last block cover whole page */ + uint length= (uint) (dir - buff) - start_of_found_block; + int2store(dir+2, length); + + buff[PAGE_TYPE_OFFSET]&= ~(byte) PAGE_CAN_BE_COMPACTED; + } + DBUG_EXECUTE("directory", _ma_print_directory(buff, block_size);); + DBUG_VOID_RETURN; +} + + + +/* + Read or initialize new head or tail page + + SYNOPSIS + get_head_or_tail_page() + info Maria handler + block Block to read + buff Suggest this buffer to key cache + length Minimum space needed + page_type HEAD_PAGE || TAIL_PAGE + res Store result position here + + NOTES + We don't decremented buff[EMPTY_SPACE_OFFSET] with the allocated data + as we don't know how much data the caller will actually use. + + RETURN + 0 ok All slots in 'res' are updated + 1 error my_errno is set +*/ + +struct st_row_pos_info +{ + byte *buff; /* page buffer */ + byte *data; /* Place for data */ + byte *dir; /* Directory */ + uint length; /* Length for data */ + uint offset; /* Offset to directory */ + uint empty_space; /* Space left on page */ +}; + +static my_bool get_head_or_tail_page(MARIA_HA *info, + MARIA_BITMAP_BLOCK *block, + byte *buff, uint length, uint page_type, + struct st_row_pos_info *res) +{ + uint block_size; + DBUG_ENTER("get_head_or_tail_page"); + + block_size= info->s->block_size; + if (block->org_bitmap_value == 0) /* Empty block */ + { + /* New page */ + bzero(buff, PAGE_HEADER_SIZE); + + /* + We zero the rest of the block to avoid getting old memory information + to disk and to allow the file to be compressed better if archived. + The rest of the code does not assume the block is zeroed above + PAGE_OVERHEAD_SIZE + */ + bzero(buff+ PAGE_HEADER_SIZE + length, + block_size - length - PAGE_HEADER_SIZE - DIR_ENTRY_SIZE - + PAGE_SUFFIX_SIZE); + buff[PAGE_TYPE_OFFSET]= (byte) page_type; + buff[DIR_ENTRY_OFFSET]= 1; + res->buff= buff; + res->empty_space= res->length= (block_size - PAGE_OVERHEAD_SIZE); + res->data= (buff + PAGE_HEADER_SIZE); + res->dir= res->data + res->length; + /* Store poistion to the first row */ + int2store(res->dir, PAGE_HEADER_SIZE); + res->offset= 0; + DBUG_ASSERT(length <= res->length); + } + else + { + byte *dir; + /* Read old page */ + if (!(res->buff= key_cache_read(info->s->key_cache, + info->dfile, + (my_off_t) block->page * block_size, 0, + buff, block_size, block_size, 0))) + DBUG_RETURN(1); + DBUG_ASSERT((res->buff[PAGE_TYPE_OFFSET] & PAGE_TYPE_MASK) == page_type); + if (!(dir= find_free_position(buff, block_size, &res->offset, + &res->length, &res->empty_space))) + { + if (res->length < length) + { + if (res->empty_space + res->length < length) + { + compact_page(res->buff, block_size, res->offset); + /* All empty space are now after current position */ + res->length= res->empty_space= uint2korr(dir+2); + } + if (res->length < length) + goto crashed; /* Wrong bitmap information */ + } + } + res->dir= dir; + res->data= res->buff + uint2korr(dir); + } + DBUG_RETURN(0); + +crashed: + my_errno= HA_ERR_WRONG_IN_RECORD; /* File crashed */ + DBUG_RETURN(1); +} + + +/* + Write tail of non-blob-data or blob + + SYNOPSIS + write_tail() + info Maria handler + block Block to tail page + row_part Data to write to page + length Length of data + + NOTES + block->page_count is updated to the directory offset for the tail + so that we can store the position in the row extent information + + RETURN + 0 ok + block->page_count is set to point (dir entry + TAIL_BIT) + + 1 error; In this case my_errno is set to the error +*/ + +static my_bool write_tail(MARIA_HA *info, + MARIA_BITMAP_BLOCK *block, + byte *row_part, uint length) +{ + MARIA_SHARE *share= share= info->s; + uint block_size= share->block_size, empty_space; + struct st_row_pos_info row_pos; + my_off_t position; + DBUG_ENTER("write_tail"); + DBUG_PRINT("enter", ("page: %lu length: %u", + (ulong) block->page, length)); + + info->keybuff_used= 1; + if (get_head_or_tail_page(info, block, info->keyread_buff, length, + TAIL_PAGE, &row_pos)) + DBUG_RETURN(1); + + memcpy(row_pos.data, row_part, length); + int2store(row_pos.dir + 2, length); + empty_space= row_pos.empty_space - length; + int2store(row_pos.buff + EMPTY_SPACE_OFFSET, empty_space); + block->page_count= row_pos.offset + TAIL_BIT; + /* + If there is less directory entries free than number of possible tails + we can write for a row, we mark the page full to ensure that we don't + during _ma_bitmap_find_place() allocate more entires on the tail page + than it can hold + */ + block->empty_space= ((uint) ((uchar*) row_pos.buff)[DIR_ENTRY_OFFSET] <= + MAX_ROWS_PER_PAGE - 1 - info->s->base.blobs ? + empty_space : 0); + block->used= BLOCKUSED_USED | BLOCKUSED_TAIL; + + /* Increase data file size, if extended */ + position= (my_off_t) block->page * block_size; + if (info->state->data_file_length <= position) + info->state->data_file_length= position + block_size; + DBUG_RETURN(key_cache_write(share->key_cache, + info->dfile, position, 0, + row_pos.buff, block_size, block_size, 1)); +} + + +/* + Write full pages + + SYNOPSIS + write_full_pages() + info Maria handler + block Where to write data + data Data to write + length Length of data + +*/ + +static my_bool write_full_pages(MARIA_HA *info, + MARIA_BITMAP_BLOCK *block, + byte *data, ulong length) +{ + my_off_t page; + MARIA_SHARE *share= share= info->s; + uint block_size= share->block_size; + uint data_size= FULL_PAGE_SIZE(block_size); + byte *buff= info->keyread_buff; + uint page_count; + my_off_t position; + DBUG_ENTER("write_full_pages"); + DBUG_PRINT("enter", ("length: %lu page: %lu page_count: %lu", + length, (ulong) block->page, block->page_count)); + + info->keybuff_used= 1; + page= block->page; + page_count= block->page_count; + + position= (my_off_t) (page + page_count) * block_size; + if (info->state->data_file_length < position) + info->state->data_file_length= position; + + /* Increase data file size, if extended */ + + for (; length; data+= data_size) + { + uint copy_length; + if (!page_count--) + { + block++; + page= block->page; + page_count= block->page_count - 1; + DBUG_PRINT("info", ("page: %lu page_count: %lu", + (ulong) block->page, block->page_count)); + + position= (page + page_count + 1) * block_size; + if (info->state->data_file_length < position) + info->state->data_file_length= position; + } + bzero(buff, LSN_SIZE); + buff[PAGE_TYPE_OFFSET]= (byte) BLOB_PAGE; + copy_length= min(data_size, length); + memcpy(buff + LSN_SIZE + PAGE_TYPE_SIZE, data, copy_length); + length-= copy_length; + + if (key_cache_write(share->key_cache, + info->dfile, (my_off_t) page * block_size, 0, + buff, block_size, block_size, 1)) + DBUG_RETURN(1); + page++; + block->used= BLOCKUSED_USED; + } + DBUG_RETURN(0); +} + + + + +/* + Store packed extent data + + SYNOPSIS + store_extent_info() + to Store first packed data here + row_extents_second_part Store rest here + first_block First block to store + count Number of blocks + + NOTES + We don't have to store the position for the head block +*/ + +static void store_extent_info(byte *to, + byte *row_extents_second_part, + MARIA_BITMAP_BLOCK *first_block, + uint count) +{ + MARIA_BITMAP_BLOCK *block, *end_block; + uint copy_length; + my_bool first_found= 0; + + for (block= first_block, end_block= first_block+count ; + block < end_block; block++) + { + /* The following is only false for marker blocks */ + if (likely(block->used)) + { + int5store(to, block->page); + int2store(to + 5, block->page_count); + to+= ROW_EXTENT_SIZE; + if (!first_found) + { + first_found= 1; + to= row_extents_second_part; + } + } + } + copy_length= (count -1) * ROW_EXTENT_SIZE; + /* + In some unlikely cases we have allocated to many blocks. Clear this + data. + */ + bzero(to, (my_size_t) (row_extents_second_part + copy_length - to)); +} + +/* + Write a record to a (set of) pages +*/ + +static my_bool write_block_record(MARIA_HA *info, const byte *record, + MARIA_ROW *row, + MARIA_BITMAP_BLOCKS *bitmap_blocks, + struct st_row_pos_info *row_pos) +{ + byte *data, *end_of_data, *tmp_data_used, *tmp_data; + byte *row_extents_first_part, *row_extents_second_part; + byte *field_length_data; + byte *page_buff; + MARIA_BITMAP_BLOCK *block, *head_block; + MARIA_SHARE *share; + MARIA_COLUMNDEF *rec, *end_field; + uint block_size, flag; + ulong *blob_lengths; + my_off_t position; + my_bool row_extents_in_use; + DBUG_ENTER("write_block_record"); + + LINT_INIT(row_extents_first_part); + LINT_INIT(row_extents_second_part); + + share= info->s; + head_block= bitmap_blocks->block; + block_size= share->block_size; + + info->cur_row.lastpos= ma_recordpos(head_block->page, row_pos->offset); + page_buff= row_pos->buff; + data= row_pos->data; + end_of_data= data + row_pos->length; + + /* Write header */ + flag= share->base.default_row_flag; + row_extents_in_use= 0; + if (unlikely(row->total_length > row_pos->length)) + { + /* Need extent */ + if (bitmap_blocks->count <= 1) + goto crashed; /* Wrong in bitmap */ + flag|= ROW_FLAG_EXTENTS; + row_extents_in_use= 1; + } + /* For now we have only a minimum header */ + *data++= (uchar) flag; + if (unlikely(flag & ROW_FLAG_NULLS_EXTENDED)) + *data++= (uchar) (share->base.null_bytes - + share->base.original_null_bytes); + if (row_extents_in_use) + { + /* Store first extent in header */ + store_key_length_inc(data, bitmap_blocks->count - 1); + row_extents_first_part= data; + data+= ROW_EXTENT_SIZE; + } + if (share->base.pack_fields) + store_key_length_inc(data, row->field_lengths_length); + if (share->calc_checksum) + *(data++)= (byte) info->cur_row.checksum; + memcpy(data, record, share->base.null_bytes); + data+= share->base.null_bytes; + memcpy(data, row->empty_bits, share->base.pack_bytes); + data+= share->base.pack_bytes; + + /* + Allocate a buffer of rest of data (except blobs) + + To avoid double copying of data, we copy as many columns that fits into + the page. The rest goes into info->packed_row. + + Using an extra buffer, instead of doing continous writes to different + pages, uses less code and we don't need to have to do a complex call + for every data segment we want to store. + */ + if (_ma_alloc_buffer(&info->rec_buff, &info->rec_buff_size, + row->head_length)) + DBUG_RETURN(1); + + tmp_data_used= 0; /* Either 0 or last used byte in 'data' */ + tmp_data= data; + + if (row_extents_in_use) + { + uint copy_length= (bitmap_blocks->count - 2) * ROW_EXTENT_SIZE; + if (!tmp_data_used && tmp_data + copy_length > end_of_data) + { + tmp_data_used= tmp_data; + tmp_data= info->rec_buff; + } + row_extents_second_part= tmp_data; + /* + We will copy the extents here when we have figured out the tail + positions. + */ + tmp_data+= copy_length; + } + + /* Copy fields that has fixed lengths (primary key etc) */ + for (rec= share->rec, end_field= rec + share->base.fixed_not_null_fields; + rec < end_field; rec++) + { + if (!tmp_data_used && tmp_data + rec->length > end_of_data) + { + tmp_data_used= tmp_data; + tmp_data= info->rec_buff; + } + memcpy(tmp_data, record + rec->offset, rec->length); + tmp_data+= rec->length; + } + + /* Copy length of data for variable length fields */ + if (!tmp_data_used && tmp_data + row->field_lengths_length > end_of_data) + { + tmp_data_used= tmp_data; + tmp_data= info->rec_buff; + } + field_length_data= row->field_lengths; + memcpy(tmp_data, field_length_data, row->field_lengths_length); + tmp_data+= row->field_lengths_length; + + /* Copy variable length fields and fields with null/zero */ + for (end_field= share->rec + share->base.fields - share->base.blobs; + rec < end_field ; + rec++) + { + const byte *field_pos; + ulong length; + if ((record[rec->null_pos] & rec->null_bit) || + (row->empty_bits[rec->empty_pos] & rec->empty_bit)) + continue; + + field_pos= record + rec->offset; + switch ((enum en_fieldtype) rec->type) { + case FIELD_NORMAL: /* Fixed length field */ + case FIELD_SKIP_PRESPACE: + case FIELD_SKIP_ZERO: /* Fixed length field */ + length= rec->length; + break; + case FIELD_SKIP_ENDSPACE: /* CHAR */ + /* Char that is space filled */ + if (rec->length <= 255) + length= (uint) (uchar) *field_length_data++; + else + { + length= uint2korr(field_length_data); + field_length_data+= 2; + } + break; + case FIELD_VARCHAR: + if (rec->length <= 256) + { + length= (uint) (uchar) *field_length_data++; + field_pos++; /* Skip length byte */ + } + else + { + length= uint2korr(field_length_data); + field_length_data+= 2; + field_pos+= 2; + } + break; + default: /* Wrong data */ + DBUG_ASSERT(0); + break; + } + if (!tmp_data_used && tmp_data + length > end_of_data) + { + /* Data didn't fit in page; Change to use tmp buffer */ + tmp_data_used= tmp_data; + tmp_data= info->rec_buff; + } + memcpy((char*) tmp_data, (char*) field_pos, length); + tmp_data+= length; + } + + block= head_block + head_block->sub_blocks; /* Point to first blob data */ + + end_field= rec + share->base.blobs; + blob_lengths= row->blob_lengths; + if (!tmp_data_used) + { + /* Still room on page; Copy as many blobs we can into this page */ + data= tmp_data; + for (; rec < end_field && *blob_lengths < (ulong) (end_of_data - data); + rec++, blob_lengths++) + { + byte *tmp_pos; + uint length; + if (!*blob_lengths) /* Null or "" */ + continue; + length= rec->length - maria_portable_sizeof_char_ptr; + memcpy_fixed((byte*) &tmp_pos, record + rec->offset + length, + sizeof(char*)); + memcpy(data, tmp_pos, *blob_lengths); + data+= *blob_lengths; + /* Skip over tail page that was to be used to store blob */ + block++; + bitmap_blocks->tail_page_skipped= 1; + } + if (head_block->sub_blocks > 1) + { + /* We have allocated pages that where not used */ + bitmap_blocks->page_skipped= 1; + } + } + else + data= tmp_data_used; /* Get last used on page */ + + { + /* Update page directory */ + uint length= (uint) (data - row_pos->data); + DBUG_PRINT("info", ("head length: %u", length)); + if (length < info->s->base.min_row_length) + length= info->s->base.min_row_length; + + int2store(row_pos->dir + 2, length); + /* update empty space at start of block */ + row_pos->empty_space-= length; + int2store(page_buff + EMPTY_SPACE_OFFSET, row_pos->empty_space); + /* Mark in bitmaps how the current page was actually used */ + head_block->empty_space= row_pos->empty_space; + if (page_buff[DIR_ENTRY_OFFSET] == (char) MAX_ROWS_PER_PAGE) + head_block->empty_space= 0; /* Page is full */ + head_block->used= BLOCKUSED_USED; + } + + /* + Now we have to write tail pages, as we need to store the position + to them in the row extent header. + + We first write out all blob tails, to be able to store them in + the current page or 'tmp_data'. + + Then we write the tail of the non-blob fields (The position to the + tail page is stored either in row header, the extents in the head + page or in the first full page of the non-blob data. It's never in + the tail page of the non-blob data) + */ + + if (row_extents_in_use) + { + if (rec != end_field) /* If blob fields */ + { + MARIA_COLUMNDEF *save_rec= rec; + MARIA_BITMAP_BLOCK *save_block= block; + MARIA_BITMAP_BLOCK *end_block; + ulong *save_blob_lengths= blob_lengths; + + for (; rec < end_field; rec++, blob_lengths++) + { + byte *blob_pos; + if (!*blob_lengths) /* Null or "" */ + continue; + if (block[block->sub_blocks - 1].used & BLOCKUSED_TAIL) + { + uint length; + length= rec->length - maria_portable_sizeof_char_ptr; + memcpy_fixed((byte *) &blob_pos, record + rec->offset + length, + sizeof(char*)); + length= *blob_lengths % FULL_PAGE_SIZE(block_size); /* tail size */ + if (write_tail(info, block + block->sub_blocks-1, + blob_pos + *blob_lengths - length, + length)) + goto disk_err; + } + for (end_block= block + block->sub_blocks; block < end_block; block++) + { + /* + Set only a bit, to not cause bitmap code to belive a block is full + when there is still a lot of entries in it + */ + block->used|= BLOCKUSED_USED; + } + } + rec= save_rec; + block= save_block; + blob_lengths= save_blob_lengths; + } + + if (tmp_data_used) /* non blob data overflows */ + { + MARIA_BITMAP_BLOCK *cur_block, *end_block; + MARIA_BITMAP_BLOCK *head_tail_block= 0; + ulong length; + ulong data_length= (tmp_data - info->rec_buff); + +#ifdef SANITY_CHECK + if (cur_block->sub_blocks == 1) + goto crashed; /* no reserved full or tails */ +#endif + + /* + Find out where to write tail for non-blob fields. + + Problem here is that the bitmap code may have allocated more + space than we need. We have to handle the following cases: + + - Bitmap code allocated a tail page we don't need. + - The last full page allocated needs to be changed to a tail page + (Because we put more data than we thought on the head page) + + The reserved pages in bitmap_blocks for the main page has one of + the following allocations: + - Full pages, with following blocks: + # * full pages + empty page ; To be used if we change last full to tail page. This + has 'count' = 0. + tail page (optional, if last full page was part full) + - One tail page + */ + + cur_block= head_block + 1; + end_block= head_block + head_block->sub_blocks; + while (data_length >= (length= (cur_block->page_count * + FULL_PAGE_SIZE(block_size)))) + { +#ifdef SANITY_CHECK + if ((cur_block == end_block) || (cur_block->used & BLOCKUSED_BIT)) + goto crashed; +#endif + data_length-= length; + (cur_block++)->used= BLOCKUSED_USED; + } + if (data_length) + { +#ifdef SANITY_CHECK + if ((cur_block == end_block)) + goto crashed; +#endif + if (cur_block->used & BLOCKUSED_TAIL) + { + DBUG_ASSERT(data_length < MAX_TAIL_SIZE(block_size)); + /* tail written to full tail page */ + cur_block->used= BLOCKUSED_USED; + head_tail_block= cur_block; + } + else if (data_length > length - MAX_TAIL_SIZE(block_size)) + { + /* tail written to full page */ + cur_block->used= BLOCKUSED_USED; + if ((cur_block != end_block - 1) && + (end_block[-1].used & BLOCKUSED_TAIL)) + bitmap_blocks->tail_page_skipped= 1; + } + else + { + /* + cur_block is a full block, followed by an empty and optional + tail block. Change cur_block to a tail block or split it + into full blocks and tail blocks. + */ + DBUG_ASSERT(cur_block[1].page_count == 0); + if (cur_block->page_count == 1) + { + /* convert full block to tail block */ + cur_block->used= BLOCKUSED_USED | BLOCKUSED_TAIL; + head_tail_block= cur_block; + } + else + { + DBUG_ASSERT(data_length < length - FULL_PAGE_SIZE(block_size)); + DBUG_PRINT("info", ("Splitting blocks into full and tail")); + cur_block[1].page= (cur_block->page + cur_block->page_count - 1); + cur_block[1].page_count= 1; + cur_block[1].used= 1; + cur_block->page_count--; + cur_block->used= BLOCKUSED_USED | BLOCKUSED_TAIL; + head_tail_block= cur_block + 1; + } + if (end_block[-1].used & BLOCKUSED_TAIL) + bitmap_blocks->tail_page_skipped= 1; + } + } + else + { + /* Must be an empty or tail page */ + DBUG_ASSERT(cur_block->page_count == 0 || + cur_block->used & BLOCKUSED_TAIL); + if (end_block[-1].used & BLOCKUSED_TAIL) + bitmap_blocks->tail_page_skipped= 1; + } + + /* + Write all extents into page or tmp_buff + + Note that we still don't have a correct position for the tail + of the non-blob fields. + */ + store_extent_info(row_extents_first_part, + row_extents_second_part, + head_block+1, bitmap_blocks->count - 1); + if (head_tail_block) + { + ulong data_length= (tmp_data - info->rec_buff); + uint length; + byte *extent_data; + + length= (uint) (data_length % FULL_PAGE_SIZE(block_size)); + if (write_tail(info, head_tail_block, data + data_length - length, + length)) + goto disk_err; + tmp_data-= length; /* Remove the tail */ + + /* Store the tail position for the non-blob fields */ + if (head_tail_block == head_block + 1) + extent_data= row_extents_first_part; + else + extent_data= row_extents_second_part + + ((head_tail_block - head_block) - 2) * ROW_EXTENT_SIZE; + int5store(extent_data, head_tail_block->page); + int2store(extent_data + 5, head_tail_block->page_count); + } + } + else + store_extent_info(row_extents_first_part, + row_extents_second_part, + head_block+1, bitmap_blocks->count - 1); + } + /* Increase data file size, if extended */ + position= (my_off_t) head_block->page * block_size; + if (info->state->data_file_length <= position) + info->state->data_file_length= position + block_size; + if (key_cache_write(share->key_cache, + info->dfile, position, 0, + page_buff, share->block_size, share->block_size, 1)) + goto disk_err; + + if (tmp_data_used) + { + /* Write data stored in info->rec_buff to pages */ + DBUG_ASSERT(bitmap_blocks->count != 0); + if (write_full_pages(info, bitmap_blocks->block + 1, info->rec_buff, + (ulong) (tmp_data - info->rec_buff))) + goto disk_err; + } + + /* Write rest of blobs (data, but no tails as they are already written) */ + for (; rec < end_field; rec++, blob_lengths++) + { + byte *blob_pos; + uint length; + ulong blob_length; + if (!*blob_lengths) /* Null or "" */ + continue; + length= rec->length - maria_portable_sizeof_char_ptr; + memcpy_fixed((byte*) &blob_pos, record + rec->offset + length, + sizeof(char*)); + /* remove tail part */ + blob_length= *blob_lengths; + if (block[block->sub_blocks - 1].used & BLOCKUSED_TAIL) + blob_length-= (blob_length % FULL_PAGE_SIZE(block_size)); + + if (write_full_pages(info, block, blob_pos, blob_length)) + goto disk_err; + block+= block->sub_blocks; + } + /* Release not used space in used pages */ + if (_ma_bitmap_release_unused(info, bitmap_blocks)) + goto disk_err; + DBUG_RETURN(0); + +crashed: + my_errno= HA_ERR_WRONG_IN_RECORD; /* File crashed */ +disk_err: + /* Something was wrong with data on record */ + DBUG_RETURN(1); +} + + +/* + Write a record (to get the row id for it) + + SYNOPSIS + _ma_write_init_block_record() + info Maria handler + record Record to write + + NOTES + This is done BEFORE we write the keys to the row! + + RETURN + HA_OFFSET_ERROR Something went wrong + # Rowid for row +*/ + +MARIA_RECORD_POS _ma_write_init_block_record(MARIA_HA *info, + const byte *record) +{ + MARIA_BITMAP_BLOCKS *blocks= &info->cur_row.insert_blocks; + struct st_row_pos_info row_pos; + DBUG_ENTER("_ma_write_init_block_record"); + + calc_record_size(info, record, &info->cur_row); + if (_ma_bitmap_find_place(info, &info->cur_row, blocks)) + DBUG_RETURN(HA_OFFSET_ERROR); /* Error reding bitmap */ + if (get_head_or_tail_page(info, blocks->block, info->buff, + info->s->base.min_row_length, HEAD_PAGE, &row_pos)) + DBUG_RETURN(HA_OFFSET_ERROR); + info->cur_row.lastpos= ma_recordpos(blocks->block->page, row_pos.offset); + if (info->s->calc_checksum) + info->cur_row.checksum= (info->s->calc_checksum)(info,record); + if (write_block_record(info, record, &info->cur_row, blocks, &row_pos)) + DBUG_RETURN(HA_OFFSET_ERROR); /* Error reading bitmap */ + DBUG_PRINT("exit", ("Rowid: %lu", (ulong) info->cur_row.lastpos)); + DBUG_RETURN(info->cur_row.lastpos); +} + + +/* + Dummy function for (*info->s->write_record)() + + Nothing to do here, as we already wrote the record in + _ma_write_init_block_record() +*/ + +my_bool _ma_write_block_record(MARIA_HA *info __attribute__ ((unused)), + const byte *record __attribute__ ((unused)) +) +{ + return 0; /* Row already written */ +} + + +/* + Remove row written by _ma_write_block_record + + SYNOPSIS + _ma_abort_write_block_record() + info Maria handler + + INFORMATION + This is called in case we got a duplicate unique key while + writing keys. + + RETURN + 0 ok + 1 error +*/ + +my_bool _ma_write_abort_block_record(MARIA_HA *info) +{ + my_bool res= 0; + MARIA_BITMAP_BLOCKS *blocks= &info->cur_row.insert_blocks; + MARIA_BITMAP_BLOCK *block, *end; + DBUG_ENTER("_ma_abort_write_block_record"); + + if (delete_head_or_tail(info, + ma_recordpos_to_page(info->cur_row.lastpos), + ma_recordpos_to_offset(info->cur_row.lastpos), 1)) + res= 1; + for (block= blocks->block + 1, end= block + blocks->count - 1; block < end; + block++) + { + if (block->used & BLOCKUSED_TAIL) + { + /* + block->page_count is set to the tail directory entry number in + write_block_record() + */ + if (delete_head_or_tail(info, block->page, block->page_count & ~TAIL_BIT, + 0)) + res= 1; + } + else + { + pthread_mutex_lock(&info->s->bitmap.bitmap_lock); + if (_ma_reset_full_page_bits(info, &info->s->bitmap, block->page, + block->page_count)) + res= 1; + pthread_mutex_unlock(&info->s->bitmap.bitmap_lock); + } + } + DBUG_RETURN(res); +} + + +/* + Update a record + + NOTES + For the moment, we assume that info->curr_row.extents is always updated + when a row is read. In the future we may decide to read this on demand + for rows split into many extents. +*/ + +my_bool _ma_update_block_record(MARIA_HA *info, MARIA_RECORD_POS record_pos, + const byte *record) +{ + MARIA_BITMAP_BLOCKS *blocks= &info->cur_row.insert_blocks; + byte *buff; + MARIA_ROW *cur_row= &info->cur_row, *new_row= &info->new_row; + uint rownr, org_empty_size, head_length; + uint block_size= info->s->block_size; + byte *dir; + ulonglong page; + struct st_row_pos_info row_pos; + DBUG_ENTER("_ma_update_block_record"); + DBUG_PRINT("enter", ("rowid: %lu", (long) record_pos)); + + calc_record_size(info, record, new_row); + page= ma_recordpos_to_page(record_pos); + + if (!(buff= key_cache_read(info->s->key_cache, + info->dfile, (my_off_t) page * block_size, 0, + info->buff, block_size, block_size, 0))) + DBUG_RETURN(1); + org_empty_size= uint2korr(buff + EMPTY_SPACE_OFFSET); + rownr= ma_recordpos_to_offset(record_pos); + dir= (buff + block_size - DIR_ENTRY_SIZE * rownr - + DIR_ENTRY_SIZE - PAGE_SUFFIX_SIZE); + + if ((org_empty_size + cur_row->head_length) >= new_row->total_length) + { + uint empty, offset, length; + MARIA_BITMAP_BLOCK block; + + /* + We can fit the new row in the same page as the original head part + of the row + */ + block.org_bitmap_value= _ma_free_size_to_head_pattern(&info->s->bitmap, + org_empty_size); + offset= uint2korr(dir); + length= uint2korr(dir + 2); + empty= 0; + if (new_row->total_length > length) + { + /* See if there is empty space after */ + if (rownr != (uint) ((uchar *) buff)[DIR_ENTRY_OFFSET] - 1) + empty= empty_pos_after_row(dir) - (offset + length); + if (new_row->total_length > length + empty) + { + compact_page(buff, info->s->block_size, rownr); + org_empty_size= 0; + length= uint2korr(dir + 2); + } + } + row_pos.buff= buff; + row_pos.offset= rownr; + row_pos.empty_space= org_empty_size + length; + row_pos.dir= dir; + row_pos.data= buff + uint2korr(dir); + row_pos.length= length + empty; + blocks->block= █ + blocks->count= 1; + block.page= page; + block.sub_blocks= 1; + block.used= BLOCKUSED_USED | BLOCKUSED_USE_ORG_BITMAP; + block.empty_space= row_pos.empty_space; + /* Update cur_row, if someone calls update at once again */ + cur_row->head_length= new_row->total_length; + if (_ma_bitmap_free_full_pages(info, cur_row->extents, + cur_row->extents_count)) + DBUG_RETURN(1); + DBUG_RETURN(write_block_record(info, record, new_row, blocks, &row_pos)); + } + /* + Allocate all size in block for record + QQ: Need to improve this to do compact if we can fit one more blob into + the head page + */ + head_length= uint2korr(dir + 2); + if (buff[PAGE_TYPE_OFFSET] & PAGE_CAN_BE_COMPACTED && org_empty_size && + (head_length < new_row->head_length || + (new_row->total_length <= head_length && + org_empty_size + head_length >= new_row->total_length))) + { + compact_page(buff, info->s->block_size, rownr); + org_empty_size= 0; + head_length= uint2korr(dir + 2); + } + + /* Delete old row */ + if (delete_tails(info, cur_row->tail_positions)) + DBUG_RETURN(1); + if (_ma_bitmap_free_full_pages(info, cur_row->extents, + cur_row->extents_count)) + DBUG_RETURN(1); + if (_ma_bitmap_find_new_place(info, new_row, page, head_length, blocks)) + DBUG_RETURN(1); + + row_pos.buff= buff; + row_pos.offset= rownr; + row_pos.empty_space= org_empty_size + head_length; + row_pos.dir= dir; + row_pos.data= buff + uint2korr(dir); + row_pos.length= head_length; + DBUG_RETURN(write_block_record(info, record, new_row, blocks, &row_pos)); +} + + +/* + Delete a head a tail part + + SYNOPSIS + delete_head_or_tail() + info Maria handler + page Page (not file offset!) on which the row is + head 1 if this is a head page + + NOTES + Uses info->keyread_buff + + RETURN + 0 ok + 1 error +*/ + +static my_bool delete_head_or_tail(MARIA_HA *info, + ulonglong page, uint record_number, + my_bool head) +{ + MARIA_SHARE *share= info->s; + uint number_of_records, empty_space, length; + uint block_size= share->block_size; + byte *buff, *dir; + my_off_t position; + DBUG_ENTER("delete_head_or_tail"); + + info->keybuff_used= 1; + if (!(buff= key_cache_read(share->key_cache, + info->dfile, page * block_size, 0, + info->keyread_buff, + block_size, block_size, 0))) + DBUG_RETURN(1); + + number_of_records= (uint) ((uchar *) buff)[DIR_ENTRY_OFFSET]; +#ifdef SANITY_CHECKS + if (record_number >= number_of_records || + record_number > MAX_ROWS_PER_PAGE || + record_number > ((block_size - LSN_SIZE - PAGE_TYPE_SIZE - 1 - + PAGE_SUFFIX_SIZE) / (DIR_ENTRY_SIZE + MIN_TAIL_SIZE))) + { + DBUG_PRINT("error", ("record_number: %u number_of_records: %u", + record_number, number_of_records)); + DBUG_RETURN(1); + } +#endif + + dir= (buff + block_size - DIR_ENTRY_SIZE * record_number - + DIR_ENTRY_SIZE - PAGE_SUFFIX_SIZE); + dir[0]= dir[1]= 0; /* Delete entry */ + length= uint2korr(dir + 2); + empty_space= uint2korr(buff + EMPTY_SPACE_OFFSET); + + if (record_number == number_of_records - 1) + { + /* Delete this entry and all following empty directory entries */ + byte *end= buff + block_size - PAGE_SUFFIX_SIZE; + do + { + number_of_records--; + dir+= DIR_ENTRY_SIZE; + empty_space+= DIR_ENTRY_SIZE; + } while (dir < end && dir[0] == 0 && dir[1] == 0); + buff[DIR_ENTRY_OFFSET]= (byte) (uchar) number_of_records; + } + empty_space+= length; + if (number_of_records != 0) + { + int2store(buff + EMPTY_SPACE_OFFSET, empty_space); + buff[PAGE_TYPE_OFFSET]|= (byte) PAGE_CAN_BE_COMPACTED; + position= (my_off_t) page * block_size; + if (key_cache_write(share->key_cache, + info->dfile, position, 0, + buff, block_size, block_size, 1)) + DBUG_RETURN(1); + } + else + { + DBUG_ASSERT(empty_space >= info->s->bitmap.sizes[0]); + } + DBUG_PRINT("info", ("empty_space: %u", empty_space)); + DBUG_RETURN(_ma_bitmap_set(info, page, head, empty_space)); +} + + +/* + delete all tails + + SYNOPSIS + delete_tails() + info Handler + tails Pointer to vector of tail positions, ending with 0 + + NOTES + Uses info->keyread_buff + + RETURN + 0 ok + 1 error +*/ + +static my_bool delete_tails(MARIA_HA *info, MARIA_RECORD_POS *tails) +{ + my_bool res= 0; + DBUG_ENTER("delete_tails"); + for (; *tails; tails++) + { + if (delete_head_or_tail(info, + ma_recordpos_to_page(*tails), + ma_recordpos_to_offset(*tails), 0)) + res= 1; + } + DBUG_RETURN(res); +} + + +/* + Delete a record + + NOTES + For the moment, we assume that info->cur_row.extents is always updated + when a row is read. In the future we may decide to read this on demand + for rows with many splits. +*/ + +my_bool _ma_delete_block_record(MARIA_HA *info) +{ + DBUG_ENTER("_ma_delete_block_record"); + if (delete_head_or_tail(info, + ma_recordpos_to_page(info->cur_row.lastpos), + ma_recordpos_to_offset(info->cur_row.lastpos), + 1) || + delete_tails(info, info->cur_row.tail_positions)) + DBUG_RETURN(1); + DBUG_RETURN(_ma_bitmap_free_full_pages(info, info->cur_row.extents, + info->cur_row.extents_count)); +} + + +/**************************************************************************** + Reading of records +****************************************************************************/ + +/* + Read position to record from record directory at end of page + + SYNOPSIS + get_record_position() + buff page buffer + block_size block size for page + record_number Record number in index + end_of_data pointer to end of data for record + + RETURN + 0 Error in data + # Pointer to start of record. + In this case *end_of_data is set. +*/ + +static byte *get_record_position(byte *buff, uint block_size, + uint record_number, byte **end_of_data) +{ + uint number_of_records= (uint) ((uchar *) buff)[DIR_ENTRY_OFFSET]; + byte *dir; + byte *data; + uint offset, length; + +#ifdef SANITY_CHECKS + if (record_number >= number_of_records || + record_number > MAX_ROWS_PER_PAGE || + record_number > ((block_size - PAGE_HEADER_SIZE - PAGE_SUFFIX_SIZE) / + (DIR_ENTRY_SIZE + MIN_TAIL_SIZE))) + { + DBUG_PRINT("error", + ("Wrong row number: record_number: %u number_of_records: %u", + record_number, number_of_records)); + return 0; + } +#endif + + dir= (buff + block_size - DIR_ENTRY_SIZE * record_number - + DIR_ENTRY_SIZE - PAGE_SUFFIX_SIZE); + offset= uint2korr(dir); + length= uint2korr(dir + 2); +#ifdef SANITY_CHECKS + if (offset < PAGE_HEADER_SIZE || + offset + length > (block_size - + number_of_records * DIR_ENTRY_SIZE - + PAGE_SUFFIX_SIZE)) + { + DBUG_PRINT("error", + ("Wrong row position: record_number: %u offset: %u " + "length: %u number_of_records: %u", + record_number, offset, length, number_of_records)); + return 0; + } +#endif + data= buff + offset; + *end_of_data= data + length; + return data; +} + + +/* + Init extent + + NOTES + extent is a cursor over which pages to read +*/ + +static void init_extent(MARIA_EXTENT_CURSOR *extent, byte *extent_info, + uint extents, MARIA_RECORD_POS *tail_positions) +{ + uint page_count; + extent->extent= extent_info; + extent->extent_count= extents; + extent->page= uint5korr(extent_info); /* First extent */ + page_count= uint2korr(extent_info+5); + extent->page_count= page_count & ~TAIL_BIT; + extent->tail= page_count & TAIL_BIT; + extent->tail_positions= tail_positions; +} + + +/* + Read next extent + + SYNOPSIS + read_next_extent() + info Maria handler + extent Pointer to current extent (this is updated to point + to next) + end_of_data Pointer to end of data in read block (out) + + NOTES + New block is read into info->buff + + RETURN + 0 Error; my_errno is set + # Pointer to start of data in read block + In this case end_of_data is updated to point to end of data. +*/ + +static byte *read_next_extent(MARIA_HA *info, MARIA_EXTENT_CURSOR *extent, + byte **end_of_data) +{ + MARIA_SHARE *share= info->s; + byte *buff, *data; + DBUG_ENTER("read_next_extent"); + + if (!extent->page_count) + { + uint page_count; + if (!--extent->extent_count) + goto crashed; + extent->extent+= ROW_EXTENT_SIZE; + extent->page= uint5korr(extent->extent); + page_count= uint2korr(extent->extent+ROW_EXTENT_PAGE_SIZE); + extent->tail= page_count & TAIL_BIT; + extent->page_count= (page_count & ~TAIL_BIT); + extent->first_extent= 0; + DBUG_PRINT("info",("New extent. Page: %lu page_count: %u tail_flag: %d", + (ulong) extent->page, extent->page_count, + extent->tail != 0)); + } + + if (info->cur_row.empty_bits != info->cur_row.empty_bits_buffer) + { + /* + First read of extents: Move data from info->buff to + internals buffers. + */ + memcpy(info->cur_row.empty_bits_buffer, info->cur_row.empty_bits, + share->base.pack_bytes); + info->cur_row.empty_bits= info->cur_row.empty_bits_buffer; + } + + if (!(buff= key_cache_read(share->key_cache, + info->dfile, extent->page * share->block_size, 0, + info->buff, + share->block_size, share->block_size, 0))) + { + /* check if we tried to read over end of file (ie: bad data in record) */ + if ((extent->page + 1) * share->block_size > info->state->data_file_length) + goto crashed; + DBUG_RETURN(0); + } + if (!extent->tail) + { + /* Full data page */ + DBUG_ASSERT((buff[PAGE_TYPE_OFFSET] & PAGE_TYPE_MASK) == BLOB_PAGE); + extent->page++; /* point to next page */ + extent->page_count--; + *end_of_data= buff + share->block_size; + info->cur_row.full_page_count++; /* For maria_chk */ + DBUG_RETURN(extent->data_start= buff + LSN_SIZE + PAGE_TYPE_SIZE); + } + /* Found tail. page_count is in this case the position in the tail page */ + + DBUG_ASSERT((buff[PAGE_TYPE_OFFSET] & PAGE_TYPE_MASK) == TAIL_PAGE); + *(extent->tail_positions++)= ma_recordpos(extent->page, + extent->page_count); + info->cur_row.tail_count++; /* For maria_chk */ + + if (!(data= get_record_position(buff, share->block_size, + extent->page_count, + end_of_data))) + goto crashed; + extent->data_start= data; + extent->page_count= 0; /* No more data in extent */ + DBUG_RETURN(data); + + +crashed: + my_errno= HA_ERR_WRONG_IN_RECORD; /* File crashed */ + DBUG_PRINT("error", ("wrong extent information")); + DBUG_RETURN(0); +} + + +/* + Read data that may be split over many blocks + + SYNOPSIS + read_long_data() + info Maria handler + to Store result string here (this is allocated) + extent Pointer to current extent position + data Current position in buffer + end_of_data End of data in buffer + + NOTES + When we have to read a new buffer, it's read into info->buff + + This loop is implemented by goto's instead of a for() loop as + the code is notable smaller and faster this way (and it's not nice + to jump into a for loop() or into a 'then' clause) + + RETURN + 0 ok + 1 error +*/ + +static my_bool read_long_data(MARIA_HA *info, byte *to, ulong length, + MARIA_EXTENT_CURSOR *extent, + byte **data, byte **end_of_data) +{ + DBUG_ENTER("read_long_data"); + DBUG_PRINT("enter", ("length: %lu", length)); + DBUG_ASSERT(*data <= *end_of_data); + + for(;;) + { + uint left_length; + left_length= (uint) (*end_of_data - *data); + if (likely(left_length >= length)) + { + memcpy(to, *data, length); + (*data)+= length; + DBUG_RETURN(0); + } + memcpy(to, *data, left_length); + to+= left_length; + length-= left_length; + if (!(*data= read_next_extent(info, extent, end_of_data))) + break; + } + DBUG_RETURN(1); +} + + +/* + Read a record from page (helper function for _ma_read_block_record()) + + SYNOPSIS + _ma_read_block_record2() + info Maria handler + record Store record here + data Start of head data for row + end_of_data End of data for row + + NOTES + The head page is already read by caller + Following data is update in info->cur_row: + + cur_row.head_length is set to size of entry in head block + cur_row.tail_positions is set to point to all tail blocks + cur_row.extents points to extents data + cur_row.extents_counts contains number of extents + cur_row.empty_bits points to empty bits part in read record + cur_row.field_lengths contains packed length of all fields + + RETURN + 0 ok + # Error code +*/ + +int _ma_read_block_record2(MARIA_HA *info, byte *record, + byte *data, byte *end_of_data) +{ + MARIA_SHARE *share= info->s; + byte *field_length_data, *blob_buffer, *start_of_data; + uint flag, null_bytes, cur_null_bytes, row_extents, field_lengths; + my_bool found_blob= 0; + MARIA_EXTENT_CURSOR extent; + MARIA_COLUMNDEF *rec, *end_field; + DBUG_ENTER("_ma_read_block_record2"); + + LINT_INIT(field_lengths); + LINT_INIT(field_length_data); + LINT_INIT(blob_buffer); + + start_of_data= data; + flag= (uint) (uchar) data[0]; + cur_null_bytes= share->base.original_null_bytes; + null_bytes= share->base.null_bytes; + info->cur_row.head_length= (uint) (end_of_data - data); + info->cur_row.full_page_count= info->cur_row.tail_count= 0; + + /* Skip trans header (for now, until we have MVCC csupport) */ + data+= total_header_size[(flag & PRECALC_HEADER_BITMASK)]; + if (flag & ROW_FLAG_NULLS_EXTENDED) + cur_null_bytes+= data[-1]; + + row_extents= 0; + if (flag & ROW_FLAG_EXTENTS) + { + uint row_extent_size; + /* + Record is split over many data pages. + Get number of extents and first extent + */ + get_key_length(row_extents, data); + info->cur_row.extents_count= row_extents; + row_extent_size= row_extents * ROW_EXTENT_SIZE; + if (info->cur_row.extents_buffer_length < row_extent_size && + _ma_alloc_buffer(&info->cur_row.extents, + &info->cur_row.extents_buffer_length, + row_extent_size)) + DBUG_RETURN(my_errno); + memcpy(info->cur_row.extents, data, ROW_EXTENT_SIZE); + data+= ROW_EXTENT_SIZE; + init_extent(&extent, info->cur_row.extents, row_extents, + info->cur_row.tail_positions); + } + else + { + info->cur_row.extents_count= 0; + (*info->cur_row.tail_positions)= 0; + extent.page_count= 0; + extent.extent_count= 1; + } + extent.first_extent= 1; + + if (share->base.max_field_lengths) + { + get_key_length(field_lengths, data); +#ifdef SANITY_CHECKS + if (field_lengths > share->base.max_field_lengths) + goto err; +#endif + } + + if (share->calc_checksum) + info->cur_row.checksum= (uint) (uchar) *data++; + /* data now points on null bits */ + memcpy(record, data, cur_null_bytes); + if (unlikely(cur_null_bytes != null_bytes)) + { + /* + This only happens if we have added more NULL columns with + ALTER TABLE and are fetching an old, not yet modified old row + */ + bzero(record + cur_null_bytes, (uint) (null_bytes - cur_null_bytes)); + } + data+= null_bytes; + info->cur_row.empty_bits= (byte*) data; /* Pointer to empty bitmask */ + data+= share->base.pack_bytes; + + /* TODO: Use field offsets, instead of just skipping them */ + data+= share->base.field_offsets * FIELD_OFFSET_SIZE; + + /* + Read row extents (note that first extent was already read into + info->cur_row.extents above) + */ + if (row_extents) + { + if (read_long_data(info, info->cur_row.extents + ROW_EXTENT_SIZE, + (row_extents - 1) * ROW_EXTENT_SIZE, + &extent, &data, &end_of_data)) + DBUG_RETURN(my_errno); + } + + /* + Data now points to start of fixed length field data that can't be null + or 'empty'. Note that these fields can't be split over blocks + */ + for (rec= share->rec, end_field= rec + share->base.fixed_not_null_fields; + rec < end_field; rec++) + { + uint rec_length= rec->length; + if (data >= end_of_data && + !(data= read_next_extent(info, &extent, &end_of_data))) + goto err; + memcpy(record + rec->offset, data, rec_length); + data+= rec_length; + } + + /* Read array of field lengths. This may be stored in several extents */ + if (share->base.max_field_lengths) + { + field_length_data= info->cur_row.field_lengths; + if (read_long_data(info, field_length_data, field_lengths, &extent, + &data, &end_of_data)) + DBUG_RETURN(my_errno); + } + + /* Read variable length data. Each of these may be split over many extents */ + for (end_field= share->rec + share->base.fields; rec < end_field; rec++) + { + enum en_fieldtype type= (enum en_fieldtype) rec->type; + byte *field_pos= record + rec->offset; + /* First check if field is present in record */ + if (record[rec->null_pos] & rec->null_bit) + continue; + else if (info->cur_row.empty_bits[rec->empty_pos] & rec->empty_bit) + { + if (type == FIELD_SKIP_ENDSPACE) + bfill(record + rec->offset, rec->length, ' '); + else + bzero(record + rec->offset, rec->fill_length); + continue; + } + switch (type) { + case FIELD_NORMAL: /* Fixed length field */ + case FIELD_SKIP_PRESPACE: + case FIELD_SKIP_ZERO: /* Fixed length field */ + if (data >= end_of_data && + !(data= read_next_extent(info, &extent, &end_of_data))) + goto err; + memcpy(field_pos, data, rec->length); + data+= rec->length; + break; + case FIELD_SKIP_ENDSPACE: /* CHAR */ + { + /* Char that is space filled */ + uint length; + if (rec->length <= 255) + length= (uint) (uchar) *field_length_data++; + else + { + length= uint2korr(field_length_data); + field_length_data+= 2; + } +#ifdef SANITY_CHECKS + if (length > rec->length) + goto err; +#endif + if (read_long_data(info, field_pos, length, &extent, &data, + &end_of_data)) + DBUG_RETURN(my_errno); + bfill(field_pos + length, rec->length - length, ' '); + break; + } + case FIELD_VARCHAR: + { + ulong length; + if (rec->length <= 256) + { + length= (uint) (uchar) (*field_pos++= *field_length_data++); + } + else + { + length= uint2korr(field_length_data); + field_pos[0]= field_length_data[0]; + field_pos[1]= field_length_data[1]; + field_pos+= 2; + field_length_data+= 2; + } + if (read_long_data(info, field_pos, length, &extent, &data, + &end_of_data)) + DBUG_RETURN(my_errno); + break; + } + case FIELD_BLOB: + { + uint size_length= rec->length - maria_portable_sizeof_char_ptr; + ulong blob_length= _ma_calc_blob_length(size_length, field_length_data); + + if (!found_blob) + { + /* Calculate total length for all blobs */ + ulong blob_lengths= 0; + byte *length_data= field_length_data; + MARIA_COLUMNDEF *blob_field= rec; + + found_blob= 1; + for (; blob_field < end_field; blob_field++) + { + uint size_length; + if ((record[blob_field->null_pos] & blob_field->null_bit) || + (info->cur_row.empty_bits[blob_field->empty_pos] & + blob_field->empty_bit)) + continue; + size_length= blob_field->length - maria_portable_sizeof_char_ptr; + blob_lengths+= _ma_calc_blob_length(size_length, length_data); + length_data+= size_length; + } + DBUG_PRINT("info", ("Total blob length: %lu", blob_lengths)); + if (_ma_alloc_buffer(&info->rec_buff, &info->rec_buff_size, + blob_lengths)) + DBUG_RETURN(my_errno); + blob_buffer= info->rec_buff; + } + + memcpy(field_pos, field_length_data, size_length); + memcpy_fixed(field_pos + size_length, (byte *) & blob_buffer, + sizeof(char*)); + field_length_data+= size_length; + + /* + After we have read one extent, then each blob is in it's own extent + */ + if (extent.first_extent && (ulong) (end_of_data - data) < blob_length) + end_of_data= data; /* Force read of next extent */ + + if (read_long_data(info, blob_buffer, blob_length, &extent, &data, + &end_of_data)) + DBUG_RETURN(my_errno); + blob_buffer+= blob_length; + break; + } +#ifdef EXTRA_DEBUG + default: + DBUG_ASSERT(0); /* purecov: deadcode */ + goto err; +#endif + } + continue; + } + + if (row_extents) + { + DBUG_PRINT("info", ("Row read: page_count: %lu extent_count: %lu", + extent.page_count, extent.extent_count)); + *extent.tail_positions= 0; /* End marker */ + if (extent.page_count) + goto err; + if (extent.extent_count > 1) + if (check_if_zero(extent.extent, + (extent.extent_count-1) * ROW_EXTENT_SIZE)) + goto err; + } + else + { + DBUG_PRINT("info", ("Row read")); + if (data != end_of_data && (uint) (end_of_data - start_of_data) >= + info->s->base.min_row_length) + goto err; + } + + info->update|= HA_STATE_AKTIV; /* We have a aktive record */ + DBUG_RETURN(0); + +err: + /* Something was wrong with data on record */ + DBUG_PRINT("error", ("Found record with wrong data")); + DBUG_RETURN((my_errno= HA_ERR_WRONG_IN_RECORD)); +} + + +/* + Read a record based on record position + + SYNOPSIS + _ma_read_block_record() + info Maria handler + record Store record here + record_pos Record position +*/ + +int _ma_read_block_record(MARIA_HA *info, byte *record, + MARIA_RECORD_POS record_pos) +{ + byte *data, *end_of_data, *buff; + my_off_t page; + uint offset; + uint block_size= info->s->block_size; + DBUG_ENTER("_ma_read_block_record"); + DBUG_PRINT("enter", ("rowid: %lu", (long) record_pos)); + + page= ma_recordpos_to_page(record_pos) * block_size; + offset= ma_recordpos_to_offset(record_pos); + + if (!(buff= key_cache_read(info->s->key_cache, + info->dfile, page, 0, info->buff, + block_size, block_size, 1))) + DBUG_RETURN(1); + DBUG_ASSERT((buff[PAGE_TYPE_OFFSET] & PAGE_TYPE_MASK) == HEAD_PAGE); + if (!(data= get_record_position(buff, block_size, offset, &end_of_data))) + { + my_errno= HA_ERR_WRONG_IN_RECORD; /* File crashed */ + DBUG_PRINT("error", ("Wrong directory entry in data block")); + DBUG_RETURN(1); + } + DBUG_RETURN(_ma_read_block_record2(info, record, data, end_of_data)); +} + + +/* compare unique constraint between stored rows */ + +my_bool _ma_cmp_block_unique(MARIA_HA *info, MARIA_UNIQUEDEF *def, + const byte *record, MARIA_RECORD_POS pos) +{ + byte *org_rec_buff, *old_record; + my_size_t org_rec_buff_size; + int error; + DBUG_ENTER("_ma_cmp_block_unique"); + + if (!(old_record= my_alloca(info->s->base.reclength))) + DBUG_RETURN(1); + + /* Don't let the compare destroy blobs that may be in use */ + org_rec_buff= info->rec_buff; + org_rec_buff_size= info->rec_buff_size; + if (info->s->base.blobs) + { + /* Force realloc of record buffer*/ + info->rec_buff= 0; + info->rec_buff_size= 0; + } + error= _ma_read_block_record(info, old_record, pos); + if (!error) + error= _ma_unique_comp(def, record, old_record, def->null_are_equal); + if (info->s->base.blobs) + { + my_free(info->rec_buff, MYF(MY_ALLOW_ZERO_PTR)); + info->rec_buff= org_rec_buff; + info->rec_buff_size= org_rec_buff_size; + } + DBUG_PRINT("exit", ("result: %d", error)); + my_afree(old_record); + DBUG_RETURN(error != 0); +} + + +/**************************************************************************** + Table scan +****************************************************************************/ + +/* + Allocate buffers for table scan + + SYNOPSIS + _ma_scan_init_block_record(MARIA_HA *info) + + IMPLEMENTATION + We allocate one buffer for the current bitmap and one buffer for the + current page +*/ + +my_bool _ma_scan_init_block_record(MARIA_HA *info) +{ + byte *ptr; + if (!(ptr= (byte *) my_malloc(info->s->block_size * 2, MYF(MY_WME)))) + return (1); + info->scan.bitmap_buff= ptr; + info->scan.page_buff= ptr + info->s->block_size; + info->scan.bitmap_end= info->scan.bitmap_buff + info->s->bitmap.total_size; + + /* Set scan variables to get _ma_scan_block() to start with reading bitmap */ + info->scan.number_of_rows= 0; + info->scan.bitmap_pos= info->scan.bitmap_end; + info->scan.bitmap_page= (ulong) - (long) info->s->bitmap.pages_covered; + /* + We have to flush bitmap as we will read the bitmap from the page cache + while scanning rows + */ + return _ma_flush_bitmap(info->s); +} + + +/* Free buffers allocated by _ma_scan_block_init() */ + +void _ma_scan_end_block_record(MARIA_HA *info) +{ + my_free(info->scan.bitmap_buff, MYF(0)); + info->scan.bitmap_buff= 0; +} + + +/* + Read next record while scanning table + + SYNOPSIS + _ma_scan_block_record() + info Maria handler + record Store found here + record_pos Value stored in info->cur_row.next_pos after last call + skip_deleted + + NOTES + - One must have called mi_scan() before this + - In this version, we don't actually need record_pos, we as easily + use a variable in info->scan + + IMPLEMENTATION + Current code uses a lot of goto's to separate the different kind of + states we may be in. This gives us a minimum of executed if's for + the normal cases. I tried several different ways to code this, but + the current one was in the end the most readable and fastest. + + RETURN + 0 ok + # Error code +*/ + +int _ma_scan_block_record(MARIA_HA *info, byte *record, + MARIA_RECORD_POS record_pos, + my_bool skip_deleted __attribute__ ((unused))) +{ + uint block_size; + my_off_t filepos; + DBUG_ENTER("_ma_scan_block_record"); + +restart_record_read: + /* Find next row in current page */ + if (likely(record_pos < info->scan.number_of_rows)) + { + uint length, offset; + byte *data, *end_of_data; + + while (!(offset= uint2korr(info->scan.dir))) + { + info->scan.dir-= DIR_ENTRY_SIZE; + record_pos++; +#ifdef SANITY_CHECKS + if (info->scan.dir < info->scan.dir_end) + goto err; +#endif + } + /* found row */ + info->cur_row.lastpos= info->scan.row_base_page + record_pos; + info->cur_row.nextpos= record_pos + 1; + data= info->scan.page_buff + offset; + length= uint2korr(info->scan.dir + 2); + end_of_data= data + length; + info->scan.dir-= DIR_ENTRY_SIZE; /* Point to previous row */ +#ifdef SANITY_CHECKS + if (end_of_data > info->scan.dir_end || + offset < PAGE_HEADER_SIZE || length < info->s->base.min_block_length) + goto err; +#endif + DBUG_PRINT("info", ("rowid: %lu", (ulong) info->cur_row.lastpos)); + DBUG_RETURN(_ma_read_block_record2(info, record, data, end_of_data)); + } + + /* Find next head page in current bitmap */ +restart_bitmap_scan: + block_size= info->s->block_size; + if (likely(info->scan.bitmap_pos < info->scan.bitmap_end)) + { + byte *data= info->scan.bitmap_pos; + longlong bits= info->scan.bits; + uint bit_pos= info->scan.bit_pos; + + do + { + while (likely(bits)) + { + uint pattern= bits & 7; + bits >>= 3; + bit_pos++; + if (pattern > 0 && pattern <= 4) + { + /* Found head page; Read it */ + ulong page; + info->scan.bitmap_pos= data; + info->scan.bits= bits; + info->scan.bit_pos= bit_pos; + page= (info->scan.bitmap_page + 1 + + (data - info->scan.bitmap_buff) / 6 * 16 + bit_pos - 1); + info->scan.row_base_page= ma_recordpos(page, 0); + if (!(key_cache_read(info->s->key_cache, + info->dfile, + (my_off_t) page * block_size, + 0, info->scan.page_buff, + block_size, block_size, 0))) + DBUG_RETURN(my_errno); + if (((info->scan.page_buff[PAGE_TYPE_OFFSET] & PAGE_TYPE_MASK) != + HEAD_PAGE) || + (info->scan.number_of_rows= + (uint) (uchar) info->scan.page_buff[DIR_ENTRY_OFFSET]) == 0) + { + DBUG_PRINT("error", ("Wrong page header")); + DBUG_RETURN((my_errno= HA_ERR_WRONG_IN_RECORD)); + } + info->scan.dir= (info->scan.page_buff + block_size - + PAGE_SUFFIX_SIZE - DIR_ENTRY_SIZE); + info->scan.dir_end= (info->scan.dir - + (info->scan.number_of_rows - 1) * + DIR_ENTRY_SIZE); + record_pos= 0; + goto restart_record_read; + } + } + for (data+= 6; data < info->scan.bitmap_end; data+= 6) + { + bits= uint6korr(data); + if (bits && ((bits & LL(04444444444444444)) != LL(04444444444444444))) + break; + } + bit_pos= 0; + } while (data < info->scan.bitmap_end); + } + + /* Read next bitmap */ + info->scan.bitmap_page+= info->s->bitmap.pages_covered; + filepos= (my_off_t) info->scan.bitmap_page * block_size; + if (unlikely(filepos >= info->state->data_file_length)) + { + DBUG_RETURN((my_errno= HA_ERR_END_OF_FILE)); + } + if (!(key_cache_read(info->s->key_cache, info->dfile, filepos, + 0, info->scan.bitmap_buff, block_size, block_size, 0))) + DBUG_RETURN(my_errno); + /* Skip scanning 'bits' in bitmap scan code */ + info->scan.bitmap_pos= info->scan.bitmap_buff - 6; + info->scan.bits= 0; + goto restart_bitmap_scan; + +err: + DBUG_PRINT("error", ("Wrong data on page")); + DBUG_RETURN((my_errno= HA_ERR_WRONG_IN_RECORD)); +} + + +/* + Compare a row against a stored one + + NOTES + Not implemented, as block record is not supposed to be used in a shared + global environment +*/ + +my_bool _ma_compare_block_record(MARIA_HA *info __attribute__ ((unused)), + const byte *record __attribute__ ((unused))) +{ + return 0; +} + + +#ifndef DBUG_OFF + +static void _ma_print_directory(byte *buff, uint block_size) +{ + uint max_entry= (uint) ((uchar *) buff)[DIR_ENTRY_OFFSET], row= 0; + uint end_of_prev_row= PAGE_HEADER_SIZE; + byte *dir, *end; + + dir= buff + block_size - DIR_ENTRY_SIZE * max_entry - PAGE_SUFFIX_SIZE; + end= buff + block_size - DIR_ENTRY_SIZE - PAGE_SUFFIX_SIZE; + + DBUG_LOCK_FILE; + fprintf(DBUG_FILE,"Directory dump (pos:length):\n"); + + for (row= 1; dir <= end ; end-= DIR_ENTRY_SIZE, row++) + { + uint offset= uint2korr(end); + uint length= uint2korr(end+2); + fprintf(DBUG_FILE, " %4u:%4u", offset, offset ? length : 0); + if (!(row % (80/12))) + fputc('\n', DBUG_FILE); + if (offset) + { + DBUG_ASSERT(offset >= end_of_prev_row); + end_of_prev_row= offset + length; + } + } + fputc('\n', DBUG_FILE); + fflush(DBUG_FILE); + DBUG_UNLOCK_FILE; +} +#endif /* DBUG_OFF */ + diff --git a/storage/maria/ma_blockrec.h b/storage/maria/ma_blockrec.h new file mode 100644 index 00000000000..ec99dbfcae2 --- /dev/null +++ b/storage/maria/ma_blockrec.h @@ -0,0 +1,160 @@ +/* Copyright (C) 2007 Michael Widenius + + 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; either version 2 of the License, or + (at your option) any later version. + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ + +/* + Storage of records in block +*/ + +#define LSN_SIZE 7 +#define DIRCOUNT_SIZE 1 /* Stores number of rows on page */ +#define EMPTY_SPACE_SIZE 2 /* Stores empty space on page */ +#define PAGE_TYPE_SIZE 1 +#define PAGE_SUFFIX_SIZE 0 /* Bytes for page suffix */ +#define PAGE_HEADER_SIZE (LSN_SIZE + DIRCOUNT_SIZE + EMPTY_SPACE_SIZE +\ + PAGE_TYPE_SIZE) +#define PAGE_OVERHEAD_SIZE (PAGE_HEADER_SIZE + DIR_ENTRY_SIZE + \ + PAGE_SUFFIX_SIZE) +#define BLOCK_RECORD_POINTER_SIZE 6 + +#define FULL_PAGE_SIZE(block_size) ((block_size) - LSN_SIZE - PAGE_TYPE_SIZE) + +#define ROW_EXTENT_PAGE_SIZE 5 +#define ROW_EXTENT_COUNT_SIZE 2 +#define ROW_EXTENT_SIZE (ROW_EXTENT_PAGE_SIZE + ROW_EXTENT_COUNT_SIZE) +#define TAIL_BIT 0x8000 /* Bit in page_count to signify tail */ +#define ELEMENTS_RESERVED_FOR_MAIN_PART 4 +#define EXTRA_LENGTH_FIELDS 3 + +#define FLAG_SIZE 1 +#define TRANSID_SIZE 6 +#define VERPTR_SIZE 7 +#define DIR_ENTRY_SIZE 4 +#define FIELD_OFFSET_SIZE 2 + +/* Minimum header size needed for a new row */ +#define BASE_ROW_HEADER_SIZE FLAG_SIZE +#define TRANS_ROW_EXTRA_HEADER_SIZE TRANSID_SIZE + +#define PAGE_TYPE_MASK 127 +enum en_page_type { UNALLOCATED_PAGE, HEAD_PAGE, TAIL_PAGE, BLOB_PAGE, MAX_PAGE_TYPE }; + +#define PAGE_TYPE_OFFSET LSN_SIZE +#define DIR_ENTRY_OFFSET LSN_SIZE+PAGE_TYPE_SIZE +#define EMPTY_SPACE_OFFSET (DIR_ENTRY_OFFSET + DIRCOUNT_SIZE) + +#define PAGE_CAN_BE_COMPACTED 128 /* Bit in PAGE_TYPE */ + +/* Bits used for flag byte (one byte, first in record) */ +#define ROW_FLAG_TRANSID 1 +#define ROW_FLAG_VER_PTR 2 +#define ROW_FLAG_DELETE_TRANSID 4 +#define ROW_FLAG_NULLS_EXTENDED 8 +#define ROW_FLAG_EXTENTS 128 +#define ROW_FLAG_ALL (1+2+4+8+128) + +/* Variables that affects how data pages are utilized */ +#define MIN_TAIL_SIZE 32 + +/* Fixed part of Max possible header size; See table in ma_blockrec.c */ +#define MAX_FIXED_HEADER_SIZE (FLAG_SIZE + 3 + ROW_EXTENT_SIZE + 3) +#define TRANS_MAX_FIXED_HEADER_SIZE (MAX_FIXED_HEADER_SIZE + \ + FLAG_SIZE + TRANSID_SIZE + VERPTR_SIZE + \ + TRANSID_SIZE) + +/* We use 1 byte in record header to store number of directory entries */ +#define MAX_ROWS_PER_PAGE 255 + +/* Bits for MARIA_BITMAP_BLOCKS->used */ +#define BLOCKUSED_USED 1 +#define BLOCKUSED_USE_ORG_BITMAP 2 +#define BLOCKUSED_TAIL 4 + +/* defines that affects allocation (density) of data */ + +/* If we fill up a block to 75 %, don't create a new tail page for it */ +#define MAX_TAIL_SIZE(block_size) ((block_size) *3 / 4) + +/* Functions to convert MARIA_RECORD_POS to/from page:offset */ + +static inline MARIA_RECORD_POS ma_recordpos(ulonglong page, uint offset) +{ + return (MARIA_RECORD_POS) ((page << 8) | offset); +} + +static inline my_off_t ma_recordpos_to_page(MARIA_RECORD_POS record_pos) +{ + return record_pos >> 8; +} + +static inline my_off_t ma_recordpos_to_offset(MARIA_RECORD_POS record_pos) +{ + return record_pos & 255; +} + +/* ma_blockrec.c */ +void _ma_init_block_record_data(void); +my_bool _ma_once_init_block_row(MARIA_SHARE *share, File dfile); +my_bool _ma_once_end_block_row(MARIA_SHARE *share); +my_bool _ma_init_block_row(MARIA_HA *info); +void _ma_end_block_row(MARIA_HA *info); + +my_bool _ma_update_block_record(MARIA_HA *info, MARIA_RECORD_POS pos, + const byte *record); +my_bool _ma_delete_block_record(MARIA_HA *info); +int _ma_read_block_record(MARIA_HA *info, byte *record, + MARIA_RECORD_POS record_pos); +int _ma_read_block_record2(MARIA_HA *info, byte *record, + byte *data, byte *end_of_data); +int _ma_scan_block_record(MARIA_HA *info, byte *record, + MARIA_RECORD_POS, my_bool); +my_bool _ma_cmp_block_unique(MARIA_HA *info, MARIA_UNIQUEDEF *def, + const byte *record, MARIA_RECORD_POS pos); +my_bool _ma_scan_init_block_record(MARIA_HA *info); +void _ma_scan_end_block_record(MARIA_HA *info); + +MARIA_RECORD_POS _ma_write_init_block_record(MARIA_HA *info, + const byte *record); +my_bool _ma_write_block_record(MARIA_HA *info, const byte *record); +my_bool _ma_write_abort_block_record(MARIA_HA *info); +my_bool _ma_compare_block_record(register MARIA_HA *info, + register const byte *record); + +/* ma_bitmap.c */ +my_bool _ma_bitmap_init(MARIA_SHARE *share, File file); +my_bool _ma_bitmap_end(MARIA_SHARE *share); +my_bool _ma_flush_bitmap(MARIA_SHARE *share); +my_bool _ma_read_bitmap_page(MARIA_SHARE *share, MARIA_FILE_BITMAP *bitmap, + ulonglong page); +my_bool _ma_bitmap_find_place(MARIA_HA *info, MARIA_ROW *row, + MARIA_BITMAP_BLOCKS *result_blocks); +my_bool _ma_bitmap_release_unused(MARIA_HA *info, MARIA_BITMAP_BLOCKS *blocks); +my_bool _ma_bitmap_free_full_pages(MARIA_HA *info, const byte *extents, + uint count); +my_bool _ma_bitmap_set(MARIA_HA *info, ulonglong pos, my_bool head, + uint empty_space); +my_bool _ma_reset_full_page_bits(MARIA_HA *info, MARIA_FILE_BITMAP *bitmap, + ulonglong page, uint page_count); +uint _ma_free_size_to_head_pattern(MARIA_FILE_BITMAP *bitmap, uint size); +my_bool _ma_bitmap_find_new_place(MARIA_HA *info, MARIA_ROW *new_row, + ulonglong page, uint free_size, + MARIA_BITMAP_BLOCKS *result_blocks); +my_bool _ma_check_bitmap_data(MARIA_HA *info, + enum en_page_type page_type, ulonglong page, + uint empty_space, uint *bitmap_pattern); +my_bool _ma_check_if_right_bitmap_type(MARIA_HA *info, + enum en_page_type page_type, + ulonglong page, + uint *bitmap_pattern); diff --git a/storage/maria/ma_check.c b/storage/maria/ma_check.c index 69d863e6366..3fece3687e1 100644 --- a/storage/maria/ma_check.c +++ b/storage/maria/ma_check.c @@ -28,43 +28,43 @@ #include #endif #include "ma_rt_index.h" +#include "ma_blockrec.h" -#ifndef USE_RAID -#define my_raid_create(A,B,C,D,E,F,G) my_create(A,B,C,G) -#define my_raid_delete(A,B,C) my_delete(A,B) -#endif - - /* Functions defined in this file */ +/* Functions defined in this file */ -static int check_k_link(HA_CHECK *param, MARIA_HA *info,uint nr); +static int check_k_link(HA_CHECK *param, MARIA_HA *info, my_off_t next_link); static int chk_index(HA_CHECK *param, MARIA_HA *info,MARIA_KEYDEF *keyinfo, - my_off_t page, uchar *buff, ha_rows *keys, + my_off_t page, byte *buff, ha_rows *keys, ha_checksum *key_checksum, uint level); static uint isam_key_length(MARIA_HA *info,MARIA_KEYDEF *keyinfo); static ha_checksum calc_checksum(ha_rows count); static int writekeys(HA_CHECK *param, MARIA_HA *info,byte *buff, my_off_t filepos); -static int sort_one_index(HA_CHECK *param, MARIA_HA *info,MARIA_KEYDEF *keyinfo, +static int sort_one_index(HA_CHECK *param, MARIA_HA *info, + MARIA_KEYDEF *keyinfo, my_off_t pagepos, File new_file); -static int sort_key_read(MARIA_SORT_PARAM *sort_param,void *key); -static int sort_maria_ft_key_read(MARIA_SORT_PARAM *sort_param,void *key); +static int sort_key_read(MARIA_SORT_PARAM *sort_param, byte *key); +static int sort_maria_ft_key_read(MARIA_SORT_PARAM *sort_param, byte *key); static int sort_get_next_record(MARIA_SORT_PARAM *sort_param); -static int sort_key_cmp(MARIA_SORT_PARAM *sort_param, const void *a,const void *b); -static int sort_maria_ft_key_write(MARIA_SORT_PARAM *sort_param, const void *a); -static int sort_key_write(MARIA_SORT_PARAM *sort_param, const void *a); +static int sort_key_cmp(MARIA_SORT_PARAM *sort_param, const void *a, + const void *b); +static int sort_maria_ft_key_write(MARIA_SORT_PARAM *sort_param, + const byte *a); +static int sort_key_write(MARIA_SORT_PARAM *sort_param, const byte *a); static my_off_t get_record_for_key(MARIA_HA *info,MARIA_KEYDEF *keyinfo, - uchar *key); + const byte *key); static int sort_insert_key(MARIA_SORT_PARAM *sort_param, reg1 SORT_KEY_BLOCKS *key_block, - uchar *key, my_off_t prev_block); + const byte *key, my_off_t prev_block); static int sort_delete_record(MARIA_SORT_PARAM *sort_param); /*static int _ma_flush_pending_blocks(HA_CHECK *param);*/ static SORT_KEY_BLOCKS *alloc_key_blocks(HA_CHECK *param, uint blocks, uint buffer_length); static ha_checksum maria_byte_checksum(const byte *buf, uint length); static void set_data_file_type(MARIA_SORT_INFO *sort_info, MARIA_SHARE *share); +static void restore_data_file_type(MARIA_SHARE *share); -void mariachk_init(HA_CHECK *param) +void maria_chk_init(HA_CHECK *param) { bzero((gptr) param,sizeof(*param)); param->opt_follow_links=1; @@ -182,7 +182,7 @@ int maria_chk_del(HA_CHECK *param, register MARIA_HA *info, uint test_flag) else { param->record_checksum+=(ha_checksum) next_link; - next_link= _ma_rec_pos(info->s,(uchar*) buff+1); + next_link= _ma_rec_pos(info->s, buff+1); empty+=info->s->base.pack_reclength; } } @@ -223,18 +223,14 @@ wrong: /* Check delete links in index file */ -static int check_k_link(HA_CHECK *param, register MARIA_HA *info, uint nr) +static int check_k_link(HA_CHECK *param, register MARIA_HA *info, + my_off_t next_link) { - my_off_t next_link; - uint block_size=(nr+1)*MARIA_MIN_KEY_BLOCK_LENGTH; + uint block_size= info->s->block_size; ha_rows records; char llbuff[21],*buff; DBUG_ENTER("check_k_link"); - if (param->testflag & T_VERBOSE) - printf("block_size %4d:",block_size); - - next_link=info->s->state.key_del[nr]; records= (ha_rows) (info->state->key_file_length / block_size); while (next_link != HA_OFFSET_ERROR && records > 0) { @@ -243,12 +239,12 @@ static int check_k_link(HA_CHECK *param, register MARIA_HA *info, uint nr) if (param->testflag & T_VERBOSE) printf("%16s",llstr(next_link,llbuff)); if (next_link > info->state->key_file_length || - next_link & (info->s->blocksize-1)) + next_link & (info->s->block_size-1)) DBUG_RETURN(1); if (!(buff=key_cache_read(info->s->key_cache, info->s->kfile, next_link, DFLT_INIT_HITS, (byte*) info->buff, - maria_block_size, block_size, 1))) + block_size, block_size, 1))) DBUG_RETURN(1); next_link=mi_sizekorr(buff); records--; @@ -274,9 +270,10 @@ int maria_chk_size(HA_CHECK *param, register MARIA_HA *info) char buff[22],buff2[22]; DBUG_ENTER("maria_chk_size"); - if (!(param->testflag & T_SILENT)) puts("- check file-size"); + if (!(param->testflag & T_SILENT)) + puts("- check file-size"); - /* The following is needed if called externally (not from mariachk) */ + /* The following is needed if called externally (not from maria_chk) */ flush_key_blocks(info->s->key_cache, info->s->kfile, FLUSH_FORCE_WRITE); @@ -291,7 +288,7 @@ int maria_chk_size(HA_CHECK *param, register MARIA_HA *info) "Size of indexfile is: %-8s Should be: %s", llstr(size,buff), llstr(skr,buff2)); } - else + else if (!(param->testflag & T_VERY_SILENT)) _ma_check_print_warning(param, "Size of indexfile is: %-8s Should be: %s", llstr(size,buff), llstr(skr,buff2)); @@ -341,7 +338,7 @@ int maria_chk_size(HA_CHECK *param, register MARIA_HA *info) } /* maria_chk_size */ - /* Check keys */ +/* Check keys */ int maria_chk_key(HA_CHECK *param, register MARIA_HA *info) { @@ -359,23 +356,22 @@ int maria_chk_key(HA_CHECK *param, register MARIA_HA *info) puts("- check key delete-chain"); param->key_file_blocks=info->s->base.keystart; - for (key=0 ; key < info->s->state.header.max_block_size_index ; key++) - if (check_k_link(param,info,key)) - { - if (param->testflag & T_VERBOSE) puts(""); - _ma_check_print_error(param,"key delete-link-chain corrupted"); - DBUG_RETURN(-1); - } + if (check_k_link(param, info, info->s->state.key_del)) + { + if (param->testflag & T_VERBOSE) puts(""); + _ma_check_print_error(param,"key delete-link-chain corrupted"); + DBUG_RETURN(-1); + } if (!(param->testflag & T_SILENT)) puts("- check index reference"); all_keydata=all_totaldata=key_totlength=0; old_record_checksum=0; init_checksum=param->record_checksum; - if (!(share->options & - (HA_OPTION_PACK_RECORD | HA_OPTION_COMPRESS_RECORD))) - old_record_checksum=calc_checksum(info->state->records+info->state->del-1)* - share->base.pack_reclength; + if (share->data_file_type == STATIC_RECORD) + old_record_checksum= (calc_checksum(info->state->records + + info->state->del-1) * + share->base.pack_reclength); rec_per_key_part= param->rec_per_key_part; for (key= 0,keyinfo= &share->keyinfo[0]; key < share->base.keys ; rec_per_key_part+=keyinfo->keysegs, key++, keyinfo++) @@ -433,11 +429,10 @@ int maria_chk_key(HA_CHECK *param, register MARIA_HA *info) result= -1; continue; } - if (found_keys - full_text_keys == 1 && - ((share->options & - (HA_OPTION_PACK_RECORD | HA_OPTION_COMPRESS_RECORD)) || - (param->testflag & T_DONT_CHECK_CHECKSUM))) - old_record_checksum=param->record_checksum; + if ((found_keys - full_text_keys == 1 && + !(share->data_file_type == STATIC_RECORD)) || + (param->testflag & T_DONT_CHECK_CHECKSUM)) + old_record_checksum= param->record_checksum; else if (old_record_checksum != param->record_checksum) { if (key) @@ -456,7 +451,7 @@ int maria_chk_key(HA_CHECK *param, register MARIA_HA *info) /* Check that auto_increment key is bigger than max key value */ ulonglong auto_increment; info->lastinx=key; - _ma_read_key_record(info, 0L, info->rec_buff); + _ma_read_key_record(info, info->rec_buff, 0); auto_increment= ma_retrieve_auto_increment(info, info->rec_buff); if (auto_increment > info->s->state.auto_increment) { @@ -479,7 +474,7 @@ int maria_chk_key(HA_CHECK *param, register MARIA_HA *info) if (!maria_rkey(info, info->rec_buff, key, (const byte*) info->lastkey, keyinfo->seg->length, HA_READ_KEY_EXACT)) { - /* Don't count this as a real warning, as mariachk can't correct it */ + /* Don't count this as a real warning, as maria_chk can't correct it */ uint save=param->warning_printed; _ma_check_print_warning(param, "Found row where the auto_increment column has the value 0"); @@ -528,21 +523,22 @@ do_stat: } /* maria_chk_key */ -static int chk_index_down(HA_CHECK *param, MARIA_HA *info, MARIA_KEYDEF *keyinfo, - my_off_t page, uchar *buff, ha_rows *keys, - ha_checksum *key_checksum, uint level) +static int chk_index_down(HA_CHECK *param, MARIA_HA *info, + MARIA_KEYDEF *keyinfo, + my_off_t page, byte *buff, ha_rows *keys, + ha_checksum *key_checksum, uint level) { char llbuff[22],llbuff2[22]; - if (page > info->state->key_file_length || (page & (info->s->blocksize -1))) + if (page > info->state->key_file_length || (page & (info->s->block_size -1))) { my_off_t max_length=my_seek(info->s->kfile,0L,MY_SEEK_END,MYF(0)); _ma_check_print_error(param,"Wrong pagepointer: %s at page: %s", llstr(page,llbuff),llstr(page,llbuff2)); - if (page+info->s->blocksize > max_length) + if (page+info->s->block_size > max_length) goto err; info->state->key_file_length=(max_length & - ~ (my_off_t) (info->s->blocksize-1)); + ~ (my_off_t) (info->s->block_size-1)); } if (!_ma_fetch_keypage(info,keyinfo,page, DFLT_INIT_HITS,buff,0)) { @@ -577,10 +573,10 @@ err: static void maria_collect_stats_nonulls_first(HA_KEYSEG *keyseg, ulonglong *notnull, - uchar *key) + const byte *key) { uint first_null, kp; - first_null= ha_find_null(keyseg, key) - keyseg; + first_null= ha_find_null(keyseg, (uchar*) key) - keyseg; /* All prefix tuples that don't include keypart_{first_null} are not-null tuples (and all others aren't), increment counters for them. @@ -617,7 +613,8 @@ void maria_collect_stats_nonulls_first(HA_KEYSEG *keyseg, ulonglong *notnull, static int maria_collect_stats_nonulls_next(HA_KEYSEG *keyseg, ulonglong *notnull, - uchar *prev_key, uchar *last_key) + const byte *prev_key, + const byte *last_key) { uint diffs[2]; uint first_null_seg, kp; @@ -631,12 +628,12 @@ int maria_collect_stats_nonulls_next(HA_KEYSEG *keyseg, ulonglong *notnull, last_key that is NULL or different from corresponding value in prev_key. */ - ha_key_cmp(keyseg, prev_key, last_key, USE_WHOLE_KEY, + ha_key_cmp(keyseg, (uchar*) prev_key, (uchar*) last_key, USE_WHOLE_KEY, SEARCH_FIND | SEARCH_NULL_ARE_NOT_EQUAL, diffs); seg= keyseg + diffs[0] - 1; /* Find first NULL in last_key */ - first_null_seg= ha_find_null(seg, last_key + diffs[1]) - keyseg; + first_null_seg= ha_find_null(seg, (uchar*) last_key + diffs[1]) - keyseg; for (kp= 0; kp < first_null_seg; kp++) notnull[kp]++; @@ -652,12 +649,12 @@ int maria_collect_stats_nonulls_next(HA_KEYSEG *keyseg, ulonglong *notnull, /* Check if index is ok */ static int chk_index(HA_CHECK *param, MARIA_HA *info, MARIA_KEYDEF *keyinfo, - my_off_t page, uchar *buff, ha_rows *keys, + my_off_t page, byte *buff, ha_rows *keys, ha_checksum *key_checksum, uint level) { int flag; uint used_length,comp_flag,nod_flag,key_length=0; - uchar key[HA_MAX_POSSIBLE_KEY_BUFF],*temp_buff,*keypos,*old_keypos,*endpos; + byte key[HA_MAX_POSSIBLE_KEY_BUFF],*temp_buff,*keypos,*old_keypos,*endpos; my_off_t next_page,record; char llbuff[22]; uint diff_pos[2]; @@ -668,7 +665,7 @@ static int chk_index(HA_CHECK *param, MARIA_HA *info, MARIA_KEYDEF *keyinfo, if (keyinfo->flag & HA_SPATIAL) DBUG_RETURN(0); - if (!(temp_buff=(uchar*) my_alloca((uint) keyinfo->block_length))) + if (!(temp_buff=(byte*) my_alloca((uint) keyinfo->block_length))) { _ma_check_print_error(param,"Not enough memory for keyblock"); DBUG_RETURN(-1); @@ -698,8 +695,8 @@ static int chk_index(HA_CHECK *param, MARIA_HA *info, MARIA_KEYDEF *keyinfo, { if (*_ma_killed_ptr(param)) goto err; - memcpy((char*) info->lastkey,(char*) key,key_length); - info->lastkey_length=key_length; + memcpy(info->lastkey, key, key_length); + info->lastkey_length= key_length; if (nod_flag) { next_page= _ma_kpos(nod_flag,keypos); @@ -713,21 +710,24 @@ static int chk_index(HA_CHECK *param, MARIA_HA *info, MARIA_KEYDEF *keyinfo, break; if (keypos > endpos) { - _ma_check_print_error(param,"Wrong key block length at page: %s",llstr(page,llbuff)); + _ma_check_print_error(param,"Wrong key block length at page: %s", + llstr(page,llbuff)); goto err; } if ((*keys)++ && - (flag=ha_key_cmp(keyinfo->seg,info->lastkey,key,key_length, - comp_flag, diff_pos)) >=0) + (flag=ha_key_cmp(keyinfo->seg, (uchar*) info->lastkey, (uchar*) key, + key_length, comp_flag, diff_pos)) >=0) { - DBUG_DUMP("old",(byte*) info->lastkey, info->lastkey_length); - DBUG_DUMP("new",(byte*) key, key_length); - DBUG_DUMP("new_in_page",(char*) old_keypos,(uint) (keypos-old_keypos)); + DBUG_DUMP("old", info->lastkey, info->lastkey_length); + DBUG_DUMP("new", key, key_length); + DBUG_DUMP("new_in_page", old_keypos, (uint) (keypos-old_keypos)); if (comp_flag & SEARCH_FIND && flag == 0) - _ma_check_print_error(param,"Found duplicated key at page %s",llstr(page,llbuff)); + _ma_check_print_error(param,"Found duplicated key at page %s", + llstr(page,llbuff)); else - _ma_check_print_error(param,"Key in wrong position at page %s",llstr(page,llbuff)); + _ma_check_print_error(param,"Key in wrong position at page %s", + llstr(page,llbuff)); goto err; } if (param->testflag & T_STATISTICS) @@ -735,14 +735,14 @@ static int chk_index(HA_CHECK *param, MARIA_HA *info, MARIA_KEYDEF *keyinfo, if (*keys != 1L) /* not first_key */ { if (param->stats_method == MI_STATS_METHOD_NULLS_NOT_EQUAL) - ha_key_cmp(keyinfo->seg,info->lastkey,key,USE_WHOLE_KEY, - SEARCH_FIND | SEARCH_NULL_ARE_NOT_EQUAL, + ha_key_cmp(keyinfo->seg, (uchar*) info->lastkey, (uchar*) key, + USE_WHOLE_KEY, SEARCH_FIND | SEARCH_NULL_ARE_NOT_EQUAL, diff_pos); else if (param->stats_method == MI_STATS_METHOD_IGNORE_NULLS) { diff_pos[0]= maria_collect_stats_nonulls_next(keyinfo->seg, - param->notnull_count, - info->lastkey, key); + param->notnull_count, + info->lastkey, key); } param->unique_count[diff_pos[0]-1]++; } @@ -795,7 +795,7 @@ static int chk_index(HA_CHECK *param, MARIA_HA *info, MARIA_KEYDEF *keyinfo, DBUG_DUMP("new_in_page",(char*) old_keypos,(uint) (keypos-old_keypos)); goto err; } - param->record_checksum+=(ha_checksum) record; + param->record_checksum+= (ha_checksum) record; } if (keypos != endpos) { @@ -852,355 +852,844 @@ static uint isam_key_length(MARIA_HA *info, register MARIA_KEYDEF *keyinfo) } /* key_length */ - /* Check that record-link is ok */ -int maria_chk_data_link(HA_CHECK *param, MARIA_HA *info,int extend) +static void record_pos_to_txt(MARIA_HA *info, my_off_t recpos, + char *buff) { - int error,got_error,flag; - uint key,left_length,b_type,field; - ha_rows records,del_blocks; - my_off_t used,empty,pos,splits,start_recpos, - del_length,link_used,start_block; - byte *record,*to; - char llbuff[22],llbuff2[22],llbuff3[22]; - ha_checksum intern_record_checksum; - ha_checksum key_checksum[HA_MAX_POSSIBLE_KEY]; - my_bool static_row_size; - MARIA_KEYDEF *keyinfo; - MARIA_BLOCK_INFO block_info; - DBUG_ENTER("maria_chk_data_link"); - - if (!(param->testflag & T_SILENT)) + if (info->s->data_file_type != BLOCK_RECORD) + llstr(recpos, buff); + else { - if (extend) - puts("- check records and index references"); - else - puts("- check record links"); + my_off_t page= ma_recordpos_to_page(recpos); + uint row= ma_recordpos_to_offset(recpos); + char *end= longlong10_to_str(page, buff, 10); + *(end++)= ':'; + longlong10_to_str(row, end, 10); } +} - if (!(record= (byte*) my_malloc(info->s->base.pack_reclength,MYF(0)))) + +/* + Check that keys in records exist in index tree + + SYNOPSIS + check_keys_in_record() + param Check paramenter + info Maria handler + extend Type of check (extended or normal) + start_recpos Position to row + record Record buffer + + NOTES + This function also calculates record checksum & number of rows +*/ + +static int check_keys_in_record(HA_CHECK *param, MARIA_HA *info, int extend, + my_off_t start_recpos, byte *record) +{ + MARIA_KEYDEF *keyinfo; + char llbuff[22+4]; + uint key; + + param->tmp_record_checksum+= (ha_checksum) start_recpos; + param->records++; + if (param->testflag & T_WRITE_LOOP && param->records % WRITE_COUNT == 0) { - _ma_check_print_error(param,"Not enough memory for record"); - DBUG_RETURN(-1); + printf("%s\r", llstr(param->records, llbuff)); + VOID(fflush(stdout)); } - records=del_blocks=0; - used=link_used=splits=del_length=0; - intern_record_checksum=param->glob_crc=0; - LINT_INIT(left_length); LINT_INIT(start_recpos); LINT_INIT(to); - got_error=error=0; - empty=info->s->pack.header_length; - /* Check how to calculate checksum of rows */ - static_row_size=1; - if (info->s->data_file_type == COMPRESSED_RECORD) + /* Check if keys match the record */ + for (key=0, keyinfo= info->s->keyinfo; key < info->s->base.keys; + key++,keyinfo++) { - for (field=0 ; field < info->s->base.fields ; field++) + if (maria_is_key_active(info->s->state.key_map, key)) { - if (info->s->rec[field].base_type == FIELD_BLOB || - info->s->rec[field].base_type == FIELD_VARCHAR) + if(!(keyinfo->flag & HA_FULLTEXT)) { - static_row_size=0; - break; + uint key_length= _ma_make_key(info,key,info->lastkey,record, + start_recpos); + if (extend) + { + /* We don't need to lock the key tree here as we don't allow + concurrent threads when running maria_chk + */ + int search_result= +#ifdef HAVE_RTREE_KEYS + (keyinfo->flag & HA_SPATIAL) ? + maria_rtree_find_first(info, key, info->lastkey, key_length, + MBR_EQUAL | MBR_DATA) : +#endif + _ma_search(info,keyinfo,info->lastkey,key_length, + SEARCH_SAME, info->s->state.key_root[key]); + if (search_result) + { + record_pos_to_txt(info, start_recpos, llbuff); + _ma_check_print_error(param,"Record at: %14s Can't find key for index: %2d", + llbuff,key+1); + if (param->err_count++ > MAXERR || !(param->testflag & T_VERBOSE)) + return -1; + } + } + else + param->tmp_key_crc[key]+= + maria_byte_checksum((byte*) info->lastkey, key_length); } } } + return 0; +} + + +/* + Functions to loop through all rows and check if they are ok + + NOTES + One function for each record format - pos=my_b_tell(¶m->read_cache); - bzero((char*) key_checksum, info->s->base.keys * sizeof(key_checksum[0])); + RESULT + 0 ok + -1 Interrupted by user + 1 Error +*/ + +static int check_static_record(HA_CHECK *param, MARIA_HA *info, int extend, + byte *record) +{ + my_off_t start_recpos, pos; + char llbuff[22]; + + pos= 0; while (pos < info->state->data_file_length) { if (*_ma_killed_ptr(param)) - goto err2; - switch (info->s->data_file_type) { - case STATIC_RECORD: - if (my_b_read(¶m->read_cache,(byte*) record, - info->s->base.pack_reclength)) - goto err; - start_recpos=pos; - pos+=info->s->base.pack_reclength; - splits++; - if (*record == '\0') + return -1; + if (my_b_read(¶m->read_cache,(byte*) record, + info->s->base.pack_reclength)) + { + _ma_check_print_error(param, + "got error: %d when reading datafile at position: %s", + my_errno, llstr(pos, llbuff)); + return 1; + } + start_recpos= pos; + pos+= info->s->base.pack_reclength; + param->splits++; + if (*record == '\0') + { + param->del_blocks++; + param->del_length+= info->s->base.pack_reclength; + continue; /* Record removed */ + } + param->glob_crc+= _ma_static_checksum(info,record); + param->used+= info->s->base.pack_reclength; + if (check_keys_in_record(param, info, extend, start_recpos, record)) + return 1; + } + return 0; +} + + +static int check_dynamic_record(HA_CHECK *param, MARIA_HA *info, int extend, + byte *record) +{ + MARIA_BLOCK_INFO block_info; + my_off_t start_recpos, start_block, pos; + byte *to; + ulong left_length; + uint b_type; + char llbuff[22],llbuff2[22],llbuff3[22]; + DBUG_ENTER("check_dynamic_record"); + + pos= 0; + while (pos < info->state->data_file_length) + { + my_bool got_error= 0; + int flag; + if (*_ma_killed_ptr(param)) + DBUG_RETURN(-1); + + flag= block_info.second_read=0; + block_info.next_filepos=pos; + do + { + if (_ma_read_cache(¶m->read_cache,(byte*) block_info.header, + (start_block=block_info.next_filepos), + sizeof(block_info.header), + (flag ? 0 : READING_NEXT) | READING_HEADER)) { - del_blocks++; - del_length+=info->s->base.pack_reclength; - continue; /* Record removed */ + _ma_check_print_error(param, + "got error: %d when reading datafile at position: %s", + my_errno, llstr(start_block, llbuff)); + DBUG_RETURN(1); } - param->glob_crc+= _ma_static_checksum(info,record); - used+=info->s->base.pack_reclength; - break; - case DYNAMIC_RECORD: - flag=block_info.second_read=0; - block_info.next_filepos=pos; - do + + if (start_block & (MARIA_DYN_ALIGN_SIZE-1)) { - if (_ma_read_cache(¶m->read_cache,(byte*) block_info.header, - (start_block=block_info.next_filepos), - sizeof(block_info.header), - (flag ? 0 : READING_NEXT) | READING_HEADER)) - goto err; - if (start_block & (MARIA_DYN_ALIGN_SIZE-1)) - { - _ma_check_print_error(param,"Wrong aligned block at %s", - llstr(start_block,llbuff)); - goto err2; - } - b_type= _ma_get_block_info(&block_info,-1,start_block); - if (b_type & (BLOCK_DELETED | BLOCK_ERROR | BLOCK_SYNC_ERROR | - BLOCK_FATAL_ERROR)) - { - if (b_type & BLOCK_SYNC_ERROR) - { - if (flag) - { - _ma_check_print_error(param,"Unexpected byte: %d at link: %s", - (int) block_info.header[0], - llstr(start_block,llbuff)); - goto err2; - } - pos=block_info.filepos+block_info.block_len; - goto next; - } - if (b_type & BLOCK_DELETED) - { - if (block_info.block_len < info->s->base.min_block_length) - { - _ma_check_print_error(param, - "Deleted block with impossible length %lu at %s", - block_info.block_len,llstr(pos,llbuff)); - goto err2; - } - if ((block_info.next_filepos != HA_OFFSET_ERROR && - block_info.next_filepos >= info->state->data_file_length) || - (block_info.prev_filepos != HA_OFFSET_ERROR && - block_info.prev_filepos >= info->state->data_file_length)) - { - _ma_check_print_error(param,"Delete link points outside datafile at %s", - llstr(pos,llbuff)); - goto err2; - } - del_blocks++; - del_length+=block_info.block_len; - pos=block_info.filepos+block_info.block_len; - splits++; - goto next; - } - _ma_check_print_error(param,"Wrong bytesec: %d-%d-%d at linkstart: %s", - block_info.header[0],block_info.header[1], - block_info.header[2], - llstr(start_block,llbuff)); - goto err2; - } - if (info->state->data_file_length < block_info.filepos+ - block_info.block_len) - { - _ma_check_print_error(param, - "Recordlink that points outside datafile at %s", - llstr(pos,llbuff)); - got_error=1; - break; - } - splits++; - if (!flag++) /* First block */ - { - start_recpos=pos; - pos=block_info.filepos+block_info.block_len; - if (block_info.rec_len > (uint) info->s->base.max_pack_length) - { - _ma_check_print_error(param,"Found too long record (%lu) at %s", - (ulong) block_info.rec_len, - llstr(start_recpos,llbuff)); - got_error=1; - break; - } - if (info->s->base.blobs) - { - if (!(to= _ma_alloc_rec_buff(info, block_info.rec_len, - &info->rec_buff))) - { - _ma_check_print_error(param, - "Not enough memory (%lu) for blob at %s", - (ulong) block_info.rec_len, - llstr(start_recpos,llbuff)); - got_error=1; - break; - } - } - else - to= info->rec_buff; - left_length=block_info.rec_len; - } - if (left_length < block_info.data_len) - { - _ma_check_print_error(param,"Found too long record (%lu) at %s", - (ulong) block_info.data_len, - llstr(start_recpos,llbuff)); - got_error=1; - break; - } - if (_ma_read_cache(¶m->read_cache,(byte*) to,block_info.filepos, - (uint) block_info.data_len, - flag == 1 ? READING_NEXT : 0)) - goto err; - to+=block_info.data_len; - link_used+= block_info.filepos-start_block; - used+= block_info.filepos - start_block + block_info.data_len; - empty+=block_info.block_len-block_info.data_len; - left_length-=block_info.data_len; - if (left_length) - { - if (b_type & BLOCK_LAST) - { - _ma_check_print_error(param, - "Wrong record length %s of %s at %s", - llstr(block_info.rec_len-left_length,llbuff), - llstr(block_info.rec_len, llbuff2), - llstr(start_recpos,llbuff3)); - got_error=1; - break; - } - if (info->state->data_file_length < block_info.next_filepos) - { - _ma_check_print_error(param, - "Found next-recordlink that points outside datafile at %s", - llstr(block_info.filepos,llbuff)); - got_error=1; - break; - } - } - } while (left_length); - if (! got_error) + _ma_check_print_error(param,"Wrong aligned block at %s", + llstr(start_block,llbuff)); + DBUG_RETURN(1); + } + b_type= _ma_get_block_info(&block_info,-1,start_block); + if (b_type & (BLOCK_DELETED | BLOCK_ERROR | BLOCK_SYNC_ERROR | + BLOCK_FATAL_ERROR)) { - if (_ma_rec_unpack(info,record,info->rec_buff,block_info.rec_len) == - MY_FILE_ERROR) - { - _ma_check_print_error(param,"Found wrong record at %s", - llstr(start_recpos,llbuff)); - got_error=1; - } - else - { - info->checksum=_ma_checksum(info,record); - if (param->testflag & (T_EXTEND | T_MEDIUM | T_VERBOSE)) - { - if (_ma_rec_check(info,record, info->rec_buff,block_info.rec_len, - test(info->s->calc_checksum))) - { - _ma_check_print_error(param,"Found wrong packed record at %s", - llstr(start_recpos,llbuff)); - got_error=1; - } - } - if (!got_error) - param->glob_crc+= info->checksum; - } + if (b_type & BLOCK_SYNC_ERROR) + { + if (flag) + { + _ma_check_print_error(param,"Unexpected byte: %d at link: %s", + (int) block_info.header[0], + llstr(start_block,llbuff)); + DBUG_RETURN(1); + } + pos=block_info.filepos+block_info.block_len; + goto next; + } + if (b_type & BLOCK_DELETED) + { + if (block_info.block_len < info->s->base.min_block_length) + { + _ma_check_print_error(param, + "Deleted block with impossible length %lu at %s", + block_info.block_len,llstr(pos,llbuff)); + DBUG_RETURN(1); + } + if ((block_info.next_filepos != HA_OFFSET_ERROR && + block_info.next_filepos >= info->state->data_file_length) || + (block_info.prev_filepos != HA_OFFSET_ERROR && + block_info.prev_filepos >= info->state->data_file_length)) + { + _ma_check_print_error(param,"Delete link points outside datafile at %s", + llstr(pos,llbuff)); + DBUG_RETURN(1); + } + param->del_blocks++; + param->del_length+= block_info.block_len; + param->splits++; + pos= block_info.filepos+block_info.block_len; + goto next; + } + _ma_check_print_error(param,"Wrong bytesec: %d-%d-%d at linkstart: %s", + block_info.header[0],block_info.header[1], + block_info.header[2], + llstr(start_block,llbuff)); + DBUG_RETURN(1); } - else if (!flag) - pos=block_info.filepos+block_info.block_len; - break; - case COMPRESSED_RECORD: - if (_ma_read_cache(¶m->read_cache,(byte*) block_info.header, pos, - info->s->pack.ref_length, READING_NEXT)) - goto err; - start_recpos=pos; - splits++; - VOID(_ma_pack_get_block_info(info,&block_info, -1, start_recpos)); - pos=block_info.filepos+block_info.rec_len; - if (block_info.rec_len < (uint) info->s->min_pack_length || - block_info.rec_len > (uint) info->s->max_pack_length) + if (info->state->data_file_length < block_info.filepos+ + block_info.block_len) { - _ma_check_print_error(param, - "Found block with wrong recordlength: %d at %s", - block_info.rec_len, llstr(start_recpos,llbuff)); - got_error=1; - break; + _ma_check_print_error(param, + "Recordlink that points outside datafile at %s", + llstr(pos,llbuff)); + got_error=1; + break; } - if (_ma_read_cache(¶m->read_cache,(byte*) info->rec_buff, - block_info.filepos, block_info.rec_len, READING_NEXT)) - goto err; - if (_ma_pack_rec_unpack(info,record,info->rec_buff,block_info.rec_len)) + param->splits++; + if (!flag++) /* First block */ { - _ma_check_print_error(param,"Found wrong record at %s", - llstr(start_recpos,llbuff)); - got_error=1; + start_recpos=pos; + pos=block_info.filepos+block_info.block_len; + if (block_info.rec_len > (uint) info->s->base.max_pack_length) + { + _ma_check_print_error(param,"Found too long record (%lu) at %s", + (ulong) block_info.rec_len, + llstr(start_recpos,llbuff)); + got_error=1; + break; + } + if (info->s->base.blobs) + { + if (_ma_alloc_buffer(&info->rec_buff, &info->rec_buff_size, + block_info.rec_len + + info->s->base.extra_rec_buff_size)) + + { + _ma_check_print_error(param, + "Not enough memory (%lu) for blob at %s", + (ulong) block_info.rec_len, + llstr(start_recpos,llbuff)); + got_error=1; + break; + } + } + to= info->rec_buff; + left_length= block_info.rec_len; + } + if (left_length < block_info.data_len) + { + _ma_check_print_error(param,"Found too long record (%lu) at %s", + (ulong) block_info.data_len, + llstr(start_recpos,llbuff)); + got_error=1; + break; + } + if (_ma_read_cache(¶m->read_cache,(byte*) to,block_info.filepos, + (uint) block_info.data_len, + flag == 1 ? READING_NEXT : 0)) + { + _ma_check_print_error(param, + "got error: %d when reading datafile at position: %s", my_errno, llstr(block_info.filepos, llbuff)); + + DBUG_RETURN(1); + } + to+=block_info.data_len; + param->link_used+= block_info.filepos-start_block; + param->used+= block_info.filepos - start_block + block_info.data_len; + param->empty+= block_info.block_len-block_info.data_len; + left_length-= block_info.data_len; + if (left_length) + { + if (b_type & BLOCK_LAST) + { + _ma_check_print_error(param, + "Wrong record length %s of %s at %s", + llstr(block_info.rec_len-left_length,llbuff), + llstr(block_info.rec_len, llbuff2), + llstr(start_recpos,llbuff3)); + got_error=1; + break; + } + if (info->state->data_file_length < block_info.next_filepos) + { + _ma_check_print_error(param, + "Found next-recordlink that points outside datafile at %s", + llstr(block_info.filepos,llbuff)); + got_error=1; + break; + } + } + } while (left_length); + + if (! got_error) + { + if (_ma_rec_unpack(info,record,info->rec_buff,block_info.rec_len) == + MY_FILE_ERROR) + { + _ma_check_print_error(param,"Found wrong record at %s", + llstr(start_recpos,llbuff)); + got_error=1; + } + else + { + info->cur_row.checksum= _ma_checksum(info,record); + if (param->testflag & (T_EXTEND | T_MEDIUM | T_VERBOSE)) + { + if (_ma_rec_check(info,record, info->rec_buff,block_info.rec_len, + test(info->s->calc_checksum))) + { + _ma_check_print_error(param,"Found wrong packed record at %s", + llstr(start_recpos,llbuff)); + got_error= 1; + } + } + param->glob_crc+= info->cur_row.checksum; + } + + if (! got_error) + { + if (check_keys_in_record(param, info, extend, start_recpos, record)) + DBUG_RETURN(1); } - if (static_row_size) - param->glob_crc+= _ma_static_checksum(info,record); else - param->glob_crc+= _ma_checksum(info,record); - link_used+= (block_info.filepos - start_recpos); - used+= (pos-start_recpos); - } /* switch */ + { + if (param->err_count++ > MAXERR || !(param->testflag & T_VERBOSE)) + DBUG_RETURN(1); + } + } + else if (!flag) + pos= block_info.filepos+block_info.block_len; +next:; + } + DBUG_RETURN(0); +} + + +static int check_compressed_record(HA_CHECK *param, MARIA_HA *info, int extend, + byte *record) +{ + my_off_t start_recpos, pos; + char llbuff[22]; + bool got_error= 0; + MARIA_BLOCK_INFO block_info; + DBUG_ENTER("check_compressed_record"); + + pos= info->s->pack.header_length; /* Skip header */ + while (pos < info->state->data_file_length) + { + if (*_ma_killed_ptr(param)) + DBUG_RETURN(-1); + + if (_ma_read_cache(¶m->read_cache,(byte*) block_info.header, pos, + info->s->pack.ref_length, READING_NEXT)) + { + _ma_check_print_error(param, + "got error: %d when reading datafile at position: %s", + my_errno, llstr(pos, llbuff)); + DBUG_RETURN(1); + } + + start_recpos= pos; + param->splits++; + VOID(_ma_pack_get_block_info(info,&block_info, -1, start_recpos)); + pos=block_info.filepos+block_info.rec_len; + if (block_info.rec_len < (uint) info->s->min_pack_length || + block_info.rec_len > (uint) info->s->max_pack_length) + { + _ma_check_print_error(param, + "Found block with wrong recordlength: %d at %s", + block_info.rec_len, llstr(start_recpos,llbuff)); + got_error=1; + goto end; + } + if (_ma_read_cache(¶m->read_cache,(byte*) info->rec_buff, + block_info.filepos, block_info.rec_len, READING_NEXT)) + { + _ma_check_print_error(param, + "got error: %d when reading datafile at position: %s", + my_errno, llstr(block_info.filepos, llbuff)); + DBUG_RETURN(1); + } + if (_ma_pack_rec_unpack(info,record,info->rec_buff,block_info.rec_len)) + { + _ma_check_print_error(param,"Found wrong record at %s", + llstr(start_recpos,llbuff)); + got_error=1; + goto end; + } + param->glob_crc+= (*info->s->calc_checksum)(info,record); + param->link_used+= (block_info.filepos - start_recpos); + param->used+= (pos-start_recpos); + +end: if (! got_error) { - intern_record_checksum+=(ha_checksum) start_recpos; - records++; - if (param->testflag & T_WRITE_LOOP && records % WRITE_COUNT == 0) + if (check_keys_in_record(param, info, extend, start_recpos, record)) + DBUG_RETURN(1); + } + else + { + got_error= 0; /* Reset for next loop */ + if (param->err_count++ > MAXERR || !(param->testflag & T_VERBOSE)) + DBUG_RETURN(1); + } + } + DBUG_RETURN(0); +} + + +/* + Check if layout on a page is ok +*/ + +static int check_page_layout(HA_CHECK *param, MARIA_HA *info, + my_off_t page_pos, byte *page, + uint row_count, uint head_empty, + uint *real_rows_found) +{ + uint empty, last_row_end, row, first_dir_entry; + byte *dir_entry; + char llbuff[22]; + DBUG_ENTER("check_page_layout"); + + empty= 0; + last_row_end= PAGE_HEADER_SIZE; + *real_rows_found= 0; + + dir_entry= page+ info->s->block_size - PAGE_SUFFIX_SIZE; + first_dir_entry= info->s->block_size - row_count* DIR_ENTRY_SIZE; + for (row= 0 ; row < row_count ; row++) + { + uint pos, length; + dir_entry-= DIR_ENTRY_SIZE; + pos= uint2korr(dir_entry); + if (!pos) + { + if (row == row_count -1) { - printf("%s\r", llstr(records,llbuff)); VOID(fflush(stdout)); + _ma_check_print_error(param, + "Page %9s: First entry in directory is 0", + llstr(page_pos, llbuff)); + if (param->err_count++ > MAXERR || !(param->testflag & T_VERBOSE)) + DBUG_RETURN(1); } + continue; /* Deleted row */ + } + (*real_rows_found)++; + length= uint2korr(dir_entry+2); + param->used+= length; + if (pos < last_row_end) + { + _ma_check_print_error(param, + "Page %9s: Row %3u overlapps with previous row", + llstr(page_pos, llbuff), row); + DBUG_RETURN(1); + } + empty+= (pos - last_row_end); + last_row_end= pos + length; + if (last_row_end > first_dir_entry) + { + _ma_check_print_error(param, + "Page %9s: Row %3u overlapps with directory", + llstr(page_pos, llbuff), row); + DBUG_RETURN(1); + } + } + empty+= (first_dir_entry - last_row_end); + + if (empty != head_empty) + { + _ma_check_print_error(param, + "Page %9s: Wrong empty size. Stored: %5u Actual: %5u", + llstr(page_pos, llbuff), head_empty, empty); + DBUG_RETURN(param->err_count++ > MAXERR || !(param->testflag & T_VERBOSE)); + } + DBUG_RETURN(0); +} + + +/* + Check all rows on head page + + NOTES + Before this, we have already called check_page_layout(), so + we know the block is logicaly correct (even if the rows may not be that) + + RETURN + 0 ok + 1 error +*/ - /* Check if keys match the record */ - for (key=0,keyinfo= info->s->keyinfo; key < info->s->base.keys; - key++,keyinfo++) +static my_bool check_head_page(HA_CHECK *param, MARIA_HA *info, byte *record, + int extend, my_off_t page_pos, byte *page_buff, + uint row_count) +{ + byte *dir_entry; + uint row; + char llbuff[22], llbuff2[22]; + DBUG_ENTER("check_head_page"); + + dir_entry= page_buff+ info->s->block_size - PAGE_SUFFIX_SIZE; + for (row= 0 ; row < row_count ; row++) + { + uint pos, length, flag; + dir_entry-= DIR_ENTRY_SIZE; + pos= uint2korr(dir_entry); + if (!pos) + continue; + length= uint2korr(dir_entry+2); + if (length < info->s->base.min_block_length) + { + _ma_check_print_error(param, + "Page %9s: Row %3u is too short (%d bytes)", + llstr(page_pos, llbuff), row, length); + DBUG_RETURN(1); + } + flag= (uint) (uchar) page_buff[pos]; + if (flag & ~(ROW_FLAG_ALL)) + _ma_check_print_error(param, + "Page %9s: Row %3u has wrong flag: %d", + llstr(page_pos, llbuff), row, flag); + + DBUG_PRINT("info", ("rowid: %s page: %lu row: %u", + llstr(ma_recordpos(page_pos/info->s->block_size, row), + llbuff), + (ulong) (page_pos / info->s->block_size), row)); + if (_ma_read_block_record2(info, record, page_buff+pos, + page_buff+pos+length)) + { + _ma_check_print_error(param, + "Page %9s: Row %3d is crashed", + llstr(page_pos, llbuff), row); + if (param->err_count++ > MAXERR || !(param->testflag & T_VERBOSE)) + DBUG_RETURN(1); + continue; + } + if (info->s->calc_checksum) + { + info->cur_row.checksum= _ma_checksum(info, record); + param->glob_crc+= info->cur_row.checksum; + } + if (info->cur_row.extents_count) + { + byte *extents= info->cur_row.extents; + uint i; + /* Check that bitmap has the right marker for the found extents */ + for (i= 0 ; i < info->cur_row.extents_count ; i++) { - if (maria_is_key_active(info->s->state.key_map, key)) - { - if(!(keyinfo->flag & HA_FULLTEXT)) - { - uint key_length= _ma_make_key(info,key,info->lastkey,record, - start_recpos); - if (extend) - { - /* We don't need to lock the key tree here as we don't allow - concurrent threads when running mariachk - */ - int search_result= -#ifdef HAVE_RTREE_KEYS - (keyinfo->flag & HA_SPATIAL) ? - maria_rtree_find_first(info, key, info->lastkey, key_length, - MBR_EQUAL | MBR_DATA) : -#endif - _ma_search(info,keyinfo,info->lastkey,key_length, - SEARCH_SAME, info->s->state.key_root[key]); - if (search_result) - { - _ma_check_print_error(param,"Record at: %10s Can't find key for index: %2d", - llstr(start_recpos,llbuff),key+1); - if (error++ > MAXERR || !(param->testflag & T_VERBOSE)) - goto err2; - } - } - else - key_checksum[key]+=maria_byte_checksum((byte*) info->lastkey, - key_length); - } - } + uint page, page_count, page_type; + page= uint5korr(extents); + page_count= uint2korr(extents+5); + extents+= ROW_EXTENT_SIZE; + page_type= BLOB_PAGE; + if (page_count & TAIL_BIT) + { + page_count= 1; + page_type= TAIL_PAGE; + } + for ( ; page_count--; page++) + { + uint bitmap_pattern; + if (_ma_check_if_right_bitmap_type(info, page_type, page, + &bitmap_pattern)) + { + _ma_check_print_error(param, + "Page %9s: Row: %3d has an extent with wrong information in bitmap: Page %9s Page_type: %d Bitmap: %d", + llstr(page_pos, llbuff), row, + llstr(page * info->s->bitmap.block_size, + llbuff2), + page_type, + bitmap_pattern); + if (param->err_count++ > MAXERR || !(param->testflag & T_VERBOSE)) + DBUG_RETURN(1); + } + } } } - else + param->full_page_count+= info->cur_row.full_page_count; + param->tail_count+= info->cur_row.tail_count; + if (check_keys_in_record(param, info, extend, + ma_recordpos(page_pos/info->s->block_size, row), + record)) + DBUG_RETURN(1); + } + DBUG_RETURN(0); +} + + + +static int check_block_record(HA_CHECK *param, MARIA_HA *info, int extend, + byte *record) +{ + my_off_t pos; + byte *page_buff, *bitmap_buff, *data; + char llbuff[22], llbuff2[22]; + uint block_size= info->s->block_size; + ha_rows full_page_count, tail_count; + my_bool full_dir; + uint offset_page, offset; + + if (_ma_scan_init_block_record(info)) + { + _ma_check_print_error(param, "got error %d when initializing scan", + my_errno); + return 1; + } + bitmap_buff= info->scan.bitmap_buff; + page_buff= info->scan.page_buff; + full_page_count= tail_count= 0; + param->full_page_count= param->tail_count= 0; + param->used= param->link_used= 0; + + for (pos= 0; + pos < info->state->data_file_length; + pos+= block_size) + { + uint row_count, real_row_count, empty_space, page_type, bitmap_pattern; + LINT_INIT(row_count); + LINT_INIT(empty_space); + + if (*_ma_killed_ptr(param)) + { + _ma_scan_end_block_record(info); + return -1; + } + if (((pos / block_size) % info->s->bitmap.pages_covered) == 0) + { + /* Bitmap page */ + if (_ma_read_cache(¶m->read_cache, bitmap_buff, pos, + block_size, READING_NEXT)) + { + _ma_check_print_error(param, + "Page %9s: Got error: %d when reading datafile", + my_errno, llstr(pos, llbuff)); + goto err; + } + param->used+= block_size; + param->link_used+= block_size; + continue; + } + /* Skip pages marked as empty in bitmap */ + offset_page= (((pos / block_size) % info->s->bitmap.pages_covered) -1) * 3; + offset= offset_page & 7; + data= bitmap_buff + offset_page / 8; + bitmap_pattern= uint2korr(data); + param->splits++; + if (!((bitmap_pattern >> offset) & 7)) + { + param->empty+= block_size; + param->del_blocks++; + continue; + } + + if (_ma_read_cache(¶m->read_cache, page_buff, pos, + block_size, READING_NEXT)) + { + _ma_check_print_error(param, + "Page %9s: Got error: %d when reading datafile", + my_errno, llstr(pos, llbuff)); + goto err; + } + page_type= page_buff[PAGE_TYPE_OFFSET] & PAGE_TYPE_MASK; + if (page_type == UNALLOCATED_PAGE || page_type >= MAX_PAGE_TYPE) + { + _ma_check_print_error(param, + "Page %9s: Found wrong page type %d\n", + llstr(pos, llbuff), page_type); + if (param->err_count++ > MAXERR || !(param->testflag & T_VERBOSE)) + goto err; + } + switch ((enum en_page_type) page_type) { + case UNALLOCATED_PAGE: + case MAX_PAGE_TYPE: + DBUG_PRINT("warning", + ("Found page with wrong page type: %d", page_type)); + DBUG_ASSERT(0); + break; + case HEAD_PAGE: + row_count= ((uchar*) page_buff)[DIR_ENTRY_OFFSET]; + empty_space= uint2korr(page_buff + EMPTY_SPACE_OFFSET); + param->used+= (PAGE_HEADER_SIZE + PAGE_SUFFIX_SIZE + + row_count * DIR_ENTRY_SIZE); + param->link_used+= (PAGE_HEADER_SIZE + PAGE_SUFFIX_SIZE + + row_count * DIR_ENTRY_SIZE); + full_dir= row_count == MAX_ROWS_PER_PAGE; + break; + case TAIL_PAGE: + row_count= ((uchar*) page_buff)[DIR_ENTRY_OFFSET]; + empty_space= uint2korr(page_buff + EMPTY_SPACE_OFFSET); + param->used+= (PAGE_HEADER_SIZE + PAGE_SUFFIX_SIZE + + row_count * DIR_ENTRY_SIZE); + param->link_used+= (PAGE_HEADER_SIZE + PAGE_SUFFIX_SIZE + + row_count * DIR_ENTRY_SIZE); + full_dir= row_count == MAX_ROWS_PER_PAGE; + break; + case BLOB_PAGE: + full_page_count++; + full_dir= 0; + empty_space= block_size; /* for error reporting */ + param->link_used+= (LSN_SIZE + PAGE_TYPE_SIZE); + param->used+= block_size; + break; + } + if (_ma_check_bitmap_data(info, page_type, pos / block_size, + full_dir ? 0 : empty_space, + &bitmap_pattern)) + { + _ma_check_print_error(param, + "Page %9s: Wrong data in bitmap. Page_type: %d empty_space: %u Bitmap: %d", + llstr(pos, llbuff), page_type, empty_space, + bitmap_pattern); + if (param->err_count++ > MAXERR || !(param->testflag & T_VERBOSE)) + goto err; + } + if ((enum en_page_type) page_type == BLOB_PAGE) + continue; + param->empty+= empty_space; + if (check_page_layout(param, info, pos, page_buff, row_count, + empty_space, &real_row_count)) + goto err; + if ((enum en_page_type) page_type == TAIL_PAGE) { - got_error=0; - if (error++ > MAXERR || !(param->testflag & T_VERBOSE)) - goto err2; + tail_count+= real_row_count; + continue; } - next:; /* Next record */ + if (check_head_page(param, info, record, extend, pos, page_buff, + row_count)) + goto err; } + + _ma_scan_end_block_record(info); + + if (full_page_count != param->full_page_count) + _ma_check_print_error(param, "Full page count read through records was %s but we found %s pages while scanning table", + llstr(param->full_page_count, llbuff), + llstr(full_page_count, llbuff2)); + if (tail_count != param->tail_count) + _ma_check_print_error(param, "Tail count read through records was %s but we found %s tails while scanning table", + llstr(param->tail_count, llbuff), + llstr(tail_count, llbuff2)); + + /* Update splits to avoid warning */ + info->s->state.split= param->splits; + info->state->del= param->del_blocks; + return param->error_printed != 0; + +err: + _ma_scan_end_block_record(info); + return 1; +} + + + /* Check that record-link is ok */ + +int maria_chk_data_link(HA_CHECK *param, MARIA_HA *info,int extend) +{ + int error; + byte *record; + char llbuff[22],llbuff2[22],llbuff3[22]; + DBUG_ENTER("maria_chk_data_link"); + + if (!(param->testflag & T_SILENT)) + { + if (extend) + puts("- check records and index references"); + else + puts("- check record links"); + } + + if (!(record= (byte*) my_malloc(info->s->base.pack_reclength,MYF(0)))) + { + _ma_check_print_error(param,"Not enough memory for record"); + DBUG_RETURN(-1); + } + param->records= param->del_blocks= 0; + param->used= param->link_used= param->splits= param->del_length= 0; + param->tmp_record_checksum= param->glob_crc= 0; + param->err_count= 0; + LINT_INIT(left_length); LINT_INIT(start_recpos); LINT_INIT(to); + error= 0; + param->empty= info->s->pack.header_length; + + bzero((char*) param->tmp_key_crc, + info->s->base.keys * sizeof(param->tmp_key_crc[0])); + + switch (info->s->data_file_type) { + case BLOCK_RECORD: + error= check_block_record(param, info, extend, record); + break; + case STATIC_RECORD: + error= check_static_record(param, info, extend, record); + break; + case DYNAMIC_RECORD: + error= check_dynamic_record(param, info, extend, record); + break; + case COMPRESSED_RECORD: + error= check_compressed_record(param, info, extend, record); + break; + } /* switch */ + + if (error) + goto err; + if (param->testflag & T_WRITE_LOOP) { VOID(fputs(" \r",stdout)); VOID(fflush(stdout)); } - if (records != info->state->records) + if (param->records != info->state->records) { - _ma_check_print_error(param,"Record-count is not ok; is %-10s Should be: %s", - llstr(records,llbuff), llstr(info->state->records,llbuff2)); + _ma_check_print_error(param, + "Record-count is not ok; found %-10s Should be: %s", + llstr(param->records,llbuff), + llstr(info->state->records,llbuff2)); error=1; } else if (param->record_checksum && - param->record_checksum != intern_record_checksum) + param->record_checksum != param->tmp_record_checksum) { _ma_check_print_error(param, - "Keypointers and record positions doesn't match"); + "Key pointers and record positions doesn't match"); error=1; } else if (param->glob_crc != info->state->checksum && @@ -1213,9 +1702,10 @@ int maria_chk_data_link(HA_CHECK *param, MARIA_HA *info,int extend) } else if (!extend) { + uint key; for (key=0 ; key < info->s->base.keys; key++) { - if (key_checksum[key] != param->key_crc[key] && + if (param->tmp_key_crc[key] != param->key_crc[key] && !(info->s->keyinfo[key].flag & (HA_FULLTEXT | HA_SPATIAL))) { _ma_check_print_error(param,"Checksum for key: %2d doesn't match checksum for records", @@ -1225,68 +1715,83 @@ int maria_chk_data_link(HA_CHECK *param, MARIA_HA *info,int extend) } } - if (del_length != info->state->empty) + if (param->del_length != info->state->empty) { _ma_check_print_warning(param, "Found %s deleted space. Should be %s", - llstr(del_length,llbuff2), + llstr(param->del_length,llbuff2), llstr(info->state->empty,llbuff)); } - if (used+empty+del_length != info->state->data_file_length) + if (param->used + param->empty + param->del_length != + info->state->data_file_length) { _ma_check_print_warning(param, - "Found %s record-data and %s unused data and %s deleted-data", - llstr(used,llbuff),llstr(empty,llbuff2), - llstr(del_length,llbuff3)); + "Found %s record data and %s unused data and %s deleted data", + llstr(param->used, llbuff), + llstr(param->empty,llbuff2), + llstr(param->del_length,llbuff3)); _ma_check_print_warning(param, - "Total %s, Should be: %s", - llstr((used+empty+del_length),llbuff), + "Total %s Should be: %s", + llstr((param->used+param->empty+param->del_length), + llbuff), llstr(info->state->data_file_length,llbuff2)); } - if (del_blocks != info->state->del) + if (param->del_blocks != info->state->del) { _ma_check_print_warning(param, "Found %10s deleted blocks Should be: %s", - llstr(del_blocks,llbuff), + llstr(param->del_blocks,llbuff), llstr(info->state->del,llbuff2)); } - if (splits != info->s->state.split) + if (param->splits != info->s->state.split) { _ma_check_print_warning(param, "Found %10s parts Should be: %s parts", - llstr(splits,llbuff), + llstr(param->splits, llbuff), llstr(info->s->state.split,llbuff2)); } if (param->testflag & T_INFO) { if (param->warning_printed || param->error_printed) puts(""); - if (used != 0 && ! param->error_printed) + if (param->used != 0 && ! param->error_printed) { - printf("Records:%18s M.recordlength:%9lu Packed:%14.0f%%\n", - llstr(records,llbuff), (long)((used-link_used)/records), - (info->s->base.blobs ? 0.0 : - (ulonglong2double((ulonglong) info->s->base.reclength*records)- - my_off_t2double(used))/ - ulonglong2double((ulonglong) info->s->base.reclength*records)*100.0)); - printf("Recordspace used:%9.0f%% Empty space:%12d%% Blocks/Record: %6.2f\n", - (ulonglong2double(used-link_used)/ulonglong2double(used-link_used+empty)*100.0), - (!records ? 100 : (int) (ulonglong2double(del_length+empty)/ - my_off_t2double(used)*100.0)), - ulonglong2double(splits - del_blocks) / records); + if (param->records) + { + printf("Records:%18s M.recordlength:%9lu Packed:%14.0f%%\n", + llstr(param->records,llbuff), + (long)((param->used - param->link_used)/param->records), + (info->s->base.blobs ? 0.0 : + (ulonglong2double((ulonglong) info->s->base.reclength * + param->records)- + my_off_t2double(param->used))/ + ulonglong2double((ulonglong) info->s->base.reclength * + param->records)*100.0)); + printf("Recordspace used:%9.0f%% Empty space:%12d%% Blocks/Record: %6.2f\n", + (ulonglong2double(param->used - param->link_used)/ + ulonglong2double(param->used-param->link_used+param->empty)*100.0), + (!param->records ? 100 : + (int) (ulonglong2double(param->del_length+param->empty)/ + my_off_t2double(param->used)*100.0)), + ulonglong2double(param->splits - param->del_blocks) / + param->records); + } + else + printf("Records:%18s\n", "0"); } printf("Record blocks:%12s Delete blocks:%10s\n", - llstr(splits-del_blocks,llbuff),llstr(del_blocks,llbuff2)); + llstr(param->splits - param->del_blocks, llbuff), + llstr(param->del_blocks, llbuff2)); printf("Record data: %12s Deleted data: %10s\n", - llstr(used-link_used,llbuff),llstr(del_length,llbuff2)); + llstr(param->used - param->link_used,llbuff), + llstr(param->del_length, llbuff2)); printf("Lost space: %12s Linkdata: %10s\n", - llstr(empty,llbuff),llstr(link_used,llbuff2)); + llstr(param->empty, llbuff),llstr(param->link_used, llbuff2)); } my_free((gptr) record,MYF(0)); DBUG_RETURN (error); + err: - _ma_check_print_error(param,"got error: %d when reading datafile at record: %s",my_errno, llstr(records,llbuff)); - err2: my_free((gptr) record,MYF(0)); param->testflag|=T_RETRY_WITHOUT_QUICK; DBUG_RETURN(1); @@ -1297,7 +1802,7 @@ int maria_chk_data_link(HA_CHECK *param, MARIA_HA *info,int extend) /* Save new datafile-name in temp_filename */ int maria_repair(HA_CHECK *param, register MARIA_HA *info, - my_string name, int rep_quick) + my_string name, int rep_quick) { int error,got_error; uint i; @@ -1348,7 +1853,8 @@ int maria_repair(HA_CHECK *param, register MARIA_HA *info, info->opt_flag|=WRITE_CACHE_USED; if (!(sort_param.record=(byte*) my_malloc((uint) share->base.pack_reclength, MYF(0))) || - !_ma_alloc_rec_buff(info, -1, &sort_param.rec_buff)) + _ma_alloc_buffer(&sort_param.rec_buff, &sort_param.rec_buff_size, + info->s->base.default_rec_buff_size)) { _ma_check_print_error(param, "Not enough memory for extra record"); goto err; @@ -1357,14 +1863,11 @@ int maria_repair(HA_CHECK *param, register MARIA_HA *info, if (!rep_quick) { /* Get real path for data file */ - if ((new_file=my_raid_create(fn_format(param->temp_filename, - share->data_file_name, "", - DATA_TMP_EXT, 2+4), - 0,param->tmpfile_createflag, - share->base.raid_type, - share->base.raid_chunks, - share->base.raid_chunksize, - MYF(0))) < 0) + if ((new_file= my_create(fn_format(param->temp_filename, + share->data_file_name, "", + DATA_TMP_EXT, 2+4), + 0,param->tmpfile_createflag, + MYF(0))) < 0) { _ma_check_print_error(param,"Can't create new tempfile: '%s'", param->temp_filename); @@ -1376,10 +1879,7 @@ int maria_repair(HA_CHECK *param, register MARIA_HA *info, info->s->state.dellink= HA_OFFSET_ERROR; info->rec_cache.file=new_file; if (param->testflag & T_UNPACK) - { - share->options&= ~HA_OPTION_COMPRESS_RECORD; - mi_int2store(share->state.header.options,share->options); - } + restore_data_file_type(share); } sort_info.info=info; sort_info.param = param; @@ -1411,8 +1911,7 @@ int maria_repair(HA_CHECK *param, register MARIA_HA *info, share->state.key_root[i]= HA_OFFSET_ERROR; /* Drop the delete chain. */ - for (i=0 ; i < share->state.header.max_block_size_index ; i++) - share->state.key_del[i]= HA_OFFSET_ERROR; + share->state.key_del= HA_OFFSET_ERROR; /* If requested, activate (enable) all keys in key_map. In this case, @@ -1436,7 +1935,7 @@ int maria_repair(HA_CHECK *param, register MARIA_HA *info, _ma_check_print_info(param,"Duplicate key %2d for record at %10s against new record at %10s", info->errkey+1, llstr(sort_param.start_recpos,llbuff), - llstr(info->dupp_key_pos,llbuff2)); + llstr(info->dup_key_pos,llbuff2)); if (param->testflag & T_VERBOSE) { VOID(_ma_make_key(info,(uint) info->errkey,info->lastkey, @@ -1530,8 +2029,7 @@ err: my_close(new_file,MYF(0)); info->dfile=new_file= -1; if (maria_change_to_newfile(share->data_file_name,MARIA_NAME_DEXT, - DATA_TMP_EXT, share->base.raid_chunks, - (param->testflag & T_BACKUP_DATA ? + DATA_TMP_EXT, (param->testflag & T_BACKUP_DATA ? MYF(MY_REDEL_MAKE_BACKUP): MYF(0))) || _ma_open_datafile(info,share,-1)) got_error=1; @@ -1545,26 +2043,20 @@ err: if (new_file >= 0) { VOID(my_close(new_file,MYF(0))); - VOID(my_raid_delete(param->temp_filename,info->s->base.raid_chunks, - MYF(MY_WME))); + VOID(my_delete(param->temp_filename, MYF(MY_WME))); info->rec_cache.file=-1; /* don't flush data to new_file, it's closed */ } maria_mark_crashed_on_repair(info); } - my_free(_ma_get_rec_buff_ptr(info, sort_param.rec_buff), - MYF(MY_ALLOW_ZERO_PTR)); + my_free(sort_param.rec_buff, MYF(MY_ALLOW_ZERO_PTR)); my_free(sort_param.record,MYF(MY_ALLOW_ZERO_PTR)); my_free(sort_info.buff,MYF(MY_ALLOW_ZERO_PTR)); VOID(end_io_cache(¶m->read_cache)); info->opt_flag&= ~(READ_CACHE_USED | WRITE_CACHE_USED); VOID(end_io_cache(&info->rec_cache)); got_error|=_ma_flush_blocks(param, share->key_cache, share->kfile); - if (!got_error && param->testflag & T_UNPACK) - { - share->state.header.options[0]&= (uchar) ~HA_OPTION_COMPRESS_RECORD; - share->pack.header_length=0; - share->data_file_type=sort_info.new_data_file_type; - } + if (!got_error && (param->testflag & T_UNPACK)) + restore_data_file_type(share); share->state.changed|= (STATE_NOT_OPTIMIZED_KEYS | STATE_NOT_SORTED_PAGES | STATE_NOT_ANALYZED); DBUG_RETURN(got_error); @@ -1577,10 +2069,10 @@ static int writekeys(HA_CHECK *param, register MARIA_HA *info, byte *buff, my_off_t filepos) { register uint i; - uchar *key; + byte *key; DBUG_ENTER("writekeys"); - key=info->lastkey+info->s->base.max_key_length; + key= info->lastkey+info->s->base.max_key_length; for (i=0 ; i < info->s->base.keys ; i++) { if (maria_is_key_active(info->s->state.key_map, i)) @@ -1632,7 +2124,7 @@ static int writekeys(HA_CHECK *param, register MARIA_HA *info, byte *buff, } /* Remove checksum that was added to glob_crc in sort_get_next_record */ if (param->calc_checksum) - param->glob_crc-= info->checksum; + param->glob_crc-= info->cur_row.checksum; DBUG_PRINT("error",("errno: %d",my_errno)); DBUG_RETURN(-1); } /* writekeys */ @@ -1640,15 +2132,16 @@ static int writekeys(HA_CHECK *param, register MARIA_HA *info, byte *buff, /* Change all key-pointers that points to a records */ -int maria_movepoint(register MARIA_HA *info, byte *record, my_off_t oldpos, - my_off_t newpos, uint prot_key) +int maria_movepoint(register MARIA_HA *info, byte *record, + MARIA_RECORD_POS oldpos, MARIA_RECORD_POS newpos, + uint prot_key) { register uint i; - uchar *key; + byte *key; uint key_length; DBUG_ENTER("maria_movepoint"); - key=info->lastkey+info->s->base.max_key_length; + key= info->lastkey+info->s->base.max_key_length; for (i=0 ; i < info->s->base.keys; i++) { if (i != prot_key && maria_is_key_active(info->s->state.key_map, i)) @@ -1764,7 +2257,7 @@ int maria_sort_index(HA_CHECK *param, register MARIA_HA *info, my_string name) index_pos[key]= HA_OFFSET_ERROR; /* No blocks */ } - /* Flush key cache for this file if we are calling this outside mariachk */ + /* Flush key cache for this file if we are calling this outside maria_chk */ flush_key_blocks(share->key_cache,share->kfile, FLUSH_IGNORE_CHANGED); share->state.version=(ulong) time((time_t*) 0); @@ -1779,8 +2272,8 @@ int maria_sort_index(HA_CHECK *param, register MARIA_HA *info, my_string name) VOID(my_close(share->kfile,MYF(MY_WME))); share->kfile = -1; VOID(my_close(new_file,MYF(MY_WME))); - if (maria_change_to_newfile(share->index_file_name,MARIA_NAME_IEXT,INDEX_TMP_EXT,0, - MYF(0)) || + if (maria_change_to_newfile(share->index_file_name, MARIA_NAME_IEXT, + INDEX_TMP_EXT, MYF(0)) || _ma_open_keyfile(share)) goto err2; info->lock_type= F_UNLCK; /* Force maria_readinfo to lock */ @@ -1795,8 +2288,7 @@ int maria_sort_index(HA_CHECK *param, register MARIA_HA *info, my_string name) info->update= (short) (HA_STATE_CHANGED | HA_STATE_ROW_CHANGED); for (key=0 ; key < info->s->base.keys ; key++) info->s->state.key_root[key]=index_pos[key]; - for (key=0 ; key < info->s->state.header.max_block_size_index ; key++) - info->s->state.key_del[key]= HA_OFFSET_ERROR; + info->s->state.key_del= HA_OFFSET_ERROR; info->s->state.changed&= ~STATE_NOT_SORTED_PAGES; DBUG_RETURN(0); @@ -1811,12 +2303,13 @@ err2: /* Sort records recursive using one index */ -static int sort_one_index(HA_CHECK *param, MARIA_HA *info, MARIA_KEYDEF *keyinfo, +static int sort_one_index(HA_CHECK *param, MARIA_HA *info, + MARIA_KEYDEF *keyinfo, my_off_t pagepos, File new_file) { uint length,nod_flag,used_length, key_length; - uchar *buff,*keypos,*endpos; - uchar key[HA_MAX_POSSIBLE_KEY_BUFF]; + byte *buff,*keypos,*endpos; + byte key[HA_MAX_POSSIBLE_KEY_BUFF]; my_off_t new_page_pos,next_page; char llbuff[22]; DBUG_ENTER("sort_one_index"); @@ -1824,7 +2317,7 @@ static int sort_one_index(HA_CHECK *param, MARIA_HA *info, MARIA_KEYDEF *keyinfo new_page_pos=param->new_file_pos; param->new_file_pos+=keyinfo->block_length; - if (!(buff=(uchar*) my_alloca((uint) keyinfo->block_length))) + if (!(buff= (byte*) my_alloca((uint) keyinfo->block_length))) { _ma_check_print_error(param,"Not enough memory for key block"); DBUG_RETURN(-1); @@ -1907,9 +2400,7 @@ err: */ int maria_change_to_newfile(const char * filename, const char * old_ext, - const char * new_ext, - uint raid_chunks __attribute__((unused)), - myf MyFlags) + const char * new_ext, myf MyFlags) { char old_filename[FN_REFLEN],new_filename[FN_REFLEN]; #ifdef USE_RAID @@ -1981,7 +2472,7 @@ err: */ int maria_repair_by_sort(HA_CHECK *param, register MARIA_HA *info, - const char * name, int rep_quick) + const char * name, int rep_quick) { int got_error; uint i; @@ -2034,7 +2525,8 @@ int maria_repair_by_sort(HA_CHECK *param, register MARIA_HA *info, if (!(sort_param.record=(byte*) my_malloc((uint) share->base.pack_reclength, MYF(0))) || - !_ma_alloc_rec_buff(info, -1, &sort_param.rec_buff)) + _ma_alloc_buffer(&sort_param.rec_buff, &sort_param.rec_buff_size, + info->s->base.default_rec_buff_size)) { _ma_check_print_error(param, "Not enough memory for extra record"); goto err; @@ -2042,14 +2534,11 @@ int maria_repair_by_sort(HA_CHECK *param, register MARIA_HA *info, if (!rep_quick) { /* Get real path for data file */ - if ((new_file=my_raid_create(fn_format(param->temp_filename, - share->data_file_name, "", - DATA_TMP_EXT, 2+4), - 0,param->tmpfile_createflag, - share->base.raid_type, - share->base.raid_chunks, - share->base.raid_chunksize, - MYF(0))) < 0) + if ((new_file=my_create(fn_format(param->temp_filename, + share->data_file_name, "", + DATA_TMP_EXT, 2+4), + 0,param->tmpfile_createflag, + MYF(0))) < 0) { _ma_check_print_error(param,"Can't create new tempfile: '%s'", param->temp_filename); @@ -2059,10 +2548,7 @@ int maria_repair_by_sort(HA_CHECK *param, register MARIA_HA *info, "datafile-header")) goto err; if (param->testflag & T_UNPACK) - { - share->options&= ~HA_OPTION_COMPRESS_RECORD; - mi_int2store(share->state.header.options,share->options); - } + restore_data_file_type(share); share->state.dellink= HA_OFFSET_ERROR; info->rec_cache.file=new_file; } @@ -2072,14 +2558,13 @@ int maria_repair_by_sort(HA_CHECK *param, register MARIA_HA *info, { /* Flush key cache for this file if we are calling this outside - mariachk + maria_chk */ flush_key_blocks(share->key_cache,share->kfile, FLUSH_IGNORE_CHANGED); /* Clear the pointers to the given rows */ for (i=0 ; i < share->base.keys ; i++) share->state.key_root[i]= HA_OFFSET_ERROR; - for (i=0 ; i < share->state.header.max_block_size_index ; i++) - share->state.key_del[i]= HA_OFFSET_ERROR; + share->state.key_del= HA_OFFSET_ERROR; info->state->key_file_length=share->base.keystart; } else @@ -2191,13 +2676,13 @@ int maria_repair_by_sort(HA_CHECK *param, register MARIA_HA *info, 10*param->sort_buffer_length/sort_param.key_length; } - sort_param.key_read=sort_maria_ft_key_read; - sort_param.key_write=sort_maria_ft_key_write; + sort_param.key_read= sort_maria_ft_key_read; + sort_param.key_write= sort_maria_ft_key_write; } else { - sort_param.key_read=sort_key_read; - sort_param.key_write=sort_key_write; + sort_param.key_read= sort_key_read; + sort_param.key_write= sort_key_write; } if (_ma_create_index_by_sort(&sort_param, @@ -2276,7 +2761,7 @@ int maria_repair_by_sort(HA_CHECK *param, register MARIA_HA *info, skr < share->base.reloc*share->base.min_pack_length) skr=share->base.reloc*share->base.min_pack_length; #endif - if (skr != sort_info.filelength && !info->s->base.raid_type) + if (skr != sort_info.filelength) if (my_chsize(info->dfile,skr,0,MYF(0))) _ma_check_print_warning(param, "Can't change size of datafile, error: %d", @@ -2315,9 +2800,9 @@ err: my_close(new_file,MYF(0)); info->dfile=new_file= -1; if (maria_change_to_newfile(share->data_file_name,MARIA_NAME_DEXT, - DATA_TMP_EXT, share->base.raid_chunks, - (param->testflag & T_BACKUP_DATA ? - MYF(MY_REDEL_MAKE_BACKUP): MYF(0))) || + DATA_TMP_EXT, + (param->testflag & T_BACKUP_DATA ? + MYF(MY_REDEL_MAKE_BACKUP): MYF(0))) || _ma_open_datafile(info,share,-1)) got_error=1; } @@ -2329,8 +2814,7 @@ err: if (new_file >= 0) { VOID(my_close(new_file,MYF(0))); - VOID(my_raid_delete(param->temp_filename,share->base.raid_chunks, - MYF(MY_WME))); + VOID(my_delete(param->temp_filename, MYF(MY_WME))); if (info->dfile == new_file) info->dfile= -1; } @@ -2340,8 +2824,7 @@ err: share->state.changed&= ~STATE_NOT_OPTIMIZED_KEYS; share->state.changed|=STATE_NOT_SORTED_PAGES; - my_free(_ma_get_rec_buff_ptr(info, sort_param.rec_buff), - MYF(MY_ALLOW_ZERO_PTR)); + my_free(sort_param.rec_buff, MYF(MY_ALLOW_ZERO_PTR)); my_free(sort_param.record,MYF(MY_ALLOW_ZERO_PTR)); my_free((gptr) sort_info.key_block,MYF(MY_ALLOW_ZERO_PTR)); my_free((gptr) sort_info.ft_buf, MYF(MY_ALLOW_ZERO_PTR)); @@ -2349,10 +2832,7 @@ err: VOID(end_io_cache(¶m->read_cache)); info->opt_flag&= ~(READ_CACHE_USED | WRITE_CACHE_USED); if (!got_error && (param->testflag & T_UNPACK)) - { - share->state.header.options[0]&= (uchar) ~HA_OPTION_COMPRESS_RECORD; - share->pack.header_length=0; - } + restore_data_file_type(share); DBUG_RETURN(got_error); } @@ -2435,15 +2915,12 @@ int maria_repair_parallel(HA_CHECK *param, register MARIA_HA *info, if (!rep_quick) { /* Get real path for data file */ - if ((new_file=my_raid_create(fn_format(param->temp_filename, - share->data_file_name, "", - DATA_TMP_EXT, - 2+4), - 0,param->tmpfile_createflag, - share->base.raid_type, - share->base.raid_chunks, - share->base.raid_chunksize, - MYF(0))) < 0) + if ((new_file= my_create(fn_format(param->temp_filename, + share->data_file_name, "", + DATA_TMP_EXT, + 2+4), + 0,param->tmpfile_createflag, + MYF(0))) < 0) { _ma_check_print_error(param,"Can't create new tempfile: '%s'", param->temp_filename); @@ -2453,10 +2930,7 @@ int maria_repair_parallel(HA_CHECK *param, register MARIA_HA *info, "datafile-header")) goto err; if (param->testflag & T_UNPACK) - { - share->options&= ~HA_OPTION_COMPRESS_RECORD; - mi_int2store(share->state.header.options,share->options); - } + restore_data_file_type(share); share->state.dellink= HA_OFFSET_ERROR; info->rec_cache.file=new_file; } @@ -2466,14 +2940,13 @@ int maria_repair_parallel(HA_CHECK *param, register MARIA_HA *info, { /* Flush key cache for this file if we are calling this outside - mariachk + maria_chk */ flush_key_blocks(share->key_cache,share->kfile, FLUSH_IGNORE_CHANGED); /* Clear the pointers to the given rows */ for (i=0 ; i < share->base.keys ; i++) share->state.key_root[i]= HA_OFFSET_ERROR; - for (i=0 ; i < share->state.header.max_block_size_index ; i++) - share->state.key_del[i]= HA_OFFSET_ERROR; + share->state.key_del= HA_OFFSET_ERROR; info->state->key_file_length=share->base.keystart; } else @@ -2573,12 +3046,12 @@ int maria_repair_parallel(HA_CHECK *param, register MARIA_HA *info, sort_param[i].record= (((char *)(sort_param+share->base.keys))+ (share->base.pack_reclength * i)); - if (!_ma_alloc_rec_buff(info, -1, &sort_param[i].rec_buff)) + if (_ma_alloc_buffer(&sort_param[i].rec_buff, &sort_param[i].rec_buff_size, + share->base.default_rec_buff_size)) { _ma_check_print_error(param,"Not enough memory!"); goto err; } - sort_param[i].key_length=share->rec_reflength; for (keyseg=sort_param[i].seg; keyseg->type != HA_KEYTYPE_END; keyseg++) @@ -2698,7 +3171,7 @@ int maria_repair_parallel(HA_CHECK *param, register MARIA_HA *info, skr < share->base.reloc*share->base.min_pack_length) skr=share->base.reloc*share->base.min_pack_length; #endif - if (skr != sort_info.filelength && !info->s->base.raid_type) + if (skr != sort_info.filelength) if (my_chsize(info->dfile,skr,0,MYF(0))) _ma_check_print_warning(param, "Can't change size of datafile, error: %d", @@ -2736,9 +3209,9 @@ err: my_close(new_file,MYF(0)); info->dfile=new_file= -1; if (maria_change_to_newfile(share->data_file_name,MARIA_NAME_DEXT, - DATA_TMP_EXT, share->base.raid_chunks, - (param->testflag & T_BACKUP_DATA ? - MYF(MY_REDEL_MAKE_BACKUP): MYF(0))) || + DATA_TMP_EXT, + (param->testflag & T_BACKUP_DATA ? + MYF(MY_REDEL_MAKE_BACKUP): MYF(0))) || _ma_open_datafile(info,share,-1)) got_error=1; } @@ -2750,8 +3223,7 @@ err: if (new_file >= 0) { VOID(my_close(new_file,MYF(0))); - VOID(my_raid_delete(param->temp_filename,share->base.raid_chunks, - MYF(MY_WME))); + VOID(my_delete(param->temp_filename, MYF(MY_WME))); if (info->dfile == new_file) info->dfile= -1; } @@ -2771,21 +3243,18 @@ err: VOID(end_io_cache(¶m->read_cache)); info->opt_flag&= ~(READ_CACHE_USED | WRITE_CACHE_USED); if (!got_error && (param->testflag & T_UNPACK)) - { - share->state.header.options[0]&= (uchar) ~HA_OPTION_COMPRESS_RECORD; - share->pack.header_length=0; - } + restore_data_file_type(share); DBUG_RETURN(got_error); #endif /* THREAD */ } /* Read next record and return next key */ -static int sort_key_read(MARIA_SORT_PARAM *sort_param, void *key) +static int sort_key_read(MARIA_SORT_PARAM *sort_param, byte *key) { int error; - MARIA_SORT_INFO *sort_info=sort_param->sort_info; - MARIA_HA *info=sort_info->info; + MARIA_SORT_INFO *sort_info= sort_param->sort_info; + MARIA_HA *info= sort_info->info; DBUG_ENTER("sort_key_read"); if ((error=sort_get_next_record(sort_param))) @@ -2799,7 +3268,7 @@ static int sort_key_read(MARIA_SORT_PARAM *sort_param, void *key) } sort_param->real_key_length= (info->s->rec_reflength+ - _ma_make_key(info, sort_param->key, (uchar*) key, + _ma_make_key(info, sort_param->key, key, sort_param->record, sort_param->filepos)); #ifdef HAVE_purify bzero(key+sort_param->real_key_length, @@ -2808,7 +3277,8 @@ static int sort_key_read(MARIA_SORT_PARAM *sort_param, void *key) DBUG_RETURN(_ma_sort_write_record(sort_param)); } /* sort_key_read */ -static int sort_maria_ft_key_read(MARIA_SORT_PARAM *sort_param, void *key) + +static int sort_maria_ft_key_read(MARIA_SORT_PARAM *sort_param, byte *key) { int error; MARIA_SORT_INFO *sort_info=sort_param->sort_info; @@ -2841,7 +3311,8 @@ static int sort_maria_ft_key_read(MARIA_SORT_PARAM *sort_param, void *key) sort_param->real_key_length=(info->s->rec_reflength+ _ma_ft_make_key(info, sort_param->key, - key, wptr++, sort_param->filepos)); + key, wptr++, + sort_param->filepos)); #ifdef HAVE_purify if (sort_param->key_length > sort_param->real_key_length) bzero(key+sort_param->real_key_length, @@ -2881,6 +3352,9 @@ static int sort_get_next_record(MARIA_SORT_PARAM *sort_param) DBUG_RETURN(1); switch (share->data_file_type) { + case BLOCK_RECORD: + DBUG_ASSERT(0); + break; case STATIC_RECORD: for (;;) { @@ -2904,7 +3378,7 @@ static int sort_get_next_record(MARIA_SORT_PARAM *sort_param) if (*sort_param->record) { if (param->calc_checksum) - param->glob_crc+= (info->checksum= + param->glob_crc+= (info->cur_row.checksum= _ma_static_checksum(info,sort_param->record)); DBUG_RETURN(0); } @@ -3088,8 +3562,11 @@ static int sort_get_next_record(MARIA_SORT_PARAM *sort_param) sort_param->pos=block_info.filepos+block_info.block_len; if (share->base.blobs) { - if (!(to=_ma_alloc_rec_buff(info,block_info.rec_len, - &(sort_param->rec_buff)))) + if (_ma_alloc_buffer(&sort_param->rec_buff, + &sort_param->rec_buff_size, + block_info.rec_len + + info->s->base.extra_rec_buff_size)) + { if (param->max_record_length >= block_info.rec_len) { @@ -3107,8 +3584,7 @@ static int sort_get_next_record(MARIA_SORT_PARAM *sort_param) } } } - else - to= sort_param->rec_buff; + to= sort_param->rec_buff; } if (left_length < block_info.data_len || ! block_info.data_len) { @@ -3158,7 +3634,7 @@ static int sort_get_next_record(MARIA_SORT_PARAM *sort_param) if (sort_param->read_cache.error < 0) DBUG_RETURN(1); if (info->s->calc_checksum) - info->checksum=_ma_checksum(info,sort_param->record); + info->cur_row.checksum= _ma_checksum(info,sort_param->record); if ((param->testflag & (T_EXTEND | T_REP)) || searching) { if (_ma_rec_check(info, sort_param->record, sort_param->rec_buff, @@ -3172,7 +3648,7 @@ static int sort_get_next_record(MARIA_SORT_PARAM *sort_param) } } if (param->calc_checksum) - param->glob_crc+= info->checksum; + param->glob_crc+= info->cur_row.checksum; DBUG_RETURN(0); } if (!searching) @@ -3230,7 +3706,6 @@ static int sort_get_next_record(MARIA_SORT_PARAM *sort_param) llstr(sort_param->pos,llbuff)); continue; } - info->checksum=_ma_checksum(info,sort_param->record); if (!sort_param->fix_datafile) { sort_param->filepos=sort_param->pos; @@ -3240,8 +3715,12 @@ static int sort_get_next_record(MARIA_SORT_PARAM *sort_param) sort_param->max_pos=(sort_param->pos=block_info.filepos+ block_info.rec_len); info->packed_length=block_info.rec_len; - if (param->calc_checksum) - param->glob_crc+= info->checksum; + + { + info->cur_row.checksum= (*info->s->calc_checksum)(info, + sort_param->record); + param->glob_crc+= info->cur_row.checksum; + } DBUG_RETURN(0); } } @@ -3267,6 +3746,9 @@ int _ma_sort_write_record(MARIA_SORT_PARAM *sort_param) if (sort_param->fix_datafile) { switch (sort_info->new_data_file_type) { + case BLOCK_RECORD: + DBUG_ASSERT(0); + break; case STATIC_RECORD: if (my_b_write(&info->rec_cache,sort_param->record, share->base.pack_reclength)) @@ -3276,7 +3758,6 @@ int _ma_sort_write_record(MARIA_SORT_PARAM *sort_param) } sort_param->filepos+=share->base.pack_reclength; info->s->state.split++; - /* sort_info->param->glob_crc+=_ma_static_checksum(info, sort_param->record); */ break; case DYNAMIC_RECORD: if (! info->blobs) @@ -3298,10 +3779,9 @@ int _ma_sort_write_record(MARIA_SORT_PARAM *sort_param) } from=sort_info->buff+ALIGN_SIZE(MARIA_MAX_DYN_BLOCK_HEADER); } - info->checksum=_ma_checksum(info,sort_param->record); + info->cur_row.checksum= _ma_checksum(info,sort_param->record); reclength= _ma_rec_pack(info,from,sort_param->record); flag=0; - /* sort_info->param->glob_crc+=info->checksum; */ do { @@ -3322,7 +3802,6 @@ int _ma_sort_write_record(MARIA_SORT_PARAM *sort_param) sort_param->filepos+=block_length; info->s->state.split++; } while (reclength); - /* sort_info->param->glob_crc+=info->checksum; */ break; case COMPRESSED_RECORD: reclength=info->packed_length; @@ -3337,7 +3816,6 @@ int _ma_sort_write_record(MARIA_SORT_PARAM *sort_param) _ma_check_print_error(param,"%d when writing to datafile",my_errno); DBUG_RETURN(1); } - /* sort_info->param->glob_crc+=info->checksum; */ sort_param->filepos+=reclength+length; info->s->state.split++; break; @@ -3369,7 +3847,7 @@ static int sort_key_cmp(MARIA_SORT_PARAM *sort_param, const void *a, } /* sort_key_cmp */ -static int sort_key_write(MARIA_SORT_PARAM *sort_param, const void *a) +static int sort_key_write(MARIA_SORT_PARAM *sort_param, const byte *a) { uint diff_pos[2]; char llbuff[22],llbuff2[22]; @@ -3379,11 +3857,11 @@ static int sort_key_write(MARIA_SORT_PARAM *sort_param, const void *a) if (sort_info->key_block->inited) { - cmp=ha_key_cmp(sort_param->seg,sort_info->key_block->lastkey, + cmp=ha_key_cmp(sort_param->seg, (uchar*) sort_info->key_block->lastkey, (uchar*) a, USE_WHOLE_KEY,SEARCH_FIND | SEARCH_UPDATE, diff_pos); if (param->stats_method == MI_STATS_METHOD_NULLS_NOT_EQUAL) - ha_key_cmp(sort_param->seg,sort_info->key_block->lastkey, + ha_key_cmp(sort_param->seg, (uchar*) sort_info->key_block->lastkey, (uchar*) a, USE_WHOLE_KEY, SEARCH_FIND | SEARCH_NULL_ARE_NOT_EQUAL, diff_pos); else if (param->stats_method == MI_STATS_METHOD_IGNORE_NULLS) @@ -3391,7 +3869,7 @@ static int sort_key_write(MARIA_SORT_PARAM *sort_param, const void *a) diff_pos[0]= maria_collect_stats_nonulls_next(sort_param->seg, sort_param->notnull, sort_info->key_block->lastkey, - (uchar*)a); + a); } sort_param->unique[diff_pos[0]-1]++; } @@ -3400,17 +3878,17 @@ static int sort_key_write(MARIA_SORT_PARAM *sort_param, const void *a) cmp= -1; if (param->stats_method == MI_STATS_METHOD_IGNORE_NULLS) maria_collect_stats_nonulls_first(sort_param->seg, sort_param->notnull, - (uchar*)a); + a); } if ((sort_param->keyinfo->flag & HA_NOSAME) && cmp == 0) { sort_info->dupp++; - sort_info->info->lastpos=get_record_for_key(sort_info->info, - sort_param->keyinfo, - (uchar*) a); + sort_info->info->cur_row.lastpos= get_record_for_key(sort_info->info, + sort_param->keyinfo, + a); _ma_check_print_warning(param, "Duplicate key for record at %10s against record at %10s", - llstr(sort_info->info->lastpos,llbuff), + llstr(sort_info->info->cur_row.lastpos, llbuff), llstr(get_record_for_key(sort_info->info, sort_param->keyinfo, sort_info->key_block-> @@ -3418,7 +3896,7 @@ static int sort_key_write(MARIA_SORT_PARAM *sort_param, const void *a) llbuff2)); param->testflag|=T_RETRY_WITHOUT_QUICK; if (sort_info->param->testflag & T_VERBOSE) - _ma_print_key(stdout,sort_param->seg,(uchar*) a, USE_WHOLE_KEY); + _ma_print_key(stdout,sort_param->seg, a, USE_WHOLE_KEY); return (sort_delete_record(sort_param)); } #ifndef DBUG_OFF @@ -3429,10 +3907,11 @@ static int sort_key_write(MARIA_SORT_PARAM *sort_param, const void *a) return(1); } #endif - return (sort_insert_key(sort_param,sort_info->key_block, - (uchar*) a, HA_OFFSET_ERROR)); + return (sort_insert_key(sort_param, sort_info->key_block, + a, HA_OFFSET_ERROR)); } /* sort_key_write */ + int _ma_sort_ft_buf_flush(MARIA_SORT_PARAM *sort_param) { MARIA_SORT_INFO *sort_info=sort_param->sort_info; @@ -3441,24 +3920,24 @@ int _ma_sort_ft_buf_flush(MARIA_SORT_PARAM *sort_param) uint val_off, val_len; int error; SORT_FT_BUF *maria_ft_buf=sort_info->ft_buf; - uchar *from, *to; + byte *from, *to; val_len=share->ft2_keyinfo.keylength; get_key_full_length_rdonly(val_off, maria_ft_buf->lastkey); - to=maria_ft_buf->lastkey+val_off; + to= maria_ft_buf->lastkey+val_off; if (maria_ft_buf->buf) { /* flushing first-level tree */ - error=sort_insert_key(sort_param,key_block,maria_ft_buf->lastkey, - HA_OFFSET_ERROR); + error= sort_insert_key(sort_param,key_block,maria_ft_buf->lastkey, + HA_OFFSET_ERROR); for (from=to+val_len; !error && from < maria_ft_buf->buf; from+= val_len) { memcpy(to, from, val_len); - error=sort_insert_key(sort_param,key_block,maria_ft_buf->lastkey, - HA_OFFSET_ERROR); + error= sort_insert_key(sort_param,key_block,maria_ft_buf->lastkey, + HA_OFFSET_ERROR); } return error; } @@ -3478,10 +3957,11 @@ int _ma_sort_ft_buf_flush(MARIA_SORT_PARAM *sort_param) maria_ft_buf->lastkey,HA_OFFSET_ERROR); } -static int sort_maria_ft_key_write(MARIA_SORT_PARAM *sort_param, const void *a) + +static int sort_maria_ft_key_write(MARIA_SORT_PARAM *sort_param, + const byte *a) { uint a_len, val_off, val_len, error; - uchar *p; MARIA_SORT_INFO *sort_info= sort_param->sort_info; SORT_FT_BUF *ft_buf= sort_info->ft_buf; SORT_KEY_BLOCKS *key_block= sort_info->key_block; @@ -3514,13 +3994,14 @@ static int sort_maria_ft_key_write(MARIA_SORT_PARAM *sort_param, const void *a) if (ha_compare_text(sort_param->seg->charset, ((uchar *)a)+1,a_len-1, - ft_buf->lastkey+1,val_off-1, 0, 0)==0) + (uchar*) ft_buf->lastkey+1,val_off-1, 0, 0)==0) { + byte *p; if (!ft_buf->buf) /* store in second-level tree */ { ft_buf->count++; return sort_insert_key(sort_param,key_block, - ((uchar *)a)+a_len, HA_OFFSET_ERROR); + a + a_len, HA_OFFSET_ERROR); } /* storing the key in the buffer. */ @@ -3569,21 +4050,22 @@ word_init_ft_buf: /* get pointer to record from a key */ static my_off_t get_record_for_key(MARIA_HA *info, MARIA_KEYDEF *keyinfo, - uchar *key) + const byte *key) { - return _ma_dpos(info,0,key + _ma_keylength(keyinfo,key)); + return _ma_dpos(info,0, key + _ma_keylength(keyinfo, key)); } /* get_record_for_key */ /* Insert a key in sort-key-blocks */ static int sort_insert_key(MARIA_SORT_PARAM *sort_param, - register SORT_KEY_BLOCKS *key_block, uchar *key, + register SORT_KEY_BLOCKS *key_block, + const byte *key, my_off_t prev_block) { uint a_length,t_length,nod_flag; my_off_t filepos,key_file_length; - uchar *anc_buff,*lastkey; + byte *anc_buff,*lastkey; MARIA_KEY_PARAM s_temp; MARIA_HA *info; MARIA_KEYDEF *keyinfo=sort_param->keyinfo; @@ -3591,7 +4073,7 @@ static int sort_insert_key(MARIA_SORT_PARAM *sort_param, HA_CHECK *param=sort_info->param; DBUG_ENTER("sort_insert_key"); - anc_buff=key_block->buff; + anc_buff= key_block->buff; info=sort_info->info; lastkey=key_block->lastkey; nod_flag= (key_block == sort_info->key_block ? 0 : @@ -3617,7 +4099,7 @@ static int sort_insert_key(MARIA_SORT_PARAM *sort_param, _ma_kpointer(info,key_block->end_pos,prev_block); t_length=(*keyinfo->pack_key)(keyinfo,nod_flag, - (uchar*) 0,lastkey,lastkey,key, + (byte*) 0,lastkey,lastkey,key, &s_temp); (*keyinfo->store_key)(keyinfo, key_block->end_pos+nod_flag,&s_temp); a_length+=t_length; @@ -3625,14 +4107,14 @@ static int sort_insert_key(MARIA_SORT_PARAM *sort_param, key_block->end_pos+=t_length; if (a_length <= keyinfo->block_length) { - VOID(_ma_move_key(keyinfo,key_block->lastkey,key)); + VOID(_ma_move_key(keyinfo, key_block->lastkey, key)); key_block->last_length=a_length-t_length; DBUG_RETURN(0); } /* Fill block with end-zero and write filled block */ maria_putint(anc_buff,key_block->last_length,nod_flag); - bzero((byte*) anc_buff+key_block->last_length, + bzero(anc_buff+key_block->last_length, keyinfo->block_length- key_block->last_length); key_file_length=info->state->key_file_length; if ((filepos= _ma_new(info,keyinfo,DFLT_INIT_HITS)) == HA_OFFSET_ERROR) @@ -3644,10 +4126,10 @@ static int sort_insert_key(MARIA_SORT_PARAM *sort_param, if (_ma_write_keypage(info, keyinfo, filepos, DFLT_INIT_HITS, anc_buff)) DBUG_RETURN(1); } - else if (my_pwrite(info->s->kfile,(byte*) anc_buff, + else if (my_pwrite(info->s->kfile,anc_buff, (uint) keyinfo->block_length,filepos, param->myf_rw)) DBUG_RETURN(1); - DBUG_DUMP("buff",(byte*) anc_buff,maria_getint(anc_buff)); + DBUG_DUMP("buff",anc_buff,maria_getint(anc_buff)); /* Write separator-key to block in next level */ if (sort_insert_key(sort_param,key_block+1,key_block->lastkey,filepos)) @@ -3665,7 +4147,7 @@ static int sort_delete_record(MARIA_SORT_PARAM *sort_param) { uint i; int old_file,error; - uchar *key; + byte *key; MARIA_SORT_INFO *sort_info=sort_param->sort_info; HA_CHECK *param=sort_info->param; MARIA_HA *info=sort_info->info; @@ -3680,7 +4162,7 @@ static int sort_delete_record(MARIA_SORT_PARAM *sort_param) if (info->s->options & HA_OPTION_COMPRESS_RECORD) { _ma_check_print_error(param, - "Recover aborted; Can't run standard recovery on compressed tables with errors in data-file. Use switch 'mariachk --safe-recover' to fix it\n",stderr);; + "Recover aborted; Can't run standard recovery on compressed tables with errors in data-file. Use switch 'maria_chk --safe-recover' to fix it\n",stderr);; DBUG_RETURN(1); } @@ -3688,8 +4170,9 @@ static int sort_delete_record(MARIA_SORT_PARAM *sort_param) info->dfile=info->rec_cache.file; if (sort_info->current_key) { - key=info->lastkey+info->s->base.max_key_length; - if ((error=(*info->s->read_rnd)(info,sort_param->record,info->lastpos,0)) && + key= info->lastkey+info->s->base.max_key_length; + if ((error=(*info->s->read_record)(info,sort_param->record, + info->cur_row.lastpos)) && error != HA_ERR_RECORD_DELETED) { _ma_check_print_error(param,"Can't read record to be removed"); @@ -3699,10 +4182,13 @@ static int sort_delete_record(MARIA_SORT_PARAM *sort_param) for (i=0 ; i < sort_info->current_key ; i++) { - uint key_length= _ma_make_key(info,i,key,sort_param->record,info->lastpos); - if (_ma_ck_delete(info,i,key,key_length)) + uint key_length= _ma_make_key(info, i, key, sort_param->record, + info->cur_row.lastpos); + if (_ma_ck_delete(info, i, key, key_length)) { - _ma_check_print_error(param,"Can't delete key %d from record to be removed",i+1); + _ma_check_print_error(param, + "Can't delete key %d from record to be removed", + i+1); info->dfile=old_file; DBUG_RETURN(1); } @@ -3738,7 +4224,7 @@ int _ma_flush_pending_blocks(MARIA_SORT_PARAM *sort_param) if (nod_flag) _ma_kpointer(info,key_block->end_pos,filepos); key_file_length=info->state->key_file_length; - bzero((byte*) key_block->buff+length, keyinfo->block_length-length); + bzero(key_block->buff+length, keyinfo->block_length-length); if ((filepos= _ma_new(info,keyinfo,DFLT_INIT_HITS)) == HA_OFFSET_ERROR) DBUG_RETURN(1); @@ -3749,10 +4235,10 @@ int _ma_flush_pending_blocks(MARIA_SORT_PARAM *sort_param) DFLT_INIT_HITS, key_block->buff)) DBUG_RETURN(1); } - else if (my_pwrite(info->s->kfile,(byte*) key_block->buff, + else if (my_pwrite(info->s->kfile,key_block->buff, (uint) keyinfo->block_length,filepos, myf_rw)) DBUG_RETURN(1); - DBUG_DUMP("buff",(byte*) key_block->buff,length); + DBUG_DUMP("buff",key_block->buff,length); nod_flag=1; } info->s->state.key_root[sort_param->key]=filepos; /* Last is root for tree */ @@ -3768,9 +4254,9 @@ static SORT_KEY_BLOCKS *alloc_key_blocks(HA_CHECK *param, uint blocks, SORT_KEY_BLOCKS *block; DBUG_ENTER("alloc_key_blocks"); - if (!(block=(SORT_KEY_BLOCKS*) my_malloc((sizeof(SORT_KEY_BLOCKS)+ - buffer_length+IO_SIZE)*blocks, - MYF(0)))) + if (!(block= (SORT_KEY_BLOCKS*) my_malloc((sizeof(SORT_KEY_BLOCKS)+ + buffer_length+IO_SIZE)*blocks, + MYF(0)))) { _ma_check_print_error(param,"Not enough memory for sort-key-blocks"); return(0); @@ -3778,7 +4264,7 @@ static SORT_KEY_BLOCKS *alloc_key_blocks(HA_CHECK *param, uint blocks, for (i=0 ; i < blocks ; i++) { block[i].inited=0; - block[i].buff=(uchar*) (block+blocks)+(buffer_length+IO_SIZE)*i; + block[i].buff= (byte*) (block+blocks)+(buffer_length+IO_SIZE)*i; } DBUG_RETURN(block); } /* alloc_key_blocks */ @@ -3821,7 +4307,8 @@ int maria_recreate_table(HA_CHECK *param, MARIA_HA **org_info, char *filename) share= *(*org_info)->s; unpack= (share.options & HA_OPTION_COMPRESS_RECORD) && (param->testflag & T_UNPACK); - if (!(keyinfo=(MARIA_KEYDEF*) my_alloca(sizeof(MARIA_KEYDEF)*share.base.keys))) + if (!(keyinfo=(MARIA_KEYDEF*) my_alloca(sizeof(MARIA_KEYDEF) * + share.base.keys))) DBUG_RETURN(0); memcpy((byte*) keyinfo,(byte*) share.keyinfo, (size_t) (sizeof(MARIA_KEYDEF)*share.base.keys)); @@ -3877,8 +4364,10 @@ int maria_recreate_table(HA_CHECK *param, MARIA_HA **org_info, char *filename) keyseg++; /* Skip end pointer */ } - /* Copy the unique definitions and change them to point at the new key - segments*/ + /* + Copy the unique definitions and change them to point at the new key + segments + */ memcpy((byte*) uniquedef,(byte*) share.uniqueinfo, (size_t) (sizeof(MARIA_UNIQUEDEF)*(share.state.header.uniques))); for (u_ptr=uniquedef,u_end=uniquedef+share.state.header.uniques; @@ -3894,7 +4383,7 @@ int maria_recreate_table(HA_CHECK *param, MARIA_HA **org_info, char *filename) (ulong) share.base.min_pack_length); else max_records=0; - unpack= (share.options & HA_OPTION_COMPRESS_RECORD) && + unpack= (share.data_file_type == COMPRESSED_RECORD) && (param->testflag & T_UNPACK); share.options&= ~HA_OPTION_TEMP_COMPRESS_RECORD; @@ -3916,21 +4405,29 @@ int maria_recreate_table(HA_CHECK *param, MARIA_HA **org_info, char *filename) create_info.language = (param->language ? param->language : share.state.header.language); create_info.key_file_length= status_info.key_file_length; + create_info.org_data_file_type= ((enum data_file_type) + share.state.header.org_data_file_type); + /* Allow for creating an auto_increment key. This has an effect only if an auto_increment key exists in the original table. */ create_info.with_auto_increment= TRUE; - /* We don't have to handle symlinks here because we are using - HA_DONT_TOUCH_DATA */ - if (maria_create(filename, - share.base.keys - share.state.header.uniques, - keyinfo, share.base.fields, recdef, - share.state.header.uniques, uniquedef, - &create_info, - HA_DONT_TOUCH_DATA)) - { - _ma_check_print_error(param,"Got error %d when trying to recreate indexfile",my_errno); + create_info.null_bytes= share.base.null_bytes; + /* + We don't have to handle symlinks here because we are using + HA_DONT_TOUCH_DATA + */ + if (maria_create(filename, share.data_file_type, + share.base.keys - share.state.header.uniques, + keyinfo, share.base.fields, recdef, + share.state.header.uniques, uniquedef, + &create_info, + HA_DONT_TOUCH_DATA)) + { + _ma_check_print_error(param, + "Got error %d when trying to recreate indexfile", + my_errno); goto end; } *org_info=maria_open(filename,O_RDWR, @@ -3939,8 +4436,9 @@ int maria_recreate_table(HA_CHECK *param, MARIA_HA **org_info, char *filename) HA_OPEN_ABORT_IF_LOCKED); if (!*org_info) { - _ma_check_print_error(param,"Got error %d when trying to open re-created indexfile", - my_errno); + _ma_check_print_error(param, + "Got error %d when trying to open re-created indexfile", + my_errno); goto end; } /* We are modifing */ @@ -3990,7 +4488,8 @@ int maria_write_data_suffix(MARIA_SORT_INFO *sort_info, my_bool fix_datafile) return 0; } - /* Update state and mariachk_time of indexfile */ + +/* Update state and maria_chk time of indexfile */ int maria_update_state_info(HA_CHECK *param, MARIA_HA *info,uint update) { @@ -4303,12 +4802,7 @@ set_data_file_type(MARIA_SORT_INFO *sort_info, MARIA_SHARE *share) COMPRESSED_RECORD && sort_info->param->testflag & T_UNPACK) { MARIA_SHARE tmp; - - if (share->options & HA_OPTION_PACK_RECORD) - sort_info->new_data_file_type = DYNAMIC_RECORD; - else - sort_info->new_data_file_type = STATIC_RECORD; - + sort_info->new_data_file_type= share->state.header.org_data_file_type; /* Set delete_function for sort_delete_record() */ memcpy((char*) &tmp, share, sizeof(*share)); tmp.options= ~HA_OPTION_COMPRESS_RECORD; @@ -4316,3 +4810,13 @@ set_data_file_type(MARIA_SORT_INFO *sort_info, MARIA_SHARE *share) share->delete_record=tmp.delete_record; } } + +static void restore_data_file_type(MARIA_SHARE *share) +{ + share->options&= ~HA_OPTION_COMPRESS_RECORD; + mi_int2store(share->state.header.options,share->options); + share->state.header.data_file_type= + share->state.header.org_data_file_type; + share->data_file_type= share->state.header.data_file_type= + share->pack.header_length= 0; +} diff --git a/storage/maria/ma_checksum.c b/storage/maria/ma_checksum.c index 054873706a4..1b0f683fe63 100644 --- a/storage/maria/ma_checksum.c +++ b/storage/maria/ma_checksum.c @@ -18,23 +18,27 @@ #include "maria_def.h" -ha_checksum _ma_checksum(MARIA_HA *info, const byte *buf) +ha_checksum _ma_checksum(MARIA_HA *info, const byte *record) { uint i; ha_checksum crc=0; MARIA_COLUMNDEF *rec=info->s->rec; - for (i=info->s->base.fields ; i-- ; buf+=(rec++)->length) + if (info->s->base.null_bytes) + crc= my_checksum(crc, record, info->s->base.null_bytes); + + for (i=info->s->base.fields ; i-- ; ) { - const byte *pos; + const byte *pos= record + rec->offset; ulong length; + switch (rec->type) { case FIELD_BLOB: { length= _ma_calc_blob_length(rec->length- maria_portable_sizeof_char_ptr, - buf); - memcpy((char*) &pos, buf+rec->length- maria_portable_sizeof_char_ptr, + pos); + memcpy((char*) &pos, pos+rec->length- maria_portable_sizeof_char_ptr, sizeof(char*)); break; } @@ -42,18 +46,17 @@ ha_checksum _ma_checksum(MARIA_HA *info, const byte *buf) { uint pack_length= HA_VARCHAR_PACKLENGTH(rec->length-1); if (pack_length == 1) - length= (ulong) *(uchar*) buf; + length= (ulong) *(uchar*) pos; else - length= uint2korr(buf); - pos= buf+pack_length; + length= uint2korr(pos); + pos+= pack_length; break; } default: - length=rec->length; - pos=buf; + length= rec->length; break; } - crc=my_checksum(crc, pos ? pos : "", length); + crc= my_checksum(crc, pos ? pos : "", length); } return crc; } diff --git a/storage/maria/ma_close.c b/storage/maria/ma_close.c index 5b940eaf4c3..e5d2985ce47 100644 --- a/storage/maria/ma_close.c +++ b/storage/maria/ma_close.c @@ -28,7 +28,7 @@ int maria_close(register MARIA_HA *info) int error=0,flag; MARIA_SHARE *share=info->s; DBUG_ENTER("maria_close"); - DBUG_PRINT("enter",("base: %lx reopen: %u locks: %u", + DBUG_PRINT("enter",("base: 0x%lx reopen: %u locks: %u", info,(uint) share->reopen, (uint) share->tot_locks)); pthread_mutex_lock(&THR_LOCK_maria); @@ -60,16 +60,22 @@ int maria_close(register MARIA_HA *info) maria_open_list=list_delete(maria_open_list,&info->open_list); pthread_mutex_unlock(&share->intern_lock); - my_free(_ma_get_rec_buff_ptr(info, info->rec_buff), MYF(MY_ALLOW_ZERO_PTR)); + my_free(info->rec_buff, MYF(MY_ALLOW_ZERO_PTR)); + (share->end)(info); + + if (info->s->data_file_type == BLOCK_RECORD) + info->dfile= -1; /* Closed in ma_end_once_block_row */ if (flag) { - if (share->kfile >= 0 && - flush_key_blocks(share->key_cache, share->kfile, - share->temporary ? FLUSH_IGNORE_CHANGED : - FLUSH_RELEASE)) - error=my_errno; if (share->kfile >= 0) { + if ((*share->once_end)(share)) + error= my_errno; + if (flush_key_blocks(share->key_cache, share->kfile, + share->temporary ? FLUSH_IGNORE_CHANGED : + FLUSH_RELEASE)) + error= my_errno; + /* If we are crashed, we can safely flush the current state as it will not change the crashed state. @@ -78,18 +84,13 @@ int maria_close(register MARIA_HA *info) */ if (share->mode != O_RDONLY && maria_is_crashed(info)) _ma_state_info_write(share->kfile, &share->state, 1); - if (my_close(share->kfile,MYF(0))) - error = my_errno; + if (my_close(share->kfile, MYF(0))) + error= my_errno; } #ifdef HAVE_MMAP if (share->file_map) _ma_unmap_file(info); #endif - if (share->decode_trees) - { - my_free((gptr) share->decode_trees,MYF(0)); - my_free((gptr) share->decode_tables,MYF(0)); - } #ifdef THREAD thr_lock_delete(&share->lock); VOID(pthread_mutex_destroy(&share->intern_lock)); diff --git a/storage/maria/ma_create.c b/storage/maria/ma_create.c index b9fb4eb0d5b..82585e78ba7 100644 --- a/storage/maria/ma_create.c +++ b/storage/maria/ma_create.c @@ -18,6 +18,7 @@ #include "ma_ftdefs.h" #include "ma_sp_defs.h" +#include "ma_blockrec.h" #if defined(MSDOS) || defined(__WIN__) #ifdef __WIN__ @@ -28,41 +29,48 @@ #endif #include +static int compare_columns(MARIA_COLUMNDEF **a, MARIA_COLUMNDEF **b); + /* Old options is used when recreating database, from maria_chk */ -int maria_create(const char *name,uint keys,MARIA_KEYDEF *keydefs, - uint columns, MARIA_COLUMNDEF *recinfo, - uint uniques, MARIA_UNIQUEDEF *uniquedefs, - MARIA_CREATE_INFO *ci,uint flags) +int maria_create(const char *name, enum data_file_type record_type, + uint keys,MARIA_KEYDEF *keydefs, + uint columns, MARIA_COLUMNDEF *recinfo, + uint uniques, MARIA_UNIQUEDEF *uniquedefs, + MARIA_CREATE_INFO *ci,uint flags) { register uint i,j; File dfile,file; int errpos,save_errno, create_mode= O_RDWR | O_TRUNC; myf create_flag; - uint fields,length,max_key_length,packed,pointer,real_length_diff, + uint length,max_key_length,packed,pack_bytes,pointer,real_length_diff, key_length,info_length,key_segs,options,min_key_length_skip, base_pos,long_varchar_count,varchar_length, - max_key_block_length,unique_key_parts,fulltext_keys,offset; - uint aligned_key_start, block_length; + unique_key_parts,fulltext_keys,offset; + uint max_field_lengths, extra_header_size; ulong reclength, real_reclength,min_pack_length; char filename[FN_REFLEN],linkname[FN_REFLEN], *linkname_ptr; ulong pack_reclength; ulonglong tot_length,max_rows, tmp; enum en_fieldtype type; + enum data_file_type org_record_type= record_type; MARIA_SHARE share; MARIA_KEYDEF *keydef,tmp_keydef; MARIA_UNIQUEDEF *uniquedef; HA_KEYSEG *keyseg,tmp_keyseg; - MARIA_COLUMNDEF *rec; + MARIA_COLUMNDEF *rec, *rec_end; ulong *rec_per_key_part; - my_off_t key_root[HA_MAX_POSSIBLE_KEY],key_del[MARIA_MAX_KEY_BLOCK_SIZE]; + my_off_t key_root[HA_MAX_POSSIBLE_KEY]; MARIA_CREATE_INFO tmp_create_info; DBUG_ENTER("maria_create"); DBUG_PRINT("enter", ("keys: %u columns: %u uniques: %u flags: %u", keys, columns, uniques, flags)); + LINT_INIT(dfile); + LINT_INIT(file); + if (!ci) { bzero((char*) &tmp_create_info,sizeof(tmp_create_info)); @@ -73,22 +81,24 @@ int maria_create(const char *name,uint keys,MARIA_KEYDEF *keydefs, { DBUG_RETURN(my_errno=HA_WRONG_CREATE_OPTION); } - LINT_INIT(dfile); - LINT_INIT(file); errpos=0; options=0; bzero((byte*) &share,sizeof(share)); if (flags & HA_DONT_TOUCH_DATA) { + org_record_type= ci->org_data_file_type; if (!(ci->old_options & HA_OPTION_TEMP_COMPRESS_RECORD)) options=ci->old_options & (HA_OPTION_COMPRESS_RECORD | HA_OPTION_PACK_RECORD | HA_OPTION_READ_ONLY_DATA | HA_OPTION_CHECKSUM | HA_OPTION_TMP_TABLE | HA_OPTION_DELAY_KEY_WRITE); else + { + /* Uncompressing rows */ options=ci->old_options & (HA_OPTION_CHECKSUM | HA_OPTION_TMP_TABLE | HA_OPTION_DELAY_KEY_WRITE); + } } if (ci->reloc_rows > ci->max_rows) @@ -101,59 +111,90 @@ int maria_create(const char *name,uint keys,MARIA_KEYDEF *keydefs, /* Start by checking fields and field-types used */ - reclength=varchar_length=long_varchar_count=packed= - min_pack_length=pack_reclength=0; - for (rec=recinfo, fields=0 ; - fields != columns ; - rec++,fields++) + varchar_length=long_varchar_count=packed= + pack_reclength= max_field_lengths= 0; + reclength= min_pack_length= ci->null_bytes; + + for (rec= recinfo, rec_end= rec + columns ; rec != rec_end ; rec++) { - reclength+=rec->length; - if ((type=(enum en_fieldtype) rec->type) != FIELD_NORMAL && - type != FIELD_CHECK) + /* Fill in not used struct parts */ + rec->offset= reclength; + rec->empty_pos= 0; + rec->empty_bit= 0; + rec->fill_length= rec->length; + + reclength+= rec->length; + type= rec->type; + if (type == FIELD_SKIP_PRESPACE && record_type == BLOCK_RECORD) + type= FIELD_NORMAL; /* SKIP_PRESPACE not supported */ + + if (type != FIELD_NORMAL && type != FIELD_CHECK) { - packed++; + rec->empty_pos= packed/8; + rec->empty_bit= (1 << (packed & 7)); if (type == FIELD_BLOB) { + packed++; share.base.blobs++; if (pack_reclength != INT_MAX32) { if (rec->length == 4+maria_portable_sizeof_char_ptr) pack_reclength= INT_MAX32; else - pack_reclength+=(1 << ((rec->length-maria_portable_sizeof_char_ptr)*8)); /* Max blob length */ + { + /* Add max possible blob length */ + pack_reclength+= (1 << ((rec->length- + maria_portable_sizeof_char_ptr)*8)); + } } + max_field_lengths+= (rec->length - maria_portable_sizeof_char_ptr); } else if (type == FIELD_SKIP_PRESPACE || type == FIELD_SKIP_ENDSPACE) { - if (pack_reclength != INT_MAX32) - pack_reclength+= rec->length > 255 ? 2 : 1; + max_field_lengths+= rec->length > 255 ? 2 : 1; min_pack_length++; + packed++; } else if (type == FIELD_VARCHAR) { varchar_length+= rec->length-1; /* Used for min_pack_length */ - packed--; pack_reclength++; min_pack_length++; + max_field_lengths++; + packed++; /* We must test for 257 as length includes pack-length */ if (test(rec->length >= 257)) { long_varchar_count++; - pack_reclength+= 2; /* May be packed on 3 bytes */ + max_field_lengths++; } } else if (type != FIELD_SKIP_ZERO) { min_pack_length+=rec->length; - packed--; /* Not a pack record type */ + rec->empty_pos= 0; + rec->empty_bit= 0; } + else + packed++; } else /* FIELD_NORMAL */ + { min_pack_length+=rec->length; + if (!rec->null_bit) + { + share.base.fixed_not_null_fields++; + share.base.fixed_not_null_fields_length+= rec->length; + } + } } if ((packed & 7) == 1) - { /* Bad packing, try to remove a zero-field */ + { + /* + Not optimal packing, try to remove a 1 byte length zero-field as + this will get same record length, but smaller pack overhead + */ while (rec != recinfo) { rec--; @@ -166,14 +207,27 @@ int maria_create(const char *name,uint keys,MARIA_KEYDEF *keydefs, } } } + share.base.null_bytes= ci->null_bytes; + share.base.original_null_bytes= ci->null_bytes; + share.base.transactional= ci->transactional; + share.base.max_field_lengths= max_field_lengths; + share.base.field_offsets= 0; /* for future */ + + if (pack_reclength != INT_MAX32) + pack_reclength+= max_field_lengths + long_varchar_count; - if (packed || (flags & HA_PACK_RECORD)) - options|=HA_OPTION_PACK_RECORD; /* Must use packed records */ - /* We can't use checksum with static length rows */ - if (!(options & HA_OPTION_PACK_RECORD)) + if (packed && record_type == STATIC_RECORD) + record_type= BLOCK_RECORD; + if (record_type == DYNAMIC_RECORD) + options|= HA_OPTION_PACK_RECORD; /* Must use packed records */ + + if (record_type == STATIC_RECORD) + { + /* We can't use checksum with static length rows */ + flags&= ~HA_CREATE_CHECKSUM; options&= ~HA_OPTION_CHECKSUM; - if (!(options & (HA_OPTION_PACK_RECORD | HA_OPTION_COMPRESS_RECORD))) min_pack_length+= varchar_length; + } if (flags & HA_CREATE_TMP_TABLE) { options|= HA_OPTION_TMP_TABLE; @@ -183,18 +237,24 @@ int maria_create(const char *name,uint keys,MARIA_KEYDEF *keydefs, { options|= HA_OPTION_CHECKSUM; min_pack_length++; + pack_reclength++; } if (flags & HA_CREATE_DELAY_KEY_WRITE) options|= HA_OPTION_DELAY_KEY_WRITE; if (flags & HA_CREATE_RELIES_ON_SQL_LAYER) options|= HA_OPTION_RELIES_ON_SQL_LAYER; - packed=(packed+7)/8; + pack_bytes= (packed + 7) / 8; if (pack_reclength != INT_MAX32) - pack_reclength+= reclength+packed + + pack_reclength+= reclength+pack_bytes + test(test_all_bits(options, HA_OPTION_CHECKSUM | HA_PACK_RECORD)); - min_pack_length+=packed; - + min_pack_length+= pack_bytes; + /* Calculate min possible row length for rows-in-block */ + extra_header_size= MAX_FIXED_HEADER_SIZE; + if (ci->transactional) + extra_header_size= TRANS_MAX_FIXED_HEADER_SIZE; + share.base.min_row_length= (extra_header_size + share.base.null_bytes + + pack_bytes); if (!ci->data_file_length && ci->max_rows) { if (pack_reclength == INT_MAX32 || @@ -204,20 +264,57 @@ int maria_create(const char *name,uint keys,MARIA_KEYDEF *keydefs, ci->data_file_length=(ulonglong) ci->max_rows*pack_reclength; } else if (!ci->max_rows) - ci->max_rows=(ha_rows) (ci->data_file_length/(min_pack_length + - ((options & HA_OPTION_PACK_RECORD) ? - 3 : 0))); - - if (options & (HA_OPTION_COMPRESS_RECORD | HA_OPTION_PACK_RECORD)) - pointer=maria_get_pointer_length(ci->data_file_length,maria_data_pointer_size); + { + if (record_type == BLOCK_RECORD) + { + uint rows_per_page= ((maria_block_size - PAGE_OVERHEAD_SIZE) / + (min_pack_length + extra_header_size + + DIR_ENTRY_SIZE)); + ulonglong data_file_length= ci->data_file_length; + if (data_file_length) + data_file_length= ((((ulonglong) 1 << ((BLOCK_RECORD_POINTER_SIZE-1) * + 8)) -1)); + if (rows_per_page > 0) + { + set_if_smaller(rows_per_page, MAX_ROWS_PER_PAGE); + ci->max_rows= ci->data_file_length / maria_block_size * rows_per_page; + } + else + ci->max_rows= ci->data_file_length / (min_pack_length + + extra_header_size + + DIR_ENTRY_SIZE); + } + else + ci->max_rows=(ha_rows) (ci->data_file_length/(min_pack_length + + ((options & + HA_OPTION_PACK_RECORD) ? + 3 : 0))); + } + max_rows= (ulonglong) ci->max_rows; + if (record_type == BLOCK_RECORD) + { + /* The + 1 is for record position withing page */ + pointer= maria_get_pointer_length((ci->data_file_length / + maria_block_size), 3) + 1; + set_if_smaller(pointer, BLOCK_RECORD_POINTER_SIZE); + + if (!max_rows) + max_rows= (((((ulonglong) 1 << ((pointer-1)*8)) -1) * maria_block_size) / + min_pack_length); + } else - pointer=maria_get_pointer_length(ci->max_rows,maria_data_pointer_size); - if (!(max_rows=(ulonglong) ci->max_rows)) - max_rows= ((((ulonglong) 1 << (pointer*8)) -1) / min_pack_length); - + { + if (record_type != STATIC_RECORD) + pointer= maria_get_pointer_length(ci->data_file_length, + maria_data_pointer_size); + else + pointer= maria_get_pointer_length(ci->max_rows, maria_data_pointer_size); + if (!max_rows) + max_rows= ((((ulonglong) 1 << (pointer*8)) -1) / min_pack_length); + } real_reclength=reclength; - if (!(options & (HA_OPTION_COMPRESS_RECORD | HA_OPTION_PACK_RECORD))) + if (record_type == STATIC_RECORD) { if (reclength <= pointer) reclength=pointer+1; /* reserve place for delete link */ @@ -227,19 +324,14 @@ int maria_create(const char *name,uint keys,MARIA_KEYDEF *keydefs, max_key_length=0; tot_length=0 ; key_segs=0; fulltext_keys=0; - max_key_block_length=0; share.state.rec_per_key_part=rec_per_key_part; share.state.key_root=key_root; - share.state.key_del=key_del; + share.state.key_del= HA_OFFSET_ERROR; if (uniques) - { - max_key_block_length= maria_block_size; - max_key_length= MARIA_UNIQUE_HASH_LENGTH + pointer; - } + max_key_length= MARIA_UNIQUE_HASH_LENGTH + pointer; for (i=0, keydef=keydefs ; i < keys ; i++ , keydef++) { - share.state.key_root[i]= HA_OFFSET_ERROR; min_key_length_skip=length=real_length_diff=0; key_length=pointer; @@ -253,11 +345,11 @@ int maria_create(const char *name,uint keys,MARIA_KEYDEF *keydefs, if (flags & HA_DONT_TOUCH_DATA) { /* - called by mariachk - i.e. table structure was taken from - MYI file and SPATIAL key *does have* additional sp_segs keysegs. - keydef->seg here points right at the GEOMETRY segment, - so we only need to decrease keydef->keysegs. - (see maria_recreate_table() in _ma_check.c) + Called by maria_chk - i.e. table structure was taken from + MYI file and SPATIAL key *does have* additional sp_segs keysegs. + keydef->seg here points right at the GEOMETRY segment, + so we only need to decrease keydef->keysegs. + (see maria_recreate_table() in _ma_check.c) */ keydef->keysegs-=sp_segs-1; } @@ -431,35 +523,22 @@ int maria_create(const char *name,uint keys,MARIA_KEYDEF *keydefs, key_segs) share.state.rec_per_key_part[key_segs-1]=1L; length+=key_length; - /* Get block length for key, if defined by user */ - block_length= (keydef->block_length ? - my_round_up_to_next_power(keydef->block_length) : - maria_block_size); - block_length= max(block_length, MARIA_MIN_KEY_BLOCK_LENGTH); - block_length= min(block_length, MARIA_MAX_KEY_BLOCK_LENGTH); - - keydef->block_length= MARIA_BLOCK_SIZE(length-real_length_diff, - pointer,MARIA_MAX_KEYPTR_SIZE, - block_length); - if (keydef->block_length > MARIA_MAX_KEY_BLOCK_LENGTH || - length >= HA_MAX_KEY_BUFF) + if (length >= min(HA_MAX_KEY_BUFF, MARIA_MAX_KEY_LENGTH)) { my_errno=HA_WRONG_CREATE_OPTION; goto err; } - set_if_bigger(max_key_block_length,keydef->block_length); + keydef->block_length= maria_block_size; keydef->keylength= (uint16) key_length; keydef->minlength= (uint16) (length-min_key_length_skip); keydef->maxlength= (uint16) length; if (length > max_key_length) max_key_length= length; - tot_length+= (max_rows/(ulong) (((uint) keydef->block_length-5)/ - (length*2)))* - (ulong) keydef->block_length; + tot_length+= ((max_rows/(ulong) (((uint) maria_block_size-5)/ + (length*2))) * + maria_block_size); } - for (i=max_key_block_length/MARIA_MIN_KEY_BLOCK_LENGTH ; i-- ; ) - key_del[i]=HA_OFFSET_ERROR; unique_key_parts=0; offset=reclength-uniques*MARIA_UNIQUE_HASH_LENGTH; @@ -476,14 +555,12 @@ int maria_create(const char *name,uint keys,MARIA_KEYDEF *keydefs, key_segs+=uniques; /* Each unique has 1 key seg */ base_pos=(MARIA_STATE_INFO_SIZE + keys * MARIA_STATE_KEY_SIZE + - max_key_block_length/MARIA_MIN_KEY_BLOCK_LENGTH* - MARIA_STATE_KEYBLOCK_SIZE+ - key_segs*MARIA_STATE_KEYSEG_SIZE); - info_length=base_pos+(uint) (MARIA_BASE_INFO_SIZE+ - keys * MARIA_KEYDEF_SIZE+ - uniques * MARIA_UNIQUEDEF_SIZE + - (key_segs + unique_key_parts)*HA_KEYSEG_SIZE+ - columns*MARIA_COLUMNDEF_SIZE); + key_segs * MARIA_STATE_KEYSEG_SIZE); + info_length= base_pos+(uint) (MARIA_BASE_INFO_SIZE+ + keys * MARIA_KEYDEF_SIZE+ + uniques * MARIA_UNIQUEDEF_SIZE + + (key_segs + unique_key_parts)*HA_KEYSEG_SIZE+ + columns*MARIA_COLUMNDEF_SIZE); DBUG_PRINT("info", ("info_length: %u", info_length)); /* There are only 16 bits for the total header length. */ @@ -505,11 +582,13 @@ int maria_create(const char *name,uint keys,MARIA_KEYDEF *keydefs, mi_int2store(share.state.header.state_info_length,MARIA_STATE_INFO_SIZE); mi_int2store(share.state.header.base_info_length,MARIA_BASE_INFO_SIZE); mi_int2store(share.state.header.base_pos,base_pos); + share.state.header.data_file_type= record_type; + share.state.header.org_data_file_type= org_record_type; share.state.header.language= (ci->language ? ci->language : default_charset_info->number); - share.state.header.max_block_size_index= max_key_block_length/MARIA_MIN_KEY_BLOCK_LENGTH; share.state.dellink = HA_OFFSET_ERROR; + share.state.first_bitmap_with_space= 0; share.state.process= (ulong) getpid(); share.state.unique= (ulong) 0; share.state.update_count=(ulong) 0; @@ -518,9 +597,11 @@ int maria_create(const char *name,uint keys,MARIA_KEYDEF *keydefs, share.state.auto_increment=ci->auto_increment; share.options=options; share.base.rec_reflength=pointer; + share.base.block_size= maria_block_size; + /* Get estimate for index file length (this may be wrong for FT keys) */ - tmp= (tot_length + max_key_block_length * keys * - MARIA_INDEX_BLOCK_MARGIN) / MARIA_MIN_KEY_BLOCK_LENGTH; + tmp= (tot_length + maria_block_size * keys * + MARIA_INDEX_BLOCK_MARGIN) / maria_block_size; /* use maximum of key_file_length we calculated and key_file_length value we got from MYI file header (see also mariapack.c:save_state) @@ -534,13 +615,10 @@ int maria_create(const char *name,uint keys,MARIA_KEYDEF *keydefs, mi_int2store(share.state.header.unique_key_parts,unique_key_parts); maria_set_all_keys_active(share.state.key_map, keys); - aligned_key_start= my_round_up_to_next_power(max_key_block_length ? - max_key_block_length : - maria_block_size); share.base.keystart = share.state.state.key_file_length= - MY_ALIGN(info_length, aligned_key_start); - share.base.max_key_block_length=max_key_block_length; + MY_ALIGN(info_length, maria_block_size); + share.base.max_key_block_length= maria_block_size; share.base.max_key_length=ALIGN_SIZE(max_key_length+4); share.base.records=ci->max_rows; share.base.reloc= ci->reloc_rows; @@ -548,9 +626,9 @@ int maria_create(const char *name,uint keys,MARIA_KEYDEF *keydefs, share.base.pack_reclength=reclength+ test(options & HA_OPTION_CHECKSUM); share.base.max_pack_length=pack_reclength; share.base.min_pack_length=min_pack_length; - share.base.pack_bits=packed; - share.base.fields=fields; - share.base.pack_fields=packed; + share.base.pack_bytes= pack_bytes; + share.base.fields= columns; + share.base.pack_fields= packed; #ifdef USE_RAID share.base.raid_type=ci->raid_type; share.base.raid_chunks=ci->raid_chunks; @@ -559,7 +637,7 @@ int maria_create(const char *name,uint keys,MARIA_KEYDEF *keydefs, /* max_data_file_length and max_key_file_length are recalculated on open */ if (options & HA_OPTION_TMP_TABLE) - share.base.max_data_file_length=(my_off_t) ci->data_file_length; + share.base.max_data_file_length= (my_off_t) ci->data_file_length; share.base.min_block_length= (share.base.pack_reclength+3 < MARIA_EXTEND_BLOCK_LENGTH && @@ -632,72 +710,63 @@ int maria_create(const char *name,uint keys,MARIA_KEYDEF *keydefs, if (!(flags & HA_DONT_TOUCH_DATA)) { -#ifdef USE_RAID - if (share.base.raid_type) - { - (void) fn_format(filename, name, "", MARIA_NAME_DEXT, - MY_UNPACK_FILENAME | MY_APPEND_EXT); - if ((dfile=my_raid_create(filename, 0, create_mode, - share.base.raid_type, - share.base.raid_chunks, - share.base.raid_chunksize, - MYF(MY_WME | MY_RAID))) < 0) - goto err; - } - else -#endif + if (ci->data_file_name) { - if (ci->data_file_name) - { - char *dext= strrchr(ci->data_file_name, '.'); - int have_dext= dext && !strcmp(dext, MARIA_NAME_DEXT); + char *dext= strrchr(ci->data_file_name, '.'); + int have_dext= dext && !strcmp(dext, MARIA_NAME_DEXT); - if (options & HA_OPTION_TMP_TABLE) - { - char *path; - /* chop off the table name, tempory tables use generated name */ - if ((path= strrchr(ci->data_file_name, FN_LIBCHAR))) - *path= '\0'; - fn_format(filename, name, ci->data_file_name, MARIA_NAME_DEXT, - MY_REPLACE_DIR | MY_UNPACK_FILENAME | MY_APPEND_EXT); - } - else - { - fn_format(filename, ci->data_file_name, "", MARIA_NAME_DEXT, - MY_UNPACK_FILENAME | - (have_dext ? MY_REPLACE_EXT : MY_APPEND_EXT)); - } - fn_format(linkname, name, "",MARIA_NAME_DEXT, - MY_UNPACK_FILENAME | MY_APPEND_EXT); - linkname_ptr=linkname; - create_flag=0; + if (options & HA_OPTION_TMP_TABLE) + { + char *path; + /* chop off the table name, tempory tables use generated name */ + if ((path= strrchr(ci->data_file_name, FN_LIBCHAR))) + *path= '\0'; + fn_format(filename, name, ci->data_file_name, MARIA_NAME_DEXT, + MY_REPLACE_DIR | MY_UNPACK_FILENAME | MY_APPEND_EXT); } else { - fn_format(filename,name,"", MARIA_NAME_DEXT, - MY_UNPACK_FILENAME | MY_APPEND_EXT); - linkname_ptr=0; - create_flag=MY_DELETE_OLD; + fn_format(filename, ci->data_file_name, "", MARIA_NAME_DEXT, + MY_UNPACK_FILENAME | + (have_dext ? MY_REPLACE_EXT : MY_APPEND_EXT)); } - if ((dfile= - my_create_with_symlink(linkname_ptr, filename, 0, create_mode, - MYF(MY_WME | create_flag))) < 0) - goto err; + fn_format(linkname, name, "",MARIA_NAME_DEXT, + MY_UNPACK_FILENAME | MY_APPEND_EXT); + linkname_ptr=linkname; + create_flag=0; + } + else + { + fn_format(filename,name,"", MARIA_NAME_DEXT, + MY_UNPACK_FILENAME | MY_APPEND_EXT); + linkname_ptr=0; + create_flag=MY_DELETE_OLD; } + if ((dfile= + my_create_with_symlink(linkname_ptr, filename, 0, create_mode, + MYF(MY_WME | create_flag))) < 0) + goto err; errpos=3; + + if (record_type == BLOCK_RECORD) + { + /* Write one bitmap page */ + char buff[IO_SIZE]; + uint i; + bzero(buff, sizeof(buff)); + for (i= 0 ; i < maria_block_size ; i+= IO_SIZE) + if (my_write(dfile, (byte*) buff, sizeof(buff), MYF(MY_NABP))) + goto err; + share.state.state.data_file_length= maria_block_size; + } } DBUG_PRINT("info", ("write state info and base info")); if (_ma_state_info_write(file, &share.state, 2) || _ma_base_info_write(file, &share.base)) goto err; -#ifndef DBUG_OFF - if ((uint) my_tell(file,MYF(0)) != base_pos+ MARIA_BASE_INFO_SIZE) - { - uint pos=(uint) my_tell(file,MYF(0)); - DBUG_PRINT("warning",("base_length: %d != used_length: %d", - base_pos+ MARIA_BASE_INFO_SIZE, pos)); - } -#endif + DBUG_PRINT("info", ("base_pos: %d base_info_size: %d", + base_pos, MARIA_BASE_INFO_SIZE)); + DBUG_ASSERT(my_tell(file,MYF(0)) == base_pos+ MARIA_BASE_INFO_SIZE); /* Write key and keyseg definitions */ DBUG_PRINT("info", ("write key and keyseg definitions")); @@ -738,7 +807,7 @@ int maria_create(const char *name,uint keys,MARIA_KEYDEF *keydefs, { tmp_keydef.keysegs=1; tmp_keydef.flag= HA_UNIQUE_CHECK; - tmp_keydef.block_length= (uint16)maria_block_size; + tmp_keydef.block_length= (uint16) maria_block_size; tmp_keydef.keylength= MARIA_UNIQUE_HASH_LENGTH + pointer; tmp_keydef.minlength=tmp_keydef.maxlength=tmp_keydef.keylength; tmp_keyseg.type= MARIA_UNIQUE_HASH_TYPE; @@ -776,6 +845,7 @@ int maria_create(const char *name,uint keys,MARIA_KEYDEF *keydefs, } break; default: + DBUG_ASSERT((keyseg->flag & HA_VAR_LENGTH_PART) == 0); break; } if (_ma_keyseg_write(file, keyseg)) @@ -783,9 +853,27 @@ int maria_create(const char *name,uint keys,MARIA_KEYDEF *keydefs, } } DBUG_PRINT("info", ("write field definitions")); - for (i=0 ; i < share.base.fields ; i++) - if (_ma_recinfo_write(file, &recinfo[i])) + if (record_type == BLOCK_RECORD) + { + /* Store columns in a more efficent order */ + MARIA_COLUMNDEF **tmp, **pos; + if (!(tmp= (MARIA_COLUMNDEF**) my_malloc(share.base.fields * + sizeof(MARIA_COLUMNDEF*), + MYF(MY_WME)))) goto err; + for (rec= recinfo, pos= tmp ; rec != rec_end ; rec++, pos++) + *pos= rec; + qsort(tmp, share.base.fields, sizeof(*tmp), (qsort_cmp) compare_columns); + for (i=0 ; i < share.base.fields ; i++) + if (_ma_recinfo_write(file, tmp[i])) + goto err; + } + else + { + for (i=0 ; i < share.base.fields ; i++) + if (_ma_recinfo_write(file, &recinfo[i])) + goto err; + } #ifndef DBUG_OFF if ((uint) my_tell(file,MYF(0)) != info_length) @@ -797,7 +885,8 @@ int maria_create(const char *name,uint keys,MARIA_KEYDEF *keydefs, #endif /* Enlarge files */ - DBUG_PRINT("info", ("enlarge to keystart: %lu", (ulong) share.base.keystart)); + DBUG_PRINT("info", ("enlarge to keystart: %lu", + (ulong) share.base.keystart)); if (my_chsize(file,(ulong) share.base.keystart,0,MYF(0))) goto err; @@ -826,7 +915,6 @@ err: VOID(my_close(dfile,MYF(0))); /* fall through */ case 2: - /* QQ: Tõnu should add a call to my_raid_delete() here */ if (! (flags & HA_DONT_TOUCH_DATA)) my_delete_with_symlink(fn_format(filename,name,"",MARIA_NAME_DEXT, MY_UNPACK_FILENAME | MY_APPEND_EXT), @@ -868,3 +956,59 @@ uint maria_get_pointer_length(ulonglong file_length, uint def) } return def; } + + +/* + Sort columns for records-in-block + + IMPLEMENTATION + Sort columns in following order: + + Fixed size, not null columns + Fixed length, null fields + Variable length fields (CHAR, VARCHAR) + Blobs + + For same kind of fields, keep fields in original order +*/ + +static inline int sign(longlong a) +{ + return a < 0 ? -1 : (a > 0 ? 1 : 0); +} + + +int compare_columns(MARIA_COLUMNDEF **a_ptr, MARIA_COLUMNDEF **b_ptr) +{ + MARIA_COLUMNDEF *a= *a_ptr, *b= *b_ptr; + enum en_fieldtype a_type, b_type; + + a_type= (a->type == FIELD_NORMAL || a->type == FIELD_CHECK ? + FIELD_NORMAL : a->type); + b_type= (b->type == FIELD_NORMAL || b->type == FIELD_CHECK ? + FIELD_NORMAL : b->type); + + if (a_type == FIELD_NORMAL && !a->null_bit) + { + if (b_type != FIELD_NORMAL || b->null_bit) + return -1; + return sign((long) (a->offset - b->offset)); + } + if (b_type == FIELD_NORMAL && !b->null_bit) + return 1; + if (a_type == b_type) + return sign((long) (a->offset - b->offset)); + if (a_type == FIELD_NORMAL) + return -1; + if (b_type == FIELD_NORMAL) + return 1; + if (a_type == FIELD_BLOB) + return 1; + if (b_type == FIELD_BLOB) + return -1; + return sign((long) (a->offset - b->offset)); +} + + + + diff --git a/storage/maria/ma_dbug.c b/storage/maria/ma_dbug.c index 7f2bff85047..596bf18bfe5 100644 --- a/storage/maria/ma_dbug.c +++ b/storage/maria/ma_dbug.c @@ -21,15 +21,15 @@ /* Print a key in user understandable format */ void _ma_print_key(FILE *stream, register HA_KEYSEG *keyseg, - const uchar *key, uint length) + const byte *key, uint length) { int flag; short int s_1; long int l_1; float f_1; double d_1; - const uchar *end; - const uchar *key_end=key+length; + const byte *end; + const byte *key_end= key + length; VOID(fputs("Key: \"",stream)); flag=0; diff --git a/storage/maria/ma_delete.c b/storage/maria/ma_delete.c index d1ad9edbed5..2669585202a 100644 --- a/storage/maria/ma_delete.c +++ b/storage/maria/ma_delete.c @@ -20,26 +20,25 @@ #include "ma_rt_index.h" static int d_search(MARIA_HA *info,MARIA_KEYDEF *keyinfo,uint comp_flag, - uchar *key,uint key_length,my_off_t page,uchar *anc_buff); -static int del(MARIA_HA *info,MARIA_KEYDEF *keyinfo,uchar *key,uchar *anc_buff, - my_off_t leaf_page,uchar *leaf_buff,uchar *keypos, - my_off_t next_block,uchar *ret_key); -static int underflow(MARIA_HA *info,MARIA_KEYDEF *keyinfo,uchar *anc_buff, - my_off_t leaf_page,uchar *leaf_buff,uchar *keypos); -static uint remove_key(MARIA_KEYDEF *keyinfo,uint nod_flag,uchar *keypos, - uchar *lastkey,uchar *page_end, + byte *key,uint key_length,my_off_t page,byte *anc_buff); +static int del(MARIA_HA *info,MARIA_KEYDEF *keyinfo,byte *key,byte *anc_buff, + my_off_t leaf_page,byte *leaf_buff,byte *keypos, + my_off_t next_block,byte *ret_key); +static int underflow(MARIA_HA *info,MARIA_KEYDEF *keyinfo,byte *anc_buff, + my_off_t leaf_page,byte *leaf_buff,byte *keypos); +static uint remove_key(MARIA_KEYDEF *keyinfo,uint nod_flag,byte *keypos, + byte *lastkey,byte *page_end, my_off_t *next_block); static int _ma_ck_real_delete(register MARIA_HA *info,MARIA_KEYDEF *keyinfo, - uchar *key, uint key_length, my_off_t *root); + byte *key, uint key_length, my_off_t *root); int maria_delete(MARIA_HA *info,const byte *record) { uint i; - uchar *old_key; + byte *old_key; int save_errno; char lastpos[8]; - MARIA_SHARE *share=info->s; DBUG_ENTER("maria_delete"); @@ -61,17 +60,15 @@ int maria_delete(MARIA_HA *info,const byte *record) } if (_ma_readinfo(info,F_WRLCK,1)) DBUG_RETURN(my_errno); - if (info->s->calc_checksum) - info->checksum=(*info->s->calc_checksum)(info,record); if ((*share->compare_record)(info,record)) goto err; /* Error on read-check */ if (_ma_mark_file_changed(info)) goto err; - /* Remove all keys from the .ISAM file */ + /* Remove all keys from the index file */ - old_key=info->lastkey2; + old_key= info->lastkey2; for (i=0 ; i < share->base.keys ; i++ ) { if (maria_is_key_active(info->s->state.key_map, i)) @@ -79,13 +76,13 @@ int maria_delete(MARIA_HA *info,const byte *record) info->s->keyinfo[i].version++; if (info->s->keyinfo[i].flag & HA_FULLTEXT ) { - if (_ma_ft_del(info,i,(char*) old_key,record,info->lastpos)) + if (_ma_ft_del(info,i,(char*) old_key,record,info->cur_row.lastpos)) goto err; } else { if (info->s->keyinfo[i].ck_delete(info,i,old_key, - _ma_make_key(info,i,old_key,record,info->lastpos))) + _ma_make_key(info,i,old_key,record,info->cur_row.lastpos))) goto err; } /* The above changed info->lastkey2. Inform maria_rnext_same(). */ @@ -95,12 +92,21 @@ int maria_delete(MARIA_HA *info,const byte *record) if ((*share->delete_record)(info)) goto err; /* Remove record from database */ - info->state->checksum-=info->checksum; + + /* + We can't use the row based checksum as this doesn't have enough + precision. + */ + if (info->s->calc_checksum) + { + info->cur_row.checksum= (*info->s->calc_checksum)(info,record); + info->state->checksum-= info->cur_row.checksum; + } info->update= HA_STATE_CHANGED+HA_STATE_DELETED+HA_STATE_ROW_CHANGED; info->state->records--; - mi_sizestore(lastpos,info->lastpos); + mi_sizestore(lastpos, info->cur_row.lastpos); VOID(_ma_writeinfo(info,WRITEINFO_UPDATE_KEYFILE)); allow_break(); /* Allow SIGHUP & SIGINT */ if (info->invalidator != 0) @@ -113,7 +119,7 @@ int maria_delete(MARIA_HA *info,const byte *record) err: save_errno=my_errno; - mi_sizestore(lastpos,info->lastpos); + mi_sizestore(lastpos, info->cur_row.lastpos); if (save_errno != HA_ERR_RECORD_CHANGED) { maria_print_error(info->s, HA_ERR_CRASHED); @@ -135,7 +141,7 @@ err: /* Remove a key from the btree index */ -int _ma_ck_delete(register MARIA_HA *info, uint keynr, uchar *key, +int _ma_ck_delete(register MARIA_HA *info, uint keynr, byte *key, uint key_length) { return _ma_ck_real_delete(info, info->s->keyinfo+keynr, key, key_length, @@ -144,12 +150,12 @@ int _ma_ck_delete(register MARIA_HA *info, uint keynr, uchar *key, static int _ma_ck_real_delete(register MARIA_HA *info, MARIA_KEYDEF *keyinfo, - uchar *key, uint key_length, my_off_t *root) + byte *key, uint key_length, my_off_t *root) { int error; uint nod_flag; my_off_t old_root; - uchar *root_buff; + byte *root_buff; DBUG_ENTER("_ma_ck_real_delete"); if ((old_root=*root) == HA_OFFSET_ERROR) @@ -157,7 +163,7 @@ static int _ma_ck_real_delete(register MARIA_HA *info, MARIA_KEYDEF *keyinfo, maria_print_error(info->s, HA_ERR_CRASHED); DBUG_RETURN(my_errno=HA_ERR_CRASHED); } - if (!(root_buff= (uchar*) my_alloca((uint) keyinfo->block_length+ + if (!(root_buff= (byte*) my_alloca((uint) keyinfo->block_length+ HA_MAX_KEY_BUFF*2))) { DBUG_PRINT("error",("Couldn't allocate memory")); @@ -212,17 +218,17 @@ err: */ static int d_search(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, - uint comp_flag, uchar *key, uint key_length, - my_off_t page, uchar *anc_buff) + uint comp_flag, byte *key, uint key_length, + my_off_t page, byte *anc_buff) { int flag,ret_value,save_flag; uint length,nod_flag,search_key_length; my_bool last_key; - uchar *leaf_buff,*keypos; + byte *leaf_buff,*keypos; my_off_t leaf_page,next_block; - uchar lastkey[HA_MAX_KEY_BUFF]; + byte lastkey[HA_MAX_KEY_BUFF]; DBUG_ENTER("d_search"); - DBUG_DUMP("page",(byte*) anc_buff,maria_getint(anc_buff)); + DBUG_DUMP("page",anc_buff,maria_getint(anc_buff)); search_key_length= (comp_flag & SEARCH_FIND) ? key_length : USE_WHOLE_KEY; flag=(*keyinfo->bin_search)(info,keyinfo,anc_buff,key, search_key_length, @@ -264,7 +270,7 @@ static int d_search(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, /* popular word. two-level tree. going down */ uint tmp_key_length; my_off_t root; - uchar *kpos=keypos; + byte *kpos=keypos; if (!(tmp_key_length=(*keyinfo->get_key)(keyinfo,nod_flag,&kpos,lastkey))) { @@ -304,8 +310,8 @@ static int d_search(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, if (nod_flag) { leaf_page= _ma_kpos(nod_flag,keypos); - if (!(leaf_buff= (uchar*) my_alloca((uint) keyinfo->block_length+ - HA_MAX_KEY_BUFF*2))) + if (!(leaf_buff= (byte*) my_alloca((uint) keyinfo->block_length+ + HA_MAX_KEY_BUFF*2))) { DBUG_PRINT("error",("Couldn't allocate memory")); my_errno=ENOMEM; @@ -366,7 +372,7 @@ static int d_search(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, if (!_ma_get_last_key(info,keyinfo,anc_buff,lastkey,keypos,&length)) goto err; ret_value= _ma_insert(info,keyinfo,key,anc_buff,keypos,lastkey, - (uchar*) 0,(uchar*) 0,(my_off_t) 0,(my_bool) 0); + (byte*) 0,(byte*) 0,(my_off_t) 0,(my_bool) 0); } } if (ret_value == 0 && maria_getint(anc_buff) > keyinfo->block_length) @@ -378,14 +384,14 @@ static int d_search(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, ret_value|= _ma_write_keypage(info,keyinfo,page,DFLT_INIT_HITS,anc_buff); else { - DBUG_DUMP("page",(byte*) anc_buff,maria_getint(anc_buff)); + DBUG_DUMP("page",anc_buff,maria_getint(anc_buff)); } - my_afree((byte*) leaf_buff); + my_afree(leaf_buff); DBUG_PRINT("exit",("Return: %d",ret_value)); DBUG_RETURN(ret_value); err: - my_afree((byte*) leaf_buff); + my_afree(leaf_buff); DBUG_PRINT("exit",("Error: %d",my_errno)); DBUG_RETURN (-1); } /* d_search */ @@ -393,24 +399,25 @@ err: /* Remove a key that has a page-reference */ -static int del(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, uchar *key, - uchar *anc_buff, my_off_t leaf_page, uchar *leaf_buff, - uchar *keypos, /* Pos to where deleted key was */ +static int del(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, + byte *key, byte *anc_buff, my_off_t leaf_page, + byte *leaf_buff, + byte *keypos, /* Pos to where deleted key was */ my_off_t next_block, - uchar *ret_key) /* key before keypos in anc_buff */ + byte *ret_key) /* key before keypos in anc_buff */ { int ret_value,length; uint a_length,nod_flag,tmp; my_off_t next_page; - uchar keybuff[HA_MAX_KEY_BUFF],*endpos,*next_buff,*key_start, *prev_key; + byte keybuff[HA_MAX_KEY_BUFF],*endpos,*next_buff,*key_start, *prev_key; MARIA_SHARE *share=info->s; MARIA_KEY_PARAM s_temp; DBUG_ENTER("del"); DBUG_PRINT("enter",("leaf_page: %ld keypos: 0x%lx", leaf_page, (ulong) keypos)); - DBUG_DUMP("leaf_buff",(byte*) leaf_buff,maria_getint(leaf_buff)); + DBUG_DUMP("leaf_buff",leaf_buff,maria_getint(leaf_buff)); - endpos=leaf_buff+maria_getint(leaf_buff); + endpos= leaf_buff+ maria_getint(leaf_buff); if (!(key_start= _ma_get_last_key(info,keyinfo,leaf_buff,keybuff,endpos, &tmp))) DBUG_RETURN(-1); @@ -418,14 +425,14 @@ static int del(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, uchar *k if ((nod_flag=_ma_test_if_nod(leaf_buff))) { next_page= _ma_kpos(nod_flag,endpos); - if (!(next_buff= (uchar*) my_alloca((uint) keyinfo->block_length+ + if (!(next_buff= (byte*) my_alloca((uint) keyinfo->block_length+ HA_MAX_KEY_BUFF*2))) DBUG_RETURN(-1); if (!_ma_fetch_keypage(info,keyinfo,next_page,DFLT_INIT_HITS,next_buff,0)) ret_value= -1; else { - DBUG_DUMP("next_page",(byte*) next_buff,maria_getint(next_buff)); + DBUG_DUMP("next_page",next_buff,maria_getint(next_buff)); if ((ret_value=del(info,keyinfo,key,anc_buff,next_page,next_buff, keypos,next_block,ret_key)) >0) { @@ -446,13 +453,13 @@ static int del(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, uchar *k &tmp)) goto err; ret_value= _ma_insert(info,keyinfo,key,leaf_buff,endpos,keybuff, - (uchar*) 0,(uchar*) 0,(my_off_t) 0,0); + (byte*) 0,(byte*) 0,(my_off_t) 0,0); } } if (_ma_write_keypage(info,keyinfo,leaf_page,DFLT_INIT_HITS,leaf_buff)) goto err; } - my_afree((byte*) next_buff); + my_afree(next_buff); DBUG_RETURN(ret_value); } @@ -472,11 +479,11 @@ static int del(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, uchar *k prev_key=(keypos == anc_buff+2+share->base.key_reflength ? 0 : ret_key); length=(*keyinfo->pack_key)(keyinfo,share->base.key_reflength, - keypos == endpos ? (uchar*) 0 : keypos, + keypos == endpos ? (byte*) 0 : keypos, prev_key, prev_key, keybuff,&s_temp); if (length > 0) - bmove_upp((byte*) endpos+length,(byte*) endpos,(uint) (endpos-keypos)); + bmove_upp(endpos+length,endpos,(uint) (endpos-keypos)); else bmove(keypos,keypos-length, (int) (endpos-keypos)+length); (*keyinfo->store_key)(keyinfo,keypos,&s_temp); @@ -497,28 +504,28 @@ err: /* Balances adjacent pages if underflow occours */ static int underflow(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, - uchar *anc_buff, + byte *anc_buff, my_off_t leaf_page,/* Ancestor page and underflow page */ - uchar *leaf_buff, - uchar *keypos) /* Position to pos after key */ + byte *leaf_buff, + byte *keypos) /* Position to pos after key */ { int t_length; uint length,anc_length,buff_length,leaf_length,p_length,s_length,nod_flag, key_reflength,key_length; my_off_t next_page; - uchar anc_key[HA_MAX_KEY_BUFF],leaf_key[HA_MAX_KEY_BUFF], - *buff,*endpos,*next_keypos,*anc_pos,*half_pos,*temp_pos,*prev_key, - *after_key; + byte anc_key[HA_MAX_KEY_BUFF],leaf_key[HA_MAX_KEY_BUFF]; + byte *buff,*endpos,*next_keypos,*anc_pos,*half_pos,*temp_pos,*prev_key; + byte *after_key; MARIA_KEY_PARAM s_temp; MARIA_SHARE *share=info->s; DBUG_ENTER("underflow"); DBUG_PRINT("enter",("leaf_page: %ld keypos: 0x%lx",(long) leaf_page, (ulong) keypos)); - DBUG_DUMP("anc_buff",(byte*) anc_buff,maria_getint(anc_buff)); - DBUG_DUMP("leaf_buff",(byte*) leaf_buff,maria_getint(leaf_buff)); + DBUG_DUMP("anc_buff",anc_buff,maria_getint(anc_buff)); + DBUG_DUMP("leaf_buff",leaf_buff,maria_getint(leaf_buff)); buff=info->buff; - info->buff_used=1; + info->keybuff_used=1; next_keypos=keypos; nod_flag=_ma_test_if_nod(leaf_buff); p_length=nod_flag+2; @@ -551,26 +558,25 @@ static int underflow(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, if (!_ma_fetch_keypage(info,keyinfo,next_page,DFLT_INIT_HITS,buff,0)) goto err; buff_length=maria_getint(buff); - DBUG_DUMP("next",(byte*) buff,buff_length); + DBUG_DUMP("next",buff,buff_length); /* find keys to make a big key-page */ - bmove((byte*) next_keypos-key_reflength,(byte*) buff+2, - key_reflength); + bmove(next_keypos-key_reflength, buff+2, key_reflength); if (!_ma_get_last_key(info,keyinfo,anc_buff,anc_key,next_keypos,&length) || !_ma_get_last_key(info,keyinfo,leaf_buff,leaf_key, leaf_buff+leaf_length,&length)) goto err; /* merge pages and put parting key from anc_buff between */ - prev_key=(leaf_length == p_length ? (uchar*) 0 : leaf_key); + prev_key=(leaf_length == p_length ? (byte*) 0 : leaf_key); t_length=(*keyinfo->pack_key)(keyinfo,nod_flag,buff+p_length, prev_key, prev_key, anc_key, &s_temp); length=buff_length-p_length; endpos=buff+length+leaf_length+t_length; /* buff will always be larger than before !*/ - bmove_upp((byte*) endpos, (byte*) buff+buff_length,length); - memcpy((byte*) buff, (byte*) leaf_buff,(size_t) leaf_length); + bmove_upp(endpos, buff+buff_length,length); + memcpy(buff, leaf_buff,(size_t) leaf_length); (*keyinfo->store_key)(keyinfo,buff+leaf_length,&s_temp); buff_length=(uint) (endpos-buff); maria_putint(buff,buff_length,nod_flag); @@ -586,14 +592,14 @@ static int underflow(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, if (buff_length <= keyinfo->block_length) { /* Keys in one page */ - memcpy((byte*) leaf_buff,(byte*) buff,(size_t) buff_length); + memcpy(leaf_buff,buff,(size_t) buff_length); if (_ma_dispose(info,keyinfo,next_page,DFLT_INIT_HITS)) goto err; } else { /* Page is full */ endpos=anc_buff+anc_length; - DBUG_PRINT("test",("anc_buff: %lx endpos: %lx",anc_buff,endpos)); + DBUG_PRINT("test",("anc_buff: 0x%lx endpos: 0x%lx",anc_buff,endpos)); if (keypos != anc_buff+2+key_reflength && !_ma_get_last_key(info,keyinfo,anc_buff,anc_key,keypos,&length)) goto err; @@ -601,22 +607,21 @@ static int underflow(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, &key_length, &after_key))) goto err; length=(uint) (half_pos-buff); - memcpy((byte*) leaf_buff,(byte*) buff,(size_t) length); + memcpy(leaf_buff,buff,(size_t) length); maria_putint(leaf_buff,length,nod_flag); /* Correct new keypointer to leaf_page */ half_pos=after_key; _ma_kpointer(info,leaf_key+key_length,next_page); /* Save key in anc_buff */ - prev_key=(keypos == anc_buff+2+key_reflength ? (uchar*) 0 : anc_key), + prev_key=(keypos == anc_buff+2+key_reflength ? (byte*) 0 : anc_key), t_length=(*keyinfo->pack_key)(keyinfo,key_reflength, - (keypos == endpos ? (uchar*) 0 : + (keypos == endpos ? (byte*) 0 : keypos), prev_key, prev_key, leaf_key, &s_temp); if (t_length >= 0) - bmove_upp((byte*) endpos+t_length,(byte*) endpos, - (uint) (endpos-keypos)); + bmove_upp(endpos+t_length, endpos, (uint) (endpos-keypos)); else bmove(keypos,keypos-t_length,(uint) (endpos-keypos)+t_length); (*keyinfo->store_key)(keyinfo,keypos,&s_temp); @@ -624,15 +629,15 @@ static int underflow(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, /* Store key first in new page */ if (nod_flag) - bmove((byte*) buff+2,(byte*) half_pos-nod_flag,(size_t) nod_flag); + bmove(buff+2,half_pos-nod_flag,(size_t) nod_flag); if (!(*keyinfo->get_key)(keyinfo,nod_flag,&half_pos,leaf_key)) goto err; - t_length=(int) (*keyinfo->pack_key)(keyinfo, nod_flag, (uchar*) 0, - (uchar*) 0, (uchar *) 0, + t_length=(int) (*keyinfo->pack_key)(keyinfo, nod_flag, (byte*) 0, + (byte*) 0, (byte*) 0, leaf_key, &s_temp); /* t_length will always be > 0 for a new page !*/ length=(uint) ((buff+maria_getint(buff))-half_pos); - bmove((byte*) buff+p_length+t_length,(byte*) half_pos,(size_t) length); + bmove(buff+p_length+t_length, half_pos, (size_t) length); (*keyinfo->store_key)(keyinfo,buff+p_length,&s_temp); maria_putint(buff,length+t_length+p_length,nod_flag); @@ -655,11 +660,10 @@ static int underflow(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, goto err; buff_length=maria_getint(buff); endpos=buff+buff_length; - DBUG_DUMP("prev",(byte*) buff,buff_length); + DBUG_DUMP("prev",buff,buff_length); /* find keys to make a big key-page */ - bmove((byte*) next_keypos - key_reflength,(byte*) leaf_buff+2, - key_reflength); + bmove(next_keypos - key_reflength, leaf_buff+2, key_reflength); next_keypos=keypos; if (!(*keyinfo->get_key)(keyinfo,key_reflength,&next_keypos, anc_key)) @@ -668,17 +672,17 @@ static int underflow(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, goto err; /* merge pages and put parting key from anc_buff between */ - prev_key=(leaf_length == p_length ? (uchar*) 0 : leaf_key); + prev_key=(leaf_length == p_length ? (byte*) 0 : leaf_key); t_length=(*keyinfo->pack_key)(keyinfo,nod_flag, (leaf_length == p_length ? - (uchar*) 0 : leaf_buff+p_length), + (byte*) 0 : leaf_buff+p_length), prev_key, prev_key, anc_key, &s_temp); if (t_length >= 0) - bmove((byte*) endpos+t_length,(byte*) leaf_buff+p_length, - (size_t) (leaf_length-p_length)); + bmove(endpos+t_length, leaf_buff+p_length, + (size_t) (leaf_length-p_length)); else /* We gained space */ - bmove((byte*) endpos,(byte*) leaf_buff+((int) p_length-t_length), + bmove(endpos,leaf_buff+((int) p_length-t_length), (size_t) (leaf_length-p_length+t_length)); (*keyinfo->store_key)(keyinfo,endpos,&s_temp); @@ -711,18 +715,17 @@ static int underflow(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, goto err; _ma_kpointer(info,leaf_key+key_length,leaf_page); /* Save key in anc_buff */ - DBUG_DUMP("anc_buff",(byte*) anc_buff,anc_length); - DBUG_DUMP("key_to_anc",(byte*) leaf_key,key_length); + DBUG_DUMP("anc_buff",anc_buff,anc_length); + DBUG_DUMP("key_to_anc",leaf_key,key_length); temp_pos=anc_buff+anc_length; t_length=(*keyinfo->pack_key)(keyinfo,key_reflength, - keypos == temp_pos ? (uchar*) 0 + keypos == temp_pos ? (byte*) 0 : keypos, anc_pos, anc_pos, leaf_key,&s_temp); if (t_length > 0) - bmove_upp((byte*) temp_pos+t_length,(byte*) temp_pos, - (uint) (temp_pos-keypos)); + bmove_upp(temp_pos+t_length, temp_pos, (uint) (temp_pos-keypos)); else bmove(keypos,keypos-t_length,(uint) (temp_pos-keypos)+t_length); (*keyinfo->store_key)(keyinfo,keypos,&s_temp); @@ -730,15 +733,15 @@ static int underflow(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, /* Store first key on new page */ if (nod_flag) - bmove((byte*) leaf_buff+2,(byte*) half_pos-nod_flag,(size_t) nod_flag); + bmove(leaf_buff+2,half_pos-nod_flag,(size_t) nod_flag); if (!(length=(*keyinfo->get_key)(keyinfo,nod_flag,&half_pos,leaf_key))) goto err; - DBUG_DUMP("key_to_leaf",(byte*) leaf_key,length); - t_length=(*keyinfo->pack_key)(keyinfo,nod_flag, (uchar*) 0, - (uchar*) 0, (uchar*) 0, leaf_key, &s_temp); + DBUG_DUMP("key_to_leaf",leaf_key,length); + t_length=(*keyinfo->pack_key)(keyinfo,nod_flag, (byte*) 0, + (byte*) 0, (byte*) 0, leaf_key, &s_temp); length=(uint) ((buff+buff_length)-half_pos); DBUG_PRINT("info",("t_length: %d length: %d",t_length,(int) length)); - bmove((byte*) leaf_buff+p_length+t_length,(byte*) half_pos, + bmove(leaf_buff+p_length+t_length,half_pos, (size_t) length); (*keyinfo->store_key)(keyinfo,leaf_buff+p_length,&s_temp); maria_putint(leaf_buff,length+t_length+p_length,nod_flag); @@ -763,15 +766,15 @@ err: */ static uint remove_key(MARIA_KEYDEF *keyinfo, uint nod_flag, - uchar *keypos, /* Where key starts */ - uchar *lastkey, /* key to be removed */ - uchar *page_end, /* End of page */ + byte *keypos, /* Where key starts */ + byte *lastkey, /* key to be removed */ + byte *page_end, /* End of page */ my_off_t *next_block) /* ptr to next block */ { int s_length; - uchar *start; + byte *start; DBUG_ENTER("remove_key"); - DBUG_PRINT("enter",("keypos: %lx page_end: %lx",keypos,page_end)); + DBUG_PRINT("enter",("keypos: 0x%lx page_end: 0x%lx", keypos, page_end)); start=keypos; if (!(keyinfo->flag & @@ -795,7 +798,7 @@ static uint remove_key(MARIA_KEYDEF *keyinfo, uint nod_flag, { if (keyinfo->flag & HA_BINARY_PACK_KEY) { - uchar *old_key=start; + byte *old_key=start; uint next_length,prev_length,prev_pack_length; get_key_length(next_length,keypos); get_key_pack_length(prev_length,prev_pack_length,old_key); @@ -882,7 +885,6 @@ static uint remove_key(MARIA_KEYDEF *keyinfo, uint nod_flag, } } end: - bmove((byte*) start,(byte*) start+s_length, - (uint) (page_end-start-s_length)); + bmove(start, start+s_length, (uint) (page_end-start-s_length)); DBUG_RETURN((uint) s_length); } /* remove_key */ diff --git a/storage/maria/ma_delete_all.c b/storage/maria/ma_delete_all.c index b16d82ed9f7..3e1448e858a 100644 --- a/storage/maria/ma_delete_all.c +++ b/storage/maria/ma_delete_all.c @@ -43,8 +43,7 @@ int maria_delete_all_rows(MARIA_HA *info) info->state->empty=info->state->key_empty=0; info->state->checksum=0; - for (i=share->base.max_key_block_length/MARIA_MIN_KEY_BLOCK_LENGTH ; i-- ; ) - state->key_del[i]= HA_OFFSET_ERROR; + state->key_del= HA_OFFSET_ERROR; for (i=0 ; i < share->base.keys ; i++) state->key_root[i]= HA_OFFSET_ERROR; diff --git a/storage/maria/ma_dynrec.c b/storage/maria/ma_dynrec.c index 047826408c3..a1d4ee58bf1 100644 --- a/storage/maria/ma_dynrec.c +++ b/storage/maria/ma_dynrec.c @@ -26,19 +26,16 @@ #include "maria_def.h" -/* Enough for comparing if number is zero */ -static char zero_string[]={0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0}; - -static int write_dynamic_record(MARIA_HA *info,const byte *record, - ulong reclength); +static my_bool write_dynamic_record(MARIA_HA *info,const byte *record, + ulong reclength); static int _ma_find_writepos(MARIA_HA *info,ulong reclength,my_off_t *filepos, ulong *length); -static int update_dynamic_record(MARIA_HA *info,my_off_t filepos,byte *record, - ulong reclength); -static int delete_dynamic_record(MARIA_HA *info,my_off_t filepos, - uint second_read); -static int _ma_cmp_buffer(File file, const byte *buff, my_off_t filepos, - uint length); +static my_bool update_dynamic_record(MARIA_HA *info, MARIA_RECORD_POS filepos, + byte *record, ulong reclength); +static my_bool delete_dynamic_record(MARIA_HA *info,MARIA_RECORD_POS filepos, + uint second_read); +static my_bool _ma_cmp_buffer(File file, const byte *buff, my_off_t filepos, + uint length); #ifdef THREAD /* Play it safe; We have a small stack when using threads */ @@ -224,19 +221,26 @@ uint _ma_nommap_pwrite(MARIA_HA *info, byte *Buffer, } -int _ma_write_dynamic_record(MARIA_HA *info, const byte *record) +my_bool _ma_write_dynamic_record(MARIA_HA *info, const byte *record) { - ulong reclength= _ma_rec_pack(info,info->rec_buff,record); - return (write_dynamic_record(info,info->rec_buff,reclength)); + ulong reclength= _ma_rec_pack(info,info->rec_buff + MARIA_REC_BUFF_OFFSET, + record); + return (write_dynamic_record(info,info->rec_buff + MARIA_REC_BUFF_OFFSET, + reclength)); } -int _ma_update_dynamic_record(MARIA_HA *info, my_off_t pos, const byte *record) +my_bool _ma_update_dynamic_record(MARIA_HA *info, MARIA_RECORD_POS pos, + const byte *record) { - uint length= _ma_rec_pack(info,info->rec_buff,record); - return (update_dynamic_record(info,pos,info->rec_buff,length)); + uint length= _ma_rec_pack(info, info->rec_buff + MARIA_REC_BUFF_OFFSET, + record); + return (update_dynamic_record(info, pos, + info->rec_buff + MARIA_REC_BUFF_OFFSET, + length)); } -int _ma_write_blob_record(MARIA_HA *info, const byte *record) + +my_bool _ma_write_blob_record(MARIA_HA *info, const byte *record) { byte *rec_buff; int error; @@ -246,31 +250,27 @@ int _ma_write_blob_record(MARIA_HA *info, const byte *record) MARIA_DYN_DELETE_BLOCK_HEADER+1); reclength= (info->s->base.pack_reclength + _ma_calc_total_blob_length(info,record)+ extra); -#ifdef NOT_USED /* We now support big rows */ - if (reclength > MARIA_DYN_MAX_ROW_LENGTH) - { - my_errno=HA_ERR_TO_BIG_ROW; - return -1; - } -#endif if (!(rec_buff=(byte*) my_alloca(reclength))) { my_errno=ENOMEM; - return(-1); + return(1); } - reclength2= _ma_rec_pack(info,rec_buff+ALIGN_SIZE(MARIA_MAX_DYN_BLOCK_HEADER), + reclength2= _ma_rec_pack(info, + rec_buff+ALIGN_SIZE(MARIA_MAX_DYN_BLOCK_HEADER), record); DBUG_PRINT("info",("reclength: %lu reclength2: %lu", reclength, reclength2)); DBUG_ASSERT(reclength2 <= reclength); - error=write_dynamic_record(info,rec_buff+ALIGN_SIZE(MARIA_MAX_DYN_BLOCK_HEADER), - reclength2); + error= write_dynamic_record(info, + rec_buff+ALIGN_SIZE(MARIA_MAX_DYN_BLOCK_HEADER), + reclength2); my_afree(rec_buff); - return(error); + return(error != 0); } -int _ma_update_blob_record(MARIA_HA *info, my_off_t pos, const byte *record) +my_bool _ma_update_blob_record(MARIA_HA *info, MARIA_RECORD_POS pos, + const byte *record) { byte *rec_buff; int error; @@ -284,13 +284,13 @@ int _ma_update_blob_record(MARIA_HA *info, my_off_t pos, const byte *record) if (reclength > MARIA_DYN_MAX_ROW_LENGTH) { my_errno=HA_ERR_TO_BIG_ROW; - return -1; + return 1; } #endif if (!(rec_buff=(byte*) my_alloca(reclength))) { my_errno=ENOMEM; - return(-1); + return(1); } reclength= _ma_rec_pack(info,rec_buff+ALIGN_SIZE(MARIA_MAX_DYN_BLOCK_HEADER), record); @@ -298,20 +298,20 @@ int _ma_update_blob_record(MARIA_HA *info, my_off_t pos, const byte *record) rec_buff+ALIGN_SIZE(MARIA_MAX_DYN_BLOCK_HEADER), reclength); my_afree(rec_buff); - return(error); + return(error != 0); } -int _ma_delete_dynamic_record(MARIA_HA *info) +my_bool _ma_delete_dynamic_record(MARIA_HA *info) { - return delete_dynamic_record(info,info->lastpos,0); + return delete_dynamic_record(info, info->cur_row.lastpos, 0); } /* Write record to data-file */ -static int write_dynamic_record(MARIA_HA *info, const byte *record, - ulong reclength) +static my_bool write_dynamic_record(MARIA_HA *info, const byte *record, + ulong reclength) { int flag; ulong length; @@ -443,8 +443,8 @@ static bool unlink_deleted_block(MARIA_HA *info, MARIA_BLOCK_INFO *block_info) (maria_rrnd() or maria_scan(), then ensure that we skip over this block when doing next maria_rrnd() or maria_scan(). */ - if (info->nextpos == block_info->filepos) - info->nextpos+=block_info->block_len; + if (info->cur_row.nextpos == block_info->filepos) + info->cur_row.nextpos+= block_info->block_len; DBUG_RETURN(0); } @@ -464,8 +464,9 @@ static bool unlink_deleted_block(MARIA_HA *info, MARIA_BLOCK_INFO *block_info) 1 error. In this case my_error is set. */ -static int update_backward_delete_link(MARIA_HA *info, my_off_t delete_block, - my_off_t filepos) +static my_bool update_backward_delete_link(MARIA_HA *info, + my_off_t delete_block, + MARIA_RECORD_POS filepos) { MARIA_BLOCK_INFO block_info; DBUG_ENTER("update_backward_delete_link"); @@ -490,11 +491,11 @@ static int update_backward_delete_link(MARIA_HA *info, my_off_t delete_block, DBUG_RETURN(0); } - /* Delete datarecord from database */ - /* info->rec_cache.seek_not_done is updated in cmp_record */ +/* Delete datarecord from database */ +/* info->rec_cache.seek_not_done is updated in cmp_record */ -static int delete_dynamic_record(MARIA_HA *info, my_off_t filepos, - uint second_read) +static my_bool delete_dynamic_record(MARIA_HA *info, MARIA_RECORD_POS filepos, + uint second_read) { uint length,b_type; MARIA_BLOCK_INFO block_info,del_block; @@ -522,7 +523,8 @@ static int delete_dynamic_record(MARIA_HA *info, my_off_t filepos, del_block.second_read=0; remove_next_block=0; if (_ma_get_block_info(&del_block,info->dfile,filepos+length) & - BLOCK_DELETED && del_block.block_len+length < MARIA_DYN_MAX_BLOCK_LENGTH) + BLOCK_DELETED && del_block.block_len+length < + MARIA_DYN_MAX_BLOCK_LENGTH) { /* We can't remove this yet as this block may be the head block */ remove_next_block=1; @@ -719,8 +721,9 @@ int _ma_write_part_record(MARIA_HA *info, else { info->rec_cache.seek_not_done=1; - if (info->s->file_write(info,(byte*) *record-head_length,length+extra_length+ - del_length,filepos,info->s->write_flag)) + if (info->s->file_write(info,(byte*) *record-head_length, + length+extra_length+ + del_length,filepos,info->s->write_flag)) goto err; } memcpy(record_end,temp,(size_t) (extra_length+del_length)); @@ -745,8 +748,8 @@ err: /* update record from datafile */ -static int update_dynamic_record(MARIA_HA *info, my_off_t filepos, byte *record, - ulong reclength) +static my_bool update_dynamic_record(MARIA_HA *info, MARIA_RECORD_POS filepos, + byte *record, ulong reclength) { int flag; uint error; @@ -784,8 +787,8 @@ static int update_dynamic_record(MARIA_HA *info, my_off_t filepos, byte *record, { /* extend file */ DBUG_PRINT("info",("Extending file with %d bytes",tmp)); - if (info->nextpos == info->state->data_file_length) - info->nextpos+= tmp; + if (info->cur_row.nextpos == info->state->data_file_length) + info->cur_row.nextpos+= tmp; info->state->data_file_length+= tmp; info->update|= HA_STATE_WRITE_AT_END | HA_STATE_EXTEND_BLOCK; length+=tmp; @@ -829,8 +832,8 @@ static int update_dynamic_record(MARIA_HA *info, my_off_t filepos, byte *record, mi_int3store(del_block.header+1, rest_length); mi_sizestore(del_block.header+4,info->s->state.dellink); bfill(del_block.header+12,8,255); - if (info->s->file_write(info,(byte*) del_block.header,20, next_pos, - MYF(MY_NABP))) + if (info->s->file_write(info,(byte*) del_block.header, 20, + next_pos, MYF(MY_NABP))) DBUG_RETURN(1); info->s->state.dellink= next_pos; info->s->state.split++; @@ -877,9 +880,18 @@ uint _ma_rec_pack(MARIA_HA *info, register byte *to, register const byte *from) MARIA_BLOB *blob; DBUG_ENTER("_ma_rec_pack"); - flag=0 ; bit=1; - startpos=packpos=to; to+= info->s->base.pack_bits; blob=info->blobs; - rec=info->s->rec; + flag= 0; + bit= 1; + startpos= packpos=to; + to+= info->s->base.pack_bytes; + blob= info->blobs; + rec= info->s->rec; + if (info->s->base.null_bytes) + { + memcpy(to, from, info->s->base.null_bytes); + from+= info->s->base.null_bytes; + to+= info->s->base.null_bytes; + } for (i=info->s->base.fields ; i-- > 0; from+= length,rec++) { @@ -903,7 +915,7 @@ uint _ma_rec_pack(MARIA_HA *info, register byte *to, register const byte *from) } else if (type == FIELD_SKIP_ZERO) { - if (memcmp((byte*) from,zero_string,length) == 0) + if (memcmp((byte*) from, maria_zero_string, length) == 0) flag|=bit; else { @@ -981,7 +993,7 @@ uint _ma_rec_pack(MARIA_HA *info, register byte *to, register const byte *from) if (bit != 1) *packpos= (char) (uchar) flag; if (info->s->calc_checksum) - *to++=(char) info->checksum; + *to++= (byte) info->cur_row.checksum; DBUG_PRINT("exit",("packed length: %d",(int) (to-startpos))); DBUG_RETURN((uint) (to-startpos)); } /* _ma_rec_pack */ @@ -989,7 +1001,7 @@ uint _ma_rec_pack(MARIA_HA *info, register byte *to, register const byte *from) /* - Check if a record was correctly packed. Used only by mariachk + Check if a record was correctly packed. Used only by maria_chk Returns 0 if record is ok. */ @@ -1002,9 +1014,11 @@ my_bool _ma_rec_check(MARIA_HA *info,const char *record, byte *rec_buff, reg3 MARIA_COLUMNDEF *rec; DBUG_ENTER("_ma_rec_check"); - packpos=rec_buff; to= rec_buff+info->s->base.pack_bits; + packpos=rec_buff; to= rec_buff+info->s->base.pack_bytes; rec=info->s->rec; flag= *packpos; bit=1; + record+= info->s->base.null_bytes; + to+= info->s->base.null_bytes; for (i=info->s->base.fields ; i-- > 0; record+= length, rec++) { @@ -1022,7 +1036,7 @@ my_bool _ma_rec_check(MARIA_HA *info,const char *record, byte *rec_buff, } else if (type == FIELD_SKIP_ZERO) { - if (memcmp((byte*) record,zero_string,length) == 0) + if (memcmp((byte*) record, maria_zero_string, length) == 0) { if (!(flag & bit)) goto err; @@ -1098,7 +1112,7 @@ my_bool _ma_rec_check(MARIA_HA *info,const char *record, byte *rec_buff, if (packed_length != (uint) (to - rec_buff) + test(info->s->calc_checksum) || (bit != 1 && (flag & ~(bit - 1)))) goto err; - if (with_checksum && ((uchar) info->checksum != (uchar) *to)) + if (with_checksum && ((uchar) info->cur_row.checksum != (uchar) *to)) { DBUG_PRINT("error",("wrong checksum for row")); goto err; @@ -1129,8 +1143,16 @@ ulong _ma_rec_unpack(register MARIA_HA *info, register byte *to, byte *from, flag= (uchar) *from; bit=1; packpos=from; if (found_length < info->s->base.min_pack_length) goto err; - from+= info->s->base.pack_bits; - min_pack_length=info->s->base.min_pack_length - info->s->base.pack_bits; + from+= info->s->base.pack_bytes; + min_pack_length= info->s->base.min_pack_length - info->s->base.pack_bytes; + + if ((length= info->s->base.null_bytes)) + { + memcpy(to, from, length); + from+= length; + to+= length; + min_pack_length-= length; + } for (rec=info->s->rec , end_field=rec+info->s->base.fields ; rec < end_field ; to+= rec_length, rec++) @@ -1234,13 +1256,13 @@ ulong _ma_rec_unpack(register MARIA_HA *info, register byte *to, byte *from, } } if (info->s->calc_checksum) - from++; + info->cur_row.checksum= (uint) (uchar) *from++; if (to == to_end && from == from_end && (bit == 1 || !(flag & ~(bit-1)))) DBUG_RETURN(found_length); err: my_errno= HA_ERR_WRONG_IN_RECORD; - DBUG_PRINT("error",("to_end: %lx -> %lx from_end: %lx -> %lx", + DBUG_PRINT("error",("to_end: 0x%lx -> 0x%lx from_end: 0x%lx -> 0x%lx", to,to_end,from,from_end)); DBUG_DUMP("from",(byte*) info->rec_buff,info->s->base.min_pack_length); DBUG_RETURN(MY_FILE_ERROR); @@ -1305,16 +1327,17 @@ void _ma_store_blob_length(byte *pos,uint pack_length,uint length) /* Read record from datafile */ - /* Returns 0 if ok, -1 if error */ + /* Returns 0 if ok, 1 if error */ -int _ma_read_dynamic_record(MARIA_HA *info, my_off_t filepos, byte *buf) +int _ma_read_dynamic_record(MARIA_HA *info, byte *buf, + MARIA_RECORD_POS filepos) { int flag; uint b_type,left_length; byte *to; MARIA_BLOCK_INFO block_info; File file; - DBUG_ENTER("maria_read_dynamic_record"); + DBUG_ENTER("_ma_read_dynamic_record"); if (filepos != HA_OFFSET_ERROR) { @@ -1349,17 +1372,18 @@ int _ma_read_dynamic_record(MARIA_HA *info, my_off_t filepos, byte *buf) goto panic; if (info->s->base.blobs) { - if (!(to=_ma_alloc_rec_buff(info, block_info.rec_len, - &info->rec_buff))) + if (_ma_alloc_buffer(&info->rec_buff, &info->rec_buff_size, + block_info.rec_len + + info->s->base.extra_rec_buff_size)) goto err; } - else - to= info->rec_buff; + to= info->rec_buff; left_length=block_info.rec_len; } if (left_length < block_info.data_len || ! block_info.data_len) goto panic; /* Wrong linked record */ - if (info->s->file_read(info,(byte*) to,block_info.data_len,block_info.filepos, + if (info->s->file_read(info,(byte*) to,block_info.data_len, + block_info.filepos, MYF(MY_NABP))) goto panic; left_length-=block_info.data_len; @@ -1369,55 +1393,61 @@ int _ma_read_dynamic_record(MARIA_HA *info, my_off_t filepos, byte *buf) info->update|= HA_STATE_AKTIV; /* We have a aktive record */ fast_ma_writeinfo(info); DBUG_RETURN(_ma_rec_unpack(info,buf,info->rec_buff,block_info.rec_len) != - MY_FILE_ERROR ? 0 : -1); + MY_FILE_ERROR ? 0 : 1); } fast_ma_writeinfo(info); - DBUG_RETURN(-1); /* Wrong data to read */ + DBUG_RETURN(1); /* Wrong data to read */ panic: my_errno=HA_ERR_WRONG_IN_RECORD; err: VOID(_ma_writeinfo(info,0)); - DBUG_RETURN(-1); + DBUG_RETURN(1); } /* compare unique constraint between stored rows */ -int _ma_cmp_dynamic_unique(MARIA_HA *info, MARIA_UNIQUEDEF *def, - const byte *record, my_off_t pos) +my_bool _ma_cmp_dynamic_unique(MARIA_HA *info, MARIA_UNIQUEDEF *def, + const byte *record, MARIA_RECORD_POS pos) { - byte *rec_buff,*old_record; - int error; + byte *old_rec_buff,*old_record; + my_off_t old_rec_buff_size; + my_bool error; DBUG_ENTER("_ma_cmp_dynamic_unique"); if (!(old_record=my_alloca(info->s->base.reclength))) DBUG_RETURN(1); /* Don't let the compare destroy blobs that may be in use */ - rec_buff=info->rec_buff; + old_rec_buff= info->rec_buff; + old_rec_buff_size= info->rec_buff_size; + if (info->s->base.blobs) - info->rec_buff=0; - error= _ma_read_dynamic_record(info,pos,old_record); + info->rec_buff= 0; + error= _ma_read_dynamic_record(info, old_record, pos) != 0; if (!error) - error=_ma_unique_comp(def, record, old_record, def->null_are_equal); + error=_ma_unique_comp(def, record, old_record, def->null_are_equal) != 0; if (info->s->base.blobs) { - my_free(_ma_get_rec_buff_ptr(info, info->rec_buff), MYF(MY_ALLOW_ZERO_PTR)); - info->rec_buff=rec_buff; + my_free(info->rec_buff, MYF(MY_ALLOW_ZERO_PTR)); + info->rec_buff= old_rec_buff; + info->rec_buff_size= old_rec_buff_size; } my_afree(old_record); DBUG_RETURN(error); } - /* Compare of record one disk with packed record in memory */ + /* Compare of record on disk with packed record in memory */ -int _ma_cmp_dynamic_record(register MARIA_HA *info, register const byte *record) +my_bool _ma_cmp_dynamic_record(register MARIA_HA *info, + register const byte *record) { - uint flag,reclength,b_type; + uint flag, reclength, b_type,cmp_length; my_off_t filepos; byte *buffer; MARIA_BLOCK_INFO block_info; + my_bool error= 1; DBUG_ENTER("_ma_cmp_dynamic_record"); /* We are going to do changes; dont let anybody disturb */ @@ -1427,7 +1457,7 @@ int _ma_cmp_dynamic_record(register MARIA_HA *info, register const byte *record) { info->update&= ~(HA_STATE_WRITE_AT_END | HA_STATE_EXTEND_BLOCK); if (flush_io_cache(&info->rec_cache)) - DBUG_RETURN(-1); + DBUG_RETURN(1); } info->rec_cache.seek_not_done=1; @@ -1440,12 +1470,12 @@ int _ma_cmp_dynamic_record(register MARIA_HA *info, register const byte *record) { if (!(buffer=(byte*) my_alloca(info->s->base.pack_reclength+ _ma_calc_total_blob_length(info,record)))) - DBUG_RETURN(-1); + DBUG_RETURN(1); } reclength= _ma_rec_pack(info,buffer,record); record= buffer; - filepos=info->lastpos; + filepos= info->cur_row.lastpos; flag=block_info.second_read=0; block_info.next_filepos=filepos; while (reclength > 0) @@ -1472,9 +1502,13 @@ int _ma_cmp_dynamic_record(register MARIA_HA *info, register const byte *record) my_errno=HA_ERR_WRONG_IN_RECORD; goto err; } - reclength-=block_info.data_len; + reclength-= block_info.data_len; + cmp_length= block_info.data_len; + if (!reclength && info->s->calc_checksum) + cmp_length--; /* 'record' may not contain checksum */ + if (_ma_cmp_buffer(info->dfile,record,block_info.filepos, - block_info.data_len)) + cmp_length)) { my_errno=HA_ERR_RECORD_CHANGED; goto err; @@ -1484,17 +1518,19 @@ int _ma_cmp_dynamic_record(register MARIA_HA *info, register const byte *record) } } my_errno=0; + error= 0; err: if (buffer != info->rec_buff) my_afree((gptr) buffer); - DBUG_RETURN(my_errno); + DBUG_PRINT("exit", ("result: %d", error)); + DBUG_RETURN(error); } /* Compare file to buffert */ -static int _ma_cmp_buffer(File file, const byte *buff, my_off_t filepos, - uint length) +static my_bool _ma_cmp_buffer(File file, const byte *buff, my_off_t filepos, + uint length) { uint next_length; char temp_buff[IO_SIZE*2]; @@ -1514,14 +1550,15 @@ static int _ma_cmp_buffer(File file, const byte *buff, my_off_t filepos, } if (my_pread(file,temp_buff,length,filepos,MYF(MY_NABP))) goto err; - DBUG_RETURN(memcmp((byte*) buff,temp_buff,length)); + DBUG_RETURN(memcmp((byte*) buff,temp_buff,length) != 0); err: DBUG_RETURN(1); } -int _ma_read_rnd_dynamic_record(MARIA_HA *info, byte *buf, - register my_off_t filepos, +int _ma_read_rnd_dynamic_record(MARIA_HA *info, + byte *buf, + register MARIA_RECORD_POS filepos, my_bool skip_deleted_blocks) { int flag,info_read,save_errno; @@ -1594,9 +1631,9 @@ int _ma_read_rnd_dynamic_record(MARIA_HA *info, byte *buf, } if (b_type & (BLOCK_DELETED | BLOCK_SYNC_ERROR)) { - my_errno=HA_ERR_RECORD_DELETED; - info->lastpos=block_info.filepos; - info->nextpos=block_info.filepos+block_info.block_len; + my_errno= HA_ERR_RECORD_DELETED; + info->cur_row.lastpos= block_info.filepos; + info->cur_row.nextpos= block_info.filepos+block_info.block_len; } goto err; } @@ -1604,15 +1641,15 @@ int _ma_read_rnd_dynamic_record(MARIA_HA *info, byte *buf, { if (block_info.rec_len > (uint) share->base.max_pack_length) goto panic; - info->lastpos=filepos; + info->cur_row.lastpos= filepos; if (share->base.blobs) { - if (!(to= _ma_alloc_rec_buff(info, block_info.rec_len, - &info->rec_buff))) + if (_ma_alloc_buffer(&info->rec_buff, &info->rec_buff_size, + block_info.rec_len + + info->s->base.extra_rec_buff_size)) goto err; } - else - to= info->rec_buff; + to= info->rec_buff; left_len=block_info.rec_len; } if (left_len < block_info.data_len) @@ -1658,7 +1695,7 @@ int _ma_read_rnd_dynamic_record(MARIA_HA *info, byte *buf, } if (flag++ == 0) { - info->nextpos=block_info.filepos+block_info.block_len; + info->cur_row.nextpos= block_info.filepos+block_info.block_len; skip_deleted_blocks=0; } left_len-=block_info.data_len; diff --git a/storage/maria/ma_extra.c b/storage/maria/ma_extra.c index d600fedb99b..a75b8084214 100644 --- a/storage/maria/ma_extra.c +++ b/storage/maria/ma_extra.c @@ -19,7 +19,8 @@ #include #endif -static void maria_extra_keyflag(MARIA_HA *info, enum ha_extra_function function); +static void maria_extra_keyflag(MARIA_HA *info, + enum ha_extra_function function); /* @@ -38,7 +39,8 @@ static void maria_extra_keyflag(MARIA_HA *info, enum ha_extra_function function) # error */ -int maria_extra(MARIA_HA *info, enum ha_extra_function function, void *extra_arg) +int maria_extra(MARIA_HA *info, enum ha_extra_function function, + void *extra_arg) { int error=0; ulong cache_size; @@ -49,7 +51,7 @@ int maria_extra(MARIA_HA *info, enum ha_extra_function function, void *extra_arg switch (function) { case HA_EXTRA_RESET_STATE: /* Reset state (don't free buffers) */ info->lastinx= 0; /* Use first index as def */ - info->last_search_keypage=info->lastpos= HA_OFFSET_ERROR; + info->last_search_keypage= info->cur_row.lastpos= HA_OFFSET_ERROR; info->page_changed=1; /* Next/prev gives first/last */ if (info->opt_flag & READ_CACHE_USED) @@ -115,7 +117,7 @@ int maria_extra(MARIA_HA *info, enum ha_extra_function function, void *extra_arg case HA_EXTRA_REINIT_CACHE: if (info->opt_flag & READ_CACHE_USED) { - reinit_io_cache(&info->rec_cache,READ_CACHE,info->nextpos, + reinit_io_cache(&info->rec_cache, READ_CACHE, info->cur_row.nextpos, (pbool) (info->lock_type != F_UNLCK), (pbool) test(info->update & HA_STATE_ROW_CHANGED)); info->update&= ~HA_STATE_ROW_CHANGED; @@ -185,7 +187,7 @@ int maria_extra(MARIA_HA *info, enum ha_extra_function function, void *extra_arg (byte*) info->lastkey,info->lastkey_length); info->save_update= info->update; info->save_lastinx= info->lastinx; - info->save_lastpos= info->lastpos; + info->save_lastpos= info->cur_row.lastpos; info->save_lastkey_length=info->lastkey_length; if (function == HA_EXTRA_REMEMBER_POS) break; @@ -203,7 +205,7 @@ int maria_extra(MARIA_HA *info, enum ha_extra_function function, void *extra_arg info->save_lastkey_length); info->update= info->save_update | HA_STATE_WRITTEN; info->lastinx= info->save_lastinx; - info->lastpos= info->save_lastpos; + info->cur_row.lastpos= info->save_lastpos; info->lastkey_length=info->save_lastkey_length; } info->read_record= share->read_record; @@ -327,8 +329,13 @@ int maria_extra(MARIA_HA *info, enum ha_extra_function function, void *extra_arg maria_mark_crashed(info); /* Fatal error found */ } } - if (share->base.blobs) - _ma_alloc_rec_buff(info, -1, &info->rec_buff); + if (share->base.blobs && info->rec_buff_size > + share->base.default_rec_buff_size) + { + info->rec_buff_size= 1; /* Force realloc */ + _ma_alloc_buffer(&info->rec_buff, &info->rec_buff_size, + share->base.default_rec_buff_size); + } break; case HA_EXTRA_NORMAL: /* Theese isn't in use */ info->quick_mode=0; @@ -419,8 +426,13 @@ int maria_reset(MARIA_HA *info) info->opt_flag&= ~(READ_CACHE_USED | WRITE_CACHE_USED); error= end_io_cache(&info->rec_cache); } - if (share->base.blobs) - _ma_alloc_rec_buff(info, -1, &info->rec_buff); + if (share->base.blobs && info->rec_buff_size > + share->base.default_rec_buff_size) + { + info->rec_buff_size= 1; /* Force realloc */ + _ma_alloc_buffer(&info->rec_buff, &info->rec_buff_size, + share->base.default_rec_buff_size); + } #if defined(HAVE_MMAP) && defined(HAVE_MADVISE) if (info->opt_flag & MEMMAP_USED) madvise(share->file_map,share->state.state.data_file_length,MADV_RANDOM); @@ -428,7 +440,7 @@ int maria_reset(MARIA_HA *info) info->opt_flag&= ~(KEY_READ_USED | REMEMBER_OLD_POS); info->quick_mode=0; info->lastinx= 0; /* Use first index as def */ - info->last_search_keypage= info->lastpos= HA_OFFSET_ERROR; + info->last_search_keypage= info->cur_row.lastpos= HA_OFFSET_ERROR; info->page_changed= 1; info->update= ((info->update & HA_STATE_CHANGED) | HA_STATE_NEXT_FOUND | HA_STATE_PREV_FOUND); diff --git a/storage/maria/ma_ft_boolean_search.c b/storage/maria/ma_ft_boolean_search.c index 83901cb5e47..40ac88bfcbf 100644 --- a/storage/maria/ma_ft_boolean_search.c +++ b/storage/maria/ma_ft_boolean_search.c @@ -332,7 +332,7 @@ static int _ft2_search(FTB *ftb, FTB_WORD *ftbw, my_bool init_search) my_bool can_go_down; MARIA_HA *info=ftb->info; uint off, extra=HA_FT_WLEN+info->s->base.rec_reflength; - byte *lastkey_buf=ftbw->word+ftbw->off; + byte *lastkey_buf= ftbw->word+ftbw->off; LINT_INIT(off); if (ftbw->flags & FTB_FLAG_TRUNC) @@ -343,7 +343,7 @@ static int _ft2_search(FTB *ftb, FTB_WORD *ftbw, my_bool init_search) ftbw->key_root=info->s->state.key_root[ftb->keynr]; ftbw->keyinfo=info->s->keyinfo+ftb->keynr; - r= _ma_search(info, ftbw->keyinfo, (uchar*) ftbw->word, ftbw->len, + r= _ma_search(info, ftbw->keyinfo, ftbw->word, ftbw->len, SEARCH_FIND | SEARCH_BIGGER, ftbw->key_root); } else @@ -352,10 +352,10 @@ static int _ft2_search(FTB *ftb, FTB_WORD *ftbw, my_bool init_search) if (ftbw->docid[0] < *ftbw->max_docid) { sflag|= SEARCH_SAME; - _ma_dpointer(info, (uchar *)(ftbw->word + ftbw->len + HA_FT_WLEN), + _ma_dpointer(info, (ftbw->word + ftbw->len + HA_FT_WLEN), *ftbw->max_docid); } - r= _ma_search(info, ftbw->keyinfo, (uchar*) lastkey_buf, + r= _ma_search(info, ftbw->keyinfo, lastkey_buf, USE_WHOLE_KEY, sflag, ftbw->key_root); } @@ -369,7 +369,7 @@ static int _ft2_search(FTB *ftb, FTB_WORD *ftbw, my_bool init_search) off=info->lastkey_length-extra; subkeys=ft_sintXkorr(info->lastkey+off); } - if (subkeys<0 || info->lastpos < info->state->data_file_length) + if (subkeys<0 || info->cur_row.lastpos < info->state->data_file_length) break; r= _ma_search_next(info, ftbw->keyinfo, info->lastkey, info->lastkey_length, @@ -379,11 +379,11 @@ static int _ft2_search(FTB *ftb, FTB_WORD *ftbw, my_bool init_search) if (!r && !ftbw->off) { r= ha_compare_text(ftb->charset, - info->lastkey+1, + (uchar*) info->lastkey+1, info->lastkey_length-extra-1, - (uchar*) ftbw->word+1, + (uchar*) ftbw->word+1, ftbw->len-1, - (my_bool) (ftbw->flags & FTB_FLAG_TRUNC),0); + (my_bool) (ftbw->flags & FTB_FLAG_TRUNC), 0); } if (r) /* not found */ @@ -405,7 +405,7 @@ static int _ft2_search(FTB *ftb, FTB_WORD *ftbw, my_bool init_search) } /* going up to the first-level tree to continue search there */ - _ma_dpointer(info, (uchar*) (lastkey_buf+HA_FT_WLEN), ftbw->key_root); + _ma_dpointer(info, (lastkey_buf+HA_FT_WLEN), ftbw->key_root); ftbw->key_root=info->s->state.key_root[ftb->keynr]; ftbw->keyinfo=info->s->keyinfo+ftb->keynr; ftbw->off=0; @@ -425,15 +425,15 @@ static int _ft2_search(FTB *ftb, FTB_WORD *ftbw, my_bool init_search) TODO here: subkey-based optimization */ ftbw->off=off; - ftbw->key_root=info->lastpos; + ftbw->key_root= info->cur_row.lastpos; ftbw->keyinfo=& info->s->ft2_keyinfo; r= _ma_search_first(info, ftbw->keyinfo, ftbw->key_root); DBUG_ASSERT(r==0); /* found something */ memcpy(lastkey_buf+off, info->lastkey, info->lastkey_length); } - ftbw->docid[0]=info->lastpos; + ftbw->docid[0]= info->cur_row.lastpos; if (ftbw->flags & FTB_FLAG_YES) - *ftbw->max_docid= info->lastpos; + *ftbw->max_docid= info->cur_row.lastpos; return 0; } @@ -796,11 +796,11 @@ int maria_ft_boolean_read_next(FT_INFO *ftb, char *record) /* but it managed already to get past this line once */ continue; - info->lastpos=curdoc; + info->cur_row.lastpos= curdoc; /* Clear all states, except that the table was updated */ info->update&= (HA_STATE_CHANGED | HA_STATE_ROW_CHANGED); - if (!(*info->read_record)(info,curdoc,record)) + if (!(*info->read_record)(info, record, curdoc)) { info->update|= HA_STATE_AKTIV; /* Record is read */ if (ftb->with_scan && maria_ft_boolean_find_relevance(ftb,record,0)==0) @@ -851,9 +851,9 @@ static int ftb_find_relevance_add_word(MYSQL_FTPARSER_PARAM *param, (uchar*)ftbw->word + 1,ftbw->len - 1, (my_bool)(ftbw->flags & FTB_FLAG_TRUNC), 0)) break; - if (ftbw->docid[1] == ftb->info->lastpos) + if (ftbw->docid[1] == ftb->info->cur_row.lastpos) continue; - ftbw->docid[1]= ftb->info->lastpos; + ftbw->docid[1]= ftb->info->cur_row.lastpos; _ftb_climb_the_tree(ftb, ftbw, ftb_param->ftsi); } return(0); @@ -877,7 +877,7 @@ float maria_ft_boolean_find_relevance(FT_INFO *ftb, byte *record, uint length) { FTB_EXPR *ftbe; FT_SEG_ITERATOR ftsi, ftsi2; - my_off_t docid=ftb->info->lastpos; + MARIA_RECORD_POS docid= ftb->info->cur_row.lastpos; MY_FTB_FIND_PARAM ftb_param; MYSQL_FTPARSER_PARAM *param; struct st_mysql_ftparser *parser= ftb->keynr == NO_SUCH_KEY ? diff --git a/storage/maria/ma_ft_nlq_search.c b/storage/maria/ma_ft_nlq_search.c index 993857aecbb..4c922516455 100644 --- a/storage/maria/ma_ft_nlq_search.c +++ b/storage/maria/ma_ft_nlq_search.c @@ -69,8 +69,8 @@ static int walk_and_match(FT_WORD *word, uint32 count, ALL_IN_ONE *aio) FT_SUPERDOC sdoc, *sptr; TREE_ELEMENT *selem; double gweight=1; - MARIA_HA *info=aio->info; - uchar *keybuff=aio->keybuff; + MARIA_HA *info= aio->info; + byte *keybuff= (byte*) aio->keybuff; MARIA_KEYDEF *keyinfo=info->s->keyinfo+aio->keynr; my_off_t key_root=info->s->state.key_root[aio->keynr]; uint extra=HA_FT_WLEN+info->s->base.rec_reflength; @@ -92,7 +92,7 @@ static int walk_and_match(FT_WORD *word, uint32 count, ALL_IN_ONE *aio) for (r= _ma_search(info, keyinfo, keybuff, keylen, SEARCH_FIND, key_root) ; !r && (subkeys=ft_sintXkorr(info->lastkey+info->lastkey_length-extra)) > 0 && - info->lastpos >= info->state->data_file_length ; + info->cur_row.lastpos >= info->state->data_file_length ; r= _ma_search_next(info, keyinfo, info->lastkey, info->lastkey_length, SEARCH_BIGGER, key_root)) ; @@ -104,8 +104,9 @@ static int walk_and_match(FT_WORD *word, uint32 count, ALL_IN_ONE *aio) { if (keylen && - ha_compare_text(aio->charset,info->lastkey+1, - info->lastkey_length-extra-1, keybuff+1,keylen-1,0,0)) + ha_compare_text(aio->charset, + (uchar*) info->lastkey+1, info->lastkey_length-extra-1, + (uchar*) keybuff+1, keylen-1, 0, 0)) break; if (subkeys<0) @@ -118,7 +119,7 @@ static int walk_and_match(FT_WORD *word, uint32 count, ALL_IN_ONE *aio) */ keybuff+=keylen; keyinfo=& info->s->ft2_keyinfo; - key_root=info->lastpos; + key_root= info->cur_row.lastpos; keylen=0; r= _ma_search_first(info, keyinfo, key_root); goto do_skip; @@ -132,7 +133,7 @@ static int walk_and_match(FT_WORD *word, uint32 count, ALL_IN_ONE *aio) if (tmp_weight==0) DBUG_RETURN(doc_cnt); /* stopword, doc_cnt should be 0 */ - sdoc.doc.dpos=info->lastpos; + sdoc.doc.dpos= info->cur_row.lastpos; /* saving document matched into dtree */ if (!(selem=tree_insert(&aio->dtree, &sdoc, 0, aio->dtree.custom_arg))) @@ -162,7 +163,7 @@ static int walk_and_match(FT_WORD *word, uint32 count, ALL_IN_ONE *aio) SEARCH_BIGGER, key_root); do_skip: while ((subkeys=ft_sintXkorr(info->lastkey+info->lastkey_length-extra)) > 0 && - !r && info->lastpos >= info->state->data_file_length) + !r && info->cur_row.lastpos >= info->state->data_file_length) r= _ma_search_next(info, keyinfo, info->lastkey, info->lastkey_length, SEARCH_BIGGER, key_root); @@ -209,22 +210,22 @@ FT_INFO *maria_ft_init_nlq_search(MARIA_HA *info, uint keynr, byte *query, ALL_IN_ONE aio; FT_DOC *dptr; FT_INFO *dlist=NULL; - my_off_t saved_lastpos=info->lastpos; + MARIA_RECORD_POS saved_lastpos= info->cur_row.lastpos; struct st_mysql_ftparser *parser; MYSQL_FTPARSER_PARAM *ftparser_param; DBUG_ENTER("maria_ft_init_nlq_search"); -/* black magic ON */ + /* black magic ON */ if ((int) (keynr = _ma_check_index(info,keynr)) < 0) DBUG_RETURN(NULL); if (_ma_readinfo(info,F_RDLCK,1)) DBUG_RETURN(NULL); -/* black magic OFF */ + /* black magic OFF */ aio.info=info; aio.keynr=keynr; aio.charset=info->s->keyinfo[keynr].seg->charset; - aio.keybuff=info->lastkey+info->s->base.max_key_length; + aio.keybuff= (uchar*) info->lastkey+info->s->base.max_key_length; parser= info->s->keyinfo[keynr].parser; if (! (ftparser_param= maria_ftparser_call_initializer(info, keynr, 0))) goto err; @@ -254,7 +255,7 @@ FT_INFO *maria_ft_init_nlq_search(MARIA_HA *info, uint keynr, byte *query, while (best.elements) { my_off_t docid=((FT_DOC *)queue_remove(& best, 0))->dpos; - if (!(*info->read_record)(info,docid,record)) + if (!(*info->read_record)(info, record, docid)) { info->update|= HA_STATE_AKTIV; ftparser_param->flags= MYSQL_FTFLAGS_NEED_COPY; @@ -296,7 +297,7 @@ FT_INFO *maria_ft_init_nlq_search(MARIA_HA *info, uint keynr, byte *query, err: delete_tree(&aio.dtree); delete_tree(&wtree); - info->lastpos=saved_lastpos; + info->cur_row.lastpos= saved_lastpos; DBUG_RETURN(dlist); } @@ -313,8 +314,8 @@ int maria_ft_nlq_read_next(FT_INFO *handler, char *record) info->update&= (HA_STATE_CHANGED | HA_STATE_ROW_CHANGED); - info->lastpos=handler->doc[handler->curdoc].dpos; - if (!(*info->read_record)(info,info->lastpos,record)) + info->cur_row.lastpos= handler->doc[handler->curdoc].dpos; + if (!(*info->read_record)(info, record, info->cur_row.lastpos)) { info->update|= HA_STATE_AKTIV; /* Record is read */ return 0; @@ -329,7 +330,7 @@ float maria_ft_nlq_find_relevance(FT_INFO *handler, { int a,b,c; FT_DOC *docs=handler->doc; - my_off_t docid=handler->info->lastpos; + MARIA_RECORD_POS docid= handler->info->cur_row.lastpos; if (docid == HA_POS_ERROR) return -5.0; diff --git a/storage/maria/ma_ft_update.c b/storage/maria/ma_ft_update.c index 965f9afc91d..25f4d5a67a0 100644 --- a/storage/maria/ma_ft_update.c +++ b/storage/maria/ma_ft_update.c @@ -141,7 +141,7 @@ static int _ma_ft_store(MARIA_HA *info, uint keynr, byte *keybuf, for (; wlist->pos; wlist++) { key_length= _ma_ft_make_key(info,keynr,keybuf,wlist,filepos); - if (_ma_ck_write(info,keynr,(uchar*) keybuf,key_length)) + if (_ma_ck_write(info, keynr, keybuf, key_length)) DBUG_RETURN(1); } DBUG_RETURN(0); @@ -156,7 +156,7 @@ static int _ma_ft_erase(MARIA_HA *info, uint keynr, byte *keybuf, for (; wlist->pos; wlist++) { key_length= _ma_ft_make_key(info,keynr,keybuf,wlist,filepos); - if (_ma_ck_delete(info,keynr,(uchar*) keybuf,key_length)) + if (_ma_ck_delete(info, keynr, keybuf, key_length)) err=1; } DBUG_RETURN(err); @@ -219,13 +219,13 @@ int _ma_ft_update(MARIA_HA *info, uint keynr, byte *keybuf, if (cmp < 0 || cmp2) { key_length= _ma_ft_make_key(info,keynr,keybuf,old_word,pos); - if ((error= _ma_ck_delete(info,keynr,(uchar*) keybuf,key_length))) + if ((error= _ma_ck_delete(info,keynr, keybuf,key_length))) goto err; } if (cmp > 0 || cmp2) { - key_length= _ma_ft_make_key(info,keynr,keybuf,new_word,pos); - if ((error= _ma_ck_write(info,keynr,(uchar*) keybuf,key_length))) + key_length= _ma_ft_make_key(info, keynr, keybuf, new_word,pos); + if ((error= _ma_ck_write(info, keynr, keybuf,key_length))) goto err; } if (cmp<=0) old_word++; @@ -277,8 +277,9 @@ int _ma_ft_del(MARIA_HA *info, uint keynr, byte *keybuf, const byte *record, DBUG_RETURN(error); } + uint _ma_ft_make_key(MARIA_HA *info, uint keynr, byte *keybuf, FT_WORD *wptr, - my_off_t filepos) + my_off_t filepos) { byte buf[HA_FT_MAXBYTELEN+16]; DBUG_ENTER("_ma_ft_make_key"); @@ -294,7 +295,7 @@ uint _ma_ft_make_key(MARIA_HA *info, uint keynr, byte *keybuf, FT_WORD *wptr, int2store(buf+HA_FT_WLEN,wptr->len); memcpy(buf+HA_FT_WLEN+2,wptr->pos,wptr->len); - DBUG_RETURN(_ma_make_key(info,keynr,(uchar*) keybuf,buf,filepos)); + DBUG_RETURN(_ma_make_key(info, keynr, keybuf, buf, filepos)); } @@ -302,12 +303,12 @@ uint _ma_ft_make_key(MARIA_HA *info, uint keynr, byte *keybuf, FT_WORD *wptr, convert key value to ft2 */ -uint _ma_ft_convert_to_ft2(MARIA_HA *info, uint keynr, uchar *key) +uint _ma_ft_convert_to_ft2(MARIA_HA *info, uint keynr, byte *key) { my_off_t root; DYNAMIC_ARRAY *da=info->ft1_to_ft2; MARIA_KEYDEF *keyinfo=&info->s->ft2_keyinfo; - uchar *key_ptr= (uchar*) dynamic_array_ptr(da, 0), *end; + byte *key_ptr= (byte*) dynamic_array_ptr(da, 0), *end; uint length, key_length; DBUG_ENTER("_ma_ft_convert_to_ft2"); @@ -329,13 +330,13 @@ uint _ma_ft_convert_to_ft2(MARIA_HA *info, uint keynr, uchar *key) /* creating pageful of keys */ maria_putint(info->buff,length+2,0); memcpy(info->buff+2, key_ptr, length); - info->buff_used=info->page_changed=1; /* info->buff is used */ + info->keybuff_used=info->page_changed=1; /* info->buff is used */ if ((root= _ma_new(info,keyinfo,DFLT_INIT_HITS)) == HA_OFFSET_ERROR || _ma_write_keypage(info,keyinfo,root,DFLT_INIT_HITS,info->buff)) DBUG_RETURN(-1); /* inserting the rest of key values */ - end= (uchar*) dynamic_array_ptr(da, da->elements); + end= (byte*) dynamic_array_ptr(da, da->elements); for (key_ptr+=length; key_ptr < end; key_ptr+=keyinfo->keylength) if(_ma_ck_real_write_btree(info, keyinfo, key_ptr, 0, &root, SEARCH_SAME)) DBUG_RETURN(-1); diff --git a/storage/maria/ma_fulltext.h b/storage/maria/ma_fulltext.h index 946a5628175..cf21471b316 100644 --- a/storage/maria/ma_fulltext.h +++ b/storage/maria/ma_fulltext.h @@ -25,4 +25,4 @@ int _ma_ft_cmp(MARIA_HA *, uint, const byte *, const byte *); int _ma_ft_add(MARIA_HA *, uint, byte *, const byte *, my_off_t); int _ma_ft_del(MARIA_HA *, uint, byte *, const byte *, my_off_t); -uint _ma_ft_convert_to_ft2(MARIA_HA *, uint, uchar *); +uint _ma_ft_convert_to_ft2(MARIA_HA *, uint, byte *); diff --git a/storage/maria/ma_info.c b/storage/maria/ma_info.c index b22ffa41833..397cd2465d4 100644 --- a/storage/maria/ma_info.c +++ b/storage/maria/ma_info.c @@ -23,9 +23,9 @@ /* Get position to last record */ -my_off_t maria_position(MARIA_HA *info) +MARIA_RECORD_POS maria_position(MARIA_HA *info) { - return info->lastpos; + return info->cur_row.lastpos; } @@ -38,7 +38,7 @@ int maria_status(MARIA_HA *info, register MARIA_INFO *x, uint flag) MARIA_SHARE *share=info->s; DBUG_ENTER("maria_status"); - x->recpos = info->lastpos; + x->recpos= info->cur_row.lastpos; if (flag == HA_STATUS_POS) DBUG_RETURN(0); /* Compatible with ISAM */ if (!(flag & HA_STATUS_NO_LOCK)) @@ -64,8 +64,8 @@ int maria_status(MARIA_HA *info, register MARIA_INFO *x, uint flag) } if (flag & HA_STATUS_ERRKEY) { - x->errkey = info->errkey; - x->dupp_key_pos= info->dupp_key_pos; + x->errkey= info->errkey; + x->dup_key_pos= info->dup_key_pos; } if (flag & HA_STATUS_CONST) { @@ -121,13 +121,17 @@ int maria_status(MARIA_HA *info, register MARIA_INFO *x, uint flag) void _ma_report_error(int errcode, const char *file_name) { - size_t lgt; + uint length; DBUG_ENTER("_ma_report_error"); DBUG_PRINT("enter",("errcode %d, table '%s'", errcode, file_name)); - if ((lgt= strlen(file_name)) > 64) - file_name+= lgt - 64; + if ((length= strlen(file_name)) > 64) + { + uint dir_length= dirname_length(file_name); + file_name+= dir_length; + if ((length-= dir_length) > 64) + file_name+= length - 64; + } my_error(errcode, MYF(ME_NOREFRESH), file_name); DBUG_VOID_RETURN; } - diff --git a/storage/maria/ma_init.c b/storage/maria/ma_init.c index 318bbe341e4..b1d119732e2 100644 --- a/storage/maria/ma_init.c +++ b/storage/maria/ma_init.c @@ -18,6 +18,7 @@ #include "maria_def.h" #include +#include "ma_blockrec.h" static int maria_inited= 0; pthread_mutex_t THR_LOCK_maria; @@ -42,6 +43,7 @@ int maria_init(void) { maria_inited= 1; pthread_mutex_init(&THR_LOCK_maria,MY_MUTEX_INIT_SLOW); + _ma_init_block_record_data(); } return 0; } diff --git a/storage/maria/ma_key.c b/storage/maria/ma_key.c index ecd51f5dc92..d366c9461d6 100644 --- a/storage/maria/ma_key.c +++ b/storage/maria/ma_key.c @@ -49,11 +49,11 @@ static int _ma_put_key_in_record(MARIA_HA *info,uint keynr,byte *record); Length of key */ -uint _ma_make_key(register MARIA_HA *info, uint keynr, uchar *key, - const byte *record, my_off_t filepos) +uint _ma_make_key(register MARIA_HA *info, uint keynr, byte *key, + const byte *record, MARIA_RECORD_POS filepos) { - byte *pos,*end; - uchar *start; + const byte *pos,*end; + byte *start; reg1 HA_KEYSEG *keyseg; my_bool is_ft= info->s->keyinfo[keynr].flag & HA_FULLTEXT; DBUG_ENTER("_ma_make_key"); @@ -64,7 +64,7 @@ uint _ma_make_key(register MARIA_HA *info, uint keynr, uchar *key, TODO: nulls processing */ #ifdef HAVE_SPATIAL - DBUG_RETURN(sp_make_key(info,keynr,key,record,filepos)); + DBUG_RETURN(sp_make_key(info,keynr, key,record,filepos)); #else DBUG_ASSERT(0); /* maria_open should check that this never happens*/ #endif @@ -91,17 +91,17 @@ uint _ma_make_key(register MARIA_HA *info, uint keynr, uchar *key, char_length= ((!is_ft && cs && cs->mbmaxlen > 1) ? length/cs->mbmaxlen : length); - pos= (byte*) record+keyseg->start; + pos= record+keyseg->start; if (type == HA_KEYTYPE_BIT) { if (keyseg->bit_length) { uchar bits= get_rec_bits((uchar*) record + keyseg->bit_pos, keyseg->bit_start, keyseg->bit_length); - *key++= bits; + *key++= (char) bits; length--; } - memcpy((byte*) key, pos, length); + memcpy(key, pos, length); key+= length; continue; } @@ -121,7 +121,7 @@ uint _ma_make_key(register MARIA_HA *info, uint keynr, uchar *key, length=(uint) (end-pos); FIX_LENGTH(cs, pos, length, char_length); store_key_length_inc(key,char_length); - memcpy((byte*) key,(byte*) pos,(size_t) char_length); + memcpy(key, pos, (size_t) char_length); key+=char_length; continue; } @@ -134,18 +134,18 @@ uint _ma_make_key(register MARIA_HA *info, uint keynr, uchar *key, set_if_smaller(length,tmp_length); FIX_LENGTH(cs, pos, length, char_length); store_key_length_inc(key,char_length); - memcpy((byte*) key,(byte*) pos,(size_t) char_length); + memcpy(key,pos,(size_t) char_length); key+= char_length; continue; } else if (keyseg->flag & HA_BLOB_PART) { uint tmp_length= _ma_calc_blob_length(keyseg->bit_start,pos); - memcpy_fixed((byte*) &pos,pos+keyseg->bit_start,sizeof(char*)); + memcpy_fixed(&pos,pos+keyseg->bit_start,sizeof(char*)); set_if_smaller(length,tmp_length); FIX_LENGTH(cs, pos, length, char_length); store_key_length_inc(key,char_length); - memcpy((byte*) key,(byte*) pos,(size_t) char_length); + memcpy(key,pos,(size_t) char_length); key+= char_length; continue; } @@ -184,14 +184,14 @@ uint _ma_make_key(register MARIA_HA *info, uint keynr, uchar *key, continue; } FIX_LENGTH(cs, pos, length, char_length); - memcpy((byte*) key, pos, char_length); + memcpy(key, pos, char_length); if (length > char_length) cs->cset->fill(cs, (char*) key+char_length, length-char_length, ' '); key+= length; } _ma_dpointer(info,key,filepos); DBUG_PRINT("exit",("keynr: %d",keynr)); - DBUG_DUMP("key",(byte*) start,(uint) (key-start)+keyseg->length); + DBUG_DUMP("key",start,(uint) (key-start)+keyseg->length); DBUG_EXECUTE("key", _ma_print_key(DBUG_FILE,info->s->keyinfo[keynr].seg,start, (uint) (key-start));); @@ -217,10 +217,10 @@ uint _ma_make_key(register MARIA_HA *info, uint keynr, uchar *key, last_use_keyseg Store pointer to the keyseg after the last used one */ -uint _ma_pack_key(register MARIA_HA *info, uint keynr, uchar *key, uchar *old, - uint k_length, HA_KEYSEG **last_used_keyseg) +uint _ma_pack_key(register MARIA_HA *info, uint keynr, byte *key, + const byte *old, uint k_length, HA_KEYSEG **last_used_keyseg) { - uchar *start_key=key; + byte *start_key=key; HA_KEYSEG *keyseg; my_bool is_ft= info->s->keyinfo[keynr].flag & HA_FULLTEXT; DBUG_ENTER("_ma_pack_key"); @@ -232,7 +232,7 @@ uint _ma_pack_key(register MARIA_HA *info, uint keynr, uchar *key, uchar *old, enum ha_base_keytype type=(enum ha_base_keytype) keyseg->type; uint length=min((uint) keyseg->length,(uint) k_length); uint char_length; - uchar *pos; + const byte *pos; CHARSET_INFO *cs=keyseg->charset; if (keyseg->null_bit) @@ -249,11 +249,12 @@ uint _ma_pack_key(register MARIA_HA *info, uint keynr, uchar *key, uchar *old, continue; /* Found NULL */ } } - char_length= (!is_ft && cs && cs->mbmaxlen > 1) ? length/cs->mbmaxlen : length; - pos=old; + char_length= ((!is_ft && cs && cs->mbmaxlen > 1) ? length/cs->mbmaxlen : + length); + pos= old; if (keyseg->flag & HA_SPACE_PACK) { - uchar *end=pos+length; + const byte *end= pos + length; if (type != HA_KEYTYPE_NUM) { while (end > pos && end[-1] == ' ') @@ -268,7 +269,7 @@ uint _ma_pack_key(register MARIA_HA *info, uint keynr, uchar *key, uchar *old, length=(uint) (end-pos); FIX_LENGTH(cs, pos, length, char_length); store_key_length_inc(key,char_length); - memcpy((byte*) key,pos,(size_t) char_length); + memcpy(key,pos,(size_t) char_length); key+= char_length; continue; } @@ -282,7 +283,7 @@ uint _ma_pack_key(register MARIA_HA *info, uint keynr, uchar *key, uchar *old, FIX_LENGTH(cs, pos, length, char_length); store_key_length_inc(key,char_length); old+=2; /* Skip length */ - memcpy((byte*) key, pos,(size_t) char_length); + memcpy(key, pos,(size_t) char_length); key+= char_length; continue; } @@ -297,7 +298,7 @@ uint _ma_pack_key(register MARIA_HA *info, uint keynr, uchar *key, uchar *old, continue; } FIX_LENGTH(cs, pos, length, char_length); - memcpy((byte*) key, pos, char_length); + memcpy(key, pos, char_length); if (length > char_length) cs->cset->fill(cs, (char*) key+char_length, length-char_length, ' '); key+= length; @@ -321,7 +322,7 @@ uint _ma_pack_key(register MARIA_HA *info, uint keynr, uchar *key, uchar *old, length+= keyseg->length; keyseg++; } while (keyseg->type); - bzero((byte*) key,length); + bzero(key,length); key+=length; } #endif @@ -358,8 +359,8 @@ static int _ma_put_key_in_record(register MARIA_HA *info, uint keynr, byte *blob_ptr; DBUG_ENTER("_ma_put_key_in_record"); - blob_ptr= (byte*) info->lastkey2; /* Place to put blob parts */ - key=(byte*) info->lastkey; /* KEy that was read */ + blob_ptr= info->lastkey2; /* Place to put blob parts */ + key=info->lastkey; /* KEy that was read */ key_end=key+info->lastkey_length; for (keyseg=info->s->keyinfo[keynr].seg ; keyseg->type ;keyseg++) { @@ -378,7 +379,7 @@ static int _ma_put_key_in_record(register MARIA_HA *info, uint keynr, if (keyseg->bit_length) { - uchar bits= *key++; + byte bits= *key++; set_rec_bits(bits, record + keyseg->bit_pos, keyseg->bit_start, keyseg->bit_length); length--; @@ -388,7 +389,7 @@ static int _ma_put_key_in_record(register MARIA_HA *info, uint keynr, clr_rec_bits(record + keyseg->bit_pos, keyseg->bit_start, keyseg->bit_length); } - memcpy(record + keyseg->start, (byte*) key, length); + memcpy(record + keyseg->start, key, length); key+= length; continue; } @@ -429,7 +430,7 @@ static int _ma_put_key_in_record(register MARIA_HA *info, uint keynr, else int2store(record+keyseg->start, length); /* And key data */ - memcpy(record+keyseg->start + keyseg->bit_start, (byte*) key, length); + memcpy(record+keyseg->start + keyseg->bit_start, key, length); key+= length; } else if (keyseg->flag & HA_BLOB_PART) @@ -472,8 +473,7 @@ static int _ma_put_key_in_record(register MARIA_HA *info, uint keynr, if (key+keyseg->length > key_end) goto err; #endif - memcpy(record+keyseg->start,(byte*) key, - (size_t) keyseg->length); + memcpy(record+keyseg->start, key, (size_t) keyseg->length); key+= keyseg->length; } } @@ -486,7 +486,7 @@ err: /* Here when key reads are used */ -int _ma_read_key_record(MARIA_HA *info, my_off_t filepos, byte *buf) +int _ma_read_key_record(MARIA_HA *info, byte *buf, MARIA_RECORD_POS filepos) { fast_ma_writeinfo(info); if (filepos != HA_OFFSET_ERROR) @@ -526,7 +526,7 @@ ulonglong ma_retrieve_auto_increment(MARIA_HA *info,const byte *record) ulonglong value= 0; /* Store unsigned values here */ longlong s_value= 0; /* Store signed values here */ HA_KEYSEG *keyseg= info->s->keyinfo[info->s->base.auto_key-1].seg; - const uchar *key= (uchar*) record + keyseg->start; + const byte *key= record + keyseg->start; switch (keyseg->type) { case HA_KEYTYPE_INT8: diff --git a/storage/maria/ma_keycache.c b/storage/maria/ma_keycache.c index 837b0fbac66..a2ea4349338 100644 --- a/storage/maria/ma_keycache.c +++ b/storage/maria/ma_keycache.c @@ -54,8 +54,9 @@ int maria_assign_to_key_cache(MARIA_HA *info, int error= 0; MARIA_SHARE* share= info->s; DBUG_ENTER("maria_assign_to_key_cache"); - DBUG_PRINT("enter",("old_key_cache_handle: %lx new_key_cache_handle: %lx", - share->key_cache, key_cache)); + DBUG_PRINT("enter", + ("old_key_cache_handle: 0x%lx new_key_cache_handle: 0x%lx", + share->key_cache, key_cache)); /* Skip operation if we didn't change key cache. This can happen if we diff --git a/storage/maria/ma_locking.c b/storage/maria/ma_locking.c index adb4b03bebe..62746eba875 100644 --- a/storage/maria/ma_locking.c +++ b/storage/maria/ma_locking.c @@ -60,13 +60,25 @@ int maria_lock_database(MARIA_HA *info, int lock_type) else count= --share->w_locks; --share->tot_locks; - if (info->lock_type == F_WRLCK && !share->w_locks && - !share->delay_key_write && flush_key_blocks(share->key_cache, - share->kfile,FLUSH_KEEP)) + if (info->lock_type == F_WRLCK && !share->w_locks) { - error=my_errno; - maria_print_error(info->s, HA_ERR_CRASHED); - maria_mark_crashed(info); /* Mark that table must be checked */ + if (!share->delay_key_write && flush_key_blocks(share->key_cache, + share->kfile, + FLUSH_KEEP)) + { + error= my_errno; + maria_print_error(info->s, HA_ERR_CRASHED); + /* Mark that table must be checked */ + maria_mark_crashed(info); + } + if (share->data_file_type == BLOCK_RECORD && + flush_key_blocks(share->key_cache, info->dfile, FLUSH_KEEP)) + { + error= my_errno; + maria_print_error(info->s, HA_ERR_CRASHED); + /* Mark that table must be checked */ + maria_mark_crashed(info); + } } if (info->opt_flag & (READ_CACHE_USED | WRITE_CACHE_USED)) { @@ -84,16 +96,17 @@ int maria_lock_database(MARIA_HA *info, int lock_type) if (share->changed && !share->w_locks) { #ifdef HAVE_MMAP - if ((info->s->mmaped_length != info->s->state.state.data_file_length) && - (info->s->nonmmaped_inserts > MAX_NONMAPPED_INSERTS)) - { - if (info->s->concurrent_insert) - rw_wrlock(&info->s->mmap_lock); - _ma_remap_file(info, info->s->state.state.data_file_length); - info->s->nonmmaped_inserts= 0; - if (info->s->concurrent_insert) - rw_unlock(&info->s->mmap_lock); - } + if ((info->s->mmaped_length != + info->s->state.state.data_file_length) && + (info->s->nonmmaped_inserts > MAX_NONMAPPED_INSERTS)) + { + if (info->s->concurrent_insert) + rw_wrlock(&info->s->mmap_lock); + _ma_remap_file(info, info->s->state.state.data_file_length); + info->s->nonmmaped_inserts= 0; + if (info->s->concurrent_insert) + rw_unlock(&info->s->mmap_lock); + } #endif share->state.process= share->last_process=share->this_process; share->state.unique= info->last_unique= info->this_unique; @@ -350,7 +363,7 @@ int _ma_readinfo(register MARIA_HA *info, int lock_type, int check_keybuffer) int _ma_writeinfo(register MARIA_HA *info, uint operation) { int error,olderror; - MARIA_SHARE *share=info->s; + MARIA_SHARE *share= info->s; DBUG_ENTER("_ma_writeinfo"); DBUG_PRINT("info",("operation: %u tot_locks: %u", operation, share->tot_locks)); @@ -358,13 +371,13 @@ int _ma_writeinfo(register MARIA_HA *info, uint operation) error=0; if (share->tot_locks == 0) { - olderror=my_errno; /* Remember last error */ if (operation) { /* Two threads can't be here */ + olderror= my_errno; /* Remember last error */ share->state.process= share->last_process= share->this_process; share->state.unique= info->last_unique= info->this_unique; share->state.update_count= info->last_loop= ++info->this_loop; - if ((error=_ma_state_info_write(share->kfile, &share->state, 1))) + if ((error= _ma_state_info_write(share->kfile, &share->state, 1))) olderror=my_errno; #ifdef __WIN__ if (maria_flush) @@ -373,8 +386,8 @@ int _ma_writeinfo(register MARIA_HA *info, uint operation) _commit(info->dfile); } #endif + my_errno=olderror; } - my_errno=olderror; } else if (operation) share->changed= 1; /* Mark keyfile changed */ diff --git a/storage/maria/ma_open.c b/storage/maria/ma_open.c index 38e71a44f8b..cbc781c589e 100644 --- a/storage/maria/ma_open.c +++ b/storage/maria/ma_open.c @@ -19,6 +19,7 @@ #include "ma_fulltext.h" #include "ma_sp_defs.h" #include "ma_rt_index.h" +#include "ma_blockrec.h" #include #if defined(MSDOS) || defined(__WIN__) @@ -33,6 +34,12 @@ #endif static void setup_key_functions(MARIA_KEYDEF *keyinfo); +static my_bool maria_scan_init_dummy(MARIA_HA *info); +static void maria_scan_end_dummy(MARIA_HA *info); +static my_bool maria_once_init_dummy(MARIA_SHARE *, File); +static my_bool maria_once_end_dummy(MARIA_SHARE *); +static byte *_ma_base_info_read(byte *ptr, MARIA_BASE_INFO *base); + #define get_next_element(to,pos,size) { memcpy((char*) to,pos,(size_t) size); \ pos+=size;} @@ -76,7 +83,7 @@ MARIA_HA *_ma_test_if_reopen(char *filename) MARIA_HA *maria_open(const char *name, int mode, uint open_flags) { int kfile,open_mode,save_errno,have_rtree=0; - uint i,j,len,errpos,head_length,base_pos,offset,info_length,keys, + uint i,j,len,errpos,head_length,base_pos,info_length,keys, key_parts,unique_key_parts,fulltext_keys,uniques; char name_buff[FN_REFLEN], org_name[FN_REFLEN], index_name[FN_REFLEN], data_name[FN_REFLEN]; @@ -84,13 +91,13 @@ MARIA_HA *maria_open(const char *name, int mode, uint open_flags) MARIA_HA info,*m_info,*old_info; MARIA_SHARE share_buff,*share; ulong rec_per_key_part[HA_MAX_POSSIBLE_KEY*HA_MAX_KEY_SEG]; - my_off_t key_root[HA_MAX_POSSIBLE_KEY],key_del[MARIA_MAX_KEY_BLOCK_SIZE]; + my_off_t key_root[HA_MAX_POSSIBLE_KEY]; ulonglong max_key_file_length, max_data_file_length; DBUG_ENTER("maria_open"); LINT_INIT(m_info); kfile= -1; - errpos=0; + errpos= 0; head_length=sizeof(share_buff.state.header); bzero((byte*) &info,sizeof(info)); @@ -103,7 +110,6 @@ MARIA_HA *maria_open(const char *name, int mode, uint open_flags) bzero((gptr) &share_buff,sizeof(share_buff)); share_buff.state.rec_per_key_part=rec_per_key_part; share_buff.state.key_root=key_root; - share_buff.state.key_del=key_del; share_buff.key_cache= multi_key_cache_search(name_buff, strlen(name_buff), maria_key_cache); @@ -121,7 +127,7 @@ MARIA_HA *maria_open(const char *name, int mode, uint open_flags) goto err; } share->mode=open_mode; - errpos=1; + errpos= 1; if (my_read(kfile,(char*) share->state.header.file_version,head_length, MYF(MY_NABP))) { @@ -165,17 +171,17 @@ MARIA_HA *maria_open(const char *name, int mode, uint open_flags) MY_APPEND_EXT|MY_UNPACK_FILENAME|MY_RESOLVE_SYMLINKS); info_length=mi_uint2korr(share->state.header.header_length); - base_pos=mi_uint2korr(share->state.header.base_pos); + base_pos= mi_uint2korr(share->state.header.base_pos); if (!(disk_cache=(char*) my_alloca(info_length+128))) { my_errno=ENOMEM; goto err; } end_pos=disk_cache+info_length; - errpos=2; + errpos= 2; VOID(my_seek(kfile,0L,MY_SEEK_SET,MYF(0))); - errpos=3; + errpos= 3; if (my_read(kfile,disk_cache,info_length,MYF(MY_NABP))) { my_errno=HA_ERR_CRASHED; @@ -195,15 +201,14 @@ MARIA_HA *maria_open(const char *name, int mode, uint open_flags) } share->state_diff_length=len-MARIA_STATE_INFO_SIZE; - _ma_state_info_read((uchar*) disk_cache, &share->state); + _ma_state_info_read(disk_cache, &share->state); len= mi_uint2korr(share->state.header.base_info_length); if (len != MARIA_BASE_INFO_SIZE) { DBUG_PRINT("warning",("saved_base_info_length: %d base_info_length: %d", len,MARIA_BASE_INFO_SIZE)); } - disk_pos= (char*) - _ma_n_base_info_read((uchar*) disk_cache + base_pos, &share->base); + disk_pos= _ma_base_info_read(disk_cache + base_pos, &share->base); share->state.state_length=base_pos; if (!(open_flags & HA_OPEN_FOR_REPAIR) && @@ -239,31 +244,13 @@ MARIA_HA *maria_open(const char *name, int mode, uint open_flags) (((ulonglong) 1 << (share->base.rec_reflength*8))-1) : (_ma_safe_mul(share->base.pack_reclength, (ulonglong) 1 << (share->base.rec_reflength*8))-1); + max_key_file_length= _ma_safe_mul(MARIA_MIN_KEY_BLOCK_LENGTH, ((ulonglong) 1 << (share->base.key_reflength*8))-1); #if SIZEOF_OFF_T == 4 set_if_smaller(max_data_file_length, INT_MAX32); set_if_smaller(max_key_file_length, INT_MAX32); -#endif -#if USE_RAID && SYSTEM_SIZEOF_OFF_T == 4 - set_if_smaller(max_key_file_length, INT_MAX32); - if (!share->base.raid_type) - { - set_if_smaller(max_data_file_length, INT_MAX32); - } - else - { - set_if_smaller(max_data_file_length, - (ulonglong) share->base.raid_chunks << 31); - } -#elif !defined(USE_RAID) - if (share->base.raid_type) - { - DBUG_PRINT("error",("Table uses RAID but we don't have RAID support")); - my_errno=HA_ERR_UNSUPPORTED; - goto err; - } #endif share->base.max_data_file_length=(my_off_t) max_data_file_length; share->base.max_key_file_length=(my_off_t) max_key_file_length; @@ -286,29 +273,25 @@ MARIA_HA *maria_open(const char *name, int mode, uint open_flags) &share->index_file_name,strlen(index_name)+1, &share->data_file_name,strlen(data_name)+1, &share->state.key_root,keys*sizeof(my_off_t), - &share->state.key_del, - (share->state.header.max_block_size_index*sizeof(my_off_t)), #ifdef THREAD &share->key_root_lock,sizeof(rw_lock_t)*keys, #endif &share->mmap_lock,sizeof(rw_lock_t), NullS)) goto err; - errpos=4; + errpos= 4; + *share=share_buff; memcpy((char*) share->state.rec_per_key_part, (char*) rec_per_key_part, sizeof(long)*key_parts); memcpy((char*) share->state.key_root, (char*) key_root, sizeof(my_off_t)*keys); - memcpy((char*) share->state.key_del, - (char*) key_del, (sizeof(my_off_t) * - share->state.header.max_block_size_index)); strmov(share->unique_file_name, name_buff); share->unique_name_length= strlen(name_buff); strmov(share->index_file_name, index_name); strmov(share->data_file_name, data_name); - share->blocksize=min(IO_SIZE,maria_block_size); + share->block_size= maria_block_size; { HA_KEYSEG *pos=share->keyparts; for (i=0 ; i < keys ; i++) @@ -319,7 +302,7 @@ MARIA_HA *maria_open(const char *name, int mode, uint open_flags) end_pos); if (share->keyinfo[i].key_alg == HA_KEY_ALG_RTREE) have_rtree=1; - set_if_smaller(share->blocksize,share->keyinfo[i].block_length); + set_if_smaller(share->block_size,share->keyinfo[i].block_length); share->keyinfo[i].seg=pos; for (j=0 ; j < share->keyinfo[i].keysegs; j++,pos++) { @@ -423,28 +406,43 @@ MARIA_HA *maria_open(const char *name, int mode, uint open_flags) } share->ftparsers= 0; } - - disk_pos_assert(disk_pos + share->base.fields *MARIA_COLUMNDEF_SIZE, end_pos); - for (i=j=offset=0 ; i < share->base.fields ; i++) + share->data_file_type= share->state.header.data_file_type; + share->base_length= (BASE_ROW_HEADER_SIZE + + share->base.is_nulls_extended + + share->base.null_bytes + + share->base.pack_bytes + + test(share->options & HA_OPTION_CHECKSUM)); + if (share->base.transactional) + share->base_length+= TRANS_ROW_EXTRA_HEADER_SIZE; + share->base.default_rec_buff_size= max(share->base.pack_reclength, + share->base.max_key_length); + if (share->data_file_type == DYNAMIC_RECORD) + { + share->base.extra_rec_buff_size= + (ALIGN_SIZE(MARIA_MAX_DYN_BLOCK_HEADER) + MARIA_SPLIT_LENGTH + + MARIA_REC_BUFF_OFFSET); + share->base.default_rec_buff_size+= share->base.extra_rec_buff_size; + } + disk_pos_assert(disk_pos + share->base.fields *MARIA_COLUMNDEF_SIZE, + end_pos); + for (i= j= 0 ; i < share->base.fields ; i++) { disk_pos=_ma_recinfo_read(disk_pos,&share->rec[i]); share->rec[i].pack_type=0; share->rec[i].huff_tree=0; - share->rec[i].offset=offset; if (share->rec[i].type == (int) FIELD_BLOB) { share->blobs[j].pack_length= share->rec[i].length-maria_portable_sizeof_char_ptr;; - share->blobs[j].offset=offset; + share->blobs[j].offset= share->rec[i].offset; j++; } - offset+=share->rec[i].length; } share->rec[i].type=(int) FIELD_LAST; /* End marker */ if (_ma_open_datafile(&info, share, -1)) goto err; - errpos=5; + errpos= 5; share->kfile=kfile; share->this_process=(ulong) getpid(); @@ -456,25 +454,12 @@ MARIA_HA *maria_open(const char *name, int mode, uint open_flags) share->rec_reflength=share->base.rec_reflength; /* May be changed */ share->base.margin_key_file_length=(share->base.max_key_file_length - (keys ? MARIA_INDEX_BLOCK_MARGIN * - share->blocksize * keys : 0)); - share->blocksize=min(IO_SIZE,maria_block_size); - share->data_file_type=STATIC_RECORD; - if (share->options & HA_OPTION_COMPRESS_RECORD) - { - share->data_file_type = COMPRESSED_RECORD; - share->options|= HA_OPTION_READ_ONLY_DATA; - info.s=share; - if (_ma_read_pack_info(&info, - (pbool) - test(!(share->options & - (HA_OPTION_PACK_RECORD | - HA_OPTION_TEMP_COMPRESS_RECORD))))) - goto err; - } - else if (share->options & HA_OPTION_PACK_RECORD) - share->data_file_type = DYNAMIC_RECORD; + share->block_size * keys : 0)); + share->block_size= share->base.block_size; my_afree((gptr) disk_cache); _ma_setup_functions(share); + if ((*share->once_init)(share, info.dfile)) + goto err; #ifdef THREAD thr_lock_init(&share->lock); VOID(pthread_mutex_init(&share->intern_lock,MY_MUTEX_INIT_FAST)); @@ -493,6 +478,7 @@ MARIA_HA *maria_open(const char *name, int mode, uint open_flags) HA_OPTION_COMPRESS_RECORD | HA_OPTION_TEMP_COMPRESS_RECORD)) || (open_flags & HA_OPEN_TMP_TABLE) || + share->data_file_type == BLOCK_RECORD || have_rtree) ? 0 : 1; if (share->concurrent_insert) { @@ -512,9 +498,11 @@ MARIA_HA *maria_open(const char *name, int mode, uint open_flags) my_errno=EACCES; /* Can't open in write mode */ goto err; } - if (_ma_open_datafile(&info, share, old_info->dfile)) + if (share->data_file_type == BLOCK_RECORD) + info.dfile= share->bitmap.file; + else if (_ma_open_datafile(&info, share, old_info->dfile)) goto err; - errpos=5; + errpos= 5; have_rtree= old_info->maria_rtree_recursion_state != NULL; } @@ -530,7 +518,7 @@ MARIA_HA *maria_open(const char *name, int mode, uint open_flags) &info.maria_rtree_recursion_state,have_rtree ? 1024 : 0, NullS)) goto err; - errpos=6; + errpos= 6; if (!have_rtree) info.maria_rtree_recursion_state= NULL; @@ -540,7 +528,7 @@ MARIA_HA *maria_open(const char *name, int mode, uint open_flags) info.lastkey2=info.lastkey+share->base.max_key_length; info.s=share; - info.lastpos= HA_OFFSET_ERROR; + info.cur_row.lastpos= HA_OFFSET_ERROR; info.update= (short) (HA_STATE_NEXT_FOUND+HA_STATE_PREV_FOUND); info.opt_flag=READ_CHECK_USED; info.this_unique= (ulong) info.dfile; /* Uniq number in process */ @@ -557,8 +545,12 @@ MARIA_HA *maria_open(const char *name, int mode, uint open_flags) info.ft1_to_ft2=0; info.errkey= -1; info.page_changed=1; + info.keyread_buff= info.buff + share->base.max_key_block_length; + if ((*share->init)(&info)) + goto err; + pthread_mutex_lock(&share->intern_lock); - info.read_record=share->read_record; + info.read_record= share->read_record; share->reopen++; share->write_flag=MYF(MY_NABP | MY_WAIT_IF_FULL); if (share->options & HA_OPTION_READ_ONLY_DATA) @@ -570,7 +562,8 @@ MARIA_HA *maria_open(const char *name, int mode, uint open_flags) if ((open_flags & HA_OPEN_TMP_TABLE) || (share->options & HA_OPTION_TMP_TABLE)) { - share->temporary=share->delay_key_write=1; + share->temporary= share->delay_key_write= 1; + share->write_flag=MYF(MY_NABP); share->w_locks++; /* We don't have to update status */ share->tot_locks++; @@ -580,15 +573,17 @@ MARIA_HA *maria_open(const char *name, int mode, uint open_flags) (share->options & HA_OPTION_DELAY_KEY_WRITE)) && maria_delay_key_write) share->delay_key_write=1; + info.state= &share->state.state; /* Change global values by default */ pthread_mutex_unlock(&share->intern_lock); /* Allocate buffer for one record */ - - /* prerequisites: bzero(info) && info->s=share; are met. */ - if (!_ma_alloc_rec_buff(&info, -1, &info.rec_buff)) + /* prerequisites: info->rec_buffer == 0 && info->rec_buff_size == 0 */ + if (_ma_alloc_buffer(&info.rec_buff, &info.rec_buff_size, + share->base.default_rec_buff_size)) goto err; - bzero(info.rec_buff, _ma_get_rec_buff_len(&info, info.rec_buff)); + + bzero(info.rec_buff, share->base.default_rec_buff_size); *m_info=info; #ifdef THREAD @@ -608,12 +603,15 @@ err: _ma_report_error(save_errno, name); switch (errpos) { case 6: + (*share->end)(&info); my_free((gptr) m_info,MYF(0)); /* fall through */ case 5: - VOID(my_close(info.dfile,MYF(0))); + if (share->data_file_type != BLOCK_RECORD) + VOID(my_close(info.dfile,MYF(0))); if (old_info) break; /* Don't remove open table */ + (*share->once_end)(share); /* fall through */ case 4: my_free((gptr) share,MYF(0)); @@ -636,38 +634,23 @@ err: } /* maria_open */ -byte *_ma_alloc_rec_buff(MARIA_HA *info, ulong length, byte **buf) -{ - uint extra; - uint32 old_length; - LINT_INIT(old_length); +/* + Reallocate a buffer, if the current buffer is not large enough +*/ - if (! *buf || length > (old_length=_ma_get_rec_buff_len(info, *buf))) +my_bool _ma_alloc_buffer(byte **old_addr, my_size_t *old_size, + my_size_t new_size) +{ + if (*old_size < new_size) { - byte *newptr = *buf; - - /* to simplify initial init of info->rec_buf in maria_open and maria_extra */ - if (length == (ulong) -1) - { - length= max(info->s->base.pack_reclength, - info->s->base.max_key_length); - /* Avoid unnecessary realloc */ - if (newptr && length == old_length) - return newptr; - } - - extra= ((info->s->options & HA_OPTION_PACK_RECORD) ? - ALIGN_SIZE(MARIA_MAX_DYN_BLOCK_HEADER)+MARIA_SPLIT_LENGTH+ - MARIA_REC_BUFF_OFFSET : 0); - if (extra && newptr) - newptr-= MARIA_REC_BUFF_OFFSET; - if (!(newptr=(byte*) my_realloc((gptr)newptr, length+extra+8, - MYF(MY_ALLOW_ZERO_PTR)))) - return newptr; - *((uint32 *) newptr)= (uint32) length; - *buf= newptr+(extra ? MARIA_REC_BUFF_OFFSET : 0); + byte *addr; + if (!(addr= (byte*) my_realloc((gptr) *old_addr, new_size, + MYF(MY_ALLOW_ZERO_PTR)))) + return 1; + *old_addr= addr; + *old_size= new_size; } - return *buf; + return 0; } @@ -684,28 +667,37 @@ ulonglong _ma_safe_mul(ulonglong a, ulonglong b) void _ma_setup_functions(register MARIA_SHARE *share) { - if (share->options & HA_OPTION_COMPRESS_RECORD) - { + share->once_init= maria_once_init_dummy; + share->once_end= maria_once_end_dummy; + share->init= maria_scan_init_dummy; + share->end= maria_scan_end_dummy; + share->scan_init= maria_scan_init_dummy; + share->scan_end= maria_scan_end_dummy; + share->write_record_init= _ma_write_init_default; + share->write_record_abort= _ma_write_abort_default; + + switch (share->data_file_type) { + case COMPRESSED_RECORD: share->read_record= _ma_read_pack_record; - share->read_rnd= _ma_read_rnd_pack_record; - if (!(share->options & HA_OPTION_TEMP_COMPRESS_RECORD)) - share->calc_checksum=0; /* No checksum */ - else if (share->options & HA_OPTION_PACK_RECORD) - share->calc_checksum= _ma_checksum; - else + share->scan= _ma_read_rnd_pack_record; + share->once_init= _ma_once_init_pack_row; + share->once_end= _ma_once_end_pack_row; + /* Calculate checksum according how the original row was stored */ + if (share->state.header.org_data_file_type == STATIC_RECORD) share->calc_checksum= _ma_static_checksum; - } - else if (share->options & HA_OPTION_PACK_RECORD) - { + else + share->calc_checksum= _ma_checksum; + share->calc_write_checksum= share->calc_checksum; + break; + case DYNAMIC_RECORD: share->read_record= _ma_read_dynamic_record; - share->read_rnd= _ma_read_rnd_dynamic_record; + share->scan= _ma_read_rnd_dynamic_record; share->delete_record= _ma_delete_dynamic_record; share->compare_record= _ma_cmp_dynamic_record; share->compare_unique= _ma_cmp_dynamic_unique; - share->calc_checksum= _ma_checksum; - + share->calc_checksum= share->calc_write_checksum= _ma_checksum; /* add bits used to pack data to pack_reclength for faster allocation */ - share->base.pack_reclength+= share->base.pack_bits; + share->base.pack_reclength+= share->base.pack_bytes; if (share->base.blobs) { share->update_record= _ma_update_blob_record; @@ -716,22 +708,42 @@ void _ma_setup_functions(register MARIA_SHARE *share) share->write_record= _ma_write_dynamic_record; share->update_record= _ma_update_dynamic_record; } - } - else - { + break; + case STATIC_RECORD: share->read_record= _ma_read_static_record; - share->read_rnd= _ma_read_rnd_static_record; + share->scan= _ma_read_rnd_static_record; share->delete_record= _ma_delete_static_record; share->compare_record= _ma_cmp_static_record; share->update_record= _ma_update_static_record; share->write_record= _ma_write_static_record; share->compare_unique= _ma_cmp_static_unique; - share->calc_checksum= _ma_static_checksum; + share->calc_checksum= share->calc_write_checksum= _ma_static_checksum; + break; + case BLOCK_RECORD: + share->once_init= _ma_once_init_block_row; + share->once_end= _ma_once_end_block_row; + share->init= _ma_init_block_row; + share->end= _ma_end_block_row; + share->write_record_init= _ma_write_init_block_record; + share->write_record_abort= _ma_write_abort_block_record; + share->scan_init= _ma_scan_init_block_record; + share->scan_end= _ma_scan_end_block_record; + share->read_record= _ma_read_block_record; + share->scan= _ma_scan_block_record; + share->delete_record= _ma_delete_block_record; + share->compare_record= _ma_compare_block_record; + share->update_record= _ma_update_block_record; + share->write_record= _ma_write_block_record; + share->compare_unique= _ma_cmp_block_unique; + share->calc_checksum= _ma_checksum; + share->calc_write_checksum= 0; + break; } share->file_read= _ma_nommap_pread; share->file_write= _ma_nommap_pwrite; - if (!(share->options & HA_OPTION_CHECKSUM)) - share->calc_checksum=0; + if (!(share->options & HA_OPTION_CHECKSUM) && + share->data_file_type != COMPRESSED_RECORD) + share->calc_checksum= share->calc_write_checksum= 0; return; } @@ -798,55 +810,53 @@ uint _ma_state_info_write(File file, MARIA_STATE_INFO *state, uint pWrite) { uchar buff[MARIA_STATE_INFO_SIZE + MARIA_STATE_EXTRA_SIZE]; uchar *ptr=buff; - uint i, keys= (uint) state->header.keys, - key_blocks=state->header.max_block_size_index; + uint i, keys= (uint) state->header.keys; DBUG_ENTER("_ma_state_info_write"); memcpy_fixed(ptr,&state->header,sizeof(state->header)); ptr+=sizeof(state->header); /* open_count must be first because of _ma_mark_file_changed ! */ - mi_int2store(ptr,state->open_count); ptr +=2; - *ptr++= (uchar)state->changed; *ptr++= state->sortkey; - mi_rowstore(ptr,state->state.records); ptr +=8; - mi_rowstore(ptr,state->state.del); ptr +=8; - mi_rowstore(ptr,state->split); ptr +=8; - mi_sizestore(ptr,state->dellink); ptr +=8; - mi_sizestore(ptr,state->state.key_file_length); ptr +=8; - mi_sizestore(ptr,state->state.data_file_length); ptr +=8; - mi_sizestore(ptr,state->state.empty); ptr +=8; - mi_sizestore(ptr,state->state.key_empty); ptr +=8; - mi_int8store(ptr,state->auto_increment); ptr +=8; - mi_int8store(ptr,(ulonglong) state->state.checksum);ptr +=8; - mi_int4store(ptr,state->process); ptr +=4; - mi_int4store(ptr,state->unique); ptr +=4; - mi_int4store(ptr,state->status); ptr +=4; - mi_int4store(ptr,state->update_count); ptr +=4; - - ptr+=state->state_diff_length; + mi_int2store(ptr,state->open_count); ptr+= 2; + *ptr++= (uchar)state->changed; + *ptr++= state->sortkey; + mi_rowstore(ptr,state->state.records); ptr+= 8; + mi_rowstore(ptr,state->state.del); ptr+= 8; + mi_rowstore(ptr,state->split); ptr+= 8; + mi_sizestore(ptr,state->dellink); ptr+= 8; + mi_sizestore(ptr,state->first_bitmap_with_space); ptr+= 8; + mi_sizestore(ptr,state->state.key_file_length); ptr+= 8; + mi_sizestore(ptr,state->state.data_file_length); ptr+= 8; + mi_sizestore(ptr,state->state.empty); ptr+= 8; + mi_sizestore(ptr,state->state.key_empty); ptr+= 8; + mi_int8store(ptr,state->auto_increment); ptr+= 8; + mi_int8store(ptr,(ulonglong) state->state.checksum); ptr+= 8; + mi_int4store(ptr,state->process); ptr+= 4; + mi_int4store(ptr,state->unique); ptr+= 4; + mi_int4store(ptr,state->status); ptr+= 4; + mi_int4store(ptr,state->update_count); ptr+= 4; + + ptr+= state->state_diff_length; for (i=0; i < keys; i++) { - mi_sizestore(ptr,state->key_root[i]); ptr +=8; - } - for (i=0; i < key_blocks; i++) - { - mi_sizestore(ptr,state->key_del[i]); ptr +=8; + mi_sizestore(ptr,state->key_root[i]); ptr+= 8; } - if (pWrite & 2) /* From isamchk */ + mi_sizestore(ptr,state->key_del); ptr+= 8; + if (pWrite & 2) /* From maria_chk */ { uint key_parts= mi_uint2korr(state->header.key_parts); - mi_int4store(ptr,state->sec_index_changed); ptr +=4; - mi_int4store(ptr,state->sec_index_used); ptr +=4; - mi_int4store(ptr,state->version); ptr +=4; - mi_int8store(ptr,state->key_map); ptr +=8; - mi_int8store(ptr,(ulonglong) state->create_time); ptr +=8; - mi_int8store(ptr,(ulonglong) state->recover_time); ptr +=8; - mi_int8store(ptr,(ulonglong) state->check_time); ptr +=8; - mi_sizestore(ptr,state->rec_per_key_rows); ptr+=8; + mi_int4store(ptr,state->sec_index_changed); ptr+= 4; + mi_int4store(ptr,state->sec_index_used); ptr+= 4; + mi_int4store(ptr,state->version); ptr+= 4; + mi_int8store(ptr,state->key_map); ptr+= 8; + mi_int8store(ptr,(ulonglong) state->create_time); ptr+= 8; + mi_int8store(ptr,(ulonglong) state->recover_time); ptr+= 8; + mi_int8store(ptr,(ulonglong) state->check_time); ptr+= 8; + mi_sizestore(ptr,state->rec_per_key_rows); ptr+= 8; for (i=0 ; i < key_parts ; i++) { - mi_int4store(ptr,state->rec_per_key_part[i]); ptr+=4; + mi_int4store(ptr,state->rec_per_key_part[i]); ptr+=4; } } @@ -858,54 +868,51 @@ uint _ma_state_info_write(File file, MARIA_STATE_INFO *state, uint pWrite) } -uchar *_ma_state_info_read(uchar *ptr, MARIA_STATE_INFO *state) +byte *_ma_state_info_read(byte *ptr, MARIA_STATE_INFO *state) { - uint i,keys,key_parts,key_blocks; + uint i,keys,key_parts; memcpy_fixed(&state->header,ptr, sizeof(state->header)); - ptr +=sizeof(state->header); - keys=(uint) state->header.keys; - key_parts=mi_uint2korr(state->header.key_parts); - key_blocks=state->header.max_block_size_index; - - state->open_count = mi_uint2korr(ptr); ptr +=2; - state->changed= (bool) *ptr++; - state->sortkey = (uint) *ptr++; - state->state.records= mi_rowkorr(ptr); ptr +=8; - state->state.del = mi_rowkorr(ptr); ptr +=8; - state->split = mi_rowkorr(ptr); ptr +=8; - state->dellink= mi_sizekorr(ptr); ptr +=8; - state->state.key_file_length = mi_sizekorr(ptr); ptr +=8; - state->state.data_file_length= mi_sizekorr(ptr); ptr +=8; - state->state.empty = mi_sizekorr(ptr); ptr +=8; - state->state.key_empty= mi_sizekorr(ptr); ptr +=8; - state->auto_increment=mi_uint8korr(ptr); ptr +=8; - state->state.checksum=(ha_checksum) mi_uint8korr(ptr); ptr +=8; - state->process= mi_uint4korr(ptr); ptr +=4; - state->unique = mi_uint4korr(ptr); ptr +=4; - state->status = mi_uint4korr(ptr); ptr +=4; - state->update_count=mi_uint4korr(ptr); ptr +=4; + ptr+= sizeof(state->header); + keys= (uint) state->header.keys; + key_parts= mi_uint2korr(state->header.key_parts); + + state->open_count = mi_uint2korr(ptr); ptr+= 2; + state->changed= (my_bool) *ptr++; + state->sortkey= (uint) *ptr++; + state->state.records= mi_rowkorr(ptr); ptr+= 8; + state->state.del = mi_rowkorr(ptr); ptr+= 8; + state->split = mi_rowkorr(ptr); ptr+= 8; + state->dellink= mi_sizekorr(ptr); ptr+= 8; + state->first_bitmap_with_space= mi_sizekorr(ptr); ptr+= 8; + state->state.key_file_length = mi_sizekorr(ptr); ptr+= 8; + state->state.data_file_length= mi_sizekorr(ptr); ptr+= 8; + state->state.empty = mi_sizekorr(ptr); ptr+= 8; + state->state.key_empty= mi_sizekorr(ptr); ptr+= 8; + state->auto_increment=mi_uint8korr(ptr); ptr+= 8; + state->state.checksum=(ha_checksum) mi_uint8korr(ptr);ptr+= 8; + state->process= mi_uint4korr(ptr); ptr+= 4; + state->unique = mi_uint4korr(ptr); ptr+= 4; + state->status = mi_uint4korr(ptr); ptr+= 4; + state->update_count=mi_uint4korr(ptr); ptr+= 4; ptr+= state->state_diff_length; for (i=0; i < keys; i++) { - state->key_root[i]= mi_sizekorr(ptr); ptr +=8; - } - for (i=0; i < key_blocks; i++) - { - state->key_del[i] = mi_sizekorr(ptr); ptr +=8; + state->key_root[i]= mi_sizekorr(ptr); ptr+= 8; } - state->sec_index_changed = mi_uint4korr(ptr); ptr +=4; - state->sec_index_used = mi_uint4korr(ptr); ptr +=4; - state->version = mi_uint4korr(ptr); ptr +=4; - state->key_map = mi_uint8korr(ptr); ptr +=8; - state->create_time = (time_t) mi_sizekorr(ptr); ptr +=8; - state->recover_time =(time_t) mi_sizekorr(ptr); ptr +=8; - state->check_time = (time_t) mi_sizekorr(ptr); ptr +=8; - state->rec_per_key_rows=mi_sizekorr(ptr); ptr +=8; + state->key_del= mi_sizekorr(ptr); ptr+= 8; + state->sec_index_changed = mi_uint4korr(ptr); ptr+= 4; + state->sec_index_used = mi_uint4korr(ptr); ptr+= 4; + state->version = mi_uint4korr(ptr); ptr+= 4; + state->key_map = mi_uint8korr(ptr); ptr+= 8; + state->create_time = (time_t) mi_sizekorr(ptr); ptr+= 8; + state->recover_time =(time_t) mi_sizekorr(ptr); ptr+= 8; + state->check_time = (time_t) mi_sizekorr(ptr); ptr+= 8; + state->rec_per_key_rows=mi_sizekorr(ptr); ptr+= 8; for (i=0 ; i < key_parts ; i++) { - state->rec_per_key_part[i]= mi_uint4korr(ptr); ptr+=4; + state->rec_per_key_part[i]= mi_uint4korr(ptr); ptr+=4; } return ptr; } @@ -924,7 +931,7 @@ uint _ma_state_info_read_dsk(File file, MARIA_STATE_INFO *state, my_bool pRead) } else if (my_read(file, buff, state->state_length,MYF(MY_NABP))) return (MY_FILE_ERROR); - _ma_state_info_read((uchar*) buff, state); + _ma_state_info_read(buff, state); } return 0; } @@ -938,74 +945,84 @@ uint _ma_base_info_write(File file, MARIA_BASE_INFO *base) { uchar buff[MARIA_BASE_INFO_SIZE], *ptr=buff; - mi_sizestore(ptr,base->keystart); ptr +=8; - mi_sizestore(ptr,base->max_data_file_length); ptr +=8; - mi_sizestore(ptr,base->max_key_file_length); ptr +=8; - mi_rowstore(ptr,base->records); ptr +=8; - mi_rowstore(ptr,base->reloc); ptr +=8; - mi_int4store(ptr,base->mean_row_length); ptr +=4; - mi_int4store(ptr,base->reclength); ptr +=4; - mi_int4store(ptr,base->pack_reclength); ptr +=4; - mi_int4store(ptr,base->min_pack_length); ptr +=4; - mi_int4store(ptr,base->max_pack_length); ptr +=4; - mi_int4store(ptr,base->min_block_length); ptr +=4; - mi_int4store(ptr,base->fields); ptr +=4; - mi_int4store(ptr,base->pack_fields); ptr +=4; - *ptr++=base->rec_reflength; - *ptr++=base->key_reflength; - *ptr++=base->keys; - *ptr++=base->auto_key; - mi_int2store(ptr,base->pack_bits); ptr +=2; - mi_int2store(ptr,base->blobs); ptr +=2; - mi_int2store(ptr,base->max_key_block_length); ptr +=2; - mi_int2store(ptr,base->max_key_length); ptr +=2; - mi_int2store(ptr,base->extra_alloc_bytes); ptr +=2; + mi_sizestore(ptr,base->keystart); ptr+= 8; + mi_sizestore(ptr,base->max_data_file_length); ptr+= 8; + mi_sizestore(ptr,base->max_key_file_length); ptr+= 8; + mi_rowstore(ptr,base->records); ptr+= 8; + mi_rowstore(ptr,base->reloc); ptr+= 8; + mi_int4store(ptr,base->mean_row_length); ptr+= 4; + mi_int4store(ptr,base->reclength); ptr+= 4; + mi_int4store(ptr,base->pack_reclength); ptr+= 4; + mi_int4store(ptr,base->min_pack_length); ptr+= 4; + mi_int4store(ptr,base->max_pack_length); ptr+= 4; + mi_int4store(ptr,base->min_block_length); ptr+= 4; + mi_int2store(ptr,base->fields); ptr+= 2; + mi_int2store(ptr,base->fixed_not_null_fields); ptr+= 2; + mi_int2store(ptr,base->fixed_not_null_fields_length); ptr+= 2; + mi_int2store(ptr,base->max_field_lengths); ptr+= 2; + mi_int2store(ptr,base->pack_fields); ptr+= 2; + mi_int2store(ptr,0); ptr+= 2; + mi_int2store(ptr,base->null_bytes); ptr+= 2; + mi_int2store(ptr,base->original_null_bytes); ptr+= 2; + mi_int2store(ptr,base->field_offsets); ptr+= 2; + mi_int2store(ptr,base->min_row_length); ptr+= 2; + mi_int2store(ptr,base->block_size); ptr+= 2; + *ptr++= base->rec_reflength; + *ptr++= base->key_reflength; + *ptr++= base->keys; + *ptr++= base->auto_key; + *ptr++= base->transactional; + *ptr++= 0; /* Reserved */ + mi_int2store(ptr,base->pack_bytes); ptr+= 2; + mi_int2store(ptr,base->blobs); ptr+= 2; + mi_int2store(ptr,base->max_key_block_length); ptr+= 2; + mi_int2store(ptr,base->max_key_length); ptr+= 2; + mi_int2store(ptr,base->extra_alloc_bytes); ptr+= 2; *ptr++= base->extra_alloc_procent; - *ptr++= base->raid_type; - mi_int2store(ptr,base->raid_chunks); ptr +=2; - mi_int4store(ptr,base->raid_chunksize); ptr +=4; - bzero(ptr,6); ptr +=6; /* extra */ + bzero(ptr,16); ptr+= 16; /* extra */ + DBUG_ASSERT((ptr - buff) == MARIA_BASE_INFO_SIZE); return my_write(file,(char*) buff, (uint) (ptr-buff), MYF(MY_NABP)); } -uchar *_ma_n_base_info_read(uchar *ptr, MARIA_BASE_INFO *base) +static byte *_ma_base_info_read(byte *ptr, MARIA_BASE_INFO *base) { - base->keystart = mi_sizekorr(ptr); ptr +=8; - base->max_data_file_length = mi_sizekorr(ptr); ptr +=8; - base->max_key_file_length = mi_sizekorr(ptr); ptr +=8; - base->records = (ha_rows) mi_sizekorr(ptr); ptr +=8; - base->reloc = (ha_rows) mi_sizekorr(ptr); ptr +=8; - base->mean_row_length = mi_uint4korr(ptr); ptr +=4; - base->reclength = mi_uint4korr(ptr); ptr +=4; - base->pack_reclength = mi_uint4korr(ptr); ptr +=4; - base->min_pack_length = mi_uint4korr(ptr); ptr +=4; - base->max_pack_length = mi_uint4korr(ptr); ptr +=4; - base->min_block_length = mi_uint4korr(ptr); ptr +=4; - base->fields = mi_uint4korr(ptr); ptr +=4; - base->pack_fields = mi_uint4korr(ptr); ptr +=4; - - base->rec_reflength = *ptr++; - base->key_reflength = *ptr++; - base->keys= *ptr++; - base->auto_key= *ptr++; - base->pack_bits = mi_uint2korr(ptr); ptr +=2; - base->blobs = mi_uint2korr(ptr); ptr +=2; - base->max_key_block_length= mi_uint2korr(ptr); ptr +=2; - base->max_key_length = mi_uint2korr(ptr); ptr +=2; - base->extra_alloc_bytes = mi_uint2korr(ptr); ptr +=2; - base->extra_alloc_procent = *ptr++; - base->raid_type= *ptr++; - base->raid_chunks= mi_uint2korr(ptr); ptr +=2; - base->raid_chunksize= mi_uint4korr(ptr); ptr +=4; - /* TO BE REMOVED: Fix for old RAID files */ - if (base->raid_type == 0) - { - base->raid_chunks=0; - base->raid_chunksize=0; - } - - ptr+=6; + base->keystart= mi_sizekorr(ptr); ptr+= 8; + base->max_data_file_length= mi_sizekorr(ptr); ptr+= 8; + base->max_key_file_length= mi_sizekorr(ptr); ptr+= 8; + base->records= (ha_rows) mi_sizekorr(ptr); ptr+= 8; + base->reloc= (ha_rows) mi_sizekorr(ptr); ptr+= 8; + base->mean_row_length= mi_uint4korr(ptr); ptr+= 4; + base->reclength= mi_uint4korr(ptr); ptr+= 4; + base->pack_reclength= mi_uint4korr(ptr); ptr+= 4; + base->min_pack_length= mi_uint4korr(ptr); ptr+= 4; + base->max_pack_length= mi_uint4korr(ptr); ptr+= 4; + base->min_block_length= mi_uint4korr(ptr); ptr+= 4; + base->fields= mi_uint2korr(ptr); ptr+= 2; + base->fixed_not_null_fields= mi_uint2korr(ptr); ptr+= 2; + base->fixed_not_null_fields_length= mi_uint2korr(ptr);ptr+= 2; + base->max_field_lengths= mi_uint2korr(ptr); ptr+= 2; + base->pack_fields= mi_uint2korr(ptr); ptr+= 2; + ptr+= 2; + base->null_bytes= mi_uint2korr(ptr); ptr+= 2; + base->original_null_bytes= mi_uint2korr(ptr); ptr+= 2; + base->field_offsets= mi_uint2korr(ptr); ptr+= 2; + base->min_row_length= mi_uint2korr(ptr); ptr+= 2; + base->block_size= mi_uint2korr(ptr); ptr+= 2; + + base->rec_reflength= *ptr++; + base->key_reflength= *ptr++; + base->keys= *ptr++; + base->auto_key= *ptr++; + base->transactional= *ptr++; + ptr++; + base->pack_bytes= mi_uint2korr(ptr); ptr+= 2; + base->blobs= mi_uint2korr(ptr); ptr+= 2; + base->max_key_block_length= mi_uint2korr(ptr); ptr+= 2; + base->max_key_length= mi_uint2korr(ptr); ptr+= 2; + base->extra_alloc_bytes= mi_uint2korr(ptr); ptr+= 2; + base->extra_alloc_procent= *ptr++; + ptr+= 16; return ptr; } @@ -1018,13 +1035,13 @@ uint _ma_keydef_write(File file, MARIA_KEYDEF *keydef) uchar buff[MARIA_KEYDEF_SIZE]; uchar *ptr=buff; - *ptr++ = (uchar) keydef->keysegs; - *ptr++ = keydef->key_alg; /* Rtree or Btree */ - mi_int2store(ptr,keydef->flag); ptr +=2; - mi_int2store(ptr,keydef->block_length); ptr +=2; - mi_int2store(ptr,keydef->keylength); ptr +=2; - mi_int2store(ptr,keydef->minlength); ptr +=2; - mi_int2store(ptr,keydef->maxlength); ptr +=2; + *ptr++= (uchar) keydef->keysegs; + *ptr++= keydef->key_alg; /* Rtree or Btree */ + mi_int2store(ptr,keydef->flag); ptr+= 2; + mi_int2store(ptr,keydef->block_length); ptr+= 2; + mi_int2store(ptr,keydef->keylength); ptr+= 2; + mi_int2store(ptr,keydef->minlength); ptr+= 2; + mi_int2store(ptr,keydef->maxlength); ptr+= 2; return my_write(file,(char*) buff, (uint) (ptr-buff), MYF(MY_NABP)); } @@ -1033,12 +1050,11 @@ char *_ma_keydef_read(char *ptr, MARIA_KEYDEF *keydef) keydef->keysegs = (uint) *ptr++; keydef->key_alg = *ptr++; /* Rtree or Btree */ - keydef->flag = mi_uint2korr(ptr); ptr +=2; - keydef->block_length = mi_uint2korr(ptr); ptr +=2; - keydef->keylength = mi_uint2korr(ptr); ptr +=2; - keydef->minlength = mi_uint2korr(ptr); ptr +=2; - keydef->maxlength = mi_uint2korr(ptr); ptr +=2; - keydef->block_size_index = keydef->block_length/MARIA_MIN_KEY_BLOCK_LENGTH-1; + keydef->flag = mi_uint2korr(ptr); ptr+= 2; + keydef->block_length = mi_uint2korr(ptr); ptr+= 2; + keydef->keylength = mi_uint2korr(ptr); ptr+= 2; + keydef->minlength = mi_uint2korr(ptr); ptr+= 2; + keydef->maxlength = mi_uint2korr(ptr); ptr+= 2; keydef->underflow_block_length=keydef->block_length/3; keydef->version = 0; /* Not saved */ keydef->parser = &ft_default_parser; @@ -1062,9 +1078,9 @@ int _ma_keyseg_write(File file, const HA_KEYSEG *keyseg) *ptr++= keyseg->bit_start; *ptr++= keyseg->bit_end; *ptr++= keyseg->bit_length; - mi_int2store(ptr,keyseg->flag); ptr+=2; - mi_int2store(ptr,keyseg->length); ptr+=2; - mi_int4store(ptr,keyseg->start); ptr+=4; + mi_int2store(ptr,keyseg->flag); ptr+= 2; + mi_int2store(ptr,keyseg->length); ptr+= 2; + mi_int4store(ptr,keyseg->start); ptr+= 4; pos= keyseg->null_bit ? keyseg->null_pos : keyseg->bit_pos; mi_int4store(ptr, pos); ptr+=4; @@ -1081,10 +1097,10 @@ char *_ma_keyseg_read(char *ptr, HA_KEYSEG *keyseg) keyseg->bit_start = *ptr++; keyseg->bit_end = *ptr++; keyseg->bit_length = *ptr++; - keyseg->flag = mi_uint2korr(ptr); ptr +=2; - keyseg->length = mi_uint2korr(ptr); ptr +=2; - keyseg->start = mi_uint4korr(ptr); ptr +=4; - keyseg->null_pos = mi_uint4korr(ptr); ptr +=4; + keyseg->flag = mi_uint2korr(ptr); ptr+= 2; + keyseg->length = mi_uint2korr(ptr); ptr+= 2; + keyseg->start = mi_uint4korr(ptr); ptr+= 4; + keyseg->null_pos = mi_uint4korr(ptr); ptr+= 4; keyseg->charset=0; /* Will be filled in later */ if (keyseg->null_bit) keyseg->bit_pos= (uint16)(keyseg->null_pos + (keyseg->null_bit == 7)); @@ -1129,47 +1145,44 @@ uint _ma_recinfo_write(File file, MARIA_COLUMNDEF *recinfo) uchar buff[MARIA_COLUMNDEF_SIZE]; uchar *ptr=buff; - mi_int2store(ptr,recinfo->type); ptr +=2; - mi_int2store(ptr,recinfo->length); ptr +=2; - *ptr++ = recinfo->null_bit; - mi_int2store(ptr,recinfo->null_pos); ptr+= 2; + mi_int6store(ptr,recinfo->offset); ptr+= 6; + mi_int2store(ptr,recinfo->type); ptr+= 2; + mi_int2store(ptr,recinfo->length); ptr+= 2; + mi_int2store(ptr,recinfo->fill_length); ptr+= 2; + mi_int2store(ptr,recinfo->null_pos); ptr+= 2; + mi_int2store(ptr,recinfo->empty_pos); ptr+= 2; + (*ptr++)= recinfo->null_bit; + (*ptr++)= recinfo->empty_bit; return my_write(file,(char*) buff, (uint) (ptr-buff), MYF(MY_NABP)); } char *_ma_recinfo_read(char *ptr, MARIA_COLUMNDEF *recinfo) { - recinfo->type= mi_sint2korr(ptr); ptr +=2; - recinfo->length=mi_uint2korr(ptr); ptr +=2; - recinfo->null_bit= (uint8) *ptr++; - recinfo->null_pos=mi_uint2korr(ptr); ptr +=2; - return ptr; + recinfo->offset= mi_uint6korr(ptr); ptr+= 6; + recinfo->type= mi_sint2korr(ptr); ptr+= 2; + recinfo->length= mi_uint2korr(ptr); ptr+= 2; + recinfo->fill_length= mi_uint2korr(ptr); ptr+= 2; + recinfo->null_pos= mi_uint2korr(ptr); ptr+= 2; + recinfo->empty_pos= mi_uint2korr(ptr); ptr+= 2; + recinfo->null_bit= (uint8) *ptr++; + recinfo->empty_bit= (uint8) *ptr++; + return ptr; } /************************************************************************** -Open data file with or without RAID -We can't use dup() here as the data file descriptors need to have different -active seek-positions. + Open data file + We can't use dup() here as the data file descriptors need to have different + active seek-positions. -The argument file_to_dup is here for the future if there would on some OS -exist a dup()-like call that would give us two different file descriptors. + The argument file_to_dup is here for the future if there would on some OS + exist a dup()-like call that would give us two different file descriptors. *************************************************************************/ -int _ma_open_datafile(MARIA_HA *info, MARIA_SHARE *share, File file_to_dup __attribute__((unused))) +int _ma_open_datafile(MARIA_HA *info, MARIA_SHARE *share, + File file_to_dup __attribute__((unused))) { -#ifdef USE_RAID - if (share->base.raid_type) - { - info->dfile=my_raid_open(share->data_file_name, - share->mode | O_SHARE, - share->base.raid_type, - share->base.raid_chunks, - share->base.raid_chunksize, - MYF(MY_WME | MY_RAID)); - } - else -#endif - info->dfile=my_open(share->data_file_name, share->mode | O_SHARE, - MYF(MY_WME)); + info->dfile= my_open(share->data_file_name, share->mode | O_SHARE, + MYF(MY_WME)); return info->dfile >= 0 ? 0 : 1; } @@ -1264,3 +1277,25 @@ int maria_indexes_are_disabled(MARIA_HA *info) return (! maria_is_any_key_active(share->state.key_map) && share->base.keys); } + + +static my_bool maria_scan_init_dummy(MARIA_HA *info __attribute__((unused))) +{ + return 0; +} + +static void maria_scan_end_dummy(MARIA_HA *info __attribute__((unused))) +{ +} + +static my_bool maria_once_init_dummy(MARIA_SHARE *share + __attribute__((unused)), + File dfile __attribute__((unused))) +{ + return 0; +} + +static my_bool maria_once_end_dummy(MARIA_SHARE *share __attribute__((unused))) +{ + return 0; +} diff --git a/storage/maria/ma_packrec.c b/storage/maria/ma_packrec.c index eb99e299f9a..69d2f15ca16 100644 --- a/storage/maria/ma_packrec.c +++ b/storage/maria/ma_packrec.c @@ -44,7 +44,10 @@ #define OFFSET_TABLE_SIZE 512 -static uint read_huff_table(MARIA_BIT_BUFF *bit_buff,MARIA_DECODE_TREE *decode_tree, +static my_bool _ma_read_pack_info(MARIA_SHARE *share, File file, + pbool fix_keys); +static uint read_huff_table(MARIA_BIT_BUFF *bit_buff, + MARIA_DECODE_TREE *decode_tree, uint16 **decode_table,byte **intervall_buff, uint16 *tmp_buff); static void make_quick_table(uint16 *to_table,uint16 *decode_table, @@ -56,55 +59,61 @@ static uint copy_decode_table(uint16 *to_pos,uint offset, uint16 *decode_table); static uint find_longest_bitstream(uint16 *table, uint16 *end); static void (*get_unpack_function(MARIA_COLUMNDEF *rec))(MARIA_COLUMNDEF *field, - MARIA_BIT_BUFF *buff, - uchar *to, - uchar *end); -static void uf_zerofill_skip_zero(MARIA_COLUMNDEF *rec,MARIA_BIT_BUFF *bit_buff, - uchar *to,uchar *end); + MARIA_BIT_BUFF *buff, + byte *to, + byte *end); +static void uf_zerofill_skip_zero(MARIA_COLUMNDEF *rec, + MARIA_BIT_BUFF *bit_buff, + byte *to,byte *end); static void uf_skip_zero(MARIA_COLUMNDEF *rec,MARIA_BIT_BUFF *bit_buff, - uchar *to,uchar *end); + byte *to,byte *end); static void uf_space_normal(MARIA_COLUMNDEF *rec,MARIA_BIT_BUFF *bit_buff, - uchar *to,uchar *end); -static void uf_space_endspace_selected(MARIA_COLUMNDEF *rec,MARIA_BIT_BUFF *bit_buff, - uchar *to, uchar *end); + byte *to,byte *end); +static void uf_space_endspace_selected(MARIA_COLUMNDEF *rec, + MARIA_BIT_BUFF *bit_buff, + byte *to, byte *end); static void uf_endspace_selected(MARIA_COLUMNDEF *rec,MARIA_BIT_BUFF *bit_buff, - uchar *to,uchar *end); + byte *to,byte *end); static void uf_space_endspace(MARIA_COLUMNDEF *rec,MARIA_BIT_BUFF *bit_buff, - uchar *to,uchar *end); + byte *to,byte *end); static void uf_endspace(MARIA_COLUMNDEF *rec,MARIA_BIT_BUFF *bit_buff, - uchar *to,uchar *end); -static void uf_space_prespace_selected(MARIA_COLUMNDEF *rec,MARIA_BIT_BUFF *bit_buff, - uchar *to, uchar *end); + byte *to,byte *end); +static void uf_space_prespace_selected(MARIA_COLUMNDEF *rec, + MARIA_BIT_BUFF *bit_buff, + byte *to, byte *end); static void uf_prespace_selected(MARIA_COLUMNDEF *rec,MARIA_BIT_BUFF *bit_buff, - uchar *to,uchar *end); + byte *to,byte *end); static void uf_space_prespace(MARIA_COLUMNDEF *rec,MARIA_BIT_BUFF *bit_buff, - uchar *to,uchar *end); + byte *to,byte *end); static void uf_prespace(MARIA_COLUMNDEF *rec,MARIA_BIT_BUFF *bit_buff, - uchar *to,uchar *end); + byte *to,byte *end); static void uf_zerofill_normal(MARIA_COLUMNDEF *rec,MARIA_BIT_BUFF *bit_buff, - uchar *to,uchar *end); + byte *to,byte *end); static void uf_constant(MARIA_COLUMNDEF *rec,MARIA_BIT_BUFF *bit_buff, - uchar *to,uchar *end); + byte *to,byte *end); static void uf_intervall(MARIA_COLUMNDEF *rec,MARIA_BIT_BUFF *bit_buff, - uchar *to,uchar *end); + byte *to,byte *end); static void uf_zero(MARIA_COLUMNDEF *rec,MARIA_BIT_BUFF *bit_buff, - uchar *to,uchar *end); + byte *to,byte *end); static void uf_blob(MARIA_COLUMNDEF *rec, MARIA_BIT_BUFF *bit_buff, - uchar *to, uchar *end); + byte *to, byte *end); static void uf_varchar1(MARIA_COLUMNDEF *rec, MARIA_BIT_BUFF *bit_buff, - uchar *to, uchar *end); + byte *to, byte *end); static void uf_varchar2(MARIA_COLUMNDEF *rec, MARIA_BIT_BUFF *bit_buff, - uchar *to, uchar *end); + byte *to, byte *end); static void decode_bytes(MARIA_COLUMNDEF *rec,MARIA_BIT_BUFF *bit_buff, - uchar *to,uchar *end); -static uint decode_pos(MARIA_BIT_BUFF *bit_buff,MARIA_DECODE_TREE *decode_tree); -static void init_bit_buffer(MARIA_BIT_BUFF *bit_buff,uchar *buffer,uint length); + byte *to,byte *end); +static uint decode_pos(MARIA_BIT_BUFF *bit_buff, + MARIA_DECODE_TREE *decode_tree); +static void init_bit_buffer(MARIA_BIT_BUFF *bit_buff,uchar *buffer, + uint length); static uint fill_and_get_bits(MARIA_BIT_BUFF *bit_buff,uint count); static void fill_buffer(MARIA_BIT_BUFF *bit_buff); static uint max_bit(uint value); static uint read_pack_length(uint version, const uchar *buf, ulong *length); #ifdef HAVE_MMAP -static uchar *_ma_mempack_get_block_info(MARIA_HA *maria,MARIA_BLOCK_INFO *info, +static uchar *_ma_mempack_get_block_info(MARIA_HA *maria, + MARIA_BLOCK_INFO *info, uchar *header); #endif @@ -121,21 +130,43 @@ static maria_bit_type mask[]= 0x01ffffff, 0x03ffffff, 0x07ffffff, 0x0fffffff, 0x1fffffff, 0x3fffffff, 0x7fffffff, 0xffffffff, #endif - }; +}; - /* Read all packed info, allocate memory and fix field structs */ +my_bool _ma_once_init_pack_row(MARIA_SHARE *share, File dfile) +{ + share->options|= HA_OPTION_READ_ONLY_DATA; + if (_ma_read_pack_info(share, dfile, + (pbool) + test(!(share->options & + (HA_OPTION_PACK_RECORD | + HA_OPTION_TEMP_COMPRESS_RECORD))))) + return 1; + return 0; +} -my_bool _ma_read_pack_info(MARIA_HA *info, pbool fix_keys) +my_bool _ma_once_end_pack_row(MARIA_SHARE *share) +{ + if (share->decode_trees) + { + my_free((gptr) share->decode_trees,MYF(0)); + my_free((gptr) share->decode_tables,MYF(0)); + } + return 0; +} + + +/* Read all packed info, allocate memory and fix field structs */ + +static my_bool _ma_read_pack_info(MARIA_SHARE *share, File file, + pbool fix_keys) { - File file; int diff_length; uint i,trees,huff_tree_bits,rec_reflength,length; uint16 *decode_table,*tmp_buff; ulong elements,intervall_length; char *disk_cache,*intervall_buff; uchar header[32]; - MARIA_SHARE *share=info->s; MARIA_BIT_BUFF bit_buff; DBUG_ENTER("_ma_read_pack_info"); @@ -144,7 +175,6 @@ my_bool _ma_read_pack_info(MARIA_HA *info, pbool fix_keys) else if (maria_quick_table_bits > MAX_QUICK_TABLE_BITS) maria_quick_table_bits=MAX_QUICK_TABLE_BITS; - file=info->dfile; my_errno=0; if (my_read(file,(byte*) header,sizeof(header),MYF(MY_NABP))) { @@ -206,7 +236,7 @@ my_bool _ma_read_pack_info(MARIA_HA *info, pbool fix_keys) share->rec[i].space_length_bits=get_bits(&bit_buff,5); share->rec[i].huff_tree=share->decode_trees+(uint) get_bits(&bit_buff, huff_tree_bits); - share->rec[i].unpack=get_unpack_function(share->rec+i); + share->rec[i].unpack= get_unpack_function(share->rec+i); } skip_to_next_byte(&bit_buff); decode_table=share->decode_tables; @@ -257,7 +287,8 @@ err0: /* Read on huff-code-table from datafile */ -static uint read_huff_table(MARIA_BIT_BUFF *bit_buff, MARIA_DECODE_TREE *decode_tree, +static uint read_huff_table(MARIA_BIT_BUFF *bit_buff, + MARIA_DECODE_TREE *decode_tree, uint16 **decode_table, byte **intervall_buff, uint16 *tmp_buff) { @@ -432,7 +463,7 @@ static uint find_longest_bitstream(uint16 *table, uint16 *end) HA_ERR_WRONG_IN_RECORD or -1 on error */ -int _ma_read_pack_record(MARIA_HA *info, my_off_t filepos, byte *buf) +int _ma_read_pack_record(MARIA_HA *info, byte *buf, MARIA_RECORD_POS filepos) { MARIA_BLOCK_INFO block_info; File file; @@ -466,15 +497,20 @@ int _ma_pack_rec_unpack(register MARIA_HA *info, register byte *to, byte *from, MARIA_SHARE *share=info->s; DBUG_ENTER("_ma_pack_rec_unpack"); - init_bit_buffer(&info->bit_buff, (uchar*) from,reclength); - + if (info->s->base.null_bytes) + { + memcpy(to, from, info->s->base.null_bytes); + to+= info->s->base.null_bytes; + from+= info->s->base.null_bytes; + reclength-= info->s->base.null_bytes; + } + init_bit_buffer(&info->bit_buff, (uchar*) from, reclength); for (current_field=share->rec, end=current_field+share->base.fields ; current_field < end ; current_field++,to=end_field) { end_field=to+current_field->length; - (*current_field->unpack)(current_field,&info->bit_buff,(uchar*) to, - (uchar*) end_field); + (*current_field->unpack)(current_field,&info->bit_buff, to, end_field); } if (! info->bit_buff.error && info->bit_buff.pos - info->bit_buff.bits/8 == info->bit_buff.end) @@ -487,7 +523,7 @@ int _ma_pack_rec_unpack(register MARIA_HA *info, register byte *to, byte *from, /* Return function to unpack field */ static void (*get_unpack_function(MARIA_COLUMNDEF *rec)) -(MARIA_COLUMNDEF *, MARIA_BIT_BUFF *, uchar *, uchar *) + (MARIA_COLUMNDEF *, MARIA_BIT_BUFF *, byte *, byte *) { switch (rec->base_type) { case FIELD_SKIP_ZERO: @@ -541,8 +577,9 @@ static void (*get_unpack_function(MARIA_COLUMNDEF *rec)) /* The different functions to unpack a field */ -static void uf_zerofill_skip_zero(MARIA_COLUMNDEF *rec, MARIA_BIT_BUFF *bit_buff, - uchar *to, uchar *end) +static void uf_zerofill_skip_zero(MARIA_COLUMNDEF *rec, + MARIA_BIT_BUFF *bit_buff, + byte *to, byte *end) { if (get_bit(bit_buff)) bzero((char*) to,(uint) (end-to)); @@ -554,8 +591,8 @@ static void uf_zerofill_skip_zero(MARIA_COLUMNDEF *rec, MARIA_BIT_BUFF *bit_buff } } -static void uf_skip_zero(MARIA_COLUMNDEF *rec, MARIA_BIT_BUFF *bit_buff, uchar *to, - uchar *end) +static void uf_skip_zero(MARIA_COLUMNDEF *rec, MARIA_BIT_BUFF *bit_buff, + byte *to, byte *end) { if (get_bit(bit_buff)) bzero((char*) to,(uint) (end-to)); @@ -563,8 +600,8 @@ static void uf_skip_zero(MARIA_COLUMNDEF *rec, MARIA_BIT_BUFF *bit_buff, uchar * decode_bytes(rec,bit_buff,to,end); } -static void uf_space_normal(MARIA_COLUMNDEF *rec, MARIA_BIT_BUFF *bit_buff, uchar *to, - uchar *end) +static void uf_space_normal(MARIA_COLUMNDEF *rec, MARIA_BIT_BUFF *bit_buff, + byte *to, byte *end) { if (get_bit(bit_buff)) bfill((byte*) to,(end-to),' '); @@ -572,8 +609,9 @@ static void uf_space_normal(MARIA_COLUMNDEF *rec, MARIA_BIT_BUFF *bit_buff, ucha decode_bytes(rec,bit_buff,to,end); } -static void uf_space_endspace_selected(MARIA_COLUMNDEF *rec, MARIA_BIT_BUFF *bit_buff, - uchar *to, uchar *end) +static void uf_space_endspace_selected(MARIA_COLUMNDEF *rec, + MARIA_BIT_BUFF *bit_buff, + byte *to, byte *end) { uint spaces; if (get_bit(bit_buff)) @@ -596,8 +634,9 @@ static void uf_space_endspace_selected(MARIA_COLUMNDEF *rec, MARIA_BIT_BUFF *bit } } -static void uf_endspace_selected(MARIA_COLUMNDEF *rec, MARIA_BIT_BUFF *bit_buff, - uchar *to, uchar *end) +static void uf_endspace_selected(MARIA_COLUMNDEF *rec, + MARIA_BIT_BUFF *bit_buff, + byte *to, byte *end) { uint spaces; if (get_bit(bit_buff)) @@ -615,8 +654,8 @@ static void uf_endspace_selected(MARIA_COLUMNDEF *rec, MARIA_BIT_BUFF *bit_buff, decode_bytes(rec,bit_buff,to,end); } -static void uf_space_endspace(MARIA_COLUMNDEF *rec, MARIA_BIT_BUFF *bit_buff, uchar *to, - uchar *end) +static void uf_space_endspace(MARIA_COLUMNDEF *rec, MARIA_BIT_BUFF *bit_buff, + byte *to, byte *end) { uint spaces; if (get_bit(bit_buff)) @@ -634,8 +673,8 @@ static void uf_space_endspace(MARIA_COLUMNDEF *rec, MARIA_BIT_BUFF *bit_buff, uc } } -static void uf_endspace(MARIA_COLUMNDEF *rec, MARIA_BIT_BUFF *bit_buff, uchar *to, - uchar *end) +static void uf_endspace(MARIA_COLUMNDEF *rec, MARIA_BIT_BUFF *bit_buff, + byte *to, byte *end) { uint spaces; if ((spaces=get_bits(bit_buff,rec->space_length_bits))+to > end) @@ -648,8 +687,9 @@ static void uf_endspace(MARIA_COLUMNDEF *rec, MARIA_BIT_BUFF *bit_buff, uchar *t bfill((byte*) end-spaces,spaces,' '); } -static void uf_space_prespace_selected(MARIA_COLUMNDEF *rec, MARIA_BIT_BUFF *bit_buff, - uchar *to, uchar *end) +static void uf_space_prespace_selected(MARIA_COLUMNDEF *rec, + MARIA_BIT_BUFF *bit_buff, + byte *to, byte *end) { uint spaces; if (get_bit(bit_buff)) @@ -673,8 +713,9 @@ static void uf_space_prespace_selected(MARIA_COLUMNDEF *rec, MARIA_BIT_BUFF *bit } -static void uf_prespace_selected(MARIA_COLUMNDEF *rec, MARIA_BIT_BUFF *bit_buff, - uchar *to, uchar *end) +static void uf_prespace_selected(MARIA_COLUMNDEF *rec, + MARIA_BIT_BUFF *bit_buff, + byte *to, byte *end) { uint spaces; if (get_bit(bit_buff)) @@ -693,8 +734,8 @@ static void uf_prespace_selected(MARIA_COLUMNDEF *rec, MARIA_BIT_BUFF *bit_buff, } -static void uf_space_prespace(MARIA_COLUMNDEF *rec, MARIA_BIT_BUFF *bit_buff, uchar *to, - uchar *end) +static void uf_space_prespace(MARIA_COLUMNDEF *rec, MARIA_BIT_BUFF *bit_buff, + byte *to, byte *end) { uint spaces; if (get_bit(bit_buff)) @@ -712,8 +753,8 @@ static void uf_space_prespace(MARIA_COLUMNDEF *rec, MARIA_BIT_BUFF *bit_buff, uc } } -static void uf_prespace(MARIA_COLUMNDEF *rec, MARIA_BIT_BUFF *bit_buff, uchar *to, - uchar *end) +static void uf_prespace(MARIA_COLUMNDEF *rec, MARIA_BIT_BUFF *bit_buff, + byte *to, byte *end) { uint spaces; if ((spaces=get_bits(bit_buff,rec->space_length_bits))+to > end) @@ -726,24 +767,24 @@ static void uf_prespace(MARIA_COLUMNDEF *rec, MARIA_BIT_BUFF *bit_buff, uchar *t decode_bytes(rec,bit_buff,to+spaces,end); } -static void uf_zerofill_normal(MARIA_COLUMNDEF *rec, MARIA_BIT_BUFF *bit_buff, uchar *to, - uchar *end) +static void uf_zerofill_normal(MARIA_COLUMNDEF *rec, MARIA_BIT_BUFF *bit_buff, + byte *to, byte *end) { end-=rec->space_length_bits; - decode_bytes(rec,bit_buff,(uchar*) to,end); + decode_bytes(rec,bit_buff, to, end); bzero((char*) end,rec->space_length_bits); } static void uf_constant(MARIA_COLUMNDEF *rec, MARIA_BIT_BUFF *bit_buff __attribute__((unused)), - uchar *to, - uchar *end) + byte *to, byte *end) { memcpy(to,rec->huff_tree->intervalls,(size_t) (end-to)); } -static void uf_intervall(MARIA_COLUMNDEF *rec, MARIA_BIT_BUFF *bit_buff, uchar *to, - uchar *end) +static void uf_intervall(MARIA_COLUMNDEF *rec, MARIA_BIT_BUFF *bit_buff, + byte *to, + byte *end) { reg1 uint field_length=(uint) (end-to); memcpy(to,rec->huff_tree->intervalls+field_length*decode_pos(bit_buff, @@ -755,16 +796,16 @@ static void uf_intervall(MARIA_COLUMNDEF *rec, MARIA_BIT_BUFF *bit_buff, uchar * /*ARGSUSED*/ static void uf_zero(MARIA_COLUMNDEF *rec __attribute__((unused)), MARIA_BIT_BUFF *bit_buff __attribute__((unused)), - uchar *to, uchar *end) + byte *to, byte *end) { - bzero((char*) to,(uint) (end-to)); + bzero(to, (uint) (end-to)); } static void uf_blob(MARIA_COLUMNDEF *rec, MARIA_BIT_BUFF *bit_buff, - uchar *to, uchar *end) + byte *to, byte *end) { if (get_bit(bit_buff)) - bzero((byte*) to,(end-to)); + bzero(to, (uint) (end-to)); else { ulong length=get_bits(bit_buff,rec->space_length_bits); @@ -775,7 +816,8 @@ static void uf_blob(MARIA_COLUMNDEF *rec, MARIA_BIT_BUFF *bit_buff, bzero((byte*) to,(end-to)); return; } - decode_bytes(rec,bit_buff,bit_buff->blob_pos,bit_buff->blob_pos+length); + decode_bytes(rec,bit_buff,(byte*) bit_buff->blob_pos, + (byte*) bit_buff->blob_pos+length); _ma_store_blob_length((byte*) to,pack_length,length); memcpy_fixed((char*) to+pack_length,(char*) &bit_buff->blob_pos, sizeof(char*)); @@ -785,21 +827,21 @@ static void uf_blob(MARIA_COLUMNDEF *rec, MARIA_BIT_BUFF *bit_buff, static void uf_varchar1(MARIA_COLUMNDEF *rec, MARIA_BIT_BUFF *bit_buff, - uchar *to, uchar *end __attribute__((unused))) + byte *to, byte *end __attribute__((unused))) { if (get_bit(bit_buff)) to[0]= 0; /* Zero lengths */ else { ulong length=get_bits(bit_buff,rec->space_length_bits); - *to= (uchar) length; + *to= (char) length; decode_bytes(rec,bit_buff,to+1,to+1+length); } } static void uf_varchar2(MARIA_COLUMNDEF *rec, MARIA_BIT_BUFF *bit_buff, - uchar *to, uchar *end __attribute__((unused))) + byte *to, byte *end __attribute__((unused))) { if (get_bit(bit_buff)) to[0]=to[1]=0; /* Zero lengths */ @@ -815,8 +857,8 @@ static void uf_varchar2(MARIA_COLUMNDEF *rec, MARIA_BIT_BUFF *bit_buff, #if BITS_SAVED == 64 -static void decode_bytes(MARIA_COLUMNDEF *rec,MARIA_BIT_BUFF *bit_buff,uchar *to, - uchar *end) +static void decode_bytes(MARIA_COLUMNDEF *rec,MARIA_BIT_BUFF *bit_buff, + byte *to, byte *end) { reg1 uint bits,low_byte; reg3 uint16 *pos; @@ -850,7 +892,7 @@ static void decode_bytes(MARIA_COLUMNDEF *rec,MARIA_BIT_BUFF *bit_buff,uchar *to low_byte=decode_tree->table[low_byte]; if (low_byte & IS_CHAR) { - *to++ = (low_byte & 255); /* Found char in quick table */ + *to++ = (char) (low_byte & 255); /* Found char in quick table */ bits-= ((low_byte >> 8) & 31); /* Remove bits used */ } else @@ -870,7 +912,7 @@ static void decode_bytes(MARIA_COLUMNDEF *rec,MARIA_BIT_BUFF *bit_buff,uchar *to decode_bytes_test_bit(7); bits-=8; } - *to++ = *pos; + *to++ = (char) *pos; } } while (to != end); @@ -880,8 +922,8 @@ static void decode_bytes(MARIA_COLUMNDEF *rec,MARIA_BIT_BUFF *bit_buff,uchar *to #else -static void decode_bytes(MARIA_COLUMNDEF *rec, MARIA_BIT_BUFF *bit_buff, uchar *to, - uchar *end) +static void decode_bytes(MARIA_COLUMNDEF *rec, MARIA_BIT_BUFF *bit_buff, + byte *to, byte *end) { reg1 uint bits,low_byte; reg3 uint16 *pos; @@ -967,7 +1009,7 @@ static void decode_bytes(MARIA_COLUMNDEF *rec, MARIA_BIT_BUFF *bit_buff, uchar * decode_bytes_test_bit(7); bits-=8; } - *to++ = (uchar) *pos; + *to++ = (char) *pos; } } while (to != end); @@ -977,7 +1019,8 @@ static void decode_bytes(MARIA_COLUMNDEF *rec, MARIA_BIT_BUFF *bit_buff, uchar * #endif /* BIT_SAVED == 64 */ -static uint decode_pos(MARIA_BIT_BUFF *bit_buff, MARIA_DECODE_TREE *decode_tree) +static uint decode_pos(MARIA_BIT_BUFF *bit_buff, + MARIA_DECODE_TREE *decode_tree) { uint16 *pos=decode_tree->table; for (;;) @@ -991,8 +1034,9 @@ static uint decode_pos(MARIA_BIT_BUFF *bit_buff, MARIA_DECODE_TREE *decode_tree) } -int _ma_read_rnd_pack_record(MARIA_HA *info, byte *buf, - register my_off_t filepos, +int _ma_read_rnd_pack_record(MARIA_HA *info, + byte *buf, + register MARIA_RECORD_POS filepos, my_bool skip_deleted_blocks) { uint b_type; @@ -1039,9 +1083,9 @@ int _ma_read_rnd_pack_record(MARIA_HA *info, byte *buf, MYF(MY_NABP))) goto err; } - info->packed_length=block_info.rec_len; - info->lastpos=filepos; - info->nextpos=block_info.filepos+block_info.rec_len; + info->packed_length= block_info.rec_len; + info->cur_row.lastpos= filepos; + info->cur_row.nextpos= block_info.filepos+block_info.rec_len; info->update|= HA_STATE_AKTIV | HA_STATE_KEY_CHANGED; DBUG_RETURN (_ma_pack_rec_unpack(info,buf,info->rec_buff, @@ -1053,10 +1097,10 @@ int _ma_read_rnd_pack_record(MARIA_HA *info, byte *buf, /* Read and process header from a huff-record-file */ -uint _ma_pack_get_block_info(MARIA_HA *maria, MARIA_BLOCK_INFO *info, File file, - my_off_t filepos) +uint _ma_pack_get_block_info(MARIA_HA *maria, MARIA_BLOCK_INFO *info, + File file, my_off_t filepos) { - uchar *header=info->header; + uchar *header= info->header; uint head_length,ref_length; LINT_INIT(ref_length); @@ -1078,8 +1122,9 @@ uint _ma_pack_get_block_info(MARIA_HA *maria, MARIA_BLOCK_INFO *info, File file, { head_length+= read_pack_length((uint) maria->s->pack.version, header + head_length, &info->blob_len); - if (!(_ma_alloc_rec_buff(maria,info->rec_len + info->blob_len, - &maria->rec_buff))) + if (_ma_alloc_buffer(&maria->rec_buff, &maria->rec_buff_size, + info->rec_len + info->blob_len + + maria->s->base.extra_rec_buff_size)) return BLOCK_FATAL_ERROR; /* not enough memory */ maria->bit_buff.blob_pos=(uchar*) maria->rec_buff+info->rec_len; maria->bit_buff.blob_end= maria->bit_buff.blob_pos+info->blob_len; @@ -1098,7 +1143,8 @@ uint _ma_pack_get_block_info(MARIA_HA *maria, MARIA_BLOCK_INFO *info, File file, /* rutines for bit buffer */ /* Note buffer must be 6 byte bigger than longest row */ -static void init_bit_buffer(MARIA_BIT_BUFF *bit_buff, uchar *buffer, uint length) +static void init_bit_buffer(MARIA_BIT_BUFF *bit_buff, uchar *buffer, + uint length) { bit_buff->pos=buffer; bit_buff->end=buffer+length; @@ -1174,8 +1220,10 @@ static uint max_bit(register uint value) #ifdef HAVE_MMAP -static int _ma_read_mempack_record(MARIA_HA *info,my_off_t filepos,byte *buf); -static int _ma_read_rnd_mempack_record(MARIA_HA*, byte *,my_off_t, my_bool); +static int _ma_read_mempack_record(MARIA_HA *info, byte *buf, + MARIA_RECORD_POS filepos); +static int _ma_read_rnd_mempack_record(MARIA_HA*, byte *, MARIA_RECORD_POS, + my_bool); my_bool _ma_memmap_file(MARIA_HA *info) { @@ -1195,7 +1243,7 @@ my_bool _ma_memmap_file(MARIA_HA *info) } info->opt_flag|= MEMMAP_USED; info->read_record= share->read_record= _ma_read_mempack_record; - share->read_rnd= _ma_read_rnd_mempack_record; + share->scan= _ma_read_rnd_mempack_record; DBUG_RETURN(1); } @@ -1207,7 +1255,8 @@ void _ma_unmap_file(MARIA_HA *info) } -static uchar *_ma_mempack_get_block_info(MARIA_HA *maria,MARIA_BLOCK_INFO *info, +static uchar *_ma_mempack_get_block_info(MARIA_HA *maria, + MARIA_BLOCK_INFO *info, uchar *header) { header+= read_pack_length((uint) maria->s->pack.version, header, @@ -1217,8 +1266,8 @@ static uchar *_ma_mempack_get_block_info(MARIA_HA *maria,MARIA_BLOCK_INFO *info, header+= read_pack_length((uint) maria->s->pack.version, header, &info->blob_len); /* _ma_alloc_rec_buff sets my_errno on error */ - if (!(_ma_alloc_rec_buff(maria, info->blob_len, - &maria->rec_buff))) + if (_ma_alloc_buffer(&maria->rec_buff, &maria->rec_buff_size, + info->blob_len + maria->s->base.extra_rec_buff_size)) return 0; /* not enough memory */ maria->bit_buff.blob_pos=(uchar*) maria->rec_buff; maria->bit_buff.blob_end= (uchar*) maria->rec_buff + info->blob_len; @@ -1227,7 +1276,8 @@ static uchar *_ma_mempack_get_block_info(MARIA_HA *maria,MARIA_BLOCK_INFO *info, } -static int _ma_read_mempack_record(MARIA_HA *info, my_off_t filepos, byte *buf) +static int _ma_read_mempack_record(MARIA_HA *info, byte *buf, + MARIA_RECORD_POS filepos) { MARIA_BLOCK_INFO block_info; MARIA_SHARE *share=info->s; @@ -1246,8 +1296,9 @@ static int _ma_read_mempack_record(MARIA_HA *info, my_off_t filepos, byte *buf) /*ARGSUSED*/ -static int _ma_read_rnd_mempack_record(MARIA_HA *info, byte *buf, - register my_off_t filepos, +static int _ma_read_rnd_mempack_record(MARIA_HA *info, + byte *buf, + register MARIA_RECORD_POS filepos, my_bool skip_deleted_blocks __attribute__((unused))) { @@ -1274,8 +1325,8 @@ static int _ma_read_rnd_mempack_record(MARIA_HA *info, byte *buf, } #endif info->packed_length=block_info.rec_len; - info->lastpos=filepos; - info->nextpos=filepos+(uint) (pos-start)+block_info.rec_len; + info->cur_row.lastpos= filepos; + info->cur_row.nextpos= filepos+(uint) (pos-start)+block_info.rec_len; info->update|= HA_STATE_AKTIV | HA_STATE_KEY_CHANGED; DBUG_RETURN (_ma_pack_rec_unpack(info,buf,pos, block_info.rec_len)); diff --git a/storage/maria/ma_page.c b/storage/maria/ma_page.c index 054b8e16468..e7ca329c3e2 100644 --- a/storage/maria/ma_page.c +++ b/storage/maria/ma_page.c @@ -20,22 +20,20 @@ /* Fetch a key-page in memory */ -uchar *_ma_fetch_keypage(register MARIA_HA *info, MARIA_KEYDEF *keyinfo, - my_off_t page, int level, - uchar *buff, int return_buffer) +byte *_ma_fetch_keypage(register MARIA_HA *info, MARIA_KEYDEF *keyinfo, + my_off_t page, int level, + byte *buff, int return_buffer) { - uchar *tmp; + byte *tmp; uint page_size; DBUG_ENTER("_ma_fetch_keypage"); DBUG_PRINT("enter",("page: %ld",page)); - tmp=(uchar*) key_cache_read(info->s->key_cache, - info->s->kfile, page, level, (byte*) buff, - (uint) keyinfo->block_length, - (uint) keyinfo->block_length, - return_buffer); + tmp= key_cache_read(info->s->key_cache, info->s->kfile, page, level, buff, + info->s->block_size, info->s->block_size, + return_buffer); if (tmp == info->buff) - info->buff_used=1; + info->keybuff_used=1; else if (!tmp) { DBUG_PRINT("error",("Got errno: %d from key_cache_read",my_errno)); @@ -53,8 +51,8 @@ uchar *_ma_fetch_keypage(register MARIA_HA *info, MARIA_KEYDEF *keyinfo, DBUG_DUMP("page", (char*) tmp, keyinfo->block_length); info->last_keypage = HA_OFFSET_ERROR; maria_print_error(info->s, HA_ERR_CRASHED); - my_errno = HA_ERR_CRASHED; - tmp = 0; + my_errno= HA_ERR_CRASHED; + tmp= 0; } DBUG_RETURN(tmp); } /* _ma_fetch_keypage */ @@ -63,7 +61,7 @@ uchar *_ma_fetch_keypage(register MARIA_HA *info, MARIA_KEYDEF *keyinfo, /* Write a key-page on disk */ int _ma_write_keypage(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, - my_off_t page, int level, uchar *buff) + my_off_t page, int level, byte *buff) { reg3 uint length; DBUG_ENTER("_ma_write_keypage"); @@ -112,8 +110,8 @@ int _ma_dispose(register MARIA_HA *info, MARIA_KEYDEF *keyinfo, my_off_t pos, DBUG_ENTER("_ma_dispose"); DBUG_PRINT("enter",("pos: %ld", (long) pos)); - old_link= info->s->state.key_del[keyinfo->block_size_index]; - info->s->state.key_del[keyinfo->block_size_index]= pos; + old_link= info->s->state.key_del; + info->s->state.key_del= pos; mi_sizestore(buff,old_link); info->s->state.changed|= STATE_NOT_SORTED_PAGES; DBUG_RETURN(key_cache_write(info->s->key_cache, @@ -129,11 +127,10 @@ int _ma_dispose(register MARIA_HA *info, MARIA_KEYDEF *keyinfo, my_off_t pos, my_off_t _ma_new(register MARIA_HA *info, MARIA_KEYDEF *keyinfo, int level) { my_off_t pos; - char buff[8]; + byte buff[8]; DBUG_ENTER("_ma_new"); - if ((pos= info->s->state.key_del[keyinfo->block_size_index]) == - HA_OFFSET_ERROR) + if ((pos= info->s->state.key_del) == HA_OFFSET_ERROR) { if (info->state->key_file_length >= info->s->base.max_key_file_length - keyinfo->block_length) @@ -153,7 +150,7 @@ my_off_t _ma_new(register MARIA_HA *info, MARIA_KEYDEF *keyinfo, int level) (uint) keyinfo->block_length,0)) pos= HA_OFFSET_ERROR; else - info->s->state.key_del[keyinfo->block_size_index]= mi_sizekorr(buff); + info->s->state.key_del= mi_sizekorr(buff); } info->s->state.changed|= STATE_NOT_SORTED_PAGES; DBUG_PRINT("exit",("Pos: %ld",(long) pos)); diff --git a/storage/maria/ma_range.c b/storage/maria/ma_range.c index 0f6883f4c9d..243e8e1b9c3 100644 --- a/storage/maria/ma_range.c +++ b/storage/maria/ma_range.c @@ -24,10 +24,10 @@ static ha_rows _ma_record_pos(MARIA_HA *info,const byte *key,uint key_len, enum ha_rkey_function search_flag); -static double _ma_search_pos(MARIA_HA *info,MARIA_KEYDEF *keyinfo,uchar *key, - uint key_len,uint nextflag,my_off_t pos); -static uint _ma_keynr(MARIA_HA *info,MARIA_KEYDEF *keyinfo,uchar *page, - uchar *keypos,uint *ret_max_key); +static double _ma_search_pos(MARIA_HA *info,MARIA_KEYDEF *keyinfo, byte *key, + uint key_len,uint nextflag, my_off_t pos); +static uint _ma_keynr(MARIA_HA *info, MARIA_KEYDEF *keyinfo, byte *page, + byte *keypos, uint *ret_max_key); /* @@ -68,12 +68,12 @@ ha_rows maria_records_in_range(MARIA_HA *info, int inx, key_range *min_key, #ifdef HAVE_RTREE_KEYS case HA_KEY_ALG_RTREE: { - uchar * key_buff; + byte *key_buff; uint start_key_len; key_buff= info->lastkey+info->s->base.max_key_length; start_key_len= _ma_pack_key(info,inx, key_buff, - (uchar*) min_key->key, min_key->length, + min_key->key, min_key->length, (HA_KEYSEG**) 0); res= maria_rtree_estimate(info, inx, key_buff, start_key_len, maria_read_vec[min_key->flag]); @@ -113,24 +113,23 @@ static ha_rows _ma_record_pos(MARIA_HA *info, const byte *key, uint key_len, { uint inx=(uint) info->lastinx, nextflag; MARIA_KEYDEF *keyinfo=info->s->keyinfo+inx; - uchar *key_buff; + byte *key_buff; double pos; - DBUG_ENTER("_ma_record_pos"); DBUG_PRINT("enter",("search_flag: %d",search_flag)); if (key_len == 0) - key_len=USE_WHOLE_KEY; + key_len= USE_WHOLE_KEY; key_buff=info->lastkey+info->s->base.max_key_length; - key_len= _ma_pack_key(info,inx,key_buff,(uchar*) key,key_len, + key_len= _ma_pack_key(info, inx, key_buff, key, key_len, (HA_KEYSEG**) 0); - DBUG_EXECUTE("key", _ma_print_key(DBUG_FILE,keyinfo->seg, - (uchar*) key_buff,key_len);); + DBUG_EXECUTE("key", _ma_print_key(DBUG_FILE, keyinfo->seg, + key_buff, key_len);); nextflag=maria_read_vec[search_flag]; if (!(nextflag & (SEARCH_FIND | SEARCH_NO_FIND | SEARCH_LAST))) key_len=USE_WHOLE_KEY; - pos= _ma_search_pos(info,keyinfo,key_buff,key_len, + pos= _ma_search_pos(info,keyinfo, key_buff, key_len, nextflag | SEARCH_SAVE_BUFF, info->s->state.key_root[inx]); if (pos >= 0.0) @@ -147,13 +146,13 @@ static ha_rows _ma_record_pos(MARIA_HA *info, const byte *key, uint key_len, static double _ma_search_pos(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, - uchar *key, uint key_len, uint nextflag, + byte *key, uint key_len, uint nextflag, register my_off_t pos) { int flag; uint nod_flag,keynr,max_keynr; my_bool after_key; - uchar *keypos,*buff; + byte *keypos, *buff; double offset; DBUG_ENTER("_ma_search_pos"); @@ -162,7 +161,7 @@ static double _ma_search_pos(register MARIA_HA *info, if (!(buff= _ma_fetch_keypage(info,keyinfo,pos,DFLT_INIT_HITS,info->buff,1))) goto err; - flag=(*keyinfo->bin_search)(info,keyinfo,buff,key,key_len,nextflag, + flag=(*keyinfo->bin_search)(info, keyinfo, buff, key, key_len, nextflag, &keypos,info->lastkey, &after_key); nod_flag=_ma_test_if_nod(buff); keynr= _ma_keynr(info,keyinfo,buff,keypos,&max_keynr); @@ -213,11 +212,11 @@ err: /* Get keynummer of current key and max number of keys in nod */ -static uint _ma_keynr(MARIA_HA *info, register MARIA_KEYDEF *keyinfo, uchar *page, - uchar *keypos, uint *ret_max_key) +static uint _ma_keynr(MARIA_HA *info, register MARIA_KEYDEF *keyinfo, + byte *page, byte *keypos, uint *ret_max_key) { uint nod_flag,keynr,max_key; - uchar t_buff[HA_MAX_KEY_BUFF],*end; + byte t_buff[HA_MAX_KEY_BUFF],*end; end= page+maria_getint(page); nod_flag=_ma_test_if_nod(page); diff --git a/storage/maria/ma_rfirst.c b/storage/maria/ma_rfirst.c index 503e8989936..6fa8af75c40 100644 --- a/storage/maria/ma_rfirst.c +++ b/storage/maria/ma_rfirst.c @@ -21,7 +21,7 @@ int maria_rfirst(MARIA_HA *info, byte *buf, int inx) { DBUG_ENTER("maria_rfirst"); - info->lastpos= HA_OFFSET_ERROR; + info->cur_row.lastpos= HA_OFFSET_ERROR; info->update|= HA_STATE_PREV_FOUND; DBUG_RETURN(maria_rnext(info,buf,inx)); } /* maria_rfirst */ diff --git a/storage/maria/ma_rkey.c b/storage/maria/ma_rkey.c index 2cb54a73b15..bd92ebd4e4c 100644 --- a/storage/maria/ma_rkey.c +++ b/storage/maria/ma_rkey.c @@ -22,16 +22,16 @@ /* Read a record using key */ /* Ordinary search_flag is 0 ; Give error if no record with key */ -int maria_rkey(MARIA_HA *info, byte *buf, int inx, const byte *key, uint key_len, - enum ha_rkey_function search_flag) +int maria_rkey(MARIA_HA *info, byte *buf, int inx, const byte *key, + uint key_len, enum ha_rkey_function search_flag) { - uchar *key_buff; + byte *key_buff; MARIA_SHARE *share=info->s; MARIA_KEYDEF *keyinfo; HA_KEYSEG *last_used_keyseg; uint pack_key_length, use_key_length, nextflag; DBUG_ENTER("maria_rkey"); - DBUG_PRINT("enter", ("base: %lx buf: %lx inx: %d search_flag: %d", + DBUG_PRINT("enter", ("base: 0x%lx buf: 0x%lx inx: %d search_flag: %d", (long) info, (long) buf, inx, search_flag)); if ((inx = _ma_check_index(info,inx)) < 0) @@ -47,7 +47,7 @@ int maria_rkey(MARIA_HA *info, byte *buf, int inx, const byte *key, uint key_len /* key is already packed!; This happens when we are using a MERGE TABLE */ - key_buff=info->lastkey+info->s->base.max_key_length; + key_buff= info->lastkey+info->s->base.max_key_length; pack_key_length= key_len; bmove(key_buff,key,key_len); last_used_keyseg= 0; @@ -58,7 +58,7 @@ int maria_rkey(MARIA_HA *info, byte *buf, int inx, const byte *key, uint key_len key_len=USE_WHOLE_KEY; /* Save the packed key for later use in the second buffer of lastkey. */ key_buff=info->lastkey+info->s->base.max_key_length; - pack_key_length= _ma_pack_key(info,(uint) inx, key_buff, (uchar*) key, + pack_key_length= _ma_pack_key(info,(uint) inx, key_buff, key, key_len, &last_used_keyseg); /* Save packed_key_length for use by the MERGE engine. */ info->pack_key_length= pack_key_length; @@ -82,15 +82,17 @@ int maria_rkey(MARIA_HA *info, byte *buf, int inx, const byte *key, uint key_len if (maria_rtree_find_first(info,inx,key_buff,use_key_length,nextflag) < 0) { maria_print_error(info->s, HA_ERR_CRASHED); - my_errno=HA_ERR_CRASHED; - goto err; + my_errno= HA_ERR_CRASHED; + info->cur_row.lastpos= HA_OFFSET_ERROR; } break; #endif case HA_KEY_ALG_BTREE: default: if (!_ma_search(info, keyinfo, key_buff, use_key_length, - maria_read_vec[search_flag], info->s->state.key_root[inx])) + maria_read_vec[search_flag], + info->s->state.key_root[inx]) && + share->concurrent_insert) { /* If we are searching for an exact key (including the data pointer) @@ -98,50 +100,65 @@ int maria_rkey(MARIA_HA *info, byte *buf, int inx, const byte *key, uint key_len then the result is "key not found". */ if ((search_flag == HA_READ_KEY_EXACT) && - (info->lastpos >= info->state->data_file_length)) + (info->cur_row.lastpos >= info->state->data_file_length)) { my_errno= HA_ERR_KEY_NOT_FOUND; - info->lastpos= HA_OFFSET_ERROR; + info->cur_row.lastpos= HA_OFFSET_ERROR; } - else while (info->lastpos >= info->state->data_file_length) + else { - /* - Skip rows that are inserted by other threads since we got a lock - Note that this can only happen if we are not searching after an - exact key, because the keys are sorted according to position - */ - - if (_ma_search_next(info, keyinfo, info->lastkey, - info->lastkey_length, - maria_readnext_vec[search_flag], - info->s->state.key_root[inx])) - break; + while (info->cur_row.lastpos >= info->state->data_file_length) + { + /* + Skip rows that are inserted by other threads since we got a lock + Note that this can only happen if we are not searching after an + exact key, because the keys are sorted according to position + */ + + if (_ma_search_next(info, keyinfo, info->lastkey, + info->lastkey_length, + maria_readnext_vec[search_flag], + info->s->state.key_root[inx])) + { + info->cur_row.lastpos= HA_OFFSET_ERROR; + break; + } + } } } } if (share->concurrent_insert) rw_unlock(&share->key_root_lock[inx]); + if (info->cur_row.lastpos == HA_OFFSET_ERROR) + { + fast_ma_writeinfo(info); + goto err; + } + /* Calculate length of the found key; Used by maria_rnext_same */ - if ((keyinfo->flag & HA_VAR_LENGTH_KEY) && last_used_keyseg && - info->lastpos != HA_OFFSET_ERROR) + if ((keyinfo->flag & HA_VAR_LENGTH_KEY) && last_used_keyseg) info->last_rkey_length= _ma_keylength_part(keyinfo, info->lastkey, last_used_keyseg); else info->last_rkey_length= pack_key_length; + /* Check if we don't want to have record back, only error message */ if (!buf) - DBUG_RETURN(info->lastpos == HA_OFFSET_ERROR ? my_errno : 0); - - if (!(*info->read_record)(info,info->lastpos,buf)) + { + fast_ma_writeinfo(info); + DBUG_RETURN(0); + } + if (!(*info->read_record)(info, buf, info->cur_row.lastpos)) { info->update|= HA_STATE_AKTIV; /* Record is read */ DBUG_RETURN(0); } - info->lastpos = HA_OFFSET_ERROR; /* Didn't find key */ + info->cur_row.lastpos= HA_OFFSET_ERROR; /* Didn't find row */ +err: /* Store last used key as a base for read next */ memcpy(info->lastkey,key_buff,pack_key_length); info->last_rkey_length= pack_key_length; @@ -150,6 +167,5 @@ int maria_rkey(MARIA_HA *info, byte *buf, int inx, const byte *key, uint key_len if (search_flag == HA_READ_AFTER_KEY) info->update|=HA_STATE_NEXT_FOUND; /* Previous gives last row */ -err: DBUG_RETURN(my_errno); } /* _ma_rkey */ diff --git a/storage/maria/ma_rlast.c b/storage/maria/ma_rlast.c index 8ce26afa78d..504cc89aed3 100644 --- a/storage/maria/ma_rlast.c +++ b/storage/maria/ma_rlast.c @@ -21,7 +21,7 @@ int maria_rlast(MARIA_HA *info, byte *buf, int inx) { DBUG_ENTER("maria_rlast"); - info->lastpos= HA_OFFSET_ERROR; + info->cur_row.lastpos= HA_OFFSET_ERROR; info->update|= HA_STATE_NEXT_FOUND; DBUG_RETURN(maria_rprev(info,buf,inx)); } /* maria_rlast */ diff --git a/storage/maria/ma_rnext.c b/storage/maria/ma_rnext.c index 8f342c6a8d2..c7feded933e 100644 --- a/storage/maria/ma_rnext.c +++ b/storage/maria/ma_rnext.c @@ -34,7 +34,8 @@ int maria_rnext(MARIA_HA *info, byte *buf, int inx) if ((inx = _ma_check_index(info,inx)) < 0) DBUG_RETURN(my_errno); flag=SEARCH_BIGGER; /* Read next */ - if (info->lastpos == HA_OFFSET_ERROR && info->update & HA_STATE_PREV_FOUND) + if (info->cur_row.lastpos == HA_OFFSET_ERROR && + info->update & HA_STATE_PREV_FOUND) flag=0; /* Read first */ if (fast_ma_readinfo(info)) @@ -86,7 +87,7 @@ int maria_rnext(MARIA_HA *info, byte *buf, int inx) { if (!error) { - while (info->lastpos >= info->state->data_file_length) + while (info->cur_row.lastpos >= info->state->data_file_length) { /* Skip rows inserted by other threads since we got a lock */ if ((error= _ma_search_next(info,info->s->keyinfo+inx, @@ -110,9 +111,9 @@ int maria_rnext(MARIA_HA *info, byte *buf, int inx) } else if (!buf) { - DBUG_RETURN(info->lastpos==HA_OFFSET_ERROR ? my_errno : 0); + DBUG_RETURN(info->cur_row.lastpos == HA_OFFSET_ERROR ? my_errno : 0); } - else if (!(*info->read_record)(info,info->lastpos,buf)) + else if (!(*info->read_record)(info, buf, info->cur_row.lastpos)) { info->update|= HA_STATE_AKTIV; /* Record is read */ DBUG_RETURN(0); diff --git a/storage/maria/ma_rnext_same.c b/storage/maria/ma_rnext_same.c index b53639073e3..a5ce0cfe15c 100644 --- a/storage/maria/ma_rnext_same.c +++ b/storage/maria/ma_rnext_same.c @@ -17,13 +17,14 @@ #include "maria_def.h" #include "ma_rt_index.h" - /* - Read next row with the same key as previous read, but abort if - the key changes. - One may have done a write, update or delete of the previous row. - NOTE! Even if one changes the previous row, the next read is done - based on the position of the last used key! - */ +/* + Read next row with the same key as previous read, but abort if + the key changes. + One may have done a write, update or delete of the previous row. + + NOTE! Even if one changes the previous row, the next read is done + based on the position of the last used key! +*/ int maria_rnext_same(MARIA_HA *info, byte *buf) { @@ -32,9 +33,10 @@ int maria_rnext_same(MARIA_HA *info, byte *buf) MARIA_KEYDEF *keyinfo; DBUG_ENTER("maria_rnext_same"); - if ((int) (inx=info->lastinx) < 0 || info->lastpos == HA_OFFSET_ERROR) + if ((int) (inx= info->lastinx) < 0 || + info->cur_row.lastpos == HA_OFFSET_ERROR) DBUG_RETURN(my_errno=HA_ERR_WRONG_INDEX); - keyinfo=info->s->keyinfo+inx; + keyinfo= info->s->keyinfo+inx; if (fast_ma_readinfo(info)) DBUG_RETURN(my_errno); @@ -50,7 +52,7 @@ int maria_rnext_same(MARIA_HA *info, byte *buf) { error=1; my_errno=HA_ERR_END_OF_FILE; - info->lastpos= HA_OFFSET_ERROR; + info->cur_row.lastpos= HA_OFFSET_ERROR; break; } break; @@ -68,16 +70,17 @@ int maria_rnext_same(MARIA_HA *info, byte *buf) info->lastkey_length,SEARCH_BIGGER, info->s->state.key_root[inx]))) break; - if (ha_key_cmp(keyinfo->seg, info->lastkey, info->lastkey2, + if (ha_key_cmp(keyinfo->seg, (uchar*) info->lastkey, + (uchar*) info->lastkey2, info->last_rkey_length, SEARCH_FIND, not_used)) { error=1; my_errno=HA_ERR_END_OF_FILE; - info->lastpos= HA_OFFSET_ERROR; + info->cur_row.lastpos= HA_OFFSET_ERROR; break; } /* Skip rows that are inserted by other threads since we got a lock */ - if (info->lastpos < info->state->data_file_length) + if (info->cur_row.lastpos < info->state->data_file_length) break; } } @@ -94,9 +97,9 @@ int maria_rnext_same(MARIA_HA *info, byte *buf) } else if (!buf) { - DBUG_RETURN(info->lastpos==HA_OFFSET_ERROR ? my_errno : 0); + DBUG_RETURN(info->cur_row.lastpos == HA_OFFSET_ERROR ? my_errno : 0); } - else if (!(*info->read_record)(info,info->lastpos,buf)) + else if (!(*info->read_record)(info, buf, info->cur_row.lastpos)) { info->update|= HA_STATE_AKTIV; /* Record is read */ DBUG_RETURN(0); diff --git a/storage/maria/ma_rprev.c b/storage/maria/ma_rprev.c index 8dd4498cf8b..ea562359ded 100644 --- a/storage/maria/ma_rprev.c +++ b/storage/maria/ma_rprev.c @@ -33,7 +33,8 @@ int maria_rprev(MARIA_HA *info, byte *buf, int inx) if ((inx = _ma_check_index(info,inx)) < 0) DBUG_RETURN(my_errno); flag=SEARCH_SMALLER; /* Read previous */ - if (info->lastpos == HA_OFFSET_ERROR && info->update & HA_STATE_NEXT_FOUND) + if (info->cur_row.lastpos == HA_OFFSET_ERROR && + info->update & HA_STATE_NEXT_FOUND) flag=0; /* Read last */ if (fast_ma_readinfo(info)) @@ -56,7 +57,7 @@ int maria_rprev(MARIA_HA *info, byte *buf, int inx) { if (!error) { - while (info->lastpos >= info->state->data_file_length) + while (info->cur_row.lastpos >= info->state->data_file_length) { /* Skip rows that are inserted by other threads since we got a lock */ if ((error= _ma_search_next(info,share->keyinfo+inx,info->lastkey, @@ -77,9 +78,9 @@ int maria_rprev(MARIA_HA *info, byte *buf, int inx) } else if (!buf) { - DBUG_RETURN(info->lastpos==HA_OFFSET_ERROR ? my_errno : 0); + DBUG_RETURN(info->cur_row.lastpos == HA_OFFSET_ERROR ? my_errno : 0); } - else if (!(*info->read_record)(info,info->lastpos,buf)) + else if (!(*info->read_record)(info, buf, info->cur_row.lastpos)) { info->update|= HA_STATE_AKTIV; /* Record is read */ DBUG_RETURN(0); diff --git a/storage/maria/ma_rrnd.c b/storage/maria/ma_rrnd.c index 2f01c0e92c5..33940d5f23f 100644 --- a/storage/maria/ma_rrnd.c +++ b/storage/maria/ma_rrnd.c @@ -21,40 +21,34 @@ #include "maria_def.h" /* - Read a row based on position. - If filepos= HA_OFFSET_ERROR then read next row - Return values - Returns one of following values: - 0 = Ok. - HA_ERR_RECORD_DELETED = Record is deleted. - HA_ERR_END_OF_FILE = EOF. + Read a row based on position. + + RETURN + 0 Ok. + HA_ERR_RECORD_DELETED Record is deleted. + HA_ERR_END_OF_FILE EOF. */ -int maria_rrnd(MARIA_HA *info, byte *buf, register my_off_t filepos) +int maria_rrnd(MARIA_HA *info, byte *buf, MARIA_RECORD_POS filepos) { - my_bool skip_deleted_blocks; DBUG_ENTER("maria_rrnd"); - skip_deleted_blocks=0; - + DBUG_ASSERT(filepos != HA_OFFSET_ERROR); +#ifdef NOT_USED if (filepos == HA_OFFSET_ERROR) { skip_deleted_blocks=1; - if (info->lastpos == HA_OFFSET_ERROR) /* First read ? */ - filepos= info->s->pack.header_length; /* Read first record */ + if (info->cur_row.lastpos == HA_OFFSET_ERROR) /* First read ? */ + filepos= info->s->pack.header_length; /* Read first record */ else - filepos= info->nextpos; + filepos= info->cur_row.nextpos; } +#endif - if (info->once_flags & RRND_PRESERVE_LASTINX) - info->once_flags&= ~RRND_PRESERVE_LASTINX; - else - info->lastinx= -1; /* Can't forward or backward */ /* Init all but update-flag */ info->update&= (HA_STATE_CHANGED | HA_STATE_ROW_CHANGED); - if (info->opt_flag & WRITE_CACHE_USED && flush_io_cache(&info->rec_cache)) DBUG_RETURN(my_errno); - DBUG_RETURN ((*info->s->read_rnd)(info,buf,filepos,skip_deleted_blocks)); + DBUG_RETURN((*info->s->read_record)(info, buf, filepos)); } diff --git a/storage/maria/ma_rsame.c b/storage/maria/ma_rsame.c index 913ae3b4370..7556c1e7332 100644 --- a/storage/maria/ma_rsame.c +++ b/storage/maria/ma_rsame.c @@ -16,14 +16,17 @@ #include "maria_def.h" - /* - ** Find current row with read on position or read on key - ** If inx >= 0 find record using key - ** Return values: - ** 0 = Ok. - ** HA_ERR_KEY_NOT_FOUND = Row is deleted - ** HA_ERR_END_OF_FILE = End of file - */ +/* + Find current row with read on position or read on key + + NOTES + If inx >= 0 find record using key + + RETURN + 0 Ok + HA_ERR_KEY_NOT_FOUND Row is deleted + HA_ERR_END_OF_FILE End of file +*/ int maria_rsame(MARIA_HA *info, byte *record, int inx) @@ -34,7 +37,8 @@ int maria_rsame(MARIA_HA *info, byte *record, int inx) { DBUG_RETURN(my_errno=HA_ERR_WRONG_INDEX); } - if (info->lastpos == HA_OFFSET_ERROR || info->update & HA_STATE_DELETED) + if (info->cur_row.lastpos == HA_OFFSET_ERROR || + info->update & HA_STATE_DELETED) { DBUG_RETURN(my_errno=HA_ERR_KEY_NOT_FOUND); /* No current record */ } @@ -48,7 +52,7 @@ int maria_rsame(MARIA_HA *info, byte *record, int inx) { info->lastinx=inx; info->lastkey_length= _ma_make_key(info,(uint) inx,info->lastkey,record, - info->lastpos); + info->cur_row.lastpos); if (info->s->concurrent_insert) rw_rdlock(&info->s->key_root_lock[inx]); VOID(_ma_search(info,info->s->keyinfo+inx,info->lastkey, USE_WHOLE_KEY, @@ -58,7 +62,7 @@ int maria_rsame(MARIA_HA *info, byte *record, int inx) rw_unlock(&info->s->key_root_lock[inx]); } - if (!(*info->read_record)(info,info->lastpos,record)) + if (!(*info->read_record)(info, record, info->cur_row.lastpos)) DBUG_RETURN(0); if (my_errno == HA_ERR_RECORD_DELETED) my_errno=HA_ERR_KEY_NOT_FOUND; diff --git a/storage/maria/ma_rsamepos.c b/storage/maria/ma_rsamepos.c index 09861c03c32..859f0da9b51 100644 --- a/storage/maria/ma_rsamepos.c +++ b/storage/maria/ma_rsamepos.c @@ -28,29 +28,31 @@ ** HA_ERR_END_OF_FILE = End of file */ -int maria_rsame_with_pos(MARIA_HA *info, byte *record, int inx, my_off_t filepos) +int maria_rsame_with_pos(MARIA_HA *info, byte *record, int inx, + MARIA_RECORD_POS filepos) { DBUG_ENTER("maria_rsame_with_pos"); DBUG_PRINT("enter",("index: %d filepos: %ld", inx, (long) filepos)); - if (inx < -1 || (inx >= 0 && !maria_is_key_active(info->s->state.key_map, inx))) + if (inx < -1 || + (inx >= 0 && !maria_is_key_active(info->s->state.key_map, inx))) { DBUG_RETURN(my_errno=HA_ERR_WRONG_INDEX); } info->update&= (HA_STATE_CHANGED | HA_STATE_ROW_CHANGED); - if ((*info->s->read_rnd)(info,record,filepos,0)) + if ((*info->s->read_record)(info, record, filepos)) { if (my_errno == HA_ERR_RECORD_DELETED) my_errno=HA_ERR_KEY_NOT_FOUND; DBUG_RETURN(my_errno); } - info->lastpos=filepos; - info->lastinx=inx; + info->cur_row.lastpos= filepos; + info->lastinx= inx; if (inx >= 0) { info->lastkey_length= _ma_make_key(info,(uint) inx,info->lastkey,record, - info->lastpos); + info->cur_row.lastpos); info->update|=HA_STATE_KEY_CHANGED; /* Don't use indexposition */ } DBUG_RETURN(0); diff --git a/storage/maria/ma_rt_index.c b/storage/maria/ma_rt_index.c index 83ced5b8167..8e8ec6c991b 100644 --- a/storage/maria/ma_rt_index.c +++ b/storage/maria/ma_rt_index.c @@ -53,18 +53,17 @@ typedef struct st_page_list 1 Not found */ -static int maria_rtree_find_req(MARIA_HA *info, MARIA_KEYDEF *keyinfo, uint search_flag, - uint nod_cmp_flag, my_off_t page, int level) +static int maria_rtree_find_req(MARIA_HA *info, MARIA_KEYDEF *keyinfo, + uint search_flag, + uint nod_cmp_flag, my_off_t page, int level) { - uchar *k; - uchar *last; uint nod_flag; int res; - uchar *page_buf; + byte *page_buf, *k, *last; int k_len; uint *saved_key = (uint*) (info->maria_rtree_recursion_state) + level; - if (!(page_buf = (uchar*)my_alloca((uint)keyinfo->block_length))) + if (!(page_buf = (byte*) my_alloca((uint)keyinfo->block_length))) { my_errno = HA_ERR_OUT_OF_MEM; return -1; @@ -77,24 +76,27 @@ static int maria_rtree_find_req(MARIA_HA *info, MARIA_KEYDEF *keyinfo, uint sear if(info->maria_rtree_recursion_depth >= level) { - k = page_buf + *saved_key; + k= page_buf + *saved_key; } else { k = rt_PAGE_FIRST_KEY(page_buf, nod_flag); } - last = rt_PAGE_END(page_buf); + last= rt_PAGE_END(page_buf); for (; k < last; k = rt_PAGE_NEXT_KEY(k, k_len, nod_flag)) { if (nod_flag) { /* this is an internal node in the tree */ - if (!(res = maria_rtree_key_cmp(keyinfo->seg, info->first_mbr_key, k, - info->last_rkey_length, nod_cmp_flag))) + if (!(res = maria_rtree_key_cmp(keyinfo->seg, + info->first_mbr_key, k, + info->last_rkey_length, nod_cmp_flag))) { - switch ((res = maria_rtree_find_req(info, keyinfo, search_flag, nod_cmp_flag, - _ma_kpos(nod_flag, k), level + 1))) + switch ((res = maria_rtree_find_req(info, keyinfo, search_flag, + nod_cmp_flag, + _ma_kpos(nod_flag, k), + level + 1))) { case 0: /* found - exit from recursion */ *saved_key = k - page_buf; @@ -111,11 +113,11 @@ static int maria_rtree_find_req(MARIA_HA *info, MARIA_KEYDEF *keyinfo, uint sear else { /* this is a leaf */ - if (!maria_rtree_key_cmp(keyinfo->seg, info->first_mbr_key, k, - info->last_rkey_length, search_flag)) + if (!maria_rtree_key_cmp(keyinfo->seg, info->first_mbr_key, + k, info->last_rkey_length, search_flag)) { - uchar *after_key = rt_PAGE_NEXT_KEY(k, k_len, nod_flag); - info->lastpos = _ma_dpos(info, 0, after_key); + byte *after_key = (byte*) rt_PAGE_NEXT_KEY(k, k_len, nod_flag); + info->cur_row.lastpos = _ma_dpos(info, 0, after_key); info->lastkey_length = k_len + info->s->base.rec_reflength; memcpy(info->lastkey, k, info->lastkey_length); info->maria_rtree_recursion_depth = level; @@ -126,11 +128,11 @@ static int maria_rtree_find_req(MARIA_HA *info, MARIA_KEYDEF *keyinfo, uint sear info->int_keypos = info->buff; info->int_maxpos = info->buff + (last - after_key); memcpy(info->buff, after_key, last - after_key); - info->buff_used = 0; + info->keybuff_used = 0; } else { - info->buff_used = 1; + info->keybuff_used = 1; } res = 0; @@ -138,7 +140,7 @@ static int maria_rtree_find_req(MARIA_HA *info, MARIA_KEYDEF *keyinfo, uint sear } } } - info->lastpos = HA_OFFSET_ERROR; + info->cur_row.lastpos = HA_OFFSET_ERROR; my_errno = HA_ERR_KEY_NOT_FOUND; res = 1; @@ -148,7 +150,7 @@ ok: err1: my_afree((byte*)page_buf); - info->lastpos = HA_OFFSET_ERROR; + info->cur_row.lastpos = HA_OFFSET_ERROR; return -1; } @@ -170,8 +172,8 @@ err1: 1 Not found */ -int maria_rtree_find_first(MARIA_HA *info, uint keynr, uchar *key, uint key_length, - uint search_flag) +int maria_rtree_find_first(MARIA_HA *info, uint keynr, byte *key, + uint key_length, uint search_flag) { my_off_t root; uint nod_cmp_flag; @@ -191,11 +193,12 @@ int maria_rtree_find_first(MARIA_HA *info, uint keynr, uchar *key, uint key_leng info->last_rkey_length = key_length; info->maria_rtree_recursion_depth = -1; - info->buff_used = 1; + info->keybuff_used = 1; - nod_cmp_flag = ((search_flag & (MBR_EQUAL | MBR_WITHIN)) ? - MBR_WITHIN : MBR_INTERSECT); - return maria_rtree_find_req(info, keyinfo, search_flag, nod_cmp_flag, root, 0); + nod_cmp_flag= ((search_flag & (MBR_EQUAL | MBR_WITHIN)) ? + MBR_WITHIN : MBR_INTERSECT); + return maria_rtree_find_req(info, keyinfo, search_flag, nod_cmp_flag, root, + 0); } @@ -221,27 +224,29 @@ int maria_rtree_find_next(MARIA_HA *info, uint keynr, uint search_flag) MARIA_KEYDEF *keyinfo = info->s->keyinfo + keynr; if (info->update & HA_STATE_DELETED) - return maria_rtree_find_first(info, keynr, info->lastkey, info->lastkey_length, - search_flag); + return maria_rtree_find_first(info, keynr, info->lastkey, + info->lastkey_length, + search_flag); - if (!info->buff_used) + if (!info->keybuff_used) { - uchar *key= info->int_keypos; + byte *key= info->int_keypos; while (key < info->int_maxpos) { - if (!maria_rtree_key_cmp(keyinfo->seg, info->first_mbr_key, key, - info->last_rkey_length, search_flag)) + if (!maria_rtree_key_cmp(keyinfo->seg, + info->first_mbr_key, key, + info->last_rkey_length, search_flag)) { - uchar *after_key = key + keyinfo->keylength; + byte *after_key= key + keyinfo->keylength; - info->lastpos= _ma_dpos(info, 0, after_key); + info->cur_row.lastpos= _ma_dpos(info, 0, after_key); memcpy(info->lastkey, key, info->lastkey_length); if (after_key < info->int_maxpos) info->int_keypos= after_key; else - info->buff_used= 1; + info->keybuff_used= 1; return 0; } key+= keyinfo->keylength; @@ -274,15 +279,12 @@ int maria_rtree_find_next(MARIA_HA *info, uint keynr, uint search_flag) static int maria_rtree_get_req(MARIA_HA *info, MARIA_KEYDEF *keyinfo, uint key_length, my_off_t page, int level) { - uchar *k; - uchar *last; - uint nod_flag; + byte *page_buf, *last, *k; + uint nod_flag, k_len; int res; - uchar *page_buf; - uint k_len; - uint *saved_key = (uint*) (info->maria_rtree_recursion_state) + level; + uint *saved_key= (uint*) (info->maria_rtree_recursion_state) + level; - if (!(page_buf = (uchar*)my_alloca((uint)keyinfo->block_length))) + if (!(page_buf= (byte*) my_alloca((uint)keyinfo->block_length))) return -1; if (!_ma_fetch_keypage(info, keyinfo, page, DFLT_INIT_HITS, page_buf, 0)) goto err1; @@ -312,7 +314,7 @@ static int maria_rtree_get_req(MARIA_HA *info, MARIA_KEYDEF *keyinfo, uint key_l { /* this is an internal node in the tree */ switch ((res = maria_rtree_get_req(info, keyinfo, key_length, - _ma_kpos(nod_flag, k), level + 1))) + _ma_kpos(nod_flag, k), level + 1))) { case 0: /* found - exit from recursion */ *saved_key = k - page_buf; @@ -328,8 +330,8 @@ static int maria_rtree_get_req(MARIA_HA *info, MARIA_KEYDEF *keyinfo, uint key_l else { /* this is a leaf */ - uchar *after_key = rt_PAGE_NEXT_KEY(k, k_len, nod_flag); - info->lastpos = _ma_dpos(info, 0, after_key); + byte *after_key = rt_PAGE_NEXT_KEY(k, k_len, nod_flag); + info->cur_row.lastpos = _ma_dpos(info, 0, after_key); info->lastkey_length = k_len + info->s->base.rec_reflength; memcpy(info->lastkey, k, info->lastkey_length); @@ -338,21 +340,21 @@ static int maria_rtree_get_req(MARIA_HA *info, MARIA_KEYDEF *keyinfo, uint key_l if (after_key < last) { - info->int_keypos = (uchar*)saved_key; + info->int_keypos = (byte*) saved_key; memcpy(info->buff, page_buf, keyinfo->block_length); info->int_maxpos = rt_PAGE_END(info->buff); - info->buff_used = 0; + info->keybuff_used = 0; } else { - info->buff_used = 1; + info->keybuff_used = 1; } res = 0; goto ok; } } - info->lastpos = HA_OFFSET_ERROR; + info->cur_row.lastpos = HA_OFFSET_ERROR; my_errno = HA_ERR_KEY_NOT_FOUND; res = 1; @@ -362,7 +364,7 @@ ok: err1: my_afree((byte*)page_buf); - info->lastpos = HA_OFFSET_ERROR; + info->cur_row.lastpos = HA_OFFSET_ERROR; return -1; } @@ -388,7 +390,7 @@ int maria_rtree_get_first(MARIA_HA *info, uint keynr, uint key_length) } info->maria_rtree_recursion_depth = -1; - info->buff_used = 1; + info->keybuff_used = 1; return maria_rtree_get_req(info, &keyinfo[keynr], key_length, root, 0); } @@ -408,23 +410,23 @@ int maria_rtree_get_next(MARIA_HA *info, uint keynr, uint key_length) my_off_t root; MARIA_KEYDEF *keyinfo = info->s->keyinfo + keynr; - if (!info->buff_used) + if (!info->keybuff_used) { uint k_len = keyinfo->keylength - info->s->base.rec_reflength; /* rt_PAGE_NEXT_KEY(info->int_keypos) */ - uchar *key = info->buff + *(int*)info->int_keypos + k_len + + byte *key = info->buff + *(int*)info->int_keypos + k_len + info->s->base.rec_reflength; /* rt_PAGE_NEXT_KEY(key) */ - uchar *after_key = key + k_len + info->s->base.rec_reflength; + byte *after_key = key + k_len + info->s->base.rec_reflength; - info->lastpos = _ma_dpos(info, 0, after_key); + info->cur_row.lastpos = _ma_dpos(info, 0, after_key); info->lastkey_length = k_len + info->s->base.rec_reflength; memcpy(info->lastkey, key, k_len + info->s->base.rec_reflength); *(int*)info->int_keypos = key - info->buff; if (after_key >= info->int_maxpos) { - info->buff_used = 1; + info->keybuff_used = 1; } return 0; @@ -447,8 +449,10 @@ int maria_rtree_get_next(MARIA_HA *info, uint keynr, uint key_length) */ #ifdef PICK_BY_PERIMETER -static uchar *maria_rtree_pick_key(MARIA_HA *info, MARIA_KEYDEF *keyinfo, uchar *key, - uint key_length, uchar *page_buf, uint nod_flag) +static uchar *maria_rtree_pick_key(MARIA_HA *info, MARIA_KEYDEF *keyinfo, + uchar *key, + uint key_length, byte *page_buf, + uint nod_flag) { double increase; double best_incr = DBL_MAX; @@ -480,16 +484,18 @@ static uchar *maria_rtree_pick_key(MARIA_HA *info, MARIA_KEYDEF *keyinfo, uchar #endif /*PICK_BY_PERIMETER*/ #ifdef PICK_BY_AREA -static uchar *maria_rtree_pick_key(MARIA_HA *info, MARIA_KEYDEF *keyinfo, uchar *key, - uint key_length, uchar *page_buf, uint nod_flag) +static byte *maria_rtree_pick_key(MARIA_HA *info, MARIA_KEYDEF *keyinfo, + byte *key, + uint key_length, byte *page_buf, + uint nod_flag) { double increase; double best_incr = DBL_MAX; double area; double best_area; - uchar *best_key; - uchar *k = rt_PAGE_FIRST_KEY(page_buf, nod_flag); - uchar *last = rt_PAGE_END(page_buf); + byte *best_key; + byte *k = rt_PAGE_FIRST_KEY(page_buf, nod_flag); + byte *last = rt_PAGE_END(page_buf); LINT_INIT(best_area); LINT_INIT(best_key); @@ -498,7 +504,7 @@ static uchar *maria_rtree_pick_key(MARIA_HA *info, MARIA_KEYDEF *keyinfo, uchar { /* The following is safe as -1.0 is an exact number */ if ((increase = maria_rtree_area_increase(keyinfo->seg, k, key, key_length, - &area)) == -1.0) + &area)) == -1.0) return NULL; /* The following should be safe, even if we compare doubles */ if (increase < best_incr) @@ -532,16 +538,17 @@ static uchar *maria_rtree_pick_key(MARIA_HA *info, MARIA_KEYDEF *keyinfo, uchar 1 Child was split */ -static int maria_rtree_insert_req(MARIA_HA *info, MARIA_KEYDEF *keyinfo, uchar *key, - uint key_length, my_off_t page, my_off_t *new_page, - int ins_level, int level) +static int maria_rtree_insert_req(MARIA_HA *info, MARIA_KEYDEF *keyinfo, + byte *key, + uint key_length, my_off_t page, + my_off_t *new_page, + int ins_level, int level) { - uchar *k; uint nod_flag; - uchar *page_buf; int res; + byte *page_buf, *k; - if (!(page_buf = (uchar*)my_alloca((uint)keyinfo->block_length + + if (!(page_buf= (byte*) my_alloca((uint)keyinfo->block_length + HA_MAX_KEY_BUFF))) { my_errno = HA_ERR_OUT_OF_MEM; @@ -555,10 +562,11 @@ static int maria_rtree_insert_req(MARIA_HA *info, MARIA_KEYDEF *keyinfo, uchar * (ins_level > -1 && ins_level > level)) /* branch: go down to ins_level */ { if ((k = maria_rtree_pick_key(info, keyinfo, key, key_length, page_buf, - nod_flag)) == NULL) + nod_flag)) == NULL) goto err1; switch ((res = maria_rtree_insert_req(info, keyinfo, key, key_length, - _ma_kpos(nod_flag, k), new_page, ins_level, level + 1))) + _ma_kpos(nod_flag, k), new_page, + ins_level, level + 1))) { case 0: /* child was not split */ { @@ -569,14 +577,15 @@ static int maria_rtree_insert_req(MARIA_HA *info, MARIA_KEYDEF *keyinfo, uchar * } case 1: /* child was split */ { - uchar *new_key = page_buf + keyinfo->block_length + nod_flag; + byte *new_key = page_buf + keyinfo->block_length + nod_flag; /* set proper MBR for key */ if (maria_rtree_set_key_mbr(info, keyinfo, k, key_length, - _ma_kpos(nod_flag, k))) + _ma_kpos(nod_flag, k))) goto err1; /* add new key for new page */ _ma_kpointer(info, new_key - nod_flag, *new_page); - if (maria_rtree_set_key_mbr(info, keyinfo, new_key, key_length, *new_page)) + if (maria_rtree_set_key_mbr(info, keyinfo, new_key, key_length, + *new_page)) goto err1; res = maria_rtree_add_key(info, keyinfo, new_key, key_length, page_buf, new_page); @@ -593,18 +602,18 @@ static int maria_rtree_insert_req(MARIA_HA *info, MARIA_KEYDEF *keyinfo, uchar * } else { - res = maria_rtree_add_key(info, keyinfo, key, key_length, page_buf, new_page); + res = maria_rtree_add_key(info, keyinfo, key, key_length, page_buf, + new_page); if (_ma_write_keypage(info, keyinfo, page, DFLT_INIT_HITS, page_buf)) goto err1; - goto ok; } ok: - my_afree((byte*)page_buf); + my_afree(page_buf); return res; err1: - my_afree((byte*)page_buf); + my_afree(page_buf); return -1; } @@ -618,8 +627,8 @@ err1: 1 Root was split */ -static int maria_rtree_insert_level(MARIA_HA *info, uint keynr, uchar *key, - uint key_length, int ins_level) +static int maria_rtree_insert_level(MARIA_HA *info, uint keynr, byte *key, + uint key_length, int ins_level) { my_off_t old_root; MARIA_KEYDEF *keyinfo = info->s->keyinfo + keynr; @@ -632,7 +641,7 @@ static int maria_rtree_insert_level(MARIA_HA *info, uint keynr, uchar *key, if ((old_root = _ma_new(info, keyinfo, DFLT_INIT_HITS)) == HA_OFFSET_ERROR) return -1; - info->buff_used = 1; + info->keybuff_used = 1; maria_putint(info->buff, 2, 0); res = maria_rtree_add_key(info, keyinfo, key, key_length, info->buff, NULL); if (_ma_write_keypage(info, keyinfo, old_root, DFLT_INIT_HITS, info->buff)) @@ -650,13 +659,12 @@ static int maria_rtree_insert_level(MARIA_HA *info, uint keynr, uchar *key, } case 1: /* root was split, grow a new root */ { - uchar *new_root_buf; + byte *new_root_buf, *new_key; my_off_t new_root; - uchar *new_key; uint nod_flag = info->s->base.key_reflength; - if (!(new_root_buf = (uchar*)my_alloca((uint)keyinfo->block_length + - HA_MAX_KEY_BUFF))) + if (!(new_root_buf= (byte*) my_alloca((uint)keyinfo->block_length + + HA_MAX_KEY_BUFF))) { my_errno = HA_ERR_OUT_OF_MEM; return -1; @@ -670,15 +678,19 @@ static int maria_rtree_insert_level(MARIA_HA *info, uint keynr, uchar *key, new_key = new_root_buf + keyinfo->block_length + nod_flag; _ma_kpointer(info, new_key - nod_flag, old_root); - if (maria_rtree_set_key_mbr(info, keyinfo, new_key, key_length, old_root)) + if (maria_rtree_set_key_mbr(info, keyinfo, new_key, key_length, + old_root)) goto err1; - if (maria_rtree_add_key(info, keyinfo, new_key, key_length, new_root_buf, NULL) + if (maria_rtree_add_key(info, keyinfo, new_key, key_length, new_root_buf, + NULL) == -1) goto err1; _ma_kpointer(info, new_key - nod_flag, new_page); - if (maria_rtree_set_key_mbr(info, keyinfo, new_key, key_length, new_page)) + if (maria_rtree_set_key_mbr(info, keyinfo, new_key, key_length, + new_page)) goto err1; - if (maria_rtree_add_key(info, keyinfo, new_key, key_length, new_root_buf, NULL) + if (maria_rtree_add_key(info, keyinfo, new_key, key_length, new_root_buf, + NULL) == -1) goto err1; if (_ma_write_keypage(info, keyinfo, new_root, @@ -710,10 +722,11 @@ err1: 0 OK */ -int maria_rtree_insert(MARIA_HA *info, uint keynr, uchar *key, uint key_length) +int maria_rtree_insert(MARIA_HA *info, uint keynr, byte *key, uint key_length) { return (!key_length || - (maria_rtree_insert_level(info, keynr, key, key_length, -1) == -1)) ? -1 : 0; + (maria_rtree_insert_level(info, keynr, key, key_length, -1) == -1)) ? + -1 : 0; } @@ -756,18 +769,18 @@ err1: 2 Empty leaf */ -static int maria_rtree_delete_req(MARIA_HA *info, MARIA_KEYDEF *keyinfo, uchar *key, - uint key_length, my_off_t page, uint *page_size, - stPageList *ReinsertList, int level) +static int maria_rtree_delete_req(MARIA_HA *info, MARIA_KEYDEF *keyinfo, + byte *key, + uint key_length, my_off_t page, + uint *page_size, + stPageList *ReinsertList, int level) { - uchar *k; - uchar *last; ulong i; uint nod_flag; - uchar *page_buf; int res; + byte *page_buf, *last, *k; - if (!(page_buf = (uchar*)my_alloca((uint)keyinfo->block_length))) + if (!(page_buf = (byte*) my_alloca((uint)keyinfo->block_length))) { my_errno = HA_ERR_OUT_OF_MEM; return -1; @@ -779,7 +792,7 @@ static int maria_rtree_delete_req(MARIA_HA *info, MARIA_KEYDEF *keyinfo, uchar * k = rt_PAGE_FIRST_KEY(page_buf, nod_flag); last = rt_PAGE_END(page_buf); - for (i = 0; k < last; k = rt_PAGE_NEXT_KEY(k, key_length, nod_flag), ++i) + for (i = 0; k < last; k = rt_PAGE_NEXT_KEY(k, key_length, nod_flag), i++) { if (nod_flag) { @@ -792,7 +805,8 @@ static int maria_rtree_delete_req(MARIA_HA *info, MARIA_KEYDEF *keyinfo, uchar * case 0: /* deleted */ { /* test page filling */ - if (*page_size + key_length >= rt_PAGE_MIN_SIZE(keyinfo->block_length)) + if (*page_size + key_length >= + rt_PAGE_MIN_SIZE(keyinfo->block_length)) { /* OK */ if (maria_rtree_set_key_mbr(info, keyinfo, k, key_length, @@ -805,7 +819,8 @@ static int maria_rtree_delete_req(MARIA_HA *info, MARIA_KEYDEF *keyinfo, uchar * else { /* too small: delete key & add it descendant to reinsert list */ - if (maria_rtree_fill_reinsert_list(ReinsertList, _ma_kpos(nod_flag, k), + if (maria_rtree_fill_reinsert_list(ReinsertList, + _ma_kpos(nod_flag, k), level + 1)) goto err1; maria_rtree_delete_key(info, page_buf, k, key_length, nod_flag); @@ -883,7 +898,7 @@ err1: 0 Deleted */ -int maria_rtree_delete(MARIA_HA *info, uint keynr, uchar *key, uint key_length) +int maria_rtree_delete(MARIA_HA *info, uint keynr, byte *key, uint key_length) { uint page_size; stPageList ReinsertList; @@ -914,12 +929,10 @@ int maria_rtree_delete(MARIA_HA *info, uint keynr, uchar *key, uint key_length) ulong i; for (i = 0; i < ReinsertList.n_pages; ++i) { - uchar *page_buf; uint nod_flag; - uchar *k; - uchar *last; + byte *page_buf, *k, *last; - if (!(page_buf = (uchar*)my_alloca((uint)keyinfo->block_length))) + if (!(page_buf = (byte*) my_alloca((uint)keyinfo->block_length))) { my_errno = HA_ERR_OUT_OF_MEM; goto err1; @@ -935,11 +948,11 @@ int maria_rtree_delete(MARIA_HA *info, uint keynr, uchar *key, uint key_length) if (maria_rtree_insert_level(info, keynr, k, key_length, ReinsertList.pages[i].level) == -1) { - my_afree((byte*)page_buf); + my_afree(page_buf); goto err1; } } - my_afree((byte*)page_buf); + my_afree(page_buf); if (_ma_dispose(info, keyinfo, ReinsertList.pages[i].offs, DFLT_INIT_HITS)) goto err1; @@ -969,16 +982,14 @@ int maria_rtree_delete(MARIA_HA *info, uint keynr, uchar *key, uint key_length) err1: return -1; } - case 1: /* not found */ + case 1: /* not found */ { my_errno = HA_ERR_KEY_NOT_FOUND; return -1; } default: - case -1: /* error */ - { + case -1: /* error */ return -1; - } } } @@ -990,17 +1001,14 @@ err1: estimated value */ -ha_rows maria_rtree_estimate(MARIA_HA *info, uint keynr, uchar *key, +ha_rows maria_rtree_estimate(MARIA_HA *info, uint keynr, byte *key, uint key_length, uint flag) { MARIA_KEYDEF *keyinfo = info->s->keyinfo + keynr; my_off_t root; uint i = 0; - uchar *k; - uchar *last; - uint nod_flag; - uchar *page_buf; - uint k_len; + uint nod_flag, k_len; + byte *page_buf, *k, *last; double area = 0; ha_rows res = 0; @@ -1009,7 +1017,7 @@ ha_rows maria_rtree_estimate(MARIA_HA *info, uint keynr, uchar *key, if ((root = info->s->state.key_root[keynr]) == HA_OFFSET_ERROR) return HA_POS_ERROR; - if (!(page_buf = (uchar*)my_alloca((uint)keyinfo->block_length))) + if (!(page_buf= (byte*) my_alloca((uint)keyinfo->block_length))) return HA_POS_ERROR; if (!_ma_fetch_keypage(info, keyinfo, root, DFLT_INIT_HITS, page_buf, 0)) goto err1; @@ -1020,7 +1028,7 @@ ha_rows maria_rtree_estimate(MARIA_HA *info, uint keynr, uchar *key, k = rt_PAGE_FIRST_KEY(page_buf, nod_flag); last = rt_PAGE_END(page_buf); - for (; k < last; k = rt_PAGE_NEXT_KEY(k, k_len, nod_flag), ++i) + for (; k < last; k = rt_PAGE_NEXT_KEY(k, k_len, nod_flag), i++) { if (nod_flag) { @@ -1035,7 +1043,8 @@ ha_rows maria_rtree_estimate(MARIA_HA *info, uint keynr, uchar *key, } else if (flag & (MBR_WITHIN | MBR_EQUAL)) { - if (!maria_rtree_key_cmp(keyinfo->seg, key, k, key_length, MBR_WITHIN)) + if (!maria_rtree_key_cmp(keyinfo->seg, key, k, key_length, + MBR_WITHIN)) area += 1; } else @@ -1045,14 +1054,15 @@ ha_rows maria_rtree_estimate(MARIA_HA *info, uint keynr, uchar *key, { if (flag & (MBR_CONTAIN | MBR_INTERSECT)) { - area += maria_rtree_overlapping_area(keyinfo->seg, key, k, key_length) / - k_area; + area+= maria_rtree_overlapping_area(keyinfo->seg, key, k, + key_length) / k_area; } else if (flag & (MBR_WITHIN | MBR_EQUAL)) { - if (!maria_rtree_key_cmp(keyinfo->seg, key, k, key_length, MBR_WITHIN)) - area += maria_rtree_rect_volume(keyinfo->seg, key, key_length) / - k_area; + if (!maria_rtree_key_cmp(keyinfo->seg, key, k, key_length, + MBR_WITHIN)) + area+= (maria_rtree_rect_volume(keyinfo->seg, key, key_length) / + k_area); } else goto err1; @@ -1076,7 +1086,7 @@ ha_rows maria_rtree_estimate(MARIA_HA *info, uint keynr, uchar *key, return res; err1: - my_afree((byte*)page_buf); + my_afree(page_buf); return HA_POS_ERROR; } diff --git a/storage/maria/ma_rt_index.h b/storage/maria/ma_rt_index.h index ff431d81372..76b6c6e230c 100644 --- a/storage/maria/ma_rt_index.h +++ b/storage/maria/ma_rt_index.h @@ -27,21 +27,22 @@ #define rt_PAGE_MIN_SIZE(block_length) ((uint)(block_length) / 3) -int maria_rtree_insert(MARIA_HA *info, uint keynr, uchar *key, uint key_length); -int maria_rtree_delete(MARIA_HA *info, uint keynr, uchar *key, uint key_length); +int maria_rtree_insert(MARIA_HA *info, uint keynr, byte *key, uint key_length); +int maria_rtree_delete(MARIA_HA *info, uint keynr, byte *key, uint key_length); -int maria_rtree_find_first(MARIA_HA *info, uint keynr, uchar *key, uint key_length, - uint search_flag); +int maria_rtree_find_first(MARIA_HA *info, uint keynr, byte *key, + uint key_length, uint search_flag); int maria_rtree_find_next(MARIA_HA *info, uint keynr, uint search_flag); int maria_rtree_get_first(MARIA_HA *info, uint keynr, uint key_length); int maria_rtree_get_next(MARIA_HA *info, uint keynr, uint key_length); -ha_rows maria_rtree_estimate(MARIA_HA *info, uint keynr, uchar *key, - uint key_length, uint flag); +ha_rows maria_rtree_estimate(MARIA_HA *info, uint keynr, byte *key, + uint key_length, uint flag); -int maria_rtree_split_page(MARIA_HA *info, MARIA_KEYDEF *keyinfo, uchar *page, - uchar *key, uint key_length, my_off_t *new_page_offs); +int maria_rtree_split_page(MARIA_HA *info, MARIA_KEYDEF *keyinfo, byte *page, + byte *key, uint key_length, + my_off_t *new_page_offs); #endif /*HAVE_RTREE_KEYS*/ #endif /* _rt_index_h */ diff --git a/storage/maria/ma_rt_key.c b/storage/maria/ma_rt_key.c index 2732fefffbe..1453195d263 100644 --- a/storage/maria/ma_rt_key.c +++ b/storage/maria/ma_rt_key.c @@ -30,8 +30,8 @@ 1 Split */ -int maria_rtree_add_key(MARIA_HA *info, MARIA_KEYDEF *keyinfo, uchar *key, - uint key_length, uchar *page_buf, my_off_t *new_page) +int maria_rtree_add_key(MARIA_HA *info, MARIA_KEYDEF *keyinfo, byte *key, + uint key_length, byte *page_buf, my_off_t *new_page) { uint page_size = maria_getint(page_buf); uint nod_flag = _ma_test_if_nod(page_buf); @@ -61,14 +61,16 @@ int maria_rtree_add_key(MARIA_HA *info, MARIA_KEYDEF *keyinfo, uchar *key, new_page) ? -1 : 1); } + /* Delete key from the page */ -int maria_rtree_delete_key(MARIA_HA *info, uchar *page_buf, uchar *key, + +int maria_rtree_delete_key(MARIA_HA *info, byte *page_buf, byte *key, uint key_length, uint nod_flag) { uint16 page_size = maria_getint(page_buf); - uchar *key_start; + byte *key_start; key_start= key - nod_flag; if (!nod_flag) @@ -87,7 +89,7 @@ int maria_rtree_delete_key(MARIA_HA *info, uchar *page_buf, uchar *key, Calculate and store key MBR */ -int maria_rtree_set_key_mbr(MARIA_HA *info, MARIA_KEYDEF *keyinfo, uchar *key, +int maria_rtree_set_key_mbr(MARIA_HA *info, MARIA_KEYDEF *keyinfo, byte *key, uint key_length, my_off_t child_page) { if (!_ma_fetch_keypage(info, keyinfo, child_page, diff --git a/storage/maria/ma_rt_key.h b/storage/maria/ma_rt_key.h index 448024ed8c5..f44251782c1 100644 --- a/storage/maria/ma_rt_key.h +++ b/storage/maria/ma_rt_key.h @@ -22,12 +22,12 @@ #ifdef HAVE_RTREE_KEYS -int maria_rtree_add_key(MARIA_HA *info, MARIA_KEYDEF *keyinfo, uchar *key, - uint key_length, uchar *page_buf, my_off_t *new_page); -int maria_rtree_delete_key(MARIA_HA *info, uchar *page, uchar *key, - uint key_length, uint nod_flag); -int maria_rtree_set_key_mbr(MARIA_HA *info, MARIA_KEYDEF *keyinfo, uchar *key, - uint key_length, my_off_t child_page); +int maria_rtree_add_key(MARIA_HA *info, MARIA_KEYDEF *keyinfo, byte *key, + uint key_length, byte *page_buf, my_off_t *new_page); +int maria_rtree_delete_key(MARIA_HA *info, byte *page, byte *key, + uint key_length, uint nod_flag); +int maria_rtree_set_key_mbr(MARIA_HA *info, MARIA_KEYDEF *keyinfo, byte *key, + uint key_length, my_off_t child_page); #endif /*HAVE_RTREE_KEYS*/ #endif /* _rt_key_h */ diff --git a/storage/maria/ma_rt_mbr.c b/storage/maria/ma_rt_mbr.c index 67b1d59f505..851618a4300 100644 --- a/storage/maria/ma_rt_mbr.c +++ b/storage/maria/ma_rt_mbr.c @@ -93,8 +93,9 @@ MBR_DATA(a,b) Data reference is the same Returns 0 on success. */ -int maria_rtree_key_cmp(HA_KEYSEG *keyseg, uchar *b, uchar *a, uint key_length, - uint nextflag) + +int maria_rtree_key_cmp(HA_KEYSEG *keyseg, byte *b, byte *a, uint key_length, + uint nextflag) { for (; (int) key_length > 0; keyseg += 2 ) { @@ -153,7 +154,7 @@ int maria_rtree_key_cmp(HA_KEYSEG *keyseg, uchar *b, uchar *a, uint key_length, end: if (nextflag & MBR_DATA) { - uchar *end = a + keyseg->length; + byte *end = a + keyseg->length; do { if (*a++ != *b++) @@ -182,7 +183,7 @@ end: /* Calculates rectangle volume */ -double maria_rtree_rect_volume(HA_KEYSEG *keyseg, uchar *a, uint key_length) +double maria_rtree_rect_volume(HA_KEYSEG *keyseg, byte *a, uint key_length) { double res = 1; for (; (int)key_length > 0; keyseg += 2) @@ -263,7 +264,7 @@ double maria_rtree_rect_volume(HA_KEYSEG *keyseg, uchar *a, uint key_length) Creates an MBR as an array of doubles. */ -int maria_rtree_d_mbr(HA_KEYSEG *keyseg, uchar *a, uint key_length, double *res) +int maria_rtree_d_mbr(HA_KEYSEG *keyseg, byte *a, uint key_length, double *res) { for (; (int)key_length > 0; keyseg += 2) { @@ -352,7 +353,7 @@ int maria_rtree_d_mbr(HA_KEYSEG *keyseg, uchar *a, uint key_length, double *res) Result is written to c */ -int maria_rtree_combine_rect(HA_KEYSEG *keyseg, uchar* a, uchar* b, uchar* c, +int maria_rtree_combine_rect(HA_KEYSEG *keyseg, byte* a, byte* b, byte* c, uint key_length) { for ( ; (int) key_length > 0 ; keyseg += 2) @@ -443,7 +444,7 @@ int maria_rtree_combine_rect(HA_KEYSEG *keyseg, uchar* a, uchar* b, uchar* c, /* Calculates overlapping area of two MBRs a & b */ -double maria_rtree_overlapping_area(HA_KEYSEG *keyseg, uchar* a, uchar* b, +double maria_rtree_overlapping_area(HA_KEYSEG *keyseg, byte* a, byte* b, uint key_length) { double res = 1; @@ -525,10 +526,11 @@ double maria_rtree_overlapping_area(HA_KEYSEG *keyseg, uchar* a, uchar* b, } /* -Calculates MBR_AREA(a+b) - MBR_AREA(a) + Calculates MBR_AREA(a+b) - MBR_AREA(a) */ -double maria_rtree_area_increase(HA_KEYSEG *keyseg, uchar* a, uchar* b, - uint key_length, double *ab_area) + +double maria_rtree_area_increase(HA_KEYSEG *keyseg, byte *a, byte *b, + uint key_length, double *ab_area) { double a_area= 1.0; double loc_ab_area= 1.0; @@ -620,7 +622,7 @@ safe_end: /* Calculates MBR_PERIMETER(a+b) - MBR_PERIMETER(a) */ -double maria_rtree_perimeter_increase(HA_KEYSEG *keyseg, uchar* a, uchar* b, +double maria_rtree_perimeter_increase(HA_KEYSEG *keyseg, byte* a, byte* b, uint key_length, double *ab_perim) { double a_perim = 0.0; @@ -731,16 +733,16 @@ double maria_rtree_perimeter_increase(HA_KEYSEG *keyseg, uchar* a, uchar* b, } /* -Calculates key page total MBR = MBR(key1) + MBR(key2) + ... + Calculates key page total MBR = MBR(key1) + MBR(key2) + ... */ -int maria_rtree_page_mbr(MARIA_HA *info, HA_KEYSEG *keyseg, uchar *page_buf, - uchar *c, uint key_length) +int maria_rtree_page_mbr(MARIA_HA *info, HA_KEYSEG *keyseg, byte *page_buf, + byte *c, uint key_length) { uint inc = 0; uint k_len = key_length; uint nod_flag = _ma_test_if_nod(page_buf); - uchar *k; - uchar *last = rt_PAGE_END(page_buf); + byte *k; + byte *last = rt_PAGE_END(page_buf); for (; (int)key_length > 0; keyseg += 2) { diff --git a/storage/maria/ma_rt_mbr.h b/storage/maria/ma_rt_mbr.h index 81e2a6851d4..3282ee0d7a3 100644 --- a/storage/maria/ma_rt_mbr.h +++ b/storage/maria/ma_rt_mbr.h @@ -20,19 +20,20 @@ #ifdef HAVE_RTREE_KEYS -int maria_rtree_key_cmp(HA_KEYSEG *keyseg, uchar *a, uchar *b, uint key_length, - uint nextflag); -int maria_rtree_combine_rect(HA_KEYSEG *keyseg,uchar *, uchar *, uchar*, - uint key_length); -double maria_rtree_rect_volume(HA_KEYSEG *keyseg, uchar*, uint key_length); -int maria_rtree_d_mbr(HA_KEYSEG *keyseg, uchar *a, uint key_length, double *res); -double maria_rtree_overlapping_area(HA_KEYSEG *keyseg, uchar *a, uchar *b, - uint key_length); -double maria_rtree_area_increase(HA_KEYSEG *keyseg, uchar *a, uchar *b, - uint key_length, double *ab_area); -double maria_rtree_perimeter_increase(HA_KEYSEG *keyseg, uchar* a, uchar* b, - uint key_length, double *ab_perim); -int maria_rtree_page_mbr(MARIA_HA *info, HA_KEYSEG *keyseg, uchar *page_buf, - uchar* c, uint key_length); +int maria_rtree_key_cmp(HA_KEYSEG *keyseg, byte *a, byte *b, uint key_length, + uint nextflag); +int maria_rtree_combine_rect(HA_KEYSEG *keyseg,byte *, byte *, byte*, + uint key_length); +double maria_rtree_rect_volume(HA_KEYSEG *keyseg, byte*, uint key_length); +int maria_rtree_d_mbr(HA_KEYSEG *keyseg, byte *a, uint key_length, + double *res); +double maria_rtree_overlapping_area(HA_KEYSEG *keyseg, byte *a, byte *b, + uint key_length); +double maria_rtree_area_increase(HA_KEYSEG *keyseg, byte *a, byte *b, + uint key_length, double *ab_area); +double maria_rtree_perimeter_increase(HA_KEYSEG *keyseg, byte* a, byte* b, + uint key_length, double *ab_perim); +int maria_rtree_page_mbr(MARIA_HA *info, HA_KEYSEG *keyseg, byte *page_buf, + byte* c, uint key_length); #endif /*HAVE_RTREE_KEYS*/ #endif /* _rt_mbr_h */ diff --git a/storage/maria/ma_rt_split.c b/storage/maria/ma_rt_split.c index 034799efd89..00c8d18f5e5 100644 --- a/storage/maria/ma_rt_split.c +++ b/storage/maria/ma_rt_split.c @@ -27,7 +27,7 @@ typedef struct { double square; int n_node; - uchar *key; + byte *key; double *coords; } SplitStruct; @@ -243,8 +243,9 @@ static int split_maria_rtree_node(SplitStruct *node, int n_entries, return 0; } -int maria_rtree_split_page(MARIA_HA *info, MARIA_KEYDEF *keyinfo, uchar *page, uchar *key, - uint key_length, my_off_t *new_page_offs) +int maria_rtree_split_page(MARIA_HA *info, MARIA_KEYDEF *keyinfo, + byte *page, byte *key, + uint key_length, my_off_t *new_page_offs) { int n1, n2; /* Number of items in groups */ @@ -255,8 +256,8 @@ int maria_rtree_split_page(MARIA_HA *info, MARIA_KEYDEF *keyinfo, uchar *page, u double *next_coord; double *old_coord; int n_dim; - uchar *source_cur, *cur1, *cur2; - uchar *new_page; + byte *source_cur, *cur1, *cur2; + byte *new_page; int err_code= 0; uint nod_flag= _ma_test_if_nod(page); uint full_length= key_length + (nod_flag ? nod_flag : @@ -300,7 +301,7 @@ int maria_rtree_split_page(MARIA_HA *info, MARIA_KEYDEF *keyinfo, uchar *page, u goto split_err; } - if (!(new_page = (uchar*)my_alloca((uint)keyinfo->block_length))) + if (!(new_page = (byte*) my_alloca((uint)keyinfo->block_length))) { err_code= -1; goto split_err; @@ -313,7 +314,7 @@ int maria_rtree_split_page(MARIA_HA *info, MARIA_KEYDEF *keyinfo, uchar *page, u n1= n2 = 0; for (cur = task; cur < stop; ++cur) { - uchar *to; + byte *to; if (cur->n_node == 1) { to = cur1; diff --git a/storage/maria/ma_rt_test.c b/storage/maria/ma_rt_test.c index ca4825c2ce2..04b0c88c222 100644 --- a/storage/maria/ma_rt_test.c +++ b/storage/maria/ma_rt_test.c @@ -153,10 +153,11 @@ static int run_test(const char *filename) create_info.max_rows=10000000; if (maria_create(filename, - 1, /* keys */ - keyinfo, - 1+2*ndims+opt_unique, /* columns */ - recinfo,uniques,&uniquedef,&create_info,create_flag)) + DYNAMIC_RECORD, + 1, /* keys */ + keyinfo, + 1+2*ndims+opt_unique, /* columns */ + recinfo,uniques,&uniquedef,&create_info,create_flag)) goto err; if (!silent) diff --git a/storage/maria/ma_scan.c b/storage/maria/ma_scan.c index c9c988722b7..4538c87e2be 100644 --- a/storage/maria/ma_scan.c +++ b/storage/maria/ma_scan.c @@ -21,26 +21,41 @@ int maria_scan_init(register MARIA_HA *info) { DBUG_ENTER("maria_scan_init"); - info->nextpos=info->s->pack.header_length; /* Read first record */ + + info->cur_row.nextpos= info->s->pack.header_length; /* Read first record */ info->lastinx= -1; /* Can't forward or backward */ if (info->opt_flag & WRITE_CACHE_USED && flush_io_cache(&info->rec_cache)) DBUG_RETURN(my_errno); + + if ((*info->s->scan_init)(info)) + DBUG_RETURN(my_errno); DBUG_RETURN(0); } /* - Read a row based on position. - If filepos= HA_OFFSET_ERROR then read next row - Return values - Returns one of following values: - 0 = Ok. - HA_ERR_END_OF_FILE = EOF. + Read a row based on position. + + SYNOPSIS + maria_scan() + info Maria handler + record Read data here + + RETURN + 0 ok + HA_ERR_END_OF_FILE End of file + # Error code */ -int maria_scan(MARIA_HA *info, byte *buf) +int maria_scan(MARIA_HA *info, byte *record) { DBUG_ENTER("maria_scan"); /* Init all but update-flag */ info->update&= (HA_STATE_CHANGED | HA_STATE_ROW_CHANGED); - DBUG_RETURN ((*info->s->read_rnd)(info,buf,info->nextpos,1)); + DBUG_RETURN((*info->s->scan)(info, record, info->cur_row.nextpos, 1)); +} + + +void maria_scan_end(MARIA_HA *info) +{ + (*info->s->scan_end)(info); } diff --git a/storage/maria/ma_search.c b/storage/maria/ma_search.c index af25be06a09..d8738ae4639 100644 --- a/storage/maria/ma_search.c +++ b/storage/maria/ma_search.c @@ -19,8 +19,9 @@ #include "ma_fulltext.h" #include "m_ctype.h" -static my_bool _ma_get_prev_key(MARIA_HA *info, MARIA_KEYDEF *keyinfo, uchar *page, - uchar *key, uchar *keypos, +static my_bool _ma_get_prev_key(MARIA_HA *info, MARIA_KEYDEF *keyinfo, + byte *page, + byte *key, byte *keypos, uint *return_key_length); /* Check index */ @@ -55,31 +56,32 @@ int _ma_check_index(MARIA_HA *info, int inx) */ int _ma_search(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, - uchar *key, uint key_len, uint nextflag, register my_off_t pos) + byte *key, uint key_len, uint nextflag, register my_off_t pos) { my_bool last_key; int error,flag; uint nod_flag; - uchar *keypos,*maxpos; - uchar lastkey[HA_MAX_KEY_BUFF],*buff; + byte *keypos,*maxpos; + byte lastkey[HA_MAX_KEY_BUFF],*buff; DBUG_ENTER("_ma_search"); DBUG_PRINT("enter",("pos: %lu nextflag: %u lastpos: %lu", - (ulong) pos, nextflag, (ulong) info->lastpos)); + (ulong) pos, nextflag, (ulong) info->cur_row.lastpos)); DBUG_EXECUTE("key", _ma_print_key(DBUG_FILE,keyinfo->seg,key,key_len);); if (pos == HA_OFFSET_ERROR) { my_errno=HA_ERR_KEY_NOT_FOUND; /* Didn't find key */ - info->lastpos= HA_OFFSET_ERROR; + info->cur_row.lastpos= HA_OFFSET_ERROR; if (!(nextflag & (SEARCH_SMALLER | SEARCH_BIGGER | SEARCH_LAST))) DBUG_RETURN(-1); /* Not found ; return error */ DBUG_RETURN(1); /* Search at upper levels */ } - if (!(buff= _ma_fetch_keypage(info,keyinfo,pos,DFLT_INIT_HITS,info->buff, + if (!(buff= _ma_fetch_keypage(info,keyinfo,pos,DFLT_INIT_HITS, + info->keyread_buff, test(!(nextflag & SEARCH_SAVE_BUFF))))) goto err; - DBUG_DUMP("page",(byte*) buff,maria_getint(buff)); + DBUG_DUMP("page", buff, maria_getint(buff)); flag=(*keyinfo->bin_search)(info,keyinfo,buff,key,key_len,nextflag, &keypos,lastkey, &last_key); @@ -118,9 +120,10 @@ int _ma_search(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, } if (pos != info->last_keypage) { - uchar *old_buff=buff; - if (!(buff= _ma_fetch_keypage(info,keyinfo,pos,DFLT_INIT_HITS,info->buff, - test(!(nextflag & SEARCH_SAVE_BUFF))))) + byte *old_buff=buff; + if (!(buff= _ma_fetch_keypage(info,keyinfo,pos,DFLT_INIT_HITS, + info->keyread_buff, + test(!(nextflag & SEARCH_SAVE_BUFF))))) goto err; keypos=buff+(keypos-old_buff); maxpos=buff+(maxpos-old_buff); @@ -133,8 +136,8 @@ int _ma_search(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, &info->lastkey_length)) goto err; if (!(nextflag & SEARCH_SMALLER) && - ha_key_cmp(keyinfo->seg, info->lastkey, key, key_len, SEARCH_FIND, - not_used)) + ha_key_cmp(keyinfo->seg, (uchar*) info->lastkey, (uchar*) key, key_len, + SEARCH_FIND, not_used)) { my_errno=HA_ERR_KEY_NOT_FOUND; /* Didn't find key */ goto err; @@ -147,22 +150,22 @@ int _ma_search(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, goto err; memcpy(info->lastkey,lastkey,info->lastkey_length); } - info->lastpos= _ma_dpos(info,0,info->lastkey+info->lastkey_length); + info->cur_row.lastpos= _ma_dpos(info,0,info->lastkey+info->lastkey_length); /* Save position for a possible read next / previous */ - info->int_keypos=info->buff+ (keypos-buff); - info->int_maxpos=info->buff+ (maxpos-buff); + info->int_keypos= info->keyread_buff+ (keypos-buff); + info->int_maxpos= info->keyread_buff+ (maxpos-buff); info->int_nod_flag=nod_flag; info->int_keytree_version=keyinfo->version; info->last_search_keypage=info->last_keypage; info->page_changed=0; - info->buff_used= (info->buff != buff); /* If we have to reread buff */ + info->keybuff_used= (info->keyread_buff != buff); /* If we have to reread */ - DBUG_PRINT("exit",("found key at %lu",(ulong) info->lastpos)); + DBUG_PRINT("exit",("found key at %lu",(ulong) info->cur_row.lastpos)); DBUG_RETURN(0); err: DBUG_PRINT("exit",("Error: %d",my_errno)); - info->lastpos= HA_OFFSET_ERROR; + info->cur_row.lastpos= HA_OFFSET_ERROR; info->page_changed=1; DBUG_RETURN (-1); } /* _ma_search */ @@ -173,9 +176,9 @@ err: /* ret_pos point to where find or bigger key starts */ /* ARGSUSED */ -int _ma_bin_search(MARIA_HA *info, register MARIA_KEYDEF *keyinfo, uchar *page, - uchar *key, uint key_len, uint comp_flag, uchar **ret_pos, - uchar *buff __attribute__((unused)), my_bool *last_key) +int _ma_bin_search(MARIA_HA *info, register MARIA_KEYDEF *keyinfo, byte *page, + byte *key, uint key_len, uint comp_flag, byte **ret_pos, + byte *buff __attribute__((unused)), my_bool *last_key) { reg4 int start,mid,end,save_end; int flag; @@ -192,16 +195,16 @@ int _ma_bin_search(MARIA_HA *info, register MARIA_KEYDEF *keyinfo, uchar *page, while (start != end) { mid= (start+end)/2; - if ((flag=ha_key_cmp(keyinfo->seg,page+(uint) mid*totlength,key,key_len, - comp_flag, not_used)) + if ((flag=ha_key_cmp(keyinfo->seg,(uchar*) page+(uint) mid*totlength, + (uchar*) key, key_len, comp_flag, not_used)) >= 0) end=mid; else start=mid+1; } if (mid != start) - flag=ha_key_cmp(keyinfo->seg,page+(uint) start*totlength,key,key_len, - comp_flag, not_used); + flag=ha_key_cmp(keyinfo->seg, (uchar*) page+(uint) start*totlength, + (uchar*) key, key_len, comp_flag, not_used); if (flag < 0) start++; /* point at next, bigger key */ *ret_pos=page+(uint) start*totlength; @@ -237,13 +240,13 @@ int _ma_bin_search(MARIA_HA *info, register MARIA_KEYDEF *keyinfo, uchar *page, < 0 Not found. */ -int _ma_seq_search(MARIA_HA *info, register MARIA_KEYDEF *keyinfo, uchar *page, - uchar *key, uint key_len, uint comp_flag, uchar **ret_pos, - uchar *buff, my_bool *last_key) +int _ma_seq_search(MARIA_HA *info, register MARIA_KEYDEF *keyinfo, byte *page, + byte *key, uint key_len, uint comp_flag, byte **ret_pos, + byte *buff, my_bool *last_key) { int flag; uint nod_flag,length,not_used[2]; - uchar t_buff[HA_MAX_KEY_BUFF],*end; + byte t_buff[HA_MAX_KEY_BUFF],*end; DBUG_ENTER("_ma_seq_search"); LINT_INIT(flag); LINT_INIT(length); @@ -264,8 +267,8 @@ int _ma_seq_search(MARIA_HA *info, register MARIA_KEYDEF *keyinfo, uchar *page, length, (long) page, (long) end)); DBUG_RETURN(MARIA_FOUND_WRONG_KEY); } - if ((flag=ha_key_cmp(keyinfo->seg,t_buff,key,key_len,comp_flag, - not_used)) >= 0) + if ((flag= ha_key_cmp(keyinfo->seg, (uchar*) t_buff,(uchar*) key, + key_len,comp_flag, not_used)) >= 0) break; #ifdef EXTRA_DEBUG DBUG_PRINT("loop",("page: 0x%lx key: '%s' flag: %d", (long) page, t_buff, @@ -282,9 +285,9 @@ int _ma_seq_search(MARIA_HA *info, register MARIA_KEYDEF *keyinfo, uchar *page, } /* _ma_seq_search */ -int _ma_prefix_search(MARIA_HA *info, register MARIA_KEYDEF *keyinfo, uchar *page, - uchar *key, uint key_len, uint nextflag, uchar **ret_pos, - uchar *buff, my_bool *last_key) +int _ma_prefix_search(MARIA_HA *info, register MARIA_KEYDEF *keyinfo, + byte *page, byte *key, uint key_len, uint nextflag, + byte **ret_pos, byte *buff, my_bool *last_key) { /* my_flag is raw comparison result to be changed according to @@ -295,10 +298,11 @@ int _ma_prefix_search(MARIA_HA *info, register MARIA_KEYDEF *keyinfo, uchar *pag uint nod_flag, length, len, matched, cmplen, kseg_len; uint prefix_len,suffix_len; int key_len_skip, seg_len_pack, key_len_left; - uchar *end, *kseg, *vseg; - uchar *sort_order=keyinfo->seg->charset->sort_order; - uchar tt_buff[HA_MAX_KEY_BUFF+2], *t_buff=tt_buff+2; - uchar *saved_from, *saved_to, *saved_vseg; + byte *end; + uchar *kseg, *vseg, *saved_vseg, *saved_from; + uchar *sort_order= keyinfo->seg->charset->sort_order; + byte tt_buff[HA_MAX_KEY_BUFF+2], *t_buff=tt_buff+2; + byte *saved_to; uint saved_length=0, saved_prefix_len=0; uint length_pack; DBUG_ENTER("_ma_prefix_search"); @@ -315,9 +319,9 @@ int _ma_prefix_search(MARIA_HA *info, register MARIA_KEYDEF *keyinfo, uchar *pag nod_flag=_ma_test_if_nod(page); page+=2+nod_flag; *ret_pos=page; - kseg=key; + kseg= (uchar*) key; - get_key_pack_length(kseg_len,length_pack,kseg); + get_key_pack_length(kseg_len, length_pack, kseg); key_len_skip=length_pack+kseg_len; key_len_left=(int) key_len- (int) key_len_skip; /* If key_len is 0, then lenght_pack is 1, then key_len_left is -1. */ @@ -344,7 +348,7 @@ int _ma_prefix_search(MARIA_HA *info, register MARIA_KEYDEF *keyinfo, uchar *pag { uint packed= *page & 128; - vseg=page; + vseg= (uchar*) page; if (keyinfo->seg->length >= 127) { suffix_len=mi_uint2korr(vseg) & 32767; @@ -387,7 +391,7 @@ int _ma_prefix_search(MARIA_HA *info, register MARIA_KEYDEF *keyinfo, uchar *pag DBUG_PRINT("loop",("page: '%.*s%.*s'",prefix_len,t_buff+seg_len_pack, suffix_len,vseg)); { - uchar *from=vseg+suffix_len; + uchar *from= vseg+suffix_len; HA_KEYSEG *keyseg; uint l; @@ -408,9 +412,9 @@ int _ma_prefix_search(MARIA_HA *info, register MARIA_KEYDEF *keyinfo, uchar *pag from+=l; } - from+=keyseg->length; - page=from+nod_flag; - length=from-vseg; + from+= keyseg->length; + page= (byte*) from+nod_flag; + length= (uint) (from-vseg); } if (page > end) @@ -427,7 +431,7 @@ int _ma_prefix_search(MARIA_HA *info, register MARIA_KEYDEF *keyinfo, uchar *pag { /* We have to compare. But we can still skip part of the key */ uint left; - uchar *k=kseg+prefix_len; + uchar *k= kseg+prefix_len; /* If prefix_len > cmplen then we are in the end-space comparison @@ -477,7 +481,7 @@ int _ma_prefix_search(MARIA_HA *info, register MARIA_KEYDEF *keyinfo, uchar *pag for ( ; k < end && *k == ' '; k++) ; if (k == end) goto cmp_rest; /* should never happen */ - if (*k < (uchar) ' ') + if ((uchar) *k < (uchar) ' ') { my_flag= 1; /* Compared string is smaller */ break; @@ -493,11 +497,11 @@ int _ma_prefix_search(MARIA_HA *info, register MARIA_KEYDEF *keyinfo, uchar *pag /* We have to compare k and vseg as if they were space extended */ for (end=vseg + (len-cmplen) ; - vseg < end && *vseg == (uchar) ' '; + vseg < end && *vseg == (byte) ' '; vseg++, matched++) ; DBUG_ASSERT(vseg < end); - if (*vseg > (uchar) ' ') + if ((uchar) *vseg > (uchar) ' ') { my_flag= 1; /* Compared string is smaller */ break; @@ -534,8 +538,8 @@ int _ma_prefix_search(MARIA_HA *info, register MARIA_KEYDEF *keyinfo, uchar *pag /* else (matched < prefix_len) ---> do nothing. */ memcpy(buff,t_buff,saved_length=seg_len_pack+prefix_len); - saved_to=buff+saved_length; - saved_from=saved_vseg; + saved_to= buff+saved_length; + saved_from= saved_vseg; saved_length=length; *ret_pos=page; } @@ -544,12 +548,12 @@ int _ma_prefix_search(MARIA_HA *info, register MARIA_KEYDEF *keyinfo, uchar *pag if (flag == 0) { memcpy(buff,t_buff,saved_length=seg_len_pack+prefix_len); - saved_to=buff+saved_length; - saved_from=saved_vseg; + saved_to= buff+saved_length; + saved_from= saved_vseg; saved_length=length; } if (saved_length) - memcpy(saved_to,saved_from,saved_length); + memcpy(saved_to, (byte*) saved_from, saved_length); *last_key= page == end; @@ -560,7 +564,7 @@ int _ma_prefix_search(MARIA_HA *info, register MARIA_KEYDEF *keyinfo, uchar *pag /* Get pos to a key_block */ -my_off_t _ma_kpos(uint nod_flag, uchar *after_key) +my_off_t _ma_kpos(uint nod_flag, byte *after_key) { after_key-=nod_flag; switch (nod_flag) { @@ -596,7 +600,7 @@ my_off_t _ma_kpos(uint nod_flag, uchar *after_key) /* Save pos to a key_block */ -void _ma_kpointer(register MARIA_HA *info, register uchar *buff, my_off_t pos) +void _ma_kpointer(register MARIA_HA *info, register byte *buff, my_off_t pos) { pos/=MARIA_MIN_KEY_BLOCK_LENGTH; switch (info->s->base.key_reflength) { @@ -615,7 +619,7 @@ void _ma_kpointer(register MARIA_HA *info, register uchar *buff, my_off_t pos) case 4: mi_int4store(buff,pos); break; case 3: mi_int3store(buff,pos); break; case 2: mi_int2store(buff,(uint) pos); break; - case 1: buff[0]= (uchar) pos; break; + case 1: buff[0]= (char) (uchar) pos; break; default: abort(); /* impossible */ } } /* _ma_kpointer */ @@ -624,7 +628,7 @@ void _ma_kpointer(register MARIA_HA *info, register uchar *buff, my_off_t pos) /* Calc pos to a data-record from a key */ -my_off_t _ma_dpos(MARIA_HA *info, uint nod_flag, uchar *after_key) +my_off_t _ma_dpos(MARIA_HA *info, uint nod_flag, const byte *after_key) { my_off_t pos; after_key-=(nod_flag + info->s->rec_reflength); @@ -646,15 +650,14 @@ my_off_t _ma_dpos(MARIA_HA *info, uint nod_flag, uchar *after_key) default: pos=0L; /* Shut compiler up */ } - return (info->s->options & - (HA_OPTION_PACK_RECORD | HA_OPTION_COMPRESS_RECORD)) ? pos : - pos*info->s->base.pack_reclength; + return ((info->s->data_file_type == STATIC_RECORD) ? + pos * info->s->base.pack_reclength : pos); } /* Calc position from a record pointer ( in delete link chain ) */ -my_off_t _ma_rec_pos(MARIA_SHARE *s, uchar *ptr) +my_off_t _ma_rec_pos(MARIA_SHARE *s, byte *ptr) { my_off_t pos; switch (s->rec_reflength) { @@ -704,20 +707,18 @@ my_off_t _ma_rec_pos(MARIA_SHARE *s, uchar *ptr) break; default: abort(); /* Impossible */ } - return ((s->options & - (HA_OPTION_PACK_RECORD | HA_OPTION_COMPRESS_RECORD)) ? pos : - pos*s->base.pack_reclength); + return ((s->data_file_type == STATIC_RECORD) ? + pos * s->base.pack_reclength : pos); } /* save position to record */ -void _ma_dpointer(MARIA_HA *info, uchar *buff, my_off_t pos) +void _ma_dpointer(MARIA_HA *info, byte *buff, my_off_t pos) { - if (!(info->s->options & - (HA_OPTION_PACK_RECORD | HA_OPTION_COMPRESS_RECORD)) && + if (info->s->data_file_type == STATIC_RECORD && pos != HA_OFFSET_ERROR) - pos/=info->s->base.pack_reclength; + pos/= info->s->base.pack_reclength; switch (info->s->rec_reflength) { #if SIZEOF_OFF_T > 4 @@ -752,7 +753,7 @@ void _ma_dpointer(MARIA_HA *info, uchar *buff, my_off_t pos) /* same as _ma_get_key but used with fixed length keys */ uint _ma_get_static_key(register MARIA_KEYDEF *keyinfo, uint nod_flag, - register uchar **page, register uchar *key) + register byte **page, register byte *key) { memcpy((byte*) key,(byte*) *page, (size_t) (keyinfo->keylength+nod_flag)); @@ -776,10 +777,10 @@ uint _ma_get_static_key(register MARIA_KEYDEF *keyinfo, uint nod_flag, */ uint _ma_get_pack_key(register MARIA_KEYDEF *keyinfo, uint nod_flag, - register uchar **page_pos, register uchar *key) + register byte **page_pos, register byte *key) { reg1 HA_KEYSEG *keyseg; - uchar *start_key,*page=*page_pos; + byte *start_key,*page=*page_pos; uint length; start_key=key; @@ -788,7 +789,7 @@ uint _ma_get_pack_key(register MARIA_KEYDEF *keyinfo, uint nod_flag, if (keyseg->flag & HA_PACK_KEY) { /* key with length, packed to previous key */ - uchar *start=key; + byte *start= key; uint packed= *page & 128,tot_length,rest_length; if (keyseg->length >= 127) { @@ -834,7 +835,7 @@ uint _ma_get_pack_key(register MARIA_KEYDEF *keyinfo, uint nod_flag, tot_length=rest_length+length; /* If the stored length has changed, we must move the key */ - if (tot_length >= 255 && *start != 255) + if (tot_length >= 255 && *start != (char) 255) { /* length prefix changed from a length of one to a length of 3 */ bmove_upp((char*) key+length+3,(char*) key+length+1,length); @@ -842,7 +843,7 @@ uint _ma_get_pack_key(register MARIA_KEYDEF *keyinfo, uint nod_flag, mi_int2store(key+1,tot_length); key+=3+length; } - else if (tot_length < 255 && *start == 255) + else if (tot_length < 255 && *start == (char) 255) { bmove(key+1,key+3,length); *key=tot_length; @@ -891,7 +892,7 @@ uint _ma_get_pack_key(register MARIA_KEYDEF *keyinfo, uint nod_flag, if (keyseg->flag & (HA_VAR_LENGTH_PART | HA_BLOB_PART | HA_SPACE_PACK)) { - uchar *tmp=page; + byte *tmp=page; get_key_length(length,tmp); length+=(uint) (tmp-page); } @@ -913,10 +914,10 @@ uint _ma_get_pack_key(register MARIA_KEYDEF *keyinfo, uint nod_flag, /* key that is packed relatively to previous */ uint _ma_get_binary_pack_key(register MARIA_KEYDEF *keyinfo, uint nod_flag, - register uchar **page_pos, register uchar *key) + register byte **page_pos, register byte *key) { reg1 HA_KEYSEG *keyseg; - uchar *start_key,*page,*page_end,*from,*from_end; + byte *start_key,*page,*page_end,*from,*from_end; uint length,tmp; DBUG_ENTER("_ma_get_binary_pack_key"); @@ -1018,8 +1019,8 @@ uint _ma_get_binary_pack_key(register MARIA_KEYDEF *keyinfo, uint nod_flag, /* Get key at position without knowledge of previous key */ /* Returns pointer to next key */ -uchar *_ma_get_key(MARIA_HA *info, MARIA_KEYDEF *keyinfo, uchar *page, - uchar *key, uchar *keypos, uint *return_key_length) +byte *_ma_get_key(MARIA_HA *info, MARIA_KEYDEF *keyinfo, byte *page, + byte *key, byte *keypos, uint *return_key_length) { uint nod_flag; DBUG_ENTER("_ma_get_key"); @@ -1054,8 +1055,8 @@ uchar *_ma_get_key(MARIA_HA *info, MARIA_KEYDEF *keyinfo, uchar *page, /* Get key at position without knowledge of previous key */ /* Returns 0 if ok */ -static my_bool _ma_get_prev_key(MARIA_HA *info, MARIA_KEYDEF *keyinfo, uchar *page, - uchar *key, uchar *keypos, +static my_bool _ma_get_prev_key(MARIA_HA *info, MARIA_KEYDEF *keyinfo, + byte *page, byte *key, byte *keypos, uint *return_key_length) { uint nod_flag; @@ -1092,11 +1093,11 @@ static my_bool _ma_get_prev_key(MARIA_HA *info, MARIA_KEYDEF *keyinfo, uchar *pa /* Get last key from key-page */ /* Return pointer to where key starts */ -uchar *_ma_get_last_key(MARIA_HA *info, MARIA_KEYDEF *keyinfo, uchar *page, - uchar *lastkey, uchar *endpos, uint *return_key_length) +byte *_ma_get_last_key(MARIA_HA *info, MARIA_KEYDEF *keyinfo, byte *page, + byte *lastkey, byte *endpos, uint *return_key_length) { uint nod_flag; - uchar *lastpos; + byte *lastpos; DBUG_ENTER("_ma_get_last_key"); DBUG_PRINT("enter",("page: 0x%lx endpos: 0x%lx", (long) page, (long) endpos)); @@ -1135,15 +1136,15 @@ uchar *_ma_get_last_key(MARIA_HA *info, MARIA_KEYDEF *keyinfo, uchar *page, /* Calculate length of key */ -uint _ma_keylength(MARIA_KEYDEF *keyinfo, register uchar *key) +uint _ma_keylength(MARIA_KEYDEF *keyinfo, register const byte *key) { reg1 HA_KEYSEG *keyseg; - uchar *start; + const byte *start; if (! (keyinfo->flag & (HA_VAR_LENGTH_KEY | HA_BINARY_PACK_KEY))) return (keyinfo->keylength); - start=key; + start= key; for (keyseg=keyinfo->seg ; keyseg->type ; keyseg++) { if (keyseg->flag & HA_NULL_PART) @@ -1170,11 +1171,11 @@ uint _ma_keylength(MARIA_KEYDEF *keyinfo, register uchar *key) after '0xDF' but find 'ss' */ -uint _ma_keylength_part(MARIA_KEYDEF *keyinfo, register uchar *key, +uint _ma_keylength_part(MARIA_KEYDEF *keyinfo, register const byte *key, HA_KEYSEG *end) { reg1 HA_KEYSEG *keyseg; - uchar *start= key; + const byte *start= key; for (keyseg=keyinfo->seg ; keyseg != end ; keyseg++) { @@ -1193,29 +1194,35 @@ uint _ma_keylength_part(MARIA_KEYDEF *keyinfo, register uchar *key, return (uint) (key-start); } - /* Move a key */ -uchar *_ma_move_key(MARIA_KEYDEF *keyinfo, uchar *to, uchar *from) +/* Move a key */ + +byte *_ma_move_key(MARIA_KEYDEF *keyinfo, byte *to, const byte *from) { reg1 uint length; - memcpy((byte*) to, (byte*) from, - (size_t) (length= _ma_keylength(keyinfo,from))); + memcpy(to, from, (size_t) (length= _ma_keylength(keyinfo, from))); return to+length; } - /* Find next/previous record with same key */ - /* This can't be used when database is touched after last read */ + +/* + Find next/previous record with same key + + WARNING + This can't be used when database is touched after last read +*/ int _ma_search_next(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, - uchar *key, uint key_length, uint nextflag, my_off_t pos) + byte *key, uint key_length, uint nextflag, my_off_t pos) { int error; uint nod_flag; - uchar lastkey[HA_MAX_KEY_BUFF]; + byte lastkey[HA_MAX_KEY_BUFF]; DBUG_ENTER("_ma_search_next"); - DBUG_PRINT("enter",("nextflag: %u lastpos: %lu int_keypos: %lu", - nextflag, (ulong) info->lastpos, - (ulong) info->int_keypos)); + DBUG_PRINT("enter",("nextflag: %u lastpos: %lu int_keypos: %lu page_changed %d keybuff_used: %d", + nextflag, (ulong) info->cur_row.lastpos, + (ulong) info->int_keypos, + info->page_changed, info->keybuff_used)); DBUG_EXECUTE("key", _ma_print_key(DBUG_FILE,keyinfo->seg,key,key_length);); /* Force full read if we are at last key or if we are not on a leaf @@ -1228,20 +1235,20 @@ int _ma_search_next(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, if (((nextflag & SEARCH_BIGGER) && info->int_keypos >= info->int_maxpos) || info->page_changed || (info->int_keytree_version != keyinfo->version && - (info->int_nod_flag || info->buff_used))) + (info->int_nod_flag || info->keybuff_used))) DBUG_RETURN(_ma_search(info,keyinfo,key, USE_WHOLE_KEY, nextflag | SEARCH_SAVE_BUFF, pos)); - if (info->buff_used) + if (info->keybuff_used) { if (!_ma_fetch_keypage(info,keyinfo,info->last_search_keypage, - DFLT_INIT_HITS,info->buff,0)) + DFLT_INIT_HITS,info->keyread_buff,0)) DBUG_RETURN(-1); - info->buff_used=0; + info->keybuff_used=0; } - /* Last used buffer is in info->buff */ - nod_flag=_ma_test_if_nod(info->buff); + /* Last used buffer is in info->keyread_buff */ + nod_flag=_ma_test_if_nod(info->keyread_buff); if (nextflag & SEARCH_BIGGER) /* Next key */ { @@ -1261,11 +1268,11 @@ int _ma_search_next(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, { uint length; /* Find start of previous key */ - info->int_keypos= _ma_get_last_key(info,keyinfo,info->buff,lastkey, + info->int_keypos= _ma_get_last_key(info,keyinfo,info->keyread_buff,lastkey, info->int_keypos, &length); if (!info->int_keypos) DBUG_RETURN(-1); - if (info->int_keypos == info->buff+2) + if (info->int_keypos == info->keyread_buff+2) DBUG_RETURN(_ma_search(info,keyinfo,key, USE_WHOLE_KEY, nextflag | SEARCH_SAVE_BUFF, pos)); if ((error= _ma_search(info,keyinfo,key, USE_WHOLE_KEY, @@ -1274,84 +1281,84 @@ int _ma_search_next(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, DBUG_RETURN(error); /* QQ: We should be able to optimize away the following call */ - if (! _ma_get_last_key(info,keyinfo,info->buff,lastkey, + if (! _ma_get_last_key(info,keyinfo,info->keyread_buff,lastkey, info->int_keypos,&info->lastkey_length)) DBUG_RETURN(-1); } memcpy(info->lastkey,lastkey,info->lastkey_length); - info->lastpos= _ma_dpos(info,0,info->lastkey+info->lastkey_length); - DBUG_PRINT("exit",("found key at %lu",(ulong) info->lastpos)); + info->cur_row.lastpos= _ma_dpos(info,0,info->lastkey+info->lastkey_length); + DBUG_PRINT("exit",("found key at %lu",(ulong) info->cur_row.lastpos)); DBUG_RETURN(0); } /* _ma_search_next */ /* Search after position for the first row in an index */ - /* This is stored in info->lastpos */ + /* This is stored in info->cur_row.lastpos */ int _ma_search_first(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, register my_off_t pos) { uint nod_flag; - uchar *page; + byte *page; DBUG_ENTER("_ma_search_first"); if (pos == HA_OFFSET_ERROR) { my_errno=HA_ERR_KEY_NOT_FOUND; - info->lastpos= HA_OFFSET_ERROR; + info->cur_row.lastpos= HA_OFFSET_ERROR; DBUG_RETURN(-1); } do { - if (!_ma_fetch_keypage(info,keyinfo,pos,DFLT_INIT_HITS,info->buff,0)) + if (!_ma_fetch_keypage(info,keyinfo,pos,DFLT_INIT_HITS,info->keyread_buff,0)) { - info->lastpos= HA_OFFSET_ERROR; + info->cur_row.lastpos= HA_OFFSET_ERROR; DBUG_RETURN(-1); } - nod_flag=_ma_test_if_nod(info->buff); - page=info->buff+2+nod_flag; + nod_flag=_ma_test_if_nod(info->keyread_buff); + page=info->keyread_buff+2+nod_flag; } while ((pos= _ma_kpos(nod_flag,page)) != HA_OFFSET_ERROR); if (!(info->lastkey_length=(*keyinfo->get_key)(keyinfo,nod_flag,&page, info->lastkey))) DBUG_RETURN(-1); /* Crashed */ - info->int_keypos=page; info->int_maxpos=info->buff+maria_getint(info->buff)-1; + info->int_keypos=page; info->int_maxpos=info->keyread_buff+maria_getint(info->keyread_buff)-1; info->int_nod_flag=nod_flag; info->int_keytree_version=keyinfo->version; info->last_search_keypage=info->last_keypage; - info->page_changed=info->buff_used=0; - info->lastpos= _ma_dpos(info,0,info->lastkey+info->lastkey_length); + info->page_changed=info->keybuff_used=0; + info->cur_row.lastpos= _ma_dpos(info,0,info->lastkey+info->lastkey_length); - DBUG_PRINT("exit",("found key at %lu", (ulong) info->lastpos)); + DBUG_PRINT("exit",("found key at %lu", (ulong) info->cur_row.lastpos)); DBUG_RETURN(0); } /* _ma_search_first */ /* Search after position for the last row in an index */ - /* This is stored in info->lastpos */ + /* This is stored in info->cur_row.lastpos */ int _ma_search_last(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, register my_off_t pos) { uint nod_flag; - uchar *buff,*page; + byte *buff,*page; DBUG_ENTER("_ma_search_last"); if (pos == HA_OFFSET_ERROR) { my_errno=HA_ERR_KEY_NOT_FOUND; /* Didn't find key */ - info->lastpos= HA_OFFSET_ERROR; + info->cur_row.lastpos= HA_OFFSET_ERROR; DBUG_RETURN(-1); } - buff=info->buff; + buff=info->keyread_buff; do { if (!_ma_fetch_keypage(info,keyinfo,pos,DFLT_INIT_HITS,buff,0)) { - info->lastpos= HA_OFFSET_ERROR; + info->cur_row.lastpos= HA_OFFSET_ERROR; DBUG_RETURN(-1); } page= buff+maria_getint(buff); @@ -1361,14 +1368,14 @@ int _ma_search_last(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, if (!_ma_get_last_key(info,keyinfo,buff,info->lastkey,page, &info->lastkey_length)) DBUG_RETURN(-1); - info->lastpos= _ma_dpos(info,0,info->lastkey+info->lastkey_length); + info->cur_row.lastpos= _ma_dpos(info,0,info->lastkey+info->lastkey_length); info->int_keypos=info->int_maxpos=page; info->int_nod_flag=nod_flag; info->int_keytree_version=keyinfo->version; info->last_search_keypage=info->last_keypage; - info->page_changed=info->buff_used=0; + info->page_changed=info->keybuff_used=0; - DBUG_PRINT("exit",("found key at %lu",(ulong) info->lastpos)); + DBUG_PRINT("exit",("found key at %lu",(ulong) info->cur_row.lastpos)); DBUG_RETURN(0); } /* _ma_search_last */ @@ -1391,12 +1398,12 @@ int _ma_search_last(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, int _ma_calc_static_key_length(MARIA_KEYDEF *keyinfo,uint nod_flag, - uchar *next_pos __attribute__((unused)), - uchar *org_key __attribute__((unused)), - uchar *prev_key __attribute__((unused)), - uchar *key, MARIA_KEY_PARAM *s_temp) + byte *next_pos __attribute__((unused)), + byte *org_key __attribute__((unused)), + byte *prev_key __attribute__((unused)), + const byte *key, MARIA_KEY_PARAM *s_temp) { - s_temp->key=key; + s_temp->key= key; return (int) (s_temp->totlength=keyinfo->keylength+nod_flag); } @@ -1404,18 +1411,18 @@ _ma_calc_static_key_length(MARIA_KEYDEF *keyinfo,uint nod_flag, int _ma_calc_var_key_length(MARIA_KEYDEF *keyinfo,uint nod_flag, - uchar *next_pos __attribute__((unused)), - uchar *org_key __attribute__((unused)), - uchar *prev_key __attribute__((unused)), - uchar *key, MARIA_KEY_PARAM *s_temp) + byte *next_pos __attribute__((unused)), + byte *org_key __attribute__((unused)), + byte *prev_key __attribute__((unused)), + const byte *key, MARIA_KEY_PARAM *s_temp) { - s_temp->key=key; + s_temp->key= key; return (int) (s_temp->totlength= _ma_keylength(keyinfo,key)+nod_flag); } /* length of key with a variable length first segment which is prefix - compressed (mariachk reports 'packed + stripped') + compressed (maria_chk reports 'packed + stripped') Keys are compressed the following way: @@ -1434,15 +1441,16 @@ _ma_calc_var_key_length(MARIA_KEYDEF *keyinfo,uint nod_flag, int _ma_calc_var_pack_key_length(MARIA_KEYDEF *keyinfo, uint nod_flag, - uchar *next_key, - uchar *org_key, uchar *prev_key, uchar *key, + byte *next_key, + byte *org_key, byte *prev_key, const byte *key, MARIA_KEY_PARAM *s_temp) { reg1 HA_KEYSEG *keyseg; int length; uint key_length,ref_length,org_key_length=0, length_pack,new_key_length,diff_flag,pack_marker; - uchar *start,*end,*key_end,*sort_order; + const byte *start,*end,*key_end; + uchar *sort_order; bool same_length; length_pack=s_temp->ref_length=s_temp->n_ref_length=s_temp->n_length=0; @@ -1455,7 +1463,7 @@ _ma_calc_var_pack_key_length(MARIA_KEYDEF *keyinfo, uint nod_flag, (keyseg->type == HA_KEYTYPE_VARTEXT1) || (keyseg->type == HA_KEYTYPE_VARTEXT2)) && !use_strnxfrm(keyseg->charset)) - sort_order=keyseg->charset->sort_order; + sort_order= keyseg->charset->sort_order; /* diff flag contains how many bytes is needed to pack key */ if (keyseg->length >= 127) @@ -1475,10 +1483,10 @@ _ma_calc_var_pack_key_length(MARIA_KEYDEF *keyinfo, uint nod_flag, { if (!*key++) { - s_temp->key=key; - s_temp->key_length=0; - s_temp->totlength=key_length-1+diff_flag; - s_temp->next_key_pos=0; /* No next key */ + s_temp->key= key; + s_temp->key_length= 0; + s_temp->totlength= key_length-1+diff_flag; + s_temp->next_key_pos= 0; /* No next key */ return (s_temp->totlength); } s_temp->store_not_null=1; @@ -1490,13 +1498,13 @@ _ma_calc_var_pack_key_length(MARIA_KEYDEF *keyinfo, uint nod_flag, } else s_temp->store_not_null=0; - s_temp->prev_key=org_key; + s_temp->prev_key= org_key; /* The key part will start with a packed length */ get_key_pack_length(new_key_length,length_pack,key); - end=key_end= key+ new_key_length; - start=key; + end= key_end= key+ new_key_length; + start= key; /* Calc how many characters are identical between this and the prev. key */ if (prev_key) @@ -1507,11 +1515,12 @@ _ma_calc_var_pack_key_length(MARIA_KEYDEF *keyinfo, uint nod_flag, if (new_key_length && new_key_length == org_key_length) same_length=1; else if (new_key_length > org_key_length) - end=key + org_key_length; + end= key + org_key_length; if (sort_order) /* SerG */ { - while (key < end && sort_order[*key] == sort_order[*prev_key]) + while (key < end && + sort_order[* (uchar*) key] == sort_order[* (uchar*) prev_key]) { key++; prev_key++; } @@ -1592,7 +1601,8 @@ _ma_calc_var_pack_key_length(MARIA_KEYDEF *keyinfo, uint nod_flag, key=start; if (sort_order) /* SerG */ { - while (key < end && sort_order[*key] == sort_order[*org_key]) + while (key < end && + sort_order[*(uchar*) key] == sort_order[*(uchar*) org_key]) { key++; org_key++; } @@ -1672,11 +1682,11 @@ _ma_calc_var_pack_key_length(MARIA_KEYDEF *keyinfo, uint nod_flag, uint tmp_length; key=(start+=ref_length); if (key+n_length < key_end) /* Normalize length based */ - key_end=key+n_length; + key_end= key+n_length; if (sort_order) /* SerG */ { - while (key < key_end && sort_order[*key] == - sort_order[*next_key]) + while (key < key_end && + sort_order[*(uchar*) key] == sort_order[*(uchar*) next_key]) { key++; next_key++; } @@ -1716,8 +1726,9 @@ _ma_calc_var_pack_key_length(MARIA_KEYDEF *keyinfo, uint nod_flag, /* Length of key which is prefix compressed */ int _ma_calc_bin_pack_key_length(MARIA_KEYDEF *keyinfo, uint nod_flag, - uchar *next_key, - uchar *org_key, uchar *prev_key, uchar *key, + byte *next_key, + byte *org_key, byte *prev_key, + const byte *key, MARIA_KEY_PARAM *s_temp) { uint length,key_length,ref_length; @@ -1732,10 +1743,10 @@ int _ma_calc_bin_pack_key_length(MARIA_KEYDEF *keyinfo, uint nod_flag, { /* pack key against previous key */ /* - As keys may be identical when running a sort in mariachk, we + As keys may be identical when running a sort in maria_chk, we have to guard against the case where keys may be identical */ - uchar *end; + const byte *end; end=key+key_length; for ( ; *key == *prev_key && key < end; key++,prev_key++) ; s_temp->ref_length= ref_length=(uint) (key-s_temp->key); @@ -1756,7 +1767,7 @@ int _ma_calc_bin_pack_key_length(MARIA_KEYDEF *keyinfo, uint nod_flag, /* If first key and next key is packed (only on delete) */ if (!prev_key && org_key && next_length) { - uchar *end; + const byte *end; for (key= s_temp->key, end=key+next_length ; *key == *org_key && key < end; key++,org_key++) ; @@ -1798,7 +1809,7 @@ int _ma_calc_bin_pack_key_length(MARIA_KEYDEF *keyinfo, uint nod_flag, /* store key without compression */ void _ma_store_static_key(MARIA_KEYDEF *keyinfo __attribute__((unused)), - register uchar *key_pos, + register byte *key_pos, register MARIA_KEY_PARAM *s_temp) { memcpy((byte*) key_pos,(byte*) s_temp->key,(size_t) s_temp->totlength); @@ -1813,11 +1824,11 @@ void _ma_store_static_key(MARIA_KEYDEF *keyinfo __attribute__((unused)), void _ma_store_var_pack_key(MARIA_KEYDEF *keyinfo __attribute__((unused)), - register uchar *key_pos, + register byte *key_pos, register MARIA_KEY_PARAM *s_temp) { uint length; - uchar *start; + byte *start; start=key_pos; @@ -1876,7 +1887,7 @@ void _ma_store_var_pack_key(MARIA_KEYDEF *keyinfo __attribute__((unused)), /* variable length key with prefix compression */ void _ma_store_bin_pack_key(MARIA_KEYDEF *keyinfo __attribute__((unused)), - register uchar *key_pos, + register byte *key_pos, register MARIA_KEY_PARAM *s_temp) { store_key_length_inc(key_pos,s_temp->ref_length); diff --git a/storage/maria/ma_sort.c b/storage/maria/ma_sort.c index 5ae23c37261..3859bd96149 100644 --- a/storage/maria/ma_sort.c +++ b/storage/maria/ma_sort.c @@ -44,31 +44,31 @@ extern void print_error _VARARGS((const char *fmt,...)); /* Functions defined in this file */ static ha_rows NEAR_F find_all_keys(MARIA_SORT_PARAM *info,uint keys, - uchar **sort_keys, + byte **sort_keys, DYNAMIC_ARRAY *buffpek,int *maxbuffer, IO_CACHE *tempfile, IO_CACHE *tempfile_for_exceptions); -static int NEAR_F write_keys(MARIA_SORT_PARAM *info,uchar **sort_keys, +static int NEAR_F write_keys(MARIA_SORT_PARAM *info, byte **sort_keys, uint count, BUFFPEK *buffpek,IO_CACHE *tempfile); -static int NEAR_F write_key(MARIA_SORT_PARAM *info, uchar *key, +static int NEAR_F write_key(MARIA_SORT_PARAM *info, byte *key, IO_CACHE *tempfile); -static int NEAR_F write_index(MARIA_SORT_PARAM *info,uchar * *sort_keys, +static int NEAR_F write_index(MARIA_SORT_PARAM *info, byte **sort_keys, uint count); static int NEAR_F merge_many_buff(MARIA_SORT_PARAM *info,uint keys, - uchar * *sort_keys, + byte **sort_keys, BUFFPEK *buffpek,int *maxbuffer, IO_CACHE *t_file); static uint NEAR_F read_to_buffer(IO_CACHE *fromfile,BUFFPEK *buffpek, uint sort_length); static int NEAR_F merge_buffers(MARIA_SORT_PARAM *info,uint keys, IO_CACHE *from_file, IO_CACHE *to_file, - uchar * *sort_keys, BUFFPEK *lastbuff, + byte **sort_keys, BUFFPEK *lastbuff, BUFFPEK *Fb, BUFFPEK *Tb); -static int NEAR_F merge_index(MARIA_SORT_PARAM *,uint,uchar **,BUFFPEK *, int, +static int NEAR_F merge_index(MARIA_SORT_PARAM *,uint, byte **,BUFFPEK *, int, IO_CACHE *); static int flush_maria_ft_buf(MARIA_SORT_PARAM *info); -static int NEAR_F write_keys_varlen(MARIA_SORT_PARAM *info,uchar **sort_keys, +static int NEAR_F write_keys_varlen(MARIA_SORT_PARAM *info, byte **sort_keys, uint count, BUFFPEK *buffpek, IO_CACHE *tempfile); static uint NEAR_F read_to_buffer_varlen(IO_CACHE *fromfile,BUFFPEK *buffpek, @@ -96,27 +96,27 @@ my_var_write(MARIA_SORT_PARAM *info, IO_CACHE *to_file, byte *bufs); <> 0 Error */ -int _ma_create_index_by_sort(MARIA_SORT_PARAM *info,my_bool no_messages, - ulong sortbuff_size) +int _ma_create_index_by_sort(MARIA_SORT_PARAM *info, my_bool no_messages, + ulong sortbuff_size) { int error,maxbuffer,skr; uint memavl,old_memavl,keys,sort_length; DYNAMIC_ARRAY buffpek; ha_rows records; - uchar **sort_keys; + byte **sort_keys; IO_CACHE tempfile, tempfile_for_exceptions; DBUG_ENTER("_ma_create_index_by_sort"); DBUG_PRINT("enter",("sort_length: %d", info->key_length)); if (info->keyinfo->flag & HA_VAR_LENGTH_KEY) { - info->write_keys=write_keys_varlen; + info->write_keys= write_keys_varlen; info->read_to_buffer=read_to_buffer_varlen; info->write_key=write_merge_key_varlen; } else { - info->write_keys=write_keys; + info->write_keys= write_keys; info->read_to_buffer=read_to_buffer; info->write_key=write_merge_key; } @@ -124,7 +124,7 @@ int _ma_create_index_by_sort(MARIA_SORT_PARAM *info,my_bool no_messages, my_b_clear(&tempfile); my_b_clear(&tempfile_for_exceptions); bzero((char*) &buffpek,sizeof(buffpek)); - sort_keys= (uchar **) NULL; error= 1; + sort_keys= (byte **) NULL; error= 1; maxbuffer=1; memavl=max(sortbuff_size,MIN_SORT_MEMORY); @@ -152,8 +152,8 @@ int _ma_create_index_by_sort(MARIA_SORT_PARAM *info,my_bool no_messages, } while ((maxbuffer= (int) (records/(keys-1)+1)) != skr); - if ((sort_keys=(uchar **)my_malloc(keys*(sort_length+sizeof(char*))+ - HA_FT_MAXBYTELEN, MYF(0)))) + if ((sort_keys=(byte**) my_malloc(keys*(sort_length+sizeof(char*))+ + HA_FT_MAXBYTELEN, MYF(0)))) { if (my_init_dynamic_array(&buffpek, sizeof(BUFFPEK), maxbuffer, maxbuffer/2)) @@ -230,7 +230,7 @@ int _ma_create_index_by_sort(MARIA_SORT_PARAM *info,my_bool no_messages, && !my_b_read(&tempfile_for_exceptions,(byte*)sort_keys, (uint) key_length)) { - if (_ma_ck_write(index,keyno,(uchar*) sort_keys,key_length-ref_length)) + if (_ma_ck_write(index,keyno,(byte*) sort_keys,key_length-ref_length)) goto err; } } @@ -251,7 +251,7 @@ err: /* Search after all keys and place them in a temp. file */ static ha_rows NEAR_F find_all_keys(MARIA_SORT_PARAM *info, uint keys, - uchar **sort_keys, DYNAMIC_ARRAY *buffpek, + byte **sort_keys, DYNAMIC_ARRAY *buffpek, int *maxbuffer, IO_CACHE *tempfile, IO_CACHE *tempfile_for_exceptions) { @@ -260,7 +260,7 @@ static ha_rows NEAR_F find_all_keys(MARIA_SORT_PARAM *info, uint keys, DBUG_ENTER("find_all_keys"); idx=error=0; - sort_keys[0]=(uchar*) (sort_keys+keys); + sort_keys[0]= (byte*) (sort_keys+keys); while (!(error=(*info->key_read)(info,sort_keys[idx]))) { @@ -277,7 +277,7 @@ static ha_rows NEAR_F find_all_keys(MARIA_SORT_PARAM *info, uint keys, tempfile)) DBUG_RETURN(HA_POS_ERROR); /* purecov: inspected */ - sort_keys[0]=(uchar*) (sort_keys+keys); + sort_keys[0]=(byte*) (sort_keys+keys); memcpy(sort_keys[0],sort_keys[idx-1],(size_t) info->key_length); idx=1; } @@ -308,7 +308,7 @@ pthread_handler_t _ma_thr_find_all_keys(void *arg) int error; uint memavl,old_memavl,keys,sort_length; uint idx, maxbuffer; - uchar **sort_keys=0; + byte **sort_keys= 0; LINT_INIT(keys); @@ -336,7 +336,6 @@ pthread_handler_t _ma_thr_find_all_keys(void *arg) my_b_clear(&info->tempfile_for_exceptions); bzero((char*) &info->buffpek,sizeof(info->buffpek)); bzero((char*) &info->unique, sizeof(info->unique)); - sort_keys= (uchar **) NULL; memavl=max(info->sortbuff_size, MIN_SORT_MEMORY); idx= info->sort_info->max_records; @@ -365,15 +364,15 @@ pthread_handler_t _ma_thr_find_all_keys(void *arg) } while ((maxbuffer= (int) (idx/(keys-1)+1)) != skr); } - if ((sort_keys=(uchar **)my_malloc(keys*(sort_length+sizeof(char*))+ - ((info->keyinfo->flag & HA_FULLTEXT) ? - HA_FT_MAXBYTELEN : 0), MYF(0)))) + if ((sort_keys=(byte**) my_malloc(keys*(sort_length+sizeof(char*))+ + ((info->keyinfo->flag & HA_FULLTEXT) ? + HA_FT_MAXBYTELEN : 0), MYF(0)))) { if (my_init_dynamic_array(&info->buffpek, sizeof(BUFFPEK), maxbuffer, maxbuffer/2)) { my_free((gptr) sort_keys,MYF(0)); - sort_keys= (uchar **) NULL; /* for err: label */ + sort_keys= (byte**) NULL; /* for err: label */ } else break; @@ -393,7 +392,7 @@ pthread_handler_t _ma_thr_find_all_keys(void *arg) info->sort_keys=sort_keys; idx=error=0; - sort_keys[0]=(uchar*) (sort_keys+keys); + sort_keys[0]=(byte*) (sort_keys+keys); while (!(error=info->sort_info->got_error) && !(error=(*info->key_read)(info,sort_keys[idx]))) @@ -411,7 +410,7 @@ pthread_handler_t _ma_thr_find_all_keys(void *arg) (BUFFPEK *)alloc_dynamic(&info->buffpek), &info->tempfile)) goto err; - sort_keys[0]=(uchar*) (sort_keys+keys); + sort_keys[0]=(byte*) (sort_keys+keys); memcpy(sort_keys[0],sort_keys[idx-1],(size_t) info->key_length); idx=1; } @@ -422,7 +421,8 @@ pthread_handler_t _ma_thr_find_all_keys(void *arg) if (info->buffpek.elements) { if (info->write_keys(info,sort_keys, idx, - (BUFFPEK *) alloc_dynamic(&info->buffpek), &info->tempfile)) + (BUFFPEK *) alloc_dynamic(&info->buffpek), + &info->tempfile)) goto err; info->keys=(info->buffpek.elements-1)*(keys-1)+idx; } @@ -434,8 +434,7 @@ pthread_handler_t _ma_thr_find_all_keys(void *arg) err: info->sort_info->got_error=1; /* no need to protect this with a mutex */ - if (sort_keys) - my_free((gptr) sort_keys,MYF(0)); + my_free((gptr) sort_keys, MYF(MY_ALLOW_ZERO_PTR)); info->sort_keys=0; delete_dynamic(& info->buffpek); close_cached_file(&info->tempfile); @@ -499,8 +498,7 @@ int _ma_thr_write_keys(MARIA_SORT_PARAM *sort_param) } } my_free((gptr) sinfo->sort_keys,MYF(0)); - my_free(_ma_get_rec_buff_ptr(info, sinfo->rec_buff), - MYF(MY_ALLOW_ZERO_PTR)); + my_free(sinfo->rec_buff, MYF(MY_ALLOW_ZERO_PTR)); sinfo->sort_keys=0; } @@ -548,7 +546,7 @@ int _ma_thr_write_keys(MARIA_SORT_PARAM *sort_param) { if (param->testflag & T_VERBOSE) printf("Key %d - Merging %u keys\n",sinfo->key+1, sinfo->keys); - if (merge_many_buff(sinfo, keys, (uchar **)mergebuf, + if (merge_many_buff(sinfo, keys, (byte **) mergebuf, dynamic_element(&sinfo->buffpek, 0, BUFFPEK *), (int*) &maxbuffer, &sinfo->tempfile)) { @@ -564,7 +562,7 @@ int _ma_thr_write_keys(MARIA_SORT_PARAM *sort_param) } if (param->testflag & T_VERBOSE) printf("Key %d - Last merge and dumping keys\n", sinfo->key+1); - if (merge_index(sinfo, keys, (uchar **)mergebuf, + if (merge_index(sinfo, keys, (byte**) mergebuf, dynamic_element(&sinfo->buffpek,0,BUFFPEK *), maxbuffer,&sinfo->tempfile) || flush_maria_ft_buf(sinfo) || @@ -596,7 +594,7 @@ int _ma_thr_write_keys(MARIA_SORT_PARAM *sort_param) if (key_length > sizeof(maria_ft_buf) || my_b_read(&sinfo->tempfile_for_exceptions, (byte*)maria_ft_buf, (uint)key_length) || - _ma_ck_write(info, sinfo->key, (uchar*)maria_ft_buf, + _ma_ck_write(info, sinfo->key, maria_ft_buf, key_length - info->s->rec_reflength)) got_error=1; } @@ -607,12 +605,14 @@ int _ma_thr_write_keys(MARIA_SORT_PARAM *sort_param) } #endif /* THREAD */ - /* Write all keys in memory to file for later merge */ -static int NEAR_F write_keys(MARIA_SORT_PARAM *info, register uchar **sort_keys, +/* Write all keys in memory to file for later merge */ + +static int NEAR_F write_keys(MARIA_SORT_PARAM *info, + register byte **sort_keys, uint count, BUFFPEK *buffpek, IO_CACHE *tempfile) { - uchar **end; + byte **end; uint sort_length=info->key_length; DBUG_ENTER("write_keys"); @@ -628,7 +628,7 @@ static int NEAR_F write_keys(MARIA_SORT_PARAM *info, register uchar **sort_keys, for (end=sort_keys+count ; sort_keys != end ; sort_keys++) { - if (my_b_write(tempfile,(byte*) *sort_keys,(uint) sort_length)) + if (my_b_write(tempfile, *sort_keys, (uint) sort_length)) DBUG_RETURN(1); /* purecov: inspected */ } DBUG_RETURN(0); @@ -639,7 +639,7 @@ static inline int my_var_write(MARIA_SORT_PARAM *info, IO_CACHE *to_file, byte *bufs) { int err; - uint16 len = _ma_keylength(info->keyinfo, (uchar*) bufs); + uint16 len= _ma_keylength(info->keyinfo, bufs); /* The following is safe as this is a local file */ if ((err= my_b_write(to_file, (byte*)&len, sizeof(len)))) @@ -651,11 +651,11 @@ my_var_write(MARIA_SORT_PARAM *info, IO_CACHE *to_file, byte *bufs) static int NEAR_F write_keys_varlen(MARIA_SORT_PARAM *info, - register uchar **sort_keys, + register byte **sort_keys, uint count, BUFFPEK *buffpek, IO_CACHE *tempfile) { - uchar **end; + byte **end; int err; DBUG_ENTER("write_keys_varlen"); @@ -670,14 +670,14 @@ static int NEAR_F write_keys_varlen(MARIA_SORT_PARAM *info, buffpek->count=count; for (end=sort_keys+count ; sort_keys != end ; sort_keys++) { - if ((err= my_var_write(info,tempfile, (byte*) *sort_keys))) + if ((err= my_var_write(info,tempfile, *sort_keys))) DBUG_RETURN(err); } DBUG_RETURN(0); } /* write_keys_varlen */ -static int NEAR_F write_key(MARIA_SORT_PARAM *info, uchar *key, +static int NEAR_F write_key(MARIA_SORT_PARAM *info, byte *key, IO_CACHE *tempfile) { uint key_length=info->real_key_length; @@ -688,8 +688,8 @@ static int NEAR_F write_key(MARIA_SORT_PARAM *info, uchar *key, DISK_BUFFER_SIZE, info->sort_info->param->myf_rw)) DBUG_RETURN(1); - if (my_b_write(tempfile,(byte*)&key_length,sizeof(key_length)) || - my_b_write(tempfile,(byte*)key,(uint) key_length)) + if (my_b_write(tempfile, (byte*)&key_length,sizeof(key_length)) || + my_b_write(tempfile, key, (uint) key_length)) DBUG_RETURN(1); DBUG_RETURN(0); } /* write_key */ @@ -697,7 +697,8 @@ static int NEAR_F write_key(MARIA_SORT_PARAM *info, uchar *key, /* Write index */ -static int NEAR_F write_index(MARIA_SORT_PARAM *info, register uchar **sort_keys, +static int NEAR_F write_index(MARIA_SORT_PARAM *info, + register byte **sort_keys, register uint count) { DBUG_ENTER("write_index"); @@ -706,7 +707,7 @@ static int NEAR_F write_index(MARIA_SORT_PARAM *info, register uchar **sort_keys (qsort2_cmp) info->key_cmp,info); while (count--) { - if ((*info->key_write)(info,*sort_keys++)) + if ((*info->key_write)(info, *sort_keys++)) DBUG_RETURN(-1); /* purecov: inspected */ } DBUG_RETURN(0); @@ -716,7 +717,7 @@ static int NEAR_F write_index(MARIA_SORT_PARAM *info, register uchar **sort_keys /* Merge buffers to make < MERGEBUFF2 buffers */ static int NEAR_F merge_many_buff(MARIA_SORT_PARAM *info, uint keys, - uchar **sort_keys, BUFFPEK *buffpek, + byte **sort_keys, BUFFPEK *buffpek, int *maxbuffer, IO_CACHE *t_file) { register int i; @@ -797,11 +798,11 @@ static uint NEAR_F read_to_buffer_varlen(IO_CACHE *fromfile, BUFFPEK *buffpek, register uint count; uint16 length_of_key = 0; uint idx; - uchar *buffp; + byte *buffp; if ((count=(uint) min((ha_rows) buffpek->max_keys,buffpek->count))) { - buffp = buffpek->base; + buffp= buffpek->base; for (idx=1;idx<=count;idx++) { @@ -855,18 +856,17 @@ static int NEAR_F write_merge_key(MARIA_SORT_PARAM *info __attribute__((unused)) static int NEAR_F merge_buffers(MARIA_SORT_PARAM *info, uint keys, IO_CACHE *from_file, - IO_CACHE *to_file, uchar **sort_keys, BUFFPEK *lastbuff, + IO_CACHE *to_file, byte **sort_keys, BUFFPEK *lastbuff, BUFFPEK *Fb, BUFFPEK *Tb) { int error; uint sort_length,maxcount; ha_rows count; my_off_t to_start_filepos; - uchar *strpos; + byte *strpos; BUFFPEK *buffpek,**refpek; QUEUE queue; volatile int *killed= _ma_killed_ptr(info->sort_info->param); - DBUG_ENTER("merge_buffers"); count=error=0; @@ -874,7 +874,7 @@ merge_buffers(MARIA_SORT_PARAM *info, uint keys, IO_CACHE *from_file, LINT_INIT(to_start_filepos); if (to_file) to_start_filepos=my_b_tell(to_file); - strpos=(uchar*) sort_keys; + strpos= (byte*) sort_keys; sort_length=info->key_length; if (init_queue(&queue,(uint) (Tb-Fb)+1,offsetof(BUFFPEK,key),0, @@ -923,7 +923,7 @@ merge_buffers(MARIA_SORT_PARAM *info, uint keys, IO_CACHE *from_file, { if (!(error=(int) info->read_to_buffer(from_file,buffpek,sort_length))) { - uchar *base=buffpek->base; + byte *base= buffpek->base; uint max_keys=buffpek->max_keys; VOID(queue_remove(&queue,0)); @@ -955,7 +955,7 @@ merge_buffers(MARIA_SORT_PARAM *info, uint keys, IO_CACHE *from_file, } } buffpek=(BUFFPEK*) queue_top(&queue); - buffpek->base=(uchar *) sort_keys; + buffpek->base= (byte*) sort_keys; buffpek->max_keys=keys; do { @@ -969,21 +969,21 @@ merge_buffers(MARIA_SORT_PARAM *info, uint keys, IO_CACHE *from_file, } else { - register uchar *end; + register byte *end; strpos= buffpek->key; - for (end=strpos+buffpek->mem_count*sort_length; + for (end= strpos+buffpek->mem_count*sort_length; strpos != end ; strpos+=sort_length) { - if ((*info->key_write)(info,(void*) strpos)) + if ((*info->key_write)(info, (byte*) strpos)) { error=1; goto err; /* purecov: inspected */ } } } } - while ((error=(int) info->read_to_buffer(from_file,buffpek,sort_length)) != -1 && - error != 0); + while ((error=(int) info->read_to_buffer(from_file,buffpek,sort_length)) != + -1 && error != 0); lastbuff->count=count; if (to_file) @@ -997,7 +997,7 @@ err: /* Do a merge to output-file (save only positions) */ static int NEAR_F -merge_index(MARIA_SORT_PARAM *info, uint keys, uchar **sort_keys, +merge_index(MARIA_SORT_PARAM *info, uint keys, byte **sort_keys, BUFFPEK *buffpek, int maxbuffer, IO_CACHE *tempfile) { DBUG_ENTER("merge_index"); @@ -1007,8 +1007,8 @@ merge_index(MARIA_SORT_PARAM *info, uint keys, uchar **sort_keys, DBUG_RETURN(0); } /* merge_index */ -static int -flush_maria_ft_buf(MARIA_SORT_PARAM *info) + +static int flush_maria_ft_buf(MARIA_SORT_PARAM *info) { int err=0; if (info->sort_info->ft_buf) diff --git a/storage/maria/ma_sp_defs.h b/storage/maria/ma_sp_defs.h index a7e282f0ddc..8b9dd204ded 100644 --- a/storage/maria/ma_sp_defs.h +++ b/storage/maria/ma_sp_defs.h @@ -41,7 +41,7 @@ enum wkbByteOrder wkbNDR = 1 /* Little Endian */ }; -uint sp_make_key(register MARIA_HA *info, uint keynr, uchar *key, +uint sp_make_key(register MARIA_HA *info, uint keynr, byte *key, const byte *record, my_off_t filepos); #endif /*HAVE_SPATIAL*/ diff --git a/storage/maria/ma_sp_key.c b/storage/maria/ma_sp_key.c index b9841fed1e7..79345550dd9 100644 --- a/storage/maria/ma_sp_key.c +++ b/storage/maria/ma_sp_key.c @@ -37,7 +37,7 @@ static void get_double(double *d, const byte *pos) float8get(*d, pos); } -uint sp_make_key(register MARIA_HA *info, uint keynr, uchar *key, +uint sp_make_key(register MARIA_HA *info, uint keynr, byte *key, const byte *record, my_off_t filepos) { HA_KEYSEG *keyseg; diff --git a/storage/maria/ma_sp_test.c b/storage/maria/ma_sp_test.c index ea812974c8c..1ac1a74d7d7 100644 --- a/storage/maria/ma_sp_test.c +++ b/storage/maria/ma_sp_test.c @@ -109,10 +109,11 @@ int run_test(const char *filename) create_info.max_rows=10000000; if (maria_create(filename, - 1, /* keys */ - keyinfo, - 2, /* columns */ - recinfo,uniques,&uniquedef,&create_info,create_flag)) + DYNAMIC_RECORD, + 1, /* keys */ + keyinfo, + 2, /* columns */ + recinfo,uniques,&uniquedef,&create_info,create_flag)) goto err; if (!silent) diff --git a/storage/maria/ma_static.c b/storage/maria/ma_static.c index 511c5507aaf..c5580e1e981 100644 --- a/storage/maria/ma_static.c +++ b/storage/maria/ma_static.c @@ -25,12 +25,13 @@ LIST *maria_open_list=0; uchar NEAR maria_file_magic[]= -{ (uchar) 254, (uchar) 254,'\007', '\001', }; +{ (uchar) 254, (uchar) 254, (uchar) 9, '\001', }; uchar NEAR maria_pack_file_magic[]= -{ (uchar) 254, (uchar) 254,'\010', '\002', }; +{ (uchar) 254, (uchar) 254, (uchar) 10, '\001', }; uint maria_quick_table_bits=9; ulong maria_block_size= MARIA_KEY_BLOCK_LENGTH; -my_bool maria_flush=0, maria_delay_key_write=0, maria_single_user=0; +my_bool maria_flush= 0, maria_single_user= 0; +my_bool maria_delay_key_write= 0; #if defined(THREAD) && !defined(DONT_USE_RW_LOCKS) ulong maria_concurrent_insert= 2; #else @@ -38,11 +39,14 @@ ulong maria_concurrent_insert= 0; #endif my_off_t maria_max_temp_length= MAX_FILE_SIZE; ulong maria_bulk_insert_tree_size=8192*1024; -ulong maria_data_pointer_size=4; +ulong maria_data_pointer_size= 4; KEY_CACHE maria_key_cache_var; KEY_CACHE *maria_key_cache= &maria_key_cache_var; +/* Enough for comparing if number is zero */ +byte maria_zero_string[]= {0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0}; + /* read_vec[] is used for converting between P_READ_KEY.. and SEARCH_ Position is , == , >= , <= , > , < diff --git a/storage/maria/ma_statrec.c b/storage/maria/ma_statrec.c index 0aef24f40a9..11049d6f279 100644 --- a/storage/maria/ma_statrec.c +++ b/storage/maria/ma_statrec.c @@ -19,9 +19,9 @@ #include "maria_def.h" -int _ma_write_static_record(MARIA_HA *info, const byte *record) +my_bool _ma_write_static_record(MARIA_HA *info, const byte *record) { - uchar temp[8]; /* max pointer length */ + byte temp[8]; /* max pointer length */ if (info->s->state.dellink != HA_OFFSET_ERROR && !info->append_insert_at_end) { @@ -86,7 +86,8 @@ int _ma_write_static_record(MARIA_HA *info, const byte *record) return 1; } -int _ma_update_static_record(MARIA_HA *info, my_off_t pos, const byte *record) +my_bool _ma_update_static_record(MARIA_HA *info, MARIA_RECORD_POS pos, + const byte *record) { info->rec_cache.seek_not_done=1; /* We have done a seek */ return (info->s->file_write(info, @@ -96,22 +97,22 @@ int _ma_update_static_record(MARIA_HA *info, my_off_t pos, const byte *record) } -int _ma_delete_static_record(MARIA_HA *info) +my_bool _ma_delete_static_record(MARIA_HA *info) { - uchar temp[9]; /* 1+sizeof(uint32) */ - + byte temp[9]; /* 1+sizeof(uint32) */ info->state->del++; info->state->empty+=info->s->base.pack_reclength; temp[0]= '\0'; /* Mark that record is deleted */ _ma_dpointer(info,temp+1,info->s->state.dellink); - info->s->state.dellink = info->lastpos; + info->s->state.dellink= info->cur_row.lastpos; info->rec_cache.seek_not_done=1; - return (info->s->file_write(info,(byte*) temp, 1+info->s->rec_reflength, - info->lastpos, MYF(MY_NABP)) != 0); + return (info->s->file_write(info, temp, 1+info->s->rec_reflength, + info->cur_row.lastpos, MYF(MY_NABP)) != 0); } -int _ma_cmp_static_record(register MARIA_HA *info, register const byte *old) +my_bool _ma_cmp_static_record(register MARIA_HA *info, + register const byte *old) { DBUG_ENTER("_ma_cmp_static_record"); @@ -122,7 +123,7 @@ int _ma_cmp_static_record(register MARIA_HA *info, register const byte *old) { if (flush_io_cache(&info->rec_cache)) { - DBUG_RETURN(-1); + DBUG_RETURN(1); } info->rec_cache.seek_not_done=1; /* We have done a seek */ } @@ -130,10 +131,11 @@ int _ma_cmp_static_record(register MARIA_HA *info, register const byte *old) if ((info->opt_flag & READ_CHECK_USED)) { /* If check isn't disabled */ info->rec_cache.seek_not_done=1; /* We have done a seek */ - if (info->s->file_read(info, (char*) info->rec_buff, info->s->base.reclength, - info->lastpos, - MYF(MY_NABP))) - DBUG_RETURN(-1); + if (info->s->file_read(info, (char*) info->rec_buff, + info->s->base.reclength, + info->cur_row.lastpos, + MYF(MY_NABP))) + DBUG_RETURN(1); if (memcmp((byte*) info->rec_buff, (byte*) old, (uint) info->s->base.reclength)) { @@ -147,27 +149,31 @@ int _ma_cmp_static_record(register MARIA_HA *info, register const byte *old) } -int _ma_cmp_static_unique(MARIA_HA *info, MARIA_UNIQUEDEF *def, - const byte *record, my_off_t pos) +my_bool _ma_cmp_static_unique(MARIA_HA *info, MARIA_UNIQUEDEF *def, + const byte *record, MARIA_RECORD_POS pos) { DBUG_ENTER("_ma_cmp_static_unique"); info->rec_cache.seek_not_done=1; /* We have done a seek */ if (info->s->file_read(info, (char*) info->rec_buff, info->s->base.reclength, pos, MYF(MY_NABP))) - DBUG_RETURN(-1); - DBUG_RETURN(_ma_unique_comp(def, record, info->rec_buff, - def->null_are_equal)); + DBUG_RETURN(1); + DBUG_RETURN(_ma_unique_comp(def, record, (byte*) info->rec_buff, + def->null_are_equal)); } - /* Read a fixed-length-record */ - /* Returns 0 if Ok. */ - /* 1 if record is deleted */ - /* MY_FILE_ERROR on read-error or locking-error */ +/* + Read a fixed-length-record + + RETURN + 0 Ok + 1 record delete + -1 on read-error or locking-error +*/ -int _ma_read_static_record(register MARIA_HA *info, register my_off_t pos, - register byte *record) +int _ma_read_static_record(register MARIA_HA *info, register byte *record, + MARIA_RECORD_POS pos) { int error; @@ -180,7 +186,7 @@ int _ma_read_static_record(register MARIA_HA *info, register my_off_t pos, info->rec_cache.seek_not_done=1; /* We have done a seek */ error=info->s->file_read(info,(char*) record,info->s->base.reclength, - pos,MYF(MY_NABP)) != 0; + pos, MYF(MY_NABP)) != 0; fast_ma_writeinfo(info); if (! error) { @@ -201,8 +207,8 @@ int _ma_read_static_record(register MARIA_HA *info, register my_off_t pos, int _ma_read_rnd_static_record(MARIA_HA *info, byte *buf, - register my_off_t filepos, - my_bool skip_deleted_blocks) + MARIA_RECORD_POS filepos, + my_bool skip_deleted_blocks) { int locked,error,cache_read; uint cache_length; @@ -211,10 +217,6 @@ int _ma_read_rnd_static_record(MARIA_HA *info, byte *buf, cache_read=0; cache_length=0; - if (info->opt_flag & WRITE_CACHE_USED && - (info->rec_cache.pos_in_file <= filepos || skip_deleted_blocks) && - flush_io_cache(&info->rec_cache)) - DBUG_RETURN(my_errno); if (info->opt_flag & READ_CACHE_USED) { /* Cache in use */ if (filepos == my_b_tell(&info->rec_cache) && @@ -256,12 +258,12 @@ int _ma_read_rnd_static_record(MARIA_HA *info, byte *buf, fast_ma_writeinfo(info); DBUG_RETURN(my_errno=HA_ERR_END_OF_FILE); } - info->lastpos= filepos; - info->nextpos= filepos+share->base.pack_reclength; + info->cur_row.lastpos= filepos; + info->cur_row.nextpos= filepos+share->base.pack_reclength; if (! cache_read) /* No cacheing */ { - if ((error= _ma_read_static_record(info,filepos,buf))) + if ((error= _ma_read_static_record(info, buf, filepos))) { if (error > 0) error=my_errno=HA_ERR_RECORD_DELETED; diff --git a/storage/maria/ma_test1.c b/storage/maria/ma_test1.c index 69d432a5d95..0f37391c1d4 100644 --- a/storage/maria/ma_test1.c +++ b/storage/maria/ma_test1.c @@ -28,12 +28,13 @@ static int rec_pointer_size=0, flags[50]; static int key_field=FIELD_SKIP_PRESPACE,extra_field=FIELD_SKIP_ENDSPACE; static int key_type=HA_KEYTYPE_NUM; static int create_flag=0; +static enum data_file_type record_type= DYNAMIC_RECORD; static uint insert_count, update_count, remove_count; static uint pack_keys=0, pack_seg=0, key_length; static uint unique_key=HA_NOSAME; static my_bool key_cacheing, null_fields, silent, skip_update, opt_unique, - verbose; + verbose, skip_delete; static MARIA_COLUMNDEF recinfo[4]; static MARIA_KEYDEF keyinfo[10]; static HA_KEYSEG keyseg[10]; @@ -63,31 +64,30 @@ static int run_test(const char *filename) MARIA_HA *file; int i,j,error,deleted,rec_length,uniques=0; ha_rows found,row_count; - my_off_t pos; char record[MAX_REC_LENGTH],key[MAX_REC_LENGTH],read_record[MAX_REC_LENGTH]; MARIA_UNIQUEDEF uniquedef; MARIA_CREATE_INFO create_info; bzero((char*) recinfo,sizeof(recinfo)); + bzero((char*) &create_info,sizeof(create_info)); /* First define 2 columns */ - recinfo[0].type=FIELD_NORMAL; recinfo[0].length=1; /* For NULL bits */ - recinfo[1].type=key_field; - recinfo[1].length= (key_field == FIELD_BLOB ? 4+maria_portable_sizeof_char_ptr : + create_info.null_bytes= 1; + recinfo[0].type= key_field; + recinfo[0].length= (key_field == FIELD_BLOB ? 4+maria_portable_sizeof_char_ptr : key_length); if (key_field == FIELD_VARCHAR) - recinfo[1].length+= HA_VARCHAR_PACKLENGTH(key_length);; - recinfo[2].type=extra_field; - recinfo[2].length= (extra_field == FIELD_BLOB ? 4 + maria_portable_sizeof_char_ptr : 24); + recinfo[0].length+= HA_VARCHAR_PACKLENGTH(key_length); + recinfo[1].type=extra_field; + recinfo[1].length= (extra_field == FIELD_BLOB ? 4 + maria_portable_sizeof_char_ptr : 24); if (extra_field == FIELD_VARCHAR) - recinfo[2].length+= HA_VARCHAR_PACKLENGTH(recinfo[2].length); + recinfo[1].length+= HA_VARCHAR_PACKLENGTH(recinfo[1].length); if (opt_unique) { - recinfo[3].type=FIELD_CHECK; - recinfo[3].length=MARIA_UNIQUE_HASH_LENGTH; + recinfo[2].type=FIELD_CHECK; + recinfo[2].length=MARIA_UNIQUE_HASH_LENGTH; } - rec_length=recinfo[0].length+recinfo[1].length+recinfo[2].length+ - recinfo[3].length; + rec_length= recinfo[0].length+recinfo[1].length+recinfo[2].length; if (key_type == HA_KEYTYPE_VARTEXT1 && key_length > 255) @@ -125,8 +125,8 @@ static int run_test(const char *filename) for (i=0, start=1 ; i < 2 ; i++) { uniqueseg[i].start=start; - start+=recinfo[i+1].length; - uniqueseg[i].length=recinfo[i+1].length; + start+=recinfo[i].length; + uniqueseg[i].length=recinfo[i].length; uniqueseg[i].language= default_charset_info->number; } uniqueseg[0].type= key_type; @@ -139,18 +139,21 @@ static int run_test(const char *filename) uniqueseg[1].flag|= HA_BLOB_PART; } else if (extra_field == FIELD_VARCHAR) + { uniqueseg[1].flag|= HA_VAR_LENGTH_PART; + uniqueseg[1].type= (HA_VARCHAR_PACKLENGTH(recinfo[1].length-1) == 1 ? + HA_KEYTYPE_VARTEXT1 : HA_KEYTYPE_VARTEXT2); + } } else uniques=0; if (!silent) - printf("- Creating isam-file\n"); - bzero((char*) &create_info,sizeof(create_info)); + printf("- Creating maria file\n"); create_info.max_rows=(ulong) (rec_pointer_size ? (1L << (rec_pointer_size*8))/40 : 0); - if (maria_create(filename,1,keyinfo,3+opt_unique,recinfo, + if (maria_create(filename, record_type, 1, keyinfo,2+opt_unique,recinfo, uniques, &uniquedef, &create_info, create_flag)) goto err; @@ -223,9 +226,10 @@ static int run_test(const char *filename) } /* Read through all rows and update them */ - pos=(my_off_t) 0; + assert(maria_scan_init(file) == 0); + found=0; - while ((error=maria_rrnd(file,read_record,pos)) == 0) + while ((error= maria_scan(file,read_record)) == 0) { if (update_count-- == 0) { VOID(maria_close(file)) ; exit(0) ; } memcpy(record,read_record,rec_length); @@ -236,17 +240,17 @@ static int run_test(const char *filename) keyinfo[0].seg[0].length,record+1,my_errno); } found++; - pos=HA_OFFSET_ERROR; } if (found != row_count) printf("Found %ld of %ld rows\n", (ulong) found, (ulong) row_count); + maria_scan_end(file); } if (!silent) printf("- Reopening file\n"); if (maria_close(file)) goto err; if (!(file=maria_open(filename,2,HA_OPEN_ABORT_IF_LOCKED))) goto err; - if (!skip_update) + if (!skip_delete) { if (!silent) printf("- Removing keys\n"); @@ -254,7 +258,13 @@ static int run_test(const char *filename) for (i=0 ; i <= 10 ; i++) { /* testing */ - if (remove_count-- == 0) { VOID(maria_close(file)) ; exit(0) ; } + if (remove_count-- == 0) + { + fprintf(stderr, + "delete-rows number of rows deleted; Going down hard!\n"); + VOID(maria_close(file)); + exit(0) ; + } j=i*2; if (!flags[j]) continue; @@ -283,6 +293,7 @@ static int run_test(const char *filename) } if (!silent) printf("- Reading rows with key\n"); + record[1]= 0; /* For nicer printf */ for (i=0 ; i <= 25 ; i++) { create_key(key,i); @@ -299,10 +310,16 @@ static int run_test(const char *filename) if (!silent) printf("- Reading rows with position\n"); + if (maria_scan_init(file)) + { + fprintf(stderr, "maria_scan_init failed\n"); + goto err; + } + for (i=1,found=0 ; i <= 30 ; i++) { my_errno=0; - if ((error=maria_rrnd(file,read_record,i == 1 ? 0L : HA_OFFSET_ERROR)) == -1) + if ((error= maria_scan(file, read_record)) == HA_ERR_END_OF_FILE) { if (found != row_count-deleted) printf("Found only %ld of %ld rows\n", (ulong) found, @@ -318,7 +335,8 @@ static int run_test(const char *filename) i-1,error,my_errno,read_record+1); } } - if (maria_close(file)) goto err; + if (maria_close(file)) + goto err; maria_end(); my_end(MY_CHECK_ERROR); @@ -346,7 +364,7 @@ static void create_key_part(char *key,uint rownr) if ((rownr & 7) == 0) { /* Change the key to force a unpack of the next key */ - bfill(key+3,keyinfo[0].seg[0].length-4,rownr < 10 ? 'a' : 'b'); + bfill(key+3,keyinfo[0].seg[0].length-5,rownr < 10 ? 'a' : 'b'); } } else @@ -375,7 +393,7 @@ static void create_key(char *key,uint rownr) if (rownr == 0) { key[0]=1; /* null key */ - key[1]=0; /* Fore easy print of key */ + key[1]=0; /* For easy print of key */ return; } *key++=0; @@ -405,7 +423,7 @@ static void create_record(char *record,uint rownr) record[0]|=keyinfo[0].seg[0].null_bit; /* Null key */ pos=record+1; - if (recinfo[1].type == FIELD_BLOB) + if (recinfo[0].type == FIELD_BLOB) { uint tmp; char *ptr; @@ -414,25 +432,25 @@ static void create_record(char *record,uint rownr) int4store(pos,tmp); ptr=blob_key; memcpy_fixed(pos+4,&ptr,sizeof(char*)); - pos+=recinfo[1].length; + pos+=recinfo[0].length; } - else if (recinfo[1].type == FIELD_VARCHAR) + else if (recinfo[0].type == FIELD_VARCHAR) { - uint tmp, pack_length= HA_VARCHAR_PACKLENGTH(recinfo[1].length-1); + uint tmp, pack_length= HA_VARCHAR_PACKLENGTH(recinfo[0].length-1); create_key_part(pos+pack_length,rownr); tmp= strlen(pos+pack_length); if (pack_length == 1) *(uchar*) pos= (uchar) tmp; else int2store(pos,tmp); - pos+= recinfo[1].length; + pos+= recinfo[0].length; } else { create_key_part(pos,rownr); - pos+=recinfo[1].length; + pos+=recinfo[0].length; } - if (recinfo[2].type == FIELD_BLOB) + if (recinfo[1].type == FIELD_BLOB) { uint tmp; char *ptr;; @@ -443,7 +461,7 @@ static void create_record(char *record,uint rownr) ptr=blob_record; memcpy_fixed(pos+4,&ptr,sizeof(char*)); } - else if (recinfo[2].type == FIELD_VARCHAR) + else if (recinfo[1].type == FIELD_VARCHAR) { uint tmp, pack_length= HA_VARCHAR_PACKLENGTH(recinfo[1].length-1); sprintf(pos+pack_length, "... row: %d", rownr); @@ -456,7 +474,7 @@ static void create_record(char *record,uint rownr) else { sprintf(pos,"... row: %d", rownr); - strappend(pos,recinfo[2].length,' '); + strappend(pos,recinfo[1].length,' '); } } @@ -465,7 +483,7 @@ static void create_record(char *record,uint rownr) static void update_record(char *record) { char *pos=record+1; - if (recinfo[1].type == FIELD_BLOB) + if (recinfo[0].type == FIELD_BLOB) { char *column,*ptr; int length; @@ -477,16 +495,16 @@ static void update_record(char *record) if (keyinfo[0].seg[0].type != HA_KEYTYPE_NUM) default_charset_info->cset->casedn(default_charset_info, blob_key, length, blob_key, length); - pos+=recinfo[1].length; + pos+=recinfo[0].length; } - else if (recinfo[1].type == FIELD_VARCHAR) + else if (recinfo[0].type == FIELD_VARCHAR) { - uint pack_length= HA_VARCHAR_PACKLENGTH(recinfo[1].length-1); + uint pack_length= HA_VARCHAR_PACKLENGTH(recinfo[0].length-1); uint length= pack_length == 1 ? (uint) *(uchar*) pos : uint2korr(pos); default_charset_info->cset->casedn(default_charset_info, pos + pack_length, length, pos + pack_length, length); - pos+=recinfo[1].length; + pos+=recinfo[0].length; } else { @@ -494,10 +512,10 @@ static void update_record(char *record) default_charset_info->cset->casedn(default_charset_info, pos, keyinfo[0].seg[0].length, pos, keyinfo[0].seg[0].length); - pos+=recinfo[1].length; + pos+=recinfo[0].length; } - if (recinfo[2].type == FIELD_BLOB) + if (recinfo[1].type == FIELD_BLOB) { char *column; int length; @@ -510,13 +528,14 @@ static void update_record(char *record) column=blob_record; memcpy_fixed(pos+4,&column,sizeof(char*)); } - else if (recinfo[2].type == FIELD_VARCHAR) + else if (recinfo[1].type == FIELD_VARCHAR) { /* Second field is longer than 10 characters */ uint pack_length= HA_VARCHAR_PACKLENGTH(recinfo[1].length-1); uint length= pack_length == 1 ? (uint) *(uchar*) pos : uint2korr(pos); - bfill(pos+pack_length+length,recinfo[2].length-length-pack_length,'.'); - length=recinfo[2].length-pack_length; + pos= record+ recinfo[1].offset; + bfill(pos+pack_length+length,recinfo[1].length-length-pack_length,'.'); + length=recinfo[1].length-pack_length; if (pack_length == 1) *(uchar*) pos= (uchar) length; else @@ -524,7 +543,7 @@ static void update_record(char *record) } else { - bfill(pos+recinfo[2].length-10,10,'.'); + bfill(pos+recinfo[1].length-10,10,'.'); } } @@ -537,44 +556,49 @@ static struct my_option my_long_options[] = {"debug", '#', "Undocumented", 0, 0, 0, GET_STR, REQUIRED_ARG, 0, 0, 0, 0, 0, 0}, #endif - {"delete_rows", 'd', "Undocumented", (gptr*) &remove_count, - (gptr*) &remove_count, 0, GET_UINT, REQUIRED_ARG, 1000, 0, 0, 0, 0, 0}, + {"delete-rows", 'd', "Abort after this many rows has been deleted", + (gptr*) &remove_count, (gptr*) &remove_count, 0, GET_UINT, REQUIRED_ARG, + 1000, 0, 0, 0, 0, 0}, {"help", '?', "Display help and exit", 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, - {"insert_rows", 'i', "Undocumented", (gptr*) &insert_count, + {"insert-rows", 'i', "Undocumented", (gptr*) &insert_count, (gptr*) &insert_count, 0, GET_UINT, REQUIRED_ARG, 1000, 0, 0, 0, 0, 0}, - {"key_alpha", 'a', "Use a key of type HA_KEYTYPE_TEXT", + {"key-alpha", 'a', "Use a key of type HA_KEYTYPE_TEXT", 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, - {"key_binary_pack", 'B', "Undocumented", + {"key-binary-pack", 'B', "Undocumented", 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, - {"key_blob", 'b', "Undocumented", + {"key-blob", 'b', "Undocumented", 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, - {"key_cache", 'K', "Undocumented", (gptr*) &key_cacheing, + {"key-cache", 'K', "Undocumented", (gptr*) &key_cacheing, (gptr*) &key_cacheing, 0, GET_BOOL, NO_ARG, 0, 0, 0, 0, 0, 0}, - {"key_length", 'k', "Undocumented", (gptr*) &key_length, (gptr*) &key_length, + {"key-length", 'k', "Undocumented", (gptr*) &key_length, (gptr*) &key_length, 0, GET_UINT, REQUIRED_ARG, 6, 0, 0, 0, 0, 0}, - {"key_multiple", 'm', "Undocumented", + {"key-multiple", 'm', "Undocumented", 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, - {"key_prefix_pack", 'P', "Undocumented", + {"key-prefix_pack", 'P', "Undocumented", 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, - {"key_space_pack", 'p', "Undocumented", + {"key-space_pack", 'p', "Undocumented", 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, - {"key_varchar", 'w', "Test VARCHAR keys", + {"key-varchar", 'w', "Test VARCHAR keys", 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, - {"null_fields", 'N', "Define fields with NULL", + {"null-fields", 'N', "Define fields with NULL", (gptr*) &null_fields, (gptr*) &null_fields, 0, GET_BOOL, NO_ARG, 0, 0, 0, 0, 0, 0}, - {"row_fixed_size", 'S', "Undocumented", + {"row-fixed-size", 'S', "Fixed size records", + 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, + {"rows-in-block", 'M', "Store rows in block format", 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, - {"row_pointer_size", 'R', "Undocumented", (gptr*) &rec_pointer_size, + {"row-pointer-size", 'R', "Undocumented", (gptr*) &rec_pointer_size, (gptr*) &rec_pointer_size, 0, GET_INT, REQUIRED_ARG, 0, 0, 0, 0, 0, 0}, {"silent", 's', "Undocumented", (gptr*) &silent, (gptr*) &silent, 0, GET_BOOL, NO_ARG, 0, 0, 0, 0, 0, 0}, - {"skip_update", 'U', "Undocumented", (gptr*) &skip_update, + {"skip-delete", 'U', "Don't test deletes", (gptr*) &skip_delete, + (gptr*) &skip_delete, 0, GET_BOOL, NO_ARG, 0, 0, 0, 0, 0, 0}, + {"skip-update", 'D', "Don't test updates", (gptr*) &skip_update, (gptr*) &skip_update, 0, GET_BOOL, NO_ARG, 0, 0, 0, 0, 0, 0}, {"unique", 'C', "Undocumented", (gptr*) &opt_unique, (gptr*) &opt_unique, 0, GET_BOOL, NO_ARG, 0, 0, 0, 0, 0, 0}, - {"update_rows", 'u', "Undocumented", (gptr*) &update_count, + {"update-rows", 'u', "Undocumented", (gptr*) &update_count, (gptr*) &update_count, 0, GET_UINT, REQUIRED_ARG, 1000, 0, 0, 0, 0, 0}, {"verbose", 'v', "Be more verbose", (gptr*) &verbose, (gptr*) &verbose, 0, GET_BOOL, NO_ARG, 0, 0, 0, 0, 0, 0}, @@ -605,15 +629,20 @@ get_one_option(int optid, const struct my_option *opt __attribute__((unused)), case 'B': pack_keys= HA_BINARY_PACK_KEY; /* Use binary compression */ break; + case 'M': + record_type= BLOCK_RECORD; + break; case 'S': if (key_field == FIELD_VARCHAR) { create_flag=0; /* Static sized varchar */ + record_type= STATIC_RECORD; } else if (key_field != FIELD_BLOB) { key_field=FIELD_NORMAL; /* static-size record */ extra_field=FIELD_NORMAL; + record_type= STATIC_RECORD; } break; case 'p': @@ -629,6 +658,8 @@ get_one_option(int optid, const struct my_option *opt __attribute__((unused)), extra_field= FIELD_BLOB; pack_seg|= HA_BLOB_PART; key_type= HA_KEYTYPE_VARTEXT1; + if (record_type == STATIC_RECORD) + record_type= DYNAMIC_RECORD; break; case 'k': if (key_length < 4 || key_length > HA_MAX_KEY_LENGTH) @@ -642,7 +673,8 @@ get_one_option(int optid, const struct my_option *opt __attribute__((unused)), extra_field= FIELD_VARCHAR; key_type= HA_KEYTYPE_VARTEXT1; pack_seg|= HA_VAR_LENGTH_PART; - create_flag|= HA_PACK_RECORD; + if (record_type == STATIC_RECORD) + record_type= DYNAMIC_RECORD; break; case 'K': /* Use key cacheing */ key_cacheing=1; diff --git a/storage/maria/ma_test2.c b/storage/maria/ma_test2.c index 840ecb2eeb7..6d54a078f25 100644 --- a/storage/maria/ma_test2.c +++ b/storage/maria/ma_test2.c @@ -38,7 +38,8 @@ static void get_options(int argc, char *argv[]); static uint rnd(uint max_value); static void fix_length(byte *record,uint length); -static void put_blob_in_record(char *blob_pos,char **blob_buffer); +static void put_blob_in_record(char *blob_pos,char **blob_buffer, + ulong *length); static void copy_key(struct st_maria_info *info,uint inx, uchar *record,uchar *key); @@ -46,10 +47,11 @@ static int verbose=0,testflag=0, first_key=0,async_io=0,key_cacheing=0,write_cacheing=0,locking=0, rec_pointer_size=0,pack_fields=1,silent=0, opt_quick_mode=0; -static int pack_seg=HA_SPACE_PACK,pack_type=HA_PACK_KEY,remove_count=-1, - create_flag=0; +static int pack_seg=HA_SPACE_PACK,pack_type=HA_PACK_KEY,remove_count=-1; +static int create_flag= 0, srand_arg= 0; static ulong key_cache_size=IO_SIZE*16; static uint key_cache_block_size= KEY_CACHE_BLOCK_SIZE; +static enum data_file_type record_type= DYNAMIC_RECORD; static uint keys=MARIA_KEYS,recant=1000; static uint use_blob=0; @@ -201,16 +203,14 @@ int main(int argc, char *argv[]) for (i=4999 ; i>0 ; i--) key3[i]=0; if (!silent) - printf("- Creating isam-file\n"); - /* DBUG_PUSH(""); */ - /* my_delete(filename,MYF(0)); */ /* Remove old locks under gdb */ + printf("- Creating maria-file\n"); file= 0; bzero((char*) &create_info,sizeof(create_info)); create_info.max_rows=(ha_rows) (rec_pointer_size ? (1L << (rec_pointer_size*8))/ reclength : 0); create_info.reloc_rows=(ha_rows) 100; - if (maria_create(filename,keys,&keyinfo[first_key], + if (maria_create(filename, record_type, keys,&keyinfo[first_key], use_blob ? 7 : 6, &recinfo[0], 0,(MARIA_UNIQUEDEF*) 0, &create_info,create_flag)) @@ -230,12 +230,13 @@ int main(int argc, char *argv[]) for (i=0 ; i < recant ; i++) { + ulong blob_length; n1=rnd(1000); n2=rnd(100); n3=rnd(5000); sprintf(record,"%6d:%4d:%8d:Pos: %4d ",n1,n2,n3,write_count); int4store(record+STANDARD_LENGTH-4,(long) i); fix_length(record,(uint) STANDARD_LENGTH+rnd(60)); - put_blob_in_record(record+blob_pos,&blob_buffer); - DBUG_PRINT("test",("record: %d",i)); + put_blob_in_record(record+blob_pos,&blob_buffer, &blob_length); + DBUG_PRINT("test",("record: %d blob_length: %lu", i, blob_length)); if (maria_write(file,record)) { @@ -257,7 +258,7 @@ int main(int argc, char *argv[]) } /* Check if we can find key without flushing database */ - if (i == recant/2) + if (i % 10 == 0) { for (j=rnd(1000)+1 ; j>0 && key1[j] == 0 ; j--) ; if (!j) @@ -270,7 +271,8 @@ int main(int argc, char *argv[]) } } } - if (testflag==1) goto end; + if (testflag == 1) + goto end; if (write_cacheing) { @@ -285,6 +287,8 @@ int main(int argc, char *argv[]) if (!silent) printf("- Delete\n"); + if (srand_arg) + srand(srand_arg); for (i=0 ; i0 && key1[j] == 0 ; j--) ; @@ -296,6 +300,12 @@ int main(int argc, char *argv[]) printf("can't find key1: \"%s\"\n",key); goto err; } + if (bcmp(read_record+keyinfo[0].seg[0].start, + key, keyinfo[0].seg[0].length)) + { + printf("Found wrong record when searching for key: \"%s\"\n",key); + goto err; + } if (opt_delete == (uint) remove_count) /* While testing */ goto end; if (maria_delete(file,read_record)) @@ -310,10 +320,13 @@ int main(int argc, char *argv[]) else puts("Warning: Skipping delete test because no dupplicate keys"); } - if (testflag==2) goto end; + if (testflag == 2) + goto end; if (!silent) printf("- Update\n"); + if (srand_arg) + srand(srand_arg); for (i=0 ; i 0 ;) @@ -602,7 +637,7 @@ int main(int argc, char *argv[]) if (maria_rsame(file,read_record2,(int) i)) goto err; if (bcmp(read_record,read_record2,reclength) != 0) { - printf("is_rsame didn't find same record\n"); + printf("maria_rsame didn't find same record\n"); goto end; } } @@ -716,12 +751,14 @@ int main(int argc, char *argv[]) } } ant=0; - while ((error=maria_rrnd(file,record,HA_OFFSET_ERROR)) != HA_ERR_END_OF_FILE && + assert(maria_scan_init(file) == 0); + while ((error= maria_scan(file,record)) != HA_ERR_END_OF_FILE && ant < write_count + 10) - ant+= error ? 0 : 1; + ant+= error ? 0 : 1; + maria_scan_end(file); if (ant != write_count-opt_delete) { - printf("rrnd with cache: I can only find: %d records of %d\n", + printf("scan with cache: I can only find: %d records of %d\n", ant,write_count-opt_delete); goto end; } @@ -743,7 +780,8 @@ int main(int argc, char *argv[]) goto end; } - if (testflag == 4) goto end; + if (testflag == 4) + goto end; if (!silent) printf("- Removing keys\n"); @@ -752,8 +790,8 @@ int main(int argc, char *argv[]) /* DBUG_POP(); */ maria_reset(file); found_parts=0; - while ((error=maria_rrnd(file,read_record,HA_OFFSET_ERROR)) != - HA_ERR_END_OF_FILE) + maria_scan_init(file); + while ((error= maria_scan(file,read_record)) != HA_ERR_END_OF_FILE) { info.recpos=maria_position(file); if (lastpos >= info.recpos && lastpos != HA_OFFSET_ERROR) @@ -767,7 +805,7 @@ int main(int argc, char *argv[]) { if (opt_delete == (uint) remove_count) /* While testing */ goto end; - if (maria_rsame(file,read_record,-1)) + if (rnd(2) == 1 && maria_rsame(file,read_record,-1)) { printf("can't find record %lx\n",(long) info.recpos); goto err; @@ -783,9 +821,10 @@ int main(int argc, char *argv[]) { if (ptr[pos] != (uchar) (blob_length+pos)) { - printf("found blob with wrong info at %ld\n",(long) lastpos); - use_blob=0; - break; + printf("Found blob with wrong info at %ld\n",(long) lastpos); + maria_scan_end(file); + my_errno= 0; + goto err; } } } @@ -793,6 +832,7 @@ int main(int argc, char *argv[]) { printf("can't delete record: %6.6s, delete_count: %d\n", read_record, opt_delete); + maria_scan_end(file); goto err; } opt_delete++; @@ -800,6 +840,7 @@ int main(int argc, char *argv[]) else found_parts++; } + maria_scan_end(file); if (my_errno != HA_ERR_END_OF_FILE && my_errno != HA_ERR_RECORD_DELETED) printf("error: %d from maria_rrnd\n",my_errno); if (write_count != opt_delete) @@ -851,8 +892,7 @@ reads: %10lu\n", (ulong) maria_key_cache->global_cache_read); } end_key_cache(maria_key_cache,1); - if (blob_buffer) - my_free(blob_buffer,MYF(0)); + my_free(blob_buffer, MYF(MY_ALLOW_ZERO_PTR)); my_end(silent ? MY_CHECK_ERROR : MY_CHECK_ERROR | MY_GIVE_INFO); return(0); err: @@ -864,8 +904,7 @@ err: } /* main */ - /* l{ser optioner */ - /* OBS! intierar endast DEBUG - ingen debuggning h{r ! */ +/* Read options */ static void get_options(int argc, char **argv) { @@ -879,7 +918,9 @@ static void get_options(int argc, char **argv) pack_type= HA_BINARY_PACK_KEY; break; case 'b': - use_blob=1; + use_blob= 1; + if (*++pos) + use_blob= atol(pos); break; case 'K': /* Use key cacheing */ key_cacheing=1; @@ -896,7 +937,7 @@ static void get_options(int argc, char **argv) break; case 'i': if (*++pos) - srand(atoi(pos)); + srand(srand_arg= atoi(pos)); break; case 'L': locking=1; @@ -910,9 +951,9 @@ static void get_options(int argc, char **argv) verbose=1; break; case 'm': /* records */ - if ((recant=atoi(++pos)) < 10) + if ((recant=atoi(++pos)) < 10 && testflag > 1) { - fprintf(stderr,"record count must be >= 10\n"); + fprintf(stderr,"record count must be >= 10 (if testflag != 1)\n"); exit(1); } break; @@ -943,6 +984,9 @@ static void get_options(int argc, char **argv) keys > (uint) (MARIA_KEYS-first_key)) keys=MARIA_KEYS-first_key; break; + case 'M': + record_type= BLOCK_RECORD; + break; case 'P': pack_type=0; /* Don't use DIFF_LENGTH */ pack_seg=0; @@ -954,6 +998,7 @@ static void get_options(int argc, char **argv) break; case 'S': pack_fields=0; /* Static-length-records */ + record_type= STATIC_RECORD; break; case 's': silent=1; @@ -973,7 +1018,7 @@ static void get_options(int argc, char **argv) case '?': case 'I': case 'V': - printf("%s Ver 1.2 for %s at %s\n",progname,SYSTEM_TYPE,MACHINE_TYPE); + printf("%s Ver 1.0 for %s at %s\n",progname,SYSTEM_TYPE,MACHINE_TYPE); puts("By Monty, for your professional use\n"); printf("Usage: %s [-?AbBcDIKLPRqSsVWltv] [-k#] [-f#] [-m#] [-e#] [-E#] [-t#]\n", progname); @@ -1010,7 +1055,8 @@ static void fix_length(byte *rec, uint length) /* Put maybe a blob in record */ -static void put_blob_in_record(char *blob_pos, char **blob_buffer) +static void put_blob_in_record(char *blob_pos, char **blob_buffer, + ulong *blob_length) { ulong i,length; if (use_blob) @@ -1028,10 +1074,12 @@ static void put_blob_in_record(char *blob_pos, char **blob_buffer) (*blob_buffer)[i]=(char) (length+i); int4store(blob_pos,length); memcpy_fixed(blob_pos+4,(char*) blob_buffer,sizeof(char*)); + *blob_length= length; } else { int4store(blob_pos,0); + *blob_length= 0; } } return; diff --git a/storage/maria/ma_test3.c b/storage/maria/ma_test3.c index 96b896b03c6..4911d13f2f1 100644 --- a/storage/maria/ma_test3.c +++ b/storage/maria/ma_test3.c @@ -97,8 +97,8 @@ int main(int argc,char **argv) puts("- Creating maria-file"); my_delete(filename,MYF(0)); /* Remove old locks under gdb */ - if (maria_create(filename,2,&keyinfo[0],2,&recinfo[0],0,(MARIA_UNIQUEDEF*) 0, - (MARIA_CREATE_INFO*) 0,0)) + if (maria_create(filename,BLOCK_RECORD, 2, &keyinfo[0],2,&recinfo[0],0, + (MARIA_UNIQUEDEF*) 0, (MARIA_CREATE_INFO*) 0,0)) exit(1); rnd_init(0); diff --git a/storage/maria/ma_test_all.sh b/storage/maria/ma_test_all.sh index f857127dca9..17e654ac51f 100755 --- a/storage/maria/ma_test_all.sh +++ b/storage/maria/ma_test_all.sh @@ -5,143 +5,188 @@ valgrind="valgrind --alignment=8 --leak-check=yes" silent="-s" +suffix="" +#set -x -v -e -if test -f ma_test1$MACH ; then suffix=$MACH ; else suffix=""; fi -./ma_test1$suffix $silent -./maria_chk$suffix -se test1 -./ma_test1$suffix $silent -N -S -./maria_chk$suffix -se test1 -./ma_test1$suffix $silent -P --checksum -./maria_chk$suffix -se test1 -./ma_test1$suffix $silent -P -N -S -./maria_chk$suffix -se test1 -./ma_test1$suffix $silent -B -N -R2 -./maria_chk$suffix -sm test1 -./ma_test1$suffix $silent -a -k 480 --unique -./maria_chk$suffix -sm test1 -./ma_test1$suffix $silent -a -N -S -R1 -./maria_chk$suffix -sm test1 -./ma_test1$suffix $silent -p -S -./maria_chk$suffix -sm test1 -./ma_test1$suffix $silent -p -S -N --unique -./maria_chk$suffix -sm test1 -./ma_test1$suffix $silent -p -S -N --key_length=127 --checksum -./maria_chk$suffix -sm test1 -./ma_test1$suffix $silent -p -S -N --key_length=128 -./maria_chk$suffix -sm test1 -./ma_test1$suffix $silent -p -S --key_length=480 -./maria_chk$suffix -sm test1 -./ma_test1$suffix $silent -a -B -./maria_chk$suffix -sm test1 -./ma_test1$suffix $silent -a -B --key_length=64 --unique -./maria_chk$suffix -sm test1 -./ma_test1$suffix $silent -a -B -k 480 --checksum -./maria_chk$suffix -sm test1 -./ma_test1$suffix $silent -a -B -k 480 -N --unique --checksum -./maria_chk$suffix -sm test1 -./ma_test1$suffix $silent -a -m -./maria_chk$suffix -sm test1 -./ma_test1$suffix $silent -a -m -P --unique --checksum -./maria_chk$suffix -sm test1 -./ma_test1$suffix $silent -a -m -P --key_length=480 --key_cache -./maria_chk$suffix -sm test1 -./ma_test1$suffix $silent -m -p -./maria_chk$suffix -sm test1 -./ma_test1$suffix $silent -w -S --unique -./maria_chk$suffix -sm test1 -./ma_test1$suffix $silent -a -w --key_length=64 --checksum -./maria_chk$suffix -sm test1 -./ma_test1$suffix $silent -a -w -N --key_length=480 -./maria_chk$suffix -sm test1 -./ma_test1$suffix $silent -a -w -S --key_length=480 --checksum -./maria_chk$suffix -sm test1 -./ma_test1$suffix $silent -a -b -N -./maria_chk$suffix -sm test1 -./ma_test1$suffix $silent -a -b --key_length=480 -./maria_chk$suffix -sm test1 -./ma_test1$suffix $silent -p -B --key_length=480 -./maria_chk$suffix -sm test1 +run_tests() +{ + row_type=$1 + # + # First some simple tests + # + ./ma_test1$suffix $silent $row_type + ./maria_chk$suffix -se test1 + ./ma_test1$suffix $silent -N $row_type + ./maria_chk$suffix -se test1 + ./ma_test1$suffix $silent -P --checksum $row_type + ./maria_chk$suffix -se test1 + ./ma_test1$suffix $silent -P -N $row_type + ./maria_chk$suffix -se test1 + ./ma_test1$suffix $silent -B -N -R2 $row_type + ./maria_chk$suffix -sm test1 + ./ma_test1$suffix $silent -a -k 480 --unique $row_type + ./maria_chk$suffix -sm test1 + ./ma_test1$suffix $silent -a -N -R1 $row_type + ./maria_chk$suffix -sm test1 + ./ma_test1$suffix $silent -p $row_type + ./maria_chk$suffix -sm test1 + ./ma_test1$suffix $silent -p -N --unique $row_type + ./maria_chk$suffix -sm test1 + ./ma_test1$suffix $silent -p -N --key_length=127 --checksum $row_type + ./maria_chk$suffix -sm test1 + ./ma_test1$suffix $silent -p -N --key_length=128 $row_type + ./maria_chk$suffix -sm test1 + ./ma_test1$suffix $silent -p --key_length=480 $row_type + ./maria_chk$suffix -sm test1 + ./ma_test1$suffix $silent -a -B $row_type + ./maria_chk$suffix -sm test1 + ./ma_test1$suffix $silent -a -B --key_length=64 --unique $row_type + ./maria_chk$suffix -sm test1 + ./ma_test1$suffix $silent -a -B -k 480 --checksum $row_type + ./maria_chk$suffix -sm test1 + ./ma_test1$suffix $silent -a -B -k 480 -N --unique --checksum $row_type + ./maria_chk$suffix -sm test1 + ./ma_test1$suffix $silent -a -m $row_type + ./maria_chk$suffix -sm test1 + ./ma_test1$suffix $silent -a -m -P --unique --checksum $row_type + ./maria_chk$suffix -sm test1 + ./ma_test1$suffix $silent -a -m -P --key_length=480 --key_cache $row_type + ./maria_chk$suffix -sm test1 + ./ma_test1$suffix $silent -m -p $row_type + ./maria_chk$suffix -sm test1 + ./ma_test1$suffix $silent -w --unique $row_type + ./maria_chk$suffix -sm test1 + ./ma_test1$suffix $silent -a -w --key_length=64 --checksum $row_type + ./maria_chk$suffix -sm test1 + ./ma_test1$suffix $silent -a -w -N --key_length=480 $row_type + ./maria_chk$suffix -sm test1 + ./ma_test1$suffix $silent -a -w --key_length=480 --checksum $row_type + ./maria_chk$suffix -sm test1 + ./ma_test1$suffix $silent -a -b -N $row_type + ./maria_chk$suffix -sm test1 + ./ma_test1$suffix $silent -a -b --key_length=480 $row_type + ./maria_chk$suffix -sm test1 + ./ma_test1$suffix $silent -p -B --key_length=480 $row_type + ./maria_chk$suffix -sm test1 + ./ma_test1$suffix $silent --checksum --unique $row_type + ./maria_chk$suffix -se test1 + ./ma_test1$suffix $silent --unique $row_type + ./maria_chk$suffix -se test1 + + ./ma_test1$suffix $silent --key_multiple -N -S $row_type + ./maria_chk$suffix -sm test1 + ./ma_test1$suffix $silent --key_multiple -a -p --key_length=480 $row_type + ./maria_chk$suffix -sm test1 + ./ma_test1$suffix $silent --key_multiple -a -B --key_length=480 $row_type + ./maria_chk$suffix -sm test1 + ./ma_test1$suffix $silent --key_multiple -P -S $row_type + ./maria_chk$suffix -sm test1 + + ./maria_pack$suffix --force -s test1 + ./maria_chk$suffix -ess test1 + + ./ma_test2$suffix $silent -L -K -W -P $row_type + ./maria_chk$suffix -sm test2 + ./ma_test2$suffix $silent -L -K -W -P -A $row_type + ./maria_chk$suffix -sm test2 + ./ma_test2$suffix $silent -L -K -P -R3 -m50 -b1000000 $row_type + ./maria_chk$suffix -sm test2 + ./ma_test2$suffix $silent -L -B $row_type + ./maria_chk$suffix -sm test2 + ./ma_test2$suffix $silent -D -B -c $row_type + ./maria_chk$suffix -sm test2 + ./ma_test2$suffix $silent -m10000 -e4096 -K $row_type + ./maria_chk$suffix -sm test2 + ./ma_test2$suffix $silent -m10000 -e8192 -K $row_type + ./maria_chk$suffix -sm test2 + ./ma_test2$suffix $silent -m10000 -e16384 -E16384 -K -L $row_type + ./maria_chk$suffix -sm test2 +} -./ma_test1$suffix $silent --checksum -./maria_chk$suffix -se test1 -./maria_chk$suffix -rs test1 -./maria_chk$suffix -se test1 -./maria_chk$suffix -rqs test1 -./maria_chk$suffix -se test1 -./maria_chk$suffix -rs --correct-checksum test1 -./maria_chk$suffix -se test1 -./maria_chk$suffix -rqs --correct-checksum test1 -./maria_chk$suffix -se test1 -./maria_chk$suffix -ros --correct-checksum test1 -./maria_chk$suffix -se test1 -./maria_chk$suffix -rqos --correct-checksum test1 -./maria_chk$suffix -se test1 +run_repair_tests() +{ + row_type=$1 + ./ma_test1$suffix $silent --checksum $row_type + ./maria_chk$suffix -se test1 + ./maria_chk$suffix -rs test1 + ./maria_chk$suffix -se test1 + ./maria_chk$suffix -rqs test1 + ./maria_chk$suffix -se test1 + ./maria_chk$suffix -rs --correct-checksum test1 + ./maria_chk$suffix -se test1 + ./maria_chk$suffix -rqs --correct-checksum test1 + ./maria_chk$suffix -se test1 + ./maria_chk$suffix -ros --correct-checksum test1 + ./maria_chk$suffix -se test1 + ./maria_chk$suffix -rqos --correct-checksum test1 + ./maria_chk$suffix -se test1 +} -# check of maria_pack / maria_chk -./maria_pack$suffix --force -s test1 -./maria_chk$suffix -es test1 -./maria_chk$suffix -rqs test1 -./maria_chk$suffix -es test1 -./maria_chk$suffix -rs test1 -./maria_chk$suffix -es test1 -./maria_chk$suffix -rus test1 -./maria_chk$suffix -es test1 +run_pack_tests() +{ + row_type=$1 + # check of maria_pack / maria_chk + ./ma_test1$suffix $silent --checksum $row_type + ./maria_pack$suffix --force -s test1 + ./maria_chk$suffix -ess test1 + ./maria_chk$suffix -rqs test1 + ./maria_chk$suffix -es test1 + ./maria_chk$suffix -rs test1 + ./maria_chk$suffix -es test1 + ./maria_chk$suffix -rus test1 + ./maria_chk$suffix -es test1 + + ./ma_test1$suffix $silent --checksum -S $row_type + ./maria_chk$suffix -se test1 + ./maria_chk$suffix -ros test1 + ./maria_chk$suffix -rqs test1 + ./maria_chk$suffix -se test1 + + ./maria_pack$suffix --force -s test1 + ./maria_chk$suffix -rqs test1 + ./maria_chk$suffix -es test1 + ./maria_chk$suffix -rus test1 + ./maria_chk$suffix -es test1 +} -./ma_test1$suffix $silent --checksum -S -./maria_chk$suffix -se test1 -./maria_chk$suffix -ros test1 -./maria_chk$suffix -rqs test1 -./maria_chk$suffix -se test1 +echo "Running tests with dynamic row format" +run_tests "" +run_repair_tests "" +run_pack_tests "" -./maria_pack$suffix --force -s test1 -./maria_chk$suffix -rqs test1 -./maria_chk$suffix -es test1 -./maria_chk$suffix -rus test1 -./maria_chk$suffix -es test1 +echo "Running tests with static row format" +run_tests -S +run_repair_tests -S +run_pack_tests -S -./ma_test1$suffix $silent --checksum --unique -./maria_chk$suffix -se test1 -./ma_test1$suffix $silent --unique -S -./maria_chk$suffix -se test1 +echo "Running tests with block row format" +run_tests -M +# +# Tests that gives warnings +# -./ma_test1$suffix $silent --key_multiple -N -S -./maria_chk$suffix -sm test1 -./ma_test1$suffix $silent --key_multiple -a -p --key_length=480 -./maria_chk$suffix -sm test1 -./ma_test1$suffix $silent --key_multiple -a -B --key_length=480 -./maria_chk$suffix -sm test1 -./ma_test1$suffix $silent --key_multiple -P -S -./maria_chk$suffix -sm test1 - -./ma_test2$suffix $silent -L -K -W -P -./maria_chk$suffix -sm test2 -./ma_test2$suffix $silent -L -K -W -P -A -./maria_chk$suffix -sm test2 ./ma_test2$suffix $silent -L -K -W -P -S -R1 -m500 -echo "ma_test2$suffix $silent -L -K -R1 -m2000 ; Should give error 135" ./maria_chk$suffix -sm test2 +echo "ma_test2$suffix $silent -L -K -R1 -m2000 ; Should give error 135" ./ma_test2$suffix $silent -L -K -R1 -m2000 +echo "./maria_chk$suffix -sm test2 will warn that 'Datafile is almost full'" ./maria_chk$suffix -sm test2 -./ma_test2$suffix $silent -L -K -P -S -R3 -m50 -b1000000 -./maria_chk$suffix -sm test2 -./ma_test2$suffix $silent -L -B -./maria_chk$suffix -sm test2 -./ma_test2$suffix $silent -D -B -c -./maria_chk$suffix -sm test2 -./ma_test2$suffix $silent -m10000 -e8192 -K -./maria_chk$suffix -sm test2 -./ma_test2$suffix $silent -m10000 -e16384 -E16384 -K -L -./maria_chk$suffix -sm test2 +./maria_chk$suffix -ssm test2 -./ma_test2$suffix $silent -L -K -W -P -m50 -l -./maria_log$suffix -./ma_test2$suffix $silent -L -K -W -P -m50 -l -b100 -./maria_log$suffix +# +# Some timing tests +# time ./ma_test2$suffix $silent +time ./ma_test2$suffix $silent -S +time ./ma_test2$suffix $silent -M +time ./ma_test2$suffix $silent -B +time ./ma_test2$suffix $silent -L +time ./ma_test2$suffix $silent -K time ./ma_test2$suffix $silent -K -B time ./ma_test2$suffix $silent -L -B time ./ma_test2$suffix $silent -L -K -B time ./ma_test2$suffix $silent -L -K -W -B -time ./ma_test2$suffix $silent -L -K -W -S -B -time ./ma_test2$suffix $silent -D -K -W -S -B +time ./ma_test2$suffix $silent -L -K -W -B -S +time ./ma_test2$suffix $silent -L -K -W -B -M +time ./ma_test2$suffix $silent -D -K -W -B -S diff --git a/storage/maria/ma_unique.c b/storage/maria/ma_unique.c index bc1aa71966b..8348bfbd84b 100644 --- a/storage/maria/ma_unique.c +++ b/storage/maria/ma_unique.c @@ -22,10 +22,11 @@ my_bool _ma_check_unique(MARIA_HA *info, MARIA_UNIQUEDEF *def, byte *record, ha_checksum unique_hash, my_off_t disk_pos) { - my_off_t lastpos=info->lastpos; + my_off_t lastpos=info->cur_row.lastpos; MARIA_KEYDEF *key= &info->s->keyinfo[def->key]; - uchar *key_buff=info->lastkey2; + byte *key_buff= info->lastkey2; DBUG_ENTER("_ma_check_unique"); + DBUG_PRINT("enter",("unique_hash: %lu", unique_hash)); maria_unique_store(record+key->seg->start, unique_hash); _ma_make_key(info,def->key,key_buff,record,0); @@ -33,24 +34,25 @@ my_bool _ma_check_unique(MARIA_HA *info, MARIA_UNIQUEDEF *def, byte *record, /* The above changed info->lastkey2. Inform maria_rnext_same(). */ info->update&= ~HA_STATE_RNEXT_SAME; - if (_ma_search(info,info->s->keyinfo+def->key,key_buff,MARIA_UNIQUE_HASH_LENGTH, + if (_ma_search(info,info->s->keyinfo+def->key,key_buff, + MARIA_UNIQUE_HASH_LENGTH, SEARCH_FIND,info->s->state.key_root[def->key])) { info->page_changed=1; /* Can't optimize read next */ - info->lastpos= lastpos; + info->cur_row.lastpos= lastpos; DBUG_RETURN(0); /* No matching rows */ } for (;;) { - if (info->lastpos != disk_pos && - !(*info->s->compare_unique)(info,def,record,info->lastpos)) + if (info->cur_row.lastpos != disk_pos && + !(*info->s->compare_unique)(info,def,record,info->cur_row.lastpos)) { my_errno=HA_ERR_FOUND_DUPP_UNIQUE; info->errkey= (int) def->key; - info->dupp_key_pos= info->lastpos; - info->page_changed=1; /* Can't optimize read next */ - info->lastpos=lastpos; + info->dup_key_pos= info->cur_row.lastpos; + info->page_changed= 1; /* Can't optimize read next */ + info->cur_row.lastpos= lastpos; DBUG_PRINT("info",("Found duplicate")); DBUG_RETURN(1); /* Found identical */ } @@ -60,8 +62,8 @@ my_bool _ma_check_unique(MARIA_HA *info, MARIA_UNIQUEDEF *def, byte *record, memcmp((char*) info->lastkey, (char*) key_buff, MARIA_UNIQUE_HASH_LENGTH)) { - info->page_changed=1; /* Can't optimize read next */ - info->lastpos=lastpos; + info->page_changed= 1; /* Can't optimize read next */ + info->cur_row.lastpos= lastpos; DBUG_RETURN(0); /* end of tree */ } } @@ -144,11 +146,11 @@ ha_checksum _ma_unique_hash(MARIA_UNIQUEDEF *def, const byte *record) RETURN 0 if both rows have equal unique value - # Rows are different + 1 Rows are different */ -int _ma_unique_comp(MARIA_UNIQUEDEF *def, const byte *a, const byte *b, - my_bool null_are_equal) +my_bool _ma_unique_comp(MARIA_UNIQUEDEF *def, const byte *a, const byte *b, + my_bool null_are_equal) { const byte *pos_a, *pos_b, *end; HA_KEYSEG *keyseg; diff --git a/storage/maria/ma_update.c b/storage/maria/ma_update.c index 248b17ce2c9..c86ebd16c6f 100644 --- a/storage/maria/ma_update.c +++ b/storage/maria/ma_update.c @@ -24,7 +24,7 @@ int maria_update(register MARIA_HA *info, const byte *oldrec, byte *newrec) int flag,key_changed,save_errno; reg3 my_off_t pos; uint i; - uchar old_key[HA_MAX_KEY_BUFF],*new_key; + byte old_key[HA_MAX_KEY_BUFF],*new_key; bool auto_key_changed=0; ulonglong changed; MARIA_SHARE *share=info->s; @@ -49,18 +49,26 @@ int maria_update(register MARIA_HA *info, const byte *oldrec, byte *newrec) { DBUG_RETURN(my_errno=HA_ERR_INDEX_FILE_FULL); } - pos=info->lastpos; + pos= info->cur_row.lastpos; if (_ma_readinfo(info,F_WRLCK,1)) DBUG_RETURN(my_errno); - if (share->calc_checksum) - old_checksum=info->checksum=(*share->calc_checksum)(info,oldrec); if ((*share->compare_record)(info,oldrec)) { - save_errno=my_errno; + save_errno= my_errno; + DBUG_PRINT("warning", ("Got error from compare record")); goto err_end; /* Record has changed */ } + if (share->calc_checksum) + { + /* + We can't use the row based checksum as this doesn't have enough + precision. + */ + if (info->s->calc_checksum) + old_checksum= (*info->s->calc_checksum)(info, oldrec); + } /* Calculate and check all unique constraints */ key_changed=0; @@ -69,7 +77,7 @@ int maria_update(register MARIA_HA *info, const byte *oldrec, byte *newrec) MARIA_UNIQUEDEF *def=share->uniqueinfo+i; if (_ma_unique_comp(def, newrec, oldrec,1) && _ma_check_unique(info, def, newrec, _ma_unique_hash(def, newrec), - info->lastpos)) + pos)) { save_errno=my_errno; goto err_end; @@ -83,7 +91,7 @@ int maria_update(register MARIA_HA *info, const byte *oldrec, byte *newrec) /* Check which keys changed from the original row */ - new_key=info->lastkey2; + new_key= info->lastkey2; changed=0; for (i=0 ; i < share->base.keys ; i++) { @@ -116,7 +124,7 @@ int maria_update(register MARIA_HA *info, const byte *oldrec, byte *newrec) info->update&= ~HA_STATE_RNEXT_SAME; if (new_length != old_length || - memcmp((byte*) old_key,(byte*) new_key,new_length)) + memcmp(old_key, new_key, new_length)) { if ((int) i == info->lastinx) key_changed|=HA_STATE_WRITTEN; /* Mark that keyfile changed */ @@ -139,7 +147,7 @@ int maria_update(register MARIA_HA *info, const byte *oldrec, byte *newrec) if (share->calc_checksum) { - info->checksum=(*share->calc_checksum)(info,newrec); + info->cur_row.checksum= (*share->calc_checksum)(info,newrec); /* Store new checksum in index file header */ key_changed|= HA_STATE_CHANGED; } @@ -167,10 +175,13 @@ int maria_update(register MARIA_HA *info, const byte *oldrec, byte *newrec) set_if_bigger(info->s->state.auto_increment, ma_retrieve_auto_increment(info, newrec)); if (share->calc_checksum) - info->state->checksum+=(info->checksum - old_checksum); + info->state->checksum+= (info->cur_row.checksum - old_checksum); - info->update= (HA_STATE_CHANGED | HA_STATE_ROW_CHANGED | HA_STATE_AKTIV | - key_changed); + /* + We can't yet have HA_STATE_ACTIVE here, as block_record dosn't support + it + */ + info->update= (HA_STATE_CHANGED | HA_STATE_ROW_CHANGED | key_changed); VOID(_ma_writeinfo(info,key_changed ? WRITEINFO_UPDATE_KEYFILE : 0)); allow_break(); /* Allow SIGHUP & SIGINT */ if (info->invalidator != 0) @@ -184,8 +195,6 @@ int maria_update(register MARIA_HA *info, const byte *oldrec, byte *newrec) err: DBUG_PRINT("error",("key: %d errno: %d",i,my_errno)); save_errno=my_errno; - if (changed) - key_changed|= HA_STATE_CHANGED; if (my_errno == HA_ERR_FOUND_DUPP_KEY || my_errno == HA_ERR_RECORD_FILE_FULL) { info->errkey= (int) i; diff --git a/storage/maria/ma_write.c b/storage/maria/ma_write.c index 24768b36c89..a4fe5506c8e 100644 --- a/storage/maria/ma_write.c +++ b/storage/maria/ma_write.c @@ -24,33 +24,52 @@ /* Functions declared in this file */ static int w_search(MARIA_HA *info,MARIA_KEYDEF *keyinfo, - uint comp_flag, uchar *key, - uint key_length, my_off_t pos, uchar *father_buff, - uchar *father_keypos, my_off_t father_page, + uint comp_flag, byte *key, + uint key_length, my_off_t pos, byte *father_buff, + byte *father_keypos, my_off_t father_page, my_bool insert_last); -static int _ma_balance_page(MARIA_HA *info,MARIA_KEYDEF *keyinfo,uchar *key, - uchar *curr_buff,uchar *father_buff, - uchar *father_keypos,my_off_t father_page); -static uchar *_ma_find_last_pos(MARIA_KEYDEF *keyinfo, uchar *page, - uchar *key, uint *return_key_length, - uchar **after_key); -int _ma_ck_write_tree(register MARIA_HA *info, uint keynr,uchar *key, +static int _ma_balance_page(MARIA_HA *info,MARIA_KEYDEF *keyinfo,byte *key, + byte *curr_buff,byte *father_buff, + byte *father_keypos,my_off_t father_page); +static byte *_ma_find_last_pos(MARIA_KEYDEF *keyinfo, byte *page, + byte *key, uint *return_key_length, + byte **after_key); +int _ma_ck_write_tree(register MARIA_HA *info, uint keynr,byte *key, uint key_length); -int _ma_ck_write_btree(register MARIA_HA *info, uint keynr,uchar *key, +int _ma_ck_write_btree(register MARIA_HA *info, uint keynr,byte *key, uint key_length); - /* Write new record to database */ + +MARIA_RECORD_POS _ma_write_init_default(MARIA_HA *info, + const byte *record + __attribute__((unused))) +{ + return ((info->s->state.dellink != HA_OFFSET_ERROR && + !info->append_insert_at_end) ? + info->s->state.dellink : + info->state->data_file_length); +} + +my_bool _ma_write_abort_default(MARIA_HA *info __attribute__((unused))) +{ + return 0; +} + + +/* Write new record to a table */ int maria_write(MARIA_HA *info, byte *record) { MARIA_SHARE *share=info->s; uint i; int save_errno; - my_off_t filepos; - uchar *buff; + MARIA_RECORD_POS filepos; + byte *buff; my_bool lock_tree= share->concurrent_insert; + my_bool fatal_error; DBUG_ENTER("maria_write"); - DBUG_PRINT("enter",("isam: %d data: %d",info->s->kfile,info->dfile)); + DBUG_PRINT("enter",("index_file: %d data_file: %d", + info->s->kfile,info->dfile)); DBUG_EXECUTE_IF("maria_pretend_crashed_table_on_usage", maria_print_error(info->s, HA_ERR_CRASHED); @@ -62,10 +81,6 @@ int maria_write(MARIA_HA *info, byte *record) if (_ma_readinfo(info,F_WRLCK,1)) DBUG_RETURN(my_errno); dont_break(); /* Dont allow SIGHUP or SIGINT */ - filepos= ((share->state.dellink != HA_OFFSET_ERROR && - !info->append_insert_at_end) ? - share->state.dellink : - info->state->data_file_length); if (share->base.reloc == (ha_rows) 1 && share->base.records == (ha_rows) 1 && @@ -86,14 +101,26 @@ int maria_write(MARIA_HA *info, byte *record) for (i=0 ; i < share->state.header.uniques ; i++) { if (_ma_check_unique(info,share->uniqueinfo+i,record, - _ma_unique_hash(share->uniqueinfo+i,record), - HA_OFFSET_ERROR)) + _ma_unique_hash(share->uniqueinfo+i,record), + HA_OFFSET_ERROR)) goto err2; } - /* Write all keys to indextree */ + if ((info->opt_flag & OPT_NO_ROWS)) + filepos= HA_OFFSET_ERROR; + else + { + /* + This may either calculate a record or, or write the record and return + the record id + */ + if ((filepos= (*share->write_record_init)(info, record)) == + HA_OFFSET_ERROR) + goto err2; + } - buff=info->lastkey2; + /* Write all keys to indextree */ + buff= info->lastkey2; for (i=0 ; i < share->base.keys ; i++) { if (maria_is_key_active(share->state.key_map, i)) @@ -136,13 +163,13 @@ int maria_write(MARIA_HA *info, byte *record) rw_unlock(&share->key_root_lock[i]); } } - if (share->calc_checksum) - info->checksum=(*share->calc_checksum)(info,record); - if (!(info->opt_flag & OPT_NO_ROWS)) + if (share->calc_write_checksum) + info->cur_row.checksum= (*share->calc_write_checksum)(info,record); + if (filepos != HA_OFFSET_ERROR) { if ((*share->write_record)(info,record)) goto err; - info->state->checksum+=info->checksum; + info->state->checksum+= info->cur_row.checksum; } if (share->base.auto_key) set_if_bigger(info->s->state.auto_increment, @@ -150,7 +177,7 @@ int maria_write(MARIA_HA *info, byte *record) info->update= (HA_STATE_CHANGED | HA_STATE_AKTIV | HA_STATE_WRITTEN | HA_STATE_ROW_CHANGED); info->state->records++; - info->lastpos=filepos; + info->cur_row.lastpos= filepos; VOID(_ma_writeinfo(info, WRITEINFO_UPDATE_KEYFILE)); if (info->invalidator != 0) { @@ -162,8 +189,10 @@ int maria_write(MARIA_HA *info, byte *record) DBUG_RETURN(0); err: - save_errno=my_errno; - if (my_errno == HA_ERR_FOUND_DUPP_KEY || my_errno == HA_ERR_RECORD_FILE_FULL || + save_errno= my_errno; + fatal_error= 0; + if (my_errno == HA_ERR_FOUND_DUPP_KEY || + my_errno == HA_ERR_RECORD_FILE_FULL || my_errno == HA_ERR_NULL_IN_SPATIAL) { if (info->bulk_insert) @@ -207,14 +236,21 @@ err: } } else + fatal_error= 1; + + if ((*share->write_record_abort)(info)) + fatal_error= 1; + if (fatal_error) { maria_print_error(info->s, HA_ERR_CRASHED); maria_mark_crashed(info); } + info->update= (HA_STATE_CHANGED | HA_STATE_WRITTEN | HA_STATE_ROW_CHANGED); my_errno=save_errno; err2: save_errno=my_errno; + DBUG_PRINT("error", ("got error: %d", save_errno)); VOID(_ma_writeinfo(info,WRITEINFO_UPDATE_KEYFILE)); allow_break(); /* Allow SIGHUP & SIGINT */ DBUG_RETURN(my_errno=save_errno); @@ -223,7 +259,7 @@ err2: /* Write one key to btree */ -int _ma_ck_write(MARIA_HA *info, uint keynr, uchar *key, uint key_length) +int _ma_ck_write(MARIA_HA *info, uint keynr, byte *key, uint key_length) { DBUG_ENTER("_ma_ck_write"); @@ -242,7 +278,7 @@ int _ma_ck_write(MARIA_HA *info, uint keynr, uchar *key, uint key_length) * Normal insert code * **********************************************************************/ -int _ma_ck_write_btree(register MARIA_HA *info, uint keynr, uchar *key, +int _ma_ck_write_btree(register MARIA_HA *info, uint keynr, byte *key, uint key_length) { int error; @@ -275,15 +311,17 @@ int _ma_ck_write_btree(register MARIA_HA *info, uint keynr, uchar *key, DBUG_RETURN(error); } /* _ma_ck_write_btree */ + int _ma_ck_real_write_btree(MARIA_HA *info, MARIA_KEYDEF *keyinfo, - uchar *key, uint key_length, my_off_t *root, uint comp_flag) + byte *key, uint key_length, my_off_t *root, + uint comp_flag) { int error; DBUG_ENTER("_ma_ck_real_write_btree"); /* key_length parameter is used only if comp_flag is SEARCH_FIND */ if (*root == HA_OFFSET_ERROR || (error=w_search(info, keyinfo, comp_flag, key, key_length, - *root, (uchar *) 0, (uchar*) 0, + *root, (byte*) 0, (byte*) 0, (my_off_t) 0, 1)) > 0) error= _ma_enlarge_root(info,keyinfo,key,root); DBUG_RETURN(error); @@ -292,7 +330,7 @@ int _ma_ck_real_write_btree(MARIA_HA *info, MARIA_KEYDEF *keyinfo, /* Make a new root with key as only pointer */ -int _ma_enlarge_root(MARIA_HA *info, MARIA_KEYDEF *keyinfo, uchar *key, +int _ma_enlarge_root(MARIA_HA *info, MARIA_KEYDEF *keyinfo, byte *key, my_off_t *root) { uint t_length,nod_flag; @@ -302,11 +340,11 @@ int _ma_enlarge_root(MARIA_HA *info, MARIA_KEYDEF *keyinfo, uchar *key, nod_flag= (*root != HA_OFFSET_ERROR) ? share->base.key_reflength : 0; _ma_kpointer(info,info->buff+2,*root); /* if nod */ - t_length=(*keyinfo->pack_key)(keyinfo,nod_flag,(uchar*) 0, - (uchar*) 0, (uchar*) 0, key,&s_temp); + t_length=(*keyinfo->pack_key)(keyinfo,nod_flag,(byte*) 0, + (byte*) 0, (byte*) 0, key,&s_temp); maria_putint(info->buff,t_length+2+nod_flag,nod_flag); (*keyinfo->store_key)(keyinfo,info->buff+2+nod_flag,&s_temp); - info->buff_used=info->page_changed=1; /* info->buff is used */ + info->keybuff_used=info->page_changed=1; /* info->buff is used */ if ((*root= _ma_new(info,keyinfo,DFLT_INIT_HITS)) == HA_OFFSET_ERROR || _ma_write_keypage(info,keyinfo,*root,DFLT_INIT_HITS,info->buff)) DBUG_RETURN(-1); @@ -322,21 +360,21 @@ int _ma_enlarge_root(MARIA_HA *info, MARIA_KEYDEF *keyinfo, uchar *key, */ static int w_search(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, - uint comp_flag, uchar *key, uint key_length, my_off_t page, - uchar *father_buff, uchar *father_keypos, + uint comp_flag, byte *key, uint key_length, my_off_t page, + byte *father_buff, byte *father_keypos, my_off_t father_page, my_bool insert_last) { int error,flag; uint nod_flag, search_key_length; - uchar *temp_buff,*keypos; - uchar keybuff[HA_MAX_KEY_BUFF]; + byte *temp_buff,*keypos; + byte keybuff[HA_MAX_KEY_BUFF]; my_bool was_last_key; - my_off_t next_page, dupp_key_pos; + my_off_t next_page, dup_key_pos; DBUG_ENTER("w_search"); DBUG_PRINT("enter",("page: %ld",page)); search_key_length= (comp_flag & SEARCH_FIND) ? key_length : USE_WHOLE_KEY; - if (!(temp_buff= (uchar*) my_alloca((uint) keyinfo->block_length+ + if (!(temp_buff= (byte*) my_alloca((uint) keyinfo->block_length+ HA_MAX_KEY_BUFF*2))) DBUG_RETURN(-1); if (!_ma_fetch_keypage(info,keyinfo,page,DFLT_INIT_HITS,temp_buff,0)) @@ -351,9 +389,9 @@ static int w_search(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, /* get position to record with duplicated key */ tmp_key_length=(*keyinfo->get_key)(keyinfo,nod_flag,&keypos,keybuff); if (tmp_key_length) - dupp_key_pos= _ma_dpos(info,0,keybuff+tmp_key_length); + dup_key_pos= _ma_dpos(info,0,keybuff+tmp_key_length); else - dupp_key_pos= HA_OFFSET_ERROR; + dup_key_pos= HA_OFFSET_ERROR; if (keyinfo->flag & HA_FULLTEXT) { @@ -373,7 +411,7 @@ static int w_search(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, else { /* popular word. two-level tree. going down */ - my_off_t root=dupp_key_pos; + my_off_t root=dup_key_pos; keyinfo=&info->s->ft2_keyinfo; get_key_full_length_rdonly(off, key); key+=off; @@ -392,7 +430,7 @@ static int w_search(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, } else /* not HA_FULLTEXT, normal HA_NOSAME key */ { - info->dupp_key_pos= dupp_key_pos; + info->dup_key_pos= dup_key_pos; my_afree((byte*) temp_buff); my_errno=HA_ERR_FOUND_DUPP_KEY; DBUG_RETURN(-1); @@ -447,25 +485,25 @@ err: */ int _ma_insert(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, - uchar *key, uchar *anc_buff, uchar *key_pos, uchar *key_buff, - uchar *father_buff, uchar *father_key_pos, my_off_t father_page, + byte *key, byte *anc_buff, byte *key_pos, byte *key_buff, + byte *father_buff, byte *father_key_pos, my_off_t father_page, my_bool insert_last) { uint a_length,nod_flag; int t_length; - uchar *endpos, *prev_key; + byte *endpos, *prev_key; MARIA_KEY_PARAM s_temp; DBUG_ENTER("_ma_insert"); - DBUG_PRINT("enter",("key_pos: %lx",key_pos)); + DBUG_PRINT("enter",("key_pos: 0x%lx", (ulong) key_pos)); DBUG_EXECUTE("key", _ma_print_key(DBUG_FILE,keyinfo->seg,key, USE_WHOLE_KEY);); nod_flag=_ma_test_if_nod(anc_buff); a_length=maria_getint(anc_buff); endpos= anc_buff+ a_length; - prev_key=(key_pos == anc_buff+2+nod_flag ? (uchar*) 0 : key_buff); + prev_key=(key_pos == anc_buff+2+nod_flag ? (byte*) 0 : key_buff); t_length=(*keyinfo->pack_key)(keyinfo,nod_flag, - (key_pos == endpos ? (uchar*) 0 : key_pos), + (key_pos == endpos ? (byte*) 0 : key_pos), prev_key, prev_key, key,&s_temp); #ifndef DBUG_OFF @@ -478,7 +516,7 @@ int _ma_insert(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, { DBUG_PRINT("test",("t_length: %d ref_len: %d", t_length,s_temp.ref_length)); - DBUG_PRINT("test",("n_ref_len: %d n_length: %d key_pos: %lx", + DBUG_PRINT("test",("n_ref_len: %d n_length: %d key_pos: 0x%lx", s_temp.n_ref_length,s_temp.n_length,s_temp.key)); } #endif @@ -517,19 +555,20 @@ int _ma_insert(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, Let's consider converting. We'll compare 'key' and the first key at anc_buff */ - uchar *a=key, *b=anc_buff+2+nod_flag; + byte *a=key, *b=anc_buff+2+nod_flag; uint alen, blen, ft2len=info->s->ft2_keyinfo.keylength; /* the very first key on the page is always unpacked */ DBUG_ASSERT((*b & 128) == 0); #if HA_FT_MAXLEN >= 127 blen= mi_uint2korr(b); b+=2; #else - blen= *b++; + blen= *(uchar*) b++; #endif get_key_length(alen,a); DBUG_ASSERT(info->ft1_to_ft2==0); if (alen == blen && - ha_compare_text(keyinfo->seg->charset, a, alen, b, blen, 0, 0)==0) + ha_compare_text(keyinfo->seg->charset, (uchar*) a, alen, + (uchar*) b, blen, 0, 0) == 0) { /* yup. converting */ info->ft1_to_ft2=(DYNAMIC_ARRAY *) @@ -570,11 +609,11 @@ int _ma_insert(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, /* split a full page in two and assign emerging item to key */ int _ma_split_page(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, - uchar *key, uchar *buff, uchar *key_buff, + byte *key, byte *buff, byte *key_buff, my_bool insert_last_key) { uint length,a_length,key_ref_length,t_length,nod_flag,key_length; - uchar *key_pos,*pos, *after_key; + byte *key_pos,*pos, *after_key; my_off_t new_pos; MARIA_KEY_PARAM s_temp; DBUG_ENTER("maria_split_page"); @@ -582,7 +621,7 @@ int _ma_split_page(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, if (info->s->keyinfo+info->lastinx == keyinfo) info->page_changed=1; /* Info->buff is used */ - info->buff_used=1; + info->keybuff_used=1; nod_flag=_ma_test_if_nod(buff); key_ref_length=2+nod_flag; if (insert_last_key) @@ -614,8 +653,8 @@ int _ma_split_page(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, if (!(*keyinfo->get_key)(keyinfo,nod_flag,&key_pos,key_buff)) DBUG_RETURN(-1); - t_length=(*keyinfo->pack_key)(keyinfo,nod_flag,(uchar *) 0, - (uchar*) 0, (uchar*) 0, + t_length=(*keyinfo->pack_key)(keyinfo,nod_flag,(byte *) 0, + (byte*) 0, (byte*) 0, key_buff, &s_temp); length=(uint) ((buff+a_length)-key_pos); memcpy((byte*) info->buff+key_ref_length+t_length,(byte*) key_pos, @@ -638,12 +677,12 @@ int _ma_split_page(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, after_key will contain the position to where the next key starts */ -uchar *_ma_find_half_pos(uint nod_flag, MARIA_KEYDEF *keyinfo, uchar *page, - uchar *key, uint *return_key_length, - uchar **after_key) +byte *_ma_find_half_pos(uint nod_flag, MARIA_KEYDEF *keyinfo, byte *page, + byte *key, uint *return_key_length, + byte **after_key) { uint keys,length,key_ref_length; - uchar *end,*lastpos; + byte *end,*lastpos; DBUG_ENTER("_ma_find_half_pos"); key_ref_length=2+nod_flag; @@ -672,24 +711,25 @@ uchar *_ma_find_half_pos(uint nod_flag, MARIA_KEYDEF *keyinfo, uchar *page, } while (page < end); *return_key_length=length; *after_key=page; - DBUG_PRINT("exit",("returns: %lx page: %lx half: %lx",lastpos,page,end)); + DBUG_PRINT("exit",("returns: 0x%lx page: 0x%lx half: 0x%lx", + lastpos, page, end)); DBUG_RETURN(lastpos); } /* _ma_find_half_pos */ - /* - Split buffer at last key - Returns pointer to the start of the key before the last key - key will contain the last key - */ +/* + Split buffer at last key + Returns pointer to the start of the key before the last key + key will contain the last key +*/ -static uchar *_ma_find_last_pos(MARIA_KEYDEF *keyinfo, uchar *page, - uchar *key, uint *return_key_length, - uchar **after_key) +static byte *_ma_find_last_pos(MARIA_KEYDEF *keyinfo, byte *page, + byte *key, uint *return_key_length, + byte **after_key) { uint keys,length,last_length,key_ref_length; - uchar *end,*lastpos,*prevpos; - uchar key_buff[HA_MAX_KEY_BUFF]; + byte *end,*lastpos,*prevpos; + byte key_buff[HA_MAX_KEY_BUFF]; DBUG_ENTER("_ma_find_last_pos"); key_ref_length=2; @@ -727,7 +767,8 @@ static uchar *_ma_find_last_pos(MARIA_KEYDEF *keyinfo, uchar *page, } *return_key_length=last_length; *after_key=lastpos; - DBUG_PRINT("exit",("returns: %lx page: %lx end: %lx",prevpos,page,end)); + DBUG_PRINT("exit",("returns: 0x%lx page: 0x%lx end: 0x%lx", + prevpos, page, end)); DBUG_RETURN(prevpos); } /* _ma_find_last_pos */ @@ -736,14 +777,14 @@ static uchar *_ma_find_last_pos(MARIA_KEYDEF *keyinfo, uchar *page, /* returns 0 if balance was done */ static int _ma_balance_page(register MARIA_HA *info, MARIA_KEYDEF *keyinfo, - uchar *key, uchar *curr_buff, uchar *father_buff, - uchar *father_key_pos, my_off_t father_page) + byte *key, byte *curr_buff, byte *father_buff, + byte *father_key_pos, my_off_t father_page) { my_bool right; uint k_length,father_length,father_keylength,nod_flag,curr_keylength, right_length,left_length,new_right_length,new_left_length,extra_length, length,keys; - uchar *pos,*buff,*extra_buff; + byte *pos,*buff,*extra_buff; my_off_t next_page,new_pos; byte tmp_part_key[HA_MAX_KEY_BUFF]; DBUG_ENTER("_ma_balance_page"); @@ -880,7 +921,8 @@ typedef struct { uint keynr; } bulk_insert_param; -int _ma_ck_write_tree(register MARIA_HA *info, uint keynr, uchar *key, + +int _ma_ck_write_tree(register MARIA_HA *info, uint keynr, byte *key, uint key_length) { int error; @@ -896,22 +938,22 @@ int _ma_ck_write_tree(register MARIA_HA *info, uint keynr, uchar *key, /* typeof(_ma_keys_compare)=qsort_cmp2 */ -static int keys_compare(bulk_insert_param *param, uchar *key1, uchar *key2) +static int keys_compare(bulk_insert_param *param, byte *key1, byte *key2) { uint not_used[2]; return ha_key_cmp(param->info->s->keyinfo[param->keynr].seg, - key1, key2, USE_WHOLE_KEY, SEARCH_SAME, + (uchar*) key1, (uchar*) key2, USE_WHOLE_KEY, SEARCH_SAME, not_used); } -static int keys_free(uchar *key, TREE_FREE mode, bulk_insert_param *param) +static int keys_free(byte *key, TREE_FREE mode, bulk_insert_param *param) { /* Probably I can use info->lastkey here, but I'm not sure, and to be safe I'd better use local lastkey. */ - uchar lastkey[HA_MAX_KEY_BUFF]; + byte lastkey[HA_MAX_KEY_BUFF]; uint keylen; MARIA_KEYDEF *keyinfo; diff --git a/storage/maria/maria_chk.c b/storage/maria/maria_chk.c index e423a3f5c36..8ea00fa9776 100644 --- a/storage/maria/maria_chk.c +++ b/storage/maria/maria_chk.c @@ -36,21 +36,23 @@ SET_STACK_SIZE(9000) /* Minimum stack size for program */ static uint decode_bits; static char **default_argv; -static const char *load_default_groups[]= { "mariachk", 0 }; +static const char *load_default_groups[]= { "maria_chk", 0 }; static const char *set_collation_name, *opt_tmpdir; static CHARSET_INFO *set_collation; static long opt_maria_block_size; static long opt_key_cache_block_size; static const char *my_progname_short; static int stopwords_inited= 0; -static MY_TMPDIR mariachk_tmpdir; +static MY_TMPDIR maria_chk_tmpdir; static const char *type_names[]= -{ "impossible","char","binary", "short", "long", "float", +{ + "impossible","char","binary", "short", "long", "float", "double","number","unsigned short", "unsigned long","longlong","ulonglong","int24", "uint24","int8","varchar", "varbin","?", - "?"}; + "?" +}; static const char *prefix_packed_txt="packed ", *bin_packed_txt="prefix ", @@ -59,23 +61,30 @@ static const char *prefix_packed_txt="packed ", *blob_txt="BLOB "; static const char *field_pack[]= -{"","no endspace", "no prespace", +{ + "","no endspace", "no prespace", "no zeros", "blob", "constant", "table-lockup", - "always zero","varchar","unique-hash","?","?"}; + "always zero","varchar","unique-hash","?","?" +}; + +static const char *record_formats[]= +{ + "Fixed length", "Packed", "Compressed", "Block", "?" +}; static const char *maria_stats_method_str="nulls_unequal"; static void get_options(int *argc,char * * *argv); static void print_version(void); static void usage(void); -static int mariachk(HA_CHECK *param, char *filename); +static int maria_chk(HA_CHECK *param, char *filename); static void descript(HA_CHECK *param, register MARIA_HA *info, my_string name); static int maria_sort_records(HA_CHECK *param, register MARIA_HA *info, - my_string name, uint sort_key, - my_bool write_info, my_bool update_index); + my_string name, uint sort_key, + my_bool write_info, my_bool update_index); static int sort_record_index(MARIA_SORT_PARAM *sort_param, MARIA_HA *info, MARIA_KEYDEF *keyinfo, - my_off_t page,uchar *buff,uint sortkey, + my_off_t page, byte *buff,uint sortkey, File new_file, my_bool update_index); HA_CHECK check_param; @@ -88,7 +97,7 @@ int main(int argc, char **argv) MY_INIT(argv[0]); my_progname_short= my_progname+dirname_length(my_progname); - mariachk_init(&check_param); + maria_chk_init(&check_param); check_param.opt_lock_memory= 1; /* Lock memory if possible */ check_param.using_global_keycache = 0; get_options(&argc,(char***) &argv); @@ -98,7 +107,7 @@ int main(int argc, char **argv) while (--argc >= 0) { - int new_error=mariachk(&check_param, *(argv++)); + int new_error=maria_chk(&check_param, *(argv++)); if ((check_param.testflag & T_REP_ANY) != T_REP) check_param.testflag&= ~T_REP; VOID(fflush(stdout)); @@ -112,7 +121,7 @@ int main(int argc, char **argv) if (!(check_param.testflag & T_REP)) check_param.testflag|= T_REP_BY_SORT; check_param.testflag&= ~T_EXTEND; /* Don't needed */ - error|=mariachk(&check_param, argv[-1]); + error|=maria_chk(&check_param, argv[-1]); check_param.testflag= old_testflag; VOID(fflush(stdout)); VOID(fflush(stderr)); @@ -134,7 +143,7 @@ int main(int argc, char **argv) llstr(check_param.total_deleted,buff2)); } free_defaults(default_argv); - free_tmpdir(&mariachk_tmpdir); + free_tmpdir(&maria_chk_tmpdir); maria_end(); my_end(check_param.testflag & T_INFO ? MY_CHECK_ERROR | MY_GIVE_INFO : MY_CHECK_ERROR); @@ -218,7 +227,7 @@ static struct my_option my_long_options[] = (gptr*) &check_param.keys_in_use, 0, GET_ULL, REQUIRED_ARG, -1, 0, 0, 0, 0, 0}, {"max-record-length", OPT_MAX_RECORD_LENGTH, - "Skip rows bigger than this if mariachk can't allocate memory to hold it", + "Skip rows bigger than this if maria_chk can't allocate memory to hold it", (gptr*) &check_param.max_record_length, (gptr*) &check_param.max_record_length, 0, GET_ULL, REQUIRED_ARG, LONGLONG_MAX, 0, LONGLONG_MAX, 0, 0, 0}, @@ -259,7 +268,7 @@ static struct my_option my_long_options[] = "Change the value of a variable. Please note that this option is deprecated; you can set variables directly with --variable-name=value.", 0, 0, 0, GET_STR, REQUIRED_ARG, 0, 0, 0, 0, 0, 0}, {"silent", 's', - "Only print errors. One can use two -s to make mariachk very silent.", + "Only print errors. One can use two -s to make maria_chk very silent.", 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, {"sort-index", 'S', "Sort index blocks. This speeds up 'read-next' in applications.", @@ -346,7 +355,7 @@ static struct my_option my_long_options[] = static void print_version(void) { - printf("%s Ver 2.7 for %s at %s\n", my_progname, SYSTEM_TYPE, + printf("%s Ver 1.0 for %s at %s\n", my_progname, SYSTEM_TYPE, MACHINE_TYPE); NETWARE_SET_SCREEN_MODE(1); } @@ -381,7 +390,7 @@ static void usage(void) printf(", they will be used\n\ in a round-robin fashion.\n\ -s, --silent Only print errors. One can use two -s to make\n\ - mariachk very silent.\n\ + maria_chk very silent.\n\ -v, --verbose Print more information. This can be used with\n\ --description and --check. Use many -v for more verbosity.\n\ -V, --version Print version and exit.\n\ @@ -390,10 +399,10 @@ static void usage(void) puts(" --start-check-pos=# Start reading file at given offset.\n"); #endif - puts("Check options (check is the default action for mariachk):\n\ + puts("Check options (check is the default action for maria_chk):\n\ -c, --check Check table for errors.\n\ -e, --extend-check Check the table VERY throughly. Only use this in\n\ - extreme cases as mariachk should normally be able to\n\ + extreme cases as maria_chk should normally be able to\n\ find out if the table is ok even without this switch.\n\ -F, --fast Check only tables that haven't been closed properly.\n\ -C, --check-only-changed\n\ @@ -419,7 +428,7 @@ static void usage(void) bit mask of which keys to use. This can be used to\n\ get faster inserts.\n\ --max-record-length=#\n\ - Skip rows bigger than this if mariachk can't allocate\n\ + Skip rows bigger than this if maria_chk can't allocate\n\ memory to hold it.\n\ -r, --recover Can fix almost anything except unique keys that aren't\n\ unique.\n\ @@ -436,7 +445,7 @@ static void usage(void) --set-collation=name\n\ Change the collation used by the index.\n\ -q, --quick Faster repair by not modifying the data file.\n\ - One can give a second '-q' to force mariachk to\n\ + One can give a second '-q' to force maria_chk to\n\ modify the original datafile in case of duplicate keys.\n\ NOTE: Tables where the data file is currupted can't be\n\ fixed with this option.\n\ @@ -684,7 +693,7 @@ get_one_option(int optid, } else { - DBUG_PUSH(argument ? argument : "d:t:o,/tmp/mariachk.trace"); + DBUG_PUSH(argument ? argument : "d:t:o,/tmp/maria_chk.trace"); } break; case 'V': @@ -700,6 +709,7 @@ get_one_option(int optid, { int method; enum_handler_stats_method method_conv; + LINT_INIT(method_conv); maria_stats_method_str= argument; if ((method=find_type(argument, &maria_stats_method_typelib, 2)) <= 0) { @@ -778,10 +788,10 @@ static void get_options(register int *argc,register char ***argv) exit(1); } - if (init_tmpdir(&mariachk_tmpdir, opt_tmpdir)) + if (init_tmpdir(&maria_chk_tmpdir, opt_tmpdir)) exit(1); - check_param.tmpdir=&mariachk_tmpdir; + check_param.tmpdir=&maria_chk_tmpdir; check_param.key_cache_block_size= opt_key_cache_block_size; if (set_collation_name) @@ -796,17 +806,16 @@ static void get_options(register int *argc,register char ***argv) /* Check table */ -static int mariachk(HA_CHECK *param, my_string filename) +static int maria_chk(HA_CHECK *param, my_string filename) { int error,lock_type,recreate; int rep_quick= param->testflag & (T_QUICK | T_FORCE_UNIQUENESS); - uint raid_chunks; MARIA_HA *info; File datafile; char llbuff[22],llbuff2[22]; my_bool state_updated=0; MARIA_SHARE *share; - DBUG_ENTER("mariachk"); + DBUG_ENTER("maria_chk"); param->out_flag=error=param->warning_printed=param->error_printed= recreate=0; @@ -849,7 +858,8 @@ static int mariachk(HA_CHECK *param, my_string filename) _ma_check_print_error(param,"File '%s' doesn't exist",filename); break; case EACCES: - _ma_check_print_error(param,"You don't have permission to use '%s'",filename); + _ma_check_print_error(param,"You don't have permission to use '%s'", + filename); break; default: _ma_check_print_error(param,"%d when opening MARIA-table '%s'", @@ -862,7 +872,18 @@ static int mariachk(HA_CHECK *param, my_string filename) share->options&= ~HA_OPTION_READ_ONLY_DATA; /* We are modifing it */ share->tot_locks-= share->r_locks; share->r_locks=0; - raid_chunks=share->base.raid_chunks; + + if (share->data_file_type == BLOCK_RECORD && + (param->testflag & (T_REP_ANY | T_SORT_RECORDS | T_FAST | T_STATISTICS | + T_CHECK | T_CHECK_ONLY_CHANGED))) + { + _ma_check_print_error(param, + "Record format used by '%s' is is not yet supported with repair/check", + filename); + param->error_printed= 0; + error= 1; + goto end2; + } /* Skip the checking of the file if: @@ -871,7 +892,8 @@ static int mariachk(HA_CHECK *param, my_string filename) */ if (param->testflag & (T_FAST | T_CHECK_ONLY_CHANGED)) { - my_bool need_to_check= maria_is_crashed(info) || share->state.open_count != 0; + my_bool need_to_check= (maria_is_crashed(info) || + share->state.open_count != 0); if ((param->testflag & (T_REP_ANY | T_SORT_RECORDS)) && ((share->state.changed & (STATE_CHANGED | STATE_CRASHED | @@ -969,6 +991,7 @@ static int mariachk(HA_CHECK *param, my_string filename) _ma_check_print_error(param,"Can't lock indexfile of '%s', error: %d", filename,my_errno); param->error_printed=0; + error= 1; goto end2; } /* @@ -989,7 +1012,8 @@ static int mariachk(HA_CHECK *param, my_string filename) if (tmp != share->state.key_map) info->update|=HA_STATE_CHANGED; } - if (rep_quick && maria_chk_del(param, info, param->testflag & ~T_VERBOSE)) + if (rep_quick && + maria_chk_del(param, info, param->testflag & ~T_VERBOSE)) { if (param->testflag & T_FORCE_CREATE) { @@ -1032,8 +1056,7 @@ static int mariachk(HA_CHECK *param, my_string filename) { /* Change temp file to org file */ VOID(my_close(info->dfile,MYF(MY_WME))); /* Close new file */ error|=maria_change_to_newfile(filename,MARIA_NAME_DEXT,DATA_TMP_EXT, - raid_chunks, - MYF(0)); + MYF(0)); if (_ma_open_datafile(info,info->s, -1)) error=1; param->out_flag&= ~O_NEW_DATA; /* We are using new datafile */ @@ -1112,8 +1135,7 @@ static int mariachk(HA_CHECK *param, my_string filename) 1, MYF(MY_WME))); maria_lock_memory(param); - if ((info->s->options & (HA_OPTION_PACK_RECORD | - HA_OPTION_COMPRESS_RECORD)) || + if ((info->s->data_file_type != STATIC_RECORD) || (param->testflag & (T_EXTEND | T_MEDIUM))) error|=maria_chk_data_link(param, info, param->testflag & T_EXTEND); error|=_ma_flush_blocks(param, share->key_cache, share->kfile); @@ -1163,12 +1185,11 @@ end2: { if (param->out_flag & O_NEW_DATA) error|=maria_change_to_newfile(filename,MARIA_NAME_DEXT,DATA_TMP_EXT, - raid_chunks, - ((param->testflag & T_BACKUP_DATA) ? - MYF(MY_REDEL_MAKE_BACKUP) : MYF(0))); + ((param->testflag & T_BACKUP_DATA) ? + MYF(MY_REDEL_MAKE_BACKUP) : MYF(0))); if (param->out_flag & O_NEW_INDEX) - error|=maria_change_to_newfile(filename,MARIA_NAME_IEXT,INDEX_TMP_EXT,0, - MYF(0)); + error|=maria_change_to_newfile(filename,MARIA_NAME_IEXT,INDEX_TMP_EXT, + MYF(0)); } VOID(fflush(stdout)); VOID(fflush(stderr)); if (param->error_printed) @@ -1195,10 +1216,10 @@ end2: filename)); VOID(fflush(stderr)); DBUG_RETURN(error); -} /* mariachk */ +} /* maria_chk */ - /* Write info about table */ +/* Write info about table */ static void descript(HA_CHECK *param, register MARIA_HA *info, my_string name) { @@ -1212,14 +1233,8 @@ static void descript(HA_CHECK *param, register MARIA_HA *info, my_string name) char llbuff[22],llbuff2[22]; DBUG_ENTER("describe"); - printf("\nMARIA file: %s\n",name); - fputs("Record format: ",stdout); - if (share->options & HA_OPTION_COMPRESS_RECORD) - puts("Compressed"); - else if (share->options & HA_OPTION_PACK_RECORD) - puts("Packed"); - else - puts("Fixed length"); + printf("\nMARIA file: %s\n",name); + printf("Record format: %s\n", record_formats[share->data_file_type]); printf("Character set: %s (%d)\n", get_charset_name(share->state.header.language), share->state.header.language); @@ -1260,25 +1275,18 @@ static void descript(HA_CHECK *param, register MARIA_HA *info, my_string name) printf("Status: %s\n",buff); if (share->base.auto_key) { - printf("Auto increment key: %13d Last value: %13s\n", + printf("Auto increment key: %16d Last value: %18s\n", share->base.auto_key, llstr(share->state.auto_increment,llbuff)); } - if (share->base.raid_type) - { - printf("RAID: Type: %u Chunks: %u Chunksize: %lu\n", - share->base.raid_type, - share->base.raid_chunks, - share->base.raid_chunksize); - } if (share->options & (HA_OPTION_CHECKSUM | HA_OPTION_COMPRESS_RECORD)) - printf("Checksum: %23s\n",llstr(info->state->checksum,llbuff)); + printf("Checksum: %26s\n",llstr(info->state->checksum,llbuff)); ; if (share->options & HA_OPTION_DELAY_KEY_WRITE) printf("Keys are only flushed at close\n"); } - printf("Data records: %13s Deleted blocks: %13s\n", + printf("Data records: %16s Deleted blocks: %18s\n", llstr(info->state->records,llbuff),llstr(info->state->del,llbuff2)); if (param->testflag & T_SILENT) DBUG_VOID_RETURN; /* This is enough */ @@ -1286,14 +1294,14 @@ static void descript(HA_CHECK *param, register MARIA_HA *info, my_string name) if (param->testflag & T_VERBOSE) { #ifdef USE_RELOC - printf("Init-relocation: %13s\n",llstr(share->base.reloc,llbuff)); + printf("Init-relocation: %16s\n",llstr(share->base.reloc,llbuff)); #endif - printf("Datafile parts: %13s Deleted data: %13s\n", + printf("Datafile parts: %16s Deleted data: %18s\n", llstr(share->state.split,llbuff), llstr(info->state->empty,llbuff2)); - printf("Datafile pointer (bytes):%9d Keyfile pointer (bytes):%9d\n", + printf("Datafile pointer (bytes): %11d Keyfile pointer (bytes): %13d\n", share->rec_reflength,share->base.key_reflength); - printf("Datafile length: %13s Keyfile length: %13s\n", + printf("Datafile length: %16s Keyfile length: %18s\n", llstr(info->state->data_file_length,llbuff), llstr(info->state->key_file_length,llbuff2)); @@ -1303,13 +1311,13 @@ static void descript(HA_CHECK *param, register MARIA_HA *info, my_string name) { if (share->base.max_data_file_length != HA_OFFSET_ERROR || share->base.max_key_file_length != HA_OFFSET_ERROR) - printf("Max datafile length: %13s Max keyfile length: %13s\n", + printf("Max datafile length: %16s Max keyfile length: %18s\n", llstr(share->base.max_data_file_length-1,llbuff), llstr(share->base.max_key_file_length-1,llbuff2)); } } - - printf("Recordlength: %13d\n",(int) share->base.pack_reclength); + printf("Block_size: %16d\n",(int) share->block_size); + printf("Recordlength: %16d\n",(int) share->base.pack_reclength); if (! maria_is_all_keys_active(share->state.key_map, share->base.keys)) { longlong2str(share->state.key_map,buff,2); @@ -1417,7 +1425,7 @@ static void descript(HA_CHECK *param, register MARIA_HA *info, my_string name) if (share->options & HA_OPTION_COMPRESS_RECORD) printf(" Huff tree Bits"); VOID(putchar('\n')); - start=1; + for (field=0 ; field < share->base.fields ; field++) { if (share->options & HA_OPTION_COMPRESS_RECORD) @@ -1446,8 +1454,9 @@ static void descript(HA_CHECK *param, register MARIA_HA *info, my_string name) sprintf(null_bit,"%d",share->rec[field].null_bit); sprintf(null_pos,"%d",share->rec[field].null_pos+1); } - printf("%-6d%-6d%-7s%-8s%-8s%-35s",field+1,start,length, - null_pos, null_bit, buff); + printf("%-6d%-6u%-7s%-8s%-8s%-35s",field+1, + (uint) share->rec[field].offset+1, + length, null_pos, null_bit, buff); if (share->options & HA_OPTION_COMPRESS_RECORD) { if (share->rec[field].huff_tree) @@ -1475,7 +1484,7 @@ static int maria_sort_records(HA_CHECK *param, uint key; MARIA_KEYDEF *keyinfo; File new_file; - uchar *temp_buff; + byte *temp_buff; ha_rows old_record_count; MARIA_SHARE *share=info->s; char llbuff[22],llbuff2[22]; @@ -1532,7 +1541,7 @@ static int maria_sort_records(HA_CHECK *param, goto err; info->opt_flag|=WRITE_CACHE_USED; - if (!(temp_buff=(uchar*) my_alloca((uint) keyinfo->block_length))) + if (!(temp_buff=(byte*) my_alloca((uint) keyinfo->block_length))) { _ma_check_print_error(param,"Not enough memory for key block"); goto err; @@ -1544,14 +1553,11 @@ static int maria_sort_records(HA_CHECK *param, goto err; } fn_format(param->temp_filename,name,"", MARIA_NAME_DEXT,2+4+32); - new_file=my_raid_create(fn_format(param->temp_filename, - param->temp_filename,"", - DATA_TMP_EXT,2+4), - 0,param->tmpfile_createflag, - share->base.raid_type, - share->base.raid_chunks, - share->base.raid_chunksize, - MYF(0)); + new_file= my_create(fn_format(param->temp_filename, + param->temp_filename,"", + DATA_TMP_EXT,2+4), + 0,param->tmpfile_createflag, + MYF(0)); if (new_file < 0) { _ma_check_print_error(param,"Can't create new tempfile: '%s'", @@ -1568,7 +1574,7 @@ static int maria_sort_records(HA_CHECK *param, for (key=0 ; key < share->base.keys ; key++) share->keyinfo[key].flag|= HA_SORT_ALLOWS_SAME; - if (my_pread(share->kfile,(byte*) temp_buff, + if (my_pread(share->kfile, temp_buff, (uint) keyinfo->block_length, share->state.key_root[sort_key], MYF(MY_NABP+MY_WME))) @@ -1589,7 +1595,8 @@ static int maria_sort_records(HA_CHECK *param, if (sort_info.new_data_file_type != COMPRESSED_RECORD) info->state->checksum=0; - if (sort_record_index(&sort_param,info,keyinfo,share->state.key_root[sort_key], + if (sort_record_index(&sort_param,info,keyinfo, + share->state.key_root[sort_key], temp_buff, sort_key,new_file,update_index) || maria_write_data_suffix(&sort_info,1) || flush_io_cache(&info->rec_cache)) @@ -1626,8 +1633,7 @@ err: { VOID(end_io_cache(&info->rec_cache)); (void) my_close(new_file,MYF(MY_WME)); - (void) my_raid_delete(param->temp_filename, share->base.raid_chunks, - MYF(MY_WME)); + (void) my_delete(param->temp_filename, MYF(MY_WME)); } if (temp_buff) { @@ -1644,17 +1650,17 @@ err: } /* sort_records */ - /* Sort records recursive using one index */ +/* Sort records recursive using one index */ static int sort_record_index(MARIA_SORT_PARAM *sort_param,MARIA_HA *info, MARIA_KEYDEF *keyinfo, - my_off_t page, uchar *buff, uint sort_key, + my_off_t page, byte *buff, uint sort_key, File new_file,my_bool update_index) { uint nod_flag,used_length,key_length; - uchar *temp_buff,*keypos,*endpos; + byte *temp_buff,*keypos,*endpos; my_off_t next_page,rec_pos; - uchar lastkey[HA_MAX_KEY_BUFF]; + byte lastkey[HA_MAX_KEY_BUFF]; char llbuff[22]; MARIA_SORT_INFO *sort_info= sort_param->sort_info; HA_CHECK *param=sort_info->param; @@ -1665,7 +1671,7 @@ static int sort_record_index(MARIA_SORT_PARAM *sort_param,MARIA_HA *info, if (nod_flag) { - if (!(temp_buff=(uchar*) my_alloca((uint) keyinfo->block_length))) + if (!(temp_buff= (byte*) my_alloca((uint) keyinfo->block_length))) { _ma_check_print_error(param,"Not Enough memory"); DBUG_RETURN(-1); @@ -1679,7 +1685,7 @@ static int sort_record_index(MARIA_SORT_PARAM *sort_param,MARIA_HA *info, _sanity(__FILE__,__LINE__); if (nod_flag) { - next_page= _ma_kpos(nod_flag,keypos); + next_page= _ma_kpos(nod_flag, keypos); if (my_pread(info->s->kfile,(byte*) temp_buff, (uint) keyinfo->block_length, next_page, MYF(MY_NABP+MY_WME))) @@ -1688,7 +1694,8 @@ static int sort_record_index(MARIA_SORT_PARAM *sort_param,MARIA_HA *info, llstr(next_page,llbuff)); goto err; } - if (sort_record_index(sort_param, info,keyinfo,next_page,temp_buff,sort_key, + if (sort_record_index(sort_param, info,keyinfo,next_page,temp_buff, + sort_key, new_file, update_index)) goto err; } @@ -1699,7 +1706,7 @@ static int sort_record_index(MARIA_SORT_PARAM *sort_param,MARIA_HA *info, break; rec_pos= _ma_dpos(info,0,lastkey+key_length); - if ((*info->s->read_rnd)(info,sort_param->record,rec_pos,0)) + if ((*info->s->read_record)(info,sort_param->record,rec_pos)) { _ma_check_print_error(param,"%d when reading datafile",my_errno); goto err; @@ -1738,7 +1745,7 @@ err: /* - Check if mariachk was killed by a signal + Check if maria_chk was killed by a signal This is overloaded by other programs that want to be able to abort sorting */ diff --git a/storage/maria/maria_def.h b/storage/maria/maria_def.h index ecd93807a06..778c5817e4f 100644 --- a/storage/maria/maria_def.h +++ b/storage/maria/maria_def.h @@ -55,14 +55,15 @@ typedef struct st_maria_state_info uchar keys; /* number of keys in file */ uchar uniques; /* number of UNIQUE definitions */ uchar language; /* Language for indexes */ - uchar max_block_size_index; /* max keyblock size */ uchar fulltext_keys; - uchar not_used; /* To align to 8 */ + uchar data_file_type; + uchar org_data_file_type; /* Used by mariapack to store dft */ } header; MARIA_STATUS_INFO state; ha_rows split; /* number of split blocks */ my_off_t dellink; /* Link to next removed block */ + ulonglong first_bitmap_with_space; ulonglong auto_increment; ulong process; /* process that updated table last */ ulong unique; /* Unique number for this process */ @@ -70,7 +71,7 @@ typedef struct st_maria_state_info ulong status; ulong *rec_per_key_part; my_off_t *key_root; /* Start of key trees */ - my_off_t *key_del; /* delete links for trees */ + my_off_t key_del; /* delete links for trees */ my_off_t rec_per_key_rows; /* Rows when calculating rec_per_key */ ulong sec_index_changed; /* Updated when new sec_index */ @@ -91,60 +92,73 @@ typedef struct st_maria_state_info } MARIA_STATE_INFO; -#define MARIA_STATE_INFO_SIZE (24+14*8+7*4+2*2+8) +#define MARIA_STATE_INFO_SIZE (24 + 4 + 11*8 + 4*4 + 8 + 3*4 + 5*8) #define MARIA_STATE_KEY_SIZE 8 #define MARIA_STATE_KEYBLOCK_SIZE 8 #define MARIA_STATE_KEYSEG_SIZE 4 -#define MARIA_STATE_EXTRA_SIZE ((MARIA_MAX_KEY+MARIA_MAX_KEY_BLOCK_SIZE)*MARIA_STATE_KEY_SIZE + MARIA_MAX_KEY*HA_MAX_KEY_SEG*MARIA_STATE_KEYSEG_SIZE) +#define MARIA_STATE_EXTRA_SIZE (MARIA_MAX_KEY*MARIA_STATE_KEY_SIZE + MARIA_MAX_KEY*HA_MAX_KEY_SEG*MARIA_STATE_KEYSEG_SIZE) #define MARIA_KEYDEF_SIZE (2+ 5*2) #define MARIA_UNIQUEDEF_SIZE (2+1+1) #define HA_KEYSEG_SIZE (6+ 2*2 + 4*2) -#define MARIA_COLUMNDEF_SIZE (2*3+1) -#define MARIA_BASE_INFO_SIZE (5*8 + 8*4 + 4 + 4*2 + 16) +#define MARIA_COLUMNDEF_SIZE (6+2+2+2+2+2+1+1) +#define MARIA_BASE_INFO_SIZE (5*8 + 6*4 + 11*2 + 6 + 5*2 + 1 + 16) #define MARIA_INDEX_BLOCK_MARGIN 16 /* Safety margin for .MYI tables */ -typedef struct st__ma_base_info +typedef struct st_ma_base_info { - my_off_t keystart; /* Start of keys */ + my_off_t keystart; /* Start of keys */ my_off_t max_data_file_length; my_off_t max_key_file_length; my_off_t margin_key_file_length; - ha_rows records, reloc; /* Create information */ - ulong mean_row_length; /* Create information */ - ulong reclength; /* length of unpacked record */ - ulong pack_reclength; /* Length of full packed rec */ + ha_rows records, reloc; /* Create information */ + ulong mean_row_length; /* Create information */ + ulong reclength; /* length of unpacked record */ + ulong pack_reclength; /* Length of full packed rec */ ulong min_pack_length; - ulong max_pack_length; /* Max possibly length of - packed rec. */ + ulong max_pack_length; /* Max possibly length of packed rec */ ulong min_block_length; - ulong fields, /* fields in table */ - pack_fields; /* packed fields in table */ - uint rec_reflength; /* = 2-8 */ - uint key_reflength; /* = 2-8 */ - uint keys; /* same as in state.header */ - uint auto_key; /* Which key-1 is a auto key */ - uint blobs; /* Number of blobs */ - uint pack_bits; /* Length of packed bits */ - uint max_key_block_length; /* Max block length */ - uint max_key_length; /* Max key length */ + uint fields; /* fields in table */ + uint fixed_not_null_fields; + uint fixed_not_null_fields_length; + uint max_field_lengths; + uint pack_fields; /* packed fields in table */ + uint varlength_fields; /* char/varchar/blobs */ + uint rec_reflength; /* = 2-8 */ + uint key_reflength; /* = 2-8 */ + uint keys; /* same as in state.header */ + uint auto_key; /* Which key-1 is a auto key */ + uint blobs; /* Number of blobs */ + /* Length of packed bits (when table was created first time) */ + uint pack_bytes; + /* Length of null bits (when table was created first time) */ + uint original_null_bytes; + uint null_bytes; /* Null bytes in record */ + uint field_offsets; /* Number of field offsets */ + uint max_key_block_length; /* Max block length */ + uint max_key_length; /* Max key length */ /* Extra allocation when using dynamic record format */ uint extra_alloc_bytes; uint extra_alloc_procent; - /* Info about raid */ - uint raid_type, raid_chunks; - ulong raid_chunksize; + uint is_nulls_extended; /* 1 if new null bytes */ + uint min_row_length; + uint default_row_flag; /* 0 or ROW_FLAG_NULLS_EXTENDED */ + uint block_size; + uint default_rec_buff_size; + uint extra_rec_buff_size; + /* The following are from the header */ uint key_parts, all_key_parts; + my_bool transactional; } MARIA_BASE_INFO; - /* Structs used intern in database */ +/* Structs used intern in database */ -typedef struct st_maria_blob /* Info of record */ +typedef struct st_maria_blob /* Info of record */ { - ulong offset; /* Offset to blob in record */ - uint pack_length; /* Type of packed length */ - ulong length; /* Calc:ed for each record */ + ulong offset; /* Offset to blob in record */ + uint pack_length; /* Type of packed length */ + ulong length; /* Calc:ed for each record */ } MARIA_BLOB; @@ -155,6 +169,26 @@ typedef struct st_maria_pack uchar version; } MARIA_PACK; +typedef struct st_maria_file_bitmap +{ + uchar *map; + ulonglong page; /* Page number for current bitmap */ + uint used_size; /* Size of bitmap that is not 0 */ + File file; + + my_bool changed; + +#ifdef THREAD + pthread_mutex_t bitmap_lock; +#endif + /* Constants, allocated when initiating bitmaps */ + uint sizes[8]; /* Size per bit combination */ + uint total_size; /* Total usable size of bitmap page */ + uint block_size; /* Block size of file */ + ulong pages_covered; /* Pages covered by bitmap + 1 */ +} MARIA_FILE_BITMAP; + + #define MAX_NONMAPPED_INSERTS 1000 typedef struct st_maria_share @@ -175,28 +209,36 @@ typedef struct st_maria_share symlinks */ *index_file_name; byte *file_map; /* mem-map of file if possible */ - KEY_CACHE *key_cache; /* ref to the current key cache - */ + KEY_CACHE *key_cache; /* ref to the current key cache */ MARIA_DECODE_TREE *decode_trees; uint16 *decode_tables; - int(*read_record) (struct st_maria_info *, my_off_t, byte *); - int(*write_record) (struct st_maria_info *, const byte *); - int(*update_record) (struct st_maria_info *, my_off_t, const byte *); - int(*delete_record) (struct st_maria_info *); - int(*read_rnd) (struct st_maria_info *, byte *, my_off_t, my_bool); - int(*compare_record) (struct st_maria_info *, const byte *); - ha_checksum(*calc_checksum) (struct st_maria_info *, const byte *); - int(*compare_unique) (struct st_maria_info *, MARIA_UNIQUEDEF *, - const byte *record, my_off_t pos); - uint(*file_read) (MARIA_HA *, byte *, uint, my_off_t, myf); - uint(*file_write) (MARIA_HA *, byte *, uint, my_off_t, myf); + my_bool (*once_init)(struct st_maria_share *, File); + my_bool (*once_end)(struct st_maria_share *); + my_bool (*init)(struct st_maria_info *); + void (*end)(struct st_maria_info *); + int (*read_record)(struct st_maria_info *, byte *, MARIA_RECORD_POS); + my_bool (*scan_init)(struct st_maria_info *); + int (*scan)(struct st_maria_info *, byte *, MARIA_RECORD_POS, my_bool); + void (*scan_end)(struct st_maria_info *); + MARIA_RECORD_POS (*write_record_init)(struct st_maria_info *, const byte *); + my_bool (*write_record)(struct st_maria_info *, const byte *); + my_bool (*write_record_abort)(struct st_maria_info *); + my_bool (*update_record)(struct st_maria_info *, MARIA_RECORD_POS, + const byte *); + my_bool (*delete_record)(struct st_maria_info *); + my_bool (*compare_record)(struct st_maria_info *, const byte *); + ha_checksum(*calc_checksum) (struct st_maria_info *, const byte *); + ha_checksum(*calc_write_checksum) (struct st_maria_info *, const byte *); + my_bool (*compare_unique) (struct st_maria_info *, MARIA_UNIQUEDEF *, + const byte *record, MARIA_RECORD_POS pos); + uint(*file_read) (MARIA_HA *, byte *, uint, my_off_t, myf); + uint(*file_write) (MARIA_HA *, byte *, uint, my_off_t, myf); invalidator_by_filename invalidator; /* query cache invalidator */ ulong this_process; /* processid */ ulong last_process; /* For table-change-check */ ulong last_version; /* Version on start */ ulong options; /* Options used */ - ulong min_pack_length; /* Theese are used by packed - data */ + ulong min_pack_length; /* These are used by packed data */ ulong max_pack_length; ulong state_diff_length; uint rec_reflength; /* rec_reflength in use now */ @@ -208,21 +250,24 @@ typedef struct st_maria_share int mode; /* mode of file on open */ uint reopen; /* How many times reopened */ uint w_locks, r_locks, tot_locks; /* Number of read/write locks */ - uint blocksize; /* blocksize of keyfile */ + uint block_size; /* block_size of keyfile & data file*/ + uint base_length; myf write_flag; enum data_file_type data_file_type; + my_bool temporary; my_bool changed, /* If changed since lock */ global_changed, /* If changed since open */ - not_flushed, temporary, delay_key_write, concurrent_insert; + not_flushed, concurrent_insert; + my_bool delay_key_write; #ifdef THREAD THR_LOCK lock; - pthread_mutex_t intern_lock; /* Locking for use with - _locking */ + pthread_mutex_t intern_lock; /* Locking for use with _locking */ rw_lock_t *key_root_lock; #endif my_off_t mmaped_length; uint nonmmaped_inserts; /* counter of writing in non-mmaped area */ + MARIA_FILE_BITMAP bitmap; rw_lock_t mmap_lock; } MARIA_SHARE; @@ -237,45 +282,106 @@ typedef struct st_maria_bit_buff uint error; } MARIA_BIT_BUFF; +typedef byte MARIA_BITMAP_BUFFER; + +typedef struct st_maria_bitmap_block +{ + ulonglong page; /* Page number */ + /* Number of continuous pages. TAIL_BIT is set if this is a tail page */ + uint page_count; + uint empty_space; /* Set for head and tail pages */ + /* + Number of BLOCKS for block-region (holds all non-blob-fields or one blob) + */ + uint sub_blocks; + /* set to <> 0 in write_record() if this block was actually used */ + uint8 used; + uint8 org_bitmap_value; +} MARIA_BITMAP_BLOCK; + + +typedef struct st_maria_bitmap_blocks +{ + MARIA_BITMAP_BLOCK *block; + uint count; + my_bool tail_page_skipped; /* If some tail pages was not used */ + my_bool page_skipped; /* If some full pages was not used */ +} MARIA_BITMAP_BLOCKS; + + +/* Data about the currently read row */ +typedef struct st_maria_row +{ + MARIA_BITMAP_BLOCKS insert_blocks; + MARIA_BITMAP_BUFFER *extents; + MARIA_RECORD_POS lastpos, nextpos; + MARIA_RECORD_POS *tail_positions; + ha_checksum checksum; + byte *empty_bits, *field_lengths; + byte *empty_bits_buffer; /* For storing cur_row.empty_bits */ + uint *null_field_lengths; /* All null field lengths */ + ulong *blob_lengths; /* Length for each blob */ + ulong base_length, normal_length, char_length, varchar_length, blob_length; + ulong head_length, total_length; + my_size_t extents_buffer_length; /* Size of 'extents' buffer */ + uint field_lengths_length; /* Length of data in field_lengths */ + uint extents_count; /* number of extents in 'extents' */ + uint full_page_count, tail_count; /* For maria_chk */ +} MARIA_ROW; + +/* Data to scan row in blocked format */ +typedef struct st_maria_block_scan +{ + byte *bitmap_buff, *bitmap_pos, *bitmap_end, *page_buff; + byte *dir, *dir_end; + ulong bitmap_page; + ulonglong bits; + uint number_of_rows, bit_pos; + MARIA_RECORD_POS row_base_page; +} MARIA_BLOCK_SCAN; + + struct st_maria_info { MARIA_SHARE *s; /* Shared between open:s */ MARIA_STATUS_INFO *state, save_state; + MARIA_ROW cur_row, new_row; + MARIA_BLOCK_SCAN scan; MARIA_BLOB *blobs; /* Pointer to blobs */ MARIA_BIT_BUFF bit_buff; + DYNAMIC_ARRAY bitmap_blocks; /* accumulate indexfile changes between write's */ TREE *bulk_insert; DYNAMIC_ARRAY *ft1_to_ft2; /* used only in ft1->ft2 conversion */ MEM_ROOT ft_memroot; /* used by the parser */ MYSQL_FTPARSER_PARAM *ftparser_param; /* share info between init/deinit */ char *filename; /* parameter to open filename */ - uchar *buff, /* Temp area for key */ - *lastkey, *lastkey2; /* Last used search key */ - uchar *first_mbr_key; /* Searhed spatial key */ - byte *rec_buff; /* Tempbuff for recordpack */ - uchar *int_keypos, /* Save position for next/previous */ + byte *buff; /* page buffer */ + byte *keyread_buff; /* Buffer for last key read */ + byte *lastkey, *lastkey2; /* Last used search key */ + byte *first_mbr_key; /* Searhed spatial key */ + byte *rec_buff; /* Temp buffer for recordpack */ + byte *int_keypos, /* Save position for next/previous */ *int_maxpos; /* -""- */ uint int_nod_flag; /* -""- */ uint32 int_keytree_version; /* -""- */ - int(*read_record) (struct st_maria_info *, my_off_t, byte *); + int (*read_record) (struct st_maria_info *, byte*, MARIA_RECORD_POS); invalidator_by_filename invalidator; /* query cache invalidator */ ulong this_unique; /* uniq filenumber or thread */ ulong last_unique; /* last unique number */ ulong this_loop; /* counter for this open */ ulong last_loop; /* last used counter */ - my_off_t lastpos, /* Last record position */ - nextpos; /* Position to next record */ - my_off_t save_lastpos; + MARIA_RECORD_POS save_lastpos; + MARIA_RECORD_POS dup_key_pos; my_off_t pos; /* Intern variable */ my_off_t last_keypage; /* Last key page read */ my_off_t last_search_keypage; /* Last keypage when searching */ - my_off_t dupp_key_pos; - ha_checksum checksum; /* QQ: the folloing two xxx_length fields should be removed, as they are not compatible with parallel repair */ ulong packed_length, blob_length; /* Length of found, packed record */ + my_size_t rec_buff_size; int dfile; /* The datafile */ uint opt_flag; /* Optim. for space/speed */ uint update; /* If file changed since open */ @@ -298,19 +404,19 @@ struct st_maria_info my_bool was_locked; /* Was locked in panic */ my_bool append_insert_at_end; /* Set if concurrent insert */ my_bool quick_mode; - /* If info->buff can't be used for rnext */ + /* If info->keyread_buff can't be used for rnext */ my_bool page_changed; - /* If info->buff has to be reread for rnext */ - my_bool buff_used; - my_bool once_flags; /* For MARIAMRG */ + /* If info->keyread_buff has to be reread for rnext */ + my_bool keybuff_used; + my_bool once_flags; /* For MARIA_MRG */ #ifdef THREAD THR_LOCK_DATA lock; #endif - uchar *maria_rtree_recursion_state; /* For RTREE */ + uchar *maria_rtree_recursion_state; /* For RTREE */ int maria_rtree_recursion_depth; }; -/* Some defines used by isam-funktions */ +/* Some defines used by maria-functions */ #define USE_WHOLE_KEY HA_MAX_KEY_BUFF*2 /* Use whole key in _search() */ #define F_EXTRA_LCK -1 @@ -361,15 +467,15 @@ struct st_maria_info } #define get_key_full_length(length,key) \ -{ if ((uchar) *(key) != 255) \ - length= ((uint) (uchar) *((key)++))+1; \ + { if (*(uchar*) (key) != 255) \ + length= ((uint) *(uchar*) ((key)++))+1; \ else \ { length=mi_uint2korr((key)+1)+3; (key)+=3; } \ } #define get_key_full_length_rdonly(length,key) \ -{ if ((uchar) *(key) != 255) \ - length= ((uint) (uchar) *((key)))+1; \ +{ if (*(uchar*) (key) != 255) \ + length= ((uint) *(uchar*) ((key)))+1; \ else \ { length=mi_uint2korr((key)+1)+3; } \ } @@ -398,7 +504,6 @@ struct st_maria_info #define PACK_TYPE_ZERO_FILL 4 #define MARIA_FOUND_WRONG_KEY 32738 /* Impossible value from ha_key_cmp */ -#define MARIA_MAX_KEY_BLOCK_SIZE (MARIA_MAX_KEY_BLOCK_LENGTH/MARIA_MIN_KEY_BLOCK_LENGTH) #define MARIA_BLOCK_SIZE(key_length,data_pointer,key_pointer,block_size) (((((key_length)+(data_pointer)+(key_pointer))*4+(key_pointer)+2)/(block_size)+1)*(block_size)) #define MARIA_MAX_KEYPTR_SIZE 5 /* For calculating block lengths */ #define MARIA_MIN_KEYBLOCK_LENGTH 50 /* When to split delete blocks */ @@ -428,85 +533,88 @@ extern LIST *maria_open_list; extern uchar NEAR maria_file_magic[], NEAR maria_pack_file_magic[]; extern uint NEAR maria_read_vec[], NEAR maria_readnext_vec[]; extern uint maria_quick_table_bits; +extern byte maria_zero_string[]; /* This is used by _ma_calc_xxx_key_length och _ma_store_key */ typedef struct st_maria_s_param { - uint ref_length, key_length, - n_ref_length, - n_length, totlength, part_of_prev_key, prev_length, pack_marker; - uchar *key, *prev_key, *next_key_pos; + uint ref_length, key_length, n_ref_length; + uint n_length, totlength, part_of_prev_key, prev_length, pack_marker; + const byte *key; + byte *prev_key, *next_key_pos; bool store_not_null; } MARIA_KEY_PARAM; /* Prototypes for intern functions */ -extern int _ma_read_dynamic_record(MARIA_HA *info, my_off_t filepos, - byte *buf); -extern int _ma_write_dynamic_record(MARIA_HA *, const byte *); -extern int _ma_update_dynamic_record(MARIA_HA *, my_off_t, const byte *); -extern int _ma_delete_dynamic_record(MARIA_HA *info); -extern int _ma_cmp_dynamic_record(MARIA_HA *info, const byte *record); -extern int _ma_read_rnd_dynamic_record(MARIA_HA *, byte *, my_off_t, +extern int _ma_read_dynamic_record(MARIA_HA *, byte *, MARIA_RECORD_POS); +extern int _ma_read_rnd_dynamic_record(MARIA_HA *, byte *, MARIA_RECORD_POS, my_bool); -extern int _ma_write_blob_record(MARIA_HA *, const byte *); -extern int _ma_update_blob_record(MARIA_HA *, my_off_t, const byte *); -extern int _ma_read_static_record(MARIA_HA *info, my_off_t filepos, - byte *buf); -extern int _ma_write_static_record(MARIA_HA *, const byte *); -extern int _ma_update_static_record(MARIA_HA *, my_off_t, const byte *); -extern int _ma_delete_static_record(MARIA_HA *info); -extern int _ma_cmp_static_record(MARIA_HA *info, const byte *record); -extern int _ma_read_rnd_static_record(MARIA_HA *, byte *, my_off_t, my_bool); -extern int _ma_ck_write(MARIA_HA *info, uint keynr, uchar *key, +extern my_bool _ma_write_dynamic_record(MARIA_HA *, const byte *); +extern my_bool _ma_update_dynamic_record(MARIA_HA *, MARIA_RECORD_POS, + const byte *); +extern my_bool _ma_delete_dynamic_record(MARIA_HA *info); +extern my_bool _ma_cmp_dynamic_record(MARIA_HA *info, const byte *record); +extern my_bool _ma_write_blob_record(MARIA_HA *, const byte *); +extern my_bool _ma_update_blob_record(MARIA_HA *, MARIA_RECORD_POS, + const byte *); +extern int _ma_read_static_record(MARIA_HA *info, byte *, MARIA_RECORD_POS); +extern int _ma_read_rnd_static_record(MARIA_HA *, byte *, MARIA_RECORD_POS, + my_bool); +extern my_bool _ma_write_static_record(MARIA_HA *, const byte *); +extern my_bool _ma_update_static_record(MARIA_HA *, MARIA_RECORD_POS, + const byte *); +extern my_bool _ma_delete_static_record(MARIA_HA *info); +extern my_bool _ma_cmp_static_record(MARIA_HA *info, const byte *record); +extern int _ma_ck_write(MARIA_HA *info, uint keynr, byte *key, uint length); extern int _ma_ck_real_write_btree(MARIA_HA *info, MARIA_KEYDEF *keyinfo, - uchar *key, uint key_length, - my_off_t *root, uint comp_flag); + byte *key, uint key_length, + MARIA_RECORD_POS *root, uint comp_flag); extern int _ma_enlarge_root(MARIA_HA *info, MARIA_KEYDEF *keyinfo, - uchar *key, my_off_t *root); -extern int _ma_insert(MARIA_HA *info, MARIA_KEYDEF *keyinfo, uchar *key, - uchar *anc_buff, uchar *key_pos, uchar *key_buff, - uchar *father_buff, uchar *father_keypos, + byte *key, MARIA_RECORD_POS *root); +extern int _ma_insert(MARIA_HA *info, MARIA_KEYDEF *keyinfo, byte *key, + byte *anc_buff, byte *key_pos, byte *key_buff, + byte *father_buff, byte *father_keypos, my_off_t father_page, my_bool insert_last); extern int _ma_split_page(MARIA_HA *info, MARIA_KEYDEF *keyinfo, - uchar *key, uchar *buff, uchar *key_buff, + byte *key, byte *buff, byte *key_buff, my_bool insert_last); -extern uchar *_ma_find_half_pos(uint nod_flag, MARIA_KEYDEF *keyinfo, - uchar *page, uchar *key, +extern byte *_ma_find_half_pos(uint nod_flag, MARIA_KEYDEF *keyinfo, + byte *page, byte *key, uint *return_key_length, - uchar ** after_key); + byte ** after_key); extern int _ma_calc_static_key_length(MARIA_KEYDEF *keyinfo, uint nod_flag, - uchar *key_pos, uchar *org_key, - uchar *key_buff, uchar *key, + byte *key_pos, byte *org_key, + byte *key_buff, const byte *key, MARIA_KEY_PARAM *s_temp); extern int _ma_calc_var_key_length(MARIA_KEYDEF *keyinfo, uint nod_flag, - uchar *key_pos, uchar *org_key, - uchar *key_buff, uchar *key, + byte *key_pos, byte *org_key, + byte *key_buff, const byte *key, MARIA_KEY_PARAM *s_temp); extern int _ma_calc_var_pack_key_length(MARIA_KEYDEF *keyinfo, - uint nod_flag, uchar *key_pos, - uchar *org_key, uchar *prev_key, - uchar *key, + uint nod_flag, byte *key_pos, + byte *org_key, byte *prev_key, + const byte *key, MARIA_KEY_PARAM *s_temp); extern int _ma_calc_bin_pack_key_length(MARIA_KEYDEF *keyinfo, - uint nod_flag, uchar *key_pos, - uchar *org_key, uchar *prev_key, - uchar *key, + uint nod_flag, byte *key_pos, + byte *org_key, byte *prev_key, + const byte *key, MARIA_KEY_PARAM *s_temp); -void _ma_store_static_key(MARIA_KEYDEF *keyinfo, uchar *key_pos, +void _ma_store_static_key(MARIA_KEYDEF *keyinfo, byte *key_pos, MARIA_KEY_PARAM *s_temp); -void _ma_store_var_pack_key(MARIA_KEYDEF *keyinfo, uchar *key_pos, +void _ma_store_var_pack_key(MARIA_KEYDEF *keyinfo, byte *key_pos, MARIA_KEY_PARAM *s_temp); #ifdef NOT_USED -void _ma_store_pack_key(MARIA_KEYDEF *keyinfo, uchar *key_pos, +void _ma_store_pack_key(MARIA_KEYDEF *keyinfo, byte *key_pos, MARIA_KEY_PARAM *s_temp); #endif -void _ma_store_bin_pack_key(MARIA_KEYDEF *keyinfo, uchar *key_pos, +void _ma_store_bin_pack_key(MARIA_KEYDEF *keyinfo, byte *key_pos, MARIA_KEY_PARAM *s_temp); -extern int _ma_ck_delete(MARIA_HA *info, uint keynr, uchar *key, +extern int _ma_ck_delete(MARIA_HA *info, uint keynr, byte *key, uint key_length); extern int _ma_readinfo(MARIA_HA *info, int lock_flag, int check_keybuffer); extern int _ma_writeinfo(MARIA_HA *info, uint options); @@ -514,74 +622,69 @@ extern int _ma_test_if_changed(MARIA_HA *info); extern int _ma_mark_file_changed(MARIA_HA *info); extern int _ma_decrement_open_count(MARIA_HA *info); extern int _ma_check_index(MARIA_HA *info, int inx); -extern int _ma_search(MARIA_HA *info, MARIA_KEYDEF *keyinfo, uchar *key, +extern int _ma_search(MARIA_HA *info, MARIA_KEYDEF *keyinfo, byte *key, uint key_len, uint nextflag, my_off_t pos); extern int _ma_bin_search(struct st_maria_info *info, MARIA_KEYDEF *keyinfo, - uchar *page, uchar *key, uint key_len, - uint comp_flag, uchar **ret_pos, uchar *buff, + byte *page, byte *key, uint key_len, + uint comp_flag, byte **ret_pos, byte *buff, my_bool *was_last_key); extern int _ma_seq_search(MARIA_HA *info, MARIA_KEYDEF *keyinfo, - uchar *page, uchar *key, uint key_len, - uint comp_flag, uchar ** ret_pos, uchar *buff, + byte *page, byte *key, uint key_len, + uint comp_flag, byte ** ret_pos, byte *buff, my_bool *was_last_key); extern int _ma_prefix_search(MARIA_HA *info, MARIA_KEYDEF *keyinfo, - uchar *page, uchar *key, uint key_len, - uint comp_flag, uchar ** ret_pos, uchar *buff, + byte *page, byte *key, uint key_len, + uint comp_flag, byte ** ret_pos, byte *buff, my_bool *was_last_key); -extern my_off_t _ma_kpos(uint nod_flag, uchar *after_key); -extern void _ma_kpointer(MARIA_HA *info, uchar *buff, my_off_t pos); -extern my_off_t _ma_dpos(MARIA_HA *info, uint nod_flag, uchar *after_key); -extern my_off_t _ma_rec_pos(MARIA_SHARE *info, uchar *ptr); -extern void _ma_dpointer(MARIA_HA *info, uchar *buff, my_off_t pos); +extern my_off_t _ma_kpos(uint nod_flag, byte *after_key); +extern void _ma_kpointer(MARIA_HA *info, byte *buff, my_off_t pos); +extern MARIA_RECORD_POS _ma_dpos(MARIA_HA *info, uint nod_flag, + const byte *after_key); +extern MARIA_RECORD_POS _ma_rec_pos(MARIA_SHARE *info, byte *ptr); +extern void _ma_dpointer(MARIA_HA *info, byte *buff, MARIA_RECORD_POS pos); extern uint _ma_get_static_key(MARIA_KEYDEF *keyinfo, uint nod_flag, - uchar **page, uchar *key); + byte **page, byte *key); extern uint _ma_get_pack_key(MARIA_KEYDEF *keyinfo, uint nod_flag, - uchar **page, uchar *key); + byte **page, byte *key); extern uint _ma_get_binary_pack_key(MARIA_KEYDEF *keyinfo, uint nod_flag, - uchar ** page_pos, uchar *key); -extern uchar *_ma_get_last_key(MARIA_HA *info, MARIA_KEYDEF *keyinfo, - uchar *keypos, uchar *lastkey, - uchar *endpos, uint *return_key_length); -extern uchar *_ma_get_key(MARIA_HA *info, MARIA_KEYDEF *keyinfo, - uchar *page, uchar *key, uchar *keypos, + byte ** page_pos, byte *key); +extern byte *_ma_get_last_key(MARIA_HA *info, MARIA_KEYDEF *keyinfo, + byte *keypos, byte *lastkey, + byte *endpos, uint *return_key_length); +extern byte *_ma_get_key(MARIA_HA *info, MARIA_KEYDEF *keyinfo, + byte *page, byte *key, byte *keypos, uint *return_key_length); -extern uint _ma_keylength(MARIA_KEYDEF *keyinfo, uchar *key); -extern uint _ma_keylength_part(MARIA_KEYDEF *keyinfo, register uchar *key, +extern uint _ma_keylength(MARIA_KEYDEF *keyinfo, const byte *key); +extern uint _ma_keylength_part(MARIA_KEYDEF *keyinfo, register const byte *key, HA_KEYSEG *end); -extern uchar *_ma_move_key(MARIA_KEYDEF *keyinfo, uchar *to, uchar *from); +extern byte *_ma_move_key(MARIA_KEYDEF *keyinfo, byte *to, const byte *from); extern int _ma_search_next(MARIA_HA *info, MARIA_KEYDEF *keyinfo, - uchar *key, uint key_length, uint nextflag, + byte *key, uint key_length, uint nextflag, my_off_t pos); extern int _ma_search_first(MARIA_HA *info, MARIA_KEYDEF *keyinfo, my_off_t pos); extern int _ma_search_last(MARIA_HA *info, MARIA_KEYDEF *keyinfo, my_off_t pos); -extern uchar *_ma_fetch_keypage(MARIA_HA *info, MARIA_KEYDEF *keyinfo, - my_off_t page, int level, uchar *buff, +extern byte *_ma_fetch_keypage(MARIA_HA *info, MARIA_KEYDEF *keyinfo, + my_off_t page, int level, byte *buff, int return_buffer); extern int _ma_write_keypage(MARIA_HA *info, MARIA_KEYDEF *keyinfo, - my_off_t page, int level, uchar *buff); + my_off_t page, int level, byte *buff); extern int _ma_dispose(MARIA_HA *info, MARIA_KEYDEF *keyinfo, my_off_t pos, int level); extern my_off_t _ma_new(MARIA_HA *info, MARIA_KEYDEF *keyinfo, int level); -extern uint _ma_make_key(MARIA_HA *info, uint keynr, uchar *key, - const byte *record, my_off_t filepos); -extern uint _ma_pack_key(MARIA_HA *info, uint keynr, uchar *key, - uchar *old, uint key_length, +extern uint _ma_make_key(MARIA_HA *info, uint keynr, byte *key, + const byte *record, MARIA_RECORD_POS filepos); +extern uint _ma_pack_key(MARIA_HA *info, uint keynr, byte *key, + const byte *old, uint key_length, HA_KEYSEG ** last_used_keyseg); -extern int _ma_read_key_record(MARIA_HA *info, my_off_t filepos, - byte *buf); -extern int _ma_read_cache(IO_CACHE *info, byte *buff, my_off_t pos, +extern int _ma_read_key_record(MARIA_HA *info, byte *buf, MARIA_RECORD_POS); +extern int _ma_read_cache(IO_CACHE *info, byte *buff, MARIA_RECORD_POS pos, uint length, int re_read_if_possibly); extern ulonglong ma_retrieve_auto_increment(MARIA_HA *info, const byte *record); -extern byte *_ma_alloc_rec_buff(MARIA_HA *, ulong, byte **); -#define _ma_get_rec_buff_ptr(info,buf) \ - ((((info)->s->options & HA_OPTION_PACK_RECORD) && (buf)) ? \ - (buf) - MARIA_REC_BUFF_OFFSET : (buf)) -#define _ma_get_rec_buff_len(info,buf) \ - (*((uint32 *)(_ma_get_rec_buff_ptr(info,buf)))) - +extern my_bool _ma_alloc_buffer(byte **old_addr, my_size_t *old_size, + my_size_t new_size); extern ulong _ma_rec_unpack(MARIA_HA *info, byte *to, byte *from, ulong reclength); extern my_bool _ma_rec_check(MARIA_HA *info, const char *record, @@ -592,11 +695,13 @@ extern int _ma_write_part_record(MARIA_HA *info, my_off_t filepos, byte ** record, ulong *reclength, int *flag); extern void _ma_print_key(FILE *stream, HA_KEYSEG *keyseg, - const uchar *key, uint length); -extern my_bool _ma_read_pack_info(MARIA_HA *info, pbool fix_keys); -extern int _ma_read_pack_record(MARIA_HA *info, my_off_t filepos, - byte *buf); -extern int _ma_read_rnd_pack_record(MARIA_HA *, byte *, my_off_t, my_bool); + const byte *key, uint length); +extern my_bool _ma_once_init_pack_row(MARIA_SHARE *share, File dfile); +extern my_bool _ma_once_end_pack_row(MARIA_SHARE *share); +extern int _ma_read_pack_record(MARIA_HA *info, byte *buf, + MARIA_RECORD_POS filepos); +extern int _ma_read_rnd_pack_record(MARIA_HA *, byte *, MARIA_RECORD_POS, + my_bool); extern int _ma_pack_rec_unpack(MARIA_HA *info, byte *to, byte *from, ulong reclength); extern ulonglong _ma_safe_mul(ulonglong a, ulonglong b); @@ -613,9 +718,9 @@ typedef struct st_maria_block_info ulong data_len; ulong block_len; ulong blob_len; - my_off_t filepos; - my_off_t next_filepos; - my_off_t prev_filepos; + MARIA_RECORD_POS filepos; + MARIA_RECORD_POS next_filepos; + MARIA_RECORD_POS prev_filepos; uint second_read; uint offset; } MARIA_BLOCK_INFO; @@ -672,11 +777,10 @@ extern uint _ma_nommap_pwrite(MARIA_HA *info, byte *Buffer, uint Count, my_off_t offset, myf MyFlags); uint _ma_state_info_write(File file, MARIA_STATE_INFO *state, uint pWrite); -uchar *_ma_state_info_read(uchar *ptr, MARIA_STATE_INFO *state); +byte *_ma_state_info_read(byte *ptr, MARIA_STATE_INFO *state); uint _ma_state_info_read_dsk(File file, MARIA_STATE_INFO *state, my_bool pRead); uint _ma_base_info_write(File file, MARIA_BASE_INFO *base); -uchar *_ma_n_base_info_read(uchar *ptr, MARIA_BASE_INFO *base); int _ma_keyseg_write(File file, const HA_KEYSEG *keyseg); char *_ma_keyseg_read(char *ptr, HA_KEYSEG *keyseg); uint _ma_keydef_write(File file, MARIA_KEYDEF *keydef); @@ -690,14 +794,14 @@ ha_checksum _ma_checksum(MARIA_HA *info, const byte *buf); ha_checksum _ma_static_checksum(MARIA_HA *info, const byte *buf); my_bool _ma_check_unique(MARIA_HA *info, MARIA_UNIQUEDEF *def, byte *record, ha_checksum unique_hash, - my_off_t pos); + MARIA_RECORD_POS pos); ha_checksum _ma_unique_hash(MARIA_UNIQUEDEF *def, const byte *buf); -int _ma_cmp_static_unique(MARIA_HA *info, MARIA_UNIQUEDEF *def, - const byte *record, my_off_t pos); -int _ma_cmp_dynamic_unique(MARIA_HA *info, MARIA_UNIQUEDEF *def, - const byte *record, my_off_t pos); -int _ma_unique_comp(MARIA_UNIQUEDEF *def, const byte *a, const byte *b, - my_bool null_are_equal); +my_bool _ma_cmp_static_unique(MARIA_HA *info, MARIA_UNIQUEDEF *def, + const byte *record, MARIA_RECORD_POS pos); +my_bool _ma_cmp_dynamic_unique(MARIA_HA *info, MARIA_UNIQUEDEF *def, + const byte *record, MARIA_RECORD_POS pos); +my_bool _ma_unique_comp(MARIA_UNIQUEDEF *def, const byte *a, const byte *b, + my_bool null_are_equal); void _ma_get_status(void *param, int concurrent_insert); void _ma_update_status(void *param); void _ma_copy_status(void *to, void *from); @@ -711,6 +815,9 @@ void _ma_setup_functions(register MARIA_SHARE *share); my_bool _ma_dynmap_file(MARIA_HA *info, my_off_t size); void _ma_remap_file(MARIA_HA *info, my_off_t size); +MARIA_RECORD_POS _ma_write_init_default(MARIA_HA *info, const byte *record); +my_bool _ma_write_abort_default(MARIA_HA *info); + /* Functions needed by _ma_check (are overrided in MySQL) */ C_MODE_START volatile int *_ma_killed_ptr(HA_CHECK *param); diff --git a/storage/maria/maria_ftdump.c b/storage/maria/maria_ftdump.c index b840072aed0..0faa64327eb 100644 --- a/storage/maria/maria_ftdump.c +++ b/storage/maria/maria_ftdump.c @@ -106,7 +106,7 @@ int main(int argc,char *argv[]) maria_lock_database(info, F_EXTRA_LCK); - info->lastpos= HA_OFFSET_ERROR; + info->cur_row.lastpos= HA_OFFSET_ERROR; info->update|= HA_STATE_PREV_FOUND; while (!(error=maria_rnext(info,NULL,inx))) @@ -157,9 +157,9 @@ int main(int argc,char *argv[]) if (dump) { if (subkeys>=0) - printf("%9lx %20.7f %s\n", (long) info->lastpos,weight,buf); + printf("%9lx %20.7f %s\n", (long) info->cur_row.lastpos,weight,buf); else - printf("%9lx => %17d %s\n",(long) info->lastpos,-subkeys,buf); + printf("%9lx => %17d %s\n",(long) info->cur_row.lastpos,-subkeys,buf); } if (verbose && (total%HOW_OFTEN_TO_WRITE)==0) printf("%10ld\r",total); diff --git a/storage/maria/maria_pack.c b/storage/maria/maria_pack.c index c5a53b1ffac..e4e38cb6e9d 100644 --- a/storage/maria/maria_pack.c +++ b/storage/maria/maria_pack.c @@ -239,7 +239,7 @@ int main(int argc, char **argv) } } if (ok && isamchk_neaded && !silent) - puts("Remember to run mariachk -rq on compressed tables"); + puts("Remember to run maria_chk -rq on compressed tables"); VOID(fflush(stdout)); VOID(fflush(stderr)); free_defaults(default_argv); @@ -294,7 +294,7 @@ static struct my_option my_long_options[] = static void print_version(void) { - VOID(printf("%s Ver 1.23 for %s on %s\n", + VOID(printf("%s Ver 1.0 for %s on %s\n", my_progname, SYSTEM_TYPE, MACHINE_TYPE)); NETWARE_SET_SCREEN_MODE(1); } @@ -308,7 +308,7 @@ static void usage(void) puts("and you are welcome to modify and redistribute it under the GPL license\n"); puts("Pack a MARIA-table to take much less space."); - puts("Keys are not updated, you must run mariachk -rq on the datafile"); + puts("Keys are not updated, you must run maria_chk -rq on the datafile"); puts("afterwards to update the keys."); puts("You should give the .MYI file as the filename argument."); @@ -359,7 +359,7 @@ get_one_option(int optid, const struct my_option *opt __attribute__((unused)), silent= 0; break; case '#': - DBUG_PUSH(argument ? argument : "d:t:o"); + DBUG_PUSH(argument ? argument : "d:t:o,/tmp/maria_pack.trace"); break; case 'V': print_version(); @@ -665,7 +665,7 @@ static int compress(PACK_MRG_INFO *mrg,char *result_table) /* Display statistics. */ DBUG_PRINT("info", ("Min record length: %6d Max length: %6d " - "Mean total length: %6ld\n", + "Mean total length: %6ld", mrg->min_pack_length, mrg->max_pack_length, (ulong) (mrg->records ? (new_length/mrg->records) : 0))); if (verbose && mrg->records) @@ -681,6 +681,7 @@ static int compress(PACK_MRG_INFO *mrg,char *result_table) { error|=my_close(isam_file->dfile,MYF(MY_WME)); isam_file->dfile= -1; /* Tell maria_close file is closed */ + isam_file->s->bitmap.file= -1; } } @@ -841,32 +842,27 @@ static void free_counts_and_tree_and_queue(HUFF_TREE *huff_trees, uint trees, static int get_statistic(PACK_MRG_INFO *mrg,HUFF_COUNTS *huff_counts) { int error; - uint length; + uint length, null_bytes; ulong reclength,max_blob_length; byte *record,*pos,*next_pos,*end_pos,*start_pos; ha_rows record_count; - my_bool static_row_size; HUFF_COUNTS *count,*end_count; TREE_ELEMENT *element; + ha_checksum(*calc_checksum) (struct st_maria_info *, const byte *); DBUG_ENTER("get_statistic"); - reclength=mrg->file[0]->s->base.reclength; + reclength= mrg->file[0]->s->base.reclength; + null_bytes= mrg->file[0]->s->base.null_bytes; record=(byte*) my_alloca(reclength); end_count=huff_counts+mrg->file[0]->s->base.fields; record_count=0; glob_crc=0; max_blob_length=0; /* Check how to calculate checksum */ - static_row_size=1; - for (count=huff_counts ; count < end_count ; count++) - { - if (count->field_type == FIELD_BLOB || - count->field_type == FIELD_VARCHAR) - { - static_row_size=0; - break; - } - } + if (mrg->file[0]->s->data_file_type == STATIC_RECORD) + calc_checksum= _ma_static_checksum; + else + calc_checksum= _ma_checksum; mrg_reset(mrg); while ((error=mrg_rrnd(mrg,record)) != HA_ERR_END_OF_FILE) @@ -875,13 +871,10 @@ static int get_statistic(PACK_MRG_INFO *mrg,HUFF_COUNTS *huff_counts) if (! error) { /* glob_crc is a checksum over all bytes of all records. */ - if (static_row_size) - glob_crc+=_ma_static_checksum(mrg->file[0],record); - else - glob_crc+=_ma_checksum(mrg->file[0],record); + glob_crc+= (*calc_checksum)(mrg->file[0],record); /* Count the incidence of values separately for every column. */ - for (pos=record,count=huff_counts ; + for (pos=record + null_bytes, count=huff_counts ; count < end_count ; count++, pos=next_pos) @@ -1109,14 +1102,14 @@ static int get_statistic(PACK_MRG_INFO *mrg,HUFF_COUNTS *huff_counts) DBUG_PRINT("info", ("column: %3lu", count - huff_counts + 1)); if (verbose >= 2) - VOID(printf("column: %3lu\n", count - huff_counts + 1)); + VOID(printf("column: %3u\n", count - huff_counts + 1)); if (count->tree_buff) { DBUG_PRINT("info", ("number of distinct values: %lu", (count->tree_pos - count->tree_buff) / count->field_length)); if (verbose >= 2) - VOID(printf("number of distinct values: %lu\n", + VOID(printf("number of distinct values: %u\n", (count->tree_pos - count->tree_buff) / count->field_length)); } @@ -1368,7 +1361,8 @@ static void check_counts(HUFF_COUNTS *huff_counts, uint trees, DBUG_VOID_RETURN; } - /* Test if we can use space-compression and empty-field-compression */ + +/* Test if we can use space-compression and empty-field-compression */ static int test_space_compress(HUFF_COUNTS *huff_counts, my_off_t records, @@ -2281,7 +2275,7 @@ static my_off_t write_huff_tree(HUFF_TREE *huff_tree, uint trees) if (bits > 8 * sizeof(code)) { VOID(fflush(stdout)); - VOID(fprintf(stderr, "error: Huffman code too long: %u/%lu\n", + VOID(fprintf(stderr, "error: Huffman code too long: %u/%u\n", bits, 8 * sizeof(code))); errors++; break; @@ -2410,8 +2404,8 @@ static uint max_bit(register uint value) static int compress_isam_file(PACK_MRG_INFO *mrg, HUFF_COUNTS *huff_counts) { int error; - uint i,max_calc_length,pack_ref_length,min_record_length,max_record_length, - intervall,field_length,max_pack_length,pack_blob_length; + uint i,max_calc_length,pack_ref_length,min_record_length,max_record_length; + uint intervall,field_length,max_pack_length,pack_blob_length, null_bytes; my_off_t record_count; char llbuf[32]; ulong length,pack_length; @@ -2429,6 +2423,7 @@ static int compress_isam_file(PACK_MRG_INFO *mrg, HUFF_COUNTS *huff_counts) end_count=huff_counts+isam_file->s->base.fields; min_record_length= (uint) ~0; max_record_length=0; + null_bytes= isam_file->s->base.null_bytes; /* Calculate the maximum number of bits required to pack the records. @@ -2439,7 +2434,8 @@ static int compress_isam_file(PACK_MRG_INFO *mrg, HUFF_COUNTS *huff_counts) Empty blobs and varchar are encoded with a single 1 bit. Other blobs and varchar get a leading 0 bit. */ - for (i=max_calc_length=0 ; i < isam_file->s->base.fields ; i++) + max_calc_length= null_bytes; + for (i= 0 ; i < isam_file->s->base.fields ; i++) { if (!(huff_counts[i].pack_type & PACK_TYPE_ZERO_FILL)) huff_counts[i].max_zero_fill=0; @@ -2475,8 +2471,16 @@ static int compress_isam_file(PACK_MRG_INFO *mrg, HUFF_COUNTS *huff_counts) if (flush_buffer((ulong) max_calc_length + (ulong) max_pack_length)) break; record_pos= (byte*) file_buffer.pos; - file_buffer.pos+=max_pack_length; - for (start_pos=record, count= huff_counts; count < end_count ; count++) + file_buffer.pos+= max_pack_length; + if (null_bytes) + { + /* Copy null bits 'as is' */ + memcpy(file_buffer.pos, record, null_bytes); + file_buffer.pos+= null_bytes; + } + for (start_pos=record+null_bytes, count= huff_counts; + count < end_count ; + count++) { end_pos=start_pos+(field_length=count->field_length); tree=count->tree; @@ -2738,8 +2742,9 @@ static int compress_isam_file(PACK_MRG_INFO *mrg, HUFF_COUNTS *huff_counts) length=(ulong) ((byte*) file_buffer.pos - record_pos) - max_pack_length; pack_length= _ma_save_pack_length(pack_version, record_pos, length); if (pack_blob_length) - pack_length+= _ma_save_pack_length(pack_version, record_pos + pack_length, - tot_blob_length); + pack_length+= _ma_save_pack_length(pack_version, + record_pos + pack_length, + tot_blob_length); DBUG_PRINT("fields", ("record: %lu length: %lu blob-length: %lu " "length-bytes: %lu", (ulong) record_count, length, tot_blob_length, pack_length)); @@ -2934,7 +2939,8 @@ static void flush_bits(void) ** functions to handle the joined files ****************************************************************************/ -static int save_state(MARIA_HA *isam_file,PACK_MRG_INFO *mrg,my_off_t new_length, +static int save_state(MARIA_HA *isam_file,PACK_MRG_INFO *mrg, + my_off_t new_length, ha_checksum crc) { MARIA_SHARE *share=isam_file->s; @@ -2944,6 +2950,8 @@ static int save_state(MARIA_HA *isam_file,PACK_MRG_INFO *mrg,my_off_t new_length options|= HA_OPTION_COMPRESS_RECORD | HA_OPTION_READ_ONLY_DATA; mi_int2store(share->state.header.options,options); + share->state.header.org_data_file_type= share->state.header.data_file_type; + share->state.header.data_file_type= COMPRESSED_RECORD; share->state.state.data_file_length=new_length; share->state.state.del=0; @@ -2962,14 +2970,13 @@ static int save_state(MARIA_HA *isam_file,PACK_MRG_INFO *mrg,my_off_t new_length } /* If there are no disabled indexes, keep key_file_length value from - original file so "mariachk -rq" can use this value (this is necessary + original file so "maria_chk -rq" can use this value (this is necessary because index size cannot be easily calculated for fulltext keys) */ maria_clear_all_keys_active(share->state.key_map); for (key=0 ; key < share->base.keys ; key++) share->state.key_root[key]= HA_OFFSET_ERROR; - for (key=0 ; key < share->state.header.max_block_size_index ; key++) - share->state.key_del[key]= HA_OFFSET_ERROR; + share->state.key_del= HA_OFFSET_ERROR; isam_file->state->checksum=crc; /* Save crc here */ share->changed=1; /* Force write of header */ share->state.open_count=0; @@ -3037,21 +3044,18 @@ static int mrg_rrnd(PACK_MRG_INFO *info,byte *buf) info->end=info->current+info->count; maria_reset(isam_info); maria_extra(isam_info, HA_EXTRA_CACHE, 0); - filepos=isam_info->s->pack.header_length; + if ((error= maria_scan_init(isam_info))) + return(error); } else - { isam_info= *info->current; - filepos= isam_info->nextpos; - } for (;;) { - isam_info->update&= HA_STATE_CHANGED; - if (!(error=(*isam_info->s->read_rnd)(isam_info,(byte*) buf, - filepos, 1)) || + if (!(error= maria_scan(isam_info, buf)) || error != HA_ERR_END_OF_FILE) return (error); + maria_scan_end(isam_info); maria_extra(isam_info,HA_EXTRA_NO_CACHE, 0); if (info->current+1 == info->end) return(HA_ERR_END_OF_FILE); @@ -3060,6 +3064,8 @@ static int mrg_rrnd(PACK_MRG_INFO *info,byte *buf) filepos=isam_info->s->pack.header_length; maria_reset(isam_info); maria_extra(isam_info,HA_EXTRA_CACHE, 0); + if ((error= maria_scan_init(isam_info))) + return(error); } } @@ -3068,11 +3074,13 @@ static int mrg_close(PACK_MRG_INFO *mrg) { uint i; int error=0; + DBUG_ENTER("mrg_close"); + for (i=0 ; i < mrg->count ; i++) error|=maria_close(mrg->file[i]); if (mrg->free_file) my_free((gptr) mrg->file,MYF(0)); - return error; + DBUG_RETURN(error); } diff --git a/storage/myisam/mi_check.c b/storage/myisam/mi_check.c index 22860698a3a..05d4e5d77cf 100644 --- a/storage/myisam/mi_check.c +++ b/storage/myisam/mi_check.c @@ -911,6 +911,9 @@ int chk_data_link(HA_CHECK *param, MI_INFO *info,int extend) if (*killed_ptr(param)) goto err2; switch (info->s->data_file_type) { + case BLOCK_RECORD: + DBUG_ASSERT(0); /* Impossible */ + break; case STATIC_RECORD: if (my_b_read(¶m->read_cache,(byte*) record, info->s->base.pack_reclength)) @@ -2897,6 +2900,9 @@ static int sort_get_next_record(MI_SORT_PARAM *sort_param) DBUG_RETURN(1); switch (share->data_file_type) { + case BLOCK_RECORD: + DBUG_ASSERT(0); /* Impossible */ + break; case STATIC_RECORD: for (;;) { @@ -3283,6 +3289,9 @@ int sort_write_record(MI_SORT_PARAM *sort_param) if (sort_param->fix_datafile) { switch (sort_info->new_data_file_type) { + case BLOCK_RECORD: + DBUG_ASSERT(0); /* Impossible */ + break; case STATIC_RECORD: if (my_b_write(&info->rec_cache,sort_param->record, share->base.pack_reclength)) @@ -3395,18 +3404,19 @@ static int sort_key_write(MI_SORT_PARAM *sort_param, const void *a) if (sort_info->key_block->inited) { - cmp=ha_key_cmp(sort_param->seg,sort_info->key_block->lastkey, + cmp=ha_key_cmp(sort_param->seg, (uchar*) sort_info->key_block->lastkey, (uchar*) a, USE_WHOLE_KEY,SEARCH_FIND | SEARCH_UPDATE, diff_pos); if (param->stats_method == MI_STATS_METHOD_NULLS_NOT_EQUAL) - ha_key_cmp(sort_param->seg,sort_info->key_block->lastkey, + ha_key_cmp(sort_param->seg, (uchar*) sort_info->key_block->lastkey, (uchar*) a, USE_WHOLE_KEY, SEARCH_FIND | SEARCH_NULL_ARE_NOT_EQUAL, diff_pos); else if (param->stats_method == MI_STATS_METHOD_IGNORE_NULLS) { diff_pos[0]= mi_collect_stats_nonulls_next(sort_param->seg, sort_param->notnull, - sort_info->key_block->lastkey, + (uchar*) sort_info-> + key_block->lastkey, (uchar*)a); } sort_param->unique[diff_pos[0]-1]++; @@ -3429,8 +3439,8 @@ static int sort_key_write(MI_SORT_PARAM *sort_param, const void *a) llstr(sort_info->info->lastpos,llbuff), llstr(get_record_for_key(sort_info->info, sort_param->keyinfo, - sort_info->key_block-> - lastkey), + (uchar*) sort_info-> + key_block->lastkey), llbuff2)); param->testflag|=T_RETRY_WITHOUT_QUICK; if (sort_info->param->testflag & T_VERBOSE) @@ -3461,19 +3471,19 @@ int sort_ft_buf_flush(MI_SORT_PARAM *sort_param) val_len=share->ft2_keyinfo.keylength; get_key_full_length_rdonly(val_off, ft_buf->lastkey); - to=ft_buf->lastkey+val_off; + to= (uchar*) ft_buf->lastkey+val_off; if (ft_buf->buf) { /* flushing first-level tree */ - error=sort_insert_key(sort_param,key_block,ft_buf->lastkey, + error=sort_insert_key(sort_param,key_block, (uchar*) ft_buf->lastkey, HA_OFFSET_ERROR); for (from=to+val_len; - !error && from < ft_buf->buf; + !error && from < (uchar*) ft_buf->buf; from+= val_len) { memcpy(to, from, val_len); - error=sort_insert_key(sort_param,key_block,ft_buf->lastkey, + error=sort_insert_key(sort_param,key_block, (uchar*) ft_buf->lastkey, HA_OFFSET_ERROR); } return error; @@ -3482,8 +3492,8 @@ int sort_ft_buf_flush(MI_SORT_PARAM *sort_param) error=flush_pending_blocks(sort_param); /* updating lastkey with second-level tree info */ ft_intXstore(ft_buf->lastkey+val_off, -ft_buf->count); - _mi_dpointer(sort_info->info, ft_buf->lastkey+val_off+HA_FT_WLEN, - share->state.key_root[sort_param->key]); + _mi_dpointer(sort_info->info, (uchar*) ft_buf->lastkey+val_off+HA_FT_WLEN, + share->state.key_root[sort_param->key]); /* restoring first level tree data in sort_info/sort_param */ sort_info->key_block=sort_info->key_block_end- sort_info->param->sort_key_blocks; sort_param->keyinfo=share->keyinfo+sort_param->key; @@ -3491,7 +3501,7 @@ int sort_ft_buf_flush(MI_SORT_PARAM *sort_param) /* writing lastkey in first-level tree */ return error ? error : sort_insert_key(sort_param,sort_info->key_block, - ft_buf->lastkey,HA_OFFSET_ERROR); + (uchar*) ft_buf->lastkey,HA_OFFSET_ERROR); } static int sort_ft_key_write(MI_SORT_PARAM *sort_param, const void *a) @@ -3530,7 +3540,7 @@ static int sort_ft_key_write(MI_SORT_PARAM *sort_param, const void *a) if (ha_compare_text(sort_param->seg->charset, ((uchar *)a)+1,a_len-1, - ft_buf->lastkey+1,val_off-1, 0, 0)==0) + (uchar*) ft_buf->lastkey+1,val_off-1, 0, 0)==0) { if (!ft_buf->buf) /* store in second-level tree */ { @@ -3546,16 +3556,16 @@ static int sort_ft_key_write(MI_SORT_PARAM *sort_param, const void *a) return 0; /* converting to two-level tree */ - p=ft_buf->lastkey+val_off; + p= (uchar*) ft_buf->lastkey+val_off; while (key_block->inited) key_block++; sort_info->key_block=key_block; sort_param->keyinfo=& sort_info->info->s->ft2_keyinfo; - ft_buf->count=(ft_buf->buf - p)/val_len; + ft_buf->count=((uchar*) ft_buf->buf - p)/val_len; /* flushing buffer to second-level tree */ - for (error=0; !error && p < ft_buf->buf; p+= val_len) + for (error=0; !error && p < (uchar*) ft_buf->buf; p+= val_len) error=sort_insert_key(sort_param,key_block,p,HA_OFFSET_ERROR); ft_buf->buf=0; return error; @@ -3607,9 +3617,9 @@ static int sort_insert_key(MI_SORT_PARAM *sort_param, HA_CHECK *param=sort_info->param; DBUG_ENTER("sort_insert_key"); - anc_buff=key_block->buff; + anc_buff= (uchar*) key_block->buff; info=sort_info->info; - lastkey=key_block->lastkey; + lastkey= (uchar*) key_block->lastkey; nod_flag= (key_block == sort_info->key_block ? 0 : info->s->base.key_reflength); @@ -3622,7 +3632,7 @@ static int sort_insert_key(MI_SORT_PARAM *sort_param, DBUG_RETURN(1); } a_length=2+nod_flag; - key_block->end_pos=anc_buff+2; + key_block->end_pos= (char*) anc_buff+2; lastkey=0; /* No previous key in block */ } else @@ -3630,18 +3640,18 @@ static int sort_insert_key(MI_SORT_PARAM *sort_param, /* Save pointer to previous block */ if (nod_flag) - _mi_kpointer(info,key_block->end_pos,prev_block); + _mi_kpointer(info,(uchar*) key_block->end_pos,prev_block); t_length=(*keyinfo->pack_key)(keyinfo,nod_flag, (uchar*) 0,lastkey,lastkey,key, &s_temp); - (*keyinfo->store_key)(keyinfo, key_block->end_pos+nod_flag,&s_temp); + (*keyinfo->store_key)(keyinfo, (uchar*) key_block->end_pos+nod_flag,&s_temp); a_length+=t_length; mi_putint(anc_buff,a_length,nod_flag); key_block->end_pos+=t_length; if (a_length <= keyinfo->block_length) { - VOID(_mi_move_key(keyinfo,key_block->lastkey,key)); + VOID(_mi_move_key(keyinfo,(uchar*) key_block->lastkey,key)); key_block->last_length=a_length-t_length; DBUG_RETURN(0); } @@ -3666,7 +3676,8 @@ static int sort_insert_key(MI_SORT_PARAM *sort_param, DBUG_DUMP("buff",(byte*) anc_buff,mi_getint(anc_buff)); /* Write separator-key to block in next level */ - if (sort_insert_key(sort_param,key_block+1,key_block->lastkey,filepos)) + if (sort_insert_key(sort_param,key_block+1,(uchar*) key_block->lastkey, + filepos)) DBUG_RETURN(1); /* clear old block and write new key in it */ @@ -3752,7 +3763,7 @@ int flush_pending_blocks(MI_SORT_PARAM *sort_param) key_block->inited=0; length=mi_getint(key_block->buff); if (nod_flag) - _mi_kpointer(info,key_block->end_pos,filepos); + _mi_kpointer(info,(uchar*) key_block->end_pos,filepos); key_file_length=info->state->key_file_length; bzero((byte*) key_block->buff+length, keyinfo->block_length-length); if ((filepos=_mi_new(info,keyinfo,DFLT_INIT_HITS)) == HA_OFFSET_ERROR) @@ -3762,7 +3773,7 @@ int flush_pending_blocks(MI_SORT_PARAM *sort_param) if (key_file_length == info->state->key_file_length) { if (_mi_write_keypage(info, keyinfo, filepos, - DFLT_INIT_HITS, key_block->buff)) + DFLT_INIT_HITS, (uchar*) key_block->buff)) DBUG_RETURN(1); } else if (my_pwrite(info->s->kfile,(byte*) key_block->buff, @@ -3794,7 +3805,7 @@ static SORT_KEY_BLOCKS *alloc_key_blocks(HA_CHECK *param, uint blocks, for (i=0 ; i < blocks ; i++) { block[i].inited=0; - block[i].buff=(uchar*) (block+blocks)+(buffer_length+IO_SIZE)*i; + block[i].buff=(byte*) (block+blocks)+(buffer_length+IO_SIZE)*i; } DBUG_RETURN(block); } /* alloc_key_blocks */ diff --git a/storage/myisam/mi_create.c b/storage/myisam/mi_create.c index ad824bef009..cbd431cc222 100644 --- a/storage/myisam/mi_create.c +++ b/storage/myisam/mi_create.c @@ -41,7 +41,7 @@ int mi_create(const char *name,uint keys,MI_KEYDEF *keydefs, File dfile,file; int errpos,save_errno, create_mode= O_RDWR | O_TRUNC; myf create_flag; - uint fields,length,max_key_length,packed,pointer,real_length_diff, + uint fields,length,max_key_length,packed,pack_bytes,pointer,real_length_diff, key_length,info_length,key_segs,options,min_key_length_skip, base_pos,long_varchar_count,varchar_length, max_key_block_length,unique_key_parts,fulltext_keys,offset; @@ -189,11 +189,11 @@ int mi_create(const char *name,uint keys,MI_KEYDEF *keydefs, if (flags & HA_CREATE_RELIES_ON_SQL_LAYER) options|= HA_OPTION_RELIES_ON_SQL_LAYER; - packed=(packed+7)/8; + pack_bytes= (packed+7)/8; if (pack_reclength != INT_MAX32) pack_reclength+= reclength+packed + test(test_all_bits(options, HA_OPTION_CHECKSUM | HA_PACK_RECORD)); - min_pack_length+=packed; + min_pack_length+= pack_bytes; if (!ci->data_file_length && ci->max_rows) { @@ -547,9 +547,9 @@ int mi_create(const char *name,uint keys,MI_KEYDEF *keydefs, share.base.pack_reclength=reclength+ test(options & HA_OPTION_CHECKSUM); share.base.max_pack_length=pack_reclength; share.base.min_pack_length=min_pack_length; - share.base.pack_bits=packed; + share.base.pack_bits= pack_bytes; share.base.fields=fields; - share.base.pack_fields=packed; + share.base.pack_fields= packed; #ifdef USE_RAID share.base.raid_type=ci->raid_type; share.base.raid_chunks=ci->raid_chunks; diff --git a/storage/myisam/mi_rsamepos.c b/storage/myisam/mi_rsamepos.c index c4bd5fa16fa..d2dba64b0fd 100644 --- a/storage/myisam/mi_rsamepos.c +++ b/storage/myisam/mi_rsamepos.c @@ -33,7 +33,8 @@ int mi_rsame_with_pos(MI_INFO *info, byte *record, int inx, my_off_t filepos) DBUG_ENTER("mi_rsame_with_pos"); DBUG_PRINT("enter",("index: %d filepos: %ld", inx, (long) filepos)); - if (inx < -1 || inx >= 0 && ! mi_is_key_active(info->s->state.key_map, inx)) + if (inx < -1 || + (inx >= 0 && ! mi_is_key_active(info->s->state.key_map, inx))) { DBUG_RETURN(my_errno=HA_ERR_WRONG_INDEX); } diff --git a/storage/myisam/mi_test2.c b/storage/myisam/mi_test2.c index 357128b7a40..cf603f85630 100644 --- a/storage/myisam/mi_test2.c +++ b/storage/myisam/mi_test2.c @@ -603,7 +603,7 @@ int main(int argc, char *argv[]) if (mi_rsame(file,read_record2,(int) i)) goto err; if (bcmp(read_record,read_record2,reclength) != 0) { - printf("is_rsame didn't find same record\n"); + printf("mi_rsame didn't find same record\n"); goto end; } } diff --git a/storage/myisam/sort.c b/storage/myisam/sort.c index 0369be5db9d..50ba22e8beb 100644 --- a/storage/myisam/sort.c +++ b/storage/myisam/sort.c @@ -802,7 +802,7 @@ static uint NEAR_F read_to_buffer_varlen(IO_CACHE *fromfile, BUFFPEK *buffpek, register uint count; uint16 length_of_key = 0; uint idx; - uchar *buffp; + byte *buffp; if ((count=(uint) min((ha_rows) buffpek->max_keys,buffpek->count))) { @@ -889,7 +889,7 @@ merge_buffers(MI_SORT_PARAM *info, uint keys, IO_CACHE *from_file, for (buffpek= Fb ; buffpek <= Tb ; buffpek++) { count+= buffpek->count; - buffpek->base= strpos; + buffpek->base= (byte*) strpos; buffpek->max_keys=maxcount; strpos+= (uint) (error=(int) info->read_to_buffer(from_file,buffpek, sort_length)); @@ -927,7 +927,7 @@ merge_buffers(MI_SORT_PARAM *info, uint keys, IO_CACHE *from_file, { if (!(error=(int) info->read_to_buffer(from_file,buffpek,sort_length))) { - uchar *base=buffpek->base; + byte *base= buffpek->base; uint max_keys=buffpek->max_keys; VOID(queue_remove(&queue,0)); @@ -959,7 +959,7 @@ merge_buffers(MI_SORT_PARAM *info, uint keys, IO_CACHE *from_file, } } buffpek=(BUFFPEK*) queue_top(&queue); - buffpek->base=(uchar *) sort_keys; + buffpek->base= (byte*) sort_keys; buffpek->max_keys=keys; do { @@ -974,7 +974,7 @@ merge_buffers(MI_SORT_PARAM *info, uint keys, IO_CACHE *from_file, else { register uchar *end; - strpos= buffpek->key; + strpos= (uchar*) buffpek->key; for (end=strpos+buffpek->mem_count*sort_length; strpos != end ; strpos+=sort_length) diff --git a/support-files/magic b/support-files/magic index 9844142ba93..c8ca5875211 100644 --- a/support-files/magic +++ b/support-files/magic @@ -4,9 +4,13 @@ # 0 beshort 0xfe01 MySQL table definition file >2 byte x Version %d -0 belong&0xffffff00 0xfefe0300 MySQL MISAM index file +0 belong&0xffffff00 0xfefe0700 MySQL MISAM index file >3 byte x Version %d -0 belong&0xffffff00 0xfefe0700 MySQL MISAM compressed data file +0 belong&0xffffff00 0xfefe0800 MySQL MISAM compressed data file +>3 byte x Version %d +0 belong&0xffffff00 0xfefe0900 MySQL Maria index file +>3 byte x Version %d +0 belong&0xffffff00 0xfefe0A00 MySQL Maria compressed data file >3 byte x Version %d 0 belong&0xffffff00 0xfefe0500 MySQL ISAM index file >3 byte x Version %d -- cgit v1.2.1 From 7412f0fa0cd498f06fe04966a6f6161e8b32d0a2 Mon Sep 17 00:00:00 2001 From: unknown Date: Fri, 26 Jan 2007 13:32:02 +0200 Subject: After merge fixes Removed compiler warnings Fixed clashing function name in maria Disable maria tests from MySQL level for now BitKeeper/deleted/.del-ha_maria.cc: Rename: libmysqld/ha_maria.cc -> BitKeeper/deleted/.del-ha_maria.cc BitKeeper/etc/ignore: added libmysqld/ha_maria.cc --- added storage/maria/unittest/maria_control unittest/maria_control --- added *.Tpo --- added unittest/page_cache_test_file_1 --- added unittest/pagecache_debug.log --- added unittest/mysys/mf_pagecache_consist_1k-t-big unittest/mysys/mf_pagecache_consist_1kHC-t-big unittest/mysys/mf_pagecache_consist_1kRD-t-big unittest/mysys/mf_pagecache_consist_1kWR-t-big unittest/mysys/mf_pagecache_consist_64k-t-big unittest/mysys/mf_pagecache_consist_64kHC-t-big unittest/mysys/mf_pagecache_consist_64kRD-t-big unittest/mysys/mf_pagecache_consist_64kWR-t-big --- added unittest/mysys/mf_pagecache_single_64k-t-big Makefile.am: Don't run 'test-unit' by default (takes too long time) client/mysqldump.c: Fixed compiler warning include/lf.h: Remove compiler warnings about not used require_pins constant include/pagecache.h: LSN should be of type ulonglong (This fixes some compiler warnings) mysql-test/r/events_logs_tests.result: Make test predictable mysql-test/r/view.result: Make test results predictable mysql-test/t/disabled.def: Disable maria tests for a while mysql-test/t/events_logs_tests.test: Make test predictable mysql-test/t/view.test: Make test results predictable mysys/lf_alloc-pin.c: #warning ->QQ mysys/lf_hash.c: #warning ->QQ Removed compiler warnings mysys/mf_pagecache.c: Removed compiler warnings mysys/my_rename.c: Removed compiler warnings plugin/daemon_example/daemon_example.c: Remove compiler warning sql/ha_ndbcluster.cc: Remove compiler warning sql/udf_example.c: Remove compiler warning storage/maria/lockman.c: Changed #warnings to QQ comment Removed compiler warnings storage/maria/ma_blockrec.c: Removed compiler warnings storage/maria/ma_check.c: After merge fixes storage/maria/ma_key.c: After merge fixes storage/maria/ma_packrec.c: After merge fixes storage/maria/ma_rkey.c: After merge fixes storage/maria/ma_sort.c: After merge fixes storage/maria/ma_sp_defs.h: Rename clashing function name storage/maria/ma_sp_key.c: Rename clashing function name storage/maria/ma_test_all.res: New test results storage/maria/ma_unique.c: Fixed compiler warning storage/maria/tablockman.c: #warning -> QQ storage/maria/tablockman.h: #warning -> QQ storage/maria/trnman.c: #warning -> QQ storage/maria/unittest/lockman2-t.c: Removed compiler warnings storage/maria/unittest/ma_control_file-t.c: Removed warning for 'maria_control' file not found storage/maria/unittest/trnman-t.c: Removed compiler warnings storage/ndb/src/mgmapi/mgmapi.cpp: Remove compiler warnings unittest/mysys/mf_pagecache_consist.c: Removed compiler warnings unittest/mysys/my_atomic-t.c: Removed compiler warnings --- .bzrignore | 15 +++++ Makefile.am | 4 +- client/mysqldump.c | 4 +- include/lf.h | 3 +- include/pagecache.h | 2 +- mysql-test/r/events_logs_tests.result | 2 +- mysql-test/r/view.result | 11 ++++ mysql-test/t/disabled.def | 2 + mysql-test/t/events_logs_tests.test | 2 +- mysql-test/t/view.test | 18 +++++- mysys/lf_alloc-pin.c | 2 +- mysys/lf_hash.c | 13 ++-- mysys/mf_pagecache.c | 98 ++++++++++++++++-------------- mysys/my_rename.c | 3 +- plugin/daemon_example/daemon_example.c | 4 +- sql/ha_ndbcluster.cc | 4 +- sql/udf_example.c | 2 +- storage/maria/lockman.c | 27 ++++---- storage/maria/ma_blockrec.c | 11 ++-- storage/maria/ma_check.c | 36 +++++++---- storage/maria/ma_key.c | 6 +- storage/maria/ma_packrec.c | 2 +- storage/maria/ma_rkey.c | 3 +- storage/maria/ma_sort.c | 3 +- storage/maria/ma_sp_defs.h | 4 +- storage/maria/ma_sp_key.c | 4 +- storage/maria/ma_test_all.res | 91 ++++++++++++++------------- storage/maria/ma_unique.c | 2 +- storage/maria/tablockman.c | 6 +- storage/maria/tablockman.h | 2 +- storage/maria/trnman.c | 2 +- storage/maria/unittest/lockman2-t.c | 21 ++++--- storage/maria/unittest/ma_control_file-t.c | 12 ++-- storage/maria/unittest/trnman-t.c | 5 +- storage/ndb/src/mgmapi/mgmapi.cpp | 12 ++-- unittest/mysys/mf_pagecache_consist.c | 2 +- unittest/mysys/my_atomic-t.c | 2 +- 37 files changed, 261 insertions(+), 181 deletions(-) diff --git a/.bzrignore b/.bzrignore index bd7941f77ab..7dd012950d8 100644 --- a/.bzrignore +++ b/.bzrignore @@ -2946,3 +2946,18 @@ win/vs71cache.txt win/vs8cache.txt zlib/*.ds? zlib/*.vcproj +libmysqld/ha_maria.cc +storage/maria/unittest/maria_control +unittest/maria_control +*.Tpo +unittest/page_cache_test_file_1 +unittest/pagecache_debug.log +unittest/mysys/mf_pagecache_consist_1k-t-big +unittest/mysys/mf_pagecache_consist_1kHC-t-big +unittest/mysys/mf_pagecache_consist_1kRD-t-big +unittest/mysys/mf_pagecache_consist_1kWR-t-big +unittest/mysys/mf_pagecache_consist_64k-t-big +unittest/mysys/mf_pagecache_consist_64kHC-t-big +unittest/mysys/mf_pagecache_consist_64kRD-t-big +unittest/mysys/mf_pagecache_consist_64kWR-t-big +unittest/mysys/mf_pagecache_single_64k-t-big diff --git a/Makefile.am b/Makefile.am index 81656d67f74..ed2f8d99414 100644 --- a/Makefile.am +++ b/Makefile.am @@ -137,9 +137,9 @@ test-binlog-statement: cd mysql-test ; \ ./mysql-test-run.pl $(force) --mysqld=--binlog-format=statement -test: test-unit test-ns test-pr +test: test-ns test-pr -test-full: test test-nr test-ps +test-full: test-unit test test-nr test-ps test-force: $(MAKE) force=--force test diff --git a/client/mysqldump.c b/client/mysqldump.c index 5d87701fe10..0d6b5678e52 100644 --- a/client/mysqldump.c +++ b/client/mysqldump.c @@ -2877,7 +2877,7 @@ static int dump_tablespaces_for_tables(char *db, char **table_names, int tables) dynstr_trunc(&where, 1); dynstr_append(&where,"))"); - DBUG_PRINT("info",("Dump TS for Tables where: %s",where)); + DBUG_PRINT("info",("Dump TS for Tables where: %s",where.str)); r= dump_tablespaces(where.str); dynstr_free(&where); return r; @@ -2907,7 +2907,7 @@ static int dump_tablespaces_for_databases(char** databases) dynstr_trunc(&where, 1); dynstr_append(&where,"))"); - DBUG_PRINT("info",("Dump TS for DBs where: %s",where)); + DBUG_PRINT("info",("Dump TS for DBs where: %s",where.str)); r= dump_tablespaces(where.str); dynstr_free(&where); return r; diff --git a/include/lf.h b/include/lf.h index c161bd5044d..39a47e6568a 100644 --- a/include/lf.h +++ b/include/lf.h @@ -135,8 +135,9 @@ typedef struct { */ #if defined(__GNUC__) && defined(MY_LF_EXTRA_DEBUG) #define LF_REQUIRE_PINS(N) \ - static const char require_pins[LF_PINBOX_PINS-N]; \ + static const char require_pins[LF_PINBOX_PINS-N] __attribute__ ((unused)); \ static const int LF_NUM_PINS_IN_THIS_FILE= N + #define _lf_pin(PINS, PIN, ADDR) \ ( \ assert(PIN < LF_NUM_PINS_IN_THIS_FILE), \ diff --git a/include/pagecache.h b/include/pagecache.h index 418e0203f64..b698a36a5e4 100644 --- a/include/pagecache.h +++ b/include/pagecache.h @@ -73,7 +73,7 @@ typedef void *PAGECACHE_PAGE_LINK; /* TODO: move to loghandler emulator */ typedef void LOG_HANDLER; -typedef void *LSN; +typedef ulonglong LSN; /* file descriptor for Maria */ typedef struct st_pagecache_file diff --git a/mysql-test/r/events_logs_tests.result b/mysql-test/r/events_logs_tests.result index 335ca848387..b067b2011e1 100644 --- a/mysql-test/r/events_logs_tests.result +++ b/mysql-test/r/events_logs_tests.result @@ -83,7 +83,7 @@ slo_val val 4 0 1 0 "Check slow log. Should see 1 row because 4 is over the threshold of 3 for GLOBAL, though under SESSION which is 10" -SELECT user_host, query_time, db, sql_text FROM mysql.slow_log; +SELECT user_host, query_time, db, sql_text FROM mysql.slow_log where sql_text <> "DROP EVENT long_event"; user_host query_time db sql_text USER_HOST SLEEPVAL events_test INSERT INTO slow_event_test SELECT @@long_query_time, SLEEP(2) DROP EVENT long_event2; diff --git a/mysql-test/r/view.result b/mysql-test/r/view.result index 4bc122cc97b..eb090658560 100644 --- a/mysql-test/r/view.result +++ b/mysql-test/r/view.result @@ -2684,10 +2684,12 @@ FROM t1 HAVING Age < 75; SHOW CREATE VIEW v1; View Create View v1 CREATE ALGORITHM=UNDEFINED DEFINER=`root`@`localhost` SQL SECURITY DEFINER VIEW `v1` AS select (year(now()) - year(`t1`.`DOB`)) AS `Age` from `t1` having (`Age` < 75) +set timestamp=1136066400; SELECT (year(now())-year(DOB)) AS Age FROM t1 HAVING Age < 75; Age 42 38 +set timestamp=1136066400; SELECT * FROM v1; Age 42 @@ -3014,6 +3016,15 @@ i j 6 3 DROP VIEW v1, v2; DROP TABLE t1; +DROP VIEW IF EXISTS v1; +CREATE VIEW v1 AS SELECT 'The\ZEnd'; +SELECT * FROM v1; +TheEnd +ThezEnd +SHOW CREATE VIEW v1; +View Create View +v1 CREATE ALGORITHM=UNDEFINED DEFINER=`root`@`localhost` SQL SECURITY DEFINER VIEW `v1` AS select _latin1'ThezEnd' AS `TheEnd` +DROP VIEW v1; End of 5.0 tests. DROP DATABASE IF EXISTS `d-1`; CREATE DATABASE `d-1`; diff --git a/mysql-test/t/disabled.def b/mysql-test/t/disabled.def index c36ac1bd168..3b063841512 100644 --- a/mysql-test/t/disabled.def +++ b/mysql-test/t/disabled.def @@ -38,3 +38,5 @@ ndb_autodiscover3 : bug#21806 flush2 : Bug#24805 Pushbuild can't handle test with --disable-log-bin +maria : Until maria is fully functional +ps_maria : Until maria is fully functional diff --git a/mysql-test/t/events_logs_tests.test b/mysql-test/t/events_logs_tests.test index aee16595ef9..001004386e4 100644 --- a/mysql-test/t/events_logs_tests.test +++ b/mysql-test/t/events_logs_tests.test @@ -91,7 +91,7 @@ CREATE EVENT long_event2 ON SCHEDULE EVERY 1 MINUTE DO INSERT INTO slow_event_te SELECT * FROM slow_event_test; --echo "Check slow log. Should see 1 row because 4 is over the threshold of 3 for GLOBAL, though under SESSION which is 10" --replace_column 1 USER_HOST 2 SLEEPVAL -SELECT user_host, query_time, db, sql_text FROM mysql.slow_log; +SELECT user_host, query_time, db, sql_text FROM mysql.slow_log where sql_text <> "DROP EVENT long_event"; DROP EVENT long_event2; --echo "Make it quite long" SET SESSION long_query_time=300; diff --git a/mysql-test/t/view.test b/mysql-test/t/view.test index 7175db1db4e..3c5308a935c 100644 --- a/mysql-test/t/view.test +++ b/mysql-test/t/view.test @@ -2553,7 +2553,9 @@ CREATE VIEW v1 AS FROM t1 HAVING Age < 75; SHOW CREATE VIEW v1; +set timestamp=1136066400; SELECT (year(now())-year(DOB)) AS Age FROM t1 HAVING Age < 75; +set timestamp=1136066400; SELECT * FROM v1; DROP VIEW v1; @@ -2906,6 +2908,7 @@ DROP FUNCTION f1; DROP VIEW v1; DROP TABLE t1; +# # Bug #16813 (WITH CHECK OPTION doesn't work with UPDATE) # CREATE TABLE t1(id INT UNSIGNED NOT NULL AUTO_INCREMENT PRIMARY KEY, val INT UNSIGNED NOT NULL); @@ -2919,7 +2922,6 @@ UPDATE v1 SET val=6 WHERE id=2; DROP VIEW v1; DROP TABLE t1; - # # BUG#22584: last_insert_id not updated after inserting a record # through a updatable view @@ -2957,6 +2959,20 @@ SELECT * FROM t1; DROP VIEW v1, v2; DROP TABLE t1; +# +# BUG#24293: '\Z' token is not handled correctly in views +# + +--disable_warnings +DROP VIEW IF EXISTS v1; +--enable_warnings + +CREATE VIEW v1 AS SELECT 'The\ZEnd'; +SELECT * FROM v1; + +SHOW CREATE VIEW v1; + +DROP VIEW v1; --echo End of 5.0 tests. diff --git a/mysys/lf_alloc-pin.c b/mysys/lf_alloc-pin.c index d0fa29ddaaf..e964553a64c 100644 --- a/mysys/lf_alloc-pin.c +++ b/mysys/lf_alloc-pin.c @@ -1,4 +1,4 @@ -#warning TODO multi-pinbox +/* QQ: TODO multi-pinbox */ /* Copyright (C) 2000 MySQL AB This program is free software; you can redistribute it and/or modify diff --git a/mysys/lf_hash.c b/mysys/lf_hash.c index b2ad7a93ace..fb2fb88492f 100644 --- a/mysys/lf_hash.c +++ b/mysys/lf_hash.c @@ -86,7 +86,7 @@ retry: if (!cursor->curr) return 0; do { -#warning XXX or goto retry ? + /* QQ: XXX or goto retry ? */ link= cursor->curr->link; cursor->next= PTR(link); _lf_pin(pins, 0, cursor->next); @@ -105,7 +105,8 @@ retry: { int r= 1; if (cur_hashnr > hashnr || - (r= my_strnncoll(cs, cur_key, cur_keylen, key, keylen)) >= 0) + (r= my_strnncoll(cs, (uchar*) cur_key, cur_keylen, (uchar*) key, + keylen)) >= 0) return !r; } cursor->prev= &(cursor->curr->link); @@ -243,7 +244,8 @@ static inline const byte* hash_key(const LF_HASH *hash, static inline uint calc_hash(LF_HASH *hash, const byte *key, uint keylen) { ulong nr1= 1, nr2= 4; - hash->charset->coll->hash_sort(hash->charset, key, keylen, &nr1, &nr2); + hash->charset->coll->hash_sort(hash->charset, (uchar*) key, keylen, + &nr1, &nr2); return nr1 & INT_MAX32; } @@ -375,7 +377,7 @@ void *lf_hash_search(LF_HASH *hash, LF_PINS *pins, const void *key, uint keylen) return found ? found+1 : 0; } -static char *dummy_key= ""; +static const char *dummy_key= ""; static void initialize_bucket(LF_HASH *hash, LF_SLIST * volatile *node, uint bucket, LF_PINS *pins) @@ -387,7 +389,7 @@ static void initialize_bucket(LF_HASH *hash, LF_SLIST * volatile *node, if (*el == NULL && bucket) initialize_bucket(hash, el, parent, pins); dummy->hashnr= my_reverse_bits(bucket); - dummy->key= dummy_key; + dummy->key= (char*) dummy_key; dummy->keylen= 0; if ((cur= linsert(el, hash->charset, dummy, pins, 0))) { @@ -396,4 +398,3 @@ static void initialize_bucket(LF_HASH *hash, LF_SLIST * volatile *node, } my_atomic_casptr((void **)node, (void **)&tmp, dummy); } - diff --git a/mysys/mf_pagecache.c b/mysys/mf_pagecache.c index 5f4e8d1a97d..0116dfabfa1 100755 --- a/mysys/mf_pagecache.c +++ b/mysys/mf_pagecache.c @@ -47,7 +47,6 @@ #include #include - /* Some compilation flags have been added specifically for this module to control the following: @@ -429,7 +428,6 @@ error: #define FLUSH_CACHE 2000 /* sort this many blocks at once */ -static int flush_all_key_blocks(PAGECACHE *pagecache); #ifdef THREAD static void link_into_queue(PAGECACHE_WQUEUE *wqueue, struct st_my_thread_var *thread); @@ -793,6 +791,40 @@ err: } +/* + Flush all blocks in the key cache to disk +*/ + +#ifdef NOT_USED +static int flush_all_key_blocks(PAGECACHE *pagecache) +{ +#if defined(PAGECACHE_DEBUG) + uint cnt=0; +#endif + while (pagecache->blocks_changed > 0) + { + PAGECACHE_BLOCK_LINK *block; + for (block= pagecache->used_last->next_used ; ; block=block->next_used) + { + if (block->hash_link) + { +#if defined(PAGECACHE_DEBUG) + cnt++; + KEYCACHE_DBUG_ASSERT(cnt <= pagecache->blocks_used); +#endif + if (flush_pagecache_blocks_int(pagecache, &block->hash_link->file, + FLUSH_RELEASE)) + return 1; + break; + } + if (block == pagecache->used_last) + break; + } + } + return 0; +} +#endif /* NOT_USED */ + /* Resize a key cache @@ -827,7 +859,7 @@ err: resizing, due to the page locking specific to this page cache. So we disable it for now. */ -#if 0 /* keep disabled until code is fixed see above !! */ +#if NOT_USED /* keep disabled until code is fixed see above !! */ int resize_pagecache(PAGECACHE *pagecache, my_size_t use_mem, uint division_limit, uint age_threshold) @@ -1383,7 +1415,7 @@ static void unlink_block(PAGECACHE *pagecache, PAGECACHE_BLOCK_LINK *block) (ulong)block, BLOCK_NUMBER(pagecache, block), block->status, block->requests, pagecache->blocks_available)); BLOCK_INFO(block); - KEYCACHE_DBUG_ASSERT(pagecache->blocks_available >= 0); + KEYCACHE_DBUG_ASSERT((int) pagecache->blocks_available >= 0); #endif } @@ -2511,7 +2543,7 @@ void pagecache_unlock_page(PAGECACHE *pagecache, DBUG_ASSERT(lock == PAGECACHE_LOCK_WRITE_UNLOCK && pin == PAGECACHE_UNPIN); /* TODO: insert LSN writing code */ - DBUG_ASSERT(first_REDO_LSN_for_page > 0); + DBUG_ASSERT(first_REDO_LSN_for_page != 0); set_if_bigger(block->rec_lsn, first_REDO_LSN_for_page); } @@ -2675,7 +2707,7 @@ void pagecache_unlock(PAGECACHE *pagecache, DBUG_ASSERT(lock == PAGECACHE_LOCK_WRITE_UNLOCK && pin == PAGECACHE_UNPIN); /* TODO: insert LSN writing code */ - DBUG_ASSERT(first_REDO_LSN_for_page > 0); + DBUG_ASSERT(first_REDO_LSN_for_page != 0); set_if_bigger(block->rec_lsn, first_REDO_LSN_for_page); } @@ -3251,7 +3283,9 @@ restart: #ifndef DBUG_OFF int rc= #endif -#warning we are doing an unlock here, so need to give the page its rec_lsn! + /* + QQ: We are doing an unlock here, so need to give the page its rec_lsn + */ pagecache_make_lock_and_pin(pagecache, block, write_lock_change_table[lock].unlock_lock, write_pin_change_table[pin].unlock_pin); @@ -3612,11 +3646,14 @@ restart: else { /* Link the block into a list of blocks 'in switch' */ -#warning this unlink_changed() is a serious problem for Maria's Checkpoint: it \ -removes a page from the list of dirty pages, while it's still dirty. A \ - solution is to abandon first_in_switch, just wait for this page to be \ - flushed by somebody else, and loop. TODO: check all places where we remove a \ - page from the list of dirty pages + /* QQ: + #warning this unlink_changed() is a serious problem for + Maria's Checkpoint: it removes a page from the list of dirty + pages, while it's still dirty. A solution is to abandon + first_in_switch, just wait for this page to be + flushed by somebody else, and loop. TODO: check all places + where we remove a page from the list of dirty pages + */ unlink_changed(block); link_changed(block, &first_in_switch); } @@ -3727,39 +3764,6 @@ int flush_pagecache_blocks(PAGECACHE *pagecache, } -/* - Flush all blocks in the key cache to disk -*/ - -static int flush_all_key_blocks(PAGECACHE *pagecache) -{ -#if defined(PAGECACHE_DEBUG) - uint cnt=0; -#endif - while (pagecache->blocks_changed > 0) - { - PAGECACHE_BLOCK_LINK *block; - for (block= pagecache->used_last->next_used ; ; block=block->next_used) - { - if (block->hash_link) - { -#if defined(PAGECACHE_DEBUG) - cnt++; - KEYCACHE_DBUG_ASSERT(cnt <= pagecache->blocks_used); -#endif - if (flush_pagecache_blocks_int(pagecache, &block->hash_link->file, - FLUSH_RELEASE)) - return 1; - break; - } - if (block == pagecache->used_last) - break; - } - } - return 0; -} - - /* Reset the counters of a key cache. @@ -3876,7 +3880,7 @@ my_bool pagecache_collect_changed_blocks_with_lsn(PAGECACHE *pagecache, 1) and 2) are critical problems. TODO: fix this when Monty has explained how he writes BLOB pages. */ - if (0 == block->rec_lsn) + if (block->rec_lsn == 0) { DBUG_ASSERT(0); goto err; @@ -3908,7 +3912,7 @@ my_bool pagecache_collect_changed_blocks_with_lsn(PAGECACHE *pagecache, ptr+= 4; int4store(ptr, block->hash_link->pageno); ptr+= 4; - int8store(ptr, (ulonglong)block->rec_lsn); + int8store(ptr, (ulonglong) block->rec_lsn); ptr+= 8; set_if_bigger(*max_lsn, block->rec_lsn); } diff --git a/mysys/my_rename.c b/mysys/my_rename.c index 8c2a354324b..c3870db177d 100644 --- a/mysys/my_rename.c +++ b/mysys/my_rename.c @@ -17,8 +17,9 @@ #include "mysys_priv.h" #include #include "mysys_err.h" - +#include "m_string.h" #undef my_rename + /* On unix rename deletes to file if it exists */ int my_rename(const char *from, const char *to, myf MyFlags) diff --git a/plugin/daemon_example/daemon_example.c b/plugin/daemon_example/daemon_example.c index d302aec6515..786ff199c93 100644 --- a/plugin/daemon_example/daemon_example.c +++ b/plugin/daemon_example/daemon_example.c @@ -40,7 +40,7 @@ 1 failure (cannot happen) */ -static int daemon_example_plugin_init(void *p) +static int daemon_example_plugin_init(void *p __attribute__ ((unused))) { return(0); } @@ -59,7 +59,7 @@ static int daemon_example_plugin_init(void *p) */ -static int daemon_example_plugin_deinit(void *p) +static int daemon_example_plugin_deinit(void *p __attribute__ ((unused))) { return(0); } diff --git a/sql/ha_ndbcluster.cc b/sql/ha_ndbcluster.cc index 0362b8bf215..0c3a79baacc 100644 --- a/sql/ha_ndbcluster.cc +++ b/sql/ha_ndbcluster.cc @@ -7297,9 +7297,9 @@ static byte *ndbcluster_get_key(NDB_SHARE *share,uint *length, static void print_share(const char* where, NDB_SHARE* share) { fprintf(DBUG_FILE, - "%s %s.%s: use_count: %u, commit_count: %llu\n", + "%s %s.%s: use_count: %u, commit_count: %lu\n", where, share->db, share->table_name, share->use_count, - share->commit_count); + (ulong) share->commit_count); fprintf(DBUG_FILE, " - key: %s, key_length: %d\n", share->key, share->key_length); diff --git a/sql/udf_example.c b/sql/udf_example.c index bbab47e253d..a71cc4b44f2 100644 --- a/sql/udf_example.c +++ b/sql/udf_example.c @@ -1087,7 +1087,7 @@ my_bool is_const_init(UDF_INIT *initid, UDF_ARGS *args, char *message) strmov(message, "IS_CONST accepts only one argument"); return 1; } - initid->ptr= (char*)((args->args[0] != NULL) ? 1 : 0); + initid->ptr= (char*)(size_t)((args->args[0] != NULL) ? 1 : 0); return 0; } diff --git a/storage/maria/lockman.c b/storage/maria/lockman.c index 7672fadb068..fdacda84875 100644 --- a/storage/maria/lockman.c +++ b/storage/maria/lockman.c @@ -1,6 +1,7 @@ -#warning TODO - allocate everything from dynarrays !!! (benchmark) -#warning TODO instant duration locks -#warning automatically place S instead of LS if possible +/* QQ: TODO - allocate everything from dynarrays !!! (benchmark) */ +/* QQ: TODO instant duration locks */ +/* QQ: #warning automatically place S instead of LS if possible */ + /* Copyright (C) 2006 MySQL AB This program is free software; you can redistribute it and/or modify @@ -218,7 +219,7 @@ typedef struct lockman_lock { struct lockman_lock *lonext; intptr volatile link; uint32 hashnr; -#warning TODO - remove hashnr from LOCK + /* QQ: TODO - remove hashnr from LOCK */ uint16 loid; uchar lock; /* sizeof(uchar) <= sizeof(enum) */ uchar flags; @@ -428,9 +429,11 @@ static int lockinsert(LOCK * volatile *head, LOCK *node, LF_PINS *pins, } if (res & LOCK_UPGRADE) cursor.upgrade_from->flags|= IGNORE_ME; -#warning is this OK ? if a reader has already read upgrade_from, \ - it may find it conflicting with node :( -#warning another bug - see the last test from test_lockman_simple() + /* + QQ: is this OK ? if a reader has already read upgrade_from, + it may find it conflicting with node :( + - see the last test from test_lockman_simple() + */ } } while (res == REPEAT_ONCE_MORE); @@ -673,7 +676,7 @@ enum lockman_getlock_result lockman_getlock(LOCKMAN *lm, LOCK_OWNER *lo, belong to _some_ LOCK_OWNER. It means, we can never free() a LOCK_OWNER, if there're other active LOCK_OWNERs. */ -#warning race condition here + /* QQ: race condition here */ pthread_mutex_lock(wait_for_lo->mutex); if (DELETED(blocker->link)) { @@ -749,7 +752,7 @@ int lockman_release_locks(LOCKMAN *lm, LOCK_OWNER *lo) } #ifdef MY_LF_EXTRA_DEBUG -static char *lock2str[]= +static const char *lock2str[]= { "N", "S", "X", "IS", "IX", "SIX", "LS", "LX", "SLX", "LSIX" }; /* NOTE @@ -764,8 +767,9 @@ void print_lockhash(LOCKMAN *lm) intptr next= el->link; if (el->hashnr & 1) { - printf("0x%08x { resource %llu, loid %u, lock %s", - el->hashnr, el->resource, el->loid, lock2str[el->lock]); + printf("0x%08lx { resource %lu, loid %u, lock %s", + (long) el->hashnr, (ulong) el->resource, el->loid, + lock2str[el->lock]); if (el->flags & IGNORE_ME) printf(" IGNORE_ME"); if (el->flags & UPGRADED) printf(" UPGRADED"); if (el->flags & ACTIVE) printf(" ACTIVE"); @@ -781,4 +785,3 @@ void print_lockhash(LOCKMAN *lm) } } #endif - diff --git a/storage/maria/ma_blockrec.c b/storage/maria/ma_blockrec.c index 6eac151d76e..f1345b2c2f3 100644 --- a/storage/maria/ma_blockrec.c +++ b/storage/maria/ma_blockrec.c @@ -967,7 +967,8 @@ static my_bool write_full_pages(MARIA_HA *info, my_off_t position; DBUG_ENTER("write_full_pages"); DBUG_PRINT("enter", ("length: %lu page: %lu page_count: %lu", - length, (ulong) block->page, block->page_count)); + (ulong) length, (ulong) block->page, + (ulong) block->page_count)); info->keybuff_used= 1; page= block->page; @@ -988,7 +989,7 @@ static my_bool write_full_pages(MARIA_HA *info, page= block->page; page_count= block->page_count - 1; DBUG_PRINT("info", ("page: %lu page_count: %lu", - (ulong) block->page, block->page_count)); + (ulong) block->page, (ulong) block->page_count)); position= (page + page_count + 1) * block_size; if (info->state->data_file_length < position) @@ -2387,18 +2388,18 @@ int _ma_read_block_record2(MARIA_HA *info, byte *record, blob_buffer+= blob_length; break; } -#ifdef EXTRA_DEBUG default: +#ifdef EXTRA_DEBUG DBUG_ASSERT(0); /* purecov: deadcode */ - goto err; #endif + goto err; } continue; } if (row_extents) { - DBUG_PRINT("info", ("Row read: page_count: %lu extent_count: %lu", + DBUG_PRINT("info", ("Row read: page_count: %u extent_count: %u", extent.page_count, extent.extent_count)); *extent.tail_positions= 0; /* End marker */ if (extent.page_count) diff --git a/storage/maria/ma_check.c b/storage/maria/ma_check.c index 682ce276a5b..ccce19de994 100644 --- a/storage/maria/ma_check.c +++ b/storage/maria/ma_check.c @@ -1078,6 +1078,10 @@ static int check_dynamic_record(HA_CHECK *param, MARIA_HA *info, int extend, char llbuff[22],llbuff2[22],llbuff3[22]; DBUG_ENTER("check_dynamic_record"); + LINT_INIT(left_length); + LINT_INIT(start_recpos); + LINT_INIT(to); + pos= 0; while (pos < info->state->data_file_length) { @@ -1096,7 +1100,8 @@ static int check_dynamic_record(HA_CHECK *param, MARIA_HA *info, int extend, (flag ? 0 : READING_NEXT) | READING_HEADER)) { _ma_check_print_error(param, - "got error: %d when reading datafile at position: %s", + "got error: %d when reading datafile at " + "position: %s", my_errno, llstr(start_block, llbuff)); DBUG_RETURN(1); } @@ -1309,7 +1314,8 @@ static int check_compressed_record(HA_CHECK *param, MARIA_HA *info, int extend, start_recpos= pos; param->splits++; VOID(_ma_pack_get_block_info(info, &info->bit_buff, &block_info, - &info->rec_buff, -1, start_recpos)); + &info->rec_buff, &info->rec_buff_size, -1, + start_recpos)); pos=block_info.filepos+block_info.rec_len; if (block_info.rec_len < (uint) info->s->min_pack_length || block_info.rec_len > (uint) info->s->max_pack_length) @@ -1727,7 +1733,7 @@ int maria_chk_data_link(HA_CHECK *param, MARIA_HA *info,int extend) param->used= param->link_used= param->splits= param->del_length= 0; param->tmp_record_checksum= param->glob_crc= 0; param->err_count= 0; - LINT_INIT(left_length); LINT_INIT(start_recpos); LINT_INIT(to); + error= 0; param->empty= info->s->pack.header_length; @@ -2206,7 +2212,7 @@ static int writekeys(MARIA_SORT_PARAM *sort_param) } /* Remove checksum that was added to glob_crc in sort_get_next_record */ if (sort_param->calc_checksum) - sort_param->glob_crc-= info->cur_row.checksum; + sort_param->sort_info->param->glob_crc-= info->cur_row.checksum; DBUG_PRINT("error",("errno: %d",my_errno)); DBUG_RETURN(-1); } /* writekeys */ @@ -2317,8 +2323,7 @@ int maria_sort_index(HA_CHECK *param, register MARIA_HA *info, my_string name) param->temp_filename); DBUG_RETURN(-1); } - if (new_header_length && - maria_filecopy(param, new_file,share->kfile,0L, + if (maria_filecopy(param, new_file,share->kfile,0L, (ulong) share->base.keystart, "headerblock")) goto err; @@ -3070,8 +3075,9 @@ int maria_repair_parallel(HA_CHECK *param, register MARIA_HA *info, param->temp_filename); goto err; } - if (maria_filecopy(param, new_file,info->dfile,0L,new_header_length, - "datafile-header")) + if (new_header_length && + maria_filecopy(param, new_file,info->dfile,0L,new_header_length, + "datafile-header")) goto err; if (param->testflag & T_UNPACK) restore_data_file_type(share); @@ -3814,16 +3820,18 @@ static int sort_get_next_record(MARIA_SORT_PARAM *sort_param) if (left_length < block_info.data_len || ! block_info.data_len) { _ma_check_print_info(param, - "Found block with too small length at %s; Skipped", - llstr(sort_param->start_recpos,llbuff)); + "Found block with too small length at %s; " + "Skipped", + llstr(sort_param->start_recpos,llbuff)); goto try_next; } if (block_info.filepos + block_info.data_len > sort_param->read_cache.end_of_file) { _ma_check_print_info(param, - "Found block that points outside data file at %s", - llstr(sort_param->start_recpos,llbuff)); + "Found block that points outside data file " + "at %s", + llstr(sort_param->start_recpos,llbuff)); goto try_next; } /* @@ -3923,7 +3931,9 @@ static int sort_get_next_record(MARIA_SORT_PARAM *sort_param) } sort_param->start_recpos=sort_param->pos; if (_ma_pack_get_block_info(info, &sort_param->bit_buff, &block_info, - &sort_param->rec_buff, -1, sort_param->pos)) + &sort_param->rec_buff, + &sort_param->rec_buff_size, -1, + sort_param->pos)) DBUG_RETURN(-1); if (!block_info.rec_len && sort_param->pos + MEMMAP_EXTRA_MARGIN == diff --git a/storage/maria/ma_key.c b/storage/maria/ma_key.c index deab2b9c983..036fd305c4d 100644 --- a/storage/maria/ma_key.c +++ b/storage/maria/ma_key.c @@ -64,7 +64,7 @@ uint _ma_make_key(register MARIA_HA *info, uint keynr, byte *key, TODO: nulls processing */ #ifdef HAVE_SPATIAL - DBUG_RETURN(sp_make_key(info,keynr, key,record,filepos)); + DBUG_RETURN(_ma_sp_make_key(info,keynr, key,record,filepos)); #else DBUG_ASSERT(0); /* maria_open should check that this never happens*/ #endif @@ -113,10 +113,10 @@ uint _ma_make_key(register MARIA_HA *info, uint keynr, byte *key, } else { - byte *end= pos + length; + const byte *end= pos + length; while (pos < end && pos[0] == ' ') pos++; - length=(uint) (end-pos); + length= (uint) (end-pos); } FIX_LENGTH(cs, pos, length, char_length); store_key_length_inc(key,char_length); diff --git a/storage/maria/ma_packrec.c b/storage/maria/ma_packrec.c index 6e481c0bc6d..7c875e7b91d 100644 --- a/storage/maria/ma_packrec.c +++ b/storage/maria/ma_packrec.c @@ -1288,7 +1288,7 @@ _ma_mempack_get_block_info(MARIA_HA *maria, header+= read_pack_length((uint) maria->s->pack.version, header, &info->blob_len); /* _ma_alloc_rec_buff sets my_errno on error */ - if (_ma_alloc_buffer(rec_buff_p, rec_buff_size, + if (_ma_alloc_buffer(rec_buff_p, rec_buff_size_p, info->blob_len + maria->s->base.extra_rec_buff_size)) return 0; /* not enough memory */ bit_buff->blob_pos= (uchar*) *rec_buff_p; diff --git a/storage/maria/ma_rkey.c b/storage/maria/ma_rkey.c index 17eae760a3a..ad27d3c286c 100644 --- a/storage/maria/ma_rkey.c +++ b/storage/maria/ma_rkey.c @@ -123,7 +123,8 @@ int maria_rkey(MARIA_HA *info, byte *buf, int inx, const byte *key, value. */ if (search_flag == HA_READ_KEY_EXACT && - ha_key_cmp(keyinfo->seg, key_buff, info->lastkey, use_key_length, + ha_key_cmp(keyinfo->seg, (uchar*) key_buff, + (uchar*) info->lastkey, use_key_length, SEARCH_FIND, not_used)) { my_errno= HA_ERR_KEY_NOT_FOUND; diff --git a/storage/maria/ma_sort.c b/storage/maria/ma_sort.c index b63f99379ea..9134645f847 100644 --- a/storage/maria/ma_sort.c +++ b/storage/maria/ma_sort.c @@ -534,8 +534,7 @@ int _ma_thr_write_keys(MARIA_SORT_PARAM *sort_param) (ulonglong) info->state->records); } my_free((gptr) sinfo->sort_keys,MYF(0)); - my_free(_ma_get_rec_buff_ptr(info, sinfo->rec_buff), - MYF(MY_ALLOW_ZERO_PTR)); + my_free(sinfo->rec_buff, MYF(MY_ALLOW_ZERO_PTR)); sinfo->sort_keys=0; } diff --git a/storage/maria/ma_sp_defs.h b/storage/maria/ma_sp_defs.h index 8b9dd204ded..6aac741bb2c 100644 --- a/storage/maria/ma_sp_defs.h +++ b/storage/maria/ma_sp_defs.h @@ -41,8 +41,8 @@ enum wkbByteOrder wkbNDR = 1 /* Little Endian */ }; -uint sp_make_key(register MARIA_HA *info, uint keynr, byte *key, - const byte *record, my_off_t filepos); +uint _ma_sp_make_key(register MARIA_HA *info, uint keynr, byte *key, + const byte *record, my_off_t filepos); #endif /*HAVE_SPATIAL*/ #endif /* _SP_DEFS_H */ diff --git a/storage/maria/ma_sp_key.c b/storage/maria/ma_sp_key.c index 79345550dd9..b365f8deb0f 100644 --- a/storage/maria/ma_sp_key.c +++ b/storage/maria/ma_sp_key.c @@ -37,8 +37,8 @@ static void get_double(double *d, const byte *pos) float8get(*d, pos); } -uint sp_make_key(register MARIA_HA *info, uint keynr, byte *key, - const byte *record, my_off_t filepos) +uint _ma_sp_make_key(register MARIA_HA *info, uint keynr, byte *key, + const byte *record, my_off_t filepos) { HA_KEYSEG *keyseg; MARIA_KEYDEF *keyinfo = &info->s->keyinfo[keynr]; diff --git a/storage/maria/ma_test_all.res b/storage/maria/ma_test_all.res index 7ffd3378b51..57b0feeeae8 100644 --- a/storage/maria/ma_test_all.res +++ b/storage/maria/ma_test_all.res @@ -1,53 +1,62 @@ -maria_chk: MARIA file test1 -maria_chk: warning: Size of indexfile is: 8192 Should be: 16384 -MARIA-table 'test1' is usable but should be fixed +Running tests with dynamic row format +Running tests with static row format +Running tests with block row format ma_test2 -s -L -K -R1 -m2000 ; Should give error 135 -Error: 135 in write at record: 1105 +Error: 135 in write at record: 1099 got error: 135 when using MARIA-database +./maria_chk -sm test2 will warn that 'Datafile is almost full' maria_chk: MARIA file test2 -maria_chk: warning: Datafile is almost full, 65532 of 65534 used +maria_chk: warning: Datafile is almost full, 65516 of 65534 used MARIA-table 'test2' is usable but should be fixed -Commands Used count Errors Recover errors -open 1 0 0 -write 50 0 0 -update 5 0 0 -delete 50 0 0 -close 1 0 0 -extra 6 0 0 -Total 113 0 0 -Commands Used count Errors Recover errors -open 2 0 0 -write 100 0 0 -update 10 0 0 -delete 100 0 0 -close 2 0 0 -extra 12 0 0 -Total 226 0 0 - -real 0m0.994s -user 0m0.432s -sys 0m0.184s - -real 0m2.153s -user 0m1.196s -sys 0m0.228s - -real 0m1.483s -user 0m0.772s + +real 0m0.808s +user 0m0.584s +sys 0m0.212s + +real 0m0.780s +user 0m0.584s +sys 0m0.176s + +real 0m0.809s +user 0m0.616s sys 0m0.180s -real 0m1.992s -user 0m1.180s +real 0m1.356s +user 0m1.140s sys 0m0.188s -real 0m2.028s +real 0m0.783s +user 0m0.600s +sys 0m0.176s + +real 0m1.390s user 0m1.184s sys 0m0.152s -real 0m1.878s -user 0m1.028s -sys 0m0.136s +real 0m1.875s +user 0m1.632s +sys 0m0.244s + +real 0m1.313s +user 0m1.148s +sys 0m0.160s + +real 0m1.846s +user 0m1.644s +sys 0m0.188s + +real 0m1.875s +user 0m1.632s +sys 0m0.212s + +real 0m1.819s +user 0m1.672s +sys 0m0.124s + +real 0m2.117s +user 0m1.816s +sys 0m0.292s -real 0m1.980s -user 0m1.116s -sys 0m0.192s +real 0m1.871s +user 0m1.636s +sys 0m0.196s diff --git a/storage/maria/ma_unique.c b/storage/maria/ma_unique.c index b79c5558933..5d0133c8ac1 100644 --- a/storage/maria/ma_unique.c +++ b/storage/maria/ma_unique.c @@ -26,7 +26,7 @@ my_bool _ma_check_unique(MARIA_HA *info, MARIA_UNIQUEDEF *def, byte *record, MARIA_KEYDEF *key= &info->s->keyinfo[def->key]; byte *key_buff= info->lastkey2; DBUG_ENTER("_ma_check_unique"); - DBUG_PRINT("enter",("unique_hash: %lu", unique_hash)); + DBUG_PRINT("enter",("unique_hash: %lu", (ulong) unique_hash)); maria_unique_store(record+key->seg->start, unique_hash); _ma_make_key(info,def->key,key_buff,record,0); diff --git a/storage/maria/tablockman.c b/storage/maria/tablockman.c index d8dffa09a5e..810c6c12ea4 100644 --- a/storage/maria/tablockman.c +++ b/storage/maria/tablockman.c @@ -1,5 +1,5 @@ -#warning TODO - allocate everything from dynarrays !!! (benchmark) -#warning automatically place S instead of LS if possible +/* QQ: TODO - allocate everything from dynarrays !!! (benchmark) */ +/* QQ: automatically place S instead of LS if possible */ /* Copyright (C) 2006 MySQL AB This program is free software; you can redistribute it and/or modify @@ -219,7 +219,7 @@ static const enum lockman_getlock_result getlock_result[10][10]= */ struct st_table_lock { -#warning do we need upgraded_from ? + /* QQ: do we need upgraded_from ? */ struct st_table_lock *next_in_lo, *upgraded_from, *next, *prev; struct st_locked_table *table; uint16 loid; diff --git a/storage/maria/tablockman.h b/storage/maria/tablockman.h index 4498e7027b4..2c6fb6996a3 100644 --- a/storage/maria/tablockman.h +++ b/storage/maria/tablockman.h @@ -33,7 +33,7 @@ LSIX - Loose Shared + Intention eXclusive */ #ifndef _lockman_h -#warning TODO remove N-locks +/* QQ: TODO remove N-locks */ enum lock_type { N, S, X, IS, IX, SIX, LS, LX, SLX, LSIX, LOCK_TYPE_LAST }; enum lockman_getlock_result { NO_MEMORY_FOR_LOCK=1, DEADLOCK, LOCK_TIMEOUT, diff --git a/storage/maria/trnman.c b/storage/maria/trnman.c index f289f6fcc5b..37978e4ff76 100644 --- a/storage/maria/trnman.c +++ b/storage/maria/trnman.c @@ -351,8 +351,8 @@ void trnman_end_trn(TRN *trn, my_bool commit) those lists, and thus nobody may want to free them. Now we don't need a mutex to access free_me list */ + /* QQ: send them to the purge thread */ while (free_me) -#warning XXX send them to the purge thread { TRN *t= free_me; free_me= free_me->next; diff --git a/storage/maria/unittest/lockman2-t.c b/storage/maria/unittest/lockman2-t.c index 2a8090ab9ac..01af1a03d22 100644 --- a/storage/maria/unittest/lockman2-t.c +++ b/storage/maria/unittest/lockman2-t.c @@ -171,10 +171,12 @@ void run_test(const char *test, pthread_handler handler, int n, int m) static void reinit_tlo(TABLOCKMAN *lm, TABLE_LOCK_OWNER *lo) { +#ifdef NOT_USED_YET TABLE_LOCK_OWNER backup= *lo; +#endif tablockman_release_locks(lm, lo); - /* +#ifdef NOT_USED_YET pthread_mutex_destroy(lo->mutex); pthread_cond_destroy(lo->cond); bzero(lo, sizeof(*lo)); @@ -183,7 +185,8 @@ static void reinit_tlo(TABLOCKMAN *lm, TABLE_LOCK_OWNER *lo) lo->cond= backup.cond; lo->loid= backup.loid; pthread_mutex_init(lo->mutex, MY_MUTEX_INIT_FAST); - pthread_cond_init(lo->cond, 0);*/ + pthread_cond_init(lo->cond, 0); +#endif } pthread_mutex_t rt_mutex; @@ -191,8 +194,8 @@ int Nrows= 100; int Ntables= 10; int table_lock_ratio= 10; enum lock_type lock_array[6]= {S, X, LS, LX, IS, IX}; -char *lock2str[6]= {"S", "X", "LS", "LX", "IS", "IX"}; -char *res2str[]= { +const char *lock2str[6]= {"S", "X", "LS", "LX", "IS", "IX"}; +const char *res2str[]= { 0, "OUT OF MEMORY", "DEADLOCK", @@ -200,6 +203,7 @@ char *res2str[]= { "GOT THE LOCK", "GOT THE LOCK NEED TO LOCK A SUBRESOURCE", "GOT THE LOCK NEED TO INSTANT LOCK A SUBRESOURCE"}; + pthread_handler_t test_lockman(void *arg) { int m= (*(int *)arg); @@ -215,13 +219,16 @@ pthread_handler_t test_lockman(void *arg) for (x= ((int)(intptr)(&m)); m > 0; m--) { - x= (x*3628273133 + 1500450271) % 9576890767; /* three prime numbers */ + /* three prime numbers */ + x= (uint) ((x*LL(3628273133) + LL(1500450271)) % LL(9576890767)); row= x % Nrows + Ntables; table= row % Ntables; locklevel= (x/Nrows) & 3; if (table_lock_ratio && (x/Nrows/4) % table_lock_ratio == 0) - { /* table lock */ - res= tablockman_getlock(&tablockman, lo1, ltarray+table, lock_array[locklevel]); + { + /* table lock */ + res= tablockman_getlock(&tablockman, lo1, ltarray+table, + lock_array[locklevel]); DIAG(("loid %2d, table %d, lock %s, res %s", loid, table, lock2str[locklevel], res2str[res])); if (res < GOT_THE_LOCK) diff --git a/storage/maria/unittest/ma_control_file-t.c b/storage/maria/unittest/ma_control_file-t.c index beb86843dd3..49575378c78 100644 --- a/storage/maria/unittest/ma_control_file-t.c +++ b/storage/maria/unittest/ma_control_file-t.c @@ -42,7 +42,7 @@ char file_name[FN_REFLEN]; LSN expect_checkpoint_lsn; uint32 expect_logno; -static int delete_file(); +static int delete_file(myf my_flags); /* Those are test-specific wrappers around the module's API functions: after calling the module's API functions they perform checks on the result. @@ -91,7 +91,7 @@ int main(int argc,char *argv[]) get_options(argc,argv); diag("Deleting control file at startup, if there is an old one"); - RET_ERR_UNLESS(0 == delete_file()); /* if fails, can't continue */ + RET_ERR_UNLESS(0 == delete_file(0)); /* if fails, can't continue */ diag("Tests of normal conditions"); ok(0 == test_one_log(), "test of creating one log"); @@ -111,7 +111,7 @@ int main(int argc,char *argv[]) } -static int delete_file() +static int delete_file(myf my_flags) { RET_ERR_UNLESS(fn_format(file_name, CONTROL_FILE_BASE_NAME, maria_data_root, "", MYF(MY_WME)) != NullS); @@ -119,7 +119,7 @@ static int delete_file() Maybe file does not exist, ignore error. The error will however be printed on stderr. */ - my_delete(file_name, MYF(MY_WME)); + my_delete(file_name, my_flags); expect_checkpoint_lsn= CONTROL_FILE_IMPOSSIBLE_LSN; expect_logno= CONTROL_FILE_IMPOSSIBLE_FILENO; @@ -365,7 +365,7 @@ static int test_bad_size() int fd; /* A too short file */ - RET_ERR_UNLESS(delete_file() == 0); + RET_ERR_UNLESS(delete_file(MYF(MY_WME)) == 0); RET_ERR_UNLESS((fd= my_open(file_name, O_BINARY | O_RDWR | O_CREAT, MYF(MY_WME))) >= 0); @@ -378,7 +378,7 @@ static int test_bad_size() RET_ERR_UNLESS(my_close(fd, MYF(MY_WME)) == 0); /* Leave a correct control file */ - RET_ERR_UNLESS(delete_file() == 0); + RET_ERR_UNLESS(delete_file(MYF(MY_WME)) == 0); RET_ERR_UNLESS(create_or_open_file() == CONTROL_FILE_OK); RET_ERR_UNLESS(close_file() == 0); diff --git a/storage/maria/unittest/trnman-t.c b/storage/maria/unittest/trnman-t.c index 3c70d10c440..7d97794b685 100644 --- a/storage/maria/unittest/trnman-t.c +++ b/storage/maria/unittest/trnman-t.c @@ -35,7 +35,7 @@ int litmus; pthread_handler_t test_trnman(void *arg) { int m= (*(int *)arg); - uint x, y, i, j, n; + uint x, y, i, n; TRN *trn[MAX_ITER]; pthread_mutex_t mutexes[MAX_ITER]; pthread_cond_t conds[MAX_ITER]; @@ -48,7 +48,7 @@ pthread_handler_t test_trnman(void *arg) for (x= ((int)(intptr)(&m)); m > 0; ) { - y= x= (x*3628273133 + 1500450271) % 9576890767; /* three prime numbers */ + y= x= (x*LL(3628273133) + LL(1500450271)) % LL(9576890767); /* three prime numbers */ m-= n= x % MAX_ITER; for (i= 0; i < n; i++) { @@ -65,7 +65,6 @@ pthread_handler_t test_trnman(void *arg) trnman_end_trn(trn[i], y & 1); } } -end: for (i= 0; i < MAX_ITER; i++) { pthread_mutex_destroy(&mutexes[i]); diff --git a/storage/ndb/src/mgmapi/mgmapi.cpp b/storage/ndb/src/mgmapi/mgmapi.cpp index 614d0fea8e2..60201eac431 100644 --- a/storage/ndb/src/mgmapi/mgmapi.cpp +++ b/storage/ndb/src/mgmapi/mgmapi.cpp @@ -1274,13 +1274,13 @@ ndb_mgm_get_clusterlog_severity_filter(NdbMgmHandle handle, MGM_ARG(clusterlog_severity_names[5], Int, Mandatory, ""), MGM_ARG(clusterlog_severity_names[6], Int, Mandatory, ""), }; - CHECK_HANDLE(handle, NULL); - CHECK_CONNECTED(handle, NULL); + CHECK_HANDLE(handle, 0); + CHECK_CONNECTED(handle, 0); Properties args; const Properties *reply; reply = ndb_mgm_call(handle, getinfo_reply, "get info clusterlog", &args); - CHECK_REPLY(reply, NULL); + CHECK_REPLY(reply, 0); for(unsigned int i=0; i < severity_size; i++) { reply->get(clusterlog_severity_names[severity[i].category], &severity[i].value); @@ -1431,13 +1431,13 @@ ndb_mgm_get_clusterlog_loglevel(NdbMgmHandle handle, MGM_ARG(clusterlog_names[10], Int, Mandatory, ""), MGM_ARG(clusterlog_names[11], Int, Mandatory, ""), }; - CHECK_HANDLE(handle, NULL); - CHECK_CONNECTED(handle, NULL); + CHECK_HANDLE(handle, 0); + CHECK_CONNECTED(handle, 0); Properties args; const Properties *reply; reply = ndb_mgm_call(handle, getloglevel_reply, "get cluster loglevel", &args); - CHECK_REPLY(reply, NULL); + CHECK_REPLY(reply, 0); for(int i=0; i < loglevel_count; i++) { reply->get(clusterlog_names[loglevel[i].category], &loglevel[i].value); diff --git a/unittest/mysys/mf_pagecache_consist.c b/unittest/mysys/mf_pagecache_consist.c index d31c19cc746..fd824e27051 100755 --- a/unittest/mysys/mf_pagecache_consist.c +++ b/unittest/mysys/mf_pagecache_consist.c @@ -101,7 +101,7 @@ uint check_page(uchar *buff, ulong offset, int page_locked, int page_no, (page_locked ? "locked" : "unlocked"), end, num, tag); h= my_open("wrong_page", O_CREAT | O_TRUNC | O_RDWR, MYF(0)); - my_pwrite(h, buff, PAGE_SIZE, 0, MYF(0)); + my_pwrite(h, (byte*) buff, PAGE_SIZE, 0, MYF(0)); my_close(h, MYF(0)); goto err; } diff --git a/unittest/mysys/my_atomic-t.c b/unittest/mysys/my_atomic-t.c index 02e07acd67b..c16a2f118e5 100644 --- a/unittest/mysys/my_atomic-t.c +++ b/unittest/mysys/my_atomic-t.c @@ -252,11 +252,11 @@ void test_atomic(const char *test, pthread_handler handler, int n, int m) for (i= 0 ; i < n ; i++) pthread_join(threads[i], 0); now= my_getsystime()-now; -err: ok(a32 == 0, "tested %s in %g secs (%d)", test, ((double)now)/1e7, a32); my_free((void *)threads, MYF(0)); } + int main() { int err; -- cgit v1.2.1 From 025400922118f11a15be54c66455f20e2f72c0b4 Mon Sep 17 00:00:00 2001 From: unknown Date: Fri, 2 Feb 2007 09:41:32 +0200 Subject: postreview changes for page cache and pre review commit for loghandler storage/maria/unittest/test_file.c: Rename: unittest/mysys/test_file.c -> storage/maria/unittest/test_file.c storage/maria/unittest/test_file.h: Rename: unittest/mysys/test_file.h -> storage/maria/unittest/test_file.h include/pagecache.h: A waiting queue mechanism moved to separate file wqueue.* Pointer name changed for compatibility mysys/Makefile.am: A waiting queue mechanism moved to separate file wqueue.* mysys/mf_keycache.c: fixed unsigned comparison mysys/mf_pagecache.c: A waiting queue mechanism moved to separate file wqueue.* Fixed bug in unregistering block during write storage/maria/Makefile.am: The loghandler files added storage/maria/ma_control_file.h: Now we have loghandler and can compile control file storage/maria/maria_def.h: Including files need for compilation of maria storage/maria/unittest/Makefile.am: unit tests of loghandler storage/maria/unittest/ma_control_file-t.c: Used maria def storage/maria/unittest/mf_pagecache_consist.c: fixed memory overrun storage/maria/unittest/mf_pagecache_single.c: fixed used uninitialized memory unittest/mysys/Makefile.am: unittests of pagecache moved to maria becase pagecache need loghandler include/wqueue.h: New BitKeeper file ``include/wqueue.h'' mysys/wqueue.c: New BitKeeper file ``mysys/wqueue.c'' storage/maria/ma_loghandler.c: New BitKeeper file ``storage/maria/ma_loghandler.c'' storage/maria/ma_loghandler.h: New BitKeeper file ``storage/maria/ma_loghandler.h'' storage/maria/ma_loghandler_lsn.h: New BitKeeper file ``storage/maria/ma_loghandler_lsn.h'' storage/maria/unittest/ma_test_loghandler-t.c: New BitKeeper file ``storage/maria/unittest/ma_test_loghandler-t.c'' storage/maria/unittest/ma_test_loghandler_multigroup-t.c: New BitKeeper file ``storage/maria/unittest/ma_test_loghandler_multigroup-t.c'' storage/maria/unittest/ma_test_loghandler_multithread-t.c: New BitKeeper file ``storage/maria/unittest/ma_test_loghandler_multithread-t.c'' storage/maria/unittest/ma_test_loghandler_pagecache-t.c: New BitKeeper file ``storage/maria/unittest/ma_test_loghandler_pagecache-t.c'' --- include/pagecache.h | 65 +- include/wqueue.h | 26 + mysys/Makefile.am | 2 +- mysys/mf_keycache.c | 2 +- mysys/mf_pagecache.c | 776 ++- mysys/wqueue.c | 167 + storage/maria/Makefile.am | 2 +- storage/maria/ma_control_file.h | 22 +- storage/maria/ma_loghandler.c | 5417 ++++++++++++++++++++ storage/maria/ma_loghandler.h | 314 ++ storage/maria/ma_loghandler_lsn.h | 39 + storage/maria/maria_def.h | 5 + storage/maria/unittest/Makefile.am | 60 +- storage/maria/unittest/ma_control_file-t.c | 2 +- storage/maria/unittest/ma_test_loghandler-t.c | 540 ++ .../unittest/ma_test_loghandler_multigroup-t.c | 570 ++ .../unittest/ma_test_loghandler_multithread-t.c | 468 ++ .../unittest/ma_test_loghandler_pagecache-t.c | 140 + storage/maria/unittest/mf_pagecache_consist.c | 468 ++ storage/maria/unittest/mf_pagecache_single.c | 589 +++ storage/maria/unittest/test_file.c | 68 + storage/maria/unittest/test_file.h | 14 + unittest/mysys/Makefile.am | 43 +- unittest/mysys/mf_pagecache_consist.c | 445 -- unittest/mysys/mf_pagecache_single.c | 587 --- unittest/mysys/test_file.c | 68 - unittest/mysys/test_file.h | 14 - 27 files changed, 9267 insertions(+), 1646 deletions(-) create mode 100644 include/wqueue.h create mode 100644 mysys/wqueue.c create mode 100644 storage/maria/ma_loghandler.c create mode 100644 storage/maria/ma_loghandler.h create mode 100644 storage/maria/ma_loghandler_lsn.h create mode 100644 storage/maria/unittest/ma_test_loghandler-t.c create mode 100644 storage/maria/unittest/ma_test_loghandler_multigroup-t.c create mode 100644 storage/maria/unittest/ma_test_loghandler_multithread-t.c create mode 100644 storage/maria/unittest/ma_test_loghandler_pagecache-t.c create mode 100755 storage/maria/unittest/mf_pagecache_consist.c create mode 100644 storage/maria/unittest/mf_pagecache_single.c create mode 100644 storage/maria/unittest/test_file.c create mode 100644 storage/maria/unittest/test_file.h delete mode 100755 unittest/mysys/mf_pagecache_consist.c delete mode 100644 unittest/mysys/mf_pagecache_single.c delete mode 100644 unittest/mysys/test_file.c delete mode 100644 unittest/mysys/test_file.h diff --git a/include/pagecache.h b/include/pagecache.h index 4d64070ad62..2f745eae0b3 100644 --- a/include/pagecache.h +++ b/include/pagecache.h @@ -20,11 +20,13 @@ #define _pagecache_h C_MODE_START +#include "../storage/maria/ma_loghandler_lsn.h" + /* Type of the page */ enum pagecache_page_type { #ifndef DBUG_OFF - /* used only for control page type chenging during debugging */ + /* used only for control page type changing during debugging */ PAGECACHE_EMPTY_PAGE, #endif /* the page does not contain LSN */ @@ -34,7 +36,7 @@ enum pagecache_page_type }; /* - This enum describe lock status changing. every typr of page cache will + This enum describe lock status changing. every type of page cache will interpret WRITE/READ lock as it need. */ enum pagecache_page_lock @@ -71,9 +73,7 @@ enum pagecache_write_mode typedef void *PAGECACHE_PAGE_LINK; -/* TODO: move to loghandler emulator */ -typedef void LOG_HANDLER; -typedef void *LSN; +typedef void *LSN_PTR; /* file descriptor for Maria */ typedef struct st_pagecache_file @@ -82,7 +82,7 @@ typedef struct st_pagecache_file } PAGECACHE_FILE; /* page number for maria */ -typedef uint32 maria_page_no_t; +typedef uint32 pgcache_page_no_t; /* declare structures that is used by st_pagecache */ @@ -93,11 +93,9 @@ typedef struct st_pagecache_page PAGECACHE_PAGE; struct st_pagecache_hash_link; typedef struct st_pagecache_hash_link PAGECACHE_HASH_LINK; -/* info about requests in a waiting queue */ -typedef struct st_pagecache_wqueue -{ - struct st_my_thread_var *last_thread; /* circular list of waiting threads */ -} PAGECACHE_WQUEUE; +#include + +typedef my_bool (*pagecache_disk_read_validator)(byte *page, gptr data); #define PAGECACHE_CHANGED_BLOCKS_HASH 128 /* must be power of 2 */ @@ -136,16 +134,14 @@ typedef struct st_pagecache PAGECACHE_BLOCK_LINK *used_last;/* ptr to the last block of the LRU chain */ PAGECACHE_BLOCK_LINK *used_ins;/* ptr to the insertion block in LRU chain */ pthread_mutex_t cache_lock; /* to lock access to the cache structure */ - PAGECACHE_WQUEUE resize_queue; /* threads waiting during resize operation */ - PAGECACHE_WQUEUE waiting_for_hash_link;/* waiting for a free hash link */ - PAGECACHE_WQUEUE waiting_for_block; /* requests waiting for a free block */ + WQUEUE resize_queue; /* threads waiting during resize operation */ + WQUEUE waiting_for_hash_link;/* waiting for a free hash link */ + WQUEUE waiting_for_block; /* requests waiting for a free block */ /* hash for dirty file bl.*/ PAGECACHE_BLOCK_LINK *changed_blocks[PAGECACHE_CHANGED_BLOCKS_HASH]; /* hash for other file bl.*/ PAGECACHE_BLOCK_LINK *file_blocks[PAGECACHE_CHANGED_BLOCKS_HASH]; - LOG_HANDLER *loghandler; /* loghandler structure */ - /* The following variables are and variables used to hold parameters for initializing the key cache. @@ -169,24 +165,29 @@ typedef struct st_pagecache extern int init_pagecache(PAGECACHE *pagecache, my_size_t use_mem, uint division_limit, uint age_threshold, - uint block_size, - LOG_HANDLER *loghandler); + uint block_size); extern int resize_pagecache(PAGECACHE *pagecache, my_size_t use_mem, uint division_limit, uint age_threshold); extern void change_pagecache_param(PAGECACHE *pagecache, uint division_limit, uint age_threshold); -extern byte *pagecache_read(PAGECACHE *pagecache, - PAGECACHE_FILE *file, - maria_page_no_t pageno, - uint level, - byte *buff, - enum pagecache_page_type type, - enum pagecache_page_lock lock, - PAGECACHE_PAGE_LINK *link); + +#define pagecache_read(P,F,N,L,B,T,K,I) \ + pagecache_valid_read(P,F,N,L,B,T,K,I,0,0) + +extern byte *pagecache_valid_read(PAGECACHE *pagecache, + PAGECACHE_FILE *file, + pgcache_page_no_t pageno, + uint level, + byte *buff, + enum pagecache_page_type type, + enum pagecache_page_lock lock, + PAGECACHE_PAGE_LINK *link, + pagecache_disk_read_validator validator, + gptr validator_data); extern my_bool pagecache_write(PAGECACHE *pagecache, PAGECACHE_FILE *file, - maria_page_no_t pageno, + pgcache_page_no_t pageno, uint level, byte *buff, enum pagecache_page_type type, @@ -196,20 +197,20 @@ extern my_bool pagecache_write(PAGECACHE *pagecache, PAGECACHE_PAGE_LINK *link); extern void pagecache_unlock_page(PAGECACHE *pagecache, PAGECACHE_FILE *file, - maria_page_no_t pageno, + pgcache_page_no_t pageno, enum pagecache_page_lock lock, enum pagecache_page_pin pin, my_bool stamp_this_page, - LSN first_REDO_LSN_for_page); + LSN_PTR first_REDO_LSN_for_page); extern void pagecache_unlock(PAGECACHE *pagecache, PAGECACHE_PAGE_LINK *link, enum pagecache_page_lock lock, enum pagecache_page_pin pin, my_bool stamp_this_page, - LSN first_REDO_LSN_for_page); + LSN_PTR first_REDO_LSN_for_page); extern void pagecache_unpin_page(PAGECACHE *pagecache, PAGECACHE_FILE *file, - maria_page_no_t pageno); + pgcache_page_no_t pageno); extern void pagecache_unpin(PAGECACHE *pagecache, PAGECACHE_PAGE_LINK *link); extern int flush_pagecache_blocks(PAGECACHE *keycache, @@ -217,7 +218,7 @@ extern int flush_pagecache_blocks(PAGECACHE *keycache, enum flush_type type); extern my_bool pagecache_delete_page(PAGECACHE *pagecache, PAGECACHE_FILE *file, - maria_page_no_t pageno, + pgcache_page_no_t pageno, enum pagecache_page_lock lock, my_bool flush); extern void end_pagecache(PAGECACHE *keycache, my_bool cleanup); diff --git a/include/wqueue.h b/include/wqueue.h new file mode 100644 index 00000000000..bacabb8c401 --- /dev/null +++ b/include/wqueue.h @@ -0,0 +1,26 @@ + +#ifndef _wqueue_h +#define _wqueue_h + +#include +#include + +/* info about requests in a waiting queue */ +typedef struct st_pagecache_wqueue +{ + struct st_my_thread_var *last_thread; /* circular list of waiting + threads */ +} WQUEUE; + +#ifdef THREAD +void wqueue_link_into_queue(WQUEUE *wqueue, struct st_my_thread_var *thread); +void wqueue_unlink_from_queue(WQUEUE *wqueue, struct st_my_thread_var *thread); +void wqueue_add_to_queue(WQUEUE *wqueue, struct st_my_thread_var *thread); +void wqueue_add_and_wait(WQUEUE *wqueue, + struct st_my_thread_var *thread, + pthread_mutex_t *lock); +void wqueue_release_queue(WQUEUE *wqueue); + +#endif + +#endif diff --git a/mysys/Makefile.am b/mysys/Makefile.am index 4d9570febbd..612411404c4 100644 --- a/mysys/Makefile.am +++ b/mysys/Makefile.am @@ -56,7 +56,7 @@ libmysys_a_SOURCES = my_init.c my_getwd.c mf_getdate.c my_mmap.c \ my_handler.c my_netware.c my_largepage.c \ my_memmem.c \ my_windac.c my_access.c base64.c my_libwrap.c \ - mf_pagecache.c + mf_pagecache.c wqueue.c EXTRA_DIST = thr_alarm.c thr_lock.c my_pthread.c my_thr_init.c \ thr_mutex.c thr_rwlock.c \ CMakeLists.txt mf_soundex.c \ diff --git a/mysys/mf_keycache.c b/mysys/mf_keycache.c index 9a99a278bc5..9cb428ab200 100644 --- a/mysys/mf_keycache.c +++ b/mysys/mf_keycache.c @@ -1008,12 +1008,12 @@ static void unlink_block(KEY_CACHE *keycache, BLOCK_LINK *block) KEYCACHE_THREAD_TRACE("unlink_block"); #if defined(KEYCACHE_DEBUG) + KEYCACHE_DBUG_ASSERT(keycache->blocks_available != 0); keycache->blocks_available--; KEYCACHE_DBUG_PRINT("unlink_block", ("unlinked block %u status=%x #requests=%u #available=%u", BLOCK_NUMBER(block), block->status, block->requests, keycache->blocks_available)); - KEYCACHE_DBUG_ASSERT(keycache->blocks_available >= 0); #endif } diff --git a/mysys/mf_pagecache.c b/mysys/mf_pagecache.c index 4b92f68d9bf..97cb542f329 100755 --- a/mysys/mf_pagecache.c +++ b/mysys/mf_pagecache.c @@ -26,7 +26,7 @@ When a new block is required it is first tried to pop one from the stack. If the stack is empty, it is tried to get a never-used block from the pool. If this is empty too, then a block is taken from the LRU ring, flushing it - to disk, if necessary. This is handled in find_key_block(). + to disk, if necessary. This is handled in find_block(). With the new free list, the blocks can have three temperatures: hot, warm and cold (which is free). This is remembered in the block header by the enum BLOCK_TEMPERATURE temperature variable. Remembering the @@ -91,13 +91,16 @@ /* In key cache we have external raw locking here we use SERIALIZED_READ_FROM_CACHE to avoid problem of reading - not consistent data from te page + not consistent data from the page. + (keycache functions (key_cache_read(), key_cache_insert() and + key_cache_write()) rely on external MyISAM lock, we don't) */ #define SERIALIZED_READ_FROM_CACHE yes #define BLOCK_INFO(B) \ DBUG_PRINT("info", \ - ("block 0x%lx, file %lu, page %lu, s %0x, hshL 0x%lx, req %u/%u", \ + ("block 0x%lx file %lu page %lu s %0x hshL 0x%lx req %u/%u " \ + "wrlock: %c", \ (ulong)(B), \ (ulong)((B)->hash_link ? \ (B)->hash_link->file.file : \ @@ -110,7 +113,8 @@ (uint) (B)->requests, \ (uint)((B)->hash_link ? \ (B)->hash_link->requests : \ - 0))) + 0), \ + ((block->status & BLOCK_WRLOCK)?'Y':'N'))) /* TODO: put it to my_static.c */ my_bool my_disable_flush_pagecache_blocks= 0; @@ -138,7 +142,7 @@ typedef pthread_cond_t KEYCACHE_CONDVAR; struct st_pagecache_page { PAGECACHE_FILE file; /* file to which the page belongs to */ - maria_page_no_t pageno; /* number of the page in the file */ + pgcache_page_no_t pageno; /* number of the page in the file */ }; /* element in the chain of a hash table bucket */ @@ -149,7 +153,7 @@ struct st_pagecache_hash_link struct st_pagecache_block_link *block; /* reference to the block for the page: */ PAGECACHE_FILE file; /* from such a file */ - maria_page_no_t pageno; /* this page */ + pgcache_page_no_t pageno; /* this page */ uint requests; /* number of requests for the page */ }; @@ -162,7 +166,7 @@ struct st_pagecache_hash_link #define BLOCK_CHANGED 32 /* block buffer contains a dirty page */ #define BLOCK_WRLOCK 64 /* write locked block */ -/* page status, returned by find_key_block */ +/* page status, returned by find_block */ #define PAGE_READ 0 #define PAGE_TO_BE_READ 1 #define PAGE_WAIT_TO_BE_READ 2 @@ -232,7 +236,7 @@ typedef struct st_pagecache_lock_info node the node which should be linked */ -void info_link(PAGECACHE_PIN_INFO **list, PAGECACHE_PIN_INFO *node) +static void info_link(PAGECACHE_PIN_INFO **list, PAGECACHE_PIN_INFO *node) { if ((node->next= *list)) node->next->prev= &(node->next); @@ -249,7 +253,7 @@ void info_link(PAGECACHE_PIN_INFO **list, PAGECACHE_PIN_INFO *node) node the node which should be unlinked */ -void info_unlink(PAGECACHE_PIN_INFO *node) +static void info_unlink(PAGECACHE_PIN_INFO *node) { if ((*node->prev= node->next)) node->next->prev= node->prev; @@ -271,8 +275,8 @@ void info_unlink(PAGECACHE_PIN_INFO *node) pointer to the information node of the thread in the list */ -PAGECACHE_PIN_INFO *info_find(PAGECACHE_PIN_INFO *list, - struct st_my_thread_var *thread) +static PAGECACHE_PIN_INFO *info_find(PAGECACHE_PIN_INFO *list, + struct st_my_thread_var *thread) { register PAGECACHE_PIN_INFO *i= list; for(; i != 0; i= i->next) @@ -291,7 +295,7 @@ struct st_pagecache_block_link *next_changed, **prev_changed; /* for lists of file dirty/clean blocks */ struct st_pagecache_hash_link *hash_link; /* backward ptr to referring hash_link */ - PAGECACHE_WQUEUE + WQUEUE wqueue[COND_SIZE]; /* queues on waiting requests for new/old pages */ uint requests; /* number of requests for the block */ byte *buffer; /* buffer for the block page */ @@ -310,8 +314,8 @@ struct st_pagecache_block_link #ifdef PAGECACHE_DEBUG /* debug checks */ -my_bool info_check_pin(PAGECACHE_BLOCK_LINK *block, - enum pagecache_page_pin mode) +static my_bool info_check_pin(PAGECACHE_BLOCK_LINK *block, + enum pagecache_page_pin mode) { struct st_my_thread_var *thread= my_thread_var; DBUG_ENTER("info_check_pin"); @@ -367,9 +371,9 @@ my_bool info_check_pin(PAGECACHE_BLOCK_LINK *block, 1 - Error */ -my_bool info_check_lock(PAGECACHE_BLOCK_LINK *block, - enum pagecache_page_lock lock, - enum pagecache_page_pin pin) +static my_bool info_check_lock(PAGECACHE_BLOCK_LINK *block, + enum pagecache_page_lock lock, + enum pagecache_page_pin pin) { struct st_my_thread_var *thread= my_thread_var; DBUG_ENTER("info_check_lock"); @@ -379,47 +383,47 @@ my_bool info_check_lock(PAGECACHE_BLOCK_LINK *block, switch(lock) { case PAGECACHE_LOCK_LEFT_UNLOCKED: - DBUG_ASSERT(pin == PAGECACHE_PIN_LEFT_UNPINNED); - if (info) + if (pin != PAGECACHE_PIN_LEFT_UNPINNED || + info) goto error; break; case PAGECACHE_LOCK_LEFT_READLOCKED: - DBUG_ASSERT(pin == PAGECACHE_PIN_LEFT_UNPINNED || - pin == PAGECACHE_PIN_LEFT_PINNED); - if (info == 0 || info->write_lock) + if ((pin != PAGECACHE_PIN_LEFT_UNPINNED && + pin != PAGECACHE_PIN_LEFT_PINNED) || + info == 0 || info->write_lock) goto error; break; case PAGECACHE_LOCK_LEFT_WRITELOCKED: - DBUG_ASSERT(pin == PAGECACHE_PIN_LEFT_PINNED); - if (info == 0 || !info->write_lock) + if (pin != PAGECACHE_PIN_LEFT_PINNED || + info == 0 || !info->write_lock) goto error; break; case PAGECACHE_LOCK_READ: - DBUG_ASSERT(pin == PAGECACHE_PIN_LEFT_UNPINNED || - pin == PAGECACHE_PIN); - if (info != 0) + if ((pin != PAGECACHE_PIN_LEFT_UNPINNED && + pin != PAGECACHE_PIN) || + info != 0) goto error; break; case PAGECACHE_LOCK_WRITE: - DBUG_ASSERT(pin == PAGECACHE_PIN); - if (info != 0) + if (pin != PAGECACHE_PIN || + info != 0) goto error; break; case PAGECACHE_LOCK_READ_UNLOCK: - DBUG_ASSERT(pin == PAGECACHE_PIN_LEFT_UNPINNED || - pin == PAGECACHE_UNPIN); - if (info == 0 || info->write_lock) + if ((pin != PAGECACHE_PIN_LEFT_UNPINNED && + pin != PAGECACHE_UNPIN) || + info == 0 || info->write_lock) goto error; break; case PAGECACHE_LOCK_WRITE_UNLOCK: - DBUG_ASSERT(pin == PAGECACHE_UNPIN); - if (info == 0 || !info->write_lock) + if (pin != PAGECACHE_UNPIN || + info == 0 || !info->write_lock) goto error; break; case PAGECACHE_LOCK_WRITE_TO_READ: - DBUG_ASSERT(pin == PAGECACHE_PIN_LEFT_PINNED || - pin == PAGECACHE_UNPIN); - if (info == 0 || !info->write_lock) + if ((pin != PAGECACHE_PIN_LEFT_PINNED && + pin != PAGECACHE_UNPIN) || + info == 0 || !info->write_lock) goto error; break; } @@ -439,12 +443,6 @@ error: #define FLUSH_CACHE 2000 /* sort this many blocks at once */ static int flush_all_key_blocks(PAGECACHE *pagecache); -#ifdef THREAD -static void link_into_queue(PAGECACHE_WQUEUE *wqueue, - struct st_my_thread_var *thread); -static void unlink_from_queue(PAGECACHE_WQUEUE *wqueue, - struct st_my_thread_var *thread); -#endif static void free_block(PAGECACHE *pagecache, PAGECACHE_BLOCK_LINK *block); static void test_key_cache(PAGECACHE *pagecache, const char *where, my_bool lock); @@ -551,6 +549,7 @@ static int ___pagecache_pthread_cond_signal(pthread_cond_t *cond); #define pagecache_pthread_cond_signal pthread_cond_signal #endif /* defined(PAGECACHE_DEBUG) */ +extern my_bool translog_flush(LSN *lsn); /* Write page to the disk @@ -567,18 +566,28 @@ static int ___pagecache_pthread_cond_signal(pthread_cond_t *cond); 0 - OK !=0 - Error */ -uint pagecache_fwrite(PAGECACHE *pagecache, - PAGECACHE_FILE *filedesc, - byte *buffer, - maria_page_no_t pageno, - enum pagecache_page_type type, - myf flags) + +static uint pagecache_fwrite(PAGECACHE *pagecache, + PAGECACHE_FILE *filedesc, + byte *buffer, + pgcache_page_no_t pageno, + enum pagecache_page_type type, + myf flags) { DBUG_ENTER("pagecache_fwrite"); if (type == PAGECACHE_LSN_PAGE) { + LSN lsn; DBUG_PRINT("info", ("Log handler call")); - /* TODO: put here loghandler call */ + /* TODO: integrate with page format */ +#define PAGE_LSN_OFFSET 0 + lsn7korr(&lsn, buffer + PAGE_LSN_OFFSET); + /* + check CONTROL_FILE_IMPOSSIBLE_FILENO & + CONTROL_FILE_IMPOSSIBLE_LOG_OFFSET + */ + DBUG_ASSERT(lsn.file_no != 0 && lsn.rec_offset != 0); + translog_flush(&lsn); } DBUG_RETURN(my_pwrite(filedesc->file, buffer, pagecache->block_size, (pageno)<<(pagecache->shift), flags)); @@ -628,8 +637,6 @@ static uint next_power(uint value) division_limit division limit (may be zero) age_threshold age threshold (may be zero) block_size size of block (should be power of 2) - loghandler logfandler pointer to call it in case of - pages with LSN RETURN VALUE number of blocks in the key cache, if successful, @@ -647,12 +654,11 @@ static uint next_power(uint value) int init_pagecache(PAGECACHE *pagecache, my_size_t use_mem, uint division_limit, uint age_threshold, - uint block_size, - LOG_HANDLER *loghandler) + uint block_size) { - int blocks, hash_links, length; + uint blocks, hash_links, length; int error; - DBUG_ENTER("init_key_cache"); + DBUG_ENTER("init_pagecache"); DBUG_ASSERT(block_size >= 512); PAGECACHE_DEBUG_OPEN; @@ -662,8 +668,6 @@ int init_pagecache(PAGECACHE *pagecache, my_size_t use_mem, DBUG_RETURN(0); } - pagecache->loghandler= loghandler; - pagecache->global_cache_w_requests= pagecache->global_cache_r_requests= 0; pagecache->global_cache_read= pagecache->global_cache_write= 0; pagecache->disk_blocks= -1; @@ -692,8 +696,8 @@ int init_pagecache(PAGECACHE *pagecache, my_size_t use_mem, for ( ; ; ) { /* Set my_hash_entries to the next bigger 2 power */ - if ((pagecache->hash_entries= next_power((uint)blocks)) < - ((uint)blocks) * 5/4) + if ((pagecache->hash_entries= next_power(blocks)) < + (blocks) * 5/4) pagecache->hash_entries<<= 1; hash_links= 2 * blocks; #if defined(MAX_THREADS) @@ -704,7 +708,7 @@ int init_pagecache(PAGECACHE *pagecache, my_size_t use_mem, ALIGN_SIZE(hash_links * sizeof(PAGECACHE_HASH_LINK)) + ALIGN_SIZE(sizeof(PAGECACHE_HASH_LINK*) * pagecache->hash_entries))) + - ((ulong) blocks << pagecache->shift) > use_mem) + (((ulong) blocks) << pagecache->shift) > use_mem) blocks--; /* Allocate memory for cache page buffers */ if ((pagecache->block_mem= @@ -760,10 +764,10 @@ int init_pagecache(PAGECACHE *pagecache, my_size_t use_mem, pagecache->warm_blocks= 0; pagecache->min_warm_blocks= (division_limit ? blocks * division_limit / 100 + 1 : - (ulong)blocks); + blocks); pagecache->age_threshold= (age_threshold ? blocks * age_threshold / 100 : - (ulong)blocks); + blocks); pagecache->cnt_for_resize_op= 0; pagecache->resize_in_flush= 0; @@ -842,7 +846,8 @@ int resize_pagecache(PAGECACHE *pagecache, { int blocks; struct st_my_thread_var *thread; - PAGECACHE_WQUEUE *wqueue; + WQUEUE *wqueue; + DBUG_ENTER("resize_pagecache"); if (!pagecache->inited) @@ -859,7 +864,7 @@ int resize_pagecache(PAGECACHE *pagecache, #ifdef THREAD wqueue= &pagecache->resize_queue; thread= my_thread_var; - link_into_queue(wqueue, thread); + wqueue_link_into_queue(wqueue, thread); while (wqueue->last_thread->next != thread) { @@ -892,12 +897,11 @@ int resize_pagecache(PAGECACHE *pagecache, end_pagecache(pagecache, 0); /* Don't free mutex */ /* The following will work even if use_mem is 0 */ blocks= init_pagecache(pagecache, pagecache->block_size, use_mem, - division_limit, age_threshold, - pagecache->loghandler); + division_limit, age_threshold); finish: #ifdef THREAD - unlink_from_queue(wqueue, thread); + wqueue_unlink_from_queue(wqueue, thread); /* Signal for the next resize request to proceeed if any */ if (wqueue->last_thread) { @@ -1027,146 +1031,6 @@ void end_pagecache(PAGECACHE *pagecache, my_bool cleanup) } /* end_pagecache */ -#ifdef THREAD -/* - Link a thread into double-linked queue of waiting threads. - - SYNOPSIS - link_into_queue() - wqueue pointer to the queue structure - thread pointer to the thread to be added to the queue - - RETURN VALUE - none - - NOTES. - Queue is represented by a circular list of the thread structures - The list is double-linked of the type (**prev,*next), accessed by - a pointer to the last element. -*/ - -static void link_into_queue(PAGECACHE_WQUEUE *wqueue, - struct st_my_thread_var *thread) -{ - struct st_my_thread_var *last; - if (! (last= wqueue->last_thread)) - { - /* Queue is empty */ - thread->next= thread; - thread->prev= &thread->next; - } - else - { - thread->prev= last->next->prev; - last->next->prev= &thread->next; - thread->next= last->next; - last->next= thread; - } - wqueue->last_thread= thread; -} - -/* - Unlink a thread from double-linked queue of waiting threads - - SYNOPSIS - unlink_from_queue() - wqueue pointer to the queue structure - thread pointer to the thread to be removed from the queue - - RETURN VALUE - none - - NOTES. - See NOTES for link_into_queue -*/ - -static void unlink_from_queue(PAGECACHE_WQUEUE *wqueue, - struct st_my_thread_var *thread) -{ - KEYCACHE_DBUG_PRINT("unlink_from_queue", ("thread %ld", thread->id)); - if (thread->next == thread) - /* The queue contains only one member */ - wqueue->last_thread= NULL; - else - { - thread->next->prev= thread->prev; - *thread->prev=thread->next; - if (wqueue->last_thread == thread) - wqueue->last_thread= STRUCT_PTR(struct st_my_thread_var, next, - thread->prev); - } - thread->next= NULL; -} - - -/* - Add a thread to single-linked queue of waiting threads - - SYNOPSIS - add_to_queue() - wqueue pointer to the queue structure - thread pointer to the thread to be added to the queue - - RETURN VALUE - none - - NOTES. - Queue is represented by a circular list of the thread structures - The list is single-linked of the type (*next), accessed by a pointer - to the last element. -*/ - -static inline void add_to_queue(PAGECACHE_WQUEUE *wqueue, - struct st_my_thread_var *thread) -{ - struct st_my_thread_var *last; - if (! (last= wqueue->last_thread)) - thread->next= thread; - else - { - thread->next= last->next; - last->next= thread; - } - wqueue->last_thread= thread; -} - - -/* - Remove all threads from queue signaling them to proceed - - SYNOPSIS - realease_queue() - wqueue pointer to the queue structure - thread pointer to the thread to be added to the queue - - RETURN VALUE - none - - NOTES. - See notes for add_to_queue - When removed from the queue each thread is signaled via condition - variable thread->suspend. -*/ - -static void release_queue(PAGECACHE_WQUEUE *wqueue) -{ - struct st_my_thread_var *last= wqueue->last_thread; - struct st_my_thread_var *next= last->next; - struct st_my_thread_var *thread; - do - { - thread=next; - KEYCACHE_DBUG_PRINT("release_queue: signal", ("thread %ld", thread->id)); - pagecache_pthread_cond_signal(&thread->suspend); - next=thread->next; - thread->next= NULL; - } - while (thread != last); - wqueue->last_thread= NULL; -} -#endif - - /* Unlink a block from the chain of dirty/clean blocks */ @@ -1273,6 +1137,7 @@ static void link_block(PAGECACHE *pagecache, PAGECACHE_BLOCK_LINK *block, PAGECACHE_BLOCK_LINK *ins; PAGECACHE_BLOCK_LINK **ptr_ins; + BLOCK_INFO(block); KEYCACHE_DBUG_ASSERT(! (block->hash_link && block->hash_link->requests)); #ifdef THREAD if (!hot && pagecache->waiting_for_block.last_thread) @@ -1297,7 +1162,7 @@ static void link_block(PAGECACHE *pagecache, PAGECACHE_BLOCK_LINK *block, { KEYCACHE_DBUG_PRINT("link_block: signal", ("thread %ld", thread->id)); pagecache_pthread_cond_signal(&thread->suspend); - unlink_from_queue(&pagecache->waiting_for_block, thread); + wqueue_unlink_from_queue(&pagecache->waiting_for_block, thread); block->requests++; } } @@ -1363,6 +1228,8 @@ static void link_block(PAGECACHE *pagecache, PAGECACHE_BLOCK_LINK *block, static void unlink_block(PAGECACHE *pagecache, PAGECACHE_BLOCK_LINK *block) { + DBUG_ENTER("unlink_block"); + DBUG_PRINT("unlink_block", ("unlink 0x%lx", (ulong)block)); if (block->next_used == block) /* The list contains only one member */ pagecache->used_last= pagecache->used_ins= NULL; @@ -1381,14 +1248,15 @@ static void unlink_block(PAGECACHE *pagecache, PAGECACHE_BLOCK_LINK *block) KEYCACHE_THREAD_TRACE("unlink_block"); #if defined(PAGECACHE_DEBUG) + KEYCACHE_DBUG_ASSERT(pagecache->blocks_available != 0); pagecache->blocks_available--; KEYCACHE_DBUG_PRINT("unlink_block", ("unlinked block 0x%lx (%u) status=%x #requests=%u #available=%u", (ulong)block, BLOCK_NUMBER(pagecache, block), block->status, block->requests, pagecache->blocks_available)); BLOCK_INFO(block); - KEYCACHE_DBUG_ASSERT(pagecache->blocks_available >= 0); #endif + DBUG_VOID_RETURN; } @@ -1591,7 +1459,7 @@ static void unlink_hash(PAGECACHE *pagecache, PAGECACHE_HASH_LINK *hash_link) { KEYCACHE_DBUG_PRINT("unlink_hash: signal", ("thread %ld", thread->id)); pagecache_pthread_cond_signal(&thread->suspend); - unlink_from_queue(&pagecache->waiting_for_hash_link, thread); + wqueue_unlink_from_queue(&pagecache->waiting_for_hash_link, thread); } } while (thread != last_thread); @@ -1618,7 +1486,7 @@ static void unlink_hash(PAGECACHE *pagecache, PAGECACHE_HASH_LINK *hash_link) pagecache Pagecache reference file file ID pageno page number in the file - start where to put pointer to found hash link (for + start where to put pointer to found hash bucket (for direct referring it) RETURN @@ -1627,7 +1495,7 @@ static void unlink_hash(PAGECACHE *pagecache, PAGECACHE_HASH_LINK *hash_link) static PAGECACHE_HASH_LINK *get_present_hash_link(PAGECACHE *pagecache, PAGECACHE_FILE *file, - maria_page_no_t pageno, + pgcache_page_no_t pageno, PAGECACHE_HASH_LINK ***start) { reg1 PAGECACHE_HASH_LINK *hash_link; @@ -1670,6 +1538,12 @@ static PAGECACHE_HASH_LINK *get_present_hash_link(PAGECACHE *pagecache, KEYCACHE_DBUG_ASSERT(cnt <= pagecache->hash_links_used); #endif } + if (hash_link) + { + /* Register the request for the page */ + hash_link->requests++; + } + DBUG_RETURN(hash_link); } @@ -1680,7 +1554,7 @@ static PAGECACHE_HASH_LINK *get_present_hash_link(PAGECACHE *pagecache, static PAGECACHE_HASH_LINK *get_hash_link(PAGECACHE *pagecache, PAGECACHE_FILE *file, - maria_page_no_t pageno) + pgcache_page_no_t pageno) { reg1 PAGECACHE_HASH_LINK *hash_link; PAGECACHE_HASH_LINK **start; @@ -1693,7 +1567,7 @@ restart: /* try to find the page in the cache */ hash_link= get_present_hash_link(pagecache, file, pageno, &start); - if (! hash_link) + if (!hash_link) { /* There is no hash link in the hash table for the pair (file, pageno) */ if (pagecache->free_hash_list) @@ -1714,7 +1588,7 @@ restart: page.file= *file; page.pageno= pageno; thread->opt_info= (void *) &page; - link_into_queue(&pagecache->waiting_for_hash_link, thread); + wqueue_link_into_queue(&pagecache->waiting_for_hash_link, thread); KEYCACHE_DBUG_PRINT("get_hash_link: wait", ("suspend thread %ld", thread->id)); pagecache_pthread_cond_wait(&thread->suspend, @@ -1723,14 +1597,15 @@ restart: #else KEYCACHE_DBUG_ASSERT(0); #endif + DBUG_PRINT("info", ("restarting...")); goto restart; } hash_link->file= *file; hash_link->pageno= pageno; link_hash(start, hash_link); + /* Register the request for the page */ + hash_link->requests++; } - /* Register the request for the page */ - hash_link->requests++; return hash_link; } @@ -1743,7 +1618,7 @@ restart: SYNOPSIS - find_key_block() + find_block() pagecache pointer to a page cache data structure file handler for the file to read page from pageno number of the page in the file @@ -1773,29 +1648,29 @@ restart: waits until first of this operations links any block back. */ -static PAGECACHE_BLOCK_LINK *find_key_block(PAGECACHE *pagecache, - PAGECACHE_FILE *file, - maria_page_no_t pageno, - int init_hits_left, - my_bool wrmode, - my_bool reg_req, - int *page_st) +static PAGECACHE_BLOCK_LINK *find_block(PAGECACHE *pagecache, + PAGECACHE_FILE *file, + pgcache_page_no_t pageno, + int init_hits_left, + my_bool wrmode, + my_bool reg_req, + int *page_st) { PAGECACHE_HASH_LINK *hash_link; PAGECACHE_BLOCK_LINK *block; int error= 0; int page_status; - DBUG_ENTER("find_key_block"); - KEYCACHE_THREAD_TRACE("find_key_block:begin"); + DBUG_ENTER("find_block"); + KEYCACHE_THREAD_TRACE("find_block:begin"); DBUG_PRINT("enter", ("fd: %u pos %lu wrmode: %lu", (uint) file->file, (ulong) pageno, (uint) wrmode)); - KEYCACHE_DBUG_PRINT("find_key_block", ("fd: %u pos: %lu wrmode: %lu", - (uint) file->file, (ulong) pageno, - (uint) wrmode)); + KEYCACHE_DBUG_PRINT("find_block", ("fd: %u pos: %lu wrmode: %lu", + (uint) file->file, (ulong) pageno, + (uint) wrmode)); #if !defined(DBUG_OFF) && defined(EXTRA_DEBUG) DBUG_EXECUTE("check_pagecache", - test_key_cache(pagecache, "start of find_key_block", 0);); + test_key_cache(pagecache, "start of find_block", 0);); #endif restart: @@ -1840,10 +1715,10 @@ restart: { #ifdef THREAD struct st_my_thread_var *thread= my_thread_var; - add_to_queue(&block->wqueue[COND_FOR_SAVED], thread); + wqueue_add_to_queue(&block->wqueue[COND_FOR_SAVED], thread); do { - KEYCACHE_DBUG_PRINT("find_key_block: wait", + KEYCACHE_DBUG_PRINT("find_block: wait", ("suspend thread %ld", thread->id)); pagecache_pthread_cond_wait(&thread->suspend, &pagecache->cache_lock); @@ -1871,7 +1746,7 @@ restart: { /* This is a request for a page to be removed from cache */ - KEYCACHE_DBUG_PRINT("find_key_block", + KEYCACHE_DBUG_PRINT("find_block", ("request for old page in block %u " "wrmode: %d block->status: %d", BLOCK_NUMBER(pagecache, block), wrmode, @@ -1888,17 +1763,17 @@ restart: else { hash_link->requests--; - KEYCACHE_DBUG_PRINT("find_key_block", + KEYCACHE_DBUG_PRINT("find_block", ("request waiting for old page to be saved")); { #ifdef THREAD struct st_my_thread_var *thread= my_thread_var; /* Put the request into the queue of those waiting for the old page */ - add_to_queue(&block->wqueue[COND_FOR_SAVED], thread); + wqueue_add_to_queue(&block->wqueue[COND_FOR_SAVED], thread); /* Wait until the request can be resubmitted */ do { - KEYCACHE_DBUG_PRINT("find_key_block: wait", + KEYCACHE_DBUG_PRINT("find_block: wait", ("suspend thread %ld", thread->id)); pagecache_pthread_cond_wait(&thread->suspend, &pagecache->cache_lock); @@ -1909,11 +1784,13 @@ restart: /* No parallel requests in single-threaded case */ #endif } - KEYCACHE_DBUG_PRINT("find_key_block", + KEYCACHE_DBUG_PRINT("find_block", ("request for old page resubmitted")); + DBUG_PRINT("info", ("restarting...")); /* Resubmit the request */ goto restart; } + block->status&= ~BLOCK_IN_SWITCH; } else { @@ -1941,7 +1818,8 @@ restart: pagecache->blocks_used++; } pagecache->blocks_unused--; - DBUG_ASSERT((block->status & BLOCK_WRLOCK) == 0); + DBUG_ASSERT((block->status & BLOCK_WRLOCK)); + DBUG_ASSERT(block->pins > 0); block->status= 0; #ifndef DBUG_OFF block->type= PAGECACHE_EMPTY_PAGE; @@ -1954,7 +1832,9 @@ restart: block->hash_link= hash_link; hash_link->block= block; page_status= PAGE_TO_BE_READ; - KEYCACHE_DBUG_PRINT("find_key_block", + DBUG_PRINT("info", ("page to be read set for page 0x%lx", + (ulong)block)); + KEYCACHE_DBUG_PRINT("find_block", ("got free or never used block %u", BLOCK_NUMBER(pagecache, block))); } @@ -1973,10 +1853,10 @@ restart: { struct st_my_thread_var *thread= my_thread_var; thread->opt_info= (void *) hash_link; - link_into_queue(&pagecache->waiting_for_block, thread); + wqueue_link_into_queue(&pagecache->waiting_for_block, thread); do { - KEYCACHE_DBUG_PRINT("find_key_block: wait", + KEYCACHE_DBUG_PRINT("find_block: wait", ("suspend thread %ld", thread->id)); pagecache_pthread_cond_wait(&thread->suspend, &pagecache->cache_lock); @@ -2001,19 +1881,18 @@ restart: reg_requests(pagecache, block,1); hash_link->block= block; } - else - { - DBUG_ASSERT((block->status & BLOCK_WRLOCK) == 0); - } + DBUG_ASSERT((block->status & BLOCK_WRLOCK) == 0); + DBUG_ASSERT(block->pins > 0); if (block->hash_link != hash_link && ! (block->status & BLOCK_IN_SWITCH) ) { /* this is a primary request for a new page */ DBUG_ASSERT((block->status & BLOCK_WRLOCK) == 0); + DBUG_ASSERT(block->pins > 0); block->status|= (BLOCK_IN_SWITCH | BLOCK_WRLOCK); - KEYCACHE_DBUG_PRINT("find_key_block", + KEYCACHE_DBUG_PRINT("find_block", ("got block %u for new page", BLOCK_NUMBER(pagecache, block))); @@ -2021,7 +1900,7 @@ restart: { /* The block contains a dirty page - push it out of the cache */ - KEYCACHE_DBUG_PRINT("find_key_block", ("block is dirty")); + KEYCACHE_DBUG_PRINT("find_block", ("block is dirty")); pagecache_pthread_mutex_unlock(&pagecache->cache_lock); /* @@ -2054,7 +1933,7 @@ restart: unlink_hash(pagecache, block->hash_link); /* All pending requests for this page must be resubmitted */ if (block->wqueue[COND_FOR_SAVED].last_thread) - release_queue(&block->wqueue[COND_FOR_SAVED]); + wqueue_release_queue(&block->wqueue[COND_FOR_SAVED]); } link_to_file_list(pagecache, block, file, (my_bool)(block->hash_link ? 1 : 0)); @@ -2065,6 +1944,8 @@ restart: #endif block->hash_link= hash_link; page_status= PAGE_TO_BE_READ; + DBUG_PRINT("info", ("page to be read set for page 0x%lx", + (ulong)block)); KEYCACHE_DBUG_ASSERT(block->hash_link->block == block); KEYCACHE_DBUG_ASSERT(hash_link->block->hash_link == hash_link); @@ -2072,7 +1953,7 @@ restart: else { /* This is for secondary requests for a new page only */ - KEYCACHE_DBUG_PRINT("find_key_block", + KEYCACHE_DBUG_PRINT("find_block", ("block->hash_link: %p hash_link: %p " "block->status: %u", block->hash_link, hash_link, block->status )); @@ -2087,7 +1968,7 @@ restart: { if (reg_req) reg_requests(pagecache, block, 1); - KEYCACHE_DBUG_PRINT("find_key_block", + KEYCACHE_DBUG_PRINT("find_block", ("block->hash_link: %p hash_link: %p " "block->status: %u", block->hash_link, hash_link, block->status )); @@ -2098,12 +1979,12 @@ restart: } KEYCACHE_DBUG_ASSERT(page_status != -1); - *page_st=page_status; + *page_st= page_status; DBUG_PRINT("info", ("block: 0x%lx fd: %u pos %lu block->status %u page_status %lu", (ulong) block, (uint) file->file, (ulong) pageno, block->status, (uint) page_status)); - KEYCACHE_DBUG_PRINT("find_key_block", + KEYCACHE_DBUG_PRINT("find_block", ("block: 0x%lx fd: %u pos %lu block->status %u page_status %lu", (ulong) block, (uint) file->file, (ulong) pageno, block->status, @@ -2111,16 +1992,16 @@ restart: #if !defined(DBUG_OFF) && defined(EXTRA_DEBUG) DBUG_EXECUTE("check_pagecache", - test_key_cache(pagecache, "end of find_key_block",0);); + test_key_cache(pagecache, "end of find_block",0);); #endif - KEYCACHE_THREAD_TRACE("find_key_block:end"); + KEYCACHE_THREAD_TRACE("find_block:end"); DBUG_RETURN(block); } -void pagecache_add_pin(PAGECACHE_BLOCK_LINK *block) +static void add_pin(PAGECACHE_BLOCK_LINK *block) { - DBUG_ENTER("pagecache_add_pin"); + DBUG_ENTER("add_pin"); DBUG_PRINT("enter", ("block 0x%lx pins: %u", (ulong) block, block->pins)); @@ -2137,9 +2018,9 @@ void pagecache_add_pin(PAGECACHE_BLOCK_LINK *block) DBUG_VOID_RETURN; } -void pagecache_remove_pin(PAGECACHE_BLOCK_LINK *block) +static void remove_pin(PAGECACHE_BLOCK_LINK *block) { - DBUG_ENTER("pagecache_remove_pin"); + DBUG_ENTER("remove_pin"); DBUG_PRINT("enter", ("block 0x%lx pins: %u", (ulong) block, block->pins)); @@ -2157,7 +2038,7 @@ void pagecache_remove_pin(PAGECACHE_BLOCK_LINK *block) DBUG_VOID_RETURN; } #ifdef PAGECACHE_DEBUG -void pagecache_add_lock(PAGECACHE_BLOCK_LINK *block, my_bool wl) +static void info_add_lock(PAGECACHE_BLOCK_LINK *block, my_bool wl) { PAGECACHE_LOCK_INFO *info= (PAGECACHE_LOCK_INFO *)my_malloc(sizeof(PAGECACHE_LOCK_INFO), MYF(0)); @@ -2166,7 +2047,7 @@ void pagecache_add_lock(PAGECACHE_BLOCK_LINK *block, my_bool wl) info_link((PAGECACHE_PIN_INFO **)&block->lock_list, (PAGECACHE_PIN_INFO *)info); } -void pagecache_remove_lock(PAGECACHE_BLOCK_LINK *block) +static void info_remove_lock(PAGECACHE_BLOCK_LINK *block) { PAGECACHE_LOCK_INFO *info= (PAGECACHE_LOCK_INFO *)info_find((PAGECACHE_PIN_INFO *)block->lock_list, @@ -2175,7 +2056,7 @@ void pagecache_remove_lock(PAGECACHE_BLOCK_LINK *block) info_unlink((PAGECACHE_PIN_INFO *)info); my_free((gptr)info, MYF(0)); } -void pagecache_change_lock(PAGECACHE_BLOCK_LINK *block, my_bool wl) +static void info_change_lock(PAGECACHE_BLOCK_LINK *block, my_bool wl) { PAGECACHE_LOCK_INFO *info= (PAGECACHE_LOCK_INFO *)info_find((PAGECACHE_PIN_INFO *)block->lock_list, @@ -2184,40 +2065,47 @@ void pagecache_change_lock(PAGECACHE_BLOCK_LINK *block, my_bool wl) info->write_lock= wl; } #else -#define pagecache_add_lock(B,W) -#define pagecache_remove_lock(B) -#define pagecache_change_lock(B,W) +#define info_add_lock(B,W) +#define info_remove_lock(B) +#define info_change_lock(B,W) #endif /* - Put on the block "update" type lock + Put on the block write lock SYNOPSIS - pagecache_lock_block() + get_wrlock() pagecache pointer to a page cache data structure block the block to work with RETURN 0 - OK - 1 - Try to lock the block failed + 1 - Can't lock this block, need retry */ -my_bool pagecache_lock_block(PAGECACHE *pagecache, - PAGECACHE_BLOCK_LINK *block) -{ - DBUG_ENTER("pagecache_lock_block"); +static my_bool get_wrlock(PAGECACHE *pagecache, + PAGECACHE_BLOCK_LINK *block) +{ + PAGECACHE_FILE file= block->hash_link->file; + pgcache_page_no_t pageno= block->hash_link->pageno; + DBUG_ENTER("get_wrlock"); + DBUG_PRINT("info", ("the block 0x%lx " + "files %d(%d) pages %d(%d)", + (ulong)block, + file.file, block->hash_link->file.file, + pageno, block->hash_link->pageno)); BLOCK_INFO(block); while (block->status & BLOCK_WRLOCK) { - DBUG_PRINT("info", ("fail to lock, waiting...")); + DBUG_PRINT("info", ("fail to lock, waiting... 0x%lx", (ulong)block)); /* Lock failed we will wait */ #ifdef THREAD struct st_my_thread_var *thread= my_thread_var; - add_to_queue(&block->wqueue[COND_FOR_WRLOCK], thread); + wqueue_add_to_queue(&block->wqueue[COND_FOR_WRLOCK], thread); dec_counter_for_resize_op(pagecache); do { - KEYCACHE_DBUG_PRINT("pagecache_lock_block: wait", + KEYCACHE_DBUG_PRINT("get_wrlock: wait", ("suspend thread %ld", thread->id)); pagecache_pthread_cond_wait(&thread->suspend, &pagecache->cache_lock); @@ -2227,35 +2115,61 @@ my_bool pagecache_lock_block(PAGECACHE *pagecache, DBUG_ASSERT(0); #endif BLOCK_INFO(block); - DBUG_RETURN(1); + if ((block->status & (BLOCK_REASSIGNED | BLOCK_IN_SWITCH)) || + file.file != block->hash_link->file.file || + pageno != block->hash_link->pageno) + { + DBUG_PRINT("info", ("the block 0x%lx changed => need retry" + "status %x files %d != %d or pages %d !=%d", + (ulong)block, block->status, + file.file, block->hash_link->file.file, + pageno, block->hash_link->pageno)); + DBUG_RETURN(1); + } } - /* we are doing it by global cache mutex protectio, so it is OK */ + DBUG_ASSERT(block->pins == 0); + /* we are doing it by global cache mutex protection, so it is OK */ block->status|= BLOCK_WRLOCK; DBUG_PRINT("info", ("WR lock set, block 0x%lx", (ulong)block)); DBUG_RETURN(0); } -void pagecache_unlock_block(PAGECACHE_BLOCK_LINK *block) + +/* + Remove write lock from the block + + SYNOPSIS + release_wrlock() + pagecache pointer to a page cache data structure + block the block to work with + + RETURN + 0 - OK +*/ + +static void release_wrlock(PAGECACHE_BLOCK_LINK *block) { - DBUG_ENTER("pagecache_unlock_block"); + DBUG_ENTER("release_wrlock"); BLOCK_INFO(block); DBUG_ASSERT(block->status & BLOCK_WRLOCK); + DBUG_ASSERT(block->pins > 0); block->status&= ~BLOCK_WRLOCK; DBUG_PRINT("info", ("WR lock reset, block 0x%lx", (ulong)block)); #ifdef THREAD /* release all threads waiting for write lock */ if (block->wqueue[COND_FOR_WRLOCK].last_thread) - release_queue(&block->wqueue[COND_FOR_WRLOCK]); + wqueue_release_queue(&block->wqueue[COND_FOR_WRLOCK]); #endif BLOCK_INFO(block); DBUG_VOID_RETURN; } + /* - Try to lock/uplock and pin/unpin the block + Try to lock/unlock and pin/unpin the block SYNOPSIS - pagecache_make_lock_and_pin() + make_lock_and_pin() pagecache pointer to a page cache data structure block the block to work with lock lock change mode @@ -2266,12 +2180,12 @@ void pagecache_unlock_block(PAGECACHE_BLOCK_LINK *block) 1 - Try to lock the block failed */ -my_bool pagecache_make_lock_and_pin(PAGECACHE *pagecache, - PAGECACHE_BLOCK_LINK *block, - enum pagecache_page_lock lock, - enum pagecache_page_pin pin) +static my_bool make_lock_and_pin(PAGECACHE *pagecache, + PAGECACHE_BLOCK_LINK *block, + enum pagecache_page_lock lock, + enum pagecache_page_pin pin) { - DBUG_ENTER("pagecache_make_lock_and_pin"); + DBUG_ENTER("make_lock_and_pin"); DBUG_PRINT("enter", ("block: 0x%lx (%u), wrlock: %c pins: %u, lock %s, pin: %s", (ulong)block, BLOCK_NUMBER(pagecache, block), ((block->status & BLOCK_WRLOCK)?'Y':'N'), @@ -2287,53 +2201,47 @@ my_bool pagecache_make_lock_and_pin(PAGECACHE *pagecache, { case PAGECACHE_LOCK_WRITE: /* free -> write */ /* Writelock and pin the buffer */ - if (pagecache_lock_block(pagecache, block)) + if (get_wrlock(pagecache, block)) { - DBUG_PRINT("info", ("restart")); - /* in case of fail pagecache_lock_block unlock cache */ - DBUG_RETURN(1); + /* can't lock => need retry */ + goto retry; } - /* The cache is locked so nothing afraid off */ - pagecache_add_pin(block); - pagecache_add_lock(block, 1); + + /* The cache is locked so nothing afraid of */ + add_pin(block); + info_add_lock(block, 1); break; case PAGECACHE_LOCK_WRITE_TO_READ: /* write -> read */ case PAGECACHE_LOCK_WRITE_UNLOCK: /* write -> free */ /* - Removes writelog and puts read lock (which is nothing in our + Removes write lock and puts read lock (which is nothing in our implementation) */ - pagecache_unlock_block(block); + release_wrlock(block); case PAGECACHE_LOCK_READ_UNLOCK: /* read -> free */ case PAGECACHE_LOCK_LEFT_READLOCKED: /* read -> read */ -#ifndef DBUG_OFF if (pin == PAGECACHE_UNPIN) { - pagecache_remove_pin(block); + remove_pin(block); } -#endif -#ifdef PAGECACHE_DEBUG if (lock == PAGECACHE_LOCK_WRITE_TO_READ) { - pagecache_change_lock(block, 0); + info_change_lock(block, 0); } else if (lock == PAGECACHE_LOCK_WRITE_UNLOCK || lock == PAGECACHE_LOCK_READ_UNLOCK) { - pagecache_remove_lock(block); + info_remove_lock(block); } -#endif break; case PAGECACHE_LOCK_READ: /* free -> read */ -#ifndef DBUG_OFF if (pin == PAGECACHE_PIN) { /* The cache is locked so nothing afraid off */ - pagecache_add_pin(block); + add_pin(block); } - pagecache_add_lock(block, 0); + info_add_lock(block, 0); break; -#endif case PAGECACHE_LOCK_LEFT_UNLOCKED: /* free -> free */ case PAGECACHE_LOCK_LEFT_WRITELOCKED: /* write -> write */ break; /* do nothing */ @@ -2343,6 +2251,16 @@ my_bool pagecache_make_lock_and_pin(PAGECACHE *pagecache, BLOCK_INFO(block); DBUG_RETURN(0); +retry: + DBUG_PRINT("INFO", ("Retry block 0x%lx", (ulong)block)); + BLOCK_INFO(block); + DBUG_ASSERT(block->hash_link->requests != 0); + block->hash_link->requests--; + DBUG_ASSERT(block->requests != 0); + unreg_request(pagecache, block, 1); + BLOCK_INFO(block); + DBUG_RETURN(1); + } @@ -2355,6 +2273,8 @@ my_bool pagecache_make_lock_and_pin(PAGECACHE *pagecache, pagecache pointer to a page cache data structure block block to which buffer the data is to be read primary <-> the current thread will read the data + validator validator of read from the disk data + validator_data pointer to the data need by the validator RETURN VALUE None @@ -2368,13 +2288,15 @@ my_bool pagecache_make_lock_and_pin(PAGECACHE *pagecache, static void read_block(PAGECACHE *pagecache, PAGECACHE_BLOCK_LINK *block, - my_bool primary) + my_bool primary, + pagecache_disk_read_validator validator, + gptr validator_data) { uint got_length; /* On entry cache_lock is locked */ - KEYCACHE_THREAD_TRACE("read_block"); + DBUG_ENTER("read_block"); if (primary) { /* @@ -2382,8 +2304,8 @@ static void read_block(PAGECACHE *pagecache, that submitted primary requests */ - KEYCACHE_DBUG_PRINT("read_block", - ("page to be read by primary request")); + DBUG_PRINT("read_block", + ("page to be read by primary request")); /* Page is not in buffer yet, is to be read from disk */ pagecache_pthread_mutex_unlock(&pagecache->cache_lock); @@ -2400,11 +2322,15 @@ static void read_block(PAGECACHE *pagecache, else block->status= (BLOCK_READ | (block->status & BLOCK_WRLOCK)); - KEYCACHE_DBUG_PRINT("read_block", - ("primary request: new page in cache")); + if (validator != NULL && + (*validator)(block->buffer, validator_data)) + block->status|= BLOCK_ERROR; + + DBUG_PRINT("read_block", + ("primary request: new page in cache")); /* Signal that all pending requests for this page now can be processed */ if (block->wqueue[COND_FOR_REQUESTED].last_thread) - release_queue(&block->wqueue[COND_FOR_REQUESTED]); + wqueue_release_queue(&block->wqueue[COND_FOR_REQUESTED]); } else { @@ -2412,17 +2338,17 @@ static void read_block(PAGECACHE *pagecache, This code is executed only by threads that submitted secondary requests */ - KEYCACHE_DBUG_PRINT("read_block", - ("secondary request waiting for new page to be read")); + DBUG_PRINT("read_block", + ("secondary request waiting for new page to be read")); { #ifdef THREAD struct st_my_thread_var *thread= my_thread_var; /* Put the request into a queue and wait until it can be processed */ - add_to_queue(&block->wqueue[COND_FOR_REQUESTED], thread); + wqueue_add_to_queue(&block->wqueue[COND_FOR_REQUESTED], thread); do { - KEYCACHE_DBUG_PRINT("read_block: wait", - ("suspend thread %ld", thread->id)); + DBUG_PRINT("read_block: wait", + ("suspend thread %ld", thread->id)); pagecache_pthread_cond_wait(&thread->suspend, &pagecache->cache_lock); } @@ -2432,9 +2358,10 @@ static void read_block(PAGECACHE *pagecache, /* No parallel requests in single-threaded case */ #endif } - KEYCACHE_DBUG_PRINT("read_block", - ("secondary request: new page in cache")); + DBUG_PRINT("read_block", + ("secondary request: new page in cache")); } + DBUG_VOID_RETURN; } @@ -2454,11 +2381,11 @@ static void read_block(PAGECACHE *pagecache, void pagecache_unlock_page(PAGECACHE *pagecache, PAGECACHE_FILE *file, - maria_page_no_t pageno, + pgcache_page_no_t pageno, enum pagecache_page_lock lock, enum pagecache_page_pin pin, my_bool stamp_this_page, - LSN first_REDO_LSN_for_page) + LSN_PTR first_REDO_LSN_for_page) { PAGECACHE_BLOCK_LINK *block; int page_st; @@ -2471,24 +2398,6 @@ void pagecache_unlock_page(PAGECACHE *pagecache, DBUG_ASSERT(pin != PAGECACHE_PIN && lock != PAGECACHE_LOCK_READ && lock != PAGECACHE_LOCK_WRITE); - if (pin == PAGECACHE_PIN_LEFT_UNPINNED && - lock == PAGECACHE_LOCK_READ_UNLOCK) - { -#ifndef DBUG_OFF - if ( -#endif - /* block do not need here so we do not provide it */ - pagecache_make_lock_and_pin(pagecache, 0, lock, pin) -#ifndef DBUG_OFF - ) - { - DBUG_ASSERT(0); /* should not happend */ - } -#else - ; -#endif - DBUG_VOID_RETURN; - } pagecache_pthread_mutex_lock(&pagecache->cache_lock); /* @@ -2498,7 +2407,7 @@ void pagecache_unlock_page(PAGECACHE *pagecache, DBUG_ASSERT(pagecache->can_be_used); inc_counter_for_resize_op(pagecache); - block= find_key_block(pagecache, file, pageno, 0, 0, 0, &page_st); + block= find_block(pagecache, file, pageno, 0, 0, 0, &page_st); BLOCK_INFO(block); DBUG_ASSERT(block != 0 && page_st == PAGE_READ); if (stamp_this_page) @@ -2511,7 +2420,7 @@ void pagecache_unlock_page(PAGECACHE *pagecache, #ifndef DBUG_OFF if ( #endif - pagecache_make_lock_and_pin(pagecache, block, lock, pin) + make_lock_and_pin(pagecache, block, lock, pin) #ifndef DBUG_OFF ) { @@ -2549,7 +2458,7 @@ void pagecache_unlock_page(PAGECACHE *pagecache, void pagecache_unpin_page(PAGECACHE *pagecache, PAGECACHE_FILE *file, - maria_page_no_t pageno) + pgcache_page_no_t pageno) { PAGECACHE_BLOCK_LINK *block; int page_st; @@ -2565,7 +2474,7 @@ void pagecache_unpin_page(PAGECACHE *pagecache, DBUG_ASSERT(pagecache->can_be_used); inc_counter_for_resize_op(pagecache); - block= find_key_block(pagecache, file, pageno, 0, 0, 0, &page_st); + block= find_block(pagecache, file, pageno, 0, 0, 0, &page_st); DBUG_ASSERT(block != 0 && page_st == PAGE_READ); #ifndef DBUG_OFF @@ -2576,9 +2485,9 @@ void pagecache_unpin_page(PAGECACHE *pagecache, a) we can't pin without any lock b) we can't unpin keeping write lock */ - pagecache_make_lock_and_pin(pagecache, block, - PAGECACHE_LOCK_LEFT_READLOCKED, - PAGECACHE_UNPIN) + make_lock_and_pin(pagecache, block, + PAGECACHE_LOCK_LEFT_READLOCKED, + PAGECACHE_UNPIN) #ifndef DBUG_OFF ) { @@ -2622,7 +2531,7 @@ void pagecache_unlock(PAGECACHE *pagecache, enum pagecache_page_lock lock, enum pagecache_page_pin pin, my_bool stamp_this_page, - LSN first_REDO_LSN_for_page) + LSN_PTR first_REDO_LSN_for_page) { PAGECACHE_BLOCK_LINK *block= (PAGECACHE_BLOCK_LINK *)link; DBUG_ENTER("pagecache_unlock"); @@ -2643,7 +2552,7 @@ void pagecache_unlock(PAGECACHE *pagecache, if ( #endif /* block do not need here so we do not provide it */ - pagecache_make_lock_and_pin(pagecache, 0, lock, pin) + make_lock_and_pin(pagecache, 0, lock, pin) #ifndef DBUG_OFF ) { @@ -2673,7 +2582,7 @@ void pagecache_unlock(PAGECACHE *pagecache, #ifndef DBUG_OFF if ( #endif - pagecache_make_lock_and_pin(pagecache, block, lock, pin) + make_lock_and_pin(pagecache, block, lock, pin) #ifndef DBUG_OFF ) { @@ -2736,9 +2645,9 @@ void pagecache_unpin(PAGECACHE *pagecache, a) we can't pin without any lock b) we can't unpin keeping write lock */ - pagecache_make_lock_and_pin(pagecache, block, - PAGECACHE_LOCK_LEFT_READLOCKED, - PAGECACHE_UNPIN) + make_lock_and_pin(pagecache, block, + PAGECACHE_LOCK_LEFT_READLOCKED, + PAGECACHE_UNPIN) #ifndef DBUG_OFF ) { @@ -2767,7 +2676,7 @@ void pagecache_unpin(PAGECACHE *pagecache, Read a block of data from a cached file into a buffer; SYNOPSIS - pagecache_read() + pagecache_valid_read() pagecache pointer to a page cache data structure file handler for the file for the block of data to be read pageno number of the block of data in the file @@ -2776,16 +2685,12 @@ void pagecache_unpin(PAGECACHE *pagecache, type type of the page lock lock change link link to the page if we pin it + validator validator of read from the disk data + validator_data pointer to the data need by the validator RETURN VALUE Returns address from where the data is placed if sucessful, 0 - otherwise. - NOTES. - - The function ensures that a block of data of size length from file - positioned at pageno is in the buffers for some key cache blocks. - Then the function copies the data into the buffer buff. - Pin will be choosen according to lock parameter (see lock_to_pin) */ static enum pagecache_page_pin lock_to_pin[]= @@ -2800,19 +2705,21 @@ static enum pagecache_page_pin lock_to_pin[]= PAGECACHE_UNPIN /*PAGECACHE_LOCK_WRITE_TO_READ*/ }; -byte *pagecache_read(PAGECACHE *pagecache, - PAGECACHE_FILE *file, - maria_page_no_t pageno, - uint level, - byte *buff, - enum pagecache_page_type type, - enum pagecache_page_lock lock, - PAGECACHE_PAGE_LINK *link) +byte *pagecache_valid_read(PAGECACHE *pagecache, + PAGECACHE_FILE *file, + pgcache_page_no_t pageno, + uint level, + byte *buff, + enum pagecache_page_type type, + enum pagecache_page_lock lock, + PAGECACHE_PAGE_LINK *link, + pagecache_disk_read_validator validator, + gptr validator_data) { int error= 0; enum pagecache_page_pin pin= lock_to_pin[lock]; PAGECACHE_PAGE_LINK fake_link; - DBUG_ENTER("page_cache_read"); + DBUG_ENTER("pagecache_valid_read"); DBUG_PRINT("enter", ("fd: %u page: %lu level: %u t:%s l%s p%s", (uint) file->file, (ulong) pageno, level, page_cache_page_type_str[type], @@ -2829,7 +2736,7 @@ restart: if (pagecache->can_be_used) { /* Key cache is used */ - reg1 PAGECACHE_BLOCK_LINK *block; + PAGECACHE_BLOCK_LINK *block; uint status; int page_st; @@ -2842,29 +2749,33 @@ restart: inc_counter_for_resize_op(pagecache); pagecache->global_cache_r_requests++; - block= find_key_block(pagecache, file, pageno, level, - ((lock == PAGECACHE_LOCK_WRITE) ? 1 : 0), - (((pin == PAGECACHE_PIN_LEFT_PINNED) || - (pin == PAGECACHE_UNPIN)) ? 0 : 1), - &page_st); + block= find_block(pagecache, file, pageno, level, + test(lock == PAGECACHE_LOCK_WRITE), + test((pin == PAGECACHE_PIN_LEFT_PINNED) || + (pin == PAGECACHE_UNPIN)), + &page_st); DBUG_ASSERT(block->type == PAGECACHE_EMPTY_PAGE || block->type == type); block->type= type; - if (pagecache_make_lock_and_pin(pagecache, block, lock, pin)) + if (block->status != BLOCK_ERROR && page_st != PAGE_READ) + { + DBUG_PRINT("info", ("read block 0x%lx", (ulong)block)); + /* The requested page is to be read into the block buffer */ + read_block(pagecache, block, + (my_bool)(page_st == PAGE_TO_BE_READ), + validator, validator_data); + DBUG_PRINT("info", ("read is done")); + } + if (make_lock_and_pin(pagecache, block, lock, pin)) { /* - We failed to write lock the block, cache is unlocked, and last write - lock is released, we will try to get the block again. + We failed to write lock the block, cache is unlocked, + we will try to get the block again. */ pagecache_pthread_mutex_unlock(&pagecache->cache_lock); + DBUG_PRINT("info", ("restarting...")); goto restart; } - if (block->status != BLOCK_ERROR && page_st != PAGE_READ) - { - /* The requested page is to be read into the block buffer */ - read_block(pagecache, block, - (my_bool)(page_st == PAGE_TO_BE_READ)); - } if (! ((status= block->status) & BLOCK_ERROR)) { @@ -2933,7 +2844,7 @@ no_key_cache: /* Key cache is not used */ */ my_bool pagecache_delete_page(PAGECACHE *pagecache, PAGECACHE_FILE *file, - maria_page_no_t pageno, + pgcache_page_no_t pageno, enum pagecache_page_lock lock, my_bool flush) { @@ -2969,13 +2880,14 @@ restart: } block= link->block; DBUG_ASSERT(block != 0); - if (pagecache_make_lock_and_pin(pagecache, block, lock, pin)) + if (make_lock_and_pin(pagecache, block, lock, pin)) { /* We failed to writelock the block, cache is unlocked, and last write lock is released, we will try to get the block again. */ pagecache_pthread_mutex_unlock(&pagecache->cache_lock); + DBUG_PRINT("info", ("restarting...")); goto restart; } @@ -2983,7 +2895,7 @@ restart: { /* The block contains a dirty page - push it out of the cache */ - KEYCACHE_DBUG_PRINT("find_key_block", ("block is dirty")); + KEYCACHE_DBUG_PRINT("find_block", ("block is dirty")); pagecache_pthread_mutex_unlock(&pagecache->cache_lock); /* @@ -3015,9 +2927,10 @@ restart: } /* Cache is locked, so we can relese page before freeing it */ - pagecache_make_lock_and_pin(pagecache, block, - PAGECACHE_LOCK_WRITE_UNLOCK, - PAGECACHE_UNPIN); + make_lock_and_pin(pagecache, block, + PAGECACHE_LOCK_WRITE_UNLOCK, + PAGECACHE_UNPIN); + link->requests--; if (pin == PAGECACHE_PIN_LEFT_PINNED) unreg_request(pagecache, block, 1); free_block(pagecache, block); @@ -3053,11 +2966,12 @@ end: 0 if a success, 1 - otherwise. */ +/* description of how to change lock before and after write */ struct write_lock_change { - int need_lock_change; - enum pagecache_page_lock new_lock; - enum pagecache_page_lock unlock_lock; + int need_lock_change; /* need changing of lock at the end of write */ + enum pagecache_page_lock new_lock; /* lock at the beginning */ + enum pagecache_page_lock unlock_lock; /* lock at the end */ }; static struct write_lock_change write_lock_change_table[]= @@ -3084,10 +2998,11 @@ static struct write_lock_change write_lock_change_table[]= PAGECACHE_LOCK_WRITE_TO_READ}/*PAGECACHE_LOCK_WRITE_TO_READ*/ }; +/* description of how to change pin before and after write */ struct write_pin_change { - enum pagecache_page_pin new_pin; - enum pagecache_page_pin unlock_pin; + enum pagecache_page_pin new_pin; /* pin status at the beginning */ + enum pagecache_page_pin unlock_pin; /* pin status at the end */ }; static struct write_pin_change write_pin_change_table[]= @@ -3104,7 +3019,7 @@ static struct write_pin_change write_pin_change_table[]= my_bool pagecache_write(PAGECACHE *pagecache, PAGECACHE_FILE *file, - maria_page_no_t pageno, + pgcache_page_no_t pageno, uint level, byte *buff, enum pagecache_page_type type, @@ -3113,7 +3028,7 @@ my_bool pagecache_write(PAGECACHE *pagecache, enum pagecache_write_mode write_mode, PAGECACHE_PAGE_LINK *link) { - reg1 PAGECACHE_BLOCK_LINK *block; + reg1 PAGECACHE_BLOCK_LINK *block= NULL; PAGECACHE_PAGE_LINK fake_link; int error= 0; int need_lock_change= write_lock_change_table[lock].need_lock_change; @@ -3133,7 +3048,7 @@ my_bool pagecache_write(PAGECACHE *pagecache, if (write_mode == PAGECACHE_WRITE_NOW) { - /* we allow direct write if wwe do not use long term lockings */ + /* we allow direct write if we do not use long term lockings */ DBUG_ASSERT(lock == PAGECACHE_LOCK_LEFT_UNLOCKED); /* Force writing from buff into disk */ pagecache->global_cache_write++; @@ -3167,10 +3082,10 @@ restart: lock != PAGECACHE_LOCK_LEFT_WRITELOCKED && lock != PAGECACHE_LOCK_WRITE_UNLOCK && lock != PAGECACHE_LOCK_WRITE_TO_READ); - block= find_key_block(pagecache, file, pageno, level, - (need_wrlock ? 1 : 0), - (need_wrlock ? 1 : 0), - &page_st); + block= find_block(pagecache, file, pageno, level, + (need_wrlock ? 1 : 0), + (need_wrlock ? 1 : 0), + &page_st); } if (!block) { @@ -3186,24 +3101,25 @@ restart: block->type == type); block->type= type; - if (pagecache_make_lock_and_pin(pagecache, block, - write_lock_change_table[lock].new_lock, - (need_lock_change ? - write_pin_change_table[pin].new_pin : - pin))) + if (make_lock_and_pin(pagecache, block, + write_lock_change_table[lock].new_lock, + (need_lock_change ? + write_pin_change_table[pin].new_pin : + pin))) { /* We failed to writelock the block, cache is unlocked, and last write lock is released, we will try to get the block again. */ pagecache_pthread_mutex_unlock(&pagecache->cache_lock); + DBUG_PRINT("info", ("restarting...")); goto restart; } if (write_mode == PAGECACHE_WRITE_DONE) { - if (block->status != BLOCK_ERROR && page_st != PAGE_READ) + if ((block->status & BLOCK_ERROR) && page_st != PAGE_READ) { /* Copy data from buff */ bmove512(block->buffer, buff, pagecache->block_size); @@ -3212,7 +3128,7 @@ restart: ("primary request: new page in cache")); /* Signal that all pending requests for this now can be processed. */ if (block->wqueue[COND_FOR_REQUESTED].last_thread) - release_queue(&block->wqueue[COND_FOR_REQUESTED]); + wqueue_release_queue(&block->wqueue[COND_FOR_REQUESTED]); } } else @@ -3220,7 +3136,8 @@ restart: if (write_mode == PAGECACHE_WRITE_NOW) { /* buff has been written to disk at start */ - if (block->status & BLOCK_CHANGED) + if ((block->status & BLOCK_CHANGED) && + !(block->status & BLOCK_ERROR)) link_to_file_list(pagecache, block, &block->hash_link->file, 1); } else @@ -3231,8 +3148,8 @@ restart: if (! (block->status & BLOCK_ERROR)) { bmove512(block->buffer, buff, pagecache->block_size); + block->status|= BLOCK_READ; } - block->status|= BLOCK_READ; } @@ -3242,9 +3159,9 @@ restart: int rc= #endif #warning we are doing an unlock here, so need to give the page its rec_lsn! - pagecache_make_lock_and_pin(pagecache, block, - write_lock_change_table[lock].unlock_lock, - write_pin_change_table[pin].unlock_pin); + make_lock_and_pin(pagecache, block, + write_lock_change_table[lock].unlock_lock, + write_pin_change_table[pin].unlock_pin); #ifndef DBUG_OFF DBUG_ASSERT(rc == 0); #endif @@ -3255,10 +3172,7 @@ restart: block->hash_link->requests--; if (pin != PAGECACHE_PIN_LEFT_PINNED && pin != PAGECACHE_PIN) { - if (write_mode != PAGECACHE_WRITE_DONE) - { - unreg_request(pagecache, block, 1); - } + unreg_request(pagecache, block, 1); } else *link= (PAGECACHE_PAGE_LINK)block; @@ -3290,6 +3204,7 @@ end: DBUG_EXECUTE("exec", test_key_cache(pagecache, "end of key_cache_write", 1);); #endif + BLOCK_INFO(block); DBUG_RETURN(error); } @@ -3321,6 +3236,7 @@ static void free_block(PAGECACHE *pagecache, PAGECACHE_BLOCK_LINK *block) unlink_changed(block); DBUG_ASSERT((block->status & BLOCK_WRLOCK) == 0); + DBUG_ASSERT(block->pins > 0); block->status= 0; #ifndef DBUG_OFF block->type= PAGECACHE_EMPTY_PAGE; @@ -3344,7 +3260,7 @@ static void free_block(PAGECACHE *pagecache, PAGECACHE_BLOCK_LINK *block) /* All pending requests for this page must be resubmitted. */ if (block->wqueue[COND_FOR_SAVED].last_thread) - release_queue(&block->wqueue[COND_FOR_SAVED]); + wqueue_release_queue(&block->wqueue[COND_FOR_SAVED]); } @@ -3398,12 +3314,13 @@ static int flush_cached_blocks(PAGECACHE *pagecache, } /* if the block is not pinned then it is not write locked */ DBUG_ASSERT((block->status & BLOCK_WRLOCK) == 0); + DBUG_ASSERT(block->pins > 0); #ifndef DBUG_OFF { int rc= #endif - pagecache_make_lock_and_pin(pagecache, block, - PAGECACHE_LOCK_WRITE, PAGECACHE_PIN); + make_lock_and_pin(pagecache, block, + PAGECACHE_LOCK_WRITE, PAGECACHE_PIN); #ifndef DBUG_OFF DBUG_ASSERT(rc == 0); } @@ -3427,9 +3344,9 @@ static int flush_cached_blocks(PAGECACHE *pagecache, MYF(MY_NABP | MY_WAIT_IF_FULL)); pagecache_pthread_mutex_lock(&pagecache->cache_lock); - pagecache_make_lock_and_pin(pagecache, block, - PAGECACHE_LOCK_WRITE_UNLOCK, - PAGECACHE_UNPIN); + make_lock_and_pin(pagecache, block, + PAGECACHE_LOCK_WRITE_UNLOCK, + PAGECACHE_UNPIN); pagecache->global_cache_write++; if (error) @@ -3443,7 +3360,7 @@ static int flush_cached_blocks(PAGECACHE *pagecache, It might happen only during an operation to resize the key cache. */ if (block->wqueue[COND_FOR_SAVED].last_thread) - release_queue(&block->wqueue[COND_FOR_SAVED]); + wqueue_release_queue(&block->wqueue[COND_FOR_SAVED]); /* type will never be FLUSH_IGNORE_CHANGED here */ if (! (type == FLUSH_KEEP || type == FLUSH_FORCE_WRITE)) { @@ -3577,6 +3494,7 @@ restart: if ((error= flush_cached_blocks(pagecache, file, cache, end,type))) last_errno=error; + DBUG_PRINT("info", ("restarting...")); /* Restart the scan as some other thread might have changed the changed blocks chain: the blocks that were in switch @@ -3622,7 +3540,7 @@ removes a page from the list of dirty pages, while it's still dirty. A \ { #ifdef THREAD struct st_my_thread_var *thread= my_thread_var; - add_to_queue(&block->wqueue[COND_FOR_SAVED], thread); + wqueue_add_to_queue(&block->wqueue[COND_FOR_SAVED], thread); do { KEYCACHE_DBUG_PRINT("flush_pagecache_blocks_int: wait", @@ -3761,7 +3679,7 @@ static int flush_all_key_blocks(PAGECACHE *pagecache) 0 on success (always because it can't fail) */ -int reset_key_cache_counters(const char *name, PAGECACHE *key_cache) +static int reset_key_cache_counters(const char *name, PAGECACHE *key_cache) { DBUG_ENTER("reset_key_cache_counters"); if (!key_cache->inited) diff --git a/mysys/wqueue.c b/mysys/wqueue.c new file mode 100644 index 00000000000..28e044ff606 --- /dev/null +++ b/mysys/wqueue.c @@ -0,0 +1,167 @@ + +#include + +#define STRUCT_PTR(TYPE, MEMBER, a) \ + (TYPE *) ((char *) (a) - offsetof(TYPE, MEMBER)) +/* + Link a thread into double-linked queue of waiting threads. + + SYNOPSIS + wqueue_link_into_queue() + wqueue pointer to the queue structure + thread pointer to the thread to be added to the queue + + RETURN VALUE + none + + NOTES. + Queue is represented by a circular list of the thread structures + The list is double-linked of the type (**prev,*next), accessed by + a pointer to the last element. +*/ + +void wqueue_link_into_queue(WQUEUE *wqueue, struct st_my_thread_var *thread) +{ + struct st_my_thread_var *last; + if (!(last= wqueue->last_thread)) + { + /* Queue is empty */ + thread->next= thread; + thread->prev= &thread->next; + } + else + { + thread->prev= last->next->prev; + last->next->prev= &thread->next; + thread->next= last->next; + last->next= thread; + } + wqueue->last_thread= thread; +} + + +/* + Add a thread to single-linked queue of waiting threads + + SYNOPSIS + wqueue_add_to_queue() + wqueue pointer to the queue structure + thread pointer to the thread to be added to the queue + + RETURN VALUE + none + + NOTES. + Queue is represented by a circular list of the thread structures + The list is single-linked of the type (*next), accessed by a pointer + to the last element. +*/ + +void wqueue_add_to_queue(WQUEUE *wqueue, struct st_my_thread_var *thread) +{ + struct st_my_thread_var *last; + if (!(last= wqueue->last_thread)) + thread->next= thread; + else + { + thread->next= last->next; + last->next= thread; + } + wqueue->last_thread= thread; +} + +/* + Unlink a thread from double-linked queue of waiting threads + + SYNOPSIS + wqueue_unlink_from_queue() + wqueue pointer to the queue structure + thread pointer to the thread to be removed from the queue + + RETURN VALUE + none + + NOTES. + See NOTES for link_into_queue +*/ + +void wqueue_unlink_from_queue(WQUEUE *wqueue, struct st_my_thread_var *thread) +{ + if (thread->next == thread) + /* The queue contains only one member */ + wqueue->last_thread= NULL; + else + { + thread->next->prev= thread->prev; + *thread->prev= thread->next; + if (wqueue->last_thread == thread) + wqueue->last_thread= STRUCT_PTR(struct st_my_thread_var, next, + thread->prev); + } + thread->next= NULL; +} + + +/* + Remove all threads from queue signaling them to proceed + + SYNOPSIS + wqueue_realease_queue() + wqueue pointer to the queue structure + thread pointer to the thread to be added to the queue + + RETURN VALUE + none + + NOTES. + See notes for add_to_queue + When removed from the queue each thread is signaled via condition + variable thread->suspend. +*/ + +void wqueue_release_queue(WQUEUE *wqueue) +{ + struct st_my_thread_var *last= wqueue->last_thread; + struct st_my_thread_var *next= last->next; + struct st_my_thread_var *thread; + do + { + thread= next; + pthread_cond_signal(&thread->suspend); + next= thread->next; + thread->next= NULL; + } + while (thread != last); + wqueue->last_thread= NULL; +} + + +/* + Add thread and wait + + SYNOPSYS + wqueue_add_and_wait() + wqueue queue to add to + thread thread which is waiting + lock mutex need for the operation +*/ + +void wqueue_add_and_wait(WQUEUE *wqueue, + struct st_my_thread_var *thread, pthread_mutex_t *lock) +{ + DBUG_ENTER("wqueue_add_and_wait"); + DBUG_PRINT("enter", ("thread ox%lxcond 0x%lx, mutex 0x%lx", + (ulong) thread, (ulong) &thread->suspend, (ulong) lock)); + wqueue_add_to_queue(wqueue, thread); + do + { + DBUG_PRINT("info", ("wait... cond 0x%lx, mutex 0x%lx", + (ulong) &thread->suspend, (ulong) lock)); + pthread_cond_wait(&thread->suspend, lock); + DBUG_PRINT("info", ("wait done cond 0x%lx, mutex 0x%lx, next 0x%lx", + (ulong) &thread->suspend, (ulong) lock, + (ulong) thread->next)); + } + while (thread->next); + DBUG_VOID_RETURN; +} diff --git a/storage/maria/Makefile.am b/storage/maria/Makefile.am index 24636f139ab..2aa9a8a36cb 100644 --- a/storage/maria/Makefile.am +++ b/storage/maria/Makefile.am @@ -110,7 +110,7 @@ libmaria_a_SOURCES = ma_init.c ma_open.c ma_extra.c ma_info.c ma_rkey.c \ ma_ft_nlq_search.c ft_maria.c ma_sort.c \ ha_maria.cc trnman.c lockman.c tablockman.c \ ma_rt_index.c ma_rt_key.c ma_rt_mbr.c ma_rt_split.c \ - ma_sp_key.c ma_control_file.c + ma_sp_key.c ma_control_file.c ma_loghandler.c CLEANFILES = test?.MA? FT?.MA? isam.log ma_test_all ma_rt_test.MA? sp_test.MA? SUFFIXES = .sh diff --git a/storage/maria/ma_control_file.h b/storage/maria/ma_control_file.h index 9a99a721469..4b5ddd006c1 100644 --- a/storage/maria/ma_control_file.h +++ b/storage/maria/ma_control_file.h @@ -17,28 +17,10 @@ /* WL#3234 Maria control file First version written by Guilhem Bichot on 2006-04-27. - Does not compile yet. */ -#ifndef _control_file_h -#define _control_file_h - -/* - Not everybody needs to call the control file that's why control_file.h is - not in maria_def.h. However, policy or habit may want to change this. -*/ - -#ifndef REMOVE_WHEN_SANJA_PUSHES_LOG_HANDLER -/* - this is to get the control file to compile, until Sanja pushes the log - handler which will supersede those definitions. -*/ -typedef struct st_lsn { - uint32 file_no; - uint32 rec_offset; -} LSN; -#define maria_data_root "." -#endif +#ifndef _ma_control_file_h +#define _ma_control_file_h #define CONTROL_FILE_BASE_NAME "maria_control" /* diff --git a/storage/maria/ma_loghandler.c b/storage/maria/ma_loghandler.c new file mode 100644 index 00000000000..d5c19e29ce2 --- /dev/null +++ b/storage/maria/ma_loghandler.c @@ -0,0 +1,5417 @@ +#include "maria_def.h" +#include + +/* number of opened log files in the pagecache (should be at lesst 2) */ +#define OPENED_FILES_NUM 3 + +/* records buffer size (should be LOG_PAGE_SIZE * n) */ +#define TRANSLOG_WRITE_BUFFER (1024*1024) +/* min chunk length */ +#define TRANSLOG_MIN_CHUNK 3 +/* + Number of buffers used by loghandler + + Should be at least 4, because one thread can block up to 2 buffers in + normal circumstances (less then half of one and full other, or just + switched one and other), But if we met end of the file in the middle and + have to switch buffer it will be 3. + 1 or 2 buffer for flushing/writing. +*/ +#define TRANSLOG_BUFFERS_NO 5 +/* number of bytes which is worth to be left on first page */ +#define TRANSLOG_MINCHUNK_CONTENT 1 +/* length of transaction log file name maria_log.XXXXXXXX*/ +#define TRANSLOG_FILE_NAME_LENGTH 18 +/* version of log file */ +#define TRANSLOG_VERSION_ID 10000 + +#define UNRECOVERABLE_ERROR(E) \ + do { \ + DBUG_PRINT("error", E); \ + printf E; \ + putchar('\n'); \ + } while(0); + + +/* record part descriptor */ +struct st_translog_part +{ + translog_size_t len; + uchar *buff; +}; + +/* record parts descriptor */ +struct st_translog_parts +{ + /* full record length */ + translog_size_t record_length; + /* full record length with chunk headers */ + translog_size_t total_record_length; + /* array of parts (st_translog_part) */ + DYNAMIC_ARRAY parts; + /* current part index */ + uint current; +}; + +/* log write buffer descriptor */ +struct st_translog_buffer +{ + LSN last_lsn; + /* This buffer offset in the file */ + TRANSLOG_ADDRESS offset; + /* + How much written (or will be written when copy_to_buffer_in_progress + become 0) to this buffer + */ + uint32 size; + /* This Buffer File */ + File file; + /* Threads which are waiting for buffer filling/freeing */ + WQUEUE waiting_filling_buffer; + /* Number of record which are in copy progress */ + int16 copy_to_buffer_in_progress; + /* list of waiting buffer ready threads */ + struct st_my_thread_var *waiting_flush; + /* lock for the buffer. Current buffer also lock the handler */ + pthread_mutex_t mutex; + struct st_translog_buffer *overlay; +#ifndef DBUG_OFF + struct st_my_thread_var *locked_by; + uint8 buffer_no; +#endif + /* IO cache for current log */ + uchar buffer[TRANSLOG_WRITE_BUFFER]; +}; + + +struct st_buffer_cursor +{ + /* pointer on the buffer */ + uchar *ptr; + /* current page fill */ + uint16 current_page_size; + /* how many times we finish this page to write it */ + uint16 write_counter; + /* previous write offset */ + uint16 previous_offset; + /* current buffer and its number */ + struct st_translog_buffer *buffer; + uint8 buffer_no; + my_bool chaser, protected; +}; + + +struct st_translog_descriptor +{ + /* *** Parameters of the log handler *** */ + + /* Directory to store files */ + char directory[FN_REFLEN]; + /* max size of one log size (for new logs creation) */ + uint32 log_file_max_size; + /* server version */ + uint32 server_version; + /* server ID */ + uint32 server_id; + /* Page cache for the log reads */ + PAGECACHE *pagecache; + /* Flags */ + uint flags; + /* Page overhead calculated by flags */ + uint16 page_overhead; + /* Page capacity calculated by flags (TRANSLOG_PAGE_SIZE-page_overhead-1) */ + uint16 page_capacity_chunk_2; + /* Loghandler's buffer capacity in case of chunk 2 filling */ + uint32 buffer_capacity_chunk_2; + /* Half of the buffer capacity in case of chunk 2 filling */ + uint32 half_buffer_capacity_chunk_2; + + /* *** Current state of the log handler *** */ + /* Current and (OPENED_FILES_NUM-1) last logs number in page cache */ + File log_file_num[OPENED_FILES_NUM]; + /* buffers for log writing */ + struct st_translog_buffer buffers[TRANSLOG_BUFFERS_NO]; + /* + horizon - visible end of the log (here is absolute end of the log: + position where next chunk can start + */ + TRANSLOG_ADDRESS horizon; + /* horizon buffer cursor */ + struct st_buffer_cursor bc; + + /* Last flushed LSN */ + LSN flushed; + LSN sent_to_file; + pthread_mutex_t sent_to_file_lock; + File directory_fd; +}; + +static struct st_translog_descriptor log_descriptor; + +static uchar end_of_log= 0; + +/* record classes */ +enum record_class +{ + LOGRECTYPE_NOT_ALLOWED, + LOGRECTYPE_VARIABLE_LENGTH, + LOGRECTYPE_PSEUDOFIXEDLENGTH, + LOGRECTYPE_FIXEDLENGTH +}; + +/* chunk types */ +#define TRANSLOG_CHUNK_LSN 0x00 /* 0 chunk refer as LSN (head + or tail */ +#define TRANSLOG_CHUNK_FIXED 0x40 /* 1 (pseudo)fixed record (also + LSN) */ +#define TRANSLOG_CHUNK_NOHDR 0x80 /* 2 no header chunk (till page + end) */ +#define TRANSLOG_CHUNK_LNGTH 0xC0 /* 3 chunk with chunk length */ +#define TRANSLOG_CHUNK_TYPE 0xC0 /* Mask to get chunk type */ +#define TRANSLOG_REC_TYPE 0x3F /* Mask to get record type */ + +/* compressed (relative) LSN constants */ +#define TRANSLOG_CLSN_LEN_BITS 0xC0 /* Mask to get compressed LSN + length */ +#define TRANSLOG_CLSN_MAX_LEN 5 /* Maximum length of compressed + LSN */ + +typedef my_bool(*prewrite_rec_hook) (enum translog_record_type type, + void *tcb, + struct st_translog_parts *parts); + +typedef my_bool(*inwrite_rec_hook) (enum translog_record_type type, + void *tcb, + LSN *lsn, + struct st_translog_parts *parts); + +typedef int16(*read_rec_hook) (enum translog_record_type type, + int16 read_length, uchar *read_buff, + uchar *decoded_buff); + +/* Descriptor of log record type */ +struct st_log_record_type_descriptor +{ + /* internal class of the record */ + enum record_class class; + /* length for fixed-size record, or maximum length of pseudo-fixed */ + uint16 fixed_length; + /* how much record body (belonged to headers too) read with headers */ + uint16 read_header_len; + /* HOOK for writing the record called before lock */ + prewrite_rec_hook prewrite_hook; + /* HOOK for writing the record called when LSN is known */ + inwrite_rec_hook inwrite_hook; + /* HOOK for reading headers */ + read_rec_hook read_hook; + /* + For pseudo fixed records number of compressed LSNs followed by + system header + */ + int16 compresed_LSN; +}; + +static struct st_log_record_type_descriptor + log_record_type_descriptor[LOGREC_NUMBER_OF_TYPES]= +{ + /*LOGREC_RESERVED_FOR_CHUNKS23= 0 */ + {LOGRECTYPE_NOT_ALLOWED, 0, 0, NULL, NULL, NULL, 0}, + /*LOGREC_REDO_INSERT_ROW_HEAD= 1 */ + {LOGRECTYPE_VARIABLE_LENGTH, 0, 9, NULL, NULL, NULL, 0}, + /*LOGREC_REDO_INSERT_ROW_TAIL= 2 */ + {LOGRECTYPE_VARIABLE_LENGTH, 0, 9, NULL, NULL, NULL, 0}, + /*LOGREC_REDO_INSERT_ROW_BLOB= 3 */ + {LOGRECTYPE_VARIABLE_LENGTH, 0, 8, NULL, NULL, NULL, 0}, + /*LOGREC_REDO_INSERT_ROW_BLOBS= 4 */ + {LOGRECTYPE_VARIABLE_LENGTH, 0, 10, NULL, NULL, NULL, 0}, + /*LOGREC_REDO_PURGE_ROW= 5 */ + {LOGRECTYPE_FIXEDLENGTH, 9, 9, NULL, NULL, NULL, 0}, + /*LOGREC_REDO_PURGE_BLOCKS= 6 */ + {LOGRECTYPE_FIXEDLENGTH, 10, 10, NULL, NULL, NULL, 0}, + /*LOGREC_REDO_DELETE_ROW= 7 */ + {LOGRECTYPE_FIXEDLENGTH, 16, 16, NULL, NULL, NULL, 0}, + /*LOGREC_REDO_UPDATE_ROW_HEAD= 8 */ + {LOGRECTYPE_VARIABLE_LENGTH, 0, 9, NULL, NULL, NULL, 0}, + /*LOGREC_REDO_INDEX= 9 */ + {LOGRECTYPE_VARIABLE_LENGTH, 0, 9, NULL, NULL, NULL, 0}, + /*LOGREC_REDO_UNDELETE_ROW= 10 */ + {LOGRECTYPE_FIXEDLENGTH, 16, 16, NULL, NULL, NULL, 0}, + /*LOGREC_CLR_END= 11 */ + {LOGRECTYPE_PSEUDOFIXEDLENGTH, 5, 5, NULL, NULL, NULL, 1}, + /*LOGREC_PURGE_END= 12 */ + {LOGRECTYPE_PSEUDOFIXEDLENGTH, 5, 5, NULL, NULL, NULL, 1}, + /*LOGREC_UNDO_ROW_INSERT= 13 */ + {LOGRECTYPE_PSEUDOFIXEDLENGTH, 14, 14, NULL, NULL, NULL, 1}, + /*LOGREC_UNDO_ROW_DELETE= 14 */ + {LOGRECTYPE_PSEUDOFIXEDLENGTH, 19, 19, NULL, NULL, NULL, 2}, + /*LOGREC_UNDO_ROW_UPDATE= 15 */ + {LOGRECTYPE_VARIABLE_LENGTH, 0, 14, NULL, NULL, NULL, 2}, + /*LOGREC_UNDO_KEY_INSERT= 16 */ + {LOGRECTYPE_VARIABLE_LENGTH, 0, 10, NULL, NULL, NULL, 1}, + /*LOGREC_UNDO_KEY_DELETE= 17 */ + {LOGRECTYPE_VARIABLE_LENGTH, 0, 15, NULL, NULL, NULL, 2}, + /*LOGREC_PREPARE= 18 */ + {LOGRECTYPE_VARIABLE_LENGTH, 0, 0, NULL, NULL, NULL, 0}, + /*LOGREC_PREPARE_WITH_UNDO_PURGE= 19 */ + {LOGRECTYPE_VARIABLE_LENGTH, 0, 5, NULL, NULL, NULL, 1}, + /*LOGREC_COMMIT= 20 */ + {LOGRECTYPE_FIXEDLENGTH, 0, 0, NULL, NULL, NULL, 0}, + /*LOGREC_COMMIT_WITH_UNDO_PURGE= 21 */ + {LOGRECTYPE_PSEUDOFIXEDLENGTH, 5, 5, NULL, NULL, NULL, 1}, + /*LOGREC_CHECKPOINT_PAGE= 22 */ + {LOGRECTYPE_VARIABLE_LENGTH, 0, 6, NULL, NULL, NULL, 0}, + /*LOGREC_CHECKPOINT_TRAN= 23 */ + {LOGRECTYPE_VARIABLE_LENGTH, 0, 0, NULL, NULL, NULL, 0}, + /*LOGREC_CHECKPOINT_TABL= 24 */ + {LOGRECTYPE_VARIABLE_LENGTH, 0, 8, NULL, NULL, NULL, 0}, + /*LOGREC_REDO_CREATE_TABLE= 25 */ + {LOGRECTYPE_VARIABLE_LENGTH, 0, 0, NULL, NULL, NULL, 0}, + /*LOGREC_REDO_RENAME_TABLE= 26 */ + {LOGRECTYPE_VARIABLE_LENGTH, 0, 0, NULL, NULL, NULL, 0}, + /*LOGREC_REDO_DROP_TABLE= 27 */ + {LOGRECTYPE_VARIABLE_LENGTH, 0, 0, NULL, NULL, NULL, 0}, + /*LOGREC_REDO_TRUNCATE_TABLE= 28 */ + {LOGRECTYPE_VARIABLE_LENGTH, 0, 0, NULL, NULL, NULL, 0}, + /*LOGREC_FILE_ID= 29 */ + {LOGRECTYPE_VARIABLE_LENGTH, 0, 4, NULL, NULL, NULL, 0}, + /*LOGREC_LONG_TRANSACTION_ID= 30 */ + {LOGRECTYPE_FIXEDLENGTH, 6, 6, NULL, NULL, NULL, 0}, + /*31 */ + {LOGRECTYPE_NOT_ALLOWED, 0, 0, NULL, NULL, NULL, 0}, + /*32 */ + {LOGRECTYPE_NOT_ALLOWED, 0, 0, NULL, NULL, NULL, 0}, + /*33 */ + {LOGRECTYPE_NOT_ALLOWED, 0, 0, NULL, NULL, NULL, 0}, + /*34 */ + {LOGRECTYPE_NOT_ALLOWED, 0, 0, NULL, NULL, NULL, 0}, + /*35 */ + {LOGRECTYPE_NOT_ALLOWED, 0, 0, NULL, NULL, NULL, 0}, + /*36 */ + {LOGRECTYPE_NOT_ALLOWED, 0, 0, NULL, NULL, NULL, 0}, + /*37 */ + {LOGRECTYPE_NOT_ALLOWED, 0, 0, NULL, NULL, NULL, 0}, + /*38 */ + {LOGRECTYPE_NOT_ALLOWED, 0, 0, NULL, NULL, NULL, 0}, + /*39 */ + {LOGRECTYPE_NOT_ALLOWED, 0, 0, NULL, NULL, NULL, 0}, + /*40 */ + {LOGRECTYPE_NOT_ALLOWED, 0, 0, NULL, NULL, NULL, 0}, + /*41 */ + {LOGRECTYPE_NOT_ALLOWED, 0, 0, NULL, NULL, NULL, 0}, + /*42 */ + {LOGRECTYPE_NOT_ALLOWED, 0, 0, NULL, NULL, NULL, 0}, + /*43 */ + {LOGRECTYPE_NOT_ALLOWED, 0, 0, NULL, NULL, NULL, 0}, + /*44 */ + {LOGRECTYPE_NOT_ALLOWED, 0, 0, NULL, NULL, NULL, 0}, + /*45 */ + {LOGRECTYPE_NOT_ALLOWED, 0, 0, NULL, NULL, NULL, 0}, + /*46 */ + {LOGRECTYPE_NOT_ALLOWED, 0, 0, NULL, NULL, NULL, 0}, + /*47 */ + {LOGRECTYPE_NOT_ALLOWED, 0, 0, NULL, NULL, NULL, 0}, + /*48 */ + {LOGRECTYPE_NOT_ALLOWED, 0, 0, NULL, NULL, NULL, 0}, + /*49 */ + {LOGRECTYPE_NOT_ALLOWED, 0, 0, NULL, NULL, NULL, 0}, + /*50 */ + {LOGRECTYPE_NOT_ALLOWED, 0, 0, NULL, NULL, NULL, 0}, + /*51 */ + {LOGRECTYPE_NOT_ALLOWED, 0, 0, NULL, NULL, NULL, 0}, + /*52 */ + {LOGRECTYPE_NOT_ALLOWED, 0, 0, NULL, NULL, NULL, 0}, + /*53 */ + {LOGRECTYPE_NOT_ALLOWED, 0, 0, NULL, NULL, NULL, 0}, + /*54 */ + {LOGRECTYPE_NOT_ALLOWED, 0, 0, NULL, NULL, NULL, 0}, + /*55 */ + {LOGRECTYPE_NOT_ALLOWED, 0, 0, NULL, NULL, NULL, 0}, + /*56 */ + {LOGRECTYPE_NOT_ALLOWED, 0, 0, NULL, NULL, NULL, 0}, + /*57 */ + {LOGRECTYPE_NOT_ALLOWED, 0, 0, NULL, NULL, NULL, 0}, + /*58 */ + {LOGRECTYPE_NOT_ALLOWED, 0, 0, NULL, NULL, NULL, 0}, + /*59 */ + {LOGRECTYPE_NOT_ALLOWED, 0, 0, NULL, NULL, NULL, 0}, + /*60 */ + {LOGRECTYPE_NOT_ALLOWED, 0, 0, NULL, NULL, NULL, 0}, + /*61 */ + {LOGRECTYPE_NOT_ALLOWED, 0, 0, NULL, NULL, NULL, 0}, + /*62 */ + {LOGRECTYPE_NOT_ALLOWED, 0, 0, NULL, NULL, NULL, 0}, + /*LOGREC_RESERVED_FUTURE_EXTENSION= 63 */ + {LOGRECTYPE_NOT_ALLOWED, 0, 0, NULL, NULL, NULL, 0} +}; + + +typedef struct st_translog_validator_data +{ + TRANSLOG_ADDRESS *addr; + my_bool was_recovered; +} TRANSLOG_VALIDATOR_DATA; + + +const char *maria_data_root; + + +/* + Get file name of the log by log number + + SYNOPSIS + translog_filename_by_fileno() + file_no Number of the log we want to open + path Pointer to buffer where file name will be + stored (must be FN_REFLEN bytes at least + RETURN + pointer to path +*/ + +char *translog_filename_by_fileno(uint32 file_no, char *path) +{ + char file_name[10 + 8 + 1]; + char *res; + DBUG_ENTER("translog_filename_by_fileno"); + my_sprintf(file_name, (file_name, "maria_log.%08u", file_no)); + res= fn_format(path, file_name, log_descriptor.directory, "", MYF(MY_WME)); + DBUG_PRINT("info", ("Path '%s', path: 0x%lx, res: 0x%lx", + res, (ulong) path, (ulong) res)); + DBUG_RETURN(res); +} + + +/* + Open log file with given number without cache + + SYNOPSIS + open_logfile_by_number_no_cache() + file_no Number of the log we want to open + + RETURN + 0 error + file descriptor number +*/ + +File open_logfile_by_number_no_cache(uint32 file_no) +{ + File file; + char path[FN_REFLEN]; + DBUG_ENTER("open_logfile_by_number_no_cache"); + + if ((file= my_open(translog_filename_by_fileno(file_no, path), O_CREAT | O_BINARY | /* O_DIRECT + | + */ O_RDWR, + MYF(MY_WME))) < 0) + { + UNRECOVERABLE_ERROR(("Error %d during opening file '%s'", errno, path)); + DBUG_RETURN(0); + } + DBUG_PRINT("info", ("File '%s', handler %d", path, file)); + DBUG_RETURN(file); +} + + +/* + Write log file page header in the just opened new log file + + SYNOPSIS + translog_write_file_header(); + + RETURN + 0 OK + 1 ERROR +*/ + +my_bool translog_write_file_header() +{ + ulonglong timestamp; + char page[TRANSLOG_PAGE_SIZE]; + DBUG_ENTER("translog_write_file_header"); + + /* file tag */ + strnmov(page, "MARIALOG", 8); + /* timestamp */ + timestamp= my_getsystime(); + int8store(page + 8, timestamp); + /* maria version */ + int4store(page + (8 + 8), TRANSLOG_VERSION_ID); + /* mysql version (MYSQL_VERSION_ID) */ + int4store(page + (8 + 8 + 4), log_descriptor.server_version); + /* server ID */ + int4store(page + (8 + 8 + 4 + 4), log_descriptor.server_id); + /* loghandler page size/512 */ + int2store(page + (8 + 8 + 4 + 4 + 4), TRANSLOG_PAGE_SIZE / 512); + /* file number */ + int3store(page + (8 + 8 + 4 + 4 + 4 + 2), log_descriptor.horizon.file_no); + + bzero(page + (8 + 8 + 4 + 4 + 4 + 2 + 3), + TRANSLOG_PAGE_SIZE - (8 + 8 + 4 + 4 + 4 + 2 + 3)); + + if (my_pwrite(log_descriptor.log_file_num[0], page, + TRANSLOG_PAGE_SIZE, 0, MYF(MY_WME)) != TRANSLOG_PAGE_SIZE) + DBUG_RETURN(1); + + DBUG_RETURN(0); +} + + +/* + Initialize transaction log file buffer + + SYNOPSIS + translog_buffer_init() + buffer The buffer to initialize + + RETURN + 0 - OK + 1 - Error +*/ + +my_bool translog_buffer_init(struct st_translog_buffer *buffer) +{ + DBUG_ENTER("translog_buffer_init"); + /* This buffer offset */ + buffer->last_lsn.file_no= buffer->offset.file_no= 0; + buffer->last_lsn.rec_offset= buffer->offset.rec_offset= 0; + /* This Buffer File */ + buffer->file= 0; + buffer->overlay= 0; + /* IO cache for current log */ + bzero(buffer->buffer, TRANSLOG_WRITE_BUFFER); + /* Buffer size */ + buffer->size= 0; + /* cond of thread which is waiting for buffer filling */ + buffer->waiting_filling_buffer.last_thread= 0; + /* Number of record which are in copy progress */ + buffer->copy_to_buffer_in_progress= 0; + /* list of waiting buffer ready threads */ + buffer->waiting_flush= 0; + /* lock for the buffer. Current buffer also lock the handler */ + if (pthread_mutex_init(&buffer->mutex, MY_MUTEX_INIT_FAST)) + DBUG_RETURN(1); + DBUG_PRINT("info", ("Init buffer #%u: 0x%lx", + (uint) buffer->buffer_no, (ulong) buffer)); + DBUG_RETURN(0); +} + + +/* + Close transaction log file by descriptor + + SYNOPSIS + translog_close_log_file() + file file descriptor + + RETURN + 0 OK + 1 Error +*/ + +static my_bool translog_close_log_file(File file) +{ + PAGECACHE_FILE fl= + { + file + }; + flush_pagecache_blocks(log_descriptor.pagecache, &fl, FLUSH_RELEASE); + return test(my_close(file, MYF(MY_WME))); +} + + +/* + Create and fill header of new file + + SYNOPSIS + translog_create_new_file() + + RETURN + 0 OK + 1 Error +*/ + +my_bool translog_create_new_file() +{ + int i; + + DBUG_ENTER("translog_create_new_file"); + + if (log_descriptor.log_file_num[OPENED_FILES_NUM - 1] && + translog_close_log_file(log_descriptor.log_file_num[OPENED_FILES_NUM - + 1])) + DBUG_RETURN(1); + for (i= OPENED_FILES_NUM - 1; i > 0; i--) + { + log_descriptor.log_file_num[i]= log_descriptor.log_file_num[i - 1]; + } + + if ((log_descriptor.log_file_num[0]= + open_logfile_by_number_no_cache(log_descriptor.horizon.file_no)) <= 0 || + translog_write_file_header()) + DBUG_RETURN(1); + + if (ma_control_file_write_and_force(NULL, log_descriptor.horizon.file_no, + CONTROL_FILE_UPDATE_ONLY_LOGNO)) + DBUG_RETURN(1); + + DBUG_RETURN(0); +} + + +/* + Lock the loghandler buffer + + SYNOPSIS + translog_buffer_lock() + buffer This buffer which should be locked + + RETURN + 0 - OK + 1 - Error +*/ + +#ifndef DBUG_OFF +static my_bool translog_buffer_lock(struct st_translog_buffer *buffer) +{ + int res; + DBUG_ENTER("translog_buffer_lock"); + DBUG_PRINT("enter", ("Lock buffer #%u (0x%lx): locked by:0x%lx, mutex: 0x%lx", + (uint) buffer->buffer_no, (ulong) buffer, + (ulong) buffer->locked_by, (ulong) &buffer->mutex)); + res= (pthread_mutex_lock(&buffer->mutex) != 0); +#ifndef DBUG_OFF + if (res == 0) + { + DBUG_ASSERT(buffer->locked_by == 0); + buffer->locked_by= my_thread_var; + } + else + DBUG_PRINT("error", ("Can't lock mutex 0x%lx (locked by0x%lx) errno: %d", + (ulong) &buffer->mutex, + (ulong) buffer->locked_by, res)); +#endif + DBUG_RETURN(res); +} +#else +#define translog_buffer_lock(B) \ + pthread_mutex_lock(&B->mutex); +#endif + + +/* + Unlock the loghandler buffer + + SYNOPSIS + translog_buffer_unlock() + buffer This buffer which should be unlocked + + RETURN + 0 - OK + 1 - Error +*/ + +#ifndef DBUG_OFF +static my_bool translog_buffer_unlock(struct st_translog_buffer *buffer) +{ + int res; + DBUG_ENTER("translog_buffer_unlock"); + DBUG_PRINT("enter", ("Unlock buffer... #%u (0x%lx) :locked by:0x%lx (0x%lx)," + " mutex: 0x%lx", + (uint) buffer->buffer_no, (ulong) buffer, + (ulong) buffer->locked_by, (ulong) my_thread_var, + (ulong) &buffer->mutex)); + DBUG_ASSERT(buffer->locked_by == my_thread_var); + + buffer->locked_by= 0; + res= (pthread_mutex_unlock(&buffer->mutex) != 0); + DBUG_PRINT("enter", ("Unlocked buffer... #%u: 0x%lx, mutex: 0x%lx", + (uint) buffer->buffer_no, (ulong) buffer, + (ulong) &buffer->mutex)); + DBUG_RETURN(res); +} +#else +#define translog_buffer_unlock(B) \ + pthread_mutex_unlock(&B->mutex); +#endif + + +/* + Write page header. + + SYNOPSIS + translog_new_page_header() + horizon Where to write the page + cursor Where to write the page + + NOTE + - space for page header should be checked before +*/ + +static void translog_new_page_header(TRANSLOG_ADDRESS *horizon, + struct st_buffer_cursor *cursor) +{ + uchar *ptr; + + DBUG_ENTER("translog_new_page_header"); + DBUG_ASSERT(cursor->ptr !=NULL); + + cursor->protected= 0; + + ptr= cursor->ptr; + /* Page number */ + int3store(ptr, horizon->rec_offset / TRANSLOG_PAGE_SIZE); + ptr +=3; + /* File number */ + int3store(ptr, horizon->file_no); + ptr +=3; + *(ptr ++)= (uchar) log_descriptor.flags; + if (log_descriptor.flags & TRANSLOG_PAGE_CRC) + { +#ifndef DBUG_OFF + DBUG_PRINT("info", ("write 0x11223344 CRC to (%lu,0x%lx)", + (ulong) horizon->file_no, (ulong) horizon->rec_offset)); + int4store(ptr, 0x11223344); +#endif + ptr +=4; /* CRC will be put when page + will be finished */ + } + if (log_descriptor.flags & TRANSLOG_SECTOR_PROTECTION) + { + time_t tm; + int2store(ptr, time(&tm) & 0xFFFF); + ptr +=(TRANSLOG_PAGE_SIZE / 512) * 2; + } + { + uint len= (ptr -cursor->ptr); + horizon->rec_offset+= len; + cursor->current_page_size= len; + if (!cursor->chaser) + cursor->buffer->size+= len; + } + cursor->ptr= ptr; + DBUG_PRINT("info", ("NewP buffer #%u: 0x%lx, chaser: %d, Size: %lu (%lu)", + (uint) cursor->buffer->buffer_no, (ulong) cursor->buffer, + cursor->chaser, (ulong) cursor->buffer->size, + (ulong) (cursor->ptr -cursor->buffer->buffer))); + DBUG_ASSERT(cursor->chaser || + ((ulong) (cursor->ptr -cursor->buffer->buffer) == + cursor->buffer->size)); + DBUG_ASSERT(cursor->buffer->buffer_no == cursor->buffer_no); + DBUG_ASSERT(cursor->current_page_size <= TRANSLOG_PAGE_SIZE); + DBUG_VOID_RETURN; +} + + +/* + Put sector protection on the page image + + SYNOPSIS + translog_put_sector_protection() + page reference on the page content + cursor cursor of the buffer +*/ + +static void translog_put_sector_protection(uchar *page, + struct st_buffer_cursor *cursor) +{ + uchar *table= page + log_descriptor.page_overhead - + (TRANSLOG_PAGE_SIZE / 512) * 2; + uint16 value= uint2korr(table) + cursor->write_counter; + uint16 last_protected_sector= (cursor->previous_offset - 1) / 512; + uint16 start_sector= cursor->previous_offset / 512; + uint i, offset; + + DBUG_ENTER("translog_put_sector_protection"); + if (start_sector == 0) + start_sector= 1; + + DBUG_PRINT("enter", ("Write counter %u, value %u, offset %u, " + "last protected %u, start sector %u", + (uint) cursor->write_counter, + (uint) value, + (uint) cursor->previous_offset, + (uint) last_protected_sector, (uint) start_sector)); + if (last_protected_sector == start_sector) + { + i= last_protected_sector * 2; + offset= last_protected_sector * 512; + /* restore data, because we modified sector which was protected */ + if (offset < cursor->previous_offset) + page[offset]= table[i]; + offset++; + if (offset < cursor->previous_offset) + page[offset]= table[i + 1]; + } + for (i= start_sector * 2, offset= start_sector * 512; + i < (TRANSLOG_PAGE_SIZE / 512) * 2; (i+= 2), (offset+= 512)) + { + DBUG_PRINT("info", ("sector %u, offset %u, data 0x%x%x", + i / 2, offset, (uint) page[offset], + (uint) page[offset + 1])); + table[i]= page[offset]; + table[i + 1]= page[offset + 1]; + /**((uint16 *)(table + i))= *((uint16* )(page + offset));*/ + int2store(page + offset, value); + DBUG_PRINT("info", ("sector %u, offset %u, data 0x%x%x", + i / 2, offset, (uint) page[offset], + (uint) page[offset + 1])); + } + DBUG_VOID_RETURN; +} + + +/* + Calculate adler CRC of given area + + SYNOPSIS + translog_adler_crc() + area Pointer of the area beginning + length The Area length + + RETURN + Adler CRC32 +*/ + +uint32 translog_adler_crc(uchar *area, uint length) +{ + uint32 a= 1, b= 0; +#define MOD_ADLER 65521 + + while (length) + { + uint tlen= length > 5550 ? 5550 : length; + length-= tlen; + do + { + a+= *area++; + b+= a; + } while (--tlen); + a= (a & 0xffff) + (a >> 16) * (65536 - MOD_ADLER); + b= (b & 0xffff) + (b >> 16) * (65536 - MOD_ADLER); + } + /* It can be shown that a <= 0x1013a here, so a single subtract will do. */ + if (a >= MOD_ADLER) + a-= MOD_ADLER; + /* It can be shown that b can reach 0xffef1 here. */ + b= (b & 0xffff) + (b >> 16) * (65536 - MOD_ADLER); + if (b >= MOD_ADLER) + b-= MOD_ADLER; + return (b << 16) | a; +} + + +/* + Finish current page with zeros + + SYNOPSIS + translog_finish_page() + horizon \ horizon & buffer pointers + cursor / +*/ + +static void translog_finish_page(TRANSLOG_ADDRESS *horizon, + struct st_buffer_cursor *cursor) +{ + uint16 left= TRANSLOG_PAGE_SIZE - cursor->current_page_size; + uchar *page= cursor->ptr -cursor->current_page_size; + DBUG_ENTER("translog_finish_page"); + + DBUG_PRINT("enter", ("Buffer #%u 0x%lx, " + "Buffer addr (%lu,0x%lx), " + "Page addr: (%lu,0x%lx), " + "size %lu (%lu), Pg: %u, left: %u", + (uint) cursor->buffer_no, (ulong) cursor->buffer, + (ulong) cursor->buffer->offset.file_no, + (ulong) cursor->buffer->offset.rec_offset, + (ulong) horizon->file_no, + (ulong) (horizon->rec_offset - + cursor->current_page_size), + (ulong) cursor->buffer->size, + (ulong) (cursor->ptr -cursor->buffer->buffer), + (uint) cursor->current_page_size, (uint) left)); + DBUG_ASSERT(cursor->ptr !=NULL); + DBUG_ASSERT((cursor->ptr -cursor->buffer->buffer) %TRANSLOG_PAGE_SIZE == + cursor->current_page_size % TRANSLOG_PAGE_SIZE); + DBUG_ASSERT(horizon->file_no == cursor->buffer->offset.file_no); + DBUG_ASSERT(cursor->buffer->offset.rec_offset + + (cursor->ptr -cursor->buffer->buffer) == horizon->rec_offset); + if (cursor->protected) + { + DBUG_PRINT("info", ("Already protected and finished")); + DBUG_VOID_RETURN; + } + if (left != TRANSLOG_PAGE_SIZE && left != 0) + { + DBUG_PRINT("info", ("left %u", (uint) left)); + bzero(cursor->ptr, left); + cursor->ptr +=left; + horizon->rec_offset+= left; + if (!cursor->chaser) + cursor->buffer->size+= left; + cursor->current_page_size= 0; + DBUG_PRINT("info", ("Finish Page buffer #%u: 0x%lx, " + "chaser: %d, Size: %lu (%lu)", + (uint) cursor->buffer->buffer_no, + (ulong) cursor->buffer, cursor->chaser, + (ulong) cursor->buffer->size, + (ulong) (cursor->ptr -cursor->buffer->buffer))); + DBUG_ASSERT(cursor->chaser + || ((ulong) (cursor->ptr -cursor->buffer->buffer) == + cursor->buffer->size)); + DBUG_ASSERT(cursor->buffer->buffer_no == cursor->buffer_no); + } + if (log_descriptor.flags & TRANSLOG_SECTOR_PROTECTION) + { + translog_put_sector_protection(page, cursor); + DBUG_PRINT("info", ("drop write_counter")); + cursor->write_counter= 0; + cursor->previous_offset= 0; + } + if (log_descriptor.flags & TRANSLOG_PAGE_CRC) + { + uint32 crc= translog_adler_crc(page + log_descriptor.page_overhead, + TRANSLOG_PAGE_SIZE - + log_descriptor.page_overhead); + DBUG_PRINT("info", ("CRC: 0x%lx", (ulong) crc)); + int4store(page + 3 + 3 + 1, crc); + } + cursor->protected= 1; + DBUG_VOID_RETURN; +} + + +/* + Wait until all thread finish filling this buffer + + SYNOPSIS + translog_wait_for_writers() + buffer This buffer should be check + + NOTE + This buffer should be locked +*/ +static void translog_wait_for_writers(struct st_translog_buffer *buffer) +{ + struct st_my_thread_var *thread; + DBUG_ENTER("translog_wait_for_writers"); + DBUG_PRINT("enter", ("Buffer #%u 0x%lx, copies in progress: %u", + (uint) buffer->buffer_no, (ulong) buffer, + (int) buffer->copy_to_buffer_in_progress)); + + if (!buffer->copy_to_buffer_in_progress) + DBUG_VOID_RETURN; + + thread= my_thread_var; + + DBUG_ASSERT(buffer->file != 0); + + do + { + DBUG_PRINT("info", ("wait for writers... , thread 0x%lx, " + "buffer #%u 0x%lx, locked by 0x%lx (0x%lx), " + "mutex: 0x%lx", + thread, (uint) buffer->buffer_no, (ulong) buffer, + (ulong) buffer->locked_by, (ulong) thread, + (ulong) &buffer->mutex)); +#ifndef DBUG_OFF + DBUG_ASSERT(buffer->locked_by == thread); + buffer->locked_by= 0; +#endif + wqueue_add_and_wait(&buffer->waiting_filling_buffer, thread, + &buffer->mutex); + DBUG_PRINT("info", ("wait for writers done, thread 0x%lx, " + "buffer #%u 0x%lx, locked by 0x%lx (0x%lx), " + "mutex: 0x%lx", + thread, (uint) buffer->buffer_no, (ulong) buffer, + (ulong) buffer->locked_by, (ulong) thread, + (ulong) &buffer->mutex)); +#ifndef DBUG_OFF + DBUG_ASSERT(buffer->locked_by == 0); + buffer->locked_by= thread; +#endif + } while (buffer->copy_to_buffer_in_progress != 0); + + DBUG_VOID_RETURN; +} + + +/* + + Wait for this buffer become free + + SYNOPSIS + translog_wait_for_buffer_free() + buffer The buffer to initialize + + NOTE + - this buffer should be locked +*/ + +static void translog_wait_for_buffer_free(struct st_translog_buffer *buffer) +{ + struct st_my_thread_var *thread= my_thread_var; + DBUG_ENTER("translog_wait_for_buffer_free"); + DBUG_PRINT("enter", ("Buffer #%u 0x%lx, copies in progress: %u size 0x%lu", + (uint) buffer->buffer_no, (ulong) buffer, + (int) buffer->copy_to_buffer_in_progress, + (ulong) buffer->size)); + + translog_wait_for_writers(buffer); + + if (!buffer->file) + DBUG_VOID_RETURN; + + thread= my_thread_var; + + do + { + DBUG_PRINT("info", ("wait for writers... , thread 0x%lx, " + "buffer #%u 0x%lx, locked by 0x%lx (0x%lx), " + "mutex: 0x%lx", + thread, (uint) buffer->buffer_no, (ulong) buffer, + (ulong) buffer->locked_by, (ulong) thread, + (ulong) &buffer->mutex)); +#ifndef DBUG_OFF + DBUG_ASSERT(buffer->locked_by == thread); + buffer->locked_by= 0; +#endif + wqueue_add_and_wait(&buffer->waiting_filling_buffer, thread, + &buffer->mutex); + DBUG_PRINT("info", ("wait for writers done, thread 0x%lx, " + "buffer #%u 0x%lx, locked by 0x%lx (0x%lx), " + "mutex: 0x%lx", + thread, (uint) buffer->buffer_no, (ulong) buffer, + (ulong) buffer->locked_by, (ulong) thread, + (ulong) &buffer->mutex)); +#ifndef DBUG_OFF + DBUG_ASSERT(buffer->locked_by == 0); + buffer->locked_by= thread; +#endif + } while (buffer->copy_to_buffer_in_progress != 0); + DBUG_VOID_RETURN; +} + + +/* + Set cursor on the buffer beginning + + SYNOPSIS + translog_cursor_init() + buffer The buffer + cursor It's cursor + buffer_no Number of buffer +*/ + +static void translog_cursor_init(struct st_buffer_cursor *cursor, + struct st_translog_buffer *buffer, + uint8 buffer_no) +{ + DBUG_ENTER("translog_cursor_init"); + cursor->ptr= buffer->buffer; + cursor->buffer= buffer; + cursor->buffer_no= buffer_no; + cursor->current_page_size= 0; + cursor->chaser= (cursor != &log_descriptor.bc); + DBUG_PRINT("info", ("drop write_counter")); + cursor->write_counter= 0; + cursor->previous_offset= 0; + cursor->protected= 0; + DBUG_VOID_RETURN; +} + + +/* + Initialize buffer for current file + + SYNOPSIS + translog_start_buffer() + buffer The buffer + cursor It's cursor + buffer_no Number of buffer +*/ +static void translog_start_buffer(struct st_translog_buffer *buffer, + struct st_buffer_cursor *cursor, + uint8 buffer_no) +{ + DBUG_ENTER("translog_start_buffer"); + DBUG_PRINT("enter", + ("Assign buffer #%u (0x%lx) to file %u, offset 0x%lx(%lu)", + (uint) buffer->buffer_no, (ulong) buffer, + (uint) log_descriptor.log_file_num[0], + (ulong) log_descriptor.horizon.rec_offset, + (ulong) log_descriptor.horizon.rec_offset)); + DBUG_ASSERT(buffer_no == buffer->buffer_no); + buffer->last_lsn.file_no= 0; + buffer->last_lsn.rec_offset= 0; + buffer->offset= log_descriptor.horizon; + buffer->file= log_descriptor.log_file_num[0]; + buffer->overlay= 0; + buffer->size= 0; + translog_cursor_init(cursor, buffer, buffer_no); + DBUG_PRINT("info", ("init cursor #%u: 0x%lx, chaser: %d, Size: %lu (%lu)", + (uint) cursor->buffer->buffer_no, (ulong) cursor->buffer, + cursor->chaser, (ulong) cursor->buffer->size, + (ulong) (cursor->ptr -cursor->buffer->buffer))); + DBUG_ASSERT(cursor->chaser || + ((ulong) (cursor->ptr -cursor->buffer->buffer) == + cursor->buffer->size)); + DBUG_ASSERT(cursor->buffer->buffer_no == cursor->buffer_no); + DBUG_VOID_RETURN; +} + + +/* + Switch to the next buffer in a chain + + SYNOPSIS + translog_buffer_next() + horizon \ Pointers on current position in file and buffer + cursor / + next_file Also start new file + + NOTE: + - loghandler should be locked + - after return new and old buffer still are locked + + RETURN + 0 - OK + 1 - Error +*/ + +static my_bool translog_buffer_next(TRANSLOG_ADDRESS *horizon, + struct st_buffer_cursor *cursor, + my_bool new_file) +{ + uint8 old_buffer_no= cursor->buffer_no; + uint8 new_buffer_no= (old_buffer_no + 1) % TRANSLOG_BUFFERS_NO; + struct st_translog_buffer *new_buffer= log_descriptor.buffers + new_buffer_no; + my_bool chasing= cursor->chaser; + DBUG_ENTER("translog_buffer_next"); + + DBUG_PRINT("info", ("horizon (%u,0x%lx), chasing: %d", + (uint) log_descriptor.horizon.file_no, + (ulong) log_descriptor.horizon.rec_offset, chasing)); + + DBUG_ASSERT(cmp_translog_addr(log_descriptor.horizon, *horizon) >= 0); + + translog_finish_page(horizon, cursor); + + if (!chasing) + { + translog_buffer_lock(new_buffer); + translog_wait_for_buffer_free(new_buffer); + } +#ifndef DBUG_OFF + else + DBUG_ASSERT(new_buffer->file != 0); +#endif + if (new_file) + { + horizon->file_no++; + horizon->rec_offset= TRANSLOG_PAGE_SIZE; /* header page */ + if (!chasing && translog_create_new_file()) + { + DBUG_RETURN(1); + } + } + + /* prepare next page */ + if (chasing) + translog_cursor_init(cursor, new_buffer, new_buffer_no); + else + translog_start_buffer(new_buffer, cursor, new_buffer_no); + translog_new_page_header(horizon, cursor); + DBUG_RETURN(0); +} + + +/* + Set max LSN send to file + + SYNOPSIS + translog_set_sent_to_file() + lsn LSN to assign +*/ + +static void translog_set_sent_to_file(LSN *lsn) +{ + DBUG_ENTER("translog_set_sent_to_file"); + pthread_mutex_lock(&log_descriptor.sent_to_file_lock); + DBUG_ASSERT(cmp_translog_addr(*lsn, log_descriptor.sent_to_file) >= 0); + log_descriptor.sent_to_file= *lsn; + pthread_mutex_unlock(&log_descriptor.sent_to_file_lock); + DBUG_VOID_RETURN; +} + + +/* + Get max LSN send to file + + SYNOPSIS + translog_get_sent_to_file() + lsn LSN to value +*/ + +static void translog_get_sent_to_file(LSN *lsn) +{ + DBUG_ENTER("translog_get_sent_to_file"); + pthread_mutex_lock(&log_descriptor.sent_to_file_lock); + *lsn= log_descriptor.sent_to_file; + pthread_mutex_unlock(&log_descriptor.sent_to_file_lock); + DBUG_VOID_RETURN; +} + + +/* + Get first chunk address on the given page + + SYNOPSIS + translog_get_first_chunk_offset() + page The page where to find first chunk + + RETURN + first chunk offset + 0 - Error +*/ + +static my_bool translog_get_first_chunk_offset(uchar *page) +{ + uint16 page_header= 7; + DBUG_ENTER("translog_get_first_chunk_offset"); + + if (page[6] & TRANSLOG_PAGE_CRC) + { + page_header+= 4; + } + if (page[6] & TRANSLOG_SECTOR_PROTECTION) + { + page_header+= (TRANSLOG_PAGE_SIZE / 512) * 2; + } + DBUG_RETURN(page_header); +} + + +/* + Write coded length of record + + SYNOPSIS + translog_write_variable_record_1group_code_len + dst Destination buffer pointer + length Length which should be coded + header_len Calculated total header length +*/ + +static void +translog_write_variable_record_1group_code_len(uchar *dst, + translog_size_t length, + uint16 header_len) +{ + switch (header_len) { + case 6: /* (5 + 1) */ + DBUG_ASSERT(length <= 250); + *dst= (uint8) length; + return; + case 8: /* (5 + 3) */ + DBUG_ASSERT(length <= 0xFFFF); + *dst= 251; + int2store(dst + 1, length); + return; + case 9: /* (5 + 4) */ + DBUG_ASSERT(length <= 0xFFFFFF); + *dst= 252; + int3store(dst + 1, length); + return; + case 10: /* (5 + 5) */ + *dst= 253; + int4store(dst + 1, length); + return; + default: + DBUG_ASSERT(0); + } + return; +} + + +/* + Decode record data length and advance given pointer to the next field + + SYNOPSIS + translog_variable_record_1group_decode_len() + src The pointer to the pointer to the length beginning + + RETURN + decoded length +*/ + +static translog_size_t translog_variable_record_1group_decode_len(uchar **src) +{ + uint8 first= (uint8) (**src); + switch (first) { + case 251: + *src+= 3; + return (uint2korr((*src) - 2)); + case 252: + *src+= 4; + return (uint3korr((*src) - 3)); + case 253: + *src+= 5; + return (uint4korr((*src) - 4)); + case 254: + case 255: + DBUG_ASSERT(0); /* reserved for future use */ + return (0); + default: + (*src)++; + return (first); + } +} + + +/* + Get total length of this chunk (not only body) + + SYNOPSIS + translog_get_total_chunk_length() + page The page where chunk placed + offset Offset of the chunk on this place + + RETURN + total length of the chunk + 0 - Error +*/ + +uint16 translog_get_total_chunk_length(uchar *page, uint16 offset) +{ + DBUG_ENTER("translog_get_total_chunk_length"); + switch (page[offset] & TRANSLOG_CHUNK_TYPE) { + case TRANSLOG_CHUNK_LSN: /* 0 chunk referred as LSN + (head or tail) */ + { + translog_size_t rec_len; + uchar *start= page + offset; + uchar *ptr= start + 1 + 2; + uint16 chunk_len, header_len, page_rest; + DBUG_PRINT("info", ("TRANSLOG_CHUNK_LSN")); + rec_len= translog_variable_record_1group_decode_len(&ptr); + chunk_len= uint2korr(ptr); + header_len= (ptr -start) +2; + DBUG_PRINT("info", ("rec len: %lu, chunk len: %u, header len: %u", + (ulong) rec_len, (uint) chunk_len, (uint) header_len)); + if (chunk_len) + { + DBUG_PRINT("info", ("chunk len: %u + %u = %u", + (uint) header_len, (uint) chunk_len, + (uint) (chunk_len + header_len))); + DBUG_RETURN(chunk_len + header_len); + } + page_rest= TRANSLOG_PAGE_SIZE - offset; + DBUG_PRINT("info", ("page_rest %u", (uint) page_rest)); + if (rec_len + header_len < page_rest) + DBUG_RETURN(rec_len + header_len); + DBUG_RETURN(page_rest); + break; + } + case TRANSLOG_CHUNK_FIXED: /* 1 (pseudo)fixed record (also + LSN) */ + { + DBUG_PRINT("info", ("TRANSLOG_CHUNK_FIXED")); + uint type= page[offset] & TRANSLOG_REC_TYPE; + DBUG_ASSERT(log_record_type_descriptor[type].class == + LOGRECTYPE_FIXEDLENGTH || + log_record_type_descriptor[type].class == + LOGRECTYPE_PSEUDOFIXEDLENGTH); + if (log_record_type_descriptor[type].class == LOGRECTYPE_FIXEDLENGTH) + { + DBUG_PRINT("info", + ("Fixed length: %u", + (uint) (log_record_type_descriptor[type].fixed_length + 3))); + DBUG_RETURN(log_record_type_descriptor[type].fixed_length + 3); + } + { + uchar *ptr= page + offset + 3; /* first compressed LSN */ + int i= 0; + uint length= log_record_type_descriptor[type].fixed_length + 3; + for (; i < log_record_type_descriptor[type].compresed_LSN; i++) + { + /* first 2 bits is length - 2 */ + uint len= ((((uint8) (*ptr)) & TRANSLOG_CLSN_LEN_BITS) >> 6) + 2; + ptr+= len; + length-= (TRANSLOG_CLSN_MAX_LEN - len); /* subtract economized + bytes */ + } + DBUG_PRINT("info", ("Pseudo-fixed length: %u", length)); + DBUG_RETURN(length); + } + break; + } + case TRANSLOG_CHUNK_NOHDR: /* 2 no header chunk (till page + end) */ + DBUG_PRINT("info", ("TRANSLOG_CHUNK_NOHDR, length: %u", + (uint) (TRANSLOG_PAGE_SIZE - offset))); + DBUG_RETURN(TRANSLOG_PAGE_SIZE - offset); + break; + case TRANSLOG_CHUNK_LNGTH: /* 3 chunk with chunk length */ + DBUG_PRINT("info", ("TRANSLOG_CHUNK_LNGTH")); + DBUG_ASSERT(TRANSLOG_PAGE_SIZE - offset >= 3); + DBUG_PRINT("info", ("Length %u", uint2korr(page + offset + 1) + 3)); + DBUG_RETURN(uint2korr(page + offset + 1) + 3); + break; + default: + DBUG_ASSERT(0); + } +} + + +/* + Flush given buffer + + SYNOPSIS + translog_buffer_flush() + buffer This buffer should be flushed + + RETURN + 0 - OK + 1 - Error +*/ + +static my_bool translog_buffer_flush(struct st_translog_buffer *buffer) +{ + uint32 i; + DBUG_ENTER("translog_buffer_flush"); + DBUG_PRINT("enter", + ("Buffer #%u 0x%lx: locked by 0x%lx (0x%lx), " + "file: %u, offset (%lu,0x%lx), size %lu", + (uint) buffer->buffer_no, (ulong) buffer, + (ulong) buffer->locked_by, my_thread_var, + (uint) buffer->file, + (ulong) buffer->offset.file_no, (ulong) buffer->offset.rec_offset, + (ulong) buffer->size)); + + DBUG_ASSERT(buffer->locked_by == my_thread_var); + DBUG_ASSERT(buffer->file != 0); + + translog_wait_for_writers(buffer); + if (buffer->overlay && buffer->overlay->file) + { + struct st_translog_buffer *overlay= buffer->overlay; + translog_buffer_unlock(buffer); + translog_buffer_lock(overlay); + translog_wait_for_buffer_free(overlay); + translog_buffer_unlock(overlay); + translog_buffer_lock(buffer); + } + + for (i= 0; i < buffer->size; i+= TRANSLOG_PAGE_SIZE) + { + PAGECACHE_FILE file= + { + buffer->file + }; + if (pagecache_write(log_descriptor.pagecache, + &file, + (buffer->offset.rec_offset + i) / TRANSLOG_PAGE_SIZE, + 3, + buffer->buffer + i, + PAGECACHE_PLAIN_PAGE, + PAGECACHE_LOCK_LEFT_UNLOCKED, + PAGECACHE_PIN_LEFT_UNPINNED, PAGECACHE_WRITE_DONE, 0)) + { + UNRECOVERABLE_ERROR(("Cant't write page (%lu,0x%lx) to pagecacte", + (ulong) buffer->file, + (ulong) (buffer->offset.rec_offset + i))); + } + } + if (my_pwrite(buffer->file, (char*) buffer->buffer, + buffer->size, buffer->offset.rec_offset, + MYF(MY_WME)) != buffer->size) + { + UNRECOVERABLE_ERROR(("Cant't buffer (%lu,0x%lx) size %lu to the disk (%d)", + (ulong) buffer->file, + (ulong) buffer->offset.rec_offset, + (ulong) buffer->size, errno)); + DBUG_RETURN(1); + } + if (buffer->last_lsn.rec_offset != 0) /* if buffer->last_lsn is set */ + translog_set_sent_to_file(&buffer->last_lsn); + /* Free buffer */ + buffer->file= 0; + buffer->overlay= 0; + if (buffer->waiting_filling_buffer.last_thread != NULL) + { + wqueue_release_queue(&buffer->waiting_filling_buffer); + } + DBUG_RETURN(0); +} + + +/* + Recover page with sector protection (wipe out failed chunks) + + SYNOPSYS + translog_recover_page_up_to_sector() + page reference on the page + offset offset of failed sector + + RETURN + 0 - OK + 1 - Error +*/ + +static my_bool translog_recover_page_up_to_sector(uchar *page, uint16 offset) +{ + uint16 chunk_offset= translog_get_first_chunk_offset(page), valid_chunk_end; + DBUG_ENTER("translog_recover_page_up_to_sector"); + DBUG_PRINT("enter", ("offset %u, first chunk %u", + (uint) offset, (uint) chunk_offset)); + + if (chunk_offset == 0) + DBUG_RETURN(1); + + while (page[chunk_offset] != '\0' && chunk_offset < offset) + { + uint16 chunk_length; + if ((chunk_length= + translog_get_total_chunk_length(page, chunk_offset)) == 0) + { + UNRECOVERABLE_ERROR(("cant get chunk length (offset %u)", + (uint) chunk_offset)); + DBUG_RETURN(1); + } + DBUG_PRINT("info", ("chunk: offset: %u, length %u", + (uint) chunk_offset, (uint) chunk_length)); + if (((ulong) chunk_offset) + ((ulong) chunk_length) > TRANSLOG_PAGE_SIZE) + { + UNRECOVERABLE_ERROR(("demaged chunk (offset %u) in trusted area", + (uint) chunk_offset)); + DBUG_RETURN(1); + } + chunk_offset+= chunk_length; + } + + valid_chunk_end= chunk_offset; + /*end of trusted area - sector parsing */ + while (page[chunk_offset] != '\0') + { + uint16 chunk_length; + if ((chunk_length= + translog_get_total_chunk_length(page, chunk_offset)) == 0) + { + break; + } + DBUG_PRINT("info", ("chunk: offset: %u, length %u", + (uint) chunk_offset, (uint) chunk_length)); + if (((ulong) chunk_offset) + ((ulong) chunk_length) > (uint) (offset + 512)) + { + break; + } + chunk_offset+= chunk_length; + valid_chunk_end= chunk_offset; + } + DBUG_PRINT("info", ("valid chunk end offset: %u", (uint) valid_chunk_end)); + + bzero(page + valid_chunk_end, TRANSLOG_PAGE_SIZE - valid_chunk_end); + + DBUG_RETURN(0); +} + + +/* + Log page validator + + SYNOPSIS + translog_page_validator() + page_addr The page to check + data data, need for validation (address in this case) + + RETURN + 0 - OK + 1 - Error +*/ +static my_bool translog_page_validator(byte *page_addr, gptr data) +{ + uint8 flags; + uchar *page= (uchar*) page_addr; + DBUG_ENTER("translog_page_validator"); + TRANSLOG_ADDRESS *addr= ((TRANSLOG_VALIDATOR_DATA*) data)->addr; + + ((TRANSLOG_VALIDATOR_DATA*) data)->was_recovered= 0; + + if (uint3korr(page) != addr->rec_offset / TRANSLOG_PAGE_SIZE || + uint3korr(page + 3) != addr->file_no) + { + UNRECOVERABLE_ERROR(("Page (%lu,0x%lx): " + "page address written in the page is incorrect :" + "File %lu instead of %lu or page %lu instead of %lu", + (ulong) addr->file_no, (ulong) addr->rec_offset, + (ulong) uint3korr(page + 3), (ulong) addr->file_no, + (ulong) uint3korr(page), + (ulong) addr->rec_offset / TRANSLOG_PAGE_SIZE)); + DBUG_RETURN(1); + } + flags= page[3 + 3]; + if (flags & ~(TRANSLOG_PAGE_CRC | TRANSLOG_SECTOR_PROTECTION | + TRANSLOG_RECORD_CRC)) + { + UNRECOVERABLE_ERROR(("Page (%lu,0x%lx): " + "Garbage in the page flags field detected : %x", + (ulong) addr->file_no, (ulong) addr->rec_offset, + (uint) flags)); + DBUG_RETURN(1); + } + if (flags & TRANSLOG_PAGE_CRC) + { + uint32 crc= translog_adler_crc(page + log_descriptor.page_overhead, + TRANSLOG_PAGE_SIZE - + log_descriptor.page_overhead); + if (crc != uint4korr(page + 3 + 3 + 1)) + { + UNRECOVERABLE_ERROR(("Page (%lu,0x%lx): " + "CRC mismatch: calculated: %lx on the page %lx", + (ulong) addr->file_no, (ulong) addr->rec_offset, + (ulong) crc, (ulong) uint4korr(page + 3 + 3 + 1))); + DBUG_RETURN(1); + } + } + if (flags & TRANSLOG_SECTOR_PROTECTION) + { + uint i, offset; + uchar *table= (page + 3 + 3 + 1 + ((flags & TRANSLOG_PAGE_CRC) ? 4 : 0)); + uint16 current= uint2korr(table); + for (i= 2, offset= 512; + i < (TRANSLOG_PAGE_SIZE / 512) * 2; i+= 2, offset+= 512) + { + /* + TODO: add cunk counting for "suspecting" sectors (difference is + more that 1-2) + */ + uint16 test= uint2korr(page + offset); + DBUG_PRINT("info", ("sector #%u offset %u current %lx " + "read 0x%lx stored 0x%x%x", + i / 2, offset, current, + (uint) uint2korr(page + offset), (uint) table[i], + (uint) table[i + 1])); + if (test < current) + { + if (0xFFFFLL - current + test > 512 / 3) + { + /* it is not overflow */ + if (translog_recover_page_up_to_sector(page, offset)) + DBUG_RETURN(1); + ((TRANSLOG_VALIDATOR_DATA*) data)->was_recovered= 1; + DBUG_RETURN(0); + } + } + else if (test - current > 512 / 3) + { + if (translog_recover_page_up_to_sector(page, offset)) + DBUG_RETURN(1); + ((TRANSLOG_VALIDATOR_DATA*) data)->was_recovered= 1; + DBUG_RETURN(0); + } + + /* Return value on the page */ + page[offset]= table[i]; + page[offset + 1]= table[i + 1]; + /**((uint16*)page + offset)= *((uint16*)(table + i));*/ + + current= test; + DBUG_PRINT("info", ("sector #%u offset %u current %lx " + "read 0x%lx stored 0x%x%x", + i / 2, offset, current, + (uint) uint2korr(page + offset), (uint) table[i], + (uint) table[i + 1])); + } + } + DBUG_RETURN(0); +} + +/* + Get log page by file number and offset of the beginning of the page + + SYNOPSIS + translog_get_page() + data validator data, which contains the page address + buffer buffer for page placing + (might not be used in some cache implementations) + + RETURN + pointer to the page cache which should be used to read this page + NULL - Error +*/ + +uchar *translog_get_page(TRANSLOG_VALIDATOR_DATA *data, uchar *buffer) +{ + uint cache_index; + DBUG_ENTER("translog_get_page"); + DBUG_PRINT("enter", ("File %lu, Offset %lu(0x%lx)", + (ulong) data->addr->file_no, + (ulong) data->addr->rec_offset, + (ulong) data->addr->rec_offset)); + + /* it is really page address */ + DBUG_ASSERT(data->addr->rec_offset % TRANSLOG_PAGE_SIZE == 0); + + if ((cache_index= log_descriptor.horizon.file_no - data->addr->file_no) < + OPENED_FILES_NUM) + { + PAGECACHE_FILE file; + /* file in the cache */ + if (log_descriptor.log_file_num[cache_index] == 0) + { + if ((log_descriptor.log_file_num[cache_index]= + open_logfile_by_number_no_cache(data->addr->file_no)) == 0) + { + DBUG_RETURN(NULL); + } + } + file.file= log_descriptor.log_file_num[cache_index]; + + buffer= (uchar*) + pagecache_valid_read(log_descriptor.pagecache, &file, + data->addr->rec_offset / TRANSLOG_PAGE_SIZE, + 3, (char*) buffer, + PAGECACHE_PLAIN_PAGE, + PAGECACHE_LOCK_LEFT_UNLOCKED, 0, + &translog_page_validator, (gptr) data); + } + else + { + File file= open_logfile_by_number_no_cache(data->addr->file_no); + if (my_pread(file, (char*) buffer, TRANSLOG_PAGE_SIZE, + data->addr->rec_offset, MYF(MY_FNABP | MY_WME))) + buffer= NULL; + else if (translog_page_validator((byte*) buffer, (gptr) data)) + buffer= NULL; + my_close(file, MYF(MY_WME)); + } + DBUG_RETURN(buffer); +} + + +/* + Finds last page of the given log file + + SYNOPSIS + translog_get_last_page_addr() + addr address structure to fill with data, which contain + file number of the log file + last_page_ok assigned 1 if last page was OK + + RETURN + 0 - OK + 1 - Error +*/ + +static my_bool translog_get_last_page_addr(TRANSLOG_ADDRESS *addr, + my_bool *last_page_ok) +{ + MY_STAT stat_buff, *stat; + char path[FN_REFLEN]; + DBUG_ENTER("translog_get_last_page_addr"); + + if ((stat= my_stat (translog_filename_by_fileno(addr->file_no, + path), + &stat_buff, MYF(MY_WME))) == NULL) + DBUG_RETURN(1); + DBUG_PRINT("info", ("File size %lu", (ulong) stat->st_size)); + if (stat->st_size > TRANSLOG_PAGE_SIZE) + { + addr->rec_offset= (((stat->st_size / TRANSLOG_PAGE_SIZE) - 1) * + TRANSLOG_PAGE_SIZE); + *last_page_ok= (stat->st_size == addr->rec_offset + TRANSLOG_PAGE_SIZE); + } + else + { + *last_page_ok= 0; + addr->rec_offset= 0; + } + DBUG_PRINT("info", ("Last page: 0x%lx, ok %d", (ulong) addr->rec_offset, + *last_page_ok)); + DBUG_RETURN(0); +} + + +/* + Get number bytes for record length storing + + SYNOPSIS + translog_variable_record_length_bytes() + length Record length wich will be codded + + RETURN + 1,3,4,5 - number of bytes to store given length +*/ +static uint translog_variable_record_length_bytes(translog_size_t length) +{ + if (length < 250) + return 1; + else if (length < 0xFFFF) + return 3; + else if (length < 0xFFFFFF) + return 4; + return 5; +} + + +/* + Get header of this chunk + + SYNOPSIS + translog_get_chunk_header_length() + page The page where chunk placed + offset Offset of the chunk on this place + + RETURN + total length of the chunk + 0 - Error +*/ + +uint16 translog_get_chunk_header_length(uchar *page, uint16 offset) +{ + DBUG_ENTER("translog_get_chunk_header_length"); + switch (page[offset] & TRANSLOG_CHUNK_TYPE) { + case TRANSLOG_CHUNK_LSN: /* 0 chunk referred as LSN + (head or tail) */ + { + translog_size_t rec_len; + uchar *start= page + offset; + uchar *ptr= start + 1 + 2; + uint16 chunk_len, header_len; + DBUG_PRINT("info", ("TRANSLOG_CHUNK_LSN")); + rec_len= translog_variable_record_1group_decode_len(&ptr); + chunk_len= uint2korr(ptr); + header_len= (ptr -start) +2; + DBUG_PRINT("info", ("rec len: %lu, chunk len: %u, header len: %u", + (ulong) rec_len, (uint) chunk_len, (uint) header_len)); + if (chunk_len) + { + /*TODO: fine header end */ + DBUG_ASSERT(0); + } + DBUG_RETURN(header_len); + break; + } + case TRANSLOG_CHUNK_FIXED: /* 1 (pseudo)fixed record (also + LSN) */ + { + DBUG_PRINT("info", ("TRANSLOG_CHUNK_FIXED = 3")); + DBUG_RETURN(3); + } + case TRANSLOG_CHUNK_NOHDR: /* 2 no header chunk (till page + end) */ + DBUG_PRINT("info", ("TRANSLOG_CHUNK_NOHDR = 1")); + DBUG_RETURN(1); + break; + case TRANSLOG_CHUNK_LNGTH: /* 3 chunk with chunk length */ + DBUG_PRINT("info", ("TRANSLOG_CHUNK_LNGTH = 3")); + DBUG_RETURN(3); + break; + default: + DBUG_ASSERT(0); + } +} + + +/* + Initialize transaction log + + SYNOPSIS + translog_init() + directory Directory where log files are put + log_file_max_size max size of one log size (for new logs creation) + server_version version of MySQL server (MYSQL_VERSION_ID) + server_id server ID (replication & Co) + pagecache Page cache for the log reads + flags flags (TRANSLOG_PAGE_CRC, TRANSLOG_SECTOR_PROTECTION + TRANSLOG_RECORD_CRC) + + RETURN + 0 - OK + 1 - Error +*/ + +my_bool translog_init(const char *directory, + uint32 log_file_max_size, + uint32 server_version, + uint32 server_id, PAGECACHE *pagecache, uint flags) +{ + int i; + int old_log_was_recovered= 0, logs_found= 0; + TRANSLOG_ADDRESS sure_page, last_page, last_valid_page; + DBUG_ENTER("translog_init"); + + + if (pthread_mutex_init(&log_descriptor.sent_to_file_lock, MY_MUTEX_INIT_FAST)) + DBUG_RETURN(1); + + /* Directory to store files */ + unpack_dirname(log_descriptor.directory, directory); + + if ((log_descriptor.directory_fd= my_open(log_descriptor.directory, + O_RDONLY, MYF(MY_WME))) < 0) + { + UNRECOVERABLE_ERROR(("Error %d during opening directory '%s'", + errno, log_descriptor.directory)); + DBUG_RETURN(1); + } + + /* max size of one log size (for new logs creation) */ + log_descriptor.log_file_max_size= + log_file_max_size - (log_file_max_size % TRANSLOG_PAGE_SIZE); + /* server version */ + log_descriptor.server_version= server_version; + /* server ID */ + log_descriptor.server_id= server_id; + /* Page cache for the log reads */ + log_descriptor.pagecache= pagecache; + /* Flags */ + DBUG_ASSERT((flags & + ~(TRANSLOG_PAGE_CRC | TRANSLOG_SECTOR_PROTECTION | + TRANSLOG_RECORD_CRC)) == 0); + log_descriptor.flags= flags; + log_descriptor.page_overhead= 7; + if (flags & TRANSLOG_PAGE_CRC) + log_descriptor.page_overhead+= 4; + if (flags & TRANSLOG_SECTOR_PROTECTION) + log_descriptor.page_overhead+= (TRANSLOG_PAGE_SIZE / 512) * 2; + log_descriptor.page_capacity_chunk_2= + TRANSLOG_PAGE_SIZE - log_descriptor.page_overhead - 1; + DBUG_ASSERT(TRANSLOG_WRITE_BUFFER % TRANSLOG_PAGE_SIZE == 0); + log_descriptor.buffer_capacity_chunk_2= + (TRANSLOG_WRITE_BUFFER / TRANSLOG_PAGE_SIZE) * + log_descriptor.page_capacity_chunk_2; + log_descriptor.half_buffer_capacity_chunk_2= + log_descriptor.buffer_capacity_chunk_2 / 2; + DBUG_PRINT("info", + ("Overhead: %u, pc2: %u, bc2: %u, bc2/2: %u", + log_descriptor.page_overhead, + log_descriptor.page_capacity_chunk_2, + log_descriptor.buffer_capacity_chunk_2, + log_descriptor.half_buffer_capacity_chunk_2)); + + /* *** Current state of the log handler *** */ + + /* Init log handler file handlers cache */ + for (i= 0; i < OPENED_FILES_NUM; i++) + { + log_descriptor.log_file_num[i]= 0; + } + + /* just to init it somehow */ + translog_start_buffer(log_descriptor.buffers, &log_descriptor.bc, 0); + + /* Buffers for log writing */ + for (i= 0; i < TRANSLOG_BUFFERS_NO; i++) + { +#ifndef DBUG_OFF + log_descriptor.buffers[i].buffer_no= (uint8) i; + log_descriptor.buffers[i].locked_by= NULL; +#endif + if (translog_buffer_init(log_descriptor.buffers + i)) + DBUG_RETURN(1); + } + + logs_found= (last_logno != CONTROL_FILE_IMPOSSIBLE_FILENO); + + if (logs_found) + { + my_bool pageok; + /* + TODO: scan directory for maria_log.XXXXXXXX files and find + highest XXXXXXXX & set logs_found + */ + + /* TODO: check that last checkpoint within present log addresses space */ + /* find the log end */ + if (last_checkpoint_lsn.file_no == CONTROL_FILE_IMPOSSIBLE_FILENO) + { + DBUG_ASSERT(last_checkpoint_lsn.rec_offset == 0); + /* there was no checkpoints we will read from the beginning */ + sure_page.file_no= 1; + sure_page.rec_offset= TRANSLOG_PAGE_SIZE; + } + else + { + sure_page= last_checkpoint_lsn; + DBUG_ASSERT(sure_page.rec_offset % TRANSLOG_PAGE_SIZE != 0); + sure_page.rec_offset-= sure_page.rec_offset % TRANSLOG_PAGE_SIZE; + } + log_descriptor.horizon.file_no= last_page.file_no= last_logno; + if (translog_get_last_page_addr(&last_page, &pageok)) + DBUG_RETURN(1); + if (last_page.rec_offset == 0) + { + if (last_page.file_no == 1) + { + logs_found= 0; /* file #1 has no pages */ + } + else + { + last_page.file_no--; + if (translog_get_last_page_addr(&last_page, &pageok)) + DBUG_RETURN(1); + } + } + } + if (logs_found) + { + TRANSLOG_ADDRESS current_page= sure_page; + my_bool pageok; + + DBUG_ASSERT(sure_page.file_no < last_page.file_no || + (sure_page.file_no == last_page.file_no && + sure_page.rec_offset <= last_page.rec_offset)); + + /* TODO: check page size */ + + last_valid_page.file_no= CONTROL_FILE_IMPOSSIBLE_FILENO; + last_valid_page.rec_offset= 0; + /* scan and validate pages */ + do + { + TRANSLOG_ADDRESS current_file_last_page; + current_file_last_page.file_no= current_page.file_no; + if (translog_get_last_page_addr(¤t_file_last_page, &pageok)) + DBUG_RETURN(1); + if (!pageok) + { + DBUG_PRINT("error", ("File %u have no complete last page", + (uint) current_file_last_page.file_no)); + old_log_was_recovered= 1; + /* This file is not written till the end so it should be last */ + last_page= current_file_last_page; + /* TODO: issue warning */ + } + do + { + TRANSLOG_VALIDATOR_DATA data= + { + ¤t_page, 0 + }; + uchar buffer[TRANSLOG_PAGE_SIZE], *page; + if ((page= translog_get_page(&data, buffer)) == NULL) + DBUG_RETURN(1); + if (data.was_recovered) + { + DBUG_PRINT("error", ("file no %u (%d), rec_offset 0x%lx (%lu) (%d)", + (uint) current_page.file_no, + (uint3korr(page + 3) != current_page.file_no), + (ulong) current_page.rec_offset, + (ulong) (current_page.rec_offset / + TRANSLOG_PAGE_SIZE), + (uint3korr(page) != + current_page.rec_offset / TRANSLOG_PAGE_SIZE))); + old_log_was_recovered= 1; + break; + } + last_valid_page= current_page; + current_page.rec_offset+= TRANSLOG_PAGE_SIZE; + } while (current_page.rec_offset <= current_file_last_page.rec_offset); + current_page.file_no++; + current_page.rec_offset= TRANSLOG_PAGE_SIZE; + } while (current_page.file_no <= last_page.file_no && + !old_log_was_recovered); + if (last_valid_page.file_no == CONTROL_FILE_IMPOSSIBLE_FILENO) + { + DBUG_ASSERT(last_valid_page.rec_offset == 0); + + /* Panic!!! Even page which should be valid is invalid */ + /* TODO: issue error */ + DBUG_RETURN(1); + } + DBUG_PRINT("info", ("Last valid page is in file %lu offset %lu (0x%lx), " + "Logs found: %d, was recovered: %d", + (ulong) last_valid_page.file_no, + (ulong) last_valid_page.rec_offset, + (ulong) last_valid_page.rec_offset, + logs_found, old_log_was_recovered)); + + /* TODO: check server ID */ + if (logs_found && !old_log_was_recovered) + { + TRANSLOG_VALIDATOR_DATA data= + { + &last_valid_page, 0 + }; + uchar buffer[TRANSLOG_PAGE_SIZE], *page; + uint16 chunk_offset; + /* continue old log */ + DBUG_ASSERT(last_valid_page.file_no == log_descriptor.horizon.file_no); + if ((page= translog_get_page(&data, + buffer)) == NULL || + (chunk_offset= translog_get_first_chunk_offset(page)) == 0) + DBUG_RETURN(1); + + /* Puts filled part of old page in the buffer */ + log_descriptor.horizon= last_valid_page; + translog_start_buffer(log_descriptor.buffers, &log_descriptor.bc, 0); + /* + Free space if filled with 0 and first byte of + real chunk can't be 0 + */ + while (chunk_offset < TRANSLOG_PAGE_SIZE && page[chunk_offset] != '\0') + { + uint16 chunk_length; + if ((chunk_length= + translog_get_total_chunk_length(page, chunk_offset)) == 0) + DBUG_RETURN(1); + DBUG_PRINT("info", ("chunk: offset: %u, length %u", + (uint) chunk_offset, (uint) chunk_length)); + chunk_offset+= chunk_length; + + /* chunk can't cross the page border */ + DBUG_ASSERT(chunk_offset <= TRANSLOG_PAGE_SIZE); + } + memmove(log_descriptor.buffers->buffer, page, chunk_offset); + log_descriptor.bc.buffer->size+= chunk_offset; + log_descriptor.bc.ptr+= chunk_offset; + log_descriptor.bc.current_page_size= chunk_offset; + log_descriptor.horizon.rec_offset= + chunk_offset + last_valid_page.rec_offset; + DBUG_PRINT("info", ("Move Page #%u: 0x%lx, chaser: %d, Size: %lu (%lu)", + (uint) log_descriptor.bc.buffer_no, + (ulong) log_descriptor.bc.buffer, + log_descriptor.bc.chaser, + (ulong) log_descriptor.bc.buffer->size, + (ulong) (log_descriptor.bc.ptr -log_descriptor.bc. + buffer->buffer))); + DBUG_ASSERT(log_descriptor.bc.chaser + || + ((ulong) + (log_descriptor.bc.ptr -log_descriptor.bc.buffer->buffer) == + log_descriptor.bc.buffer->size)); + DBUG_ASSERT(log_descriptor.bc.buffer->buffer_no == + log_descriptor.bc.buffer_no); + DBUG_ASSERT(log_descriptor.bc.current_page_size <= TRANSLOG_PAGE_SIZE); + } + } + DBUG_PRINT("info", ("Logs found: %d, was recovered %d", + logs_found, old_log_was_recovered)); + if (!logs_found) + { + /* Start new log system from scratch */ + /* Current log number */ + log_descriptor.horizon.file_no= 1; + /* Used space */ + log_descriptor.horizon.rec_offset= TRANSLOG_PAGE_SIZE; // header page + /* Current logs file number in page cache */ + log_descriptor.log_file_num[0]= + open_logfile_by_number_no_cache(log_descriptor.horizon.file_no); + if (translog_write_file_header()) + DBUG_RETURN(1); + if (ma_control_file_write_and_force(NULL, log_descriptor.horizon.file_no, + CONTROL_FILE_UPDATE_ONLY_LOGNO)) + DBUG_RETURN(1); + /* assign buffer 0 */ + translog_start_buffer(log_descriptor.buffers, &log_descriptor.bc, 0); + translog_new_page_header(&log_descriptor.horizon, &log_descriptor.bc); + } + else if (old_log_was_recovered) + { + int buffer_touched= log_descriptor.bc.buffer->file; + if (buffer_touched) + { + struct st_translog_buffer *buffer= log_descriptor.bc.buffer; + /* + We are in initialization so we can use translog_buffer_lock instead + of translog_lock, because there is no other threads which can lock + the loghandler. + */ + if (translog_buffer_lock(buffer) || + translog_buffer_next(&log_descriptor.horizon, &log_descriptor.bc, + 1) || + translog_buffer_unlock(log_descriptor.bc.buffer) || + translog_buffer_flush(buffer) || translog_buffer_unlock(buffer)) + DBUG_RETURN(1); + } + else + { + log_descriptor.horizon.file_no++; /* leave the demaged file + untouched */ + log_descriptor.horizon.rec_offset= TRANSLOG_PAGE_SIZE; /* header page */ + if (translog_create_new_file()) + DBUG_RETURN(1); + /* + Buffer system left untouched after recovery => we should init it + (starting from buffer 0) + */ + translog_start_buffer(log_descriptor.buffers, &log_descriptor.bc, 0); + translog_new_page_header(&log_descriptor.horizon, &log_descriptor.bc); + } + } + + /* all LSNs that are on disk are flushed */ + log_descriptor.sent_to_file= log_descriptor.flushed= log_descriptor.horizon; + log_descriptor.flushed.rec_offset--; + log_descriptor.sent_to_file.rec_offset--; + + DBUG_RETURN(0); +} + + +/* + Free transaction log file buffer + + SYNOPSIS + translog_buffer_destroy() + buffer_no The buffer to free + + NOTE + This buffer should be locked; +*/ + +static void translog_buffer_destroy(struct st_translog_buffer *buffer) +{ + DBUG_ENTER("translog_buffer_destroy"); + DBUG_PRINT("enter", + ("Buffer #%u: 0x%lx, file: %u, offset (%u,0x%lx), size %lu", + (uint) buffer->buffer_no, (ulong) buffer, + (uint) buffer->file, + (ulong) buffer->offset.file_no, (ulong) buffer->offset.rec_offset, + (ulong) buffer->size)); + DBUG_ASSERT(buffer->waiting_filling_buffer.last_thread == 0); + if (buffer->file) + { + /* + We ignore error here, because we can't do something about it + (it is shutting down) + */ + translog_buffer_flush(buffer); + } + DBUG_PRINT("info", ("Unlock mutex 0x%lx", (ulong) &buffer->mutex)); + pthread_mutex_unlock(&buffer->mutex); + DBUG_PRINT("info", ("Destroy mutex 0x%lx", (ulong) &buffer->mutex)); + pthread_mutex_destroy(&buffer->mutex); + DBUG_VOID_RETURN; +} + + +/* + Free log handler resources + + SYNOPSIS + translog_destroy() +*/ + +void translog_destroy() +{ + int i; + DBUG_ENTER("translog_destroy"); + if (log_descriptor.bc.buffer->file != 0) + translog_finish_page(&log_descriptor.horizon, &log_descriptor.bc); + + for (i= 0; i < TRANSLOG_BUFFERS_NO; i++) + { + struct st_translog_buffer *buffer= log_descriptor.buffers + i; + translog_buffer_lock(buffer); + translog_buffer_destroy(buffer); + } + /* close files */ + for (i= 0; i < OPENED_FILES_NUM; i++) + { + if (log_descriptor.log_file_num[i]) + translog_close_log_file(log_descriptor.log_file_num[i]); + } + pthread_mutex_destroy(&log_descriptor.sent_to_file_lock); + my_close(log_descriptor.directory_fd, MYF(MY_WME)); + DBUG_VOID_RETURN; +} + + +/* + Lock the loghandler + + SYNOPSIS + translog_lock() + + RETURN + 0 - OK + 1 - Error +*/ + +static my_bool translog_lock() +{ + struct st_translog_buffer *current_buffer; + DBUG_ENTER("translog_lock"); + + /* + locking the loghandler mean locking current buffer, but it can change + during locking, so we should check it + */ + for (;;) + { + current_buffer= log_descriptor.bc.buffer; + if (translog_buffer_lock(current_buffer)) + DBUG_RETURN(1); + if (log_descriptor.bc.buffer == current_buffer) + break; + translog_buffer_unlock(current_buffer); + } + DBUG_RETURN(0); +} + + +/* + Unlock the loghandler + + SYNOPSIS + translog_unlock() + + RETURN + 0 - OK + 1 - Error +*/ + +#ifndef DBUG_OFF +static my_bool translog_unlock() +{ + DBUG_ENTER("translog_unlock"); + translog_buffer_unlock(log_descriptor.bc.buffer); + + DBUG_RETURN(0); +} +#else +#define translog_unlock() \ + translog_buffer_unlock(log_descriptor.bc.buffer); +#endif + +/* + Start new page + + SYNOPSIS + translog_page_next() + horizon \ Position in file and buffer where we are + cursor / + prev_buffer Buffer which should be flushed will be assigned + here if it is need + + NOTE + handler should be locked + + RETURN + 0 - OK + 1 - Error +*/ + +static my_bool translog_page_next(TRANSLOG_ADDRESS *horizon, + struct st_buffer_cursor *cursor, + struct st_translog_buffer **prev_buffer) +{ + struct st_translog_buffer *buffer= cursor->buffer; + DBUG_ENTER("translog_page_next"); + + if ((cursor->ptr +TRANSLOG_PAGE_SIZE > + cursor->buffer->buffer + TRANSLOG_WRITE_BUFFER) || + (horizon->rec_offset + TRANSLOG_PAGE_SIZE > + log_descriptor.log_file_max_size)) + { + DBUG_PRINT("info", ("Switch to next buffer, Buffer Size %lu (%lu) => %d, " + "File size %lu max %lu => %d", + (ulong) cursor->buffer->size, + (ulong) (cursor->ptr -cursor->buffer->buffer), + (cursor->ptr +TRANSLOG_PAGE_SIZE > + cursor->buffer->buffer + TRANSLOG_WRITE_BUFFER), + (ulong) horizon->rec_offset, + (ulong) log_descriptor.log_file_max_size, + (horizon->rec_offset + TRANSLOG_PAGE_SIZE > + log_descriptor.log_file_max_size))); + if (translog_buffer_next(horizon, cursor, + (horizon->rec_offset + + TRANSLOG_PAGE_SIZE) > + log_descriptor.log_file_max_size)) + DBUG_RETURN(1); + *prev_buffer= buffer; + DBUG_PRINT("info", ("Buffer #%u (0x%lu) have to be flushed", + (uint) buffer->buffer_no, (ulong) buffer)); + } + else + { + DBUG_PRINT("info", ("Use the same buffer #%u (0x%lu), " + "Buffer Size %lu (%lu)", + (uint) buffer->buffer_no, + (ulong) buffer, + (ulong) cursor->buffer->size, + (ulong) (cursor->ptr -cursor->buffer->buffer))); + translog_finish_page(horizon, cursor); + translog_new_page_header(horizon, cursor); + *prev_buffer= NULL; + } + DBUG_RETURN(0); +} + + +/* + Write data of given length to the current page + + SYNOPSIS + translog_write_data_on_page() + horizon \ Pointers on file and buffer + cursor / + length IN length of the chunk + buffer buffer with data + + RETURN + 0 - OK + 1 - Error +*/ + +my_bool translog_write_data_on_page(TRANSLOG_ADDRESS *horizon, + struct st_buffer_cursor *cursor, + translog_size_t length, uchar *buffer) +{ + DBUG_ENTER("translog_write_data_on_page"); + DBUG_PRINT("enter", ("Chunk length: %lu Page size %u", + (ulong) length, (uint) cursor->current_page_size)); + DBUG_ASSERT(length > 0); + DBUG_ASSERT(length + cursor->current_page_size <= TRANSLOG_PAGE_SIZE); + DBUG_ASSERT(length + cursor->ptr <=cursor->buffer->buffer + + TRANSLOG_WRITE_BUFFER); + + memmove(cursor->ptr, buffer, length); + cursor->ptr+= length; + horizon->rec_offset+= length; + cursor->current_page_size+= length; + if (!cursor->chaser) + cursor->buffer->size+= length; + DBUG_PRINT("info", ("Write data buffer #%u: 0x%lx," + "chaser: %d, Size: %lu (%lu)", + (uint) cursor->buffer->buffer_no, (ulong) cursor->buffer, + cursor->chaser, (ulong) cursor->buffer->size, + (ulong) (cursor->ptr -cursor->buffer->buffer))); + DBUG_ASSERT(cursor->chaser || + ((ulong) (cursor->ptr -cursor->buffer->buffer) == + cursor->buffer->size)); + DBUG_ASSERT(cursor->buffer->buffer_no == cursor->buffer_no); + DBUG_ASSERT(cursor->current_page_size <= TRANSLOG_PAGE_SIZE); + + DBUG_RETURN(0); +} + + +/* + Write data from parts of given length to the current page + + SYNOPSIS + translog_write_parts_on_page() + horizon \ Pointers on file and buffer + cursor / + length IN length of the chunk + parts IN/OUT chunk source + + RETURN + 0 - OK + 1 - Error +*/ + +my_bool translog_write_parts_on_page(TRANSLOG_ADDRESS *horizon, + struct st_buffer_cursor *cursor, + translog_size_t length, + struct st_translog_parts *parts) +{ + translog_size_t left= length; + uint cur= (uint) parts->current; + DBUG_ENTER("translog_write_parts_on_page"); + DBUG_PRINT("enter", ("Chunk length: %lu, parts %u of %u. Page size %u, " + "Buffer size: %lu (%lu)", + (ulong) length, + (uint) (cur + 1), (uint) parts->parts.elements, + (uint) cursor->current_page_size, + (ulong) cursor->buffer->size, + (ulong) (cursor->ptr -cursor->buffer->buffer))); + DBUG_ASSERT(length > 0); + DBUG_ASSERT(length + cursor->current_page_size <= TRANSLOG_PAGE_SIZE); + DBUG_ASSERT(length + cursor->ptr <=cursor->buffer->buffer + + TRANSLOG_WRITE_BUFFER); + + do + { + translog_size_t len; + struct st_translog_part part; + uchar *buff; + + DBUG_ASSERT(cur < parts->parts.elements); + get_dynamic(&parts->parts, (gptr) &part, cur); + buff= part.buff; + DBUG_PRINT("info", ("Part %u, Length: %lu, left: %lu", + (uint) (cur + 1), (ulong) part.len, (ulong) left)); + + if (part.len > left) + { + /* we should write less then the current part */ + len= left; + part.len-= len; + part.buff+= len; + if (set_dynamic(&parts->parts, (gptr) &part, cur)) + DBUG_RETURN(1); + DBUG_PRINT("info", ("Set new part %u, Length: %lu", + (uint) (cur + 1), (ulong) part.len)); + } + else + { + len= part.len; + cur++; + DBUG_PRINT("info", ("moved to next part (len: %lu)", (ulong) len)); + } + DBUG_PRINT("info", ("copy: 0x%lx <- 0x%lx %u", + (ulong) cursor->ptr, (ulong)buff, (uint)len)); + memmove(cursor->ptr, buff, len); + left-= len; + cursor->ptr+= len; + } while (left); + + parts->current= cur; + horizon->rec_offset+= length; + cursor->current_page_size+= length; + if (!cursor->chaser) + cursor->buffer->size+= length; + DBUG_PRINT("info", ("Write parts buffer #%u: 0x%lx, " + "chaser: %d, Size: %lu (%lu)", + (uint) cursor->buffer->buffer_no, (ulong) cursor->buffer, + cursor->chaser, (ulong) cursor->buffer->size, + (ulong) (cursor->ptr -cursor->buffer->buffer))); + DBUG_ASSERT(cursor->chaser || + ((ulong) (cursor->ptr -cursor->buffer->buffer) == + cursor->buffer->size)); + DBUG_ASSERT(cursor->buffer->buffer_no == cursor->buffer_no); + DBUG_ASSERT((cursor->ptr -cursor->buffer->buffer) %TRANSLOG_PAGE_SIZE == + cursor->current_page_size % TRANSLOG_PAGE_SIZE); + DBUG_ASSERT(cursor->current_page_size <= TRANSLOG_PAGE_SIZE); + + DBUG_RETURN(0); +} + + +/* + Put 1 group chunk type 0 header into parts array + + SYNOPSIS + translog_write_variable_record_1group_header() + parts Descriptor of record source parts + type the log record type + short_trid Sort transaction ID or 0 if it has no sense + header_length Calculated header length of chunk type 0 + chunk0_header Buffer for the chunk header writing +*/ + +static void +translog_write_variable_record_1group_header(struct st_translog_parts *parts, + enum translog_record_type type, + SHORT_TRANSACTION_ID short_trid, + uint16 header_length, + uchar *chunk0_header) +{ + struct st_translog_part part; + DBUG_ASSERT(parts->current != 0); /* first part is left for + header */ + parts->total_record_length+= (part.len= header_length); + part.buff= chunk0_header; + *chunk0_header= (uchar) (type |TRANSLOG_CHUNK_LSN); + int2store(chunk0_header + 1, short_trid); + translog_write_variable_record_1group_code_len(chunk0_header + 3, + parts->record_length, + header_length); + int2store(chunk0_header + header_length - 2, 0); + parts->current--; + set_dynamic(&parts->parts, (gptr) &part, parts->current); +} + + +/* + Increase number of writers for this buffer + + SYNOPSIS + translog_buffer_increase_writers() + buffer target buffer +*/ + +#ifndef DBUG_OFF +static void translog_buffer_increase_writers(struct st_translog_buffer *buffer) +{ + DBUG_ENTER("translog_buffer_increase_writers"); + buffer->copy_to_buffer_in_progress++; + DBUG_PRINT("info", ("copy_to_buffer_in_progress, buffer #%u 0x%lx: %d", + (uint) buffer->buffer_no, (ulong) buffer, + buffer->copy_to_buffer_in_progress)); + DBUG_VOID_RETURN; +} +#else +#define translog_buffer_increase_writers(B) \ + (B)->copy_to_buffer_in_progress++; +#endif + + +/* + Decrease number of writers for this buffer + + SYNOPSIS + translog_buffer_decrease_writers() + buffer target buffer +*/ + + +static void translog_buffer_decrease_writers(struct st_translog_buffer *buffer) +{ + DBUG_ENTER("translog_buffer_decrease_writers"); + buffer->copy_to_buffer_in_progress--; + DBUG_PRINT("info", ("copy_to_buffer_in_progress, buffer #%u 0x%lx: %d", + (uint) buffer->buffer_no, (ulong) buffer, + buffer->copy_to_buffer_in_progress)); + if (buffer->copy_to_buffer_in_progress == 0 && + buffer->waiting_filling_buffer.last_thread != NULL) + { + wqueue_release_queue(&buffer->waiting_filling_buffer); + } + DBUG_VOID_RETURN; +} + + +/* + Put chunk 2 from new page beginning + + SYNOPSIS + translog_write_variable_record_chunk2_page() + parts Descriptor of record source parts + horizon \ Pointers on file position and buffer + cursor / + + RETURN + 0 - OK + 1 - Error +*/ + +static my_bool +translog_write_variable_record_chunk2_page(struct st_translog_parts *parts, + TRANSLOG_ADDRESS *horizon, + struct st_buffer_cursor *cursor) +{ + struct st_translog_buffer *buffer_to_flush= 0; + int rc; + uchar chunk2_header[1]= + { + TRANSLOG_CHUNK_NOHDR + }; + + DBUG_ENTER("translog_write_variable_record_chunk2_page"); + + rc= translog_page_next(horizon, cursor, &buffer_to_flush); + if (buffer_to_flush != NULL) + { + rc|= translog_buffer_lock(buffer_to_flush); + translog_buffer_decrease_writers(buffer_to_flush); + if (!rc) + rc= translog_buffer_flush(buffer_to_flush); + rc|= translog_buffer_unlock(buffer_to_flush); + } + if (rc) + DBUG_RETURN(1); + + translog_write_data_on_page(horizon, cursor, 1, chunk2_header); + translog_write_parts_on_page(horizon, cursor, + log_descriptor.page_capacity_chunk_2, parts); + DBUG_RETURN(0); +} + + +/* + Put chunk 3 of requested length in the buffer from new page beginning + + SYNOPSIS + translog_write_variable_record_chunk3_page() + parts Descriptor of record source parts + length Length of this chunk + horizon \ Pointers on file position and buffer + cursor / + + RETURN + 0 - OK + 1 - Error +*/ + +static my_bool +translog_write_variable_record_chunk3_page(struct st_translog_parts *parts, + uint16 length, + TRANSLOG_ADDRESS *horizon, + struct st_buffer_cursor *cursor) +{ + struct st_translog_buffer *buffer_to_flush= 0; + struct st_translog_part part; + int rc; + uchar chunk3_header[1 + 2]; + + DBUG_ENTER("translog_write_variable_record_chunk3_page"); + + rc= translog_page_next(horizon, cursor, &buffer_to_flush); + if (buffer_to_flush != NULL) + { + rc|= translog_buffer_lock(buffer_to_flush); + translog_buffer_decrease_writers(buffer_to_flush); + if (!rc) + rc= translog_buffer_flush(buffer_to_flush); + rc|= translog_buffer_unlock(buffer_to_flush); + } + if (rc) + DBUG_RETURN(1); + if (length == 0) + { + /* It was call to write page header only (no data for chunk 3) */ + DBUG_PRINT("info", ("It is a call to make page header only")); + DBUG_RETURN(0); + } + + DBUG_ASSERT(parts->current != 0); /* first part is left for + header */ + parts->total_record_length+= (part.len= 1 + 2); + part.buff= chunk3_header; + *chunk3_header= (uchar) (TRANSLOG_CHUNK_LNGTH); + int2store(chunk3_header + 1, length); + parts->current--; + set_dynamic(&parts->parts, (gptr) &part, parts->current); + + translog_write_parts_on_page(horizon, cursor, length + 1 + 2, parts); + DBUG_RETURN(0); +} + +/* + Move log pointer (horizon) on given number pages starting from next page, + and given offset on the last page + + SYNOPSIS + translog_advance_pointer() + pages Number of full pages starting from the next one + last_page_data Plus this data on the last page + + RETURN + 0 - OK + 1 - Error +*/ + +static my_bool translog_advance_pointer(uint pages, uint16 last_page_data) +{ + translog_size_t last_page_offset= + log_descriptor.page_overhead + last_page_data; + translog_size_t offset= TRANSLOG_PAGE_SIZE - log_descriptor.bc.current_page_size /* next + page + */ + + pages * TRANSLOG_PAGE_SIZE + last_page_offset; + translog_size_t buffer_end_offset, file_end_offset, min_offset; + DBUG_ENTER("translog_advance_pointer"); + DBUG_PRINT("enter", ("Pointer: (%u, 0x%lx) + %u + %u pages + %u + %u", + (uint) log_descriptor.horizon.file_no, + (ulong) log_descriptor.horizon.rec_offset, + (uint) (TRANSLOG_PAGE_SIZE - + log_descriptor.bc.current_page_size), + pages, (uint) log_descriptor.page_overhead, + (uint) last_page_data)); + + for (;;) + { + uint8 new_buffer_no= + (log_descriptor.bc.buffer_no + 1) % TRANSLOG_BUFFERS_NO; + struct st_translog_buffer *new_buffer; + struct st_translog_buffer *old_buffer; + buffer_end_offset= TRANSLOG_WRITE_BUFFER - log_descriptor.bc.buffer->size; + file_end_offset= + log_descriptor.log_file_max_size - log_descriptor.horizon.rec_offset; + DBUG_PRINT("info", ("offset: %lu, buffer_end_offs: %lu, " + "file_end_offs: %lu", + (ulong) offset, (ulong) buffer_end_offset, + (ulong) file_end_offset)); + DBUG_PRINT("info", ("Buff #%u %u (0x%lx) offset 0x%lx + size 0x%lx = " + "0x%lx (0x%lx)", + (uint) log_descriptor.bc.buffer->buffer_no, + (uint) log_descriptor.bc.buffer_no, + (ulong) log_descriptor.bc.buffer, + (ulong) log_descriptor.bc.buffer->offset.rec_offset, + (ulong) log_descriptor.bc.buffer->size, + (ulong) (log_descriptor.bc.buffer->offset.rec_offset + + log_descriptor.bc.buffer->size), + (ulong) log_descriptor.horizon.rec_offset)); + DBUG_ASSERT(log_descriptor.bc.buffer->offset.rec_offset + + log_descriptor.bc.buffer->size == + log_descriptor.horizon.rec_offset); + + if (offset <= buffer_end_offset && offset <= file_end_offset) + break; + old_buffer= log_descriptor.bc.buffer; + new_buffer_no= (log_descriptor.bc.buffer_no + 1) % TRANSLOG_BUFFERS_NO; + new_buffer= log_descriptor.buffers + new_buffer_no; + + translog_buffer_lock(new_buffer); + translog_wait_for_buffer_free(new_buffer); + + min_offset= (buffer_end_offset < file_end_offset ? + buffer_end_offset : file_end_offset); + log_descriptor.bc.buffer->size+= min_offset; + log_descriptor.bc.ptr +=min_offset; + DBUG_PRINT("info", ("NewP buffer #%u: 0x%lx, chaser: %d, Size: %lu (%lu)", + (uint) log_descriptor.bc.buffer->buffer_no, + (ulong) log_descriptor.bc.buffer, + log_descriptor.bc.chaser, + (ulong) log_descriptor.bc.buffer->size, + (ulong) (log_descriptor.bc.ptr -log_descriptor.bc. + buffer->buffer))); + DBUG_ASSERT((ulong) + (log_descriptor.bc.ptr -log_descriptor.bc.buffer->buffer) == + log_descriptor.bc.buffer->size); + DBUG_ASSERT(log_descriptor.bc.buffer->buffer_no == + log_descriptor.bc.buffer_no); + translog_buffer_increase_writers(log_descriptor.bc.buffer); + + if (file_end_offset <= buffer_end_offset) + { + log_descriptor.horizon.file_no++; + log_descriptor.horizon.rec_offset= TRANSLOG_PAGE_SIZE; + DBUG_PRINT("info", ("New file %d", log_descriptor.horizon.file_no)); + if (translog_create_new_file()) + { + DBUG_RETURN(1); + } + } + else + { + DBUG_PRINT("info", ("The same file")); + log_descriptor.horizon.rec_offset+= min_offset; + } + translog_start_buffer(new_buffer, &log_descriptor.bc, new_buffer_no); + if (translog_buffer_unlock(old_buffer)) + { + DBUG_RETURN(1); + } + offset-= min_offset; + } + log_descriptor.bc.ptr+= offset; + log_descriptor.bc.buffer->size+= offset; + translog_buffer_increase_writers(log_descriptor.bc.buffer); + log_descriptor.horizon.rec_offset+= offset; + log_descriptor.bc.current_page_size= last_page_offset; + DBUG_PRINT("info", ("drop write_counter")); + log_descriptor.bc.write_counter= 0; + log_descriptor.bc.previous_offset= 0; + DBUG_PRINT("info", ("NewP buffer #%u: 0x%lx, chaser: %d, Size: %lu (%lu), " + "offset: %u last page: %u", + (uint) log_descriptor.bc.buffer->buffer_no, + (ulong) log_descriptor.bc.buffer, + log_descriptor.bc.chaser, + (ulong) log_descriptor.bc.buffer->size, + (ulong) (log_descriptor.bc.ptr -log_descriptor.bc.buffer-> + buffer), (uint) offset, + (uint) last_page_offset)); + DBUG_ASSERT(log_descriptor.bc.chaser + || + ((ulong) (log_descriptor.bc.ptr -log_descriptor.bc.buffer->buffer) + == log_descriptor.bc.buffer->size)); + DBUG_ASSERT(log_descriptor.bc.buffer->buffer_no == + log_descriptor.bc.buffer_no); + DBUG_PRINT("info", + ("pointer moved to: (%u, 0x%lx)", + (uint) log_descriptor.horizon.file_no, + (ulong) log_descriptor.horizon.rec_offset)); + DBUG_ASSERT((log_descriptor.bc.ptr -log_descriptor.bc.buffer-> + buffer) %TRANSLOG_PAGE_SIZE == + log_descriptor.bc.current_page_size % TRANSLOG_PAGE_SIZE); + DBUG_ASSERT(log_descriptor.bc.current_page_size <= TRANSLOG_PAGE_SIZE); + log_descriptor.bc.protected= 0; + DBUG_RETURN(0); +} + + + +/* + Get page rest + + SYNOPSIS + translog_get_current_page_rest() + + NOTE loghandler should be locked + + RETURN + number of bytes left on the current page +*/ + +#define translog_get_current_page_rest() \ + (TRANSLOG_PAGE_SIZE - log_descriptor.bc.current_page_size) + +/* + Get buffer rest in full pages + + SYNOPSIS + translog_get_current_buffer_rest() + + NOTE loghandler should be locked + + RETURN + number of full pages left on the current buffer +*/ + +#define translog_get_current_buffer_rest() \ + ((log_descriptor.bc.buffer->buffer + TRANSLOG_WRITE_BUFFER - \ + log_descriptor.bc.ptr) / \ + TRANSLOG_PAGE_SIZE) + +/* + Calculate possible group size without first (current) page + + SYNOPSIS + translog_get_current_group_size() + + NOTE loghandler should be locked + + RETURN + group size without first (current) page +*/ + +static translog_size_t translog_get_current_group_size() +{ + /* buffer rest in full pages */ + translog_size_t buffer_rest= translog_get_current_buffer_rest(); + + DBUG_ENTER("translog_get_current_group_size"); + + DBUG_PRINT("info", ("buffer_rest in pages %lu", buffer_rest)); + buffer_rest*= log_descriptor.page_capacity_chunk_2; + /* in case of only half of buffer free we can write this and next buffer */ + if (buffer_rest < log_descriptor.half_buffer_capacity_chunk_2) + { + DBUG_PRINT("info", ("buffer_rest %lu -> add %lu", + buffer_rest, + (ulong) log_descriptor.buffer_capacity_chunk_2)); + buffer_rest+= log_descriptor.buffer_capacity_chunk_2; + } + + DBUG_PRINT("info", ("buffer_rest %lu", buffer_rest)); + + DBUG_RETURN(buffer_rest); +} + + +/* + Write variable record in 1 group + + SYNOPSIS + translog_write_variable_record_1group() + lsn LSN of the record will be written here + type the log record type + short_trid Sort transaction ID or 0 if it has no sense + parts Descriptor of record source parts + buffer_to_flush Buffer which have to be flushed if it is not 0 + header_length Calculated header length of chunk type 0 + tcb Transaction control block pointer for hooks by + record log type + + RETURN + 0 - OK + 1 - Error +*/ + +static my_bool +translog_write_variable_record_1group(LSN *lsn, + enum translog_record_type type, + SHORT_TRANSACTION_ID short_trid, + struct st_translog_parts *parts, + struct st_translog_buffer + *buffer_to_flush, uint16 header_length, + void *tcb) +{ + TRANSLOG_ADDRESS horizon; + struct st_buffer_cursor cursor; + int rc= 0; + uint i; + translog_size_t record_rest, full_pages, first_page; + uint additional_chunk3_page= 0; + uchar chunk0_header[1 + 2 + 5 + 2]; + + DBUG_ENTER("translog_write_variable_record_1group"); + + *lsn= horizon= log_descriptor.horizon; + if (log_record_type_descriptor[type].inwrite_hook && + (*log_record_type_descriptor[type].inwrite_hook)(type, tcb, + lsn, parts)) + { + DBUG_RETURN(1); + } + cursor= log_descriptor.bc; + cursor.chaser= 1; + + /* + Advance pointer To be able unlock the loghandler + */ + first_page= translog_get_current_page_rest(); + record_rest= parts->record_length - (first_page - header_length); + full_pages= record_rest / log_descriptor.page_capacity_chunk_2; + record_rest= (record_rest % log_descriptor.page_capacity_chunk_2); + + if (record_rest + 1 == log_descriptor.page_capacity_chunk_2) + { + DBUG_PRINT("info", ("2 chunks type 3 is needed")); + /* We will write 2 chunks type 3 at the end of this group */ + additional_chunk3_page= 1; + record_rest= 1; + } + + DBUG_PRINT("info", ("first_page: %u (%u), full_pages: %u (%lu), " + "additional: %u (%u), rest %u = %u", + first_page, first_page - header_length, + full_pages, + (ulong) full_pages * + log_descriptor.page_capacity_chunk_2, + additional_chunk3_page, + additional_chunk3_page * + (log_descriptor.page_capacity_chunk_2 - 1), + record_rest, parts->record_length)); + /* record_rest + 3 is chunk type 3 overhead + record_rest */ + translog_advance_pointer(full_pages + additional_chunk3_page, + (record_rest ? record_rest + 3 : 0)); + log_descriptor.bc.buffer->last_lsn= *lsn; + + rc|= translog_unlock(); + + /* + check if we switched buffer and need process it (current buffer is + unlocked already => we will not delay other threads + */ + if (buffer_to_flush != NULL) + { + if (!rc) + rc= translog_buffer_flush(buffer_to_flush); + rc|= translog_buffer_unlock(buffer_to_flush); + } + + if (rc) + DBUG_RETURN(1); + + translog_write_variable_record_1group_header(parts, type, short_trid, + header_length, chunk0_header); + + /* fill the pages */ + translog_write_parts_on_page(&horizon, &cursor, first_page, parts); + + + DBUG_PRINT("info", ("absolute horizon (%u,0x%lx), local (%u,0x%lx)", + (uint) log_descriptor.horizon.file_no, + (ulong) log_descriptor.horizon.rec_offset, + (uint) horizon.file_no, (ulong) horizon.rec_offset)); + + for (i= 0; i < full_pages; i++) + { + if (translog_write_variable_record_chunk2_page(parts, &horizon, &cursor)) + DBUG_RETURN(1); + + DBUG_PRINT("info", ("absolute horizon (%u,0x%lx), local (%u,0x%lx)", + (uint) log_descriptor.horizon.file_no, + (ulong) log_descriptor.horizon.rec_offset, + (uint) horizon.file_no, (ulong) horizon.rec_offset)); + } + + if (additional_chunk3_page) + { + if (translog_write_variable_record_chunk3_page(parts, + log_descriptor. + page_capacity_chunk_2 - 2, + &horizon, &cursor)) + DBUG_RETURN(1); + DBUG_PRINT("info", ("absolute horizon (%u,0x%lx), local (%u,0x%lx)", + (uint) log_descriptor.horizon.file_no, + (ulong) log_descriptor.horizon.rec_offset, + (uint) horizon.file_no, (ulong) horizon.rec_offset)); + DBUG_ASSERT(cursor.current_page_size == TRANSLOG_PAGE_SIZE); + } + + if (translog_write_variable_record_chunk3_page(parts, + record_rest, + &horizon, &cursor)) + DBUG_RETURN(1); + DBUG_PRINT("info", ("absolute horizon (%u,0x%lx), local (%u,0x%lx)", + (uint) log_descriptor.horizon.file_no, + (ulong) log_descriptor.horizon.rec_offset, + (uint) horizon.file_no, (ulong) horizon.rec_offset)); + + rc= translog_buffer_lock(cursor.buffer); + if (!rc) + { + /* + check if we wrote something on lst not full page and need to reconstruct + CRC and sector protection + if (buffer->offset.rec_offset + buffer->size - horizon->rec_offset > + */ + translog_buffer_decrease_writers(cursor.buffer); + } + rc|= translog_buffer_unlock(cursor.buffer); + DBUG_RETURN(rc); +} + + +/* + Write variable record in 1 chunk + + SYNOPSIS + translog_write_variable_record_1chunk() + lsn LSN of the record will be written here + type the log record type + short_trid Sort transaction ID or 0 if it has no sense + parts Descriptor of record source parts + buffer_to_flush Buffer which have to be flushed if it is not 0 + header_length Calculated header length of chunk type 0 + tcb Transaction control block pointer for hooks by + record log type + + RETURN + 0 - OK + 1 - Error +*/ + +static my_bool +translog_write_variable_record_1chunk(LSN *lsn, + enum translog_record_type type, + SHORT_TRANSACTION_ID short_trid, + struct st_translog_parts *parts, + struct st_translog_buffer + *buffer_to_flush, uint16 header_length, + void *tcb) +{ + int rc; + uchar chunk0_header[1 + 2 + 5 + 2]; + DBUG_ENTER("translog_write_variable_record_1chunk"); + + translog_write_variable_record_1group_header(parts, type, short_trid, + header_length, chunk0_header); + + *lsn= log_descriptor.horizon; + if (log_record_type_descriptor[type].inwrite_hook && + (*log_record_type_descriptor[type].inwrite_hook) (type, tcb, + lsn, parts)) + { + DBUG_RETURN(1); + } + + rc= translog_write_parts_on_page(&log_descriptor.horizon, + &log_descriptor.bc, + parts->total_record_length, parts); + log_descriptor.bc.buffer->last_lsn= *lsn; + rc|= translog_unlock(); + + /* + check if we switched buffer and need process it (current buffer is + unlocked already => we will not delay other threads + */ + if (buffer_to_flush != NULL) + { + if (!rc) + rc= translog_buffer_flush(buffer_to_flush); + rc|= translog_buffer_unlock(buffer_to_flush); + } + + DBUG_RETURN(rc); +} + + +/* + Calculate and write LSN difference (compressed LSN) + + SYNOPSIS + translog_put_LSN_diff() + base_lsn LSN from which we calculate difference + lsn LSN for codding + dst pointer before which result should be written + + NOTE: + to store an LSN in a compact way we will use the following compression: + + if a log record has LSN1, and it contains the lSN2 as a back reference, + instead of LSN2 we write LSN1-LSN2, encoded as: + + two bits the number N (see below) + 14 bits + N bytes + + that is, LSN is encoded in 2..5 bytes, and the number of bytes minus 2 + is stored in the first two bits. + + RETURN + pointer on coded LSN + NULL - error +*/ + +static uchar *translog_put_LSN_diff(LSN *base_lsn, LSN *lsn, uchar *dst) +{ + DBUG_ENTER("translog_put_LSN_diff"); + DBUG_PRINT("enter", ("Base: (0x%lx,0x%lx), val: (0x%lx,0x%lx), dst 0x%lx", + (ulong) base_lsn->file_no, + (ulong) base_lsn->rec_offset, + (ulong) lsn->file_no, + (ulong) lsn->rec_offset, (ulong) dst)); + if (base_lsn->file_no == lsn->file_no) + { + uint32 diff; + DBUG_ASSERT(base_lsn->rec_offset > lsn->rec_offset); + diff= base_lsn->rec_offset - lsn->rec_offset; + if (diff <= 0x3FFF) + { + dst-= 2; + dst[0]= diff >> 8; + dst[1]= (diff & 0xFF); + } + else if (diff <= 0x3FFFFF) + { + dst-= 3; + dst[0]= 0x40 | (diff >> 16); + int2store(dst + 1, diff & 0xFFFF); + } + else if (diff <= 0x3FFFFFFF) + { + dst-= 4; + dst[0]= 0x80 | (diff >> 24); + int3store(dst + 1, diff & 0xFFFFFF); + } + else + { + dst-= 5; + dst[0]= 0xC0; + int4store(dst + 1, diff); + } + } + else + { + uint32 diff; + uint32 offset_diff; + ulonglong base_offset= base_lsn->rec_offset; + DBUG_ASSERT(base_lsn->file_no > lsn->file_no); + diff= base_lsn->file_no - lsn->file_no; + if (base_offset < lsn->rec_offset) + { + /* take 1 from file offset */ + diff--; + base_offset+= 0x100000000LL; + } + offset_diff= base_offset - lsn->rec_offset; + if (diff > 0x3f) + { + /*TODO: error - too long transaction - panic!!! */ + UNRECOVERABLE_ERROR(("Too big file diff: %lu", (ulong) diff)); + DBUG_RETURN(NULL); + } + dst-= 5; + *dst= (0xC0 | diff); + int4store(dst + 1, offset_diff); + } + DBUG_PRINT("info", ("new dst: 0x%lx", (ulong) dst)); + DBUG_RETURN(dst); +} + + +/* + Get LSN from LSN-difference (compressed LSN) + + SYNOPSIS + translog_get_LSN_from_diff() + base_lsn LSN from which we calculate difference + src pointer to coded lsn + dst pointer to buffer where to write 7byte LSN + + NOTE: + to store an LSN in a compact way we use the following compression: + + If a log record has LSN1, and it contains the lSN2 as a back reference, + instead of LSN2 we write LSN1-LSN2, encoded as: + + two bits the number N (see below) + 14 bits + N bytes + + That is, LSN is encoded in 2..5 bytes, and the number of bytes minus 2 + is stored in the first two bits. + + RETURN + pointer to buffer after decoded LSN +*/ + +static uchar *translog_get_LSN_from_diff(LSN *base_lsn, uchar *src, uchar *dst) +{ + LSN lsn; + uint32 diff; + uint32 first_byte; + uint8 code; + DBUG_ENTER("translog_get_LSN_from_diff"); + DBUG_PRINT("enter", ("Base: (0x%lx,0x%lx), src: 0x%lx, dst 0x%lx", + (ulong) base_lsn->file_no, + (ulong) base_lsn->rec_offset, (ulong) src, (ulong) dst)); + first_byte= *((uint8*) src); + code= first_byte & 0xC0; + first_byte &= 0x3F; + switch (code) { + case 0x00: + lsn.file_no= base_lsn->file_no; + lsn.rec_offset= + base_lsn->rec_offset - ((first_byte << 8) + *((uint8*) (src + 1))); + src+= 2; + break; + case 0x40: + lsn.file_no= base_lsn->file_no; + diff= uint2korr(src + 1); + lsn.rec_offset= base_lsn->rec_offset - ((first_byte << 16) + diff); + src+= 3; + break; + case 0x80: + lsn.file_no= base_lsn->file_no; + diff= uint3korr(src + 1); + lsn.rec_offset= base_lsn->rec_offset - ((first_byte << 24) + diff); + src+= 4; + break; + case 0xC0: + { + ulonglong base_offset= base_lsn->rec_offset; + diff= uint4korr(src + 1); + if (diff > base_lsn->rec_offset) + { + /* take 1 from file offset */ + first_byte++; + base_offset+= 0x100000000LL; + } + lsn.file_no= base_lsn->file_no - first_byte; + lsn.rec_offset= base_offset - diff; + src+= 5; + break; + } + default: + DBUG_ASSERT(0); + DBUG_RETURN(NULL); + } + lsn7store(dst, &lsn); + DBUG_PRINT("info", ("new src: 0x%lx", (ulong) dst)); + DBUG_RETURN(src); +} + + +/* + Encode relative LSNs listed in the parameters + + SYNOPSIS + translog_relative_LSN_encode() + parts Parts list with encoded LSN(s) + base_lsn LSN which is base for encoding + lsns number of LSN(s) to encode + compressed_LSNs buffer which can be used for storing compressed LSN(s) + + RETURN + 0 - OK + 1 - Error +*/ + +static my_bool translog_relative_LSN_encode(struct st_translog_parts *parts, + LSN *base_lsn, + uint lsns, uchar *compressed_LSNs) +{ + struct st_translog_part part; + uint lsns_len= lsns * 7; + + DBUG_ENTER("translog_relative_LSN_encode"); + + get_dynamic(&parts->parts, (gptr) &part, parts->current); + /* collect all LSN(s) in one chunk if it (they) is (are) divided */ + if (part.len < lsns_len) + { + uint copied= part.len; + DBUG_PRINT("info", ("Using buffer 0x%lx", (ulong) compressed_LSNs)); + memmove(compressed_LSNs, part.buff, part.len); + do + { + get_dynamic(&parts->parts, (gptr) &part, parts->current + 1); + if ((part.len + copied) < lsns_len) + { + memmove(compressed_LSNs + copied, part.buff, part.len); + copied+= part.len; + delete_dynamic_element(&parts->parts, parts->current + 1); + } + else + { + uint len= lsns_len - copied; + memmove(compressed_LSNs + copied, part.buff, len); + copied= lsns_len; + part.buff+= len; + part.len-= len; + /* + We do not check result of set_dynamic, because we are sure that + it will not grow + */ + set_dynamic(&parts->parts, (gptr) &part, parts->current + 1); + } + } while (copied < lsns_len); + part.len= lsns_len; + part.buff= compressed_LSNs; + } + { + /* Compress */ + LSN ref; + uint economy; + uchar *ref_ptr= part.buff + lsns_len - 7; + uchar *dst_ptr= part.buff + lsns_len; + uint i; + for (i= 0; i < lsns; i++, ref_ptr-= 7) + { + lsn7korr(&ref, ref_ptr); + if ((dst_ptr= translog_put_LSN_diff(base_lsn, &ref, dst_ptr)) == NULL) + DBUG_RETURN(1); + } + economy= (dst_ptr - part.buff); + DBUG_PRINT("info", ("Economy %u", economy)); + part.len-= economy; + parts->record_length-= economy; + parts->total_record_length-= economy; + part.buff= dst_ptr; + } + /* + We do not check result of set_dynamic, because we are sure that + it will not grow + */ + set_dynamic(&parts->parts, (gptr) &part, parts->current); + DBUG_RETURN(0); +} + + +/* + Write multi-group variable-size record + + SYNOPSIS + translog_write_variable_record_mgroup() + lsn LSN of the record will be written here + type the log record type + short_trid Sort transaction ID or 0 if it has no sense + parts Descriptor of record source parts + buffer_to_flush Buffer which have to be flushed if it is not 0 + header_length Header length calculated for 1 group + buffer_rest Beginning from which we plan to write in full pages + tcb Transaction control block pointer for hooks by + record log type + + RETURN + 0 - OK + 1 - Error +*/ + +static my_bool +translog_write_variable_record_mgroup(LSN *lsn, + enum translog_record_type type, + SHORT_TRANSACTION_ID short_trid, + struct st_translog_parts *parts, + struct st_translog_buffer + *buffer_to_flush, + uint16 header_length, + translog_size_t buffer_rest, + void *tcb) +{ + TRANSLOG_ADDRESS horizon; + struct st_buffer_cursor cursor; + int rc= 0; + uint i, chunk2_page, full_pages; + uint curr_group= 0; + translog_size_t record_rest, first_page, chunk3_pages, chunk0_pages= 1; + translog_size_t done= 0; + struct st_translog_group_descriptor group; + DYNAMIC_ARRAY groups; + uint16 chunk3_size; + uint16 page_capacity= log_descriptor.page_capacity_chunk_2 + 1; + uint16 last_page_capacity; + my_bool new_page_before_chunk0= 1, first_chunk0= 1; + uchar chunk0_header[1 + 2 + 5 + 2 + 2], group_desc[7 + 1]; + uchar chunk2_header[1]= + { + TRANSLOG_CHUNK_NOHDR + }; + uint header_fixed_part= header_length + 2; + uint groups_per_page= (page_capacity - header_fixed_part) / (7 + 1); + + DBUG_ENTER("translog_write_variable_record_mgroup"); + + if (init_dynamic_array(&groups, sizeof(struct st_translog_group_descriptor), + 10, 10 CALLER_INFO)) + { + UNRECOVERABLE_ERROR(("init array failed")); + DBUG_RETURN(1); + } + + first_page= translog_get_current_page_rest(); + record_rest= parts->record_length - (first_page - 1); + DBUG_PRINT("info", ("Record Rest: %lu", (ulong) record_rest)); + + if (record_rest < buffer_rest) + { + DBUG_PRINT("info", ("too many free space because changing header")); + buffer_rest-= log_descriptor.page_capacity_chunk_2; + DBUG_ASSERT(record_rest >= buffer_rest); + } + + do + { + group.addr= horizon= log_descriptor.horizon; + cursor= log_descriptor.bc; + cursor.chaser= 1; + if ((full_pages= buffer_rest / log_descriptor.page_capacity_chunk_2) > 255) + { + /* suzeof(uint8) == 256 is max number of chunk in multi-chunks group */ + full_pages= 255; + buffer_rest= full_pages * log_descriptor.page_capacity_chunk_2; + } + /* + group chunks = + full pages + first page (which actually can be full, too. + But here we assign number of chunks - 1 + */ + group.num= full_pages; + if (insert_dynamic(&groups, (gptr) &group)) + { + translog_unlock(); + delete_dynamic(&groups); + UNRECOVERABLE_ERROR(("insert into array failed")); + DBUG_RETURN(1); + } + + DBUG_PRINT("info", ("chunk #%u first_page: %u (%u), full_pages: %u (%lu), " + "Left %lu", + groups.elements, + first_page, first_page - 1, + full_pages, + (ulong) full_pages * + log_descriptor.page_capacity_chunk_2, + parts->record_length - (first_page - 1 + buffer_rest) - + done)); + translog_advance_pointer(full_pages, 0); + + rc|= translog_unlock(); + + if (buffer_to_flush != NULL) + { + rc|= translog_buffer_lock(buffer_to_flush); + translog_buffer_decrease_writers(buffer_to_flush); + if (!rc) + rc= translog_buffer_flush(buffer_to_flush); + rc|= translog_buffer_unlock(buffer_to_flush); + buffer_to_flush= NULL; + } + if (rc) + { + delete_dynamic(&groups); + UNRECOVERABLE_ERROR(("flush of unlock buffer failed")); + DBUG_RETURN(1); + } + + translog_write_data_on_page(&horizon, &cursor, 1, chunk2_header); + translog_write_parts_on_page(&horizon, &cursor, first_page - 1, parts); + DBUG_PRINT("info", ("absolute horizon (%u,0x%lx), local (%u,0x%lx) " + "Left: %lu", + (uint) log_descriptor.horizon.file_no, + (ulong) log_descriptor.horizon.rec_offset, + (uint) horizon.file_no, + (ulong) horizon.rec_offset, + (ulong) (parts->record_length - (first_page - 1) - + done))); + + for (i= 0; i < full_pages; i++) + { + if (translog_write_variable_record_chunk2_page(parts, &horizon, &cursor)) + { + delete_dynamic(&groups); + DBUG_RETURN(1); + } + + DBUG_PRINT("info", ("absolute horizon (%u,0x%lx), local (%u,0x%lx)" + "Left: %lu", + (uint) log_descriptor.horizon.file_no, + (ulong) log_descriptor.horizon.rec_offset, + (uint) horizon.file_no, + (ulong) horizon.rec_offset, + (ulong) (parts->record_length - (first_page - 1) - + i * log_descriptor.page_capacity_chunk_2 - + done))); + } + + done+= (first_page - 1 + buffer_rest); + + /* TODO: made separate function for following */ + rc= translog_page_next(&horizon, &cursor, &buffer_to_flush); + if (buffer_to_flush != NULL) + { + rc|= translog_buffer_lock(buffer_to_flush); + translog_buffer_decrease_writers(buffer_to_flush); + if (!rc) + rc= translog_buffer_flush(buffer_to_flush); + rc|= translog_buffer_unlock(buffer_to_flush); + buffer_to_flush= NULL; + } + if (rc) + { + delete_dynamic(&groups); + UNRECOVERABLE_ERROR(("flush of unlock buffer failed")); + DBUG_RETURN(1); + } + rc= translog_buffer_lock(cursor.buffer); + if (!rc) + translog_buffer_decrease_writers(cursor.buffer); + rc|= translog_buffer_unlock(cursor.buffer); + if (rc) + { + delete_dynamic(&groups); + DBUG_RETURN(1); + } + + translog_lock(); + + first_page= translog_get_current_page_rest(); + buffer_rest= translog_get_current_group_size(); + } while (first_page + buffer_rest < (uint) (parts->record_length - done)); + + group.addr= horizon= log_descriptor.horizon; + cursor= log_descriptor.bc; + cursor.chaser= 1; + group.num= 0; /* 0 because it does not matter + */ + if (insert_dynamic(&groups, (gptr) &group)) + { + delete_dynamic(&groups); + translog_unlock(); + UNRECOVERABLE_ERROR(("insert into array failed")); + DBUG_RETURN(1); + } + record_rest= parts->record_length - done; + DBUG_PRINT("info", ("Record rest: %lu", (ulong) record_rest)); + if (first_page <= record_rest + 1) + { + chunk2_page= 1; + record_rest-= (first_page - 1); + full_pages= record_rest / log_descriptor.page_capacity_chunk_2; + record_rest= (record_rest % log_descriptor.page_capacity_chunk_2); + last_page_capacity= page_capacity; + } + else + { + chunk2_page= full_pages= 0; + last_page_capacity= first_page; + } + chunk3_size= 0; + chunk3_pages= 0; + if (last_page_capacity > record_rest + 1 && record_rest != 0) + { + if (last_page_capacity > + record_rest + header_fixed_part + groups.elements * (7 + 1)) + { + /* 1 record of type 0 */ + chunk3_pages= 0; + } + else + { + chunk3_pages= 1; + if (record_rest + 2 == last_page_capacity) + { + chunk3_size= record_rest - 1; + record_rest= 1; + } + else + { + chunk3_size= record_rest; + record_rest= 0; + } + } + } + /* + A first non-full page will hold type 0 chunk only if it fit in it with + all its headers + */ + while (page_capacity < + record_rest + header_fixed_part + + (groups.elements - groups_per_page * (chunk0_pages - 1)) * (7 + 1)) + chunk0_pages++; + DBUG_PRINT("info", ("chunk0_pages %u, groups %u, groups per full page %u, " + "Group on last page %u", + chunk0_pages, groups.elements, + groups_per_page, + (groups.elements - + ((page_capacity - header_fixed_part) / (7 + 1)) * + (chunk0_pages - 1)))); + DBUG_PRINT("info", ("first_page: %u, chunk2 %u full_pages: %u (%lu), " + "chunk3 %u (%u), rest %u", + first_page, + chunk2_page, full_pages, + (ulong) full_pages * + log_descriptor.page_capacity_chunk_2, + chunk3_pages, (uint) chunk3_size, (uint) record_rest)); + translog_advance_pointer(full_pages + chunk3_pages + + (chunk0_pages - 1), + record_rest + header_fixed_part + + (groups.elements - + ((page_capacity - header_fixed_part) / (7 + 1)) * + (chunk0_pages - 1)) * (7 + 1)); + translog_unlock(); + + if (chunk2_page) + { + DBUG_PRINT("info", ("chunk 2 to finish first page")); + translog_write_data_on_page(&horizon, &cursor, 1, chunk2_header); + translog_write_parts_on_page(&horizon, &cursor, first_page - 1, parts); + DBUG_PRINT("info", ("absolute horizon (%u,0x%lx), local (%u,0x%lx) " + "Left: %lu", + (uint) log_descriptor.horizon.file_no, + (ulong) log_descriptor.horizon.rec_offset, + (uint) horizon.file_no, + (ulong) horizon.rec_offset, + (ulong) (parts->record_length - (first_page - 1) - + done))); + } + else if (chunk3_pages) + { + DBUG_PRINT("info", ("chunk 3")); + DBUG_ASSERT(full_pages == 0); + uchar chunk3_header[3]; + chunk3_header[0]= TRANSLOG_CHUNK_LNGTH; + int2store(chunk3_header + 1, chunk3_size); + translog_write_data_on_page(&horizon, &cursor, 3, chunk3_header); + translog_write_parts_on_page(&horizon, &cursor, chunk3_size, parts); + DBUG_PRINT("info", ("absolute horizon (%u,0x%lx), local (%u,0x%lx) " + "Left: %lu", + (uint) log_descriptor.horizon.file_no, + (ulong) log_descriptor.horizon.rec_offset, + (uint) horizon.file_no, + (ulong) horizon.rec_offset, + (ulong) (parts->record_length - chunk3_size - done))); + chunk3_pages= 0; + } + else + { + DBUG_PRINT("info", ("no new_page_before_chunk0")); + new_page_before_chunk0= 0; + } + + for (i= 0; i < full_pages; i++) + { + DBUG_ASSERT(chunk2_page != 0); + if (translog_write_variable_record_chunk2_page(parts, &horizon, &cursor)) + { + delete_dynamic(&groups); + DBUG_RETURN(1); + } + + DBUG_PRINT("info", ("absolute horizon (%u,0x%lx), local (%u,0x%lx) " + "Left: %lu", + (uint) log_descriptor.horizon.file_no, + (ulong) log_descriptor.horizon.rec_offset, + (uint) horizon.file_no, + (ulong) horizon.rec_offset, + (ulong) (parts->record_length - (first_page - 1) - + i * log_descriptor.page_capacity_chunk_2 - + done))); + } + + if (chunk3_pages && + translog_write_variable_record_chunk3_page(parts, + chunk3_size, + &horizon, &cursor)) + { + delete_dynamic(&groups); + DBUG_RETURN(1); + } + DBUG_PRINT("info", ("absolute horizon (%u,0x%lx), local (%u,0x%lx)", + (uint) log_descriptor.horizon.file_no, + (ulong) log_descriptor.horizon.rec_offset, + (uint) horizon.file_no, (ulong) horizon.rec_offset)); + + + *chunk0_header= (uchar) (type |TRANSLOG_CHUNK_LSN); + int2store(chunk0_header + 1, short_trid); + translog_write_variable_record_1group_code_len(chunk0_header + 3, + parts->record_length, + header_length); + do + { + int limit; + if (new_page_before_chunk0) + { + rc= translog_page_next(&horizon, &cursor, &buffer_to_flush); + if (buffer_to_flush != NULL) + { + rc|= translog_buffer_lock(buffer_to_flush); + translog_buffer_decrease_writers(buffer_to_flush); + if (!rc) + rc= translog_buffer_flush(buffer_to_flush); + rc|= translog_buffer_unlock(buffer_to_flush); + buffer_to_flush= NULL; + } + if (rc) + { + delete_dynamic(&groups); + UNRECOVERABLE_ERROR(("flush of unlock buffer failed")); + DBUG_RETURN(1); + } + } + new_page_before_chunk0= 1; + + if (first_chunk0) + { + *lsn= horizon; + if (log_record_type_descriptor[type].inwrite_hook && + (*log_record_type_descriptor[type].inwrite_hook) (type, tcb, + lsn, parts)) + { + DBUG_RETURN(1); + } + + first_chunk0= 0; + } + + /* + A first non-full page will hold type 0 chunk only if it fit in it with + all its headers => the fist page is full or number of groups less then + possible number of full page. + */ + limit= (groups_per_page < groups.elements - curr_group ? + groups_per_page : groups.elements - curr_group); + DBUG_PRINT("info", ("Groups: %u curr %u, limit %u", + (uint) groups.elements, (uint) curr_group, + (uint) limit)); + + if (chunk0_pages == 1) + { + DBUG_PRINT("info", ("chunk_len: 2 + %u * (7+1) + %u = %u", + (uint) limit, (uint) record_rest, + (uint) (2 + limit * (7 + 1) + record_rest))); + int2store(chunk0_header + header_length - 2, + 2 + limit * (7 + 1) + record_rest); + } + else + { + DBUG_PRINT("info", ("chunk_len: 2 + %u * (7+1) = %u", + (uint) limit, (uint) (2 + limit * (7 + 1)))); + int2store(chunk0_header + header_length - 2, 2 + limit * (7 + 1)); + } + int2store(chunk0_header + header_length, groups.elements - curr_group); + translog_write_data_on_page(&horizon, &cursor, header_fixed_part, + chunk0_header); + for (i= curr_group; i < limit + curr_group; i++) + { + get_dynamic(&groups, (gptr) &group, i); + lsn7store(group_desc, &group.addr); + group_desc[7]= group.num; + translog_write_data_on_page(&horizon, &cursor, (7 + 1), group_desc); + } + + if (chunk0_pages == 1 && record_rest != 0) + translog_write_parts_on_page(&horizon, &cursor, record_rest, parts); + + chunk0_pages--; + curr_group+= limit; + + } while (chunk0_pages != 0); + rc= translog_buffer_lock(cursor.buffer); + if (cmp_translog_addr(cursor.buffer->last_lsn, *lsn) < 0) + cursor.buffer->last_lsn= *lsn; + translog_buffer_decrease_writers(cursor.buffer); + rc|= translog_buffer_unlock(cursor.buffer); + + delete_dynamic(&groups); + DBUG_RETURN(rc); +} + + +/* + Write the variable length log record + + SYNOPSIS + translog_write_variable_record() + lsn LSN of the record will be written here + type the log record type + short_trid Sort transaction ID or 0 if it has no sense + parts Descriptor of record source parts + tcb Transaction control block pointer for hooks by + record log type + + RETURN + 0 - OK + 1 - Error +*/ + +static my_bool translog_write_variable_record(LSN *lsn, + enum translog_record_type type, + SHORT_TRANSACTION_ID short_trid, + struct st_translog_parts *parts, + void *tcb) +{ + struct st_translog_buffer *buffer_to_flush= NULL; + uint header_length1= 1 + 2 + 2 + + translog_variable_record_length_bytes(parts->record_length); + ulong buffer_rest; + uint page_rest; + uchar compressed_LSNs[2 * 7]; /* Max number of such LSNs per + record is 2 */ + + DBUG_ENTER("translog_write_variable_record"); + + translog_lock(); + DBUG_PRINT("info", ("horizon (%u,0x%lx)", + (uint) log_descriptor.horizon.file_no, + (ulong) log_descriptor.horizon.rec_offset)); + page_rest= TRANSLOG_PAGE_SIZE - log_descriptor.bc.current_page_size; + DBUG_PRINT("info", ("header length %u, page_rest: %u", + header_length1, page_rest)); + + /* + header and part which we should read have to fit in one chunk + TODO: allow to divide readable header + */ + if (page_rest < + (header_length1 + log_record_type_descriptor[type].read_header_len)) + { + DBUG_PRINT("info", + ("Next page, size: %u, header: %u + %u", + log_descriptor.bc.current_page_size, + header_length1, + log_record_type_descriptor[type].read_header_len)); + translog_page_next(&log_descriptor.horizon, &log_descriptor.bc, + &buffer_to_flush); + page_rest= log_descriptor.page_capacity_chunk_2 + 1; + DBUG_PRINT("info", ("page_rest: %u", page_rest)); + } + + /* + To minimize compressed size we will compress always relative to + very first chunk address (log_descriptor.horizon for now) + */ + if (log_record_type_descriptor[type].compresed_LSN > 0) + { + if (translog_relative_LSN_encode(parts, &log_descriptor.horizon, + log_record_type_descriptor[type]. + compresed_LSN, compressed_LSNs)) + { + int rc= translog_unlock(); + if (buffer_to_flush != NULL) + { + if (!rc) + rc= translog_buffer_flush(buffer_to_flush); + rc|= translog_buffer_unlock(buffer_to_flush); + } + DBUG_RETURN(1); + } + /* recalculate header length after compression */ + header_length1= 1 + 2 + 2 + + translog_variable_record_length_bytes(parts->record_length); + DBUG_PRINT("info", ("after compressing LSN(s) header length %u, " + "record length %lu", + header_length1, parts->record_length)); + } + + /* TODO: check space on current page for header + few bytes */ + if (page_rest >= parts->record_length + header_length1) + { + /* following function makes translog_unlock(); */ + DBUG_RETURN(translog_write_variable_record_1chunk(lsn, type, short_trid, + parts, buffer_to_flush, + header_length1, tcb)); + } + + buffer_rest= translog_get_current_group_size(); + + if (buffer_rest >= parts->record_length + header_length1 - page_rest) + { + /* following function makes translog_unlock(); */ + DBUG_RETURN(translog_write_variable_record_1group(lsn, type, short_trid, + parts, buffer_to_flush, + header_length1, tcb)); + } + /* following function makes translog_unlock(); */ + DBUG_RETURN(translog_write_variable_record_mgroup(lsn, type, short_trid, + parts, buffer_to_flush, + header_length1, + buffer_rest, tcb)); + DBUG_RETURN(0); +} + + +/* + Write the fixed and pseudo-fixed log record + + SYNOPSIS + translog_write_fixed_record() + lsn LSN of the record will be written here + type the log record type + short_trid Sort transaction ID or 0 if it has no sense + parts Descriptor of record source parts + tcb Transaction control block pointer for hooks by + record log type + + RETURN + 0 - OK + 1 - Error +*/ + +static my_bool translog_write_fixed_record(LSN *lsn, + enum translog_record_type type, + SHORT_TRANSACTION_ID short_trid, + struct st_translog_parts *parts, + void *tcb) +{ + struct st_translog_buffer *buffer_to_flush= NULL; + uchar chunk1_header[1 + 2]; + uchar compressed_LSNs[2 * 7]; /* Max number of such LSNs per + record is 2 */ + struct st_translog_part part; + int rc; + DBUG_ENTER("translog_write_fixed_record"); + DBUG_ASSERT((log_record_type_descriptor[type].class == + LOGRECTYPE_FIXEDLENGTH && + parts->record_length == + log_record_type_descriptor[type].fixed_length) || + (log_record_type_descriptor[type].class == + LOGRECTYPE_PSEUDOFIXEDLENGTH && + (parts->record_length - + log_record_type_descriptor[type].compresed_LSN * 2) <= + log_record_type_descriptor[type].fixed_length)); + + translog_lock(); + DBUG_PRINT("info", ("horizon (%u,0x%lx)", + (uint) log_descriptor.horizon.file_no, + (ulong) log_descriptor.horizon.rec_offset)); + + DBUG_ASSERT(log_descriptor.bc.current_page_size <= TRANSLOG_PAGE_SIZE); + DBUG_PRINT("info", + ("Page size: %u, record %u, next cond %d", + log_descriptor.bc.current_page_size, + (parts->record_length - + log_record_type_descriptor[type].compresed_LSN * 2 + 3), + ((((uint) log_descriptor.bc.current_page_size) + + (parts->record_length - + log_record_type_descriptor[type].compresed_LSN * 2 + 3)) > + TRANSLOG_PAGE_SIZE))); + /* + check that there is enough place on current page: + (log_record_type_descriptor[type].fixed_length - economized on compressed + LSNs) bytes + */ + if ((((uint) log_descriptor.bc.current_page_size) + + (parts->record_length - + log_record_type_descriptor[type].compresed_LSN * 2 + 3)) > + TRANSLOG_PAGE_SIZE) + { + DBUG_PRINT("info", ("Next page")); + translog_page_next(&log_descriptor.horizon, &log_descriptor.bc, + &buffer_to_flush); + } + + *lsn= log_descriptor.horizon; + if (log_record_type_descriptor[type].inwrite_hook && + (*log_record_type_descriptor[type].inwrite_hook) (type, tcb, + lsn, parts)) + { + DBUG_RETURN(1); + } + + + /* compress LSNs */ + if (log_record_type_descriptor[type].class == LOGRECTYPE_PSEUDOFIXEDLENGTH) + { + DBUG_ASSERT(log_record_type_descriptor[type].compresed_LSN > 0); + if (translog_relative_LSN_encode(parts, lsn, + log_record_type_descriptor[type]. + compresed_LSN, compressed_LSNs)) + { + rc= 1; + goto err; + } + } + + /* + Write the whole record at once (we sure that there is enough place on + the destination page + */ + DBUG_ASSERT(parts->current != 0); /* first part is left for + header */ + parts->total_record_length+= (part.len= 1 + 2); + part.buff= chunk1_header; + *chunk1_header= (uchar) (type |TRANSLOG_CHUNK_FIXED); + int2store(chunk1_header + 1, short_trid); + parts->current--; + set_dynamic(&parts->parts, (gptr) &part, parts->current); + + rc= translog_write_parts_on_page(&log_descriptor.horizon, + &log_descriptor.bc, + parts->total_record_length, parts); + + log_descriptor.bc.buffer->last_lsn= *lsn; +err: + rc|= translog_unlock(); + + /* + check if we switched buffer and need process it (current buffer is + unlocked already => we will not delay other threads + */ + if (buffer_to_flush != NULL) + { + if (!rc) + rc= translog_buffer_flush(buffer_to_flush); + rc|= translog_buffer_unlock(buffer_to_flush); + } + + DBUG_RETURN(rc); +} + + +/* + Write the log record + + SYNOPSIS + translog_write_record() + lsn LSN of the record will be written here + type the log record type + short_trid Sort transaction ID or 0 if it has no sense + tcb Transaction control block pointer for hooks by + record log type + partN_length length of Ns part of the log + partN_buffer pointer on Ns part buffer + 0 sign of the end of parts + + RETURN + 0 - OK + 1 - Error +*/ + +my_bool translog_write_record(LSN *lsn, + enum translog_record_type type, + SHORT_TRANSACTION_ID short_trid, + void *tcb, + translog_size_t part1_length, + uchar *part1_buff, ...) +{ + struct st_translog_parts parts; + va_list pvar; + int rc; + DBUG_ENTER("translog_write_record"); + DBUG_PRINT("enter", ("type %u, ShortTrID %u", (uint) type, (uint)short_trid)); + + /* move information about parts into dynamic array */ + if (init_dynamic_array(&parts.parts, sizeof(struct st_translog_part), + 10, 10 CALLER_INFO)) + { + UNRECOVERABLE_ERROR(("init array failed")); + DBUG_RETURN(1); + } + { + struct st_translog_part part; + + /* reserve place for header */ + parts.current= 1; + part.len= 0; + part.buff= 0; + if (insert_dynamic(&parts.parts, (gptr) &part)) + { + UNRECOVERABLE_ERROR(("insert into array failed")); + DBUG_RETURN(1); + } + + parts.record_length= part.len= part1_length; + part.buff= part1_buff; + if (insert_dynamic(&parts.parts, (gptr) &part)) + { + UNRECOVERABLE_ERROR(("insert into array failed")); + DBUG_RETURN(1); + } + DBUG_PRINT("info", ("record length: %lu, %lu ...", + (ulong) parts.record_length, + (ulong) parts.total_record_length)); + + /* count record length */ + va_start(pvar, part1_buff); + for (;;) + { + part.len= va_arg(pvar, translog_size_t); + if (part.len == 0) + break; + parts.record_length+= part.len; + part.buff= va_arg(pvar, uchar*); + if (insert_dynamic(&parts.parts, (gptr) &part)) + { + UNRECOVERABLE_ERROR(("insert into array failed")); + DBUG_RETURN(1); + } + DBUG_PRINT("info", ("record length: %lu, %lu ...", + (ulong) parts.record_length, + (ulong) parts.total_record_length)); + } + va_end(pvar); + + /* + start total_record_length from record_length then overhead will + be add + */ + parts.total_record_length= parts.record_length; + } + va_end(pvar); + DBUG_PRINT("info", ("record length: %lu, %lu", + (ulong) parts.record_length, + (ulong) parts.total_record_length)); + + /* process this parts */ + if (!(rc= (log_record_type_descriptor[type].prewrite_hook && + (*log_record_type_descriptor[type].prewrite_hook) (type, tcb, + &parts)))) + { + switch (log_record_type_descriptor[type].class) + { + case LOGRECTYPE_VARIABLE_LENGTH: + { + rc= translog_write_variable_record(lsn, type, short_trid, &parts, tcb); + break; + } + case LOGRECTYPE_PSEUDOFIXEDLENGTH: + case LOGRECTYPE_FIXEDLENGTH: + { + rc= translog_write_fixed_record(lsn, type, short_trid, &parts, tcb); + break; + } + case LOGRECTYPE_NOT_ALLOWED: + default: + DBUG_ASSERT(0); + rc= 1; + } + } + + delete_dynamic(&parts.parts); + DBUG_RETURN(rc); +} + + +/* + Decode compressed (relative) LSN(s) + + SYNOPSIS + translog_relative_lsn_decode() + base_lsn LSN for encoding + src Decode LSN(s) from here + dst Put decoded LSNs here + lsns number of LSN(s) + + RETURN + position in sources after decoded LSN(s) +*/ + +static uchar *translog_relative_LSN_decode(LSN *base_lsn, + uchar *src, uchar *dst, uint lsns) +{ + uint i; + for (i= 0; i < lsns; i++, dst+= 7) + { + src= translog_get_LSN_from_diff(base_lsn, src, dst); + } + return src; +} + +/* + Get header of fixed/pseudo length record and call hook for it processing + + SYNOPSIS + translog_fixed_length_header() + page Pointer to the buffer with page where LSN chunk is + placed + page_offset Offset of the first chunk in the page + buff Buffer to be filled with header data + + RETURN + 0 - error + number of bytes in TRANSLOG_HEADER_BUFFER::header where stored decoded + part of the header +*/ + +translog_size_t translog_fixed_length_header(uchar *page, + translog_size_t page_offset, + TRANSLOG_HEADER_BUFFER *buff) +{ + struct st_log_record_type_descriptor *desc= + log_record_type_descriptor + buff->type; + uchar *src= page + page_offset + 3; + uchar *dst= buff->header; + uchar *start= src; + uint lsns= desc->compresed_LSN; + uint length= desc->fixed_length + (lsns * 2); + + DBUG_ENTER("translog_fixed_length_header"); + + buff->record_length= length; + + if (desc->class == LOGRECTYPE_PSEUDOFIXEDLENGTH) + { + DBUG_ASSERT(lsns > 0); + src= translog_relative_LSN_decode(&buff->lsn, src, dst, lsns); + lsns*= 7; + dst+= lsns; + length-= lsns; + buff->compressed_LSN_economy= (uint16) (lsns - (src - start)); + } + else + buff->compressed_LSN_economy= 0; + + memmove(dst, src, length); + buff->non_header_data_start_offset= page_offset + + ((src + length) - (page + page_offset)); + buff->non_header_data_len= 0; + DBUG_RETURN(buff->record_length); +} + + +/* + Free resources used by TRANSLOG_HEADER_BUFFER + + SYNOPSIS + translog_free_record_header(); +*/ + +void translog_free_record_header(TRANSLOG_HEADER_BUFFER *buff) +{ + DBUG_ENTER("translog_free_record_header"); + if (buff->groups_no != 0) + { + my_free((gptr) buff->groups, MYF(0)); + buff->groups_no= 0; + } + DBUG_VOID_RETURN; +} + + +/* + Set current horizon in the scanner data structure + + SYNOPSIS + translog_scanner_set_horizon() + scanner Information about current chunk during scanning +*/ + +static void translog_scanner_set_horizon(struct st_translog_scanner_data + *scanner) +{ + translog_lock(); + scanner->horizon= log_descriptor.horizon; + translog_unlock(); +} + + +/* + Set last page in the scanner data structure + + SYNOPSIS + translog_scanner_set_last_page() + scanner Information about current chunk during scanning + + RETURN + 0 - OK + 1 - Error +*/ + +static my_bool translog_scanner_set_last_page(struct st_translog_scanner_data + *scanner) +{ + my_bool page_ok; + scanner->last_file_page= scanner->page_addr; + if (translog_get_last_page_addr(&scanner->last_file_page, &page_ok)) + return (1); + return (0); +} + + +/* + Init scanner + + SYNOPSIS + translog_init_scanner() + lsn LSN with which it have to be inited + fixed_horizon true if it is OK do not read records which was written + after scanning beginning + scanner scanner which have to be inited + + RETURN + 0 - OK + 1 - Error +*/ +static my_bool translog_init_scanner(LSN *lsn, + my_bool fixed_horizon, + struct st_translog_scanner_data *scanner) +{ + TRANSLOG_VALIDATOR_DATA data= + { + &scanner->page_addr, 0 + }; + + DBUG_ENTER("translog_init_scanner"); + DBUG_PRINT("enter", ("LSN: (0x%lx,0x%lx)", + (ulong) lsn->file_no, (ulong) lsn->rec_offset)); + DBUG_ASSERT(lsn->rec_offset % TRANSLOG_PAGE_SIZE != 0); + scanner->page_offset= lsn->rec_offset % TRANSLOG_PAGE_SIZE; + + scanner->fixed_horizon= fixed_horizon; + + translog_scanner_set_horizon(scanner); + DBUG_PRINT("info", ("Horizon: (0x%lx,0x%lx)", + (ulong) scanner->horizon.file_no, + (ulong) scanner->horizon.rec_offset)); + + /* lsn < horizon */ + DBUG_ASSERT(lsn->file_no < scanner->horizon.file_no || + (lsn->file_no == scanner->horizon.file_no && + lsn->rec_offset < scanner->horizon.rec_offset)); + + scanner->page_addr= *lsn; + scanner->page_addr.rec_offset-= scanner->page_offset; + + if (translog_scanner_set_last_page(scanner)) + DBUG_RETURN(1); + + if ((scanner->page= translog_get_page(&data, scanner->buffer)) == NULL) + DBUG_RETURN(1); + DBUG_RETURN(0); +} + + +/* + Checks End of the Log + + SYNOPSIS + translog_scanner_eol() + scanner Information about current chunk during scanning + + RETURN + 1 - End of the Log + 0 - OK +*/ +static my_bool translog_scanner_eol(struct st_translog_scanner_data *scanner) +{ + DBUG_ENTER("translog_scanner_eol"); + DBUG_PRINT("enter", + ("Horizon: (%lu, 0x%lx), Current: (%lu, 0x%lx+0x%x=0x%lx)", + (ulong) scanner->horizon.file_no, + (ulong) scanner->horizon.rec_offset, + (ulong) scanner->page_addr.file_no, + (ulong) scanner->page_addr.rec_offset, + (uint) scanner->page_offset, + (ulong) (scanner->page_addr.rec_offset + scanner->page_offset))); + if (scanner->horizon.file_no > scanner->page_addr.file_no || + (scanner->horizon.file_no == scanner->page_addr.file_no && + scanner->horizon.rec_offset > (scanner->page_addr.rec_offset + + scanner->page_offset))) + { + DBUG_PRINT("info", ("Horizon is not reached")); + DBUG_RETURN(0); + } + if (scanner->fixed_horizon) + { + DBUG_PRINT("info", ("Horizon is fixed and reached")); + DBUG_RETURN(1); + } + translog_scanner_set_horizon(scanner); + DBUG_PRINT("info", + ("Horizon is re-read, EOL: %d", + scanner->horizon.file_no <= scanner->page_addr.file_no && + (scanner->horizon.file_no != scanner->page_addr.file_no || + scanner->horizon.rec_offset <= (scanner->page_addr.rec_offset + + scanner->page_offset)))); + DBUG_RETURN(scanner->horizon.file_no <= scanner->page_addr.file_no && + (scanner->horizon.file_no != scanner->page_addr.file_no || + scanner->horizon.rec_offset <= (scanner->page_addr.rec_offset + + scanner->page_offset))); +} + + +/* + Cheks End of the Page + + SYNOPSIS + translog_scanner_eop() + scanner Information about current chunk during scanning + + RETURN + 1 - End of the Page + 0 - OK +*/ +static my_bool translog_scanner_eop(struct st_translog_scanner_data *scanner) +{ + DBUG_ENTER("translog_scanner_eop"); + DBUG_RETURN(scanner->page_offset >= TRANSLOG_PAGE_SIZE || + scanner->page[scanner->page_offset] == 0); +} + + +/* + Checks End of the File (I.e. we are scanning last page, which do not + mean end of this page) + + SYNOPSIS + translog_scanner_eof() + scanner Information about current chunk during scanning + + RETURN + 1 - End of the File + 0 - OK +*/ +static my_bool translog_scanner_eof(struct st_translog_scanner_data *scanner) +{ + DBUG_ENTER("translog_scanner_eof"); + DBUG_ASSERT(scanner->page_addr.file_no == scanner->last_file_page.file_no); + DBUG_PRINT("enter", ("curr Page 0x%lx, last page 0x%lx, " + "normal EOF %d", + scanner->page_addr.rec_offset, + scanner->last_file_page.rec_offset, + scanner->page_addr.rec_offset == + scanner->last_file_page.rec_offset)); + /* + TODO: detect damaged file EOF, + TODO: issue warning if damaged file EOF detected + */ + DBUG_RETURN(scanner->page_addr.rec_offset == + scanner->last_file_page.rec_offset); +} + + +/* + Move scanner to the next chunk + + SYNOPSIS + translog_get_next_chunk() + scanner Information about current chunk during scanning + + RETURN + 0 - OK + 1 - Error +*/ + +static my_bool translog_get_next_chunk(struct st_translog_scanner_data *scanner) +{ + DBUG_ENTER("translog_get_next_chunk"); + uint16 len= translog_get_total_chunk_length(scanner->page, + scanner->page_offset); + if (len == 0) + DBUG_RETURN(1); + scanner->page_offset+= len; + + if (translog_scanner_eol(scanner)) + { + scanner->page= &end_of_log; + scanner->page_offset= 0; + DBUG_RETURN(0); + } + if (translog_scanner_eop(scanner)) + { + if (translog_scanner_eof(scanner)) + { + DBUG_PRINT("info", ("horizon (%lu,0x%lx) pageaddr (%lu,0x%lx)", + (ulong) scanner->horizon.file_no, + (ulong) scanner->horizon.rec_offset, + (ulong) scanner->page_addr.file_no, + (ulong) scanner->page_addr.rec_offset)); + /* if it is log end it have to be caught before */ + DBUG_ASSERT(scanner->horizon.file_no > scanner->page_addr.file_no); + scanner->page_addr.file_no++; + scanner->page_addr.rec_offset= TRANSLOG_PAGE_SIZE; + if (translog_scanner_set_last_page(scanner)) + DBUG_RETURN(1); + } + else + { + scanner->page_addr.rec_offset+= TRANSLOG_PAGE_SIZE; + } + { + TRANSLOG_VALIDATOR_DATA data= + { + &scanner->page_addr, 0 + }; + if ((scanner->page= translog_get_page(&data, scanner->buffer)) == NULL) + DBUG_RETURN(1); + } + scanner->page_offset= translog_get_first_chunk_offset(scanner->page); + if (translog_scanner_eol(scanner)) + { + scanner->page= &end_of_log; + scanner->page_offset= 0; + DBUG_RETURN(0); + } + DBUG_ASSERT(scanner->page[scanner->page_offset] != 0); + } + DBUG_RETURN(0); +} + + +/* + Get header of variable length record and call hook for it processing + + SYNOPSIS + translog_variable_length_header() + page Pointer to the buffer with page where LSN chunk is + placed + page_offset Offset of the first chunk in the page + buff Buffer to be filled with header data + scanner If present should be moved to the header page if + it differ from LSN page + + RETURN + 0 - error + number of bytes in TRANSLOG_HEADER_BUFFER::header where stored decoded + part of the header +*/ + +translog_size_t translog_variable_length_header(uchar *page, + translog_size_t page_offset, + TRANSLOG_HEADER_BUFFER *buff, + struct + st_translog_scanner_data + *scanner) +{ + struct st_log_record_type_descriptor *desc= + log_record_type_descriptor + buff->type; + uchar *src= page + page_offset + 1 + 2; + uchar *dst= buff->header; + LSN base_lsn; + uint lsns= desc->compresed_LSN; + uint16 chunk_len; + uint16 length= desc->read_header_len + (lsns * 2); + uint16 buffer_length= length; + uint16 body_len; + struct st_translog_scanner_data internal_scanner; + + DBUG_ENTER("translog_variable_length_header"); + + buff->record_length= translog_variable_record_1group_decode_len(&src); + chunk_len= uint2korr(src); + DBUG_PRINT("info", ("rec len: %lu, chunk len: %u, length %u, bufflen %u", + (ulong) buff->record_length, (uint) chunk_len, + (uint) length, (uint) buffer_length)); + if (chunk_len == 0) + { + uint16 page_rest; + DBUG_PRINT("info", ("1 group")); + src+= 2; + page_rest= TRANSLOG_PAGE_SIZE - (src - page); + + base_lsn= buff->lsn; + body_len= (page_rest < buff->record_length ? + page_rest : buff->record_length); + } + else + { + uint grp_no, curr; + uint header_to_skip; + uint16 page_rest; + + DBUG_PRINT("info", ("multi-group")); + grp_no= buff->groups_no= uint2korr(src + 2); + if ((buff->groups= + (TRANSLOG_GROUP*) my_malloc(sizeof(TRANSLOG_GROUP) * buff->groups_no, + MYF(0))) == 0) + DBUG_RETURN(0); + DBUG_PRINT("info", ("Groups: %u", (uint) grp_no)); + src+= (2 + 2); + page_rest= TRANSLOG_PAGE_SIZE - (src - page); + curr= 0; + header_to_skip= src - (page + page_offset); + buff->chunk0_pages= 0; + + for (;;) + { + uint i; + uint read= grp_no; + + buff->chunk0_pages++; + if (page_rest < grp_no * (7 + 1)) + read= page_rest / (7 + 1); + DBUG_PRINT("info", ("Read chunk0 page#%u read %u left %u start from %u", + buff->chunk0_pages, read, grp_no, curr)); + for (i= 0; i < read; i++, curr++) + { + DBUG_ASSERT(curr < buff->groups_no); + lsn7korr(&buff->groups[curr].addr, src + i * (7 + 1)); + buff->groups[curr].num= src[i * (7 + 1) + 7]; + DBUG_PRINT("info", ("group #%u (%u,0x%lx) chunks %u", + curr, + (uint) buff->groups[curr].addr.file_no, + (ulong) buff->groups[curr].addr.rec_offset, + (uint) buff->groups[curr].num)); + } + grp_no-= read; + if (grp_no == 0) + { + if (scanner) + { + buff->chunk0_data_addr= scanner->page_addr; + buff->chunk0_data_addr.rec_offset+= (page_offset + header_to_skip + + i * (7 + 1)); + } + else + { + buff->chunk0_data_addr= buff->lsn; + buff->chunk0_data_addr.rec_offset+= (header_to_skip + i * (7 + 1)); + } + buff->chunk0_data_len= chunk_len - 2 - i * (7 + 1); + DBUG_PRINT("info", ("Data address (%u,0x%lx), len: %u", + (uint) buff->chunk0_data_addr.file_no, + (ulong) buff->chunk0_data_addr.rec_offset, + buff->chunk0_data_len)); + break; + } + if (scanner == NULL) + { + DBUG_PRINT("info", ("use internal scanner for header reding")); + scanner= &internal_scanner; + translog_init_scanner(&buff->lsn, 1, scanner); + } + translog_get_next_chunk(scanner); + page= scanner->page; + page_offset= scanner->page_offset; + src= page + page_offset + header_to_skip; + chunk_len= uint2korr(src - 2 - 2); + DBUG_PRINT("info", ("Chunk len: %u", (uint) chunk_len)); + page_rest= TRANSLOG_PAGE_SIZE - (src - page); + } + + if (scanner == NULL) + { + DBUG_PRINT("info", ("use internal scanner")); + scanner= &internal_scanner; + } + + base_lsn= buff->groups[0].addr; + translog_init_scanner(&base_lsn, 1, scanner); + /* first group chunk is always chunk type 2 */ + page= scanner->page; + page_offset= scanner->page_offset; + src= page + page_offset + 1; + page_rest= TRANSLOG_PAGE_SIZE - (src - page); + body_len= page_rest; + } + if (lsns) + { + uchar *start= src; + src= translog_relative_LSN_decode(&base_lsn, src, dst, lsns); + lsns*= 7; + dst+= lsns; + length-= lsns; + buff->record_length+= (buff->compressed_LSN_economy= + (uint16) (lsns - (src - start))); + DBUG_PRINT("info", ("lsns: %u, length %u, economy %u, new length %lu", + lsns / 7, (uint) length, + (uint) buff->compressed_LSN_economy, + (ulong) buff->record_length)); + body_len-= (src - start); + } + else + buff->compressed_LSN_economy= 0; + + DBUG_ASSERT(body_len >= length); + body_len-= length; + memmove(dst, src, length); + buff->non_header_data_start_offset= src + length - page; + buff->non_header_data_len= body_len; + DBUG_PRINT("info", ("non_header_data_start_offset %u len %u buffer %u", + buff->non_header_data_start_offset, + buff->non_header_data_len, buffer_length)); + DBUG_RETURN(buffer_length); +} + + +/* + Read record header from the given buffer + + SYNOPSIS + translog_read_record_header_from_buffer() + page page content buffer + page_offset offset of the chunk in the page + buff destination buffer + scanner if it is need this scanner will be moved to the + record header page (differ from LSN page in case of + multi-group records +*/ + +translog_size_t +translog_read_record_header_from_buffer(uchar *page, + uint16 page_offset, + TRANSLOG_HEADER_BUFFER *buff, + struct + st_translog_scanner_data *scanner) +{ + DBUG_ENTER("translog_read_record_header_from_buffer"); + DBUG_ASSERT((page[page_offset] & TRANSLOG_CHUNK_TYPE) == + TRANSLOG_CHUNK_LSN || + (page[page_offset] & TRANSLOG_CHUNK_TYPE) == + TRANSLOG_CHUNK_FIXED); + buff->type= (page[page_offset] & TRANSLOG_REC_TYPE); + buff->short_trid= uint2korr(page + page_offset + 1); + DBUG_PRINT("info", ("Type %u, Sort TrID %u, LSN (%u,0x%lx)", + (uint) buff->type, (uint)buff->short_trid, + buff->lsn.file_no, buff->lsn.rec_offset)); + /* Read required bytes from the header and call hook */ + switch (log_record_type_descriptor[buff->type].class) + { + case LOGRECTYPE_VARIABLE_LENGTH: + DBUG_RETURN(translog_variable_length_header(page, page_offset, buff, + scanner)); + case LOGRECTYPE_PSEUDOFIXEDLENGTH: + case LOGRECTYPE_FIXEDLENGTH: + DBUG_RETURN(translog_fixed_length_header(page, page_offset, buff)); + default: + DBUG_ASSERT(0); + } + DBUG_RETURN(0); +} + + +/* + Read record header and some fixed part of a record (the part depend on + record type). + + SYNOPSIS + translog_read_record_header() + lsn log record serial number (address of the record) + buff log record header buffer + + NOTE + - lsn can point to TRANSLOG_HEADER_BUFFER::lsn and it will be processed + correctly. + - Some type of record can be read completely by this call + - "Decoded" header stored in TRANSLOG_HEADER_BUFFER::header (relative + LSN can be translated to absolute one), some fields can be added + (like actual header length in the record if the header has variable + length) + + RETURN + 0 - error + number of bytes in TRANSLOG_HEADER_BUFFER::header where stored decoded + part of the header +*/ + +translog_size_t translog_read_record_header(LSN *lsn, + TRANSLOG_HEADER_BUFFER *buff) +{ + uchar buffer[TRANSLOG_PAGE_SIZE], *page; + translog_size_t page_offset= lsn->rec_offset % TRANSLOG_PAGE_SIZE; + + DBUG_ENTER("translog_read_record_header"); + DBUG_PRINT("enter", ("LSN: (0x%lx,0x%lx)", + (ulong) lsn->file_no, (ulong) lsn->rec_offset)); + DBUG_ASSERT(lsn->rec_offset % TRANSLOG_PAGE_SIZE != 0); + + buff->lsn= *lsn; + buff->groups_no= 0; + { + TRANSLOG_ADDRESS addr= *lsn; + TRANSLOG_VALIDATOR_DATA data= + { + &addr, 0 + }; + addr.rec_offset-= page_offset; + if ((page= translog_get_page(&data, buffer)) == NULL) + DBUG_RETURN(0); + } + + DBUG_RETURN(translog_read_record_header_from_buffer(page, page_offset, + buff, 0)); +} + + +/* + Read record header and some fixed part of a record (the part depend on + record type). + + SYNOPSIS + translog_read_record_header_scan() + scan scanner position to read + buff log record header buffer + move_scanner request to move scanner to the header position + + NOTE + - Some type of record can be read completely by this call + - "Decoded" header stored in TRANSLOG_HEADER_BUFFER::header (relative + LSN can be translated to absolute one), some fields can be added + (like actual header length in the record if the header has variable + length) + + RETURN + 0 - error + number of bytes in TRANSLOG_HEADER_BUFFER::header where stored decoded + part of the header +*/ + +translog_size_t +translog_read_record_header_scan(struct st_translog_scanner_data + *scanner, + TRANSLOG_HEADER_BUFFER *buff, + my_bool move_scanner) +{ + DBUG_ENTER("translog_read_record_header_scan"); + DBUG_PRINT("enter", ("Scanner: Cur: (%u, 0x%lx), Hrz: (%u, 0x%lx), " + "Lst: (%u, 0x%lx), Offset: %u(%x), fixed %d", + (uint) scanner->page_addr.file_no, + (ulong) scanner->page_addr.rec_offset, + (uint) scanner->horizon.file_no, + (ulong) scanner->horizon.rec_offset, + (uint) scanner->last_file_page.file_no, + (ulong) scanner->last_file_page.rec_offset, + (uint) scanner->page_offset, + (uint) scanner->page_offset, scanner->fixed_horizon)); + buff->groups_no= 0; + buff->lsn= scanner->page_addr; + buff->lsn.rec_offset+= scanner->page_offset; + DBUG_RETURN(translog_read_record_header_from_buffer(scanner->page, + scanner->page_offset, + buff, + (move_scanner ? + scanner : 0))); +} + + +/* + Read record header and some fixed part of the next record (the part + depend on record type). + + SYNOPSIS + translog_read_next_record_header() + lsn log record serial number (address of the record) + previous to the record which will be read + If LSN present scanner will be initialized from it, + do not use LSN after initialization for fast scanning. + buff log record header buffer + fixed_horizon true if it is OK do not read records which was written + after scanning beginning + scanner data for scanning if lsn is NULL scanner data + will be used for continue scanning. + The scanner can be NULL. + + NOTE + - lsn can point to TRANSLOG_HEADER_BUFFER::lsn and it will be processed + correctly (lsn in buffer will be replaced by next record, but initial + lsn will be read correctly). + - it is like translog_read_record_header, but read next record, so see + its NOTES. + - in case of end of the log buff->lsn will be set to + (CONTROL_FILE_IMPOSSIBLE_FILENO, 0) + RETURN + 0 - error + TRANSLOG_RECORD_HEADER_MAX_SIZE + 1 - End of the log + number of bytes in TRANSLOG_HEADER_BUFFER::header where stored decoded + part of the header +*/ +translog_size_t translog_read_next_record_header(LSN *lsn, + TRANSLOG_HEADER_BUFFER *buff, + my_bool fixed_horizon, + struct + st_translog_scanner_data + *scanner) +{ + struct st_translog_scanner_data internal_scanner; + uint8 chunk_type; + + buff->groups_no= 0; /* to be sure that we will free + it right */ + + DBUG_ENTER("translog_read_next_record_header"); + DBUG_PRINT("enter", ("scanner: 0x%lx", (ulong) scanner)); + if (scanner == NULL) + { + DBUG_ASSERT(lsn != NULL); + scanner= &internal_scanner; + } + if (lsn) + { + if (translog_init_scanner(lsn, fixed_horizon, scanner)) + DBUG_RETURN(0); + DBUG_ASSERT(lsn->rec_offset % TRANSLOG_PAGE_SIZE != 0); + } + DBUG_PRINT("info", ("Scanner: Cur: (%u, 0x%lx), Hrz: (%u, 0x%lx), " + "Lst: (%u, 0x%lx), Offset: %u(%x), fixed %d", + (uint) scanner->page_addr.file_no, + (ulong) scanner->page_addr.rec_offset, + (uint) scanner->horizon.file_no, + (ulong) scanner->horizon.rec_offset, + (uint) scanner->last_file_page.file_no, + (ulong) scanner->last_file_page.rec_offset, + (uint) scanner->page_offset, + (uint) scanner->page_offset, scanner->fixed_horizon)); + + do + { + if (translog_get_next_chunk(scanner)) + DBUG_RETURN(0); + chunk_type= scanner->page[scanner->page_offset] & TRANSLOG_CHUNK_TYPE; + DBUG_PRINT("info", ("type %x, byte %x", (uint) chunk_type, + (uint) scanner->page[scanner->page_offset])); + } while (chunk_type != TRANSLOG_CHUNK_LSN && chunk_type != + TRANSLOG_CHUNK_FIXED && scanner->page[scanner->page_offset] != 0); + + if (scanner->page[scanner->page_offset] == 0) + { + /* Last record was read */ + buff->lsn.file_no= CONTROL_FILE_IMPOSSIBLE_FILENO; + buff->lsn.rec_offset= 0; + DBUG_RETURN(TRANSLOG_RECORD_HEADER_MAX_SIZE + 1); /* just it is not error + */ + } + DBUG_RETURN(translog_read_record_header_scan(scanner, buff, 0)); +} + + +/* + Moves record data reader to the next chunk and fill the data reader + information about that chunk. + + SYNOPSIS + translog_record_read_next_chunk() + data data cursor + + RETURN + 0 - OK + 1 - Error +*/ +static my_bool translog_record_read_next_chunk(struct st_translog_reader_data + *data) +{ + translog_size_t new_current_offset= data->current_offset + data->chunk_size; + uint16 chunk_header_len, chunk_len; + uint8 type; + + DBUG_ENTER("translog_record_read_next_chunk"); + + if (data->eor) + { + DBUG_PRINT("info", ("end of the record flag set")); + DBUG_RETURN(1); + } + + if (data->header.groups_no && + data->header.groups_no - 1 != data->current_group && + data->header.groups[data->current_group].num == data->current_chunk) + { + /* Goto next group */ + data->current_group++; + data->current_chunk= 0; + DBUG_PRINT("info", ("skip to group #%u", data->current_group)); + translog_init_scanner(&data->header.groups[data->current_group].addr, + 1, &data->scanner); + } + else + { + data->current_chunk++; + if (translog_get_next_chunk(&data->scanner)) + DBUG_RETURN(1); + } + type= data->scanner.page[data->scanner.page_offset] & TRANSLOG_CHUNK_TYPE; + + if (type == TRANSLOG_CHUNK_LSN && data->header.groups_no) + { + DBUG_PRINT("info", + ("Last chunk: data len %u, offset %u group %u of %u", + data->header.chunk0_data_len, data->scanner.page_offset, + data->current_group, data->header.groups_no - 1)); + DBUG_ASSERT(data->header.groups_no - 1 == data->current_group); + DBUG_ASSERT(data->header.lsn.file_no == data->scanner.page_addr.file_no && + data->header.lsn.rec_offset == + data->scanner.page_addr.rec_offset + data->scanner.page_offset); + translog_init_scanner(&data->header.chunk0_data_addr, 1, &data->scanner); + data->chunk_size= data->header.chunk0_data_len; + data->body_offset= data->scanner.page_offset; + data->current_offset= new_current_offset; + data->eor= 1; + DBUG_RETURN(0); + } + + if (type == TRANSLOG_CHUNK_LSN || type == TRANSLOG_CHUNK_FIXED) + { + data->eor= 1; + DBUG_RETURN(1); /* End of record */ + } + + chunk_header_len= + translog_get_chunk_header_length(data->scanner.page, + data->scanner.page_offset); + chunk_len= translog_get_total_chunk_length(data->scanner.page, + data->scanner.page_offset); + data->chunk_size= chunk_len - chunk_header_len; + data->body_offset= data->scanner.page_offset + chunk_header_len; + data->current_offset= new_current_offset; + DBUG_PRINT("info", ("grp: %u chunk %u body_offset %u, chunk_size %u, " + "current_offset %lu", + (uint) data->current_group, + (uint) data->current_chunk, + (uint) data->body_offset, + (uint) data->chunk_size, (ulong) data->current_offset)); + DBUG_RETURN(0); +} + + +/* + Initialize record reader data from LSN + + SYNOPSIS + translog_init_reader_data() + lsn reference to LSN we should start from + data reader data to initialize + + RETURN + 0 - OK + 1 - Error +*/ + +static my_bool translog_init_reader_data(LSN *lsn, + struct st_translog_reader_data *data) +{ + DBUG_ENTER("translog_init_reader_data"); + if (translog_init_scanner(lsn, 1, &data->scanner) || + (data->read_header= + translog_read_record_header_scan(&data->scanner, &data->header, 1)) == 0) + { + DBUG_RETURN(1); + } + data->body_offset= data->header.non_header_data_start_offset; + data->chunk_size= data->header.non_header_data_len; + data->current_offset= data->read_header; + data->current_group= 0; + data->current_chunk= 0; + data->eor= 0; + DBUG_PRINT("info", ("read_header %u, " + "body_offset %u, chunk_size %u, current_offset %lu", + (uint) data->read_header, + (uint) data->body_offset, + (uint) data->chunk_size, (ulong) data->current_offset)); + DBUG_RETURN(0); +} + + +/* + Read a part of the record. + + SYNOPSIS + translog_read_record_header() + lsn log record serial number (address of the record) + offset from the beginning of the record beginning (read + by translog_read_record_header). + length length of record part which have to be read. + buffer buffer where to read the record part (have to be at + least 'length' bytes length) + + RETURN + length of data actually read +*/ + +translog_size_t translog_read_record(LSN *lsn, + translog_size_t offset, + translog_size_t length, + uchar *buffer, + struct st_translog_reader_data *data) +{ + translog_size_t requested_length= length; + translog_size_t end= offset + length; + struct st_translog_reader_data internal_data; + + DBUG_ENTER("translog_read_record"); + + if (data == NULL) + { + DBUG_ASSERT(lsn != NULL); + data= &internal_data; + } + if (lsn || + (offset < data->current_offset && + !(offset < data->read_header && offset + length < data->read_header))) + { + if (translog_init_reader_data(lsn, data)) + DBUG_RETURN(0); + } + DBUG_PRINT("info", ("Offset %lu, length %lu " + "Scanner: Cur: (%u, 0x%lx), Hrz: (%u, 0x%lx), " + "Lst: (%u, 0x%lx), Offset: %u(%x), fixed %d", + (ulong) offset, (ulong) length, + (uint) data->scanner.page_addr.file_no, + (ulong) data->scanner.page_addr.rec_offset, + (uint) data->scanner.horizon.file_no, + (ulong) data->scanner.horizon.rec_offset, + (uint) data->scanner.last_file_page.file_no, + (ulong) data->scanner.last_file_page.rec_offset, + (uint) data->scanner.page_offset, + (uint) data->scanner.page_offset, + data->scanner.fixed_horizon)); + if (offset < data->read_header) + { + DBUG_PRINT("info", + ("enter header offset %lu, length %lu", + (ulong) offset, (ulong) length)); + uint16 len= (data->read_header < end ? data->read_header : end) - offset; + memmove(buffer, data->header.header + offset, len); + length-= len; + if (length == 0) + DBUG_RETURN(requested_length); + offset+= len; + buffer+= len; + DBUG_PRINT("info", + ("len: %u, offset %lu, curr %lu, length %lu", + len, (ulong) offset, (ulong) data->current_offset, + (ulong) length)); + } + /* TODO: find first page which we should read by offset */ + + /* read the record chunk by chunk */ + do + { + uint page_end= data->current_offset + data->chunk_size; + DBUG_PRINT("info", + ("enter body offset %lu, curr %lu, length %lu page_end %lu", + (ulong) offset, (ulong) data->current_offset, (ulong) length, + (ulong) page_end)); + if (offset < page_end) + { + DBUG_ASSERT(offset >= data->current_offset); + uint len= page_end - offset; + memmove(buffer, + data->scanner.page + data->body_offset + + (offset - data->current_offset), len); + length-= len; + if (length == 0) + DBUG_RETURN(requested_length); + offset+= len; + buffer+= len; + DBUG_PRINT("info", + ("len: %u, offset %lu, curr %lu, length %lu", + len, (ulong) offset, (ulong) data->current_offset, + (ulong) length)); + } + if (translog_record_read_next_chunk(data)) + DBUG_RETURN(requested_length - length); + } while (length != 0); + + DBUG_RETURN(requested_length); +} + + +/* + Force skipping to the next buffer + + SYNOPSIS + translog_force_current_buffer_to_finish() +*/ + +static void translog_force_current_buffer_to_finish() +{ + TRANSLOG_ADDRESS new_buff_begunning; + uint8 old_buffer_no= log_descriptor.bc.buffer_no; + uint8 new_buffer_no= (old_buffer_no + 1) % TRANSLOG_BUFFERS_NO; + struct st_translog_buffer *new_buffer= log_descriptor.buffers + new_buffer_no; + struct st_translog_buffer *old_buffer= log_descriptor.bc.buffer; + uchar *data= log_descriptor.bc.ptr -log_descriptor.bc.current_page_size; + uint16 left= TRANSLOG_PAGE_SIZE - log_descriptor.bc.current_page_size; + uint16 current_page_size; + + new_buff_begunning= log_descriptor.bc.buffer->offset; + new_buff_begunning.rec_offset+= log_descriptor.bc.buffer->size; + + DBUG_ENTER("translog_force_current_buffer_to_finish"); + DBUG_PRINT("enter", ("Buffer #%u 0x%lx, " + "Buffer addr (%lu,0x%lx), " + "Page addr: (%lu,0x%lx), " + "New Buff: (%lu,0x%lx), " + "size %lu (%lu), Pg: %u, left: %u", + (uint) log_descriptor.bc.buffer_no, + (ulong) log_descriptor.bc.buffer, + (ulong) log_descriptor.bc.buffer->offset.file_no, + (ulong) log_descriptor.bc.buffer->offset.rec_offset, + (ulong) log_descriptor.horizon.file_no, + (ulong) (log_descriptor.horizon.rec_offset - + log_descriptor.bc.current_page_size), + (ulong) new_buff_begunning.file_no, + (ulong) new_buff_begunning.rec_offset, + (ulong) log_descriptor.bc.buffer->size, + (ulong) (log_descriptor.bc.ptr -log_descriptor.bc. + buffer->buffer), + (uint) log_descriptor.bc.current_page_size, + (uint) left)); + DBUG_ASSERT(log_descriptor.bc.ptr !=NULL); + DBUG_ASSERT((log_descriptor.bc.ptr -log_descriptor.bc.buffer->buffer) + %TRANSLOG_PAGE_SIZE == + log_descriptor.bc.current_page_size % TRANSLOG_PAGE_SIZE); + DBUG_ASSERT(log_descriptor.horizon.file_no == + log_descriptor.bc.buffer->offset.file_no); + DBUG_ASSERT(log_descriptor.bc.buffer->offset.rec_offset + + (log_descriptor.bc.ptr -log_descriptor.bc.buffer->buffer) == + log_descriptor.horizon.rec_offset); + if (left != TRANSLOG_PAGE_SIZE && left != 0) + { + /* + TODO: if 'left' is so small that can't hold any other record + then do not move the page + */ + DBUG_PRINT("info", ("left %u", (uint) left)); + + new_buff_begunning.rec_offset-= log_descriptor.bc.current_page_size; + current_page_size= log_descriptor.bc.current_page_size; + + bzero(log_descriptor.bc.ptr, left); + log_descriptor.bc.buffer->size+= left; + DBUG_PRINT("info", ("Finish Page buffer #%u: 0x%lx, " + "Size: %lu", + (uint) log_descriptor.bc.buffer->buffer_no, + (ulong) log_descriptor.bc.buffer, + (ulong) log_descriptor.bc.buffer->size)); + DBUG_ASSERT(log_descriptor.bc.buffer->buffer_no == + log_descriptor.bc.buffer_no); + } + else + { + left= 0; + log_descriptor.bc.current_page_size= 0; + } + + translog_buffer_lock(new_buffer); + translog_wait_for_buffer_free(new_buffer); + + { + uint16 write_counter= log_descriptor.bc.write_counter; + uint16 previous_offset= log_descriptor.bc.previous_offset; + translog_start_buffer(new_buffer, &log_descriptor.bc, new_buffer_no); + log_descriptor.bc.buffer->offset= new_buff_begunning; + log_descriptor.bc.write_counter= write_counter; + log_descriptor.bc.previous_offset= previous_offset; + } + + if (log_descriptor.flags & TRANSLOG_SECTOR_PROTECTION) + { + translog_put_sector_protection(data, &log_descriptor.bc); + if (left) + { + log_descriptor.bc.write_counter++; + log_descriptor.bc.previous_offset= current_page_size; + } + else + { + DBUG_PRINT("info", ("drop write_counter")); + log_descriptor.bc.write_counter= 0; + log_descriptor.bc.previous_offset= 0; + } + } + + if (log_descriptor.flags & TRANSLOG_PAGE_CRC) + { + uint32 crc= translog_adler_crc(data + log_descriptor.page_overhead, + TRANSLOG_PAGE_SIZE - + log_descriptor.page_overhead); + DBUG_PRINT("info", ("CRC: 0x%lx", (ulong) crc)); + int4store(data + 3 + 3 + 1, crc); + } + + if (left) + { + memmove(new_buffer->buffer, data, current_page_size); + log_descriptor.bc.ptr +=current_page_size; + log_descriptor.bc.buffer->size= log_descriptor.bc.current_page_size= + current_page_size; + new_buffer->overlay= old_buffer; + } + else + translog_new_page_header(&log_descriptor.horizon, &log_descriptor.bc); + + DBUG_VOID_RETURN; +} + +/* + Flush the log up to given LSN (included) + + SYNOPSIS + translog_flush() + lsn log record serial number up to which (inclusive) + the log have to be flushed + + RETURN + 0 - OK + 1 - Error +*/ + +my_bool translog_flush(LSN *lsn) +{ + LSN old_flushed, sent_to_file; + int rc= 0; + uint i; + my_bool full_circle= 0; + + DBUG_ENTER("translog_flush"); + DBUG_PRINT("enter", ("Flush up to LSN (%u,0x%lx)", + (uint) lsn->file_no, (ulong) lsn->rec_offset)); + + translog_lock(); + old_flushed= log_descriptor.flushed; + for (;;) + { + uint8 buffer_no= log_descriptor.bc.buffer_no; + uint8 buffer_start= buffer_no; + struct st_translog_buffer *buffer_unlock= log_descriptor.bc.buffer; + + struct st_translog_buffer *buffer= log_descriptor.bc.buffer; + /* we can't flush in future */ + DBUG_ASSERT(cmp_translog_addr(log_descriptor.horizon, *lsn) >= 0); + if (cmp_translog_addr(log_descriptor.flushed, *lsn) >= 0) + { + DBUG_PRINT("info", ("already flushed (%u,0x%lx)", + (uint) log_descriptor.flushed.file_no, + (ulong) log_descriptor.flushed.rec_offset)); + translog_unlock(); + DBUG_RETURN(0); + } + /* send to the file if it is not sent */ + translog_get_sent_to_file(&sent_to_file); + if (cmp_translog_addr(sent_to_file, *lsn) >= 0) + break; + + do + { + buffer_no= (buffer_no + 1) % TRANSLOG_BUFFERS_NO; + buffer= log_descriptor.buffers + buffer_no; + translog_buffer_lock(buffer); + translog_buffer_unlock(buffer_unlock); + buffer_unlock= buffer; + if (buffer->file) + { + buffer_unlock= NULL; + if (buffer_start == buffer_no) + { + /* we made a circle */ + full_circle= 1; + translog_force_current_buffer_to_finish(); + } + break; + } + } while ((buffer_start != buffer_no) && + cmp_translog_addr(log_descriptor.flushed, *lsn) < 0); + if (buffer_unlock != NULL) + translog_buffer_unlock(buffer_unlock); + if (translog_buffer_flush(buffer)) + { + translog_buffer_unlock(buffer); + DBUG_RETURN(1); + } + translog_buffer_unlock(buffer); + if (!full_circle) + translog_lock(); + } + + for (i= old_flushed.file_no; i <= lsn->file_no; i++) + { + uint cache_index; + File file; + + if ((cache_index= log_descriptor.horizon.file_no - i) < OPENED_FILES_NUM) + { + /* file in the cache */ + if (log_descriptor.log_file_num[cache_index] == 0) + { + if ((log_descriptor.log_file_num[cache_index]= + open_logfile_by_number_no_cache(i)) == 0) + { + translog_unlock(); + DBUG_RETURN(1); + } + } + file= log_descriptor.log_file_num[cache_index]; + rc|= my_sync(file, MYF(MY_WME)); + } + else + { + /* very unlike situation with extremely small file size */ + File file= open_logfile_by_number_no_cache(i); + rc|= my_sync(file, MYF(MY_WME)); + my_close(file, MYF(MY_WME)); + } + } + log_descriptor.flushed= sent_to_file; + rc|= my_sync(log_descriptor.directory_fd, MYF(MY_WME)); + translog_unlock(); + DBUG_RETURN(rc); +} diff --git a/storage/maria/ma_loghandler.h b/storage/maria/ma_loghandler.h new file mode 100644 index 00000000000..f4d939786fc --- /dev/null +++ b/storage/maria/ma_loghandler.h @@ -0,0 +1,314 @@ + +#ifndef _ma_loghandler_h +#define _ma_loghandler_h + +/* Transaction log flags */ +#define TRANSLOG_PAGE_CRC 1 +#define TRANSLOG_SECTOR_PROTECTION (1<<1) +#define TRANSLOG_RECORD_CRC (1<<2) + +/* page size in transaction log */ +#define TRANSLOG_PAGE_SIZE (8*1024) + +#include "ma_loghandler_lsn.h" + +/* short transaction ID type */ +typedef uint16 SHORT_TRANSACTION_ID; + +/* types of records in the transaction log */ +enum translog_record_type +{ + LOGREC_RESERVED_FOR_CHUNKS23= 0, + LOGREC_REDO_INSERT_ROW_HEAD= 1, + LOGREC_REDO_INSERT_ROW_TAIL= 2, + LOGREC_REDO_INSERT_ROW_BLOB= 3, + LOGREC_REDO_INSERT_ROW_BLOBS= 4, + LOGREC_REDO_PURGE_ROW= 5, + eLOGREC_REDO_PURGE_BLOCKS= 6, + LOGREC_REDO_DELETE_ROW= 7, + LOGREC_REDO_UPDATE_ROW_HEAD= 8, + LOGREC_REDO_INDEX= 9, + LOGREC_REDO_UNDELETE_ROW= 10, + LOGREC_CLR_END= 11, + LOGREC_PURGE_END= 12, + LOGREC_UNDO_ROW_INSERT= 13, + LOGREC_UNDO_ROW_DELETE= 14, + LOGREC_UNDO_ROW_UPDATE= 15, + LOGREC_UNDO_KEY_INSERT= 16, + LOGREC_UNDO_KEY_DELETE= 17, + LOGREC_PREPARE= 18, + LOGREC_PREPARE_WITH_UNDO_PURGE= 19, + LOGREC_COMMIT= 20, + LOGREC_COMMIT_WITH_UNDO_PURGE= 21, + LOGREC_CHECKPOINT_PAGE= 22, + LOGREC_CHECKPOINT_TRAN= 23, + LOGREC_CHECKPOINT_TABL= 24, + LOGREC_REDO_CREATE_TABLE= 25, + LOGREC_REDO_RENAME_TABLE= 26, + LOGREC_REDO_DROP_TABLE= 27, + LOGREC_REDO_TRUNCATE_TABLE= 28, + LOGREC_FILE_ID= 29, + LOGREC_LONG_TRANSACTION_ID= 30, + LOGREC_RESERVED_FUTURE_EXTENSION= 63 +}; +#define LOGREC_NUMBER_OF_TYPES 64 + +typedef uint32 translog_size_t; + +#define TRANSLOG_RECORD_HEADER_MAX_SIZE 1024 + +typedef struct st_translog_group_descriptor +{ + TRANSLOG_ADDRESS addr; + uint8 num; +} TRANSLOG_GROUP; + + +typedef struct st_translog_header_buffer +{ + /* LSN of the read record */ + LSN lsn; + /* type of the read record */ + enum translog_record_type type; + /* short transaction ID or 0 if it has no sense for the record */ + SHORT_TRANSACTION_ID short_trid; + /* + The Record length in buffer (including read header, but excluding + hidden part of record (type, short TrID, length) + */ + translog_size_t record_length; + /* + Real compressed LSN(s) size economy (*7 - ) + */ + uint16 compressed_LSN_economy; + /* + Buffer for write decoded header of the record (depend on the record + type) + */ + uchar header[TRANSLOG_RECORD_HEADER_MAX_SIZE]; + /* non read body data offset on the page */ + uint16 non_header_data_start_offset; + /* non read body data length in this first chunk */ + uint16 non_header_data_len; + /* number of groups listed in */ + uint groups_no; + /* array of groups descriptors, can be used only if groups_no > 0 */ + TRANSLOG_GROUP *groups; + /* in multi-group number of chunk0 pages (valid only if groups_no > 0) */ + uint chunk0_pages; + /* chunk 0 data address (valid only if groups_no > 0) */ + TRANSLOG_ADDRESS chunk0_data_addr; + /* chunk 0 data size (valid only if groups_no > 0) */ + uint16 chunk0_data_len; +} TRANSLOG_HEADER_BUFFER; + + +struct st_translog_scanner_data +{ + uchar buffer[TRANSLOG_PAGE_SIZE]; /* buffer for page content */ + TRANSLOG_ADDRESS page_addr; /* current page address */ + TRANSLOG_ADDRESS horizon; /* end of the log which we saw + last time */ + TRANSLOG_ADDRESS last_file_page; /* Last page on in this file */ + uchar *page; /* page content pointer */ + translog_size_t page_offset; /* offset of the chunk in the + page */ + my_bool fixed_horizon; /* set horizon only once at + init */ +}; + + +struct st_translog_reader_data +{ + TRANSLOG_HEADER_BUFFER header; /* Header */ + struct st_translog_scanner_data scanner; /* chunks scanner */ + translog_size_t body_offset; /* current chunk body offset */ + translog_size_t current_offset; /* data offset from the record + beginning */ + uint16 read_header; /* number of bytes read in + header */ + uint16 chunk_size; /* current chunk size */ + uint current_group; /* current group */ + uint current_chunk; /* current chunk in the group */ + my_bool eor; /* end of the record */ +}; + + +/* + Initialize transaction log + + SYNOPSIS + translog_init() + directory Directory where log files are put + log_file_max_size max size of one log size (for new logs creation) + server_version version of MySQL servger (MYSQL_VERSION_ID) + server_id server ID (replication & Co) + pagecache Page cache for the log reads + flags flags (TRANSLOG_PAGE_CRC, TRANSLOG_SECTOR_PROTECTION + TRANSLOG_RECORD_CRC) + + RETURN + 0 - OK + 1 - Error +*/ + +my_bool translog_init(const char *directory, uint32 log_file_max_size, + uint32 server_version, + uint32 server_id, PAGECACHE *pagecache, uint flags); + + +/* + Write the log record + + SYNOPSIS + translog_write_record() + lsn LSN of the record will be writen here + type the log record type + short_trid Sort transaction ID or 0 if it has no sense + tcb Transaction control block pointer for hooks by + record log type + partN_length length of Ns part of the log + partN_buffer pointer on Ns part buffer + 0 sign of the end of parts + + RETURN + 0 - OK + 1 - Error +*/ + +my_bool translog_write_record(LSN *lsn, + enum translog_record_type type, + SHORT_TRANSACTION_ID short_trid, + void *tcb, + translog_size_t part1_length, + uchar *part1_buff, ...); + + +/* + Free log handler resources + + SYNOPSIS + translog_destroy() +*/ + +void translog_destroy(); + + +/* + Read record header and some fixed part of a record (the part depend on + record type). + + SYNOPSIS + translog_read_record_header() + lsn log record serial number (address of the record) + buff log record header buffer + + NOTE + - lsn can point to TRANSLOG_HEADER_BUFFER::lsn and it will be processed + correctly. + - Some type of record can be read completely by this call + - "Decoded" header stored in TRANSLOG_HEADER_BUFFER::header (relative + LSN can be translated to absolute one), some fields can be added + (like actual header length in the record if the header has variable + length) + + RETURN + 0 - error + number of bytes in TRANSLOG_HEADER_BUFFER::header where stored decoded + part of the header +*/ + +translog_size_t translog_read_record_header(LSN *lsn, + TRANSLOG_HEADER_BUFFER *buff); + + +/* + Free resources used by TRANSLOG_HEADER_BUFFER + + SYNOPSIS + translog_free_record_header(); +*/ + +void translog_free_record_header(TRANSLOG_HEADER_BUFFER *buff); + + +/* + Read a part of the record. + + SYNOPSIS + translog_read_record_header() + lsn log record serial number (address of the record) + offset from the beginning of the record beginning (read + by translog_read_record_header). + length length of record part which have to be read. + buffer buffer where to read the record part (have to be at + least 'length' bytes length) + + RETURN + 0 - error (or read out of the record) + length of data actually read +*/ + +translog_size_t translog_read_record(LSN *lsn, + translog_size_t offset, + translog_size_t length, + uchar *buffer, + struct st_translog_reader_data *data); + + +/* + Flush the log up to given LSN (included) + + SYNOPSIS + translog_flush() + lsn log record serial number up to which (inclusive) + the log have to be flushed + + RETURN + 0 - OK + 1 - Error +*/ + +my_bool translog_flush(LSN *lsn); + + +/* + Read record header and some fixed part of the next record (the part + depend on record type). + + SYNOPSIS + translog_read_next_record_header() + lsn log record serial number (address of the record) + previous to the record which will be read + If LSN present scanner will be initialized from it, + do not use LSN after initialization for fast scanning. + buff log record header buffer + fixed_horizon true if it is OK do not read records which was written + after scaning begining + scanner data for scaning if lsn is NULL scanner data + will be used for continue scaning. + scanner can be NULL. + + NOTE + - lsn can point to TRANSLOG_HEADER_BUFFER::lsn and it will be processed + correctly (lsn in buffer will be replaced by next record, but initial + lsn will be read correctly). + - it is like translog_read_record_header, but read next record, so see + its NOTES. + - in case of end of the log buff->lsn will be set to + (CONTROL_FILE_IMPOSSIBLE_LOGNO, 0) + RETURN + 0 - error + TRANSLOG_RECORD_HEADER_MAX_SIZE + 1 - End of the log + number of bytes in TRANSLOG_HEADER_BUFFER::header where stored decoded + part of the header +*/ + +translog_size_t translog_read_next_record_header(LSN *lsn, + TRANSLOG_HEADER_BUFFER *buff, + my_bool fixed_horizon, + struct + st_translog_scanner_data + *scanner); + +#endif diff --git a/storage/maria/ma_loghandler_lsn.h b/storage/maria/ma_loghandler_lsn.h new file mode 100644 index 00000000000..9576d4d734d --- /dev/null +++ b/storage/maria/ma_loghandler_lsn.h @@ -0,0 +1,39 @@ +#ifndef _ma_loghandler_lsn_h +#define _ma_loghandler_lsn_h + +/* Transaction log record address (file_no is int24 on the disk) */ +typedef struct st_translog_address +{ + uint32 file_no; + uint32 rec_offset; +} TRANSLOG_ADDRESS; + +/* + Compare addresses + A1 > A2 -> result > 0 + A1 == A2 -> 0 + A1 < A2 -> result < 0 +*/ +#define cmp_translog_addr(A1,A2) \ + ((A1).file_no == (A2).file_no ? \ + ((int64)(A1).rec_offset) - (int64)(A2).rec_offset : \ + ((int64)(A1).file_no - (int64)(A2).file_no)) + +/* LSN type (address of certain log record chank */ +typedef TRANSLOG_ADDRESS LSN; + +/* Puts LSN into buffer (dst) */ +#define lsn7store(dst, lsn) \ + do { \ + int3store((dst), (lsn)->file_no); \ + int4store((dst) + 3, (lsn)->rec_offset); \ + } while (0) + +/* Unpacks LSN from the buffer (P) */ +#define lsn7korr(lsn, P) \ + do { \ + (lsn)->file_no= uint3korr(P); \ + (lsn)->rec_offset= uint4korr((P) + 3); \ + } while (0) + +#endif diff --git a/storage/maria/maria_def.h b/storage/maria/maria_def.h index 506bdbc71ca..380c42c105e 100644 --- a/storage/maria/maria_def.h +++ b/storage/maria/maria_def.h @@ -26,6 +26,10 @@ #include #endif +#include +#include "ma_loghandler.h" +#include "ma_control_file.h" + /* undef map from my_nosys; We need test-if-disk full */ #undef my_write @@ -438,6 +442,7 @@ extern LIST *maria_open_list; extern uchar NEAR maria_file_magic[], NEAR maria_pack_file_magic[]; extern uint NEAR maria_read_vec[], NEAR maria_readnext_vec[]; extern uint maria_quick_table_bits; +extern const char *maria_data_root; extern my_bool maria_inited; /* This is used by _ma_calc_xxx_key_length och _ma_store_key */ diff --git a/storage/maria/unittest/Makefile.am b/storage/maria/unittest/Makefile.am index d0b247d65e1..78b285edd70 100644 --- a/storage/maria/unittest/Makefile.am +++ b/storage/maria/unittest/Makefile.am @@ -14,8 +14,10 @@ # along with this program; if not, write to the Free Software # Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA -AM_CPPFLAGS = @ZLIB_INCLUDES@ -I$(top_builddir)/include -AM_CPPFLAGS += -I$(top_srcdir)/include -I$(top_srcdir)/unittest/mytap +AM_CPPFLAGS = @ZLIB_INCLUDES@ -I$(top_builddir)/include \ + -I$(top_srcdir)/include -I$(top_srcdir)/unittest/mytap +INCLUDES = @ZLIB_INCLUDES@ -I$(top_builddir)/include \ + -I$(top_srcdir)/include -I$(top_srcdir)/unittest/mytap # Only reason to link with libmyisam.a here is that it's where some fulltext # pieces are (but soon we'll remove fulltext dependencies from Maria). @@ -24,6 +26,54 @@ LDADD= $(top_builddir)/unittest/mytap/libmytap.a \ $(top_builddir)/storage/myisam/libmyisam.a \ $(top_builddir)/mysys/libmysys.a \ $(top_builddir)/dbug/libdbug.a \ - $(top_builddir)/strings/libmystrings.a @ZLIB_LIBS@ -noinst_PROGRAMS = ma_control_file-t trnman-t lockman-t lockman1-t lockman2-t -CLEANFILES = maria_control + $(top_builddir)/strings/libmystrings.a @ZLIB_LIBS@ \ + $(top_builddir)/storage/maria/ma_loghandler.o +noinst_PROGRAMS = ma_control_file-t trnman-t lockman-t lockman1-t \ + lockman2-t \ + mf_pagecache_single_1k-t mf_pagecache_single_8k-t \ + mf_pagecache_single_64k-t \ + mf_pagecache_consist_1k-t mf_pagecache_consist_64k-t \ + mf_pagecache_consist_1kHC-t \ + mf_pagecache_consist_64kHC-t \ + mf_pagecache_consist_1kRD-t \ + mf_pagecache_consist_64kRD-t \ + mf_pagecache_consist_1kWR-t \ + mf_pagecache_consist_64kWR-t \ + ma_test_loghandler-t \ + ma_test_loghandler_multigroup-t \ + ma_test_loghandler_multithread-t \ + ma_test_loghandler_pagecache-t + +mf_pagecache_single_src = mf_pagecache_single.c $(top_srcdir)/mysys/mf_pagecache.c test_file.c +mf_pagecache_consist_src = mf_pagecache_consist.c $(top_srcdir)/mysys/mf_pagecache.c test_file.c +mf_pagecache_common_cppflags = -DEXTRA_DEBUG -DPAGECACHE_DEBUG -DMAIN + +mf_pagecache_single_1k_t_SOURCES = $(mf_pagecache_single_src) +mf_pagecache_single_8k_t_SOURCES = $(mf_pagecache_single_src) +mf_pagecache_single_64k_t_SOURCES = $(mf_pagecache_single_src) +mf_pagecache_single_1k_t_CPPFLAGS = $(mf_pagecache_common_cppflags) -DPAGE_SIZE=1024 +mf_pagecache_single_8k_t_CPPFLAGS = $(mf_pagecache_common_cppflags) -DPAGE_SIZE=8192 +mf_pagecache_single_64k_t_CPPFLAGS = $(mf_pagecache_common_cppflags) -DPAGE_SIZE=65536 + +mf_pagecache_consist_1k_t_SOURCES = $(mf_pagecache_consist_src) +mf_pagecache_consist_1k_t_CPPFLAGS = $(mf_pagecache_common_cppflags) -DPAGE_SIZE=1024 +mf_pagecache_consist_64k_t_SOURCES = $(mf_pagecache_consist_src) +mf_pagecache_consist_64k_t_CPPFLAGS = $(mf_pagecache_common_cppflags) -DPAGE_SIZE=65536 + +mf_pagecache_consist_1kHC_t_SOURCES = $(mf_pagecache_consist_src) +mf_pagecache_consist_1kHC_t_CPPFLAGS = $(mf_pagecache_common_cppflags) -DPAGE_SIZE=1024 -DTEST_HIGH_CONCURENCY +mf_pagecache_consist_64kHC_t_SOURCES = $(mf_pagecache_consist_src) +mf_pagecache_consist_64kHC_t_CPPFLAGS = $(mf_pagecache_common_cppflags) -DPAGE_SIZE=65536 -DTEST_HIGH_CONCURENCY + +mf_pagecache_consist_1kRD_t_SOURCES = $(mf_pagecache_consist_src) +mf_pagecache_consist_1kRD_t_CPPFLAGS = $(mf_pagecache_common_cppflags) -DPAGE_SIZE=1024 -DTEST_READERS +mf_pagecache_consist_64kRD_t_SOURCES = $(mf_pagecache_consist_src) +mf_pagecache_consist_64kRD_t_CPPFLAGS = $(mf_pagecache_common_cppflags) -DPAGE_SIZE=65536 -DTEST_READERS + +mf_pagecache_consist_1kWR_t_SOURCES = $(mf_pagecache_consist_src) +mf_pagecache_consist_1kWR_t_CPPFLAGS = $(mf_pagecache_common_cppflags) -DPAGE_SIZE=1024 -DTEST_WRITERS +mf_pagecache_consist_64kWR_t_SOURCES = $(mf_pagecache_consist_src) +mf_pagecache_consist_64kWR_t_CPPFLAGS = $(mf_pagecache_common_cppflags) -DPAGE_SIZE=65536 -DTEST_WRITERS + +CLEANFILES = maria_control page_cache_test_file_1 \ + maria_log.???????? maria_control diff --git a/storage/maria/unittest/ma_control_file-t.c b/storage/maria/unittest/ma_control_file-t.c index beb86843dd3..7b7e1454cc3 100644 --- a/storage/maria/unittest/ma_control_file-t.c +++ b/storage/maria/unittest/ma_control_file-t.c @@ -33,7 +33,7 @@ #endif #include "maria.h" -#include "../../../storage/maria/ma_control_file.h" +#include "../../../storage/maria/maria_def.h" #include char file_name[FN_REFLEN]; diff --git a/storage/maria/unittest/ma_test_loghandler-t.c b/storage/maria/unittest/ma_test_loghandler-t.c new file mode 100644 index 00000000000..1cbfcac504e --- /dev/null +++ b/storage/maria/unittest/ma_test_loghandler-t.c @@ -0,0 +1,540 @@ +#include "../maria_def.h" +#include +#include + +#ifndef DBUG_OFF +static const char *default_dbug_option; +#endif + +#define PCACHE_SIZE (1024*1024*10) + +#define LONG_BUFFER_SIZE (100 * 1024) + + +#define LOG_FLAGS TRANSLOG_SECTOR_PROTECTION | TRANSLOG_PAGE_CRC +#define LOG_FILE_SIZE 1024L*1024L*3L +#define ITERATIONS 1600 + +/* +#define LOG_FLAGS 0 +#define LOG_FILE_SIZE 1024L*1024L*1024L +#define ITERATIONS 181000 +*/ + +/* +#define LOG_FLAGS 0 +#define LOG_FILE_SIZE 1024L*1024L*3L +#define ITERATIONS 1600 +*/ + +/* +#define LOG_FLAGS 0 +#define LOG_FILE_SIZE 1024L*1024L*100L +#define ITERATIONS 65000 +*/ + +/* + Check that the buffer filled correctly + + SYNOPSIS + check_content() + ptr Pointer to the buffer + length length of the buffer + + RETURN + 0 - OK + 1 - Error +*/ + +static my_bool check_content(uchar *ptr, ulong length) +{ + ulong i; + uchar buff[2]; + for (i= 0; i < length; i++) + { + if (i % 2 == 0) + int2store(buff, i >> 1); + if (ptr[i] != buff[i % 2]) + { + fprintf(stderr, "Byte # %lu is %x instead of %x", + i, (uint) ptr[i], (uint) buff[i % 2]); + return 1; + } + } + return 0; +} + + +/* + Read whole record content, and check content (put with offset) + + SYNOPSIS + read_and_check_content() + rec The record header buffer + buffer The buffer to read the record in + skip Skip this number of bytes ot the record content + + RETURN + 0 - OK + 1 - Error +*/ + +static my_bool read_and_check_content(TRANSLOG_HEADER_BUFFER *rec, + uchar *buffer, uint skip) +{ + DBUG_ASSERT(rec->record_length < LONG_BUFFER_SIZE * 2 + 7 * 2 + 2); + if (translog_read_record(&rec->lsn, 0, rec->record_length, buffer, NULL) != + rec->record_length) + return 1; + return check_content(buffer + skip, rec->record_length - skip); +} + +int main(int argc, char *argv[]) +{ + uint32 i; + uint32 rec_len; + uint pagen; + uchar long_tr_id[6]; + uchar lsn_buff[23]= + { + 0x55, 0xAA, 0x55, 0xAA, 0x55, 0xAA, 0x55, 0xAA, + 0x55, 0xAA, 0x55, 0xAA, 0x55, 0xAA, 0x55, 0xAA, + 0x55, 0xAA, 0x55, 0xAA, 0x55, 0xAA, 0x55 + }; + uchar long_buffer[LONG_BUFFER_SIZE * 2 + 7 * 2 + 2]; + PAGECACHE pagecache; + LSN lsn, lsn_base, first_lsn, *lsn_ptr; + TRANSLOG_HEADER_BUFFER rec; + struct st_translog_scanner_data scanner; + int rc; + + MY_INIT(argv[0]); + + bzero(&pagecache, sizeof(pagecache)); + maria_data_root= "."; + + for (i= 0; i < (LONG_BUFFER_SIZE + 7 * 2 + 2); i+= 2) + { + int2store(long_buffer + i, (i >> 1)); + /* long_buffer[i]= (i & 0xFF); */ + } + + bzero(long_tr_id, 6); +#ifndef DBUG_OFF +#if defined(__WIN__) + default_dbug_option= "d:t:i:O,\\ma_test_loghandler.trace"; +#else + default_dbug_option= "d:t:i:o,/tmp/ma_test_loghandler.trace"; +#endif + if (argc > 1) + { + DBUG_SET(default_dbug_option); + DBUG_SET_INITIAL(default_dbug_option); + } +#endif + + if (ma_control_file_create_or_open()) + { + fprintf(stderr, "Can't init control file (%d)\n", errno); + exit(1); + } + if ((pagen= init_pagecache(&pagecache, PCACHE_SIZE, 0, 0, + TRANSLOG_PAGE_SIZE)) == 0) + { + fprintf(stderr, "Got error: init_pagecache() (errno: %d)\n", errno); + exit(1); + } + if (translog_init(".", LOG_FILE_SIZE, 50112, 0, &pagecache, LOG_FLAGS)) + { + fprintf(stderr, "Can't init loghandler (%d)\n", errno); + translog_destroy(); + exit(1); + } + + srandom(122334817L); + + long_tr_id[5]= 0xff; + + int4store(long_tr_id, 0); + if (translog_write_record(&lsn, + LOGREC_LONG_TRANSACTION_ID, + 0, NULL, 6, long_tr_id, 0)) + { + fprintf(stderr, "Can't write record #%lu\n", (ulong) 0); + translog_destroy(); + exit(1); + } + lsn_base= first_lsn= lsn; + + for (i= 1; i < ITERATIONS; i++) + { + if (i % 1000 == 0) + printf("write %d\n", i); + if (i % 2) + { + lsn7store(lsn_buff, &lsn_base); + if (translog_write_record(&lsn, + LOGREC_CLR_END, + (i % 0xFFFF), NULL, 7, lsn_buff, 0)) + { + fprintf(stderr, "1 Can't write reference defore record #%lu\n", + (ulong) i); + translog_destroy(); + exit(1); + } + lsn7store(lsn_buff, &lsn_base); + if ((rec_len= random() / (RAND_MAX / (LONG_BUFFER_SIZE + 1))) < 12) + rec_len= 12; + if (translog_write_record(&lsn, + LOGREC_UNDO_KEY_INSERT, + (i % 0xFFFF), + NULL, 7, lsn_buff, rec_len, long_buffer, 0)) + { + fprintf(stderr, "1 Can't write var reference defore record #%lu\n", + (ulong) i); + translog_destroy(); + exit(1); + } + } + else + { + lsn7store(lsn_buff, &lsn_base); + lsn7store(lsn_buff + 7, &first_lsn); + if (translog_write_record(&lsn, + LOGREC_UNDO_ROW_DELETE, + (i % 0xFFFF), NULL, 23, lsn_buff, 0)) + { + fprintf(stderr, "0 Can't write reference defore record #%lu\n", + (ulong) i); + translog_destroy(); + exit(1); + } + lsn7store(lsn_buff, &lsn_base); + lsn7store(lsn_buff + 7, &first_lsn); + if ((rec_len= random() / (RAND_MAX / (LONG_BUFFER_SIZE + 1))) < 19) + rec_len= 19; + if (translog_write_record(&lsn, + LOGREC_UNDO_KEY_DELETE, + (i % 0xFFFF), + NULL, 14, lsn_buff, rec_len, long_buffer, 0)) + { + fprintf(stderr, "0 Can't write var reference defore record #%lu\n", + (ulong) i); + translog_destroy(); + exit(1); + } + } + int4store(long_tr_id, i); + if (translog_write_record(&lsn, + LOGREC_LONG_TRANSACTION_ID, + (i % 0xFFFF), NULL, 6, long_tr_id, 0)) + { + fprintf(stderr, "Can't write record #%lu\n", (ulong) i); + translog_destroy(); + exit(1); + } + + lsn_base= lsn; + + if ((rec_len= random() / (RAND_MAX / (LONG_BUFFER_SIZE + 1))) < 9) + rec_len= 9; + if (translog_write_record(&lsn, + LOGREC_REDO_INSERT_ROW_HEAD, + (i % 0xFFFF), NULL, rec_len, long_buffer, 0)) + { + fprintf(stderr, "Can't write variable record #%lu\n", (ulong) i); + translog_destroy(); + exit(1); + } + if (translog_flush(&lsn)) + { + fprintf(stderr, "Can't flush #%lu\n", (ulong) i); + translog_destroy(); + exit(1); + } + } + + translog_destroy(); + end_pagecache(&pagecache, 1); + ma_control_file_end(); + + + if (ma_control_file_create_or_open()) + { + fprintf(stderr, "pass2: Can't init control file (%d)\n", errno); + exit(1); + } + if ((pagen= init_pagecache(&pagecache, PCACHE_SIZE, 0, 0, + TRANSLOG_PAGE_SIZE)) == 0) + { + fprintf(stderr, "pass2: Got error: init_pagecache() (errno: %d)\n", errno); + exit(1); + } + if (translog_init(".", LOG_FILE_SIZE, 50112, 0, &pagecache, LOG_FLAGS)) + { + fprintf(stderr, "pass2: Can't init loghandler (%d)\n", errno); + translog_destroy(); + exit(1); + } + srandom(122334817L); + + + rc= 1; + + { + translog_size_t len= translog_read_record_header(&first_lsn, &rec); + if (len == 0) + { + fprintf(stderr, "translog_read_record_header failed (%d)\n", errno); + goto err; + } + if (rec.type !=LOGREC_LONG_TRANSACTION_ID || rec.short_trid != 0 || + rec.record_length != 6 || uint4korr(rec.header) != 0 || + (uint)rec.header[4] != 0 || rec.header[5] != 0xFF || + first_lsn.file_no != rec.lsn.file_no || + first_lsn.rec_offset != rec.lsn.rec_offset) + { + fprintf(stderr, "Incorrect LOGREC_LONG_TRANSACTION_ID data read(0)\n" + "type %u, strid %u, len %u, i: %u, 4: %u 5: %u, " + "lsn(0x%lx,0x%lx)\n", + (uint) rec.type, (uint) rec.short_trid, (uint) rec.record_length, + uint4korr(rec.header), (uint) rec.header[4], (uint) rec.header[5], + (ulong) rec.lsn.file_no, (ulong) rec.lsn.rec_offset); + goto err; + } + lsn= first_lsn; + lsn_ptr= &first_lsn; + for (i= 1;; i++) + { + if (i % 1000 == 0) + printf("read %d\n", i); + len= translog_read_next_record_header(lsn_ptr, &rec, 1, &scanner); + if (len == 0) + { + fprintf(stderr, "1-%d translog_read_next_record_header failed (%d)\n", + i, errno); + goto err; + } + if (rec.lsn.file_no == CONTROL_FILE_IMPOSSIBLE_FILENO) + { + if (i != ITERATIONS) + { + fprintf(stderr, "EOL met at iteration %u instead of %u\n", + i, ITERATIONS); + goto err; + } + break; + } + lsn_ptr= NULL; /* use scanner after its + initialization */ + if (i % 2) + { + LSN ref; + lsn7korr(&ref, rec.header); + if (rec.type !=LOGREC_CLR_END || rec.short_trid != (i % 0xFFFF) || + rec.record_length != 7 || ref.file_no != lsn.file_no || + ref.rec_offset != lsn.rec_offset) + { + fprintf(stderr, "Incorrect LOGREC_CLR_END data read(%d)" + "type %u, strid %u, len %u, ref(%u,0x%lx), lsn(%u,0x%lx)\n", + i, (uint) rec.type, (uint) rec.short_trid, + (uint) rec.record_length, + (uint) ref.file_no, (ulong) ref.rec_offset, + (uint) rec.lsn.file_no, (ulong) rec.lsn.rec_offset); + goto err; + } + } + else + { + LSN ref1, ref2; + lsn7korr(&ref1, rec.header); + lsn7korr(&ref2, rec.header + 7); + if (rec.type !=LOGREC_UNDO_ROW_DELETE || + rec.short_trid != (i % 0xFFFF) || + rec.record_length != 23 || + ref1.file_no != lsn.file_no || + ref1.rec_offset != lsn.rec_offset || + ref2.file_no != first_lsn.file_no || + ref2.rec_offset != first_lsn.rec_offset || + rec.header[22] != 0x55 || rec.header[21] != 0xAA || + rec.header[20] != 0x55 || rec.header[19] != 0xAA || + rec.header[18] != 0x55 || rec.header[17] != 0xAA || + rec.header[16] != 0x55 || rec.header[15] != 0xAA || + rec.header[14] != 0x55) + { + fprintf(stderr, "Incorrect LOGREC_UNDO_ROW_DELETE data read(%d)" + "type %u, strid %u, len %u, ref1(%u,0x%lx), " + "ref2(%u,0x%lx) %x%x%x%x%x%x%x%x%x " + "lsn(%u,0x%lx)\n", + i, (uint) rec.type, (uint) rec.short_trid, + (uint) rec.record_length, + (uint) ref1.file_no, (ulong) ref1.rec_offset, + (uint) ref2.file_no, (ulong) ref2.rec_offset, + (uint) rec.header[14], (uint) rec.header[15], + (uint) rec.header[16], (uint) rec.header[17], + (uint) rec.header[18], (uint) rec.header[19], + (uint) rec.header[20], (uint) rec.header[21], + (uint) rec.header[22], + (uint) rec.lsn.file_no, (ulong) rec.lsn.rec_offset); + goto err; + } + } + len= translog_read_next_record_header(lsn_ptr, &rec, 1, &scanner); + if (len == 0) + { + fprintf(stderr, "1-%d translog_read_next_record_header (var) " + "failed (%d)\n", i, errno); + goto err; + } + if (rec.lsn.file_no == CONTROL_FILE_IMPOSSIBLE_FILENO) + { + fprintf(stderr, "EOL met at the middle of iteration (first var) %u " + "instead of beginning of %u\n", i, ITERATIONS); + goto err; + } + if (i % 2) + { + LSN ref; + lsn7korr(&ref, rec.header); + if ((rec_len= random() / (RAND_MAX / (LONG_BUFFER_SIZE + 1))) < 12) + rec_len= 12; + if (rec.type !=LOGREC_UNDO_KEY_INSERT || + rec.short_trid != (i % 0xFFFF) || + rec.record_length != rec_len + 7 || + len != 12 || ref.file_no != lsn.file_no || + ref.rec_offset != lsn.rec_offset || + check_content(rec.header + 7, len - 7)) + { + fprintf(stderr, "Incorrect LOGREC_UNDO_KEY_INSERT data read(%d)" + "type %u (%d), strid %u (%d), len %lu, %lu + 7 (%d), " + "hdr len: %u (%d), " + "ref(%u,0x%lx), lsn(%u,0x%lx) (%d), content: %d\n", + i, (uint) rec.type, + rec.type !=LOGREC_UNDO_KEY_INSERT, + (uint) rec.short_trid, + rec.short_trid != (i % 0xFFFF), + (ulong) rec.record_length, (ulong) rec_len, + rec.record_length != rec_len + 7, + (uint) len, + len != 12, + (uint) ref.file_no, (ulong) ref.rec_offset, + (uint) rec.lsn.file_no, (ulong) rec.lsn.rec_offset, + (len != 12 || ref.file_no != lsn.file_no || + ref.rec_offset != lsn.rec_offset), + check_content(rec.header + 7, len - 7)); + goto err; + } + if (read_and_check_content(&rec, long_buffer, 7)) + { + fprintf(stderr, + "Incorrect LOGREC_UNDO_KEY_INSERT in whole rec read " + "lsn(%u,0x%lx)\n", + (uint) rec.lsn.file_no, (ulong) rec.lsn.rec_offset); + goto err; + } + } + else + { + LSN ref1, ref2; + lsn7korr(&ref1, rec.header); + lsn7korr(&ref2, rec.header + 7); + if ((rec_len= random() / (RAND_MAX / (LONG_BUFFER_SIZE + 1))) < 19) + rec_len= 19; + if (rec.type !=LOGREC_UNDO_KEY_DELETE || + rec.short_trid != (i % 0xFFFF) || + rec.record_length != rec_len + 14 || + len != 19 || + ref1.file_no != lsn.file_no || + ref1.rec_offset != lsn.rec_offset || + ref2.file_no != first_lsn.file_no || + ref2.rec_offset != first_lsn.rec_offset || + check_content(rec.header + 14, len - 14)) + { + fprintf(stderr, "Incorrect LOGREC_UNDO_KEY_DELETE data read(%d)" + "type %u, strid %u, len %lu != %lu + 7, hdr len: %u, " + "ref1(%u,0x%lx), ref2(%u,0x%lx), " + "lsn(%u,0x%lx)\n", + i, (uint) rec.type, (uint) rec.short_trid, + (ulong) rec.record_length, (ulong) rec_len, + (uint) len, + (uint) ref1.file_no, (ulong) ref1.rec_offset, + (uint) ref2.file_no, (ulong) ref2.rec_offset, + (uint) rec.lsn.file_no, (ulong) rec.lsn.rec_offset); + goto err; + } + if (read_and_check_content(&rec, long_buffer, 14)) + { + fprintf(stderr, + "Incorrect LOGREC_UNDO_KEY_DELETE in whole rec read " + "lsn(%u,0x%lx)\n", + (uint) rec.lsn.file_no, (ulong) rec.lsn.rec_offset); + goto err; + } + } + + len= translog_read_next_record_header(lsn_ptr, &rec, 1, &scanner); + if (len == 0) + { + fprintf(stderr, "1-%d translog_read_next_record_header failed (%d)\n", + i, errno); + goto err; + } + if (rec.lsn.file_no == CONTROL_FILE_IMPOSSIBLE_FILENO) + { + fprintf(stderr, "EOL met at the middle of iteration %u " + "instead of beginning of %u\n", i, ITERATIONS); + goto err; + } + if (rec.type !=LOGREC_LONG_TRANSACTION_ID || + rec.short_trid != (i % 0xFFFF) || + rec.record_length != 6 || uint4korr(rec.header) != i || + rec.header[4] != 0 || rec.header[5] != 0xFF) + { + fprintf(stderr, "Incorrect LOGREC_LONG_TRANSACTION_ID data read(%d)\n" + "type %u, strid %u, len %u, i: %u, 4: %u 5: %u " + "lsn(%u,0x%lx)\n", + i, (uint) rec.type, (uint) rec.short_trid, + (uint) rec.record_length, + uint4korr(rec.header), (uint) rec.header[4], + (uint) rec.header[5], + (uint) rec.lsn.file_no, (ulong) rec.lsn.rec_offset); + goto err; + } + + lsn= rec.lsn; + + len= translog_read_next_record_header(lsn_ptr, &rec, 1, &scanner); + if ((rec_len= random() / (RAND_MAX / (LONG_BUFFER_SIZE + 1))) < 9) + rec_len= 9; + if (rec.type !=LOGREC_REDO_INSERT_ROW_HEAD || + rec.short_trid != (i % 0xFFFF) || + rec.record_length != rec_len || + len != 9 || check_content(rec.header, len)) + { + fprintf(stderr, "Incorrect LOGREC_REDO_INSERT_ROW_HEAD data read(%d)" + "type %u, strid %u, len %lu != %lu, hdr len: %u, " + "lsn(%u,0x%lx)\n", + i, (uint) rec.type, (uint) rec.short_trid, + (ulong) rec.record_length, (ulong) rec_len, + (uint) len, (uint) rec.lsn.file_no, (ulong) rec.lsn.rec_offset); + goto err; + } + if (read_and_check_content(&rec, long_buffer, 0)) + { + fprintf(stderr, + "Incorrect LOGREC_UNDO_KEY_DELETE in whole rec read " + "lsn(%u,0x%lx)\n", + (uint) rec.lsn.file_no, (ulong) rec.lsn.rec_offset); + goto err; + } + } + } + + rc= 1; +err: + translog_destroy(); + end_pagecache(&pagecache, 1); + ma_control_file_end(); + + return(test(exit_status() || rc)); +} diff --git a/storage/maria/unittest/ma_test_loghandler_multigroup-t.c b/storage/maria/unittest/ma_test_loghandler_multigroup-t.c new file mode 100644 index 00000000000..abb12faa015 --- /dev/null +++ b/storage/maria/unittest/ma_test_loghandler_multigroup-t.c @@ -0,0 +1,570 @@ +#include "../maria_def.h" +#include +#include + +#ifndef DBUG_OFF +static const char *default_dbug_option; +#endif + +#define PCACHE_SIZE (1024*1024*10) + +#define LONG_BUFFER_SIZE ((1024L*1024L*1024L) + (1024L*1024L*512)) + +#define MIN_REC_LENGTH (1024L*1024L + 1024L*512L + 1) + +#define SHOW_DIVIDER 2 + +#define LOG_FILE_SIZE (1024L*1024L*1024L + 1024L*1024L*512) +#define ITERATIONS 2 +/*#define ITERATIONS 63 */ + +/* +#define LOG_FILE_SIZE 1024L*1024L*3L +#define ITERATIONS 1600 +*/ +/* +#define LOG_FILE_SIZE 1024L*1024L*100L +#define ITERATIONS 65000 +*/ + + +/* + Check that the buffer filled correctly + + SYNOPSIS + check_content() + ptr Pointer to the buffer + length length of the buffer + + RETURN + 0 - OK + 1 - Error +*/ + +static my_bool check_content(uchar *ptr, ulong length) +{ + ulong i; + uchar buff[4]; + DBUG_ENTER("check_content"); + for (i= 0; i < length; i++) + { + if (i % 4 == 0) + int4store(buff, (i >> 2)); + if (ptr[i] != buff[i % 4]) + { + fprintf(stderr, "Byte # %lu is %x instead of %x", + i, (uint) ptr[i], (uint) buff[i % 4]); + DBUG_DUMP("mem", ptr +(ulong) (i > 16 ? i - 16 : 0), + (i > 16 ? 16 : i) + (i + 16 < length ? 16 : length - i)); + DBUG_RETURN(1); + } + } + DBUG_RETURN(0); +} + + +/* + Read whole record content, and check content (put with offset) + + SYNOPSIS + read_and_check_content() + rec The record header buffer + buffer The buffer to read the record in + skip Skip this number of bytes ot the record content + + RETURN + 0 - OK + 1 - Error +*/ + +static my_bool read_and_check_content(TRANSLOG_HEADER_BUFFER *rec, + uchar *buffer, uint skip) +{ + int res= 0; + translog_size_t len; + DBUG_ENTER("read_and_check_content"); + DBUG_ASSERT(rec->record_length < LONG_BUFFER_SIZE + 7 * 2 + 2); + if ((len= translog_read_record(&rec->lsn, 0, rec->record_length, + buffer, NULL)) != rec->record_length) + { + fprintf(stderr, "Requested %lu byte, read %lu\n", + (ulong) rec->record_length, (ulong) len); + res= 1; + } + res|= check_content(buffer + skip, rec->record_length - skip); + DBUG_RETURN(res); +} + + +static uint32 get_len() +{ + uint32 rec_len; + do + { + rec_len= random() / + (RAND_MAX / (LONG_BUFFER_SIZE - MIN_REC_LENGTH - 1)) + MIN_REC_LENGTH; + } while (rec_len >= LONG_BUFFER_SIZE); + return rec_len; +} + +int main(int argc, char *argv[]) +{ + uint32 i; + uint32 rec_len; + uint pagen; + uchar long_tr_id[6]; + uchar lsn_buff[23]= + { + 0x55, 0xAA, 0x55, 0xAA, 0x55, 0xAA, 0x55, 0xAA, + 0x55, 0xAA, 0x55, 0xAA, 0x55, 0xAA, 0x55, 0xAA, + 0x55, 0xAA, 0x55, 0xAA, 0x55, 0xAA, 0x55 + }; + uchar *long_buffer= malloc(LONG_BUFFER_SIZE + 7 * 2 + 2); + PAGECACHE pagecache; + LSN lsn, lsn_base, first_lsn, *lsn_ptr; + TRANSLOG_HEADER_BUFFER rec; + struct st_translog_scanner_data scanner; + int rc; + + MY_INIT(argv[0]); + + bzero(&pagecache, sizeof(pagecache)); + maria_data_root= "."; + + { + uchar buff[4]; + for (i= 0; i < (LONG_BUFFER_SIZE + 7 * 2 + 2); i++) + { + if (i % 4 == 0) + int4store(buff, (i >> 2)); + long_buffer[i]= buff[i % 4]; + } + } + + bzero(long_tr_id, 6); +#ifndef DBUG_OFF +#if defined(__WIN__) + default_dbug_option= "d:t:i:O,\\ma_test_loghandler.trace"; +#else + default_dbug_option= "d:t:i:o,/tmp/ma_test_loghandler.trace"; +#endif + if (argc > 1) + { + DBUG_SET(default_dbug_option); + DBUG_SET_INITIAL(default_dbug_option); + } +#endif + + if (ma_control_file_create_or_open()) + { + fprintf(stderr, "Can't init control file (%d)\n", errno); + exit(1); + } + if ((pagen= init_pagecache(&pagecache, PCACHE_SIZE, 0, 0, + TRANSLOG_PAGE_SIZE)) == 0) + { + fprintf(stderr, "Got error: init_pagecache() (errno: %d)\n", errno); + exit(1); + } + if (translog_init(".", LOG_FILE_SIZE, 50112, 0, &pagecache, 0)) + { + fprintf(stderr, "Can't init loghandler (%d)\n", errno); + translog_destroy(); + exit(1); + } + + srandom(122334817L); + + long_tr_id[5]= 0xff; + + int4store(long_tr_id, 0); + if (translog_write_record(&lsn, + LOGREC_LONG_TRANSACTION_ID, + 0, NULL, 6, long_tr_id, 0)) + { + fprintf(stderr, "Can't write record #%lu\n", (ulong) 0); + translog_destroy(); + exit(1); + } + lsn_base= first_lsn= lsn; + + for (i= 1; i < ITERATIONS; i++) + { + if (i % SHOW_DIVIDER == 0) + printf("write %d\n", i); + if (i % 2) + { + lsn7store(lsn_buff, &lsn_base); + if (translog_write_record(&lsn, + LOGREC_CLR_END, + (i % 0xFFFF), NULL, 7, lsn_buff, 0)) + { + fprintf(stderr, "1 Can't write reference before record #%lu\n", + (ulong) i); + translog_destroy(); + exit(1); + } + lsn7store(lsn_buff, &lsn_base); + rec_len= get_len(); + if (translog_write_record(&lsn, + LOGREC_UNDO_KEY_INSERT, + (i % 0xFFFF), + NULL, 7, lsn_buff, rec_len, long_buffer, 0)) + { + fprintf(stderr, "1 Can't write var reference before record #%lu\n", + (ulong) i); + translog_destroy(); + exit(1); + } + } + else + { + lsn7store(lsn_buff, &lsn_base); + lsn7store(lsn_buff + 7, &first_lsn); + if (translog_write_record(&lsn, + LOGREC_UNDO_ROW_DELETE, + (i % 0xFFFF), NULL, 23, lsn_buff, 0)) + { + fprintf(stderr, "0 Can't write reference before record #%lu\n", + (ulong) i); + translog_destroy(); + exit(1); + } + lsn7store(lsn_buff, &lsn_base); + lsn7store(lsn_buff + 7, &first_lsn); + rec_len= get_len(); + if (translog_write_record(&lsn, + LOGREC_UNDO_KEY_DELETE, + (i % 0xFFFF), + NULL, 14, lsn_buff, rec_len, long_buffer, 0)) + { + fprintf(stderr, "0 Can't write var reference before record #%lu\n", + (ulong) i); + translog_destroy(); + exit(1); + } + } + int4store(long_tr_id, i); + if (translog_write_record(&lsn, + LOGREC_LONG_TRANSACTION_ID, + (i % 0xFFFF), NULL, 6, long_tr_id, 0)) + { + fprintf(stderr, "Can't write record #%lu\n", (ulong) i); + translog_destroy(); + exit(1); + } + + lsn_base= lsn; + + rec_len= get_len(); + if (translog_write_record(&lsn, + LOGREC_REDO_INSERT_ROW_HEAD, + (i % 0xFFFF), NULL, rec_len, long_buffer, 0)) + { + fprintf(stderr, "Can't write variable record #%lu\n", (ulong) i); + translog_destroy(); + exit(1); + } + } + + translog_destroy(); + end_pagecache(&pagecache, 1); + ma_control_file_end(); + + if (ma_control_file_create_or_open()) + { + fprintf(stderr, "pass2: Can't init control file (%d)\n", errno); + exit(1); + } + if ((pagen= init_pagecache(&pagecache, PCACHE_SIZE, 0, 0, + TRANSLOG_PAGE_SIZE)) == 0) + { + fprintf(stderr, "pass2: Got error: init_pagecache() (errno: %d)\n", errno); + exit(1); + } + if (translog_init(".", LOG_FILE_SIZE, 50112, 0, &pagecache, 0)) + { + fprintf(stderr, "pass2: Can't init loghandler (%d)\n", errno); + translog_destroy(); + exit(1); + } + + srandom(122334817L); + + rc= 1; + + { + translog_size_t len= translog_read_record_header(&first_lsn, &rec); + if (len == 0) + { + fprintf(stderr, "translog_read_record_header failed (%d)\n", errno); + translog_free_record_header(&rec); + goto err; + } + if (rec.type !=LOGREC_LONG_TRANSACTION_ID || rec.short_trid != 0 || + rec.record_length != 6 || uint4korr(rec.header) != 0 || + (uint)rec.header[4] != 0 || rec.header[5] != 0xFF || + first_lsn.file_no != rec.lsn.file_no || + first_lsn.rec_offset != rec.lsn.rec_offset) + { + fprintf(stderr, "Incorrect LOGREC_LONG_TRANSACTION_ID data read(0)\n" + "type %u, strid %u, len %u, i: %u, 4: %u 5: %u, " + "lsn(0x%lx,0x%lx)\n", + (uint) rec.type, (uint) rec.short_trid, (uint) rec.record_length, + uint4korr(rec.header), (uint) rec.header[4], (uint) rec.header[5], + (ulong) rec.lsn.file_no, (ulong) rec.lsn.rec_offset); + translog_free_record_header(&rec); + goto err; + } + translog_free_record_header(&rec); + lsn= first_lsn; + lsn_ptr= &first_lsn; + for (i= 1;; i++) + { + if (i % SHOW_DIVIDER == 0) + printf("read %d\n", i); + len= translog_read_next_record_header(lsn_ptr, &rec, 1, &scanner); + if (len == 0) + { + fprintf(stderr, "1-%d translog_read_next_record_header failed (%d)\n", + i, errno); + translog_free_record_header(&rec); + goto err; + } + if (rec.lsn.file_no == CONTROL_FILE_IMPOSSIBLE_FILENO) + { + if (i != ITERATIONS) + { + fprintf(stderr, "EOL met at iteration %u instead of %u\n", + i, ITERATIONS); + translog_free_record_header(&rec); + goto err; + } + break; + } + lsn_ptr= NULL; /* use scanner after its + initialization */ + + if (i % 2) + { + LSN ref; + lsn7korr(&ref, rec.header); + if (rec.type !=LOGREC_CLR_END || rec.short_trid != (i % 0xFFFF) || + rec.record_length != 7 || ref.file_no != lsn.file_no || + ref.rec_offset != lsn.rec_offset) + { + fprintf(stderr, "Incorrect LOGREC_CLR_END data read(%d)" + "type %u, strid %u, len %u, ref(%u,0x%lx), lsn(%u,0x%lx)\n", + i, (uint) rec.type, (uint) rec.short_trid, + (uint) rec.record_length, + (uint) ref.file_no, (ulong) ref.rec_offset, + (uint) rec.lsn.file_no, (ulong) rec.lsn.rec_offset); + translog_free_record_header(&rec); + goto err; + } + } + else + { + LSN ref1, ref2; + lsn7korr(&ref1, rec.header); + lsn7korr(&ref2, rec.header + 7); + if (rec.type !=LOGREC_UNDO_ROW_DELETE || + rec.short_trid != (i % 0xFFFF) || + rec.record_length != 23 || + ref1.file_no != lsn.file_no || + ref1.rec_offset != lsn.rec_offset || + ref2.file_no != first_lsn.file_no || + ref2.rec_offset != first_lsn.rec_offset || + rec.header[22] != 0x55 || rec.header[21] != 0xAA || + rec.header[20] != 0x55 || rec.header[19] != 0xAA || + rec.header[18] != 0x55 || rec.header[17] != 0xAA || + rec.header[16] != 0x55 || rec.header[15] != 0xAA || + rec.header[14] != 0x55) + { + fprintf(stderr, "Incorrect LOGREC_UNDO_ROW_DELETE data read(%d)" + "type %u, strid %u, len %u, ref1(%u,0x%lx), " + "ref2(%u,0x%lx) %x%x%x%x%x%x%x%x%x " + "lsn(%u,0x%lx)\n", + i, (uint) rec.type, (uint) rec.short_trid, + (uint) rec.record_length, + (uint) ref1.file_no, (ulong) ref1.rec_offset, + (uint) ref2.file_no, (ulong) ref2.rec_offset, + (uint) rec.header[14], (uint) rec.header[15], + (uint) rec.header[16], (uint) rec.header[17], + (uint) rec.header[18], (uint) rec.header[19], + (uint) rec.header[20], (uint) rec.header[21], + (uint) rec.header[22], + (uint) rec.lsn.file_no, (ulong) rec.lsn.rec_offset); + translog_free_record_header(&rec); + goto err; + } + } + translog_free_record_header(&rec); + + len= translog_read_next_record_header(lsn_ptr, &rec, 1, &scanner); + if (len == 0) + { + fprintf(stderr, "1-%d translog_read_next_record_header (var) " + "failed (%d)\n", i, errno); + goto err; + } + if (rec.lsn.file_no == CONTROL_FILE_IMPOSSIBLE_FILENO) + { + fprintf(stderr, "EOL met at the middle of iteration (first var) %u " + "instead of beginning of %u\n", i, ITERATIONS); + goto err; + } + if (i % 2) + { + LSN ref; + lsn7korr(&ref, rec.header); + rec_len= get_len(); + if (rec.type !=LOGREC_UNDO_KEY_INSERT || + rec.short_trid != (i % 0xFFFF) || + rec.record_length != rec_len + 7 || + len != 12 || ref.file_no != lsn.file_no || + ref.rec_offset != lsn.rec_offset || + check_content(rec.header + 7, len - 7)) + { + fprintf(stderr, "Incorrect LOGREC_UNDO_KEY_INSERT data read(%d)" + "type %u (%d), strid %u (%d), len %lu, %lu + 7 (%d), " + "hdr len: %u (%d), " + "ref(%u,0x%lx), lsn(%u,0x%lx) (%d), content: %d\n", + i, (uint) rec.type, + rec.type !=LOGREC_UNDO_KEY_INSERT, + (uint) rec.short_trid, + rec.short_trid != (i % 0xFFFF), + (ulong) rec.record_length, (ulong) rec_len, + rec.record_length != rec_len + 7, + (uint) len, + len != 12, + (uint) ref.file_no, (ulong) ref.rec_offset, + (uint) rec.lsn.file_no, (ulong) rec.lsn.rec_offset, + (ref.file_no != lsn.file_no || + ref.rec_offset != lsn.rec_offset), + check_content(rec.header + 7, len - 7)); + translog_free_record_header(&rec); + goto err; + } + if (read_and_check_content(&rec, long_buffer, 7)) + { + fprintf(stderr, + "Incorrect LOGREC_UNDO_KEY_INSERT in whole rec read " + "lsn(%u,0x%lx)\n", + (uint) rec.lsn.file_no, (ulong) rec.lsn.rec_offset); + translog_free_record_header(&rec); + goto err; + } + } + else + { + LSN ref1, ref2; + lsn7korr(&ref1, rec.header); + lsn7korr(&ref2, rec.header + 7); + rec_len= get_len(); + if (rec.type !=LOGREC_UNDO_KEY_DELETE || + rec.short_trid != (i % 0xFFFF) || + rec.record_length != rec_len + 14 || + len != 19 || + ref1.file_no != lsn.file_no || + ref1.rec_offset != lsn.rec_offset || + ref2.file_no != first_lsn.file_no || + ref2.rec_offset != first_lsn.rec_offset || + check_content(rec.header + 14, len - 14)) + { + fprintf(stderr, "Incorrect LOGREC_UNDO_KEY_DELETE data read(%d)" + "type %u, strid %u, len %lu != %lu + 7, hdr len: %u, " + "ref1(%u,0x%lx), ref2(%u,0x%lx), " + "lsn(%u,0x%lx)\n", + i, (uint) rec.type, (uint) rec.short_trid, + (ulong) rec.record_length, (ulong) rec_len, + (uint) len, + (uint) ref1.file_no, (ulong) ref1.rec_offset, + (uint) ref2.file_no, (ulong) ref2.rec_offset, + (uint) rec.lsn.file_no, (ulong) rec.lsn.rec_offset); + translog_free_record_header(&rec); + goto err; + } + if (read_and_check_content(&rec, long_buffer, 14)) + { + fprintf(stderr, + "Incorrect LOGREC_UNDO_KEY_DELETE in whole rec read " + "lsn(%u,0x%lx)\n", + (uint) rec.lsn.file_no, (ulong) rec.lsn.rec_offset); + translog_free_record_header(&rec); + goto err; + } + } + translog_free_record_header(&rec); + + len= translog_read_next_record_header(lsn_ptr, &rec, 1, &scanner); + if (len == 0) + { + fprintf(stderr, "1-%d translog_read_next_record_header failed (%d)\n", + i, errno); + translog_free_record_header(&rec); + goto err; + } + if (rec.lsn.file_no == CONTROL_FILE_IMPOSSIBLE_FILENO) + { + fprintf(stderr, "EOL met at the middle of iteration %u " + "instead of beginning of %u\n", i, ITERATIONS); + translog_free_record_header(&rec); + goto err; + } + if (rec.type !=LOGREC_LONG_TRANSACTION_ID || + rec.short_trid != (i % 0xFFFF) || + rec.record_length != 6 || uint4korr(rec.header) != i || + rec.header[4] != 0 || rec.header[5] != 0xFF) + { + fprintf(stderr, "Incorrect LOGREC_LONG_TRANSACTION_ID data read(%d)\n" + "type %u, strid %u, len %u, i: %u, 4: %u 5: %u " + "lsn(%u,0x%lx)\n", + i, (uint) rec.type, (uint) rec.short_trid, + (uint) rec.record_length, + uint4korr(rec.header), (uint) rec.header[4], + (uint) rec.header[5], + (uint) rec.lsn.file_no, (ulong) rec.lsn.rec_offset); + translog_free_record_header(&rec); + goto err; + } + translog_free_record_header(&rec); + + lsn= rec.lsn; + + len= translog_read_next_record_header(lsn_ptr, &rec, 1, &scanner); + rec_len= get_len(); + if (rec.type !=LOGREC_REDO_INSERT_ROW_HEAD || + rec.short_trid != (i % 0xFFFF) || + rec.record_length != rec_len || + len != 9 || check_content(rec.header, len)) + { + fprintf(stderr, "Incorrect LOGREC_REDO_INSERT_ROW_HEAD data read(%d)" + "type %u, strid %u, len %lu != %lu, hdr len: %u, " + "lsn(%u,0x%lx)\n", + i, (uint) rec.type, (uint) rec.short_trid, + (ulong) rec.record_length, (ulong) rec_len, + (uint) len, (uint) rec.lsn.file_no, (ulong) rec.lsn.rec_offset); + translog_free_record_header(&rec); + goto err; + } + if (read_and_check_content(&rec, long_buffer, 0)) + { + fprintf(stderr, + "Incorrect LOGREC_UNDO_KEY_DELETE in whole rec read " + "lsn(%u,0x%lx)\n", + (uint) rec.lsn.file_no, (ulong) rec.lsn.rec_offset); + translog_free_record_header(&rec); + goto err; + } + } + } + + rc= 0; +err: + translog_destroy(); + end_pagecache(&pagecache, 1); + ma_control_file_end(); + + return (test(exit_status() || rc)); +} diff --git a/storage/maria/unittest/ma_test_loghandler_multithread-t.c b/storage/maria/unittest/ma_test_loghandler_multithread-t.c new file mode 100644 index 00000000000..794dc6dd255 --- /dev/null +++ b/storage/maria/unittest/ma_test_loghandler_multithread-t.c @@ -0,0 +1,468 @@ +#include "../maria_def.h" +#include +#include + +#ifndef DBUG_OFF +static const char *default_dbug_option; +#endif + +#define PCACHE_SIZE (1024*1024*10) + +/*#define LOG_FLAGS TRANSLOG_SECTOR_PROTECTION | TRANSLOG_PAGE_CRC */ +#define LOG_FLAGS 0 +/*#define LONG_BUFFER_SIZE (1024L*1024L*1024L + 1024L*1024L*512)*/ +#define LONG_BUFFER_SIZE (1024L*1024L*1024L) +#define MIN_REC_LENGTH 30 +#define SHOW_DIVIDER 10 +#define LOG_FILE_SIZE (1024L*1024L*1024L + 1024L*1024L*512) +#define ITERATIONS 3 +#define WRITERS 3 +static uint number_of_writers= WRITERS; + +static pthread_cond_t COND_thread_count; +static pthread_mutex_t LOCK_thread_count; +static uint thread_count; + +static ulong lens[WRITERS][ITERATIONS]; +static LSN lsns1[WRITERS][ITERATIONS]; +static LSN lsns2[WRITERS][ITERATIONS]; +static uchar *long_buffer; + +/* + Get pseudo-random length of the field in + limits [MIN_REC_LENGTH..LONG_BUFFER_SIZE] + + SYNOPSYS + get_len() + + RETURN + length - length >= 0 length <= LONG_BUFFER_SIZE +*/ + +static uint32 get_len() +{ + uint32 rec_len; + do + { + rec_len= random() / + (RAND_MAX / (LONG_BUFFER_SIZE - MIN_REC_LENGTH - 1)) + MIN_REC_LENGTH; + } while (rec_len >= LONG_BUFFER_SIZE); + return rec_len; +} + + +/* + Check that the buffer filled correctly + + SYNOPSIS + check_content() + ptr Pointer to the buffer + length length of the buffer + + RETURN + 0 - OK + 1 - Error +*/ + +static my_bool check_content(uchar *ptr, ulong length) +{ + ulong i; + for (i= 0; i < length; i++) + { + if (ptr[i] != (i & 0xFF)) + { + fprintf(stderr, "Byte # %lu is %x instead of %x", + i, (uint) ptr[i], (uint) (i & 0xFF)); + return 1; + } + } + return 0; +} + + +/* + Read whole record content, and check content (put with offset) + + SYNOPSIS + read_and_check_content() + rec The record header buffer + buffer The buffer to read the record in + skip Skip this number of bytes ot the record content + + RETURN + 0 - OK + 1 - Error +*/ + + +static my_bool read_and_check_content(TRANSLOG_HEADER_BUFFER *rec, + uchar *buffer, uint skip) +{ + int res= 0; + translog_size_t len; + DBUG_ENTER("read_and_check_content"); + DBUG_ASSERT(rec->record_length < LONG_BUFFER_SIZE + 7 * 2 + 2); + if ((len= translog_read_record(&rec->lsn, 0, rec->record_length, + buffer, NULL)) != rec->record_length) + { + fprintf(stderr, "Requested %lu byte, read %lu\n", + (ulong) rec->record_length, (ulong) len); + res= 1; + } + res|= check_content(buffer + skip, rec->record_length - skip); + DBUG_RETURN(res); +} + +void writer(int num) +{ + LSN lsn; + uchar long_tr_id[6]; + uint i; + DBUG_ENTER("writer"); + + for (i= 0; i < ITERATIONS; i++) + { + uint len= get_len(); + lens[num][i]= len; + + int2store(long_tr_id, num); + int4store(long_tr_id + 2, i); + if (translog_write_record(&lsn, + LOGREC_LONG_TRANSACTION_ID, + num, NULL, 6, long_tr_id, 0)) + { + fprintf(stderr, "Can't write LOGREC_LONG_TRANSACTION_ID record #%lu " + "thread %i\n", (ulong) i, num); + translog_destroy(); + return; + } + lsns1[num][i]= lsn; + if (translog_write_record(&lsn, + LOGREC_REDO_INSERT_ROW_HEAD, + num, NULL, len, long_buffer, 0)) + { + fprintf(stderr, "Can't write variable record #%lu\n", (ulong) i); + translog_destroy(); + return; + } + lsns2[num][i]= lsn; + DBUG_PRINT("info", ("thread: %u, iteration: %u, len: %lu, " + "lsn1 (%lu,0x%lx) lsn2 (%lu,0x%lx)", + num, i, (ulong) lens[num][i], + (ulong) lsns1[num][i].file_no, + (ulong) lsns1[num][i].rec_offset, + (ulong) lsns2[num][i].file_no, + (ulong) lsns2[num][i].rec_offset)); + printf("thread: %u, iteration: %u, len: %lu, " + "lsn1 (%lu,0x%lx) lsn2 (%lu,0x%lx)\n", + num, i, (ulong) lens[num][i], + (ulong) lsns1[num][i].file_no, + (ulong) lsns1[num][i].rec_offset, + (ulong) lsns2[num][i].file_no, (ulong) lsns2[num][i].rec_offset); + } + DBUG_VOID_RETURN; +} + + +static void *test_thread_writer(void *arg) +{ + int param= *((int*) arg); + + my_thread_init(); + DBUG_ENTER("test_writer"); + DBUG_PRINT("enter", ("param: %d", param)); + + writer(param); + + DBUG_PRINT("info", ("Thread %s ended\n", my_thread_name())); + pthread_mutex_lock(&LOCK_thread_count); + thread_count--; + VOID(pthread_cond_signal(&COND_thread_count)); /* Tell main we are + ready */ + pthread_mutex_unlock(&LOCK_thread_count); + free((gptr) arg); + my_thread_end(); + DBUG_RETURN(0); +} + + +int main(int argc, char **argv __attribute__ ((unused))) +{ + uint32 i; + uint pagen; + PAGECACHE pagecache; + LSN first_lsn, *lsn_ptr; + TRANSLOG_HEADER_BUFFER rec; + struct st_translog_scanner_data scanner; + pthread_t tid; + pthread_attr_t thr_attr; + int *param, error; + int rc; + + bzero(&pagecache, sizeof(pagecache)); + maria_data_root= "."; + long_buffer= malloc(LONG_BUFFER_SIZE + 7 * 2 + 2); + if (long_buffer == 0) + { + fprintf(stderr, "End of memory\n"); + exit(1); + } + for (i= 0; i < (LONG_BUFFER_SIZE + 7 * 2 + 2); i++) + long_buffer[i]= (i & 0xFF); + + + MY_INIT(argv[0]); + +#ifndef DBUG_OFF +#if defined(__WIN__) + default_dbug_option= "d:t:i:O,\\ma_test_loghandler.trace"; +#else + default_dbug_option= "d:t:i:o,/tmp/ma_test_loghandler.trace"; +#endif + if (argc > 1) + { + DBUG_SET(default_dbug_option); + DBUG_SET_INITIAL(default_dbug_option); + } +#endif + + DBUG_ENTER("main"); + DBUG_PRINT("info", ("Main thread: %s\n", my_thread_name())); + + if ((error= pthread_cond_init(&COND_thread_count, NULL))) + { + fprintf(stderr, "COND_thread_count: %d from pthread_cond_init " + "(errno: %d)\n", error, errno); + exit(1); + } + if ((error= pthread_mutex_init(&LOCK_thread_count, MY_MUTEX_INIT_FAST))) + { + fprintf(stderr, "LOCK_thread_count: %d from pthread_cond_init " + "(errno: %d)\n", error, errno); + exit(1); + } + if ((error= pthread_attr_init(&thr_attr))) + { + fprintf(stderr, "Got error: %d from pthread_attr_init " + "(errno: %d)\n", error, errno); + exit(1); + } + if ((error= pthread_attr_setdetachstate(&thr_attr, PTHREAD_CREATE_DETACHED))) + { + fprintf(stderr, + "Got error: %d from pthread_attr_setdetachstate (errno: %d)\n", + error, errno); + exit(1); + } +#ifndef pthread_attr_setstacksize /* void return value */ + if ((error= pthread_attr_setstacksize(&thr_attr, 65536L))) + { + fprintf(stderr, "Got error: %d from pthread_attr_setstacksize " + "(errno: %d)\n", error, errno); + exit(1); + } +#endif +#ifdef HAVE_THR_SETCONCURRENCY + VOID(thr_setconcurrency(2)); +#endif + + my_thread_global_init(); + + if (ma_control_file_create_or_open()) + { + fprintf(stderr, "Can't init control file (%d)\n", errno); + exit(1); + } + if ((pagen= init_pagecache(&pagecache, PCACHE_SIZE, 0, 0, + TRANSLOG_PAGE_SIZE)) == 0) + { + fprintf(stderr, "Got error: init_pagecache() (errno: %d)\n", errno); + exit(1); + } + if (translog_init(".", LOG_FILE_SIZE, 50112, 0, &pagecache, LOG_FLAGS)) + { + fprintf(stderr, "Can't init loghandler (%d)\n", errno); + translog_destroy(); + exit(1); + } + + srandom(122334817L); + { + uchar long_tr_id[6]= + { + 0x11, 0x22, 0x33, 0x44, 0x55, 0x66 + }; + + if (translog_write_record(&first_lsn, + LOGREC_LONG_TRANSACTION_ID, + 0, NULL, 6, long_tr_id, 0)) + { + fprintf(stderr, "Can't write the first record\n"); + translog_destroy(); + exit(1); + } + } + + + if ((error= pthread_mutex_lock(&LOCK_thread_count))) + { + fprintf(stderr, "LOCK_thread_count: %d from pthread_mutex_lock " + "(errno: %d)\n", error, errno); + exit(1); + } + + while (number_of_writers != 0) + { + param= (int*) malloc(sizeof(int)); + *param= number_of_writers - 1; + if ((error= pthread_create(&tid, &thr_attr, test_thread_writer, + (void*) param))) + { + fprintf(stderr, "Got error: %d from pthread_create (errno: %d)\n", + error, errno); + exit(1); + } + thread_count++; + number_of_writers--; + } + DBUG_PRINT("info", ("All threads are started")); + pthread_mutex_unlock(&LOCK_thread_count); + + pthread_attr_destroy(&thr_attr); + + /* wait finishing */ + if ((error= pthread_mutex_lock(&LOCK_thread_count))) + fprintf(stderr, "LOCK_thread_count: %d from pthread_mutex_lock\n", error); + while (thread_count) + { + if ((error= pthread_cond_wait(&COND_thread_count, &LOCK_thread_count))) + fprintf(stderr, "COND_thread_count: %d from pthread_cond_wait\n", error); + } + if ((error= pthread_mutex_unlock(&LOCK_thread_count))) + fprintf(stderr, "LOCK_thread_count: %d from pthread_mutex_unlock\n", error); + DBUG_PRINT("info", ("All threads ended")); + + /* Find last LSN and flush up to it (all our log) */ + { + LSN max= + { + 0, 0 + }; + for (i= 0; i < WRITERS; i++) + { + if (cmp_translog_addr(lsns2[i][ITERATIONS - 1], max) > 0) + max= lsns2[i][ITERATIONS - 1]; + } + DBUG_PRINT("info", ("first lsn: (%lu,0x%lx), max lsn: (%lu,0x%lx)", + (ulong) first_lsn.file_no, + (ulong) first_lsn.rec_offset, + (ulong) max.file_no, (ulong) max.rec_offset)); + translog_flush(&max); + } + + rc= 1; + + { + uint indeces[WRITERS]; + uint index, len, stage; + bzero(indeces, sizeof(uint) * WRITERS); + + bzero(indeces, sizeof(indeces)); + + lsn_ptr= &first_lsn; + for (i= 0;; i++) + { + len= translog_read_next_record_header(lsn_ptr, &rec, 1, &scanner); + lsn_ptr= NULL; + + if (len == 0) + { + fprintf(stderr, "1-%d translog_read_next_record_header failed (%d)\n", + i, errno); + translog_free_record_header(&rec); + goto err; + } + if (rec.lsn.file_no == CONTROL_FILE_IMPOSSIBLE_FILENO) + { + if (i != WRITERS * ITERATIONS * 2) + { + fprintf(stderr, "EOL met at iteration %u instead of %u\n", + i, ITERATIONS * WRITERS * 2); + translog_free_record_header(&rec); + goto err; + } + break; + } + index= indeces[rec.short_trid] / 2; + stage= indeces[rec.short_trid] % 2; + printf("read(%d) thread: %d, iteration %d, stage %d\n", + i, (uint) rec.short_trid, index, stage); + if (stage == 0) + { + if (rec.type !=LOGREC_LONG_TRANSACTION_ID || + rec.record_length != 6 || + uint2korr(rec.header) != rec.short_trid || + index != uint4korr(rec.header + 2) || + cmp_translog_addr(lsns1[rec.short_trid][index], rec.lsn) != 0) + { + fprintf(stderr, "Incorrect LOGREC_LONG_TRANSACTION_ID data read(%d)\n" + "type %u, strid %u %u, len %u, i: %u %u, " + "lsn(%lu,0x%lx) (%lu,0x%lx)\n", + i, (uint) rec.type, + (uint) rec.short_trid, (uint) uint2korr(rec.header), + (uint) rec.record_length, + (uint) index, (uint) uint4korr(rec.header + 2), + (ulong) rec.lsn.file_no, (ulong) rec.lsn.rec_offset, + (ulong) lsns1[rec.short_trid][index].file_no, + (ulong) lsns1[rec.short_trid][index].rec_offset); + translog_free_record_header(&rec); + goto err; + } + } + else + { + if (rec.type !=LOGREC_REDO_INSERT_ROW_HEAD || + len != 9 || + rec.record_length != lens[rec.short_trid][index] || + cmp_translog_addr(lsns2[rec.short_trid][index], rec.lsn) != 0 || + check_content(rec.header, len)) + { + fprintf(stderr, + "Incorrect LOGREC_REDO_INSERT_ROW_HEAD data read(%d) " + " thread: %d, iteration %d, stage %d\n" + "type %u (%d), len %u, length %lu %lu (%d) " + "lsn(%lu,0x%lx) (%lu,0x%lx)\n", + i, (uint) rec.short_trid, index, stage, + (uint) rec.type, (rec.type !=LOGREC_REDO_INSERT_ROW_HEAD), + (uint) len, + (ulong) rec.record_length, lens[rec.short_trid][index], + (rec.record_length != lens[rec.short_trid][index]), + (ulong) rec.lsn.file_no, (ulong) rec.lsn.rec_offset, + (ulong) lsns2[rec.short_trid][index].file_no, + (ulong) lsns2[rec.short_trid][index].rec_offset); + translog_free_record_header(&rec); + goto err; + } + if (read_and_check_content(&rec, long_buffer, 0)) + { + fprintf(stderr, + "Incorrect LOGREC_REDO_INSERT_ROW_HEAD in whole rec read " + "lsn(%u,0x%lx)\n", + (uint) rec.lsn.file_no, (ulong) rec.lsn.rec_offset); + translog_free_record_header(&rec); + goto err; + } + } + translog_free_record_header(&rec); + indeces[rec.short_trid]++; + } + } + + rc= 0; +err: + translog_destroy(); + end_pagecache(&pagecache, 1); + ma_control_file_end(); + + DBUG_RETURN(test(exit_status() || rc)); +} diff --git a/storage/maria/unittest/ma_test_loghandler_pagecache-t.c b/storage/maria/unittest/ma_test_loghandler_pagecache-t.c new file mode 100644 index 00000000000..6771b5f888d --- /dev/null +++ b/storage/maria/unittest/ma_test_loghandler_pagecache-t.c @@ -0,0 +1,140 @@ +#include "../maria_def.h" +#include +#include + +#ifndef DBUG_OFF +static const char *default_dbug_option; +#endif + +#define PCACHE_SIZE (1024*1024*10) +#define PCACHE_PAGE TRANSLOG_PAGE_SIZE +#define LOG_FILE_SIZE (1024L*1024L*1024L + 1024L*1024L*512) +#define LOG_FLAGS 0 + +static char *first_translog_file= (char*)"maria_log.00000001"; +static char *file1_name= (char*)"page_cache_test_file_1"; +static PAGECACHE_FILE file1; + +int main(int argc, char *argv[]) +{ + uint pagen; + uchar long_tr_id[6]; + PAGECACHE pagecache; + LSN lsn; + MY_STAT st, *stat; + + MY_INIT(argv[0]); + + bzero(&pagecache, sizeof(pagecache)); + maria_data_root= "."; + /* be sure that we have no logs in the directory*/ + if (my_stat(CONTROL_FILE_BASE_NAME, &st, MYF(0))) + my_delete(CONTROL_FILE_BASE_NAME, MYF(0)); + if (my_stat(first_translog_file, &st, MYF(0))) + my_delete(first_translog_file, MYF(0)); + + bzero(long_tr_id, 6); +#ifndef DBUG_OFF +#if defined(__WIN__) + default_dbug_option= "d:t:i:O,\\ma_test_loghandler_pagecache.trace"; +#else + default_dbug_option= "d:t:i:o,/tmp/ma_test_loghandler_pagecache.trace"; +#endif + if (argc > 1) + { + DBUG_SET(default_dbug_option); + DBUG_SET_INITIAL(default_dbug_option); + } +#endif + + if (ma_control_file_create_or_open()) + { + fprintf(stderr, "Can't init control file (%d)\n", errno); + exit(1); + } + if ((pagen= init_pagecache(&pagecache, PCACHE_SIZE, 0, 0, + PCACHE_PAGE)) == 0) + { + fprintf(stderr, "Got error: init_pagecache() (errno: %d)\n", errno); + exit(1); + } + if (translog_init(".", LOG_FILE_SIZE, 50112, 0, &pagecache, LOG_FLAGS)) + { + fprintf(stderr, "Can't init loghandler (%d)\n", errno); + translog_destroy(); + exit(1); + } + + if ((stat= my_stat(first_translog_file, &st, MYF(0))) == 0) + { + fprintf(stderr, "There is no %s (%d)\n", first_translog_file, errno); + exit(1); + } + if (st.st_size != TRANSLOG_PAGE_SIZE) + { + fprintf(stderr, + "incorrect initial size of %s: %ld instead of %ld\n", + first_translog_file, (long)st.st_size, (long)TRANSLOG_PAGE_SIZE); + exit(1); + } + int4store(long_tr_id, 0); + if (translog_write_record(&lsn, + LOGREC_LONG_TRANSACTION_ID, + 0, NULL, 6, long_tr_id, 0)) + { + fprintf(stderr, "Can't write record #%lu\n", (ulong) 0); + translog_destroy(); + exit(1); + } + + if ((file1.file= my_open(file1_name, + O_CREAT | O_TRUNC | O_RDWR, MYF(0))) == -1) + { + fprintf(stderr, "Got error during file1 creation from open() (errno: %d)\n", + errno); + exit(1); + } + if (chmod(file1_name, S_IRWXU | S_IRWXG | S_IRWXO) != 0) + { + fprintf(stderr, "Got error during file1 chmod() (errno: %d)\n", + errno); + exit(1); + } + + { + uchar page[PCACHE_PAGE]; + + bzero(page, PCACHE_PAGE); +#define PAGE_LSN_OFFSET 0 + lsn7store(page + PAGE_LSN_OFFSET, &lsn); + pagecache_write(&pagecache, &file1, 0, 3, (char*)page, + PAGECACHE_LSN_PAGE, + PAGECACHE_LOCK_LEFT_UNLOCKED, + PAGECACHE_PIN_LEFT_UNPINNED, + PAGECACHE_WRITE_DELAY, + 0); + flush_pagecache_blocks(&pagecache, &file1, FLUSH_FORCE_WRITE); + } + if ((stat= my_stat(first_translog_file, &st, MYF(0))) == 0) + { + fprintf(stderr, "can't stat %s (%d)\n", first_translog_file, errno); + exit(1); + } + if (st.st_size != TRANSLOG_PAGE_SIZE * 2) + { + fprintf(stderr, + "incorrect initial size of %s: %ld instead of %ld\n", + first_translog_file, + (long)st.st_size, (long)(TRANSLOG_PAGE_SIZE * 2)); + exit(1); + } + + translog_destroy(); + end_pagecache(&pagecache, 1); + ma_control_file_end(); + my_delete(CONTROL_FILE_BASE_NAME, MYF(0)); + my_delete(first_translog_file, MYF(0)); + my_delete(file1_name, MYF(0)); + + exit(0); +} diff --git a/storage/maria/unittest/mf_pagecache_consist.c b/storage/maria/unittest/mf_pagecache_consist.c new file mode 100755 index 00000000000..9389e5a093c --- /dev/null +++ b/storage/maria/unittest/mf_pagecache_consist.c @@ -0,0 +1,468 @@ +/* + TODO: use pthread_join instead of wait_for_thread_count_to_be_zero, like in + my_atomic-t.c (see BUG#22320). + Use diag() instead of fprintf(stderr). Use ok() and plan(). +*/ + +#include +#include +#include +#include "test_file.h" + +#define PCACHE_SIZE (PAGE_SIZE*1024*8) + +#ifndef DBUG_OFF +static const char* default_dbug_option; +#endif + +static char *file1_name= (char*)"page_cache_test_file_1"; +static PAGECACHE_FILE file1; +static pthread_cond_t COND_thread_count; +static pthread_mutex_t LOCK_thread_count; +static uint thread_count; +static PAGECACHE pagecache; + +#ifdef TEST_HIGH_CONCURENCY +static uint number_of_readers= 10; +static uint number_of_writers= 20; +static uint number_of_tests= 30000; +static uint record_length_limit= PAGE_SIZE/200; +static uint number_of_pages= 20; +static uint flush_divider= 1000; +#else /*TEST_HIGH_CONCURENCY*/ +#ifdef TEST_READERS +static uint number_of_readers= 10; +static uint number_of_writers= 1; +static uint number_of_tests= 30000; +static uint record_length_limit= PAGE_SIZE/200; +static uint number_of_pages= 20; +static uint flush_divider= 1000; +#else /*TEST_READERS*/ +#ifdef TEST_WRITERS +static uint number_of_readers= 0; +static uint number_of_writers= 10; +static uint number_of_tests= 30000; +static uint record_length_limit= PAGE_SIZE/200; +static uint number_of_pages= 20; +static uint flush_divider= 1000; +#else /*TEST_WRITERS*/ +static uint number_of_readers= 10; +static uint number_of_writers= 10; +static uint number_of_tests= 50000; +static uint record_length_limit= PAGE_SIZE/200; +static uint number_of_pages= 20000; +static uint flush_divider= 1000; +#endif /*TEST_WRITERS*/ +#endif /*TEST_READERS*/ +#endif /*TEST_HIGH_CONCURENCY*/ + + +/* + Get pseudo-random length of the field in (0;limit) + + SYNOPSYS + get_len() + limit limit for generated value + + RETURN + length where length >= 0 & length < limit +*/ + +static uint get_len(uint limit) +{ + uint32 rec_len; + do + { + rec_len= random() / + (RAND_MAX / limit); + } while (rec_len >= limit || rec_len == 0); + return rec_len; +} + + +/* check page consistency */ +uint check_page(uchar *buff, ulong offset, int page_locked, int page_no, + int tag) +{ + uint end= sizeof(uint); + uint num= *((uint *)buff); + uint i; + DBUG_ENTER("check_page"); + + for (i= 0; i < num; i++) + { + uint len= *((uint *)(buff + end)); + uint j; + end+= sizeof(uint) + sizeof(uint); + if (len + end > PAGE_SIZE) + { + diag("incorrect field header #%u by offset %lu\n", i, offset + end + j); + goto err; + } + for(j= 0; j < len; j++) + { + if (buff[end + j] != (uchar)((i+1) % 256)) + { + diag("incorrect %lu byte\n", offset + end + j); + goto err; + } + } + end+= len; + } + for(i= end; i < PAGE_SIZE; i++) + { + if (buff[i] != 0) + { + int h; + DBUG_PRINT("err", + ("byte %lu (%lu + %u), page %u (%s, end: %u, recs: %u, tag: %d) should be 0\n", + offset + i, offset, i, page_no, + (page_locked ? "locked" : "unlocked"), + end, num, tag)); + diag("byte %lu (%lu + %u), page %u (%s, end: %u, recs: %u, tag: %d) should be 0\n", + offset + i, offset, i, page_no, + (page_locked ? "locked" : "unlocked"), + end, num, tag); + h= my_open("wrong_page", O_CREAT | O_TRUNC | O_RDWR, MYF(0)); + my_pwrite(h, buff, PAGE_SIZE, 0, MYF(0)); + my_close(h, MYF(0)); + goto err; + } + } + DBUG_RETURN(end); +err: + DBUG_PRINT("err", ("try to flush")); + if (page_locked) + { + pagecache_delete_page(&pagecache, &file1, page_no, + PAGECACHE_LOCK_LEFT_WRITELOCKED, 1); + } + else + { + flush_pagecache_blocks(&pagecache, &file1, FLUSH_RELEASE); + } + exit(1); +} + +void put_rec(uchar *buff, uint end, uint len, uint tag) +{ + uint i; + uint num= *((uint *)buff); + if (!len) + len= 1; + if (end + sizeof(uint)*2 + len > PAGE_SIZE) + return; + *((uint *)(buff + end))= len; + end+= sizeof(uint); + *((uint *)(buff + end))= tag; + end+= sizeof(uint); + num++; + *((uint *)buff)= num; + *((uint*)(buff + end))= len; + for (i= end; i < (len + end); i++) + { + buff[i]= (uchar) num % 256; + } +} + +/* + Recreate and reopen a file for test + + SYNOPSIS + reset_file() + file File to reset + file_name Path (and name) of file which should be reset +*/ + +void reset_file(PAGECACHE_FILE file, char *file_name) +{ + flush_pagecache_blocks(&pagecache, &file1, FLUSH_RELEASE); + if (my_close(file1.file, MYF(0)) != 0) + { + diag("Got error during %s closing from close() (errno: %d)\n", + file_name, errno); + exit(1); + } + my_delete(file_name, MYF(0)); + if ((file.file= my_open(file_name, + O_CREAT | O_TRUNC | O_RDWR, MYF(0))) == -1) + { + diag("Got error during %s creation from open() (errno: %d)\n", + file_name, errno); + exit(1); + } +} + + +void reader(int num) +{ + unsigned char *buffr= malloc(PAGE_SIZE); + uint i; + + for (i= 0; i < number_of_tests; i++) + { + uint page= get_len(number_of_pages); + pagecache_read(&pagecache, &file1, page, 3, (char*)buffr, + PAGECACHE_PLAIN_PAGE, + PAGECACHE_LOCK_LEFT_UNLOCKED, + 0); + check_page(buffr, page * PAGE_SIZE, 0, page, -num); + if (i % 500 == 0) + printf("reader%d: %d\n", num, i); + + } + printf("reader%d: done\n", num); + free(buffr); +} + + +void writer(int num) +{ + unsigned char *buffr= malloc(PAGE_SIZE); + uint i; + + for (i= 0; i < number_of_tests; i++) + { + uint end; + uint page= get_len(number_of_pages); + pagecache_read(&pagecache, &file1, page, 3, (char*)buffr, + PAGECACHE_PLAIN_PAGE, + PAGECACHE_LOCK_WRITE, + 0); + end= check_page(buffr, page * PAGE_SIZE, 1, page, num); + put_rec(buffr, end, get_len(record_length_limit), num); + pagecache_write(&pagecache, &file1, page, 3, (char*)buffr, + PAGECACHE_PLAIN_PAGE, + PAGECACHE_LOCK_WRITE_UNLOCK, + PAGECACHE_UNPIN, + PAGECACHE_WRITE_DELAY, + 0); + + if (i % flush_divider == 0) + flush_pagecache_blocks(&pagecache, &file1, FLUSH_FORCE_WRITE); + if (i % 500 == 0) + printf("writer%d: %d\n", num, i); + } + printf("writer%d: done\n", num); + free(buffr); +} + + +static void *test_thread_reader(void *arg) +{ + int param=*((int*) arg); + + my_thread_init(); + DBUG_ENTER("test_reader"); + DBUG_PRINT("enter", ("param: %d", param)); + + reader(param); + + DBUG_PRINT("info", ("Thread %s ended\n", my_thread_name())); + pthread_mutex_lock(&LOCK_thread_count); + thread_count--; + VOID(pthread_cond_signal(&COND_thread_count)); /* Tell main we are ready */ + pthread_mutex_unlock(&LOCK_thread_count); + free((gptr) arg); + my_thread_end(); + DBUG_RETURN(0); +} + +static void *test_thread_writer(void *arg) +{ + int param=*((int*) arg); + + my_thread_init(); + DBUG_ENTER("test_writer"); + DBUG_PRINT("enter", ("param: %d", param)); + + writer(param); + + DBUG_PRINT("info", ("Thread %s ended\n", my_thread_name())); + pthread_mutex_lock(&LOCK_thread_count); + thread_count--; + VOID(pthread_cond_signal(&COND_thread_count)); /* Tell main we are ready */ + pthread_mutex_unlock(&LOCK_thread_count); + free((gptr) arg); + my_thread_end(); + DBUG_RETURN(0); +} + +int main(int argc, char **argv __attribute__((unused))) +{ + pthread_t tid; + pthread_attr_t thr_attr; + int *param, error, pagen; + + MY_INIT(argv[0]); + +#ifndef DBUG_OFF +#if defined(__WIN__) + default_dbug_option= "d:t:i:O,\\test_pagecache_consist.trace"; +#else + default_dbug_option= "d:t:i:o,/tmp/test_pagecache_consist.trace"; +#endif + if (argc > 1) + { + DBUG_SET(default_dbug_option); + DBUG_SET_INITIAL(default_dbug_option); + } +#endif + + + DBUG_ENTER("main"); + DBUG_PRINT("info", ("Main thread: %s\n", my_thread_name())); + if ((file1.file= my_open(file1_name, + O_CREAT | O_TRUNC | O_RDWR, MYF(0))) == -1) + { + fprintf(stderr, "Got error during file1 creation from open() (errno: %d)\n", + errno); + exit(1); + } + DBUG_PRINT("info", ("file1: %d", file1.file)); + if (chmod(file1_name, S_IRWXU | S_IRWXG | S_IRWXO) != 0) + { + fprintf(stderr, "Got error during file1 chmod() (errno: %d)\n", + errno); + exit(1); + } + my_pwrite(file1.file, "test file", 9, 0, MYF(0)); + + if ((error= pthread_cond_init(&COND_thread_count, NULL))) + { + fprintf(stderr, "COND_thread_count: %d from pthread_cond_init (errno: %d)\n", + error, errno); + exit(1); + } + if ((error= pthread_mutex_init(&LOCK_thread_count, MY_MUTEX_INIT_FAST))) + { + fprintf(stderr, "LOCK_thread_count: %d from pthread_cond_init (errno: %d)\n", + error, errno); + exit(1); + } + + if ((error= pthread_attr_init(&thr_attr))) + { + fprintf(stderr,"Got error: %d from pthread_attr_init (errno: %d)\n", + error,errno); + exit(1); + } + if ((error= pthread_attr_setdetachstate(&thr_attr, PTHREAD_CREATE_DETACHED))) + { + fprintf(stderr, + "Got error: %d from pthread_attr_setdetachstate (errno: %d)\n", + error,errno); + exit(1); + } + +#ifndef pthread_attr_setstacksize /* void return value */ + if ((error= pthread_attr_setstacksize(&thr_attr, 65536L))) + { + fprintf(stderr,"Got error: %d from pthread_attr_setstacksize (errno: %d)\n", + error,errno); + exit(1); + } +#endif +#ifdef HAVE_THR_SETCONCURRENCY + VOID(thr_setconcurrency(2)); +#endif + + my_thread_global_init(); + + + if ((pagen= init_pagecache(&pagecache, PCACHE_SIZE, 0, 0, + PAGE_SIZE)) == 0) + { + fprintf(stderr,"Got error: init_pagecache() (errno: %d)\n", + errno); + exit(1); + } + DBUG_PRINT("info", ("Page cache %d pages", pagen)); + { + unsigned char *buffr= malloc(PAGE_SIZE); + uint i; + memset(buffr, '\0', PAGE_SIZE); + for (i= 0; i < number_of_pages; i++) + { + pagecache_write(&pagecache, &file1, i, 3, (char*)buffr, + PAGECACHE_PLAIN_PAGE, + PAGECACHE_LOCK_LEFT_UNLOCKED, + PAGECACHE_PIN_LEFT_UNPINNED, + PAGECACHE_WRITE_DELAY, + 0); + } + flush_pagecache_blocks(&pagecache, &file1, FLUSH_FORCE_WRITE); + free(buffr); + } + if ((error= pthread_mutex_lock(&LOCK_thread_count))) + { + fprintf(stderr,"LOCK_thread_count: %d from pthread_mutex_lock (errno: %d)\n", + error,errno); + exit(1); + } + while (number_of_readers != 0 || number_of_writers != 0) + { + if (number_of_readers != 0) + { + param=(int*) malloc(sizeof(int)); + *param= number_of_readers; + if ((error= pthread_create(&tid, &thr_attr, test_thread_reader, + (void*) param))) + { + fprintf(stderr,"Got error: %d from pthread_create (errno: %d)\n", + error,errno); + exit(1); + } + thread_count++; + number_of_readers--; + } + if (number_of_writers != 0) + { + param=(int*) malloc(sizeof(int)); + *param= number_of_writers; + if ((error= pthread_create(&tid, &thr_attr, test_thread_writer, + (void*) param))) + { + fprintf(stderr,"Got error: %d from pthread_create (errno: %d)\n", + error,errno); + exit(1); + } + thread_count++; + number_of_writers--; + } + } + DBUG_PRINT("info", ("Thread started")); + pthread_mutex_unlock(&LOCK_thread_count); + + pthread_attr_destroy(&thr_attr); + + /* wait finishing */ + if ((error= pthread_mutex_lock(&LOCK_thread_count))) + fprintf(stderr,"LOCK_thread_count: %d from pthread_mutex_lock\n",error); + while (thread_count) + { + if ((error= pthread_cond_wait(&COND_thread_count,&LOCK_thread_count))) + fprintf(stderr,"COND_thread_count: %d from pthread_cond_wait\n",error); + } + if ((error= pthread_mutex_unlock(&LOCK_thread_count))) + fprintf(stderr,"LOCK_thread_count: %d from pthread_mutex_unlock\n",error); + DBUG_PRINT("info", ("thread ended")); + + end_pagecache(&pagecache, 1); + DBUG_PRINT("info", ("Page cache ended")); + + if (my_close(file1.file, MYF(0)) != 0) + { + fprintf(stderr, "Got error during file1 closing from close() (errno: %d)\n", + errno); + exit(1); + } + /*my_delete(file1_name, MYF(0));*/ + my_thread_global_end(); + + DBUG_PRINT("info", ("file1 (%d) closed", file1.file)); + + DBUG_PRINT("info", ("Program end")); + + DBUG_RETURN(exit_status()); +} diff --git a/storage/maria/unittest/mf_pagecache_single.c b/storage/maria/unittest/mf_pagecache_single.c new file mode 100644 index 00000000000..3c4a3642fe9 --- /dev/null +++ b/storage/maria/unittest/mf_pagecache_single.c @@ -0,0 +1,589 @@ +/* + TODO: use pthread_join instead of wait_for_thread_count_to_be_zero, like in + my_atomic-t.c (see BUG#22320). + Use diag() instead of fprintf(stderr). +*/ +#include +#include +#include +#include "test_file.h" + +#define PCACHE_SIZE (PAGE_SIZE*1024*10) + +#ifndef DBUG_OFF +static const char* default_dbug_option; +#endif + +static char *file1_name= (char*)"page_cache_test_file_1"; +static PAGECACHE_FILE file1; +static pthread_cond_t COND_thread_count; +static pthread_mutex_t LOCK_thread_count; +static uint thread_count; +static PAGECACHE pagecache; + +/* + File contance descriptors +*/ +static struct file_desc simple_read_write_test_file[]= +{ + {PAGE_SIZE, '\1'}, + { 0, 0} +}; +static struct file_desc simple_read_change_write_read_test_file[]= +{ + {PAGE_SIZE/2, '\65'}, + {PAGE_SIZE/2, '\1'}, + { 0, 0} +}; +static struct file_desc simple_pin_test_file1[]= +{ + {PAGE_SIZE*2, '\1'}, + { 0, 0} +}; +static struct file_desc simple_pin_test_file2[]= +{ + {PAGE_SIZE/2, '\1'}, + {PAGE_SIZE/2, (unsigned char)129}, + {PAGE_SIZE, '\1'}, + { 0, 0} +}; +static struct file_desc simple_delete_forget_test_file[]= +{ + {PAGE_SIZE, '\1'}, + { 0, 0} +}; +static struct file_desc simple_delete_flush_test_file[]= +{ + {PAGE_SIZE, '\2'}, + { 0, 0} +}; + + +/* + Recreate and reopen a file for test + + SYNOPSIS + reset_file() + file File to reset + file_name Path (and name) of file which should be reset +*/ + +void reset_file(PAGECACHE_FILE file, char *file_name) +{ + flush_pagecache_blocks(&pagecache, &file1, FLUSH_RELEASE); + if (my_close(file1.file, MYF(0)) != 0) + { + diag("Got error during %s closing from close() (errno: %d)\n", + file_name, errno); + exit(1); + } + my_delete(file_name, MYF(0)); + if ((file.file= my_open(file_name, + O_CREAT | O_TRUNC | O_RDWR, MYF(0))) == -1) + { + diag("Got error during %s creation from open() (errno: %d)\n", + file_name, errno); + exit(1); + } +} + +/* + Write then read page, check file on disk +*/ + +int simple_read_write_test() +{ + unsigned char *buffw= malloc(PAGE_SIZE); + unsigned char *buffr= malloc(PAGE_SIZE); + int res; + DBUG_ENTER("simple_read_write_test"); + bfill(buffw, PAGE_SIZE, '\1'); + pagecache_write(&pagecache, &file1, 0, 3, (char*)buffw, + PAGECACHE_PLAIN_PAGE, + PAGECACHE_LOCK_LEFT_UNLOCKED, + PAGECACHE_PIN_LEFT_UNPINNED, + PAGECACHE_WRITE_DELAY, + 0); + pagecache_read(&pagecache, &file1, 0, 3, (char*)buffr, + PAGECACHE_PLAIN_PAGE, + PAGECACHE_LOCK_LEFT_UNLOCKED, + 0); + ok((res= test(memcmp(buffr, buffw, PAGE_SIZE) == 0)), + "Simple write-read page "); + flush_pagecache_blocks(&pagecache, &file1, FLUSH_FORCE_WRITE); + ok((res&= test(test_file(file1, file1_name, PAGE_SIZE, PAGE_SIZE, + simple_read_write_test_file))), + "Simple write-read page file"); + if (res) + reset_file(file1, file1_name); + free(buffw); + free(buffr); + DBUG_RETURN(res); +} + + +/* + Prepare page, then read (and lock), change (write new value and unlock), + then check the page in the cache and on the disk +*/ +int simple_read_change_write_read_test() +{ + unsigned char *buffw= malloc(PAGE_SIZE); + unsigned char *buffr= malloc(PAGE_SIZE); + int res; + DBUG_ENTER("simple_read_change_write_read_test"); + /* prepare the file */ + bfill(buffw, PAGE_SIZE, '\1'); + pagecache_write(&pagecache, &file1, 0, 3, (char*)buffw, + PAGECACHE_PLAIN_PAGE, + PAGECACHE_LOCK_LEFT_UNLOCKED, + PAGECACHE_PIN_LEFT_UNPINNED, + PAGECACHE_WRITE_DELAY, + 0); + flush_pagecache_blocks(&pagecache, &file1, FLUSH_FORCE_WRITE); + /* test */ + pagecache_read(&pagecache, &file1, 0, 3, (char*)buffw, + PAGECACHE_PLAIN_PAGE, + PAGECACHE_LOCK_WRITE, + 0); + bfill(buffw, PAGE_SIZE/2, '\65'); + pagecache_write(&pagecache, &file1, 0, 3, (char*)buffw, + PAGECACHE_PLAIN_PAGE, + PAGECACHE_LOCK_WRITE_UNLOCK, + PAGECACHE_UNPIN, + PAGECACHE_WRITE_DELAY, + 0); + + pagecache_read(&pagecache, &file1, 0, 3, (char*)buffr, + PAGECACHE_PLAIN_PAGE, + PAGECACHE_LOCK_LEFT_UNLOCKED, + 0); + ok((res= test(memcmp(buffr, buffw, PAGE_SIZE) == 0)), + "Simple read-change-write-read page "); + flush_pagecache_blocks(&pagecache, &file1, FLUSH_FORCE_WRITE); + ok((res&= test(test_file(file1, file1_name, PAGE_SIZE, PAGE_SIZE, + simple_read_change_write_read_test_file))), + "Simple read-change-write-read page file"); + if (res) + reset_file(file1, file1_name); + free(buffw); + free(buffr); + DBUG_RETURN(res); +} + + +/* + Prepare page, read page 0 (and pin) then write page 1 and page 0. + Flush the file (shold flush only page 1 and return 1 (page 0 is + still pinned). + Check file on the disk. + Unpin and flush. + Check file on the disk. +*/ +int simple_pin_test() +{ + unsigned char *buffw= malloc(PAGE_SIZE); + unsigned char *buffr= malloc(PAGE_SIZE); + int res; + DBUG_ENTER("simple_pin_test"); + /* prepare the file */ + bfill(buffw, PAGE_SIZE, '\1'); + pagecache_write(&pagecache, &file1, 0, 3, (char*)buffw, + PAGECACHE_PLAIN_PAGE, + PAGECACHE_LOCK_LEFT_UNLOCKED, + PAGECACHE_PIN_LEFT_UNPINNED, + PAGECACHE_WRITE_DELAY, + 0); + /* test */ + if (flush_pagecache_blocks(&pagecache, &file1, FLUSH_FORCE_WRITE)) + { + diag("error in flush_pagecache_blocks\n"); + exit(1); + } + pagecache_read(&pagecache, &file1, 0, 3, (char*)buffw, + PAGECACHE_PLAIN_PAGE, + PAGECACHE_LOCK_WRITE, + 0); + pagecache_write(&pagecache, &file1, 1, 3, (char*)buffw, + PAGECACHE_PLAIN_PAGE, + PAGECACHE_LOCK_LEFT_UNLOCKED, + PAGECACHE_PIN_LEFT_UNPINNED, + PAGECACHE_WRITE_DELAY, + 0); + bfill(buffw + PAGE_SIZE/2, PAGE_SIZE/2, ((unsigned char) 129)); + pagecache_write(&pagecache, &file1, 0, 3, (char*)buffw, + PAGECACHE_PLAIN_PAGE, + PAGECACHE_LOCK_WRITE_TO_READ, + PAGECACHE_PIN_LEFT_PINNED, + PAGECACHE_WRITE_DELAY, + 0); + /* + We have to get error because one page of the file is pinned, + other page should be flushed + */ + if (!flush_pagecache_blocks(&pagecache, &file1, FLUSH_FORCE_WRITE)) + { + diag("Did not get error in flush_pagecache_blocks\n"); + res= 0; + goto err; + } + ok((res= test(test_file(file1, file1_name, PAGE_SIZE*2, PAGE_SIZE*2, + simple_pin_test_file1))), + "Simple pin page file with pin"); + pagecache_unlock_page(&pagecache, + &file1, + 0, + PAGECACHE_LOCK_READ_UNLOCK, + PAGECACHE_UNPIN, + 0, 0); + if (flush_pagecache_blocks(&pagecache, &file1, FLUSH_FORCE_WRITE)) + { + diag("Got error in flush_pagecache_blocks\n"); + res= 0; + goto err; + } + ok((res&= test(test_file(file1, file1_name, PAGE_SIZE*2, PAGE_SIZE, + simple_pin_test_file2))), + "Simple pin page result file"); + if (res) + reset_file(file1, file1_name); +err: + free(buffw); + free(buffr); + DBUG_RETURN(res); +} + +/* + Prepare page, write new value, then delete page from cache without flush, + on the disk should be page with old content written during preparation +*/ + +int simple_delete_forget_test() +{ + unsigned char *buffw= malloc(PAGE_SIZE); + unsigned char *buffr= malloc(PAGE_SIZE); + int res; + DBUG_ENTER("simple_delete_forget_test"); + /* prepare the file */ + bfill(buffw, PAGE_SIZE, '\1'); + pagecache_write(&pagecache, &file1, 0, 3, (char*)buffw, + PAGECACHE_PLAIN_PAGE, + PAGECACHE_LOCK_LEFT_UNLOCKED, + PAGECACHE_PIN_LEFT_UNPINNED, + PAGECACHE_WRITE_DELAY, + 0); + flush_pagecache_blocks(&pagecache, &file1, FLUSH_FORCE_WRITE); + /* test */ + bfill(buffw, PAGE_SIZE, '\2'); + pagecache_write(&pagecache, &file1, 0, 3, (char*)buffw, + PAGECACHE_PLAIN_PAGE, + PAGECACHE_LOCK_LEFT_UNLOCKED, + PAGECACHE_PIN_LEFT_UNPINNED, + PAGECACHE_WRITE_DELAY, + 0); + pagecache_delete_page(&pagecache, &file1, 0, + PAGECACHE_LOCK_WRITE, 0); + flush_pagecache_blocks(&pagecache, &file1, FLUSH_FORCE_WRITE); + ok((res= test(test_file(file1, file1_name, PAGE_SIZE, PAGE_SIZE, + simple_delete_forget_test_file))), + "Simple delete-forget page file"); + if (res) + reset_file(file1, file1_name); + free(buffw); + free(buffr); + DBUG_RETURN(res); +} + +/* + Prepare page with locking, write new content to the page, + delete page with flush and on existing lock, + check that page on disk contain new value. +*/ + +int simple_delete_flush_test() +{ + unsigned char *buffw= malloc(PAGE_SIZE); + unsigned char *buffr= malloc(PAGE_SIZE); + int res; + DBUG_ENTER("simple_delete_flush_test"); + /* prepare the file */ + bfill(buffw, PAGE_SIZE, '\1'); + pagecache_write(&pagecache, &file1, 0, 3, (char*)buffw, + PAGECACHE_PLAIN_PAGE, + PAGECACHE_LOCK_WRITE, + PAGECACHE_PIN, + PAGECACHE_WRITE_DELAY, + 0); + flush_pagecache_blocks(&pagecache, &file1, FLUSH_FORCE_WRITE); + /* test */ + bfill(buffw, PAGE_SIZE, '\2'); + pagecache_write(&pagecache, &file1, 0, 3, (char*)buffw, + PAGECACHE_PLAIN_PAGE, + PAGECACHE_LOCK_LEFT_WRITELOCKED, + PAGECACHE_PIN_LEFT_PINNED, + PAGECACHE_WRITE_DELAY, + 0); + pagecache_delete_page(&pagecache, &file1, 0, + PAGECACHE_LOCK_LEFT_WRITELOCKED, 1); + flush_pagecache_blocks(&pagecache, &file1, FLUSH_FORCE_WRITE); + ok((res= test(test_file(file1, file1_name, PAGE_SIZE, PAGE_SIZE, + simple_delete_flush_test_file))), + "Simple delete-forget page file"); + if (res) + reset_file(file1, file1_name); + free(buffw); + free(buffr); + DBUG_RETURN(res); +} + + +/* + write then read file bigger then cache +*/ + +int simple_big_test() +{ + unsigned char *buffw= (unsigned char *)malloc(PAGE_SIZE); + unsigned char *buffr= (unsigned char *)malloc(PAGE_SIZE); + struct file_desc *desc= + (struct file_desc *)malloc((PCACHE_SIZE/(PAGE_SIZE/2) + 1) * + sizeof(struct file_desc)); + int res, i; + DBUG_ENTER("simple_big_test"); + /* prepare the file twice larger then cache */ + for (i= 0; i < PCACHE_SIZE/(PAGE_SIZE/2); i++) + { + bfill(buffw, PAGE_SIZE, (unsigned char) (i & 0xff)); + desc[i].length= PAGE_SIZE; + desc[i].content= (i & 0xff); + pagecache_write(&pagecache, &file1, i, 3, (char*)buffw, + PAGECACHE_PLAIN_PAGE, + PAGECACHE_LOCK_LEFT_UNLOCKED, + PAGECACHE_PIN_LEFT_UNPINNED, + PAGECACHE_WRITE_DELAY, + 0); + } + desc[i].length= 0; + desc[i].content= NULL; + ok(1, "Simple big file write"); + /* check written pages sequentally read */ + for (i= 0; i < PCACHE_SIZE/(PAGE_SIZE/2); i++) + { + int j; + pagecache_read(&pagecache, &file1, i, 3, (char*)buffr, + PAGECACHE_PLAIN_PAGE, + PAGECACHE_LOCK_LEFT_UNLOCKED, + 0); + for(j= 0; j < PAGE_SIZE; j++) + { + if (buffr[j] != (i & 0xff)) + { + diag("simple_big_test seq: page %u byte %u mismatch\n", i, j); + return 0; + } + } + } + ok(1, "simple big file sequentally read"); + /* chack random reads */ + for (i= 0; i < PCACHE_SIZE/(PAGE_SIZE); i++) + { + int j, page; + page= rand() % (PCACHE_SIZE/(PAGE_SIZE/2)); + pagecache_read(&pagecache, &file1, page, 3, (char*)buffr, + PAGECACHE_PLAIN_PAGE, + PAGECACHE_LOCK_LEFT_UNLOCKED, + 0); + for(j= 0; j < PAGE_SIZE; j++) + { + if (buffr[j] != (page & 0xff)) + { + diag("simple_big_test rnd: page %u byte %u mismatch\n", page, j); + return 0; + } + } + } + ok(1, "simple big file random read"); + flush_pagecache_blocks(&pagecache, &file1, FLUSH_FORCE_WRITE); + + ok((res= test(test_file(file1, file1_name, PCACHE_SIZE*2, PAGE_SIZE, + desc))), + "Simple big file"); + if (res) + reset_file(file1, file1_name); + free(buffw); + free(buffr); + DBUG_RETURN(res); +} +/* + Thread function +*/ + +static void *test_thread(void *arg) +{ + int param=*((int*) arg); + + my_thread_init(); + DBUG_ENTER("test_thread"); + + DBUG_PRINT("enter", ("param: %d", param)); + + if (!simple_read_write_test() || + !simple_read_change_write_read_test() || + !simple_pin_test() || + !simple_delete_forget_test() || + !simple_delete_flush_test() || + !simple_big_test()) + exit(1); + + DBUG_PRINT("info", ("Thread %s ended\n", my_thread_name())); + pthread_mutex_lock(&LOCK_thread_count); + thread_count--; + VOID(pthread_cond_signal(&COND_thread_count)); /* Tell main we are ready */ + pthread_mutex_unlock(&LOCK_thread_count); + free((gptr) arg); + my_thread_end(); + DBUG_RETURN(0); +} + + +int main(int argc, char **argv __attribute__((unused))) +{ + pthread_t tid; + pthread_attr_t thr_attr; + int *param, error, pagen; + + MY_INIT(argv[0]); + +#ifndef DBUG_OFF +#if defined(__WIN__) + default_dbug_option= "d:t:i:O,\\test_pagecache_single.trace"; +#else + default_dbug_option= "d:t:i:o,/tmp/test_pagecache_single.trace"; +#endif + if (argc > 1) + { + DBUG_SET(default_dbug_option); + DBUG_SET_INITIAL(default_dbug_option); + } +#endif + + + DBUG_ENTER("main"); + DBUG_PRINT("info", ("Main thread: %s\n", my_thread_name())); + if ((file1.file= my_open(file1_name, + O_CREAT | O_TRUNC | O_RDWR, MYF(0))) == -1) + { + fprintf(stderr, "Got error during file1 creation from open() (errno: %d)\n", + errno); + exit(1); + } + DBUG_PRINT("info", ("file1: %d", file1.file)); + if (chmod(file1_name, S_IRWXU | S_IRWXG | S_IRWXO) != 0) + { + fprintf(stderr, "Got error during file1 chmod() (errno: %d)\n", + errno); + exit(1); + } + my_pwrite(file1.file, "test file", 9, 0, MYF(0)); + + if ((error= pthread_cond_init(&COND_thread_count, NULL))) + { + fprintf(stderr, "Got error: %d from pthread_cond_init (errno: %d)\n", + error, errno); + exit(1); + } + if ((error= pthread_mutex_init(&LOCK_thread_count, MY_MUTEX_INIT_FAST))) + { + fprintf(stderr, "Got error: %d from pthread_cond_init (errno: %d)\n", + error, errno); + exit(1); + } + + if ((error= pthread_attr_init(&thr_attr))) + { + fprintf(stderr,"Got error: %d from pthread_attr_init (errno: %d)\n", + error,errno); + exit(1); + } + if ((error= pthread_attr_setdetachstate(&thr_attr, PTHREAD_CREATE_DETACHED))) + { + fprintf(stderr, + "Got error: %d from pthread_attr_setdetachstate (errno: %d)\n", + error,errno); + exit(1); + } + +#ifndef pthread_attr_setstacksize /* void return value */ + if ((error= pthread_attr_setstacksize(&thr_attr, 65536L))) + { + fprintf(stderr,"Got error: %d from pthread_attr_setstacksize (errno: %d)\n", + error,errno); + exit(1); + } +#endif +#ifdef HAVE_THR_SETCONCURRENCY + VOID(thr_setconcurrency(2)); +#endif + + my_thread_global_init(); + + plan(12); + + if ((pagen= init_pagecache(&pagecache, PCACHE_SIZE, 0, 0, + PAGE_SIZE)) == 0) + { + fprintf(stderr,"Got error: init_pagecache() (errno: %d)\n", + errno); + exit(1); + } + DBUG_PRINT("info", ("Page cache %d pages", pagen)); + + if ((error=pthread_mutex_lock(&LOCK_thread_count))) + { + fprintf(stderr,"Got error: %d from pthread_mutex_lock (errno: %d)\n", + error,errno); + exit(1); + } + param=(int*) malloc(sizeof(int)); + *param= 1; + if ((error= pthread_create(&tid, &thr_attr, test_thread, (void*) param))) + { + fprintf(stderr,"Got error: %d from pthread_create (errno: %d)\n", + error,errno); + exit(1); + } + thread_count++; + DBUG_PRINT("info", ("Thread started")); + pthread_mutex_unlock(&LOCK_thread_count); + + pthread_attr_destroy(&thr_attr); + + if ((error= pthread_mutex_lock(&LOCK_thread_count))) + fprintf(stderr,"Got error: %d from pthread_mutex_lock\n",error); + while (thread_count) + { + if ((error= pthread_cond_wait(&COND_thread_count,&LOCK_thread_count))) + fprintf(stderr,"Got error: %d from pthread_cond_wait\n",error); + } + if ((error= pthread_mutex_unlock(&LOCK_thread_count))) + fprintf(stderr,"Got error: %d from pthread_mutex_unlock\n",error); + DBUG_PRINT("info", ("thread ended")); + + end_pagecache(&pagecache, 1); + DBUG_PRINT("info", ("Page cache ended")); + + if (my_close(file1.file, MYF(0)) != 0) + { + fprintf(stderr, "Got error during file1 closing from close() (errno: %d)\n", + errno); + exit(1); + } + /*my_delete(file1_name, MYF(0));*/ + my_thread_global_end(); + + DBUG_PRINT("info", ("file1 (%d) closed", file1.file)); + + DBUG_PRINT("info", ("Program end")); + + DBUG_RETURN(exit_status()); +} diff --git a/storage/maria/unittest/test_file.c b/storage/maria/unittest/test_file.c new file mode 100644 index 00000000000..758d0bfa81b --- /dev/null +++ b/storage/maria/unittest/test_file.c @@ -0,0 +1,68 @@ +#include +#include +#include +#include "test_file.h" + + +/* + Check that file contance correspond to descriptor + + SYNOPSIS + test_file() + file File to test + file_name Path (and name) of file which is tested + size size of file + buff_size size of buffer which is enought to check the file + desc file descriptor to check with + + RETURN + 1 file if OK + 0 error +*/ + +int test_file(PAGECACHE_FILE file, char *file_name, + off_t size, size_t buff_size, struct file_desc *desc) +{ + MY_STAT stat_buff, *stat; + unsigned char *buffr= malloc(buff_size); + off_t pos= 0; + size_t byte; + int step= 0; + + if ((stat= my_stat(file_name, &stat_buff, MYF(0))) == NULL) + { + diag("Can't stat() %s (errno: %d)\n", file_name, errno); + return 0; + } + if (stat->st_size != size) + { + diag("file %s size is %lu (should be %lu)\n", + file_name, (ulong) stat->st_size, (ulong) size); + return 0; + } + /* check content */ + my_seek(file.file, 0, SEEK_SET, MYF(0)); + while (desc[step].length != 0) + { + if (my_read(file.file, (char*)buffr, desc[step].length, MYF(0)) != + desc[step].length) + { + diag("Can't read %u bytes from %s (errno: %d)\n", + (uint)desc[step].length, file_name, errno); + return 0; + } + for (byte= 0; byte < desc[step].length; byte++) + { + if (buffr[byte] != desc[step].content) + { + diag("content of %s mismatch 0x%x in position %lu instead of 0x%x\n", + file_name, (uint) buffr[byte], (ulong) (pos + byte), + desc[step].content); + return 0; + } + } + pos+= desc[step].length; + step++; + } + return 1; +} diff --git a/storage/maria/unittest/test_file.h b/storage/maria/unittest/test_file.h new file mode 100644 index 00000000000..ea787c123ed --- /dev/null +++ b/storage/maria/unittest/test_file.h @@ -0,0 +1,14 @@ + +#include + +/* + File content descriptor +*/ +struct file_desc +{ + unsigned int length; + unsigned char content; +}; + +int test_file(PAGECACHE_FILE file, char *file_name, + off_t size, size_t buff_size, struct file_desc *desc); diff --git a/unittest/mysys/Makefile.am b/unittest/mysys/Makefile.am index 9b230272329..a32dbb9b0c3 100644 --- a/unittest/mysys/Makefile.am +++ b/unittest/mysys/Makefile.am @@ -7,45 +7,4 @@ LDADD = $(top_builddir)/unittest/mytap/libmytap.a \ $(top_builddir)/dbug/libdbug.a \ $(top_builddir)/strings/libmystrings.a -noinst_PROGRAMS = bitmap-t base64-t my_atomic-t \ - mf_pagecache_single_1k-t mf_pagecache_single_8k-t \ - mf_pagecache_single_64k-t \ - mf_pagecache_consist_1k-t mf_pagecache_consist_64k-t \ - mf_pagecache_consist_1kHC-t mf_pagecache_consist_64kHC-t \ - mf_pagecache_consist_1kRD-t mf_pagecache_consist_64kRD-t \ - mf_pagecache_consist_1kWR-t mf_pagecache_consist_64kWR-t - -# tests for mysys/mf_pagecache.c - -mf_pagecache_single_src = mf_pagecache_single.c $(top_srcdir)/mysys/mf_pagecache.c test_file.c -mf_pagecache_consist_src = mf_pagecache_consist.c $(top_srcdir)/mysys/mf_pagecache.c test_file.c -mf_pagecache_common_cppflags = -DEXTRA_DEBUG -DPAGECACHE_DEBUG -DMAIN - -mf_pagecache_single_1k_t_SOURCES = $(mf_pagecache_single_src) -mf_pagecache_single_8k_t_SOURCES = $(mf_pagecache_single_src) -mf_pagecache_single_64k_t_SOURCES = $(mf_pagecache_single_src) -mf_pagecache_single_1k_t_CPPFLAGS = $(mf_pagecache_common_cppflags) -DPAGE_SIZE=1024 -mf_pagecache_single_8k_t_CPPFLAGS = $(mf_pagecache_common_cppflags) -DPAGE_SIZE=8192 -mf_pagecache_single_64k_t_CPPFLAGS = $(mf_pagecache_common_cppflags) -DPAGE_SIZE=65536 - -mf_pagecache_consist_1k_t_SOURCES = $(mf_pagecache_consist_src) -mf_pagecache_consist_1k_t_CPPFLAGS = $(mf_pagecache_common_cppflags) -DPAGE_SIZE=1024 -mf_pagecache_consist_64k_t_SOURCES = $(mf_pagecache_consist_src) -mf_pagecache_consist_64k_t_CPPFLAGS = $(mf_pagecache_common_cppflags) -DPAGE_SIZE=65536 - -mf_pagecache_consist_1kHC_t_SOURCES = $(mf_pagecache_consist_src) -mf_pagecache_consist_1kHC_t_CPPFLAGS = $(mf_pagecache_common_cppflags) -DPAGE_SIZE=1024 -DTEST_HIGH_CONCURENCY -mf_pagecache_consist_64kHC_t_SOURCES = $(mf_pagecache_consist_src) -mf_pagecache_consist_64kHC_t_CPPFLAGS = $(mf_pagecache_common_cppflags) -DPAGE_SIZE=65536 -DTEST_HIGH_CONCURENCY - -mf_pagecache_consist_1kRD_t_SOURCES = $(mf_pagecache_consist_src) -mf_pagecache_consist_1kRD_t_CPPFLAGS = $(mf_pagecache_common_cppflags) -DPAGE_SIZE=1024 -DTEST_READERS -mf_pagecache_consist_64kRD_t_SOURCES = $(mf_pagecache_consist_src) -mf_pagecache_consist_64kRD_t_CPPFLAGS = $(mf_pagecache_common_cppflags) -DPAGE_SIZE=65536 -DTEST_READERS - -mf_pagecache_consist_1kWR_t_SOURCES = $(mf_pagecache_consist_src) -mf_pagecache_consist_1kWR_t_CPPFLAGS = $(mf_pagecache_common_cppflags) -DPAGE_SIZE=1024 -DTEST_WRITERS -mf_pagecache_consist_64kWR_t_SOURCES = $(mf_pagecache_consist_src) -mf_pagecache_consist_64kWR_t_CPPFLAGS = $(mf_pagecache_common_cppflags) -DPAGE_SIZE=65536 -DTEST_WRITERS - -CLEANFILES = my_pagecache_debug.log page_cache_test_file_1 +noinst_PROGRAMS = bitmap-t base64-t my_atomic-t diff --git a/unittest/mysys/mf_pagecache_consist.c b/unittest/mysys/mf_pagecache_consist.c deleted file mode 100755 index f7724222a09..00000000000 --- a/unittest/mysys/mf_pagecache_consist.c +++ /dev/null @@ -1,445 +0,0 @@ -/* - TODO: use pthread_join instead of wait_for_thread_count_to_be_zero, like in - my_atomic-t.c (see BUG#22320). - Use diag() instead of fprintf(stderr). Use ok() and plan(). -*/ - -#include -#include -#include -#include "test_file.h" - -#define PCACHE_SIZE (PAGE_SIZE*1024*8) - -#ifndef DBUG_OFF -static const char* default_dbug_option; -#endif - -static char *file1_name= (char*)"page_cache_test_file_1"; -static PAGECACHE_FILE file1; -static pthread_cond_t COND_thread_count; -static pthread_mutex_t LOCK_thread_count; -static uint thread_count; -static PAGECACHE pagecache; - -#ifdef TEST_HIGH_CONCURENCY -static uint number_of_readers= 10; -static uint number_of_writers= 20; -static uint number_of_tests= 30000; -static uint record_length_limit= PAGE_SIZE/200; -static uint number_of_pages= 20; -static uint flush_divider= 1000; -#else /*TEST_HIGH_CONCURENCY*/ -#ifdef TEST_READERS -static uint number_of_readers= 10; -static uint number_of_writers= 1; -static uint number_of_tests= 30000; -static uint record_length_limit= PAGE_SIZE/200; -static uint number_of_pages= 20; -static uint flush_divider= 1000; -#else /*TEST_READERS*/ -#ifdef TEST_WRITERS -static uint number_of_readers= 0; -static uint number_of_writers= 10; -static uint number_of_tests= 30000; -static uint record_length_limit= PAGE_SIZE/200; -static uint number_of_pages= 20; -static uint flush_divider= 1000; -#else /*TEST_WRITERS*/ -static uint number_of_readers= 10; -static uint number_of_writers= 10; -static uint number_of_tests= 50000; -static uint record_length_limit= PAGE_SIZE/200; -static uint number_of_pages= 20000; -static uint flush_divider= 1000; -#endif /*TEST_WRITERS*/ -#endif /*TEST_READERS*/ -#endif /*TEST_HIGH_CONCURENCY*/ - - -/* check page consistency */ -uint check_page(uchar *buff, ulong offset, int page_locked, int page_no, - int tag) -{ - uint end= sizeof(uint); - uint num= *((uint *)buff); - uint i; - DBUG_ENTER("check_page"); - - for (i= 0; i < num; i++) - { - uint len= *((uint *)(buff + end)); - uint j; - end+= sizeof(uint)+ sizeof(uint); - if (len + end > PAGE_SIZE) - { - diag("incorrect field header #%u by offset %lu\n", i, offset + end + j); - goto err; - } - for(j= 0; j < len; j++) - { - if (buff[end + j] != (uchar)((i+1) % 256)) - { - diag("incorrect %lu byte\n", offset + end + j); - goto err; - } - } - end+= len; - } - for(i= end; i < PAGE_SIZE; i++) - { - if (buff[i] != 0) - { - int h; - DBUG_PRINT("err", - ("byte %lu (%lu + %u), page %u (%s, end: %u, recs: %u, tag: %d) should be 0\n", - offset + i, offset, i, page_no, - (page_locked ? "locked" : "unlocked"), - end, num, tag)); - diag("byte %lu (%lu + %u), page %u (%s, end: %u, recs: %u, tag: %d) should be 0\n", - offset + i, offset, i, page_no, - (page_locked ? "locked" : "unlocked"), - end, num, tag); - h= my_open("wrong_page", O_CREAT | O_TRUNC | O_RDWR, MYF(0)); - my_pwrite(h, buff, PAGE_SIZE, 0, MYF(0)); - my_close(h, MYF(0)); - goto err; - } - } - DBUG_RETURN(end); -err: - DBUG_PRINT("err", ("try to flush")); - if (page_locked) - { - pagecache_delete_page(&pagecache, &file1, page_no, - PAGECACHE_LOCK_LEFT_WRITELOCKED, 1); - } - else - { - flush_pagecache_blocks(&pagecache, &file1, FLUSH_RELEASE); - } - exit(1); -} - -void put_rec(uchar *buff, uint end, uint len, uint tag) -{ - uint i; - uint num= *((uint *)buff); - if (!len) - len= 1; - if (end + sizeof(uint)*2 + len > PAGE_SIZE) - return; - *((uint *)(buff + end))= len; - end+= sizeof(uint); - *((uint *)(buff + end))= tag; - end+= sizeof(uint); - num++; - *((uint *)buff)= num; - *((uint*)(buff + end))= len; - for (i= end; i < (len + end); i++) - { - buff[i]= (uchar) num % 256; - } -} - -/* - Recreate and reopen a file for test - - SYNOPSIS - reset_file() - file File to reset - file_name Path (and name) of file which should be reset -*/ - -void reset_file(PAGECACHE_FILE file, char *file_name) -{ - flush_pagecache_blocks(&pagecache, &file1, FLUSH_RELEASE); - if (my_close(file1.file, MYF(0)) != 0) - { - diag("Got error during %s closing from close() (errno: %d)\n", - file_name, errno); - exit(1); - } - my_delete(file_name, MYF(0)); - if ((file.file= my_open(file_name, - O_CREAT | O_TRUNC | O_RDWR, MYF(0))) == -1) - { - diag("Got error during %s creation from open() (errno: %d)\n", - file_name, errno); - exit(1); - } -} - - -void reader(int num) -{ - unsigned char *buffr= malloc(PAGE_SIZE); - uint i; - - for (i= 0; i < number_of_tests; i++) - { - uint page= rand()/(RAND_MAX/number_of_pages); - pagecache_read(&pagecache, &file1, page, 3, (char*)buffr, - PAGECACHE_PLAIN_PAGE, - PAGECACHE_LOCK_LEFT_UNLOCKED, - 0); - check_page(buffr, page * PAGE_SIZE, 0, page, -num); - if (i % 500 == 0) - printf("reader%d: %d\n", num, i); - - } - printf("reader%d: done\n", num); - free(buffr); -} - - -void writer(int num) -{ - unsigned char *buffr= malloc(PAGE_SIZE); - uint i; - - for (i= 0; i < number_of_tests; i++) - { - uint end; - uint page= rand()/(RAND_MAX/number_of_pages); - pagecache_read(&pagecache, &file1, page, 3, (char*)buffr, - PAGECACHE_PLAIN_PAGE, - PAGECACHE_LOCK_WRITE, - 0); - end= check_page(buffr, page * PAGE_SIZE, 1, page, num); - put_rec(buffr, end, rand()/(RAND_MAX/record_length_limit), num); - pagecache_write(&pagecache, &file1, page, 3, (char*)buffr, - PAGECACHE_PLAIN_PAGE, - PAGECACHE_LOCK_WRITE_UNLOCK, - PAGECACHE_UNPIN, - PAGECACHE_WRITE_DELAY, - 0); - - if (i % flush_divider == 0) - flush_pagecache_blocks(&pagecache, &file1, FLUSH_FORCE_WRITE); - if (i % 500 == 0) - printf("writer%d: %d\n", num, i); - } - printf("writer%d: done\n", num); - free(buffr); -} - - -static void *test_thread_reader(void *arg) -{ - int param=*((int*) arg); - - my_thread_init(); - DBUG_ENTER("test_reader"); - DBUG_PRINT("enter", ("param: %d", param)); - - reader(param); - - DBUG_PRINT("info", ("Thread %s ended\n", my_thread_name())); - pthread_mutex_lock(&LOCK_thread_count); - thread_count--; - VOID(pthread_cond_signal(&COND_thread_count)); /* Tell main we are ready */ - pthread_mutex_unlock(&LOCK_thread_count); - free((gptr) arg); - my_thread_end(); - DBUG_RETURN(0); -} - -static void *test_thread_writer(void *arg) -{ - int param=*((int*) arg); - - my_thread_init(); - DBUG_ENTER("test_writer"); - DBUG_PRINT("enter", ("param: %d", param)); - - writer(param); - - DBUG_PRINT("info", ("Thread %s ended\n", my_thread_name())); - pthread_mutex_lock(&LOCK_thread_count); - thread_count--; - VOID(pthread_cond_signal(&COND_thread_count)); /* Tell main we are ready */ - pthread_mutex_unlock(&LOCK_thread_count); - free((gptr) arg); - my_thread_end(); - DBUG_RETURN(0); -} - -int main(int argc, char **argv __attribute__((unused))) -{ - pthread_t tid; - pthread_attr_t thr_attr; - int *param, error, pagen; - - MY_INIT(argv[0]); - -#ifndef DBUG_OFF -#if defined(__WIN__) - default_dbug_option= "d:t:i:O,\\test_pagecache_consist.trace"; -#else - default_dbug_option= "d:t:i:o,/tmp/test_pagecache_consist.trace"; -#endif - if (argc > 1) - { - DBUG_SET(default_dbug_option); - DBUG_SET_INITIAL(default_dbug_option); - } -#endif - - - DBUG_ENTER("main"); - DBUG_PRINT("info", ("Main thread: %s\n", my_thread_name())); - if ((file1.file= my_open(file1_name, - O_CREAT | O_TRUNC | O_RDWR, MYF(0))) == -1) - { - fprintf(stderr, "Got error during file1 creation from open() (errno: %d)\n", - errno); - exit(1); - } - DBUG_PRINT("info", ("file1: %d", file1.file)); - if (chmod(file1_name, S_IRWXU | S_IRWXG | S_IRWXO) != 0) - { - fprintf(stderr, "Got error during file1 chmod() (errno: %d)\n", - errno); - exit(1); - } - my_pwrite(file1.file, "test file", 9, 0, MYF(0)); - - if ((error= pthread_cond_init(&COND_thread_count, NULL))) - { - fprintf(stderr, "COND_thread_count: %d from pthread_cond_init (errno: %d)\n", - error, errno); - exit(1); - } - if ((error= pthread_mutex_init(&LOCK_thread_count, MY_MUTEX_INIT_FAST))) - { - fprintf(stderr, "LOCK_thread_count: %d from pthread_cond_init (errno: %d)\n", - error, errno); - exit(1); - } - - if ((error= pthread_attr_init(&thr_attr))) - { - fprintf(stderr,"Got error: %d from pthread_attr_init (errno: %d)\n", - error,errno); - exit(1); - } - if ((error= pthread_attr_setdetachstate(&thr_attr, PTHREAD_CREATE_DETACHED))) - { - fprintf(stderr, - "Got error: %d from pthread_attr_setdetachstate (errno: %d)\n", - error,errno); - exit(1); - } - -#ifndef pthread_attr_setstacksize /* void return value */ - if ((error= pthread_attr_setstacksize(&thr_attr, 65536L))) - { - fprintf(stderr,"Got error: %d from pthread_attr_setstacksize (errno: %d)\n", - error,errno); - exit(1); - } -#endif -#ifdef HAVE_THR_SETCONCURRENCY - VOID(thr_setconcurrency(2)); -#endif - - my_thread_global_init(); - - - if ((pagen= init_pagecache(&pagecache, PCACHE_SIZE, 0, 0, - PAGE_SIZE, 0)) == 0) - { - fprintf(stderr,"Got error: init_pagecache() (errno: %d)\n", - errno); - exit(1); - } - DBUG_PRINT("info", ("Page cache %d pages", pagen)); - { - unsigned char *buffr= malloc(PAGE_SIZE); - uint i; - memset(buffr, '\0', PAGE_SIZE); - for (i= 0; i < number_of_pages; i++) - { - pagecache_write(&pagecache, &file1, i, 3, (char*)buffr, - PAGECACHE_PLAIN_PAGE, - PAGECACHE_LOCK_LEFT_UNLOCKED, - PAGECACHE_PIN_LEFT_UNPINNED, - PAGECACHE_WRITE_DELAY, - 0); - } - flush_pagecache_blocks(&pagecache, &file1, FLUSH_FORCE_WRITE); - free(buffr); - } - if ((error= pthread_mutex_lock(&LOCK_thread_count))) - { - fprintf(stderr,"LOCK_thread_count: %d from pthread_mutex_lock (errno: %d)\n", - error,errno); - exit(1); - } - while (number_of_readers != 0 || number_of_writers != 0) - { - if (number_of_readers != 0) - { - param=(int*) malloc(sizeof(int)); - *param= number_of_readers; - if ((error= pthread_create(&tid, &thr_attr, test_thread_reader, - (void*) param))) - { - fprintf(stderr,"Got error: %d from pthread_create (errno: %d)\n", - error,errno); - exit(1); - } - thread_count++; - number_of_readers--; - } - if (number_of_writers != 0) - { - param=(int*) malloc(sizeof(int)); - *param= number_of_writers; - if ((error= pthread_create(&tid, &thr_attr, test_thread_writer, - (void*) param))) - { - fprintf(stderr,"Got error: %d from pthread_create (errno: %d)\n", - error,errno); - exit(1); - } - thread_count++; - number_of_writers--; - } - } - DBUG_PRINT("info", ("Thread started")); - pthread_mutex_unlock(&LOCK_thread_count); - - pthread_attr_destroy(&thr_attr); - - /* wait finishing */ - if ((error= pthread_mutex_lock(&LOCK_thread_count))) - fprintf(stderr,"LOCK_thread_count: %d from pthread_mutex_lock\n",error); - while (thread_count) - { - if ((error= pthread_cond_wait(&COND_thread_count,&LOCK_thread_count))) - fprintf(stderr,"COND_thread_count: %d from pthread_cond_wait\n",error); - } - if ((error= pthread_mutex_unlock(&LOCK_thread_count))) - fprintf(stderr,"LOCK_thread_count: %d from pthread_mutex_unlock\n",error); - DBUG_PRINT("info", ("thread ended")); - - end_pagecache(&pagecache, 1); - DBUG_PRINT("info", ("Page cache ended")); - - if (my_close(file1.file, MYF(0)) != 0) - { - fprintf(stderr, "Got error during file1 closing from close() (errno: %d)\n", - errno); - exit(1); - } - /*my_delete(file1_name, MYF(0));*/ - my_thread_global_end(); - - DBUG_PRINT("info", ("file1 (%d) closed", file1.file)); - - DBUG_PRINT("info", ("Program end")); - - DBUG_RETURN(exit_status()); -} diff --git a/unittest/mysys/mf_pagecache_single.c b/unittest/mysys/mf_pagecache_single.c deleted file mode 100644 index 49ecd2986ab..00000000000 --- a/unittest/mysys/mf_pagecache_single.c +++ /dev/null @@ -1,587 +0,0 @@ -/* - TODO: use pthread_join instead of wait_for_thread_count_to_be_zero, like in - my_atomic-t.c (see BUG#22320). - Use diag() instead of fprintf(stderr). -*/ -#include -#include -#include -#include "test_file.h" - -#define PCACHE_SIZE (PAGE_SIZE*1024*10) - -#ifndef DBUG_OFF -static const char* default_dbug_option; -#endif - -static char *file1_name= (char*)"page_cache_test_file_1"; -static PAGECACHE_FILE file1; -static pthread_cond_t COND_thread_count; -static pthread_mutex_t LOCK_thread_count; -static uint thread_count; -static PAGECACHE pagecache; - -/* - File contance descriptors -*/ -static struct file_desc simple_read_write_test_file[]= -{ - {PAGE_SIZE, '\1'}, - { 0, 0} -}; -static struct file_desc simple_read_change_write_read_test_file[]= -{ - {PAGE_SIZE/2, '\65'}, - {PAGE_SIZE/2, '\1'}, - { 0, 0} -}; -static struct file_desc simple_pin_test_file1[]= -{ - {PAGE_SIZE*2, '\1'}, - { 0, 0} -}; -static struct file_desc simple_pin_test_file2[]= -{ - {PAGE_SIZE/2, '\1'}, - {PAGE_SIZE/2, (unsigned char)129}, - {PAGE_SIZE, '\1'}, - { 0, 0} -}; -static struct file_desc simple_delete_forget_test_file[]= -{ - {PAGE_SIZE, '\1'}, - { 0, 0} -}; -static struct file_desc simple_delete_flush_test_file[]= -{ - {PAGE_SIZE, '\2'}, - { 0, 0} -}; - - -/* - Recreate and reopen a file for test - - SYNOPSIS - reset_file() - file File to reset - file_name Path (and name) of file which should be reset -*/ - -void reset_file(PAGECACHE_FILE file, char *file_name) -{ - flush_pagecache_blocks(&pagecache, &file1, FLUSH_RELEASE); - if (my_close(file1.file, MYF(0)) != 0) - { - diag("Got error during %s closing from close() (errno: %d)\n", - file_name, errno); - exit(1); - } - my_delete(file_name, MYF(0)); - if ((file.file= my_open(file_name, - O_CREAT | O_TRUNC | O_RDWR, MYF(0))) == -1) - { - diag("Got error during %s creation from open() (errno: %d)\n", - file_name, errno); - exit(1); - } -} - -/* - Write then read page, check file on disk -*/ - -int simple_read_write_test() -{ - unsigned char *buffw= malloc(PAGE_SIZE); - unsigned char *buffr= malloc(PAGE_SIZE); - int res; - DBUG_ENTER("simple_read_write_test"); - bfill(buffw, PAGE_SIZE, '\1'); - pagecache_write(&pagecache, &file1, 0, 3, (char*)buffw, - PAGECACHE_PLAIN_PAGE, - PAGECACHE_LOCK_LEFT_UNLOCKED, - PAGECACHE_PIN_LEFT_UNPINNED, - PAGECACHE_WRITE_DELAY, - 0); - pagecache_read(&pagecache, &file1, 0, 3, (char*)buffr, - PAGECACHE_PLAIN_PAGE, - PAGECACHE_LOCK_LEFT_UNLOCKED, - 0); - ok((res= test(memcmp(buffr, buffw, PAGE_SIZE) == 0)), - "Simple write-read page "); - flush_pagecache_blocks(&pagecache, &file1, FLUSH_FORCE_WRITE); - ok((res&= test(test_file(file1, file1_name, PAGE_SIZE, PAGE_SIZE, - simple_read_write_test_file))), - "Simple write-read page file"); - if (res) - reset_file(file1, file1_name); - free(buffw); - free(buffr); - DBUG_RETURN(res); -} - - -/* - Prepare page, then read (and lock), change (write new value and unlock), - then check the page in the cache and on the disk -*/ -int simple_read_change_write_read_test() -{ - unsigned char *buffw= malloc(PAGE_SIZE); - unsigned char *buffr= malloc(PAGE_SIZE); - int res; - DBUG_ENTER("simple_read_change_write_read_test"); - /* prepare the file */ - bfill(buffw, PAGE_SIZE, '\1'); - pagecache_write(&pagecache, &file1, 0, 3, (char*)buffw, - PAGECACHE_PLAIN_PAGE, - PAGECACHE_LOCK_LEFT_UNLOCKED, - PAGECACHE_PIN_LEFT_UNPINNED, - PAGECACHE_WRITE_DELAY, - 0); - flush_pagecache_blocks(&pagecache, &file1, FLUSH_FORCE_WRITE); - /* test */ - pagecache_read(&pagecache, &file1, 0, 3, (char*)buffw, - PAGECACHE_PLAIN_PAGE, - PAGECACHE_LOCK_WRITE, - 0); - bfill(buffw, PAGE_SIZE/2, '\65'); - pagecache_write(&pagecache, &file1, 0, 3, (char*)buffw, - PAGECACHE_PLAIN_PAGE, - PAGECACHE_LOCK_WRITE_UNLOCK, - PAGECACHE_UNPIN, - PAGECACHE_WRITE_DELAY, - 0); - - pagecache_read(&pagecache, &file1, 0, 3, (char*)buffr, - PAGECACHE_PLAIN_PAGE, - PAGECACHE_LOCK_LEFT_UNLOCKED, - 0); - ok((res= test(memcmp(buffr, buffw, PAGE_SIZE) == 0)), - "Simple read-change-write-read page "); - flush_pagecache_blocks(&pagecache, &file1, FLUSH_FORCE_WRITE); - ok((res&= test(test_file(file1, file1_name, PAGE_SIZE, PAGE_SIZE, - simple_read_change_write_read_test_file))), - "Simple read-change-write-read page file"); - if (res) - reset_file(file1, file1_name); - free(buffw); - free(buffr); - DBUG_RETURN(res); -} - - -/* - Prepare page, read page 0 (and pin) then write page 1 and page 0. - Flush the file (shold flush only page 1 and return 1 (page 0 is - still pinned). - Check file on the disk. - Unpin and flush. - Check file on the disk. -*/ -int simple_pin_test() -{ - unsigned char *buffw= malloc(PAGE_SIZE); - unsigned char *buffr= malloc(PAGE_SIZE); - int res; - DBUG_ENTER("simple_pin_test"); - /* prepare the file */ - bfill(buffw, PAGE_SIZE, '\1'); - pagecache_write(&pagecache, &file1, 0, 3, (char*)buffw, - PAGECACHE_PLAIN_PAGE, - PAGECACHE_LOCK_LEFT_UNLOCKED, - PAGECACHE_PIN_LEFT_UNPINNED, - PAGECACHE_WRITE_DELAY, - 0); - /* test */ - if (flush_pagecache_blocks(&pagecache, &file1, FLUSH_FORCE_WRITE)) - { - diag("error in flush_pagecache_blocks\n"); - exit(1); - } - pagecache_read(&pagecache, &file1, 0, 3, (char*)buffw, - PAGECACHE_PLAIN_PAGE, - PAGECACHE_LOCK_WRITE, - 0); - pagecache_write(&pagecache, &file1, 1, 3, (char*)buffw, - PAGECACHE_PLAIN_PAGE, - PAGECACHE_LOCK_LEFT_UNLOCKED, - PAGECACHE_PIN_LEFT_UNPINNED, - PAGECACHE_WRITE_DELAY, - 0); - bfill(buffw + PAGE_SIZE/2, PAGE_SIZE/2, ((unsigned char) 129)); - pagecache_write(&pagecache, &file1, 0, 3, (char*)buffw, - PAGECACHE_PLAIN_PAGE, - PAGECACHE_LOCK_WRITE_TO_READ, - PAGECACHE_PIN_LEFT_PINNED, - PAGECACHE_WRITE_DELAY, - 0); - /* - We have to get error because one page of the file is pinned, - other page should be flushed - */ - if (!flush_pagecache_blocks(&pagecache, &file1, FLUSH_FORCE_WRITE)) - { - diag("Did not get error in flush_pagecache_blocks\n"); - res= 0; - goto err; - } - ok((res= test(test_file(file1, file1_name, PAGE_SIZE*2, PAGE_SIZE*2, - simple_pin_test_file1))), - "Simple pin page file with pin"); - pagecache_unlock_page(&pagecache, - &file1, - 0, - PAGECACHE_LOCK_READ_UNLOCK, - PAGECACHE_UNPIN, - 0, 0); - if (flush_pagecache_blocks(&pagecache, &file1, FLUSH_FORCE_WRITE)) - { - diag("Got error in flush_pagecache_blocks\n"); - res= 0; - goto err; - } - ok((res&= test(test_file(file1, file1_name, PAGE_SIZE*2, PAGE_SIZE, - simple_pin_test_file2))), - "Simple pin page result file"); - if (res) - reset_file(file1, file1_name); -err: - free(buffw); - free(buffr); - DBUG_RETURN(res); -} - -/* - Prepare page, write new value, then delete page from cache without flush, - on the disk should be page with old content written during preparation -*/ - -int simple_delete_forget_test() -{ - unsigned char *buffw= malloc(PAGE_SIZE); - unsigned char *buffr= malloc(PAGE_SIZE); - int res; - DBUG_ENTER("simple_delete_forget_test"); - /* prepare the file */ - bfill(buffw, PAGE_SIZE, '\1'); - pagecache_write(&pagecache, &file1, 0, 3, (char*)buffw, - PAGECACHE_PLAIN_PAGE, - PAGECACHE_LOCK_LEFT_UNLOCKED, - PAGECACHE_PIN_LEFT_UNPINNED, - PAGECACHE_WRITE_DELAY, - 0); - flush_pagecache_blocks(&pagecache, &file1, FLUSH_FORCE_WRITE); - /* test */ - bfill(buffw, PAGE_SIZE, '\2'); - pagecache_write(&pagecache, &file1, 0, 3, (char*)buffw, - PAGECACHE_PLAIN_PAGE, - PAGECACHE_LOCK_LEFT_UNLOCKED, - PAGECACHE_PIN_LEFT_UNPINNED, - PAGECACHE_WRITE_DELAY, - 0); - pagecache_delete_page(&pagecache, &file1, 0, - PAGECACHE_LOCK_WRITE, 0); - flush_pagecache_blocks(&pagecache, &file1, FLUSH_FORCE_WRITE); - ok((res= test(test_file(file1, file1_name, PAGE_SIZE, PAGE_SIZE, - simple_delete_forget_test_file))), - "Simple delete-forget page file"); - if (res) - reset_file(file1, file1_name); - free(buffw); - free(buffr); - DBUG_RETURN(res); -} - -/* - Prepare page with locking, write new content to the page, - delete page with flush and on existing lock, - check that page on disk contain new value. -*/ - -int simple_delete_flush_test() -{ - unsigned char *buffw= malloc(PAGE_SIZE); - unsigned char *buffr= malloc(PAGE_SIZE); - int res; - DBUG_ENTER("simple_delete_flush_test"); - /* prepare the file */ - bfill(buffw, PAGE_SIZE, '\1'); - pagecache_write(&pagecache, &file1, 0, 3, (char*)buffw, - PAGECACHE_PLAIN_PAGE, - PAGECACHE_LOCK_WRITE, - PAGECACHE_PIN, - PAGECACHE_WRITE_DELAY, - 0); - flush_pagecache_blocks(&pagecache, &file1, FLUSH_FORCE_WRITE); - /* test */ - bfill(buffw, PAGE_SIZE, '\2'); - pagecache_write(&pagecache, &file1, 0, 3, (char*)buffw, - PAGECACHE_PLAIN_PAGE, - PAGECACHE_LOCK_LEFT_WRITELOCKED, - PAGECACHE_PIN_LEFT_PINNED, - PAGECACHE_WRITE_DELAY, - 0); - pagecache_delete_page(&pagecache, &file1, 0, - PAGECACHE_LOCK_LEFT_WRITELOCKED, 1); - flush_pagecache_blocks(&pagecache, &file1, FLUSH_FORCE_WRITE); - ok((res= test(test_file(file1, file1_name, PAGE_SIZE, PAGE_SIZE, - simple_delete_flush_test_file))), - "Simple delete-forget page file"); - if (res) - reset_file(file1, file1_name); - free(buffw); - free(buffr); - DBUG_RETURN(res); -} - - -/* - write then read file bigger then cache -*/ - -int simple_big_test() -{ - unsigned char *buffw= (unsigned char *)malloc(PAGE_SIZE); - unsigned char *buffr= (unsigned char *)malloc(PAGE_SIZE); - struct file_desc *desc= - (struct file_desc *)malloc((PCACHE_SIZE/(PAGE_SIZE/2)) * - sizeof(struct file_desc)); - int res, i; - DBUG_ENTER("simple_big_test"); - /* prepare the file twice larger then cache */ - for (i= 0; i < PCACHE_SIZE/(PAGE_SIZE/2); i++) - { - bfill(buffw, PAGE_SIZE, (unsigned char) (i & 0xff)); - desc[i].length= PAGE_SIZE; - desc[i].content= (i & 0xff); - pagecache_write(&pagecache, &file1, i, 3, (char*)buffw, - PAGECACHE_PLAIN_PAGE, - PAGECACHE_LOCK_LEFT_UNLOCKED, - PAGECACHE_PIN_LEFT_UNPINNED, - PAGECACHE_WRITE_DELAY, - 0); - } - ok(1, "Simple big file write"); - /* check written pages sequentally read */ - for (i= 0; i < PCACHE_SIZE/(PAGE_SIZE/2); i++) - { - int j; - pagecache_read(&pagecache, &file1, i, 3, (char*)buffr, - PAGECACHE_PLAIN_PAGE, - PAGECACHE_LOCK_LEFT_UNLOCKED, - 0); - for(j= 0; j < PAGE_SIZE; j++) - { - if (buffr[j] != (i & 0xff)) - { - diag("simple_big_test seq: page %u byte %u mismatch\n", i, j); - return 0; - } - } - } - ok(1, "simple big file sequentally read"); - /* chack random reads */ - for (i= 0; i < PCACHE_SIZE/(PAGE_SIZE); i++) - { - int j, page; - page= rand() % (PCACHE_SIZE/(PAGE_SIZE/2)); - pagecache_read(&pagecache, &file1, page, 3, (char*)buffr, - PAGECACHE_PLAIN_PAGE, - PAGECACHE_LOCK_LEFT_UNLOCKED, - 0); - for(j= 0; j < PAGE_SIZE; j++) - { - if (buffr[j] != (page & 0xff)) - { - diag("simple_big_test rnd: page %u byte %u mismatch\n", page, j); - return 0; - } - } - } - ok(1, "simple big file random read"); - flush_pagecache_blocks(&pagecache, &file1, FLUSH_FORCE_WRITE); - - ok((res= test(test_file(file1, file1_name, PCACHE_SIZE*2, PAGE_SIZE, - desc))), - "Simple big file"); - if (res) - reset_file(file1, file1_name); - free(buffw); - free(buffr); - DBUG_RETURN(res); -} -/* - Thread function -*/ - -static void *test_thread(void *arg) -{ - int param=*((int*) arg); - - my_thread_init(); - DBUG_ENTER("test_thread"); - - DBUG_PRINT("enter", ("param: %d", param)); - - if (!simple_read_write_test() || - !simple_read_change_write_read_test() || - !simple_pin_test() || - !simple_delete_forget_test() || - !simple_delete_flush_test() || - !simple_big_test()) - exit(1); - - DBUG_PRINT("info", ("Thread %s ended\n", my_thread_name())); - pthread_mutex_lock(&LOCK_thread_count); - thread_count--; - VOID(pthread_cond_signal(&COND_thread_count)); /* Tell main we are ready */ - pthread_mutex_unlock(&LOCK_thread_count); - free((gptr) arg); - my_thread_end(); - DBUG_RETURN(0); -} - - -int main(int argc, char **argv __attribute__((unused))) -{ - pthread_t tid; - pthread_attr_t thr_attr; - int *param, error, pagen; - - MY_INIT(argv[0]); - -#ifndef DBUG_OFF -#if defined(__WIN__) - default_dbug_option= "d:t:i:O,\\test_pagecache_single.trace"; -#else - default_dbug_option= "d:t:i:o,/tmp/test_pagecache_single.trace"; -#endif - if (argc > 1) - { - DBUG_SET(default_dbug_option); - DBUG_SET_INITIAL(default_dbug_option); - } -#endif - - - DBUG_ENTER("main"); - DBUG_PRINT("info", ("Main thread: %s\n", my_thread_name())); - if ((file1.file= my_open(file1_name, - O_CREAT | O_TRUNC | O_RDWR, MYF(0))) == -1) - { - fprintf(stderr, "Got error during file1 creation from open() (errno: %d)\n", - errno); - exit(1); - } - DBUG_PRINT("info", ("file1: %d", file1.file)); - if (chmod(file1_name, S_IRWXU | S_IRWXG | S_IRWXO) != 0) - { - fprintf(stderr, "Got error during file1 chmod() (errno: %d)\n", - errno); - exit(1); - } - my_pwrite(file1.file, "test file", 9, 0, MYF(0)); - - if ((error= pthread_cond_init(&COND_thread_count, NULL))) - { - fprintf(stderr, "Got error: %d from pthread_cond_init (errno: %d)\n", - error, errno); - exit(1); - } - if ((error= pthread_mutex_init(&LOCK_thread_count, MY_MUTEX_INIT_FAST))) - { - fprintf(stderr, "Got error: %d from pthread_cond_init (errno: %d)\n", - error, errno); - exit(1); - } - - if ((error= pthread_attr_init(&thr_attr))) - { - fprintf(stderr,"Got error: %d from pthread_attr_init (errno: %d)\n", - error,errno); - exit(1); - } - if ((error= pthread_attr_setdetachstate(&thr_attr, PTHREAD_CREATE_DETACHED))) - { - fprintf(stderr, - "Got error: %d from pthread_attr_setdetachstate (errno: %d)\n", - error,errno); - exit(1); - } - -#ifndef pthread_attr_setstacksize /* void return value */ - if ((error= pthread_attr_setstacksize(&thr_attr, 65536L))) - { - fprintf(stderr,"Got error: %d from pthread_attr_setstacksize (errno: %d)\n", - error,errno); - exit(1); - } -#endif -#ifdef HAVE_THR_SETCONCURRENCY - VOID(thr_setconcurrency(2)); -#endif - - my_thread_global_init(); - - plan(12); - - if ((pagen= init_pagecache(&pagecache, PCACHE_SIZE, 0, 0, - PAGE_SIZE, 0)) == 0) - { - fprintf(stderr,"Got error: init_pagecache() (errno: %d)\n", - errno); - exit(1); - } - DBUG_PRINT("info", ("Page cache %d pages", pagen)); - - if ((error=pthread_mutex_lock(&LOCK_thread_count))) - { - fprintf(stderr,"Got error: %d from pthread_mutex_lock (errno: %d)\n", - error,errno); - exit(1); - } - param=(int*) malloc(sizeof(int)); - *param= 1; - if ((error= pthread_create(&tid, &thr_attr, test_thread, (void*) param))) - { - fprintf(stderr,"Got error: %d from pthread_create (errno: %d)\n", - error,errno); - exit(1); - } - thread_count++; - DBUG_PRINT("info", ("Thread started")); - pthread_mutex_unlock(&LOCK_thread_count); - - pthread_attr_destroy(&thr_attr); - - if ((error= pthread_mutex_lock(&LOCK_thread_count))) - fprintf(stderr,"Got error: %d from pthread_mutex_lock\n",error); - while (thread_count) - { - if ((error= pthread_cond_wait(&COND_thread_count,&LOCK_thread_count))) - fprintf(stderr,"Got error: %d from pthread_cond_wait\n",error); - } - if ((error= pthread_mutex_unlock(&LOCK_thread_count))) - fprintf(stderr,"Got error: %d from pthread_mutex_unlock\n",error); - DBUG_PRINT("info", ("thread ended")); - - end_pagecache(&pagecache, 1); - DBUG_PRINT("info", ("Page cache ended")); - - if (my_close(file1.file, MYF(0)) != 0) - { - fprintf(stderr, "Got error during file1 closing from close() (errno: %d)\n", - errno); - exit(1); - } - /*my_delete(file1_name, MYF(0));*/ - my_thread_global_end(); - - DBUG_PRINT("info", ("file1 (%d) closed", file1.file)); - - DBUG_PRINT("info", ("Program end")); - - DBUG_RETURN(exit_status()); -} diff --git a/unittest/mysys/test_file.c b/unittest/mysys/test_file.c deleted file mode 100644 index 758d0bfa81b..00000000000 --- a/unittest/mysys/test_file.c +++ /dev/null @@ -1,68 +0,0 @@ -#include -#include -#include -#include "test_file.h" - - -/* - Check that file contance correspond to descriptor - - SYNOPSIS - test_file() - file File to test - file_name Path (and name) of file which is tested - size size of file - buff_size size of buffer which is enought to check the file - desc file descriptor to check with - - RETURN - 1 file if OK - 0 error -*/ - -int test_file(PAGECACHE_FILE file, char *file_name, - off_t size, size_t buff_size, struct file_desc *desc) -{ - MY_STAT stat_buff, *stat; - unsigned char *buffr= malloc(buff_size); - off_t pos= 0; - size_t byte; - int step= 0; - - if ((stat= my_stat(file_name, &stat_buff, MYF(0))) == NULL) - { - diag("Can't stat() %s (errno: %d)\n", file_name, errno); - return 0; - } - if (stat->st_size != size) - { - diag("file %s size is %lu (should be %lu)\n", - file_name, (ulong) stat->st_size, (ulong) size); - return 0; - } - /* check content */ - my_seek(file.file, 0, SEEK_SET, MYF(0)); - while (desc[step].length != 0) - { - if (my_read(file.file, (char*)buffr, desc[step].length, MYF(0)) != - desc[step].length) - { - diag("Can't read %u bytes from %s (errno: %d)\n", - (uint)desc[step].length, file_name, errno); - return 0; - } - for (byte= 0; byte < desc[step].length; byte++) - { - if (buffr[byte] != desc[step].content) - { - diag("content of %s mismatch 0x%x in position %lu instead of 0x%x\n", - file_name, (uint) buffr[byte], (ulong) (pos + byte), - desc[step].content); - return 0; - } - } - pos+= desc[step].length; - step++; - } - return 1; -} diff --git a/unittest/mysys/test_file.h b/unittest/mysys/test_file.h deleted file mode 100644 index ea787c123ed..00000000000 --- a/unittest/mysys/test_file.h +++ /dev/null @@ -1,14 +0,0 @@ - -#include - -/* - File content descriptor -*/ -struct file_desc -{ - unsigned int length; - unsigned char content; -}; - -int test_file(PAGECACHE_FILE file, char *file_name, - off_t size, size_t buff_size, struct file_desc *desc); -- cgit v1.2.1 From 91a8199773a8ee6b4d5a00b337d9b49a69dfc1ea Mon Sep 17 00:00:00 2001 From: unknown Date: Mon, 12 Feb 2007 14:23:43 +0200 Subject: Postmerge fix (including changing type of LSN) Some debug info and comments added include/pagecache.h: postmerge fix mysys/mf_pagecache.c: Postmerge fix (including changing type of LSN) Additional DBUG_ASSERTs added Comment about pinning mechanism added storage/maria/ma_control_file.c: Used the same LSN storing procedure everywhere Postmerge fix (including changing type of LSN) storage/maria/ma_control_file.h: Postmerge fix (including changing type of LSN) storage/maria/ma_loghandler.c: Postmerge fix (including changing type of LSN) storage/maria/ma_loghandler.h: Postmerge fix (including changing type of LSN) storage/maria/ma_loghandler_lsn.h: Postmerge fix (including changing type of LSN) storage/maria/unittest/Makefile.am: Postmerge fix storage/maria/unittest/ma_control_file-t.c: Postmerge fix (including changing type of LSN) storage/maria/unittest/ma_test_loghandler-t.c: Postmerge fix (including changing type of LSN) storage/maria/unittest/ma_test_loghandler_multigroup-t.c: Postmerge fix (including changing type of LSN) storage/maria/unittest/ma_test_loghandler_multithread-t.c: Postmerge fix (including changing type of LSN) storage/maria/unittest/ma_test_loghandler_pagecache-t.c: Postmerge fix (including changing type of LSN) storage/maria/unittest/mf_pagecache_consist.c: Postmerge fix (including changing type of LSN) storage/maria/unittest/mf_pagecache_single.c: Postmerge fix (including changing type of LSN) --- include/pagecache.h | 14 +- mysys/mf_pagecache.c | 158 ++-- storage/maria/ma_control_file.c | 34 +- storage/maria/ma_control_file.h | 10 +- storage/maria/ma_loghandler.c | 827 +++++++++++---------- storage/maria/ma_loghandler.h | 8 +- storage/maria/ma_loghandler_lsn.h | 41 +- storage/maria/unittest/Makefile.am | 56 +- storage/maria/unittest/ma_control_file-t.c | 57 +- storage/maria/unittest/ma_test_loghandler-t.c | 136 ++-- .../unittest/ma_test_loghandler_multigroup-t.c | 133 ++-- .../unittest/ma_test_loghandler_multithread-t.c | 59 +- .../unittest/ma_test_loghandler_pagecache-t.c | 3 +- storage/maria/unittest/mf_pagecache_consist.c | 1 + storage/maria/unittest/mf_pagecache_single.c | 5 +- 15 files changed, 789 insertions(+), 753 deletions(-) diff --git a/include/pagecache.h b/include/pagecache.h index b507c1fb4ae..32e23fb54ea 100644 --- a/include/pagecache.h +++ b/include/pagecache.h @@ -21,6 +21,7 @@ C_MODE_START #include "../storage/maria/ma_loghandler_lsn.h" +#include /* Type of the page */ enum pagecache_page_type @@ -73,8 +74,6 @@ enum pagecache_write_mode typedef void *PAGECACHE_PAGE_LINK; -typedef uint64 LSN; - /* file descriptor for Maria */ typedef struct st_pagecache_file { @@ -152,8 +151,8 @@ typedef struct st_pagecache ulong param_division_limit; /* min. percentage of warm blocks */ ulong param_age_threshold; /* determines when hot block is downgraded */ - /* Statistics variables. These are reset in reset_key_cache_counters(). */ - ulong global_blocks_changed; /* number of currently dirty blocks */ + /* Statistics variables. These are reset in reset_pagecache_counters(). */ + ulong global_blocks_changed; /* number of currently dirty blocks */ ulonglong global_cache_w_requests;/* number of write requests (write hits) */ ulonglong global_cache_write; /* number of writes from cache to files */ ulonglong global_cache_r_requests;/* number of read requests (read hits) */ @@ -200,14 +199,12 @@ extern void pagecache_unlock_page(PAGECACHE *pagecache, pgcache_page_no_t pageno, enum pagecache_page_lock lock, enum pagecache_page_pin pin, - my_bool stamp_this_page, - LSN_PTR first_REDO_LSN_for_page); + LSN first_REDO_LSN_for_page); extern void pagecache_unlock(PAGECACHE *pagecache, PAGECACHE_PAGE_LINK *link, enum pagecache_page_lock lock, enum pagecache_page_pin pin, - my_bool stamp_this_page, - LSN_PTR first_REDO_LSN_for_page); + LSN first_REDO_LSN_for_page); extern void pagecache_unpin_page(PAGECACHE *pagecache, PAGECACHE_FILE *file, pgcache_page_no_t pageno); @@ -225,6 +222,7 @@ extern void end_pagecache(PAGECACHE *keycache, my_bool cleanup); extern my_bool pagecache_collect_changed_blocks_with_lsn(PAGECACHE *pagecache, LEX_STRING *str, LSN *max_lsn); +extern int reset_pagecache_counters(const char *name, PAGECACHE *pagecache); C_MODE_END #endif /* _keycache_h */ diff --git a/mysys/mf_pagecache.c b/mysys/mf_pagecache.c index 3773119109a..9ef3d5841b5 100755 --- a/mysys/mf_pagecache.c +++ b/mysys/mf_pagecache.c @@ -538,7 +538,7 @@ static int ___pagecache_pthread_cond_signal(pthread_cond_t *cond); #define pagecache_pthread_cond_signal pthread_cond_signal #endif /* defined(PAGECACHE_DEBUG) */ -extern my_bool translog_flush(LSN *lsn); +extern my_bool translog_flush(LSN lsn); /* Write page to the disk @@ -570,13 +570,13 @@ static uint pagecache_fwrite(PAGECACHE *pagecache, DBUG_PRINT("info", ("Log handler call")); /* TODO: integrate with page format */ #define PAGE_LSN_OFFSET 0 - lsn7korr(&lsn, buffer + PAGE_LSN_OFFSET); + lsn= lsn7korr(buffer + PAGE_LSN_OFFSET); /* check CONTROL_FILE_IMPOSSIBLE_FILENO & CONTROL_FILE_IMPOSSIBLE_LOG_OFFSET */ - DBUG_ASSERT(lsn.file_no != 0 && lsn.rec_offset != 0); - translog_flush(&lsn); + DBUG_ASSERT(lsn != 0); + translog_flush(lsn); } DBUG_RETURN(my_pwrite(filedesc->file, buffer, pagecache->block_size, (pageno)<<(pagecache->shift), flags)); @@ -1354,6 +1354,7 @@ static void unreg_request(PAGECACHE *pagecache, (ulong)block, BLOCK_NUMBER(pagecache, block), block->status, block->requests)); BLOCK_INFO(block); + DBUG_ASSERT(block->requests > 0); if (! --block->requests) { my_bool hot; @@ -1400,6 +1401,7 @@ static inline void remove_reader(PAGECACHE_BLOCK_LINK *block) { DBUG_ENTER("remove_reader"); BLOCK_INFO(block); + DBUG_ASSERT(block->hash_link->requests > 0); #ifdef THREAD if (! --block->hash_link->requests && block->condvar) pagecache_pthread_cond_signal(block->condvar); @@ -1722,12 +1724,14 @@ restart: if (page_status != PAGE_READ) { /* We don't need the page in the cache: we are going to write on disk */ + DBUG_ASSERT(hash_link->requests > 0); hash_link->requests--; unlink_hash(pagecache, hash_link); return 0; } if (!(block->status & BLOCK_IN_FLUSH)) { + DBUG_ASSERT(hash_link->requests > 0); hash_link->requests--; /* Remove block to invalidate the page in the block buffer @@ -1744,6 +1748,7 @@ restart: return 0; } /* Wait until the page is flushed on disk */ + DBUG_ASSERT(hash_link->requests > 0); hash_link->requests--; { #ifdef THREAD @@ -1795,6 +1800,7 @@ restart: } else { + DBUG_ASSERT(hash_link->requests > 0); hash_link->requests--; KEYCACHE_DBUG_PRINT("find_block", ("request waiting for old page to be saved")); @@ -1851,8 +1857,8 @@ restart: pagecache->blocks_used++; } pagecache->blocks_unused--; - DBUG_ASSERT((block->status & BLOCK_WRLOCK)); - DBUG_ASSERT(block->pins > 0); + DBUG_ASSERT((block->status & BLOCK_WRLOCK) == 0); + DBUG_ASSERT(block->pins == 0); block->status= 0; #ifndef DBUG_OFF block->type= PAGECACHE_EMPTY_PAGE; @@ -1911,18 +1917,19 @@ restart: block->hits_left= init_hits_left; block->last_hit_time= 0; if (reg_req) - reg_requests(pagecache, block,1); + reg_requests(pagecache, block, 1); hash_link->block= block; } + BLOCK_INFO(block); DBUG_ASSERT((block->status & BLOCK_WRLOCK) == 0); - DBUG_ASSERT(block->pins > 0); + DBUG_ASSERT(block->pins == 0); if (block->hash_link != hash_link && ! (block->status & BLOCK_IN_SWITCH) ) { /* this is a primary request for a new page */ DBUG_ASSERT((block->status & BLOCK_WRLOCK) == 0); - DBUG_ASSERT(block->pins > 0); + DBUG_ASSERT(block->pins == 0); block->status|= (BLOCK_IN_SWITCH | BLOCK_WRLOCK); KEYCACHE_DBUG_PRINT("find_block", @@ -2289,9 +2296,9 @@ static my_bool make_lock_and_pin(PAGECACHE *pagecache, retry: DBUG_PRINT("INFO", ("Retry block 0x%lx", (ulong)block)); BLOCK_INFO(block); - DBUG_ASSERT(block->hash_link->requests != 0); + DBUG_ASSERT(block->hash_link->requests > 0); block->hash_link->requests--; - DBUG_ASSERT(block->requests != 0); + DBUG_ASSERT(block->requests > 0); unreg_request(pagecache, block, 1); BLOCK_INFO(block); DBUG_RETURN(1); @@ -2412,8 +2419,19 @@ static void read_block(PAGECACHE *pagecache, pageno number of the block of data in the file lock lock change pin pin page - stamp_this_page put LSN stamp on the page - first_REDO_LSN_for_page + first_REDO_LSN_for_page do not set it if it is zero + + NOTE + Pininig uses requests registration mechanism it works following way: + | beginnig | ending | + | of func. | of func. | + ----------------------------+-------------+---------------+ + PAGECACHE_PIN_LEFT_PINNED | - | - | + PAGECACHE_PIN_LEFT_UNPINNED | reg request | unreg request | + PAGECACHE_PIN | reg request | - | + PAGECACHE_UNPIN | - | unreg request | + + */ void pagecache_unlock_page(PAGECACHE *pagecache, @@ -2421,8 +2439,7 @@ void pagecache_unlock_page(PAGECACHE *pagecache, pgcache_page_no_t pageno, enum pagecache_page_lock lock, enum pagecache_page_pin pin, - my_bool stamp_this_page, - LSN_PTR first_REDO_LSN_for_page) + LSN first_REDO_LSN_for_page) { PAGECACHE_BLOCK_LINK *block; int page_st; @@ -2444,15 +2461,15 @@ void pagecache_unlock_page(PAGECACHE *pagecache, DBUG_ASSERT(pagecache->can_be_used); inc_counter_for_resize_op(pagecache); - block= find_block(pagecache, file, pageno, 0, 0, 0, &page_st); + /* See NOTE for pagecache_unlock_page about registering requests */ + block= find_block(pagecache, file, pageno, 0, 0, + test(pin == PAGECACHE_PIN_LEFT_UNPINNED), &page_st); BLOCK_INFO(block); DBUG_ASSERT(block != 0 && page_st == PAGE_READ); - if (stamp_this_page) + if (first_REDO_LSN_for_page) { DBUG_ASSERT(lock == PAGECACHE_LOCK_WRITE_UNLOCK && pin == PAGECACHE_UNPIN); - /* TODO: insert LSN writing code */ - DBUG_ASSERT(first_REDO_LSN_for_page != 0); set_if_bigger(block->rec_lsn, first_REDO_LSN_for_page); } @@ -2472,7 +2489,8 @@ void pagecache_unlock_page(PAGECACHE *pagecache, remove_reader(block); /* Link the block into the LRU chain if it's the last submitted request - for the block and block will not be pinned + for the block and block will not be pinned. + See NOTE for pagecache_unlock_page about registering requests. */ if (pin != PAGECACHE_PIN_LEFT_PINNED) unreg_request(pagecache, block, 1); @@ -2504,7 +2522,6 @@ void pagecache_unpin_page(PAGECACHE *pagecache, DBUG_ENTER("pagecache_unpin_page"); DBUG_PRINT("enter", ("fd: %u page: %lu", (uint) file->file, (ulong) pageno)); - pagecache_pthread_mutex_lock(&pagecache->cache_lock); /* As soon as we keep lock cache can be used, and we have lock bacause want @@ -2513,6 +2530,7 @@ void pagecache_unpin_page(PAGECACHE *pagecache, DBUG_ASSERT(pagecache->can_be_used); inc_counter_for_resize_op(pagecache); + /* See NOTE for pagecache_unlock_page about registering requests */ block= find_block(pagecache, file, pageno, 0, 0, 0, &page_st); DBUG_ASSERT(block != 0 && page_st == PAGE_READ); @@ -2539,7 +2557,8 @@ void pagecache_unpin_page(PAGECACHE *pagecache, remove_reader(block); /* Link the block into the LRU chain if it's the last submitted request - for the block and block will not be pinned + for the block and block will not be pinned. + See NOTE for pagecache_unlock_page about registering requests */ unreg_request(pagecache, block, 1); @@ -2561,16 +2580,14 @@ void pagecache_unpin_page(PAGECACHE *pagecache, link direct link to page (returned by read or write) lock lock change pin pin page - stamp_this_page put LSN stamp on the page - first_REDO_LSN_for_page + first_REDO_LSN_for_page do not set it if it is zero */ void pagecache_unlock(PAGECACHE *pagecache, PAGECACHE_PAGE_LINK *link, enum pagecache_page_lock lock, enum pagecache_page_pin pin, - my_bool stamp_this_page, - LSN_PTR first_REDO_LSN_for_page) + LSN first_REDO_LSN_for_page) { PAGECACHE_BLOCK_LINK *block= (PAGECACHE_BLOCK_LINK *)link; DBUG_ENTER("pagecache_unlock"); @@ -2580,8 +2597,12 @@ void pagecache_unlock(PAGECACHE *pagecache, (ulong) block->hash_link->pageno, page_cache_page_lock_str[lock], page_cache_page_pin_str[pin])); - /* we do not allow any lock/pin increasing here */ + /* + We do not allow any lock/pin increasing here and page can't be + unpinned because we use direct link. + */ DBUG_ASSERT(pin != PAGECACHE_PIN && + pin != PAGECACHE_PIN_LEFT_UNPINNED && lock != PAGECACHE_LOCK_READ && lock != PAGECACHE_LOCK_WRITE); if (pin == PAGECACHE_PIN_LEFT_UNPINNED && @@ -2611,12 +2632,10 @@ void pagecache_unlock(PAGECACHE *pagecache, DBUG_ASSERT(pagecache->can_be_used); inc_counter_for_resize_op(pagecache); - if (stamp_this_page) + if (first_REDO_LSN_for_page) { DBUG_ASSERT(lock == PAGECACHE_LOCK_WRITE_UNLOCK && pin == PAGECACHE_UNPIN); - /* TODO: insert LSN writing code */ - DBUG_ASSERT(first_REDO_LSN_for_page != 0); set_if_bigger(block->rec_lsn, first_REDO_LSN_for_page); } @@ -2636,7 +2655,8 @@ void pagecache_unlock(PAGECACHE *pagecache, remove_reader(block); /* Link the block into the LRU chain if it's the last submitted request - for the block and block will not be pinned + for the block and block will not be pinned. + See NOTE for pagecache_unlock_page about registering requests. */ if (pin != PAGECACHE_PIN_LEFT_PINNED) unreg_request(pagecache, block, 1); @@ -2701,7 +2721,8 @@ void pagecache_unpin(PAGECACHE *pagecache, remove_reader(block); /* Link the block into the LRU chain if it's the last submitted request - for the block and block will not be pinned + for the block and block will not be pinned. + See NOTE for pagecache_unlock_page about registering requests. */ unreg_request(pagecache, block, 1); @@ -2790,10 +2811,11 @@ restart: inc_counter_for_resize_op(pagecache); pagecache->global_cache_r_requests++; + /* See NOTE for pagecache_unlock_page about registering requests. */ block= find_block(pagecache, file, pageno, level, test(lock == PAGECACHE_LOCK_WRITE), - test((pin == PAGECACHE_PIN_LEFT_PINNED) || - (pin == PAGECACHE_UNPIN)), + test((pin == PAGECACHE_PIN_LEFT_UNPINNED) || + (pin == PAGECACHE_PIN)), &page_st); DBUG_ASSERT(block->type == PAGECACHE_EMPTY_PAGE || block->type == type); @@ -2836,9 +2858,10 @@ restart: remove_reader(block); /* Link the block into the LRU chain if it's the last submitted request - for the block and block will not be pinned + for the block and block will not be pinned. + See NOTE for pagecache_unlock_page about registering requests. */ - if (pin != PAGECACHE_PIN_LEFT_PINNED && pin != PAGECACHE_PIN) + if (pin == PAGECACHE_PIN_LEFT_UNPINNED || pin == PAGECACHE_UNPIN) unreg_request(pagecache, block, 1); else *link= (PAGECACHE_PAGE_LINK)block; @@ -2898,6 +2921,8 @@ my_bool pagecache_delete_page(PAGECACHE *pagecache, page_cache_page_pin_str[pin])); DBUG_ASSERT(lock == PAGECACHE_LOCK_WRITE || lock == PAGECACHE_LOCK_LEFT_WRITELOCKED); + DBUG_ASSERT(pin == PAGECACHE_PIN || + pin == PAGECACHE_PIN_LEFT_PINNED); restart: @@ -2920,6 +2945,9 @@ restart: DBUG_RETURN(0); } block= link->block; + /* See NOTE for pagecache_unlock_page about registering requests. */ + if (pin == PAGECACHE_PIN) + reg_requests(pagecache, block, 1); DBUG_ASSERT(block != 0); if (make_lock_and_pin(pagecache, block, lock, pin)) { @@ -2972,9 +3000,9 @@ restart: make_lock_and_pin(pagecache, block, PAGECACHE_LOCK_WRITE_UNLOCK, PAGECACHE_UNPIN); + DBUG_ASSERT(link->requests > 0); link->requests--; - if (pin == PAGECACHE_PIN_LEFT_PINNED) - unreg_request(pagecache, block, 1); + /* See NOTE for pagecache_unlock_page about registering requests. */ free_block(pagecache, block); err: @@ -3119,16 +3147,15 @@ restart: inc_counter_for_resize_op(pagecache); pagecache->global_cache_w_requests++; - { - int need_wrlock= (write_mode != PAGECACHE_WRITE_DONE && - lock != PAGECACHE_LOCK_LEFT_WRITELOCKED && - lock != PAGECACHE_LOCK_WRITE_UNLOCK && - lock != PAGECACHE_LOCK_WRITE_TO_READ); - block= find_block(pagecache, file, pageno, level, - (need_wrlock ? 1 : 0), - (need_wrlock ? 1 : 0), - &page_st); - } + /* See NOTE for pagecache_unlock_page about registering requests. */ + block= find_block(pagecache, file, pageno, level, + test(write_mode != PAGECACHE_WRITE_DONE && + lock != PAGECACHE_LOCK_LEFT_WRITELOCKED && + lock != PAGECACHE_LOCK_WRITE_UNLOCK && + lock != PAGECACHE_LOCK_WRITE_TO_READ), + test((pin == PAGECACHE_PIN_LEFT_UNPINNED) || + (pin == PAGECACHE_PIN)), + &page_st); if (!block) { DBUG_ASSERT(write_mode != PAGECACHE_WRITE_DONE); @@ -3212,12 +3239,11 @@ restart: } /* Unregister the request */ - + DBUG_ASSERT(block->hash_link->requests > 0); block->hash_link->requests--; - if (pin != PAGECACHE_PIN_LEFT_PINNED && pin != PAGECACHE_PIN) - { + /* See NOTE for pagecache_unlock_page about registering requests. */ + if (pin == PAGECACHE_PIN_LEFT_UNPINNED || pin == PAGECACHE_UNPIN) unreg_request(pagecache, block, 1); - } else *link= (PAGECACHE_PAGE_LINK)block; @@ -3280,7 +3306,7 @@ static void free_block(PAGECACHE *pagecache, PAGECACHE_BLOCK_LINK *block) unlink_changed(block); DBUG_ASSERT((block->status & BLOCK_WRLOCK) == 0); - DBUG_ASSERT(block->pins > 0); + DBUG_ASSERT(block->pins == 0); block->status= 0; #ifndef DBUG_OFF block->type= PAGECACHE_EMPTY_PAGE; @@ -3361,7 +3387,7 @@ static int flush_cached_blocks(PAGECACHE *pagecache, } /* if the block is not pinned then it is not write locked */ DBUG_ASSERT((block->status & BLOCK_WRLOCK) == 0); - DBUG_ASSERT(block->pins > 0); + DBUG_ASSERT(block->pins == 0); #ifndef DBUG_OFF { int rc= @@ -3686,33 +3712,33 @@ int flush_pagecache_blocks(PAGECACHE *pagecache, Reset the counters of a key cache. SYNOPSIS - reset_key_cache_counters() + reset_pagecache_counters() name the name of a key cache - key_cache pointer to the key kache to be reset + pagecache pointer to the pagecache to be reset DESCRIPTION - This procedure is used by process_key_caches() to reset the counters of all - currently used key caches, both the default one and the named ones. + This procedure is used to reset the counters of all currently used key + caches, both the default one and the named ones. RETURN 0 on success (always because it can't fail) */ -static int reset_key_cache_counters(const char *name, PAGECACHE *key_cache) +int reset_pagecache_counters(const char *name, PAGECACHE *pagecache) { - DBUG_ENTER("reset_key_cache_counters"); - if (!key_cache->inited) + DBUG_ENTER("reset_pagecache_counters"); + if (!pagecache->inited) { DBUG_PRINT("info", ("Key cache %s not initialized.", name)); DBUG_RETURN(0); } DBUG_PRINT("info", ("Resetting counters for key cache %s.", name)); - key_cache->global_blocks_changed= 0; /* Key_blocks_not_flushed */ - key_cache->global_cache_r_requests= 0; /* Key_read_requests */ - key_cache->global_cache_read= 0; /* Key_reads */ - key_cache->global_cache_w_requests= 0; /* Key_write_requests */ - key_cache->global_cache_write= 0; /* Key_writes */ + pagecache->global_blocks_changed= 0; /* Key_blocks_not_flushed */ + pagecache->global_cache_r_requests= 0; /* Key_read_requests */ + pagecache->global_cache_read= 0; /* Key_reads */ + pagecache->global_cache_w_requests= 0; /* Key_write_requests */ + pagecache->global_cache_write= 0; /* Key_writes */ DBUG_RETURN(0); } diff --git a/storage/maria/ma_control_file.c b/storage/maria/ma_control_file.c index 47583466cd7..3f9af34b2f1 100644 --- a/storage/maria/ma_control_file.c +++ b/storage/maria/ma_control_file.c @@ -37,7 +37,7 @@ #define CONTROL_FILE_CHECKSUM_OFFSET (CONTROL_FILE_MAGIC_STRING_OFFSET + CONTROL_FILE_MAGIC_STRING_SIZE) #define CONTROL_FILE_CHECKSUM_SIZE 1 #define CONTROL_FILE_LSN_OFFSET (CONTROL_FILE_CHECKSUM_OFFSET + CONTROL_FILE_CHECKSUM_SIZE) -#define CONTROL_FILE_LSN_SIZE (4+4) +#define CONTROL_FILE_LSN_SIZE (3+4) #define CONTROL_FILE_FILENO_OFFSET (CONTROL_FILE_LSN_OFFSET + CONTROL_FILE_LSN_SIZE) #define CONTROL_FILE_FILENO_SIZE 4 #define CONTROL_FILE_SIZE (CONTROL_FILE_FILENO_OFFSET + CONTROL_FILE_FILENO_SIZE) @@ -59,20 +59,6 @@ uint32 last_logno; */ static int control_file_fd= -1; -static void lsn8store(char *buffer, const LSN *lsn) -{ - int4store(buffer, lsn->file_no); - int4store(buffer + CONTROL_FILE_FILENO_SIZE, lsn->rec_offset); -} - -static LSN lsn8korr(char *buffer) -{ - LSN tmp; - tmp.file_no= uint4korr(buffer); - tmp.rec_offset= uint4korr(buffer + CONTROL_FILE_FILENO_SIZE); - return tmp; -} - static char simple_checksum(char *buffer, uint size) { /* TODO: improve this sum if we want */ @@ -120,7 +106,7 @@ CONTROL_FILE_ERROR ma_control_file_create_or_open() "*store" and "*korr" calls in this file, and can even create backward compatibility problems. Beware! */ - DBUG_ASSERT(CONTROL_FILE_LSN_SIZE == (4+4)); + DBUG_ASSERT(CONTROL_FILE_LSN_SIZE == (3+4)); DBUG_ASSERT(CONTROL_FILE_FILENO_SIZE == 4); if (control_file_fd >= 0) /* already open */ @@ -154,11 +140,9 @@ CONTROL_FILE_ERROR ma_control_file_create_or_open() usable as soon as it has been written to the log). */ - LSN imposs_lsn= CONTROL_FILE_IMPOSSIBLE_LSN; - uint32 imposs_logno= CONTROL_FILE_IMPOSSIBLE_FILENO; - /* init the file with these "undefined" values */ - DBUG_RETURN(ma_control_file_write_and_force(&imposs_lsn, imposs_logno, + DBUG_RETURN(ma_control_file_write_and_force(CONTROL_FILE_IMPOSSIBLE_LSN, + CONTROL_FILE_IMPOSSIBLE_FILENO, CONTROL_FILE_UPDATE_ALL)); } @@ -216,7 +200,7 @@ CONTROL_FILE_ERROR ma_control_file_create_or_open() error= CONTROL_FILE_BAD_CHECKSUM; goto err; } - last_checkpoint_lsn= lsn8korr(buffer + CONTROL_FILE_LSN_OFFSET); + last_checkpoint_lsn= lsn7korr(buffer + CONTROL_FILE_LSN_OFFSET); last_logno= uint4korr(buffer + CONTROL_FILE_FILENO_OFFSET); DBUG_RETURN(0); @@ -253,7 +237,7 @@ err: 1 - Error */ -int ma_control_file_write_and_force(const LSN *checkpoint_lsn, uint32 logno, +int ma_control_file_write_and_force(const LSN checkpoint_lsn, uint32 logno, uint objs_to_write) { char buffer[CONTROL_FILE_SIZE]; @@ -277,9 +261,9 @@ int ma_control_file_write_and_force(const LSN *checkpoint_lsn, uint32 logno, DBUG_ASSERT(0); if (update_checkpoint_lsn) - lsn8store(buffer + CONTROL_FILE_LSN_OFFSET, checkpoint_lsn); + lsn7store(buffer + CONTROL_FILE_LSN_OFFSET, checkpoint_lsn); else /* store old value == change nothing */ - lsn8store(buffer + CONTROL_FILE_LSN_OFFSET, &last_checkpoint_lsn); + lsn7store(buffer + CONTROL_FILE_LSN_OFFSET, last_checkpoint_lsn); if (update_logno) int4store(buffer + CONTROL_FILE_FILENO_OFFSET, logno); @@ -297,7 +281,7 @@ int ma_control_file_write_and_force(const LSN *checkpoint_lsn, uint32 logno, /* TODO: you need some protection to be able to write last_* global vars */ if (update_checkpoint_lsn) - last_checkpoint_lsn= *checkpoint_lsn; + last_checkpoint_lsn= checkpoint_lsn; if (update_logno) last_logno= logno; diff --git a/storage/maria/ma_control_file.h b/storage/maria/ma_control_file.h index 4b5ddd006c1..8e5dafac24c 100644 --- a/storage/maria/ma_control_file.h +++ b/storage/maria/ma_control_file.h @@ -30,16 +30,14 @@ #define CONTROL_FILE_IMPOSSIBLE_FILENO 0 /* logs always have a header */ #define CONTROL_FILE_IMPOSSIBLE_LOG_OFFSET 0 -/* - indicate absence of LSN. -*/ -#define CONTROL_FILE_IMPOSSIBLE_LSN ((LSN){CONTROL_FILE_IMPOSSIBLE_FILENO,CONTROL_FILE_IMPOSSIBLE_LOG_OFFSET}) +/* indicate absence of LSN. */ +#define CONTROL_FILE_IMPOSSIBLE_LSN ((LSN)0) /* Here is the interface of this module */ /* LSN of the last checkoint - (if last_checkpoint_lsn.file_no == CONTROL_FILE_IMPOSSIBLE_FILENO + (if last_checkpoint_lsn == CONTROL_FILE_IMPOSSIBLE_LSN then there was never a checkpoint) */ extern LSN last_checkpoint_lsn; @@ -72,7 +70,7 @@ CONTROL_FILE_ERROR ma_control_file_create_or_open(); #define CONTROL_FILE_UPDATE_ALL 0 #define CONTROL_FILE_UPDATE_ONLY_LSN 1 #define CONTROL_FILE_UPDATE_ONLY_LOGNO 2 -int ma_control_file_write_and_force(const LSN *checkpoint_lsn, uint32 logno, +int ma_control_file_write_and_force(const LSN checkpoint_lsn, uint32 logno, uint objs_to_write); diff --git a/storage/maria/ma_loghandler.c b/storage/maria/ma_loghandler.c index d5c19e29ce2..01b4c68f12f 100644 --- a/storage/maria/ma_loghandler.c +++ b/storage/maria/ma_loghandler.c @@ -366,7 +366,7 @@ const char *maria_data_root; pointer to path */ -char *translog_filename_by_fileno(uint32 file_no, char *path) +static char *translog_filename_by_fileno(uint32 file_no, char *path) { char file_name[10 + 8 + 1]; char *res; @@ -391,7 +391,7 @@ char *translog_filename_by_fileno(uint32 file_no, char *path) file descriptor number */ -File open_logfile_by_number_no_cache(uint32 file_no) +static File open_logfile_by_number_no_cache(uint32 file_no) { File file; char path[FN_REFLEN]; @@ -421,7 +421,7 @@ File open_logfile_by_number_no_cache(uint32 file_no) 1 ERROR */ -my_bool translog_write_file_header() +static my_bool translog_write_file_header() { ulonglong timestamp; char page[TRANSLOG_PAGE_SIZE]; @@ -441,7 +441,8 @@ my_bool translog_write_file_header() /* loghandler page size/512 */ int2store(page + (8 + 8 + 4 + 4 + 4), TRANSLOG_PAGE_SIZE / 512); /* file number */ - int3store(page + (8 + 8 + 4 + 4 + 4 + 2), log_descriptor.horizon.file_no); + int3store(page + (8 + 8 + 4 + 4 + 4 + 2), + LSN_FILE_NO(log_descriptor.horizon)); bzero(page + (8 + 8 + 4 + 4 + 4 + 2 + 3), TRANSLOG_PAGE_SIZE - (8 + 8 + 4 + 4 + 4 + 2 + 3)); @@ -466,12 +467,11 @@ my_bool translog_write_file_header() 1 - Error */ -my_bool translog_buffer_init(struct st_translog_buffer *buffer) +static my_bool translog_buffer_init(struct st_translog_buffer *buffer) { DBUG_ENTER("translog_buffer_init"); /* This buffer offset */ - buffer->last_lsn.file_no= buffer->offset.file_no= 0; - buffer->last_lsn.rec_offset= buffer->offset.rec_offset= 0; + buffer->last_lsn= CONTROL_FILE_IMPOSSIBLE_LSN; /* This Buffer File */ buffer->file= 0; buffer->overlay= 0; @@ -528,9 +528,10 @@ static my_bool translog_close_log_file(File file) 1 Error */ -my_bool translog_create_new_file() +static my_bool translog_create_new_file() { int i; + uint32 file_no= LSN_FILE_NO(log_descriptor.horizon); DBUG_ENTER("translog_create_new_file"); @@ -544,11 +545,11 @@ my_bool translog_create_new_file() } if ((log_descriptor.log_file_num[0]= - open_logfile_by_number_no_cache(log_descriptor.horizon.file_no)) <= 0 || + open_logfile_by_number_no_cache(file_no)) <= 0 || translog_write_file_header()) DBUG_RETURN(1); - if (ma_control_file_write_and_force(NULL, log_descriptor.horizon.file_no, + if (ma_control_file_write_and_force(CONTROL_FILE_IMPOSSIBLE_LSN, file_no, CONTROL_FILE_UPDATE_ONLY_LOGNO)) DBUG_RETURN(1); @@ -657,31 +658,32 @@ static void translog_new_page_header(TRANSLOG_ADDRESS *horizon, ptr= cursor->ptr; /* Page number */ - int3store(ptr, horizon->rec_offset / TRANSLOG_PAGE_SIZE); - ptr +=3; + int3store(ptr, LSN_OFFSET(*horizon) / TRANSLOG_PAGE_SIZE); + ptr+= 3; /* File number */ - int3store(ptr, horizon->file_no); - ptr +=3; + int3store(ptr, LSN_FILE_NO(*horizon)); + ptr+= 3; *(ptr ++)= (uchar) log_descriptor.flags; if (log_descriptor.flags & TRANSLOG_PAGE_CRC) { #ifndef DBUG_OFF DBUG_PRINT("info", ("write 0x11223344 CRC to (%lu,0x%lx)", - (ulong) horizon->file_no, (ulong) horizon->rec_offset)); + (ulong) LSN_FILE_NO(*horizon), + (ulong) LSN_OFFSET(*horizon))); int4store(ptr, 0x11223344); #endif - ptr +=4; /* CRC will be put when page - will be finished */ + /* CRC will be put when page will be finished */ + ptr+= 4; } if (log_descriptor.flags & TRANSLOG_SECTOR_PROTECTION) { time_t tm; int2store(ptr, time(&tm) & 0xFFFF); - ptr +=(TRANSLOG_PAGE_SIZE / 512) * 2; + ptr+= (TRANSLOG_PAGE_SIZE / 512) * 2; } { uint len= (ptr -cursor->ptr); - horizon->rec_offset+= len; + *horizon+= len; /* it is increasing of offset part of the address */ cursor->current_page_size= len; if (!cursor->chaser) cursor->buffer->size+= len; @@ -819,10 +821,10 @@ static void translog_finish_page(TRANSLOG_ADDRESS *horizon, "Page addr: (%lu,0x%lx), " "size %lu (%lu), Pg: %u, left: %u", (uint) cursor->buffer_no, (ulong) cursor->buffer, - (ulong) cursor->buffer->offset.file_no, - (ulong) cursor->buffer->offset.rec_offset, - (ulong) horizon->file_no, - (ulong) (horizon->rec_offset - + (ulong) LSN_FILE_NO(cursor->buffer->offset), + (ulong) LSN_OFFSET(cursor->buffer->offset), + (ulong) LSN_FILE_NO(*horizon), + (ulong) (LSN_OFFSET(*horizon) - cursor->current_page_size), (ulong) cursor->buffer->size, (ulong) (cursor->ptr -cursor->buffer->buffer), @@ -830,9 +832,9 @@ static void translog_finish_page(TRANSLOG_ADDRESS *horizon, DBUG_ASSERT(cursor->ptr !=NULL); DBUG_ASSERT((cursor->ptr -cursor->buffer->buffer) %TRANSLOG_PAGE_SIZE == cursor->current_page_size % TRANSLOG_PAGE_SIZE); - DBUG_ASSERT(horizon->file_no == cursor->buffer->offset.file_no); - DBUG_ASSERT(cursor->buffer->offset.rec_offset + - (cursor->ptr -cursor->buffer->buffer) == horizon->rec_offset); + DBUG_ASSERT(LSN_FILE_NO(*horizon) == LSN_FILE_NO(cursor->buffer->offset)); + DBUG_ASSERT(LSN_OFFSET(cursor->buffer->offset) + + (cursor->ptr -cursor->buffer->buffer) == LSN_OFFSET(*horizon)); if (cursor->protected) { DBUG_PRINT("info", ("Already protected and finished")); @@ -843,7 +845,7 @@ static void translog_finish_page(TRANSLOG_ADDRESS *horizon, DBUG_PRINT("info", ("left %u", (uint) left)); bzero(cursor->ptr, left); cursor->ptr +=left; - horizon->rec_offset+= left; + *horizon+= left; /* offset increasing */ if (!cursor->chaser) cursor->buffer->size+= left; cursor->current_page_size= 0; @@ -888,6 +890,7 @@ static void translog_finish_page(TRANSLOG_ADDRESS *horizon, NOTE This buffer should be locked */ + static void translog_wait_for_writers(struct st_translog_buffer *buffer) { struct st_my_thread_var *thread; @@ -908,7 +911,8 @@ static void translog_wait_for_writers(struct st_translog_buffer *buffer) DBUG_PRINT("info", ("wait for writers... , thread 0x%lx, " "buffer #%u 0x%lx, locked by 0x%lx (0x%lx), " "mutex: 0x%lx", - thread, (uint) buffer->buffer_no, (ulong) buffer, + (ulong) thread, + (uint) buffer->buffer_no, (ulong) buffer, (ulong) buffer->locked_by, (ulong) thread, (ulong) &buffer->mutex)); #ifndef DBUG_OFF @@ -920,7 +924,8 @@ static void translog_wait_for_writers(struct st_translog_buffer *buffer) DBUG_PRINT("info", ("wait for writers done, thread 0x%lx, " "buffer #%u 0x%lx, locked by 0x%lx (0x%lx), " "mutex: 0x%lx", - thread, (uint) buffer->buffer_no, (ulong) buffer, + (ulong) thread, + (uint) buffer->buffer_no, (ulong) buffer, (ulong) buffer->locked_by, (ulong) thread, (ulong) &buffer->mutex)); #ifndef DBUG_OFF @@ -966,7 +971,8 @@ static void translog_wait_for_buffer_free(struct st_translog_buffer *buffer) DBUG_PRINT("info", ("wait for writers... , thread 0x%lx, " "buffer #%u 0x%lx, locked by 0x%lx (0x%lx), " "mutex: 0x%lx", - thread, (uint) buffer->buffer_no, (ulong) buffer, + (ulong) thread, + (uint) buffer->buffer_no, (ulong) buffer, (ulong) buffer->locked_by, (ulong) thread, (ulong) &buffer->mutex)); #ifndef DBUG_OFF @@ -978,7 +984,8 @@ static void translog_wait_for_buffer_free(struct st_translog_buffer *buffer) DBUG_PRINT("info", ("wait for writers done, thread 0x%lx, " "buffer #%u 0x%lx, locked by 0x%lx (0x%lx), " "mutex: 0x%lx", - thread, (uint) buffer->buffer_no, (ulong) buffer, + (ulong) thread, + (uint) buffer->buffer_no, (ulong) buffer, (ulong) buffer->locked_by, (ulong) thread, (ulong) &buffer->mutex)); #ifndef DBUG_OFF @@ -1027,6 +1034,7 @@ static void translog_cursor_init(struct st_buffer_cursor *cursor, cursor It's cursor buffer_no Number of buffer */ + static void translog_start_buffer(struct st_translog_buffer *buffer, struct st_buffer_cursor *cursor, uint8 buffer_no) @@ -1036,11 +1044,10 @@ static void translog_start_buffer(struct st_translog_buffer *buffer, ("Assign buffer #%u (0x%lx) to file %u, offset 0x%lx(%lu)", (uint) buffer->buffer_no, (ulong) buffer, (uint) log_descriptor.log_file_num[0], - (ulong) log_descriptor.horizon.rec_offset, - (ulong) log_descriptor.horizon.rec_offset)); + (ulong) LSN_OFFSET(log_descriptor.horizon), + (ulong) LSN_OFFSET(log_descriptor.horizon))); DBUG_ASSERT(buffer_no == buffer->buffer_no); - buffer->last_lsn.file_no= 0; - buffer->last_lsn.rec_offset= 0; + buffer->last_lsn= CONTROL_FILE_IMPOSSIBLE_LSN; buffer->offset= log_descriptor.horizon; buffer->file= log_descriptor.log_file_num[0]; buffer->overlay= 0; @@ -1086,9 +1093,9 @@ static my_bool translog_buffer_next(TRANSLOG_ADDRESS *horizon, my_bool chasing= cursor->chaser; DBUG_ENTER("translog_buffer_next"); - DBUG_PRINT("info", ("horizon (%u,0x%lx), chasing: %d", - (uint) log_descriptor.horizon.file_no, - (ulong) log_descriptor.horizon.rec_offset, chasing)); + DBUG_PRINT("info", ("horizon (%lu,0x%lx), chasing: %d", + (ulong) LSN_FILE_NO(log_descriptor.horizon), + (ulong) LSN_OFFSET(log_descriptor.horizon), chasing)); DBUG_ASSERT(cmp_translog_addr(log_descriptor.horizon, *horizon) >= 0); @@ -1105,8 +1112,9 @@ static my_bool translog_buffer_next(TRANSLOG_ADDRESS *horizon, #endif if (new_file) { - horizon->file_no++; - horizon->rec_offset= TRANSLOG_PAGE_SIZE; /* header page */ + /* move the horizon to the next file and its header page */ + *horizon+= LSN_ONE_FILE; + *horizon= LSN_REPLACE_OFFSET(*horizon, TRANSLOG_PAGE_SIZE); if (!chasing && translog_create_new_file()) { DBUG_RETURN(1); @@ -1278,7 +1286,7 @@ static translog_size_t translog_variable_record_1group_decode_len(uchar **src) 0 - Error */ -uint16 translog_get_total_chunk_length(uchar *page, uint16 offset) +static uint16 translog_get_total_chunk_length(uchar *page, uint16 offset) { DBUG_ENTER("translog_get_total_chunk_length"); switch (page[offset] & TRANSLOG_CHUNK_TYPE) { @@ -1380,9 +1388,10 @@ static my_bool translog_buffer_flush(struct st_translog_buffer *buffer) ("Buffer #%u 0x%lx: locked by 0x%lx (0x%lx), " "file: %u, offset (%lu,0x%lx), size %lu", (uint) buffer->buffer_no, (ulong) buffer, - (ulong) buffer->locked_by, my_thread_var, + (ulong) buffer->locked_by, (ulong) my_thread_var, (uint) buffer->file, - (ulong) buffer->offset.file_no, (ulong) buffer->offset.rec_offset, + (ulong) LSN_FILE_NO(buffer->offset), + (ulong) LSN_OFFSET(buffer->offset), (ulong) buffer->size)); DBUG_ASSERT(buffer->locked_by == my_thread_var); @@ -1407,7 +1416,7 @@ static my_bool translog_buffer_flush(struct st_translog_buffer *buffer) }; if (pagecache_write(log_descriptor.pagecache, &file, - (buffer->offset.rec_offset + i) / TRANSLOG_PAGE_SIZE, + (LSN_OFFSET(buffer->offset) + i) / TRANSLOG_PAGE_SIZE, 3, buffer->buffer + i, PAGECACHE_PLAIN_PAGE, @@ -1416,20 +1425,20 @@ static my_bool translog_buffer_flush(struct st_translog_buffer *buffer) { UNRECOVERABLE_ERROR(("Cant't write page (%lu,0x%lx) to pagecacte", (ulong) buffer->file, - (ulong) (buffer->offset.rec_offset + i))); + (ulong) (LSN_OFFSET(buffer->offset)+ i))); } } if (my_pwrite(buffer->file, (char*) buffer->buffer, - buffer->size, buffer->offset.rec_offset, + buffer->size, LSN_OFFSET(buffer->offset), MYF(MY_WME)) != buffer->size) { UNRECOVERABLE_ERROR(("Cant't buffer (%lu,0x%lx) size %lu to the disk (%d)", (ulong) buffer->file, - (ulong) buffer->offset.rec_offset, + (ulong) LSN_OFFSET(buffer->offset), (ulong) buffer->size, errno)); DBUG_RETURN(1); } - if (buffer->last_lsn.rec_offset != 0) /* if buffer->last_lsn is set */ + if (LSN_OFFSET(buffer->last_lsn) != 0) /* if buffer->last_lsn is set */ translog_set_sent_to_file(&buffer->last_lsn); /* Free buffer */ buffer->file= 0; @@ -1530,20 +1539,20 @@ static my_bool translog_page_validator(byte *page_addr, gptr data) uint8 flags; uchar *page= (uchar*) page_addr; DBUG_ENTER("translog_page_validator"); - TRANSLOG_ADDRESS *addr= ((TRANSLOG_VALIDATOR_DATA*) data)->addr; + TRANSLOG_ADDRESS addr= *((TRANSLOG_VALIDATOR_DATA*) data)->addr; ((TRANSLOG_VALIDATOR_DATA*) data)->was_recovered= 0; - if (uint3korr(page) != addr->rec_offset / TRANSLOG_PAGE_SIZE || - uint3korr(page + 3) != addr->file_no) + if (uint3korr(page) != LSN_OFFSET(addr) / TRANSLOG_PAGE_SIZE || + uint3korr(page + 3) != LSN_FILE_NO(addr)) { UNRECOVERABLE_ERROR(("Page (%lu,0x%lx): " "page address written in the page is incorrect :" "File %lu instead of %lu or page %lu instead of %lu", - (ulong) addr->file_no, (ulong) addr->rec_offset, - (ulong) uint3korr(page + 3), (ulong) addr->file_no, + (ulong) LSN_FILE_NO(addr), (ulong) LSN_OFFSET(addr), + (ulong) uint3korr(page + 3), (ulong) LSN_FILE_NO(addr), (ulong) uint3korr(page), - (ulong) addr->rec_offset / TRANSLOG_PAGE_SIZE)); + (ulong) LSN_OFFSET(addr) / TRANSLOG_PAGE_SIZE)); DBUG_RETURN(1); } flags= page[3 + 3]; @@ -1552,7 +1561,7 @@ static my_bool translog_page_validator(byte *page_addr, gptr data) { UNRECOVERABLE_ERROR(("Page (%lu,0x%lx): " "Garbage in the page flags field detected : %x", - (ulong) addr->file_no, (ulong) addr->rec_offset, + (ulong) LSN_FILE_NO(addr), (ulong) LSN_OFFSET(addr), (uint) flags)); DBUG_RETURN(1); } @@ -1565,7 +1574,7 @@ static my_bool translog_page_validator(byte *page_addr, gptr data) { UNRECOVERABLE_ERROR(("Page (%lu,0x%lx): " "CRC mismatch: calculated: %lx on the page %lx", - (ulong) addr->file_no, (ulong) addr->rec_offset, + (ulong) LSN_FILE_NO(addr), (ulong) LSN_OFFSET(addr), (ulong) crc, (ulong) uint4korr(page + 3 + 3 + 1))); DBUG_RETURN(1); } @@ -1584,8 +1593,8 @@ static my_bool translog_page_validator(byte *page_addr, gptr data) */ uint16 test= uint2korr(page + offset); DBUG_PRINT("info", ("sector #%u offset %u current %lx " - "read 0x%lx stored 0x%x%x", - i / 2, offset, current, + "read 0x%x stored 0x%x%x", + i / 2, offset, (ulong) current, (uint) uint2korr(page + offset), (uint) table[i], (uint) table[i + 1])); if (test < current) @@ -1614,8 +1623,8 @@ static my_bool translog_page_validator(byte *page_addr, gptr data) current= test; DBUG_PRINT("info", ("sector #%u offset %u current %lx " - "read 0x%lx stored 0x%x%x", - i / 2, offset, current, + "read 0x%x stored 0x%x%x", + i / 2, offset, (ulong) current, (uint) uint2korr(page + offset), (uint) table[i], (uint) table[i + 1])); } @@ -1623,6 +1632,7 @@ static my_bool translog_page_validator(byte *page_addr, gptr data) DBUG_RETURN(0); } + /* Get log page by file number and offset of the beginning of the page @@ -1637,19 +1647,21 @@ static my_bool translog_page_validator(byte *page_addr, gptr data) NULL - Error */ -uchar *translog_get_page(TRANSLOG_VALIDATOR_DATA *data, uchar *buffer) +static uchar *translog_get_page(TRANSLOG_VALIDATOR_DATA *data, uchar *buffer) { + TRANSLOG_ADDRESS addr= *(data->addr); uint cache_index; + uint32 file_no= LSN_FILE_NO(addr); DBUG_ENTER("translog_get_page"); DBUG_PRINT("enter", ("File %lu, Offset %lu(0x%lx)", - (ulong) data->addr->file_no, - (ulong) data->addr->rec_offset, - (ulong) data->addr->rec_offset)); + (ulong) file_no, + (ulong) LSN_OFFSET(addr), + (ulong) LSN_OFFSET(addr))); /* it is really page address */ - DBUG_ASSERT(data->addr->rec_offset % TRANSLOG_PAGE_SIZE == 0); + DBUG_ASSERT(LSN_OFFSET(addr) % TRANSLOG_PAGE_SIZE == 0); - if ((cache_index= log_descriptor.horizon.file_no - data->addr->file_no) < + if ((cache_index= LSN_FILE_NO(log_descriptor.horizon) - file_no) < OPENED_FILES_NUM) { PAGECACHE_FILE file; @@ -1657,7 +1669,7 @@ uchar *translog_get_page(TRANSLOG_VALIDATOR_DATA *data, uchar *buffer) if (log_descriptor.log_file_num[cache_index] == 0) { if ((log_descriptor.log_file_num[cache_index]= - open_logfile_by_number_no_cache(data->addr->file_no)) == 0) + open_logfile_by_number_no_cache(file_no)) == 0) { DBUG_RETURN(NULL); } @@ -1666,7 +1678,7 @@ uchar *translog_get_page(TRANSLOG_VALIDATOR_DATA *data, uchar *buffer) buffer= (uchar*) pagecache_valid_read(log_descriptor.pagecache, &file, - data->addr->rec_offset / TRANSLOG_PAGE_SIZE, + LSN_OFFSET(addr) / TRANSLOG_PAGE_SIZE, 3, (char*) buffer, PAGECACHE_PLAIN_PAGE, PAGECACHE_LOCK_LEFT_UNLOCKED, 0, @@ -1674,9 +1686,9 @@ uchar *translog_get_page(TRANSLOG_VALIDATOR_DATA *data, uchar *buffer) } else { - File file= open_logfile_by_number_no_cache(data->addr->file_no); + File file= open_logfile_by_number_no_cache(file_no); if (my_pread(file, (char*) buffer, TRANSLOG_PAGE_SIZE, - data->addr->rec_offset, MYF(MY_FNABP | MY_WME))) + LSN_OFFSET(addr), MYF(MY_FNABP | MY_WME))) buffer= NULL; else if (translog_page_validator((byte*) buffer, (gptr) data)) buffer= NULL; @@ -1705,25 +1717,28 @@ static my_bool translog_get_last_page_addr(TRANSLOG_ADDRESS *addr, { MY_STAT stat_buff, *stat; char path[FN_REFLEN]; + uint32 rec_offset; + uint32 file_no= LSN_FILE_NO(*addr); DBUG_ENTER("translog_get_last_page_addr"); - if ((stat= my_stat (translog_filename_by_fileno(addr->file_no, + if ((stat= my_stat (translog_filename_by_fileno(file_no, path), &stat_buff, MYF(MY_WME))) == NULL) DBUG_RETURN(1); DBUG_PRINT("info", ("File size %lu", (ulong) stat->st_size)); if (stat->st_size > TRANSLOG_PAGE_SIZE) { - addr->rec_offset= (((stat->st_size / TRANSLOG_PAGE_SIZE) - 1) * + rec_offset= (((stat->st_size / TRANSLOG_PAGE_SIZE) - 1) * TRANSLOG_PAGE_SIZE); - *last_page_ok= (stat->st_size == addr->rec_offset + TRANSLOG_PAGE_SIZE); + *last_page_ok= (stat->st_size == rec_offset + TRANSLOG_PAGE_SIZE); } else { *last_page_ok= 0; - addr->rec_offset= 0; + rec_offset= 0; } - DBUG_PRINT("info", ("Last page: 0x%lx, ok %d", (ulong) addr->rec_offset, + *addr= MAKE_LSN(file_no, rec_offset); + DBUG_PRINT("info", ("Last page: 0x%lx, ok %d", (ulong) rec_offset, *last_page_ok)); DBUG_RETURN(0); } @@ -1739,6 +1754,7 @@ static my_bool translog_get_last_page_addr(TRANSLOG_ADDRESS *addr, RETURN 1,3,4,5 - number of bytes to store given length */ + static uint translog_variable_record_length_bytes(translog_size_t length) { if (length < 250) @@ -1764,7 +1780,7 @@ static uint translog_variable_record_length_bytes(translog_size_t length) 0 - Error */ -uint16 translog_get_chunk_header_length(uchar *page, uint16 offset) +static uint16 translog_get_chunk_header_length(uchar *page, uint16 offset) { DBUG_ENTER("translog_get_chunk_header_length"); switch (page[offset] & TRANSLOG_CHUNK_TYPE) { @@ -1921,31 +1937,30 @@ my_bool translog_init(const char *directory, /* TODO: check that last checkpoint within present log addresses space */ /* find the log end */ - if (last_checkpoint_lsn.file_no == CONTROL_FILE_IMPOSSIBLE_FILENO) + if (LSN_FILE_NO(last_checkpoint_lsn) == CONTROL_FILE_IMPOSSIBLE_FILENO) { - DBUG_ASSERT(last_checkpoint_lsn.rec_offset == 0); + DBUG_ASSERT(LSN_OFFSET(last_checkpoint_lsn) == 0); /* there was no checkpoints we will read from the beginning */ - sure_page.file_no= 1; - sure_page.rec_offset= TRANSLOG_PAGE_SIZE; + sure_page= (LSN_ONE_FILE | TRANSLOG_PAGE_SIZE); } else { sure_page= last_checkpoint_lsn; - DBUG_ASSERT(sure_page.rec_offset % TRANSLOG_PAGE_SIZE != 0); - sure_page.rec_offset-= sure_page.rec_offset % TRANSLOG_PAGE_SIZE; + DBUG_ASSERT(LSN_OFFSET(sure_page) % TRANSLOG_PAGE_SIZE != 0); + sure_page-= LSN_OFFSET(sure_page) % TRANSLOG_PAGE_SIZE; } - log_descriptor.horizon.file_no= last_page.file_no= last_logno; + log_descriptor.horizon= last_page= MAKE_LSN(last_logno,0); if (translog_get_last_page_addr(&last_page, &pageok)) DBUG_RETURN(1); - if (last_page.rec_offset == 0) + if (LSN_OFFSET(last_page) == 0) { - if (last_page.file_no == 1) + if (LSN_FILE_NO(last_page) == 1) { logs_found= 0; /* file #1 has no pages */ } else { - last_page.file_no--; + last_page-= LSN_ONE_FILE; if (translog_get_last_page_addr(&last_page, &pageok)) DBUG_RETURN(1); } @@ -1956,25 +1971,22 @@ my_bool translog_init(const char *directory, TRANSLOG_ADDRESS current_page= sure_page; my_bool pageok; - DBUG_ASSERT(sure_page.file_no < last_page.file_no || - (sure_page.file_no == last_page.file_no && - sure_page.rec_offset <= last_page.rec_offset)); + DBUG_ASSERT(sure_page <= last_page); /* TODO: check page size */ - last_valid_page.file_no= CONTROL_FILE_IMPOSSIBLE_FILENO; - last_valid_page.rec_offset= 0; + last_valid_page= CONTROL_FILE_IMPOSSIBLE_LSN; /* scan and validate pages */ do { TRANSLOG_ADDRESS current_file_last_page; - current_file_last_page.file_no= current_page.file_no; + current_file_last_page= current_page; if (translog_get_last_page_addr(¤t_file_last_page, &pageok)) DBUG_RETURN(1); if (!pageok) { - DBUG_PRINT("error", ("File %u have no complete last page", - (uint) current_file_last_page.file_no)); + DBUG_PRINT("error", ("File %lu have no complete last page", + (ulong) LSN_FILE_NO(current_file_last_page))); old_log_was_recovered= 1; /* This file is not written till the end so it should be last */ last_page= current_file_last_page; @@ -1992,36 +2004,36 @@ my_bool translog_init(const char *directory, if (data.was_recovered) { DBUG_PRINT("error", ("file no %u (%d), rec_offset 0x%lx (%lu) (%d)", - (uint) current_page.file_no, - (uint3korr(page + 3) != current_page.file_no), - (ulong) current_page.rec_offset, - (ulong) (current_page.rec_offset / + (uint) LSN_FILE_NO(current_page), + (uint3korr(page + 3) != + LSN_FILE_NO(current_page)), + (ulong) LSN_OFFSET(current_page), + (ulong) (LSN_OFFSET(current_page) / TRANSLOG_PAGE_SIZE), (uint3korr(page) != - current_page.rec_offset / TRANSLOG_PAGE_SIZE))); + LSN_OFFSET(current_page) / + TRANSLOG_PAGE_SIZE))); old_log_was_recovered= 1; break; } last_valid_page= current_page; - current_page.rec_offset+= TRANSLOG_PAGE_SIZE; - } while (current_page.rec_offset <= current_file_last_page.rec_offset); - current_page.file_no++; - current_page.rec_offset= TRANSLOG_PAGE_SIZE; - } while (current_page.file_no <= last_page.file_no && + current_page+= TRANSLOG_PAGE_SIZE; /* increase offset */ + } while (current_page <= current_file_last_page); + current_page+= LSN_ONE_FILE; + current_page= LSN_REPLACE_OFFSET(current_page, TRANSLOG_PAGE_SIZE); + } while (LSN_FILE_NO(current_page) <= LSN_FILE_NO(last_page) && !old_log_was_recovered); - if (last_valid_page.file_no == CONTROL_FILE_IMPOSSIBLE_FILENO) + if (last_valid_page == CONTROL_FILE_IMPOSSIBLE_LSN) { - DBUG_ASSERT(last_valid_page.rec_offset == 0); - /* Panic!!! Even page which should be valid is invalid */ /* TODO: issue error */ DBUG_RETURN(1); } DBUG_PRINT("info", ("Last valid page is in file %lu offset %lu (0x%lx), " "Logs found: %d, was recovered: %d", - (ulong) last_valid_page.file_no, - (ulong) last_valid_page.rec_offset, - (ulong) last_valid_page.rec_offset, + (ulong) LSN_FILE_NO(last_valid_page), + (ulong) LSN_OFFSET(last_valid_page), + (ulong) LSN_OFFSET(last_valid_page), logs_found, old_log_was_recovered)); /* TODO: check server ID */ @@ -2034,7 +2046,8 @@ my_bool translog_init(const char *directory, uchar buffer[TRANSLOG_PAGE_SIZE], *page; uint16 chunk_offset; /* continue old log */ - DBUG_ASSERT(last_valid_page.file_no == log_descriptor.horizon.file_no); + DBUG_ASSERT(LSN_FILE_NO(last_valid_page)== + LSN_FILE_NO(log_descriptor.horizon)); if ((page= translog_get_page(&data, buffer)) == NULL || (chunk_offset= translog_get_first_chunk_offset(page)) == 0) @@ -2064,8 +2077,9 @@ my_bool translog_init(const char *directory, log_descriptor.bc.buffer->size+= chunk_offset; log_descriptor.bc.ptr+= chunk_offset; log_descriptor.bc.current_page_size= chunk_offset; - log_descriptor.horizon.rec_offset= - chunk_offset + last_valid_page.rec_offset; + log_descriptor.horizon= LSN_REPLACE_OFFSET(log_descriptor.horizon, + (chunk_offset + + LSN_OFFSET(last_valid_page))); DBUG_PRINT("info", ("Move Page #%u: 0x%lx, chaser: %d, Size: %lu (%lu)", (uint) log_descriptor.bc.buffer_no, (ulong) log_descriptor.bc.buffer, @@ -2088,16 +2102,14 @@ my_bool translog_init(const char *directory, if (!logs_found) { /* Start new log system from scratch */ - /* Current log number */ - log_descriptor.horizon.file_no= 1; /* Used space */ - log_descriptor.horizon.rec_offset= TRANSLOG_PAGE_SIZE; // header page + log_descriptor.horizon= MAKE_LSN(1, TRANSLOG_PAGE_SIZE); // header page /* Current logs file number in page cache */ log_descriptor.log_file_num[0]= - open_logfile_by_number_no_cache(log_descriptor.horizon.file_no); + open_logfile_by_number_no_cache(1); if (translog_write_file_header()) DBUG_RETURN(1); - if (ma_control_file_write_and_force(NULL, log_descriptor.horizon.file_no, + if (ma_control_file_write_and_force(CONTROL_FILE_IMPOSSIBLE_LSN, 1, CONTROL_FILE_UPDATE_ONLY_LOGNO)) DBUG_RETURN(1); /* assign buffer 0 */ @@ -2124,9 +2136,11 @@ my_bool translog_init(const char *directory, } else { - log_descriptor.horizon.file_no++; /* leave the demaged file - untouched */ - log_descriptor.horizon.rec_offset= TRANSLOG_PAGE_SIZE; /* header page */ + /* leave the demaged file untouched */ + log_descriptor.horizon+= LSN_ONE_FILE; + /* header page */ + log_descriptor.horizon= LSN_REPLACE_OFFSET(log_descriptor.horizon, + TRANSLOG_PAGE_SIZE); if (translog_create_new_file()) DBUG_RETURN(1); /* @@ -2140,8 +2154,8 @@ my_bool translog_init(const char *directory, /* all LSNs that are on disk are flushed */ log_descriptor.sent_to_file= log_descriptor.flushed= log_descriptor.horizon; - log_descriptor.flushed.rec_offset--; - log_descriptor.sent_to_file.rec_offset--; + log_descriptor.flushed--; /* offset decreased */ + log_descriptor.sent_to_file--; /* offset decreased */ DBUG_RETURN(0); } @@ -2162,10 +2176,11 @@ static void translog_buffer_destroy(struct st_translog_buffer *buffer) { DBUG_ENTER("translog_buffer_destroy"); DBUG_PRINT("enter", - ("Buffer #%u: 0x%lx, file: %u, offset (%u,0x%lx), size %lu", + ("Buffer #%u: 0x%lx, file: %u, offset (%lu,0x%lx), size %lu", (uint) buffer->buffer_no, (ulong) buffer, (uint) buffer->file, - (ulong) buffer->offset.file_no, (ulong) buffer->offset.rec_offset, + (ulong) LSN_FILE_NO(buffer->offset), + (ulong) LSN_OFFSET(buffer->offset), (ulong) buffer->size)); DBUG_ASSERT(buffer->waiting_filling_buffer.last_thread == 0); if (buffer->file) @@ -2300,23 +2315,24 @@ static my_bool translog_page_next(TRANSLOG_ADDRESS *horizon, if ((cursor->ptr +TRANSLOG_PAGE_SIZE > cursor->buffer->buffer + TRANSLOG_WRITE_BUFFER) || - (horizon->rec_offset + TRANSLOG_PAGE_SIZE > - log_descriptor.log_file_max_size)) + (LSN_OFFSET(*horizon) > + log_descriptor.log_file_max_size - TRANSLOG_PAGE_SIZE)) { DBUG_PRINT("info", ("Switch to next buffer, Buffer Size %lu (%lu) => %d, " "File size %lu max %lu => %d", (ulong) cursor->buffer->size, (ulong) (cursor->ptr -cursor->buffer->buffer), - (cursor->ptr +TRANSLOG_PAGE_SIZE > + (cursor->ptr + TRANSLOG_PAGE_SIZE > cursor->buffer->buffer + TRANSLOG_WRITE_BUFFER), - (ulong) horizon->rec_offset, + (ulong) LSN_OFFSET(*horizon), (ulong) log_descriptor.log_file_max_size, - (horizon->rec_offset + TRANSLOG_PAGE_SIZE > - log_descriptor.log_file_max_size))); + (LSN_OFFSET(*horizon) > + (log_descriptor.log_file_max_size - + TRANSLOG_PAGE_SIZE)))); if (translog_buffer_next(horizon, cursor, - (horizon->rec_offset + - TRANSLOG_PAGE_SIZE) > - log_descriptor.log_file_max_size)) + LSN_OFFSET(*horizon) > + (log_descriptor.log_file_max_size - + TRANSLOG_PAGE_SIZE))) DBUG_RETURN(1); *prev_buffer= buffer; DBUG_PRINT("info", ("Buffer #%u (0x%lu) have to be flushed", @@ -2353,9 +2369,10 @@ static my_bool translog_page_next(TRANSLOG_ADDRESS *horizon, 1 - Error */ -my_bool translog_write_data_on_page(TRANSLOG_ADDRESS *horizon, - struct st_buffer_cursor *cursor, - translog_size_t length, uchar *buffer) +static my_bool translog_write_data_on_page(TRANSLOG_ADDRESS *horizon, + struct st_buffer_cursor *cursor, + translog_size_t length, + uchar *buffer) { DBUG_ENTER("translog_write_data_on_page"); DBUG_PRINT("enter", ("Chunk length: %lu Page size %u", @@ -2367,7 +2384,7 @@ my_bool translog_write_data_on_page(TRANSLOG_ADDRESS *horizon, memmove(cursor->ptr, buffer, length); cursor->ptr+= length; - horizon->rec_offset+= length; + *horizon+= length; /* adds offset */ cursor->current_page_size+= length; if (!cursor->chaser) cursor->buffer->size+= length; @@ -2401,10 +2418,10 @@ my_bool translog_write_data_on_page(TRANSLOG_ADDRESS *horizon, 1 - Error */ -my_bool translog_write_parts_on_page(TRANSLOG_ADDRESS *horizon, - struct st_buffer_cursor *cursor, - translog_size_t length, - struct st_translog_parts *parts) +static my_bool translog_write_parts_on_page(TRANSLOG_ADDRESS *horizon, + struct st_buffer_cursor *cursor, + translog_size_t length, + struct st_translog_parts *parts) { translog_size_t left= length; uint cur= (uint) parts->current; @@ -2457,16 +2474,25 @@ my_bool translog_write_parts_on_page(TRANSLOG_ADDRESS *horizon, cursor->ptr+= len; } while (left); + DBUG_PRINT("info", ("Horizon (%lu,0x%lx) Length %lu(0x%lx)", + (ulong) LSN_FILE_NO(*horizon), + (ulong) LSN_OFFSET(*horizon), + (ulong) length, (ulong) length)); parts->current= cur; - horizon->rec_offset+= length; + *horizon+= length; /* offset increasing */ cursor->current_page_size+= length; if (!cursor->chaser) cursor->buffer->size+= length; - DBUG_PRINT("info", ("Write parts buffer #%u: 0x%lx, " - "chaser: %d, Size: %lu (%lu)", + DBUG_PRINT("info", ("Write parts buffer #%u: 0x%lx " + "chaser: %d Size: %lu (%lu) " + "Horizon (%lu,0x%lx) buff offset 0x%lx", (uint) cursor->buffer->buffer_no, (ulong) cursor->buffer, cursor->chaser, (ulong) cursor->buffer->size, - (ulong) (cursor->ptr -cursor->buffer->buffer))); + (ulong) (cursor->ptr -cursor->buffer->buffer), + (ulong) LSN_FILE_NO(*horizon), + (ulong) LSN_OFFSET(*horizon), + (ulong) (LSN_OFFSET(cursor->buffer->offset) + + cursor->buffer->size))); DBUG_ASSERT(cursor->chaser || ((ulong) (cursor->ptr -cursor->buffer->buffer) == cursor->buffer->size)); @@ -2687,15 +2713,14 @@ static my_bool translog_advance_pointer(uint pages, uint16 last_page_data) { translog_size_t last_page_offset= log_descriptor.page_overhead + last_page_data; - translog_size_t offset= TRANSLOG_PAGE_SIZE - log_descriptor.bc.current_page_size /* next - page - */ + - pages * TRANSLOG_PAGE_SIZE + last_page_offset; + translog_size_t offset= (TRANSLOG_PAGE_SIZE - + log_descriptor.bc.current_page_size + + pages * TRANSLOG_PAGE_SIZE + last_page_offset); translog_size_t buffer_end_offset, file_end_offset, min_offset; DBUG_ENTER("translog_advance_pointer"); - DBUG_PRINT("enter", ("Pointer: (%u, 0x%lx) + %u + %u pages + %u + %u", - (uint) log_descriptor.horizon.file_no, - (ulong) log_descriptor.horizon.rec_offset, + DBUG_PRINT("enter", ("Pointer: (%lu, 0x%lx) + %u + %u pages + %u + %u", + (ulong) LSN_FILE_NO(log_descriptor.horizon), + (ulong) LSN_OFFSET(log_descriptor.horizon), (uint) (TRANSLOG_PAGE_SIZE - log_descriptor.bc.current_page_size), pages, (uint) log_descriptor.page_overhead, @@ -2709,7 +2734,7 @@ static my_bool translog_advance_pointer(uint pages, uint16 last_page_data) struct st_translog_buffer *old_buffer; buffer_end_offset= TRANSLOG_WRITE_BUFFER - log_descriptor.bc.buffer->size; file_end_offset= - log_descriptor.log_file_max_size - log_descriptor.horizon.rec_offset; + log_descriptor.log_file_max_size - LSN_OFFSET(log_descriptor.horizon); DBUG_PRINT("info", ("offset: %lu, buffer_end_offs: %lu, " "file_end_offs: %lu", (ulong) offset, (ulong) buffer_end_offset, @@ -2719,14 +2744,14 @@ static my_bool translog_advance_pointer(uint pages, uint16 last_page_data) (uint) log_descriptor.bc.buffer->buffer_no, (uint) log_descriptor.bc.buffer_no, (ulong) log_descriptor.bc.buffer, - (ulong) log_descriptor.bc.buffer->offset.rec_offset, + (ulong) LSN_OFFSET(log_descriptor.bc.buffer->offset), (ulong) log_descriptor.bc.buffer->size, - (ulong) (log_descriptor.bc.buffer->offset.rec_offset + + (ulong) (LSN_OFFSET(log_descriptor.bc.buffer->offset) + log_descriptor.bc.buffer->size), - (ulong) log_descriptor.horizon.rec_offset)); - DBUG_ASSERT(log_descriptor.bc.buffer->offset.rec_offset + + (ulong) LSN_OFFSET(log_descriptor.horizon))); + DBUG_ASSERT(LSN_OFFSET(log_descriptor.bc.buffer->offset) + log_descriptor.bc.buffer->size == - log_descriptor.horizon.rec_offset); + LSN_OFFSET(log_descriptor.horizon)); if (offset <= buffer_end_offset && offset <= file_end_offset) break; @@ -2740,7 +2765,7 @@ static my_bool translog_advance_pointer(uint pages, uint16 last_page_data) min_offset= (buffer_end_offset < file_end_offset ? buffer_end_offset : file_end_offset); log_descriptor.bc.buffer->size+= min_offset; - log_descriptor.bc.ptr +=min_offset; + log_descriptor.bc.ptr+= min_offset; DBUG_PRINT("info", ("NewP buffer #%u: 0x%lx, chaser: %d, Size: %lu (%lu)", (uint) log_descriptor.bc.buffer->buffer_no, (ulong) log_descriptor.bc.buffer, @@ -2757,9 +2782,11 @@ static my_bool translog_advance_pointer(uint pages, uint16 last_page_data) if (file_end_offset <= buffer_end_offset) { - log_descriptor.horizon.file_no++; - log_descriptor.horizon.rec_offset= TRANSLOG_PAGE_SIZE; - DBUG_PRINT("info", ("New file %d", log_descriptor.horizon.file_no)); + log_descriptor.horizon+= LSN_ONE_FILE; + log_descriptor.horizon= LSN_REPLACE_OFFSET(log_descriptor.horizon, + TRANSLOG_PAGE_SIZE); + DBUG_PRINT("info", ("New file %lu", + (ulong) LSN_FILE_NO(log_descriptor.horizon))); if (translog_create_new_file()) { DBUG_RETURN(1); @@ -2768,7 +2795,7 @@ static my_bool translog_advance_pointer(uint pages, uint16 last_page_data) else { DBUG_PRINT("info", ("The same file")); - log_descriptor.horizon.rec_offset+= min_offset; + log_descriptor.horizon+= min_offset; /* offset increasing */ } translog_start_buffer(new_buffer, &log_descriptor.bc, new_buffer_no); if (translog_buffer_unlock(old_buffer)) @@ -2780,7 +2807,7 @@ static my_bool translog_advance_pointer(uint pages, uint16 last_page_data) log_descriptor.bc.ptr+= offset; log_descriptor.bc.buffer->size+= offset; translog_buffer_increase_writers(log_descriptor.bc.buffer); - log_descriptor.horizon.rec_offset+= offset; + log_descriptor.horizon+= offset; /* offset increasing */ log_descriptor.bc.current_page_size= last_page_offset; DBUG_PRINT("info", ("drop write_counter")); log_descriptor.bc.write_counter= 0; @@ -2801,9 +2828,9 @@ static my_bool translog_advance_pointer(uint pages, uint16 last_page_data) DBUG_ASSERT(log_descriptor.bc.buffer->buffer_no == log_descriptor.bc.buffer_no); DBUG_PRINT("info", - ("pointer moved to: (%u, 0x%lx)", - (uint) log_descriptor.horizon.file_no, - (ulong) log_descriptor.horizon.rec_offset)); + ("pointer moved to: (%lu, 0x%lx)", + (ulong) LSN_FILE_NO(log_descriptor.horizon), + (ulong) LSN_OFFSET(log_descriptor.horizon))); DBUG_ASSERT((log_descriptor.bc.ptr -log_descriptor.bc.buffer-> buffer) %TRANSLOG_PAGE_SIZE == log_descriptor.bc.current_page_size % TRANSLOG_PAGE_SIZE); @@ -2865,18 +2892,18 @@ static translog_size_t translog_get_current_group_size() DBUG_ENTER("translog_get_current_group_size"); - DBUG_PRINT("info", ("buffer_rest in pages %lu", buffer_rest)); + DBUG_PRINT("info", ("buffer_rest in pages %u", buffer_rest)); buffer_rest*= log_descriptor.page_capacity_chunk_2; /* in case of only half of buffer free we can write this and next buffer */ if (buffer_rest < log_descriptor.half_buffer_capacity_chunk_2) { - DBUG_PRINT("info", ("buffer_rest %lu -> add %lu", + DBUG_PRINT("info", ("buffer_rest %u -> add %lu", buffer_rest, (ulong) log_descriptor.buffer_capacity_chunk_2)); buffer_rest+= log_descriptor.buffer_capacity_chunk_2; } - DBUG_PRINT("info", ("buffer_rest %lu", buffer_rest)); + DBUG_PRINT("info", ("buffer_rest %u", buffer_rest)); DBUG_RETURN(buffer_rest); } @@ -2984,20 +3011,22 @@ translog_write_variable_record_1group(LSN *lsn, translog_write_parts_on_page(&horizon, &cursor, first_page, parts); - DBUG_PRINT("info", ("absolute horizon (%u,0x%lx), local (%u,0x%lx)", - (uint) log_descriptor.horizon.file_no, - (ulong) log_descriptor.horizon.rec_offset, - (uint) horizon.file_no, (ulong) horizon.rec_offset)); + DBUG_PRINT("info", ("absolute horizon (%lu,0x%lx), local (%lu,0x%lx)", + (ulong) LSN_FILE_NO(log_descriptor.horizon), + (ulong) LSN_OFFSET(log_descriptor.horizon), + (ulong) LSN_FILE_NO(horizon), + (ulong) LSN_OFFSET(horizon))); for (i= 0; i < full_pages; i++) { if (translog_write_variable_record_chunk2_page(parts, &horizon, &cursor)) DBUG_RETURN(1); - DBUG_PRINT("info", ("absolute horizon (%u,0x%lx), local (%u,0x%lx)", - (uint) log_descriptor.horizon.file_no, - (ulong) log_descriptor.horizon.rec_offset, - (uint) horizon.file_no, (ulong) horizon.rec_offset)); + DBUG_PRINT("info", ("absolute horizon (%lu,0x%lx), local (%lu,0x%lx)", + (ulong) LSN_FILE_NO(log_descriptor.horizon), + (ulong) LSN_OFFSET(log_descriptor.horizon), + (ulong) LSN_FILE_NO(horizon), + (ulong) LSN_OFFSET(horizon))); } if (additional_chunk3_page) @@ -3007,10 +3036,11 @@ translog_write_variable_record_1group(LSN *lsn, page_capacity_chunk_2 - 2, &horizon, &cursor)) DBUG_RETURN(1); - DBUG_PRINT("info", ("absolute horizon (%u,0x%lx), local (%u,0x%lx)", - (uint) log_descriptor.horizon.file_no, - (ulong) log_descriptor.horizon.rec_offset, - (uint) horizon.file_no, (ulong) horizon.rec_offset)); + DBUG_PRINT("info", ("absolute horizon (%lu,0x%lx), local (%lu,0x%lx)", + (ulong) LSN_FILE_NO(log_descriptor.horizon), + (ulong) LSN_OFFSET(log_descriptor.horizon), + (ulong) LSN_FILE_NO(horizon), + (ulong) LSN_OFFSET(horizon))); DBUG_ASSERT(cursor.current_page_size == TRANSLOG_PAGE_SIZE); } @@ -3018,10 +3048,11 @@ translog_write_variable_record_1group(LSN *lsn, record_rest, &horizon, &cursor)) DBUG_RETURN(1); - DBUG_PRINT("info", ("absolute horizon (%u,0x%lx), local (%u,0x%lx)", - (uint) log_descriptor.horizon.file_no, - (ulong) log_descriptor.horizon.rec_offset, - (uint) horizon.file_no, (ulong) horizon.rec_offset)); + DBUG_PRINT("info", ("absolute horizon (%lu,0x%lx), local (%lu,0x%lx)", + (ulong) LSN_FILE_NO(log_descriptor.horizon), + (ulong) LSN_OFFSET(log_descriptor.horizon), + (ulong) LSN_FILE_NO(horizon), + (ulong) LSN_OFFSET(horizon))); rc= translog_buffer_lock(cursor.buffer); if (!rc) @@ -3029,7 +3060,6 @@ translog_write_variable_record_1group(LSN *lsn, /* check if we wrote something on lst not full page and need to reconstruct CRC and sector protection - if (buffer->offset.rec_offset + buffer->size - horizon->rec_offset > */ translog_buffer_decrease_writers(cursor.buffer); } @@ -3129,19 +3159,19 @@ translog_write_variable_record_1chunk(LSN *lsn, NULL - error */ -static uchar *translog_put_LSN_diff(LSN *base_lsn, LSN *lsn, uchar *dst) +static uchar *translog_put_LSN_diff(LSN base_lsn, LSN lsn, uchar *dst) { DBUG_ENTER("translog_put_LSN_diff"); - DBUG_PRINT("enter", ("Base: (0x%lx,0x%lx), val: (0x%lx,0x%lx), dst 0x%lx", - (ulong) base_lsn->file_no, - (ulong) base_lsn->rec_offset, - (ulong) lsn->file_no, - (ulong) lsn->rec_offset, (ulong) dst)); - if (base_lsn->file_no == lsn->file_no) + DBUG_PRINT("enter", ("Base: (0x%lu,0x%lx), val: (0x%lu,0x%lx), dst 0x%lx", + (ulong) LSN_FILE_NO(base_lsn), + (ulong) LSN_OFFSET(base_lsn), + (ulong) LSN_FILE_NO(lsn), + (ulong) LSN_OFFSET(lsn), (ulong) dst)); + if (LSN_FILE_NO(base_lsn) == LSN_FILE_NO(lsn)) { uint32 diff; - DBUG_ASSERT(base_lsn->rec_offset > lsn->rec_offset); - diff= base_lsn->rec_offset - lsn->rec_offset; + DBUG_ASSERT(base_lsn > lsn); + diff= base_lsn - lsn; if (diff <= 0x3FFF) { dst-= 2; @@ -3171,16 +3201,16 @@ static uchar *translog_put_LSN_diff(LSN *base_lsn, LSN *lsn, uchar *dst) { uint32 diff; uint32 offset_diff; - ulonglong base_offset= base_lsn->rec_offset; - DBUG_ASSERT(base_lsn->file_no > lsn->file_no); - diff= base_lsn->file_no - lsn->file_no; - if (base_offset < lsn->rec_offset) + ulonglong base_offset= LSN_OFFSET(base_lsn); + DBUG_ASSERT(base_lsn > lsn); + diff= LSN_FILE_NO(base_lsn) - LSN_FILE_NO(lsn); + if (base_offset < LSN_OFFSET(lsn)) { /* take 1 from file offset */ diff--; base_offset+= 0x100000000LL; } - offset_diff= base_offset - lsn->rec_offset; + offset_diff= base_offset - LSN_OFFSET(lsn); if (diff > 0x3f) { /*TODO: error - too long transaction - panic!!! */ @@ -3222,7 +3252,7 @@ static uchar *translog_put_LSN_diff(LSN *base_lsn, LSN *lsn, uchar *dst) pointer to buffer after decoded LSN */ -static uchar *translog_get_LSN_from_diff(LSN *base_lsn, uchar *src, uchar *dst) +static uchar *translog_get_LSN_from_diff(LSN base_lsn, uchar *src, uchar *dst) { LSN lsn; uint32 diff; @@ -3230,42 +3260,39 @@ static uchar *translog_get_LSN_from_diff(LSN *base_lsn, uchar *src, uchar *dst) uint8 code; DBUG_ENTER("translog_get_LSN_from_diff"); DBUG_PRINT("enter", ("Base: (0x%lx,0x%lx), src: 0x%lx, dst 0x%lx", - (ulong) base_lsn->file_no, - (ulong) base_lsn->rec_offset, (ulong) src, (ulong) dst)); + (ulong) LSN_FILE_NO(base_lsn), + (ulong) LSN_OFFSET(base_lsn), + (ulong) src, (ulong) dst)); first_byte= *((uint8*) src); code= first_byte & 0xC0; first_byte &= 0x3F; switch (code) { case 0x00: - lsn.file_no= base_lsn->file_no; - lsn.rec_offset= - base_lsn->rec_offset - ((first_byte << 8) + *((uint8*) (src + 1))); + lsn= base_lsn - ((first_byte << 8) + *((uint8*) (src + 1))); src+= 2; break; case 0x40: - lsn.file_no= base_lsn->file_no; diff= uint2korr(src + 1); - lsn.rec_offset= base_lsn->rec_offset - ((first_byte << 16) + diff); + lsn= base_lsn - ((first_byte << 16) + diff); src+= 3; break; case 0x80: - lsn.file_no= base_lsn->file_no; diff= uint3korr(src + 1); - lsn.rec_offset= base_lsn->rec_offset - ((first_byte << 24) + diff); + lsn= base_lsn - ((first_byte << 24) + diff); src+= 4; break; case 0xC0: { - ulonglong base_offset= base_lsn->rec_offset; diff= uint4korr(src + 1); - if (diff > base_lsn->rec_offset) + ulonglong base_offset= LSN_OFFSET(base_lsn); + if (diff > LSN_OFFSET(base_lsn)) { /* take 1 from file offset */ first_byte++; base_offset+= 0x100000000LL; } - lsn.file_no= base_lsn->file_no - first_byte; - lsn.rec_offset= base_offset - diff; + lsn= MAKE_LSN(LSN_FILE_NO(base_lsn) - first_byte, + base_offset - diff); src+= 5; break; } @@ -3273,7 +3300,7 @@ static uchar *translog_get_LSN_from_diff(LSN *base_lsn, uchar *src, uchar *dst) DBUG_ASSERT(0); DBUG_RETURN(NULL); } - lsn7store(dst, &lsn); + lsn7store(dst, lsn); DBUG_PRINT("info", ("new src: 0x%lx", (ulong) dst)); DBUG_RETURN(src); } @@ -3295,7 +3322,7 @@ static uchar *translog_get_LSN_from_diff(LSN *base_lsn, uchar *src, uchar *dst) */ static my_bool translog_relative_LSN_encode(struct st_translog_parts *parts, - LSN *base_lsn, + LSN base_lsn, uint lsns, uchar *compressed_LSNs) { struct st_translog_part part; @@ -3345,8 +3372,8 @@ static my_bool translog_relative_LSN_encode(struct st_translog_parts *parts, uint i; for (i= 0; i < lsns; i++, ref_ptr-= 7) { - lsn7korr(&ref, ref_ptr); - if ((dst_ptr= translog_put_LSN_diff(base_lsn, &ref, dst_ptr)) == NULL) + ref= lsn7korr(ref_ptr); + if ((dst_ptr= translog_put_LSN_diff(base_lsn, ref, dst_ptr)) == NULL) DBUG_RETURN(1); } economy= (dst_ptr - part.buff); @@ -3462,14 +3489,15 @@ translog_write_variable_record_mgroup(LSN *lsn, DBUG_RETURN(1); } - DBUG_PRINT("info", ("chunk #%u first_page: %u (%u), full_pages: %u (%lu), " + DBUG_PRINT("info", ("chunk #%u first_page: %u (%u), full_pages: %lu (%lu), " "Left %lu", groups.elements, first_page, first_page - 1, - full_pages, + (ulong) full_pages, (ulong) full_pages * log_descriptor.page_capacity_chunk_2, - parts->record_length - (first_page - 1 + buffer_rest) - + (ulong)parts->record_length - + (first_page - 1 + buffer_rest) - done)); translog_advance_pointer(full_pages, 0); @@ -3493,12 +3521,12 @@ translog_write_variable_record_mgroup(LSN *lsn, translog_write_data_on_page(&horizon, &cursor, 1, chunk2_header); translog_write_parts_on_page(&horizon, &cursor, first_page - 1, parts); - DBUG_PRINT("info", ("absolute horizon (%u,0x%lx), local (%u,0x%lx) " + DBUG_PRINT("info", ("absolute horizon (%lu,0x%lx), local (%lu,0x%lx) " "Left: %lu", - (uint) log_descriptor.horizon.file_no, - (ulong) log_descriptor.horizon.rec_offset, - (uint) horizon.file_no, - (ulong) horizon.rec_offset, + (ulong) LSN_FILE_NO(log_descriptor.horizon), + (ulong) LSN_OFFSET(log_descriptor.horizon), + (ulong) LSN_FILE_NO(horizon), + (ulong) LSN_OFFSET(horizon), (ulong) (parts->record_length - (first_page - 1) - done))); @@ -3510,12 +3538,12 @@ translog_write_variable_record_mgroup(LSN *lsn, DBUG_RETURN(1); } - DBUG_PRINT("info", ("absolute horizon (%u,0x%lx), local (%u,0x%lx)" + DBUG_PRINT("info", ("absolute horizon (%lu,0x%lx), local (%lu,0x%lx)" "Left: %lu", - (uint) log_descriptor.horizon.file_no, - (ulong) log_descriptor.horizon.rec_offset, - (uint) horizon.file_no, - (ulong) horizon.rec_offset, + (ulong) LSN_FILE_NO(log_descriptor.horizon), + (ulong) LSN_OFFSET(log_descriptor.horizon), + (ulong) LSN_FILE_NO(horizon), + (ulong) LSN_OFFSET(horizon), (ulong) (parts->record_length - (first_page - 1) - i * log_descriptor.page_capacity_chunk_2 - done))); @@ -3559,8 +3587,7 @@ translog_write_variable_record_mgroup(LSN *lsn, group.addr= horizon= log_descriptor.horizon; cursor= log_descriptor.bc; cursor.chaser= 1; - group.num= 0; /* 0 because it does not matter - */ + group.num= 0; /* 0 because it does not matter */ if (insert_dynamic(&groups, (gptr) &group)) { delete_dynamic(&groups); @@ -3643,12 +3670,12 @@ translog_write_variable_record_mgroup(LSN *lsn, DBUG_PRINT("info", ("chunk 2 to finish first page")); translog_write_data_on_page(&horizon, &cursor, 1, chunk2_header); translog_write_parts_on_page(&horizon, &cursor, first_page - 1, parts); - DBUG_PRINT("info", ("absolute horizon (%u,0x%lx), local (%u,0x%lx) " + DBUG_PRINT("info", ("absolute horizon (%lu,0x%lx), local (%lu,0x%lx) " "Left: %lu", - (uint) log_descriptor.horizon.file_no, - (ulong) log_descriptor.horizon.rec_offset, - (uint) horizon.file_no, - (ulong) horizon.rec_offset, + (ulong) LSN_FILE_NO(log_descriptor.horizon), + (ulong) LSN_OFFSET(log_descriptor.horizon), + (ulong) LSN_FILE_NO(horizon), + (ulong) LSN_OFFSET(horizon), (ulong) (parts->record_length - (first_page - 1) - done))); } @@ -3661,12 +3688,12 @@ translog_write_variable_record_mgroup(LSN *lsn, int2store(chunk3_header + 1, chunk3_size); translog_write_data_on_page(&horizon, &cursor, 3, chunk3_header); translog_write_parts_on_page(&horizon, &cursor, chunk3_size, parts); - DBUG_PRINT("info", ("absolute horizon (%u,0x%lx), local (%u,0x%lx) " + DBUG_PRINT("info", ("absolute horizon (%lu,0x%lx), local (%lu,0x%lx) " "Left: %lu", - (uint) log_descriptor.horizon.file_no, - (ulong) log_descriptor.horizon.rec_offset, - (uint) horizon.file_no, - (ulong) horizon.rec_offset, + (ulong) LSN_FILE_NO(log_descriptor.horizon), + (ulong) LSN_OFFSET(log_descriptor.horizon), + (ulong) LSN_FILE_NO(horizon), + (ulong) LSN_OFFSET(horizon), (ulong) (parts->record_length - chunk3_size - done))); chunk3_pages= 0; } @@ -3685,12 +3712,12 @@ translog_write_variable_record_mgroup(LSN *lsn, DBUG_RETURN(1); } - DBUG_PRINT("info", ("absolute horizon (%u,0x%lx), local (%u,0x%lx) " + DBUG_PRINT("info", ("absolute horizon (%lu,0x%lx), local (%lu,0x%lx) " "Left: %lu", - (uint) log_descriptor.horizon.file_no, - (ulong) log_descriptor.horizon.rec_offset, - (uint) horizon.file_no, - (ulong) horizon.rec_offset, + (ulong) LSN_FILE_NO(log_descriptor.horizon), + (ulong) LSN_OFFSET(log_descriptor.horizon), + (ulong) LSN_FILE_NO(horizon), + (ulong) LSN_OFFSET(horizon), (ulong) (parts->record_length - (first_page - 1) - i * log_descriptor.page_capacity_chunk_2 - done))); @@ -3704,10 +3731,11 @@ translog_write_variable_record_mgroup(LSN *lsn, delete_dynamic(&groups); DBUG_RETURN(1); } - DBUG_PRINT("info", ("absolute horizon (%u,0x%lx), local (%u,0x%lx)", - (uint) log_descriptor.horizon.file_no, - (ulong) log_descriptor.horizon.rec_offset, - (uint) horizon.file_no, (ulong) horizon.rec_offset)); + DBUG_PRINT("info", ("absolute horizon (%lu,0x%lx), local (%lu,0x%lx)", + (ulong) LSN_FILE_NO(log_descriptor.horizon), + (ulong) LSN_OFFSET(log_descriptor.horizon), + (ulong) LSN_FILE_NO(horizon), + (ulong) LSN_OFFSET(horizon))); *chunk0_header= (uchar) (type |TRANSLOG_CHUNK_LSN); @@ -3783,7 +3811,7 @@ translog_write_variable_record_mgroup(LSN *lsn, for (i= curr_group; i < limit + curr_group; i++) { get_dynamic(&groups, (gptr) &group, i); - lsn7store(group_desc, &group.addr); + lsn7store(group_desc, group.addr); group_desc[7]= group.num; translog_write_data_on_page(&horizon, &cursor, (7 + 1), group_desc); } @@ -3840,9 +3868,9 @@ static my_bool translog_write_variable_record(LSN *lsn, DBUG_ENTER("translog_write_variable_record"); translog_lock(); - DBUG_PRINT("info", ("horizon (%u,0x%lx)", - (uint) log_descriptor.horizon.file_no, - (ulong) log_descriptor.horizon.rec_offset)); + DBUG_PRINT("info", ("horizon (%lu,0x%lx)", + (ulong) LSN_FILE_NO(log_descriptor.horizon), + (ulong) LSN_OFFSET(log_descriptor.horizon))); page_rest= TRANSLOG_PAGE_SIZE - log_descriptor.bc.current_page_size; DBUG_PRINT("info", ("header length %u, page_rest: %u", header_length1, page_rest)); @@ -3871,7 +3899,7 @@ static my_bool translog_write_variable_record(LSN *lsn, */ if (log_record_type_descriptor[type].compresed_LSN > 0) { - if (translog_relative_LSN_encode(parts, &log_descriptor.horizon, + if (translog_relative_LSN_encode(parts, log_descriptor.horizon, log_record_type_descriptor[type]. compresed_LSN, compressed_LSNs)) { @@ -3889,7 +3917,7 @@ static my_bool translog_write_variable_record(LSN *lsn, translog_variable_record_length_bytes(parts->record_length); DBUG_PRINT("info", ("after compressing LSN(s) header length %u, " "record length %lu", - header_length1, parts->record_length)); + header_length1, (ulong)parts->record_length)); } /* TODO: check space on current page for header + few bytes */ @@ -3960,9 +3988,9 @@ static my_bool translog_write_fixed_record(LSN *lsn, log_record_type_descriptor[type].fixed_length)); translog_lock(); - DBUG_PRINT("info", ("horizon (%u,0x%lx)", - (uint) log_descriptor.horizon.file_no, - (ulong) log_descriptor.horizon.rec_offset)); + DBUG_PRINT("info", ("horizon (%lu,0x%lx)", + (ulong) LSN_FILE_NO(log_descriptor.horizon), + (ulong) LSN_OFFSET(log_descriptor.horizon))); DBUG_ASSERT(log_descriptor.bc.current_page_size <= TRANSLOG_PAGE_SIZE); DBUG_PRINT("info", @@ -4002,7 +4030,7 @@ static my_bool translog_write_fixed_record(LSN *lsn, if (log_record_type_descriptor[type].class == LOGRECTYPE_PSEUDOFIXEDLENGTH) { DBUG_ASSERT(log_record_type_descriptor[type].compresed_LSN > 0); - if (translog_relative_LSN_encode(parts, lsn, + if (translog_relative_LSN_encode(parts, *lsn, log_record_type_descriptor[type]. compresed_LSN, compressed_LSNs)) { @@ -4185,7 +4213,7 @@ my_bool translog_write_record(LSN *lsn, position in sources after decoded LSN(s) */ -static uchar *translog_relative_LSN_decode(LSN *base_lsn, +static uchar *translog_relative_LSN_decode(LSN base_lsn, uchar *src, uchar *dst, uint lsns) { uint i; @@ -4231,7 +4259,7 @@ translog_size_t translog_fixed_length_header(uchar *page, if (desc->class == LOGRECTYPE_PSEUDOFIXEDLENGTH) { DBUG_ASSERT(lsns > 0); - src= translog_relative_LSN_decode(&buff->lsn, src, dst, lsns); + src= translog_relative_LSN_decode(buff->lsn, src, dst, lsns); lsns*= 7; dst+= lsns; length-= lsns; @@ -4321,7 +4349,7 @@ static my_bool translog_scanner_set_last_page(struct st_translog_scanner_data 0 - OK 1 - Error */ -static my_bool translog_init_scanner(LSN *lsn, +static my_bool translog_init_scanner(LSN lsn, my_bool fixed_horizon, struct st_translog_scanner_data *scanner) { @@ -4331,25 +4359,24 @@ static my_bool translog_init_scanner(LSN *lsn, }; DBUG_ENTER("translog_init_scanner"); - DBUG_PRINT("enter", ("LSN: (0x%lx,0x%lx)", - (ulong) lsn->file_no, (ulong) lsn->rec_offset)); - DBUG_ASSERT(lsn->rec_offset % TRANSLOG_PAGE_SIZE != 0); - scanner->page_offset= lsn->rec_offset % TRANSLOG_PAGE_SIZE; + DBUG_PRINT("enter", ("LSN: (0x%lu,0x%lx)", + (ulong) LSN_FILE_NO(lsn), + (ulong) LSN_OFFSET(lsn)); + DBUG_ASSERT(LSN_OFFSET(lsn) % TRANSLOG_PAGE_SIZE != 0); + scanner->page_offset= LSN_OFFSET(lsn) % TRANSLOG_PAGE_SIZE; scanner->fixed_horizon= fixed_horizon; translog_scanner_set_horizon(scanner); - DBUG_PRINT("info", ("Horizon: (0x%lx,0x%lx)", - (ulong) scanner->horizon.file_no, - (ulong) scanner->horizon.rec_offset)); + DBUG_PRINT("info", ("Horizon: (0x%lu,0x%lx)", + (ulong) LSN_FILE_NO(scanner->horizon), + (ulong) LSN_OFFSET(scanner->horizon))); /* lsn < horizon */ - DBUG_ASSERT(lsn->file_no < scanner->horizon.file_no || - (lsn->file_no == scanner->horizon.file_no && - lsn->rec_offset < scanner->horizon.rec_offset)); + DBUG_ASSERT(lsn < scanner->horizon)); - scanner->page_addr= *lsn; - scanner->page_addr.rec_offset-= scanner->page_offset; + scanner->page_addr= lsn; + scanner->page_addr-= scanner->page_offset; /*decrease offset */ if (translog_scanner_set_last_page(scanner)) DBUG_RETURN(1); @@ -4376,16 +4403,14 @@ static my_bool translog_scanner_eol(struct st_translog_scanner_data *scanner) DBUG_ENTER("translog_scanner_eol"); DBUG_PRINT("enter", ("Horizon: (%lu, 0x%lx), Current: (%lu, 0x%lx+0x%x=0x%lx)", - (ulong) scanner->horizon.file_no, - (ulong) scanner->horizon.rec_offset, - (ulong) scanner->page_addr.file_no, - (ulong) scanner->page_addr.rec_offset, + (ulong) LSN_FILE_NO(scanner->horizon), + (ulong) LSN_OFFSET(scanner->horizon), + (ulong) LSN_FILE_NO(scanner->page_addr), + (ulong) LSN_OFFSET(scanner->page_addr), (uint) scanner->page_offset, - (ulong) (scanner->page_addr.rec_offset + scanner->page_offset))); - if (scanner->horizon.file_no > scanner->page_addr.file_no || - (scanner->horizon.file_no == scanner->page_addr.file_no && - scanner->horizon.rec_offset > (scanner->page_addr.rec_offset + - scanner->page_offset))) + (ulong) (LSN_OFFSET(scanner->page_addr) + scanner->page_offset))); + if (scanner->horizon > (scanner->page_addr + + scanner->page_offset)) { DBUG_PRINT("info", ("Horizon is not reached")); DBUG_RETURN(0); @@ -4398,14 +4423,10 @@ static my_bool translog_scanner_eol(struct st_translog_scanner_data *scanner) translog_scanner_set_horizon(scanner); DBUG_PRINT("info", ("Horizon is re-read, EOL: %d", - scanner->horizon.file_no <= scanner->page_addr.file_no && - (scanner->horizon.file_no != scanner->page_addr.file_no || - scanner->horizon.rec_offset <= (scanner->page_addr.rec_offset + - scanner->page_offset)))); - DBUG_RETURN(scanner->horizon.file_no <= scanner->page_addr.file_no && - (scanner->horizon.file_no != scanner->page_addr.file_no || - scanner->horizon.rec_offset <= (scanner->page_addr.rec_offset + - scanner->page_offset))); + scanner->horizon <= (scanner->page_addr + + scanner->page_offset))); + DBUG_RETURN(scanner->horizon <= (scanner->page_addr + + scanner->page_offset)); } @@ -4443,19 +4464,20 @@ static my_bool translog_scanner_eop(struct st_translog_scanner_data *scanner) static my_bool translog_scanner_eof(struct st_translog_scanner_data *scanner) { DBUG_ENTER("translog_scanner_eof"); - DBUG_ASSERT(scanner->page_addr.file_no == scanner->last_file_page.file_no); + DBUG_ASSERT(LSN_FILE_NO(scanner->page_addr) == + LSN_FILE_NO(scanner->last_file_page)); DBUG_PRINT("enter", ("curr Page 0x%lx, last page 0x%lx, " "normal EOF %d", - scanner->page_addr.rec_offset, - scanner->last_file_page.rec_offset, - scanner->page_addr.rec_offset == - scanner->last_file_page.rec_offset)); + (ulong) LSN_OFFSET(scanner->page_addr), + (ulong) LSN_OFFSET(scanner->last_file_page), + LSN_OFFSET(scanner->page_addr) == + LSN_OFFSET(scanner->last_file_page))); /* TODO: detect damaged file EOF, TODO: issue warning if damaged file EOF detected */ - DBUG_RETURN(scanner->page_addr.rec_offset == - scanner->last_file_page.rec_offset); + DBUG_RETURN(scanner->page_addr == + scanner->last_file_page); } @@ -4491,20 +4513,22 @@ static my_bool translog_get_next_chunk(struct st_translog_scanner_data *scanner) if (translog_scanner_eof(scanner)) { DBUG_PRINT("info", ("horizon (%lu,0x%lx) pageaddr (%lu,0x%lx)", - (ulong) scanner->horizon.file_no, - (ulong) scanner->horizon.rec_offset, - (ulong) scanner->page_addr.file_no, - (ulong) scanner->page_addr.rec_offset)); + (ulong) LSN_FILE_NO(scanner->horizon), + (ulong) LSN_OFFSET(scanner->horizon), + (ulong) LSN_FILE_NO(scanner->page_addr), + (ulong) LSN_OFFSET(scanner->page_addr))); /* if it is log end it have to be caught before */ - DBUG_ASSERT(scanner->horizon.file_no > scanner->page_addr.file_no); - scanner->page_addr.file_no++; - scanner->page_addr.rec_offset= TRANSLOG_PAGE_SIZE; + DBUG_ASSERT(LSN_FILE_NO(scanner->horizon) > + LSN_FILE_NO(scanner->page_addr)); + scanner->page_addr+= LSN_ONE_FILE; + scanner->page_addr= LSN_REPLACE_OFFSET(scanner->page_addr, + TRANSLOG_PAGE_SIZE); if (translog_scanner_set_last_page(scanner)) DBUG_RETURN(1); } else { - scanner->page_addr.rec_offset+= TRANSLOG_PAGE_SIZE; + scanner->page_addr+= TRANSLOG_PAGE_SIZE; /* offset increased */ } { TRANSLOG_VALIDATOR_DATA data= @@ -4614,12 +4638,12 @@ translog_size_t translog_variable_length_header(uchar *page, for (i= 0; i < read; i++, curr++) { DBUG_ASSERT(curr < buff->groups_no); - lsn7korr(&buff->groups[curr].addr, src + i * (7 + 1)); + buff->groups[curr].addr= lsn7korr(src + i * (7 + 1)); buff->groups[curr].num= src[i * (7 + 1) + 7]; - DBUG_PRINT("info", ("group #%u (%u,0x%lx) chunks %u", + DBUG_PRINT("info", ("group #%u (%lu,0x%lx) chunks %u", curr, - (uint) buff->groups[curr].addr.file_no, - (ulong) buff->groups[curr].addr.rec_offset, + (ulong) LSN_FILE_NO(buff->groups[curr].addr), + (ulong) LSN_OFFSET(buff->groups[curr].addr), (uint) buff->groups[curr].num)); } grp_no-= read; @@ -4628,18 +4652,19 @@ translog_size_t translog_variable_length_header(uchar *page, if (scanner) { buff->chunk0_data_addr= scanner->page_addr; - buff->chunk0_data_addr.rec_offset+= (page_offset + header_to_skip + - i * (7 + 1)); + buff->chunk0_data_addr+= (page_offset + header_to_skip + + i * (7 + 1)); /* offset increased */ } else { buff->chunk0_data_addr= buff->lsn; - buff->chunk0_data_addr.rec_offset+= (header_to_skip + i * (7 + 1)); + /* offset increased */ + buff->chunk0_data_addr+= (header_to_skip + i * (7 + 1)); } buff->chunk0_data_len= chunk_len - 2 - i * (7 + 1); - DBUG_PRINT("info", ("Data address (%u,0x%lx), len: %u", - (uint) buff->chunk0_data_addr.file_no, - (ulong) buff->chunk0_data_addr.rec_offset, + DBUG_PRINT("info", ("Data address (%lu,0x%lx), len: %u", + (ulong) LSN_FILE_NO(buff->chunk0_data_addr), + (ulong) LSN_OFFSET(buff->chunk0_data_addr), buff->chunk0_data_len)); break; } @@ -4647,7 +4672,7 @@ translog_size_t translog_variable_length_header(uchar *page, { DBUG_PRINT("info", ("use internal scanner for header reding")); scanner= &internal_scanner; - translog_init_scanner(&buff->lsn, 1, scanner); + translog_init_scanner(buff->lsn, 1, scanner); } translog_get_next_chunk(scanner); page= scanner->page; @@ -4665,7 +4690,7 @@ translog_size_t translog_variable_length_header(uchar *page, } base_lsn= buff->groups[0].addr; - translog_init_scanner(&base_lsn, 1, scanner); + translog_init_scanner(base_lsn, 1, scanner); /* first group chunk is always chunk type 2 */ page= scanner->page; page_offset= scanner->page_offset; @@ -4676,7 +4701,7 @@ translog_size_t translog_variable_length_header(uchar *page, if (lsns) { uchar *start= src; - src= translog_relative_LSN_decode(&base_lsn, src, dst, lsns); + src= translog_relative_LSN_decode(base_lsn, src, dst, lsns); lsns*= 7; dst+= lsns; length-= lsns; @@ -4730,9 +4755,10 @@ translog_read_record_header_from_buffer(uchar *page, TRANSLOG_CHUNK_FIXED); buff->type= (page[page_offset] & TRANSLOG_REC_TYPE); buff->short_trid= uint2korr(page + page_offset + 1); - DBUG_PRINT("info", ("Type %u, Sort TrID %u, LSN (%u,0x%lx)", + DBUG_PRINT("info", ("Type %u, Sort TrID %u, LSN (%lu,0x%lx)", (uint) buff->type, (uint)buff->short_trid, - buff->lsn.file_no, buff->lsn.rec_offset)); + (ulong) LSN_FILE_NO(buff->lsn), + (ulong) LSN_OFFSET(buff->lsn))); /* Read required bytes from the header and call hook */ switch (log_record_type_descriptor[buff->type].class) { @@ -4773,26 +4799,26 @@ translog_read_record_header_from_buffer(uchar *page, part of the header */ -translog_size_t translog_read_record_header(LSN *lsn, +translog_size_t translog_read_record_header(LSN lsn, TRANSLOG_HEADER_BUFFER *buff) { uchar buffer[TRANSLOG_PAGE_SIZE], *page; - translog_size_t page_offset= lsn->rec_offset % TRANSLOG_PAGE_SIZE; + translog_size_t page_offset= LSN_OFFSET(lsn) % TRANSLOG_PAGE_SIZE; DBUG_ENTER("translog_read_record_header"); - DBUG_PRINT("enter", ("LSN: (0x%lx,0x%lx)", - (ulong) lsn->file_no, (ulong) lsn->rec_offset)); - DBUG_ASSERT(lsn->rec_offset % TRANSLOG_PAGE_SIZE != 0); + DBUG_PRINT("enter", ("LSN: (0x%lu,0x%lx)", + (ulong) LSN_FILE_NO(lsn), (ulong) LSN_OFFSET(lsn))); + DBUG_ASSERT(LSN_OFFSET(lsn) % TRANSLOG_PAGE_SIZE != 0); - buff->lsn= *lsn; + buff->lsn= lsn; buff->groups_no= 0; { - TRANSLOG_ADDRESS addr= *lsn; + TRANSLOG_ADDRESS addr= lsn; TRANSLOG_VALIDATOR_DATA data= { &addr, 0 }; - addr.rec_offset-= page_offset; + addr-= page_offset; /* offset decreasing */ if ((page= translog_get_page(&data, buffer)) == NULL) DBUG_RETURN(0); } @@ -4832,19 +4858,19 @@ translog_read_record_header_scan(struct st_translog_scanner_data my_bool move_scanner) { DBUG_ENTER("translog_read_record_header_scan"); - DBUG_PRINT("enter", ("Scanner: Cur: (%u, 0x%lx), Hrz: (%u, 0x%lx), " - "Lst: (%u, 0x%lx), Offset: %u(%x), fixed %d", - (uint) scanner->page_addr.file_no, - (ulong) scanner->page_addr.rec_offset, - (uint) scanner->horizon.file_no, - (ulong) scanner->horizon.rec_offset, - (uint) scanner->last_file_page.file_no, - (ulong) scanner->last_file_page.rec_offset, + DBUG_PRINT("enter", ("Scanner: Cur: (%lu,0x%lx), Hrz: (%lu,0x%lx), " + "Lst: (%lu,0x%lx), Offset: %u(%x), fixed %d", + (ulong) LSN_FILE_NO(scanner->page_addr), + (ulong) LSN_OFFSET(scanner->page_addr), + (ulong) LSN_FILE_NO(scanner->horizon), + (ulong) LSN_OFFSET(scanner->horizon), + (ulong) LSN_FILE_NO(scanner->last_file_page), + (ulong) LSN_OFFSET(scanner->last_file_page), (uint) scanner->page_offset, (uint) scanner->page_offset, scanner->fixed_horizon)); buff->groups_no= 0; buff->lsn= scanner->page_addr; - buff->lsn.rec_offset+= scanner->page_offset; + buff->lsn+= scanner->page_offset; /* offset increasing */ DBUG_RETURN(translog_read_record_header_from_buffer(scanner->page, scanner->page_offset, buff, @@ -4884,7 +4910,8 @@ translog_read_record_header_scan(struct st_translog_scanner_data number of bytes in TRANSLOG_HEADER_BUFFER::header where stored decoded part of the header */ -translog_size_t translog_read_next_record_header(LSN *lsn, + +translog_size_t translog_read_next_record_header(LSN lsn, TRANSLOG_HEADER_BUFFER *buff, my_bool fixed_horizon, struct @@ -4894,30 +4921,29 @@ translog_size_t translog_read_next_record_header(LSN *lsn, struct st_translog_scanner_data internal_scanner; uint8 chunk_type; - buff->groups_no= 0; /* to be sure that we will free - it right */ + buff->groups_no= 0; /* to be sure that we will free it right */ DBUG_ENTER("translog_read_next_record_header"); DBUG_PRINT("enter", ("scanner: 0x%lx", (ulong) scanner)); if (scanner == NULL) { - DBUG_ASSERT(lsn != NULL); + DBUG_ASSERT(lsn != CONTROL_FILE_IMPOSSIBLE_LSN); scanner= &internal_scanner; } if (lsn) { if (translog_init_scanner(lsn, fixed_horizon, scanner)) DBUG_RETURN(0); - DBUG_ASSERT(lsn->rec_offset % TRANSLOG_PAGE_SIZE != 0); + DBUG_ASSERT(LSN_OFFSET(lsn) % TRANSLOG_PAGE_SIZE != 0); } - DBUG_PRINT("info", ("Scanner: Cur: (%u, 0x%lx), Hrz: (%u, 0x%lx), " - "Lst: (%u, 0x%lx), Offset: %u(%x), fixed %d", - (uint) scanner->page_addr.file_no, - (ulong) scanner->page_addr.rec_offset, - (uint) scanner->horizon.file_no, - (ulong) scanner->horizon.rec_offset, - (uint) scanner->last_file_page.file_no, - (ulong) scanner->last_file_page.rec_offset, + DBUG_PRINT("info", ("Scanner: Cur: (%lu,0x%lx), Hrz: (%lu,0x%lx), " + "Lst: (%lu,0x%lx), Offset: %u(%x), fixed %d", + (ulong) LSN_FILE_NO(scanner->page_addr), + (ulong) LSN_OFFSET(scanner->page_addr), + (ulong) LSN_FILE_NO(scanner->horizon), + (ulong) LSN_OFFSET(scanner->horizon), + (ulong) LSN_FILE_NO(scanner->last_file_page), + (ulong) LSN_OFFSET(scanner->last_file_page), (uint) scanner->page_offset, (uint) scanner->page_offset, scanner->fixed_horizon)); @@ -4934,10 +4960,8 @@ translog_size_t translog_read_next_record_header(LSN *lsn, if (scanner->page[scanner->page_offset] == 0) { /* Last record was read */ - buff->lsn.file_no= CONTROL_FILE_IMPOSSIBLE_FILENO; - buff->lsn.rec_offset= 0; - DBUG_RETURN(TRANSLOG_RECORD_HEADER_MAX_SIZE + 1); /* just it is not error - */ + buff->lsn= CONTROL_FILE_IMPOSSIBLE_LSN; + DBUG_RETURN(TRANSLOG_RECORD_HEADER_MAX_SIZE + 1); /* just it is not error */ } DBUG_RETURN(translog_read_record_header_scan(scanner, buff, 0)); } @@ -4955,6 +4979,7 @@ translog_size_t translog_read_next_record_header(LSN *lsn, 0 - OK 1 - Error */ + static my_bool translog_record_read_next_chunk(struct st_translog_reader_data *data) { @@ -4978,7 +5003,7 @@ static my_bool translog_record_read_next_chunk(struct st_translog_reader_data data->current_group++; data->current_chunk= 0; DBUG_PRINT("info", ("skip to group #%u", data->current_group)); - translog_init_scanner(&data->header.groups[data->current_group].addr, + translog_init_scanner(data->header.groups[data->current_group].addr, 1, &data->scanner); } else @@ -4996,10 +5021,9 @@ static my_bool translog_record_read_next_chunk(struct st_translog_reader_data data->header.chunk0_data_len, data->scanner.page_offset, data->current_group, data->header.groups_no - 1)); DBUG_ASSERT(data->header.groups_no - 1 == data->current_group); - DBUG_ASSERT(data->header.lsn.file_no == data->scanner.page_addr.file_no && - data->header.lsn.rec_offset == - data->scanner.page_addr.rec_offset + data->scanner.page_offset); - translog_init_scanner(&data->header.chunk0_data_addr, 1, &data->scanner); + DBUG_ASSERT(data->header.lsn == + data->scanner.page_addr + data->scanner.page_offset); + translog_init_scanner(data->header.chunk0_data_addr, 1, &data->scanner); data->chunk_size= data->header.chunk0_data_len; data->body_offset= data->scanner.page_offset; data->current_offset= new_current_offset; @@ -5044,7 +5068,7 @@ static my_bool translog_record_read_next_chunk(struct st_translog_reader_data 1 - Error */ -static my_bool translog_init_reader_data(LSN *lsn, +static my_bool translog_init_reader_data(LSN lsn, struct st_translog_reader_data *data) { DBUG_ENTER("translog_init_reader_data"); @@ -5085,7 +5109,7 @@ static my_bool translog_init_reader_data(LSN *lsn, length of data actually read */ -translog_size_t translog_read_record(LSN *lsn, +translog_size_t translog_read_record(LSN lsn, translog_size_t offset, translog_size_t length, uchar *buffer, @@ -5099,7 +5123,7 @@ translog_size_t translog_read_record(LSN *lsn, if (data == NULL) { - DBUG_ASSERT(lsn != NULL); + DBUG_ASSERT(lsn != CONTROL_FILE_IMPOSSIBLE_LSN); data= &internal_data; } if (lsn || @@ -5110,15 +5134,15 @@ translog_size_t translog_read_record(LSN *lsn, DBUG_RETURN(0); } DBUG_PRINT("info", ("Offset %lu, length %lu " - "Scanner: Cur: (%u, 0x%lx), Hrz: (%u, 0x%lx), " - "Lst: (%u, 0x%lx), Offset: %u(%x), fixed %d", + "Scanner: Cur: (%lu,0x%lx), Hrz: (%lu,0x%lx), " + "Lst: (%lu,0x%lx), Offset: %u(%x), fixed %d", (ulong) offset, (ulong) length, - (uint) data->scanner.page_addr.file_no, - (ulong) data->scanner.page_addr.rec_offset, - (uint) data->scanner.horizon.file_no, - (ulong) data->scanner.horizon.rec_offset, - (uint) data->scanner.last_file_page.file_no, - (ulong) data->scanner.last_file_page.rec_offset, + (ulong) LSN_FILE_NO(data->scanner.page_addr), + (ulong) LSN_OFFSET(data->scanner.page_addr), + (ulong) LSN_FILE_NO(data->scanner.horizon), + (ulong) LSN_OFFSET(data->scanner.horizon), + (ulong) LSN_FILE_NO(data->scanner.last_file_page), + (ulong) LSN_OFFSET(data->scanner.last_file_page), (uint) data->scanner.page_offset, (uint) data->scanner.page_offset, data->scanner.fixed_horizon)); @@ -5193,7 +5217,7 @@ static void translog_force_current_buffer_to_finish() uint16 current_page_size; new_buff_begunning= log_descriptor.bc.buffer->offset; - new_buff_begunning.rec_offset+= log_descriptor.bc.buffer->size; + new_buff_begunning+= log_descriptor.bc.buffer->size; /* increase offset */ DBUG_ENTER("translog_force_current_buffer_to_finish"); DBUG_PRINT("enter", ("Buffer #%u 0x%lx, " @@ -5203,13 +5227,13 @@ static void translog_force_current_buffer_to_finish() "size %lu (%lu), Pg: %u, left: %u", (uint) log_descriptor.bc.buffer_no, (ulong) log_descriptor.bc.buffer, - (ulong) log_descriptor.bc.buffer->offset.file_no, - (ulong) log_descriptor.bc.buffer->offset.rec_offset, - (ulong) log_descriptor.horizon.file_no, - (ulong) (log_descriptor.horizon.rec_offset - + (ulong) LSN_FILE_NO(log_descriptor.bc.buffer->offset), + (ulong) LSN_OFFSET(log_descriptor.bc.buffer->offset), + (ulong) LSN_FILE_NO(log_descriptor.horizon), + (ulong) (LSN_OFFSET(log_descriptor.horizon) - log_descriptor.bc.current_page_size), - (ulong) new_buff_begunning.file_no, - (ulong) new_buff_begunning.rec_offset, + (ulong) LSN_FILE_NO(new_buff_begunning), + (ulong) LSN_OFFSET(new_buff_begunning), (ulong) log_descriptor.bc.buffer->size, (ulong) (log_descriptor.bc.ptr -log_descriptor.bc. buffer->buffer), @@ -5219,11 +5243,11 @@ static void translog_force_current_buffer_to_finish() DBUG_ASSERT((log_descriptor.bc.ptr -log_descriptor.bc.buffer->buffer) %TRANSLOG_PAGE_SIZE == log_descriptor.bc.current_page_size % TRANSLOG_PAGE_SIZE); - DBUG_ASSERT(log_descriptor.horizon.file_no == - log_descriptor.bc.buffer->offset.file_no); - DBUG_ASSERT(log_descriptor.bc.buffer->offset.rec_offset + + DBUG_ASSERT(LSN_FILE_NO(log_descriptor.horizon) == + LSN_FILE_NO(log_descriptor.bc.buffer->offset)); + DBUG_ASSERT(LSN_OFFSET(log_descriptor.bc.buffer->offset) + (log_descriptor.bc.ptr -log_descriptor.bc.buffer->buffer) == - log_descriptor.horizon.rec_offset); + LSN_OFFSET(log_descriptor.horizon)); if (left != TRANSLOG_PAGE_SIZE && left != 0) { /* @@ -5232,7 +5256,8 @@ static void translog_force_current_buffer_to_finish() */ DBUG_PRINT("info", ("left %u", (uint) left)); - new_buff_begunning.rec_offset-= log_descriptor.bc.current_page_size; + /* decrease offset */ + new_buff_begunning-= log_descriptor.bc.current_page_size; current_page_size= log_descriptor.bc.current_page_size; bzero(log_descriptor.bc.ptr, left); @@ -5315,7 +5340,7 @@ static void translog_force_current_buffer_to_finish() 1 - Error */ -my_bool translog_flush(LSN *lsn) +my_bool translog_flush(LSN lsn) { LSN old_flushed, sent_to_file; int rc= 0; @@ -5323,8 +5348,9 @@ my_bool translog_flush(LSN *lsn) my_bool full_circle= 0; DBUG_ENTER("translog_flush"); - DBUG_PRINT("enter", ("Flush up to LSN (%u,0x%lx)", - (uint) lsn->file_no, (ulong) lsn->rec_offset)); + DBUG_PRINT("enter", ("Flush up to LSN (%lu,0x%lx)", + (ulong) LSN_FILE_NO(lsn), + (ulong) LSN_OFFSET(lsn))); translog_lock(); old_flushed= log_descriptor.flushed; @@ -5336,18 +5362,18 @@ my_bool translog_flush(LSN *lsn) struct st_translog_buffer *buffer= log_descriptor.bc.buffer; /* we can't flush in future */ - DBUG_ASSERT(cmp_translog_addr(log_descriptor.horizon, *lsn) >= 0); - if (cmp_translog_addr(log_descriptor.flushed, *lsn) >= 0) + DBUG_ASSERT(cmp_translog_addr(log_descriptor.horizon, lsn) >= 0); + if (cmp_translog_addr(log_descriptor.flushed, lsn) >= 0) { - DBUG_PRINT("info", ("already flushed (%u,0x%lx)", - (uint) log_descriptor.flushed.file_no, - (ulong) log_descriptor.flushed.rec_offset)); + DBUG_PRINT("info", ("already flushed (%lu,0x%lx)", + (ulong) LSN_FILE_NO(log_descriptor.flushed), + (ulong) LSN_OFFSET(log_descriptor.flushed))); translog_unlock(); DBUG_RETURN(0); } /* send to the file if it is not sent */ translog_get_sent_to_file(&sent_to_file); - if (cmp_translog_addr(sent_to_file, *lsn) >= 0) + if (cmp_translog_addr(sent_to_file, lsn) >= 0) break; do @@ -5369,7 +5395,7 @@ my_bool translog_flush(LSN *lsn) break; } } while ((buffer_start != buffer_no) && - cmp_translog_addr(log_descriptor.flushed, *lsn) < 0); + cmp_translog_addr(log_descriptor.flushed, lsn) < 0); if (buffer_unlock != NULL) translog_buffer_unlock(buffer_unlock); if (translog_buffer_flush(buffer)) @@ -5382,12 +5408,13 @@ my_bool translog_flush(LSN *lsn) translog_lock(); } - for (i= old_flushed.file_no; i <= lsn->file_no; i++) + for (i= LSN_FILE_NO(old_flushed); i <= LSN_FILE_NO(lsn); i++) { uint cache_index; File file; - if ((cache_index= log_descriptor.horizon.file_no - i) < OPENED_FILES_NUM) + if ((cache_index= LSN_FILE_NO(log_descriptor.horizon) - i) < + OPENED_FILES_NUM) { /* file in the cache */ if (log_descriptor.log_file_num[cache_index] == 0) diff --git a/storage/maria/ma_loghandler.h b/storage/maria/ma_loghandler.h index f4d939786fc..02c272361a4 100644 --- a/storage/maria/ma_loghandler.h +++ b/storage/maria/ma_loghandler.h @@ -218,7 +218,7 @@ void translog_destroy(); part of the header */ -translog_size_t translog_read_record_header(LSN *lsn, +translog_size_t translog_read_record_header(LSN lsn, TRANSLOG_HEADER_BUFFER *buff); @@ -249,7 +249,7 @@ void translog_free_record_header(TRANSLOG_HEADER_BUFFER *buff); length of data actually read */ -translog_size_t translog_read_record(LSN *lsn, +translog_size_t translog_read_record(LSN lsn, translog_size_t offset, translog_size_t length, uchar *buffer, @@ -269,7 +269,7 @@ translog_size_t translog_read_record(LSN *lsn, 1 - Error */ -my_bool translog_flush(LSN *lsn); +my_bool translog_flush(LSN lsn); /* @@ -304,7 +304,7 @@ my_bool translog_flush(LSN *lsn); part of the header */ -translog_size_t translog_read_next_record_header(LSN *lsn, +translog_size_t translog_read_next_record_header(LSN lsn, TRANSLOG_HEADER_BUFFER *buff, my_bool fixed_horizon, struct diff --git a/storage/maria/ma_loghandler_lsn.h b/storage/maria/ma_loghandler_lsn.h index 9576d4d734d..472298de07c 100644 --- a/storage/maria/ma_loghandler_lsn.h +++ b/storage/maria/ma_loghandler_lsn.h @@ -2,11 +2,7 @@ #define _ma_loghandler_lsn_h /* Transaction log record address (file_no is int24 on the disk) */ -typedef struct st_translog_address -{ - uint32 file_no; - uint32 rec_offset; -} TRANSLOG_ADDRESS; +typedef int64 TRANSLOG_ADDRESS; /* Compare addresses @@ -14,26 +10,39 @@ typedef struct st_translog_address A1 == A2 -> 0 A1 < A2 -> result < 0 */ -#define cmp_translog_addr(A1,A2) \ - ((A1).file_no == (A2).file_no ? \ - ((int64)(A1).rec_offset) - (int64)(A2).rec_offset : \ - ((int64)(A1).file_no - (int64)(A2).file_no)) +#define cmp_translog_addr(A1,A2) ((A1) - (A2)) /* LSN type (address of certain log record chank */ typedef TRANSLOG_ADDRESS LSN; +/* Gets file number part of a LSN/log address */ +#define LSN_FILE_NO(L) ((L) >> 32) + +/* Gets raw file number part of a LSN/log address */ +#define LSN_FINE_NO_PART(L) ((L) & ((int64)0xFFFFFF00000000LL)) + +/* Gets record offset of a LSN/log address */ +#define LSN_OFFSET(L) ((L) & 0xFFFFFFFFL) + +/* Makes lsn/log address from file number and record offset */ +#define MAKE_LSN(F,S) ((((uint64)(F)) << 32) | (S)) + +/* checks LSN */ +#define LSN_VALID(L) DBUG_ASSERT((L) >= 0 && (L) < (uint64)0xFFFFFFFFFFFFFFLL) + /* Puts LSN into buffer (dst) */ #define lsn7store(dst, lsn) \ do { \ - int3store((dst), (lsn)->file_no); \ - int4store((dst) + 3, (lsn)->rec_offset); \ + int3store((dst), LSN_FILE_NO(lsn)); \ + int4store((dst) + 3, LSN_OFFSET(lsn)); \ } while (0) /* Unpacks LSN from the buffer (P) */ -#define lsn7korr(lsn, P) \ - do { \ - (lsn)->file_no= uint3korr(P); \ - (lsn)->rec_offset= uint4korr((P) + 3); \ - } while (0) +#define lsn7korr(P) MAKE_LSN(uint3korr(P), uint4korr((P) + 3)) + +/* what we need to add to LSN to increase it on one file */ +#define LSN_ONE_FILE ((int64)0x100000000LL) + +#define LSN_REPLACE_OFFSET(L, S) (LSN_FINE_NO_PART(L) | (S)) #endif diff --git a/storage/maria/unittest/Makefile.am b/storage/maria/unittest/Makefile.am index 92988b088ab..74d9059ca7d 100644 --- a/storage/maria/unittest/Makefile.am +++ b/storage/maria/unittest/Makefile.am @@ -30,14 +30,15 @@ LDADD= $(top_builddir)/unittest/mytap/libmytap.a \ $(top_builddir)/storage/maria/ma_loghandler.o noinst_PROGRAMS = ma_control_file-t trnman-t lockman2-t \ mf_pagecache_single_1k-t mf_pagecache_single_8k-t \ - mf_pagecache_single_64k-t \ - mf_pagecache_consist_1k-t mf_pagecache_consist_64k-t \ - mf_pagecache_consist_1kHC-t \ - mf_pagecache_consist_64kHC-t \ - mf_pagecache_consist_1kRD-t \ - mf_pagecache_consist_64kRD-t \ - mf_pagecache_consist_1kWR-t \ - mf_pagecache_consist_64kWR-t \ + mf_pagecache_single_64k-t-big \ + mf_pagecache_consist_1k-t-big \ + mf_pagecache_consist_64k-t-big \ + mf_pagecache_consist_1kHC-t-big \ + mf_pagecache_consist_64kHC-t-big \ + mf_pagecache_consist_1kRD-t-big \ + mf_pagecache_consist_64kRD-t-big \ + mf_pagecache_consist_1kWR-t-big \ + mf_pagecache_consist_64kWR-t-big \ ma_test_loghandler-t \ ma_test_loghandler_multigroup-t \ ma_test_loghandler_multithread-t \ @@ -49,34 +50,33 @@ mf_pagecache_common_cppflags = -DEXTRA_DEBUG -DPAGECACHE_DEBUG -DMAIN mf_pagecache_single_1k_t_SOURCES = $(mf_pagecache_single_src) mf_pagecache_single_8k_t_SOURCES = $(mf_pagecache_single_src) -mf_pagecache_single_64k_t_SOURCES = $(mf_pagecache_single_src) +mf_pagecache_single_64k_t_big_SOURCES = $(mf_pagecache_single_src) mf_pagecache_single_1k_t_CPPFLAGS = $(mf_pagecache_common_cppflags) -DPAGE_SIZE=1024 mf_pagecache_single_8k_t_CPPFLAGS = $(mf_pagecache_common_cppflags) -DPAGE_SIZE=8192 -mf_pagecache_single_64k_t_CPPFLAGS = $(mf_pagecache_common_cppflags) -DPAGE_SIZE=65536 +mf_pagecache_single_64k_t_big_CPPFLAGS = $(mf_pagecache_common_cppflags) -DPAGE_SIZE=65536 -mf_pagecache_consist_1k_t_SOURCES = $(mf_pagecache_consist_src) -mf_pagecache_consist_1k_t_CPPFLAGS = $(mf_pagecache_common_cppflags) -DPAGE_SIZE=1024 -mf_pagecache_consist_64k_t_SOURCES = $(mf_pagecache_consist_src) -mf_pagecache_consist_64k_t_CPPFLAGS = $(mf_pagecache_common_cppflags) -DPAGE_SIZE=65536 +mf_pagecache_consist_1k_t_big_SOURCES = $(mf_pagecache_consist_src) +mf_pagecache_consist_1k_t_big_CPPFLAGS = $(mf_pagecache_common_cppflags) -DPAGE_SIZE=1024 +mf_pagecache_consist_64k_t_big_SOURCES = $(mf_pagecache_consist_src) +mf_pagecache_consist_64k_t_big_CPPFLAGS = $(mf_pagecache_common_cppflags) -DPAGE_SIZE=65536 -mf_pagecache_consist_1kHC_t_SOURCES = $(mf_pagecache_consist_src) -mf_pagecache_consist_1kHC_t_CPPFLAGS = $(mf_pagecache_common_cppflags) -DPAGE_SIZE=1024 -DTEST_HIGH_CONCURENCY -mf_pagecache_consist_64kHC_t_SOURCES = $(mf_pagecache_consist_src) -mf_pagecache_consist_64kHC_t_CPPFLAGS = $(mf_pagecache_common_cppflags) -DPAGE_SIZE=65536 -DTEST_HIGH_CONCURENCY +mf_pagecache_consist_1kHC_t_big_SOURCES = $(mf_pagecache_consist_src) +mf_pagecache_consist_1kHC_t_big_CPPFLAGS = $(mf_pagecache_common_cppflags) -DPAGE_SIZE=1024 -DTEST_HIGH_CONCURENCY +mf_pagecache_consist_64kHC_t_big_SOURCES = $(mf_pagecache_consist_src) +mf_pagecache_consist_64kHC_t_big_CPPFLAGS = $(mf_pagecache_common_cppflags) -DPAGE_SIZE=65536 -DTEST_HIGH_CONCURENCY -mf_pagecache_consist_1kRD_t_SOURCES = $(mf_pagecache_consist_src) -mf_pagecache_consist_1kRD_t_CPPFLAGS = $(mf_pagecache_common_cppflags) -DPAGE_SIZE=1024 -DTEST_READERS -mf_pagecache_consist_64kRD_t_SOURCES = $(mf_pagecache_consist_src) -mf_pagecache_consist_64kRD_t_CPPFLAGS = $(mf_pagecache_common_cppflags) -DPAGE_SIZE=65536 -DTEST_READERS +mf_pagecache_consist_1kRD_t_big_SOURCES = $(mf_pagecache_consist_src) +mf_pagecache_consist_1kRD_t_big_CPPFLAGS = $(mf_pagecache_common_cppflags) -DPAGE_SIZE=1024 -DTEST_READERS +mf_pagecache_consist_64kRD_t_big_SOURCES = $(mf_pagecache_consist_src) +mf_pagecache_consist_64kRD_t_big_CPPFLAGS = $(mf_pagecache_common_cppflags) -DPAGE_SIZE=65536 -DTEST_READERS -mf_pagecache_consist_1kWR_t_SOURCES = $(mf_pagecache_consist_src) -mf_pagecache_consist_1kWR_t_CPPFLAGS = $(mf_pagecache_common_cppflags) -DPAGE_SIZE=1024 -DTEST_WRITERS -mf_pagecache_consist_64kWR_t_SOURCES = $(mf_pagecache_consist_src) -mf_pagecache_consist_64kWR_t_CPPFLAGS = $(mf_pagecache_common_cppflags) -DPAGE_SIZE=65536 -DTEST_WRITERS +mf_pagecache_consist_1kWR_t_big_SOURCES = $(mf_pagecache_consist_src) +mf_pagecache_consist_1kWR_t_big_CPPFLAGS = $(mf_pagecache_common_cppflags) -DPAGE_SIZE=1024 -DTEST_WRITERS +mf_pagecache_consist_64kWR_t_big_SOURCES = $(mf_pagecache_consist_src) +mf_pagecache_consist_64kWR_t_big_CPPFLAGS = $(mf_pagecache_common_cppflags) -DPAGE_SIZE=65536 -DTEST_WRITERS # the generic lock manager may not be used in the end and lockman1-t crashes, # so we don't build lockman-t and lockman1-t CLEANFILES = maria_control page_cache_test_file_1 \ - maria_log.???????? maria_control -noinst_PROGRAMS = ma_control_file-t trnman-t lockman2-t + maria_log.???????? diff --git a/storage/maria/unittest/ma_control_file-t.c b/storage/maria/unittest/ma_control_file-t.c index 00b3382cc06..9f6a6c9cf56 100644 --- a/storage/maria/unittest/ma_control_file-t.c +++ b/storage/maria/unittest/ma_control_file-t.c @@ -133,10 +133,8 @@ static int delete_file(myf my_flags) static int verify_module_values_match_expected() { RET_ERR_UNLESS(last_logno == expect_logno); - RET_ERR_UNLESS(last_checkpoint_lsn.file_no == - expect_checkpoint_lsn.file_no); - RET_ERR_UNLESS(last_checkpoint_lsn.rec_offset == - expect_checkpoint_lsn.rec_offset); + RET_ERR_UNLESS(last_checkpoint_lsn == + expect_checkpoint_lsn); return 0; } @@ -148,10 +146,8 @@ static int verify_module_values_match_expected() static int verify_module_values_are_impossible() { RET_ERR_UNLESS(last_logno == CONTROL_FILE_IMPOSSIBLE_FILENO); - RET_ERR_UNLESS(last_checkpoint_lsn.file_no == - CONTROL_FILE_IMPOSSIBLE_FILENO); - RET_ERR_UNLESS(last_checkpoint_lsn.rec_offset == - CONTROL_FILE_IMPOSSIBLE_LOG_OFFSET); + RET_ERR_UNLESS(last_checkpoint_lsn == + CONTROL_FILE_IMPOSSIBLE_LSN); return 0; } @@ -173,9 +169,9 @@ static int create_or_open_file() return 0; } -static int write_file(const LSN *checkpoint_lsn, - uint32 logno, - uint objs_to_write) +static int write_file(const LSN checkpoint_lsn, + uint32 logno, + uint objs_to_write) { RET_ERR_UNLESS(ma_control_file_write_and_force(checkpoint_lsn, logno, objs_to_write) == 0); @@ -191,8 +187,9 @@ static int test_one_log() RET_ERR_UNLESS(create_or_open_file() == CONTROL_FILE_OK); objs_to_write= CONTROL_FILE_UPDATE_ONLY_LOGNO; expect_logno= 123; - RET_ERR_UNLESS(write_file(NULL, expect_logno, - objs_to_write) == 0); + RET_ERR_UNLESS(write_file(CONTROL_FILE_IMPOSSIBLE_LSN, + expect_logno, + objs_to_write) == 0); RET_ERR_UNLESS(close_file() == 0); return 0; } @@ -208,8 +205,8 @@ static int test_five_logs() for (i= 0; i<5; i++) { expect_logno*= 3; - RET_ERR_UNLESS(write_file(NULL, expect_logno, - objs_to_write) == 0); + RET_ERR_UNLESS(write_file(CONTROL_FILE_IMPOSSIBLE_LSN, expect_logno, + objs_to_write) == 0); } RET_ERR_UNLESS(close_file() == 0); return 0; @@ -224,29 +221,29 @@ static int test_3_checkpoints_and_2_logs() */ RET_ERR_UNLESS(create_or_open_file() == CONTROL_FILE_OK); objs_to_write= CONTROL_FILE_UPDATE_ONLY_LSN; - expect_checkpoint_lsn= (LSN){5, 10000}; - RET_ERR_UNLESS(write_file(&expect_checkpoint_lsn, - expect_logno, objs_to_write) == 0); + expect_checkpoint_lsn= MAKE_LSN(5, 10000); + RET_ERR_UNLESS(write_file(expect_checkpoint_lsn, + expect_logno, objs_to_write) == 0); objs_to_write= CONTROL_FILE_UPDATE_ONLY_LOGNO; expect_logno= 17; - RET_ERR_UNLESS(write_file(&expect_checkpoint_lsn, - expect_logno, objs_to_write) == 0); + RET_ERR_UNLESS(write_file(expect_checkpoint_lsn, + expect_logno, objs_to_write) == 0); objs_to_write= CONTROL_FILE_UPDATE_ONLY_LSN; - expect_checkpoint_lsn= (LSN){17, 20000}; - RET_ERR_UNLESS(write_file(&expect_checkpoint_lsn, - expect_logno, objs_to_write) == 0); + expect_checkpoint_lsn= MAKE_LSN(17, 20000); + RET_ERR_UNLESS(write_file(expect_checkpoint_lsn, + expect_logno, objs_to_write) == 0); objs_to_write= CONTROL_FILE_UPDATE_ONLY_LSN; - expect_checkpoint_lsn= (LSN){17, 45000}; - RET_ERR_UNLESS(write_file(&expect_checkpoint_lsn, - expect_logno, objs_to_write) == 0); + expect_checkpoint_lsn= MAKE_LSN(17, 45000); + RET_ERR_UNLESS(write_file(expect_checkpoint_lsn, + expect_logno, objs_to_write) == 0); objs_to_write= CONTROL_FILE_UPDATE_ONLY_LOGNO; expect_logno= 19; - RET_ERR_UNLESS(write_file(&expect_checkpoint_lsn, - expect_logno, objs_to_write) == 0); + RET_ERR_UNLESS(write_file(expect_checkpoint_lsn, + expect_logno, objs_to_write) == 0); RET_ERR_UNLESS(close_file() == 0); return 0; } @@ -274,9 +271,9 @@ static int test_binary_content() RET_ERR_UNLESS(my_close(fd, MYF(MY_WME)) == 0); RET_ERR_UNLESS(create_or_open_file() == CONTROL_FILE_OK); i= uint4korr(buffer+5); - RET_ERR_UNLESS(i == last_checkpoint_lsn.file_no); + RET_ERR_UNLESS(i == LSN_FILE_NO(last_checkpoint_lsn)); i= uint4korr(buffer+9); - RET_ERR_UNLESS(i == last_checkpoint_lsn.rec_offset); + RET_ERR_UNLESS(i == LSN_OFFSET(last_checkpoint_lsn)); i= uint4korr(buffer+13); RET_ERR_UNLESS(i == last_logno); RET_ERR_UNLESS(close_file() == 0); diff --git a/storage/maria/unittest/ma_test_loghandler-t.c b/storage/maria/unittest/ma_test_loghandler-t.c index 1cbfcac504e..afbef150b62 100644 --- a/storage/maria/unittest/ma_test_loghandler-t.c +++ b/storage/maria/unittest/ma_test_loghandler-t.c @@ -1,6 +1,7 @@ #include "../maria_def.h" #include #include +#include #ifndef DBUG_OFF static const char *default_dbug_option; @@ -83,7 +84,7 @@ static my_bool read_and_check_content(TRANSLOG_HEADER_BUFFER *rec, uchar *buffer, uint skip) { DBUG_ASSERT(rec->record_length < LONG_BUFFER_SIZE * 2 + 7 * 2 + 2); - if (translog_read_record(&rec->lsn, 0, rec->record_length, buffer, NULL) != + if (translog_read_record(rec->lsn, 0, rec->record_length, buffer, NULL) != rec->record_length) return 1; return check_content(buffer + skip, rec->record_length - skip); @@ -103,7 +104,7 @@ int main(int argc, char *argv[]) }; uchar long_buffer[LONG_BUFFER_SIZE * 2 + 7 * 2 + 2]; PAGECACHE pagecache; - LSN lsn, lsn_base, first_lsn, *lsn_ptr; + LSN lsn, lsn_base, first_lsn, lsn_ptr; TRANSLOG_HEADER_BUFFER rec; struct st_translog_scanner_data scanner; int rc; @@ -172,7 +173,7 @@ int main(int argc, char *argv[]) printf("write %d\n", i); if (i % 2) { - lsn7store(lsn_buff, &lsn_base); + lsn7store(lsn_buff, lsn_base); if (translog_write_record(&lsn, LOGREC_CLR_END, (i % 0xFFFF), NULL, 7, lsn_buff, 0)) @@ -182,7 +183,7 @@ int main(int argc, char *argv[]) translog_destroy(); exit(1); } - lsn7store(lsn_buff, &lsn_base); + lsn7store(lsn_buff, lsn_base); if ((rec_len= random() / (RAND_MAX / (LONG_BUFFER_SIZE + 1))) < 12) rec_len= 12; if (translog_write_record(&lsn, @@ -198,8 +199,8 @@ int main(int argc, char *argv[]) } else { - lsn7store(lsn_buff, &lsn_base); - lsn7store(lsn_buff + 7, &first_lsn); + lsn7store(lsn_buff, lsn_base); + lsn7store(lsn_buff + 7, first_lsn); if (translog_write_record(&lsn, LOGREC_UNDO_ROW_DELETE, (i % 0xFFFF), NULL, 23, lsn_buff, 0)) @@ -209,8 +210,8 @@ int main(int argc, char *argv[]) translog_destroy(); exit(1); } - lsn7store(lsn_buff, &lsn_base); - lsn7store(lsn_buff + 7, &first_lsn); + lsn7store(lsn_buff, lsn_base); + lsn7store(lsn_buff + 7, first_lsn); if ((rec_len= random() / (RAND_MAX / (LONG_BUFFER_SIZE + 1))) < 19) rec_len= 19; if (translog_write_record(&lsn, @@ -246,7 +247,7 @@ int main(int argc, char *argv[]) translog_destroy(); exit(1); } - if (translog_flush(&lsn)) + if (translog_flush(lsn)) { fprintf(stderr, "Can't flush #%lu\n", (ulong) i); translog_destroy(); @@ -282,7 +283,7 @@ int main(int argc, char *argv[]) rc= 1; { - translog_size_t len= translog_read_record_header(&first_lsn, &rec); + translog_size_t len= translog_read_record_header(first_lsn, &rec); if (len == 0) { fprintf(stderr, "translog_read_record_header failed (%d)\n", errno); @@ -291,19 +292,19 @@ int main(int argc, char *argv[]) if (rec.type !=LOGREC_LONG_TRANSACTION_ID || rec.short_trid != 0 || rec.record_length != 6 || uint4korr(rec.header) != 0 || (uint)rec.header[4] != 0 || rec.header[5] != 0xFF || - first_lsn.file_no != rec.lsn.file_no || - first_lsn.rec_offset != rec.lsn.rec_offset) + first_lsn != rec.lsn) { fprintf(stderr, "Incorrect LOGREC_LONG_TRANSACTION_ID data read(0)\n" "type %u, strid %u, len %u, i: %u, 4: %u 5: %u, " - "lsn(0x%lx,0x%lx)\n", + "lsn(%lu,0x%lx)\n", (uint) rec.type, (uint) rec.short_trid, (uint) rec.record_length, - uint4korr(rec.header), (uint) rec.header[4], (uint) rec.header[5], - (ulong) rec.lsn.file_no, (ulong) rec.lsn.rec_offset); + (uint) uint4korr(rec.header), (uint) rec.header[4], + (uint) rec.header[5], + (ulong) LSN_FILE_NO(rec.lsn), (ulong) LSN_OFFSET(rec.lsn)); goto err; } lsn= first_lsn; - lsn_ptr= &first_lsn; + lsn_ptr= first_lsn; for (i= 1;; i++) { if (i % 1000 == 0) @@ -315,7 +316,7 @@ int main(int argc, char *argv[]) i, errno); goto err; } - if (rec.lsn.file_no == CONTROL_FILE_IMPOSSIBLE_FILENO) + if (rec.lsn == CONTROL_FILE_IMPOSSIBLE_LSN) { if (i != ITERATIONS) { @@ -325,37 +326,35 @@ int main(int argc, char *argv[]) } break; } - lsn_ptr= NULL; /* use scanner after its - initialization */ + /* use scanner after its initialization */ + lsn_ptr= 0; if (i % 2) { LSN ref; - lsn7korr(&ref, rec.header); + ref= lsn7korr(rec.header); if (rec.type !=LOGREC_CLR_END || rec.short_trid != (i % 0xFFFF) || - rec.record_length != 7 || ref.file_no != lsn.file_no || - ref.rec_offset != lsn.rec_offset) + rec.record_length != 7 || ref != lsn) { - fprintf(stderr, "Incorrect LOGREC_CLR_END data read(%d)" - "type %u, strid %u, len %u, ref(%u,0x%lx), lsn(%u,0x%lx)\n", + fprintf(stderr, "Incorrect LOGREC_CLR_END data read(%d) " + "type %u, strid %u, len %u, ref(%lu,0x%lx), " + "lsn(%lu,0x%lx)\n", i, (uint) rec.type, (uint) rec.short_trid, (uint) rec.record_length, - (uint) ref.file_no, (ulong) ref.rec_offset, - (uint) rec.lsn.file_no, (ulong) rec.lsn.rec_offset); + (ulong) LSN_FILE_NO(ref), (ulong) LSN_OFFSET(ref), + (ulong) LSN_FILE_NO(rec.lsn), (ulong) LSN_OFFSET(rec.lsn)); goto err; } } else { LSN ref1, ref2; - lsn7korr(&ref1, rec.header); - lsn7korr(&ref2, rec.header + 7); + ref1= lsn7korr(rec.header); + ref2= lsn7korr(rec.header + 7); if (rec.type !=LOGREC_UNDO_ROW_DELETE || rec.short_trid != (i % 0xFFFF) || rec.record_length != 23 || - ref1.file_no != lsn.file_no || - ref1.rec_offset != lsn.rec_offset || - ref2.file_no != first_lsn.file_no || - ref2.rec_offset != first_lsn.rec_offset || + ref1 != lsn || + ref2 != first_lsn || rec.header[22] != 0x55 || rec.header[21] != 0xAA || rec.header[20] != 0x55 || rec.header[19] != 0xAA || rec.header[18] != 0x55 || rec.header[17] != 0xAA || @@ -363,19 +362,19 @@ int main(int argc, char *argv[]) rec.header[14] != 0x55) { fprintf(stderr, "Incorrect LOGREC_UNDO_ROW_DELETE data read(%d)" - "type %u, strid %u, len %u, ref1(%u,0x%lx), " - "ref2(%u,0x%lx) %x%x%x%x%x%x%x%x%x " - "lsn(%u,0x%lx)\n", + "type %u, strid %u, len %u, ref1(%lu,0x%lx), " + "ref2(%lu,0x%lx) %x%x%x%x%x%x%x%x%x " + "lsn(%lu,0x%lx)\n", i, (uint) rec.type, (uint) rec.short_trid, (uint) rec.record_length, - (uint) ref1.file_no, (ulong) ref1.rec_offset, - (uint) ref2.file_no, (ulong) ref2.rec_offset, + (ulong) LSN_FILE_NO(ref1), (ulong) LSN_OFFSET(ref1), + (ulong) LSN_FILE_NO(ref2), (ulong) LSN_OFFSET(ref2), (uint) rec.header[14], (uint) rec.header[15], (uint) rec.header[16], (uint) rec.header[17], (uint) rec.header[18], (uint) rec.header[19], (uint) rec.header[20], (uint) rec.header[21], (uint) rec.header[22], - (uint) rec.lsn.file_no, (ulong) rec.lsn.rec_offset); + (ulong) LSN_FILE_NO(rec.lsn), (ulong) LSN_OFFSET(rec.lsn)); goto err; } } @@ -386,7 +385,7 @@ int main(int argc, char *argv[]) "failed (%d)\n", i, errno); goto err; } - if (rec.lsn.file_no == CONTROL_FILE_IMPOSSIBLE_FILENO) + if (rec.lsn == CONTROL_FILE_IMPOSSIBLE_LSN) { fprintf(stderr, "EOL met at the middle of iteration (first var) %u " "instead of beginning of %u\n", i, ITERATIONS); @@ -395,20 +394,19 @@ int main(int argc, char *argv[]) if (i % 2) { LSN ref; - lsn7korr(&ref, rec.header); + ref= lsn7korr(rec.header); if ((rec_len= random() / (RAND_MAX / (LONG_BUFFER_SIZE + 1))) < 12) rec_len= 12; if (rec.type !=LOGREC_UNDO_KEY_INSERT || rec.short_trid != (i % 0xFFFF) || rec.record_length != rec_len + 7 || - len != 12 || ref.file_no != lsn.file_no || - ref.rec_offset != lsn.rec_offset || + len != 12 || ref != lsn || check_content(rec.header + 7, len - 7)) { fprintf(stderr, "Incorrect LOGREC_UNDO_KEY_INSERT data read(%d)" "type %u (%d), strid %u (%d), len %lu, %lu + 7 (%d), " "hdr len: %u (%d), " - "ref(%u,0x%lx), lsn(%u,0x%lx) (%d), content: %d\n", + "ref(%lu,0x%lx), lsn(%lu,0x%lx) (%d), content: %d\n", i, (uint) rec.type, rec.type !=LOGREC_UNDO_KEY_INSERT, (uint) rec.short_trid, @@ -417,10 +415,9 @@ int main(int argc, char *argv[]) rec.record_length != rec_len + 7, (uint) len, len != 12, - (uint) ref.file_no, (ulong) ref.rec_offset, - (uint) rec.lsn.file_no, (ulong) rec.lsn.rec_offset, - (len != 12 || ref.file_no != lsn.file_no || - ref.rec_offset != lsn.rec_offset), + (ulong) LSN_FILE_NO(ref), (ulong) LSN_OFFSET(ref), + (ulong) LSN_FILE_NO(rec.lsn), (ulong) LSN_OFFSET(rec.lsn), + (len != 12 || ref != lsn), check_content(rec.header + 7, len - 7)); goto err; } @@ -428,46 +425,44 @@ int main(int argc, char *argv[]) { fprintf(stderr, "Incorrect LOGREC_UNDO_KEY_INSERT in whole rec read " - "lsn(%u,0x%lx)\n", - (uint) rec.lsn.file_no, (ulong) rec.lsn.rec_offset); + "lsn(%lu,0x%lx)\n", + (ulong) LSN_FILE_NO(rec.lsn), (ulong) LSN_OFFSET(rec.lsn)); goto err; } } else { LSN ref1, ref2; - lsn7korr(&ref1, rec.header); - lsn7korr(&ref2, rec.header + 7); + ref1= lsn7korr(rec.header); + ref2= lsn7korr(rec.header + 7); if ((rec_len= random() / (RAND_MAX / (LONG_BUFFER_SIZE + 1))) < 19) rec_len= 19; if (rec.type !=LOGREC_UNDO_KEY_DELETE || rec.short_trid != (i % 0xFFFF) || rec.record_length != rec_len + 14 || len != 19 || - ref1.file_no != lsn.file_no || - ref1.rec_offset != lsn.rec_offset || - ref2.file_no != first_lsn.file_no || - ref2.rec_offset != first_lsn.rec_offset || + ref1 != lsn || + ref2 != first_lsn || check_content(rec.header + 14, len - 14)) { fprintf(stderr, "Incorrect LOGREC_UNDO_KEY_DELETE data read(%d)" "type %u, strid %u, len %lu != %lu + 7, hdr len: %u, " - "ref1(%u,0x%lx), ref2(%u,0x%lx), " - "lsn(%u,0x%lx)\n", + "ref1(%lu,0x%lx), ref2(%lu,0x%lx), " + "lsn(%lu,0x%lx)\n", i, (uint) rec.type, (uint) rec.short_trid, (ulong) rec.record_length, (ulong) rec_len, (uint) len, - (uint) ref1.file_no, (ulong) ref1.rec_offset, - (uint) ref2.file_no, (ulong) ref2.rec_offset, - (uint) rec.lsn.file_no, (ulong) rec.lsn.rec_offset); + (ulong) LSN_FILE_NO(ref1), (ulong) LSN_OFFSET(ref1), + (ulong) LSN_FILE_NO(ref2), (ulong) LSN_OFFSET(ref2), + (ulong) LSN_FILE_NO(rec.lsn), (ulong) LSN_OFFSET(rec.lsn)); goto err; } if (read_and_check_content(&rec, long_buffer, 14)) { fprintf(stderr, "Incorrect LOGREC_UNDO_KEY_DELETE in whole rec read " - "lsn(%u,0x%lx)\n", - (uint) rec.lsn.file_no, (ulong) rec.lsn.rec_offset); + "lsn(%lu,0x%lx)\n", + (ulong) LSN_FILE_NO(rec.lsn), (ulong) LSN_OFFSET(rec.lsn)); goto err; } } @@ -479,7 +474,7 @@ int main(int argc, char *argv[]) i, errno); goto err; } - if (rec.lsn.file_no == CONTROL_FILE_IMPOSSIBLE_FILENO) + if (rec.lsn == CONTROL_FILE_IMPOSSIBLE_LSN) { fprintf(stderr, "EOL met at the middle of iteration %u " "instead of beginning of %u\n", i, ITERATIONS); @@ -492,12 +487,12 @@ int main(int argc, char *argv[]) { fprintf(stderr, "Incorrect LOGREC_LONG_TRANSACTION_ID data read(%d)\n" "type %u, strid %u, len %u, i: %u, 4: %u 5: %u " - "lsn(%u,0x%lx)\n", + "lsn(%lu,0x%lx)\n", i, (uint) rec.type, (uint) rec.short_trid, (uint) rec.record_length, - uint4korr(rec.header), (uint) rec.header[4], + (uint) uint4korr(rec.header), (uint) rec.header[4], (uint) rec.header[5], - (uint) rec.lsn.file_no, (ulong) rec.lsn.rec_offset); + (ulong) LSN_FILE_NO(rec.lsn), (ulong) LSN_OFFSET(rec.lsn)); goto err; } @@ -513,18 +508,19 @@ int main(int argc, char *argv[]) { fprintf(stderr, "Incorrect LOGREC_REDO_INSERT_ROW_HEAD data read(%d)" "type %u, strid %u, len %lu != %lu, hdr len: %u, " - "lsn(%u,0x%lx)\n", + "lsn(%lu,0x%lx)\n", i, (uint) rec.type, (uint) rec.short_trid, (ulong) rec.record_length, (ulong) rec_len, - (uint) len, (uint) rec.lsn.file_no, (ulong) rec.lsn.rec_offset); + (uint) len, + (ulong) LSN_FILE_NO(rec.lsn), (ulong) LSN_OFFSET(rec.lsn)); goto err; } if (read_and_check_content(&rec, long_buffer, 0)) { fprintf(stderr, "Incorrect LOGREC_UNDO_KEY_DELETE in whole rec read " - "lsn(%u,0x%lx)\n", - (uint) rec.lsn.file_no, (ulong) rec.lsn.rec_offset); + "lsn(%lu,0x%lx)\n", + (ulong) LSN_FILE_NO(rec.lsn), (ulong) LSN_OFFSET(rec.lsn)); goto err; } } diff --git a/storage/maria/unittest/ma_test_loghandler_multigroup-t.c b/storage/maria/unittest/ma_test_loghandler_multigroup-t.c index abb12faa015..1d5ad9c81d8 100644 --- a/storage/maria/unittest/ma_test_loghandler_multigroup-t.c +++ b/storage/maria/unittest/ma_test_loghandler_multigroup-t.c @@ -1,6 +1,7 @@ #include "../maria_def.h" #include #include +#include #ifndef DBUG_OFF static const char *default_dbug_option; @@ -84,7 +85,7 @@ static my_bool read_and_check_content(TRANSLOG_HEADER_BUFFER *rec, translog_size_t len; DBUG_ENTER("read_and_check_content"); DBUG_ASSERT(rec->record_length < LONG_BUFFER_SIZE + 7 * 2 + 2); - if ((len= translog_read_record(&rec->lsn, 0, rec->record_length, + if ((len= translog_read_record(rec->lsn, 0, rec->record_length, buffer, NULL)) != rec->record_length) { fprintf(stderr, "Requested %lu byte, read %lu\n", @@ -121,7 +122,7 @@ int main(int argc, char *argv[]) }; uchar *long_buffer= malloc(LONG_BUFFER_SIZE + 7 * 2 + 2); PAGECACHE pagecache; - LSN lsn, lsn_base, first_lsn, *lsn_ptr; + LSN lsn, lsn_base, first_lsn, lsn_ptr; TRANSLOG_HEADER_BUFFER rec; struct st_translog_scanner_data scanner; int rc; @@ -194,7 +195,7 @@ int main(int argc, char *argv[]) printf("write %d\n", i); if (i % 2) { - lsn7store(lsn_buff, &lsn_base); + lsn7store(lsn_buff, lsn_base); if (translog_write_record(&lsn, LOGREC_CLR_END, (i % 0xFFFF), NULL, 7, lsn_buff, 0)) @@ -204,7 +205,7 @@ int main(int argc, char *argv[]) translog_destroy(); exit(1); } - lsn7store(lsn_buff, &lsn_base); + lsn7store(lsn_buff, lsn_base); rec_len= get_len(); if (translog_write_record(&lsn, LOGREC_UNDO_KEY_INSERT, @@ -219,8 +220,8 @@ int main(int argc, char *argv[]) } else { - lsn7store(lsn_buff, &lsn_base); - lsn7store(lsn_buff + 7, &first_lsn); + lsn7store(lsn_buff, lsn_base); + lsn7store(lsn_buff + 7, first_lsn); if (translog_write_record(&lsn, LOGREC_UNDO_ROW_DELETE, (i % 0xFFFF), NULL, 23, lsn_buff, 0)) @@ -230,8 +231,8 @@ int main(int argc, char *argv[]) translog_destroy(); exit(1); } - lsn7store(lsn_buff, &lsn_base); - lsn7store(lsn_buff + 7, &first_lsn); + lsn7store(lsn_buff, lsn_base); + lsn7store(lsn_buff + 7, first_lsn); rec_len= get_len(); if (translog_write_record(&lsn, LOGREC_UNDO_KEY_DELETE, @@ -294,7 +295,7 @@ int main(int argc, char *argv[]) rc= 1; { - translog_size_t len= translog_read_record_header(&first_lsn, &rec); + translog_size_t len= translog_read_record_header(first_lsn, &rec); if (len == 0) { fprintf(stderr, "translog_read_record_header failed (%d)\n", errno); @@ -304,21 +305,21 @@ int main(int argc, char *argv[]) if (rec.type !=LOGREC_LONG_TRANSACTION_ID || rec.short_trid != 0 || rec.record_length != 6 || uint4korr(rec.header) != 0 || (uint)rec.header[4] != 0 || rec.header[5] != 0xFF || - first_lsn.file_no != rec.lsn.file_no || - first_lsn.rec_offset != rec.lsn.rec_offset) + first_lsn != rec.lsn) { fprintf(stderr, "Incorrect LOGREC_LONG_TRANSACTION_ID data read(0)\n" "type %u, strid %u, len %u, i: %u, 4: %u 5: %u, " - "lsn(0x%lx,0x%lx)\n", + "lsn(0x%lu,0x%lx)\n", (uint) rec.type, (uint) rec.short_trid, (uint) rec.record_length, - uint4korr(rec.header), (uint) rec.header[4], (uint) rec.header[5], - (ulong) rec.lsn.file_no, (ulong) rec.lsn.rec_offset); + (uint)uint4korr(rec.header), (uint) rec.header[4], + (uint) rec.header[5], + (ulong) LSN_FILE_NO(rec.lsn), (ulong) LSN_OFFSET(rec.lsn)); translog_free_record_header(&rec); goto err; } translog_free_record_header(&rec); lsn= first_lsn; - lsn_ptr= &first_lsn; + lsn_ptr= first_lsn; for (i= 1;; i++) { if (i % SHOW_DIVIDER == 0) @@ -331,7 +332,7 @@ int main(int argc, char *argv[]) translog_free_record_header(&rec); goto err; } - if (rec.lsn.file_no == CONTROL_FILE_IMPOSSIBLE_FILENO) + if (rec.lsn == CONTROL_FILE_IMPOSSIBLE_LSN) { if (i != ITERATIONS) { @@ -342,23 +343,22 @@ int main(int argc, char *argv[]) } break; } - lsn_ptr= NULL; /* use scanner after its - initialization */ + /* use scanner after its initialization */ + lsn_ptr= 0; if (i % 2) { LSN ref; - lsn7korr(&ref, rec.header); - if (rec.type !=LOGREC_CLR_END || rec.short_trid != (i % 0xFFFF) || - rec.record_length != 7 || ref.file_no != lsn.file_no || - ref.rec_offset != lsn.rec_offset) + ref= lsn7korr(rec.header); + if (rec.type != LOGREC_CLR_END || rec.short_trid != (i % 0xFFFF) || + rec.record_length != 7 || ref != lsn) { fprintf(stderr, "Incorrect LOGREC_CLR_END data read(%d)" - "type %u, strid %u, len %u, ref(%u,0x%lx), lsn(%u,0x%lx)\n", + "type %u, strid %u, len %u, ref(%lu,0x%lx), lsn(%lu,0x%lx)\n", i, (uint) rec.type, (uint) rec.short_trid, (uint) rec.record_length, - (uint) ref.file_no, (ulong) ref.rec_offset, - (uint) rec.lsn.file_no, (ulong) rec.lsn.rec_offset); + (ulong) LSN_FILE_NO(ref), (ulong) LSN_OFFSET(ref), + (ulong) LSN_FILE_NO(rec.lsn), (ulong) LSN_OFFSET(rec.lsn)); translog_free_record_header(&rec); goto err; } @@ -366,15 +366,13 @@ int main(int argc, char *argv[]) else { LSN ref1, ref2; - lsn7korr(&ref1, rec.header); - lsn7korr(&ref2, rec.header + 7); + ref1= lsn7korr(rec.header); + ref2= lsn7korr(rec.header + 7); if (rec.type !=LOGREC_UNDO_ROW_DELETE || rec.short_trid != (i % 0xFFFF) || rec.record_length != 23 || - ref1.file_no != lsn.file_no || - ref1.rec_offset != lsn.rec_offset || - ref2.file_no != first_lsn.file_no || - ref2.rec_offset != first_lsn.rec_offset || + ref1 != lsn || + ref2 != first_lsn || rec.header[22] != 0x55 || rec.header[21] != 0xAA || rec.header[20] != 0x55 || rec.header[19] != 0xAA || rec.header[18] != 0x55 || rec.header[17] != 0xAA || @@ -382,19 +380,19 @@ int main(int argc, char *argv[]) rec.header[14] != 0x55) { fprintf(stderr, "Incorrect LOGREC_UNDO_ROW_DELETE data read(%d)" - "type %u, strid %u, len %u, ref1(%u,0x%lx), " - "ref2(%u,0x%lx) %x%x%x%x%x%x%x%x%x " - "lsn(%u,0x%lx)\n", + "type %u, strid %u, len %u, ref1(%lu,0x%lx), " + "ref2(%lu,0x%lx) %x%x%x%x%x%x%x%x%x " + "lsn(%lu,0x%lx)\n", i, (uint) rec.type, (uint) rec.short_trid, (uint) rec.record_length, - (uint) ref1.file_no, (ulong) ref1.rec_offset, - (uint) ref2.file_no, (ulong) ref2.rec_offset, + (ulong) LSN_FILE_NO(ref1), (ulong) LSN_OFFSET(ref1), + (ulong) LSN_FILE_NO(ref2), (ulong) LSN_OFFSET(ref2), (uint) rec.header[14], (uint) rec.header[15], (uint) rec.header[16], (uint) rec.header[17], (uint) rec.header[18], (uint) rec.header[19], (uint) rec.header[20], (uint) rec.header[21], (uint) rec.header[22], - (uint) rec.lsn.file_no, (ulong) rec.lsn.rec_offset); + (ulong) LSN_FILE_NO(rec.lsn), (ulong) LSN_OFFSET(rec.lsn)); translog_free_record_header(&rec); goto err; } @@ -408,7 +406,7 @@ int main(int argc, char *argv[]) "failed (%d)\n", i, errno); goto err; } - if (rec.lsn.file_no == CONTROL_FILE_IMPOSSIBLE_FILENO) + if (rec.lsn == CONTROL_FILE_IMPOSSIBLE_LSN) { fprintf(stderr, "EOL met at the middle of iteration (first var) %u " "instead of beginning of %u\n", i, ITERATIONS); @@ -417,19 +415,18 @@ int main(int argc, char *argv[]) if (i % 2) { LSN ref; - lsn7korr(&ref, rec.header); + ref= lsn7korr(rec.header); rec_len= get_len(); if (rec.type !=LOGREC_UNDO_KEY_INSERT || rec.short_trid != (i % 0xFFFF) || rec.record_length != rec_len + 7 || - len != 12 || ref.file_no != lsn.file_no || - ref.rec_offset != lsn.rec_offset || + len != 12 || ref != lsn || check_content(rec.header + 7, len - 7)) { fprintf(stderr, "Incorrect LOGREC_UNDO_KEY_INSERT data read(%d)" "type %u (%d), strid %u (%d), len %lu, %lu + 7 (%d), " "hdr len: %u (%d), " - "ref(%u,0x%lx), lsn(%u,0x%lx) (%d), content: %d\n", + "ref(%lu,0x%lx), lsn(%lu,0x%lx) (%d), content: %d\n", i, (uint) rec.type, rec.type !=LOGREC_UNDO_KEY_INSERT, (uint) rec.short_trid, @@ -438,10 +435,9 @@ int main(int argc, char *argv[]) rec.record_length != rec_len + 7, (uint) len, len != 12, - (uint) ref.file_no, (ulong) ref.rec_offset, - (uint) rec.lsn.file_no, (ulong) rec.lsn.rec_offset, - (ref.file_no != lsn.file_no || - ref.rec_offset != lsn.rec_offset), + (ulong) LSN_FILE_NO(ref), (ulong) LSN_OFFSET(ref), + (ulong) LSN_FILE_NO(rec.lsn), (ulong) LSN_OFFSET(rec.lsn), + (ref != lsn), check_content(rec.header + 7, len - 7)); translog_free_record_header(&rec); goto err; @@ -450,8 +446,8 @@ int main(int argc, char *argv[]) { fprintf(stderr, "Incorrect LOGREC_UNDO_KEY_INSERT in whole rec read " - "lsn(%u,0x%lx)\n", - (uint) rec.lsn.file_no, (ulong) rec.lsn.rec_offset); + "lsn(%lu,0x%lx)\n", + (ulong) LSN_FILE_NO(rec.lsn), (ulong) LSN_OFFSET(rec.lsn)); translog_free_record_header(&rec); goto err; } @@ -459,29 +455,27 @@ int main(int argc, char *argv[]) else { LSN ref1, ref2; - lsn7korr(&ref1, rec.header); - lsn7korr(&ref2, rec.header + 7); + ref1= lsn7korr(rec.header); + ref2= lsn7korr(rec.header + 7); rec_len= get_len(); if (rec.type !=LOGREC_UNDO_KEY_DELETE || rec.short_trid != (i % 0xFFFF) || rec.record_length != rec_len + 14 || len != 19 || - ref1.file_no != lsn.file_no || - ref1.rec_offset != lsn.rec_offset || - ref2.file_no != first_lsn.file_no || - ref2.rec_offset != first_lsn.rec_offset || + ref1 != lsn || + ref2 != first_lsn || check_content(rec.header + 14, len - 14)) { fprintf(stderr, "Incorrect LOGREC_UNDO_KEY_DELETE data read(%d)" "type %u, strid %u, len %lu != %lu + 7, hdr len: %u, " - "ref1(%u,0x%lx), ref2(%u,0x%lx), " - "lsn(%u,0x%lx)\n", + "ref1(%lu,0x%lx), ref2(%lu,0x%lx), " + "lsn(%lu,0x%lx)\n", i, (uint) rec.type, (uint) rec.short_trid, (ulong) rec.record_length, (ulong) rec_len, (uint) len, - (uint) ref1.file_no, (ulong) ref1.rec_offset, - (uint) ref2.file_no, (ulong) ref2.rec_offset, - (uint) rec.lsn.file_no, (ulong) rec.lsn.rec_offset); + (ulong) LSN_FILE_NO(ref1), (ulong) LSN_OFFSET(ref1), + (ulong) LSN_FILE_NO(ref2), (ulong) LSN_OFFSET(ref2), + (ulong) LSN_FILE_NO(rec.lsn), (ulong) LSN_OFFSET(rec.lsn)); translog_free_record_header(&rec); goto err; } @@ -489,8 +483,8 @@ int main(int argc, char *argv[]) { fprintf(stderr, "Incorrect LOGREC_UNDO_KEY_DELETE in whole rec read " - "lsn(%u,0x%lx)\n", - (uint) rec.lsn.file_no, (ulong) rec.lsn.rec_offset); + "lsn(%lu,0x%lx)\n", + (ulong) LSN_FILE_NO(rec.lsn), (ulong) LSN_OFFSET(rec.lsn)); translog_free_record_header(&rec); goto err; } @@ -505,7 +499,7 @@ int main(int argc, char *argv[]) translog_free_record_header(&rec); goto err; } - if (rec.lsn.file_no == CONTROL_FILE_IMPOSSIBLE_FILENO) + if (rec.lsn == CONTROL_FILE_IMPOSSIBLE_LSN) { fprintf(stderr, "EOL met at the middle of iteration %u " "instead of beginning of %u\n", i, ITERATIONS); @@ -519,12 +513,12 @@ int main(int argc, char *argv[]) { fprintf(stderr, "Incorrect LOGREC_LONG_TRANSACTION_ID data read(%d)\n" "type %u, strid %u, len %u, i: %u, 4: %u 5: %u " - "lsn(%u,0x%lx)\n", + "lsn(%lu,0x%lx)\n", i, (uint) rec.type, (uint) rec.short_trid, (uint) rec.record_length, - uint4korr(rec.header), (uint) rec.header[4], + (uint)uint4korr(rec.header), (uint) rec.header[4], (uint) rec.header[5], - (uint) rec.lsn.file_no, (ulong) rec.lsn.rec_offset); + (ulong) LSN_FILE_NO(rec.lsn), (ulong) LSN_OFFSET(rec.lsn)); translog_free_record_header(&rec); goto err; } @@ -541,10 +535,11 @@ int main(int argc, char *argv[]) { fprintf(stderr, "Incorrect LOGREC_REDO_INSERT_ROW_HEAD data read(%d)" "type %u, strid %u, len %lu != %lu, hdr len: %u, " - "lsn(%u,0x%lx)\n", + "lsn(%lu,0x%lx)\n", i, (uint) rec.type, (uint) rec.short_trid, (ulong) rec.record_length, (ulong) rec_len, - (uint) len, (uint) rec.lsn.file_no, (ulong) rec.lsn.rec_offset); + (uint) len, + (ulong) LSN_FILE_NO(rec.lsn), (ulong) LSN_OFFSET(rec.lsn)); translog_free_record_header(&rec); goto err; } @@ -552,8 +547,8 @@ int main(int argc, char *argv[]) { fprintf(stderr, "Incorrect LOGREC_UNDO_KEY_DELETE in whole rec read " - "lsn(%u,0x%lx)\n", - (uint) rec.lsn.file_no, (ulong) rec.lsn.rec_offset); + "lsn(%lu,0x%lx)\n", + (ulong) LSN_FILE_NO(rec.lsn), (ulong) LSN_OFFSET(rec.lsn)); translog_free_record_header(&rec); goto err; } diff --git a/storage/maria/unittest/ma_test_loghandler_multithread-t.c b/storage/maria/unittest/ma_test_loghandler_multithread-t.c index 794dc6dd255..ed5479026ef 100644 --- a/storage/maria/unittest/ma_test_loghandler_multithread-t.c +++ b/storage/maria/unittest/ma_test_loghandler_multithread-t.c @@ -1,6 +1,7 @@ #include "../maria_def.h" #include #include +#include #ifndef DBUG_OFF static const char *default_dbug_option; @@ -102,7 +103,7 @@ static my_bool read_and_check_content(TRANSLOG_HEADER_BUFFER *rec, translog_size_t len; DBUG_ENTER("read_and_check_content"); DBUG_ASSERT(rec->record_length < LONG_BUFFER_SIZE + 7 * 2 + 2); - if ((len= translog_read_record(&rec->lsn, 0, rec->record_length, + if ((len= translog_read_record(rec->lsn, 0, rec->record_length, buffer, NULL)) != rec->record_length) { fprintf(stderr, "Requested %lu byte, read %lu\n", @@ -149,16 +150,17 @@ void writer(int num) DBUG_PRINT("info", ("thread: %u, iteration: %u, len: %lu, " "lsn1 (%lu,0x%lx) lsn2 (%lu,0x%lx)", num, i, (ulong) lens[num][i], - (ulong) lsns1[num][i].file_no, - (ulong) lsns1[num][i].rec_offset, - (ulong) lsns2[num][i].file_no, - (ulong) lsns2[num][i].rec_offset)); + (ulong) LSN_FILE_NO(lsns1[num][i]), + (ulong) LSN_OFFSET(lsns1[num][i]), + (ulong) LSN_FILE_NO(lsns2[num][i]), + (ulong) LSN_OFFSET(lsns2[num][i]))); printf("thread: %u, iteration: %u, len: %lu, " "lsn1 (%lu,0x%lx) lsn2 (%lu,0x%lx)\n", num, i, (ulong) lens[num][i], - (ulong) lsns1[num][i].file_no, - (ulong) lsns1[num][i].rec_offset, - (ulong) lsns2[num][i].file_no, (ulong) lsns2[num][i].rec_offset); + (ulong) LSN_FILE_NO(lsns1[num][i]), + (ulong) LSN_OFFSET(lsns1[num][i]), + (ulong) LSN_FILE_NO(lsns2[num][i]), + (ulong) LSN_OFFSET(lsns2[num][i])); } DBUG_VOID_RETURN; } @@ -191,7 +193,7 @@ int main(int argc, char **argv __attribute__ ((unused))) uint32 i; uint pagen; PAGECACHE pagecache; - LSN first_lsn, *lsn_ptr; + LSN first_lsn, lsn_ptr; TRANSLOG_HEADER_BUFFER rec; struct st_translog_scanner_data scanner; pthread_t tid; @@ -344,20 +346,18 @@ int main(int argc, char **argv __attribute__ ((unused))) /* Find last LSN and flush up to it (all our log) */ { - LSN max= - { - 0, 0 - }; + LSN max= 0; for (i= 0; i < WRITERS; i++) { if (cmp_translog_addr(lsns2[i][ITERATIONS - 1], max) > 0) max= lsns2[i][ITERATIONS - 1]; } DBUG_PRINT("info", ("first lsn: (%lu,0x%lx), max lsn: (%lu,0x%lx)", - (ulong) first_lsn.file_no, - (ulong) first_lsn.rec_offset, - (ulong) max.file_no, (ulong) max.rec_offset)); - translog_flush(&max); + (ulong) LSN_FILE_NO(first_lsn), + (ulong) LSN_OFFSET(first_lsn), + (ulong) LSN_FILE_NO(max), + (ulong) LSN_OFFSET(max))); + translog_flush(max); } rc= 1; @@ -369,11 +369,11 @@ int main(int argc, char **argv __attribute__ ((unused))) bzero(indeces, sizeof(indeces)); - lsn_ptr= &first_lsn; + lsn_ptr= first_lsn; for (i= 0;; i++) { len= translog_read_next_record_header(lsn_ptr, &rec, 1, &scanner); - lsn_ptr= NULL; + lsn_ptr= 0; if (len == 0) { @@ -382,7 +382,7 @@ int main(int argc, char **argv __attribute__ ((unused))) translog_free_record_header(&rec); goto err; } - if (rec.lsn.file_no == CONTROL_FILE_IMPOSSIBLE_FILENO) + if (rec.lsn == CONTROL_FILE_IMPOSSIBLE_LSN) { if (i != WRITERS * ITERATIONS * 2) { @@ -412,9 +412,10 @@ int main(int argc, char **argv __attribute__ ((unused))) (uint) rec.short_trid, (uint) uint2korr(rec.header), (uint) rec.record_length, (uint) index, (uint) uint4korr(rec.header + 2), - (ulong) rec.lsn.file_no, (ulong) rec.lsn.rec_offset, - (ulong) lsns1[rec.short_trid][index].file_no, - (ulong) lsns1[rec.short_trid][index].rec_offset); + (ulong) LSN_FILE_NO(rec.lsn), + (ulong) LSN_OFFSET(rec.lsn), + (ulong) LSN_FILE_NO(lsns1[rec.short_trid][index]), + (ulong) LSN_OFFSET(lsns1[rec.short_trid][index])); translog_free_record_header(&rec); goto err; } @@ -437,9 +438,10 @@ int main(int argc, char **argv __attribute__ ((unused))) (uint) len, (ulong) rec.record_length, lens[rec.short_trid][index], (rec.record_length != lens[rec.short_trid][index]), - (ulong) rec.lsn.file_no, (ulong) rec.lsn.rec_offset, - (ulong) lsns2[rec.short_trid][index].file_no, - (ulong) lsns2[rec.short_trid][index].rec_offset); + (ulong) LSN_FILE_NO(rec.lsn), + (ulong) LSN_OFFSET(rec.lsn), + (ulong) LSN_FILE_NO(lsns2[rec.short_trid][index]), + (ulong) LSN_OFFSET(lsns2[rec.short_trid][index])); translog_free_record_header(&rec); goto err; } @@ -447,8 +449,9 @@ int main(int argc, char **argv __attribute__ ((unused))) { fprintf(stderr, "Incorrect LOGREC_REDO_INSERT_ROW_HEAD in whole rec read " - "lsn(%u,0x%lx)\n", - (uint) rec.lsn.file_no, (ulong) rec.lsn.rec_offset); + "lsn(%lu,0x%lx)\n", + (ulong) LSN_FILE_NO(rec.lsn), + (ulong) LSN_OFFSET(rec.lsn)); translog_free_record_header(&rec); goto err; } diff --git a/storage/maria/unittest/ma_test_loghandler_pagecache-t.c b/storage/maria/unittest/ma_test_loghandler_pagecache-t.c index 6771b5f888d..f215805d829 100644 --- a/storage/maria/unittest/ma_test_loghandler_pagecache-t.c +++ b/storage/maria/unittest/ma_test_loghandler_pagecache-t.c @@ -1,6 +1,7 @@ #include "../maria_def.h" #include #include +#include #ifndef DBUG_OFF static const char *default_dbug_option; @@ -106,7 +107,7 @@ int main(int argc, char *argv[]) bzero(page, PCACHE_PAGE); #define PAGE_LSN_OFFSET 0 - lsn7store(page + PAGE_LSN_OFFSET, &lsn); + lsn7store(page + PAGE_LSN_OFFSET, lsn); pagecache_write(&pagecache, &file1, 0, 3, (char*)page, PAGECACHE_LSN_PAGE, PAGECACHE_LOCK_LEFT_UNLOCKED, diff --git a/storage/maria/unittest/mf_pagecache_consist.c b/storage/maria/unittest/mf_pagecache_consist.c index 14105cb85af..8ea0094762c 100755 --- a/storage/maria/unittest/mf_pagecache_consist.c +++ b/storage/maria/unittest/mf_pagecache_consist.c @@ -8,6 +8,7 @@ #include #include #include "test_file.h" +#include #define PCACHE_SIZE (PAGE_SIZE*1024*8) diff --git a/storage/maria/unittest/mf_pagecache_single.c b/storage/maria/unittest/mf_pagecache_single.c index 6b1d57f5f91..91cceee618d 100644 --- a/storage/maria/unittest/mf_pagecache_single.c +++ b/storage/maria/unittest/mf_pagecache_single.c @@ -7,6 +7,7 @@ #include #include #include "test_file.h" +#include #define PCACHE_SIZE (PAGE_SIZE*1024*10) @@ -235,7 +236,7 @@ int simple_pin_test() 0, PAGECACHE_LOCK_READ_UNLOCK, PAGECACHE_UNPIN, - 0, 0); + 0); if (flush_pagecache_blocks(&pagecache, &file1, FLUSH_FORCE_WRITE)) { diag("Got error in flush_pagecache_blocks\n"); @@ -364,7 +365,7 @@ int simple_big_test() 0); } desc[i].length= 0; - desc[i].content= NULL; + desc[i].content= '\0'; ok(1, "Simple big file write"); /* check written pages sequentally read */ for (i= 0; i < PCACHE_SIZE/(PAGE_SIZE/2); i++) -- cgit v1.2.1 From 3bc8f629dd35d832cbee14a26c187cb76e78bf6d Mon Sep 17 00:00:00 2001 From: unknown Date: Mon, 19 Feb 2007 23:01:27 +0200 Subject: =?UTF-8?q?Postreview=20changes.=20Fixed=20befaviour=20when=20logh?= =?UTF-8?q?andler=20flags=20changed=20from=20one=20run=20to=20another=20on?= =?UTF-8?q?e.=20Description=20of=20maria=20transaction=20log=20and=20contr?= =?UTF-8?q?ol=20file=20added=20to=20the=20file=20command=C3=8Ds=20magic=20?= =?UTF-8?q?number=20file.?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit mysys/mf_pagecache.c: postreview changes storage/maria/ma_control_file.c: Postreview changes. storage/maria/ma_control_file.h: Postreview changes. storage/maria/ma_loghandler.c: Postreview changes. Fixed befaviour when loghandler flags changed from one run to another one. storage/maria/ma_loghandler.h: Postreview changes. Functions comment left only near the function body. storage/maria/ma_loghandler_lsn.h: Postreview changes. storage/maria/unittest/ma_test_loghandler-t.c: Postreview changes. storage/maria/unittest/ma_test_loghandler_multigroup-t.c: Postreview changes. storage/maria/unittest/ma_test_loghandler_multithread-t.c: Postreview changes. storage/maria/unittest/ma_test_loghandler_pagecache-t.c: Postreview changes. support-files/magic: Description of maria transaction log and control file added to the file commandÕs magic number file. --- mysys/mf_pagecache.c | 2 +- storage/maria/ma_control_file.c | 11 +- storage/maria/ma_control_file.h | 25 +- storage/maria/ma_loghandler.c | 2212 +++++++++----------- storage/maria/ma_loghandler.h | 246 +-- storage/maria/ma_loghandler_lsn.h | 7 +- storage/maria/unittest/ma_test_loghandler-t.c | 100 +- .../unittest/ma_test_loghandler_multigroup-t.c | 109 +- .../unittest/ma_test_loghandler_multithread-t.c | 23 +- .../unittest/ma_test_loghandler_pagecache-t.c | 2 +- support-files/magic | 7 + 11 files changed, 1262 insertions(+), 1482 deletions(-) diff --git a/mysys/mf_pagecache.c b/mysys/mf_pagecache.c index 9ef3d5841b5..1b9d48c80e6 100755 --- a/mysys/mf_pagecache.c +++ b/mysys/mf_pagecache.c @@ -570,7 +570,7 @@ static uint pagecache_fwrite(PAGECACHE *pagecache, DBUG_PRINT("info", ("Log handler call")); /* TODO: integrate with page format */ #define PAGE_LSN_OFFSET 0 - lsn= lsn7korr(buffer + PAGE_LSN_OFFSET); + lsn= lsn_korr(buffer + PAGE_LSN_OFFSET); /* check CONTROL_FILE_IMPOSSIBLE_FILENO & CONTROL_FILE_IMPOSSIBLE_LOG_OFFSET diff --git a/storage/maria/ma_control_file.c b/storage/maria/ma_control_file.c index 3f9af34b2f1..07eddb956a2 100644 --- a/storage/maria/ma_control_file.c +++ b/storage/maria/ma_control_file.c @@ -21,7 +21,6 @@ */ #include "maria_def.h" -#include "ma_control_file.h" /* Here is the implementation of this module */ @@ -31,13 +30,13 @@ */ /* total size should be < sector size for atomic write operation */ -#define CONTROL_FILE_MAGIC_STRING "MACF" +#define CONTROL_FILE_MAGIC_STRING "\xfe\xfe\xc\1MACF" #define CONTROL_FILE_MAGIC_STRING_OFFSET 0 #define CONTROL_FILE_MAGIC_STRING_SIZE (sizeof(CONTROL_FILE_MAGIC_STRING)-1) #define CONTROL_FILE_CHECKSUM_OFFSET (CONTROL_FILE_MAGIC_STRING_OFFSET + CONTROL_FILE_MAGIC_STRING_SIZE) #define CONTROL_FILE_CHECKSUM_SIZE 1 #define CONTROL_FILE_LSN_OFFSET (CONTROL_FILE_CHECKSUM_OFFSET + CONTROL_FILE_CHECKSUM_SIZE) -#define CONTROL_FILE_LSN_SIZE (3+4) +#define CONTROL_FILE_LSN_SIZE LSN_STORE_SIZE #define CONTROL_FILE_FILENO_OFFSET (CONTROL_FILE_LSN_OFFSET + CONTROL_FILE_LSN_SIZE) #define CONTROL_FILE_FILENO_SIZE 4 #define CONTROL_FILE_SIZE (CONTROL_FILE_FILENO_OFFSET + CONTROL_FILE_FILENO_SIZE) @@ -200,7 +199,7 @@ CONTROL_FILE_ERROR ma_control_file_create_or_open() error= CONTROL_FILE_BAD_CHECKSUM; goto err; } - last_checkpoint_lsn= lsn7korr(buffer + CONTROL_FILE_LSN_OFFSET); + last_checkpoint_lsn= lsn_korr(buffer + CONTROL_FILE_LSN_OFFSET); last_logno= uint4korr(buffer + CONTROL_FILE_FILENO_OFFSET); DBUG_RETURN(0); @@ -261,9 +260,9 @@ int ma_control_file_write_and_force(const LSN checkpoint_lsn, uint32 logno, DBUG_ASSERT(0); if (update_checkpoint_lsn) - lsn7store(buffer + CONTROL_FILE_LSN_OFFSET, checkpoint_lsn); + lsn_store(buffer + CONTROL_FILE_LSN_OFFSET, checkpoint_lsn); else /* store old value == change nothing */ - lsn7store(buffer + CONTROL_FILE_LSN_OFFSET, last_checkpoint_lsn); + lsn_store(buffer + CONTROL_FILE_LSN_OFFSET, last_checkpoint_lsn); if (update_logno) int4store(buffer + CONTROL_FILE_FILENO_OFFSET, logno); diff --git a/storage/maria/ma_control_file.h b/storage/maria/ma_control_file.h index 8e5dafac24c..616babc3bb2 100644 --- a/storage/maria/ma_control_file.h +++ b/storage/maria/ma_control_file.h @@ -19,9 +19,6 @@ First version written by Guilhem Bichot on 2006-04-27. */ -#ifndef _ma_control_file_h -#define _ma_control_file_h - #define CONTROL_FILE_BASE_NAME "maria_control" /* indicate absence of the log file number; first log is always number 1, 0 is @@ -47,11 +44,6 @@ extern LSN last_checkpoint_lsn; */ extern uint32 last_logno; -/* - Looks for the control file. If absent, it's a fresh start, create file. - If present, read it to find out last checkpoint's LSN and last log. - Called at engine's start. -*/ typedef enum enum_control_file_error { CONTROL_FILE_OK= 0, CONTROL_FILE_TOO_SMALL, @@ -60,21 +52,26 @@ typedef enum enum_control_file_error { CONTROL_FILE_BAD_CHECKSUM, CONTROL_FILE_UNKNOWN_ERROR /* any other error */ } CONTROL_FILE_ERROR; -CONTROL_FILE_ERROR ma_control_file_create_or_open(); + +#define CONTROL_FILE_UPDATE_ALL 0 +#define CONTROL_FILE_UPDATE_ONLY_LSN 1 +#define CONTROL_FILE_UPDATE_ONLY_LOGNO 2 + +/* + Looks for the control file. If absent, it's a fresh start, create file. + If present, read it to find out last checkpoint's LSN and last log. + Called at engine's start. +*/ +CONTROL_FILE_ERROR ma_control_file_create_or_open(); /* Write information durably to the control file. Called when we have created a new log (after syncing this log's creation) and when we have written a checkpoint (after syncing this log record). */ -#define CONTROL_FILE_UPDATE_ALL 0 -#define CONTROL_FILE_UPDATE_ONLY_LSN 1 -#define CONTROL_FILE_UPDATE_ONLY_LOGNO 2 int ma_control_file_write_and_force(const LSN checkpoint_lsn, uint32 logno, uint objs_to_write); /* Free resources taken by control file subsystem */ int ma_control_file_end(); - -#endif diff --git a/storage/maria/ma_loghandler.c b/storage/maria/ma_loghandler.c index 01b4c68f12f..dd12c73770b 100644 --- a/storage/maria/ma_loghandler.c +++ b/storage/maria/ma_loghandler.c @@ -1,7 +1,6 @@ #include "maria_def.h" -#include -/* number of opened log files in the pagecache (should be at lesst 2) */ +/* number of opened log files in the pagecache (should be at least 2) */ #define OPENED_FILES_NUM 3 /* records buffer size (should be LOG_PAGE_SIZE * n) */ @@ -14,16 +13,18 @@ Should be at least 4, because one thread can block up to 2 buffers in normal circumstances (less then half of one and full other, or just switched one and other), But if we met end of the file in the middle and - have to switch buffer it will be 3. + 1 or 2 buffer for flushing/writing. + have to switch buffer it will be 3. + 1 buffer for flushing/writing. + We have a bigger number here for higher concurrency. */ #define TRANSLOG_BUFFERS_NO 5 -/* number of bytes which is worth to be left on first page */ +/* number of bytes (+ header) which can be unused on first page in sequence */ #define TRANSLOG_MINCHUNK_CONTENT 1 -/* length of transaction log file name maria_log.XXXXXXXX*/ -#define TRANSLOG_FILE_NAME_LENGTH 18 /* version of log file */ -#define TRANSLOG_VERSION_ID 10000 +#define TRANSLOG_VERSION_ID 10000 /* 1.00.00 */ +#define TRANSLOG_PAGE_FLAGS 6 /* transaction log page flags offset */ + +/* QQ: For temporary debugging */ #define UNRECOVERABLE_ERROR(E) \ do { \ DBUG_PRINT("error", E); \ @@ -32,13 +33,15 @@ } while(0); + /* record part descriptor */ struct st_translog_part { translog_size_t len; - uchar *buff; + byte *buff; }; + /* record parts descriptor */ struct st_translog_parts { @@ -62,39 +65,39 @@ struct st_translog_buffer How much written (or will be written when copy_to_buffer_in_progress become 0) to this buffer */ - uint32 size; - /* This Buffer File */ + translog_size_t size; + /* File handler for this buffer */ File file; /* Threads which are waiting for buffer filling/freeing */ WQUEUE waiting_filling_buffer; /* Number of record which are in copy progress */ - int16 copy_to_buffer_in_progress; + uint copy_to_buffer_in_progress; /* list of waiting buffer ready threads */ struct st_my_thread_var *waiting_flush; - /* lock for the buffer. Current buffer also lock the handler */ - pthread_mutex_t mutex; struct st_translog_buffer *overlay; #ifndef DBUG_OFF - struct st_my_thread_var *locked_by; - uint8 buffer_no; + uint buffer_no; #endif + /* lock for the buffer. Current buffer also lock the handler */ + pthread_mutex_t mutex; /* IO cache for current log */ - uchar buffer[TRANSLOG_WRITE_BUFFER]; + byte buffer[TRANSLOG_WRITE_BUFFER]; }; struct st_buffer_cursor { /* pointer on the buffer */ - uchar *ptr; + byte *ptr; + /* current buffer */ + struct st_translog_buffer *buffer; /* current page fill */ - uint16 current_page_size; + uint16 current_page_fill; /* how many times we finish this page to write it */ uint16 write_counter; /* previous write offset */ uint16 previous_offset; - /* current buffer and its number */ - struct st_translog_buffer *buffer; + /* Number of current buffer */ uint8 buffer_no; my_bool chaser, protected; }; @@ -104,30 +107,31 @@ struct st_translog_descriptor { /* *** Parameters of the log handler *** */ - /* Directory to store files */ - char directory[FN_REFLEN]; + /* Page cache for the log reads */ + PAGECACHE *pagecache; + /* Flags */ + uint flags; /* max size of one log size (for new logs creation) */ uint32 log_file_max_size; /* server version */ uint32 server_version; /* server ID */ uint32 server_id; - /* Page cache for the log reads */ - PAGECACHE *pagecache; - /* Flags */ - uint flags; - /* Page overhead calculated by flags */ - uint16 page_overhead; - /* Page capacity calculated by flags (TRANSLOG_PAGE_SIZE-page_overhead-1) */ - uint16 page_capacity_chunk_2; /* Loghandler's buffer capacity in case of chunk 2 filling */ uint32 buffer_capacity_chunk_2; /* Half of the buffer capacity in case of chunk 2 filling */ uint32 half_buffer_capacity_chunk_2; + /* Page overhead calculated by flags */ + uint16 page_overhead; + /* Page capacity calculated by flags (TRANSLOG_PAGE_SIZE-page_overhead-1) */ + uint16 page_capacity_chunk_2; + /* Directory to store files */ + char directory[FN_REFLEN]; /* *** Current state of the log handler *** */ /* Current and (OPENED_FILES_NUM-1) last logs number in page cache */ File log_file_num[OPENED_FILES_NUM]; + File directory_fd; /* buffers for log writing */ struct st_translog_buffer buffers[TRANSLOG_BUFFERS_NO]; /* @@ -142,12 +146,12 @@ struct st_translog_descriptor LSN flushed; LSN sent_to_file; pthread_mutex_t sent_to_file_lock; - File directory_fd; }; static struct st_translog_descriptor log_descriptor; -static uchar end_of_log= 0; +/* Marker for end of log */ +static byte end_of_log= 0; /* record classes */ enum record_class @@ -159,21 +163,16 @@ enum record_class }; /* chunk types */ -#define TRANSLOG_CHUNK_LSN 0x00 /* 0 chunk refer as LSN (head - or tail */ -#define TRANSLOG_CHUNK_FIXED 0x40 /* 1 (pseudo)fixed record (also - LSN) */ -#define TRANSLOG_CHUNK_NOHDR 0x80 /* 2 no header chunk (till page - end) */ -#define TRANSLOG_CHUNK_LNGTH 0xC0 /* 3 chunk with chunk length */ -#define TRANSLOG_CHUNK_TYPE 0xC0 /* Mask to get chunk type */ +#define TRANSLOG_CHUNK_LSN 0x00 /* 0 chunk refer as LSN (head or tail */ +#define TRANSLOG_CHUNK_FIXED (1 << 6) /* 1 (pseudo)fixed record (also LSN) */ +#define TRANSLOG_CHUNK_NOHDR (2 << 6) /* 2 no head chunk (till page end) */ +#define TRANSLOG_CHUNK_LNGTH (3 << 6) /* 3 chunk with chunk length */ +#define TRANSLOG_CHUNK_TYPE (3 << 6) /* Mask to get chunk type */ #define TRANSLOG_REC_TYPE 0x3F /* Mask to get record type */ /* compressed (relative) LSN constants */ -#define TRANSLOG_CLSN_LEN_BITS 0xC0 /* Mask to get compressed LSN - length */ -#define TRANSLOG_CLSN_MAX_LEN 5 /* Maximum length of compressed - LSN */ +#define TRANSLOG_CLSN_LEN_BITS 0xC0 /* Mask to get compressed LSN length */ +#define TRANSLOG_CLSN_MAX_LEN 5 /* Maximum length of compressed LSN */ typedef my_bool(*prewrite_rec_hook) (enum translog_record_type type, void *tcb, @@ -184,11 +183,14 @@ typedef my_bool(*inwrite_rec_hook) (enum translog_record_type type, LSN *lsn, struct st_translog_parts *parts); -typedef int16(*read_rec_hook) (enum translog_record_type type, - int16 read_length, uchar *read_buff, - uchar *decoded_buff); +typedef uint16(*read_rec_hook) (enum translog_record_type type, + uint16 read_length, uchar *read_buff, + byte *decoded_buff); -/* Descriptor of log record type */ +/* + Descriptor of log record type + Note: Don't reorder because of constructs later... +*/ struct st_log_record_type_descriptor { /* internal class of the record */ @@ -207,9 +209,10 @@ struct st_log_record_type_descriptor For pseudo fixed records number of compressed LSNs followed by system header */ - int16 compresed_LSN; + int16 compressed_LSN; }; + static struct st_log_record_type_descriptor log_record_type_descriptor[LOGREC_NUMBER_OF_TYPES]= { @@ -343,6 +346,8 @@ static struct st_log_record_type_descriptor {LOGRECTYPE_NOT_ALLOWED, 0, 0, NULL, NULL, NULL, 0} }; +/* all possible flags page overheads */ +static uint page_overhead[TRANSLOG_FLAGS_NUM]; typedef struct st_translog_validator_data { @@ -368,12 +373,13 @@ const char *maria_data_root; static char *translog_filename_by_fileno(uint32 file_no, char *path) { - char file_name[10 + 8 + 1]; + char file_name[10 + 8 + 1]; /* See my_sprintf */ char *res; DBUG_ENTER("translog_filename_by_fileno"); + DBUG_ASSERT(file_no <= 0xfffffff); my_sprintf(file_name, (file_name, "maria_log.%08u", file_no)); res= fn_format(path, file_name, log_descriptor.directory, "", MYF(MY_WME)); - DBUG_PRINT("info", ("Path '%s', path: 0x%lx, res: 0x%lx", + DBUG_PRINT("info", ("Path: '%s' path: 0x%lx res: 0x%lx", res, (ulong) path, (ulong) res)); DBUG_RETURN(res); } @@ -387,8 +393,8 @@ static char *translog_filename_by_fileno(uint32 file_no, char *path) file_no Number of the log we want to open RETURN - 0 error - file descriptor number + -1 error + # file descriptor number */ static File open_logfile_by_number_no_cache(uint32 file_no) @@ -397,15 +403,15 @@ static File open_logfile_by_number_no_cache(uint32 file_no) char path[FN_REFLEN]; DBUG_ENTER("open_logfile_by_number_no_cache"); - if ((file= my_open(translog_filename_by_fileno(file_no, path), O_CREAT | O_BINARY | /* O_DIRECT - | - */ O_RDWR, + /* Todo: add O_DIRECT to open flags (when buffer is aligned) */ + if ((file= my_open(translog_filename_by_fileno(file_no, path), + O_CREAT | O_BINARY | O_RDWR, MYF(MY_WME))) < 0) { UNRECOVERABLE_ERROR(("Error %d during opening file '%s'", errno, path)); - DBUG_RETURN(0); + DBUG_RETURN(-1); } - DBUG_PRINT("info", ("File '%s', handler %d", path, file)); + DBUG_PRINT("info", ("File: '%s' handler: %d", path, file)); DBUG_RETURN(file); } @@ -416,42 +422,50 @@ static File open_logfile_by_number_no_cache(uint32 file_no) SYNOPSIS translog_write_file_header(); + NOTES + First page is just a marker page; We don't store any real log data in it. + RETURN 0 OK 1 ERROR */ +uchar NEAR maria_trans_file_magic[]= +{ (uchar) 254, (uchar) 254, (uchar) 11, '\001', 'M', 'A', 'R', 'I', 'A', + 'L', 'O', 'G' }; + static my_bool translog_write_file_header() { ulonglong timestamp; - char page[TRANSLOG_PAGE_SIZE]; + byte page_buff[TRANSLOG_PAGE_SIZE], *page= page_buff; DBUG_ENTER("translog_write_file_header"); /* file tag */ - strnmov(page, "MARIALOG", 8); + memcpy(page, maria_trans_file_magic, sizeof(maria_trans_file_magic)); + page+= sizeof(maria_trans_file_magic); /* timestamp */ timestamp= my_getsystime(); - int8store(page + 8, timestamp); + int8store(page, timestamp); + page+= 8; /* maria version */ - int4store(page + (8 + 8), TRANSLOG_VERSION_ID); + int4store(page, TRANSLOG_VERSION_ID); + page+= 4; /* mysql version (MYSQL_VERSION_ID) */ - int4store(page + (8 + 8 + 4), log_descriptor.server_version); + int4store(page, log_descriptor.server_version); + page+= 4; /* server ID */ - int4store(page + (8 + 8 + 4 + 4), log_descriptor.server_id); - /* loghandler page size/512 */ - int2store(page + (8 + 8 + 4 + 4 + 4), TRANSLOG_PAGE_SIZE / 512); + int4store(page, log_descriptor.server_id); + page+= 4; + /* loghandler page_size/DISK_DRIVE_SECTOR_SIZE */ + int2store(page, TRANSLOG_PAGE_SIZE / DISK_DRIVE_SECTOR_SIZE); + page+= 2; /* file number */ - int3store(page + (8 + 8 + 4 + 4 + 4 + 2), - LSN_FILE_NO(log_descriptor.horizon)); - - bzero(page + (8 + 8 + 4 + 4 + 4 + 2 + 3), - TRANSLOG_PAGE_SIZE - (8 + 8 + 4 + 4 + 4 + 2 + 3)); + int3store(page, LSN_FILE_NO(log_descriptor.horizon)); + page+= 3; + bzero(page, sizeof(page_buff) - (page- page_buff)); - if (my_pwrite(log_descriptor.log_file_num[0], page, - TRANSLOG_PAGE_SIZE, 0, MYF(MY_WME)) != TRANSLOG_PAGE_SIZE) - DBUG_RETURN(1); - - DBUG_RETURN(0); + DBUG_RETURN(my_pwrite(log_descriptor.log_file_num[0], page_buff, + sizeof(page_buff), 0, MYF(MY_WME | MY_NABP)) != 0); } @@ -463,8 +477,8 @@ static my_bool translog_write_file_header() buffer The buffer to initialize RETURN - 0 - OK - 1 - Error + 0 OK + 1 Error */ static my_bool translog_buffer_init(struct st_translog_buffer *buffer) @@ -473,7 +487,7 @@ static my_bool translog_buffer_init(struct st_translog_buffer *buffer) /* This buffer offset */ buffer->last_lsn= CONTROL_FILE_IMPOSSIBLE_LSN; /* This Buffer File */ - buffer->file= 0; + buffer->file= -1; buffer->overlay= 0; /* IO cache for current log */ bzero(buffer->buffer, TRANSLOG_WRITE_BUFFER); @@ -488,8 +502,6 @@ static my_bool translog_buffer_init(struct st_translog_buffer *buffer) /* lock for the buffer. Current buffer also lock the handler */ if (pthread_mutex_init(&buffer->mutex, MY_MUTEX_INIT_FAST)) DBUG_RETURN(1); - DBUG_PRINT("info", ("Init buffer #%u: 0x%lx", - (uint) buffer->buffer_no, (ulong) buffer)); DBUG_RETURN(0); } @@ -502,18 +514,23 @@ static my_bool translog_buffer_init(struct st_translog_buffer *buffer) file file descriptor RETURN - 0 OK - 1 Error + 0 OK + 1 Error */ static my_bool translog_close_log_file(File file) { - PAGECACHE_FILE fl= - { - file - }; + int rc; + PAGECACHE_FILE fl; + fl.file= file; flush_pagecache_blocks(log_descriptor.pagecache, &fl, FLUSH_RELEASE); - return test(my_close(file, MYF(MY_WME))); + /* + Sync file when we close it + TODO: sync only we have changed the log + */ + rc= my_sync(file, MYF(MY_WME)); + rc|= my_close(file, MYF(MY_WME)); + return test(rc); } @@ -532,20 +549,17 @@ static my_bool translog_create_new_file() { int i; uint32 file_no= LSN_FILE_NO(log_descriptor.horizon); - DBUG_ENTER("translog_create_new_file"); - if (log_descriptor.log_file_num[OPENED_FILES_NUM - 1] && + if (log_descriptor.log_file_num[OPENED_FILES_NUM - 1] != -1 && translog_close_log_file(log_descriptor.log_file_num[OPENED_FILES_NUM - 1])) DBUG_RETURN(1); for (i= OPENED_FILES_NUM - 1; i > 0; i--) - { log_descriptor.log_file_num[i]= log_descriptor.log_file_num[i - 1]; - } if ((log_descriptor.log_file_num[0]= - open_logfile_by_number_no_cache(file_no)) <= 0 || + open_logfile_by_number_no_cache(file_no)) == -1 || translog_write_file_header()) DBUG_RETURN(1); @@ -565,8 +579,8 @@ static my_bool translog_create_new_file() buffer This buffer which should be locked RETURN - 0 - OK - 1 - Error + 0 OK + 1 Error */ #ifndef DBUG_OFF @@ -574,21 +588,11 @@ static my_bool translog_buffer_lock(struct st_translog_buffer *buffer) { int res; DBUG_ENTER("translog_buffer_lock"); - DBUG_PRINT("enter", ("Lock buffer #%u (0x%lx): locked by:0x%lx, mutex: 0x%lx", - (uint) buffer->buffer_no, (ulong) buffer, - (ulong) buffer->locked_by, (ulong) &buffer->mutex)); + DBUG_PRINT("enter", + ("Lock buffer #%u: (0x%lx) mutex: 0x%lx", + (uint) buffer->buffer_no, (ulong) buffer, + (ulong) &buffer->mutex)); res= (pthread_mutex_lock(&buffer->mutex) != 0); -#ifndef DBUG_OFF - if (res == 0) - { - DBUG_ASSERT(buffer->locked_by == 0); - buffer->locked_by= my_thread_var; - } - else - DBUG_PRINT("error", ("Can't lock mutex 0x%lx (locked by0x%lx) errno: %d", - (ulong) &buffer->mutex, - (ulong) buffer->locked_by, res)); -#endif DBUG_RETURN(res); } #else @@ -605,8 +609,8 @@ static my_bool translog_buffer_lock(struct st_translog_buffer *buffer) buffer This buffer which should be unlocked RETURN - 0 - OK - 1 - Error + 0 OK + 1 Error */ #ifndef DBUG_OFF @@ -614,16 +618,13 @@ static my_bool translog_buffer_unlock(struct st_translog_buffer *buffer) { int res; DBUG_ENTER("translog_buffer_unlock"); - DBUG_PRINT("enter", ("Unlock buffer... #%u (0x%lx) :locked by:0x%lx (0x%lx)," - " mutex: 0x%lx", + DBUG_PRINT("enter", ("Unlock buffer... #%u (0x%lx) " + "mutex: 0x%lx", (uint) buffer->buffer_no, (ulong) buffer, - (ulong) buffer->locked_by, (ulong) my_thread_var, (ulong) &buffer->mutex)); - DBUG_ASSERT(buffer->locked_by == my_thread_var); - buffer->locked_by= 0; res= (pthread_mutex_unlock(&buffer->mutex) != 0); - DBUG_PRINT("enter", ("Unlocked buffer... #%u: 0x%lx, mutex: 0x%lx", + DBUG_PRINT("enter", ("Unlocked buffer... #%u: 0x%lx mutex: 0x%lx", (uint) buffer->buffer_no, (ulong) buffer, (ulong) &buffer->mutex)); DBUG_RETURN(res); @@ -635,7 +636,7 @@ static my_bool translog_buffer_unlock(struct st_translog_buffer *buffer) /* - Write page header. + Write a header on the page SYNOPSIS translog_new_page_header() @@ -649,10 +650,10 @@ static my_bool translog_buffer_unlock(struct st_translog_buffer *buffer) static void translog_new_page_header(TRANSLOG_ADDRESS *horizon, struct st_buffer_cursor *cursor) { - uchar *ptr; + byte *ptr; DBUG_ENTER("translog_new_page_header"); - DBUG_ASSERT(cursor->ptr !=NULL); + DBUG_ASSERT(cursor->ptr); cursor->protected= 0; @@ -663,41 +664,43 @@ static void translog_new_page_header(TRANSLOG_ADDRESS *horizon, /* File number */ int3store(ptr, LSN_FILE_NO(*horizon)); ptr+= 3; - *(ptr ++)= (uchar) log_descriptor.flags; + *(ptr++)= (byte) log_descriptor.flags; if (log_descriptor.flags & TRANSLOG_PAGE_CRC) { #ifndef DBUG_OFF DBUG_PRINT("info", ("write 0x11223344 CRC to (%lu,0x%lx)", (ulong) LSN_FILE_NO(*horizon), (ulong) LSN_OFFSET(*horizon))); + /* This will be overwritten by real CRC; This is just for debugging */ int4store(ptr, 0x11223344); #endif - /* CRC will be put when page will be finished */ - ptr+= 4; + /* CRC will be put when page is finished */ + ptr+= CRC_LENGTH; } if (log_descriptor.flags & TRANSLOG_SECTOR_PROTECTION) { time_t tm; - int2store(ptr, time(&tm) & 0xFFFF); - ptr+= (TRANSLOG_PAGE_SIZE / 512) * 2; + uint16 tmp_time= time(&tm); + int2store(ptr, tmp_time); + ptr+= (TRANSLOG_PAGE_SIZE / DISK_DRIVE_SECTOR_SIZE) * 2; } { - uint len= (ptr -cursor->ptr); - *horizon+= len; /* it is increasing of offset part of the address */ - cursor->current_page_size= len; + uint len= (ptr - cursor->ptr); + (*horizon)+= len; /* it is increasing of offset part of the address */ + cursor->current_page_fill= len; if (!cursor->chaser) cursor->buffer->size+= len; } cursor->ptr= ptr; - DBUG_PRINT("info", ("NewP buffer #%u: 0x%lx, chaser: %d, Size: %lu (%lu)", + DBUG_PRINT("info", ("NewP buffer #%u: 0x%lx chaser: %d Size: %lu (%lu)", (uint) cursor->buffer->buffer_no, (ulong) cursor->buffer, cursor->chaser, (ulong) cursor->buffer->size, - (ulong) (cursor->ptr -cursor->buffer->buffer))); + (ulong) (cursor->ptr - cursor->buffer->buffer))); DBUG_ASSERT(cursor->chaser || - ((ulong) (cursor->ptr -cursor->buffer->buffer) == + ((ulong) (cursor->ptr - cursor->buffer->buffer) == cursor->buffer->size)); DBUG_ASSERT(cursor->buffer->buffer_no == cursor->buffer_no); - DBUG_ASSERT(cursor->current_page_size <= TRANSLOG_PAGE_SIZE); + DBUG_ASSERT(cursor->current_page_fill <= TRANSLOG_PAGE_SIZE); DBUG_VOID_RETURN; } @@ -709,24 +712,29 @@ static void translog_new_page_header(TRANSLOG_ADDRESS *horizon, translog_put_sector_protection() page reference on the page content cursor cursor of the buffer + + NOTES + We put a sector protection on all following sectors on the page, + except the first sector that is protected by page header. */ -static void translog_put_sector_protection(uchar *page, +static void translog_put_sector_protection(byte *page, struct st_buffer_cursor *cursor) { - uchar *table= page + log_descriptor.page_overhead - - (TRANSLOG_PAGE_SIZE / 512) * 2; + byte *table= page + log_descriptor.page_overhead - + (TRANSLOG_PAGE_SIZE / DISK_DRIVE_SECTOR_SIZE) * 2; uint16 value= uint2korr(table) + cursor->write_counter; - uint16 last_protected_sector= (cursor->previous_offset - 1) / 512; - uint16 start_sector= cursor->previous_offset / 512; + uint16 last_protected_sector= ((cursor->previous_offset - 1) / + DISK_DRIVE_SECTOR_SIZE); + uint16 start_sector= cursor->previous_offset / DISK_DRIVE_SECTOR_SIZE; uint i, offset; - DBUG_ENTER("translog_put_sector_protection"); + if (start_sector == 0) - start_sector= 1; + start_sector= 1; /* First sector is protected */ - DBUG_PRINT("enter", ("Write counter %u, value %u, offset %u, " - "last protected %u, start sector %u", + DBUG_PRINT("enter", ("Write counter:%u value:%u offset:%u, " + "last protected:%u start sector:%u", (uint) cursor->write_counter, (uint) value, (uint) cursor->previous_offset, @@ -734,7 +742,7 @@ static void translog_put_sector_protection(uchar *page, if (last_protected_sector == start_sector) { i= last_protected_sector * 2; - offset= last_protected_sector * 512; + offset= last_protected_sector * DISK_DRIVE_SECTOR_SIZE; /* restore data, because we modified sector which was protected */ if (offset < cursor->previous_offset) page[offset]= table[i]; @@ -742,17 +750,17 @@ static void translog_put_sector_protection(uchar *page, if (offset < cursor->previous_offset) page[offset]= table[i + 1]; } - for (i= start_sector * 2, offset= start_sector * 512; - i < (TRANSLOG_PAGE_SIZE / 512) * 2; (i+= 2), (offset+= 512)) + for (i= start_sector * 2, offset= start_sector * DISK_DRIVE_SECTOR_SIZE; + i < (TRANSLOG_PAGE_SIZE / DISK_DRIVE_SECTOR_SIZE) * 2; + (i+= 2), (offset+= DISK_DRIVE_SECTOR_SIZE)) { - DBUG_PRINT("info", ("sector %u, offset %u, data 0x%x%x", + DBUG_PRINT("info", ("sector:%u offset:%u data 0x%x%x", i / 2, offset, (uint) page[offset], (uint) page[offset + 1])); table[i]= page[offset]; table[i + 1]= page[offset + 1]; - /**((uint16 *)(table + i))= *((uint16* )(page + offset));*/ int2store(page + offset, value); - DBUG_PRINT("info", ("sector %u, offset %u, data 0x%x%x", + DBUG_PRINT("info", ("sector:%u offset:%u data 0x%x%x", i / 2, offset, (uint) page[offset], (uint) page[offset + 1])); } @@ -761,42 +769,20 @@ static void translog_put_sector_protection(uchar *page, /* - Calculate adler CRC of given area + Calculate CRC32 of given area SYNOPSIS - translog_adler_crc() + translog_crc() area Pointer of the area beginning length The Area length RETURN - Adler CRC32 + CRC32 */ -uint32 translog_adler_crc(uchar *area, uint length) +static uint32 translog_crc(byte *area, uint length) { - uint32 a= 1, b= 0; -#define MOD_ADLER 65521 - - while (length) - { - uint tlen= length > 5550 ? 5550 : length; - length-= tlen; - do - { - a+= *area++; - b+= a; - } while (--tlen); - a= (a & 0xffff) + (a >> 16) * (65536 - MOD_ADLER); - b= (b & 0xffff) + (b >> 16) * (65536 - MOD_ADLER); - } - /* It can be shown that a <= 0x1013a here, so a single subtract will do. */ - if (a >= MOD_ADLER) - a-= MOD_ADLER; - /* It can be shown that b can reach 0xffef1 here. */ - b= (b & 0xffff) + (b >> 16) * (65536 - MOD_ADLER); - if (b >= MOD_ADLER) - b-= MOD_ADLER; - return (b << 16) | a; + return crc32(0L, area, length); } @@ -812,26 +798,25 @@ uint32 translog_adler_crc(uchar *area, uint length) static void translog_finish_page(TRANSLOG_ADDRESS *horizon, struct st_buffer_cursor *cursor) { - uint16 left= TRANSLOG_PAGE_SIZE - cursor->current_page_size; - uchar *page= cursor->ptr -cursor->current_page_size; + uint16 left= TRANSLOG_PAGE_SIZE - cursor->current_page_fill; + byte *page= cursor->ptr -cursor->current_page_fill; DBUG_ENTER("translog_finish_page"); - - DBUG_PRINT("enter", ("Buffer #%u 0x%lx, " - "Buffer addr (%lu,0x%lx), " - "Page addr: (%lu,0x%lx), " - "size %lu (%lu), Pg: %u, left: %u", + DBUG_PRINT("enter", ("Buffer: #%u 0x%lx " + "Buffer addr: (%lu,0x%lx) " + "Page addr: (%lu,0x%lx) " + "size:%lu (%lu) Pg:%u left:%u", (uint) cursor->buffer_no, (ulong) cursor->buffer, (ulong) LSN_FILE_NO(cursor->buffer->offset), (ulong) LSN_OFFSET(cursor->buffer->offset), (ulong) LSN_FILE_NO(*horizon), (ulong) (LSN_OFFSET(*horizon) - - cursor->current_page_size), + cursor->current_page_fill), (ulong) cursor->buffer->size, (ulong) (cursor->ptr -cursor->buffer->buffer), - (uint) cursor->current_page_size, (uint) left)); + (uint) cursor->current_page_fill, (uint) left)); DBUG_ASSERT(cursor->ptr !=NULL); DBUG_ASSERT((cursor->ptr -cursor->buffer->buffer) %TRANSLOG_PAGE_SIZE == - cursor->current_page_size % TRANSLOG_PAGE_SIZE); + cursor->current_page_fill % TRANSLOG_PAGE_SIZE); DBUG_ASSERT(LSN_FILE_NO(*horizon) == LSN_FILE_NO(cursor->buffer->offset)); DBUG_ASSERT(LSN_OFFSET(cursor->buffer->offset) + (cursor->ptr -cursor->buffer->buffer) == LSN_OFFSET(*horizon)); @@ -840,48 +825,51 @@ static void translog_finish_page(TRANSLOG_ADDRESS *horizon, DBUG_PRINT("info", ("Already protected and finished")); DBUG_VOID_RETURN; } - if (left != TRANSLOG_PAGE_SIZE && left != 0) + cursor->protected= 1; + + DBUG_ASSERT(left < TRANSLOG_PAGE_SIZE); + if (left != 0) { - DBUG_PRINT("info", ("left %u", (uint) left)); + DBUG_PRINT("info", ("left: %u", (uint) left)); bzero(cursor->ptr, left); cursor->ptr +=left; - *horizon+= left; /* offset increasing */ + (*horizon)+= left; /* offset increasing */ if (!cursor->chaser) cursor->buffer->size+= left; - cursor->current_page_size= 0; - DBUG_PRINT("info", ("Finish Page buffer #%u: 0x%lx, " - "chaser: %d, Size: %lu (%lu)", + cursor->current_page_fill= 0; + DBUG_PRINT("info", ("Finish Page buffer #%u: 0x%lx " + "chaser: %d Size: %lu (%lu)", (uint) cursor->buffer->buffer_no, (ulong) cursor->buffer, cursor->chaser, (ulong) cursor->buffer->size, - (ulong) (cursor->ptr -cursor->buffer->buffer))); + (ulong) (cursor->ptr - cursor->buffer->buffer))); DBUG_ASSERT(cursor->chaser - || ((ulong) (cursor->ptr -cursor->buffer->buffer) == + || ((ulong) (cursor->ptr - cursor->buffer->buffer) == cursor->buffer->size)); DBUG_ASSERT(cursor->buffer->buffer_no == cursor->buffer_no); } - if (log_descriptor.flags & TRANSLOG_SECTOR_PROTECTION) + if (page[TRANSLOG_PAGE_FLAGS] & TRANSLOG_SECTOR_PROTECTION) { translog_put_sector_protection(page, cursor); DBUG_PRINT("info", ("drop write_counter")); cursor->write_counter= 0; cursor->previous_offset= 0; } - if (log_descriptor.flags & TRANSLOG_PAGE_CRC) + if (page[TRANSLOG_PAGE_FLAGS] & TRANSLOG_PAGE_CRC) { - uint32 crc= translog_adler_crc(page + log_descriptor.page_overhead, - TRANSLOG_PAGE_SIZE - - log_descriptor.page_overhead); - DBUG_PRINT("info", ("CRC: 0x%lx", (ulong) crc)); + uint32 crc= translog_crc(page + log_descriptor.page_overhead, + TRANSLOG_PAGE_SIZE - + log_descriptor.page_overhead); + DBUG_PRINT("info", ("CRC: %lx", (ulong) crc)); + /* We have page number, file number and flag before crc */ int4store(page + 3 + 3 + 1, crc); } - cursor->protected= 1; DBUG_VOID_RETURN; } /* - Wait until all thread finish filling this buffer + Wait until all thread finish filling this buffer SYNOPSIS translog_wait_for_writers() @@ -893,46 +881,28 @@ static void translog_finish_page(TRANSLOG_ADDRESS *horizon, static void translog_wait_for_writers(struct st_translog_buffer *buffer) { - struct st_my_thread_var *thread; + struct st_my_thread_var *thread= my_thread_var; DBUG_ENTER("translog_wait_for_writers"); - DBUG_PRINT("enter", ("Buffer #%u 0x%lx, copies in progress: %u", + DBUG_PRINT("enter", ("Buffer #%u 0x%lx copies in progress: %u", (uint) buffer->buffer_no, (ulong) buffer, (int) buffer->copy_to_buffer_in_progress)); - if (!buffer->copy_to_buffer_in_progress) - DBUG_VOID_RETURN; - - thread= my_thread_var; - - DBUG_ASSERT(buffer->file != 0); - - do + while (buffer->copy_to_buffer_in_progress) { - DBUG_PRINT("info", ("wait for writers... , thread 0x%lx, " - "buffer #%u 0x%lx, locked by 0x%lx (0x%lx), " + DBUG_PRINT("info", ("wait for writers... " + "buffer: #%u 0x%lx " "mutex: 0x%lx", - (ulong) thread, (uint) buffer->buffer_no, (ulong) buffer, - (ulong) buffer->locked_by, (ulong) thread, (ulong) &buffer->mutex)); -#ifndef DBUG_OFF - DBUG_ASSERT(buffer->locked_by == thread); - buffer->locked_by= 0; -#endif + DBUG_ASSERT(buffer->file != -1); wqueue_add_and_wait(&buffer->waiting_filling_buffer, thread, &buffer->mutex); - DBUG_PRINT("info", ("wait for writers done, thread 0x%lx, " - "buffer #%u 0x%lx, locked by 0x%lx (0x%lx), " + DBUG_PRINT("info", ("wait for writers done " + "buffer: #%u 0x%lx " "mutex: 0x%lx", - (ulong) thread, (uint) buffer->buffer_no, (ulong) buffer, - (ulong) buffer->locked_by, (ulong) thread, (ulong) &buffer->mutex)); -#ifndef DBUG_OFF - DBUG_ASSERT(buffer->locked_by == 0); - buffer->locked_by= thread; -#endif - } while (buffer->copy_to_buffer_in_progress != 0); + } DBUG_VOID_RETURN; } @@ -940,11 +910,11 @@ static void translog_wait_for_writers(struct st_translog_buffer *buffer) /* - Wait for this buffer become free + Wait for buffer to become free SYNOPSIS translog_wait_for_buffer_free() - buffer The buffer to initialize + buffer The buffer we are waiting for NOTE - this buffer should be locked @@ -954,51 +924,36 @@ static void translog_wait_for_buffer_free(struct st_translog_buffer *buffer) { struct st_my_thread_var *thread= my_thread_var; DBUG_ENTER("translog_wait_for_buffer_free"); - DBUG_PRINT("enter", ("Buffer #%u 0x%lx, copies in progress: %u size 0x%lu", + DBUG_PRINT("enter", ("Buffer: #%u 0x%lx copies in progress: %u " + "File: %d size: 0x%lu", (uint) buffer->buffer_no, (ulong) buffer, (int) buffer->copy_to_buffer_in_progress, - (ulong) buffer->size)); + buffer->file, (ulong) buffer->size)); translog_wait_for_writers(buffer); - if (!buffer->file) - DBUG_VOID_RETURN; - - thread= my_thread_var; - - do + while (buffer->file != -1) { - DBUG_PRINT("info", ("wait for writers... , thread 0x%lx, " - "buffer #%u 0x%lx, locked by 0x%lx (0x%lx), " + DBUG_PRINT("info", ("wait for writers... " + "buffer: #%u 0x%lx " "mutex: 0x%lx", - (ulong) thread, (uint) buffer->buffer_no, (ulong) buffer, - (ulong) buffer->locked_by, (ulong) thread, (ulong) &buffer->mutex)); -#ifndef DBUG_OFF - DBUG_ASSERT(buffer->locked_by == thread); - buffer->locked_by= 0; -#endif wqueue_add_and_wait(&buffer->waiting_filling_buffer, thread, &buffer->mutex); - DBUG_PRINT("info", ("wait for writers done, thread 0x%lx, " - "buffer #%u 0x%lx, locked by 0x%lx (0x%lx), " + DBUG_PRINT("info", ("wait for writers done. " + "buffer: #%u 0x%lx " "mutex: 0x%lx", - (ulong) thread, (uint) buffer->buffer_no, (ulong) buffer, - (ulong) buffer->locked_by, (ulong) thread, (ulong) &buffer->mutex)); -#ifndef DBUG_OFF - DBUG_ASSERT(buffer->locked_by == 0); - buffer->locked_by= thread; -#endif - } while (buffer->copy_to_buffer_in_progress != 0); + } + DBUG_ASSERT(buffer->copy_to_buffer_in_progress == 0); DBUG_VOID_RETURN; } /* - Set cursor on the buffer beginning + Initialize the cursor for a buffer SYNOPSIS translog_cursor_init() @@ -1015,9 +970,8 @@ static void translog_cursor_init(struct st_buffer_cursor *cursor, cursor->ptr= buffer->buffer; cursor->buffer= buffer; cursor->buffer_no= buffer_no; - cursor->current_page_size= 0; + cursor->current_page_fill= 0; cursor->chaser= (cursor != &log_descriptor.bc); - DBUG_PRINT("info", ("drop write_counter")); cursor->write_counter= 0; cursor->previous_offset= 0; cursor->protected= 0; @@ -1037,13 +991,13 @@ static void translog_cursor_init(struct st_buffer_cursor *cursor, static void translog_start_buffer(struct st_translog_buffer *buffer, struct st_buffer_cursor *cursor, - uint8 buffer_no) + uint buffer_no) { DBUG_ENTER("translog_start_buffer"); DBUG_PRINT("enter", - ("Assign buffer #%u (0x%lx) to file %u, offset 0x%lx(%lu)", + ("Assign buffer: #%u (0x%lx) to file: %d offset: 0x%lx(%lu)", (uint) buffer->buffer_no, (ulong) buffer, - (uint) log_descriptor.log_file_num[0], + log_descriptor.log_file_num[0], (ulong) LSN_OFFSET(log_descriptor.horizon), (ulong) LSN_OFFSET(log_descriptor.horizon))); DBUG_ASSERT(buffer_no == buffer->buffer_no); @@ -1053,12 +1007,12 @@ static void translog_start_buffer(struct st_translog_buffer *buffer, buffer->overlay= 0; buffer->size= 0; translog_cursor_init(cursor, buffer, buffer_no); - DBUG_PRINT("info", ("init cursor #%u: 0x%lx, chaser: %d, Size: %lu (%lu)", + DBUG_PRINT("info", ("init cursor #%u: 0x%lx chaser: %d Size: %lu (%lu)", (uint) cursor->buffer->buffer_no, (ulong) cursor->buffer, cursor->chaser, (ulong) cursor->buffer->size, - (ulong) (cursor->ptr -cursor->buffer->buffer))); + (ulong) (cursor->ptr - cursor->buffer->buffer))); DBUG_ASSERT(cursor->chaser || - ((ulong) (cursor->ptr -cursor->buffer->buffer) == + ((ulong) (cursor->ptr - cursor->buffer->buffer) == cursor->buffer->size)); DBUG_ASSERT(cursor->buffer->buffer_no == cursor->buffer_no); DBUG_VOID_RETURN; @@ -1079,21 +1033,21 @@ static void translog_start_buffer(struct st_translog_buffer *buffer, - after return new and old buffer still are locked RETURN - 0 - OK - 1 - Error + 0 OK + 1 Error */ static my_bool translog_buffer_next(TRANSLOG_ADDRESS *horizon, struct st_buffer_cursor *cursor, my_bool new_file) { - uint8 old_buffer_no= cursor->buffer_no; - uint8 new_buffer_no= (old_buffer_no + 1) % TRANSLOG_BUFFERS_NO; + uint old_buffer_no= cursor->buffer_no; + uint new_buffer_no= (old_buffer_no + 1) % TRANSLOG_BUFFERS_NO; struct st_translog_buffer *new_buffer= log_descriptor.buffers + new_buffer_no; my_bool chasing= cursor->chaser; DBUG_ENTER("translog_buffer_next"); - DBUG_PRINT("info", ("horizon (%lu,0x%lx), chasing: %d", + DBUG_PRINT("info", ("horizon: (%lu,0x%lx) chasing: %d", (ulong) LSN_FILE_NO(log_descriptor.horizon), (ulong) LSN_OFFSET(log_descriptor.horizon), chasing)); @@ -1113,8 +1067,8 @@ static my_bool translog_buffer_next(TRANSLOG_ADDRESS *horizon, if (new_file) { /* move the horizon to the next file and its header page */ - *horizon+= LSN_ONE_FILE; - *horizon= LSN_REPLACE_OFFSET(*horizon, TRANSLOG_PAGE_SIZE); + (*horizon)+= LSN_ONE_FILE; + (*horizon)= LSN_REPLACE_OFFSET(*horizon, TRANSLOG_PAGE_SIZE); if (!chasing && translog_create_new_file()) { DBUG_RETURN(1); @@ -1177,22 +1131,17 @@ static void translog_get_sent_to_file(LSN *lsn) RETURN first chunk offset - 0 - Error */ -static my_bool translog_get_first_chunk_offset(uchar *page) +static my_bool translog_get_first_chunk_offset(byte *page) { uint16 page_header= 7; DBUG_ENTER("translog_get_first_chunk_offset"); - if (page[6] & TRANSLOG_PAGE_CRC) - { + if (page[TRANSLOG_PAGE_FLAGS] & TRANSLOG_PAGE_CRC) page_header+= 4; - } - if (page[6] & TRANSLOG_SECTOR_PROTECTION) - { - page_header+= (TRANSLOG_PAGE_SIZE / 512) * 2; - } + if (page[TRANSLOG_PAGE_FLAGS] & TRANSLOG_SECTOR_PROTECTION) + page_header+= (TRANSLOG_PAGE_SIZE / DISK_DRIVE_SECTOR_SIZE) * 2; DBUG_RETURN(page_header); } @@ -1208,7 +1157,7 @@ static my_bool translog_get_first_chunk_offset(uchar *page) */ static void -translog_write_variable_record_1group_code_len(uchar *dst, +translog_write_variable_record_1group_code_len(byte *dst, translog_size_t length, uint16 header_len) { @@ -1223,7 +1172,7 @@ translog_write_variable_record_1group_code_len(uchar *dst, int2store(dst + 1, length); return; case 9: /* (5 + 4) */ - DBUG_ASSERT(length <= 0xFFFFFF); + DBUG_ASSERT(length <= (ulong) 0xFFFFFF); *dst= 252; int3store(dst + 1, length); return; @@ -1249,18 +1198,18 @@ translog_write_variable_record_1group_code_len(uchar *dst, decoded length */ -static translog_size_t translog_variable_record_1group_decode_len(uchar **src) +static translog_size_t translog_variable_record_1group_decode_len(byte **src) { uint8 first= (uint8) (**src); switch (first) { case 251: - *src+= 3; + (*src)+= 3; return (uint2korr((*src) - 2)); case 252: - *src+= 4; + (*src)+= 4; return (uint3korr((*src) - 3)); case 253: - *src+= 5; + (*src)+= 5; return (uint4korr((*src) - 4)); case 254: case 255: @@ -1283,25 +1232,24 @@ static translog_size_t translog_variable_record_1group_decode_len(uchar **src) RETURN total length of the chunk - 0 - Error */ -static uint16 translog_get_total_chunk_length(uchar *page, uint16 offset) +static uint16 translog_get_total_chunk_length(byte *page, uint16 offset) { DBUG_ENTER("translog_get_total_chunk_length"); switch (page[offset] & TRANSLOG_CHUNK_TYPE) { - case TRANSLOG_CHUNK_LSN: /* 0 chunk referred as LSN - (head or tail) */ + case TRANSLOG_CHUNK_LSN: { + /* 0 chunk referred as LSN (head or tail) */ translog_size_t rec_len; - uchar *start= page + offset; - uchar *ptr= start + 1 + 2; + byte *start= page + offset; + byte *ptr= start + 1 + 2; uint16 chunk_len, header_len, page_rest; DBUG_PRINT("info", ("TRANSLOG_CHUNK_LSN")); rec_len= translog_variable_record_1group_decode_len(&ptr); chunk_len= uint2korr(ptr); - header_len= (ptr -start) +2; - DBUG_PRINT("info", ("rec len: %lu, chunk len: %u, header len: %u", + header_len= (ptr -start) + 2; + DBUG_PRINT("info", ("rec len: %lu chunk len: %u header len: %u", (ulong) rec_len, (uint) chunk_len, (uint) header_len)); if (chunk_len) { @@ -1317,11 +1265,14 @@ static uint16 translog_get_total_chunk_length(uchar *page, uint16 offset) DBUG_RETURN(page_rest); break; } - case TRANSLOG_CHUNK_FIXED: /* 1 (pseudo)fixed record (also - LSN) */ + case TRANSLOG_CHUNK_FIXED: { - DBUG_PRINT("info", ("TRANSLOG_CHUNK_FIXED")); + byte *ptr; uint type= page[offset] & TRANSLOG_REC_TYPE; + uint length; + int i; + /* 1 (pseudo)fixed record (also LSN) */ + DBUG_PRINT("info", ("TRANSLOG_CHUNK_FIXED")); DBUG_ASSERT(log_record_type_descriptor[type].class == LOGRECTYPE_FIXEDLENGTH || log_record_type_descriptor[type].class == @@ -1334,32 +1285,31 @@ static uint16 translog_get_total_chunk_length(uchar *page, uint16 offset) DBUG_RETURN(log_record_type_descriptor[type].fixed_length + 3); } { - uchar *ptr= page + offset + 3; /* first compressed LSN */ - int i= 0; - uint length= log_record_type_descriptor[type].fixed_length + 3; - for (; i < log_record_type_descriptor[type].compresed_LSN; i++) + ptr= page + offset + 3; /* first compressed LSN */ + length= log_record_type_descriptor[type].fixed_length + 3; + for (i= 0; i < log_record_type_descriptor[type].compressed_LSN; i++) { /* first 2 bits is length - 2 */ uint len= ((((uint8) (*ptr)) & TRANSLOG_CLSN_LEN_BITS) >> 6) + 2; ptr+= len; - length-= (TRANSLOG_CLSN_MAX_LEN - len); /* subtract economized - bytes */ + /* subtract economized bytes */ + length-= (TRANSLOG_CLSN_MAX_LEN - len); } DBUG_PRINT("info", ("Pseudo-fixed length: %u", length)); DBUG_RETURN(length); } break; } - case TRANSLOG_CHUNK_NOHDR: /* 2 no header chunk (till page - end) */ - DBUG_PRINT("info", ("TRANSLOG_CHUNK_NOHDR, length: %u", + case TRANSLOG_CHUNK_NOHDR: + /* 2 no header chunk (till page end) */ + DBUG_PRINT("info", ("TRANSLOG_CHUNK_NOHDR length: %u", (uint) (TRANSLOG_PAGE_SIZE - offset))); DBUG_RETURN(TRANSLOG_PAGE_SIZE - offset); break; case TRANSLOG_CHUNK_LNGTH: /* 3 chunk with chunk length */ DBUG_PRINT("info", ("TRANSLOG_CHUNK_LNGTH")); DBUG_ASSERT(TRANSLOG_PAGE_SIZE - offset >= 3); - DBUG_PRINT("info", ("Length %u", uint2korr(page + offset + 1) + 3)); + DBUG_PRINT("info", ("length: %u", uint2korr(page + offset + 1) + 3)); DBUG_RETURN(uint2korr(page + offset + 1) + 3); break; default: @@ -1376,8 +1326,8 @@ static uint16 translog_get_total_chunk_length(uchar *page, uint16 offset) buffer This buffer should be flushed RETURN - 0 - OK - 1 - Error + 0 OK + 1 Error */ static my_bool translog_buffer_flush(struct st_translog_buffer *buffer) @@ -1385,20 +1335,18 @@ static my_bool translog_buffer_flush(struct st_translog_buffer *buffer) uint32 i; DBUG_ENTER("translog_buffer_flush"); DBUG_PRINT("enter", - ("Buffer #%u 0x%lx: locked by 0x%lx (0x%lx), " - "file: %u, offset (%lu,0x%lx), size %lu", + ("Buffer: #%u 0x%lx: " + "file: %d offset: (%lu,0x%lx) size: %lu", (uint) buffer->buffer_no, (ulong) buffer, - (ulong) buffer->locked_by, (ulong) my_thread_var, - (uint) buffer->file, + buffer->file, (ulong) LSN_FILE_NO(buffer->offset), (ulong) LSN_OFFSET(buffer->offset), (ulong) buffer->size)); - DBUG_ASSERT(buffer->locked_by == my_thread_var); - DBUG_ASSERT(buffer->file != 0); + DBUG_ASSERT(buffer->file != -1); translog_wait_for_writers(buffer); - if (buffer->overlay && buffer->overlay->file) + if (buffer->overlay && buffer->overlay->file != -1) { struct st_translog_buffer *overlay= buffer->overlay; translog_buffer_unlock(buffer); @@ -1410,10 +1358,8 @@ static my_bool translog_buffer_flush(struct st_translog_buffer *buffer) for (i= 0; i < buffer->size; i+= TRANSLOG_PAGE_SIZE) { - PAGECACHE_FILE file= - { - buffer->file - }; + PAGECACHE_FILE file; + file.file= buffer->file; if (pagecache_write(log_descriptor.pagecache, &file, (LSN_OFFSET(buffer->offset) + i) / TRANSLOG_PAGE_SIZE, @@ -1423,16 +1369,17 @@ static my_bool translog_buffer_flush(struct st_translog_buffer *buffer) PAGECACHE_LOCK_LEFT_UNLOCKED, PAGECACHE_PIN_LEFT_UNPINNED, PAGECACHE_WRITE_DONE, 0)) { - UNRECOVERABLE_ERROR(("Cant't write page (%lu,0x%lx) to pagecacte", + UNRECOVERABLE_ERROR(("Can't write page (%lu,0x%lx) to pagecache", (ulong) buffer->file, (ulong) (LSN_OFFSET(buffer->offset)+ i))); } } if (my_pwrite(buffer->file, (char*) buffer->buffer, buffer->size, LSN_OFFSET(buffer->offset), - MYF(MY_WME)) != buffer->size) + MYF(MY_WME | MY_NABP))) { - UNRECOVERABLE_ERROR(("Cant't buffer (%lu,0x%lx) size %lu to the disk (%d)", + UNRECOVERABLE_ERROR(("Can't write buffer (%lu,0x%lx) size %lu " + "to the disk (%d)", (ulong) buffer->file, (ulong) LSN_OFFSET(buffer->offset), (ulong) buffer->size, errno)); @@ -1440,10 +1387,11 @@ static my_bool translog_buffer_flush(struct st_translog_buffer *buffer) } if (LSN_OFFSET(buffer->last_lsn) != 0) /* if buffer->last_lsn is set */ translog_set_sent_to_file(&buffer->last_lsn); + /* Free buffer */ - buffer->file= 0; + buffer->file= -1; buffer->overlay= 0; - if (buffer->waiting_filling_buffer.last_thread != NULL) + if (buffer->waiting_filling_buffer.last_thread) { wqueue_release_queue(&buffer->waiting_filling_buffer); } @@ -1460,20 +1408,17 @@ static my_bool translog_buffer_flush(struct st_translog_buffer *buffer) offset offset of failed sector RETURN - 0 - OK - 1 - Error + 0 OK + 1 Error */ -static my_bool translog_recover_page_up_to_sector(uchar *page, uint16 offset) +static my_bool translog_recover_page_up_to_sector(byte *page, uint16 offset) { uint16 chunk_offset= translog_get_first_chunk_offset(page), valid_chunk_end; DBUG_ENTER("translog_recover_page_up_to_sector"); - DBUG_PRINT("enter", ("offset %u, first chunk %u", + DBUG_PRINT("enter", ("offset: %u first chunk: %u", (uint) offset, (uint) chunk_offset)); - if (chunk_offset == 0) - DBUG_RETURN(1); - while (page[chunk_offset] != '\0' && chunk_offset < offset) { uint16 chunk_length; @@ -1484,11 +1429,11 @@ static my_bool translog_recover_page_up_to_sector(uchar *page, uint16 offset) (uint) chunk_offset)); DBUG_RETURN(1); } - DBUG_PRINT("info", ("chunk: offset: %u, length %u", + DBUG_PRINT("info", ("chunk: offset: %u length %u", (uint) chunk_offset, (uint) chunk_length)); if (((ulong) chunk_offset) + ((ulong) chunk_length) > TRANSLOG_PAGE_SIZE) { - UNRECOVERABLE_ERROR(("demaged chunk (offset %u) in trusted area", + UNRECOVERABLE_ERROR(("damaged chunk (offset %u) in trusted area", (uint) chunk_offset)); DBUG_RETURN(1); } @@ -1496,21 +1441,20 @@ static my_bool translog_recover_page_up_to_sector(uchar *page, uint16 offset) } valid_chunk_end= chunk_offset; - /*end of trusted area - sector parsing */ + /* end of trusted area - sector parsing */ while (page[chunk_offset] != '\0') { uint16 chunk_length; if ((chunk_length= translog_get_total_chunk_length(page, chunk_offset)) == 0) - { break; - } - DBUG_PRINT("info", ("chunk: offset: %u, length %u", + + DBUG_PRINT("info", ("chunk: offset: %u length %u", (uint) chunk_offset, (uint) chunk_length)); - if (((ulong) chunk_offset) + ((ulong) chunk_length) > (uint) (offset + 512)) - { + if (((ulong) chunk_offset) + ((ulong) chunk_length) > + (uint) (offset + DISK_DRIVE_SECTOR_SIZE)) break; - } + chunk_offset+= chunk_length; valid_chunk_end= chunk_offset; } @@ -1531,23 +1475,25 @@ static my_bool translog_recover_page_up_to_sector(uchar *page, uint16 offset) data data, need for validation (address in this case) RETURN - 0 - OK - 1 - Error + 0 OK + 1 Error */ -static my_bool translog_page_validator(byte *page_addr, gptr data) +static my_bool translog_page_validator(byte *page_addr, gptr data_ptr) { - uint8 flags; - uchar *page= (uchar*) page_addr; + uint this_page_page_overhead; + uint flags; + byte *page= (byte*) page_addr, *page_pos; + TRANSLOG_VALIDATOR_DATA *data= (TRANSLOG_VALIDATOR_DATA *) data_ptr; + TRANSLOG_ADDRESS addr= *(data->addr); DBUG_ENTER("translog_page_validator"); - TRANSLOG_ADDRESS addr= *((TRANSLOG_VALIDATOR_DATA*) data)->addr; - ((TRANSLOG_VALIDATOR_DATA*) data)->was_recovered= 0; + data->was_recovered= 0; if (uint3korr(page) != LSN_OFFSET(addr) / TRANSLOG_PAGE_SIZE || uint3korr(page + 3) != LSN_FILE_NO(addr)) { UNRECOVERABLE_ERROR(("Page (%lu,0x%lx): " - "page address written in the page is incorrect :" + "page address written in the page is incorrect: " "File %lu instead of %lu or page %lu instead of %lu", (ulong) LSN_FILE_NO(addr), (ulong) LSN_OFFSET(addr), (ulong) uint3korr(page + 3), (ulong) LSN_FILE_NO(addr), @@ -1555,7 +1501,8 @@ static my_bool translog_page_validator(byte *page_addr, gptr data) (ulong) LSN_OFFSET(addr) / TRANSLOG_PAGE_SIZE)); DBUG_RETURN(1); } - flags= page[3 + 3]; + flags= (uint)(page[TRANSLOG_PAGE_FLAGS]); + this_page_page_overhead= page_overhead[flags]; if (flags & ~(TRANSLOG_PAGE_CRC | TRANSLOG_SECTOR_PROTECTION | TRANSLOG_RECORD_CRC)) { @@ -1565,65 +1512,58 @@ static my_bool translog_page_validator(byte *page_addr, gptr data) (uint) flags)); DBUG_RETURN(1); } + page_pos= page + (3 + 3 + 1); if (flags & TRANSLOG_PAGE_CRC) { - uint32 crc= translog_adler_crc(page + log_descriptor.page_overhead, - TRANSLOG_PAGE_SIZE - - log_descriptor.page_overhead); - if (crc != uint4korr(page + 3 + 3 + 1)) + uint32 crc= translog_crc(page + this_page_page_overhead, + TRANSLOG_PAGE_SIZE - + this_page_page_overhead); + if (crc != uint4korr(page_pos)) { UNRECOVERABLE_ERROR(("Page (%lu,0x%lx): " "CRC mismatch: calculated: %lx on the page %lx", (ulong) LSN_FILE_NO(addr), (ulong) LSN_OFFSET(addr), - (ulong) crc, (ulong) uint4korr(page + 3 + 3 + 1))); + (ulong) crc, (ulong) uint4korr(page_pos))); DBUG_RETURN(1); } + page_pos+= CRC_LENGTH; /* Skip crc */ } if (flags & TRANSLOG_SECTOR_PROTECTION) { uint i, offset; - uchar *table= (page + 3 + 3 + 1 + ((flags & TRANSLOG_PAGE_CRC) ? 4 : 0)); + byte *table= page_pos; uint16 current= uint2korr(table); - for (i= 2, offset= 512; - i < (TRANSLOG_PAGE_SIZE / 512) * 2; i+= 2, offset+= 512) + for (i= 2, offset= DISK_DRIVE_SECTOR_SIZE; + i < (TRANSLOG_PAGE_SIZE / DISK_DRIVE_SECTOR_SIZE) * 2; + i+= 2, offset+= DISK_DRIVE_SECTOR_SIZE) { /* - TODO: add cunk counting for "suspecting" sectors (difference is - more that 1-2) + TODO: add chunk counting for "suspecting" sectors (difference is + more than 1-2) */ uint16 test= uint2korr(page + offset); - DBUG_PRINT("info", ("sector #%u offset %u current %lx " - "read 0x%x stored 0x%x%x", + DBUG_PRINT("info", ("sector: #%u offset: %u current: %lx " + "read: 0x%x stored: 0x%x%x", i / 2, offset, (ulong) current, (uint) uint2korr(page + offset), (uint) table[i], (uint) table[i + 1])); - if (test < current) - { - if (0xFFFFLL - current + test > 512 / 3) - { - /* it is not overflow */ - if (translog_recover_page_up_to_sector(page, offset)) - DBUG_RETURN(1); - ((TRANSLOG_VALIDATOR_DATA*) data)->was_recovered= 1; - DBUG_RETURN(0); - } - } - else if (test - current > 512 / 3) + if (((test < current) && + (LL(0xFFFF) - current + test > DISK_DRIVE_SECTOR_SIZE / 3)) || + ((test >= current) && + (test - current > DISK_DRIVE_SECTOR_SIZE / 3))) { if (translog_recover_page_up_to_sector(page, offset)) DBUG_RETURN(1); - ((TRANSLOG_VALIDATOR_DATA*) data)->was_recovered= 1; + data->was_recovered= 1; DBUG_RETURN(0); } /* Return value on the page */ page[offset]= table[i]; page[offset + 1]= table[i + 1]; - /**((uint16*)page + offset)= *((uint16*)(table + i));*/ - current= test; - DBUG_PRINT("info", ("sector #%u offset %u current %lx " - "read 0x%x stored 0x%x%x", + DBUG_PRINT("info", ("sector: #%u offset: %u current: %lx " + "read: 0x%x stored: 0x%x%x", i / 2, offset, (ulong) current, (uint) uint2korr(page + offset), (uint) table[i], (uint) table[i + 1])); @@ -1643,17 +1583,17 @@ static my_bool translog_page_validator(byte *page_addr, gptr data) (might not be used in some cache implementations) RETURN - pointer to the page cache which should be used to read this page NULL - Error + # pointer to the page cache which should be used to read this page */ -static uchar *translog_get_page(TRANSLOG_VALIDATOR_DATA *data, uchar *buffer) +static byte *translog_get_page(TRANSLOG_VALIDATOR_DATA *data, byte *buffer) { TRANSLOG_ADDRESS addr= *(data->addr); uint cache_index; uint32 file_no= LSN_FILE_NO(addr); DBUG_ENTER("translog_get_page"); - DBUG_PRINT("enter", ("File %lu, Offset %lu(0x%lx)", + DBUG_PRINT("enter", ("File: %lu Offset: %lu(0x%lx)", (ulong) file_no, (ulong) LSN_OFFSET(addr), (ulong) LSN_OFFSET(addr))); @@ -1666,17 +1606,15 @@ static uchar *translog_get_page(TRANSLOG_VALIDATOR_DATA *data, uchar *buffer) { PAGECACHE_FILE file; /* file in the cache */ - if (log_descriptor.log_file_num[cache_index] == 0) + if (log_descriptor.log_file_num[cache_index] == -1) { if ((log_descriptor.log_file_num[cache_index]= - open_logfile_by_number_no_cache(file_no)) == 0) - { + open_logfile_by_number_no_cache(file_no)) == -1) DBUG_RETURN(NULL); - } } file.file= log_descriptor.log_file_num[cache_index]; - buffer= (uchar*) + buffer= (byte*) pagecache_valid_read(log_descriptor.pagecache, &file, LSN_OFFSET(addr) / TRANSLOG_PAGE_SIZE, 3, (char*) buffer, @@ -1686,7 +1624,17 @@ static uchar *translog_get_page(TRANSLOG_VALIDATOR_DATA *data, uchar *buffer) } else { + /* + TODO: WE KEEP THE LAST OPENED_FILES_NUM FILES IN THE LOG CACHE, NOT + THE LAST USED FILES. THIS WILL BE A NOTABLE PROBLEM IF WE ARE + FOLLOWING AN UNDO CHAIN THAT GOES OVER MANY OLD LOG FILES. WE WILL + PROBABLY NEED SPECIAL HANDLING OF THIS OR HAVE A FILO FOR THE LOG + FILES. + */ + File file= open_logfile_by_number_no_cache(file_no); + if (file == -1) + DBUG_RETURN(NULL); if (my_pread(file, (char*) buffer, TRANSLOG_PAGE_SIZE, LSN_OFFSET(addr), MYF(MY_FNABP | MY_WME))) buffer= NULL; @@ -1708,8 +1656,8 @@ static uchar *translog_get_page(TRANSLOG_VALIDATOR_DATA *data, uchar *buffer) last_page_ok assigned 1 if last page was OK RETURN - 0 - OK - 1 - Error + 0 OK + 1 Error */ static my_bool translog_get_last_page_addr(TRANSLOG_ADDRESS *addr, @@ -1721,11 +1669,10 @@ static my_bool translog_get_last_page_addr(TRANSLOG_ADDRESS *addr, uint32 file_no= LSN_FILE_NO(*addr); DBUG_ENTER("translog_get_last_page_addr"); - if ((stat= my_stat (translog_filename_by_fileno(file_no, - path), - &stat_buff, MYF(MY_WME))) == NULL) + if (!(stat= my_stat(translog_filename_by_fileno(file_no, path), + &stat_buff, MYF(MY_WME)))) DBUG_RETURN(1); - DBUG_PRINT("info", ("File size %lu", (ulong) stat->st_size)); + DBUG_PRINT("info", ("File size: %lu", (ulong) stat->st_size)); if (stat->st_size > TRANSLOG_PAGE_SIZE) { rec_offset= (((stat->st_size / TRANSLOG_PAGE_SIZE) - 1) * @@ -1738,7 +1685,7 @@ static my_bool translog_get_last_page_addr(TRANSLOG_ADDRESS *addr, rec_offset= 0; } *addr= MAKE_LSN(file_no, rec_offset); - DBUG_PRINT("info", ("Last page: 0x%lx, ok %d", (ulong) rec_offset, + DBUG_PRINT("info", ("Last page: 0x%lx ok: %d", (ulong) rec_offset, *last_page_ok)); DBUG_RETURN(0); } @@ -1759,9 +1706,9 @@ static uint translog_variable_record_length_bytes(translog_size_t length) { if (length < 250) return 1; - else if (length < 0xFFFF) + if (length < 0xFFFF) return 3; - else if (length < 0xFFFFFF) + if (length < (ulong) 0xFFFFFF) return 4; return 5; } @@ -1776,47 +1723,49 @@ static uint translog_variable_record_length_bytes(translog_size_t length) offset Offset of the chunk on this place RETURN - total length of the chunk - 0 - Error + # total length of the chunk + 0 Error */ -static uint16 translog_get_chunk_header_length(uchar *page, uint16 offset) +static uint16 translog_get_chunk_header_length(byte *page, uint16 offset) { DBUG_ENTER("translog_get_chunk_header_length"); - switch (page[offset] & TRANSLOG_CHUNK_TYPE) { - case TRANSLOG_CHUNK_LSN: /* 0 chunk referred as LSN - (head or tail) */ + page+= offset; + switch (*page & TRANSLOG_CHUNK_TYPE) { + case TRANSLOG_CHUNK_LSN: { + /* 0 chunk referred as LSN (head or tail) */ translog_size_t rec_len; - uchar *start= page + offset; - uchar *ptr= start + 1 + 2; + byte *start= page; + byte *ptr= start + 1 + 2; uint16 chunk_len, header_len; DBUG_PRINT("info", ("TRANSLOG_CHUNK_LSN")); rec_len= translog_variable_record_1group_decode_len(&ptr); chunk_len= uint2korr(ptr); - header_len= (ptr -start) +2; - DBUG_PRINT("info", ("rec len: %lu, chunk len: %u, header len: %u", + header_len= (ptr - start) +2; + DBUG_PRINT("info", ("rec len: %lu chunk len: %u header len: %u", (ulong) rec_len, (uint) chunk_len, (uint) header_len)); if (chunk_len) { - /*TODO: fine header end */ + /* TODO: fine header end */ DBUG_ASSERT(0); } DBUG_RETURN(header_len); break; } - case TRANSLOG_CHUNK_FIXED: /* 1 (pseudo)fixed record (also - LSN) */ + case TRANSLOG_CHUNK_FIXED: { + /* 1 (pseudo)fixed record (also LSN) */ DBUG_PRINT("info", ("TRANSLOG_CHUNK_FIXED = 3")); DBUG_RETURN(3); } - case TRANSLOG_CHUNK_NOHDR: /* 2 no header chunk (till page - end) */ + case TRANSLOG_CHUNK_NOHDR: + /* 2 no header chunk (till page end) */ DBUG_PRINT("info", ("TRANSLOG_CHUNK_NOHDR = 1")); DBUG_RETURN(1); break; - case TRANSLOG_CHUNK_LNGTH: /* 3 chunk with chunk length */ + case TRANSLOG_CHUNK_LNGTH: + /* 3 chunk with chunk length */ DBUG_PRINT("info", ("TRANSLOG_CHUNK_LNGTH = 3")); DBUG_RETURN(3); break; @@ -1840,8 +1789,8 @@ static uint16 translog_get_chunk_header_length(uchar *page, uint16 offset) TRANSLOG_RECORD_CRC) RETURN - 0 - OK - 1 - Error + 0 OK + 1 Error */ my_bool translog_init(const char *directory, @@ -1851,11 +1800,13 @@ my_bool translog_init(const char *directory, { int i; int old_log_was_recovered= 0, logs_found= 0; + uint old_flags= flags; TRANSLOG_ADDRESS sure_page, last_page, last_valid_page; DBUG_ENTER("translog_init"); - if (pthread_mutex_init(&log_descriptor.sent_to_file_lock, MY_MUTEX_INIT_FAST)) + if (pthread_mutex_init(&log_descriptor.sent_to_file_lock, + MY_MUTEX_INIT_FAST)) DBUG_RETURN(1); /* Directory to store files */ @@ -1883,11 +1834,16 @@ my_bool translog_init(const char *directory, ~(TRANSLOG_PAGE_CRC | TRANSLOG_SECTOR_PROTECTION | TRANSLOG_RECORD_CRC)) == 0); log_descriptor.flags= flags; - log_descriptor.page_overhead= 7; - if (flags & TRANSLOG_PAGE_CRC) - log_descriptor.page_overhead+= 4; - if (flags & TRANSLOG_SECTOR_PROTECTION) - log_descriptor.page_overhead+= (TRANSLOG_PAGE_SIZE / 512) * 2; + for (i= 0; i < TRANSLOG_FLAGS_NUM; i++) + { + page_overhead[i]= 7; + if (i & TRANSLOG_PAGE_CRC) + page_overhead[i]+= CRC_LENGTH; + if (i & TRANSLOG_SECTOR_PROTECTION) + page_overhead[i]+= (TRANSLOG_PAGE_SIZE / + DISK_DRIVE_SECTOR_SIZE) * 2; + } + log_descriptor.page_overhead= page_overhead[flags]; log_descriptor.page_capacity_chunk_2= TRANSLOG_PAGE_SIZE - log_descriptor.page_overhead - 1; DBUG_ASSERT(TRANSLOG_WRITE_BUFFER % TRANSLOG_PAGE_SIZE == 0); @@ -1897,7 +1853,7 @@ my_bool translog_init(const char *directory, log_descriptor.half_buffer_capacity_chunk_2= log_descriptor.buffer_capacity_chunk_2 / 2; DBUG_PRINT("info", - ("Overhead: %u, pc2: %u, bc2: %u, bc2/2: %u", + ("Overhead: %u pc2: %u bc2: %u, bc2/2: %u", log_descriptor.page_overhead, log_descriptor.page_capacity_chunk_2, log_descriptor.buffer_capacity_chunk_2, @@ -1907,9 +1863,7 @@ my_bool translog_init(const char *directory, /* Init log handler file handlers cache */ for (i= 0; i < OPENED_FILES_NUM; i++) - { - log_descriptor.log_file_num[i]= 0; - } + log_descriptor.log_file_num[i]= -1; /* just to init it somehow */ translog_start_buffer(log_descriptor.buffers, &log_descriptor.bc, 0); @@ -1917,12 +1871,13 @@ my_bool translog_init(const char *directory, /* Buffers for log writing */ for (i= 0; i < TRANSLOG_BUFFERS_NO; i++) { + if (translog_buffer_init(log_descriptor.buffers + i)) + DBUG_RETURN(1); #ifndef DBUG_OFF log_descriptor.buffers[i].buffer_no= (uint8) i; - log_descriptor.buffers[i].locked_by= NULL; #endif - if (translog_buffer_init(log_descriptor.buffers + i)) - DBUG_RETURN(1); + DBUG_PRINT("info", ("translog_buffer buffer #%u: 0x%lx", + i, (ulong) log_descriptor.buffers + i)); } logs_found= (last_logno != CONTROL_FILE_IMPOSSIBLE_FILENO); @@ -1931,12 +1886,12 @@ my_bool translog_init(const char *directory, { my_bool pageok; /* - TODO: scan directory for maria_log.XXXXXXXX files and find + TODO: scan directory for maria_log.XXXXXXXX files and find highest XXXXXXXX & set logs_found - */ + TODO: check that last checkpoint within present log addresses space - /* TODO: check that last checkpoint within present log addresses space */ - /* find the log end */ + find the log end + */ if (LSN_FILE_NO(last_checkpoint_lsn) == CONTROL_FILE_IMPOSSIBLE_FILENO) { DBUG_ASSERT(LSN_OFFSET(last_checkpoint_lsn) == 0); @@ -1994,17 +1949,16 @@ my_bool translog_init(const char *directory, } do { - TRANSLOG_VALIDATOR_DATA data= - { - ¤t_page, 0 - }; - uchar buffer[TRANSLOG_PAGE_SIZE], *page; + TRANSLOG_VALIDATOR_DATA data; + byte buffer[TRANSLOG_PAGE_SIZE], *page; + data.addr= ¤t_page; if ((page= translog_get_page(&data, buffer)) == NULL) DBUG_RETURN(1); if (data.was_recovered) { - DBUG_PRINT("error", ("file no %u (%d), rec_offset 0x%lx (%lu) (%d)", - (uint) LSN_FILE_NO(current_page), + DBUG_PRINT("error", ("file no: %lu (%d) " + "rec_offset: 0x%lx (%lu) (%d)", + (ulong) LSN_FILE_NO(current_page), (uint3korr(page + 3) != LSN_FILE_NO(current_page)), (ulong) LSN_OFFSET(current_page), @@ -2016,6 +1970,7 @@ my_bool translog_init(const char *directory, old_log_was_recovered= 1; break; } + old_flags= page[TRANSLOG_PAGE_FLAGS]; last_valid_page= current_page; current_page+= TRANSLOG_PAGE_SIZE; /* increase offset */ } while (current_page <= current_file_last_page); @@ -2029,27 +1984,27 @@ my_bool translog_init(const char *directory, /* TODO: issue error */ DBUG_RETURN(1); } - DBUG_PRINT("info", ("Last valid page is in file %lu offset %lu (0x%lx), " - "Logs found: %d, was recovered: %d", + DBUG_PRINT("info", ("Last valid page is in file: %lu " + "offset: %lu (0x%lx) " + "Logs found: %d was recovered: %d " + "flags match: %d", (ulong) LSN_FILE_NO(last_valid_page), (ulong) LSN_OFFSET(last_valid_page), (ulong) LSN_OFFSET(last_valid_page), - logs_found, old_log_was_recovered)); + logs_found, old_log_was_recovered, + (old_flags == flags))); /* TODO: check server ID */ - if (logs_found && !old_log_was_recovered) + if (logs_found && !old_log_was_recovered && old_flags == flags) { - TRANSLOG_VALIDATOR_DATA data= - { - &last_valid_page, 0 - }; - uchar buffer[TRANSLOG_PAGE_SIZE], *page; + TRANSLOG_VALIDATOR_DATA data; + byte buffer[TRANSLOG_PAGE_SIZE], *page; uint16 chunk_offset; + data.addr= &last_valid_page; /* continue old log */ DBUG_ASSERT(LSN_FILE_NO(last_valid_page)== LSN_FILE_NO(log_descriptor.horizon)); - if ((page= translog_get_page(&data, - buffer)) == NULL || + if ((page= translog_get_page(&data, buffer)) == NULL || (chunk_offset= translog_get_first_chunk_offset(page)) == 0) DBUG_RETURN(1); @@ -2066,38 +2021,37 @@ my_bool translog_init(const char *directory, if ((chunk_length= translog_get_total_chunk_length(page, chunk_offset)) == 0) DBUG_RETURN(1); - DBUG_PRINT("info", ("chunk: offset: %u, length %u", + DBUG_PRINT("info", ("chunk: offset: %u length: %u", (uint) chunk_offset, (uint) chunk_length)); chunk_offset+= chunk_length; /* chunk can't cross the page border */ DBUG_ASSERT(chunk_offset <= TRANSLOG_PAGE_SIZE); } - memmove(log_descriptor.buffers->buffer, page, chunk_offset); + memcpy(log_descriptor.buffers->buffer, page, chunk_offset); log_descriptor.bc.buffer->size+= chunk_offset; log_descriptor.bc.ptr+= chunk_offset; - log_descriptor.bc.current_page_size= chunk_offset; + log_descriptor.bc.current_page_fill= chunk_offset; log_descriptor.horizon= LSN_REPLACE_OFFSET(log_descriptor.horizon, (chunk_offset + LSN_OFFSET(last_valid_page))); - DBUG_PRINT("info", ("Move Page #%u: 0x%lx, chaser: %d, Size: %lu (%lu)", + DBUG_PRINT("info", ("Move Page #%u: 0x%lx chaser: %d Size: %lu (%lu)", (uint) log_descriptor.bc.buffer_no, (ulong) log_descriptor.bc.buffer, log_descriptor.bc.chaser, (ulong) log_descriptor.bc.buffer->size, - (ulong) (log_descriptor.bc.ptr -log_descriptor.bc. + (ulong) (log_descriptor.bc.ptr - log_descriptor.bc. buffer->buffer))); - DBUG_ASSERT(log_descriptor.bc.chaser - || + DBUG_ASSERT(log_descriptor.bc.chaser || ((ulong) (log_descriptor.bc.ptr -log_descriptor.bc.buffer->buffer) == log_descriptor.bc.buffer->size)); DBUG_ASSERT(log_descriptor.bc.buffer->buffer_no == log_descriptor.bc.buffer_no); - DBUG_ASSERT(log_descriptor.bc.current_page_size <= TRANSLOG_PAGE_SIZE); + DBUG_ASSERT(log_descriptor.bc.current_page_fill <= TRANSLOG_PAGE_SIZE); } } - DBUG_PRINT("info", ("Logs found: %d, was recovered %d", + DBUG_PRINT("info", ("Logs found: %d was recovered: %d", logs_found, old_log_was_recovered)); if (!logs_found) { @@ -2105,9 +2059,9 @@ my_bool translog_init(const char *directory, /* Used space */ log_descriptor.horizon= MAKE_LSN(1, TRANSLOG_PAGE_SIZE); // header page /* Current logs file number in page cache */ - log_descriptor.log_file_num[0]= - open_logfile_by_number_no_cache(1); - if (translog_write_file_header()) + if ((log_descriptor.log_file_num[0]= + open_logfile_by_number_no_cache(1)) == -1 || + translog_write_file_header()) DBUG_RETURN(1); if (ma_control_file_write_and_force(CONTROL_FILE_IMPOSSIBLE_LSN, 1, CONTROL_FILE_UPDATE_ONLY_LOGNO)) @@ -2116,44 +2070,29 @@ my_bool translog_init(const char *directory, translog_start_buffer(log_descriptor.buffers, &log_descriptor.bc, 0); translog_new_page_header(&log_descriptor.horizon, &log_descriptor.bc); } - else if (old_log_was_recovered) - { - int buffer_touched= log_descriptor.bc.buffer->file; - if (buffer_touched) - { - struct st_translog_buffer *buffer= log_descriptor.bc.buffer; - /* - We are in initialization so we can use translog_buffer_lock instead - of translog_lock, because there is no other threads which can lock - the loghandler. - */ - if (translog_buffer_lock(buffer) || - translog_buffer_next(&log_descriptor.horizon, &log_descriptor.bc, - 1) || - translog_buffer_unlock(log_descriptor.bc.buffer) || - translog_buffer_flush(buffer) || translog_buffer_unlock(buffer)) - DBUG_RETURN(1); - } - else - { - /* leave the demaged file untouched */ - log_descriptor.horizon+= LSN_ONE_FILE; - /* header page */ - log_descriptor.horizon= LSN_REPLACE_OFFSET(log_descriptor.horizon, - TRANSLOG_PAGE_SIZE); - if (translog_create_new_file()) - DBUG_RETURN(1); - /* - Buffer system left untouched after recovery => we should init it - (starting from buffer 0) - */ - translog_start_buffer(log_descriptor.buffers, &log_descriptor.bc, 0); - translog_new_page_header(&log_descriptor.horizon, &log_descriptor.bc); - } + else if (old_log_was_recovered || old_flags != flags) + { + /* leave the damaged file untouched */ + log_descriptor.horizon+= LSN_ONE_FILE; + /* header page */ + log_descriptor.horizon= LSN_REPLACE_OFFSET(log_descriptor.horizon, + TRANSLOG_PAGE_SIZE); + if (translog_create_new_file()) + DBUG_RETURN(1); + /* + Buffer system left untouched after recovery => we should init it + (starting from buffer 0) + */ + translog_start_buffer(log_descriptor.buffers, &log_descriptor.bc, 0); + translog_new_page_header(&log_descriptor.horizon, &log_descriptor.bc); } /* all LSNs that are on disk are flushed */ log_descriptor.sent_to_file= log_descriptor.flushed= log_descriptor.horizon; + /* + horizon is (potentially) address of the next LSN we need decrease + it to signal that all LSNs before it are flushed + */ log_descriptor.flushed--; /* offset decreased */ log_descriptor.sent_to_file--; /* offset decreased */ @@ -2169,31 +2108,31 @@ my_bool translog_init(const char *directory, buffer_no The buffer to free NOTE - This buffer should be locked; + This buffer should be locked */ static void translog_buffer_destroy(struct st_translog_buffer *buffer) { DBUG_ENTER("translog_buffer_destroy"); DBUG_PRINT("enter", - ("Buffer #%u: 0x%lx, file: %u, offset (%lu,0x%lx), size %lu", + ("Buffer #%u: 0x%lx file: %d offset: (%lu,0x%lx) size: %lu", (uint) buffer->buffer_no, (ulong) buffer, - (uint) buffer->file, + buffer->file, (ulong) LSN_FILE_NO(buffer->offset), (ulong) LSN_OFFSET(buffer->offset), (ulong) buffer->size)); DBUG_ASSERT(buffer->waiting_filling_buffer.last_thread == 0); - if (buffer->file) + if (buffer->file != -1) { /* - We ignore error here, because we can't do something about it + We ignore errors here, because we can't do something about it (it is shutting down) */ translog_buffer_flush(buffer); } - DBUG_PRINT("info", ("Unlock mutex 0x%lx", (ulong) &buffer->mutex)); + DBUG_PRINT("info", ("Unlock mutex: 0x%lx", (ulong) &buffer->mutex)); pthread_mutex_unlock(&buffer->mutex); - DBUG_PRINT("info", ("Destroy mutex 0x%lx", (ulong) &buffer->mutex)); + DBUG_PRINT("info", ("Destroy mutex: 0x%lx", (ulong) &buffer->mutex)); pthread_mutex_destroy(&buffer->mutex); DBUG_VOID_RETURN; } @@ -2208,21 +2147,25 @@ static void translog_buffer_destroy(struct st_translog_buffer *buffer) void translog_destroy() { - int i; + uint i; DBUG_ENTER("translog_destroy"); - if (log_descriptor.bc.buffer->file != 0) + if (log_descriptor.bc.buffer->file != -1) translog_finish_page(&log_descriptor.horizon, &log_descriptor.bc); for (i= 0; i < TRANSLOG_BUFFERS_NO; i++) { struct st_translog_buffer *buffer= log_descriptor.buffers + i; + /* + Lock the buffer just for safety, there should not be other + threads running. + */ translog_buffer_lock(buffer); translog_buffer_destroy(buffer); } /* close files */ for (i= 0; i < OPENED_FILES_NUM; i++) { - if (log_descriptor.log_file_num[i]) + if (log_descriptor.log_file_num[i] != -1) translog_close_log_file(log_descriptor.log_file_num[i]); } pthread_mutex_destroy(&log_descriptor.sent_to_file_lock); @@ -2238,8 +2181,8 @@ void translog_destroy() translog_lock() RETURN - 0 - OK - 1 - Error + 0 OK + 1 Error */ static my_bool translog_lock() @@ -2248,7 +2191,7 @@ static my_bool translog_lock() DBUG_ENTER("translog_lock"); /* - locking the loghandler mean locking current buffer, but it can change + Locking the loghandler mean locking current buffer, but it can change during locking, so we should check it */ for (;;) @@ -2271,22 +2214,18 @@ static my_bool translog_lock() translog_unlock() RETURN - 0 - OK - 1 - Error + 0 OK + 1 Error */ -#ifndef DBUG_OFF -static my_bool translog_unlock() +static inline my_bool translog_unlock() { DBUG_ENTER("translog_unlock"); translog_buffer_unlock(log_descriptor.bc.buffer); DBUG_RETURN(0); } -#else -#define translog_unlock() \ - translog_buffer_unlock(log_descriptor.bc.buffer); -#endif + /* Start new page @@ -2296,14 +2235,14 @@ static my_bool translog_unlock() horizon \ Position in file and buffer where we are cursor / prev_buffer Buffer which should be flushed will be assigned - here if it is need + here if it is need. This is always set. NOTE handler should be locked RETURN - 0 - OK - 1 - Error + 0 OK + 1 Error */ static my_bool translog_page_next(TRANSLOG_ADDRESS *horizon, @@ -2318,10 +2257,10 @@ static my_bool translog_page_next(TRANSLOG_ADDRESS *horizon, (LSN_OFFSET(*horizon) > log_descriptor.log_file_max_size - TRANSLOG_PAGE_SIZE)) { - DBUG_PRINT("info", ("Switch to next buffer, Buffer Size %lu (%lu) => %d, " - "File size %lu max %lu => %d", + DBUG_PRINT("info", ("Switch to next buffer Buffer Size: %lu (%lu) => %d " + "File size: %lu max: %lu => %d", (ulong) cursor->buffer->size, - (ulong) (cursor->ptr -cursor->buffer->buffer), + (ulong) (cursor->ptr - cursor->buffer->buffer), (cursor->ptr + TRANSLOG_PAGE_SIZE > cursor->buffer->buffer + TRANSLOG_WRITE_BUFFER), (ulong) LSN_OFFSET(*horizon), @@ -2335,17 +2274,17 @@ static my_bool translog_page_next(TRANSLOG_ADDRESS *horizon, TRANSLOG_PAGE_SIZE))) DBUG_RETURN(1); *prev_buffer= buffer; - DBUG_PRINT("info", ("Buffer #%u (0x%lu) have to be flushed", + DBUG_PRINT("info", ("Buffer #%u (0x%lu): have to be flushed", (uint) buffer->buffer_no, (ulong) buffer)); } else { - DBUG_PRINT("info", ("Use the same buffer #%u (0x%lu), " - "Buffer Size %lu (%lu)", + DBUG_PRINT("info", ("Use the same buffer #%u (0x%lu): " + "Buffer Size: %lu (%lu)", (uint) buffer->buffer_no, (ulong) buffer, (ulong) cursor->buffer->size, - (ulong) (cursor->ptr -cursor->buffer->buffer))); + (ulong) (cursor->ptr - cursor->buffer->buffer))); translog_finish_page(horizon, cursor); translog_new_page_header(horizon, cursor); *prev_buffer= NULL; @@ -2365,39 +2304,39 @@ static my_bool translog_page_next(TRANSLOG_ADDRESS *horizon, buffer buffer with data RETURN - 0 - OK - 1 - Error + 0 OK + 1 Error */ static my_bool translog_write_data_on_page(TRANSLOG_ADDRESS *horizon, struct st_buffer_cursor *cursor, translog_size_t length, - uchar *buffer) + byte *buffer) { DBUG_ENTER("translog_write_data_on_page"); - DBUG_PRINT("enter", ("Chunk length: %lu Page size %u", - (ulong) length, (uint) cursor->current_page_size)); + DBUG_PRINT("enter", ("Chunk length: %lu Page size %u", + (ulong) length, (uint) cursor->current_page_fill)); DBUG_ASSERT(length > 0); - DBUG_ASSERT(length + cursor->current_page_size <= TRANSLOG_PAGE_SIZE); + DBUG_ASSERT(length + cursor->current_page_fill <= TRANSLOG_PAGE_SIZE); DBUG_ASSERT(length + cursor->ptr <=cursor->buffer->buffer + TRANSLOG_WRITE_BUFFER); - memmove(cursor->ptr, buffer, length); + memcpy(cursor->ptr, buffer, length); cursor->ptr+= length; - *horizon+= length; /* adds offset */ - cursor->current_page_size+= length; + (*horizon)+= length; /* adds offset */ + cursor->current_page_fill+= length; if (!cursor->chaser) cursor->buffer->size+= length; - DBUG_PRINT("info", ("Write data buffer #%u: 0x%lx," - "chaser: %d, Size: %lu (%lu)", + DBUG_PRINT("info", ("Write data buffer #%u: 0x%lx " + "chaser: %d Size: %lu (%lu)", (uint) cursor->buffer->buffer_no, (ulong) cursor->buffer, cursor->chaser, (ulong) cursor->buffer->size, - (ulong) (cursor->ptr -cursor->buffer->buffer))); + (ulong) (cursor->ptr - cursor->buffer->buffer))); DBUG_ASSERT(cursor->chaser || - ((ulong) (cursor->ptr -cursor->buffer->buffer) == + ((ulong) (cursor->ptr - cursor->buffer->buffer) == cursor->buffer->size)); DBUG_ASSERT(cursor->buffer->buffer_no == cursor->buffer_no); - DBUG_ASSERT(cursor->current_page_size <= TRANSLOG_PAGE_SIZE); + DBUG_ASSERT(cursor->current_page_fill <= TRANSLOG_PAGE_SIZE); DBUG_RETURN(0); } @@ -2414,8 +2353,8 @@ static my_bool translog_write_data_on_page(TRANSLOG_ADDRESS *horizon, parts IN/OUT chunk source RETURN - 0 - OK - 1 - Error + 0 OK + 1 Error */ static my_bool translog_write_parts_on_page(TRANSLOG_ADDRESS *horizon, @@ -2426,80 +2365,82 @@ static my_bool translog_write_parts_on_page(TRANSLOG_ADDRESS *horizon, translog_size_t left= length; uint cur= (uint) parts->current; DBUG_ENTER("translog_write_parts_on_page"); - DBUG_PRINT("enter", ("Chunk length: %lu, parts %u of %u. Page size %u, " + DBUG_PRINT("enter", ("Chunk length: %lu parts: %u of %u. Page size: %u " "Buffer size: %lu (%lu)", (ulong) length, (uint) (cur + 1), (uint) parts->parts.elements, - (uint) cursor->current_page_size, + (uint) cursor->current_page_fill, (ulong) cursor->buffer->size, - (ulong) (cursor->ptr -cursor->buffer->buffer))); + (ulong) (cursor->ptr - cursor->buffer->buffer))); DBUG_ASSERT(length > 0); - DBUG_ASSERT(length + cursor->current_page_size <= TRANSLOG_PAGE_SIZE); + DBUG_ASSERT(length + cursor->current_page_fill <= TRANSLOG_PAGE_SIZE); DBUG_ASSERT(length + cursor->ptr <=cursor->buffer->buffer + TRANSLOG_WRITE_BUFFER); do { translog_size_t len; - struct st_translog_part part; - uchar *buff; + struct st_translog_part *part; + byte *buff; DBUG_ASSERT(cur < parts->parts.elements); - get_dynamic(&parts->parts, (gptr) &part, cur); - buff= part.buff; - DBUG_PRINT("info", ("Part %u, Length: %lu, left: %lu", - (uint) (cur + 1), (ulong) part.len, (ulong) left)); + part= dynamic_element(&parts->parts, cur, struct st_translog_part *); + buff= part->buff; + DBUG_PRINT("info", ("Part: %u Length: %lu left: %lu", + (uint) (cur + 1), (ulong) part->len, (ulong) left)); - if (part.len > left) + if (part->len > left) { /* we should write less then the current part */ len= left; - part.len-= len; - part.buff+= len; - if (set_dynamic(&parts->parts, (gptr) &part, cur)) - DBUG_RETURN(1); - DBUG_PRINT("info", ("Set new part %u, Length: %lu", - (uint) (cur + 1), (ulong) part.len)); + part->len-= len; + part->buff+= len; + DBUG_PRINT("info", ("Set new part: %u Length: %lu", + (uint) (cur + 1), (ulong) part->len)); } else { - len= part.len; + len= part->len; cur++; DBUG_PRINT("info", ("moved to next part (len: %lu)", (ulong) len)); } DBUG_PRINT("info", ("copy: 0x%lx <- 0x%lx %u", (ulong) cursor->ptr, (ulong)buff, (uint)len)); - memmove(cursor->ptr, buff, len); + memcpy(cursor->ptr, buff, len); left-= len; cursor->ptr+= len; } while (left); - DBUG_PRINT("info", ("Horizon (%lu,0x%lx) Length %lu(0x%lx)", + DBUG_PRINT("info", ("Horizon: (%lu,0x%lx) Length %lu(0x%lx)", (ulong) LSN_FILE_NO(*horizon), (ulong) LSN_OFFSET(*horizon), (ulong) length, (ulong) length)); parts->current= cur; - *horizon+= length; /* offset increasing */ - cursor->current_page_size+= length; + (*horizon)+= length; /* offset increasing */ + cursor->current_page_fill+= length; if (!cursor->chaser) cursor->buffer->size+= length; DBUG_PRINT("info", ("Write parts buffer #%u: 0x%lx " "chaser: %d Size: %lu (%lu) " - "Horizon (%lu,0x%lx) buff offset 0x%lx", + "Horizon: (%lu,0x%lx) buff offset: 0x%lx", (uint) cursor->buffer->buffer_no, (ulong) cursor->buffer, cursor->chaser, (ulong) cursor->buffer->size, - (ulong) (cursor->ptr -cursor->buffer->buffer), + (ulong) (cursor->ptr - cursor->buffer->buffer), (ulong) LSN_FILE_NO(*horizon), (ulong) LSN_OFFSET(*horizon), (ulong) (LSN_OFFSET(cursor->buffer->offset) + cursor->buffer->size))); + /* + TODO: make one check function for the buffer/loghandler + */ + DBUG_ASSERT(cursor->chaser || - ((ulong) (cursor->ptr -cursor->buffer->buffer) == + ((ulong) (cursor->ptr - cursor->buffer->buffer) == cursor->buffer->size)); DBUG_ASSERT(cursor->buffer->buffer_no == cursor->buffer_no); DBUG_ASSERT((cursor->ptr -cursor->buffer->buffer) %TRANSLOG_PAGE_SIZE == - cursor->current_page_size % TRANSLOG_PAGE_SIZE); - DBUG_ASSERT(cursor->current_page_size <= TRANSLOG_PAGE_SIZE); + cursor->current_page_fill % TRANSLOG_PAGE_SIZE); + DBUG_ASSERT(cursor->current_page_fill <= TRANSLOG_PAGE_SIZE); DBUG_RETURN(0); } @@ -2511,7 +2452,7 @@ static my_bool translog_write_parts_on_page(TRANSLOG_ADDRESS *horizon, SYNOPSIS translog_write_variable_record_1group_header() parts Descriptor of record source parts - type the log record type + type The log record type short_trid Sort transaction ID or 0 if it has no sense header_length Calculated header length of chunk type 0 chunk0_header Buffer for the chunk header writing @@ -2522,18 +2463,20 @@ translog_write_variable_record_1group_header(struct st_translog_parts *parts, enum translog_record_type type, SHORT_TRANSACTION_ID short_trid, uint16 header_length, - uchar *chunk0_header) + byte *chunk0_header) { struct st_translog_part part; - DBUG_ASSERT(parts->current != 0); /* first part is left for - header */ + DBUG_ASSERT(parts->current != 0); /* first part is left for header */ parts->total_record_length+= (part.len= header_length); part.buff= chunk0_header; - *chunk0_header= (uchar) (type |TRANSLOG_CHUNK_LSN); + /* puts chunk type */ + *chunk0_header= (byte) (type | TRANSLOG_CHUNK_LSN); int2store(chunk0_header + 1, short_trid); + /* puts record length */ translog_write_variable_record_1group_code_len(chunk0_header + 3, parts->record_length, header_length); + /* puts 0 as chunk length which indicate 1 group record */ int2store(chunk0_header + header_length - 2, 0); parts->current--; set_dynamic(&parts->parts, (gptr) &part, parts->current); @@ -2548,20 +2491,16 @@ translog_write_variable_record_1group_header(struct st_translog_parts *parts, buffer target buffer */ -#ifndef DBUG_OFF -static void translog_buffer_increase_writers(struct st_translog_buffer *buffer) +static inline void +translog_buffer_increase_writers(struct st_translog_buffer *buffer) { DBUG_ENTER("translog_buffer_increase_writers"); buffer->copy_to_buffer_in_progress++; - DBUG_PRINT("info", ("copy_to_buffer_in_progress, buffer #%u 0x%lx: %d", + DBUG_PRINT("info", ("copy_to_buffer_in_progress. Buffer #%u 0x%lx: %d", (uint) buffer->buffer_no, (ulong) buffer, buffer->copy_to_buffer_in_progress)); DBUG_VOID_RETURN; } -#else -#define translog_buffer_increase_writers(B) \ - (B)->copy_to_buffer_in_progress++; -#endif /* @@ -2577,14 +2516,12 @@ static void translog_buffer_decrease_writers(struct st_translog_buffer *buffer) { DBUG_ENTER("translog_buffer_decrease_writers"); buffer->copy_to_buffer_in_progress--; - DBUG_PRINT("info", ("copy_to_buffer_in_progress, buffer #%u 0x%lx: %d", + DBUG_PRINT("info", ("copy_to_buffer_in_progress. Buffer #%u 0x%lx: %d", (uint) buffer->buffer_no, (ulong) buffer, buffer->copy_to_buffer_in_progress)); if (buffer->copy_to_buffer_in_progress == 0 && buffer->waiting_filling_buffer.last_thread != NULL) - { wqueue_release_queue(&buffer->waiting_filling_buffer); - } DBUG_VOID_RETURN; } @@ -2599,8 +2536,8 @@ static void translog_buffer_decrease_writers(struct st_translog_buffer *buffer) cursor / RETURN - 0 - OK - 1 - Error + 0 OK + 1 Error */ static my_bool @@ -2608,14 +2545,11 @@ translog_write_variable_record_chunk2_page(struct st_translog_parts *parts, TRANSLOG_ADDRESS *horizon, struct st_buffer_cursor *cursor) { - struct st_translog_buffer *buffer_to_flush= 0; + struct st_translog_buffer *buffer_to_flush; int rc; - uchar chunk2_header[1]= - { - TRANSLOG_CHUNK_NOHDR - }; - + byte chunk2_header[1]; DBUG_ENTER("translog_write_variable_record_chunk2_page"); + chunk2_header[0]= TRANSLOG_CHUNK_NOHDR; rc= translog_page_next(horizon, cursor, &buffer_to_flush); if (buffer_to_flush != NULL) @@ -2629,7 +2563,9 @@ translog_write_variable_record_chunk2_page(struct st_translog_parts *parts, if (rc) DBUG_RETURN(1); + /* Puts chunk type */ translog_write_data_on_page(horizon, cursor, 1, chunk2_header); + /* Puts chunk body */ translog_write_parts_on_page(horizon, cursor, log_descriptor.page_capacity_chunk_2, parts); DBUG_RETURN(0); @@ -2647,8 +2583,8 @@ translog_write_variable_record_chunk2_page(struct st_translog_parts *parts, cursor / RETURN - 0 - OK - 1 - Error + 0 OK + 1 Error */ static my_bool @@ -2657,11 +2593,10 @@ translog_write_variable_record_chunk3_page(struct st_translog_parts *parts, TRANSLOG_ADDRESS *horizon, struct st_buffer_cursor *cursor) { - struct st_translog_buffer *buffer_to_flush= 0; + struct st_translog_buffer *buffer_to_flush; struct st_translog_part part; int rc; - uchar chunk3_header[1 + 2]; - + byte chunk3_header[1 + 2]; DBUG_ENTER("translog_write_variable_record_chunk3_page"); rc= translog_page_next(horizon, cursor, &buffer_to_flush); @@ -2682,11 +2617,12 @@ translog_write_variable_record_chunk3_page(struct st_translog_parts *parts, DBUG_RETURN(0); } - DBUG_ASSERT(parts->current != 0); /* first part is left for - header */ + DBUG_ASSERT(parts->current != 0); /* first part is left for header */ parts->total_record_length+= (part.len= 1 + 2); part.buff= chunk3_header; - *chunk3_header= (uchar) (TRANSLOG_CHUNK_LNGTH); + /* Puts chunk type */ + *chunk3_header= (byte) (TRANSLOG_CHUNK_LNGTH); + /* Puts chunk length */ int2store(chunk3_header + 1, length); parts->current--; set_dynamic(&parts->parts, (gptr) &part, parts->current); @@ -2705,16 +2641,16 @@ translog_write_variable_record_chunk3_page(struct st_translog_parts *parts, last_page_data Plus this data on the last page RETURN - 0 - OK - 1 - Error + 0 OK + 1 Error */ static my_bool translog_advance_pointer(uint pages, uint16 last_page_data) { - translog_size_t last_page_offset= - log_descriptor.page_overhead + last_page_data; + translog_size_t last_page_offset= (log_descriptor.page_overhead + + last_page_data); translog_size_t offset= (TRANSLOG_PAGE_SIZE - - log_descriptor.bc.current_page_size + + log_descriptor.bc.current_page_fill + pages * TRANSLOG_PAGE_SIZE + last_page_offset); translog_size_t buffer_end_offset, file_end_offset, min_offset; DBUG_ENTER("translog_advance_pointer"); @@ -2722,20 +2658,19 @@ static my_bool translog_advance_pointer(uint pages, uint16 last_page_data) (ulong) LSN_FILE_NO(log_descriptor.horizon), (ulong) LSN_OFFSET(log_descriptor.horizon), (uint) (TRANSLOG_PAGE_SIZE - - log_descriptor.bc.current_page_size), + log_descriptor.bc.current_page_fill), pages, (uint) log_descriptor.page_overhead, (uint) last_page_data)); for (;;) { - uint8 new_buffer_no= - (log_descriptor.bc.buffer_no + 1) % TRANSLOG_BUFFERS_NO; + uint8 new_buffer_no; struct st_translog_buffer *new_buffer; struct st_translog_buffer *old_buffer; buffer_end_offset= TRANSLOG_WRITE_BUFFER - log_descriptor.bc.buffer->size; - file_end_offset= - log_descriptor.log_file_max_size - LSN_OFFSET(log_descriptor.horizon); - DBUG_PRINT("info", ("offset: %lu, buffer_end_offs: %lu, " + file_end_offset= (log_descriptor.log_file_max_size - + LSN_OFFSET(log_descriptor.horizon)); + DBUG_PRINT("info", ("offset: %lu buffer_end_offs: %lu, " "file_end_offs: %lu", (ulong) offset, (ulong) buffer_end_offset, (ulong) file_end_offset)); @@ -2762,19 +2697,21 @@ static my_bool translog_advance_pointer(uint pages, uint16 last_page_data) translog_buffer_lock(new_buffer); translog_wait_for_buffer_free(new_buffer); - min_offset= (buffer_end_offset < file_end_offset ? - buffer_end_offset : file_end_offset); + min_offset= min(buffer_end_offset, file_end_offset); + /* + TODO: check is it ptr or size enough + */ log_descriptor.bc.buffer->size+= min_offset; log_descriptor.bc.ptr+= min_offset; - DBUG_PRINT("info", ("NewP buffer #%u: 0x%lx, chaser: %d, Size: %lu (%lu)", + DBUG_PRINT("info", ("NewP buffer #%u: 0x%lx chaser: %d Size: %lu (%lu)", (uint) log_descriptor.bc.buffer->buffer_no, (ulong) log_descriptor.bc.buffer, log_descriptor.bc.chaser, (ulong) log_descriptor.bc.buffer->size, (ulong) (log_descriptor.bc.ptr -log_descriptor.bc. buffer->buffer))); - DBUG_ASSERT((ulong) - (log_descriptor.bc.ptr -log_descriptor.bc.buffer->buffer) == + DBUG_ASSERT((ulong) (log_descriptor.bc.ptr - + log_descriptor.bc.buffer->buffer) == log_descriptor.bc.buffer->size); DBUG_ASSERT(log_descriptor.bc.buffer->buffer_no == log_descriptor.bc.buffer_no); @@ -2785,7 +2722,7 @@ static my_bool translog_advance_pointer(uint pages, uint16 last_page_data) log_descriptor.horizon+= LSN_ONE_FILE; log_descriptor.horizon= LSN_REPLACE_OFFSET(log_descriptor.horizon, TRANSLOG_PAGE_SIZE); - DBUG_PRINT("info", ("New file %lu", + DBUG_PRINT("info", ("New file: %lu", (ulong) LSN_FILE_NO(log_descriptor.horizon))); if (translog_create_new_file()) { @@ -2799,42 +2736,41 @@ static my_bool translog_advance_pointer(uint pages, uint16 last_page_data) } translog_start_buffer(new_buffer, &log_descriptor.bc, new_buffer_no); if (translog_buffer_unlock(old_buffer)) - { DBUG_RETURN(1); - } offset-= min_offset; } log_descriptor.bc.ptr+= offset; log_descriptor.bc.buffer->size+= offset; translog_buffer_increase_writers(log_descriptor.bc.buffer); log_descriptor.horizon+= offset; /* offset increasing */ - log_descriptor.bc.current_page_size= last_page_offset; + log_descriptor.bc.current_page_fill= last_page_offset; DBUG_PRINT("info", ("drop write_counter")); log_descriptor.bc.write_counter= 0; log_descriptor.bc.previous_offset= 0; - DBUG_PRINT("info", ("NewP buffer #%u: 0x%lx, chaser: %d, Size: %lu (%lu), " - "offset: %u last page: %u", + DBUG_PRINT("info", ("NewP buffer #%u: 0x%lx chaser: %d Size: %lu (%lu) " + "offset: %u last page: %u", (uint) log_descriptor.bc.buffer->buffer_no, (ulong) log_descriptor.bc.buffer, log_descriptor.bc.chaser, (ulong) log_descriptor.bc.buffer->size, - (ulong) (log_descriptor.bc.ptr -log_descriptor.bc.buffer-> + (ulong) (log_descriptor.bc.ptr - + log_descriptor.bc.buffer-> buffer), (uint) offset, (uint) last_page_offset)); - DBUG_ASSERT(log_descriptor.bc.chaser - || - ((ulong) (log_descriptor.bc.ptr -log_descriptor.bc.buffer->buffer) - == log_descriptor.bc.buffer->size)); - DBUG_ASSERT(log_descriptor.bc.buffer->buffer_no == - log_descriptor.bc.buffer_no); DBUG_PRINT("info", ("pointer moved to: (%lu, 0x%lx)", (ulong) LSN_FILE_NO(log_descriptor.horizon), (ulong) LSN_OFFSET(log_descriptor.horizon))); - DBUG_ASSERT((log_descriptor.bc.ptr -log_descriptor.bc.buffer-> + DBUG_ASSERT(log_descriptor.bc.chaser || + ((ulong) (log_descriptor.bc.ptr - + log_descriptor.bc.buffer->buffer) + == log_descriptor.bc.buffer->size)); + DBUG_ASSERT(log_descriptor.bc.buffer->buffer_no == + log_descriptor.bc.buffer_no); + DBUG_ASSERT((log_descriptor.bc.ptr - log_descriptor.bc.buffer-> buffer) %TRANSLOG_PAGE_SIZE == - log_descriptor.bc.current_page_size % TRANSLOG_PAGE_SIZE); - DBUG_ASSERT(log_descriptor.bc.current_page_size <= TRANSLOG_PAGE_SIZE); + log_descriptor.bc.current_page_fill % TRANSLOG_PAGE_SIZE); + DBUG_ASSERT(log_descriptor.bc.current_page_fill <= TRANSLOG_PAGE_SIZE); log_descriptor.bc.protected= 0; DBUG_RETURN(0); } @@ -2854,7 +2790,7 @@ static my_bool translog_advance_pointer(uint pages, uint16 last_page_data) */ #define translog_get_current_page_rest() \ - (TRANSLOG_PAGE_SIZE - log_descriptor.bc.current_page_size) + (TRANSLOG_PAGE_SIZE - log_descriptor.bc.current_page_fill) /* Get buffer rest in full pages @@ -2889,21 +2825,20 @@ static translog_size_t translog_get_current_group_size() { /* buffer rest in full pages */ translog_size_t buffer_rest= translog_get_current_buffer_rest(); - DBUG_ENTER("translog_get_current_group_size"); + DBUG_PRINT("info", ("buffer_rest in pages: %u", buffer_rest)); - DBUG_PRINT("info", ("buffer_rest in pages %u", buffer_rest)); buffer_rest*= log_descriptor.page_capacity_chunk_2; /* in case of only half of buffer free we can write this and next buffer */ if (buffer_rest < log_descriptor.half_buffer_capacity_chunk_2) { - DBUG_PRINT("info", ("buffer_rest %u -> add %lu", - buffer_rest, + DBUG_PRINT("info", ("buffer_rest: %lu -> add %lu", + (ulong) buffer_rest, (ulong) log_descriptor.buffer_capacity_chunk_2)); buffer_rest+= log_descriptor.buffer_capacity_chunk_2; } - DBUG_PRINT("info", ("buffer_rest %u", buffer_rest)); + DBUG_PRINT("info", ("buffer_rest: %lu", (ulong) buffer_rest)); DBUG_RETURN(buffer_rest); } @@ -2924,8 +2859,8 @@ static translog_size_t translog_get_current_group_size() record log type RETURN - 0 - OK - 1 - Error + 0 OK + 1 Error */ static my_bool @@ -2943,23 +2878,20 @@ translog_write_variable_record_1group(LSN *lsn, uint i; translog_size_t record_rest, full_pages, first_page; uint additional_chunk3_page= 0; - uchar chunk0_header[1 + 2 + 5 + 2]; - + byte chunk0_header[1 + 2 + 5 + 2]; DBUG_ENTER("translog_write_variable_record_1group"); *lsn= horizon= log_descriptor.horizon; if (log_record_type_descriptor[type].inwrite_hook && - (*log_record_type_descriptor[type].inwrite_hook)(type, tcb, - lsn, parts)) + (*log_record_type_descriptor[type].inwrite_hook)(type, tcb, lsn, parts)) { + translog_unlock(); DBUG_RETURN(1); } cursor= log_descriptor.bc; cursor.chaser= 1; - /* - Advance pointer To be able unlock the loghandler - */ + /* Advance pointer To be able unlock the loghandler */ first_page= translog_get_current_page_rest(); record_rest= parts->record_length - (first_page - header_length); full_pages= record_rest / log_descriptor.page_capacity_chunk_2; @@ -2973,8 +2905,8 @@ translog_write_variable_record_1group(LSN *lsn, record_rest= 1; } - DBUG_PRINT("info", ("first_page: %u (%u), full_pages: %u (%lu), " - "additional: %u (%u), rest %u = %u", + DBUG_PRINT("info", ("first_page: %u (%u) full_pages: %u (%lu) " + "additional: %u (%u) rest %u = %u", first_page, first_page - header_length, full_pages, (ulong) full_pages * @@ -2984,14 +2916,14 @@ translog_write_variable_record_1group(LSN *lsn, (log_descriptor.page_capacity_chunk_2 - 1), record_rest, parts->record_length)); /* record_rest + 3 is chunk type 3 overhead + record_rest */ - translog_advance_pointer(full_pages + additional_chunk3_page, - (record_rest ? record_rest + 3 : 0)); + rc|= translog_advance_pointer(full_pages + additional_chunk3_page, + (record_rest ? record_rest + 3 : 0)); log_descriptor.bc.buffer->last_lsn= *lsn; rc|= translog_unlock(); /* - check if we switched buffer and need process it (current buffer is + Check if we switched buffer and need process it (current buffer is unlocked already => we will not delay other threads */ if (buffer_to_flush != NULL) @@ -3000,7 +2932,6 @@ translog_write_variable_record_1group(LSN *lsn, rc= translog_buffer_flush(buffer_to_flush); rc|= translog_buffer_unlock(buffer_to_flush); } - if (rc) DBUG_RETURN(1); @@ -3011,7 +2942,7 @@ translog_write_variable_record_1group(LSN *lsn, translog_write_parts_on_page(&horizon, &cursor, first_page, parts); - DBUG_PRINT("info", ("absolute horizon (%lu,0x%lx), local (%lu,0x%lx)", + DBUG_PRINT("info", ("absolute horizon: (%lu,0x%lx) local: (%lu,0x%lx)", (ulong) LSN_FILE_NO(log_descriptor.horizon), (ulong) LSN_OFFSET(log_descriptor.horizon), (ulong) LSN_FILE_NO(horizon), @@ -3022,7 +2953,7 @@ translog_write_variable_record_1group(LSN *lsn, if (translog_write_variable_record_chunk2_page(parts, &horizon, &cursor)) DBUG_RETURN(1); - DBUG_PRINT("info", ("absolute horizon (%lu,0x%lx), local (%lu,0x%lx)", + DBUG_PRINT("info", ("absolute horizon: (%lu,0x%lx) local: (%lu,0x%lx)", (ulong) LSN_FILE_NO(log_descriptor.horizon), (ulong) LSN_OFFSET(log_descriptor.horizon), (ulong) LSN_FILE_NO(horizon), @@ -3036,29 +2967,28 @@ translog_write_variable_record_1group(LSN *lsn, page_capacity_chunk_2 - 2, &horizon, &cursor)) DBUG_RETURN(1); - DBUG_PRINT("info", ("absolute horizon (%lu,0x%lx), local (%lu,0x%lx)", + DBUG_PRINT("info", ("absolute horizon: (%lu,0x%lx) local: (%lu,0x%lx)", (ulong) LSN_FILE_NO(log_descriptor.horizon), (ulong) LSN_OFFSET(log_descriptor.horizon), (ulong) LSN_FILE_NO(horizon), (ulong) LSN_OFFSET(horizon))); - DBUG_ASSERT(cursor.current_page_size == TRANSLOG_PAGE_SIZE); + DBUG_ASSERT(cursor.current_page_fill == TRANSLOG_PAGE_SIZE); } if (translog_write_variable_record_chunk3_page(parts, record_rest, &horizon, &cursor)) DBUG_RETURN(1); - DBUG_PRINT("info", ("absolute horizon (%lu,0x%lx), local (%lu,0x%lx)", + DBUG_PRINT("info", ("absolute horizon: (%lu,0x%lx) local: (%lu,0x%lx)", (ulong) LSN_FILE_NO(log_descriptor.horizon), (ulong) LSN_OFFSET(log_descriptor.horizon), (ulong) LSN_FILE_NO(horizon), (ulong) LSN_OFFSET(horizon))); - rc= translog_buffer_lock(cursor.buffer); - if (!rc) + if (!(rc= translog_buffer_lock(cursor.buffer))) { /* - check if we wrote something on lst not full page and need to reconstruct + Check if we wrote something on 1:st not full page and need to reconstruct CRC and sector protection */ translog_buffer_decrease_writers(cursor.buffer); @@ -3083,8 +3013,8 @@ translog_write_variable_record_1group(LSN *lsn, record log type RETURN - 0 - OK - 1 - Error + 0 OK + 1 Error */ static my_bool @@ -3097,7 +3027,7 @@ translog_write_variable_record_1chunk(LSN *lsn, void *tcb) { int rc; - uchar chunk0_header[1 + 2 + 5 + 2]; + byte chunk0_header[1 + 2 + 5 + 2]; DBUG_ENTER("translog_write_variable_record_1chunk"); translog_write_variable_record_1group_header(parts, type, short_trid, @@ -3105,9 +3035,10 @@ translog_write_variable_record_1chunk(LSN *lsn, *lsn= log_descriptor.horizon; if (log_record_type_descriptor[type].inwrite_hook && - (*log_record_type_descriptor[type].inwrite_hook) (type, tcb, - lsn, parts)) + (*log_record_type_descriptor[type].inwrite_hook)(type, tcb, + lsn, parts)) { + translog_unlock(); DBUG_RETURN(1); } @@ -3139,30 +3070,30 @@ translog_write_variable_record_1chunk(LSN *lsn, translog_put_LSN_diff() base_lsn LSN from which we calculate difference lsn LSN for codding - dst pointer before which result should be written + dst Result will be written to dst[-pack_length] .. dst[-1] NOTE: - to store an LSN in a compact way we will use the following compression: + To store an LSN in a compact way we will use the following compression: - if a log record has LSN1, and it contains the lSN2 as a back reference, - instead of LSN2 we write LSN1-LSN2, encoded as: + If a log record has LSN1, and it contains the lSN2 as a back reference, + Instead of LSN2 we write LSN1-LSN2, encoded as: two bits the number N (see below) 14 bits N bytes - that is, LSN is encoded in 2..5 bytes, and the number of bytes minus 2 + That is, LSN is encoded in 2..5 bytes, and the number of bytes minus 2 is stored in the first two bits. RETURN - pointer on coded LSN - NULL - error + # pointer on coded LSN + NULL Error */ -static uchar *translog_put_LSN_diff(LSN base_lsn, LSN lsn, uchar *dst) +static byte *translog_put_LSN_diff(LSN base_lsn, LSN lsn, byte *dst) { DBUG_ENTER("translog_put_LSN_diff"); - DBUG_PRINT("enter", ("Base: (0x%lu,0x%lx), val: (0x%lu,0x%lx), dst 0x%lx", + DBUG_PRINT("enter", ("Base: (0x%lu,0x%lx) val: (0x%lu,0x%lx) dst: 0x%lx", (ulong) LSN_FILE_NO(base_lsn), (ulong) LSN_OFFSET(base_lsn), (ulong) LSN_FILE_NO(lsn), @@ -3172,9 +3103,14 @@ static uchar *translog_put_LSN_diff(LSN base_lsn, LSN lsn, uchar *dst) uint32 diff; DBUG_ASSERT(base_lsn > lsn); diff= base_lsn - lsn; + DBUG_PRINT("info", ("File is the same. Diff: 0x%lx", (ulong) diff)); if (diff <= 0x3FFF) { dst-= 2; + /* + Note we store this high byte first to ensure that first byte has + 0 in the 3 upper bits. + */ dst[0]= diff >> 8; dst[1]= (diff & 0xFF); } @@ -3204,11 +3140,13 @@ static uchar *translog_put_LSN_diff(LSN base_lsn, LSN lsn, uchar *dst) ulonglong base_offset= LSN_OFFSET(base_lsn); DBUG_ASSERT(base_lsn > lsn); diff= LSN_FILE_NO(base_lsn) - LSN_FILE_NO(lsn); + DBUG_PRINT("info", ("File is different. Diff: 0x%lx", (ulong) diff)); + if (base_offset < LSN_OFFSET(lsn)) { /* take 1 from file offset */ diff--; - base_offset+= 0x100000000LL; + base_offset+= LL(0x100000000); } offset_diff= base_offset - LSN_OFFSET(lsn); if (diff > 0x3f) @@ -3236,71 +3174,74 @@ static uchar *translog_put_LSN_diff(LSN base_lsn, LSN lsn, uchar *dst) dst pointer to buffer where to write 7byte LSN NOTE: - to store an LSN in a compact way we use the following compression: + To store an LSN in a compact way we will use the following compression: If a log record has LSN1, and it contains the lSN2 as a back reference, - instead of LSN2 we write LSN1-LSN2, encoded as: + Instead of LSN2 we write LSN1-LSN2, encoded as: two bits the number N (see below) 14 bits N bytes - That is, LSN is encoded in 2..5 bytes, and the number of bytes minus 2 - is stored in the first two bits. + That is, LSN is encoded in 2..5 bytes, and the number of bytes minus 2 + is stored in the first two bits. RETURN pointer to buffer after decoded LSN */ -static uchar *translog_get_LSN_from_diff(LSN base_lsn, uchar *src, uchar *dst) +static byte *translog_get_LSN_from_diff(LSN base_lsn, byte *src, byte *dst) { LSN lsn; uint32 diff; uint32 first_byte; + uint32 file_no, rec_offset; uint8 code; DBUG_ENTER("translog_get_LSN_from_diff"); - DBUG_PRINT("enter", ("Base: (0x%lx,0x%lx), src: 0x%lx, dst 0x%lx", + DBUG_PRINT("enter", ("Base: (0x%lx,0x%lx) src: 0x%lx dst 0x%lx", (ulong) LSN_FILE_NO(base_lsn), (ulong) LSN_OFFSET(base_lsn), (ulong) src, (ulong) dst)); first_byte= *((uint8*) src); - code= first_byte & 0xC0; - first_byte &= 0x3F; + code= first_byte >> 6; /* Length in 2 upmost bits */ + first_byte&= 0x3F; + src++; /* Skip length + encode */ + file_no= LSN_FILE_NO(base_lsn); /* Assume relative */ + DBUG_PRINT("info", ("code: %u first byte: %lu", + (uint) code, (ulong) first_byte)); switch (code) { - case 0x00: - lsn= base_lsn - ((first_byte << 8) + *((uint8*) (src + 1))); - src+= 2; + case 0: + rec_offset= LSN_OFFSET(base_lsn) - ((first_byte << 8) + *((uint8*)src)); break; - case 0x40: - diff= uint2korr(src + 1); - lsn= base_lsn - ((first_byte << 16) + diff); - src+= 3; + case 1: + diff= uint2korr(src); + rec_offset= LSN_OFFSET(base_lsn) - ((first_byte << 16) + diff); break; - case 0x80: - diff= uint3korr(src + 1); - lsn= base_lsn - ((first_byte << 24) + diff); - src+= 4; + case 2: + diff= uint3korr(src); + rec_offset= LSN_OFFSET(base_lsn) - ((first_byte << 24) + diff); break; - case 0xC0: + case 3: { - diff= uint4korr(src + 1); ulonglong base_offset= LSN_OFFSET(base_lsn); + diff= uint4korr(src); if (diff > LSN_OFFSET(base_lsn)) { /* take 1 from file offset */ first_byte++; - base_offset+= 0x100000000LL; + base_offset+= LL(0x100000000); } - lsn= MAKE_LSN(LSN_FILE_NO(base_lsn) - first_byte, - base_offset - diff); - src+= 5; + file_no= LSN_FILE_NO(base_lsn) - first_byte; + rec_offset= base_offset - diff; break; } default: DBUG_ASSERT(0); DBUG_RETURN(NULL); } - lsn7store(dst, lsn); + lsn= MAKE_LSN(file_no, rec_offset); + src+= code + 1; + lsn_store(dst, lsn); DBUG_PRINT("info", ("new src: 0x%lx", (ulong) dst)); DBUG_RETURN(src); } @@ -3317,77 +3258,70 @@ static uchar *translog_get_LSN_from_diff(LSN base_lsn, uchar *src, uchar *dst) compressed_LSNs buffer which can be used for storing compressed LSN(s) RETURN - 0 - OK - 1 - Error + 0 OK + 1 Error */ static my_bool translog_relative_LSN_encode(struct st_translog_parts *parts, LSN base_lsn, - uint lsns, uchar *compressed_LSNs) + uint lsns, byte *compressed_LSNs) { - struct st_translog_part part; - uint lsns_len= lsns * 7; + struct st_translog_part *part; + uint lsns_len= lsns * LSN_STORE_SIZE; DBUG_ENTER("translog_relative_LSN_encode"); - get_dynamic(&parts->parts, (gptr) &part, parts->current); + part= dynamic_element(&parts->parts, parts->current, + struct st_translog_part *); /* collect all LSN(s) in one chunk if it (they) is (are) divided */ - if (part.len < lsns_len) + if (part->len < lsns_len) { - uint copied= part.len; - DBUG_PRINT("info", ("Using buffer 0x%lx", (ulong) compressed_LSNs)); - memmove(compressed_LSNs, part.buff, part.len); + uint copied= part->len; + DBUG_PRINT("info", ("Using buffer: 0x%lx", (ulong) compressed_LSNs)); + memcpy(compressed_LSNs, part->buff, part->len); do { - get_dynamic(&parts->parts, (gptr) &part, parts->current + 1); - if ((part.len + copied) < lsns_len) + struct st_translog_part *next_part; + next_part= dynamic_element(&parts->parts, parts->current + 1, + struct st_translog_part *); + if ((next_part->len + copied) < lsns_len) { - memmove(compressed_LSNs + copied, part.buff, part.len); - copied+= part.len; + memcpy(compressed_LSNs + copied, next_part->buff, next_part->len); + copied+= next_part->len; delete_dynamic_element(&parts->parts, parts->current + 1); } else { uint len= lsns_len - copied; - memmove(compressed_LSNs + copied, part.buff, len); + memcpy(compressed_LSNs + copied, next_part->buff, len); copied= lsns_len; - part.buff+= len; - part.len-= len; - /* - We do not check result of set_dynamic, because we are sure that - it will not grow - */ - set_dynamic(&parts->parts, (gptr) &part, parts->current + 1); + next_part->buff+= len; + next_part->len-= len; } } while (copied < lsns_len); - part.len= lsns_len; - part.buff= compressed_LSNs; + part->len= lsns_len; + part->buff= compressed_LSNs; } { /* Compress */ LSN ref; uint economy; - uchar *ref_ptr= part.buff + lsns_len - 7; - uchar *dst_ptr= part.buff + lsns_len; - uint i; - for (i= 0; i < lsns; i++, ref_ptr-= 7) + byte *ref_ptr= part->buff + lsns_len - LSN_STORE_SIZE; + byte *dst_ptr= part->buff + lsns_len; + for (; ref_ptr >= part->buff ; ref_ptr-= LSN_STORE_SIZE) { - ref= lsn7korr(ref_ptr); + ref= lsn_korr(ref_ptr); if ((dst_ptr= translog_put_LSN_diff(base_lsn, ref, dst_ptr)) == NULL) DBUG_RETURN(1); } - economy= (dst_ptr - part.buff); - DBUG_PRINT("info", ("Economy %u", economy)); - part.len-= economy; + /* Note that dst_ptr did grow downward ! */ + economy= (uint) (dst_ptr - part->buff); + DBUG_PRINT("info", ("Economy: %u", economy)); + part->len-= economy; parts->record_length-= economy; parts->total_record_length-= economy; - part.buff= dst_ptr; + part->buff= dst_ptr; } - /* - We do not check result of set_dynamic, because we are sure that - it will not grow - */ - set_dynamic(&parts->parts, (gptr) &part, parts->current); DBUG_RETURN(0); } @@ -3408,8 +3342,8 @@ static my_bool translog_relative_LSN_encode(struct st_translog_parts *parts, record log type RETURN - 0 - OK - 1 - Error + 0 OK + 1 Error */ static my_bool @@ -3436,19 +3370,18 @@ translog_write_variable_record_mgroup(LSN *lsn, uint16 page_capacity= log_descriptor.page_capacity_chunk_2 + 1; uint16 last_page_capacity; my_bool new_page_before_chunk0= 1, first_chunk0= 1; - uchar chunk0_header[1 + 2 + 5 + 2 + 2], group_desc[7 + 1]; - uchar chunk2_header[1]= - { - TRANSLOG_CHUNK_NOHDR - }; + byte chunk0_header[1 + 2 + 5 + 2 + 2], group_desc[7 + 1]; + byte chunk2_header[1]; uint header_fixed_part= header_length + 2; uint groups_per_page= (page_capacity - header_fixed_part) / (7 + 1); - DBUG_ENTER("translog_write_variable_record_mgroup"); + chunk2_header[0]= TRANSLOG_CHUNK_NOHDR; + if (init_dynamic_array(&groups, sizeof(struct st_translog_group_descriptor), 10, 10 CALLER_INFO)) { + translog_unlock(); UNRECOVERABLE_ERROR(("init array failed")); DBUG_RETURN(1); } @@ -3471,35 +3404,34 @@ translog_write_variable_record_mgroup(LSN *lsn, cursor.chaser= 1; if ((full_pages= buffer_rest / log_descriptor.page_capacity_chunk_2) > 255) { - /* suzeof(uint8) == 256 is max number of chunk in multi-chunks group */ + /* sizeof(uint8) == 256 is max number of chunk in multi-chunks group */ full_pages= 255; buffer_rest= full_pages * log_descriptor.page_capacity_chunk_2; } /* group chunks = - full pages + first page (which actually can be full, too. + full pages + first page (which actually can be full, too). But here we assign number of chunks - 1 */ group.num= full_pages; if (insert_dynamic(&groups, (gptr) &group)) { - translog_unlock(); - delete_dynamic(&groups); UNRECOVERABLE_ERROR(("insert into array failed")); - DBUG_RETURN(1); + goto err_unlock; } - DBUG_PRINT("info", ("chunk #%u first_page: %u (%u), full_pages: %lu (%lu), " + DBUG_PRINT("info", ("chunk: #%u first_page: %u (%u) " + "full_pages: %lu (%lu) " "Left %lu", groups.elements, first_page, first_page - 1, (ulong) full_pages, - (ulong) full_pages * - log_descriptor.page_capacity_chunk_2, - (ulong)parts->record_length - - (first_page - 1 + buffer_rest) - - done)); - translog_advance_pointer(full_pages, 0); + (ulong) (full_pages * + log_descriptor.page_capacity_chunk_2), + (ulong)(parts->record_length - (first_page - 1 + + buffer_rest) - + done))); + rc|= translog_advance_pointer(full_pages, 0); rc|= translog_unlock(); @@ -3514,15 +3446,14 @@ translog_write_variable_record_mgroup(LSN *lsn, } if (rc) { - delete_dynamic(&groups); UNRECOVERABLE_ERROR(("flush of unlock buffer failed")); - DBUG_RETURN(1); + goto err; } translog_write_data_on_page(&horizon, &cursor, 1, chunk2_header); translog_write_parts_on_page(&horizon, &cursor, first_page - 1, parts); - DBUG_PRINT("info", ("absolute horizon (%lu,0x%lx), local (%lu,0x%lx) " - "Left: %lu", + DBUG_PRINT("info", ("absolute horizon: (%lu,0x%lx) local: (%lu,0x%lx) " + "Left %lu", (ulong) LSN_FILE_NO(log_descriptor.horizon), (ulong) LSN_OFFSET(log_descriptor.horizon), (ulong) LSN_FILE_NO(horizon), @@ -3533,12 +3464,10 @@ translog_write_variable_record_mgroup(LSN *lsn, for (i= 0; i < full_pages; i++) { if (translog_write_variable_record_chunk2_page(parts, &horizon, &cursor)) - { - delete_dynamic(&groups); - DBUG_RETURN(1); - } + goto err; - DBUG_PRINT("info", ("absolute horizon (%lu,0x%lx), local (%lu,0x%lx)" + DBUG_PRINT("info", ("absolute horizon: (%lu,0x%lx) " + "local: (%lu,0x%lx) " "Left: %lu", (ulong) LSN_FILE_NO(log_descriptor.horizon), (ulong) LSN_OFFSET(log_descriptor.horizon), @@ -3551,7 +3480,7 @@ translog_write_variable_record_mgroup(LSN *lsn, done+= (first_page - 1 + buffer_rest); - /* TODO: made separate function for following */ + /* TODO: make separate function for following */ rc= translog_page_next(&horizon, &cursor, &buffer_to_flush); if (buffer_to_flush != NULL) { @@ -3564,19 +3493,15 @@ translog_write_variable_record_mgroup(LSN *lsn, } if (rc) { - delete_dynamic(&groups); UNRECOVERABLE_ERROR(("flush of unlock buffer failed")); - DBUG_RETURN(1); + goto err; } rc= translog_buffer_lock(cursor.buffer); if (!rc) translog_buffer_decrease_writers(cursor.buffer); rc|= translog_buffer_unlock(cursor.buffer); if (rc) - { - delete_dynamic(&groups); - DBUG_RETURN(1); - } + goto err; translog_lock(); @@ -3590,10 +3515,8 @@ translog_write_variable_record_mgroup(LSN *lsn, group.num= 0; /* 0 because it does not matter */ if (insert_dynamic(&groups, (gptr) &group)) { - delete_dynamic(&groups); - translog_unlock(); UNRECOVERABLE_ERROR(("insert into array failed")); - DBUG_RETURN(1); + goto err_unlock; } record_rest= parts->record_length - done; DBUG_PRINT("info", ("Record rest: %lu", (ulong) record_rest)); @@ -3643,34 +3566,37 @@ translog_write_variable_record_mgroup(LSN *lsn, record_rest + header_fixed_part + (groups.elements - groups_per_page * (chunk0_pages - 1)) * (7 + 1)) chunk0_pages++; - DBUG_PRINT("info", ("chunk0_pages %u, groups %u, groups per full page %u, " - "Group on last page %u", + DBUG_PRINT("info", ("chunk0_pages: %u groups %u groups per full page: %u " + "Group on last page: %u", chunk0_pages, groups.elements, groups_per_page, (groups.elements - ((page_capacity - header_fixed_part) / (7 + 1)) * (chunk0_pages - 1)))); - DBUG_PRINT("info", ("first_page: %u, chunk2 %u full_pages: %u (%lu), " - "chunk3 %u (%u), rest %u", + DBUG_PRINT("info", ("first_page: %u chunk2: %u full_pages: %u (%lu) " + "chunk3: %u (%u) rest: %u", first_page, chunk2_page, full_pages, (ulong) full_pages * log_descriptor.page_capacity_chunk_2, chunk3_pages, (uint) chunk3_size, (uint) record_rest)); - translog_advance_pointer(full_pages + chunk3_pages + - (chunk0_pages - 1), - record_rest + header_fixed_part + - (groups.elements - - ((page_capacity - header_fixed_part) / (7 + 1)) * - (chunk0_pages - 1)) * (7 + 1)); - translog_unlock(); + rc= translog_advance_pointer(full_pages + chunk3_pages + + (chunk0_pages - 1), + record_rest + header_fixed_part + + (groups.elements - + ((page_capacity - + header_fixed_part) / (7 + 1)) * + (chunk0_pages - 1)) * (7 + 1)); + rc|= translog_unlock(); + if (rc) + goto err; if (chunk2_page) { DBUG_PRINT("info", ("chunk 2 to finish first page")); translog_write_data_on_page(&horizon, &cursor, 1, chunk2_header); translog_write_parts_on_page(&horizon, &cursor, first_page - 1, parts); - DBUG_PRINT("info", ("absolute horizon (%lu,0x%lx), local (%lu,0x%lx) " + DBUG_PRINT("info", ("absolute horizon: (%lu,0x%lx) local: (%lu,0x%lx) " "Left: %lu", (ulong) LSN_FILE_NO(log_descriptor.horizon), (ulong) LSN_OFFSET(log_descriptor.horizon), @@ -3683,19 +3609,19 @@ translog_write_variable_record_mgroup(LSN *lsn, { DBUG_PRINT("info", ("chunk 3")); DBUG_ASSERT(full_pages == 0); - uchar chunk3_header[3]; + byte chunk3_header[3]; + chunk3_pages= 0; chunk3_header[0]= TRANSLOG_CHUNK_LNGTH; int2store(chunk3_header + 1, chunk3_size); translog_write_data_on_page(&horizon, &cursor, 3, chunk3_header); translog_write_parts_on_page(&horizon, &cursor, chunk3_size, parts); - DBUG_PRINT("info", ("absolute horizon (%lu,0x%lx), local (%lu,0x%lx) " + DBUG_PRINT("info", ("absolute horizon: (%lu,0x%lx) local: (%lu,0x%lx) " "Left: %lu", (ulong) LSN_FILE_NO(log_descriptor.horizon), (ulong) LSN_OFFSET(log_descriptor.horizon), (ulong) LSN_FILE_NO(horizon), (ulong) LSN_OFFSET(horizon), (ulong) (parts->record_length - chunk3_size - done))); - chunk3_pages= 0; } else { @@ -3707,12 +3633,9 @@ translog_write_variable_record_mgroup(LSN *lsn, { DBUG_ASSERT(chunk2_page != 0); if (translog_write_variable_record_chunk2_page(parts, &horizon, &cursor)) - { - delete_dynamic(&groups); - DBUG_RETURN(1); - } + goto err; - DBUG_PRINT("info", ("absolute horizon (%lu,0x%lx), local (%lu,0x%lx) " + DBUG_PRINT("info", ("absolute horizon: (%lu,0x%lx) local: (%lu,0x%lx) " "Left: %lu", (ulong) LSN_FILE_NO(log_descriptor.horizon), (ulong) LSN_OFFSET(log_descriptor.horizon), @@ -3727,18 +3650,15 @@ translog_write_variable_record_mgroup(LSN *lsn, translog_write_variable_record_chunk3_page(parts, chunk3_size, &horizon, &cursor)) - { - delete_dynamic(&groups); - DBUG_RETURN(1); - } - DBUG_PRINT("info", ("absolute horizon (%lu,0x%lx), local (%lu,0x%lx)", + goto err; + DBUG_PRINT("info", ("absolute horizon: (%lu,0x%lx) local: (%lu,0x%lx)", (ulong) LSN_FILE_NO(log_descriptor.horizon), (ulong) LSN_OFFSET(log_descriptor.horizon), (ulong) LSN_FILE_NO(horizon), (ulong) LSN_OFFSET(horizon))); - *chunk0_header= (uchar) (type |TRANSLOG_CHUNK_LSN); + *chunk0_header= (byte) (type |TRANSLOG_CHUNK_LSN); int2store(chunk0_header + 1, short_trid); translog_write_variable_record_1group_code_len(chunk0_header + 3, parts->record_length, @@ -3760,24 +3680,20 @@ translog_write_variable_record_mgroup(LSN *lsn, } if (rc) { - delete_dynamic(&groups); UNRECOVERABLE_ERROR(("flush of unlock buffer failed")); - DBUG_RETURN(1); + goto err; } } new_page_before_chunk0= 1; if (first_chunk0) { + first_chunk0= 0; *lsn= horizon; if (log_record_type_descriptor[type].inwrite_hook && (*log_record_type_descriptor[type].inwrite_hook) (type, tcb, lsn, parts)) - { - DBUG_RETURN(1); - } - - first_chunk0= 0; + goto err; } /* @@ -3787,7 +3703,7 @@ translog_write_variable_record_mgroup(LSN *lsn, */ limit= (groups_per_page < groups.elements - curr_group ? groups_per_page : groups.elements - curr_group); - DBUG_PRINT("info", ("Groups: %u curr %u, limit %u", + DBUG_PRINT("info", ("Groups: %u curr: %u limit: %u", (uint) groups.elements, (uint) curr_group, (uint) limit)); @@ -3810,9 +3726,11 @@ translog_write_variable_record_mgroup(LSN *lsn, chunk0_header); for (i= curr_group; i < limit + curr_group; i++) { - get_dynamic(&groups, (gptr) &group, i); - lsn7store(group_desc, group.addr); - group_desc[7]= group.num; + struct st_translog_group_descriptor *grp_ptr; + grp_ptr= dynamic_element(&groups, i, + struct st_translog_group_descriptor *); + lsn_store(group_desc, grp_ptr->addr); + group_desc[7]= grp_ptr->num; translog_write_data_on_page(&horizon, &cursor, (7 + 1), group_desc); } @@ -3831,6 +3749,12 @@ translog_write_variable_record_mgroup(LSN *lsn, delete_dynamic(&groups); DBUG_RETURN(rc); + +err_unlock: + translog_unlock(); +err: + delete_dynamic(&groups); + DBUG_RETURN(1); } @@ -3847,8 +3771,8 @@ translog_write_variable_record_mgroup(LSN *lsn, record log type RETURN - 0 - OK - 1 - Error + 0 OK + 1 Error */ static my_bool translog_write_variable_record(LSN *lsn, @@ -3862,17 +3786,17 @@ static my_bool translog_write_variable_record(LSN *lsn, translog_variable_record_length_bytes(parts->record_length); ulong buffer_rest; uint page_rest; - uchar compressed_LSNs[2 * 7]; /* Max number of such LSNs per - record is 2 */ + /* Max number of such LSNs per record is 2 */ + byte compressed_LSNs[2 * LSN_STORE_SIZE]; DBUG_ENTER("translog_write_variable_record"); translog_lock(); - DBUG_PRINT("info", ("horizon (%lu,0x%lx)", + DBUG_PRINT("info", ("horizon: (%lu,0x%lx)", (ulong) LSN_FILE_NO(log_descriptor.horizon), (ulong) LSN_OFFSET(log_descriptor.horizon))); - page_rest= TRANSLOG_PAGE_SIZE - log_descriptor.bc.current_page_size; - DBUG_PRINT("info", ("header length %u, page_rest: %u", + page_rest= TRANSLOG_PAGE_SIZE - log_descriptor.bc.current_page_fill; + DBUG_PRINT("info", ("header length: %u page_rest: %u", header_length1, page_rest)); /* @@ -3883,12 +3807,13 @@ static my_bool translog_write_variable_record(LSN *lsn, (header_length1 + log_record_type_descriptor[type].read_header_len)) { DBUG_PRINT("info", - ("Next page, size: %u, header: %u + %u", - log_descriptor.bc.current_page_size, + ("Next page, size: %u header: %u + %u", + log_descriptor.bc.current_page_fill, header_length1, log_record_type_descriptor[type].read_header_len)); translog_page_next(&log_descriptor.horizon, &log_descriptor.bc, &buffer_to_flush); + /* Chunk 2 header is 1 byte, so full page capacity will be one byte more */ page_rest= log_descriptor.page_capacity_chunk_2 + 1; DBUG_PRINT("info", ("page_rest: %u", page_rest)); } @@ -3897,26 +3822,30 @@ static my_bool translog_write_variable_record(LSN *lsn, To minimize compressed size we will compress always relative to very first chunk address (log_descriptor.horizon for now) */ - if (log_record_type_descriptor[type].compresed_LSN > 0) + if (log_record_type_descriptor[type].compressed_LSN > 0) { if (translog_relative_LSN_encode(parts, log_descriptor.horizon, log_record_type_descriptor[type]. - compresed_LSN, compressed_LSNs)) + compressed_LSN, compressed_LSNs)) { - int rc= translog_unlock(); + translog_unlock(); if (buffer_to_flush != NULL) { - if (!rc) - rc= translog_buffer_flush(buffer_to_flush); - rc|= translog_buffer_unlock(buffer_to_flush); + /* + It is just try to finish log in nice way in case of error, so we + do not check result of the following functions, because we are + going return error state in any case + */ + translog_buffer_flush(buffer_to_flush); + translog_buffer_unlock(buffer_to_flush); } DBUG_RETURN(1); } /* recalculate header length after compression */ header_length1= 1 + 2 + 2 + translog_variable_record_length_bytes(parts->record_length); - DBUG_PRINT("info", ("after compressing LSN(s) header length %u, " - "record length %lu", + DBUG_PRINT("info", ("after compressing LSN(s) header length: %u " + "record length: %lu", header_length1, (ulong)parts->record_length)); } @@ -3943,7 +3872,6 @@ static my_bool translog_write_variable_record(LSN *lsn, parts, buffer_to_flush, header_length1, buffer_rest, tcb)); - DBUG_RETURN(0); } @@ -3960,8 +3888,8 @@ static my_bool translog_write_variable_record(LSN *lsn, record log type RETURN - 0 - OK - 1 - Error + 0 OK + 1 Error */ static my_bool translog_write_fixed_record(LSN *lsn, @@ -3971,9 +3899,9 @@ static my_bool translog_write_fixed_record(LSN *lsn, void *tcb) { struct st_translog_buffer *buffer_to_flush= NULL; - uchar chunk1_header[1 + 2]; - uchar compressed_LSNs[2 * 7]; /* Max number of such LSNs per - record is 2 */ + byte chunk1_header[1 + 2]; + /* Max number of such LSNs per record is 2 */ + byte compressed_LSNs[2 * LSN_STORE_SIZE]; struct st_translog_part part; int rc; DBUG_ENTER("translog_write_fixed_record"); @@ -3984,32 +3912,32 @@ static my_bool translog_write_fixed_record(LSN *lsn, (log_record_type_descriptor[type].class == LOGRECTYPE_PSEUDOFIXEDLENGTH && (parts->record_length - - log_record_type_descriptor[type].compresed_LSN * 2) <= + log_record_type_descriptor[type].compressed_LSN * 2) <= log_record_type_descriptor[type].fixed_length)); translog_lock(); - DBUG_PRINT("info", ("horizon (%lu,0x%lx)", + DBUG_PRINT("info", ("horizon: (%lu,0x%lx)", (ulong) LSN_FILE_NO(log_descriptor.horizon), (ulong) LSN_OFFSET(log_descriptor.horizon))); - DBUG_ASSERT(log_descriptor.bc.current_page_size <= TRANSLOG_PAGE_SIZE); + DBUG_ASSERT(log_descriptor.bc.current_page_fill <= TRANSLOG_PAGE_SIZE); DBUG_PRINT("info", - ("Page size: %u, record %u, next cond %d", - log_descriptor.bc.current_page_size, + ("Page size: %u record: %u next cond: %d", + log_descriptor.bc.current_page_fill, (parts->record_length - - log_record_type_descriptor[type].compresed_LSN * 2 + 3), - ((((uint) log_descriptor.bc.current_page_size) + + log_record_type_descriptor[type].compressed_LSN * 2 + 3), + ((((uint) log_descriptor.bc.current_page_fill) + (parts->record_length - - log_record_type_descriptor[type].compresed_LSN * 2 + 3)) > + log_record_type_descriptor[type].compressed_LSN * 2 + 3)) > TRANSLOG_PAGE_SIZE))); /* check that there is enough place on current page: (log_record_type_descriptor[type].fixed_length - economized on compressed LSNs) bytes */ - if ((((uint) log_descriptor.bc.current_page_size) + + if ((((uint) log_descriptor.bc.current_page_fill) + (parts->record_length - - log_record_type_descriptor[type].compresed_LSN * 2 + 3)) > + log_record_type_descriptor[type].compressed_LSN * 2 + 3)) > TRANSLOG_PAGE_SIZE) { DBUG_PRINT("info", ("Next page")); @@ -4022,17 +3950,17 @@ static my_bool translog_write_fixed_record(LSN *lsn, (*log_record_type_descriptor[type].inwrite_hook) (type, tcb, lsn, parts)) { - DBUG_RETURN(1); + rc= 1; + goto err; } - /* compress LSNs */ if (log_record_type_descriptor[type].class == LOGRECTYPE_PSEUDOFIXEDLENGTH) { - DBUG_ASSERT(log_record_type_descriptor[type].compresed_LSN > 0); + DBUG_ASSERT(log_record_type_descriptor[type].compressed_LSN > 0); if (translog_relative_LSN_encode(parts, *lsn, log_record_type_descriptor[type]. - compresed_LSN, compressed_LSNs)) + compressed_LSN, compressed_LSNs)) { rc= 1; goto err; @@ -4040,14 +3968,13 @@ static my_bool translog_write_fixed_record(LSN *lsn, } /* - Write the whole record at once (we sure that there is enough place on - the destination page + Write the whole record at once (we know that there is enough place on + the destination page) */ - DBUG_ASSERT(parts->current != 0); /* first part is left for - header */ + DBUG_ASSERT(parts->current != 0); /* first part is left for header */ parts->total_record_length+= (part.len= 1 + 2); part.buff= chunk1_header; - *chunk1_header= (uchar) (type |TRANSLOG_CHUNK_FIXED); + *chunk1_header= (byte) (type | TRANSLOG_CHUNK_FIXED); int2store(chunk1_header + 1, short_trid); parts->current--; set_dynamic(&parts->parts, (gptr) &part, parts->current); @@ -4057,6 +3984,7 @@ static my_bool translog_write_fixed_record(LSN *lsn, parts->total_record_length, parts); log_descriptor.bc.buffer->last_lsn= *lsn; + err: rc|= translog_unlock(); @@ -4090,8 +4018,8 @@ err: 0 sign of the end of parts RETURN - 0 - OK - 1 - Error + 0 OK + 1 Error */ my_bool translog_write_record(LSN *lsn, @@ -4099,13 +4027,15 @@ my_bool translog_write_record(LSN *lsn, SHORT_TRANSACTION_ID short_trid, void *tcb, translog_size_t part1_length, - uchar *part1_buff, ...) + byte *part1_buff, ...) { struct st_translog_parts parts; + struct st_translog_part part; va_list pvar; int rc; DBUG_ENTER("translog_write_record"); - DBUG_PRINT("enter", ("type %u, ShortTrID %u", (uint) type, (uint)short_trid)); + DBUG_PRINT("enter", ("type: %u ShortTrID: %u", + (uint) type, (uint)short_trid)); /* move information about parts into dynamic array */ if (init_dynamic_array(&parts.parts, sizeof(struct st_translog_part), @@ -4114,58 +4044,55 @@ my_bool translog_write_record(LSN *lsn, UNRECOVERABLE_ERROR(("init array failed")); DBUG_RETURN(1); } + + /* reserve place for header */ + parts.current= 1; + part.len= 0; + part.buff= 0; + if (insert_dynamic(&parts.parts, (gptr) &part)) { - struct st_translog_part part; + UNRECOVERABLE_ERROR(("insert into array failed")); + DBUG_RETURN(1); + } - /* reserve place for header */ - parts.current= 1; - part.len= 0; - part.buff= 0; - if (insert_dynamic(&parts.parts, (gptr) &part)) - { - UNRECOVERABLE_ERROR(("insert into array failed")); - DBUG_RETURN(1); - } + parts.record_length= part.len= part1_length; + part.buff= part1_buff; + if (insert_dynamic(&parts.parts, (gptr) &part)) + { + UNRECOVERABLE_ERROR(("insert into array failed")); + DBUG_RETURN(1); + } + DBUG_PRINT("info", ("record length: %lu %lu ...", + (ulong) parts.record_length, + (ulong) parts.total_record_length)); - parts.record_length= part.len= part1_length; - part.buff= part1_buff; + /* count record length */ + va_start(pvar, part1_buff); + for (;;) + { + part.len= va_arg(pvar, translog_size_t); + if (part.len == 0) + break; + parts.record_length+= part.len; + part.buff= va_arg(pvar, byte*); if (insert_dynamic(&parts.parts, (gptr) &part)) { UNRECOVERABLE_ERROR(("insert into array failed")); DBUG_RETURN(1); } - DBUG_PRINT("info", ("record length: %lu, %lu ...", + DBUG_PRINT("info", ("record length: %lu %lu ...", (ulong) parts.record_length, (ulong) parts.total_record_length)); - - /* count record length */ - va_start(pvar, part1_buff); - for (;;) - { - part.len= va_arg(pvar, translog_size_t); - if (part.len == 0) - break; - parts.record_length+= part.len; - part.buff= va_arg(pvar, uchar*); - if (insert_dynamic(&parts.parts, (gptr) &part)) - { - UNRECOVERABLE_ERROR(("insert into array failed")); - DBUG_RETURN(1); - } - DBUG_PRINT("info", ("record length: %lu, %lu ...", - (ulong) parts.record_length, - (ulong) parts.total_record_length)); - } - va_end(pvar); - - /* - start total_record_length from record_length then overhead will - be add - */ - parts.total_record_length= parts.record_length; } va_end(pvar); - DBUG_PRINT("info", ("record length: %lu, %lu", + + /* + Start total_record_length from record_length then overhead will + be add + */ + parts.total_record_length= parts.record_length; + va_end(pvar); + DBUG_PRINT("info", ("record length: %lu %lu", (ulong) parts.record_length, (ulong) parts.total_record_length)); @@ -4174,19 +4101,14 @@ my_bool translog_write_record(LSN *lsn, (*log_record_type_descriptor[type].prewrite_hook) (type, tcb, &parts)))) { - switch (log_record_type_descriptor[type].class) - { + switch (log_record_type_descriptor[type].class) { case LOGRECTYPE_VARIABLE_LENGTH: - { rc= translog_write_variable_record(lsn, type, short_trid, &parts, tcb); break; - } case LOGRECTYPE_PSEUDOFIXEDLENGTH: case LOGRECTYPE_FIXEDLENGTH: - { rc= translog_write_fixed_record(lsn, type, short_trid, &parts, tcb); break; - } case LOGRECTYPE_NOT_ALLOWED: default: DBUG_ASSERT(0); @@ -4213,11 +4135,11 @@ my_bool translog_write_record(LSN *lsn, position in sources after decoded LSN(s) */ -static uchar *translog_relative_LSN_decode(LSN base_lsn, - uchar *src, uchar *dst, uint lsns) +static byte *translog_relative_LSN_decode(LSN base_lsn, + byte *src, byte *dst, uint lsns) { uint i; - for (i= 0; i < lsns; i++, dst+= 7) + for (i= 0; i < lsns; i++, dst+= LSN_STORE_SIZE) { src= translog_get_LSN_from_diff(base_lsn, src, dst); } @@ -4235,21 +4157,21 @@ static uchar *translog_relative_LSN_decode(LSN base_lsn, buff Buffer to be filled with header data RETURN - 0 - error - number of bytes in TRANSLOG_HEADER_BUFFER::header where stored decoded - part of the header + 0 error + # number of bytes in TRANSLOG_HEADER_BUFFER::header where stored decoded + part of the header */ -translog_size_t translog_fixed_length_header(uchar *page, +translog_size_t translog_fixed_length_header(byte *page, translog_size_t page_offset, TRANSLOG_HEADER_BUFFER *buff) { struct st_log_record_type_descriptor *desc= log_record_type_descriptor + buff->type; - uchar *src= page + page_offset + 3; - uchar *dst= buff->header; - uchar *start= src; - uint lsns= desc->compresed_LSN; + byte *src= page + page_offset + 3; + byte *dst= buff->header; + byte *start= src; + uint lsns= desc->compressed_LSN; uint length= desc->fixed_length + (lsns * 2); DBUG_ENTER("translog_fixed_length_header"); @@ -4260,7 +4182,7 @@ translog_size_t translog_fixed_length_header(uchar *page, { DBUG_ASSERT(lsns > 0); src= translog_relative_LSN_decode(buff->lsn, src, dst, lsns); - lsns*= 7; + lsns*= LSN_STORE_SIZE; dst+= lsns; length-= lsns; buff->compressed_LSN_economy= (uint16) (lsns - (src - start)); @@ -4268,7 +4190,7 @@ translog_size_t translog_fixed_length_header(uchar *page, else buff->compressed_LSN_economy= 0; - memmove(dst, src, length); + memcpy(dst, src, length); buff->non_header_data_start_offset= page_offset + ((src + length) - (page + page_offset)); buff->non_header_data_len= 0; @@ -4320,23 +4242,21 @@ static void translog_scanner_set_horizon(struct st_translog_scanner_data scanner Information about current chunk during scanning RETURN - 0 - OK - 1 - Error + 0 OK + 1 Error */ -static my_bool translog_scanner_set_last_page(struct st_translog_scanner_data +static my_bool translog_scanner_set_last_page(TRANSLOG_SCANNER_DATA *scanner) { my_bool page_ok; scanner->last_file_page= scanner->page_addr; - if (translog_get_last_page_addr(&scanner->last_file_page, &page_ok)) - return (1); - return (0); + return (translog_get_last_page_addr(&scanner->last_file_page, &page_ok)); } /* - Init scanner + Initialize reader scanner SYNOPSIS translog_init_scanner() @@ -4346,29 +4266,30 @@ static my_bool translog_scanner_set_last_page(struct st_translog_scanner_data scanner scanner which have to be inited RETURN - 0 - OK - 1 - Error + 0 OK + 1 Error */ -static my_bool translog_init_scanner(LSN lsn, - my_bool fixed_horizon, - struct st_translog_scanner_data *scanner) -{ - TRANSLOG_VALIDATOR_DATA data= - { - &scanner->page_addr, 0 - }; +my_bool translog_init_scanner(LSN lsn, + my_bool fixed_horizon, + struct st_translog_scanner_data *scanner) +{ + TRANSLOG_VALIDATOR_DATA data; DBUG_ENTER("translog_init_scanner"); DBUG_PRINT("enter", ("LSN: (0x%lu,0x%lx)", (ulong) LSN_FILE_NO(lsn), (ulong) LSN_OFFSET(lsn)); DBUG_ASSERT(LSN_OFFSET(lsn) % TRANSLOG_PAGE_SIZE != 0); + + data.addr= &scanner->page_addr; + data.was_recovered= 0; + scanner->page_offset= LSN_OFFSET(lsn) % TRANSLOG_PAGE_SIZE; scanner->fixed_horizon= fixed_horizon; translog_scanner_set_horizon(scanner); - DBUG_PRINT("info", ("Horizon: (0x%lu,0x%lx)", + DBUG_PRINT("info", ("horizon: (0x%lu,0x%lx)", (ulong) LSN_FILE_NO(scanner->horizon), (ulong) LSN_OFFSET(scanner->horizon))); @@ -4395,14 +4316,14 @@ static my_bool translog_init_scanner(LSN lsn, scanner Information about current chunk during scanning RETURN - 1 - End of the Log - 0 - OK + 1 End of the Log + 0 OK */ -static my_bool translog_scanner_eol(struct st_translog_scanner_data *scanner) +static my_bool translog_scanner_eol(TRANSLOG_SCANNER_DATA *scanner) { DBUG_ENTER("translog_scanner_eol"); DBUG_PRINT("enter", - ("Horizon: (%lu, 0x%lx), Current: (%lu, 0x%lx+0x%x=0x%lx)", + ("Horizon: (%lu, 0x%lx) Current: (%lu, 0x%lx+0x%x=0x%lx)", (ulong) LSN_FILE_NO(scanner->horizon), (ulong) LSN_OFFSET(scanner->horizon), (ulong) LSN_FILE_NO(scanner->page_addr), @@ -4438,10 +4359,10 @@ static my_bool translog_scanner_eol(struct st_translog_scanner_data *scanner) scanner Information about current chunk during scanning RETURN - 1 - End of the Page - 0 - OK + 1 End of the Page + 0 OK */ -static my_bool translog_scanner_eop(struct st_translog_scanner_data *scanner) +static my_bool translog_scanner_eop(TRANSLOG_SCANNER_DATA *scanner) { DBUG_ENTER("translog_scanner_eop"); DBUG_RETURN(scanner->page_offset >= TRANSLOG_PAGE_SIZE || @@ -4458,16 +4379,16 @@ static my_bool translog_scanner_eop(struct st_translog_scanner_data *scanner) scanner Information about current chunk during scanning RETURN - 1 - End of the File - 0 - OK + 1 End of the File + 0 OK */ -static my_bool translog_scanner_eof(struct st_translog_scanner_data *scanner) +static my_bool translog_scanner_eof(TRANSLOG_SCANNER_DATA *scanner) { DBUG_ENTER("translog_scanner_eof"); DBUG_ASSERT(LSN_FILE_NO(scanner->page_addr) == LSN_FILE_NO(scanner->last_file_page)); - DBUG_PRINT("enter", ("curr Page 0x%lx, last page 0x%lx, " - "normal EOF %d", + DBUG_PRINT("enter", ("curr Page: 0x%lx last page: 0x%lx " + "normal EOF: %d", (ulong) LSN_OFFSET(scanner->page_addr), (ulong) LSN_OFFSET(scanner->last_file_page), LSN_OFFSET(scanner->page_addr) == @@ -4489,16 +4410,19 @@ static my_bool translog_scanner_eof(struct st_translog_scanner_data *scanner) scanner Information about current chunk during scanning RETURN - 0 - OK - 1 - Error + 0 OK + 1 Error */ -static my_bool translog_get_next_chunk(struct st_translog_scanner_data *scanner) +static my_bool +translog_get_next_chunk(TRANSLOG_SCANNER_DATA *scanner) { + uint16 len; + TRANSLOG_VALIDATOR_DATA data; DBUG_ENTER("translog_get_next_chunk"); - uint16 len= translog_get_total_chunk_length(scanner->page, - scanner->page_offset); - if (len == 0) + + if ((len= translog_get_total_chunk_length(scanner->page, + scanner->page_offset)) == 0) DBUG_RETURN(1); scanner->page_offset+= len; @@ -4512,7 +4436,7 @@ static my_bool translog_get_next_chunk(struct st_translog_scanner_data *scanner) { if (translog_scanner_eof(scanner)) { - DBUG_PRINT("info", ("horizon (%lu,0x%lx) pageaddr (%lu,0x%lx)", + DBUG_PRINT("info", ("horizon: (%lu,0x%lx) pageaddr: (%lu,0x%lx)", (ulong) LSN_FILE_NO(scanner->horizon), (ulong) LSN_OFFSET(scanner->horizon), (ulong) LSN_FILE_NO(scanner->page_addr), @@ -4530,14 +4454,12 @@ static my_bool translog_get_next_chunk(struct st_translog_scanner_data *scanner) { scanner->page_addr+= TRANSLOG_PAGE_SIZE; /* offset increased */ } - { - TRANSLOG_VALIDATOR_DATA data= - { - &scanner->page_addr, 0 - }; - if ((scanner->page= translog_get_page(&data, scanner->buffer)) == NULL) - DBUG_RETURN(1); - } + + data.addr= &scanner->page_addr; + data.was_recovered= 0; + if ((scanner->page= translog_get_page(&data, scanner->buffer)) == NULL) + DBUG_RETURN(1); + scanner->page_offset= translog_get_first_chunk_offset(scanner->page); if (translog_scanner_eol(scanner)) { @@ -4545,7 +4467,7 @@ static my_bool translog_get_next_chunk(struct st_translog_scanner_data *scanner) scanner->page_offset= 0; DBUG_RETURN(0); } - DBUG_ASSERT(scanner->page[scanner->page_offset] != 0); + DBUG_ASSERT(scanner->page[scanner->page_offset]); } DBUG_RETURN(0); } @@ -4564,35 +4486,34 @@ static my_bool translog_get_next_chunk(struct st_translog_scanner_data *scanner) it differ from LSN page RETURN - 0 - error - number of bytes in TRANSLOG_HEADER_BUFFER::header where stored decoded - part of the header + 0 error + # number of bytes in TRANSLOG_HEADER_BUFFER::header where stored decoded + part of the header */ -translog_size_t translog_variable_length_header(uchar *page, +translog_size_t translog_variable_length_header(byte *page, translog_size_t page_offset, TRANSLOG_HEADER_BUFFER *buff, - struct - st_translog_scanner_data + TRANSLOG_SCANNER_DATA *scanner) { - struct st_log_record_type_descriptor *desc= - log_record_type_descriptor + buff->type; - uchar *src= page + page_offset + 1 + 2; - uchar *dst= buff->header; + struct st_log_record_type_descriptor *desc= (log_record_type_descriptor + + buff->type); + byte *src= page + page_offset + 1 + 2; + byte *dst= buff->header; LSN base_lsn; - uint lsns= desc->compresed_LSN; + uint lsns= desc->compressed_LSN; uint16 chunk_len; uint16 length= desc->read_header_len + (lsns * 2); uint16 buffer_length= length; uint16 body_len; - struct st_translog_scanner_data internal_scanner; + TRANSLOG_SCANNER_DATA internal_scanner; DBUG_ENTER("translog_variable_length_header"); buff->record_length= translog_variable_record_1group_decode_len(&src); chunk_len= uint2korr(src); - DBUG_PRINT("info", ("rec len: %lu, chunk len: %u, length %u, bufflen %u", + DBUG_PRINT("info", ("rec len: %lu chunk len: %u length: %u bufflen: %u", (ulong) buff->record_length, (uint) chunk_len, (uint) length, (uint) buffer_length)); if (chunk_len == 0) @@ -4603,8 +4524,7 @@ translog_size_t translog_variable_length_header(uchar *page, page_rest= TRANSLOG_PAGE_SIZE - (src - page); base_lsn= buff->lsn; - body_len= (page_rest < buff->record_length ? - page_rest : buff->record_length); + body_len= min(page_rest, buff->record_length); } else { @@ -4614,9 +4534,9 @@ translog_size_t translog_variable_length_header(uchar *page, DBUG_PRINT("info", ("multi-group")); grp_no= buff->groups_no= uint2korr(src + 2); - if ((buff->groups= - (TRANSLOG_GROUP*) my_malloc(sizeof(TRANSLOG_GROUP) * buff->groups_no, - MYF(0))) == 0) + if (!(buff->groups= + (TRANSLOG_GROUP*) my_malloc(sizeof(TRANSLOG_GROUP) * grp_no, + MYF(0)))) DBUG_RETURN(0); DBUG_PRINT("info", ("Groups: %u", (uint) grp_no)); src+= (2 + 2); @@ -4627,20 +4547,20 @@ translog_size_t translog_variable_length_header(uchar *page, for (;;) { - uint i; - uint read= grp_no; + uint i, read= grp_no; buff->chunk0_pages++; if (page_rest < grp_no * (7 + 1)) read= page_rest / (7 + 1); - DBUG_PRINT("info", ("Read chunk0 page#%u read %u left %u start from %u", + DBUG_PRINT("info", ("Read chunk0 page#%u read: %u left: %u " + "start from: %u", buff->chunk0_pages, read, grp_no, curr)); for (i= 0; i < read; i++, curr++) { DBUG_ASSERT(curr < buff->groups_no); - buff->groups[curr].addr= lsn7korr(src + i * (7 + 1)); + buff->groups[curr].addr= lsn_korr(src + i * (7 + 1)); buff->groups[curr].num= src[i * (7 + 1) + 7]; - DBUG_PRINT("info", ("group #%u (%lu,0x%lx) chunks %u", + DBUG_PRINT("info", ("group #%u (%lu,0x%lx) chunks: %u", curr, (ulong) LSN_FILE_NO(buff->groups[curr].addr), (ulong) LSN_OFFSET(buff->groups[curr].addr), @@ -4653,16 +4573,16 @@ translog_size_t translog_variable_length_header(uchar *page, { buff->chunk0_data_addr= scanner->page_addr; buff->chunk0_data_addr+= (page_offset + header_to_skip + - i * (7 + 1)); /* offset increased */ + read * (7 + 1)); /* offset increased */ } else { buff->chunk0_data_addr= buff->lsn; /* offset increased */ - buff->chunk0_data_addr+= (header_to_skip + i * (7 + 1)); + buff->chunk0_data_addr+= (header_to_skip + read * (7 + 1)); } - buff->chunk0_data_len= chunk_len - 2 - i * (7 + 1); - DBUG_PRINT("info", ("Data address (%lu,0x%lx), len: %u", + buff->chunk0_data_len= chunk_len - 2 - read * (7 + 1); + DBUG_PRINT("info", ("Data address: (%lu,0x%lx) len: %u", (ulong) LSN_FILE_NO(buff->chunk0_data_addr), (ulong) LSN_OFFSET(buff->chunk0_data_addr), buff->chunk0_data_len)); @@ -4670,7 +4590,7 @@ translog_size_t translog_variable_length_header(uchar *page, } if (scanner == NULL) { - DBUG_PRINT("info", ("use internal scanner for header reding")); + DBUG_PRINT("info", ("use internal scanner for header reading")); scanner= &internal_scanner; translog_init_scanner(buff->lsn, 1, scanner); } @@ -4700,15 +4620,15 @@ translog_size_t translog_variable_length_header(uchar *page, } if (lsns) { - uchar *start= src; + byte *start= src; src= translog_relative_LSN_decode(base_lsn, src, dst, lsns); - lsns*= 7; + lsns*= LSN_STORE_SIZE; dst+= lsns; length-= lsns; buff->record_length+= (buff->compressed_LSN_economy= (uint16) (lsns - (src - start))); - DBUG_PRINT("info", ("lsns: %u, length %u, economy %u, new length %lu", - lsns / 7, (uint) length, + DBUG_PRINT("info", ("lsns: %u length: %u economy: %u new length: %lu", + lsns / LSN_STORE_SIZE, (uint) length, (uint) buff->compressed_LSN_economy, (ulong) buff->record_length)); body_len-= (src - start); @@ -4718,10 +4638,10 @@ translog_size_t translog_variable_length_header(uchar *page, DBUG_ASSERT(body_len >= length); body_len-= length; - memmove(dst, src, length); + memcpy(dst, src, length); buff->non_header_data_start_offset= src + length - page; buff->non_header_data_len= body_len; - DBUG_PRINT("info", ("non_header_data_start_offset %u len %u buffer %u", + DBUG_PRINT("info", ("non_header_data_start_offset: %u len: %u buffer: %u", buff->non_header_data_start_offset, buff->non_header_data_len, buffer_length)); DBUG_RETURN(buffer_length); @@ -4736,17 +4656,16 @@ translog_size_t translog_variable_length_header(uchar *page, page page content buffer page_offset offset of the chunk in the page buff destination buffer - scanner if it is need this scanner will be moved to the + scanner If this is set the scanner will be moved to the record header page (differ from LSN page in case of - multi-group records + multi-group records) */ translog_size_t -translog_read_record_header_from_buffer(uchar *page, +translog_read_record_header_from_buffer(byte *page, uint16 page_offset, TRANSLOG_HEADER_BUFFER *buff, - struct - st_translog_scanner_data *scanner) + TRANSLOG_SCANNER_DATA *scanner) { DBUG_ENTER("translog_read_record_header_from_buffer"); DBUG_ASSERT((page[page_offset] & TRANSLOG_CHUNK_TYPE) == @@ -4760,8 +4679,7 @@ translog_read_record_header_from_buffer(uchar *page, (ulong) LSN_FILE_NO(buff->lsn), (ulong) LSN_OFFSET(buff->lsn))); /* Read required bytes from the header and call hook */ - switch (log_record_type_descriptor[buff->type].class) - { + switch (log_record_type_descriptor[buff->type].class) { case LOGRECTYPE_VARIABLE_LENGTH: DBUG_RETURN(translog_variable_length_header(page, page_offset, buff, scanner)); @@ -4771,7 +4689,7 @@ translog_read_record_header_from_buffer(uchar *page, default: DBUG_ASSERT(0); } - DBUG_RETURN(0); + DBUG_RETURN(0); /* purecov: deadcode */ } @@ -4794,17 +4712,18 @@ translog_read_record_header_from_buffer(uchar *page, length) RETURN - 0 - error - number of bytes in TRANSLOG_HEADER_BUFFER::header where stored decoded - part of the header + 0 error + # number of bytes in TRANSLOG_HEADER_BUFFER::header where stored decoded + part of the header */ translog_size_t translog_read_record_header(LSN lsn, TRANSLOG_HEADER_BUFFER *buff) { - uchar buffer[TRANSLOG_PAGE_SIZE], *page; + byte buffer[TRANSLOG_PAGE_SIZE], *page; translog_size_t page_offset= LSN_OFFSET(lsn) % TRANSLOG_PAGE_SIZE; - + TRANSLOG_ADDRESS addr; + TRANSLOG_VALIDATOR_DATA data; DBUG_ENTER("translog_read_record_header"); DBUG_PRINT("enter", ("LSN: (0x%lu,0x%lx)", (ulong) LSN_FILE_NO(lsn), (ulong) LSN_OFFSET(lsn))); @@ -4812,16 +4731,12 @@ translog_size_t translog_read_record_header(LSN lsn, buff->lsn= lsn; buff->groups_no= 0; - { - TRANSLOG_ADDRESS addr= lsn; - TRANSLOG_VALIDATOR_DATA data= - { - &addr, 0 - }; - addr-= page_offset; /* offset decreasing */ - if ((page= translog_get_page(&data, buffer)) == NULL) - DBUG_RETURN(0); - } + data.addr= &addr; + data.was_recovered= 0; + addr= lsn; + addr-= page_offset; /* offset decreasing */ + if (!(page= translog_get_page(&data, buffer))) + DBUG_RETURN(0); DBUG_RETURN(translog_read_record_header_from_buffer(page, page_offset, buff, 0)); @@ -4846,20 +4761,20 @@ translog_size_t translog_read_record_header(LSN lsn, length) RETURN - 0 - error - number of bytes in TRANSLOG_HEADER_BUFFER::header where stored decoded - part of the header + 0 error + # number of bytes in TRANSLOG_HEADER_BUFFER::header where stored decoded + part of the header */ translog_size_t -translog_read_record_header_scan(struct st_translog_scanner_data +translog_read_record_header_scan(TRANSLOG_SCANNER_DATA *scanner, TRANSLOG_HEADER_BUFFER *buff, my_bool move_scanner) { DBUG_ENTER("translog_read_record_header_scan"); - DBUG_PRINT("enter", ("Scanner: Cur: (%lu,0x%lx), Hrz: (%lu,0x%lx), " - "Lst: (%lu,0x%lx), Offset: %u(%x), fixed %d", + DBUG_PRINT("enter", ("Scanner: Cur: (%lu,0x%lx) Hrz: (%lu,0x%lx) " + "Lst: (%lu,0x%lx) Offset: %u(%x) fixed %d", (ulong) LSN_FILE_NO(scanner->page_addr), (ulong) LSN_OFFSET(scanner->page_addr), (ulong) LSN_FILE_NO(scanner->horizon), @@ -4885,59 +4800,37 @@ translog_read_record_header_scan(struct st_translog_scanner_data SYNOPSIS translog_read_next_record_header() - lsn log record serial number (address of the record) - previous to the record which will be read - If LSN present scanner will be initialized from it, - do not use LSN after initialization for fast scanning. - buff log record header buffer - fixed_horizon true if it is OK do not read records which was written - after scanning beginning scanner data for scanning if lsn is NULL scanner data will be used for continue scanning. The scanner can be NULL. + buff log record header buffer NOTE - - lsn can point to TRANSLOG_HEADER_BUFFER::lsn and it will be processed - correctly (lsn in buffer will be replaced by next record, but initial - lsn will be read correctly). - it is like translog_read_record_header, but read next record, so see its NOTES. - in case of end of the log buff->lsn will be set to - (CONTROL_FILE_IMPOSSIBLE_FILENO, 0) + (CONTROL_FILE_IMPOSSIBLE_LSN) RETURN - 0 - error - TRANSLOG_RECORD_HEADER_MAX_SIZE + 1 - End of the log - number of bytes in TRANSLOG_HEADER_BUFFER::header where stored decoded - part of the header + 0 error + TRANSLOG_RECORD_HEADER_MAX_SIZE + 1 End of the log + # number of bytes in + TRANSLOG_HEADER_BUFFER::header + where stored decoded + part of the header */ -translog_size_t translog_read_next_record_header(LSN lsn, - TRANSLOG_HEADER_BUFFER *buff, - my_bool fixed_horizon, - struct - st_translog_scanner_data - *scanner) +translog_size_t translog_read_next_record_header(TRANSLOG_SCANNER_DATA + *scanner, + TRANSLOG_HEADER_BUFFER *buff) { - struct st_translog_scanner_data internal_scanner; uint8 chunk_type; buff->groups_no= 0; /* to be sure that we will free it right */ DBUG_ENTER("translog_read_next_record_header"); DBUG_PRINT("enter", ("scanner: 0x%lx", (ulong) scanner)); - if (scanner == NULL) - { - DBUG_ASSERT(lsn != CONTROL_FILE_IMPOSSIBLE_LSN); - scanner= &internal_scanner; - } - if (lsn) - { - if (translog_init_scanner(lsn, fixed_horizon, scanner)) - DBUG_RETURN(0); - DBUG_ASSERT(LSN_OFFSET(lsn) % TRANSLOG_PAGE_SIZE != 0); - } - DBUG_PRINT("info", ("Scanner: Cur: (%lu,0x%lx), Hrz: (%lu,0x%lx), " - "Lst: (%lu,0x%lx), Offset: %u(%x), fixed %d", + DBUG_PRINT("info", ("Scanner: Cur: (%lu,0x%lx) Hrz: (%lu,0x%lx) " + "Lst: (%lu,0x%lx) Offset: %u(%x) fixed: %d", (ulong) LSN_FILE_NO(scanner->page_addr), (ulong) LSN_OFFSET(scanner->page_addr), (ulong) LSN_FILE_NO(scanner->horizon), @@ -4952,7 +4845,7 @@ translog_size_t translog_read_next_record_header(LSN lsn, if (translog_get_next_chunk(scanner)) DBUG_RETURN(0); chunk_type= scanner->page[scanner->page_offset] & TRANSLOG_CHUNK_TYPE; - DBUG_PRINT("info", ("type %x, byte %x", (uint) chunk_type, + DBUG_PRINT("info", ("type: %x byte: %x", (uint) chunk_type, (uint) scanner->page[scanner->page_offset])); } while (chunk_type != TRANSLOG_CHUNK_LSN && chunk_type != TRANSLOG_CHUNK_FIXED && scanner->page[scanner->page_offset] != 0); @@ -4961,7 +4854,8 @@ translog_size_t translog_read_next_record_header(LSN lsn, { /* Last record was read */ buff->lsn= CONTROL_FILE_IMPOSSIBLE_LSN; - DBUG_RETURN(TRANSLOG_RECORD_HEADER_MAX_SIZE + 1); /* just it is not error */ + /* Return 'end of log' marker */ + DBUG_RETURN(TRANSLOG_RECORD_HEADER_MAX_SIZE + 1); } DBUG_RETURN(translog_read_record_header_scan(scanner, buff, 0)); } @@ -4976,8 +4870,8 @@ translog_size_t translog_read_next_record_header(LSN lsn, data data cursor RETURN - 0 - OK - 1 - Error + 0 OK + 1 Error */ static my_bool translog_record_read_next_chunk(struct st_translog_reader_data @@ -4986,7 +4880,6 @@ static my_bool translog_record_read_next_chunk(struct st_translog_reader_data translog_size_t new_current_offset= data->current_offset + data->chunk_size; uint16 chunk_header_len, chunk_len; uint8 type; - DBUG_ENTER("translog_record_read_next_chunk"); if (data->eor) @@ -5002,7 +4895,7 @@ static my_bool translog_record_read_next_chunk(struct st_translog_reader_data /* Goto next group */ data->current_group++; data->current_chunk= 0; - DBUG_PRINT("info", ("skip to group #%u", data->current_group)); + DBUG_PRINT("info", ("skip to group: #%u", data->current_group)); translog_init_scanner(data->header.groups[data->current_group].addr, 1, &data->scanner); } @@ -5017,7 +4910,7 @@ static my_bool translog_record_read_next_chunk(struct st_translog_reader_data if (type == TRANSLOG_CHUNK_LSN && data->header.groups_no) { DBUG_PRINT("info", - ("Last chunk: data len %u, offset %u group %u of %u", + ("Last chunk: data len: %u offset: %u group: %u of %u", data->header.chunk0_data_len, data->scanner.page_offset, data->current_group, data->header.groups_no - 1)); DBUG_ASSERT(data->header.groups_no - 1 == data->current_group); @@ -5045,8 +4938,8 @@ static my_bool translog_record_read_next_chunk(struct st_translog_reader_data data->chunk_size= chunk_len - chunk_header_len; data->body_offset= data->scanner.page_offset + chunk_header_len; data->current_offset= new_current_offset; - DBUG_PRINT("info", ("grp: %u chunk %u body_offset %u, chunk_size %u, " - "current_offset %lu", + DBUG_PRINT("info", ("grp: %u chunk: %u body_offset: %u chunk_size: %u " + "current_offset: %lu", (uint) data->current_group, (uint) data->current_chunk, (uint) data->body_offset, @@ -5064,8 +4957,8 @@ static my_bool translog_record_read_next_chunk(struct st_translog_reader_data data reader data to initialize RETURN - 0 - OK - 1 - Error + 0 OK + 1 Error */ static my_bool translog_init_reader_data(LSN lsn, @@ -5073,8 +4966,8 @@ static my_bool translog_init_reader_data(LSN lsn, { DBUG_ENTER("translog_init_reader_data"); if (translog_init_scanner(lsn, 1, &data->scanner) || - (data->read_header= - translog_read_record_header_scan(&data->scanner, &data->header, 1)) == 0) + !(data->read_header= + translog_read_record_header_scan(&data->scanner, &data->header, 1))) { DBUG_RETURN(1); } @@ -5084,8 +4977,8 @@ static my_bool translog_init_reader_data(LSN lsn, data->current_group= 0; data->current_chunk= 0; data->eor= 0; - DBUG_PRINT("info", ("read_header %u, " - "body_offset %u, chunk_size %u, current_offset %lu", + DBUG_PRINT("info", ("read_header: %u " + "body_offset: %u chunk_size: %u current_offset: %lu", (uint) data->read_header, (uint) data->body_offset, (uint) data->chunk_size, (ulong) data->current_offset)); @@ -5099,10 +4992,10 @@ static my_bool translog_init_reader_data(LSN lsn, SYNOPSIS translog_read_record_header() lsn log record serial number (address of the record) - offset from the beginning of the record beginning (read + offset From the beginning of the record beginning (read§ by translog_read_record_header). - length length of record part which have to be read. - buffer buffer where to read the record part (have to be at + length Length of record part which have to be read. + buffer Buffer where to read the record part (have to be at least 'length' bytes length) RETURN @@ -5112,13 +5005,12 @@ static my_bool translog_init_reader_data(LSN lsn, translog_size_t translog_read_record(LSN lsn, translog_size_t offset, translog_size_t length, - uchar *buffer, + byte *buffer, struct st_translog_reader_data *data) { translog_size_t requested_length= length; translog_size_t end= offset + length; struct st_translog_reader_data internal_data; - DBUG_ENTER("translog_read_record"); if (data == NULL) @@ -5133,9 +5025,9 @@ translog_size_t translog_read_record(LSN lsn, if (translog_init_reader_data(lsn, data)) DBUG_RETURN(0); } - DBUG_PRINT("info", ("Offset %lu, length %lu " - "Scanner: Cur: (%lu,0x%lx), Hrz: (%lu,0x%lx), " - "Lst: (%lu,0x%lx), Offset: %u(%x), fixed %d", + DBUG_PRINT("info", ("Offset: %lu length: %lu " + "Scanner: Cur: (%lu,0x%lx) Hrz: (%lu,0x%lx) " + "Lst: (%lu,0x%lx) Offset: %u(%x) fixed: %d", (ulong) offset, (ulong) length, (ulong) LSN_FILE_NO(data->scanner.page_addr), (ulong) LSN_OFFSET(data->scanner.page_addr), @@ -5148,36 +5040,37 @@ translog_size_t translog_read_record(LSN lsn, data->scanner.fixed_horizon)); if (offset < data->read_header) { + uint16 len= min(data->read_header, end) - offset; DBUG_PRINT("info", - ("enter header offset %lu, length %lu", + ("enter header offset: %lu length: %lu", (ulong) offset, (ulong) length)); - uint16 len= (data->read_header < end ? data->read_header : end) - offset; - memmove(buffer, data->header.header + offset, len); + memcpy(buffer, data->header.header + offset, len); length-= len; if (length == 0) DBUG_RETURN(requested_length); offset+= len; buffer+= len; DBUG_PRINT("info", - ("len: %u, offset %lu, curr %lu, length %lu", + ("len: %u offset: %lu curr: %lu length: %lu", len, (ulong) offset, (ulong) data->current_offset, (ulong) length)); } /* TODO: find first page which we should read by offset */ /* read the record chunk by chunk */ - do + for(;;) { uint page_end= data->current_offset + data->chunk_size; DBUG_PRINT("info", - ("enter body offset %lu, curr %lu, length %lu page_end %lu", + ("enter body offset: %lu curr: %lu " + "length: %lu page_end: %lu", (ulong) offset, (ulong) data->current_offset, (ulong) length, (ulong) page_end)); if (offset < page_end) { - DBUG_ASSERT(offset >= data->current_offset); uint len= page_end - offset; - memmove(buffer, + DBUG_ASSERT(offset >= data->current_offset); + memcpy(buffer, data->scanner.page + data->body_offset + (offset - data->current_offset), len); length-= len; @@ -5186,15 +5079,13 @@ translog_size_t translog_read_record(LSN lsn, offset+= len; buffer+= len; DBUG_PRINT("info", - ("len: %u, offset %lu, curr %lu, length %lu", + ("len: %u offset: %lu curr: %lu length: %lu", len, (ulong) offset, (ulong) data->current_offset, (ulong) length)); } if (translog_record_read_next_chunk(data)) DBUG_RETURN(requested_length - length); - } while (length != 0); - - DBUG_RETURN(requested_length); + } } @@ -5208,61 +5099,63 @@ translog_size_t translog_read_record(LSN lsn, static void translog_force_current_buffer_to_finish() { TRANSLOG_ADDRESS new_buff_begunning; - uint8 old_buffer_no= log_descriptor.bc.buffer_no; - uint8 new_buffer_no= (old_buffer_no + 1) % TRANSLOG_BUFFERS_NO; - struct st_translog_buffer *new_buffer= log_descriptor.buffers + new_buffer_no; + uint16 old_buffer_no= log_descriptor.bc.buffer_no; + uint16 new_buffer_no= (old_buffer_no + 1) % TRANSLOG_BUFFERS_NO; + struct st_translog_buffer *new_buffer= (log_descriptor.buffers + + new_buffer_no); struct st_translog_buffer *old_buffer= log_descriptor.bc.buffer; - uchar *data= log_descriptor.bc.ptr -log_descriptor.bc.current_page_size; - uint16 left= TRANSLOG_PAGE_SIZE - log_descriptor.bc.current_page_size; - uint16 current_page_size; - - new_buff_begunning= log_descriptor.bc.buffer->offset; - new_buff_begunning+= log_descriptor.bc.buffer->size; /* increase offset */ - + byte *data= log_descriptor.bc.ptr -log_descriptor.bc.current_page_fill; + uint16 left= TRANSLOG_PAGE_SIZE - log_descriptor.bc.current_page_fill; + uint16 current_page_fill, write_counter, previous_offset; DBUG_ENTER("translog_force_current_buffer_to_finish"); - DBUG_PRINT("enter", ("Buffer #%u 0x%lx, " - "Buffer addr (%lu,0x%lx), " - "Page addr: (%lu,0x%lx), " - "New Buff: (%lu,0x%lx), " - "size %lu (%lu), Pg: %u, left: %u", + DBUG_PRINT("enter", ("Buffer #%u 0x%lx " + "Buffer addr: (%lu,0x%lx) " + "Page addr: (%lu,0x%lx) " + "New Buff: (%lu,0x%lx) " + "size: %lu (%lu) Pg: %u left: %u", (uint) log_descriptor.bc.buffer_no, (ulong) log_descriptor.bc.buffer, (ulong) LSN_FILE_NO(log_descriptor.bc.buffer->offset), (ulong) LSN_OFFSET(log_descriptor.bc.buffer->offset), (ulong) LSN_FILE_NO(log_descriptor.horizon), (ulong) (LSN_OFFSET(log_descriptor.horizon) - - log_descriptor.bc.current_page_size), + log_descriptor.bc.current_page_fill), (ulong) LSN_FILE_NO(new_buff_begunning), (ulong) LSN_OFFSET(new_buff_begunning), (ulong) log_descriptor.bc.buffer->size, (ulong) (log_descriptor.bc.ptr -log_descriptor.bc. buffer->buffer), - (uint) log_descriptor.bc.current_page_size, + (uint) log_descriptor.bc.current_page_fill, (uint) left)); + + new_buff_begunning= log_descriptor.bc.buffer->offset; + new_buff_begunning+= log_descriptor.bc.buffer->size; /* increase offset */ + DBUG_ASSERT(log_descriptor.bc.ptr !=NULL); DBUG_ASSERT((log_descriptor.bc.ptr -log_descriptor.bc.buffer->buffer) - %TRANSLOG_PAGE_SIZE == - log_descriptor.bc.current_page_size % TRANSLOG_PAGE_SIZE); + % TRANSLOG_PAGE_SIZE == + log_descriptor.bc.current_page_fill % TRANSLOG_PAGE_SIZE); DBUG_ASSERT(LSN_FILE_NO(log_descriptor.horizon) == LSN_FILE_NO(log_descriptor.bc.buffer->offset)); DBUG_ASSERT(LSN_OFFSET(log_descriptor.bc.buffer->offset) + (log_descriptor.bc.ptr -log_descriptor.bc.buffer->buffer) == LSN_OFFSET(log_descriptor.horizon)); - if (left != TRANSLOG_PAGE_SIZE && left != 0) + DBUG_ASSERT(left < TRANSLOG_PAGE_SIZE); + if (left != 0) { /* TODO: if 'left' is so small that can't hold any other record then do not move the page */ - DBUG_PRINT("info", ("left %u", (uint) left)); + DBUG_PRINT("info", ("left: %u", (uint) left)); /* decrease offset */ - new_buff_begunning-= log_descriptor.bc.current_page_size; - current_page_size= log_descriptor.bc.current_page_size; + new_buff_begunning-= log_descriptor.bc.current_page_fill; + current_page_fill= log_descriptor.bc.current_page_fill; bzero(log_descriptor.bc.ptr, left); log_descriptor.bc.buffer->size+= left; - DBUG_PRINT("info", ("Finish Page buffer #%u: 0x%lx, " + DBUG_PRINT("info", ("Finish Page buffer #%u: 0x%lx " "Size: %lu", (uint) log_descriptor.bc.buffer->buffer_no, (ulong) log_descriptor.bc.buffer, @@ -5273,28 +5166,26 @@ static void translog_force_current_buffer_to_finish() else { left= 0; - log_descriptor.bc.current_page_size= 0; + log_descriptor.bc.current_page_fill= 0; } translog_buffer_lock(new_buffer); translog_wait_for_buffer_free(new_buffer); - { - uint16 write_counter= log_descriptor.bc.write_counter; - uint16 previous_offset= log_descriptor.bc.previous_offset; - translog_start_buffer(new_buffer, &log_descriptor.bc, new_buffer_no); - log_descriptor.bc.buffer->offset= new_buff_begunning; - log_descriptor.bc.write_counter= write_counter; - log_descriptor.bc.previous_offset= previous_offset; - } + write_counter= log_descriptor.bc.write_counter; + previous_offset= log_descriptor.bc.previous_offset; + translog_start_buffer(new_buffer, &log_descriptor.bc, new_buffer_no); + log_descriptor.bc.buffer->offset= new_buff_begunning; + log_descriptor.bc.write_counter= write_counter; + log_descriptor.bc.previous_offset= previous_offset; - if (log_descriptor.flags & TRANSLOG_SECTOR_PROTECTION) + if (data[TRANSLOG_PAGE_FLAGS] & TRANSLOG_SECTOR_PROTECTION) { translog_put_sector_protection(data, &log_descriptor.bc); if (left) { log_descriptor.bc.write_counter++; - log_descriptor.bc.previous_offset= current_page_size; + log_descriptor.bc.previous_offset= current_page_fill; } else { @@ -5304,21 +5195,21 @@ static void translog_force_current_buffer_to_finish() } } - if (log_descriptor.flags & TRANSLOG_PAGE_CRC) + if (data[TRANSLOG_PAGE_FLAGS] & TRANSLOG_PAGE_CRC) { - uint32 crc= translog_adler_crc(data + log_descriptor.page_overhead, - TRANSLOG_PAGE_SIZE - - log_descriptor.page_overhead); + uint32 crc= translog_crc(data + log_descriptor.page_overhead, + TRANSLOG_PAGE_SIZE - + log_descriptor.page_overhead); DBUG_PRINT("info", ("CRC: 0x%lx", (ulong) crc)); int4store(data + 3 + 3 + 1, crc); } if (left) { - memmove(new_buffer->buffer, data, current_page_size); - log_descriptor.bc.ptr +=current_page_size; - log_descriptor.bc.buffer->size= log_descriptor.bc.current_page_size= - current_page_size; + memcpy(new_buffer->buffer, data, current_page_fill); + log_descriptor.bc.ptr +=current_page_fill; + log_descriptor.bc.buffer->size= log_descriptor.bc.current_page_fill= + current_page_fill; new_buffer->overlay= old_buffer; } else @@ -5327,6 +5218,7 @@ static void translog_force_current_buffer_to_finish() DBUG_VOID_RETURN; } + /* Flush the log up to given LSN (included) @@ -5336,8 +5228,8 @@ static void translog_force_current_buffer_to_finish() the log have to be flushed RETURN - 0 - OK - 1 - Error + 0 OK + 1 Error */ my_bool translog_flush(LSN lsn) @@ -5346,9 +5238,8 @@ my_bool translog_flush(LSN lsn) int rc= 0; uint i; my_bool full_circle= 0; - DBUG_ENTER("translog_flush"); - DBUG_PRINT("enter", ("Flush up to LSN (%lu,0x%lx)", + DBUG_PRINT("enter", ("Flush up to LSN: (%lu,0x%lx)", (ulong) LSN_FILE_NO(lsn), (ulong) LSN_OFFSET(lsn))); @@ -5356,16 +5247,15 @@ my_bool translog_flush(LSN lsn) old_flushed= log_descriptor.flushed; for (;;) { - uint8 buffer_no= log_descriptor.bc.buffer_no; - uint8 buffer_start= buffer_no; + uint16 buffer_no= log_descriptor.bc.buffer_no; + uint16 buffer_start= buffer_no; struct st_translog_buffer *buffer_unlock= log_descriptor.bc.buffer; - struct st_translog_buffer *buffer= log_descriptor.bc.buffer; /* we can't flush in future */ DBUG_ASSERT(cmp_translog_addr(log_descriptor.horizon, lsn) >= 0); if (cmp_translog_addr(log_descriptor.flushed, lsn) >= 0) { - DBUG_PRINT("info", ("already flushed (%lu,0x%lx)", + DBUG_PRINT("info", ("already flushed: (%lu,0x%lx)", (ulong) LSN_FILE_NO(log_descriptor.flushed), (ulong) LSN_OFFSET(log_descriptor.flushed))); translog_unlock(); @@ -5383,7 +5273,7 @@ my_bool translog_flush(LSN lsn) translog_buffer_lock(buffer); translog_buffer_unlock(buffer_unlock); buffer_unlock= buffer; - if (buffer->file) + if (buffer->file != -1) { buffer_unlock= NULL; if (buffer_start == buffer_no) @@ -5398,12 +5288,10 @@ my_bool translog_flush(LSN lsn) cmp_translog_addr(log_descriptor.flushed, lsn) < 0); if (buffer_unlock != NULL) translog_buffer_unlock(buffer_unlock); - if (translog_buffer_flush(buffer)) - { - translog_buffer_unlock(buffer); - DBUG_RETURN(1); - } + rc= translog_buffer_flush(buffer); translog_buffer_unlock(buffer); + if (rc) + DBUG_RETURN(1); if (!full_circle) translog_lock(); } @@ -5417,10 +5305,10 @@ my_bool translog_flush(LSN lsn) OPENED_FILES_NUM) { /* file in the cache */ - if (log_descriptor.log_file_num[cache_index] == 0) + if (log_descriptor.log_file_num[cache_index] == -1) { if ((log_descriptor.log_file_num[cache_index]= - open_logfile_by_number_no_cache(i)) == 0) + open_logfile_by_number_no_cache(i)) == -1) { translog_unlock(); DBUG_RETURN(1); @@ -5429,13 +5317,7 @@ my_bool translog_flush(LSN lsn) file= log_descriptor.log_file_num[cache_index]; rc|= my_sync(file, MYF(MY_WME)); } - else - { - /* very unlike situation with extremely small file size */ - File file= open_logfile_by_number_no_cache(i); - rc|= my_sync(file, MYF(MY_WME)); - my_close(file, MYF(MY_WME)); - } + /* We sync file when we are closing it => do nothing if file closed */ } log_descriptor.flushed= sent_to_file; rc|= my_sync(log_descriptor.directory_fd, MYF(MY_WME)); diff --git a/storage/maria/ma_loghandler.h b/storage/maria/ma_loghandler.h index 02c272361a4..3646afcf52a 100644 --- a/storage/maria/ma_loghandler.h +++ b/storage/maria/ma_loghandler.h @@ -1,13 +1,16 @@ -#ifndef _ma_loghandler_h -#define _ma_loghandler_h - /* Transaction log flags */ #define TRANSLOG_PAGE_CRC 1 #define TRANSLOG_SECTOR_PROTECTION (1<<1) #define TRANSLOG_RECORD_CRC (1<<2) +#define TRANSLOG_FLAGS_NUM ((TRANSLOG_PAGE_CRC | TRANSLOG_SECTOR_PROTECTION | \ + TRANSLOG_RECORD_CRC) + 1) -/* page size in transaction log */ +/* + Page size in transaction log + It should be Power of 2 and multiple of DISK_DRIVE_SECTOR_SIZE + (DISK_DRIVE_SECTOR_SIZE * 2^N) +*/ #define TRANSLOG_PAGE_SIZE (8*1024) #include "ma_loghandler_lsn.h" @@ -15,6 +18,14 @@ /* short transaction ID type */ typedef uint16 SHORT_TRANSACTION_ID; +/* Length of CRC at end of pages */ +#define CRC_LENGTH 4 +/* + Length of disk drive sector size (we assume that writing it + to disk is atomic operation) +*/ +#define DISK_DRIVE_SECTOR_SIZE 512 + /* types of records in the transaction log */ enum translog_record_type { @@ -51,8 +62,9 @@ enum translog_record_type LOGREC_LONG_TRANSACTION_ID= 30, LOGREC_RESERVED_FUTURE_EXTENSION= 63 }; -#define LOGREC_NUMBER_OF_TYPES 64 +#define LOGREC_NUMBER_OF_TYPES 64 /* Maximum, can't be extended */ +/* Size of log file; One log file is restricted to 4G */ typedef uint32 translog_size_t; #define TRANSLOG_RECORD_HEADER_MAX_SIZE 1024 @@ -68,8 +80,8 @@ typedef struct st_translog_header_buffer { /* LSN of the read record */ LSN lsn; - /* type of the read record */ - enum translog_record_type type; + /* array of groups descriptors, can be used only if groups_no > 0 */ + TRANSLOG_GROUP *groups; /* short transaction ID or 0 if it has no sense for the record */ SHORT_TRANSACTION_ID short_trid; /* @@ -77,56 +89,56 @@ typedef struct st_translog_header_buffer hidden part of record (type, short TrID, length) */ translog_size_t record_length; - /* - Real compressed LSN(s) size economy (*7 - ) - */ - uint16 compressed_LSN_economy; /* Buffer for write decoded header of the record (depend on the record type) */ - uchar header[TRANSLOG_RECORD_HEADER_MAX_SIZE]; - /* non read body data offset on the page */ - uint16 non_header_data_start_offset; - /* non read body data length in this first chunk */ - uint16 non_header_data_len; + byte header[TRANSLOG_RECORD_HEADER_MAX_SIZE]; /* number of groups listed in */ uint groups_no; - /* array of groups descriptors, can be used only if groups_no > 0 */ - TRANSLOG_GROUP *groups; /* in multi-group number of chunk0 pages (valid only if groups_no > 0) */ uint chunk0_pages; + /* type of the read record */ + enum translog_record_type type; /* chunk 0 data address (valid only if groups_no > 0) */ TRANSLOG_ADDRESS chunk0_data_addr; + /* + Real compressed LSN(s) size economy (*7 - ) + */ + uint16 compressed_LSN_economy; + /* short transaction ID or 0 if it has no sense for the record */ + uint16 non_header_data_start_offset; + /* non read body data length in this first chunk */ + uint16 non_header_data_len; /* chunk 0 data size (valid only if groups_no > 0) */ uint16 chunk0_data_len; } TRANSLOG_HEADER_BUFFER; -struct st_translog_scanner_data +typedef struct st_translog_scanner_data { - uchar buffer[TRANSLOG_PAGE_SIZE]; /* buffer for page content */ - TRANSLOG_ADDRESS page_addr; /* current page address */ - TRANSLOG_ADDRESS horizon; /* end of the log which we saw - last time */ - TRANSLOG_ADDRESS last_file_page; /* Last page on in this file */ - uchar *page; /* page content pointer */ - translog_size_t page_offset; /* offset of the chunk in the - page */ - my_bool fixed_horizon; /* set horizon only once at - init */ -}; + byte buffer[TRANSLOG_PAGE_SIZE]; /* buffer for page content */ + TRANSLOG_ADDRESS page_addr; /* current page address */ + /* end of the log which we saw last time */ + TRANSLOG_ADDRESS horizon; + TRANSLOG_ADDRESS last_file_page; /* Last page on in this file */ + byte *page; /* page content pointer */ + /* offset of the chunk in the page */ + translog_size_t page_offset; + /* set horizon only once at init */ + my_bool fixed_horizon; +} TRANSLOG_SCANNER_DATA; struct st_translog_reader_data { TRANSLOG_HEADER_BUFFER header; /* Header */ - struct st_translog_scanner_data scanner; /* chunks scanner */ + TRANSLOG_SCANNER_DATA scanner; /* chunks scanner */ translog_size_t body_offset; /* current chunk body offset */ - translog_size_t current_offset; /* data offset from the record - beginning */ - uint16 read_header; /* number of bytes read in - header */ + /* data offset from the record beginning */ + translog_size_t current_offset; + /* number of bytes read in header */ + uint16 read_header; uint16 chunk_size; /* current chunk size */ uint current_group; /* current group */ uint current_chunk; /* current chunk in the group */ @@ -134,181 +146,37 @@ struct st_translog_reader_data }; -/* - Initialize transaction log - - SYNOPSIS - translog_init() - directory Directory where log files are put - log_file_max_size max size of one log size (for new logs creation) - server_version version of MySQL servger (MYSQL_VERSION_ID) - server_id server ID (replication & Co) - pagecache Page cache for the log reads - flags flags (TRANSLOG_PAGE_CRC, TRANSLOG_SECTOR_PROTECTION - TRANSLOG_RECORD_CRC) - - RETURN - 0 - OK - 1 - Error -*/ - my_bool translog_init(const char *directory, uint32 log_file_max_size, - uint32 server_version, - uint32 server_id, PAGECACHE *pagecache, uint flags); - - -/* - Write the log record - - SYNOPSIS - translog_write_record() - lsn LSN of the record will be writen here - type the log record type - short_trid Sort transaction ID or 0 if it has no sense - tcb Transaction control block pointer for hooks by - record log type - partN_length length of Ns part of the log - partN_buffer pointer on Ns part buffer - 0 sign of the end of parts - - RETURN - 0 - OK - 1 - Error -*/ + uint32 server_version, uint32 server_id, + PAGECACHE *pagecache, uint flags); my_bool translog_write_record(LSN *lsn, enum translog_record_type type, SHORT_TRANSACTION_ID short_trid, void *tcb, translog_size_t part1_length, - uchar *part1_buff, ...); - - -/* - Free log handler resources - - SYNOPSIS - translog_destroy() -*/ + byte *part1_buff, ...); void translog_destroy(); - -/* - Read record header and some fixed part of a record (the part depend on - record type). - - SYNOPSIS - translog_read_record_header() - lsn log record serial number (address of the record) - buff log record header buffer - - NOTE - - lsn can point to TRANSLOG_HEADER_BUFFER::lsn and it will be processed - correctly. - - Some type of record can be read completely by this call - - "Decoded" header stored in TRANSLOG_HEADER_BUFFER::header (relative - LSN can be translated to absolute one), some fields can be added - (like actual header length in the record if the header has variable - length) - - RETURN - 0 - error - number of bytes in TRANSLOG_HEADER_BUFFER::header where stored decoded - part of the header -*/ - translog_size_t translog_read_record_header(LSN lsn, TRANSLOG_HEADER_BUFFER *buff); - -/* - Free resources used by TRANSLOG_HEADER_BUFFER - - SYNOPSIS - translog_free_record_header(); -*/ - void translog_free_record_header(TRANSLOG_HEADER_BUFFER *buff); - -/* - Read a part of the record. - - SYNOPSIS - translog_read_record_header() - lsn log record serial number (address of the record) - offset from the beginning of the record beginning (read - by translog_read_record_header). - length length of record part which have to be read. - buffer buffer where to read the record part (have to be at - least 'length' bytes length) - - RETURN - 0 - error (or read out of the record) - length of data actually read -*/ - translog_size_t translog_read_record(LSN lsn, translog_size_t offset, translog_size_t length, - uchar *buffer, + byte *buffer, struct st_translog_reader_data *data); - -/* - Flush the log up to given LSN (included) - - SYNOPSIS - translog_flush() - lsn log record serial number up to which (inclusive) - the log have to be flushed - - RETURN - 0 - OK - 1 - Error -*/ - my_bool translog_flush(LSN lsn); +my_bool translog_init_scanner(LSN lsn, + my_bool fixed_horizon, + struct st_translog_scanner_data *scanner); -/* - Read record header and some fixed part of the next record (the part - depend on record type). - - SYNOPSIS - translog_read_next_record_header() - lsn log record serial number (address of the record) - previous to the record which will be read - If LSN present scanner will be initialized from it, - do not use LSN after initialization for fast scanning. - buff log record header buffer - fixed_horizon true if it is OK do not read records which was written - after scaning begining - scanner data for scaning if lsn is NULL scanner data - will be used for continue scaning. - scanner can be NULL. - - NOTE - - lsn can point to TRANSLOG_HEADER_BUFFER::lsn and it will be processed - correctly (lsn in buffer will be replaced by next record, but initial - lsn will be read correctly). - - it is like translog_read_record_header, but read next record, so see - its NOTES. - - in case of end of the log buff->lsn will be set to - (CONTROL_FILE_IMPOSSIBLE_LOGNO, 0) - RETURN - 0 - error - TRANSLOG_RECORD_HEADER_MAX_SIZE + 1 - End of the log - number of bytes in TRANSLOG_HEADER_BUFFER::header where stored decoded - part of the header -*/ - -translog_size_t translog_read_next_record_header(LSN lsn, - TRANSLOG_HEADER_BUFFER *buff, - my_bool fixed_horizon, - struct - st_translog_scanner_data - *scanner); +translog_size_t translog_read_next_record_header(TRANSLOG_SCANNER_DATA + *scanner, + TRANSLOG_HEADER_BUFFER *buff); -#endif diff --git a/storage/maria/ma_loghandler_lsn.h b/storage/maria/ma_loghandler_lsn.h index 472298de07c..9625f109864 100644 --- a/storage/maria/ma_loghandler_lsn.h +++ b/storage/maria/ma_loghandler_lsn.h @@ -30,15 +30,18 @@ typedef TRANSLOG_ADDRESS LSN; /* checks LSN */ #define LSN_VALID(L) DBUG_ASSERT((L) >= 0 && (L) < (uint64)0xFFFFFFFFFFFFFFLL) +/* size of stored LSN on a disk */ +#define LSN_STORE_SIZE 7 + /* Puts LSN into buffer (dst) */ -#define lsn7store(dst, lsn) \ +#define lsn_store(dst, lsn) \ do { \ int3store((dst), LSN_FILE_NO(lsn)); \ int4store((dst) + 3, LSN_OFFSET(lsn)); \ } while (0) /* Unpacks LSN from the buffer (P) */ -#define lsn7korr(P) MAKE_LSN(uint3korr(P), uint4korr((P) + 3)) +#define lsn_korr(P) MAKE_LSN(uint3korr(P), uint4korr((P) + 3)) /* what we need to add to LSN to increase it on one file */ #define LSN_ONE_FILE ((int64)0x100000000LL) diff --git a/storage/maria/unittest/ma_test_loghandler-t.c b/storage/maria/unittest/ma_test_loghandler-t.c index afbef150b62..54ed4b0d746 100644 --- a/storage/maria/unittest/ma_test_loghandler-t.c +++ b/storage/maria/unittest/ma_test_loghandler-t.c @@ -47,10 +47,10 @@ static const char *default_dbug_option; 1 - Error */ -static my_bool check_content(uchar *ptr, ulong length) +static my_bool check_content(byte *ptr, ulong length) { ulong i; - uchar buff[2]; + byte buff[2]; for (i= 0; i < length; i++) { if (i % 2 == 0) @@ -81,7 +81,7 @@ static my_bool check_content(uchar *ptr, ulong length) */ static my_bool read_and_check_content(TRANSLOG_HEADER_BUFFER *rec, - uchar *buffer, uint skip) + byte *buffer, uint skip) { DBUG_ASSERT(rec->record_length < LONG_BUFFER_SIZE * 2 + 7 * 2 + 2); if (translog_read_record(rec->lsn, 0, rec->record_length, buffer, NULL) != @@ -95,16 +95,16 @@ int main(int argc, char *argv[]) uint32 i; uint32 rec_len; uint pagen; - uchar long_tr_id[6]; - uchar lsn_buff[23]= + byte long_tr_id[6]; + byte lsn_buff[23]= { 0x55, 0xAA, 0x55, 0xAA, 0x55, 0xAA, 0x55, 0xAA, 0x55, 0xAA, 0x55, 0xAA, 0x55, 0xAA, 0x55, 0xAA, 0x55, 0xAA, 0x55, 0xAA, 0x55, 0xAA, 0x55 }; - uchar long_buffer[LONG_BUFFER_SIZE * 2 + 7 * 2 + 2]; + byte long_buffer[LONG_BUFFER_SIZE * 2 + LSN_STORE_SIZE * 2 + 2]; PAGECACHE pagecache; - LSN lsn, lsn_base, first_lsn, lsn_ptr; + LSN lsn, lsn_base, first_lsn; TRANSLOG_HEADER_BUFFER rec; struct st_translog_scanner_data scanner; int rc; @@ -114,7 +114,7 @@ int main(int argc, char *argv[]) bzero(&pagecache, sizeof(pagecache)); maria_data_root= "."; - for (i= 0; i < (LONG_BUFFER_SIZE + 7 * 2 + 2); i+= 2) + for (i= 0; i < (LONG_BUFFER_SIZE + LSN_STORE_SIZE * 2 + 2); i+= 2) { int2store(long_buffer + i, (i >> 1)); /* long_buffer[i]= (i & 0xFF); */ @@ -173,7 +173,7 @@ int main(int argc, char *argv[]) printf("write %d\n", i); if (i % 2) { - lsn7store(lsn_buff, lsn_base); + lsn_store(lsn_buff, lsn_base); if (translog_write_record(&lsn, LOGREC_CLR_END, (i % 0xFFFF), NULL, 7, lsn_buff, 0)) @@ -183,7 +183,7 @@ int main(int argc, char *argv[]) translog_destroy(); exit(1); } - lsn7store(lsn_buff, lsn_base); + lsn_store(lsn_buff, lsn_base); if ((rec_len= random() / (RAND_MAX / (LONG_BUFFER_SIZE + 1))) < 12) rec_len= 12; if (translog_write_record(&lsn, @@ -199,8 +199,8 @@ int main(int argc, char *argv[]) } else { - lsn7store(lsn_buff, lsn_base); - lsn7store(lsn_buff + 7, first_lsn); + lsn_store(lsn_buff, lsn_base); + lsn_store(lsn_buff + LSN_STORE_SIZE, first_lsn); if (translog_write_record(&lsn, LOGREC_UNDO_ROW_DELETE, (i % 0xFFFF), NULL, 23, lsn_buff, 0)) @@ -210,8 +210,8 @@ int main(int argc, char *argv[]) translog_destroy(); exit(1); } - lsn7store(lsn_buff, lsn_base); - lsn7store(lsn_buff + 7, first_lsn); + lsn_store(lsn_buff, lsn_base); + lsn_store(lsn_buff + LSN_STORE_SIZE, first_lsn); if ((rec_len= random() / (RAND_MAX / (LONG_BUFFER_SIZE + 1))) < 19) rec_len= 19; if (translog_write_record(&lsn, @@ -291,7 +291,7 @@ int main(int argc, char *argv[]) } if (rec.type !=LOGREC_LONG_TRANSACTION_ID || rec.short_trid != 0 || rec.record_length != 6 || uint4korr(rec.header) != 0 || - (uint)rec.header[4] != 0 || rec.header[5] != 0xFF || + ((uchar)rec.header[4]) != 0 || ((uchar)rec.header[5]) != 0xFF || first_lsn != rec.lsn) { fprintf(stderr, "Incorrect LOGREC_LONG_TRANSACTION_ID data read(0)\n" @@ -304,12 +304,16 @@ int main(int argc, char *argv[]) goto err; } lsn= first_lsn; - lsn_ptr= first_lsn; + if (translog_init_scanner(first_lsn, 1, &scanner)) + { + fprintf(stderr, "scanner init failed\n"); + goto err; + } for (i= 1;; i++) { if (i % 1000 == 0) printf("read %d\n", i); - len= translog_read_next_record_header(lsn_ptr, &rec, 1, &scanner); + len= translog_read_next_record_header(&scanner, &rec); if (len == 0) { fprintf(stderr, "1-%d translog_read_next_record_header failed (%d)\n", @@ -326,21 +330,21 @@ int main(int argc, char *argv[]) } break; } - /* use scanner after its initialization */ - lsn_ptr= 0; if (i % 2) { LSN ref; - ref= lsn7korr(rec.header); + ref= lsn_korr(rec.header); if (rec.type !=LOGREC_CLR_END || rec.short_trid != (i % 0xFFFF) || rec.record_length != 7 || ref != lsn) { fprintf(stderr, "Incorrect LOGREC_CLR_END data read(%d) " - "type %u, strid %u, len %u, ref(%lu,0x%lx), " + "type: %u strid: %u len: %u" + "ref: (%lu,0x%lx) (%lu,0x%lx) " "lsn(%lu,0x%lx)\n", i, (uint) rec.type, (uint) rec.short_trid, (uint) rec.record_length, (ulong) LSN_FILE_NO(ref), (ulong) LSN_OFFSET(ref), + (ulong) LSN_FILE_NO(lsn), (ulong) LSN_OFFSET(lsn), (ulong) LSN_FILE_NO(rec.lsn), (ulong) LSN_OFFSET(rec.lsn)); goto err; } @@ -348,18 +352,22 @@ int main(int argc, char *argv[]) else { LSN ref1, ref2; - ref1= lsn7korr(rec.header); - ref2= lsn7korr(rec.header + 7); - if (rec.type !=LOGREC_UNDO_ROW_DELETE || + ref1= lsn_korr(rec.header); + ref2= lsn_korr(rec.header + LSN_STORE_SIZE); + if (rec.type != LOGREC_UNDO_ROW_DELETE || rec.short_trid != (i % 0xFFFF) || rec.record_length != 23 || ref1 != lsn || ref2 != first_lsn || - rec.header[22] != 0x55 || rec.header[21] != 0xAA || - rec.header[20] != 0x55 || rec.header[19] != 0xAA || - rec.header[18] != 0x55 || rec.header[17] != 0xAA || - rec.header[16] != 0x55 || rec.header[15] != 0xAA || - rec.header[14] != 0x55) + ((uchar)rec.header[22]) != 0x55 || + ((uchar)rec.header[21]) != 0xAA || + ((uchar)rec.header[20]) != 0x55 || + ((uchar)rec.header[19]) != 0xAA || + ((uchar)rec.header[18]) != 0x55 || + ((uchar)rec.header[17]) != 0xAA || + ((uchar)rec.header[16]) != 0x55 || + ((uchar)rec.header[15]) != 0xAA || + ((uchar)rec.header[14]) != 0x55) { fprintf(stderr, "Incorrect LOGREC_UNDO_ROW_DELETE data read(%d)" "type %u, strid %u, len %u, ref1(%lu,0x%lx), " @@ -378,7 +386,7 @@ int main(int argc, char *argv[]) goto err; } } - len= translog_read_next_record_header(lsn_ptr, &rec, 1, &scanner); + len= translog_read_next_record_header(&scanner, &rec); if (len == 0) { fprintf(stderr, "1-%d translog_read_next_record_header (var) " @@ -394,14 +402,14 @@ int main(int argc, char *argv[]) if (i % 2) { LSN ref; - ref= lsn7korr(rec.header); + ref= lsn_korr(rec.header); if ((rec_len= random() / (RAND_MAX / (LONG_BUFFER_SIZE + 1))) < 12) rec_len= 12; if (rec.type !=LOGREC_UNDO_KEY_INSERT || rec.short_trid != (i % 0xFFFF) || - rec.record_length != rec_len + 7 || + rec.record_length != rec_len + LSN_STORE_SIZE || len != 12 || ref != lsn || - check_content(rec.header + 7, len - 7)) + check_content(rec.header + LSN_STORE_SIZE, len - LSN_STORE_SIZE)) { fprintf(stderr, "Incorrect LOGREC_UNDO_KEY_INSERT data read(%d)" "type %u (%d), strid %u (%d), len %lu, %lu + 7 (%d), " @@ -412,16 +420,17 @@ int main(int argc, char *argv[]) (uint) rec.short_trid, rec.short_trid != (i % 0xFFFF), (ulong) rec.record_length, (ulong) rec_len, - rec.record_length != rec_len + 7, + rec.record_length != rec_len + LSN_STORE_SIZE, (uint) len, len != 12, (ulong) LSN_FILE_NO(ref), (ulong) LSN_OFFSET(ref), (ulong) LSN_FILE_NO(rec.lsn), (ulong) LSN_OFFSET(rec.lsn), (len != 12 || ref != lsn), - check_content(rec.header + 7, len - 7)); + check_content(rec.header + LSN_STORE_SIZE, + len - LSN_STORE_SIZE)); goto err; } - if (read_and_check_content(&rec, long_buffer, 7)) + if (read_and_check_content(&rec, long_buffer, LSN_STORE_SIZE)) { fprintf(stderr, "Incorrect LOGREC_UNDO_KEY_INSERT in whole rec read " @@ -433,20 +442,21 @@ int main(int argc, char *argv[]) else { LSN ref1, ref2; - ref1= lsn7korr(rec.header); - ref2= lsn7korr(rec.header + 7); + ref1= lsn_korr(rec.header); + ref2= lsn_korr(rec.header + LSN_STORE_SIZE); if ((rec_len= random() / (RAND_MAX / (LONG_BUFFER_SIZE + 1))) < 19) rec_len= 19; if (rec.type !=LOGREC_UNDO_KEY_DELETE || rec.short_trid != (i % 0xFFFF) || - rec.record_length != rec_len + 14 || + rec.record_length != rec_len + LSN_STORE_SIZE * 2 || len != 19 || ref1 != lsn || ref2 != first_lsn || - check_content(rec.header + 14, len - 14)) + check_content(rec.header + LSN_STORE_SIZE * 2, + len - LSN_STORE_SIZE * 2)) { fprintf(stderr, "Incorrect LOGREC_UNDO_KEY_DELETE data read(%d)" - "type %u, strid %u, len %lu != %lu + 7, hdr len: %u, " + "type %u, strid %u, len %lu != %lu + 14, hdr len: %u, " "ref1(%lu,0x%lx), ref2(%lu,0x%lx), " "lsn(%lu,0x%lx)\n", i, (uint) rec.type, (uint) rec.short_trid, @@ -457,7 +467,7 @@ int main(int argc, char *argv[]) (ulong) LSN_FILE_NO(rec.lsn), (ulong) LSN_OFFSET(rec.lsn)); goto err; } - if (read_and_check_content(&rec, long_buffer, 14)) + if (read_and_check_content(&rec, long_buffer, LSN_STORE_SIZE * 2)) { fprintf(stderr, "Incorrect LOGREC_UNDO_KEY_DELETE in whole rec read " @@ -467,7 +477,7 @@ int main(int argc, char *argv[]) } } - len= translog_read_next_record_header(lsn_ptr, &rec, 1, &scanner); + len= translog_read_next_record_header(&scanner, &rec); if (len == 0) { fprintf(stderr, "1-%d translog_read_next_record_header failed (%d)\n", @@ -483,7 +493,7 @@ int main(int argc, char *argv[]) if (rec.type !=LOGREC_LONG_TRANSACTION_ID || rec.short_trid != (i % 0xFFFF) || rec.record_length != 6 || uint4korr(rec.header) != i || - rec.header[4] != 0 || rec.header[5] != 0xFF) + ((uchar)rec.header[4]) != 0 || ((uchar)rec.header[5]) != 0xFF) { fprintf(stderr, "Incorrect LOGREC_LONG_TRANSACTION_ID data read(%d)\n" "type %u, strid %u, len %u, i: %u, 4: %u 5: %u " @@ -498,7 +508,7 @@ int main(int argc, char *argv[]) lsn= rec.lsn; - len= translog_read_next_record_header(lsn_ptr, &rec, 1, &scanner); + len= translog_read_next_record_header(&scanner, &rec); if ((rec_len= random() / (RAND_MAX / (LONG_BUFFER_SIZE + 1))) < 9) rec_len= 9; if (rec.type !=LOGREC_REDO_INSERT_ROW_HEAD || diff --git a/storage/maria/unittest/ma_test_loghandler_multigroup-t.c b/storage/maria/unittest/ma_test_loghandler_multigroup-t.c index 1d5ad9c81d8..3a2525a089e 100644 --- a/storage/maria/unittest/ma_test_loghandler_multigroup-t.c +++ b/storage/maria/unittest/ma_test_loghandler_multigroup-t.c @@ -42,10 +42,10 @@ static const char *default_dbug_option; 1 - Error */ -static my_bool check_content(uchar *ptr, ulong length) +static my_bool check_content(byte *ptr, ulong length) { ulong i; - uchar buff[4]; + byte buff[4]; DBUG_ENTER("check_content"); for (i= 0; i < length; i++) { @@ -79,12 +79,12 @@ static my_bool check_content(uchar *ptr, ulong length) */ static my_bool read_and_check_content(TRANSLOG_HEADER_BUFFER *rec, - uchar *buffer, uint skip) + byte *buffer, uint skip) { int res= 0; translog_size_t len; DBUG_ENTER("read_and_check_content"); - DBUG_ASSERT(rec->record_length < LONG_BUFFER_SIZE + 7 * 2 + 2); + DBUG_ASSERT(rec->record_length < LONG_BUFFER_SIZE + LSN_STORE_SIZE * 2 + 2); if ((len= translog_read_record(rec->lsn, 0, rec->record_length, buffer, NULL)) != rec->record_length) { @@ -113,16 +113,16 @@ int main(int argc, char *argv[]) uint32 i; uint32 rec_len; uint pagen; - uchar long_tr_id[6]; - uchar lsn_buff[23]= + byte long_tr_id[6]; + byte lsn_buff[23]= { 0x55, 0xAA, 0x55, 0xAA, 0x55, 0xAA, 0x55, 0xAA, 0x55, 0xAA, 0x55, 0xAA, 0x55, 0xAA, 0x55, 0xAA, 0x55, 0xAA, 0x55, 0xAA, 0x55, 0xAA, 0x55 }; - uchar *long_buffer= malloc(LONG_BUFFER_SIZE + 7 * 2 + 2); + byte *long_buffer= malloc(LONG_BUFFER_SIZE + LSN_STORE_SIZE * 2 + 2); PAGECACHE pagecache; - LSN lsn, lsn_base, first_lsn, lsn_ptr; + LSN lsn, lsn_base, first_lsn; TRANSLOG_HEADER_BUFFER rec; struct st_translog_scanner_data scanner; int rc; @@ -133,8 +133,8 @@ int main(int argc, char *argv[]) maria_data_root= "."; { - uchar buff[4]; - for (i= 0; i < (LONG_BUFFER_SIZE + 7 * 2 + 2); i++) + byte buff[4]; + for (i= 0; i < (LONG_BUFFER_SIZE + LSN_STORE_SIZE * 2 + 2); i++) { if (i % 4 == 0) int4store(buff, (i >> 2)); @@ -195,22 +195,24 @@ int main(int argc, char *argv[]) printf("write %d\n", i); if (i % 2) { - lsn7store(lsn_buff, lsn_base); + lsn_store(lsn_buff, lsn_base); if (translog_write_record(&lsn, LOGREC_CLR_END, - (i % 0xFFFF), NULL, 7, lsn_buff, 0)) + (i % 0xFFFF), NULL, + LSN_STORE_SIZE, lsn_buff, 0)) { fprintf(stderr, "1 Can't write reference before record #%lu\n", (ulong) i); translog_destroy(); exit(1); } - lsn7store(lsn_buff, lsn_base); + lsn_store(lsn_buff, lsn_base); rec_len= get_len(); if (translog_write_record(&lsn, LOGREC_UNDO_KEY_INSERT, (i % 0xFFFF), - NULL, 7, lsn_buff, rec_len, long_buffer, 0)) + NULL, LSN_STORE_SIZE, lsn_buff, + rec_len, long_buffer, 0)) { fprintf(stderr, "1 Can't write var reference before record #%lu\n", (ulong) i); @@ -220,8 +222,8 @@ int main(int argc, char *argv[]) } else { - lsn7store(lsn_buff, lsn_base); - lsn7store(lsn_buff + 7, first_lsn); + lsn_store(lsn_buff, lsn_base); + lsn_store(lsn_buff + LSN_STORE_SIZE, first_lsn); if (translog_write_record(&lsn, LOGREC_UNDO_ROW_DELETE, (i % 0xFFFF), NULL, 23, lsn_buff, 0)) @@ -231,13 +233,14 @@ int main(int argc, char *argv[]) translog_destroy(); exit(1); } - lsn7store(lsn_buff, lsn_base); - lsn7store(lsn_buff + 7, first_lsn); + lsn_store(lsn_buff, lsn_base); + lsn_store(lsn_buff + LSN_STORE_SIZE, first_lsn); rec_len= get_len(); if (translog_write_record(&lsn, LOGREC_UNDO_KEY_DELETE, (i % 0xFFFF), - NULL, 14, lsn_buff, rec_len, long_buffer, 0)) + NULL, LSN_STORE_SIZE * 2, lsn_buff, + rec_len, long_buffer, 0)) { fprintf(stderr, "0 Can't write var reference before record #%lu\n", (ulong) i); @@ -304,7 +307,7 @@ int main(int argc, char *argv[]) } if (rec.type !=LOGREC_LONG_TRANSACTION_ID || rec.short_trid != 0 || rec.record_length != 6 || uint4korr(rec.header) != 0 || - (uint)rec.header[4] != 0 || rec.header[5] != 0xFF || + ((uchar)rec.header[4]) != 0 || ((uchar)rec.header[5]) != 0xFF || first_lsn != rec.lsn) { fprintf(stderr, "Incorrect LOGREC_LONG_TRANSACTION_ID data read(0)\n" @@ -319,12 +322,16 @@ int main(int argc, char *argv[]) } translog_free_record_header(&rec); lsn= first_lsn; - lsn_ptr= first_lsn; + if (translog_init_scanner(first_lsn, 1, &scanner)) + { + fprintf(stderr, "scanner init failed\n"); + goto err; + } for (i= 1;; i++) { if (i % SHOW_DIVIDER == 0) printf("read %d\n", i); - len= translog_read_next_record_header(lsn_ptr, &rec, 1, &scanner); + len= translog_read_next_record_header(&scanner, &rec); if (len == 0) { fprintf(stderr, "1-%d translog_read_next_record_header failed (%d)\n", @@ -343,15 +350,13 @@ int main(int argc, char *argv[]) } break; } - /* use scanner after its initialization */ - lsn_ptr= 0; if (i % 2) { LSN ref; - ref= lsn7korr(rec.header); + ref= lsn_korr(rec.header); if (rec.type != LOGREC_CLR_END || rec.short_trid != (i % 0xFFFF) || - rec.record_length != 7 || ref != lsn) + rec.record_length != LSN_STORE_SIZE || ref != lsn) { fprintf(stderr, "Incorrect LOGREC_CLR_END data read(%d)" "type %u, strid %u, len %u, ref(%lu,0x%lx), lsn(%lu,0x%lx)\n", @@ -366,18 +371,22 @@ int main(int argc, char *argv[]) else { LSN ref1, ref2; - ref1= lsn7korr(rec.header); - ref2= lsn7korr(rec.header + 7); + ref1= lsn_korr(rec.header); + ref2= lsn_korr(rec.header + LSN_STORE_SIZE); if (rec.type !=LOGREC_UNDO_ROW_DELETE || rec.short_trid != (i % 0xFFFF) || rec.record_length != 23 || ref1 != lsn || ref2 != first_lsn || - rec.header[22] != 0x55 || rec.header[21] != 0xAA || - rec.header[20] != 0x55 || rec.header[19] != 0xAA || - rec.header[18] != 0x55 || rec.header[17] != 0xAA || - rec.header[16] != 0x55 || rec.header[15] != 0xAA || - rec.header[14] != 0x55) + ((uchar)rec.header[22]) != 0x55 || + ((uchar)rec.header[21]) != 0xAA || + ((uchar)rec.header[20]) != 0x55 || + ((uchar)rec.header[19]) != 0xAA || + ((uchar)rec.header[18]) != 0x55 || + ((uchar)rec.header[17]) != 0xAA || + ((uchar)rec.header[16]) != 0x55 || + ((uchar)rec.header[15]) != 0xAA || + ((uchar)rec.header[14]) != 0x55) { fprintf(stderr, "Incorrect LOGREC_UNDO_ROW_DELETE data read(%d)" "type %u, strid %u, len %u, ref1(%lu,0x%lx), " @@ -399,7 +408,7 @@ int main(int argc, char *argv[]) } translog_free_record_header(&rec); - len= translog_read_next_record_header(lsn_ptr, &rec, 1, &scanner); + len= translog_read_next_record_header(&scanner, &rec); if (len == 0) { fprintf(stderr, "1-%d translog_read_next_record_header (var) " @@ -415,13 +424,13 @@ int main(int argc, char *argv[]) if (i % 2) { LSN ref; - ref= lsn7korr(rec.header); + ref= lsn_korr(rec.header); rec_len= get_len(); if (rec.type !=LOGREC_UNDO_KEY_INSERT || rec.short_trid != (i % 0xFFFF) || - rec.record_length != rec_len + 7 || + rec.record_length != rec_len + LSN_STORE_SIZE || len != 12 || ref != lsn || - check_content(rec.header + 7, len - 7)) + check_content(rec.header + LSN_STORE_SIZE, len - LSN_STORE_SIZE)) { fprintf(stderr, "Incorrect LOGREC_UNDO_KEY_INSERT data read(%d)" "type %u (%d), strid %u (%d), len %lu, %lu + 7 (%d), " @@ -432,17 +441,18 @@ int main(int argc, char *argv[]) (uint) rec.short_trid, rec.short_trid != (i % 0xFFFF), (ulong) rec.record_length, (ulong) rec_len, - rec.record_length != rec_len + 7, + rec.record_length != rec_len + LSN_STORE_SIZE, (uint) len, len != 12, (ulong) LSN_FILE_NO(ref), (ulong) LSN_OFFSET(ref), (ulong) LSN_FILE_NO(rec.lsn), (ulong) LSN_OFFSET(rec.lsn), (ref != lsn), - check_content(rec.header + 7, len - 7)); + check_content(rec.header + LSN_STORE_SIZE, + len - LSN_STORE_SIZE)); translog_free_record_header(&rec); goto err; } - if (read_and_check_content(&rec, long_buffer, 7)) + if (read_and_check_content(&rec, long_buffer, LSN_STORE_SIZE)) { fprintf(stderr, "Incorrect LOGREC_UNDO_KEY_INSERT in whole rec read " @@ -455,19 +465,20 @@ int main(int argc, char *argv[]) else { LSN ref1, ref2; - ref1= lsn7korr(rec.header); - ref2= lsn7korr(rec.header + 7); + ref1= lsn_korr(rec.header); + ref2= lsn_korr(rec.header + LSN_STORE_SIZE); rec_len= get_len(); if (rec.type !=LOGREC_UNDO_KEY_DELETE || rec.short_trid != (i % 0xFFFF) || - rec.record_length != rec_len + 14 || + rec.record_length != rec_len + LSN_STORE_SIZE * 2 || len != 19 || ref1 != lsn || ref2 != first_lsn || - check_content(rec.header + 14, len - 14)) + check_content(rec.header + LSN_STORE_SIZE * 2, + len - LSN_STORE_SIZE * 2)) { fprintf(stderr, "Incorrect LOGREC_UNDO_KEY_DELETE data read(%d)" - "type %u, strid %u, len %lu != %lu + 7, hdr len: %u, " + "type %u, strid %u, len %lu != %lu + 14, hdr len: %u, " "ref1(%lu,0x%lx), ref2(%lu,0x%lx), " "lsn(%lu,0x%lx)\n", i, (uint) rec.type, (uint) rec.short_trid, @@ -479,7 +490,7 @@ int main(int argc, char *argv[]) translog_free_record_header(&rec); goto err; } - if (read_and_check_content(&rec, long_buffer, 14)) + if (read_and_check_content(&rec, long_buffer, LSN_STORE_SIZE * 2)) { fprintf(stderr, "Incorrect LOGREC_UNDO_KEY_DELETE in whole rec read " @@ -491,7 +502,7 @@ int main(int argc, char *argv[]) } translog_free_record_header(&rec); - len= translog_read_next_record_header(lsn_ptr, &rec, 1, &scanner); + len= translog_read_next_record_header(&scanner, &rec); if (len == 0) { fprintf(stderr, "1-%d translog_read_next_record_header failed (%d)\n", @@ -509,7 +520,7 @@ int main(int argc, char *argv[]) if (rec.type !=LOGREC_LONG_TRANSACTION_ID || rec.short_trid != (i % 0xFFFF) || rec.record_length != 6 || uint4korr(rec.header) != i || - rec.header[4] != 0 || rec.header[5] != 0xFF) + ((uchar)rec.header[4]) != 0 || ((uchar)rec.header[5]) != 0xFF) { fprintf(stderr, "Incorrect LOGREC_LONG_TRANSACTION_ID data read(%d)\n" "type %u, strid %u, len %u, i: %u, 4: %u 5: %u " @@ -526,7 +537,7 @@ int main(int argc, char *argv[]) lsn= rec.lsn; - len= translog_read_next_record_header(lsn_ptr, &rec, 1, &scanner); + len= translog_read_next_record_header(&scanner, &rec); rec_len= get_len(); if (rec.type !=LOGREC_REDO_INSERT_ROW_HEAD || rec.short_trid != (i % 0xFFFF) || diff --git a/storage/maria/unittest/ma_test_loghandler_multithread-t.c b/storage/maria/unittest/ma_test_loghandler_multithread-t.c index ed5479026ef..eb64d15868c 100644 --- a/storage/maria/unittest/ma_test_loghandler_multithread-t.c +++ b/storage/maria/unittest/ma_test_loghandler_multithread-t.c @@ -27,7 +27,7 @@ static uint thread_count; static ulong lens[WRITERS][ITERATIONS]; static LSN lsns1[WRITERS][ITERATIONS]; static LSN lsns2[WRITERS][ITERATIONS]; -static uchar *long_buffer; +static byte *long_buffer; /* Get pseudo-random length of the field in @@ -65,12 +65,12 @@ static uint32 get_len() 1 - Error */ -static my_bool check_content(uchar *ptr, ulong length) +static my_bool check_content(byte *ptr, ulong length) { ulong i; for (i= 0; i < length; i++) { - if (ptr[i] != (i & 0xFF)) + if (((uchar)ptr[i]) != (i & 0xFF)) { fprintf(stderr, "Byte # %lu is %x instead of %x", i, (uint) ptr[i], (uint) (i & 0xFF)); @@ -97,7 +97,7 @@ static my_bool check_content(uchar *ptr, ulong length) static my_bool read_and_check_content(TRANSLOG_HEADER_BUFFER *rec, - uchar *buffer, uint skip) + byte *buffer, uint skip) { int res= 0; translog_size_t len; @@ -117,7 +117,7 @@ static my_bool read_and_check_content(TRANSLOG_HEADER_BUFFER *rec, void writer(int num) { LSN lsn; - uchar long_tr_id[6]; + byte long_tr_id[6]; uint i; DBUG_ENTER("writer"); @@ -193,7 +193,7 @@ int main(int argc, char **argv __attribute__ ((unused))) uint32 i; uint pagen; PAGECACHE pagecache; - LSN first_lsn, lsn_ptr; + LSN first_lsn; TRANSLOG_HEADER_BUFFER rec; struct st_translog_scanner_data scanner; pthread_t tid; @@ -290,7 +290,7 @@ int main(int argc, char **argv __attribute__ ((unused))) srandom(122334817L); { - uchar long_tr_id[6]= + byte long_tr_id[6]= { 0x11, 0x22, 0x33, 0x44, 0x55, 0x66 }; @@ -369,11 +369,14 @@ int main(int argc, char **argv __attribute__ ((unused))) bzero(indeces, sizeof(indeces)); - lsn_ptr= first_lsn; + if (translog_init_scanner(first_lsn, 1, &scanner)) + { + fprintf(stderr, "scanner init failed\n"); + goto err; + } for (i= 0;; i++) { - len= translog_read_next_record_header(lsn_ptr, &rec, 1, &scanner); - lsn_ptr= 0; + len= translog_read_next_record_header(&scanner, &rec); if (len == 0) { diff --git a/storage/maria/unittest/ma_test_loghandler_pagecache-t.c b/storage/maria/unittest/ma_test_loghandler_pagecache-t.c index f215805d829..fc9b34ff980 100644 --- a/storage/maria/unittest/ma_test_loghandler_pagecache-t.c +++ b/storage/maria/unittest/ma_test_loghandler_pagecache-t.c @@ -107,7 +107,7 @@ int main(int argc, char *argv[]) bzero(page, PCACHE_PAGE); #define PAGE_LSN_OFFSET 0 - lsn7store(page + PAGE_LSN_OFFSET, lsn); + lsn_store(page + PAGE_LSN_OFFSET, lsn); pagecache_write(&pagecache, &file1, 0, 3, (char*)page, PAGECACHE_LSN_PAGE, PAGECACHE_LOCK_LEFT_UNLOCKED, diff --git a/support-files/magic b/support-files/magic index c8ca5875211..e2bd4d70fb5 100644 --- a/support-files/magic +++ b/support-files/magic @@ -17,3 +17,10 @@ 0 belong&0xffffff00 0xfefe0600 MySQL ISAM compressed data file >3 byte x Version %d 0 string \376bin MySQL replication log +0 belong&0xffffff00 0xfefe0b00 +>4 string MARIALOG MySQL Maria transaction log file +>>3 byte x Version %d +0 belong&0xffffff00 0xfefe0c00 +>4 string MACF MySQL Maria control file +>>3 byte x Version %d + -- cgit v1.2.1 From 3b0e794aca0239a7b5a63c590b52efe2b4e0141d Mon Sep 17 00:00:00 2001 From: unknown Date: Wed, 21 Feb 2007 15:54:08 +0200 Subject: unittest fixed storage/maria/unittest/Makefile.am: Unittest cleunup added storage/maria/unittest/ma_control_file-t.c: Unittest fixed according new file format storage/maria/unittest/ma_test_loghandler-t.c: Unittest cleunup added Unittest progress report added storage/maria/unittest/ma_test_loghandler_multigroup-t.c: Unittest cleunup added Unittest progress report added storage/maria/unittest/ma_test_loghandler_multithread-t.c: Unittest cleunup added Unittest progress report added storage/maria/unittest/ma_test_loghandler_pagecache-t.c: Unittest cleunup added Unittest progress report added storage/maria/unittest/ma_maria_log_cleanup.c: New BitKeeper file ``storage/maria/unittest/ma_maria_log_cleanup.c'' --- storage/maria/unittest/Makefile.am | 5 ++ storage/maria/unittest/ma_control_file-t.c | 15 ++--- storage/maria/unittest/ma_maria_log_cleanup.c | 45 +++++++++++++++ storage/maria/unittest/ma_test_loghandler-t.c | 46 +++++++++++++--- .../unittest/ma_test_loghandler_multigroup-t.c | 38 ++++++++++--- .../unittest/ma_test_loghandler_multithread-t.c | 64 +++++++++------------- .../unittest/ma_test_loghandler_pagecache-t.c | 8 +++ 7 files changed, 163 insertions(+), 58 deletions(-) create mode 100644 storage/maria/unittest/ma_maria_log_cleanup.c diff --git a/storage/maria/unittest/Makefile.am b/storage/maria/unittest/Makefile.am index 74d9059ca7d..33f1bfed560 100644 --- a/storage/maria/unittest/Makefile.am +++ b/storage/maria/unittest/Makefile.am @@ -44,6 +44,11 @@ noinst_PROGRAMS = ma_control_file-t trnman-t lockman2-t \ ma_test_loghandler_multithread-t \ ma_test_loghandler_pagecache-t +ma_test_loghandler_t_SOURCES= ma_test_loghandler-t.c ma_maria_log_cleanup.c +ma_test_loghandler_multigroup_t_SOURCES= ma_test_loghandler_multigroup-t.c ma_maria_log_cleanup.c +ma_test_loghandler_multithread_t_SOURCES= ma_test_loghandler_multithread-t.c ma_maria_log_cleanup.c +ma_test_loghandler_pagecache_t_SOURCES= ma_test_loghandler_pagecache-t.c ma_maria_log_cleanup.c + mf_pagecache_single_src = mf_pagecache_single.c $(top_srcdir)/mysys/mf_pagecache.c test_file.c mf_pagecache_consist_src = mf_pagecache_consist.c $(top_srcdir)/mysys/mf_pagecache.c test_file.c mf_pagecache_common_cppflags = -DEXTRA_DEBUG -DPAGECACHE_DEBUG -DMAIN diff --git a/storage/maria/unittest/ma_control_file-t.c b/storage/maria/unittest/ma_control_file-t.c index 9f6a6c9cf56..1b37ee3f53c 100644 --- a/storage/maria/unittest/ma_control_file-t.c +++ b/storage/maria/unittest/ma_control_file-t.c @@ -83,6 +83,7 @@ static void get_options(int argc, char *argv[]); int main(int argc,char *argv[]) { MY_INIT(argv[0]); + maria_data_root= "."; plan(9); @@ -263,18 +264,18 @@ static int test_binary_content() future change/breakage. */ - char buffer[17]; + char buffer[20]; RET_ERR_UNLESS((fd= my_open(file_name, O_BINARY | O_RDWR, MYF(MY_WME))) >= 0); - RET_ERR_UNLESS(my_read(fd, buffer, 17, MYF(MY_FNABP | MY_WME)) == 0); + RET_ERR_UNLESS(my_read(fd, buffer, 20, MYF(MY_FNABP | MY_WME)) == 0); RET_ERR_UNLESS(my_close(fd, MYF(MY_WME)) == 0); RET_ERR_UNLESS(create_or_open_file() == CONTROL_FILE_OK); - i= uint4korr(buffer+5); + i= uint3korr(buffer+9); RET_ERR_UNLESS(i == LSN_FILE_NO(last_checkpoint_lsn)); - i= uint4korr(buffer+9); + i= uint4korr(buffer+12); RET_ERR_UNLESS(i == LSN_OFFSET(last_checkpoint_lsn)); - i= uint4korr(buffer+13); + i= uint4korr(buffer+16); RET_ERR_UNLESS(i == last_logno); RET_ERR_UNLESS(close_file() == 0); return 0; @@ -341,9 +342,9 @@ static int test_bad_checksum() RET_ERR_UNLESS((fd= my_open(file_name, O_BINARY | O_RDWR, MYF(MY_WME))) >= 0); - RET_ERR_UNLESS(my_pread(fd, buffer, 1, 4, MYF(MY_FNABP | MY_WME)) == 0); + RET_ERR_UNLESS(my_pread(fd, buffer, 1, 8, MYF(MY_FNABP | MY_WME)) == 0); buffer[0]+= 3; /* mangle checksum */ - RET_ERR_UNLESS(my_pwrite(fd, buffer, 1, 4, MYF(MY_FNABP | MY_WME)) == 0); + RET_ERR_UNLESS(my_pwrite(fd, buffer, 1, 8, MYF(MY_FNABP | MY_WME)) == 0); /* Check that control file module sees the problem */ RET_ERR_UNLESS(ma_control_file_create_or_open() == CONTROL_FILE_BAD_CHECKSUM); diff --git a/storage/maria/unittest/ma_maria_log_cleanup.c b/storage/maria/unittest/ma_maria_log_cleanup.c new file mode 100644 index 00000000000..c5917764b9b --- /dev/null +++ b/storage/maria/unittest/ma_maria_log_cleanup.c @@ -0,0 +1,45 @@ +#include "../maria_def.h" +#include + +my_bool maria_log_remove() +{ + MY_DIR *dirp; + uint i; + MY_STAT stat_buff; + char file_name[FN_REFLEN]; + + /* Removes control file */ + if (fn_format(file_name, CONTROL_FILE_BASE_NAME, + maria_data_root, "", MYF(MY_WME)) == NullS) + return 1; + if (my_stat(file_name, &stat_buff, MYF(0)) && + my_delete(file_name, MYF(MY_WME)) != 0) + return 1; + + /* Finds and removes transaction log files */ + if (!(dirp = my_dir(maria_data_root, MYF(MY_DONT_SORT)))) + return 1; + + for (i= 0; i < dirp->number_off_files; i++) + { + char *file= dirp->dir_entry[i].name; + if (strncmp(file, "maria_log.", 10) == 0 && + file[10] >= '0' && file[10] <= '9' && + file[11] >= '0' && file[11] <= '9' && + file[12] >= '0' && file[12] <= '9' && + file[13] >= '0' && file[13] <= '9' && + file[14] >= '0' && file[14] <= '9' && + file[15] >= '0' && file[15] <= '9' && + file[16] >= '0' && file[16] <= '9' && + file[17] >= '0' && file[17] <= '9' && + file[18] == '\0') + { + if (fn_format(file_name, file, + maria_data_root, "", MYF(MY_WME)) == NullS || + my_delete(file_name, MYF(MY_WME)) != 0) + return 1; + } + } + return 0; +} + diff --git a/storage/maria/unittest/ma_test_loghandler-t.c b/storage/maria/unittest/ma_test_loghandler-t.c index 54ed4b0d746..757520322c8 100644 --- a/storage/maria/unittest/ma_test_loghandler-t.c +++ b/storage/maria/unittest/ma_test_loghandler-t.c @@ -3,6 +3,8 @@ #include #include +extern my_bool maria_log_remove(); + #ifndef DBUG_OFF static const char *default_dbug_option; #endif @@ -113,6 +115,8 @@ int main(int argc, char *argv[]) bzero(&pagecache, sizeof(pagecache)); maria_data_root= "."; + if (maria_log_remove()) + exit(1); for (i= 0; i < (LONG_BUFFER_SIZE + LSN_STORE_SIZE * 2 + 2); i+= 2) { @@ -152,6 +156,8 @@ int main(int argc, char *argv[]) exit(1); } + plan(((ITERATIONS - 1) * 4 + 1)*2 + ITERATIONS - 1); + srandom(122334817L); long_tr_id[5]= 0xff; @@ -163,14 +169,14 @@ int main(int argc, char *argv[]) { fprintf(stderr, "Can't write record #%lu\n", (ulong) 0); translog_destroy(); + ok(0, "write LOGREC_LONG_TRANSACTION_ID"); exit(1); } + ok(1, "write LOGREC_LONG_TRANSACTION_ID"); lsn_base= first_lsn= lsn; for (i= 1; i < ITERATIONS; i++) { - if (i % 1000 == 0) - printf("write %d\n", i); if (i % 2) { lsn_store(lsn_buff, lsn_base); @@ -181,8 +187,10 @@ int main(int argc, char *argv[]) fprintf(stderr, "1 Can't write reference defore record #%lu\n", (ulong) i); translog_destroy(); + ok(0, "write LOGREC_CLR_END"); exit(1); } + ok(1, "write LOGREC_CLR_END"); lsn_store(lsn_buff, lsn_base); if ((rec_len= random() / (RAND_MAX / (LONG_BUFFER_SIZE + 1))) < 12) rec_len= 12; @@ -194,8 +202,10 @@ int main(int argc, char *argv[]) fprintf(stderr, "1 Can't write var reference defore record #%lu\n", (ulong) i); translog_destroy(); + ok(0, "write LOGREC_UNDO_KEY_INSERT"); exit(1); } + ok(1, "write LOGREC_UNDO_KEY_INSERT"); } else { @@ -208,8 +218,10 @@ int main(int argc, char *argv[]) fprintf(stderr, "0 Can't write reference defore record #%lu\n", (ulong) i); translog_destroy(); + ok(0, "write LOGREC_UNDO_ROW_DELETE"); exit(1); } + ok(1, "write LOGREC_UNDO_ROW_DELETE"); lsn_store(lsn_buff, lsn_base); lsn_store(lsn_buff + LSN_STORE_SIZE, first_lsn); if ((rec_len= random() / (RAND_MAX / (LONG_BUFFER_SIZE + 1))) < 19) @@ -222,8 +234,10 @@ int main(int argc, char *argv[]) fprintf(stderr, "0 Can't write var reference defore record #%lu\n", (ulong) i); translog_destroy(); + ok(0, "write LOGREC_UNDO_KEY_DELETE"); exit(1); } + ok(1, "write LOGREC_UNDO_KEY_DELETE"); } int4store(long_tr_id, i); if (translog_write_record(&lsn, @@ -232,8 +246,10 @@ int main(int argc, char *argv[]) { fprintf(stderr, "Can't write record #%lu\n", (ulong) i); translog_destroy(); + ok(0, "write LOGREC_LONG_TRANSACTION_ID"); exit(1); } + ok(1, "write LOGREC_LONG_TRANSACTION_ID"); lsn_base= lsn; @@ -245,14 +261,18 @@ int main(int argc, char *argv[]) { fprintf(stderr, "Can't write variable record #%lu\n", (ulong) i); translog_destroy(); + ok(0, "write LOGREC_REDO_INSERT_ROW_HEAD"); exit(1); } + ok(1, "write LOGREC_REDO_INSERT_ROW_HEAD"); if (translog_flush(lsn)) { fprintf(stderr, "Can't flush #%lu\n", (ulong) i); translog_destroy(); + ok(0, "flush"); exit(1); } + ok(1, "flush"); } translog_destroy(); @@ -303,6 +323,8 @@ int main(int argc, char *argv[]) (ulong) LSN_FILE_NO(rec.lsn), (ulong) LSN_OFFSET(rec.lsn)); goto err; } + ok(1, "read record"); + translog_free_record_header(&rec); lsn= first_lsn; if (translog_init_scanner(first_lsn, 1, &scanner)) { @@ -311,8 +333,6 @@ int main(int argc, char *argv[]) } for (i= 1;; i++) { - if (i % 1000 == 0) - printf("read %d\n", i); len= translog_read_next_record_header(&scanner, &rec); if (len == 0) { @@ -386,6 +406,9 @@ int main(int argc, char *argv[]) goto err; } } + ok(1, "read record"); + translog_free_record_header(&rec); + len= translog_read_next_record_header(&scanner, &rec); if (len == 0) { @@ -476,6 +499,8 @@ int main(int argc, char *argv[]) goto err; } } + ok(1, "read record"); + translog_free_record_header(&rec); len= translog_read_next_record_header(&scanner, &rec); if (len == 0) @@ -505,8 +530,9 @@ int main(int argc, char *argv[]) (ulong) LSN_FILE_NO(rec.lsn), (ulong) LSN_OFFSET(rec.lsn)); goto err; } - lsn= rec.lsn; + ok(1, "read record"); + translog_free_record_header(&rec); len= translog_read_next_record_header(&scanner, &rec); if ((rec_len= random() / (RAND_MAX / (LONG_BUFFER_SIZE + 1))) < 9) @@ -533,14 +559,20 @@ int main(int argc, char *argv[]) (ulong) LSN_FILE_NO(rec.lsn), (ulong) LSN_OFFSET(rec.lsn)); goto err; } + ok(1, "read record"); + translog_free_record_header(&rec); } } - rc= 1; + rc= 0; err: + if (rc) + ok(0, "read record"); translog_destroy(); end_pagecache(&pagecache, 1); ma_control_file_end(); - return(test(exit_status() || rc)); + if (maria_log_remove()) + exit(1); + return(test(exit_status())); } diff --git a/storage/maria/unittest/ma_test_loghandler_multigroup-t.c b/storage/maria/unittest/ma_test_loghandler_multigroup-t.c index 3a2525a089e..4aaf30bd9a3 100644 --- a/storage/maria/unittest/ma_test_loghandler_multigroup-t.c +++ b/storage/maria/unittest/ma_test_loghandler_multigroup-t.c @@ -3,6 +3,8 @@ #include #include +extern my_bool maria_log_remove(); + #ifndef DBUG_OFF static const char *default_dbug_option; #endif @@ -13,8 +15,6 @@ static const char *default_dbug_option; #define MIN_REC_LENGTH (1024L*1024L + 1024L*512L + 1) -#define SHOW_DIVIDER 2 - #define LOG_FILE_SIZE (1024L*1024L*1024L + 1024L*1024L*512) #define ITERATIONS 2 /*#define ITERATIONS 63 */ @@ -131,6 +131,8 @@ int main(int argc, char *argv[]) bzero(&pagecache, sizeof(pagecache)); maria_data_root= "."; + if (maria_log_remove()) + exit(1); { byte buff[4]; @@ -174,6 +176,8 @@ int main(int argc, char *argv[]) exit(1); } + plan(((ITERATIONS - 1) * 4 + 1) * 2); + srandom(122334817L); long_tr_id[5]= 0xff; @@ -185,14 +189,14 @@ int main(int argc, char *argv[]) { fprintf(stderr, "Can't write record #%lu\n", (ulong) 0); translog_destroy(); + ok(0, "write LOGREC_LONG_TRANSACTION_ID"); exit(1); } + ok(1, "write LOGREC_LONG_TRANSACTION_ID"); lsn_base= first_lsn= lsn; for (i= 1; i < ITERATIONS; i++) { - if (i % SHOW_DIVIDER == 0) - printf("write %d\n", i); if (i % 2) { lsn_store(lsn_buff, lsn_base); @@ -204,8 +208,10 @@ int main(int argc, char *argv[]) fprintf(stderr, "1 Can't write reference before record #%lu\n", (ulong) i); translog_destroy(); + ok(0, "write LOGREC_CLR_END"); exit(1); } + ok(1, "write LOGREC_CLR_END"); lsn_store(lsn_buff, lsn_base); rec_len= get_len(); if (translog_write_record(&lsn, @@ -217,8 +223,10 @@ int main(int argc, char *argv[]) fprintf(stderr, "1 Can't write var reference before record #%lu\n", (ulong) i); translog_destroy(); + ok(0, "write LOGREC_UNDO_KEY_INSERT"); exit(1); } + ok(1, "write LOGREC_UNDO_KEY_INSERT"); } else { @@ -231,8 +239,10 @@ int main(int argc, char *argv[]) fprintf(stderr, "0 Can't write reference before record #%lu\n", (ulong) i); translog_destroy(); + ok(0, "write LOGREC_UNDO_ROW_DELETE"); exit(1); } + ok(1, "write LOGREC_UNDO_ROW_DELETE"); lsn_store(lsn_buff, lsn_base); lsn_store(lsn_buff + LSN_STORE_SIZE, first_lsn); rec_len= get_len(); @@ -245,8 +255,10 @@ int main(int argc, char *argv[]) fprintf(stderr, "0 Can't write var reference before record #%lu\n", (ulong) i); translog_destroy(); + ok(0, "write LOGREC_UNDO_KEY_DELETE"); exit(1); } + ok(1, "write LOGREC_UNDO_KEY_DELETE"); } int4store(long_tr_id, i); if (translog_write_record(&lsn, @@ -255,8 +267,10 @@ int main(int argc, char *argv[]) { fprintf(stderr, "Can't write record #%lu\n", (ulong) i); translog_destroy(); + ok(0, "write LOGREC_LONG_TRANSACTION_ID"); exit(1); } + ok(1, "write LOGREC_LONG_TRANSACTION_ID"); lsn_base= lsn; @@ -267,8 +281,10 @@ int main(int argc, char *argv[]) { fprintf(stderr, "Can't write variable record #%lu\n", (ulong) i); translog_destroy(); + ok(0, "write LOGREC_REDO_INSERT_ROW_HEAD"); exit(1); } + ok(1, "write LOGREC_REDO_INSERT_ROW_HEAD"); } translog_destroy(); @@ -320,6 +336,7 @@ int main(int argc, char *argv[]) translog_free_record_header(&rec); goto err; } + ok(1, "read record"); translog_free_record_header(&rec); lsn= first_lsn; if (translog_init_scanner(first_lsn, 1, &scanner)) @@ -329,8 +346,6 @@ int main(int argc, char *argv[]) } for (i= 1;; i++) { - if (i % SHOW_DIVIDER == 0) - printf("read %d\n", i); len= translog_read_next_record_header(&scanner, &rec); if (len == 0) { @@ -406,6 +421,7 @@ int main(int argc, char *argv[]) goto err; } } + ok(1, "read record"); translog_free_record_header(&rec); len= translog_read_next_record_header(&scanner, &rec); @@ -500,6 +516,7 @@ int main(int argc, char *argv[]) goto err; } } + ok(1, "read record"); translog_free_record_header(&rec); len= translog_read_next_record_header(&scanner, &rec); @@ -533,6 +550,7 @@ int main(int argc, char *argv[]) translog_free_record_header(&rec); goto err; } + ok(1, "read record"); translog_free_record_header(&rec); lsn= rec.lsn; @@ -563,14 +581,20 @@ int main(int argc, char *argv[]) translog_free_record_header(&rec); goto err; } + ok(1, "read record"); + translog_free_record_header(&rec); } } rc= 0; err: + if (rc) + ok(0, "read record"); translog_destroy(); end_pagecache(&pagecache, 1); ma_control_file_end(); + if (maria_log_remove()) + exit(1); - return (test(exit_status() || rc)); + return (test(exit_status())); } diff --git a/storage/maria/unittest/ma_test_loghandler_multithread-t.c b/storage/maria/unittest/ma_test_loghandler_multithread-t.c index eb64d15868c..78f526aa3cf 100644 --- a/storage/maria/unittest/ma_test_loghandler_multithread-t.c +++ b/storage/maria/unittest/ma_test_loghandler_multithread-t.c @@ -2,6 +2,7 @@ #include #include #include +extern my_bool maria_log_remove(); #ifndef DBUG_OFF static const char *default_dbug_option; @@ -33,7 +34,7 @@ static byte *long_buffer; Get pseudo-random length of the field in limits [MIN_REC_LENGTH..LONG_BUFFER_SIZE] - SYNOPSYS + SYNOPSIS get_len() RETURN @@ -101,8 +102,7 @@ static my_bool read_and_check_content(TRANSLOG_HEADER_BUFFER *rec, { int res= 0; translog_size_t len; - DBUG_ENTER("read_and_check_content"); - DBUG_ASSERT(rec->record_length < LONG_BUFFER_SIZE + 7 * 2 + 2); + if ((len= translog_read_record(rec->lsn, 0, rec->record_length, buffer, NULL)) != rec->record_length) { @@ -111,7 +111,7 @@ static my_bool read_and_check_content(TRANSLOG_HEADER_BUFFER *rec, res= 1; } res|= check_content(buffer + skip, rec->record_length - skip); - DBUG_RETURN(res); + return(res); } void writer(int num) @@ -119,7 +119,6 @@ void writer(int num) LSN lsn; byte long_tr_id[6]; uint i; - DBUG_ENTER("writer"); for (i= 0; i < ITERATIONS; i++) { @@ -135,6 +134,9 @@ void writer(int num) fprintf(stderr, "Can't write LOGREC_LONG_TRANSACTION_ID record #%lu " "thread %i\n", (ulong) i, num); translog_destroy(); + pthread_mutex_lock(&LOCK_thread_count); + ok(0, "write records"); + pthread_mutex_unlock(&LOCK_thread_count); return; } lsns1[num][i]= lsn; @@ -144,25 +146,17 @@ void writer(int num) { fprintf(stderr, "Can't write variable record #%lu\n", (ulong) i); translog_destroy(); + pthread_mutex_lock(&LOCK_thread_count); + ok(0, "write records"); + pthread_mutex_unlock(&LOCK_thread_count); return; } lsns2[num][i]= lsn; - DBUG_PRINT("info", ("thread: %u, iteration: %u, len: %lu, " - "lsn1 (%lu,0x%lx) lsn2 (%lu,0x%lx)", - num, i, (ulong) lens[num][i], - (ulong) LSN_FILE_NO(lsns1[num][i]), - (ulong) LSN_OFFSET(lsns1[num][i]), - (ulong) LSN_FILE_NO(lsns2[num][i]), - (ulong) LSN_OFFSET(lsns2[num][i]))); - printf("thread: %u, iteration: %u, len: %lu, " - "lsn1 (%lu,0x%lx) lsn2 (%lu,0x%lx)\n", - num, i, (ulong) lens[num][i], - (ulong) LSN_FILE_NO(lsns1[num][i]), - (ulong) LSN_OFFSET(lsns1[num][i]), - (ulong) LSN_FILE_NO(lsns2[num][i]), - (ulong) LSN_OFFSET(lsns2[num][i])); + pthread_mutex_lock(&LOCK_thread_count); + ok(1, "write records"); + pthread_mutex_unlock(&LOCK_thread_count); } - DBUG_VOID_RETURN; + return; } @@ -171,20 +165,18 @@ static void *test_thread_writer(void *arg) int param= *((int*) arg); my_thread_init(); - DBUG_ENTER("test_writer"); - DBUG_PRINT("enter", ("param: %d", param)); writer(param); - DBUG_PRINT("info", ("Thread %s ended\n", my_thread_name())); pthread_mutex_lock(&LOCK_thread_count); thread_count--; + ok(1, "writer finished"); /* just to show progress */ VOID(pthread_cond_signal(&COND_thread_count)); /* Tell main we are ready */ pthread_mutex_unlock(&LOCK_thread_count); free((gptr) arg); my_thread_end(); - DBUG_RETURN(0); + return(0); } @@ -201,6 +193,8 @@ int main(int argc, char **argv __attribute__ ((unused))) int *param, error; int rc; + plan(WRITERS + ITERATIONS * WRITERS * 3); + bzero(&pagecache, sizeof(pagecache)); maria_data_root= "."; long_buffer= malloc(LONG_BUFFER_SIZE + 7 * 2 + 2); @@ -212,8 +206,10 @@ int main(int argc, char **argv __attribute__ ((unused))) for (i= 0; i < (LONG_BUFFER_SIZE + 7 * 2 + 2); i++) long_buffer[i]= (i & 0xFF); - MY_INIT(argv[0]); + if (maria_log_remove()) + exit(1); + #ifndef DBUG_OFF #if defined(__WIN__) @@ -228,8 +224,6 @@ int main(int argc, char **argv __attribute__ ((unused))) } #endif - DBUG_ENTER("main"); - DBUG_PRINT("info", ("Main thread: %s\n", my_thread_name())); if ((error= pthread_cond_init(&COND_thread_count, NULL))) { @@ -327,7 +321,6 @@ int main(int argc, char **argv __attribute__ ((unused))) thread_count++; number_of_writers--; } - DBUG_PRINT("info", ("All threads are started")); pthread_mutex_unlock(&LOCK_thread_count); pthread_attr_destroy(&thr_attr); @@ -342,7 +335,6 @@ int main(int argc, char **argv __attribute__ ((unused))) } if ((error= pthread_mutex_unlock(&LOCK_thread_count))) fprintf(stderr, "LOCK_thread_count: %d from pthread_mutex_unlock\n", error); - DBUG_PRINT("info", ("All threads ended")); /* Find last LSN and flush up to it (all our log) */ { @@ -352,11 +344,6 @@ int main(int argc, char **argv __attribute__ ((unused))) if (cmp_translog_addr(lsns2[i][ITERATIONS - 1], max) > 0) max= lsns2[i][ITERATIONS - 1]; } - DBUG_PRINT("info", ("first lsn: (%lu,0x%lx), max lsn: (%lu,0x%lx)", - (ulong) LSN_FILE_NO(first_lsn), - (ulong) LSN_OFFSET(first_lsn), - (ulong) LSN_FILE_NO(max), - (ulong) LSN_OFFSET(max))); translog_flush(max); } @@ -398,8 +385,6 @@ int main(int argc, char **argv __attribute__ ((unused))) } index= indeces[rec.short_trid] / 2; stage= indeces[rec.short_trid] % 2; - printf("read(%d) thread: %d, iteration %d, stage %d\n", - i, (uint) rec.short_trid, index, stage); if (stage == 0) { if (rec.type !=LOGREC_LONG_TRANSACTION_ID || @@ -459,6 +444,7 @@ int main(int argc, char **argv __attribute__ ((unused))) goto err; } } + ok(1, "record read"); translog_free_record_header(&rec); indeces[rec.short_trid]++; } @@ -466,9 +452,13 @@ int main(int argc, char **argv __attribute__ ((unused))) rc= 0; err: + if (rc) + ok(0, "record read"); translog_destroy(); end_pagecache(&pagecache, 1); ma_control_file_end(); + if (maria_log_remove()) + exit(1); - DBUG_RETURN(test(exit_status() || rc)); + return(exit_status()); } diff --git a/storage/maria/unittest/ma_test_loghandler_pagecache-t.c b/storage/maria/unittest/ma_test_loghandler_pagecache-t.c index fc9b34ff980..211eb6dea9e 100644 --- a/storage/maria/unittest/ma_test_loghandler_pagecache-t.c +++ b/storage/maria/unittest/ma_test_loghandler_pagecache-t.c @@ -3,6 +3,8 @@ #include #include +extern my_bool maria_log_remove(); + #ifndef DBUG_OFF static const char *default_dbug_option; #endif @@ -26,8 +28,12 @@ int main(int argc, char *argv[]) MY_INIT(argv[0]); + plan(1); + bzero(&pagecache, sizeof(pagecache)); maria_data_root= "."; + if (maria_log_remove()) + exit(1); /* be sure that we have no logs in the directory*/ if (my_stat(CONTROL_FILE_BASE_NAME, &st, MYF(0))) my_delete(CONTROL_FILE_BASE_NAME, MYF(0)); @@ -127,8 +133,10 @@ int main(int argc, char *argv[]) "incorrect initial size of %s: %ld instead of %ld\n", first_translog_file, (long)st.st_size, (long)(TRANSLOG_PAGE_SIZE * 2)); + ok(0, "log triggered"); exit(1); } + ok(1, "log triggered"); translog_destroy(); end_pagecache(&pagecache, 1); -- cgit v1.2.1 From 3f4bd8e8a7f905dc8574a83cbbd7829c34d7a811 Mon Sep 17 00:00:00 2001 From: unknown Date: Wed, 21 Feb 2007 16:03:04 +0200 Subject: Temporary disable unstable test --- mysql-test/t/disabled.def | 1 + 1 file changed, 1 insertion(+) diff --git a/mysql-test/t/disabled.def b/mysql-test/t/disabled.def index 3b063841512..c7748318e51 100644 --- a/mysql-test/t/disabled.def +++ b/mysql-test/t/disabled.def @@ -40,3 +40,4 @@ flush2 : Bug#24805 Pushbuild can't handle test with --disable- maria : Until maria is fully functional ps_maria : Until maria is fully functional +rpl_ndb_dd_advance : unstable test disable till next merge -- cgit v1.2.1 From bce8576412a6db26304dd2d069fa24c1c5293e9f Mon Sep 17 00:00:00 2001 From: unknown Date: Wed, 21 Feb 2007 23:30:58 +0200 Subject: More postreview changes storage/maria/ma_loghandler.c: Cursor checks made as one function. --- storage/maria/ma_loghandler.c | 93 +++++++++++++++---------------------------- 1 file changed, 32 insertions(+), 61 deletions(-) diff --git a/storage/maria/ma_loghandler.c b/storage/maria/ma_loghandler.c index dd12c73770b..07cf48acd60 100644 --- a/storage/maria/ma_loghandler.c +++ b/storage/maria/ma_loghandler.c @@ -359,6 +359,27 @@ typedef struct st_translog_validator_data const char *maria_data_root; +/* + Check cursor/buffer consistence + + SYNOPSIS + translog_check_cursor + cursor cursor which will be checked +*/ + +#ifndef DBUG_OFF +static void translog_check_cursor(struct st_buffer_cursor *cursor) +{ + DBUG_ASSERT(cursor->chaser || + ((ulong) (cursor->ptr - cursor->buffer->buffer) == + cursor->buffer->size)); + DBUG_ASSERT(cursor->buffer->buffer_no == cursor->buffer_no); + DBUG_ASSERT((cursor->ptr -cursor->buffer->buffer) %TRANSLOG_PAGE_SIZE == + cursor->current_page_fill % TRANSLOG_PAGE_SIZE); + DBUG_ASSERT(cursor->current_page_fill <= TRANSLOG_PAGE_SIZE); +} +#endif + /* Get file name of the log by log number @@ -696,11 +717,7 @@ static void translog_new_page_header(TRANSLOG_ADDRESS *horizon, (uint) cursor->buffer->buffer_no, (ulong) cursor->buffer, cursor->chaser, (ulong) cursor->buffer->size, (ulong) (cursor->ptr - cursor->buffer->buffer))); - DBUG_ASSERT(cursor->chaser || - ((ulong) (cursor->ptr - cursor->buffer->buffer) == - cursor->buffer->size)); - DBUG_ASSERT(cursor->buffer->buffer_no == cursor->buffer_no); - DBUG_ASSERT(cursor->current_page_fill <= TRANSLOG_PAGE_SIZE); + DBUG_EXECUTE("info", translog_check_cursor(cursor);); DBUG_VOID_RETURN; } @@ -814,12 +831,8 @@ static void translog_finish_page(TRANSLOG_ADDRESS *horizon, (ulong) cursor->buffer->size, (ulong) (cursor->ptr -cursor->buffer->buffer), (uint) cursor->current_page_fill, (uint) left)); - DBUG_ASSERT(cursor->ptr !=NULL); - DBUG_ASSERT((cursor->ptr -cursor->buffer->buffer) %TRANSLOG_PAGE_SIZE == - cursor->current_page_fill % TRANSLOG_PAGE_SIZE); DBUG_ASSERT(LSN_FILE_NO(*horizon) == LSN_FILE_NO(cursor->buffer->offset)); - DBUG_ASSERT(LSN_OFFSET(cursor->buffer->offset) + - (cursor->ptr -cursor->buffer->buffer) == LSN_OFFSET(*horizon)); + DBUG_EXECUTE("info", translog_check_cursor(cursor);); if (cursor->protected) { DBUG_PRINT("info", ("Already protected and finished")); @@ -843,10 +856,7 @@ static void translog_finish_page(TRANSLOG_ADDRESS *horizon, (ulong) cursor->buffer, cursor->chaser, (ulong) cursor->buffer->size, (ulong) (cursor->ptr - cursor->buffer->buffer))); - DBUG_ASSERT(cursor->chaser - || ((ulong) (cursor->ptr - cursor->buffer->buffer) == - cursor->buffer->size)); - DBUG_ASSERT(cursor->buffer->buffer_no == cursor->buffer_no); + DBUG_EXECUTE("info", translog_check_cursor(cursor);); } if (page[TRANSLOG_PAGE_FLAGS] & TRANSLOG_SECTOR_PROTECTION) { @@ -1011,10 +1021,7 @@ static void translog_start_buffer(struct st_translog_buffer *buffer, (uint) cursor->buffer->buffer_no, (ulong) cursor->buffer, cursor->chaser, (ulong) cursor->buffer->size, (ulong) (cursor->ptr - cursor->buffer->buffer))); - DBUG_ASSERT(cursor->chaser || - ((ulong) (cursor->ptr - cursor->buffer->buffer) == - cursor->buffer->size)); - DBUG_ASSERT(cursor->buffer->buffer_no == cursor->buffer_no); + DBUG_EXECUTE("info", translog_check_cursor(cursor);); DBUG_VOID_RETURN; } @@ -1890,7 +1897,7 @@ my_bool translog_init(const char *directory, highest XXXXXXXX & set logs_found TODO: check that last checkpoint within present log addresses space - find the log end + find the log end */ if (LSN_FILE_NO(last_checkpoint_lsn) == CONTROL_FILE_IMPOSSIBLE_FILENO) { @@ -2042,13 +2049,7 @@ my_bool translog_init(const char *directory, (ulong) log_descriptor.bc.buffer->size, (ulong) (log_descriptor.bc.ptr - log_descriptor.bc. buffer->buffer))); - DBUG_ASSERT(log_descriptor.bc.chaser || - ((ulong) - (log_descriptor.bc.ptr -log_descriptor.bc.buffer->buffer) == - log_descriptor.bc.buffer->size)); - DBUG_ASSERT(log_descriptor.bc.buffer->buffer_no == - log_descriptor.bc.buffer_no); - DBUG_ASSERT(log_descriptor.bc.current_page_fill <= TRANSLOG_PAGE_SIZE); + DBUG_EXECUTE("info", translog_check_cursor(&log_descriptor.bc);); } } DBUG_PRINT("info", ("Logs found: %d was recovered: %d", @@ -2332,11 +2333,7 @@ static my_bool translog_write_data_on_page(TRANSLOG_ADDRESS *horizon, (uint) cursor->buffer->buffer_no, (ulong) cursor->buffer, cursor->chaser, (ulong) cursor->buffer->size, (ulong) (cursor->ptr - cursor->buffer->buffer))); - DBUG_ASSERT(cursor->chaser || - ((ulong) (cursor->ptr - cursor->buffer->buffer) == - cursor->buffer->size)); - DBUG_ASSERT(cursor->buffer->buffer_no == cursor->buffer_no); - DBUG_ASSERT(cursor->current_page_fill <= TRANSLOG_PAGE_SIZE); + DBUG_EXECUTE("info", translog_check_cursor(cursor);); DBUG_RETURN(0); } @@ -2430,17 +2427,7 @@ static my_bool translog_write_parts_on_page(TRANSLOG_ADDRESS *horizon, (ulong) LSN_OFFSET(*horizon), (ulong) (LSN_OFFSET(cursor->buffer->offset) + cursor->buffer->size))); - /* - TODO: make one check function for the buffer/loghandler - */ - - DBUG_ASSERT(cursor->chaser || - ((ulong) (cursor->ptr - cursor->buffer->buffer) == - cursor->buffer->size)); - DBUG_ASSERT(cursor->buffer->buffer_no == cursor->buffer_no); - DBUG_ASSERT((cursor->ptr -cursor->buffer->buffer) %TRANSLOG_PAGE_SIZE == - cursor->current_page_fill % TRANSLOG_PAGE_SIZE); - DBUG_ASSERT(cursor->current_page_fill <= TRANSLOG_PAGE_SIZE); + DBUG_EXECUTE("info", translog_check_cursor(cursor);); DBUG_RETURN(0); } @@ -2698,9 +2685,7 @@ static my_bool translog_advance_pointer(uint pages, uint16 last_page_data) translog_wait_for_buffer_free(new_buffer); min_offset= min(buffer_end_offset, file_end_offset); - /* - TODO: check is it ptr or size enough - */ + /* TODO: check is it ptr or size enough */ log_descriptor.bc.buffer->size+= min_offset; log_descriptor.bc.ptr+= min_offset; DBUG_PRINT("info", ("NewP buffer #%u: 0x%lx chaser: %d Size: %lu (%lu)", @@ -2761,16 +2746,7 @@ static my_bool translog_advance_pointer(uint pages, uint16 last_page_data) ("pointer moved to: (%lu, 0x%lx)", (ulong) LSN_FILE_NO(log_descriptor.horizon), (ulong) LSN_OFFSET(log_descriptor.horizon))); - DBUG_ASSERT(log_descriptor.bc.chaser || - ((ulong) (log_descriptor.bc.ptr - - log_descriptor.bc.buffer->buffer) - == log_descriptor.bc.buffer->size)); - DBUG_ASSERT(log_descriptor.bc.buffer->buffer_no == - log_descriptor.bc.buffer_no); - DBUG_ASSERT((log_descriptor.bc.ptr - log_descriptor.bc.buffer-> - buffer) %TRANSLOG_PAGE_SIZE == - log_descriptor.bc.current_page_fill % TRANSLOG_PAGE_SIZE); - DBUG_ASSERT(log_descriptor.bc.current_page_fill <= TRANSLOG_PAGE_SIZE); + DBUG_EXECUTE("info", translog_check_cursor(&log_descriptor.bc);); log_descriptor.bc.protected= 0; DBUG_RETURN(0); } @@ -5132,14 +5108,9 @@ static void translog_force_current_buffer_to_finish() new_buff_begunning+= log_descriptor.bc.buffer->size; /* increase offset */ DBUG_ASSERT(log_descriptor.bc.ptr !=NULL); - DBUG_ASSERT((log_descriptor.bc.ptr -log_descriptor.bc.buffer->buffer) - % TRANSLOG_PAGE_SIZE == - log_descriptor.bc.current_page_fill % TRANSLOG_PAGE_SIZE); DBUG_ASSERT(LSN_FILE_NO(log_descriptor.horizon) == LSN_FILE_NO(log_descriptor.bc.buffer->offset)); - DBUG_ASSERT(LSN_OFFSET(log_descriptor.bc.buffer->offset) + - (log_descriptor.bc.ptr -log_descriptor.bc.buffer->buffer) == - LSN_OFFSET(log_descriptor.horizon)); + DBUG_EXECUTE("info", translog_check_cursor(&log_descriptor.bc);); DBUG_ASSERT(left < TRANSLOG_PAGE_SIZE); if (left != 0) { -- cgit v1.2.1 From 2eb27365dfdd82b8db9c93db0df0f244b5ff8e80 Mon Sep 17 00:00:00 2001 From: unknown Date: Wed, 21 Feb 2007 23:54:07 +0200 Subject: fixed C++ comment --- storage/maria/ma_loghandler.c | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/storage/maria/ma_loghandler.c b/storage/maria/ma_loghandler.c index 07cf48acd60..1dc4e9171e3 100644 --- a/storage/maria/ma_loghandler.c +++ b/storage/maria/ma_loghandler.c @@ -2058,7 +2058,7 @@ my_bool translog_init(const char *directory, { /* Start new log system from scratch */ /* Used space */ - log_descriptor.horizon= MAKE_LSN(1, TRANSLOG_PAGE_SIZE); // header page + log_descriptor.horizon= MAKE_LSN(1, TRANSLOG_PAGE_SIZE); /* header page */ /* Current logs file number in page cache */ if ((log_descriptor.log_file_num[0]= open_logfile_by_number_no_cache(1)) == -1 || -- cgit v1.2.1 From a3f9083bb4649530b210423e21589803c075bb9e Mon Sep 17 00:00:00 2001 From: unknown Date: Thu, 22 Feb 2007 10:19:52 +0200 Subject: removed pthread_attr_setstacksize (not really nead but incompatible with some platforms) --- storage/maria/unittest/ma_test_loghandler_multithread-t.c | 9 +-------- 1 file changed, 1 insertion(+), 8 deletions(-) diff --git a/storage/maria/unittest/ma_test_loghandler_multithread-t.c b/storage/maria/unittest/ma_test_loghandler_multithread-t.c index 78f526aa3cf..1834e720328 100644 --- a/storage/maria/unittest/ma_test_loghandler_multithread-t.c +++ b/storage/maria/unittest/ma_test_loghandler_multithread-t.c @@ -250,14 +250,7 @@ int main(int argc, char **argv __attribute__ ((unused))) error, errno); exit(1); } -#ifndef pthread_attr_setstacksize /* void return value */ - if ((error= pthread_attr_setstacksize(&thr_attr, 65536L))) - { - fprintf(stderr, "Got error: %d from pthread_attr_setstacksize " - "(errno: %d)\n", error, errno); - exit(1); - } -#endif + #ifdef HAVE_THR_SETCONCURRENCY VOID(thr_setconcurrency(2)); #endif -- cgit v1.2.1 From fdf847fb62a0fcdf0edf25d6c8654b19eaa9a9ad Mon Sep 17 00:00:00 2001 From: unknown Date: Tue, 27 Feb 2007 10:59:01 +0200 Subject: comments fixed storage/maria/ma_loghandler.c: The function comment fixed. storage/maria/ma_loghandler_lsn.h: Comment of LSN/TRANSLOG_ADDRESS type fixed --- storage/maria/ma_loghandler.c | 3 +-- storage/maria/ma_loghandler_lsn.h | 7 ++++++- 2 files changed, 7 insertions(+), 3 deletions(-) diff --git a/storage/maria/ma_loghandler.c b/storage/maria/ma_loghandler.c index 1dc4e9171e3..1e9afc7571d 100644 --- a/storage/maria/ma_loghandler.c +++ b/storage/maria/ma_loghandler.c @@ -4679,8 +4679,6 @@ translog_read_record_header_from_buffer(byte *page, buff log record header buffer NOTE - - lsn can point to TRANSLOG_HEADER_BUFFER::lsn and it will be processed - correctly. - Some type of record can be read completely by this call - "Decoded" header stored in TRANSLOG_HEADER_BUFFER::header (relative LSN can be translated to absolute one), some fields can be added @@ -4786,6 +4784,7 @@ translog_read_record_header_scan(TRANSLOG_SCANNER_DATA its NOTES. - in case of end of the log buff->lsn will be set to (CONTROL_FILE_IMPOSSIBLE_LSN) + RETURN 0 error TRANSLOG_RECORD_HEADER_MAX_SIZE + 1 End of the log diff --git a/storage/maria/ma_loghandler_lsn.h b/storage/maria/ma_loghandler_lsn.h index 9625f109864..1789d3ce61b 100644 --- a/storage/maria/ma_loghandler_lsn.h +++ b/storage/maria/ma_loghandler_lsn.h @@ -1,7 +1,12 @@ #ifndef _ma_loghandler_lsn_h #define _ma_loghandler_lsn_h -/* Transaction log record address (file_no is int24 on the disk) */ +/* + Transaction log record address: + file_no << 32 | offset + file_no is only 3 bytes so we can use signed integer to make + comparison more simple. +*/ typedef int64 TRANSLOG_ADDRESS; /* -- cgit v1.2.1 From 3411bfe05a2a77c6c5b9911237792eb436f16543 Mon Sep 17 00:00:00 2001 From: unknown Date: Thu, 1 Mar 2007 18:23:58 +0100 Subject: merge from MyISAM into Maria (last step of merge of 5.1 into Maria). Tests: "maria" and "ps_maria" fail like before merge (assertions), "ma_test_all" fails like before merge (ma_test2 segfaults, I'll try to find out why). mysys/mf_pagecache.c: using a more distinctive tag storage/maria/ha_maria.cc: merge from MyISAM into Maria storage/maria/ma_check.c: merge from MyISAM into Maria storage/maria/ma_close.c: TODO as a word storage/maria/ma_create.c: merge from MyISAM into Maria storage/maria/ma_delete_all.c: TODO as a word storage/maria/ma_delete_table.c: TODO as a word storage/maria/ma_dynrec.c: merge from MyISAM into Maria storage/maria/ma_extra.c: merge from MyISAM into Maria storage/maria/ma_ft_boolean_search.c: merge from MyISAM into Maria storage/maria/ma_locking.c: merge from MyISAM into Maria storage/maria/ma_loghandler.c: fix for compiler warning storage/maria/ma_open.c: merge from MyISAM into Maria. I will ask Monty to check the ASKMONTY-marked piece of code. storage/maria/ma_packrec.c: merge from MyISAM into Maria storage/maria/ma_range.c: merge from MyISAM into Maria storage/maria/ma_rename.c: TODO as a word storage/maria/ma_rt_index.c: merge from MyISAM into Maria storage/maria/ma_rt_split.c: merge from MyISAM into Maria storage/maria/ma_search.c: merge from MyISAM into Maria storage/maria/ma_sort.c: merge from MyISAM into Maria storage/maria/ma_update.c: merge from MyISAM into Maria storage/maria/ma_write.c: merge from MyISAM into Maria storage/maria/maria_chk.c: merge from MyISAM into Maria storage/maria/maria_def.h: merge from MyISAM into Maria storage/maria/maria_pack.c: merge from MyISAM into Maria storage/maria/unittest/ma_test_loghandler_pagecache-t.c: fix for compiler warning storage/myisam/ha_myisam.cc: merge from MyISAM into Maria storage/myisammrg/ha_myisammrg.cc: merge from MyISAM into Maria --- mysys/mf_pagecache.c | 17 +- storage/maria/ha_maria.cc | 608 +++++++++++++-------- storage/maria/ma_check.c | 8 + storage/maria/ma_close.c | 2 +- storage/maria/ma_create.c | 15 +- storage/maria/ma_delete_all.c | 12 +- storage/maria/ma_delete_table.c | 4 +- storage/maria/ma_dynrec.c | 18 +- storage/maria/ma_extra.c | 7 +- storage/maria/ma_ft_boolean_search.c | 16 +- storage/maria/ma_locking.c | 9 + storage/maria/ma_loghandler.c | 2 +- storage/maria/ma_open.c | 57 +- storage/maria/ma_packrec.c | 343 +++++++++++- storage/maria/ma_range.c | 1 + storage/maria/ma_rename.c | 4 +- storage/maria/ma_rt_index.c | 3 - storage/maria/ma_rt_split.c | 4 + storage/maria/ma_search.c | 14 +- storage/maria/ma_sort.c | 6 +- storage/maria/ma_update.c | 3 +- storage/maria/ma_write.c | 4 +- storage/maria/maria_chk.c | 1 + storage/maria/maria_def.h | 1 + storage/maria/maria_pack.c | 9 +- .../unittest/ma_test_loghandler_pagecache-t.c | 2 +- storage/myisam/ha_myisam.cc | 12 +- storage/myisammrg/ha_myisammrg.cc | 18 +- 28 files changed, 877 insertions(+), 323 deletions(-) diff --git a/mysys/mf_pagecache.c b/mysys/mf_pagecache.c index 1b9d48c80e6..8fa651bdf36 100755 --- a/mysys/mf_pagecache.c +++ b/mysys/mf_pagecache.c @@ -3228,7 +3228,8 @@ restart: int rc= #endif /* - QQ: We are doing an unlock here, so need to give the page its rec_lsn + RECOVERY TODO BUG We are doing an unlock here, so need to give the + page its rec_lsn */ make_lock_and_pin(pagecache, block, write_lock_change_table[lock].unlock_lock, @@ -3590,13 +3591,13 @@ restart: else { /* Link the block into a list of blocks 'in switch' */ - /* QQ: - #warning this unlink_changed() is a serious problem for - Maria's Checkpoint: it removes a page from the list of dirty - pages, while it's still dirty. A solution is to abandon - first_in_switch, just wait for this page to be - flushed by somebody else, and loop. TODO: check all places - where we remove a page from the list of dirty pages + /* + RECOVERY TODO BUG this unlink_changed() is a serious problem for + Maria's Checkpoint: it removes a page from the list of dirty + pages, while it's still dirty. A solution is to abandon + first_in_switch, just wait for this page to be + flushed by somebody else, and loop. TODO: check all places + where we remove a page from the list of dirty pages */ unlink_changed(block); link_changed(block, &first_in_switch); diff --git a/storage/maria/ha_maria.cc b/storage/maria/ha_maria.cc index 102f987f01a..a5e0722fd4a 100644 --- a/storage/maria/ha_maria.cc +++ b/storage/maria/ha_maria.cc @@ -108,6 +108,315 @@ static void _ma_check_print_msg(HA_CHECK *param, const char *msg_type, return; } + + +/* + Convert TABLE object to Maria key and column definition + + SYNOPSIS + table2maria() + table_arg in TABLE object. + keydef_out out Maria key definition. + recinfo_out out Maria column definition. + records_out out Number of fields. + + DESCRIPTION + This function will allocate and initialize Maria key and column + definition for further use in ma_create or for a check for underlying + table conformance in merge engine. + + RETURN VALUE + 0 OK + !0 error code +*/ + +int table2maria(TABLE *table_arg, MARIA_KEYDEF **keydef_out, + MARIA_COLUMNDEF **recinfo_out, uint *records_out) +{ + uint i, j, recpos, minpos, fieldpos, temp_length, length; + enum ha_base_keytype type= HA_KEYTYPE_BINARY; + byte *record; + KEY *pos; + MARIA_KEYDEF *keydef; + MARIA_COLUMNDEF *recinfo, *recinfo_pos; + HA_KEYSEG *keyseg; + TABLE_SHARE *share= table_arg->s; + uint options= share->db_options_in_use; + DBUG_ENTER("table2maria"); + if (!(my_multi_malloc(MYF(MY_WME), + recinfo_out, (share->fields * 2 + 2) * sizeof(MARIA_COLUMNDEF), + keydef_out, share->keys * sizeof(MARIA_KEYDEF), + &keyseg, + (share->key_parts + share->keys) * sizeof(HA_KEYSEG), + NullS))) + DBUG_RETURN(HA_ERR_OUT_OF_MEM); /* purecov: inspected */ + keydef= *keydef_out; + recinfo= *recinfo_out; + pos= table_arg->key_info; + for (i= 0; i < share->keys; i++, pos++) + { + keydef[i].flag= (pos->flags & (HA_NOSAME | HA_FULLTEXT | HA_SPATIAL)); + keydef[i].key_alg= pos->algorithm == HA_KEY_ALG_UNDEF ? + (pos->flags & HA_SPATIAL ? HA_KEY_ALG_RTREE : HA_KEY_ALG_BTREE) : + pos->algorithm; + keydef[i].block_length= pos->block_size; + keydef[i].seg= keyseg; + keydef[i].keysegs= pos->key_parts; + for (j= 0; j < pos->key_parts; j++) + { + Field *field= pos->key_part[j].field; + type= field->key_type(); + keydef[i].seg[j].flag= pos->key_part[j].key_part_flag; + + if (options & HA_OPTION_PACK_KEYS || + (pos->flags & (HA_PACK_KEY | HA_BINARY_PACK_KEY | + HA_SPACE_PACK_USED))) + { + if (pos->key_part[j].length > 8 && + (type == HA_KEYTYPE_TEXT || + type == HA_KEYTYPE_NUM || + (type == HA_KEYTYPE_BINARY && !field->zero_pack()))) + { + /* No blobs here */ + if (j == 0) + keydef[i].flag|= HA_PACK_KEY; + if (!(field->flags & ZEROFILL_FLAG) && + (field->type() == MYSQL_TYPE_STRING || + field->type() == MYSQL_TYPE_VAR_STRING || + ((int) (pos->key_part[j].length - field->decimals())) >= 4)) + keydef[i].seg[j].flag|= HA_SPACE_PACK; + } + else if (j == 0 && (!(pos->flags & HA_NOSAME) || pos->key_length > 16)) + keydef[i].flag|= HA_BINARY_PACK_KEY; + } + keydef[i].seg[j].type= (int) type; + keydef[i].seg[j].start= pos->key_part[j].offset; + keydef[i].seg[j].length= pos->key_part[j].length; + keydef[i].seg[j].bit_start= keydef[i].seg[j].bit_end= + keydef[i].seg[j].bit_length= 0; + keydef[i].seg[j].bit_pos= 0; + keydef[i].seg[j].language= field->charset()->number; + + if (field->null_ptr) + { + keydef[i].seg[j].null_bit= field->null_bit; + keydef[i].seg[j].null_pos= (uint) (field->null_ptr- + (uchar*) table_arg->record[0]); + } + else + { + keydef[i].seg[j].null_bit= 0; + keydef[i].seg[j].null_pos= 0; + } + if (field->type() == MYSQL_TYPE_BLOB || + field->type() == MYSQL_TYPE_GEOMETRY) + { + keydef[i].seg[j].flag|= HA_BLOB_PART; + /* save number of bytes used to pack length */ + keydef[i].seg[j].bit_start= (uint) (field->pack_length() - + share->blob_ptr_size); + } + else if (field->type() == MYSQL_TYPE_BIT) + { + keydef[i].seg[j].bit_length= ((Field_bit *) field)->bit_len; + keydef[i].seg[j].bit_start= ((Field_bit *) field)->bit_ofs; + keydef[i].seg[j].bit_pos= (uint) (((Field_bit *) field)->bit_ptr - + (uchar*) table_arg->record[0]); + } + } + keyseg+= pos->key_parts; + } + if (table_arg->found_next_number_field) + keydef[share->next_number_index].flag|= HA_AUTO_KEY; + record= table_arg->record[0]; + recpos= 0; + recinfo_pos= recinfo; + while (recpos < (uint) share->reclength) + { + Field **field, *found= 0; + minpos= share->reclength; + length= 0; + + for (field= table_arg->field; *field; field++) + { + if ((fieldpos= (*field)->offset(record)) >= recpos && + fieldpos <= minpos) + { + /* skip null fields */ + if (!(temp_length= (*field)->pack_length_in_rec())) + continue; /* Skip null-fields */ + if (! found || fieldpos < minpos || + (fieldpos == minpos && temp_length < length)) + { + minpos= fieldpos; + found= *field; + length= temp_length; + } + } + } + DBUG_PRINT("loop", ("found: 0x%lx recpos: %d minpos: %d length: %d", + (long) found, recpos, minpos, length)); + if (recpos != minpos) + { // Reserved space (Null bits?) + bzero((char*) recinfo_pos, sizeof(*recinfo_pos)); + recinfo_pos->type= FIELD_NORMAL; + recinfo_pos++->length= (uint16) (minpos - recpos); + } + if (!found) + break; + + if (found->flags & BLOB_FLAG) + recinfo_pos->type= FIELD_BLOB; + else if (found->type() == MYSQL_TYPE_VARCHAR) + recinfo_pos->type= FIELD_VARCHAR; + else if (!(options & HA_OPTION_PACK_RECORD)) + recinfo_pos->type= FIELD_NORMAL; + else if (found->zero_pack()) + recinfo_pos->type= FIELD_SKIP_ZERO; + else + recinfo_pos->type= ((length <= 3 || + (found->flags & ZEROFILL_FLAG)) ? + FIELD_NORMAL : + found->type() == MYSQL_TYPE_STRING || + found->type() == MYSQL_TYPE_VAR_STRING ? + FIELD_SKIP_ENDSPACE : + FIELD_SKIP_PRESPACE); + if (found->null_ptr) + { + recinfo_pos->null_bit= found->null_bit; + recinfo_pos->null_pos= (uint) (found->null_ptr - + (uchar*) table_arg->record[0]); + } + else + { + recinfo_pos->null_bit= 0; + recinfo_pos->null_pos= 0; + } + (recinfo_pos++)->length= (uint16) length; + recpos= minpos + length; + DBUG_PRINT("loop", ("length: %d type: %d", + recinfo_pos[-1].length,recinfo_pos[-1].type)); + } + *records_out= (uint) (recinfo_pos - recinfo); + DBUG_RETURN(0); +} + + +/* + Check for underlying table conformance + + SYNOPSIS + maria_check_definition() + t1_keyinfo in First table key definition + t1_recinfo in First table record definition + t1_keys in Number of keys in first table + t1_recs in Number of records in first table + t2_keyinfo in Second table key definition + t2_recinfo in Second table record definition + t2_keys in Number of keys in second table + t2_recs in Number of records in second table + strict in Strict check switch + + DESCRIPTION + This function compares two Maria definitions. By intention it was done + to compare merge table definition against underlying table definition. + It may also be used to compare dot-frm and MAI definitions of Maria + table as well to compare different Maria table definitions. + + For merge table it is not required that number of keys in merge table + must exactly match number of keys in underlying table. When calling this + function for underlying table conformance check, 'strict' flag must be + set to false, and converted merge definition must be passed as t1_*. + + Otherwise 'strict' flag must be set to 1 and it is not required to pass + converted dot-frm definition as t1_*. + + RETURN VALUE + 0 - Equal definitions. + 1 - Different definitions. + + NOTES + This is currently not used. In MyISAM the corresponding function + (myisam_check_definition()) is used only by MERGE tables + (in ha_myisammrg.cc). +*/ +int maria_check_definition(MARIA_KEYDEF *t1_keyinfo, + MARIA_COLUMNDEF *t1_recinfo, + uint t1_keys, uint t1_recs, + MARIA_KEYDEF *t2_keyinfo, + MARIA_COLUMNDEF *t2_recinfo, + uint t2_keys, uint t2_recs, bool strict) +{ + uint i, j; + DBUG_ENTER("maria_check_definition"); + if ((strict ? t1_keys != t2_keys : t1_keys > t2_keys)) + { + DBUG_PRINT("error", ("Number of keys differs: t1_keys=%u, t2_keys=%u", + t1_keys, t2_keys)); + DBUG_RETURN(1); + } + if (t1_recs != t2_recs) + { + DBUG_PRINT("error", ("Number of recs differs: t1_recs=%u, t2_recs=%u", + t1_recs, t2_recs)); + DBUG_RETURN(1); + } + for (i= 0; i < t1_keys; i++) + { + HA_KEYSEG *t1_keysegs= t1_keyinfo[i].seg; + HA_KEYSEG *t2_keysegs= t2_keyinfo[i].seg; + if (t1_keyinfo[i].keysegs != t2_keyinfo[i].keysegs || + t1_keyinfo[i].key_alg != t2_keyinfo[i].key_alg) + { + DBUG_PRINT("error", ("Key %d has different definition", i)); + DBUG_PRINT("error", ("t1_keysegs=%d, t1_key_alg=%d", + t1_keyinfo[i].keysegs, t1_keyinfo[i].key_alg)); + DBUG_PRINT("error", ("t2_keysegs=%d, t2_key_alg=%d", + t2_keyinfo[i].keysegs, t2_keyinfo[i].key_alg)); + DBUG_RETURN(1); + } + for (j= t1_keyinfo[i].keysegs; j--;) + { + if (t1_keysegs[j].type != t2_keysegs[j].type || + t1_keysegs[j].language != t2_keysegs[j].language || + t1_keysegs[j].null_bit != t2_keysegs[j].null_bit || + t1_keysegs[j].length != t2_keysegs[j].length) + { + DBUG_PRINT("error", ("Key segment %d (key %d) has different " + "definition", j, i)); + DBUG_PRINT("error", ("t1_type=%d, t1_language=%d, t1_null_bit=%d, " + "t1_length=%d", + t1_keysegs[j].type, t1_keysegs[j].language, + t1_keysegs[j].null_bit, t1_keysegs[j].length)); + DBUG_PRINT("error", ("t2_type=%d, t2_language=%d, t2_null_bit=%d, " + "t2_length=%d", + t2_keysegs[j].type, t2_keysegs[j].language, + t2_keysegs[j].null_bit, t2_keysegs[j].length)); + + DBUG_RETURN(1); + } + } + } + for (i= 0; i < t1_recs; i++) + { + MARIA_COLUMNDEF *t1_rec= &t1_recinfo[i]; + MARIA_COLUMNDEF *t2_rec= &t2_recinfo[i]; + if (t1_rec->type != t2_rec->type || + t1_rec->length != t2_rec->length || + t1_rec->null_bit != t2_rec->null_bit) + { + DBUG_PRINT("error", ("Field %d has different definition", i)); + DBUG_PRINT("error", ("t1_type=%d, t1_length=%d, t1_null_bit=%d", + t1_rec->type, t1_rec->length, t1_rec->null_bit)); + DBUG_PRINT("error", ("t2_type=%d, t2_length=%d, t2_null_bit=%d", + t2_rec->type, t2_rec->length, t2_rec->null_bit)); + DBUG_RETURN(1); + } + } + DBUG_RETURN(0); +} + + extern "C" { volatile int *_ma_killed_ptr(HA_CHECK *param) @@ -315,17 +624,33 @@ bool ha_maria::check_if_locking_is_allowed(uint sql_command, int ha_maria::open(const char *name, int mode, uint test_if_locked) { uint i; + +#ifdef NOT_USED + /* + If the user wants to have memory mapped data files, add an + open_flag. Do not memory map temporary tables because they are + expected to be inserted and thus extended a lot. Memory mapping is + efficient for files that keep their size, but very inefficient for + growing files. Using an open_flag instead of calling ma_extra(... + HA_EXTRA_MMAP ...) after maxs_open() has the advantage that the + mapping is not repeated for every open, but just done on the initial + open, when the MyISAM share is created. Everytime the server + requires to open a new instance of a table it calls this method. We + will always supply HA_OPEN_MMAP for a permanent table. However, the + Maria storage engine will ignore this flag if this is a secondary + open of a table that is in use by other threads already (if the + Maria share exists already). + */ + if (!(test_if_locked & HA_OPEN_TMP_TABLE) && opt_maria_use_mmap) + test_if_locked|= HA_OPEN_MMAP; +#endif + if (!(file= maria_open(name, mode, test_if_locked | HA_OPEN_FROM_SQL_LAYER))) return (my_errno ? my_errno : -1); if (test_if_locked & (HA_OPEN_IGNORE_IF_LOCKED | HA_OPEN_TMP_TABLE)) VOID(maria_extra(file, HA_EXTRA_NO_WAIT_LOCK, 0)); -#ifdef NOT_USED - if (!(test_if_locked & HA_OPEN_TMP_TABLE) && opt_maria_use_mmap) - VOID(maria_extra(file, HA_EXTRA_MMAP, 0)); -#endif - info(HA_STATUS_NO_LOCK | HA_STATUS_VARIABLE | HA_STATUS_CONST); if (!(test_if_locked & HA_OPEN_WAIT_IF_LOCKED)) VOID(maria_extra(file, HA_EXTRA_WAIT_LOCK, 0)); @@ -684,11 +1009,11 @@ int ha_maria::optimize(THD * thd, HA_CHECK_OPT *check_opt) } -int ha_maria::repair(THD *thd, HA_CHECK ¶m, bool optimize) +int ha_maria::repair(THD *thd, HA_CHECK ¶m, bool do_optimize) { int error= 0; uint local_testflag= param.testflag; - bool optimize_done= !optimize, statistics_done= 0; + bool optimize_done= !do_optimize, statistics_done= 0; const char *old_proc_info= thd->proc_info; char fixed_name[FN_REFLEN]; MARIA_SHARE *share= file->s; @@ -712,7 +1037,7 @@ int ha_maria::repair(THD *thd, HA_CHECK ¶m, bool optimize) DBUG_RETURN(HA_ADMIN_FAILED); } - if (!optimize || + if (!do_optimize || ((file->state->del || share->state.split != file->state->records) && (!(param.testflag & T_QUICK) || !(share->state.changed & STATE_NOT_OPTIMIZED_KEYS)))) @@ -1364,47 +1689,47 @@ int ha_maria::rnd_pos(byte * buf, byte *pos) void ha_maria::position(const byte * record) { - my_off_t position= maria_position(file); - my_store_ptr(ref, ref_length, position); + my_off_t row_position= maria_position(file); + my_store_ptr(ref, ref_length, row_position); } int ha_maria::info(uint flag) { - MARIA_INFO info; + MARIA_INFO maria_info; char name_buff[FN_REFLEN]; - (void) maria_status(file, &info, flag); + (void) maria_status(file, &maria_info, flag); if (flag & HA_STATUS_VARIABLE) { - stats.records= info.records; - stats.deleted= info.deleted; - stats.data_file_length= info.data_file_length; - stats.index_file_length= info.index_file_length; - stats.delete_length= info.delete_length; - stats.check_time= info.check_time; - stats.mean_rec_length= info.mean_reclength; + stats.records= maria_info.records; + stats.deleted= maria_info.deleted; + stats.data_file_length= maria_info.data_file_length; + stats.index_file_length= maria_info.index_file_length; + stats.delete_length= maria_info.delete_length; + stats.check_time= maria_info.check_time; + stats.mean_rec_length= maria_info.mean_reclength; } if (flag & HA_STATUS_CONST) { TABLE_SHARE *share= table->s; - stats.max_data_file_length= info.max_data_file_length; - stats.max_index_file_length= info.max_index_file_length; - stats.create_time= info.create_time; - ref_length= info.reflength; - share->db_options_in_use= info.options; + stats.max_data_file_length= maria_info.max_data_file_length; + stats.max_index_file_length= maria_info.max_index_file_length; + stats.create_time= maria_info.create_time; + ref_length= maria_info.reflength; + share->db_options_in_use= maria_info.options; stats.block_size= maria_block_size; /* Update share */ if (share->tmp_table == NO_TMP_TABLE) pthread_mutex_lock(&share->mutex); share->keys_in_use.set_prefix(share->keys); - share->keys_in_use.intersect_extended(info.key_map); + share->keys_in_use.intersect_extended(maria_info.key_map); share->keys_for_keyread.intersect(share->keys_in_use); - share->db_record_offset= info.record_offset; + share->db_record_offset= maria_info.record_offset; if (share->key_parts) memcpy((char*) table->key_info[0].rec_per_key, - (char*) info.rec_per_key, + (char*) maria_info.rec_per_key, sizeof(table->key_info[0].rec_per_key) * share->key_parts); if (share->tmp_table == NO_TMP_TABLE) pthread_mutex_unlock(&share->mutex); @@ -1414,21 +1739,23 @@ int ha_maria::info(uint flag) if table is symlinked (Ie; Real name is not same as generated name) */ data_file_name= index_file_name= 0; - fn_format(name_buff, file->filename, "", MARIA_NAME_DEXT, MY_APPEND_EXT); - if (strcmp(name_buff, info.data_file_name)) - data_file_name= info.data_file_name; - fn_format(name_buff, file->filename, "", MARIA_NAME_IEXT, MY_APPEND_EXT); - if (strcmp(name_buff, info.index_file_name)) - index_file_name= info.index_file_name; + fn_format(name_buff, file->filename, "", MARIA_NAME_DEXT, + MY_APPEND_EXT | MY_UNPACK_FILENAME); + if (strcmp(name_buff, maria_info.data_file_name)) + data_file_name=maria_info.data_file_name; + fn_format(name_buff, file->filename, "", MARIA_NAME_IEXT, + MY_APPEND_EXT | MY_UNPACK_FILENAME); + if (strcmp(name_buff, maria_info.index_file_name)) + index_file_name=maria_info.index_file_name; } if (flag & HA_STATUS_ERRKEY) { - errkey= info.errkey; - my_store_ptr(dup_ref, ref_length, info.dup_key_pos); + errkey= maria_info.errkey; + my_store_ptr(dup_ref, ref_length, maria_info.dup_key_pos); } /* Faster to always update, than to do it based on flag */ - stats.update_time= info.update_time; - stats.auto_increment_value= info.auto_increment; + stats.update_time= maria_info.update_time; + stats.auto_increment_value= maria_info.auto_increment; return 0; } @@ -1500,208 +1827,50 @@ void ha_maria::update_create_info(HA_CREATE_INFO *create_info) int ha_maria::create(const char *name, register TABLE *table_arg, - HA_CREATE_INFO *info) + HA_CREATE_INFO *ha_create_info) { int error; - uint i, j, recpos, minpos, fieldpos, temp_length, length, create_flags= 0; - bool found_real_auto_increment= 0; - enum ha_base_keytype type; + uint create_flags= 0, records, i; char buff[FN_REFLEN]; - byte *record; - KEY *pos; MARIA_KEYDEF *keydef; - MARIA_COLUMNDEF *recinfo, *recinfo_pos; - HA_KEYSEG *keyseg; + MARIA_COLUMNDEF *recinfo; + MARIA_CREATE_INFO create_info; TABLE_SHARE *share= table_arg->s; uint options= share->db_options_in_use; enum data_file_type row_type; DBUG_ENTER("ha_maria::create"); - - type= HA_KEYTYPE_BINARY; // Keep compiler happy - if (!(my_multi_malloc(MYF(MY_WME), - &recinfo, (share->fields * 2 + 2) * - sizeof(MARIA_COLUMNDEF), - &keydef, share->keys * sizeof(MARIA_KEYDEF), - &keyseg, - ((share->key_parts + share->keys) * - sizeof(HA_KEYSEG)), NullS))) - DBUG_RETURN(HA_ERR_OUT_OF_MEM); - - pos= table_arg->key_info; - for (i= 0; i < share->keys; i++, pos++) - { - if (pos->flags & HA_USES_PARSER) - create_flags |= HA_CREATE_RELIES_ON_SQL_LAYER; - keydef[i].flag= (pos->flags & (HA_NOSAME | HA_FULLTEXT | HA_SPATIAL)); - keydef[i].key_alg= pos->algorithm == HA_KEY_ALG_UNDEF ? - (pos->flags & HA_SPATIAL ? HA_KEY_ALG_RTREE : HA_KEY_ALG_BTREE) : - pos->algorithm; - keydef[i].block_length= pos->block_size; - - keydef[i].seg= keyseg; - keydef[i].keysegs= pos->key_parts; - for (j= 0; j < pos->key_parts; j++) - { - Field *field= pos->key_part[j].field; - type= field->key_type(); - keydef[i].seg[j].flag= pos->key_part[j].key_part_flag; - - if (options & HA_OPTION_PACK_KEYS || - (pos->flags & (HA_PACK_KEY | HA_BINARY_PACK_KEY | - HA_SPACE_PACK_USED))) - { - if (pos->key_part[j].length > 8 && - (type == HA_KEYTYPE_TEXT || - type == HA_KEYTYPE_NUM || - (type == HA_KEYTYPE_BINARY && !field->zero_pack()))) - { - /* No blobs here */ - if (j == 0) - keydef[i].flag |= HA_PACK_KEY; - if (!(field->flags & ZEROFILL_FLAG) && - (field->type() == MYSQL_TYPE_STRING || - field->type() == MYSQL_TYPE_VAR_STRING || - ((int) (pos->key_part[j].length - field->decimals())) >= 4)) - keydef[i].seg[j].flag |= HA_SPACE_PACK; - } - else if (j == 0 && (!(pos->flags & HA_NOSAME) || pos->key_length > 16)) - keydef[i].flag |= HA_BINARY_PACK_KEY; - } - keydef[i].seg[j].type= (int) type; - keydef[i].seg[j].start= pos->key_part[j].offset; - keydef[i].seg[j].length= pos->key_part[j].length; - keydef[i].seg[j].bit_start= keydef[i].seg[j].bit_end= - keydef[i].seg[j].bit_length= 0; - keydef[i].seg[j].bit_pos= 0; - keydef[i].seg[j].language= field->charset()->number; - - if (field->null_ptr) - { - keydef[i].seg[j].null_bit= field->null_bit; - keydef[i].seg[j].null_pos= (uint) (field->null_ptr - - (uchar *) table_arg->record[0]); - } - else - { - keydef[i].seg[j].null_bit= 0; - keydef[i].seg[j].null_pos= 0; - } - if (field->type() == FIELD_TYPE_BLOB || - field->type() == FIELD_TYPE_GEOMETRY) - { - keydef[i].seg[j].flag |= HA_BLOB_PART; - /* save number of bytes used to pack length */ - keydef[i].seg[j].bit_start= (uint) (field->pack_length() - - share->blob_ptr_size); - } - else if (field->type() == FIELD_TYPE_BIT) - { - keydef[i].seg[j].bit_length= ((Field_bit *) field)->bit_len; - keydef[i].seg[j].bit_start= ((Field_bit *) field)->bit_ofs; - keydef[i].seg[j].bit_pos= (uint) (((Field_bit *) field)->bit_ptr - - (uchar *) table_arg->record[0]); - } - } - keyseg += pos->key_parts; - } - - if (table_arg->found_next_number_field) + for (i= 0; i < share->keys; i++) { - keydef[share->next_number_index].flag |= HA_AUTO_KEY; - found_real_auto_increment= share->next_number_key_offset == 0; - } - - record= table_arg->record[0]; - recpos= 0; - recinfo_pos= recinfo; - while (recpos < (uint) share->reclength) - { - Field **field, *found= 0; - minpos= share->reclength; - length= 0; - - for (field= table_arg->field; *field; field++) + if (table_arg->key_info[i].flags & HA_USES_PARSER) { - if ((fieldpos=(*field)->offset(record)) >= recpos && - fieldpos <= minpos) - { - /* skip null fields */ - if (!(temp_length= (*field)->pack_length_in_rec())) - continue; /* Skip null-fields */ - if (!found || fieldpos < minpos || - (fieldpos == minpos && temp_length < length)) - { - minpos= fieldpos; - found= *field; - length= temp_length; - } - } - } - DBUG_PRINT("loop", ("found: 0x%lx recpos: %d minpos: %d length: %d", - (long) found, recpos, minpos, length)); - if (recpos != minpos) - { // Reserved space (Null bits?) - bzero((char*) recinfo_pos, sizeof(*recinfo_pos)); - recinfo_pos->type= FIELD_NORMAL; - recinfo_pos++->length= (uint16) (minpos - recpos); - } - if (!found) + create_flags|= HA_CREATE_RELIES_ON_SQL_LAYER; break; - - if (found->flags & BLOB_FLAG) - recinfo_pos->type= FIELD_BLOB; - else if (found->type() == MYSQL_TYPE_VARCHAR) - recinfo_pos->type= FIELD_VARCHAR; - else if (!(options & HA_OPTION_PACK_RECORD)) - recinfo_pos->type= FIELD_NORMAL; - else if (found->zero_pack()) - recinfo_pos->type= FIELD_SKIP_ZERO; - else - recinfo_pos->type= ((length <= 3 || - (found->flags & ZEROFILL_FLAG)) ? - FIELD_NORMAL : - found->type() == MYSQL_TYPE_STRING || - found->type() == MYSQL_TYPE_VAR_STRING ? - FIELD_SKIP_ENDSPACE : FIELD_SKIP_PRESPACE); - if (found->null_ptr) - { - recinfo_pos->null_bit= found->null_bit; - recinfo_pos->null_pos= (uint) (found->null_ptr - - (uchar *) table_arg->record[0]); } - else - { - recinfo_pos->null_bit= 0; - recinfo_pos->null_pos= 0; - } - (recinfo_pos++)->length= (uint16) length; - recpos= minpos + length; - DBUG_PRINT("loop", ("length: %d type: %d", - recinfo_pos[-1].length, recinfo_pos[-1].type)); - } - MARIA_CREATE_INFO create_info; + if ((error= table2maria(table_arg, &keydef, &recinfo, &records))) + DBUG_RETURN(error); /* purecov: inspected */ bzero((char*) &create_info, sizeof(create_info)); create_info.max_rows= share->max_rows; create_info.reloc_rows= share->min_rows; - create_info.with_auto_increment= found_real_auto_increment; - create_info.auto_increment= (info->auto_increment_value ? - info->auto_increment_value - 1 : (ulonglong) 0); + create_info.with_auto_increment= share->next_number_key_offset == 0; + create_info.auto_increment= (ha_create_info->auto_increment_value ? + ha_create_info->auto_increment_value -1 : + (ulonglong) 0); create_info.data_file_length= ((ulonglong) share->max_rows * share->avg_row_length); - create_info.data_file_name= info->data_file_name; - create_info.index_file_name= info->index_file_name; + create_info.data_file_name= ha_create_info->data_file_name; + create_info.index_file_name= ha_create_info->index_file_name; - if (info->options & HA_LEX_CREATE_TMP_TABLE) - create_flags |= HA_CREATE_TMP_TABLE; + if (ha_create_info->options & HA_LEX_CREATE_TMP_TABLE) + create_flags|= HA_CREATE_TMP_TABLE; if (options & HA_OPTION_PACK_RECORD) - create_flags |= HA_PACK_RECORD; + create_flags|= HA_PACK_RECORD; if (options & HA_OPTION_CHECKSUM) - create_flags |= HA_CREATE_CHECKSUM; + create_flags|= HA_CREATE_CHECKSUM; if (options & HA_OPTION_DELAY_KEY_WRITE) - create_flags |= HA_CREATE_DELAY_KEY_WRITE; + create_flags|= HA_CREATE_DELAY_KEY_WRITE; - switch (info->row_type) { + switch (ha_create_info->row_type) { case ROW_TYPE_FIXED: row_type= STATIC_RECORD; break; @@ -1718,9 +1887,10 @@ int ha_maria::create(const char *name, register TABLE *table_arg, error= maria_create(fn_format(buff, name, "", "", MY_UNPACK_FILENAME | MY_APPEND_EXT), - row_type, share->keys, keydef, (uint) (recinfo_pos - recinfo), - recinfo, 0, (MARIA_UNIQUEDEF *) 0, &create_info, - create_flags); + row_type, share->keys, keydef, + records, recinfo, + 0, (MARIA_UNIQUEDEF *) 0, + &create_info, create_flags); my_free((gptr) recinfo, MYF(0)); DBUG_RETURN(error); diff --git a/storage/maria/ma_check.c b/storage/maria/ma_check.c index ccce19de994..a247d36144c 100644 --- a/storage/maria/ma_check.c +++ b/storage/maria/ma_check.c @@ -2310,6 +2310,12 @@ int maria_sort_index(HA_CHECK *param, register MARIA_HA *info, my_string name) MARIA_STATE_INFO old_state; DBUG_ENTER("maria_sort_index"); + /* cannot sort index files with R-tree indexes */ + for (key= 0,keyinfo= &share->keyinfo[0]; key < share->base.keys ; + key++,keyinfo++) + if (keyinfo->key_alg == HA_KEY_ALG_RTREE) + DBUG_RETURN(0); + if (!(param->testflag & T_SILENT)) printf("- Sorting index for MARIA-table '%s'\n",name); @@ -2402,6 +2408,8 @@ static int sort_one_index(HA_CHECK *param, MARIA_HA *info, char llbuff[22]; DBUG_ENTER("sort_one_index"); + /* cannot walk over R-tree indices */ + DBUG_ASSERT(keyinfo->key_alg != HA_KEY_ALG_RTREE); new_page_pos=param->new_file_pos; param->new_file_pos+=keyinfo->block_length; diff --git a/storage/maria/ma_close.c b/storage/maria/ma_close.c index 2a874079961..508334fcf67 100644 --- a/storage/maria/ma_close.c +++ b/storage/maria/ma_close.c @@ -59,7 +59,7 @@ int maria_close(register MARIA_HA *info) } flag= !--share->reopen; /* - RECOVERYTODO: + RECOVERY TODO: If "flag" is TRUE, in the line below we are going to make the table unknown to future checkpoints, so it needs to have fsync'ed itself entirely (bitmap, pages, etc) at this point. diff --git a/storage/maria/ma_create.c b/storage/maria/ma_create.c index 59f5f7d1a4d..bd9382ff9b0 100644 --- a/storage/maria/ma_create.c +++ b/storage/maria/ma_create.c @@ -921,7 +921,7 @@ int maria_create(const char *name, enum data_file_type record_type, if (my_close(file,MYF(0))) goto err; /* - RECOVERYTODO + RECOVERY TODO Write a log record describing the CREATE operation (just the file names, link names, and the full header's content). For this record to be of any use for Recovery, we need the upper @@ -967,18 +967,19 @@ uint maria_get_pointer_length(ulonglong file_length, uint def) if (file_length) /* If not default */ { #ifdef NOT_YET_READY_FOR_8_BYTE_POINTERS - if (file_length >= (longlong) 1 << 56) + if (file_length >= (ULL(1) << 56)) def=8; + else #endif - if (file_length >= (longlong) 1 << 48) + if (file_length >= (ULL(1) << 48)) def=7; - if (file_length >= (longlong) 1 << 40) + else if (file_length >= (ULL(1) << 40)) def=6; - else if (file_length >= (longlong) 1 << 32) + else if (file_length >= (ULL(1) << 32)) def=5; - else if (file_length >= (1L << 24)) + else if (file_length >= (ULL(1) << 24)) def=4; - else if (file_length >= (1L << 16)) + else if (file_length >= (ULL(1) << 16)) def=3; else def=2; diff --git a/storage/maria/ma_delete_all.c b/storage/maria/ma_delete_all.c index 616147c1067..428724ec313 100644 --- a/storage/maria/ma_delete_all.c +++ b/storage/maria/ma_delete_all.c @@ -30,7 +30,7 @@ int maria_delete_all_rows(MARIA_HA *info) { DBUG_RETURN(my_errno=EACCES); } - /* LOCKTODO take X-lock on table here */ + /* LOCK TODO take X-lock on table here */ if (_ma_readinfo(info,F_WRLCK,1)) DBUG_RETURN(my_errno); if (_ma_mark_file_changed(info)) @@ -54,7 +54,7 @@ int maria_delete_all_rows(MARIA_HA *info) */ flush_key_blocks(share->key_cache, share->kfile, FLUSH_IGNORE_CHANGED); /* - RECOVERYTODO Log the two chsize and header modifications and force the + RECOVERY TODO Log the two chsize and header modifications and force the log. So that if crash between the two chsize, we finish the work at Recovery. For this scenario: "TRUNCATE TABLE t1; DROP TABLE t1; RENAME TABLE t2 to t1; crash;" @@ -66,7 +66,7 @@ int maria_delete_all_rows(MARIA_HA *info) my_chsize(share->kfile, share->base.keystart, 0, MYF(MY_WME)) ) goto err; /* - RECOVERYTODO Consider updating ZeroDirtyPagesLSN here. It is + RECOVERY TODO Consider updating ZeroDirtyPagesLSN here. It is not a necessity (it is one only in RENAME commands) but an optional optimization which will allow some REDO skipping at Recovery. */ @@ -78,7 +78,7 @@ int maria_delete_all_rows(MARIA_HA *info) rw_unlock(&info->s->mmap_lock); #endif /* - RECOVERYTODO Until we have the TRUNCATE log record and take it into + RECOVERY TODO Until we have the TRUNCATE log record and take it into account for log-low-water-mark calculation and use it in Recovery, we need to sync. */ @@ -90,10 +90,10 @@ int maria_delete_all_rows(MARIA_HA *info) err: { int save_errno=my_errno; - /* RECOVERYTODO log the header modifications */ + /* RECOVERY TODO log the header modifications */ VOID(_ma_writeinfo(info,WRITEINFO_UPDATE_KEYFILE)); info->update|=HA_STATE_WRITTEN; /* Buffer changed */ - /* RECOVERYTODO until we log above we have to sync */ + /* RECOVERY TODO until we log above we have to sync */ if (_ma_sync_table_files(info) && !save_errno) save_errno= my_errno; allow_break(); /* Allow SIGHUP & SIGINT */ diff --git a/storage/maria/ma_delete_table.c b/storage/maria/ma_delete_table.c index 5c7b4337b20..47ba56e031c 100644 --- a/storage/maria/ma_delete_table.c +++ b/storage/maria/ma_delete_table.c @@ -31,7 +31,7 @@ int maria_delete_table(const char *name) #ifdef EXTRA_DEBUG _ma_check_table_is_closed(name,"delete"); #endif - /* LOCKTODO take X-lock on table here */ + /* LOCK TODO take X-lock on table here */ #ifdef USE_RAID { MARIA_HA *info; @@ -61,7 +61,7 @@ int maria_delete_table(const char *name) fn_format(from,name,"",MARIA_NAME_IEXT,MY_UNPACK_FILENAME|MY_APPEND_EXT); /* - RECOVERYTODO log the two deletes below. + RECOVERY TODO log the two deletes below. Then do the file deletions. For this log record to be of any use for Recovery, we need the upper MySQL layer to be crash-safe in DDLs; when it is we should reconsider the moment diff --git a/storage/maria/ma_dynrec.c b/storage/maria/ma_dynrec.c index c3d0bf1fb0a..52780ab4f6f 100644 --- a/storage/maria/ma_dynrec.c +++ b/storage/maria/ma_dynrec.c @@ -69,6 +69,14 @@ my_bool _ma_dynmap_file(MARIA_HA *info, my_off_t size) DBUG_PRINT("warning", ("File is too large for mmap")); DBUG_RETURN(1); } + /* + Ingo wonders if it is good to use MAP_NORESERVE. From the Linux man page: + MAP_NORESERVE + Do not reserve swap space for this mapping. When swap space is + reserved, one has the guarantee that it is possible to modify the + mapping. When swap space is not reserved one might get SIGSEGV + upon a write if no physical memory is available. + */ info->s->file_map= (byte*) my_mmap(0, (size_t)(size + MEMMAP_EXTRA_MARGIN), info->s->mode==O_RDONLY ? PROT_READ : @@ -252,7 +260,7 @@ my_bool _ma_write_blob_record(MARIA_HA *info, const byte *record) _ma_calc_total_blob_length(info,record)+ extra); if (!(rec_buff=(byte*) my_alloca(reclength))) { - my_errno=ENOMEM; + my_errno= HA_ERR_OUT_OF_MEM; /* purecov: inspected */ return(1); } reclength2= _ma_rec_pack(info, @@ -289,7 +297,7 @@ my_bool _ma_update_blob_record(MARIA_HA *info, MARIA_RECORD_POS pos, #endif if (!(rec_buff=(byte*) my_alloca(reclength))) { - my_errno=ENOMEM; + my_errno= HA_ERR_OUT_OF_MEM; /* purecov: inspected */ return(1); } reclength= _ma_rec_pack(info,rec_buff+ALIGN_SIZE(MARIA_MAX_DYN_BLOCK_HEADER), @@ -1225,8 +1233,10 @@ ulong _ma_rec_unpack(register MARIA_HA *info, register byte *to, byte *from, { uint size_length=rec_length- maria_portable_sizeof_char_ptr; ulong blob_length= _ma_calc_blob_length(size_length,from); - if ((ulong) (from_end-from) - size_length < blob_length || - min_pack_length > (uint) (from_end -(from+size_length+blob_length))) + ulong from_left= (ulong) (from_end - from); + if (from_left < size_length || + from_left - size_length < blob_length || + from_left - size_length - blob_length < min_pack_length) goto err; memcpy((byte*) to,(byte*) from,(size_t) size_length); from+=size_length; diff --git a/storage/maria/ma_extra.c b/storage/maria/ma_extra.c index 90e79362442..59ae18d949a 100644 --- a/storage/maria/ma_extra.c +++ b/storage/maria/ma_extra.c @@ -355,7 +355,12 @@ int maria_extra(MARIA_HA *info, enum ha_extra_function function, case HA_EXTRA_MMAP: #ifdef HAVE_MMAP pthread_mutex_lock(&share->intern_lock); - if (!share->file_map) + /* + Memory map the data file if it is not already mapped and if there + are no other threads using this table. intern_lock prevents other + threads from starting to use the table while we are mapping it. + */ + if (!share->file_map && (share->tot_locks == 1)) { if (_ma_dynmap_file(info, share->state.state.data_file_length)) { diff --git a/storage/maria/ma_ft_boolean_search.c b/storage/maria/ma_ft_boolean_search.c index 06b044e9e38..af76f0858a7 100644 --- a/storage/maria/ma_ft_boolean_search.c +++ b/storage/maria/ma_ft_boolean_search.c @@ -676,7 +676,7 @@ static void _ftb_climb_the_tree(FTB *ftb, FTB_WORD *ftbw, FT_SEG_ITERATOR *ftsi_ FT_SEG_ITERATOR ftsi; FTB_EXPR *ftbe; float weight=ftbw->weight; - int yn=ftbw->flags, ythresh, mode=(ftsi_orig != 0); + int yn_flag= ftbw->flags, ythresh, mode=(ftsi_orig != 0); my_off_t curdoc=ftbw->docid[mode]; struct st_mysql_ftparser *parser= ftb->keynr == NO_SUCH_KEY ? &ft_default_parser : @@ -693,13 +693,13 @@ static void _ftb_climb_the_tree(FTB *ftb, FTB_WORD *ftbw, FT_SEG_ITERATOR *ftsi_ } if (ftbe->nos) break; - if (yn & FTB_FLAG_YES) + if (yn_flag & FTB_FLAG_YES) { weight /= ftbe->ythresh; ftbe->cur_weight += weight; if ((int) ++ftbe->yesses == ythresh) { - yn=ftbe->flags; + yn_flag=ftbe->flags; weight=ftbe->cur_weight*ftbe->weight; if (mode && ftbe->phrase) { @@ -720,14 +720,14 @@ static void _ftb_climb_the_tree(FTB *ftb, FTB_WORD *ftbw, FT_SEG_ITERATOR *ftsi_ break; } else - if (yn & FTB_FLAG_NO) + if (yn_flag & FTB_FLAG_NO) { /* NOTE: special sort function of queue assures that all - (yn & FTB_FLAG_NO) != 0 + (yn_flag & FTB_FLAG_NO) != 0 events for every particular subexpression will "auto-magically" happen BEFORE all the - (yn & FTB_FLAG_YES) != 0 events. So no + (yn_flag & FTB_FLAG_YES) != 0 events. So no already matched expression can become not-matched again. */ ++ftbe->nos; @@ -740,8 +740,8 @@ static void _ftb_climb_the_tree(FTB *ftb, FTB_WORD *ftbw, FT_SEG_ITERATOR *ftsi_ ftbe->cur_weight += weight; if ((int) ftbe->yesses < ythresh) break; - if (!(yn & FTB_FLAG_WONLY)) - yn= ((int) ftbe->yesses++ == ythresh) ? ftbe->flags : FTB_FLAG_WONLY ; + if (!(yn_flag & FTB_FLAG_WONLY)) + yn_flag= ((int) ftbe->yesses++ == ythresh) ? ftbe->flags : FTB_FLAG_WONLY ; weight*= ftbe->weight; } } diff --git a/storage/maria/ma_locking.c b/storage/maria/ma_locking.c index 6a0bbe82dcb..e41b2d490dc 100644 --- a/storage/maria/ma_locking.c +++ b/storage/maria/ma_locking.c @@ -291,6 +291,15 @@ void _ma_update_status(void* param) } } + +void _ma_restore_status(void *param) +{ + MARIA_HA *info= (MARIA_HA*) param; + info->state= &info->s->state.state; + info->append_insert_at_end= 0; +} + + void _ma_copy_status(void* to,void *from) { ((MARIA_HA*) to)->state= &((MARIA_HA*) from)->save_state; diff --git a/storage/maria/ma_loghandler.c b/storage/maria/ma_loghandler.c index 1e9afc7571d..d8a821cb538 100644 --- a/storage/maria/ma_loghandler.c +++ b/storage/maria/ma_loghandler.c @@ -799,7 +799,7 @@ static void translog_put_sector_protection(byte *page, static uint32 translog_crc(byte *area, uint length) { - return crc32(0L, area, length); + return crc32(0L, (unsigned char*) area, length); } diff --git a/storage/maria/ma_open.c b/storage/maria/ma_open.c index 19113196f15..7c4ad4c4a0a 100644 --- a/storage/maria/ma_open.c +++ b/storage/maria/ma_open.c @@ -310,7 +310,13 @@ MARIA_HA *maria_open(const char *name, int mode, uint open_flags) for (j=0 ; j < share->keyinfo[i].keysegs; j++,pos++) { disk_pos=_ma_keyseg_read(disk_pos, pos); - + if (pos->flag & HA_BLOB_PART && + ! (share->options & (HA_OPTION_COMPRESS_RECORD | + HA_OPTION_PACK_RECORD))) + { + my_errno= HA_ERR_CRASHED; + goto err; + } if (pos->type == HA_KEYTYPE_TEXT || pos->type == HA_KEYTYPE_VARTEXT1 || pos->type == HA_KEYTYPE_VARTEXT2) @@ -346,11 +352,11 @@ MARIA_HA *maria_open(const char *name, int mode, uint open_flags) } else { - uint j; + uint k; share->keyinfo[i].seg=pos; - for (j=0; j < FT_SEGS; j++) + for (k=0; k < FT_SEGS; k++) { - *pos= ft_keysegs[j]; + *pos= ft_keysegs[k]; pos[0].language= pos[-1].language; if (!(pos[0].charset= pos[-1].charset)) { @@ -444,6 +450,32 @@ MARIA_HA *maria_open(const char *name, int mode, uint open_flags) } } share->rec[i].type=(int) FIELD_LAST; /* End marker */ +#ifdef ASKMONTY + /* + This code was added to mi_open.c in this cset: + "ChangeSet 1.1616.2941.5 2007/01/22 16:34:58 svoj@mysql.com + BUG#24401 - MySQL server crashes if you try to retrieve data from + corrupted table + Accessing a table with corrupted column definition results in server + crash. + This is fixed by refusing to open such tables. Affects MyISAM only. + No test case, since it requires crashed table. + storage/myisam/mi_open.c 1.80.2.10 2007/01/22 16:34:57 svoj@mysql.com + Refuse to open MyISAM table with summary columns length bigger than + length of the record." + + The problem is that the "offset" variable was removed (by Monty in the + rows-in-block patch). Monty will know how to merge that. + Guilhem will make sure to notify him. + */ + if (offset > share->base.reclength) + { + /* purecov: begin inspected */ + my_errno= HA_ERR_CRASHED; + goto err; + /* purecov: end */ + } +#endif /* ASKMONTY */ if (_ma_open_datafile(&info, share, -1)) goto err; @@ -465,6 +497,22 @@ MARIA_HA *maria_open(const char *name, int mode, uint open_flags) _ma_setup_functions(share); if ((*share->once_init)(share, info.dfile)) goto err; + if (open_flags & HA_OPEN_MMAP) + { + info.s= share; + if (_ma_dynmap_file(&info, share->state.state.data_file_length)) + { + /* purecov: begin inspected */ + /* Ignore if mmap fails. Use file I/O instead. */ + DBUG_PRINT("warning", ("mmap failed: errno: %d", errno)); + /* purecov: end */ + } + else + { + share->file_read= _ma_mmap_pread; + share->file_write= _ma_mmap_pwrite; + } + } share->is_log_table= FALSE; #ifdef THREAD thr_lock_init(&share->lock); @@ -491,6 +539,7 @@ MARIA_HA *maria_open(const char *name, int mode, uint open_flags) share->lock.get_status=_ma_get_status; share->lock.copy_status=_ma_copy_status; share->lock.update_status=_ma_update_status; + share->lock.restore_status=_ma_restore_status; share->lock.check_status=_ma_check_status; } } diff --git a/storage/maria/ma_packrec.c b/storage/maria/ma_packrec.c index 7c875e7b91d..4e74653192e 100644 --- a/storage/maria/ma_packrec.c +++ b/storage/maria/ma_packrec.c @@ -20,7 +20,10 @@ #define IS_CHAR ((uint) 32768) /* Bit if char (not offset) in tree */ -#if INT_MAX > 65536L +/* Some definitions to keep in sync with maria_pack.c */ +#define HEAD_LENGTH 32 /* Length of fixed header */ + +#if INT_MAX > 32767 #define BITS_SAVED 32 #define MAX_QUICK_TABLE_BITS 9 /* Because we may shift in 24 bits */ #else @@ -42,6 +45,7 @@ { bits-=(bit+1); break; } \ pos+= *pos +/* Size in uint16 of a Huffman tree for byte compression of 256 byte values. */ #define OFFSET_TABLE_SIZE 512 static my_bool _ma_read_pack_info(MARIA_SHARE *share, File file, @@ -169,7 +173,7 @@ static my_bool _ma_read_pack_info(MARIA_SHARE *share, File file, uint16 *decode_table,*tmp_buff; ulong elements,intervall_length; char *disk_cache,*intervall_buff; - uchar header[32]; + uchar header[HEAD_LENGTH]; MARIA_BIT_BUFF bit_buff; DBUG_ENTER("_ma_read_pack_info"); @@ -185,12 +189,13 @@ static my_bool _ma_read_pack_info(MARIA_SHARE *share, File file, my_errno=HA_ERR_END_OF_FILE; goto err0; } + /* Only the first three bytes of magic number are independent of version. */ if (memcmp((byte*) header, (byte*) maria_pack_file_magic, 3)) { my_errno=HA_ERR_WRONG_IN_RECORD; goto err0; } - share->pack.version= header[3]; + share->pack.version= header[3]; /* fourth byte of magic number */ share->pack.header_length= uint4korr(header+4); share->min_pack_length=(uint) uint4korr(header+8); share->max_pack_length=(uint) uint4korr(header+12); @@ -206,7 +211,23 @@ static my_bool _ma_read_pack_info(MARIA_SHARE *share, File file, share->base.min_block_length=share->min_pack_length+1; if (share->min_pack_length > 254) share->base.min_block_length+=2; - + DBUG_PRINT("info", ("fixed header length: %u", HEAD_LENGTH)); + DBUG_PRINT("info", ("total header length: %lu", share->pack.header_length)); + DBUG_PRINT("info", ("pack file version: %u", share->pack.version)); + DBUG_PRINT("info", ("min pack length: %lu", share->min_pack_length)); + DBUG_PRINT("info", ("max pack length: %lu", share->max_pack_length)); + DBUG_PRINT("info", ("elements of all trees: %lu", elements)); + DBUG_PRINT("info", ("distinct values bytes: %lu", intervall_length)); + DBUG_PRINT("info", ("number of code trees: %u", trees)); + DBUG_PRINT("info", ("bytes for record lgt: %u", share->pack.ref_length)); + DBUG_PRINT("info", ("record pointer length: %u", rec_reflength)); + + + /* + Memory segment #1: + - Decode tree heads + - Distinct column values + */ if (!(share->decode_trees=(MARIA_DECODE_TREE*) my_malloc((uint) (trees*sizeof(MARIA_DECODE_TREE)+ intervall_length*sizeof(byte)), @@ -214,10 +235,18 @@ static my_bool _ma_read_pack_info(MARIA_SHARE *share, File file, goto err0; intervall_buff=(byte*) (share->decode_trees+trees); + /* + Memory segment #2: + - Decode tables + - Quick decode tables + - Temporary decode table + - Compressed data file header cache + This segment will be reallocated after construction of the tables. + */ length=(uint) (elements*2+trees*(1 << maria_quick_table_bits)); if (!(share->decode_tables=(uint16*) my_malloc((length+OFFSET_TABLE_SIZE)*sizeof(uint16)+ - (uint) (share->pack.header_length+7), + (uint) (share->pack.header_length - sizeof(header)), MYF(MY_WME | MY_ZEROFILL)))) goto err1; tmp_buff=share->decode_tables+length; @@ -240,17 +269,26 @@ static my_bool _ma_read_pack_info(MARIA_SHARE *share, File file, share->rec[i].huff_tree=share->decode_trees+(uint) get_bits(&bit_buff, huff_tree_bits); share->rec[i].unpack= get_unpack_function(share->rec+i); + DBUG_PRINT("info", ("col: %2u type: %2u pack: %u slbits: %2u", + i, share->rec[i].base_type, share->rec[i].pack_type, + share->rec[i].space_length_bits)); } skip_to_next_byte(&bit_buff); + /* + Construct the decoding tables from the file header. Keep track of + the used memory. + */ decode_table=share->decode_tables; for (i=0 ; i < trees ; i++) if (read_huff_table(&bit_buff,share->decode_trees+i,&decode_table, &intervall_buff,tmp_buff)) goto err3; + /* Reallocate the decoding tables to the used size. */ decode_table=(uint16*) my_realloc((gptr) share->decode_tables, (uint) ((byte*) decode_table - (byte*) share->decode_tables), MYF(MY_HOLD_ON_ERROR)); + /* Fix the table addresses in the tree heads. */ { long diff=PTR_BYTE_DIFF(decode_table,share->decode_tables); share->decode_tables=decode_table; @@ -296,8 +334,23 @@ err0: } - /* Read on huff-code-table from datafile */ +/* + Read a huff-code-table from datafile. + SYNOPSIS + read_huff_table() + bit_buff Bit buffer pointing at start of the + decoding table in the file header cache. + decode_tree Pointer to the decode tree head. + decode_table IN/OUT Address of a pointer to the next free space. + intervall_buff IN/OUT Address of a pointer to the next unused values. + tmp_buff Buffer for temporary extraction of a full + decoding table as read from bit_buff. + + RETURN + 0 OK. + 1 Error. +*/ static uint read_huff_table(MARIA_BIT_BUFF *bit_buff, MARIA_DECODE_TREE *decode_tree, uint16 **decode_table, byte **intervall_buff, @@ -306,19 +359,33 @@ static uint read_huff_table(MARIA_BIT_BUFF *bit_buff, uint min_chr,elements,char_bits,offset_bits,size,intervall_length,table_bits, next_free_offset; uint16 *ptr,*end; + DBUG_ENTER("read_huff_table"); - LINT_INIT(ptr); if (!get_bits(bit_buff,1)) { + /* Byte value compression. */ min_chr=get_bits(bit_buff,8); elements=get_bits(bit_buff,9); char_bits=get_bits(bit_buff,5); offset_bits=get_bits(bit_buff,5); intervall_length=0; ptr=tmp_buff; + ptr=tmp_buff; + DBUG_PRINT("info", ("byte value compression")); + DBUG_PRINT("info", ("minimum byte value: %u", min_chr)); + DBUG_PRINT("info", ("number of tree nodes: %u", elements)); + DBUG_PRINT("info", ("bits for values: %u", char_bits)); + DBUG_PRINT("info", ("bits for tree offsets: %u", offset_bits)); + if (elements > 256) + { + DBUG_PRINT("error", ("ERROR: illegal number of tree elements: %u", + elements)); + DBUG_RETURN(1); + } } else { + /* Distinct column value compression. */ min_chr=0; elements=get_bits(bit_buff,15); intervall_length=get_bits(bit_buff,16); @@ -326,13 +393,28 @@ static uint read_huff_table(MARIA_BIT_BUFF *bit_buff, offset_bits=get_bits(bit_buff,5); decode_tree->quick_table_bits=0; ptr= *decode_table; + DBUG_PRINT("info", ("distinct column value compression")); + DBUG_PRINT("info", ("number of tree nodes: %u", elements)); + DBUG_PRINT("info", ("value buffer length: %u", intervall_length)); + DBUG_PRINT("info", ("bits for value index: %u", char_bits)); + DBUG_PRINT("info", ("bits for tree offsets: %u", offset_bits)); } size=elements*2-2; + DBUG_PRINT("info", ("tree size in uint16: %u", size)); + DBUG_PRINT("info", ("tree size in bytes: %u", + size * (uint) sizeof(uint16))); for (end=ptr+size ; ptr < end ; ptr++) { if (get_bit(bit_buff)) + { *ptr= (uint16) get_bits(bit_buff,offset_bits); + if ((ptr + *ptr >= end) || !*ptr) + { + DBUG_PRINT("error", ("ERROR: illegal pointer in decode tree")); + DBUG_RETURN(1); + } + } else *ptr= (uint16) (IS_CHAR + (get_bits(bit_buff,char_bits) + min_chr)); } @@ -342,11 +424,15 @@ static uint read_huff_table(MARIA_BIT_BUFF *bit_buff, decode_tree->intervalls= *intervall_buff; if (! intervall_length) { - table_bits=find_longest_bitstream(tmp_buff, tmp_buff+OFFSET_TABLE_SIZE); - if (table_bits == (uint) ~0) - return 1; + /* Byte value compression. ptr started from tmp_buff. */ + /* Find longest Huffman code from begin to end of tree in bits. */ + table_bits= find_longest_bitstream(tmp_buff, ptr); + if (table_bits >= OFFSET_TABLE_SIZE) + DBUG_RETURN(1); if (table_bits > maria_quick_table_bits) table_bits=maria_quick_table_bits; + DBUG_PRINT("info", ("table bits: %u", table_bits)); + next_free_offset= (1 << table_bits); make_quick_table(*decode_table,tmp_buff,&next_free_offset,0,table_bits, table_bits); @@ -355,96 +441,265 @@ static uint read_huff_table(MARIA_BIT_BUFF *bit_buff, } else { + /* Distinct column value compression. ptr started from *decode_table */ (*decode_table)=end; + /* + get_bits() moves some bytes to a cache buffer in advance. May need + to step back. + */ bit_buff->pos-= bit_buff->bits/8; + /* Copy the distinct column values from the buffer. */ memcpy(*intervall_buff,bit_buff->pos,(size_t) intervall_length); (*intervall_buff)+=intervall_length; bit_buff->pos+=intervall_length; bit_buff->bits=0; } - return 0; + DBUG_RETURN(0); } +/* + Make a quick_table for faster decoding. + + SYNOPSIS + make_quick_table() + to_table Target quick_table and remaining decode table. + decode_table Source Huffman (sub-)tree within tmp_buff. + next_free_offset IN/OUT Next free offset from to_table. + Starts behind quick_table on the top-level. + value Huffman bits found so far. + bits Remaining bits to be collected. + max_bits Total number of bits to collect (table_bits). + + DESCRIPTION + + The quick table is an array of 16-bit values. There exists one value + for each possible code representable by max_bits (table_bits) bits. + In most cases table_bits is 9. So there are 512 16-bit values. + + If the high-order bit (16) is set (IS_CHAR) then the array slot for + this value is a valid Huffman code for a resulting byte value. + + The low-order 8 bits (1..8) are the resulting byte value. + + Bits 9..14 are the length of the Huffman code for this byte value. + This means so many bits from the input stream were needed to + represent this byte value. The remaining bits belong to later + Huffman codes. This also means that for every Huffman code shorter + than table_bits there are multiple entires in the array, which + differ just in the unused bits. + + If the high-order bit (16) is clear (0) then the remaining bits are + the position of the remaining Huffman decode tree segment behind the + quick table. + + RETURN + void +*/ + static void make_quick_table(uint16 *to_table, uint16 *decode_table, uint *next_free_offset, uint value, uint bits, uint max_bits) { + DBUG_ENTER("make_quick_table"); + + /* + When down the table to the requested maximum, copy the rest of the + Huffman table. + */ if (!bits--) { + /* + Remaining left Huffman tree segment starts behind quick table. + Remaining right Huffman tree segment starts behind left segment. + */ to_table[value]= (uint16) *next_free_offset; + /* + Re-construct the remaining Huffman tree segment at + next_free_offset in to_table. + */ *next_free_offset=copy_decode_table(to_table, *next_free_offset, decode_table); - return; + DBUG_VOID_RETURN; } + + /* Descent on the left side. Left side bits are clear (0). */ if (!(*decode_table & IS_CHAR)) { + /* Not a leaf. Follow the pointer. */ make_quick_table(to_table,decode_table+ *decode_table, next_free_offset,value,bits,max_bits); } else + { + /* + A leaf. A Huffman code is complete. Fill the quick_table + array for all possible bit strings starting with this Huffman + code. + */ fill_quick_table(to_table+value,bits,max_bits,(uint) *decode_table); + } + + /* Descent on the right side. Right side bits are set (1). */ decode_table++; value|= (1 << bits); if (!(*decode_table & IS_CHAR)) { + /* Not a leaf. Follow the pointer. */ make_quick_table(to_table,decode_table+ *decode_table, next_free_offset,value,bits,max_bits); } else + { + /* + A leaf. A Huffman code is complete. Fill the quick_table + array for all possible bit strings starting with this Huffman + code. + */ fill_quick_table(to_table+value,bits,max_bits,(uint) *decode_table); - return; + } + + DBUG_VOID_RETURN; } +/* + Fill quick_table for all possible values starting with this Huffman code. + + SYNOPSIS + fill_quick_table() + table Target quick_table position. + bits Unused bits from max_bits. + max_bits Total number of bits to collect (table_bits). + value The byte encoded by the found Huffman code. + + DESCRIPTION + + Fill the segment (all slots) of the quick_table array with the + resulting value for the found Huffman code. There are as many slots + as there are combinations representable by the unused bits. + + In most cases we use 9 table bits. Assume a 3-bit Huffman code. Then + there are 6 unused bits. Hence we fill 2**6 = 64 slots with the + value. + + RETURN + void +*/ + static void fill_quick_table(uint16 *table, uint bits, uint max_bits, uint value) { uint16 *end; - value|=(max_bits-bits) << 8; - for (end=table+ (1 << bits) ; - table < end ; - *table++ = (uint16) value | IS_CHAR) ; + DBUG_ENTER("fill_quick_table"); + + /* + Bits 1..8 of value represent the decoded byte value. + Bits 9..14 become the length of the Huffman code for this byte value. + Bit 16 flags a valid code (IS_CHAR). + */ + value|= (max_bits - bits) << 8 | IS_CHAR; + + for (end= table + (uint) (((uint) 1 << bits)); table < end; table++) + { + *table= (uint16) value; + } + DBUG_VOID_RETURN; } +/* + Reconstruct a decode subtree at the target position. + + SYNOPSIS + copy_decode_table() + to_pos Target quick_table and remaining decode table. + offset Next free offset from to_pos. + decode_table Source Huffman subtree within tmp_buff. + + NOTE + Pointers in the decode tree are relative to the pointers position. + + RETURN + next free offset from to_pos. +*/ + static uint copy_decode_table(uint16 *to_pos, uint offset, uint16 *decode_table) { - uint prev_offset; - prev_offset= offset; + uint prev_offset= offset; + DBUG_ENTER("copy_decode_table"); + /* Descent on the left side. */ if (!(*decode_table & IS_CHAR)) { + /* Set a pointer to the next target node. */ to_pos[offset]=2; + /* Copy the left hand subtree there. */ offset=copy_decode_table(to_pos,offset+2,decode_table+ *decode_table); } else { + /* Copy the byte value. */ to_pos[offset]= *decode_table; + /* Step behind this node. */ offset+=2; } - decode_table++; + /* Descent on the right side. */ + decode_table++; if (!(*decode_table & IS_CHAR)) { + /* Set a pointer to the next free target node. */ to_pos[prev_offset+1]=(uint16) (offset-prev_offset-1); + /* Copy the right hand subtree to the entry of that node. */ offset=copy_decode_table(to_pos,offset,decode_table+ *decode_table); } else + { + /* Copy the byte value. */ to_pos[prev_offset+1]= *decode_table; - return offset; + } + DBUG_RETURN(offset); } +/* + Find the length of the longest Huffman code in this table in bits. + + SYNOPSIS + find_longest_bitstream() + table Code (sub-)table start. + end End of code table. + + IMPLEMENTATION + + Recursively follow the branch(es) of the code pair on every level of + the tree until two byte values (and no branch) are found. Add one to + each level when returning back from each recursion stage. + + 'end' is used for error checking only. A clean tree terminates + before reaching 'end'. Hence the exact value of 'end' is not too + important. However having it higher than necessary could lead to + misbehaviour should 'next' jump into the dirty area. + + RETURN + length Length of longest Huffman code in bits. + >= OFFSET_TABLE_SIZE Error, broken tree. It does not end before 'end'. +*/ + static uint find_longest_bitstream(uint16 *table, uint16 *end) { - uint length=1,length2; + uint length=1; + uint length2; if (!(*table & IS_CHAR)) { uint16 *next= table + *table; if (next > end || next == table) - return ~0; + { + DBUG_PRINT("error", ("ERROR: illegal pointer in decode tree")); + return OFFSET_TABLE_SIZE; + } length=find_longest_bitstream(next, end)+1; } table++; @@ -452,8 +707,11 @@ static uint find_longest_bitstream(uint16 *table, uint16 *end) { uint16 *next= table + *table; if (next > end || next == table) - return ~0; - length2=find_longest_bitstream(table+ *table, end)+1; + { + DBUG_PRINT("error", ("ERROR: illegal pointer in decode tree")); + return OFFSET_TABLE_SIZE; + } + length2= find_longest_bitstream(next, end) + 1; length=max(length,length2); } return length; @@ -901,18 +1159,46 @@ static void decode_bytes(MARIA_COLUMNDEF *rec,MARIA_BIT_BUFF *bit_buff, bit_buff->pos+=4; bits+=32; } - /* First use info in quick_table */ + /* + First use info in quick_table. + + The quick table is an array of 16-bit values. There exists one + value for each possible code representable by table_bits bits. + In most cases table_bits is 9. So there are 512 16-bit values. + + If the high-order bit (16) is set (IS_CHAR) then the array slot + for this value is a valid Huffman code for a resulting byte value. + + The low-order 8 bits (1..8) are the resulting byte value. + + Bits 9..14 are the length of the Huffman code for this byte value. + This means so many bits from the input stream were needed to + represent this byte value. The remaining bits belong to later + Huffman codes. This also means that for every Huffman code shorter + than table_bits there are multiple entires in the array, which + differ just in the unused bits. + + If the high-order bit (16) is clear (0) then the remaining bits are + the position of the remaining Huffman decode tree segment behind the + quick table. + */ low_byte=(uint) (bit_buff->current_byte >> (bits - table_bits)) & table_and; low_byte=decode_tree->table[low_byte]; if (low_byte & IS_CHAR) { + /* + All Huffman codes of less or equal table_bits length are in the + quick table. This is one of them. + */ *to++ = (char) (low_byte & 255); /* Found char in quick table */ bits-= ((low_byte >> 8) & 31); /* Remove bits used */ } else { /* Map through rest of decode-table */ + /* This means that the Huffman code must be longer than table_bits. */ pos=decode_tree->table+low_byte; bits-=table_bits; + /* NOTE: decode_bytes_test_bit() is a macro wich contains a break !!! */ for (;;) { low_byte=(uint) (bit_buff->current_byte >> (bits-8)); @@ -1140,6 +1426,11 @@ uint _ma_pack_get_block_info(MARIA_HA *maria, MARIA_BIT_BUFF *bit_buff, { head_length+= read_pack_length((uint) maria->s->pack.version, header + head_length, &info->blob_len); + /* + Ensure that the record buffer is big enough for the compressed + record plus all expanded blobs. [We do not have an extra buffer + for the resulting blobs. Sigh.] + */ if (_ma_alloc_buffer(rec_buff_p, rec_buff_size_p, info->rec_len + info->blob_len + maria->s->base.extra_rec_buff_size)) diff --git a/storage/maria/ma_range.c b/storage/maria/ma_range.c index d3806aa79bc..4c5a037eb67 100644 --- a/storage/maria/ma_range.c +++ b/storage/maria/ma_range.c @@ -170,6 +170,7 @@ static double _ma_search_pos(register MARIA_HA *info, byte *keypos, *buff; double offset; DBUG_ENTER("_ma_search_pos"); + LINT_INIT(max_keynr); if (pos == HA_OFFSET_ERROR) DBUG_RETURN(0.5); diff --git a/storage/maria/ma_rename.c b/storage/maria/ma_rename.c index 5d89cc063d7..59a2315b18b 100644 --- a/storage/maria/ma_rename.c +++ b/storage/maria/ma_rename.c @@ -33,7 +33,7 @@ int maria_rename(const char *old_name, const char *new_name) _ma_check_table_is_closed(old_name,"rename old_table"); _ma_check_table_is_closed(new_name,"rename new table2"); #endif - /* LOCKTODO take X-lock on table here */ + /* LOCK TODO take X-lock on table here */ #ifdef USE_RAID { MARIA_HA *info; @@ -51,7 +51,7 @@ int maria_rename(const char *old_name, const char *new_name) fn_format(from,old_name,"",MARIA_NAME_IEXT,MY_UNPACK_FILENAME|MY_APPEND_EXT); fn_format(to,new_name,"",MARIA_NAME_IEXT,MY_UNPACK_FILENAME|MY_APPEND_EXT); /* - RECOVERYTODO log the two renames below. Update + RECOVERY TODO log the two renames below. Update ZeroDirtyPagesLSN of the table on disk (=> sync the files), this is needed so that Recovery does not pick a wrong table. Then do the file renames. diff --git a/storage/maria/ma_rt_index.c b/storage/maria/ma_rt_index.c index 8e8ec6c991b..cefbcffcbf9 100644 --- a/storage/maria/ma_rt_index.c +++ b/storage/maria/ma_rt_index.c @@ -637,8 +637,6 @@ static int maria_rtree_insert_level(MARIA_HA *info, uint keynr, byte *key, if ((old_root = info->s->state.key_root[keynr]) == HA_OFFSET_ERROR) { - int res; - if ((old_root = _ma_new(info, keyinfo, DFLT_INIT_HITS)) == HA_OFFSET_ERROR) return -1; info->keybuff_used = 1; @@ -929,7 +927,6 @@ int maria_rtree_delete(MARIA_HA *info, uint keynr, byte *key, uint key_length) ulong i; for (i = 0; i < ReinsertList.n_pages; ++i) { - uint nod_flag; byte *page_buf, *k, *last; if (!(page_buf = (byte*) my_alloca((uint)keyinfo->block_length))) diff --git a/storage/maria/ma_rt_split.c b/storage/maria/ma_rt_split.c index 00c8d18f5e5..a81b2b932ec 100644 --- a/storage/maria/ma_rt_split.c +++ b/storage/maria/ma_rt_split.c @@ -188,6 +188,10 @@ static int split_maria_rtree_node(SplitStruct *node, int n_entries, int next_node; int i; SplitStruct *end = node + n_entries; + LINT_INIT(a); + LINT_INIT(b); + LINT_INIT(next); + LINT_INIT(next_node); if (all_size < min_size * 2) { diff --git a/storage/maria/ma_search.c b/storage/maria/ma_search.c index d8738ae4639..991115ad181 100644 --- a/storage/maria/ma_search.c +++ b/storage/maria/ma_search.c @@ -477,9 +477,9 @@ int _ma_prefix_search(MARIA_HA *info, register MARIA_KEYDEF *keyinfo, else { /* We have to compare k and vseg as if they were space extended */ - uchar *end= k+ (cmplen - len); - for ( ; k < end && *k == ' '; k++) ; - if (k == end) + uchar *k_end= k+ (cmplen - len); + for ( ; k < k_end && *k == ' '; k++) ; + if (k == k_end) goto cmp_rest; /* should never happen */ if ((uchar) *k < (uchar) ' ') { @@ -491,15 +491,15 @@ int _ma_prefix_search(MARIA_HA *info, register MARIA_KEYDEF *keyinfo, } else if (len > cmplen) { - uchar *end; + uchar *vseg_end; if ((nextflag & SEARCH_PREFIX) && key_len_left == 0) goto fix_flag; /* We have to compare k and vseg as if they were space extended */ - for (end=vseg + (len-cmplen) ; - vseg < end && *vseg == (byte) ' '; + for (vseg_end= vseg + (len-cmplen) ; + vseg < vseg_end && *vseg == (byte) ' '; vseg++, matched++) ; - DBUG_ASSERT(vseg < end); + DBUG_ASSERT(vseg < vseg_end); if ((uchar) *vseg > (uchar) ' ') { diff --git a/storage/maria/ma_sort.c b/storage/maria/ma_sort.c index 9134645f847..2c5524906ad 100644 --- a/storage/maria/ma_sort.c +++ b/storage/maria/ma_sort.c @@ -221,9 +221,9 @@ int _ma_create_index_by_sort(MARIA_SORT_PARAM *info, my_bool no_messages, if (my_b_inited(&tempfile_for_exceptions)) { - MARIA_HA *index=info->sort_info->info; + MARIA_HA *idx=info->sort_info->info; uint keyno=info->key; - uint key_length, ref_length=index->s->rec_reflength; + uint key_length, ref_length=idx->s->rec_reflength; if (!no_messages) printf(" - Adding exceptions\n"); /* purecov: tested */ @@ -236,7 +236,7 @@ int _ma_create_index_by_sort(MARIA_SORT_PARAM *info, my_bool no_messages, && !my_b_read(&tempfile_for_exceptions,(byte*)sort_keys, (uint) key_length)) { - if (_ma_ck_write(index,keyno,(byte*) sort_keys,key_length-ref_length)) + if (_ma_ck_write(idx,keyno,(byte*) sort_keys,key_length-ref_length)) goto err; } } diff --git a/storage/maria/ma_update.c b/storage/maria/ma_update.c index e3e391dcccc..29018b47bda 100644 --- a/storage/maria/ma_update.c +++ b/storage/maria/ma_update.c @@ -206,7 +206,8 @@ int maria_update(register MARIA_HA *info, const byte *oldrec, byte *newrec) err: DBUG_PRINT("error",("key: %d errno: %d",i,my_errno)); save_errno=my_errno; - if (my_errno == HA_ERR_FOUND_DUPP_KEY || my_errno == HA_ERR_RECORD_FILE_FULL) + if (my_errno == HA_ERR_FOUND_DUPP_KEY || my_errno == HA_ERR_OUT_OF_MEM || + my_errno == HA_ERR_RECORD_FILE_FULL) { info->errkey= (int) i; flag=0; diff --git a/storage/maria/ma_write.c b/storage/maria/ma_write.c index 3f1ca59a00a..b672ed1fe23 100644 --- a/storage/maria/ma_write.c +++ b/storage/maria/ma_write.c @@ -205,7 +205,8 @@ err: fatal_error= 0; if (my_errno == HA_ERR_FOUND_DUPP_KEY || my_errno == HA_ERR_RECORD_FILE_FULL || - my_errno == HA_ERR_NULL_IN_SPATIAL) + my_errno == HA_ERR_NULL_IN_SPATIAL || + my_errno == HA_ERR_OUT_OF_MEM) { if (info->bulk_insert) { @@ -629,6 +630,7 @@ int _ma_split_page(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, my_off_t new_pos; MARIA_KEY_PARAM s_temp; DBUG_ENTER("maria_split_page"); + LINT_INIT(after_key); DBUG_DUMP("buff",(byte*) buff,maria_getint(buff)); if (info->s->keyinfo+info->lastinx == keyinfo) diff --git a/storage/maria/maria_chk.c b/storage/maria/maria_chk.c index 8471d36bec9..533766e35f9 100644 --- a/storage/maria/maria_chk.c +++ b/storage/maria/maria_chk.c @@ -727,6 +727,7 @@ get_one_option(int optid, case 2: method_conv= MI_STATS_METHOD_IGNORE_NULLS; break; + default: assert(0); /* Impossible */ } check_param.stats_method= method_conv; break; diff --git a/storage/maria/maria_def.h b/storage/maria/maria_def.h index b174df2fa3e..f4684735938 100644 --- a/storage/maria/maria_def.h +++ b/storage/maria/maria_def.h @@ -817,6 +817,7 @@ my_bool _ma_unique_comp(MARIA_UNIQUEDEF *def, const byte *a, const byte *b, my_bool null_are_equal); void _ma_get_status(void *param, int concurrent_insert); void _ma_update_status(void *param); +void _ma_restore_status(void *param); void _ma_copy_status(void *to, void *from); my_bool _ma_check_status(void *param); diff --git a/storage/maria/maria_pack.c b/storage/maria/maria_pack.c index 41c519693d5..a76b1027179 100644 --- a/storage/maria/maria_pack.c +++ b/storage/maria/maria_pack.c @@ -2696,8 +2696,9 @@ static int compress_isam_file(PACK_MRG_INFO *mrg, HUFF_COUNTS *huff_counts) } case FIELD_VARCHAR: { - uint pack_length= HA_VARCHAR_PACKLENGTH(count->field_length-1); - ulong col_length= (pack_length == 1 ? (uint) *(uchar*) start_pos : + uint var_pack_length= HA_VARCHAR_PACKLENGTH(count->field_length-1); + ulong col_length= (var_pack_length == 1 ? + (uint) *(uchar*) start_pos : uint2korr(start_pos)); /* Empty varchar are encoded with a single 1 bit. */ if (!col_length) @@ -2707,7 +2708,7 @@ static int compress_isam_file(PACK_MRG_INFO *mrg, HUFF_COUNTS *huff_counts) } else { - byte *end=start_pos+pack_length+col_length; + byte *end= start_pos + var_pack_length + col_length; DBUG_PRINT("fields", ("FIELD_VARCHAR not empty, bits: 1")); write_bits(0,1); /* Write the varchar length. */ @@ -2715,7 +2716,7 @@ static int compress_isam_file(PACK_MRG_INFO *mrg, HUFF_COUNTS *huff_counts) col_length, count->length_bits)); write_bits(col_length,count->length_bits); /* Encode the varchar bytes. */ - for (start_pos+=pack_length ; start_pos < end ; start_pos++) + for (start_pos+= var_pack_length ; start_pos < end ; start_pos++) { DBUG_PRINT("fields", ("value: 0x%02x code: 0x%s bits: %2u bin: %s", diff --git a/storage/maria/unittest/ma_test_loghandler_pagecache-t.c b/storage/maria/unittest/ma_test_loghandler_pagecache-t.c index 211eb6dea9e..9204d531ea1 100644 --- a/storage/maria/unittest/ma_test_loghandler_pagecache-t.c +++ b/storage/maria/unittest/ma_test_loghandler_pagecache-t.c @@ -21,7 +21,7 @@ static PAGECACHE_FILE file1; int main(int argc, char *argv[]) { uint pagen; - uchar long_tr_id[6]; + byte long_tr_id[6]; PAGECACHE pagecache; LSN lsn; MY_STAT st, *stat; diff --git a/storage/myisam/ha_myisam.cc b/storage/myisam/ha_myisam.cc index 0113fe80a08..f0bee43c4c6 100644 --- a/storage/myisam/ha_myisam.cc +++ b/storage/myisam/ha_myisam.cc @@ -293,7 +293,7 @@ int table2myisam(TABLE *table_arg, MI_KEYDEF **keydef_out, Check for underlying table conformance SYNOPSIS - check_definition() + myisam_check_definition() t1_keyinfo in First table key definition t1_recinfo in First table record definition t1_keys in Number of keys in first table @@ -323,13 +323,13 @@ int table2myisam(TABLE *table_arg, MI_KEYDEF **keydef_out, 1 - Different definitions. */ -int check_definition(MI_KEYDEF *t1_keyinfo, MI_COLUMNDEF *t1_recinfo, - uint t1_keys, uint t1_recs, - MI_KEYDEF *t2_keyinfo, MI_COLUMNDEF *t2_recinfo, - uint t2_keys, uint t2_recs, bool strict) +int myisam_check_definition(MI_KEYDEF *t1_keyinfo, MI_COLUMNDEF *t1_recinfo, + uint t1_keys, uint t1_recs, + MI_KEYDEF *t2_keyinfo, MI_COLUMNDEF *t2_recinfo, + uint t2_keys, uint t2_recs, bool strict) { uint i, j; - DBUG_ENTER("check_definition"); + DBUG_ENTER("myisam_check_definition"); if ((strict ? t1_keys != t2_keys : t1_keys > t2_keys)) { DBUG_PRINT("error", ("Number of keys differs: t1_keys=%u, t2_keys=%u", diff --git a/storage/myisammrg/ha_myisammrg.cc b/storage/myisammrg/ha_myisammrg.cc index d9e7e1d5700..73f84e307c1 100644 --- a/storage/myisammrg/ha_myisammrg.cc +++ b/storage/myisammrg/ha_myisammrg.cc @@ -48,10 +48,12 @@ static const char *ha_myisammrg_exts[] = { }; extern int table2myisam(TABLE *table_arg, MI_KEYDEF **keydef_out, MI_COLUMNDEF **recinfo_out, uint *records_out); -extern int check_definition(MI_KEYDEF *t1_keyinfo, MI_COLUMNDEF *t1_recinfo, - uint t1_keys, uint t1_recs, - MI_KEYDEF *t2_keyinfo, MI_COLUMNDEF *t2_recinfo, - uint t2_keys, uint t2_recs, bool strict); +extern int myisam_check_definition(MI_KEYDEF *t1_keyinfo, + MI_COLUMNDEF *t1_recinfo, + uint t1_keys, uint t1_recs, + MI_KEYDEF *t2_keyinfo, + MI_COLUMNDEF *t2_recinfo, + uint t2_keys, uint t2_recs, bool strict); const char **ha_myisammrg::bas_ext() const { @@ -115,10 +117,10 @@ int ha_myisammrg::open(const char *name, int mode, uint test_if_locked) } for (u_table= file->open_tables; u_table < file->end_table; u_table++) { - if (check_definition(keyinfo, recinfo, keys, recs, - u_table->table->s->keyinfo, u_table->table->s->rec, - u_table->table->s->base.keys, - u_table->table->s->base.fields, false)) + if (myisam_check_definition(keyinfo, recinfo, keys, recs, + u_table->table->s->keyinfo, u_table->table->s->rec, + u_table->table->s->base.keys, + u_table->table->s->base.fields, false)) { my_free((gptr) recinfo, MYF(0)); error= HA_ERR_WRONG_MRG_TABLE_DEF; -- cgit v1.2.1 From a19644ceeefe80c5392810d932bf536ad6a4b23b Mon Sep 17 00:00:00 2001 From: unknown Date: Thu, 1 Mar 2007 22:36:46 +0100 Subject: adding files needed for building from the source tarball include/Makefile.am: needed for source tarball storage/maria/Makefile.am: needed for source tarball --- include/Makefile.am | 2 +- storage/maria/Makefile.am | 3 ++- 2 files changed, 3 insertions(+), 2 deletions(-) diff --git a/include/Makefile.am b/include/Makefile.am index f92110cb4ce..59f671dd04a 100644 --- a/include/Makefile.am +++ b/include/Makefile.am @@ -37,7 +37,7 @@ noinst_HEADERS = config-win.h config-netware.h lf.h my_bit.h \ mysql_version.h.in my_handler.h my_time.h decimal.h \ my_vle.h my_user.h my_atomic.h atomic/nolock.h \ atomic/rwlock.h atomic/x86-gcc.h atomic/x86-msvc.h \ - my_libwrap.h pagecache.h + my_libwrap.h pagecache.h wqueue.h # Remove built files and the symlinked directories CLEANFILES = $(BUILT_SOURCES) readline openssl diff --git a/storage/maria/Makefile.am b/storage/maria/Makefile.am index b6c713571d7..8f3de170e16 100644 --- a/storage/maria/Makefile.am +++ b/storage/maria/Makefile.am @@ -54,7 +54,8 @@ noinst_PROGRAMS = ma_test1 ma_test2 ma_test3 ma_rt_test ma_sp_test noinst_HEADERS = maria_def.h ma_rt_index.h ma_rt_key.h ma_rt_mbr.h \ ma_sp_defs.h ma_fulltext.h ma_ftdefs.h ma_ft_test1.h \ ma_ft_eval.h trnman.h lockman.h tablockman.h \ - ma_control_file.h ha_maria.h ma_blockrec.h + ma_control_file.h ha_maria.h ma_blockrec.h \ + ma_loghandler.h ma_loghandler_lsn.h ma_test1_DEPENDENCIES= $(LIBRARIES) ma_test1_LDADD= @CLIENT_EXTRA_LDFLAGS@ libmaria.a \ $(top_builddir)/storage/myisam/libmyisam.a \ -- cgit v1.2.1 From 46922b5125e1cdbc0a79e7ba161aaa5fc515ae6b Mon Sep 17 00:00:00 2001 From: unknown Date: Fri, 2 Mar 2007 11:20:23 +0100 Subject: GPL license update (same change as was done for all files in 5.1). storage/maria/Makefile.am: GPL license update storage/maria/ft_maria.c: GPL license update storage/maria/ha_maria.cc: GPL license update storage/maria/ha_maria.h: GPL license update storage/maria/lockman.c: GPL license update storage/maria/lockman.h: GPL license update storage/maria/ma_bitmap.c: GPL license update storage/maria/ma_blockrec.c: GPL license update storage/maria/ma_blockrec.h: GPL license update storage/maria/ma_cache.c: GPL license update storage/maria/ma_changed.c: GPL license update storage/maria/ma_check.c: GPL license update storage/maria/ma_checkpoint.c: GPL license update storage/maria/ma_checkpoint.h: GPL license update storage/maria/ma_checksum.c: GPL license update storage/maria/ma_close.c: GPL license update storage/maria/ma_control_file.c: GPL license update storage/maria/ma_control_file.h: GPL license update storage/maria/ma_create.c: GPL license update storage/maria/ma_dbug.c: GPL license update storage/maria/ma_delete.c: GPL license update storage/maria/ma_delete_all.c: GPL license update storage/maria/ma_delete_table.c: GPL license update storage/maria/ma_dynrec.c: GPL license update storage/maria/ma_extra.c: GPL license update storage/maria/ma_ft_boolean_search.c: GPL license update storage/maria/ma_ft_eval.c: GPL license update storage/maria/ma_ft_eval.h: GPL license update storage/maria/ma_ft_nlq_search.c: GPL license update storage/maria/ma_ft_parser.c: GPL license update storage/maria/ma_ft_stem.c: GPL license update storage/maria/ma_ft_test1.c: GPL license update storage/maria/ma_ft_test1.h: GPL license update storage/maria/ma_ft_update.c: GPL license update storage/maria/ma_ftdefs.h: GPL license update storage/maria/ma_fulltext.h: GPL license update storage/maria/ma_info.c: GPL license update storage/maria/ma_init.c: GPL license update storage/maria/ma_key.c: GPL license update storage/maria/ma_keycache.c: GPL license update storage/maria/ma_least_recently_dirtied.c: GPL license update storage/maria/ma_least_recently_dirtied.h: GPL license update storage/maria/ma_locking.c: GPL license update storage/maria/ma_open.c: GPL license update storage/maria/ma_packrec.c: GPL license update storage/maria/ma_page.c: GPL license update storage/maria/ma_panic.c: GPL license update storage/maria/ma_preload.c: GPL license update storage/maria/ma_range.c: GPL license update storage/maria/ma_recovery.c: GPL license update storage/maria/ma_recovery.h: GPL license update storage/maria/ma_rename.c: GPL license update storage/maria/ma_rfirst.c: GPL license update storage/maria/ma_rkey.c: GPL license update storage/maria/ma_rlast.c: GPL license update storage/maria/ma_rnext.c: GPL license update storage/maria/ma_rnext_same.c: GPL license update storage/maria/ma_rprev.c: GPL license update storage/maria/ma_rrnd.c: GPL license update storage/maria/ma_rsame.c: GPL license update storage/maria/ma_rsamepos.c: GPL license update storage/maria/ma_rt_index.c: GPL license update storage/maria/ma_rt_index.h: GPL license update storage/maria/ma_rt_key.c: GPL license update storage/maria/ma_rt_key.h: GPL license update storage/maria/ma_rt_mbr.c: GPL license update storage/maria/ma_rt_mbr.h: GPL license update storage/maria/ma_rt_split.c: GPL license update storage/maria/ma_rt_test.c: GPL license update storage/maria/ma_scan.c: GPL license update storage/maria/ma_search.c: GPL license update storage/maria/ma_sort.c: GPL license update storage/maria/ma_sp_defs.h: GPL license update storage/maria/ma_sp_key.c: GPL license update storage/maria/ma_sp_test.c: GPL license update storage/maria/ma_static.c: GPL license update storage/maria/ma_statrec.c: GPL license update storage/maria/ma_test1.c: GPL license update storage/maria/ma_test2.c: GPL license update storage/maria/ma_test3.c: GPL license update storage/maria/ma_unique.c: GPL license update storage/maria/ma_update.c: GPL license update storage/maria/ma_write.c: GPL license update storage/maria/maria_chk.c: GPL license update storage/maria/maria_def.h: GPL license update storage/maria/maria_ftdump.c: GPL license update storage/maria/maria_pack.c: GPL license update storage/maria/tablockman.c: GPL license update storage/maria/tablockman.h: GPL license update storage/maria/trnman.c: GPL license update storage/maria/trnman.h: GPL license update --- storage/maria/Makefile.am | 3 +-- storage/maria/ft_maria.c | 3 +-- storage/maria/ha_maria.cc | 3 +-- storage/maria/ha_maria.h | 3 +-- storage/maria/lockman.c | 3 +-- storage/maria/lockman.h | 3 +-- storage/maria/ma_bitmap.c | 3 +-- storage/maria/ma_blockrec.c | 3 +-- storage/maria/ma_blockrec.h | 3 +-- storage/maria/ma_cache.c | 3 +-- storage/maria/ma_changed.c | 3 +-- storage/maria/ma_check.c | 3 +-- storage/maria/ma_checkpoint.c | 3 +-- storage/maria/ma_checkpoint.h | 3 +-- storage/maria/ma_checksum.c | 3 +-- storage/maria/ma_close.c | 3 +-- storage/maria/ma_control_file.c | 3 +-- storage/maria/ma_control_file.h | 3 +-- storage/maria/ma_create.c | 3 +-- storage/maria/ma_dbug.c | 3 +-- storage/maria/ma_delete.c | 3 +-- storage/maria/ma_delete_all.c | 3 +-- storage/maria/ma_delete_table.c | 3 +-- storage/maria/ma_dynrec.c | 3 +-- storage/maria/ma_extra.c | 3 +-- storage/maria/ma_ft_boolean_search.c | 3 +-- storage/maria/ma_ft_eval.c | 3 +-- storage/maria/ma_ft_eval.h | 3 +-- storage/maria/ma_ft_nlq_search.c | 3 +-- storage/maria/ma_ft_parser.c | 3 +-- storage/maria/ma_ft_stem.c | 3 +-- storage/maria/ma_ft_test1.c | 3 +-- storage/maria/ma_ft_test1.h | 3 +-- storage/maria/ma_ft_update.c | 3 +-- storage/maria/ma_ftdefs.h | 3 +-- storage/maria/ma_fulltext.h | 3 +-- storage/maria/ma_info.c | 3 +-- storage/maria/ma_init.c | 3 +-- storage/maria/ma_key.c | 3 +-- storage/maria/ma_keycache.c | 3 +-- storage/maria/ma_least_recently_dirtied.c | 3 +-- storage/maria/ma_least_recently_dirtied.h | 3 +-- storage/maria/ma_locking.c | 3 +-- storage/maria/ma_open.c | 3 +-- storage/maria/ma_packrec.c | 3 +-- storage/maria/ma_page.c | 3 +-- storage/maria/ma_panic.c | 3 +-- storage/maria/ma_preload.c | 3 +-- storage/maria/ma_range.c | 3 +-- storage/maria/ma_recovery.c | 3 +-- storage/maria/ma_recovery.h | 3 +-- storage/maria/ma_rename.c | 3 +-- storage/maria/ma_rfirst.c | 3 +-- storage/maria/ma_rkey.c | 3 +-- storage/maria/ma_rlast.c | 3 +-- storage/maria/ma_rnext.c | 3 +-- storage/maria/ma_rnext_same.c | 3 +-- storage/maria/ma_rprev.c | 3 +-- storage/maria/ma_rrnd.c | 3 +-- storage/maria/ma_rsame.c | 3 +-- storage/maria/ma_rsamepos.c | 3 +-- storage/maria/ma_rt_index.c | 3 +-- storage/maria/ma_rt_index.h | 3 +-- storage/maria/ma_rt_key.c | 3 +-- storage/maria/ma_rt_key.h | 3 +-- storage/maria/ma_rt_mbr.c | 3 +-- storage/maria/ma_rt_mbr.h | 3 +-- storage/maria/ma_rt_split.c | 3 +-- storage/maria/ma_rt_test.c | 3 +-- storage/maria/ma_scan.c | 3 +-- storage/maria/ma_search.c | 3 +-- storage/maria/ma_sort.c | 3 +-- storage/maria/ma_sp_defs.h | 3 +-- storage/maria/ma_sp_key.c | 3 +-- storage/maria/ma_sp_test.c | 3 +-- storage/maria/ma_static.c | 3 +-- storage/maria/ma_statrec.c | 3 +-- storage/maria/ma_test1.c | 3 +-- storage/maria/ma_test2.c | 3 +-- storage/maria/ma_test3.c | 3 +-- storage/maria/ma_unique.c | 3 +-- storage/maria/ma_update.c | 3 +-- storage/maria/ma_write.c | 3 +-- storage/maria/maria_chk.c | 3 +-- storage/maria/maria_def.h | 3 +-- storage/maria/maria_ftdump.c | 3 +-- storage/maria/maria_pack.c | 3 +-- storage/maria/tablockman.c | 3 +-- storage/maria/tablockman.h | 3 +-- storage/maria/trnman.c | 3 +-- storage/maria/trnman.h | 3 +-- 91 files changed, 91 insertions(+), 182 deletions(-) diff --git a/storage/maria/Makefile.am b/storage/maria/Makefile.am index 8f3de170e16..78c711693d3 100644 --- a/storage/maria/Makefile.am +++ b/storage/maria/Makefile.am @@ -2,8 +2,7 @@ # # 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; either version 2 of the License, or -# (at your option) any later version. +# 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 diff --git a/storage/maria/ft_maria.c b/storage/maria/ft_maria.c index 7104c6704ba..06e7c4bd59c 100644 --- a/storage/maria/ft_maria.c +++ b/storage/maria/ft_maria.c @@ -2,8 +2,7 @@ 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; either version 2 of the License, or - (at your option) any later version. + 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 diff --git a/storage/maria/ha_maria.cc b/storage/maria/ha_maria.cc index a5e0722fd4a..5ffd991d58a 100644 --- a/storage/maria/ha_maria.cc +++ b/storage/maria/ha_maria.cc @@ -2,8 +2,7 @@ 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; either version 2 of the License, or - (at your option) any later version. + 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 diff --git a/storage/maria/ha_maria.h b/storage/maria/ha_maria.h index 1f243f9ec59..e856189829e 100644 --- a/storage/maria/ha_maria.h +++ b/storage/maria/ha_maria.h @@ -2,8 +2,7 @@ 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; either version 2 of the License, or - (at your option) any later version. + 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 diff --git a/storage/maria/lockman.c b/storage/maria/lockman.c index fdacda84875..cb305dc9bd6 100644 --- a/storage/maria/lockman.c +++ b/storage/maria/lockman.c @@ -6,8 +6,7 @@ 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; either version 2 of the License, or - (at your option) any later version. + 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 diff --git a/storage/maria/lockman.h b/storage/maria/lockman.h index fd96f8930d5..279a5537f76 100644 --- a/storage/maria/lockman.h +++ b/storage/maria/lockman.h @@ -2,8 +2,7 @@ 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; either version 2 of the License, or - (at your option) any later version. + 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 diff --git a/storage/maria/ma_bitmap.c b/storage/maria/ma_bitmap.c index 5ed5a776658..f5cbaefb6bc 100644 --- a/storage/maria/ma_bitmap.c +++ b/storage/maria/ma_bitmap.c @@ -2,8 +2,7 @@ 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; either version 2 of the License, or - (at your option) any later version. + 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 diff --git a/storage/maria/ma_blockrec.c b/storage/maria/ma_blockrec.c index f1345b2c2f3..94209b8e41c 100644 --- a/storage/maria/ma_blockrec.c +++ b/storage/maria/ma_blockrec.c @@ -2,8 +2,7 @@ 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; either version 2 of the License, or - (at your option) any later version. + 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 diff --git a/storage/maria/ma_blockrec.h b/storage/maria/ma_blockrec.h index ec99dbfcae2..bdb86eb717d 100644 --- a/storage/maria/ma_blockrec.h +++ b/storage/maria/ma_blockrec.h @@ -2,8 +2,7 @@ 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; either version 2 of the License, or - (at your option) any later version. + 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 diff --git a/storage/maria/ma_cache.c b/storage/maria/ma_cache.c index d6061c647ec..e8a4b20571b 100644 --- a/storage/maria/ma_cache.c +++ b/storage/maria/ma_cache.c @@ -2,8 +2,7 @@ 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; either version 2 of the License, or - (at your option) any later version. + 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 diff --git a/storage/maria/ma_changed.c b/storage/maria/ma_changed.c index 9e86212baa6..4d0964581f6 100644 --- a/storage/maria/ma_changed.c +++ b/storage/maria/ma_changed.c @@ -2,8 +2,7 @@ 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; either version 2 of the License, or - (at your option) any later version. + 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 diff --git a/storage/maria/ma_check.c b/storage/maria/ma_check.c index a247d36144c..f262917b706 100644 --- a/storage/maria/ma_check.c +++ b/storage/maria/ma_check.c @@ -2,8 +2,7 @@ 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; either version 2 of the License, or - (at your option) any later version. + 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 diff --git a/storage/maria/ma_checkpoint.c b/storage/maria/ma_checkpoint.c index 02d887f758a..ed5520f66bf 100644 --- a/storage/maria/ma_checkpoint.c +++ b/storage/maria/ma_checkpoint.c @@ -2,8 +2,7 @@ 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; either version 2 of the License, or - (at your option) any later version. + 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 diff --git a/storage/maria/ma_checkpoint.h b/storage/maria/ma_checkpoint.h index 1b8064fa755..1ce2ccb7012 100644 --- a/storage/maria/ma_checkpoint.h +++ b/storage/maria/ma_checkpoint.h @@ -2,8 +2,7 @@ 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; either version 2 of the License, or - (at your option) any later version. + 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 diff --git a/storage/maria/ma_checksum.c b/storage/maria/ma_checksum.c index 1b0f683fe63..140f500e64d 100644 --- a/storage/maria/ma_checksum.c +++ b/storage/maria/ma_checksum.c @@ -2,8 +2,7 @@ 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; either version 2 of the License, or - (at your option) any later version. + 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 diff --git a/storage/maria/ma_close.c b/storage/maria/ma_close.c index 508334fcf67..b38ce2a8cc3 100644 --- a/storage/maria/ma_close.c +++ b/storage/maria/ma_close.c @@ -2,8 +2,7 @@ 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; either version 2 of the License, or - (at your option) any later version. + 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 diff --git a/storage/maria/ma_control_file.c b/storage/maria/ma_control_file.c index 07eddb956a2..2caa0038df2 100644 --- a/storage/maria/ma_control_file.c +++ b/storage/maria/ma_control_file.c @@ -2,8 +2,7 @@ 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; either version 2 of the License, or - (at your option) any later version. + 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 diff --git a/storage/maria/ma_control_file.h b/storage/maria/ma_control_file.h index 616babc3bb2..159cd15b3d6 100644 --- a/storage/maria/ma_control_file.h +++ b/storage/maria/ma_control_file.h @@ -2,8 +2,7 @@ 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; either version 2 of the License, or - (at your option) any later version. + 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 diff --git a/storage/maria/ma_create.c b/storage/maria/ma_create.c index bd9382ff9b0..2e491448a71 100644 --- a/storage/maria/ma_create.c +++ b/storage/maria/ma_create.c @@ -2,8 +2,7 @@ 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; either version 2 of the License, or - (at your option) any later version. + 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 diff --git a/storage/maria/ma_dbug.c b/storage/maria/ma_dbug.c index 596bf18bfe5..10c570c5794 100644 --- a/storage/maria/ma_dbug.c +++ b/storage/maria/ma_dbug.c @@ -2,8 +2,7 @@ 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; either version 2 of the License, or - (at your option) any later version. + 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 diff --git a/storage/maria/ma_delete.c b/storage/maria/ma_delete.c index 9198989dcb7..b576eec5e5e 100644 --- a/storage/maria/ma_delete.c +++ b/storage/maria/ma_delete.c @@ -2,8 +2,7 @@ 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; either version 2 of the License, or - (at your option) any later version. + 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 diff --git a/storage/maria/ma_delete_all.c b/storage/maria/ma_delete_all.c index 428724ec313..5a5ec98fe06 100644 --- a/storage/maria/ma_delete_all.c +++ b/storage/maria/ma_delete_all.c @@ -2,8 +2,7 @@ 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; either version 2 of the License, or - (at your option) any later version. + 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 diff --git a/storage/maria/ma_delete_table.c b/storage/maria/ma_delete_table.c index 47ba56e031c..aafe7a1dee9 100644 --- a/storage/maria/ma_delete_table.c +++ b/storage/maria/ma_delete_table.c @@ -2,8 +2,7 @@ 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; either version 2 of the License, or - (at your option) any later version. + 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 diff --git a/storage/maria/ma_dynrec.c b/storage/maria/ma_dynrec.c index 52780ab4f6f..a3fd323d059 100644 --- a/storage/maria/ma_dynrec.c +++ b/storage/maria/ma_dynrec.c @@ -2,8 +2,7 @@ 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; either version 2 of the License, or - (at your option) any later version. + 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 diff --git a/storage/maria/ma_extra.c b/storage/maria/ma_extra.c index 59ae18d949a..d654d5b2656 100644 --- a/storage/maria/ma_extra.c +++ b/storage/maria/ma_extra.c @@ -2,8 +2,7 @@ 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; either version 2 of the License, or - (at your option) any later version. + 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 diff --git a/storage/maria/ma_ft_boolean_search.c b/storage/maria/ma_ft_boolean_search.c index af76f0858a7..6e95262fe84 100644 --- a/storage/maria/ma_ft_boolean_search.c +++ b/storage/maria/ma_ft_boolean_search.c @@ -2,8 +2,7 @@ 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; either version 2 of the License, or - (at your option) any later version. + 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 diff --git a/storage/maria/ma_ft_eval.c b/storage/maria/ma_ft_eval.c index fe4900aeb64..50584459b7d 100644 --- a/storage/maria/ma_ft_eval.c +++ b/storage/maria/ma_ft_eval.c @@ -1,8 +1,7 @@ /* Copyright (C) 2006 MySQL AB & MySQL Finland AB & TCX DataKonsult AB 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; either version 2 of the License, or - (at your option) any later version. + 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 diff --git a/storage/maria/ma_ft_eval.h b/storage/maria/ma_ft_eval.h index d9b5c51642c..481943dfb0b 100644 --- a/storage/maria/ma_ft_eval.h +++ b/storage/maria/ma_ft_eval.h @@ -2,8 +2,7 @@ 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; either version 2 of the License, or - (at your option) any later version. + 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 diff --git a/storage/maria/ma_ft_nlq_search.c b/storage/maria/ma_ft_nlq_search.c index 4c922516455..145b7891dd1 100644 --- a/storage/maria/ma_ft_nlq_search.c +++ b/storage/maria/ma_ft_nlq_search.c @@ -2,8 +2,7 @@ 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; either version 2 of the License, or - (at your option) any later version. + 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 diff --git a/storage/maria/ma_ft_parser.c b/storage/maria/ma_ft_parser.c index 24713c1344f..f41b53bf3f7 100644 --- a/storage/maria/ma_ft_parser.c +++ b/storage/maria/ma_ft_parser.c @@ -2,8 +2,7 @@ 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; either version 2 of the License, or - (at your option) any later version. + 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 diff --git a/storage/maria/ma_ft_stem.c b/storage/maria/ma_ft_stem.c index 7a2f8cfd7c5..06fc0b2df6c 100644 --- a/storage/maria/ma_ft_stem.c +++ b/storage/maria/ma_ft_stem.c @@ -2,8 +2,7 @@ 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; either version 2 of the License, or - (at your option) any later version. + 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 diff --git a/storage/maria/ma_ft_test1.c b/storage/maria/ma_ft_test1.c index 595c31e774c..2b087dde35e 100644 --- a/storage/maria/ma_ft_test1.c +++ b/storage/maria/ma_ft_test1.c @@ -2,8 +2,7 @@ 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; either version 2 of the License, or - (at your option) any later version. + 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 diff --git a/storage/maria/ma_ft_test1.h b/storage/maria/ma_ft_test1.h index 9449f063125..5883c42f5c5 100644 --- a/storage/maria/ma_ft_test1.h +++ b/storage/maria/ma_ft_test1.h @@ -2,8 +2,7 @@ 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; either version 2 of the License, or - (at your option) any later version. + 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 diff --git a/storage/maria/ma_ft_update.c b/storage/maria/ma_ft_update.c index 25f4d5a67a0..cd2e121d0ed 100644 --- a/storage/maria/ma_ft_update.c +++ b/storage/maria/ma_ft_update.c @@ -2,8 +2,7 @@ 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; either version 2 of the License, or - (at your option) any later version. + 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 diff --git a/storage/maria/ma_ftdefs.h b/storage/maria/ma_ftdefs.h index def7e92e6e0..e25687d54b9 100644 --- a/storage/maria/ma_ftdefs.h +++ b/storage/maria/ma_ftdefs.h @@ -2,8 +2,7 @@ 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; either version 2 of the License, or - (at your option) any later version. + 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 diff --git a/storage/maria/ma_fulltext.h b/storage/maria/ma_fulltext.h index cf21471b316..778ffd49196 100644 --- a/storage/maria/ma_fulltext.h +++ b/storage/maria/ma_fulltext.h @@ -2,8 +2,7 @@ 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; either version 2 of the License, or - (at your option) any later version. + 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 diff --git a/storage/maria/ma_info.c b/storage/maria/ma_info.c index 397cd2465d4..45a67db605c 100644 --- a/storage/maria/ma_info.c +++ b/storage/maria/ma_info.c @@ -2,8 +2,7 @@ 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; either version 2 of the License, or - (at your option) any later version. + 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 diff --git a/storage/maria/ma_init.c b/storage/maria/ma_init.c index 679aa8c6f8f..271eac6c6d1 100644 --- a/storage/maria/ma_init.c +++ b/storage/maria/ma_init.c @@ -2,8 +2,7 @@ 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; either version 2 of the License, or - (at your option) any later version. + 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 diff --git a/storage/maria/ma_key.c b/storage/maria/ma_key.c index 036fd305c4d..920b59b5b54 100644 --- a/storage/maria/ma_key.c +++ b/storage/maria/ma_key.c @@ -2,8 +2,7 @@ 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; either version 2 of the License, or - (at your option) any later version. + 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 diff --git a/storage/maria/ma_keycache.c b/storage/maria/ma_keycache.c index 64725baae86..c52188a1717 100644 --- a/storage/maria/ma_keycache.c +++ b/storage/maria/ma_keycache.c @@ -2,8 +2,7 @@ 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; either version 2 of the License, or - (at your option) any later version. + 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 diff --git a/storage/maria/ma_least_recently_dirtied.c b/storage/maria/ma_least_recently_dirtied.c index 170e59a601a..3d2c85bbf98 100644 --- a/storage/maria/ma_least_recently_dirtied.c +++ b/storage/maria/ma_least_recently_dirtied.c @@ -2,8 +2,7 @@ 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; either version 2 of the License, or - (at your option) any later version. + 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 diff --git a/storage/maria/ma_least_recently_dirtied.h b/storage/maria/ma_least_recently_dirtied.h index f6d7420febc..1d57f3596f8 100644 --- a/storage/maria/ma_least_recently_dirtied.h +++ b/storage/maria/ma_least_recently_dirtied.h @@ -2,8 +2,7 @@ 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; either version 2 of the License, or - (at your option) any later version. + 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 diff --git a/storage/maria/ma_locking.c b/storage/maria/ma_locking.c index e41b2d490dc..f280af130ac 100644 --- a/storage/maria/ma_locking.c +++ b/storage/maria/ma_locking.c @@ -2,8 +2,7 @@ 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; either version 2 of the License, or - (at your option) any later version. + 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 diff --git a/storage/maria/ma_open.c b/storage/maria/ma_open.c index 7c4ad4c4a0a..86376404f74 100644 --- a/storage/maria/ma_open.c +++ b/storage/maria/ma_open.c @@ -2,8 +2,7 @@ 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; either version 2 of the License, or - (at your option) any later version. + 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 diff --git a/storage/maria/ma_packrec.c b/storage/maria/ma_packrec.c index 4e74653192e..7134297710d 100644 --- a/storage/maria/ma_packrec.c +++ b/storage/maria/ma_packrec.c @@ -2,8 +2,7 @@ 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; either version 2 of the License, or - (at your option) any later version. + 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 diff --git a/storage/maria/ma_page.c b/storage/maria/ma_page.c index d0a11ad08ab..1b013b6a0da 100644 --- a/storage/maria/ma_page.c +++ b/storage/maria/ma_page.c @@ -2,8 +2,7 @@ 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; either version 2 of the License, or - (at your option) any later version. + 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 diff --git a/storage/maria/ma_panic.c b/storage/maria/ma_panic.c index c1312cb1e77..e2e582bb390 100644 --- a/storage/maria/ma_panic.c +++ b/storage/maria/ma_panic.c @@ -2,8 +2,7 @@ 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; either version 2 of the License, or - (at your option) any later version. + 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 diff --git a/storage/maria/ma_preload.c b/storage/maria/ma_preload.c index f387f2b7de3..4b2df2bbf17 100644 --- a/storage/maria/ma_preload.c +++ b/storage/maria/ma_preload.c @@ -2,8 +2,7 @@ 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; either version 2 of the License, or - (at your option) any later version. + 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 diff --git a/storage/maria/ma_range.c b/storage/maria/ma_range.c index 4c5a037eb67..798ca348f92 100644 --- a/storage/maria/ma_range.c +++ b/storage/maria/ma_range.c @@ -2,8 +2,7 @@ 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; either version 2 of the License, or - (at your option) any later version. + 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 diff --git a/storage/maria/ma_recovery.c b/storage/maria/ma_recovery.c index b6739b86874..a42fbdf0458 100644 --- a/storage/maria/ma_recovery.c +++ b/storage/maria/ma_recovery.c @@ -2,8 +2,7 @@ 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; either version 2 of the License, or - (at your option) any later version. + 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 diff --git a/storage/maria/ma_recovery.h b/storage/maria/ma_recovery.h index 05026f4b52a..d2901f5724c 100644 --- a/storage/maria/ma_recovery.h +++ b/storage/maria/ma_recovery.h @@ -2,8 +2,7 @@ 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; either version 2 of the License, or - (at your option) any later version. + 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 diff --git a/storage/maria/ma_rename.c b/storage/maria/ma_rename.c index 59a2315b18b..a80bbcd398f 100644 --- a/storage/maria/ma_rename.c +++ b/storage/maria/ma_rename.c @@ -2,8 +2,7 @@ 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; either version 2 of the License, or - (at your option) any later version. + 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 diff --git a/storage/maria/ma_rfirst.c b/storage/maria/ma_rfirst.c index 6fa8af75c40..04c496d9c56 100644 --- a/storage/maria/ma_rfirst.c +++ b/storage/maria/ma_rfirst.c @@ -2,8 +2,7 @@ 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; either version 2 of the License, or - (at your option) any later version. + 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 diff --git a/storage/maria/ma_rkey.c b/storage/maria/ma_rkey.c index ad27d3c286c..c02c18094e8 100644 --- a/storage/maria/ma_rkey.c +++ b/storage/maria/ma_rkey.c @@ -2,8 +2,7 @@ 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; either version 2 of the License, or - (at your option) any later version. + 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 diff --git a/storage/maria/ma_rlast.c b/storage/maria/ma_rlast.c index 504cc89aed3..ebd039843c8 100644 --- a/storage/maria/ma_rlast.c +++ b/storage/maria/ma_rlast.c @@ -2,8 +2,7 @@ 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; either version 2 of the License, or - (at your option) any later version. + 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 diff --git a/storage/maria/ma_rnext.c b/storage/maria/ma_rnext.c index c7feded933e..ccca05ff3ad 100644 --- a/storage/maria/ma_rnext.c +++ b/storage/maria/ma_rnext.c @@ -2,8 +2,7 @@ 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; either version 2 of the License, or - (at your option) any later version. + 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 diff --git a/storage/maria/ma_rnext_same.c b/storage/maria/ma_rnext_same.c index a5ce0cfe15c..207a438e10b 100644 --- a/storage/maria/ma_rnext_same.c +++ b/storage/maria/ma_rnext_same.c @@ -2,8 +2,7 @@ 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; either version 2 of the License, or - (at your option) any later version. + 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 diff --git a/storage/maria/ma_rprev.c b/storage/maria/ma_rprev.c index ea562359ded..5e7cfc9f41a 100644 --- a/storage/maria/ma_rprev.c +++ b/storage/maria/ma_rprev.c @@ -2,8 +2,7 @@ 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; either version 2 of the License, or - (at your option) any later version. + 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 diff --git a/storage/maria/ma_rrnd.c b/storage/maria/ma_rrnd.c index 33940d5f23f..8d1bf9aa4f6 100644 --- a/storage/maria/ma_rrnd.c +++ b/storage/maria/ma_rrnd.c @@ -2,8 +2,7 @@ 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; either version 2 of the License, or - (at your option) any later version. + 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 diff --git a/storage/maria/ma_rsame.c b/storage/maria/ma_rsame.c index 7556c1e7332..052fe79af58 100644 --- a/storage/maria/ma_rsame.c +++ b/storage/maria/ma_rsame.c @@ -2,8 +2,7 @@ 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; either version 2 of the License, or - (at your option) any later version. + 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 diff --git a/storage/maria/ma_rsamepos.c b/storage/maria/ma_rsamepos.c index 92138693c62..1e09bdb8db4 100644 --- a/storage/maria/ma_rsamepos.c +++ b/storage/maria/ma_rsamepos.c @@ -2,8 +2,7 @@ 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; either version 2 of the License, or - (at your option) any later version. + 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 diff --git a/storage/maria/ma_rt_index.c b/storage/maria/ma_rt_index.c index cefbcffcbf9..b941c77f44d 100644 --- a/storage/maria/ma_rt_index.c +++ b/storage/maria/ma_rt_index.c @@ -3,8 +3,7 @@ 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; either version 2 of the License, or - (at your option) any later version. + 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 diff --git a/storage/maria/ma_rt_index.h b/storage/maria/ma_rt_index.h index 76b6c6e230c..c98422144e2 100644 --- a/storage/maria/ma_rt_index.h +++ b/storage/maria/ma_rt_index.h @@ -3,8 +3,7 @@ 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; either version 2 of the License, or - (at your option) any later version. + 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 diff --git a/storage/maria/ma_rt_key.c b/storage/maria/ma_rt_key.c index 1453195d263..d88b2582be4 100644 --- a/storage/maria/ma_rt_key.c +++ b/storage/maria/ma_rt_key.c @@ -2,8 +2,7 @@ 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; either version 2 of the License, or - (at your option) any later version. + 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 diff --git a/storage/maria/ma_rt_key.h b/storage/maria/ma_rt_key.h index f44251782c1..03c9ef46438 100644 --- a/storage/maria/ma_rt_key.h +++ b/storage/maria/ma_rt_key.h @@ -3,8 +3,7 @@ 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; either version 2 of the License, or - (at your option) any later version. + 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 diff --git a/storage/maria/ma_rt_mbr.c b/storage/maria/ma_rt_mbr.c index 851618a4300..2da4ffea7a4 100644 --- a/storage/maria/ma_rt_mbr.c +++ b/storage/maria/ma_rt_mbr.c @@ -3,8 +3,7 @@ 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; either version 2 of the License, or - (at your option) any later version. + 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 diff --git a/storage/maria/ma_rt_mbr.h b/storage/maria/ma_rt_mbr.h index 3282ee0d7a3..01da74418a6 100644 --- a/storage/maria/ma_rt_mbr.h +++ b/storage/maria/ma_rt_mbr.h @@ -3,8 +3,7 @@ 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; either version 2 of the License, or - (at your option) any later version. + 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 diff --git a/storage/maria/ma_rt_split.c b/storage/maria/ma_rt_split.c index a81b2b932ec..4e0abdcdb6d 100644 --- a/storage/maria/ma_rt_split.c +++ b/storage/maria/ma_rt_split.c @@ -3,8 +3,7 @@ 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; either version 2 of the License, or - (at your option) any later version. + 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 diff --git a/storage/maria/ma_rt_test.c b/storage/maria/ma_rt_test.c index 04b0c88c222..4360e81c550 100644 --- a/storage/maria/ma_rt_test.c +++ b/storage/maria/ma_rt_test.c @@ -2,8 +2,7 @@ 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; either version 2 of the License, or - (at your option) any later version. + 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 diff --git a/storage/maria/ma_scan.c b/storage/maria/ma_scan.c index 4538c87e2be..4ed4027378e 100644 --- a/storage/maria/ma_scan.c +++ b/storage/maria/ma_scan.c @@ -2,8 +2,7 @@ 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; either version 2 of the License, or - (at your option) any later version. + 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 diff --git a/storage/maria/ma_search.c b/storage/maria/ma_search.c index 991115ad181..5db09cc9618 100644 --- a/storage/maria/ma_search.c +++ b/storage/maria/ma_search.c @@ -2,8 +2,7 @@ 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; either version 2 of the License, or - (at your option) any later version. + 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 diff --git a/storage/maria/ma_sort.c b/storage/maria/ma_sort.c index 2c5524906ad..d6256deb39c 100644 --- a/storage/maria/ma_sort.c +++ b/storage/maria/ma_sort.c @@ -2,8 +2,7 @@ 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; either version 2 of the License, or - (at your option) any later version. + 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 diff --git a/storage/maria/ma_sp_defs.h b/storage/maria/ma_sp_defs.h index 6aac741bb2c..a2870bfa062 100644 --- a/storage/maria/ma_sp_defs.h +++ b/storage/maria/ma_sp_defs.h @@ -3,8 +3,7 @@ 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; either version 2 of the License, or - (at your option) any later version. + 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 diff --git a/storage/maria/ma_sp_key.c b/storage/maria/ma_sp_key.c index b365f8deb0f..06769e97d30 100644 --- a/storage/maria/ma_sp_key.c +++ b/storage/maria/ma_sp_key.c @@ -2,8 +2,7 @@ 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; either version 2 of the License, or - (at your option) any later version. + 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 diff --git a/storage/maria/ma_sp_test.c b/storage/maria/ma_sp_test.c index 1ac1a74d7d7..bab346ca18d 100644 --- a/storage/maria/ma_sp_test.c +++ b/storage/maria/ma_sp_test.c @@ -2,8 +2,7 @@ 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; either version 2 of the License, or - (at your option) any later version. + 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 diff --git a/storage/maria/ma_static.c b/storage/maria/ma_static.c index c5580e1e981..c747d07bc8e 100644 --- a/storage/maria/ma_static.c +++ b/storage/maria/ma_static.c @@ -2,8 +2,7 @@ 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; either version 2 of the License, or - (at your option) any later version. + 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 diff --git a/storage/maria/ma_statrec.c b/storage/maria/ma_statrec.c index 72c5e10d9ab..68864e7c170 100644 --- a/storage/maria/ma_statrec.c +++ b/storage/maria/ma_statrec.c @@ -2,8 +2,7 @@ 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; either version 2 of the License, or - (at your option) any later version. + 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 diff --git a/storage/maria/ma_test1.c b/storage/maria/ma_test1.c index 0f37391c1d4..1cb8c3e002a 100644 --- a/storage/maria/ma_test1.c +++ b/storage/maria/ma_test1.c @@ -2,8 +2,7 @@ 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; either version 2 of the License, or - (at your option) any later version. + 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 diff --git a/storage/maria/ma_test2.c b/storage/maria/ma_test2.c index 46b8c710d4a..4ec5afd35e7 100644 --- a/storage/maria/ma_test2.c +++ b/storage/maria/ma_test2.c @@ -2,8 +2,7 @@ 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; either version 2 of the License, or - (at your option) any later version. + 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 diff --git a/storage/maria/ma_test3.c b/storage/maria/ma_test3.c index f6ed248ce16..9177b87ac4b 100644 --- a/storage/maria/ma_test3.c +++ b/storage/maria/ma_test3.c @@ -2,8 +2,7 @@ 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; either version 2 of the License, or - (at your option) any later version. + 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 diff --git a/storage/maria/ma_unique.c b/storage/maria/ma_unique.c index 5d0133c8ac1..06d29b9b037 100644 --- a/storage/maria/ma_unique.c +++ b/storage/maria/ma_unique.c @@ -2,8 +2,7 @@ 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; either version 2 of the License, or - (at your option) any later version. + 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 diff --git a/storage/maria/ma_update.c b/storage/maria/ma_update.c index 29018b47bda..8c8e46bc024 100644 --- a/storage/maria/ma_update.c +++ b/storage/maria/ma_update.c @@ -2,8 +2,7 @@ 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; either version 2 of the License, or - (at your option) any later version. + 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 diff --git a/storage/maria/ma_write.c b/storage/maria/ma_write.c index b672ed1fe23..ec84eabd421 100644 --- a/storage/maria/ma_write.c +++ b/storage/maria/ma_write.c @@ -2,8 +2,7 @@ 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; either version 2 of the License, or - (at your option) any later version. + 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 diff --git a/storage/maria/maria_chk.c b/storage/maria/maria_chk.c index 533766e35f9..5796ae1a196 100644 --- a/storage/maria/maria_chk.c +++ b/storage/maria/maria_chk.c @@ -2,8 +2,7 @@ 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; either version 2 of the License, or - (at your option) any later version. + 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 diff --git a/storage/maria/maria_def.h b/storage/maria/maria_def.h index f4684735938..8f54001ad45 100644 --- a/storage/maria/maria_def.h +++ b/storage/maria/maria_def.h @@ -2,8 +2,7 @@ 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; either version 2 of the License, or - (at your option) any later version. + 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 diff --git a/storage/maria/maria_ftdump.c b/storage/maria/maria_ftdump.c index ef30540bfce..f1d80d90828 100644 --- a/storage/maria/maria_ftdump.c +++ b/storage/maria/maria_ftdump.c @@ -2,8 +2,7 @@ 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; either version 2 of the License, or - (at your option) any later version. + 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 diff --git a/storage/maria/maria_pack.c b/storage/maria/maria_pack.c index a76b1027179..dbd48e62d29 100644 --- a/storage/maria/maria_pack.c +++ b/storage/maria/maria_pack.c @@ -2,8 +2,7 @@ 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; either version 2 of the License, or - (at your option) any later version. + 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 diff --git a/storage/maria/tablockman.c b/storage/maria/tablockman.c index 810c6c12ea4..4634f60a085 100644 --- a/storage/maria/tablockman.c +++ b/storage/maria/tablockman.c @@ -4,8 +4,7 @@ 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; either version 2 of the License, or - (at your option) any later version. + 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 diff --git a/storage/maria/tablockman.h b/storage/maria/tablockman.h index 2c6fb6996a3..58c852b5a21 100644 --- a/storage/maria/tablockman.h +++ b/storage/maria/tablockman.h @@ -2,8 +2,7 @@ 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; either version 2 of the License, or - (at your option) any later version. + 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 diff --git a/storage/maria/trnman.c b/storage/maria/trnman.c index 37978e4ff76..7918c1aa00d 100644 --- a/storage/maria/trnman.c +++ b/storage/maria/trnman.c @@ -2,8 +2,7 @@ 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; either version 2 of the License, or - (at your option) any later version. + 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 diff --git a/storage/maria/trnman.h b/storage/maria/trnman.h index 267e3cabd7a..87107ab52fb 100644 --- a/storage/maria/trnman.h +++ b/storage/maria/trnman.h @@ -2,8 +2,7 @@ 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; either version 2 of the License, or - (at your option) any later version. + 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 -- cgit v1.2.1 From 1a77ebd14c5ba2244db4b515c367e62bf97e8d8e Mon Sep 17 00:00:00 2001 From: unknown Date: Fri, 2 Mar 2007 12:15:19 +0100 Subject: Maria: a stronger checksum in the control file, to test integrity. 4 bytes, using my_checksum() (the old checksum was one byte and just a sum of the bytes - that was before I saw we have my_checksum :) storage/maria/ma_control_file.c: stronger checksum (4 bytes instead of 1, and using CRC instead of simple byte sum). storage/maria/unittest/ma_control_file-t.c: Checksum is now 4 bytes (total length of control file is now 23), so LSN and LAST_LOGNO move. --- storage/maria/ma_control_file.c | 29 +++++++++++------------------ storage/maria/unittest/ma_control_file-t.c | 8 ++++---- 2 files changed, 15 insertions(+), 22 deletions(-) diff --git a/storage/maria/ma_control_file.c b/storage/maria/ma_control_file.c index 2caa0038df2..f53da8a5881 100644 --- a/storage/maria/ma_control_file.c +++ b/storage/maria/ma_control_file.c @@ -33,7 +33,7 @@ #define CONTROL_FILE_MAGIC_STRING_OFFSET 0 #define CONTROL_FILE_MAGIC_STRING_SIZE (sizeof(CONTROL_FILE_MAGIC_STRING)-1) #define CONTROL_FILE_CHECKSUM_OFFSET (CONTROL_FILE_MAGIC_STRING_OFFSET + CONTROL_FILE_MAGIC_STRING_SIZE) -#define CONTROL_FILE_CHECKSUM_SIZE 1 +#define CONTROL_FILE_CHECKSUM_SIZE 4 #define CONTROL_FILE_LSN_OFFSET (CONTROL_FILE_CHECKSUM_OFFSET + CONTROL_FILE_CHECKSUM_SIZE) #define CONTROL_FILE_LSN_SIZE LSN_STORE_SIZE #define CONTROL_FILE_FILENO_OFFSET (CONTROL_FILE_LSN_OFFSET + CONTROL_FILE_LSN_SIZE) @@ -57,16 +57,6 @@ uint32 last_logno; */ static int control_file_fd= -1; -static char simple_checksum(char *buffer, uint size) -{ - /* TODO: improve this sum if we want */ - char s= 0; - uint i; - for (i= 0; i= 0); - RET_ERR_UNLESS(my_read(fd, buffer, 20, MYF(MY_FNABP | MY_WME)) == 0); + RET_ERR_UNLESS(my_read(fd, buffer, 23, MYF(MY_FNABP | MY_WME)) == 0); RET_ERR_UNLESS(my_close(fd, MYF(MY_WME)) == 0); RET_ERR_UNLESS(create_or_open_file() == CONTROL_FILE_OK); - i= uint3korr(buffer+9); + i= uint3korr(buffer+12); RET_ERR_UNLESS(i == LSN_FILE_NO(last_checkpoint_lsn)); - i= uint4korr(buffer+12); + i= uint4korr(buffer+15); RET_ERR_UNLESS(i == LSN_OFFSET(last_checkpoint_lsn)); - i= uint4korr(buffer+16); + i= uint4korr(buffer+19); RET_ERR_UNLESS(i == last_logno); RET_ERR_UNLESS(close_file() == 0); return 0; -- cgit v1.2.1 From 39d64a1d2565b09307d11b2a665f3f2c6bc8106e Mon Sep 17 00:00:00 2001 From: unknown Date: Wed, 4 Apr 2007 23:37:09 +0300 Subject: Pagecache integration for review. storage/maria/unittest/ma_pagecache_single.c: Rename: storage/maria/unittest/mf_pagecache_single.c -> storage/maria/unittest/ma_pagecache_single.c include/maria.h: Pagecache integration. include/myisamchk.h: Pagecache integration. include/pagecache.h: removed WRITE_NOW mode Pagecache parameters management. mysys/Makefile.am: Safe hash procedures moved to the separate file. Pagecache moved to maria engine directory. mysys/mf_keycaches.c: Safe hash procedures moved to the separate file. sql/handler.cc: Pageccahe integration. sql/handler.h: Pagecache integration. sql/mysql_priv.h: pagecache integration sql/mysqld.cc: pagecache integration sql/set_var.cc: Pagecache integration. sql/set_var.h: Pagecache integration. storage/maria/Makefile.am: Pagecache integration and moving to maria engine directory. storage/maria/ha_maria.cc: File changed on PAGECCAHE_FILE. storage/maria/ma_bitmap.c: Pagecache integration. storage/maria/ma_blockrec.c: Pagecache integration. storage/maria/ma_check.c: File changed on PAGECCAHE_FILE. Pagecache integration. storage/maria/ma_close.c: File changed on PAGECCAHE_FILE. storage/maria/ma_delete_all.c: File changed on PAGECCAHE_FILE. storage/maria/ma_dynrec.c: File changed on PAGECCAHE_FILE. storage/maria/ma_extra.c: File changed on PAGECCAHE_FILE. storage/maria/ma_info.c: File changed on PAGECCAHE_FILE. storage/maria/ma_keycache.c: Pagecache integration. storage/maria/ma_locking.c: File changed on PAGECCAHE_FILE. storage/maria/ma_loghandler.c: Assert added. storage/maria/ma_loghandler.h: extern specifier added. storage/maria/ma_open.c: Pagecache integration. File changed on PAGECCAHE_FILE. storage/maria/ma_packrec.c: File changed on PAGECCAHE_FILE. storage/maria/ma_page.c: Pagecache integration. storage/maria/ma_pagecache.c: Pagecache renamed and moved to the maria directory. BLOCK_* defines renamed to avoid conflict with BLOCK_ERROR defined in maria_def.h storage/maria/ma_panic.c: File changed on PAGECCAHE_FILE. storage/maria/ma_preload.c: Pagecache integration. File changed on PAGECCAHE_FILE. storage/maria/ma_static.c: Pagecache integration. storage/maria/ma_test1.c: Pagecache integration. storage/maria/ma_test2.c: Pagecache integration. storage/maria/ma_test3.c: Pagecache integration. storage/maria/ma_write.c: File changed on PAGECCAHE_FILE. storage/maria/maria_chk.c: Pagecache integration. File changed on PAGECCAHE_FILE. storage/maria/maria_def.h: Pagecache integration. File changed on PAGECCAHE_FILE. storage/maria/maria_ftdump.c: Pagecache integration. storage/maria/maria_pack.c: File changed on PAGECCAHE_FILE. storage/maria/unittest/Makefile.am: Pagecache moved to the maria directory. storage/maria/unittest/ma_pagecache_consist.c: fixed using uninitialized variable storage/maria/ma_pagecaches.c: New BitKeeper file ``storage/maria/ma_pagecaches.c'' mysys/my_safehash.h: New BitKeeper file ``mysys/my_safehash.h'' --- include/maria.h | 14 +- include/myisamchk.h | 2 +- include/pagecache.h | 44 +- mysys/Makefile.am | 3 +- mysys/mf_keycaches.c | 269 +- mysys/mf_pagecache.c | 4102 ------------------------- mysys/my_safehash.h | 58 + sql/handler.cc | 97 + sql/handler.h | 13 +- sql/mysql_priv.h | 3 + sql/mysqld.cc | 78 +- sql/set_var.cc | 176 +- sql/set_var.h | 42 +- storage/maria/Makefile.am | 3 +- storage/maria/ha_maria.cc | 10 +- storage/maria/ma_bitmap.c | 19 +- storage/maria/ma_blockrec.c | 129 +- storage/maria/ma_check.c | 156 +- storage/maria/ma_close.c | 19 +- storage/maria/ma_delete_all.c | 7 +- storage/maria/ma_dynrec.c | 49 +- storage/maria/ma_extra.c | 27 +- storage/maria/ma_info.c | 4 +- storage/maria/ma_keycache.c | 50 +- storage/maria/ma_locking.c | 37 +- storage/maria/ma_loghandler.c | 1 + storage/maria/ma_loghandler.h | 69 +- storage/maria/ma_open.c | 25 +- storage/maria/ma_packrec.c | 8 +- storage/maria/ma_page.c | 84 +- storage/maria/ma_pagecache.c | 4100 ++++++++++++++++++++++++ storage/maria/ma_pagecaches.c | 105 + storage/maria/ma_panic.c | 30 +- storage/maria/ma_preload.c | 28 +- storage/maria/ma_static.c | 4 +- storage/maria/ma_test1.c | 12 +- storage/maria/ma_test2.c | 49 +- storage/maria/ma_test3.c | 10 +- storage/maria/ma_write.c | 2 +- storage/maria/maria_chk.c | 45 +- storage/maria/maria_def.h | 11 +- storage/maria/maria_ftdump.c | 3 +- storage/maria/maria_pack.c | 10 +- storage/maria/unittest/Makefile.am | 73 +- storage/maria/unittest/ma_pagecache_consist.c | 458 +++ storage/maria/unittest/ma_pagecache_single.c | 580 ++++ storage/maria/unittest/mf_pagecache_consist.c | 458 --- storage/maria/unittest/mf_pagecache_single.c | 580 ---- 48 files changed, 6281 insertions(+), 5875 deletions(-) delete mode 100755 mysys/mf_pagecache.c create mode 100644 mysys/my_safehash.h create mode 100755 storage/maria/ma_pagecache.c create mode 100644 storage/maria/ma_pagecaches.c create mode 100755 storage/maria/unittest/ma_pagecache_consist.c create mode 100644 storage/maria/unittest/ma_pagecache_single.c delete mode 100755 storage/maria/unittest/mf_pagecache_consist.c delete mode 100644 storage/maria/unittest/mf_pagecache_single.c diff --git a/include/maria.h b/include/maria.h index 94152ce9bfa..6777d8b1b63 100644 --- a/include/maria.h +++ b/include/maria.h @@ -27,8 +27,8 @@ extern "C" { #ifndef _m_ctype_h #include #endif -#ifndef _keycache_h -#include "keycache.h" +#ifndef _pagecache_h +#include "pagecache.h" #endif #include "my_handler.h" #include "ft_global.h" @@ -248,7 +248,7 @@ extern my_bool maria_flush, maria_single_user; extern my_bool maria_delay_key_write, maria_delay_rec_write; extern my_off_t maria_max_temp_length; extern ulong maria_bulk_insert_tree_size, maria_data_pointer_size; -extern KEY_CACHE maria_key_cache_var, *maria_key_cache; +extern PAGECACHE maria_pagecache_var, *maria_pagecache; /* Prototypes for maria-functions */ @@ -417,10 +417,10 @@ my_bool maria_test_if_sort_rep(MARIA_HA *info, ha_rows rows, ulonglong key_map, int maria_init_bulk_insert(MARIA_HA *info, ulong cache_size, ha_rows rows); void maria_flush_bulk_insert(MARIA_HA *info, uint inx); void maria_end_bulk_insert(MARIA_HA *info); -int maria_assign_to_key_cache(MARIA_HA *info, ulonglong key_map, - KEY_CACHE *key_cache); -void maria_change_key_cache(KEY_CACHE *old_key_cache, - KEY_CACHE *new_key_cache); +int maria_assign_to_pagecache(MARIA_HA *info, ulonglong key_map, + PAGECACHE *key_cache); +void maria_change_pagecache(PAGECACHE *old_key_cache, + PAGECACHE *new_key_cache); int maria_preload(MARIA_HA *info, ulonglong key_map, my_bool ignore_leaves); /* fulltext functions */ diff --git a/include/myisamchk.h b/include/myisamchk.h index 66d0a77b62f..887cf835b87 100644 --- a/include/myisamchk.h +++ b/include/myisamchk.h @@ -133,7 +133,7 @@ typedef struct st_handler_check_param ulong rec_per_key_part[HA_MAX_KEY_SEG * HA_MAX_POSSIBLE_KEY]; uint out_flag, warning_printed, error_printed, verbose; uint opt_sort_key, total_files, max_level; - uint testflag, key_cache_block_size; + uint testflag, key_cache_block_size, pagecache_block_size; int tmpfile_createflag, err_count; myf myf_rw; uint8 language; diff --git a/include/pagecache.h b/include/pagecache.h index 32e23fb54ea..08ecc7e5f4a 100644 --- a/include/pagecache.h +++ b/include/pagecache.h @@ -66,8 +66,6 @@ enum pagecache_write_mode { /* do not write immediately, i.e. it will be dirty page */ PAGECACHE_WRITE_DELAY, - /* write page to the file and put it to the cache */ - PAGECACHE_WRITE_NOW, /* page already is in the file. (key cache insert analogue) */ PAGECACHE_WRITE_DONE }; @@ -162,6 +160,9 @@ typedef struct st_pagecache my_bool in_init; /* Set to 1 in MySQL during init/resize */ } PAGECACHE; +/* The default key cache */ +extern PAGECACHE dflt_pagecache_var, *dflt_pagecache; + extern int init_pagecache(PAGECACHE *pagecache, my_size_t use_mem, uint division_limit, uint age_threshold, uint block_size); @@ -184,16 +185,22 @@ extern byte *pagecache_valid_read(PAGECACHE *pagecache, PAGECACHE_PAGE_LINK *link, pagecache_disk_read_validator validator, gptr validator_data); -extern my_bool pagecache_write(PAGECACHE *pagecache, - PAGECACHE_FILE *file, - pgcache_page_no_t pageno, - uint level, - byte *buff, - enum pagecache_page_type type, - enum pagecache_page_lock lock, - enum pagecache_page_pin pin, - enum pagecache_write_mode write_mode, - PAGECACHE_PAGE_LINK *link); + +#define pagecache_write(P,F,N,L,B,T,O,I,M,K) \ + pagecache_write_part(P,F,N,L,B,T,O,I,M,K,0,(P)->block_size) + +extern my_bool pagecache_write_part(PAGECACHE *pagecache, + PAGECACHE_FILE *file, + pgcache_page_no_t pageno, + uint level, + byte *buff, + enum pagecache_page_type type, + enum pagecache_page_lock lock, + enum pagecache_page_pin pin, + enum pagecache_write_mode write_mode, + PAGECACHE_PAGE_LINK *link, + uint offset, + uint size); extern void pagecache_unlock_page(PAGECACHE *pagecache, PAGECACHE_FILE *file, pgcache_page_no_t pageno, @@ -224,5 +231,18 @@ extern my_bool pagecache_collect_changed_blocks_with_lsn(PAGECACHE *pagecache, LSN *max_lsn); extern int reset_pagecache_counters(const char *name, PAGECACHE *pagecache); + +/* Functions to handle multiple key caches */ +extern my_bool multi_pagecache_init(void); +extern void multi_pagecache_free(void); +extern PAGECACHE *multi_pagecache_search(byte *key, uint length, + PAGECACHE *def); +extern my_bool multi_pagecache_set(const byte *key, uint length, + PAGECACHE *pagecache); +extern void multi_pagecache_change(PAGECACHE *old_data, + PAGECACHE *new_data); +extern int reset_pagecache_counters(const char *name, + PAGECACHE *pagecache); + C_MODE_END #endif /* _keycache_h */ diff --git a/mysys/Makefile.am b/mysys/Makefile.am index bec72263ba4..3c686733b59 100644 --- a/mysys/Makefile.am +++ b/mysys/Makefile.am @@ -26,6 +26,7 @@ libmysys_a_SOURCES = my_init.c my_getwd.c mf_getdate.c my_mmap.c \ mf_path.c mf_loadpath.c my_file.c \ my_open.c my_create.c my_dup.c my_seek.c my_read.c \ my_pread.c my_write.c my_getpagesize.c \ + my_safehash.c \ mf_keycache.c mf_keycaches.c my_crc32.c \ mf_iocache.c mf_iocache2.c mf_cache.c mf_tempfile.c \ mf_tempdir.c my_lock.c mf_brkhant.c my_alarm.c \ @@ -55,7 +56,7 @@ libmysys_a_SOURCES = my_init.c my_getwd.c mf_getdate.c my_mmap.c \ my_handler.c my_netware.c my_largepage.c \ my_memmem.c \ my_windac.c my_access.c base64.c my_libwrap.c \ - mf_pagecache.c wqueue.c + wqueue.c EXTRA_DIST = thr_alarm.c thr_lock.c my_pthread.c my_thr_init.c \ thr_mutex.c thr_rwlock.c \ CMakeLists.txt mf_soundex.c \ diff --git a/mysys/mf_keycaches.c b/mysys/mf_keycaches.c index 4c20f69053d..39ff536d3e0 100644 --- a/mysys/mf_keycaches.c +++ b/mysys/mf_keycaches.c @@ -1,4 +1,4 @@ -/* Copyright (C) 2003 MySQL AB +/* Copyright (C) 2003-2007 MySQL AB This program is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by @@ -26,270 +26,7 @@ #include #include #include - -/***************************************************************************** - General functions to handle SAFE_HASH objects. - - A SAFE_HASH object is used to store the hash, the mutex and default value - needed by the rest of the key cache code. - This is a separate struct to make it easy to later reuse the code for other - purposes - - All entries are linked in a list to allow us to traverse all elements - and delete selected ones. (HASH doesn't allow any easy ways to do this). -*****************************************************************************/ - -/* - Struct to store a key and pointer to object -*/ - -typedef struct st_safe_hash_entry -{ - byte *key; - uint length; - byte *data; - struct st_safe_hash_entry *next, **prev; -} SAFE_HASH_ENTRY; - - -typedef struct st_safe_hash_with_default -{ -#ifdef THREAD - rw_lock_t mutex; -#endif - HASH hash; - byte *default_value; - SAFE_HASH_ENTRY *root; -} SAFE_HASH; - - -/* - Free a SAFE_HASH_ENTRY - - This function is called by the hash object on delete -*/ - -static void safe_hash_entry_free(SAFE_HASH_ENTRY *entry) -{ - DBUG_ENTER("free_assign_entry"); - my_free((gptr) entry, MYF(0)); - DBUG_VOID_RETURN; -} - - -/* Get key and length for a SAFE_HASH_ENTRY */ - -static byte *safe_hash_entry_get(SAFE_HASH_ENTRY *entry, uint *length, - my_bool not_used __attribute__((unused))) -{ - *length=entry->length; - return (byte*) entry->key; -} - - -/* - Init a SAFE_HASH object - - SYNOPSIS - safe_hash_init() - hash safe_hash handler - elements Expected max number of elements - default_value default value - - NOTES - In case of error we set hash->default_value to 0 to allow one to call - safe_hash_free on an object that couldn't be initialized. - - RETURN - 0 ok - 1 error -*/ - -static my_bool safe_hash_init(SAFE_HASH *hash, uint elements, - byte *default_value) -{ - DBUG_ENTER("safe_hash"); - if (hash_init(&hash->hash, &my_charset_bin, elements, - 0, 0, (hash_get_key) safe_hash_entry_get, - (void (*)(void*)) safe_hash_entry_free, 0)) - { - hash->default_value= 0; - DBUG_RETURN(1); - } - my_rwlock_init(&hash->mutex, 0); - hash->default_value= default_value; - hash->root= 0; - DBUG_RETURN(0); -} - - -/* - Free a SAFE_HASH object - - NOTES - This is safe to call on any object that has been sent to safe_hash_init() -*/ - -static void safe_hash_free(SAFE_HASH *hash) -{ - /* - Test if safe_hash_init succeeded. This will also guard us against multiple - free calls. - */ - if (hash->default_value) - { - hash_free(&hash->hash); - rwlock_destroy(&hash->mutex); - hash->default_value=0; - } -} - -/* - Return the value stored for a key or default value if no key -*/ - -static byte *safe_hash_search(SAFE_HASH *hash, const byte *key, uint length, - byte *def) -{ - byte *result; - DBUG_ENTER("safe_hash_search"); - rw_rdlock(&hash->mutex); - result= hash_search(&hash->hash, key, length); - rw_unlock(&hash->mutex); - if (!result) - result= def; - else - result= ((SAFE_HASH_ENTRY*) result)->data; - DBUG_PRINT("exit",("data: 0x%lx", (long) result)); - DBUG_RETURN(result); -} - - -/* - Associate a key with some data - - SYONOPSIS - safe_hash_set() - hash Hash handle - key key (path to table etc..) - length Length of key - data data to to associate with the data - - NOTES - This can be used both to insert a new entry and change an existing - entry. - If one associates a key with the default key cache, the key is deleted - - RETURN - 0 ok - 1 error (Can only be EOM). In this case my_message() is called. -*/ - -static my_bool safe_hash_set(SAFE_HASH *hash, const byte *key, uint length, - byte *data) -{ - SAFE_HASH_ENTRY *entry; - my_bool error= 0; - DBUG_ENTER("safe_hash_set"); - DBUG_PRINT("enter",("key: %.*s data: 0x%lx", length, key, (long) data)); - - rw_wrlock(&hash->mutex); - entry= (SAFE_HASH_ENTRY*) hash_search(&hash->hash, key, length); - - if (data == hash->default_value) - { - /* - The key is to be associated with the default entry. In this case - we can just delete the entry (if it existed) from the hash as a - search will return the default entry - */ - if (!entry) /* nothing to do */ - goto end; - /* unlink entry from list */ - if ((*entry->prev= entry->next)) - entry->next->prev= entry->prev; - hash_delete(&hash->hash, (byte*) entry); - goto end; - } - if (entry) - { - /* Entry existed; Just change the pointer to point at the new data */ - entry->data= data; - } - else - { - if (!(entry= (SAFE_HASH_ENTRY *) my_malloc(sizeof(*entry) + length, - MYF(MY_WME)))) - { - error= 1; - goto end; - } - entry->key= (byte*) (entry +1); - memcpy((char*) entry->key, (char*) key, length); - entry->length= length; - entry->data= data; - /* Link entry to list */ - if ((entry->next= hash->root)) - entry->next->prev= &entry->next; - entry->prev= &hash->root; - hash->root= entry; - if (my_hash_insert(&hash->hash, (byte*) entry)) - { - /* This can only happen if hash got out of memory */ - my_free((char*) entry, MYF(0)); - error= 1; - goto end; - } - } - -end: - rw_unlock(&hash->mutex); - DBUG_RETURN(error); -} - - -/* - Change all entres with one data value to another data value - - SYONOPSIS - safe_hash_change() - hash Hash handle - old_data Old data - new_data Change all 'old_data' to this - - NOTES - We use the linked list to traverse all elements in the hash as - this allows us to delete elements in the case where 'new_data' is the - default value. -*/ - -static void safe_hash_change(SAFE_HASH *hash, byte *old_data, byte *new_data) -{ - SAFE_HASH_ENTRY *entry, *next; - DBUG_ENTER("safe_hash_set"); - - rw_wrlock(&hash->mutex); - - for (entry= hash->root ; entry ; entry= next) - { - next= entry->next; - if (entry->data == old_data) - { - if (new_data == hash->default_value) - { - if ((*entry->prev= entry->next)) - entry->next->prev= entry->prev; - hash_delete(&hash->hash, (byte*) entry); - } - else - entry->data= new_data; - } - } - - rw_unlock(&hash->mutex); - DBUG_VOID_RETURN; -} - +#include "my_safehash.h" /***************************************************************************** Functions to handle the key cache objects @@ -366,3 +103,5 @@ void multi_key_cache_change(KEY_CACHE *old_data, { safe_hash_change(&key_cache_hash, (byte*) old_data, (byte*) new_data); } + + diff --git a/mysys/mf_pagecache.c b/mysys/mf_pagecache.c deleted file mode 100755 index 1b9d48c80e6..00000000000 --- a/mysys/mf_pagecache.c +++ /dev/null @@ -1,4102 +0,0 @@ -/* Copyright (C) 2000-2006 MySQL AB - - 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; either version 2 of the License, or - (at your option) any later version. - - 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ - -/* - These functions handle page cacheing for Maria tables. - - One cache can handle many files. - It must contain buffers of the same blocksize. - init_pagecache() should be used to init cache handler. - - The free list (free_block_list) is a stack like structure. - When a block is freed by free_block(), it is pushed onto the stack. - When a new block is required it is first tried to pop one from the stack. - If the stack is empty, it is tried to get a never-used block from the pool. - If this is empty too, then a block is taken from the LRU ring, flushing it - to disk, if necessary. This is handled in find_block(). - With the new free list, the blocks can have three temperatures: - hot, warm and cold (which is free). This is remembered in the block header - by the enum BLOCK_TEMPERATURE temperature variable. Remembering the - temperature is necessary to correctly count the number of warm blocks, - which is required to decide when blocks are allowed to become hot. Whenever - a block is inserted to another (sub-)chain, we take the old and new - temperature into account to decide if we got one more or less warm block. - blocks_unused is the sum of never used blocks in the pool and of currently - free blocks. blocks_used is the number of blocks fetched from the pool and - as such gives the maximum number of in-use blocks at any time. -*/ - -#include "mysys_priv.h" -#include -#include -#include "my_static.h" -#include -#include -#include - -/* - Some compilation flags have been added specifically for this module - to control the following: - - not to let a thread to yield the control when reading directly - from page cache, which might improve performance in many cases; - to enable this add: - #define SERIALIZED_READ_FROM_CACHE - - to set an upper bound for number of threads simultaneously - using the page cache; this setting helps to determine an optimal - size for hash table and improve performance when the number of - blocks in the page cache much less than the number of threads - accessing it; - to set this number equal to add - #define MAX_THREADS - - to substitute calls of pthread_cond_wait for calls of - pthread_cond_timedwait (wait with timeout set up); - this setting should be used only when you want to trap a deadlock - situation, which theoretically should not happen; - to set timeout equal to seconds add - #define PAGECACHE_TIMEOUT - - to enable the module traps and to send debug information from - page cache module to a special debug log add: - #define PAGECACHE_DEBUG - the name of this debug log file can be set through: - #define PAGECACHE_DEBUG_LOG - if the name is not defined, it's set by default; - if the PAGECACHE_DEBUG flag is not set up and we are in a debug - mode, i.e. when ! defined(DBUG_OFF), the debug information from the - module is sent to the regular debug log. - - Example of the settings: - #define SERIALIZED_READ_FROM_CACHE - #define MAX_THREADS 100 - #define PAGECACHE_TIMEOUT 1 - #define PAGECACHE_DEBUG - #define PAGECACHE_DEBUG_LOG "my_pagecache_debug.log" -*/ - -/* - In key cache we have external raw locking here we use - SERIALIZED_READ_FROM_CACHE to avoid problem of reading - not consistent data from the page. - (keycache functions (key_cache_read(), key_cache_insert() and - key_cache_write()) rely on external MyISAM lock, we don't) -*/ -#define SERIALIZED_READ_FROM_CACHE yes - -#define BLOCK_INFO(B) \ - DBUG_PRINT("info", \ - ("block 0x%lx file %lu page %lu s %0x hshL 0x%lx req %u/%u " \ - "wrlock: %c", \ - (ulong)(B), \ - (ulong)((B)->hash_link ? \ - (B)->hash_link->file.file : \ - 0), \ - (ulong)((B)->hash_link ? \ - (B)->hash_link->pageno : \ - 0), \ - (B)->status, \ - (ulong)(B)->hash_link, \ - (uint) (B)->requests, \ - (uint)((B)->hash_link ? \ - (B)->hash_link->requests : \ - 0), \ - ((block->status & BLOCK_WRLOCK)?'Y':'N'))) - -/* TODO: put it to my_static.c */ -my_bool my_disable_flush_pagecache_blocks= 0; - -#define STRUCT_PTR(TYPE, MEMBER, a) \ - (TYPE *) ((char *) (a) - offsetof(TYPE, MEMBER)) - -/* types of condition variables */ -#define COND_FOR_REQUESTED 0 /* queue of thread waiting for read operation */ -#define COND_FOR_SAVED 1 /* queue of thread waiting for flush */ -#define COND_FOR_WRLOCK 2 /* queue of write lock */ -#define COND_SIZE 3 /* number of COND_* queues */ - -typedef pthread_cond_t KEYCACHE_CONDVAR; - -/* descriptor of the page in the page cache block buffer */ -struct st_pagecache_page -{ - PAGECACHE_FILE file; /* file to which the page belongs to */ - pgcache_page_no_t pageno; /* number of the page in the file */ -}; - -/* element in the chain of a hash table bucket */ -struct st_pagecache_hash_link -{ - struct st_pagecache_hash_link - *next, **prev; /* to connect links in the same bucket */ - struct st_pagecache_block_link - *block; /* reference to the block for the page: */ - PAGECACHE_FILE file; /* from such a file */ - pgcache_page_no_t pageno; /* this page */ - uint requests; /* number of requests for the page */ -}; - -/* simple states of a block */ -#define BLOCK_ERROR 1 /* an error occurred when performing disk i/o */ -#define BLOCK_READ 2 /* the is page in the block buffer */ -#define BLOCK_IN_SWITCH 4 /* block is preparing to read new page */ -#define BLOCK_REASSIGNED 8 /* block does not accept requests for old page */ -#define BLOCK_IN_FLUSH 16 /* block is in flush operation */ -#define BLOCK_CHANGED 32 /* block buffer contains a dirty page */ -#define BLOCK_WRLOCK 64 /* write locked block */ - -/* page status, returned by find_block */ -#define PAGE_READ 0 -#define PAGE_TO_BE_READ 1 -#define PAGE_WAIT_TO_BE_READ 2 - -/* block temperature determines in which (sub-)chain the block currently is */ -enum BLOCK_TEMPERATURE { BLOCK_COLD /*free*/ , BLOCK_WARM , BLOCK_HOT }; - -/* debug info */ -#ifndef DBUG_OFF -static char *page_cache_page_type_str[]= -{ - (char*)"PLAIN", - (char*)"LSN" -}; -static char *page_cache_page_write_mode_str[]= -{ - (char*)"DELAY", - (char*)"NOW", - (char*)"DONE" -}; -static char *page_cache_page_lock_str[]= -{ - (char*)"free -> free ", - (char*)"read -> read ", - (char*)"write -> write", - (char*)"free -> read ", - (char*)"free -> write", - (char*)"read -> free ", - (char*)"write -> free ", - (char*)"write -> read " -}; -static char *page_cache_page_pin_str[]= -{ - (char*)"pinned -> pinned ", - (char*)"unpinned -> unpinned", - (char*)"unpinned -> pinned ", - (char*)"pinned -> unpinned" -}; -#endif -#ifdef PAGECACHE_DEBUG -typedef struct st_pagecache_pin_info -{ - struct st_pagecache_pin_info *next, **prev; - struct st_my_thread_var *thread; -} PAGECACHE_PIN_INFO; -/* - st_pagecache_lock_info structure should be kept in next, prev, thread part - compatible with st_pagecache_pin_info to be compatible in functions. -*/ -typedef struct st_pagecache_lock_info -{ - struct st_pagecache_lock_info *next, **prev; - struct st_my_thread_var *thread; - my_bool write_lock; -} PAGECACHE_LOCK_INFO; - - -/* service functions maintain debugging info about pin & lock */ - - -/* - Links information about thread pinned/locked the block to the list - - SYNOPSIS - info_link() - list the list to link in - node the node which should be linked -*/ - -static void info_link(PAGECACHE_PIN_INFO **list, PAGECACHE_PIN_INFO *node) -{ - if ((node->next= *list)) - node->next->prev= &(node->next); - *list= node; - node->prev= list; -} - - -/* - Unlinks information about thread pinned/locked the block from the list - - SYNOPSIS - info_unlink() - node the node which should be unlinked -*/ - -static void info_unlink(PAGECACHE_PIN_INFO *node) -{ - if ((*node->prev= node->next)) - node->next->prev= node->prev; -} - - -/* - Finds information about given thread in the list of threads which - pinned/locked this block. - - SYNOPSIS - info_find() - list the list where to find the thread - thread thread ID (reference to the st_my_thread_var - of the thread) - - RETURN - 0 - the thread was not found - pointer to the information node of the thread in the list -*/ - -static PAGECACHE_PIN_INFO *info_find(PAGECACHE_PIN_INFO *list, - struct st_my_thread_var *thread) -{ - register PAGECACHE_PIN_INFO *i= list; - for(; i != 0; i= i->next) - if (i->thread == thread) - return i; - return 0; -} -#endif - -/* page cache block */ -struct st_pagecache_block_link -{ - struct st_pagecache_block_link - *next_used, **prev_used; /* to connect links in the LRU chain (ring) */ - struct st_pagecache_block_link - *next_changed, **prev_changed; /* for lists of file dirty/clean blocks */ - struct st_pagecache_hash_link - *hash_link; /* backward ptr to referring hash_link */ - WQUEUE - wqueue[COND_SIZE]; /* queues on waiting requests for new/old pages */ - uint requests; /* number of requests for the block */ - byte *buffer; /* buffer for the block page */ - uint status; /* state of the block */ - uint pins; /* pin counter */ -#ifdef PAGECACHE_DEBUG - PAGECACHE_PIN_INFO *pin_list; - PAGECACHE_LOCK_INFO *lock_list; -#endif - enum BLOCK_TEMPERATURE temperature; /* block temperature: cold, warm, hot */ - enum pagecache_page_type type; /* type of the block */ - uint hits_left; /* number of hits left until promotion */ - ulonglong last_hit_time; /* timestamp of the last hit */ - LSN rec_lsn; /* LSN when first became dirty */ - KEYCACHE_CONDVAR *condvar; /* condition variable for 'no readers' event */ -}; - -#ifdef PAGECACHE_DEBUG -/* debug checks */ -static my_bool info_check_pin(PAGECACHE_BLOCK_LINK *block, - enum pagecache_page_pin mode) -{ - struct st_my_thread_var *thread= my_thread_var; - PAGECACHE_PIN_INFO *info= info_find(block->pin_list, thread); - DBUG_ENTER("info_check_pin"); - if (info) - { - if (mode == PAGECACHE_PIN_LEFT_UNPINNED) - { - DBUG_PRINT("info", - ("info_check_pin: thread: 0x%lx block 0x%lx: LEFT_UNPINNED!!!", - (ulong)thread, (ulong)block)); - DBUG_RETURN(1); - } - else if (mode == PAGECACHE_PIN) - { - DBUG_PRINT("info", - ("info_check_pin: thread: 0x%lx block 0x%lx: PIN!!!", - (ulong)thread, (ulong)block)); - DBUG_RETURN(1); - } - } - else - { - if (mode == PAGECACHE_PIN_LEFT_PINNED) - { - DBUG_PRINT("info", - ("info_check_pin: thread: 0x%lx block 0x%lx: LEFT_PINNED!!!", - (ulong)thread, (ulong)block)); - DBUG_RETURN(1); - } - else if (mode == PAGECACHE_UNPIN) - { - DBUG_PRINT("info", - ("info_check_pin: thread: 0x%lx block 0x%lx: UNPIN!!!", - (ulong)thread, (ulong)block)); - DBUG_RETURN(1); - } - } - DBUG_RETURN(0); -} - - -/* - Debug function which checks current lock/pin state and requested changes - - SYNOPSIS - info_check_lock() - lock requested lock changes - pin requested pin changes - - RETURN - 0 - OK - 1 - Error -*/ - -static my_bool info_check_lock(PAGECACHE_BLOCK_LINK *block, - enum pagecache_page_lock lock, - enum pagecache_page_pin pin) -{ - struct st_my_thread_var *thread= my_thread_var; - PAGECACHE_LOCK_INFO *info= - (PAGECACHE_LOCK_INFO *) info_find((PAGECACHE_PIN_INFO *) block->lock_list, - thread); - DBUG_ENTER("info_check_lock"); - switch(lock) - { - case PAGECACHE_LOCK_LEFT_UNLOCKED: - if (pin != PAGECACHE_PIN_LEFT_UNPINNED || - info) - goto error; - break; - case PAGECACHE_LOCK_LEFT_READLOCKED: - if ((pin != PAGECACHE_PIN_LEFT_UNPINNED && - pin != PAGECACHE_PIN_LEFT_PINNED) || - info == 0 || info->write_lock) - goto error; - break; - case PAGECACHE_LOCK_LEFT_WRITELOCKED: - if (pin != PAGECACHE_PIN_LEFT_PINNED || - info == 0 || !info->write_lock) - goto error; - break; - case PAGECACHE_LOCK_READ: - if ((pin != PAGECACHE_PIN_LEFT_UNPINNED && - pin != PAGECACHE_PIN) || - info != 0) - goto error; - break; - case PAGECACHE_LOCK_WRITE: - if (pin != PAGECACHE_PIN || - info != 0) - goto error; - break; - case PAGECACHE_LOCK_READ_UNLOCK: - if ((pin != PAGECACHE_PIN_LEFT_UNPINNED && - pin != PAGECACHE_UNPIN) || - info == 0 || info->write_lock) - goto error; - break; - case PAGECACHE_LOCK_WRITE_UNLOCK: - if (pin != PAGECACHE_UNPIN || - info == 0 || !info->write_lock) - goto error; - break; - case PAGECACHE_LOCK_WRITE_TO_READ: - if ((pin != PAGECACHE_PIN_LEFT_PINNED && - pin != PAGECACHE_UNPIN) || - info == 0 || !info->write_lock) - goto error; - break; - } - DBUG_RETURN(0); -error: - DBUG_PRINT("info", - ("info_check_lock: thread: 0x%lx block 0x%lx: info: %d wrt: %d," - "to lock: %s, to pin: %s", - (ulong)thread, (ulong)block, test(info), - (info ? info->write_lock : 0), - page_cache_page_lock_str[lock], - page_cache_page_pin_str[pin])); - DBUG_RETURN(1); -} -#endif - -#define FLUSH_CACHE 2000 /* sort this many blocks at once */ - -static void free_block(PAGECACHE *pagecache, PAGECACHE_BLOCK_LINK *block); -static void test_key_cache(PAGECACHE *pagecache, - const char *where, my_bool lock); - -#define PAGECACHE_HASH(p, f, pos) (((ulong) (pos) + \ - (ulong) (f).file) & (p->hash_entries-1)) -#define FILE_HASH(f) ((uint) (f).file & (PAGECACHE_CHANGED_BLOCKS_HASH - 1)) - -#define DEFAULT_PAGECACHE_DEBUG_LOG "pagecache_debug.log" - -#if defined(PAGECACHE_DEBUG) && ! defined(PAGECACHE_DEBUG_LOG) -#define PAGECACHE_DEBUG_LOG DEFAULT_PAGECACHE_DEBUG_LOG -#endif - -#if defined(PAGECACHE_DEBUG_LOG) -static FILE *pagecache_debug_log= NULL; -static void pagecache_debug_print _VARARGS((const char *fmt, ...)); -#define PAGECACHE_DEBUG_OPEN \ - if (!pagecache_debug_log) \ - { \ - pagecache_debug_log= fopen(PAGECACHE_DEBUG_LOG, "w"); \ - (void) setvbuf(pagecache_debug_log, NULL, _IOLBF, BUFSIZ); \ - } - -#define PAGECACHE_DEBUG_CLOSE \ - if (pagecache_debug_log) \ - { \ - fclose(pagecache_debug_log); \ - pagecache_debug_log= 0; \ - } -#else -#define PAGECACHE_DEBUG_OPEN -#define PAGECACHE_DEBUG_CLOSE -#endif /* defined(PAGECACHE_DEBUG_LOG) */ - -#if defined(PAGECACHE_DEBUG_LOG) && defined(PAGECACHE_DEBUG) -#define KEYCACHE_DBUG_PRINT(l, m) \ - { if (pagecache_debug_log) \ - fprintf(pagecache_debug_log, "%s: ", l); \ - pagecache_debug_print m; } - -#define KEYCACHE_DBUG_ASSERT(a) \ - { if (! (a) && pagecache_debug_log) \ - fclose(pagecache_debug_log); \ - assert(a); } -#else -#define KEYCACHE_DBUG_PRINT(l, m) DBUG_PRINT(l, m) -#define KEYCACHE_DBUG_ASSERT(a) DBUG_ASSERT(a) -#endif /* defined(PAGECACHE_DEBUG_LOG) && defined(PAGECACHE_DEBUG) */ - -#if defined(PAGECACHE_DEBUG) || !defined(DBUG_OFF) -#ifdef THREAD -static long pagecache_thread_id; -#define KEYCACHE_THREAD_TRACE(l) \ - KEYCACHE_DBUG_PRINT(l,("|thread %ld",pagecache_thread_id)) - -#define KEYCACHE_THREAD_TRACE_BEGIN(l) \ - { struct st_my_thread_var *thread_var= my_thread_var; \ - pagecache_thread_id= thread_var->id; \ - KEYCACHE_DBUG_PRINT(l,("[thread %ld",pagecache_thread_id)) } - -#define KEYCACHE_THREAD_TRACE_END(l) \ - KEYCACHE_DBUG_PRINT(l,("]thread %ld",pagecache_thread_id)) -#else /* THREAD */ -#define KEYCACHE_THREAD_TRACE(l) KEYCACHE_DBUG_PRINT(l,("")) -#define KEYCACHE_THREAD_TRACE_BEGIN(l) KEYCACHE_DBUG_PRINT(l,("")) -#define KEYCACHE_THREAD_TRACE_END(l) KEYCACHE_DBUG_PRINT(l,("")) -#endif /* THREAD */ -#else -#define KEYCACHE_THREAD_TRACE_BEGIN(l) -#define KEYCACHE_THREAD_TRACE_END(l) -#define KEYCACHE_THREAD_TRACE(l) -#endif /* defined(PAGECACHE_DEBUG) || !defined(DBUG_OFF) */ - -#define BLOCK_NUMBER(p, b) \ - ((uint) (((char*)(b)-(char *) p->block_root)/sizeof(PAGECACHE_BLOCK_LINK))) -#define PAGECACHE_HASH_LINK_NUMBER(p, h) \ - ((uint) (((char*)(h)-(char *) p->hash_link_root)/ \ - sizeof(PAGECACHE_HASH_LINK))) - -#if (defined(PAGECACHE_TIMEOUT) && !defined(__WIN__)) || defined(PAGECACHE_DEBUG) -static int pagecache_pthread_cond_wait(pthread_cond_t *cond, - pthread_mutex_t *mutex); -#else -#define pagecache_pthread_cond_wait pthread_cond_wait -#endif - -#if defined(PAGECACHE_DEBUG) -static int ___pagecache_pthread_mutex_lock(pthread_mutex_t *mutex); -static void ___pagecache_pthread_mutex_unlock(pthread_mutex_t *mutex); -static int ___pagecache_pthread_cond_signal(pthread_cond_t *cond); -#define pagecache_pthread_mutex_lock(M) \ -{ DBUG_PRINT("lock", ("mutex lock 0x%lx %u", (ulong)(M), __LINE__)); \ - ___pagecache_pthread_mutex_lock(M);} -#define pagecache_pthread_mutex_unlock(M) \ -{ DBUG_PRINT("lock", ("mutex unlock 0x%lx %u", (ulong)(M), __LINE__)); \ - ___pagecache_pthread_mutex_unlock(M);} -#define pagecache_pthread_cond_signal(M) \ -{ DBUG_PRINT("lock", ("signal 0x%lx %u", (ulong)(M), __LINE__)); \ - ___pagecache_pthread_cond_signal(M);} -#else -#define pagecache_pthread_mutex_lock pthread_mutex_lock -#define pagecache_pthread_mutex_unlock pthread_mutex_unlock -#define pagecache_pthread_cond_signal pthread_cond_signal -#endif /* defined(PAGECACHE_DEBUG) */ - -extern my_bool translog_flush(LSN lsn); - -/* - Write page to the disk - - SYNOPSIS - pagecache_fwrite() - pagecache - page cache pointer - filedesc - pagecache file descriptor structure - buffer - buffer which we will write - type - page type (plain or with LSN) - flags - MYF() flags - - RETURN - 0 - OK - !=0 - Error -*/ - -static uint pagecache_fwrite(PAGECACHE *pagecache, - PAGECACHE_FILE *filedesc, - byte *buffer, - pgcache_page_no_t pageno, - enum pagecache_page_type type, - myf flags) -{ - DBUG_ENTER("pagecache_fwrite"); - if (type == PAGECACHE_LSN_PAGE) - { - LSN lsn; - DBUG_PRINT("info", ("Log handler call")); - /* TODO: integrate with page format */ -#define PAGE_LSN_OFFSET 0 - lsn= lsn_korr(buffer + PAGE_LSN_OFFSET); - /* - check CONTROL_FILE_IMPOSSIBLE_FILENO & - CONTROL_FILE_IMPOSSIBLE_LOG_OFFSET - */ - DBUG_ASSERT(lsn != 0); - translog_flush(lsn); - } - DBUG_RETURN(my_pwrite(filedesc->file, buffer, pagecache->block_size, - (pageno)<<(pagecache->shift), flags)); -} - - -/* - Read page from the disk - - SYNOPSIS - pagecache_fread() - pagecache - page cache pointer - filedesc - pagecache file descriptor structure - buffer - buffer in which we will read - pageno - page number - flags - MYF() flags -*/ -#define pagecache_fread(pagecache, filedesc, buffer, pageno, flags) \ - my_pread((filedesc)->file, buffer, pagecache->block_size, \ - (pageno)<<(pagecache->shift), flags) - - -/* - next_power(value) is 2 at the power of (1+floor(log2(value))); - e.g. next_power(2)=4, next_power(3)=4. -*/ -static inline uint next_power(uint value) -{ - return (uint) my_round_up_to_next_power((uint32) value) << 1; -} - - -/* - Initialize a page cache - - SYNOPSIS - init_pagecache() - pagecache pointer to a page cache data structure - key_cache_block_size size of blocks to keep cached data - use_mem total memory to use for the key cache - division_limit division limit (may be zero) - age_threshold age threshold (may be zero) - block_size size of block (should be power of 2) - - RETURN VALUE - number of blocks in the key cache, if successful, - 0 - otherwise. - - NOTES. - if pagecache->inited != 0 we assume that the key cache - is already initialized. This is for now used by myisamchk, but shouldn't - be something that a program should rely on! - - It's assumed that no two threads call this function simultaneously - referring to the same key cache handle. - -*/ - -int init_pagecache(PAGECACHE *pagecache, my_size_t use_mem, - uint division_limit, uint age_threshold, - uint block_size) -{ - uint blocks, hash_links, length; - int error; - DBUG_ENTER("init_pagecache"); - DBUG_ASSERT(block_size >= 512); - - PAGECACHE_DEBUG_OPEN; - if (pagecache->inited && pagecache->disk_blocks > 0) - { - DBUG_PRINT("warning",("key cache already in use")); - DBUG_RETURN(0); - } - - pagecache->global_cache_w_requests= pagecache->global_cache_r_requests= 0; - pagecache->global_cache_read= pagecache->global_cache_write= 0; - pagecache->disk_blocks= -1; - if (! pagecache->inited) - { - pagecache->inited= 1; - pagecache->in_init= 0; - pthread_mutex_init(&pagecache->cache_lock, MY_MUTEX_INIT_FAST); - pagecache->resize_queue.last_thread= NULL; - } - - pagecache->mem_size= use_mem; - pagecache->block_size= block_size; - pagecache->shift= my_bit_log2(block_size); - DBUG_PRINT("info", ("block_size: %u", - block_size)); - DBUG_ASSERT(((uint)(1 << pagecache->shift)) == block_size); - - blocks= (int) (use_mem / (sizeof(PAGECACHE_BLOCK_LINK) + - 2 * sizeof(PAGECACHE_HASH_LINK) + - sizeof(PAGECACHE_HASH_LINK*) * - 5/4 + block_size)); - /* It doesn't make sense to have too few blocks (less than 8) */ - if (blocks >= 8 && pagecache->disk_blocks < 0) - { - for ( ; ; ) - { - /* Set my_hash_entries to the next bigger 2 power */ - if ((pagecache->hash_entries= next_power(blocks)) < - (blocks) * 5/4) - pagecache->hash_entries<<= 1; - hash_links= 2 * blocks; -#if defined(MAX_THREADS) - if (hash_links < MAX_THREADS + blocks - 1) - hash_links= MAX_THREADS + blocks - 1; -#endif - while ((length= (ALIGN_SIZE(blocks * sizeof(PAGECACHE_BLOCK_LINK)) + - ALIGN_SIZE(hash_links * sizeof(PAGECACHE_HASH_LINK)) + - ALIGN_SIZE(sizeof(PAGECACHE_HASH_LINK*) * - pagecache->hash_entries))) + - (((ulong) blocks) << pagecache->shift) > use_mem) - blocks--; - /* Allocate memory for cache page buffers */ - if ((pagecache->block_mem= - my_large_malloc((ulong) blocks * pagecache->block_size, - MYF(MY_WME)))) - { - /* - Allocate memory for blocks, hash_links and hash entries; - For each block 2 hash links are allocated - */ - if ((pagecache->block_root= - (PAGECACHE_BLOCK_LINK*) my_malloc((uint) length, - MYF(0)))) - break; - my_large_free(pagecache->block_mem, MYF(0)); - pagecache->block_mem= 0; - } - if (blocks < 8) - { - my_errno= ENOMEM; - goto err; - } - blocks= blocks / 4*3; - } - pagecache->blocks_unused= (ulong) blocks; - pagecache->disk_blocks= (int) blocks; - pagecache->hash_links= hash_links; - pagecache->hash_root= - (PAGECACHE_HASH_LINK**) ((char*) pagecache->block_root + - ALIGN_SIZE(blocks*sizeof(PAGECACHE_BLOCK_LINK))); - pagecache->hash_link_root= - (PAGECACHE_HASH_LINK*) ((char*) pagecache->hash_root + - ALIGN_SIZE((sizeof(PAGECACHE_HASH_LINK*) * - pagecache->hash_entries))); - bzero((byte*) pagecache->block_root, - pagecache->disk_blocks * sizeof(PAGECACHE_BLOCK_LINK)); - bzero((byte*) pagecache->hash_root, - pagecache->hash_entries * sizeof(PAGECACHE_HASH_LINK*)); - bzero((byte*) pagecache->hash_link_root, - pagecache->hash_links * sizeof(PAGECACHE_HASH_LINK)); - pagecache->hash_links_used= 0; - pagecache->free_hash_list= NULL; - pagecache->blocks_used= pagecache->blocks_changed= 0; - - pagecache->global_blocks_changed= 0; - pagecache->blocks_available=0; /* For debugging */ - - /* The LRU chain is empty after initialization */ - pagecache->used_last= NULL; - pagecache->used_ins= NULL; - pagecache->free_block_list= NULL; - pagecache->time= 0; - pagecache->warm_blocks= 0; - pagecache->min_warm_blocks= (division_limit ? - blocks * division_limit / 100 + 1 : - blocks); - pagecache->age_threshold= (age_threshold ? - blocks * age_threshold / 100 : - blocks); - - pagecache->cnt_for_resize_op= 0; - pagecache->resize_in_flush= 0; - pagecache->can_be_used= 1; - - pagecache->waiting_for_hash_link.last_thread= NULL; - pagecache->waiting_for_block.last_thread= NULL; - DBUG_PRINT("exit", - ("disk_blocks: %d block_root: 0x%lx hash_entries: %d\ - hash_root: 0x%lx hash_links: %d hash_link_root: 0x%lx", - pagecache->disk_blocks, (long) pagecache->block_root, - pagecache->hash_entries, (long) pagecache->hash_root, - pagecache->hash_links, (long) pagecache->hash_link_root)); - bzero((gptr) pagecache->changed_blocks, - sizeof(pagecache->changed_blocks[0]) * - PAGECACHE_CHANGED_BLOCKS_HASH); - bzero((gptr) pagecache->file_blocks, - sizeof(pagecache->file_blocks[0]) * - PAGECACHE_CHANGED_BLOCKS_HASH); - } - - pagecache->blocks= pagecache->disk_blocks > 0 ? pagecache->disk_blocks : 0; - DBUG_RETURN((uint) pagecache->blocks); - -err: - error= my_errno; - pagecache->disk_blocks= 0; - pagecache->blocks= 0; - if (pagecache->block_mem) - { - my_large_free((gptr) pagecache->block_mem, MYF(0)); - pagecache->block_mem= NULL; - } - if (pagecache->block_root) - { - my_free((gptr) pagecache->block_root, MYF(0)); - pagecache->block_root= NULL; - } - my_errno= error; - pagecache->can_be_used= 0; - DBUG_RETURN(0); -} - - -/* - Flush all blocks in the key cache to disk -*/ - -#ifdef NOT_USED -static int flush_all_key_blocks(PAGECACHE *pagecache) -{ -#if defined(PAGECACHE_DEBUG) - uint cnt=0; -#endif - while (pagecache->blocks_changed > 0) - { - PAGECACHE_BLOCK_LINK *block; - for (block= pagecache->used_last->next_used ; ; block=block->next_used) - { - if (block->hash_link) - { -#if defined(PAGECACHE_DEBUG) - cnt++; - KEYCACHE_DBUG_ASSERT(cnt <= pagecache->blocks_used); -#endif - if (flush_pagecache_blocks_int(pagecache, &block->hash_link->file, - FLUSH_RELEASE)) - return 1; - break; - } - if (block == pagecache->used_last) - break; - } - } - return 0; -} -#endif /* NOT_USED */ - -/* - Resize a key cache - - SYNOPSIS - resize_pagecache() - pagecache pointer to a page cache data structure - use_mem total memory to use for the new key cache - division_limit new division limit (if not zero) - age_threshold new age threshold (if not zero) - - RETURN VALUE - number of blocks in the key cache, if successful, - 0 - otherwise. - - NOTES. - The function first compares the memory size parameter - with the key cache value. - - If they differ the function free the the memory allocated for the - old key cache blocks by calling the end_pagecache function and - then rebuilds the key cache with new blocks by calling - init_key_cache. - - The function starts the operation only when all other threads - performing operations with the key cache let her to proceed - (when cnt_for_resize=0). - - Before being usable, this function needs: - - to receive fixes for BUG#17332 "changing key_buffer_size on a running - server can crash under load" similar to those done to the key cache - - to have us (Sanja) look at the additional constraints placed on - resizing, due to the page locking specific to this page cache. - So we disable it for now. -*/ -#if NOT_USED /* keep disabled until code is fixed see above !! */ -int resize_pagecache(PAGECACHE *pagecache, - my_size_t use_mem, uint division_limit, - uint age_threshold) -{ - int blocks; -#ifdef THREAD - struct st_my_thread_var *thread; - WQUEUE *wqueue; - -#endif - DBUG_ENTER("resize_pagecache"); - - if (!pagecache->inited) - DBUG_RETURN(pagecache->disk_blocks); - - if(use_mem == pagecache->mem_size) - { - change_pagecache_param(pagecache, division_limit, age_threshold); - DBUG_RETURN(pagecache->disk_blocks); - } - - pagecache_pthread_mutex_lock(&pagecache->cache_lock); - -#ifdef THREAD - wqueue= &pagecache->resize_queue; - thread= my_thread_var; - wqueue_link_into_queue(wqueue, thread); - - while (wqueue->last_thread->next != thread) - { - pagecache_pthread_cond_wait(&thread->suspend, &pagecache->cache_lock); - } -#endif - - pagecache->resize_in_flush= 1; - if (flush_all_key_blocks(pagecache)) - { - /* TODO: if this happens, we should write a warning in the log file ! */ - pagecache->resize_in_flush= 0; - blocks= 0; - pagecache->can_be_used= 0; - goto finish; - } - pagecache->resize_in_flush= 0; - pagecache->can_be_used= 0; -#ifdef THREAD - while (pagecache->cnt_for_resize_op) - { - KEYCACHE_DBUG_PRINT("resize_pagecache: wait", - ("suspend thread %ld", thread->id)); - pagecache_pthread_cond_wait(&thread->suspend, &pagecache->cache_lock); - } -#else - KEYCACHE_DBUG_ASSERT(pagecache->cnt_for_resize_op == 0); -#endif - - end_pagecache(pagecache, 0); /* Don't free mutex */ - /* The following will work even if use_mem is 0 */ - blocks= init_pagecache(pagecache, pagecache->block_size, use_mem, - division_limit, age_threshold); - -finish: -#ifdef THREAD - wqueue_unlink_from_queue(wqueue, thread); - /* Signal for the next resize request to proceeed if any */ - if (wqueue->last_thread) - { - KEYCACHE_DBUG_PRINT("resize_pagecache: signal", - ("thread %ld", wqueue->last_thread->next->id)); - pagecache_pthread_cond_signal(&wqueue->last_thread->next->suspend); - } -#endif - pagecache_pthread_mutex_unlock(&pagecache->cache_lock); - DBUG_RETURN(blocks); -} -#endif /* 0 */ - - -/* - Increment counter blocking resize key cache operation -*/ -static inline void inc_counter_for_resize_op(PAGECACHE *pagecache) -{ - pagecache->cnt_for_resize_op++; -} - - -/* - Decrement counter blocking resize key cache operation; - Signal the operation to proceed when counter becomes equal zero -*/ -static inline void dec_counter_for_resize_op(PAGECACHE *pagecache) -{ -#ifdef THREAD - struct st_my_thread_var *last_thread; - if (!--pagecache->cnt_for_resize_op && - (last_thread= pagecache->resize_queue.last_thread)) - { - KEYCACHE_DBUG_PRINT("dec_counter_for_resize_op: signal", - ("thread %ld", last_thread->next->id)); - pagecache_pthread_cond_signal(&last_thread->next->suspend); - } -#else - pagecache->cnt_for_resize_op--; -#endif -} - -/* - Change the page cache parameters - - SYNOPSIS - change_pagecache_param() - pagecache pointer to a page cache data structure - division_limit new division limit (if not zero) - age_threshold new age threshold (if not zero) - - RETURN VALUE - none - - NOTES. - Presently the function resets the key cache parameters - concerning midpoint insertion strategy - division_limit and - age_threshold. -*/ - -void change_pagecache_param(PAGECACHE *pagecache, uint division_limit, - uint age_threshold) -{ - DBUG_ENTER("change_pagecache_param"); - - pagecache_pthread_mutex_lock(&pagecache->cache_lock); - if (division_limit) - pagecache->min_warm_blocks= (pagecache->disk_blocks * - division_limit / 100 + 1); - if (age_threshold) - pagecache->age_threshold= (pagecache->disk_blocks * - age_threshold / 100); - pagecache_pthread_mutex_unlock(&pagecache->cache_lock); - DBUG_VOID_RETURN; -} - - -/* - Removes page cache from memory. Does NOT flush pages to disk. - - SYNOPSIS - end_pagecache() - pagecache page cache handle - cleanup Complete free (Free also mutex for key cache) - - RETURN VALUE - none -*/ - -void end_pagecache(PAGECACHE *pagecache, my_bool cleanup) -{ - DBUG_ENTER("end_pagecache"); - DBUG_PRINT("enter", ("key_cache: 0x%lx", (long) pagecache)); - - if (!pagecache->inited) - DBUG_VOID_RETURN; - - if (pagecache->disk_blocks > 0) - { - if (pagecache->block_mem) - { - my_large_free((gptr) pagecache->block_mem, MYF(0)); - pagecache->block_mem= NULL; - my_free((gptr) pagecache->block_root, MYF(0)); - pagecache->block_root= NULL; - } - pagecache->disk_blocks= -1; - /* Reset blocks_changed to be safe if flush_all_key_blocks is called */ - pagecache->blocks_changed= 0; - } - - DBUG_PRINT("status", ("used: %lu changed: %lu w_requests: %lu " - "writes: %lu r_requests: %lu reads: %lu", - pagecache->blocks_used, pagecache->global_blocks_changed, - (ulong) pagecache->global_cache_w_requests, - (ulong) pagecache->global_cache_write, - (ulong) pagecache->global_cache_r_requests, - (ulong) pagecache->global_cache_read)); - - if (cleanup) - { - pthread_mutex_destroy(&pagecache->cache_lock); - pagecache->inited= pagecache->can_be_used= 0; - PAGECACHE_DEBUG_CLOSE; - } - DBUG_VOID_RETURN; -} /* end_pagecache */ - - -/* - Unlink a block from the chain of dirty/clean blocks -*/ - -static inline void unlink_changed(PAGECACHE_BLOCK_LINK *block) -{ - if (block->next_changed) - block->next_changed->prev_changed= block->prev_changed; - *block->prev_changed= block->next_changed; -} - - -/* - Link a block into the chain of dirty/clean blocks -*/ - -static inline void link_changed(PAGECACHE_BLOCK_LINK *block, - PAGECACHE_BLOCK_LINK **phead) -{ - block->prev_changed= phead; - if ((block->next_changed= *phead)) - (*phead)->prev_changed= &block->next_changed; - *phead= block; -} - - -/* - Unlink a block from the chain of dirty/clean blocks, if it's asked for, - and link it to the chain of clean blocks for the specified file -*/ - -static void link_to_file_list(PAGECACHE *pagecache, - PAGECACHE_BLOCK_LINK *block, - PAGECACHE_FILE *file, my_bool unlink) -{ - if (unlink) - unlink_changed(block); - link_changed(block, &pagecache->file_blocks[FILE_HASH(*file)]); - if (block->status & BLOCK_CHANGED) - { - block->status&= ~BLOCK_CHANGED; - block->rec_lsn= 0; - pagecache->blocks_changed--; - pagecache->global_blocks_changed--; - } -} - - -/* - Unlink a block from the chain of clean blocks for the specified - file and link it to the chain of dirty blocks for this file -*/ - -static inline void link_to_changed_list(PAGECACHE *pagecache, - PAGECACHE_BLOCK_LINK *block) -{ - unlink_changed(block); - link_changed(block, - &pagecache->changed_blocks[FILE_HASH(block->hash_link->file)]); - block->status|=BLOCK_CHANGED; - pagecache->blocks_changed++; - pagecache->global_blocks_changed++; -} - - -/* - Link a block to the LRU chain at the beginning or at the end of - one of two parts. - - SYNOPSIS - link_block() - pagecache pointer to a page cache data structure - block pointer to the block to link to the LRU chain - hot <-> to link the block into the hot subchain - at_end <-> to link the block at the end of the subchain - - RETURN VALUE - none - - NOTES. - The LRU chain is represented by a curcular list of block structures. - The list is double-linked of the type (**prev,*next) type. - The LRU chain is divided into two parts - hot and warm. - There are two pointers to access the last blocks of these two - parts. The beginning of the warm part follows right after the - end of the hot part. - Only blocks of the warm part can be used for replacement. - The first block from the beginning of this subchain is always - taken for eviction (pagecache->last_used->next) - - LRU chain: +------+ H O T +------+ - +----| end |----...<----| beg |----+ - | +------+last +------+ | - v<-link in latest hot (new end) | - | link in latest warm (new end)->^ - | +------+ W A R M +------+ | - +----| beg |---->...----| end |----+ - +------+ +------+ins - first for eviction -*/ - -static void link_block(PAGECACHE *pagecache, PAGECACHE_BLOCK_LINK *block, - my_bool hot, my_bool at_end) -{ - PAGECACHE_BLOCK_LINK *ins; - PAGECACHE_BLOCK_LINK **ptr_ins; - - BLOCK_INFO(block); - KEYCACHE_DBUG_ASSERT(! (block->hash_link && block->hash_link->requests)); -#ifdef THREAD - if (!hot && pagecache->waiting_for_block.last_thread) - { - /* Signal that in the LRU warm sub-chain an available block has appeared */ - struct st_my_thread_var *last_thread= - pagecache->waiting_for_block.last_thread; - struct st_my_thread_var *first_thread= last_thread->next; - struct st_my_thread_var *next_thread= first_thread; - PAGECACHE_HASH_LINK *hash_link= - (PAGECACHE_HASH_LINK *) first_thread->opt_info; - struct st_my_thread_var *thread; - do - { - thread= next_thread; - next_thread= thread->next; - /* - We notify about the event all threads that ask - for the same page as the first thread in the queue - */ - if ((PAGECACHE_HASH_LINK *) thread->opt_info == hash_link) - { - KEYCACHE_DBUG_PRINT("link_block: signal", ("thread %ld", thread->id)); - pagecache_pthread_cond_signal(&thread->suspend); - wqueue_unlink_from_queue(&pagecache->waiting_for_block, thread); - block->requests++; - } - } - while (thread != last_thread); - hash_link->block= block; - KEYCACHE_THREAD_TRACE("link_block: after signaling"); -#if defined(PAGECACHE_DEBUG) - KEYCACHE_DBUG_PRINT("link_block", - ("linked,unlinked block %u status=%x #requests=%u #available=%u", - BLOCK_NUMBER(pagecache, block), block->status, - block->requests, pagecache->blocks_available)); -#endif - return; - } -#else /* THREAD */ - KEYCACHE_DBUG_ASSERT(! (!hot && pagecache->waiting_for_block.last_thread)); - /* Condition not transformed using DeMorgan, to keep the text identical */ -#endif /* THREAD */ - ptr_ins= hot ? &pagecache->used_ins : &pagecache->used_last; - ins= *ptr_ins; - if (ins) - { - ins->next_used->prev_used= &block->next_used; - block->next_used= ins->next_used; - block->prev_used= &ins->next_used; - ins->next_used= block; - if (at_end) - *ptr_ins= block; - } - else - { - /* The LRU chain is empty */ - pagecache->used_last= pagecache->used_ins= block->next_used= block; - block->prev_used= &block->next_used; - } - KEYCACHE_THREAD_TRACE("link_block"); -#if defined(PAGECACHE_DEBUG) - pagecache->blocks_available++; - KEYCACHE_DBUG_PRINT("link_block", - ("linked block %u:%1u status=%x #requests=%u #available=%u", - BLOCK_NUMBER(pagecache, block), at_end, block->status, - block->requests, pagecache->blocks_available)); - KEYCACHE_DBUG_ASSERT((ulong) pagecache->blocks_available <= - pagecache->blocks_used); -#endif -} - - -/* - Unlink a block from the LRU chain - - SYNOPSIS - unlink_block() - pagecache pointer to a page cache data structure - block pointer to the block to unlink from the LRU chain - - RETURN VALUE - none - - NOTES. - See NOTES for link_block -*/ - -static void unlink_block(PAGECACHE *pagecache, PAGECACHE_BLOCK_LINK *block) -{ - DBUG_ENTER("unlink_block"); - DBUG_PRINT("unlink_block", ("unlink 0x%lx", (ulong)block)); - if (block->next_used == block) - /* The list contains only one member */ - pagecache->used_last= pagecache->used_ins= NULL; - else - { - block->next_used->prev_used= block->prev_used; - *block->prev_used= block->next_used; - if (pagecache->used_last == block) - pagecache->used_last= STRUCT_PTR(PAGECACHE_BLOCK_LINK, - next_used, block->prev_used); - if (pagecache->used_ins == block) - pagecache->used_ins= STRUCT_PTR(PAGECACHE_BLOCK_LINK, - next_used, block->prev_used); - } - block->next_used= NULL; - - KEYCACHE_THREAD_TRACE("unlink_block"); -#if defined(PAGECACHE_DEBUG) - KEYCACHE_DBUG_ASSERT(pagecache->blocks_available != 0); - pagecache->blocks_available--; - KEYCACHE_DBUG_PRINT("unlink_block", - ("unlinked block 0x%lx (%u) status=%x #requests=%u #available=%u", - (ulong)block, BLOCK_NUMBER(pagecache, block), block->status, - block->requests, pagecache->blocks_available)); - BLOCK_INFO(block); -#endif - DBUG_VOID_RETURN; -} - - -/* - Register requests for a block - - SYNOPSIS - reg_requests() - pagecache this page cache reference - block the block we request reference - count how many requests we register (it is 1 everywhere) - - NOTE - Registration of request means we are going to use this block so we exclude - it from the LRU if it is first request -*/ -static void reg_requests(PAGECACHE *pagecache, PAGECACHE_BLOCK_LINK *block, - int count) -{ - DBUG_ENTER("reg_requests"); - DBUG_PRINT("enter", ("block 0x%lx (%u) status=%x, reqs: %u", - (ulong)block, BLOCK_NUMBER(pagecache, block), - block->status, block->requests)); - BLOCK_INFO(block); - if (! block->requests) - /* First request for the block unlinks it */ - unlink_block(pagecache, block); - block->requests+= count; - DBUG_VOID_RETURN; -} - - -/* - Unregister request for a block - linking it to the LRU chain if it's the last request - - SYNOPSIS - unreg_request() - pagecache pointer to a page cache data structure - block pointer to the block to link to the LRU chain - at_end <-> to link the block at the end of the LRU chain - - RETURN VALUE - none - - NOTES. - Every linking to the LRU chain decrements by one a special block - counter (if it's positive). If the at_end parameter is TRUE the block is - added either at the end of warm sub-chain or at the end of hot sub-chain. - It is added to the hot subchain if its counter is zero and number of - blocks in warm sub-chain is not less than some low limit (determined by - the division_limit parameter). Otherwise the block is added to the warm - sub-chain. If the at_end parameter is FALSE the block is always added - at beginning of the warm sub-chain. - Thus a warm block can be promoted to the hot sub-chain when its counter - becomes zero for the first time. - At the same time the block at the very beginning of the hot subchain - might be moved to the beginning of the warm subchain if it stays untouched - for a too long time (this time is determined by parameter age_threshold). -*/ - -static void unreg_request(PAGECACHE *pagecache, - PAGECACHE_BLOCK_LINK *block, int at_end) -{ - DBUG_ENTER("unreg_request"); - DBUG_PRINT("enter", ("block 0x%lx (%u) status=%x, reqs: %u", - (ulong)block, BLOCK_NUMBER(pagecache, block), - block->status, block->requests)); - BLOCK_INFO(block); - DBUG_ASSERT(block->requests > 0); - if (! --block->requests) - { - my_bool hot; - if (block->hits_left) - block->hits_left--; - hot= !block->hits_left && at_end && - pagecache->warm_blocks > pagecache->min_warm_blocks; - if (hot) - { - if (block->temperature == BLOCK_WARM) - pagecache->warm_blocks--; - block->temperature= BLOCK_HOT; - KEYCACHE_DBUG_PRINT("unreg_request", ("#warm_blocks: %lu", - pagecache->warm_blocks)); - } - link_block(pagecache, block, hot, (my_bool)at_end); - block->last_hit_time= pagecache->time; - pagecache->time++; - - block= pagecache->used_ins; - /* Check if we should link a hot block to the warm block */ - if (block && pagecache->time - block->last_hit_time > - pagecache->age_threshold) - { - unlink_block(pagecache, block); - link_block(pagecache, block, 0, 0); - if (block->temperature != BLOCK_WARM) - { - pagecache->warm_blocks++; - block->temperature= BLOCK_WARM; - } - KEYCACHE_DBUG_PRINT("unreg_request", ("#warm_blocks: %lu", - pagecache->warm_blocks)); - } - } - DBUG_VOID_RETURN; -} - -/* - Remove a reader of the page in block -*/ - -static inline void remove_reader(PAGECACHE_BLOCK_LINK *block) -{ - DBUG_ENTER("remove_reader"); - BLOCK_INFO(block); - DBUG_ASSERT(block->hash_link->requests > 0); -#ifdef THREAD - if (! --block->hash_link->requests && block->condvar) - pagecache_pthread_cond_signal(block->condvar); -#else - --block->hash_link->requests; -#endif - DBUG_VOID_RETURN; -} - - -/* - Wait until the last reader of the page in block - signals on its termination -*/ - -static inline void wait_for_readers(PAGECACHE *pagecache - __attribute__((unused)), - PAGECACHE_BLOCK_LINK *block) -{ -#ifdef THREAD - struct st_my_thread_var *thread= my_thread_var; - while (block->hash_link->requests) - { - KEYCACHE_DBUG_PRINT("wait_for_readers: wait", - ("suspend thread %ld block %u", - thread->id, BLOCK_NUMBER(pagecache, block))); - block->condvar= &thread->suspend; - pagecache_pthread_cond_wait(&thread->suspend, &pagecache->cache_lock); - block->condvar= NULL; - } -#else - KEYCACHE_DBUG_ASSERT(block->hash_link->requests == 0); -#endif -} - - -/* - Add a hash link to a bucket in the hash_table -*/ - -static inline void link_hash(PAGECACHE_HASH_LINK **start, - PAGECACHE_HASH_LINK *hash_link) -{ - if (*start) - (*start)->prev= &hash_link->next; - hash_link->next= *start; - hash_link->prev= start; - *start= hash_link; -} - - -/* - Remove a hash link from the hash table -*/ - -static void unlink_hash(PAGECACHE *pagecache, PAGECACHE_HASH_LINK *hash_link) -{ - KEYCACHE_DBUG_PRINT("unlink_hash", ("fd: %u pos_ %lu #requests=%u", - (uint) hash_link->file.file, (ulong) hash_link->pageno, - hash_link->requests)); - KEYCACHE_DBUG_ASSERT(hash_link->requests == 0); - if ((*hash_link->prev= hash_link->next)) - hash_link->next->prev= hash_link->prev; - hash_link->block= NULL; -#ifdef THREAD - if (pagecache->waiting_for_hash_link.last_thread) - { - /* Signal that a free hash link has appeared */ - struct st_my_thread_var *last_thread= - pagecache->waiting_for_hash_link.last_thread; - struct st_my_thread_var *first_thread= last_thread->next; - struct st_my_thread_var *next_thread= first_thread; - PAGECACHE_PAGE *first_page= (PAGECACHE_PAGE *) (first_thread->opt_info); - struct st_my_thread_var *thread; - - hash_link->file= first_page->file; - hash_link->pageno= first_page->pageno; - do - { - PAGECACHE_PAGE *page; - thread= next_thread; - page= (PAGECACHE_PAGE *) thread->opt_info; - next_thread= thread->next; - /* - We notify about the event all threads that ask - for the same page as the first thread in the queue - */ - if (page->file.file == hash_link->file.file && - page->pageno == hash_link->pageno) - { - KEYCACHE_DBUG_PRINT("unlink_hash: signal", ("thread %ld", thread->id)); - pagecache_pthread_cond_signal(&thread->suspend); - wqueue_unlink_from_queue(&pagecache->waiting_for_hash_link, thread); - } - } - while (thread != last_thread); - link_hash(&pagecache->hash_root[PAGECACHE_HASH(pagecache, - hash_link->file, - hash_link->pageno)], - hash_link); - return; - } -#else /* THREAD */ - KEYCACHE_DBUG_ASSERT(! (pagecache->waiting_for_hash_link.last_thread)); -#endif /* THREAD */ - hash_link->next= pagecache->free_hash_list; - pagecache->free_hash_list= hash_link; -} - - -/* - Get the hash link for the page if it is in the cache (do not put the - page in the cache if it is absent there) - - SYNOPSIS - get_present_hash_link() - pagecache Pagecache reference - file file ID - pageno page number in the file - start where to put pointer to found hash bucket (for - direct referring it) - - RETURN - found hashlink pointer -*/ - -static PAGECACHE_HASH_LINK *get_present_hash_link(PAGECACHE *pagecache, - PAGECACHE_FILE *file, - pgcache_page_no_t pageno, - PAGECACHE_HASH_LINK ***start) -{ - reg1 PAGECACHE_HASH_LINK *hash_link; -#if defined(PAGECACHE_DEBUG) - int cnt; -#endif - DBUG_ENTER("get_present_hash_link"); - - KEYCACHE_DBUG_PRINT("get_present_hash_link", ("fd: %u pos: %lu", - (uint) file->file, (ulong) pageno)); - - /* - Find the bucket in the hash table for the pair (file, pageno); - start contains the head of the bucket list, - hash_link points to the first member of the list - */ - hash_link= *(*start= &pagecache->hash_root[PAGECACHE_HASH(pagecache, - *file, pageno)]); -#if defined(PAGECACHE_DEBUG) - cnt= 0; -#endif - /* Look for an element for the pair (file, pageno) in the bucket chain */ - while (hash_link && - (hash_link->pageno != pageno || - hash_link->file.file != file->file)) - { - hash_link= hash_link->next; -#if defined(PAGECACHE_DEBUG) - cnt++; - if (! (cnt <= pagecache->hash_links_used)) - { - int i; - for (i=0, hash_link= **start ; - i < cnt ; i++, hash_link= hash_link->next) - { - KEYCACHE_DBUG_PRINT("get_present_hash_link", ("fd: %u pos: %lu", - (uint) hash_link->file.file, (ulong) hash_link->pageno)); - } - } - KEYCACHE_DBUG_ASSERT(cnt <= pagecache->hash_links_used); -#endif - } - if (hash_link) - { - /* Register the request for the page */ - hash_link->requests++; - } - - DBUG_RETURN(hash_link); -} - - -/* - Get the hash link for a page -*/ - -static PAGECACHE_HASH_LINK *get_hash_link(PAGECACHE *pagecache, - PAGECACHE_FILE *file, - pgcache_page_no_t pageno) -{ - reg1 PAGECACHE_HASH_LINK *hash_link; - PAGECACHE_HASH_LINK **start; - - KEYCACHE_DBUG_PRINT("get_hash_link", ("fd: %u pos: %lu", - (uint) file->file, (ulong) pageno)); - -restart: - /* try to find the page in the cache */ - hash_link= get_present_hash_link(pagecache, file, pageno, - &start); - if (!hash_link) - { - /* There is no hash link in the hash table for the pair (file, pageno) */ - if (pagecache->free_hash_list) - { - hash_link= pagecache->free_hash_list; - pagecache->free_hash_list= hash_link->next; - } - else if (pagecache->hash_links_used < pagecache->hash_links) - { - hash_link= &pagecache->hash_link_root[pagecache->hash_links_used++]; - } - else - { -#ifdef THREAD - /* Wait for a free hash link */ - struct st_my_thread_var *thread= my_thread_var; - PAGECACHE_PAGE page; - KEYCACHE_DBUG_PRINT("get_hash_link", ("waiting")); - page.file= *file; - page.pageno= pageno; - thread->opt_info= (void *) &page; - wqueue_link_into_queue(&pagecache->waiting_for_hash_link, thread); - KEYCACHE_DBUG_PRINT("get_hash_link: wait", - ("suspend thread %ld", thread->id)); - pagecache_pthread_cond_wait(&thread->suspend, - &pagecache->cache_lock); - thread->opt_info= NULL; -#else - KEYCACHE_DBUG_ASSERT(0); -#endif - DBUG_PRINT("info", ("restarting...")); - goto restart; - } - hash_link->file= *file; - hash_link->pageno= pageno; - link_hash(start, hash_link); - /* Register the request for the page */ - hash_link->requests++; - } - - return hash_link; -} - - -/* - Get a block for the file page requested by a pagecache read/write operation; - If the page is not in the cache return a free block, if there is none - return the lru block after saving its buffer if the page is dirty. - - SYNOPSIS - - find_block() - pagecache pointer to a page cache data structure - file handler for the file to read page from - pageno number of the page in the file - init_hits_left how initialize the block counter for the page - wrmode <-> get for writing - reg_req Register request to thye page - page_st out {PAGE_READ,PAGE_TO_BE_READ,PAGE_WAIT_TO_BE_READ} - - RETURN VALUE - Pointer to the found block if successful, 0 - otherwise - - NOTES. - For the page from file positioned at pageno the function checks whether - the page is in the key cache specified by the first parameter. - If this is the case it immediately returns the block. - If not, the function first chooses a block for this page. If there is - no not used blocks in the key cache yet, the function takes the block - at the very beginning of the warm sub-chain. It saves the page in that - block if it's dirty before returning the pointer to it. - The function returns in the page_st parameter the following values: - PAGE_READ - if page already in the block, - PAGE_TO_BE_READ - if it is to be read yet by the current thread - WAIT_TO_BE_READ - if it is to be read by another thread - If an error occurs THE BLOCK_ERROR bit is set in the block status. - It might happen that there are no blocks in LRU chain (in warm part) - - all blocks are unlinked for some read/write operations. Then the function - waits until first of this operations links any block back. -*/ - -static PAGECACHE_BLOCK_LINK *find_block(PAGECACHE *pagecache, - PAGECACHE_FILE *file, - pgcache_page_no_t pageno, - int init_hits_left, - my_bool wrmode, - my_bool reg_req, - int *page_st) -{ - PAGECACHE_HASH_LINK *hash_link; - PAGECACHE_BLOCK_LINK *block; - int error= 0; - int page_status; - - DBUG_ENTER("find_block"); - KEYCACHE_THREAD_TRACE("find_block:begin"); - DBUG_PRINT("enter", ("fd: %d pos: %lu wrmode: %d", - file->file, (ulong) pageno, wrmode)); - KEYCACHE_DBUG_PRINT("find_block", ("fd: %d pos: %lu wrmode: %d", - file->file, (ulong) pageno, - wrmode)); -#if !defined(DBUG_OFF) && defined(EXTRA_DEBUG) - DBUG_EXECUTE("check_pagecache", - test_key_cache(pagecache, "start of find_block", 0);); -#endif - -restart: - /* Find the hash link for the requested page (file, pageno) */ - hash_link= get_hash_link(pagecache, file, pageno); - - page_status= -1; - if ((block= hash_link->block) && - block->hash_link == hash_link && (block->status & BLOCK_READ)) - page_status= PAGE_READ; - - if (wrmode && pagecache->resize_in_flush) - { - /* This is a write request during the flush phase of a resize operation */ - - if (page_status != PAGE_READ) - { - /* We don't need the page in the cache: we are going to write on disk */ - DBUG_ASSERT(hash_link->requests > 0); - hash_link->requests--; - unlink_hash(pagecache, hash_link); - return 0; - } - if (!(block->status & BLOCK_IN_FLUSH)) - { - DBUG_ASSERT(hash_link->requests > 0); - hash_link->requests--; - /* - Remove block to invalidate the page in the block buffer - as we are going to write directly on disk. - Although we have an exclusive lock for the updated key part - the control can be yielded by the current thread as we might - have unfinished readers of other key parts in the block - buffer. Still we are guaranteed not to have any readers - of the key part we are writing into until the block is - removed from the cache as we set the BLOCK_REASSIGNED - flag (see the code below that handles reading requests). - */ - free_block(pagecache, block); - return 0; - } - /* Wait until the page is flushed on disk */ - DBUG_ASSERT(hash_link->requests > 0); - hash_link->requests--; - { -#ifdef THREAD - struct st_my_thread_var *thread= my_thread_var; - wqueue_add_to_queue(&block->wqueue[COND_FOR_SAVED], thread); - do - { - KEYCACHE_DBUG_PRINT("find_block: wait", - ("suspend thread %ld", thread->id)); - pagecache_pthread_cond_wait(&thread->suspend, - &pagecache->cache_lock); - } - while(thread->next); -#else - KEYCACHE_DBUG_ASSERT(0); - /* - Given the use of "resize_in_flush", it seems impossible - that this whole branch is ever entered in single-threaded case - because "(wrmode && pagecache->resize_in_flush)" cannot be true. - TODO: Check this, and then put the whole branch into the - "#ifdef THREAD" guard. - */ -#endif - } - /* Invalidate page in the block if it has not been done yet */ - if (block->status) - free_block(pagecache, block); - return 0; - } - - if (page_status == PAGE_READ && - (block->status & (BLOCK_IN_SWITCH | BLOCK_REASSIGNED))) - { - /* This is a request for a page to be removed from cache */ - - KEYCACHE_DBUG_PRINT("find_block", - ("request for old page in block %u " - "wrmode: %d block->status: %d", - BLOCK_NUMBER(pagecache, block), wrmode, - block->status)); - /* - Only reading requests can proceed until the old dirty page is flushed, - all others are to be suspended, then resubmitted - */ - if (!wrmode && !(block->status & BLOCK_REASSIGNED)) - { - if (reg_req) - reg_requests(pagecache, block, 1); - } - else - { - DBUG_ASSERT(hash_link->requests > 0); - hash_link->requests--; - KEYCACHE_DBUG_PRINT("find_block", - ("request waiting for old page to be saved")); - { -#ifdef THREAD - struct st_my_thread_var *thread= my_thread_var; - /* Put the request into the queue of those waiting for the old page */ - wqueue_add_to_queue(&block->wqueue[COND_FOR_SAVED], thread); - /* Wait until the request can be resubmitted */ - do - { - KEYCACHE_DBUG_PRINT("find_block: wait", - ("suspend thread %ld", thread->id)); - pagecache_pthread_cond_wait(&thread->suspend, - &pagecache->cache_lock); - } - while(thread->next); -#else - KEYCACHE_DBUG_ASSERT(0); - /* No parallel requests in single-threaded case */ -#endif - } - KEYCACHE_DBUG_PRINT("find_block", - ("request for old page resubmitted")); - DBUG_PRINT("info", ("restarting...")); - /* Resubmit the request */ - goto restart; - } - block->status&= ~BLOCK_IN_SWITCH; - } - else - { - /* This is a request for a new page or for a page not to be removed */ - if (! block) - { - /* No block is assigned for the page yet */ - if (pagecache->blocks_unused) - { - if (pagecache->free_block_list) - { - /* There is a block in the free list. */ - block= pagecache->free_block_list; - pagecache->free_block_list= block->next_used; - block->next_used= NULL; - } - else - { - /* There are some never used blocks, take first of them */ - block= &pagecache->block_root[pagecache->blocks_used]; - block->buffer= ADD_TO_PTR(pagecache->block_mem, - ((ulong) pagecache->blocks_used* - pagecache->block_size), - byte*); - pagecache->blocks_used++; - } - pagecache->blocks_unused--; - DBUG_ASSERT((block->status & BLOCK_WRLOCK) == 0); - DBUG_ASSERT(block->pins == 0); - block->status= 0; -#ifndef DBUG_OFF - block->type= PAGECACHE_EMPTY_PAGE; -#endif - block->requests= 1; - block->temperature= BLOCK_COLD; - block->hits_left= init_hits_left; - block->last_hit_time= 0; - link_to_file_list(pagecache, block, file, 0); - block->hash_link= hash_link; - hash_link->block= block; - page_status= PAGE_TO_BE_READ; - DBUG_PRINT("info", ("page to be read set for page 0x%lx", - (ulong)block)); - KEYCACHE_DBUG_PRINT("find_block", - ("got free or never used block %u", - BLOCK_NUMBER(pagecache, block))); - } - else - { - /* There are no never used blocks, use a block from the LRU chain */ - - /* - Wait until a new block is added to the LRU chain; - several threads might wait here for the same page, - all of them must get the same block - */ - -#ifdef THREAD - if (! pagecache->used_last) - { - struct st_my_thread_var *thread= my_thread_var; - thread->opt_info= (void *) hash_link; - wqueue_link_into_queue(&pagecache->waiting_for_block, thread); - do - { - KEYCACHE_DBUG_PRINT("find_block: wait", - ("suspend thread %ld", thread->id)); - pagecache_pthread_cond_wait(&thread->suspend, - &pagecache->cache_lock); - } - while (thread->next); - thread->opt_info= NULL; - } -#else - KEYCACHE_DBUG_ASSERT(pagecache->used_last); -#endif - block= hash_link->block; - if (! block) - { - /* - Take the first block from the LRU chain - unlinking it from the chain - */ - block= pagecache->used_last->next_used; - block->hits_left= init_hits_left; - block->last_hit_time= 0; - if (reg_req) - reg_requests(pagecache, block, 1); - hash_link->block= block; - } - BLOCK_INFO(block); - DBUG_ASSERT((block->status & BLOCK_WRLOCK) == 0); - DBUG_ASSERT(block->pins == 0); - - if (block->hash_link != hash_link && - ! (block->status & BLOCK_IN_SWITCH) ) - { - /* this is a primary request for a new page */ - DBUG_ASSERT((block->status & BLOCK_WRLOCK) == 0); - DBUG_ASSERT(block->pins == 0); - block->status|= (BLOCK_IN_SWITCH | BLOCK_WRLOCK); - - KEYCACHE_DBUG_PRINT("find_block", - ("got block %u for new page", - BLOCK_NUMBER(pagecache, block))); - - if (block->status & BLOCK_CHANGED) - { - /* The block contains a dirty page - push it out of the cache */ - - KEYCACHE_DBUG_PRINT("find_block", ("block is dirty")); - - pagecache_pthread_mutex_unlock(&pagecache->cache_lock); - /* - The call is thread safe because only the current - thread might change the block->hash_link value - */ - DBUG_ASSERT(block->pins == 0); - error= pagecache_fwrite(pagecache, - &block->hash_link->file, - block->buffer, - block->hash_link->pageno, - block->type, - MYF(MY_NABP | MY_WAIT_IF_FULL)); - pagecache_pthread_mutex_lock(&pagecache->cache_lock); - pagecache->global_cache_write++; - } - - block->status|= BLOCK_REASSIGNED; - if (block->hash_link) - { - /* - Wait until all pending read requests - for this page are executed - (we could have avoided this waiting, if we had read - a page in the cache in a sweep, without yielding control) - */ - wait_for_readers(pagecache, block); - - /* Remove the hash link for this page from the hash table */ - unlink_hash(pagecache, block->hash_link); - /* All pending requests for this page must be resubmitted */ -#ifdef THREAD - if (block->wqueue[COND_FOR_SAVED].last_thread) - wqueue_release_queue(&block->wqueue[COND_FOR_SAVED]); -#endif - } - link_to_file_list(pagecache, block, file, - (my_bool)(block->hash_link ? 1 : 0)); - BLOCK_INFO(block); - block->status= error? BLOCK_ERROR : 0; -#ifndef DBUG_OFF - block->type= PAGECACHE_EMPTY_PAGE; -#endif - block->hash_link= hash_link; - page_status= PAGE_TO_BE_READ; - DBUG_PRINT("info", ("page to be read set for page 0x%lx", - (ulong)block)); - - KEYCACHE_DBUG_ASSERT(block->hash_link->block == block); - KEYCACHE_DBUG_ASSERT(hash_link->block->hash_link == hash_link); - } - else - { - /* This is for secondary requests for a new page only */ - KEYCACHE_DBUG_PRINT("find_block", - ("block->hash_link: %p hash_link: %p " - "block->status: %u", block->hash_link, - hash_link, block->status )); - page_status= (((block->hash_link == hash_link) && - (block->status & BLOCK_READ)) ? - PAGE_READ : PAGE_WAIT_TO_BE_READ); - } - } - pagecache->global_cache_read++; - } - else - { - if (reg_req) - reg_requests(pagecache, block, 1); - KEYCACHE_DBUG_PRINT("find_block", - ("block->hash_link: %p hash_link: %p " - "block->status: %u", block->hash_link, - hash_link, block->status )); - page_status= (((block->hash_link == hash_link) && - (block->status & BLOCK_READ)) ? - PAGE_READ : PAGE_WAIT_TO_BE_READ); - } - } - - KEYCACHE_DBUG_ASSERT(page_status != -1); - *page_st= page_status; - DBUG_PRINT("info", - ("block: 0x%lx fd: %u pos %lu block->status %u page_status %u", - (ulong) block, (uint) file->file, - (ulong) pageno, block->status, (uint) page_status)); - KEYCACHE_DBUG_PRINT("find_block", - ("block: 0x%lx fd: %d pos: %lu block->status: %u page_status: %d", - (ulong) block, - file->file, (ulong) pageno, block->status, - page_status)); - -#if !defined(DBUG_OFF) && defined(EXTRA_DEBUG) - DBUG_EXECUTE("check_pagecache", - test_key_cache(pagecache, "end of find_block",0);); -#endif - KEYCACHE_THREAD_TRACE("find_block:end"); - DBUG_RETURN(block); -} - - -static void add_pin(PAGECACHE_BLOCK_LINK *block) -{ - DBUG_ENTER("add_pin"); - DBUG_PRINT("enter", ("block 0x%lx pins: %u", - (ulong) block, - block->pins)); - BLOCK_INFO(block); - block->pins++; -#ifdef PAGECACHE_DEBUG - { - PAGECACHE_PIN_INFO *info= - (PAGECACHE_PIN_INFO *)my_malloc(sizeof(PAGECACHE_PIN_INFO), MYF(0)); - info->thread= my_thread_var; - info_link(&block->pin_list, info); - } -#endif - DBUG_VOID_RETURN; -} - -static void remove_pin(PAGECACHE_BLOCK_LINK *block) -{ - DBUG_ENTER("remove_pin"); - DBUG_PRINT("enter", ("block 0x%lx pins: %u", - (ulong) block, - block->pins)); - BLOCK_INFO(block); - DBUG_ASSERT(block->pins > 0); - block->pins--; -#ifdef PAGECACHE_DEBUG - { - PAGECACHE_PIN_INFO *info= info_find(block->pin_list, my_thread_var); - DBUG_ASSERT(info != 0); - info_unlink(info); - my_free((gptr) info, MYF(0)); - } -#endif - DBUG_VOID_RETURN; -} -#ifdef PAGECACHE_DEBUG -static void info_add_lock(PAGECACHE_BLOCK_LINK *block, my_bool wl) -{ - PAGECACHE_LOCK_INFO *info= - (PAGECACHE_LOCK_INFO *)my_malloc(sizeof(PAGECACHE_LOCK_INFO), MYF(0)); - info->thread= my_thread_var; - info->write_lock= wl; - info_link((PAGECACHE_PIN_INFO **)&block->lock_list, - (PAGECACHE_PIN_INFO *)info); -} -static void info_remove_lock(PAGECACHE_BLOCK_LINK *block) -{ - PAGECACHE_LOCK_INFO *info= - (PAGECACHE_LOCK_INFO *)info_find((PAGECACHE_PIN_INFO *)block->lock_list, - my_thread_var); - DBUG_ASSERT(info != 0); - info_unlink((PAGECACHE_PIN_INFO *)info); - my_free((gptr)info, MYF(0)); -} -static void info_change_lock(PAGECACHE_BLOCK_LINK *block, my_bool wl) -{ - PAGECACHE_LOCK_INFO *info= - (PAGECACHE_LOCK_INFO *)info_find((PAGECACHE_PIN_INFO *)block->lock_list, - my_thread_var); - DBUG_ASSERT(info != 0 && info->write_lock != wl); - info->write_lock= wl; -} -#else -#define info_add_lock(B,W) -#define info_remove_lock(B) -#define info_change_lock(B,W) -#endif - -/* - Put on the block write lock - - SYNOPSIS - get_wrlock() - pagecache pointer to a page cache data structure - block the block to work with - - RETURN - 0 - OK - 1 - Can't lock this block, need retry -*/ - -static my_bool get_wrlock(PAGECACHE *pagecache, - PAGECACHE_BLOCK_LINK *block) -{ - PAGECACHE_FILE file= block->hash_link->file; - pgcache_page_no_t pageno= block->hash_link->pageno; - DBUG_ENTER("get_wrlock"); - DBUG_PRINT("info", ("the block 0x%lx " - "files %d(%d) pages %d(%d)", - (ulong)block, - file.file, block->hash_link->file.file, - pageno, block->hash_link->pageno)); - BLOCK_INFO(block); - while (block->status & BLOCK_WRLOCK) - { - /* Lock failed we will wait */ -#ifdef THREAD - struct st_my_thread_var *thread= my_thread_var; - DBUG_PRINT("info", ("fail to lock, waiting... 0x%lx", (ulong)block)); - wqueue_add_to_queue(&block->wqueue[COND_FOR_WRLOCK], thread); - dec_counter_for_resize_op(pagecache); - do - { - KEYCACHE_DBUG_PRINT("get_wrlock: wait", - ("suspend thread %ld", thread->id)); - pagecache_pthread_cond_wait(&thread->suspend, - &pagecache->cache_lock); - } - while(thread->next); -#else - DBUG_ASSERT(0); -#endif - BLOCK_INFO(block); - if ((block->status & (BLOCK_REASSIGNED | BLOCK_IN_SWITCH)) || - file.file != block->hash_link->file.file || - pageno != block->hash_link->pageno) - { - DBUG_PRINT("info", ("the block 0x%lx changed => need retry" - "status %x files %d != %d or pages %d !=%d", - (ulong)block, block->status, - file.file, block->hash_link->file.file, - pageno, block->hash_link->pageno)); - DBUG_RETURN(1); - } - } - DBUG_ASSERT(block->pins == 0); - /* we are doing it by global cache mutex protection, so it is OK */ - block->status|= BLOCK_WRLOCK; - DBUG_PRINT("info", ("WR lock set, block 0x%lx", (ulong)block)); - DBUG_RETURN(0); -} - - -/* - Remove write lock from the block - - SYNOPSIS - release_wrlock() - pagecache pointer to a page cache data structure - block the block to work with - - RETURN - 0 - OK -*/ - -static void release_wrlock(PAGECACHE_BLOCK_LINK *block) -{ - DBUG_ENTER("release_wrlock"); - BLOCK_INFO(block); - DBUG_ASSERT(block->status & BLOCK_WRLOCK); - DBUG_ASSERT(block->pins > 0); - block->status&= ~BLOCK_WRLOCK; - DBUG_PRINT("info", ("WR lock reset, block 0x%lx", (ulong)block)); -#ifdef THREAD - /* release all threads waiting for write lock */ - if (block->wqueue[COND_FOR_WRLOCK].last_thread) - wqueue_release_queue(&block->wqueue[COND_FOR_WRLOCK]); -#endif - BLOCK_INFO(block); - DBUG_VOID_RETURN; -} - - -/* - Try to lock/unlock and pin/unpin the block - - SYNOPSIS - make_lock_and_pin() - pagecache pointer to a page cache data structure - block the block to work with - lock lock change mode - pin pinchange mode - - RETURN - 0 - OK - 1 - Try to lock the block failed -*/ - -static my_bool make_lock_and_pin(PAGECACHE *pagecache, - PAGECACHE_BLOCK_LINK *block, - enum pagecache_page_lock lock, - enum pagecache_page_pin pin) -{ - DBUG_ENTER("make_lock_and_pin"); - DBUG_PRINT("enter", ("block: 0x%lx (%u), wrlock: %c pins: %u, lock %s, pin: %s", - (ulong)block, BLOCK_NUMBER(pagecache, block), - ((block->status & BLOCK_WRLOCK)?'Y':'N'), - block->pins, - page_cache_page_lock_str[lock], - page_cache_page_pin_str[pin])); - BLOCK_INFO(block); -#ifdef PAGECACHE_DEBUG - DBUG_ASSERT(info_check_pin(block, pin) == 0 && - info_check_lock(block, lock, pin) == 0); -#endif - switch (lock) - { - case PAGECACHE_LOCK_WRITE: /* free -> write */ - /* Writelock and pin the buffer */ - if (get_wrlock(pagecache, block)) - { - /* can't lock => need retry */ - goto retry; - } - - /* The cache is locked so nothing afraid of */ - add_pin(block); - info_add_lock(block, 1); - break; - case PAGECACHE_LOCK_WRITE_TO_READ: /* write -> read */ - case PAGECACHE_LOCK_WRITE_UNLOCK: /* write -> free */ - /* - Removes write lock and puts read lock (which is nothing in our - implementation) - */ - release_wrlock(block); - case PAGECACHE_LOCK_READ_UNLOCK: /* read -> free */ - case PAGECACHE_LOCK_LEFT_READLOCKED: /* read -> read */ - if (pin == PAGECACHE_UNPIN) - { - remove_pin(block); - } - if (lock == PAGECACHE_LOCK_WRITE_TO_READ) - { - info_change_lock(block, 0); - } - else if (lock == PAGECACHE_LOCK_WRITE_UNLOCK || - lock == PAGECACHE_LOCK_READ_UNLOCK) - { - info_remove_lock(block); - } - break; - case PAGECACHE_LOCK_READ: /* free -> read */ - if (pin == PAGECACHE_PIN) - { - /* The cache is locked so nothing afraid off */ - add_pin(block); - } - info_add_lock(block, 0); - break; - case PAGECACHE_LOCK_LEFT_UNLOCKED: /* free -> free */ - case PAGECACHE_LOCK_LEFT_WRITELOCKED: /* write -> write */ - break; /* do nothing */ - default: - DBUG_ASSERT(0); /* Never should happened */ - } - - BLOCK_INFO(block); - DBUG_RETURN(0); -retry: - DBUG_PRINT("INFO", ("Retry block 0x%lx", (ulong)block)); - BLOCK_INFO(block); - DBUG_ASSERT(block->hash_link->requests > 0); - block->hash_link->requests--; - DBUG_ASSERT(block->requests > 0); - unreg_request(pagecache, block, 1); - BLOCK_INFO(block); - DBUG_RETURN(1); - -} - - -/* - Read into a key cache block buffer from disk. - - SYNOPSIS - - read_block() - pagecache pointer to a page cache data structure - block block to which buffer the data is to be read - primary <-> the current thread will read the data - validator validator of read from the disk data - validator_data pointer to the data need by the validator - - RETURN VALUE - None - - NOTES. - The function either reads a page data from file to the block buffer, - or waits until another thread reads it. What page to read is determined - by a block parameter - reference to a hash link for this page. - If an error occurs THE BLOCK_ERROR bit is set in the block status. -*/ - -static void read_block(PAGECACHE *pagecache, - PAGECACHE_BLOCK_LINK *block, - my_bool primary, - pagecache_disk_read_validator validator, - gptr validator_data) -{ - uint got_length; - - /* On entry cache_lock is locked */ - - DBUG_ENTER("read_block"); - if (primary) - { - /* - This code is executed only by threads - that submitted primary requests - */ - - DBUG_PRINT("read_block", - ("page to be read by primary request")); - - /* Page is not in buffer yet, is to be read from disk */ - pagecache_pthread_mutex_unlock(&pagecache->cache_lock); - /* - Here other threads may step in and register as secondary readers. - They will register in block->wqueue[COND_FOR_REQUESTED]. - */ - got_length= pagecache_fread(pagecache, &block->hash_link->file, - block->buffer, - block->hash_link->pageno, MYF(0)); - pagecache_pthread_mutex_lock(&pagecache->cache_lock); - if (got_length < pagecache->block_size) - block->status|= BLOCK_ERROR; - else - block->status= (BLOCK_READ | (block->status & BLOCK_WRLOCK)); - - if (validator != NULL && - (*validator)(block->buffer, validator_data)) - block->status|= BLOCK_ERROR; - - DBUG_PRINT("read_block", - ("primary request: new page in cache")); - /* Signal that all pending requests for this page now can be processed */ -#ifdef THREAD - if (block->wqueue[COND_FOR_REQUESTED].last_thread) - wqueue_release_queue(&block->wqueue[COND_FOR_REQUESTED]); -#endif - } - else - { - /* - This code is executed only by threads - that submitted secondary requests - */ - DBUG_PRINT("read_block", - ("secondary request waiting for new page to be read")); - { -#ifdef THREAD - struct st_my_thread_var *thread= my_thread_var; - /* Put the request into a queue and wait until it can be processed */ - wqueue_add_to_queue(&block->wqueue[COND_FOR_REQUESTED], thread); - do - { - DBUG_PRINT("read_block: wait", - ("suspend thread %ld", thread->id)); - pagecache_pthread_cond_wait(&thread->suspend, - &pagecache->cache_lock); - } - while (thread->next); -#else - KEYCACHE_DBUG_ASSERT(0); - /* No parallel requests in single-threaded case */ -#endif - } - DBUG_PRINT("read_block", - ("secondary request: new page in cache")); - } - DBUG_VOID_RETURN; -} - - -/* - Unlock/unpin page and put LSN stamp if it need - - SYNOPSIS - pagecache_unlock_page() - pagecache pointer to a page cache data structure - file handler for the file for the block of data to be read - pageno number of the block of data in the file - lock lock change - pin pin page - first_REDO_LSN_for_page do not set it if it is zero - - NOTE - Pininig uses requests registration mechanism it works following way: - | beginnig | ending | - | of func. | of func. | - ----------------------------+-------------+---------------+ - PAGECACHE_PIN_LEFT_PINNED | - | - | - PAGECACHE_PIN_LEFT_UNPINNED | reg request | unreg request | - PAGECACHE_PIN | reg request | - | - PAGECACHE_UNPIN | - | unreg request | - - -*/ - -void pagecache_unlock_page(PAGECACHE *pagecache, - PAGECACHE_FILE *file, - pgcache_page_no_t pageno, - enum pagecache_page_lock lock, - enum pagecache_page_pin pin, - LSN first_REDO_LSN_for_page) -{ - PAGECACHE_BLOCK_LINK *block; - int page_st; - DBUG_ENTER("pagecache_unlock_page"); - DBUG_PRINT("enter", ("fd: %u page: %lu l%s p%s", - (uint) file->file, (ulong) pageno, - page_cache_page_lock_str[lock], - page_cache_page_pin_str[pin])); - /* we do not allow any lock/pin increasing here */ - DBUG_ASSERT(pin != PAGECACHE_PIN && - lock != PAGECACHE_LOCK_READ && - lock != PAGECACHE_LOCK_WRITE); - - pagecache_pthread_mutex_lock(&pagecache->cache_lock); - /* - As soon as we keep lock cache can be used, and we have lock because want - to unlock. - */ - DBUG_ASSERT(pagecache->can_be_used); - - inc_counter_for_resize_op(pagecache); - /* See NOTE for pagecache_unlock_page about registering requests */ - block= find_block(pagecache, file, pageno, 0, 0, - test(pin == PAGECACHE_PIN_LEFT_UNPINNED), &page_st); - BLOCK_INFO(block); - DBUG_ASSERT(block != 0 && page_st == PAGE_READ); - if (first_REDO_LSN_for_page) - { - DBUG_ASSERT(lock == PAGECACHE_LOCK_WRITE_UNLOCK && - pin == PAGECACHE_UNPIN); - set_if_bigger(block->rec_lsn, first_REDO_LSN_for_page); - } - -#ifndef DBUG_OFF - if ( -#endif - make_lock_and_pin(pagecache, block, lock, pin) -#ifndef DBUG_OFF - ) - { - DBUG_ASSERT(0); /* should not happend */ - } -#else - ; -#endif - - remove_reader(block); - /* - Link the block into the LRU chain if it's the last submitted request - for the block and block will not be pinned. - See NOTE for pagecache_unlock_page about registering requests. - */ - if (pin != PAGECACHE_PIN_LEFT_PINNED) - unreg_request(pagecache, block, 1); - - dec_counter_for_resize_op(pagecache); - - pagecache_pthread_mutex_unlock(&pagecache->cache_lock); - - DBUG_VOID_RETURN; -} - - -/* - Unpin page - - SYNOPSIS - pagecache_unpin_page() - pagecache pointer to a page cache data structure - file handler for the file for the block of data to be read - pageno number of the block of data in the file -*/ - -void pagecache_unpin_page(PAGECACHE *pagecache, - PAGECACHE_FILE *file, - pgcache_page_no_t pageno) -{ - PAGECACHE_BLOCK_LINK *block; - int page_st; - DBUG_ENTER("pagecache_unpin_page"); - DBUG_PRINT("enter", ("fd: %u page: %lu", - (uint) file->file, (ulong) pageno)); - pagecache_pthread_mutex_lock(&pagecache->cache_lock); - /* - As soon as we keep lock cache can be used, and we have lock bacause want - aunlock. - */ - DBUG_ASSERT(pagecache->can_be_used); - - inc_counter_for_resize_op(pagecache); - /* See NOTE for pagecache_unlock_page about registering requests */ - block= find_block(pagecache, file, pageno, 0, 0, 0, &page_st); - DBUG_ASSERT(block != 0 && page_st == PAGE_READ); - -#ifndef DBUG_OFF - if ( -#endif - /* - we can just unpin only with keeping read lock because: - a) we can't pin without any lock - b) we can't unpin keeping write lock - */ - make_lock_and_pin(pagecache, block, - PAGECACHE_LOCK_LEFT_READLOCKED, - PAGECACHE_UNPIN) -#ifndef DBUG_OFF - ) - { - DBUG_ASSERT(0); /* should not happend */ - } -#else - ; -#endif - - remove_reader(block); - /* - Link the block into the LRU chain if it's the last submitted request - for the block and block will not be pinned. - See NOTE for pagecache_unlock_page about registering requests - */ - unreg_request(pagecache, block, 1); - - dec_counter_for_resize_op(pagecache); - - pagecache_pthread_mutex_unlock(&pagecache->cache_lock); - - DBUG_VOID_RETURN; -} - - -/* - Unlock/unpin page and put LSN stamp if it need - (uses direct block/page pointer) - - SYNOPSIS - pagecache_unlock() - pagecache pointer to a page cache data structure - link direct link to page (returned by read or write) - lock lock change - pin pin page - first_REDO_LSN_for_page do not set it if it is zero -*/ - -void pagecache_unlock(PAGECACHE *pagecache, - PAGECACHE_PAGE_LINK *link, - enum pagecache_page_lock lock, - enum pagecache_page_pin pin, - LSN first_REDO_LSN_for_page) -{ - PAGECACHE_BLOCK_LINK *block= (PAGECACHE_BLOCK_LINK *)link; - DBUG_ENTER("pagecache_unlock"); - DBUG_PRINT("enter", ("block: 0x%lx fd: %u page: %lu l%s p%s", - (ulong) block, - (uint) block->hash_link->file.file, - (ulong) block->hash_link->pageno, - page_cache_page_lock_str[lock], - page_cache_page_pin_str[pin])); - /* - We do not allow any lock/pin increasing here and page can't be - unpinned because we use direct link. - */ - DBUG_ASSERT(pin != PAGECACHE_PIN && - pin != PAGECACHE_PIN_LEFT_UNPINNED && - lock != PAGECACHE_LOCK_READ && - lock != PAGECACHE_LOCK_WRITE); - if (pin == PAGECACHE_PIN_LEFT_UNPINNED && - lock == PAGECACHE_LOCK_READ_UNLOCK) - { -#ifndef DBUG_OFF - if ( -#endif - /* block do not need here so we do not provide it */ - make_lock_and_pin(pagecache, 0, lock, pin) -#ifndef DBUG_OFF - ) - { - DBUG_ASSERT(0); /* should not happend */ - } -#else - ; -#endif - DBUG_VOID_RETURN; - } - - pagecache_pthread_mutex_lock(&pagecache->cache_lock); - /* - As soon as we keep lock cache can be used, and we have lock bacause want - aunlock. - */ - DBUG_ASSERT(pagecache->can_be_used); - - inc_counter_for_resize_op(pagecache); - if (first_REDO_LSN_for_page) - { - DBUG_ASSERT(lock == PAGECACHE_LOCK_WRITE_UNLOCK && - pin == PAGECACHE_UNPIN); - set_if_bigger(block->rec_lsn, first_REDO_LSN_for_page); - } - -#ifndef DBUG_OFF - if ( -#endif - make_lock_and_pin(pagecache, block, lock, pin) -#ifndef DBUG_OFF - ) - { - DBUG_ASSERT(0); /* should not happend */ - } -#else - ; -#endif - - remove_reader(block); - /* - Link the block into the LRU chain if it's the last submitted request - for the block and block will not be pinned. - See NOTE for pagecache_unlock_page about registering requests. - */ - if (pin != PAGECACHE_PIN_LEFT_PINNED) - unreg_request(pagecache, block, 1); - - dec_counter_for_resize_op(pagecache); - - pagecache_pthread_mutex_unlock(&pagecache->cache_lock); - - DBUG_VOID_RETURN; -} - - -/* - Unpin page - (uses direct block/page pointer) - - SYNOPSIS - pagecache_unpin_page() - pagecache pointer to a page cache data structure - link direct link to page (returned by read or write) -*/ - -void pagecache_unpin(PAGECACHE *pagecache, - PAGECACHE_PAGE_LINK *link) -{ - PAGECACHE_BLOCK_LINK *block= (PAGECACHE_BLOCK_LINK *)link; - DBUG_ENTER("pagecache_unpin"); - DBUG_PRINT("enter", ("block: 0x%lx fd: %u page: %lu", - (ulong) block, - (uint) block->hash_link->file.file, - (ulong) block->hash_link->pageno)); - - pagecache_pthread_mutex_lock(&pagecache->cache_lock); - /* - As soon as we keep lock cache can be used, and we have lock bacause want - aunlock. - */ - DBUG_ASSERT(pagecache->can_be_used); - - inc_counter_for_resize_op(pagecache); - -#ifndef DBUG_OFF - if ( -#endif - /* - we can just unpin only with keeping read lock because: - a) we can't pin without any lock - b) we can't unpin keeping write lock - */ - make_lock_and_pin(pagecache, block, - PAGECACHE_LOCK_LEFT_READLOCKED, - PAGECACHE_UNPIN) -#ifndef DBUG_OFF - ) - { - DBUG_ASSERT(0); /* should not happend */ - } -#else - ; -#endif - - remove_reader(block); - /* - Link the block into the LRU chain if it's the last submitted request - for the block and block will not be pinned. - See NOTE for pagecache_unlock_page about registering requests. - */ - unreg_request(pagecache, block, 1); - - dec_counter_for_resize_op(pagecache); - - pagecache_pthread_mutex_unlock(&pagecache->cache_lock); - - DBUG_VOID_RETURN; -} - - -/* - Read a block of data from a cached file into a buffer; - - SYNOPSIS - pagecache_valid_read() - pagecache pointer to a page cache data structure - file handler for the file for the block of data to be read - pageno number of the block of data in the file - level determines the weight of the data - buff buffer to where the data must be placed - type type of the page - lock lock change - link link to the page if we pin it - validator validator of read from the disk data - validator_data pointer to the data need by the validator - - RETURN VALUE - Returns address from where the data is placed if sucessful, 0 - otherwise. - - Pin will be choosen according to lock parameter (see lock_to_pin) -*/ -static enum pagecache_page_pin lock_to_pin[]= -{ - PAGECACHE_PIN_LEFT_UNPINNED /*PAGECACHE_LOCK_LEFT_UNLOCKED*/, - PAGECACHE_PIN_LEFT_UNPINNED /*PAGECACHE_LOCK_LEFT_READLOCKED*/, - PAGECACHE_PIN_LEFT_PINNED /*PAGECACHE_LOCK_LEFT_WRITELOCKED*/, - PAGECACHE_PIN_LEFT_UNPINNED /*PAGECACHE_LOCK_READ*/, - PAGECACHE_PIN /*PAGECACHE_LOCK_WRITE*/, - PAGECACHE_PIN_LEFT_UNPINNED /*PAGECACHE_LOCK_READ_UNLOCK*/, - PAGECACHE_UNPIN /*PAGECACHE_LOCK_WRITE_UNLOCK*/, - PAGECACHE_UNPIN /*PAGECACHE_LOCK_WRITE_TO_READ*/ -}; - -byte *pagecache_valid_read(PAGECACHE *pagecache, - PAGECACHE_FILE *file, - pgcache_page_no_t pageno, - uint level, - byte *buff, - enum pagecache_page_type type, - enum pagecache_page_lock lock, - PAGECACHE_PAGE_LINK *link, - pagecache_disk_read_validator validator, - gptr validator_data) -{ - int error= 0; - enum pagecache_page_pin pin= lock_to_pin[lock]; - PAGECACHE_PAGE_LINK fake_link; - DBUG_ENTER("pagecache_valid_read"); - DBUG_PRINT("enter", ("fd: %u page: %lu level: %u t:%s l%s p%s", - (uint) file->file, (ulong) pageno, level, - page_cache_page_type_str[type], - page_cache_page_lock_str[lock], - page_cache_page_pin_str[pin])); - - if (!link) - link= &fake_link; - else - *link= 0; - -restart: - - if (pagecache->can_be_used) - { - /* Key cache is used */ - PAGECACHE_BLOCK_LINK *block; - uint status; - int page_st; - - pagecache_pthread_mutex_lock(&pagecache->cache_lock); - if (!pagecache->can_be_used) - { - pagecache_pthread_mutex_unlock(&pagecache->cache_lock); - goto no_key_cache; - } - - inc_counter_for_resize_op(pagecache); - pagecache->global_cache_r_requests++; - /* See NOTE for pagecache_unlock_page about registering requests. */ - block= find_block(pagecache, file, pageno, level, - test(lock == PAGECACHE_LOCK_WRITE), - test((pin == PAGECACHE_PIN_LEFT_UNPINNED) || - (pin == PAGECACHE_PIN)), - &page_st); - DBUG_ASSERT(block->type == PAGECACHE_EMPTY_PAGE || - block->type == type); - block->type= type; - if (block->status != BLOCK_ERROR && page_st != PAGE_READ) - { - DBUG_PRINT("info", ("read block 0x%lx", (ulong)block)); - /* The requested page is to be read into the block buffer */ - read_block(pagecache, block, - (my_bool)(page_st == PAGE_TO_BE_READ), - validator, validator_data); - DBUG_PRINT("info", ("read is done")); - } - if (make_lock_and_pin(pagecache, block, lock, pin)) - { - /* - We failed to write lock the block, cache is unlocked, - we will try to get the block again. - */ - pagecache_pthread_mutex_unlock(&pagecache->cache_lock); - DBUG_PRINT("info", ("restarting...")); - goto restart; - } - - if (! ((status= block->status) & BLOCK_ERROR)) - { -#if !defined(SERIALIZED_READ_FROM_CACHE) - pagecache_pthread_mutex_unlock(&pagecache->cache_lock); -#endif - - DBUG_ASSERT((pagecache->block_size & 511) == 0); - /* Copy data from the cache buffer */ - bmove512(buff, block->buffer, pagecache->block_size); - -#if !defined(SERIALIZED_READ_FROM_CACHE) - pagecache_pthread_mutex_lock(&pagecache->cache_lock); -#endif - } - - remove_reader(block); - /* - Link the block into the LRU chain if it's the last submitted request - for the block and block will not be pinned. - See NOTE for pagecache_unlock_page about registering requests. - */ - if (pin == PAGECACHE_PIN_LEFT_UNPINNED || pin == PAGECACHE_UNPIN) - unreg_request(pagecache, block, 1); - else - *link= (PAGECACHE_PAGE_LINK)block; - - dec_counter_for_resize_op(pagecache); - - pagecache_pthread_mutex_unlock(&pagecache->cache_lock); - - if (status & BLOCK_ERROR) - DBUG_RETURN((byte *) 0); - - DBUG_RETURN(buff); - } - -no_key_cache: /* Key cache is not used */ - - /* We can't use mutex here as the key cache may not be initialized */ - pagecache->global_cache_r_requests++; - pagecache->global_cache_read++; - if (pagecache_fread(pagecache, file, (byte*) buff, pageno, MYF(MY_NABP))) - error= 1; - DBUG_RETURN(error ? (byte*) 0 : buff); -} - - -/* - Delete page from the buffer - - SYNOPSIS - pagecache_delete_page() - pagecache pointer to a page cache data structure - file handler for the file for the block of data to be read - pageno number of the block of data in the file - lock lock change - flush flush page if it is dirty - - RETURN VALUE - 0 - deleted or was not present at all - 1 - error - - NOTES. - lock can be only PAGECACHE_LOCK_LEFT_WRITELOCKED (page was write locked - before) or PAGECACHE_LOCK_WRITE (delete will write lock page before delete) -*/ -my_bool pagecache_delete_page(PAGECACHE *pagecache, - PAGECACHE_FILE *file, - pgcache_page_no_t pageno, - enum pagecache_page_lock lock, - my_bool flush) -{ - int error= 0; - enum pagecache_page_pin pin= lock_to_pin[lock]; - DBUG_ENTER("pagecache_delete_page"); - DBUG_PRINT("enter", ("fd: %u page: %lu l%s p%s", - (uint) file->file, (ulong) pageno, - page_cache_page_lock_str[lock], - page_cache_page_pin_str[pin])); - DBUG_ASSERT(lock == PAGECACHE_LOCK_WRITE || - lock == PAGECACHE_LOCK_LEFT_WRITELOCKED); - DBUG_ASSERT(pin == PAGECACHE_PIN || - pin == PAGECACHE_PIN_LEFT_PINNED); - -restart: - - if (pagecache->can_be_used) - { - /* Key cache is used */ - reg1 PAGECACHE_BLOCK_LINK *block; - PAGECACHE_HASH_LINK **unused_start, *link; - - pagecache_pthread_mutex_lock(&pagecache->cache_lock); - if (!pagecache->can_be_used) - goto end; - - inc_counter_for_resize_op(pagecache); - link= get_present_hash_link(pagecache, file, pageno, &unused_start); - if (!link) - { - DBUG_PRINT("info", ("There is no such page in the cache")); - pagecache_pthread_mutex_unlock(&pagecache->cache_lock); - DBUG_RETURN(0); - } - block= link->block; - /* See NOTE for pagecache_unlock_page about registering requests. */ - if (pin == PAGECACHE_PIN) - reg_requests(pagecache, block, 1); - DBUG_ASSERT(block != 0); - if (make_lock_and_pin(pagecache, block, lock, pin)) - { - /* - We failed to writelock the block, cache is unlocked, and last write - lock is released, we will try to get the block again. - */ - pagecache_pthread_mutex_unlock(&pagecache->cache_lock); - DBUG_PRINT("info", ("restarting...")); - goto restart; - } - - if (block->status & BLOCK_CHANGED) - { - if (flush) - { - /* The block contains a dirty page - push it out of the cache */ - - KEYCACHE_DBUG_PRINT("find_block", ("block is dirty")); - - pagecache_pthread_mutex_unlock(&pagecache->cache_lock); - /* - The call is thread safe because only the current - thread might change the block->hash_link value - */ - DBUG_ASSERT(block->pins == 1); - error= pagecache_fwrite(pagecache, - &block->hash_link->file, - block->buffer, - block->hash_link->pageno, - block->type, - MYF(MY_NABP | MY_WAIT_IF_FULL)); - pagecache_pthread_mutex_lock(&pagecache->cache_lock); - pagecache->global_cache_write++; - - if (error) - { - block->status|= BLOCK_ERROR; - goto err; - } - } - pagecache->blocks_changed--; - pagecache->global_blocks_changed--; - /* - free_block() will change the status and rec_lsn of the block so no - need to change them here. - */ - } - /* Cache is locked, so we can relese page before freeing it */ - make_lock_and_pin(pagecache, block, - PAGECACHE_LOCK_WRITE_UNLOCK, - PAGECACHE_UNPIN); - DBUG_ASSERT(link->requests > 0); - link->requests--; - /* See NOTE for pagecache_unlock_page about registering requests. */ - free_block(pagecache, block); - -err: - dec_counter_for_resize_op(pagecache); -end: - pagecache_pthread_mutex_unlock(&pagecache->cache_lock); - } - - DBUG_RETURN(error); -} - - -/* - Write a buffer into a cached file. - - SYNOPSIS - - pagecache_write() - pagecache pointer to a page cache data structure - file handler for the file to write data to - pageno number of the block of data in the file - level determines the weight of the data - buff buffer to where the data must be placed - type type of the page - lock lock change - pin pin page - write_mode how to write page - link link to the page if we pin it - - RETURN VALUE - 0 if a success, 1 - otherwise. -*/ - -/* description of how to change lock before and after write */ -struct write_lock_change -{ - int need_lock_change; /* need changing of lock at the end of write */ - enum pagecache_page_lock new_lock; /* lock at the beginning */ - enum pagecache_page_lock unlock_lock; /* lock at the end */ -}; - -static struct write_lock_change write_lock_change_table[]= -{ - {1, - PAGECACHE_LOCK_WRITE, - PAGECACHE_LOCK_WRITE_UNLOCK} /*PAGECACHE_LOCK_LEFT_UNLOCKED*/, - {0, /*unsupported*/ - PAGECACHE_LOCK_LEFT_UNLOCKED, - PAGECACHE_LOCK_LEFT_UNLOCKED} /*PAGECACHE_LOCK_LEFT_READLOCKED*/, - {0, PAGECACHE_LOCK_LEFT_WRITELOCKED, 0} /*PAGECACHE_LOCK_LEFT_WRITELOCKED*/, - {1, - PAGECACHE_LOCK_WRITE, - PAGECACHE_LOCK_WRITE_TO_READ} /*PAGECACHE_LOCK_READ*/, - {0, PAGECACHE_LOCK_WRITE, 0} /*PAGECACHE_LOCK_WRITE*/, - {0, /*unsupported*/ - PAGECACHE_LOCK_LEFT_UNLOCKED, - PAGECACHE_LOCK_LEFT_UNLOCKED} /*PAGECACHE_LOCK_READ_UNLOCK*/, - {1, - PAGECACHE_LOCK_LEFT_WRITELOCKED, - PAGECACHE_LOCK_WRITE_UNLOCK } /*PAGECACHE_LOCK_WRITE_UNLOCK*/, - {1, - PAGECACHE_LOCK_LEFT_WRITELOCKED, - PAGECACHE_LOCK_WRITE_TO_READ}/*PAGECACHE_LOCK_WRITE_TO_READ*/ -}; - -/* description of how to change pin before and after write */ -struct write_pin_change -{ - enum pagecache_page_pin new_pin; /* pin status at the beginning */ - enum pagecache_page_pin unlock_pin; /* pin status at the end */ -}; - -static struct write_pin_change write_pin_change_table[]= -{ - {PAGECACHE_PIN_LEFT_PINNED, - PAGECACHE_PIN_LEFT_PINNED} /*PAGECACHE_PIN_LEFT_PINNED*/, - {PAGECACHE_PIN, - PAGECACHE_UNPIN} /*PAGECACHE_PIN_LEFT_UNPINNED*/, - {PAGECACHE_PIN, - PAGECACHE_PIN_LEFT_PINNED} /*PAGECACHE_PIN*/, - {PAGECACHE_PIN_LEFT_PINNED, - PAGECACHE_UNPIN} /*PAGECACHE_UNPIN*/ -}; - -my_bool pagecache_write(PAGECACHE *pagecache, - PAGECACHE_FILE *file, - pgcache_page_no_t pageno, - uint level, - byte *buff, - enum pagecache_page_type type, - enum pagecache_page_lock lock, - enum pagecache_page_pin pin, - enum pagecache_write_mode write_mode, - PAGECACHE_PAGE_LINK *link) -{ - reg1 PAGECACHE_BLOCK_LINK *block= NULL; - PAGECACHE_PAGE_LINK fake_link; - int error= 0; - int need_lock_change= write_lock_change_table[lock].need_lock_change; - DBUG_ENTER("pagecache_write"); - DBUG_PRINT("enter", ("fd: %u page: %lu level: %u t:%s l%s p%s m%s", - (uint) file->file, (ulong) pageno, level, - page_cache_page_type_str[type], - page_cache_page_lock_str[lock], - page_cache_page_pin_str[pin], - page_cache_page_write_mode_str[write_mode])); - DBUG_ASSERT(lock != PAGECACHE_LOCK_LEFT_READLOCKED && - lock != PAGECACHE_LOCK_READ_UNLOCK); - if (!link) - link= &fake_link; - else - *link= 0; - - if (write_mode == PAGECACHE_WRITE_NOW) - { - /* we allow direct write if we do not use long term lockings */ - DBUG_ASSERT(lock == PAGECACHE_LOCK_LEFT_UNLOCKED); - /* Force writing from buff into disk */ - pagecache->global_cache_write++; - if (pagecache_fwrite(pagecache, file, buff, pageno, type, - MYF(MY_NABP | MY_WAIT_IF_FULL))) - DBUG_RETURN(1); - } -restart: - -#if !defined(DBUG_OFF) && defined(EXTRA_DEBUG) - DBUG_EXECUTE("check_pagecache", - test_key_cache(pagecache, "start of key_cache_write", 1);); -#endif - - if (pagecache->can_be_used) - { - /* Key cache is used */ - int page_st; - - pagecache_pthread_mutex_lock(&pagecache->cache_lock); - if (!pagecache->can_be_used) - { - pagecache_pthread_mutex_unlock(&pagecache->cache_lock); - goto no_key_cache; - } - - inc_counter_for_resize_op(pagecache); - pagecache->global_cache_w_requests++; - /* See NOTE for pagecache_unlock_page about registering requests. */ - block= find_block(pagecache, file, pageno, level, - test(write_mode != PAGECACHE_WRITE_DONE && - lock != PAGECACHE_LOCK_LEFT_WRITELOCKED && - lock != PAGECACHE_LOCK_WRITE_UNLOCK && - lock != PAGECACHE_LOCK_WRITE_TO_READ), - test((pin == PAGECACHE_PIN_LEFT_UNPINNED) || - (pin == PAGECACHE_PIN)), - &page_st); - if (!block) - { - DBUG_ASSERT(write_mode != PAGECACHE_WRITE_DONE); - /* It happens only for requests submitted during resize operation */ - dec_counter_for_resize_op(pagecache); - pagecache_pthread_mutex_unlock(&pagecache->cache_lock); - /* Write to the disk key cache is in resize at the moment*/ - goto no_key_cache; - } - - DBUG_ASSERT(block->type == PAGECACHE_EMPTY_PAGE || - block->type == type); - block->type= type; - - if (make_lock_and_pin(pagecache, block, - write_lock_change_table[lock].new_lock, - (need_lock_change ? - write_pin_change_table[pin].new_pin : - pin))) - { - /* - We failed to writelock the block, cache is unlocked, and last write - lock is released, we will try to get the block again. - */ - pagecache_pthread_mutex_unlock(&pagecache->cache_lock); - DBUG_PRINT("info", ("restarting...")); - goto restart; - } - - - if (write_mode == PAGECACHE_WRITE_DONE) - { - if ((block->status & BLOCK_ERROR) && page_st != PAGE_READ) - { - /* Copy data from buff */ - bmove512(block->buffer, buff, pagecache->block_size); - block->status= (BLOCK_READ | (block->status & BLOCK_WRLOCK)); - KEYCACHE_DBUG_PRINT("key_cache_insert", - ("primary request: new page in cache")); -#ifdef THREAD - /* Signal that all pending requests for this now can be processed. */ - if (block->wqueue[COND_FOR_REQUESTED].last_thread) - wqueue_release_queue(&block->wqueue[COND_FOR_REQUESTED]); -#endif - } - } - else - { - if (write_mode == PAGECACHE_WRITE_NOW) - { - /* buff has been written to disk at start */ - if ((block->status & BLOCK_CHANGED) && - !(block->status & BLOCK_ERROR)) - link_to_file_list(pagecache, block, &block->hash_link->file, 1); - } - else if (! (block->status & BLOCK_CHANGED)) - link_to_changed_list(pagecache, block); - - if (! (block->status & BLOCK_ERROR)) - { - bmove512(block->buffer, buff, pagecache->block_size); - block->status|= BLOCK_READ; - } - } - - - if (need_lock_change) - { -#ifndef DBUG_OFF - int rc= -#endif - /* - QQ: We are doing an unlock here, so need to give the page its rec_lsn - */ - make_lock_and_pin(pagecache, block, - write_lock_change_table[lock].unlock_lock, - write_pin_change_table[pin].unlock_pin); -#ifndef DBUG_OFF - DBUG_ASSERT(rc == 0); -#endif - } - - /* Unregister the request */ - DBUG_ASSERT(block->hash_link->requests > 0); - block->hash_link->requests--; - /* See NOTE for pagecache_unlock_page about registering requests. */ - if (pin == PAGECACHE_PIN_LEFT_UNPINNED || pin == PAGECACHE_UNPIN) - unreg_request(pagecache, block, 1); - else - *link= (PAGECACHE_PAGE_LINK)block; - - - if (block->status & BLOCK_ERROR) - error= 1; - - dec_counter_for_resize_op(pagecache); - - pagecache_pthread_mutex_unlock(&pagecache->cache_lock); - - goto end; - } - -no_key_cache: - /* Key cache is not used */ - if (write_mode == PAGECACHE_WRITE_DELAY) - { - pagecache->global_cache_w_requests++; - pagecache->global_cache_write++; - if (pagecache_fwrite(pagecache, file, (byte*) buff, pageno, type, - MYF(MY_NABP | MY_WAIT_IF_FULL))) - error=1; - } - -end: -#if !defined(DBUG_OFF) && defined(EXTRA_DEBUG) - DBUG_EXECUTE("exec", - test_key_cache(pagecache, "end of key_cache_write", 1);); -#endif - BLOCK_INFO(block); - DBUG_RETURN(error); -} - - -/* - Free block: remove reference to it from hash table, - remove it from the chain file of dirty/clean blocks - and add it to the free list. -*/ - -static void free_block(PAGECACHE *pagecache, PAGECACHE_BLOCK_LINK *block) -{ - KEYCACHE_THREAD_TRACE("free block"); - KEYCACHE_DBUG_PRINT("free_block", - ("block %u to be freed, hash_link %p", - BLOCK_NUMBER(pagecache, block), block->hash_link)); - if (block->hash_link) - { - /* - While waiting for readers to finish, new readers might request the - block. But since we set block->status|= BLOCK_REASSIGNED, they - will wait on block->wqueue[COND_FOR_SAVED]. They must be signalled - later. - */ - block->status|= BLOCK_REASSIGNED; - wait_for_readers(pagecache, block); - unlink_hash(pagecache, block->hash_link); - } - - unlink_changed(block); - DBUG_ASSERT((block->status & BLOCK_WRLOCK) == 0); - DBUG_ASSERT(block->pins == 0); - block->status= 0; -#ifndef DBUG_OFF - block->type= PAGECACHE_EMPTY_PAGE; -#endif - block->rec_lsn= 0; - KEYCACHE_THREAD_TRACE("free block"); - KEYCACHE_DBUG_PRINT("free_block", - ("block is freed")); - unreg_request(pagecache, block, 0); - block->hash_link= NULL; - - /* Remove the free block from the LRU ring. */ - unlink_block(pagecache, block); - if (block->temperature == BLOCK_WARM) - pagecache->warm_blocks--; - block->temperature= BLOCK_COLD; - /* Insert the free block in the free list. */ - block->next_used= pagecache->free_block_list; - pagecache->free_block_list= block; - /* Keep track of the number of currently unused blocks. */ - pagecache->blocks_unused++; - -#ifdef THREAD - /* All pending requests for this page must be resubmitted. */ - if (block->wqueue[COND_FOR_SAVED].last_thread) - wqueue_release_queue(&block->wqueue[COND_FOR_SAVED]); -#endif -} - - -static int cmp_sec_link(PAGECACHE_BLOCK_LINK **a, PAGECACHE_BLOCK_LINK **b) -{ - return (((*a)->hash_link->pageno < (*b)->hash_link->pageno) ? -1 : - ((*a)->hash_link->pageno > (*b)->hash_link->pageno) ? 1 : 0); -} - - -/* - Flush a portion of changed blocks to disk, - free used blocks if requested -*/ - -static int flush_cached_blocks(PAGECACHE *pagecache, - PAGECACHE_FILE *file, - PAGECACHE_BLOCK_LINK **cache, - PAGECACHE_BLOCK_LINK **end, - enum flush_type type) -{ - int error; - int last_errno= 0; - uint count= (uint) (end-cache); - DBUG_ENTER("flush_cached_blocks"); - - /* Don't lock the cache during the flush */ - pagecache_pthread_mutex_unlock(&pagecache->cache_lock); - /* - As all blocks referred in 'cache' are marked by BLOCK_IN_FLUSH - we are guarunteed no thread will change them - */ - qsort((byte*) cache, count, sizeof(*cache), (qsort_cmp) cmp_sec_link); - - pagecache_pthread_mutex_lock(&pagecache->cache_lock); - for (; cache != end; cache++) - { - PAGECACHE_BLOCK_LINK *block= *cache; - - if (block->pins) - { - KEYCACHE_DBUG_PRINT("flush_cached_blocks", - ("block %u (0x%lx) pinned", - BLOCK_NUMBER(pagecache, block), (ulong)block)); - DBUG_PRINT("info", ("block %u (0x%lx) pinned", - BLOCK_NUMBER(pagecache, block), (ulong)block)); - BLOCK_INFO(block); - last_errno= -1; - unreg_request(pagecache, block, 1); - continue; - } - /* if the block is not pinned then it is not write locked */ - DBUG_ASSERT((block->status & BLOCK_WRLOCK) == 0); - DBUG_ASSERT(block->pins == 0); -#ifndef DBUG_OFF - { - int rc= -#endif - make_lock_and_pin(pagecache, block, - PAGECACHE_LOCK_WRITE, PAGECACHE_PIN); -#ifndef DBUG_OFF - DBUG_ASSERT(rc == 0); - } -#endif - - KEYCACHE_DBUG_PRINT("flush_cached_blocks", - ("block %u (0x%lx) to be flushed", - BLOCK_NUMBER(pagecache, block), (ulong)block)); - DBUG_PRINT("info", ("block %u (0x%lx) to be flushed", - BLOCK_NUMBER(pagecache, block), (ulong)block)); - BLOCK_INFO(block); - pagecache_pthread_mutex_unlock(&pagecache->cache_lock); - DBUG_PRINT("info", ("block %u (0x%lx) pins: %u", - BLOCK_NUMBER(pagecache, block), (ulong)block, - block->pins)); - DBUG_ASSERT(block->pins == 1); - error= pagecache_fwrite(pagecache, file, - block->buffer, - block->hash_link->pageno, - block->type, - MYF(MY_NABP | MY_WAIT_IF_FULL)); - pagecache_pthread_mutex_lock(&pagecache->cache_lock); - - make_lock_and_pin(pagecache, block, - PAGECACHE_LOCK_WRITE_UNLOCK, - PAGECACHE_UNPIN); - - pagecache->global_cache_write++; - if (error) - { - block->status|= BLOCK_ERROR; - if (!last_errno) - last_errno= errno ? errno : -1; - } -#ifdef THREAD - /* - Let to proceed for possible waiting requests to write to the block page. - It might happen only during an operation to resize the key cache. - */ - if (block->wqueue[COND_FOR_SAVED].last_thread) - wqueue_release_queue(&block->wqueue[COND_FOR_SAVED]); -#endif - /* type will never be FLUSH_IGNORE_CHANGED here */ - if (! (type == FLUSH_KEEP || type == FLUSH_FORCE_WRITE)) - { - pagecache->blocks_changed--; - pagecache->global_blocks_changed--; - free_block(pagecache, block); - } - else - { - block->status&= ~BLOCK_IN_FLUSH; - link_to_file_list(pagecache, block, file, 1); - unreg_request(pagecache, block, 1); - } - } - DBUG_RETURN(last_errno); -} - - -/* - flush all key blocks for a file to disk, but don't do any mutex locks - - flush_pagecache_blocks_int() - pagecache pointer to a key cache data structure - file handler for the file to flush to - flush_type type of the flush - - NOTES - This function doesn't do any mutex locks because it needs to be called - both from flush_pagecache_blocks and flush_all_key_blocks (the later one - does the mutex lock in the resize_pagecache() function). - - RETURN - 0 ok - 1 error -*/ - -static int flush_pagecache_blocks_int(PAGECACHE *pagecache, - PAGECACHE_FILE *file, - enum flush_type type) -{ - PAGECACHE_BLOCK_LINK *cache_buff[FLUSH_CACHE],**cache; - int last_errno= 0; - DBUG_ENTER("flush_pagecache_blocks_int"); - DBUG_PRINT("enter",("file: %d blocks_used: %lu blocks_changed: %lu", - file->file, pagecache->blocks_used, pagecache->blocks_changed)); - -#if !defined(DBUG_OFF) && defined(EXTRA_DEBUG) - DBUG_EXECUTE("check_pagecache", - test_key_cache(pagecache, - "start of flush_pagecache_blocks", 0);); -#endif - - cache= cache_buff; - if (pagecache->disk_blocks > 0 && - (!my_disable_flush_pagecache_blocks || type != FLUSH_KEEP)) - { - /* Key cache exists and flush is not disabled */ - int error= 0; - uint count= 0; - PAGECACHE_BLOCK_LINK **pos, **end; - PAGECACHE_BLOCK_LINK *first_in_switch= NULL; - PAGECACHE_BLOCK_LINK *block, *next; -#if defined(PAGECACHE_DEBUG) - uint cnt= 0; -#endif - - if (type != FLUSH_IGNORE_CHANGED) - { - /* - Count how many key blocks we have to cache to be able - to flush all dirty pages with minimum seek moves - */ - for (block= pagecache->changed_blocks[FILE_HASH(*file)] ; - block; - block= block->next_changed) - { - if (block->hash_link->file.file == file->file) - { - count++; - KEYCACHE_DBUG_ASSERT(count<= pagecache->blocks_used); - } - } - /* Allocate a new buffer only if its bigger than the one we have */ - if (count > FLUSH_CACHE && - !(cache= - (PAGECACHE_BLOCK_LINK**) - my_malloc(sizeof(PAGECACHE_BLOCK_LINK*)*count, MYF(0)))) - { - cache= cache_buff; - count= FLUSH_CACHE; - } - } - - /* Retrieve the blocks and write them to a buffer to be flushed */ -restart: - end= (pos= cache)+count; - for (block= pagecache->changed_blocks[FILE_HASH(*file)] ; - block; - block= next) - { -#if defined(PAGECACHE_DEBUG) - cnt++; - KEYCACHE_DBUG_ASSERT(cnt <= pagecache->blocks_used); -#endif - next= block->next_changed; - if (block->hash_link->file.file == file->file) - { - /* - Mark the block with BLOCK_IN_FLUSH in order not to let - other threads to use it for new pages and interfere with - our sequence ot flushing dirty file pages - */ - block->status|= BLOCK_IN_FLUSH; - - if (! (block->status & BLOCK_IN_SWITCH)) - { - /* - We care only for the blocks for which flushing was not - initiated by other threads as a result of page swapping - */ - reg_requests(pagecache, block, 1); - if (type != FLUSH_IGNORE_CHANGED) - { - /* It's not a temporary file */ - if (pos == end) - { - /* - This happens only if there is not enough - memory for the big block - */ - if ((error= flush_cached_blocks(pagecache, file, cache, - end,type))) - last_errno=error; - DBUG_PRINT("info", ("restarting...")); - /* - Restart the scan as some other thread might have changed - the changed blocks chain: the blocks that were in switch - state before the flush started have to be excluded - */ - goto restart; - } - *pos++= block; - } - else - { - /* It's a temporary file */ - pagecache->blocks_changed--; - pagecache->global_blocks_changed--; - free_block(pagecache, block); - } - } - else - { - /* Link the block into a list of blocks 'in switch' */ - /* QQ: - #warning this unlink_changed() is a serious problem for - Maria's Checkpoint: it removes a page from the list of dirty - pages, while it's still dirty. A solution is to abandon - first_in_switch, just wait for this page to be - flushed by somebody else, and loop. TODO: check all places - where we remove a page from the list of dirty pages - */ - unlink_changed(block); - link_changed(block, &first_in_switch); - } - } - } - if (pos != cache) - { - if ((error= flush_cached_blocks(pagecache, file, cache, pos, type))) - last_errno= error; - } - /* Wait until list of blocks in switch is empty */ - while (first_in_switch) - { -#if defined(PAGECACHE_DEBUG) - cnt= 0; -#endif - block= first_in_switch; - { -#ifdef THREAD - struct st_my_thread_var *thread= my_thread_var; - wqueue_add_to_queue(&block->wqueue[COND_FOR_SAVED], thread); - do - { - KEYCACHE_DBUG_PRINT("flush_pagecache_blocks_int: wait", - ("suspend thread %ld", thread->id)); - pagecache_pthread_cond_wait(&thread->suspend, - &pagecache->cache_lock); - } - while (thread->next); -#else - KEYCACHE_DBUG_ASSERT(0); - /* No parallel requests in single-threaded case */ -#endif - } -#if defined(PAGECACHE_DEBUG) - cnt++; - KEYCACHE_DBUG_ASSERT(cnt <= pagecache->blocks_used); -#endif - } - /* The following happens very seldom */ - if (! (type == FLUSH_KEEP || type == FLUSH_FORCE_WRITE)) - { -#if defined(PAGECACHE_DEBUG) - cnt=0; -#endif - for (block= pagecache->file_blocks[FILE_HASH(*file)] ; - block; - block= next) - { -#if defined(PAGECACHE_DEBUG) - cnt++; - KEYCACHE_DBUG_ASSERT(cnt <= pagecache->blocks_used); -#endif - next= block->next_changed; - if (block->hash_link->file.file == file->file && - (! (block->status & BLOCK_CHANGED) - || type == FLUSH_IGNORE_CHANGED)) - { - reg_requests(pagecache, block, 1); - free_block(pagecache, block); - } - } - } - } - -#ifndef DBUG_OFF - DBUG_EXECUTE("check_pagecache", - test_key_cache(pagecache, "end of flush_pagecache_blocks", 0);); -#endif - if (cache != cache_buff) - my_free((gptr) cache, MYF(0)); - if (last_errno) - errno=last_errno; /* Return first error */ - DBUG_RETURN(last_errno != 0); -} - - -/* - Flush all blocks for a file to disk - - SYNOPSIS - - flush_pagecache_blocks() - pagecache pointer to a page cache data structure - file handler for the file to flush to - flush_type type of the flush - - RETURN - 0 ok - 1 error -*/ - -int flush_pagecache_blocks(PAGECACHE *pagecache, - PAGECACHE_FILE *file, enum flush_type type) -{ - int res; - DBUG_ENTER("flush_pagecache_blocks"); - DBUG_PRINT("enter", ("pagecache: 0x%lx", (long) pagecache)); - - if (pagecache->disk_blocks <= 0) - DBUG_RETURN(0); - pagecache_pthread_mutex_lock(&pagecache->cache_lock); - inc_counter_for_resize_op(pagecache); - res= flush_pagecache_blocks_int(pagecache, file, type); - dec_counter_for_resize_op(pagecache); - pagecache_pthread_mutex_unlock(&pagecache->cache_lock); - DBUG_RETURN(res); -} - - -/* - Reset the counters of a key cache. - - SYNOPSIS - reset_pagecache_counters() - name the name of a key cache - pagecache pointer to the pagecache to be reset - - DESCRIPTION - This procedure is used to reset the counters of all currently used key - caches, both the default one and the named ones. - - RETURN - 0 on success (always because it can't fail) -*/ - -int reset_pagecache_counters(const char *name, PAGECACHE *pagecache) -{ - DBUG_ENTER("reset_pagecache_counters"); - if (!pagecache->inited) - { - DBUG_PRINT("info", ("Key cache %s not initialized.", name)); - DBUG_RETURN(0); - } - DBUG_PRINT("info", ("Resetting counters for key cache %s.", name)); - - pagecache->global_blocks_changed= 0; /* Key_blocks_not_flushed */ - pagecache->global_cache_r_requests= 0; /* Key_read_requests */ - pagecache->global_cache_read= 0; /* Key_reads */ - pagecache->global_cache_w_requests= 0; /* Key_write_requests */ - pagecache->global_cache_write= 0; /* Key_writes */ - DBUG_RETURN(0); -} - - -/* - Allocates a buffer and stores in it some information about all dirty pages - of type PAGECACHE_LSN_PAGE. - - SYNOPSIS - pagecache_collect_changed_blocks_with_lsn() - pagecache pointer to the page cache - str (OUT) pointer to a LEX_STRING where the allocated buffer, and - its size, will be put - max_lsn (OUT) pointer to a LSN where the maximum rec_lsn of all - relevant dirty pages will be put - - DESCRIPTION - Does the allocation because the caller cannot know the size itself. - Memory freeing is to be done by the caller (if the "str" member of the - LEX_STRING is not NULL). - Ignores all pages of another type than PAGECACHE_LSN_PAGE, because they - are not interesting for a checkpoint record. - The caller has the intention of doing checkpoints. - - RETURN - 0 on success - 1 on error -*/ -my_bool pagecache_collect_changed_blocks_with_lsn(PAGECACHE *pagecache, - LEX_STRING *str, - LSN *max_lsn) -{ - my_bool error; - ulong stored_list_size= 0; - uint file_hash; - char *ptr; - DBUG_ENTER("pagecache_collect_changed_blocks_with_LSN"); - - *max_lsn= 0; - DBUG_ASSERT(NULL == str->str); - /* - We lock the entire cache but will be quick, just reading/writing a few MBs - of memory at most. - When we enter here, we must be sure that no "first_in_switch" situation - is happening or will happen (either we have to get rid of - first_in_switch in the code or, first_in_switch has to increment a - "danger" counter for this function to know it has to wait). TODO. - */ - pagecache_pthread_mutex_lock(&pagecache->cache_lock); - - /* Count how many dirty pages are interesting */ - for (file_hash= 0; file_hash < PAGECACHE_CHANGED_BLOCKS_HASH; file_hash++) - { - PAGECACHE_BLOCK_LINK *block; - for (block= pagecache->changed_blocks[file_hash] ; - block; - block= block->next_changed) - { - /* - Q: is there somthing subtle with block->hash_link: can it be NULL? - does it have to be == hash_link->block... ? - */ - DBUG_ASSERT(block->hash_link != NULL); - DBUG_ASSERT(block->status & BLOCK_CHANGED); - if (block->type != PAGECACHE_LSN_PAGE) - continue; /* no need to store it */ - /* - In the current pagecache, rec_lsn is not set correctly: - 1) it is set on pagecache_unlock(), too late (a page is dirty - (BLOCK_CHANGED) since the first pagecache_write()). So in this - scenario: - thread1: thread2: - write_REDO - pagecache_write() checkpoint : reclsn not known - pagecache_unlock(sets rec_lsn) - commit - crash, - at recovery we will wrongly skip the REDO. It also affects the - low-water mark's computation. - 2) sometimes the unlocking can be an implicit action of - pagecache_write(), without any call to pagecache_unlock(), then - rec_lsn is not set. - 1) and 2) are critical problems. - TODO: fix this when Monty has explained how he writes BLOB pages. - */ - if (block->rec_lsn == 0) - { - DBUG_ASSERT(0); - goto err; - } - stored_list_size++; - } - } - - str->length= 8+(4+4+8)*stored_list_size; - if (NULL == (str->str= my_malloc(str->length, MYF(MY_WME)))) - goto err; - ptr= str->str; - int8store(ptr, stored_list_size); - ptr+= 8; - if (0 == stored_list_size) - goto end; - for (file_hash= 0; file_hash < PAGECACHE_CHANGED_BLOCKS_HASH; file_hash++) - { - PAGECACHE_BLOCK_LINK *block; - for (block= pagecache->changed_blocks[file_hash] ; - block; - block= block->next_changed) - { - if (block->type != PAGECACHE_LSN_PAGE) - continue; /* no need to store it in the checkpoint record */ - DBUG_ASSERT((4 == sizeof(block->hash_link->file.file)) && - (4 == sizeof(block->hash_link->pageno))); - int4store(ptr, block->hash_link->file.file); - ptr+= 4; - int4store(ptr, block->hash_link->pageno); - ptr+= 4; - int8store(ptr, (ulonglong) block->rec_lsn); - ptr+= 8; - set_if_bigger(*max_lsn, block->rec_lsn); - } - } - error= 0; - goto end; -err: - error= 1; -end: - pagecache_pthread_mutex_unlock(&pagecache->cache_lock); - DBUG_RETURN(error); -} - - -#ifndef DBUG_OFF -/* - Test if disk-cache is ok -*/ -static void test_key_cache(PAGECACHE *pagecache __attribute__((unused)), - const char *where __attribute__((unused)), - my_bool lock __attribute__((unused))) -{ - /* TODO */ -} -#endif - -#if defined(PAGECACHE_TIMEOUT) - -#define KEYCACHE_DUMP_FILE "pagecache_dump.txt" -#define MAX_QUEUE_LEN 100 - - -static void pagecache_dump(PAGECACHE *pagecache) -{ - FILE *pagecache_dump_file=fopen(KEYCACHE_DUMP_FILE, "w"); - struct st_my_thread_var *last; - struct st_my_thread_var *thread; - PAGECACHE_BLOCK_LINK *block; - PAGECACHE_HASH_LINK *hash_link; - PAGECACHE_PAGE *page; - uint i; - - fprintf(pagecache_dump_file, "thread:%u\n", thread->id); - - i=0; - thread=last=waiting_for_hash_link.last_thread; - fprintf(pagecache_dump_file, "queue of threads waiting for hash link\n"); - if (thread) - do - { - thread= thread->next; - page= (PAGECACHE_PAGE *) thread->opt_info; - fprintf(pagecache_dump_file, - "thread:%u, (file,pageno)=(%u,%lu)\n", - thread->id,(uint) page->file.file,(ulong) page->pageno); - if (++i == MAX_QUEUE_LEN) - break; - } - while (thread != last); - - i=0; - thread=last=waiting_for_block.last_thread; - fprintf(pagecache_dump_file, "queue of threads waiting for block\n"); - if (thread) - do - { - thread=thread->next; - hash_link= (PAGECACHE_HASH_LINK *) thread->opt_info; - fprintf(pagecache_dump_file, - "thread:%u hash_link:%u (file,pageno)=(%u,%lu)\n", - thread->id, (uint) PAGECACHE_HASH_LINK_NUMBER(pagecache, hash_link), - (uint) hash_link->file.file,(ulong) hash_link->pageno); - if (++i == MAX_QUEUE_LEN) - break; - } - while (thread != last); - - for (i=0 ; i < pagecache->blocks_used ; i++) - { - int j; - block= &pagecache->block_root[i]; - hash_link= block->hash_link; - fprintf(pagecache_dump_file, - "block:%u hash_link:%d status:%x #requests=%u waiting_for_readers:%d\n", - i, (int) (hash_link ? - PAGECACHE_HASH_LINK_NUMBER(pagecache, hash_link) : - -1), - block->status, block->requests, block->condvar ? 1 : 0); - for (j=0 ; j < COND_SIZE; j++) - { - PAGECACHE_WQUEUE *wqueue=&block->wqueue[j]; - thread= last= wqueue->last_thread; - fprintf(pagecache_dump_file, "queue #%d\n", j); - if (thread) - { - do - { - thread=thread->next; - fprintf(pagecache_dump_file, - "thread:%u\n", thread->id); - if (++i == MAX_QUEUE_LEN) - break; - } - while (thread != last); - } - } - } - fprintf(pagecache_dump_file, "LRU chain:"); - block= pagecache= used_last; - if (block) - { - do - { - block= block->next_used; - fprintf(pagecache_dump_file, - "block:%u, ", BLOCK_NUMBER(pagecache, block)); - } - while (block != pagecache->used_last); - } - fprintf(pagecache_dump_file, "\n"); - - fclose(pagecache_dump_file); -} - -#endif /* defined(PAGECACHE_TIMEOUT) */ - -#if defined(PAGECACHE_TIMEOUT) && !defined(__WIN__) - - -static int pagecache_pthread_cond_wait(pthread_cond_t *cond, - pthread_mutex_t *mutex) -{ - int rc; - struct timeval now; /* time when we started waiting */ - struct timespec timeout; /* timeout value for the wait function */ - struct timezone tz; -#if defined(PAGECACHE_DEBUG) - int cnt=0; -#endif - - /* Get current time */ - gettimeofday(&now, &tz); - /* Prepare timeout value */ - timeout.tv_sec= now.tv_sec + PAGECACHE_TIMEOUT; - /* - timeval uses microseconds. - timespec uses nanoseconds. - 1 nanosecond = 1000 micro seconds - */ - timeout.tv_nsec= now.tv_usec * 1000; - KEYCACHE_THREAD_TRACE_END("started waiting"); -#if defined(PAGECACHE_DEBUG) - cnt++; - if (cnt % 100 == 0) - fprintf(pagecache_debug_log, "waiting...\n"); - fflush(pagecache_debug_log); -#endif - rc= pthread_cond_timedwait(cond, mutex, &timeout); - KEYCACHE_THREAD_TRACE_BEGIN("finished waiting"); - if (rc == ETIMEDOUT || rc == ETIME) - { -#if defined(PAGECACHE_DEBUG) - fprintf(pagecache_debug_log,"aborted by pagecache timeout\n"); - fclose(pagecache_debug_log); - abort(); -#endif - pagecache_dump(); - } - -#if defined(PAGECACHE_DEBUG) - KEYCACHE_DBUG_ASSERT(rc != ETIMEDOUT); -#else - assert(rc != ETIMEDOUT); -#endif - return rc; -} -#else -#if defined(PAGECACHE_DEBUG) -static int pagecache_pthread_cond_wait(pthread_cond_t *cond, - pthread_mutex_t *mutex) -{ - int rc; - KEYCACHE_THREAD_TRACE_END("started waiting"); - rc= pthread_cond_wait(cond, mutex); - KEYCACHE_THREAD_TRACE_BEGIN("finished waiting"); - return rc; -} -#endif -#endif /* defined(PAGECACHE_TIMEOUT) && !defined(__WIN__) */ - -#if defined(PAGECACHE_DEBUG) -static int ___pagecache_pthread_mutex_lock(pthread_mutex_t *mutex) -{ - int rc; - rc= pthread_mutex_lock(mutex); - KEYCACHE_THREAD_TRACE_BEGIN(""); - return rc; -} - - -static void ___pagecache_pthread_mutex_unlock(pthread_mutex_t *mutex) -{ - KEYCACHE_THREAD_TRACE_END(""); - pthread_mutex_unlock(mutex); -} - - -static int ___pagecache_pthread_cond_signal(pthread_cond_t *cond) -{ - int rc; - KEYCACHE_THREAD_TRACE("signal"); - rc= pthread_cond_signal(cond); - return rc; -} - - -#if defined(PAGECACHE_DEBUG_LOG) - - -static void pagecache_debug_print(const char * fmt, ...) -{ - va_list args; - va_start(args,fmt); - if (pagecache_debug_log) - { - VOID(vfprintf(pagecache_debug_log, fmt, args)); - VOID(fputc('\n',pagecache_debug_log)); - } - va_end(args); -} -#endif /* defined(PAGECACHE_DEBUG_LOG) */ - -#if defined(PAGECACHE_DEBUG_LOG) - - -void pagecache_debug_log_close(void) -{ - if (pagecache_debug_log) - fclose(pagecache_debug_log); -} -#endif /* defined(PAGECACHE_DEBUG_LOG) */ - -#endif /* defined(PAGECACHE_DEBUG) */ diff --git a/mysys/my_safehash.h b/mysys/my_safehash.h new file mode 100644 index 00000000000..53845a5fec7 --- /dev/null +++ b/mysys/my_safehash.h @@ -0,0 +1,58 @@ +/* Copyright (C) 2003 MySQL AB + + 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; either version 2 of the License, or + (at your option) any later version. + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ + +/* + Handling of multiple key caches + + The idea is to have a thread safe hash on the table name, + with a default key cache value that is returned if the table name is not in + the cache. +*/ + +#include + +/* + Struct to store a key and pointer to object +*/ + +typedef struct st_safe_hash_entry +{ + byte *key; + uint length; + byte *data; + struct st_safe_hash_entry *next, **prev; +} SAFE_HASH_ENTRY; + + +typedef struct st_safe_hash_with_default +{ +#ifdef THREAD + rw_lock_t mutex; +#endif + HASH hash; + byte *default_value; + SAFE_HASH_ENTRY *root; +} SAFE_HASH; + + +my_bool safe_hash_init(SAFE_HASH *hash, uint elements, + byte *default_value); +void safe_hash_free(SAFE_HASH *hash); +byte *safe_hash_search(SAFE_HASH *hash, const byte *key, uint length, + byte *def); +my_bool safe_hash_set(SAFE_HASH *hash, const byte *key, uint length, + byte *data); +void safe_hash_change(SAFE_HASH *hash, byte *old_data, byte *new_data); diff --git a/sql/handler.cc b/sql/handler.cc index ce9dc342713..fe31f912381 100644 --- a/sql/handler.cc +++ b/sql/handler.cc @@ -28,6 +28,10 @@ #include #include +#ifdef WITH_MARIA_STORAGE_ENGINE +#include +#endif + #ifdef WITH_PARTITION_STORAGE_ENGINE #include "ha_partition.h" #endif @@ -2773,6 +2777,99 @@ int ha_change_key_cache(KEY_CACHE *old_key_cache, } +/***************************************************************************** + pagecache handling. + + This code is only relevant for maria tables + + pagecache->cache may be 0 only in the case where a key cache is not + initialized or when we where not able to init the key cache in a previous + call to ha_init_pagecache() (probably out of memory) +*****************************************************************************/ + + +#ifdef WITH_MARIA_STORAGE_ENGINE + +/* Init a pagecache if it has not been initied before */ + +int ha_init_pagecache(const char *name, PAGECACHE *pagecache) +{ + DBUG_ENTER("ha_init_key_cache"); + + if (!pagecache->inited) + { + pthread_mutex_lock(&LOCK_global_system_variables); + long tmp_buff_size= (long) pagecache->param_buff_size; + long tmp_block_size= (long) pagecache->param_block_size; + uint division_limit= pagecache->param_division_limit; + uint age_threshold= pagecache->param_age_threshold; + pthread_mutex_unlock(&LOCK_global_system_variables); + DBUG_RETURN(!init_pagecache(pagecache, + tmp_buff_size, division_limit, age_threshold, + tmp_block_size)); + } + DBUG_RETURN(0); +} + + +/* Resize key cache */ +/* +TODO: uncomment when resize will be implemented +int ha_resize_pagecache(PAGECACHE *pagecache) +{ + DBUG_ENTER("ha_resize_pagecache"); + + if (pagecache->inited) + { + pthread_mutex_lock(&LOCK_global_system_variables); + long tmp_buff_size= (long) pagecache->param_buff_size; + long tmp_block_size= (long) pagecache->param_block_size; + uint division_limit= pagecache->param_division_limit; + uint age_threshold= pagecache->param_age_threshold; + pthread_mutex_unlock(&LOCK_global_system_variables); + DBUG_RETURN(!resize_pagecache(pagecache, tmp_block_size, + tmp_buff_size, + division_limit, age_threshold)); + } + DBUG_RETURN(0); +} +*/ + + +/* Change parameters for key cache (like size) */ + +int ha_change_pagecache_param(PAGECACHE *pagecache) +{ + if (pagecache->inited) + { + pthread_mutex_lock(&LOCK_global_system_variables); + uint division_limit= pagecache->param_division_limit; + uint age_threshold= pagecache->param_age_threshold; + pthread_mutex_unlock(&LOCK_global_system_variables); + change_pagecache_param(pagecache, division_limit, age_threshold); + } + return 0; +} + +/* Free memory allocated by a key cache */ + +int ha_end_pagecache(PAGECACHE *pagecache) +{ + end_pagecache(pagecache, 1); // Can never fail + return 0; +} + +/* Move all tables from one key cache to another one */ + +int ha_change_pagecache(PAGECACHE *old_pagecache, + PAGECACHE *new_pagecache) +{ + maria_change_pagecache(old_pagecache, new_pagecache); + return 0; +} + +#endif /* WITH_MARIA_STORAGE_ENGINE */ + /* Try to discover one table from handler(s) diff --git a/sql/handler.h b/sql/handler.h index 17e98803209..552e506ac4c 100644 --- a/sql/handler.h +++ b/sql/handler.h @@ -24,6 +24,7 @@ #include #include #include +#include #ifndef NO_HASH #define NO_HASH /* Not yet implemented */ @@ -823,7 +824,8 @@ typedef struct st_ha_check_opt ulong sort_buffer_size; uint flags; /* isam layer flags (e.g. for myisamchk) */ uint sql_flags; /* sql layer flags - for something myisamchk cannot do */ - KEY_CACHE *key_cache; /* new key cache when changing key cache */ + KEY_CACHE *key_cache; /* new key cache when changing key cache */ + PAGECACHE *pagecache; /* new pagecache when changing pagecache */ void init(); } HA_CHECK_OPT; @@ -1694,6 +1696,15 @@ int ha_resize_key_cache(KEY_CACHE *key_cache); int ha_change_key_cache_param(KEY_CACHE *key_cache); int ha_change_key_cache(KEY_CACHE *old_key_cache, KEY_CACHE *new_key_cache); int ha_end_key_cache(KEY_CACHE *key_cache); +/* pagecache */ +int ha_init_pagecache(const char *name, PAGECACHE *pagecache); +/* +TODO: uncomment when resizing will be implemented +int ha_resize_pagecache(PAGECACHE *pagecache); +*/ +int ha_change_pagecache_param(PAGECACHE *pagecache); +int ha_change_pagecache(PAGECACHE *old_pagecache, PAGECACHE *new_pagecache); +int ha_end_pagecache(PAGECACHE *pagecache); /* report to InnoDB that control passes to the client */ int ha_release_temporary_latches(THD *thd); diff --git a/sql/mysql_priv.h b/sql/mysql_priv.h index 82e71f59b2c..f6cfa52f006 100644 --- a/sql/mysql_priv.h +++ b/sql/mysql_priv.h @@ -1629,6 +1629,9 @@ extern pthread_cond_t COND_global_read_lock; extern pthread_attr_t connection_attrib; extern I_List threads; extern I_List key_caches; +#ifdef WITH_MARIA_STORAGE_ENGINE +extern I_List pagecaches; +#endif /* WITH_MARIA_STORAGE_ENGINE */ extern MY_BITMAP temp_pool; extern String my_empty_string; extern const String my_null_string; diff --git a/sql/mysqld.cc b/sql/mysqld.cc index d3cada2405f..3fae670a579 100644 --- a/sql/mysqld.cc +++ b/sql/mysqld.cc @@ -551,6 +551,9 @@ FILE *stderror_file=0; I_List threads; I_List key_caches; +#ifdef WITH_MARIA_STORAGE_ENGINE +I_List pagecaches; +#endif /* WITH_MARIA_STORAGE_ENGINE */ Rpl_filter* rpl_filter; Rpl_filter* binlog_filter; @@ -1209,7 +1212,13 @@ void clean_up(bool print_message) tc_log->close(); xid_cache_free(); delete_elements(&key_caches, (void (*)(const char*, gptr)) free_key_cache); +#ifdef WITH_MARIA_STORAGE_ENGINE + delete_elements(&pagecaches, (void (*)(const char*, gptr)) free_pagecache); +#endif /* WITH_MARIA_STORAGE_ENGINE */ multi_keycache_free(); +#ifdef WITH_MARIA_STORAGE_ENGINE + multi_pagecache_free(); +#endif free_status_vars(); end_thr_alarm(1); /* Free allocated memory */ my_free_open_file_info(); @@ -2105,6 +2114,10 @@ the problem, but since we have already crashed, something is definitely wrong\n\ and this may fail.\n\n"); fprintf(stderr, "key_buffer_size=%lu\n", (ulong) dflt_key_cache->key_cache_mem_size); +#ifdef WITH_MARIA_STORAGE_ENGINE + fprintf(stderr, "page_buffer_size=%lu\n", + (ulong) dflt_pagecache->mem_size); +#endif /* WITH_MARIA_STORAGE_ENGINE */ fprintf(stderr, "read_buffer_size=%ld\n", (long) global_system_variables.read_buff_size); fprintf(stderr, "max_used_connections=%lu\n", max_used_connections); fprintf(stderr, "max_connections=%lu\n", max_connections); @@ -2112,6 +2125,9 @@ and this may fail.\n\n"); fprintf(stderr, "It is possible that mysqld could use up to \n\ key_buffer_size + (read_buffer_size + sort_buffer_size)*max_connections = %lu K\n\ bytes of memory\n", ((ulong) dflt_key_cache->key_cache_mem_size + +#ifdef WITH_MARIA_STORAGE_ENGINE + (ulong) dflt_pagecache->mem_size + +#endif /* WITH_MARIA_STORAGE_ENGINE */ (global_system_variables.read_buff_size + global_system_variables.sortbuff_size) * max_connections)/ 1024); @@ -2667,6 +2683,16 @@ static int init_common_variables(const char *conf_file_name, int argc, get_options(argc,argv); set_server_version(); +#ifdef WITH_MARIA_STORAGE_ENGINE + if (!(dflt_pagecache= get_or_create_pagecache(maria_pagecache_base.str, + maria_pagecache_base.length))) + exit(1); +/* + maria_pagecache->param_buff_size= maria_pagecache_var.param_buff_size; + maria_pagecache->param_block_size= maria_block_size; +*/ +#endif + DBUG_PRINT("info",("%s Ver %s for %s on %s\n",my_progname, server_version, SYSTEM_TYPE,MACHINE_TYPE)); @@ -3345,6 +3371,9 @@ server."); /* call ha_init_key_cache() on all key caches to init them */ process_key_caches(&ha_init_key_cache); +#ifdef WITH_MARIA_STORAGE_ENGINE + process_pagecaches(&ha_init_pagecache); +#endif /* WITH_MARIA_STORAGE_ENGINE */ #if defined(HAVE_MLOCKALL) && defined(MCL_CURRENT) && !defined(EMBEDDED_LIBRARY) if (locked_in_memory && !getuid()) @@ -5949,7 +5978,7 @@ log and this option does nothing anymore.", "this to get better index handling (for all reads and multiple writes) to " "as much as you can afford; 64M on a 256M machine that mainly runs MySQL " "is quite common.", - (gptr*) &maria_key_cache_var.param_buff_size, (gptr*) 0, + (gptr*) &maria_pagecache_var.param_buff_size, (gptr*) 0, 0, (GET_ULL | GET_ASK_ADDR), REQUIRED_ARG, KEY_CACHE_SIZE, MALLOC_OVERHEAD, ~(ulong) 0, MALLOC_OVERHEAD, IO_SIZE, 0}, @@ -6143,6 +6172,33 @@ The minimum value for this variable is 4096.", (gptr*) &global_system_variables.optimizer_search_depth, (gptr*) &max_system_variables.optimizer_search_depth, 0, GET_ULONG, OPT_ARG, MAX_TABLES+1, 0, MAX_TABLES+2, 0, 1, 0}, +#ifdef WITH_MARIA_STORAGE_ENGINE + {"pagecache_age_threshold", OPT_KEY_CACHE_AGE_THRESHOLD, + "This characterizes the number of hits a hot block has to be untouched until it is considered aged enough to be downgraded to a warm block. This specifies the percentage ratio of that number of hits to the total number of blocks in key cache", + (gptr*) &dflt_pagecache_var.param_age_threshold, + (gptr*) 0, + 0, (GET_ULONG | GET_ASK_ADDR), REQUIRED_ARG, + 300, 100, ~0L, 0, 100, 0}, + {"pagecache_block_size", OPT_KEY_CACHE_BLOCK_SIZE, + "The default size of key cache blocks", + (gptr*) &dflt_pagecache_var.param_block_size, + (gptr*) 0, + 0, (GET_ULONG | GET_ASK_ADDR), REQUIRED_ARG, + KEY_CACHE_BLOCK_SIZE , 512, 1024*16, MALLOC_OVERHEAD, 512, 0}, + {"pagecache_buffer_size", OPT_KEY_BUFFER_SIZE, + "The size of the buffer used for index blocks for MyISAM tables. Increase this to get better index handling (for all reads and multiple writes) to as much as you can afford; 64M on a 256M machine that mainly runs MySQL is quite common.", + (gptr*) &dflt_pagecache_var.param_buff_size, + (gptr*) 0, + 0, (GET_ULL | GET_ASK_ADDR), + REQUIRED_ARG, KEY_CACHE_SIZE, MALLOC_OVERHEAD, ~(ulong) 0, MALLOC_OVERHEAD, + IO_SIZE, 0}, + {"pagecache_division_limit", OPT_KEY_CACHE_DIVISION_LIMIT, + "The minimum percentage of warm blocks in key cache", + (gptr*) &dflt_pagecache_var.param_division_limit, + (gptr*) 0, + 0, (GET_ULONG | GET_ASK_ADDR) , REQUIRED_ARG, 100, + 1, 100, 0, 1, 0}, +#endif /* WITH_MARIA_STORAGE_ENGINE */ {"plugin_dir", OPT_PLUGIN_DIR, "Directory for plugins.", (gptr*) &opt_plugin_dir_ptr, (gptr*) &opt_plugin_dir_ptr, 0, @@ -7035,20 +7091,21 @@ static void mysql_init_variables(void) threads.empty(); thread_cache.empty(); key_caches.empty(); +#ifdef WITH_MARIA_STORAGE_ENGINE + pagecaches.empty(); +#endif /* WITH_MARIA_STORAGE_ENGINE */ if (!(dflt_key_cache= get_or_create_key_cache(default_key_cache_base.str, default_key_cache_base.length))) exit(1); -#ifdef WITH_MARIA_STORAGE_ENGINE - if (!(maria_key_cache= get_or_create_key_cache(maria_key_cache_base.str, - maria_key_cache_base.length))) - exit(1); - maria_key_cache->param_buff_size= maria_key_cache_var.param_buff_size; - maria_key_cache->param_block_size= maria_block_size; -#endif - /* set key_cache_hash.default_value = dflt_key_cache */ + /* set key_cache_hash.default_value = dflt_key_cache */ multi_keycache_init(); +#ifdef WITH_MARIA_STORAGE_ENGINE + /* set pagecache_hash.default_value = dflt_pagecache */ + multi_pagecache_init(); +#endif + /* Set directory paths */ strmake(language, LANGUAGE, sizeof(language)-1); strmake(mysql_real_data_home, get_relative_path(DATADIR), @@ -8168,6 +8225,9 @@ void refresh_status(THD *thd) /* Reset the counters of all key caches (default and named). */ process_key_caches(reset_key_cache_counters); +#ifdef WITH_MARIA_STORAGE_ENGINE + process_pagecaches(reset_pagecache_counters); +#endif /* WITH_MARIA_STORAGE_ENGINE */ pthread_mutex_unlock(&LOCK_status); /* diff --git a/sql/set_var.cc b/sql/set_var.cc index bd68863ff97..5dd438ba95a 100644 --- a/sql/set_var.cc +++ b/sql/set_var.cc @@ -156,6 +156,9 @@ static void fix_thd_mem_root(THD *thd, enum_var_type type); static void fix_trans_mem_root(THD *thd, enum_var_type type); static void fix_server_id(THD *thd, enum_var_type type); static KEY_CACHE *create_key_cache(const char *name, uint length); +#ifdef WITH_MARIA_STORAGE_ENGINE +static PAGECACHE *create_pagecache(const char *name, uint length); +#endif /* WITH_MARIA_STORAGE_ENGINE */ void fix_sql_mode_var(THD *thd, enum_var_type type); static byte *get_error_count(THD *thd); static byte *get_warning_count(THD *thd); @@ -258,6 +261,14 @@ sys_var_key_cache_long sys_key_cache_division_limit("key_cache_division_limit", sys_var_key_cache_long sys_key_cache_age_threshold("key_cache_age_threshold", offsetof(KEY_CACHE, param_age_threshold)); +#ifdef WITH_MARIA_STORAGE_ENGINE +sys_var_pagecache_long sys_pagecache_division_limit("pagecache_division_limit", + offsetof(PAGECACHE, + param_division_limit)); +sys_var_pagecache_long sys_pagecache_age_threshold("pagecache_age_threshold", + offsetof(KEY_CACHE, + param_age_threshold)); +#endif /* WITH_MARIA_STORAGE_ENGINE */ sys_var_bool_ptr sys_local_infile("local_infile", &opt_local_infile); sys_var_trust_routine_creators @@ -855,6 +866,12 @@ SHOW_VAR init_vars[]= { SHOW_SYS}, {sys_key_cache_division_limit.name, (char*) &sys_key_cache_division_limit, SHOW_SYS}, +#ifdef WITH_MARIA_STORAGE_ENGINE + {sys_pagecache_age_threshold.name, (char*) &sys_pagecache_age_threshold, + SHOW_SYS}, + {sys_pagecache_division_limit.name, (char*) &sys_pagecache_division_limit, + SHOW_SYS}, +#endif /* WITH_MARIA_STORAGE_ENGINE */ {"language", language, SHOW_CHAR}, {"large_files_support", (char*) &opt_large_files, SHOW_BOOL}, {"large_page_size", (char*) &opt_large_page_size, SHOW_INT}, @@ -2396,10 +2413,13 @@ void sys_var_collation_server::set_default(THD *thd, enum_var_type type) LEX_STRING default_key_cache_base= {(char *) "default", 7 }; -LEX_STRING maria_key_cache_base= {(char *) "maria", 5 }; - static KEY_CACHE zero_key_cache; +#ifdef WITH_MARIA_STORAGE_ENGINE +LEX_STRING maria_pagecache_base= {(char *) "default", 7 }; +static PAGECACHE zero_pagecache; +#endif /* WITH_MARIA_STORAGE_ENGINE */ + KEY_CACHE *get_key_cache(LEX_STRING *cache_name) { safe_mutex_assert_owner(&LOCK_global_system_variables); @@ -2410,6 +2430,18 @@ KEY_CACHE *get_key_cache(LEX_STRING *cache_name) } +#ifdef WITH_MARIA_STORAGE_ENGINE +PAGECACHE *get_pagecache(LEX_STRING *cache_name) +{ + safe_mutex_assert_owner(&LOCK_global_system_variables); + if (!cache_name || ! cache_name->length) + cache_name= &default_key_cache_base; + return ((PAGECACHE*) find_named(&pagecaches, + cache_name->str, cache_name->length, 0)); +} +#endif /* WITH_MARIA_STORAGE_ENGINE */ + + byte *sys_var_key_cache_param::value_ptr(THD *thd, enum_var_type type, LEX_STRING *base) { @@ -2420,6 +2452,18 @@ byte *sys_var_key_cache_param::value_ptr(THD *thd, enum_var_type type, } +#ifdef WITH_MARIA_STORAGE_ENGINE +byte *sys_var_pagecache_param::value_ptr(THD *thd, enum_var_type type, + LEX_STRING *base) +{ + PAGECACHE *pagecache= get_pagecache(base); + if (!pagecache) + pagecache= &zero_pagecache; + return (byte*) pagecache + offset ; +} +#endif /* WITH_MARIA_STORAGE_ENGINE */ + + bool sys_var_key_buffer_size::update(THD *thd, set_var *var) { ulonglong tmp= var->save_result.ulonglong_value; @@ -2556,6 +2600,60 @@ end: } +#ifdef WITH_MARIA_STORAGE_ENGINE +bool sys_var_pagecache_long::update(THD *thd, set_var *var) +{ + ulong tmp= (ulong) var->value->val_int(); + LEX_STRING *base_name= &var->base; + bool error= 0; + + if (!base_name->length) + base_name= &maria_pagecache_base; + + pthread_mutex_lock(&LOCK_global_system_variables); + PAGECACHE *pagecache= get_pagecache(base_name); + + if (!pagecache && !(pagecache= create_pagecache(base_name->str, + base_name->length))) + { + error= 1; + goto end; + } + + /* + Abort if some other thread is changing the key cache + TODO: This should be changed so that we wait until the previous + assignment is done and then do the new assign + */ + if (pagecache->in_init) + goto end; + + *((ulong*) (((char*) pagecache) + offset))= + (ulong) getopt_ull_limit_value(tmp, option_limits); + + /* + Don't create a new key cache if it didn't exist + (pagecaches are created only when the user sets block_size) + */ + pagecache->in_init= 1; + + pthread_mutex_unlock(&LOCK_global_system_variables); + + /* + TODO: uncomment whan it will be implemented + error= (bool) (ha_resize_pagecache(pagecache)); + */ + + pthread_mutex_lock(&LOCK_global_system_variables); + pagecache->in_init= 0; + +end: + pthread_mutex_unlock(&LOCK_global_system_variables); + return error; +} +#endif /* WITH_MARIA_STORAGE_ENGINE */ + + bool sys_var_log_state::update(THD *thd, set_var *var) { bool res= 0; @@ -3872,12 +3970,86 @@ bool process_key_caches(int (* func) (const char *name, KEY_CACHE *)) } +#ifdef WITH_MARIA_STORAGE_ENGINE + +static PAGECACHE *create_pagecache(const char *name, uint length) +{ + PAGECACHE *pagecache; + DBUG_ENTER("create_pagecache"); + DBUG_PRINT("enter",("name: %.*s", length, name)); + + if ((pagecache= (PAGECACHE*) my_malloc(sizeof(PAGECACHE), + MYF(MY_ZEROFILL | MY_WME)))) + { + if (!new NAMED_LIST(&pagecaches, name, length, (gptr) pagecache)) + { + my_free((char*) pagecache, MYF(0)); + pagecache= 0; + } + else + { + /* + Set default values for a key cache + The values in dflt_pagecache_var is set by my_getopt() at startup + We don't set 'buff_size' as this is used to enable the key cache + */ + pagecache->param_buff_size= (dflt_pagecache_var.param_buff_size ? + dflt_pagecache_var.param_buff_size: + KEY_CACHE_SIZE); + pagecache->param_block_size= dflt_pagecache_var.param_block_size; + pagecache->param_division_limit= dflt_pagecache_var.param_division_limit; + pagecache->param_age_threshold= dflt_pagecache_var.param_age_threshold; + } + } + DBUG_RETURN(pagecache); +} + + +PAGECACHE *get_or_create_pagecache(const char *name, uint length) +{ + LEX_STRING pagecache_name; + PAGECACHE *pagecache; + + pagecache_name.str= (char *) name; + pagecache_name.length= length; + pthread_mutex_lock(&LOCK_global_system_variables); + if (!(pagecache= get_pagecache(&pagecache_name))) + pagecache= create_pagecache(name, length); + pthread_mutex_unlock(&LOCK_global_system_variables); + return pagecache; +} + + +void free_pagecache(const char *name, PAGECACHE *pagecache) +{ + ha_end_pagecache(pagecache); + my_free((char*) pagecache, MYF(0)); +} + + +bool process_pagecaches(int (* func) (const char *name, PAGECACHE *)) +{ + I_List_iterator it(pagecaches); + NAMED_LIST *element; + + while ((element= it++)) + { + PAGECACHE *pagecache= (PAGECACHE *) element->data; + func(element->name, pagecache); + } + return 0; +} + +#endif /* WITH_MARIA_STORAGE_ENGINE */ + + void sys_var_trust_routine_creators::warn_deprecated(THD *thd) { WARN_DEPRECATED(thd, "5.2", "log_bin_trust_routine_creators", "'log_bin_trust_function_creators'"); } + void sys_var_trust_routine_creators::set_default(THD *thd, enum_var_type type) { warn_deprecated(thd); diff --git a/sql/set_var.h b/sql/set_var.h index 17f9b248c07..a415f629cb9 100644 --- a/sql/set_var.h +++ b/sql/set_var.h @@ -749,6 +749,33 @@ public: }; +#ifdef WITH_MARIA_STORAGE_ENGINE +class sys_var_pagecache_param :public sys_var +{ +protected: + size_t offset; +public: + sys_var_pagecache_param(const char *name_arg, size_t offset_arg) + :sys_var(name_arg), offset(offset_arg) + {} + byte *value_ptr(THD *thd, enum_var_type type, LEX_STRING *base); + bool check_default(enum_var_type type) { return 1; } + bool is_struct() { return 1; } +}; + + +class sys_var_pagecache_long :public sys_var_pagecache_param +{ +public: + sys_var_pagecache_long(const char *name_arg, size_t offset_arg) + :sys_var_pagecache_param(name_arg, offset_arg) + {} + bool update(THD *thd, set_var *var); + SHOW_TYPE type() { return SHOW_LONG; } +}; +#endif /* WITH_MARIA_STORAGE_ENGINE */ + + class sys_var_thd_date_time_format :public sys_var_thd { DATE_TIME_FORMAT *SV::*offset; @@ -1095,7 +1122,11 @@ public: my_free((char*) name, MYF(0)); } friend bool process_key_caches(int (* func) (const char *name, - KEY_CACHE *)); + KEY_CACHE *)); +#ifdef WITH_MARIA_STORAGE_ENGINE + friend bool process_pagecaches(int (* func) (const char *name, + PAGECACHE *)); +#endif /* WITH_MARIA_STORAGE_ENGINE */ friend void delete_elements(I_List *list, void (*free_element)(const char*, gptr)); }; @@ -1105,7 +1136,9 @@ public: extern sys_var_thd_bool sys_old_alter_table; extern sys_var_thd_bool sys_old_passwords; extern LEX_STRING default_key_cache_base; -extern LEX_STRING maria_key_cache_base; +#ifdef WITH_MARIA_STORAGE_ENGINE +extern LEX_STRING maria_pagecache_base; +#endif /* WITH_MARIA_STORAGE_ENGINE */ /* For sql_yacc */ struct sys_var_with_base @@ -1143,3 +1176,8 @@ void free_key_cache(const char *name, KEY_CACHE *key_cache); bool process_key_caches(int (* func) (const char *name, KEY_CACHE *)); void delete_elements(I_List *list, void (*free_element)(const char*, gptr)); +#ifdef WITH_MARIA_STORAGE_ENGINE +PAGECACHE *get_or_create_pagecache(const char *name, uint length); +void free_pagecache(const char *name, PAGECACHE *pagecache); +bool process_pagecaches(int (* func) (const char *name, PAGECACHE *)); +#endif /* WITH_MARIA_STORAGE_ENGINE */ diff --git a/storage/maria/Makefile.am b/storage/maria/Makefile.am index b6c713571d7..cacc88147a9 100644 --- a/storage/maria/Makefile.am +++ b/storage/maria/Makefile.am @@ -111,7 +111,8 @@ libmaria_a_SOURCES = ma_init.c ma_open.c ma_extra.c ma_info.c ma_rkey.c \ ma_ft_nlq_search.c ft_maria.c ma_sort.c \ ha_maria.cc trnman.c lockman.c tablockman.c \ ma_rt_index.c ma_rt_key.c ma_rt_mbr.c ma_rt_split.c \ - ma_sp_key.c ma_control_file.c ma_loghandler.c + ma_sp_key.c ma_control_file.c ma_loghandler.c \ + ma_pagecache.c ma_pagecaches.c CLEANFILES = test?.MA? FT?.MA? isam.log ma_test_all ma_rt_test.MA? sp_test.MA? SUFFIXES = .sh diff --git a/storage/maria/ha_maria.cc b/storage/maria/ha_maria.cc index 102f987f01a..cb64cf6f693 100644 --- a/storage/maria/ha_maria.cc +++ b/storage/maria/ha_maria.cc @@ -198,7 +198,7 @@ const char *ha_maria::index_type(uint key_number) #ifdef HAVE_REPLICATION int ha_maria::net_read_dump(NET * net) { - int data_fd= file->dfile; + int data_fd= file->dfile.file; int error= 0; my_seek(data_fd, 0L, MY_SEEK_SET, MYF(MY_WME)); @@ -231,7 +231,7 @@ int ha_maria::dump(THD * thd, int fd) NET *net= &thd->net; uint block_size= share->block_size; my_off_t bytes_to_read= share->state.state.data_file_length; - int data_fd= file->dfile; + int data_fd= file->dfile.file; byte *buf= (byte *) my_malloc(block_size, MYF(MY_WME)); if (!buf) return ENOMEM; @@ -424,7 +424,7 @@ int ha_maria::check(THD * thd, HA_CHECK_OPT * check_opt) { uint old_testflag= param.testflag; param.testflag |= T_MEDIUM; - if (!(error= init_io_cache(¶m.read_cache, file->dfile, + if (!(error= init_io_cache(¶m.read_cache, file->dfile.file, my_default_record_cache_size, READ_CACHE, share->pack.header_length, 1, MYF(MY_WME)))) { @@ -827,7 +827,7 @@ int ha_maria::repair(THD *thd, HA_CHECK ¶m, bool optimize) int ha_maria::assign_to_keycache(THD * thd, HA_CHECK_OPT *check_opt) { - KEY_CACHE *new_key_cache= check_opt->key_cache; + PAGECACHE *new_pagecache= check_opt->pagecache; const char *errmsg= 0; int error= HA_ADMIN_OK; ulonglong map= ~(ulonglong) 0; @@ -848,7 +848,7 @@ int ha_maria::assign_to_keycache(THD * thd, HA_CHECK_OPT *check_opt) map= kmap.to_ulonglong(); } - if ((error= maria_assign_to_key_cache(file, map, new_key_cache))) + if ((error= maria_assign_to_pagecache(file, map, new_pagecache))) { char buf[STRING_BUFFER_USUAL_SIZE]; my_snprintf(buf, sizeof(buf), diff --git a/storage/maria/ma_bitmap.c b/storage/maria/ma_bitmap.c index 5ed5a776658..1192396ca19 100644 --- a/storage/maria/ma_bitmap.c +++ b/storage/maria/ma_bitmap.c @@ -122,10 +122,13 @@ static inline my_bool write_changed_bitmap(MARIA_SHARE *share, MARIA_FILE_BITMAP *bitmap) { - return (key_cache_write(share->key_cache, - bitmap->file, bitmap->page * bitmap->block_size, 0, - (byte*) bitmap->map, - bitmap->block_size, bitmap->block_size, 1)); + DBUG_ASSERT(share->pagecache->block_size == bitmap->block_size); + return (pagecache_write(share->pagecache, + (PAGECACHE_FILE*)&bitmap->file, bitmap->page, 0, + (byte*) bitmap->map, PAGECACHE_PLAIN_PAGE, + PAGECACHE_LOCK_LEFT_UNLOCKED, + PAGECACHE_PIN_LEFT_PINNED, + PAGECACHE_WRITE_DELAY, 0)); } /* @@ -398,10 +401,12 @@ my_bool _ma_read_bitmap_page(MARIA_SHARE *share, MARIA_FILE_BITMAP *bitmap, DBUG_RETURN(0); } bitmap->used_size= bitmap->total_size; - res= key_cache_read(share->key_cache, - bitmap->file, position, 0, + DBUG_ASSERT(share->pagecache->block_size == bitmap->block_size); + res= pagecache_read(share->pagecache, + (PAGECACHE_FILE*)&bitmap->file, page, 0, (byte*) bitmap->map, - bitmap->block_size, bitmap->block_size, 0) == 0; + PAGECACHE_PLAIN_PAGE, + PAGECACHE_LOCK_LEFT_UNLOCKED, 0) == 0; #ifndef DBUG_OFF if (!res) memcpy(bitmap->map+ bitmap->block_size, bitmap->map, bitmap->block_size); diff --git a/storage/maria/ma_blockrec.c b/storage/maria/ma_blockrec.c index f1345b2c2f3..f1a1df23358 100644 --- a/storage/maria/ma_blockrec.c +++ b/storage/maria/ma_blockrec.c @@ -342,7 +342,7 @@ my_bool _ma_once_init_block_row(MARIA_SHARE *share, File data_file) my_bool _ma_once_end_block_row(MARIA_SHARE *share) { int res= _ma_bitmap_end(share); - if (flush_key_blocks(share->key_cache, share->bitmap.file, + if (flush_pagecache_blocks(share->pagecache, (PAGECACHE_FILE*)&share->bitmap, share->temporary ? FLUSH_IGNORE_CHANGED : FLUSH_RELEASE)) res= 1; @@ -845,11 +845,14 @@ static my_bool get_head_or_tail_page(MARIA_HA *info, else { byte *dir; + /* TODO: lock the page */ /* Read old page */ - if (!(res->buff= key_cache_read(info->s->key_cache, - info->dfile, - (my_off_t) block->page * block_size, 0, - buff, block_size, block_size, 0))) + DBUG_ASSERT(info->s->pagecache->block_size == block_size); + if (!(res->buff= pagecache_read(info->s->pagecache, + &info->dfile, + (my_off_t) block->page, 0, + buff, PAGECACHE_PLAIN_PAGE, + PAGECACHE_LOCK_LEFT_UNLOCKED, 0))) DBUG_RETURN(1); DBUG_ASSERT((res->buff[PAGE_TYPE_OFFSET] & PAGE_TYPE_MASK) == page_type); if (!(dir= find_free_position(buff, block_size, &res->offset, @@ -912,6 +915,7 @@ static my_bool write_tail(MARIA_HA *info, (ulong) block->page, length)); info->keybuff_used= 1; + /* page will be pinned & locked by get_head_or_tail_page */ if (get_head_or_tail_page(info, block, info->keyread_buff, length, TAIL_PAGE, &row_pos)) DBUG_RETURN(1); @@ -936,9 +940,15 @@ static my_bool write_tail(MARIA_HA *info, position= (my_off_t) block->page * block_size; if (info->state->data_file_length <= position) info->state->data_file_length= position + block_size; - DBUG_RETURN(key_cache_write(share->key_cache, - info->dfile, position, 0, - row_pos.buff, block_size, block_size, 1)); + /* TODO: left the page pinned (or pin it if it is new) and unlock\ + the page (do not lock if it is new */ + DBUG_ASSERT(share->pagecache->block_size == block_size); + DBUG_RETURN(pagecache_write(share->pagecache, + &info->dfile, block->page, 0, + row_pos.buff,PAGECACHE_PLAIN_PAGE, + PAGECACHE_LOCK_LEFT_UNLOCKED, + PAGECACHE_PIN_LEFT_PINNED, + PAGECACHE_WRITE_DELAY, 0)); } @@ -969,7 +979,7 @@ static my_bool write_full_pages(MARIA_HA *info, DBUG_PRINT("enter", ("length: %lu page: %lu page_count: %lu", (ulong) length, (ulong) block->page, (ulong) block->page_count)); - + info->keybuff_used= 1; page= block->page; page_count= block->page_count; @@ -1001,9 +1011,18 @@ static my_bool write_full_pages(MARIA_HA *info, memcpy(buff + LSN_SIZE + PAGE_TYPE_SIZE, data, copy_length); length-= copy_length; - if (key_cache_write(share->key_cache, - info->dfile, (my_off_t) page * block_size, 0, - buff, block_size, block_size, 1)) + /* + TODO: replace PAGECACHE_PLAIN_PAGE with PAGECACHE_LSN_PAGE when + LSN on the pages will be implemented + */ + DBUG_ASSERT(share->pagecache->block_size == block_size); + if (pagecache_write(share->pagecache, + &info->dfile, page, 0, + buff, PAGECACHE_PLAIN_PAGE, + PAGECACHE_LOCK_LEFT_UNLOCKED, + PAGECACHE_PIN_LEFT_PINNED, + PAGECACHE_WRITE_DELAY, + 0)) DBUG_RETURN(1); page++; block->used= BLOCKUSED_USED; @@ -1133,7 +1152,7 @@ static my_bool write_block_record(MARIA_HA *info, const byte *record, To avoid double copying of data, we copy as many columns that fits into the page. The rest goes into info->packed_row. - Using an extra buffer, instead of doing continous writes to different + Using an extra buffer, instead of doing continuous writes to different pages, uses less code and we don't need to have to do a complex call for every data segment we want to store. */ @@ -1363,14 +1382,14 @@ static my_bool write_block_record(MARIA_HA *info, const byte *record, - Bitmap code allocated a tail page we don't need. - The last full page allocated needs to be changed to a tail page (Because we put more data than we thought on the head page) - - The reserved pages in bitmap_blocks for the main page has one of + + The reserved pages in bitmap_blocks for the main page has one of the following allocations: - Full pages, with following blocks: # * full pages empty page ; To be used if we change last full to tail page. This has 'count' = 0. - tail page (optional, if last full page was part full) + tail page (optional, if last full page was part full) - One tail page */ @@ -1485,9 +1504,13 @@ static my_bool write_block_record(MARIA_HA *info, const byte *record, position= (my_off_t) head_block->page * block_size; if (info->state->data_file_length <= position) info->state->data_file_length= position + block_size; - if (key_cache_write(share->key_cache, - info->dfile, position, 0, - page_buff, share->block_size, share->block_size, 1)) + DBUG_ASSERT(share->pagecache->block_size == block_size); + if (pagecache_write(share->pagecache, + &info->dfile, head_block->page, 0, + page_buff, PAGECACHE_PLAIN_PAGE, + PAGECACHE_LOCK_LEFT_UNLOCKED, + PAGECACHE_PIN_LEFT_PINNED, + PAGECACHE_WRITE_DELAY, 0)) goto disk_err; if (tmp_data_used) @@ -1557,7 +1580,8 @@ MARIA_RECORD_POS _ma_write_init_block_record(MARIA_HA *info, calc_record_size(info, record, &info->cur_row); if (_ma_bitmap_find_place(info, &info->cur_row, blocks)) - DBUG_RETURN(HA_OFFSET_ERROR); /* Error reding bitmap */ + DBUG_RETURN(HA_OFFSET_ERROR); /* Error reading bitmap */ + /* page will be pinned & locked by get_head_or_tail_page */ if (get_head_or_tail_page(info, blocks->block, info->buff, info->s->base.min_row_length, HEAD_PAGE, &row_pos)) DBUG_RETURN(HA_OFFSET_ERROR); @@ -1665,9 +1689,11 @@ my_bool _ma_update_block_record(MARIA_HA *info, MARIA_RECORD_POS record_pos, calc_record_size(info, record, new_row); page= ma_recordpos_to_page(record_pos); - if (!(buff= key_cache_read(info->s->key_cache, - info->dfile, (my_off_t) page * block_size, 0, - info->buff, block_size, block_size, 0))) + DBUG_ASSERT(info->s->pagecache->block_size == block_size); + if (!(buff= pagecache_read(info->s->pagecache, + &info->dfile, (my_off_t) page, 0, + info->buff, PAGECACHE_PLAIN_PAGE, + PAGECACHE_LOCK_LEFT_UNLOCKED, 0))) DBUG_RETURN(1); org_empty_size= uint2korr(buff + EMPTY_SPACE_OFFSET); rownr= ma_recordpos_to_offset(record_pos); @@ -1779,14 +1805,15 @@ static my_bool delete_head_or_tail(MARIA_HA *info, uint number_of_records, empty_space, length; uint block_size= share->block_size; byte *buff, *dir; - my_off_t position; DBUG_ENTER("delete_head_or_tail"); info->keybuff_used= 1; - if (!(buff= key_cache_read(share->key_cache, - info->dfile, page * block_size, 0, + DBUG_ASSERT(info->s->pagecache->block_size == block_size); + if (!(buff= pagecache_read(share->pagecache, + &info->dfile, page, 0, info->keyread_buff, - block_size, block_size, 0))) + PAGECACHE_PLAIN_PAGE, + PAGECACHE_LOCK_LEFT_UNLOCKED, 0))) DBUG_RETURN(1); number_of_records= (uint) ((uchar *) buff)[DIR_ENTRY_OFFSET]; @@ -1825,10 +1852,13 @@ static my_bool delete_head_or_tail(MARIA_HA *info, { int2store(buff + EMPTY_SPACE_OFFSET, empty_space); buff[PAGE_TYPE_OFFSET]|= (byte) PAGE_CAN_BE_COMPACTED; - position= (my_off_t) page * block_size; - if (key_cache_write(share->key_cache, - info->dfile, position, 0, - buff, block_size, block_size, 1)) + DBUG_ASSERT(share->pagecache->block_size == block_size); + if (pagecache_write(share->pagecache, + &info->dfile, page, 0, + buff, PAGECACHE_PLAIN_PAGE, + PAGECACHE_LOCK_LEFT_UNLOCKED, + PAGECACHE_PIN_LEFT_PINNED, + PAGECACHE_WRITE_DELAY, 0)) DBUG_RETURN(1); } else @@ -2032,10 +2062,11 @@ static byte *read_next_extent(MARIA_HA *info, MARIA_EXTENT_CURSOR *extent, info->cur_row.empty_bits= info->cur_row.empty_bits_buffer; } - if (!(buff= key_cache_read(share->key_cache, - info->dfile, extent->page * share->block_size, 0, - info->buff, - share->block_size, share->block_size, 0))) + DBUG_ASSERT(share->pagecache->block_size == share->block_size); + if (!(buff= pagecache_read(share->pagecache, + &info->dfile, extent->page, 0, + info->buff, PAGECACHE_PLAIN_PAGE, + PAGECACHE_LOCK_LEFT_UNLOCKED, 0))) { /* check if we tried to read over end of file (ie: bad data in record) */ if ((extent->page + 1) * share->block_size > info->state->data_file_length) @@ -2441,18 +2472,18 @@ int _ma_read_block_record(MARIA_HA *info, byte *record, MARIA_RECORD_POS record_pos) { byte *data, *end_of_data, *buff; - my_off_t page; uint offset; uint block_size= info->s->block_size; DBUG_ENTER("_ma_read_block_record"); DBUG_PRINT("enter", ("rowid: %lu", (long) record_pos)); - page= ma_recordpos_to_page(record_pos) * block_size; offset= ma_recordpos_to_offset(record_pos); - - if (!(buff= key_cache_read(info->s->key_cache, - info->dfile, page, 0, info->buff, - block_size, block_size, 1))) + + DBUG_ASSERT(info->s->pagecache->block_size == block_size); + if (!(buff= pagecache_read(info->s->pagecache, + &info->dfile, ma_recordpos_to_page(record_pos), 0, + info->buff, PAGECACHE_PLAIN_PAGE, + PAGECACHE_LOCK_LEFT_UNLOCKED, 0))) DBUG_RETURN(1); DBUG_ASSERT((buff[PAGE_TYPE_OFFSET] & PAGE_TYPE_MASK) == HEAD_PAGE); if (!(data= get_record_position(buff, block_size, offset, &end_of_data))) @@ -2639,11 +2670,11 @@ restart_bitmap_scan: page= (info->scan.bitmap_page + 1 + (data - info->scan.bitmap_buff) / 6 * 16 + bit_pos - 1); info->scan.row_base_page= ma_recordpos(page, 0); - if (!(key_cache_read(info->s->key_cache, - info->dfile, - (my_off_t) page * block_size, - 0, info->scan.page_buff, - block_size, block_size, 0))) + if (!(pagecache_read(info->s->pagecache, + &info->dfile, + page, 0, info->scan.page_buff, + PAGECACHE_PLAIN_PAGE, + PAGECACHE_LOCK_LEFT_UNLOCKED, 0))) DBUG_RETURN(my_errno); if (((info->scan.page_buff[PAGE_TYPE_OFFSET] & PAGE_TYPE_MASK) != HEAD_PAGE) || @@ -2679,8 +2710,10 @@ restart_bitmap_scan: { DBUG_RETURN((my_errno= HA_ERR_END_OF_FILE)); } - if (!(key_cache_read(info->s->key_cache, info->dfile, filepos, - 0, info->scan.bitmap_buff, block_size, block_size, 0))) + if (!(pagecache_read(info->s->pagecache, &info->dfile, + info->scan.bitmap_page, + 0, info->scan.bitmap_buff, PAGECACHE_PLAIN_PAGE, + PAGECACHE_LOCK_LEFT_UNLOCKED, 0))) DBUG_RETURN(my_errno); /* Skip scanning 'bits' in bitmap scan code */ info->scan.bitmap_pos= info->scan.bitmap_buff - 6; diff --git a/storage/maria/ma_check.c b/storage/maria/ma_check.c index ccce19de994..98c3d56000a 100644 --- a/storage/maria/ma_check.c +++ b/storage/maria/ma_check.c @@ -104,7 +104,7 @@ void maria_chk_init(HA_CHECK *param) param->myf_rw=MYF(MY_NABP | MY_WME | MY_WAIT_IF_FULL); param->start_check_pos=0; param->max_record_length= LONGLONG_MAX; - param->key_cache_block_size= KEY_CACHE_BLOCK_SIZE; + param->pagecache_block_size= KEY_CACHE_BLOCK_SIZE; param->stats_method= MI_STATS_METHOD_NULLS_NOT_EQUAL; } @@ -175,7 +175,7 @@ int maria_chk_del(HA_CHECK *param, register MARIA_HA *info, uint test_flag) printf(" %9s",llstr(next_link,buff)); if (next_link >= info->state->data_file_length) goto wrong; - if (my_pread(info->dfile,(char*) buff,delete_link_length, + if (my_pread(info->dfile.file, (char*) buff, delete_link_length, next_link,MYF(MY_NABP))) { if (test_flag & T_VERBOSE) puts(""); @@ -287,10 +287,13 @@ static int check_k_link(HA_CHECK *param, register MARIA_HA *info, /* purecov: end */ } - if (!(buff=key_cache_read(info->s->key_cache, - info->s->kfile, next_link, DFLT_INIT_HITS, - (byte*) info->buff, - block_size, block_size, 1))) + DBUG_ASSERT(info->s->pagecache->block_size == block_size); + if (!(buff= pagecache_read(info->s->pagecache, + &info->s->kfile, next_link/block_size, + DFLT_INIT_HITS, + (byte*) info->buff, + PAGECACHE_PLAIN_PAGE, + PAGECACHE_LOCK_LEFT_UNLOCKED, 0))) { /* purecov: begin tested */ _ma_check_print_error(param, "key cache read error for block: %s", @@ -326,10 +329,10 @@ int maria_chk_size(HA_CHECK *param, register MARIA_HA *info) puts("- check file-size"); /* The following is needed if called externally (not from maria_chk) */ - flush_key_blocks(info->s->key_cache, - info->s->kfile, FLUSH_FORCE_WRITE); + flush_pagecache_blocks(info->s->pagecache, + &info->s->kfile, FLUSH_FORCE_WRITE); - size=my_seek(info->s->kfile,0L,MY_SEEK_END,MYF(0)); + size= my_seek(info->s->kfile.file, 0L, MY_SEEK_END, MYF(0)); if ((skr=(my_off_t) info->state->key_file_length) != size) { /* Don't give error if file generated by mariapack */ @@ -353,7 +356,7 @@ int maria_chk_size(HA_CHECK *param, register MARIA_HA *info) llstr(info->state->key_file_length,buff), llstr(info->s->base.max_key_file_length-1,buff)); - size=my_seek(info->dfile,0L,MY_SEEK_END,MYF(0)); + size= my_seek(info->dfile.file, 0L, MY_SEEK_END, MYF(0)); skr=(my_off_t) info->state->data_file_length; if (info->s->options & HA_OPTION_COMPRESS_RECORD) skr+= MEMMAP_EXTRA_MARGIN; @@ -588,7 +591,7 @@ static int chk_index_down(HA_CHECK *param, MARIA_HA *info, { /* purecov: begin tested */ /* Give it a chance to fit in the real file size. */ - my_off_t max_length= my_seek(info->s->kfile, 0L, MY_SEEK_END, MYF(0)); + my_off_t max_length= my_seek(info->s->kfile.file, 0L, MY_SEEK_END, MYF(0)); _ma_check_print_error(param, "Invalid key block position: %s " "key block size: %u file_length: %s", llstr(page, llbuff), keyinfo->block_length, @@ -1920,10 +1923,10 @@ int maria_repair(HA_CHECK *param, register MARIA_HA *info, param->testflag|=T_CALC_CHECKSUM; if (!param->using_global_keycache) - VOID(init_key_cache(maria_key_cache, param->key_cache_block_size, - param->use_buffers, 0, 0)); + VOID(init_pagecache(maria_pagecache, param->use_buffers, 0, 0, + param->pagecache_block_size)); - if (init_io_cache(¶m->read_cache,info->dfile, + if (init_io_cache(¶m->read_cache, info->dfile.file, (uint) param->read_buffer_length, READ_CACHE,share->pack.header_length,1,MYF(MY_WME))) { @@ -1959,8 +1962,8 @@ int maria_repair(HA_CHECK *param, register MARIA_HA *info, goto err; } if (new_header_length && - maria_filecopy(param,new_file,info->dfile,0L,new_header_length, - "datafile-header")) + maria_filecopy(param, new_file, info->dfile.file, 0L, + new_header_length, "datafile-header")) goto err; info->s->state.dellink= HA_OFFSET_ERROR; info->rec_cache.file=new_file; @@ -1973,7 +1976,7 @@ int maria_repair(HA_CHECK *param, register MARIA_HA *info, sort_param.pos=sort_param.max_pos=share->pack.header_length; sort_param.filepos=new_header_length; param->read_cache.end_of_file=sort_info.filelength= - my_seek(info->dfile,0L,MY_SEEK_END,MYF(0)); + my_seek(info->dfile.file, 0L, MY_SEEK_END, MYF(0)); sort_info.dupp=0; sort_param.fix_datafile= (my_bool) (! rep_quick); sort_param.master=1; @@ -2049,7 +2052,7 @@ int maria_repair(HA_CHECK *param, register MARIA_HA *info, { VOID(fputs(" \r",stdout)); VOID(fflush(stdout)); } - if (my_chsize(share->kfile,info->state->key_file_length,0,MYF(0))) + if (my_chsize(share->kfile.file, info->state->key_file_length, 0, MYF(0))) { _ma_check_print_warning(param, "Can't change size of indexfile, error: %d", @@ -2079,8 +2082,8 @@ int maria_repair(HA_CHECK *param, register MARIA_HA *info, if (!rep_quick) { - my_close(info->dfile,MYF(0)); - info->dfile=new_file; + my_close(info->dfile.file, MYF(0)); + info->dfile.file= new_file; info->state->data_file_length=sort_param.filepos; share->state.version=(ulong) time((time_t*) 0); /* Force reopen */ } @@ -2113,7 +2116,7 @@ err: if (new_file >= 0) { my_close(new_file,MYF(0)); - info->dfile=new_file= -1; + info->dfile.file= new_file= -1; if (maria_change_to_newfile(share->data_file_name,MARIA_NAME_DEXT, DATA_TMP_EXT, (param->testflag & T_BACKUP_DATA ? MYF(MY_REDEL_MAKE_BACKUP): MYF(0))) || @@ -2140,7 +2143,7 @@ err: VOID(end_io_cache(¶m->read_cache)); info->opt_flag&= ~(READ_CACHE_USED | WRITE_CACHE_USED); VOID(end_io_cache(&info->rec_cache)); - got_error|=_ma_flush_blocks(param, share->key_cache, share->kfile); + got_error|=_ma_flush_blocks(param, share->pagecache, &share->kfile); if (!got_error && (param->testflag & T_UNPACK)) restore_data_file_type(share); share->state.changed|= (STATE_NOT_OPTIMIZED_KEYS | STATE_NOT_SORTED_PAGES | @@ -2283,15 +2286,16 @@ void maria_lock_memory(HA_CHECK *param __attribute__((unused))) /* Flush all changed blocks to disk */ -int _ma_flush_blocks(HA_CHECK *param, KEY_CACHE *key_cache, File file) +int _ma_flush_blocks(HA_CHECK *param, PAGECACHE *pagecache, + PAGECACHE_FILE *file) { - if (flush_key_blocks(key_cache, file, FLUSH_RELEASE)) + if (flush_pagecache_blocks(pagecache, file, FLUSH_RELEASE)) { _ma_check_print_error(param,"%d when trying to write bufferts",my_errno); return(1); } if (!param->using_global_keycache) - end_key_cache(key_cache,1); + end_pagecache(pagecache,1); return 0; } /* _ma_flush_blocks */ @@ -2323,7 +2327,7 @@ int maria_sort_index(HA_CHECK *param, register MARIA_HA *info, my_string name) param->temp_filename); DBUG_RETURN(-1); } - if (maria_filecopy(param, new_file,share->kfile,0L, + if (maria_filecopy(param, new_file, share->kfile.file, 0L, (ulong) share->base.keystart, "headerblock")) goto err; @@ -2346,7 +2350,8 @@ int maria_sort_index(HA_CHECK *param, register MARIA_HA *info, my_string name) } /* Flush key cache for this file if we are calling this outside maria_chk */ - flush_key_blocks(share->key_cache,share->kfile, FLUSH_IGNORE_CHANGED); + flush_pagecache_blocks(share->pagecache, &share->kfile, + FLUSH_IGNORE_CHANGED); share->state.version=(ulong) time((time_t*) 0); old_state= share->state; /* save state if not stored */ @@ -2357,8 +2362,8 @@ int maria_sort_index(HA_CHECK *param, register MARIA_HA *info, my_string name) /* Put same locks as old file */ share->r_locks= share->w_locks= share->tot_locks= 0; (void) _ma_writeinfo(info,WRITEINFO_UPDATE_KEYFILE); - VOID(my_close(share->kfile,MYF(MY_WME))); - share->kfile = -1; + VOID(my_close(share->kfile.file, MYF(MY_WME))); + share->kfile.file = -1; VOID(my_close(new_file,MYF(MY_WME))); if (maria_change_to_newfile(share->index_file_name, MARIA_NAME_IEXT, INDEX_TMP_EXT, MYF(0)) || @@ -2598,18 +2603,18 @@ int maria_repair_by_sort(HA_CHECK *param, register MARIA_HA *info, alloc_key_blocks(param, (uint) param->sort_key_blocks, share->base.max_key_block_length)) - || init_io_cache(¶m->read_cache,info->dfile, + || init_io_cache(¶m->read_cache, info->dfile.file, (uint) param->read_buffer_length, READ_CACHE,share->pack.header_length,1,MYF(MY_WME)) || (! rep_quick && - init_io_cache(&info->rec_cache,info->dfile, + init_io_cache(&info->rec_cache, info->dfile.file, (uint) param->write_buffer_length, WRITE_CACHE,new_header_length,1, MYF(MY_WME | MY_WAIT_IF_FULL) & param->myf_rw))) goto err; sort_info.key_block_end=sort_info.key_block+param->sort_key_blocks; info->opt_flag|=WRITE_CACHE_USED; - info->rec_cache.file=info->dfile; /* for sort_delete_record */ + info->rec_cache.file= info->dfile.file; /* for sort_delete_record */ if (!(sort_param.record=(byte*) my_malloc((uint) share->base.pack_reclength, MYF(0))) || @@ -2633,8 +2638,8 @@ int maria_repair_by_sort(HA_CHECK *param, register MARIA_HA *info, goto err; } if (new_header_length && - maria_filecopy(param, new_file,info->dfile,0L,new_header_length, - "datafile-header")) + maria_filecopy(param, new_file, info->dfile.file, 0L, + new_header_length, "datafile-header")) goto err; if (param->testflag & T_UNPACK) restore_data_file_type(share); @@ -2649,7 +2654,8 @@ int maria_repair_by_sort(HA_CHECK *param, register MARIA_HA *info, Flush key cache for this file if we are calling this outside maria_chk */ - flush_key_blocks(share->key_cache,share->kfile, FLUSH_IGNORE_CHANGED); + flush_pagecache_blocks(share->pagecache, &share->kfile, + FLUSH_IGNORE_CHANGED); /* Clear the pointers to the given rows */ for (i=0 ; i < share->base.keys ; i++) share->state.key_root[i]= HA_OFFSET_ERROR; @@ -2658,7 +2664,8 @@ int maria_repair_by_sort(HA_CHECK *param, register MARIA_HA *info, } else { - if (flush_key_blocks(share->key_cache,share->kfile, FLUSH_FORCE_WRITE)) + if (flush_pagecache_blocks(share->pagecache, &share->kfile, + FLUSH_FORCE_WRITE)) goto err; key_map= ~key_map; /* Create the missing keys */ } @@ -2812,8 +2819,8 @@ int maria_repair_by_sort(HA_CHECK *param, register MARIA_HA *info, sort_param.filepos; /* Only whole records */ share->state.version=(ulong) time((time_t*) 0); - my_close(info->dfile,MYF(0)); - info->dfile=new_file; + my_close(info->dfile.file, MYF(0)); + info->dfile.file= new_file; share->data_file_type=sort_info.new_data_file_type; share->pack.header_length=(ulong) new_header_length; sort_param.fix_datafile=0; @@ -2821,7 +2828,7 @@ int maria_repair_by_sort(HA_CHECK *param, register MARIA_HA *info, else info->state->data_file_length=sort_param.max_pos; - param->read_cache.file=info->dfile; /* re-init read cache */ + param->read_cache.file= info->dfile.file; /* re-init read cache */ reinit_io_cache(¶m->read_cache,READ_CACHE,share->pack.header_length, 1,1); } @@ -2852,7 +2859,7 @@ int maria_repair_by_sort(HA_CHECK *param, register MARIA_HA *info, skr=share->base.reloc*share->base.min_pack_length; #endif if (skr != sort_info.filelength) - if (my_chsize(info->dfile,skr,0,MYF(0))) + if (my_chsize(info->dfile.file, skr, 0, MYF(0))) _ma_check_print_warning(param, "Can't change size of datafile, error: %d", my_errno); @@ -2860,7 +2867,7 @@ int maria_repair_by_sort(HA_CHECK *param, register MARIA_HA *info, if (param->testflag & T_CALC_CHECKSUM) info->state->checksum=param->glob_crc; - if (my_chsize(share->kfile,info->state->key_file_length,0,MYF(0))) + if (my_chsize(share->kfile.file, info->state->key_file_length, 0, MYF(0))) _ma_check_print_warning(param, "Can't change size of indexfile, error: %d", my_errno); @@ -2880,7 +2887,7 @@ int maria_repair_by_sort(HA_CHECK *param, register MARIA_HA *info, memcpy( &share->state.state, info->state, sizeof(*info->state)); err: - got_error|= _ma_flush_blocks(param, share->key_cache, share->kfile); + got_error|= _ma_flush_blocks(param, share->pagecache, &share->kfile); VOID(end_io_cache(&info->rec_cache)); if (!got_error) { @@ -2888,7 +2895,7 @@ err: if (new_file >= 0) { my_close(new_file,MYF(0)); - info->dfile=new_file= -1; + info->dfile.file= new_file= -1; if (maria_change_to_newfile(share->data_file_name,MARIA_NAME_DEXT, DATA_TMP_EXT, (param->testflag & T_BACKUP_DATA ? @@ -2905,8 +2912,8 @@ err: { VOID(my_close(new_file,MYF(0))); VOID(my_delete(param->temp_filename, MYF(MY_WME))); - if (info->dfile == new_file) - info->dfile= -1; + if (info->dfile.file == new_file) + info->dfile.file= -1; } maria_mark_crashed_on_repair(info); } @@ -3010,14 +3017,14 @@ int maria_repair_parallel(HA_CHECK *param, register MARIA_HA *info, /* Quick repair (not touching data file, rebuilding indexes): { - Read cache is (MI_CHECK *param)->read_cache using info->dfile. + Read cache is (MI_CHECK *param)->read_cache using info->dfile.file. } Non-quick repair (rebuilding data file and indexes): { Master thread: - Read cache is (MI_CHECK *param)->read_cache using info->dfile. + Read cache is (MI_CHECK *param)->read_cache using info->dfile.file. Write cache is (MI_INFO *info)->rec_cache using new_file. Slave threads: @@ -3044,11 +3051,11 @@ int maria_repair_parallel(HA_CHECK *param, register MARIA_HA *info, if (!(sort_info.key_block= alloc_key_blocks(param, (uint) param->sort_key_blocks, share->base.max_key_block_length)) || - init_io_cache(¶m->read_cache, info->dfile, + init_io_cache(¶m->read_cache, info->dfile.file, (uint) param->read_buffer_length, READ_CACHE, share->pack.header_length, 1, MYF(MY_WME)) || (!rep_quick && - (init_io_cache(&info->rec_cache, info->dfile, + (init_io_cache(&info->rec_cache, info->dfile.file, (uint) param->write_buffer_length, WRITE_CACHE, new_header_length, 1, MYF(MY_WME | MY_WAIT_IF_FULL) & param->myf_rw) || @@ -3059,7 +3066,7 @@ int maria_repair_parallel(HA_CHECK *param, register MARIA_HA *info, goto err; sort_info.key_block_end=sort_info.key_block+param->sort_key_blocks; info->opt_flag|=WRITE_CACHE_USED; - info->rec_cache.file=info->dfile; /* for sort_delete_record */ + info->rec_cache.file= info->dfile.file; /* for sort_delete_record */ if (!rep_quick) { @@ -3076,7 +3083,7 @@ int maria_repair_parallel(HA_CHECK *param, register MARIA_HA *info, goto err; } if (new_header_length && - maria_filecopy(param, new_file,info->dfile,0L,new_header_length, + maria_filecopy(param, new_file, info->dfile.file,0L,new_header_length, "datafile-header")) goto err; if (param->testflag & T_UNPACK) @@ -3092,7 +3099,8 @@ int maria_repair_parallel(HA_CHECK *param, register MARIA_HA *info, Flush key cache for this file if we are calling this outside maria_chk */ - flush_key_blocks(share->key_cache,share->kfile, FLUSH_IGNORE_CHANGED); + flush_pagecache_blocks(share->pagecache, &share->kfile, + FLUSH_IGNORE_CHANGED); /* Clear the pointers to the given rows */ for (i=0 ; i < share->base.keys ; i++) share->state.key_root[i]= HA_OFFSET_ERROR; @@ -3101,7 +3109,8 @@ int maria_repair_parallel(HA_CHECK *param, register MARIA_HA *info, } else { - if (flush_key_blocks(share->key_cache,share->kfile, FLUSH_FORCE_WRITE)) + if (flush_pagecache_blocks(share->pagecache, &share->kfile, + FLUSH_FORCE_WRITE)) goto err; key_map= ~key_map; /* Create the missing keys */ } @@ -3331,8 +3340,8 @@ int maria_repair_parallel(HA_CHECK *param, register MARIA_HA *info, Exchange the data file descriptor of the table, so that we use the new file from now on. */ - my_close(info->dfile,MYF(0)); - info->dfile=new_file; + my_close(info->dfile.file, MYF(0)); + info->dfile.file= new_file; share->data_file_type=sort_info.new_data_file_type; share->pack.header_length=(ulong) new_header_length; @@ -3360,7 +3369,7 @@ int maria_repair_parallel(HA_CHECK *param, register MARIA_HA *info, skr=share->base.reloc*share->base.min_pack_length; #endif if (skr != sort_info.filelength) - if (my_chsize(info->dfile,skr,0,MYF(0))) + if (my_chsize(info->dfile.file, skr, 0, MYF(0))) _ma_check_print_warning(param, "Can't change size of datafile, error: %d", my_errno); @@ -3368,7 +3377,7 @@ int maria_repair_parallel(HA_CHECK *param, register MARIA_HA *info, if (param->testflag & T_CALC_CHECKSUM) info->state->checksum=param->glob_crc; - if (my_chsize(share->kfile,info->state->key_file_length,0,MYF(0))) + if (my_chsize(share->kfile.file, info->state->key_file_length, 0, MYF(0))) _ma_check_print_warning(param, "Can't change size of indexfile, error: %d", my_errno); @@ -3387,7 +3396,7 @@ int maria_repair_parallel(HA_CHECK *param, register MARIA_HA *info, memcpy(&share->state.state, info->state, sizeof(*info->state)); err: - got_error|= _ma_flush_blocks(param, share->key_cache, share->kfile); + got_error|= _ma_flush_blocks(param, share->pagecache, &share->kfile); /* Destroy the write cache. The master thread did already detach from the share by remove_io_thread() or it was not yet started (if the @@ -3408,7 +3417,7 @@ err: if (new_file >= 0) { my_close(new_file,MYF(0)); - info->dfile=new_file= -1; + info->dfile.file= new_file= -1; if (maria_change_to_newfile(share->data_file_name,MARIA_NAME_DEXT, DATA_TMP_EXT, (param->testflag & T_BACKUP_DATA ? @@ -3425,8 +3434,8 @@ err: { VOID(my_close(new_file,MYF(0))); VOID(my_delete(param->temp_filename, MYF(MY_WME))); - if (info->dfile == new_file) - info->dfile= -1; + if (info->dfile.file == new_file) + info->dfile.file= -1; } maria_mark_crashed_on_repair(info); } @@ -4400,7 +4409,7 @@ static int sort_insert_key(MARIA_SORT_PARAM *sort_param, if (_ma_write_keypage(info, keyinfo, filepos, DFLT_INIT_HITS, anc_buff)) DBUG_RETURN(1); } - else if (my_pwrite(info->s->kfile,anc_buff, + else if (my_pwrite(info->s->kfile.file, anc_buff, (uint) keyinfo->block_length,filepos, param->myf_rw)) DBUG_RETURN(1); DBUG_DUMP("buff",anc_buff,maria_getint(anc_buff)); @@ -4440,8 +4449,8 @@ static int sort_delete_record(MARIA_SORT_PARAM *sort_param) DBUG_RETURN(1); } - old_file=info->dfile; - info->dfile=info->rec_cache.file; + old_file= info->dfile.file; + info->dfile.file= info->rec_cache.file; if (sort_info->current_key) { key= info->lastkey+info->s->base.max_key_length; @@ -4450,7 +4459,7 @@ static int sort_delete_record(MARIA_SORT_PARAM *sort_param) error != HA_ERR_RECORD_DELETED) { _ma_check_print_error(param,"Can't read record to be removed"); - info->dfile=old_file; + info->dfile.file= old_file; DBUG_RETURN(1); } @@ -4463,7 +4472,7 @@ static int sort_delete_record(MARIA_SORT_PARAM *sort_param) _ma_check_print_error(param, "Can't delete key %d from record to be removed", i+1); - info->dfile=old_file; + info->dfile.file= old_file; DBUG_RETURN(1); } } @@ -4471,7 +4480,7 @@ static int sort_delete_record(MARIA_SORT_PARAM *sort_param) param->glob_crc-=(*info->s->calc_checksum)(info, sort_param->record); } error=flush_io_cache(&info->rec_cache) || (*info->s->delete_record)(info); - info->dfile=old_file; /* restore actual value */ + info->dfile.file= old_file; /* restore actual value */ info->state->records--; DBUG_RETURN(error); } /* sort_delete_record */ @@ -4509,7 +4518,7 @@ int _ma_flush_pending_blocks(MARIA_SORT_PARAM *sort_param) DFLT_INIT_HITS, key_block->buff)) DBUG_RETURN(1); } - else if (my_pwrite(info->s->kfile,key_block->buff, + else if (my_pwrite(info->s->kfile.file, key_block->buff, (uint) keyinfo->block_length,filepos, myf_rw)) DBUG_RETURN(1); DBUG_DUMP("buff",key_block->buff,length); @@ -4550,9 +4559,9 @@ int maria_test_if_almost_full(MARIA_HA *info) { if (info->s->options & HA_OPTION_COMPRESS_RECORD) return 0; - return (my_seek(info->s->kfile,0L,MY_SEEK_END,MYF(0))/10*9 > + return (my_seek(info->s->kfile.file, 0L, MY_SEEK_END, MYF(0))/10*9 > (my_off_t) (info->s->base.max_key_file_length) || - my_seek(info->dfile,0L,MY_SEEK_END,MYF(0))/10*9 > + my_seek(info->dfile.file, 0L, MY_SEEK_END, MYF(0)) / 10 * 9 > (my_off_t) info->s->base.max_data_file_length); } @@ -4653,7 +4662,8 @@ int maria_recreate_table(HA_CHECK *param, MARIA_HA **org_info, char *filename) if (share.options & HA_OPTION_COMPRESS_RECORD) share.base.records=max_records=info.state->records; else if (share.base.min_pack_length) - max_records=(ha_rows) (my_seek(info.dfile,0L,MY_SEEK_END,MYF(0)) / + max_records=(ha_rows) (my_seek(info.dfile.file, 0L, MY_SEEK_END, + MYF(0)) / (ulong) share.base.min_pack_length); else max_records=0; @@ -4661,7 +4671,7 @@ int maria_recreate_table(HA_CHECK *param, MARIA_HA **org_info, char *filename) (param->testflag & T_UNPACK); share.options&= ~HA_OPTION_TEMP_COMPRESS_RECORD; - file_length=(ulonglong) my_seek(info.dfile,0L,MY_SEEK_END,MYF(0)); + file_length=(ulonglong) my_seek(info.dfile.file, 0L, MY_SEEK_END, MYF(0)); tmp_length= file_length+file_length/10; set_if_bigger(file_length,param->max_data_file_length); set_if_bigger(file_length,tmp_length); @@ -4804,7 +4814,7 @@ int maria_update_state_info(HA_CHECK *param, MARIA_HA *info,uint update) */ if (info->lock_type == F_WRLCK) share->state.state= *info->state; - if (_ma_state_info_write(share->kfile,&share->state,1+2)) + if (_ma_state_info_write(share->kfile.file, &share->state, 1 + 2)) goto err; share->changed=0; } diff --git a/storage/maria/ma_close.c b/storage/maria/ma_close.c index 2a874079961..05259218e6d 100644 --- a/storage/maria/ma_close.c +++ b/storage/maria/ma_close.c @@ -36,7 +36,7 @@ int maria_close(register MARIA_HA *info) if (info->lock_type == F_EXTRA_LCK) info->lock_type=F_UNLCK; /* HA_EXTRA_NO_USER_CHANGE */ - if (share->reopen == 1 && share->kfile >= 0) + if (share->reopen == 1 && share->kfile.file >= 0) _ma_decrement_open_count(info); if (info->lock_type != F_UNLCK) @@ -73,16 +73,17 @@ int maria_close(register MARIA_HA *info) (share->end)(info); if (info->s->data_file_type == BLOCK_RECORD) - info->dfile= -1; /* Closed in ma_end_once_block_row */ + info->dfile.file= -1; /* Closed in ma_end_once_block_row */ if (flag) { - if (share->kfile >= 0) + if (share->kfile.file >= 0) { if ((*share->once_end)(share)) error= my_errno; - if (flush_key_blocks(share->key_cache, share->kfile, - share->temporary ? FLUSH_IGNORE_CHANGED : - FLUSH_RELEASE)) + if (flush_pagecache_blocks(share->pagecache, &share->kfile, + (share->temporary ? + FLUSH_IGNORE_CHANGED : + FLUSH_RELEASE))) error= my_errno; /* @@ -92,8 +93,8 @@ int maria_close(register MARIA_HA *info) may be using the file at this point */ if (share->mode != O_RDONLY && maria_is_crashed(info)) - _ma_state_info_write(share->kfile, &share->state, 1); - if (my_close(share->kfile, MYF(0))) + _ma_state_info_write(share->kfile.file, &share->state, 1); + if (my_close(share->kfile.file, MYF(0))) error= my_errno; } #ifdef HAVE_MMAP @@ -120,7 +121,7 @@ int maria_close(register MARIA_HA *info) my_free((gptr)info->ftparser_param, MYF(0)); info->ftparser_param= 0; } - if (info->dfile >= 0 && my_close(info->dfile,MYF(0))) + if (info->dfile.file >= 0 && my_close(info->dfile.file, MYF(0))) error = my_errno; my_free((gptr) info,MYF(0)); diff --git a/storage/maria/ma_delete_all.c b/storage/maria/ma_delete_all.c index 616147c1067..d12e58c49c9 100644 --- a/storage/maria/ma_delete_all.c +++ b/storage/maria/ma_delete_all.c @@ -52,7 +52,8 @@ int maria_delete_all_rows(MARIA_HA *info) If we are using delayed keys or if the user has done changes to the tables since it was locked then there may be key blocks in the key cache */ - flush_key_blocks(share->key_cache, share->kfile, FLUSH_IGNORE_CHANGED); + flush_pagecache_blocks(share->pagecache, &share->kfile, + FLUSH_IGNORE_CHANGED); /* RECOVERYTODO Log the two chsize and header modifications and force the log. So that if crash between the two chsize, we finish the work at @@ -62,8 +63,8 @@ int maria_delete_all_rows(MARIA_HA *info) should be applied only if t1 exists and its ZeroDirtyPagesLSN is smaller than the records'. See more comments below. */ - if (my_chsize(info->dfile, 0, 0, MYF(MY_WME)) || - my_chsize(share->kfile, share->base.keystart, 0, MYF(MY_WME)) ) + if (my_chsize(info->dfile.file, 0, 0, MYF(MY_WME)) || + my_chsize(share->kfile.file, share->base.keystart, 0, MYF(MY_WME)) ) goto err; /* RECOVERYTODO Consider updating ZeroDirtyPagesLSN here. It is diff --git a/storage/maria/ma_dynrec.c b/storage/maria/ma_dynrec.c index c3d0bf1fb0a..aeedd4519c1 100644 --- a/storage/maria/ma_dynrec.c +++ b/storage/maria/ma_dynrec.c @@ -74,7 +74,7 @@ my_bool _ma_dynmap_file(MARIA_HA *info, my_off_t size) info->s->mode==O_RDONLY ? PROT_READ : PROT_READ | PROT_WRITE, MAP_SHARED | MAP_NORESERVE, - info->dfile, 0L); + info->dfile.file, 0L); if (info->s->file_map == (byte*) MAP_FAILED) { info->s->file_map= NULL; @@ -128,7 +128,7 @@ void _ma_remap_file(MARIA_HA *info, my_off_t size) uint _ma_mmap_pread(MARIA_HA *info, byte *Buffer, uint Count, my_off_t offset, myf MyFlags) { - DBUG_PRINT("info", ("maria_read with mmap %d\n", info->dfile)); + DBUG_PRINT("info", ("maria_read with mmap %d\n", info->dfile.file)); if (info->s->concurrent_insert) rw_rdlock(&info->s->mmap_lock); @@ -150,7 +150,7 @@ uint _ma_mmap_pread(MARIA_HA *info, byte *Buffer, { if (info->s->concurrent_insert) rw_unlock(&info->s->mmap_lock); - return my_pread(info->dfile, Buffer, Count, offset, MyFlags); + return my_pread(info->dfile.file, Buffer, Count, offset, MyFlags); } } @@ -160,7 +160,7 @@ uint _ma_mmap_pread(MARIA_HA *info, byte *Buffer, uint _ma_nommap_pread(MARIA_HA *info, byte *Buffer, uint Count, my_off_t offset, myf MyFlags) { - return my_pread(info->dfile, Buffer, Count, offset, MyFlags); + return my_pread(info->dfile.file, Buffer, Count, offset, MyFlags); } @@ -183,7 +183,7 @@ uint _ma_nommap_pread(MARIA_HA *info, byte *Buffer, uint _ma_mmap_pwrite(MARIA_HA *info, byte *Buffer, uint Count, my_off_t offset, myf MyFlags) { - DBUG_PRINT("info", ("maria_write with mmap %d\n", info->dfile)); + DBUG_PRINT("info", ("maria_write with mmap %d\n", info->dfile.file)); if (info->s->concurrent_insert) rw_rdlock(&info->s->mmap_lock); @@ -206,7 +206,7 @@ uint _ma_mmap_pwrite(MARIA_HA *info, byte *Buffer, info->s->nonmmaped_inserts++; if (info->s->concurrent_insert) rw_unlock(&info->s->mmap_lock); - return my_pwrite(info->dfile, Buffer, Count, offset, MyFlags); + return my_pwrite(info->dfile.file, Buffer, Count, offset, MyFlags); } } @@ -217,7 +217,7 @@ uint _ma_mmap_pwrite(MARIA_HA *info, byte *Buffer, uint _ma_nommap_pwrite(MARIA_HA *info, byte *Buffer, uint Count, my_off_t offset, myf MyFlags) { - return my_pwrite(info->dfile, Buffer, Count, offset, MyFlags); + return my_pwrite(info->dfile.file, Buffer, Count, offset, MyFlags); } @@ -354,7 +354,8 @@ static int _ma_find_writepos(MARIA_HA *info, *filepos=info->s->state.dellink; block_info.second_read=0; info->rec_cache.seek_not_done=1; - if (!(_ma_get_block_info(&block_info,info->dfile,info->s->state.dellink) & + if (!(_ma_get_block_info(&block_info, info->dfile.file, + info->s->state.dellink) & BLOCK_DELETED)) { DBUG_PRINT("error",("Delete link crashed")); @@ -413,7 +414,7 @@ static bool unlink_deleted_block(MARIA_HA *info, MARIA_BLOCK_INFO *block_info) MARIA_BLOCK_INFO tmp; tmp.second_read=0; /* Unlink block from the previous block */ - if (!(_ma_get_block_info(&tmp,info->dfile,block_info->prev_filepos) + if (!(_ma_get_block_info(&tmp, info->dfile.file, block_info->prev_filepos) & BLOCK_DELETED)) DBUG_RETURN(1); /* Something is wrong */ mi_sizestore(tmp.header+4,block_info->next_filepos); @@ -423,7 +424,8 @@ static bool unlink_deleted_block(MARIA_HA *info, MARIA_BLOCK_INFO *block_info) /* Unlink block from next block */ if (block_info->next_filepos != HA_OFFSET_ERROR) { - if (!(_ma_get_block_info(&tmp,info->dfile,block_info->next_filepos) + if (!(_ma_get_block_info(&tmp, info->dfile.file, + block_info->next_filepos) & BLOCK_DELETED)) DBUG_RETURN(1); /* Something is wrong */ mi_sizestore(tmp.header+12,block_info->prev_filepos); @@ -474,7 +476,7 @@ static my_bool update_backward_delete_link(MARIA_HA *info, if (delete_block != HA_OFFSET_ERROR) { block_info.second_read=0; - if (_ma_get_block_info(&block_info,info->dfile,delete_block) + if (_ma_get_block_info(&block_info, info->dfile.file, delete_block) & BLOCK_DELETED) { char buff[8]; @@ -510,7 +512,7 @@ static my_bool delete_dynamic_record(MARIA_HA *info, MARIA_RECORD_POS filepos, do { /* Remove block at 'filepos' */ - if ((b_type= _ma_get_block_info(&block_info,info->dfile,filepos)) + if ((b_type= _ma_get_block_info(&block_info, info->dfile.file, filepos)) & (BLOCK_DELETED | BLOCK_ERROR | BLOCK_SYNC_ERROR | BLOCK_FATAL_ERROR) || (length=(uint) (block_info.filepos-filepos) +block_info.block_len) < @@ -522,7 +524,7 @@ static my_bool delete_dynamic_record(MARIA_HA *info, MARIA_RECORD_POS filepos, /* Check if next block is a delete block */ del_block.second_read=0; remove_next_block=0; - if (_ma_get_block_info(&del_block,info->dfile,filepos+length) & + if (_ma_get_block_info(&del_block, info->dfile.file, filepos + length) & BLOCK_DELETED && del_block.block_len+length < MARIA_DYN_MAX_BLOCK_LENGTH) { @@ -682,7 +684,7 @@ int _ma_write_part_record(MARIA_HA *info, if (next_block < info->state->data_file_length && info->s->state.dellink != HA_OFFSET_ERROR) { - if ((_ma_get_block_info(&del_block,info->dfile,next_block) + if ((_ma_get_block_info(&del_block, info->dfile.file, next_block) & BLOCK_DELETED) && res_length + del_block.block_len < MARIA_DYN_MAX_BLOCK_LENGTH) { @@ -763,7 +765,7 @@ static my_bool update_dynamic_record(MARIA_HA *info, MARIA_RECORD_POS filepos, if (filepos != info->s->state.dellink) { block_info.next_filepos= HA_OFFSET_ERROR; - if ((error= _ma_get_block_info(&block_info,info->dfile,filepos)) + if ((error= _ma_get_block_info(&block_info, info->dfile.file, filepos)) & (BLOCK_DELETED | BLOCK_ERROR | BLOCK_SYNC_ERROR | BLOCK_FATAL_ERROR)) { @@ -804,7 +806,7 @@ static my_bool update_dynamic_record(MARIA_HA *info, MARIA_RECORD_POS filepos, MARIA_BLOCK_INFO del_block; del_block.second_read=0; - if (_ma_get_block_info(&del_block,info->dfile, + if (_ma_get_block_info(&del_block, info->dfile.file, block_info.filepos + block_info.block_len) & BLOCK_DELETED) { @@ -1370,7 +1372,7 @@ int _ma_read_dynamic_record(MARIA_HA *info, byte *buf, { LINT_INIT(to); LINT_INIT(left_length); - file=info->dfile; + file= info->dfile.file; block_of_record= 0; /* First block of record is numbered as zero. */ block_info.second_read= 0; do @@ -1534,7 +1536,7 @@ my_bool _ma_cmp_dynamic_record(register MARIA_HA *info, block_info.next_filepos=filepos; while (reclength > 0) { - if ((b_type= _ma_get_block_info(&block_info,info->dfile, + if ((b_type= _ma_get_block_info(&block_info, info->dfile.file, block_info.next_filepos)) & (BLOCK_DELETED | BLOCK_ERROR | BLOCK_SYNC_ERROR | BLOCK_FATAL_ERROR)) @@ -1561,7 +1563,7 @@ my_bool _ma_cmp_dynamic_record(register MARIA_HA *info, if (!reclength && info->s->calc_checksum) cmp_length--; /* 'record' may not contain checksum */ - if (_ma_cmp_buffer(info->dfile,record,block_info.filepos, + if (_ma_cmp_buffer(info->dfile.file, record, block_info.filepos, cmp_length)) { my_errno=HA_ERR_RECORD_CHANGED; @@ -1680,7 +1682,7 @@ int _ma_read_rnd_dynamic_record(MARIA_HA *info, { /* Check if changed */ info_read=1; info->rec_cache.seek_not_done=1; - if (_ma_state_info_read_dsk(share->kfile,&share->state,1)) + if (_ma_state_info_read_dsk(share->kfile.file, &share->state, 1)) goto panic; } if (filepos >= info->state->data_file_length) @@ -1705,7 +1707,7 @@ int _ma_read_rnd_dynamic_record(MARIA_HA *info, flush_io_cache(&info->rec_cache)) DBUG_RETURN(my_errno); info->rec_cache.seek_not_done=1; - b_type= _ma_get_block_info(&block_info,info->dfile,filepos); + b_type= _ma_get_block_info(&block_info, info->dfile.file, filepos); } if (b_type & (BLOCK_DELETED | BLOCK_ERROR | BLOCK_SYNC_ERROR | @@ -1779,8 +1781,9 @@ int _ma_read_rnd_dynamic_record(MARIA_HA *info, block_info.filepos + block_info.data_len && flush_io_cache(&info->rec_cache)) goto err; - /* VOID(my_seek(info->dfile,filepos,MY_SEEK_SET,MYF(0))); */ - if (my_read(info->dfile,(byte*) to,block_info.data_len,MYF(MY_NABP))) + /* VOID(my_seek(info->dfile.file, filepos, MY_SEEK_SET, MYF(0))); */ + if (my_read(info->dfile.file, (byte*)to, block_info.data_len, + MYF(MY_NABP))) { if (my_errno == -1) my_errno= HA_ERR_WRONG_IN_RECORD; /* Unexpected end of file */ diff --git a/storage/maria/ma_extra.c b/storage/maria/ma_extra.c index 90e79362442..3c35d7d20fc 100644 --- a/storage/maria/ma_extra.c +++ b/storage/maria/ma_extra.c @@ -101,7 +101,7 @@ int maria_extra(MARIA_HA *info, enum ha_extra_function function, { cache_size= (extra_arg ? *(ulong*) extra_arg : my_default_record_cache_size); - if (!(init_io_cache(&info->rec_cache,info->dfile, + if (!(init_io_cache(&info->rec_cache, info->dfile.file, (uint) min(info->state->data_file_length+1, cache_size), READ_CACHE,0L,(pbool) (info->lock_type != F_UNLCK), @@ -137,7 +137,7 @@ int maria_extra(MARIA_HA *info, enum ha_extra_function function, if (!(info->opt_flag & (READ_CACHE_USED | WRITE_CACHE_USED | OPT_NO_ROWS)) && !share->state.header.uniques) - if (!(init_io_cache(&info->rec_cache,info->dfile, cache_size, + if (!(init_io_cache(&info->rec_cache, info->dfile.file, cache_size, WRITE_CACHE,info->state->data_file_length, (pbool) (info->lock_type != F_UNLCK), MYF(share->write_flag & MY_WAIT_IF_FULL)))) @@ -250,7 +250,7 @@ int maria_extra(MARIA_HA *info, enum ha_extra_function function, } } share->state.state= *info->state; - error=_ma_state_info_write(share->kfile,&share->state,1 | 2); + error=_ma_state_info_write(share->kfile.file, &share->state, (1 | 2)); } break; case HA_EXTRA_FORCE_REOPEN: @@ -264,8 +264,8 @@ int maria_extra(MARIA_HA *info, enum ha_extra_function function, #ifdef __WIN__ /* Close the isam and data files as Win32 can't drop an open table */ pthread_mutex_lock(&share->intern_lock); - if (flush_key_blocks(share->key_cache, share->kfile, - (function == HA_EXTRA_FORCE_REOPEN ? + if (flush_pagecache_blocks(share->pagecache, &share->kfile, + (function == HA_EXTRA_FORCE_REOPEN ? FLUSH_RELEASE : FLUSH_IGNORE_CHANGED))) { error=my_errno; @@ -285,9 +285,9 @@ int maria_extra(MARIA_HA *info, enum ha_extra_function function, error=my_errno; info->lock_type = F_UNLCK; } - if (share->kfile >= 0) + if (share->kfile.file >= 0) _ma_decrement_open_count(info); - if (share->kfile >= 0 && my_close(share->kfile,MYF(0))) + if (share->kfile.file >= 0 && my_close(share->kfile,MYF(0))) error=my_errno; { LIST *list_element ; @@ -298,20 +298,21 @@ int maria_extra(MARIA_HA *info, enum ha_extra_function function, MARIA_HA *tmpinfo=(MARIA_HA*) list_element->data; if (tmpinfo->s == info->s) { - if (tmpinfo->dfile >= 0 && my_close(tmpinfo->dfile,MYF(0))) + if (tmpinfo->dfile.file >= 0 && + my_close(tmpinfo->dfile.file, MYF(0))) error = my_errno; - tmpinfo->dfile= -1; + tmpinfo->dfile.file= -1; } } } - share->kfile= -1; /* Files aren't open anymore */ + share->kfile.file= -1; /* Files aren't open anymore */ pthread_mutex_unlock(&share->intern_lock); #endif pthread_mutex_unlock(&THR_LOCK_maria); break; case HA_EXTRA_FLUSH: if (!share->temporary) - flush_key_blocks(share->key_cache, share->kfile, FLUSH_KEEP); + flush_pagecache_blocks(share->pagecache, &share->kfile, FLUSH_KEEP); #ifdef HAVE_PWRITE _ma_decrement_open_count(info); #endif @@ -453,6 +454,6 @@ int maria_reset(MARIA_HA *info) int _ma_sync_table_files(const MARIA_HA *info) { - return (my_sync(info->dfile, MYF(0)) || - my_sync(info->s->kfile, MYF(0))); + return (my_sync(info->dfile.file, MYF(0)) || + my_sync(info->s->kfile.file, MYF(0))); } diff --git a/storage/maria/ma_info.c b/storage/maria/ma_info.c index 397cd2465d4..e76692960c6 100644 --- a/storage/maria/ma_info.c +++ b/storage/maria/ma_info.c @@ -72,7 +72,7 @@ int maria_status(MARIA_HA *info, register MARIA_INFO *x, uint flag) x->reclength = share->base.reclength; x->max_data_file_length=share->base.max_data_file_length; x->max_index_file_length=info->s->base.max_key_file_length; - x->filenr = info->dfile; + x->filenr = info->dfile.file; x->options = share->options; x->create_time=share->state.create_time; x->reflength= maria_get_pointer_length(share->base.max_data_file_length, @@ -86,7 +86,7 @@ int maria_status(MARIA_HA *info, register MARIA_INFO *x, uint flag) x->data_file_name = share->data_file_name; x->index_file_name = share->index_file_name; } - if ((flag & HA_STATUS_TIME) && !my_fstat(info->dfile,&state,MYF(0))) + if ((flag & HA_STATUS_TIME) && !my_fstat(info->dfile.file, &state, MYF(0))) x->update_time=state.st_mtime; else x->update_time=0; diff --git a/storage/maria/ma_keycache.c b/storage/maria/ma_keycache.c index 64725baae86..a9df52eadf9 100644 --- a/storage/maria/ma_keycache.c +++ b/storage/maria/ma_keycache.c @@ -24,10 +24,10 @@ Assign pages of the index file for a table to a key cache SYNOPSIS - maria_assign_to_key_cache() + maria_assign_to_pagecache() info open table key_map map of indexes to assign to the key cache - key_cache_ptr pointer to the key cache handle + pagecache_ptr pointer to the key cache handle assign_lock Mutex to lock during assignment PREREQUESTS @@ -47,22 +47,22 @@ # Error code */ -int maria_assign_to_key_cache(MARIA_HA *info, - ulonglong key_map __attribute__((unused)), - KEY_CACHE *key_cache) +int maria_assign_to_pagecache(MARIA_HA *info, + ulonglong key_map __attribute__((unused)), + PAGECACHE *pagecache) { int error= 0; MARIA_SHARE* share= info->s; - DBUG_ENTER("maria_assign_to_key_cache"); + DBUG_ENTER("maria_assign_to_pagecache"); DBUG_PRINT("enter", - ("old_key_cache_handle: 0x%lx new_key_cache_handle: 0x%lx", - (long) share->key_cache, (long) key_cache)); + ("old_pagecache_handle: 0x%lx new_pagecache_handle: 0x%lx", + (long) share->pagecache, (long) pagecache)); /* Skip operation if we didn't change key cache. This can happen if we call this for all open instances of the same table */ - if (share->key_cache == key_cache) + if (share->pagecache == pagecache) DBUG_RETURN(0); /* @@ -77,7 +77,7 @@ int maria_assign_to_key_cache(MARIA_HA *info, in the old key cache. */ - if (flush_key_blocks(share->key_cache, share->kfile, FLUSH_RELEASE)) + if (flush_pagecache_blocks(share->pagecache, &share->kfile, FLUSH_RELEASE)) { error= my_errno; maria_print_error(info->s, HA_ERR_CRASHED); @@ -92,10 +92,10 @@ int maria_assign_to_key_cache(MARIA_HA *info, (This can never fail as there is never any not written data in the new key cache) */ - (void) flush_key_blocks(key_cache, share->kfile, FLUSH_RELEASE); + (void) flush_pagecache_blocks(pagecache, &share->kfile, FLUSH_RELEASE); /* - ensure that setting the key cache and changing the multi_key_cache + ensure that setting the key cache and changing the multi_pagecache is done atomicly */ pthread_mutex_lock(&share->intern_lock); @@ -103,11 +103,11 @@ int maria_assign_to_key_cache(MARIA_HA *info, Tell all threads to use the new key cache This should be seen at the lastes for the next call to an maria function. */ - share->key_cache= key_cache; + share->pagecache= pagecache; /* store the key cache in the global hash structure for future opens */ - if (multi_key_cache_set(share->unique_file_name, share->unique_name_length, - share->key_cache)) + if (multi_pagecache_set(share->unique_file_name, share->unique_name_length, + share->pagecache)) error= my_errno; pthread_mutex_unlock(&share->intern_lock); DBUG_RETURN(error); @@ -118,16 +118,16 @@ int maria_assign_to_key_cache(MARIA_HA *info, Change all MARIA entries that uses one key cache to another key cache SYNOPSIS - maria_change_key_cache() - old_key_cache Old key cache - new_key_cache New key cache + maria_change_pagecache() + old_pagecache Old key cache + new_pagecache New key cache NOTES This is used when we delete one key cache. To handle the case where some other threads tries to open an MARIA table associated with the to-be-deleted key cache while this operation - is running, we have to call 'multi_key_cache_change()' from this + is running, we have to call 'multi_pagecache_change()' from this function while we have a lock on the MARIA table list structure. This is safe as long as it's only MARIA that is using this specific @@ -135,11 +135,11 @@ int maria_assign_to_key_cache(MARIA_HA *info, */ -void maria_change_key_cache(KEY_CACHE *old_key_cache, - KEY_CACHE *new_key_cache) +void maria_change_pagecache(PAGECACHE *old_pagecache, + PAGECACHE *new_pagecache) { LIST *pos; - DBUG_ENTER("maria_change_key_cache"); + DBUG_ENTER("maria_change_pagecache"); /* Lock list to ensure that no one can close the table while we manipulate it @@ -149,8 +149,8 @@ void maria_change_key_cache(KEY_CACHE *old_key_cache, { MARIA_HA *info= (MARIA_HA*) pos->data; MARIA_SHARE *share= info->s; - if (share->key_cache == old_key_cache) - maria_assign_to_key_cache(info, (ulonglong) ~0, new_key_cache); + if (share->pagecache == old_pagecache) + maria_assign_to_pagecache(info, (ulonglong) ~0, new_pagecache); } /* @@ -158,7 +158,7 @@ void maria_change_key_cache(KEY_CACHE *old_key_cache, MARIA list structure to ensure that another thread is not trying to open a new table that will be associted with the old key cache */ - multi_key_cache_change(old_key_cache, new_key_cache); + multi_pagecache_change(old_pagecache, new_pagecache); pthread_mutex_unlock(&THR_LOCK_maria); DBUG_VOID_RETURN; } diff --git a/storage/maria/ma_locking.c b/storage/maria/ma_locking.c index 6a0bbe82dcb..d59eca7d36b 100644 --- a/storage/maria/ma_locking.c +++ b/storage/maria/ma_locking.c @@ -50,7 +50,7 @@ int maria_lock_database(MARIA_HA *info, int lock_type) error=0; pthread_mutex_lock(&share->intern_lock); - if (share->kfile >= 0) /* May only be false on windows */ + if (share->kfile.file >= 0) /* May only be false on windows */ { switch (lock_type) { case F_UNLCK: @@ -62,9 +62,9 @@ int maria_lock_database(MARIA_HA *info, int lock_type) --share->tot_locks; if (info->lock_type == F_WRLCK && !share->w_locks) { - if (!share->delay_key_write && flush_key_blocks(share->key_cache, - share->kfile, - FLUSH_KEEP)) + if (!share->delay_key_write && + flush_pagecache_blocks(share->pagecache, &share->kfile, + FLUSH_KEEP)) { error= my_errno; maria_print_error(info->s, HA_ERR_CRASHED); @@ -72,7 +72,7 @@ int maria_lock_database(MARIA_HA *info, int lock_type) maria_mark_crashed(info); } if (share->data_file_type == BLOCK_RECORD && - flush_key_blocks(share->key_cache, info->dfile, FLUSH_KEEP)) + flush_pagecache_blocks(share->pagecache, &info->dfile, FLUSH_KEEP)) { error= my_errno; maria_print_error(info->s, HA_ERR_CRASHED); @@ -111,7 +111,7 @@ int maria_lock_database(MARIA_HA *info, int lock_type) share->state.process= share->last_process=share->this_process; share->state.unique= info->last_unique= info->this_unique; share->state.update_count= info->last_loop= ++info->this_loop; - if (_ma_state_info_write(share->kfile, &share->state, 1)) + if (_ma_state_info_write(share->kfile.file, &share->state, 1)) error=my_errno; share->changed=0; if (maria_flush) @@ -147,7 +147,7 @@ int maria_lock_database(MARIA_HA *info, int lock_type) } if (!share->r_locks && !share->w_locks) { - if (_ma_state_info_read_dsk(share->kfile, &share->state, 1)) + if (_ma_state_info_read_dsk(share->kfile.file, &share->state, 1)) { error=my_errno; break; @@ -175,7 +175,7 @@ int maria_lock_database(MARIA_HA *info, int lock_type) { if (!share->r_locks) { - if (_ma_state_info_read_dsk(share->kfile, &share->state, 1)) + if (_ma_state_info_read_dsk(share->kfile.file, &share->state, 1)) { error=my_errno; break; @@ -203,7 +203,7 @@ int maria_lock_database(MARIA_HA *info, int lock_type) a crash on windows if the table is renamed and later on referenced by the merge table. */ - if( info->owned_by_merge && (info->s)->kfile < 0 ) + if( info->owned_by_merge && (info->s)->kfile.file < 0 ) { error = HA_ERR_NO_SUCH_TABLE; } @@ -348,7 +348,7 @@ int _ma_readinfo(register MARIA_HA *info, int lock_type, int check_keybuffer) MARIA_SHARE *share=info->s; if (!share->tot_locks) { - if (_ma_state_info_read_dsk(share->kfile, &share->state, 1)) + if (_ma_state_info_read_dsk(share->kfile.file, &share->state, 1)) { int error=my_errno ? my_errno : -1; my_errno=error; @@ -390,13 +390,13 @@ int _ma_writeinfo(register MARIA_HA *info, uint operation) share->state.process= share->last_process= share->this_process; share->state.unique= info->last_unique= info->this_unique; share->state.update_count= info->last_loop= ++info->this_loop; - if ((error= _ma_state_info_write(share->kfile, &share->state, 1))) + if ((error= _ma_state_info_write(share->kfile.file, &share->state, 1))) olderror=my_errno; #ifdef __WIN__ if (maria_flush) { - _commit(share->kfile); - _commit(info->dfile); + _commit(share->kfile.file); + _commit(info->dfile.file); } #endif my_errno=olderror; @@ -420,7 +420,8 @@ int _ma_test_if_changed(register MARIA_HA *info) { /* Keyfile has changed */ DBUG_PRINT("info",("index file changed")); if (share->state.process != share->this_process) - VOID(flush_key_blocks(share->key_cache, share->kfile, FLUSH_RELEASE)); + VOID(flush_pagecache_blocks(share->pagecache, &share->kfile, + FLUSH_RELEASE)); share->last_process=share->state.process; info->last_unique= share->state.unique; info->last_loop= share->state.update_count; @@ -472,7 +473,7 @@ int _ma_mark_file_changed(MARIA_HA *info) { mi_int2store(buff,share->state.open_count); buff[2]=1; /* Mark that it's changed */ - DBUG_RETURN(my_pwrite(share->kfile,buff,sizeof(buff), + DBUG_RETURN(my_pwrite(share->kfile.file, buff, sizeof(buff), sizeof(share->state.header), MYF(MY_NABP))); } @@ -501,9 +502,9 @@ int _ma_decrement_open_count(MARIA_HA *info) { share->state.open_count--; mi_int2store(buff,share->state.open_count); - write_error=my_pwrite(share->kfile,buff,sizeof(buff), - sizeof(share->state.header), - MYF(MY_NABP)); + write_error= my_pwrite(share->kfile.file, buff, sizeof(buff), + sizeof(share->state.header), + MYF(MY_NABP)); } if (!lock_error) lock_error=maria_lock_database(info,old_lock); diff --git a/storage/maria/ma_loghandler.c b/storage/maria/ma_loghandler.c index 1dc4e9171e3..5db7ce8b163 100644 --- a/storage/maria/ma_loghandler.c +++ b/storage/maria/ma_loghandler.c @@ -1367,6 +1367,7 @@ static my_bool translog_buffer_flush(struct st_translog_buffer *buffer) { PAGECACHE_FILE file; file.file= buffer->file; + DBUG_ASSERT(log_descriptor.pagecache->block_size == TRANSLOG_PAGE_SIZE); if (pagecache_write(log_descriptor.pagecache, &file, (LSN_OFFSET(buffer->offset) + i) / TRANSLOG_PAGE_SIZE, diff --git a/storage/maria/ma_loghandler.h b/storage/maria/ma_loghandler.h index 3646afcf52a..48101814063 100644 --- a/storage/maria/ma_loghandler.h +++ b/storage/maria/ma_loghandler.h @@ -146,37 +146,40 @@ struct st_translog_reader_data }; -my_bool translog_init(const char *directory, uint32 log_file_max_size, - uint32 server_version, uint32 server_id, - PAGECACHE *pagecache, uint flags); - -my_bool translog_write_record(LSN *lsn, - enum translog_record_type type, - SHORT_TRANSACTION_ID short_trid, - void *tcb, - translog_size_t part1_length, - byte *part1_buff, ...); - -void translog_destroy(); - -translog_size_t translog_read_record_header(LSN lsn, - TRANSLOG_HEADER_BUFFER *buff); - -void translog_free_record_header(TRANSLOG_HEADER_BUFFER *buff); - -translog_size_t translog_read_record(LSN lsn, - translog_size_t offset, - translog_size_t length, - byte *buffer, - struct st_translog_reader_data *data); - -my_bool translog_flush(LSN lsn); - -my_bool translog_init_scanner(LSN lsn, - my_bool fixed_horizon, - struct st_translog_scanner_data *scanner); - -translog_size_t translog_read_next_record_header(TRANSLOG_SCANNER_DATA - *scanner, - TRANSLOG_HEADER_BUFFER *buff); +extern my_bool translog_init(const char *directory, uint32 log_file_max_size, + uint32 server_version, uint32 server_id, + PAGECACHE *pagecache, uint flags); + +extern my_bool translog_write_record(LSN *lsn, + enum translog_record_type type, + SHORT_TRANSACTION_ID short_trid, + void *tcb, + translog_size_t part1_length, + byte *part1_buff, ...); + +extern void translog_destroy(); + +extern translog_size_t translog_read_record_header(LSN lsn, + TRANSLOG_HEADER_BUFFER + *buff); + +extern void translog_free_record_header(TRANSLOG_HEADER_BUFFER *buff); + +extern translog_size_t translog_read_record(LSN lsn, + translog_size_t offset, + translog_size_t length, + byte *buffer, + struct st_translog_reader_data + *data); + +extern my_bool translog_flush(LSN lsn); + +extern my_bool translog_init_scanner(LSN lsn, + my_bool fixed_horizon, + struct st_translog_scanner_data *scanner); + +extern translog_size_t translog_read_next_record_header(TRANSLOG_SCANNER_DATA + *scanner, + TRANSLOG_HEADER_BUFFER + *buff); diff --git a/storage/maria/ma_open.c b/storage/maria/ma_open.c index 19113196f15..96b2971aab6 100644 --- a/storage/maria/ma_open.c +++ b/storage/maria/ma_open.c @@ -110,8 +110,8 @@ MARIA_HA *maria_open(const char *name, int mode, uint open_flags) bzero((gptr) &share_buff,sizeof(share_buff)); share_buff.state.rec_per_key_part=rec_per_key_part; share_buff.state.key_root=key_root; - share_buff.key_cache= multi_key_cache_search(name_buff, strlen(name_buff), - maria_key_cache); + share_buff.pagecache= multi_pagecache_search(name_buff, strlen(name_buff), + maria_pagecache); DBUG_EXECUTE_IF("maria_pretend_crashed_table_on_open", if (strstr(name, "/t1")) @@ -449,7 +449,7 @@ MARIA_HA *maria_open(const char *name, int mode, uint open_flags) goto err; errpos= 5; - share->kfile=kfile; + share->kfile.file= kfile; share->this_process=(ulong) getpid(); share->last_process= share->state.process; share->base.key_parts=key_parts; @@ -463,7 +463,7 @@ MARIA_HA *maria_open(const char *name, int mode, uint open_flags) share->block_size= share->base.block_size; my_afree((gptr) disk_cache); _ma_setup_functions(share); - if ((*share->once_init)(share, info.dfile)) + if ((*share->once_init)(share, info.dfile.file)) goto err; share->is_log_table= FALSE; #ifdef THREAD @@ -505,8 +505,8 @@ MARIA_HA *maria_open(const char *name, int mode, uint open_flags) goto err; } if (share->data_file_type == BLOCK_RECORD) - info.dfile= share->bitmap.file; - else if (_ma_open_datafile(&info, share, old_info->dfile)) + info.dfile.file= share->bitmap.file; + else if (_ma_open_datafile(&info, share, old_info->dfile.file)) goto err; errpos= 5; have_rtree= old_info->maria_rtree_recursion_state != NULL; @@ -537,7 +537,7 @@ MARIA_HA *maria_open(const char *name, int mode, uint open_flags) info.cur_row.lastpos= HA_OFFSET_ERROR; info.update= (short) (HA_STATE_NEXT_FOUND+HA_STATE_PREV_FOUND); info.opt_flag=READ_CHECK_USED; - info.this_unique= (ulong) info.dfile; /* Uniq number in process */ + info.this_unique= (ulong) info.dfile.file; /* Uniq number in process */ if (share->data_file_type == COMPRESSED_RECORD) info.this_unique= share->state.unique; info.this_loop=0; /* Update counter */ @@ -614,7 +614,7 @@ err: /* fall through */ case 5: if (share->data_file_type != BLOCK_RECORD) - VOID(my_close(info.dfile,MYF(0))); + VOID(my_close(info.dfile.file, MYF(0))); if (old_info) break; /* Don't remove open table */ (*share->once_end)(share); @@ -1187,15 +1187,16 @@ char *_ma_recinfo_read(char *ptr, MARIA_COLUMNDEF *recinfo) int _ma_open_datafile(MARIA_HA *info, MARIA_SHARE *share, File file_to_dup __attribute__((unused))) { - info->dfile= my_open(share->data_file_name, share->mode | O_SHARE, - MYF(MY_WME)); - return info->dfile >= 0 ? 0 : 1; + info->dfile.file= my_open(share->data_file_name, share->mode | O_SHARE, + MYF(MY_WME)); + return info->dfile.file >= 0 ? 0 : 1; } int _ma_open_keyfile(MARIA_SHARE *share) { - if ((share->kfile=my_open(share->unique_file_name, share->mode | O_SHARE, + if ((share->kfile.file= my_open(share->unique_file_name, + share->mode | O_SHARE, MYF(MY_WME))) < 0) return 1; return 0; diff --git a/storage/maria/ma_packrec.c b/storage/maria/ma_packrec.c index 7c875e7b91d..30b7d0b0eec 100644 --- a/storage/maria/ma_packrec.c +++ b/storage/maria/ma_packrec.c @@ -483,7 +483,7 @@ int _ma_read_pack_record(MARIA_HA *info, byte *buf, MARIA_RECORD_POS filepos) if (filepos == HA_OFFSET_ERROR) DBUG_RETURN(-1); /* _search() didn't find record */ - file=info->dfile; + file= info->dfile.file; if (_ma_pack_get_block_info(info, &info->bit_buff, &block_info, &info->rec_buff, &info->rec_buff_size, file, filepos)) @@ -1064,7 +1064,7 @@ int _ma_read_rnd_pack_record(MARIA_HA *info, goto err; } - file= info->dfile; + file= info->dfile.file; if (info->opt_flag & READ_CACHE_USED) { if (_ma_read_cache(&info->rec_cache, (byte*) block_info.header, @@ -1094,7 +1094,7 @@ int _ma_read_rnd_pack_record(MARIA_HA *info, } else { - if (my_read(info->dfile,(byte*) info->rec_buff + block_info.offset, + if (my_read(info->dfile.file, (byte*)info->rec_buff + block_info.offset, block_info.rec_len-block_info.offset, MYF(MY_NABP))) goto err; @@ -1250,7 +1250,7 @@ my_bool _ma_memmap_file(MARIA_HA *info) if (!info->s->file_map) { - if (my_seek(info->dfile,0L,MY_SEEK_END,MYF(0)) < + if (my_seek(info->dfile.file, 0L, MY_SEEK_END, MYF(0)) < share->state.state.data_file_length+MEMMAP_EXTRA_MARGIN) { DBUG_PRINT("warning",("File isn't extended for memmap")); diff --git a/storage/maria/ma_page.c b/storage/maria/ma_page.c index d0a11ad08ab..96907e0ffb0 100644 --- a/storage/maria/ma_page.c +++ b/storage/maria/ma_page.c @@ -29,14 +29,20 @@ byte *_ma_fetch_keypage(register MARIA_HA *info, MARIA_KEYDEF *keyinfo, DBUG_ENTER("_ma_fetch_keypage"); DBUG_PRINT("enter",("page: %ld", (long) page)); - tmp= key_cache_read(info->s->key_cache, info->s->kfile, page, level, buff, - info->s->block_size, info->s->block_size, - return_buffer); + DBUG_ASSERT(info->s->pagecache->block_size == keyinfo->block_length && + info->s->pagecache->block_size == info->s->block_size); + /* + TODO: replace PAGECACHE_PLAIN_PAGE with PAGECACHE_LSN_PAGE when + LSN on the pages will be implemented + */ + tmp= pagecache_read(info->s->pagecache, &info->s->kfile, + page / keyinfo->block_length, level, buff, + PAGECACHE_PLAIN_PAGE, PAGECACHE_LOCK_LEFT_UNLOCKED, 0); if (tmp == info->buff) info->keybuff_used=1; else if (!tmp) { - DBUG_PRINT("error",("Got errno: %d from key_cache_read",my_errno)); + DBUG_PRINT("error",("Got errno: %d from pagecache_read",my_errno)); info->last_keypage=HA_OFFSET_ERROR; maria_print_error(info->s, HA_ERR_CRASHED); my_errno=HA_ERR_CRASHED; @@ -63,7 +69,6 @@ byte *_ma_fetch_keypage(register MARIA_HA *info, MARIA_KEYDEF *keyinfo, int _ma_write_keypage(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, my_off_t page, int level, byte *buff) { - reg3 uint length; DBUG_ENTER("_ma_write_keypage"); #ifndef FAST /* Safety check */ @@ -71,7 +76,8 @@ int _ma_write_keypage(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, page+keyinfo->block_length > info->state->key_file_length || (page & (MARIA_MIN_KEY_BLOCK_LENGTH-1))) { - DBUG_PRINT("error",("Trying to write inside key status region: key_start: %lu length: %lu page: %lu", + DBUG_PRINT("error",("Trying to write inside key status region: " + "key_start: %lu length: %lu page: %lu", (long) info->s->base.keystart, (long) info->state->key_file_length, (long) page)); @@ -82,21 +88,18 @@ int _ma_write_keypage(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, DBUG_DUMP("buff",(byte*) buff,maria_getint(buff)); #endif - if ((length=keyinfo->block_length) > IO_SIZE*2 && - info->state->key_file_length != page+length) - length= ((maria_getint(buff)+IO_SIZE-1) & (uint) ~(IO_SIZE-1)); -#ifdef HAVE_purify - { - length=maria_getint(buff); - bzero((byte*) buff+length,keyinfo->block_length-length); - length=keyinfo->block_length; - } -#endif - DBUG_RETURN((key_cache_write(info->s->key_cache, - info->s->kfile,page, level, (byte*) buff,length, - (uint) keyinfo->block_length, - (int) ((info->lock_type != F_UNLCK) || - info->s->delay_key_write)))); + DBUG_ASSERT(info->s->pagecache->block_size == keyinfo->block_length); + DBUG_ASSERT(info->s->pagecache->block_size == info->s->block_size); + /* + TODO: replace PAGECACHE_PLAIN_PAGE with PAGECACHE_LSN_PAGE when + LSN on the pages will be implemented + */ + DBUG_RETURN(pagecache_write(info->s->pagecache, + &info->s->kfile, page / keyinfo->block_length, + level, buff, PAGECACHE_PLAIN_PAGE, + PAGECACHE_LOCK_LEFT_UNLOCKED, + PAGECACHE_PIN_LEFT_PINNED, + PAGECACHE_WRITE_DELAY, 0)); } /* maria_write_keypage */ @@ -107,18 +110,31 @@ int _ma_dispose(register MARIA_HA *info, MARIA_KEYDEF *keyinfo, my_off_t pos, { my_off_t old_link; char buff[8]; + uint offset; + pgcache_page_no_t page_no; DBUG_ENTER("_ma_dispose"); DBUG_PRINT("enter",("pos: %ld", (long) pos)); old_link= info->s->state.key_del; info->s->state.key_del= pos; + page_no= pos / keyinfo->block_length; + offset= pos % keyinfo->block_length; mi_sizestore(buff,old_link); info->s->state.changed|= STATE_NOT_SORTED_PAGES; - DBUG_RETURN(key_cache_write(info->s->key_cache, - info->s->kfile, pos , level, buff, - sizeof(buff), - (uint) keyinfo->block_length, - (int) (info->lock_type != F_UNLCK))); + + DBUG_ASSERT(info->s->pagecache->block_size == keyinfo->block_length && + info->s->pagecache->block_size == info->s->block_size); + /* + TODO: replace PAGECACHE_PLAIN_PAGE with PAGECACHE_LSN_PAGE when + LSN on the pages will be implemented + */ + DBUG_RETURN(pagecache_write_part(info->s->pagecache, + &info->s->kfile, page_no, level, buff, + PAGECACHE_PLAIN_PAGE, + PAGECACHE_LOCK_LEFT_UNLOCKED, + PAGECACHE_PIN_LEFT_PINNED, + PAGECACHE_WRITE_DELAY, 0, + offset, sizeof(buff))); } /* _ma_dispose */ @@ -143,11 +159,17 @@ my_off_t _ma_new(register MARIA_HA *info, MARIA_KEYDEF *keyinfo, int level) } else { - if (!key_cache_read(info->s->key_cache, - info->s->kfile, pos, level, - buff, - (uint) sizeof(buff), - (uint) keyinfo->block_length,0)) + DBUG_ASSERT(info->s->pagecache->block_size == keyinfo->block_length && + info->s->pagecache->block_size == info->s->block_size); + /* + TODO: replace PAGECACHE_PLAIN_PAGE with PAGECACHE_LSN_PAGE when + LSN on the pages will be implemented + */ + DBUG_ASSERT(info->s->pagecache->block_size == keyinfo->block_length); + if (!pagecache_read(info->s->pagecache, + &info->s->kfile, pos / keyinfo->block_length, level, + buff, PAGECACHE_PLAIN_PAGE, + PAGECACHE_LOCK_LEFT_UNLOCKED, 0)) pos= HA_OFFSET_ERROR; else info->s->state.key_del= mi_sizekorr(buff); diff --git a/storage/maria/ma_pagecache.c b/storage/maria/ma_pagecache.c new file mode 100755 index 00000000000..ea63210ce95 --- /dev/null +++ b/storage/maria/ma_pagecache.c @@ -0,0 +1,4100 @@ +/* Copyright (C) 2000-2006 MySQL AB + + 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; either version 2 of the License, or + (at your option) any later version. + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ + +/* + These functions handle page cacheing for Maria tables. + + One cache can handle many files. + It must contain buffers of the same blocksize. + init_pagecache() should be used to init cache handler. + + The free list (free_block_list) is a stack like structure. + When a block is freed by free_block(), it is pushed onto the stack. + When a new block is required it is first tried to pop one from the stack. + If the stack is empty, it is tried to get a never-used block from the pool. + If this is empty too, then a block is taken from the LRU ring, flushing it + to disk, if necessary. This is handled in find_block(). + With the new free list, the blocks can have three temperatures: + hot, warm and cold (which is free). This is remembered in the block header + by the enum PCBLOCK_TEMPERATURE temperature variable. Remembering the + temperature is necessary to correctly count the number of warm blocks, + which is required to decide when blocks are allowed to become hot. Whenever + a block is inserted to another (sub-)chain, we take the old and new + temperature into account to decide if we got one more or less warm block. + blocks_unused is the sum of never used blocks in the pool and of currently + free blocks. blocks_used is the number of blocks fetched from the pool and + as such gives the maximum number of in-use blocks at any time. +*/ + +#include "maria_def.h" +#include +#include +#include +#include +#include + +/* + Some compilation flags have been added specifically for this module + to control the following: + - not to let a thread to yield the control when reading directly + from page cache, which might improve performance in many cases; + to enable this add: + #define SERIALIZED_READ_FROM_CACHE + - to set an upper bound for number of threads simultaneously + using the page cache; this setting helps to determine an optimal + size for hash table and improve performance when the number of + blocks in the page cache much less than the number of threads + accessing it; + to set this number equal to add + #define MAX_THREADS + - to substitute calls of pthread_cond_wait for calls of + pthread_cond_timedwait (wait with timeout set up); + this setting should be used only when you want to trap a deadlock + situation, which theoretically should not happen; + to set timeout equal to seconds add + #define PAGECACHE_TIMEOUT + - to enable the module traps and to send debug information from + page cache module to a special debug log add: + #define PAGECACHE_DEBUG + the name of this debug log file can be set through: + #define PAGECACHE_DEBUG_LOG + if the name is not defined, it's set by default; + if the PAGECACHE_DEBUG flag is not set up and we are in a debug + mode, i.e. when ! defined(DBUG_OFF), the debug information from the + module is sent to the regular debug log. + + Example of the settings: + #define SERIALIZED_READ_FROM_CACHE + #define MAX_THREADS 100 + #define PAGECACHE_TIMEOUT 1 + #define PAGECACHE_DEBUG + #define PAGECACHE_DEBUG_LOG "my_pagecache_debug.log" +*/ + +/* + In key cache we have external raw locking here we use + SERIALIZED_READ_FROM_CACHE to avoid problem of reading + not consistent data from the page. + (keycache functions (key_cache_read(), key_cache_insert() and + key_cache_write()) rely on external MyISAM lock, we don't) +*/ +#define SERIALIZED_READ_FROM_CACHE yes + +#define PCBLOCK_INFO(B) \ + DBUG_PRINT("info", \ + ("block 0x%lx file %lu page %lu s %0x hshL 0x%lx req %u/%u " \ + "wrlock: %c", \ + (ulong)(B), \ + (ulong)((B)->hash_link ? \ + (B)->hash_link->file.file : \ + 0), \ + (ulong)((B)->hash_link ? \ + (B)->hash_link->pageno : \ + 0), \ + (B)->status, \ + (ulong)(B)->hash_link, \ + (uint) (B)->requests, \ + (uint)((B)->hash_link ? \ + (B)->hash_link->requests : \ + 0), \ + ((block->status & PCBLOCK_WRLOCK)?'Y':'N'))) + +/* TODO: put it to my_static.c */ +my_bool my_disable_flush_pagecache_blocks= 0; + +#define STRUCT_PTR(TYPE, MEMBER, a) \ + (TYPE *) ((char *) (a) - offsetof(TYPE, MEMBER)) + +/* types of condition variables */ +#define COND_FOR_REQUESTED 0 /* queue of thread waiting for read operation */ +#define COND_FOR_SAVED 1 /* queue of thread waiting for flush */ +#define COND_FOR_WRLOCK 2 /* queue of write lock */ +#define COND_SIZE 3 /* number of COND_* queues */ + +typedef pthread_cond_t KEYCACHE_CONDVAR; + +/* descriptor of the page in the page cache block buffer */ +struct st_pagecache_page +{ + PAGECACHE_FILE file; /* file to which the page belongs to */ + pgcache_page_no_t pageno; /* number of the page in the file */ +}; + +/* element in the chain of a hash table bucket */ +struct st_pagecache_hash_link +{ + struct st_pagecache_hash_link + *next, **prev; /* to connect links in the same bucket */ + struct st_pagecache_block_link + *block; /* reference to the block for the page: */ + PAGECACHE_FILE file; /* from such a file */ + pgcache_page_no_t pageno; /* this page */ + uint requests; /* number of requests for the page */ +}; + +/* simple states of a block */ +#define PCBLOCK_ERROR 1 /* an error occurred when performing disk i/o */ +#define PCBLOCK_READ 2 /* the is page in the block buffer */ +#define PCBLOCK_IN_SWITCH 4 /* block is preparing to read new page */ +#define PCBLOCK_REASSIGNED 8 /* block does not accept requests for old page */ +#define PCBLOCK_IN_FLUSH 16 /* block is in flush operation */ +#define PCBLOCK_CHANGED 32 /* block buffer contains a dirty page */ +#define PCBLOCK_WRLOCK 64 /* write locked block */ + +/* page status, returned by find_block */ +#define PAGE_READ 0 +#define PAGE_TO_BE_READ 1 +#define PAGE_WAIT_TO_BE_READ 2 + +/* block temperature determines in which (sub-)chain the block currently is */ +enum PCBLOCK_TEMPERATURE { PCBLOCK_COLD /*free*/ , PCBLOCK_WARM , PCBLOCK_HOT }; + +/* debug info */ +#ifndef DBUG_OFF +static char *page_cache_page_type_str[]= +{ + (char*)"PLAIN", + (char*)"LSN" +}; +static char *page_cache_page_write_mode_str[]= +{ + (char*)"DELAY", + (char*)"NOW", + (char*)"DONE" +}; +static char *page_cache_page_lock_str[]= +{ + (char*)"free -> free ", + (char*)"read -> read ", + (char*)"write -> write", + (char*)"free -> read ", + (char*)"free -> write", + (char*)"read -> free ", + (char*)"write -> free ", + (char*)"write -> read " +}; +static char *page_cache_page_pin_str[]= +{ + (char*)"pinned -> pinned ", + (char*)"unpinned -> unpinned", + (char*)"unpinned -> pinned ", + (char*)"pinned -> unpinned" +}; +#endif +#ifdef PAGECACHE_DEBUG +typedef struct st_pagecache_pin_info +{ + struct st_pagecache_pin_info *next, **prev; + struct st_my_thread_var *thread; +} PAGECACHE_PIN_INFO; +/* + st_pagecache_lock_info structure should be kept in next, prev, thread part + compatible with st_pagecache_pin_info to be compatible in functions. +*/ +typedef struct st_pagecache_lock_info +{ + struct st_pagecache_lock_info *next, **prev; + struct st_my_thread_var *thread; + my_bool write_lock; +} PAGECACHE_LOCK_INFO; + + +/* service functions maintain debugging info about pin & lock */ + + +/* + Links information about thread pinned/locked the block to the list + + SYNOPSIS + info_link() + list the list to link in + node the node which should be linked +*/ + +static void info_link(PAGECACHE_PIN_INFO **list, PAGECACHE_PIN_INFO *node) +{ + if ((node->next= *list)) + node->next->prev= &(node->next); + *list= node; + node->prev= list; +} + + +/* + Unlinks information about thread pinned/locked the block from the list + + SYNOPSIS + info_unlink() + node the node which should be unlinked +*/ + +static void info_unlink(PAGECACHE_PIN_INFO *node) +{ + if ((*node->prev= node->next)) + node->next->prev= node->prev; +} + + +/* + Finds information about given thread in the list of threads which + pinned/locked this block. + + SYNOPSIS + info_find() + list the list where to find the thread + thread thread ID (reference to the st_my_thread_var + of the thread) + + RETURN + 0 - the thread was not found + pointer to the information node of the thread in the list +*/ + +static PAGECACHE_PIN_INFO *info_find(PAGECACHE_PIN_INFO *list, + struct st_my_thread_var *thread) +{ + register PAGECACHE_PIN_INFO *i= list; + for(; i != 0; i= i->next) + if (i->thread == thread) + return i; + return 0; +} +#endif + +/* page cache block */ +struct st_pagecache_block_link +{ + struct st_pagecache_block_link + *next_used, **prev_used; /* to connect links in the LRU chain (ring) */ + struct st_pagecache_block_link + *next_changed, **prev_changed; /* for lists of file dirty/clean blocks */ + struct st_pagecache_hash_link + *hash_link; /* backward ptr to referring hash_link */ + WQUEUE + wqueue[COND_SIZE]; /* queues on waiting requests for new/old pages */ + uint requests; /* number of requests for the block */ + byte *buffer; /* buffer for the block page */ + uint status; /* state of the block */ + uint pins; /* pin counter */ +#ifdef PAGECACHE_DEBUG + PAGECACHE_PIN_INFO *pin_list; + PAGECACHE_LOCK_INFO *lock_list; +#endif + enum PCBLOCK_TEMPERATURE temperature; /* block temperature: cold, warm, hot */ + enum pagecache_page_type type; /* type of the block */ + uint hits_left; /* number of hits left until promotion */ + ulonglong last_hit_time; /* timestamp of the last hit */ + LSN rec_lsn; /* LSN when first became dirty */ + KEYCACHE_CONDVAR *condvar; /* condition variable for 'no readers' event */ +}; + +PAGECACHE dflt_pagecache_var; +PAGECACHE *dflt_pagecache= &dflt_pagecache_var; + +#ifdef PAGECACHE_DEBUG +/* debug checks */ +static my_bool info_check_pin(PAGECACHE_BLOCK_LINK *block, + enum pagecache_page_pin mode) +{ + struct st_my_thread_var *thread= my_thread_var; + PAGECACHE_PIN_INFO *info= info_find(block->pin_list, thread); + DBUG_ENTER("info_check_pin"); + if (info) + { + if (mode == PAGECACHE_PIN_LEFT_UNPINNED) + { + DBUG_PRINT("info", + ("info_check_pin: thread: 0x%lx block 0x%lx: LEFT_UNPINNED!!!", + (ulong)thread, (ulong)block)); + DBUG_RETURN(1); + } + else if (mode == PAGECACHE_PIN) + { + DBUG_PRINT("info", + ("info_check_pin: thread: 0x%lx block 0x%lx: PIN!!!", + (ulong)thread, (ulong)block)); + DBUG_RETURN(1); + } + } + else + { + if (mode == PAGECACHE_PIN_LEFT_PINNED) + { + DBUG_PRINT("info", + ("info_check_pin: thread: 0x%lx block 0x%lx: LEFT_PINNED!!!", + (ulong)thread, (ulong)block)); + DBUG_RETURN(1); + } + else if (mode == PAGECACHE_UNPIN) + { + DBUG_PRINT("info", + ("info_check_pin: thread: 0x%lx block 0x%lx: UNPIN!!!", + (ulong)thread, (ulong)block)); + DBUG_RETURN(1); + } + } + DBUG_RETURN(0); +} + + +/* + Debug function which checks current lock/pin state and requested changes + + SYNOPSIS + info_check_lock() + lock requested lock changes + pin requested pin changes + + RETURN + 0 - OK + 1 - Error +*/ + +static my_bool info_check_lock(PAGECACHE_BLOCK_LINK *block, + enum pagecache_page_lock lock, + enum pagecache_page_pin pin) +{ + struct st_my_thread_var *thread= my_thread_var; + PAGECACHE_LOCK_INFO *info= + (PAGECACHE_LOCK_INFO *) info_find((PAGECACHE_PIN_INFO *) block->lock_list, + thread); + DBUG_ENTER("info_check_lock"); + switch(lock) + { + case PAGECACHE_LOCK_LEFT_UNLOCKED: + if (pin != PAGECACHE_PIN_LEFT_UNPINNED || + info) + goto error; + break; + case PAGECACHE_LOCK_LEFT_READLOCKED: + if ((pin != PAGECACHE_PIN_LEFT_UNPINNED && + pin != PAGECACHE_PIN_LEFT_PINNED) || + info == 0 || info->write_lock) + goto error; + break; + case PAGECACHE_LOCK_LEFT_WRITELOCKED: + if (pin != PAGECACHE_PIN_LEFT_PINNED || + info == 0 || !info->write_lock) + goto error; + break; + case PAGECACHE_LOCK_READ: + if ((pin != PAGECACHE_PIN_LEFT_UNPINNED && + pin != PAGECACHE_PIN) || + info != 0) + goto error; + break; + case PAGECACHE_LOCK_WRITE: + if (pin != PAGECACHE_PIN || + info != 0) + goto error; + break; + case PAGECACHE_LOCK_READ_UNLOCK: + if ((pin != PAGECACHE_PIN_LEFT_UNPINNED && + pin != PAGECACHE_UNPIN) || + info == 0 || info->write_lock) + goto error; + break; + case PAGECACHE_LOCK_WRITE_UNLOCK: + if (pin != PAGECACHE_UNPIN || + info == 0 || !info->write_lock) + goto error; + break; + case PAGECACHE_LOCK_WRITE_TO_READ: + if ((pin != PAGECACHE_PIN_LEFT_PINNED && + pin != PAGECACHE_UNPIN) || + info == 0 || !info->write_lock) + goto error; + break; + } + DBUG_RETURN(0); +error: + DBUG_PRINT("info", + ("info_check_lock: thread: 0x%lx block 0x%lx: info: %d wrt: %d," + "to lock: %s, to pin: %s", + (ulong)thread, (ulong)block, test(info), + (info ? info->write_lock : 0), + page_cache_page_lock_str[lock], + page_cache_page_pin_str[pin])); + DBUG_RETURN(1); +} +#endif + +#define FLUSH_CACHE 2000 /* sort this many blocks at once */ + +static void free_block(PAGECACHE *pagecache, PAGECACHE_BLOCK_LINK *block); +static void test_key_cache(PAGECACHE *pagecache, + const char *where, my_bool lock); + +#define PAGECACHE_HASH(p, f, pos) (((ulong) (pos) + \ + (ulong) (f).file) & (p->hash_entries-1)) +#define FILE_HASH(f) ((uint) (f).file & (PAGECACHE_CHANGED_BLOCKS_HASH - 1)) + +#define DEFAULT_PAGECACHE_DEBUG_LOG "pagecache_debug.log" + +#if defined(PAGECACHE_DEBUG) && ! defined(PAGECACHE_DEBUG_LOG) +#define PAGECACHE_DEBUG_LOG DEFAULT_PAGECACHE_DEBUG_LOG +#endif + +#if defined(PAGECACHE_DEBUG_LOG) +static FILE *pagecache_debug_log= NULL; +static void pagecache_debug_print _VARARGS((const char *fmt, ...)); +#define PAGECACHE_DEBUG_OPEN \ + if (!pagecache_debug_log) \ + { \ + pagecache_debug_log= fopen(PAGECACHE_DEBUG_LOG, "w"); \ + (void) setvbuf(pagecache_debug_log, NULL, _IOLBF, BUFSIZ); \ + } + +#define PAGECACHE_DEBUG_CLOSE \ + if (pagecache_debug_log) \ + { \ + fclose(pagecache_debug_log); \ + pagecache_debug_log= 0; \ + } +#else +#define PAGECACHE_DEBUG_OPEN +#define PAGECACHE_DEBUG_CLOSE +#endif /* defined(PAGECACHE_DEBUG_LOG) */ + +#if defined(PAGECACHE_DEBUG_LOG) && defined(PAGECACHE_DEBUG) +#define KEYCACHE_DBUG_PRINT(l, m) \ + { if (pagecache_debug_log) \ + fprintf(pagecache_debug_log, "%s: ", l); \ + pagecache_debug_print m; } + +#define KEYCACHE_DBUG_ASSERT(a) \ + { if (! (a) && pagecache_debug_log) \ + fclose(pagecache_debug_log); \ + assert(a); } +#else +#define KEYCACHE_DBUG_PRINT(l, m) DBUG_PRINT(l, m) +#define KEYCACHE_DBUG_ASSERT(a) DBUG_ASSERT(a) +#endif /* defined(PAGECACHE_DEBUG_LOG) && defined(PAGECACHE_DEBUG) */ + +#if defined(PAGECACHE_DEBUG) || !defined(DBUG_OFF) +#ifdef THREAD +static long pagecache_thread_id; +#define KEYCACHE_THREAD_TRACE(l) \ + KEYCACHE_DBUG_PRINT(l,("|thread %ld",pagecache_thread_id)) + +#define KEYCACHE_THREAD_TRACE_BEGIN(l) \ + { struct st_my_thread_var *thread_var= my_thread_var; \ + pagecache_thread_id= thread_var->id; \ + KEYCACHE_DBUG_PRINT(l,("[thread %ld",pagecache_thread_id)) } + +#define KEYCACHE_THREAD_TRACE_END(l) \ + KEYCACHE_DBUG_PRINT(l,("]thread %ld",pagecache_thread_id)) +#else /* THREAD */ +#define KEYCACHE_THREAD_TRACE(l) KEYCACHE_DBUG_PRINT(l,("")) +#define KEYCACHE_THREAD_TRACE_BEGIN(l) KEYCACHE_DBUG_PRINT(l,("")) +#define KEYCACHE_THREAD_TRACE_END(l) KEYCACHE_DBUG_PRINT(l,("")) +#endif /* THREAD */ +#else +#define KEYCACHE_THREAD_TRACE_BEGIN(l) +#define KEYCACHE_THREAD_TRACE_END(l) +#define KEYCACHE_THREAD_TRACE(l) +#endif /* defined(PAGECACHE_DEBUG) || !defined(DBUG_OFF) */ + +#define PCBLOCK_NUMBER(p, b) \ + ((uint) (((char*)(b)-(char *) p->block_root)/sizeof(PAGECACHE_BLOCK_LINK))) +#define PAGECACHE_HASH_LINK_NUMBER(p, h) \ + ((uint) (((char*)(h)-(char *) p->hash_link_root)/ \ + sizeof(PAGECACHE_HASH_LINK))) + +#if (defined(PAGECACHE_TIMEOUT) && !defined(__WIN__)) || defined(PAGECACHE_DEBUG) +static int pagecache_pthread_cond_wait(pthread_cond_t *cond, + pthread_mutex_t *mutex); +#else +#define pagecache_pthread_cond_wait pthread_cond_wait +#endif + +#if defined(PAGECACHE_DEBUG) +static int ___pagecache_pthread_mutex_lock(pthread_mutex_t *mutex); +static void ___pagecache_pthread_mutex_unlock(pthread_mutex_t *mutex); +static int ___pagecache_pthread_cond_signal(pthread_cond_t *cond); +#define pagecache_pthread_mutex_lock(M) \ +{ DBUG_PRINT("lock", ("mutex lock 0x%lx %u", (ulong)(M), __LINE__)); \ + ___pagecache_pthread_mutex_lock(M);} +#define pagecache_pthread_mutex_unlock(M) \ +{ DBUG_PRINT("lock", ("mutex unlock 0x%lx %u", (ulong)(M), __LINE__)); \ + ___pagecache_pthread_mutex_unlock(M);} +#define pagecache_pthread_cond_signal(M) \ +{ DBUG_PRINT("lock", ("signal 0x%lx %u", (ulong)(M), __LINE__)); \ + ___pagecache_pthread_cond_signal(M);} +#else +#define pagecache_pthread_mutex_lock pthread_mutex_lock +#define pagecache_pthread_mutex_unlock pthread_mutex_unlock +#define pagecache_pthread_cond_signal pthread_cond_signal +#endif /* defined(PAGECACHE_DEBUG) */ + +extern my_bool translog_flush(LSN lsn); + +/* + Write page to the disk + + SYNOPSIS + pagecache_fwrite() + pagecache - page cache pointer + filedesc - pagecache file descriptor structure + buffer - buffer which we will write + type - page type (plain or with LSN) + flags - MYF() flags + + RETURN + 0 - OK + !=0 - Error +*/ + +static uint pagecache_fwrite(PAGECACHE *pagecache, + PAGECACHE_FILE *filedesc, + byte *buffer, + pgcache_page_no_t pageno, + enum pagecache_page_type type, + myf flags) +{ + DBUG_ENTER("pagecache_fwrite"); + if (type == PAGECACHE_LSN_PAGE) + { + LSN lsn; + DBUG_PRINT("info", ("Log handler call")); + /* TODO: integrate with page format */ +#define PAGE_LSN_OFFSET 0 + lsn= lsn_korr(buffer + PAGE_LSN_OFFSET); + /* + check CONTROL_FILE_IMPOSSIBLE_FILENO & + CONTROL_FILE_IMPOSSIBLE_LOG_OFFSET + */ + DBUG_ASSERT(lsn != 0); + translog_flush(lsn); + } + DBUG_RETURN(my_pwrite(filedesc->file, buffer, pagecache->block_size, + (pageno)<<(pagecache->shift), flags)); +} + + +/* + Read page from the disk + + SYNOPSIS + pagecache_fread() + pagecache - page cache pointer + filedesc - pagecache file descriptor structure + buffer - buffer in which we will read + pageno - page number + flags - MYF() flags +*/ +#define pagecache_fread(pagecache, filedesc, buffer, pageno, flags) \ + my_pread((filedesc)->file, buffer, pagecache->block_size, \ + (pageno)<<(pagecache->shift), flags) + + +/* + next_power(value) is 2 at the power of (1+floor(log2(value))); + e.g. next_power(2)=4, next_power(3)=4. +*/ +static inline uint next_power(uint value) +{ + return (uint) my_round_up_to_next_power((uint32) value) << 1; +} + + +/* + Initialize a page cache + + SYNOPSIS + init_pagecache() + pagecache pointer to a page cache data structure + key_cache_block_size size of blocks to keep cached data + use_mem total memory to use for the key cache + division_limit division limit (may be zero) + age_threshold age threshold (may be zero) + block_size size of block (should be power of 2) + + RETURN VALUE + number of blocks in the key cache, if successful, + 0 - otherwise. + + NOTES. + if pagecache->inited != 0 we assume that the key cache + is already initialized. This is for now used by myisamchk, but shouldn't + be something that a program should rely on! + + It's assumed that no two threads call this function simultaneously + referring to the same key cache handle. + +*/ + +int init_pagecache(PAGECACHE *pagecache, my_size_t use_mem, + uint division_limit, uint age_threshold, + uint block_size) +{ + uint blocks, hash_links, length; + int error; + DBUG_ENTER("init_pagecache"); + DBUG_ASSERT(block_size >= 512); + + PAGECACHE_DEBUG_OPEN; + if (pagecache->inited && pagecache->disk_blocks > 0) + { + DBUG_PRINT("warning",("key cache already in use")); + DBUG_RETURN(0); + } + + pagecache->global_cache_w_requests= pagecache->global_cache_r_requests= 0; + pagecache->global_cache_read= pagecache->global_cache_write= 0; + pagecache->disk_blocks= -1; + if (! pagecache->inited) + { + pagecache->inited= 1; + pagecache->in_init= 0; + pthread_mutex_init(&pagecache->cache_lock, MY_MUTEX_INIT_FAST); + pagecache->resize_queue.last_thread= NULL; + } + + pagecache->mem_size= use_mem; + pagecache->block_size= block_size; + pagecache->shift= my_bit_log2(block_size); + DBUG_PRINT("info", ("block_size: %u", + block_size)); + DBUG_ASSERT(((uint)(1 << pagecache->shift)) == block_size); + + blocks= (int) (use_mem / (sizeof(PAGECACHE_BLOCK_LINK) + + 2 * sizeof(PAGECACHE_HASH_LINK) + + sizeof(PAGECACHE_HASH_LINK*) * + 5/4 + block_size)); + /* It doesn't make sense to have too few blocks (less than 8) */ + if (blocks >= 8 && pagecache->disk_blocks < 0) + { + for ( ; ; ) + { + /* Set my_hash_entries to the next bigger 2 power */ + if ((pagecache->hash_entries= next_power(blocks)) < + (blocks) * 5/4) + pagecache->hash_entries<<= 1; + hash_links= 2 * blocks; +#if defined(MAX_THREADS) + if (hash_links < MAX_THREADS + blocks - 1) + hash_links= MAX_THREADS + blocks - 1; +#endif + while ((length= (ALIGN_SIZE(blocks * sizeof(PAGECACHE_BLOCK_LINK)) + + ALIGN_SIZE(hash_links * sizeof(PAGECACHE_HASH_LINK)) + + ALIGN_SIZE(sizeof(PAGECACHE_HASH_LINK*) * + pagecache->hash_entries))) + + (((ulong) blocks) << pagecache->shift) > use_mem) + blocks--; + /* Allocate memory for cache page buffers */ + if ((pagecache->block_mem= + my_large_malloc((ulong) blocks * pagecache->block_size, + MYF(MY_WME)))) + { + /* + Allocate memory for blocks, hash_links and hash entries; + For each block 2 hash links are allocated + */ + if ((pagecache->block_root= + (PAGECACHE_BLOCK_LINK*) my_malloc((uint) length, + MYF(0)))) + break; + my_large_free(pagecache->block_mem, MYF(0)); + pagecache->block_mem= 0; + } + if (blocks < 8) + { + my_errno= ENOMEM; + goto err; + } + blocks= blocks / 4*3; + } + pagecache->blocks_unused= (ulong) blocks; + pagecache->disk_blocks= (int) blocks; + pagecache->hash_links= hash_links; + pagecache->hash_root= + (PAGECACHE_HASH_LINK**) ((char*) pagecache->block_root + + ALIGN_SIZE(blocks*sizeof(PAGECACHE_BLOCK_LINK))); + pagecache->hash_link_root= + (PAGECACHE_HASH_LINK*) ((char*) pagecache->hash_root + + ALIGN_SIZE((sizeof(PAGECACHE_HASH_LINK*) * + pagecache->hash_entries))); + bzero((byte*) pagecache->block_root, + pagecache->disk_blocks * sizeof(PAGECACHE_BLOCK_LINK)); + bzero((byte*) pagecache->hash_root, + pagecache->hash_entries * sizeof(PAGECACHE_HASH_LINK*)); + bzero((byte*) pagecache->hash_link_root, + pagecache->hash_links * sizeof(PAGECACHE_HASH_LINK)); + pagecache->hash_links_used= 0; + pagecache->free_hash_list= NULL; + pagecache->blocks_used= pagecache->blocks_changed= 0; + + pagecache->global_blocks_changed= 0; + pagecache->blocks_available=0; /* For debugging */ + + /* The LRU chain is empty after initialization */ + pagecache->used_last= NULL; + pagecache->used_ins= NULL; + pagecache->free_block_list= NULL; + pagecache->time= 0; + pagecache->warm_blocks= 0; + pagecache->min_warm_blocks= (division_limit ? + blocks * division_limit / 100 + 1 : + blocks); + pagecache->age_threshold= (age_threshold ? + blocks * age_threshold / 100 : + blocks); + + pagecache->cnt_for_resize_op= 0; + pagecache->resize_in_flush= 0; + pagecache->can_be_used= 1; + + pagecache->waiting_for_hash_link.last_thread= NULL; + pagecache->waiting_for_block.last_thread= NULL; + DBUG_PRINT("exit", + ("disk_blocks: %d block_root: 0x%lx hash_entries: %d\ + hash_root: 0x%lx hash_links: %d hash_link_root: 0x%lx", + pagecache->disk_blocks, (long) pagecache->block_root, + pagecache->hash_entries, (long) pagecache->hash_root, + pagecache->hash_links, (long) pagecache->hash_link_root)); + bzero((gptr) pagecache->changed_blocks, + sizeof(pagecache->changed_blocks[0]) * + PAGECACHE_CHANGED_BLOCKS_HASH); + bzero((gptr) pagecache->file_blocks, + sizeof(pagecache->file_blocks[0]) * + PAGECACHE_CHANGED_BLOCKS_HASH); + } + + pagecache->blocks= pagecache->disk_blocks > 0 ? pagecache->disk_blocks : 0; + DBUG_RETURN((uint) pagecache->blocks); + +err: + error= my_errno; + pagecache->disk_blocks= 0; + pagecache->blocks= 0; + if (pagecache->block_mem) + { + my_large_free((gptr) pagecache->block_mem, MYF(0)); + pagecache->block_mem= NULL; + } + if (pagecache->block_root) + { + my_free((gptr) pagecache->block_root, MYF(0)); + pagecache->block_root= NULL; + } + my_errno= error; + pagecache->can_be_used= 0; + DBUG_RETURN(0); +} + + +/* + Flush all blocks in the key cache to disk +*/ + +#ifdef NOT_USED +static int flush_all_key_blocks(PAGECACHE *pagecache) +{ +#if defined(PAGECACHE_DEBUG) + uint cnt=0; +#endif + while (pagecache->blocks_changed > 0) + { + PAGECACHE_BLOCK_LINK *block; + for (block= pagecache->used_last->next_used ; ; block=block->next_used) + { + if (block->hash_link) + { +#if defined(PAGECACHE_DEBUG) + cnt++; + KEYCACHE_DBUG_ASSERT(cnt <= pagecache->blocks_used); +#endif + if (flush_pagecache_blocks_int(pagecache, &block->hash_link->file, + FLUSH_RELEASE)) + return 1; + break; + } + if (block == pagecache->used_last) + break; + } + } + return 0; +} +#endif /* NOT_USED */ + +/* + Resize a key cache + + SYNOPSIS + resize_pagecache() + pagecache pointer to a page cache data structure + use_mem total memory to use for the new key cache + division_limit new division limit (if not zero) + age_threshold new age threshold (if not zero) + + RETURN VALUE + number of blocks in the key cache, if successful, + 0 - otherwise. + + NOTES. + The function first compares the memory size parameter + with the key cache value. + + If they differ the function free the the memory allocated for the + old key cache blocks by calling the end_pagecache function and + then rebuilds the key cache with new blocks by calling + init_key_cache. + + The function starts the operation only when all other threads + performing operations with the key cache let her to proceed + (when cnt_for_resize=0). + + Before being usable, this function needs: + - to receive fixes for BUG#17332 "changing key_buffer_size on a running + server can crash under load" similar to those done to the key cache + - to have us (Sanja) look at the additional constraints placed on + resizing, due to the page locking specific to this page cache. + So we disable it for now. +*/ +#if NOT_USED /* keep disabled until code is fixed see above !! */ +int resize_pagecache(PAGECACHE *pagecache, + my_size_t use_mem, uint division_limit, + uint age_threshold) +{ + int blocks; +#ifdef THREAD + struct st_my_thread_var *thread; + WQUEUE *wqueue; + +#endif + DBUG_ENTER("resize_pagecache"); + + if (!pagecache->inited) + DBUG_RETURN(pagecache->disk_blocks); + + if(use_mem == pagecache->mem_size) + { + change_pagecache_param(pagecache, division_limit, age_threshold); + DBUG_RETURN(pagecache->disk_blocks); + } + + pagecache_pthread_mutex_lock(&pagecache->cache_lock); + +#ifdef THREAD + wqueue= &pagecache->resize_queue; + thread= my_thread_var; + wqueue_link_into_queue(wqueue, thread); + + while (wqueue->last_thread->next != thread) + { + pagecache_pthread_cond_wait(&thread->suspend, &pagecache->cache_lock); + } +#endif + + pagecache->resize_in_flush= 1; + if (flush_all_key_blocks(pagecache)) + { + /* TODO: if this happens, we should write a warning in the log file ! */ + pagecache->resize_in_flush= 0; + blocks= 0; + pagecache->can_be_used= 0; + goto finish; + } + pagecache->resize_in_flush= 0; + pagecache->can_be_used= 0; +#ifdef THREAD + while (pagecache->cnt_for_resize_op) + { + KEYCACHE_DBUG_PRINT("resize_pagecache: wait", + ("suspend thread %ld", thread->id)); + pagecache_pthread_cond_wait(&thread->suspend, &pagecache->cache_lock); + } +#else + KEYCACHE_DBUG_ASSERT(pagecache->cnt_for_resize_op == 0); +#endif + + end_pagecache(pagecache, 0); /* Don't free mutex */ + /* The following will work even if use_mem is 0 */ + blocks= init_pagecache(pagecache, pagecache->block_size, use_mem, + division_limit, age_threshold); + +finish: +#ifdef THREAD + wqueue_unlink_from_queue(wqueue, thread); + /* Signal for the next resize request to proceeed if any */ + if (wqueue->last_thread) + { + KEYCACHE_DBUG_PRINT("resize_pagecache: signal", + ("thread %ld", wqueue->last_thread->next->id)); + pagecache_pthread_cond_signal(&wqueue->last_thread->next->suspend); + } +#endif + pagecache_pthread_mutex_unlock(&pagecache->cache_lock); + DBUG_RETURN(blocks); +} +#endif /* 0 */ + + +/* + Increment counter blocking resize key cache operation +*/ +static inline void inc_counter_for_resize_op(PAGECACHE *pagecache) +{ + pagecache->cnt_for_resize_op++; +} + + +/* + Decrement counter blocking resize key cache operation; + Signal the operation to proceed when counter becomes equal zero +*/ +static inline void dec_counter_for_resize_op(PAGECACHE *pagecache) +{ +#ifdef THREAD + struct st_my_thread_var *last_thread; + if (!--pagecache->cnt_for_resize_op && + (last_thread= pagecache->resize_queue.last_thread)) + { + KEYCACHE_DBUG_PRINT("dec_counter_for_resize_op: signal", + ("thread %ld", last_thread->next->id)); + pagecache_pthread_cond_signal(&last_thread->next->suspend); + } +#else + pagecache->cnt_for_resize_op--; +#endif +} + +/* + Change the page cache parameters + + SYNOPSIS + change_pagecache_param() + pagecache pointer to a page cache data structure + division_limit new division limit (if not zero) + age_threshold new age threshold (if not zero) + + RETURN VALUE + none + + NOTES. + Presently the function resets the key cache parameters + concerning midpoint insertion strategy - division_limit and + age_threshold. +*/ + +void change_pagecache_param(PAGECACHE *pagecache, uint division_limit, + uint age_threshold) +{ + DBUG_ENTER("change_pagecache_param"); + + pagecache_pthread_mutex_lock(&pagecache->cache_lock); + if (division_limit) + pagecache->min_warm_blocks= (pagecache->disk_blocks * + division_limit / 100 + 1); + if (age_threshold) + pagecache->age_threshold= (pagecache->disk_blocks * + age_threshold / 100); + pagecache_pthread_mutex_unlock(&pagecache->cache_lock); + DBUG_VOID_RETURN; +} + + +/* + Removes page cache from memory. Does NOT flush pages to disk. + + SYNOPSIS + end_pagecache() + pagecache page cache handle + cleanup Complete free (Free also mutex for key cache) + + RETURN VALUE + none +*/ + +void end_pagecache(PAGECACHE *pagecache, my_bool cleanup) +{ + DBUG_ENTER("end_pagecache"); + DBUG_PRINT("enter", ("key_cache: 0x%lx", (long) pagecache)); + + if (!pagecache->inited) + DBUG_VOID_RETURN; + + if (pagecache->disk_blocks > 0) + { + if (pagecache->block_mem) + { + my_large_free((gptr) pagecache->block_mem, MYF(0)); + pagecache->block_mem= NULL; + my_free((gptr) pagecache->block_root, MYF(0)); + pagecache->block_root= NULL; + } + pagecache->disk_blocks= -1; + /* Reset blocks_changed to be safe if flush_all_key_blocks is called */ + pagecache->blocks_changed= 0; + } + + DBUG_PRINT("status", ("used: %lu changed: %lu w_requests: %lu " + "writes: %lu r_requests: %lu reads: %lu", + pagecache->blocks_used, pagecache->global_blocks_changed, + (ulong) pagecache->global_cache_w_requests, + (ulong) pagecache->global_cache_write, + (ulong) pagecache->global_cache_r_requests, + (ulong) pagecache->global_cache_read)); + + if (cleanup) + { + pthread_mutex_destroy(&pagecache->cache_lock); + pagecache->inited= pagecache->can_be_used= 0; + PAGECACHE_DEBUG_CLOSE; + } + DBUG_VOID_RETURN; +} /* end_pagecache */ + + +/* + Unlink a block from the chain of dirty/clean blocks +*/ + +static inline void unlink_changed(PAGECACHE_BLOCK_LINK *block) +{ + if (block->next_changed) + block->next_changed->prev_changed= block->prev_changed; + *block->prev_changed= block->next_changed; +} + + +/* + Link a block into the chain of dirty/clean blocks +*/ + +static inline void link_changed(PAGECACHE_BLOCK_LINK *block, + PAGECACHE_BLOCK_LINK **phead) +{ + block->prev_changed= phead; + if ((block->next_changed= *phead)) + (*phead)->prev_changed= &block->next_changed; + *phead= block; +} + + +/* + Unlink a block from the chain of dirty/clean blocks, if it's asked for, + and link it to the chain of clean blocks for the specified file +*/ + +static void link_to_file_list(PAGECACHE *pagecache, + PAGECACHE_BLOCK_LINK *block, + PAGECACHE_FILE *file, my_bool unlink) +{ + if (unlink) + unlink_changed(block); + link_changed(block, &pagecache->file_blocks[FILE_HASH(*file)]); + if (block->status & PCBLOCK_CHANGED) + { + block->status&= ~PCBLOCK_CHANGED; + block->rec_lsn= 0; + pagecache->blocks_changed--; + pagecache->global_blocks_changed--; + } +} + + +/* + Unlink a block from the chain of clean blocks for the specified + file and link it to the chain of dirty blocks for this file +*/ + +static inline void link_to_changed_list(PAGECACHE *pagecache, + PAGECACHE_BLOCK_LINK *block) +{ + unlink_changed(block); + link_changed(block, + &pagecache->changed_blocks[FILE_HASH(block->hash_link->file)]); + block->status|=PCBLOCK_CHANGED; + pagecache->blocks_changed++; + pagecache->global_blocks_changed++; +} + + +/* + Link a block to the LRU chain at the beginning or at the end of + one of two parts. + + SYNOPSIS + link_block() + pagecache pointer to a page cache data structure + block pointer to the block to link to the LRU chain + hot <-> to link the block into the hot subchain + at_end <-> to link the block at the end of the subchain + + RETURN VALUE + none + + NOTES. + The LRU chain is represented by a curcular list of block structures. + The list is double-linked of the type (**prev,*next) type. + The LRU chain is divided into two parts - hot and warm. + There are two pointers to access the last blocks of these two + parts. The beginning of the warm part follows right after the + end of the hot part. + Only blocks of the warm part can be used for replacement. + The first block from the beginning of this subchain is always + taken for eviction (pagecache->last_used->next) + + LRU chain: +------+ H O T +------+ + +----| end |----...<----| beg |----+ + | +------+last +------+ | + v<-link in latest hot (new end) | + | link in latest warm (new end)->^ + | +------+ W A R M +------+ | + +----| beg |---->...----| end |----+ + +------+ +------+ins + first for eviction +*/ + +static void link_block(PAGECACHE *pagecache, PAGECACHE_BLOCK_LINK *block, + my_bool hot, my_bool at_end) +{ + PAGECACHE_BLOCK_LINK *ins; + PAGECACHE_BLOCK_LINK **ptr_ins; + + PCBLOCK_INFO(block); + KEYCACHE_DBUG_ASSERT(! (block->hash_link && block->hash_link->requests)); +#ifdef THREAD + if (!hot && pagecache->waiting_for_block.last_thread) + { + /* Signal that in the LRU warm sub-chain an available block has appeared */ + struct st_my_thread_var *last_thread= + pagecache->waiting_for_block.last_thread; + struct st_my_thread_var *first_thread= last_thread->next; + struct st_my_thread_var *next_thread= first_thread; + PAGECACHE_HASH_LINK *hash_link= + (PAGECACHE_HASH_LINK *) first_thread->opt_info; + struct st_my_thread_var *thread; + do + { + thread= next_thread; + next_thread= thread->next; + /* + We notify about the event all threads that ask + for the same page as the first thread in the queue + */ + if ((PAGECACHE_HASH_LINK *) thread->opt_info == hash_link) + { + KEYCACHE_DBUG_PRINT("link_block: signal", ("thread %ld", thread->id)); + pagecache_pthread_cond_signal(&thread->suspend); + wqueue_unlink_from_queue(&pagecache->waiting_for_block, thread); + block->requests++; + } + } + while (thread != last_thread); + hash_link->block= block; + KEYCACHE_THREAD_TRACE("link_block: after signaling"); +#if defined(PAGECACHE_DEBUG) + KEYCACHE_DBUG_PRINT("link_block", + ("linked,unlinked block %u status=%x #requests=%u #available=%u", + PCBLOCK_NUMBER(pagecache, block), block->status, + block->requests, pagecache->blocks_available)); +#endif + return; + } +#else /* THREAD */ + KEYCACHE_DBUG_ASSERT(! (!hot && pagecache->waiting_for_block.last_thread)); + /* Condition not transformed using DeMorgan, to keep the text identical */ +#endif /* THREAD */ + ptr_ins= hot ? &pagecache->used_ins : &pagecache->used_last; + ins= *ptr_ins; + if (ins) + { + ins->next_used->prev_used= &block->next_used; + block->next_used= ins->next_used; + block->prev_used= &ins->next_used; + ins->next_used= block; + if (at_end) + *ptr_ins= block; + } + else + { + /* The LRU chain is empty */ + pagecache->used_last= pagecache->used_ins= block->next_used= block; + block->prev_used= &block->next_used; + } + KEYCACHE_THREAD_TRACE("link_block"); +#if defined(PAGECACHE_DEBUG) + pagecache->blocks_available++; + KEYCACHE_DBUG_PRINT("link_block", + ("linked block %u:%1u status=%x #requests=%u #available=%u", + PCBLOCK_NUMBER(pagecache, block), at_end, block->status, + block->requests, pagecache->blocks_available)); + KEYCACHE_DBUG_ASSERT((ulong) pagecache->blocks_available <= + pagecache->blocks_used); +#endif +} + + +/* + Unlink a block from the LRU chain + + SYNOPSIS + unlink_block() + pagecache pointer to a page cache data structure + block pointer to the block to unlink from the LRU chain + + RETURN VALUE + none + + NOTES. + See NOTES for link_block +*/ + +static void unlink_block(PAGECACHE *pagecache, PAGECACHE_BLOCK_LINK *block) +{ + DBUG_ENTER("unlink_block"); + DBUG_PRINT("unlink_block", ("unlink 0x%lx", (ulong)block)); + if (block->next_used == block) + /* The list contains only one member */ + pagecache->used_last= pagecache->used_ins= NULL; + else + { + block->next_used->prev_used= block->prev_used; + *block->prev_used= block->next_used; + if (pagecache->used_last == block) + pagecache->used_last= STRUCT_PTR(PAGECACHE_BLOCK_LINK, + next_used, block->prev_used); + if (pagecache->used_ins == block) + pagecache->used_ins= STRUCT_PTR(PAGECACHE_BLOCK_LINK, + next_used, block->prev_used); + } + block->next_used= NULL; + + KEYCACHE_THREAD_TRACE("unlink_block"); +#if defined(PAGECACHE_DEBUG) + KEYCACHE_DBUG_ASSERT(pagecache->blocks_available != 0); + pagecache->blocks_available--; + KEYCACHE_DBUG_PRINT("unlink_block", + ("unlinked block 0x%lx (%u) status=%x #requests=%u #available=%u", + (ulong)block, PCBLOCK_NUMBER(pagecache, block), block->status, + block->requests, pagecache->blocks_available)); + PCBLOCK_INFO(block); +#endif + DBUG_VOID_RETURN; +} + + +/* + Register requests for a block + + SYNOPSIS + reg_requests() + pagecache this page cache reference + block the block we request reference + count how many requests we register (it is 1 everywhere) + + NOTE + Registration of request means we are going to use this block so we exclude + it from the LRU if it is first request +*/ +static void reg_requests(PAGECACHE *pagecache, PAGECACHE_BLOCK_LINK *block, + int count) +{ + DBUG_ENTER("reg_requests"); + DBUG_PRINT("enter", ("block 0x%lx (%u) status=%x, reqs: %u", + (ulong)block, PCBLOCK_NUMBER(pagecache, block), + block->status, block->requests)); + PCBLOCK_INFO(block); + if (! block->requests) + /* First request for the block unlinks it */ + unlink_block(pagecache, block); + block->requests+= count; + DBUG_VOID_RETURN; +} + + +/* + Unregister request for a block + linking it to the LRU chain if it's the last request + + SYNOPSIS + unreg_request() + pagecache pointer to a page cache data structure + block pointer to the block to link to the LRU chain + at_end <-> to link the block at the end of the LRU chain + + RETURN VALUE + none + + NOTES. + Every linking to the LRU chain decrements by one a special block + counter (if it's positive). If the at_end parameter is TRUE the block is + added either at the end of warm sub-chain or at the end of hot sub-chain. + It is added to the hot subchain if its counter is zero and number of + blocks in warm sub-chain is not less than some low limit (determined by + the division_limit parameter). Otherwise the block is added to the warm + sub-chain. If the at_end parameter is FALSE the block is always added + at beginning of the warm sub-chain. + Thus a warm block can be promoted to the hot sub-chain when its counter + becomes zero for the first time. + At the same time the block at the very beginning of the hot subchain + might be moved to the beginning of the warm subchain if it stays untouched + for a too long time (this time is determined by parameter age_threshold). +*/ + +static void unreg_request(PAGECACHE *pagecache, + PAGECACHE_BLOCK_LINK *block, int at_end) +{ + DBUG_ENTER("unreg_request"); + DBUG_PRINT("enter", ("block 0x%lx (%u) status=%x, reqs: %u", + (ulong)block, PCBLOCK_NUMBER(pagecache, block), + block->status, block->requests)); + PCBLOCK_INFO(block); + DBUG_ASSERT(block->requests > 0); + if (! --block->requests) + { + my_bool hot; + if (block->hits_left) + block->hits_left--; + hot= !block->hits_left && at_end && + pagecache->warm_blocks > pagecache->min_warm_blocks; + if (hot) + { + if (block->temperature == PCBLOCK_WARM) + pagecache->warm_blocks--; + block->temperature= PCBLOCK_HOT; + KEYCACHE_DBUG_PRINT("unreg_request", ("#warm_blocks: %lu", + pagecache->warm_blocks)); + } + link_block(pagecache, block, hot, (my_bool)at_end); + block->last_hit_time= pagecache->time; + pagecache->time++; + + block= pagecache->used_ins; + /* Check if we should link a hot block to the warm block */ + if (block && pagecache->time - block->last_hit_time > + pagecache->age_threshold) + { + unlink_block(pagecache, block); + link_block(pagecache, block, 0, 0); + if (block->temperature != PCBLOCK_WARM) + { + pagecache->warm_blocks++; + block->temperature= PCBLOCK_WARM; + } + KEYCACHE_DBUG_PRINT("unreg_request", ("#warm_blocks: %lu", + pagecache->warm_blocks)); + } + } + DBUG_VOID_RETURN; +} + +/* + Remove a reader of the page in block +*/ + +static inline void remove_reader(PAGECACHE_BLOCK_LINK *block) +{ + DBUG_ENTER("remove_reader"); + PCBLOCK_INFO(block); + DBUG_ASSERT(block->hash_link->requests > 0); +#ifdef THREAD + if (! --block->hash_link->requests && block->condvar) + pagecache_pthread_cond_signal(block->condvar); +#else + --block->hash_link->requests; +#endif + DBUG_VOID_RETURN; +} + + +/* + Wait until the last reader of the page in block + signals on its termination +*/ + +static inline void wait_for_readers(PAGECACHE *pagecache + __attribute__((unused)), + PAGECACHE_BLOCK_LINK *block) +{ +#ifdef THREAD + struct st_my_thread_var *thread= my_thread_var; + while (block->hash_link->requests) + { + KEYCACHE_DBUG_PRINT("wait_for_readers: wait", + ("suspend thread %ld block %u", + thread->id, PCBLOCK_NUMBER(pagecache, block))); + block->condvar= &thread->suspend; + pagecache_pthread_cond_wait(&thread->suspend, &pagecache->cache_lock); + block->condvar= NULL; + } +#else + KEYCACHE_DBUG_ASSERT(block->hash_link->requests == 0); +#endif +} + + +/* + Add a hash link to a bucket in the hash_table +*/ + +static inline void link_hash(PAGECACHE_HASH_LINK **start, + PAGECACHE_HASH_LINK *hash_link) +{ + if (*start) + (*start)->prev= &hash_link->next; + hash_link->next= *start; + hash_link->prev= start; + *start= hash_link; +} + + +/* + Remove a hash link from the hash table +*/ + +static void unlink_hash(PAGECACHE *pagecache, PAGECACHE_HASH_LINK *hash_link) +{ + KEYCACHE_DBUG_PRINT("unlink_hash", ("fd: %u pos_ %lu #requests=%u", + (uint) hash_link->file.file, (ulong) hash_link->pageno, + hash_link->requests)); + KEYCACHE_DBUG_ASSERT(hash_link->requests == 0); + if ((*hash_link->prev= hash_link->next)) + hash_link->next->prev= hash_link->prev; + hash_link->block= NULL; +#ifdef THREAD + if (pagecache->waiting_for_hash_link.last_thread) + { + /* Signal that a free hash link has appeared */ + struct st_my_thread_var *last_thread= + pagecache->waiting_for_hash_link.last_thread; + struct st_my_thread_var *first_thread= last_thread->next; + struct st_my_thread_var *next_thread= first_thread; + PAGECACHE_PAGE *first_page= (PAGECACHE_PAGE *) (first_thread->opt_info); + struct st_my_thread_var *thread; + + hash_link->file= first_page->file; + hash_link->pageno= first_page->pageno; + do + { + PAGECACHE_PAGE *page; + thread= next_thread; + page= (PAGECACHE_PAGE *) thread->opt_info; + next_thread= thread->next; + /* + We notify about the event all threads that ask + for the same page as the first thread in the queue + */ + if (page->file.file == hash_link->file.file && + page->pageno == hash_link->pageno) + { + KEYCACHE_DBUG_PRINT("unlink_hash: signal", ("thread %ld", thread->id)); + pagecache_pthread_cond_signal(&thread->suspend); + wqueue_unlink_from_queue(&pagecache->waiting_for_hash_link, thread); + } + } + while (thread != last_thread); + link_hash(&pagecache->hash_root[PAGECACHE_HASH(pagecache, + hash_link->file, + hash_link->pageno)], + hash_link); + return; + } +#else /* THREAD */ + KEYCACHE_DBUG_ASSERT(! (pagecache->waiting_for_hash_link.last_thread)); +#endif /* THREAD */ + hash_link->next= pagecache->free_hash_list; + pagecache->free_hash_list= hash_link; +} + + +/* + Get the hash link for the page if it is in the cache (do not put the + page in the cache if it is absent there) + + SYNOPSIS + get_present_hash_link() + pagecache Pagecache reference + file file ID + pageno page number in the file + start where to put pointer to found hash bucket (for + direct referring it) + + RETURN + found hashlink pointer +*/ + +static PAGECACHE_HASH_LINK *get_present_hash_link(PAGECACHE *pagecache, + PAGECACHE_FILE *file, + pgcache_page_no_t pageno, + PAGECACHE_HASH_LINK ***start) +{ + reg1 PAGECACHE_HASH_LINK *hash_link; +#if defined(PAGECACHE_DEBUG) + int cnt; +#endif + DBUG_ENTER("get_present_hash_link"); + + KEYCACHE_DBUG_PRINT("get_present_hash_link", ("fd: %u pos: %lu", + (uint) file->file, (ulong) pageno)); + + /* + Find the bucket in the hash table for the pair (file, pageno); + start contains the head of the bucket list, + hash_link points to the first member of the list + */ + hash_link= *(*start= &pagecache->hash_root[PAGECACHE_HASH(pagecache, + *file, pageno)]); +#if defined(PAGECACHE_DEBUG) + cnt= 0; +#endif + /* Look for an element for the pair (file, pageno) in the bucket chain */ + while (hash_link && + (hash_link->pageno != pageno || + hash_link->file.file != file->file)) + { + hash_link= hash_link->next; +#if defined(PAGECACHE_DEBUG) + cnt++; + if (! (cnt <= pagecache->hash_links_used)) + { + int i; + for (i=0, hash_link= **start ; + i < cnt ; i++, hash_link= hash_link->next) + { + KEYCACHE_DBUG_PRINT("get_present_hash_link", ("fd: %u pos: %lu", + (uint) hash_link->file.file, (ulong) hash_link->pageno)); + } + } + KEYCACHE_DBUG_ASSERT(cnt <= pagecache->hash_links_used); +#endif + } + if (hash_link) + { + /* Register the request for the page */ + hash_link->requests++; + } + + DBUG_RETURN(hash_link); +} + + +/* + Get the hash link for a page +*/ + +static PAGECACHE_HASH_LINK *get_hash_link(PAGECACHE *pagecache, + PAGECACHE_FILE *file, + pgcache_page_no_t pageno) +{ + reg1 PAGECACHE_HASH_LINK *hash_link; + PAGECACHE_HASH_LINK **start; + + KEYCACHE_DBUG_PRINT("get_hash_link", ("fd: %u pos: %lu", + (uint) file->file, (ulong) pageno)); + +restart: + /* try to find the page in the cache */ + hash_link= get_present_hash_link(pagecache, file, pageno, + &start); + if (!hash_link) + { + /* There is no hash link in the hash table for the pair (file, pageno) */ + if (pagecache->free_hash_list) + { + hash_link= pagecache->free_hash_list; + pagecache->free_hash_list= hash_link->next; + } + else if (pagecache->hash_links_used < pagecache->hash_links) + { + hash_link= &pagecache->hash_link_root[pagecache->hash_links_used++]; + } + else + { +#ifdef THREAD + /* Wait for a free hash link */ + struct st_my_thread_var *thread= my_thread_var; + PAGECACHE_PAGE page; + KEYCACHE_DBUG_PRINT("get_hash_link", ("waiting")); + page.file= *file; + page.pageno= pageno; + thread->opt_info= (void *) &page; + wqueue_link_into_queue(&pagecache->waiting_for_hash_link, thread); + KEYCACHE_DBUG_PRINT("get_hash_link: wait", + ("suspend thread %ld", thread->id)); + pagecache_pthread_cond_wait(&thread->suspend, + &pagecache->cache_lock); + thread->opt_info= NULL; +#else + KEYCACHE_DBUG_ASSERT(0); +#endif + DBUG_PRINT("info", ("restarting...")); + goto restart; + } + hash_link->file= *file; + hash_link->pageno= pageno; + link_hash(start, hash_link); + /* Register the request for the page */ + hash_link->requests++; + } + + return hash_link; +} + + +/* + Get a block for the file page requested by a pagecache read/write operation; + If the page is not in the cache return a free block, if there is none + return the lru block after saving its buffer if the page is dirty. + + SYNOPSIS + + find_block() + pagecache pointer to a page cache data structure + file handler for the file to read page from + pageno number of the page in the file + init_hits_left how initialize the block counter for the page + wrmode <-> get for writing + reg_req Register request to thye page + page_st out {PAGE_READ,PAGE_TO_BE_READ,PAGE_WAIT_TO_BE_READ} + + RETURN VALUE + Pointer to the found block if successful, 0 - otherwise + + NOTES. + For the page from file positioned at pageno the function checks whether + the page is in the key cache specified by the first parameter. + If this is the case it immediately returns the block. + If not, the function first chooses a block for this page. If there is + no not used blocks in the key cache yet, the function takes the block + at the very beginning of the warm sub-chain. It saves the page in that + block if it's dirty before returning the pointer to it. + The function returns in the page_st parameter the following values: + PAGE_READ - if page already in the block, + PAGE_TO_BE_READ - if it is to be read yet by the current thread + WAIT_TO_BE_READ - if it is to be read by another thread + If an error occurs THE PCBLOCK_ERROR bit is set in the block status. + It might happen that there are no blocks in LRU chain (in warm part) - + all blocks are unlinked for some read/write operations. Then the function + waits until first of this operations links any block back. +*/ + +static PAGECACHE_BLOCK_LINK *find_block(PAGECACHE *pagecache, + PAGECACHE_FILE *file, + pgcache_page_no_t pageno, + int init_hits_left, + my_bool wrmode, + my_bool reg_req, + int *page_st) +{ + PAGECACHE_HASH_LINK *hash_link; + PAGECACHE_BLOCK_LINK *block; + int error= 0; + int page_status; + + DBUG_ENTER("find_block"); + KEYCACHE_THREAD_TRACE("find_block:begin"); + DBUG_PRINT("enter", ("fd: %d pos: %lu wrmode: %d", + file->file, (ulong) pageno, wrmode)); + KEYCACHE_DBUG_PRINT("find_block", ("fd: %d pos: %lu wrmode: %d", + file->file, (ulong) pageno, + wrmode)); +#if !defined(DBUG_OFF) && defined(EXTRA_DEBUG) + DBUG_EXECUTE("check_pagecache", + test_key_cache(pagecache, "start of find_block", 0);); +#endif + +restart: + /* Find the hash link for the requested page (file, pageno) */ + hash_link= get_hash_link(pagecache, file, pageno); + + page_status= -1; + if ((block= hash_link->block) && + block->hash_link == hash_link && (block->status & PCBLOCK_READ)) + page_status= PAGE_READ; + + if (wrmode && pagecache->resize_in_flush) + { + /* This is a write request during the flush phase of a resize operation */ + + if (page_status != PAGE_READ) + { + /* We don't need the page in the cache: we are going to write on disk */ + DBUG_ASSERT(hash_link->requests > 0); + hash_link->requests--; + unlink_hash(pagecache, hash_link); + return 0; + } + if (!(block->status & PCBLOCK_IN_FLUSH)) + { + DBUG_ASSERT(hash_link->requests > 0); + hash_link->requests--; + /* + Remove block to invalidate the page in the block buffer + as we are going to write directly on disk. + Although we have an exclusive lock for the updated key part + the control can be yielded by the current thread as we might + have unfinished readers of other key parts in the block + buffer. Still we are guaranteed not to have any readers + of the key part we are writing into until the block is + removed from the cache as we set the PCBLOCK_REASSIGNED + flag (see the code below that handles reading requests). + */ + free_block(pagecache, block); + return 0; + } + /* Wait until the page is flushed on disk */ + DBUG_ASSERT(hash_link->requests > 0); + hash_link->requests--; + { +#ifdef THREAD + struct st_my_thread_var *thread= my_thread_var; + wqueue_add_to_queue(&block->wqueue[COND_FOR_SAVED], thread); + do + { + KEYCACHE_DBUG_PRINT("find_block: wait", + ("suspend thread %ld", thread->id)); + pagecache_pthread_cond_wait(&thread->suspend, + &pagecache->cache_lock); + } + while(thread->next); +#else + KEYCACHE_DBUG_ASSERT(0); + /* + Given the use of "resize_in_flush", it seems impossible + that this whole branch is ever entered in single-threaded case + because "(wrmode && pagecache->resize_in_flush)" cannot be true. + TODO: Check this, and then put the whole branch into the + "#ifdef THREAD" guard. + */ +#endif + } + /* Invalidate page in the block if it has not been done yet */ + if (block->status) + free_block(pagecache, block); + return 0; + } + + if (page_status == PAGE_READ && + (block->status & (PCBLOCK_IN_SWITCH | PCBLOCK_REASSIGNED))) + { + /* This is a request for a page to be removed from cache */ + + KEYCACHE_DBUG_PRINT("find_block", + ("request for old page in block %u " + "wrmode: %d block->status: %d", + PCBLOCK_NUMBER(pagecache, block), wrmode, + block->status)); + /* + Only reading requests can proceed until the old dirty page is flushed, + all others are to be suspended, then resubmitted + */ + if (!wrmode && !(block->status & PCBLOCK_REASSIGNED)) + { + if (reg_req) + reg_requests(pagecache, block, 1); + } + else + { + DBUG_ASSERT(hash_link->requests > 0); + hash_link->requests--; + KEYCACHE_DBUG_PRINT("find_block", + ("request waiting for old page to be saved")); + { +#ifdef THREAD + struct st_my_thread_var *thread= my_thread_var; + /* Put the request into the queue of those waiting for the old page */ + wqueue_add_to_queue(&block->wqueue[COND_FOR_SAVED], thread); + /* Wait until the request can be resubmitted */ + do + { + KEYCACHE_DBUG_PRINT("find_block: wait", + ("suspend thread %ld", thread->id)); + pagecache_pthread_cond_wait(&thread->suspend, + &pagecache->cache_lock); + } + while(thread->next); +#else + KEYCACHE_DBUG_ASSERT(0); + /* No parallel requests in single-threaded case */ +#endif + } + KEYCACHE_DBUG_PRINT("find_block", + ("request for old page resubmitted")); + DBUG_PRINT("info", ("restarting...")); + /* Resubmit the request */ + goto restart; + } + block->status&= ~PCBLOCK_IN_SWITCH; + } + else + { + /* This is a request for a new page or for a page not to be removed */ + if (! block) + { + /* No block is assigned for the page yet */ + if (pagecache->blocks_unused) + { + if (pagecache->free_block_list) + { + /* There is a block in the free list. */ + block= pagecache->free_block_list; + pagecache->free_block_list= block->next_used; + block->next_used= NULL; + } + else + { + /* There are some never used blocks, take first of them */ + block= &pagecache->block_root[pagecache->blocks_used]; + block->buffer= ADD_TO_PTR(pagecache->block_mem, + ((ulong) pagecache->blocks_used* + pagecache->block_size), + byte*); + pagecache->blocks_used++; + } + pagecache->blocks_unused--; + DBUG_ASSERT((block->status & PCBLOCK_WRLOCK) == 0); + DBUG_ASSERT(block->pins == 0); + block->status= 0; +#ifndef DBUG_OFF + block->type= PAGECACHE_EMPTY_PAGE; +#endif + block->requests= 1; + block->temperature= PCBLOCK_COLD; + block->hits_left= init_hits_left; + block->last_hit_time= 0; + link_to_file_list(pagecache, block, file, 0); + block->hash_link= hash_link; + hash_link->block= block; + page_status= PAGE_TO_BE_READ; + DBUG_PRINT("info", ("page to be read set for page 0x%lx", + (ulong)block)); + KEYCACHE_DBUG_PRINT("find_block", + ("got free or never used block %u", + PCBLOCK_NUMBER(pagecache, block))); + } + else + { + /* There are no never used blocks, use a block from the LRU chain */ + + /* + Wait until a new block is added to the LRU chain; + several threads might wait here for the same page, + all of them must get the same block + */ + +#ifdef THREAD + if (! pagecache->used_last) + { + struct st_my_thread_var *thread= my_thread_var; + thread->opt_info= (void *) hash_link; + wqueue_link_into_queue(&pagecache->waiting_for_block, thread); + do + { + KEYCACHE_DBUG_PRINT("find_block: wait", + ("suspend thread %ld", thread->id)); + pagecache_pthread_cond_wait(&thread->suspend, + &pagecache->cache_lock); + } + while (thread->next); + thread->opt_info= NULL; + } +#else + KEYCACHE_DBUG_ASSERT(pagecache->used_last); +#endif + block= hash_link->block; + if (! block) + { + /* + Take the first block from the LRU chain + unlinking it from the chain + */ + block= pagecache->used_last->next_used; + block->hits_left= init_hits_left; + block->last_hit_time= 0; + if (reg_req) + reg_requests(pagecache, block, 1); + hash_link->block= block; + } + PCBLOCK_INFO(block); + DBUG_ASSERT((block->status & PCBLOCK_WRLOCK) == 0); + DBUG_ASSERT(block->pins == 0); + + if (block->hash_link != hash_link && + ! (block->status & PCBLOCK_IN_SWITCH) ) + { + /* this is a primary request for a new page */ + DBUG_ASSERT((block->status & PCBLOCK_WRLOCK) == 0); + DBUG_ASSERT(block->pins == 0); + block->status|= (PCBLOCK_IN_SWITCH | PCBLOCK_WRLOCK); + + KEYCACHE_DBUG_PRINT("find_block", + ("got block %u for new page", + PCBLOCK_NUMBER(pagecache, block))); + + if (block->status & PCBLOCK_CHANGED) + { + /* The block contains a dirty page - push it out of the cache */ + + KEYCACHE_DBUG_PRINT("find_block", ("block is dirty")); + + pagecache_pthread_mutex_unlock(&pagecache->cache_lock); + /* + The call is thread safe because only the current + thread might change the block->hash_link value + */ + DBUG_ASSERT(block->pins == 0); + error= pagecache_fwrite(pagecache, + &block->hash_link->file, + block->buffer, + block->hash_link->pageno, + block->type, + MYF(MY_NABP | MY_WAIT_IF_FULL)); + pagecache_pthread_mutex_lock(&pagecache->cache_lock); + pagecache->global_cache_write++; + } + + block->status|= PCBLOCK_REASSIGNED; + if (block->hash_link) + { + /* + Wait until all pending read requests + for this page are executed + (we could have avoided this waiting, if we had read + a page in the cache in a sweep, without yielding control) + */ + wait_for_readers(pagecache, block); + + /* Remove the hash link for this page from the hash table */ + unlink_hash(pagecache, block->hash_link); + /* All pending requests for this page must be resubmitted */ +#ifdef THREAD + if (block->wqueue[COND_FOR_SAVED].last_thread) + wqueue_release_queue(&block->wqueue[COND_FOR_SAVED]); +#endif + } + link_to_file_list(pagecache, block, file, + (my_bool)(block->hash_link ? 1 : 0)); + PCBLOCK_INFO(block); + block->status= error? PCBLOCK_ERROR : 0; +#ifndef DBUG_OFF + block->type= PAGECACHE_EMPTY_PAGE; +#endif + block->hash_link= hash_link; + page_status= PAGE_TO_BE_READ; + DBUG_PRINT("info", ("page to be read set for page 0x%lx", + (ulong)block)); + + KEYCACHE_DBUG_ASSERT(block->hash_link->block == block); + KEYCACHE_DBUG_ASSERT(hash_link->block->hash_link == hash_link); + } + else + { + /* This is for secondary requests for a new page only */ + KEYCACHE_DBUG_PRINT("find_block", + ("block->hash_link: %p hash_link: %p " + "block->status: %u", block->hash_link, + hash_link, block->status )); + page_status= (((block->hash_link == hash_link) && + (block->status & PCBLOCK_READ)) ? + PAGE_READ : PAGE_WAIT_TO_BE_READ); + } + } + pagecache->global_cache_read++; + } + else + { + if (reg_req) + reg_requests(pagecache, block, 1); + KEYCACHE_DBUG_PRINT("find_block", + ("block->hash_link: %p hash_link: %p " + "block->status: %u", block->hash_link, + hash_link, block->status )); + page_status= (((block->hash_link == hash_link) && + (block->status & PCBLOCK_READ)) ? + PAGE_READ : PAGE_WAIT_TO_BE_READ); + } + } + + KEYCACHE_DBUG_ASSERT(page_status != -1); + *page_st= page_status; + DBUG_PRINT("info", + ("block: 0x%lx fd: %u pos %lu block->status %u page_status %u", + (ulong) block, (uint) file->file, + (ulong) pageno, block->status, (uint) page_status)); + KEYCACHE_DBUG_PRINT("find_block", + ("block: 0x%lx fd: %d pos: %lu block->status: %u page_status: %d", + (ulong) block, + file->file, (ulong) pageno, block->status, + page_status)); + +#if !defined(DBUG_OFF) && defined(EXTRA_DEBUG) + DBUG_EXECUTE("check_pagecache", + test_key_cache(pagecache, "end of find_block",0);); +#endif + KEYCACHE_THREAD_TRACE("find_block:end"); + DBUG_RETURN(block); +} + + +static void add_pin(PAGECACHE_BLOCK_LINK *block) +{ + DBUG_ENTER("add_pin"); + DBUG_PRINT("enter", ("block 0x%lx pins: %u", + (ulong) block, + block->pins)); + PCBLOCK_INFO(block); + block->pins++; +#ifdef PAGECACHE_DEBUG + { + PAGECACHE_PIN_INFO *info= + (PAGECACHE_PIN_INFO *)my_malloc(sizeof(PAGECACHE_PIN_INFO), MYF(0)); + info->thread= my_thread_var; + info_link(&block->pin_list, info); + } +#endif + DBUG_VOID_RETURN; +} + +static void remove_pin(PAGECACHE_BLOCK_LINK *block) +{ + DBUG_ENTER("remove_pin"); + DBUG_PRINT("enter", ("block 0x%lx pins: %u", + (ulong) block, + block->pins)); + PCBLOCK_INFO(block); + DBUG_ASSERT(block->pins > 0); + block->pins--; +#ifdef PAGECACHE_DEBUG + { + PAGECACHE_PIN_INFO *info= info_find(block->pin_list, my_thread_var); + DBUG_ASSERT(info != 0); + info_unlink(info); + my_free((gptr) info, MYF(0)); + } +#endif + DBUG_VOID_RETURN; +} +#ifdef PAGECACHE_DEBUG +static void info_add_lock(PAGECACHE_BLOCK_LINK *block, my_bool wl) +{ + PAGECACHE_LOCK_INFO *info= + (PAGECACHE_LOCK_INFO *)my_malloc(sizeof(PAGECACHE_LOCK_INFO), MYF(0)); + info->thread= my_thread_var; + info->write_lock= wl; + info_link((PAGECACHE_PIN_INFO **)&block->lock_list, + (PAGECACHE_PIN_INFO *)info); +} +static void info_remove_lock(PAGECACHE_BLOCK_LINK *block) +{ + PAGECACHE_LOCK_INFO *info= + (PAGECACHE_LOCK_INFO *)info_find((PAGECACHE_PIN_INFO *)block->lock_list, + my_thread_var); + DBUG_ASSERT(info != 0); + info_unlink((PAGECACHE_PIN_INFO *)info); + my_free((gptr)info, MYF(0)); +} +static void info_change_lock(PAGECACHE_BLOCK_LINK *block, my_bool wl) +{ + PAGECACHE_LOCK_INFO *info= + (PAGECACHE_LOCK_INFO *)info_find((PAGECACHE_PIN_INFO *)block->lock_list, + my_thread_var); + DBUG_ASSERT(info != 0 && info->write_lock != wl); + info->write_lock= wl; +} +#else +#define info_add_lock(B,W) +#define info_remove_lock(B) +#define info_change_lock(B,W) +#endif + +/* + Put on the block write lock + + SYNOPSIS + get_wrlock() + pagecache pointer to a page cache data structure + block the block to work with + + RETURN + 0 - OK + 1 - Can't lock this block, need retry +*/ + +static my_bool get_wrlock(PAGECACHE *pagecache, + PAGECACHE_BLOCK_LINK *block) +{ + PAGECACHE_FILE file= block->hash_link->file; + pgcache_page_no_t pageno= block->hash_link->pageno; + DBUG_ENTER("get_wrlock"); + DBUG_PRINT("info", ("the block 0x%lx " + "files %d(%d) pages %d(%d)", + (ulong)block, + file.file, block->hash_link->file.file, + pageno, block->hash_link->pageno)); + PCBLOCK_INFO(block); + while (block->status & PCBLOCK_WRLOCK) + { + /* Lock failed we will wait */ +#ifdef THREAD + struct st_my_thread_var *thread= my_thread_var; + DBUG_PRINT("info", ("fail to lock, waiting... 0x%lx", (ulong)block)); + wqueue_add_to_queue(&block->wqueue[COND_FOR_WRLOCK], thread); + dec_counter_for_resize_op(pagecache); + do + { + KEYCACHE_DBUG_PRINT("get_wrlock: wait", + ("suspend thread %ld", thread->id)); + pagecache_pthread_cond_wait(&thread->suspend, + &pagecache->cache_lock); + } + while(thread->next); +#else + DBUG_ASSERT(0); +#endif + PCBLOCK_INFO(block); + if ((block->status & (PCBLOCK_REASSIGNED | PCBLOCK_IN_SWITCH)) || + file.file != block->hash_link->file.file || + pageno != block->hash_link->pageno) + { + DBUG_PRINT("info", ("the block 0x%lx changed => need retry" + "status %x files %d != %d or pages %d !=%d", + (ulong)block, block->status, + file.file, block->hash_link->file.file, + pageno, block->hash_link->pageno)); + DBUG_RETURN(1); + } + } + DBUG_ASSERT(block->pins == 0); + /* we are doing it by global cache mutex protection, so it is OK */ + block->status|= PCBLOCK_WRLOCK; + DBUG_PRINT("info", ("WR lock set, block 0x%lx", (ulong)block)); + DBUG_RETURN(0); +} + + +/* + Remove write lock from the block + + SYNOPSIS + release_wrlock() + pagecache pointer to a page cache data structure + block the block to work with + + RETURN + 0 - OK +*/ + +static void release_wrlock(PAGECACHE_BLOCK_LINK *block) +{ + DBUG_ENTER("release_wrlock"); + PCBLOCK_INFO(block); + DBUG_ASSERT(block->status & PCBLOCK_WRLOCK); + DBUG_ASSERT(block->pins > 0); + block->status&= ~PCBLOCK_WRLOCK; + DBUG_PRINT("info", ("WR lock reset, block 0x%lx", (ulong)block)); +#ifdef THREAD + /* release all threads waiting for write lock */ + if (block->wqueue[COND_FOR_WRLOCK].last_thread) + wqueue_release_queue(&block->wqueue[COND_FOR_WRLOCK]); +#endif + PCBLOCK_INFO(block); + DBUG_VOID_RETURN; +} + + +/* + Try to lock/unlock and pin/unpin the block + + SYNOPSIS + make_lock_and_pin() + pagecache pointer to a page cache data structure + block the block to work with + lock lock change mode + pin pinchange mode + + RETURN + 0 - OK + 1 - Try to lock the block failed +*/ + +static my_bool make_lock_and_pin(PAGECACHE *pagecache, + PAGECACHE_BLOCK_LINK *block, + enum pagecache_page_lock lock, + enum pagecache_page_pin pin) +{ + DBUG_ENTER("make_lock_and_pin"); + DBUG_PRINT("enter", ("block: 0x%lx (%u), wrlock: %c pins: %u, lock %s, pin: %s", + (ulong)block, PCBLOCK_NUMBER(pagecache, block), + ((block->status & PCBLOCK_WRLOCK)?'Y':'N'), + block->pins, + page_cache_page_lock_str[lock], + page_cache_page_pin_str[pin])); + PCBLOCK_INFO(block); +#ifdef PAGECACHE_DEBUG + DBUG_ASSERT(info_check_pin(block, pin) == 0 && + info_check_lock(block, lock, pin) == 0); +#endif + switch (lock) + { + case PAGECACHE_LOCK_WRITE: /* free -> write */ + /* Writelock and pin the buffer */ + if (get_wrlock(pagecache, block)) + { + /* can't lock => need retry */ + goto retry; + } + + /* The cache is locked so nothing afraid of */ + add_pin(block); + info_add_lock(block, 1); + break; + case PAGECACHE_LOCK_WRITE_TO_READ: /* write -> read */ + case PAGECACHE_LOCK_WRITE_UNLOCK: /* write -> free */ + /* + Removes write lock and puts read lock (which is nothing in our + implementation) + */ + release_wrlock(block); + case PAGECACHE_LOCK_READ_UNLOCK: /* read -> free */ + case PAGECACHE_LOCK_LEFT_READLOCKED: /* read -> read */ + if (pin == PAGECACHE_UNPIN) + { + remove_pin(block); + } + if (lock == PAGECACHE_LOCK_WRITE_TO_READ) + { + info_change_lock(block, 0); + } + else if (lock == PAGECACHE_LOCK_WRITE_UNLOCK || + lock == PAGECACHE_LOCK_READ_UNLOCK) + { + info_remove_lock(block); + } + break; + case PAGECACHE_LOCK_READ: /* free -> read */ + if (pin == PAGECACHE_PIN) + { + /* The cache is locked so nothing afraid off */ + add_pin(block); + } + info_add_lock(block, 0); + break; + case PAGECACHE_LOCK_LEFT_UNLOCKED: /* free -> free */ + case PAGECACHE_LOCK_LEFT_WRITELOCKED: /* write -> write */ + break; /* do nothing */ + default: + DBUG_ASSERT(0); /* Never should happened */ + } + + PCBLOCK_INFO(block); + DBUG_RETURN(0); +retry: + DBUG_PRINT("INFO", ("Retry block 0x%lx", (ulong)block)); + PCBLOCK_INFO(block); + DBUG_ASSERT(block->hash_link->requests > 0); + block->hash_link->requests--; + DBUG_ASSERT(block->requests > 0); + unreg_request(pagecache, block, 1); + PCBLOCK_INFO(block); + DBUG_RETURN(1); + +} + + +/* + Read into a key cache block buffer from disk. + + SYNOPSIS + + read_block() + pagecache pointer to a page cache data structure + block block to which buffer the data is to be read + primary <-> the current thread will read the data + validator validator of read from the disk data + validator_data pointer to the data need by the validator + + RETURN VALUE + None + + NOTES. + The function either reads a page data from file to the block buffer, + or waits until another thread reads it. What page to read is determined + by a block parameter - reference to a hash link for this page. + If an error occurs THE PCBLOCK_ERROR bit is set in the block status. +*/ + +static void read_block(PAGECACHE *pagecache, + PAGECACHE_BLOCK_LINK *block, + my_bool primary, + pagecache_disk_read_validator validator, + gptr validator_data) +{ + uint got_length; + + /* On entry cache_lock is locked */ + + DBUG_ENTER("read_block"); + if (primary) + { + /* + This code is executed only by threads + that submitted primary requests + */ + + DBUG_PRINT("read_block", + ("page to be read by primary request")); + + /* Page is not in buffer yet, is to be read from disk */ + pagecache_pthread_mutex_unlock(&pagecache->cache_lock); + /* + Here other threads may step in and register as secondary readers. + They will register in block->wqueue[COND_FOR_REQUESTED]. + */ + got_length= pagecache_fread(pagecache, &block->hash_link->file, + block->buffer, + block->hash_link->pageno, MYF(0)); + pagecache_pthread_mutex_lock(&pagecache->cache_lock); + if (got_length < pagecache->block_size) + block->status|= PCBLOCK_ERROR; + else + block->status= (PCBLOCK_READ | (block->status & PCBLOCK_WRLOCK)); + + if (validator != NULL && + (*validator)(block->buffer, validator_data)) + block->status|= PCBLOCK_ERROR; + + DBUG_PRINT("read_block", + ("primary request: new page in cache")); + /* Signal that all pending requests for this page now can be processed */ +#ifdef THREAD + if (block->wqueue[COND_FOR_REQUESTED].last_thread) + wqueue_release_queue(&block->wqueue[COND_FOR_REQUESTED]); +#endif + } + else + { + /* + This code is executed only by threads + that submitted secondary requests + */ + DBUG_PRINT("read_block", + ("secondary request waiting for new page to be read")); + { +#ifdef THREAD + struct st_my_thread_var *thread= my_thread_var; + /* Put the request into a queue and wait until it can be processed */ + wqueue_add_to_queue(&block->wqueue[COND_FOR_REQUESTED], thread); + do + { + DBUG_PRINT("read_block: wait", + ("suspend thread %ld", thread->id)); + pagecache_pthread_cond_wait(&thread->suspend, + &pagecache->cache_lock); + } + while (thread->next); +#else + KEYCACHE_DBUG_ASSERT(0); + /* No parallel requests in single-threaded case */ +#endif + } + DBUG_PRINT("read_block", + ("secondary request: new page in cache")); + } + DBUG_VOID_RETURN; +} + + +/* + Unlock/unpin page and put LSN stamp if it need + + SYNOPSIS + pagecache_unlock_page() + pagecache pointer to a page cache data structure + file handler for the file for the block of data to be read + pageno number of the block of data in the file + lock lock change + pin pin page + first_REDO_LSN_for_page do not set it if it is zero + + NOTE + Pininig uses requests registration mechanism it works following way: + | beginnig | ending | + | of func. | of func. | + ----------------------------+-------------+---------------+ + PAGECACHE_PIN_LEFT_PINNED | - | - | + PAGECACHE_PIN_LEFT_UNPINNED | reg request | unreg request | + PAGECACHE_PIN | reg request | - | + PAGECACHE_UNPIN | - | unreg request | + + +*/ + +void pagecache_unlock_page(PAGECACHE *pagecache, + PAGECACHE_FILE *file, + pgcache_page_no_t pageno, + enum pagecache_page_lock lock, + enum pagecache_page_pin pin, + LSN first_REDO_LSN_for_page) +{ + PAGECACHE_BLOCK_LINK *block; + int page_st; + DBUG_ENTER("pagecache_unlock_page"); + DBUG_PRINT("enter", ("fd: %u page: %lu l%s p%s", + (uint) file->file, (ulong) pageno, + page_cache_page_lock_str[lock], + page_cache_page_pin_str[pin])); + /* we do not allow any lock/pin increasing here */ + DBUG_ASSERT(pin != PAGECACHE_PIN && + lock != PAGECACHE_LOCK_READ && + lock != PAGECACHE_LOCK_WRITE); + + pagecache_pthread_mutex_lock(&pagecache->cache_lock); + /* + As soon as we keep lock cache can be used, and we have lock because want + to unlock. + */ + DBUG_ASSERT(pagecache->can_be_used); + + inc_counter_for_resize_op(pagecache); + /* See NOTE for pagecache_unlock_page about registering requests */ + block= find_block(pagecache, file, pageno, 0, 0, + test(pin == PAGECACHE_PIN_LEFT_UNPINNED), &page_st); + PCBLOCK_INFO(block); + DBUG_ASSERT(block != 0 && page_st == PAGE_READ); + if (first_REDO_LSN_for_page) + { + DBUG_ASSERT(lock == PAGECACHE_LOCK_WRITE_UNLOCK && + pin == PAGECACHE_UNPIN); + set_if_bigger(block->rec_lsn, first_REDO_LSN_for_page); + } + +#ifndef DBUG_OFF + if ( +#endif + make_lock_and_pin(pagecache, block, lock, pin) +#ifndef DBUG_OFF + ) + { + DBUG_ASSERT(0); /* should not happend */ + } +#else + ; +#endif + + remove_reader(block); + /* + Link the block into the LRU chain if it's the last submitted request + for the block and block will not be pinned. + See NOTE for pagecache_unlock_page about registering requests. + */ + if (pin != PAGECACHE_PIN_LEFT_PINNED) + unreg_request(pagecache, block, 1); + + dec_counter_for_resize_op(pagecache); + + pagecache_pthread_mutex_unlock(&pagecache->cache_lock); + + DBUG_VOID_RETURN; +} + + +/* + Unpin page + + SYNOPSIS + pagecache_unpin_page() + pagecache pointer to a page cache data structure + file handler for the file for the block of data to be read + pageno number of the block of data in the file +*/ + +void pagecache_unpin_page(PAGECACHE *pagecache, + PAGECACHE_FILE *file, + pgcache_page_no_t pageno) +{ + PAGECACHE_BLOCK_LINK *block; + int page_st; + DBUG_ENTER("pagecache_unpin_page"); + DBUG_PRINT("enter", ("fd: %u page: %lu", + (uint) file->file, (ulong) pageno)); + pagecache_pthread_mutex_lock(&pagecache->cache_lock); + /* + As soon as we keep lock cache can be used, and we have lock bacause want + aunlock. + */ + DBUG_ASSERT(pagecache->can_be_used); + + inc_counter_for_resize_op(pagecache); + /* See NOTE for pagecache_unlock_page about registering requests */ + block= find_block(pagecache, file, pageno, 0, 0, 0, &page_st); + DBUG_ASSERT(block != 0 && page_st == PAGE_READ); + +#ifndef DBUG_OFF + if ( +#endif + /* + we can just unpin only with keeping read lock because: + a) we can't pin without any lock + b) we can't unpin keeping write lock + */ + make_lock_and_pin(pagecache, block, + PAGECACHE_LOCK_LEFT_READLOCKED, + PAGECACHE_UNPIN) +#ifndef DBUG_OFF + ) + { + DBUG_ASSERT(0); /* should not happend */ + } +#else + ; +#endif + + remove_reader(block); + /* + Link the block into the LRU chain if it's the last submitted request + for the block and block will not be pinned. + See NOTE for pagecache_unlock_page about registering requests + */ + unreg_request(pagecache, block, 1); + + dec_counter_for_resize_op(pagecache); + + pagecache_pthread_mutex_unlock(&pagecache->cache_lock); + + DBUG_VOID_RETURN; +} + + +/* + Unlock/unpin page and put LSN stamp if it need + (uses direct block/page pointer) + + SYNOPSIS + pagecache_unlock() + pagecache pointer to a page cache data structure + link direct link to page (returned by read or write) + lock lock change + pin pin page + first_REDO_LSN_for_page do not set it if it is zero +*/ + +void pagecache_unlock(PAGECACHE *pagecache, + PAGECACHE_PAGE_LINK *link, + enum pagecache_page_lock lock, + enum pagecache_page_pin pin, + LSN first_REDO_LSN_for_page) +{ + PAGECACHE_BLOCK_LINK *block= (PAGECACHE_BLOCK_LINK *)link; + DBUG_ENTER("pagecache_unlock"); + DBUG_PRINT("enter", ("block: 0x%lx fd: %u page: %lu l%s p%s", + (ulong) block, + (uint) block->hash_link->file.file, + (ulong) block->hash_link->pageno, + page_cache_page_lock_str[lock], + page_cache_page_pin_str[pin])); + /* + We do not allow any lock/pin increasing here and page can't be + unpinned because we use direct link. + */ + DBUG_ASSERT(pin != PAGECACHE_PIN && + pin != PAGECACHE_PIN_LEFT_UNPINNED && + lock != PAGECACHE_LOCK_READ && + lock != PAGECACHE_LOCK_WRITE); + if (pin == PAGECACHE_PIN_LEFT_UNPINNED && + lock == PAGECACHE_LOCK_READ_UNLOCK) + { +#ifndef DBUG_OFF + if ( +#endif + /* block do not need here so we do not provide it */ + make_lock_and_pin(pagecache, 0, lock, pin) +#ifndef DBUG_OFF + ) + { + DBUG_ASSERT(0); /* should not happend */ + } +#else + ; +#endif + DBUG_VOID_RETURN; + } + + pagecache_pthread_mutex_lock(&pagecache->cache_lock); + /* + As soon as we keep lock cache can be used, and we have lock bacause want + aunlock. + */ + DBUG_ASSERT(pagecache->can_be_used); + + inc_counter_for_resize_op(pagecache); + if (first_REDO_LSN_for_page) + { + DBUG_ASSERT(lock == PAGECACHE_LOCK_WRITE_UNLOCK && + pin == PAGECACHE_UNPIN); + set_if_bigger(block->rec_lsn, first_REDO_LSN_for_page); + } + +#ifndef DBUG_OFF + if ( +#endif + make_lock_and_pin(pagecache, block, lock, pin) +#ifndef DBUG_OFF + ) + { + DBUG_ASSERT(0); /* should not happend */ + } +#else + ; +#endif + + remove_reader(block); + /* + Link the block into the LRU chain if it's the last submitted request + for the block and block will not be pinned. + See NOTE for pagecache_unlock_page about registering requests. + */ + if (pin != PAGECACHE_PIN_LEFT_PINNED) + unreg_request(pagecache, block, 1); + + dec_counter_for_resize_op(pagecache); + + pagecache_pthread_mutex_unlock(&pagecache->cache_lock); + + DBUG_VOID_RETURN; +} + + +/* + Unpin page + (uses direct block/page pointer) + + SYNOPSIS + pagecache_unpin_page() + pagecache pointer to a page cache data structure + link direct link to page (returned by read or write) +*/ + +void pagecache_unpin(PAGECACHE *pagecache, + PAGECACHE_PAGE_LINK *link) +{ + PAGECACHE_BLOCK_LINK *block= (PAGECACHE_BLOCK_LINK *)link; + DBUG_ENTER("pagecache_unpin"); + DBUG_PRINT("enter", ("block: 0x%lx fd: %u page: %lu", + (ulong) block, + (uint) block->hash_link->file.file, + (ulong) block->hash_link->pageno)); + + pagecache_pthread_mutex_lock(&pagecache->cache_lock); + /* + As soon as we keep lock cache can be used, and we have lock bacause want + aunlock. + */ + DBUG_ASSERT(pagecache->can_be_used); + + inc_counter_for_resize_op(pagecache); + +#ifndef DBUG_OFF + if ( +#endif + /* + we can just unpin only with keeping read lock because: + a) we can't pin without any lock + b) we can't unpin keeping write lock + */ + make_lock_and_pin(pagecache, block, + PAGECACHE_LOCK_LEFT_READLOCKED, + PAGECACHE_UNPIN) +#ifndef DBUG_OFF + ) + { + DBUG_ASSERT(0); /* should not happend */ + } +#else + ; +#endif + + remove_reader(block); + /* + Link the block into the LRU chain if it's the last submitted request + for the block and block will not be pinned. + See NOTE for pagecache_unlock_page about registering requests. + */ + unreg_request(pagecache, block, 1); + + dec_counter_for_resize_op(pagecache); + + pagecache_pthread_mutex_unlock(&pagecache->cache_lock); + + DBUG_VOID_RETURN; +} + + +/* + Read a block of data from a cached file into a buffer; + + SYNOPSIS + pagecache_valid_read() + pagecache pointer to a page cache data structure + file handler for the file for the block of data to be read + pageno number of the block of data in the file + level determines the weight of the data + buff buffer to where the data must be placed + type type of the page + lock lock change + link link to the page if we pin it + validator validator of read from the disk data + validator_data pointer to the data need by the validator + + RETURN VALUE + Returns address from where the data is placed if sucessful, 0 - otherwise. + + Pin will be choosen according to lock parameter (see lock_to_pin) +*/ +static enum pagecache_page_pin lock_to_pin[]= +{ + PAGECACHE_PIN_LEFT_UNPINNED /*PAGECACHE_LOCK_LEFT_UNLOCKED*/, + PAGECACHE_PIN_LEFT_UNPINNED /*PAGECACHE_LOCK_LEFT_READLOCKED*/, + PAGECACHE_PIN_LEFT_PINNED /*PAGECACHE_LOCK_LEFT_WRITELOCKED*/, + PAGECACHE_PIN_LEFT_UNPINNED /*PAGECACHE_LOCK_READ*/, + PAGECACHE_PIN /*PAGECACHE_LOCK_WRITE*/, + PAGECACHE_PIN_LEFT_UNPINNED /*PAGECACHE_LOCK_READ_UNLOCK*/, + PAGECACHE_UNPIN /*PAGECACHE_LOCK_WRITE_UNLOCK*/, + PAGECACHE_UNPIN /*PAGECACHE_LOCK_WRITE_TO_READ*/ +}; + +byte *pagecache_valid_read(PAGECACHE *pagecache, + PAGECACHE_FILE *file, + pgcache_page_no_t pageno, + uint level, + byte *buff, + enum pagecache_page_type type, + enum pagecache_page_lock lock, + PAGECACHE_PAGE_LINK *link, + pagecache_disk_read_validator validator, + gptr validator_data) +{ + int error= 0; + enum pagecache_page_pin pin= lock_to_pin[lock]; + PAGECACHE_PAGE_LINK fake_link; + DBUG_ENTER("pagecache_valid_read"); + DBUG_PRINT("enter", ("fd: %u page: %lu level: %u t:%s l%s p%s", + (uint) file->file, (ulong) pageno, level, + page_cache_page_type_str[type], + page_cache_page_lock_str[lock], + page_cache_page_pin_str[pin])); + + if (!link) + link= &fake_link; + else + *link= 0; + +restart: + + if (pagecache->can_be_used) + { + /* Key cache is used */ + PAGECACHE_BLOCK_LINK *block; + uint status; + int page_st; + + pagecache_pthread_mutex_lock(&pagecache->cache_lock); + if (!pagecache->can_be_used) + { + pagecache_pthread_mutex_unlock(&pagecache->cache_lock); + goto no_key_cache; + } + + inc_counter_for_resize_op(pagecache); + pagecache->global_cache_r_requests++; + /* See NOTE for pagecache_unlock_page about registering requests. */ + block= find_block(pagecache, file, pageno, level, + test(lock == PAGECACHE_LOCK_WRITE), + test((pin == PAGECACHE_PIN_LEFT_UNPINNED) || + (pin == PAGECACHE_PIN)), + &page_st); + DBUG_ASSERT(block->type == PAGECACHE_EMPTY_PAGE || + block->type == type); + block->type= type; + if (block->status != PCBLOCK_ERROR && page_st != PAGE_READ) + { + DBUG_PRINT("info", ("read block 0x%lx", (ulong)block)); + /* The requested page is to be read into the block buffer */ + read_block(pagecache, block, + (my_bool)(page_st == PAGE_TO_BE_READ), + validator, validator_data); + DBUG_PRINT("info", ("read is done")); + } + if (make_lock_and_pin(pagecache, block, lock, pin)) + { + /* + We failed to write lock the block, cache is unlocked, + we will try to get the block again. + */ + pagecache_pthread_mutex_unlock(&pagecache->cache_lock); + DBUG_PRINT("info", ("restarting...")); + goto restart; + } + + if (! ((status= block->status) & PCBLOCK_ERROR)) + { +#if !defined(SERIALIZED_READ_FROM_CACHE) + pagecache_pthread_mutex_unlock(&pagecache->cache_lock); +#endif + + DBUG_ASSERT((pagecache->block_size & 511) == 0); + /* Copy data from the cache buffer */ + bmove512(buff, block->buffer, pagecache->block_size); + +#if !defined(SERIALIZED_READ_FROM_CACHE) + pagecache_pthread_mutex_lock(&pagecache->cache_lock); +#endif + } + + remove_reader(block); + /* + Link the block into the LRU chain if it's the last submitted request + for the block and block will not be pinned. + See NOTE for pagecache_unlock_page about registering requests. + */ + if (pin == PAGECACHE_PIN_LEFT_UNPINNED || pin == PAGECACHE_UNPIN) + unreg_request(pagecache, block, 1); + else + *link= (PAGECACHE_PAGE_LINK)block; + + dec_counter_for_resize_op(pagecache); + + pagecache_pthread_mutex_unlock(&pagecache->cache_lock); + + if (status & PCBLOCK_ERROR) + DBUG_RETURN((byte *) 0); + + DBUG_RETURN(buff); + } + +no_key_cache: /* Key cache is not used */ + + /* We can't use mutex here as the key cache may not be initialized */ + pagecache->global_cache_r_requests++; + pagecache->global_cache_read++; + if (pagecache_fread(pagecache, file, (byte*) buff, pageno, MYF(MY_NABP))) + error= 1; + DBUG_RETURN(error ? (byte*) 0 : buff); +} + + +/* + Delete page from the buffer + + SYNOPSIS + pagecache_delete_page() + pagecache pointer to a page cache data structure + file handler for the file for the block of data to be read + pageno number of the block of data in the file + lock lock change + flush flush page if it is dirty + + RETURN VALUE + 0 - deleted or was not present at all + 1 - error + + NOTES. + lock can be only PAGECACHE_LOCK_LEFT_WRITELOCKED (page was write locked + before) or PAGECACHE_LOCK_WRITE (delete will write lock page before delete) +*/ +my_bool pagecache_delete_page(PAGECACHE *pagecache, + PAGECACHE_FILE *file, + pgcache_page_no_t pageno, + enum pagecache_page_lock lock, + my_bool flush) +{ + int error= 0; + enum pagecache_page_pin pin= lock_to_pin[lock]; + DBUG_ENTER("pagecache_delete_page"); + DBUG_PRINT("enter", ("fd: %u page: %lu l%s p%s", + (uint) file->file, (ulong) pageno, + page_cache_page_lock_str[lock], + page_cache_page_pin_str[pin])); + DBUG_ASSERT(lock == PAGECACHE_LOCK_WRITE || + lock == PAGECACHE_LOCK_LEFT_WRITELOCKED); + DBUG_ASSERT(pin == PAGECACHE_PIN || + pin == PAGECACHE_PIN_LEFT_PINNED); + +restart: + + if (pagecache->can_be_used) + { + /* Key cache is used */ + reg1 PAGECACHE_BLOCK_LINK *block; + PAGECACHE_HASH_LINK **unused_start, *link; + + pagecache_pthread_mutex_lock(&pagecache->cache_lock); + if (!pagecache->can_be_used) + goto end; + + inc_counter_for_resize_op(pagecache); + link= get_present_hash_link(pagecache, file, pageno, &unused_start); + if (!link) + { + DBUG_PRINT("info", ("There is no such page in the cache")); + pagecache_pthread_mutex_unlock(&pagecache->cache_lock); + DBUG_RETURN(0); + } + block= link->block; + /* See NOTE for pagecache_unlock_page about registering requests. */ + if (pin == PAGECACHE_PIN) + reg_requests(pagecache, block, 1); + DBUG_ASSERT(block != 0); + if (make_lock_and_pin(pagecache, block, lock, pin)) + { + /* + We failed to writelock the block, cache is unlocked, and last write + lock is released, we will try to get the block again. + */ + pagecache_pthread_mutex_unlock(&pagecache->cache_lock); + DBUG_PRINT("info", ("restarting...")); + goto restart; + } + + if (block->status & PCBLOCK_CHANGED) + { + if (flush) + { + /* The block contains a dirty page - push it out of the cache */ + + KEYCACHE_DBUG_PRINT("find_block", ("block is dirty")); + + pagecache_pthread_mutex_unlock(&pagecache->cache_lock); + /* + The call is thread safe because only the current + thread might change the block->hash_link value + */ + DBUG_ASSERT(block->pins == 1); + error= pagecache_fwrite(pagecache, + &block->hash_link->file, + block->buffer, + block->hash_link->pageno, + block->type, + MYF(MY_NABP | MY_WAIT_IF_FULL)); + pagecache_pthread_mutex_lock(&pagecache->cache_lock); + pagecache->global_cache_write++; + + if (error) + { + block->status|= PCBLOCK_ERROR; + goto err; + } + } + pagecache->blocks_changed--; + pagecache->global_blocks_changed--; + /* + free_block() will change the status and rec_lsn of the block so no + need to change them here. + */ + } + /* Cache is locked, so we can relese page before freeing it */ + make_lock_and_pin(pagecache, block, + PAGECACHE_LOCK_WRITE_UNLOCK, + PAGECACHE_UNPIN); + DBUG_ASSERT(link->requests > 0); + link->requests--; + /* See NOTE for pagecache_unlock_page about registering requests. */ + free_block(pagecache, block); + +err: + dec_counter_for_resize_op(pagecache); +end: + pagecache_pthread_mutex_unlock(&pagecache->cache_lock); + } + + DBUG_RETURN(error); +} + + +/* + Write a buffer into a cached file. + + SYNOPSIS + + pagecache_write() + pagecache pointer to a page cache data structure + file handler for the file to write data to + pageno number of the block of data in the file + level determines the weight of the data + buff buffer to where the data must be placed + type type of the page + lock lock change + pin pin page + write_mode how to write page + link link to the page if we pin it + + RETURN VALUE + 0 if a success, 1 - otherwise. +*/ + +/* description of how to change lock before and after write */ +struct write_lock_change +{ + int need_lock_change; /* need changing of lock at the end of write */ + enum pagecache_page_lock new_lock; /* lock at the beginning */ + enum pagecache_page_lock unlock_lock; /* lock at the end */ +}; + +static struct write_lock_change write_lock_change_table[]= +{ + {1, + PAGECACHE_LOCK_WRITE, + PAGECACHE_LOCK_WRITE_UNLOCK} /*PAGECACHE_LOCK_LEFT_UNLOCKED*/, + {0, /*unsupported*/ + PAGECACHE_LOCK_LEFT_UNLOCKED, + PAGECACHE_LOCK_LEFT_UNLOCKED} /*PAGECACHE_LOCK_LEFT_READLOCKED*/, + {0, PAGECACHE_LOCK_LEFT_WRITELOCKED, 0} /*PAGECACHE_LOCK_LEFT_WRITELOCKED*/, + {1, + PAGECACHE_LOCK_WRITE, + PAGECACHE_LOCK_WRITE_TO_READ} /*PAGECACHE_LOCK_READ*/, + {0, PAGECACHE_LOCK_WRITE, 0} /*PAGECACHE_LOCK_WRITE*/, + {0, /*unsupported*/ + PAGECACHE_LOCK_LEFT_UNLOCKED, + PAGECACHE_LOCK_LEFT_UNLOCKED} /*PAGECACHE_LOCK_READ_UNLOCK*/, + {1, + PAGECACHE_LOCK_LEFT_WRITELOCKED, + PAGECACHE_LOCK_WRITE_UNLOCK } /*PAGECACHE_LOCK_WRITE_UNLOCK*/, + {1, + PAGECACHE_LOCK_LEFT_WRITELOCKED, + PAGECACHE_LOCK_WRITE_TO_READ}/*PAGECACHE_LOCK_WRITE_TO_READ*/ +}; + +/* description of how to change pin before and after write */ +struct write_pin_change +{ + enum pagecache_page_pin new_pin; /* pin status at the beginning */ + enum pagecache_page_pin unlock_pin; /* pin status at the end */ +}; + +static struct write_pin_change write_pin_change_table[]= +{ + {PAGECACHE_PIN_LEFT_PINNED, + PAGECACHE_PIN_LEFT_PINNED} /*PAGECACHE_PIN_LEFT_PINNED*/, + {PAGECACHE_PIN, + PAGECACHE_UNPIN} /*PAGECACHE_PIN_LEFT_UNPINNED*/, + {PAGECACHE_PIN, + PAGECACHE_PIN_LEFT_PINNED} /*PAGECACHE_PIN*/, + {PAGECACHE_PIN_LEFT_PINNED, + PAGECACHE_UNPIN} /*PAGECACHE_UNPIN*/ +}; + +my_bool pagecache_write_part(PAGECACHE *pagecache, + PAGECACHE_FILE *file, + pgcache_page_no_t pageno, + uint level, + byte *buff, + enum pagecache_page_type type, + enum pagecache_page_lock lock, + enum pagecache_page_pin pin, + enum pagecache_write_mode write_mode, + PAGECACHE_PAGE_LINK *link, + uint offset, uint size) +{ + PAGECACHE_BLOCK_LINK *block= NULL; + PAGECACHE_PAGE_LINK fake_link; + int error= 0; + int need_lock_change= write_lock_change_table[lock].need_lock_change; + DBUG_ENTER("pagecache_write_part"); + DBUG_PRINT("enter", ("fd: %u page: %lu level: %u type: %s lock: %s " + "pin: %s mode: %s offset: %u size %u", + (uint) file->file, (ulong) pageno, level, + page_cache_page_type_str[type], + page_cache_page_lock_str[lock], + page_cache_page_pin_str[pin], + page_cache_page_write_mode_str[write_mode], + offset, size)); + DBUG_ASSERT(lock != PAGECACHE_LOCK_LEFT_READLOCKED && + lock != PAGECACHE_LOCK_READ_UNLOCK); + DBUG_ASSERT(offset + size <= pagecache->block_size); + if (!link) + link= &fake_link; + else + *link= 0; + +restart: + +#if !defined(DBUG_OFF) && defined(EXTRA_DEBUG) + DBUG_EXECUTE("check_pagecache", + test_key_cache(pagecache, "start of key_cache_write", 1);); +#endif + + if (pagecache->can_be_used) + { + /* Key cache is used */ + int page_st; + + pagecache_pthread_mutex_lock(&pagecache->cache_lock); + if (!pagecache->can_be_used) + { + pagecache_pthread_mutex_unlock(&pagecache->cache_lock); + goto no_key_cache; + } + + inc_counter_for_resize_op(pagecache); + pagecache->global_cache_w_requests++; + /* See NOTE for pagecache_unlock_page about registering requests. */ + block= find_block(pagecache, file, pageno, level, + test(write_mode != PAGECACHE_WRITE_DONE && + lock != PAGECACHE_LOCK_LEFT_WRITELOCKED && + lock != PAGECACHE_LOCK_WRITE_UNLOCK && + lock != PAGECACHE_LOCK_WRITE_TO_READ), + test((pin == PAGECACHE_PIN_LEFT_UNPINNED) || + (pin == PAGECACHE_PIN)), + &page_st); + if (!block) + { + DBUG_ASSERT(write_mode != PAGECACHE_WRITE_DONE); + /* It happens only for requests submitted during resize operation */ + dec_counter_for_resize_op(pagecache); + pagecache_pthread_mutex_unlock(&pagecache->cache_lock); + /* Write to the disk key cache is in resize at the moment*/ + goto no_key_cache; + } + + DBUG_ASSERT(block->type == PAGECACHE_EMPTY_PAGE || + block->type == type); + block->type= type; + + if (make_lock_and_pin(pagecache, block, + write_lock_change_table[lock].new_lock, + (need_lock_change ? + write_pin_change_table[pin].new_pin : + pin))) + { + /* + We failed to writelock the block, cache is unlocked, and last write + lock is released, we will try to get the block again. + */ + pagecache_pthread_mutex_unlock(&pagecache->cache_lock); + DBUG_PRINT("info", ("restarting...")); + goto restart; + } + + + if (write_mode == PAGECACHE_WRITE_DONE) + { + if ((block->status & PCBLOCK_ERROR) && page_st != PAGE_READ) + { + /* Copy data from buff */ + if (!(size & 511)) + bmove512(block->buffer + offset, buff, size); + else + memcpy(block->buffer + offset, buff, size); + block->status= (PCBLOCK_READ | (block->status & PCBLOCK_WRLOCK)); + KEYCACHE_DBUG_PRINT("key_cache_insert", + ("primary request: new page in cache")); +#ifdef THREAD + /* Signal that all pending requests for this now can be processed. */ + if (block->wqueue[COND_FOR_REQUESTED].last_thread) + wqueue_release_queue(&block->wqueue[COND_FOR_REQUESTED]); +#endif + } + } + else + { + if (! (block->status & PCBLOCK_CHANGED)) + link_to_changed_list(pagecache, block); + + if (! (block->status & PCBLOCK_ERROR)) + { + if (!(size & 511)) + bmove512(block->buffer + offset, buff, size); + else + memcpy(block->buffer + offset, buff, size); + block->status|= PCBLOCK_READ; + } + } + + + if (need_lock_change) + { +#ifndef DBUG_OFF + int rc= +#endif + /* + QQ: We are doing an unlock here, so need to give the page its rec_lsn + */ + make_lock_and_pin(pagecache, block, + write_lock_change_table[lock].unlock_lock, + write_pin_change_table[pin].unlock_pin); +#ifndef DBUG_OFF + DBUG_ASSERT(rc == 0); +#endif + } + + /* Unregister the request */ + DBUG_ASSERT(block->hash_link->requests > 0); + block->hash_link->requests--; + /* See NOTE for pagecache_unlock_page about registering requests. */ + if (pin == PAGECACHE_PIN_LEFT_UNPINNED || pin == PAGECACHE_UNPIN) + unreg_request(pagecache, block, 1); + else + *link= (PAGECACHE_PAGE_LINK)block; + + + if (block->status & PCBLOCK_ERROR) + error= 1; + + dec_counter_for_resize_op(pagecache); + + pagecache_pthread_mutex_unlock(&pagecache->cache_lock); + + goto end; + } + +no_key_cache: + /* Key cache is not used */ + if (write_mode == PAGECACHE_WRITE_DELAY) + { + pagecache->global_cache_w_requests++; + pagecache->global_cache_write++; + if (pagecache_fwrite(pagecache, file, (byte*) buff, pageno, type, + MYF(MY_NABP | MY_WAIT_IF_FULL))) + error=1; + } + +end: +#if !defined(DBUG_OFF) && defined(EXTRA_DEBUG) + DBUG_EXECUTE("exec", + test_key_cache(pagecache, "end of key_cache_write", 1);); +#endif + if (block) + PCBLOCK_INFO(block); + else + DBUG_PRINT("info", ("No block")); + DBUG_RETURN(error); +} + + +/* + Free block: remove reference to it from hash table, + remove it from the chain file of dirty/clean blocks + and add it to the free list. +*/ + +static void free_block(PAGECACHE *pagecache, PAGECACHE_BLOCK_LINK *block) +{ + KEYCACHE_THREAD_TRACE("free block"); + KEYCACHE_DBUG_PRINT("free_block", + ("block %u to be freed, hash_link %p", + PCBLOCK_NUMBER(pagecache, block), block->hash_link)); + if (block->hash_link) + { + /* + While waiting for readers to finish, new readers might request the + block. But since we set block->status|= PCBLOCK_REASSIGNED, they + will wait on block->wqueue[COND_FOR_SAVED]. They must be signalled + later. + */ + block->status|= PCBLOCK_REASSIGNED; + wait_for_readers(pagecache, block); + unlink_hash(pagecache, block->hash_link); + } + + unlink_changed(block); + DBUG_ASSERT((block->status & PCBLOCK_WRLOCK) == 0); + DBUG_ASSERT(block->pins == 0); + block->status= 0; +#ifndef DBUG_OFF + block->type= PAGECACHE_EMPTY_PAGE; +#endif + block->rec_lsn= 0; + KEYCACHE_THREAD_TRACE("free block"); + KEYCACHE_DBUG_PRINT("free_block", + ("block is freed")); + unreg_request(pagecache, block, 0); + block->hash_link= NULL; + + /* Remove the free block from the LRU ring. */ + unlink_block(pagecache, block); + if (block->temperature == PCBLOCK_WARM) + pagecache->warm_blocks--; + block->temperature= PCBLOCK_COLD; + /* Insert the free block in the free list. */ + block->next_used= pagecache->free_block_list; + pagecache->free_block_list= block; + /* Keep track of the number of currently unused blocks. */ + pagecache->blocks_unused++; + +#ifdef THREAD + /* All pending requests for this page must be resubmitted. */ + if (block->wqueue[COND_FOR_SAVED].last_thread) + wqueue_release_queue(&block->wqueue[COND_FOR_SAVED]); +#endif +} + + +static int cmp_sec_link(PAGECACHE_BLOCK_LINK **a, PAGECACHE_BLOCK_LINK **b) +{ + return (((*a)->hash_link->pageno < (*b)->hash_link->pageno) ? -1 : + ((*a)->hash_link->pageno > (*b)->hash_link->pageno) ? 1 : 0); +} + + +/* + Flush a portion of changed blocks to disk, + free used blocks if requested +*/ + +static int flush_cached_blocks(PAGECACHE *pagecache, + PAGECACHE_FILE *file, + PAGECACHE_BLOCK_LINK **cache, + PAGECACHE_BLOCK_LINK **end, + enum flush_type type) +{ + int error; + int last_errno= 0; + uint count= (uint) (end-cache); + DBUG_ENTER("flush_cached_blocks"); + + /* Don't lock the cache during the flush */ + pagecache_pthread_mutex_unlock(&pagecache->cache_lock); + /* + As all blocks referred in 'cache' are marked by PCBLOCK_IN_FLUSH + we are guarunteed no thread will change them + */ + qsort((byte*) cache, count, sizeof(*cache), (qsort_cmp) cmp_sec_link); + + pagecache_pthread_mutex_lock(&pagecache->cache_lock); + for (; cache != end; cache++) + { + PAGECACHE_BLOCK_LINK *block= *cache; + + if (block->pins) + { + KEYCACHE_DBUG_PRINT("flush_cached_blocks", + ("block %u (0x%lx) pinned", + PCBLOCK_NUMBER(pagecache, block), (ulong)block)); + DBUG_PRINT("info", ("block %u (0x%lx) pinned", + PCBLOCK_NUMBER(pagecache, block), (ulong)block)); + PCBLOCK_INFO(block); + last_errno= -1; + unreg_request(pagecache, block, 1); + continue; + } + /* if the block is not pinned then it is not write locked */ + DBUG_ASSERT((block->status & PCBLOCK_WRLOCK) == 0); + DBUG_ASSERT(block->pins == 0); +#ifndef DBUG_OFF + { + int rc= +#endif + make_lock_and_pin(pagecache, block, + PAGECACHE_LOCK_WRITE, PAGECACHE_PIN); +#ifndef DBUG_OFF + DBUG_ASSERT(rc == 0); + } +#endif + + KEYCACHE_DBUG_PRINT("flush_cached_blocks", + ("block %u (0x%lx) to be flushed", + PCBLOCK_NUMBER(pagecache, block), (ulong)block)); + DBUG_PRINT("info", ("block %u (0x%lx) to be flushed", + PCBLOCK_NUMBER(pagecache, block), (ulong)block)); + PCBLOCK_INFO(block); + pagecache_pthread_mutex_unlock(&pagecache->cache_lock); + DBUG_PRINT("info", ("block %u (0x%lx) pins: %u", + PCBLOCK_NUMBER(pagecache, block), (ulong)block, + block->pins)); + DBUG_ASSERT(block->pins == 1); + error= pagecache_fwrite(pagecache, file, + block->buffer, + block->hash_link->pageno, + block->type, + MYF(MY_NABP | MY_WAIT_IF_FULL)); + pagecache_pthread_mutex_lock(&pagecache->cache_lock); + + make_lock_and_pin(pagecache, block, + PAGECACHE_LOCK_WRITE_UNLOCK, + PAGECACHE_UNPIN); + + pagecache->global_cache_write++; + if (error) + { + block->status|= PCBLOCK_ERROR; + if (!last_errno) + last_errno= errno ? errno : -1; + } +#ifdef THREAD + /* + Let to proceed for possible waiting requests to write to the block page. + It might happen only during an operation to resize the key cache. + */ + if (block->wqueue[COND_FOR_SAVED].last_thread) + wqueue_release_queue(&block->wqueue[COND_FOR_SAVED]); +#endif + /* type will never be FLUSH_IGNORE_CHANGED here */ + if (! (type == FLUSH_KEEP || type == FLUSH_FORCE_WRITE)) + { + pagecache->blocks_changed--; + pagecache->global_blocks_changed--; + free_block(pagecache, block); + } + else + { + block->status&= ~PCBLOCK_IN_FLUSH; + link_to_file_list(pagecache, block, file, 1); + unreg_request(pagecache, block, 1); + } + } + DBUG_RETURN(last_errno); +} + + +/* + flush all key blocks for a file to disk, but don't do any mutex locks + + flush_pagecache_blocks_int() + pagecache pointer to a key cache data structure + file handler for the file to flush to + flush_type type of the flush + + NOTES + This function doesn't do any mutex locks because it needs to be called + both from flush_pagecache_blocks and flush_all_key_blocks (the later one + does the mutex lock in the resize_pagecache() function). + + RETURN + 0 ok + 1 error +*/ + +static int flush_pagecache_blocks_int(PAGECACHE *pagecache, + PAGECACHE_FILE *file, + enum flush_type type) +{ + PAGECACHE_BLOCK_LINK *cache_buff[FLUSH_CACHE],**cache; + int last_errno= 0; + DBUG_ENTER("flush_pagecache_blocks_int"); + DBUG_PRINT("enter",("file: %d blocks_used: %lu blocks_changed: %lu", + file->file, pagecache->blocks_used, pagecache->blocks_changed)); + +#if !defined(DBUG_OFF) && defined(EXTRA_DEBUG) + DBUG_EXECUTE("check_pagecache", + test_key_cache(pagecache, + "start of flush_pagecache_blocks", 0);); +#endif + + cache= cache_buff; + if (pagecache->disk_blocks > 0 && + (!my_disable_flush_pagecache_blocks || type != FLUSH_KEEP)) + { + /* Key cache exists and flush is not disabled */ + int error= 0; + uint count= 0; + PAGECACHE_BLOCK_LINK **pos, **end; + PAGECACHE_BLOCK_LINK *first_in_switch= NULL; + PAGECACHE_BLOCK_LINK *block, *next; +#if defined(PAGECACHE_DEBUG) + uint cnt= 0; +#endif + + if (type != FLUSH_IGNORE_CHANGED) + { + /* + Count how many key blocks we have to cache to be able + to flush all dirty pages with minimum seek moves + */ + for (block= pagecache->changed_blocks[FILE_HASH(*file)] ; + block; + block= block->next_changed) + { + if (block->hash_link->file.file == file->file) + { + count++; + KEYCACHE_DBUG_ASSERT(count<= pagecache->blocks_used); + } + } + /* Allocate a new buffer only if its bigger than the one we have */ + if (count > FLUSH_CACHE && + !(cache= + (PAGECACHE_BLOCK_LINK**) + my_malloc(sizeof(PAGECACHE_BLOCK_LINK*)*count, MYF(0)))) + { + cache= cache_buff; + count= FLUSH_CACHE; + } + } + + /* Retrieve the blocks and write them to a buffer to be flushed */ +restart: + end= (pos= cache)+count; + for (block= pagecache->changed_blocks[FILE_HASH(*file)] ; + block; + block= next) + { +#if defined(PAGECACHE_DEBUG) + cnt++; + KEYCACHE_DBUG_ASSERT(cnt <= pagecache->blocks_used); +#endif + next= block->next_changed; + if (block->hash_link->file.file == file->file) + { + /* + Mark the block with BLOCK_IN_FLUSH in order not to let + other threads to use it for new pages and interfere with + our sequence ot flushing dirty file pages + */ + block->status|= PCBLOCK_IN_FLUSH; + + if (! (block->status & PCBLOCK_IN_SWITCH)) + { + /* + We care only for the blocks for which flushing was not + initiated by other threads as a result of page swapping + */ + reg_requests(pagecache, block, 1); + if (type != FLUSH_IGNORE_CHANGED) + { + /* It's not a temporary file */ + if (pos == end) + { + /* + This happens only if there is not enough + memory for the big block + */ + if ((error= flush_cached_blocks(pagecache, file, cache, + end,type))) + last_errno=error; + DBUG_PRINT("info", ("restarting...")); + /* + Restart the scan as some other thread might have changed + the changed blocks chain: the blocks that were in switch + state before the flush started have to be excluded + */ + goto restart; + } + *pos++= block; + } + else + { + /* It's a temporary file */ + pagecache->blocks_changed--; + pagecache->global_blocks_changed--; + free_block(pagecache, block); + } + } + else + { + /* Link the block into a list of blocks 'in switch' */ + /* QQ: + #warning this unlink_changed() is a serious problem for + Maria's Checkpoint: it removes a page from the list of dirty + pages, while it's still dirty. A solution is to abandon + first_in_switch, just wait for this page to be + flushed by somebody else, and loop. TODO: check all places + where we remove a page from the list of dirty pages + */ + unlink_changed(block); + link_changed(block, &first_in_switch); + } + } + } + if (pos != cache) + { + if ((error= flush_cached_blocks(pagecache, file, cache, pos, type))) + last_errno= error; + } + /* Wait until list of blocks in switch is empty */ + while (first_in_switch) + { +#if defined(PAGECACHE_DEBUG) + cnt= 0; +#endif + block= first_in_switch; + { +#ifdef THREAD + struct st_my_thread_var *thread= my_thread_var; + wqueue_add_to_queue(&block->wqueue[COND_FOR_SAVED], thread); + do + { + KEYCACHE_DBUG_PRINT("flush_pagecache_blocks_int: wait", + ("suspend thread %ld", thread->id)); + pagecache_pthread_cond_wait(&thread->suspend, + &pagecache->cache_lock); + } + while (thread->next); +#else + KEYCACHE_DBUG_ASSERT(0); + /* No parallel requests in single-threaded case */ +#endif + } +#if defined(PAGECACHE_DEBUG) + cnt++; + KEYCACHE_DBUG_ASSERT(cnt <= pagecache->blocks_used); +#endif + } + /* The following happens very seldom */ + if (! (type == FLUSH_KEEP || type == FLUSH_FORCE_WRITE)) + { +#if defined(PAGECACHE_DEBUG) + cnt=0; +#endif + for (block= pagecache->file_blocks[FILE_HASH(*file)] ; + block; + block= next) + { +#if defined(PAGECACHE_DEBUG) + cnt++; + KEYCACHE_DBUG_ASSERT(cnt <= pagecache->blocks_used); +#endif + next= block->next_changed; + if (block->hash_link->file.file == file->file && + (! (block->status & PCBLOCK_CHANGED) + || type == FLUSH_IGNORE_CHANGED)) + { + reg_requests(pagecache, block, 1); + free_block(pagecache, block); + } + } + } + } + +#ifndef DBUG_OFF + DBUG_EXECUTE("check_pagecache", + test_key_cache(pagecache, "end of flush_pagecache_blocks", 0);); +#endif + if (cache != cache_buff) + my_free((gptr) cache, MYF(0)); + if (last_errno) + errno=last_errno; /* Return first error */ + DBUG_RETURN(last_errno != 0); +} + + +/* + Flush all blocks for a file to disk + + SYNOPSIS + + flush_pagecache_blocks() + pagecache pointer to a page cache data structure + file handler for the file to flush to + flush_type type of the flush + + RETURN + 0 ok + 1 error +*/ + +int flush_pagecache_blocks(PAGECACHE *pagecache, + PAGECACHE_FILE *file, enum flush_type type) +{ + int res; + DBUG_ENTER("flush_pagecache_blocks"); + DBUG_PRINT("enter", ("pagecache: 0x%lx", (long) pagecache)); + + if (pagecache->disk_blocks <= 0) + DBUG_RETURN(0); + pagecache_pthread_mutex_lock(&pagecache->cache_lock); + inc_counter_for_resize_op(pagecache); + res= flush_pagecache_blocks_int(pagecache, file, type); + dec_counter_for_resize_op(pagecache); + pagecache_pthread_mutex_unlock(&pagecache->cache_lock); + DBUG_RETURN(res); +} + + +/* + Reset the counters of a key cache. + + SYNOPSIS + reset_pagecache_counters() + name the name of a key cache + pagecache pointer to the pagecache to be reset + + DESCRIPTION + This procedure is used to reset the counters of all currently used key + caches, both the default one and the named ones. + + RETURN + 0 on success (always because it can't fail) +*/ + +int reset_pagecache_counters(const char *name, PAGECACHE *pagecache) +{ + DBUG_ENTER("reset_pagecache_counters"); + if (!pagecache->inited) + { + DBUG_PRINT("info", ("Key cache %s not initialized.", name)); + DBUG_RETURN(0); + } + DBUG_PRINT("info", ("Resetting counters for key cache %s.", name)); + + pagecache->global_blocks_changed= 0; /* Key_blocks_not_flushed */ + pagecache->global_cache_r_requests= 0; /* Key_read_requests */ + pagecache->global_cache_read= 0; /* Key_reads */ + pagecache->global_cache_w_requests= 0; /* Key_write_requests */ + pagecache->global_cache_write= 0; /* Key_writes */ + DBUG_RETURN(0); +} + + +/* + Allocates a buffer and stores in it some information about all dirty pages + of type PAGECACHE_LSN_PAGE. + + SYNOPSIS + pagecache_collect_changed_blocks_with_lsn() + pagecache pointer to the page cache + str (OUT) pointer to a LEX_STRING where the allocated buffer, and + its size, will be put + max_lsn (OUT) pointer to a LSN where the maximum rec_lsn of all + relevant dirty pages will be put + + DESCRIPTION + Does the allocation because the caller cannot know the size itself. + Memory freeing is to be done by the caller (if the "str" member of the + LEX_STRING is not NULL). + Ignores all pages of another type than PAGECACHE_LSN_PAGE, because they + are not interesting for a checkpoint record. + The caller has the intention of doing checkpoints. + + RETURN + 0 on success + 1 on error +*/ +my_bool pagecache_collect_changed_blocks_with_lsn(PAGECACHE *pagecache, + LEX_STRING *str, + LSN *max_lsn) +{ + my_bool error; + ulong stored_list_size= 0; + uint file_hash; + char *ptr; + DBUG_ENTER("pagecache_collect_changed_blocks_with_LSN"); + + *max_lsn= 0; + DBUG_ASSERT(NULL == str->str); + /* + We lock the entire cache but will be quick, just reading/writing a few MBs + of memory at most. + When we enter here, we must be sure that no "first_in_switch" situation + is happening or will happen (either we have to get rid of + first_in_switch in the code or, first_in_switch has to increment a + "danger" counter for this function to know it has to wait). TODO. + */ + pagecache_pthread_mutex_lock(&pagecache->cache_lock); + + /* Count how many dirty pages are interesting */ + for (file_hash= 0; file_hash < PAGECACHE_CHANGED_BLOCKS_HASH; file_hash++) + { + PAGECACHE_BLOCK_LINK *block; + for (block= pagecache->changed_blocks[file_hash] ; + block; + block= block->next_changed) + { + /* + Q: is there somthing subtle with block->hash_link: can it be NULL? + does it have to be == hash_link->block... ? + */ + DBUG_ASSERT(block->hash_link != NULL); + DBUG_ASSERT(block->status & PCBLOCK_CHANGED); + if (block->type != PAGECACHE_LSN_PAGE) + continue; /* no need to store it */ + /* + In the current pagecache, rec_lsn is not set correctly: + 1) it is set on pagecache_unlock(), too late (a page is dirty + (PCBLOCK_CHANGED) since the first pagecache_write()). So in this + scenario: + thread1: thread2: + write_REDO + pagecache_write() checkpoint : reclsn not known + pagecache_unlock(sets rec_lsn) + commit + crash, + at recovery we will wrongly skip the REDO. It also affects the + low-water mark's computation. + 2) sometimes the unlocking can be an implicit action of + pagecache_write(), without any call to pagecache_unlock(), then + rec_lsn is not set. + 1) and 2) are critical problems. + TODO: fix this when Monty has explained how he writes BLOB pages. + */ + if (block->rec_lsn == 0) + { + DBUG_ASSERT(0); + goto err; + } + stored_list_size++; + } + } + + str->length= 8+(4+4+8)*stored_list_size; + if (NULL == (str->str= my_malloc(str->length, MYF(MY_WME)))) + goto err; + ptr= str->str; + int8store(ptr, stored_list_size); + ptr+= 8; + if (0 == stored_list_size) + goto end; + for (file_hash= 0; file_hash < PAGECACHE_CHANGED_BLOCKS_HASH; file_hash++) + { + PAGECACHE_BLOCK_LINK *block; + for (block= pagecache->changed_blocks[file_hash] ; + block; + block= block->next_changed) + { + if (block->type != PAGECACHE_LSN_PAGE) + continue; /* no need to store it in the checkpoint record */ + DBUG_ASSERT((4 == sizeof(block->hash_link->file.file)) && + (4 == sizeof(block->hash_link->pageno))); + int4store(ptr, block->hash_link->file.file); + ptr+= 4; + int4store(ptr, block->hash_link->pageno); + ptr+= 4; + int8store(ptr, (ulonglong) block->rec_lsn); + ptr+= 8; + set_if_bigger(*max_lsn, block->rec_lsn); + } + } + error= 0; + goto end; +err: + error= 1; +end: + pagecache_pthread_mutex_unlock(&pagecache->cache_lock); + DBUG_RETURN(error); +} + + +#ifndef DBUG_OFF +/* + Test if disk-cache is ok +*/ +static void test_key_cache(PAGECACHE *pagecache __attribute__((unused)), + const char *where __attribute__((unused)), + my_bool lock __attribute__((unused))) +{ + /* TODO */ +} +#endif + +#if defined(PAGECACHE_TIMEOUT) + +#define KEYCACHE_DUMP_FILE "pagecache_dump.txt" +#define MAX_QUEUE_LEN 100 + + +static void pagecache_dump(PAGECACHE *pagecache) +{ + FILE *pagecache_dump_file=fopen(KEYCACHE_DUMP_FILE, "w"); + struct st_my_thread_var *last; + struct st_my_thread_var *thread; + PAGECACHE_BLOCK_LINK *block; + PAGECACHE_HASH_LINK *hash_link; + PAGECACHE_PAGE *page; + uint i; + + fprintf(pagecache_dump_file, "thread:%u\n", thread->id); + + i=0; + thread=last=waiting_for_hash_link.last_thread; + fprintf(pagecache_dump_file, "queue of threads waiting for hash link\n"); + if (thread) + do + { + thread= thread->next; + page= (PAGECACHE_PAGE *) thread->opt_info; + fprintf(pagecache_dump_file, + "thread:%u, (file,pageno)=(%u,%lu)\n", + thread->id,(uint) page->file.file,(ulong) page->pageno); + if (++i == MAX_QUEUE_LEN) + break; + } + while (thread != last); + + i=0; + thread=last=waiting_for_block.last_thread; + fprintf(pagecache_dump_file, "queue of threads waiting for block\n"); + if (thread) + do + { + thread=thread->next; + hash_link= (PAGECACHE_HASH_LINK *) thread->opt_info; + fprintf(pagecache_dump_file, + "thread:%u hash_link:%u (file,pageno)=(%u,%lu)\n", + thread->id, (uint) PAGECACHE_HASH_LINK_NUMBER(pagecache, hash_link), + (uint) hash_link->file.file,(ulong) hash_link->pageno); + if (++i == MAX_QUEUE_LEN) + break; + } + while (thread != last); + + for (i=0 ; i < pagecache->blocks_used ; i++) + { + int j; + block= &pagecache->block_root[i]; + hash_link= block->hash_link; + fprintf(pagecache_dump_file, + "block:%u hash_link:%d status:%x #requests=%u waiting_for_readers:%d\n", + i, (int) (hash_link ? + PAGECACHE_HASH_LINK_NUMBER(pagecache, hash_link) : + -1), + block->status, block->requests, block->condvar ? 1 : 0); + for (j=0 ; j < COND_SIZE; j++) + { + PAGECACHE_WQUEUE *wqueue=&block->wqueue[j]; + thread= last= wqueue->last_thread; + fprintf(pagecache_dump_file, "queue #%d\n", j); + if (thread) + { + do + { + thread=thread->next; + fprintf(pagecache_dump_file, + "thread:%u\n", thread->id); + if (++i == MAX_QUEUE_LEN) + break; + } + while (thread != last); + } + } + } + fprintf(pagecache_dump_file, "LRU chain:"); + block= pagecache= used_last; + if (block) + { + do + { + block= block->next_used; + fprintf(pagecache_dump_file, + "block:%u, ", PCBLOCK_NUMBER(pagecache, block)); + } + while (block != pagecache->used_last); + } + fprintf(pagecache_dump_file, "\n"); + + fclose(pagecache_dump_file); +} + +#endif /* defined(PAGECACHE_TIMEOUT) */ + +#if defined(PAGECACHE_TIMEOUT) && !defined(__WIN__) + + +static int pagecache_pthread_cond_wait(pthread_cond_t *cond, + pthread_mutex_t *mutex) +{ + int rc; + struct timeval now; /* time when we started waiting */ + struct timespec timeout; /* timeout value for the wait function */ + struct timezone tz; +#if defined(PAGECACHE_DEBUG) + int cnt=0; +#endif + + /* Get current time */ + gettimeofday(&now, &tz); + /* Prepare timeout value */ + timeout.tv_sec= now.tv_sec + PAGECACHE_TIMEOUT; + /* + timeval uses microseconds. + timespec uses nanoseconds. + 1 nanosecond = 1000 micro seconds + */ + timeout.tv_nsec= now.tv_usec * 1000; + KEYCACHE_THREAD_TRACE_END("started waiting"); +#if defined(PAGECACHE_DEBUG) + cnt++; + if (cnt % 100 == 0) + fprintf(pagecache_debug_log, "waiting...\n"); + fflush(pagecache_debug_log); +#endif + rc= pthread_cond_timedwait(cond, mutex, &timeout); + KEYCACHE_THREAD_TRACE_BEGIN("finished waiting"); + if (rc == ETIMEDOUT || rc == ETIME) + { +#if defined(PAGECACHE_DEBUG) + fprintf(pagecache_debug_log,"aborted by pagecache timeout\n"); + fclose(pagecache_debug_log); + abort(); +#endif + pagecache_dump(); + } + +#if defined(PAGECACHE_DEBUG) + KEYCACHE_DBUG_ASSERT(rc != ETIMEDOUT); +#else + assert(rc != ETIMEDOUT); +#endif + return rc; +} +#else +#if defined(PAGECACHE_DEBUG) +static int pagecache_pthread_cond_wait(pthread_cond_t *cond, + pthread_mutex_t *mutex) +{ + int rc; + KEYCACHE_THREAD_TRACE_END("started waiting"); + rc= pthread_cond_wait(cond, mutex); + KEYCACHE_THREAD_TRACE_BEGIN("finished waiting"); + return rc; +} +#endif +#endif /* defined(PAGECACHE_TIMEOUT) && !defined(__WIN__) */ + +#if defined(PAGECACHE_DEBUG) +static int ___pagecache_pthread_mutex_lock(pthread_mutex_t *mutex) +{ + int rc; + rc= pthread_mutex_lock(mutex); + KEYCACHE_THREAD_TRACE_BEGIN(""); + return rc; +} + + +static void ___pagecache_pthread_mutex_unlock(pthread_mutex_t *mutex) +{ + KEYCACHE_THREAD_TRACE_END(""); + pthread_mutex_unlock(mutex); +} + + +static int ___pagecache_pthread_cond_signal(pthread_cond_t *cond) +{ + int rc; + KEYCACHE_THREAD_TRACE("signal"); + rc= pthread_cond_signal(cond); + return rc; +} + + +#if defined(PAGECACHE_DEBUG_LOG) + + +static void pagecache_debug_print(const char * fmt, ...) +{ + va_list args; + va_start(args,fmt); + if (pagecache_debug_log) + { + VOID(vfprintf(pagecache_debug_log, fmt, args)); + VOID(fputc('\n',pagecache_debug_log)); + } + va_end(args); +} +#endif /* defined(PAGECACHE_DEBUG_LOG) */ + +#if defined(PAGECACHE_DEBUG_LOG) + + +void pagecache_debug_log_close(void) +{ + if (pagecache_debug_log) + fclose(pagecache_debug_log); +} +#endif /* defined(PAGECACHE_DEBUG_LOG) */ + +#endif /* defined(PAGECACHE_DEBUG) */ diff --git a/storage/maria/ma_pagecaches.c b/storage/maria/ma_pagecaches.c new file mode 100644 index 00000000000..e635709c11e --- /dev/null +++ b/storage/maria/ma_pagecaches.c @@ -0,0 +1,105 @@ +/* Copyright (C) 2003-2007 MySQL AB + + 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; either version 2 of the License, or + (at your option) any later version. + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ + +/* + Handling of multiple key caches + + The idea is to have a thread safe hash on the table name, + with a default key cache value that is returned if the table name is not in + the cache. +*/ + +#include "maria_def.h" +#include +#include +#include +#include "../../mysys/my_safehash.h" + +/***************************************************************************** + Functions to handle the pagecache objects +*****************************************************************************/ + +/* Variable to store all key cache objects */ +static SAFE_HASH pagecache_hash; + + +my_bool multi_pagecache_init(void) +{ + return safe_hash_init(&pagecache_hash, 16, (byte*) dflt_pagecache); +} + + +void multi_pagecache_free(void) +{ + safe_hash_free(&pagecache_hash); +} + +/* + Get a key cache to be used for a specific table. + + SYNOPSIS + multi_pagecache_search() + key key to find (usually table path) + uint length Length of key. + def Default value if no key cache + + NOTES + This function is coded in such a way that we will return the + default key cache even if one never called multi_pagecache_init. + This will ensure that it works with old MyISAM clients. + + RETURN + key cache to use +*/ + +PAGECACHE *multi_pagecache_search(byte *key, uint length, + PAGECACHE *def) +{ + if (!pagecache_hash.hash.records) + return def; + return (PAGECACHE*) safe_hash_search(&pagecache_hash, key, length, + (void*) def); +} + + +/* + Assosiate a key cache with a key + + + SYONOPSIS + multi_pagecache_set() + key key (path to table etc..) + length Length of key + pagecache cache to assococite with the table + + NOTES + This can be used both to insert a new entry and change an existing + entry +*/ + + +my_bool multi_pagecache_set(const byte *key, uint length, + PAGECACHE *pagecache) +{ + return safe_hash_set(&pagecache_hash, key, length, (byte*) pagecache); +} + + +void multi_pagecache_change(PAGECACHE *old_data, + PAGECACHE *new_data) +{ + safe_hash_change(&pagecache_hash, (byte*) old_data, (byte*) new_data); +} diff --git a/storage/maria/ma_panic.c b/storage/maria/ma_panic.c index c1312cb1e77..204c3538011 100644 --- a/storage/maria/ma_panic.c +++ b/storage/maria/ma_panic.c @@ -63,7 +63,8 @@ int maria_panic(enum ha_panic_function flag) if (info->s->options & HA_OPTION_READ_ONLY_DATA) break; #endif - if (flush_key_blocks(info->s->key_cache, info->s->kfile, FLUSH_RELEASE)) + if (flush_pagecache_blocks(info->s->pagecache, &info->s->kfile, + FLUSH_RELEASE)) error=my_errno; if (info->opt_flag & WRITE_CACHE_USED) if (flush_io_cache(&info->rec_cache)) @@ -82,29 +83,32 @@ int maria_panic(enum ha_panic_function flag) error=my_errno; } #ifdef CANT_OPEN_FILES_TWICE - if (info->s->kfile >= 0 && my_close(info->s->kfile,MYF(0))) + if (info->s->kfile.file >= 0 && my_close(info->s->kfile.file, MYF(0))) error = my_errno; - if (info->dfile >= 0 && my_close(info->dfile,MYF(0))) + if (info->dfile.file >= 0 && my_close(info->dfile.file, MYF(0))) error = my_errno; - info->s->kfile=info->dfile= -1; /* Files aren't open anymore */ + info->s->kfile.file= info->dfile.file= -1;/* Files aren't open anymore */ break; #endif case HA_PANIC_READ: /* Restore to before WRITE */ #ifdef CANT_OPEN_FILES_TWICE { /* Open closed files */ char name_buff[FN_REFLEN]; - if (info->s->kfile < 0) - if ((info->s->kfile= my_open(fn_format(name_buff,info->filename,"", - N_NAME_IEXT,4),info->mode, - MYF(MY_WME))) < 0) + if (info->s->kfile.file < 0) + if ((info->s->kfile.file= my_open(fn_format(name_buff, + info->filename, "", + N_NAME_IEXT,4), + info->mode, + MYF(MY_WME))) < 0) error = my_errno; - if (info->dfile < 0) + if (info->dfile.file < 0) { - if ((info->dfile= my_open(fn_format(name_buff,info->filename,"", - N_NAME_DEXT,4),info->mode, - MYF(MY_WME))) < 0) + if ((info->dfile.file= my_open(fn_format(name_buff, info->filename, + "", N_NAME_DEXT, 4), + info->mode, + MYF(MY_WME))) < 0) error = my_errno; - info->rec_cache.file=info->dfile; + info->rec_cache.file= info->dfile.file; } } #endif diff --git a/storage/maria/ma_preload.c b/storage/maria/ma_preload.c index f387f2b7de3..5dcfcb35129 100644 --- a/storage/maria/ma_preload.c +++ b/storage/maria/ma_preload.c @@ -69,7 +69,7 @@ int maria_preload(MARIA_HA *info, ulonglong key_map, my_bool ignore_leaves) if (!(buff= (uchar *) my_malloc(length, MYF(MY_WME)))) DBUG_RETURN(my_errno= HA_ERR_OUT_OF_MEM); - if (flush_key_blocks(share->key_cache,share->kfile, FLUSH_RELEASE)) + if (flush_pagecache_blocks(share->pagecache, &share->kfile, FLUSH_RELEASE)) goto err; do @@ -77,7 +77,8 @@ int maria_preload(MARIA_HA *info, ulonglong key_map, my_bool ignore_leaves) /* Read the next block of index file into the preload buffer */ if ((my_off_t) length > (key_file_length-pos)) length= (ulong) (key_file_length-pos); - if (my_pread(share->kfile, (byte*) buff, length, pos, MYF(MY_FAE|MY_FNABP))) + if (my_pread(share->kfile.file, (byte*) buff, length, pos, + MYF(MY_FAE|MY_FNABP))) goto err; if (ignore_leaves) @@ -87,9 +88,15 @@ int maria_preload(MARIA_HA *info, ulonglong key_map, my_bool ignore_leaves) { if (_ma_test_if_nod(buff)) { - if (key_cache_insert(share->key_cache, - share->kfile, pos, DFLT_INIT_HITS, - (byte*) buff, block_length)) + DBUG_ASSERT(share->pagecache->block_size == block_length); + if (pagecache_write(share->pagecache, + &share->kfile, pos / block_length, + DFLT_INIT_HITS, + (byte*) buff, + PAGECACHE_PLAIN_PAGE, + PAGECACHE_LOCK_LEFT_UNLOCKED, + PAGECACHE_PIN_LEFT_PINNED, + PAGECACHE_WRITE_DONE, 0)) goto err; } pos+= block_length; @@ -99,9 +106,14 @@ int maria_preload(MARIA_HA *info, ulonglong key_map, my_bool ignore_leaves) } else { - if (key_cache_insert(share->key_cache, - share->kfile, pos, DFLT_INIT_HITS, - (byte*) buff, length)) + if (pagecache_write(share->pagecache, + &share->kfile, pos / block_length, + DFLT_INIT_HITS, + (byte*) buff, + PAGECACHE_PLAIN_PAGE, + PAGECACHE_LOCK_LEFT_UNLOCKED, + PAGECACHE_PIN_LEFT_PINNED, + PAGECACHE_WRITE_DONE, 0)) goto err; pos+= length; } diff --git a/storage/maria/ma_static.c b/storage/maria/ma_static.c index c5580e1e981..7d5c61d05c8 100644 --- a/storage/maria/ma_static.c +++ b/storage/maria/ma_static.c @@ -41,8 +41,8 @@ my_off_t maria_max_temp_length= MAX_FILE_SIZE; ulong maria_bulk_insert_tree_size=8192*1024; ulong maria_data_pointer_size= 4; -KEY_CACHE maria_key_cache_var; -KEY_CACHE *maria_key_cache= &maria_key_cache_var; +PAGECACHE maria_pagecache_var; +PAGECACHE *maria_pagecache= &maria_pagecache_var; /* Enough for comparing if number is zero */ byte maria_zero_string[]= {0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0}; diff --git a/storage/maria/ma_test1.c b/storage/maria/ma_test1.c index 0f37391c1d4..7f536f71c80 100644 --- a/storage/maria/ma_test1.c +++ b/storage/maria/ma_test1.c @@ -33,7 +33,7 @@ static enum data_file_type record_type= DYNAMIC_RECORD; static uint insert_count, update_count, remove_count; static uint pack_keys=0, pack_seg=0, key_length; static uint unique_key=HA_NOSAME; -static my_bool key_cacheing, null_fields, silent, skip_update, opt_unique, +static my_bool pagecacheing, null_fields, silent, skip_update, opt_unique, verbose, skip_delete; static MARIA_COLUMNDEF recinfo[4]; static MARIA_KEYDEF keyinfo[10]; @@ -51,9 +51,9 @@ int main(int argc,char *argv[]) MY_INIT(argv[0]); my_init(); maria_init(); - if (key_cacheing) - init_key_cache(maria_key_cache,KEY_CACHE_BLOCK_SIZE,IO_SIZE*16,0,0); get_options(argc,argv); + if (pagecacheing) + init_pagecache(maria_pagecache, IO_SIZE*16, 0, 0, MARIA_KEY_BLOCK_LENGTH); exit(run_test("test1")); } @@ -569,8 +569,8 @@ static struct my_option my_long_options[] = 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, {"key-blob", 'b', "Undocumented", 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, - {"key-cache", 'K', "Undocumented", (gptr*) &key_cacheing, - (gptr*) &key_cacheing, 0, GET_BOOL, NO_ARG, 0, 0, 0, 0, 0, 0}, + {"key-cache", 'K', "Undocumented", (gptr*) &pagecacheing, + (gptr*) &pagecacheing, 0, GET_BOOL, NO_ARG, 0, 0, 0, 0, 0, 0}, {"key-length", 'k', "Undocumented", (gptr*) &key_length, (gptr*) &key_length, 0, GET_UINT, REQUIRED_ARG, 6, 0, 0, 0, 0, 0}, {"key-multiple", 'm', "Undocumented", @@ -677,7 +677,7 @@ get_one_option(int optid, const struct my_option *opt __attribute__((unused)), record_type= DYNAMIC_RECORD; break; case 'K': /* Use key cacheing */ - key_cacheing=1; + pagecacheing=1; break; case 'V': printf("test1 Ver 1.2 \n"); diff --git a/storage/maria/ma_test2.c b/storage/maria/ma_test2.c index 46b8c710d4a..a120562bafc 100644 --- a/storage/maria/ma_test2.c +++ b/storage/maria/ma_test2.c @@ -45,13 +45,13 @@ static void copy_key(struct st_maria_info *info,uint inx, uchar *record,uchar *key); static int verbose=0,testflag=0, - first_key=0,async_io=0,key_cacheing=0,write_cacheing=0,locking=0, + first_key=0,async_io=0,pagecacheing=0,write_cacheing=0,locking=0, rec_pointer_size=0,pack_fields=1,silent=0, opt_quick_mode=0; static int pack_seg=HA_SPACE_PACK,pack_type=HA_PACK_KEY,remove_count=-1; static int create_flag= 0, srand_arg= 0; -static ulong key_cache_size=IO_SIZE*16; -static uint key_cache_block_size= KEY_CACHE_BLOCK_SIZE; +static ulong pagecache_size=IO_SIZE*16; +static uint pagecache_block_size= MARIA_KEY_BLOCK_LENGTH; static enum data_file_type record_type= DYNAMIC_RECORD; static uint keys=MARIA_KEYS,recant=1000; @@ -220,8 +220,8 @@ int main(int argc, char *argv[]) goto err; if (!silent) printf("- Writing key:s\n"); - if (key_cacheing) - init_key_cache(maria_key_cache,key_cache_block_size,key_cache_size,0,0); + if (pagecacheing) + init_pagecache(maria_pagecache, pagecache_size, 0, 0, pagecache_block_size); if (locking) maria_lock_database(file,F_WRLCK); if (write_cacheing) @@ -283,9 +283,12 @@ int main(int argc, char *argv[]) goto end; } } - if (key_cacheing) - resize_key_cache(maria_key_cache,key_cache_block_size,key_cache_size*2,0,0); - + /* + TODO: uncomment when resize will be implemented + if (pagecacheing) + resize_pagecache(maria_pagecache, pagecache_block_size, + pagecache_size * 2, 0, 0); + */ if (!silent) printf("- Delete\n"); if (srand_arg) @@ -861,10 +864,10 @@ end: if (rec_pointer_size) printf("Record pointer size: %d\n",rec_pointer_size); printf("maria_block_size: %lu\n", maria_block_size); - if (key_cacheing) + if (pagecacheing) { puts("Key cache used"); - printf("key_cache_block_size: %u\n", key_cache_block_size); + printf("pagecache_block_size: %u\n", pagecache_block_size); if (write_cacheing) puts("Key cache resized"); } @@ -885,14 +888,14 @@ w_requests: %10lu\n\ writes: %10lu\n\ r_requests: %10lu\n\ reads: %10lu\n", - maria_key_cache->blocks_used, - maria_key_cache->global_blocks_changed, - (ulong) maria_key_cache->global_cache_w_requests, - (ulong) maria_key_cache->global_cache_write, - (ulong) maria_key_cache->global_cache_r_requests, - (ulong) maria_key_cache->global_cache_read); + maria_pagecache->blocks_used, + maria_pagecache->global_blocks_changed, + (ulong) maria_pagecache->global_cache_w_requests, + (ulong) maria_pagecache->global_cache_write, + (ulong) maria_pagecache->global_cache_r_requests, + (ulong) maria_pagecache->global_cache_read); } - end_key_cache(maria_key_cache,1); + end_pagecache(maria_pagecache,1); my_free(blob_buffer, MYF(MY_ALLOW_ZERO_PTR)); my_end(silent ? MY_CHECK_ERROR : MY_CHECK_ERROR | MY_GIVE_INFO); return(0); @@ -924,9 +927,9 @@ static void get_options(int argc, char **argv) use_blob= atol(pos); break; case 'K': /* Use key cacheing */ - key_cacheing=1; + pagecacheing=1; if (*++pos) - key_cache_size=atol(pos); + pagecache_size=atol(pos); break; case 'W': /* Use write cacheing */ write_cacheing=1; @@ -968,13 +971,13 @@ static void get_options(int argc, char **argv) maria_block_size= my_round_up_to_next_power(maria_block_size); break; case 'E': /* maria_block_length */ - if ((key_cache_block_size=atoi(++pos)) < MARIA_MIN_KEY_BLOCK_LENGTH || - key_cache_block_size > MARIA_MAX_KEY_BLOCK_LENGTH) + if ((pagecache_block_size=atoi(++pos)) < MARIA_MIN_KEY_BLOCK_LENGTH || + pagecache_block_size > MARIA_MAX_KEY_BLOCK_LENGTH) { - fprintf(stderr,"Wrong key_cache_block_size\n"); + fprintf(stderr,"Wrong pagecache_block_size\n"); exit(1); } - key_cache_block_size= my_round_up_to_next_power(key_cache_block_size); + pagecache_block_size= my_round_up_to_next_power(pagecache_block_size); break; case 'f': if ((first_key=atoi(++pos)) < 0 || first_key >= MARIA_KEYS) diff --git a/storage/maria/ma_test3.c b/storage/maria/ma_test3.c index f6ed248ce16..b784151ad3a 100644 --- a/storage/maria/ma_test3.c +++ b/storage/maria/ma_test3.c @@ -41,7 +41,7 @@ const char *filename= "test3"; -uint tests=10,forks=10,key_cacheing=0; +uint tests=10,forks=10,pagecacheing=0; static void get_options(int argc, char *argv[]); void start_test(int id); @@ -140,10 +140,10 @@ static void get_options(int argc, char **argv) tests=atoi(++pos); break; case 'K': /* Use key cacheing */ - key_cacheing=1; + pagecacheing=1; break; case 'A': /* All flags */ - key_cacheing=1; + pagecacheing=1; break; case '?': case 'I': @@ -178,8 +178,8 @@ void start_test(int id) fprintf(stderr,"Can't open isam-file: %s\n",filename); exit(1); } - if (key_cacheing && rnd(2) == 0) - init_key_cache(maria_key_cache, KEY_CACHE_BLOCK_SIZE, 65536L, 0, 0); + if (pagecacheing && rnd(2) == 0) + init_pagecache(maria_pagecache, 65536L, 0, 0, MARIA_KEY_BLOCK_LENGTH); printf("Process %d, pid: %d\n",id,getpid()); fflush(stdout); for (error=i=0 ; i < tests && !error; i++) diff --git a/storage/maria/ma_write.c b/storage/maria/ma_write.c index 3f1ca59a00a..891f354397c 100644 --- a/storage/maria/ma_write.c +++ b/storage/maria/ma_write.c @@ -69,7 +69,7 @@ int maria_write(MARIA_HA *info, byte *record) my_bool fatal_error; DBUG_ENTER("maria_write"); DBUG_PRINT("enter",("index_file: %d data_file: %d", - info->s->kfile,info->dfile)); + info->s->kfile.file, info->dfile.file)); DBUG_EXECUTE_IF("maria_pretend_crashed_table_on_usage", maria_print_error(info->s, HA_ERR_CRASHED); diff --git a/storage/maria/maria_chk.c b/storage/maria/maria_chk.c index 8471d36bec9..1f0787ef32a 100644 --- a/storage/maria/maria_chk.c +++ b/storage/maria/maria_chk.c @@ -41,7 +41,7 @@ static const char *load_default_groups[]= { "maria_chk", 0 }; static const char *set_collation_name, *opt_tmpdir; static CHARSET_INFO *set_collation; static long opt_maria_block_size; -static long opt_key_cache_block_size; +static long opt_pagecache_block_size; static const char *my_progname_short; static int stopwords_inited= 0; static MY_TMPDIR maria_chk_tmpdir; @@ -302,9 +302,9 @@ static struct my_option my_long_options[] = (gptr*) &check_param.use_buffers, (gptr*) &check_param.use_buffers, 0, GET_ULONG, REQUIRED_ARG, (long) USE_BUFFER_INIT, (long) MALLOC_OVERHEAD, (long) ~0L, (long) MALLOC_OVERHEAD, (long) IO_SIZE, 0}, - { "key_cache_block_size", OPT_KEY_CACHE_BLOCK_SIZE, "", - (gptr*) &opt_key_cache_block_size, - (gptr*) &opt_key_cache_block_size, 0, + { "pagecache_block_size", OPT_KEY_CACHE_BLOCK_SIZE, "", + (gptr*) &opt_pagecache_block_size, + (gptr*) &opt_pagecache_block_size, 0, GET_LONG, REQUIRED_ARG, MARIA_KEY_BLOCK_LENGTH, MARIA_MIN_KEY_BLOCK_LENGTH, MARIA_MAX_KEY_BLOCK_LENGTH, 0, MARIA_MIN_KEY_BLOCK_LENGTH, 0}, { "maria_block_size", OPT_MARIA_BLOCK_SIZE, "", @@ -793,7 +793,7 @@ static void get_options(register int *argc,register char ***argv) exit(1); check_param.tmpdir=&maria_chk_tmpdir; - check_param.key_cache_block_size= opt_key_cache_block_size; + check_param.pagecache_block_size= opt_pagecache_block_size; if (set_collation_name) if (!(set_collation= get_charset_by_name(set_collation_name, @@ -1001,7 +1001,7 @@ static int maria_chk(HA_CHECK *param, my_string filename) use functions that only works on locked tables (like row caching). */ maria_lock_database(info, F_EXTRA_LCK); - datafile=info->dfile; + datafile= info->dfile.file; if (param->testflag & (T_REP_ANY | T_SORT_RECORDS | T_SORT_INDEX)) { @@ -1055,13 +1055,13 @@ static int maria_chk(HA_CHECK *param, my_string filename) #ifndef TO_BE_REMOVED if (param->out_flag & O_NEW_DATA) { /* Change temp file to org file */ - VOID(my_close(info->dfile,MYF(MY_WME))); /* Close new file */ + VOID(my_close(info->dfile.file, MYF(MY_WME))); /* Close new file */ error|=maria_change_to_newfile(filename,MARIA_NAME_DEXT,DATA_TMP_EXT, MYF(0)); if (_ma_open_datafile(info,info->s, -1)) error=1; param->out_flag&= ~O_NEW_DATA; /* We are using new datafile */ - param->read_cache.file=info->dfile; + param->read_cache.file= info->dfile.file; } #endif if (! error) @@ -1080,7 +1080,7 @@ static int maria_chk(HA_CHECK *param, my_string filename) /* what is the following parameter for ? */ (my_bool) !(param->testflag & T_REP), update_index); - datafile=info->dfile; /* This is now locked */ + datafile= info->dfile.file; /* This is now locked */ if (!error && !update_index) { if (param->verbose) @@ -1125,8 +1125,8 @@ static int maria_chk(HA_CHECK *param, my_string filename) !(param->testflag & (T_FAST | T_FORCE_CREATE))) { if (param->testflag & (T_EXTEND | T_MEDIUM)) - VOID(init_key_cache(maria_key_cache,opt_key_cache_block_size, - param->use_buffers, 0, 0)); + VOID(init_pagecache(maria_pagecache, param->use_buffers, 0, 0, + opt_pagecache_block_size)); VOID(init_io_cache(¶m->read_cache,datafile, (uint) param->read_buffer_length, READ_CACHE, @@ -1139,7 +1139,7 @@ static int maria_chk(HA_CHECK *param, my_string filename) if ((info->s->data_file_type != STATIC_RECORD) || (param->testflag & (T_EXTEND | T_MEDIUM))) error|=maria_chk_data_link(param, info, param->testflag & T_EXTEND); - error|=_ma_flush_blocks(param, share->key_cache, share->kfile); + error|=_ma_flush_blocks(param, share->pagecache, &share->kfile); VOID(end_io_cache(¶m->read_cache)); } if (!error) @@ -1534,8 +1534,8 @@ static int maria_sort_records(HA_CHECK *param, if (share->state.key_root[sort_key] == HA_OFFSET_ERROR) DBUG_RETURN(0); /* Nothing to do */ - init_key_cache(maria_key_cache, opt_key_cache_block_size, param->use_buffers, - 0, 0); + init_pagecache(maria_pagecache, param->use_buffers, + 0, 0, opt_pagecache_block_size); if (init_io_cache(&info->rec_cache,-1,(uint) param->write_buffer_length, WRITE_CACHE,share->pack.header_length,1, MYF(MY_WME | MY_WAIT_IF_FULL))) @@ -1566,8 +1566,9 @@ static int maria_sort_records(HA_CHECK *param, goto err; } if (share->pack.header_length) - if (maria_filecopy(param,new_file,info->dfile,0L,share->pack.header_length, - "datafile-header")) + if (maria_filecopy(param, new_file, info->dfile.file, 0L, + share->pack.header_length, + "datafile-header")) goto err; info->rec_cache.file=new_file; /* Use this file for cacheing*/ @@ -1575,7 +1576,7 @@ static int maria_sort_records(HA_CHECK *param, for (key=0 ; key < share->base.keys ; key++) share->keyinfo[key].flag|= HA_SORT_ALLOWS_SAME; - if (my_pread(share->kfile, temp_buff, + if (my_pread(share->kfile.file, temp_buff, (uint) keyinfo->block_length, share->state.key_root[sort_key], MYF(MY_NABP+MY_WME))) @@ -1611,9 +1612,9 @@ static int maria_sort_records(HA_CHECK *param, goto err; } - VOID(my_close(info->dfile,MYF(MY_WME))); + VOID(my_close(info->dfile.file, MYF(MY_WME))); param->out_flag|=O_NEW_DATA; /* Data in new file */ - info->dfile=new_file; /* Use new datafile */ + info->dfile.file= new_file; /* Use new datafile */ info->state->del=0; info->state->empty=0; share->state.dellink= HA_OFFSET_ERROR; @@ -1646,7 +1647,7 @@ err: my_free(sort_info.buff,MYF(MY_ALLOW_ZERO_PTR)); sort_info.buff=0; share->state.sortkey=sort_key; - DBUG_RETURN(_ma_flush_blocks(param, share->key_cache, share->kfile) | + DBUG_RETURN(_ma_flush_blocks(param, share->pagecache, &share->kfile) | got_error); } /* sort_records */ @@ -1687,7 +1688,7 @@ static int sort_record_index(MARIA_SORT_PARAM *sort_param,MARIA_HA *info, if (nod_flag) { next_page= _ma_kpos(nod_flag, keypos); - if (my_pread(info->s->kfile,(byte*) temp_buff, + if (my_pread(info->s->kfile.file, (byte*)temp_buff, (uint) keyinfo->block_length, next_page, MYF(MY_NABP+MY_WME))) { @@ -1728,7 +1729,7 @@ static int sort_record_index(MARIA_SORT_PARAM *sort_param,MARIA_HA *info, } /* Clear end of block to get better compression if the table is backuped */ bzero((byte*) buff+used_length,keyinfo->block_length-used_length); - if (my_pwrite(info->s->kfile,(byte*) buff,(uint) keyinfo->block_length, + if (my_pwrite(info->s->kfile.file, (byte*)buff, (uint)keyinfo->block_length, page,param->myf_rw)) { _ma_check_print_error(param,"%d when updating keyblock",my_errno); diff --git a/storage/maria/maria_def.h b/storage/maria/maria_def.h index b174df2fa3e..2d11dd07900 100644 --- a/storage/maria/maria_def.h +++ b/storage/maria/maria_def.h @@ -176,10 +176,10 @@ typedef struct st_maria_pack typedef struct st_maria_file_bitmap { + File file; /* should be first for compatibility with PAGECACHE_FILE */ uchar *map; ulonglong page; /* Page number for current bitmap */ uint used_size; /* Size of bitmap that is not 0 */ - File file; my_bool changed; @@ -214,7 +214,7 @@ typedef struct st_maria_share symlinks */ *index_file_name; byte *file_map; /* mem-map of file if possible */ - KEY_CACHE *key_cache; /* ref to the current key cache */ + PAGECACHE *pagecache; /* ref to the current key cache */ MARIA_DECODE_TREE *decode_trees; uint16 *decode_tables; my_bool (*once_init)(struct st_maria_share *, File); @@ -250,7 +250,7 @@ typedef struct st_maria_share uint unique_name_length; uint32 ftparsers; /* Number of distinct ftparsers + 1 */ - File kfile; /* Shared keyfile */ + PAGECACHE_FILE kfile; /* Shared keyfile */ File data_file; /* Shared data file */ int mode; /* mode of file on open */ uint reopen; /* How many times reopened */ @@ -381,7 +381,7 @@ struct st_maria_info */ ulong packed_length, blob_length; /* Length of found, packed record */ my_size_t rec_buff_size; - int dfile; /* The datafile */ + PAGECACHE_FILE dfile; /* The datafile */ uint opt_flag; /* Optim. for space/speed */ uint update; /* If file changed since open */ int lastinx; /* Last used index */ @@ -845,7 +845,8 @@ int _ma_thr_write_keys(MARIA_SORT_PARAM *sort_param); #ifdef THREAD pthread_handler_t _ma_thr_find_all_keys(void *arg); #endif -int _ma_flush_blocks(HA_CHECK *param, KEY_CACHE *key_cache, File file); +int _ma_flush_blocks(HA_CHECK *param, PAGECACHE *pagecache, + PAGECACHE_FILE *file); int _ma_sort_write_record(MARIA_SORT_PARAM *sort_param); int _ma_create_index_by_sort(MARIA_SORT_PARAM *info, my_bool no_messages, diff --git a/storage/maria/maria_ftdump.c b/storage/maria/maria_ftdump.c index ef30540bfce..fecbe7575c8 100644 --- a/storage/maria/maria_ftdump.c +++ b/storage/maria/maria_ftdump.c @@ -85,7 +85,8 @@ int main(int argc,char *argv[]) usage(); } - init_key_cache(maria_key_cache,MARIA_KEY_BLOCK_LENGTH,USE_BUFFER_INIT, 0, 0); + init_pagecache(maria_pagecache, USE_BUFFER_INIT, 0, 0, + MARIA_KEY_BLOCK_LENGTH); if (!(info=maria_open(argv[0], O_RDONLY, HA_OPEN_ABORT_IF_LOCKED|HA_OPEN_FROM_SQL_LAYER))) diff --git a/storage/maria/maria_pack.c b/storage/maria/maria_pack.c index 41c519693d5..57d5956f38e 100644 --- a/storage/maria/maria_pack.c +++ b/storage/maria/maria_pack.c @@ -525,7 +525,7 @@ static int compress(PACK_MRG_INFO *mrg,char *result_table) length=(uint) share->base.keystart; if (!(buff=my_malloc(length,MYF(MY_WME)))) goto err; - if (my_pread(share->kfile,buff,length,0L,MYF(MY_WME | MY_NABP)) || + if (my_pread(share->kfile.file, buff, length, 0L, MYF(MY_WME | MY_NABP)) || my_write(join_isam_file,buff,length, MYF(MY_WME | MY_NABP | MY_WAIT_IF_FULL))) { @@ -679,8 +679,8 @@ static int compress(PACK_MRG_INFO *mrg,char *result_table) error|=my_close(new_file,MYF(MY_WME)); if (!result_table) { - error|=my_close(isam_file->dfile,MYF(MY_WME)); - isam_file->dfile= -1; /* Tell maria_close file is closed */ + error|=my_close(isam_file->dfile.file, MYF(MY_WME)); + isam_file->dfile.file= -1; /* Tell maria_close file is closed */ isam_file->s->bitmap.file= -1; } } @@ -2983,10 +2983,10 @@ static int save_state(MARIA_HA *isam_file,PACK_MRG_INFO *mrg, share->changed=1; /* Force write of header */ share->state.open_count=0; share->global_changed=0; - VOID(my_chsize(share->kfile, share->base.keystart, 0, MYF(0))); + VOID(my_chsize(share->kfile.file, share->base.keystart, 0, MYF(0))); if (share->base.keys) isamchk_neaded=1; - DBUG_RETURN(_ma_state_info_write(share->kfile,&share->state,1+2)); + DBUG_RETURN(_ma_state_info_write(share->kfile.file, &share->state, (1 + 2))); } diff --git a/storage/maria/unittest/Makefile.am b/storage/maria/unittest/Makefile.am index 33f1bfed560..76c8aa24779 100644 --- a/storage/maria/unittest/Makefile.am +++ b/storage/maria/unittest/Makefile.am @@ -26,19 +26,18 @@ LDADD= $(top_builddir)/unittest/mytap/libmytap.a \ $(top_builddir)/storage/myisam/libmyisam.a \ $(top_builddir)/mysys/libmysys.a \ $(top_builddir)/dbug/libdbug.a \ - $(top_builddir)/strings/libmystrings.a @ZLIB_LIBS@ \ - $(top_builddir)/storage/maria/ma_loghandler.o + $(top_builddir)/strings/libmystrings.a @ZLIB_LIBS@ noinst_PROGRAMS = ma_control_file-t trnman-t lockman2-t \ - mf_pagecache_single_1k-t mf_pagecache_single_8k-t \ - mf_pagecache_single_64k-t-big \ - mf_pagecache_consist_1k-t-big \ - mf_pagecache_consist_64k-t-big \ - mf_pagecache_consist_1kHC-t-big \ - mf_pagecache_consist_64kHC-t-big \ - mf_pagecache_consist_1kRD-t-big \ - mf_pagecache_consist_64kRD-t-big \ - mf_pagecache_consist_1kWR-t-big \ - mf_pagecache_consist_64kWR-t-big \ + ma_pagecache_single_1k-t ma_pagecache_single_8k-t \ + ma_pagecache_single_64k-t-big \ + ma_pagecache_consist_1k-t-big \ + ma_pagecache_consist_64k-t-big \ + ma_pagecache_consist_1kHC-t-big \ + ma_pagecache_consist_64kHC-t-big \ + ma_pagecache_consist_1kRD-t-big \ + ma_pagecache_consist_64kRD-t-big \ + ma_pagecache_consist_1kWR-t-big \ + ma_pagecache_consist_64kWR-t-big \ ma_test_loghandler-t \ ma_test_loghandler_multigroup-t \ ma_test_loghandler_multithread-t \ @@ -49,36 +48,36 @@ ma_test_loghandler_multigroup_t_SOURCES= ma_test_loghandler_multigroup-t.c ma_ma ma_test_loghandler_multithread_t_SOURCES= ma_test_loghandler_multithread-t.c ma_maria_log_cleanup.c ma_test_loghandler_pagecache_t_SOURCES= ma_test_loghandler_pagecache-t.c ma_maria_log_cleanup.c -mf_pagecache_single_src = mf_pagecache_single.c $(top_srcdir)/mysys/mf_pagecache.c test_file.c -mf_pagecache_consist_src = mf_pagecache_consist.c $(top_srcdir)/mysys/mf_pagecache.c test_file.c -mf_pagecache_common_cppflags = -DEXTRA_DEBUG -DPAGECACHE_DEBUG -DMAIN +ma_pagecache_single_src = ma_pagecache_single.c test_file.c +ma_pagecache_consist_src = ma_pagecache_consist.c test_file.c +ma_pagecache_common_cppflags = -DEXTRA_DEBUG -DPAGECACHE_DEBUG -DMAIN -mf_pagecache_single_1k_t_SOURCES = $(mf_pagecache_single_src) -mf_pagecache_single_8k_t_SOURCES = $(mf_pagecache_single_src) -mf_pagecache_single_64k_t_big_SOURCES = $(mf_pagecache_single_src) -mf_pagecache_single_1k_t_CPPFLAGS = $(mf_pagecache_common_cppflags) -DPAGE_SIZE=1024 -mf_pagecache_single_8k_t_CPPFLAGS = $(mf_pagecache_common_cppflags) -DPAGE_SIZE=8192 -mf_pagecache_single_64k_t_big_CPPFLAGS = $(mf_pagecache_common_cppflags) -DPAGE_SIZE=65536 +ma_pagecache_single_1k_t_SOURCES = $(ma_pagecache_single_src) +ma_pagecache_single_8k_t_SOURCES = $(ma_pagecache_single_src) +ma_pagecache_single_64k_t_big_SOURCES = $(ma_pagecache_single_src) +ma_pagecache_single_1k_t_CPPFLAGS = $(ma_pagecache_common_cppflags) -DPAGE_SIZE=1024 +ma_pagecache_single_8k_t_CPPFLAGS = $(ma_pagecache_common_cppflags) -DPAGE_SIZE=8192 +ma_pagecache_single_64k_t_big_CPPFLAGS = $(ma_pagecache_common_cppflags) -DPAGE_SIZE=65536 -mf_pagecache_consist_1k_t_big_SOURCES = $(mf_pagecache_consist_src) -mf_pagecache_consist_1k_t_big_CPPFLAGS = $(mf_pagecache_common_cppflags) -DPAGE_SIZE=1024 -mf_pagecache_consist_64k_t_big_SOURCES = $(mf_pagecache_consist_src) -mf_pagecache_consist_64k_t_big_CPPFLAGS = $(mf_pagecache_common_cppflags) -DPAGE_SIZE=65536 +ma_pagecache_consist_1k_t_big_SOURCES = $(ma_pagecache_consist_src) +ma_pagecache_consist_1k_t_big_CPPFLAGS = $(ma_pagecache_common_cppflags) -DPAGE_SIZE=1024 +ma_pagecache_consist_64k_t_big_SOURCES = $(ma_pagecache_consist_src) +ma_pagecache_consist_64k_t_big_CPPFLAGS = $(ma_pagecache_common_cppflags) -DPAGE_SIZE=65536 -mf_pagecache_consist_1kHC_t_big_SOURCES = $(mf_pagecache_consist_src) -mf_pagecache_consist_1kHC_t_big_CPPFLAGS = $(mf_pagecache_common_cppflags) -DPAGE_SIZE=1024 -DTEST_HIGH_CONCURENCY -mf_pagecache_consist_64kHC_t_big_SOURCES = $(mf_pagecache_consist_src) -mf_pagecache_consist_64kHC_t_big_CPPFLAGS = $(mf_pagecache_common_cppflags) -DPAGE_SIZE=65536 -DTEST_HIGH_CONCURENCY +ma_pagecache_consist_1kHC_t_big_SOURCES = $(ma_pagecache_consist_src) +ma_pagecache_consist_1kHC_t_big_CPPFLAGS = $(ma_pagecache_common_cppflags) -DPAGE_SIZE=1024 -DTEST_HIGH_CONCURENCY +ma_pagecache_consist_64kHC_t_big_SOURCES = $(ma_pagecache_consist_src) +ma_pagecache_consist_64kHC_t_big_CPPFLAGS = $(ma_pagecache_common_cppflags) -DPAGE_SIZE=65536 -DTEST_HIGH_CONCURENCY -mf_pagecache_consist_1kRD_t_big_SOURCES = $(mf_pagecache_consist_src) -mf_pagecache_consist_1kRD_t_big_CPPFLAGS = $(mf_pagecache_common_cppflags) -DPAGE_SIZE=1024 -DTEST_READERS -mf_pagecache_consist_64kRD_t_big_SOURCES = $(mf_pagecache_consist_src) -mf_pagecache_consist_64kRD_t_big_CPPFLAGS = $(mf_pagecache_common_cppflags) -DPAGE_SIZE=65536 -DTEST_READERS +ma_pagecache_consist_1kRD_t_big_SOURCES = $(ma_pagecache_consist_src) +ma_pagecache_consist_1kRD_t_big_CPPFLAGS = $(ma_pagecache_common_cppflags) -DPAGE_SIZE=1024 -DTEST_READERS +ma_pagecache_consist_64kRD_t_big_SOURCES = $(ma_pagecache_consist_src) +ma_pagecache_consist_64kRD_t_big_CPPFLAGS = $(ma_pagecache_common_cppflags) -DPAGE_SIZE=65536 -DTEST_READERS -mf_pagecache_consist_1kWR_t_big_SOURCES = $(mf_pagecache_consist_src) -mf_pagecache_consist_1kWR_t_big_CPPFLAGS = $(mf_pagecache_common_cppflags) -DPAGE_SIZE=1024 -DTEST_WRITERS -mf_pagecache_consist_64kWR_t_big_SOURCES = $(mf_pagecache_consist_src) -mf_pagecache_consist_64kWR_t_big_CPPFLAGS = $(mf_pagecache_common_cppflags) -DPAGE_SIZE=65536 -DTEST_WRITERS +ma_pagecache_consist_1kWR_t_big_SOURCES = $(ma_pagecache_consist_src) +ma_pagecache_consist_1kWR_t_big_CPPFLAGS = $(ma_pagecache_common_cppflags) -DPAGE_SIZE=1024 -DTEST_WRITERS +ma_pagecache_consist_64kWR_t_big_SOURCES = $(ma_pagecache_consist_src) +ma_pagecache_consist_64kWR_t_big_CPPFLAGS = $(ma_pagecache_common_cppflags) -DPAGE_SIZE=65536 -DTEST_WRITERS # the generic lock manager may not be used in the end and lockman1-t crashes, # so we don't build lockman-t and lockman1-t diff --git a/storage/maria/unittest/ma_pagecache_consist.c b/storage/maria/unittest/ma_pagecache_consist.c new file mode 100755 index 00000000000..3346160429d --- /dev/null +++ b/storage/maria/unittest/ma_pagecache_consist.c @@ -0,0 +1,458 @@ +/* + TODO: use pthread_join instead of wait_for_thread_count_to_be_zero, like in + my_atomic-t.c (see BUG#22320). + Use diag() instead of fprintf(stderr). Use ok() and plan(). +*/ + +#include +#include +#include +#include "test_file.h" +#include + +#define PCACHE_SIZE (PAGE_SIZE*1024*8) + +#ifndef DBUG_OFF +static const char* default_dbug_option; +#endif + +static char *file1_name= (char*)"page_cache_test_file_1"; +static PAGECACHE_FILE file1; +static pthread_cond_t COND_thread_count; +static pthread_mutex_t LOCK_thread_count; +static uint thread_count; +static PAGECACHE pagecache; + +#ifdef TEST_HIGH_CONCURENCY +static uint number_of_readers= 10; +static uint number_of_writers= 20; +static uint number_of_tests= 30000; +static uint record_length_limit= PAGE_SIZE/200; +static uint number_of_pages= 20; +static uint flush_divider= 1000; +#else /*TEST_HIGH_CONCURENCY*/ +#ifdef TEST_READERS +static uint number_of_readers= 10; +static uint number_of_writers= 1; +static uint number_of_tests= 30000; +static uint record_length_limit= PAGE_SIZE/200; +static uint number_of_pages= 20; +static uint flush_divider= 1000; +#else /*TEST_READERS*/ +#ifdef TEST_WRITERS +static uint number_of_readers= 0; +static uint number_of_writers= 10; +static uint number_of_tests= 30000; +static uint record_length_limit= PAGE_SIZE/200; +static uint number_of_pages= 20; +static uint flush_divider= 1000; +#else /*TEST_WRITERS*/ +static uint number_of_readers= 10; +static uint number_of_writers= 10; +static uint number_of_tests= 50000; +static uint record_length_limit= PAGE_SIZE/200; +static uint number_of_pages= 20000; +static uint flush_divider= 1000; +#endif /*TEST_WRITERS*/ +#endif /*TEST_READERS*/ +#endif /*TEST_HIGH_CONCURENCY*/ + + +/* + Get pseudo-random length of the field in (0;limit) + + SYNOPSYS + get_len() + limit limit for generated value + + RETURN + length where length >= 0 & length < limit +*/ + +static uint get_len(uint limit) +{ + uint32 rec_len; + do + { + rec_len= random() / + (RAND_MAX / limit); + } while (rec_len >= limit || rec_len == 0); + return rec_len; +} + + +/* check page consistency */ +uint check_page(uchar *buff, ulong offset, int page_locked, int page_no, + int tag) +{ + uint end= sizeof(uint); + uint num= *((uint *)buff); + uint i; + DBUG_ENTER("check_page"); + + for (i= 0; i < num; i++) + { + uint len= *((uint *)(buff + end)); + uint j; + end+= sizeof(uint) + sizeof(uint); + if (len + end > PAGE_SIZE) + { + diag("incorrect field header #%u by offset %lu\n", i, offset + end); + goto err; + } + for(j= 0; j < len; j++) + { + if (buff[end + j] != (uchar)((i+1) % 256)) + { + diag("incorrect %lu byte\n", offset + end + j); + goto err; + } + } + end+= len; + } + for(i= end; i < PAGE_SIZE; i++) + { + if (buff[i] != 0) + { + int h; + DBUG_PRINT("err", + ("byte %lu (%lu + %u), page %u (%s, end: %u, recs: %u, tag: %d) should be 0\n", + offset + i, offset, i, page_no, + (page_locked ? "locked" : "unlocked"), + end, num, tag)); + diag("byte %lu (%lu + %u), page %u (%s, end: %u, recs: %u, tag: %d) should be 0\n", + offset + i, offset, i, page_no, + (page_locked ? "locked" : "unlocked"), + end, num, tag); + h= my_open("wrong_page", O_CREAT | O_TRUNC | O_RDWR, MYF(0)); + my_pwrite(h, (byte*) buff, PAGE_SIZE, 0, MYF(0)); + my_close(h, MYF(0)); + goto err; + } + } + DBUG_RETURN(end); +err: + DBUG_PRINT("err", ("try to flush")); + if (page_locked) + { + pagecache_delete_page(&pagecache, &file1, page_no, + PAGECACHE_LOCK_LEFT_WRITELOCKED, 1); + } + else + { + flush_pagecache_blocks(&pagecache, &file1, FLUSH_RELEASE); + } + exit(1); +} + +void put_rec(uchar *buff, uint end, uint len, uint tag) +{ + uint i; + uint num= *((uint *)buff); + if (!len) + len= 1; + if (end + sizeof(uint)*2 + len > PAGE_SIZE) + return; + *((uint *)(buff + end))= len; + end+= sizeof(uint); + *((uint *)(buff + end))= tag; + end+= sizeof(uint); + num++; + *((uint *)buff)= num; + *((uint*)(buff + end))= len; + for (i= end; i < (len + end); i++) + { + buff[i]= (uchar) num % 256; + } +} + +/* + Recreate and reopen a file for test + + SYNOPSIS + reset_file() + file File to reset + file_name Path (and name) of file which should be reset +*/ + +void reset_file(PAGECACHE_FILE file, char *file_name) +{ + flush_pagecache_blocks(&pagecache, &file1, FLUSH_RELEASE); + if (my_close(file1.file, MYF(0)) != 0) + { + diag("Got error during %s closing from close() (errno: %d)\n", + file_name, errno); + exit(1); + } + my_delete(file_name, MYF(0)); + if ((file.file= my_open(file_name, + O_CREAT | O_TRUNC | O_RDWR, MYF(0))) == -1) + { + diag("Got error during %s creation from open() (errno: %d)\n", + file_name, errno); + exit(1); + } +} + + +void reader(int num) +{ + unsigned char *buffr= malloc(PAGE_SIZE); + uint i; + + for (i= 0; i < number_of_tests; i++) + { + uint page= get_len(number_of_pages); + pagecache_read(&pagecache, &file1, page, 3, (char*)buffr, + PAGECACHE_PLAIN_PAGE, + PAGECACHE_LOCK_LEFT_UNLOCKED, + 0); + check_page(buffr, page * PAGE_SIZE, 0, page, -num); + if (i % 500 == 0) + printf("reader%d: %d\n", num, i); + + } + printf("reader%d: done\n", num); + free(buffr); +} + + +void writer(int num) +{ + unsigned char *buffr= malloc(PAGE_SIZE); + uint i; + + for (i= 0; i < number_of_tests; i++) + { + uint end; + uint page= get_len(number_of_pages); + pagecache_read(&pagecache, &file1, page, 3, (char*)buffr, + PAGECACHE_PLAIN_PAGE, + PAGECACHE_LOCK_WRITE, + 0); + end= check_page(buffr, page * PAGE_SIZE, 1, page, num); + put_rec(buffr, end, get_len(record_length_limit), num); + pagecache_write(&pagecache, &file1, page, 3, (char*)buffr, + PAGECACHE_PLAIN_PAGE, + PAGECACHE_LOCK_WRITE_UNLOCK, + PAGECACHE_UNPIN, + PAGECACHE_WRITE_DELAY, + 0); + + if (i % flush_divider == 0) + flush_pagecache_blocks(&pagecache, &file1, FLUSH_FORCE_WRITE); + if (i % 500 == 0) + printf("writer%d: %d\n", num, i); + } + printf("writer%d: done\n", num); + free(buffr); +} + + +static void *test_thread_reader(void *arg) +{ + int param=*((int*) arg); + + my_thread_init(); + DBUG_ENTER("test_reader"); + DBUG_PRINT("enter", ("param: %d", param)); + + reader(param); + + DBUG_PRINT("info", ("Thread %s ended\n", my_thread_name())); + pthread_mutex_lock(&LOCK_thread_count); + thread_count--; + VOID(pthread_cond_signal(&COND_thread_count)); /* Tell main we are ready */ + pthread_mutex_unlock(&LOCK_thread_count); + free((gptr) arg); + my_thread_end(); + DBUG_RETURN(0); +} + +static void *test_thread_writer(void *arg) +{ + int param=*((int*) arg); + + my_thread_init(); + DBUG_ENTER("test_writer"); + DBUG_PRINT("enter", ("param: %d", param)); + + writer(param); + + DBUG_PRINT("info", ("Thread %s ended\n", my_thread_name())); + pthread_mutex_lock(&LOCK_thread_count); + thread_count--; + VOID(pthread_cond_signal(&COND_thread_count)); /* Tell main we are ready */ + pthread_mutex_unlock(&LOCK_thread_count); + free((gptr) arg); + my_thread_end(); + DBUG_RETURN(0); +} + +int main(int argc, char **argv __attribute__((unused))) +{ + pthread_t tid; + pthread_attr_t thr_attr; + int *param, error, pagen; + + MY_INIT(argv[0]); + +#ifndef DBUG_OFF +#if defined(__WIN__) + default_dbug_option= "d:t:i:O,\\test_pagecache_consist.trace"; +#else + default_dbug_option= "d:t:i:o,/tmp/test_pagecache_consist.trace"; +#endif + if (argc > 1) + { + DBUG_SET(default_dbug_option); + DBUG_SET_INITIAL(default_dbug_option); + } +#endif + + + DBUG_ENTER("main"); + DBUG_PRINT("info", ("Main thread: %s\n", my_thread_name())); + if ((file1.file= my_open(file1_name, + O_CREAT | O_TRUNC | O_RDWR, MYF(0))) == -1) + { + fprintf(stderr, "Got error during file1 creation from open() (errno: %d)\n", + errno); + exit(1); + } + DBUG_PRINT("info", ("file1: %d", file1.file)); + if (chmod(file1_name, S_IRWXU | S_IRWXG | S_IRWXO) != 0) + { + fprintf(stderr, "Got error during file1 chmod() (errno: %d)\n", + errno); + exit(1); + } + my_pwrite(file1.file, "test file", 9, 0, MYF(0)); + + if ((error= pthread_cond_init(&COND_thread_count, NULL))) + { + fprintf(stderr, "COND_thread_count: %d from pthread_cond_init (errno: %d)\n", + error, errno); + exit(1); + } + if ((error= pthread_mutex_init(&LOCK_thread_count, MY_MUTEX_INIT_FAST))) + { + fprintf(stderr, "LOCK_thread_count: %d from pthread_cond_init (errno: %d)\n", + error, errno); + exit(1); + } + + if ((error= pthread_attr_init(&thr_attr))) + { + fprintf(stderr,"Got error: %d from pthread_attr_init (errno: %d)\n", + error,errno); + exit(1); + } + if ((error= pthread_attr_setdetachstate(&thr_attr, PTHREAD_CREATE_DETACHED))) + { + fprintf(stderr, + "Got error: %d from pthread_attr_setdetachstate (errno: %d)\n", + error,errno); + exit(1); + } + +#ifdef HAVE_THR_SETCONCURRENCY + VOID(thr_setconcurrency(2)); +#endif + + if ((pagen= init_pagecache(&pagecache, PCACHE_SIZE, 0, 0, + PAGE_SIZE)) == 0) + { + fprintf(stderr,"Got error: init_pagecache() (errno: %d)\n", + errno); + exit(1); + } + DBUG_PRINT("info", ("Page cache %d pages", pagen)); + { + unsigned char *buffr= malloc(PAGE_SIZE); + uint i; + memset(buffr, '\0', PAGE_SIZE); + for (i= 0; i < number_of_pages; i++) + { + pagecache_write(&pagecache, &file1, i, 3, (char*)buffr, + PAGECACHE_PLAIN_PAGE, + PAGECACHE_LOCK_LEFT_UNLOCKED, + PAGECACHE_PIN_LEFT_UNPINNED, + PAGECACHE_WRITE_DELAY, + 0); + } + flush_pagecache_blocks(&pagecache, &file1, FLUSH_FORCE_WRITE); + free(buffr); + } + if ((error= pthread_mutex_lock(&LOCK_thread_count))) + { + fprintf(stderr,"LOCK_thread_count: %d from pthread_mutex_lock (errno: %d)\n", + error,errno); + exit(1); + } + while (number_of_readers != 0 || number_of_writers != 0) + { + if (number_of_readers != 0) + { + param=(int*) malloc(sizeof(int)); + *param= number_of_readers; + if ((error= pthread_create(&tid, &thr_attr, test_thread_reader, + (void*) param))) + { + fprintf(stderr,"Got error: %d from pthread_create (errno: %d)\n", + error,errno); + exit(1); + } + thread_count++; + number_of_readers--; + } + if (number_of_writers != 0) + { + param=(int*) malloc(sizeof(int)); + *param= number_of_writers; + if ((error= pthread_create(&tid, &thr_attr, test_thread_writer, + (void*) param))) + { + fprintf(stderr,"Got error: %d from pthread_create (errno: %d)\n", + error,errno); + exit(1); + } + thread_count++; + number_of_writers--; + } + } + DBUG_PRINT("info", ("Thread started")); + pthread_mutex_unlock(&LOCK_thread_count); + + pthread_attr_destroy(&thr_attr); + + /* wait finishing */ + if ((error= pthread_mutex_lock(&LOCK_thread_count))) + fprintf(stderr,"LOCK_thread_count: %d from pthread_mutex_lock\n",error); + while (thread_count) + { + if ((error= pthread_cond_wait(&COND_thread_count,&LOCK_thread_count))) + fprintf(stderr,"COND_thread_count: %d from pthread_cond_wait\n",error); + } + if ((error= pthread_mutex_unlock(&LOCK_thread_count))) + fprintf(stderr,"LOCK_thread_count: %d from pthread_mutex_unlock\n",error); + DBUG_PRINT("info", ("thread ended")); + + end_pagecache(&pagecache, 1); + DBUG_PRINT("info", ("Page cache ended")); + + if (my_close(file1.file, MYF(0)) != 0) + { + fprintf(stderr, "Got error during file1 closing from close() (errno: %d)\n", + errno); + exit(1); + } + /*my_delete(file1_name, MYF(0));*/ + my_end(0); + + DBUG_PRINT("info", ("file1 (%d) closed", file1.file)); + + DBUG_PRINT("info", ("Program end")); + + DBUG_RETURN(exit_status()); +} diff --git a/storage/maria/unittest/ma_pagecache_single.c b/storage/maria/unittest/ma_pagecache_single.c new file mode 100644 index 00000000000..91cceee618d --- /dev/null +++ b/storage/maria/unittest/ma_pagecache_single.c @@ -0,0 +1,580 @@ +/* + TODO: use pthread_join instead of wait_for_thread_count_to_be_zero, like in + my_atomic-t.c (see BUG#22320). + Use diag() instead of fprintf(stderr). +*/ +#include +#include +#include +#include "test_file.h" +#include + +#define PCACHE_SIZE (PAGE_SIZE*1024*10) + +#ifndef DBUG_OFF +static const char* default_dbug_option; +#endif + +static char *file1_name= (char*)"page_cache_test_file_1"; +static PAGECACHE_FILE file1; +static pthread_cond_t COND_thread_count; +static pthread_mutex_t LOCK_thread_count; +static uint thread_count; +static PAGECACHE pagecache; + +/* + File contance descriptors +*/ +static struct file_desc simple_read_write_test_file[]= +{ + {PAGE_SIZE, '\1'}, + { 0, 0} +}; +static struct file_desc simple_read_change_write_read_test_file[]= +{ + {PAGE_SIZE/2, '\65'}, + {PAGE_SIZE/2, '\1'}, + { 0, 0} +}; +static struct file_desc simple_pin_test_file1[]= +{ + {PAGE_SIZE*2, '\1'}, + { 0, 0} +}; +static struct file_desc simple_pin_test_file2[]= +{ + {PAGE_SIZE/2, '\1'}, + {PAGE_SIZE/2, (unsigned char)129}, + {PAGE_SIZE, '\1'}, + { 0, 0} +}; +static struct file_desc simple_delete_forget_test_file[]= +{ + {PAGE_SIZE, '\1'}, + { 0, 0} +}; +static struct file_desc simple_delete_flush_test_file[]= +{ + {PAGE_SIZE, '\2'}, + { 0, 0} +}; + + +/* + Recreate and reopen a file for test + + SYNOPSIS + reset_file() + file File to reset + file_name Path (and name) of file which should be reset +*/ + +void reset_file(PAGECACHE_FILE file, char *file_name) +{ + flush_pagecache_blocks(&pagecache, &file1, FLUSH_RELEASE); + if (my_close(file1.file, MYF(0)) != 0) + { + diag("Got error during %s closing from close() (errno: %d)\n", + file_name, errno); + exit(1); + } + my_delete(file_name, MYF(0)); + if ((file.file= my_open(file_name, + O_CREAT | O_TRUNC | O_RDWR, MYF(0))) == -1) + { + diag("Got error during %s creation from open() (errno: %d)\n", + file_name, errno); + exit(1); + } +} + +/* + Write then read page, check file on disk +*/ + +int simple_read_write_test() +{ + unsigned char *buffw= malloc(PAGE_SIZE); + unsigned char *buffr= malloc(PAGE_SIZE); + int res; + DBUG_ENTER("simple_read_write_test"); + bfill(buffw, PAGE_SIZE, '\1'); + pagecache_write(&pagecache, &file1, 0, 3, (char*)buffw, + PAGECACHE_PLAIN_PAGE, + PAGECACHE_LOCK_LEFT_UNLOCKED, + PAGECACHE_PIN_LEFT_UNPINNED, + PAGECACHE_WRITE_DELAY, + 0); + pagecache_read(&pagecache, &file1, 0, 3, (char*)buffr, + PAGECACHE_PLAIN_PAGE, + PAGECACHE_LOCK_LEFT_UNLOCKED, + 0); + ok((res= test(memcmp(buffr, buffw, PAGE_SIZE) == 0)), + "Simple write-read page "); + flush_pagecache_blocks(&pagecache, &file1, FLUSH_FORCE_WRITE); + ok((res&= test(test_file(file1, file1_name, PAGE_SIZE, PAGE_SIZE, + simple_read_write_test_file))), + "Simple write-read page file"); + if (res) + reset_file(file1, file1_name); + free(buffw); + free(buffr); + DBUG_RETURN(res); +} + + +/* + Prepare page, then read (and lock), change (write new value and unlock), + then check the page in the cache and on the disk +*/ +int simple_read_change_write_read_test() +{ + unsigned char *buffw= malloc(PAGE_SIZE); + unsigned char *buffr= malloc(PAGE_SIZE); + int res; + DBUG_ENTER("simple_read_change_write_read_test"); + /* prepare the file */ + bfill(buffw, PAGE_SIZE, '\1'); + pagecache_write(&pagecache, &file1, 0, 3, (char*)buffw, + PAGECACHE_PLAIN_PAGE, + PAGECACHE_LOCK_LEFT_UNLOCKED, + PAGECACHE_PIN_LEFT_UNPINNED, + PAGECACHE_WRITE_DELAY, + 0); + flush_pagecache_blocks(&pagecache, &file1, FLUSH_FORCE_WRITE); + /* test */ + pagecache_read(&pagecache, &file1, 0, 3, (char*)buffw, + PAGECACHE_PLAIN_PAGE, + PAGECACHE_LOCK_WRITE, + 0); + bfill(buffw, PAGE_SIZE/2, '\65'); + pagecache_write(&pagecache, &file1, 0, 3, (char*)buffw, + PAGECACHE_PLAIN_PAGE, + PAGECACHE_LOCK_WRITE_UNLOCK, + PAGECACHE_UNPIN, + PAGECACHE_WRITE_DELAY, + 0); + + pagecache_read(&pagecache, &file1, 0, 3, (char*)buffr, + PAGECACHE_PLAIN_PAGE, + PAGECACHE_LOCK_LEFT_UNLOCKED, + 0); + ok((res= test(memcmp(buffr, buffw, PAGE_SIZE) == 0)), + "Simple read-change-write-read page "); + flush_pagecache_blocks(&pagecache, &file1, FLUSH_FORCE_WRITE); + ok((res&= test(test_file(file1, file1_name, PAGE_SIZE, PAGE_SIZE, + simple_read_change_write_read_test_file))), + "Simple read-change-write-read page file"); + if (res) + reset_file(file1, file1_name); + free(buffw); + free(buffr); + DBUG_RETURN(res); +} + + +/* + Prepare page, read page 0 (and pin) then write page 1 and page 0. + Flush the file (shold flush only page 1 and return 1 (page 0 is + still pinned). + Check file on the disk. + Unpin and flush. + Check file on the disk. +*/ +int simple_pin_test() +{ + unsigned char *buffw= malloc(PAGE_SIZE); + unsigned char *buffr= malloc(PAGE_SIZE); + int res; + DBUG_ENTER("simple_pin_test"); + /* prepare the file */ + bfill(buffw, PAGE_SIZE, '\1'); + pagecache_write(&pagecache, &file1, 0, 3, (char*)buffw, + PAGECACHE_PLAIN_PAGE, + PAGECACHE_LOCK_LEFT_UNLOCKED, + PAGECACHE_PIN_LEFT_UNPINNED, + PAGECACHE_WRITE_DELAY, + 0); + /* test */ + if (flush_pagecache_blocks(&pagecache, &file1, FLUSH_FORCE_WRITE)) + { + diag("error in flush_pagecache_blocks\n"); + exit(1); + } + pagecache_read(&pagecache, &file1, 0, 3, (char*)buffw, + PAGECACHE_PLAIN_PAGE, + PAGECACHE_LOCK_WRITE, + 0); + pagecache_write(&pagecache, &file1, 1, 3, (char*)buffw, + PAGECACHE_PLAIN_PAGE, + PAGECACHE_LOCK_LEFT_UNLOCKED, + PAGECACHE_PIN_LEFT_UNPINNED, + PAGECACHE_WRITE_DELAY, + 0); + bfill(buffw + PAGE_SIZE/2, PAGE_SIZE/2, ((unsigned char) 129)); + pagecache_write(&pagecache, &file1, 0, 3, (char*)buffw, + PAGECACHE_PLAIN_PAGE, + PAGECACHE_LOCK_WRITE_TO_READ, + PAGECACHE_PIN_LEFT_PINNED, + PAGECACHE_WRITE_DELAY, + 0); + /* + We have to get error because one page of the file is pinned, + other page should be flushed + */ + if (!flush_pagecache_blocks(&pagecache, &file1, FLUSH_FORCE_WRITE)) + { + diag("Did not get error in flush_pagecache_blocks\n"); + res= 0; + goto err; + } + ok((res= test(test_file(file1, file1_name, PAGE_SIZE*2, PAGE_SIZE*2, + simple_pin_test_file1))), + "Simple pin page file with pin"); + pagecache_unlock_page(&pagecache, + &file1, + 0, + PAGECACHE_LOCK_READ_UNLOCK, + PAGECACHE_UNPIN, + 0); + if (flush_pagecache_blocks(&pagecache, &file1, FLUSH_FORCE_WRITE)) + { + diag("Got error in flush_pagecache_blocks\n"); + res= 0; + goto err; + } + ok((res&= test(test_file(file1, file1_name, PAGE_SIZE*2, PAGE_SIZE, + simple_pin_test_file2))), + "Simple pin page result file"); + if (res) + reset_file(file1, file1_name); +err: + free(buffw); + free(buffr); + DBUG_RETURN(res); +} + +/* + Prepare page, write new value, then delete page from cache without flush, + on the disk should be page with old content written during preparation +*/ + +int simple_delete_forget_test() +{ + unsigned char *buffw= malloc(PAGE_SIZE); + unsigned char *buffr= malloc(PAGE_SIZE); + int res; + DBUG_ENTER("simple_delete_forget_test"); + /* prepare the file */ + bfill(buffw, PAGE_SIZE, '\1'); + pagecache_write(&pagecache, &file1, 0, 3, (char*)buffw, + PAGECACHE_PLAIN_PAGE, + PAGECACHE_LOCK_LEFT_UNLOCKED, + PAGECACHE_PIN_LEFT_UNPINNED, + PAGECACHE_WRITE_DELAY, + 0); + flush_pagecache_blocks(&pagecache, &file1, FLUSH_FORCE_WRITE); + /* test */ + bfill(buffw, PAGE_SIZE, '\2'); + pagecache_write(&pagecache, &file1, 0, 3, (char*)buffw, + PAGECACHE_PLAIN_PAGE, + PAGECACHE_LOCK_LEFT_UNLOCKED, + PAGECACHE_PIN_LEFT_UNPINNED, + PAGECACHE_WRITE_DELAY, + 0); + pagecache_delete_page(&pagecache, &file1, 0, + PAGECACHE_LOCK_WRITE, 0); + flush_pagecache_blocks(&pagecache, &file1, FLUSH_FORCE_WRITE); + ok((res= test(test_file(file1, file1_name, PAGE_SIZE, PAGE_SIZE, + simple_delete_forget_test_file))), + "Simple delete-forget page file"); + if (res) + reset_file(file1, file1_name); + free(buffw); + free(buffr); + DBUG_RETURN(res); +} + +/* + Prepare page with locking, write new content to the page, + delete page with flush and on existing lock, + check that page on disk contain new value. +*/ + +int simple_delete_flush_test() +{ + unsigned char *buffw= malloc(PAGE_SIZE); + unsigned char *buffr= malloc(PAGE_SIZE); + int res; + DBUG_ENTER("simple_delete_flush_test"); + /* prepare the file */ + bfill(buffw, PAGE_SIZE, '\1'); + pagecache_write(&pagecache, &file1, 0, 3, (char*)buffw, + PAGECACHE_PLAIN_PAGE, + PAGECACHE_LOCK_WRITE, + PAGECACHE_PIN, + PAGECACHE_WRITE_DELAY, + 0); + flush_pagecache_blocks(&pagecache, &file1, FLUSH_FORCE_WRITE); + /* test */ + bfill(buffw, PAGE_SIZE, '\2'); + pagecache_write(&pagecache, &file1, 0, 3, (char*)buffw, + PAGECACHE_PLAIN_PAGE, + PAGECACHE_LOCK_LEFT_WRITELOCKED, + PAGECACHE_PIN_LEFT_PINNED, + PAGECACHE_WRITE_DELAY, + 0); + pagecache_delete_page(&pagecache, &file1, 0, + PAGECACHE_LOCK_LEFT_WRITELOCKED, 1); + flush_pagecache_blocks(&pagecache, &file1, FLUSH_FORCE_WRITE); + ok((res= test(test_file(file1, file1_name, PAGE_SIZE, PAGE_SIZE, + simple_delete_flush_test_file))), + "Simple delete-forget page file"); + if (res) + reset_file(file1, file1_name); + free(buffw); + free(buffr); + DBUG_RETURN(res); +} + + +/* + write then read file bigger then cache +*/ + +int simple_big_test() +{ + unsigned char *buffw= (unsigned char *)malloc(PAGE_SIZE); + unsigned char *buffr= (unsigned char *)malloc(PAGE_SIZE); + struct file_desc *desc= + (struct file_desc *)malloc((PCACHE_SIZE/(PAGE_SIZE/2) + 1) * + sizeof(struct file_desc)); + int res, i; + DBUG_ENTER("simple_big_test"); + /* prepare the file twice larger then cache */ + for (i= 0; i < PCACHE_SIZE/(PAGE_SIZE/2); i++) + { + bfill(buffw, PAGE_SIZE, (unsigned char) (i & 0xff)); + desc[i].length= PAGE_SIZE; + desc[i].content= (i & 0xff); + pagecache_write(&pagecache, &file1, i, 3, (char*)buffw, + PAGECACHE_PLAIN_PAGE, + PAGECACHE_LOCK_LEFT_UNLOCKED, + PAGECACHE_PIN_LEFT_UNPINNED, + PAGECACHE_WRITE_DELAY, + 0); + } + desc[i].length= 0; + desc[i].content= '\0'; + ok(1, "Simple big file write"); + /* check written pages sequentally read */ + for (i= 0; i < PCACHE_SIZE/(PAGE_SIZE/2); i++) + { + int j; + pagecache_read(&pagecache, &file1, i, 3, (char*)buffr, + PAGECACHE_PLAIN_PAGE, + PAGECACHE_LOCK_LEFT_UNLOCKED, + 0); + for(j= 0; j < PAGE_SIZE; j++) + { + if (buffr[j] != (i & 0xff)) + { + diag("simple_big_test seq: page %u byte %u mismatch\n", i, j); + return 0; + } + } + } + ok(1, "simple big file sequentally read"); + /* chack random reads */ + for (i= 0; i < PCACHE_SIZE/(PAGE_SIZE); i++) + { + int j, page; + page= rand() % (PCACHE_SIZE/(PAGE_SIZE/2)); + pagecache_read(&pagecache, &file1, page, 3, (char*)buffr, + PAGECACHE_PLAIN_PAGE, + PAGECACHE_LOCK_LEFT_UNLOCKED, + 0); + for(j= 0; j < PAGE_SIZE; j++) + { + if (buffr[j] != (page & 0xff)) + { + diag("simple_big_test rnd: page %u byte %u mismatch\n", page, j); + return 0; + } + } + } + ok(1, "simple big file random read"); + flush_pagecache_blocks(&pagecache, &file1, FLUSH_FORCE_WRITE); + + ok((res= test(test_file(file1, file1_name, PCACHE_SIZE*2, PAGE_SIZE, + desc))), + "Simple big file"); + if (res) + reset_file(file1, file1_name); + free(buffw); + free(buffr); + DBUG_RETURN(res); +} +/* + Thread function +*/ + +static void *test_thread(void *arg) +{ + int param=*((int*) arg); + + my_thread_init(); + DBUG_ENTER("test_thread"); + + DBUG_PRINT("enter", ("param: %d", param)); + + if (!simple_read_write_test() || + !simple_read_change_write_read_test() || + !simple_pin_test() || + !simple_delete_forget_test() || + !simple_delete_flush_test() || + !simple_big_test()) + exit(1); + + DBUG_PRINT("info", ("Thread %s ended\n", my_thread_name())); + pthread_mutex_lock(&LOCK_thread_count); + thread_count--; + VOID(pthread_cond_signal(&COND_thread_count)); /* Tell main we are ready */ + pthread_mutex_unlock(&LOCK_thread_count); + free((gptr) arg); + my_thread_end(); + DBUG_RETURN(0); +} + + +int main(int argc, char **argv __attribute__((unused))) +{ + pthread_t tid; + pthread_attr_t thr_attr; + int *param, error, pagen; + + MY_INIT(argv[0]); + +#ifndef DBUG_OFF +#if defined(__WIN__) + default_dbug_option= "d:t:i:O,\\test_pagecache_single.trace"; +#else + default_dbug_option= "d:t:i:o,/tmp/test_pagecache_single.trace"; +#endif + if (argc > 1) + { + DBUG_SET(default_dbug_option); + DBUG_SET_INITIAL(default_dbug_option); + } +#endif + + + DBUG_ENTER("main"); + DBUG_PRINT("info", ("Main thread: %s\n", my_thread_name())); + if ((file1.file= my_open(file1_name, + O_CREAT | O_TRUNC | O_RDWR, MYF(0))) == -1) + { + fprintf(stderr, "Got error during file1 creation from open() (errno: %d)\n", + errno); + exit(1); + } + DBUG_PRINT("info", ("file1: %d", file1.file)); + if (chmod(file1_name, S_IRWXU | S_IRWXG | S_IRWXO) != 0) + { + fprintf(stderr, "Got error during file1 chmod() (errno: %d)\n", + errno); + exit(1); + } + my_pwrite(file1.file, "test file", 9, 0, MYF(0)); + + if ((error= pthread_cond_init(&COND_thread_count, NULL))) + { + fprintf(stderr, "Got error: %d from pthread_cond_init (errno: %d)\n", + error, errno); + exit(1); + } + if ((error= pthread_mutex_init(&LOCK_thread_count, MY_MUTEX_INIT_FAST))) + { + fprintf(stderr, "Got error: %d from pthread_cond_init (errno: %d)\n", + error, errno); + exit(1); + } + + if ((error= pthread_attr_init(&thr_attr))) + { + fprintf(stderr,"Got error: %d from pthread_attr_init (errno: %d)\n", + error,errno); + exit(1); + } + if ((error= pthread_attr_setdetachstate(&thr_attr, PTHREAD_CREATE_DETACHED))) + { + fprintf(stderr, + "Got error: %d from pthread_attr_setdetachstate (errno: %d)\n", + error,errno); + exit(1); + } + +#ifdef HAVE_THR_SETCONCURRENCY + VOID(thr_setconcurrency(2)); +#endif + + plan(12); + + if ((pagen= init_pagecache(&pagecache, PCACHE_SIZE, 0, 0, + PAGE_SIZE)) == 0) + { + fprintf(stderr,"Got error: init_pagecache() (errno: %d)\n", + errno); + exit(1); + } + DBUG_PRINT("info", ("Page cache %d pages", pagen)); + + if ((error=pthread_mutex_lock(&LOCK_thread_count))) + { + fprintf(stderr,"Got error: %d from pthread_mutex_lock (errno: %d)\n", + error,errno); + exit(1); + } + param=(int*) malloc(sizeof(int)); + *param= 1; + if ((error= pthread_create(&tid, &thr_attr, test_thread, (void*) param))) + { + fprintf(stderr,"Got error: %d from pthread_create (errno: %d)\n", + error,errno); + exit(1); + } + thread_count++; + DBUG_PRINT("info", ("Thread started")); + pthread_mutex_unlock(&LOCK_thread_count); + + pthread_attr_destroy(&thr_attr); + + if ((error= pthread_mutex_lock(&LOCK_thread_count))) + fprintf(stderr,"Got error: %d from pthread_mutex_lock\n",error); + while (thread_count) + { + if ((error= pthread_cond_wait(&COND_thread_count,&LOCK_thread_count))) + fprintf(stderr,"Got error: %d from pthread_cond_wait\n",error); + } + if ((error= pthread_mutex_unlock(&LOCK_thread_count))) + fprintf(stderr,"Got error: %d from pthread_mutex_unlock\n",error); + DBUG_PRINT("info", ("thread ended")); + + end_pagecache(&pagecache, 1); + DBUG_PRINT("info", ("Page cache ended")); + + if (my_close(file1.file, MYF(0)) != 0) + { + fprintf(stderr, "Got error during file1 closing from close() (errno: %d)\n", + errno); + exit(1); + } + /*my_delete(file1_name, MYF(0));*/ + my_end(0); + + DBUG_PRINT("info", ("file1 (%d) closed", file1.file)); + + DBUG_PRINT("info", ("Program end")); + + DBUG_RETURN(exit_status()); +} diff --git a/storage/maria/unittest/mf_pagecache_consist.c b/storage/maria/unittest/mf_pagecache_consist.c deleted file mode 100755 index 8ea0094762c..00000000000 --- a/storage/maria/unittest/mf_pagecache_consist.c +++ /dev/null @@ -1,458 +0,0 @@ -/* - TODO: use pthread_join instead of wait_for_thread_count_to_be_zero, like in - my_atomic-t.c (see BUG#22320). - Use diag() instead of fprintf(stderr). Use ok() and plan(). -*/ - -#include -#include -#include -#include "test_file.h" -#include - -#define PCACHE_SIZE (PAGE_SIZE*1024*8) - -#ifndef DBUG_OFF -static const char* default_dbug_option; -#endif - -static char *file1_name= (char*)"page_cache_test_file_1"; -static PAGECACHE_FILE file1; -static pthread_cond_t COND_thread_count; -static pthread_mutex_t LOCK_thread_count; -static uint thread_count; -static PAGECACHE pagecache; - -#ifdef TEST_HIGH_CONCURENCY -static uint number_of_readers= 10; -static uint number_of_writers= 20; -static uint number_of_tests= 30000; -static uint record_length_limit= PAGE_SIZE/200; -static uint number_of_pages= 20; -static uint flush_divider= 1000; -#else /*TEST_HIGH_CONCURENCY*/ -#ifdef TEST_READERS -static uint number_of_readers= 10; -static uint number_of_writers= 1; -static uint number_of_tests= 30000; -static uint record_length_limit= PAGE_SIZE/200; -static uint number_of_pages= 20; -static uint flush_divider= 1000; -#else /*TEST_READERS*/ -#ifdef TEST_WRITERS -static uint number_of_readers= 0; -static uint number_of_writers= 10; -static uint number_of_tests= 30000; -static uint record_length_limit= PAGE_SIZE/200; -static uint number_of_pages= 20; -static uint flush_divider= 1000; -#else /*TEST_WRITERS*/ -static uint number_of_readers= 10; -static uint number_of_writers= 10; -static uint number_of_tests= 50000; -static uint record_length_limit= PAGE_SIZE/200; -static uint number_of_pages= 20000; -static uint flush_divider= 1000; -#endif /*TEST_WRITERS*/ -#endif /*TEST_READERS*/ -#endif /*TEST_HIGH_CONCURENCY*/ - - -/* - Get pseudo-random length of the field in (0;limit) - - SYNOPSYS - get_len() - limit limit for generated value - - RETURN - length where length >= 0 & length < limit -*/ - -static uint get_len(uint limit) -{ - uint32 rec_len; - do - { - rec_len= random() / - (RAND_MAX / limit); - } while (rec_len >= limit || rec_len == 0); - return rec_len; -} - - -/* check page consistency */ -uint check_page(uchar *buff, ulong offset, int page_locked, int page_no, - int tag) -{ - uint end= sizeof(uint); - uint num= *((uint *)buff); - uint i; - DBUG_ENTER("check_page"); - - for (i= 0; i < num; i++) - { - uint len= *((uint *)(buff + end)); - uint j; - end+= sizeof(uint) + sizeof(uint); - if (len + end > PAGE_SIZE) - { - diag("incorrect field header #%u by offset %lu\n", i, offset + end + j); - goto err; - } - for(j= 0; j < len; j++) - { - if (buff[end + j] != (uchar)((i+1) % 256)) - { - diag("incorrect %lu byte\n", offset + end + j); - goto err; - } - } - end+= len; - } - for(i= end; i < PAGE_SIZE; i++) - { - if (buff[i] != 0) - { - int h; - DBUG_PRINT("err", - ("byte %lu (%lu + %u), page %u (%s, end: %u, recs: %u, tag: %d) should be 0\n", - offset + i, offset, i, page_no, - (page_locked ? "locked" : "unlocked"), - end, num, tag)); - diag("byte %lu (%lu + %u), page %u (%s, end: %u, recs: %u, tag: %d) should be 0\n", - offset + i, offset, i, page_no, - (page_locked ? "locked" : "unlocked"), - end, num, tag); - h= my_open("wrong_page", O_CREAT | O_TRUNC | O_RDWR, MYF(0)); - my_pwrite(h, (byte*) buff, PAGE_SIZE, 0, MYF(0)); - my_close(h, MYF(0)); - goto err; - } - } - DBUG_RETURN(end); -err: - DBUG_PRINT("err", ("try to flush")); - if (page_locked) - { - pagecache_delete_page(&pagecache, &file1, page_no, - PAGECACHE_LOCK_LEFT_WRITELOCKED, 1); - } - else - { - flush_pagecache_blocks(&pagecache, &file1, FLUSH_RELEASE); - } - exit(1); -} - -void put_rec(uchar *buff, uint end, uint len, uint tag) -{ - uint i; - uint num= *((uint *)buff); - if (!len) - len= 1; - if (end + sizeof(uint)*2 + len > PAGE_SIZE) - return; - *((uint *)(buff + end))= len; - end+= sizeof(uint); - *((uint *)(buff + end))= tag; - end+= sizeof(uint); - num++; - *((uint *)buff)= num; - *((uint*)(buff + end))= len; - for (i= end; i < (len + end); i++) - { - buff[i]= (uchar) num % 256; - } -} - -/* - Recreate and reopen a file for test - - SYNOPSIS - reset_file() - file File to reset - file_name Path (and name) of file which should be reset -*/ - -void reset_file(PAGECACHE_FILE file, char *file_name) -{ - flush_pagecache_blocks(&pagecache, &file1, FLUSH_RELEASE); - if (my_close(file1.file, MYF(0)) != 0) - { - diag("Got error during %s closing from close() (errno: %d)\n", - file_name, errno); - exit(1); - } - my_delete(file_name, MYF(0)); - if ((file.file= my_open(file_name, - O_CREAT | O_TRUNC | O_RDWR, MYF(0))) == -1) - { - diag("Got error during %s creation from open() (errno: %d)\n", - file_name, errno); - exit(1); - } -} - - -void reader(int num) -{ - unsigned char *buffr= malloc(PAGE_SIZE); - uint i; - - for (i= 0; i < number_of_tests; i++) - { - uint page= get_len(number_of_pages); - pagecache_read(&pagecache, &file1, page, 3, (char*)buffr, - PAGECACHE_PLAIN_PAGE, - PAGECACHE_LOCK_LEFT_UNLOCKED, - 0); - check_page(buffr, page * PAGE_SIZE, 0, page, -num); - if (i % 500 == 0) - printf("reader%d: %d\n", num, i); - - } - printf("reader%d: done\n", num); - free(buffr); -} - - -void writer(int num) -{ - unsigned char *buffr= malloc(PAGE_SIZE); - uint i; - - for (i= 0; i < number_of_tests; i++) - { - uint end; - uint page= get_len(number_of_pages); - pagecache_read(&pagecache, &file1, page, 3, (char*)buffr, - PAGECACHE_PLAIN_PAGE, - PAGECACHE_LOCK_WRITE, - 0); - end= check_page(buffr, page * PAGE_SIZE, 1, page, num); - put_rec(buffr, end, get_len(record_length_limit), num); - pagecache_write(&pagecache, &file1, page, 3, (char*)buffr, - PAGECACHE_PLAIN_PAGE, - PAGECACHE_LOCK_WRITE_UNLOCK, - PAGECACHE_UNPIN, - PAGECACHE_WRITE_DELAY, - 0); - - if (i % flush_divider == 0) - flush_pagecache_blocks(&pagecache, &file1, FLUSH_FORCE_WRITE); - if (i % 500 == 0) - printf("writer%d: %d\n", num, i); - } - printf("writer%d: done\n", num); - free(buffr); -} - - -static void *test_thread_reader(void *arg) -{ - int param=*((int*) arg); - - my_thread_init(); - DBUG_ENTER("test_reader"); - DBUG_PRINT("enter", ("param: %d", param)); - - reader(param); - - DBUG_PRINT("info", ("Thread %s ended\n", my_thread_name())); - pthread_mutex_lock(&LOCK_thread_count); - thread_count--; - VOID(pthread_cond_signal(&COND_thread_count)); /* Tell main we are ready */ - pthread_mutex_unlock(&LOCK_thread_count); - free((gptr) arg); - my_thread_end(); - DBUG_RETURN(0); -} - -static void *test_thread_writer(void *arg) -{ - int param=*((int*) arg); - - my_thread_init(); - DBUG_ENTER("test_writer"); - DBUG_PRINT("enter", ("param: %d", param)); - - writer(param); - - DBUG_PRINT("info", ("Thread %s ended\n", my_thread_name())); - pthread_mutex_lock(&LOCK_thread_count); - thread_count--; - VOID(pthread_cond_signal(&COND_thread_count)); /* Tell main we are ready */ - pthread_mutex_unlock(&LOCK_thread_count); - free((gptr) arg); - my_thread_end(); - DBUG_RETURN(0); -} - -int main(int argc, char **argv __attribute__((unused))) -{ - pthread_t tid; - pthread_attr_t thr_attr; - int *param, error, pagen; - - MY_INIT(argv[0]); - -#ifndef DBUG_OFF -#if defined(__WIN__) - default_dbug_option= "d:t:i:O,\\test_pagecache_consist.trace"; -#else - default_dbug_option= "d:t:i:o,/tmp/test_pagecache_consist.trace"; -#endif - if (argc > 1) - { - DBUG_SET(default_dbug_option); - DBUG_SET_INITIAL(default_dbug_option); - } -#endif - - - DBUG_ENTER("main"); - DBUG_PRINT("info", ("Main thread: %s\n", my_thread_name())); - if ((file1.file= my_open(file1_name, - O_CREAT | O_TRUNC | O_RDWR, MYF(0))) == -1) - { - fprintf(stderr, "Got error during file1 creation from open() (errno: %d)\n", - errno); - exit(1); - } - DBUG_PRINT("info", ("file1: %d", file1.file)); - if (chmod(file1_name, S_IRWXU | S_IRWXG | S_IRWXO) != 0) - { - fprintf(stderr, "Got error during file1 chmod() (errno: %d)\n", - errno); - exit(1); - } - my_pwrite(file1.file, "test file", 9, 0, MYF(0)); - - if ((error= pthread_cond_init(&COND_thread_count, NULL))) - { - fprintf(stderr, "COND_thread_count: %d from pthread_cond_init (errno: %d)\n", - error, errno); - exit(1); - } - if ((error= pthread_mutex_init(&LOCK_thread_count, MY_MUTEX_INIT_FAST))) - { - fprintf(stderr, "LOCK_thread_count: %d from pthread_cond_init (errno: %d)\n", - error, errno); - exit(1); - } - - if ((error= pthread_attr_init(&thr_attr))) - { - fprintf(stderr,"Got error: %d from pthread_attr_init (errno: %d)\n", - error,errno); - exit(1); - } - if ((error= pthread_attr_setdetachstate(&thr_attr, PTHREAD_CREATE_DETACHED))) - { - fprintf(stderr, - "Got error: %d from pthread_attr_setdetachstate (errno: %d)\n", - error,errno); - exit(1); - } - -#ifdef HAVE_THR_SETCONCURRENCY - VOID(thr_setconcurrency(2)); -#endif - - if ((pagen= init_pagecache(&pagecache, PCACHE_SIZE, 0, 0, - PAGE_SIZE)) == 0) - { - fprintf(stderr,"Got error: init_pagecache() (errno: %d)\n", - errno); - exit(1); - } - DBUG_PRINT("info", ("Page cache %d pages", pagen)); - { - unsigned char *buffr= malloc(PAGE_SIZE); - uint i; - memset(buffr, '\0', PAGE_SIZE); - for (i= 0; i < number_of_pages; i++) - { - pagecache_write(&pagecache, &file1, i, 3, (char*)buffr, - PAGECACHE_PLAIN_PAGE, - PAGECACHE_LOCK_LEFT_UNLOCKED, - PAGECACHE_PIN_LEFT_UNPINNED, - PAGECACHE_WRITE_DELAY, - 0); - } - flush_pagecache_blocks(&pagecache, &file1, FLUSH_FORCE_WRITE); - free(buffr); - } - if ((error= pthread_mutex_lock(&LOCK_thread_count))) - { - fprintf(stderr,"LOCK_thread_count: %d from pthread_mutex_lock (errno: %d)\n", - error,errno); - exit(1); - } - while (number_of_readers != 0 || number_of_writers != 0) - { - if (number_of_readers != 0) - { - param=(int*) malloc(sizeof(int)); - *param= number_of_readers; - if ((error= pthread_create(&tid, &thr_attr, test_thread_reader, - (void*) param))) - { - fprintf(stderr,"Got error: %d from pthread_create (errno: %d)\n", - error,errno); - exit(1); - } - thread_count++; - number_of_readers--; - } - if (number_of_writers != 0) - { - param=(int*) malloc(sizeof(int)); - *param= number_of_writers; - if ((error= pthread_create(&tid, &thr_attr, test_thread_writer, - (void*) param))) - { - fprintf(stderr,"Got error: %d from pthread_create (errno: %d)\n", - error,errno); - exit(1); - } - thread_count++; - number_of_writers--; - } - } - DBUG_PRINT("info", ("Thread started")); - pthread_mutex_unlock(&LOCK_thread_count); - - pthread_attr_destroy(&thr_attr); - - /* wait finishing */ - if ((error= pthread_mutex_lock(&LOCK_thread_count))) - fprintf(stderr,"LOCK_thread_count: %d from pthread_mutex_lock\n",error); - while (thread_count) - { - if ((error= pthread_cond_wait(&COND_thread_count,&LOCK_thread_count))) - fprintf(stderr,"COND_thread_count: %d from pthread_cond_wait\n",error); - } - if ((error= pthread_mutex_unlock(&LOCK_thread_count))) - fprintf(stderr,"LOCK_thread_count: %d from pthread_mutex_unlock\n",error); - DBUG_PRINT("info", ("thread ended")); - - end_pagecache(&pagecache, 1); - DBUG_PRINT("info", ("Page cache ended")); - - if (my_close(file1.file, MYF(0)) != 0) - { - fprintf(stderr, "Got error during file1 closing from close() (errno: %d)\n", - errno); - exit(1); - } - /*my_delete(file1_name, MYF(0));*/ - my_end(0); - - DBUG_PRINT("info", ("file1 (%d) closed", file1.file)); - - DBUG_PRINT("info", ("Program end")); - - DBUG_RETURN(exit_status()); -} diff --git a/storage/maria/unittest/mf_pagecache_single.c b/storage/maria/unittest/mf_pagecache_single.c deleted file mode 100644 index 91cceee618d..00000000000 --- a/storage/maria/unittest/mf_pagecache_single.c +++ /dev/null @@ -1,580 +0,0 @@ -/* - TODO: use pthread_join instead of wait_for_thread_count_to_be_zero, like in - my_atomic-t.c (see BUG#22320). - Use diag() instead of fprintf(stderr). -*/ -#include -#include -#include -#include "test_file.h" -#include - -#define PCACHE_SIZE (PAGE_SIZE*1024*10) - -#ifndef DBUG_OFF -static const char* default_dbug_option; -#endif - -static char *file1_name= (char*)"page_cache_test_file_1"; -static PAGECACHE_FILE file1; -static pthread_cond_t COND_thread_count; -static pthread_mutex_t LOCK_thread_count; -static uint thread_count; -static PAGECACHE pagecache; - -/* - File contance descriptors -*/ -static struct file_desc simple_read_write_test_file[]= -{ - {PAGE_SIZE, '\1'}, - { 0, 0} -}; -static struct file_desc simple_read_change_write_read_test_file[]= -{ - {PAGE_SIZE/2, '\65'}, - {PAGE_SIZE/2, '\1'}, - { 0, 0} -}; -static struct file_desc simple_pin_test_file1[]= -{ - {PAGE_SIZE*2, '\1'}, - { 0, 0} -}; -static struct file_desc simple_pin_test_file2[]= -{ - {PAGE_SIZE/2, '\1'}, - {PAGE_SIZE/2, (unsigned char)129}, - {PAGE_SIZE, '\1'}, - { 0, 0} -}; -static struct file_desc simple_delete_forget_test_file[]= -{ - {PAGE_SIZE, '\1'}, - { 0, 0} -}; -static struct file_desc simple_delete_flush_test_file[]= -{ - {PAGE_SIZE, '\2'}, - { 0, 0} -}; - - -/* - Recreate and reopen a file for test - - SYNOPSIS - reset_file() - file File to reset - file_name Path (and name) of file which should be reset -*/ - -void reset_file(PAGECACHE_FILE file, char *file_name) -{ - flush_pagecache_blocks(&pagecache, &file1, FLUSH_RELEASE); - if (my_close(file1.file, MYF(0)) != 0) - { - diag("Got error during %s closing from close() (errno: %d)\n", - file_name, errno); - exit(1); - } - my_delete(file_name, MYF(0)); - if ((file.file= my_open(file_name, - O_CREAT | O_TRUNC | O_RDWR, MYF(0))) == -1) - { - diag("Got error during %s creation from open() (errno: %d)\n", - file_name, errno); - exit(1); - } -} - -/* - Write then read page, check file on disk -*/ - -int simple_read_write_test() -{ - unsigned char *buffw= malloc(PAGE_SIZE); - unsigned char *buffr= malloc(PAGE_SIZE); - int res; - DBUG_ENTER("simple_read_write_test"); - bfill(buffw, PAGE_SIZE, '\1'); - pagecache_write(&pagecache, &file1, 0, 3, (char*)buffw, - PAGECACHE_PLAIN_PAGE, - PAGECACHE_LOCK_LEFT_UNLOCKED, - PAGECACHE_PIN_LEFT_UNPINNED, - PAGECACHE_WRITE_DELAY, - 0); - pagecache_read(&pagecache, &file1, 0, 3, (char*)buffr, - PAGECACHE_PLAIN_PAGE, - PAGECACHE_LOCK_LEFT_UNLOCKED, - 0); - ok((res= test(memcmp(buffr, buffw, PAGE_SIZE) == 0)), - "Simple write-read page "); - flush_pagecache_blocks(&pagecache, &file1, FLUSH_FORCE_WRITE); - ok((res&= test(test_file(file1, file1_name, PAGE_SIZE, PAGE_SIZE, - simple_read_write_test_file))), - "Simple write-read page file"); - if (res) - reset_file(file1, file1_name); - free(buffw); - free(buffr); - DBUG_RETURN(res); -} - - -/* - Prepare page, then read (and lock), change (write new value and unlock), - then check the page in the cache and on the disk -*/ -int simple_read_change_write_read_test() -{ - unsigned char *buffw= malloc(PAGE_SIZE); - unsigned char *buffr= malloc(PAGE_SIZE); - int res; - DBUG_ENTER("simple_read_change_write_read_test"); - /* prepare the file */ - bfill(buffw, PAGE_SIZE, '\1'); - pagecache_write(&pagecache, &file1, 0, 3, (char*)buffw, - PAGECACHE_PLAIN_PAGE, - PAGECACHE_LOCK_LEFT_UNLOCKED, - PAGECACHE_PIN_LEFT_UNPINNED, - PAGECACHE_WRITE_DELAY, - 0); - flush_pagecache_blocks(&pagecache, &file1, FLUSH_FORCE_WRITE); - /* test */ - pagecache_read(&pagecache, &file1, 0, 3, (char*)buffw, - PAGECACHE_PLAIN_PAGE, - PAGECACHE_LOCK_WRITE, - 0); - bfill(buffw, PAGE_SIZE/2, '\65'); - pagecache_write(&pagecache, &file1, 0, 3, (char*)buffw, - PAGECACHE_PLAIN_PAGE, - PAGECACHE_LOCK_WRITE_UNLOCK, - PAGECACHE_UNPIN, - PAGECACHE_WRITE_DELAY, - 0); - - pagecache_read(&pagecache, &file1, 0, 3, (char*)buffr, - PAGECACHE_PLAIN_PAGE, - PAGECACHE_LOCK_LEFT_UNLOCKED, - 0); - ok((res= test(memcmp(buffr, buffw, PAGE_SIZE) == 0)), - "Simple read-change-write-read page "); - flush_pagecache_blocks(&pagecache, &file1, FLUSH_FORCE_WRITE); - ok((res&= test(test_file(file1, file1_name, PAGE_SIZE, PAGE_SIZE, - simple_read_change_write_read_test_file))), - "Simple read-change-write-read page file"); - if (res) - reset_file(file1, file1_name); - free(buffw); - free(buffr); - DBUG_RETURN(res); -} - - -/* - Prepare page, read page 0 (and pin) then write page 1 and page 0. - Flush the file (shold flush only page 1 and return 1 (page 0 is - still pinned). - Check file on the disk. - Unpin and flush. - Check file on the disk. -*/ -int simple_pin_test() -{ - unsigned char *buffw= malloc(PAGE_SIZE); - unsigned char *buffr= malloc(PAGE_SIZE); - int res; - DBUG_ENTER("simple_pin_test"); - /* prepare the file */ - bfill(buffw, PAGE_SIZE, '\1'); - pagecache_write(&pagecache, &file1, 0, 3, (char*)buffw, - PAGECACHE_PLAIN_PAGE, - PAGECACHE_LOCK_LEFT_UNLOCKED, - PAGECACHE_PIN_LEFT_UNPINNED, - PAGECACHE_WRITE_DELAY, - 0); - /* test */ - if (flush_pagecache_blocks(&pagecache, &file1, FLUSH_FORCE_WRITE)) - { - diag("error in flush_pagecache_blocks\n"); - exit(1); - } - pagecache_read(&pagecache, &file1, 0, 3, (char*)buffw, - PAGECACHE_PLAIN_PAGE, - PAGECACHE_LOCK_WRITE, - 0); - pagecache_write(&pagecache, &file1, 1, 3, (char*)buffw, - PAGECACHE_PLAIN_PAGE, - PAGECACHE_LOCK_LEFT_UNLOCKED, - PAGECACHE_PIN_LEFT_UNPINNED, - PAGECACHE_WRITE_DELAY, - 0); - bfill(buffw + PAGE_SIZE/2, PAGE_SIZE/2, ((unsigned char) 129)); - pagecache_write(&pagecache, &file1, 0, 3, (char*)buffw, - PAGECACHE_PLAIN_PAGE, - PAGECACHE_LOCK_WRITE_TO_READ, - PAGECACHE_PIN_LEFT_PINNED, - PAGECACHE_WRITE_DELAY, - 0); - /* - We have to get error because one page of the file is pinned, - other page should be flushed - */ - if (!flush_pagecache_blocks(&pagecache, &file1, FLUSH_FORCE_WRITE)) - { - diag("Did not get error in flush_pagecache_blocks\n"); - res= 0; - goto err; - } - ok((res= test(test_file(file1, file1_name, PAGE_SIZE*2, PAGE_SIZE*2, - simple_pin_test_file1))), - "Simple pin page file with pin"); - pagecache_unlock_page(&pagecache, - &file1, - 0, - PAGECACHE_LOCK_READ_UNLOCK, - PAGECACHE_UNPIN, - 0); - if (flush_pagecache_blocks(&pagecache, &file1, FLUSH_FORCE_WRITE)) - { - diag("Got error in flush_pagecache_blocks\n"); - res= 0; - goto err; - } - ok((res&= test(test_file(file1, file1_name, PAGE_SIZE*2, PAGE_SIZE, - simple_pin_test_file2))), - "Simple pin page result file"); - if (res) - reset_file(file1, file1_name); -err: - free(buffw); - free(buffr); - DBUG_RETURN(res); -} - -/* - Prepare page, write new value, then delete page from cache without flush, - on the disk should be page with old content written during preparation -*/ - -int simple_delete_forget_test() -{ - unsigned char *buffw= malloc(PAGE_SIZE); - unsigned char *buffr= malloc(PAGE_SIZE); - int res; - DBUG_ENTER("simple_delete_forget_test"); - /* prepare the file */ - bfill(buffw, PAGE_SIZE, '\1'); - pagecache_write(&pagecache, &file1, 0, 3, (char*)buffw, - PAGECACHE_PLAIN_PAGE, - PAGECACHE_LOCK_LEFT_UNLOCKED, - PAGECACHE_PIN_LEFT_UNPINNED, - PAGECACHE_WRITE_DELAY, - 0); - flush_pagecache_blocks(&pagecache, &file1, FLUSH_FORCE_WRITE); - /* test */ - bfill(buffw, PAGE_SIZE, '\2'); - pagecache_write(&pagecache, &file1, 0, 3, (char*)buffw, - PAGECACHE_PLAIN_PAGE, - PAGECACHE_LOCK_LEFT_UNLOCKED, - PAGECACHE_PIN_LEFT_UNPINNED, - PAGECACHE_WRITE_DELAY, - 0); - pagecache_delete_page(&pagecache, &file1, 0, - PAGECACHE_LOCK_WRITE, 0); - flush_pagecache_blocks(&pagecache, &file1, FLUSH_FORCE_WRITE); - ok((res= test(test_file(file1, file1_name, PAGE_SIZE, PAGE_SIZE, - simple_delete_forget_test_file))), - "Simple delete-forget page file"); - if (res) - reset_file(file1, file1_name); - free(buffw); - free(buffr); - DBUG_RETURN(res); -} - -/* - Prepare page with locking, write new content to the page, - delete page with flush and on existing lock, - check that page on disk contain new value. -*/ - -int simple_delete_flush_test() -{ - unsigned char *buffw= malloc(PAGE_SIZE); - unsigned char *buffr= malloc(PAGE_SIZE); - int res; - DBUG_ENTER("simple_delete_flush_test"); - /* prepare the file */ - bfill(buffw, PAGE_SIZE, '\1'); - pagecache_write(&pagecache, &file1, 0, 3, (char*)buffw, - PAGECACHE_PLAIN_PAGE, - PAGECACHE_LOCK_WRITE, - PAGECACHE_PIN, - PAGECACHE_WRITE_DELAY, - 0); - flush_pagecache_blocks(&pagecache, &file1, FLUSH_FORCE_WRITE); - /* test */ - bfill(buffw, PAGE_SIZE, '\2'); - pagecache_write(&pagecache, &file1, 0, 3, (char*)buffw, - PAGECACHE_PLAIN_PAGE, - PAGECACHE_LOCK_LEFT_WRITELOCKED, - PAGECACHE_PIN_LEFT_PINNED, - PAGECACHE_WRITE_DELAY, - 0); - pagecache_delete_page(&pagecache, &file1, 0, - PAGECACHE_LOCK_LEFT_WRITELOCKED, 1); - flush_pagecache_blocks(&pagecache, &file1, FLUSH_FORCE_WRITE); - ok((res= test(test_file(file1, file1_name, PAGE_SIZE, PAGE_SIZE, - simple_delete_flush_test_file))), - "Simple delete-forget page file"); - if (res) - reset_file(file1, file1_name); - free(buffw); - free(buffr); - DBUG_RETURN(res); -} - - -/* - write then read file bigger then cache -*/ - -int simple_big_test() -{ - unsigned char *buffw= (unsigned char *)malloc(PAGE_SIZE); - unsigned char *buffr= (unsigned char *)malloc(PAGE_SIZE); - struct file_desc *desc= - (struct file_desc *)malloc((PCACHE_SIZE/(PAGE_SIZE/2) + 1) * - sizeof(struct file_desc)); - int res, i; - DBUG_ENTER("simple_big_test"); - /* prepare the file twice larger then cache */ - for (i= 0; i < PCACHE_SIZE/(PAGE_SIZE/2); i++) - { - bfill(buffw, PAGE_SIZE, (unsigned char) (i & 0xff)); - desc[i].length= PAGE_SIZE; - desc[i].content= (i & 0xff); - pagecache_write(&pagecache, &file1, i, 3, (char*)buffw, - PAGECACHE_PLAIN_PAGE, - PAGECACHE_LOCK_LEFT_UNLOCKED, - PAGECACHE_PIN_LEFT_UNPINNED, - PAGECACHE_WRITE_DELAY, - 0); - } - desc[i].length= 0; - desc[i].content= '\0'; - ok(1, "Simple big file write"); - /* check written pages sequentally read */ - for (i= 0; i < PCACHE_SIZE/(PAGE_SIZE/2); i++) - { - int j; - pagecache_read(&pagecache, &file1, i, 3, (char*)buffr, - PAGECACHE_PLAIN_PAGE, - PAGECACHE_LOCK_LEFT_UNLOCKED, - 0); - for(j= 0; j < PAGE_SIZE; j++) - { - if (buffr[j] != (i & 0xff)) - { - diag("simple_big_test seq: page %u byte %u mismatch\n", i, j); - return 0; - } - } - } - ok(1, "simple big file sequentally read"); - /* chack random reads */ - for (i= 0; i < PCACHE_SIZE/(PAGE_SIZE); i++) - { - int j, page; - page= rand() % (PCACHE_SIZE/(PAGE_SIZE/2)); - pagecache_read(&pagecache, &file1, page, 3, (char*)buffr, - PAGECACHE_PLAIN_PAGE, - PAGECACHE_LOCK_LEFT_UNLOCKED, - 0); - for(j= 0; j < PAGE_SIZE; j++) - { - if (buffr[j] != (page & 0xff)) - { - diag("simple_big_test rnd: page %u byte %u mismatch\n", page, j); - return 0; - } - } - } - ok(1, "simple big file random read"); - flush_pagecache_blocks(&pagecache, &file1, FLUSH_FORCE_WRITE); - - ok((res= test(test_file(file1, file1_name, PCACHE_SIZE*2, PAGE_SIZE, - desc))), - "Simple big file"); - if (res) - reset_file(file1, file1_name); - free(buffw); - free(buffr); - DBUG_RETURN(res); -} -/* - Thread function -*/ - -static void *test_thread(void *arg) -{ - int param=*((int*) arg); - - my_thread_init(); - DBUG_ENTER("test_thread"); - - DBUG_PRINT("enter", ("param: %d", param)); - - if (!simple_read_write_test() || - !simple_read_change_write_read_test() || - !simple_pin_test() || - !simple_delete_forget_test() || - !simple_delete_flush_test() || - !simple_big_test()) - exit(1); - - DBUG_PRINT("info", ("Thread %s ended\n", my_thread_name())); - pthread_mutex_lock(&LOCK_thread_count); - thread_count--; - VOID(pthread_cond_signal(&COND_thread_count)); /* Tell main we are ready */ - pthread_mutex_unlock(&LOCK_thread_count); - free((gptr) arg); - my_thread_end(); - DBUG_RETURN(0); -} - - -int main(int argc, char **argv __attribute__((unused))) -{ - pthread_t tid; - pthread_attr_t thr_attr; - int *param, error, pagen; - - MY_INIT(argv[0]); - -#ifndef DBUG_OFF -#if defined(__WIN__) - default_dbug_option= "d:t:i:O,\\test_pagecache_single.trace"; -#else - default_dbug_option= "d:t:i:o,/tmp/test_pagecache_single.trace"; -#endif - if (argc > 1) - { - DBUG_SET(default_dbug_option); - DBUG_SET_INITIAL(default_dbug_option); - } -#endif - - - DBUG_ENTER("main"); - DBUG_PRINT("info", ("Main thread: %s\n", my_thread_name())); - if ((file1.file= my_open(file1_name, - O_CREAT | O_TRUNC | O_RDWR, MYF(0))) == -1) - { - fprintf(stderr, "Got error during file1 creation from open() (errno: %d)\n", - errno); - exit(1); - } - DBUG_PRINT("info", ("file1: %d", file1.file)); - if (chmod(file1_name, S_IRWXU | S_IRWXG | S_IRWXO) != 0) - { - fprintf(stderr, "Got error during file1 chmod() (errno: %d)\n", - errno); - exit(1); - } - my_pwrite(file1.file, "test file", 9, 0, MYF(0)); - - if ((error= pthread_cond_init(&COND_thread_count, NULL))) - { - fprintf(stderr, "Got error: %d from pthread_cond_init (errno: %d)\n", - error, errno); - exit(1); - } - if ((error= pthread_mutex_init(&LOCK_thread_count, MY_MUTEX_INIT_FAST))) - { - fprintf(stderr, "Got error: %d from pthread_cond_init (errno: %d)\n", - error, errno); - exit(1); - } - - if ((error= pthread_attr_init(&thr_attr))) - { - fprintf(stderr,"Got error: %d from pthread_attr_init (errno: %d)\n", - error,errno); - exit(1); - } - if ((error= pthread_attr_setdetachstate(&thr_attr, PTHREAD_CREATE_DETACHED))) - { - fprintf(stderr, - "Got error: %d from pthread_attr_setdetachstate (errno: %d)\n", - error,errno); - exit(1); - } - -#ifdef HAVE_THR_SETCONCURRENCY - VOID(thr_setconcurrency(2)); -#endif - - plan(12); - - if ((pagen= init_pagecache(&pagecache, PCACHE_SIZE, 0, 0, - PAGE_SIZE)) == 0) - { - fprintf(stderr,"Got error: init_pagecache() (errno: %d)\n", - errno); - exit(1); - } - DBUG_PRINT("info", ("Page cache %d pages", pagen)); - - if ((error=pthread_mutex_lock(&LOCK_thread_count))) - { - fprintf(stderr,"Got error: %d from pthread_mutex_lock (errno: %d)\n", - error,errno); - exit(1); - } - param=(int*) malloc(sizeof(int)); - *param= 1; - if ((error= pthread_create(&tid, &thr_attr, test_thread, (void*) param))) - { - fprintf(stderr,"Got error: %d from pthread_create (errno: %d)\n", - error,errno); - exit(1); - } - thread_count++; - DBUG_PRINT("info", ("Thread started")); - pthread_mutex_unlock(&LOCK_thread_count); - - pthread_attr_destroy(&thr_attr); - - if ((error= pthread_mutex_lock(&LOCK_thread_count))) - fprintf(stderr,"Got error: %d from pthread_mutex_lock\n",error); - while (thread_count) - { - if ((error= pthread_cond_wait(&COND_thread_count,&LOCK_thread_count))) - fprintf(stderr,"Got error: %d from pthread_cond_wait\n",error); - } - if ((error= pthread_mutex_unlock(&LOCK_thread_count))) - fprintf(stderr,"Got error: %d from pthread_mutex_unlock\n",error); - DBUG_PRINT("info", ("thread ended")); - - end_pagecache(&pagecache, 1); - DBUG_PRINT("info", ("Page cache ended")); - - if (my_close(file1.file, MYF(0)) != 0) - { - fprintf(stderr, "Got error during file1 closing from close() (errno: %d)\n", - errno); - exit(1); - } - /*my_delete(file1_name, MYF(0));*/ - my_end(0); - - DBUG_PRINT("info", ("file1 (%d) closed", file1.file)); - - DBUG_PRINT("info", ("Program end")); - - DBUG_RETURN(exit_status()); -} -- cgit v1.2.1 From 0abffa05a60e3fccb31a81828fdc99986ce4b7c5 Mon Sep 17 00:00:00 2001 From: unknown Date: Thu, 5 Apr 2007 14:38:05 +0300 Subject: Fixed that maria.test works BUILD/SETUP.sh: Update from 5.1 include/maria.h: Moved structs into size order mysql-test/include/varchar.inc: Fixed error numbers (as in 5.1) mysql-test/mysql-test-run.pl: Updated from 5.1 Create a dummy mysql.err file if using --valgrind --debug mysql-test/lib/init_db.sql: Update from 5.1 mysql-test/lib/mtr_cases.pl: Update from 5.1 mysql-test/lib/mtr_diff.pl: Update from 5.1 mysql-test/lib/mtr_gcov.pl: Update from 5.1 mysql-test/lib/mtr_gprof.pl: Update from 5.1 mysql-test/lib/mtr_im.pl: Update from 5.1 mysql-test/lib/mtr_io.pl: Update from 5.1 mysql-test/lib/mtr_match.pl: Update from 5.1 mysql-test/lib/mtr_misc.pl: Update from 5.1 mysql-test/lib/mtr_process.pl: Update from 5.1 mysql-test/lib/mtr_report.pl: Update from 5.1 mysql-test/lib/mtr_stress.pl: Update from 5.1 mysql-test/lib/mtr_timer.pl: Update from 5.1 mysql-test/lib/mtr_unique.pl: Update from 5.1 mysql-test/r/maria.result: Updated results. The reason for the new results are: - Maria doesn't support REPAIR TABLE or OPTIMIZE table yet - Some statistics information is different, so MySQL prefers index reads instead of table scans - No support for concurrent writes in the default BLOCK_RECORD mode - No support for different KEY_BLOCK sizes (will not be fixed) mysql-test/t/disabled.def: Enable maria test mysql-test/t/maria.test: No support for concurrent writes in the default BLOCK_RECORD mode No support for different KEY_BLOCK sizes (will not be fixed) mysql-test/t/myisam.test: Fix to be able to run with --extern mysql-test/t/query_cache_notembedded.test: Fix to be able to run with --extern sql/filesort.cc: Fixed compiler warning sql/handler.cc: Use new error message (as in 5.1) sql/share/errmsg.txt: Update error messages (as in 5.1) sql/slave.cc: Fixed compiler warning sql/slave.h: Fixed compiler warning sql/sql_table.cc: Fixed compiler warning storage/maria/ha_maria.cc: Added better scan_time() Disble REPAIR on BLOCK_RECORD tables Added rnd_end() to free memory after scan Don't pack numerical primary keys Don't allow fast alter table if row type changes storage/maria/ha_maria.h: Added get_row_type(), scan_time() and rnd_end() BitKeeper/etc/ignore: Added storage/maria/unittest/mf_pagecache_consist_1k-t-big storage/maria/unittest/mf_pagecache_consist_1kHC-t-big storage/maria/unittest/mf_pagecache_consist_1kRD-t-big storage/maria/unittest/mf_pagecache_consist_1kWR-t-big storage/maria/unittest/mf_pagecache_consist_64k-t-big storage/maria/unittest/mf_pagecache_consist_64kHC-t-big storage/maria/unittest/mf_pagecache_consist_64kRD-t-big storage/maria/unittest/mf_pagecache_consist_64kWR-t-big storage/maria/unittest/mf_pagecache_single_64k-t-big to the ignore list storage/maria/ma_bitmap.c: Fixed some bugs found with maria.test Added more DBUG_PRINT and some more comments storage/maria/ma_blockrec.c: Fixed some bugs found with maria.test Simplified code More comments storage/maria/ma_blockrec.h: Added DBUG_ASSERT() storage/maria/ma_check.c: Don't check record data links with block_records Update state.changed properly storage/maria/ma_checksum.c: Fixed bug in checksum handling (only first field was calculated) storage/maria/ma_create.c: Set rec->fill_length properly Added extra testing needed for BLOCK_RECORD Fixed bug in unlock of not locked mutex Fixed memory leak storage/maria/ma_delete.c: Update state.changed storage/maria/ma_delete_all.c: Update state.changed storage/maria/ma_extra.c: Disable caching of rows if we are using BLOCK_RECORD (scan_init will enable caching of rows when using BLOCK_RECORD) storage/maria/ma_info.c: Added data_file_type storage/maria/ma_search.c: Fixed bug with signed bytes storage/maria/ma_test2.c: Fixed wrong pointer handling (caused crash on 64 bit machines) storage/maria/ma_write.c: Added DBUG_ statements storage/maria/maria_def.h: Added STATE_NOT_OPTIMIZED_ROWS storage/myisam/mi_create.c: Fixed bug with unlocking of not locked mutex (in case of error condition) storage/myisam/mi_test2.c: Fixed wrong pointer handling (caused crash on 64 bit machines) --- .bzrignore | 9 + BUILD/SETUP.sh | 31 +- include/maria.h | 17 +- mysql-test/include/varchar.inc | 12 +- mysql-test/lib/init_db.sql | 14 + mysql-test/lib/mtr_cases.pl | 24 +- mysql-test/lib/mtr_diff.pl | 14 + mysql-test/lib/mtr_gcov.pl | 14 + mysql-test/lib/mtr_gprof.pl | 14 + mysql-test/lib/mtr_im.pl | 14 + mysql-test/lib/mtr_io.pl | 14 + mysql-test/lib/mtr_match.pl | 14 + mysql-test/lib/mtr_misc.pl | 14 + mysql-test/lib/mtr_process.pl | 17 +- mysql-test/lib/mtr_report.pl | 20 +- mysql-test/lib/mtr_stress.pl | 14 + mysql-test/lib/mtr_timer.pl | 14 + mysql-test/lib/mtr_unique.pl | 16 + mysql-test/mysql-test-run.pl | 317 +++++--- mysql-test/r/maria.result | 79 +- mysql-test/t/disabled.def | 1 - mysql-test/t/maria.test | 93 +-- mysql-test/t/myisam.test | 2 +- mysql-test/t/query_cache_notembedded.test | 4 +- sql/filesort.cc | 2 +- sql/handler.cc | 6 +- sql/share/errmsg.txt | 1141 +++++++++++++++-------------- sql/slave.cc | 2 + sql/slave.h | 2 - sql/sql_table.cc | 3 +- storage/maria/ha_maria.cc | 82 ++- storage/maria/ha_maria.h | 4 + storage/maria/ma_bitmap.c | 19 +- storage/maria/ma_blockrec.c | 269 +++++-- storage/maria/ma_blockrec.h | 3 +- storage/maria/ma_check.c | 15 +- storage/maria/ma_checksum.c | 5 +- storage/maria/ma_create.c | 77 +- storage/maria/ma_delete.c | 3 +- storage/maria/ma_delete_all.c | 1 + storage/maria/ma_extra.c | 19 +- storage/maria/ma_info.c | 6 +- storage/maria/ma_search.c | 9 +- storage/maria/ma_test2.c | 3 +- storage/maria/ma_write.c | 2 + storage/maria/maria_def.h | 1 + storage/myisam/mi_create.c | 21 +- storage/myisam/mi_test2.c | 3 +- 48 files changed, 1536 insertions(+), 944 deletions(-) diff --git a/.bzrignore b/.bzrignore index 7dd012950d8..6f579b34311 100644 --- a/.bzrignore +++ b/.bzrignore @@ -2961,3 +2961,12 @@ unittest/mysys/mf_pagecache_consist_64kHC-t-big unittest/mysys/mf_pagecache_consist_64kRD-t-big unittest/mysys/mf_pagecache_consist_64kWR-t-big unittest/mysys/mf_pagecache_single_64k-t-big +storage/maria/unittest/mf_pagecache_consist_1k-t-big +storage/maria/unittest/mf_pagecache_consist_1kHC-t-big +storage/maria/unittest/mf_pagecache_consist_1kRD-t-big +storage/maria/unittest/mf_pagecache_consist_1kWR-t-big +storage/maria/unittest/mf_pagecache_consist_64k-t-big +storage/maria/unittest/mf_pagecache_consist_64kHC-t-big +storage/maria/unittest/mf_pagecache_consist_64kRD-t-big +storage/maria/unittest/mf_pagecache_consist_64kWR-t-big +storage/maria/unittest/mf_pagecache_single_64k-t-big diff --git a/BUILD/SETUP.sh b/BUILD/SETUP.sh index e2049c2ca79..11dd67d5de0 100755 --- a/BUILD/SETUP.sh +++ b/BUILD/SETUP.sh @@ -92,9 +92,13 @@ if [ "x$warning_mode" != "xpedantic" ]; then # Both C and C++ warnings warnings="-Wimplicit -Wreturn-type -Wswitch -Wtrigraphs -Wcomment -W" warnings="$warnings -Wchar-subscripts -Wformat -Wparentheses -Wsign-compare" - warnings="$warnings -Wwrite-strings" + warnings="$warnings -Wwrite-strings -Wunused-function -Wunused-label -Wunused-value -Wunused-variable" + +# For more warnings, uncomment the following line +# warnings="$global_warnings -Wshadow" + # C warnings - c_warnings="$warnings -Wunused" + c_warnings="$warnings -Wunused-parameter" # C++ warnings cxx_warnings="$warnings -Woverloaded-virtual -Wsign-promo -Wreorder" cxx_warnings="$warnings -Wctor-dtor-privacy -Wnon-virtual-dtor" @@ -117,9 +121,8 @@ valgrind_flags="-USAFEMALLOC -UFORCE_INIT_OF_VARS -DHAVE_purify " valgrind_flags="$valgrind_flags -DMYSQL_SERVER_SUFFIX=-valgrind-max" # # Used in -debug builds -debug_cflags="-DUNIV_MUST_NOT_INLINE -DEXTRA_DEBUG -DFORCE_INIT_OF_VARS" +debug_cflags="-DUNIV_MUST_NOT_INLINE -DEXTRA_DEBUG -DFORCE_INIT_OF_VARS " debug_cflags="$debug_cflags -DSAFEMALLOC -DPEDANTIC_SAFEMALLOC -DSAFE_MUTEX" -debug_cflags="$debug_cflags -DMY_LF_EXTRA_DEBUG" error_inject="--with-error-inject " # # Base C++ flags for all builds @@ -140,9 +143,19 @@ fi # base_configs="--prefix=$prefix --enable-assembler " base_configs="$base_configs --with-extra-charsets=complex " -base_configs="$base_configs --enable-thread-safe-client --with-readline " +base_configs="$base_configs --enable-thread-safe-client " base_configs="$base_configs --with-big-tables" +if test -d "$path/../cmd-line-utils/readline" +then + base_configs="$base_configs --with-readline" +elif test -d "$path/../cmd-line-utils/libedit" +then + base_configs="$base_configs --with-libedit" +fi + +static_link="--with-mysqld-ldflags=-all-static " +static_link="$static_link --with-client-ldflags=-all-static" # we need local-infile in all binaries for rpl000001 # if you need to disable local-infile in the client, write a build script # and unset local_infile_configs @@ -182,7 +195,13 @@ fi # (http://samba.org/ccache) is installed, use it. # We use 'grep' and hope 'grep' will work as expected # (returns 0 if finds lines) -if ccache -V > /dev/null 2>&1 && test "$CCACHE_GCOV_VERSION_ENABLED" == "1" +if test "$USING_GCOV" != "1" +then + # Not using gcov; Safe to use ccache + CCACHE_GCOV_VERSION_ENABLED=1 +fi + +if ccache -V > /dev/null 2>&1 && test "$CCACHE_GCOV_VERSION_ENABLED" = "1" then echo "$CC" | grep "ccache" > /dev/null || CC="ccache $CC" echo "$CXX" | grep "ccache" > /dev/null || CXX="ccache $CXX" diff --git a/include/maria.h b/include/maria.h index 94152ce9bfa..a8c5cbf794b 100644 --- a/include/maria.h +++ b/include/maria.h @@ -122,22 +122,23 @@ typedef struct st_maria_isaminfo /* Struct from h_info */ my_off_t data_file_length; /* Length of data file */ my_off_t max_data_file_length, index_file_length; my_off_t max_index_file_length, delete_length; - ulong reclength; /* Recordlength */ - ulong mean_reclength; /* Mean recordlength (if packed) */ ulonglong auto_increment; ulonglong key_map; /* Which keys are used */ + time_t create_time; /* When table was created */ + time_t check_time; + time_t update_time; + ulong record_offset; + ulong *rec_per_key; /* for sql optimizing */ + ulong reclength; /* Recordlength */ + ulong mean_reclength; /* Mean recordlength (if packed) */ char *data_file_name, *index_file_name; + enum data_file_type data_file_type; uint keys; /* Number of keys in use */ uint options; /* HA_OPTION_... used */ + uint reflength; int errkey, /* With key was dupplicated on err */ sortkey; /* clustered by this key */ File filenr; /* (uniq) filenr for datafile */ - time_t create_time; /* When table was created */ - time_t check_time; - time_t update_time; - uint reflength; - ulong record_offset; - ulong *rec_per_key; /* for sql optimizing */ } MARIA_INFO; diff --git a/mysql-test/include/varchar.inc b/mysql-test/include/varchar.inc index 70b563e871c..1cc2090ff8c 100644 --- a/mysql-test/include/varchar.inc +++ b/mysql-test/include/varchar.inc @@ -81,7 +81,7 @@ explain select count(*) from t1 where v between 'a' and 'a '; --replace_column 9 # explain select count(*) from t1 where v between 'a' and 'a ' and v between 'a ' and 'b\n'; ---error 1062 +--error 1582 alter table t1 add unique(v); alter table t1 add key(v); select concat('*',v,'*',c,'*',t,'*') as qq from t1 where v='a'; @@ -179,17 +179,17 @@ drop table t1; create table t1 (a char(10), unique (a)); insert into t1 values ('a '); ---error 1062 +--error 1582 insert into t1 values ('a '); alter table t1 modify a varchar(10); ---error 1062 +--error 1582 insert into t1 values ('a '),('a '),('a '),('a '); ---error 1062 +--error 1582 insert into t1 values ('a '); ---error 1062 +--error 1582 insert into t1 values ('a '); ---error 1062 +--error 1582 insert into t1 values ('a '); update t1 set a='a ' where a like 'a%'; select concat(a,'.') from t1; diff --git a/mysql-test/lib/init_db.sql b/mysql-test/lib/init_db.sql index fcef17eab9b..c6c0f1d81dd 100644 --- a/mysql-test/lib/init_db.sql +++ b/mysql-test/lib/init_db.sql @@ -114,6 +114,20 @@ REPLACE INTO user VALUES ('127.0.0.1' ,'root','','Y','Y','Y','Y','Y','Y','Y','Y INSERT INTO user (host,user) VALUES ('localhost',''); INSERT INTO user (host,user) VALUES ('@HOSTNAME@%',''); +CREATE TABLE servers ( + Server_name char(64) NOT NULL DEFAULT '', + Host char(64) NOT NULL DEFAULT '', + Db char(64) NOT NULL DEFAULT '', + Username char(64) NOT NULL DEFAULT '', + Password char(64) NOT NULL DEFAULT '', + Port INT(4) NOT NULL DEFAULT '0', + Socket char(64) NOT NULL DEFAULT '', + Wrapper char(64) NOT NULL DEFAULT '', + Owner char(64) NOT NULL DEFAULT '', + PRIMARY KEY (Server_name)) + CHARACTER SET utf8 comment='MySQL Foreign Servers table'; + +INSERT INTO servers VALUES ('test','localhost','test','root','', 0,'','mysql','root'); CREATE TABLE func ( name char(64) binary DEFAULT '' NOT NULL, diff --git a/mysql-test/lib/mtr_cases.pl b/mysql-test/lib/mtr_cases.pl index 9e943fec9ef..22290a88d39 100644 --- a/mysql-test/lib/mtr_cases.pl +++ b/mysql-test/lib/mtr_cases.pl @@ -1,4 +1,18 @@ # -*- cperl -*- +# Copyright (C) 2005-2006 MySQL AB +# +# 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 St, Fifth Floor, Boston, MA 02110-1301 USA # This is a library file used by the Perl version of mysql-test-run, # and is part of the translation of the Bourne shell script with the @@ -288,6 +302,7 @@ sub collect_one_test_case($$$$$$$) { $tinfo->{'timezone'}= "GMT-3"; # for UNIX_TIMESTAMP tests to work $tinfo->{'slave_num'}= 0; # Default, no slave + $tinfo->{'master_num'}= 1; # Default, 1 master if ( defined mtr_match_prefix($tname,"rpl") ) { if ( $::opt_skip_rpl ) @@ -297,13 +312,8 @@ sub collect_one_test_case($$$$$$$) { return; } - $tinfo->{'slave_num'}= 1; # Default for rpl* tests, use one slave - if ( $tname eq 'rpl_failsafe' or $tname eq 'rpl_chain_temp_table' ) - { - # $tinfo->{'slave_num'}= 3; # Not 3 ? Check old code, strange - } } if ( defined mtr_match_prefix($tname,"federated") ) @@ -541,6 +551,8 @@ sub collect_one_test_case($$$$$$$) { $tinfo->{'comment'}= "No ndbcluster tests(--skip-ndbcluster)"; return; } + # Ndb tests run with two mysqld masters + $tinfo->{'master_num'}= 2; } else { @@ -556,7 +568,7 @@ sub collect_one_test_case($$$$$$$) { if ( $tinfo->{'innodb_test'} ) { - # This is a test that need inndob + # This is a test that need innodb if ( $::mysqld_variables{'innodb'} eq "FALSE" ) { # innodb is not supported, skip it diff --git a/mysql-test/lib/mtr_diff.pl b/mysql-test/lib/mtr_diff.pl index 4e927ff4e37..26e556de5e8 100644 --- a/mysql-test/lib/mtr_diff.pl +++ b/mysql-test/lib/mtr_diff.pl @@ -1,4 +1,18 @@ # -*- cperl -*- +# Copyright (C) 2005 MySQL AB +# +# 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 St, Fifth Floor, Boston, MA 02110-1301 USA # This is a library file used by the Perl version of mysql-test-run, # and is part of the translation of the Bourne shell script with the diff --git a/mysql-test/lib/mtr_gcov.pl b/mysql-test/lib/mtr_gcov.pl index 71d3d6a2a43..a2de1fcbdff 100644 --- a/mysql-test/lib/mtr_gcov.pl +++ b/mysql-test/lib/mtr_gcov.pl @@ -1,4 +1,18 @@ # -*- cperl -*- +# Copyright (C) 2004, 2006 MySQL AB +# +# 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 St, Fifth Floor, Boston, MA 02110-1301 USA # This is a library file used by the Perl version of mysql-test-run, # and is part of the translation of the Bourne shell script with the diff --git a/mysql-test/lib/mtr_gprof.pl b/mysql-test/lib/mtr_gprof.pl index cc874eebfe5..f6615301dd7 100644 --- a/mysql-test/lib/mtr_gprof.pl +++ b/mysql-test/lib/mtr_gprof.pl @@ -1,4 +1,18 @@ # -*- cperl -*- +# Copyright (C) 2004 MySQL AB +# +# 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 St, Fifth Floor, Boston, MA 02110-1301 USA # This is a library file used by the Perl version of mysql-test-run, # and is part of the translation of the Bourne shell script with the diff --git a/mysql-test/lib/mtr_im.pl b/mysql-test/lib/mtr_im.pl index 967e92dfcdd..c8e332498d7 100644 --- a/mysql-test/lib/mtr_im.pl +++ b/mysql-test/lib/mtr_im.pl @@ -1,4 +1,18 @@ # -*- cperl -*- +# Copyright (C) 2006 MySQL AB +# +# 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 St, Fifth Floor, Boston, MA 02110-1301 USA # This is a library file used by the Perl version of mysql-test-run, # and is part of the translation of the Bourne shell script with the diff --git a/mysql-test/lib/mtr_io.pl b/mysql-test/lib/mtr_io.pl index 842dc09413d..570a58875c2 100644 --- a/mysql-test/lib/mtr_io.pl +++ b/mysql-test/lib/mtr_io.pl @@ -1,4 +1,18 @@ # -*- cperl -*- +# Copyright (C) 2004-2006 MySQL AB +# +# 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 St, Fifth Floor, Boston, MA 02110-1301 USA # This is a library file used by the Perl version of mysql-test-run, # and is part of the translation of the Bourne shell script with the diff --git a/mysql-test/lib/mtr_match.pl b/mysql-test/lib/mtr_match.pl index 66b639c7f8e..96aa43f4fa2 100644 --- a/mysql-test/lib/mtr_match.pl +++ b/mysql-test/lib/mtr_match.pl @@ -1,4 +1,18 @@ # -*- cperl -*- +# Copyright (C) 2004-2006 MySQL AB +# +# 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 St, Fifth Floor, Boston, MA 02110-1301 USA # This is a library file used by the Perl version of mysql-test-run, # and is part of the translation of the Bourne shell script with the diff --git a/mysql-test/lib/mtr_misc.pl b/mysql-test/lib/mtr_misc.pl index f28e25e5966..97b0a6fc131 100644 --- a/mysql-test/lib/mtr_misc.pl +++ b/mysql-test/lib/mtr_misc.pl @@ -1,4 +1,18 @@ # -*- cperl -*- +# Copyright (C) 2004-2006 MySQL AB +# +# 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 St, Fifth Floor, Boston, MA 02110-1301 USA # This is a library file used by the Perl version of mysql-test-run, # and is part of the translation of the Bourne shell script with the diff --git a/mysql-test/lib/mtr_process.pl b/mysql-test/lib/mtr_process.pl index 330dfddf1d3..f63009cd24c 100644 --- a/mysql-test/lib/mtr_process.pl +++ b/mysql-test/lib/mtr_process.pl @@ -1,4 +1,18 @@ # -*- cperl -*- +# Copyright (C) 2004-2006 MySQL AB +# +# 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 St, Fifth Floor, Boston, MA 02110-1301 USA # This is a library file used by the Perl version of mysql-test-run, # and is part of the translation of the Bourne shell script with the @@ -220,7 +234,8 @@ sub spawn_parent_impl { my $ret_pid= waitpid($pid,0); if ( $ret_pid != $pid ) { - mtr_error("$path ($pid) got lost somehow"); + mtr_error("waitpid($pid, 0) returned $ret_pid " . + "when waiting for '$path'"); } return mtr_process_exit_status($?); diff --git a/mysql-test/lib/mtr_report.pl b/mysql-test/lib/mtr_report.pl index f840c1275d3..78986674c62 100644 --- a/mysql-test/lib/mtr_report.pl +++ b/mysql-test/lib/mtr_report.pl @@ -1,4 +1,18 @@ # -*- cperl -*- +# Copyright (C) 2004-2006 MySQL AB +# +# 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 St, Fifth Floor, Boston, MA 02110-1301 USA # This is a library file used by the Perl version of mysql-test-run, # and is part of the translation of the Bourne shell script with the @@ -222,8 +236,10 @@ sub mtr_report_stats ($) { "the documentation at\n", "http://www.mysql.com/doc/en/MySQL_test_suite.html\n"; } - print - "The servers were restarted $tot_restarts times\n"; + if (!$::opt_extern) + { + print "The servers where restarted $tot_restarts times\n"; + } if ( $::opt_timer ) { diff --git a/mysql-test/lib/mtr_stress.pl b/mysql-test/lib/mtr_stress.pl index 1371eaa44c6..93b06b32c5f 100644 --- a/mysql-test/lib/mtr_stress.pl +++ b/mysql-test/lib/mtr_stress.pl @@ -1,4 +1,18 @@ # -*- cperl -*- +# Copyright (C) 2006 MySQL AB +# +# 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 St, Fifth Floor, Boston, MA 02110-1301 USA # This is a library file used by the Perl version of mysql-test-run, # and is part of the translation of the Bourne shell script with the diff --git a/mysql-test/lib/mtr_timer.pl b/mysql-test/lib/mtr_timer.pl index 06374716c62..523799f7cf5 100644 --- a/mysql-test/lib/mtr_timer.pl +++ b/mysql-test/lib/mtr_timer.pl @@ -1,4 +1,18 @@ # -*- cperl -*- +# Copyright (C) 2005-2006 MySQL AB +# +# 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 St, Fifth Floor, Boston, MA 02110-1301 USA # This is a library file used by the Perl version of mysql-test-run, # and is part of the translation of the Bourne shell script with the diff --git a/mysql-test/lib/mtr_unique.pl b/mysql-test/lib/mtr_unique.pl index 3adc8413576..a668fc097c7 100644 --- a/mysql-test/lib/mtr_unique.pl +++ b/mysql-test/lib/mtr_unique.pl @@ -1,3 +1,19 @@ +# -*- cperl -*- +# Copyright (C) 2006 MySQL AB +# +# 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 St, Fifth Floor, Boston, MA 02110-1301 USA + # # This file is used from mysql-test-run.pl when choosing # port numbers and directories to use for running mysqld. diff --git a/mysql-test/mysql-test-run.pl b/mysql-test/mysql-test-run.pl index 73755cd2df9..7aec7ba0c77 100755 --- a/mysql-test/mysql-test-run.pl +++ b/mysql-test/mysql-test-run.pl @@ -58,6 +58,7 @@ $Devel::Trace::TRACE= 0; # Don't trace boring init stuff use File::Path; use File::Basename; use File::Copy; +use File::Temp qw / tempdir /; use Cwd; use Getopt::Long; use Sys::Hostname; @@ -159,6 +160,7 @@ our $exe_im; our $exe_my_print_defaults; our $exe_perror; our $lib_udf_example; +our $lib_example_plugin; our $exe_libtool; our $opt_bench= 0; @@ -230,10 +232,12 @@ our $opt_ndbconnectstring_slave; our $opt_record; my $opt_report_features; our $opt_check_testcases; +our $opt_mark_progress; our $opt_skip; our $opt_skip_rpl; our $max_slave_num= 0; +our $max_master_num= 1; our $use_innodb; our $opt_skip_test; our $opt_skip_im; @@ -306,6 +310,8 @@ our %mysqld_variables; my $source_dist= 0; +our $opt_max_save_core= 5; +my $num_saved_cores= 0; # Number of core files saved in vardir/log/ so far. ###################################################################### # @@ -403,14 +409,23 @@ sub main () { { next if $test->{skip}; - $need_ndbcluster||= $test->{ndb_test}; - $need_im||= $test->{component_id} eq 'im'; - - # Count max number of slaves used by a test case - if ( $test->{slave_num} > $max_slave_num) + if (!$opt_extern) { - $max_slave_num= $test->{slave_num}; - mtr_error("Too many slaves") if $max_slave_num > 3; + $need_ndbcluster||= $test->{ndb_test}; + $need_im||= $test->{component_id} eq 'im'; + + # Count max number of slaves used by a test case + if ( $test->{slave_num} > $max_slave_num) { + $max_slave_num= $test->{slave_num}; + mtr_error("Too many slaves") if $max_slave_num > 3; + } + + # Count max number of masters used by a test case + if ( $test->{master_num} > $max_master_num) { + $max_master_num= $test->{master_num}; + mtr_error("Too many masters") if $max_master_num > 2; + mtr_error("Too few masters") if $max_master_num < 1; + } } $use_innodb||= $test->{'innodb_test'}; } @@ -535,6 +550,7 @@ sub command_line_setup () { # Test case authoring 'record' => \$opt_record, 'check-testcases' => \$opt_check_testcases, + 'mark-progress' => \$opt_mark_progress, # Extra options used when starting mysqld 'mysqld=s' => \@opt_extra_mysqld_opt, @@ -556,6 +572,7 @@ sub command_line_setup () { 'strace-client' => \$opt_strace_client, 'master-binary=s' => \$exe_master_mysqld, 'slave-binary=s' => \$exe_slave_mysqld, + 'max-save-core=i' => \$opt_max_save_core, # Coverage, profiling etc 'gcov' => \$opt_gcov, @@ -683,7 +700,9 @@ sub command_line_setup () { "$glob_basedir/client", "$glob_basedir/bin"); - $exe_mysqld= mtr_exe_exists (vs_config_dirs('sql', 'mysqld'), + if (!$opt_extern) + { + $exe_mysqld= mtr_exe_exists (vs_config_dirs('sql', 'mysqld'), "$glob_basedir/sql/mysqld", "$path_client_bindir/mysqld-max-nt", "$path_client_bindir/mysqld-max", @@ -695,8 +714,16 @@ sub command_line_setup () { "$glob_basedir/bin/mysqld", "$glob_basedir/sbin/mysqld"); - # Use the mysqld found above to find out what features are available - collect_mysqld_features(); + # Use the mysqld found above to find out what features are available + collect_mysqld_features(); + } + else + { + $mysqld_variables{'port'}= 3306; + $mysqld_variables{'master-port'}= 3306; + $opt_skip_ndbcluster= 1; + $opt_skip_im= 1; + } if ( $opt_comment ) { @@ -733,7 +760,7 @@ sub command_line_setup () { # -------------------------------------------------------------------------- # NOTE if the default binlog format is changed, this has to be changed $used_binlog_format= "stmt"; - if ( $mysql_version_id >= 50100 ) + if (!$opt_extern && $mysql_version_id >= 50100 ) { $used_binlog_format= "mixed"; # Default value for binlog format @@ -819,19 +846,20 @@ sub command_line_setup () { # -------------------------------------------------------------------------- # Check im suport # -------------------------------------------------------------------------- - if ( $mysql_version_id < 50000 ) + if (!$opt_extern) { - # Instance manager is not supported until 5.0 - $opt_skip_im= 1; + if ( $mysql_version_id < 50000 ) { + # Instance manager is not supported until 5.0 + $opt_skip_im= 1; - } + } - if ( $glob_win32 ) - { - mtr_report("Disable Instance manager - not supported on Windows"); - $opt_skip_im= 1; - } + if ( $glob_win32 ) { + mtr_report("Disable Instance manager - not supported on Windows"); + $opt_skip_im= 1; + } + } # -------------------------------------------------------------------------- # Record flag # -------------------------------------------------------------------------- @@ -1020,6 +1048,11 @@ sub command_line_setup () { my $sockdir = $opt_tmpdir; $sockdir =~ s|/+$||; + # On some operating systems, there is a limit to the length of a + # UNIX domain socket's path far below PATH_MAX, so try to avoid long + # socket path names. + $sockdir = tempdir(CLEANUP => 0) if ( length($sockdir) > 80 ); + # Put this into a hash, will be a C struct $master->[0]= @@ -1190,7 +1223,7 @@ sub command_line_setup () { $opt_skip_rpl= 1; # Setup master->[0] with the settings for the extern server - $master->[0]->{'path_sock'}= $opt_socket if $opt_socket; + $master->[0]->{'path_sock'}= $opt_socket ? $opt_socket : "/tmp/mysql.sock"; mtr_report("Using extern server at '$master->[0]->{path_sock}'"); } else @@ -1253,9 +1286,10 @@ sub set_mtr_build_thread_ports($) { sub datadir_list_setup () { # Make a list of all data_dirs - @data_dir_lst = ( - $master->[0]->{'path_myddir'}, - $master->[1]->{'path_myddir'}); + for (my $idx= 0; $idx < $max_master_num; $idx++) + { + push(@data_dir_lst, $master->[$idx]->{'path_myddir'}); + } for (my $idx= 0; $idx < $max_slave_num; $idx++) { @@ -1442,56 +1476,66 @@ sub executable_setup () { $exe_mysqlbinlog= mtr_exe_exists("$path_client_bindir/mysqlbinlog"); $exe_mysqladmin= mtr_exe_exists("$path_client_bindir/mysqladmin"); $exe_mysql= mtr_exe_exists("$path_client_bindir/mysql"); - if ( $mysql_version_id >= 50100 ) - { - $exe_mysqlslap= mtr_exe_exists("$path_client_bindir/mysqlslap"); - } - if ( $mysql_version_id >= 50000 and !$glob_use_embedded_server ) - { - $exe_mysql_upgrade= mtr_exe_exists("$path_client_bindir/mysql_upgrade") - } - else - { - $exe_mysql_upgrade= ""; - } - if ( ! $glob_win32 ) + if (!$opt_extern) { - # Look for mysql_fix_system_table script - $exe_mysql_fix_system_tables= - mtr_script_exists("$glob_basedir/scripts/mysql_fix_privilege_tables", - "$path_client_bindir/mysql_fix_privilege_tables"); - } + if ( $mysql_version_id >= 50100 ) + { + $exe_mysqlslap= mtr_exe_exists("$path_client_bindir/mysqlslap"); + } + if ( $mysql_version_id >= 50000 and !$glob_use_embedded_server ) + { + $exe_mysql_upgrade= mtr_exe_exists("$path_client_bindir/mysql_upgrade") + } + else + { + $exe_mysql_upgrade= ""; + } - # Look for mysql_fix_privilege_tables.sql script - $file_mysql_fix_privilege_tables= - mtr_file_exists("$glob_basedir/scripts/mysql_fix_privilege_tables.sql", - "$glob_basedir/share/mysql_fix_privilege_tables.sql"); + if ( ! $glob_win32 ) + { + # Look for mysql_fix_system_table script + $exe_mysql_fix_system_tables= + mtr_script_exists("$glob_basedir/scripts/mysql_fix_privilege_tables", + "$path_client_bindir/mysql_fix_privilege_tables"); + } - if ( ! $opt_skip_ndbcluster and executable_setup_ndb()) - { - mtr_warning("Could not find all required ndb binaries, " . - "all ndb tests will fail, use --skip-ndbcluster to " . - "skip testing it."); + # Look for mysql_fix_privilege_tables.sql script + $file_mysql_fix_privilege_tables= + mtr_file_exists("$glob_basedir/scripts/mysql_fix_privilege_tables.sql", + "$glob_basedir/share/mysql_fix_privilege_tables.sql"); - foreach my $cluster (@{$clusters}) + if ( ! $opt_skip_ndbcluster and executable_setup_ndb()) { - $cluster->{"executable_setup_failed"}= 1; + mtr_warning("Could not find all required ndb binaries, " . + "all ndb tests will fail, use --skip-ndbcluster to " . + "skip testing it."); + + foreach my $cluster (@{$clusters}) + { + $cluster->{"executable_setup_failed"}= 1; + } } - } - if ( ! $opt_skip_im and executable_setup_im()) - { - mtr_warning("Could not find all required instance manager binaries, " . - "all im tests will fail, use --skip-im to " . - "continue without instance manager"); - $instance_manager->{"executable_setup_failed"}= 1; - } + if ( ! $opt_skip_im and executable_setup_im()) + { + mtr_warning("Could not find all required instance manager binaries, " . + "all im tests will fail, use --skip-im to " . + "continue without instance manager"); + $instance_manager->{"executable_setup_failed"}= 1; + } + + # Look for the udf_example library + $lib_udf_example= + mtr_file_exists(vs_config_dirs('sql', 'udf_example.dll'), + "$glob_basedir/sql/.libs/udf_example.so",); + + # Look for the ha_example library + $lib_example_plugin= + mtr_file_exists(vs_config_dirs('storage/example', 'ha_example.dll'), + "$glob_basedir/storage/example/.libs/ha_example.so",); - # Look for the udf_example library - $lib_udf_example= - mtr_file_exists(vs_config_dirs('sql', 'udf_example.dll'), - "$glob_basedir/sql/.libs/udf_example.so",); + } # Look for mysqltest executable if ( $glob_use_embedded_server ) @@ -1558,7 +1602,7 @@ sub mysql_client_test_arguments() mtr_add_arg($args, "--port=$master->[0]->{'port'}"); mtr_add_arg($args, "--socket=$master->[0]->{'path_sock'}"); - if ( $mysql_version_id >= 50000 ) + if ( $opt_extern || $mysql_version_id >= 50000 ) { mtr_add_arg($args, "--vardir=$opt_vardir") } @@ -1648,6 +1692,14 @@ sub environment_setup () { push(@ld_library_paths, dirname($lib_udf_example)); } + # -------------------------------------------------------------------------- + # Add the path where mysqld will find ha_example.so + # -------------------------------------------------------------------------- + if ( $lib_example_plugin ) + { + push(@ld_library_paths, dirname($lib_example_plugin)); + } + # -------------------------------------------------------------------------- # Valgrind need to be run with debug libraries otherwise it's almost # impossible to add correct supressions, that means if "/usr/lib/debug" @@ -1669,7 +1721,7 @@ sub environment_setup () { } $ENV{'LD_LIBRARY_PATH'}= join(":", @ld_library_paths, - $ENV{'LD_LIBRARY_PATHS'} ? + $ENV{'LD_LIBRARY_PATH'} ? split(':', $ENV{'LD_LIBRARY_PATH'}) : ()); mtr_debug("LD_LIBRARY_PATH: $ENV{'LD_LIBRARY_PATH'}"); @@ -1849,8 +1901,8 @@ sub environment_setup () { # ---------------------------------------------------- my $cmdline_mysqlbinlog= "$exe_mysqlbinlog" . - " --no-defaults --debug-info --local-load=$opt_tmpdir"; - if ( $mysql_version_id >= 50000 ) + " --no-defaults --disable-force-if-open --debug-info --local-load=$opt_tmpdir"; + if ( !$opt_extern && $mysql_version_id >= 50000 ) { $cmdline_mysqlbinlog .=" --character-sets-dir=$path_charsetsdir"; } @@ -1881,7 +1933,7 @@ sub environment_setup () { # ---------------------------------------------------- # Setup env so childs can execute mysql_upgrade # ---------------------------------------------------- - if ( $mysql_version_id >= 50000 ) + if ( !$opt_extern && $mysql_version_id >= 50000 ) { $ENV{'MYSQL_UPGRADE'}= mysql_upgrade_arguments(); } @@ -1889,7 +1941,7 @@ sub environment_setup () { # ---------------------------------------------------- # Setup env so childs can execute mysql_fix_system_tables # ---------------------------------------------------- - if ( ! $glob_win32 ) + if ( !$opt_extern && ! $glob_win32 ) { my $cmdline_mysql_fix_system_tables= "$exe_mysql_fix_system_tables --no-defaults --host=localhost " . @@ -1899,7 +1951,10 @@ sub environment_setup () { "--socket=$master->[0]->{'path_sock'}"; $ENV{'MYSQL_FIX_SYSTEM_TABLES'}= $cmdline_mysql_fix_system_tables; } - $ENV{'MYSQL_FIX_PRIVILEGE_TABLES'}= $file_mysql_fix_privilege_tables; + if (!$opt_extern) + { + $ENV{'MYSQL_FIX_PRIVILEGE_TABLES'}= $file_mysql_fix_privilege_tables; + } # ---------------------------------------------------- # Setup env so childs can execute my_print_defaults @@ -1922,10 +1977,11 @@ sub environment_setup () { $ENV{'UDF_EXAMPLE_LIB'}= ($lib_udf_example ? basename($lib_udf_example) : ""); - $ENV{'LD_LIBRARY_PATH'}= - ($lib_udf_example ? dirname($lib_udf_example) : "") . - ($ENV{'LD_LIBRARY_PATH'} ? ":$ENV{'LD_LIBRARY_PATH'}" : ""); - + # ---------------------------------------------------- + # Add the path where mysqld will find ha_example.so + # ---------------------------------------------------- + $ENV{'EXAMPLE_PLUGIN'}= + ($lib_example_plugin ? basename($lib_example_plugin) : ""); # ---------------------------------------------------- # We are nice and report a bit about our settings @@ -2168,6 +2224,25 @@ sub setup_vardir() { { unlink($name); } + if ( $opt_valgrind and $opt_debug ) + { + # When both --valgrind and --debug is selected, send + # all output to the trace file, making it possible to + # see the exact location where valgrind complains + foreach my $mysqld (@{$master}, @{$slave}) + { + my $sidx= $mysqld->{idx} ? "$mysqld->{idx}" : ""; + my $trace_name= "$opt_vardir/log/" . $mysqld->{type} . "$sidx.trace"; + open(LOG, ">$mysqld->{path_myerr}") or die "Can't create $mysqld->{path_myerr}\n"; + print LOG " +NOTE: When running with --valgrind --debug the output from the .err file is +stored together with the trace file to make it easier to find the exact +position for valgrind errors. +See trace file $trace_name.\n"; + close(LOG); + $mysqld->{path_myerr}= $trace_name; + } + } } @@ -2203,7 +2278,10 @@ sub check_ssl_support ($) { if ($opt_skip_ssl || $opt_extern) { - mtr_report("Skipping SSL"); + if (!$opt_extern) + { + mtr_report("Skipping SSL"); + } $opt_ssl_supported= 0; $opt_ssl= 0; return; @@ -2278,9 +2356,12 @@ sub vs_config_dirs ($$) { sub check_ndbcluster_support ($) { my $mysqld_variables= shift; - if ($opt_skip_ndbcluster) + if ($opt_skip_ndbcluster || $opt_extern) { - mtr_report("Skipping ndbcluster"); + if (!$opt_extern) + { + mtr_report("Skipping ndbcluster"); + } $opt_skip_ndbcluster_slave= 1; return; } @@ -2696,7 +2777,10 @@ sub initialize_servers () { } else { - mtr_report("No need to create '$opt_vardir' it already exists"); + if ($opt_verbose) + { + mtr_report("No need to create '$opt_vardir' it already exists"); + } } } else @@ -2724,8 +2808,10 @@ sub mysql_install_db () { install_db('master', $master->[0]->{'path_myddir'}); - # FIXME check if testcase really is using second master - copy_install_db('master', $master->[1]->{'path_myddir'}); + if ($max_master_num) + { + copy_install_db('master', $master->[1]->{'path_myddir'}); + } # Install the number of slave databses needed for (my $idx= 0; $idx < $max_slave_num; $idx++) @@ -2867,12 +2953,19 @@ sub install_db ($$) { mtr_add_arg($args, "--character-sets-dir=%s", $path_charsetsdir); } + # If DISABLE_GRANT_OPTIONS is defined when the server is compiled (e.g., + # configure --disable-grant-options), mysqld will not recognize the + # --bootstrap or --skip-grant-tables options. The user can set + # MYSQLD_BOOTSTRAP to the full path to a mysqld which does accept + # --bootstrap, to accommodate this. + my $exe_mysqld_bootstrap = $ENV{'MYSQLD_BOOTSTRAP'} || $exe_mysqld; + # Log bootstrap command my $path_bootstrap_log= "$opt_vardir/log/bootstrap.log"; mtr_tofile($path_bootstrap_log, - "$exe_mysqld " . join(" ", @$args) . "\n"); + "$exe_mysqld_bootstrap " . join(" ", @$args) . "\n"); - if ( mtr_run($exe_mysqld, $args, $init_db_sql_tmp, + if ( mtr_run($exe_mysqld_bootstrap, $args, $init_db_sql_tmp, $path_bootstrap_log, $path_bootstrap_log, "", { append_log_file => 1 }) != 0 ) @@ -3084,17 +3177,17 @@ sub do_before_run_mysqltest($) unlink("$result_dir/$tname.log"); unlink("$result_dir/$tname.warnings"); - if ( $mysql_version_id < 50000 ) - { - # Set environment variable NDB_STATUS_OK to 1 - # if script decided to run mysqltest cluster _is_ installed ok - $ENV{'NDB_STATUS_OK'} = "1"; - } - elsif ( $mysql_version_id < 50100 ) + if (!$opt_extern) { - # Set environment variable NDB_STATUS_OK to YES - # if script decided to run mysqltest cluster _is_ installed ok - $ENV{'NDB_STATUS_OK'} = "YES"; + if ( $mysql_version_id < 50000 ) { + # Set environment variable NDB_STATUS_OK to 1 + # if script decided to run mysqltest cluster _is_ installed ok + $ENV{'NDB_STATUS_OK'} = "1"; + } elsif ( $mysql_version_id < 50100 ) { + # Set environment variable NDB_STATUS_OK to YES + # if script decided to run mysqltest cluster _is_ installed ok + $ENV{'NDB_STATUS_OK'} = "YES"; + } } } @@ -3362,10 +3455,12 @@ sub save_files_before_restore($$) { # Look for core files foreach my $core_file ( glob("$data_dir/core*") ) { + last if $opt_max_save_core > 0 && $num_saved_cores >= $opt_max_save_core; my $core_name= basename($core_file); mtr_report("Saving $core_name"); mkdir($save_name) if ! -d $save_name; rename("$core_file", "$save_name/$core_name"); + ++$num_saved_cores; } } @@ -3527,11 +3622,10 @@ sub mysqld_arguments ($$$$$) { if ( $glob_use_embedded_server ) { $prefix= "--server-arg="; - } else { - # We can't pass embedded server --no-defaults - mtr_add_arg($args, "--no-defaults"); } + mtr_add_arg($args, "%s--no-defaults", $prefix); + mtr_add_arg($args, "%s--console", $prefix); mtr_add_arg($args, "%s--basedir=%s", $prefix, $path_my_basedir); mtr_add_arg($args, "%s--character-sets-dir=%s", $prefix, $path_charsetsdir); @@ -3555,13 +3649,14 @@ sub mysqld_arguments ($$$$$) { } } - my $pidfile; + # Check if "extra_opt" contains --skip-log-bin + my $skip_binlog= grep(/^--skip-log-bin/, @$extra_opt); if ( $type eq 'master' ) { my $id= $idx > 0 ? $idx + 101 : 1; - if (! $opt_skip_master_binlog) + if (! ($opt_skip_master_binlog || $skip_binlog) ) { mtr_add_arg($args, "%s--log-bin=%s/log/master-bin%s", $prefix, $opt_vardir, $sidx); @@ -3610,6 +3705,9 @@ sub mysqld_arguments ($$$$$) { # Turn on logging, will be sent to tables mtr_add_arg($args, "%s--log=", $prefix); } + + mtr_add_arg($args, "%s--plugin_dir=%s", $prefix, + dirname($lib_example_plugin)); } if ( $type eq 'slave' ) @@ -3620,7 +3718,7 @@ sub mysqld_arguments ($$$$$) { mtr_add_arg($args, "%s--datadir=%s", $prefix, $slave->[$idx]->{'path_myddir'}); mtr_add_arg($args, "%s--init-rpl-role=slave", $prefix); - if (! $opt_skip_slave_binlog) + if (! ( $opt_skip_slave_binlog || $skip_binlog )) { mtr_add_arg($args, "%s--log-bin=%s/log/slave%s-bin", $prefix, $opt_vardir, $sidx); # FIXME use own dir for binlogs @@ -3757,6 +3855,10 @@ sub mysqld_arguments ($$$$$) { { $found_skip_core= 1; } + elsif ($skip_binlog and mtr_match_prefix($arg, "--binlog-format")) + { + ; # Dont add --binlog-format when running without binlog + } else { mtr_add_arg($args, "%s%s", $prefix, $arg); @@ -4260,7 +4362,8 @@ sub run_testcase_start_servers($) { } - if ( $clusters->[0]->{'pid'} and ! $master->[1]->{'pid'} ) + if ( $clusters->[0]->{'pid'} and ! $master->[1]->{'pid'} and + $tinfo->{'master_num'} > 1 ) { # Test needs cluster, start an extra mysqld connected to cluster @@ -4472,6 +4575,10 @@ sub run_mysqltest ($) { mtr_add_arg($args, "--character-sets-dir=%s", $path_charsetsdir); mtr_add_arg($args, "--logdir=%s/log", $opt_vardir); + # Log line number and time for each line in .test file + mtr_add_arg($args, "--mark-progress") + if $opt_mark_progress; + if ($tinfo->{'component_id'} eq 'im') { mtr_add_arg($args, "--socket=%s", $instance_manager->{'path_sock'}); @@ -4859,10 +4966,10 @@ sub usage ($) { if ( $message ) { - print STDERR "$message \n"; + print STDERR "$message\n"; } - print STDERR <.progress Options that pass on options @@ -4956,6 +5064,9 @@ Options for debugging the product master-binary=PATH Specify the master "mysqld" to use slave-binary=PATH Specify the slave "mysqld" to use strace-client Create strace output for mysqltest client + max-save-core Limit the number of core files saved (to avoid filling + up disks for heavily crashing server). Defaults to + $opt_max_save_core, set to 0 for no limit. Options for coverage, profiling etc diff --git a/mysql-test/r/maria.result b/mysql-test/r/maria.result index 7812f04336a..4a708cde307 100644 --- a/mysql-test/r/maria.result +++ b/mysql-test/r/maria.result @@ -348,11 +348,11 @@ t1 1 c_2 2 a A 5 NULL NULL BTREE explain select * from t1,t2 where t1.a=t2.a; id select_type table type possible_keys key key_len ref rows Extra 1 SIMPLE t2 ALL a NULL NULL NULL 2 -1 SIMPLE t1 ALL a NULL NULL NULL 5 Using where +1 SIMPLE t1 ref a a 4 test.t2.a 3 explain select * from t1,t2 force index(a) where t1.a=t2.a; id select_type table type possible_keys key key_len ref rows Extra 1 SIMPLE t2 ALL a NULL NULL NULL 2 -1 SIMPLE t1 ALL a NULL NULL NULL 5 Using where +1 SIMPLE t1 ref a a 4 test.t2.a 3 explain select * from t1 force index(a),t2 force index(a) where t1.a=t2.a; id select_type table type possible_keys key key_len ref rows Extra 1 SIMPLE t2 ALL a NULL NULL NULL 2 @@ -364,10 +364,10 @@ id select_type table type possible_keys key key_len ref rows Extra explain select * from t1,t2 force index(c) where t1.a=t2.a; id select_type table type possible_keys key key_len ref rows Extra 1 SIMPLE t2 ALL NULL NULL NULL NULL 2 -1 SIMPLE t1 ALL a NULL NULL NULL 5 Using where +1 SIMPLE t1 ref a a 4 test.t2.a 3 explain select * from t1 where a=0 or a=2; id select_type table type possible_keys key key_len ref rows Extra -1 SIMPLE t1 ALL a NULL NULL NULL 5 Using where +1 SIMPLE t1 range a a 4 NULL 4 Using where explain select * from t1 force index (a) where a=0 or a=2; id select_type table type possible_keys key key_len ref rows Extra 1 SIMPLE t1 range a a 4 NULL 4 Using where @@ -430,9 +430,6 @@ select * from t1 where a='807780' and b='477' and c='165'; a b c 807780 477 165 drop table t1; -DROP TABLE IF EXISTS t1; -Warnings: -Note 1051 Unknown table 't1' CREATE TABLE t1 (a varchar(150) NOT NULL, KEY (a)); INSERT t1 VALUES ("can \tcan"); INSERT t1 VALUES ("can can"); @@ -517,8 +514,8 @@ select c1 from t1 order by c1 limit 1; c1 a drop table t1; -create table t1 (a int not null, primary key(a)); -create table t2 (a int not null, b int not null, primary key(a,b)); +create table t1 (a int not null, primary key(a)) ROW_FORMAT=FIXED; +create table t2 (a int not null, b int not null, primary key(a,b)) ROW_FORMAT=FIXED; insert into t1 values (1),(2),(3),(4),(5),(6); insert into t2 values (1,1),(2,1); lock tables t1 read local, t2 read local; @@ -532,8 +529,8 @@ a a b 1 1 1 2 2 1 drop table t1,t2; -CREATE TABLE t1 (c1 varchar(250) NOT NULL); -CREATE TABLE t2 (c1 varchar(250) NOT NULL, PRIMARY KEY (c1)); +CREATE TABLE t1 (c1 varchar(250) NOT NULL) ROW_FORMAT=DYNAMIC; +CREATE TABLE t2 (c1 varchar(250) NOT NULL, PRIMARY KEY (c1)) ROW_FORMAT=DYNAMIC; INSERT INTO t1 VALUES ('test000001'), ('test000002'), ('test000003'); INSERT INTO t2 VALUES ('test000002'), ('test000003'), ('test000004'); LOCK TABLES t1 READ LOCAL, t2 READ LOCAL; @@ -590,7 +587,7 @@ t1 1 a 1 a A NULL NULL NULL YES BTREE disabled alter table t1 enable keys; show keys from t1; Table Non_unique Key_name Seq_in_index Column_name Collation Cardinality Sub_part Packed Null Index_type Comment -t1 1 a 1 a A 1000 NULL NULL YES BTREE +t1 1 a 1 a A NULL NULL NULL YES BTREE disabled alter table t1 engine=heap; alter table t1 disable keys; Warnings: @@ -715,7 +712,7 @@ t1 1 a 4 d A 4 NULL NULL YES BTREE delete from t1; analyze table t1; Table Op Msg_type Msg_text -test.t1 analyze status OK +test.t1 analyze status Table is already up to date show index from t1; Table Non_unique Key_name Seq_in_index Column_name Collation Cardinality Sub_part Packed Null Index_type Comment t1 1 a 1 a A 0 NULL NULL YES BTREE @@ -800,7 +797,7 @@ CREATE TABLE t1 ( PRIMARY KEY (`_id`), UNIQUE KEY `sequence_name_index` (`name`(50)), KEY (`length_`) -) ENGINE=maria DEFAULT CHARSET=latin1; +) DEFAULT CHARSET=latin1; INSERT INTO t1 VALUES (1,NULL,NULL,NULL,NULL,NULL,NULL,NULL,NULL,NULL,'sample1',''), (2,NULL,NULL,NULL,NULL,NULL,NULL,NULL,NULL,NULL,'sample2',''), @@ -825,19 +822,19 @@ _id DELETE FROM t1 WHERE _id < 8; SHOW TABLE STATUS LIKE 't1'; Name Engine Version Row_format Rows Avg_row_length Data_length Max_data_length Index_length Data_free Auto_increment Create_time Update_time Check_time Collation Checksum Create_options Comment -t1 MARIA 10 Dynamic 2 # # # # 140 # # # # # # +t1 MARIA 10 Paged 2 # # # # 0 # # # # # # CHECK TABLE t1 EXTENDED; Table Op Msg_type Msg_text test.t1 check status OK OPTIMIZE TABLE t1; Table Op Msg_type Msg_text -test.t1 optimize status OK +test.t1 optimize status Table is already up to date CHECK TABLE t1 EXTENDED; Table Op Msg_type Msg_text test.t1 check status OK SHOW TABLE STATUS LIKE 't1'; Name Engine Version Row_format Rows Avg_row_length Data_length Max_data_length Index_length Data_free Auto_increment Create_time Update_time Check_time Collation Checksum Create_options Comment -t1 MARIA 10 Dynamic 2 # # # # 0 # # # # # # +t1 MARIA 10 Paged 2 # # # # 0 # # # # # # SELECT _id FROM t1; _id 8 @@ -859,7 +856,7 @@ CREATE TABLE t1 ( PRIMARY KEY (`_id`), UNIQUE KEY `sequence_name_index` (`name`(50)), KEY (`length_`) -) ENGINE=maria DEFAULT CHARSET=latin1; +) DEFAULT CHARSET=latin1; INSERT INTO t1 VALUES (1,NULL,NULL,NULL,NULL,NULL,NULL,NULL,NULL,NULL,'sample1',''), (2,NULL,NULL,NULL,NULL,NULL,NULL,NULL,NULL,NULL,'sample2',''), @@ -884,7 +881,7 @@ _id DELETE FROM t1 WHERE _id < 8; SHOW TABLE STATUS LIKE 't1'; Name Engine Version Row_format Rows Avg_row_length Data_length Max_data_length Index_length Data_free Auto_increment Create_time Update_time Check_time Collation Checksum Create_options Comment -t1 MARIA 10 Dynamic 2 # # # # 140 # # # # # # +t1 MARIA 10 Paged 2 # # # # 0 # # # # # # CHECK TABLE t1 EXTENDED; Table Op Msg_type Msg_text test.t1 check status OK @@ -896,7 +893,7 @@ Table Op Msg_type Msg_text test.t1 check status OK SHOW TABLE STATUS LIKE 't1'; Name Engine Version Row_format Rows Avg_row_length Data_length Max_data_length Index_length Data_free Auto_increment Create_time Update_time Check_time Collation Checksum Create_options Comment -t1 MARIA 10 Dynamic 2 # # # # 140 # # # # # # +t1 MARIA 10 Paged 2 # # # # 0 # # # # # # SELECT _id FROM t1; _id 8 @@ -1503,7 +1500,7 @@ create table t1 (v varchar(65535)); ERROR 42000: Row size too large. The maximum row size for the used table type, not counting BLOBs, is 65535. You have to change some columns to TEXT or BLOBs set @save_concurrent_insert=@@concurrent_insert; set global concurrent_insert=1; -create table t1 (a int); +create table t1 (a int) ROW_FORMAT=FIXED; insert into t1 values (1),(2),(3),(4),(5); lock table t1 read local; insert into t1 values(6),(7); @@ -1530,7 +1527,7 @@ check table t1; Table Op Msg_type Msg_text test.t1 check status OK drop table t1; -create table t1 (a int, b varchar(30) default "hello"); +create table t1 (a int, b varchar(30) default "hello") ROW_FORMAT=DYNAMIC; insert into t1 (a) values (1),(2),(3),(4),(5); lock table t1 read local; insert into t1 (a) values(6),(7); @@ -1570,7 +1567,7 @@ alter table t1 disable keys; alter table t1 enable keys; show keys from t1; Table Non_unique Key_name Seq_in_index Column_name Collation Cardinality Sub_part Packed Null Index_type Comment -t1 1 a 1 a A 8 NULL NULL YES BTREE +t1 1 a 1 a A 8 NULL NULL YES BTREE disabled drop table t1; show create table t1; show create table t1; @@ -1592,13 +1589,12 @@ create table t3 (c1 int) pack_keys=default; create table t4 (c1 int) pack_keys=2; ERROR 42000: You have an error in your SQL syntax; check the manual that corresponds to your MySQL server version for the right syntax to use near '2' at line 1 drop table t1, t2, t3; -End of 5.0 tests create table t1 (a int not null, key `a` (a) key_block_size=1024); show create table t1; Table Create Table t1 CREATE TABLE `t1` ( `a` int(11) NOT NULL, - KEY `a` (`a`) KEY_BLOCK_SIZE=1024 + KEY `a` (`a`) KEY_BLOCK_SIZE=8192 ) ENGINE=MARIA DEFAULT CHARSET=latin1 drop table t1; create table t1 (a int not null, key `a` (a) key_block_size=2048); @@ -1606,7 +1602,7 @@ show create table t1; Table Create Table t1 CREATE TABLE `t1` ( `a` int(11) NOT NULL, - KEY `a` (`a`) KEY_BLOCK_SIZE=2048 + KEY `a` (`a`) KEY_BLOCK_SIZE=8192 ) ENGINE=MARIA DEFAULT CHARSET=latin1 drop table t1; create table t1 (a varchar(2048), key `a` (a)); @@ -1626,7 +1622,7 @@ show create table t1; Table Create Table t1 CREATE TABLE `t1` ( `a` varchar(2048) DEFAULT NULL, - KEY `a` (`a`(1000)) KEY_BLOCK_SIZE=4096 + KEY `a` (`a`(1000)) KEY_BLOCK_SIZE=8192 ) ENGINE=MARIA DEFAULT CHARSET=latin1 drop table t1; create table t1 (a int not null, b varchar(2048), key (a), key(b)) key_block_size=1024; @@ -1637,8 +1633,8 @@ Table Create Table t1 CREATE TABLE `t1` ( `a` int(11) NOT NULL, `b` varchar(2048) DEFAULT NULL, - KEY `a` (`a`), - KEY `b` (`b`(1000)) KEY_BLOCK_SIZE=4096 + KEY `a` (`a`) KEY_BLOCK_SIZE=8192, + KEY `b` (`b`(1000)) KEY_BLOCK_SIZE=8192 ) ENGINE=MARIA DEFAULT CHARSET=latin1 KEY_BLOCK_SIZE=1024 alter table t1 key_block_size=2048; show create table t1; @@ -1646,8 +1642,8 @@ Table Create Table t1 CREATE TABLE `t1` ( `a` int(11) NOT NULL, `b` varchar(2048) DEFAULT NULL, - KEY `a` (`a`) KEY_BLOCK_SIZE=1024, - KEY `b` (`b`(1000)) KEY_BLOCK_SIZE=4096 + KEY `a` (`a`) KEY_BLOCK_SIZE=8192, + KEY `b` (`b`(1000)) KEY_BLOCK_SIZE=8192 ) ENGINE=MARIA DEFAULT CHARSET=latin1 KEY_BLOCK_SIZE=2048 alter table t1 add c int, add key (c); show create table t1; @@ -1656,9 +1652,9 @@ t1 CREATE TABLE `t1` ( `a` int(11) NOT NULL, `b` varchar(2048) DEFAULT NULL, `c` int(11) DEFAULT NULL, - KEY `a` (`a`) KEY_BLOCK_SIZE=1024, - KEY `b` (`b`(1000)) KEY_BLOCK_SIZE=4096, - KEY `c` (`c`) + KEY `a` (`a`) KEY_BLOCK_SIZE=8192, + KEY `b` (`b`(1000)) KEY_BLOCK_SIZE=8192, + KEY `c` (`c`) KEY_BLOCK_SIZE=8192 ) ENGINE=MARIA DEFAULT CHARSET=latin1 KEY_BLOCK_SIZE=2048 alter table t1 key_block_size=0; alter table t1 add d int, add key (d); @@ -1669,9 +1665,9 @@ t1 CREATE TABLE `t1` ( `b` varchar(2048) DEFAULT NULL, `c` int(11) DEFAULT NULL, `d` int(11) DEFAULT NULL, - KEY `a` (`a`) KEY_BLOCK_SIZE=1024, - KEY `b` (`b`(1000)) KEY_BLOCK_SIZE=4096, - KEY `c` (`c`) KEY_BLOCK_SIZE=2048, + KEY `a` (`a`) KEY_BLOCK_SIZE=8192, + KEY `b` (`b`(1000)) KEY_BLOCK_SIZE=8192, + KEY `c` (`c`) KEY_BLOCK_SIZE=8192, KEY `d` (`d`) ) ENGINE=MARIA DEFAULT CHARSET=latin1 drop table t1; @@ -1695,7 +1691,7 @@ Table Create Table t1 CREATE TABLE `t1` ( `a` int(11) NOT NULL, `b` varchar(2048) DEFAULT NULL, - KEY `a` (`a`) KEY_BLOCK_SIZE=1024, + KEY `a` (`a`), KEY `b` (`b`(1000)) ) ENGINE=MARIA DEFAULT CHARSET=latin1 KEY_BLOCK_SIZE=8192 drop table t1; @@ -1705,7 +1701,7 @@ Table Create Table t1 CREATE TABLE `t1` ( `a` int(11) NOT NULL, `b` int(11) DEFAULT NULL, - KEY `a` (`a`) KEY_BLOCK_SIZE=1024, + KEY `a` (`a`) KEY_BLOCK_SIZE=8192, KEY `b` (`b`) KEY_BLOCK_SIZE=8192 ) ENGINE=MARIA DEFAULT CHARSET=latin1 KEY_BLOCK_SIZE=16384 drop table t1; @@ -1714,7 +1710,7 @@ show create table t1; Table Create Table t1 CREATE TABLE `t1` ( `a` int(11) NOT NULL, - KEY `a` (`a`) KEY_BLOCK_SIZE=1024 + KEY `a` (`a`) KEY_BLOCK_SIZE=8192 ) ENGINE=MARIA DEFAULT CHARSET=latin1 drop table t1; create table t1 (a varchar(2048), key `a` (a) key_block_size=1000000000000000000); @@ -1732,7 +1728,7 @@ show create table t1; Table Create Table t1 CREATE TABLE `t1` ( `a` int(11) NOT NULL, - KEY `a` (`a`) KEY_BLOCK_SIZE=2048 + KEY `a` (`a`) KEY_BLOCK_SIZE=8192 ) ENGINE=MARIA DEFAULT CHARSET=latin1 drop table t1; create table t1 (a int not null, key key_block_size=1024 (a)); @@ -1784,5 +1780,4 @@ CHECK TABLE t1; Table Op Msg_type Msg_text test.t1 check status OK DROP TABLE t1; -End of 5.1 tests set global storage_engine=MyISAM; diff --git a/mysql-test/t/disabled.def b/mysql-test/t/disabled.def index c7748318e51..13c96f1611f 100644 --- a/mysql-test/t/disabled.def +++ b/mysql-test/t/disabled.def @@ -38,6 +38,5 @@ ndb_autodiscover3 : bug#21806 flush2 : Bug#24805 Pushbuild can't handle test with --disable-log-bin -maria : Until maria is fully functional ps_maria : Until maria is fully functional rpl_ndb_dd_advance : unstable test disable till next merge diff --git a/mysql-test/t/maria.test b/mysql-test/t/maria.test index 87a38ff60b0..898268c9bb0 100644 --- a/mysql-test/t/maria.test +++ b/mysql-test/t/maria.test @@ -1,6 +1,8 @@ # -# Test bugs in the MARIA code +# Testing of potential probelms in Maria +# This code was initially taken from myisam.test # + -- source include/have_maria.inc let $default=`select @@global.storage_engine`; @@ -48,7 +50,7 @@ while ($1) } SET SQL_WARNINGS=1; --enable_warnings -enable_query_log; +--enable_query_log check table t1; repair table t1; delete from t1 where (a & 1); @@ -404,7 +406,7 @@ check table t1; drop table t1; # -# two bugs in maria-space-stripping feature +# Test space-stripping features # create table t1 ( a text not null, key a (a(20))); insert into t1 values ('aaa '),('aaa'),('aa'); @@ -417,7 +419,7 @@ select concat(a,'.') from t1; drop table t1; # -# Third bug in the same code (BUG#2295) +# More space testing # create table t1(a text not null, b text not null, c text not null, index (a(10),b(10),c(10))); @@ -428,9 +430,9 @@ select * from t1 where a='807780' and b='477' and c='165'; drop table t1; # -# space-stripping in _mi_prefix_search: BUG#5284 +# Space-stripping in prefix_search # -DROP TABLE IF EXISTS t1; + CREATE TABLE t1 (a varchar(150) NOT NULL, KEY (a)); INSERT t1 VALUES ("can \tcan"); INSERT t1 VALUES ("can can"); @@ -442,6 +444,7 @@ DROP TABLE t1; # # Verify blob handling # + create table t1 (a blob); insert into t1 values('a '),('a'); select concat(a,'.') from t1 where a='a'; @@ -458,9 +461,9 @@ create table t1 (a int not null auto_increment primary key, b text not null, uni insert into t1 (b) values ('a'),('b'),('c'); select concat(b,'.') from t1; update t1 set b='b ' where a=2; ---error 1062 +--error 1582 update t1 set b='b ' where a > 1; ---error 1062 +--error 1582 insert into t1 (b) values ('b'); select * from t1; delete from t1 where b='b'; @@ -468,7 +471,7 @@ select a,concat(b,'.') from t1; drop table t1; # -# Test keys with 0 segments. (Bug #3203) +# Test keys with 0 segments # create table t1 (a int not null); create table t2 (a int not null, primary key (a)); @@ -482,7 +485,7 @@ explain select distinct t1.a from t1,t2 order by t2.a; drop table t1,t2; # -# Bug#14616 - Freshly imported table returns error 124 when using LIMIT +# Test freshly imported table and LIMIT # create table t1 ( c1 varchar(32), @@ -494,16 +497,18 @@ select c1 from t1 order by c1 limit 1; drop table t1; # -# Bug #14400 Join could miss concurrently inserted row +# Test join that could miss concurrently inserted row +# Note that for the moment Maria only supports multiple writers if we have +# static or dynamic row format # # Partial key. -create table t1 (a int not null, primary key(a)); -create table t2 (a int not null, b int not null, primary key(a,b)); +create table t1 (a int not null, primary key(a)) ROW_FORMAT=FIXED; +create table t2 (a int not null, b int not null, primary key(a,b)) ROW_FORMAT=FIXED; insert into t1 values (1),(2),(3),(4),(5),(6); insert into t2 values (1,1),(2,1); lock tables t1 read local, t2 read local; select straight_join * from t1,t2 force index (primary) where t1.a=t2.a; -connect (root,localhost,root,,test,$MASTER_MYPORT,master.sock); +connect (root,localhost,root,,test,$MASTER_MYPORT,$MASTER_MYSOCK); insert into t2 values(2,0); disconnect root; connection default; @@ -511,8 +516,8 @@ select straight_join * from t1,t2 force index (primary) where t1.a=t2.a; drop table t1,t2; # # Full key. -CREATE TABLE t1 (c1 varchar(250) NOT NULL); -CREATE TABLE t2 (c1 varchar(250) NOT NULL, PRIMARY KEY (c1)); +CREATE TABLE t1 (c1 varchar(250) NOT NULL) ROW_FORMAT=DYNAMIC; +CREATE TABLE t2 (c1 varchar(250) NOT NULL, PRIMARY KEY (c1)) ROW_FORMAT=DYNAMIC; INSERT INTO t1 VALUES ('test000001'), ('test000002'), ('test000003'); INSERT INTO t2 VALUES ('test000002'), ('test000003'), ('test000004'); LOCK TABLES t1 READ LOCAL, t2 READ LOCAL; @@ -527,8 +532,6 @@ SELECT t1.c1 AS t1c1, t2.c1 AS t2c1 FROM t1, t2 WHERE t1.c1 = t2.c1 HAVING t1c1 != t2c1; DROP TABLE t1,t2; -# End of 4.0 tests - # # Test RTREE index # @@ -566,13 +569,16 @@ insert t1 select * from t2; show keys from t1; alter table t1 enable keys; show keys from t1; +#TODO after we have repair: delete the following --disable-warnings +--disable_warnings alter table t1 engine=heap; +--enable_warnings alter table t1 disable keys; show keys from t1; drop table t1,t2; # -# index search for NULL in blob. Bug #4816 +# Index search for NULL in blob # create table t1 ( a tinytext, b char(1), index idx (a(1),b) ); insert into t1 values (null,''), (null,''); @@ -581,7 +587,7 @@ select count(*) from t1 where a is null; drop table t1; # -# bug9188 - Corruption Can't open file: 'table.MYI' (errno: 145) +# Test corruption Can't open file: 'table.MYI' (errno: 145) # create table t1 (c1 int, c2 varchar(4) not null default '', key(c2(3))) default charset=utf8; @@ -592,9 +598,9 @@ drop table t1; # -# Bug#12296 - CHECKSUM TABLE reports 0 for the table -# This happened if the first record was marked as deleted. +# Test CHECKSUM TABLE # + create table t1 (c1 int); insert into t1 values (1),(2),(3),(4); checksum table t1; @@ -607,7 +613,7 @@ checksum table t2; drop table t1, t2; # -# BUG#12232: New maria_stats_method variable. +# maria_stats_method variable. # show variables like 'maria_stats_method'; @@ -677,7 +683,9 @@ show index from t1; set maria_stats_method=DEFAULT; drop table t1; -# BUG#13814 - key value packed incorrectly for TINYBLOBs +# +# Test key value packing for TINYBLOBs +# create table t1( cip INT NOT NULL, @@ -695,8 +703,9 @@ select * from t1 where bob is null and cip=1; drop table t1; # -# Bug#14980 - COUNT(*) incorrect on MARIA table with certain INDEX +# Test COUNT(*) table with different INDEX # + create table t1 ( id1 int not null auto_increment, id2 int not null default '0', @@ -713,9 +722,9 @@ select count(id1) from t1 where id2 = 10; drop table t1; # -# BUG##20357 - Got error 124 from storage engine using MIN and MAX functions -# in queries +# Test MIN and MAX functions in queries # + CREATE TABLE t1(a TINYINT, KEY(a)); INSERT INTO t1 VALUES(1); SELECT MAX(a) FROM t1 IGNORE INDEX(a); @@ -725,7 +734,7 @@ SELECT MAX(a) FROM t1 IGNORE INDEX(a); DROP TABLE t1; # -# BUG#18036 - update of table joined to self reports table as crashed +# Test update of table joined to self # CREATE TABLE t1(a CHAR(9), b VARCHAR(7)); INSERT INTO t1(a) VALUES('xxxxxxxxx'),('xxxxxxxxx'); @@ -734,7 +743,7 @@ SELECT * FROM t1; DROP TABLE t1; # -# Bug#8283 - OPTIMIZE TABLE causes data loss +# OPTIMIZE TABLE with multiple threads # SET @@maria_repair_threads=2; SHOW VARIABLES LIKE 'maria_repair%'; @@ -756,7 +765,7 @@ CREATE TABLE t1 ( PRIMARY KEY (`_id`), UNIQUE KEY `sequence_name_index` (`name`(50)), KEY (`length_`) -) ENGINE=maria DEFAULT CHARSET=latin1; +) DEFAULT CHARSET=latin1; # INSERT INTO t1 VALUES (1,NULL,NULL,NULL,NULL,NULL,NULL,NULL,NULL,NULL,'sample1',''), @@ -798,7 +807,7 @@ CREATE TABLE t1 ( PRIMARY KEY (`_id`), UNIQUE KEY `sequence_name_index` (`name`(50)), KEY (`length_`) -) ENGINE=maria DEFAULT CHARSET=latin1; +) DEFAULT CHARSET=latin1; # INSERT INTO t1 VALUES (1,NULL,NULL,NULL,NULL,NULL,NULL,NULL,NULL,NULL,'sample1',''), @@ -825,6 +834,7 @@ DROP TABLE t1; # SET @@maria_repair_threads=1; SHOW VARIABLES LIKE 'maria_repair%'; + # # Test varchar # @@ -854,7 +864,7 @@ create table t1 (v varchar(65535)); # set @save_concurrent_insert=@@concurrent_insert; set global concurrent_insert=1; -create table t1 (a int); +create table t1 (a int) ROW_FORMAT=FIXED; insert into t1 values (1),(2),(3),(4),(5); lock table t1 read local; connect (con1,localhost,root,,); @@ -879,7 +889,7 @@ drop table t1; disconnect con1; # Same test with dynamic record length -create table t1 (a int, b varchar(30) default "hello"); +create table t1 (a int, b varchar(30) default "hello") ROW_FORMAT=DYNAMIC; insert into t1 (a) values (1),(2),(3),(4),(5); lock table t1 read local; connect (con1,localhost,root,,); @@ -904,9 +914,10 @@ drop table t1; disconnect con1; set global concurrent_insert=@save_concurrent_insert; +# +# ANALYZE TABLE and ALTER TABLE .. ENABLE INDEX +# -# BUG#9622 - ANALYZE TABLE and ALTER TABLE .. ENABLE INDEX produce -# different statistics on the same table with NULL values. create table t1 (a int, key(a)); insert into t1 values (1),(2),(3),(4),(NULL),(NULL),(NULL),(NULL); @@ -920,7 +931,7 @@ show keys from t1; drop table t1; # -# Bug#8706 - temporary table with data directory option fails +# Test temporary table with data directory option # connect (session1,localhost,root,,); connect (session2,localhost,root,,); @@ -957,7 +968,7 @@ drop table t1; --echo End of 4.1 tests # -# Bug#10056 - PACK_KEYS option take values greater than 1 while creating table +# Test if PACK_KEYS option takes values greater than 1 while creating table # create table t1 (c1 int) pack_keys=0; create table t2 (c1 int) pack_keys=1; @@ -966,8 +977,6 @@ create table t3 (c1 int) pack_keys=default; create table t4 (c1 int) pack_keys=2; drop table t1, t2, t3; ---echo End of 5.0 tests - # # Test of key_block_size # @@ -1033,7 +1042,7 @@ create table t1 (a int not null, key `a` key_block_size=1024 (a)); # -# Bug#22119 - Changing MI_KEY_BLOCK_LENGTH makes a wrong myisamchk +# Test of changing MI_KEY_BLOCK_LENGTH # CREATE TABLE t1 ( c1 INT, @@ -1075,8 +1084,6 @@ DELETE FROM t1 WHERE c1 >= 10; CHECK TABLE t1; DROP TABLE t1; ---echo End of 5.1 tests +# End of 5.2 tests eval set global storage_engine=$default; - -# End of 5.2 tests diff --git a/mysql-test/t/myisam.test b/mysql-test/t/myisam.test index 5a663718878..8d5502fe28f 100644 --- a/mysql-test/t/myisam.test +++ b/mysql-test/t/myisam.test @@ -498,7 +498,7 @@ insert into t1 values (1),(2),(3),(4),(5),(6); insert into t2 values (1,1),(2,1); lock tables t1 read local, t2 read local; select straight_join * from t1,t2 force index (primary) where t1.a=t2.a; -connect (root,localhost,root,,test,$MASTER_MYPORT,master.sock); +connect (root,localhost,root,,test,$MASTER_MYPORT,$MASTER_MYSOCK); insert into t2 values(2,0); disconnect root; connection default; diff --git a/mysql-test/t/query_cache_notembedded.test b/mysql-test/t/query_cache_notembedded.test index 802022d061b..a0085c0ba31 100644 --- a/mysql-test/t/query_cache_notembedded.test +++ b/mysql-test/t/query_cache_notembedded.test @@ -81,12 +81,12 @@ drop table t1, t2, t3, t11, t21; # # do not use QC if tables locked (BUG#12385) # -connect (root,localhost,root,,test,$MASTER_MYPORT,master.sock); +connect (root,localhost,root,,test,$MASTER_MYPORT,$MASTER_MYSOCK); connection root; CREATE TABLE t1 ( a INT NOT NULL PRIMARY KEY AUTO_INCREMENT ) ENGINE = MyISAM; LOCK TABLE t1 READ LOCAL; -connect (root2,localhost,root,,test,$MASTER_MYPORT,master.sock); +connect (root2,localhost,root,,test,$MASTER_MYPORT,$MASTER_MYSOCK); connection root2; INSERT INTO t1 VALUES (), (), (); connection root; diff --git a/sql/filesort.cc b/sql/filesort.cc index 77365499da0..6fafa4c46bf 100644 --- a/sql/filesort.cc +++ b/sql/filesort.cc @@ -451,7 +451,7 @@ static ha_rows find_all_keys(SORTPARAM *param, SQL_SELECT *select, ref_pos= ref_buff; quick_select=select && select->quick; record=0; - flag= ((!indexfile && file->ha_table_flags() & HA_REC_NOT_IN_SEQ) + flag= ((!indexfile && (file->ha_table_flags() & HA_REC_NOT_IN_SEQ)) || quick_select); if (indexfile || flag) ref_pos= &file->ref[0]; diff --git a/sql/handler.cc b/sql/handler.cc index ce9dc342713..89e48fd1ab5 100644 --- a/sql/handler.cc +++ b/sql/handler.cc @@ -2029,7 +2029,7 @@ void handler::print_keydup_error(uint key_nr, const char *msg) { /* Key is unknown */ str.copy("", 0, system_charset_info); - my_printf_error(ER_DUP_ENTRY, msg, + my_printf_error(ER_DUP_ENTRY_WITH_KEY_NAME, msg, MYF(0), str.c_ptr(), "*UNKNOWN*"); } else @@ -2042,7 +2042,7 @@ void handler::print_keydup_error(uint key_nr, const char *msg) str.length(max_length-4); str.append(STRING_WITH_LEN("...")); } - my_printf_error(ER_DUP_ENTRY, msg, + my_printf_error(ER_DUP_ENTRY_WITH_KEY_NAME, msg, MYF(0), str.c_ptr(), table->key_info[key_nr].name); } } @@ -2087,7 +2087,7 @@ void handler::print_error(int error, myf errflag) uint key_nr=get_dup_key(error); if ((int) key_nr >= 0) { - print_keydup_error(key_nr, ER(ER_DUP_ENTRY)); + print_keydup_error(key_nr, ER(ER_DUP_ENTRY_WITH_KEY_NAME)); DBUG_VOID_RETURN; } textno=ER_DUP_KEY; diff --git a/sql/share/errmsg.txt b/sql/share/errmsg.txt index c64f4da045e..8159f014e93 100644 --- a/sql/share/errmsg.txt +++ b/sql/share/errmsg.txt @@ -51,54 +51,54 @@ ER_YES spa "SI" ukr "ôáë" ER_CANT_CREATE_FILE - cze "Nemohu vytvo-Bøit soubor '%-.64s' (chybový kód: %d)" - dan "Kan ikke oprette filen '%-.64s' (Fejlkode: %d)" - nla "Kan file '%-.64s' niet aanmaken (Errcode: %d)" + cze "Nemohu vytvo-Bøit soubor '%-.200s' (chybový kód: %d)" + dan "Kan ikke oprette filen '%-.200s' (Fejlkode: %d)" + nla "Kan file '%-.200s' niet aanmaken (Errcode: %d)" eng "Can't create file '%-.200s' (errno: %d)" - est "Ei suuda luua faili '%-.64s' (veakood: %d)" - fre "Ne peut créer le fichier '%-.64s' (Errcode: %d)" - ger "Kann Datei '%-.64s' nicht erzeugen (Fehler: %d)" - greek "Áäýíáôç ç äçìéïõñãßá ôïõ áñ÷åßïõ '%-.64s' (êùäéêüò ëÜèïõò: %d)" - hun "A '%-.64s' file nem hozhato letre (hibakod: %d)" - ita "Impossibile creare il file '%-.64s' (errno: %d)" - jpn "'%-.64s' ¥Õ¥¡¥¤¥ë¤¬ºî¤ì¤Þ¤»¤ó (errno: %d)" - kor "È­ÀÏ '%-.64s'¸¦ ¸¸µéÁö ¸øÇß½À´Ï´Ù. (¿¡·¯¹øÈ£: %d)" - nor "Kan ikke opprette fila '%-.64s' (Feilkode: %d)" - norwegian-ny "Kan ikkje opprette fila '%-.64s' (Feilkode: %d)" - pol "Nie mo¿na stworzyæ pliku '%-.64s' (Kod b³êdu: %d)" - por "Não pode criar o arquivo '%-.64s' (erro no. %d)" - rum "Nu pot sa creez fisierul '%-.64s' (Eroare: %d)" - rus "îÅ×ÏÚÍÏÖÎÏ ÓÏÚÄÁÔØ ÆÁÊÌ '%-.64s' (ÏÛÉÂËÁ: %d)" - serbian "Ne mogu da kreiram file '%-.64s' (errno: %d)" - slo "Nemô¾em vytvori» súbor '%-.64s' (chybový kód: %d)" - spa "No puedo crear archivo '%-.64s' (Error: %d)" - swe "Kan inte skapa filen '%-.64s' (Felkod: %d)" - ukr "îÅ ÍÏÖÕ ÓÔ×ÏÒÉÔÉ ÆÁÊÌ '%-.64s' (ÐÏÍÉÌËÁ: %d)" + est "Ei suuda luua faili '%-.200s' (veakood: %d)" + fre "Ne peut créer le fichier '%-.200s' (Errcode: %d)" + ger "Kann Datei '%-.200s' nicht erzeugen (Fehler: %d)" + greek "Áäýíáôç ç äçìéïõñãßá ôïõ áñ÷åßïõ '%-.200s' (êùäéêüò ëÜèïõò: %d)" + hun "A '%-.200s' file nem hozhato letre (hibakod: %d)" + ita "Impossibile creare il file '%-.200s' (errno: %d)" + jpn "'%-.200s' ¥Õ¥¡¥¤¥ë¤¬ºî¤ì¤Þ¤»¤ó (errno: %d)" + kor "È­ÀÏ '%-.200s'¸¦ ¸¸µéÁö ¸øÇß½À´Ï´Ù. (¿¡·¯¹øÈ£: %d)" + nor "Kan ikke opprette fila '%-.200s' (Feilkode: %d)" + norwegian-ny "Kan ikkje opprette fila '%-.200s' (Feilkode: %d)" + pol "Nie mo¿na stworzyæ pliku '%-.200s' (Kod b³êdu: %d)" + por "Não pode criar o arquivo '%-.200s' (erro no. %d)" + rum "Nu pot sa creez fisierul '%-.200s' (Eroare: %d)" + rus "îÅ×ÏÚÍÏÖÎÏ ÓÏÚÄÁÔØ ÆÁÊÌ '%-.200s' (ÏÛÉÂËÁ: %d)" + serbian "Ne mogu da kreiram file '%-.200s' (errno: %d)" + slo "Nemô¾em vytvori» súbor '%-.200s' (chybový kód: %d)" + spa "No puedo crear archivo '%-.200s' (Error: %d)" + swe "Kan inte skapa filen '%-.200s' (Felkod: %d)" + ukr "îÅ ÍÏÖÕ ÓÔ×ÏÒÉÔÉ ÆÁÊÌ '%-.200s' (ÐÏÍÉÌËÁ: %d)" ER_CANT_CREATE_TABLE - cze "Nemohu vytvo-Bøit tabulku '%-.64s' (chybový kód: %d)" - dan "Kan ikke oprette tabellen '%-.64s' (Fejlkode: %d)" - nla "Kan tabel '%-.64s' niet aanmaken (Errcode: %d)" - eng "Can't create table '%-.64s' (errno: %d)" - jps "'%-.64s' ƒe[ƒuƒ‹‚ªì‚ê‚Ü‚¹‚ñ.(errno: %d)", - est "Ei suuda luua tabelit '%-.64s' (veakood: %d)" - fre "Ne peut créer la table '%-.64s' (Errcode: %d)" - ger "Kann Tabelle '%-.64s' nicht erzeugen (Fehler: %d)" - greek "Áäýíáôç ç äçìéïõñãßá ôïõ ðßíáêá '%-.64s' (êùäéêüò ëÜèïõò: %d)" - hun "A '%-.64s' tabla nem hozhato letre (hibakod: %d)" - ita "Impossibile creare la tabella '%-.64s' (errno: %d)" - jpn "'%-.64s' ¥Æ¡¼¥Ö¥ë¤¬ºî¤ì¤Þ¤»¤ó.(errno: %d)" - kor "Å×À̺í '%-.64s'¸¦ ¸¸µéÁö ¸øÇß½À´Ï´Ù. (¿¡·¯¹øÈ£: %d)" - nor "Kan ikke opprette tabellen '%-.64s' (Feilkode: %d)" - norwegian-ny "Kan ikkje opprette tabellen '%-.64s' (Feilkode: %d)" - pol "Nie mo¿na stworzyæ tabeli '%-.64s' (Kod b³êdu: %d)" - por "Não pode criar a tabela '%-.64s' (erro no. %d)" - rum "Nu pot sa creez tabla '%-.64s' (Eroare: %d)" - rus "îÅ×ÏÚÍÏÖÎÏ ÓÏÚÄÁÔØ ÔÁÂÌÉÃÕ '%-.64s' (ÏÛÉÂËÁ: %d)" - serbian "Ne mogu da kreiram tabelu '%-.64s' (errno: %d)" - slo "Nemô¾em vytvori» tabuµku '%-.64s' (chybový kód: %d)" - spa "No puedo crear tabla '%-.64s' (Error: %d)" - swe "Kan inte skapa tabellen '%-.64s' (Felkod: %d)" - ukr "îÅ ÍÏÖÕ ÓÔ×ÏÒÉÔÉ ÔÁÂÌÉÃÀ '%-.64s' (ÐÏÍÉÌËÁ: %d)" + cze "Nemohu vytvo-Bøit tabulku '%-.200s' (chybový kód: %d)" + dan "Kan ikke oprette tabellen '%-.200s' (Fejlkode: %d)" + nla "Kan tabel '%-.200s' niet aanmaken (Errcode: %d)" + eng "Can't create table '%-.200s' (errno: %d)" + jps "'%-.200s' ƒe[ƒuƒ‹‚ªì‚ê‚Ü‚¹‚ñ.(errno: %d)", + est "Ei suuda luua tabelit '%-.200s' (veakood: %d)" + fre "Ne peut créer la table '%-.200s' (Errcode: %d)" + ger "Kann Tabelle '%-.200s' nicht erzeugen (Fehler: %d)" + greek "Áäýíáôç ç äçìéïõñãßá ôïõ ðßíáêá '%-.200s' (êùäéêüò ëÜèïõò: %d)" + hun "A '%-.200s' tabla nem hozhato letre (hibakod: %d)" + ita "Impossibile creare la tabella '%-.200s' (errno: %d)" + jpn "'%-.200s' ¥Æ¡¼¥Ö¥ë¤¬ºî¤ì¤Þ¤»¤ó.(errno: %d)" + kor "Å×À̺í '%-.200s'¸¦ ¸¸µéÁö ¸øÇß½À´Ï´Ù. (¿¡·¯¹øÈ£: %d)" + nor "Kan ikke opprette tabellen '%-.200s' (Feilkode: %d)" + norwegian-ny "Kan ikkje opprette tabellen '%-.200s' (Feilkode: %d)" + pol "Nie mo¿na stworzyæ tabeli '%-.200s' (Kod b³êdu: %d)" + por "Não pode criar a tabela '%-.200s' (erro no. %d)" + rum "Nu pot sa creez tabla '%-.200s' (Eroare: %d)" + rus "îÅ×ÏÚÍÏÖÎÏ ÓÏÚÄÁÔØ ÔÁÂÌÉÃÕ '%-.200s' (ÏÛÉÂËÁ: %d)" + serbian "Ne mogu da kreiram tabelu '%-.200s' (errno: %d)" + slo "Nemô¾em vytvori» tabuµku '%-.200s' (chybový kód: %d)" + spa "No puedo crear tabla '%-.200s' (Error: %d)" + swe "Kan inte skapa tabellen '%-.200s' (Felkod: %d)" + ukr "îÅ ÍÏÖÕ ÓÔ×ÏÒÉÔÉ ÔÁÂÌÉÃÀ '%-.200s' (ÐÏÍÉÌËÁ: %d)" ER_CANT_CREATE_DB cze "Nemohu vytvo-Bøit databázi '%-.64s' (chybový kód: %d)" dan "Kan ikke oprette databasen '%-.64s' (Fejlkode: %d)" @@ -275,30 +275,30 @@ ER_CANT_FIND_SYSTEM_REC swe "Hittar inte posten i systemregistret" ukr "îÅ ÍÏÖÕ ÚÞÉÔÁÔÉ ÚÁÐÉÓ Ú ÓÉÓÔÅÍÎϧ ÔÁÂÌÉæ" ER_CANT_GET_STAT - cze "Nemohu z-Bískat stav '%-.64s' (chybový kód: %d)" - dan "Kan ikke læse status af '%-.64s' (Fejlkode: %d)" - nla "Kan de status niet krijgen van '%-.64s' (Errcode: %d)" + cze "Nemohu z-Bískat stav '%-.200s' (chybový kód: %d)" + dan "Kan ikke læse status af '%-.200s' (Fejlkode: %d)" + nla "Kan de status niet krijgen van '%-.200s' (Errcode: %d)" eng "Can't get status of '%-.200s' (errno: %d)" - jps "'%-.64s' ‚̃XƒeƒCƒ^ƒX‚ª“¾‚ç‚ê‚Ü‚¹‚ñ. (errno: %d)", - est "Ei suuda lugeda '%-.64s' olekut (veakood: %d)" - fre "Ne peut obtenir le status de '%-.64s' (Errcode: %d)" - ger "Kann Status von '%-.64s' nicht ermitteln (Fehler: %d)" - greek "Áäýíáôç ç ëÞøç ðëçñïöïñéþí ãéá ôçí êáôÜóôáóç ôïõ '%-.64s' (êùäéêüò ëÜèïõò: %d)" - hun "A(z) '%-.64s' statusza nem allapithato meg (hibakod: %d)" - ita "Impossibile leggere lo stato di '%-.64s' (errno: %d)" - jpn "'%-.64s' ¤Î¥¹¥Æ¥¤¥¿¥¹¤¬ÆÀ¤é¤ì¤Þ¤»¤ó. (errno: %d)" - kor "'%-.64s'ÀÇ »óŸ¦ ¾òÁö ¸øÇß½À´Ï´Ù. (¿¡·¯¹øÈ£: %d)" - nor "Kan ikke lese statusen til '%-.64s' (Feilkode: %d)" - norwegian-ny "Kan ikkje lese statusen til '%-.64s' (Feilkode: %d)" - pol "Nie mo¿na otrzymaæ statusu '%-.64s' (Kod b³êdu: %d)" - por "Não pode obter o status de '%-.64s' (erro no. %d)" - rum "Nu pot sa obtin statusul lui '%-.64s' (Eroare: %d)" - rus "îÅ×ÏÚÍÏÖÎÏ ÐÏÌÕÞÉÔØ ÓÔÁÔÕÓÎÕÀ ÉÎÆÏÒÍÁÃÉÀ Ï '%-.64s' (ÏÛÉÂËÁ: %d)" - serbian "Ne mogu da dobijem stanje file-a '%-.64s' (errno: %d)" - slo "Nemô¾em zisti» stav '%-.64s' (chybový kód: %d)" - spa "No puedo obtener el estado de '%-.64s' (Error: %d)" - swe "Kan inte läsa filinformationen (stat) från '%-.64s' (Felkod: %d)" - ukr "îÅ ÍÏÖÕ ÏÔÒÉÍÁÔÉ ÓÔÁÔÕÓ '%-.64s' (ÐÏÍÉÌËÁ: %d)" + jps "'%-.200s' ‚̃XƒeƒCƒ^ƒX‚ª“¾‚ç‚ê‚Ü‚¹‚ñ. (errno: %d)", + est "Ei suuda lugeda '%-.200s' olekut (veakood: %d)" + fre "Ne peut obtenir le status de '%-.200s' (Errcode: %d)" + ger "Kann Status von '%-.200s' nicht ermitteln (Fehler: %d)" + greek "Áäýíáôç ç ëÞøç ðëçñïöïñéþí ãéá ôçí êáôÜóôáóç ôïõ '%-.200s' (êùäéêüò ëÜèïõò: %d)" + hun "A(z) '%-.200s' statusza nem allapithato meg (hibakod: %d)" + ita "Impossibile leggere lo stato di '%-.200s' (errno: %d)" + jpn "'%-.200s' ¤Î¥¹¥Æ¥¤¥¿¥¹¤¬ÆÀ¤é¤ì¤Þ¤»¤ó. (errno: %d)" + kor "'%-.200s'ÀÇ »óŸ¦ ¾òÁö ¸øÇß½À´Ï´Ù. (¿¡·¯¹øÈ£: %d)" + nor "Kan ikke lese statusen til '%-.200s' (Feilkode: %d)" + norwegian-ny "Kan ikkje lese statusen til '%-.200s' (Feilkode: %d)" + pol "Nie mo¿na otrzymaæ statusu '%-.200s' (Kod b³êdu: %d)" + por "Não pode obter o status de '%-.200s' (erro no. %d)" + rum "Nu pot sa obtin statusul lui '%-.200s' (Eroare: %d)" + rus "îÅ×ÏÚÍÏÖÎÏ ÐÏÌÕÞÉÔØ ÓÔÁÔÕÓÎÕÀ ÉÎÆÏÒÍÁÃÉÀ Ï '%-.200s' (ÏÛÉÂËÁ: %d)" + serbian "Ne mogu da dobijem stanje file-a '%-.200s' (errno: %d)" + slo "Nemô¾em zisti» stav '%-.200s' (chybový kód: %d)" + spa "No puedo obtener el estado de '%-.200s' (Error: %d)" + swe "Kan inte läsa filinformationen (stat) från '%-.200s' (Felkod: %d)" + ukr "îÅ ÍÏÖÕ ÏÔÒÉÍÁÔÉ ÓÔÁÔÕÓ '%-.200s' (ÐÏÍÉÌËÁ: %d)" ER_CANT_GET_WD cze "Chyba p-Bøi zji¹»ování pracovní adresáø (chybový kód: %d)" dan "Kan ikke læse aktive folder (Fejlkode: %d)" @@ -350,55 +350,55 @@ ER_CANT_LOCK swe "Kan inte låsa filen. (Felkod: %d)" ukr "îÅ ÍÏÖÕ ÚÁÂÌÏËÕ×ÁÔÉ ÆÁÊÌ (ÐÏÍÉÌËÁ: %d)" ER_CANT_OPEN_FILE - cze "Nemohu otev-Bøít soubor '%-.64s' (chybový kód: %d)" - dan "Kan ikke åbne fil: '%-.64s' (Fejlkode: %d)" - nla "Kan de file '%-.64s' niet openen (Errcode: %d)" + cze "Nemohu otev-Bøít soubor '%-.200s' (chybový kód: %d)" + dan "Kan ikke åbne fil: '%-.200s' (Fejlkode: %d)" + nla "Kan de file '%-.200s' niet openen (Errcode: %d)" eng "Can't open file: '%-.200s' (errno: %d)" - jps "'%-.64s' ƒtƒ@ƒCƒ‹‚ðŠJ‚­Ž–‚ª‚Å‚«‚Ü‚¹‚ñ (errno: %d)", - est "Ei suuda avada faili '%-.64s' (veakood: %d)" - fre "Ne peut ouvrir le fichier: '%-.64s' (Errcode: %d)" - ger "Kann Datei '%-.64s' nicht öffnen (Fehler: %d)" - greek "Äåí åßíáé äõíáôü íá áíïé÷ôåß ôï áñ÷åßï: '%-.64s' (êùäéêüò ëÜèïõò: %d)" - hun "A '%-.64s' file nem nyithato meg (hibakod: %d)" - ita "Impossibile aprire il file: '%-.64s' (errno: %d)" - jpn "'%-.64s' ¥Õ¥¡¥¤¥ë¤ò³«¤¯»ö¤¬¤Ç¤­¤Þ¤»¤ó (errno: %d)" - kor "È­ÀÏÀ» ¿­Áö ¸øÇß½À´Ï´Ù.: '%-.64s' (¿¡·¯¹øÈ£: %d)" - nor "Kan ikke åpne fila: '%-.64s' (Feilkode: %d)" - norwegian-ny "Kan ikkje åpne fila: '%-.64s' (Feilkode: %d)" - pol "Nie mo¿na otworzyæ pliku: '%-.64s' (Kod b³êdu: %d)" - por "Não pode abrir o arquivo '%-.64s' (erro no. %d)" - rum "Nu pot sa deschid fisierul: '%-.64s' (Eroare: %d)" - rus "îÅ×ÏÚÍÏÖÎÏ ÏÔËÒÙÔØ ÆÁÊÌ: '%-.64s' (ÏÛÉÂËÁ: %d)" - serbian "Ne mogu da otvorim file: '%-.64s' (errno: %d)" - slo "Nemô¾em otvori» súbor: '%-.64s' (chybový kód: %d)" - spa "No puedo abrir archivo: '%-.64s' (Error: %d)" - swe "Kan inte använda '%-.64s' (Felkod: %d)" - ukr "îÅ ÍÏÖÕ ×¦ÄËÒÉÔÉ ÆÁÊÌ: '%-.64s' (ÐÏÍÉÌËÁ: %d)" + jps "'%-.200s' ƒtƒ@ƒCƒ‹‚ðŠJ‚­Ž–‚ª‚Å‚«‚Ü‚¹‚ñ (errno: %d)", + est "Ei suuda avada faili '%-.200s' (veakood: %d)" + fre "Ne peut ouvrir le fichier: '%-.200s' (Errcode: %d)" + ger "Kann Datei '%-.200s' nicht öffnen (Fehler: %d)" + greek "Äåí åßíáé äõíáôü íá áíïé÷ôåß ôï áñ÷åßï: '%-.200s' (êùäéêüò ëÜèïõò: %d)" + hun "A '%-.200s' file nem nyithato meg (hibakod: %d)" + ita "Impossibile aprire il file: '%-.200s' (errno: %d)" + jpn "'%-.200s' ¥Õ¥¡¥¤¥ë¤ò³«¤¯»ö¤¬¤Ç¤­¤Þ¤»¤ó (errno: %d)" + kor "È­ÀÏÀ» ¿­Áö ¸øÇß½À´Ï´Ù.: '%-.200s' (¿¡·¯¹øÈ£: %d)" + nor "Kan ikke åpne fila: '%-.200s' (Feilkode: %d)" + norwegian-ny "Kan ikkje åpne fila: '%-.200s' (Feilkode: %d)" + pol "Nie mo¿na otworzyæ pliku: '%-.200s' (Kod b³êdu: %d)" + por "Não pode abrir o arquivo '%-.200s' (erro no. %d)" + rum "Nu pot sa deschid fisierul: '%-.200s' (Eroare: %d)" + rus "îÅ×ÏÚÍÏÖÎÏ ÏÔËÒÙÔØ ÆÁÊÌ: '%-.200s' (ÏÛÉÂËÁ: %d)" + serbian "Ne mogu da otvorim file: '%-.200s' (errno: %d)" + slo "Nemô¾em otvori» súbor: '%-.200s' (chybový kód: %d)" + spa "No puedo abrir archivo: '%-.200s' (Error: %d)" + swe "Kan inte använda '%-.200s' (Felkod: %d)" + ukr "îÅ ÍÏÖÕ ×¦ÄËÒÉÔÉ ÆÁÊÌ: '%-.200s' (ÐÏÍÉÌËÁ: %d)" ER_FILE_NOT_FOUND - cze "Nemohu naj-Bít soubor '%-.64s' (chybový kód: %d)" - dan "Kan ikke finde fila: '%-.64s' (Fejlkode: %d)" - nla "Kan de file: '%-.64s' niet vinden (Errcode: %d)" + cze "Nemohu naj-Bít soubor '%-.200s' (chybový kód: %d)" + dan "Kan ikke finde fila: '%-.200s' (Fejlkode: %d)" + nla "Kan de file: '%-.200s' niet vinden (Errcode: %d)" eng "Can't find file: '%-.200s' (errno: %d)" - jps "'%-.64s' ƒtƒ@ƒCƒ‹‚ðŒ©•t‚¯‚鎖‚ª‚Å‚«‚Ü‚¹‚ñ.(errno: %d)", - est "Ei suuda leida faili '%-.64s' (veakood: %d)" - fre "Ne peut trouver le fichier: '%-.64s' (Errcode: %d)" - ger "Kann Datei '%-.64s' nicht finden (Fehler: %d)" - greek "Äåí âñÝèçêå ôï áñ÷åßï: '%-.64s' (êùäéêüò ëÜèïõò: %d)" - hun "A(z) '%-.64s' file nem talalhato (hibakod: %d)" - ita "Impossibile trovare il file: '%-.64s' (errno: %d)" - jpn "'%-.64s' ¥Õ¥¡¥¤¥ë¤ò¸«ÉÕ¤±¤ë»ö¤¬¤Ç¤­¤Þ¤»¤ó.(errno: %d)" - kor "È­ÀÏÀ» ãÁö ¸øÇß½À´Ï´Ù.: '%-.64s' (¿¡·¯¹øÈ£: %d)" - nor "Kan ikke finne fila: '%-.64s' (Feilkode: %d)" - norwegian-ny "Kan ikkje finne fila: '%-.64s' (Feilkode: %d)" - pol "Nie mo¿na znale¥æ pliku: '%-.64s' (Kod b³êdu: %d)" - por "Não pode encontrar o arquivo '%-.64s' (erro no. %d)" - rum "Nu pot sa gasesc fisierul: '%-.64s' (Eroare: %d)" - rus "îÅ×ÏÚÍÏÖÎÏ ÎÁÊÔÉ ÆÁÊÌ: '%-.64s' (ÏÛÉÂËÁ: %d)" - serbian "Ne mogu da pronaðem file: '%-.64s' (errno: %d)" - slo "Nemô¾em nájs» súbor: '%-.64s' (chybový kód: %d)" - spa "No puedo encontrar archivo: '%-.64s' (Error: %d)" - swe "Hittar inte filen '%-.64s' (Felkod: %d)" - ukr "îÅ ÍÏÖÕ ÚÎÁÊÔÉ ÆÁÊÌ: '%-.64s' (ÐÏÍÉÌËÁ: %d)" + jps "'%-.200s' ƒtƒ@ƒCƒ‹‚ðŒ©•t‚¯‚鎖‚ª‚Å‚«‚Ü‚¹‚ñ.(errno: %d)", + est "Ei suuda leida faili '%-.200s' (veakood: %d)" + fre "Ne peut trouver le fichier: '%-.200s' (Errcode: %d)" + ger "Kann Datei '%-.200s' nicht finden (Fehler: %d)" + greek "Äåí âñÝèçêå ôï áñ÷åßï: '%-.200s' (êùäéêüò ëÜèïõò: %d)" + hun "A(z) '%-.200s' file nem talalhato (hibakod: %d)" + ita "Impossibile trovare il file: '%-.200s' (errno: %d)" + jpn "'%-.200s' ¥Õ¥¡¥¤¥ë¤ò¸«ÉÕ¤±¤ë»ö¤¬¤Ç¤­¤Þ¤»¤ó.(errno: %d)" + kor "È­ÀÏÀ» ãÁö ¸øÇß½À´Ï´Ù.: '%-.200s' (¿¡·¯¹øÈ£: %d)" + nor "Kan ikke finne fila: '%-.200s' (Feilkode: %d)" + norwegian-ny "Kan ikkje finne fila: '%-.200s' (Feilkode: %d)" + pol "Nie mo¿na znale¥æ pliku: '%-.200s' (Kod b³êdu: %d)" + por "Não pode encontrar o arquivo '%-.200s' (erro no. %d)" + rum "Nu pot sa gasesc fisierul: '%-.200s' (Eroare: %d)" + rus "îÅ×ÏÚÍÏÖÎÏ ÎÁÊÔÉ ÆÁÊÌ: '%-.200s' (ÏÛÉÂËÁ: %d)" + serbian "Ne mogu da pronaðem file: '%-.200s' (errno: %d)" + slo "Nemô¾em nájs» súbor: '%-.200s' (chybový kód: %d)" + spa "No puedo encontrar archivo: '%-.200s' (Error: %d)" + swe "Hittar inte filen '%-.200s' (Felkod: %d)" + ukr "îÅ ÍÏÖÕ ÚÎÁÊÔÉ ÆÁÊÌ: '%-.200s' (ÐÏÍÉÌËÁ: %d)" ER_CANT_READ_DIR cze "Nemohu -Bèíst adresáø '%-.64s' (chybový kód: %d)" dan "Kan ikke læse folder '%-.64s' (Fejlkode: %d)" @@ -480,7 +480,7 @@ ER_DISK_FULL jps "Disk full (%s). ’N‚©‚ª‰½‚©‚ðŒ¸‚ç‚·‚Ü‚Å‚Ü‚Á‚Ä‚­‚¾‚³‚¢...", est "Ketas täis (%s). Ootame kuni tekib vaba ruumi..." fre "Disque plein (%s). J'attend que quelqu'un libère de l'espace..." - ger "Festplatte voll (%-.64s). Warte, bis jemand Platz schafft ..." + ger "Festplatte voll (%s). Warte, bis jemand Platz schafft ..." greek "Äåí õðÜñ÷åé ÷þñïò óôï äßóêï (%s). Ðáñáêáëþ, ðåñéìÝíåôå íá åëåõèåñùèåß ÷þñïò..." hun "A lemez megtelt (%s)." ita "Disco pieno (%s). In attesa che qualcuno liberi un po' di spazio..." @@ -546,80 +546,80 @@ ER_ERROR_ON_CLOSE swe "Fick fel vid stängning av '%-.64s' (Felkod: %d)" ukr "îÅ ÍÏÖÕ ÚÁËÒÉÔÉ '%-.64s' (ÐÏÍÉÌËÁ: %d)" ER_ERROR_ON_READ - cze "Chyba p-Bøi ètení souboru '%-.64s' (chybový kód: %d)" - dan "Fejl ved læsning af '%-.64s' (Fejlkode: %d)" - nla "Fout bij het lezen van file '%-.64s' (Errcode: %d)" + cze "Chyba p-Bøi ètení souboru '%-.200s' (chybový kód: %d)" + dan "Fejl ved læsning af '%-.200s' (Fejlkode: %d)" + nla "Fout bij het lezen van file '%-.200s' (Errcode: %d)" eng "Error reading file '%-.200s' (errno: %d)" - jps "'%-.64s' ƒtƒ@ƒCƒ‹‚Ì“Ç‚Ýž‚݃Gƒ‰[ (errno: %d)", - est "Viga faili '%-.64s' lugemisel (veakood: %d)" - fre "Erreur en lecture du fichier '%-.64s' (Errcode: %d)" - ger "Fehler beim Lesen der Datei '%-.64s' (Fehler: %d)" - greek "Ðñüâëçìá êáôÜ ôçí áíÜãíùóç ôïõ áñ÷åßïõ '%-.64s' (êùäéêüò ëÜèïõò: %d)" - hun "Hiba a '%-.64s'file olvasasakor. (hibakod: %d)" - ita "Errore durante la lettura del file '%-.64s' (errno: %d)" - jpn "'%-.64s' ¥Õ¥¡¥¤¥ë¤ÎÆɤ߹þ¤ß¥¨¥é¡¼ (errno: %d)" - kor "'%-.64s'È­ÀÏ Àб⠿¡·¯ (¿¡·¯¹øÈ£: %d)" - nor "Feil ved lesing av '%-.64s' (Feilkode: %d)" - norwegian-ny "Feil ved lesing av '%-.64s' (Feilkode: %d)" - pol "B³?d podczas odczytu pliku '%-.64s' (Kod b³êdu: %d)" - por "Erro ao ler arquivo '%-.64s' (erro no. %d)" - rum "Eroare citind fisierul '%-.64s' (errno: %d)" - rus "ïÛÉÂËÁ ÞÔÅÎÉÑ ÆÁÊÌÁ '%-.64s' (ÏÛÉÂËÁ: %d)" - serbian "Greška pri èitanju file-a '%-.64s' (errno: %d)" - slo "Chyba pri èítaní súboru '%-.64s' (chybový kód: %d)" - spa "Error leyendo el fichero '%-.64s' (Error: %d)" - swe "Fick fel vid läsning av '%-.64s' (Felkod %d)" - ukr "îÅ ÍÏÖÕ ÐÒÏÞÉÔÁÔÉ ÆÁÊÌ '%-.64s' (ÐÏÍÉÌËÁ: %d)" + jps "'%-.200s' ƒtƒ@ƒCƒ‹‚Ì“Ç‚Ýž‚݃Gƒ‰[ (errno: %d)", + est "Viga faili '%-.200s' lugemisel (veakood: %d)" + fre "Erreur en lecture du fichier '%-.200s' (Errcode: %d)" + ger "Fehler beim Lesen der Datei '%-.200s' (Fehler: %d)" + greek "Ðñüâëçìá êáôÜ ôçí áíÜãíùóç ôïõ áñ÷åßïõ '%-.200s' (êùäéêüò ëÜèïõò: %d)" + hun "Hiba a '%-.200s'file olvasasakor. (hibakod: %d)" + ita "Errore durante la lettura del file '%-.200s' (errno: %d)" + jpn "'%-.200s' ¥Õ¥¡¥¤¥ë¤ÎÆɤ߹þ¤ß¥¨¥é¡¼ (errno: %d)" + kor "'%-.200s'È­ÀÏ Àб⠿¡·¯ (¿¡·¯¹øÈ£: %d)" + nor "Feil ved lesing av '%-.200s' (Feilkode: %d)" + norwegian-ny "Feil ved lesing av '%-.200s' (Feilkode: %d)" + pol "B³?d podczas odczytu pliku '%-.200s' (Kod b³êdu: %d)" + por "Erro ao ler arquivo '%-.200s' (erro no. %d)" + rum "Eroare citind fisierul '%-.200s' (errno: %d)" + rus "ïÛÉÂËÁ ÞÔÅÎÉÑ ÆÁÊÌÁ '%-.200s' (ÏÛÉÂËÁ: %d)" + serbian "Greška pri èitanju file-a '%-.200s' (errno: %d)" + slo "Chyba pri èítaní súboru '%-.200s' (chybový kód: %d)" + spa "Error leyendo el fichero '%-.200s' (Error: %d)" + swe "Fick fel vid läsning av '%-.200s' (Felkod %d)" + ukr "îÅ ÍÏÖÕ ÐÒÏÞÉÔÁÔÉ ÆÁÊÌ '%-.200s' (ÐÏÍÉÌËÁ: %d)" ER_ERROR_ON_RENAME - cze "Chyba p-Bøi pøejmenování '%-.64s' na '%-.64s' (chybový kód: %d)" - dan "Fejl ved omdøbning af '%-.64s' til '%-.64s' (Fejlkode: %d)" - nla "Fout bij het hernoemen van '%-.64s' naar '%-.64s' (Errcode: %d)" - eng "Error on rename of '%-.64s' to '%-.64s' (errno: %d)" - jps "'%-.64s' ‚ð '%-.64s' ‚É rename ‚Å‚«‚Ü‚¹‚ñ (errno: %d)", - est "Viga faili '%-.64s' ümbernimetamisel '%-.64s'-ks (veakood: %d)" - fre "Erreur en renommant '%-.64s' en '%-.64s' (Errcode: %d)" - ger "Fehler beim Umbenennen von '%-.64s' in '%-.64s' (Fehler: %d)" - greek "Ðñüâëçìá êáôÜ ôçí ìåôïíïìáóßá ôïõ áñ÷åßïõ '%-.64s' to '%-.64s' (êùäéêüò ëÜèïõò: %d)" - hun "Hiba a '%-.64s' file atnevezesekor. (hibakod: %d)" - ita "Errore durante la rinominazione da '%-.64s' a '%-.64s' (errno: %d)" - jpn "'%-.64s' ¤ò '%-.64s' ¤Ë rename ¤Ç¤­¤Þ¤»¤ó (errno: %d)" - kor "'%-.64s'¸¦ '%-.64s'·Î À̸§ º¯°æÁß ¿¡·¯ (¿¡·¯¹øÈ£: %d)" - nor "Feil ved omdøping av '%-.64s' til '%-.64s' (Feilkode: %d)" - norwegian-ny "Feil ved omdøyping av '%-.64s' til '%-.64s' (Feilkode: %d)" - pol "B³?d podczas zmieniania nazwy '%-.64s' na '%-.64s' (Kod b³êdu: %d)" - por "Erro ao renomear '%-.64s' para '%-.64s' (erro no. %d)" - rum "Eroare incercind sa renumesc '%-.64s' in '%-.64s' (errno: %d)" - rus "ïÛÉÂËÁ ÐÒÉ ÐÅÒÅÉÍÅÎÏ×ÁÎÉÉ '%-.64s' × '%-.64s' (ÏÛÉÂËÁ: %d)" - serbian "Greška pri promeni imena '%-.64s' na '%-.64s' (errno: %d)" - slo "Chyba pri premenovávaní '%-.64s' na '%-.64s' (chybový kód: %d)" - spa "Error en el renombrado de '%-.64s' a '%-.64s' (Error: %d)" - swe "Kan inte byta namn från '%-.64s' till '%-.64s' (Felkod: %d)" - ukr "îÅ ÍÏÖÕ ÐÅÒÅÊÍÅÎÕ×ÁÔÉ '%-.64s' Õ '%-.64s' (ÐÏÍÉÌËÁ: %d)" + cze "Chyba p-Bøi pøejmenování '%-.150s' na '%-.150s' (chybový kód: %d)" + dan "Fejl ved omdøbning af '%-.150s' til '%-.150s' (Fejlkode: %d)" + nla "Fout bij het hernoemen van '%-.150s' naar '%-.150s' (Errcode: %d)" + eng "Error on rename of '%-.150s' to '%-.150s' (errno: %d)" + jps "'%-.150s' ‚ð '%-.150s' ‚É rename ‚Å‚«‚Ü‚¹‚ñ (errno: %d)", + est "Viga faili '%-.150s' ümbernimetamisel '%-.150s'-ks (veakood: %d)" + fre "Erreur en renommant '%-.150s' en '%-.150s' (Errcode: %d)" + ger "Fehler beim Umbenennen von '%-.150s' in '%-.150s' (Fehler: %d)" + greek "Ðñüâëçìá êáôÜ ôçí ìåôïíïìáóßá ôïõ áñ÷åßïõ '%-.150s' to '%-.150s' (êùäéêüò ëÜèïõò: %d)" + hun "Hiba a '%-.150s' file atnevezesekor '%-.150s'. (hibakod: %d)" + ita "Errore durante la rinominazione da '%-.150s' a '%-.150s' (errno: %d)" + jpn "'%-.150s' ¤ò '%-.150s' ¤Ë rename ¤Ç¤­¤Þ¤»¤ó (errno: %d)" + kor "'%-.150s'¸¦ '%-.150s'·Î À̸§ º¯°æÁß ¿¡·¯ (¿¡·¯¹øÈ£: %d)" + nor "Feil ved omdøping av '%-.150s' til '%-.150s' (Feilkode: %d)" + norwegian-ny "Feil ved omdøyping av '%-.150s' til '%-.150s' (Feilkode: %d)" + pol "B³?d podczas zmieniania nazwy '%-.150s' na '%-.150s' (Kod b³êdu: %d)" + por "Erro ao renomear '%-.150s' para '%-.150s' (erro no. %d)" + rum "Eroare incercind sa renumesc '%-.150s' in '%-.150s' (errno: %d)" + rus "ïÛÉÂËÁ ÐÒÉ ÐÅÒÅÉÍÅÎÏ×ÁÎÉÉ '%-.150s' × '%-.150s' (ÏÛÉÂËÁ: %d)" + serbian "Greška pri promeni imena '%-.150s' na '%-.150s' (errno: %d)" + slo "Chyba pri premenovávaní '%-.150s' na '%-.150s' (chybový kód: %d)" + spa "Error en el renombrado de '%-.150s' a '%-.150s' (Error: %d)" + swe "Kan inte byta namn från '%-.150s' till '%-.150s' (Felkod: %d)" + ukr "îÅ ÍÏÖÕ ÐÅÒÅÊÍÅÎÕ×ÁÔÉ '%-.150s' Õ '%-.150s' (ÐÏÍÉÌËÁ: %d)" ER_ERROR_ON_WRITE - cze "Chyba p-Bøi zápisu do souboru '%-.64s' (chybový kód: %d)" - dan "Fejl ved skriving av filen '%-.64s' (Fejlkode: %d)" - nla "Fout bij het wegschrijven van file '%-.64s' (Errcode: %d)" + cze "Chyba p-Bøi zápisu do souboru '%-.200s' (chybový kód: %d)" + dan "Fejl ved skriving av filen '%-.200s' (Fejlkode: %d)" + nla "Fout bij het wegschrijven van file '%-.200s' (Errcode: %d)" eng "Error writing file '%-.200s' (errno: %d)" - jps "'%-.64s' ƒtƒ@ƒCƒ‹‚ð‘‚­Ž–‚ª‚Å‚«‚Ü‚¹‚ñ (errno: %d)", - est "Viga faili '%-.64s' kirjutamisel (veakood: %d)" - fre "Erreur d'écriture du fichier '%-.64s' (Errcode: %d)" - ger "Fehler beim Speichern der Datei '%-.64s' (Fehler: %d)" - greek "Ðñüâëçìá êáôÜ ôçí áðïèÞêåõóç ôïõ áñ÷åßïõ '%-.64s' (êùäéêüò ëÜèïõò: %d)" - hun "Hiba a '%-.64s' file irasakor. (hibakod: %d)" - ita "Errore durante la scrittura del file '%-.64s' (errno: %d)" - jpn "'%-.64s' ¥Õ¥¡¥¤¥ë¤ò½ñ¤¯»ö¤¬¤Ç¤­¤Þ¤»¤ó (errno: %d)" - kor "'%-.64s'È­ÀÏ ±â·Ï Áß ¿¡·¯ (¿¡·¯¹øÈ£: %d)" - nor "Feil ved skriving av fila '%-.64s' (Feilkode: %d)" - norwegian-ny "Feil ved skriving av fila '%-.64s' (Feilkode: %d)" - pol "B³?d podczas zapisywania pliku '%-.64s' (Kod b³êdu: %d)" - por "Erro ao gravar arquivo '%-.64s' (erro no. %d)" - rum "Eroare scriind fisierul '%-.64s' (errno: %d)" - rus "ïÛÉÂËÁ ÚÁÐÉÓÉ × ÆÁÊÌ '%-.64s' (ÏÛÉÂËÁ: %d)" - serbian "Greška pri upisu '%-.64s' (errno: %d)" - slo "Chyba pri zápise do súboru '%-.64s' (chybový kód: %d)" - spa "Error escribiendo el archivo '%-.64s' (Error: %d)" - swe "Fick fel vid skrivning till '%-.64s' (Felkod %d)" - ukr "îÅ ÍÏÖÕ ÚÁÐÉÓÁÔÉ ÆÁÊÌ '%-.64s' (ÐÏÍÉÌËÁ: %d)" + jps "'%-.200s' ƒtƒ@ƒCƒ‹‚ð‘‚­Ž–‚ª‚Å‚«‚Ü‚¹‚ñ (errno: %d)", + est "Viga faili '%-.200s' kirjutamisel (veakood: %d)" + fre "Erreur d'écriture du fichier '%-.200s' (Errcode: %d)" + ger "Fehler beim Speichern der Datei '%-.200s' (Fehler: %d)" + greek "Ðñüâëçìá êáôÜ ôçí áðïèÞêåõóç ôïõ áñ÷åßïõ '%-.200s' (êùäéêüò ëÜèïõò: %d)" + hun "Hiba a '%-.200s' file irasakor. (hibakod: %d)" + ita "Errore durante la scrittura del file '%-.200s' (errno: %d)" + jpn "'%-.200s' ¥Õ¥¡¥¤¥ë¤ò½ñ¤¯»ö¤¬¤Ç¤­¤Þ¤»¤ó (errno: %d)" + kor "'%-.200s'È­ÀÏ ±â·Ï Áß ¿¡·¯ (¿¡·¯¹øÈ£: %d)" + nor "Feil ved skriving av fila '%-.200s' (Feilkode: %d)" + norwegian-ny "Feil ved skriving av fila '%-.200s' (Feilkode: %d)" + pol "B³?d podczas zapisywania pliku '%-.200s' (Kod b³êdu: %d)" + por "Erro ao gravar arquivo '%-.200s' (erro no. %d)" + rum "Eroare scriind fisierul '%-.200s' (errno: %d)" + rus "ïÛÉÂËÁ ÚÁÐÉÓÉ × ÆÁÊÌ '%-.200s' (ÏÛÉÂËÁ: %d)" + serbian "Greška pri upisu '%-.200s' (errno: %d)" + slo "Chyba pri zápise do súboru '%-.200s' (chybový kód: %d)" + spa "Error escribiendo el archivo '%-.200s' (Error: %d)" + swe "Fick fel vid skrivning till '%-.200s' (Felkod %d)" + ukr "îÅ ÍÏÖÕ ÚÁÐÉÓÁÔÉ ÆÁÊÌ '%-.200s' (ÐÏÍÉÌËÁ: %d)" ER_FILE_USED cze "'%-.64s' je zam-Bèen proti zmìnám" dan "'%-.64s' er låst mod opdateringer" @@ -741,7 +741,7 @@ ER_ILLEGAL_HA serbian "Handler tabela za '%-.64s' nema ovu opciju" slo "Obsluha tabuµky '%-.64s' nemá tento parameter" spa "El manejador de la tabla de '%-.64s' no tiene esta opcion" - swe "Registrets databas har inte denna facilitet" + swe "Tabellhanteraren for tabell '%-.64s' stödjer ej detta" ukr "äÅÓËÒÉÐÔÏÒ ÔÁÂÌÉæ '%-.64s' ÎÅ ÍÁ¤ 椧 ×ÌÁÓÔÉ×ÏÓÔ¦" ER_KEY_NOT_FOUND cze "Nemohu naj-Bít záznam v '%-.64s'" @@ -766,58 +766,58 @@ ER_KEY_NOT_FOUND serbian "Ne mogu da pronaðem slog u '%-.64s'" slo "Nemô¾em nájs» záznam v '%-.64s'" spa "No puedo encontrar el registro en '%-.64s'" - swe "Hittar inte posten" + swe "Hittar inte posten '%-.64s'" ukr "îÅ ÍÏÖÕ ÚÁÐÉÓÁÔÉ Õ '%-.64s'" ER_NOT_FORM_FILE - cze "Nespr-Bávná informace v souboru '%-.64s'" - dan "Forkert indhold i: '%-.64s'" - nla "Verkeerde info in file: '%-.64s'" + cze "Nespr-Bávná informace v souboru '%-.200s'" + dan "Forkert indhold i: '%-.200s'" + nla "Verkeerde info in file: '%-.200s'" eng "Incorrect information in file: '%-.200s'" - jps "ƒtƒ@ƒCƒ‹ '%-.64s' ‚Ì info ‚ªŠÔˆá‚Á‚Ä‚¢‚é‚悤‚Å‚·", - est "Vigane informatsioon failis '%-.64s'" - fre "Information erronnée dans le fichier: '%-.64s'" - ger "Falsche Information in Datei '%-.64s'" - greek "ËÜèïò ðëçñïöïñßåò óôï áñ÷åßï: '%-.64s'" - hun "Ervenytelen info a file-ban: '%-.64s'" - ita "Informazione errata nel file: '%-.64s'" - jpn "¥Õ¥¡¥¤¥ë '%-.64s' ¤Î info ¤¬´Ö°ã¤Ã¤Æ¤¤¤ë¤è¤¦¤Ç¤¹" - kor "È­ÀÏÀÇ ºÎÁ¤È®ÇÑ Á¤º¸: '%-.64s'" - nor "Feil informasjon i filen: '%-.64s'" - norwegian-ny "Feil informasjon i fila: '%-.64s'" - pol "Niew³a?ciwa informacja w pliku: '%-.64s'" - por "Informação incorreta no arquivo '%-.64s'" - rum "Informatie incorecta in fisierul: '%-.64s'" - rus "îÅËÏÒÒÅËÔÎÁÑ ÉÎÆÏÒÍÁÃÉÑ × ÆÁÊÌÅ '%-.64s'" - serbian "Pogrešna informacija u file-u: '%-.64s'" - slo "Nesprávna informácia v súbore: '%-.64s'" - spa "Informacion erronea en el archivo: '%-.64s'" - swe "Felaktig fil: '%-.64s'" - ukr "èÉÂÎÁ ¦ÎÆÏÒÍÁÃ¦Ñ Õ ÆÁÊ̦: '%-.64s'" + jps "ƒtƒ@ƒCƒ‹ '%-.200s' ‚Ì info ‚ªŠÔˆá‚Á‚Ä‚¢‚é‚悤‚Å‚·", + est "Vigane informatsioon failis '%-.200s'" + fre "Information erronnée dans le fichier: '%-.200s'" + ger "Falsche Information in Datei '%-.200s'" + greek "ËÜèïò ðëçñïöïñßåò óôï áñ÷åßï: '%-.200s'" + hun "Ervenytelen info a file-ban: '%-.200s'" + ita "Informazione errata nel file: '%-.200s'" + jpn "¥Õ¥¡¥¤¥ë '%-.200s' ¤Î info ¤¬´Ö°ã¤Ã¤Æ¤¤¤ë¤è¤¦¤Ç¤¹" + kor "È­ÀÏÀÇ ºÎÁ¤È®ÇÑ Á¤º¸: '%-.200s'" + nor "Feil informasjon i filen: '%-.200s'" + norwegian-ny "Feil informasjon i fila: '%-.200s'" + pol "Niew³a?ciwa informacja w pliku: '%-.200s'" + por "Informação incorreta no arquivo '%-.200s'" + rum "Informatie incorecta in fisierul: '%-.200s'" + rus "îÅËÏÒÒÅËÔÎÁÑ ÉÎÆÏÒÍÁÃÉÑ × ÆÁÊÌÅ '%-.200s'" + serbian "Pogrešna informacija u file-u: '%-.200s'" + slo "Nesprávna informácia v súbore: '%-.200s'" + spa "Informacion erronea en el archivo: '%-.200s'" + swe "Felaktig fil: '%-.200s'" + ukr "èÉÂÎÁ ¦ÎÆÏÒÍÁÃ¦Ñ Õ ÆÁÊ̦: '%-.200s'" ER_NOT_KEYFILE - cze "Nespr-Bávný klíè pro tabulku '%-.64s'; pokuste se ho opravit" - dan "Fejl i indeksfilen til tabellen '%-.64s'; prøv at reparere den" - nla "Verkeerde zoeksleutel file voor tabel: '%-.64s'; probeer het te repareren" + cze "Nespr-Bávný klíè pro tabulku '%-.200s'; pokuste se ho opravit" + dan "Fejl i indeksfilen til tabellen '%-.200s'; prøv at reparere den" + nla "Verkeerde zoeksleutel file voor tabel: '%-.200s'; probeer het te repareren" eng "Incorrect key file for table '%-.200s'; try to repair it" - jps "'%-.64s' ƒe[ƒuƒ‹‚Ì key file ‚ªŠÔˆá‚Á‚Ä‚¢‚é‚悤‚Å‚·. C•œ‚ð‚µ‚Ä‚­‚¾‚³‚¢", - est "Tabeli '%-.64s' võtmefail on vigane; proovi seda parandada" - fre "Index corrompu dans la table: '%-.64s'; essayez de le réparer" - ger "Fehlerhafte Index-Datei für Tabelle '%-.64s'; versuche zu reparieren" - greek "ËÜèïò áñ÷åßï ôáîéíüìéóçò (key file) ãéá ôïí ðßíáêá: '%-.64s'; Ðáñáêáëþ, äéïñèþóôå ôï!" - hun "Ervenytelen kulcsfile a tablahoz: '%-.64s'; probalja kijavitani!" - ita "File chiave errato per la tabella : '%-.64s'; prova a riparalo" - jpn "'%-.64s' ¥Æ¡¼¥Ö¥ë¤Î key file ¤¬´Ö°ã¤Ã¤Æ¤¤¤ë¤è¤¦¤Ç¤¹. ½¤Éü¤ò¤·¤Æ¤¯¤À¤µ¤¤" - kor "'%-.64s' Å×À̺íÀÇ ºÎÁ¤È®ÇÑ Å° Á¸Àç. ¼öÁ¤ÇϽÿÀ!" - nor "Tabellen '%-.64s' har feil i nøkkelfilen; forsøk å reparer den" - norwegian-ny "Tabellen '%-.64s' har feil i nykkelfila; prøv å reparere den" - pol "Niew³a?ciwy plik kluczy dla tabeli: '%-.64s'; spróbuj go naprawiæ" - por "Arquivo de índice incorreto para tabela '%-.64s'; tente repará-lo" - rum "Cheia fisierului incorecta pentru tabela: '%-.64s'; incearca s-o repari" - rus "îÅËÏÒÒÅËÔÎÙÊ ÉÎÄÅËÓÎÙÊ ÆÁÊÌ ÄÌÑ ÔÁÂÌÉÃÙ: '%-.64s'. ðÏÐÒÏÂÕÊÔÅ ×ÏÓÓÔÁÎÏ×ÉÔØ ÅÇÏ" - serbian "Pogrešan key file za tabelu: '%-.64s'; probajte da ga ispravite" - slo "Nesprávny kµúè pre tabuµku '%-.64s'; pokúste sa ho opravi»" - spa "Clave de archivo erronea para la tabla: '%-.64s'; intente repararlo" - swe "Fatalt fel vid hantering av register '%-.64s'; kör en reparation" - ukr "èÉÂÎÉÊ ÆÁÊÌ ËÌÀÞÅÊ ÄÌÑ ÔÁÂÌÉæ: '%-.64s'; óÐÒÏÂÕÊÔÅ ÊÏÇÏ ×¦ÄÎÏ×ÉÔÉ" + jps "'%-.200s' ƒe[ƒuƒ‹‚Ì key file ‚ªŠÔˆá‚Á‚Ä‚¢‚é‚悤‚Å‚·. C•œ‚ð‚µ‚Ä‚­‚¾‚³‚¢", + est "Tabeli '%-.200s' võtmefail on vigane; proovi seda parandada" + fre "Index corrompu dans la table: '%-.200s'; essayez de le réparer" + ger "Fehlerhafte Index-Datei für Tabelle '%-.200s'; versuche zu reparieren" + greek "ËÜèïò áñ÷åßï ôáîéíüìéóçò (key file) ãéá ôïí ðßíáêá: '%-.200s'; Ðáñáêáëþ, äéïñèþóôå ôï!" + hun "Ervenytelen kulcsfile a tablahoz: '%-.200s'; probalja kijavitani!" + ita "File chiave errato per la tabella : '%-.200s'; prova a riparalo" + jpn "'%-.200s' ¥Æ¡¼¥Ö¥ë¤Î key file ¤¬´Ö°ã¤Ã¤Æ¤¤¤ë¤è¤¦¤Ç¤¹. ½¤Éü¤ò¤·¤Æ¤¯¤À¤µ¤¤" + kor "'%-.200s' Å×À̺íÀÇ ºÎÁ¤È®ÇÑ Å° Á¸Àç. ¼öÁ¤ÇϽÿÀ!" + nor "Tabellen '%-.200s' har feil i nøkkelfilen; forsøk å reparer den" + norwegian-ny "Tabellen '%-.200s' har feil i nykkelfila; prøv å reparere den" + pol "Niew³a?ciwy plik kluczy dla tabeli: '%-.200s'; spróbuj go naprawiæ" + por "Arquivo de índice incorreto para tabela '%-.200s'; tente repará-lo" + rum "Cheia fisierului incorecta pentru tabela: '%-.200s'; incearca s-o repari" + rus "îÅËÏÒÒÅËÔÎÙÊ ÉÎÄÅËÓÎÙÊ ÆÁÊÌ ÄÌÑ ÔÁÂÌÉÃÙ: '%-.200s'. ðÏÐÒÏÂÕÊÔÅ ×ÏÓÓÔÁÎÏ×ÉÔØ ÅÇÏ" + serbian "Pogrešan key file za tabelu: '%-.200s'; probajte da ga ispravite" + slo "Nesprávny kµúè pre tabuµku '%-.200s'; pokúste sa ho opravi»" + spa "Clave de archivo erronea para la tabla: '%-.200s'; intente repararlo" + swe "Fatalt fel vid hantering av register '%-.200s'; kör en reparation" + ukr "èÉÂÎÉÊ ÆÁÊÌ ËÌÀÞÅÊ ÄÌÑ ÔÁÂÌÉæ: '%-.200s'; óÐÒÏÂÕÊÔÅ ÊÏÇÏ ×¦ÄÎÏ×ÉÔÉ" ER_OLD_KEYFILE cze "Star-Bý klíèový soubor pro '%-.64s'; opravte ho." dan "Gammel indeksfil for tabellen '%-.64s'; reparer den" @@ -950,7 +950,7 @@ ER_CON_COUNT_ERROR 08004 eng "Too many connections" jps "Ú‘±‚ª‘½‚·‚¬‚Ü‚·", est "Liiga palju samaaegseid ühendusi" - fre "Trop de connections" + fre "Trop de connexions" ger "Zu viele Verbindungen" greek "ÕðÜñ÷ïõí ðïëëÝò óõíäÝóåéò..." hun "Tul sok kapcsolat" @@ -1072,7 +1072,7 @@ ER_ACCESS_DENIED_ERROR 28000 jps "ƒ†[ƒU[ '%-.32s'@'%-.64s' ‚ð‹‘”Û‚µ‚Ü‚·.uUsing password: %s)", est "Ligipääs keelatud kasutajale '%-.32s'@'%-.64s' (kasutab parooli: %s)" fre "Accès refusé pour l'utilisateur: '%-.32s'@'@%-.64s' (mot de passe: %s)" - ger "Benutzer '%-.32s'@'%-.64s' hat keine Zugriffsberechtigung (verwendetes Passwort: %-.64s)" + ger "Benutzer '%-.32s'@'%-.64s' hat keine Zugriffsberechtigung (verwendetes Passwort: %s)" greek "Äåí åðéôÝñåôáé ç ðñüóâáóç óôï ÷ñÞóôç: '%-.32s'@'%-.64s' (÷ñÞóç password: %s)" hun "A(z) '%-.32s'@'%-.64s' felhasznalo szamara tiltott eleres. (Hasznalja a jelszot: %s)" ita "Accesso non consentito per l'utente: '%-.32s'@'%-.64s' (Password: %s)" @@ -1239,28 +1239,28 @@ ER_BAD_TABLE_ERROR 42S02 swe "Okänd tabell '%-.100s'" ukr "îÅצÄÏÍÁ ÔÁÂÌÉÃÑ '%-.100s'" ER_NON_UNIQ_ERROR 23000 - cze "Sloupec '%-.64s' v %s nen-Bí zcela jasný" - dan "Felt: '%-.64s' i tabel %s er ikke entydigt" - nla "Kolom: '%-.64s' in %s is niet eenduidig" + cze "Sloupec '%-.64s' v %-.64s nen-Bí zcela jasný" + dan "Felt: '%-.64s' i tabel %-.64s er ikke entydigt" + nla "Kolom: '%-.64s' in %-.64s is niet eenduidig" eng "Column '%-.64s' in %-.64s is ambiguous" est "Väli '%-.64s' %-.64s-s ei ole ühene" - fre "Champ: '%-.64s' dans %s est ambigu" + fre "Champ: '%-.64s' dans %-.64s est ambigu" ger "Feld '%-.64s' in %-.64s ist nicht eindeutig" greek "Ôï ðåäßï: '%-.64s' óå %-.64s äåí Ý÷åé êáèïñéóôåß" hun "A(z) '%-.64s' oszlop %-.64s-ben ketertelmu" ita "Colonna: '%-.64s' di %-.64s e` ambigua" jpn "Column: '%-.64s' in %-.64s is ambiguous" kor "Ä®·³: '%-.64s' in '%-.64s' ÀÌ ¸ðÈ£ÇÔ" - nor "Felt: '%-.64s' i tabell %s er ikke entydig" - norwegian-ny "Kolonne: '%-.64s' i tabell %s er ikkje eintydig" - pol "Kolumna: '%-.64s' w %s jest dwuznaczna" + nor "Felt: '%-.64s' i tabell %-.64s er ikke entydig" + norwegian-ny "Kolonne: '%-.64s' i tabell %-.64s er ikkje eintydig" + pol "Kolumna: '%-.64s' w %-.64s jest dwuznaczna" por "Coluna '%-.64s' em '%-.64s' é ambígua" rum "Coloana: '%-.64s' in %-.64s este ambigua" rus "óÔÏÌÂÅà '%-.64s' × %-.64s ÚÁÄÁÎ ÎÅÏÄÎÏÚÎÁÞÎÏ" serbian "Kolona '%-.64s' u %-.64s nije jedinstvena u kontekstu" slo "Pole: '%-.64s' v %-.64s je nejasné" - spa "La columna: '%-.64s' en %s es ambigua" - swe "Kolumn '%-.64s' i %s är inte unik" + spa "La columna: '%-.64s' en %-.64s es ambigua" + swe "Kolumn '%-.64s' i %-.64s är inte unik" ukr "óÔÏ×ÂÅÃØ '%-.64s' Õ %-.64s ×ÉÚÎÁÞÅÎÉÊ ÎÅÏÄÎÏÚÎÁÞÎÏ" ER_SERVER_SHUTDOWN 08S01 cze "Prob-Bíhá ukonèování práce serveru" @@ -1288,29 +1288,29 @@ ER_SERVER_SHUTDOWN 08S01 swe "Servern går nu ned" ukr "úÁ×ÅÒÛÕ¤ÔØÓÑ ÒÁÂÏÔÁ ÓÅÒ×ÅÒÁ" ER_BAD_FIELD_ERROR 42S22 S0022 - cze "Nezn-Bámý sloupec '%-.64s' v %s" - dan "Ukendt kolonne '%-.64s' i tabel %s" - nla "Onbekende kolom '%-.64s' in %s" + cze "Nezn-Bámý sloupec '%-.64s' v %-.64s" + dan "Ukendt kolonne '%-.64s' i tabel %-.64s" + nla "Onbekende kolom '%-.64s' in %-.64s" eng "Unknown column '%-.64s' in '%-.64s'" jps "'%-.64s' column ‚Í '%-.64s' ‚É‚Í‚ ‚è‚Ü‚¹‚ñ.", est "Tundmatu tulp '%-.64s' '%-.64s'-s" - fre "Champ '%-.64s' inconnu dans %s" + fre "Champ '%-.64s' inconnu dans %-.64s" ger "Unbekanntes Tabellenfeld '%-.64s' in %-.64s" greek "Áãíùóôï ðåäßï '%-.64s' óå '%-.64s'" hun "A(z) '%-.64s' oszlop ervenytelen '%-.64s'-ben" ita "Colonna sconosciuta '%-.64s' in '%-.64s'" jpn "'%-.64s' column ¤Ï '%-.64s' ¤Ë¤Ï¤¢¤ê¤Þ¤»¤ó." kor "Unknown Ä®·³ '%-.64s' in '%-.64s'" - nor "Ukjent kolonne '%-.64s' i tabell %s" - norwegian-ny "Ukjent felt '%-.64s' i tabell %s" - pol "Nieznana kolumna '%-.64s' w %s" + nor "Ukjent kolonne '%-.64s' i tabell %-.64s" + norwegian-ny "Ukjent felt '%-.64s' i tabell %-.64s" + pol "Nieznana kolumna '%-.64s' w %-.64s" por "Coluna '%-.64s' desconhecida em '%-.64s'" rum "Coloana invalida '%-.64s' in '%-.64s'" rus "îÅÉÚ×ÅÓÔÎÙÊ ÓÔÏÌÂÅà '%-.64s' × '%-.64s'" serbian "Nepoznata kolona '%-.64s' u '%-.64s'" slo "Neznáme pole '%-.64s' v '%-.64s'" - spa "La columna '%-.64s' en %s es desconocida" - swe "Okänd kolumn '%-.64s' i %s" + spa "La columna '%-.64s' en %-.64s es desconocida" + swe "Okänd kolumn '%-.64s' i %-.64s" ukr "îÅצÄÏÍÉÊ ÓÔÏ×ÂÅÃØ '%-.64s' Õ '%-.64s'" ER_WRONG_FIELD_WITH_GROUP 42000 S1009 cze "Pou-B¾ité '%-.64s' nebylo v group by" @@ -1404,29 +1404,29 @@ ER_WRONG_VALUE_COUNT 21S01 swe "Antalet kolumner motsvarar inte antalet värden" ukr "ë¦ÌØ˦ÓÔØ ÓÔÏ×ÂÃ¦× ÎÅ ÓЦ×ÐÁÄÁ¤ Ú Ë¦ÌØ˦ÓÔÀ ÚÎÁÞÅÎØ" ER_TOO_LONG_IDENT 42000 S1009 - cze "Jm-Béno identifikátoru '%-.64s' je pøíli¹ dlouhé" - dan "Navnet '%-.64s' er for langt" - nla "Naam voor herkenning '%-.64s' is te lang" + cze "Jm-Béno identifikátoru '%-.100s' je pøíli¹ dlouhé" + dan "Navnet '%-.100s' er for langt" + nla "Naam voor herkenning '%-.100s' is te lang" eng "Identifier name '%-.100s' is too long" jps "Identifier name '%-.100s' ‚Í’·‚·‚¬‚Ü‚·", est "Identifikaatori '%-.100s' nimi on liiga pikk" - fre "Le nom de l'identificateur '%-.64s' est trop long" + fre "Le nom de l'identificateur '%-.100s' est trop long" ger "Name des Bezeichners '%-.100s' ist zu lang" greek "Ôï identifier name '%-.100s' åßíáé ðïëý ìåãÜëï" hun "A(z) '%-.100s' azonositonev tul hosszu." ita "Il nome dell'identificatore '%-.100s' e` troppo lungo" jpn "Identifier name '%-.100s' ¤ÏŤ¹¤®¤Þ¤¹" kor "Identifier '%-.100s'´Â ³Ê¹« ±æ±º¿ä." - nor "Identifikator '%-.64s' er for lang" - norwegian-ny "Identifikator '%-.64s' er for lang" - pol "Nazwa identyfikatora '%-.64s' jest zbyt d³uga" + nor "Identifikator '%-.100s' er for lang" + norwegian-ny "Identifikator '%-.100s' er for lang" + pol "Nazwa identyfikatora '%-.100s' jest zbyt d³uga" por "Nome identificador '%-.100s' é longo demais" rum "Numele indentificatorului '%-.100s' este prea lung" rus "óÌÉÛËÏÍ ÄÌÉÎÎÙÊ ÉÄÅÎÔÉÆÉËÁÔÏÒ '%-.100s'" serbian "Ime '%-.100s' je predugaèko" slo "Meno identifikátora '%-.100s' je príli¹ dlhé" - spa "El nombre del identificador '%-.64s' es demasiado grande" - swe "Kolumnnamn '%-.64s' är för långt" + spa "El nombre del identificador '%-.100s' es demasiado grande" + swe "Kolumnnamn '%-.100s' är för långt" ukr "¶Í'Ñ ¦ÄÅÎÔÉƦËÁÔÏÒÁ '%-.100s' ÚÁÄÏ×ÇÅ" ER_DUP_FIELDNAME 42S21 S1009 cze "Zdvojen-Bé jméno sloupce '%-.64s'" @@ -1479,30 +1479,30 @@ ER_DUP_KEYNAME 42000 S1009 swe "Nyckelnamn '%-.64s' finns flera gånger" ukr "äÕÂÌÀÀÞÅ ¦Í'Ñ ËÌÀÞÁ '%-.64s'" ER_DUP_ENTRY 23000 S1009 - cze "Zvojen-Bý klíè '%-.64s' (èíslo klíèe '%-.64s')" - dan "Ens værdier '%-.64s' for indeks '%-.64s'" - nla "Dubbele ingang '%-.64s' voor zoeksleutel '%-.64s'" - eng "Duplicate entry '%-.64s' for key '%-.64s'" - jps "'%-.64s' ‚Í key '%-.64s' ‚É‚¨‚¢‚Äd•¡‚µ‚Ä‚¢‚Ü‚·", - est "Kattuv väärtus '%-.64s' võtmele '%-.64s'" - fre "Duplicata du champ '%-.64s' pour la clef '%-.64s'" - ger "Doppelter Eintrag '%-.64s' für Schlüssel '%-.64s'" - greek "ÄéðëÞ åããñáöÞ '%-.64s' ãéá ôï êëåéäß '%-.64s'" - hun "Duplikalt bejegyzes '%-.64s' a '%-.64s' kulcs szerint." - ita "Valore duplicato '%-.64s' per la chiave '%-.64s'" - jpn "'%-.64s' ¤Ï key '%-.64s' ¤Ë¤ª¤¤¤Æ½ÅÊ£¤·¤Æ¤¤¤Þ¤¹" - kor "Áߺ¹µÈ ÀÔ·Â °ª '%-.64s': key '%-.64s'" - nor "Like verdier '%-.64s' for nøkkel '%-.64s'" - norwegian-ny "Like verdiar '%-.64s' for nykkel '%-.64s'" - pol "Powtórzone wyst?pienie '%-.64s' dla klucza '%-.64s'" - por "Entrada '%-.64s' duplicada para a chave '%-.64s'" - rum "Cimpul '%-.64s' e duplicat pentru cheia '%-.64s'" - rus "äÕÂÌÉÒÕÀÝÁÑÓÑ ÚÁÐÉÓØ '%-.64s' ÐÏ ËÌÀÞÕ '%-.64s'" - serbian "Dupliran unos '%-.64s' za kljuè '%-.64s'" - slo "Opakovaný kµúè '%-.64s' (èíslo kµúèa '%-.64s')" - spa "Entrada duplicada '%-.64s' para la clave '%-.64s'" - swe "Dubbel nyckel '%-.64s' för nyckel '%-.64s'" - ukr "äÕÂÌÀÀÞÉÊ ÚÁÐÉÓ '%-.64s' ÄÌÑ ËÌÀÞÁ '%-.64s'" + cze "Zdvojen-Bý klíè '%-.64s' (èíslo klíèe %d)" + dan "Ens værdier '%-.64s' for indeks %d" + nla "Dubbele ingang '%-.64s' voor zoeksleutel %d" + eng "Duplicate entry '%-.64s' for key %d" + jps "'%-.64s' ‚Í key %d ‚É‚¨‚¢‚Äd•¡‚µ‚Ä‚¢‚Ü‚·", + est "Kattuv väärtus '%-.64s' võtmele %d" + fre "Duplicata du champ '%-.64s' pour la clef %d" + ger "Doppelter Eintrag '%-.64s' für Schlüssel %d" + greek "ÄéðëÞ åããñáöÞ '%-.64s' ãéá ôï êëåéäß %d" + hun "Duplikalt bejegyzes '%-.64s' a %d kulcs szerint." + ita "Valore duplicato '%-.64s' per la chiave %d" + jpn "'%-.64s' ¤Ï key %d ¤Ë¤ª¤¤¤Æ½ÅÊ£¤·¤Æ¤¤¤Þ¤¹" + kor "Áߺ¹µÈ ÀÔ·Â °ª '%-.64s': key %d" + nor "Like verdier '%-.64s' for nøkkel %d" + norwegian-ny "Like verdiar '%-.64s' for nykkel %d" + pol "Powtórzone wyst?pienie '%-.64s' dla klucza %d" + por "Entrada '%-.64s' duplicada para a chave %d" + rum "Cimpul '%-.64s' e duplicat pentru cheia %d" + rus "äÕÂÌÉÒÕÀÝÁÑÓÑ ÚÁÐÉÓØ '%-.64s' ÐÏ ËÌÀÞÕ %d" + serbian "Dupliran unos '%-.64s' za kljuè '%d'" + slo "Opakovaný kµúè '%-.64s' (èíslo kµúèa %d)" + spa "Entrada duplicada '%-.64s' para la clave %d" + swe "Dubbel nyckel '%-.64s' för nyckel %d" + ukr "äÕÂÌÀÀÞÉÊ ÚÁÐÉÓ '%-.64s' ÄÌÑ ËÌÀÞÁ %d" ER_WRONG_FIELD_SPEC 42000 S1009 cze "Chybn-Bá specifikace sloupce '%-.64s'" dan "Forkert kolonnespecifikaton for felt '%-.64s'" @@ -1526,30 +1526,30 @@ ER_WRONG_FIELD_SPEC 42000 S1009 spa "Especificador de columna erroneo para la columna '%-.64s'" swe "Felaktigt kolumntyp för kolumn '%-.64s'" ukr "îÅצÒÎÉÊ ÓÐÅÃÉƦËÁÔÏÒ ÓÔÏ×ÂÃÑ '%-.64s'" -ER_PARSE_ERROR 42000 - cze "%s bl-Bízko '%-.64s' na øádku %d" - dan "%s nær '%-.64s' på linje %d" - nla "%s bij '%-.64s' in regel %d" +ER_PARSE_ERROR 42000 s1009 + cze "%s bl-Bízko '%-.80s' na øádku %d" + dan "%s nær '%-.80s' på linje %d" + nla "%s bij '%-.80s' in regel %d" eng "%s near '%-.80s' at line %d" jps "%s : '%-.80s' •t‹ß : %d s–Ú", est "%s '%-.80s' ligidal real %d" - fre "%s près de '%-.64s' à la ligne %d" + fre "%s près de '%-.80s' à la ligne %d" ger "%s bei '%-.80s' in Zeile %d" greek "%s ðëçóßïí '%-.80s' óôç ãñáììÞ %d" hun "A %s a '%-.80s'-hez kozeli a %d sorban" ita "%s vicino a '%-.80s' linea %d" jpn "%s : '%-.80s' ÉÕ¶á : %d ¹ÔÌÜ" - kor "'%-.64s' ¿¡·¯ °°À¾´Ï´Ù. ('%-.80s' ¸í·É¾î ¶óÀÎ %d)" - nor "%s nær '%-.64s' på linje %d" - norwegian-ny "%s attmed '%-.64s' på line %d" - pol "%s obok '%-.64s' w linii %d" + kor "'%s' ¿¡·¯ °°À¾´Ï´Ù. ('%-.80s' ¸í·É¾î ¶óÀÎ %d)" + nor "%s nær '%-.80s' på linje %d" + norwegian-ny "%s attmed '%-.80s' på line %d" + pol "%s obok '%-.80s' w linii %d" por "%s próximo a '%-.80s' na linha %d" rum "%s linga '%-.80s' pe linia %d" rus "%s ÏËÏÌÏ '%-.80s' ÎÁ ÓÔÒÏËÅ %d" serbian "'%s' u iskazu '%-.80s' na liniji %d" slo "%s blízko '%-.80s' na riadku %d" - spa "%s cerca '%-.64s' en la linea %d" - swe "%s nära '%-.64s' på rad %d" + spa "%s cerca '%-.80s' en la linea %d" + swe "%s nära '%-.80s' på rad %d" ukr "%s ¦ÌÑ '%-.80s' × ÓÔÒÏæ %d" ER_EMPTY_QUERY 42000 cze "V-Býsledek dotazu je prázdný" @@ -1821,30 +1821,30 @@ ER_WRONG_AUTO_KEY 42000 S1009 swe "Det får finnas endast ett AUTO_INCREMENT-fält och detta måste vara en nyckel" ukr "îÅצÒÎÅ ×ÉÚÎÁÞÅÎÎÑ ÔÁÂÌÉæ; íÏÖÅ ÂÕÔÉ ÌÉÛÅ ÏÄÉÎ Á×ÔÏÍÁÔÉÞÎÉÊ ÓÔÏ×ÂÅÃØ, ÝÏ ÐÏ×ÉÎÅÎ ÂÕÔÉ ×ÉÚÎÁÞÅÎÉÊ ÑË ËÌÀÞ" ER_READY - cze "%s: p-Bøipraven na spojení" - dan "%s: klar til tilslutninger" - nla "%s: klaar voor verbindingen" + cze "%s: p-Bøipraven na spojení\nVersion: '%s' socket: '%s' port: %d"" + dan "%s: klar til tilslutninger\nVersion: '%s' socket: '%s' port: %d"" + nla "%s: klaar voor verbindingen\nVersion: '%s' socket: '%s' port: %d"" eng "%s: ready for connections.\nVersion: '%s' socket: '%s' port: %d" - jps "%s: €”õŠ®—¹", - est "%s: ootab ühendusi" - fre "%s: Prêt pour des connections" - ger "%-.64s: Bereit für Verbindungen.\nVersion: '%2' Socket: '%s' Port: %d" - greek "%s: óå áíáìïíÞ óõíäÝóåùí" - hun "%s: kapcsolatra kesz" - ita "%s: Pronto per le connessioni\n" - jpn "%s: ½àÈ÷´°Î»" - kor "%s: ¿¬°á ÁغñÁßÀÔ´Ï´Ù" - nor "%s: klar for tilkoblinger" - norwegian-ny "%s: klar for tilkoblingar" - pol "%s: gotowe do po³?czenia" - por "%s: Pronto para conexões" - rum "%s: sint gata pentru conectii" - rus "%s: çÏÔÏ× ÐÒÉÎÉÍÁÔØ ÓÏÅÄÉÎÅÎÉÑ.\n÷ÅÒÓÉÑ: '%s' ÓÏËÅÔ: '%s' ÐÏÒÔ: %d %s" - serbian "%s: Spreman za konekcije\n" - slo "%s: pripravený na spojenie" - spa "%s: preparado para conexiones" - swe "%s: klar att ta emot klienter" - ukr "%s: çÏÔÏ×ÉÊ ÄÌÑ Ú'¤ÄÎÁÎØ!" + jps "%s: €”õŠ®—¹\nVersion: '%s' socket: '%s' port: %d"", + est "%s: ootab ühendusi\nVersion: '%s' socket: '%s' port: %d"" + fre "%s: Prêt pour des connexions\nVersion: '%s' socket: '%s' port: %d"" + ger "%s: Bereit für Verbindungen.\nVersion: '%s' Socket: '%s' Port: %d" + greek "%s: óå áíáìïíÞ óõíäÝóåùí\nVersion: '%s' socket: '%s' port: %d"" + hun "%s: kapcsolatra kesz\nVersion: '%s' socket: '%s' port: %d"" + ita "%s: Pronto per le connessioni\nVersion: '%s' socket: '%s' port: %d"" + jpn "%s: ½àÈ÷´°Î»\nVersion: '%s' socket: '%s' port: %d"" + kor "%s: ¿¬°á ÁغñÁßÀÔ´Ï´Ù\nVersion: '%s' socket: '%s' port: %d"" + nor "%s: klar for tilkoblinger\nVersion: '%s' socket: '%s' port: %d"" + norwegian-ny "%s: klar for tilkoblingar\nVersion: '%s' socket: '%s' port: %d"" + pol "%s: gotowe do po³?czenia\nVersion: '%s' socket: '%s' port: %d"" + por "%s: Pronto para conexões\nVersion: '%s' socket: '%s' port: %d"" + rum "%s: sint gata pentru conectii\nVersion: '%s' socket: '%s' port: %d"" + rus "%s: çÏÔÏ× ÐÒÉÎÉÍÁÔØ ÓÏÅÄÉÎÅÎÉÑ.\n÷ÅÒÓÉÑ: '%s' ÓÏËÅÔ: '%s' ÐÏÒÔ: %d" + serbian "%s: Spreman za konekcije\nVersion: '%s' socket: '%s' port: %d"" + slo "%s: pripravený na spojenie\nVersion: '%s' socket: '%s' port: %d"" + spa "%s: preparado para conexiones\nVersion: '%s' socket: '%s' port: %d"" + swe "%s: klar att ta emot klienter\nVersion: '%s' socket: '%s' port: %d"" + ukr "%s: çÏÔÏ×ÉÊ ÄÌÑ Ú'¤ÄÎÁÎØ!\nVersion: '%s' socket: '%s' port: %d"" ER_NORMAL_SHUTDOWN cze "%s: norm-Bální ukonèení\n" dan "%s: Normal nedlukning\n" @@ -1852,7 +1852,7 @@ ER_NORMAL_SHUTDOWN eng "%s: Normal shutdown\n" est "%s: MySQL lõpetas\n" fre "%s: Arrêt normal du serveur\n" - ger "%-.64s: Normal heruntergefahren\n" + ger "%s: Normal heruntergefahren\n" greek "%s: ÖõóéïëïãéêÞ äéáäéêáóßá shutdown\n" hun "%s: Normal leallitas\n" ita "%s: Shutdown normale\n" @@ -1876,7 +1876,7 @@ ER_GOT_SIGNAL jps "%s: Got signal %d. ’†’f!\n", est "%s: sain signaali %d. Lõpetan!\n" fre "%s: Reçu le signal %d. Abandonne!\n" - ger "%-.64s: Signal %d erhalten. Abbruch!\n" + ger "%s: Signal %d erhalten. Abbruch!\n" greek "%s: ÅëÞöèç ôï ìÞíõìá %d. Ç äéáäéêáóßá åãêáôáëåßðåôáé!\n" hun "%s: %d jelzes. Megszakitva!\n" ita "%s: Ricevuto segnale %d. Interruzione!\n" @@ -1901,7 +1901,7 @@ ER_SHUTDOWN_COMPLETE jps "%s: Shutdown Š®—¹\n", est "%s: Lõpp\n" fre "%s: Arrêt du serveur terminé\n" - ger "%-.64s: Herunterfahren beendet\n" + ger "%s: Herunterfahren beendet\n" greek "%s: Ç äéáäéêáóßá Shutdown ïëïêëçñþèçêå\n" hun "%s: A leallitas kesz\n" ita "%s: Shutdown completato\n" @@ -1919,29 +1919,29 @@ ER_SHUTDOWN_COMPLETE swe "%s: Avslutning klar\n" ukr "%s: òÏÂÏÔÕ ÚÁ×ÅÒÛÅÎÏ\n" ER_FORCING_CLOSE 08S01 - cze "%s: n-Básilné uzavøení threadu %ld u¾ivatele '%-.64s'\n" - dan "%s: Forceret nedlukning af tråd: %ld bruger: '%-.64s'\n" - nla "%s: Afsluiten afgedwongen van thread %ld gebruiker: '%-.64s'\n" + cze "%s: n-Básilné uzavøení threadu %ld u¾ivatele '%-.32s'\n" + dan "%s: Forceret nedlukning af tråd: %ld bruger: '%-.32s'\n" + nla "%s: Afsluiten afgedwongen van thread %ld gebruiker: '%-.32s'\n" eng "%s: Forcing close of thread %ld user: '%-.32s'\n" - jps "%s: ƒXƒŒƒbƒh %ld ‹­§I—¹ user: '%-.64s'\n", + jps "%s: ƒXƒŒƒbƒh %ld ‹­§I—¹ user: '%-.32s'\n", est "%s: Sulgen jõuga lõime %ld kasutaja: '%-.32s'\n" - fre "%s: Arrêt forcé de la tâche (thread) %ld utilisateur: '%-.64s'\n" + fre "%s: Arrêt forcé de la tâche (thread) %ld utilisateur: '%-.32s'\n" ger "%s: Thread %ld zwangsweise beendet. Benutzer: '%-.32s'\n" - greek "%s: Ôï thread èá êëåßóåé %ld user: '%-.64s'\n" - hun "%s: A(z) %ld thread kenyszeritett zarasa. Felhasznalo: '%-.64s'\n" - ita "%s: Forzata la chiusura del thread %ld utente: '%-.64s'\n" - jpn "%s: ¥¹¥ì¥Ã¥É %ld ¶¯À©½ªÎ» user: '%-.64s'\n" - kor "%s: thread %ldÀÇ °­Á¦ Á¾·á user: '%-.64s'\n" - nor "%s: Påtvinget avslutning av tråd %ld bruker: '%-.64s'\n" - norwegian-ny "%s: Påtvinga avslutning av tråd %ld brukar: '%-.64s'\n" - pol "%s: Wymuszenie zamkniêcia w?tku %ld u¿ytkownik: '%-.64s'\n" + greek "%s: Ôï thread èá êëåßóåé %ld user: '%-.32s'\n" + hun "%s: A(z) %ld thread kenyszeritett zarasa. Felhasznalo: '%-.32s'\n" + ita "%s: Forzata la chiusura del thread %ld utente: '%-.32s'\n" + jpn "%s: ¥¹¥ì¥Ã¥É %ld ¶¯À©½ªÎ» user: '%-.32s'\n" + kor "%s: thread %ldÀÇ °­Á¦ Á¾·á user: '%-.32s'\n" + nor "%s: Påtvinget avslutning av tråd %ld bruker: '%-.32s'\n" + norwegian-ny "%s: Påtvinga avslutning av tråd %ld brukar: '%-.32s'\n" + pol "%s: Wymuszenie zamkniêcia w?tku %ld u¿ytkownik: '%-.32s'\n" por "%s: Forçando finalização da 'thread' %ld - usuário '%-.32s'\n" rum "%s: Terminare fortata a thread-ului %ld utilizatorului: '%-.32s'\n" rus "%s: ðÒÉÎÕÄÉÔÅÌØÎÏ ÚÁËÒÙ×ÁÅÍ ÐÏÔÏË %ld ÐÏÌØÚÏ×ÁÔÅÌÑ: '%-.32s'\n" serbian "%s: Usiljeno gašenje thread-a %ld koji pripada korisniku: '%-.32s'\n" - slo "%s: násilné ukonèenie vlákna %ld u¾ívateµa '%-.64s'\n" - spa "%s: Forzando a cerrar el thread %ld usuario: '%-.64s'\n" - swe "%s: Stänger av tråd %ld; användare: '%-.64s'\n" + slo "%s: násilné ukonèenie vlákna %ld u¾ívateµa '%-.32s'\n" + spa "%s: Forzando a cerrar el thread %ld usuario: '%-.32s'\n" + swe "%s: Stänger av tråd %ld; användare: '%-.32s'\n" ukr "%s: ðÒÉÓËÏÒÀÀ ÚÁËÒÉÔÔÑ Ç¦ÌËÉ %ld ËÏÒÉÓÔÕ×ÁÞÁ: '%-.32s'\n" ER_IPSOCK_ERROR 08S01 cze "Nemohu vytvo-Bøit IP socket" @@ -1950,7 +1950,7 @@ ER_IPSOCK_ERROR 08S01 eng "Can't create IP socket" jps "IP socket ‚ªì‚ê‚Ü‚¹‚ñ", est "Ei suuda luua IP socketit" - fre "Ne peut créer la connection IP (socket)" + fre "Ne peut créer la connexion IP (socket)" ger "Kann IP-Socket nicht erzeugen" greek "Äåí åßíáé äõíáôÞ ç äçìéïõñãßá IP socket" hun "Az IP socket nem hozhato letre" @@ -2041,55 +2041,55 @@ ER_BLOBS_AND_NO_TERMINATED 42000 S1009 swe "Man kan inte använda fast radlängd med blobs. Använd 'fields terminated by'" ukr "îÅ ÍÏÖÎÁ ×ÉËÏÒÉÓÔÏ×Õ×ÁÔÉ ÓÔÁÌÕ ÄÏ×ÖÉÎÕ ÓÔÒÏËÉ Ú BLOB. úËÏÒÉÓÔÁÊÔÅÓÑ 'fields terminated by'" ER_TEXTFILE_NOT_READABLE - cze "Soubor '%-.64s' mus-Bí být v adresáøi databáze nebo èitelný pro v¹echny" - dan "Filen '%-.64s' skal være i database-folderen og kunne læses af alle" - nla "Het bestand '%-.64s' dient in de database directory voor the komen of leesbaar voor iedereen te zijn." + cze "Soubor '%-.128s' mus-Bí být v adresáøi databáze nebo èitelný pro v¹echny" + dan "Filen '%-.128s' skal være i database-folderen og kunne læses af alle" + nla "Het bestand '%-.128s' dient in de database directory voor the komen of leesbaar voor iedereen te zijn." eng "The file '%-.128s' must be in the database directory or be readable by all" - jps "ƒtƒ@ƒCƒ‹ '%-.64s' ‚Í databse ‚Ì directory ‚É‚ ‚é‚©‘S‚Ẵ†[ƒU[‚ª“Ç‚ß‚é‚悤‚É‹–‰Â‚³‚ê‚Ä‚¢‚È‚¯‚ê‚΂Ȃè‚Ü‚¹‚ñ.", - est "Fail '%-.64s' peab asuma andmebaasi kataloogis või olema kõigile loetav" - fre "Le fichier '%-.64s' doit être dans le répertoire de la base et lisible par tous" - ger "Datei '%-.64s' muss im Datenbank-Verzeichnis vorhanden oder lesbar für alle sein" - greek "Ôï áñ÷åßï '%-.64s' ðñÝðåé íá õðÜñ÷åé óôï database directory Þ íá ìðïñåß íá äéáâáóôåß áðü üëïõò" - hun "A(z) '%-.64s'-nak az adatbazis konyvtarban kell lennie, vagy mindenki szamara olvashatonak" - ita "Il file '%-.64s' deve essere nella directory del database e deve essere leggibile da tutti" - jpn "¥Õ¥¡¥¤¥ë '%-.64s' ¤Ï databse ¤Î directory ¤Ë¤¢¤ë¤«Á´¤Æ¤Î¥æ¡¼¥¶¡¼¤¬Æɤá¤ë¤è¤¦¤Ëµö²Ä¤µ¤ì¤Æ¤¤¤Ê¤±¤ì¤Ð¤Ê¤ê¤Þ¤»¤ó." - kor "'%-.64s' È­ÀÏ´Â µ¥ÀÌŸº£À̽º µð·ºÅ丮¿¡ Á¸ÀçÇϰųª ¸ðµÎ¿¡°Ô Àб⠰¡´ÉÇÏ¿©¾ß ÇÕ´Ï´Ù." - nor "Filen '%-.64s' må være i database-katalogen for å være lesbar for alle" - norwegian-ny "Filen '%-.64s' må være i database-katalogen for å være lesbar for alle" - pol "Plik '%-.64s' musi znajdowaæ sie w katalogu bazy danych lub mieæ prawa czytania przez wszystkich" - por "Arquivo '%-.64s' tem que estar no diretório do banco de dados ou ter leitura possível para todos" - rum "Fisierul '%-.64s' trebuie sa fie in directorul bazei de data sau trebuie sa poata sa fie citit de catre toata lumea (verifica permisiile)" - rus "æÁÊÌ '%-.64s' ÄÏÌÖÅÎ ÎÁÈÏÄÉÔØÓÑ × ÔÏÍ ÖÅ ËÁÔÁÌÏÇÅ, ÞÔÏ É ÂÁÚÁ ÄÁÎÎÙÈ, ÉÌÉ ÂÙÔØ ÏÂÝÅÄÏÓÔÕÐÎÙÍ ÄÌÑ ÞÔÅÎÉÑ" - serbian "File '%-.64s' mora biti u direktorijumu gde su file-ovi baze i mora imati odgovarajuæa prava pristupa" - slo "Súbor '%-.64s' musí by» v adresári databázy, alebo èitateµný pre v¹etkých" - spa "El archivo '%-.64s' debe estar en el directorio de la base de datos o ser de lectura por todos" - swe "Textfilen '%.64s' måste finnas i databasbiblioteket eller vara läsbar för alla" - ukr "æÁÊÌ '%-.64s' ÐÏ×ÉÎÅÎ ÂÕÔÉ Õ ÔÅæ ÂÁÚÉ ÄÁÎÎÉÈ ÁÂÏ ÍÁÔÉ ×ÓÔÁÎÏ×ÌÅÎÅ ÐÒÁ×Ï ÎÁ ÞÉÔÁÎÎÑ ÄÌÑ ÕÓ¦È" + jps "ƒtƒ@ƒCƒ‹ '%-.128s' ‚Í databse ‚Ì directory ‚É‚ ‚é‚©‘S‚Ẵ†[ƒU[‚ª“Ç‚ß‚é‚悤‚É‹–‰Â‚³‚ê‚Ä‚¢‚È‚¯‚ê‚΂Ȃè‚Ü‚¹‚ñ.", + est "Fail '%-.128s' peab asuma andmebaasi kataloogis või olema kõigile loetav" + fre "Le fichier '%-.128s' doit être dans le répertoire de la base et lisible par tous" + ger "Datei '%-.128s' muss im Datenbank-Verzeichnis vorhanden oder lesbar für alle sein" + greek "Ôï áñ÷åßï '%-.128s' ðñÝðåé íá õðÜñ÷åé óôï database directory Þ íá ìðïñåß íá äéáâáóôåß áðü üëïõò" + hun "A(z) '%-.128s'-nak az adatbazis konyvtarban kell lennie, vagy mindenki szamara olvashatonak" + ita "Il file '%-.128s' deve essere nella directory del database e deve essere leggibile da tutti" + jpn "¥Õ¥¡¥¤¥ë '%-.128s' ¤Ï databse ¤Î directory ¤Ë¤¢¤ë¤«Á´¤Æ¤Î¥æ¡¼¥¶¡¼¤¬Æɤá¤ë¤è¤¦¤Ëµö²Ä¤µ¤ì¤Æ¤¤¤Ê¤±¤ì¤Ð¤Ê¤ê¤Þ¤»¤ó." + kor "'%-.128s' È­ÀÏ´Â µ¥ÀÌŸº£À̽º µð·ºÅ丮¿¡ Á¸ÀçÇϰųª ¸ðµÎ¿¡°Ô Àб⠰¡´ÉÇÏ¿©¾ß ÇÕ´Ï´Ù." + nor "Filen '%-.128s' må være i database-katalogen for å være lesbar for alle" + norwegian-ny "Filen '%-.128s' må være i database-katalogen for å være lesbar for alle" + pol "Plik '%-.128s' musi znajdowaæ sie w katalogu bazy danych lub mieæ prawa czytania przez wszystkich" + por "Arquivo '%-.128s' tem que estar no diretório do banco de dados ou ter leitura possível para todos" + rum "Fisierul '%-.128s' trebuie sa fie in directorul bazei de data sau trebuie sa poata sa fie citit de catre toata lumea (verifica permisiile)" + rus "æÁÊÌ '%-.128s' ÄÏÌÖÅÎ ÎÁÈÏÄÉÔØÓÑ × ÔÏÍ ÖÅ ËÁÔÁÌÏÇÅ, ÞÔÏ É ÂÁÚÁ ÄÁÎÎÙÈ, ÉÌÉ ÂÙÔØ ÏÂÝÅÄÏÓÔÕÐÎÙÍ ÄÌÑ ÞÔÅÎÉÑ" + serbian "File '%-.128s' mora biti u direktorijumu gde su file-ovi baze i mora imati odgovarajuæa prava pristupa" + slo "Súbor '%-.128s' musí by» v adresári databázy, alebo èitateµný pre v¹etkých" + spa "El archivo '%-.128s' debe estar en el directorio de la base de datos o ser de lectura por todos" + swe "Textfilen '%-.128s' måste finnas i databasbiblioteket eller vara läsbar för alla" + ukr "æÁÊÌ '%-.128s' ÐÏ×ÉÎÅÎ ÂÕÔÉ Õ ÔÅæ ÂÁÚÉ ÄÁÎÎÉÈ ÁÂÏ ÍÁÔÉ ×ÓÔÁÎÏ×ÌÅÎÅ ÐÒÁ×Ï ÎÁ ÞÉÔÁÎÎÑ ÄÌÑ ÕÓ¦È" ER_FILE_EXISTS_ERROR - cze "Soubor '%-.64s' ji-B¾ existuje" - dan "Filen '%-.64s' eksisterer allerede" - nla "Het bestand '%-.64s' bestaat reeds" + cze "Soubor '%-.200s' ji-B¾ existuje" + dan "Filen '%-.200s' eksisterer allerede" + nla "Het bestand '%-.200s' bestaat reeds" eng "File '%-.200s' already exists" - jps "File '%-.64s' ‚ÍŠù‚É‘¶Ý‚µ‚Ü‚·", - est "Fail '%-.80s' juba eksisteerib" - fre "Le fichier '%-.64s' existe déjà" - ger "Datei '%-.80s' bereits vorhanden" - greek "Ôï áñ÷åßï '%-.64s' õðÜñ÷åé Þäç" - hun "A '%-.64s' file mar letezik." - ita "Il file '%-.64s' esiste gia`" - jpn "File '%-.64s' ¤Ï´û¤Ë¸ºß¤·¤Þ¤¹" - kor "'%-.64s' È­ÀÏÀº ÀÌ¹Ì Á¸ÀçÇÕ´Ï´Ù." - nor "Filen '%-.64s' eksisterte allerede" - norwegian-ny "Filen '%-.64s' eksisterte allereide" - pol "Plik '%-.64s' ju¿ istnieje" - por "Arquivo '%-.80s' já existe" - rum "Fisierul '%-.80s' exista deja" - rus "æÁÊÌ '%-.80s' ÕÖÅ ÓÕÝÅÓÔ×ÕÅÔ" - serbian "File '%-.80s' veæ postoji" - slo "Súbor '%-.64s' u¾ existuje" - spa "El archivo '%-.64s' ya existe" - swe "Filen '%-.64s' existerar redan" - ukr "æÁÊÌ '%-.80s' ×ÖÅ ¦ÓÎÕ¤" + jps "File '%-.200s' ‚ÍŠù‚É‘¶Ý‚µ‚Ü‚·", + est "Fail '%-.200s' juba eksisteerib" + fre "Le fichier '%-.200s' existe déjà" + ger "Datei '%-.200s' bereits vorhanden" + greek "Ôï áñ÷åßï '%-.200s' õðÜñ÷åé Þäç" + hun "A '%-.200s' file mar letezik." + ita "Il file '%-.200s' esiste gia`" + jpn "File '%-.200s' ¤Ï´û¤Ë¸ºß¤·¤Þ¤¹" + kor "'%-.200s' È­ÀÏÀº ÀÌ¹Ì Á¸ÀçÇÕ´Ï´Ù." + nor "Filen '%-.200s' eksisterte allerede" + norwegian-ny "Filen '%-.200s' eksisterte allereide" + pol "Plik '%-.200s' ju¿ istnieje" + por "Arquivo '%-.200s' já existe" + rum "Fisierul '%-.200s' exista deja" + rus "æÁÊÌ '%-.200s' ÕÖÅ ÓÕÝÅÓÔ×ÕÅÔ" + serbian "File '%-.200s' veæ postoji" + slo "Súbor '%-.200s' u¾ existuje" + spa "El archivo '%-.200s' ya existe" + swe "Filen '%-.200s' existerar redan" + ukr "æÁÊÌ '%-.200s' ×ÖÅ ¦ÓÎÕ¤" ER_LOAD_INFO cze "Z-Báznamù: %ld Vymazáno: %ld Pøeskoèeno: %ld Varování: %ld" dan "Poster: %ld Fjernet: %ld Sprunget over: %ld Advarsler: %ld" @@ -2319,51 +2319,51 @@ ER_NO_TABLES_USED swe "Inga tabeller angivna" ukr "îÅ ×ÉËÏÒÉÓÔÁÎÏ ÔÁÂÌÉÃØ" ER_TOO_BIG_SET - cze "P-Bøíli¹ mnoho øetìzcù pro sloupec %s a SET" + cze "P-Bøíli¹ mnoho øetìzcù pro sloupec %-.64s a SET" dan "For mange tekststrenge til specifikationen af SET i kolonne %-.64s" - nla "Teveel strings voor kolom %s en SET" + nla "Teveel strings voor kolom %-.64s en SET" eng "Too many strings for column %-.64s and SET" est "Liiga palju string tulbale %-.64s tüübile SET" - fre "Trop de chaînes dans la colonne %s avec SET" + fre "Trop de chaînes dans la colonne %-.64s avec SET" ger "Zu viele Strings für Feld %-.64s und SET angegeben" greek "ÐÜñá ðïëëÜ strings ãéá ôï ðåäßï %-.64s êáé SET" hun "Tul sok karakter: %-.64s es SET" ita "Troppe stringhe per la colonna %-.64s e la SET" kor "Ä®·³ %-.64s¿Í SET¿¡¼­ ½ºÆ®¸µÀÌ ³Ê¹« ¸¹½À´Ï´Ù." - nor "For mange tekststrenger kolonne %s og SET" - norwegian-ny "For mange tekststrengar felt %s og SET" - pol "Zbyt wiele ³añcuchów dla kolumny %s i polecenia SET" + nor "For mange tekststrenger kolonne %-.64s og SET" + norwegian-ny "For mange tekststrengar felt %-.64s og SET" + pol "Zbyt wiele ³añcuchów dla kolumny %-.64s i polecenia SET" por "'Strings' demais para coluna '%-.64s' e SET" rum "Prea multe siruri pentru coloana %-.64s si SET" rus "óÌÉÛËÏÍ ÍÎÏÇÏ ÚÎÁÞÅÎÉÊ ÄÌÑ ÓÔÏÌÂÃÁ %-.64s × SET" serbian "Previše string-ova za kolonu '%-.64s' i komandu 'SET'" slo "Príli¹ mnoho re»azcov pre pole %-.64s a SET" - spa "Muchas strings para columna %s y SET" - swe "För många alternativ till kolumn %s för SET" + spa "Muchas strings para columna %-.64s y SET" + swe "För många alternativ till kolumn %-.64s för SET" ukr "úÁÂÁÇÁÔÏ ÓÔÒÏË ÄÌÑ ÓÔÏ×ÂÃÑ %-.64s ÔÁ SET" ER_NO_UNIQUE_LOGFILE - cze "Nemohu vytvo-Bøit jednoznaèné jméno logovacího souboru %s.(1-999)\n" - dan "Kan ikke lave unikt log-filnavn %s.(1-999)\n" - nla "Het is niet mogelijk een unieke naam te maken voor de logfile %s.(1-999)\n" + cze "Nemohu vytvo-Bøit jednoznaèné jméno logovacího souboru %-.200s.(1-999)\n" + dan "Kan ikke lave unikt log-filnavn %-.200s.(1-999)\n" + nla "Het is niet mogelijk een unieke naam te maken voor de logfile %-.200s.(1-999)\n" eng "Can't generate a unique log-filename %-.200s.(1-999)\n" - est "Ei suuda luua unikaalset logifaili nime %-.64s.(1-999)\n" - fre "Ne peut générer un unique nom de journal %s.(1-999)\n" - ger "Kann keinen eindeutigen Dateinamen für die Logdatei %-.64s(1-999) erzeugen\n" - greek "Áäýíáôç ç äçìéïõñãßá unique log-filename %-.64s.(1-999)\n" - hun "Egyedi log-filenev nem generalhato: %-.64s.(1-999)\n" - ita "Impossibile generare un nome del file log unico %-.64s.(1-999)\n" - kor "Unique ·Î±×È­ÀÏ '%-.64s'¸¦ ¸¸µé¼ö ¾ø½À´Ï´Ù.(1-999)\n" - nor "Kan ikke lage unikt loggfilnavn %s.(1-999)\n" - norwegian-ny "Kan ikkje lage unikt loggfilnavn %s.(1-999)\n" - pol "Nie mo¿na stworzyæ unikalnej nazwy pliku z logiem %s.(1-999)\n" - por "Não pode gerar um nome de arquivo de 'log' único '%-.64s'.(1-999)\n" - rum "Nu pot sa generez un nume de log unic %-.64s.(1-999)\n" - rus "îÅ×ÏÚÍÏÖÎÏ ÓÏÚÄÁÔØ ÕÎÉËÁÌØÎÏÅ ÉÍÑ ÆÁÊÌÁ ÖÕÒÎÁÌÁ %-.64s.(1-999)\n" - serbian "Ne mogu da generišem jedinstveno ime log-file-a: '%-.64s.(1-999)'\n" - slo "Nemô¾em vytvori» unikátne meno log-súboru %-.64s.(1-999)\n" - spa "No puede crear un unico archivo log %s.(1-999)\n" - swe "Kan inte generera ett unikt filnamn %s.(1-999)\n" - ukr "îÅ ÍÏÖÕ ÚÇÅÎÅÒÕ×ÁÔÉ ÕΦËÁÌØÎÅ ¦Í'Ñ log-ÆÁÊÌÕ %-.64s.(1-999)\n" + est "Ei suuda luua unikaalset logifaili nime %-.200s.(1-999)\n" + fre "Ne peut générer un unique nom de journal %-.200s.(1-999)\n" + ger "Kann keinen eindeutigen Dateinamen für die Logdatei %-.200s(1-999) erzeugen\n" + greek "Áäýíáôç ç äçìéïõñãßá unique log-filename %-.200s.(1-999)\n" + hun "Egyedi log-filenev nem generalhato: %-.200s.(1-999)\n" + ita "Impossibile generare un nome del file log unico %-.200s.(1-999)\n" + kor "Unique ·Î±×È­ÀÏ '%-.200s'¸¦ ¸¸µé¼ö ¾ø½À´Ï´Ù.(1-999)\n" + nor "Kan ikke lage unikt loggfilnavn %-.200s.(1-999)\n" + norwegian-ny "Kan ikkje lage unikt loggfilnavn %-.200s.(1-999)\n" + pol "Nie mo¿na stworzyæ unikalnej nazwy pliku z logiem %-.200s.(1-999)\n" + por "Não pode gerar um nome de arquivo de 'log' único '%-.200s'.(1-999)\n" + rum "Nu pot sa generez un nume de log unic %-.200s.(1-999)\n" + rus "îÅ×ÏÚÍÏÖÎÏ ÓÏÚÄÁÔØ ÕÎÉËÁÌØÎÏÅ ÉÍÑ ÆÁÊÌÁ ÖÕÒÎÁÌÁ %-.200s.(1-999)\n" + serbian "Ne mogu da generišem jedinstveno ime log-file-a: '%-.200s.(1-999)'\n" + slo "Nemô¾em vytvori» unikátne meno log-súboru %-.200s.(1-999)\n" + spa "No puede crear un unico archivo log %-.200s.(1-999)\n" + swe "Kan inte generera ett unikt filnamn %-.200s.(1-999)\n" + ukr "îÅ ÍÏÖÕ ÚÇÅÎÅÒÕ×ÁÔÉ ÕΦËÁÌØÎÅ ¦Í'Ñ log-ÆÁÊÌÕ %-.200s.(1-999)\n" ER_TABLE_NOT_LOCKED_FOR_WRITE cze "Tabulka '%-.64s' byla zam-Bèena s READ a nemù¾e být zmìnìna" dan "Tabellen '%-.64s' var låst med READ lås og kan ikke opdateres" @@ -2439,54 +2439,54 @@ ER_BLOB_CANT_HAVE_DEFAULT 42000 swe "BLOB fält '%-.64s' kan inte ha ett DEFAULT-värde" ukr "óÔÏ×ÂÅÃØ BLOB '%-.64s' ÎÅ ÍÏÖÅ ÍÁÔÉ ÚÎÁÞÅÎÎÑ ÐÏ ÚÁÍÏ×ÞÕ×ÁÎÎÀ" ER_WRONG_DB_NAME 42000 - cze "Nep-Bøípustné jméno databáze '%-.64s'" - dan "Ugyldigt database navn '%-.64s'" - nla "Databasenaam '%-.64s' is niet getoegestaan" + cze "Nep-Bøípustné jméno databáze '%-.100s'" + dan "Ugyldigt database navn '%-.100s'" + nla "Databasenaam '%-.100s' is niet getoegestaan" eng "Incorrect database name '%-.100s'" jps "Žw’肵‚½ database –¼ '%-.100s' ‚ªŠÔˆá‚Á‚Ä‚¢‚Ü‚·", est "Vigane andmebaasi nimi '%-.100s'" - fre "Nom de base de donnée illégal: '%-.64s'" + fre "Nom de base de donnée illégal: '%-.100s'" ger "Unerlaubter Datenbankname '%-.100s'" greek "ËÜèïò üíïìá âÜóçò äåäïìÝíùí '%-.100s'" hun "Hibas adatbazisnev: '%-.100s'" ita "Nome database errato '%-.100s'" jpn "»ØÄꤷ¤¿ database ̾ '%-.100s' ¤¬´Ö°ã¤Ã¤Æ¤¤¤Þ¤¹" kor "'%-.100s' µ¥ÀÌŸº£À̽ºÀÇ À̸§ÀÌ ºÎÁ¤È®ÇÕ´Ï´Ù." - nor "Ugyldig database navn '%-.64s'" - norwegian-ny "Ugyldig database namn '%-.64s'" - pol "Niedozwolona nazwa bazy danych '%-.64s'" + nor "Ugyldig database navn '%-.100s'" + norwegian-ny "Ugyldig database namn '%-.100s'" + pol "Niedozwolona nazwa bazy danych '%-.100s'" por "Nome de banco de dados '%-.100s' incorreto" rum "Numele bazei de date este incorect '%-.100s'" rus "îÅËÏÒÒÅËÔÎÏÅ ÉÍÑ ÂÁÚÙ ÄÁÎÎÙÈ '%-.100s'" serbian "Pogrešno ime baze '%-.100s'" slo "Neprípustné meno databázy '%-.100s'" - spa "Nombre de base de datos ilegal '%-.64s'" - swe "Felaktigt databasnamn '%-.64s'" + spa "Nombre de base de datos ilegal '%-.100s'" + swe "Felaktigt databasnamn '%-.100s'" ukr "îÅצÒÎÅ ¦Í'Ñ ÂÁÚÉ ÄÁÎÎÉÈ '%-.100s'" ER_WRONG_TABLE_NAME 42000 - cze "Nep-Bøípustné jméno tabulky '%-.64s'" - dan "Ugyldigt tabel navn '%-.64s'" - nla "Niet toegestane tabelnaam '%-.64s'" + cze "Nep-Bøípustné jméno tabulky '%-.100s'" + dan "Ugyldigt tabel navn '%-.100s'" + nla "Niet toegestane tabelnaam '%-.100s'" eng "Incorrect table name '%-.100s'" jps "Žw’肵‚½ table –¼ '%-.100s' ‚Í‚Ü‚¿‚ª‚Á‚Ä‚¢‚Ü‚·", est "Vigane tabeli nimi '%-.100s'" - fre "Nom de table illégal: '%-.64s'" + fre "Nom de table illégal: '%-.100s'" ger "Unerlaubter Tabellenname '%-.100s'" greek "ËÜèïò üíïìá ðßíáêá '%-.100s'" hun "Hibas tablanev: '%-.100s'" ita "Nome tabella errato '%-.100s'" jpn "»ØÄꤷ¤¿ table ̾ '%-.100s' ¤Ï¤Þ¤Á¤¬¤Ã¤Æ¤¤¤Þ¤¹" kor "'%-.100s' Å×À̺í À̸§ÀÌ ºÎÁ¤È®ÇÕ´Ï´Ù." - nor "Ugyldig tabell navn '%-.64s'" - norwegian-ny "Ugyldig tabell namn '%-.64s'" - pol "Niedozwolona nazwa tabeli '%-.64s'..." + nor "Ugyldig tabell navn '%-.100s'" + norwegian-ny "Ugyldig tabell namn '%-.100s'" + pol "Niedozwolona nazwa tabeli '%-.100s'..." por "Nome de tabela '%-.100s' incorreto" rum "Numele tabelei este incorect '%-.100s'" rus "îÅËÏÒÒÅËÔÎÏÅ ÉÍÑ ÔÁÂÌÉÃÙ '%-.100s'" serbian "Pogrešno ime tabele '%-.100s'" slo "Neprípustné meno tabuµky '%-.100s'" - spa "Nombre de tabla ilegal '%-.64s'" - swe "Felaktigt tabellnamn '%-.64s'" + spa "Nombre de tabla ilegal '%-.100s'" + swe "Felaktigt tabellnamn '%-.100s'" ukr "îÅצÒÎÅ ¦Í'Ñ ÔÁÂÌÉæ '%-.100s'" ER_TOO_BIG_SELECT 42000 cze "Zadan-Bý SELECT by procházel pøíli¹ mnoho záznamù a trval velmi dlouho. Zkontrolujte tvar WHERE a je-li SELECT v poøádku, pou¾ijte SET SQL_BIG_SELECTS=1" @@ -2534,97 +2534,97 @@ ER_UNKNOWN_ERROR swe "Oidentifierat fel" ukr "îÅצÄÏÍÁ ÐÏÍÉÌËÁ" ER_UNKNOWN_PROCEDURE 42000 - cze "Nezn-Bámá procedura %s" - dan "Ukendt procedure %s" - nla "Onbekende procedure %s" + cze "Nezn-Bámá procedura %-.64s" + dan "Ukendt procedure %-.64s" + nla "Onbekende procedure %-.64s" eng "Unknown procedure '%-.64s'" est "Tundmatu protseduur '%-.64s'" - fre "Procédure %s inconnue" + fre "Procédure %-.64s inconnue" ger "Unbekannte Prozedur '%-.64s'" greek "Áãíùóôç äéáäéêáóßá '%-.64s'" hun "Ismeretlen eljaras: '%-.64s'" ita "Procedura '%-.64s' sconosciuta" kor "¾Ë¼ö ¾ø´Â ¼öÇ๮ : '%-.64s'" - nor "Ukjent prosedyre %s" - norwegian-ny "Ukjend prosedyre %s" - pol "Unkown procedure %s" + nor "Ukjent prosedyre %-.64s" + norwegian-ny "Ukjend prosedyre %-.64s" + pol "Unkown procedure %-.64s" por "'Procedure' '%-.64s' desconhecida" rum "Procedura unknown '%-.64s'" rus "îÅÉÚ×ÅÓÔÎÁÑ ÐÒÏÃÅÄÕÒÁ '%-.64s'" serbian "Nepoznata procedura '%-.64s'" slo "Neznámá procedúra '%-.64s'" - spa "Procedimiento desconocido %s" - swe "Okänd procedur: %s" + spa "Procedimiento desconocido %-.64s" + swe "Okänd procedur: %-.64s" ukr "îÅצÄÏÍÁ ÐÒÏÃÅÄÕÒÁ '%-.64s'" ER_WRONG_PARAMCOUNT_TO_PROCEDURE 42000 - cze "Chybn-Bý poèet parametrù procedury %s" - dan "Forkert antal parametre til proceduren %s" - nla "Foutief aantal parameters doorgegeven aan procedure %s" + cze "Chybn-Bý poèet parametrù procedury %-.64s" + dan "Forkert antal parametre til proceduren %-.64s" + nla "Foutief aantal parameters doorgegeven aan procedure %-.64s" eng "Incorrect parameter count to procedure '%-.64s'" est "Vale parameetrite hulk protseduurile '%-.64s'" - fre "Mauvais nombre de paramètres pour la procedure %s" + fre "Mauvais nombre de paramètres pour la procedure %-.64s" ger "Falsche Parameterzahl für Prozedur '%-.64s'" greek "ËÜèïò áñéèìüò ðáñáìÝôñùí óôç äéáäéêáóßá '%-.64s'" hun "Rossz parameter a(z) '%-.64s'eljaras szamitasanal" ita "Numero di parametri errato per la procedura '%-.64s'" kor "'%-.64s' ¼öÇ๮¿¡ ´ëÇÑ ºÎÁ¤È®ÇÑ ÆĶó¸ÞÅÍ" - nor "Feil parameter antall til prosedyren %s" - norwegian-ny "Feil parameter tal til prosedyra %s" - pol "Incorrect parameter count to procedure %s" + nor "Feil parameter antall til prosedyren %-.64s" + norwegian-ny "Feil parameter tal til prosedyra %-.64s" + pol "Incorrect parameter count to procedure %-.64s" por "Número de parâmetros incorreto para a 'procedure' '%-.64s'" rum "Procedura '%-.64s' are un numar incorect de parametri" rus "îÅËÏÒÒÅËÔÎÏÅ ËÏÌÉÞÅÓÔ×Ï ÐÁÒÁÍÅÔÒÏ× ÄÌÑ ÐÒÏÃÅÄÕÒÙ '%-.64s'" serbian "Pogrešan broj parametara za proceduru '%-.64s'" slo "Chybný poèet parametrov procedúry '%-.64s'" - spa "Equivocado parametro count para procedimiento %s" - swe "Felaktigt antal parametrar till procedur %s" + spa "Equivocado parametro count para procedimiento %-.64s" + swe "Felaktigt antal parametrar till procedur %-.64s" ukr "èÉÂÎÁ ˦ÌØ˦ÓÔØ ÐÁÒÁÍÅÔÒ¦× ÐÒÏÃÅÄÕÒÉ '%-.64s'" ER_WRONG_PARAMETERS_TO_PROCEDURE - cze "Chybn-Bé parametry procedury %s" - dan "Forkert(e) parametre til proceduren %s" - nla "Foutieve parameters voor procedure %s" + cze "Chybn-Bé parametry procedury %-.64s" + dan "Forkert(e) parametre til proceduren %-.64s" + nla "Foutieve parameters voor procedure %-.64s" eng "Incorrect parameters to procedure '%-.64s'" est "Vigased parameetrid protseduurile '%-.64s'" - fre "Paramètre erroné pour la procedure %s" + fre "Paramètre erroné pour la procedure %-.64s" ger "Falsche Parameter für Prozedur '%-.64s'" greek "ËÜèïò ðáñÜìåôñïé óôçí äéáäéêáóßá '%-.64s'" hun "Rossz parameter a(z) '%-.64s' eljarasban" ita "Parametri errati per la procedura '%-.64s'" kor "'%-.64s' ¼öÇ๮¿¡ ´ëÇÑ ºÎÁ¤È®ÇÑ ÆĶó¸ÞÅÍ" - nor "Feil parametre til prosedyren %s" - norwegian-ny "Feil parameter til prosedyra %s" - pol "Incorrect parameters to procedure %s" + nor "Feil parametre til prosedyren %-.64s" + norwegian-ny "Feil parameter til prosedyra %-.64s" + pol "Incorrect parameters to procedure %-.64s" por "Parâmetros incorretos para a 'procedure' '%-.64s'" rum "Procedura '%-.64s' are parametrii incorecti" rus "îÅËÏÒÒÅËÔÎÙÅ ÐÁÒÁÍÅÔÒÙ ÄÌÑ ÐÒÏÃÅÄÕÒÙ '%-.64s'" serbian "Pogrešni parametri prosleðeni proceduri '%-.64s'" slo "Chybné parametre procedúry '%-.64s'" - spa "Equivocados parametros para procedimiento %s" - swe "Felaktiga parametrar till procedur %s" + spa "Equivocados parametros para procedimiento %-.64s" + swe "Felaktiga parametrar till procedur %-.64s" ukr "èÉÂÎÉÊ ÐÁÒÁÍÅÔÅÒ ÐÒÏÃÅÄÕÒÉ '%-.64s'" ER_UNKNOWN_TABLE 42S02 - cze "Nezn-Bámá tabulka '%-.64s' v %s" - dan "Ukendt tabel '%-.64s' i %s" - nla "Onbekende tabel '%-.64s' in %s" + cze "Nezn-Bámá tabulka '%-.64s' v %-.32s" + dan "Ukendt tabel '%-.64s' i %-.32s" + nla "Onbekende tabel '%-.64s' in %-.32s" eng "Unknown table '%-.64s' in %-.32s" est "Tundmatu tabel '%-.64s' %-.32s-s" - fre "Table inconnue '%-.64s' dans %s" - ger "Unbekannte Tabelle '%-.64s' in '%-.64s'" - greek "Áãíùóôïò ðßíáêáò '%-.64s' óå %s" - hun "Ismeretlen tabla: '%-.64s' %s-ban" - ita "Tabella '%-.64s' sconosciuta in %s" - jpn "Unknown table '%-.64s' in %s" - kor "¾Ë¼ö ¾ø´Â Å×À̺í '%-.64s' (µ¥ÀÌŸº£À̽º %s)" - nor "Ukjent tabell '%-.64s' i %s" - norwegian-ny "Ukjend tabell '%-.64s' i %s" - pol "Unknown table '%-.64s' in %s" + fre "Table inconnue '%-.64s' dans %-.32s" + ger "Unbekannte Tabelle '%-.64s' in '%-.32s'" + greek "Áãíùóôïò ðßíáêáò '%-.64s' óå %-.32s" + hun "Ismeretlen tabla: '%-.64s' %-.32s-ban" + ita "Tabella '%-.64s' sconosciuta in %-.32s" + jpn "Unknown table '%-.64s' in %-.32s" + kor "¾Ë¼ö ¾ø´Â Å×À̺í '%-.64s' (µ¥ÀÌŸº£À̽º %-.32s)" + nor "Ukjent tabell '%-.64s' i %-.32s" + norwegian-ny "Ukjend tabell '%-.64s' i %-.32s" + pol "Unknown table '%-.64s' in %-.32s" por "Tabela '%-.64s' desconhecida em '%-.32s'" rum "Tabla '%-.64s' invalida in %-.32s" rus "îÅÉÚ×ÅÓÔÎÁÑ ÔÁÂÌÉÃÁ '%-.64s' × %-.32s" serbian "Nepoznata tabela '%-.64s' u '%-.32s'" - slo "Neznáma tabuµka '%-.64s' v %s" - spa "Tabla desconocida '%-.64s' in %s" - swe "Okänd tabell '%-.64s' i '%-.64s'" + slo "Neznáma tabuµka '%-.64s' v %-.32s" + spa "Tabla desconocida '%-.64s' in %-.32s" + swe "Okänd tabell '%-.64s' i '%-.32s'" ukr "îÅצÄÏÍÁ ÔÁÂÌÉÃÑ '%-.64s' Õ %-.32s" ER_FIELD_SPECIFIED_TWICE 42000 cze "Polo-B¾ka '%-.64s' je zadána dvakrát" @@ -2803,27 +2803,27 @@ ER_TOO_MANY_FIELDS swe "För många fält" ukr "úÁÂÁÇÁÔÏ ÓÔÏ×Âæ×" ER_TOO_BIG_ROWSIZE 42000 - cze "-BØádek je pøíli¹ velký. Maximální velikost øádku, nepoèítaje polo¾ky blob, je %d. Musíte zmìnit nìkteré polo¾ky na blob" - dan "For store poster. Max post størrelse, uden BLOB's, er %d. Du må lave nogle felter til BLOB's" - nla "Rij-grootte is groter dan toegestaan. Maximale rij grootte, blobs niet meegeteld, is %d. U dient sommige velden in blobs te veranderen." + cze "-BØádek je pøíli¹ velký. Maximální velikost øádku, nepoèítaje polo¾ky blob, je %ld. Musíte zmìnit nìkteré polo¾ky na blob" + dan "For store poster. Max post størrelse, uden BLOB's, er %ld. Du må lave nogle felter til BLOB's" + nla "Rij-grootte is groter dan toegestaan. Maximale rij grootte, blobs niet meegeteld, is %ld. U dient sommige velden in blobs te veranderen." eng "Row size too large. The maximum row size for the used table type, not counting BLOBs, is %ld. You have to change some columns to TEXT or BLOBs" - jps "row size ‚ª‘å‚«‚·‚¬‚Ü‚·. BLOB ‚ðŠÜ‚Ü‚È‚¢ê‡‚Ì row size ‚ÌÅ‘å‚Í %d ‚Å‚·. ‚¢‚­‚‚©‚Ì field ‚ð BLOB ‚É•Ï‚¦‚Ä‚­‚¾‚³‚¢.", - est "Liiga pikk kirje. Kirje maksimumpikkus arvestamata BLOB-tüüpi välju on %d. Muuda mõned väljad BLOB-tüüpi väljadeks" - fre "Ligne trop grande. Le taille maximale d'une ligne, sauf les BLOBs, est %d. Changez le type de quelques colonnes en BLOB" + jps "row size ‚ª‘å‚«‚·‚¬‚Ü‚·. BLOB ‚ðŠÜ‚Ü‚È‚¢ê‡‚Ì row size ‚ÌÅ‘å‚Í %ld ‚Å‚·. ‚¢‚­‚‚©‚Ì field ‚ð BLOB ‚É•Ï‚¦‚Ä‚­‚¾‚³‚¢.", + est "Liiga pikk kirje. Kirje maksimumpikkus arvestamata BLOB-tüüpi välju on %ld. Muuda mõned väljad BLOB-tüüpi väljadeks" + fre "Ligne trop grande. Le taille maximale d'une ligne, sauf les BLOBs, est %ld. Changez le type de quelques colonnes en BLOB" ger "Zeilenlänge zu groß. Die maximale Zeilenlänge für den verwendeten Tabellentyp (ohne BLOB-Felder) beträgt %ld. Einige Felder müssen in BLOB oder TEXT umgewandelt werden" - greek "Ðïëý ìåãÜëï ìÝãåèïò åããñáöÞò. Ôï ìÝãéóôï ìÝãåèïò åããñáöÞò, ÷ùñßò íá õðïëïãßæïíôáé ôá blobs, åßíáé %d. ÐñÝðåé íá ïñßóåôå êÜðïéá ðåäßá óáí blobs" - hun "Tul nagy sormeret. A maximalis sormeret (nem szamolva a blob objektumokat) %d. Nehany mezot meg kell valtoztatnia" - ita "Riga troppo grande. La massima grandezza di una riga, non contando i BLOB, e` %d. Devi cambiare alcuni campi in BLOB" - jpn "row size ¤¬Â礭¤¹¤®¤Þ¤¹. BLOB ¤ò´Þ¤Þ¤Ê¤¤¾ì¹ç¤Î row size ¤ÎºÇÂç¤Ï %d ¤Ç¤¹. ¤¤¤¯¤Ä¤«¤Î field ¤ò BLOB ¤ËÊѤ¨¤Æ¤¯¤À¤µ¤¤." - kor "³Ê¹« Å« row »çÀÌÁîÀÔ´Ï´Ù. BLOB¸¦ °è»êÇÏÁö ¾Ê°í ÃÖ´ë row »çÀÌÁî´Â %dÀÔ´Ï´Ù. ¾ó¸¶°£ÀÇ ÇʵåµéÀ» BLOB·Î ¹Ù²Ù¼Å¾ß °Ú±º¿ä.." - por "Tamanho de linha grande demais. O máximo tamanho de linha, não contando BLOBs, é %d. Você tem que mudar alguns campos para BLOBs" - rum "Marimea liniei (row) prea mare. Marimea maxima a liniei, excluzind BLOB-urile este de %d. Trebuie sa schimbati unele cimpuri in BLOB-uri" - rus "óÌÉÛËÏÍ ÂÏÌØÛÏÊ ÒÁÚÍÅÒ ÚÁÐÉÓÉ. íÁËÓÉÍÁÌØÎÙÊ ÒÁÚÍÅÒ ÓÔÒÏËÉ, ÉÓËÌÀÞÁÑ ÐÏÌÑ BLOB, - %d. ÷ÏÚÍÏÖÎÏ, ×ÁÍ ÓÌÅÄÕÅÔ ÉÚÍÅÎÉÔØ ÔÉÐ ÎÅËÏÔÏÒÙÈ ÐÏÌÅÊ ÎÁ BLOB" - serbian "Prevelik slog. Maksimalna velièina sloga, ne raèunajuæi BLOB polja, je %d. Trebali bi da promenite tip nekih polja u BLOB" - slo "Riadok je príli¹ veµký. Maximálna veµkos» riadku, okrem 'BLOB', je %d. Musíte zmeni» niektoré polo¾ky na BLOB" - spa "Tamaño de línea muy grande. Máximo tamaño de línea, no contando blob, es %d. Tu tienes que cambiar algunos campos para blob" - swe "För stor total radlängd. Den högst tillåtna radlängden, förutom BLOBs, är %d. Ändra några av dina fält till BLOB" - ukr "úÁÄÏ×ÇÁ ÓÔÒÏËÁ. îÁʦÌØÛÏÀ ÄÏ×ÖÉÎÏÀ ÓÔÒÏËÉ, ÎÅ ÒÁÈÕÀÞÉ BLOB, ¤ %d. ÷ÁÍ ÐÏÔÒ¦ÂÎÏ ÐÒÉ×ÅÓÔÉ ÄÅÑ˦ ÓÔÏ×Âæ ÄÏ ÔÉÐÕ BLOB" + greek "Ðïëý ìåãÜëï ìÝãåèïò åããñáöÞò. Ôï ìÝãéóôï ìÝãåèïò åããñáöÞò, ÷ùñßò íá õðïëïãßæïíôáé ôá blobs, åßíáé %ld. ÐñÝðåé íá ïñßóåôå êÜðïéá ðåäßá óáí blobs" + hun "Tul nagy sormeret. A maximalis sormeret (nem szamolva a blob objektumokat) %ld. Nehany mezot meg kell valtoztatnia" + ita "Riga troppo grande. La massima grandezza di una riga, non contando i BLOB, e` %ld. Devi cambiare alcuni campi in BLOB" + jpn "row size ¤¬Â礭¤¹¤®¤Þ¤¹. BLOB ¤ò´Þ¤Þ¤Ê¤¤¾ì¹ç¤Î row size ¤ÎºÇÂç¤Ï %ld ¤Ç¤¹. ¤¤¤¯¤Ä¤«¤Î field ¤ò BLOB ¤ËÊѤ¨¤Æ¤¯¤À¤µ¤¤." + kor "³Ê¹« Å« row »çÀÌÁîÀÔ´Ï´Ù. BLOB¸¦ °è»êÇÏÁö ¾Ê°í ÃÖ´ë row »çÀÌÁî´Â %ldÀÔ´Ï´Ù. ¾ó¸¶°£ÀÇ ÇʵåµéÀ» BLOB·Î ¹Ù²Ù¼Å¾ß °Ú±º¿ä.." + por "Tamanho de linha grande demais. O máximo tamanho de linha, não contando BLOBs, é %ld. Você tem que mudar alguns campos para BLOBs" + rum "Marimea liniei (row) prea mare. Marimea maxima a liniei, excluzind BLOB-urile este de %ld. Trebuie sa schimbati unele cimpuri in BLOB-uri" + rus "óÌÉÛËÏÍ ÂÏÌØÛÏÊ ÒÁÚÍÅÒ ÚÁÐÉÓÉ. íÁËÓÉÍÁÌØÎÙÊ ÒÁÚÍÅÒ ÓÔÒÏËÉ, ÉÓËÌÀÞÁÑ ÐÏÌÑ BLOB, - %ld. ÷ÏÚÍÏÖÎÏ, ×ÁÍ ÓÌÅÄÕÅÔ ÉÚÍÅÎÉÔØ ÔÉÐ ÎÅËÏÔÏÒÙÈ ÐÏÌÅÊ ÎÁ BLOB" + serbian "Prevelik slog. Maksimalna velièina sloga, ne raèunajuæi BLOB polja, je %ld. Trebali bi da promenite tip nekih polja u BLOB" + slo "Riadok je príli¹ veµký. Maximálna veµkos» riadku, okrem 'BLOB', je %ld. Musíte zmeni» niektoré polo¾ky na BLOB" + spa "Tamaño de línea muy grande. Máximo tamaño de línea, no contando blob, es %ld. Tu tienes que cambiar algunos campos para blob" + swe "För stor total radlängd. Den högst tillåtna radlängden, förutom BLOBs, är %ld. Ändra några av dina fält till BLOB" + ukr "úÁÄÏ×ÇÁ ÓÔÒÏËÁ. îÁʦÌØÛÏÀ ÄÏ×ÖÉÎÏÀ ÓÔÒÏËÉ, ÎÅ ÒÁÈÕÀÞÉ BLOB, ¤ %ld. ÷ÁÍ ÐÏÔÒ¦ÂÎÏ ÐÒÉ×ÅÓÔÉ ÄÅÑ˦ ÓÔÏ×Âæ ÄÏ ÔÉÐÕ BLOB" ER_STACK_OVERRUN cze "P-Bøeteèení zásobníku threadu: pou¾ito %ld z %ld. Pou¾ijte 'mysqld -O thread_stack=#' k zadání vìt¹ího zásobníku" dan "Thread stack brugt: Brugt: %ld af en %ld stak. Brug 'mysqld -O thread_stack=#' for at allokere en større stak om nødvendigt" @@ -3026,14 +3026,14 @@ ER_FUNCTION_NOT_DEFINED ukr "æÕÎËæÀ '%-.64s' ÎÅ ×ÉÚÎÁÞÅÎÏ" ER_HOST_IS_BLOCKED cze "Stroj '%-.64s' je zablokov-Bán kvùli mnoha chybám pøi pøipojování. Odblokujete pou¾itím 'mysqladmin flush-hosts'" - dan "Værten er blokeret på grund af mange fejlforespørgsler. Lås op med 'mysqladmin flush-hosts'" + dan "Værten '%-.64s' er blokeret på grund af mange fejlforespørgsler. Lås op med 'mysqladmin flush-hosts'" nla "Host '%-.64s' is geblokkeeerd vanwege te veel verbindings fouten. Deblokkeer met 'mysqladmin flush-hosts'" eng "Host '%-.64s' is blocked because of many connection errors; unblock with 'mysqladmin flush-hosts'" jps "Host '%-.64s' ‚Í many connection error ‚Ì‚½‚ßA‹‘”Û‚³‚ê‚Ü‚µ‚½. 'mysqladmin flush-hosts' ‚ʼn𜂵‚Ä‚­‚¾‚³‚¢", est "Masin '%-.64s' on blokeeritud hulgaliste ühendusvigade tõttu. Blokeeringu saab tühistada 'mysqladmin flush-hosts' käsuga" - fre "L'hôte '%-.64s' est bloqué à cause d'un trop grand nombre d'erreur de connection. Débloquer le par 'mysqladmin flush-hosts'" + fre "L'hôte '%-.64s' est bloqué à cause d'un trop grand nombre d'erreur de connexion. Débloquer le par 'mysqladmin flush-hosts'" ger "Host '%-.64s' blockiert wegen zu vieler Verbindungsfehler. Aufheben der Blockierung mit 'mysqladmin flush-hosts'" - greek "Ï õðïëïãéóôÞò Ý÷åé áðïêëåéóèåß ëüãù ðïëëáðëþí ëáèþí óýíäåóçò. ÐñïóðáèÞóôå íá äéïñþóåôå ìå 'mysqladmin flush-hosts'" + greek "Ï õðïëïãéóôÞò '%-.64s' Ý÷åé áðïêëåéóèåß ëüãù ðïëëáðëþí ëáèþí óýíäåóçò. ÐñïóðáèÞóôå íá äéïñþóåôå ìå 'mysqladmin flush-hosts'" hun "A '%-.64s' host blokkolodott, tul sok kapcsolodasi hiba miatt. Hasznalja a 'mysqladmin flush-hosts' parancsot" ita "Sistema '%-.64s' bloccato a causa di troppi errori di connessione. Per sbloccarlo: 'mysqladmin flush-hosts'" jpn "Host '%-.64s' ¤Ï many connection error ¤Î¤¿¤á¡¢µñÈݤµ¤ì¤Þ¤·¤¿. 'mysqladmin flush-hosts' ¤Ç²ò½ü¤·¤Æ¤¯¤À¤µ¤¤" @@ -3054,7 +3054,7 @@ ER_HOST_NOT_PRIVILEGED est "Masinal '%-.64s' puudub ligipääs sellele MySQL serverile" fre "Le hôte '%-.64s' n'est pas authorisé à se connecter à ce serveur MySQL" ger "Host '%-.64s' hat keine Berechtigung, sich mit diesem MySQL-Server zu verbinden" - greek "Ï õðïëïãéóôÞò äåí Ý÷åé äéêáßùìá óýíäåóçò ìå ôïí MySQL server" + greek "Ï õðïëïãéóôÞò '%-.64s' äåí Ý÷åé äéêáßùìá óýíäåóçò ìå ôïí MySQL server" hun "A '%-.64s' host szamara nem engedelyezett a kapcsolodas ehhez a MySQL szerverhez" ita "Al sistema '%-.64s' non e` consentita la connessione a questo server MySQL" jpn "Host '%-.64s' ¤Ï MySQL server ¤ËÀܳ¤òµö²Ä¤µ¤ì¤Æ¤¤¤Þ¤»¤ó" @@ -3368,27 +3368,27 @@ ER_GRANT_WRONG_HOST_OR_USER 42000 swe "Felaktigt maskinnamn eller användarnamn använt med GRANT" ukr "áÒÇÕÍÅÎÔ host ÁÂÏ user ÄÌÑ GRANT ÚÁÄÏ×ÇÉÊ" ER_NO_SUCH_TABLE 42S02 - cze "Tabulka '%-.64s.%s' neexistuje" + cze "Tabulka '%-.64s.%-.64s' neexistuje" dan "Tabellen '%-.64s.%-.64s' eksisterer ikke" - nla "Tabel '%-.64s.%s' bestaat niet" + nla "Tabel '%-.64s.%-.64s' bestaat niet" eng "Table '%-.64s.%-.64s' doesn't exist" est "Tabelit '%-.64s.%-.64s' ei eksisteeri" - fre "La table '%-.64s.%s' n'existe pas" + fre "La table '%-.64s.%-.64s' n'existe pas" ger "Tabelle '%-.64s.%-.64s' existiert nicht" - hun "A '%-.64s.%s' tabla nem letezik" - ita "La tabella '%-.64s.%s' non esiste" - jpn "Table '%-.64s.%s' doesn't exist" - kor "Å×À̺í '%-.64s.%s' ´Â Á¸ÀçÇÏÁö ¾Ê½À´Ï´Ù." - nor "Table '%-.64s.%s' doesn't exist" - norwegian-ny "Table '%-.64s.%s' doesn't exist" - pol "Table '%-.64s.%s' doesn't exist" + hun "A '%-.64s.%-.64s' tabla nem letezik" + ita "La tabella '%-.64s.%-.64s' non esiste" + jpn "Table '%-.64s.%-.64s' doesn't exist" + kor "Å×À̺í '%-.64s.%-.64s' ´Â Á¸ÀçÇÏÁö ¾Ê½À´Ï´Ù." + nor "Table '%-.64s.%-.64s' doesn't exist" + norwegian-ny "Table '%-.64s.%-.64s' doesn't exist" + pol "Table '%-.64s.%-.64s' doesn't exist" por "Tabela '%-.64s.%-.64s' não existe" rum "Tabela '%-.64s.%-.64s' nu exista" rus "ôÁÂÌÉÃÁ '%-.64s.%-.64s' ÎÅ ÓÕÝÅÓÔ×ÕÅÔ" serbian "Tabela '%-.64s.%-.64s' ne postoji" - slo "Table '%-.64s.%s' doesn't exist" - spa "Tabla '%-.64s.%s' no existe" - swe "Det finns ingen tabell som heter '%-.64s.%s'" + slo "Table '%-.64s.%-.64s' doesn't exist" + spa "Tabla '%-.64s.%-.64s' no existe" + swe "Det finns ingen tabell som heter '%-.64s.%-.64s'" ukr "ôÁÂÌÉÃÑ '%-.64s.%-.64s' ÎÅ ¦ÓÎÕ¤" ER_NONEXISTING_TABLE_GRANT 42000 cze "Neexistuje odpov-Bídající grant pro u¾ivatele '%-.32s' na stroji '%-.64s' pro tabulku '%-.64s'" @@ -3487,27 +3487,27 @@ ER_TOO_MANY_DELAYED_THREADS swe "Det finns redan 'max_delayed_threads' trådar i använding" ukr "úÁÂÁÇÁÔÏ ÚÁÔÒÉÍÁÎÉÈ Ç¦ÌÏË ×ÉËÏÒÉÓÔÏ×Õ¤ÔØÓÑ" ER_ABORTING_CONNECTION 08S01 - cze "Zru-B¹eno spojení %ld do databáze: '%-.64s' u¾ivatel: '%-.64s' (%s)" - dan "Afbrudt forbindelse %ld til database: '%-.64s' bruger: '%-.64s' (%-.64s)" - nla "Afgebroken verbinding %ld naar db: '%-.64s' gebruiker: '%-.64s' (%s)" + cze "Zru-B¹eno spojení %ld do databáze: '%-.64s' u¾ivatel: '%-.32s' (%-.64s)" + dan "Afbrudt forbindelse %ld til database: '%-.64s' bruger: '%-.32s' (%-.64s)" + nla "Afgebroken verbinding %ld naar db: '%-.64s' gebruiker: '%-.32s' (%-.64s)" eng "Aborted connection %ld to db: '%-.64s' user: '%-.32s' (%-.64s)" est "Ühendus katkestatud %ld andmebaasile: '%-.64s' kasutajale: '%-.32s' (%-.64s)" - fre "Connection %ld avortée vers la bd: '%-.64s' utilisateur: '%-.64s' (%s)" - ger "Abbruch der Verbindung %ld zur Datenbank '%-.64s'. Benutzer: '%-.64s' (%-.64s)" - hun "Megszakitott kapcsolat %ld db: '%-.64s' adatbazishoz, felhasznalo: '%-.64s' (%s)" - ita "Interrotta la connessione %ld al db: '%-.64s' utente: '%-.64s' (%s)" - jpn "Aborted connection %ld to db: '%-.64s' user: '%-.64s' (%s)" - kor "µ¥ÀÌŸº£À̽º Á¢¼ÓÀ» À§ÇÑ ¿¬°á %ld°¡ Áß´ÜµÊ : '%-.64s' »ç¿ëÀÚ: '%-.64s' (%s)" - nor "Aborted connection %ld to db: '%-.64s' user: '%-.64s' (%s)" - norwegian-ny "Aborted connection %ld to db: '%-.64s' user: '%-.64s' (%s)" - pol "Aborted connection %ld to db: '%-.64s' user: '%-.64s' (%s)" + fre "Connection %ld avortée vers la bd: '%-.64s' utilisateur: '%-.32s' (%-.64s)" + ger "Abbruch der Verbindung %ld zur Datenbank '%-.64s'. Benutzer: '%-.32s' (%-.64s)" + hun "Megszakitott kapcsolat %ld db: '%-.64s' adatbazishoz, felhasznalo: '%-.32s' (%-.64s)" + ita "Interrotta la connessione %ld al db: '%-.64s' utente: '%-.32s' (%-.64s)" + jpn "Aborted connection %ld to db: '%-.64s' user: '%-.32s' (%-.64s)" + kor "µ¥ÀÌŸº£À̽º Á¢¼ÓÀ» À§ÇÑ ¿¬°á %ld°¡ Áß´ÜµÊ : '%-.64s' »ç¿ëÀÚ: '%-.32s' (%-.64s)" + nor "Aborted connection %ld to db: '%-.64s' user: '%-.32s' (%-.64s)" + norwegian-ny "Aborted connection %ld to db: '%-.64s' user: '%-.32s' (%-.64s)" + pol "Aborted connection %ld to db: '%-.64s' user: '%-.32s' (%-.64s)" por "Conexão %ld abortou para o banco de dados '%-.64s' - usuário '%-.32s' (%-.64s)" rum "Conectie terminata %ld la baza de date: '%-.64s' utilizator: '%-.32s' (%-.64s)" rus "ðÒÅÒ×ÁÎÏ ÓÏÅÄÉÎÅÎÉÅ %ld Ë ÂÁÚÅ ÄÁÎÎÙÈ '%-.64s' ÐÏÌØÚÏ×ÁÔÅÌÑ '%-.32s' (%-.64s)" serbian "Prekinuta konekcija broj %ld ka bazi: '%-.64s' korisnik je bio: '%-.32s' (%-.64s)" - slo "Aborted connection %ld to db: '%-.64s' user: '%-.64s' (%s)" - spa "Conexión abortada %ld para db: '%-.64s' usuario: '%-.64s' (%s)" - swe "Avbröt länken för tråd %ld till db '%-.64s', användare '%-.64s' (%s)" + slo "Aborted connection %ld to db: '%-.64s' user: '%-.32s' (%-.64s)" + spa "Conexión abortada %ld para db: '%-.64s' usuario: '%-.32s' (%-.64s)" + swe "Avbröt länken för tråd %ld till db '%-.64s', användare '%-.32s' (%-.64s)" ukr "ðÅÒÅÒ×ÁÎÏ Ú'¤ÄÎÁÎÎÑ %ld ÄÏ ÂÁÚÉ ÄÁÎÎÉÈ: '%-.64s' ËÏÒÉÓÔÕ×ÁÞÁ: '%-.32s' (%-.64s)" ER_NET_PACKET_TOO_LARGE 08S01 cze "Zji-B¹tìn pøíchozí packet del¹í ne¾ 'max_allowed_packet'" @@ -3533,7 +3533,7 @@ ER_NET_READ_ERROR_FROM_PIPE 08S01 nla "Kreeg leesfout van de verbindings pipe" eng "Got a read error from the connection pipe" est "Viga ühendustoru lugemisel" - fre "Erreur de lecture reçue du pipe de connection" + fre "Erreur de lecture reçue du pipe de connexion" ger "Lese-Fehler bei einer Verbindungs-Pipe" hun "Olvasasi hiba a kapcsolat soran" ita "Rilevato un errore di lettura dalla pipe di connessione" @@ -3986,7 +3986,7 @@ ER_CHECK_NOT_IMPLEMENTED 42000 por "O manipulador de tabela não suporta %s" rum "The handler for the table doesn't support %s" rus "ïÂÒÁÂÏÔÞÉË ÔÁÂÌÉÃÙ ÎÅ ÐÏÄÄÅÒÖÉ×ÁÅÔ ÜÔÏÇÏ: %s" - serbian "Handler za ovu tabelu ne dozvoljava 'check' odnosno 'repair' komande" + serbian "Handler za ovu tabelu ne dozvoljava %s komande" slo "The handler for the table doesn't support %s" spa "El manipulador de la tabla no permite soporte para %s" swe "Tabellhanteraren för denna tabell kan inte göra %s" @@ -4331,7 +4331,7 @@ ER_TOO_MANY_USER_CONNECTIONS 42000 nla "Gebruiker %-.64s heeft reeds meer dan 'max_user_connections' actieve verbindingen" eng "User %-.64s already has more than 'max_user_connections' active connections" est "Kasutajal %-.64s on juba rohkem ühendusi kui lubatud 'max_user_connections' muutujaga" - fre "L'utilisateur %-.64s possède déjà plus de 'max_user_connections' connections actives" + fre "L'utilisateur %-.64s possède déjà plus de 'max_user_connections' connexions actives" ger "Benutzer '%-.64s' hat mehr als 'max_user_connections' aktive Verbindungen" ita "L'utente %-.64s ha gia' piu' di 'max_user_connections' connessioni attive" por "Usuário '%-.64s' já possui mais que o valor máximo de conexões (max_user_connections) ativas" @@ -4654,14 +4654,14 @@ ER_NO_DEFAULT 42000 spa "Variable '%-.64s' no tiene un valor patrón" swe "Variabel '%-.64s' har inte ett DEFAULT-värde" ER_WRONG_VALUE_FOR_VAR 42000 - nla "Variabele '%-.64s' kan niet worden gewijzigd naar de waarde '%-.64s'" - eng "Variable '%-.64s' can't be set to the value of '%-.64s'" - ger "Variable '%-.64s' kann nicht auf '%-.64s' gesetzt werden" - ita "Alla variabile '%-.64s' non puo' essere assegato il valore '%-.64s'" - por "Variável '%-.64s' não pode ser configurada para o valor de '%-.64s'" - rus "ðÅÒÅÍÅÎÎÁÑ '%-.64s' ÎÅ ÍÏÖÅÔ ÂÙÔØ ÕÓÔÁÎÏ×ÌÅÎÁ × ÚÎÁÞÅÎÉÅ '%-.64s'" - spa "Variable '%-.64s' no puede ser configurada para el valor de '%-.64s'" - swe "Variabel '%-.64s' kan inte sättas till '%-.64s'" + nla "Variabele '%-.64s' kan niet worden gewijzigd naar de waarde '%-.200s'" + eng "Variable '%-.64s' can't be set to the value of '%-.200s'" + ger "Variable '%-.64s' kann nicht auf '%-.200s' gesetzt werden" + ita "Alla variabile '%-.64s' non puo' essere assegato il valore '%-.200s'" + por "Variável '%-.64s' não pode ser configurada para o valor de '%-.200s'" + rus "ðÅÒÅÍÅÎÎÁÑ '%-.64s' ÎÅ ÍÏÖÅÔ ÂÙÔØ ÕÓÔÁÎÏ×ÌÅÎÁ × ÚÎÁÞÅÎÉÅ '%-.200s'" + spa "Variable '%-.64s' no puede ser configurada para el valor de '%-.200s'" + swe "Variabel '%-.64s' kan inte sättas till '%-.200s'" ER_WRONG_TYPE_FOR_VAR 42000 nla "Foutief argumenttype voor variabele '%-.64s'" eng "Incorrect argument type to variable '%-.64s'" @@ -5220,9 +5220,9 @@ ER_FPARSER_BAD_HEADER ukr "îÅצÒÎÉÊ ÚÁÇÏÌÏ×ÏË ÔÉÐÕ Õ ÆÁÊ̦ '%-.64s'" ER_FPARSER_EOF_IN_COMMENT eng "Unexpected end of file while parsing comment '%-.200s'" - ger "Unerwartetes Dateiende beim Parsen des Kommentars '%-.64s'" - rus "îÅÏÖÉÄÁÎÎÙÊ ËÏÎÅà ÆÁÊÌÁ × ËÏÍÅÎÔÁÒÉÉ '%-.64s'" - ukr "îÅÓÐÏĦ×ÁÎÎÉÊ Ë¦ÎÅÃØ ÆÁÊÌÕ Õ ËÏÍÅÎÔÁÒ¦ '%-.64s'" + ger "Unerwartetes Dateiende beim Parsen des Kommentars '%-.200s'" + rus "îÅÏÖÉÄÁÎÎÙÊ ËÏÎÅà ÆÁÊÌÁ × ËÏÍÅÎÔÁÒÉÉ '%-.200s'" + ukr "îÅÓÐÏĦ×ÁÎÎÉÊ Ë¦ÎÅÃØ ÆÁÊÌÕ Õ ËÏÍÅÎÔÁÒ¦ '%-.200s'" ER_FPARSER_ERROR_IN_PARAMETER eng "Error while parsing parameter '%-.64s' (line: '%-.64s')" ger "Fehler beim Parsen des Parameters '%-.64s' (Zeile: '%-.64s')" @@ -5307,7 +5307,7 @@ ER_TRG_ON_VIEW_OR_TEMP_TABLE ger "'%-.64s' des Triggers ist View oder temporäre Tabelle" ER_TRG_CANT_CHANGE_ROW eng "Updating of %s row is not allowed in %strigger" - ger "Aktualisieren einer %s-Zeile ist in einem %-Trigger nicht erlaubt" + ger "Aktualisieren einer %s-Zeile ist in einem %s-Trigger nicht erlaubt" ER_TRG_NO_SUCH_ROW_IN_TRG eng "There is no %s row in %s trigger" ger "Es gibt keine %s-Zeile im %s-Trigger" @@ -5389,7 +5389,7 @@ ER_LOGGING_PROHIBIT_CHANGING_OF ger "Binärlogs und Replikation verhindern Wechsel des globalen Servers %s" ER_NO_FILE_MAPPING eng "Can't map file: %-.200s, errno: %d" - ger "Kann Datei nicht abbilden: %-.64s, Fehler: %d" + ger "Kann Datei nicht abbilden: %-.200s, Fehler: %d" ER_WRONG_MAGIC eng "Wrong magic in %-.64s" ger "Falsche magische Zahlen in %-.64s" @@ -5518,13 +5518,12 @@ ER_M_BIGGER_THAN_D 42000 S1009 eng "For float(M,D), double(M,D) or decimal(M,D), M must be >= D (column '%-.64s')." ger "Für FLOAT(M,D), DOUBLE(M,D) oder DECIMAL(M,D) muss M >= D sein (Feld '%-.64s')" ER_WRONG_LOCK_OF_SYSTEM_TABLE - eng "You can't combine write-locking of system '%-.64s.%-.64s' table with other tables" - ger "Sie können Schreibsperren auf der Systemtabelle '%-.64s.%-.64s' nicht mit anderen Tabellen kombinieren" + eng "You can't combine write-locking of system tables with other tables or lock types" ER_CONNECT_TO_FOREIGN_DATA_SOURCE eng "Unable to connect to foreign data source: %.64s" ger "Kann nicht mit Fremddatenquelle verbinden: %.64s" ER_QUERY_ON_FOREIGN_DATA_SOURCE - eng "There was a problem processing the query on the foreign data source. Data source error: %-.64" + eng "There was a problem processing the query on the foreign data source. Data source error: %-.64s" ger "Bei der Verarbeitung der Abfrage ist in der Fremddatenquelle ein Problem aufgetreten. Datenquellenfehlermeldung: %-.64s" ER_FOREIGN_DATA_SOURCE_DOESNT_EXIST eng "The foreign data source you are trying to reference does not exist. Data source error: %-.64s" @@ -5606,7 +5605,9 @@ ER_SP_RECURSION_LIMIT ger "Rekursionsgrenze %d (durch Variable max_sp_recursion_depth gegeben) wurde für Routine %.64s überschritten" ER_SP_PROC_TABLE_CORRUPT eng "Failed to load routine %-.64s. The table mysql.proc is missing, corrupt, or contains bad data (internal code %d)" - ger "Routine %-64s konnte nicht geladen werden. Die Tabelle mysql.proc fehlt, ist beschädigt, oder enthält fehlerhaften Daten (interner Code: %d)" + ger "Routine %-.64s konnte nicht geladen werden. Die Tabelle mysql.proc fehlt, ist beschädigt, oder enthält fehlerhaften Daten (interner Code: %d)" +ER_FOREIGN_SERVER_EXISTS + eng "The foreign server, %s, you are trying to create already exists." ER_SP_WRONG_NAME 42000 eng "Incorrect routine name '%-.64s'" ger "Ungültiger Routinenname '%-.64s'" @@ -5847,6 +5848,9 @@ ER_BINLOG_ROW_WRONG_TABLE_DEF ER_BINLOG_ROW_RBR_TO_SBR eng "Slave running with --log-slave-updates must use row-based binary logging to be able to replicate row-based binary log events" ger "Slave, die mit --log-slave-updates laufen, müssen zeilenbasiertes Loggen verwenden, um zeilenbasierte Binärlog-Ereignisse loggen zu können" +ER_FOREIGN_SERVER_DOESNT_EXIST + eng "The foreign server name you are trying to reference does not exist. Data source error: %-.64s" + ger "Die externe Verbindung, auf die Sie zugreifen wollen, existiert nicht. Datenquellenfehlermeldung: %-.64s" ER_EVENT_ALREADY_EXISTS eng "Event '%-.64s' already exists" ger "Event '%-.64s' existiert bereits" @@ -5869,8 +5873,7 @@ ER_EVENT_ENDS_BEFORE_STARTS eng "ENDS is either invalid or before STARTS" ger "ENDS ist entweder ungültig oder liegt vor STARTS" ER_EVENT_EXEC_TIME_IN_THE_PAST - eng "Activation (AT) time is in the past" - ger "Aktivierungszeit (AT) liegt in der Vergangenheit" + eng "Event execution time is in the past. Event has been disabled" ER_EVENT_OPEN_TABLE_FAILED eng "Failed to open mysql.event" ger "Öffnen von mysql.event fehlgeschlagen" @@ -6004,15 +6007,51 @@ ER_BAD_LOG_STATEMENT ger "Sie können eine Logtabelle nicht '%s', wenn Loggen angeschaltet ist" ER_NON_INSERTABLE_TABLE eng "The target table %-.100s of the %s is not insertable-into" + ger "Die Zieltabelle %-.100s von %s ist nicht einfügbar" ER_CANT_RENAME_LOG_TABLE eng "Cannot rename '%s'. When logging enabled, rename to/from log table must rename two tables: the log table to an archive table and another table back to '%s'" + ger "Kann '%s' nicht umbenennen. Wenn Loggen angeschaltet ist, müssen beim Umbenennen zu/von einer Logtabelle zwei Tabellen angegeben werden: die Logtabelle zu einer Archivtabelle und eine weitere Tabelle zurück zu '%s'" ER_WRONG_PARAMCOUNT_TO_NATIVE_FCT 42000 eng "Incorrect parameter count in the call to native function '%-.64s'" + ger "Falsche Anzahl von Parametern beim Aufruf der nativen Funktion '%-.64s'" ER_WRONG_PARAMETERS_TO_NATIVE_FCT 42000 eng "Incorrect parameters in the call to native function '%-.64s'" + ger "Falscher Parameter beim Aufruf der nativen Funktion '%-.64s'" ER_WRONG_PARAMETERS_TO_STORED_FCT 42000 eng "Incorrect parameters in the call to stored function '%-.64s'" + ger "Falsche Parameter beim Aufruf der gespeicherten Funktion '%-.64s'" ER_NATIVE_FCT_NAME_COLLISION - eng "This function '%-.64s' has the same name as a native function." + eng "This function '%-.64s' has the same name as a native function" + ger "Die Funktion '%-.64s' hat denselben Namen wie eine native Funktion" +ER_DUP_ENTRY_WITH_KEY_NAME 23000 S1009 + cze "Zvojen-Bý klíè '%-.64s' (èíslo klíèe '%-.64s')" + dan "Ens værdier '%-.64s' for indeks '%-.64s'" + nla "Dubbele ingang '%-.64s' voor zoeksleutel '%-.64s'" + eng "Duplicate entry '%-.64s' for key '%-.64s'" + jps "'%-.64s' ‚Í key '%-.64s' ‚É‚¨‚¢‚Äd•¡‚µ‚Ä‚¢‚Ü‚·", + est "Kattuv väärtus '%-.64s' võtmele '%-.64s'" + fre "Duplicata du champ '%-.64s' pour la clef '%-.64s'" + ger "Doppelter Eintrag '%-.64s' für Schlüssel '%-.64s'" + greek "ÄéðëÞ åããñáöÞ '%-.64s' ãéá ôï êëåéäß '%-.64s'" + hun "Duplikalt bejegyzes '%-.64s' a '%-.64s' kulcs szerint." + ita "Valore duplicato '%-.64s' per la chiave '%-.64s'" + jpn "'%-.64s' ¤Ï key '%-.64s' ¤Ë¤ª¤¤¤Æ½ÅÊ£¤·¤Æ¤¤¤Þ¤¹" + kor "Áߺ¹µÈ ÀÔ·Â °ª '%-.64s': key '%-.64s'" + nor "Like verdier '%-.64s' for nøkkel '%-.64s'" + norwegian-ny "Like verdiar '%-.64s' for nykkel '%-.64s'" + pol "Powtórzone wyst?pienie '%-.64s' dla klucza '%-.64s'" + por "Entrada '%-.64s' duplicada para a chave '%-.64s'" + rum "Cimpul '%-.64s' e duplicat pentru cheia '%-.64s'" + rus "äÕÂÌÉÒÕÀÝÁÑÓÑ ÚÁÐÉÓØ '%-.64s' ÐÏ ËÌÀÞÕ '%-.64s'" + serbian "Dupliran unos '%-.64s' za kljuè '%-.64s'" + slo "Opakovaný kµúè '%-.64s' (èíslo kµúèa '%-.64s')" + spa "Entrada duplicada '%-.64s' para la clave '%-.64s'" + swe "Dubbel nyckel '%-.64s' för nyckel '%-.64s'" + ukr "äÕÂÌÀÀÞÉÊ ÚÁÐÉÓ '%-.64s' ÄÌÑ ËÌÀÞÁ '%-.64s'" ER_BINLOG_PURGE_EMFILE eng "Too many files opened, please execute the command again" + ger "Zu viele offene Dateien, bitte führen Sie den Befehl noch einmal aus" +ER_EVENT_CANNOT_CREATE_IN_THE_PAST + eng "Event execution time is in the past and ON COMPLETION NOT PRESERVE is set. Event has not been created" +ER_EVENT_CANNOT_ALTER_IN_THE_PAST + eng "Event execution time is in the past and ON COMPLETION NOT PRESERVE is set. Event has not been altered" diff --git a/sql/slave.cc b/sql/slave.cc index 67e8ba20c4f..45b7acf9875 100644 --- a/sql/slave.cc +++ b/sql/slave.cc @@ -27,6 +27,8 @@ #include #include +static Log_event* next_event(RELAY_LOG_INFO* rli); + #ifdef HAVE_REPLICATION #include "rpl_tblmap.h" diff --git a/sql/slave.h b/sql/slave.h index 24ba09d78d3..b29cd2b2975 100644 --- a/sql/slave.h +++ b/sql/slave.h @@ -112,8 +112,6 @@ extern ulonglong relay_log_space_limit; #define MYSQL_SLAVE_RUN_NOT_CONNECT 1 #define MYSQL_SLAVE_RUN_CONNECT 2 -static Log_event* next_event(RELAY_LOG_INFO* rli); - #define RPL_LOG_NAME (rli->group_master_log_name[0] ? rli->group_master_log_name :\ "FIRST") #define IO_RPL_LOG_NAME (mi->master_log_name[0] ? mi->master_log_name :\ diff --git a/sql/sql_table.cc b/sql/sql_table.cc index c8f6e09fecb..b53efbe021f 100644 --- a/sql/sql_table.cc +++ b/sql/sql_table.cc @@ -599,7 +599,6 @@ bool read_ddl_log_entry(uint read_entry, DDL_LOG_ENTRY *ddl_log_entry) static bool init_ddl_log() { - bool error= FALSE; char file_name[FN_REFLEN]; DBUG_ENTER("init_ddl_log"); @@ -6805,7 +6804,7 @@ copy_data_between_tables(TABLE *from,TABLE *to, uint key_nr= to->file->get_dup_key(error); if ((int) key_nr >= 0) { - const char *err_msg= ER(ER_DUP_ENTRY); + const char *err_msg= ER(ER_DUP_ENTRY_WITH_KEY_NAME); if (key_nr == 0 && (to->key_info[0].key_part[0].field->flags & AUTO_INCREMENT_FLAG)) diff --git a/storage/maria/ha_maria.cc b/storage/maria/ha_maria.cc index 102f987f01a..7cd0b763dcc 100644 --- a/storage/maria/ha_maria.cc +++ b/storage/maria/ha_maria.cc @@ -195,6 +195,14 @@ const char *ha_maria::index_type(uint key_number) } +double ha_maria::scan_time() +{ + if (file->s->data_file_type == BLOCK_RECORD) + return ulonglong2double(stats.data_file_length - file->s->block_size) / max(file->s->block_size / 2, IO_SIZE) + 2; + return handler::scan_time(); +} + + #ifdef HAVE_REPLICATION int ha_maria::net_read_dump(NET * net) { @@ -329,7 +337,7 @@ int ha_maria::open(const char *name, int mode, uint test_if_locked) info(HA_STATUS_NO_LOCK | HA_STATUS_VARIABLE | HA_STATUS_CONST); if (!(test_if_locked & HA_OPEN_WAIT_IF_LOCKED)) VOID(maria_extra(file, HA_EXTRA_WAIT_LOCK, 0)); - if (!table->s->db_record_offset) + if (file->s->data_file_type != STATIC_RECORD) int_table_flags |= HA_REC_NOT_IN_SEQ; if (file->s->options & (HA_OPTION_CHECKSUM | HA_OPTION_COMPRESS_RECORD)) int_table_flags |= HA_HAS_CHECKSUM; @@ -704,6 +712,16 @@ int ha_maria::repair(THD *thd, HA_CHECK ¶m, bool optimize) param.out_flag= 0; strmov(fixed_name, file->filename); +#ifndef TO_BE_FIXED + /* QQ: Until we have repair for block format, lie that it succeded */ + if (file->s->data_file_type == BLOCK_RECORD) + { + if (optimize) + DBUG_RETURN(analyze(thd, (HA_CHECK_OPT*) 0)); + DBUG_RETURN(HA_ADMIN_OK); + } +#endif + // Don't lock tables if we have used LOCK TABLE if (!thd->locked_tables && maria_lock_database(file, table->s->tmp_table ? F_EXTRA_LCK : F_WRLCK)) @@ -715,7 +733,8 @@ int ha_maria::repair(THD *thd, HA_CHECK ¶m, bool optimize) if (!optimize || ((file->state->del || share->state.split != file->state->records) && (!(param.testflag & T_QUICK) || - !(share->state.changed & STATE_NOT_OPTIMIZED_KEYS)))) + (share->state.changed & (STATE_NOT_OPTIMIZED_KEYS | + STATE_NOT_OPTIMIZED_ROWS))))) { ulonglong key_map= ((local_testflag & T_CREATE_MISSING_KEYS) ? maria_get_mask_all_keys_active(share->base.keys) : @@ -1125,6 +1144,8 @@ void ha_maria::start_bulk_insert(ha_rows rows) can_enable_indexes= maria_is_all_keys_active(file->s->state.key_map, file->s->base.keys); + /* TODO: Remove when we have repair() working */ + can_enable_indexes= 0; if (!(specialflag & SPECIAL_SAFE_MODE)) { @@ -1163,10 +1184,12 @@ void ha_maria::start_bulk_insert(ha_rows rows) int ha_maria::end_bulk_insert() { + int err; + DBUG_ENTER("ha_maria::end_bulk_insert"); maria_end_bulk_insert(file); - int err= maria_extra(file, HA_EXTRA_NO_CACHE, 0); - return err ? err : can_enable_indexes ? - enable_indexes(HA_KEY_SWITCH_NONUNIQ_SAVE) : 0; + err= maria_extra(file, HA_EXTRA_NO_CACHE, 0); + DBUG_RETURN(err ? err : can_enable_indexes ? + enable_indexes(HA_KEY_SWITCH_NONUNIQ_SAVE) : 0); } @@ -1336,6 +1359,14 @@ int ha_maria::rnd_init(bool scan) } +int ha_maria::rnd_end() +{ + /* Safe to call even if we don't have started a scan */ + maria_scan_end(file); + return 0; +} + + int ha_maria::rnd_next(byte *buf) { statistic_increment(table->in_use->status_var.ha_read_rnd_next_count, @@ -1499,6 +1530,29 @@ void ha_maria::update_create_info(HA_CREATE_INFO *create_info) } +enum row_type ha_maria::get_row_type() const +{ + switch (file->s->data_file_type) { + case STATIC_RECORD: return ROW_TYPE_FIXED; + case DYNAMIC_RECORD: return ROW_TYPE_DYNAMIC; + case BLOCK_RECORD: return ROW_TYPE_PAGES; + case COMPRESSED_RECORD: return ROW_TYPE_COMPRESSED; + default: return ROW_TYPE_NOT_USED; + } +} + + +static enum data_file_type maria_row_type(HA_CREATE_INFO *info) +{ + switch (info->row_type) { + case ROW_TYPE_FIXED: return STATIC_RECORD; + case ROW_TYPE_DYNAMIC: return DYNAMIC_RECORD; + default: return BLOCK_RECORD; + } +} + + + int ha_maria::create(const char *name, register TABLE *table_arg, HA_CREATE_INFO *info) { @@ -1518,6 +1572,7 @@ int ha_maria::create(const char *name, register TABLE *table_arg, DBUG_ENTER("ha_maria::create"); type= HA_KEYTYPE_BINARY; // Keep compiler happy + row_type= maria_row_type(info); if (!(my_multi_malloc(MYF(MY_WME), &recinfo, (share->fields * 2 + 2) * sizeof(MARIA_COLUMNDEF), @@ -1652,7 +1707,8 @@ int ha_maria::create(const char *name, register TABLE *table_arg, recinfo_pos->type= FIELD_BLOB; else if (found->type() == MYSQL_TYPE_VARCHAR) recinfo_pos->type= FIELD_VARCHAR; - else if (!(options & HA_OPTION_PACK_RECORD)) + else if (!(options & HA_OPTION_PACK_RECORD) || + (found->zero_pack() && (found->flags & PRI_KEY_FLAG))) recinfo_pos->type= FIELD_NORMAL; else if (found->zero_pack()) recinfo_pos->type= FIELD_SKIP_ZERO; @@ -1701,19 +1757,6 @@ int ha_maria::create(const char *name, register TABLE *table_arg, if (options & HA_OPTION_DELAY_KEY_WRITE) create_flags |= HA_CREATE_DELAY_KEY_WRITE; - switch (info->row_type) { - case ROW_TYPE_FIXED: - row_type= STATIC_RECORD; - break; - case ROW_TYPE_DYNAMIC: - row_type= DYNAMIC_RECORD; - break; - default: - case ROW_TYPE_PAGES: - row_type= BLOCK_RECORD; - break; - } - /* TODO: Check that the following fn_format is really needed */ error= maria_create(fn_format(buff, name, "", "", @@ -1843,6 +1886,7 @@ bool ha_maria::check_if_incompatible_data(HA_CREATE_INFO *info, if (info->auto_increment_value != stats.auto_increment_value || info->data_file_name != data_file_name || info->index_file_name != index_file_name || + maria_row_type(info) != data_file_type || table_changes == IS_EQUAL_NO || table_changes & IS_EQUAL_PACK_LENGTH) // Not implemented yet return COMPATIBLE_DATA_NO; diff --git a/storage/maria/ha_maria.h b/storage/maria/ha_maria.h index 1f243f9ec59..d16e06e35d5 100644 --- a/storage/maria/ha_maria.h +++ b/storage/maria/ha_maria.h @@ -38,6 +38,7 @@ class ha_maria :public handler MARIA_HA *file; ulonglong int_table_flags; char *data_file_name, *index_file_name; + enum data_file_type data_file_type; bool can_enable_indexes; int repair(THD * thd, HA_CHECK ¶m, bool optimize); @@ -63,7 +64,9 @@ public: { return HA_MAX_KEY_LENGTH; } uint max_supported_key_part_length() const { return HA_MAX_KEY_LENGTH; } + enum row_type get_row_type() const; uint checksum() const; + virtual double scan_time(); virtual bool check_if_locking_is_allowed(uint sql_command, ulong type, TABLE * table, @@ -99,6 +102,7 @@ public: } int ft_read(byte * buf); int rnd_init(bool scan); + int rnd_end(void); int rnd_next(byte * buf); int rnd_pos(byte * buf, byte * pos); int restart_rnd_next(byte * buf, byte * pos); diff --git a/storage/maria/ma_bitmap.c b/storage/maria/ma_bitmap.c index 5ed5a776658..c632efba669 100644 --- a/storage/maria/ma_bitmap.c +++ b/storage/maria/ma_bitmap.c @@ -937,7 +937,7 @@ static my_bool find_mid(MARIA_HA *info, ulong pages, uint position) MARIA_BITMAP_BLOCK *block; block= dynamic_element(&info->bitmap_blocks, position, MARIA_BITMAP_BLOCK *); - while (allocate_full_pages(bitmap, pages, block, 1)) + while (!allocate_full_pages(bitmap, pages, block, 1)) { if (move_to_next_bitmap(info, bitmap)) return 1; @@ -1101,6 +1101,9 @@ static my_bool write_rest_of_head(MARIA_HA *info, uint position, MARIA_SHARE *share= info->s; uint full_page_size= FULL_PAGE_SIZE(share->block_size); MARIA_BITMAP_BLOCK *block; + DBUG_ENTER("write_rest_of_head"); + DBUG_PRINT("enter", ("position: %u rest_length: %lu", position, + rest_length)); if (position == 0) { @@ -1114,8 +1117,8 @@ static my_bool write_rest_of_head(MARIA_HA *info, uint position, pages++; rest_length= 0; } - if (find_mid(info, rest_length / full_page_size, 1)) - return 1; + if (find_mid(info, pages, 1)) + DBUG_RETURN(1); /* Insert empty block after full pages, to allow write_block_record() to split segment into used + free page @@ -1127,7 +1130,7 @@ static my_bool write_rest_of_head(MARIA_HA *info, uint position, if (rest_length) { if (find_tail(info, rest_length, ELEMENTS_RESERVED_FOR_MAIN_PART - 1)) - return 1; + DBUG_RETURN(1); } else { @@ -1138,7 +1141,7 @@ static my_bool write_rest_of_head(MARIA_HA *info, uint position, block->page_count= 0; block->used= 0; } - return 0; + DBUG_RETURN(0); } @@ -1510,6 +1513,9 @@ my_bool _ma_reset_full_page_bits(MARIA_HA *info, MARIA_FILE_BITMAP *bitmap, For the first block (head block) the logic is same as for a tail block + Note that we may have 'filler blocks' that are used to split a block + in half; These can be recognized by that they have page_count == 0. + RETURN 0 ok 1 error (Couldn't write or read bitmap page) @@ -1548,6 +1554,9 @@ my_bool _ma_bitmap_release_unused(MARIA_HA *info, MARIA_BITMAP_BLOCKS *blocks) /* Handle all full pages and tail pages (for head page and blob) */ for (block++; block < end; block++) { + if (!block->page_count) + continue; /* Skip 'filler blocks' */ + if (block->used & BLOCKUSED_TAIL) { if (block->used & BLOCKUSED_USED) diff --git a/storage/maria/ma_blockrec.c b/storage/maria/ma_blockrec.c index f1345b2c2f3..e8a3d9aa5fd 100644 --- a/storage/maria/ma_blockrec.c +++ b/storage/maria/ma_blockrec.c @@ -266,6 +266,8 @@ static my_bool delete_head_or_tail(MARIA_HA *info, ulonglong page, uint record_number, my_bool head); static void _ma_print_directory(byte *buff, uint block_size); +static void compact_page(byte *buff, uint block_size, uint rownr, + my_bool extend_block); /**************************************************************************** Initialization @@ -365,7 +367,8 @@ my_bool _ma_init_block_row(MARIA_HA *info) &row->blob_lengths, sizeof(ulong) * info->s->base.blobs, &row->null_field_lengths, (sizeof(uint) * (info->s->base.fields - - info->s->base.blobs)), + info->s->base.blobs + + EXTRA_LENGTH_FIELDS)), &row->tail_positions, (sizeof(MARIA_RECORD_POS) * (info->s->base.blobs + 2)), &new_row->empty_bits_buffer, info->s->base.pack_bytes, @@ -375,7 +378,8 @@ my_bool _ma_init_block_row(MARIA_HA *info) sizeof(ulong) * info->s->base.blobs, &new_row->null_field_lengths, (sizeof(uint) * (info->s->base.fields - - info->s->base.blobs)), + info->s->base.blobs + + EXTRA_LENGTH_FIELDS)), NullS, 0)) DBUG_RETURN(1); if (my_init_dynamic_array(&info->bitmap_blocks, @@ -383,9 +387,18 @@ my_bool _ma_init_block_row(MARIA_HA *info) ELEMENTS_RESERVED_FOR_MAIN_PART, 16)) my_free((char*) &info->bitmap_blocks, MYF(0)); row->base_length= new_row->base_length= info->s->base_length; + + /* + We need to reserve 'EXTRA_LENGTH_FIELDS' number of parts in + null_field_lengths to allow splitting of rows in 'find_where_to_split_row' + */ + + row->null_field_lengths+= EXTRA_LENGTH_FIELDS; + new_row->null_field_lengths+= EXTRA_LENGTH_FIELDS; DBUG_RETURN(0); } + void _ma_end_block_row(MARIA_HA *info) { DBUG_ENTER("_ma_end_block_row"); @@ -444,8 +457,9 @@ static my_bool check_if_zero(byte *pos, uint length) are stored on disk in inverse directory order, which makes life easier for 'compact_page()' and to know if there is free space after any block. - If there is no free entry (entry with postion == 0), then we create - a new one. + If there is no free entry (entry with position == 0), then we create + a new one. If there is not space for the directory entry (because + the last block overlapps with the directory), we compact the page. We will update the offset and the length of the found dir entry to match the position and empty space found. @@ -453,7 +467,7 @@ static my_bool check_if_zero(byte *pos, uint length) buff[EMPTY_SPACE_OFFSET] is NOT updated but left up to the caller RETURN - 0 Error (directory full) + 0 Error (directory full or last block goes over directory) # Pointer to directory entry on page */ @@ -463,6 +477,8 @@ static byte *find_free_position(byte *buff, uint block_size, uint *res_rownr, uint max_entry= (uint) ((uchar*) buff)[DIR_ENTRY_OFFSET]; uint entry, length, first_pos; byte *dir, *end; + DBUG_ENTER("find_free_position"); + DBUG_PRINT("info", ("max_entry: %u", max_entry)); dir= (buff + block_size - DIR_ENTRY_SIZE * max_entry - PAGE_SUFFIX_SIZE); end= buff + block_size - PAGE_SUFFIX_SIZE - DIR_ENTRY_SIZE; @@ -471,7 +487,7 @@ static byte *find_free_position(byte *buff, uint block_size, uint *res_rownr, *empty_space= uint2korr(buff + EMPTY_SPACE_OFFSET); /* Search after first empty position */ - for (entry= 0 ; dir <= end ; end-= DIR_ENTRY_SIZE, entry--) + for (entry= 0 ; dir <= end ; end-= DIR_ENTRY_SIZE, entry++) { if (end[0] == 0 && end[1] == 0) /* Found not used entry */ { @@ -480,15 +496,25 @@ static byte *find_free_position(byte *buff, uint block_size, uint *res_rownr, int2store(end + 2, length); *res_rownr= entry; *res_length= length; - return end; + DBUG_RETURN(end); } first_pos= uint2korr(end) + uint2korr(end + 2); } /* No empty places in dir; create a new one */ + dir= end; + /* Check if there is place for the directory entry */ if (max_entry == MAX_ROWS_PER_PAGE) - return 0; + DBUG_RETURN(0); + /* Check if there is place for the directory entry */ + if ((dir - buff) < first_pos) + { + /* Create place for directory */ + compact_page(buff, block_size, max_entry-1, 0); + first_pos= (uint2korr(end + DIR_ENTRY_SIZE) + + uint2korr(end + DIR_ENTRY_SIZE+ 2)); + *empty_space= uint2korr(buff + EMPTY_SPACE_OFFSET); + } buff[DIR_ENTRY_OFFSET]= (byte) (uchar) max_entry+1; - dir-= DIR_ENTRY_SIZE; length= (uint) (dir - buff - first_pos); DBUG_ASSERT(length <= *empty_space - DIR_ENTRY_SIZE); int2store(dir, first_pos); @@ -498,7 +524,7 @@ static byte *find_free_position(byte *buff, uint block_size, uint *res_rownr, /* Reduce directory entry size from free space size */ (*empty_space)-= DIR_ENTRY_SIZE; - return dir; + DBUG_RETURN(dir); } @@ -508,6 +534,17 @@ static byte *find_free_position(byte *buff, uint block_size, uint *res_rownr, /* Calculate length of all the different field parts + + SYNOPSIS + calc_record_size() + info Maria handler + record Row to store + row Store statistics about row here + + NOTES + The statistics is used to find out how much space a row will need + and also where we can split a row when we need to split it into several + extents. */ static void calc_record_size(MARIA_HA *info, const byte *record, @@ -516,7 +553,8 @@ static void calc_record_size(MARIA_HA *info, const byte *record, MARIA_SHARE *share= info->s; byte *field_length_data; MARIA_COLUMNDEF *rec, *end_field; - uint blob_count= 0, *null_field_lengths= row->null_field_lengths; + uint *null_field_lengths= row->null_field_lengths; + ulong *blob_lengths= row->blob_lengths; row->normal_length= row->char_length= row->varchar_length= row->blob_length= row->extents_count= 0; @@ -533,6 +571,8 @@ static void calc_record_size(MARIA_HA *info, const byte *record, { if (rec->type != FIELD_BLOB) *null_field_lengths= 0; + else + *blob_lengths++= 0; continue; } switch ((enum en_fieldtype) rec->type) { @@ -586,32 +626,31 @@ static void calc_record_size(MARIA_HA *info, const byte *record, } case FIELD_VARCHAR: { - uint length; + uint length, field_length_data_length; const byte *field_pos= record + rec->offset; /* 256 is correct as this includes the length byte */ + + field_length_data[0]= field_pos[0]; if (rec->length <= 256) { - if (!(length= (uint) (uchar) *field_pos)) - { - row->empty_bits[rec->empty_pos]|= rec->empty_bit; - *null_field_lengths= 0; - break; - } - *field_length_data++= *field_pos; + length= (uint) (uchar) *field_pos; + field_length_data_length= 1; } else { - if (!(length= uint2korr(field_pos))) - { - row->empty_bits[rec->empty_pos]|= rec->empty_bit; - break; - } - field_length_data[0]= field_pos[0]; + length= uint2korr(field_pos); field_length_data[1]= field_pos[1]; - field_length_data+= 2; + field_length_data_length= 2; + } + *null_field_lengths= length; + if (!length) + { + row->empty_bits[rec->empty_pos]|= rec->empty_bit; + break; } row->varchar_length+= length; *null_field_lengths= length; + field_length_data+= field_length_data_length; break; } case FIELD_BLOB: @@ -619,16 +658,16 @@ static void calc_record_size(MARIA_HA *info, const byte *record, const byte *field_pos= record + rec->offset; uint size_length= rec->length - maria_portable_sizeof_char_ptr; ulong blob_length= _ma_calc_blob_length(size_length, field_pos); + + *blob_lengths++= blob_length; if (!blob_length) - { row->empty_bits[rec->empty_pos]|= rec->empty_bit; - row->blob_lengths[blob_count++]= 0; - break; + else + { + row->blob_length+= blob_length; + memcpy(field_length_data, field_pos, size_length); + field_length_data+= size_length; } - row->blob_length+= blob_length; - row->blob_lengths[blob_count++]= blob_length; - memcpy(field_length_data, field_pos, size_length); - field_length_data+= size_length; break; } default: @@ -663,10 +702,13 @@ static void calc_record_size(MARIA_HA *info, const byte *record, buff Page to compact block_size Size of page recnr Put empty data after this row + extend_block If 1, extend the block at 'rownr' to cover the + whole block. */ -void compact_page(byte *buff, uint block_size, uint rownr) +static void compact_page(byte *buff, uint block_size, uint rownr, + my_bool extend_block) { uint max_entry= (uint) ((uchar *) buff)[DIR_ENTRY_OFFSET]; uint page_pos, next_free_pos, start_of_found_block, diff, end_of_found_block; @@ -765,10 +807,12 @@ void compact_page(byte *buff, uint block_size, uint rownr) } else { - /* Extend last block cover whole page */ - uint length= (uint) (dir - buff) - start_of_found_block; - int2store(dir+2, length); - + if (extend_block) + { + /* Extend last block cover whole page */ + uint length= (uint) (dir - buff) - start_of_found_block; + int2store(dir+2, length); + } buff[PAGE_TYPE_OFFSET]&= ~(byte) PAGE_CAN_BE_COMPACTED; } DBUG_EXECUTE("directory", _ma_print_directory(buff, block_size);); @@ -837,9 +881,9 @@ static my_bool get_head_or_tail_page(MARIA_HA *info, res->empty_space= res->length= (block_size - PAGE_OVERHEAD_SIZE); res->data= (buff + PAGE_HEADER_SIZE); res->dir= res->data + res->length; + res->offset= 0; /* Store poistion to the first row */ int2store(res->dir, PAGE_HEADER_SIZE); - res->offset= 0; DBUG_ASSERT(length <= res->length); } else @@ -852,20 +896,22 @@ static my_bool get_head_or_tail_page(MARIA_HA *info, buff, block_size, block_size, 0))) DBUG_RETURN(1); DBUG_ASSERT((res->buff[PAGE_TYPE_OFFSET] & PAGE_TYPE_MASK) == page_type); - if (!(dir= find_free_position(buff, block_size, &res->offset, + if (!(dir= find_free_position(res->buff, block_size, &res->offset, &res->length, &res->empty_space))) + goto crashed; + + if (res->length < length) { - if (res->length < length) + if (res->empty_space + res->length < length) { - if (res->empty_space + res->length < length) - { - compact_page(res->buff, block_size, res->offset); - /* All empty space are now after current position */ - res->length= res->empty_space= uint2korr(dir+2); - } - if (res->length < length) - goto crashed; /* Wrong bitmap information */ + compact_page(res->buff, block_size, res->offset, 1); + /* All empty space are now after current position */ + dir= (res->buff + block_size - DIR_ENTRY_SIZE * res->offset - + PAGE_SUFFIX_SIZE); + res->length= res->empty_space= uint2korr(dir+2); } + if (res->length < length) + goto crashed; /* Wrong bitmap information */ } res->dir= dir; res->data= res->buff + uint2korr(dir); @@ -1041,8 +1087,9 @@ static void store_extent_info(byte *to, block < end_block; block++) { /* The following is only false for marker blocks */ - if (likely(block->used)) + if (likely(block->used & BLOCKUSED_USED)) { + DBUG_ASSERT(block->page_count != 0); int5store(to, block->page); int2store(to + 5, block->page_count); to+= ROW_EXTENT_SIZE; @@ -1053,7 +1100,7 @@ static void store_extent_info(byte *to, } } } - copy_length= (count -1) * ROW_EXTENT_SIZE; + copy_length= (count - 1) * ROW_EXTENT_SIZE; /* In some unlikely cases we have allocated to many blocks. Clear this data. @@ -1063,6 +1110,18 @@ static void store_extent_info(byte *to, /* Write a record to a (set of) pages + + SYNOPSIS + write_block_record() + info Maria handler + record Record we should write + row Statistics about record (calculated by calc_record_size()) + map_blocks On which pages the record should be stored + row_pos Position on head page where to put head part of record + + RETURN + 0 ok + 1 error */ static my_bool write_block_record(MARIA_HA *info, const byte *record, @@ -1090,8 +1149,8 @@ static my_bool write_block_record(MARIA_HA *info, const byte *record, head_block= bitmap_blocks->block; block_size= share->block_size; - info->cur_row.lastpos= ma_recordpos(head_block->page, row_pos->offset); page_buff= row_pos->buff; + /* Position on head page where we should store the head part */ data= row_pos->data; end_of_data= data + row_pos->length; @@ -1344,7 +1403,7 @@ static my_bool write_block_record(MARIA_HA *info, const byte *record, if (tmp_data_used) /* non blob data overflows */ { - MARIA_BITMAP_BLOCK *cur_block, *end_block; + MARIA_BITMAP_BLOCK *cur_block, *end_block, *last_head_block; MARIA_BITMAP_BLOCK *head_tail_block= 0; ulong length; ulong data_length= (tmp_data - info->rec_buff); @@ -1362,8 +1421,9 @@ static my_bool write_block_record(MARIA_HA *info, const byte *record, - Bitmap code allocated a tail page we don't need. - The last full page allocated needs to be changed to a tail page - (Because we put more data than we thought on the head page) - + (Because we where able to put more data on the head page than + the bitmap allocation assumed) + The reserved pages in bitmap_blocks for the main page has one of the following allocations: - Full pages, with following blocks: @@ -1376,8 +1436,13 @@ static my_bool write_block_record(MARIA_HA *info, const byte *record, cur_block= head_block + 1; end_block= head_block + head_block->sub_blocks; + /* + Loop until we have find a block bigger than we need or + we find the the empty page block. + */ while (data_length >= (length= (cur_block->page_count * - FULL_PAGE_SIZE(block_size)))) + FULL_PAGE_SIZE(block_size))) && + cur_block->page_count) { #ifdef SANITY_CHECK if ((cur_block == end_block) || (cur_block->used & BLOCKUSED_BIT)) @@ -1386,10 +1451,16 @@ static my_bool write_block_record(MARIA_HA *info, const byte *record, data_length-= length; (cur_block++)->used= BLOCKUSED_USED; } + last_head_block= cur_block; if (data_length) { + if (cur_block->page_count == 0) + { + /* Skip empty filler block */ + cur_block++; + } #ifdef SANITY_CHECK - if ((cur_block == end_block)) + if ((cur_block >= end_block)) goto crashed; #endif if (cur_block->used & BLOCKUSED_TAIL) @@ -1413,6 +1484,11 @@ static my_bool write_block_record(MARIA_HA *info, const byte *record, cur_block is a full block, followed by an empty and optional tail block. Change cur_block to a tail block or split it into full blocks and tail blocks. + + TODO: + If there is enough space on the following tail block, use + this instead of creating a new tail block. + */ DBUG_ASSERT(cur_block[1].page_count == 0); if (cur_block->page_count == 1) @@ -1426,11 +1502,11 @@ static my_bool write_block_record(MARIA_HA *info, const byte *record, DBUG_ASSERT(data_length < length - FULL_PAGE_SIZE(block_size)); DBUG_PRINT("info", ("Splitting blocks into full and tail")); cur_block[1].page= (cur_block->page + cur_block->page_count - 1); - cur_block[1].page_count= 1; - cur_block[1].used= 1; + cur_block[1].page_count= 1; /* Avoid DBUG_ASSERT */ + cur_block[1].used= BLOCKUSED_USED | BLOCKUSED_TAIL; cur_block->page_count--; - cur_block->used= BLOCKUSED_USED | BLOCKUSED_TAIL; - head_tail_block= cur_block + 1; + cur_block->used= BLOCKUSED_USED; + last_head_block= head_tail_block= cur_block+1; } if (end_block[-1].used & BLOCKUSED_TAIL) bitmap_blocks->tail_page_skipped= 1; @@ -1446,7 +1522,7 @@ static my_bool write_block_record(MARIA_HA *info, const byte *record, } /* - Write all extents into page or tmp_buff + Write all extents into page or tmp_data Note that we still don't have a correct position for the tail of the non-blob fields. @@ -1461,17 +1537,32 @@ static my_bool write_block_record(MARIA_HA *info, const byte *record, byte *extent_data; length= (uint) (data_length % FULL_PAGE_SIZE(block_size)); - if (write_tail(info, head_tail_block, data + data_length - length, + if (write_tail(info, head_tail_block, + info->rec_buff + data_length - length, length)) goto disk_err; tmp_data-= length; /* Remove the tail */ /* Store the tail position for the non-blob fields */ if (head_tail_block == head_block + 1) + { + /* + We had a head block + tail block, which means that the + tail block is the first extent + */ extent_data= row_extents_first_part; + } else + { + /* + We have a head block + some full blocks + tail block + last_head_block is pointing after the last used extent + for the head block. + */ extent_data= row_extents_second_part + - ((head_tail_block - head_block) - 2) * ROW_EXTENT_SIZE; + ((last_head_block - head_block) - 2) * ROW_EXTENT_SIZE; + } + DBUG_ASSERT(uint2korr(extent_data+5) & TAIL_BIT); int5store(extent_data, head_tail_block->page); int2store(extent_data + 5, head_tail_block->page_count); } @@ -1492,9 +1583,12 @@ static my_bool write_block_record(MARIA_HA *info, const byte *record, if (tmp_data_used) { - /* Write data stored in info->rec_buff to pages */ + /* + Write data stored in info->rec_buff to pages + This is the char/varchar data that didn't fit into the head page. + */ DBUG_ASSERT(bitmap_blocks->count != 0); - if (write_full_pages(info, bitmap_blocks->block + 1, info->rec_buff, + if (write_full_pages(info, head_block + 1, info->rec_buff, (ulong) (tmp_data - info->rec_buff))) goto disk_err; } @@ -1567,6 +1661,7 @@ MARIA_RECORD_POS _ma_write_init_block_record(MARIA_HA *info, if (write_block_record(info, record, &info->cur_row, blocks, &row_pos)) DBUG_RETURN(HA_OFFSET_ERROR); /* Error reading bitmap */ DBUG_PRINT("exit", ("Rowid: %lu", (ulong) info->cur_row.lastpos)); + info->s->state.split++; DBUG_RETURN(info->cur_row.lastpos); } @@ -1695,7 +1790,7 @@ my_bool _ma_update_block_record(MARIA_HA *info, MARIA_RECORD_POS record_pos, empty= empty_pos_after_row(dir) - (offset + length); if (new_row->total_length > length + empty) { - compact_page(buff, info->s->block_size, rownr); + compact_page(buff, info->s->block_size, rownr, 1); org_empty_size= 0; length= uint2korr(dir + 2); } @@ -1730,7 +1825,7 @@ my_bool _ma_update_block_record(MARIA_HA *info, MARIA_RECORD_POS record_pos, (new_row->total_length <= head_length && org_empty_size + head_length >= new_row->total_length))) { - compact_page(buff, info->s->block_size, rownr); + compact_page(buff, info->s->block_size, rownr, 1); org_empty_size= 0; head_length= uint2korr(dir + 2); } @@ -1792,9 +1887,8 @@ static my_bool delete_head_or_tail(MARIA_HA *info, number_of_records= (uint) ((uchar *) buff)[DIR_ENTRY_OFFSET]; #ifdef SANITY_CHECKS if (record_number >= number_of_records || - record_number > MAX_ROWS_PER_PAGE || record_number > ((block_size - LSN_SIZE - PAGE_TYPE_SIZE - 1 - - PAGE_SUFFIX_SIZE) / (DIR_ENTRY_SIZE + MIN_TAIL_SIZE))) + PAGE_SUFFIX_SIZE) / DIR_ENTRY_SIZE)) { DBUG_PRINT("error", ("record_number: %u number_of_records: %u", record_number, number_of_records)); @@ -1889,6 +1983,7 @@ my_bool _ma_delete_block_record(MARIA_HA *info) 1) || delete_tails(info, info->cur_row.tail_positions)) DBUG_RETURN(1); + info->s->state.split--; DBUG_RETURN(_ma_bitmap_free_full_pages(info, info->cur_row.extents, info->cur_row.extents_count)); } @@ -1924,9 +2019,8 @@ static byte *get_record_position(byte *buff, uint block_size, #ifdef SANITY_CHECKS if (record_number >= number_of_records || - record_number > MAX_ROWS_PER_PAGE || record_number > ((block_size - PAGE_HEADER_SIZE - PAGE_SUFFIX_SIZE) / - (DIR_ENTRY_SIZE + MIN_TAIL_SIZE))) + DIR_ENTRY_SIZE)) { DBUG_PRINT("error", ("Wrong row number: record_number: %u number_of_records: %u", @@ -2013,6 +2107,7 @@ static byte *read_next_extent(MARIA_HA *info, MARIA_EXTENT_CURSOR *extent, extent->extent+= ROW_EXTENT_SIZE; extent->page= uint5korr(extent->extent); page_count= uint2korr(extent->extent+ROW_EXTENT_PAGE_SIZE); + DBUG_ASSERT(page_count != 0); extent->tail= page_count & TAIL_BIT; extent->page_count= (page_count & ~TAIL_BIT); extent->first_extent= 0; @@ -2279,9 +2374,8 @@ int _ma_read_block_record2(MARIA_HA *info, byte *record, enum en_fieldtype type= (enum en_fieldtype) rec->type; byte *field_pos= record + rec->offset; /* First check if field is present in record */ - if (record[rec->null_pos] & rec->null_bit) - continue; - else if (info->cur_row.empty_bits[rec->empty_pos] & rec->empty_bit) + if ((record[rec->null_pos] & rec->null_bit) || + (info->cur_row.empty_bits[rec->empty_pos] & rec->empty_bit)) { if (type == FIELD_SKIP_ENDSPACE) bfill(record + rec->offset, rec->length, ' '); @@ -2405,19 +2499,24 @@ int _ma_read_block_record2(MARIA_HA *info, byte *record, if (extent.page_count) goto err; if (extent.extent_count > 1) - if (check_if_zero(extent.extent, + if (check_if_zero(extent.extent + ROW_EXTENT_SIZE, (extent.extent_count-1) * ROW_EXTENT_SIZE)) goto err; } else { DBUG_PRINT("info", ("Row read")); - if (data != end_of_data && (uint) (end_of_data - start_of_data) >= + /* + data should normally point to end_of_date. The only exception is if + the row is very short in which case we allocated 'min_row_length' data + for allowing the row to expand. + */ + if (data != end_of_data && (uint) (end_of_data - start_of_data) > info->s->base.min_row_length) goto err; } - info->update|= HA_STATE_AKTIV; /* We have a aktive record */ + info->update|= HA_STATE_AKTIV; /* We have an active record */ DBUG_RETURN(0); err: @@ -2447,6 +2546,7 @@ int _ma_read_block_record(MARIA_HA *info, byte *record, DBUG_ENTER("_ma_read_block_record"); DBUG_PRINT("enter", ("rowid: %lu", (long) record_pos)); + info->cur_row.lastpos= record_pos; page= ma_recordpos_to_page(record_pos) * block_size; offset= ma_recordpos_to_offset(record_pos); @@ -2515,13 +2615,18 @@ my_bool _ma_cmp_block_unique(MARIA_HA *info, MARIA_UNIQUEDEF *def, IMPLEMENTATION We allocate one buffer for the current bitmap and one buffer for the current page + + RETURN + 0 ok + 1 error (couldn't allocate memory or disk error) */ my_bool _ma_scan_init_block_record(MARIA_HA *info) { byte *ptr; + DBUG_ENTER("_ma_scan_init_block_record"); if (!(ptr= (byte *) my_malloc(info->s->block_size * 2, MYF(MY_WME)))) - return (1); + DBUG_RETURN(1); info->scan.bitmap_buff= ptr; info->scan.page_buff= ptr + info->s->block_size; info->scan.bitmap_end= info->scan.bitmap_buff + info->s->bitmap.total_size; @@ -2534,7 +2639,7 @@ my_bool _ma_scan_init_block_record(MARIA_HA *info) We have to flush bitmap as we will read the bitmap from the page cache while scanning rows */ - return _ma_flush_bitmap(info->s); + DBUG_RETURN(_ma_flush_bitmap(info->s)); } @@ -2542,8 +2647,10 @@ my_bool _ma_scan_init_block_record(MARIA_HA *info) void _ma_scan_end_block_record(MARIA_HA *info) { - my_free(info->scan.bitmap_buff, MYF(0)); + DBUG_ENTER("_ma_scan_end_block_record"); + my_free(info->scan.bitmap_buff, MYF(MY_ALLOW_ZERO_PTR)); info->scan.bitmap_buff= 0; + DBUG_VOID_RETURN; } diff --git a/storage/maria/ma_blockrec.h b/storage/maria/ma_blockrec.h index ec99dbfcae2..1e18b823ef8 100644 --- a/storage/maria/ma_blockrec.h +++ b/storage/maria/ma_blockrec.h @@ -71,7 +71,7 @@ enum en_page_type { UNALLOCATED_PAGE, HEAD_PAGE, TAIL_PAGE, BLOB_PAGE, MAX_PAGE_ /* Fixed part of Max possible header size; See table in ma_blockrec.c */ #define MAX_FIXED_HEADER_SIZE (FLAG_SIZE + 3 + ROW_EXTENT_SIZE + 3) #define TRANS_MAX_FIXED_HEADER_SIZE (MAX_FIXED_HEADER_SIZE + \ - FLAG_SIZE + TRANSID_SIZE + VERPTR_SIZE + \ + TRANSID_SIZE + VERPTR_SIZE + \ TRANSID_SIZE) /* We use 1 byte in record header to store number of directory entries */ @@ -91,6 +91,7 @@ enum en_page_type { UNALLOCATED_PAGE, HEAD_PAGE, TAIL_PAGE, BLOB_PAGE, MAX_PAGE_ static inline MARIA_RECORD_POS ma_recordpos(ulonglong page, uint offset) { + DBUG_ASSERT(offset <= 255); return (MARIA_RECORD_POS) ((page << 8) | offset); } diff --git a/storage/maria/ma_check.c b/storage/maria/ma_check.c index ccce19de994..673a8081f6c 100644 --- a/storage/maria/ma_check.c +++ b/storage/maria/ma_check.c @@ -136,7 +136,9 @@ int maria_chk_status(HA_CHECK *param, register MARIA_HA *info) return 0; } - /* Check delete links */ +/* + Check delete links in row data +*/ int maria_chk_del(HA_CHECK *param, register MARIA_HA *info, uint test_flag) { @@ -147,6 +149,10 @@ int maria_chk_del(HA_CHECK *param, register MARIA_HA *info, uint test_flag) DBUG_ENTER("maria_chk_del"); LINT_INIT(old_link); + + if (info->s->data_file_type == BLOCK_RECORD) + DBUG_RETURN(0); /* No delete links here */ + param->record_checksum=0; delete_link_length=((info->s->options & HA_OPTION_PACK_RECORD) ? 20 : info->s->rec_reflength+1); @@ -2145,6 +2151,7 @@ err: restore_data_file_type(share); share->state.changed|= (STATE_NOT_OPTIMIZED_KEYS | STATE_NOT_SORTED_PAGES | STATE_NOT_ANALYZED); + share->state.changed&= ~STATE_NOT_OPTIMIZED_ROWS; DBUG_RETURN(got_error); } @@ -2912,7 +2919,8 @@ err: } else if (key_map == share->state.key_map) share->state.changed&= ~STATE_NOT_OPTIMIZED_KEYS; - share->state.changed|=STATE_NOT_SORTED_PAGES; + share->state.changed|= STATE_NOT_SORTED_PAGES; + share->state.changed&= ~STATE_NOT_OPTIMIZED_ROWS; my_free(sort_param.rec_buff, MYF(MY_ALLOW_ZERO_PTR)); my_free(sort_param.record,MYF(MY_ALLOW_ZERO_PTR)); @@ -3432,7 +3440,8 @@ err: } else if (key_map == share->state.key_map) share->state.changed&= ~STATE_NOT_OPTIMIZED_KEYS; - share->state.changed|=STATE_NOT_SORTED_PAGES; + share->state.changed|= STATE_NOT_SORTED_PAGES; + share->state.changed&= ~STATE_NOT_OPTIMIZED_ROWS; pthread_cond_destroy (&sort_info.cond); pthread_mutex_destroy(&sort_info.mutex); diff --git a/storage/maria/ma_checksum.c b/storage/maria/ma_checksum.c index 1b0f683fe63..859feb9cf0b 100644 --- a/storage/maria/ma_checksum.c +++ b/storage/maria/ma_checksum.c @@ -20,14 +20,13 @@ ha_checksum _ma_checksum(MARIA_HA *info, const byte *record) { - uint i; ha_checksum crc=0; - MARIA_COLUMNDEF *rec=info->s->rec; + MARIA_COLUMNDEF *rec= info->s->rec, *rec_end= rec+ info->s->base.fields; if (info->s->base.null_bytes) crc= my_checksum(crc, record, info->s->base.null_bytes); - for (i=info->s->base.fields ; i-- ; ) + for ( ; rec != rec_end ; rec++) { const byte *pos= record + rec->offset; ulong length; diff --git a/storage/maria/ma_create.c b/storage/maria/ma_create.c index 59f5f7d1a4d..966a8644e67 100644 --- a/storage/maria/ma_create.c +++ b/storage/maria/ma_create.c @@ -44,7 +44,7 @@ int maria_create(const char *name, enum data_file_type record_type, { register uint i,j; File dfile,file; - int errpos,save_errno, create_mode= O_RDWR | O_TRUNC; + int errpos,save_errno, create_mode= O_RDWR | O_TRUNC, res; myf create_flag; uint length,max_key_length,packed,pack_bytes,pointer,real_length_diff, key_length,info_length,key_segs,options,min_key_length_skip, @@ -156,35 +156,41 @@ int maria_create(const char *name, enum data_file_type record_type, type == FIELD_SKIP_ENDSPACE) { max_field_lengths+= rec->length > 255 ? 2 : 1; - min_pack_length++; + if (record_type != BLOCK_RECORD) + min_pack_length++; packed++; } else if (type == FIELD_VARCHAR) { varchar_length+= rec->length-1; /* Used for min_pack_length */ pack_reclength++; - min_pack_length++; + if (record_type != BLOCK_RECORD) + min_pack_length++; max_field_lengths++; packed++; + rec->fill_length= 1; /* We must test for 257 as length includes pack-length */ if (test(rec->length >= 257)) { long_varchar_count++; max_field_lengths++; + rec->fill_length= 2; } } - else if (type != FIELD_SKIP_ZERO) + else if (type == FIELD_SKIP_ZERO) + packed++; + else { - min_pack_length+=rec->length; + if (record_type != BLOCK_RECORD || !rec->null_bit) + min_pack_length+= rec->length; rec->empty_pos= 0; rec->empty_bit= 0; } - else - packed++; } else /* FIELD_NORMAL */ { - min_pack_length+=rec->length; + if (record_type != BLOCK_RECORD || !rec->null_bit) + min_pack_length+= rec->length; if (!rec->null_bit) { share.base.fixed_not_null_fields++; @@ -204,6 +210,8 @@ int maria_create(const char *name, enum data_file_type record_type, if (rec->type == (int) FIELD_SKIP_ZERO && rec->length == 1) { rec->type=(int) FIELD_NORMAL; + rec->empty_pos= 0; + rec->empty_bit= 0; packed--; min_pack_length++; break; @@ -365,7 +373,7 @@ int maria_create(const char *name, enum data_file_type record_type, keyseg->type != HA_KEYTYPE_VARBINARY2) { my_errno=HA_WRONG_CREATE_OPTION; - goto err; + goto err_no_lock; } } keydef->keysegs+=sp_segs; @@ -374,7 +382,7 @@ int maria_create(const char *name, enum data_file_type record_type, min_key_length_skip+=SPLEN*2*SPDIMS; #else my_errno= HA_ERR_UNSUPPORTED; - goto err; + goto err_no_lock; #endif /*HAVE_SPATIAL*/ } else if (keydef->flag & HA_FULLTEXT) @@ -390,7 +398,7 @@ int maria_create(const char *name, enum data_file_type record_type, keyseg->type != HA_KEYTYPE_VARTEXT2) { my_errno=HA_WRONG_CREATE_OPTION; - goto err; + goto err_no_lock; } if (!(keyseg->flag & HA_BLOB_PART) && (keyseg->type == HA_KEYTYPE_VARTEXT1 || @@ -515,7 +523,7 @@ int maria_create(const char *name, enum data_file_type record_type, if (keydef->keysegs > HA_MAX_KEY_SEG) { my_errno=HA_WRONG_CREATE_OPTION; - goto err; + goto err_no_lock; } /* key_segs may be 0 in the case when we only want to be able to @@ -526,10 +534,10 @@ int maria_create(const char *name, enum data_file_type record_type, key_segs) share.state.rec_per_key_part[key_segs-1]=1L; length+=key_length; - if (length >= min(HA_MAX_KEY_BUFF, MARIA_MAX_KEY_LENGTH)) + if (length >= HA_MAX_KEY_BUFF) { my_errno=HA_WRONG_CREATE_OPTION; - goto err; + goto err_no_lock; } keydef->block_length= maria_block_size; keydef->keylength= (uint16) key_length; @@ -573,7 +581,7 @@ int maria_create(const char *name, enum data_file_type record_type, "indexes and/or unique constraints.", MYF(0), name + dirname_length(name)); my_errno= HA_WRONG_CREATE_OPTION; - goto err; + goto err_no_lock; } bmove(share.state.header.file_version,(byte*) maria_file_magic,4); @@ -646,11 +654,16 @@ int maria_create(const char *name, enum data_file_type record_type, share.base.max_data_file_length= (my_off_t) ci->data_file_length; } - share.base.min_block_length= - (share.base.pack_reclength+3 < MARIA_EXTEND_BLOCK_LENGTH && - ! share.base.blobs) ? - max(share.base.pack_reclength,MARIA_MIN_BLOCK_LENGTH) : - MARIA_EXTEND_BLOCK_LENGTH; + if (record_type == BLOCK_RECORD) + share.base.min_block_length= share.base.min_row_length; + else + { + share.base.min_block_length= + (share.base.pack_reclength+3 < MARIA_EXTEND_BLOCK_LENGTH && + ! share.base.blobs) ? + max(share.base.pack_reclength,MARIA_MIN_BLOCK_LENGTH) : + MARIA_EXTEND_BLOCK_LENGTH; + } if (! (flags & HA_DONT_TOUCH_DATA)) share.state.create_time= (long) time((time_t*) 0); @@ -869,17 +882,24 @@ int maria_create(const char *name, enum data_file_type record_type, if (record_type == BLOCK_RECORD) { /* Store columns in a more efficent order */ - MARIA_COLUMNDEF **tmp, **pos; - if (!(tmp= (MARIA_COLUMNDEF**) my_malloc(share.base.fields * + MARIA_COLUMNDEF **col_order, **pos; + if (!(col_order= (MARIA_COLUMNDEF**) my_malloc(share.base.fields * sizeof(MARIA_COLUMNDEF*), MYF(MY_WME)))) goto err; - for (rec= recinfo, pos= tmp ; rec != rec_end ; rec++, pos++) + for (rec= recinfo, pos= col_order ; rec != rec_end ; rec++, pos++) *pos= rec; - qsort(tmp, share.base.fields, sizeof(*tmp), (qsort_cmp) compare_columns); + qsort(col_order, share.base.fields, sizeof(*col_order), + (qsort_cmp) compare_columns); for (i=0 ; i < share.base.fields ; i++) - if (_ma_recinfo_write(file, tmp[i])) + { + if (_ma_recinfo_write(file, col_order[i])) + { + my_free((gptr) col_order, MYF(0)); goto err; + } + } + my_free((gptr) col_order, MYF(0)); } else { @@ -918,8 +938,9 @@ int maria_create(const char *name, enum data_file_type record_type, } errpos=0; pthread_mutex_unlock(&THR_LOCK_maria); + res= 0; if (my_close(file,MYF(0))) - goto err; + res= my_errno; /* RECOVERYTODO Write a log record describing the CREATE operation (just the file @@ -934,10 +955,12 @@ int maria_create(const char *name, enum data_file_type record_type, will clean up the frm, so we needn't write anything to the log. */ my_free((char*) rec_per_key_part,MYF(0)); - DBUG_RETURN(0); + DBUG_RETURN(res); err: pthread_mutex_unlock(&THR_LOCK_maria); + +err_no_lock: save_errno=my_errno; switch (errpos) { case 3: diff --git a/storage/maria/ma_delete.c b/storage/maria/ma_delete.c index 9198989dcb7..5dfa7b81595 100644 --- a/storage/maria/ma_delete.c +++ b/storage/maria/ma_delete.c @@ -105,7 +105,8 @@ int maria_delete(MARIA_HA *info,const byte *record) info->update= HA_STATE_CHANGED+HA_STATE_DELETED+HA_STATE_ROW_CHANGED; info->state->records--; - + share->state.changed|= STATE_NOT_OPTIMIZED_ROWS; + mi_sizestore(lastpos, info->cur_row.lastpos); VOID(_ma_writeinfo(info,WRITEINFO_UPDATE_KEYFILE)); allow_break(); /* Allow SIGHUP & SIGINT */ diff --git a/storage/maria/ma_delete_all.c b/storage/maria/ma_delete_all.c index 616147c1067..04dcb29b39d 100644 --- a/storage/maria/ma_delete_all.c +++ b/storage/maria/ma_delete_all.c @@ -37,6 +37,7 @@ int maria_delete_all_rows(MARIA_HA *info) goto err; info->state->records=info->state->del=state->split=0; + state->changed= 0; /* File is optimized */ state->dellink = HA_OFFSET_ERROR; state->sortkey= (ushort) ~0; info->state->key_file_length=share->base.keystart; diff --git a/storage/maria/ma_extra.c b/storage/maria/ma_extra.c index 90e79362442..cd5e87d1d2b 100644 --- a/storage/maria/ma_extra.c +++ b/storage/maria/ma_extra.c @@ -45,6 +45,8 @@ int maria_extra(MARIA_HA *info, enum ha_extra_function function, int error=0; ulong cache_size; MARIA_SHARE *share=info->s; + my_bool block_records= share->data_file_type == BLOCK_RECORD; + DBUG_ENTER("maria_extra"); DBUG_PRINT("enter",("function: %d",(int) function)); @@ -65,6 +67,9 @@ int maria_extra(MARIA_HA *info, enum ha_extra_function function, HA_STATE_PREV_FOUND); break; case HA_EXTRA_CACHE: + if (block_records) + break; /* Not supported */ + if (info->lock_type == F_UNLCK && (share->options & HA_OPTION_PACK_RECORD)) { @@ -128,9 +133,11 @@ int maria_extra(MARIA_HA *info, enum ha_extra_function function, case HA_EXTRA_WRITE_CACHE: if (info->lock_type == F_UNLCK) { - error=1; /* Not possibly if not locked */ + error=1; /* Not possibly if not locked */ break; } + if (block_records) + break; /* Not supported */ cache_size= (extra_arg ? *(ulong*) extra_arg : my_default_record_cache_size); @@ -354,6 +361,8 @@ int maria_extra(MARIA_HA *info, enum ha_extra_function function, break; case HA_EXTRA_MMAP: #ifdef HAVE_MMAP + if (block_records) + break; /* Not supported */ pthread_mutex_lock(&share->intern_lock); if (!share->file_map) { @@ -390,9 +399,11 @@ int maria_extra(MARIA_HA *info, enum ha_extra_function function, /* - Start/Stop Inserting Duplicates Into a Table, WL#1648. - */ -static void maria_extra_keyflag(MARIA_HA *info, enum ha_extra_function function) + Start/Stop Inserting Duplicates Into a Table, WL#1648. +*/ + +static void maria_extra_keyflag(MARIA_HA *info, + enum ha_extra_function function) { uint idx; diff --git a/storage/maria/ma_info.c b/storage/maria/ma_info.c index 397cd2465d4..7b8cea20297 100644 --- a/storage/maria/ma_info.c +++ b/storage/maria/ma_info.c @@ -77,14 +77,14 @@ int maria_status(MARIA_HA *info, register MARIA_INFO *x, uint flag) x->create_time=share->state.create_time; x->reflength= maria_get_pointer_length(share->base.max_data_file_length, maria_data_pointer_size); - x->record_offset= ((share->options & - (HA_OPTION_PACK_RECORD | HA_OPTION_COMPRESS_RECORD)) ? - 0L : share->base.pack_reclength); + x->record_offset= (info->s->data_file_type == STATIC_RECORD ? + share->base.pack_reclength: 0); x->sortkey= -1; /* No clustering */ x->rec_per_key = share->state.rec_per_key_part; x->key_map = share->state.key_map; x->data_file_name = share->data_file_name; x->index_file_name = share->index_file_name; + x->data_file_type = share->data_file_type; } if ((flag & HA_STATUS_TIME) && !my_fstat(info->dfile,&state,MYF(0))) x->update_time=state.st_mtime; diff --git a/storage/maria/ma_search.c b/storage/maria/ma_search.c index d8738ae4639..1d712b8d7d9 100644 --- a/storage/maria/ma_search.c +++ b/storage/maria/ma_search.c @@ -189,7 +189,7 @@ int _ma_bin_search(MARIA_HA *info, register MARIA_KEYDEF *keyinfo, byte *page, totlength=keyinfo->keylength+(nod_flag=_ma_test_if_nod(page)); start=0; mid=1; save_end=end=(int) ((maria_getint(page)-2-nod_flag)/totlength-1); - DBUG_PRINT("test",("maria_getint: %d end: %d",maria_getint(page),end)); + DBUG_PRINT("test",("page_length: %d end: %d",maria_getint(page),end)); page+=2+nod_flag; while (start != end) @@ -971,12 +971,12 @@ uint _ma_get_binary_pack_key(register MARIA_KEYDEF *keyinfo, uint nod_flag, { /* Get length of dynamic length key part */ if (from == from_end) { from=page; from_end=page_end; } - if ((length= (*key++ = *from++)) == 255) + if ((length= (uint) (uchar) (*key++ = *from++)) == 255) { if (from == from_end) { from=page; from_end=page_end; } - length= (uint) ((*key++ = *from++)) << 8; + length= ((uint) (uchar) ((*key++ = *from++))) << 8; if (from == from_end) { from=page; from_end=page_end; } - length+= (uint) ((*key++ = *from++)); + length+= (uint) (uchar) ((*key++ = *from++)); } } else @@ -988,6 +988,7 @@ uint _ma_get_binary_pack_key(register MARIA_KEYDEF *keyinfo, uint nod_flag, length-=tmp; from=page; from_end=page_end; } + DBUG_ASSERT((int) length >= 0); DBUG_PRINT("info",("key: 0x%lx from: 0x%lx length: %u", (long) key, (long) from, length)); memmove((byte*) key, (byte*) from, (size_t) length); diff --git a/storage/maria/ma_test2.c b/storage/maria/ma_test2.c index 46b8c710d4a..552d78de7b0 100644 --- a/storage/maria/ma_test2.c +++ b/storage/maria/ma_test2.c @@ -815,8 +815,7 @@ int main(int argc, char *argv[]) { ulong blob_length,pos; uchar *ptr; - longget(blob_length,read_record+blob_pos+4); - ptr=(uchar*) blob_length; + memcpy_fixed(&ptr, read_record+blob_pos+4, sizeof(ptr)); longget(blob_length,read_record+blob_pos); for (pos=0 ; pos < blob_length ; pos++) { diff --git a/storage/maria/ma_write.c b/storage/maria/ma_write.c index 3f1ca59a00a..64fae81d944 100644 --- a/storage/maria/ma_write.c +++ b/storage/maria/ma_write.c @@ -1065,6 +1065,7 @@ void maria_flush_bulk_insert(MARIA_HA *info, uint inx) void maria_end_bulk_insert(MARIA_HA *info) { + DBUG_ENTER("maria_end_bulk_insert"); if (info->bulk_insert) { uint i; @@ -1078,4 +1079,5 @@ void maria_end_bulk_insert(MARIA_HA *info) my_free((void *)info->bulk_insert, MYF(0)); info->bulk_insert=0; } + DBUG_VOID_RETURN; } diff --git a/storage/maria/maria_def.h b/storage/maria/maria_def.h index b174df2fa3e..0b171323886 100644 --- a/storage/maria/maria_def.h +++ b/storage/maria/maria_def.h @@ -442,6 +442,7 @@ struct st_maria_info #define STATE_NOT_ANALYZED 8 #define STATE_NOT_OPTIMIZED_KEYS 16 #define STATE_NOT_SORTED_PAGES 32 +#define STATE_NOT_OPTIMIZED_ROWS 64 /* options to maria_read_cache */ diff --git a/storage/myisam/mi_create.c b/storage/myisam/mi_create.c index a675325f0a1..b98c7af19cc 100644 --- a/storage/myisam/mi_create.c +++ b/storage/myisam/mi_create.c @@ -46,7 +46,7 @@ int mi_create(const char *name,uint keys,MI_KEYDEF *keydefs, key_length,info_length,key_segs,options,min_key_length_skip, base_pos,long_varchar_count,varchar_length, max_key_block_length,unique_key_parts,fulltext_keys,offset; - uint aligned_key_start, block_length; + uint aligned_key_start, block_length, res; ulong reclength, real_reclength,min_pack_length; char filename[FN_REFLEN],linkname[FN_REFLEN], *linkname_ptr; ulong pack_reclength; @@ -271,7 +271,7 @@ int mi_create(const char *name,uint keys,MI_KEYDEF *keydefs, keyseg->type != HA_KEYTYPE_VARBINARY2) { my_errno=HA_WRONG_CREATE_OPTION; - goto err; + goto err_no_lock; } } keydef->keysegs+=sp_segs; @@ -280,7 +280,7 @@ int mi_create(const char *name,uint keys,MI_KEYDEF *keydefs, min_key_length_skip+=SPLEN*2*SPDIMS; #else my_errno= HA_ERR_UNSUPPORTED; - goto err; + goto err_no_lock; #endif /*HAVE_SPATIAL*/ } else if (keydef->flag & HA_FULLTEXT) @@ -296,7 +296,7 @@ int mi_create(const char *name,uint keys,MI_KEYDEF *keydefs, keyseg->type != HA_KEYTYPE_VARTEXT2) { my_errno=HA_WRONG_CREATE_OPTION; - goto err; + goto err_no_lock; } if (!(keyseg->flag & HA_BLOB_PART) && (keyseg->type == HA_KEYTYPE_VARTEXT1 || @@ -421,7 +421,7 @@ int mi_create(const char *name,uint keys,MI_KEYDEF *keydefs, if (keydef->keysegs > HA_MAX_KEY_SEG) { my_errno=HA_WRONG_CREATE_OPTION; - goto err; + goto err_no_lock; } /* key_segs may be 0 in the case when we only want to be able to @@ -446,7 +446,7 @@ int mi_create(const char *name,uint keys,MI_KEYDEF *keydefs, length >= HA_MAX_KEY_BUFF) { my_errno=HA_WRONG_CREATE_OPTION; - goto err; + goto err_no_lock; } set_if_bigger(max_key_block_length,keydef->block_length); keydef->keylength= (uint16) key_length; @@ -493,7 +493,7 @@ int mi_create(const char *name,uint keys,MI_KEYDEF *keydefs, "indexes and/or unique constraints.", MYF(0), name + dirname_length(name)); my_errno= HA_WRONG_CREATE_OPTION; - goto err; + goto err_no_lock; } bmove(share.state.header.file_version,(byte*) myisam_file_magic,4); @@ -815,13 +815,16 @@ int mi_create(const char *name,uint keys,MI_KEYDEF *keydefs, } errpos=0; pthread_mutex_unlock(&THR_LOCK_myisam); + res= 0; if (my_close(file,MYF(0))) - goto err; + res= my_errno; my_free((char*) rec_per_key_part,MYF(0)); - DBUG_RETURN(0); + DBUG_RETURN(res); err: pthread_mutex_unlock(&THR_LOCK_myisam); +err_no_lock: + save_errno=my_errno; switch (errpos) { case 3: diff --git a/storage/myisam/mi_test2.c b/storage/myisam/mi_test2.c index 9cf47e1555b..f68c9746cc2 100644 --- a/storage/myisam/mi_test2.c +++ b/storage/myisam/mi_test2.c @@ -778,8 +778,7 @@ int main(int argc, char *argv[]) { ulong blob_length,pos; uchar *ptr; - longget(blob_length,read_record+blob_pos+4); - ptr=(uchar*) blob_length; + memcpy_fixed(&ptr, read_record+blob_pos+4, sizeof(ptr)); longget(blob_length,read_record+blob_pos); for (pos=0 ; pos < blob_length ; pos++) { -- cgit v1.2.1 From 1a392bf63e95383be9a5e5276dfb1d85cd4cb6a2 Mon Sep 17 00:00:00 2001 From: unknown Date: Fri, 6 Apr 2007 04:58:05 +0300 Subject: After merge fixes Fixed test for row based replication mysql-test/mysql-test-run.pl: After merge fix mysql-test/r/maria.result: Fixed test for row based replication mysql-test/t/maria.test: Fixed test for row based replication storage/maria/ha_maria.cc: After merge fix storage/maria/ma_blockrec.c: Better to clear whole page, as 'length' may be bigger than what we need. storage/maria/ma_loghandler.c: Fix compiler warning Removed access to not initialized memory storage/maria/ma_open.c: Remove wrong (not needed) test --- mysql-test/mysql-test-run.pl | 13 ------------- mysql-test/r/maria.result | 3 +-- mysql-test/t/maria.test | 8 ++++++++ storage/maria/ha_maria.cc | 2 +- storage/maria/ma_blockrec.c | 6 +++--- storage/maria/ma_loghandler.c | 14 ++++++-------- storage/maria/ma_open.c | 7 ------- 7 files changed, 19 insertions(+), 34 deletions(-) diff --git a/mysql-test/mysql-test-run.pl b/mysql-test/mysql-test-run.pl index b5137e06fb8..7aec7ba0c77 100755 --- a/mysql-test/mysql-test-run.pl +++ b/mysql-test/mysql-test-run.pl @@ -1238,19 +1238,6 @@ sub command_line_setup () { $path_ndb_testrun_log= "$opt_vardir/log/ndb_testrun.log"; $path_snapshot= "$opt_tmpdir/snapshot_$opt_master_myport/"; - - if ( $opt_valgrind and $opt_debug ) - { - # When both --valgrind and --debug is selected, send - # all output to the trace file, making it possible to - # see the exact location where valgrind complains - foreach my $mysqld (@{$master}, @{$slave}) - { - my $sidx= $mysqld->{idx} ? "$mysqld->{idx}" : ""; - $mysqld->{path_myerr}= - "$opt_vardir/log/" . $mysqld->{type} . "$sidx.trace"; - } - } } # diff --git a/mysql-test/r/maria.result b/mysql-test/r/maria.result index 4a708cde307..d52f243e186 100644 --- a/mysql-test/r/maria.result +++ b/mysql-test/r/maria.result @@ -712,7 +712,7 @@ t1 1 a 4 d A 4 NULL NULL YES BTREE delete from t1; analyze table t1; Table Op Msg_type Msg_text -test.t1 analyze status Table is already up to date +test.t1 analyze status OK show index from t1; Table Non_unique Key_name Seq_in_index Column_name Collation Cardinality Sub_part Packed Null Index_type Comment t1 1 a 1 a A 0 NULL NULL YES BTREE @@ -1780,4 +1780,3 @@ CHECK TABLE t1; Table Op Msg_type Msg_text test.t1 check status OK DROP TABLE t1; -set global storage_engine=MyISAM; diff --git a/mysql-test/t/maria.test b/mysql-test/t/maria.test index 898268c9bb0..f368efcdfa2 100644 --- a/mysql-test/t/maria.test +++ b/mysql-test/t/maria.test @@ -677,6 +677,10 @@ insert into t1 values ('bce','def2', NULL, 'quux'); analyze table t1; show index from t1; delete from t1; +# This will give you different messages depending on if we are using +# row base or stmt based replication as stmt base replication will use +# truncate and row based will delete things row by row. +--replace_result "Table is already up to date" "OK" analyze table t1; show index from t1; @@ -1086,4 +1090,8 @@ DROP TABLE t1; # End of 5.2 tests +--disable_result_log +--disable_query_log eval set global storage_engine=$default; +--enable_result_log +--enable_query_log diff --git a/storage/maria/ha_maria.cc b/storage/maria/ha_maria.cc index 1522bf80a28..ba8bb654a7d 100644 --- a/storage/maria/ha_maria.cc +++ b/storage/maria/ha_maria.cc @@ -1042,7 +1042,7 @@ int ha_maria::repair(THD *thd, HA_CHECK ¶m, bool do_optimize) /* QQ: Until we have repair for block format, lie that it succeded */ if (file->s->data_file_type == BLOCK_RECORD) { - if (optimize) + if (do_optimize) DBUG_RETURN(analyze(thd, (HA_CHECK_OPT*) 0)); DBUG_RETURN(HA_ADMIN_OK); } diff --git a/storage/maria/ma_blockrec.c b/storage/maria/ma_blockrec.c index 1db372cdffc..29c06cba7e5 100644 --- a/storage/maria/ma_blockrec.c +++ b/storage/maria/ma_blockrec.c @@ -858,6 +858,7 @@ static my_bool get_head_or_tail_page(MARIA_HA *info, { uint block_size; DBUG_ENTER("get_head_or_tail_page"); + DBUG_PRINT("enter", ("length: %u", length)); block_size= info->s->block_size; if (block->org_bitmap_value == 0) /* Empty block */ @@ -871,9 +872,8 @@ static my_bool get_head_or_tail_page(MARIA_HA *info, The rest of the code does not assume the block is zeroed above PAGE_OVERHEAD_SIZE */ - bzero(buff+ PAGE_HEADER_SIZE + length, - block_size - length - PAGE_HEADER_SIZE - DIR_ENTRY_SIZE - - PAGE_SUFFIX_SIZE); + bzero(buff+ PAGE_HEADER_SIZE, block_size - PAGE_HEADER_SIZE); + buff[PAGE_TYPE_OFFSET]= (byte) page_type; buff[DIR_ENTRY_OFFSET]= 1; res->buff= buff; diff --git a/storage/maria/ma_loghandler.c b/storage/maria/ma_loghandler.c index d8a821cb538..d83ac86aa9b 100644 --- a/storage/maria/ma_loghandler.c +++ b/storage/maria/ma_loghandler.c @@ -5073,7 +5073,7 @@ translog_size_t translog_read_record(LSN lsn, static void translog_force_current_buffer_to_finish() { - TRANSLOG_ADDRESS new_buff_begunning; + TRANSLOG_ADDRESS new_buff_beginning; uint16 old_buffer_no= log_descriptor.bc.buffer_no; uint16 new_buffer_no= (old_buffer_no + 1) % TRANSLOG_BUFFERS_NO; struct st_translog_buffer *new_buffer= (log_descriptor.buffers + @@ -5086,7 +5086,6 @@ static void translog_force_current_buffer_to_finish() DBUG_PRINT("enter", ("Buffer #%u 0x%lx " "Buffer addr: (%lu,0x%lx) " "Page addr: (%lu,0x%lx) " - "New Buff: (%lu,0x%lx) " "size: %lu (%lu) Pg: %u left: %u", (uint) log_descriptor.bc.buffer_no, (ulong) log_descriptor.bc.buffer, @@ -5095,16 +5094,15 @@ static void translog_force_current_buffer_to_finish() (ulong) LSN_FILE_NO(log_descriptor.horizon), (ulong) (LSN_OFFSET(log_descriptor.horizon) - log_descriptor.bc.current_page_fill), - (ulong) LSN_FILE_NO(new_buff_begunning), - (ulong) LSN_OFFSET(new_buff_begunning), (ulong) log_descriptor.bc.buffer->size, (ulong) (log_descriptor.bc.ptr -log_descriptor.bc. buffer->buffer), (uint) log_descriptor.bc.current_page_fill, (uint) left)); - new_buff_begunning= log_descriptor.bc.buffer->offset; - new_buff_begunning+= log_descriptor.bc.buffer->size; /* increase offset */ + LINT_INIT(current_page_fill); + new_buff_beginning= log_descriptor.bc.buffer->offset; + new_buff_beginning+= log_descriptor.bc.buffer->size; /* increase offset */ DBUG_ASSERT(log_descriptor.bc.ptr !=NULL); DBUG_ASSERT(LSN_FILE_NO(log_descriptor.horizon) == @@ -5120,7 +5118,7 @@ static void translog_force_current_buffer_to_finish() DBUG_PRINT("info", ("left: %u", (uint) left)); /* decrease offset */ - new_buff_begunning-= log_descriptor.bc.current_page_fill; + new_buff_beginning-= log_descriptor.bc.current_page_fill; current_page_fill= log_descriptor.bc.current_page_fill; bzero(log_descriptor.bc.ptr, left); @@ -5145,7 +5143,7 @@ static void translog_force_current_buffer_to_finish() write_counter= log_descriptor.bc.write_counter; previous_offset= log_descriptor.bc.previous_offset; translog_start_buffer(new_buffer, &log_descriptor.bc, new_buffer_no); - log_descriptor.bc.buffer->offset= new_buff_begunning; + log_descriptor.bc.buffer->offset= new_buff_beginning; log_descriptor.bc.write_counter= write_counter; log_descriptor.bc.previous_offset= previous_offset; diff --git a/storage/maria/ma_open.c b/storage/maria/ma_open.c index 86376404f74..8ef7a7375c1 100644 --- a/storage/maria/ma_open.c +++ b/storage/maria/ma_open.c @@ -309,13 +309,6 @@ MARIA_HA *maria_open(const char *name, int mode, uint open_flags) for (j=0 ; j < share->keyinfo[i].keysegs; j++,pos++) { disk_pos=_ma_keyseg_read(disk_pos, pos); - if (pos->flag & HA_BLOB_PART && - ! (share->options & (HA_OPTION_COMPRESS_RECORD | - HA_OPTION_PACK_RECORD))) - { - my_errno= HA_ERR_CRASHED; - goto err; - } if (pos->type == HA_KEYTYPE_TEXT || pos->type == HA_KEYTYPE_VARTEXT1 || pos->type == HA_KEYTYPE_VARTEXT2) -- cgit v1.2.1 From 1f57759ac86587d16259544b321cc8965a984926 Mon Sep 17 00:00:00 2001 From: unknown Date: Wed, 11 Apr 2007 09:40:54 +0300 Subject: Fixed buffer size in the control file test --- storage/maria/unittest/ma_control_file-t.c | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/storage/maria/unittest/ma_control_file-t.c b/storage/maria/unittest/ma_control_file-t.c index d13fc25232f..71a1157f1ba 100644 --- a/storage/maria/unittest/ma_control_file-t.c +++ b/storage/maria/unittest/ma_control_file-t.c @@ -264,7 +264,7 @@ static int test_binary_content() future change/breakage. */ - char buffer[20]; + char buffer[23]; RET_ERR_UNLESS((fd= my_open(file_name, O_BINARY | O_RDWR, MYF(MY_WME))) >= 0); -- cgit v1.2.1 From 09cb2fe629fb2ed08408f28b842c04ddef210917 Mon Sep 17 00:00:00 2001 From: unknown Date: Thu, 12 Apr 2007 11:35:05 +0300 Subject: Postreview changes. include/my_sys.h: The comments added. include/pagecache.h: Functions names changed. storage/maria/unittest/mf_pagecache_consist.c: Functions names changed. storage/maria/unittest/mf_pagecache_single.c: Functions names changed. --- include/my_sys.h | 7 +- include/pagecache.h | 32 ++--- mysys/mf_pagecache.c | 185 ++++++++++++++------------ storage/maria/unittest/mf_pagecache_consist.c | 4 +- storage/maria/unittest/mf_pagecache_single.c | 20 +-- 5 files changed, 134 insertions(+), 114 deletions(-) diff --git a/include/my_sys.h b/include/my_sys.h index c8362455970..52953a86a0d 100644 --- a/include/my_sys.h +++ b/include/my_sys.h @@ -280,7 +280,12 @@ enum cache_type enum flush_type { - FLUSH_KEEP, FLUSH_RELEASE, FLUSH_IGNORE_CHANGED, FLUSH_FORCE_WRITE + FLUSH_KEEP, /* flush block and keep it in the cache */ + FLUSH_RELEASE, /* flush block and remove it from the cache */ + FLUSH_IGNORE_CHANGED, /* remove block from the cache */ + /* as my_disable_flush_pagecache_blocks is always 0, it is + strictly equivalent to FLUSH_KEEP */ + FLUSH_FORCE_WRITE }; typedef struct st_record_cache /* Used when cacheing records */ diff --git a/include/pagecache.h b/include/pagecache.h index 32e23fb54ea..8e9a373ff16 100644 --- a/include/pagecache.h +++ b/include/pagecache.h @@ -194,30 +194,30 @@ extern my_bool pagecache_write(PAGECACHE *pagecache, enum pagecache_page_pin pin, enum pagecache_write_mode write_mode, PAGECACHE_PAGE_LINK *link); -extern void pagecache_unlock_page(PAGECACHE *pagecache, - PAGECACHE_FILE *file, - pgcache_page_no_t pageno, - enum pagecache_page_lock lock, - enum pagecache_page_pin pin, - LSN first_REDO_LSN_for_page); extern void pagecache_unlock(PAGECACHE *pagecache, - PAGECACHE_PAGE_LINK *link, + PAGECACHE_FILE *file, + pgcache_page_no_t pageno, enum pagecache_page_lock lock, enum pagecache_page_pin pin, LSN first_REDO_LSN_for_page); -extern void pagecache_unpin_page(PAGECACHE *pagecache, - PAGECACHE_FILE *file, - pgcache_page_no_t pageno); +extern void pagecache_unlock_by_link(PAGECACHE *pagecache, + PAGECACHE_PAGE_LINK *link, + enum pagecache_page_lock lock, + enum pagecache_page_pin pin, + LSN first_REDO_LSN_for_page); extern void pagecache_unpin(PAGECACHE *pagecache, - PAGECACHE_PAGE_LINK *link); + PAGECACHE_FILE *file, + pgcache_page_no_t pageno); +extern void pagecache_unpin_by_link(PAGECACHE *pagecache, + PAGECACHE_PAGE_LINK *link); extern int flush_pagecache_blocks(PAGECACHE *keycache, PAGECACHE_FILE *file, enum flush_type type); -extern my_bool pagecache_delete_page(PAGECACHE *pagecache, - PAGECACHE_FILE *file, - pgcache_page_no_t pageno, - enum pagecache_page_lock lock, - my_bool flush); +extern my_bool pagecache_delete(PAGECACHE *pagecache, + PAGECACHE_FILE *file, + pgcache_page_no_t pageno, + enum pagecache_page_lock lock, + my_bool flush); extern void end_pagecache(PAGECACHE *keycache, my_bool cleanup); extern my_bool pagecache_collect_changed_blocks_with_lsn(PAGECACHE *pagecache, LEX_STRING *str, diff --git a/mysys/mf_pagecache.c b/mysys/mf_pagecache.c index 8fa651bdf36..f9ac33a30a1 100755 --- a/mysys/mf_pagecache.c +++ b/mysys/mf_pagecache.c @@ -773,7 +773,7 @@ int init_pagecache(PAGECACHE *pagecache, my_size_t use_mem, } pagecache->blocks= pagecache->disk_blocks > 0 ? pagecache->disk_blocks : 0; - DBUG_RETURN((uint) pagecache->blocks); + DBUG_RETURN((int) pagecache->disk_blocks); err: error= my_errno; @@ -1578,7 +1578,10 @@ static PAGECACHE_HASH_LINK *get_present_hash_link(PAGECACHE *pagecache, /* Register the request for the page */ hash_link->requests++; } - + /* + As soon as the caller will release the page cache's lock, "hash_link" + will be potentially obsolete (unusable) information. + */ DBUG_RETURN(hash_link); } @@ -2103,7 +2106,8 @@ static void info_change_lock(PAGECACHE_BLOCK_LINK *block, my_bool wl) PAGECACHE_LOCK_INFO *info= (PAGECACHE_LOCK_INFO *)info_find((PAGECACHE_PIN_INFO *)block->lock_list, my_thread_var); - DBUG_ASSERT(info != 0 && info->write_lock != wl); + DBUG_ASSERT(info != 0); + DBUG_ASSERT(info->write_lock != wl); info->write_lock= wl; } #else @@ -2228,16 +2232,23 @@ static my_bool make_lock_and_pin(PAGECACHE *pagecache, enum pagecache_page_pin pin) { DBUG_ENTER("make_lock_and_pin"); - DBUG_PRINT("enter", ("block: 0x%lx (%u), wrlock: %c pins: %u, lock %s, pin: %s", - (ulong)block, BLOCK_NUMBER(pagecache, block), - ((block->status & BLOCK_WRLOCK)?'Y':'N'), - block->pins, - page_cache_page_lock_str[lock], - page_cache_page_pin_str[pin])); - BLOCK_INFO(block); + + DBUG_PRINT("enter", ("block: 0x%lx", (ulong)block)); +#ifndef DBUG_OFF + if (block) + { + DBUG_PRINT("enter", ("block %u wrlock: %c pins: %u, lock %s, pin: %s", + BLOCK_NUMBER(pagecache, block), + ((block->status & BLOCK_WRLOCK)?'Y':'N'), + block->pins, + page_cache_page_lock_str[lock], + page_cache_page_pin_str[pin])); + BLOCK_INFO(block); + } +#endif #ifdef PAGECACHE_DEBUG - DBUG_ASSERT(info_check_pin(block, pin) == 0 && - info_check_lock(block, lock, pin) == 0); + DBUG_ASSERT(info_check_pin(block, pin) == 0); + DBUG_ASSERT(info_check_lock(block, lock, pin) == 0); #endif switch (lock) { @@ -2291,7 +2302,10 @@ static my_bool make_lock_and_pin(PAGECACHE *pagecache, DBUG_ASSERT(0); /* Never should happened */ } - BLOCK_INFO(block); +#ifndef DBUG_OFF + if (block) + BLOCK_INFO(block); +#endif DBUG_RETURN(0); retry: DBUG_PRINT("INFO", ("Retry block 0x%lx", (ulong)block)); @@ -2413,7 +2427,7 @@ static void read_block(PAGECACHE *pagecache, Unlock/unpin page and put LSN stamp if it need SYNOPSIS - pagecache_unlock_page() + pagecache_unlock() pagecache pointer to a page cache data structure file handler for the file for the block of data to be read pageno number of the block of data in the file @@ -2434,24 +2448,24 @@ static void read_block(PAGECACHE *pagecache, */ -void pagecache_unlock_page(PAGECACHE *pagecache, - PAGECACHE_FILE *file, - pgcache_page_no_t pageno, - enum pagecache_page_lock lock, - enum pagecache_page_pin pin, - LSN first_REDO_LSN_for_page) +void pagecache_unlock(PAGECACHE *pagecache, + PAGECACHE_FILE *file, + pgcache_page_no_t pageno, + enum pagecache_page_lock lock, + enum pagecache_page_pin pin, + LSN first_REDO_LSN_for_page) { PAGECACHE_BLOCK_LINK *block; int page_st; - DBUG_ENTER("pagecache_unlock_page"); + DBUG_ENTER("pagecache_unlock"); DBUG_PRINT("enter", ("fd: %u page: %lu l%s p%s", (uint) file->file, (ulong) pageno, page_cache_page_lock_str[lock], page_cache_page_pin_str[pin])); /* we do not allow any lock/pin increasing here */ - DBUG_ASSERT(pin != PAGECACHE_PIN && - lock != PAGECACHE_LOCK_READ && - lock != PAGECACHE_LOCK_WRITE); + DBUG_ASSERT(pin != PAGECACHE_PIN); + DBUG_ASSERT(lock != PAGECACHE_LOCK_READ); + DBUG_ASSERT(lock != PAGECACHE_LOCK_WRITE); pagecache_pthread_mutex_lock(&pagecache->cache_lock); /* @@ -2461,15 +2475,15 @@ void pagecache_unlock_page(PAGECACHE *pagecache, DBUG_ASSERT(pagecache->can_be_used); inc_counter_for_resize_op(pagecache); - /* See NOTE for pagecache_unlock_page about registering requests */ + /* See NOTE for pagecache_unlock about registering requests */ block= find_block(pagecache, file, pageno, 0, 0, test(pin == PAGECACHE_PIN_LEFT_UNPINNED), &page_st); BLOCK_INFO(block); DBUG_ASSERT(block != 0 && page_st == PAGE_READ); if (first_REDO_LSN_for_page) { - DBUG_ASSERT(lock == PAGECACHE_LOCK_WRITE_UNLOCK && - pin == PAGECACHE_UNPIN); + DBUG_ASSERT(lock == PAGECACHE_LOCK_WRITE_UNLOCK); + DBUG_ASSERT(pin == PAGECACHE_UNPIN); set_if_bigger(block->rec_lsn, first_REDO_LSN_for_page); } @@ -2490,7 +2504,7 @@ void pagecache_unlock_page(PAGECACHE *pagecache, /* Link the block into the LRU chain if it's the last submitted request for the block and block will not be pinned. - See NOTE for pagecache_unlock_page about registering requests. + See NOTE for pagecache_unlock about registering requests. */ if (pin != PAGECACHE_PIN_LEFT_PINNED) unreg_request(pagecache, block, 1); @@ -2507,19 +2521,19 @@ void pagecache_unlock_page(PAGECACHE *pagecache, Unpin page SYNOPSIS - pagecache_unpin_page() + pagecache_unpin() pagecache pointer to a page cache data structure file handler for the file for the block of data to be read pageno number of the block of data in the file */ -void pagecache_unpin_page(PAGECACHE *pagecache, - PAGECACHE_FILE *file, - pgcache_page_no_t pageno) +void pagecache_unpin(PAGECACHE *pagecache, + PAGECACHE_FILE *file, + pgcache_page_no_t pageno) { PAGECACHE_BLOCK_LINK *block; int page_st; - DBUG_ENTER("pagecache_unpin_page"); + DBUG_ENTER("pagecache_unpin"); DBUG_PRINT("enter", ("fd: %u page: %lu", (uint) file->file, (ulong) pageno)); pagecache_pthread_mutex_lock(&pagecache->cache_lock); @@ -2530,9 +2544,10 @@ void pagecache_unpin_page(PAGECACHE *pagecache, DBUG_ASSERT(pagecache->can_be_used); inc_counter_for_resize_op(pagecache); - /* See NOTE for pagecache_unlock_page about registering requests */ + /* See NOTE for pagecache_unlock about registering requests */ block= find_block(pagecache, file, pageno, 0, 0, 0, &page_st); - DBUG_ASSERT(block != 0 && page_st == PAGE_READ); + DBUG_ASSERT(block != 0); + DBUG_ASSERT(page_st == PAGE_READ); #ifndef DBUG_OFF if ( @@ -2558,7 +2573,7 @@ void pagecache_unpin_page(PAGECACHE *pagecache, /* Link the block into the LRU chain if it's the last submitted request for the block and block will not be pinned. - See NOTE for pagecache_unlock_page about registering requests + See NOTE for pagecache_unlock about registering requests */ unreg_request(pagecache, block, 1); @@ -2575,7 +2590,7 @@ void pagecache_unpin_page(PAGECACHE *pagecache, (uses direct block/page pointer) SYNOPSIS - pagecache_unlock() + pagecache_unlock_by_link() pagecache pointer to a page cache data structure link direct link to page (returned by read or write) lock lock change @@ -2583,14 +2598,14 @@ void pagecache_unpin_page(PAGECACHE *pagecache, first_REDO_LSN_for_page do not set it if it is zero */ -void pagecache_unlock(PAGECACHE *pagecache, - PAGECACHE_PAGE_LINK *link, - enum pagecache_page_lock lock, - enum pagecache_page_pin pin, - LSN first_REDO_LSN_for_page) +void pagecache_unlock_by_link(PAGECACHE *pagecache, + PAGECACHE_PAGE_LINK *link, + enum pagecache_page_lock lock, + enum pagecache_page_pin pin, + LSN first_REDO_LSN_for_page) { PAGECACHE_BLOCK_LINK *block= (PAGECACHE_BLOCK_LINK *)link; - DBUG_ENTER("pagecache_unlock"); + DBUG_ENTER("pagecache_unlock_by_link"); DBUG_PRINT("enter", ("block: 0x%lx fd: %u page: %lu l%s p%s", (ulong) block, (uint) block->hash_link->file.file, @@ -2601,10 +2616,10 @@ void pagecache_unlock(PAGECACHE *pagecache, We do not allow any lock/pin increasing here and page can't be unpinned because we use direct link. */ - DBUG_ASSERT(pin != PAGECACHE_PIN && - pin != PAGECACHE_PIN_LEFT_UNPINNED && - lock != PAGECACHE_LOCK_READ && - lock != PAGECACHE_LOCK_WRITE); + DBUG_ASSERT(pin != PAGECACHE_PIN); + DBUG_ASSERT(pin != PAGECACHE_PIN_LEFT_UNPINNED); + DBUG_ASSERT(lock != PAGECACHE_LOCK_READ); + DBUG_ASSERT(lock != PAGECACHE_LOCK_WRITE); if (pin == PAGECACHE_PIN_LEFT_UNPINNED && lock == PAGECACHE_LOCK_READ_UNLOCK) { @@ -2626,16 +2641,16 @@ void pagecache_unlock(PAGECACHE *pagecache, pagecache_pthread_mutex_lock(&pagecache->cache_lock); /* - As soon as we keep lock cache can be used, and we have lock bacause want - aunlock. + As soon as we keep lock cache can be used, and we have lock because want + unlock. */ DBUG_ASSERT(pagecache->can_be_used); inc_counter_for_resize_op(pagecache); if (first_REDO_LSN_for_page) { - DBUG_ASSERT(lock == PAGECACHE_LOCK_WRITE_UNLOCK && - pin == PAGECACHE_UNPIN); + DBUG_ASSERT(lock == PAGECACHE_LOCK_WRITE_UNLOCK); + DBUG_ASSERT(pin == PAGECACHE_UNPIN); set_if_bigger(block->rec_lsn, first_REDO_LSN_for_page); } @@ -2646,7 +2661,7 @@ void pagecache_unlock(PAGECACHE *pagecache, #ifndef DBUG_OFF ) { - DBUG_ASSERT(0); /* should not happend */ + DBUG_ASSERT(0); /* should not happened */ } #else ; @@ -2656,7 +2671,7 @@ void pagecache_unlock(PAGECACHE *pagecache, /* Link the block into the LRU chain if it's the last submitted request for the block and block will not be pinned. - See NOTE for pagecache_unlock_page about registering requests. + See NOTE for pagecache_unlock about registering requests. */ if (pin != PAGECACHE_PIN_LEFT_PINNED) unreg_request(pagecache, block, 1); @@ -2674,16 +2689,16 @@ void pagecache_unlock(PAGECACHE *pagecache, (uses direct block/page pointer) SYNOPSIS - pagecache_unpin_page() + pagecache_unpin_by_link() pagecache pointer to a page cache data structure link direct link to page (returned by read or write) */ -void pagecache_unpin(PAGECACHE *pagecache, - PAGECACHE_PAGE_LINK *link) +void pagecache_unpin_by_link(PAGECACHE *pagecache, + PAGECACHE_PAGE_LINK *link) { PAGECACHE_BLOCK_LINK *block= (PAGECACHE_BLOCK_LINK *)link; - DBUG_ENTER("pagecache_unpin"); + DBUG_ENTER("pagecache_unpin_by_link"); DBUG_PRINT("enter", ("block: 0x%lx fd: %u page: %lu", (ulong) block, (uint) block->hash_link->file.file, @@ -2691,8 +2706,8 @@ void pagecache_unpin(PAGECACHE *pagecache, pagecache_pthread_mutex_lock(&pagecache->cache_lock); /* - As soon as we keep lock cache can be used, and we have lock bacause want - aunlock. + As soon as we keep lock cache can be used, and we have lock because want + unlock. */ DBUG_ASSERT(pagecache->can_be_used); @@ -2712,7 +2727,7 @@ void pagecache_unpin(PAGECACHE *pagecache, #ifndef DBUG_OFF ) { - DBUG_ASSERT(0); /* should not happend */ + DBUG_ASSERT(0); /* should not happened */ } #else ; @@ -2722,7 +2737,7 @@ void pagecache_unpin(PAGECACHE *pagecache, /* Link the block into the LRU chain if it's the last submitted request for the block and block will not be pinned. - See NOTE for pagecache_unlock_page about registering requests. + See NOTE for pagecache_unlock about registering requests. */ unreg_request(pagecache, block, 1); @@ -2751,9 +2766,9 @@ void pagecache_unpin(PAGECACHE *pagecache, validator_data pointer to the data need by the validator RETURN VALUE - Returns address from where the data is placed if sucessful, 0 - otherwise. + Returns address from where the data is placed if successful, 0 - otherwise. - Pin will be choosen according to lock parameter (see lock_to_pin) + Pin will be chosen according to lock parameter (see lock_to_pin) */ static enum pagecache_page_pin lock_to_pin[]= { @@ -2811,7 +2826,7 @@ restart: inc_counter_for_resize_op(pagecache); pagecache->global_cache_r_requests++; - /* See NOTE for pagecache_unlock_page about registering requests. */ + /* See NOTE for pagecache_unlock about registering requests. */ block= find_block(pagecache, file, pageno, level, test(lock == PAGECACHE_LOCK_WRITE), test((pin == PAGECACHE_PIN_LEFT_UNPINNED) || @@ -2820,7 +2835,7 @@ restart: DBUG_ASSERT(block->type == PAGECACHE_EMPTY_PAGE || block->type == type); block->type= type; - if (block->status != BLOCK_ERROR && page_st != PAGE_READ) + if (((block->status & BLOCK_ERROR) == 0) && (page_st != PAGE_READ)) { DBUG_PRINT("info", ("read block 0x%lx", (ulong)block)); /* The requested page is to be read into the block buffer */ @@ -2859,7 +2874,7 @@ restart: /* Link the block into the LRU chain if it's the last submitted request for the block and block will not be pinned. - See NOTE for pagecache_unlock_page about registering requests. + See NOTE for pagecache_unlock about registering requests. */ if (pin == PAGECACHE_PIN_LEFT_UNPINNED || pin == PAGECACHE_UNPIN) unreg_request(pagecache, block, 1); @@ -2891,7 +2906,7 @@ no_key_cache: /* Key cache is not used */ Delete page from the buffer SYNOPSIS - pagecache_delete_page() + pagecache_delete() pagecache pointer to a page cache data structure file handler for the file for the block of data to be read pageno number of the block of data in the file @@ -2906,15 +2921,15 @@ no_key_cache: /* Key cache is not used */ lock can be only PAGECACHE_LOCK_LEFT_WRITELOCKED (page was write locked before) or PAGECACHE_LOCK_WRITE (delete will write lock page before delete) */ -my_bool pagecache_delete_page(PAGECACHE *pagecache, - PAGECACHE_FILE *file, - pgcache_page_no_t pageno, - enum pagecache_page_lock lock, - my_bool flush) +my_bool pagecache_delete(PAGECACHE *pagecache, + PAGECACHE_FILE *file, + pgcache_page_no_t pageno, + enum pagecache_page_lock lock, + my_bool flush) { int error= 0; enum pagecache_page_pin pin= lock_to_pin[lock]; - DBUG_ENTER("pagecache_delete_page"); + DBUG_ENTER("pagecache_delete"); DBUG_PRINT("enter", ("fd: %u page: %lu l%s p%s", (uint) file->file, (ulong) pageno, page_cache_page_lock_str[lock], @@ -2945,7 +2960,7 @@ restart: DBUG_RETURN(0); } block= link->block; - /* See NOTE for pagecache_unlock_page about registering requests. */ + /* See NOTE for pagecache_unlock about registering requests. */ if (pin == PAGECACHE_PIN) reg_requests(pagecache, block, 1); DBUG_ASSERT(block != 0); @@ -3002,7 +3017,7 @@ restart: PAGECACHE_UNPIN); DBUG_ASSERT(link->requests > 0); link->requests--; - /* See NOTE for pagecache_unlock_page about registering requests. */ + /* See NOTE for pagecache_unlock about registering requests. */ free_block(pagecache, block); err: @@ -3025,7 +3040,7 @@ end: file handler for the file to write data to pageno number of the block of data in the file level determines the weight of the data - buff buffer to where the data must be placed + buff buffer with the data type type of the page lock lock change pin pin page @@ -3049,7 +3064,7 @@ static struct write_lock_change write_lock_change_table[]= {1, PAGECACHE_LOCK_WRITE, PAGECACHE_LOCK_WRITE_UNLOCK} /*PAGECACHE_LOCK_LEFT_UNLOCKED*/, - {0, /*unsupported*/ + {0, /*unsupported (we can't write having the block read locked) */ PAGECACHE_LOCK_LEFT_UNLOCKED, PAGECACHE_LOCK_LEFT_UNLOCKED} /*PAGECACHE_LOCK_LEFT_READLOCKED*/, {0, PAGECACHE_LOCK_LEFT_WRITELOCKED, 0} /*PAGECACHE_LOCK_LEFT_WRITELOCKED*/, @@ -3057,7 +3072,7 @@ static struct write_lock_change write_lock_change_table[]= PAGECACHE_LOCK_WRITE, PAGECACHE_LOCK_WRITE_TO_READ} /*PAGECACHE_LOCK_READ*/, {0, PAGECACHE_LOCK_WRITE, 0} /*PAGECACHE_LOCK_WRITE*/, - {0, /*unsupported*/ + {0, /*unsupported (we can't write having the block read locked) */ PAGECACHE_LOCK_LEFT_UNLOCKED, PAGECACHE_LOCK_LEFT_UNLOCKED} /*PAGECACHE_LOCK_READ_UNLOCK*/, {1, @@ -3065,7 +3080,7 @@ static struct write_lock_change write_lock_change_table[]= PAGECACHE_LOCK_WRITE_UNLOCK } /*PAGECACHE_LOCK_WRITE_UNLOCK*/, {1, PAGECACHE_LOCK_LEFT_WRITELOCKED, - PAGECACHE_LOCK_WRITE_TO_READ}/*PAGECACHE_LOCK_WRITE_TO_READ*/ + PAGECACHE_LOCK_WRITE_TO_READ} /*PAGECACHE_LOCK_WRITE_TO_READ*/ }; /* description of how to change pin before and after write */ @@ -3109,8 +3124,8 @@ my_bool pagecache_write(PAGECACHE *pagecache, page_cache_page_lock_str[lock], page_cache_page_pin_str[pin], page_cache_page_write_mode_str[write_mode])); - DBUG_ASSERT(lock != PAGECACHE_LOCK_LEFT_READLOCKED && - lock != PAGECACHE_LOCK_READ_UNLOCK); + DBUG_ASSERT(lock != PAGECACHE_LOCK_LEFT_READLOCKED); + DBUG_ASSERT(lock != PAGECACHE_LOCK_READ_UNLOCK); if (!link) link= &fake_link; else @@ -3147,7 +3162,7 @@ restart: inc_counter_for_resize_op(pagecache); pagecache->global_cache_w_requests++; - /* See NOTE for pagecache_unlock_page about registering requests. */ + /* See NOTE for pagecache_unlock about registering requests. */ block= find_block(pagecache, file, pageno, level, test(write_mode != PAGECACHE_WRITE_DONE && lock != PAGECACHE_LOCK_LEFT_WRITELOCKED && @@ -3242,7 +3257,7 @@ restart: /* Unregister the request */ DBUG_ASSERT(block->hash_link->requests > 0); block->hash_link->requests--; - /* See NOTE for pagecache_unlock_page about registering requests. */ + /* See NOTE for pagecache_unlock about registering requests. */ if (pin == PAGECACHE_PIN_LEFT_UNPINNED || pin == PAGECACHE_UNPIN) unreg_request(pagecache, block, 1); else @@ -3851,8 +3866,8 @@ my_bool pagecache_collect_changed_blocks_with_lsn(PAGECACHE *pagecache, { if (block->type != PAGECACHE_LSN_PAGE) continue; /* no need to store it in the checkpoint record */ - DBUG_ASSERT((4 == sizeof(block->hash_link->file.file)) && - (4 == sizeof(block->hash_link->pageno))); + DBUG_ASSERT((4 == sizeof(block->hash_link->file.file))); + DBUG_ASSERT((4 == sizeof(block->hash_link->pageno))); int4store(ptr, block->hash_link->file.file); ptr+= 4; int4store(ptr, block->hash_link->pageno); diff --git a/storage/maria/unittest/mf_pagecache_consist.c b/storage/maria/unittest/mf_pagecache_consist.c index 8ea0094762c..e5f994bbdba 100755 --- a/storage/maria/unittest/mf_pagecache_consist.c +++ b/storage/maria/unittest/mf_pagecache_consist.c @@ -135,8 +135,8 @@ err: DBUG_PRINT("err", ("try to flush")); if (page_locked) { - pagecache_delete_page(&pagecache, &file1, page_no, - PAGECACHE_LOCK_LEFT_WRITELOCKED, 1); + pagecache_delete(&pagecache, &file1, page_no, + PAGECACHE_LOCK_LEFT_WRITELOCKED, 1); } else { diff --git a/storage/maria/unittest/mf_pagecache_single.c b/storage/maria/unittest/mf_pagecache_single.c index 91cceee618d..0b5208a60ff 100644 --- a/storage/maria/unittest/mf_pagecache_single.c +++ b/storage/maria/unittest/mf_pagecache_single.c @@ -231,12 +231,12 @@ int simple_pin_test() ok((res= test(test_file(file1, file1_name, PAGE_SIZE*2, PAGE_SIZE*2, simple_pin_test_file1))), "Simple pin page file with pin"); - pagecache_unlock_page(&pagecache, - &file1, - 0, - PAGECACHE_LOCK_READ_UNLOCK, - PAGECACHE_UNPIN, - 0); + pagecache_unlock(&pagecache, + &file1, + 0, + PAGECACHE_LOCK_READ_UNLOCK, + PAGECACHE_UNPIN, + 0); if (flush_pagecache_blocks(&pagecache, &file1, FLUSH_FORCE_WRITE)) { diag("Got error in flush_pagecache_blocks\n"); @@ -282,8 +282,8 @@ int simple_delete_forget_test() PAGECACHE_PIN_LEFT_UNPINNED, PAGECACHE_WRITE_DELAY, 0); - pagecache_delete_page(&pagecache, &file1, 0, - PAGECACHE_LOCK_WRITE, 0); + pagecache_delete(&pagecache, &file1, 0, + PAGECACHE_LOCK_WRITE, 0); flush_pagecache_blocks(&pagecache, &file1, FLUSH_FORCE_WRITE); ok((res= test(test_file(file1, file1_name, PAGE_SIZE, PAGE_SIZE, simple_delete_forget_test_file))), @@ -324,8 +324,8 @@ int simple_delete_flush_test() PAGECACHE_PIN_LEFT_PINNED, PAGECACHE_WRITE_DELAY, 0); - pagecache_delete_page(&pagecache, &file1, 0, - PAGECACHE_LOCK_LEFT_WRITELOCKED, 1); + pagecache_delete(&pagecache, &file1, 0, + PAGECACHE_LOCK_LEFT_WRITELOCKED, 1); flush_pagecache_blocks(&pagecache, &file1, FLUSH_FORCE_WRITE); ok((res= test(test_file(file1, file1_name, PAGE_SIZE, PAGE_SIZE, simple_delete_flush_test_file))), -- cgit v1.2.1 From bd65a4f56a694c61aa34c5ba1600d676625a85a6 Mon Sep 17 00:00:00 2001 From: unknown Date: Thu, 12 Apr 2007 12:05:30 +0300 Subject: Enabled ps_maria.test Fixed bug in field-is-zero detection Fixed bug in truncate file (datafile was not properly initialized) mysql-test/t/disabled.def: Enable ps_maria storage/maria/ma_bitmap.c: Added reset of bitmap (for truncate) storage/maria/ma_blockrec.c: Fixed bug in zero detection storage/maria/ma_blockrec.h: New prototype storage/maria/ma_create.c: Moved initialzation of datafile to separate function storage/maria/ma_delete_all.c: Added initializtion of data file storage/maria/maria_def.h: New prototype --- mysql-test/t/disabled.def | 1 - storage/maria/ma_bitmap.c | 14 ++++++++++++++ storage/maria/ma_blockrec.c | 2 +- storage/maria/ma_blockrec.h | 1 + storage/maria/ma_create.c | 35 ++++++++++++++++++++++------------- storage/maria/ma_delete_all.c | 4 ++++ storage/maria/maria_def.h | 1 + 7 files changed, 43 insertions(+), 15 deletions(-) diff --git a/mysql-test/t/disabled.def b/mysql-test/t/disabled.def index e0812f54b4f..f351b4079fa 100644 --- a/mysql-test/t/disabled.def +++ b/mysql-test/t/disabled.def @@ -39,4 +39,3 @@ mysql_upgrade : Bug#25074 mysql_upgrade gives inconsisten results plugin : Bug#25659 memory leak via "plugins" test rpl_ndb_dd_advance : Bug#25913 rpl_ndb_dd_advance fails randomly ndb_alter_table : Bug##25774 ndb_alter_table.test fails in DBUG_ASSERT() on Linux x64 -ps_maria : Until maria is fully functional diff --git a/storage/maria/ma_bitmap.c b/storage/maria/ma_bitmap.c index b32a2a11bfc..202f695e30c 100644 --- a/storage/maria/ma_bitmap.c +++ b/storage/maria/ma_bitmap.c @@ -191,6 +191,7 @@ my_bool _ma_bitmap_end(MARIA_SHARE *share) _ma_flush_bitmap(share); pthread_mutex_destroy(&share->bitmap.bitmap_lock); my_free((byte*) share->bitmap.map, MYF(MY_ALLOW_ZERO_PTR)); + share->bitmap.map= 0; return res; } @@ -216,6 +217,19 @@ my_bool _ma_flush_bitmap(MARIA_SHARE *share) } +void _ma_bitmap_delete_all(MARIA_SHARE *share) +{ + MARIA_FILE_BITMAP *bitmap= &share->bitmap; + if (bitmap->map) /* Not in create */ + { + bzero(bitmap->map, share->block_size); + bitmap->changed= 0; + bitmap->page= 0; + bitmap->used_size= bitmap->total_size; + } +} + + /* Return bitmap pattern for the smallest head block that can hold 'size' diff --git a/storage/maria/ma_blockrec.c b/storage/maria/ma_blockrec.c index 29c06cba7e5..c56522a9072 100644 --- a/storage/maria/ma_blockrec.c +++ b/storage/maria/ma_blockrec.c @@ -585,7 +585,7 @@ static void calc_record_size(MARIA_HA *info, const byte *record, *null_field_lengths= rec->length; break; case FIELD_SKIP_ZERO: /* Fixed length field */ - if (memcmp(record+ rec->null_pos, maria_zero_string, + if (memcmp(record+ rec->offset, maria_zero_string, rec->length) == 0) { row->empty_bits[rec->empty_pos] |= rec->empty_bit; diff --git a/storage/maria/ma_blockrec.h b/storage/maria/ma_blockrec.h index e54ce45114f..54145319b83 100644 --- a/storage/maria/ma_blockrec.h +++ b/storage/maria/ma_blockrec.h @@ -158,3 +158,4 @@ my_bool _ma_check_if_right_bitmap_type(MARIA_HA *info, enum en_page_type page_type, ulonglong page, uint *bitmap_pattern); +void _ma_bitmap_delete_all(MARIA_SHARE *share); diff --git a/storage/maria/ma_create.c b/storage/maria/ma_create.c index 69c18f54910..00bf949a43c 100644 --- a/storage/maria/ma_create.c +++ b/storage/maria/ma_create.c @@ -773,17 +773,8 @@ int maria_create(const char *name, enum data_file_type record_type, goto err; errpos=3; - if (record_type == BLOCK_RECORD) - { - /* Write one bitmap page */ - char buff[IO_SIZE]; - uint i; - bzero(buff, sizeof(buff)); - for (i= 0 ; i < maria_block_size ; i+= IO_SIZE) - if (my_write(dfile, (byte*) buff, sizeof(buff), MYF(MY_NABP))) - goto err; - share.state.state.data_file_length= maria_block_size; - } + if (_ma_initialize_data_file(dfile, &share)) + goto err; } DBUG_PRINT("info", ("write state info and base info")); if (_ma_state_info_write(file, &share.state, 2) || @@ -1030,7 +1021,7 @@ static inline int sign(longlong a) } -int compare_columns(MARIA_COLUMNDEF **a_ptr, MARIA_COLUMNDEF **b_ptr) +static int compare_columns(MARIA_COLUMNDEF **a_ptr, MARIA_COLUMNDEF **b_ptr) { MARIA_COLUMNDEF *a= *a_ptr, *b= *b_ptr; enum en_fieldtype a_type, b_type; @@ -1062,5 +1053,23 @@ int compare_columns(MARIA_COLUMNDEF **a_ptr, MARIA_COLUMNDEF **b_ptr) } +/* Initialize data file */ - +int _ma_initialize_data_file(File dfile, MARIA_SHARE *share) +{ + if (share->data_file_type == BLOCK_RECORD) + { + /* Write one bitmap page */ + byte buff[IO_SIZE]; + uint i; + bzero((char*) buff, sizeof(buff)); + if (my_seek(dfile, 0, SEEK_SET, 0)) + return 1; + for (i= 0 ; i < maria_block_size ; i+= IO_SIZE) + if (my_write(dfile, buff, sizeof(buff), MYF(MY_NABP))) + return 1; + share->state.state.data_file_length= maria_block_size; + _ma_bitmap_delete_all(share); + } + return 0; +} diff --git a/storage/maria/ma_delete_all.c b/storage/maria/ma_delete_all.c index 80022b1ae26..7880a692fc9 100644 --- a/storage/maria/ma_delete_all.c +++ b/storage/maria/ma_delete_all.c @@ -65,6 +65,10 @@ int maria_delete_all_rows(MARIA_HA *info) if (my_chsize(info->dfile, 0, 0, MYF(MY_WME)) || my_chsize(share->kfile, share->base.keystart, 0, MYF(MY_WME)) ) goto err; + + if (_ma_initialize_data_file(info->dfile, info->s)) + goto err; + /* RECOVERY TODO Consider updating ZeroDirtyPagesLSN here. It is not a necessity (it is one only in RENAME commands) but an optional diff --git a/storage/maria/maria_def.h b/storage/maria/maria_def.h index 8d4cf75f09d..553e8efb787 100644 --- a/storage/maria/maria_def.h +++ b/storage/maria/maria_def.h @@ -852,3 +852,4 @@ int _ma_sort_write_record(MARIA_SORT_PARAM *sort_param); int _ma_create_index_by_sort(MARIA_SORT_PARAM *info, my_bool no_messages, ulong); int _ma_sync_table_files(const MARIA_HA *info); +int _ma_initialize_data_file(File dfile, MARIA_SHARE *share); -- cgit v1.2.1 From 92e99ce4243b5ffdc069f1681136e858e888d646 Mon Sep 17 00:00:00 2001 From: unknown Date: Wed, 18 Apr 2007 12:55:09 +0300 Subject: Postmerge fixes. added forgoten file. The patch broke maria.test (will be fixed later) sql/handler.cc: Pagecache block should be equal maria block. sql/mysqld.cc: parameters Fixed. storage/maria/ma_bitmap.c: fixed typo. storage/maria/ma_blockrec.c: fixed typo. storage/maria/ma_delete_all.c: fixed typo. storage/maria/ma_page.c: fixed typo. storage/maria/ma_pagecache.c: pin/lock debugging protection activated by default. storage/maria/ma_pagecaches.c: parameters Fixed. storage/maria/ma_preload.c: fixed typo. mysys/my_safehash.c: New BitKeeper file ``mysys/my_safehash.c'' --- mysys/my_safehash.c | 297 ++++++++++++++++++++++++++++++++++++++++++ sql/handler.cc | 3 +- sql/mysqld.cc | 31 ++--- storage/maria/ma_bitmap.c | 2 +- storage/maria/ma_blockrec.c | 8 +- storage/maria/ma_delete_all.c | 2 +- storage/maria/ma_page.c | 7 +- storage/maria/ma_pagecache.c | 20 ++- storage/maria/ma_pagecaches.c | 2 +- storage/maria/ma_preload.c | 4 +- 10 files changed, 328 insertions(+), 48 deletions(-) create mode 100644 mysys/my_safehash.c diff --git a/mysys/my_safehash.c b/mysys/my_safehash.c new file mode 100644 index 00000000000..00ca1569b91 --- /dev/null +++ b/mysys/my_safehash.c @@ -0,0 +1,297 @@ +/* Copyright (C) 2003-2007 MySQL AB + + 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; either version 2 of the License, or + (at your option) any later version. + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ + +/* + Handling of multiple key caches + + The idea is to have a thread safe hash on the table name, + with a default key cache value that is returned if the table name is not in + the cache. +*/ + +#include "mysys_priv.h" +#include +#include "my_safehash.h" + +/***************************************************************************** + General functions to handle SAFE_HASH objects. + + A SAFE_HASH object is used to store the hash, the mutex and default value + needed by the rest of the key cache code. + This is a separate struct to make it easy to later reuse the code for other + purposes + + All entries are linked in a list to allow us to traverse all elements + and delete selected ones. (HASH doesn't allow any easy ways to do this). +*****************************************************************************/ + + +/* + Free a SAFE_HASH_ENTRY + + SYNOPSIS + safe_hash_entry_free() + entry The entry which should be freed + + NOTE + This function is called by the hash object on delete +*/ + +static void safe_hash_entry_free(SAFE_HASH_ENTRY *entry) +{ + DBUG_ENTER("free_assign_entry"); + my_free((gptr) entry, MYF(0)); + DBUG_VOID_RETURN; +} + + +/* + Get key and length for a SAFE_HASH_ENTRY + + SYNOPSIS + safe_hash_entry_get() + entry The entry for which the key should be returned + length Length of the key + + RETURN + # reference on the key +*/ + +static byte *safe_hash_entry_get(SAFE_HASH_ENTRY *entry, uint *length, + my_bool not_used __attribute__((unused))) +{ + *length= entry->length; + return (byte*) entry->key; +} + + +/* + Init a SAFE_HASH object + + SYNOPSIS + safe_hash_init() + hash safe_hash handler + elements Expected max number of elements + default_value default value + + NOTES + In case of error we set hash->default_value to 0 to allow one to call + safe_hash_free on an object that couldn't be initialized. + + RETURN + 0 OK + 1 error +*/ + +my_bool safe_hash_init(SAFE_HASH *hash, uint elements, + byte *default_value) +{ + DBUG_ENTER("safe_hash"); + if (hash_init(&hash->hash, &my_charset_bin, elements, + 0, 0, (hash_get_key) safe_hash_entry_get, + (void (*)(void*)) safe_hash_entry_free, 0)) + { + hash->default_value= 0; + DBUG_RETURN(1); + } + my_rwlock_init(&hash->mutex, 0); + hash->default_value= default_value; + hash->root= 0; + DBUG_RETURN(0); +} + + +/* + Free a SAFE_HASH object + + SYNOPSIS + safe_hash_free() + hash Hash handle + + NOTES + This is safe to call on any object that has been sent to safe_hash_init() +*/ + +void safe_hash_free(SAFE_HASH *hash) +{ + /* + Test if safe_hash_init succeeded. This will also guard us against multiple + free calls. + */ + if (hash->default_value) + { + hash_free(&hash->hash); + rwlock_destroy(&hash->mutex); + hash->default_value=0; + } +} + + +/* + Return the value stored for a key or default value if no key + + SYNOPSIS + safe_hash_search() + hash Hash handle + key key (path to table etc..) + length Length of key + def Default value of data + + RETURN + # data associated with the key of default value if data was not found +*/ + +byte *safe_hash_search(SAFE_HASH *hash, const byte *key, uint length, + byte *def) +{ + byte *result; + DBUG_ENTER("safe_hash_search"); + rw_rdlock(&hash->mutex); + result= hash_search(&hash->hash, key, length); + rw_unlock(&hash->mutex); + if (!result) + result= def; + else + result= ((SAFE_HASH_ENTRY*) result)->data; + DBUG_PRINT("exit",("data: 0x%lx", (long) result)); + DBUG_RETURN(result); +} + + +/* + Associate a key with some data + + SYNOPSIS + safe_hash_set() + hash Hash handle + key key (path to table etc..) + length Length of key + data data to to associate with the data + + NOTES + This can be used both to insert a new entry and change an existing + entry. + If one associates a key with the default key cache, the key is deleted + + RETURN + 0 OK + 1 error (Can only be EOM). In this case my_message() is called. +*/ + +my_bool safe_hash_set(SAFE_HASH *hash, const byte *key, uint length, + byte *data) +{ + SAFE_HASH_ENTRY *entry; + my_bool error= 0; + DBUG_ENTER("safe_hash_set"); + DBUG_PRINT("enter",("key: %.*s data: 0x%lx", length, key, (long) data)); + + rw_wrlock(&hash->mutex); + entry= (SAFE_HASH_ENTRY*) hash_search(&hash->hash, key, length); + + if (data == hash->default_value) + { + /* + The key is to be associated with the default entry. In this case + we can just delete the entry (if it existed) from the hash as a + search will return the default entry + */ + if (!entry) /* nothing to do */ + goto end; + /* unlink entry from list */ + if ((*entry->prev= entry->next)) + entry->next->prev= entry->prev; + hash_delete(&hash->hash, (byte*) entry); + goto end; + } + if (entry) + { + /* Entry existed; Just change the pointer to point at the new data */ + entry->data= data; + } + else + { + if (!(entry= (SAFE_HASH_ENTRY *) my_malloc(sizeof(*entry) + length, + MYF(MY_WME)))) + { + error= 1; + goto end; + } + entry->key= (byte*) (entry +1); + memcpy((char*) entry->key, (char*) key, length); + entry->length= length; + entry->data= data; + /* Link entry to list */ + if ((entry->next= hash->root)) + entry->next->prev= &entry->next; + entry->prev= &hash->root; + hash->root= entry; + if (my_hash_insert(&hash->hash, (byte*) entry)) + { + /* This can only happen if hash got out of memory */ + my_free((char*) entry, MYF(0)); + error= 1; + goto end; + } + } + +end: + rw_unlock(&hash->mutex); + DBUG_RETURN(error); +} + + +/* + Change all entries with one data value to another data value + + SYNOPSIS + safe_hash_change() + hash Hash handle + old_data Old data + new_data Change all 'old_data' to this + + NOTES + We use the linked list to traverse all elements in the hash as + this allows us to delete elements in the case where 'new_data' is the + default value. +*/ + +void safe_hash_change(SAFE_HASH *hash, byte *old_data, byte *new_data) +{ + SAFE_HASH_ENTRY *entry, *next; + DBUG_ENTER("safe_hash_set"); + + rw_wrlock(&hash->mutex); + + for (entry= hash->root ; entry ; entry= next) + { + next= entry->next; + if (entry->data == old_data) + { + if (new_data == hash->default_value) + { + if ((*entry->prev= entry->next)) + entry->next->prev= entry->prev; + hash_delete(&hash->hash, (byte*) entry); + } + else + entry->data= new_data; + } + } + + rw_unlock(&hash->mutex); + DBUG_VOID_RETURN; +} diff --git a/sql/handler.cc b/sql/handler.cc index 625ccae6d17..b32098bfc78 100644 --- a/sql/handler.cc +++ b/sql/handler.cc @@ -2796,13 +2796,12 @@ int ha_init_pagecache(const char *name, PAGECACHE *pagecache) { pthread_mutex_lock(&LOCK_global_system_variables); long tmp_buff_size= (long) pagecache->param_buff_size; - long tmp_block_size= (long) pagecache->param_block_size; uint division_limit= pagecache->param_division_limit; uint age_threshold= pagecache->param_age_threshold; pthread_mutex_unlock(&LOCK_global_system_variables); DBUG_RETURN(!init_pagecache(pagecache, tmp_buff_size, division_limit, age_threshold, - tmp_block_size)); + MARIA_KEY_BLOCK_LENGTH)); } DBUG_RETURN(0); } diff --git a/sql/mysqld.cc b/sql/mysqld.cc index ee5957c5cf2..a49e4005c67 100644 --- a/sql/mysqld.cc +++ b/sql/mysqld.cc @@ -2210,7 +2210,7 @@ and this may fail.\n\n"); (ulong) dflt_key_cache->key_cache_mem_size); #ifdef WITH_MARIA_STORAGE_ENGINE fprintf(stderr, "page_buffer_size=%lu\n", - (ulong) dflt_pagecache->mem_size); + (ulong) maria_pagecache->mem_size); #endif /* WITH_MARIA_STORAGE_ENGINE */ fprintf(stderr, "read_buffer_size=%ld\n", (long) global_system_variables.read_buff_size); fprintf(stderr, "max_used_connections=%lu\n", max_used_connections); @@ -2220,7 +2220,7 @@ and this may fail.\n\n"); key_buffer_size + (read_buffer_size + sort_buffer_size)*max_threads = %lu K\n\ bytes of memory\n", ((ulong) dflt_key_cache->key_cache_mem_size + #ifdef WITH_MARIA_STORAGE_ENGINE - (ulong) dflt_pagecache->mem_size + + (ulong) maria_pagecache->mem_size + #endif /* WITH_MARIA_STORAGE_ENGINE */ (global_system_variables.read_buff_size + global_system_variables.sortbuff_size) * @@ -2778,7 +2778,7 @@ static int init_common_variables(const char *conf_file_name, int argc, set_server_version(); #ifdef WITH_MARIA_STORAGE_ENGINE - if (!(dflt_pagecache= get_or_create_pagecache(maria_pagecache_base.str, + if (!(maria_pagecache= get_or_create_pagecache(maria_pagecache_base.str, maria_pagecache_base.length))) exit(1); /* @@ -6091,15 +6091,6 @@ log and this option does nothing anymore.", MARIA_KEY_BLOCK_LENGTH, MARIA_MIN_KEY_BLOCK_LENGTH, MARIA_MAX_KEY_BLOCK_LENGTH, 0, MARIA_MIN_KEY_BLOCK_LENGTH, 0}, - {"maria_key_buffer_size", OPT_KEY_BUFFER_SIZE, - "The size of the buffer used for index blocks for Maria tables. Increase " - "this to get better index handling (for all reads and multiple writes) to " - "as much as you can afford; 64M on a 256M machine that mainly runs MySQL " - "is quite common.", - (gptr*) &maria_pagecache_var.param_buff_size, (gptr*) 0, - 0, (GET_ULL | GET_ASK_ADDR), - REQUIRED_ARG, KEY_CACHE_SIZE, MALLOC_OVERHEAD, ~(ulong) 0, MALLOC_OVERHEAD, - IO_SIZE, 0}, {"maria_max_sort_file_size", OPT_MARIA_MAX_SORT_FILE_SIZE, "Don't use the fast sort index method to created index if the temporary " "file would get bigger than this.", @@ -6293,26 +6284,20 @@ The minimum value for this variable is 4096.", #ifdef WITH_MARIA_STORAGE_ENGINE {"pagecache_age_threshold", OPT_KEY_CACHE_AGE_THRESHOLD, "This characterizes the number of hits a hot block has to be untouched until it is considered aged enough to be downgraded to a warm block. This specifies the percentage ratio of that number of hits to the total number of blocks in key cache", - (gptr*) &dflt_pagecache_var.param_age_threshold, + (gptr*) &maria_pagecache_var.param_age_threshold, (gptr*) 0, 0, (GET_ULONG | GET_ASK_ADDR), REQUIRED_ARG, 300, 100, ~0L, 0, 100, 0}, - {"pagecache_block_size", OPT_KEY_CACHE_BLOCK_SIZE, - "The default size of key cache blocks", - (gptr*) &dflt_pagecache_var.param_block_size, - (gptr*) 0, - 0, (GET_ULONG | GET_ASK_ADDR), REQUIRED_ARG, - KEY_CACHE_BLOCK_SIZE , 512, 1024*16, MALLOC_OVERHEAD, 512, 0}, {"pagecache_buffer_size", OPT_KEY_BUFFER_SIZE, "The size of the buffer used for index blocks for MyISAM tables. Increase this to get better index handling (for all reads and multiple writes) to as much as you can afford; 64M on a 256M machine that mainly runs MySQL is quite common.", - (gptr*) &dflt_pagecache_var.param_buff_size, + (gptr*) &maria_pagecache_var.param_buff_size, (gptr*) 0, 0, (GET_ULL | GET_ASK_ADDR), REQUIRED_ARG, KEY_CACHE_SIZE, MALLOC_OVERHEAD, ~(ulong) 0, MALLOC_OVERHEAD, - IO_SIZE, 0}, + IO_SIZE, KEY_CACHE_SIZE}, {"pagecache_division_limit", OPT_KEY_CACHE_DIVISION_LIMIT, "The minimum percentage of warm blocks in key cache", - (gptr*) &dflt_pagecache_var.param_division_limit, + (gptr*) &maria_pagecache_var.param_division_limit, (gptr*) 0, 0, (GET_ULONG | GET_ASK_ADDR) , REQUIRED_ARG, 100, 1, 100, 0, 1, 0}, @@ -7230,7 +7215,7 @@ static void mysql_init_variables(void) multi_keycache_init(); #ifdef WITH_MARIA_STORAGE_ENGINE - /* set pagecache_hash.default_value = dflt_pagecache */ + /* set pagecache_hash.default_value = maria_pagecache */ multi_pagecache_init(); #endif diff --git a/storage/maria/ma_bitmap.c b/storage/maria/ma_bitmap.c index 2b2bad58346..03bd536d000 100644 --- a/storage/maria/ma_bitmap.c +++ b/storage/maria/ma_bitmap.c @@ -126,7 +126,7 @@ static inline my_bool write_changed_bitmap(MARIA_SHARE *share, (PAGECACHE_FILE*)&bitmap->file, bitmap->page, 0, (byte*) bitmap->map, PAGECACHE_PLAIN_PAGE, PAGECACHE_LOCK_LEFT_UNLOCKED, - PAGECACHE_PIN_LEFT_PINNED, + PAGECACHE_PIN_LEFT_UNPINNED, PAGECACHE_WRITE_DELAY, 0)); } diff --git a/storage/maria/ma_blockrec.c b/storage/maria/ma_blockrec.c index 870837ca991..da2aeb09524 100644 --- a/storage/maria/ma_blockrec.c +++ b/storage/maria/ma_blockrec.c @@ -992,7 +992,7 @@ static my_bool write_tail(MARIA_HA *info, &info->dfile, block->page, 0, row_pos.buff,PAGECACHE_PLAIN_PAGE, PAGECACHE_LOCK_LEFT_UNLOCKED, - PAGECACHE_PIN_LEFT_PINNED, + PAGECACHE_PIN_LEFT_UNPINNED, PAGECACHE_WRITE_DELAY, 0)); } @@ -1065,7 +1065,7 @@ static my_bool write_full_pages(MARIA_HA *info, &info->dfile, page, 0, buff, PAGECACHE_PLAIN_PAGE, PAGECACHE_LOCK_LEFT_UNLOCKED, - PAGECACHE_PIN_LEFT_PINNED, + PAGECACHE_PIN_LEFT_UNPINNED, PAGECACHE_WRITE_DELAY, 0)) DBUG_RETURN(1); @@ -1599,7 +1599,7 @@ static my_bool write_block_record(MARIA_HA *info, const byte *record, &info->dfile, head_block->page, 0, page_buff, PAGECACHE_PLAIN_PAGE, PAGECACHE_LOCK_LEFT_UNLOCKED, - PAGECACHE_PIN_LEFT_PINNED, + PAGECACHE_PIN_LEFT_UNPINNED, PAGECACHE_WRITE_DELAY, 0)) goto disk_err; @@ -1950,7 +1950,7 @@ static my_bool delete_head_or_tail(MARIA_HA *info, &info->dfile, page, 0, buff, PAGECACHE_PLAIN_PAGE, PAGECACHE_LOCK_LEFT_UNLOCKED, - PAGECACHE_PIN_LEFT_PINNED, + PAGECACHE_PIN_LEFT_UNPINNED, PAGECACHE_WRITE_DELAY, 0)) DBUG_RETURN(1); } diff --git a/storage/maria/ma_delete_all.c b/storage/maria/ma_delete_all.c index b2c984c02c7..2d85b347662 100644 --- a/storage/maria/ma_delete_all.c +++ b/storage/maria/ma_delete_all.c @@ -67,7 +67,7 @@ int maria_delete_all_rows(MARIA_HA *info) my_chsize(share->kfile.file, share->base.keystart, 0, MYF(MY_WME)) ) goto err; - if (_ma_initialize_data_file(info->dfile, info->s)) + if (_ma_initialize_data_file(info->dfile.file, info->s)) goto err; /* diff --git a/storage/maria/ma_page.c b/storage/maria/ma_page.c index 5a037337882..03c8a787697 100644 --- a/storage/maria/ma_page.c +++ b/storage/maria/ma_page.c @@ -97,7 +97,7 @@ int _ma_write_keypage(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, &info->s->kfile, page / keyinfo->block_length, level, buff, PAGECACHE_PLAIN_PAGE, PAGECACHE_LOCK_LEFT_UNLOCKED, - PAGECACHE_PIN_LEFT_PINNED, + PAGECACHE_PIN_LEFT_UNPINNED, PAGECACHE_WRITE_DELAY, 0)); } /* maria_write_keypage */ @@ -131,7 +131,7 @@ int _ma_dispose(register MARIA_HA *info, MARIA_KEYDEF *keyinfo, my_off_t pos, &info->s->kfile, page_no, level, buff, PAGECACHE_PLAIN_PAGE, PAGECACHE_LOCK_LEFT_UNLOCKED, - PAGECACHE_PIN_LEFT_PINNED, + PAGECACHE_PIN_LEFT_UNPINNED, PAGECACHE_WRITE_DELAY, 0, offset, sizeof(buff))); } /* _ma_dispose */ @@ -142,7 +142,7 @@ int _ma_dispose(register MARIA_HA *info, MARIA_KEYDEF *keyinfo, my_off_t pos, my_off_t _ma_new(register MARIA_HA *info, MARIA_KEYDEF *keyinfo, int level) { my_off_t pos; - byte buff[8]; + byte *buff; DBUG_ENTER("_ma_new"); if ((pos= info->s->state.key_del) == HA_OFFSET_ERROR) @@ -158,6 +158,7 @@ my_off_t _ma_new(register MARIA_HA *info, MARIA_KEYDEF *keyinfo, int level) } else { + buff= alloca(info->s->block_size); DBUG_ASSERT(info->s->pagecache->block_size == keyinfo->block_length && info->s->pagecache->block_size == info->s->block_size); /* diff --git a/storage/maria/ma_pagecache.c b/storage/maria/ma_pagecache.c index 2cf8a9ca2f1..3d5c3026173 100755 --- a/storage/maria/ma_pagecache.c +++ b/storage/maria/ma_pagecache.c @@ -194,7 +194,7 @@ static char *page_cache_page_pin_str[]= (char*)"pinned -> unpinned" }; #endif -#ifdef PAGECACHE_DEBUG +#ifndef DBUG_OFF typedef struct st_pagecache_pin_info { struct st_pagecache_pin_info *next, **prev; @@ -289,7 +289,7 @@ struct st_pagecache_block_link byte *buffer; /* buffer for the block page */ uint status; /* state of the block */ uint pins; /* pin counter */ -#ifdef PAGECACHE_DEBUG +#ifndef DBUG_OFF PAGECACHE_PIN_INFO *pin_list; PAGECACHE_LOCK_INFO *lock_list; #endif @@ -301,10 +301,7 @@ struct st_pagecache_block_link KEYCACHE_CONDVAR *condvar; /* condition variable for 'no readers' event */ }; -PAGECACHE dflt_pagecache_var; -PAGECACHE *dflt_pagecache= &dflt_pagecache_var; - -#ifdef PAGECACHE_DEBUG +#ifndef DBUG_OFF /* debug checks */ static my_bool info_check_pin(PAGECACHE_BLOCK_LINK *block, enum pagecache_page_pin mode) @@ -312,6 +309,9 @@ static my_bool info_check_pin(PAGECACHE_BLOCK_LINK *block, struct st_my_thread_var *thread= my_thread_var; PAGECACHE_PIN_INFO *info= info_find(block->pin_list, thread); DBUG_ENTER("info_check_pin"); + DBUG_PRINT("enter", ("info_check_pin: thread: 0x%lx pin: %s", + (ulong)thread, + page_cache_page_pin_str[mode])); if (info) { if (mode == PAGECACHE_PIN_LEFT_UNPINNED) @@ -2051,7 +2051,7 @@ static void add_pin(PAGECACHE_BLOCK_LINK *block) block->pins)); PCBLOCK_INFO(block); block->pins++; -#ifdef PAGECACHE_DEBUG +#ifndef DBUG_OFF { PAGECACHE_PIN_INFO *info= (PAGECACHE_PIN_INFO *)my_malloc(sizeof(PAGECACHE_PIN_INFO), MYF(0)); @@ -2071,7 +2071,7 @@ static void remove_pin(PAGECACHE_BLOCK_LINK *block) PCBLOCK_INFO(block); DBUG_ASSERT(block->pins > 0); block->pins--; -#ifdef PAGECACHE_DEBUG +#ifndef DBUG_OFF { PAGECACHE_PIN_INFO *info= info_find(block->pin_list, my_thread_var); DBUG_ASSERT(info != 0); @@ -2081,7 +2081,7 @@ static void remove_pin(PAGECACHE_BLOCK_LINK *block) #endif DBUG_VOID_RETURN; } -#ifdef PAGECACHE_DEBUG +#ifndef DBUG_OFF static void info_add_lock(PAGECACHE_BLOCK_LINK *block, my_bool wl) { PAGECACHE_LOCK_INFO *info= @@ -2237,10 +2237,8 @@ static my_bool make_lock_and_pin(PAGECACHE *pagecache, page_cache_page_lock_str[lock], page_cache_page_pin_str[pin])); PCBLOCK_INFO(block); -#ifdef PAGECACHE_DEBUG DBUG_ASSERT(info_check_pin(block, pin) == 0 && info_check_lock(block, lock, pin) == 0); -#endif switch (lock) { case PAGECACHE_LOCK_WRITE: /* free -> write */ diff --git a/storage/maria/ma_pagecaches.c b/storage/maria/ma_pagecaches.c index e635709c11e..1a120131016 100644 --- a/storage/maria/ma_pagecaches.c +++ b/storage/maria/ma_pagecaches.c @@ -38,7 +38,7 @@ static SAFE_HASH pagecache_hash; my_bool multi_pagecache_init(void) { - return safe_hash_init(&pagecache_hash, 16, (byte*) dflt_pagecache); + return safe_hash_init(&pagecache_hash, 16, (byte*) maria_pagecache); } diff --git a/storage/maria/ma_preload.c b/storage/maria/ma_preload.c index fc818b5b277..44fc12f8571 100644 --- a/storage/maria/ma_preload.c +++ b/storage/maria/ma_preload.c @@ -94,7 +94,7 @@ int maria_preload(MARIA_HA *info, ulonglong key_map, my_bool ignore_leaves) (byte*) buff, PAGECACHE_PLAIN_PAGE, PAGECACHE_LOCK_LEFT_UNLOCKED, - PAGECACHE_PIN_LEFT_PINNED, + PAGECACHE_PIN_LEFT_UNPINNED, PAGECACHE_WRITE_DONE, 0)) goto err; } @@ -111,7 +111,7 @@ int maria_preload(MARIA_HA *info, ulonglong key_map, my_bool ignore_leaves) (byte*) buff, PAGECACHE_PLAIN_PAGE, PAGECACHE_LOCK_LEFT_UNLOCKED, - PAGECACHE_PIN_LEFT_PINNED, + PAGECACHE_PIN_LEFT_UNPINNED, PAGECACHE_WRITE_DONE, 0)) goto err; pos+= length; -- cgit v1.2.1 From ee8f8dd128ffa6fae5270918154526039b8eb80c Mon Sep 17 00:00:00 2001 From: unknown Date: Wed, 18 Apr 2007 14:45:32 +0300 Subject: Fixed dubug info. sql/set_var.cc: fixed parameters. --- mysys/my_safehash.c | 6 +++--- sql/set_var.cc | 11 +++++------ 2 files changed, 8 insertions(+), 9 deletions(-) diff --git a/mysys/my_safehash.c b/mysys/my_safehash.c index 00ca1569b91..57f408942bf 100644 --- a/mysys/my_safehash.c +++ b/mysys/my_safehash.c @@ -52,7 +52,7 @@ static void safe_hash_entry_free(SAFE_HASH_ENTRY *entry) { - DBUG_ENTER("free_assign_entry"); + DBUG_ENTER("safe_hash_entry_free"); my_free((gptr) entry, MYF(0)); DBUG_VOID_RETURN; } @@ -99,7 +99,7 @@ static byte *safe_hash_entry_get(SAFE_HASH_ENTRY *entry, uint *length, my_bool safe_hash_init(SAFE_HASH *hash, uint elements, byte *default_value) { - DBUG_ENTER("safe_hash"); + DBUG_ENTER("safe_hash_init"); if (hash_init(&hash->hash, &my_charset_bin, elements, 0, 0, (hash_get_key) safe_hash_entry_get, (void (*)(void*)) safe_hash_entry_free, 0)) @@ -272,7 +272,7 @@ end: void safe_hash_change(SAFE_HASH *hash, byte *old_data, byte *new_data) { SAFE_HASH_ENTRY *entry, *next; - DBUG_ENTER("safe_hash_set"); + DBUG_ENTER("safe_hash_change"); rw_wrlock(&hash->mutex); diff --git a/sql/set_var.cc b/sql/set_var.cc index 04c80576887..55db1f6fd5c 100644 --- a/sql/set_var.cc +++ b/sql/set_var.cc @@ -4013,15 +4013,14 @@ static PAGECACHE *create_pagecache(const char *name, uint length) { /* Set default values for a key cache - The values in dflt_pagecache_var is set by my_getopt() at startup + The values in maria_pagecache_var is set by my_getopt() at startup We don't set 'buff_size' as this is used to enable the key cache */ - pagecache->param_buff_size= (dflt_pagecache_var.param_buff_size ? - dflt_pagecache_var.param_buff_size: + pagecache->param_buff_size= (maria_pagecache_var.param_buff_size ? + maria_pagecache_var.param_buff_size: KEY_CACHE_SIZE); - pagecache->param_block_size= dflt_pagecache_var.param_block_size; - pagecache->param_division_limit= dflt_pagecache_var.param_division_limit; - pagecache->param_age_threshold= dflt_pagecache_var.param_age_threshold; + pagecache->param_division_limit= maria_pagecache_var.param_division_limit; + pagecache->param_age_threshold= maria_pagecache_var.param_age_threshold; } } DBUG_RETURN(pagecache); -- cgit v1.2.1 From eb7d9500a9909ce594c4d169e70fb5cecbb33e2b Mon Sep 17 00:00:00 2001 From: unknown Date: Thu, 19 Apr 2007 13:18:56 +0300 Subject: Fixes after review of guilhem of block record patch Short overview: Changed a lot of variable, functions, defines and struct elements to use more readable names More comments (mostly function and structure slot comments) Other things: Changed 'USE_WHOLE_KEY' to a big number to not interfer with long keys Ensure that tail block are at least of size 'MIN_TAIL_SIZE' Allow longer keys and key parts than before (don't limit Maria interface by HA_MAX_KEY_LENGTH) Use ma_chsize() to write initial bitmap page Added checking if using file with wrong block_size Added issing types to type_names[] (for maria_chk -d) Added maria_max_key_length() include/maria.h: Changed maria_portable_size_char_ptr to portable_size_char_ptr and moved it to my_handler.h Removed not used variable maria_delay_rec_write. More comments include/my_handler.h: Added portable_sizeof_char_ptr include/myisam.h: Changed mi_portable_size_char_ptr to portable_size_char_ptr and moved it to my_handler.h mysql-test/r/maria.result: Fix results when we now have a longer key length mysql-test/t/maria.test: More tests mysys/my_pread.c: Code cleanup sql/net_serv.cc: Changed warning to note (as in main 5.1 tree) to avoid not critical failing tests sql/sql_select.cc: Use portable_sizeof_char_ptr storage/maria/ha_maria.cc: Added max_supported_key_length(), as this is not a trival function anymore storage/maria/ha_maria.h: Moved max_supported_key_length(), as this is not a trival function anymore storage/maria/ma_bitmap.c: Lots of new comments Added maria_bitmap_marker[] to mark 2 last bytes of each bitmap (for corruption detection) Trivial code changes (based on review comments) storage/maria/ma_blockrec.c: More code comments Renamed _block_row() functions to _block_record() Trivial code changes, based on review comments Moved Code from maria_close() to _ma_end_block_record() Some function renames to make things more understandable DIR_ENTRY_OFFSET -> DIR_COUNT_OFFSET keybuff_used -> keyread_buff_used ma_recordpos_to_offset -> ma_recordpos_to_dir_entry Changed some 'rec' named variables to 'column'. Ensure that tail block are at least of size 'MIN_TAIL_SIZE' storage/maria/ma_blockrec.h: More comments DIRCOUNT_SIZE -> DIR_COUNT_SIZE Added define for maira_bitmap_marker[] ma_recordpos_to_offset -> ma_recordpos_to_dir_entry xxx_block_row() -> xxx_block_record() Made _ma_read_bitmap_page() static storage/maria/ma_check.c: More comments ma_recordpos_to_offset() -> ma_recordpos_to_dir_entry() DIR_ENTRY_OFFSET -> DIR_COUNT_OFFSET rec variables -> column variables recdef -> columndef storage/maria/ma_checksum.c: rec -> column Avoid an 'if' in _ma_checksum() for the common case storage/maria/ma_close.c: Moved resetting of info->dfile to ma_end_once_block_record() storage/maria/ma_create.c: Some variable changes to make things more readable: recinfo -> columndef rec -> column rec_end -> end_column record_type -> datafile_type ma_recinfo_write() -> ma_columndef_write() Fixed wrong setting of 'data_file_length'; Now max_rows should be calculated correctly New check if too long key. Use ma_chsize() to write bitmap page. storage/maria/ma_delete.c: keybuff_used -> keyread_buff_used storage/maria/ma_dynrec.c: rec -> columndef rec_length -> column_length maria_portable_sizeof_char_ptr -> portable_sizeof_char_ptr Better comment for _ma_read_rnd_dynamic_record() storage/maria/ma_ft_eval.c: maria_portable_sizeof_char_ptr -> portable_sizeof_char_ptr storage/maria/ma_ft_test1.c: maria_portable_sizeof_char_ptr -> portable_sizeof_char_ptr storage/maria/ma_ft_update.c: keybuff_used -> keyread_buff_used storage/maria/ma_info.c: More comments storage/maria/ma_open.c: Added checking if using file with wrong block_size New checking of max_key_length rec -> columndef _ma_recinfo_write -> _ma_columndef_write Don't change block_size (as this is checked in ma_create()) More comments storage/maria/ma_packrec.c: Trivial code changes rec -> columndef maria_portable_sizeof_char_ptr -> portable_sizeof_char_ptr storage/maria/ma_page.c: keybuff_used -> keyread_buff_used storage/maria/ma_rkey.c: Removed not needded empty line storage/maria/ma_rrnd.c: Removed not used variable storage/maria/ma_rt_index.c: keybuff_used -> keyread_buff_used storage/maria/ma_search.c: keybuff_used -> keyread_buff_used Trivial code changes storage/maria/ma_sp_test.c: maria_portable_sizeof_char_ptr -> portable_sizeof_char_ptr storage/maria/ma_test1.c: maria_portable_sizeof_char_ptr -> portable_sizeof_char_ptr storage/maria/ma_test2.c: maria_portable_sizeof_char_ptr -> portable_sizeof_char_ptr storage/maria/ma_update.c: Updated comment storage/maria/ma_write.c: keybuff_used -> keyread_buff_used storage/maria/maria_chk.c: Added missing types to type_names[] Removed not used variable rec -> columndef Replaced some numbers with define flags storage/maria/maria_def.h: More comments Added 'MARIA_INDEX_MIN_OVERHEAD_SIZE' rec -> columndef keybuff_used -> keyread_buff_used _ma_recinfo_write -> _ma_culumndef_write _ma_recinfo_read -> _ma_columndef_read Changed 'USE_WHOLE_KEY' to a big number to not interfer with long keys Added maria_max_key_length() storage/maria/maria_pack.c: Updated message strings rec -> columndef maria_portable_sizeof_char_ptr -> portable_sizeof_char_ptr More comments storage/myisam/ft_eval.c: mi_portable_sizeof_char_ptr -> portable_sizeof_char_ptr storage/myisam/ft_test1.c: mi_portable_sizeof_char_ptr -> portable_sizeof_char_ptr storage/myisam/mi_checksum.c: mi_portable_sizeof_char_ptr -> portable_sizeof_char_ptr storage/myisam/mi_create.c: mi_portable_sizeof_char_ptr -> portable_sizeof_char_ptr storage/myisam/mi_dynrec.c: mi_portable_sizeof_char_ptr -> portable_sizeof_char_ptr storage/myisam/mi_open.c: mi_portable_sizeof_char_ptr -> portable_sizeof_char_ptr storage/myisam/mi_packrec.c: mi_portable_sizeof_char_ptr -> portable_sizeof_char_ptr storage/myisam/mi_rkey.c: Unlock mutex also in case of error storage/myisam/mi_test1.c: mi_portable_sizeof_char_ptr -> portable_sizeof_char_ptr storage/myisam/mi_test2.c: mi_portable_sizeof_char_ptr -> portable_sizeof_char_ptr storage/myisam/myisampack.c: mi_portable_sizeof_char_ptr -> portable_sizeof_char_ptr storage/myisam/sp_test.c: mi_portable_sizeof_char_ptr -> portable_sizeof_char_ptr support-files/magic: Fixed typo --- include/maria.h | 9 +- include/my_handler.h | 7 + include/myisam.h | 2 - mysql-test/r/maria.result | 40 +++-- mysql-test/t/maria.test | 4 + mysys/my_pread.c | 2 - sql/net_serv.cc | 4 +- sql/sql_select.cc | 4 +- storage/maria/ha_maria.cc | 22 ++- storage/maria/ha_maria.h | 5 +- storage/maria/ma_bitmap.c | 318 ++++++++++++++++++++++++++++++++---- storage/maria/ma_blockrec.c | 380 ++++++++++++++++++++++++++----------------- storage/maria/ma_blockrec.h | 54 +++--- storage/maria/ma_check.c | 40 +++-- storage/maria/ma_checksum.c | 29 ++-- storage/maria/ma_close.c | 4 +- storage/maria/ma_create.c | 153 ++++++++--------- storage/maria/ma_delete.c | 2 +- storage/maria/ma_dynrec.c | 105 ++++++------ storage/maria/ma_ft_eval.c | 2 +- storage/maria/ma_ft_test1.c | 4 +- storage/maria/ma_ft_update.c | 2 +- storage/maria/ma_info.c | 4 + storage/maria/ma_open.c | 88 +++++----- storage/maria/ma_packrec.c | 32 ++-- storage/maria/ma_page.c | 2 +- storage/maria/ma_rkey.c | 1 - storage/maria/ma_rrnd.c | 1 - storage/maria/ma_rt_index.c | 22 +-- storage/maria/ma_search.c | 18 +- storage/maria/ma_sp_test.c | 2 +- storage/maria/ma_test1.c | 4 +- storage/maria/ma_test2.c | 2 +- storage/maria/ma_update.c | 2 +- storage/maria/ma_write.c | 4 +- storage/maria/maria_chk.c | 40 ++--- storage/maria/maria_def.h | 74 +++++++-- storage/maria/maria_pack.c | 20 ++- storage/myisam/ft_eval.c | 2 +- storage/myisam/ft_test1.c | 4 +- storage/myisam/mi_checksum.c | 4 +- storage/myisam/mi_create.c | 4 +- storage/myisam/mi_dynrec.c | 8 +- storage/myisam/mi_open.c | 2 +- storage/myisam/mi_packrec.c | 2 +- storage/myisam/mi_rkey.c | 2 + storage/myisam/mi_test1.c | 4 +- storage/myisam/mi_test2.c | 2 +- storage/myisam/myisampack.c | 8 +- storage/myisam/sp_test.c | 2 +- support-files/magic | 4 +- 51 files changed, 999 insertions(+), 557 deletions(-) diff --git a/include/maria.h b/include/maria.h index a8c5cbf794b..dc3df955850 100644 --- a/include/maria.h +++ b/include/maria.h @@ -54,8 +54,6 @@ extern "C" { #define MARIA_KEY_BLOCK_LENGTH 8192 /* default key block length */ #define MARIA_MIN_KEY_BLOCK_LENGTH 1024 /* Min key block length */ #define MARIA_MAX_KEY_BLOCK_LENGTH 32768 -#define maria_portable_sizeof_char_ptr 8 -#define MARIA_MAX_KEY_LENGTH 1000 /* Max length in bytes */ /* In the following macros '_keyno_' is 0 .. keys-1. @@ -150,6 +148,7 @@ typedef struct st_maria_create_info ulonglong auto_increment; ulonglong data_file_length; ulonglong key_file_length; + /* Size of null bitmap at start of row */ uint null_bytes; uint old_options; enum data_file_type org_data_file_type; @@ -226,11 +225,13 @@ typedef struct st_maria_columndef /* column information */ uint64 offset; /* Offset to position in row */ enum en_fieldtype type; uint16 length; /* length of field */ + /* Intern variable (size of total storage area for the row) */ uint16 fill_length; uint16 null_pos; /* Position for null marker */ uint16 empty_pos; /* Position for empty marker */ uint8 null_bit; /* If column may be NULL */ - uint8 empty_bit; /* If column may be empty */ + /* Intern. Set if column should be zero packed (part of empty_bits) */ + uint8 empty_bit; #ifndef NOT_PACKED_DATABASES void(*unpack)(struct st_maria_columndef *rec, @@ -246,7 +247,7 @@ typedef struct st_maria_columndef /* column information */ extern ulong maria_block_size; extern ulong maria_concurrent_insert; extern my_bool maria_flush, maria_single_user; -extern my_bool maria_delay_key_write, maria_delay_rec_write; +extern my_bool maria_delay_key_write; extern my_off_t maria_max_temp_length; extern ulong maria_bulk_insert_tree_size, maria_data_pointer_size; extern KEY_CACHE maria_key_cache_var, *maria_key_cache; diff --git a/include/my_handler.h b/include/my_handler.h index 72caf67398f..13dcd01a332 100644 --- a/include/my_handler.h +++ b/include/my_handler.h @@ -111,4 +111,11 @@ extern int ha_key_cmp(register HA_KEYSEG *keyseg, register uchar *a, extern HA_KEYSEG *ha_find_null(HA_KEYSEG *keyseg, uchar *a); +/* + Inside an in-memory data record, memory pointers to pieces of the + record (like BLOBs) are stored in their native byte order and in + this amount of bytes. +*/ +#define portable_sizeof_char_ptr 8 + #endif /* _my_handler_h */ diff --git a/include/myisam.h b/include/myisam.h index d6d6426adf7..d66dd54cf98 100644 --- a/include/myisam.h +++ b/include/myisam.h @@ -55,8 +55,6 @@ extern "C" { #define MI_MIN_KEY_BLOCK_LENGTH 1024 /* Min key block length */ #define MI_MAX_KEY_BLOCK_LENGTH 16384 -#define mi_portable_sizeof_char_ptr 8 - /* In the following macros '_keyno_' is 0 .. keys-1. If there can be more keys than bits in the key_map, the highest bit diff --git a/mysql-test/r/maria.result b/mysql-test/r/maria.result index d52f243e186..515cb73eb2a 100644 --- a/mysql-test/r/maria.result +++ b/mysql-test/r/maria.result @@ -325,10 +325,14 @@ Table Op Msg_type Msg_text test.t1 check status OK drop table t1; CREATE TABLE t1 (a varchar(255), b varchar(255), c varchar(255), d varchar(255), e varchar(255), KEY t1 (a, b, c, d, e)); -ERROR 42000: Specified key was too long; max key length is 1000 bytes +ERROR 42000: Specified key was too long; max key length is 1112 bytes +CREATE TABLE t1 (a varchar(32000), unique key(a)); +ERROR 42000: Specified key was too long; max key length is 1112 bytes +CREATE TABLE t1 (a varchar(1), b varchar(1), key (a,b,a,b,a,b,a,b,a,b,a,b,a,b,a,b,a,b,a,b,a,b,a,b,a,b,a,b,a,b,a,b,a,b,a,b)); +ERROR 42000: Too many key parts specified; max 16 parts allowed CREATE TABLE t1 (a varchar(255), b varchar(255), c varchar(255), d varchar(255), e varchar(255)); ALTER TABLE t1 ADD INDEX t1 (a, b, c, d, e); -ERROR 42000: Specified key was too long; max key length is 1000 bytes +ERROR 42000: Specified key was too long; max key length is 1112 bytes DROP TABLE t1; CREATE TABLE t1 (a int not null, b int, c int, key(b), key(c), key(a,b), key(c,a)); INSERT into t1 values (0, null, 0), (0, null, 1), (0, null, 2), (0, null,3), (1,1,4); @@ -1476,7 +1480,7 @@ a b drop table t1; create table t1 (v varchar(65530), key(v)); Warnings: -Warning 1071 Specified key was too long; max key length is 1000 bytes +Warning 1071 Specified key was too long; max key length is 1112 bytes drop table if exists t1; create table t1 (v varchar(65536)); Warnings: @@ -1607,34 +1611,34 @@ t1 CREATE TABLE `t1` ( drop table t1; create table t1 (a varchar(2048), key `a` (a)); Warnings: -Warning 1071 Specified key was too long; max key length is 1000 bytes +Warning 1071 Specified key was too long; max key length is 1112 bytes show create table t1; Table Create Table t1 CREATE TABLE `t1` ( `a` varchar(2048) DEFAULT NULL, - KEY `a` (`a`(1000)) + KEY `a` (`a`(1112)) ) ENGINE=MARIA DEFAULT CHARSET=latin1 drop table t1; create table t1 (a varchar(2048), key `a` (a) key_block_size=1024); Warnings: -Warning 1071 Specified key was too long; max key length is 1000 bytes +Warning 1071 Specified key was too long; max key length is 1112 bytes show create table t1; Table Create Table t1 CREATE TABLE `t1` ( `a` varchar(2048) DEFAULT NULL, - KEY `a` (`a`(1000)) KEY_BLOCK_SIZE=8192 + KEY `a` (`a`(1112)) KEY_BLOCK_SIZE=8192 ) ENGINE=MARIA DEFAULT CHARSET=latin1 drop table t1; create table t1 (a int not null, b varchar(2048), key (a), key(b)) key_block_size=1024; Warnings: -Warning 1071 Specified key was too long; max key length is 1000 bytes +Warning 1071 Specified key was too long; max key length is 1112 bytes show create table t1; Table Create Table t1 CREATE TABLE `t1` ( `a` int(11) NOT NULL, `b` varchar(2048) DEFAULT NULL, KEY `a` (`a`) KEY_BLOCK_SIZE=8192, - KEY `b` (`b`(1000)) KEY_BLOCK_SIZE=8192 + KEY `b` (`b`(1112)) KEY_BLOCK_SIZE=8192 ) ENGINE=MARIA DEFAULT CHARSET=latin1 KEY_BLOCK_SIZE=1024 alter table t1 key_block_size=2048; show create table t1; @@ -1643,7 +1647,7 @@ t1 CREATE TABLE `t1` ( `a` int(11) NOT NULL, `b` varchar(2048) DEFAULT NULL, KEY `a` (`a`) KEY_BLOCK_SIZE=8192, - KEY `b` (`b`(1000)) KEY_BLOCK_SIZE=8192 + KEY `b` (`b`(1112)) KEY_BLOCK_SIZE=8192 ) ENGINE=MARIA DEFAULT CHARSET=latin1 KEY_BLOCK_SIZE=2048 alter table t1 add c int, add key (c); show create table t1; @@ -1653,7 +1657,7 @@ t1 CREATE TABLE `t1` ( `b` varchar(2048) DEFAULT NULL, `c` int(11) DEFAULT NULL, KEY `a` (`a`) KEY_BLOCK_SIZE=8192, - KEY `b` (`b`(1000)) KEY_BLOCK_SIZE=8192, + KEY `b` (`b`(1112)) KEY_BLOCK_SIZE=8192, KEY `c` (`c`) KEY_BLOCK_SIZE=8192 ) ENGINE=MARIA DEFAULT CHARSET=latin1 KEY_BLOCK_SIZE=2048 alter table t1 key_block_size=0; @@ -1666,33 +1670,33 @@ t1 CREATE TABLE `t1` ( `c` int(11) DEFAULT NULL, `d` int(11) DEFAULT NULL, KEY `a` (`a`) KEY_BLOCK_SIZE=8192, - KEY `b` (`b`(1000)) KEY_BLOCK_SIZE=8192, + KEY `b` (`b`(1112)) KEY_BLOCK_SIZE=8192, KEY `c` (`c`) KEY_BLOCK_SIZE=8192, KEY `d` (`d`) ) ENGINE=MARIA DEFAULT CHARSET=latin1 drop table t1; create table t1 (a int not null, b varchar(2048), key (a), key(b)) key_block_size=8192; Warnings: -Warning 1071 Specified key was too long; max key length is 1000 bytes +Warning 1071 Specified key was too long; max key length is 1112 bytes show create table t1; Table Create Table t1 CREATE TABLE `t1` ( `a` int(11) NOT NULL, `b` varchar(2048) DEFAULT NULL, KEY `a` (`a`), - KEY `b` (`b`(1000)) + KEY `b` (`b`(1112)) ) ENGINE=MARIA DEFAULT CHARSET=latin1 KEY_BLOCK_SIZE=8192 drop table t1; create table t1 (a int not null, b varchar(2048), key (a) key_block_size=1024, key(b)) key_block_size=8192; Warnings: -Warning 1071 Specified key was too long; max key length is 1000 bytes +Warning 1071 Specified key was too long; max key length is 1112 bytes show create table t1; Table Create Table t1 CREATE TABLE `t1` ( `a` int(11) NOT NULL, `b` varchar(2048) DEFAULT NULL, KEY `a` (`a`), - KEY `b` (`b`(1000)) + KEY `b` (`b`(1112)) ) ENGINE=MARIA DEFAULT CHARSET=latin1 KEY_BLOCK_SIZE=8192 drop table t1; create table t1 (a int not null, b int, key (a) key_block_size=1024, key(b) key_block_size=8192) key_block_size=16384; @@ -1715,12 +1719,12 @@ t1 CREATE TABLE `t1` ( drop table t1; create table t1 (a varchar(2048), key `a` (a) key_block_size=1000000000000000000); Warnings: -Warning 1071 Specified key was too long; max key length is 1000 bytes +Warning 1071 Specified key was too long; max key length is 1112 bytes show create table t1; Table Create Table t1 CREATE TABLE `t1` ( `a` varchar(2048) DEFAULT NULL, - KEY `a` (`a`(1000)) KEY_BLOCK_SIZE=8192 + KEY `a` (`a`(1112)) KEY_BLOCK_SIZE=8192 ) ENGINE=MARIA DEFAULT CHARSET=latin1 drop table t1; create table t1 (a int not null, key `a` (a) key_block_size=1025); diff --git a/mysql-test/t/maria.test b/mysql-test/t/maria.test index f368efcdfa2..bbf96d401f0 100644 --- a/mysql-test/t/maria.test +++ b/mysql-test/t/maria.test @@ -345,6 +345,10 @@ drop table t1; --error 1071 CREATE TABLE t1 (a varchar(255), b varchar(255), c varchar(255), d varchar(255), e varchar(255), KEY t1 (a, b, c, d, e)); +--error 1071 +CREATE TABLE t1 (a varchar(32000), unique key(a)); +--error 1070 +CREATE TABLE t1 (a varchar(1), b varchar(1), key (a,b,a,b,a,b,a,b,a,b,a,b,a,b,a,b,a,b,a,b,a,b,a,b,a,b,a,b,a,b,a,b,a,b,a,b)); CREATE TABLE t1 (a varchar(255), b varchar(255), c varchar(255), d varchar(255), e varchar(255)); --error 1071 ALTER TABLE t1 ADD INDEX t1 (a, b, c, d, e); diff --git a/mysys/my_pread.c b/mysys/my_pread.c index 2b9a994299f..339c5627a3e 100644 --- a/mysys/my_pread.c +++ b/mysys/my_pread.c @@ -87,10 +87,8 @@ uint my_pread(File Filedes, byte *Buffer, uint Count, my_off_t offset, my_error(EE_READ, MYF(ME_BELL+ME_WAITTANG), my_filename(Filedes),my_errno); else if (MyFlags & (MY_NABP | MY_FNABP)) - { my_error(EE_EOFERR, MYF(ME_BELL+ME_WAITTANG), my_filename(Filedes),my_errno); - } } if ((int) readbytes == -1 || (MyFlags & (MY_FNABP | MY_NABP))) DBUG_RETURN(MY_FILE_ERROR); /* Return with error */ diff --git a/sql/net_serv.cc b/sql/net_serv.cc index 2156888b8cf..06d12e8983d 100644 --- a/sql/net_serv.cc +++ b/sql/net_serv.cc @@ -298,8 +298,8 @@ void net_clear(NET *net, my_bool clear_buffer) { DBUG_PRINT("info",("skipped %d bytes from file: %s", count, vio_description(net->vio))); -#if defined(EXTRA_DEBUG) && (MYSQL_VERSION_ID < 51000) - fprintf(stderr,"Error: net_clear() skipped %d bytes from file: %s\n", +#if defined(EXTRA_DEBUG) && (MYSQL_VERSION_ID < 50100) + fprintf(stderr,"Note: net_clear() skipped %d bytes from file: %s\n", count, vio_description(net->vio)); #endif } diff --git a/sql/sql_select.cc b/sql/sql_select.cc index f58433cd220..026ae004919 100644 --- a/sql/sql_select.cc +++ b/sql/sql_select.cc @@ -9301,7 +9301,7 @@ create_tmp_table(THD *thd,TMP_TABLE_PARAM *param,List &fields, table->s= share; init_tmp_table_share(share, "", 0, tmpname, tmpname); share->blob_field= blob_field; - share->blob_ptr_size= mi_portable_sizeof_char_ptr; + share->blob_ptr_size= portable_sizeof_char_ptr; share->db_low_byte_first=1; // True for HEAP and MyISAM share->table_charset= param->table_charset; share->primary_key= MAX_KEY; // Indicate no primary key @@ -9856,7 +9856,7 @@ TABLE *create_virtual_tmp_table(THD *thd, List &field_list) table->s= share; share->blob_field= blob_field; share->fields= field_count; - share->blob_ptr_size= mi_portable_sizeof_char_ptr; + share->blob_ptr_size= portable_sizeof_char_ptr; setup_tmp_table_column_bitmaps(table, bitmaps); /* Create all fields and calculate the total length of record */ diff --git a/storage/maria/ha_maria.cc b/storage/maria/ha_maria.cc index ba8bb654a7d..d1237bba8f3 100644 --- a/storage/maria/ha_maria.cc +++ b/storage/maria/ha_maria.cc @@ -28,6 +28,7 @@ #include "maria_def.h" #include "ma_rt_index.h" +#include "ma_blockrec.h" ulong maria_recover_options= HA_RECOVER_NONE; @@ -108,7 +109,6 @@ static void _ma_check_print_msg(HA_CHECK *param, const char *msg_type, } - /* Convert TABLE object to Maria key and column definition @@ -512,6 +512,26 @@ double ha_maria::scan_time() return handler::scan_time(); } +/* + We need to be able to store at least two keys on an index page as the + splitting algorithms depends on this. (With only one key on a page + we also can't use any compression, which may make the index file much + larger) + We use HA_MAX_KEY_BUFF as this is a stack restriction imposed by the + handler interface. + + We also need to reserve place for a record pointer (8) and 3 bytes + per key segment to store the length of the segment + possible null bytes. + These extra bytes are required here so that maria_create() will surely + accept any keys created which the returned key data storage length. +*/ + +uint ha_maria::max_supported_key_length() const +{ + uint tmp= (maria_max_key_length() - 8 - HA_MAX_KEY_SEG*3); + return min(HA_MAX_KEY_BUFF, tmp); +} + #ifdef HAVE_REPLICATION int ha_maria::net_read_dump(NET * net) diff --git a/storage/maria/ha_maria.h b/storage/maria/ha_maria.h index 3f281711253..031a3dc3b98 100644 --- a/storage/maria/ha_maria.h +++ b/storage/maria/ha_maria.h @@ -59,10 +59,9 @@ public: } uint max_supported_keys() const { return MARIA_MAX_KEY; } - uint max_supported_key_length() const - { return HA_MAX_KEY_LENGTH; } + uint max_supported_key_length() const; uint max_supported_key_part_length() const - { return HA_MAX_KEY_LENGTH; } + { return max_supported_key_length(); } enum row_type get_row_type() const; uint checksum() const; virtual double scan_time(); diff --git a/storage/maria/ma_bitmap.c b/storage/maria/ma_bitmap.c index 202f695e30c..706743d349d 100644 --- a/storage/maria/ma_bitmap.c +++ b/storage/maria/ma_bitmap.c @@ -30,7 +30,7 @@ 2 bits are used to indicate: 0 Empty - 1 50-75 % full (at least room for 2 records) + 1 0-75 % full (at least room for 2 records) 2 75-100 % full (at least room for one record) 3 100 % full (no more room for records) @@ -89,9 +89,9 @@ Bitmaps are read on demand in response to insert/delete/update operations. The following bitmap pointers will be cached and stored on disk on close: - Current insert_bitmap; When inserting new data we will first try to - fill this one. + fill this one. - First bitmap which is not completely full. This is updated when we - free data with an update or delete. + free data with an update or delete. While flushing out bitmaps, we will cache the status of the bitmap in memory to avoid having to read a bitmap for insert of new data that will not @@ -106,7 +106,6 @@ put on disk even if they are not in the page cache). - When explicitely requested (for example on backup or after recvoery, to simplify things) - */ #include "maria_def.h" @@ -118,6 +117,15 @@ #define FULL_HEAD_PAGE 4 #define FULL_TAIL_PAGE 7 +uchar maria_bitmap_marker[2]= {(uchar) 'b',(uchar) 'm'}; + +static my_bool _ma_read_bitmap_page(MARIA_SHARE *share, + MARIA_FILE_BITMAP *bitmap, + ulonglong page); + + +/* Write bitmap page to key cache */ + static inline my_bool write_changed_bitmap(MARIA_SHARE *share, MARIA_FILE_BITMAP *bitmap) { @@ -128,7 +136,19 @@ static inline my_bool write_changed_bitmap(MARIA_SHARE *share, } /* - Initialize bitmap. This is called the first time a file is opened + Initialize bitmap variables in share + + SYNOPSIS + _ma_bitmap_init() + share Share handler + file data file handler + + NOTES + This is called the first time a file is opened. + + RETURN + 0 ok + 1 error */ my_bool _ma_bitmap_init(MARIA_SHARE *share, File file) @@ -175,20 +195,21 @@ my_bool _ma_bitmap_init(MARIA_SHARE *share, File file) Start by reading first page (assume table scan) Later code is simpler if it can assume we always have an active bitmap. */ - if (_ma_read_bitmap_page(share, bitmap, (ulonglong) 0)) - return(1); - return 0; + return _ma_read_bitmap_page(share, bitmap, (ulonglong) 0); } /* Free data allocated by _ma_bitmap_init + + SYNOPSIS + _ma_bitmap_end() + share Share handler */ my_bool _ma_bitmap_end(MARIA_SHARE *share) { - my_bool res= 0; - _ma_flush_bitmap(share); + my_bool res= _ma_flush_bitmap(share); pthread_mutex_destroy(&share->bitmap.bitmap_lock); my_free((byte*) share->bitmap.map, MYF(MY_ALLOW_ZERO_PTR)); share->bitmap.map= 0; @@ -198,6 +219,20 @@ my_bool _ma_bitmap_end(MARIA_SHARE *share) /* Flush bitmap to disk + + SYNOPSIS + _ma_flush_bitmap() + share Share handler + + NOTES + In the future, _ma_flush_bitmap() will be called to flush changes don't + by this thread (ie, checking the changed flag is ok). The reason we + check it again in the mutex is that if someone else did a flush at the + same time, we don't have to do the write. + + RETURN + 0 ok + 1 error */ my_bool _ma_flush_bitmap(MARIA_SHARE *share) @@ -217,12 +252,24 @@ my_bool _ma_flush_bitmap(MARIA_SHARE *share) } +/* + Intialize bitmap in memory to a zero bitmap + + SYNOPSIS + _ma_bitmap_delete_all() + share Share handler + + NOTES + This is called on ma_delete_all (truncate data file). +*/ + void _ma_bitmap_delete_all(MARIA_SHARE *share) { MARIA_FILE_BITMAP *bitmap= &share->bitmap; if (bitmap->map) /* Not in create */ { bzero(bitmap->map, share->block_size); + memcpy(bitmap->map + share->block_size - 2, maria_bitmap_marker, 2); bitmap->changed= 0; bitmap->page= 0; bitmap->used_size= bitmap->total_size; @@ -256,7 +303,15 @@ static uint size_to_head_pattern(MARIA_FILE_BITMAP *bitmap, uint size) /* - Return bitmap pattern for block where there is size bytes free + Return bitmap pattern for head block where there is size bytes free + + SYNOPSIS + _ma_free_size_to_head_pattern() + bitmap Bitmap + size Requested size + + RETURN + 0-4 (Possible bitmap patterns for head block) */ uint _ma_free_size_to_head_pattern(MARIA_FILE_BITMAP *bitmap, uint size) @@ -294,6 +349,18 @@ static uint size_to_tail_pattern(MARIA_FILE_BITMAP *bitmap, uint size) } +/* + Return bitmap pattern for tail block where there is size bytes free + + SYNOPSIS + free_size_to_tail_pattern() + bitmap Bitmap + size Requested size + + RETURN + 0, 5, 6, 7 For a description of the bitmap sizes, see the header +*/ + static uint free_size_to_tail_pattern(MARIA_FILE_BITMAP *bitmap, uint size) { if (size >= bitmap->sizes[0]) @@ -310,7 +377,7 @@ static uint free_size_to_tail_pattern(MARIA_FILE_BITMAP *bitmap, uint size) Return size guranteed to be available on a page SYNOPSIS - pattern_to_head_size + pattern_to_head_size() bitmap Bitmap pattern Pattern (0-7) @@ -327,6 +394,15 @@ static inline uint pattern_to_size(MARIA_FILE_BITMAP *bitmap, uint pattern) /* Print bitmap for debugging + + SYNOPSIS + _ma_print_bitmap() + bitmap Bitmap to print + + IMPLEMENTATION + Prints all changed bits since last call to _ma_print_bitmap(). + This is done by having a copy of the last bitmap in + bitmap->map+bitmap->block_size. */ #ifndef DBUG_OFF @@ -342,18 +418,24 @@ static void _ma_print_bitmap(MARIA_FILE_BITMAP *bitmap) uchar *pos, *end, *org_pos; ulong page; - end= bitmap->map+ bitmap->used_size; + end= bitmap->map + bitmap->used_size; DBUG_LOCK_FILE; fprintf(DBUG_FILE,"\nBitmap page changes at page %lu\n", (ulong) bitmap->page); page= (ulong) bitmap->page+1; - for (pos= bitmap->map, org_pos= bitmap->map+bitmap->block_size ; pos < end ; + for (pos= bitmap->map, org_pos= bitmap->map + bitmap->block_size ; + pos < end ; pos+= 6, org_pos+= 6) { ulonglong bits= uint6korr(pos); /* 6 bytes = 6*8/3= 16 patterns */ ulonglong org_bits= uint6korr(org_pos); uint i; + + /* + Test if there is any changes in the next 16 bitmaps (to not have to + loop through all bits if we know they are the same) + */ if (bits != org_bits) { for (i= 0; i < 16 ; i++, bits>>= 3, org_bits>>= 3) @@ -367,7 +449,7 @@ static void _ma_print_bitmap(MARIA_FILE_BITMAP *bitmap) } fputc('\n', DBUG_FILE); DBUG_UNLOCK_FILE; - memcpy(bitmap->map+ bitmap->block_size, bitmap->map, bitmap->block_size); + memcpy(bitmap->map + bitmap->block_size, bitmap->map, bitmap->block_size); } #endif /* DBUG_OFF */ @@ -394,8 +476,9 @@ static void _ma_print_bitmap(MARIA_FILE_BITMAP *bitmap) 1 error (Error writing old bitmap or reading bitmap page) */ -my_bool _ma_read_bitmap_page(MARIA_SHARE *share, MARIA_FILE_BITMAP *bitmap, - ulonglong page) +static my_bool _ma_read_bitmap_page(MARIA_SHARE *share, + MARIA_FILE_BITMAP *bitmap, + ulonglong page) { my_off_t position= page * bitmap->block_size; my_bool res; @@ -407,6 +490,7 @@ my_bool _ma_read_bitmap_page(MARIA_SHARE *share, MARIA_FILE_BITMAP *bitmap, { share->state.state.data_file_length= position + bitmap->block_size; bzero(bitmap->map, bitmap->block_size); + memcpy(bitmap->map + share->block_size - 2, maria_bitmap_marker, 2); bitmap->used_size= 0; DBUG_RETURN(0); } @@ -417,7 +501,7 @@ my_bool _ma_read_bitmap_page(MARIA_SHARE *share, MARIA_FILE_BITMAP *bitmap, bitmap->block_size, bitmap->block_size, 0) == 0; #ifndef DBUG_OFF if (!res) - memcpy(bitmap->map+ bitmap->block_size, bitmap->map, bitmap->block_size); + memcpy(bitmap->map + bitmap->block_size, bitmap->map, bitmap->block_size); #endif DBUG_RETURN(res); } @@ -465,6 +549,10 @@ static my_bool _ma_change_bitmap_page(MARIA_HA *info, move_to_next_bitmap() bitmap Bitmap handle + NOTES + The found bitmap may be full, so calling function may need to call this + repeatedly until it finds enough space. + TODO Add cache of bitmaps to not read something that is not usable @@ -505,7 +593,12 @@ static my_bool move_to_next_bitmap(MARIA_HA *info, MARIA_FILE_BITMAP *bitmap) best_data Pointer to best 6 byte aligned area in bitmap->map best_pos Which bit in *best_data the area starts 0 = first bit pattern, 1 second bit pattern etc + best_bits The original value of the bits at best_pos fill_pattern Bitmap pattern to store in best_data[best_pos] + + NOTES + We mark all pages to be 'TAIL's, which means that + block->page_count is really a row position inside the page. */ static void fill_block(MARIA_FILE_BITMAP *bitmap, @@ -523,7 +616,7 @@ static void fill_block(MARIA_FILE_BITMAP *bitmap, block->empty_space= pattern_to_size(bitmap, best_bits); block->sub_blocks= 1; block->org_bitmap_value= best_bits; - block->used= BLOCKUSED_TAIL; + block->used= BLOCKUSED_TAIL; /* See _ma_bitmap_release_unused() */ /* Mark place used by reading/writing 2 bytes at a time to handle @@ -533,6 +626,8 @@ static void fill_block(MARIA_FILE_BITMAP *bitmap, data= best_data+ best_pos / 8; offset= best_pos & 7; tmp= uint2korr(data); + + /* we turn off the 3 bits and replace them with fill_pattern */ tmp= (tmp & ~(7 << offset)) | (fill_pattern << offset); int2store(data, tmp); bitmap->changed= 1; @@ -546,9 +641,14 @@ static void fill_block(MARIA_FILE_BITMAP *bitmap, SYNOPSIS allocate_head() bitmap bitmap - size Size of block we need to find + size Size of data region we need to store block Store found information here + IMPLEMENTATION + Find the best-fit page to put a region of 'size' + This is defined as the first page of the set of pages + with the smallest free space that can hold 'size'. + RETURN 0 ok (block is updated) 1 error (no space in bitmap; block is not touched) @@ -586,9 +686,10 @@ static my_bool allocate_head(MARIA_FILE_BITMAP *bitmap, uint size, uint pattern= bits & 7; if (pattern <= min_bits) { + /* There is enough space here */ if (pattern == min_bits) { - /* Found perfect match */ + /* There is exactly enough space here, return this page */ best_bits= min_bits; best_data= data; best_pos= i; @@ -596,6 +697,11 @@ static my_bool allocate_head(MARIA_FILE_BITMAP *bitmap, uint size, } if ((int) pattern > (int) best_bits) { + /* + There is more than enough space here and it's better than what + we have found so far. Remember it, as we will choose it if we + don't find anything in this bitmap page. + */ best_bits= pattern; best_data= data; best_pos= i; @@ -603,10 +709,10 @@ static my_bool allocate_head(MARIA_FILE_BITMAP *bitmap, uint size, } } } - if (!best_data) + if (!best_data) /* Found no place */ { if (bitmap->used_size == bitmap->total_size) - DBUG_RETURN(1); + DBUG_RETURN(1); /* No space in bitmap */ /* Allocate data at end of bitmap */ bitmap->used_size+= 6; best_data= data; @@ -655,7 +761,12 @@ static my_bool allocate_tail(MARIA_FILE_BITMAP *bitmap, uint size, /* Skip common patterns We can skip empty pages (if we already found a match) or - the following patterns: 1-4 or 7 + the following patterns: 1-4 (head pages, not suitable for tail) or + 7 (full tail page). See 'Dynamic size records' comment at start of file. + + At the moment we only skip full tail pages (ie, all bits are + set) as this is easy to detect with one simple test and is a + quite common case if we have blobs. */ if ((!bits && best_data) || bits == LL(0xffffffffffff)) @@ -888,6 +999,13 @@ static ulong allocate_full_pages(MARIA_FILE_BITMAP *bitmap, /* Find right bitmap and position for head block + SYNOPSIS + find_head() + info Maria handler + length Size of data region we need store + position Position in bitmap_blocks where to store the + information for the head block. + RETURN 0 ok 1 error @@ -897,7 +1015,10 @@ static my_bool find_head(MARIA_HA *info, uint length, uint position) { MARIA_FILE_BITMAP *bitmap= &info->s->bitmap; MARIA_BITMAP_BLOCK *block; - /* There is always place for head blocks in bitmap_blocks */ + /* + There is always place for the head block in bitmap_blocks as these are + preallocated at _ma_init_block_record(). + */ block= dynamic_element(&info->bitmap_blocks, position, MARIA_BITMAP_BLOCK *); while (allocate_head(bitmap, length, block)) @@ -910,6 +1031,13 @@ static my_bool find_head(MARIA_HA *info, uint length, uint position) /* Find right bitmap and position for tail + SYNOPSIS + find_tail() + info Maria handler + length Size of data region we need store + position Position in bitmap_blocks where to store the + information for the head block. + RETURN 0 ok 1 error @@ -936,8 +1064,15 @@ static my_bool find_tail(MARIA_HA *info, uint length, uint position) /* Find right bitmap and position for full blocks in one extent + SYNOPSIS + find_mid() + info Maria handler. + pages How many pages to allocate. + position Position in bitmap_blocks where to store the + information for the head block. NOTES This is used to allocate the main extent after the 'head' block + (Ie, the middle part of the head-middle-tail entry) RETURN 0 ok @@ -962,6 +1097,11 @@ static my_bool find_mid(MARIA_HA *info, ulong pages, uint position) /* Find right bitmap and position for putting a blob + SYNOPSIS + find_blob() + info Maria handler. + length Length of the blob + NOTES The extents are stored last in info->bitmap_blocks @@ -1023,6 +1163,19 @@ static my_bool find_blob(MARIA_HA *info, ulong length) } +/* + Find pages to put ALL blobs + + SYNOPSIS + allocate_blobs() + info Maria handler + row Information of what is in the row (from calc_record_size()) + + RETURN + 0 ok + 1 error +*/ + static my_bool allocate_blobs(MARIA_HA *info, MARIA_ROW *row) { ulong *length, *end; @@ -1045,6 +1198,23 @@ static my_bool allocate_blobs(MARIA_HA *info, MARIA_ROW *row) } +/* + Store in the bitmap the new size for a head page + + SYNOPSIS + use_head() + info Maria handler + page Page number to update + (Note that caller guarantees this is in the active + bitmap) + size How much free space is left on the page + block_position In which info->bitmap_block we have the + information about the head block. + + NOTES + This is used on update where we are updating an existing head page +*/ + static void use_head(MARIA_HA *info, ulonglong page, uint size, uint block_position) { @@ -1078,7 +1248,18 @@ static void use_head(MARIA_HA *info, ulonglong page, uint size, /* - Find out where to split the row; + Find out where to split the row (ie, what goes in head, middle, tail etc) + + SYNOPSIS + find_where_to_split_row() + share Maria share + row Information of what is in the row (from calc_record_size()) + extents_length Number of bytes needed to store all extents + split_size Free size on the page (The head length must be less + than this) + + RETURN + row_length for the head block. */ static uint find_where_to_split_row(MARIA_SHARE *share, MARIA_ROW *row, @@ -1108,6 +1289,21 @@ static uint find_where_to_split_row(MARIA_SHARE *share, MARIA_ROW *row, } +/* + Find where to write the middle parts of the row and the tail + + SYNOPSIS + write_rest_of_head() + info Maria handler + position Position in bitmap_blocks. Is 0 for rows that needs + full blocks (ie, has a head, middle part and optional tail) + rest_length How much left of the head block to write. + + RETURN + 0 ok + 1 error +*/ + static my_bool write_rest_of_head(MARIA_HA *info, uint position, ulong rest_length) { @@ -1349,6 +1545,23 @@ abort: Clear and reset bits ****************************************************************************/ +/* + Set fill pattern for a page + + set_page_bits() + info Maria handler + bitmap Bitmap handler + page Adress to page + fill_pattern Pattern (not size) for page + + NOTES + Page may not be part of active bitmap + + RETURN + 0 ok + 1 error +*/ + static my_bool set_page_bits(MARIA_HA *info, MARIA_FILE_BITMAP *bitmap, ulonglong page, uint fill_pattern) { @@ -1390,11 +1603,10 @@ static my_bool set_page_bits(MARIA_HA *info, MARIA_FILE_BITMAP *bitmap, Get bitmap pattern for a given page SYNOPSIS - - get_page_bits() - info Maria handler - bitmap Bitmap handler - page Page number + get_page_bits() + info Maria handler + bitmap Bitmap handler + page Page number RETURN 0-7 Bitmap pattern @@ -1432,7 +1644,7 @@ static uint get_page_bits(MARIA_HA *info, MARIA_FILE_BITMAP *bitmap, Mark all pages in a region as free SYNOPSIS - reset_full_page_bits() + _ma_reset_full_page_bits() info Maria handler bitmap Bitmap handler page Start page @@ -1579,6 +1791,11 @@ my_bool _ma_bitmap_release_unused(MARIA_HA *info, MARIA_BITMAP_BLOCKS *blocks) } else bits= block->org_bitmap_value; + + /* + The page has all bits set; The following test is an optimization + to not set the bits to the same value as before. + */ if (bits != FULL_TAIL_PAGE && set_page_bits(info, bitmap, block->page, bits)) goto err; @@ -1638,8 +1855,22 @@ my_bool _ma_bitmap_free_full_pages(MARIA_HA *info, const byte *extents, } +/* + Mark in the bitmap how much free space there is on a page + + SYNOPSIS + _ma_bitmap_set() + info Mari handler + page Adress to page + head 1 if page is a head page, 0 if tail page + empty_space How much empty space there is on page + + RETURN + 0 ok + 1 error +*/ -my_bool _ma_bitmap_set(MARIA_HA *info, ulonglong pos, my_bool head, +my_bool _ma_bitmap_set(MARIA_HA *info, ulonglong page, my_bool head, uint empty_space) { MARIA_FILE_BITMAP *bitmap= &info->s->bitmap; @@ -1651,7 +1882,7 @@ my_bool _ma_bitmap_set(MARIA_HA *info, ulonglong pos, my_bool head, bits= (head ? _ma_free_size_to_head_pattern(bitmap, empty_space) : free_size_to_tail_pattern(bitmap, empty_space)); - res= set_page_bits(info, bitmap, pos, bits); + res= set_page_bits(info, bitmap, page, bits); pthread_mutex_unlock(&info->s->bitmap.bitmap_lock); DBUG_RETURN(res); } @@ -1663,6 +1894,15 @@ my_bool _ma_bitmap_set(MARIA_HA *info, ulonglong pos, my_bool head, NOTES Used in maria_chk + SYNOPSIS + _ma_check_bitmap_data() + info Maria handler + page_type What kind of page this is + page Adress to page + empty_space Empty space on page + bitmap_pattern Store here the pattern that was in the bitmap for the + page. This is always updated. + RETURN 0 ok 1 error @@ -1694,7 +1934,15 @@ my_bool _ma_check_bitmap_data(MARIA_HA *info, /* - Check that bitmap pattern is correct for a page + Check if the page type matches the one that we have in the bitmap + + SYNOPSIS + _ma_check_if_right_bitmap_type() + info Maria handler + page_type What kind of page this is + page Adress to page + bitmap_pattern Store here the pattern that was in the bitmap for the + page. This is always updated. NOTES Used in maria_chk diff --git a/storage/maria/ma_blockrec.c b/storage/maria/ma_blockrec.c index c56522a9072..8b0e700959b 100644 --- a/storage/maria/ma_blockrec.c +++ b/storage/maria/ma_blockrec.c @@ -16,16 +16,19 @@ /* Storage of records in block - Maria will have a LSN at start of each page (including the bitmap page) - Maria will for each row have the additional information: + Some clarifactions about the abbrev used: - TRANSID Transaction ID that last updated row (6 bytes) - VER_PTR Version pointer that points on the UNDO entry that - contains last version of the row versions (7 bytes) + NULL fields -> Fields that may have contain a NULL value. + Not null fields -> Fields that may not contain a NULL value. + Critical fields -> Fields that can't be null and can't be dropped without + causing a table reorganization. + + + Maria will have a LSN at start of each page (excluding the bitmap pages) The different page types that are in a data file are: - Bitmap pages Map of free pages in the next extent (8129 page size + Bitmap pages Map of free pages in the next extent (8192 page size gives us 256M of mapped pages / bitmap) Head page Start of rows are stored on this page. A rowid always points to a head page @@ -43,9 +46,9 @@ Structure of data and tail page: The page has a row directory at end of page to allow us to do deletes - without having to reorganize the page. It also allows us to store some - extra bytes after each row to allow them to grow without having to move - around other rows + without having to reorganize the page. It also allows us to later store + some more bytes after each row to allow them to grow without having to move + around other rows. Page header: @@ -59,7 +62,7 @@ Row data - Row directory of NO entires, that consist of the following for each row + Row directory of NO entries, that consist of the following for each row (in reverse order; ie, first record is stored last): Position 2 bytes Position of row on page @@ -69,7 +72,8 @@ upmost bit of the length could be used for some states of the row (in other words, we should try to keep these reserved) - eof flag 1 byte Reserved for full page read testing + eof flag 1 byte Reserved for full page read testing. (Ie, did the + previous write get the whole block on disk. ---------------- @@ -105,19 +109,27 @@ Total length of length array 1-3 byte Only used if we have char/varchar/blob fields. Row checksum 1 byte Only if table created with checksums - Null_bits .. One bit for each NULL field - Empty_bits .. One bit for each NOT NULL field. This bit is - 0 if the value is 0 or empty string. + Null_bits .. One bit for each NULL field (a field that may + have the value NULL) + Empty_bits .. One bit for each field that may be 'empty'. + (Both for null and not null fields). + This bit is 1 if the value for the field is + 0 or empty string. field_offsets 2 byte/offset - For each 32 field, there is one offset that - points to where the field information starts - in the block. This is to provide fast access - to later field in the row when we only need - to return a small set of fields. - - Things marked above as 'optional' will only be present if the corresponding - bit is set in 'Flag' field. + For each 32'th field, there is one offset + that points to where the field information + starts in the block. This is to provide + fast access to later field in the row + when we only need to return a small + set of fields. + TODO: Implement this. + + Things marked above as 'optional' will only be present if the + corresponding bit is set in 'Flag' field. Flag gives us a way to + get more space on a page when doing page compaction as we don't need + to store TRANSID that have committed before the smallest running + transaction we have in memory. Data in the following order: (Field order is precalculated when table is created) @@ -176,11 +188,6 @@ Nulls_extended_exists 3 Row is split 7 This means that 'Number_of_row_extents' exists - - This would be a way to get more space on a page when doing page - compaction as we don't need to store TRANSID that have committed - before the smallest running transaction we have in memory. - Nulls_extended is the number of new DEFAULT NULL fields in the row compared to the number of DEFAULT NULL fields when the first version of the table was created. If Nulls_extended doesn't exist in the row, @@ -198,8 +205,10 @@ fields. When storing a row, we will mark a dropped field either with a null in the null bit map or in the empty_bits and not store any data for it. + TODO: Add code for handling dropped fields. + - One ROW_EXTENT is coded as: + A ROW EXTENT is range of pages. One ROW_EXTENT is coded as: START_PAGE 5 bytes PAGE_COUNT 2 bytes. High bit is used to indicate tail page/ @@ -248,14 +257,36 @@ #include "maria_def.h" #include "ma_blockrec.h" +/* + Struct for having a cursor over a set of extent. + This is used to loop over all extents for a row when reading + the row data. It's also used to store the tail positions for + a read row to be used by a later update/delete command. +*/ + typedef struct st_maria_extent_cursor { + /* + Pointer to packed byte array of extents for the row. + Format is described above in the header + */ byte *extent; - byte *data_start; /* For error checking */ + /* Where data starts on page; Only for debugging */ + byte *data_start; + /* Position to all tails in the row. Updated when reading a row */ MARIA_RECORD_POS *tail_positions; + /* Current page */ my_off_t page; - uint extent_count, page_count; - uint tail; /* <> 0 if current extent is a tail page */ + /* How many pages in the page region */ + uint page_count; + /* Total number of extents (ie, entries in the 'extent' slot) */ + uint extent_count; + /* <> 0 if current extent is a tail page; Set while using cursor */ + uint tail; + /* + <> 1 if we are working on the first extent (ie, the one that is store in + the row header, not an extent that is stored as part of the row data). + */ my_bool first_extent; } MARIA_EXTENT_CURSOR; @@ -327,38 +358,47 @@ void _ma_init_block_record_data(void) } -my_bool _ma_once_init_block_row(MARIA_SHARE *share, File data_file) +my_bool _ma_once_init_block_record(MARIA_SHARE *share, File data_file) { share->base.max_data_file_length= (((ulonglong) 1 << ((share->base.rec_reflength-1)*8))-1) * share->block_size; #if SIZEOF_OFF_T == 4 - set_if_smaller(max_data_file_length, INT_MAX32); + set_if_smaller(share->base.max_data_file_length, INT_MAX32); #endif return _ma_bitmap_init(share, data_file); } -my_bool _ma_once_end_block_row(MARIA_SHARE *share) +my_bool _ma_once_end_block_record(MARIA_SHARE *share) { int res= _ma_bitmap_end(share); - if (flush_key_blocks(share->key_cache, share->bitmap.file, - share->temporary ? FLUSH_IGNORE_CHANGED : - FLUSH_RELEASE)) - res= 1; - if (share->bitmap.file >= 0 && my_close(share->bitmap.file, MYF(MY_WME))) - res= 1; + if (share->bitmap.file >= 0) + { + if (flush_key_blocks(share->key_cache, share->bitmap.file, + share->temporary ? FLUSH_IGNORE_CHANGED : + FLUSH_RELEASE)) + res= 1; + if (my_close(share->bitmap.file, MYF(MY_WME))) + res= 1; + /* + Trivial assignment to guard against multiple invocations + (May happen if file are closed but we want to keep the maria object + around a bit longer) + */ + share->bitmap.file= -1; + } return res; } /* Init info->cur_row structure */ -my_bool _ma_init_block_row(MARIA_HA *info) +my_bool _ma_init_block_record(MARIA_HA *info) { MARIA_ROW *row= &info->cur_row, *new_row= &info->new_row; - DBUG_ENTER("_ma_init_block_row"); + DBUG_ENTER("_ma_init_block_record"); if (!my_multi_malloc(MY_WME, &row->empty_bits_buffer, info->s->base.pack_bytes, @@ -398,12 +438,18 @@ my_bool _ma_init_block_row(MARIA_HA *info) } -void _ma_end_block_row(MARIA_HA *info) +void _ma_end_block_record(MARIA_HA *info) { - DBUG_ENTER("_ma_end_block_row"); + DBUG_ENTER("_ma_end_block_record"); my_free((gptr) info->cur_row.empty_bits_buffer, MYF(MY_ALLOW_ZERO_PTR)); delete_dynamic(&info->bitmap_blocks); my_free((gptr) info->cur_row.extents, MYF(MY_ALLOW_ZERO_PTR)); + /* + The data file is closed, when needed, in ma_once_end_block_record(). + The following protects us from doing an extra, not allowed, close + in maria_close() + */ + info->dfile= -1; DBUG_VOID_RETURN; } @@ -412,7 +458,19 @@ void _ma_end_block_row(MARIA_HA *info) Helper functions ****************************************************************************/ -static inline uint empty_pos_after_row(byte *dir) +/* + Return the next used byte on the page after a directory entry. + + SYNOPSIS + start_of_next_entry() + dir Directory entry to be used + + RETURN + # Position in page where next entry starts. + Everything between the '*dir' and this are free to be used. +*/ + +static inline uint start_of_next_entry(byte *dir) { byte *prev; /* @@ -427,6 +485,18 @@ static inline uint empty_pos_after_row(byte *dir) } +/* + Check that a region is all zero + + SYNOPSIS + check_if_zero() + pos Start of memory to check + length length of memory region + + NOTES + Used mainly to detect rows with wrong extent information +*/ + static my_bool check_if_zero(byte *pos, uint length) { byte *end; @@ -438,7 +508,7 @@ static my_bool check_if_zero(byte *pos, uint length) /* - Find free postion in directory + Find free position in directory SYNOPSIS find_free_position() @@ -450,7 +520,7 @@ static my_bool check_if_zero(byte *pos, uint length) all empty space, including the found block. NOTES - If there is a free directory entry (entry with postion == 0), + If there is a free directory entry (entry with position == 0), then use it and change it to be the size of the empty block after the previous entry. This guarantees that all row entries are stored on disk in inverse directory order, which makes life easier for @@ -473,7 +543,7 @@ static my_bool check_if_zero(byte *pos, uint length) static byte *find_free_position(byte *buff, uint block_size, uint *res_rownr, uint *res_length, uint *empty_space) { - uint max_entry= (uint) ((uchar*) buff)[DIR_ENTRY_OFFSET]; + uint max_entry= (uint) ((uchar*) buff)[DIR_COUNT_OFFSET]; uint entry, length, first_pos; byte *dir, *end; DBUG_ENTER("find_free_position"); @@ -482,22 +552,23 @@ static byte *find_free_position(byte *buff, uint block_size, uint *res_rownr, dir= (buff + block_size - DIR_ENTRY_SIZE * max_entry - PAGE_SUFFIX_SIZE); end= buff + block_size - PAGE_SUFFIX_SIZE - DIR_ENTRY_SIZE; - first_pos= PAGE_HEADER_SIZE; *empty_space= uint2korr(buff + EMPTY_SPACE_OFFSET); /* Search after first empty position */ + first_pos= PAGE_HEADER_SIZE; for (entry= 0 ; dir <= end ; end-= DIR_ENTRY_SIZE, entry++) { - if (end[0] == 0 && end[1] == 0) /* Found not used entry */ + uint tmp= uint2korr(end); + if (!tmp) /* Found not used entry */ { - length= empty_pos_after_row(end) - first_pos; + length= start_of_next_entry(end) - first_pos; int2store(end, first_pos); /* Update dir entry */ int2store(end + 2, length); *res_rownr= entry; *res_length= length; DBUG_RETURN(end); } - first_pos= uint2korr(end) + uint2korr(end + 2); + first_pos= tmp + uint2korr(end + 2); } /* No empty places in dir; create a new one */ dir= end; @@ -513,7 +584,7 @@ static byte *find_free_position(byte *buff, uint block_size, uint *res_rownr, uint2korr(end + DIR_ENTRY_SIZE+ 2)); *empty_space= uint2korr(buff + EMPTY_SPACE_OFFSET); } - buff[DIR_ENTRY_OFFSET]= (byte) (uchar) max_entry+1; + buff[DIR_COUNT_OFFSET]= (byte) (uchar) max_entry+1; length= (uint) (dir - buff - first_pos); DBUG_ASSERT(length <= *empty_space - DIR_ENTRY_SIZE); int2store(dir, first_pos); @@ -551,7 +622,7 @@ static void calc_record_size(MARIA_HA *info, const byte *record, { MARIA_SHARE *share= info->s; byte *field_length_data; - MARIA_COLUMNDEF *rec, *end_field; + MARIA_COLUMNDEF *column, *end_column; uint *null_field_lengths= row->null_field_lengths; ulong *blob_lengths= row->blob_lengths; @@ -562,56 +633,56 @@ static void calc_record_size(MARIA_HA *info, const byte *record, bzero(row->empty_bits_buffer, share->base.pack_bytes); row->empty_bits= row->empty_bits_buffer; field_length_data= row->field_lengths; - for (rec= share->rec + share->base.fixed_not_null_fields, - end_field= share->rec + share->base.fields; - rec < end_field; rec++, null_field_lengths++) + for (column= share->columndef + share->base.fixed_not_null_fields, + end_column= share->columndef + share->base.fields; + column < end_column; column++, null_field_lengths++) { - if ((record[rec->null_pos] & rec->null_bit)) + if ((record[column->null_pos] & column->null_bit)) { - if (rec->type != FIELD_BLOB) + if (column->type != FIELD_BLOB) *null_field_lengths= 0; else *blob_lengths++= 0; continue; } - switch ((enum en_fieldtype) rec->type) { + switch ((enum en_fieldtype) column->type) { case FIELD_CHECK: case FIELD_NORMAL: /* Fixed length field */ case FIELD_ZERO: - DBUG_ASSERT(rec->empty_bit == 0); + DBUG_ASSERT(column->empty_bit == 0); /* fall through */ case FIELD_SKIP_PRESPACE: /* Not packed */ - row->normal_length+= rec->length; - *null_field_lengths= rec->length; + row->normal_length+= column->length; + *null_field_lengths= column->length; break; case FIELD_SKIP_ZERO: /* Fixed length field */ - if (memcmp(record+ rec->offset, maria_zero_string, - rec->length) == 0) + if (memcmp(record+ column->offset, maria_zero_string, + column->length) == 0) { - row->empty_bits[rec->empty_pos] |= rec->empty_bit; + row->empty_bits[column->empty_pos] |= column->empty_bit; *null_field_lengths= 0; } else { - row->normal_length+= rec->length; - *null_field_lengths= rec->length; + row->normal_length+= column->length; + *null_field_lengths= column->length; } break; case FIELD_SKIP_ENDSPACE: /* CHAR */ { const char *pos, *end; - for (pos= record + rec->offset, end= pos + rec->length; + for (pos= record + column->offset, end= pos + column->length; end > pos && end[-1] == ' '; end--) ; if (pos == end) /* If empty string */ { - row->empty_bits[rec->empty_pos]|= rec->empty_bit; + row->empty_bits[column->empty_pos]|= column->empty_bit; *null_field_lengths= 0; } else { uint length= (end - pos); - if (rec->length <= 255) + if (column->length <= 255) *field_length_data++= (byte) (uchar) length; else { @@ -626,11 +697,11 @@ static void calc_record_size(MARIA_HA *info, const byte *record, case FIELD_VARCHAR: { uint length, field_length_data_length; - const byte *field_pos= record + rec->offset; + const byte *field_pos= record + column->offset; /* 256 is correct as this includes the length byte */ field_length_data[0]= field_pos[0]; - if (rec->length <= 256) + if (column->length <= 256) { length= (uint) (uchar) *field_pos; field_length_data_length= 1; @@ -644,7 +715,7 @@ static void calc_record_size(MARIA_HA *info, const byte *record, *null_field_lengths= length; if (!length) { - row->empty_bits[rec->empty_pos]|= rec->empty_bit; + row->empty_bits[column->empty_pos]|= column->empty_bit; break; } row->varchar_length+= length; @@ -654,13 +725,13 @@ static void calc_record_size(MARIA_HA *info, const byte *record, } case FIELD_BLOB: { - const byte *field_pos= record + rec->offset; - uint size_length= rec->length - maria_portable_sizeof_char_ptr; + const byte *field_pos= record + column->offset; + uint size_length= column->length - portable_sizeof_char_ptr; ulong blob_length= _ma_calc_blob_length(size_length, field_pos); *blob_lengths++= blob_length; if (!blob_length) - row->empty_bits[rec->empty_pos]|= rec->empty_bit; + row->empty_bits[column->empty_pos]|= column->empty_bit; else { row->blob_length+= blob_length; @@ -709,7 +780,7 @@ static void calc_record_size(MARIA_HA *info, const byte *record, static void compact_page(byte *buff, uint block_size, uint rownr, my_bool extend_block) { - uint max_entry= (uint) ((uchar *) buff)[DIR_ENTRY_OFFSET]; + uint max_entry= (uint) ((uchar *) buff)[DIR_COUNT_OFFSET]; uint page_pos, next_free_pos, start_of_found_block, diff, end_of_found_block; byte *dir, *end; DBUG_ENTER("compact_page"); @@ -875,7 +946,7 @@ static my_bool get_head_or_tail_page(MARIA_HA *info, bzero(buff+ PAGE_HEADER_SIZE, block_size - PAGE_HEADER_SIZE); buff[PAGE_TYPE_OFFSET]= (byte) page_type; - buff[DIR_ENTRY_OFFSET]= 1; + buff[DIR_COUNT_OFFSET]= 1; res->buff= buff; res->empty_space= res->length= (block_size - PAGE_OVERHEAD_SIZE); res->data= (buff + PAGE_HEADER_SIZE); @@ -956,12 +1027,18 @@ static my_bool write_tail(MARIA_HA *info, DBUG_PRINT("enter", ("page: %lu length: %u", (ulong) block->page, length)); - info->keybuff_used= 1; + info->keyread_buff_used= 1; if (get_head_or_tail_page(info, block, info->keyread_buff, length, TAIL_PAGE, &row_pos)) DBUG_RETURN(1); memcpy(row_pos.data, row_part, length); + /* + Don't allocate smaller block than MIN_TAIL_SIZE (we want to give rows + some place to grow in the future) + */ + if (length < MIN_TAIL_SIZE) + length= MIN_TAIL_SIZE; int2store(row_pos.dir + 2, length); empty_space= row_pos.empty_space - length; int2store(row_pos.buff + EMPTY_SPACE_OFFSET, empty_space); @@ -969,10 +1046,10 @@ static my_bool write_tail(MARIA_HA *info, /* If there is less directory entries free than number of possible tails we can write for a row, we mark the page full to ensure that we don't - during _ma_bitmap_find_place() allocate more entires on the tail page + during _ma_bitmap_find_place() allocate more entries on the tail page than it can hold */ - block->empty_space= ((uint) ((uchar*) row_pos.buff)[DIR_ENTRY_OFFSET] <= + block->empty_space= ((uint) ((uchar*) row_pos.buff)[DIR_COUNT_OFFSET] <= MAX_ROWS_PER_PAGE - 1 - info->s->base.blobs ? empty_space : 0); block->used= BLOCKUSED_USED | BLOCKUSED_TAIL; @@ -1015,7 +1092,7 @@ static my_bool write_full_pages(MARIA_HA *info, (ulong) length, (ulong) block->page, (ulong) block->page_count)); - info->keybuff_used= 1; + info->keyread_buff_used= 1; page= block->page; page_count= block->page_count; @@ -1134,7 +1211,7 @@ static my_bool write_block_record(MARIA_HA *info, const byte *record, byte *page_buff; MARIA_BITMAP_BLOCK *block, *head_block; MARIA_SHARE *share; - MARIA_COLUMNDEF *rec, *end_field; + MARIA_COLUMNDEF *column, *end_column; uint block_size, flag; ulong *blob_lengths; my_off_t position; @@ -1219,16 +1296,17 @@ static my_bool write_block_record(MARIA_HA *info, const byte *record, } /* Copy fields that has fixed lengths (primary key etc) */ - for (rec= share->rec, end_field= rec + share->base.fixed_not_null_fields; - rec < end_field; rec++) + for (column= share->columndef, + end_column= column + share->base.fixed_not_null_fields; + column < end_column; column++) { - if (!tmp_data_used && tmp_data + rec->length > end_of_data) + if (!tmp_data_used && tmp_data + column->length > end_of_data) { tmp_data_used= tmp_data; tmp_data= info->rec_buff; } - memcpy(tmp_data, record + rec->offset, rec->length); - tmp_data+= rec->length; + memcpy(tmp_data, record + column->offset, column->length); + tmp_data+= column->length; } /* Copy length of data for variable length fields */ @@ -1242,26 +1320,26 @@ static my_bool write_block_record(MARIA_HA *info, const byte *record, tmp_data+= row->field_lengths_length; /* Copy variable length fields and fields with null/zero */ - for (end_field= share->rec + share->base.fields - share->base.blobs; - rec < end_field ; - rec++) + for (end_column= share->columndef + share->base.fields - share->base.blobs; + column < end_column ; + column++) { const byte *field_pos; ulong length; - if ((record[rec->null_pos] & rec->null_bit) || - (row->empty_bits[rec->empty_pos] & rec->empty_bit)) + if ((record[column->null_pos] & column->null_bit) || + (row->empty_bits[column->empty_pos] & column->empty_bit)) continue; - field_pos= record + rec->offset; - switch ((enum en_fieldtype) rec->type) { + field_pos= record + column->offset; + switch ((enum en_fieldtype) column->type) { case FIELD_NORMAL: /* Fixed length field */ case FIELD_SKIP_PRESPACE: case FIELD_SKIP_ZERO: /* Fixed length field */ - length= rec->length; + length= column->length; break; case FIELD_SKIP_ENDSPACE: /* CHAR */ /* Char that is space filled */ - if (rec->length <= 255) + if (column->length <= 255) length= (uint) (uchar) *field_length_data++; else { @@ -1270,7 +1348,7 @@ static my_bool write_block_record(MARIA_HA *info, const byte *record, } break; case FIELD_VARCHAR: - if (rec->length <= 256) + if (column->length <= 256) { length= (uint) (uchar) *field_length_data++; field_pos++; /* Skip length byte */ @@ -1298,21 +1376,21 @@ static my_bool write_block_record(MARIA_HA *info, const byte *record, block= head_block + head_block->sub_blocks; /* Point to first blob data */ - end_field= rec + share->base.blobs; + end_column= column + share->base.blobs; blob_lengths= row->blob_lengths; if (!tmp_data_used) { /* Still room on page; Copy as many blobs we can into this page */ data= tmp_data; - for (; rec < end_field && *blob_lengths < (ulong) (end_of_data - data); - rec++, blob_lengths++) + for (; column < end_column && *blob_lengths < (ulong) (end_of_data - data); + column++, blob_lengths++) { byte *tmp_pos; uint length; if (!*blob_lengths) /* Null or "" */ continue; - length= rec->length - maria_portable_sizeof_char_ptr; - memcpy_fixed((byte*) &tmp_pos, record + rec->offset + length, + length= column->length - portable_sizeof_char_ptr; + memcpy_fixed((byte*) &tmp_pos, record + column->offset + length, sizeof(char*)); memcpy(data, tmp_pos, *blob_lengths); data+= *blob_lengths; @@ -1342,7 +1420,7 @@ static my_bool write_block_record(MARIA_HA *info, const byte *record, int2store(page_buff + EMPTY_SPACE_OFFSET, row_pos->empty_space); /* Mark in bitmaps how the current page was actually used */ head_block->empty_space= row_pos->empty_space; - if (page_buff[DIR_ENTRY_OFFSET] == (char) MAX_ROWS_PER_PAGE) + if (page_buff[DIR_COUNT_OFFSET] == (char) MAX_ROWS_PER_PAGE) head_block->empty_space= 0; /* Page is full */ head_block->used= BLOCKUSED_USED; } @@ -1362,14 +1440,14 @@ static my_bool write_block_record(MARIA_HA *info, const byte *record, if (row_extents_in_use) { - if (rec != end_field) /* If blob fields */ + if (column != end_column) /* If blob fields */ { - MARIA_COLUMNDEF *save_rec= rec; + MARIA_COLUMNDEF *save_column= column; MARIA_BITMAP_BLOCK *save_block= block; MARIA_BITMAP_BLOCK *end_block; ulong *save_blob_lengths= blob_lengths; - for (; rec < end_field; rec++, blob_lengths++) + for (; column < end_column; column++, blob_lengths++) { byte *blob_pos; if (!*blob_lengths) /* Null or "" */ @@ -1377,8 +1455,8 @@ static my_bool write_block_record(MARIA_HA *info, const byte *record, if (block[block->sub_blocks - 1].used & BLOCKUSED_TAIL) { uint length; - length= rec->length - maria_portable_sizeof_char_ptr; - memcpy_fixed((byte *) &blob_pos, record + rec->offset + length, + length= column->length - portable_sizeof_char_ptr; + memcpy_fixed((byte *) &blob_pos, record + column->offset + length, sizeof(char*)); length= *blob_lengths % FULL_PAGE_SIZE(block_size); /* tail size */ if (write_tail(info, block + block->sub_blocks-1, @@ -1395,7 +1473,7 @@ static my_bool write_block_record(MARIA_HA *info, const byte *record, block->used|= BLOCKUSED_USED; } } - rec= save_rec; + column= save_column; block= save_block; blob_lengths= save_blob_lengths; } @@ -1593,15 +1671,15 @@ static my_bool write_block_record(MARIA_HA *info, const byte *record, } /* Write rest of blobs (data, but no tails as they are already written) */ - for (; rec < end_field; rec++, blob_lengths++) + for (; column < end_column; column++, blob_lengths++) { byte *blob_pos; uint length; ulong blob_length; if (!*blob_lengths) /* Null or "" */ continue; - length= rec->length - maria_portable_sizeof_char_ptr; - memcpy_fixed((byte*) &blob_pos, record + rec->offset + length, + length= column->length - portable_sizeof_char_ptr; + memcpy_fixed((byte*) &blob_pos, record + column->offset + length, sizeof(char*)); /* remove tail part */ blob_length= *blob_lengths; @@ -1705,7 +1783,7 @@ my_bool _ma_write_abort_block_record(MARIA_HA *info) if (delete_head_or_tail(info, ma_recordpos_to_page(info->cur_row.lastpos), - ma_recordpos_to_offset(info->cur_row.lastpos), 1)) + ma_recordpos_to_dir_entry(info->cur_row.lastpos), 1)) res= 1; for (block= blocks->block + 1, end= block + blocks->count - 1; block < end; block++) @@ -1764,7 +1842,7 @@ my_bool _ma_update_block_record(MARIA_HA *info, MARIA_RECORD_POS record_pos, info->buff, block_size, block_size, 0))) DBUG_RETURN(1); org_empty_size= uint2korr(buff + EMPTY_SPACE_OFFSET); - rownr= ma_recordpos_to_offset(record_pos); + rownr= ma_recordpos_to_dir_entry(record_pos); dir= (buff + block_size - DIR_ENTRY_SIZE * rownr - DIR_ENTRY_SIZE - PAGE_SUFFIX_SIZE); @@ -1785,8 +1863,8 @@ my_bool _ma_update_block_record(MARIA_HA *info, MARIA_RECORD_POS record_pos, if (new_row->total_length > length) { /* See if there is empty space after */ - if (rownr != (uint) ((uchar *) buff)[DIR_ENTRY_OFFSET] - 1) - empty= empty_pos_after_row(dir) - (offset + length); + if (rownr != (uint) ((uchar *) buff)[DIR_COUNT_OFFSET] - 1) + empty= start_of_next_entry(dir) - (offset + length); if (new_row->total_length > length + empty) { compact_page(buff, info->s->block_size, rownr, 1); @@ -1876,14 +1954,14 @@ static my_bool delete_head_or_tail(MARIA_HA *info, my_off_t position; DBUG_ENTER("delete_head_or_tail"); - info->keybuff_used= 1; + info->keyread_buff_used= 1; if (!(buff= key_cache_read(share->key_cache, info->dfile, page * block_size, 0, info->keyread_buff, block_size, block_size, 0))) DBUG_RETURN(1); - number_of_records= (uint) ((uchar *) buff)[DIR_ENTRY_OFFSET]; + number_of_records= (uint) ((uchar *) buff)[DIR_COUNT_OFFSET]; #ifdef SANITY_CHECKS if (record_number >= number_of_records || record_number > ((block_size - LSN_SIZE - PAGE_TYPE_SIZE - 1 - @@ -1911,7 +1989,7 @@ static my_bool delete_head_or_tail(MARIA_HA *info, dir+= DIR_ENTRY_SIZE; empty_space+= DIR_ENTRY_SIZE; } while (dir < end && dir[0] == 0 && dir[1] == 0); - buff[DIR_ENTRY_OFFSET]= (byte) (uchar) number_of_records; + buff[DIR_COUNT_OFFSET]= (byte) (uchar) number_of_records; } empty_space+= length; if (number_of_records != 0) @@ -1957,7 +2035,7 @@ static my_bool delete_tails(MARIA_HA *info, MARIA_RECORD_POS *tails) { if (delete_head_or_tail(info, ma_recordpos_to_page(*tails), - ma_recordpos_to_offset(*tails), 0)) + ma_recordpos_to_dir_entry(*tails), 0)) res= 1; } DBUG_RETURN(res); @@ -1978,7 +2056,7 @@ my_bool _ma_delete_block_record(MARIA_HA *info) DBUG_ENTER("_ma_delete_block_record"); if (delete_head_or_tail(info, ma_recordpos_to_page(info->cur_row.lastpos), - ma_recordpos_to_offset(info->cur_row.lastpos), + ma_recordpos_to_dir_entry(info->cur_row.lastpos), 1) || delete_tails(info, info->cur_row.tail_positions)) DBUG_RETURN(1); @@ -2011,7 +2089,7 @@ my_bool _ma_delete_block_record(MARIA_HA *info) static byte *get_record_position(byte *buff, uint block_size, uint record_number, byte **end_of_data) { - uint number_of_records= (uint) ((uchar *) buff)[DIR_ENTRY_OFFSET]; + uint number_of_records= (uint) ((uchar *) buff)[DIR_COUNT_OFFSET]; byte *dir; byte *data; uint offset, length; @@ -2254,7 +2332,7 @@ int _ma_read_block_record2(MARIA_HA *info, byte *record, uint flag, null_bytes, cur_null_bytes, row_extents, field_lengths; my_bool found_blob= 0; MARIA_EXTENT_CURSOR extent; - MARIA_COLUMNDEF *rec, *end_field; + MARIA_COLUMNDEF *column, *end_column; DBUG_ENTER("_ma_read_block_record2"); LINT_INIT(field_lengths); @@ -2347,15 +2425,16 @@ int _ma_read_block_record2(MARIA_HA *info, byte *record, Data now points to start of fixed length field data that can't be null or 'empty'. Note that these fields can't be split over blocks */ - for (rec= share->rec, end_field= rec + share->base.fixed_not_null_fields; - rec < end_field; rec++) + for (column= share->columndef, + end_column= column + share->base.fixed_not_null_fields; + column < end_column; column++) { - uint rec_length= rec->length; + uint column_length= column->length; if (data >= end_of_data && !(data= read_next_extent(info, &extent, &end_of_data))) goto err; - memcpy(record + rec->offset, data, rec_length); - data+= rec_length; + memcpy(record + column->offset, data, column_length); + data+= column_length; } /* Read array of field lengths. This may be stored in several extents */ @@ -2368,18 +2447,19 @@ int _ma_read_block_record2(MARIA_HA *info, byte *record, } /* Read variable length data. Each of these may be split over many extents */ - for (end_field= share->rec + share->base.fields; rec < end_field; rec++) + for (end_column= share->columndef + share->base.fields; + column < end_column; column++) { - enum en_fieldtype type= (enum en_fieldtype) rec->type; - byte *field_pos= record + rec->offset; + enum en_fieldtype type= (enum en_fieldtype) column->type; + byte *field_pos= record + column->offset; /* First check if field is present in record */ - if ((record[rec->null_pos] & rec->null_bit) || - (info->cur_row.empty_bits[rec->empty_pos] & rec->empty_bit)) + if ((record[column->null_pos] & column->null_bit) || + (info->cur_row.empty_bits[column->empty_pos] & column->empty_bit)) { if (type == FIELD_SKIP_ENDSPACE) - bfill(record + rec->offset, rec->length, ' '); + bfill(record + column->offset, column->length, ' '); else - bzero(record + rec->offset, rec->fill_length); + bzero(record + column->offset, column->fill_length); continue; } switch (type) { @@ -2389,14 +2469,14 @@ int _ma_read_block_record2(MARIA_HA *info, byte *record, if (data >= end_of_data && !(data= read_next_extent(info, &extent, &end_of_data))) goto err; - memcpy(field_pos, data, rec->length); - data+= rec->length; + memcpy(field_pos, data, column->length); + data+= column->length; break; case FIELD_SKIP_ENDSPACE: /* CHAR */ { /* Char that is space filled */ uint length; - if (rec->length <= 255) + if (column->length <= 255) length= (uint) (uchar) *field_length_data++; else { @@ -2404,19 +2484,19 @@ int _ma_read_block_record2(MARIA_HA *info, byte *record, field_length_data+= 2; } #ifdef SANITY_CHECKS - if (length > rec->length) + if (length > column->length) goto err; #endif if (read_long_data(info, field_pos, length, &extent, &data, &end_of_data)) DBUG_RETURN(my_errno); - bfill(field_pos + length, rec->length - length, ' '); + bfill(field_pos + length, column->length - length, ' '); break; } case FIELD_VARCHAR: { ulong length; - if (rec->length <= 256) + if (column->length <= 256) { length= (uint) (uchar) (*field_pos++= *field_length_data++); } @@ -2435,7 +2515,7 @@ int _ma_read_block_record2(MARIA_HA *info, byte *record, } case FIELD_BLOB: { - uint size_length= rec->length - maria_portable_sizeof_char_ptr; + uint size_length= column->length - portable_sizeof_char_ptr; ulong blob_length= _ma_calc_blob_length(size_length, field_length_data); if (!found_blob) @@ -2443,17 +2523,17 @@ int _ma_read_block_record2(MARIA_HA *info, byte *record, /* Calculate total length for all blobs */ ulong blob_lengths= 0; byte *length_data= field_length_data; - MARIA_COLUMNDEF *blob_field= rec; + MARIA_COLUMNDEF *blob_field= column; found_blob= 1; - for (; blob_field < end_field; blob_field++) + for (; blob_field < end_column; blob_field++) { uint size_length; if ((record[blob_field->null_pos] & blob_field->null_bit) || (info->cur_row.empty_bits[blob_field->empty_pos] & blob_field->empty_bit)) continue; - size_length= blob_field->length - maria_portable_sizeof_char_ptr; + size_length= blob_field->length - portable_sizeof_char_ptr; blob_lengths+= _ma_calc_blob_length(size_length, length_data); length_data+= size_length; } @@ -2547,7 +2627,7 @@ int _ma_read_block_record(MARIA_HA *info, byte *record, info->cur_row.lastpos= record_pos; page= ma_recordpos_to_page(record_pos) * block_size; - offset= ma_recordpos_to_offset(record_pos); + offset= ma_recordpos_to_dir_entry(record_pos); if (!(buff= key_cache_read(info->s->key_cache, info->dfile, page, 0, info->buff, @@ -2754,7 +2834,7 @@ restart_bitmap_scan: if (((info->scan.page_buff[PAGE_TYPE_OFFSET] & PAGE_TYPE_MASK) != HEAD_PAGE) || (info->scan.number_of_rows= - (uint) (uchar) info->scan.page_buff[DIR_ENTRY_OFFSET]) == 0) + (uint) (uchar) info->scan.page_buff[DIR_COUNT_OFFSET]) == 0) { DBUG_PRINT("error", ("Wrong page header")); DBUG_RETURN((my_errno= HA_ERR_WRONG_IN_RECORD)); @@ -2818,7 +2898,7 @@ my_bool _ma_compare_block_record(MARIA_HA *info __attribute__ ((unused)), static void _ma_print_directory(byte *buff, uint block_size) { - uint max_entry= (uint) ((uchar *) buff)[DIR_ENTRY_OFFSET], row= 0; + uint max_entry= (uint) ((uchar *) buff)[DIR_COUNT_OFFSET], row= 0; uint end_of_prev_row= PAGE_HEADER_SIZE; byte *dir, *end; diff --git a/storage/maria/ma_blockrec.h b/storage/maria/ma_blockrec.h index 54145319b83..9e251a8c59d 100644 --- a/storage/maria/ma_blockrec.h +++ b/storage/maria/ma_blockrec.h @@ -18,11 +18,11 @@ */ #define LSN_SIZE 7 -#define DIRCOUNT_SIZE 1 /* Stores number of rows on page */ +#define DIR_COUNT_SIZE 1 /* Stores number of rows on page */ #define EMPTY_SPACE_SIZE 2 /* Stores empty space on page */ #define PAGE_TYPE_SIZE 1 #define PAGE_SUFFIX_SIZE 0 /* Bytes for page suffix */ -#define PAGE_HEADER_SIZE (LSN_SIZE + DIRCOUNT_SIZE + EMPTY_SPACE_SIZE +\ +#define PAGE_HEADER_SIZE (LSN_SIZE + DIR_COUNT_SIZE + EMPTY_SPACE_SIZE +\ PAGE_TYPE_SIZE) #define PAGE_OVERHEAD_SIZE (PAGE_HEADER_SIZE + DIR_ENTRY_SIZE + \ PAGE_SUFFIX_SIZE) @@ -34,14 +34,18 @@ #define ROW_EXTENT_COUNT_SIZE 2 #define ROW_EXTENT_SIZE (ROW_EXTENT_PAGE_SIZE + ROW_EXTENT_COUNT_SIZE) #define TAIL_BIT 0x8000 /* Bit in page_count to signify tail */ +/* Number of extents reserved MARIA_BITMAP_BLOCKS to store head part */ #define ELEMENTS_RESERVED_FOR_MAIN_PART 4 +/* Fields before 'row->null_field_lengths' used by find_where_to_split_row */ #define EXTRA_LENGTH_FIELDS 3 +/* Size for the different parts in the row header (and head page) */ + #define FLAG_SIZE 1 #define TRANSID_SIZE 6 #define VERPTR_SIZE 7 #define DIR_ENTRY_SIZE 4 -#define FIELD_OFFSET_SIZE 2 +#define FIELD_OFFSET_SIZE 2 /* size of pointers to field starts */ /* Minimum header size needed for a new row */ #define BASE_ROW_HEADER_SIZE FLAG_SIZE @@ -51,8 +55,8 @@ enum en_page_type { UNALLOCATED_PAGE, HEAD_PAGE, TAIL_PAGE, BLOB_PAGE, MAX_PAGE_TYPE }; #define PAGE_TYPE_OFFSET LSN_SIZE -#define DIR_ENTRY_OFFSET LSN_SIZE+PAGE_TYPE_SIZE -#define EMPTY_SPACE_OFFSET (DIR_ENTRY_OFFSET + DIRCOUNT_SIZE) +#define DIR_COUNT_OFFSET LSN_SIZE+PAGE_TYPE_SIZE +#define EMPTY_SPACE_OFFSET (DIR_COUNT_OFFSET + DIR_COUNT_SIZE) #define PAGE_CAN_BE_COMPACTED 128 /* Bit in PAGE_TYPE */ @@ -64,10 +68,15 @@ enum en_page_type { UNALLOCATED_PAGE, HEAD_PAGE, TAIL_PAGE, BLOB_PAGE, MAX_PAGE_ #define ROW_FLAG_EXTENTS 128 #define ROW_FLAG_ALL (1+2+4+8+128) -/* Variables that affects how data pages are utilized */ +/******** Variables that affects how data pages are utilized ********/ + +/* Minium size of tail segment */ #define MIN_TAIL_SIZE 32 -/* Fixed part of Max possible header size; See table in ma_blockrec.c */ +/* + Fixed length part of Max possible header size; See row data structure + table in ma_blockrec.c. +*/ #define MAX_FIXED_HEADER_SIZE (FLAG_SIZE + 3 + ROW_EXTENT_SIZE + 3) #define TRANS_MAX_FIXED_HEADER_SIZE (MAX_FIXED_HEADER_SIZE + \ TRANSID_SIZE + VERPTR_SIZE + \ @@ -77,21 +86,30 @@ enum en_page_type { UNALLOCATED_PAGE, HEAD_PAGE, TAIL_PAGE, BLOB_PAGE, MAX_PAGE_ #define MAX_ROWS_PER_PAGE 255 /* Bits for MARIA_BITMAP_BLOCKS->used */ +/* We stored data on disk in the block */ #define BLOCKUSED_USED 1 +/* Bitmap on disk is block->org_bitmap_value ; Happens only on update */ #define BLOCKUSED_USE_ORG_BITMAP 2 +/* We stored tail data on disk for the block */ #define BLOCKUSED_TAIL 4 -/* defines that affects allocation (density) of data */ +/******* defines that affects allocation (density) of data *******/ -/* If we fill up a block to 75 %, don't create a new tail page for it */ +/* + If the tail part (from the main block or a blob) uses more than 75 % of + the size of page, store the tail on a full page instead of a shared + tail page. +*/ #define MAX_TAIL_SIZE(block_size) ((block_size) *3 / 4) +extern uchar maria_bitmap_marker[2]; + /* Functions to convert MARIA_RECORD_POS to/from page:offset */ -static inline MARIA_RECORD_POS ma_recordpos(ulonglong page, uint offset) +static inline MARIA_RECORD_POS ma_recordpos(ulonglong page, uint dir_entry) { - DBUG_ASSERT(offset <= 255); - return (MARIA_RECORD_POS) ((page << 8) | offset); + DBUG_ASSERT(dir_entry <= 255); + return (MARIA_RECORD_POS) ((page << 8) | dir_entry); } static inline my_off_t ma_recordpos_to_page(MARIA_RECORD_POS record_pos) @@ -99,17 +117,17 @@ static inline my_off_t ma_recordpos_to_page(MARIA_RECORD_POS record_pos) return record_pos >> 8; } -static inline my_off_t ma_recordpos_to_offset(MARIA_RECORD_POS record_pos) +static inline my_off_t ma_recordpos_to_dir_entry(MARIA_RECORD_POS record_pos) { return record_pos & 255; } /* ma_blockrec.c */ void _ma_init_block_record_data(void); -my_bool _ma_once_init_block_row(MARIA_SHARE *share, File dfile); -my_bool _ma_once_end_block_row(MARIA_SHARE *share); -my_bool _ma_init_block_row(MARIA_HA *info); -void _ma_end_block_row(MARIA_HA *info); +my_bool _ma_once_init_block_record(MARIA_SHARE *share, File dfile); +my_bool _ma_once_end_block_record(MARIA_SHARE *share); +my_bool _ma_init_block_record(MARIA_HA *info); +void _ma_end_block_record(MARIA_HA *info); my_bool _ma_update_block_record(MARIA_HA *info, MARIA_RECORD_POS pos, const byte *record); @@ -136,8 +154,6 @@ my_bool _ma_compare_block_record(register MARIA_HA *info, my_bool _ma_bitmap_init(MARIA_SHARE *share, File file); my_bool _ma_bitmap_end(MARIA_SHARE *share); my_bool _ma_flush_bitmap(MARIA_SHARE *share); -my_bool _ma_read_bitmap_page(MARIA_SHARE *share, MARIA_FILE_BITMAP *bitmap, - ulonglong page); my_bool _ma_bitmap_find_place(MARIA_HA *info, MARIA_ROW *row, MARIA_BITMAP_BLOCKS *result_blocks); my_bool _ma_bitmap_release_unused(MARIA_HA *info, MARIA_BITMAP_BLOCKS *blocks); diff --git a/storage/maria/ma_check.c b/storage/maria/ma_check.c index a87de2bf9ed..05f85eab96b 100644 --- a/storage/maria/ma_check.c +++ b/storage/maria/ma_check.c @@ -941,7 +941,7 @@ static void record_pos_to_txt(MARIA_HA *info, my_off_t recpos, else { my_off_t page= ma_recordpos_to_page(recpos); - uint row= ma_recordpos_to_offset(recpos); + uint row= ma_recordpos_to_dir_entry(recpos); char *end= longlong10_to_str(page, buff, 10); *(end++)= ':'; longlong10_to_str(row, end, 10); @@ -1370,6 +1370,9 @@ end: /* Check if layout on a page is ok + + NOTES + This is for rows-in-block format. */ static int check_page_layout(HA_CHECK *param, MARIA_HA *info, @@ -1442,6 +1445,8 @@ static int check_page_layout(HA_CHECK *param, MARIA_HA *info, Check all rows on head page NOTES + This is for rows-in-block format. + Before this, we have already called check_page_layout(), so we know the block is logicaly correct (even if the rows may not be that) @@ -1548,6 +1553,9 @@ static my_bool check_head_page(HA_CHECK *param, MARIA_HA *info, byte *record, } +/* + Check if rows-in-block data file is consistent +*/ static int check_block_record(HA_CHECK *param, MARIA_HA *info, int extend, byte *record) @@ -1638,7 +1646,7 @@ static int check_block_record(HA_CHECK *param, MARIA_HA *info, int extend, DBUG_ASSERT(0); break; case HEAD_PAGE: - row_count= ((uchar*) page_buff)[DIR_ENTRY_OFFSET]; + row_count= ((uchar*) page_buff)[DIR_COUNT_OFFSET]; empty_space= uint2korr(page_buff + EMPTY_SPACE_OFFSET); param->used+= (PAGE_HEADER_SIZE + PAGE_SUFFIX_SIZE + row_count * DIR_ENTRY_SIZE); @@ -1647,7 +1655,7 @@ static int check_block_record(HA_CHECK *param, MARIA_HA *info, int extend, full_dir= row_count == MAX_ROWS_PER_PAGE; break; case TAIL_PAGE: - row_count= ((uchar*) page_buff)[DIR_ENTRY_OFFSET]; + row_count= ((uchar*) page_buff)[DIR_COUNT_OFFSET]; empty_space= uint2korr(page_buff + EMPTY_SPACE_OFFSET); param->used+= (PAGE_HEADER_SIZE + PAGE_SUFFIX_SIZE + row_count * DIR_ENTRY_SIZE); @@ -1712,7 +1720,7 @@ err: } - /* Check that record-link is ok */ +/* Check that record-link is ok */ int maria_chk_data_link(HA_CHECK *param, MARIA_HA *info,int extend) { @@ -4581,7 +4589,7 @@ int maria_recreate_table(HA_CHECK *param, MARIA_HA **org_info, char *filename) MARIA_SHARE share; MARIA_KEYDEF *keyinfo,*key,*key_end; HA_KEYSEG *keysegs,*keyseg; - MARIA_COLUMNDEF *recdef,*rec,*end; + MARIA_COLUMNDEF *columndef,*column,*end; MARIA_UNIQUEDEF *uniquedef,*u_ptr,*u_end; MARIA_STATUS_INFO status_info; uint unpack,key_parts; @@ -4610,7 +4618,7 @@ int maria_recreate_table(HA_CHECK *param, MARIA_HA **org_info, char *filename) my_afree((gptr) keyinfo); DBUG_RETURN(1); } - if (!(recdef=(MARIA_COLUMNDEF*) + if (!(columndef=(MARIA_COLUMNDEF*) my_alloca(sizeof(MARIA_COLUMNDEF)*(share.base.fields+1)))) { my_afree((gptr) keyinfo); @@ -4620,22 +4628,24 @@ int maria_recreate_table(HA_CHECK *param, MARIA_HA **org_info, char *filename) if (!(uniquedef=(MARIA_UNIQUEDEF*) my_alloca(sizeof(MARIA_UNIQUEDEF)*(share.state.header.uniques+1)))) { - my_afree((gptr) recdef); + my_afree((gptr) columndef); my_afree((gptr) keyinfo); my_afree((gptr) keysegs); DBUG_RETURN(1); } /* Copy the column definitions */ - memcpy((byte*) recdef,(byte*) share.rec, + memcpy((byte*) columndef,(byte*) share.columndef, (size_t) (sizeof(MARIA_COLUMNDEF)*(share.base.fields+1))); - for (rec=recdef,end=recdef+share.base.fields; rec != end ; rec++) + for (column=columndef, end= columndef+share.base.fields; + column != end ; + column++) { if (unpack && !(share.options & HA_OPTION_PACK_RECORD) && - rec->type != FIELD_BLOB && - rec->type != FIELD_VARCHAR && - rec->type != FIELD_CHECK) - rec->type=(int) FIELD_NORMAL; + column->type != FIELD_BLOB && + column->type != FIELD_VARCHAR && + column->type != FIELD_CHECK) + column->type=(int) FIELD_NORMAL; } /* Change the new key to point at the saved key segments */ @@ -4710,7 +4720,7 @@ int maria_recreate_table(HA_CHECK *param, MARIA_HA **org_info, char *filename) */ if (maria_create(filename, share.data_file_type, share.base.keys - share.state.header.uniques, - keyinfo, share.base.fields, recdef, + keyinfo, share.base.fields, columndef, share.state.header.uniques, uniquedef, &create_info, HA_DONT_TOUCH_DATA)) @@ -4751,7 +4761,7 @@ int maria_recreate_table(HA_CHECK *param, MARIA_HA **org_info, char *filename) end: my_afree((gptr) uniquedef); my_afree((gptr) keyinfo); - my_afree((gptr) recdef); + my_afree((gptr) columndef); my_afree((gptr) keysegs); DBUG_RETURN(error); } diff --git a/storage/maria/ma_checksum.c b/storage/maria/ma_checksum.c index 671e2a7358b..95555aa3129 100644 --- a/storage/maria/ma_checksum.c +++ b/storage/maria/ma_checksum.c @@ -20,29 +20,32 @@ ha_checksum _ma_checksum(MARIA_HA *info, const byte *record) { ha_checksum crc=0; - MARIA_COLUMNDEF *rec= info->s->rec, *rec_end= rec+ info->s->base.fields; + MARIA_COLUMNDEF *column= info->s->columndef; + MARIA_COLUMNDEF *column_end= column+ info->s->base.fields; if (info->s->base.null_bytes) crc= my_checksum(crc, record, info->s->base.null_bytes); - for ( ; rec != rec_end ; rec++) + for ( ; column != column_end ; column++) { - const byte *pos= record + rec->offset; + const byte *pos= record + column->offset; ulong length; - switch (rec->type) { + switch (column->type) { case FIELD_BLOB: { - length= _ma_calc_blob_length(rec->length- - maria_portable_sizeof_char_ptr, - pos); - memcpy((char*) &pos, pos+rec->length- maria_portable_sizeof_char_ptr, - sizeof(char*)); - break; + uint blob_size_length= column->length- portable_sizeof_char_ptr; + length= _ma_calc_blob_length(blob_size_length, pos); + if (length) + { + memcpy((char*) &pos, pos + blob_size_length, sizeof(char*)); + crc= my_checksum(crc, pos, length); + } + continue; } case FIELD_VARCHAR: { - uint pack_length= HA_VARCHAR_PACKLENGTH(rec->length-1); + uint pack_length= HA_VARCHAR_PACKLENGTH(column->length-1); if (pack_length == 1) length= (ulong) *(uchar*) pos; else @@ -51,10 +54,10 @@ ha_checksum _ma_checksum(MARIA_HA *info, const byte *record) break; } default: - length= rec->length; + length= column->length; break; } - crc= my_checksum(crc, pos ? pos : "", length); + crc= my_checksum(crc, pos, length); } return crc; } diff --git a/storage/maria/ma_close.c b/storage/maria/ma_close.c index b38ce2a8cc3..3278baf1dad 100644 --- a/storage/maria/ma_close.c +++ b/storage/maria/ma_close.c @@ -69,10 +69,8 @@ int maria_close(register MARIA_HA *info) pthread_mutex_unlock(&share->intern_lock); my_free(info->rec_buff, MYF(MY_ALLOW_ZERO_PTR)); - (share->end)(info); + (*share->end)(info); - if (info->s->data_file_type == BLOCK_RECORD) - info->dfile= -1; /* Closed in ma_end_once_block_row */ if (flag) { if (share->kfile >= 0) diff --git a/storage/maria/ma_create.c b/storage/maria/ma_create.c index 00bf949a43c..c38471e06a0 100644 --- a/storage/maria/ma_create.c +++ b/storage/maria/ma_create.c @@ -35,9 +35,9 @@ static int compare_columns(MARIA_COLUMNDEF **a, MARIA_COLUMNDEF **b); Old options is used when recreating database, from maria_chk */ -int maria_create(const char *name, enum data_file_type record_type, +int maria_create(const char *name, enum data_file_type datafile_type, uint keys,MARIA_KEYDEF *keydefs, - uint columns, MARIA_COLUMNDEF *recinfo, + uint columns, MARIA_COLUMNDEF *columndef, uint uniques, MARIA_UNIQUEDEF *uniquedefs, MARIA_CREATE_INFO *ci,uint flags) { @@ -55,12 +55,12 @@ int maria_create(const char *name, enum data_file_type record_type, ulong pack_reclength; ulonglong tot_length,max_rows, tmp; enum en_fieldtype type; - enum data_file_type org_record_type= record_type; + enum data_file_type org_datafile_type= datafile_type; MARIA_SHARE share; MARIA_KEYDEF *keydef,tmp_keydef; MARIA_UNIQUEDEF *uniquedef; HA_KEYSEG *keyseg,tmp_keyseg; - MARIA_COLUMNDEF *rec, *rec_end; + MARIA_COLUMNDEF *column, *end_column; ulong *rec_per_key_part; my_off_t key_root[HA_MAX_POSSIBLE_KEY]; MARIA_CREATE_INFO tmp_create_info; @@ -70,6 +70,7 @@ int maria_create(const char *name, enum data_file_type record_type, DBUG_PRINT("enter", ("keys: %u columns: %u uniques: %u flags: %u", keys, columns, uniques, flags)); + DBUG_ASSERT(maria_block_size && maria_block_size % IO_SIZE == 0); LINT_INIT(dfile); LINT_INIT(file); @@ -89,7 +90,7 @@ int maria_create(const char *name, enum data_file_type record_type, if (flags & HA_DONT_TOUCH_DATA) { - org_record_type= ci->org_data_file_type; + org_datafile_type= ci->org_data_file_type; if (!(ci->old_options & HA_OPTION_TEMP_COMPRESS_RECORD)) options=ci->old_options & (HA_OPTION_COMPRESS_RECORD | HA_OPTION_PACK_RECORD | @@ -117,83 +118,85 @@ int maria_create(const char *name, enum data_file_type record_type, pack_reclength= max_field_lengths= 0; reclength= min_pack_length= ci->null_bytes; - for (rec= recinfo, rec_end= rec + columns ; rec != rec_end ; rec++) + for (column= columndef, end_column= column + columns ; + column != end_column ; + column++) { /* Fill in not used struct parts */ - rec->offset= reclength; - rec->empty_pos= 0; - rec->empty_bit= 0; - rec->fill_length= rec->length; - - reclength+= rec->length; - type= rec->type; - if (type == FIELD_SKIP_PRESPACE && record_type == BLOCK_RECORD) + column->offset= reclength; + column->empty_pos= 0; + column->empty_bit= 0; + column->fill_length= column->length; + + reclength+= column->length; + type= column->type; + if (type == FIELD_SKIP_PRESPACE && datafile_type == BLOCK_RECORD) type= FIELD_NORMAL; /* SKIP_PRESPACE not supported */ if (type != FIELD_NORMAL && type != FIELD_CHECK) { - rec->empty_pos= packed/8; - rec->empty_bit= (1 << (packed & 7)); + column->empty_pos= packed/8; + column->empty_bit= (1 << (packed & 7)); if (type == FIELD_BLOB) { packed++; share.base.blobs++; if (pack_reclength != INT_MAX32) { - if (rec->length == 4+maria_portable_sizeof_char_ptr) + if (column->length == 4+portable_sizeof_char_ptr) pack_reclength= INT_MAX32; else { /* Add max possible blob length */ - pack_reclength+= (1 << ((rec->length- - maria_portable_sizeof_char_ptr)*8)); + pack_reclength+= (1 << ((column->length- + portable_sizeof_char_ptr)*8)); } } - max_field_lengths+= (rec->length - maria_portable_sizeof_char_ptr); + max_field_lengths+= (column->length - portable_sizeof_char_ptr); } else if (type == FIELD_SKIP_PRESPACE || type == FIELD_SKIP_ENDSPACE) { - max_field_lengths+= rec->length > 255 ? 2 : 1; - if (record_type != BLOCK_RECORD) + max_field_lengths+= column->length > 255 ? 2 : 1; + if (datafile_type != BLOCK_RECORD) min_pack_length++; packed++; } else if (type == FIELD_VARCHAR) { - varchar_length+= rec->length-1; /* Used for min_pack_length */ + varchar_length+= column->length-1; /* Used for min_pack_length */ pack_reclength++; - if (record_type != BLOCK_RECORD) + if (datafile_type != BLOCK_RECORD) min_pack_length++; max_field_lengths++; packed++; - rec->fill_length= 1; + column->fill_length= 1; /* We must test for 257 as length includes pack-length */ - if (test(rec->length >= 257)) + if (test(column->length >= 257)) { long_varchar_count++; max_field_lengths++; - rec->fill_length= 2; + column->fill_length= 2; } } else if (type == FIELD_SKIP_ZERO) packed++; else { - if (record_type != BLOCK_RECORD || !rec->null_bit) - min_pack_length+= rec->length; - rec->empty_pos= 0; - rec->empty_bit= 0; + if (datafile_type != BLOCK_RECORD || !column->null_bit) + min_pack_length+= column->length; + column->empty_pos= 0; + column->empty_bit= 0; } } else /* FIELD_NORMAL */ { - if (record_type != BLOCK_RECORD || !rec->null_bit) - min_pack_length+= rec->length; - if (!rec->null_bit) + if (datafile_type != BLOCK_RECORD || !column->null_bit) + min_pack_length+= column->length; + if (!column->null_bit) { share.base.fixed_not_null_fields++; - share.base.fixed_not_null_fields_length+= rec->length; + share.base.fixed_not_null_fields_length+= column->length; } } } @@ -203,14 +206,14 @@ int maria_create(const char *name, enum data_file_type record_type, Not optimal packing, try to remove a 1 byte length zero-field as this will get same record length, but smaller pack overhead */ - while (rec != recinfo) + while (column != columndef) { - rec--; - if (rec->type == (int) FIELD_SKIP_ZERO && rec->length == 1) + column--; + if (column->type == (int) FIELD_SKIP_ZERO && column->length == 1) { - rec->type=(int) FIELD_NORMAL; - rec->empty_pos= 0; - rec->empty_bit= 0; + column->type=(int) FIELD_NORMAL; + column->empty_pos= 0; + column->empty_bit= 0; packed--; min_pack_length++; break; @@ -226,12 +229,12 @@ int maria_create(const char *name, enum data_file_type record_type, if (pack_reclength != INT_MAX32) pack_reclength+= max_field_lengths + long_varchar_count; - if (packed && record_type == STATIC_RECORD) - record_type= BLOCK_RECORD; - if (record_type == DYNAMIC_RECORD) + if (packed && datafile_type == STATIC_RECORD) + datafile_type= BLOCK_RECORD; + if (datafile_type == DYNAMIC_RECORD) options|= HA_OPTION_PACK_RECORD; /* Must use packed records */ - if (record_type == STATIC_RECORD) + if (datafile_type == STATIC_RECORD) { /* We can't use checksum with static length rows */ flags&= ~HA_CREATE_CHECKSUM; @@ -275,24 +278,24 @@ int maria_create(const char *name, enum data_file_type record_type, } else if (!ci->max_rows) { - if (record_type == BLOCK_RECORD) + if (datafile_type == BLOCK_RECORD) { uint rows_per_page= ((maria_block_size - PAGE_OVERHEAD_SIZE) / (min_pack_length + extra_header_size + DIR_ENTRY_SIZE)); ulonglong data_file_length= ci->data_file_length; - if (data_file_length) + if (!data_file_length) data_file_length= ((((ulonglong) 1 << ((BLOCK_RECORD_POINTER_SIZE-1) * 8)) -1)); if (rows_per_page > 0) { set_if_smaller(rows_per_page, MAX_ROWS_PER_PAGE); - ci->max_rows= ci->data_file_length / maria_block_size * rows_per_page; + ci->max_rows= data_file_length / maria_block_size * rows_per_page; } else - ci->max_rows= ci->data_file_length / (min_pack_length + - extra_header_size + - DIR_ENTRY_SIZE); + ci->max_rows= data_file_length / (min_pack_length + + extra_header_size + + DIR_ENTRY_SIZE); } else ci->max_rows=(ha_rows) (ci->data_file_length/(min_pack_length + @@ -301,7 +304,7 @@ int maria_create(const char *name, enum data_file_type record_type, 3 : 0))); } max_rows= (ulonglong) ci->max_rows; - if (record_type == BLOCK_RECORD) + if (datafile_type == BLOCK_RECORD) { /* The + 1 is for record position withing page */ pointer= maria_get_pointer_length((ci->data_file_length / @@ -314,7 +317,7 @@ int maria_create(const char *name, enum data_file_type record_type, } else { - if (record_type != STATIC_RECORD) + if (datafile_type != STATIC_RECORD) pointer= maria_get_pointer_length(ci->data_file_length, maria_data_pointer_size); else @@ -324,7 +327,7 @@ int maria_create(const char *name, enum data_file_type record_type, } real_reclength=reclength; - if (record_type == STATIC_RECORD) + if (datafile_type == STATIC_RECORD) { if (reclength <= pointer) reclength=pointer+1; /* reserve place for delete link */ @@ -533,7 +536,12 @@ int maria_create(const char *name, enum data_file_type record_type, key_segs) share.state.rec_per_key_part[key_segs-1]=1L; length+=key_length; - if (length >= HA_MAX_KEY_BUFF) + /* + A key can't be longer than than half a index block (as we have + to be able to put at least 2 keys on an index block for the key + algorithms to work). + */ + if (length > maria_max_key_length()) { my_errno=HA_WRONG_CREATE_OPTION; goto err_no_lock; @@ -592,8 +600,8 @@ int maria_create(const char *name, enum data_file_type record_type, mi_int2store(share.state.header.state_info_length,MARIA_STATE_INFO_SIZE); mi_int2store(share.state.header.base_info_length,MARIA_BASE_INFO_SIZE); mi_int2store(share.state.header.base_pos,base_pos); - share.state.header.data_file_type= record_type; - share.state.header.org_data_file_type= org_record_type; + share.state.header.data_file_type= datafile_type; + share.state.header.org_data_file_type= org_datafile_type; share.state.header.language= (ci->language ? ci->language : default_charset_info->number); @@ -653,7 +661,7 @@ int maria_create(const char *name, enum data_file_type record_type, share.base.max_data_file_length= (my_off_t) ci->data_file_length; } - if (record_type == BLOCK_RECORD) + if (datafile_type == BLOCK_RECORD) share.base.min_block_length= share.base.min_row_length; else { @@ -869,21 +877,23 @@ int maria_create(const char *name, enum data_file_type record_type, } } DBUG_PRINT("info", ("write field definitions")); - if (record_type == BLOCK_RECORD) + if (datafile_type == BLOCK_RECORD) { /* Store columns in a more efficent order */ MARIA_COLUMNDEF **col_order, **pos; if (!(col_order= (MARIA_COLUMNDEF**) my_malloc(share.base.fields * - sizeof(MARIA_COLUMNDEF*), - MYF(MY_WME)))) + sizeof(MARIA_COLUMNDEF*), + MYF(MY_WME)))) goto err; - for (rec= recinfo, pos= col_order ; rec != rec_end ; rec++, pos++) - *pos= rec; + for (column= columndef, pos= col_order ; + column != end_column ; + column++, pos++) + *pos= column; qsort(col_order, share.base.fields, sizeof(*col_order), (qsort_cmp) compare_columns); for (i=0 ; i < share.base.fields ; i++) { - if (_ma_recinfo_write(file, col_order[i])) + if (_ma_columndef_write(file, col_order[i])) { my_free((gptr) col_order, MYF(0)); goto err; @@ -894,7 +904,7 @@ int maria_create(const char *name, enum data_file_type record_type, else { for (i=0 ; i < share.base.fields ; i++) - if (_ma_recinfo_write(file, &recinfo[i])) + if (_ma_columndef_write(file, &columndef[i])) goto err; } @@ -1026,9 +1036,9 @@ static int compare_columns(MARIA_COLUMNDEF **a_ptr, MARIA_COLUMNDEF **b_ptr) MARIA_COLUMNDEF *a= *a_ptr, *b= *b_ptr; enum en_fieldtype a_type, b_type; - a_type= (a->type == FIELD_NORMAL || a->type == FIELD_CHECK ? + a_type= ((a->type == FIELD_NORMAL || a->type == FIELD_CHECK) ? FIELD_NORMAL : a->type); - b_type= (b->type == FIELD_NORMAL || b->type == FIELD_CHECK ? + b_type= ((b->type == FIELD_NORMAL || b->type == FIELD_CHECK) ? FIELD_NORMAL : b->type); if (a_type == FIELD_NORMAL && !a->null_bit) @@ -1059,15 +1069,8 @@ int _ma_initialize_data_file(File dfile, MARIA_SHARE *share) { if (share->data_file_type == BLOCK_RECORD) { - /* Write one bitmap page */ - byte buff[IO_SIZE]; - uint i; - bzero((char*) buff, sizeof(buff)); - if (my_seek(dfile, 0, SEEK_SET, 0)) + if (my_chsize(dfile, maria_block_size, 0, MYF(MY_WME))) return 1; - for (i= 0 ; i < maria_block_size ; i+= IO_SIZE) - if (my_write(dfile, buff, sizeof(buff), MYF(MY_NABP))) - return 1; share->state.state.data_file_length= maria_block_size; _ma_bitmap_delete_all(share); } diff --git a/storage/maria/ma_delete.c b/storage/maria/ma_delete.c index e06bb454edb..f7b11cb6f48 100644 --- a/storage/maria/ma_delete.c +++ b/storage/maria/ma_delete.c @@ -525,7 +525,7 @@ static int underflow(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, DBUG_DUMP("leaf_buff",leaf_buff,maria_getint(leaf_buff)); buff=info->buff; - info->keybuff_used=1; + info->keyread_buff_used=1; next_keypos=keypos; nod_flag=_ma_test_if_nod(leaf_buff); p_length=nod_flag+2; diff --git a/storage/maria/ma_dynrec.c b/storage/maria/ma_dynrec.c index a3fd323d059..91a1170310a 100644 --- a/storage/maria/ma_dynrec.c +++ b/storage/maria/ma_dynrec.c @@ -883,7 +883,7 @@ uint _ma_rec_pack(MARIA_HA *info, register byte *to, register const byte *from) uint length,new_length,flag,bit,i; char *pos,*end,*startpos,*packpos; enum en_fieldtype type; - reg3 MARIA_COLUMNDEF *rec; + reg3 MARIA_COLUMNDEF *column; MARIA_BLOB *blob; DBUG_ENTER("_ma_rec_pack"); @@ -892,7 +892,7 @@ uint _ma_rec_pack(MARIA_HA *info, register byte *to, register const byte *from) startpos= packpos=to; to+= info->s->base.pack_bytes; blob= info->blobs; - rec= info->s->rec; + column= info->s->columndef; if (info->s->base.null_bytes) { memcpy(to, from, info->s->base.null_bytes); @@ -900,10 +900,10 @@ uint _ma_rec_pack(MARIA_HA *info, register byte *to, register const byte *from) to+= info->s->base.null_bytes; } - for (i=info->s->base.fields ; i-- > 0; from+= length,rec++) + for (i=info->s->base.fields ; i-- > 0; from+= length, column++) { - length=(uint) rec->length; - if ((type = (enum en_fieldtype) rec->type) != FIELD_NORMAL) + length=(uint) column->length; + if ((type = (enum en_fieldtype) column->type) != FIELD_NORMAL) { if (type == FIELD_BLOB) { @@ -912,7 +912,7 @@ uint _ma_rec_pack(MARIA_HA *info, register byte *to, register const byte *from) else { char *temp_pos; - size_t tmp_length=length-maria_portable_sizeof_char_ptr; + size_t tmp_length=length-portable_sizeof_char_ptr; memcpy((byte*) to,from,tmp_length); memcpy_fixed(&temp_pos,from+tmp_length,sizeof(char*)); memcpy(to+tmp_length,temp_pos,(size_t) blob->length); @@ -944,10 +944,10 @@ uint _ma_rec_pack(MARIA_HA *info, register byte *to, register const byte *from) pos++; } new_length=(uint) (end-pos); - if (new_length +1 + test(rec->length > 255 && new_length > 127) + if (new_length +1 + test(column->length > 255 && new_length > 127) < length) { - if (rec->length > 255 && new_length > 127) + if (column->length > 255 && new_length > 127) { to[0]=(char) ((new_length & 127)+128); to[1]=(char) (new_length >> 7); @@ -965,7 +965,7 @@ uint _ma_rec_pack(MARIA_HA *info, register byte *to, register const byte *from) } else if (type == FIELD_VARCHAR) { - uint pack_length= HA_VARCHAR_PACKLENGTH(rec->length -1); + uint pack_length= HA_VARCHAR_PACKLENGTH(column->length -1); uint tmp_length; if (pack_length == 1) { @@ -1018,28 +1018,28 @@ my_bool _ma_rec_check(MARIA_HA *info,const char *record, byte *rec_buff, uint length,new_length,flag,bit,i; char *pos,*end,*packpos,*to; enum en_fieldtype type; - reg3 MARIA_COLUMNDEF *rec; + reg3 MARIA_COLUMNDEF *column; DBUG_ENTER("_ma_rec_check"); packpos=rec_buff; to= rec_buff+info->s->base.pack_bytes; - rec=info->s->rec; + column= info->s->columndef; flag= *packpos; bit=1; record+= info->s->base.null_bytes; to+= info->s->base.null_bytes; - for (i=info->s->base.fields ; i-- > 0; record+= length, rec++) + for (i=info->s->base.fields ; i-- > 0; record+= length, column++) { - length=(uint) rec->length; - if ((type = (enum en_fieldtype) rec->type) != FIELD_NORMAL) + length=(uint) column->length; + if ((type = (enum en_fieldtype) column->type) != FIELD_NORMAL) { if (type == FIELD_BLOB) { uint blob_length= - _ma_calc_blob_length(length-maria_portable_sizeof_char_ptr,record); + _ma_calc_blob_length(length-portable_sizeof_char_ptr,record); if (!blob_length && !(flag & bit)) goto err; if (blob_length) - to+=length - maria_portable_sizeof_char_ptr+ blob_length; + to+=length - portable_sizeof_char_ptr+ blob_length; } else if (type == FIELD_SKIP_ZERO) { @@ -1066,12 +1066,12 @@ my_bool _ma_rec_check(MARIA_HA *info,const char *record, byte *rec_buff, pos++; } new_length=(uint) (end-pos); - if (new_length +1 + test(rec->length > 255 && new_length > 127) + if (new_length +1 + test(column->length > 255 && new_length > 127) < length) { if (!(flag & bit)) goto err; - if (rec->length > 255 && new_length > 127) + if (column->length > 255 && new_length > 127) { if (to[0] != (char) ((new_length & 127)+128) || to[1] != (char) (new_length >> 7)) @@ -1087,7 +1087,7 @@ my_bool _ma_rec_check(MARIA_HA *info,const char *record, byte *rec_buff, } else if (type == FIELD_VARCHAR) { - uint pack_length= HA_VARCHAR_PACKLENGTH(rec->length -1); + uint pack_length= HA_VARCHAR_PACKLENGTH(column->length -1); uint tmp_length; if (pack_length == 1) { @@ -1139,10 +1139,10 @@ err: ulong _ma_rec_unpack(register MARIA_HA *info, register byte *to, byte *from, ulong found_length) { - uint flag,bit,length,rec_length,min_pack_length; + uint flag,bit,length,min_pack_length, column_length; enum en_fieldtype type; byte *from_end,*to_end,*packpos; - reg3 MARIA_COLUMNDEF *rec,*end_field; + reg3 MARIA_COLUMNDEF *column, *end_column; DBUG_ENTER("_ma_rec_unpack"); to_end=to + info->s->base.reclength; @@ -1161,27 +1161,27 @@ ulong _ma_rec_unpack(register MARIA_HA *info, register byte *to, byte *from, min_pack_length-= length; } - for (rec=info->s->rec , end_field=rec+info->s->base.fields ; - rec < end_field ; to+= rec_length, rec++) + for (column= info->s->columndef, end_column= column + info->s->base.fields; + column < end_column ; to+= column_length, column++) { - rec_length=rec->length; - if ((type = (enum en_fieldtype) rec->type) != FIELD_NORMAL && + column_length= column->length; + if ((type = (enum en_fieldtype) column->type) != FIELD_NORMAL && (type != FIELD_CHECK)) { if (type == FIELD_VARCHAR) { - uint pack_length= HA_VARCHAR_PACKLENGTH(rec_length-1); + uint pack_length= HA_VARCHAR_PACKLENGTH(column_length-1); if (pack_length == 1) { length= (uint) *(uchar*) from; - if (length > rec_length-1) + if (length > column_length-1) goto err; *to= *from++; } else { get_key_length(length, from); - if (length > rec_length-2) + if (length > column_length-2) goto err; int2store(to,length); } @@ -1195,11 +1195,11 @@ ulong _ma_rec_unpack(register MARIA_HA *info, register byte *to, byte *from, if (flag & bit) { if (type == FIELD_BLOB || type == FIELD_SKIP_ZERO) - bzero((byte*) to,rec_length); + bzero((byte*) to,column_length); else if (type == FIELD_SKIP_ENDSPACE || type == FIELD_SKIP_PRESPACE) { - if (rec->length > 255 && *from & 128) + if (column->length > 255 && *from & 128) { if (from + 1 >= from_end) goto err; @@ -1212,25 +1212,25 @@ ulong _ma_rec_unpack(register MARIA_HA *info, register byte *to, byte *from, length= (uchar) *from++; } min_pack_length--; - if (length >= rec_length || + if (length >= column_length || min_pack_length + length > (uint) (from_end - from)) goto err; if (type == FIELD_SKIP_ENDSPACE) { memcpy(to,(byte*) from,(size_t) length); - bfill((byte*) to+length,rec_length-length,' '); + bfill((byte*) to+length,column_length-length,' '); } else { - bfill((byte*) to,rec_length-length,' '); - memcpy(to+rec_length-length,(byte*) from,(size_t) length); + bfill((byte*) to,column_length-length,' '); + memcpy(to+column_length-length,(byte*) from,(size_t) length); } from+=length; } } else if (type == FIELD_BLOB) { - uint size_length=rec_length- maria_portable_sizeof_char_ptr; + uint size_length=column_length- portable_sizeof_char_ptr; ulong blob_length= _ma_calc_blob_length(size_length,from); ulong from_left= (ulong) (from_end - from); if (from_left < size_length || @@ -1246,9 +1246,9 @@ ulong _ma_rec_unpack(register MARIA_HA *info, register byte *to, byte *from, { if (type == FIELD_SKIP_ENDSPACE || type == FIELD_SKIP_PRESPACE) min_pack_length--; - if (min_pack_length + rec_length > (uint) (from_end - from)) + if (min_pack_length + column_length > (uint) (from_end - from)) goto err; - memcpy(to,(byte*) from,(size_t) rec_length); from+=rec_length; + memcpy(to,(byte*) from,(size_t) column_length); from+=column_length; } if ((bit= bit << 1) >= 256) { @@ -1259,9 +1259,9 @@ ulong _ma_rec_unpack(register MARIA_HA *info, register byte *to, byte *from, { if (min_pack_length > (uint) (from_end - from)) goto err; - min_pack_length-=rec_length; - memcpy(to, (byte*) from, (size_t) rec_length); - from+=rec_length; + min_pack_length-=column_length; + memcpy(to, (byte*) from, (size_t) column_length); + from+=column_length; } } if (info->s->calc_checksum) @@ -1620,7 +1620,7 @@ err: /* - Read record from datafile. + Read next record from datafile during table scan. SYNOPSIS _ma_read_rnd_dynamic_record() @@ -1631,22 +1631,17 @@ err: record is found. NOTE + This is identical to _ma_read_dynamic_record(), except the following + cases: - If a write buffer is active, it needs to be flushed if its contents - intersects with the record to read. We always check if the position - of the first byte of the write buffer is lower than the position - past the last byte to read. In theory this is also true if the write - buffer is completely below the read segment. That is, if there is no - intersection. But this case is unusual. We flush anyway. Only if the - first byte in the write buffer is above the last byte to read, we do - not flush. + - If there is no active row at 'filepos', continue scanning for + an active row. (This is becasue the previous + _ma_read_rnd_dynamic_record() call stored the next block position + in filepos, but this position may not be a start block for a row + - We may have READ_CACHING enabled, in which case we use the cache + to read rows. - A dynamic record may need several reads. So this check must be done - before every read. Reading a dynamic record starts with reading the - block header. If the record does not fit into the free space of the - header, the block may be longer than the header. In this case a - second read is necessary. These one or two reads repeat for every - part of the record. + For other comments, check _ma_read_dynamic_record() RETURN 0 OK diff --git a/storage/maria/ma_ft_eval.c b/storage/maria/ma_ft_eval.c index 50584459b7d..5fc67c6c664 100644 --- a/storage/maria/ma_ft_eval.c +++ b/storage/maria/ma_ft_eval.c @@ -49,7 +49,7 @@ int main(int argc, char *argv[]) recinfo[0].type=FIELD_SKIP_ENDSPACE; recinfo[0].length=docid_length; recinfo[1].type=FIELD_BLOB; - recinfo[1].length= 4+maria_portable_sizeof_char_ptr; + recinfo[1].length= 4+portable_sizeof_char_ptr; /* Define a key over the first column */ keyinfo[0].seg=keyseg; diff --git a/storage/maria/ma_ft_test1.c b/storage/maria/ma_ft_test1.c index 2b087dde35e..4c98e766234 100644 --- a/storage/maria/ma_ft_test1.c +++ b/storage/maria/ma_ft_test1.c @@ -76,12 +76,12 @@ static int run_test(const char *filename) /* First define 2 columns */ recinfo[0].type=extra_field; - recinfo[0].length= (extra_field == FIELD_BLOB ? 4 + maria_portable_sizeof_char_ptr : + recinfo[0].length= (extra_field == FIELD_BLOB ? 4 + portable_sizeof_char_ptr : extra_length); if (extra_field == FIELD_VARCHAR) recinfo[0].length+= HA_VARCHAR_PACKLENGTH(extra_length); recinfo[1].type=key_field; - recinfo[1].length= (key_field == FIELD_BLOB ? 4+maria_portable_sizeof_char_ptr : + recinfo[1].length= (key_field == FIELD_BLOB ? 4+portable_sizeof_char_ptr : key_length); if (key_field == FIELD_VARCHAR) recinfo[1].length+= HA_VARCHAR_PACKLENGTH(key_length); diff --git a/storage/maria/ma_ft_update.c b/storage/maria/ma_ft_update.c index cd2e121d0ed..97ebdb05b42 100644 --- a/storage/maria/ma_ft_update.c +++ b/storage/maria/ma_ft_update.c @@ -329,7 +329,7 @@ uint _ma_ft_convert_to_ft2(MARIA_HA *info, uint keynr, byte *key) /* creating pageful of keys */ maria_putint(info->buff,length+2,0); memcpy(info->buff+2, key_ptr, length); - info->keybuff_used=info->page_changed=1; /* info->buff is used */ + info->keyread_buff_used=info->page_changed=1; /* info->buff is used */ if ((root= _ma_new(info,keyinfo,DFLT_INIT_HITS)) == HA_OFFSET_ERROR || _ma_write_keypage(info,keyinfo,root,DFLT_INIT_HITS,info->buff)) DBUG_RETURN(-1); diff --git a/storage/maria/ma_info.c b/storage/maria/ma_info.c index 366243ccba7..83143160b66 100644 --- a/storage/maria/ma_info.c +++ b/storage/maria/ma_info.c @@ -126,10 +126,14 @@ void _ma_report_error(int errcode, const char *file_name) if ((length= strlen(file_name)) > 64) { + /* we first remove the directory */ uint dir_length= dirname_length(file_name); file_name+= dir_length; if ((length-= dir_length) > 64) + { + /* still too long, chop start of table name */ file_name+= length - 64; + } } my_error(errcode, MYF(ME_NOREFRESH), file_name); DBUG_VOID_RETURN; diff --git a/storage/maria/ma_open.c b/storage/maria/ma_open.c index 8ef7a7375c1..933d9b32045 100644 --- a/storage/maria/ma_open.c +++ b/storage/maria/ma_open.c @@ -232,13 +232,21 @@ MARIA_HA *maria_open(const char *name, int mode, uint open_flags) } key_parts+=fulltext_keys*FT_SEGS; - if (share->base.max_key_length > HA_MAX_KEY_BUFF || keys > MARIA_MAX_KEY || - key_parts >= MARIA_MAX_KEY * HA_MAX_KEY_SEG) + if (share->base.max_key_length > maria_max_key_length() || + keys > MARIA_MAX_KEY || key_parts >= MARIA_MAX_KEY * HA_MAX_KEY_SEG) { DBUG_PRINT("error",("Wrong key info: Max_key_length: %d keys: %d key_parts: %d", share->base.max_key_length, keys, key_parts)); my_errno=HA_ERR_UNSUPPORTED; goto err; } + if (share->base.block_size != maria_block_size) + { + DBUG_PRINT("error", ("Wrong block size %u; Expected %u", + (uint) share->base.block_size, + (uint) maria_block_size)); + my_errno=HA_ERR_UNSUPPORTED; + goto err; + } /* Correct max_file_length based on length of sizeof(off_t) */ max_data_file_length= @@ -268,7 +276,7 @@ MARIA_HA *maria_open(const char *name, int mode, uint open_flags) &share->keyparts, (key_parts+unique_key_parts+keys+uniques) * sizeof(HA_KEYSEG), - &share->rec, + &share->columndef, (share->base.fields+1)*sizeof(MARIA_COLUMNDEF), &share->blobs,sizeof(MARIA_BLOB)*share->base.blobs, &share->unique_file_name,strlen(name_buff)+1, @@ -304,7 +312,6 @@ MARIA_HA *maria_open(const char *name, int mode, uint open_flags) end_pos); if (share->keyinfo[i].key_alg == HA_KEY_ALG_RTREE) have_rtree=1; - set_if_smaller(share->block_size,share->keyinfo[i].block_length); share->keyinfo[i].seg=pos; for (j=0 ; j < share->keyinfo[i].keysegs; j++,pos++) { @@ -418,7 +425,7 @@ MARIA_HA *maria_open(const char *name, int mode, uint open_flags) if (share->base.transactional) share->base_length+= TRANS_ROW_EXTRA_HEADER_SIZE; share->base.default_rec_buff_size= max(share->base.pack_reclength, - share->base.max_key_length); + share->base.max_key_length); if (share->data_file_type == DYNAMIC_RECORD) { share->base.extra_rec_buff_size= @@ -430,18 +437,18 @@ MARIA_HA *maria_open(const char *name, int mode, uint open_flags) end_pos); for (i= j= 0 ; i < share->base.fields ; i++) { - disk_pos=_ma_recinfo_read(disk_pos,&share->rec[i]); - share->rec[i].pack_type=0; - share->rec[i].huff_tree=0; - if (share->rec[i].type == (int) FIELD_BLOB) + disk_pos=_ma_columndef_read(disk_pos,&share->columndef[i]); + share->columndef[i].pack_type=0; + share->columndef[i].huff_tree=0; + if (share->columndef[i].type == (int) FIELD_BLOB) { share->blobs[j].pack_length= - share->rec[i].length-maria_portable_sizeof_char_ptr;; - share->blobs[j].offset= share->rec[i].offset; + share->columndef[i].length-portable_sizeof_char_ptr;; + share->blobs[j].offset= share->columndef[i].offset; j++; } } - share->rec[i].type=(int) FIELD_LAST; /* End marker */ + share->columndef[i].type=(int) FIELD_LAST; /* End marker */ #ifdef ASKMONTY /* This code was added to mi_open.c in this cset: @@ -718,8 +725,8 @@ void _ma_setup_functions(register MARIA_SHARE *share) share->once_end= maria_once_end_dummy; share->init= maria_scan_init_dummy; share->end= maria_scan_end_dummy; - share->scan_init= maria_scan_init_dummy; - share->scan_end= maria_scan_end_dummy; + share->scan_init= maria_scan_init_dummy;/* Compat. dummy function */ + share->scan_end= maria_scan_end_dummy;/* Compat. dummy function */ share->write_record_init= _ma_write_init_default; share->write_record_abort= _ma_write_abort_default; @@ -729,7 +736,10 @@ void _ma_setup_functions(register MARIA_SHARE *share) share->scan= _ma_read_rnd_pack_record; share->once_init= _ma_once_init_pack_row; share->once_end= _ma_once_end_pack_row; - /* Calculate checksum according how the original row was stored */ + /* + Calculate checksum according to data in the original, not compressed, + row. + */ if (share->state.header.org_data_file_type == STATIC_RECORD) share->calc_checksum= _ma_static_checksum; else @@ -767,10 +777,10 @@ void _ma_setup_functions(register MARIA_SHARE *share) share->calc_checksum= share->calc_write_checksum= _ma_static_checksum; break; case BLOCK_RECORD: - share->once_init= _ma_once_init_block_row; - share->once_end= _ma_once_end_block_row; - share->init= _ma_init_block_row; - share->end= _ma_end_block_row; + share->once_init= _ma_once_init_block_record; + share->once_end= _ma_once_end_block_record; + share->init= _ma_init_block_record; + share->end= _ma_end_block_record; share->write_record_init= _ma_write_init_block_record; share->write_record_abort= _ma_write_abort_block_record; share->scan_init= _ma_scan_init_block_record; @@ -783,6 +793,10 @@ void _ma_setup_functions(register MARIA_SHARE *share) share->write_record= _ma_write_block_record; share->compare_unique= _ma_cmp_block_unique; share->calc_checksum= _ma_checksum; + /* + write_block_record() will calculate the checksum; Tell maria_write() + that it doesn't have to do this. + */ share->calc_write_checksum= 0; break; } @@ -1187,32 +1201,32 @@ char *_ma_uniquedef_read(char *ptr, MARIA_UNIQUEDEF *def) ** MARIA_COLUMNDEF ***************************************************************************/ -uint _ma_recinfo_write(File file, MARIA_COLUMNDEF *recinfo) +uint _ma_columndef_write(File file, MARIA_COLUMNDEF *columndef) { uchar buff[MARIA_COLUMNDEF_SIZE]; uchar *ptr=buff; - mi_int6store(ptr,recinfo->offset); ptr+= 6; - mi_int2store(ptr,recinfo->type); ptr+= 2; - mi_int2store(ptr,recinfo->length); ptr+= 2; - mi_int2store(ptr,recinfo->fill_length); ptr+= 2; - mi_int2store(ptr,recinfo->null_pos); ptr+= 2; - mi_int2store(ptr,recinfo->empty_pos); ptr+= 2; - (*ptr++)= recinfo->null_bit; - (*ptr++)= recinfo->empty_bit; + mi_int6store(ptr,columndef->offset); ptr+= 6; + mi_int2store(ptr,columndef->type); ptr+= 2; + mi_int2store(ptr,columndef->length); ptr+= 2; + mi_int2store(ptr,columndef->fill_length); ptr+= 2; + mi_int2store(ptr,columndef->null_pos); ptr+= 2; + mi_int2store(ptr,columndef->empty_pos); ptr+= 2; + (*ptr++)= columndef->null_bit; + (*ptr++)= columndef->empty_bit; return my_write(file,(char*) buff, (uint) (ptr-buff), MYF(MY_NABP)); } -char *_ma_recinfo_read(char *ptr, MARIA_COLUMNDEF *recinfo) +char *_ma_columndef_read(char *ptr, MARIA_COLUMNDEF *columndef) { - recinfo->offset= mi_uint6korr(ptr); ptr+= 6; - recinfo->type= mi_sint2korr(ptr); ptr+= 2; - recinfo->length= mi_uint2korr(ptr); ptr+= 2; - recinfo->fill_length= mi_uint2korr(ptr); ptr+= 2; - recinfo->null_pos= mi_uint2korr(ptr); ptr+= 2; - recinfo->empty_pos= mi_uint2korr(ptr); ptr+= 2; - recinfo->null_bit= (uint8) *ptr++; - recinfo->empty_bit= (uint8) *ptr++; + columndef->offset= mi_uint6korr(ptr); ptr+= 6; + columndef->type= mi_sint2korr(ptr); ptr+= 2; + columndef->length= mi_uint2korr(ptr); ptr+= 2; + columndef->fill_length= mi_uint2korr(ptr); ptr+= 2; + columndef->null_pos= mi_uint2korr(ptr); ptr+= 2; + columndef->empty_pos= mi_uint2korr(ptr); ptr+= 2; + columndef->null_bit= (uint8) *ptr++; + columndef->empty_bit= (uint8) *ptr++; return ptr; } diff --git a/storage/maria/ma_packrec.c b/storage/maria/ma_packrec.c index 7134297710d..2c489f69233 100644 --- a/storage/maria/ma_packrec.c +++ b/storage/maria/ma_packrec.c @@ -142,15 +142,14 @@ static maria_bit_type mask[]= my_bool _ma_once_init_pack_row(MARIA_SHARE *share, File dfile) { share->options|= HA_OPTION_READ_ONLY_DATA; - if (_ma_read_pack_info(share, dfile, - (pbool) - test(!(share->options & - (HA_OPTION_PACK_RECORD | - HA_OPTION_TEMP_COMPRESS_RECORD))))) - return 1; - return 0; + return (_ma_read_pack_info(share, dfile, + (pbool) + test(!(share->options & + (HA_OPTION_PACK_RECORD | + HA_OPTION_TEMP_COMPRESS_RECORD))))); } + my_bool _ma_once_end_pack_row(MARIA_SHARE *share) { if (share->decode_trees) @@ -262,15 +261,16 @@ static my_bool _ma_read_pack_info(MARIA_SHARE *share, File file, /* Read new info for each field */ for (i=0 ; i < share->base.fields ; i++) { - share->rec[i].base_type=(enum en_fieldtype) get_bits(&bit_buff,5); - share->rec[i].pack_type=(uint) get_bits(&bit_buff,6); - share->rec[i].space_length_bits=get_bits(&bit_buff,5); - share->rec[i].huff_tree=share->decode_trees+(uint) get_bits(&bit_buff, + share->columndef[i].base_type=(enum en_fieldtype) get_bits(&bit_buff,5); + share->columndef[i].pack_type=(uint) get_bits(&bit_buff,6); + share->columndef[i].space_length_bits=get_bits(&bit_buff,5); + share->columndef[i].huff_tree=share->decode_trees+(uint) get_bits(&bit_buff, huff_tree_bits); - share->rec[i].unpack= get_unpack_function(share->rec+i); + share->columndef[i].unpack= get_unpack_function(share->columndef + i); DBUG_PRINT("info", ("col: %2u type: %2u pack: %u slbits: %2u", - i, share->rec[i].base_type, share->rec[i].pack_type, - share->rec[i].space_length_bits)); + i, share->columndef[i].base_type, + share->columndef[i].pack_type, + share->columndef[i].space_length_bits)); } skip_to_next_byte(&bit_buff); /* @@ -776,7 +776,7 @@ int _ma_pack_rec_unpack(register MARIA_HA *info, MARIA_BIT_BUFF *bit_buff, reclength-= info->s->base.null_bytes; } init_bit_buffer(bit_buff, (uchar*) from, reclength); - for (current_field=share->rec, end=current_field+share->base.fields ; + for (current_field=share->columndef, end=current_field+share->base.fields ; current_field < end ; current_field++,to=end_field) { @@ -1080,7 +1080,7 @@ static void uf_blob(MARIA_COLUMNDEF *rec, MARIA_BIT_BUFF *bit_buff, else { ulong length=get_bits(bit_buff,rec->space_length_bits); - uint pack_length=(uint) (end-to)-maria_portable_sizeof_char_ptr; + uint pack_length=(uint) (end-to)-portable_sizeof_char_ptr; if (bit_buff->blob_pos+length > bit_buff->blob_end) { bit_buff->error=1; diff --git a/storage/maria/ma_page.c b/storage/maria/ma_page.c index 1b013b6a0da..b1f9ddde97c 100644 --- a/storage/maria/ma_page.c +++ b/storage/maria/ma_page.c @@ -32,7 +32,7 @@ byte *_ma_fetch_keypage(register MARIA_HA *info, MARIA_KEYDEF *keyinfo, info->s->block_size, info->s->block_size, return_buffer); if (tmp == info->buff) - info->keybuff_used=1; + info->keyread_buff_used=1; else if (!tmp) { DBUG_PRINT("error",("Got errno: %d from key_cache_read",my_errno)); diff --git a/storage/maria/ma_rkey.c b/storage/maria/ma_rkey.c index c02c18094e8..6158935472b 100644 --- a/storage/maria/ma_rkey.c +++ b/storage/maria/ma_rkey.c @@ -150,7 +150,6 @@ int maria_rkey(MARIA_HA *info, byte *buf, int inx, const byte *key, else info->last_rkey_length= pack_key_length; - /* Check if we don't want to have record back, only error message */ if (!buf) { diff --git a/storage/maria/ma_rrnd.c b/storage/maria/ma_rrnd.c index 8d1bf9aa4f6..8e2b12dc60d 100644 --- a/storage/maria/ma_rrnd.c +++ b/storage/maria/ma_rrnd.c @@ -36,7 +36,6 @@ int maria_rrnd(MARIA_HA *info, byte *buf, MARIA_RECORD_POS filepos) #ifdef NOT_USED if (filepos == HA_OFFSET_ERROR) { - skip_deleted_blocks=1; if (info->cur_row.lastpos == HA_OFFSET_ERROR) /* First read ? */ filepos= info->s->pack.header_length; /* Read first record */ else diff --git a/storage/maria/ma_rt_index.c b/storage/maria/ma_rt_index.c index b941c77f44d..b61e7ed49a8 100644 --- a/storage/maria/ma_rt_index.c +++ b/storage/maria/ma_rt_index.c @@ -127,11 +127,11 @@ static int maria_rtree_find_req(MARIA_HA *info, MARIA_KEYDEF *keyinfo, info->int_keypos = info->buff; info->int_maxpos = info->buff + (last - after_key); memcpy(info->buff, after_key, last - after_key); - info->keybuff_used = 0; + info->keyread_buff_used = 0; } else { - info->keybuff_used = 1; + info->keyread_buff_used = 1; } res = 0; @@ -192,7 +192,7 @@ int maria_rtree_find_first(MARIA_HA *info, uint keynr, byte *key, info->last_rkey_length = key_length; info->maria_rtree_recursion_depth = -1; - info->keybuff_used = 1; + info->keyread_buff_used = 1; nod_cmp_flag= ((search_flag & (MBR_EQUAL | MBR_WITHIN)) ? MBR_WITHIN : MBR_INTERSECT); @@ -227,7 +227,7 @@ int maria_rtree_find_next(MARIA_HA *info, uint keynr, uint search_flag) info->lastkey_length, search_flag); - if (!info->keybuff_used) + if (!info->keyread_buff_used) { byte *key= info->int_keypos; @@ -245,7 +245,7 @@ int maria_rtree_find_next(MARIA_HA *info, uint keynr, uint search_flag) if (after_key < info->int_maxpos) info->int_keypos= after_key; else - info->keybuff_used= 1; + info->keyread_buff_used= 1; return 0; } key+= keyinfo->keylength; @@ -342,11 +342,11 @@ static int maria_rtree_get_req(MARIA_HA *info, MARIA_KEYDEF *keyinfo, uint key_l info->int_keypos = (byte*) saved_key; memcpy(info->buff, page_buf, keyinfo->block_length); info->int_maxpos = rt_PAGE_END(info->buff); - info->keybuff_used = 0; + info->keyread_buff_used = 0; } else { - info->keybuff_used = 1; + info->keyread_buff_used = 1; } res = 0; @@ -389,7 +389,7 @@ int maria_rtree_get_first(MARIA_HA *info, uint keynr, uint key_length) } info->maria_rtree_recursion_depth = -1; - info->keybuff_used = 1; + info->keyread_buff_used = 1; return maria_rtree_get_req(info, &keyinfo[keynr], key_length, root, 0); } @@ -409,7 +409,7 @@ int maria_rtree_get_next(MARIA_HA *info, uint keynr, uint key_length) my_off_t root; MARIA_KEYDEF *keyinfo = info->s->keyinfo + keynr; - if (!info->keybuff_used) + if (!info->keyread_buff_used) { uint k_len = keyinfo->keylength - info->s->base.rec_reflength; /* rt_PAGE_NEXT_KEY(info->int_keypos) */ @@ -425,7 +425,7 @@ int maria_rtree_get_next(MARIA_HA *info, uint keynr, uint key_length) *(int*)info->int_keypos = key - info->buff; if (after_key >= info->int_maxpos) { - info->keybuff_used = 1; + info->keyread_buff_used = 1; } return 0; @@ -638,7 +638,7 @@ static int maria_rtree_insert_level(MARIA_HA *info, uint keynr, byte *key, { if ((old_root = _ma_new(info, keyinfo, DFLT_INIT_HITS)) == HA_OFFSET_ERROR) return -1; - info->keybuff_used = 1; + info->keyread_buff_used = 1; maria_putint(info->buff, 2, 0); res = maria_rtree_add_key(info, keyinfo, key, key_length, info->buff, NULL); if (_ma_write_keypage(info, keyinfo, old_root, DFLT_INIT_HITS, info->buff)) diff --git a/storage/maria/ma_search.c b/storage/maria/ma_search.c index d171dca8729..0ec36db59c5 100644 --- a/storage/maria/ma_search.c +++ b/storage/maria/ma_search.c @@ -157,7 +157,7 @@ int _ma_search(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, info->int_keytree_version=keyinfo->version; info->last_search_keypage=info->last_keypage; info->page_changed=0; - info->keybuff_used= (info->keyread_buff != buff); /* If we have to reread */ + info->keyread_buff_used= (info->keyread_buff != buff); /* If we have to reread */ DBUG_PRINT("exit",("found key at %lu",(ulong) info->cur_row.lastpos)); DBUG_RETURN(0); @@ -618,7 +618,7 @@ void _ma_kpointer(register MARIA_HA *info, register byte *buff, my_off_t pos) case 4: mi_int4store(buff,pos); break; case 3: mi_int3store(buff,pos); break; case 2: mi_int2store(buff,(uint) pos); break; - case 1: buff[0]= (char) (uchar) pos; break; + case 1: buff[0]= (byte) pos; break; default: abort(); /* impossible */ } } /* _ma_kpointer */ @@ -1219,10 +1219,10 @@ int _ma_search_next(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, uint nod_flag; byte lastkey[HA_MAX_KEY_BUFF]; DBUG_ENTER("_ma_search_next"); - DBUG_PRINT("enter",("nextflag: %u lastpos: %lu int_keypos: %lu page_changed %d keybuff_used: %d", + DBUG_PRINT("enter",("nextflag: %u lastpos: %lu int_keypos: %lu page_changed %d keyread_buff_used: %d", nextflag, (ulong) info->cur_row.lastpos, (ulong) info->int_keypos, - info->page_changed, info->keybuff_used)); + info->page_changed, info->keyread_buff_used)); DBUG_EXECUTE("key", _ma_print_key(DBUG_FILE,keyinfo->seg,key,key_length);); /* Force full read if we are at last key or if we are not on a leaf @@ -1235,16 +1235,16 @@ int _ma_search_next(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, if (((nextflag & SEARCH_BIGGER) && info->int_keypos >= info->int_maxpos) || info->page_changed || (info->int_keytree_version != keyinfo->version && - (info->int_nod_flag || info->keybuff_used))) + (info->int_nod_flag || info->keyread_buff_used))) DBUG_RETURN(_ma_search(info,keyinfo,key, USE_WHOLE_KEY, nextflag | SEARCH_SAVE_BUFF, pos)); - if (info->keybuff_used) + if (info->keyread_buff_used) { if (!_ma_fetch_keypage(info,keyinfo,info->last_search_keypage, DFLT_INIT_HITS,info->keyread_buff,0)) DBUG_RETURN(-1); - info->keybuff_used=0; + info->keyread_buff_used=0; } /* Last used buffer is in info->keyread_buff */ @@ -1328,7 +1328,7 @@ int _ma_search_first(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, info->int_nod_flag=nod_flag; info->int_keytree_version=keyinfo->version; info->last_search_keypage=info->last_keypage; - info->page_changed=info->keybuff_used=0; + info->page_changed=info->keyread_buff_used=0; info->cur_row.lastpos= _ma_dpos(info,0,info->lastkey+info->lastkey_length); DBUG_PRINT("exit",("found key at %lu", (ulong) info->cur_row.lastpos)); @@ -1373,7 +1373,7 @@ int _ma_search_last(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, info->int_nod_flag=nod_flag; info->int_keytree_version=keyinfo->version; info->last_search_keypage=info->last_keypage; - info->page_changed=info->keybuff_used=0; + info->page_changed=info->keyread_buff_used=0; DBUG_PRINT("exit",("found key at %lu",(ulong) info->cur_row.lastpos)); DBUG_RETURN(0); diff --git a/storage/maria/ma_sp_test.c b/storage/maria/ma_sp_test.c index bab346ca18d..7a413f68135 100644 --- a/storage/maria/ma_sp_test.c +++ b/storage/maria/ma_sp_test.c @@ -80,7 +80,7 @@ int run_test(const char *filename) /* Define spatial column */ recinfo[1].type=FIELD_BLOB; - recinfo[1].length=4 + maria_portable_sizeof_char_ptr; + recinfo[1].length=4 + portable_sizeof_char_ptr; diff --git a/storage/maria/ma_test1.c b/storage/maria/ma_test1.c index 1cb8c3e002a..223ef3850aa 100644 --- a/storage/maria/ma_test1.c +++ b/storage/maria/ma_test1.c @@ -73,12 +73,12 @@ static int run_test(const char *filename) /* First define 2 columns */ create_info.null_bytes= 1; recinfo[0].type= key_field; - recinfo[0].length= (key_field == FIELD_BLOB ? 4+maria_portable_sizeof_char_ptr : + recinfo[0].length= (key_field == FIELD_BLOB ? 4+portable_sizeof_char_ptr : key_length); if (key_field == FIELD_VARCHAR) recinfo[0].length+= HA_VARCHAR_PACKLENGTH(key_length); recinfo[1].type=extra_field; - recinfo[1].length= (extra_field == FIELD_BLOB ? 4 + maria_portable_sizeof_char_ptr : 24); + recinfo[1].length= (extra_field == FIELD_BLOB ? 4 + portable_sizeof_char_ptr : 24); if (extra_field == FIELD_VARCHAR) recinfo[1].length+= HA_VARCHAR_PACKLENGTH(recinfo[1].length); if (opt_unique) diff --git a/storage/maria/ma_test2.c b/storage/maria/ma_test2.c index 92fa5b63650..6b4a7f5ec92 100644 --- a/storage/maria/ma_test2.c +++ b/storage/maria/ma_test2.c @@ -191,7 +191,7 @@ int main(int argc, char *argv[]) if (use_blob) { recinfo[6].type=FIELD_BLOB; - recinfo[6].length=4+maria_portable_sizeof_char_ptr; + recinfo[6].length=4+portable_sizeof_char_ptr; recinfo[6].null_bit=0; recinfo[6].null_pos=0; } diff --git a/storage/maria/ma_update.c b/storage/maria/ma_update.c index 8c8e46bc024..ee0f638ea7c 100644 --- a/storage/maria/ma_update.c +++ b/storage/maria/ma_update.c @@ -177,7 +177,7 @@ int maria_update(register MARIA_HA *info, const byte *oldrec, byte *newrec) info->state->checksum+= (info->cur_row.checksum - old_checksum); /* - We can't yet have HA_STATE_ACTIVE here, as block_record dosn't support + We can't yet have HA_STATE_AKTIV here, as block_record dosn't support it */ info->update= (HA_STATE_CHANGED | HA_STATE_ROW_CHANGED | key_changed); diff --git a/storage/maria/ma_write.c b/storage/maria/ma_write.c index b05cafb0469..da363ddabc8 100644 --- a/storage/maria/ma_write.c +++ b/storage/maria/ma_write.c @@ -356,7 +356,7 @@ int _ma_enlarge_root(MARIA_HA *info, MARIA_KEYDEF *keyinfo, byte *key, (byte*) 0, (byte*) 0, key,&s_temp); maria_putint(info->buff,t_length+2+nod_flag,nod_flag); (*keyinfo->store_key)(keyinfo,info->buff+2+nod_flag,&s_temp); - info->keybuff_used=info->page_changed=1; /* info->buff is used */ + info->keyread_buff_used=info->page_changed=1; /* info->buff is used */ if ((*root= _ma_new(info,keyinfo,DFLT_INIT_HITS)) == HA_OFFSET_ERROR || _ma_write_keypage(info,keyinfo,*root,DFLT_INIT_HITS,info->buff)) DBUG_RETURN(-1); @@ -634,7 +634,7 @@ int _ma_split_page(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, if (info->s->keyinfo+info->lastinx == keyinfo) info->page_changed=1; /* Info->buff is used */ - info->keybuff_used=1; + info->keyread_buff_used=1; nod_flag=_ma_test_if_nod(buff); key_ref_length=2+nod_flag; if (insert_last_key) diff --git a/storage/maria/maria_chk.c b/storage/maria/maria_chk.c index 5796ae1a196..d9f00a7f4db 100644 --- a/storage/maria/maria_chk.c +++ b/storage/maria/maria_chk.c @@ -50,8 +50,8 @@ static const char *type_names[]= "impossible","char","binary", "short", "long", "float", "double","number","unsigned short", "unsigned long","longlong","ulonglong","int24", - "uint24","int8","varchar", "varbin","?", - "?" + "uint24","int8","varchar", "varbin", "varchar2", "varbin2", "bit", + "?","?" }; static const char *prefix_packed_txt="packed ", @@ -1224,7 +1224,7 @@ end2: static void descript(HA_CHECK *param, register MARIA_HA *info, my_string name) { - uint key,keyseg_nr,field,start; + uint key,keyseg_nr,field; reg3 MARIA_KEYDEF *keyinfo; reg2 HA_KEYSEG *keyseg; reg4 const char *text; @@ -1430,43 +1430,42 @@ static void descript(HA_CHECK *param, register MARIA_HA *info, my_string name) for (field=0 ; field < share->base.fields ; field++) { if (share->options & HA_OPTION_COMPRESS_RECORD) - type=share->rec[field].base_type; + type=share->columndef[field].base_type; else - type=(enum en_fieldtype) share->rec[field].type; + type=(enum en_fieldtype) share->columndef[field].type; end=strmov(buff,field_pack[type]); if (share->options & HA_OPTION_COMPRESS_RECORD) { - if (share->rec[field].pack_type & PACK_TYPE_SELECTED) + if (share->columndef[field].pack_type & PACK_TYPE_SELECTED) end=strmov(end,", not_always"); - if (share->rec[field].pack_type & PACK_TYPE_SPACE_FIELDS) + if (share->columndef[field].pack_type & PACK_TYPE_SPACE_FIELDS) end=strmov(end,", no empty"); - if (share->rec[field].pack_type & PACK_TYPE_ZERO_FILL) + if (share->columndef[field].pack_type & PACK_TYPE_ZERO_FILL) { - sprintf(end,", zerofill(%d)",share->rec[field].space_length_bits); + sprintf(end,", zerofill(%d)",share->columndef[field].space_length_bits); end=strend(end); } } if (buff[0] == ',') strmov(buff,buff+2); - int10_to_str((long) share->rec[field].length,length,10); + int10_to_str((long) share->columndef[field].length,length,10); null_bit[0]=null_pos[0]=0; - if (share->rec[field].null_bit) + if (share->columndef[field].null_bit) { - sprintf(null_bit,"%d",share->rec[field].null_bit); - sprintf(null_pos,"%d",share->rec[field].null_pos+1); + sprintf(null_bit,"%d",share->columndef[field].null_bit); + sprintf(null_pos,"%d",share->columndef[field].null_pos+1); } printf("%-6d%-6u%-7s%-8s%-8s%-35s",field+1, - (uint) share->rec[field].offset+1, + (uint) share->columndef[field].offset+1, length, null_pos, null_bit, buff); if (share->options & HA_OPTION_COMPRESS_RECORD) { - if (share->rec[field].huff_tree) + if (share->columndef[field].huff_tree) printf("%3d %2d", - (uint) (share->rec[field].huff_tree-share->decode_trees)+1, - share->rec[field].huff_tree->quick_table_bits); + (uint) (share->columndef[field].huff_tree-share->decode_trees)+1, + share->columndef[field].huff_tree->quick_table_bits); } VOID(putchar('\n')); - start+=share->rec[field].length; } } DBUG_VOID_RETURN; @@ -1556,8 +1555,9 @@ static int maria_sort_records(HA_CHECK *param, fn_format(param->temp_filename,name,"", MARIA_NAME_DEXT,2+4+32); new_file= my_create(fn_format(param->temp_filename, param->temp_filename,"", - DATA_TMP_EXT,2+4), - 0,param->tmpfile_createflag, + DATA_TMP_EXT, + MY_REPLACE_EXT | MY_UNPACK_FILENAME), + 0, param->tmpfile_createflag, MYF(0)); if (new_file < 0) { diff --git a/storage/maria/maria_def.h b/storage/maria/maria_def.h index 553e8efb787..6377d3dbae6 100644 --- a/storage/maria/maria_def.h +++ b/storage/maria/maria_def.h @@ -60,7 +60,8 @@ typedef struct st_maria_state_info uchar language; /* Language for indexes */ uchar fulltext_keys; uchar data_file_type; - uchar org_data_file_type; /* Used by mariapack to store dft */ + /* Used by mariapack to store the original data_file_type */ + uchar org_data_file_type; } header; MARIA_STATUS_INFO state; @@ -75,7 +76,7 @@ typedef struct st_maria_state_info ulong *rec_per_key_part; ha_checksum checksum; /* Table checksum */ my_off_t *key_root; /* Start of key trees */ - my_off_t key_del; /* delete links for trees */ + my_off_t key_del; /* delete links for index pages */ my_off_t rec_per_key_rows; /* Rows when calculating rec_per_key */ ulong sec_index_changed; /* Updated when new sec_index */ @@ -107,6 +108,13 @@ typedef struct st_maria_state_info #define MARIA_COLUMNDEF_SIZE (6+2+2+2+2+2+1+1) #define MARIA_BASE_INFO_SIZE (5*8 + 6*4 + 11*2 + 6 + 5*2 + 1 + 16) #define MARIA_INDEX_BLOCK_MARGIN 16 /* Safety margin for .MYI tables */ +/* Internal management bytes needed to store 2 keys on an index page */ +#define MARIA_INDEX_MIN_OVERHEAD_SIZE (4 + (TRANSID_SIZE+1) * 2) + +/* + Basic information of the Maria table. This is stored on disk + and not changed (unless we do DLL changes). +*/ typedef struct st_ma_base_info { @@ -127,7 +135,9 @@ typedef struct st_ma_base_info uint max_field_lengths; uint pack_fields; /* packed fields in table */ uint varlength_fields; /* char/varchar/blobs */ - uint rec_reflength; /* = 2-8 */ + /* Number of bytes in the index used to refer to a row (2-8) */ + uint rec_reflength; + /* Number of bytes in the index used to refer to another index page (2-8) */ uint key_reflength; /* = 2-8 */ uint keys; /* same as in state.header */ uint auto_key; /* Which key-1 is a auto key */ @@ -144,14 +154,17 @@ typedef struct st_ma_base_info uint extra_alloc_bytes; uint extra_alloc_procent; uint is_nulls_extended; /* 1 if new null bytes */ - uint min_row_length; + uint min_row_length; /* Min possible length of a row */ uint default_row_flag; /* 0 or ROW_FLAG_NULLS_EXTENDED */ uint block_size; + /* Size of initial record buffer */ uint default_rec_buff_size; + /* Extra number of bytes the row format require in the record buffer */ uint extra_rec_buff_size; /* The following are from the header */ uint key_parts, all_key_parts; + /* If false, we disable logging, versioning etc */ my_bool transactional; } MARIA_BASE_INFO; @@ -177,8 +190,8 @@ typedef struct st_maria_file_bitmap { uchar *map; ulonglong page; /* Page number for current bitmap */ - uint used_size; /* Size of bitmap that is not 0 */ - File file; + uint used_size; /* Size of bitmap head that is not 0 */ + File file; /* Datafile, where bitmap is stored */ my_bool changed; @@ -204,8 +217,7 @@ typedef struct st_maria_share MARIA_KEYDEF *keyinfo; /* Key definitions */ MARIA_UNIQUEDEF *uniqueinfo; /* unique definitions */ HA_KEYSEG *keyparts; /* key part info */ - MARIA_COLUMNDEF *rec; /* Pointer to field information - */ + MARIA_COLUMNDEF *columndef; /* Pointer to column information */ MARIA_PACK pack; /* Data about packed records */ MARIA_BLOB *blobs; /* Pointer to blobs */ char *unique_file_name; /* realpath() of index file */ @@ -216,25 +228,43 @@ typedef struct st_maria_share KEY_CACHE *key_cache; /* ref to the current key cache */ MARIA_DECODE_TREE *decode_trees; uint16 *decode_tables; + /* Called the first time the table instance is opened */ my_bool (*once_init)(struct st_maria_share *, File); + /* Called when the last instance of the table is closed */ my_bool (*once_end)(struct st_maria_share *); + /* Is called for every open of the table */ my_bool (*init)(struct st_maria_info *); + /* Is called for every close of the table */ void (*end)(struct st_maria_info *); + /* Called when we want to read a record from a specific position */ int (*read_record)(struct st_maria_info *, byte *, MARIA_RECORD_POS); + /* Initialize a scan */ my_bool (*scan_init)(struct st_maria_info *); + /* Read next record while scanning */ int (*scan)(struct st_maria_info *, byte *, MARIA_RECORD_POS, my_bool); + /* End scan */ void (*scan_end)(struct st_maria_info *); + /* Pre-write of row (some handlers may do the actual write here) */ MARIA_RECORD_POS (*write_record_init)(struct st_maria_info *, const byte *); + /* Write record (or accept write_record_init) */ my_bool (*write_record)(struct st_maria_info *, const byte *); + /* Called when write failed */ my_bool (*write_record_abort)(struct st_maria_info *); my_bool (*update_record)(struct st_maria_info *, MARIA_RECORD_POS, const byte *); my_bool (*delete_record)(struct st_maria_info *); my_bool (*compare_record)(struct st_maria_info *, const byte *); - ha_checksum(*calc_checksum) (struct st_maria_info *, const byte *); + /* calculate checksum for a row */ + ha_checksum(*calc_checksum)(struct st_maria_info *, const byte *); + /* + Calculate checksum for a row during write. May be 0 if we calculate + the checksum in write_record_init() + */ ha_checksum(*calc_write_checksum) (struct st_maria_info *, const byte *); - my_bool (*compare_unique) (struct st_maria_info *, MARIA_UNIQUEDEF *, - const byte *record, MARIA_RECORD_POS pos); + /* Compare a row in memory with a row on disk */ + my_bool (*compare_unique)(struct st_maria_info *, MARIA_UNIQUEDEF *, + const byte *record, MARIA_RECORD_POS pos); + /* Mapings to read/write the data file */ uint (*file_read)(MARIA_HA *, byte *, uint, my_off_t, myf); uint (*file_write)(MARIA_HA *, byte *, uint, my_off_t, myf); invalidator_by_filename invalidator; /* query cache invalidator */ @@ -255,6 +285,7 @@ typedef struct st_maria_share uint reopen; /* How many times reopened */ uint w_locks, r_locks, tot_locks; /* Number of read/write locks */ uint block_size; /* block_size of keyfile & data file*/ + /* Fixed length part of a packed row in BLOCK_RECORD format */ uint base_length; myf write_flag; enum data_file_type data_file_type; @@ -342,7 +373,8 @@ struct st_maria_info { MARIA_SHARE *s; /* Shared between open:s */ MARIA_STATUS_INFO *state, save_state; - MARIA_ROW cur_row, new_row; + MARIA_ROW cur_row; /* The active row that we just read */ + MARIA_ROW new_row; /* Storage for a row during update */ MARIA_BLOCK_SCAN scan; MARIA_BLOB *blobs; /* Pointer to blobs */ MARIA_BIT_BUFF bit_buff; @@ -404,8 +436,8 @@ struct st_maria_info my_bool quick_mode; /* If info->keyread_buff can't be used for rnext */ my_bool page_changed; - /* If info->keyread_buff has to be reread for rnext */ - my_bool keybuff_used; + /* If info->keyread_buff has to be re-read for rnext */ + my_bool keyread_buff_used; my_bool once_flags; /* For MARIA_MRG */ #ifdef __WIN__ my_bool owned_by_merge; /* This Maria table is part of a merge union */ @@ -419,7 +451,7 @@ struct st_maria_info /* Some defines used by maria-functions */ -#define USE_WHOLE_KEY HA_MAX_KEY_BUFF*2 /* Use whole key in _search() */ +#define USE_WHOLE_KEY 65535 /* Use whole key in _search() */ #define F_EXTRA_LCK -1 /* bits in opt_flag */ @@ -489,6 +521,7 @@ struct st_maria_info { length=mi_uint2korr((key)+1)+3; } \ } +#define maria_max_key_length() ((maria_block_size - MARIA_INDEX_MIN_OVERHEAD_SIZE)/2) #define get_pack_length(length) ((length) >= 255 ? 3 : 1) #define MARIA_MIN_BLOCK_LENGTH 20 /* Because of delete-link */ @@ -720,7 +753,11 @@ extern int _ma_ft_update(MARIA_HA *info, uint keynr, byte *keybuf, const byte *oldrec, const byte *newrec, my_off_t pos); -/* Parameter to _ma_get_block_info */ +/* + Parameter to _ma_get_block_info + The dynamic row header is read into this struct. For an explanation of + the fields, look at the function _ma_get_block_info(). +*/ typedef struct st_maria_block_info { @@ -736,6 +773,7 @@ typedef struct st_maria_block_info uint offset; } MARIA_BLOCK_INFO; + /* bits in return from _ma_get_block_info */ #define BLOCK_FIRST 1 @@ -800,8 +838,8 @@ uint _ma_keydef_write(File file, MARIA_KEYDEF *keydef); char *_ma_keydef_read(char *ptr, MARIA_KEYDEF *keydef); uint _ma_uniquedef_write(File file, MARIA_UNIQUEDEF *keydef); char *_ma_uniquedef_read(char *ptr, MARIA_UNIQUEDEF *keydef); -uint _ma_recinfo_write(File file, MARIA_COLUMNDEF *recinfo); -char *_ma_recinfo_read(char *ptr, MARIA_COLUMNDEF *recinfo); +uint _ma_columndef_write(File file, MARIA_COLUMNDEF *columndef); +char *_ma_columndef_read(char *ptr, MARIA_COLUMNDEF *columndef); ulong _ma_calc_total_blob_length(MARIA_HA *info, const byte *record); ha_checksum _ma_checksum(MARIA_HA *info, const byte *buf); ha_checksum _ma_static_checksum(MARIA_HA *info, const byte *buf); diff --git a/storage/maria/maria_pack.c b/storage/maria/maria_pack.c index dbd48e62d29..e4d0bcebbb9 100644 --- a/storage/maria/maria_pack.c +++ b/storage/maria/maria_pack.c @@ -307,9 +307,10 @@ static void usage(void) puts("and you are welcome to modify and redistribute it under the GPL license\n"); puts("Pack a MARIA-table to take much less space."); - puts("Keys are not updated, you must run maria_chk -rq on the datafile"); + puts("Keys are not updated, you must run maria_chk -rq on the index (.MAI) file"); puts("afterwards to update the keys."); - puts("You should give the .MYI file as the filename argument."); + puts("You should give the .MAI file as the filename argument."); + puts("To unpack a packed table, run maria_chk -u on the table"); VOID(printf("\nUsage: %s [OPTIONS] filename...\n", my_progname)); my_print_help(my_long_options); @@ -462,9 +463,9 @@ static bool open_isam_files(PACK_MRG_INFO *mrg,char **names,uint count) if (mrg->file[j]->s->base.reclength != mrg->file[j+1]->s->base.reclength || mrg->file[j]->s->base.fields != mrg->file[j+1]->s->base.fields) goto diff_file; - m1=mrg->file[j]->s->rec; + m1=mrg->file[j]->s->columndef; end=m1+mrg->file[j]->s->base.fields; - m2=mrg->file[j+1]->s->rec; + m2=mrg->file[j+1]->s->columndef; for ( ; m1 != end ; m1++,m2++) { if (m1->type != m2->type || m1->length != m2->length) @@ -773,8 +774,8 @@ static HUFF_COUNTS *init_huff_count(MARIA_HA *info,my_off_t records) for (i=0 ; i < info->s->base.fields ; i++) { enum en_fieldtype type; - count[i].field_length=info->s->rec[i].length; - type= count[i].field_type= (enum en_fieldtype) info->s->rec[i].type; + count[i].field_length=info->s->columndef[i].length; + type= count[i].field_type= (enum en_fieldtype) info->s->columndef[i].type; if (type == FIELD_INTERVALL || type == FIELD_CONSTANT || type == FIELD_ZERO) @@ -1003,7 +1004,7 @@ static int get_statistic(PACK_MRG_INFO *mrg,HUFF_COUNTS *huff_counts) /* Calculate pos, end_pos, and max_length for variable length fields. */ if (count->field_type == FIELD_BLOB) { - uint field_length=count->field_length -maria_portable_sizeof_char_ptr; + uint field_length=count->field_length -portable_sizeof_char_ptr; ulong blob_length= _ma_calc_blob_length(field_length, start_pos); memcpy_fixed((char*) &pos, start_pos+field_length,sizeof(char*)); end_pos=pos+blob_length; @@ -2656,7 +2657,7 @@ static int compress_isam_file(PACK_MRG_INFO *mrg, HUFF_COUNTS *huff_counts) case FIELD_BLOB: { ulong blob_length= _ma_calc_blob_length(field_length- - maria_portable_sizeof_char_ptr, + portable_sizeof_char_ptr, start_pos); /* Empty blobs are encoded with a single 1 bit. */ if (!blob_length) @@ -2673,7 +2674,7 @@ static int compress_isam_file(PACK_MRG_INFO *mrg, HUFF_COUNTS *huff_counts) DBUG_PRINT("fields", ("FIELD_BLOB %lu bytes, bits: %2u", blob_length, count->length_bits)); write_bits(blob_length,count->length_bits); - memcpy_fixed(&blob,end_pos-maria_portable_sizeof_char_ptr, + memcpy_fixed(&blob,end_pos-portable_sizeof_char_ptr, sizeof(char*)); blob_end=blob+blob_length; /* Encode the blob bytes. */ @@ -2952,6 +2953,7 @@ static int save_state(MARIA_HA *isam_file,PACK_MRG_INFO *mrg, options|= HA_OPTION_COMPRESS_RECORD | HA_OPTION_READ_ONLY_DATA; mi_int2store(share->state.header.options,options); + /* Save the original file type of we have to undo the packing later */ share->state.header.org_data_file_type= share->state.header.data_file_type; share->state.header.data_file_type= COMPRESSED_RECORD; diff --git a/storage/myisam/ft_eval.c b/storage/myisam/ft_eval.c index 7eb78861e5e..de01510fdd7 100644 --- a/storage/myisam/ft_eval.c +++ b/storage/myisam/ft_eval.c @@ -48,7 +48,7 @@ int main(int argc, char *argv[]) recinfo[0].type=FIELD_SKIP_ENDSPACE; recinfo[0].length=docid_length; recinfo[1].type=FIELD_BLOB; - recinfo[1].length= 4+mi_portable_sizeof_char_ptr; + recinfo[1].length= 4+portable_sizeof_char_ptr; /* Define a key over the first column */ keyinfo[0].seg=keyseg; diff --git a/storage/myisam/ft_test1.c b/storage/myisam/ft_test1.c index e49c47bb268..b37935a0d7a 100644 --- a/storage/myisam/ft_test1.c +++ b/storage/myisam/ft_test1.c @@ -75,12 +75,12 @@ static int run_test(const char *filename) /* First define 2 columns */ recinfo[0].type=extra_field; - recinfo[0].length= (extra_field == FIELD_BLOB ? 4 + mi_portable_sizeof_char_ptr : + recinfo[0].length= (extra_field == FIELD_BLOB ? 4 + portable_sizeof_char_ptr : extra_length); if (extra_field == FIELD_VARCHAR) recinfo[0].length+= HA_VARCHAR_PACKLENGTH(extra_length); recinfo[1].type=key_field; - recinfo[1].length= (key_field == FIELD_BLOB ? 4+mi_portable_sizeof_char_ptr : + recinfo[1].length= (key_field == FIELD_BLOB ? 4+portable_sizeof_char_ptr : key_length); if (key_field == FIELD_VARCHAR) recinfo[1].length+= HA_VARCHAR_PACKLENGTH(key_length); diff --git a/storage/myisam/mi_checksum.c b/storage/myisam/mi_checksum.c index 711e87c1547..273b0779e26 100644 --- a/storage/myisam/mi_checksum.c +++ b/storage/myisam/mi_checksum.c @@ -31,9 +31,9 @@ ha_checksum mi_checksum(MI_INFO *info, const byte *buf) case FIELD_BLOB: { length=_mi_calc_blob_length(rec->length- - mi_portable_sizeof_char_ptr, + portable_sizeof_char_ptr, buf); - memcpy((char*) &pos, buf+rec->length- mi_portable_sizeof_char_ptr, + memcpy((char*) &pos, buf+rec->length- portable_sizeof_char_ptr, sizeof(char*)); break; } diff --git a/storage/myisam/mi_create.c b/storage/myisam/mi_create.c index 2c2e6b9e101..9afe3d57ab6 100644 --- a/storage/myisam/mi_create.c +++ b/storage/myisam/mi_create.c @@ -117,10 +117,10 @@ int mi_create(const char *name,uint keys,MI_KEYDEF *keydefs, share.base.blobs++; if (pack_reclength != INT_MAX32) { - if (rec->length == 4+mi_portable_sizeof_char_ptr) + if (rec->length == 4+portable_sizeof_char_ptr) pack_reclength= INT_MAX32; else - pack_reclength+=(1 << ((rec->length-mi_portable_sizeof_char_ptr)*8)); /* Max blob length */ + pack_reclength+=(1 << ((rec->length-portable_sizeof_char_ptr)*8)); /* Max blob length */ } } else if (type == FIELD_SKIP_PRESPACE || diff --git a/storage/myisam/mi_dynrec.c b/storage/myisam/mi_dynrec.c index d56df7b269b..534d5d9563b 100644 --- a/storage/myisam/mi_dynrec.c +++ b/storage/myisam/mi_dynrec.c @@ -900,7 +900,7 @@ uint _mi_rec_pack(MI_INFO *info, register byte *to, register const byte *from) else { char *temp_pos; - size_t tmp_length=length-mi_portable_sizeof_char_ptr; + size_t tmp_length=length-portable_sizeof_char_ptr; memcpy((byte*) to,from,tmp_length); memcpy_fixed(&temp_pos,from+tmp_length,sizeof(char*)); memcpy(to+tmp_length,temp_pos,(size_t) blob->length); @@ -1021,11 +1021,11 @@ my_bool _mi_rec_check(MI_INFO *info,const char *record, byte *rec_buff, if (type == FIELD_BLOB) { uint blob_length= - _mi_calc_blob_length(length-mi_portable_sizeof_char_ptr,record); + _mi_calc_blob_length(length-portable_sizeof_char_ptr,record); if (!blob_length && !(flag & bit)) goto err; if (blob_length) - to+=length - mi_portable_sizeof_char_ptr+ blob_length; + to+=length - portable_sizeof_char_ptr+ blob_length; } else if (type == FIELD_SKIP_ZERO) { @@ -1208,7 +1208,7 @@ ulong _mi_rec_unpack(register MI_INFO *info, register byte *to, byte *from, } else if (type == FIELD_BLOB) { - uint size_length=rec_length- mi_portable_sizeof_char_ptr; + uint size_length=rec_length- portable_sizeof_char_ptr; ulong blob_length=_mi_calc_blob_length(size_length,from); ulong from_left= (ulong) (from_end - from); if (from_left < size_length || diff --git a/storage/myisam/mi_open.c b/storage/myisam/mi_open.c index 6fd7d7a571f..af7206dd335 100644 --- a/storage/myisam/mi_open.c +++ b/storage/myisam/mi_open.c @@ -453,7 +453,7 @@ MI_INFO *mi_open(const char *name, int mode, uint open_flags) if (share->rec[i].type == (int) FIELD_BLOB) { share->blobs[j].pack_length= - share->rec[i].length-mi_portable_sizeof_char_ptr;; + share->rec[i].length-portable_sizeof_char_ptr;; share->blobs[j].offset=offset; j++; } diff --git a/storage/myisam/mi_packrec.c b/storage/myisam/mi_packrec.c index c74bfb5af41..d2676572569 100644 --- a/storage/myisam/mi_packrec.c +++ b/storage/myisam/mi_packrec.c @@ -1036,7 +1036,7 @@ static void uf_blob(MI_COLUMNDEF *rec, MI_BIT_BUFF *bit_buff, else { ulong length=get_bits(bit_buff,rec->space_length_bits); - uint pack_length=(uint) (end-to)-mi_portable_sizeof_char_ptr; + uint pack_length=(uint) (end-to)-portable_sizeof_char_ptr; if (bit_buff->blob_pos+length > bit_buff->blob_end) { bit_buff->error=1; diff --git a/storage/myisam/mi_rkey.c b/storage/myisam/mi_rkey.c index 917ba381504..77b2783cf56 100644 --- a/storage/myisam/mi_rkey.c +++ b/storage/myisam/mi_rkey.c @@ -83,6 +83,8 @@ int mi_rkey(MI_INFO *info, byte *buf, int inx, const byte *key, uint key_len, { mi_print_error(info->s, HA_ERR_CRASHED); my_errno=HA_ERR_CRASHED; + if (share->concurrent_insert) + rw_unlock(&share->key_root_lock[inx]); goto err; } break; diff --git a/storage/myisam/mi_test1.c b/storage/myisam/mi_test1.c index c5a1ffcd5d7..0a11c3dbb74 100644 --- a/storage/myisam/mi_test1.c +++ b/storage/myisam/mi_test1.c @@ -71,12 +71,12 @@ static int run_test(const char *filename) /* First define 2 columns */ recinfo[0].type=FIELD_NORMAL; recinfo[0].length=1; /* For NULL bits */ recinfo[1].type=key_field; - recinfo[1].length= (key_field == FIELD_BLOB ? 4+mi_portable_sizeof_char_ptr : + recinfo[1].length= (key_field == FIELD_BLOB ? 4+portable_sizeof_char_ptr : key_length); if (key_field == FIELD_VARCHAR) recinfo[1].length+= HA_VARCHAR_PACKLENGTH(key_length);; recinfo[2].type=extra_field; - recinfo[2].length= (extra_field == FIELD_BLOB ? 4 + mi_portable_sizeof_char_ptr : 24); + recinfo[2].length= (extra_field == FIELD_BLOB ? 4 + portable_sizeof_char_ptr : 24); if (extra_field == FIELD_VARCHAR) recinfo[2].length+= HA_VARCHAR_PACKLENGTH(recinfo[2].length); if (opt_unique) diff --git a/storage/myisam/mi_test2.c b/storage/myisam/mi_test2.c index ef58f8776b5..6f7c1c980c5 100644 --- a/storage/myisam/mi_test2.c +++ b/storage/myisam/mi_test2.c @@ -188,7 +188,7 @@ int main(int argc, char *argv[]) if (use_blob) { recinfo[6].type=FIELD_BLOB; - recinfo[6].length=4+mi_portable_sizeof_char_ptr; + recinfo[6].length=4+portable_sizeof_char_ptr; recinfo[6].null_bit=0; recinfo[6].null_pos=0; } diff --git a/storage/myisam/myisampack.c b/storage/myisam/myisampack.c index fb631b5e63e..70dd7835c67 100644 --- a/storage/myisam/myisampack.c +++ b/storage/myisam/myisampack.c @@ -305,7 +305,7 @@ static void usage(void) puts("and you are welcome to modify and redistribute it under the GPL license\n"); puts("Pack a MyISAM-table to take much less space."); - puts("Keys are not updated, you must run myisamchk -rq on the datafile"); + puts("Keys are not updated, you must run myisamchk -rq on the index (.MYI) file"); puts("afterwards to update the keys."); puts("You should give the .MYI file as the filename argument."); @@ -1008,7 +1008,7 @@ static int get_statistic(PACK_MRG_INFO *mrg,HUFF_COUNTS *huff_counts) /* Calculate pos, end_pos, and max_length for variable length fields. */ if (count->field_type == FIELD_BLOB) { - uint field_length=count->field_length -mi_portable_sizeof_char_ptr; + uint field_length=count->field_length -portable_sizeof_char_ptr; ulong blob_length= _mi_calc_blob_length(field_length, start_pos); memcpy_fixed((char*) &pos, start_pos+field_length,sizeof(char*)); end_pos=pos+blob_length; @@ -2650,7 +2650,7 @@ static int compress_isam_file(PACK_MRG_INFO *mrg, HUFF_COUNTS *huff_counts) case FIELD_BLOB: { ulong blob_length=_mi_calc_blob_length(field_length- - mi_portable_sizeof_char_ptr, + portable_sizeof_char_ptr, start_pos); /* Empty blobs are encoded with a single 1 bit. */ if (!blob_length) @@ -2667,7 +2667,7 @@ static int compress_isam_file(PACK_MRG_INFO *mrg, HUFF_COUNTS *huff_counts) DBUG_PRINT("fields", ("FIELD_BLOB %lu bytes, bits: %2u", blob_length, count->length_bits)); write_bits(blob_length,count->length_bits); - memcpy_fixed(&blob,end_pos-mi_portable_sizeof_char_ptr, + memcpy_fixed(&blob,end_pos-portable_sizeof_char_ptr, sizeof(char*)); blob_end=blob+blob_length; /* Encode the blob bytes. */ diff --git a/storage/myisam/sp_test.c b/storage/myisam/sp_test.c index c7226589811..8078942e44a 100644 --- a/storage/myisam/sp_test.c +++ b/storage/myisam/sp_test.c @@ -79,7 +79,7 @@ int run_test(const char *filename) /* Define spatial column */ recinfo[1].type=FIELD_BLOB; - recinfo[1].length=4 + mi_portable_sizeof_char_ptr; + recinfo[1].length=4 + portable_sizeof_char_ptr; diff --git a/support-files/magic b/support-files/magic index e2bd4d70fb5..b3f3b3ea29d 100644 --- a/support-files/magic +++ b/support-files/magic @@ -4,9 +4,9 @@ # 0 beshort 0xfe01 MySQL table definition file >2 byte x Version %d -0 belong&0xffffff00 0xfefe0700 MySQL MISAM index file +0 belong&0xffffff00 0xfefe0700 MySQL MyISAM index file >3 byte x Version %d -0 belong&0xffffff00 0xfefe0800 MySQL MISAM compressed data file +0 belong&0xffffff00 0xfefe0800 MySQL MyISAM compressed data file >3 byte x Version %d 0 belong&0xffffff00 0xfefe0900 MySQL Maria index file >3 byte x Version %d -- cgit v1.2.1 From 6686a3ee53e7260047a2737b854f400f6847b452 Mon Sep 17 00:00:00 2001 From: unknown Date: Thu, 19 Apr 2007 18:48:36 +0300 Subject: After merge fixes Read blocks through page cache in check_block_record() Don't read first bitmap on ma_open() Don't require that a files block_size is equal to maria_block_size, if page cache is not setup yet. Changed ma_test1, ma_test2, maria_chk and maria_pack to always create a page cache. The above fixes so that ma_test_all now works again BitKeeper/etc/ignore: added storage/maria/unittest/ma_pagecache_consist_1k-t-big storage/maria/unittest/ma_pagecache_consist_1kHC-t-big storage/maria/unittest/ma_pagecache_consist_1kRD-t-big storage/maria/unittest/ma_pagecache_consist_1kWR-t-big storage/maria/unittest/ma_pagecache_consist_64k-t-big storage/maria/unittest/ma_pagecache_consist_64kHC-t-big storage/maria/unittest/ma_pagecache_consist_64kRD-t-big storage/maria/unittest/ma_pagecache_consist_64kWR-t-big storage/maria/unittest/ma_pagecache_single_64k-t-big include/maria.h: Added MARIA_MIN_PAGE_CACHE_SIZE include/pagecache.h: Filedescriptors should be of type File storage/maria/ma_bitmap.c: After merge fixes Create dummy bitmap on startup (can't read first bitmap becasue page cache may not be set up yet) storage/maria/ma_blockrec.c: After merge fixes storage/maria/ma_check.c: Use page cache to read rows-in-block rows. Don't initialize page cache; It's now done in maria_chk storage/maria/ma_dynrec.c: Trivial code reorganization storage/maria/ma_open.c: Don't give error for conflicting block size if page cache is not initalized. (Needed for maria_chk to be able to work on tables with different page sizes) After merge fixes storage/maria/ma_page.c: Fix compiler warning Remove net needed asserts (Guranteed by ma_create()) storage/maria/ma_pagecache.c: Allow one to create a page cache with just one block (For trivail scan of table) Trivial code simplication storage/maria/ma_test1.c: Always create a page cache (Maria now requires a page cache to work) storage/maria/ma_test2.c: Always create a page cache (Maria now requires a page cache to work) storage/maria/maria_chk.c: Remove command line options --maria_block_size and --pagecache_block_size. Set the global maria_block_size from the data file. This allows maria_chk to work with tables of different block sizes. Simply DESCRIPT handling; Allows us to remove one indentation level in maria_chk(). Always initialize page cache if we are doing check/repair. (Most of the patch is reindentation of the code) storage/maria/maria_def.h: After merge fix storage/maria/maria_pack.c: Set maria_block_size based on the files block_size. Initalize page cache (needed for getting rows-in-blocks to works) --- .bzrignore | 9 ++ include/maria.h | 2 + include/pagecache.h | 2 +- storage/maria/ma_bitmap.c | 15 +- storage/maria/ma_blockrec.c | 15 +- storage/maria/ma_check.c | 26 +-- storage/maria/ma_dynrec.c | 13 +- storage/maria/ma_open.c | 11 +- storage/maria/ma_page.c | 7 +- storage/maria/ma_pagecache.c | 19 ++- storage/maria/ma_test1.c | 4 +- storage/maria/ma_test2.c | 31 +--- storage/maria/maria_chk.c | 377 +++++++++++++++++++++---------------------- storage/maria/maria_def.h | 4 +- storage/maria/maria_pack.c | 13 +- 15 files changed, 281 insertions(+), 267 deletions(-) diff --git a/.bzrignore b/.bzrignore index cd469d20721..0c52723de3c 100644 --- a/.bzrignore +++ b/.bzrignore @@ -2989,3 +2989,12 @@ win/vs71cache.txt win/vs8cache.txt zlib/*.ds? zlib/*.vcproj +storage/maria/unittest/ma_pagecache_consist_1k-t-big +storage/maria/unittest/ma_pagecache_consist_1kHC-t-big +storage/maria/unittest/ma_pagecache_consist_1kRD-t-big +storage/maria/unittest/ma_pagecache_consist_1kWR-t-big +storage/maria/unittest/ma_pagecache_consist_64k-t-big +storage/maria/unittest/ma_pagecache_consist_64kHC-t-big +storage/maria/unittest/ma_pagecache_consist_64kRD-t-big +storage/maria/unittest/ma_pagecache_consist_64kWR-t-big +storage/maria/unittest/ma_pagecache_single_64k-t-big diff --git a/include/maria.h b/include/maria.h index 2e36a3c50ca..da0cb4ba0e5 100644 --- a/include/maria.h +++ b/include/maria.h @@ -54,6 +54,8 @@ extern "C" { #define MARIA_KEY_BLOCK_LENGTH 8192 /* default key block length */ #define MARIA_MIN_KEY_BLOCK_LENGTH 1024 /* Min key block length */ #define MARIA_MAX_KEY_BLOCK_LENGTH 32768 +/* Minimal page cache when we only want to be able to scan a table */ +#define MARIA_MIN_PAGE_CACHE_SIZE 65536 /* In the following macros '_keyno_' is 0 .. keys-1. diff --git a/include/pagecache.h b/include/pagecache.h index 08ecc7e5f4a..b917271f0c9 100644 --- a/include/pagecache.h +++ b/include/pagecache.h @@ -75,7 +75,7 @@ typedef void *PAGECACHE_PAGE_LINK; /* file descriptor for Maria */ typedef struct st_pagecache_file { - int file; /* it is for debugging purposes then it will be uint32 file_no */ + File file; } PAGECACHE_FILE; /* page number for maria */ diff --git a/storage/maria/ma_bitmap.c b/storage/maria/ma_bitmap.c index 97ebedb5ac4..ed5bd070981 100644 --- a/storage/maria/ma_bitmap.c +++ b/storage/maria/ma_bitmap.c @@ -131,7 +131,7 @@ static inline my_bool write_changed_bitmap(MARIA_SHARE *share, { DBUG_ASSERT(share->pagecache->block_size == bitmap->block_size); return (pagecache_write(share->pagecache, - (PAGECACHE_FILE*)&bitmap->file, bitmap->page, 0, + &bitmap->file, bitmap->page, 0, (byte*) bitmap->map, PAGECACHE_PLAIN_PAGE, PAGECACHE_LOCK_LEFT_UNLOCKED, PAGECACHE_PIN_LEFT_UNPINNED, @@ -168,7 +168,7 @@ my_bool _ma_bitmap_init(MARIA_SHARE *share, File file) if (!(bitmap->map= (uchar*) my_malloc(size, MYF(MY_WME)))) return 1; - bitmap->file= file; + bitmap->file.file= file; bitmap->changed= 0; bitmap->block_size= share->block_size; /* Size needs to be alligned on 6 */ @@ -195,10 +195,15 @@ my_bool _ma_bitmap_init(MARIA_SHARE *share, File file) pthread_mutex_init(&share->bitmap.bitmap_lock, MY_MUTEX_INIT_SLOW); /* - Start by reading first page (assume table scan) - Later code is simpler if it can assume we always have an active bitmap. + We can't read a page yet, as in some case we don't have an active + page cache yet. + Pretend we have a dummy, full and not changed bitmap page in memory. */ - return _ma_read_bitmap_page(share, bitmap, (ulonglong) 0); + + bitmap->page= ~(ulonglong) 0; + bitmap->used_size= bitmap->total_size; + bfill(bitmap->map, share->block_size, 255); + return 0; } diff --git a/storage/maria/ma_blockrec.c b/storage/maria/ma_blockrec.c index e05f2fcfe6f..55b16a64916 100644 --- a/storage/maria/ma_blockrec.c +++ b/storage/maria/ma_blockrec.c @@ -374,21 +374,20 @@ my_bool _ma_once_init_block_record(MARIA_SHARE *share, File data_file) my_bool _ma_once_end_block_record(MARIA_SHARE *share) { int res= _ma_bitmap_end(share); - if (share->bitmap.file >= 0) + if (share->bitmap.file.file >= 0) { - if (flush_pagecache_blocks(share->pagecache, (PAGECACHE_FILE*)&share->bitmap, - if (flush_key_blocks(share->key_cache, share->bitmap.file, - share->temporary ? FLUSH_IGNORE_CHANGED : - FLUSH_RELEASE)) + if (flush_pagecache_blocks(share->pagecache, &share->bitmap.file, + share->temporary ? FLUSH_IGNORE_CHANGED : + FLUSH_RELEASE)) res= 1; - if (my_close(share->bitmap.file, MYF(MY_WME))) + if (my_close(share->bitmap.file.file, MYF(MY_WME))) res= 1; /* Trivial assignment to guard against multiple invocations (May happen if file are closed but we want to keep the maria object around a bit longer) */ - share->bitmap.file= -1; + share->bitmap.file.file= -1; } return res; } @@ -450,7 +449,7 @@ void _ma_end_block_record(MARIA_HA *info) The following protects us from doing an extra, not allowed, close in maria_close() */ - info->dfile= -1; + info->dfile.file= -1; DBUG_VOID_RETURN; } diff --git a/storage/maria/ma_check.c b/storage/maria/ma_check.c index 259ff90478b..4087f12ba43 100644 --- a/storage/maria/ma_check.c +++ b/storage/maria/ma_check.c @@ -1599,8 +1599,12 @@ static int check_block_record(HA_CHECK *param, MARIA_HA *info, int extend, if (((pos / block_size) % info->s->bitmap.pages_covered) == 0) { /* Bitmap page */ - if (_ma_read_cache(¶m->read_cache, bitmap_buff, pos, - block_size, READING_NEXT)) + if (pagecache_read(info->s->pagecache, + &info->dfile, + (pos / block_size), 1, + bitmap_buff, + PAGECACHE_PLAIN_PAGE, + PAGECACHE_LOCK_LEFT_UNLOCKED, 0) == 0) { _ma_check_print_error(param, "Page %9s: Got error: %d when reading datafile", @@ -1624,8 +1628,12 @@ static int check_block_record(HA_CHECK *param, MARIA_HA *info, int extend, continue; } - if (_ma_read_cache(¶m->read_cache, page_buff, pos, - block_size, READING_NEXT)) + if (pagecache_read(info->s->pagecache, + &info->dfile, + (pos / block_size), 1, + page_buff, + PAGECACHE_PLAIN_PAGE, + PAGECACHE_LOCK_LEFT_UNLOCKED, 0) == 0) { _ma_check_print_error(param, "Page %9s: Got error: %d when reading datafile", @@ -1935,10 +1943,6 @@ int maria_repair(HA_CHECK *param, register MARIA_HA *info, if (info->s->options & (HA_OPTION_CHECKSUM | HA_OPTION_COMPRESS_RECORD)) param->testflag|=T_CALC_CHECKSUM; - if (!param->using_global_keycache) - VOID(init_pagecache(maria_pagecache, param->use_buffers, 0, 0, - param->pagecache_block_size)); - if (init_io_cache(¶m->read_cache, info->dfile.file, (uint) param->read_buffer_length, READ_CACHE,share->pack.header_length,1,MYF(MY_WME))) @@ -2308,8 +2312,6 @@ int _ma_flush_blocks(HA_CHECK *param, PAGECACHE *pagecache, _ma_check_print_error(param,"%d when trying to write bufferts",my_errno); return(1); } - if (!param->using_global_keycache) - end_pagecache(pagecache,1); return 0; } /* _ma_flush_blocks */ @@ -3602,7 +3604,6 @@ static int sort_get_next_record(MARIA_SORT_PARAM *sort_param) int parallel_flag; uint found_record,b_type,left_length; my_off_t pos; - byte *to; MARIA_BLOCK_INFO block_info; MARIA_SORT_INFO *sort_info=sort_param->sort_info; HA_CHECK *param=sort_info->param; @@ -3652,6 +3653,8 @@ static int sort_get_next_record(MARIA_SORT_PARAM *sort_param) } } case DYNAMIC_RECORD: + { + byte *to; LINT_INIT(to); pos=sort_param->pos; searching=(sort_param->fix_datafile && (param->testflag & T_EXTEND)); @@ -3948,6 +3951,7 @@ static int sort_get_next_record(MARIA_SORT_PARAM *sort_param) pos=(sort_param->start_recpos+=MARIA_DYN_ALIGN_SIZE); searching=1; } + } case COMPRESSED_RECORD: for (searching=0 ;; searching=1, sort_param->pos++) { diff --git a/storage/maria/ma_dynrec.c b/storage/maria/ma_dynrec.c index a5f3f3c4900..d6f7309cafa 100644 --- a/storage/maria/ma_dynrec.c +++ b/storage/maria/ma_dynrec.c @@ -1372,13 +1372,15 @@ int _ma_read_dynamic_record(MARIA_HA *info, byte *buf, { int block_of_record; uint b_type,left_length; - byte *to; MARIA_BLOCK_INFO block_info; File file; DBUG_ENTER("_ma_read_dynamic_record"); if (filepos != HA_OFFSET_ERROR) { + byte *to; + uint left_length; + LINT_INIT(to); LINT_INIT(left_length); file= info->dfile.file; @@ -1390,13 +1392,14 @@ int _ma_read_dynamic_record(MARIA_HA *info, byte *buf, if (filepos == HA_OFFSET_ERROR) goto panic; if (info->opt_flag & WRITE_CACHE_USED && - info->rec_cache.pos_in_file < filepos + MARIA_BLOCK_INFO_HEADER_LENGTH && + (info->rec_cache.pos_in_file < filepos + + MARIA_BLOCK_INFO_HEADER_LENGTH) && flush_io_cache(&info->rec_cache)) goto err; info->rec_cache.seek_not_done=1; - if ((b_type= _ma_get_block_info(&block_info, file, filepos)) - & (BLOCK_DELETED | BLOCK_ERROR | BLOCK_SYNC_ERROR | - BLOCK_FATAL_ERROR)) + if ((b_type= _ma_get_block_info(&block_info, file, filepos)) & + (BLOCK_DELETED | BLOCK_ERROR | BLOCK_SYNC_ERROR | + BLOCK_FATAL_ERROR)) { if (b_type & (BLOCK_SYNC_ERROR | BLOCK_DELETED)) my_errno=HA_ERR_RECORD_DELETED; diff --git a/storage/maria/ma_open.c b/storage/maria/ma_open.c index 3a95d705743..9b7313a469b 100644 --- a/storage/maria/ma_open.c +++ b/storage/maria/ma_open.c @@ -239,7 +239,12 @@ MARIA_HA *maria_open(const char *name, int mode, uint open_flags) my_errno=HA_ERR_UNSUPPORTED; goto err; } - if (share->base.block_size != maria_block_size) + /* + If page cache is not initialized, then assume we will create it + after the table is opened! + */ + if (share->base.block_size != maria_block_size && + share_buff.pagecache->inited != 0) { DBUG_PRINT("error", ("Wrong block size %u; Expected %u", (uint) share->base.block_size, @@ -301,7 +306,7 @@ MARIA_HA *maria_open(const char *name, int mode, uint open_flags) strmov(share->index_file_name, index_name); strmov(share->data_file_name, data_name); - share->block_size= maria_block_size; + share->block_size= share->base.block_size; { HA_KEYSEG *pos=share->keyparts; for (i=0 ; i < keys ; i++) @@ -553,7 +558,7 @@ MARIA_HA *maria_open(const char *name, int mode, uint open_flags) goto err; } if (share->data_file_type == BLOCK_RECORD) - info.dfile.file= share->bitmap.file; + info.dfile= share->bitmap.file; else if (_ma_open_datafile(&info, share, old_info->dfile.file)) goto err; errpos= 5; diff --git a/storage/maria/ma_page.c b/storage/maria/ma_page.c index 91a3b42009e..2aaabb1257d 100644 --- a/storage/maria/ma_page.c +++ b/storage/maria/ma_page.c @@ -21,15 +21,15 @@ byte *_ma_fetch_keypage(register MARIA_HA *info, MARIA_KEYDEF *keyinfo, my_off_t page, int level, - byte *buff, int return_buffer) + byte *buff, + int return_buffer __attribute__ ((unused))) { byte *tmp; uint page_size; DBUG_ENTER("_ma_fetch_keypage"); DBUG_PRINT("enter",("page: %ld", (long) page)); - DBUG_ASSERT(info->s->pagecache->block_size == keyinfo->block_length && - info->s->pagecache->block_size == info->s->block_size); + DBUG_ASSERT(info->s->pagecache->block_size == keyinfo->block_length); /* TODO: replace PAGECACHE_PLAIN_PAGE with PAGECACHE_LSN_PAGE when LSN on the pages will be implemented @@ -88,7 +88,6 @@ int _ma_write_keypage(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, #endif DBUG_ASSERT(info->s->pagecache->block_size == keyinfo->block_length); - DBUG_ASSERT(info->s->pagecache->block_size == info->s->block_size); /* TODO: replace PAGECACHE_PLAIN_PAGE with PAGECACHE_LSN_PAGE when LSN on the pages will be implemented diff --git a/storage/maria/ma_pagecache.c b/storage/maria/ma_pagecache.c index 3d5c3026173..b7f30eae625 100755 --- a/storage/maria/ma_pagecache.c +++ b/storage/maria/ma_pagecache.c @@ -675,8 +675,11 @@ int init_pagecache(PAGECACHE *pagecache, my_size_t use_mem, 2 * sizeof(PAGECACHE_HASH_LINK) + sizeof(PAGECACHE_HASH_LINK*) * 5/4 + block_size)); - /* It doesn't make sense to have too few blocks (less than 8) */ - if (blocks >= 8 && pagecache->disk_blocks < 0) + /* + We need to support page cache with just one block to be able to do + scanning of rows-in-block files + */ + if (blocks >= 1) { for ( ; ; ) { @@ -3768,7 +3771,7 @@ my_bool pagecache_collect_changed_blocks_with_lsn(PAGECACHE *pagecache, LEX_STRING *str, LSN *max_lsn) { - my_bool error; + my_bool error= 0; ulong stored_list_size= 0; uint file_hash; char *ptr; @@ -3836,7 +3839,7 @@ my_bool pagecache_collect_changed_blocks_with_lsn(PAGECACHE *pagecache, ptr= str->str; int8store(ptr, stored_list_size); ptr+= 8; - if (0 == stored_list_size) + if (!stored_list_size) goto end; for (file_hash= 0; file_hash < PAGECACHE_CHANGED_BLOCKS_HASH; file_hash++) { @@ -3858,13 +3861,13 @@ my_bool pagecache_collect_changed_blocks_with_lsn(PAGECACHE *pagecache, set_if_bigger(*max_lsn, block->rec_lsn); } } - error= 0; - goto end; -err: - error= 1; end: pagecache_pthread_mutex_unlock(&pagecache->cache_lock); DBUG_RETURN(error); + +err: + error= 1; + goto end; } diff --git a/storage/maria/ma_test1.c b/storage/maria/ma_test1.c index fc3ffca7f0e..1c69e2c95b4 100644 --- a/storage/maria/ma_test1.c +++ b/storage/maria/ma_test1.c @@ -51,8 +51,8 @@ int main(int argc,char *argv[]) my_init(); maria_init(); get_options(argc,argv); - if (pagecacheing) - init_pagecache(maria_pagecache, IO_SIZE*16, 0, 0, MARIA_KEY_BLOCK_LENGTH); + /* Maria requires that we always have a page cache */ + init_pagecache(maria_pagecache, IO_SIZE*16, 0, 0, maria_block_size); exit(run_test("test1")); } diff --git a/storage/maria/ma_test2.c b/storage/maria/ma_test2.c index c6dde12f5a3..18eaf27073b 100644 --- a/storage/maria/ma_test2.c +++ b/storage/maria/ma_test2.c @@ -50,7 +50,6 @@ static int verbose=0,testflag=0, static int pack_seg=HA_SPACE_PACK,pack_type=HA_PACK_KEY,remove_count=-1; static int create_flag= 0, srand_arg= 0; static ulong pagecache_size=IO_SIZE*16; -static uint pagecache_block_size= MARIA_KEY_BLOCK_LENGTH; static enum data_file_type record_type= DYNAMIC_RECORD; static uint keys=MARIA_KEYS,recant=1000; @@ -219,8 +218,8 @@ int main(int argc, char *argv[]) goto err; if (!silent) printf("- Writing key:s\n"); - if (pagecacheing) - init_pagecache(maria_pagecache, pagecache_size, 0, 0, pagecache_block_size); + /* Maria requires that we always have a page cache */ + init_pagecache(maria_pagecache, pagecache_size, 0, 0, maria_block_size); if (locking) maria_lock_database(file,F_WRLCK); if (write_cacheing) @@ -282,12 +281,11 @@ int main(int argc, char *argv[]) goto end; } } - /* - TODO: uncomment when resize will be implemented +#ifdef REMOVE_WHEN_WE_HAVE_RESIZE if (pagecacheing) - resize_pagecache(maria_pagecache, pagecache_block_size, + resize_pagecache(maria_pagecache, maria_block_size, pagecache_size * 2, 0, 0); - */ +#endif if (!silent) printf("- Delete\n"); if (srand_arg) @@ -862,13 +860,8 @@ end: if (rec_pointer_size) printf("Record pointer size: %d\n",rec_pointer_size); printf("maria_block_size: %lu\n", maria_block_size); - if (pagecacheing) - { - puts("Key cache used"); - printf("pagecache_block_size: %u\n", pagecache_block_size); - if (write_cacheing) - puts("Key cache resized"); - } + if (write_cacheing) + puts("Key cache resized"); if (write_cacheing) puts("Write cacheing used"); if (write_cacheing) @@ -960,6 +953,7 @@ static void get_options(int argc, char **argv) } break; case 'e': /* maria_block_length */ + case 'E': if ((maria_block_size= atoi(++pos)) < MARIA_MIN_KEY_BLOCK_LENGTH || maria_block_size > MARIA_MAX_KEY_BLOCK_LENGTH) { @@ -968,15 +962,6 @@ static void get_options(int argc, char **argv) } maria_block_size= my_round_up_to_next_power(maria_block_size); break; - case 'E': /* maria_block_length */ - if ((pagecache_block_size=atoi(++pos)) < MARIA_MIN_KEY_BLOCK_LENGTH || - pagecache_block_size > MARIA_MAX_KEY_BLOCK_LENGTH) - { - fprintf(stderr,"Wrong pagecache_block_size\n"); - exit(1); - } - pagecache_block_size= my_round_up_to_next_power(pagecache_block_size); - break; case 'f': if ((first_key=atoi(++pos)) < 0 || first_key >= MARIA_KEYS) first_key=0; diff --git a/storage/maria/maria_chk.c b/storage/maria/maria_chk.c index a1921f91092..36101e7d002 100644 --- a/storage/maria/maria_chk.c +++ b/storage/maria/maria_chk.c @@ -39,8 +39,6 @@ static char **default_argv; static const char *load_default_groups[]= { "maria_chk", 0 }; static const char *set_collation_name, *opt_tmpdir; static CHARSET_INFO *set_collation; -static long opt_maria_block_size; -static long opt_pagecache_block_size; static const char *my_progname_short; static int stopwords_inited= 0; static MY_TMPDIR maria_chk_tmpdir; @@ -301,15 +299,6 @@ static struct my_option my_long_options[] = (gptr*) &check_param.use_buffers, (gptr*) &check_param.use_buffers, 0, GET_ULONG, REQUIRED_ARG, (long) USE_BUFFER_INIT, (long) MALLOC_OVERHEAD, (long) ~0L, (long) MALLOC_OVERHEAD, (long) IO_SIZE, 0}, - { "pagecache_block_size", OPT_KEY_CACHE_BLOCK_SIZE, "", - (gptr*) &opt_pagecache_block_size, - (gptr*) &opt_pagecache_block_size, 0, - GET_LONG, REQUIRED_ARG, MARIA_KEY_BLOCK_LENGTH, MARIA_MIN_KEY_BLOCK_LENGTH, - MARIA_MAX_KEY_BLOCK_LENGTH, 0, MARIA_MIN_KEY_BLOCK_LENGTH, 0}, - { "maria_block_size", OPT_MARIA_BLOCK_SIZE, "", - (gptr*) &opt_maria_block_size, (gptr*) &opt_maria_block_size, 0, - GET_LONG, REQUIRED_ARG, MARIA_KEY_BLOCK_LENGTH, MARIA_MIN_KEY_BLOCK_LENGTH, - MARIA_MAX_KEY_BLOCK_LENGTH, 0, MARIA_MIN_KEY_BLOCK_LENGTH, 0}, { "read_buffer_size", OPT_READ_BUFFER_SIZE, "", (gptr*) &check_param.read_buffer_length, (gptr*) &check_param.read_buffer_length, 0, GET_ULONG, REQUIRED_ARG, @@ -793,14 +782,12 @@ static void get_options(register int *argc,register char ***argv) exit(1); check_param.tmpdir=&maria_chk_tmpdir; - check_param.pagecache_block_size= opt_pagecache_block_size; if (set_collation_name) if (!(set_collation= get_charset_by_name(set_collation_name, MYF(MY_WME)))) exit(1); - maria_block_size=(uint) 1 << my_bit_log2(opt_maria_block_size); return; } /* get options */ @@ -873,6 +860,7 @@ static int maria_chk(HA_CHECK *param, my_string filename) share->options&= ~HA_OPTION_READ_ONLY_DATA; /* We are modifing it */ share->tot_locks-= share->r_locks; share->r_locks=0; + maria_block_size= share->base.block_size; if (share->data_file_type == BLOCK_RECORD && (param->testflag & (T_REP_ANY | T_SORT_RECORDS | T_FAST | T_STATISTICS | @@ -944,8 +932,7 @@ static int maria_chk(HA_CHECK *param, my_string filename) maria_test_if_almost_full(info) || info->s->state.header.file_version[3] != maria_file_magic[3] || (set_collation && - set_collation->number != share->state.header.language) || - maria_block_size != MARIA_KEY_BLOCK_LENGTH)) + set_collation->number != share->state.header.language))) { if (set_collation) param->language= set_collation->number; @@ -975,211 +962,217 @@ static int maria_chk(HA_CHECK *param, my_string filename) param->total_records+=info->state->records; param->total_deleted+=info->state->del; descript(param, info, filename); + maria_close(info); /* Should always succeed */ + return(0); } + + if (!stopwords_inited++) + ft_init_stopwords(); + + if (!(param->testflag & T_READONLY)) + lock_type = F_WRLCK; /* table is changed */ else + lock_type= F_RDLCK; + if (info->lock_type == F_RDLCK) + info->lock_type=F_UNLCK; /* Read only table */ + if (_ma_readinfo(info,lock_type,0)) + { + _ma_check_print_error(param,"Can't lock indexfile of '%s', error: %d", + filename,my_errno); + param->error_printed=0; + error= 1; + goto end2; + } + /* + _ma_readinfo() has locked the table. + We mark the table as locked (without doing file locks) to be able to + use functions that only works on locked tables (like row caching). + */ + maria_lock_database(info, F_EXTRA_LCK); + datafile= info->dfile.file; + if (init_pagecache(maria_pagecache, param->use_buffers, 0, 0, + maria_block_size) == 0) { - if (!stopwords_inited++) - ft_init_stopwords(); + _ma_check_print_error(param, "Can't initialize page cache with %lu memory", + (ulong) param->use_buffers); + error= 1; + goto end2; + } - if (!(param->testflag & T_READONLY)) - lock_type = F_WRLCK; /* table is changed */ - else - lock_type= F_RDLCK; - if (info->lock_type == F_RDLCK) - info->lock_type=F_UNLCK; /* Read only table */ - if (_ma_readinfo(info,lock_type,0)) + if (param->testflag & (T_REP_ANY | T_SORT_RECORDS | T_SORT_INDEX)) + { + if (param->testflag & T_REP_ANY) { - _ma_check_print_error(param,"Can't lock indexfile of '%s', error: %d", - filename,my_errno); - param->error_printed=0; - error= 1; - goto end2; + ulonglong tmp=share->state.key_map; + maria_copy_keys_active(share->state.key_map, share->base.keys, + param->keys_in_use); + if (tmp != share->state.key_map) + info->update|=HA_STATE_CHANGED; } - /* - _ma_readinfo() has locked the table. - We mark the table as locked (without doing file locks) to be able to - use functions that only works on locked tables (like row caching). - */ - maria_lock_database(info, F_EXTRA_LCK); - datafile= info->dfile.file; - - if (param->testflag & (T_REP_ANY | T_SORT_RECORDS | T_SORT_INDEX)) + if (rep_quick && + maria_chk_del(param, info, param->testflag & ~T_VERBOSE)) { - if (param->testflag & T_REP_ANY) - { - ulonglong tmp=share->state.key_map; - maria_copy_keys_active(share->state.key_map, share->base.keys, - param->keys_in_use); - if (tmp != share->state.key_map) - info->update|=HA_STATE_CHANGED; - } - if (rep_quick && - maria_chk_del(param, info, param->testflag & ~T_VERBOSE)) - { - if (param->testflag & T_FORCE_CREATE) - { - rep_quick=0; - _ma_check_print_info(param,"Creating new data file\n"); - } - else - { - error=1; - _ma_check_print_error(param, - "Quick-recover aborted; Run recovery without switch 'q'"); - } - } - if (!error) + if (param->testflag & T_FORCE_CREATE) { - if ((param->testflag & (T_REP_BY_SORT | T_REP_PARALLEL)) && - (maria_is_any_key_active(share->state.key_map) || - (rep_quick && !param->keys_in_use && !recreate)) && - maria_test_if_sort_rep(info, info->state->records, - info->s->state.key_map, - param->force_sort)) - { - if (param->testflag & T_REP_BY_SORT) - error=maria_repair_by_sort(param,info,filename,rep_quick); - else - error=maria_repair_parallel(param,info,filename,rep_quick); - state_updated=1; - } - else if (param->testflag & T_REP_ANY) - error=maria_repair(param, info,filename,rep_quick); + rep_quick=0; + _ma_check_print_info(param,"Creating new data file\n"); } - if (!error && param->testflag & T_SORT_RECORDS) + else { - /* - The data file is nowadays reopened in the repair code so we should - soon remove the following reopen-code - */ -#ifndef TO_BE_REMOVED - if (param->out_flag & O_NEW_DATA) - { /* Change temp file to org file */ - VOID(my_close(info->dfile.file, MYF(MY_WME))); /* Close new file */ - error|=maria_change_to_newfile(filename,MARIA_NAME_DEXT,DATA_TMP_EXT, - MYF(0)); - if (_ma_open_datafile(info,info->s, -1)) - error=1; - param->out_flag&= ~O_NEW_DATA; /* We are using new datafile */ - param->read_cache.file= info->dfile.file; - } -#endif - if (! error) - { - uint key; - /* - We can't update the index in maria_sort_records if we have a - prefix compressed or fulltext index - */ - my_bool update_index=1; - for (key=0 ; key < share->base.keys; key++) - if (share->keyinfo[key].flag & (HA_BINARY_PACK_KEY|HA_FULLTEXT)) - update_index=0; - - error=maria_sort_records(param,info,filename,param->opt_sort_key, - /* what is the following parameter for ? */ - (my_bool) !(param->testflag & T_REP), - update_index); - datafile= info->dfile.file; /* This is now locked */ - if (!error && !update_index) - { - if (param->verbose) - puts("Table had a compressed index; We must now recreate the index"); - error=maria_repair_by_sort(param,info,filename,1); - } - } + error=1; + _ma_check_print_error(param, + "Quick-recover aborted; Run recovery without switch 'q'"); } - if (!error && param->testflag & T_SORT_INDEX) - error=maria_sort_index(param,info,filename); - if (!error) - share->state.changed&= ~(STATE_CHANGED | STATE_CRASHED | - STATE_CRASHED_ON_REPAIR); - else - maria_mark_crashed(info); } - else if ((param->testflag & T_CHECK) || !(param->testflag & T_AUTO_INC)) + if (!error) { - if (!(param->testflag & T_SILENT) || param->testflag & T_INFO) - printf("Checking MARIA file: %s\n",filename); - if (!(param->testflag & T_SILENT)) - printf("Data records: %7s Deleted blocks: %7s\n", - llstr(info->state->records,llbuff), - llstr(info->state->del,llbuff2)); - error =maria_chk_status(param,info); - maria_intersect_keys_active(share->state.key_map, param->keys_in_use); - error =maria_chk_size(param,info); - if (!error || !(param->testflag & (T_FAST | T_FORCE_CREATE))) - error|=maria_chk_del(param, info,param->testflag); - if ((!error || (!(param->testflag & (T_FAST | T_FORCE_CREATE)) && - !param->start_check_pos))) + if ((param->testflag & (T_REP_BY_SORT | T_REP_PARALLEL)) && + (maria_is_any_key_active(share->state.key_map) || + (rep_quick && !param->keys_in_use && !recreate)) && + maria_test_if_sort_rep(info, info->state->records, + info->s->state.key_map, + param->force_sort)) { - error|=maria_chk_key(param, info); - if (!error && (param->testflag & (T_STATISTICS | T_AUTO_INC))) - error=maria_update_state_info(param, info, - ((param->testflag & T_STATISTICS) ? - UPDATE_STAT : 0) | - ((param->testflag & T_AUTO_INC) ? - UPDATE_AUTO_INC : 0)); + if (param->testflag & T_REP_BY_SORT) + error=maria_repair_by_sort(param,info,filename,rep_quick); + else + error=maria_repair_parallel(param,info,filename,rep_quick); + state_updated=1; } - if ((!rep_quick && !error) || - !(param->testflag & (T_FAST | T_FORCE_CREATE))) - { - if (param->testflag & (T_EXTEND | T_MEDIUM)) - VOID(init_pagecache(maria_pagecache, param->use_buffers, 0, 0, - opt_pagecache_block_size)); - VOID(init_io_cache(¶m->read_cache,datafile, - (uint) param->read_buffer_length, - READ_CACHE, - (param->start_check_pos ? - param->start_check_pos : - share->pack.header_length), - 1, - MYF(MY_WME))); - maria_lock_memory(param); - if ((info->s->data_file_type != STATIC_RECORD) || - (param->testflag & (T_EXTEND | T_MEDIUM))) - error|=maria_chk_data_link(param, info, param->testflag & T_EXTEND); - error|=_ma_flush_blocks(param, share->pagecache, &share->kfile); - VOID(end_io_cache(¶m->read_cache)); + else if (param->testflag & T_REP_ANY) + error=maria_repair(param, info,filename,rep_quick); + } + if (!error && param->testflag & T_SORT_RECORDS) + { + /* + The data file is nowadays reopened in the repair code so we should + soon remove the following reopen-code + */ +#ifndef TO_BE_REMOVED + if (param->out_flag & O_NEW_DATA) + { /* Change temp file to org file */ + VOID(my_close(info->dfile.file, MYF(MY_WME))); /* Close new file */ + error|=maria_change_to_newfile(filename,MARIA_NAME_DEXT,DATA_TMP_EXT, + MYF(0)); + if (_ma_open_datafile(info,info->s, -1)) + error=1; + param->out_flag&= ~O_NEW_DATA; /* We are using new datafile */ + param->read_cache.file= info->dfile.file; } - if (!error) +#endif + if (! error) { - if ((share->state.changed & STATE_CHANGED) && - (param->testflag & T_UPDATE_STATE)) - info->update|=HA_STATE_CHANGED | HA_STATE_ROW_CHANGED; - share->state.changed&= ~(STATE_CHANGED | STATE_CRASHED | - STATE_CRASHED_ON_REPAIR); - } - else if (!maria_is_crashed(info) && - (param->testflag & T_UPDATE_STATE)) - { /* Mark crashed */ - maria_mark_crashed(info); - info->update|=HA_STATE_CHANGED | HA_STATE_ROW_CHANGED; + uint key; + /* + We can't update the index in maria_sort_records if we have a + prefix compressed or fulltext index + */ + my_bool update_index=1; + for (key=0 ; key < share->base.keys; key++) + if (share->keyinfo[key].flag & (HA_BINARY_PACK_KEY|HA_FULLTEXT)) + update_index=0; + + error=maria_sort_records(param,info,filename,param->opt_sort_key, + /* what is the following parameter for ? */ + (my_bool) !(param->testflag & T_REP), + update_index); + datafile= info->dfile.file; /* This is now locked */ + if (!error && !update_index) + { + if (param->verbose) + puts("Table had a compressed index; We must now recreate the index"); + error=maria_repair_by_sort(param,info,filename,1); + } } } + if (!error && param->testflag & T_SORT_INDEX) + error=maria_sort_index(param,info,filename); + if (!error) + share->state.changed&= ~(STATE_CHANGED | STATE_CRASHED | + STATE_CRASHED_ON_REPAIR); + else + maria_mark_crashed(info); + } + else if ((param->testflag & T_CHECK) || !(param->testflag & T_AUTO_INC)) + { + if (!(param->testflag & T_SILENT) || param->testflag & T_INFO) + printf("Checking MARIA file: %s\n",filename); + if (!(param->testflag & T_SILENT)) + printf("Data records: %7s Deleted blocks: %7s\n", + llstr(info->state->records,llbuff), + llstr(info->state->del,llbuff2)); + error =maria_chk_status(param,info); + maria_intersect_keys_active(share->state.key_map, param->keys_in_use); + error =maria_chk_size(param,info); + if (!error || !(param->testflag & (T_FAST | T_FORCE_CREATE))) + error|=maria_chk_del(param, info,param->testflag); + if ((!error || (!(param->testflag & (T_FAST | T_FORCE_CREATE)) && + !param->start_check_pos))) + { + error|=maria_chk_key(param, info); + if (!error && (param->testflag & (T_STATISTICS | T_AUTO_INC))) + error=maria_update_state_info(param, info, + ((param->testflag & T_STATISTICS) ? + UPDATE_STAT : 0) | + ((param->testflag & T_AUTO_INC) ? + UPDATE_AUTO_INC : 0)); + } + if ((!rep_quick && !error) || + !(param->testflag & (T_FAST | T_FORCE_CREATE))) + { + VOID(init_io_cache(¶m->read_cache,datafile, + (uint) param->read_buffer_length, + READ_CACHE, + (param->start_check_pos ? + param->start_check_pos : + share->pack.header_length), + 1, + MYF(MY_WME))); + maria_lock_memory(param); + if ((info->s->data_file_type != STATIC_RECORD) || + (param->testflag & (T_EXTEND | T_MEDIUM))) + error|=maria_chk_data_link(param, info, param->testflag & T_EXTEND); + error|=_ma_flush_blocks(param, share->pagecache, &share->kfile); + VOID(end_io_cache(¶m->read_cache)); + } + if (!error) + { + if ((share->state.changed & STATE_CHANGED) && + (param->testflag & T_UPDATE_STATE)) + info->update|=HA_STATE_CHANGED | HA_STATE_ROW_CHANGED; + share->state.changed&= ~(STATE_CHANGED | STATE_CRASHED | + STATE_CRASHED_ON_REPAIR); + } + else if (!maria_is_crashed(info) && + (param->testflag & T_UPDATE_STATE)) + { /* Mark crashed */ + maria_mark_crashed(info); + info->update|=HA_STATE_CHANGED | HA_STATE_ROW_CHANGED; + } } + if ((param->testflag & T_AUTO_INC) || ((param->testflag & T_REP_ANY) && info->s->base.auto_key)) _ma_update_auto_increment_key(param, info, (my_bool) !test(param->testflag & T_AUTO_INC)); - if (!(param->testflag & T_DESCRIPT)) - { - if (info->update & HA_STATE_CHANGED && ! (param->testflag & T_READONLY)) - error|=maria_update_state_info(param, info, - UPDATE_OPEN_COUNT | - (((param->testflag & T_REP_ANY) ? - UPDATE_TIME : 0) | - (state_updated ? UPDATE_STAT : 0) | - ((param->testflag & T_SORT_RECORDS) ? - UPDATE_SORT : 0))); - info->update&= ~HA_STATE_CHANGED; - } + if (info->update & HA_STATE_CHANGED && ! (param->testflag & T_READONLY)) + error|=maria_update_state_info(param, info, + UPDATE_OPEN_COUNT | + (((param->testflag & T_REP_ANY) ? + UPDATE_TIME : 0) | + (state_updated ? UPDATE_STAT : 0) | + ((param->testflag & T_SORT_RECORDS) ? + UPDATE_SORT : 0))); + info->update&= ~HA_STATE_CHANGED; maria_lock_database(info, F_UNLCK); + end2: + end_pagecache(maria_pagecache, 1); if (maria_close(info)) { - _ma_check_print_error(param,"%d when closing MARIA-table '%s'",my_errno,filename); + _ma_check_print_error(param,"%d when closing MARIA-table '%s'", + my_errno,filename); DBUG_RETURN(1); } if (error == 0) @@ -1533,8 +1526,6 @@ static int maria_sort_records(HA_CHECK *param, if (share->state.key_root[sort_key] == HA_OFFSET_ERROR) DBUG_RETURN(0); /* Nothing to do */ - init_pagecache(maria_pagecache, param->use_buffers, - 0, 0, opt_pagecache_block_size); if (init_io_cache(&info->rec_cache,-1,(uint) param->write_buffer_length, WRITE_CACHE,share->pack.header_length,1, MYF(MY_WME | MY_WAIT_IF_FULL))) diff --git a/storage/maria/maria_def.h b/storage/maria/maria_def.h index 4f3024bc427..5e56b0edc5a 100644 --- a/storage/maria/maria_def.h +++ b/storage/maria/maria_def.h @@ -190,9 +190,9 @@ typedef struct st_maria_file_bitmap { uchar *map; ulonglong page; /* Page number for current bitmap */ - PAGECACHE_FILE page_cache; uint used_size; /* Size of bitmap head that is not 0 */ - my_bool changed; /* Set to 1 if page needs to be flushed */ + my_bool changed; /* 1 if page needs to be flushed */ + PAGECACHE_FILE file; /* datafile where bitmap is stored */ #ifdef THREAD pthread_mutex_t bitmap_lock; diff --git a/storage/maria/maria_pack.c b/storage/maria/maria_pack.c index 449b6bf7d59..f1b3903c944 100644 --- a/storage/maria/maria_pack.c +++ b/storage/maria/maria_pack.c @@ -505,13 +505,21 @@ static int compress(PACK_MRG_INFO *mrg,char *result_table) trees=fields=0; huff_trees=0; huff_counts=0; + maria_block_size= isam_file->s->block_size; /* Create temporary or join file */ - if (backup) VOID(fn_format(org_name,isam_file->filename,"",MARIA_NAME_DEXT,2)); else VOID(fn_format(org_name,isam_file->filename,"",MARIA_NAME_DEXT,2+4+16)); + + if (init_pagecache(maria_pagecache, MARIA_MIN_PAGE_CACHE_SIZE, 0, 0, + maria_block_size) == 0) + { + fprintf(stderr, "Can't initialize page cache\n"); + goto err; + } + if (!test_only && result_table) { /* Make a new indexfile based on first file in list */ @@ -681,7 +689,7 @@ static int compress(PACK_MRG_INFO *mrg,char *result_table) { error|=my_close(isam_file->dfile.file, MYF(MY_WME)); isam_file->dfile.file= -1; /* Tell maria_close file is closed */ - isam_file->s->bitmap.file= -1; + isam_file->s->bitmap.file.file= -1; } } @@ -751,6 +759,7 @@ static int compress(PACK_MRG_INFO *mrg,char *result_table) DBUG_RETURN(0); err: + end_pagecache(maria_pagecache, 1); free_counts_and_tree_and_queue(huff_trees,trees,huff_counts,fields); if (new_file >= 0) VOID(my_close(new_file,MYF(0))); -- cgit v1.2.1 From d4512b20ec1ea3bb1bcfd38af5c5355d4facfb12 Mon Sep 17 00:00:00 2001 From: unknown Date: Fri, 20 Apr 2007 15:16:43 +0300 Subject: Fixed some problems caused by and unconvered when we don't read the first bitmap on open storage/maria/ma_bitmap.c: Set 'first_bitmap_with_space' to point at first page, if it's was not set before. (This is needed as we are not anymore reading the first bitmap into memory on startup) Fixed some bugs with full bitmaps and changing bitmaps that was unconvered while finding and fixing the above problem. --- storage/maria/ma_bitmap.c | 40 ++++++++++++++++++++++++++-------------- 1 file changed, 26 insertions(+), 14 deletions(-) diff --git a/storage/maria/ma_bitmap.c b/storage/maria/ma_bitmap.c index ed5bd070981..e7ced9fc1e1 100644 --- a/storage/maria/ma_bitmap.c +++ b/storage/maria/ma_bitmap.c @@ -203,6 +203,11 @@ my_bool _ma_bitmap_init(MARIA_SHARE *share, File file) bitmap->page= ~(ulonglong) 0; bitmap->used_size= bitmap->total_size; bfill(bitmap->map, share->block_size, 255); + if (share->state.first_bitmap_with_space == ~(ulonglong) 0) + { + /* Start scanning for free space from start of file */ + share->state.first_bitmap_with_space = 0; + } return 0; } @@ -540,7 +545,6 @@ static my_bool _ma_change_bitmap_page(MARIA_HA *info, ulonglong page) { DBUG_ENTER("_ma_change_bitmap_page"); - DBUG_ASSERT(page % bitmap->pages_covered == 0); if (bitmap->changed) { @@ -1158,11 +1162,18 @@ static my_bool find_blob(MARIA_HA *info, ulong length) used= allocate_full_pages(bitmap, (pages >= 65535 ? 65535 : (uint) pages), block, 0); - if (!used && move_to_next_bitmap(info, bitmap)) - DBUG_RETURN(1); - info->bitmap_blocks.elements++; - block++; - } while ((pages-= used) != 0); + if (!used) + { + if (move_to_next_bitmap(info, bitmap)) + DBUG_RETURN(1); + } + else + { + pages-= used; + info->bitmap_blocks.elements++; + block++; + } + } while (pages != 0); } if (rest_length && find_tail(info, rest_length, info->bitmap_blocks.elements++)) @@ -1390,8 +1401,9 @@ my_bool _ma_bitmap_find_place(MARIA_HA *info, MARIA_ROW *row, blocks->count= 0; blocks->tail_page_skipped= blocks->page_skipped= 0; row->extents_count= 0; + /* - Reserver place for the following blocks: + Reserve place for the following blocks: - Head block - Full page block - Marker block to allow write_block_record() to split full page blocks @@ -1580,7 +1592,7 @@ static my_bool set_page_bits(MARIA_HA *info, MARIA_FILE_BITMAP *bitmap, uchar *data; DBUG_ENTER("set_page_bits"); - bitmap_page= page / bitmap->pages_covered; + bitmap_page= page - page % bitmap->pages_covered; if (bitmap_page != bitmap->page && _ma_change_bitmap_page(info, bitmap, bitmap_page)) DBUG_RETURN(1); @@ -1603,8 +1615,8 @@ static my_bool set_page_bits(MARIA_HA *info, MARIA_FILE_BITMAP *bitmap, bitmap->changed= 1; DBUG_EXECUTE("bitmap", _ma_print_bitmap(bitmap);); if (fill_pattern != 3 && fill_pattern != 7 && - page < info->s->state.first_bitmap_with_space) - info->s->state.first_bitmap_with_space= page; + bitmap_page < info->s->state.first_bitmap_with_space) + info->s->state.first_bitmap_with_space= bitmap_page; DBUG_RETURN(0); } @@ -1631,7 +1643,7 @@ static uint get_page_bits(MARIA_HA *info, MARIA_FILE_BITMAP *bitmap, uchar *data; DBUG_ENTER("get_page_bits"); - bitmap_page= page / bitmap->pages_covered; + bitmap_page= page - page % bitmap->pages_covered; if (bitmap_page != bitmap->page && _ma_change_bitmap_page(info, bitmap, bitmap_page)) DBUG_RETURN(~ (uint) 0); @@ -1679,7 +1691,7 @@ my_bool _ma_reset_full_page_bits(MARIA_HA *info, MARIA_FILE_BITMAP *bitmap, DBUG_PRINT("enter", ("page: %lu page_count: %u", (ulong) page, page_count)); safe_mutex_assert_owner(&info->s->bitmap.bitmap_lock); - bitmap_page= page / bitmap->pages_covered; + bitmap_page= page - page % bitmap->pages_covered; if (bitmap_page != bitmap->page && _ma_change_bitmap_page(info, bitmap, bitmap_page)) DBUG_RETURN(1); @@ -1719,8 +1731,8 @@ my_bool _ma_reset_full_page_bits(MARIA_HA *info, MARIA_FILE_BITMAP *bitmap, tmp= (1 << bit_count) - 1; *data&= ~tmp; } - if (bitmap->page < (ulonglong) info->s->state.first_bitmap_with_space) - info->s->state.first_bitmap_with_space= bitmap->page; + if (bitmap_page < info->s->state.first_bitmap_with_space) + info->s->state.first_bitmap_with_space= bitmap_page; bitmap->changed= 1; DBUG_EXECUTE("bitmap", _ma_print_bitmap(bitmap);); DBUG_RETURN(0); -- cgit v1.2.1 From 486fd0a22ae11ed6bc1f425e7ab7fcdb03679b20 Mon Sep 17 00:00:00 2001 From: unknown Date: Thu, 26 Apr 2007 00:19:56 +0300 Subject: fixed mistypings in comments and spaces --- storage/maria/ma_blockrec.c | 49 +++++++++++++++++++++---------------------- storage/maria/ma_loghandler.c | 4 ++-- 2 files changed, 26 insertions(+), 27 deletions(-) diff --git a/storage/maria/ma_blockrec.c b/storage/maria/ma_blockrec.c index 55b16a64916..bb1cddd4d77 100644 --- a/storage/maria/ma_blockrec.c +++ b/storage/maria/ma_blockrec.c @@ -16,7 +16,7 @@ /* Storage of records in block - Some clarifactions about the abbrev used: + Some clarifications about the abbrev used: NULL fields -> Fields that may have contain a NULL value. Not null fields -> Fields that may not contain a NULL value. @@ -57,21 +57,21 @@ NO 1 byte Number of row/tail entries on page empty space 2 bytes Empty space on page - The upmost bit in PAGE_TYPE is set to 1 if the data on the page + The most significant bit in PAGE_TYPE is set to 1 if the data on the page can be compacted to get more space. (PAGE_CAN_BE_COMPACTED) Row data Row directory of NO entries, that consist of the following for each row - (in reverse order; ie, first record is stored last): + (in reverse order; i.e., first record is stored last): Position 2 bytes Position of row on page Length 2 bytes Length of entry - For Position and Length, the 1 upmost bit of the position and the 1 - upmost bit of the length could be used for some states of the row (in - other words, we should try to keep these reserved) - + For Position and Length, the 1 most significant bit of the position and + the 1 most significant bit of the length could be used for some states of + the row (in other words, we should try to keep these reserved) + eof flag 1 byte Reserved for full page read testing. (Ie, did the previous write get the whole block on disk. @@ -167,7 +167,7 @@ efficiently. On update and delete we would add TRANSID (if it was an old committed row) and VER_PTR to the row. On row page compaction we can easily detect rows where - TRANSID was committed before the the longest running transaction + TRANSID was committed before the longest running transaction started and we can then delete TRANSID and VER_PTR from the row to gain more space. @@ -279,12 +279,12 @@ typedef struct st_maria_extent_cursor my_off_t page; /* How many pages in the page region */ uint page_count; - /* Total number of extents (ie, entries in the 'extent' slot) */ + /* Total number of extents (i.e., entries in the 'extent' slot) */ uint extent_count; /* <> 0 if current extent is a tail page; Set while using cursor */ - uint tail; + uint tail; /* - <> 1 if we are working on the first extent (ie, the one that is store in + <> 1 if we are working on the first extent (i.e., the one that is store in the row header, not an extent that is stored as part of the row data). */ my_bool first_extent; @@ -362,7 +362,7 @@ my_bool _ma_once_init_block_record(MARIA_SHARE *share, File data_file) { share->base.max_data_file_length= - (((ulonglong) 1 << ((share->base.rec_reflength-1)*8))-1) * + (((ulonglong) 1 << ((share->base.rec_reflength-1)*8))-1) * share->block_size; #if SIZEOF_OFF_T == 4 set_if_smaller(share->base.max_data_file_length, INT_MAX32); @@ -422,7 +422,7 @@ my_bool _ma_init_block_record(MARIA_HA *info) NullS, 0)) DBUG_RETURN(1); if (my_init_dynamic_array(&info->bitmap_blocks, - sizeof(MARIA_BITMAP_BLOCK), + sizeof(MARIA_BITMAP_BLOCK), ELEMENTS_RESERVED_FOR_MAIN_PART, 16)) my_free((char*) &info->bitmap_blocks, MYF(0)); row->base_length= new_row->base_length= info->s->base_length; @@ -460,7 +460,7 @@ void _ma_end_block_record(MARIA_HA *info) /* Return the next used byte on the page after a directory entry. - + SYNOPSIS start_of_next_entry() dir Directory entry to be used @@ -468,7 +468,7 @@ void _ma_end_block_record(MARIA_HA *info) RETURN # Position in page where next entry starts. Everything between the '*dir' and this are free to be used. -*/ +*/ static inline uint start_of_next_entry(byte *dir) { @@ -491,7 +491,7 @@ static inline uint start_of_next_entry(byte *dir) SYNOPSIS check_if_zero() pos Start of memory to check - length length of memory region + length length of memory region NOTES Used mainly to detect rows with wrong extent information @@ -715,7 +715,7 @@ static void calc_record_size(MARIA_HA *info, const byte *record, *null_field_lengths= length; if (!length) { - row->empty_bits[column->empty_pos]|= column->empty_bit; + row->empty_bits[column->empty_pos]|= column->empty_bit; break; } row->varchar_length+= length; @@ -809,7 +809,7 @@ static void compact_page(byte *buff, uint block_size, uint rownr, uint length= (next_free_pos - start_of_found_block); /* There was empty space before this and prev block - Check if we have to move prevous block up to page start + Check if we have to move previous block up to page start */ if (page_pos != start_of_found_block) { @@ -817,7 +817,7 @@ static void compact_page(byte *buff, uint block_size, uint rownr, memmove(buff + page_pos, buff + start_of_found_block, length); } page_pos+= length; - /* next continous block starts here */ + /* next continuous block starts here */ start_of_found_block= offset; diff= offset - page_pos; } @@ -858,7 +858,7 @@ static void compact_page(byte *buff, uint block_size, uint rownr, memmove(buff + page_pos - length, buff + next_free_pos, length); } page_pos-= length; - /* next continous block starts here */ + /* next continuous block starts here */ end_of_found_block= row_end; diff= page_pos - row_end; } @@ -952,7 +952,7 @@ static my_bool get_head_or_tail_page(MARIA_HA *info, res->data= (buff + PAGE_HEADER_SIZE); res->dir= res->data + res->length; res->offset= 0; - /* Store poistion to the first row */ + /* Store position to the first row */ int2store(res->dir, PAGE_HEADER_SIZE); DBUG_ASSERT(length <= res->length); } @@ -1487,7 +1487,7 @@ static my_bool write_block_record(MARIA_HA *info, const byte *record, for (end_block= block + block->sub_blocks; block < end_block; block++) { /* - Set only a bit, to not cause bitmap code to belive a block is full + Set only a bit, to not cause bitmap code to believe a block is full when there is still a lot of entries in it */ block->used|= BLOCKUSED_USED; @@ -1535,7 +1535,7 @@ static my_bool write_block_record(MARIA_HA *info, const byte *record, end_block= head_block + head_block->sub_blocks; /* Loop until we have find a block bigger than we need or - we find the the empty page block. + we find the empty page block. */ while (data_length >= (length= (cur_block->page_count * FULL_PAGE_SIZE(block_size))) && @@ -1585,7 +1585,6 @@ static my_bool write_block_record(MARIA_HA *info, const byte *record, TODO: If there is enough space on the following tail block, use this instead of creating a new tail block. - */ DBUG_ASSERT(cur_block[1].page_count == 0); if (cur_block->page_count == 1) @@ -2622,7 +2621,7 @@ int _ma_read_block_record2(MARIA_HA *info, byte *record, the row is very short in which case we allocated 'min_row_length' data for allowing the row to expand. */ - if (data != end_of_data && (uint) (end_of_data - start_of_data) > + if (data != end_of_data && (uint) (end_of_data - start_of_data) > info->s->base.min_row_length) goto err; } diff --git a/storage/maria/ma_loghandler.c b/storage/maria/ma_loghandler.c index a9f1503f2c9..198be85ab8c 100644 --- a/storage/maria/ma_loghandler.c +++ b/storage/maria/ma_loghandler.c @@ -424,7 +424,7 @@ static File open_logfile_by_number_no_cache(uint32 file_no) char path[FN_REFLEN]; DBUG_ENTER("open_logfile_by_number_no_cache"); - /* Todo: add O_DIRECT to open flags (when buffer is aligned) */ + /* TODO: add O_DIRECT to open flags (when buffer is aligned) */ if ((file= my_open(translog_filename_by_fileno(file_no, path), O_CREAT | O_BINARY | O_RDWR, MYF(MY_WME))) < 0) @@ -3180,7 +3180,7 @@ static byte *translog_get_LSN_from_diff(LSN base_lsn, byte *src, byte *dst) (ulong) LSN_OFFSET(base_lsn), (ulong) src, (ulong) dst)); first_byte= *((uint8*) src); - code= first_byte >> 6; /* Length in 2 upmost bits */ + code= first_byte >> 6; /* Length is in 2 most significant bits */ first_byte&= 0x3F; src++; /* Skip length + encode */ file_no= LSN_FILE_NO(base_lsn); /* Assume relative */ -- cgit v1.2.1 From 8f39541e7d8ba812d1198af5d4179ba44d6693fa Mon Sep 17 00:00:00 2001 From: unknown Date: Tue, 29 May 2007 20:13:56 +0300 Subject: This patch is a collection of patches from from Sanja, Sergei and Monty. Added logging and pinning of pages to block format. Integration of transaction manager, log handler. Better page cache intergration Split trnman.h into two files, so that we don't have to include my_atomic.h into C++ programs. Renaming of structures, more comments, more debugging etc. Fixed problem with small head block + long varchar. Added extra argument to delete_record() and update_record() (needed for UNDO logging) Small changes to interface of pagecache and log handler. Change initialization of log_record_type_descriptors to not be depending on enum order. Use array of LEX_STRING's to send data to log handler Added 'dummy' transaction option to MARIA_INFO so that we can always assume 'trn' exists. include/lf.h: Interface fixes Rename of structures (Patch from Sergei via Sanja) include/my_atomic.h: More comments include/my_global.h: Added MY_ERRPTR include/pagecache.h: Added undo LSN when unlocking pages mysql-test/r/maria.result: Updated results mysql-test/t/maria.test: Added autocommit around lock tables (Patch from Sanja) mysys/lf_alloc-pin.c: Post-review fixes, simple optimizations More comments Struct slot renames Check amount of memory on stack (Patch from Sergei) mysys/lf_dynarray.c: More comments mysys/lf_hash.c: More comments After review fixes (Patch from Sergei) storage/maria/ha_maria.cc: Split trnman.h into two files, so that we don't have to include my_atomic.h into the .cc program. (Temporary fix to avoid bug in gcc) Move out all deferencing of the transaction structure. Transaction manager integrated (Patch from Sergei) storage/maria/ha_maria.h: Added prototype for start_stmt() storage/maria/lockman.c: Function call rename storage/maria/ma_bitmap.c: Mark deleted pages free from page cache storage/maria/ma_blockrec.c: Offset -> rownr More debugging Fixed problem with small head block + long varchar Added logging of changed pages Added logging of undo (Including only loggging of changed fields in case of update) Added pinning/unpinning of all changed pages More comments Added free_full_pages() as the same code was used in several places. fill_rows_parts() renamed as fill_insert_undo_parts() offset -> rownr Added some optimization of not transactional tables _ma_update_block_record() has new parameter, as we need original row to do efficent undo for update storage/maria/ma_blockrec.h: Added ROW_EXTENTS_ON_STACK Changed prototype for update and delete of row storage/maria/ma_check.c: Added original row to delete_record() call storage/maria/ma_control_file.h: Added ifdefs for C++ storage/maria/ma_delete.c: Added original row to delete_record() call (Needed for efficent undo logging) storage/maria/ma_dynrec.c: Added extra argument to delete_record() and update_record() Removed not used variable storage/maria/ma_init.c: Initialize log handler storage/maria/ma_loghandler.c: Removed not used variable Change initialization of log_record_type_descriptors to not be depending on enum order Use array of LEX_STRING's to send data to log handler storage/maria/ma_loghandler.h: New defines Use array of LEX_STRING's to send data to log handler storage/maria/ma_open.c: Added 'dummy' transaction option to MARIA_INFO so that we can always assume 'trn' exists. Store in MARIA_SHARE->page_type if pages will have up to date LSN's storage/maria/ma_pagecache.c: Don't decrease number of readers when using pagecache_write()/pagecache_read() In pagecache_write() decrement request count if page was left pinned Added pagecache_delete_pages() Removed some casts Make trace output consistent with rest of code Simplify calling of DBUG_ASSERT(0) Only update LSN if the LSN is bigger than what's already on the page Added LSN parameter pagecache_unpin_page(), pagecache_unpin(), and pagecache_unlock() (Part of patch from Sanja) storage/maria/ma_static.c: Added 'dummy' transaction option to MARIA_INFO so that we can always assume 'trn' exists. Added default page cache storage/maria/ma_statrec.c: Added extra argument to delete_record() and update_record() storage/maria/ma_test1.c: Added option -T for transactions storage/maria/ma_test2.c: Added option -T for transactions storage/maria/ma_test_all.sh: Test with transactions storage/maria/ma_update.c: Changed prototype for update of row storage/maria/maria_def.h: Changed prototype for update & delete of row as block records need to access the old row Store in MARIA_SHARE->page_type if pages will have up to date LSN's Added MARIA_MAX_TREE_LEVELS to allow us to calculate the number of possible pinned pages we may need. Removed not used 'empty_bits_buffer' Added pointer to transaction object Added array for pinned pages Added log_row_parts array for logging of field data. Added MARIA_PINNED_PAGE to store pinned pages storage/maria/trnman.c: Added accessor functions to transaction object Added missing DBUG_RETURN() More debugging More comments Changed // comment of code to #ifdef NOT_USED Transaction manager integrated. Post review fixes Part of patch originally from Sergei storage/maria/trnman.h: Split trnman.h into two files, so that we don't have to include my_atomic.h into the .cc program. (Temporary fix to avoid bug in gcc) storage/maria/unittest/ma_pagecache_single.c: Added missing argument Added SKIP_BIG_TESTS (Patch from Sanja) storage/maria/unittest/ma_test_loghandler-t.c: Test logging with new LEX_STRING parameter (Patch from Sanja) storage/maria/unittest/ma_test_loghandler_multigroup-t.c: Test logging with new LEX_STRING parameter (Patch from Sanja) storage/maria/unittest/ma_test_loghandler_multithread-t.c: Test logging with new LEX_STRING parameter (Patch from Sanja) storage/maria/unittest/ma_test_loghandler_pagecache-t.c: Test logging with new LEX_STRING parameter (Patch from Sanja) storage/maria/unittest/trnman-t.c: Stack overflow detection (Patch from Sergei) unittest/unit.pl: Command-line options --big and --verbose (Patch from Sergei) unittest/mytap/tap.c: Detect --big (Patch from Sergei) unittest/mytap/tap.h: Skip_big_tests and SKIP_BIG_TESTS (Patch from Sergei) storage/maria/trnman_public.h: New BitKeeper file ``storage/maria/trnman_public.h'' --- include/lf.h | 58 +- include/my_atomic.h | 5 + include/my_global.h | 2 + include/pagecache.h | 18 +- mysql-test/r/maria.result | 8 + mysql-test/t/maria.test | 8 + mysys/lf_alloc-pin.c | 188 ++- mysys/lf_dynarray.c | 10 +- mysys/lf_hash.c | 167 ++- storage/maria/ha_maria.cc | 124 +- storage/maria/ha_maria.h | 1 + storage/maria/lockman.c | 2 +- storage/maria/ma_bitmap.c | 8 +- storage/maria/ma_blockrec.c | 1260 ++++++++++++++++++-- storage/maria/ma_blockrec.h | 7 +- storage/maria/ma_check.c | 3 +- storage/maria/ma_control_file.h | 7 + storage/maria/ma_delete.c | 2 +- storage/maria/ma_dynrec.c | 7 +- storage/maria/ma_init.c | 1 + storage/maria/ma_loghandler.c | 552 +++++---- storage/maria/ma_loghandler.h | 113 +- storage/maria/ma_open.c | 5 + storage/maria/ma_pagecache.c | 351 +++--- storage/maria/ma_static.c | 7 + storage/maria/ma_statrec.c | 4 +- storage/maria/ma_test1.c | 6 +- storage/maria/ma_test2.c | 8 +- storage/maria/ma_test_all.sh | 3 + storage/maria/ma_update.c | 2 +- storage/maria/maria_def.h | 47 +- storage/maria/trnman.c | 145 ++- storage/maria/trnman.h | 32 +- storage/maria/trnman_public.h | 49 + storage/maria/unittest/ma_pagecache_single.c | 15 +- storage/maria/unittest/ma_test_loghandler-t.c | 48 +- .../unittest/ma_test_loghandler_multigroup-t.c | 49 +- .../unittest/ma_test_loghandler_multithread-t.c | 18 +- .../unittest/ma_test_loghandler_pagecache-t.c | 6 +- storage/maria/unittest/trnman-t.c | 19 +- unittest/mytap/tap.c | 6 + unittest/mytap/tap.h | 43 +- unittest/unit.pl | 12 +- 43 files changed, 2614 insertions(+), 812 deletions(-) create mode 100644 storage/maria/trnman_public.h diff --git a/include/lf.h b/include/lf.h index 39a47e6568a..4712f9c4862 100644 --- a/include/lf.h +++ b/include/lf.h @@ -1,4 +1,4 @@ -/* Copyright (C) 2006 MySQL AB +/* Copyright (C) 2007 MySQL AB 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 @@ -96,26 +96,28 @@ nolock_wrap(lf_dynarray_iterate, int, #define LF_PINBOX_PINS 4 #define LF_PURGATORY_SIZE 10 -typedef void lf_pinbox_free_func(void *, void *); +typedef void lf_pinbox_free_func(void *, void *, void*); typedef struct { - LF_DYNARRAY pinstack; + LF_DYNARRAY pinarray; lf_pinbox_free_func *free_func; void *free_func_arg; uint free_ptr_offset; uint32 volatile pinstack_top_ver; /* this is a versioned pointer */ - uint32 volatile pins_in_stack; /* number of elements in array */ + uint32 volatile pins_in_array; /* number of elements in array */ } LF_PINBOX; typedef struct { void * volatile pin[LF_PINBOX_PINS]; LF_PINBOX *pinbox; + void *stack_ends_here; void *purgatory; uint32 purgatory_count; uint32 volatile link; /* we want sizeof(LF_PINS) to be 128 to avoid false sharing */ char pad[128-sizeof(uint32)*2 -sizeof(LF_PINBOX *) + -sizeof(void*) -sizeof(void *)*(LF_PINBOX_PINS+1)]; } LF_PINS; @@ -124,9 +126,9 @@ typedef struct { (e.g. lf_hash). */ #define lf_rwlock_by_pins(PINS) \ - my_atomic_rwlock_wrlock(&(PINS)->pinbox->pinstack.lock) + my_atomic_rwlock_wrlock(&(PINS)->pinbox->pinarray.lock) #define lf_rwunlock_by_pins(PINS) \ - my_atomic_rwlock_wrunlock(&(PINS)->pinbox->pinstack.lock) + my_atomic_rwlock_wrunlock(&(PINS)->pinbox->pinarray.lock) /* compile-time assert, to require "no less than N" pins @@ -135,9 +137,9 @@ typedef struct { */ #if defined(__GNUC__) && defined(MY_LF_EXTRA_DEBUG) #define LF_REQUIRE_PINS(N) \ - static const char require_pins[LF_PINBOX_PINS-N] __attribute__ ((unused)); \ - static const int LF_NUM_PINS_IN_THIS_FILE= N - + static const char require_pins[LF_PINBOX_PINS-N] \ + __attribute__ ((unused)); \ + static const int LF_NUM_PINS_IN_THIS_FILE= N; #define _lf_pin(PINS, PIN, ADDR) \ ( \ assert(PIN < LF_NUM_PINS_IN_THIS_FILE), \ @@ -164,17 +166,17 @@ void lf_pinbox_init(LF_PINBOX *pinbox, uint free_ptr_offset, void lf_pinbox_destroy(LF_PINBOX *pinbox); lock_wrap(lf_pinbox_get_pins, LF_PINS *, - (LF_PINBOX *pinbox), - (pinbox), - &pinbox->pinstack.lock) + (LF_PINBOX *pinbox, void *stack_end), + (pinbox, stack_end), + &pinbox->pinarray.lock); lock_wrap_void(lf_pinbox_put_pins, (LF_PINS *pins), (pins), - &pins->pinbox->pinstack.lock) + &pins->pinbox->pinarray.lock); lock_wrap_void(lf_pinbox_free, (LF_PINS *pins, void *addr), (pins, addr), - &pins->pinbox->pinstack.lock) + &pins->pinbox->pinarray.lock); /* memory allocator, lf_alloc-pin.c @@ -193,23 +195,23 @@ typedef struct st_lf_allocator { void lf_alloc_init(LF_ALLOCATOR *allocator, uint size, uint free_ptr_offset); void lf_alloc_destroy(LF_ALLOCATOR *allocator); -uint lf_alloc_in_pool(LF_ALLOCATOR *allocator); +uint lf_alloc_pool_count(LF_ALLOCATOR *allocator); /* shortcut macros to access underlying pinbox functions from an LF_ALLOCATOR see _lf_pinbox_get_pins() and _lf_pinbox_put_pins() */ -#define _lf_alloc_free(PINS, PTR) _lf_pinbox_free((PINS), (PTR)) -#define lf_alloc_free(PINS, PTR) lf_pinbox_free((PINS), (PTR)) -#define _lf_alloc_get_pins(ALLOC) _lf_pinbox_get_pins(&(ALLOC)->pinbox) -#define lf_alloc_get_pins(ALLOC) lf_pinbox_get_pins(&(ALLOC)->pinbox) -#define _lf_alloc_put_pins(PINS) _lf_pinbox_put_pins(PINS) -#define lf_alloc_put_pins(PINS) lf_pinbox_put_pins(PINS) -#define lf_alloc_real_free(ALLOC, ADDR) my_free((gptr)(ADDR), MYF(0)) +#define _lf_alloc_free(PINS, PTR) _lf_pinbox_free((PINS), (PTR)) +#define lf_alloc_free(PINS, PTR) lf_pinbox_free((PINS), (PTR)) +#define _lf_alloc_get_pins(A, ST) _lf_pinbox_get_pins(&(A)->pinbox, (ST)) +#define lf_alloc_get_pins(A, ST) lf_pinbox_get_pins(&(A)->pinbox, (ST)) +#define _lf_alloc_put_pins(PINS) _lf_pinbox_put_pins(PINS) +#define lf_alloc_put_pins(PINS) lf_pinbox_put_pins(PINS) +#define lf_alloc_direct_free(ALLOC, ADDR) my_free((gptr)(ADDR), MYF(0)) lock_wrap(lf_alloc_new, void *, (LF_PINS *pins), (pins), - &pins->pinbox->pinstack.lock) + &pins->pinbox->pinarray.lock); /* extendible hash, lf_hash.c @@ -240,11 +242,11 @@ int lf_hash_delete(LF_HASH *hash, LF_PINS *pins, const void *key, uint keylen); shortcut macros to access underlying pinbox functions from an LF_HASH see _lf_pinbox_get_pins() and _lf_pinbox_put_pins() */ -#define _lf_hash_get_pins(HASH) _lf_alloc_get_pins(&(HASH)->alloc) -#define lf_hash_get_pins(HASH) lf_alloc_get_pins(&(HASH)->alloc) -#define _lf_hash_put_pins(PINS) _lf_pinbox_put_pins(PINS) -#define lf_hash_put_pins(PINS) lf_pinbox_put_pins(PINS) - +#define _lf_hash_get_pins(HASH, ST) _lf_alloc_get_pins(&(HASH)->alloc, (ST)) +#define lf_hash_get_pins(HASH, ST) lf_alloc_get_pins(&(HASH)->alloc, (ST)) +#define _lf_hash_put_pins(PINS) _lf_pinbox_put_pins(PINS) +#define lf_hash_put_pins(PINS) lf_pinbox_put_pins(PINS) +#define lf_hash_search_unpin(PINS) lf_unpin((PINS), 2) /* cleanup */ diff --git a/include/my_atomic.h b/include/my_atomic.h index 5f328f32f3f..5fe7e521c19 100644 --- a/include/my_atomic.h +++ b/include/my_atomic.h @@ -225,6 +225,11 @@ make_atomic_fas(ptr) #undef make_atomic_fas_body #undef intptr +/* + the macro below defines (as an expression) the code that + will be run in spin-loops. Intel manuals recummend to have PAUSE there. + It is expected to be defined in include/atomic/ *.h files +*/ #ifndef LF_BACKOFF #define LF_BACKOFF (1) #endif diff --git a/include/my_global.h b/include/my_global.h index 68f47a75e4b..0b089d41611 100644 --- a/include/my_global.h +++ b/include/my_global.h @@ -1034,6 +1034,8 @@ typedef long intptr; #error #endif +#define MY_ERRPTR ((void*)(intptr)1) + #ifdef USE_RAID /* The following is done with a if to not get problems with pre-processors diff --git a/include/pagecache.h b/include/pagecache.h index b917271f0c9..7951e48edb0 100644 --- a/include/pagecache.h +++ b/include/pagecache.h @@ -206,17 +206,21 @@ extern void pagecache_unlock_page(PAGECACHE *pagecache, pgcache_page_no_t pageno, enum pagecache_page_lock lock, enum pagecache_page_pin pin, - LSN first_REDO_LSN_for_page); + LSN first_REDO_LSN_for_page, + LSN lsn); extern void pagecache_unlock(PAGECACHE *pagecache, PAGECACHE_PAGE_LINK *link, enum pagecache_page_lock lock, enum pagecache_page_pin pin, - LSN first_REDO_LSN_for_page); + LSN first_REDO_LSN_for_page, + LSN lsn); extern void pagecache_unpin_page(PAGECACHE *pagecache, PAGECACHE_FILE *file, - pgcache_page_no_t pageno); + pgcache_page_no_t pageno, + LSN lsn); extern void pagecache_unpin(PAGECACHE *pagecache, - PAGECACHE_PAGE_LINK *link); + PAGECACHE_PAGE_LINK *link, + LSN lsn); extern int flush_pagecache_blocks(PAGECACHE *keycache, PAGECACHE_FILE *file, enum flush_type type); @@ -225,6 +229,12 @@ extern my_bool pagecache_delete_page(PAGECACHE *pagecache, pgcache_page_no_t pageno, enum pagecache_page_lock lock, my_bool flush); +extern my_bool pagecache_delete_pages(PAGECACHE *pagecache, + PAGECACHE_FILE *file, + pgcache_page_no_t pageno, + uint page_count, + enum pagecache_page_lock lock, + my_bool flush); extern void end_pagecache(PAGECACHE *keycache, my_bool cleanup); extern my_bool pagecache_collect_changed_blocks_with_lsn(PAGECACHE *pagecache, LEX_STRING *str, diff --git a/mysql-test/r/maria.result b/mysql-test/r/maria.result index 515cb73eb2a..93d8863f40b 100644 --- a/mysql-test/r/maria.result +++ b/mysql-test/r/maria.result @@ -78,10 +78,15 @@ explain select a,b,c from t1; id select_type table type possible_keys key key_len ref rows Extra 1 SIMPLE t1 ALL NULL NULL NULL NULL 4 drop table t1; +set autocommit=0; +begin; CREATE TABLE t1 (a INT); INSERT INTO t1 VALUES (1), (2), (3); LOCK TABLES t1 WRITE; INSERT INTO t1 VALUES (1), (2), (3); +commit; +set autocommit=1; +UNLOCK TABLES; OPTIMIZE TABLE t1; Table Op Msg_type Msg_text test.t1 optimize status OK @@ -522,12 +527,15 @@ create table t1 (a int not null, primary key(a)) ROW_FORMAT=FIXED; create table t2 (a int not null, b int not null, primary key(a,b)) ROW_FORMAT=FIXED; insert into t1 values (1),(2),(3),(4),(5),(6); insert into t2 values (1,1),(2,1); +set autocommit=0; +begin; lock tables t1 read local, t2 read local; select straight_join * from t1,t2 force index (primary) where t1.a=t2.a; a a b 1 1 1 2 2 1 insert into t2 values(2,0); +commit; select straight_join * from t1,t2 force index (primary) where t1.a=t2.a; a a b 1 1 1 diff --git a/mysql-test/t/maria.test b/mysql-test/t/maria.test index bbf96d401f0..e9a0e146475 100644 --- a/mysql-test/t/maria.test +++ b/mysql-test/t/maria.test @@ -90,10 +90,15 @@ drop table t1; # # Test of OPTIMIZE of locked and modified tables # +set autocommit=0; +begin; CREATE TABLE t1 (a INT); INSERT INTO t1 VALUES (1), (2), (3); LOCK TABLES t1 WRITE; INSERT INTO t1 VALUES (1), (2), (3); +commit; +set autocommit=1; +UNLOCK TABLES; OPTIMIZE TABLE t1; DROP TABLE t1; @@ -510,10 +515,13 @@ create table t1 (a int not null, primary key(a)) ROW_FORMAT=FIXED; create table t2 (a int not null, b int not null, primary key(a,b)) ROW_FORMAT=FIXED; insert into t1 values (1),(2),(3),(4),(5),(6); insert into t2 values (1,1),(2,1); +set autocommit=0; +begin; lock tables t1 read local, t2 read local; select straight_join * from t1,t2 force index (primary) where t1.a=t2.a; connect (root,localhost,root,,test,$MASTER_MYPORT,$MASTER_MYSOCK); insert into t2 values(2,0); +commit; disconnect root; connection default; select straight_join * from t1,t2 force index (primary) where t1.a=t2.a; diff --git a/mysys/lf_alloc-pin.c b/mysys/lf_alloc-pin.c index e964553a64c..51c4df7c94a 100644 --- a/mysys/lf_alloc-pin.c +++ b/mysys/lf_alloc-pin.c @@ -1,5 +1,5 @@ /* QQ: TODO multi-pinbox */ -/* Copyright (C) 2000 MySQL AB +/* Copyright (C) 2006 MySQL AB 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 @@ -32,10 +32,11 @@ Pins are used to solve ABA problem. To use pins one must obey a pinning protocol: + 1. Let's assume that PTR is a shared pointer to an object. Shared means that any thread may modify it anytime to point to a different object and free the old object. Later the freed object may be potentially - allocated by another thread. If we're unlucky that another thread may + allocated by another thread. If we're unlucky that other thread may set PTR to point to this object again. This is ABA problem. 2. Create a local pointer LOCAL_PTR. 3. Pin the PTR in a loop: @@ -70,12 +71,34 @@ pins you have is limited (and small), keeping an object pinned prevents its reuse and cause unnecessary mallocs. + Explanations: + + 3. The loop is important. The following can occur: + thread1> LOCAL_PTR= PTR + thread2> free(PTR); PTR=0; + thread1> pin(PTR, PIN_NUMBER); + now thread1 cannot access LOCAL_PTR, even if it's pinned, + because it points to a freed memory. That is, it *must* + verify that it has indeed pinned PTR, the shared pointer. + + 6. When a thread wants to free some LOCAL_PTR, and it scans + all lists of pins to see whether it's pinned, it does it + upwards, from low pin numbers to high. Thus another thread + must copy an address from one pin to another in the same + direction - upwards, otherwise the scanning thread may + miss it. + Implementation details: + Pins are given away from a "pinbox". Pinbox is stack-based allocator. It used dynarray for storing pins, new elements are allocated by dynarray as necessary, old are pushed in the stack for reuse. ABA is solved by - versioning a pointer - because we use an array, a pointer to pins is 32 bit, - upper 32 bits are used for a version. + versioning a pointer - because we use an array, a pointer to pins is 16 bit, + upper 16 bits are used for a version. + + It is assumed that pins belong to a thread and are not transferable + between threads (LF_PINS::stack_ends_here being a primary reason + for this limitation). */ #include @@ -93,11 +116,11 @@ static void _lf_pinbox_real_free(LF_PINS *pins); void lf_pinbox_init(LF_PINBOX *pinbox, uint free_ptr_offset, lf_pinbox_free_func *free_func, void *free_func_arg) { - DBUG_ASSERT(sizeof(LF_PINS) == 128); DBUG_ASSERT(free_ptr_offset % sizeof(void *) == 0); - lf_dynarray_init(&pinbox->pinstack, sizeof(LF_PINS)); + compile_time_assert(sizeof(LF_PINS) == 128); + lf_dynarray_init(&pinbox->pinarray, sizeof(LF_PINS)); pinbox->pinstack_top_ver= 0; - pinbox->pins_in_stack= 0; + pinbox->pins_in_array= 0; pinbox->free_ptr_offset= free_ptr_offset; pinbox->free_func= free_func; pinbox->free_func_arg= free_func_arg; @@ -105,38 +128,72 @@ void lf_pinbox_init(LF_PINBOX *pinbox, uint free_ptr_offset, void lf_pinbox_destroy(LF_PINBOX *pinbox) { - lf_dynarray_destroy(&pinbox->pinstack); + lf_dynarray_destroy(&pinbox->pinarray); } /* Get pins from a pinbox. Usually called via lf_alloc_get_pins() or lf_hash_get_pins(). + SYNOPSYS + pinbox - + stack_end - a pointer to the end (top/bottom, depending on the + STACK_DIRECTION) of stack. Used for safe alloca. There's + no safety margin deducted, a caller should take care of it, + if necessary. + DESCRIPTION get a new LF_PINS structure from a stack of unused pins, or allocate a new one out of dynarray. + + NOTE + It is assumed that pins belong to a thread and are not transferable + between threads. */ -LF_PINS *_lf_pinbox_get_pins(LF_PINBOX *pinbox) +LF_PINS *_lf_pinbox_get_pins(LF_PINBOX *pinbox, void *stack_end) { uint32 pins, next, top_ver; LF_PINS *el; - + /* + We have an array of max. 64k elements. + The highest index currently allocated is pinbox->pins_in_array. + Freed elements are in a lifo stack, pinstack_top_ver. + pinstack_top_ver is 32 bits; 16 low bits are the index in the + array, to the first element of the list. 16 high bits are a version + (every time the 16 low bits are updated, the 16 high bits are + incremented). Versioniong prevents the ABA problem. + */ top_ver= pinbox->pinstack_top_ver; do { if (!(pins= top_ver % LF_PINBOX_MAX_PINS)) { - pins= my_atomic_add32(&pinbox->pins_in_stack, 1)+1; - el= (LF_PINS *)_lf_dynarray_lvalue(&pinbox->pinstack, pins); + /* the stack of free elements is empty */ + pins= my_atomic_add32(&pinbox->pins_in_array, 1)+1; + if (unlikely(pins >= LF_PINBOX_MAX_PINS)) + return 0; + /* + note that the first allocated element has index 1 (pins==1). + index 0 is reserved to mean "NULL pointer" + */ + el= (LF_PINS *)_lf_dynarray_lvalue(&pinbox->pinarray, pins); + if (unlikely(!el)) + return 0; break; } - el= (LF_PINS *)_lf_dynarray_value(&pinbox->pinstack, pins); + el= (LF_PINS *)_lf_dynarray_value(&pinbox->pinarray, pins); next= el->link; } while (!my_atomic_cas32(&pinbox->pinstack_top_ver, &top_ver, top_ver-pins+next+LF_PINBOX_MAX_PINS)); + /* + set el->link to the index of el in the dynarray (el->link has two usages: + - if element is allocated, it's its own index + - if element is free, it's its next element in the free stack + */ el->link= pins; el->purgatory_count= 0; el->pinbox= pinbox; + el->stack_ends_here= stack_end; return el; } @@ -171,25 +228,17 @@ void _lf_pinbox_put_pins(LF_PINS *pins) _lf_pinbox_real_free(pins); if (pins->purgatory_count) { - my_atomic_rwlock_wrunlock(&pins->pinbox->pinstack.lock); + my_atomic_rwlock_wrunlock(&pins->pinbox->pinarray.lock); pthread_yield(); - my_atomic_rwlock_wrlock(&pins->pinbox->pinstack.lock); + my_atomic_rwlock_wrlock(&pins->pinbox->pinarray.lock); } } top_ver= pinbox->pinstack_top_ver; - if (nr == pinbox->pins_in_stack) - { - int32 tmp= nr; - if (my_atomic_cas32(&pinbox->pins_in_stack, &tmp, tmp-1)) - goto ret; - } - do { pins->link= top_ver % LF_PINBOX_MAX_PINS; } while (!my_atomic_cas32(&pinbox->pinstack_top_ver, &top_ver, top_ver-pins->link+nr+LF_PINBOX_MAX_PINS)); -ret: return; } @@ -228,7 +277,7 @@ struct st_harvester { /* callback for _lf_dynarray_iterate: - scan all pins or all threads and accumulate all pins + scan all pins of all threads and accumulate all pins */ static int harvest_pins(LF_PINS *el, struct st_harvester *hv) { @@ -243,13 +292,19 @@ static int harvest_pins(LF_PINS *el, struct st_harvester *hv) *hv->granary++= p; } } + /* + hv->npins may become negative below, but it means that + we're on the last dynarray page and harvest_pins() won't be + called again. We don't bother to make hv->npins() correct + (that is 0) in this case. + */ hv->npins-= LF_DYNARRAY_LEVEL_LENGTH; return 0; } /* callback for _lf_dynarray_iterate: - scan all pins or all threads and see if addr is present there + scan all pins of all threads and see if addr is present there */ static int match_pins(LF_PINS *el, void *addr) { @@ -262,28 +317,35 @@ static int match_pins(LF_PINS *el, void *addr) return 0; } +#if STACK_DIRECTION < 0 +#define available_stack_size(END,CUR) (long) ((char*)(CUR) - (char*)(END)) +#else +#define available_stack_size(END,CUR) (long) ((char*)(END) - (char*)(CUR)) +#endif + /* - Scan the purgatory as free everything that can be freed + Scan the purgatory and free everything that can be freed */ static void _lf_pinbox_real_free(LF_PINS *pins) { - int npins; - void *list; - void **addr; + int npins, alloca_size; + void *list, **addr; + struct st_lf_alloc_node *first, *last= NULL; LF_PINBOX *pinbox= pins->pinbox; - npins= pinbox->pins_in_stack+1; + npins= pinbox->pins_in_array+1; #ifdef HAVE_ALLOCA + alloca_size= sizeof(void *)*LF_PINBOX_PINS*npins; /* create a sorted list of pinned addresses, to speed up searches */ - if (sizeof(void *)*LF_PINBOX_PINS*npins < my_thread_stack_size) + if (available_stack_size(&pinbox, pins->stack_ends_here) > alloca_size) { struct st_harvester hv; - addr= (void **) alloca(sizeof(void *)*LF_PINBOX_PINS*npins); + addr= (void **) alloca(alloca_size); hv.granary= addr; hv.npins= npins; /* scan the dynarray and accumulate all pinned addresses */ - _lf_dynarray_iterate(&pinbox->pinstack, + _lf_dynarray_iterate(&pinbox->pinarray, (lf_dynarray_func)harvest_pins, &hv); npins= hv.granary-addr; @@ -307,7 +369,7 @@ static void _lf_pinbox_real_free(LF_PINS *pins) if (addr) /* use binary search */ { void **a, **b, **c; - for (a= addr, b= addr+npins-1, c= a+(b-a)/2; b-a>1; c= a+(b-a)/2) + for (a= addr, b= addr+npins-1, c= a+(b-a)/2; (b-a) > 1; c= a+(b-a)/2) if (cur == *c) a= b= c; else if (cur > *c) @@ -319,41 +381,52 @@ static void _lf_pinbox_real_free(LF_PINS *pins) } else /* no alloca - no cookie. linear search here */ { - if (_lf_dynarray_iterate(&pinbox->pinstack, + if (_lf_dynarray_iterate(&pinbox->pinarray, (lf_dynarray_func)match_pins, cur)) goto found; } } /* not pinned - freeing */ - pinbox->free_func(cur, pinbox->free_func_arg); + if (last) + last= last->next= (struct st_lf_alloc_node *)cur; + else + first= last= (struct st_lf_alloc_node *)cur; continue; found: /* pinned - keeping */ add_to_purgatory(pins, cur); } + if (last) + pinbox->free_func(first, last, pinbox->free_func_arg); } +/* lock-free memory allocator for fixed-size objects */ + +LF_REQUIRE_PINS(1); + /* - callback for _lf_pinbox_real_free to free an unpinned object - + callback for _lf_pinbox_real_free to free a list of unpinned objects - add it back to the allocator stack + + DESCRIPTION + 'first' and 'last' are the ends of the linked list of st_lf_alloc_node's: + first->el->el->....->el->last. Use first==last to free only one element. */ -static void alloc_free(struct st_lf_alloc_node *node, LF_ALLOCATOR *allocator) +static void alloc_free(struct st_lf_alloc_node *first, + struct st_lf_alloc_node *last, + LF_ALLOCATOR *allocator) { struct st_lf_alloc_node *tmp; tmp= allocator->top; do { - node->next= tmp; - } while (!my_atomic_casptr((void **)&allocator->top, (void **)&tmp, node) && + last->next= tmp; + } while (!my_atomic_casptr((void **)&allocator->top, (void **)&tmp, first) && LF_BACKOFF); } -/* lock-free memory allocator for fixed-size objects */ - -LF_REQUIRE_PINS(1); - /* - initialize lock-free allocatod. + initialize lock-free allocator SYNOPSYS allocator - @@ -362,6 +435,8 @@ LF_REQUIRE_PINS(1); memory that is guaranteed to be unused after the object is put in the purgatory. Unused by ANY thread, not only the purgatory owner. + This memory will be used to link waiting-to-be-freed + objects in a purgatory list. */ void lf_alloc_init(LF_ALLOCATOR *allocator, uint size, uint free_ptr_offset) { @@ -370,12 +445,19 @@ void lf_alloc_init(LF_ALLOCATOR *allocator, uint size, uint free_ptr_offset) allocator->top= 0; allocator->mallocs= 0; allocator->element_size= size; - DBUG_ASSERT(size >= (int)sizeof(void *)); - DBUG_ASSERT(free_ptr_offset < size); + DBUG_ASSERT(size >= sizeof(void*) + free_ptr_offset); } /* destroy the allocator, free everything that's in it + + NOTE + As every other init/destroy function here and elsewhere it + is not thread safe. No, this function is no different, ensure + that no thread needs the allocator before destroying it. + We are not responsible for any damage that may be caused by + accessing the allocator when it is being or has been destroyed. + Oh yes, and don't put your cat in a microwave. */ void lf_alloc_destroy(LF_ALLOCATOR *allocator) { @@ -410,16 +492,14 @@ void *_lf_alloc_new(LF_PINS *pins) } while (node != allocator->top && LF_BACKOFF); if (!node) { - if (!(node= (void *)my_malloc(allocator->element_size, - MYF(MY_WME|MY_ZEROFILL)))) - break; + node= (void *)my_malloc(allocator->element_size, MYF(MY_WME)); #ifdef MY_LF_EXTRA_DEBUG - my_atomic_add32(&allocator->mallocs, 1); + if (likely(node)) + my_atomic_add32(&allocator->mallocs, 1); #endif break; } - if (my_atomic_casptr((void **)&allocator->top, - (void *)&node, *(void **)node)) + if (my_atomic_casptr((void **)&allocator->top, (void *)&node, node->next)) break; } _lf_unpin(pins, 0); @@ -432,7 +512,7 @@ void *_lf_alloc_new(LF_PINS *pins) NOTE This is NOT thread-safe !!! */ -uint lf_alloc_in_pool(LF_ALLOCATOR *allocator) +uint lf_alloc_pool_count(LF_ALLOCATOR *allocator) { uint i; struct st_lf_alloc_node *node; diff --git a/mysys/lf_dynarray.c b/mysys/lf_dynarray.c index c6dd654bf03..770b1f9342b 100644 --- a/mysys/lf_dynarray.c +++ b/mysys/lf_dynarray.c @@ -1,4 +1,4 @@ -/* Copyright (C) 2000 MySQL AB +/* Copyright (C) 2006 MySQL AB 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 @@ -21,8 +21,6 @@ Memory is allocated in non-contiguous chunks. This data structure is not space efficient for sparse arrays. - The number of elements is limited to 4311810304 - Every element is aligned to sizeof(element) boundary (to avoid false sharing if element is big enough). @@ -32,6 +30,9 @@ to arrays of elements, on the second level it's an array of pointers to arrays of pointers to arrays of elements. And so on. + With four levels the number of elements is limited to 4311810304 + (but as in all functions index is uint, the real limit is 2^32-1) + Actually, it's wait-free, not lock-free ;-) */ @@ -192,6 +193,9 @@ static int recursive_iterate(LF_DYNARRAY *array, void *ptr, int level, each. _lf_dynarray_iterate() calls user-supplied function on every array from the set. It is the fastest way to scan the array, faster than for (i=0; i < N; i++) { func(_lf_dynarray_value(dynarray, i)); } + + NOTE + if func() returns non-zero, the scan is aborted */ int _lf_dynarray_iterate(LF_DYNARRAY *array, lf_dynarray_func func, void *arg) { diff --git a/mysys/lf_hash.c b/mysys/lf_hash.c index fb2fb88492f..832f0eb5852 100644 --- a/mysys/lf_hash.c +++ b/mysys/lf_hash.c @@ -1,4 +1,4 @@ -/* Copyright (C) 2000 MySQL AB +/* Copyright (C) 2006 MySQL AB 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 @@ -36,6 +36,10 @@ typedef struct { uint32 hashnr; /* reversed hash number, for sorting */ const byte *key; uint keylen; + /* + data is stored here, directly after the keylen. + thus the pointer to data is (void*)(slist_element_ptr+1) + */ } LF_SLIST; /* @@ -77,20 +81,20 @@ static int lfind(LF_SLIST * volatile *head, CHARSET_INFO *cs, uint32 hashnr, retry: cursor->prev= (intptr *)head; - do { - cursor->curr= PTR(*cursor->prev); + do { /* PTR() isn't necessary below, head is a dummy node */ + cursor->curr= (LF_SLIST *)(*cursor->prev); _lf_pin(pins, 1, cursor->curr); - } while(*cursor->prev != (intptr)cursor->curr && LF_BACKOFF); + } while (*cursor->prev != (intptr)cursor->curr && LF_BACKOFF); for (;;) { - if (!cursor->curr) - return 0; + if (unlikely(!cursor->curr)) + return 0; /* end of the list */ do { /* QQ: XXX or goto retry ? */ link= cursor->curr->link; cursor->next= PTR(link); _lf_pin(pins, 0, cursor->next); - } while(link != cursor->curr->link && LF_BACKOFF); + } while (link != cursor->curr->link && LF_BACKOFF); cur_hashnr= cursor->curr->hashnr; cur_key= cursor->curr->key; cur_keylen= cursor->curr->keylen; @@ -114,6 +118,10 @@ retry: } else { + /* + we found a deleted node - be nice, help the other thread + and remove this deleted node + */ if (my_atomic_casptr((void **)cursor->prev, (void **)&cursor->curr, cursor->next)) _lf_alloc_free(pins, cursor->curr); @@ -139,31 +147,44 @@ retry: NOTE it uses pins[0..2], on return all pins are removed. + if there're nodes with the same key value, a new node is added before them. */ static LF_SLIST *linsert(LF_SLIST * volatile *head, CHARSET_INFO *cs, LF_SLIST *node, LF_PINS *pins, uint flags) { CURSOR cursor; - int res= -1; + int res; - do + for (;;) { if (lfind(head, cs, node->hashnr, node->key, node->keylen, &cursor, pins) && (flags & LF_HASH_UNIQUE)) + { res= 0; /* duplicate found */ + break; + } else { node->link= (intptr)cursor.curr; - assert(node->link != (intptr)node); - assert(cursor.prev != &node->link); + DBUG_ASSERT(node->link != (intptr)node); /* no circular references */ + DBUG_ASSERT(cursor.prev != &node->link); /* no circular references */ if (my_atomic_casptr((void **)cursor.prev, (void **)&cursor.curr, node)) + { res= 1; /* inserted ok */ + break; + } } - } while (res == -1); + } _lf_unpin(pins, 0); _lf_unpin(pins, 1); _lf_unpin(pins, 2); + /* + Note that cursor.curr is not pinned here and the pointer is unreliable, + the object may dissapear anytime. But if it points to a dummy node, the + pointer is safe, because dummy nodes are never freed - initialize_bucket() + uses this fact. + */ return res ? 0 : cursor.curr; } @@ -183,24 +204,41 @@ static int ldelete(LF_SLIST * volatile *head, CHARSET_INFO *cs, uint32 hashnr, const byte *key, uint keylen, LF_PINS *pins) { CURSOR cursor; - int res= -1; + int res; - do + for (;;) { if (!lfind(head, cs, hashnr, key, keylen, &cursor, pins)) - res= 1; + { + res= 1; /* not found */ + break; + } else + { + /* mark the node deleted */ if (my_atomic_casptr((void **)&(cursor.curr->link), - (void **)&cursor.next, 1+(char *)cursor.next)) + (void **)&cursor.next, + (void *)(((intptr)cursor.next) | 1))) { + /* and remove it from the list */ if (my_atomic_casptr((void **)cursor.prev, (void **)&cursor.curr, cursor.next)) _lf_alloc_free(pins, cursor.curr); else + { + /* + somebody already "helped" us and removed the node ? + Let's check if we need to help that someone too! + (to ensure the number of "set DELETED flag" actions + is equal to the number of "remove from the list" actions) + */ lfind(head, cs, hashnr, key, keylen, &cursor, pins); + } res= 0; + break; } - } while (res == -1); + } + } _lf_unpin(pins, 0); _lf_unpin(pins, 1); _lf_unpin(pins, 2); @@ -226,7 +264,8 @@ static LF_SLIST *lsearch(LF_SLIST * volatile *head, CHARSET_INFO *cs, { CURSOR cursor; int res= lfind(head, cs, hashnr, key, keylen, &cursor, pins); - if (res) _lf_pin(pins, 2, cursor.curr); + if (res) + _lf_pin(pins, 2, cursor.curr); _lf_unpin(pins, 0); _lf_unpin(pins, 1); return res ? cursor.curr : 0; @@ -241,6 +280,11 @@ static inline const byte* hash_key(const LF_HASH *hash, return record + hash->key_offset; } +/* + compute the hash key value from the raw key. + note, that the hash value is limited to 2^31, because we need one + bit to distinguish between normal and dummy nodes. +*/ static inline uint calc_hash(LF_HASH *hash, const byte *key, uint keylen) { ulong nr1= 1, nr2= 4; @@ -249,8 +293,9 @@ static inline uint calc_hash(LF_HASH *hash, const byte *key, uint keylen) return nr1 & INT_MAX32; } -#define MAX_LOAD 1.0 -static void initialize_bucket(LF_HASH *, LF_SLIST * volatile*, uint, LF_PINS *); +#define MAX_LOAD 1.0 /* average number of elements in a bucket */ + +static int initialize_bucket(LF_HASH *, LF_SLIST * volatile*, uint, LF_PINS *); /* Initializes lf_hash, the arguments are compatible with hash_init @@ -261,7 +306,7 @@ void lf_hash_init(LF_HASH *hash, uint element_size, uint flags, { lf_alloc_init(&hash->alloc, sizeof(LF_SLIST)+element_size, offsetof(LF_SLIST, key)); - lf_dynarray_init(&hash->array, sizeof(LF_SLIST **)); + lf_dynarray_init(&hash->array, sizeof(LF_SLIST *)); hash->size= 1; hash->count= 0; hash->element_size= element_size; @@ -275,14 +320,19 @@ void lf_hash_init(LF_HASH *hash, uint element_size, uint flags, void lf_hash_destroy(LF_HASH *hash) { - LF_SLIST *el= *(LF_SLIST **)_lf_dynarray_lvalue(&hash->array, 0); + LF_SLIST *el, **head= (LF_SLIST **)_lf_dynarray_value(&hash->array, 0); + + if (unlikely(!head)) + return; + el= *head; + while (el) { intptr next= el->link; if (el->hashnr & 1) - lf_alloc_real_free(&hash->alloc, el); + lf_alloc_direct_free(&hash->alloc, el); /* normal node */ else - my_free((void *)el, MYF(0)); + my_free((void *)el, MYF(0)); /* dummy node */ el= (LF_SLIST *)next; } lf_alloc_destroy(&hash->alloc); @@ -297,6 +347,7 @@ void lf_hash_destroy(LF_HASH *hash) RETURN 0 - inserted 1 - didn't (unique key conflict) + -1 - out of memory NOTE see linsert() for pin usage notes @@ -308,14 +359,18 @@ int lf_hash_insert(LF_HASH *hash, LF_PINS *pins, const void *data) lf_rwlock_by_pins(pins); node= (LF_SLIST *)_lf_alloc_new(pins); + if (unlikely(!node)) + return -1; memcpy(node+1, data, hash->element_size); node->key= hash_key(hash, (byte *)(node+1), &node->keylen); hashnr= calc_hash(hash, node->key, node->keylen); bucket= hashnr % hash->size; el= _lf_dynarray_lvalue(&hash->array, bucket); - if (*el == NULL) - initialize_bucket(hash, el, bucket, pins); - node->hashnr= my_reverse_bits(hashnr) | 1; + if (unlikely(!el)) + return -1; + if (*el == NULL && unlikely(initialize_bucket(hash, el, bucket, pins))) + return -1; + node->hashnr= my_reverse_bits(hashnr) | 1; /* normal node */ if (linsert(el, hash->charset, node, pins, hash->flags)) { _lf_alloc_free(pins, node); @@ -330,9 +385,14 @@ int lf_hash_insert(LF_HASH *hash, LF_PINS *pins, const void *data) } /* + DESCRIPTION + deletes an element with the given key from the hash (if a hash is + not unique and there're many elements with this key - the "first" + matching element is deleted) RETURN 0 - deleted 1 - didn't (not found) + -1 - out of memory NOTE see ldelete() for pin usage notes */ @@ -344,8 +404,16 @@ int lf_hash_delete(LF_HASH *hash, LF_PINS *pins, const void *key, uint keylen) bucket= hashnr % hash->size; lf_rwlock_by_pins(pins); el= _lf_dynarray_lvalue(&hash->array, bucket); - if (*el == NULL) - initialize_bucket(hash, el, bucket, pins); + if (unlikely(!el)) + return -1; + /* + note that we still need to initialize_bucket here, + we cannot return "node not found", because an old bucket of that + node may've been split and the node was assigned to a new bucket + that was never accessed before and thus is not initialized. + */ + if (*el == NULL && unlikely(initialize_bucket(hash, el, bucket, pins))) + return -1; if (ldelete(el, hash->charset, my_reverse_bits(hashnr) | 1, (byte *)key, keylen, pins)) { @@ -358,6 +426,12 @@ int lf_hash_delete(LF_HASH *hash, LF_PINS *pins, const void *key, uint keylen) } /* + RETURN + a pointer to an element with the given key (if a hash is not unique and + there're many elements with this key - the "first" matching element) + NULL if nothing is found + MY_ERRPTR if OOM + NOTE see lsearch() for pin usage notes */ @@ -369,32 +443,51 @@ void *lf_hash_search(LF_HASH *hash, LF_PINS *pins, const void *key, uint keylen) bucket= hashnr % hash->size; lf_rwlock_by_pins(pins); el= _lf_dynarray_lvalue(&hash->array, bucket); - if (*el == NULL) - initialize_bucket(hash, el, bucket, pins); + if (unlikely(!el)) + return MY_ERRPTR; + if (*el == NULL && unlikely(initialize_bucket(hash, el, bucket, pins))) + return MY_ERRPTR; found= lsearch(el, hash->charset, my_reverse_bits(hashnr) | 1, (byte *)key, keylen, pins); lf_rwunlock_by_pins(pins); return found ? found+1 : 0; } -static const char *dummy_key= ""; +static const byte *dummy_key= ""; -static void initialize_bucket(LF_HASH *hash, LF_SLIST * volatile *node, +/* + RETURN + 0 - ok + -1 - out of memory +*/ +static int initialize_bucket(LF_HASH *hash, LF_SLIST * volatile *node, uint bucket, LF_PINS *pins) { uint parent= my_clear_highest_bit(bucket); LF_SLIST *dummy= (LF_SLIST *)my_malloc(sizeof(LF_SLIST), MYF(MY_WME)); LF_SLIST **tmp= 0, *cur; LF_SLIST * volatile *el= _lf_dynarray_lvalue(&hash->array, parent); - if (*el == NULL && bucket) - initialize_bucket(hash, el, parent, pins); - dummy->hashnr= my_reverse_bits(bucket); + if (unlikely(!el || !dummy)) + return -1; + if (*el == NULL && bucket && + unlikely(initialize_bucket(hash, el, parent, pins))) + return -1; + dummy->hashnr= my_reverse_bits(bucket) | 0; /* dummy node */ dummy->key= (char*) dummy_key; dummy->keylen= 0; - if ((cur= linsert(el, hash->charset, dummy, pins, 0))) + if ((cur= linsert(el, hash->charset, dummy, pins, LF_HASH_UNIQUE))) { my_free((void *)dummy, MYF(0)); dummy= cur; } my_atomic_casptr((void **)node, (void **)&tmp, dummy); + /* + note that if the CAS above failed (after linsert() succeeded), + it would mean that some other thread has executed linsert() for + the same dummy node, its linsert() failed, it picked up our + dummy node (in "dummy= cur") and executed the same CAS as above. + Which means that even if CAS above failed we don't need to retry, + and we should not free(dummy) - there's no memory leak here + */ + return 0; } diff --git a/storage/maria/ha_maria.cc b/storage/maria/ha_maria.cc index a714528a643..ecb966a4fbd 100644 --- a/storage/maria/ha_maria.cc +++ b/storage/maria/ha_maria.cc @@ -25,12 +25,14 @@ #include #include #include "ha_maria.h" +#include "trnman_public.h" #include "maria_def.h" #include "ma_rt_index.h" #include "ma_blockrec.h" ulong maria_recover_options= HA_RECOVER_NONE; +static handlerton *maria_hton; /* bits in maria_recover_options */ const char *maria_recover_names[]= @@ -464,7 +466,7 @@ ha_maria::ha_maria(handlerton *hton, TABLE_SHARE *table_arg): handler(hton, table_arg), file(0), int_table_flags(HA_NULL_IN_KEY | HA_CAN_FULLTEXT | HA_CAN_SQL_HANDLER | HA_DUPLICATE_POS | HA_CAN_INDEX_BLOBS | HA_AUTO_PART_KEY | - HA_FILE_BASED | HA_CAN_GEOMETRY | HA_NO_TRANSACTIONS | + HA_FILE_BASED | HA_CAN_GEOMETRY | HA_CAN_INSERT_DELAYED | HA_CAN_BIT_FIELD | HA_CAN_RTREEKEYS | HA_HAS_RECORDS | HA_STATS_RECORDS_IS_EXACT), can_enable_indexes(1) @@ -1846,14 +1848,69 @@ int ha_maria::delete_table(const char *name) return maria_delete_table(name); } +#define THD_TRN (*(TRN **)thd_ha_data(thd, maria_hton)) int ha_maria::external_lock(THD *thd, int lock_type) { - return maria_lock_database(file, !table->s->tmp_table ? - lock_type : ((lock_type == F_UNLCK) ? - F_UNLCK : F_EXTRA_LCK)); + TRN *trn= THD_TRN; + DBUG_ENTER("ha_maria::external_lock"); + if (!trn && lock_type != F_UNLCK) /* no transaction yet - open it now */ + { + trn= trnman_new_trn(& thd->mysys_var->mutex, + & thd->mysys_var->suspend, + thd->thread_stack + STACK_DIRECTION * + (my_thread_stack_size - STACK_MIN_SIZE)); + if (!trn) + DBUG_RETURN(HA_ERR_OUT_OF_MEM); + + DBUG_PRINT("info", ("THD_TRN set to 0x%lx", (ulong)trn)); + THD_TRN= trn; + if (thd->options & (OPTION_NOT_AUTOCOMMIT | OPTION_BEGIN)) + trans_register_ha(thd, true, maria_hton); + } + if (lock_type != F_UNLCK) + { + this->file->trn= trn; + if (!trnman_increment_locked_tables(trn)) + { + trans_register_ha(thd, FALSE, maria_hton); + trnman_new_statement(trn); + } + } + else + { + this->file->trn= 0; /* TODO: remove it also in commit and rollback */ + if (trn && trnman_has_locked_tables(trn)) + { + if (!trnman_decrement_locked_tables(trn)) + { + /* autocommit ? rollback a transaction */ + if (!(thd->options & (OPTION_NOT_AUTOCOMMIT | OPTION_BEGIN))) + { + trnman_rollback_trn(trn); + DBUG_PRINT("info", ("THD_TRN set to 0x0")); + THD_TRN= 0; + } + } + } + } + DBUG_RETURN(maria_lock_database(file, !table->s->tmp_table ? + lock_type : ((lock_type == F_UNLCK) ? + F_UNLCK : F_EXTRA_LCK))); } +int ha_maria::start_stmt(THD *thd, thr_lock_type lock_type) +{ + TRN *trn= THD_TRN; + DBUG_ASSERT(trn); // this may be called only after external_lock() + DBUG_ASSERT(lock_type != F_UNLCK); + if (!trnman_increment_locked_tables(trn)) + { + trans_register_ha(thd, false, maria_hton); + trnman_new_statement(trn); + } + return 0; +} THR_LOCK_DATA **ha_maria::store_lock(THD *thd, THR_LOCK_DATA **to, @@ -1936,6 +1993,7 @@ int ha_maria::create(const char *name, register TABLE *table_arg, share->avg_row_length); create_info.data_file_name= ha_create_info->data_file_name; create_info.index_file_name= ha_create_info->index_file_name; + create_info.transactional= row_type == BLOCK_RECORD; if (ha_create_info->options & HA_LEX_CREATE_TMP_TABLE) create_flags|= HA_CREATE_TMP_TABLE; @@ -2089,26 +2147,72 @@ bool ha_maria::check_if_incompatible_data(HA_CREATE_INFO *info, return COMPATIBLE_DATA_YES; } -extern int maria_panic(enum ha_panic_function flag); -int maria_panic(handlerton *hton, ha_panic_function flag) + +static int maria_hton_panic(handlerton *hton, ha_panic_function flag) { return maria_panic(flag); } -static int ha_maria_init(void *p) + +static int maria_commit(handlerton *hton __attribute__ ((unused)), + THD *thd, bool all) { - handlerton *maria_hton; + TRN *trn= THD_TRN; + DBUG_ENTER("maria_commit"); + trnman_reset_locked_tables(trn); + /* statement or transaction ? */ + if ((thd->options & (OPTION_NOT_AUTOCOMMIT | OPTION_BEGIN)) && !all) + DBUG_RETURN(0); // end of statement + DBUG_PRINT("info", ("THD_TRN set to 0x0")); + THD_TRN= 0; + DBUG_RETURN(trnman_commit_trn(trn) ? + HA_ERR_OUT_OF_MEM : 0); // end of transaction +} + +static int maria_rollback(handlerton *hton __attribute__ ((unused)), + THD *thd, bool all) +{ + TRN *trn= THD_TRN; + DBUG_ENTER("maria_rollback"); + trnman_reset_locked_tables(trn); + /* statement or transaction ? */ + if ((thd->options & (OPTION_NOT_AUTOCOMMIT | OPTION_BEGIN)) && !all) + { + trnman_rollback_statement(trn); + DBUG_RETURN(0); // end of statement + } + DBUG_PRINT("info", ("THD_TRN set to 0x0")); + THD_TRN= 0; + DBUG_RETURN(trnman_rollback_trn(trn) ? + HA_ERR_OUT_OF_MEM : 0); // end of transaction +} + + +static int ha_maria_init(void *p) +{ maria_hton= (handlerton *)p; maria_hton->state= SHOW_OPTION_YES; maria_hton->db_type= DB_TYPE_MARIA; maria_hton->create= maria_create_handler; - maria_hton->panic= maria_panic; + maria_hton->panic= maria_hton_panic; + maria_hton->commit= maria_commit; + maria_hton->rollback= maria_rollback; /* TODO: decide if we support Maria being used for log tables */ maria_hton->flags= HTON_CAN_RECREATE | HTON_SUPPORT_LOG_TABLES; - return test(maria_init()); + bzero(maria_log_pagecache, sizeof(*maria_log_pagecache)); + maria_data_root= mysql_real_data_home; + return (test(maria_init() || ma_control_file_create_or_open() || + (init_pagecache(maria_log_pagecache, + TRANSLOG_PAGECACHE_SIZE, 0, 0, + TRANSLOG_PAGE_SIZE) == 0) || + translog_init(maria_data_root, TRANSLOG_FILE_SIZE, + MYSQL_VERSION_ID, server_id, maria_log_pagecache, + TRANSLOG_DEFAULT_FLAGS) || + trnman_init())); } + struct st_mysql_storage_engine maria_storage_engine= { MYSQL_HANDLERTON_INTERFACE_VERSION }; diff --git a/storage/maria/ha_maria.h b/storage/maria/ha_maria.h index 031a3dc3b98..dd0a9594ef3 100644 --- a/storage/maria/ha_maria.h +++ b/storage/maria/ha_maria.h @@ -110,6 +110,7 @@ public: int extra_opt(enum ha_extra_function operation, ulong cache_size); int reset(void); int external_lock(THD * thd, int lock_type); + int start_stmt(THD *thd, thr_lock_type lock_type); int delete_all_rows(void); int disable_indexes(uint mode); int enable_indexes(uint mode); diff --git a/storage/maria/lockman.c b/storage/maria/lockman.c index cb305dc9bd6..8316d70bb29 100644 --- a/storage/maria/lockman.c +++ b/storage/maria/lockman.c @@ -538,7 +538,7 @@ void lockman_destroy(LOCKMAN *lm) { intptr next= el->link; if (el->hashnr & 1) - lf_alloc_real_free(&lm->alloc, el); + lf_alloc_direct_free(&lm->alloc, el); else my_free((void *)el, MYF(0)); el= (LOCK *)next; diff --git a/storage/maria/ma_bitmap.c b/storage/maria/ma_bitmap.c index e7ced9fc1e1..923525922da 100644 --- a/storage/maria/ma_bitmap.c +++ b/storage/maria/ma_bitmap.c @@ -1837,7 +1837,7 @@ err: /* - Free full pages from bitmap + Free full pages from bitmap and pagecache SYNOPSIS _ma_bitmap_free_full_pages() @@ -1846,7 +1846,8 @@ err: count Number of extents IMPLEMENTATION - Mark all full pages (not tails) from extents as free + Mark all full pages (not tails) from extents as free, both in bitmap + and page cache. RETURN 0 ok @@ -1865,6 +1866,9 @@ my_bool _ma_bitmap_free_full_pages(MARIA_HA *info, const byte *extents, uint page_count= uint2korr(extents + ROW_EXTENT_PAGE_SIZE); if (!(page_count & TAIL_BIT)) { + if (pagecache_delete_pages(info->s->pagecache, &info->dfile, page, + page_count, PAGECACHE_LOCK_WRITE, 1)) + DBUG_RETURN(1); if (_ma_reset_full_page_bits(info, &info->s->bitmap, page, page_count)) { pthread_mutex_unlock(&info->s->bitmap.bitmap_lock); diff --git a/storage/maria/ma_blockrec.c b/storage/maria/ma_blockrec.c index bb1cddd4d77..8d8adde46d1 100644 --- a/storage/maria/ma_blockrec.c +++ b/storage/maria/ma_blockrec.c @@ -162,7 +162,7 @@ - Store the parts in as many full-contiguous pages as possible. - The last part, that doesn't fill a full page, is stored in tail page. - When doing an insert of a new row, we don't have to have + When doing an insert of a new row, we don't have to have VER_PTR in the row. This will make rows that are not changed stored efficiently. On update and delete we would add TRANSID (if it was an old committed row) and VER_PTR to @@ -239,7 +239,7 @@ 06 00 00 00 00 80 00 First blob, stored at page 6-133 05 00 00 00 00 01 80 Tail of first blob (896 bytes) at page 5 86 00 00 00 00 80 00 Second blob, stored at page 134-262 - 05 00 00 00 00 02 80 Tail of second blob (896 bytes) at page 5 + 05 00 00 00 00 02 80 Tail of second blob (896 bytes) at page 5 05 00 5 integer FA Length of first varchar field (size 250) 00 60 Length of second varchar field (size 8192*3) @@ -256,6 +256,8 @@ #include "maria_def.h" #include "ma_blockrec.h" +#include +#include "trnman.h" /* Struct for having a cursor over a set of extent. @@ -298,6 +300,15 @@ static my_bool delete_head_or_tail(MARIA_HA *info, static void _ma_print_directory(byte *buff, uint block_size); static void compact_page(byte *buff, uint block_size, uint rownr, my_bool extend_block); +static uchar *store_page_range(uchar *to, MARIA_BITMAP_BLOCK *block, + uint block_size, ulong length); +static size_t fill_insert_undo_parts(MARIA_HA *info, const byte *record, + LEX_STRING *log_parts, + uint *log_parts_count); +static size_t fill_update_undo_parts(MARIA_HA *info, const byte *oldrec, + const byte *newrec, + LEX_STRING *log_parts, + uint *log_parts_count); /**************************************************************************** Initialization @@ -401,8 +412,9 @@ my_bool _ma_init_block_record(MARIA_HA *info) DBUG_ENTER("_ma_init_block_record"); if (!my_multi_malloc(MY_WME, - &row->empty_bits_buffer, info->s->base.pack_bytes, - &row->field_lengths, info->s->base.max_field_lengths, + &row->empty_bits, info->s->base.pack_bytes, + &row->field_lengths, + info->s->base.max_field_lengths + 2, &row->blob_lengths, sizeof(ulong) * info->s->base.blobs, &row->null_field_lengths, (sizeof(uint) * (info->s->base.fields - @@ -410,21 +422,37 @@ my_bool _ma_init_block_record(MARIA_HA *info) EXTRA_LENGTH_FIELDS)), &row->tail_positions, (sizeof(MARIA_RECORD_POS) * (info->s->base.blobs + 2)), - &new_row->empty_bits_buffer, info->s->base.pack_bytes, + &new_row->empty_bits, info->s->base.pack_bytes, &new_row->field_lengths, - info->s->base.max_field_lengths, + info->s->base.max_field_lengths + 2, &new_row->blob_lengths, sizeof(ulong) * info->s->base.blobs, &new_row->null_field_lengths, (sizeof(uint) * (info->s->base.fields - info->s->base.blobs + EXTRA_LENGTH_FIELDS)), + &info->log_row_parts, + sizeof(*info->log_row_parts) * + (TRANSLOG_INTERNAL_PARTS + 2 + + info->s->base.fields + 2), + &info->update_field_data, + (info->s->base.fields * 4 + + info->s->base.max_field_lengths + 1 + 4), NullS, 0)) DBUG_RETURN(1); + /* Skip over bytes used to store length of field length for logging */ + row->field_lengths+= 2; + new_row->field_lengths+= 2; if (my_init_dynamic_array(&info->bitmap_blocks, sizeof(MARIA_BITMAP_BLOCK), ELEMENTS_RESERVED_FOR_MAIN_PART, 16)) - my_free((char*) &info->bitmap_blocks, MYF(0)); + goto err; + /* The following should be big enough for all purposes */ + if (my_init_dynamic_array(&info->pinned_pages, + sizeof(MARIA_PINNED_PAGE), + max(info->s->base.blobs + 2, + MARIA_MAX_TREE_LEVELS*2), 16)) + goto err; row->base_length= new_row->base_length= info->s->base_length; /* @@ -434,15 +462,21 @@ my_bool _ma_init_block_record(MARIA_HA *info) row->null_field_lengths+= EXTRA_LENGTH_FIELDS; new_row->null_field_lengths+= EXTRA_LENGTH_FIELDS; + DBUG_RETURN(0); + +err: + _ma_end_block_record(info); + DBUG_RETURN(1); } void _ma_end_block_record(MARIA_HA *info) { DBUG_ENTER("_ma_end_block_record"); - my_free((gptr) info->cur_row.empty_bits_buffer, MYF(MY_ALLOW_ZERO_PTR)); + my_free((gptr) info->cur_row.empty_bits, MYF(MY_ALLOW_ZERO_PTR)); delete_dynamic(&info->bitmap_blocks); + delete_dynamic(&info->pinned_pages); my_free((gptr) info->cur_row.extents, MYF(MY_ALLOW_ZERO_PTR)); /* The data file is closed, when needed, in ma_once_end_block_record(). @@ -507,6 +541,44 @@ static my_bool check_if_zero(byte *pos, uint length) } +/* + Unpin all pinned pages + + SYNOPSIS + _ma_unpin_all_pages() + info Maria handler + undo_lsn LSN for undo pages. 0 if we shouldn't write undo (error) + + NOTE + We unpin pages in the reverse order as they where pinned; This may not + be strictly necessary but may simplify things in the future. + + RETURN + 0 ok + 1 error (fatal disk error) + +*/ + +void _ma_unpin_all_pages(MARIA_HA *info, LSN undo_lsn) +{ + MARIA_PINNED_PAGE *page_link= ((MARIA_PINNED_PAGE*) + dynamic_array_ptr(&info->pinned_pages, 0)); + MARIA_PINNED_PAGE *pinned_page= page_link + info->pinned_pages.elements; + DBUG_ENTER("_ma_unpin_all_pages"); + DBUG_PRINT("info", ("undo_lsn: %lu", (ulong) undo_lsn)); + + /* True if not disk error */ + DBUG_ASSERT(undo_lsn != 0 || info->s->base.transactional == 0); + + while (pinned_page-- != page_link) + pagecache_unlock(info->s->pagecache, pinned_page->link, + pinned_page->unlock, PAGECACHE_UNPIN, 0, undo_lsn); + + info->pinned_pages.elements= 0; + DBUG_VOID_RETURN; +} + + /* Find free position in directory @@ -576,7 +648,7 @@ static byte *find_free_position(byte *buff, uint block_size, uint *res_rownr, if (max_entry == MAX_ROWS_PER_PAGE) DBUG_RETURN(0); /* Check if there is place for the directory entry */ - if ((dir - buff) < first_pos) + if ((uint) (dir - buff) < first_pos) { /* Create place for directory */ compact_page(buff, block_size, max_entry-1, 0); @@ -595,9 +667,9 @@ static byte *find_free_position(byte *buff, uint block_size, uint *res_rownr, /* Reduce directory entry size from free space size */ (*empty_space)-= DIR_ENTRY_SIZE; DBUG_RETURN(dir); - } + /**************************************************************************** Updating records ****************************************************************************/ @@ -630,8 +702,7 @@ static void calc_record_size(MARIA_HA *info, const byte *record, row->blob_length= row->extents_count= 0; /* Create empty bitmap and calculate length of each varlength/char field */ - bzero(row->empty_bits_buffer, share->base.pack_bytes); - row->empty_bits= row->empty_bits_buffer; + bzero(row->empty_bits, share->base.pack_bytes); field_length_data= row->field_lengths; for (column= share->columndef + share->base.fixed_not_null_fields, end_column= share->columndef + share->base.fields; @@ -918,20 +989,23 @@ struct st_row_pos_info byte *data; /* Place for data */ byte *dir; /* Directory */ uint length; /* Length for data */ - uint offset; /* Offset to directory */ + uint rownr; /* Offset in directory */ uint empty_space; /* Space left on page */ }; static my_bool get_head_or_tail_page(MARIA_HA *info, MARIA_BITMAP_BLOCK *block, byte *buff, uint length, uint page_type, + enum pagecache_page_lock lock, struct st_row_pos_info *res) { uint block_size; + MARIA_PINNED_PAGE page_link; + MARIA_SHARE *share= info->s; DBUG_ENTER("get_head_or_tail_page"); DBUG_PRINT("enter", ("length: %u", length)); - block_size= info->s->block_size; + block_size= share->block_size; if (block->org_bitmap_value == 0) /* Empty block */ { /* New page */ @@ -951,7 +1025,7 @@ static my_bool get_head_or_tail_page(MARIA_HA *info, res->empty_space= res->length= (block_size - PAGE_OVERHEAD_SIZE); res->data= (buff + PAGE_HEADER_SIZE); res->dir= res->data + res->length; - res->offset= 0; + res->rownr= 0; /* Store position to the first row */ int2store(res->dir, PAGE_HEADER_SIZE); DBUG_ASSERT(length <= res->length); @@ -961,15 +1035,23 @@ static my_bool get_head_or_tail_page(MARIA_HA *info, byte *dir; /* TODO: lock the page */ /* Read old page */ - DBUG_ASSERT(info->s->pagecache->block_size == block_size); - if (!(res->buff= pagecache_read(info->s->pagecache, + DBUG_ASSERT(share->pagecache->block_size == block_size); + if (!(res->buff= pagecache_read(share->pagecache, &info->dfile, (my_off_t) block->page, 0, - buff, PAGECACHE_PLAIN_PAGE, - PAGECACHE_LOCK_LEFT_UNLOCKED, 0))) + buff, share->page_type, + lock, &page_link.link))) DBUG_RETURN(1); + if (lock != PAGECACHE_LOCK_LEFT_UNLOCKED) + { + page_link.unlock= (lock == PAGECACHE_LOCK_READ ? + PAGECACHE_LOCK_READ_UNLOCK : + PAGECACHE_LOCK_WRITE_UNLOCK); + push_dynamic(&info->pinned_pages, (void*) &page_link); + } + DBUG_ASSERT((res->buff[PAGE_TYPE_OFFSET] & PAGE_TYPE_MASK) == page_type); - if (!(dir= find_free_position(res->buff, block_size, &res->offset, + if (!(dir= find_free_position(res->buff, block_size, &res->rownr, &res->length, &res->empty_space))) goto crashed; @@ -977,9 +1059,9 @@ static my_bool get_head_or_tail_page(MARIA_HA *info, { if (res->empty_space + res->length < length) { - compact_page(res->buff, block_size, res->offset, 1); + compact_page(res->buff, block_size, res->rownr, 1); /* All empty space are now after current position */ - dir= (res->buff + block_size - DIR_ENTRY_SIZE * res->offset - + dir= (res->buff + block_size - DIR_ENTRY_SIZE * res->rownr - PAGE_SUFFIX_SIZE); res->length= res->empty_space= uint2korr(dir+2); } @@ -998,7 +1080,7 @@ crashed: /* - Write tail of non-blob-data or blob + Write tail for head data or blob SYNOPSIS write_tail() @@ -1023,9 +1105,11 @@ static my_bool write_tail(MARIA_HA *info, byte *row_part, uint length) { MARIA_SHARE *share= share= info->s; + MARIA_PINNED_PAGE page_link; uint block_size= share->block_size, empty_space; struct st_row_pos_info row_pos; my_off_t position; + my_bool res, block_is_read; DBUG_ENTER("write_tail"); DBUG_PRINT("enter", ("page: %lu length: %u", (ulong) block->page, length)); @@ -1034,10 +1118,36 @@ static my_bool write_tail(MARIA_HA *info, /* page will be pinned & locked by get_head_or_tail_page */ if (get_head_or_tail_page(info, block, info->keyread_buff, length, - TAIL_PAGE, &row_pos)) + TAIL_PAGE, PAGECACHE_LOCK_WRITE, + &row_pos)) DBUG_RETURN(1); + block_is_read= block->org_bitmap_value != 0; memcpy(row_pos.data, row_part, length); + + { + /* Log changes in tail block */ + uchar log_data[FILEID_STORE_SIZE + PAGE_STORE_SIZE + DIRPOS_STORE_SIZE]; + LEX_STRING log_array[TRANSLOG_INTERNAL_PARTS + 2]; + LSN lsn; + + /* Log REDO changes of tail page */ + fileid_store(log_data, info->dfile.file); + page_store(log_data+ FILEID_STORE_SIZE, block->page); + dirpos_store(log_data+ FILEID_STORE_SIZE + PAGE_STORE_SIZE, + row_pos.rownr); + log_array[TRANSLOG_INTERNAL_PARTS + 0].str= (char*) log_data; + log_array[TRANSLOG_INTERNAL_PARTS + 0].length= sizeof(log_data); + log_array[TRANSLOG_INTERNAL_PARTS + 1].str= (char*) row_pos.data; + log_array[TRANSLOG_INTERNAL_PARTS + 1].length= length; + if (translog_write_record(&lsn, LOGREC_REDO_INSERT_ROW_TAIL, + info->trn->short_id, NULL, share, + sizeof(log_data) + length, + TRANSLOG_INTERNAL_PARTS + 2, + log_array)) + DBUG_RETURN(1); + } + /* Don't allocate smaller block than MIN_TAIL_SIZE (we want to give rows some place to grow in the future) @@ -1047,7 +1157,7 @@ static my_bool write_tail(MARIA_HA *info, int2store(row_pos.dir + 2, length); empty_space= row_pos.empty_space - length; int2store(row_pos.buff + EMPTY_SPACE_OFFSET, empty_space); - block->page_count= row_pos.offset + TAIL_BIT; + block->page_count= row_pos.rownr + TAIL_BIT; /* If there is less directory entries free than number of possible tails we can write for a row, we mark the page full to ensure that we don't @@ -1055,7 +1165,7 @@ static my_bool write_tail(MARIA_HA *info, than it can hold */ block->empty_space= ((uint) ((uchar*) row_pos.buff)[DIR_COUNT_OFFSET] <= - MAX_ROWS_PER_PAGE - 1 - info->s->base.blobs ? + MAX_ROWS_PER_PAGE - 1 - share->base.blobs ? empty_space : 0); block->used= BLOCKUSED_USED | BLOCKUSED_TAIL; @@ -1063,15 +1173,28 @@ static my_bool write_tail(MARIA_HA *info, position= (my_off_t) block->page * block_size; if (info->state->data_file_length <= position) info->state->data_file_length= position + block_size; - /* TODO: left the page pinned (or pin it if it is new) and unlock\ - the page (do not lock if it is new */ + DBUG_ASSERT(share->pagecache->block_size == block_size); - DBUG_RETURN(pagecache_write(share->pagecache, - &info->dfile, block->page, 0, - row_pos.buff,PAGECACHE_PLAIN_PAGE, - PAGECACHE_LOCK_LEFT_UNLOCKED, - PAGECACHE_PIN_LEFT_UNPINNED, - PAGECACHE_WRITE_DELAY, 0)); + if (!(res= pagecache_write(share->pagecache, + &info->dfile, block->page, 0, + row_pos.buff,share->page_type, + block_is_read ? PAGECACHE_LOCK_WRITE_TO_READ : + PAGECACHE_LOCK_READ, + block_is_read ? PAGECACHE_PIN_LEFT_PINNED : + PAGECACHE_PIN, + PAGECACHE_WRITE_DELAY, &page_link.link))) + { + page_link.unlock= PAGECACHE_LOCK_READ_UNLOCK; + if (block_is_read) + { + /* Change the lock used when we read the page */ + set_dynamic(&info->pinned_pages, (void*) &page_link, + info->pinned_pages.elements-1); + } + else + push_dynamic(&info->pinned_pages, (void*) &page_link); + } + DBUG_RETURN(res); } @@ -1081,13 +1204,22 @@ static my_bool write_tail(MARIA_HA *info, SYNOPSIS write_full_pages() info Maria handler + lsn LSN for the undo record block Where to write data data Data to write length Length of data + NOTES + Logging of the changes to the full pages are done in the caller + write_block_record(). + + RETURN + 0 ok + 1 error on write */ static my_bool write_full_pages(MARIA_HA *info, + LSN lsn, MARIA_BITMAP_BLOCK *block, byte *data, ulong length) { @@ -1128,20 +1260,16 @@ static my_bool write_full_pages(MARIA_HA *info, if (info->state->data_file_length < position) info->state->data_file_length= position; } - bzero(buff, LSN_SIZE); + lsn_store(buff, lsn); buff[PAGE_TYPE_OFFSET]= (byte) BLOB_PAGE; copy_length= min(data_size, length); memcpy(buff + LSN_SIZE + PAGE_TYPE_SIZE, data, copy_length); length-= copy_length; - /* - TODO: replace PAGECACHE_PLAIN_PAGE with PAGECACHE_LSN_PAGE when - LSN on the pages will be implemented - */ DBUG_ASSERT(share->pagecache->block_size == block_size); if (pagecache_write(share->pagecache, &info->dfile, page, 0, - buff, PAGECACHE_PLAIN_PAGE, + buff, share->page_type, PAGECACHE_LOCK_LEFT_UNLOCKED, PAGECACHE_PIN_LEFT_UNPINNED, PAGECACHE_WRITE_DELAY, @@ -1154,6 +1282,46 @@ static my_bool write_full_pages(MARIA_HA *info, } +/* + Store ranges of full pages in compact format for logging + + SYNOPSIS + store_page_range() + to Store data here + block Where pages are to be written + block_size block size + length Length of data to be written + Normally this is full pages, except for the last + tail block that may only partly fit the last page. + + RETURN + # end position for 'to' +*/ + +static uchar *store_page_range(uchar *to, MARIA_BITMAP_BLOCK *block, + uint block_size, ulong length) +{ + uint data_size= FULL_PAGE_SIZE(block_size); + ulong pages_left= (length + data_size -1) / data_size; + uint page_count; + DBUG_ENTER("store_page_range"); + + do + { + ulonglong page; + page= block->page; + page_count= block->page_count; + block++; + if (page_count > pages_left) + page_count= pages_left; + + page_store(to, page); + to+= PAGE_STORE_SIZE; + pagerange_store(to, page_count); + to+= PAGERANGE_STORE_SIZE; + } while ((pages_left-= page_count)); + DBUG_RETURN(to); +} /* @@ -1186,8 +1354,8 @@ static void store_extent_info(byte *to, if (likely(block->used & BLOCKUSED_USED)) { DBUG_ASSERT(block->page_count != 0); - int5store(to, block->page); - int2store(to + 5, block->page_count); + page_store(to, block->page); + pagerange_store(to + PAGE_STORE_SIZE, block->page_count); to+= ROW_EXTENT_SIZE; if (!first_found) { @@ -1204,25 +1372,113 @@ static void store_extent_info(byte *to, bzero(to, (my_size_t) (row_extents_second_part + copy_length - to)); } + +/* + Free regions of pages with logging + + RETURN + 0 ok + 1 error +*/ + +static my_bool free_full_pages(MARIA_HA *info, MARIA_ROW *row) +{ + uchar log_data[FILEID_STORE_SIZE + PAGERANGE_STORE_SIZE]; + LEX_STRING log_array[TRANSLOG_INTERNAL_PARTS + 2]; + LSN lsn; + size_t extents_length= row->extents_count * ROW_EXTENT_SIZE; + DBUG_ENTER("free_full_pages"); + + fileid_store(log_data, info->dfile.file); + pagerange_store(log_data + FILEID_STORE_SIZE, + row->extents_count); + log_array[TRANSLOG_INTERNAL_PARTS + 0].str= (char*) log_data; + log_array[TRANSLOG_INTERNAL_PARTS + 0].length= sizeof(log_data); + log_array[TRANSLOG_INTERNAL_PARTS + 1].str= row->extents; + log_array[TRANSLOG_INTERNAL_PARTS + 1].length= extents_length; + if (translog_write_record(&lsn, LOGREC_REDO_PURGE_BLOCKS, + info->trn->short_id, NULL, info->s, + sizeof(log_data) + extents_length, + TRANSLOG_INTERNAL_PARTS + 2, log_array)) + DBUG_RETURN(1); + + DBUG_RETURN (_ma_bitmap_free_full_pages(info, row->extents, + row->extents_count)); +} + + +/* + Free one page range + + NOTES + This is very similar to free_full_pages() + + RETURN + 0 ok + 1 error +*/ + +static my_bool free_full_page_range(MARIA_HA *info, ulonglong page, uint count) +{ + uchar log_data[FILEID_STORE_SIZE + PAGERANGE_STORE_SIZE + + ROW_EXTENT_SIZE]; + LEX_STRING log_array[TRANSLOG_INTERNAL_PARTS + 1]; + LSN lsn; + my_bool res= 0; + + if (pagecache_delete_pages(info->s->pagecache, &info->dfile, + page, count, PAGECACHE_LOCK_WRITE, 0)) + res= 1; + + fileid_store(log_data, info->dfile.file); + pagerange_store(log_data + FILEID_STORE_SIZE, 1); + int5store(log_data + FILEID_STORE_SIZE + PAGERANGE_STORE_SIZE, + page); + int2store(log_data + FILEID_STORE_SIZE + PAGERANGE_STORE_SIZE + 5, + count); + log_array[TRANSLOG_INTERNAL_PARTS + 0].str= (char*) log_data; + log_array[TRANSLOG_INTERNAL_PARTS + 0].length= sizeof(log_data); + + if (translog_write_record(&lsn, LOGREC_REDO_PURGE_BLOCKS, + info->trn->short_id, NULL, info->s, + sizeof(log_data), + TRANSLOG_INTERNAL_PARTS + 1, log_array)) + res= 1; + + pthread_mutex_lock(&info->s->bitmap.bitmap_lock); + if (_ma_reset_full_page_bits(info, &info->s->bitmap, page, + count)) + res= 1; + pthread_mutex_unlock(&info->s->bitmap.bitmap_lock); + return res; +} + + /* Write a record to a (set of) pages SYNOPSIS write_block_record() info Maria handler + old_record Orignal record in case of update; NULL in case of insert record Record we should write row Statistics about record (calculated by calc_record_size()) map_blocks On which pages the record should be stored row_pos Position on head page where to put head part of record + NOTES + On return all pinned pages are released. + RETURN 0 ok 1 error */ -static my_bool write_block_record(MARIA_HA *info, const byte *record, +static my_bool write_block_record(MARIA_HA *info, + const byte *old_record, const byte *record, MARIA_ROW *row, MARIA_BITMAP_BLOCKS *bitmap_blocks, + my_bool head_block_is_read, struct st_row_pos_info *row_pos) { byte *data, *end_of_data, *tmp_data_used, *tmp_data; @@ -1230,18 +1486,19 @@ static my_bool write_block_record(MARIA_HA *info, const byte *record, byte *field_length_data; byte *page_buff; MARIA_BITMAP_BLOCK *block, *head_block; - MARIA_SHARE *share; + MARIA_SHARE *share= info->s; MARIA_COLUMNDEF *column, *end_column; + MARIA_PINNED_PAGE page_link; uint block_size, flag; ulong *blob_lengths; + my_bool row_extents_in_use, blob_full_pages_exists; + LSN lsn; my_off_t position; - my_bool row_extents_in_use; DBUG_ENTER("write_block_record"); LINT_INIT(row_extents_first_part); LINT_INIT(row_extents_second_part); - share= info->s; head_block= bitmap_blocks->block; block_size= share->block_size; @@ -1458,6 +1715,7 @@ static my_bool write_block_record(MARIA_HA *info, const byte *record, the tail page of the non-blob data) */ + blob_full_pages_exists= 0; if (row_extents_in_use) { if (column != end_column) /* If blob fields */ @@ -1479,11 +1737,16 @@ static my_bool write_block_record(MARIA_HA *info, const byte *record, memcpy_fixed((byte *) &blob_pos, record + column->offset + length, sizeof(char*)); length= *blob_lengths % FULL_PAGE_SIZE(block_size); /* tail size */ + if (length != *blob_lengths) + blob_full_pages_exists= 1; if (write_tail(info, block + block->sub_blocks-1, blob_pos + *blob_lengths - length, length)) goto disk_err; } + else + blob_full_pages_exists= 1; + for (end_block= block + block->sub_blocks; block < end_block; block++) { /* @@ -1668,18 +1931,196 @@ static my_bool write_block_record(MARIA_HA *info, const byte *record, row_extents_second_part, head_block+1, bitmap_blocks->count - 1); } + + if (share->base.transactional) + { + uchar log_data[FILEID_STORE_SIZE + PAGE_STORE_SIZE + DIRPOS_STORE_SIZE]; + LEX_STRING log_array[TRANSLOG_INTERNAL_PARTS + 2]; + size_t data_length= (size_t) (data - row_pos->data); + + /* Log REDO changes of head page */ + fileid_store(log_data, info->dfile.file); + page_store(log_data+ FILEID_STORE_SIZE, head_block->page); + dirpos_store(log_data+ FILEID_STORE_SIZE + PAGE_STORE_SIZE, + row_pos->rownr); + log_array[TRANSLOG_INTERNAL_PARTS + 0].str= (char*) log_data; + log_array[TRANSLOG_INTERNAL_PARTS + 0].length= sizeof(log_data); + log_array[TRANSLOG_INTERNAL_PARTS + 1].str= (char*) row_pos->data; + log_array[TRANSLOG_INTERNAL_PARTS + 1].length= data_length; + if (translog_write_record(&lsn, LOGREC_REDO_INSERT_ROW_HEAD, + info->trn->short_id, NULL, share, + sizeof(log_data) + data_length, + TRANSLOG_INTERNAL_PARTS + 2, log_array)) + goto disk_err; + } + /* Increase data file size, if extended */ position= (my_off_t) head_block->page * block_size; if (info->state->data_file_length <= position) info->state->data_file_length= position + block_size; + DBUG_ASSERT(share->pagecache->block_size == block_size); if (pagecache_write(share->pagecache, &info->dfile, head_block->page, 0, - page_buff, PAGECACHE_PLAIN_PAGE, - PAGECACHE_LOCK_LEFT_UNLOCKED, - PAGECACHE_PIN_LEFT_UNPINNED, - PAGECACHE_WRITE_DELAY, 0)) + page_buff, share->page_type, + head_block_is_read ? PAGECACHE_LOCK_WRITE_TO_READ : + PAGECACHE_LOCK_READ, + head_block_is_read ? PAGECACHE_PIN_LEFT_PINNED : + PAGECACHE_PIN, + PAGECACHE_WRITE_DELAY, &page_link.link)) goto disk_err; + page_link.unlock= PAGECACHE_LOCK_READ_UNLOCK; + if (head_block_is_read) + { + /* Head page is always the first pinned page */ + set_dynamic(&info->pinned_pages, (void*) &page_link, 0); + } + else + push_dynamic(&info->pinned_pages, (void*) &page_link); + + if (share->base.transactional && (tmp_data_used || blob_full_pages_exists)) + { + /* + Log REDO writes for all full pages (head part and all blobs) + We write all here to be able to generate the UNDO record early + so that we can write the LSN for the UNDO record to all full pages. + */ + uchar tmp_log_data[FILEID_STORE_SIZE + LSN_STORE_SIZE + PAGE_STORE_SIZE + + ROW_EXTENT_SIZE * ROW_EXTENTS_ON_STACK]; + uchar *log_data, *log_pos; + LEX_STRING tmp_log_array[TRANSLOG_INTERNAL_PARTS + 2 + + ROW_EXTENTS_ON_STACK]; + LEX_STRING *log_array_pos, *log_array; + int error; + ulong log_entry_length= 0; + + /* If few extents, then allocate things on stack to avoid a malloc call */ + if (bitmap_blocks->count < ROW_EXTENTS_ON_STACK) + { + log_array= tmp_log_array; + log_data= tmp_log_data; + } + else + { + if (my_multi_malloc(MY_WME, &log_array, + (uint) ((bitmap_blocks->count + + TRANSLOG_INTERNAL_PARTS + 2) * + sizeof(*log_array)), + &log_data, bitmap_blocks->count * ROW_EXTENT_SIZE, + NullS)) + goto disk_err; + } + fileid_store(log_data, info->dfile.file); + log_pos= log_data + FILEID_STORE_SIZE; + log_array_pos= log_array+ TRANSLOG_INTERNAL_PARTS+1; + + if (tmp_data_used) + { + /* Full head pages */ + size_t data_length= (ulong) (tmp_data - info->rec_buff); + log_pos= store_page_range(log_pos, head_block+1, block_size, + data_length); + log_array_pos->str= (char*) info->rec_buff; + log_array_pos->length= data_length; + log_entry_length+= data_length; + log_array_pos++; + } + if (blob_full_pages_exists) + { + MARIA_COLUMNDEF *tmp_column= column; + ulong *tmp_blob_lengths= blob_lengths; + MARIA_BITMAP_BLOCK *tmp_block= block; + + /* Full blob pages */ + for (; tmp_column < end_column; tmp_column++, tmp_blob_lengths++) + { + ulong blob_length; + uint length; + + if (!*tmp_blob_lengths) /* Null or "" */ + continue; + length= tmp_column->length - portable_sizeof_char_ptr; + blob_length= *tmp_blob_lengths; + if (tmp_block[tmp_block->sub_blocks - 1].used & BLOCKUSED_TAIL) + blob_length-= (blob_length % FULL_PAGE_SIZE(block_size)); + if (blob_length) + { + log_array_pos->str= (char*) record + column->offset + length; + log_array_pos->length= blob_length; + log_entry_length+= blob_length; + log_array_pos++; + + log_pos= store_page_range(log_pos, tmp_block, block_size, + blob_length); + tmp_block+= tmp_block->sub_blocks; + } + } + } + + log_array[TRANSLOG_INTERNAL_PARTS + 0].str= (char*) log_data; + log_array[TRANSLOG_INTERNAL_PARTS + 0].length= (size_t) (log_pos - + log_data); + log_entry_length+= (log_pos - log_data); + + error= translog_write_record(&lsn, LOGREC_REDO_INSERT_ROW_BLOBS, + info->trn->short_id, NULL, share, + log_entry_length, (uint) (log_array_pos - + log_array), + log_array); + if (log_array != tmp_log_array) + my_free((gptr) log_array, MYF(0)); + if (error) + goto disk_err; + } + + /* Write UNDO record */ + if (share->base.transactional) + { + uchar log_data[LSN_STORE_SIZE + FILEID_STORE_SIZE + + PAGE_STORE_SIZE + DIRPOS_STORE_SIZE]; + LEX_STRING *log_array= info->log_row_parts; + + /* LOGREC_UNDO_ROW_INSERT & LOGREC_UNDO_ROW_INSERT share same header */ + lsn_store(log_data, info->trn->undo_lsn); + fileid_store(log_data + LSN_STORE_SIZE, info->dfile.file); + page_store(log_data+ LSN_STORE_SIZE + FILEID_STORE_SIZE, + head_block->page); + dirpos_store(log_data+ LSN_STORE_SIZE + FILEID_STORE_SIZE + + PAGE_STORE_SIZE, + row_pos->rownr); + + log_array[TRANSLOG_INTERNAL_PARTS + 0].str= (char*) log_data; + log_array[TRANSLOG_INTERNAL_PARTS + 0].length= sizeof(log_data); + + if (!old_record) + { + /* Write UNDO log record for the INSERT */ + if (translog_write_record(&info->trn->undo_lsn, LOGREC_UNDO_ROW_INSERT, + info->trn->short_id, NULL, share, + sizeof(log_data), + TRANSLOG_INTERNAL_PARTS + 1, + log_array)) + goto disk_err; + } + else + { + /* Write UNDO log record for the UPDATE */ + size_t row_length; + uint row_parts_count; + row_length= fill_update_undo_parts(info, old_record, record, + info->log_row_parts + + TRANSLOG_INTERNAL_PARTS + 1, + &row_parts_count); + if (translog_write_record(&info->trn->undo_lsn, LOGREC_UNDO_ROW_UPDATE, + info->trn->short_id, NULL, share, + sizeof(log_data) + row_length, + TRANSLOG_INTERNAL_PARTS + 1 + row_parts_count, + log_array)) + goto disk_err; + } + } + + _ma_unpin_all_pages(info, info->trn->undo_lsn); if (tmp_data_used) { @@ -1688,8 +2129,8 @@ static my_bool write_block_record(MARIA_HA *info, const byte *record, This is the char/varchar data that didn't fit into the head page. */ DBUG_ASSERT(bitmap_blocks->count != 0); - if (write_full_pages(info, head_block + 1, info->rec_buff, - (ulong) (tmp_data - info->rec_buff))) + if (write_full_pages(info, info->trn->undo_lsn, head_block + 1, + info->rec_buff, (ulong) (tmp_data - info->rec_buff))) goto disk_err; } @@ -1709,7 +2150,8 @@ static my_bool write_block_record(MARIA_HA *info, const byte *record, if (block[block->sub_blocks - 1].used & BLOCKUSED_TAIL) blob_length-= (blob_length % FULL_PAGE_SIZE(block_size)); - if (write_full_pages(info, block, blob_pos, blob_length)) + if (write_full_pages(info, info->trn->undo_lsn, block, + blob_pos, blob_length)) goto disk_err; block+= block->sub_blocks; } @@ -1719,9 +2161,13 @@ static my_bool write_block_record(MARIA_HA *info, const byte *record, DBUG_RETURN(0); crashed: - my_errno= HA_ERR_WRONG_IN_RECORD; /* File crashed */ + /* Something was wrong with data on page */ + my_errno= HA_ERR_WRONG_IN_RECORD; + disk_err: - /* Something was wrong with data on record */ + /* Unpin all pinned pages to not cause problems for disk cache */ + _ma_unpin_all_pages(info, 0); + DBUG_RETURN(1); } @@ -1754,12 +2200,15 @@ MARIA_RECORD_POS _ma_write_init_block_record(MARIA_HA *info, DBUG_RETURN(HA_OFFSET_ERROR); /* Error reading bitmap */ /* page will be pinned & locked by get_head_or_tail_page */ if (get_head_or_tail_page(info, blocks->block, info->buff, - info->s->base.min_row_length, HEAD_PAGE, &row_pos)) + info->s->base.min_row_length, HEAD_PAGE, + PAGECACHE_LOCK_WRITE, &row_pos)) DBUG_RETURN(HA_OFFSET_ERROR); - info->cur_row.lastpos= ma_recordpos(blocks->block->page, row_pos.offset); + info->cur_row.lastpos= ma_recordpos(blocks->block->page, row_pos.rownr); if (info->s->calc_checksum) info->cur_row.checksum= (info->s->calc_checksum)(info,record); - if (write_block_record(info, record, &info->cur_row, blocks, &row_pos)) + if (write_block_record(info, (byte*) 0, record, &info->cur_row, + blocks, blocks->block->org_bitmap_value != 0, + &row_pos)) DBUG_RETURN(HA_OFFSET_ERROR); /* Error reading bitmap */ DBUG_PRINT("exit", ("Rowid: %lu", (ulong) info->cur_row.lastpos)); info->s->state.split++; @@ -1775,8 +2224,7 @@ MARIA_RECORD_POS _ma_write_init_block_record(MARIA_HA *info, */ my_bool _ma_write_block_record(MARIA_HA *info __attribute__ ((unused)), - const byte *record __attribute__ ((unused)) -) + const byte *record __attribute__ ((unused))) { return 0; /* Row already written */ } @@ -1822,15 +2270,36 @@ my_bool _ma_write_abort_block_record(MARIA_HA *info) 0)) res= 1; } - else + else if (block->used & BLOCKUSED_USED) { - pthread_mutex_lock(&info->s->bitmap.bitmap_lock); - if (_ma_reset_full_page_bits(info, &info->s->bitmap, block->page, - block->page_count)) + if (free_full_page_range(info, block->page, block->page_count)) res= 1; - pthread_mutex_unlock(&info->s->bitmap.bitmap_lock); } } + + if (info->s->base.transactional) + { + LEX_STRING log_array[TRANSLOG_INTERNAL_PARTS + 1]; + uchar log_data[LSN_STORE_SIZE]; + + /* + Write UNDO record + This entry is just an end marker for the abort_insert as we will never + really undo a failed insert. Note that this UNDO will cause recover + to ignore the LOGREC_UNDO_ROW_INSERT that is the previous entry + in the UNDO chain. + */ + lsn_store(log_data, info->trn->undo_lsn); + log_array[TRANSLOG_INTERNAL_PARTS + 0].str= (char*) log_data; + log_array[TRANSLOG_INTERNAL_PARTS + 0].length= sizeof(log_data); + if (translog_write_record(&info->trn->undo_lsn, LOGREC_UNDO_ROW_PURGE, + info->trn->short_id, NULL, info->s, + sizeof(log_data), + TRANSLOG_INTERNAL_PARTS + 1, + log_array)) + res= 1; + } + _ma_unpin_all_pages(info, info->trn->undo_lsn); DBUG_RETURN(res); } @@ -1845,28 +2314,33 @@ my_bool _ma_write_abort_block_record(MARIA_HA *info) */ my_bool _ma_update_block_record(MARIA_HA *info, MARIA_RECORD_POS record_pos, - const byte *record) + const byte *oldrec, const byte *record) { MARIA_BITMAP_BLOCKS *blocks= &info->cur_row.insert_blocks; byte *buff; MARIA_ROW *cur_row= &info->cur_row, *new_row= &info->new_row; + MARIA_PINNED_PAGE page_link; uint rownr, org_empty_size, head_length; uint block_size= info->s->block_size; byte *dir; ulonglong page; struct st_row_pos_info row_pos; + MARIA_SHARE *share= info->s; DBUG_ENTER("_ma_update_block_record"); DBUG_PRINT("enter", ("rowid: %lu", (long) record_pos)); calc_record_size(info, record, new_row); page= ma_recordpos_to_page(record_pos); - DBUG_ASSERT(info->s->pagecache->block_size == block_size); - if (!(buff= pagecache_read(info->s->pagecache, + DBUG_ASSERT(share->pagecache->block_size == block_size); + if (!(buff= pagecache_read(share->pagecache, &info->dfile, (my_off_t) page, 0, - info->buff, PAGECACHE_PLAIN_PAGE, - PAGECACHE_LOCK_LEFT_UNLOCKED, 0))) + info->buff, share->page_type, + PAGECACHE_LOCK_WRITE, &page_link.link))) DBUG_RETURN(1); + page_link.unlock= PAGECACHE_LOCK_WRITE_UNLOCK; + push_dynamic(&info->pinned_pages, (void*) &page_link); + org_empty_size= uint2korr(buff + EMPTY_SPACE_OFFSET); rownr= ma_recordpos_to_dir_entry(record_pos); dir= (buff + block_size - DIR_ENTRY_SIZE * rownr - @@ -1878,10 +2352,10 @@ my_bool _ma_update_block_record(MARIA_HA *info, MARIA_RECORD_POS record_pos, MARIA_BITMAP_BLOCK block; /* - We can fit the new row in the same page as the original head part - of the row + We can fit the new row in the same page as the original head part + of the row */ - block.org_bitmap_value= _ma_free_size_to_head_pattern(&info->s->bitmap, + block.org_bitmap_value= _ma_free_size_to_head_pattern(&share->bitmap, org_empty_size); offset= uint2korr(dir); length= uint2korr(dir + 2); @@ -1893,13 +2367,13 @@ my_bool _ma_update_block_record(MARIA_HA *info, MARIA_RECORD_POS record_pos, empty= start_of_next_entry(dir) - (offset + length); if (new_row->total_length > length + empty) { - compact_page(buff, info->s->block_size, rownr, 1); + compact_page(buff, share->block_size, rownr, 1); org_empty_size= 0; length= uint2korr(dir + 2); } } row_pos.buff= buff; - row_pos.offset= rownr; + row_pos.rownr= rownr; row_pos.empty_space= org_empty_size + length; row_pos.dir= dir; row_pos.data= buff + uint2korr(dir); @@ -1912,10 +2386,11 @@ my_bool _ma_update_block_record(MARIA_HA *info, MARIA_RECORD_POS record_pos, block.empty_space= row_pos.empty_space; /* Update cur_row, if someone calls update at once again */ cur_row->head_length= new_row->total_length; - if (_ma_bitmap_free_full_pages(info, cur_row->extents, - cur_row->extents_count)) - DBUG_RETURN(1); - DBUG_RETURN(write_block_record(info, record, new_row, blocks, &row_pos)); + + if (free_full_pages(info, cur_row)) + goto err; + DBUG_RETURN(write_block_record(info, oldrec, record, new_row, blocks, + 1, &row_pos)); } /* Allocate all size in block for record @@ -1928,27 +2403,31 @@ my_bool _ma_update_block_record(MARIA_HA *info, MARIA_RECORD_POS record_pos, (new_row->total_length <= head_length && org_empty_size + head_length >= new_row->total_length))) { - compact_page(buff, info->s->block_size, rownr, 1); + compact_page(buff, share->block_size, rownr, 1); org_empty_size= 0; head_length= uint2korr(dir + 2); } /* Delete old row */ if (delete_tails(info, cur_row->tail_positions)) - DBUG_RETURN(1); - if (_ma_bitmap_free_full_pages(info, cur_row->extents, - cur_row->extents_count)) - DBUG_RETURN(1); + goto err; + if (free_full_pages(info, cur_row)) + goto err; if (_ma_bitmap_find_new_place(info, new_row, page, head_length, blocks)) - DBUG_RETURN(1); + goto err; row_pos.buff= buff; - row_pos.offset= rownr; + row_pos.rownr= rownr; row_pos.empty_space= org_empty_size + head_length; row_pos.dir= dir; row_pos.data= buff + uint2korr(dir); row_pos.length= head_length; - DBUG_RETURN(write_block_record(info, record, new_row, blocks, &row_pos)); + DBUG_RETURN(write_block_record(info, oldrec, record, new_row, blocks, 1, + &row_pos)); + +err: + _ma_unpin_all_pages(info, 0); + DBUG_RETURN(1); } @@ -1977,6 +2456,8 @@ static my_bool delete_head_or_tail(MARIA_HA *info, uint number_of_records, empty_space, length; uint block_size= share->block_size; byte *buff, *dir; + LSN lsn; + MARIA_PINNED_PAGE page_link; DBUG_ENTER("delete_head_or_tail"); info->keyread_buff_used= 1; @@ -1984,9 +2465,11 @@ static my_bool delete_head_or_tail(MARIA_HA *info, if (!(buff= pagecache_read(share->pagecache, &info->dfile, page, 0, info->keyread_buff, - PAGECACHE_PLAIN_PAGE, - PAGECACHE_LOCK_LEFT_UNLOCKED, 0))) + info->s->page_type, + PAGECACHE_LOCK_WRITE, &page_link.link))) DBUG_RETURN(1); + page_link.unlock= PAGECACHE_LOCK_WRITE_UNLOCK; + push_dynamic(&info->pinned_pages, (void*) &page_link); number_of_records= (uint) ((uchar *) buff)[DIR_COUNT_OFFSET]; #ifdef SANITY_CHECKS @@ -2021,19 +2504,60 @@ static my_bool delete_head_or_tail(MARIA_HA *info, empty_space+= length; if (number_of_records != 0) { + uchar log_data[FILEID_STORE_SIZE + PAGE_STORE_SIZE + DIRPOS_STORE_SIZE]; + LEX_STRING log_array[TRANSLOG_INTERNAL_PARTS + 1]; + + /* Update directory */ int2store(buff + EMPTY_SPACE_OFFSET, empty_space); buff[PAGE_TYPE_OFFSET]|= (byte) PAGE_CAN_BE_COMPACTED; DBUG_ASSERT(share->pagecache->block_size == block_size); + + /* Log REDO data */ + fileid_store(log_data, info->dfile.file); + page_store(log_data+ FILEID_STORE_SIZE, page); + dirpos_store(log_data+ FILEID_STORE_SIZE + PAGE_STORE_SIZE, + record_number); + + log_array[TRANSLOG_INTERNAL_PARTS + 0].str= (char*) log_data; + log_array[TRANSLOG_INTERNAL_PARTS + 0].length= sizeof(log_data); + if (translog_write_record(&lsn, + (head ? LOGREC_REDO_PURGE_ROW_HEAD : + LOGREC_REDO_PURGE_ROW_TAIL), + info->trn->short_id, NULL, share, + sizeof(log_data), TRANSLOG_INTERNAL_PARTS + 1, + log_array)) + DBUG_RETURN(1); if (pagecache_write(share->pagecache, &info->dfile, page, 0, - buff, PAGECACHE_PLAIN_PAGE, - PAGECACHE_LOCK_LEFT_UNLOCKED, - PAGECACHE_PIN_LEFT_UNPINNED, - PAGECACHE_WRITE_DELAY, 0)) + buff, share->page_type, + PAGECACHE_LOCK_WRITE_TO_READ, + PAGECACHE_PIN_LEFT_PINNED, + PAGECACHE_WRITE_DELAY, &page_link.link)) DBUG_RETURN(1); + + /* Change the lock used when we read the page */ + page_link.unlock= PAGECACHE_LOCK_READ_UNLOCK; + set_dynamic(&info->pinned_pages, (void*) &page_link, + info->pinned_pages.elements-1); } else { + uchar log_data[FILEID_STORE_SIZE + PAGERANGE_STORE_SIZE + + PAGE_STORE_SIZE + PAGERANGE_STORE_SIZE]; + LEX_STRING log_array[TRANSLOG_INTERNAL_PARTS + 1]; + + fileid_store(log_data, info->dfile.file); + pagerange_store(log_data + FILEID_STORE_SIZE, 1); + page_store(log_data+ FILEID_STORE_SIZE + PAGERANGE_STORE_SIZE, page); + pagerange_store(log_data + FILEID_STORE_SIZE + PAGERANGE_STORE_SIZE + + PAGERANGE_STORE_SIZE, 1); + log_array[TRANSLOG_INTERNAL_PARTS + 0].str= (char*) log_data; + log_array[TRANSLOG_INTERNAL_PARTS + 0].length= sizeof(log_data); + if (translog_write_record(&lsn, LOGREC_REDO_PURGE_BLOCKS, + info->trn->short_id, NULL, share, + sizeof(log_data), TRANSLOG_INTERNAL_PARTS + 1, + log_array)) + DBUG_RETURN(1); DBUG_ASSERT(empty_space >= info->s->bitmap.sizes[0]); } DBUG_PRINT("info", ("empty_space: %u", empty_space)); @@ -2081,18 +2605,58 @@ static my_bool delete_tails(MARIA_HA *info, MARIA_RECORD_POS *tails) for rows with many splits. */ -my_bool _ma_delete_block_record(MARIA_HA *info) +my_bool _ma_delete_block_record(MARIA_HA *info, const byte *record) { + ulonglong page; + uint record_number; DBUG_ENTER("_ma_delete_block_record"); - if (delete_head_or_tail(info, - ma_recordpos_to_page(info->cur_row.lastpos), - ma_recordpos_to_dir_entry(info->cur_row.lastpos), - 1) || + + page= ma_recordpos_to_page(info->cur_row.lastpos); + record_number= ma_recordpos_to_dir_entry(info->cur_row.lastpos); + + if (delete_head_or_tail(info, page, record_number, 1) || delete_tails(info, info->cur_row.tail_positions)) - DBUG_RETURN(1); + goto err; + info->s->state.split--; - DBUG_RETURN(_ma_bitmap_free_full_pages(info, info->cur_row.extents, - info->cur_row.extents_count)); + + if (info->cur_row.extents && free_full_pages(info, &info->cur_row)) + goto err; + + { + uchar log_data[LSN_STORE_SIZE + FILEID_STORE_SIZE + PAGE_STORE_SIZE + + DIR_COUNT_SIZE]; + size_t row_length; + uint row_parts_count; + + /* Write UNDO record */ + lsn_store(log_data, info->trn->undo_lsn); + fileid_store(log_data+ LSN_STORE_SIZE, info->dfile.file); + page_store(log_data+ LSN_STORE_SIZE + FILEID_STORE_SIZE, page); + dirpos_store(log_data+ LSN_STORE_SIZE + FILEID_STORE_SIZE + + PAGE_STORE_SIZE, record_number); + + info->log_row_parts[TRANSLOG_INTERNAL_PARTS].str= (char*) log_data; + info->log_row_parts[TRANSLOG_INTERNAL_PARTS].length= sizeof(log_data); + row_length= fill_insert_undo_parts(info, record, info->log_row_parts + + TRANSLOG_INTERNAL_PARTS + 1, + &row_parts_count); + + if (translog_write_record(&info->trn->undo_lsn, LOGREC_UNDO_ROW_DELETE, + info->trn->short_id, NULL, info->s, + sizeof(log_data) + row_length, + TRANSLOG_INTERNAL_PARTS + 1 + row_parts_count, + info->log_row_parts)) + goto err; + + } + + _ma_unpin_all_pages(info, info->trn->undo_lsn); + DBUG_RETURN(0); + +err: + _ma_unpin_all_pages(info, 0); + DBUG_RETURN(1); } @@ -2173,7 +2737,7 @@ static void init_extent(MARIA_EXTENT_CURSOR *extent, byte *extent_info, extent->extent= extent_info; extent->extent_count= extents; extent->page= uint5korr(extent_info); /* First extent */ - page_count= uint2korr(extent_info+5); + page_count= uint2korr(extent_info + ROW_EXTENT_PAGE_SIZE); extent->page_count= page_count & ~TAIL_BIT; extent->tail= page_count & TAIL_BIT; extent->tail_positions= tail_positions; @@ -2223,21 +2787,10 @@ static byte *read_next_extent(MARIA_HA *info, MARIA_EXTENT_CURSOR *extent, extent->tail != 0)); } - if (info->cur_row.empty_bits != info->cur_row.empty_bits_buffer) - { - /* - First read of extents: Move data from info->buff to - internals buffers. - */ - memcpy(info->cur_row.empty_bits_buffer, info->cur_row.empty_bits, - share->base.pack_bytes); - info->cur_row.empty_bits= info->cur_row.empty_bits_buffer; - } - DBUG_ASSERT(share->pagecache->block_size == share->block_size); if (!(buff= pagecache_read(share->pagecache, &info->dfile, extent->page, 0, - info->buff, PAGECACHE_PLAIN_PAGE, + info->buff, share->page_type, PAGECACHE_LOCK_LEFT_UNLOCKED, 0))) { /* check if we tried to read over end of file (ie: bad data in record) */ @@ -2309,6 +2862,19 @@ static my_bool read_long_data(MARIA_HA *info, byte *to, ulong length, DBUG_PRINT("enter", ("length: %lu", length)); DBUG_ASSERT(*data <= *end_of_data); + /* + Fields are never split in middle. This means that if length > rest-of-data + we should start reading from the next extent. The reason we may have + data left on the page is that there fixed part of the row was less than + min_row_length and in this case the head block was extended to + min_row_length. + + This may change in the future, which is why we have the loop written + the way it's written. + */ + if (length > (ulong) (*end_of_data - *data)) + *end_of_data= *data; + for(;;) { uint left_length; @@ -2347,7 +2913,7 @@ static my_bool read_long_data(MARIA_HA *info, byte *to, ulong length, cur_row.tail_positions is set to point to all tail blocks cur_row.extents points to extents data cur_row.extents_counts contains number of extents - cur_row.empty_bits points to empty bits part in read record + cur_row.empty_bits is set to empty bits cur_row.field_lengths contains packed length of all fields RETURN @@ -2415,6 +2981,7 @@ int _ma_read_block_record2(MARIA_HA *info, byte *record, if (share->base.max_field_lengths) { get_key_length(field_lengths, data); + info->cur_row.field_lengths_length= field_lengths; #ifdef SANITY_CHECKS if (field_lengths > share->base.max_field_lengths) goto err; @@ -2434,7 +3001,8 @@ int _ma_read_block_record2(MARIA_HA *info, byte *record, bzero(record + cur_null_bytes, (uint) (null_bytes - cur_null_bytes)); } data+= null_bytes; - info->cur_row.empty_bits= (byte*) data; /* Pointer to empty bitmask */ + /* We copy the empty bits to be able to use them for delete/update */ + memcpy(info->cur_row.empty_bits, data, share->base.pack_bytes); data+= share->base.pack_bytes; /* TODO: Use field offsets, instead of just skipping them */ @@ -2454,7 +3022,7 @@ int _ma_read_block_record2(MARIA_HA *info, byte *record, /* Data now points to start of fixed length field data that can't be null - or 'empty'. Note that these fields can't be split over blocks + or 'empty'. Note that these fields can't be split over blocks. */ for (column= share->columndef, end_column= column + share->base.fixed_not_null_fields; @@ -2661,7 +3229,7 @@ int _ma_read_block_record(MARIA_HA *info, byte *record, DBUG_ASSERT(info->s->pagecache->block_size == block_size); if (!(buff= pagecache_read(info->s->pagecache, &info->dfile, ma_recordpos_to_page(record_pos), 0, - info->buff, PAGECACHE_PLAIN_PAGE, + info->buff, info->s->page_type, PAGECACHE_LOCK_LEFT_UNLOCKED, 0))) DBUG_RETURN(1); DBUG_ASSERT((buff[PAGE_TYPE_OFFSET] & PAGE_TYPE_MASK) == HEAD_PAGE); @@ -2780,10 +3348,10 @@ void _ma_scan_end_block_record(MARIA_HA *info) use a variable in info->scan IMPLEMENTATION - Current code uses a lot of goto's to separate the different kind of - states we may be in. This gives us a minimum of executed if's for - the normal cases. I tried several different ways to code this, but - the current one was in the end the most readable and fastest. + Current code uses a lot of goto's to separate the different kind of + states we may be in. This gives us a minimum of executed if's for + the normal cases. I tried several different ways to code this, but + the current one was in the end the most readable and fastest. RETURN 0 ok @@ -2796,6 +3364,7 @@ int _ma_scan_block_record(MARIA_HA *info, byte *record, { uint block_size; my_off_t filepos; + MARIA_SHARE *share= info->s; DBUG_ENTER("_ma_scan_block_record"); restart_record_read: @@ -2823,7 +3392,7 @@ restart_record_read: info->scan.dir-= DIR_ENTRY_SIZE; /* Point to previous row */ #ifdef SANITY_CHECKS if (end_of_data > info->scan.dir_end || - offset < PAGE_HEADER_SIZE || length < info->s->base.min_block_length) + offset < PAGE_HEADER_SIZE || length < share->base.min_block_length) goto err; #endif DBUG_PRINT("info", ("rowid: %lu", (ulong) info->cur_row.lastpos)); @@ -2832,7 +3401,7 @@ restart_record_read: /* Find next head page in current bitmap */ restart_bitmap_scan: - block_size= info->s->block_size; + block_size= share->block_size; if (likely(info->scan.bitmap_pos < info->scan.bitmap_end)) { byte *data= info->scan.bitmap_pos; @@ -2856,10 +3425,10 @@ restart_bitmap_scan: page= (info->scan.bitmap_page + 1 + (data - info->scan.bitmap_buff) / 6 * 16 + bit_pos - 1); info->scan.row_base_page= ma_recordpos(page, 0); - if (!(pagecache_read(info->s->pagecache, + if (!(pagecache_read(share->pagecache, &info->dfile, page, 0, info->scan.page_buff, - PAGECACHE_PLAIN_PAGE, + share->page_type, PAGECACHE_LOCK_LEFT_UNLOCKED, 0))) DBUG_RETURN(my_errno); if (((info->scan.page_buff[PAGE_TYPE_OFFSET] & PAGE_TYPE_MASK) != @@ -2890,13 +3459,13 @@ restart_bitmap_scan: } /* Read next bitmap */ - info->scan.bitmap_page+= info->s->bitmap.pages_covered; + info->scan.bitmap_page+= share->bitmap.pages_covered; filepos= (my_off_t) info->scan.bitmap_page * block_size; if (unlikely(filepos >= info->state->data_file_length)) { DBUG_RETURN((my_errno= HA_ERR_END_OF_FILE)); } - if (!(pagecache_read(info->s->pagecache, &info->dfile, + if (!(pagecache_read(share->pagecache, &info->dfile, info->scan.bitmap_page, 0, info->scan.bitmap_buff, PAGECACHE_PLAIN_PAGE, PAGECACHE_LOCK_LEFT_UNLOCKED, 0))) @@ -2960,3 +3529,438 @@ static void _ma_print_directory(byte *buff, uint block_size) } #endif /* DBUG_OFF */ + +/* + Store an integer with simple packing + + SYNOPSIS + ma_store_integer() + to Store the packed integer here + nr Integer to store + + NOTES + This is mostly used to store field numbers and lengths of strings. + We have to cast the result for the LL() becasue of a bug in Forte CC + compiler. + + Packing used is: + nr < 251 is stored as is (in 1 byte) + Numbers that require 1-4 bytes are stored as char(250+byte_length), data + Bigger numbers are stored as 255, data as ulonglong (not yet done). + + RETURN + Position in 'to' after the packed length +*/ + +uchar *ma_store_length(uchar *to, ulong nr) +{ + if (nr < 251) + { + *to=(uchar) nr; + return to+1; + } + if (nr < 65536) + { + if (nr <= 255) + { + to[0]= (uchar) 251; + to[1]= (uchar) nr; + return to+2; + } + to[0]= (uchar) 252; + int2store(to+1, nr); + return to+3; + } + if (nr < 16777216) + { + *to++= (uchar) 253; + int3store(to, nr); + return to+3; + } + *to++= (uchar) 254; + int4store(to, nr); + return to+4; +} + + +/* Calculate how many bytes needed to store a number */ + +uint ma_calc_length_for_store_length(ulong nr) +{ + if (nr < 251) + return 1; + if (nr < 65536) + { + if (nr <= 255) + return 2; + return 3; + } + if (nr < 16777216) + return 4; + return 5; +} + + +/* + Fill array with pointers to field parts to be stored in log for insert + + SYNOPSIS + fill_insert_undo_parts() + info Maria handler + record Inserted row + log_parts Store pointers to changed memory areas here + log_parts_count See RETURN + + NOTES + We have information in info->cur_row about the read row. + + RETURN + length of data in log_parts. + log_parts_count contains number of used log_parts +*/ + +static size_t fill_insert_undo_parts(MARIA_HA *info, const byte *record, + LEX_STRING *log_parts, + uint *log_parts_count) +{ + MARIA_SHARE *share= info->s; + MARIA_COLUMNDEF *column, *end_column; + uchar *field_lengths= info->cur_row.field_lengths; + size_t row_length; + MARIA_ROW *cur_row= &info->cur_row; + LEX_STRING *start_log_parts; + DBUG_ENTER("fill_insert_undo_parts"); + + start_log_parts= log_parts; + + /* Store null bits */ + log_parts->str= (char*) record; + log_parts->length= share->base.null_bytes; + row_length= log_parts->length; + log_parts++; + + /* Stored bitmap over packed (zero length or all-zero fields) */ + start_log_parts= log_parts; + log_parts->str= info->cur_row.empty_bits; + log_parts->length= share->base.pack_bytes; + row_length+= log_parts->length; + log_parts++; + + if (share->base.max_field_lengths) + { + /* Store field lenghts, with a prefix of number of bytes */ + log_parts->str= field_lengths-2; + log_parts->length= info->cur_row.field_lengths_length+2; + int2store(log_parts->str, info->cur_row.field_lengths_length); + row_length+= log_parts->length; + log_parts++; + } + + /* Handle constant length fields that are always present */ + for (column= share->columndef, + end_column= column+ share->base.fixed_not_null_fields; + column < end_column; + column++) + { + log_parts->str= (char*) record + column->offset; + log_parts->length= column->length; + row_length+= log_parts->length; + log_parts++; + } + + /* Handle NULL fields and CHAR/VARCHAR fields */ + for (end_column= share->columndef + share->base.fields - share->base.blobs; + column < end_column; + column++) + { + const uchar *column_pos; + size_t column_length; + if ((record[column->null_pos] & column->null_bit) || + cur_row->empty_bits[column->empty_pos] & column->empty_bit) + continue; + + column_pos= record+ column->offset; + column_length= column->length; + + switch ((enum en_fieldtype) column->type) { + case FIELD_CHECK: + case FIELD_NORMAL: /* Fixed length field */ + case FIELD_ZERO: + case FIELD_SKIP_PRESPACE: /* Not packed */ + case FIELD_SKIP_ZERO: /* Fixed length field */ + break; + case FIELD_SKIP_ENDSPACE: /* CHAR */ + { + if (column->length <= 255) + column_length= *field_lengths++; + else + { + column_length= uint2korr(field_lengths); + field_lengths+= 2; + } + break; + } + case FIELD_VARCHAR: + { + if (column->length <= 256) + { + column_length= *field_lengths; + field_lengths++; + } + else + { + column_length= uint2korr(field_lengths); + field_lengths+= 2; + } + break; + } + default: + DBUG_ASSERT(0); + } + log_parts->str= (char*) column_pos; + log_parts->length= column_length; + row_length+= log_parts->length; + log_parts++; + } + + /* Add blobs */ + for (end_column+= share->base.blobs; column < end_column; column++) + { + const byte *field_pos= record + column->offset; + uint size_length= column->length - portable_sizeof_char_ptr; + ulong blob_length= _ma_calc_blob_length(size_length, field_pos); + + /* + We don't have to check for null, as blob_length is guranteed to be 0 + if the blob is null + */ + if (blob_length) + { + char *blob_pos; + memcpy_fixed((byte*) &blob_pos, record + column->offset + size_length, + sizeof(blob_pos)); + log_parts->str= blob_pos; + log_parts->length= blob_length; + row_length+= log_parts->length; + log_parts++; + } + } + *log_parts_count= (log_parts - start_log_parts); + DBUG_RETURN(row_length); +} + + +/* + Fill array with pointers to field parts to be stored in log for update + + SYNOPSIS + fill_update_undo_parts() + info Maria handler + oldrec Original row + newrec New row + log_parts Store pointers to changed memory areas here + log_parts_count See RETURN + + IMPLEMENTATION + Format of undo record: + + Fields are stored in same order as the field array. + + Number of changed fields (packed) + + For each changed field + Fieldnumber (packed) + Length, if variable length field (packed) + + For each changed field + Data + + Packing is using ma_store_integer() + + The reason we store field numbers & length separated from data (ie, not + after each other) is to get better cpu caching when we loop over + fields (as we probably don't have to access data for each field when we + want to read and old row through the undo log record). + + As a special case, we use '255' for the field number of the null bitmap. + + RETURN + length of data in log_parts. + log_parts_count contains number of used log_parts +*/ + +static size_t fill_update_undo_parts(MARIA_HA *info, const byte *oldrec, + const byte *newrec, + LEX_STRING *log_parts, + uint *log_parts_count) +{ + MARIA_SHARE *share= info->s; + MARIA_COLUMNDEF *column, *end_column; + MARIA_ROW *old_row= &info->cur_row, *new_row= &info->new_row; + uchar *field_data, *start_field_data; + uchar *old_field_lengths= old_row->field_lengths; + uchar *new_field_lengths= new_row->field_lengths; + size_t row_length; + uint field_count= 0; + LEX_STRING *start_log_parts; + my_bool new_column_is_empty; + DBUG_ENTER("fill_update_undo_parts"); + + start_log_parts= log_parts; + + /* + First log part is for number of fields, field numbers and lengths + The +4 is to reserve place for the number of changed fields. + */ + start_field_data= field_data= info->update_field_data + 4; + log_parts++; + + if (memcmp(oldrec, newrec, share->base.null_bytes)) + { + /* Store changed null bits */ + *field_data++= (uchar) 255; /* Special case */ + field_count++; + log_parts->str= (char*) oldrec; + log_parts->length= share->base.null_bytes; + row_length= log_parts->length; + log_parts++; + } + + /* Handle constant length fields */ + for (column= share->columndef, + end_column= column+ share->base.fixed_not_null_fields; + column < end_column; + column++) + { + if (memcmp(oldrec + column->offset, newrec + column->offset, + column->length)) + { + field_data= ma_store_length(field_data, + (uint) (column - share->columndef)); + field_count++; + log_parts->str= (char*) oldrec + column->offset; + log_parts->length= column->length; + row_length+= log_parts->length; + log_parts++; + } + } + + /* Handle the rest: NULL fields and CHAR/VARCHAR fields and BLOB's */ + for (end_column= share->columndef + share->base.fields; + column < end_column; + column++) + { + const uchar *new_column_pos, *old_column_pos; + size_t new_column_length, old_column_length; + + /* First check if old column is null or empty */ + if (oldrec[column->null_pos] & column->null_bit) + { + /* + It's safe to skip this one as either the new column is also null + (no change) or the new_column is not null, in which case the null-bit + maps differed and we have already stored the null bitmap. + */ + continue; + } + if (old_row->empty_bits[column->empty_pos] & column->empty_bit) + { + if (new_row->empty_bits[column->empty_pos] & column->empty_bit) + continue; /* Both are empty; skip */ + + /* Store null length column */ + field_data= ma_store_length(field_data, + (uint) (column - share->columndef)); + field_data= ma_store_length(field_data, 0); + field_count++; + continue; + } + /* + Remember if the 'new' value is empty (as in this case we must always + log the original value + */ + new_column_is_empty= ((newrec[column->null_pos] & column->null_bit) || + (new_row->empty_bits[column->empty_pos] & + column->empty_bit)); + + old_column_pos= oldrec + column->offset; + new_column_pos= newrec + column->offset; + old_column_length= new_column_length= column->length; + + switch ((enum en_fieldtype) column->type) { + case FIELD_CHECK: + case FIELD_NORMAL: /* Fixed length field */ + case FIELD_ZERO: + case FIELD_SKIP_PRESPACE: /* Not packed */ + case FIELD_SKIP_ZERO: /* Fixed length field */ + break; + case FIELD_VARCHAR: + new_column_length--; /* Skip length prefix */ + /* Fall through */ + case FIELD_SKIP_ENDSPACE: /* CHAR */ + { + if (new_column_length <= 255) + { + old_column_length= *old_field_lengths++; + if (!new_column_is_empty) + new_column_length= *new_field_lengths++; + } + else + { + old_column_length= uint2korr(old_field_lengths); + old_field_lengths+= 2; + if (!new_column_is_empty) + { + new_column_length= uint2korr(new_field_lengths); + new_field_lengths+= 2; + } + } + break; + } + case FIELD_BLOB: + { + uint size_length= column->length - portable_sizeof_char_ptr; + old_column_length= _ma_calc_blob_length(size_length, old_column_pos); + memcpy_fixed((byte*) &old_column_pos, + oldrec + column->offset + size_length, + sizeof(old_column_pos)); + if (!new_column_is_empty) + { + new_column_length= _ma_calc_blob_length(size_length, new_column_pos); + memcpy_fixed((byte*) &new_column_pos, + newrec + column->offset + size_length, + sizeof(old_column_pos)); + } + break; + } + default: + DBUG_ASSERT(0); + } + + if (new_column_is_empty || new_column_length != old_column_length || + memcmp(old_column_pos, new_column_pos, new_column_length)) + { + field_data= ma_store_length(field_data, + (uint) (column - share->columndef)); + field_data= ma_store_length(field_data, old_column_length); + field_count++; + + log_parts->str= (char*) old_column_pos; + log_parts->length= old_column_length; + row_length+= log_parts->length; + log_parts++; + } + } + + *log_parts_count= (log_parts - start_log_parts); + + /* Store number of fields before the field/field_lengths */ + start_log_parts->str= ((char*) + (start_field_data - + ma_calc_length_for_store_length(field_count))); + ma_store_length(start_log_parts->str, field_count); + start_log_parts->length= (size_t) ((char*) field_data - + start_log_parts->str); + row_length+= start_log_parts->length; + DBUG_RETURN(row_length); +} diff --git a/storage/maria/ma_blockrec.h b/storage/maria/ma_blockrec.h index 9e251a8c59d..f45250ff39c 100644 --- a/storage/maria/ma_blockrec.h +++ b/storage/maria/ma_blockrec.h @@ -102,6 +102,9 @@ enum en_page_type { UNALLOCATED_PAGE, HEAD_PAGE, TAIL_PAGE, BLOB_PAGE, MAX_PAGE_ */ #define MAX_TAIL_SIZE(block_size) ((block_size) *3 / 4) +/* Don't allocate memory for too many row extents on the stack */ +#define ROW_EXTENTS_ON_STACK 32 + extern uchar maria_bitmap_marker[2]; /* Functions to convert MARIA_RECORD_POS to/from page:offset */ @@ -130,8 +133,8 @@ my_bool _ma_init_block_record(MARIA_HA *info); void _ma_end_block_record(MARIA_HA *info); my_bool _ma_update_block_record(MARIA_HA *info, MARIA_RECORD_POS pos, - const byte *record); -my_bool _ma_delete_block_record(MARIA_HA *info); + const byte *oldrec, const byte *newrec); +my_bool _ma_delete_block_record(MARIA_HA *info, const byte *record); int _ma_read_block_record(MARIA_HA *info, byte *record, MARIA_RECORD_POS record_pos); int _ma_read_block_record2(MARIA_HA *info, byte *record, diff --git a/storage/maria/ma_check.c b/storage/maria/ma_check.c index 4087f12ba43..82ad67c2452 100644 --- a/storage/maria/ma_check.c +++ b/storage/maria/ma_check.c @@ -4507,7 +4507,8 @@ static int sort_delete_record(MARIA_SORT_PARAM *sort_param) if (sort_param->calc_checksum) param->glob_crc-=(*info->s->calc_checksum)(info, sort_param->record); } - error=flush_io_cache(&info->rec_cache) || (*info->s->delete_record)(info); + error= (flush_io_cache(&info->rec_cache) || + (*info->s->delete_record)(info, sort_param->record)); info->dfile.file= old_file; /* restore actual value */ info->state->records--; DBUG_RETURN(error); diff --git a/storage/maria/ma_control_file.h b/storage/maria/ma_control_file.h index 159cd15b3d6..4728d719b2f 100644 --- a/storage/maria/ma_control_file.h +++ b/storage/maria/ma_control_file.h @@ -56,6 +56,9 @@ typedef enum enum_control_file_error { #define CONTROL_FILE_UPDATE_ONLY_LSN 1 #define CONTROL_FILE_UPDATE_ONLY_LOGNO 2 +#ifdef __cplusplus +extern "C" { +#endif /* Looks for the control file. If absent, it's a fresh start, create file. @@ -74,3 +77,7 @@ int ma_control_file_write_and_force(const LSN checkpoint_lsn, uint32 logno, /* Free resources taken by control file subsystem */ int ma_control_file_end(); + +#ifdef __cplusplus +} +#endif diff --git a/storage/maria/ma_delete.c b/storage/maria/ma_delete.c index f7b11cb6f48..067dd060a92 100644 --- a/storage/maria/ma_delete.c +++ b/storage/maria/ma_delete.c @@ -89,7 +89,7 @@ int maria_delete(MARIA_HA *info,const byte *record) } } - if ((*share->delete_record)(info)) + if ((*share->delete_record)(info, record)) goto err; /* Remove record from database */ /* diff --git a/storage/maria/ma_dynrec.c b/storage/maria/ma_dynrec.c index d6f7309cafa..ebf84032106 100644 --- a/storage/maria/ma_dynrec.c +++ b/storage/maria/ma_dynrec.c @@ -237,6 +237,7 @@ my_bool _ma_write_dynamic_record(MARIA_HA *info, const byte *record) } my_bool _ma_update_dynamic_record(MARIA_HA *info, MARIA_RECORD_POS pos, + const byte *oldrec __attribute__ ((unused)), const byte *record) { uint length= _ma_rec_pack(info, info->rec_buff + MARIA_REC_BUFF_OFFSET, @@ -277,6 +278,7 @@ my_bool _ma_write_blob_record(MARIA_HA *info, const byte *record) my_bool _ma_update_blob_record(MARIA_HA *info, MARIA_RECORD_POS pos, + const byte *oldrec __attribute__ ((unused)), const byte *record) { byte *rec_buff; @@ -309,7 +311,8 @@ my_bool _ma_update_blob_record(MARIA_HA *info, MARIA_RECORD_POS pos, } -my_bool _ma_delete_dynamic_record(MARIA_HA *info) +my_bool _ma_delete_dynamic_record(MARIA_HA *info, + const byte *record __attribute__ ((unused))) { return delete_dynamic_record(info, info->cur_row.lastpos, 0); } @@ -1371,7 +1374,7 @@ int _ma_read_dynamic_record(MARIA_HA *info, byte *buf, MARIA_RECORD_POS filepos) { int block_of_record; - uint b_type,left_length; + uint b_type; MARIA_BLOCK_INFO block_info; File file; DBUG_ENTER("_ma_read_dynamic_record"); diff --git a/storage/maria/ma_init.c b/storage/maria/ma_init.c index 271eac6c6d1..8f7cdf291ae 100644 --- a/storage/maria/ma_init.c +++ b/storage/maria/ma_init.c @@ -43,6 +43,7 @@ int maria_init(void) maria_inited= TRUE; pthread_mutex_init(&THR_LOCK_maria,MY_MUTEX_INIT_SLOW); _ma_init_block_record_data(); + loghandler_init(); } return 0; } diff --git a/storage/maria/ma_loghandler.c b/storage/maria/ma_loghandler.c index 198be85ab8c..d16595be24e 100644 --- a/storage/maria/ma_loghandler.c +++ b/storage/maria/ma_loghandler.c @@ -1,4 +1,5 @@ #include "maria_def.h" +#include "ma_blockrec.h" /* number of opened log files in the pagecache (should be at least 2) */ #define OPENED_FILES_NUM 3 @@ -34,14 +35,6 @@ -/* record part descriptor */ -struct st_translog_part -{ - translog_size_t len; - byte *buff; -}; - - /* record parts descriptor */ struct st_translog_parts { @@ -49,10 +42,12 @@ struct st_translog_parts translog_size_t record_length; /* full record length with chunk headers */ translog_size_t total_record_length; - /* array of parts (st_translog_part) */ - DYNAMIC_ARRAY parts; /* current part index */ uint current; + /* total number of elements in parts */ + uint elements; + /* array of parts (LEX_STRING) */ + LEX_STRING *parts; }; /* log write buffer descriptor */ @@ -175,7 +170,7 @@ enum record_class #define TRANSLOG_CLSN_MAX_LEN 5 /* Maximum length of compressed LSN */ typedef my_bool(*prewrite_rec_hook) (enum translog_record_type type, - void *tcb, + void *tcb, struct st_maria_share *share, struct st_translog_parts *parts); typedef my_bool(*inwrite_rec_hook) (enum translog_record_type type, @@ -213,139 +208,209 @@ struct st_log_record_type_descriptor }; -static struct st_log_record_type_descriptor - log_record_type_descriptor[LOGREC_NUMBER_OF_TYPES]= +/* + Initialize log_record_type_descriptors + + NOTE that after first public Maria release, these can NOT be changed +*/ + +typedef struct st_log_record_type_descriptor LOG_DESC; + +static LOG_DESC log_record_type_descriptor[LOGREC_NUMBER_OF_TYPES]; + +static LOG_DESC INIT_LOGREC_RESERVED_FOR_CHUNKS23= +{ LOGRECTYPE_NOT_ALLOWED, 0, 0, NULL, NULL, NULL, 0 }; + +static LOG_DESC INIT_LOGREC_REDO_INSERT_ROW_HEAD= +{LOGRECTYPE_VARIABLE_LENGTH, 0, + FILEID_STORE_SIZE + PAGE_STORE_SIZE + DIRPOS_STORE_SIZE, NULL, NULL, NULL, 0}; + +static LOG_DESC INIT_LOGREC_REDO_INSERT_ROW_TAIL= +{LOGRECTYPE_VARIABLE_LENGTH, 0, 9, NULL, NULL, NULL, 0}; + +static LOG_DESC INIT_LOGREC_REDO_INSERT_ROW_BLOB= +{LOGRECTYPE_VARIABLE_LENGTH, 0, 8, NULL, NULL, NULL, 0}; + +/*QQQ:TODO:header???*/ +static LOG_DESC INIT_LOGREC_REDO_INSERT_ROW_BLOBS= +{LOGRECTYPE_VARIABLE_LENGTH, 0, 0, NULL, NULL, NULL, 0}; + +static LOG_DESC INIT_LOGREC_REDO_PURGE_ROW_HEAD= +{LOGRECTYPE_FIXEDLENGTH, + FILEID_STORE_SIZE + PAGE_STORE_SIZE + DIRPOS_STORE_SIZE, + FILEID_STORE_SIZE + PAGE_STORE_SIZE + DIRPOS_STORE_SIZE, + NULL, NULL, NULL, 0}; + +static LOG_DESC INIT_LOGREC_REDO_PURGE_ROW_TAIL= +{LOGRECTYPE_FIXEDLENGTH, + FILEID_STORE_SIZE + PAGE_STORE_SIZE + DIRPOS_STORE_SIZE, + FILEID_STORE_SIZE + PAGE_STORE_SIZE + DIRPOS_STORE_SIZE, + NULL, NULL, NULL, 0}; + +/* QQQ: TODO: variable and fixed size??? */ +static LOG_DESC INIT_LOGREC_REDO_PURGE_BLOCKS= +{LOGRECTYPE_VARIABLE_LENGTH, + FILEID_STORE_SIZE + PAGERANGE_STORE_SIZE + PAGE_STORE_SIZE + + PAGERANGE_STORE_SIZE, + FILEID_STORE_SIZE + PAGERANGE_STORE_SIZE + PAGE_STORE_SIZE + + PAGERANGE_STORE_SIZE, + NULL, NULL, NULL, 0}; + +static LOG_DESC INIT_LOGREC_REDO_DELETE_ROW= +{LOGRECTYPE_FIXEDLENGTH, 16, 16, NULL, NULL, NULL, 0}; + +static LOG_DESC INIT_LOGREC_REDO_UPDATE_ROW_HEAD= +{LOGRECTYPE_VARIABLE_LENGTH, 0, 9, NULL, NULL, NULL, 0}; + +static LOG_DESC INIT_LOGREC_REDO_INDEX= +{LOGRECTYPE_VARIABLE_LENGTH, 0, 9, NULL, NULL, NULL, 0}; + +static LOG_DESC INIT_LOGREC_REDO_UNDELETE_ROW= +{LOGRECTYPE_FIXEDLENGTH, 16, 16, NULL, NULL, NULL, 0}; + +static LOG_DESC INIT_LOGREC_CLR_END= +{LOGRECTYPE_PSEUDOFIXEDLENGTH, 5, 5, NULL, NULL, NULL, 1}; + +static LOG_DESC INIT_LOGREC_PURGE_END= +{LOGRECTYPE_PSEUDOFIXEDLENGTH, 5, 5, NULL, NULL, NULL, 1}; + +static LOG_DESC INIT_LOGREC_UNDO_ROW_INSERT= +{LOGRECTYPE_FIXEDLENGTH, + LSN_STORE_SIZE + FILEID_STORE_SIZE + PAGE_STORE_SIZE + DIRPOS_STORE_SIZE, + LSN_STORE_SIZE + FILEID_STORE_SIZE + PAGE_STORE_SIZE + DIRPOS_STORE_SIZE, + NULL, NULL, NULL, 0}; + +static LOG_DESC INIT_LOGREC_UNDO_ROW_DELETE= +{LOGRECTYPE_VARIABLE_LENGTH, 0, + LSN_STORE_SIZE + FILEID_STORE_SIZE + PAGE_STORE_SIZE + DIRPOS_STORE_SIZE, + NULL, NULL, NULL, 0}; + +static LOG_DESC INIT_LOGREC_UNDO_ROW_UPDATE= +{LOGRECTYPE_VARIABLE_LENGTH, 0, + LSN_STORE_SIZE + FILEID_STORE_SIZE + PAGE_STORE_SIZE + DIRPOS_STORE_SIZE, + NULL, NULL, NULL, 2}; + +static LOG_DESC INIT_LOGREC_UNDO_ROW_PURGE= +{LOGRECTYPE_PSEUDOFIXEDLENGTH, LSN_STORE_SIZE, LSN_STORE_SIZE, + NULL, NULL, NULL, 1}; + +static LOG_DESC INIT_LOGREC_UNDO_KEY_INSERT= +{LOGRECTYPE_VARIABLE_LENGTH, 0, 10, NULL, NULL, NULL, 1}; + +static LOG_DESC INIT_LOGREC_UNDO_KEY_DELETE= +{LOGRECTYPE_VARIABLE_LENGTH, 0, 15, NULL, NULL, NULL, 0}; + +static LOG_DESC INIT_LOGREC_PREPARE= +{LOGRECTYPE_VARIABLE_LENGTH, 0, 0, NULL, NULL, NULL, 0}; + +static LOG_DESC INIT_LOGREC_PREPARE_WITH_UNDO_PURGE= +{LOGRECTYPE_VARIABLE_LENGTH, 0, 5, NULL, NULL, NULL, 1}; + +static LOG_DESC INIT_LOGREC_COMMIT= +{LOGRECTYPE_FIXEDLENGTH, 0, 0, NULL, NULL, NULL, 0}; + +static LOG_DESC INIT_LOGREC_COMMIT_WITH_UNDO_PURGE= +{LOGRECTYPE_PSEUDOFIXEDLENGTH, 5, 5, NULL, NULL, NULL, 1}; + +static LOG_DESC INIT_LOGREC_CHECKPOINT_PAGE= +{LOGRECTYPE_VARIABLE_LENGTH, 0, 6, NULL, NULL, NULL, 0}; + +static LOG_DESC INIT_LOGREC_CHECKPOINT_TRAN= +{LOGRECTYPE_VARIABLE_LENGTH, 0, 0, NULL, NULL, NULL, 0}; + +static LOG_DESC INIT_LOGREC_CHECKPOINT_TABL= +{LOGRECTYPE_VARIABLE_LENGTH, 0, 8, NULL, NULL, NULL, 0}; + +static LOG_DESC INIT_LOGREC_REDO_CREATE_TABLE= +{LOGRECTYPE_VARIABLE_LENGTH, 0, 0, NULL, NULL, NULL, 0}; + +static LOG_DESC INIT_LOGREC_REDO_RENAME_TABLE= +{LOGRECTYPE_VARIABLE_LENGTH, 0, 0, NULL, NULL, NULL, 0}; + +static LOG_DESC INIT_LOGREC_REDO_DROP_TABLE= +{LOGRECTYPE_VARIABLE_LENGTH, 0, 0, NULL, NULL, NULL, 0}; + +static LOG_DESC INIT_LOGREC_REDO_TRUNCATE_TABLE= +{LOGRECTYPE_VARIABLE_LENGTH, 0, 0, NULL, NULL, NULL, 0}; + +static LOG_DESC INIT_LOGREC_FILE_ID= +{LOGRECTYPE_VARIABLE_LENGTH, 0, 4, NULL, NULL, NULL, 0}; + +static LOG_DESC INIT_LOGREC_LONG_TRANSACTION_ID= +{LOGRECTYPE_FIXEDLENGTH, 6, 6, NULL, NULL, NULL, 0}; + + +void loghandler_init() { - /*LOGREC_RESERVED_FOR_CHUNKS23= 0 */ - {LOGRECTYPE_NOT_ALLOWED, 0, 0, NULL, NULL, NULL, 0}, - /*LOGREC_REDO_INSERT_ROW_HEAD= 1 */ - {LOGRECTYPE_VARIABLE_LENGTH, 0, 9, NULL, NULL, NULL, 0}, - /*LOGREC_REDO_INSERT_ROW_TAIL= 2 */ - {LOGRECTYPE_VARIABLE_LENGTH, 0, 9, NULL, NULL, NULL, 0}, - /*LOGREC_REDO_INSERT_ROW_BLOB= 3 */ - {LOGRECTYPE_VARIABLE_LENGTH, 0, 8, NULL, NULL, NULL, 0}, - /*LOGREC_REDO_INSERT_ROW_BLOBS= 4 */ - {LOGRECTYPE_VARIABLE_LENGTH, 0, 10, NULL, NULL, NULL, 0}, - /*LOGREC_REDO_PURGE_ROW= 5 */ - {LOGRECTYPE_FIXEDLENGTH, 9, 9, NULL, NULL, NULL, 0}, - /*LOGREC_REDO_PURGE_BLOCKS= 6 */ - {LOGRECTYPE_FIXEDLENGTH, 10, 10, NULL, NULL, NULL, 0}, - /*LOGREC_REDO_DELETE_ROW= 7 */ - {LOGRECTYPE_FIXEDLENGTH, 16, 16, NULL, NULL, NULL, 0}, - /*LOGREC_REDO_UPDATE_ROW_HEAD= 8 */ - {LOGRECTYPE_VARIABLE_LENGTH, 0, 9, NULL, NULL, NULL, 0}, - /*LOGREC_REDO_INDEX= 9 */ - {LOGRECTYPE_VARIABLE_LENGTH, 0, 9, NULL, NULL, NULL, 0}, - /*LOGREC_REDO_UNDELETE_ROW= 10 */ - {LOGRECTYPE_FIXEDLENGTH, 16, 16, NULL, NULL, NULL, 0}, - /*LOGREC_CLR_END= 11 */ - {LOGRECTYPE_PSEUDOFIXEDLENGTH, 5, 5, NULL, NULL, NULL, 1}, - /*LOGREC_PURGE_END= 12 */ - {LOGRECTYPE_PSEUDOFIXEDLENGTH, 5, 5, NULL, NULL, NULL, 1}, - /*LOGREC_UNDO_ROW_INSERT= 13 */ - {LOGRECTYPE_PSEUDOFIXEDLENGTH, 14, 14, NULL, NULL, NULL, 1}, - /*LOGREC_UNDO_ROW_DELETE= 14 */ - {LOGRECTYPE_PSEUDOFIXEDLENGTH, 19, 19, NULL, NULL, NULL, 2}, - /*LOGREC_UNDO_ROW_UPDATE= 15 */ - {LOGRECTYPE_VARIABLE_LENGTH, 0, 14, NULL, NULL, NULL, 2}, - /*LOGREC_UNDO_KEY_INSERT= 16 */ - {LOGRECTYPE_VARIABLE_LENGTH, 0, 10, NULL, NULL, NULL, 1}, - /*LOGREC_UNDO_KEY_DELETE= 17 */ - {LOGRECTYPE_VARIABLE_LENGTH, 0, 15, NULL, NULL, NULL, 2}, - /*LOGREC_PREPARE= 18 */ - {LOGRECTYPE_VARIABLE_LENGTH, 0, 0, NULL, NULL, NULL, 0}, - /*LOGREC_PREPARE_WITH_UNDO_PURGE= 19 */ - {LOGRECTYPE_VARIABLE_LENGTH, 0, 5, NULL, NULL, NULL, 1}, - /*LOGREC_COMMIT= 20 */ - {LOGRECTYPE_FIXEDLENGTH, 0, 0, NULL, NULL, NULL, 0}, - /*LOGREC_COMMIT_WITH_UNDO_PURGE= 21 */ - {LOGRECTYPE_PSEUDOFIXEDLENGTH, 5, 5, NULL, NULL, NULL, 1}, - /*LOGREC_CHECKPOINT_PAGE= 22 */ - {LOGRECTYPE_VARIABLE_LENGTH, 0, 6, NULL, NULL, NULL, 0}, - /*LOGREC_CHECKPOINT_TRAN= 23 */ - {LOGRECTYPE_VARIABLE_LENGTH, 0, 0, NULL, NULL, NULL, 0}, - /*LOGREC_CHECKPOINT_TABL= 24 */ - {LOGRECTYPE_VARIABLE_LENGTH, 0, 8, NULL, NULL, NULL, 0}, - /*LOGREC_REDO_CREATE_TABLE= 25 */ - {LOGRECTYPE_VARIABLE_LENGTH, 0, 0, NULL, NULL, NULL, 0}, - /*LOGREC_REDO_RENAME_TABLE= 26 */ - {LOGRECTYPE_VARIABLE_LENGTH, 0, 0, NULL, NULL, NULL, 0}, - /*LOGREC_REDO_DROP_TABLE= 27 */ - {LOGRECTYPE_VARIABLE_LENGTH, 0, 0, NULL, NULL, NULL, 0}, - /*LOGREC_REDO_TRUNCATE_TABLE= 28 */ - {LOGRECTYPE_VARIABLE_LENGTH, 0, 0, NULL, NULL, NULL, 0}, - /*LOGREC_FILE_ID= 29 */ - {LOGRECTYPE_VARIABLE_LENGTH, 0, 4, NULL, NULL, NULL, 0}, - /*LOGREC_LONG_TRANSACTION_ID= 30 */ - {LOGRECTYPE_FIXEDLENGTH, 6, 6, NULL, NULL, NULL, 0}, - /*31 */ - {LOGRECTYPE_NOT_ALLOWED, 0, 0, NULL, NULL, NULL, 0}, - /*32 */ - {LOGRECTYPE_NOT_ALLOWED, 0, 0, NULL, NULL, NULL, 0}, - /*33 */ - {LOGRECTYPE_NOT_ALLOWED, 0, 0, NULL, NULL, NULL, 0}, - /*34 */ - {LOGRECTYPE_NOT_ALLOWED, 0, 0, NULL, NULL, NULL, 0}, - /*35 */ - {LOGRECTYPE_NOT_ALLOWED, 0, 0, NULL, NULL, NULL, 0}, - /*36 */ - {LOGRECTYPE_NOT_ALLOWED, 0, 0, NULL, NULL, NULL, 0}, - /*37 */ - {LOGRECTYPE_NOT_ALLOWED, 0, 0, NULL, NULL, NULL, 0}, - /*38 */ - {LOGRECTYPE_NOT_ALLOWED, 0, 0, NULL, NULL, NULL, 0}, - /*39 */ - {LOGRECTYPE_NOT_ALLOWED, 0, 0, NULL, NULL, NULL, 0}, - /*40 */ - {LOGRECTYPE_NOT_ALLOWED, 0, 0, NULL, NULL, NULL, 0}, - /*41 */ - {LOGRECTYPE_NOT_ALLOWED, 0, 0, NULL, NULL, NULL, 0}, - /*42 */ - {LOGRECTYPE_NOT_ALLOWED, 0, 0, NULL, NULL, NULL, 0}, - /*43 */ - {LOGRECTYPE_NOT_ALLOWED, 0, 0, NULL, NULL, NULL, 0}, - /*44 */ - {LOGRECTYPE_NOT_ALLOWED, 0, 0, NULL, NULL, NULL, 0}, - /*45 */ - {LOGRECTYPE_NOT_ALLOWED, 0, 0, NULL, NULL, NULL, 0}, - /*46 */ - {LOGRECTYPE_NOT_ALLOWED, 0, 0, NULL, NULL, NULL, 0}, - /*47 */ - {LOGRECTYPE_NOT_ALLOWED, 0, 0, NULL, NULL, NULL, 0}, - /*48 */ - {LOGRECTYPE_NOT_ALLOWED, 0, 0, NULL, NULL, NULL, 0}, - /*49 */ - {LOGRECTYPE_NOT_ALLOWED, 0, 0, NULL, NULL, NULL, 0}, - /*50 */ - {LOGRECTYPE_NOT_ALLOWED, 0, 0, NULL, NULL, NULL, 0}, - /*51 */ - {LOGRECTYPE_NOT_ALLOWED, 0, 0, NULL, NULL, NULL, 0}, - /*52 */ - {LOGRECTYPE_NOT_ALLOWED, 0, 0, NULL, NULL, NULL, 0}, - /*53 */ - {LOGRECTYPE_NOT_ALLOWED, 0, 0, NULL, NULL, NULL, 0}, - /*54 */ - {LOGRECTYPE_NOT_ALLOWED, 0, 0, NULL, NULL, NULL, 0}, - /*55 */ - {LOGRECTYPE_NOT_ALLOWED, 0, 0, NULL, NULL, NULL, 0}, - /*56 */ - {LOGRECTYPE_NOT_ALLOWED, 0, 0, NULL, NULL, NULL, 0}, - /*57 */ - {LOGRECTYPE_NOT_ALLOWED, 0, 0, NULL, NULL, NULL, 0}, - /*58 */ - {LOGRECTYPE_NOT_ALLOWED, 0, 0, NULL, NULL, NULL, 0}, - /*59 */ - {LOGRECTYPE_NOT_ALLOWED, 0, 0, NULL, NULL, NULL, 0}, - /*60 */ - {LOGRECTYPE_NOT_ALLOWED, 0, 0, NULL, NULL, NULL, 0}, - /*61 */ - {LOGRECTYPE_NOT_ALLOWED, 0, 0, NULL, NULL, NULL, 0}, - /*62 */ - {LOGRECTYPE_NOT_ALLOWED, 0, 0, NULL, NULL, NULL, 0}, - /*LOGREC_RESERVED_FUTURE_EXTENSION= 63 */ - {LOGRECTYPE_NOT_ALLOWED, 0, 0, NULL, NULL, NULL, 0} + log_record_type_descriptor[LOGREC_RESERVED_FOR_CHUNKS23]= + INIT_LOGREC_RESERVED_FOR_CHUNKS23; + log_record_type_descriptor[LOGREC_REDO_INSERT_ROW_HEAD]= + INIT_LOGREC_REDO_INSERT_ROW_HEAD; + log_record_type_descriptor[LOGREC_REDO_INSERT_ROW_TAIL]= + INIT_LOGREC_REDO_INSERT_ROW_TAIL; + log_record_type_descriptor[LOGREC_REDO_INSERT_ROW_BLOB]= + INIT_LOGREC_REDO_INSERT_ROW_BLOB; + log_record_type_descriptor[LOGREC_REDO_INSERT_ROW_BLOBS]= + INIT_LOGREC_REDO_INSERT_ROW_BLOBS; + log_record_type_descriptor[LOGREC_REDO_PURGE_ROW_HEAD]= + INIT_LOGREC_REDO_PURGE_ROW_HEAD; + log_record_type_descriptor[LOGREC_REDO_PURGE_ROW_TAIL]= + INIT_LOGREC_REDO_PURGE_ROW_TAIL; + log_record_type_descriptor[LOGREC_REDO_PURGE_BLOCKS]= + INIT_LOGREC_REDO_PURGE_BLOCKS; + log_record_type_descriptor[LOGREC_REDO_DELETE_ROW]= + INIT_LOGREC_REDO_DELETE_ROW; + log_record_type_descriptor[LOGREC_REDO_UPDATE_ROW_HEAD]= + INIT_LOGREC_REDO_UPDATE_ROW_HEAD; + log_record_type_descriptor[LOGREC_REDO_INDEX]= + INIT_LOGREC_REDO_INDEX; + log_record_type_descriptor[LOGREC_REDO_UNDELETE_ROW]= + INIT_LOGREC_REDO_UNDELETE_ROW; + log_record_type_descriptor[LOGREC_CLR_END]= + INIT_LOGREC_CLR_END; + log_record_type_descriptor[LOGREC_PURGE_END]= + INIT_LOGREC_PURGE_END; + log_record_type_descriptor[LOGREC_UNDO_ROW_INSERT]= + INIT_LOGREC_UNDO_ROW_INSERT; + log_record_type_descriptor[LOGREC_UNDO_ROW_DELETE]= + INIT_LOGREC_UNDO_ROW_DELETE; + log_record_type_descriptor[LOGREC_UNDO_ROW_UPDATE]= + INIT_LOGREC_UNDO_ROW_UPDATE; + log_record_type_descriptor[LOGREC_UNDO_ROW_PURGE]= + INIT_LOGREC_UNDO_ROW_PURGE; + log_record_type_descriptor[LOGREC_UNDO_KEY_INSERT]= + INIT_LOGREC_UNDO_KEY_INSERT; + log_record_type_descriptor[LOGREC_UNDO_KEY_DELETE]= + INIT_LOGREC_UNDO_KEY_DELETE; + log_record_type_descriptor[LOGREC_PREPARE]= + INIT_LOGREC_PREPARE; + log_record_type_descriptor[LOGREC_PREPARE_WITH_UNDO_PURGE]= + INIT_LOGREC_PREPARE_WITH_UNDO_PURGE; + log_record_type_descriptor[LOGREC_COMMIT]= + INIT_LOGREC_COMMIT; + log_record_type_descriptor[LOGREC_COMMIT_WITH_UNDO_PURGE]= + INIT_LOGREC_COMMIT_WITH_UNDO_PURGE; + log_record_type_descriptor[LOGREC_CHECKPOINT_PAGE]= + INIT_LOGREC_CHECKPOINT_PAGE; + log_record_type_descriptor[LOGREC_CHECKPOINT_TRAN]= + INIT_LOGREC_CHECKPOINT_TRAN; + log_record_type_descriptor[LOGREC_CHECKPOINT_TABL]= + INIT_LOGREC_CHECKPOINT_TABL; + log_record_type_descriptor[LOGREC_REDO_CREATE_TABLE]= + INIT_LOGREC_REDO_CREATE_TABLE; + log_record_type_descriptor[LOGREC_REDO_RENAME_TABLE]= + INIT_LOGREC_REDO_RENAME_TABLE; + log_record_type_descriptor[LOGREC_REDO_DROP_TABLE]= + INIT_LOGREC_REDO_DROP_TABLE; + log_record_type_descriptor[LOGREC_REDO_TRUNCATE_TABLE]= + INIT_LOGREC_REDO_TRUNCATE_TABLE; + log_record_type_descriptor[LOGREC_FILE_ID]= + INIT_LOGREC_FILE_ID; + log_record_type_descriptor[LOGREC_LONG_TRANSACTION_ID]= + INIT_LOGREC_LONG_TRANSACTION_ID; }; + /* all possible flags page overheads */ static uint page_overhead[TRANSLOG_FLAGS_NUM]; @@ -1812,6 +1877,7 @@ my_bool translog_init(const char *directory, TRANSLOG_ADDRESS sure_page, last_page, last_valid_page; DBUG_ENTER("translog_init"); + loghandler_init(); if (pthread_mutex_init(&log_descriptor.sent_to_file_lock, MY_MUTEX_INIT_FAST)) @@ -2366,7 +2432,7 @@ static my_bool translog_write_parts_on_page(TRANSLOG_ADDRESS *horizon, DBUG_PRINT("enter", ("Chunk length: %lu parts: %u of %u. Page size: %u " "Buffer size: %lu (%lu)", (ulong) length, - (uint) (cur + 1), (uint) parts->parts.elements, + (uint) (cur + 1), (uint) parts->elements, (uint) cursor->current_page_fill, (ulong) cursor->buffer->size, (ulong) (cursor->ptr - cursor->buffer->buffer))); @@ -2378,35 +2444,39 @@ static my_bool translog_write_parts_on_page(TRANSLOG_ADDRESS *horizon, do { translog_size_t len; - struct st_translog_part *part; + LEX_STRING *part; byte *buff; - DBUG_ASSERT(cur < parts->parts.elements); - part= dynamic_element(&parts->parts, cur, struct st_translog_part *); - buff= part->buff; - DBUG_PRINT("info", ("Part: %u Length: %lu left: %lu", - (uint) (cur + 1), (ulong) part->len, (ulong) left)); + DBUG_ASSERT(cur < parts->elements); + part= parts->parts + cur; + buff= (byte*) part->str; + DBUG_PRINT("info", ("Part: %u Length: %lu left: %lu buff: 0x%lx", + (uint) (cur + 1), (ulong) part->length, (ulong) left, + (ulong) buff)); - if (part->len > left) + if (part->length > left) { /* we should write less then the current part */ len= left; - part->len-= len; - part->buff+= len; + part->length-= len; + part->str+= len; DBUG_PRINT("info", ("Set new part: %u Length: %lu", - (uint) (cur + 1), (ulong) part->len)); + (uint) (cur + 1), (ulong) part->length)); } else { - len= part->len; + len= part->length; cur++; DBUG_PRINT("info", ("moved to next part (len: %lu)", (ulong) len)); } DBUG_PRINT("info", ("copy: 0x%lx <- 0x%lx %u", (ulong) cursor->ptr, (ulong)buff, (uint)len)); - memcpy(cursor->ptr, buff, len); - left-= len; - cursor->ptr+= len; + if (likely(len)) + { + memcpy(cursor->ptr, buff, len); + left-= len; + cursor->ptr+= len; + } } while (left); DBUG_PRINT("info", ("Horizon: (%lu,0x%lx) Length %lu(0x%lx)", @@ -2453,10 +2523,11 @@ translog_write_variable_record_1group_header(struct st_translog_parts *parts, uint16 header_length, byte *chunk0_header) { - struct st_translog_part part; + LEX_STRING *part; DBUG_ASSERT(parts->current != 0); /* first part is left for header */ - parts->total_record_length+= (part.len= header_length); - part.buff= chunk0_header; + part= parts->parts + (--parts->current); + parts->total_record_length+= (part->length= header_length); + part->str= (char*)chunk0_header; /* puts chunk type */ *chunk0_header= (byte) (type | TRANSLOG_CHUNK_LSN); int2store(chunk0_header + 1, short_trid); @@ -2466,8 +2537,6 @@ translog_write_variable_record_1group_header(struct st_translog_parts *parts, header_length); /* puts 0 as chunk length which indicate 1 group record */ int2store(chunk0_header + header_length - 2, 0); - parts->current--; - set_dynamic(&parts->parts, (gptr) &part, parts->current); } @@ -2582,7 +2651,7 @@ translog_write_variable_record_chunk3_page(struct st_translog_parts *parts, struct st_buffer_cursor *cursor) { struct st_translog_buffer *buffer_to_flush; - struct st_translog_part part; + LEX_STRING *part; int rc; byte chunk3_header[1 + 2]; DBUG_ENTER("translog_write_variable_record_chunk3_page"); @@ -2606,14 +2675,13 @@ translog_write_variable_record_chunk3_page(struct st_translog_parts *parts, } DBUG_ASSERT(parts->current != 0); /* first part is left for header */ - parts->total_record_length+= (part.len= 1 + 2); - part.buff= chunk3_header; + part= parts->parts + (--parts->current); + parts->total_record_length+= (part->length= 1 + 2); + part->str= (char*)chunk3_header; /* Puts chunk type */ *chunk3_header= (byte) (TRANSLOG_CHUNK_LNGTH); /* Puts chunk length */ int2store(chunk3_header + 1, length); - parts->current--; - set_dynamic(&parts->parts, (gptr) &part, parts->current); translog_write_parts_on_page(horizon, cursor, length + 1 + 2, parts); DBUG_RETURN(0); @@ -3243,61 +3311,63 @@ static my_bool translog_relative_LSN_encode(struct st_translog_parts *parts, LSN base_lsn, uint lsns, byte *compressed_LSNs) { - struct st_translog_part *part; + LEX_STRING *part; uint lsns_len= lsns * LSN_STORE_SIZE; DBUG_ENTER("translog_relative_LSN_encode"); - part= dynamic_element(&parts->parts, parts->current, - struct st_translog_part *); + part= parts->parts + parts->current; /* collect all LSN(s) in one chunk if it (they) is (are) divided */ - if (part->len < lsns_len) + if (part->length < lsns_len) { - uint copied= part->len; + uint copied= part->length; + LEX_STRING *next_part; DBUG_PRINT("info", ("Using buffer: 0x%lx", (ulong) compressed_LSNs)); - memcpy(compressed_LSNs, part->buff, part->len); + memcpy(compressed_LSNs, (byte*)part->str, part->length); + next_part= parts->parts + parts->current + 1; do { - struct st_translog_part *next_part; - next_part= dynamic_element(&parts->parts, parts->current + 1, - struct st_translog_part *); - if ((next_part->len + copied) < lsns_len) + DBUG_ASSERT(next_part < parts->parts + parts->elements); + if ((next_part->length + copied) < lsns_len) { - memcpy(compressed_LSNs + copied, next_part->buff, next_part->len); - copied+= next_part->len; - delete_dynamic_element(&parts->parts, parts->current + 1); + memcpy(compressed_LSNs + copied, (byte*)next_part->str, + next_part->length); + copied+= next_part->length; + next_part->length= 0; next_part->str= 0; + /* delete_dynamic_element(&parts->parts, parts->current + 1); */ + next_part++; } else { uint len= lsns_len - copied; - memcpy(compressed_LSNs + copied, next_part->buff, len); + memcpy(compressed_LSNs + copied, (byte*)next_part->str, len); copied= lsns_len; - next_part->buff+= len; - next_part->len-= len; + next_part->str+= len; + next_part->length-= len; } } while (copied < lsns_len); - part->len= lsns_len; - part->buff= compressed_LSNs; + part->length= lsns_len; + part->str= (char*)compressed_LSNs; } { /* Compress */ LSN ref; uint economy; - byte *ref_ptr= part->buff + lsns_len - LSN_STORE_SIZE; - byte *dst_ptr= part->buff + lsns_len; - for (; ref_ptr >= part->buff ; ref_ptr-= LSN_STORE_SIZE) + byte *ref_ptr= (byte*)part->str + lsns_len - LSN_STORE_SIZE; + byte *dst_ptr= (byte*)part->str + lsns_len; + for (; ref_ptr >= (byte*)part->str ; ref_ptr-= LSN_STORE_SIZE) { ref= lsn_korr(ref_ptr); if ((dst_ptr= translog_put_LSN_diff(base_lsn, ref, dst_ptr)) == NULL) DBUG_RETURN(1); } /* Note that dst_ptr did grow downward ! */ - economy= (uint) (dst_ptr - part->buff); + economy= (uint) (dst_ptr - (byte*)part->str); DBUG_PRINT("info", ("Economy: %u", economy)); - part->len-= economy; + part->length-= economy; parts->record_length-= economy; parts->total_record_length-= economy; - part->buff= dst_ptr; + part->str= (char*)dst_ptr; } DBUG_RETURN(0); } @@ -3879,7 +3949,7 @@ static my_bool translog_write_fixed_record(LSN *lsn, byte chunk1_header[1 + 2]; /* Max number of such LSNs per record is 2 */ byte compressed_LSNs[2 * LSN_STORE_SIZE]; - struct st_translog_part part; + LEX_STRING *part; int rc; DBUG_ENTER("translog_write_fixed_record"); DBUG_ASSERT((log_record_type_descriptor[type].class == @@ -3949,12 +4019,11 @@ static my_bool translog_write_fixed_record(LSN *lsn, the destination page) */ DBUG_ASSERT(parts->current != 0); /* first part is left for header */ - parts->total_record_length+= (part.len= 1 + 2); - part.buff= chunk1_header; + part= parts->parts + (--parts->current); + parts->total_record_length+= (part->length= 1 + 2); + part->str= (char*)chunk1_header; *chunk1_header= (byte) (type | TRANSLOG_CHUNK_FIXED); int2store(chunk1_header + 1, short_trid); - parts->current--; - set_dynamic(&parts->parts, (gptr) &part, parts->current); rc= translog_write_parts_on_page(&log_descriptor.horizon, &log_descriptor.bc, @@ -3990,9 +4059,12 @@ err: short_trid Sort transaction ID or 0 if it has no sense tcb Transaction control block pointer for hooks by record log type - partN_length length of Ns part of the log - partN_buffer pointer on Ns part buffer - 0 sign of the end of parts + rec_len record length or 0 (count it) + part_no number of parts or 0 (count it) + parts_data zero ended (in case of number of parts is 0) + array of LEX_STRINGs (parts), first + TRANSLOG_INTERNAL_PARTS positions in the log + should be unused (need for loghandler) RETURN 0 OK @@ -4002,73 +4074,67 @@ err: my_bool translog_write_record(LSN *lsn, enum translog_record_type type, SHORT_TRANSACTION_ID short_trid, - void *tcb, - translog_size_t part1_length, - byte *part1_buff, ...) + void *tcb, struct st_maria_share *share, + translog_size_t rec_len, + uint part_no, + LEX_STRING *parts_data) { struct st_translog_parts parts; - struct st_translog_part part; - va_list pvar; + LEX_STRING *part; int rc; DBUG_ENTER("translog_write_record"); DBUG_PRINT("enter", ("type: %u ShortTrID: %u", (uint) type, (uint)short_trid)); - /* move information about parts into dynamic array */ - if (init_dynamic_array(&parts.parts, sizeof(struct st_translog_part), - 10, 10 CALLER_INFO)) + if (share && !share->base.transactional) { - UNRECOVERABLE_ERROR(("init array failed")); - DBUG_RETURN(1); + DBUG_PRINT("info", ("It is not transactional table")); + DBUG_RETURN(0); } - /* reserve place for header */ - parts.current= 1; - part.len= 0; - part.buff= 0; - if (insert_dynamic(&parts.parts, (gptr) &part)) - { - UNRECOVERABLE_ERROR(("insert into array failed")); - DBUG_RETURN(1); - } + parts.parts= parts_data; - parts.record_length= part.len= part1_length; - part.buff= part1_buff; - if (insert_dynamic(&parts.parts, (gptr) &part)) + /* count parts if they are not counted by upper level */ + if (part_no == 0) { - UNRECOVERABLE_ERROR(("insert into array failed")); - DBUG_RETURN(1); + for (part_no= TRANSLOG_INTERNAL_PARTS; + parts_data[part_no].length != 0; + part_no++); } - DBUG_PRINT("info", ("record length: %lu %lu ...", - (ulong) parts.record_length, - (ulong) parts.total_record_length)); + parts.elements= part_no; + parts.current= TRANSLOG_INTERNAL_PARTS; - /* count record length */ - va_start(pvar, part1_buff); - for (;;) + /* clear TRANSLOG_INTERNAL_PARTS */ + DBUG_ASSERT(TRANSLOG_INTERNAL_PARTS == 1); + parts_data[0].str= 0; + parts_data[0].length= 0; + + /* count length of the record */ + if (rec_len == 0) { - part.len= va_arg(pvar, translog_size_t); - if (part.len == 0) - break; - parts.record_length+= part.len; - part.buff= va_arg(pvar, byte*); - if (insert_dynamic(&parts.parts, (gptr) &part)) + for(part= parts_data + TRANSLOG_INTERNAL_PARTS;\ + part < parts_data + part_no; + part++) { - UNRECOVERABLE_ERROR(("insert into array failed")); - DBUG_RETURN(1); + rec_len+= part->length; } - DBUG_PRINT("info", ("record length: %lu %lu ...", - (ulong) parts.record_length, - (ulong) parts.total_record_length)); } - va_end(pvar); + parts.record_length= rec_len; +#ifndef DBUG_OFF + { + uint i; + uint len= 0; + for (i= TRANSLOG_INTERNAL_PARTS; i < part_no; i++) + len+= parts_data[i].length; + DBUG_ASSERT(len == rec_len); + } +#endif /* Start total_record_length from record_length then overhead will be add */ parts.total_record_length= parts.record_length; - va_end(pvar); DBUG_PRINT("info", ("record length: %lu %lu", (ulong) parts.record_length, (ulong) parts.total_record_length)); @@ -4076,6 +4142,7 @@ my_bool translog_write_record(LSN *lsn, /* process this parts */ if (!(rc= (log_record_type_descriptor[type].prewrite_hook && (*log_record_type_descriptor[type].prewrite_hook) (type, tcb, + share, &parts)))) { switch (log_record_type_descriptor[type].class) { @@ -4093,7 +4160,6 @@ my_bool translog_write_record(LSN *lsn, } } - delete_dynamic(&parts.parts); DBUG_RETURN(rc); } @@ -4968,7 +5034,7 @@ static my_bool translog_init_reader_data(LSN lsn, SYNOPSIS translog_read_record_header() lsn log record serial number (address of the record) - offset From the beginning of the record beginning (read§ + offset From the beginning of the record beginning (read§ by translog_read_record_header). length Length of record part which have to be read. buffer Buffer where to read the record part (have to be at diff --git a/storage/maria/ma_loghandler.h b/storage/maria/ma_loghandler.h index 48101814063..3ccb3bf9af2 100644 --- a/storage/maria/ma_loghandler.h +++ b/storage/maria/ma_loghandler.h @@ -1,3 +1,9 @@ +/* transaction log default cache size (TODO: make it global variable) */ +#define TRANSLOG_PAGECACHE_SIZE 1024*1024*2 +/* transaction log default file size (TODO: make it global variable) */ +#define TRANSLOG_FILE_SIZE 1024*1024*1024 +/* transaction log default flags (TODO: make it global variable) */ +#define TRANSLOG_DEFAULT_FLAGS 0 /* Transaction log flags */ #define TRANSLOG_PAGE_CRC 1 @@ -18,48 +24,77 @@ /* short transaction ID type */ typedef uint16 SHORT_TRANSACTION_ID; +struct st_maria_share; + /* Length of CRC at end of pages */ #define CRC_LENGTH 4 +/* Size of file id in logs */ +#define FILEID_STORE_SIZE 2 +/* Size of page reference in log */ +#define PAGE_STORE_SIZE ROW_EXTENT_PAGE_SIZE +/* Size of page ranges in log */ +#define PAGERANGE_STORE_SIZE ROW_EXTENT_COUNT_SIZE +#define DIRPOS_STORE_SIZE 1 + +/* Store methods to match the above sizes */ +#define fileid_store(T,A) int2store(T,A) +#define page_store(T,A) int5store(T,A) +#define dirpos_store(T,A) ((*(uchar*) (T)) = A) +#define pagerange_store(T,A) int2store(T,A) + /* Length of disk drive sector size (we assume that writing it to disk is atomic operation) */ #define DISK_DRIVE_SECTOR_SIZE 512 +/* + Number of empty entries we need to have in LEX_STRING for + translog_write_record() +*/ +#define LOG_INTERNAL_PARTS 1 + +/* position reserved in an array of parts of a log record */ +#define TRANSLOG_INTERNAL_PARTS 1 + /* types of records in the transaction log */ +/* Todo: Set numbers for these when we have all entries figured out */ + enum translog_record_type { LOGREC_RESERVED_FOR_CHUNKS23= 0, - LOGREC_REDO_INSERT_ROW_HEAD= 1, - LOGREC_REDO_INSERT_ROW_TAIL= 2, - LOGREC_REDO_INSERT_ROW_BLOB= 3, - LOGREC_REDO_INSERT_ROW_BLOBS= 4, - LOGREC_REDO_PURGE_ROW= 5, - eLOGREC_REDO_PURGE_BLOCKS= 6, - LOGREC_REDO_DELETE_ROW= 7, - LOGREC_REDO_UPDATE_ROW_HEAD= 8, - LOGREC_REDO_INDEX= 9, - LOGREC_REDO_UNDELETE_ROW= 10, - LOGREC_CLR_END= 11, - LOGREC_PURGE_END= 12, - LOGREC_UNDO_ROW_INSERT= 13, - LOGREC_UNDO_ROW_DELETE= 14, - LOGREC_UNDO_ROW_UPDATE= 15, - LOGREC_UNDO_KEY_INSERT= 16, - LOGREC_UNDO_KEY_DELETE= 17, - LOGREC_PREPARE= 18, - LOGREC_PREPARE_WITH_UNDO_PURGE= 19, - LOGREC_COMMIT= 20, - LOGREC_COMMIT_WITH_UNDO_PURGE= 21, - LOGREC_CHECKPOINT_PAGE= 22, - LOGREC_CHECKPOINT_TRAN= 23, - LOGREC_CHECKPOINT_TABL= 24, - LOGREC_REDO_CREATE_TABLE= 25, - LOGREC_REDO_RENAME_TABLE= 26, - LOGREC_REDO_DROP_TABLE= 27, - LOGREC_REDO_TRUNCATE_TABLE= 28, - LOGREC_FILE_ID= 29, - LOGREC_LONG_TRANSACTION_ID= 30, + LOGREC_REDO_INSERT_ROW_HEAD, + LOGREC_REDO_INSERT_ROW_TAIL, + LOGREC_REDO_INSERT_ROW_BLOB, + LOGREC_REDO_INSERT_ROW_BLOBS, + LOGREC_REDO_PURGE_ROW_HEAD, + LOGREC_REDO_PURGE_ROW_TAIL, + LOGREC_REDO_PURGE_BLOCKS, + LOGREC_REDO_DELETE_ROW, + LOGREC_REDO_UPDATE_ROW_HEAD, + LOGREC_REDO_INDEX, + LOGREC_REDO_UNDELETE_ROW, + LOGREC_CLR_END, + LOGREC_PURGE_END, + LOGREC_UNDO_ROW_INSERT, + LOGREC_UNDO_ROW_DELETE, + LOGREC_UNDO_ROW_UPDATE, + LOGREC_UNDO_ROW_PURGE, + LOGREC_UNDO_KEY_INSERT, + LOGREC_UNDO_KEY_DELETE, + LOGREC_PREPARE, + LOGREC_PREPARE_WITH_UNDO_PURGE, + LOGREC_COMMIT, + LOGREC_COMMIT_WITH_UNDO_PURGE, + LOGREC_CHECKPOINT_PAGE, + LOGREC_CHECKPOINT_TRAN, + LOGREC_CHECKPOINT_TABL, + LOGREC_REDO_CREATE_TABLE, + LOGREC_REDO_RENAME_TABLE, + LOGREC_REDO_DROP_TABLE, + LOGREC_REDO_TRUNCATE_TABLE, + LOGREC_FILE_ID, + LOGREC_LONG_TRANSACTION_ID, LOGREC_RESERVED_FUTURE_EXTENSION= 63 }; #define LOGREC_NUMBER_OF_TYPES 64 /* Maximum, can't be extended */ @@ -145,17 +180,22 @@ struct st_translog_reader_data my_bool eor; /* end of the record */ }; +#ifdef __cplusplus +extern "C" { +#endif +extern void loghandler_init(); extern my_bool translog_init(const char *directory, uint32 log_file_max_size, uint32 server_version, uint32 server_id, PAGECACHE *pagecache, uint flags); extern my_bool translog_write_record(LSN *lsn, - enum translog_record_type type, - SHORT_TRANSACTION_ID short_trid, - void *tcb, - translog_size_t part1_length, - byte *part1_buff, ...); + enum translog_record_type type, + SHORT_TRANSACTION_ID short_trid, + void *tcb, struct st_maria_share *share, + translog_size_t rec_len, + uint part_no, + LEX_STRING *parts_data); extern void translog_destroy(); @@ -182,4 +222,7 @@ extern translog_size_t translog_read_next_record_header(TRANSLOG_SCANNER_DATA *scanner, TRANSLOG_HEADER_BUFFER *buff); +#ifdef __cplusplus +} +#endif diff --git a/storage/maria/ma_open.c b/storage/maria/ma_open.c index 9b7313a469b..b3005571436 100644 --- a/storage/maria/ma_open.c +++ b/storage/maria/ma_open.c @@ -19,6 +19,7 @@ #include "ma_sp_defs.h" #include "ma_rt_index.h" #include "ma_blockrec.h" +#include "trnman.h" #include #if defined(MSDOS) || defined(__WIN__) @@ -431,6 +432,9 @@ MARIA_HA *maria_open(const char *name, int mode, uint open_flags) share->base_length+= TRANS_ROW_EXTRA_HEADER_SIZE; share->base.default_rec_buff_size= max(share->base.pack_reclength, share->base.max_key_length); + share->page_type= (share->base.transactional ? PAGECACHE_LSN_PAGE : + PAGECACHE_PLAIN_PAGE); + if (share->data_file_type == DYNAMIC_RECORD) { share->base.extra_rec_buff_size= @@ -634,6 +638,7 @@ MARIA_HA *maria_open(const char *name, int mode, uint open_flags) share->delay_key_write=1; info.state= &share->state.state; /* Change global values by default */ + info.trn= &dummy_transaction_object; pthread_mutex_unlock(&share->intern_lock); /* Allocate buffer for one record */ diff --git a/storage/maria/ma_pagecache.c b/storage/maria/ma_pagecache.c index b7f30eae625..f0c1d674f4b 100755 --- a/storage/maria/ma_pagecache.c +++ b/storage/maria/ma_pagecache.c @@ -95,7 +95,7 @@ #define PCBLOCK_INFO(B) \ DBUG_PRINT("info", \ - ("block 0x%lx file %lu page %lu s %0x hshL 0x%lx req %u/%u " \ + ("block: 0x%lx file: %lu page: %lu s: %0x hshL: 0x%lx req: %u/%u " \ "wrlock: %c", \ (ulong)(B), \ (ulong)((B)->hash_link ? \ @@ -124,6 +124,9 @@ my_bool my_disable_flush_pagecache_blocks= 0; #define COND_FOR_WRLOCK 2 /* queue of write lock */ #define COND_SIZE 3 /* number of COND_* queues */ +/* offset of LSN on the page */ +#define PAGE_LSN_OFFSET 0 + typedef pthread_cond_t KEYCACHE_CONDVAR; /* descriptor of the page in the page cache block buffer */ @@ -164,34 +167,34 @@ enum PCBLOCK_TEMPERATURE { PCBLOCK_COLD /*free*/ , PCBLOCK_WARM , PCBLOCK_HOT }; /* debug info */ #ifndef DBUG_OFF -static char *page_cache_page_type_str[]= +static const char *page_cache_page_type_str[]= { - (char*)"PLAIN", - (char*)"LSN" + "PLAIN", + "LSN" }; -static char *page_cache_page_write_mode_str[]= +static const char *page_cache_page_write_mode_str[]= { - (char*)"DELAY", - (char*)"NOW", - (char*)"DONE" + "DELAY", + "NOW", + "DONE" }; -static char *page_cache_page_lock_str[]= +static const char *page_cache_page_lock_str[]= { - (char*)"free -> free ", - (char*)"read -> read ", - (char*)"write -> write", - (char*)"free -> read ", - (char*)"free -> write", - (char*)"read -> free ", - (char*)"write -> free ", - (char*)"write -> read " + "free -> free", + "read -> read", + "write -> write", + "free -> read", + "free -> write", + "read -> free", + "write -> free", + "write -> read" }; -static char *page_cache_page_pin_str[]= +static const char *page_cache_page_pin_str[]= { - (char*)"pinned -> pinned ", - (char*)"unpinned -> unpinned", - (char*)"unpinned -> pinned ", - (char*)"pinned -> unpinned" + "pinned -> pinned", + "unpinned -> unpinned", + "unpinned -> pinned", + "pinned -> unpinned" }; #endif #ifndef DBUG_OFF @@ -309,22 +312,21 @@ static my_bool info_check_pin(PAGECACHE_BLOCK_LINK *block, struct st_my_thread_var *thread= my_thread_var; PAGECACHE_PIN_INFO *info= info_find(block->pin_list, thread); DBUG_ENTER("info_check_pin"); - DBUG_PRINT("enter", ("info_check_pin: thread: 0x%lx pin: %s", - (ulong)thread, - page_cache_page_pin_str[mode])); + DBUG_PRINT("enter", ("thread: 0x%lx pin: %s", + (ulong) thread, page_cache_page_pin_str[mode])); if (info) { if (mode == PAGECACHE_PIN_LEFT_UNPINNED) { DBUG_PRINT("info", - ("info_check_pin: thread: 0x%lx block 0x%lx: LEFT_UNPINNED!!!", + ("info_check_pin: thread: 0x%lx block: 0x%lx ; LEFT_UNPINNED!!!", (ulong)thread, (ulong)block)); DBUG_RETURN(1); } else if (mode == PAGECACHE_PIN) { DBUG_PRINT("info", - ("info_check_pin: thread: 0x%lx block 0x%lx: PIN!!!", + ("info_check_pin: thread: 0x%lx block: 0x%lx ; PIN!!!", (ulong)thread, (ulong)block)); DBUG_RETURN(1); } @@ -334,14 +336,14 @@ static my_bool info_check_pin(PAGECACHE_BLOCK_LINK *block, if (mode == PAGECACHE_PIN_LEFT_PINNED) { DBUG_PRINT("info", - ("info_check_pin: thread: 0x%lx block 0x%lx: LEFT_PINNED!!!", + ("info_check_pin: thread: 0x%lx block: 0x%lx ; LEFT_PINNED!!!", (ulong)thread, (ulong)block)); DBUG_RETURN(1); } else if (mode == PAGECACHE_UNPIN) { DBUG_PRINT("info", - ("info_check_pin: thread: 0x%lx block 0x%lx: UNPIN!!!", + ("info_check_pin: thread: 0x%lx block: 0x%lx ; UNPIN!!!", (ulong)thread, (ulong)block)); DBUG_RETURN(1); } @@ -571,7 +573,6 @@ static uint pagecache_fwrite(PAGECACHE *pagecache, LSN lsn; DBUG_PRINT("info", ("Log handler call")); /* TODO: integrate with page format */ -#define PAGE_LSN_OFFSET 0 lsn= lsn_korr(buffer + PAGE_LSN_OFFSET); /* check CONTROL_FILE_IMPOSSIBLE_FILENO & @@ -1193,7 +1194,7 @@ static void link_block(PAGECACHE *pagecache, PAGECACHE_BLOCK_LINK *block, */ if ((PAGECACHE_HASH_LINK *) thread->opt_info == hash_link) { - KEYCACHE_DBUG_PRINT("link_block: signal", ("thread %ld", thread->id)); + KEYCACHE_DBUG_PRINT("link_block: signal", ("thread: %ld", thread->id)); pagecache_pthread_cond_signal(&thread->suspend); wqueue_unlink_from_queue(&pagecache->waiting_for_block, thread); block->requests++; @@ -1204,7 +1205,7 @@ static void link_block(PAGECACHE *pagecache, PAGECACHE_BLOCK_LINK *block, KEYCACHE_THREAD_TRACE("link_block: after signaling"); #if defined(PAGECACHE_DEBUG) KEYCACHE_DBUG_PRINT("link_block", - ("linked,unlinked block %u status=%x #requests=%u #available=%u", + ("linked,unlinked block: %u status: %x #requests: %u #available: %u", PCBLOCK_NUMBER(pagecache, block), block->status, block->requests, pagecache->blocks_available)); #endif @@ -1235,9 +1236,9 @@ static void link_block(PAGECACHE *pagecache, PAGECACHE_BLOCK_LINK *block, #if defined(PAGECACHE_DEBUG) pagecache->blocks_available++; KEYCACHE_DBUG_PRINT("link_block", - ("linked block %u:%1u status=%x #requests=%u #available=%u", - PCBLOCK_NUMBER(pagecache, block), at_end, block->status, - block->requests, pagecache->blocks_available)); + ("linked block: %u:%1u status: %x #requests: %u #available: %u", + PCBLOCK_NUMBER(pagecache, block), at_end, block->status, + block->requests, pagecache->blocks_available)); KEYCACHE_DBUG_ASSERT((ulong) pagecache->blocks_available <= pagecache->blocks_used); #endif @@ -1284,9 +1285,10 @@ static void unlink_block(PAGECACHE *pagecache, PAGECACHE_BLOCK_LINK *block) KEYCACHE_DBUG_ASSERT(pagecache->blocks_available != 0); pagecache->blocks_available--; KEYCACHE_DBUG_PRINT("unlink_block", - ("unlinked block 0x%lx (%u) status=%x #requests=%u #available=%u", - (ulong)block, PCBLOCK_NUMBER(pagecache, block), block->status, - block->requests, pagecache->blocks_available)); + ("unlinked block: 0x%lx (%u) status: %x #requests: %u #available: %u", + (ulong)block, PCBLOCK_NUMBER(pagecache, block), + block->status, + block->requests, pagecache->blocks_available)); PCBLOCK_INFO(block); #endif DBUG_VOID_RETURN; @@ -1310,7 +1312,7 @@ static void reg_requests(PAGECACHE *pagecache, PAGECACHE_BLOCK_LINK *block, int count) { DBUG_ENTER("reg_requests"); - DBUG_PRINT("enter", ("block 0x%lx (%u) status=%x, reqs: %u", + DBUG_PRINT("enter", ("block: 0x%lx (%u) status: %x reqs: %u", (ulong)block, PCBLOCK_NUMBER(pagecache, block), block->status, block->requests)); PCBLOCK_INFO(block); @@ -1355,7 +1357,7 @@ static void unreg_request(PAGECACHE *pagecache, PAGECACHE_BLOCK_LINK *block, int at_end) { DBUG_ENTER("unreg_request"); - DBUG_PRINT("enter", ("block 0x%lx (%u) status=%x, reqs: %u", + DBUG_PRINT("enter", ("block 0x%lx (%u) status: %x reqs: %u", (ulong)block, PCBLOCK_NUMBER(pagecache, block), block->status, block->requests)); PCBLOCK_INFO(block); @@ -1431,7 +1433,7 @@ static inline void wait_for_readers(PAGECACHE *pagecache while (block->hash_link->requests) { KEYCACHE_DBUG_PRINT("wait_for_readers: wait", - ("suspend thread %ld block %u", + ("suspend thread: %ld block: %u", thread->id, PCBLOCK_NUMBER(pagecache, block))); block->condvar= &thread->suspend; pagecache_pthread_cond_wait(&thread->suspend, &pagecache->cache_lock); @@ -1790,7 +1792,7 @@ restart: /* This is a request for a page to be removed from cache */ KEYCACHE_DBUG_PRINT("find_block", - ("request for old page in block %u " + ("request for old page in block: %u " "wrmode: %d block->status: %d", PCBLOCK_NUMBER(pagecache, block), wrmode, block->status)); @@ -2028,11 +2030,11 @@ restart: KEYCACHE_DBUG_ASSERT(page_status != -1); *page_st= page_status; DBUG_PRINT("info", - ("block: 0x%lx fd: %u pos %lu block->status %u page_status %u", + ("block: 0x%lx fd: %u pos: %lu block->status: %u page_status: %u", (ulong) block, (uint) file->file, (ulong) pageno, block->status, (uint) page_status)); KEYCACHE_DBUG_PRINT("find_block", - ("block: 0x%lx fd: %d pos: %lu block->status: %u page_status: %d", + ("block: 0x%lx fd: %d pos: %lu block->status: %u page_status: %d", (ulong) block, file->file, (ulong) pageno, block->status, page_status)); @@ -2049,7 +2051,7 @@ restart: static void add_pin(PAGECACHE_BLOCK_LINK *block) { DBUG_ENTER("add_pin"); - DBUG_PRINT("enter", ("block 0x%lx pins: %u", + DBUG_PRINT("enter", ("block: 0x%lx pins: %u", (ulong) block, block->pins)); PCBLOCK_INFO(block); @@ -2068,7 +2070,7 @@ static void add_pin(PAGECACHE_BLOCK_LINK *block) static void remove_pin(PAGECACHE_BLOCK_LINK *block) { DBUG_ENTER("remove_pin"); - DBUG_PRINT("enter", ("block 0x%lx pins: %u", + DBUG_PRINT("enter", ("block: 0x%lx pins: %u", (ulong) block, block->pins)); PCBLOCK_INFO(block); @@ -2233,7 +2235,7 @@ static my_bool make_lock_and_pin(PAGECACHE *pagecache, enum pagecache_page_pin pin) { DBUG_ENTER("make_lock_and_pin"); - DBUG_PRINT("enter", ("block: 0x%lx (%u), wrlock: %c pins: %u, lock %s, pin: %s", + DBUG_PRINT("enter", ("block: 0x%lx (%u) wrlock: %c pins: %u lock: %s pin: %s", (ulong)block, PCBLOCK_NUMBER(pagecache, block), ((block->status & PCBLOCK_WRLOCK)?'Y':'N'), block->pins, @@ -2242,8 +2244,7 @@ static my_bool make_lock_and_pin(PAGECACHE *pagecache, PCBLOCK_INFO(block); DBUG_ASSERT(info_check_pin(block, pin) == 0 && info_check_lock(block, lock, pin) == 0); - switch (lock) - { + switch (lock) { case PAGECACHE_LOCK_WRITE: /* free -> write */ /* Writelock and pin the buffer */ if (get_wrlock(pagecache, block)) @@ -2412,6 +2413,30 @@ static void read_block(PAGECACHE *pagecache, } +/* + Set LSN on the page to the given one if the given LSN is bigger + + SYNOPSIS + check_and_set_lsn() + lsn LSN to set + block block to check and set +*/ + +static void check_and_set_lsn(LSN lsn, PAGECACHE_BLOCK_LINK *block) +{ + LSN old; + DBUG_ENTER("check_and_set_lsn"); + DBUG_ASSERT(block->type == PAGECACHE_LSN_PAGE); + old= lsn_korr(block->buffer + PAGE_LSN_OFFSET); + DBUG_PRINT("info", ("old lsn: (%lu, 0x%lx) new lsn: (%lu, 0x%lx)", + (ulong)LSN_FILE_NO(old), (ulong)LSN_OFFSET(old), + (ulong)LSN_FILE_NO(lsn), (ulong)LSN_OFFSET(lsn))); + if (cmp_translog_addr(lsn, old) > 0) + lsn_store(block->buffer + PAGE_LSN_OFFSET, lsn); + DBUG_VOID_RETURN; +} + + /* Unlock/unpin page and put LSN stamp if it need @@ -2423,6 +2448,9 @@ static void read_block(PAGECACHE *pagecache, lock lock change pin pin page first_REDO_LSN_for_page do not set it if it is zero + lsn if it is not CONTROL_FILE_IMPOSSIBLE_LSN (0) and it + is bigger then LSN on the page it will be written on + the page NOTE Pininig uses requests registration mechanism it works following way: @@ -2442,12 +2470,12 @@ void pagecache_unlock_page(PAGECACHE *pagecache, pgcache_page_no_t pageno, enum pagecache_page_lock lock, enum pagecache_page_pin pin, - LSN first_REDO_LSN_for_page) + LSN first_REDO_LSN_for_page, LSN lsn) { PAGECACHE_BLOCK_LINK *block; int page_st; DBUG_ENTER("pagecache_unlock_page"); - DBUG_PRINT("enter", ("fd: %u page: %lu l%s p%s", + DBUG_PRINT("enter", ("fd: %u page: %lu %s %s", (uint) file->file, (ulong) pageno, page_cache_page_lock_str[lock], page_cache_page_pin_str[pin])); @@ -2475,19 +2503,15 @@ void pagecache_unlock_page(PAGECACHE *pagecache, pin == PAGECACHE_UNPIN); set_if_bigger(block->rec_lsn, first_REDO_LSN_for_page); } + if (lsn != 0) + { + check_and_set_lsn(lsn, block); + } -#ifndef DBUG_OFF - if ( -#endif - make_lock_and_pin(pagecache, block, lock, pin) -#ifndef DBUG_OFF - ) + if (make_lock_and_pin(pagecache, block, lock, pin)) { DBUG_ASSERT(0); /* should not happend */ } -#else - ; -#endif remove_reader(block); /* @@ -2514,11 +2538,15 @@ void pagecache_unlock_page(PAGECACHE *pagecache, pagecache pointer to a page cache data structure file handler for the file for the block of data to be read pageno number of the block of data in the file + lsn if it is not CONTROL_FILE_IMPOSSIBLE_LSN (0) and it + is bigger then LSN on the page it will be written on + the page */ void pagecache_unpin_page(PAGECACHE *pagecache, PAGECACHE_FILE *file, - pgcache_page_no_t pageno) + pgcache_page_no_t pageno, + LSN lsn) { PAGECACHE_BLOCK_LINK *block; int page_st; @@ -2537,25 +2565,20 @@ void pagecache_unpin_page(PAGECACHE *pagecache, block= find_block(pagecache, file, pageno, 0, 0, 0, &page_st); DBUG_ASSERT(block != 0 && page_st == PAGE_READ); -#ifndef DBUG_OFF - if ( -#endif - /* - we can just unpin only with keeping read lock because: - a) we can't pin without any lock - b) we can't unpin keeping write lock - */ - make_lock_and_pin(pagecache, block, - PAGECACHE_LOCK_LEFT_READLOCKED, - PAGECACHE_UNPIN) -#ifndef DBUG_OFF - ) + if (lsn != 0) { - DBUG_ASSERT(0); /* should not happend */ + check_and_set_lsn(lsn, block); } -#else - ; -#endif + + /* + we can just unpin only with keeping read lock because: + a) we can't pin without any lock + b) we can't unpin keeping write lock + */ + if (make_lock_and_pin(pagecache, block, + PAGECACHE_LOCK_LEFT_READLOCKED, + PAGECACHE_UNPIN)) + DBUG_ASSERT(0); /* should not happend */ remove_reader(block); /* @@ -2584,17 +2607,20 @@ void pagecache_unpin_page(PAGECACHE *pagecache, lock lock change pin pin page first_REDO_LSN_for_page do not set it if it is zero + lsn if it is not CONTROL_FILE_IMPOSSIBLE_LSN (0) and it + is bigger then LSN on the page it will be written on + the page */ void pagecache_unlock(PAGECACHE *pagecache, PAGECACHE_PAGE_LINK *link, enum pagecache_page_lock lock, enum pagecache_page_pin pin, - LSN first_REDO_LSN_for_page) + LSN first_REDO_LSN_for_page, LSN lsn) { PAGECACHE_BLOCK_LINK *block= (PAGECACHE_BLOCK_LINK *)link; DBUG_ENTER("pagecache_unlock"); - DBUG_PRINT("enter", ("block: 0x%lx fd: %u page: %lu l%s p%s", + DBUG_PRINT("enter", ("block: 0x%lx fd: %u page: %lu %s %s", (ulong) block, (uint) block->hash_link->file.file, (ulong) block->hash_link->pageno, @@ -2611,19 +2637,9 @@ void pagecache_unlock(PAGECACHE *pagecache, if (pin == PAGECACHE_PIN_LEFT_UNPINNED && lock == PAGECACHE_LOCK_READ_UNLOCK) { -#ifndef DBUG_OFF - if ( -#endif - /* block do not need here so we do not provide it */ - make_lock_and_pin(pagecache, 0, lock, pin) -#ifndef DBUG_OFF - ) - { - DBUG_ASSERT(0); /* should not happend */ - } -#else - ; -#endif + /* block do not need here so we do not provide it */ + if (make_lock_and_pin(pagecache, 0, lock, pin)) + DBUG_ASSERT(0); /* should not happend */ DBUG_VOID_RETURN; } @@ -2641,19 +2657,13 @@ void pagecache_unlock(PAGECACHE *pagecache, pin == PAGECACHE_UNPIN); set_if_bigger(block->rec_lsn, first_REDO_LSN_for_page); } - -#ifndef DBUG_OFF - if ( -#endif - make_lock_and_pin(pagecache, block, lock, pin) -#ifndef DBUG_OFF - ) + if (lsn != 0) { - DBUG_ASSERT(0); /* should not happend */ + check_and_set_lsn(lsn, block); } -#else - ; -#endif + + if (make_lock_and_pin(pagecache, block, lock, pin)) + DBUG_ASSERT(0); /* should not happend */ remove_reader(block); /* @@ -2680,10 +2690,14 @@ void pagecache_unlock(PAGECACHE *pagecache, pagecache_unpin_page() pagecache pointer to a page cache data structure link direct link to page (returned by read or write) + lsn if it is not CONTROL_FILE_IMPOSSIBLE_LSN (0) and it + is bigger then LSN on the page it will be written on + the page */ void pagecache_unpin(PAGECACHE *pagecache, - PAGECACHE_PAGE_LINK *link) + PAGECACHE_PAGE_LINK *link, + LSN lsn) { PAGECACHE_BLOCK_LINK *block= (PAGECACHE_BLOCK_LINK *)link; DBUG_ENTER("pagecache_unpin"); @@ -2701,25 +2715,20 @@ void pagecache_unpin(PAGECACHE *pagecache, inc_counter_for_resize_op(pagecache); -#ifndef DBUG_OFF - if ( -#endif - /* - we can just unpin only with keeping read lock because: - a) we can't pin without any lock - b) we can't unpin keeping write lock - */ - make_lock_and_pin(pagecache, block, - PAGECACHE_LOCK_LEFT_READLOCKED, - PAGECACHE_UNPIN) -#ifndef DBUG_OFF - ) + if (lsn != 0) { - DBUG_ASSERT(0); /* should not happend */ + check_and_set_lsn(lsn, block); } -#else - ; -#endif + + /* + We can just unpin only with keeping read lock because: + a) we can't pin without any lock + b) we can't unpin keeping write lock + */ + if (make_lock_and_pin(pagecache, block, + PAGECACHE_LOCK_LEFT_READLOCKED, + PAGECACHE_UNPIN)) + DBUG_ASSERT(0); /* should not happend */ remove_reader(block); /* @@ -2785,7 +2794,7 @@ byte *pagecache_valid_read(PAGECACHE *pagecache, enum pagecache_page_pin pin= lock_to_pin[lock]; PAGECACHE_PAGE_LINK fake_link; DBUG_ENTER("pagecache_valid_read"); - DBUG_PRINT("enter", ("fd: %u page: %lu level: %u t:%s l%s p%s", + DBUG_PRINT("enter", ("fd: %u page: %lu level: %u t:%s %s %s", (uint) file->file, (ulong) pageno, level, page_cache_page_type_str[type], page_cache_page_lock_str[lock], @@ -2858,14 +2867,16 @@ restart: #endif } - remove_reader(block); /* Link the block into the LRU chain if it's the last submitted request for the block and block will not be pinned. See NOTE for pagecache_unlock_page about registering requests. */ if (pin == PAGECACHE_PIN_LEFT_UNPINNED || pin == PAGECACHE_UNPIN) + { + remove_reader(block); unreg_request(pagecache, block, 1); + } else *link= (PAGECACHE_PAGE_LINK)block; @@ -2909,6 +2920,7 @@ no_key_cache: /* Key cache is not used */ lock can be only PAGECACHE_LOCK_LEFT_WRITELOCKED (page was write locked before) or PAGECACHE_LOCK_WRITE (delete will write lock page before delete) */ + my_bool pagecache_delete_page(PAGECACHE *pagecache, PAGECACHE_FILE *file, pgcache_page_no_t pageno, @@ -2918,7 +2930,7 @@ my_bool pagecache_delete_page(PAGECACHE *pagecache, int error= 0; enum pagecache_page_pin pin= lock_to_pin[lock]; DBUG_ENTER("pagecache_delete_page"); - DBUG_PRINT("enter", ("fd: %u page: %lu l%s p%s", + DBUG_PRINT("enter", ("fd: %u page: %lu %s %s", (uint) file->file, (ulong) pageno, page_cache_page_lock_str[lock], page_cache_page_pin_str[pin])); @@ -3018,12 +3030,34 @@ end: } +my_bool pagecache_delete_pages(PAGECACHE *pagecache, + PAGECACHE_FILE *file, + pgcache_page_no_t pageno, + uint page_count, + enum pagecache_page_lock lock, + my_bool flush) +{ + ulong page_end; + DBUG_ENTER("pagecache_delete_pages"); + DBUG_ASSERT(page_count > 0); + + page_end= pageno + page_count; + do + { + if (pagecache_delete_page(pagecache, file, pageno, + lock, flush)) + DBUG_RETURN(1); + } while (++pageno != page_end); + DBUG_RETURN(0); +} + + /* Write a buffer into a cached file. SYNOPSIS - pagecache_write() + pagecache_write_part() pagecache pointer to a page cache data structure file handler for the file to write data to pageno number of the block of data in the file @@ -3107,7 +3141,7 @@ my_bool pagecache_write_part(PAGECACHE *pagecache, int error= 0; int need_lock_change= write_lock_change_table[lock].need_lock_change; DBUG_ENTER("pagecache_write_part"); - DBUG_PRINT("enter", ("fd: %u page: %lu level: %u type: %s lock: %s " + DBUG_PRINT("enter", ("fd: %u page: %lu level: %u type: %s lock: %s " "pin: %s mode: %s offset: %u size %u", (uint) file->file, (ulong) pageno, level, page_cache_page_type_str[type], @@ -3182,7 +3216,6 @@ restart: goto restart; } - if (write_mode == PAGECACHE_WRITE_DONE) { if ((block->status & PCBLOCK_ERROR) && page_st != PAGE_READ) @@ -3220,30 +3253,30 @@ restart: if (need_lock_change) { -#ifndef DBUG_OFF - int rc= -#endif - /* - RECOVERY TODO BUG We are doing an unlock here, so need to give the - page its rec_lsn - */ - make_lock_and_pin(pagecache, block, - write_lock_change_table[lock].unlock_lock, - write_pin_change_table[pin].unlock_pin); -#ifndef DBUG_OFF - DBUG_ASSERT(rc == 0); -#endif + /* + RECOVERY TODO BUG We are doing an unlock here, so need to give the + page its rec_lsn + */ + if (make_lock_and_pin(pagecache, block, + write_lock_change_table[lock].unlock_lock, + write_pin_change_table[pin].unlock_pin)) + DBUG_ASSERT(0); } - /* Unregister the request */ - DBUG_ASSERT(block->hash_link->requests > 0); - block->hash_link->requests--; /* See NOTE for pagecache_unlock_page about registering requests. */ if (pin == PAGECACHE_PIN_LEFT_UNPINNED || pin == PAGECACHE_UNPIN) + { + /* Unregister the request */ + DBUG_ASSERT(block->hash_link->requests > 0); + block->hash_link->requests--; unreg_request(pagecache, block, 1); + } else + { + if (pin == PAGECACHE_PIN_LEFT_PINNED) + block->hash_link->requests--; *link= (PAGECACHE_PAGE_LINK)block; - + } if (block->status & PCBLOCK_ERROR) error= 1; @@ -3289,8 +3322,9 @@ static void free_block(PAGECACHE *pagecache, PAGECACHE_BLOCK_LINK *block) { KEYCACHE_THREAD_TRACE("free block"); KEYCACHE_DBUG_PRINT("free_block", - ("block %u to be freed, hash_link %p", - PCBLOCK_NUMBER(pagecache, block), block->hash_link)); + ("block: %u hash_link 0x%lx", + PCBLOCK_NUMBER(pagecache, block), + (long) block->hash_link)); if (block->hash_link) { /* @@ -3376,9 +3410,9 @@ static int flush_cached_blocks(PAGECACHE *pagecache, if (block->pins) { KEYCACHE_DBUG_PRINT("flush_cached_blocks", - ("block %u (0x%lx) pinned", + ("block: %u (0x%lx) pinned", PCBLOCK_NUMBER(pagecache, block), (ulong)block)); - DBUG_PRINT("info", ("block %u (0x%lx) pinned", + DBUG_PRINT("info", ("block: %u (0x%lx) pinned", PCBLOCK_NUMBER(pagecache, block), (ulong)block)); PCBLOCK_INFO(block); last_errno= -1; @@ -3388,25 +3422,18 @@ static int flush_cached_blocks(PAGECACHE *pagecache, /* if the block is not pinned then it is not write locked */ DBUG_ASSERT((block->status & PCBLOCK_WRLOCK) == 0); DBUG_ASSERT(block->pins == 0); -#ifndef DBUG_OFF - { - int rc= -#endif - make_lock_and_pin(pagecache, block, - PAGECACHE_LOCK_WRITE, PAGECACHE_PIN); -#ifndef DBUG_OFF - DBUG_ASSERT(rc == 0); - } -#endif + if (make_lock_and_pin(pagecache, block, + PAGECACHE_LOCK_WRITE, PAGECACHE_PIN)) + DBUG_ASSERT(0); KEYCACHE_DBUG_PRINT("flush_cached_blocks", - ("block %u (0x%lx) to be flushed", + ("block: %u (0x%lx) to be flushed", PCBLOCK_NUMBER(pagecache, block), (ulong)block)); - DBUG_PRINT("info", ("block %u (0x%lx) to be flushed", + DBUG_PRINT("info", ("block: %u (0x%lx) to be flushed", PCBLOCK_NUMBER(pagecache, block), (ulong)block)); PCBLOCK_INFO(block); pagecache_pthread_mutex_unlock(&pagecache->cache_lock); - DBUG_PRINT("info", ("block %u (0x%lx) pins: %u", + DBUG_PRINT("info", ("block: %u (0x%lx) pins: %u", PCBLOCK_NUMBER(pagecache, block), (ulong)block, block->pins)); DBUG_ASSERT(block->pins == 1); diff --git a/storage/maria/ma_static.c b/storage/maria/ma_static.c index 4a6746b71d8..c77f3f512fd 100644 --- a/storage/maria/ma_static.c +++ b/storage/maria/ma_static.c @@ -20,6 +20,7 @@ #ifndef _global_h #include "maria_def.h" +#include "trnman.h" #endif LIST *maria_open_list=0; @@ -43,6 +44,12 @@ ulong maria_data_pointer_size= 4; PAGECACHE maria_pagecache_var; PAGECACHE *maria_pagecache= &maria_pagecache_var; +PAGECACHE maria_log_pagecache_var; +PAGECACHE *maria_log_pagecache= &maria_log_pagecache_var; + +/* For using maria externally */ +TRN dummy_transaction_object; + /* Enough for comparing if number is zero */ byte maria_zero_string[]= {0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0}; diff --git a/storage/maria/ma_statrec.c b/storage/maria/ma_statrec.c index 68864e7c170..8ca3a5e989d 100644 --- a/storage/maria/ma_statrec.c +++ b/storage/maria/ma_statrec.c @@ -86,6 +86,7 @@ my_bool _ma_write_static_record(MARIA_HA *info, const byte *record) } my_bool _ma_update_static_record(MARIA_HA *info, MARIA_RECORD_POS pos, + const byte *oldrec __attribute__ ((unused)), const byte *record) { info->rec_cache.seek_not_done=1; /* We have done a seek */ @@ -96,7 +97,8 @@ my_bool _ma_update_static_record(MARIA_HA *info, MARIA_RECORD_POS pos, } -my_bool _ma_delete_static_record(MARIA_HA *info) +my_bool _ma_delete_static_record(MARIA_HA *info, + const byte *record __attribute__ ((unused))) { byte temp[9]; /* 1+sizeof(uint32) */ info->state->del++; diff --git a/storage/maria/ma_test1.c b/storage/maria/ma_test1.c index 1c69e2c95b4..98d55c7d254 100644 --- a/storage/maria/ma_test1.c +++ b/storage/maria/ma_test1.c @@ -33,7 +33,7 @@ static uint insert_count, update_count, remove_count; static uint pack_keys=0, pack_seg=0, key_length; static uint unique_key=HA_NOSAME; static my_bool pagecacheing, null_fields, silent, skip_update, opt_unique, - verbose, skip_delete; + verbose, skip_delete, transactional; static MARIA_COLUMNDEF recinfo[4]; static MARIA_KEYDEF keyinfo[10]; static HA_KEYSEG keyseg[10]; @@ -152,6 +152,7 @@ static int run_test(const char *filename) create_info.max_rows=(ulong) (rec_pointer_size ? (1L << (rec_pointer_size*8))/40 : 0); + create_info.transactional= transactional; if (maria_create(filename, record_type, 1, keyinfo,2+opt_unique,recinfo, uniques, &uniquedef, &create_info, create_flag)) @@ -595,6 +596,9 @@ static struct my_option my_long_options[] = (gptr*) &skip_delete, 0, GET_BOOL, NO_ARG, 0, 0, 0, 0, 0, 0}, {"skip-update", 'D', "Don't test updates", (gptr*) &skip_update, (gptr*) &skip_update, 0, GET_BOOL, NO_ARG, 0, 0, 0, 0, 0, 0}, + {"transactional", 'T', "Test in transactional mode. (Only works with block format)", + (gptr*) &transactional, (gptr*) &transactional, 0, GET_BOOL, NO_ARG, + 0, 0, 0, 0, 0, 0}, {"unique", 'C', "Undocumented", (gptr*) &opt_unique, (gptr*) &opt_unique, 0, GET_BOOL, NO_ARG, 0, 0, 0, 0, 0, 0}, {"update-rows", 'u', "Undocumented", (gptr*) &update_count, diff --git a/storage/maria/ma_test2.c b/storage/maria/ma_test2.c index 18eaf27073b..cde8da08dca 100644 --- a/storage/maria/ma_test2.c +++ b/storage/maria/ma_test2.c @@ -46,7 +46,7 @@ static void copy_key(struct st_maria_info *info,uint inx, static int verbose=0,testflag=0, first_key=0,async_io=0,pagecacheing=0,write_cacheing=0,locking=0, rec_pointer_size=0,pack_fields=1,silent=0, - opt_quick_mode=0; + opt_quick_mode=0, transactional= 0; static int pack_seg=HA_SPACE_PACK,pack_type=HA_PACK_KEY,remove_count=-1; static int create_flag= 0, srand_arg= 0; static ulong pagecache_size=IO_SIZE*16; @@ -209,6 +209,7 @@ int main(int argc, char *argv[]) (1L << (rec_pointer_size*8))/ reclength : 0); create_info.reloc_rows=(ha_rows) 100; + create_info.transactional= transactional; if (maria_create(filename, record_type, keys,&keyinfo[first_key], use_blob ? 7 : 6, &recinfo[0], 0,(MARIA_UNIQUEDEF*) 0, @@ -993,6 +994,9 @@ static void get_options(int argc, char **argv) case 't': testflag=atoi(++pos); /* testmod */ break; + case 'T': + transactional= 1; + break; case 'q': opt_quick_mode=1; break; @@ -1007,7 +1011,7 @@ static void get_options(int argc, char **argv) case 'V': printf("%s Ver 1.0 for %s at %s\n",progname,SYSTEM_TYPE,MACHINE_TYPE); puts("By Monty, for your professional use\n"); - printf("Usage: %s [-?AbBcDIKLPRqSsVWltv] [-k#] [-f#] [-m#] [-e#] [-E#] [-t#]\n", + printf("Usage: %s [-?AbBcDIKLPRqSsTVWltv] [-k#] [-f#] [-m#] [-e#] [-E#] [-t#]\n", progname); exit(0); case '#': diff --git a/storage/maria/ma_test_all.sh b/storage/maria/ma_test_all.sh index 17e654ac51f..8ee326a9c69 100755 --- a/storage/maria/ma_test_all.sh +++ b/storage/maria/ma_test_all.sh @@ -162,6 +162,9 @@ run_pack_tests -S echo "Running tests with block row format" run_tests -M +echo "Running tests with block row format and transactions" +run_tests "-M -T" + # # Tests that gives warnings # diff --git a/storage/maria/ma_update.c b/storage/maria/ma_update.c index ee0f638ea7c..db0f5641124 100644 --- a/storage/maria/ma_update.c +++ b/storage/maria/ma_update.c @@ -162,7 +162,7 @@ int maria_update(register MARIA_HA *info, const byte *oldrec, byte *newrec) memcpy((char*) &state, (char*) info->state, sizeof(state)); org_split= share->state.split; org_delete_link= share->state.dellink; - if ((*share->update_record)(info,pos,newrec)) + if ((*share->update_record)(info, pos, oldrec, newrec)) goto err; if (!key_changed && (memcmp((char*) &state, (char*) info->state, sizeof(state)) || diff --git a/storage/maria/maria_def.h b/storage/maria/maria_def.h index 5e56b0edc5a..25e27c8d296 100644 --- a/storage/maria/maria_def.h +++ b/storage/maria/maria_def.h @@ -25,10 +25,14 @@ #include #endif -#include #include "ma_loghandler.h" #include "ma_control_file.h" +#define MAX_NONMAPPED_INSERTS 1000 +#define MARIA_MAX_TREE_LEVELS 32 + +struct st_transaction; + /* undef map from my_nosys; We need test-if-disk full */ #undef my_write @@ -205,8 +209,6 @@ typedef struct st_maria_file_bitmap } MARIA_FILE_BITMAP; -#define MAX_NONMAPPED_INSERTS 1000 - typedef struct st_maria_share { /* Shared between opens */ MARIA_STATE_INFO state; @@ -250,8 +252,8 @@ typedef struct st_maria_share /* Called when write failed */ my_bool (*write_record_abort)(struct st_maria_info *); my_bool (*update_record)(struct st_maria_info *, MARIA_RECORD_POS, - const byte *); - my_bool (*delete_record)(struct st_maria_info *); + const byte *, const byte *); + my_bool (*delete_record)(struct st_maria_info *, const byte *record); my_bool (*compare_record)(struct st_maria_info *, const byte *); /* calculate checksum for a row */ ha_checksum(*calc_checksum)(struct st_maria_info *, const byte *); @@ -288,6 +290,7 @@ typedef struct st_maria_share uint base_length; myf write_flag; enum data_file_type data_file_type; + enum pagecache_page_type page_type; /* value depending transactional */ my_bool temporary; /* Below flag is needed to make log tables work with concurrent insert */ my_bool is_log_table; @@ -345,7 +348,6 @@ typedef struct st_maria_row MARIA_RECORD_POS *tail_positions; ha_checksum checksum; byte *empty_bits, *field_lengths; - byte *empty_bits_buffer; /* For storing cur_row.empty_bits */ uint *null_field_lengths; /* All null field lengths */ ulong *blob_lengths; /* Length for each blob */ ulong base_length, normal_length, char_length, varchar_length, blob_length; @@ -371,6 +373,7 @@ typedef struct st_maria_block_scan struct st_maria_info { MARIA_SHARE *s; /* Shared between open:s */ + struct st_transaction *trn; /* Pointer to active transaction */ MARIA_STATUS_INFO *state, save_state; MARIA_ROW cur_row; /* The active row that we just read */ MARIA_ROW new_row; /* Storage for a row during update */ @@ -378,8 +381,10 @@ struct st_maria_info MARIA_BLOB *blobs; /* Pointer to blobs */ MARIA_BIT_BUFF bit_buff; DYNAMIC_ARRAY bitmap_blocks; + DYNAMIC_ARRAY pinned_pages; /* accumulate indexfile changes between write's */ TREE *bulk_insert; + LEX_STRING *log_row_parts; /* For logging */ DYNAMIC_ARRAY *ft1_to_ft2; /* used only in ft1->ft2 conversion */ MEM_ROOT ft_memroot; /* used by the parser */ MYSQL_FTPARSER_PARAM *ftparser_param; /* share info between init/deinit */ @@ -391,6 +396,7 @@ struct st_maria_info byte *rec_buff; /* Temp buffer for recordpack */ byte *int_keypos, /* Save position for next/previous */ *int_maxpos; /* -""- */ + byte *update_field_data; /* Used by update in rows-in-block */ uint int_nod_flag; /* -""- */ uint32 int_keytree_version; /* -""- */ int (*read_record) (struct st_maria_info *, byte*, MARIA_RECORD_POS); @@ -568,8 +574,8 @@ extern pthread_mutex_t THR_LOCK_maria; #define rw_unlock(A) {} #endif - /* Some extern variables */ +/* Some extern variables */ extern LIST *maria_open_list; extern uchar NEAR maria_file_magic[], NEAR maria_pack_file_magic[]; extern uint NEAR maria_read_vec[], NEAR maria_readnext_vec[]; @@ -578,8 +584,8 @@ extern const char *maria_data_root; extern byte maria_zero_string[]; extern my_bool maria_inited; - /* This is used by _ma_calc_xxx_key_length och _ma_store_key */ +/* This is used by _ma_calc_xxx_key_length och _ma_store_key */ typedef struct st_maria_s_param { uint ref_length, key_length, n_ref_length; @@ -589,26 +595,34 @@ typedef struct st_maria_s_param bool store_not_null; } MARIA_KEY_PARAM; - /* Prototypes for intern functions */ +/* Used to store reference to pinned page */ +typedef struct st_pinned_page +{ + PAGECACHE_PAGE_LINK link; + enum pagecache_page_lock unlock; +} MARIA_PINNED_PAGE; + + +/* Prototypes for intern functions */ extern int _ma_read_dynamic_record(MARIA_HA *, byte *, MARIA_RECORD_POS); extern int _ma_read_rnd_dynamic_record(MARIA_HA *, byte *, MARIA_RECORD_POS, my_bool); extern my_bool _ma_write_dynamic_record(MARIA_HA *, const byte *); extern my_bool _ma_update_dynamic_record(MARIA_HA *, MARIA_RECORD_POS, - const byte *); -extern my_bool _ma_delete_dynamic_record(MARIA_HA *info); + const byte *, const byte *); +extern my_bool _ma_delete_dynamic_record(MARIA_HA *info, const byte *record); extern my_bool _ma_cmp_dynamic_record(MARIA_HA *info, const byte *record); extern my_bool _ma_write_blob_record(MARIA_HA *, const byte *); extern my_bool _ma_update_blob_record(MARIA_HA *, MARIA_RECORD_POS, - const byte *); + const byte *, const byte *); extern int _ma_read_static_record(MARIA_HA *info, byte *, MARIA_RECORD_POS); extern int _ma_read_rnd_static_record(MARIA_HA *, byte *, MARIA_RECORD_POS, my_bool); extern my_bool _ma_write_static_record(MARIA_HA *, const byte *); extern my_bool _ma_update_static_record(MARIA_HA *, MARIA_RECORD_POS, - const byte *); -extern my_bool _ma_delete_static_record(MARIA_HA *info); + const byte *, const byte *); +extern my_bool _ma_delete_static_record(MARIA_HA *info, const byte *record); extern my_bool _ma_cmp_static_record(MARIA_HA *info, const byte *record); extern int _ma_ck_write(MARIA_HA *info, uint keynr, byte *key, uint length); @@ -891,3 +905,8 @@ int _ma_create_index_by_sort(MARIA_SORT_PARAM *info, my_bool no_messages, ulong); int _ma_sync_table_files(const MARIA_HA *info); int _ma_initialize_data_file(File dfile, MARIA_SHARE *share); + +void _ma_unpin_all_pages(MARIA_HA *info, LSN undo_lsn); + +extern PAGECACHE *maria_log_pagecache; + diff --git a/storage/maria/trnman.c b/storage/maria/trnman.c index 7918c1aa00d..9b8c36f9769 100644 --- a/storage/maria/trnman.c +++ b/storage/maria/trnman.c @@ -16,7 +16,6 @@ #include #include -#include #include #include "trnman.h" @@ -51,19 +50,38 @@ static TRN **short_trid_to_active_trn; /* locks for short_trid_to_active_trn and pool */ static my_atomic_rwlock_t LOCK_short_trid_to_trn, LOCK_pool; -static LOCKMAN maria_lockman; - /* - short transaction id is at the same time its identifier - for a lock manager - its lock owner identifier (loid) + Simple interface functions */ -#define short_id locks.loid + +uint trnman_increment_locked_tables(TRN *trn) +{ + return trn->locked_tables++; +} + +my_bool trnman_has_locked_tables(TRN *trn) +{ + return trn->locked_tables != 0; +} + +uint trnman_decrement_locked_tables(TRN *trn) +{ + return --trn->locked_tables; +} + +void trnman_reset_locked_tables(TRN *trn) +{ + trn->locked_tables= 0; +} + /* NOTE Just as short_id doubles as loid, this function doubles as short_trid_to_LOCK_OWNER. See the compile-time assert below. */ + +#ifdef NOT_USED static TRN *short_trid_to_TRN(uint16 short_trid) { TRN *trn; @@ -73,6 +91,7 @@ static TRN *short_trid_to_TRN(uint16 short_trid) my_atomic_rwlock_rdunlock(&LOCK_short_trid_to_trn); return (TRN *)trn; } +#endif static byte *trn_get_hash_key(const byte *trn, uint* len, my_bool unused __attribute__ ((unused))) @@ -83,6 +102,7 @@ static byte *trn_get_hash_key(const byte *trn, uint* len, int trnman_init() { + DBUG_ENTER("trnman_init"); /* Initialize lists. active_list_max.min_read_from must be larger than any trid, @@ -94,12 +114,12 @@ int trnman_init() */ active_list_max.trid= active_list_min.trid= 0; - active_list_max.min_read_from= ~0; + active_list_max.min_read_from= ~(ulong) 0; active_list_max.next= active_list_min.prev= 0; active_list_max.prev= &active_list_min; active_list_min.next= &active_list_max; - committed_list_max.commit_trid= ~0; + committed_list_max.commit_trid= ~(ulong) 0; committed_list_max.next= committed_list_min.prev= 0; committed_list_max.prev= &committed_list_min; committed_list_min.next= &committed_list_max; @@ -112,18 +132,21 @@ int trnman_init() global_trid_generator= 0; /* set later by the recovery code */ lf_hash_init(&trid_to_committed_trn, sizeof(TRN*), LF_HASH_UNIQUE, 0, 0, trn_get_hash_key, 0); + DBUG_PRINT("info", ("pthread_mutex_init LOCK_trn_list")); pthread_mutex_init(&LOCK_trn_list, MY_MUTEX_INIT_FAST); my_atomic_rwlock_init(&LOCK_short_trid_to_trn); my_atomic_rwlock_init(&LOCK_pool); short_trid_to_active_trn= (TRN **)my_malloc(SHORT_TRID_MAX*sizeof(TRN*), MYF(MY_WME|MY_ZEROFILL)); if (unlikely(!short_trid_to_active_trn)) - return 1; + DBUG_RETURN(1); short_trid_to_active_trn--; /* min short_trid is 1 */ +#ifdef NOT_USED lockman_init(&maria_lockman, (loid_to_lo_func *)&short_trid_to_TRN, 10000); +#endif - return 0; + DBUG_RETURN(0); } /* @@ -133,6 +156,7 @@ int trnman_init() */ void trnman_destroy() { + DBUG_ENTER("trnman_destroy"); DBUG_ASSERT(trid_to_committed_trn.count == 0); DBUG_ASSERT(trnman_active_transactions == 0); DBUG_ASSERT(trnman_committed_transactions == 0); @@ -149,11 +173,15 @@ void trnman_destroy() my_free((void *)trn, MYF(0)); } lf_hash_destroy(&trid_to_committed_trn); + DBUG_PRINT("info", ("pthread_mutex_destroy LOCK_trn_list")); pthread_mutex_destroy(&LOCK_trn_list); my_atomic_rwlock_destroy(&LOCK_short_trid_to_trn); my_atomic_rwlock_destroy(&LOCK_pool); my_free((void *)(short_trid_to_active_trn+1), MYF(0)); +#ifdef NOT_USED lockman_destroy(&maria_lockman); +#endif + DBUG_VOID_RETURN; } /* @@ -164,9 +192,11 @@ void trnman_destroy() */ static TrID new_trid() { + DBUG_ENTER("new_trid"); DBUG_ASSERT(global_trid_generator < 0xffffffffffffLL); + DBUG_PRINT("info", ("safe_mutex_assert_owner LOCK_trn_list")); safe_mutex_assert_owner(&LOCK_trn_list); - return ++global_trid_generator; + DBUG_RETURN(++global_trid_generator); } static void set_short_trid(TRN *trn) @@ -189,9 +219,12 @@ static void set_short_trid(TRN *trn) start a new transaction, allocate and initialize transaction object mutex and cond will be used for lock waits */ -TRN *trnman_new_trn(pthread_mutex_t *mutex, pthread_cond_t *cond) + +TRN *trnman_new_trn(pthread_mutex_t *mutex, pthread_cond_t *cond, + void *stack_end) { TRN *trn; + DBUG_ENTER("trnman_new_trn"); /* we have a mutex, to do simple things under it - allocate a TRN, @@ -202,6 +235,7 @@ TRN *trnman_new_trn(pthread_mutex_t *mutex, pthread_cond_t *cond) mutex. */ + DBUG_PRINT("info", ("pthread_mutex_lock LOCK_trn_list")); pthread_mutex_lock(&LOCK_trn_list); /* Allocating a new TRN structure */ @@ -222,11 +256,19 @@ TRN *trnman_new_trn(pthread_mutex_t *mutex, pthread_cond_t *cond) trn= (TRN *)my_malloc(sizeof(TRN), MYF(MY_WME)); if (unlikely(!trn)) { + DBUG_PRINT("info", ("pthread_mutex_unlock LOCK_trn_list")); pthread_mutex_unlock(&LOCK_trn_list); return 0; } trnman_allocated_transactions++; } + trn->pins= lf_hash_get_pins(&trid_to_committed_trn, stack_end); + if (!trn->pins) + { + trnman_free_trn(trn); + return 0; + } + trnman_active_transactions++; trn->min_read_from= active_list_min.next->trid; @@ -237,10 +279,9 @@ TRN *trnman_new_trn(pthread_mutex_t *mutex, pthread_cond_t *cond) trn->next= &active_list_max; trn->prev= active_list_max.prev; active_list_max.prev= trn->prev->next= trn; + DBUG_PRINT("info", ("pthread_mutex_unlock LOCK_trn_list")); pthread_mutex_unlock(&LOCK_trn_list); - trn->pins= lf_hash_get_pins(&trid_to_committed_trn); - if (unlikely(!trn->min_read_from)) trn->min_read_from= trn->trid; @@ -250,7 +291,11 @@ TRN *trnman_new_trn(pthread_mutex_t *mutex, pthread_cond_t *cond) trn->locks.cond= cond; trn->locks.waiting_for= 0; trn->locks.all_locks= 0; +#ifdef NOT_USED trn->locks.pins= lf_alloc_get_pins(&maria_lockman.alloc); +#endif + + trn->locked_tables= 0; /* only after the following function TRN is considered initialized, @@ -258,7 +303,7 @@ TRN *trnman_new_trn(pthread_mutex_t *mutex, pthread_cond_t *cond) */ set_short_trid(trn); - return trn; + DBUG_RETURN(trn); } /* @@ -273,12 +318,19 @@ TRN *trnman_new_trn(pthread_mutex_t *mutex, pthread_cond_t *cond) released arbitrarily late. In other words, when locks are released it serves as a start banner for other threads, they start to run. So everything they may need must be ready at that point. + + RETURN + 0 ok + 1 error */ -void trnman_end_trn(TRN *trn, my_bool commit) +int trnman_end_trn(TRN *trn, my_bool commit) { + int res= 1; TRN *free_me= 0; LF_PINS *pins= trn->pins; + DBUG_ENTER("trnman_end_trn"); + DBUG_PRINT("info", ("pthread_mutex_lock LOCK_trn_list")); pthread_mutex_lock(&LOCK_trn_list); /* remove from active list */ @@ -314,30 +366,41 @@ void trnman_end_trn(TRN *trn, my_bool commit) /* if transaction is committed and it was not the only active transaction - add it to the committed list (which is used for read-from relation) + TODO check in the condition below that a transaction have made some + changes, was not read-only. Something like '&& UndoLSN != 0' */ if (commit && active_list_min.next != &active_list_max) { - int res; - trn->commit_trid= global_trid_generator; trn->next= &committed_list_max; trn->prev= committed_list_max.prev; - committed_list_max.prev= trn->prev->next= trn; trnman_committed_transactions++; res= lf_hash_insert(&trid_to_committed_trn, pins, &trn); - DBUG_ASSERT(res == 0); + DBUG_ASSERT(res <= 0); } - else /* otherwise free it right away */ + if (res) { + /* + res == 1 means the condition in the if() above + was false. + res == -1 means lf_hash_insert failed + */ trn->next= free_me; free_me= trn; } + else + { + committed_list_max.prev= trn->prev->next= trn; + } trnman_active_transactions--; + DBUG_PRINT("info", ("pthread_mutex_unlock LOCK_trn_list")); pthread_mutex_unlock(&LOCK_trn_list); /* the rest is done outside of a critical section */ +#ifdef NOT_USED lockman_release_locks(&maria_lockman, &trn->locks); +#endif trn->locks.mutex= 0; trn->locks.cond= 0; my_atomic_rwlock_rdlock(&LOCK_short_trid_to_trn); @@ -356,13 +419,20 @@ void trnman_end_trn(TRN *trn, my_bool commit) TRN *t= free_me; free_me= free_me->next; - lf_hash_delete(&trid_to_committed_trn, pins, &t->trid, sizeof(TrID)); + /* + ignore OOM here. it's harmless, and there's nothing we could do, anyway + */ + (void)lf_hash_delete(&trid_to_committed_trn, pins, &t->trid, sizeof(TrID)); trnman_free_trn(t); } lf_hash_put_pins(pins); +#ifdef NOT_USED lf_pinbox_put_pins(trn->locks.pins); +#endif + + DBUG_RETURN(res < 0); } /* @@ -404,27 +474,44 @@ void trnman_free_trn(TRN *trn) found->trid >= trn->min_read_from and found->commit_trid > found->trid + + RETURN + 1 can + 0 cannot + -1 error (OOM) */ -my_bool trnman_can_read_from(TRN *trn, TrID trid) +int trnman_can_read_from(TRN *trn, TrID trid) { TRN **found; my_bool can; LF_REQUIRE_PINS(3); if (trid < trn->min_read_from) - return TRUE; /* can read */ + return 1; /* can read */ if (trid > trn->trid) - return FALSE; /* cannot read */ + return 0; /* cannot read */ found= lf_hash_search(&trid_to_committed_trn, trn->pins, &trid, sizeof(trid)); - if (!found) - return FALSE; /* not in the hash of committed transactions = cannot read */ + if (found == NULL) + return 0; /* not in the hash of committed transactions = cannot read */ + if (found == MY_ERRPTR) + return -1; can= (*found)->commit_trid < trn->trid; - lf_unpin(trn->pins, 2); + lf_hash_search_unpin(trn->pins); return can; } +/* TODO: the stubs below are waiting for savepoints to be implemented */ + +void trnman_new_statement(TRN *trn __attribute__ ((unused))) +{ +} + +void trnman_rollback_statement(TRN *trn __attribute__ ((unused))) +{ +} + /* Allocates two buffers and stores in them some information about transactions @@ -458,6 +545,7 @@ my_bool trnman_collect_transactions(LEX_STRING *str_act, LEX_STRING *str_com) DBUG_ASSERT((NULL == str_act->str) && (NULL == str_com->str)); + DBUG_PRINT("info", ("pthread_mutex_lock LOCK_trn_list")); pthread_mutex_lock(&LOCK_trn_list); str_act->length= 8+(6+2+7+7+7)*trnman_active_transactions; str_com->length= 8+(6+7+7)*trnman_committed_transactions; @@ -513,6 +601,7 @@ my_bool trnman_collect_transactions(LEX_STRING *str_act, LEX_STRING *str_com) err: error= 1; end: + DBUG_PRINT("info", ("pthread_mutex_unlock LOCK_trn_list")); pthread_mutex_unlock(&LOCK_trn_list); DBUG_RETURN(error); } diff --git a/storage/maria/trnman.h b/storage/maria/trnman.h index 87107ab52fb..24936253935 100644 --- a/storage/maria/trnman.h +++ b/storage/maria/trnman.h @@ -16,10 +16,12 @@ #ifndef _trnman_h #define _trnman_h -#include "lockman.h" +C_MODE_START -typedef uint64 TrID; /* our TrID is 6 bytes */ -typedef struct st_transaction TRN; +#include +#include "lockman.h" +#include "trnman_public.h" +#include "ma_loghandler_lsn.h" /* trid - 6 byte transaction identifier. Assigned when a transaction @@ -29,28 +31,28 @@ typedef struct st_transaction TRN; short_trid - 2-byte transaction identifier, identifies a running transaction, is reassigned when transaction ends. */ + +/* + short transaction id is at the same time its identifier + for a lock manager - its lock owner identifier (loid) +*/ + +#define short_id locks.loid + struct st_transaction { LOCK_OWNER locks; /* must be the first! see short_trid_to_TRN() */ LF_PINS *pins; TrID trid, min_read_from, commit_trid; TRN *next, *prev; + LSN undo_lsn; + uint locked_tables; /* Note! if locks.loid is 0, trn is NOT initialized */ }; -#define SHORT_TRID_MAX 65535 - -extern uint trnman_active_transactions, trnman_allocated_transactions; +TRN dummy_transaction_object; -int trnman_init(void); -void trnman_destroy(void); -TRN *trnman_new_trn(pthread_mutex_t *mutex, pthread_cond_t *cond); -void trnman_end_trn(TRN *trn, my_bool commit); -#define trnman_commit_trn(T) trnman_end_trn(T, TRUE) -#define trnman_abort_trn(T) trnman_end_trn(T, FALSE) -void trnman_free_trn(TRN *trn); -my_bool trnman_can_read_from(TRN *trn, TrID trid); -my_bool trnman_collect_transactions(LEX_STRING *str_act, LEX_STRING *str_com); +C_MODE_END #endif diff --git a/storage/maria/trnman_public.h b/storage/maria/trnman_public.h new file mode 100644 index 00000000000..4b3f8acb4b3 --- /dev/null +++ b/storage/maria/trnman_public.h @@ -0,0 +1,49 @@ +/* Copyright (C) 2006 MySQL AB + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ + + +/* + External definitions for trnman.h + We need to split this into two files as gcc 4.1.2 gives error if it tries + to include my_atomic.h in C++ code. +*/ + +C_MODE_START +typedef uint64 TrID; /* our TrID is 6 bytes */ +typedef struct st_transaction TRN; + +#define SHORT_TRID_MAX 65535 + +extern uint trnman_active_transactions, trnman_allocated_transactions; + +int trnman_init(void); +void trnman_destroy(void); +TRN *trnman_new_trn(pthread_mutex_t *, pthread_cond_t *, void *); +int trnman_end_trn(TRN *trn, my_bool commit); +#define trnman_commit_trn(T) trnman_end_trn(T, TRUE) +#define trnman_abort_trn(T) trnman_end_trn(T, FALSE) +#define trnman_rollback_trn(T) trnman_end_trn(T, FALSE) +void trnman_free_trn(TRN *trn); +int trnman_can_read_from(TRN *trn, TrID trid); +void trnman_new_statement(TRN *trn); +void trnman_rollback_statement(TRN *trn); +my_bool trnman_collect_transactions(LEX_STRING *str_act, LEX_STRING *str_com); + +uint trnman_increment_locked_tables(TRN *trn); +uint trnman_decrement_locked_tables(TRN *trn); +my_bool trnman_has_locked_tables(TRN *trn); +void trnman_reset_locked_tables(TRN *trn); + +C_MODE_END diff --git a/storage/maria/unittest/ma_pagecache_single.c b/storage/maria/unittest/ma_pagecache_single.c index 91cceee618d..2dfa4e89feb 100644 --- a/storage/maria/unittest/ma_pagecache_single.c +++ b/storage/maria/unittest/ma_pagecache_single.c @@ -236,7 +236,7 @@ int simple_pin_test() 0, PAGECACHE_LOCK_READ_UNLOCK, PAGECACHE_UNPIN, - 0); + 0, 0); if (flush_pagecache_blocks(&pagecache, &file1, FLUSH_FORCE_WRITE)) { diag("Got error in flush_pagecache_blocks\n"); @@ -384,7 +384,7 @@ int simple_big_test() } } } - ok(1, "simple big file sequentally read"); + ok(1, "Simple big file sequential read"); /* chack random reads */ for (i= 0; i < PCACHE_SIZE/(PAGE_SIZE); i++) { @@ -403,7 +403,7 @@ int simple_big_test() } } } - ok(1, "simple big file random read"); + ok(1, "Simple big file random read"); flush_pagecache_blocks(&pagecache, &file1, FLUSH_FORCE_WRITE); ok((res= test(test_file(file1, file1_name, PCACHE_SIZE*2, PAGE_SIZE, @@ -432,10 +432,15 @@ static void *test_thread(void *arg) !simple_read_change_write_read_test() || !simple_pin_test() || !simple_delete_forget_test() || - !simple_delete_flush_test() || - !simple_big_test()) + !simple_delete_flush_test()) exit(1); + SKIP_BIG_TESTS(4) + { + if (!simple_big_test()) + exit(1); + } + DBUG_PRINT("info", ("Thread %s ended\n", my_thread_name())); pthread_mutex_lock(&LOCK_thread_count); thread_count--; diff --git a/storage/maria/unittest/ma_test_loghandler-t.c b/storage/maria/unittest/ma_test_loghandler-t.c index 757520322c8..047e9c12bfc 100644 --- a/storage/maria/unittest/ma_test_loghandler-t.c +++ b/storage/maria/unittest/ma_test_loghandler-t.c @@ -108,6 +108,7 @@ int main(int argc, char *argv[]) PAGECACHE pagecache; LSN lsn, lsn_base, first_lsn; TRANSLOG_HEADER_BUFFER rec; + LEX_STRING parts[TRANSLOG_INTERNAL_PARTS + 3]; struct st_translog_scanner_data scanner; int rc; @@ -163,9 +164,12 @@ int main(int argc, char *argv[]) long_tr_id[5]= 0xff; int4store(long_tr_id, 0); + parts[TRANSLOG_INTERNAL_PARTS + 0].str= (char*)long_tr_id; + parts[TRANSLOG_INTERNAL_PARTS + 0].length= 6; if (translog_write_record(&lsn, LOGREC_LONG_TRANSACTION_ID, - 0, NULL, 6, long_tr_id, 0)) + 0, NULL, NULL, + 6, TRANSLOG_INTERNAL_PARTS + 1, parts)) { fprintf(stderr, "Can't write record #%lu\n", (ulong) 0); translog_destroy(); @@ -180,9 +184,13 @@ int main(int argc, char *argv[]) if (i % 2) { lsn_store(lsn_buff, lsn_base); - if (translog_write_record(&lsn, - LOGREC_CLR_END, - (i % 0xFFFF), NULL, 7, lsn_buff, 0)) + parts[TRANSLOG_INTERNAL_PARTS + 0].str= (char*)lsn_buff; + parts[TRANSLOG_INTERNAL_PARTS + 0].length= LSN_STORE_SIZE; + /* check auto-count feature */ + parts[TRANSLOG_INTERNAL_PARTS + 1].str= NULL; + parts[TRANSLOG_INTERNAL_PARTS + 1].length= 0; + if (translog_write_record(&lsn, LOGREC_CLR_END, (i % 0xFFFF), NULL, + NULL, LSN_STORE_SIZE, 0, parts)) { fprintf(stderr, "1 Can't write reference defore record #%lu\n", (ulong) i); @@ -194,10 +202,16 @@ int main(int argc, char *argv[]) lsn_store(lsn_buff, lsn_base); if ((rec_len= random() / (RAND_MAX / (LONG_BUFFER_SIZE + 1))) < 12) rec_len= 12; + parts[TRANSLOG_INTERNAL_PARTS + 0].str= (char*)lsn_buff; + parts[TRANSLOG_INTERNAL_PARTS + 0].length= LSN_STORE_SIZE; + parts[TRANSLOG_INTERNAL_PARTS + 1].str= (char*)long_buffer; + parts[TRANSLOG_INTERNAL_PARTS + 1].length= rec_len; + /* check record length auto-counting */ if (translog_write_record(&lsn, LOGREC_UNDO_KEY_INSERT, (i % 0xFFFF), - NULL, 7, lsn_buff, rec_len, long_buffer, 0)) + NULL, NULL, 0, TRANSLOG_INTERNAL_PARTS + 2, + parts)) { fprintf(stderr, "1 Can't write var reference defore record #%lu\n", (ulong) i); @@ -211,9 +225,12 @@ int main(int argc, char *argv[]) { lsn_store(lsn_buff, lsn_base); lsn_store(lsn_buff + LSN_STORE_SIZE, first_lsn); + parts[TRANSLOG_INTERNAL_PARTS + 0].str= (char*)lsn_buff; + parts[TRANSLOG_INTERNAL_PARTS + 0].length= 23; if (translog_write_record(&lsn, LOGREC_UNDO_ROW_DELETE, - (i % 0xFFFF), NULL, 23, lsn_buff, 0)) + (i % 0xFFFF), NULL, NULL, + 23, TRANSLOG_INTERNAL_PARTS + 1, parts)) { fprintf(stderr, "0 Can't write reference defore record #%lu\n", (ulong) i); @@ -226,10 +243,15 @@ int main(int argc, char *argv[]) lsn_store(lsn_buff + LSN_STORE_SIZE, first_lsn); if ((rec_len= random() / (RAND_MAX / (LONG_BUFFER_SIZE + 1))) < 19) rec_len= 19; + parts[TRANSLOG_INTERNAL_PARTS + 0].str= (char*)lsn_buff; + parts[TRANSLOG_INTERNAL_PARTS + 0].length= 14; + parts[TRANSLOG_INTERNAL_PARTS + 1].str= (char*)long_buffer; + parts[TRANSLOG_INTERNAL_PARTS + 1].length= rec_len; if (translog_write_record(&lsn, LOGREC_UNDO_KEY_DELETE, (i % 0xFFFF), - NULL, 14, lsn_buff, rec_len, long_buffer, 0)) + NULL, NULL, 14 + rec_len, + TRANSLOG_INTERNAL_PARTS + 2, parts)) { fprintf(stderr, "0 Can't write var reference defore record #%lu\n", (ulong) i); @@ -240,9 +262,13 @@ int main(int argc, char *argv[]) ok(1, "write LOGREC_UNDO_KEY_DELETE"); } int4store(long_tr_id, i); + parts[TRANSLOG_INTERNAL_PARTS + 0].str= (char*)long_tr_id; + parts[TRANSLOG_INTERNAL_PARTS + 0].length= 6; if (translog_write_record(&lsn, LOGREC_LONG_TRANSACTION_ID, - (i % 0xFFFF), NULL, 6, long_tr_id, 0)) + (i % 0xFFFF), NULL, NULL, 6, + TRANSLOG_INTERNAL_PARTS + 1, + parts)) { fprintf(stderr, "Can't write record #%lu\n", (ulong) i); translog_destroy(); @@ -255,9 +281,13 @@ int main(int argc, char *argv[]) if ((rec_len= random() / (RAND_MAX / (LONG_BUFFER_SIZE + 1))) < 9) rec_len= 9; + parts[TRANSLOG_INTERNAL_PARTS + 0].str= (char*)long_buffer; + parts[TRANSLOG_INTERNAL_PARTS + 0].length= rec_len; if (translog_write_record(&lsn, LOGREC_REDO_INSERT_ROW_HEAD, - (i % 0xFFFF), NULL, rec_len, long_buffer, 0)) + (i % 0xFFFF), NULL, NULL, rec_len, + TRANSLOG_INTERNAL_PARTS + 1, + parts)) { fprintf(stderr, "Can't write variable record #%lu\n", (ulong) i); translog_destroy(); diff --git a/storage/maria/unittest/ma_test_loghandler_multigroup-t.c b/storage/maria/unittest/ma_test_loghandler_multigroup-t.c index 4aaf30bd9a3..f07ceab1a49 100644 --- a/storage/maria/unittest/ma_test_loghandler_multigroup-t.c +++ b/storage/maria/unittest/ma_test_loghandler_multigroup-t.c @@ -124,6 +124,7 @@ int main(int argc, char *argv[]) PAGECACHE pagecache; LSN lsn, lsn_base, first_lsn; TRANSLOG_HEADER_BUFFER rec; + LEX_STRING parts[TRANSLOG_INTERNAL_PARTS + 2]; struct st_translog_scanner_data scanner; int rc; @@ -183,9 +184,10 @@ int main(int argc, char *argv[]) long_tr_id[5]= 0xff; int4store(long_tr_id, 0); - if (translog_write_record(&lsn, - LOGREC_LONG_TRANSACTION_ID, - 0, NULL, 6, long_tr_id, 0)) + parts[TRANSLOG_INTERNAL_PARTS + 0].str= (char*)long_tr_id; + parts[TRANSLOG_INTERNAL_PARTS + 0].length= 6; + if (translog_write_record(&lsn, LOGREC_LONG_TRANSACTION_ID, 0, NULL, NULL, + 6, TRANSLOG_INTERNAL_PARTS + 1, parts)) { fprintf(stderr, "Can't write record #%lu\n", (ulong) 0); translog_destroy(); @@ -200,10 +202,13 @@ int main(int argc, char *argv[]) if (i % 2) { lsn_store(lsn_buff, lsn_base); + parts[TRANSLOG_INTERNAL_PARTS + 0].str= (char*)lsn_buff; + parts[TRANSLOG_INTERNAL_PARTS + 0].length= LSN_STORE_SIZE; if (translog_write_record(&lsn, LOGREC_CLR_END, - (i % 0xFFFF), NULL, - LSN_STORE_SIZE, lsn_buff, 0)) + (i % 0xFFFF), NULL, NULL, + LSN_STORE_SIZE, + TRANSLOG_INTERNAL_PARTS + 1, parts)) { fprintf(stderr, "1 Can't write reference before record #%lu\n", (ulong) i); @@ -214,11 +219,16 @@ int main(int argc, char *argv[]) ok(1, "write LOGREC_CLR_END"); lsn_store(lsn_buff, lsn_base); rec_len= get_len(); + parts[TRANSLOG_INTERNAL_PARTS + 0].str= (char*)lsn_buff; + parts[TRANSLOG_INTERNAL_PARTS + 0].length= LSN_STORE_SIZE; + parts[TRANSLOG_INTERNAL_PARTS + 1].str= (char*)long_buffer; + parts[TRANSLOG_INTERNAL_PARTS + 1].length= rec_len; if (translog_write_record(&lsn, LOGREC_UNDO_KEY_INSERT, (i % 0xFFFF), - NULL, LSN_STORE_SIZE, lsn_buff, - rec_len, long_buffer, 0)) + NULL, NULL, LSN_STORE_SIZE + rec_len, + TRANSLOG_INTERNAL_PARTS + 2, + parts)) { fprintf(stderr, "1 Can't write var reference before record #%lu\n", (ulong) i); @@ -232,9 +242,13 @@ int main(int argc, char *argv[]) { lsn_store(lsn_buff, lsn_base); lsn_store(lsn_buff + LSN_STORE_SIZE, first_lsn); + parts[TRANSLOG_INTERNAL_PARTS + 1].str= (char*)lsn_buff; + parts[TRANSLOG_INTERNAL_PARTS + 1].length= 23; if (translog_write_record(&lsn, LOGREC_UNDO_ROW_DELETE, - (i % 0xFFFF), NULL, 23, lsn_buff, 0)) + (i % 0xFFFF), NULL, NULL, 23, + TRANSLOG_INTERNAL_PARTS + 1, + parts)) { fprintf(stderr, "0 Can't write reference before record #%lu\n", (ulong) i); @@ -246,11 +260,16 @@ int main(int argc, char *argv[]) lsn_store(lsn_buff, lsn_base); lsn_store(lsn_buff + LSN_STORE_SIZE, first_lsn); rec_len= get_len(); + parts[TRANSLOG_INTERNAL_PARTS + 0].str= (char*)lsn_buff; + parts[TRANSLOG_INTERNAL_PARTS + 0].length= LSN_STORE_SIZE * 2; + parts[TRANSLOG_INTERNAL_PARTS + 1].str= (char*)long_buffer; + parts[TRANSLOG_INTERNAL_PARTS + 1].length= rec_len; if (translog_write_record(&lsn, LOGREC_UNDO_KEY_DELETE, (i % 0xFFFF), - NULL, LSN_STORE_SIZE * 2, lsn_buff, - rec_len, long_buffer, 0)) + NULL, NULL, LSN_STORE_SIZE * 2 + rec_len, + TRANSLOG_INTERNAL_PARTS + 2, + parts)) { fprintf(stderr, "0 Can't write var reference before record #%lu\n", (ulong) i); @@ -261,9 +280,12 @@ int main(int argc, char *argv[]) ok(1, "write LOGREC_UNDO_KEY_DELETE"); } int4store(long_tr_id, i); + parts[TRANSLOG_INTERNAL_PARTS + 0].str= (char*)long_tr_id; + parts[TRANSLOG_INTERNAL_PARTS + 0].length= 6; if (translog_write_record(&lsn, LOGREC_LONG_TRANSACTION_ID, - (i % 0xFFFF), NULL, 6, long_tr_id, 0)) + (i % 0xFFFF), NULL, NULL, 6, + TRANSLOG_INTERNAL_PARTS + 1, parts)) { fprintf(stderr, "Can't write record #%lu\n", (ulong) i); translog_destroy(); @@ -275,9 +297,12 @@ int main(int argc, char *argv[]) lsn_base= lsn; rec_len= get_len(); + parts[TRANSLOG_INTERNAL_PARTS + 0].str= (char*)long_buffer; + parts[TRANSLOG_INTERNAL_PARTS + 0].length= rec_len; if (translog_write_record(&lsn, LOGREC_REDO_INSERT_ROW_HEAD, - (i % 0xFFFF), NULL, rec_len, long_buffer, 0)) + (i % 0xFFFF), NULL, NULL, rec_len, + TRANSLOG_INTERNAL_PARTS + 1, parts)) { fprintf(stderr, "Can't write variable record #%lu\n", (ulong) i); translog_destroy(); diff --git a/storage/maria/unittest/ma_test_loghandler_multithread-t.c b/storage/maria/unittest/ma_test_loghandler_multithread-t.c index 1834e720328..2651258e290 100644 --- a/storage/maria/unittest/ma_test_loghandler_multithread-t.c +++ b/storage/maria/unittest/ma_test_loghandler_multithread-t.c @@ -124,12 +124,16 @@ void writer(int num) { uint len= get_len(); lens[num][i]= len; + LEX_STRING parts[TRANSLOG_INTERNAL_PARTS + 1]; int2store(long_tr_id, num); int4store(long_tr_id + 2, i); + parts[TRANSLOG_INTERNAL_PARTS + 0].str= (char*)long_tr_id; + parts[TRANSLOG_INTERNAL_PARTS + 0].length= 6; if (translog_write_record(&lsn, LOGREC_LONG_TRANSACTION_ID, - num, NULL, 6, long_tr_id, 0)) + num, NULL, NULL, 6, TRANSLOG_INTERNAL_PARTS + 1, + parts)) { fprintf(stderr, "Can't write LOGREC_LONG_TRANSACTION_ID record #%lu " "thread %i\n", (ulong) i, num); @@ -140,9 +144,13 @@ void writer(int num) return; } lsns1[num][i]= lsn; + parts[TRANSLOG_INTERNAL_PARTS + 0].str= (char*)long_buffer; + parts[TRANSLOG_INTERNAL_PARTS + 0].length= len; if (translog_write_record(&lsn, LOGREC_REDO_INSERT_ROW_HEAD, - num, NULL, len, long_buffer, 0)) + num, NULL, NULL, + len, TRANSLOG_INTERNAL_PARTS + 1, + parts)) { fprintf(stderr, "Can't write variable record #%lu\n", (ulong) i); translog_destroy(); @@ -277,14 +285,18 @@ int main(int argc, char **argv __attribute__ ((unused))) srandom(122334817L); { + LEX_STRING parts[TRANSLOG_INTERNAL_PARTS + 1]; byte long_tr_id[6]= { 0x11, 0x22, 0x33, 0x44, 0x55, 0x66 }; + parts[TRANSLOG_INTERNAL_PARTS + 0].str= (char*)long_tr_id; + parts[TRANSLOG_INTERNAL_PARTS + 0].length= 6; if (translog_write_record(&first_lsn, LOGREC_LONG_TRANSACTION_ID, - 0, NULL, 6, long_tr_id, 0)) + 0, NULL, NULL, 6, TRANSLOG_INTERNAL_PARTS + 1, + parts)) { fprintf(stderr, "Can't write the first record\n"); translog_destroy(); diff --git a/storage/maria/unittest/ma_test_loghandler_pagecache-t.c b/storage/maria/unittest/ma_test_loghandler_pagecache-t.c index 9204d531ea1..13b5afe7444 100644 --- a/storage/maria/unittest/ma_test_loghandler_pagecache-t.c +++ b/storage/maria/unittest/ma_test_loghandler_pagecache-t.c @@ -25,6 +25,7 @@ int main(int argc, char *argv[]) PAGECACHE pagecache; LSN lsn; MY_STAT st, *stat; + LEX_STRING parts[TRANSLOG_INTERNAL_PARTS + 1]; MY_INIT(argv[0]); @@ -85,9 +86,12 @@ int main(int argc, char *argv[]) exit(1); } int4store(long_tr_id, 0); + parts[TRANSLOG_INTERNAL_PARTS + 0].str= (char*)long_tr_id; + parts[TRANSLOG_INTERNAL_PARTS + 0].length= 6; if (translog_write_record(&lsn, LOGREC_LONG_TRANSACTION_ID, - 0, NULL, 6, long_tr_id, 0)) + 0, NULL, NULL, 6, TRANSLOG_INTERNAL_PARTS + 1, + parts)) { fprintf(stderr, "Can't write record #%lu\n", (ulong) 0); translog_destroy(); diff --git a/storage/maria/unittest/trnman-t.c b/storage/maria/unittest/trnman-t.c index 7d97794b685..b0a087370f2 100644 --- a/storage/maria/unittest/trnman-t.c +++ b/storage/maria/unittest/trnman-t.c @@ -24,8 +24,11 @@ #include "../trnman.h" pthread_mutex_t rt_mutex; -int rt_num_threads; +pthread_attr_t attr; +size_t stacksize= 0; +#define STACK_SIZE (((int)stacksize-2048)*STACK_DIRECTION) +int rt_num_threads; int litmus; /* @@ -34,11 +37,11 @@ int litmus; #define MAX_ITER 100 pthread_handler_t test_trnman(void *arg) { - int m= (*(int *)arg); uint x, y, i, n; TRN *trn[MAX_ITER]; pthread_mutex_t mutexes[MAX_ITER]; pthread_cond_t conds[MAX_ITER]; + int m= (*(int *)arg); for (i= 0; i < MAX_ITER; i++) { @@ -52,7 +55,7 @@ pthread_handler_t test_trnman(void *arg) m-= n= x % MAX_ITER; for (i= 0; i < n; i++) { - trn[i]= trnman_new_trn(&mutexes[i], &conds[i]); + trn[i]= trnman_new_trn(&mutexes[i], &conds[i], &m + STACK_SIZE); if (!trn[i]) { diag("trnman_new_trn() failed"); @@ -96,7 +99,7 @@ void run_test(const char *test, pthread_handler handler, int n, int m) diag("Testing %s with %d threads, %d iterations... ", test, n, m); rt_num_threads= n; for (i= 0; i < n ; i++) - if (pthread_create(threads+i, 0, handler, &m)) + if (pthread_create(threads+i, &attr, handler, &m)) { diag("Could not create thread"); abort(); @@ -112,7 +115,7 @@ void run_test(const char *test, pthread_handler handler, int n, int m) i= trnman_can_read_from(trn[T1], trn[T2]->trid); \ ok(i == RES, "trn" #T1 " %s read from trn" #T2, i ? "can" : "cannot") #define start_transaction(T) \ - trn[T]= trnman_new_trn(&mutexes[T], &conds[T]) + trn[T]= trnman_new_trn(&mutexes[T], &conds[T], &i + STACK_SIZE) #define commit(T) trnman_commit_trn(trn[T]) #define abort(T) trnman_abort_trn(trn[T]) @@ -161,6 +164,12 @@ int main() return exit_status(); pthread_mutex_init(&rt_mutex, 0); + pthread_attr_init(&attr); +#ifdef HAVE_PTHREAD_ATTR_GETSTACKSIZE + pthread_attr_getstacksize(&attr, &stacksize); + if (stacksize == 0) +#endif + stacksize= PTHREAD_STACK_MIN; #define CYCLES 10000 #define THREADS 10 diff --git a/unittest/mytap/tap.c b/unittest/mytap/tap.c index a99da3fd975..f9396adbd69 100644 --- a/unittest/mytap/tap.c +++ b/unittest/mytap/tap.c @@ -166,9 +166,15 @@ static signal_entry install_signal[]= { #endif }; +int skip_big_tests= 0; + void plan(int const count) { + char *config= getenv("MYTAP_CONFIG"); + + if (config) + skip_big_tests= strcmp(config, "big"); setvbuf(tapout, 0, _IONBF, 0); /* provide output at once */ /* diff --git a/unittest/mytap/tap.h b/unittest/mytap/tap.h index 31ec47d1ef2..f92fad1101f 100644 --- a/unittest/mytap/tap.h +++ b/unittest/mytap/tap.h @@ -61,6 +61,24 @@ typedef struct TEST_DATA { extern "C" { #endif +/** + Defines whether "big" tests should be skipped. + + This variable is set by plan() function unless MYTAP_CONFIG environment + variable is set to the string "big". It is supposed to be used as + + @code + if (skip_big_tests) { + skip(1, "Big test skipped"); + } else { + ok(life_universe_and_everything() == 42, "The answer is CORRECT"); + } + @endcode + + @see SKIP_BIG_TESTS +*/ +extern int skip_big_tests; + /** @defgroup MyTAP_API MyTAP API @@ -81,7 +99,12 @@ extern "C" { that generate a core, so if you want to override these signals, do it after you have called the plan() function. - @param count The planned number of tests to run. + It will also set skip_big_tests variable if MYTAP_CONFIG environment + variable is defined. + + @see skip_big_tests + + @param count The planned number of tests to run. */ void plan(int count); @@ -160,6 +183,24 @@ void skip(int how_many, char const *reason, ...) if (SKIP_IF_TRUE) skip((COUNT),(REASON)); else +/** + Helper macro to skip a group of "big" tests. It is used in the following + manner: + + @code + SKIP_BIG_TESTS(1) + { + ok(life_universe_and_everything() == 42, "The answer is CORRECT"); + } + @endcode + + @see skip_big_tests + */ + +#define SKIP_BIG_TESTS(COUNT) \ + if (skip_big_tests) skip((COUNT), "big test"); else + + /** Print a diagnostics message. diff --git a/unittest/unit.pl b/unittest/unit.pl index 9d328985012..b83132581f9 100644 --- a/unittest/unit.pl +++ b/unittest/unit.pl @@ -14,8 +14,9 @@ # along with this program; if not, write to the Free Software # Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA -use Test::Harness qw(&runtests $verbose); +use Test::Harness; use File::Find; +use Getopt::Long; use strict; @@ -35,6 +36,15 @@ unit - Run unit tests in directory =cut +my $big=1; + +my $result = GetOptions ( + "big!" => \$big, + "verbose!" => \$Test::Harness::verbose, +); + +$ENV{'MYTAP_CONFIG'} = $big ? "big" : ""; + my $cmd = shift; if (defined $cmd && exists $dispatch{$cmd}) { -- cgit v1.2.1 From ffe437be41918062a8936c6d7cc6235cccf84d2e Mon Sep 17 00:00:00 2001 From: unknown Date: Mon, 4 Jun 2007 14:07:18 +0300 Subject: Fixed that ma_test_all works with -T (simple transaction/logging support) Some fixes from Sanja BitKeeper/etc/ignore: added storage/maria/maria_log.* include/pagecache.h: Always have enum PAGECACHE_EMPTY_PAGE available (Simpler code) storage/maria/ma_bitmap.c: Reset 'debugging' bitmap when creating new one (fixes valgrind warning) storage/maria/ma_blockrec.c: Removed duplicate (wrong) initialization Reset not initialized variable storage/maria/ma_check.c: Use right page type (Patch from Sanja) storage/maria/ma_init.c: Reset logging in maria_end() (Fixes memory leak) storage/maria/ma_loghandler.c: Add missing copyright header Added checking of duplicate calls or calls without init to translog_destroy() Don't lock mutex before destroying them (not needed as you can't use a destroyed mutex anyway) storage/maria/ma_pagecache.c: Added extra page type text Trivial indentation fixes storage/maria/ma_test1.c: Added transaction setup (Patch from Sanja) storage/maria/ma_test2.c: Added transaction setup (Patch from Sanja) --- .bzrignore | 2 ++ include/pagecache.h | 7 +++-- storage/maria/ma_bitmap.c | 3 ++ storage/maria/ma_blockrec.c | 3 +- storage/maria/ma_check.c | 2 +- storage/maria/ma_init.c | 2 ++ storage/maria/ma_loghandler.c | 66 ++++++++++++++++++++++++++++--------------- storage/maria/ma_pagecache.c | 14 +++++++-- storage/maria/ma_test1.c | 22 +++++++++++++-- storage/maria/ma_test2.c | 17 ++++++++++- 10 files changed, 103 insertions(+), 35 deletions(-) diff --git a/.bzrignore b/.bzrignore index 0c52723de3c..40623790bc6 100644 --- a/.bzrignore +++ b/.bzrignore @@ -2998,3 +2998,5 @@ storage/maria/unittest/ma_pagecache_consist_64kHC-t-big storage/maria/unittest/ma_pagecache_consist_64kRD-t-big storage/maria/unittest/ma_pagecache_consist_64kWR-t-big storage/maria/unittest/ma_pagecache_single_64k-t-big +storage/maria/maria_control +storage/maria/maria_log.* diff --git a/include/pagecache.h b/include/pagecache.h index 7951e48edb0..aa4ec60412b 100644 --- a/include/pagecache.h +++ b/include/pagecache.h @@ -26,10 +26,11 @@ C_MODE_START /* Type of the page */ enum pagecache_page_type { -#ifndef DBUG_OFF - /* used only for control page type changing during debugging */ + /* + Used only for control page type changing during debugging. This define + should only be using when using DBUG. + */ PAGECACHE_EMPTY_PAGE, -#endif /* the page does not contain LSN */ PAGECACHE_PLAIN_PAGE, /* the page contain LSN (maria tablespace page) */ diff --git a/storage/maria/ma_bitmap.c b/storage/maria/ma_bitmap.c index 923525922da..e1308bce487 100644 --- a/storage/maria/ma_bitmap.c +++ b/storage/maria/ma_bitmap.c @@ -505,6 +505,9 @@ static my_bool _ma_read_bitmap_page(MARIA_SHARE *share, bzero(bitmap->map, bitmap->block_size); memcpy(bitmap->map + share->block_size - 2, maria_bitmap_marker, 2); bitmap->used_size= 0; +#ifndef DBUG_OFF + memcpy(bitmap->map + bitmap->block_size, bitmap->map, bitmap->block_size); +#endif DBUG_RETURN(0); } bitmap->used_size= bitmap->total_size; diff --git a/storage/maria/ma_blockrec.c b/storage/maria/ma_blockrec.c index 8d8adde46d1..e381c505453 100644 --- a/storage/maria/ma_blockrec.c +++ b/storage/maria/ma_blockrec.c @@ -3640,7 +3640,6 @@ static size_t fill_insert_undo_parts(MARIA_HA *info, const byte *record, log_parts++; /* Stored bitmap over packed (zero length or all-zero fields) */ - start_log_parts= log_parts; log_parts->str= info->cur_row.empty_bits; log_parts->length= share->base.pack_bytes; row_length+= log_parts->length; @@ -3800,7 +3799,7 @@ static size_t fill_update_undo_parts(MARIA_HA *info, const byte *oldrec, uchar *field_data, *start_field_data; uchar *old_field_lengths= old_row->field_lengths; uchar *new_field_lengths= new_row->field_lengths; - size_t row_length; + size_t row_length= 0; uint field_count= 0; LEX_STRING *start_log_parts; my_bool new_column_is_empty; diff --git a/storage/maria/ma_check.c b/storage/maria/ma_check.c index 82ad67c2452..8755caf2445 100644 --- a/storage/maria/ma_check.c +++ b/storage/maria/ma_check.c @@ -1632,7 +1632,7 @@ static int check_block_record(HA_CHECK *param, MARIA_HA *info, int extend, &info->dfile, (pos / block_size), 1, page_buff, - PAGECACHE_PLAIN_PAGE, + info->s->page_type, PAGECACHE_LOCK_LEFT_UNLOCKED, 0) == 0) { _ma_check_print_error(param, diff --git a/storage/maria/ma_init.c b/storage/maria/ma_init.c index 8f7cdf291ae..9d72408ad4b 100644 --- a/storage/maria/ma_init.c +++ b/storage/maria/ma_init.c @@ -55,6 +55,8 @@ void maria_end(void) { maria_inited= FALSE; ft_free_stopwords(); + translog_destroy(); + ma_control_file_end(); pthread_mutex_destroy(&THR_LOCK_maria); } } diff --git a/storage/maria/ma_loghandler.c b/storage/maria/ma_loghandler.c index d16595be24e..8ebe17083ce 100644 --- a/storage/maria/ma_loghandler.c +++ b/storage/maria/ma_loghandler.c @@ -1,3 +1,18 @@ +/* Copyright (C) 2007 MySQL AB & Sanja Belkin + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ + #include "maria_def.h" #include "ma_blockrec.h" @@ -148,6 +163,8 @@ static struct st_translog_descriptor log_descriptor; /* Marker for end of log */ static byte end_of_log= 0; +static my_bool translog_inited; + /* record classes */ enum record_class { @@ -288,7 +305,7 @@ static LOG_DESC INIT_LOGREC_UNDO_ROW_DELETE= static LOG_DESC INIT_LOGREC_UNDO_ROW_UPDATE= {LOGRECTYPE_VARIABLE_LENGTH, 0, LSN_STORE_SIZE + FILEID_STORE_SIZE + PAGE_STORE_SIZE + DIRPOS_STORE_SIZE, - NULL, NULL, NULL, 2}; + NULL, NULL, NULL, 1}; static LOG_DESC INIT_LOGREC_UNDO_ROW_PURGE= {LOGRECTYPE_PSEUDOFIXEDLENGTH, LSN_STORE_SIZE, LSN_STORE_SIZE, @@ -1861,6 +1878,9 @@ static uint16 translog_get_chunk_header_length(byte *page, uint16 offset) flags flags (TRANSLOG_PAGE_CRC, TRANSLOG_SECTOR_PROTECTION TRANSLOG_RECORD_CRC) + TODO + Free used resources in case of error. + RETURN 0 OK 1 Error @@ -1877,7 +1897,7 @@ my_bool translog_init(const char *directory, TRANSLOG_ADDRESS sure_page, last_page, last_valid_page; DBUG_ENTER("translog_init"); - loghandler_init(); + loghandler_init(); /* Safe to do many times */ if (pthread_mutex_init(&log_descriptor.sent_to_file_lock, MY_MUTEX_INIT_FAST)) @@ -2164,6 +2184,7 @@ my_bool translog_init(const char *directory, log_descriptor.flushed--; /* offset decreased */ log_descriptor.sent_to_file--; /* offset decreased */ + translog_inited= 1; DBUG_RETURN(0); } @@ -2198,8 +2219,6 @@ static void translog_buffer_destroy(struct st_translog_buffer *buffer) */ translog_buffer_flush(buffer); } - DBUG_PRINT("info", ("Unlock mutex: 0x%lx", (ulong) &buffer->mutex)); - pthread_mutex_unlock(&buffer->mutex); DBUG_PRINT("info", ("Destroy mutex: 0x%lx", (ulong) &buffer->mutex)); pthread_mutex_destroy(&buffer->mutex); DBUG_VOID_RETURN; @@ -2217,27 +2236,28 @@ void translog_destroy() { uint i; DBUG_ENTER("translog_destroy"); - if (log_descriptor.bc.buffer->file != -1) - translog_finish_page(&log_descriptor.horizon, &log_descriptor.bc); - - for (i= 0; i < TRANSLOG_BUFFERS_NO; i++) + + if (translog_inited) { - struct st_translog_buffer *buffer= log_descriptor.buffers + i; - /* - Lock the buffer just for safety, there should not be other - threads running. - */ - translog_buffer_lock(buffer); - translog_buffer_destroy(buffer); - } - /* close files */ - for (i= 0; i < OPENED_FILES_NUM; i++) - { - if (log_descriptor.log_file_num[i] != -1) - translog_close_log_file(log_descriptor.log_file_num[i]); + if (log_descriptor.bc.buffer->file != -1) + translog_finish_page(&log_descriptor.horizon, &log_descriptor.bc); + + for (i= 0; i < TRANSLOG_BUFFERS_NO; i++) + { + struct st_translog_buffer *buffer= log_descriptor.buffers + i; + translog_buffer_destroy(buffer); + } + + /* close files */ + for (i= 0; i < OPENED_FILES_NUM; i++) + { + if (log_descriptor.log_file_num[i] != -1) + translog_close_log_file(log_descriptor.log_file_num[i]); + } + pthread_mutex_destroy(&log_descriptor.sent_to_file_lock); + my_close(log_descriptor.directory_fd, MYF(MY_WME)); + translog_inited= 0; } - pthread_mutex_destroy(&log_descriptor.sent_to_file_lock); - my_close(log_descriptor.directory_fd, MYF(MY_WME)); DBUG_VOID_RETURN; } diff --git a/storage/maria/ma_pagecache.c b/storage/maria/ma_pagecache.c index f0c1d674f4b..96fc9704054 100755 --- a/storage/maria/ma_pagecache.c +++ b/storage/maria/ma_pagecache.c @@ -169,15 +169,19 @@ enum PCBLOCK_TEMPERATURE { PCBLOCK_COLD /*free*/ , PCBLOCK_WARM , PCBLOCK_HOT }; #ifndef DBUG_OFF static const char *page_cache_page_type_str[]= { + /* used only for control page type changing during debugging */ + "EMPTY", "PLAIN", "LSN" }; + static const char *page_cache_page_write_mode_str[]= { "DELAY", "NOW", "DONE" }; + static const char *page_cache_page_lock_str[]= { "free -> free", @@ -189,6 +193,7 @@ static const char *page_cache_page_lock_str[]= "write -> free", "write -> read" }; + static const char *page_cache_page_pin_str[]= { "pinned -> pinned", @@ -196,17 +201,19 @@ static const char *page_cache_page_pin_str[]= "unpinned -> pinned", "pinned -> unpinned" }; -#endif -#ifndef DBUG_OFF + + typedef struct st_pagecache_pin_info { struct st_pagecache_pin_info *next, **prev; struct st_my_thread_var *thread; } PAGECACHE_PIN_INFO; + /* st_pagecache_lock_info structure should be kept in next, prev, thread part compatible with st_pagecache_pin_info to be compatible in functions. */ + typedef struct st_pagecache_lock_info { struct st_pagecache_lock_info *next, **prev; @@ -275,7 +282,8 @@ static PAGECACHE_PIN_INFO *info_find(PAGECACHE_PIN_INFO *list, return i; return 0; } -#endif + +#endif /* !DBUG_OFF */ /* page cache block */ struct st_pagecache_block_link diff --git a/storage/maria/ma_test1.c b/storage/maria/ma_test1.c index 98d55c7d254..3546521e0d1 100644 --- a/storage/maria/ma_test1.c +++ b/storage/maria/ma_test1.c @@ -18,6 +18,11 @@ #include "maria.h" #include #include +#include "ma_control_file.h" +#include "ma_loghandler.h" + +extern PAGECACHE *maria_log_pagecache; +extern const char *maria_data_root; #define MAX_REC_LENGTH 1024 @@ -49,10 +54,23 @@ int main(int argc,char *argv[]) { MY_INIT(argv[0]); my_init(); - maria_init(); get_options(argc,argv); + maria_data_root= "."; /* Maria requires that we always have a page cache */ - init_pagecache(maria_pagecache, IO_SIZE*16, 0, 0, maria_block_size); + if (maria_init() || + (init_pagecache(maria_pagecache, IO_SIZE*16, 0, 0, + maria_block_size) == 0) || + ma_control_file_create_or_open() || + (init_pagecache(maria_log_pagecache, + TRANSLOG_PAGECACHE_SIZE, 0, 0, + TRANSLOG_PAGE_SIZE) == 0) || + translog_init(maria_data_root, TRANSLOG_FILE_SIZE, + 0, 0, maria_log_pagecache, + TRANSLOG_DEFAULT_FLAGS)) + { + fprintf(stderr, "Error in initialization"); + exit(1); + } exit(run_test("test1")); } diff --git a/storage/maria/ma_test2.c b/storage/maria/ma_test2.c index cde8da08dca..bbbb4fca1bf 100644 --- a/storage/maria/ma_test2.c +++ b/storage/maria/ma_test2.c @@ -28,6 +28,7 @@ #include #include + #define STANDARD_LENGTH 37 #define MARIA_KEYS 6 #define MAX_PARTS 4 @@ -219,8 +220,22 @@ int main(int argc, char *argv[]) goto err; if (!silent) printf("- Writing key:s\n"); + maria_data_root= "."; /* Maria requires that we always have a page cache */ - init_pagecache(maria_pagecache, pagecache_size, 0, 0, maria_block_size); + if ((init_pagecache(maria_pagecache, pagecache_size, 0, 0, + maria_block_size) == 0) || + ma_control_file_create_or_open() || + (init_pagecache(maria_log_pagecache, + TRANSLOG_PAGECACHE_SIZE, 0, 0, + TRANSLOG_PAGE_SIZE) == 0) || + translog_init(maria_data_root, TRANSLOG_FILE_SIZE, + 0, 0, maria_log_pagecache, + TRANSLOG_DEFAULT_FLAGS)) + { + fprintf(stderr, "Error in initialization"); + exit(1); + } + if (locking) maria_lock_database(file,F_WRLCK); if (write_cacheing) -- cgit v1.2.1 From ef00dbaa7e6335ff1968642f8f6b07e9442feaeb Mon Sep 17 00:00:00 2001 From: unknown Date: Tue, 5 Jun 2007 12:14:05 +0300 Subject: postmerge changes --- include/pagecache.h | 12 ++++++------ storage/maria/ma_blockrec.c | 5 +++-- storage/maria/ma_pagecache.c | 6 +++--- 3 files changed, 12 insertions(+), 11 deletions(-) diff --git a/include/pagecache.h b/include/pagecache.h index 325347cd90c..12906676959 100644 --- a/include/pagecache.h +++ b/include/pagecache.h @@ -213,7 +213,7 @@ extern void pagecache_unlock_by_link(PAGECACHE *pagecache, PAGECACHE_PAGE_LINK *link, enum pagecache_page_lock lock, enum pagecache_page_pin pin, - pgcache_page_no_t pageno, + LSN first_REDO_LSN_for_page, LSN lsn); extern void pagecache_unpin(PAGECACHE *pagecache, PAGECACHE_FILE *file, @@ -225,11 +225,11 @@ extern void pagecache_unpin_by_link(PAGECACHE *pagecache, extern int flush_pagecache_blocks(PAGECACHE *keycache, PAGECACHE_FILE *file, enum flush_type type); -extern my_bool pagecache_delete_page(PAGECACHE *pagecache, - PAGECACHE_FILE *file, - pgcache_page_no_t pageno, - enum pagecache_page_lock lock, - my_bool flush); +extern my_bool pagecache_delete(PAGECACHE *pagecache, + PAGECACHE_FILE *file, + pgcache_page_no_t pageno, + enum pagecache_page_lock lock, + my_bool flush); extern my_bool pagecache_delete_pages(PAGECACHE *pagecache, PAGECACHE_FILE *file, pgcache_page_no_t pageno, diff --git a/storage/maria/ma_blockrec.c b/storage/maria/ma_blockrec.c index e381c505453..41599f39810 100644 --- a/storage/maria/ma_blockrec.c +++ b/storage/maria/ma_blockrec.c @@ -571,8 +571,9 @@ void _ma_unpin_all_pages(MARIA_HA *info, LSN undo_lsn) DBUG_ASSERT(undo_lsn != 0 || info->s->base.transactional == 0); while (pinned_page-- != page_link) - pagecache_unlock(info->s->pagecache, pinned_page->link, - pinned_page->unlock, PAGECACHE_UNPIN, 0, undo_lsn); + pagecache_unlock_by_link(info->s->pagecache, pinned_page->link, + pinned_page->unlock, PAGECACHE_UNPIN, + 0, undo_lsn); info->pinned_pages.elements= 0; DBUG_VOID_RETURN; diff --git a/storage/maria/ma_pagecache.c b/storage/maria/ma_pagecache.c index 503ef46c684..5b713e6797f 100755 --- a/storage/maria/ma_pagecache.c +++ b/storage/maria/ma_pagecache.c @@ -2258,7 +2258,7 @@ static my_bool make_lock_and_pin(PAGECACHE *pagecache, block->pins, page_cache_page_lock_str[lock], page_cache_page_pin_str[pin])); - BLOCK_INFO(block); + PCBLOCK_INFO(block); } #endif @@ -3067,8 +3067,8 @@ my_bool pagecache_delete_pages(PAGECACHE *pagecache, page_end= pageno + page_count; do { - if (pagecache_delete_page(pagecache, file, pageno, - lock, flush)) + if (pagecache_delete(pagecache, file, pageno, + lock, flush)) DBUG_RETURN(1); } while (++pageno != page_end); DBUG_RETURN(0); -- cgit v1.2.1 From 4423b8bfce304f94624daa754c130f280d2e16d2 Mon Sep 17 00:00:00 2001 From: unknown Date: Tue, 5 Jun 2007 14:35:34 +0300 Subject: Fixes to get maria.test and ps_maria.test to work storage/maria/ma_blockrec.c: Reset undo_lsn if not transactional table (Avoids assert() when checking LSN in pagecache code) storage/maria/ma_create.c: Don't convert simple FIXED size tables to BLOCK format. storage/maria/trnman.c: Reset undo_lsn --- storage/maria/ma_blockrec.c | 6 ++++- storage/maria/ma_create.c | 56 ++++++++++++++++++++++++++++----------------- storage/maria/trnman.c | 9 +++++++- 3 files changed, 48 insertions(+), 23 deletions(-) diff --git a/storage/maria/ma_blockrec.c b/storage/maria/ma_blockrec.c index e381c505453..aba609125ca 100644 --- a/storage/maria/ma_blockrec.c +++ b/storage/maria/ma_blockrec.c @@ -568,7 +568,10 @@ void _ma_unpin_all_pages(MARIA_HA *info, LSN undo_lsn) DBUG_PRINT("info", ("undo_lsn: %lu", (ulong) undo_lsn)); /* True if not disk error */ - DBUG_ASSERT(undo_lsn != 0 || info->s->base.transactional == 0); + DBUG_ASSERT(undo_lsn != 0 || !info->s->base.transactional); + + if (!info->s->base.transactional) + undo_lsn= 0; /* Avoid assert in key cache */ while (pinned_page-- != page_link) pagecache_unlock(info->s->pagecache, pinned_page->link, @@ -2623,6 +2626,7 @@ my_bool _ma_delete_block_record(MARIA_HA *info, const byte *record) if (info->cur_row.extents && free_full_pages(info, &info->cur_row)) goto err; + if (info->s->base.transactional) { uchar log_data[LSN_STORE_SIZE + FILEID_STORE_SIZE + PAGE_STORE_SIZE + DIR_COUNT_SIZE]; diff --git a/storage/maria/ma_create.c b/storage/maria/ma_create.c index c38471e06a0..5dd6e0e1f93 100644 --- a/storage/maria/ma_create.c +++ b/storage/maria/ma_create.c @@ -48,7 +48,7 @@ int maria_create(const char *name, enum data_file_type datafile_type, uint length,max_key_length,packed,pack_bytes,pointer,real_length_diff, key_length,info_length,key_segs,options,min_key_length_skip, base_pos,long_varchar_count,varchar_length, - unique_key_parts,fulltext_keys,offset; + unique_key_parts,fulltext_keys,offset, not_block_record_extra_length; uint max_field_lengths, extra_header_size; ulong reclength, real_reclength,min_pack_length; char filename[FN_REFLEN],linkname[FN_REFLEN], *linkname_ptr; @@ -65,6 +65,7 @@ int maria_create(const char *name, enum data_file_type datafile_type, my_off_t key_root[HA_MAX_POSSIBLE_KEY]; MARIA_CREATE_INFO tmp_create_info; my_bool tmp_table= FALSE; /* cache for presence of HA_OPTION_TMP_TABLE */ + my_bool forced_packed; myf sync_dir= MY_SYNC_DIR; DBUG_ENTER("maria_create"); DBUG_PRINT("enter", ("keys: %u columns: %u uniques: %u flags: %u", @@ -114,9 +115,10 @@ int maria_create(const char *name, enum data_file_type datafile_type, /* Start by checking fields and field-types used */ - varchar_length=long_varchar_count=packed= + varchar_length=long_varchar_count=packed= not_block_record_extra_length= pack_reclength= max_field_lengths= 0; reclength= min_pack_length= ci->null_bytes; + forced_packed= 0; for (column= columndef, end_column= column + columns ; column != end_column ; @@ -139,6 +141,7 @@ int maria_create(const char *name, enum data_file_type datafile_type, column->empty_bit= (1 << (packed & 7)); if (type == FIELD_BLOB) { + forced_packed= 1; packed++; share.base.blobs++; if (pack_reclength != INT_MAX32) @@ -157,17 +160,16 @@ int maria_create(const char *name, enum data_file_type datafile_type, else if (type == FIELD_SKIP_PRESPACE || type == FIELD_SKIP_ENDSPACE) { + forced_packed= 1; max_field_lengths+= column->length > 255 ? 2 : 1; - if (datafile_type != BLOCK_RECORD) - min_pack_length++; + not_block_record_extra_length++; packed++; } else if (type == FIELD_VARCHAR) { varchar_length+= column->length-1; /* Used for min_pack_length */ pack_reclength++; - if (datafile_type != BLOCK_RECORD) - min_pack_length++; + not_block_record_extra_length++; max_field_lengths++; packed++; column->fill_length= 1; @@ -183,23 +185,47 @@ int maria_create(const char *name, enum data_file_type datafile_type, packed++; else { - if (datafile_type != BLOCK_RECORD || !column->null_bit) + if (!column->null_bit) min_pack_length+= column->length; + else + not_block_record_extra_length+= column->length; column->empty_pos= 0; column->empty_bit= 0; } } else /* FIELD_NORMAL */ { - if (datafile_type != BLOCK_RECORD || !column->null_bit) - min_pack_length+= column->length; if (!column->null_bit) { + min_pack_length+= column->length; share.base.fixed_not_null_fields++; share.base.fixed_not_null_fields_length+= column->length; } + else + not_block_record_extra_length+= column->length; } } + + if (datafile_type == STATIC_RECORD && forced_packed) + { + /* Can't use fixed length records, revert to block records */ + datafile_type= BLOCK_RECORD; + } + + if (datafile_type == DYNAMIC_RECORD) + options|= HA_OPTION_PACK_RECORD; /* Must use packed records */ + + if (datafile_type == STATIC_RECORD) + { + /* We can't use checksum with static length rows */ + flags&= ~HA_CREATE_CHECKSUM; + options&= ~HA_OPTION_CHECKSUM; + min_pack_length+= varchar_length; + packed= 0; + } + if (datafile_type != BLOCK_RECORD) + min_pack_length+= not_block_record_extra_length; + if ((packed & 7) == 1) { /* @@ -229,18 +255,6 @@ int maria_create(const char *name, enum data_file_type datafile_type, if (pack_reclength != INT_MAX32) pack_reclength+= max_field_lengths + long_varchar_count; - if (packed && datafile_type == STATIC_RECORD) - datafile_type= BLOCK_RECORD; - if (datafile_type == DYNAMIC_RECORD) - options|= HA_OPTION_PACK_RECORD; /* Must use packed records */ - - if (datafile_type == STATIC_RECORD) - { - /* We can't use checksum with static length rows */ - flags&= ~HA_CREATE_CHECKSUM; - options&= ~HA_OPTION_CHECKSUM; - min_pack_length+= varchar_length; - } if (flags & HA_CREATE_TMP_TABLE) { options|= HA_OPTION_TMP_TABLE; diff --git a/storage/maria/trnman.c b/storage/maria/trnman.c index 9b8c36f9769..8f52f1e2825 100644 --- a/storage/maria/trnman.c +++ b/storage/maria/trnman.c @@ -253,7 +253,13 @@ TRN *trnman_new_trn(pthread_mutex_t *mutex, pthread_cond_t *cond, /* Nothing in the pool ? Allocate a new one */ if (!trn) { - trn= (TRN *)my_malloc(sizeof(TRN), MYF(MY_WME)); + /* + trn should be completely initalized at create time to allow + one to keep a known state on it. + (Like redo_lns, which is assumed to be 0 at start of row handling + and reset to zero before end of row handling) + */ + trn= (TRN *)my_malloc(sizeof(TRN), MYF(MY_WME | MY_ZEROFILL)); if (unlikely(!trn)) { DBUG_PRINT("info", ("pthread_mutex_unlock LOCK_trn_list")); @@ -286,6 +292,7 @@ TRN *trnman_new_trn(pthread_mutex_t *mutex, pthread_cond_t *cond, trn->min_read_from= trn->trid; trn->commit_trid= 0; + trn->undo_lsn= 0; trn->locks.mutex= mutex; trn->locks.cond= cond; -- cgit v1.2.1 From 75e52671d771a51bfea53082139cd88c2fba1480 Mon Sep 17 00:00:00 2001 From: unknown Date: Tue, 5 Jun 2007 16:06:08 +0200 Subject: maria cannot rollback yet --- mysql-test/r/maria.result | 19 +++++++++++++++++++ mysql-test/t/maria.test | 16 ++++++++++++++++ storage/maria/ha_maria.cc | 12 +++++++++++- 3 files changed, 46 insertions(+), 1 deletion(-) diff --git a/mysql-test/r/maria.result b/mysql-test/r/maria.result index 93d8863f40b..3ec9af0fffa 100644 --- a/mysql-test/r/maria.result +++ b/mysql-test/r/maria.result @@ -2,6 +2,25 @@ set global storage_engine=maria; set session storage_engine=maria; drop table if exists t1,t2; SET SQL_WARNINGS=1; +RESET MASTER; +set binlog_format=statement; +CREATE TABLE t1 (a int primary key); +insert t1 values (1),(2),(3); +insert t1 values (4),(2),(5); +ERROR 23000: Duplicate entry '2' for key 'PRIMARY' +select * from t1; +a +1 +2 +3 +4 +SHOW BINLOG EVENTS FROM 102; +Log_name Pos Event_type Server_id End_log_pos Info +master-bin.000001 102 Query 1 200 use `test`; CREATE TABLE t1 (a int primary key) +master-bin.000001 200 Query 1 291 use `test`; insert t1 values (1),(2),(3) +master-bin.000001 291 Query 1 382 use `test`; insert t1 values (4),(2),(5) +drop table t1; +set binlog_format=default; CREATE TABLE t1 ( STRING_DATA char(255) default NULL, KEY string_data (STRING_DATA) diff --git a/mysql-test/t/maria.test b/mysql-test/t/maria.test index e9a0e146475..763abbd9d25 100644 --- a/mysql-test/t/maria.test +++ b/mysql-test/t/maria.test @@ -15,6 +15,22 @@ drop table if exists t1,t2; --enable_warnings SET SQL_WARNINGS=1; +# +# UNIQUE key test +# +# as long as maria cannot rollback, binlog should contain both inserts +# +RESET MASTER; +set binlog_format=statement; +CREATE TABLE t1 (a int primary key); +insert t1 values (1),(2),(3); +--error 1582 +insert t1 values (4),(2),(5); +select * from t1; +SHOW BINLOG EVENTS FROM 102; +drop table t1; +set binlog_format=default; + # # Test problem with CHECK TABLE; # diff --git a/storage/maria/ha_maria.cc b/storage/maria/ha_maria.cc index ecb966a4fbd..e518261ec49 100644 --- a/storage/maria/ha_maria.cc +++ b/storage/maria/ha_maria.cc @@ -31,6 +31,11 @@ #include "ma_rt_index.h" #include "ma_blockrec.h" +#define MARIA_CANNOT_ROLLBACK HA_NO_TRANSACTIONS +#ifdef MARIA_CANNOT_ROLLBACK +#define trans_register_ha(A, B, C) do { /* nothing */ } while(0) +#endif + ulong maria_recover_options= HA_RECOVER_NONE; static handlerton *maria_hton; @@ -466,7 +471,7 @@ ha_maria::ha_maria(handlerton *hton, TABLE_SHARE *table_arg): handler(hton, table_arg), file(0), int_table_flags(HA_NULL_IN_KEY | HA_CAN_FULLTEXT | HA_CAN_SQL_HANDLER | HA_DUPLICATE_POS | HA_CAN_INDEX_BLOBS | HA_AUTO_PART_KEY | - HA_FILE_BASED | HA_CAN_GEOMETRY | + HA_FILE_BASED | HA_CAN_GEOMETRY | MARIA_CANNOT_ROLLBACK | HA_CAN_INSERT_DELAYED | HA_CAN_BIT_FIELD | HA_CAN_RTREEKEYS | HA_HAS_RECORDS | HA_STATS_RECORDS_IS_EXACT), can_enable_indexes(1) @@ -1885,12 +1890,17 @@ int ha_maria::external_lock(THD *thd, int lock_type) if (!trnman_decrement_locked_tables(trn)) { /* autocommit ? rollback a transaction */ +#ifdef MARIA_CANNOT_ROLLBACK + trnman_commit_trn(trn); + THD_TRN= 0; +#else if (!(thd->options & (OPTION_NOT_AUTOCOMMIT | OPTION_BEGIN))) { trnman_rollback_trn(trn); DBUG_PRINT("info", ("THD_TRN set to 0x0")); THD_TRN= 0; } +#endif } } } -- cgit v1.2.1 From aa40e3589ff5dee84c22e26872c45a88b63ba15e Mon Sep 17 00:00:00 2001 From: unknown Date: Tue, 5 Jun 2007 18:34:24 +0200 Subject: fix unit tests to build unittest/mysys/Makefile.am: build unit tests (they were lost in a bad merge) unittest/mysys/my_atomic-t.c: update to the new lf_pinbox* API --- unittest/mysys/Makefile.am | 1 + unittest/mysys/my_atomic-t.c | 25 ++++++++++++++++++------- 2 files changed, 19 insertions(+), 7 deletions(-) diff --git a/unittest/mysys/Makefile.am b/unittest/mysys/Makefile.am index 229e8f0339b..2802844cad5 100644 --- a/unittest/mysys/Makefile.am +++ b/unittest/mysys/Makefile.am @@ -16,6 +16,7 @@ INCLUDES = @ZLIB_INCLUDES@ -I$(top_builddir)/include \ -I$(top_srcdir)/include -I$(top_srcdir)/unittest/mytap +noinst_PROGRAMS = bitmap-t base64-t my_atomic-t LDADD = $(top_builddir)/unittest/mytap/libmytap.a \ $(top_builddir)/mysys/libmysys.a \ diff --git a/unittest/mysys/my_atomic-t.c b/unittest/mysys/my_atomic-t.c index d3be33f4163..b0b1ed7fce6 100644 --- a/unittest/mysys/my_atomic-t.c +++ b/unittest/mysys/my_atomic-t.c @@ -26,6 +26,10 @@ my_atomic_rwlock_t rwl; LF_ALLOCATOR lf_allocator; LF_HASH lf_hash; +pthread_attr_t attr; +size_t stacksize= 0; +#define STACK_SIZE (((int)stacksize-2048)*STACK_DIRECTION) + /* add and sub a random number in a loop. Must get 0 at the end */ pthread_handler_t test_atomic_add_handler(void *arg) { @@ -119,12 +123,12 @@ pthread_handler_t test_lf_pinbox(void *arg) int32 x= 0; LF_PINS *pins; - pins= lf_pinbox_get_pins(&lf_allocator.pinbox); + pins= lf_pinbox_get_pins(&lf_allocator.pinbox, &m + STACK_SIZE); for (x= ((int)(intptr)(&m)); m ; m--) { lf_pinbox_put_pins(pins); - pins= lf_pinbox_get_pins(&lf_allocator.pinbox); + pins= lf_pinbox_get_pins(&lf_allocator.pinbox, &m + STACK_SIZE); } lf_pinbox_put_pins(pins); return 0; @@ -141,7 +145,7 @@ pthread_handler_t test_lf_alloc(void *arg) int32 x,y= 0; LF_PINS *pins; - pins= lf_alloc_get_pins(&lf_allocator); + pins= lf_alloc_get_pins(&lf_allocator, &m + STACK_SIZE); for (x= ((int)(intptr)(&m)); m ; m--) { @@ -165,9 +169,9 @@ pthread_handler_t test_lf_alloc(void *arg) if (my_atomic_add32(&N, -1) == 1) { diag("%d mallocs, %d pins in stack", - lf_allocator.mallocs, lf_allocator.pinbox.pins_in_stack); + lf_allocator.mallocs, lf_allocator.pinbox.pins_in_array); #ifdef MY_LF_EXTRA_DEBUG - a32|= lf_allocator.mallocs - lf_alloc_in_pool(&lf_allocator); + a32|= lf_allocator.mallocs - lf_alloc_pool_count(&lf_allocator); #endif } my_atomic_rwlock_wrunlock(&rwl); @@ -181,7 +185,7 @@ pthread_handler_t test_lf_hash(void *arg) int32 x,y,z,sum= 0, ins= 0; LF_PINS *pins; - pins= lf_hash_get_pins(&lf_hash); + pins= lf_hash_get_pins(&lf_hash, &m + STACK_SIZE); for (x= ((int)(intptr)(&m)); m ; m--) { @@ -213,7 +217,7 @@ pthread_handler_t test_lf_hash(void *arg) if (my_atomic_add32(&N, -1) == 1) { diag("%d mallocs, %d pins in stack, %d hash size, %d inserts", - lf_hash.alloc.mallocs, lf_hash.alloc.pinbox.pins_in_stack, + lf_hash.alloc.mallocs, lf_hash.alloc.pinbox.pins_in_array, lf_hash.size, b32); a32|= lf_hash.count; } @@ -273,6 +277,13 @@ int main() lf_hash_init(&lf_hash, sizeof(int), LF_HASH_UNIQUE, 0, sizeof(int), 0, &my_charset_bin); + pthread_attr_init(&attr); +#ifdef HAVE_PTHREAD_ATTR_GETSTACKSIZE + pthread_attr_getstacksize(&attr, &stacksize); + if (stacksize == 0) +#endif + stacksize= PTHREAD_STACK_MIN; + #ifdef MY_ATOMIC_MODE_RWLOCKS #define CYCLES 3000 #else -- cgit v1.2.1 From 1a7ee974d4bcc38144a194a64ad2b924db167d0d Mon Sep 17 00:00:00 2001 From: unknown Date: Wed, 6 Jun 2007 09:51:30 +0300 Subject: Fixed pagecache unittest storage/maria/ma_pagecache.c: - remove_reader() call is removed from unlock/unpin operations which uses direct link (because find_block() was not called) - patch which broke pagecache unittest is reverted --- storage/maria/ma_pagecache.c | 18 ++++-------------- 1 file changed, 4 insertions(+), 14 deletions(-) diff --git a/storage/maria/ma_pagecache.c b/storage/maria/ma_pagecache.c index 5b713e6797f..877fcabd54b 100755 --- a/storage/maria/ma_pagecache.c +++ b/storage/maria/ma_pagecache.c @@ -2689,7 +2689,6 @@ void pagecache_unlock_by_link(PAGECACHE *pagecache, if (make_lock_and_pin(pagecache, block, lock, pin)) DBUG_ASSERT(0); /* should not happend */ - remove_reader(block); /* Link the block into the LRU chain if it's the last submitted request for the block and block will not be pinned. @@ -2754,7 +2753,6 @@ void pagecache_unpin_by_link(PAGECACHE *pagecache, PAGECACHE_UNPIN)) DBUG_ASSERT(0); /* should not happend */ - remove_reader(block); /* Link the block into the LRU chain if it's the last submitted request for the block and block will not be pinned. @@ -2891,16 +2889,14 @@ restart: #endif } + remove_reader(block); /* Link the block into the LRU chain if it's the last submitted request for the block and block will not be pinned. See NOTE for pagecache_unlock about registering requests. */ if (pin == PAGECACHE_PIN_LEFT_UNPINNED || pin == PAGECACHE_UNPIN) - { - remove_reader(block); unreg_request(pagecache, block, 1); - } else *link= (PAGECACHE_PAGE_LINK)block; @@ -3286,20 +3282,14 @@ restart: DBUG_ASSERT(0); } + /* Unregister the request */ + DBUG_ASSERT(block->hash_link->requests > 0); + block->hash_link->requests--; /* See NOTE for pagecache_unlock about registering requests. */ if (pin == PAGECACHE_PIN_LEFT_UNPINNED || pin == PAGECACHE_UNPIN) - { - /* Unregister the request */ - DBUG_ASSERT(block->hash_link->requests > 0); - block->hash_link->requests--; unreg_request(pagecache, block, 1); - } else - { - if (pin == PAGECACHE_PIN_LEFT_PINNED) - block->hash_link->requests--; *link= (PAGECACHE_PAGE_LINK)block; - } if (block->status & PCBLOCK_ERROR) error= 1; -- cgit v1.2.1 From 42cde8a7b28902e52e4b8339dedb110476e43e6e Mon Sep 17 00:00:00 2001 From: unknown Date: Thu, 7 Jun 2007 01:01:43 +0300 Subject: rec_lsn (first REDO LSN( is now given to the page cache on unpinning Added maria_clone(), needed by future REPAIR code storage/maria/unittest/ma_pagecache_consist.c: Change mode to -rw-rw-r-- storage/maria/unittest/lockman-t.c: Change mode to -rw-rw-r-- storage/maria/unittest/lockman1-t.c: Change mode to -rw-rw-r-- storage/maria/unittest/lockman2-t.c: Change mode to -rw-rw-r-- storage/maria/unittest/trnman-t.c: Change mode to -rw-rw-r-- include/maria.h: Added prototype for maria_clone (for future) storage/maria/ha_maria.cc: Move filename to share structure storage/maria/ma_blockrec.c: rec_lsn (first REDO LSN( is now given to the page cache on unpinning Removed impossible lock handling in get_head_or_tail_page() Changed calls ot translog_write_record() to remember rec_lsn Removed some logging in csse of not transactions storage/maria/ma_delete.c: info->filename -> info->s->open_file_name storage/maria/ma_loghandler.c: Indentation fixes storage/maria/ma_open.c: Added maria_clone(), needed by future REPAIR code storage/maria/ma_packrec.c: Fixed typo in comment storage/maria/ma_pagecache.c: Added comment. Allow setting REC_LSN in case of read lock storage/maria/ma_update.c: info->filename -> info->s->open_file_name storage/maria/ma_write.c: info->filename -> info->s->open_file_name storage/maria/maria_def.h: info->filename -> info->s->open_file_name Added have_rtree to simplify test in ma_clone() storage/maria/maria_ftdump.c: info->filename -> info->s->open_file_name storage/maria/maria_pack.c: info->filename -> info->s->open_file_name storage/maria/trnman.h: Added rec_lsn --- include/maria.h | 1 + storage/maria/ha_maria.cc | 6 +- storage/maria/ma_blockrec.c | 65 +++--- storage/maria/ma_delete.c | 4 +- storage/maria/ma_loghandler.c | 5 +- storage/maria/ma_open.c | 314 +++++++++++++++----------- storage/maria/ma_packrec.c | 2 +- storage/maria/ma_pagecache.c | 8 +- storage/maria/ma_update.c | 4 +- storage/maria/ma_write.c | 4 +- storage/maria/maria_def.h | 8 +- storage/maria/maria_ftdump.c | 2 +- storage/maria/maria_pack.c | 9 +- storage/maria/trnman.h | 2 +- storage/maria/unittest/ma_pagecache_consist.c | 0 15 files changed, 249 insertions(+), 185 deletions(-) mode change 100755 => 100644 storage/maria/unittest/ma_pagecache_consist.c diff --git a/include/maria.h b/include/maria.h index da0cb4ba0e5..fbf4bc68c29 100644 --- a/include/maria.h +++ b/include/maria.h @@ -263,6 +263,7 @@ extern int maria_close(struct st_maria_info *file); extern int maria_delete(struct st_maria_info *file, const byte *buff); extern struct st_maria_info *maria_open(const char *name, int mode, uint wait_if_locked); +extern struct st_maria_info *maria_clone(struct st_maria_share *share, int mode); extern int maria_panic(enum ha_panic_function function); extern int maria_rfirst(struct st_maria_info *file, byte *buf, int inx); extern int maria_rkey(struct st_maria_info *file, byte *buf, int inx, diff --git a/storage/maria/ha_maria.cc b/storage/maria/ha_maria.cc index ecb966a4fbd..f4173ce439b 100644 --- a/storage/maria/ha_maria.cc +++ b/storage/maria/ha_maria.cc @@ -1058,7 +1058,7 @@ int ha_maria::repair(THD *thd, HA_CHECK ¶m, bool do_optimize) param.thd= thd; param.tmpdir= &mysql_tmpdir_list; param.out_flag= 0; - strmov(fixed_name, file->filename); + strmov(fixed_name, file->s->open_file_name); #ifndef TO_BE_FIXED /* QQ: Until we have repair for block format, lie that it succeded */ @@ -1793,11 +1793,11 @@ int ha_maria::info(uint flag) if table is symlinked (Ie; Real name is not same as generated name) */ data_file_name= index_file_name= 0; - fn_format(name_buff, file->filename, "", MARIA_NAME_DEXT, + fn_format(name_buff, file->s->open_file_name, "", MARIA_NAME_DEXT, MY_APPEND_EXT | MY_UNPACK_FILENAME); if (strcmp(name_buff, maria_info.data_file_name)) data_file_name=maria_info.data_file_name; - fn_format(name_buff, file->filename, "", MARIA_NAME_IEXT, + fn_format(name_buff, file->s->open_file_name, "", MARIA_NAME_IEXT, MY_APPEND_EXT | MY_UNPACK_FILENAME); if (strcmp(name_buff, maria_info.index_file_name)) index_file_name=maria_info.index_file_name; diff --git a/storage/maria/ma_blockrec.c b/storage/maria/ma_blockrec.c index 3c20c2704e0..27190f4e04c 100644 --- a/storage/maria/ma_blockrec.c +++ b/storage/maria/ma_blockrec.c @@ -553,6 +553,8 @@ static my_bool check_if_zero(byte *pos, uint length) We unpin pages in the reverse order as they where pinned; This may not be strictly necessary but may simplify things in the future. + info->s->rec_lsn contains the lsn for the first REDO + RETURN 0 ok 1 error (fatal disk error) @@ -576,8 +578,9 @@ void _ma_unpin_all_pages(MARIA_HA *info, LSN undo_lsn) while (pinned_page-- != page_link) pagecache_unlock_by_link(info->s->pagecache, pinned_page->link, pinned_page->unlock, PAGECACHE_UNPIN, - 0, undo_lsn); + info->trn->rec_lsn, undo_lsn); + info->trn->rec_lsn= 0; info->pinned_pages.elements= 0; DBUG_VOID_RETURN; } @@ -1037,7 +1040,6 @@ static my_bool get_head_or_tail_page(MARIA_HA *info, else { byte *dir; - /* TODO: lock the page */ /* Read old page */ DBUG_ASSERT(share->pagecache->block_size == block_size); if (!(res->buff= pagecache_read(share->pagecache, @@ -1046,13 +1048,8 @@ static my_bool get_head_or_tail_page(MARIA_HA *info, buff, share->page_type, lock, &page_link.link))) DBUG_RETURN(1); - if (lock != PAGECACHE_LOCK_LEFT_UNLOCKED) - { - page_link.unlock= (lock == PAGECACHE_LOCK_READ ? - PAGECACHE_LOCK_READ_UNLOCK : - PAGECACHE_LOCK_WRITE_UNLOCK); - push_dynamic(&info->pinned_pages, (void*) &page_link); - } + page_link.unlock= PAGECACHE_LOCK_WRITE_UNLOCK; + push_dynamic(&info->pinned_pages, (void*) &page_link); DBUG_ASSERT((res->buff[PAGE_TYPE_OFFSET] & PAGE_TYPE_MASK) == page_type); if (!(dir= find_free_position(res->buff, block_size, &res->rownr, @@ -1144,7 +1141,8 @@ static my_bool write_tail(MARIA_HA *info, log_array[TRANSLOG_INTERNAL_PARTS + 0].length= sizeof(log_data); log_array[TRANSLOG_INTERNAL_PARTS + 1].str= (char*) row_pos.data; log_array[TRANSLOG_INTERNAL_PARTS + 1].length= length; - if (translog_write_record(&lsn, LOGREC_REDO_INSERT_ROW_TAIL, + if (translog_write_record(!info->trn->rec_lsn ? &info->trn->rec_lsn : &lsn, + LOGREC_REDO_INSERT_ROW_TAIL, info->trn->short_id, NULL, share, sizeof(log_data) + length, TRANSLOG_INTERNAL_PARTS + 2, @@ -1400,7 +1398,8 @@ static my_bool free_full_pages(MARIA_HA *info, MARIA_ROW *row) log_array[TRANSLOG_INTERNAL_PARTS + 0].length= sizeof(log_data); log_array[TRANSLOG_INTERNAL_PARTS + 1].str= row->extents; log_array[TRANSLOG_INTERNAL_PARTS + 1].length= extents_length; - if (translog_write_record(&lsn, LOGREC_REDO_PURGE_BLOCKS, + if (translog_write_record(!info->trn->rec_lsn ? &info->trn->rec_lsn : &lsn, + LOGREC_REDO_PURGE_BLOCKS, info->trn->short_id, NULL, info->s, sizeof(log_data) + extents_length, TRANSLOG_INTERNAL_PARTS + 2, log_array)) @@ -1417,6 +1416,9 @@ static my_bool free_full_pages(MARIA_HA *info, MARIA_ROW *row) NOTES This is very similar to free_full_pages() + We don't have to update trn->rec_lsn here as before calling this function + we have already generated REDO's for deleting the HEAD block. + RETURN 0 ok 1 error @@ -1427,28 +1429,32 @@ static my_bool free_full_page_range(MARIA_HA *info, ulonglong page, uint count) uchar log_data[FILEID_STORE_SIZE + PAGERANGE_STORE_SIZE + ROW_EXTENT_SIZE]; LEX_STRING log_array[TRANSLOG_INTERNAL_PARTS + 1]; - LSN lsn; my_bool res= 0; if (pagecache_delete_pages(info->s->pagecache, &info->dfile, page, count, PAGECACHE_LOCK_WRITE, 0)) res= 1; - fileid_store(log_data, info->dfile.file); - pagerange_store(log_data + FILEID_STORE_SIZE, 1); - int5store(log_data + FILEID_STORE_SIZE + PAGERANGE_STORE_SIZE, - page); - int2store(log_data + FILEID_STORE_SIZE + PAGERANGE_STORE_SIZE + 5, - count); - log_array[TRANSLOG_INTERNAL_PARTS + 0].str= (char*) log_data; - log_array[TRANSLOG_INTERNAL_PARTS + 0].length= sizeof(log_data); + if (info->s->base.transactional) + { + LSN lsn; + DBUG_ASSERT(info->trn->rec_lsn); + fileid_store(log_data, info->dfile.file); + pagerange_store(log_data + FILEID_STORE_SIZE, 1); + int5store(log_data + FILEID_STORE_SIZE + PAGERANGE_STORE_SIZE, + page); + int2store(log_data + FILEID_STORE_SIZE + PAGERANGE_STORE_SIZE + 5, + count); + log_array[TRANSLOG_INTERNAL_PARTS + 0].str= (char*) log_data; + log_array[TRANSLOG_INTERNAL_PARTS + 0].length= sizeof(log_data); - if (translog_write_record(&lsn, LOGREC_REDO_PURGE_BLOCKS, - info->trn->short_id, NULL, info->s, - sizeof(log_data), - TRANSLOG_INTERNAL_PARTS + 1, log_array)) - res= 1; + if (translog_write_record(&lsn, LOGREC_REDO_PURGE_BLOCKS, + info->trn->short_id, NULL, info->s, + sizeof(log_data), + TRANSLOG_INTERNAL_PARTS + 1, log_array)) + res= 1; + } pthread_mutex_lock(&info->s->bitmap.bitmap_lock); if (_ma_reset_full_page_bits(info, &info->s->bitmap, page, count)) @@ -1951,7 +1957,8 @@ static my_bool write_block_record(MARIA_HA *info, log_array[TRANSLOG_INTERNAL_PARTS + 0].length= sizeof(log_data); log_array[TRANSLOG_INTERNAL_PARTS + 1].str= (char*) row_pos->data; log_array[TRANSLOG_INTERNAL_PARTS + 1].length= data_length; - if (translog_write_record(&lsn, LOGREC_REDO_INSERT_ROW_HEAD, + if (translog_write_record(!info->trn->rec_lsn ? &info->trn->rec_lsn : &lsn, + LOGREC_REDO_INSERT_ROW_HEAD, info->trn->short_id, NULL, share, sizeof(log_data) + data_length, TRANSLOG_INTERNAL_PARTS + 2, log_array)) @@ -2066,6 +2073,7 @@ static my_bool write_block_record(MARIA_HA *info, log_data); log_entry_length+= (log_pos - log_data); + /* trn->rec_lsn is already set earlier in this function */ error= translog_write_record(&lsn, LOGREC_REDO_INSERT_ROW_BLOBS, info->trn->short_id, NULL, share, log_entry_length, (uint) (log_array_pos - @@ -2524,7 +2532,7 @@ static my_bool delete_head_or_tail(MARIA_HA *info, log_array[TRANSLOG_INTERNAL_PARTS + 0].str= (char*) log_data; log_array[TRANSLOG_INTERNAL_PARTS + 0].length= sizeof(log_data); - if (translog_write_record(&lsn, + if (translog_write_record(!info->trn->rec_lsn ? &info->trn->rec_lsn : &lsn, (head ? LOGREC_REDO_PURGE_ROW_HEAD : LOGREC_REDO_PURGE_ROW_TAIL), info->trn->short_id, NULL, share, @@ -2557,7 +2565,8 @@ static my_bool delete_head_or_tail(MARIA_HA *info, PAGERANGE_STORE_SIZE, 1); log_array[TRANSLOG_INTERNAL_PARTS + 0].str= (char*) log_data; log_array[TRANSLOG_INTERNAL_PARTS + 0].length= sizeof(log_data); - if (translog_write_record(&lsn, LOGREC_REDO_PURGE_BLOCKS, + if (translog_write_record(!info->trn->rec_lsn ? &info->trn->rec_lsn : &lsn, + LOGREC_REDO_PURGE_BLOCKS, info->trn->short_id, NULL, share, sizeof(log_data), TRANSLOG_INTERNAL_PARTS + 1, log_array)) diff --git a/storage/maria/ma_delete.c b/storage/maria/ma_delete.c index 067dd060a92..436a65a52ce 100644 --- a/storage/maria/ma_delete.c +++ b/storage/maria/ma_delete.c @@ -111,8 +111,8 @@ int maria_delete(MARIA_HA *info,const byte *record) allow_break(); /* Allow SIGHUP & SIGINT */ if (info->invalidator != 0) { - DBUG_PRINT("info", ("invalidator... '%s' (delete)", info->filename)); - (*info->invalidator)(info->filename); + DBUG_PRINT("info", ("invalidator... '%s' (delete)", info->s->open_file_name)); + (*info->invalidator)(info->s->open_file_name); info->invalidator=0; } DBUG_RETURN(0); diff --git a/storage/maria/ma_loghandler.c b/storage/maria/ma_loghandler.c index 8ebe17083ce..6d19f46310d 100644 --- a/storage/maria/ma_loghandler.c +++ b/storage/maria/ma_loghandler.c @@ -3855,7 +3855,6 @@ static my_bool translog_write_variable_record(LSN *lsn, uint page_rest; /* Max number of such LSNs per record is 2 */ byte compressed_LSNs[2 * LSN_STORE_SIZE]; - DBUG_ENTER("translog_write_variable_record"); translog_lock(); @@ -3867,8 +3866,8 @@ static my_bool translog_write_variable_record(LSN *lsn, header_length1, page_rest)); /* - header and part which we should read have to fit in one chunk - TODO: allow to divide readable header + header and part which we should read have to fit in one chunk + TODO: allow to divide readable header */ if (page_rest < (header_length1 + log_record_type_descriptor[type].read_header_len)) diff --git a/storage/maria/ma_open.c b/storage/maria/ma_open.c index b3005571436..b8ce6d123e7 100644 --- a/storage/maria/ma_open.c +++ b/storage/maria/ma_open.c @@ -72,8 +72,163 @@ MARIA_HA *_ma_test_if_reopen(char *filename) } +/* + Open a new instance of an already opened Maria table + + SYNOPSIS + maria_clone_internal() + share Share of already open table + mode Mode of table (O_RDONLY | O_RDWR) + data_file Filedescriptor of data file to use < 0 if one should open + open it. + + RETURN + # Maria handler + 0 Error +*/ + + +static MARIA_HA *maria_clone_internal(MARIA_SHARE *share, int mode, + File data_file) +{ + int save_errno; + uint errpos; + MARIA_HA info,*m_info; + DBUG_ENTER("maria_clone_internal"); + + errpos= 0; + bzero((byte*) &info,sizeof(info)); + + if (mode == O_RDWR && share->mode == O_RDONLY) + { + my_errno=EACCES; /* Can't open in write mode */ + goto err; + } + if (data_file >= 0) + info.dfile.file= data_file; + else if (_ma_open_datafile(&info, share, -1)) + goto err; + errpos= 5; + + /* alloc and set up private structure parts */ + if (!my_multi_malloc(MY_WME, + &m_info,sizeof(MARIA_HA), + &info.blobs,sizeof(MARIA_BLOB)*share->base.blobs, + &info.buff,(share->base.max_key_block_length*2+ + share->base.max_key_length), + &info.lastkey,share->base.max_key_length*3+1, + &info.first_mbr_key, share->base.max_key_length, + &info.maria_rtree_recursion_state, + share->have_rtree ? 1024 : 0, + NullS)) + goto err; + errpos= 6; + + memcpy(info.blobs,share->blobs,sizeof(MARIA_BLOB)*share->base.blobs); + info.lastkey2=info.lastkey+share->base.max_key_length; + + info.s=share; + info.cur_row.lastpos= HA_OFFSET_ERROR; + info.update= (short) (HA_STATE_NEXT_FOUND+HA_STATE_PREV_FOUND); + info.opt_flag=READ_CHECK_USED; + info.this_unique= (ulong) info.dfile.file; /* Uniq number in process */ + if (share->data_file_type == COMPRESSED_RECORD) + info.this_unique= share->state.unique; + info.this_loop=0; /* Update counter */ + info.last_unique= share->state.unique; + info.last_loop= share->state.update_count; + info.lock_type=F_UNLCK; + info.quick_mode=0; + info.bulk_insert=0; + info.ft1_to_ft2=0; + info.errkey= -1; + info.page_changed=1; + info.keyread_buff= info.buff + share->base.max_key_block_length; + if ((*share->init)(&info)) + goto err; + + pthread_mutex_lock(&share->intern_lock); + info.read_record= share->read_record; + share->reopen++; + share->write_flag=MYF(MY_NABP | MY_WAIT_IF_FULL); + if (share->options & HA_OPTION_READ_ONLY_DATA) + { + info.lock_type=F_RDLCK; + share->r_locks++; + share->tot_locks++; + } + if (share->options & HA_OPTION_TMP_TABLE) + { + share->temporary= share->delay_key_write= 1; + + share->write_flag=MYF(MY_NABP); + share->w_locks++; /* We don't have to update status */ + share->tot_locks++; + info.lock_type=F_WRLCK; + } + if ((share->options & HA_OPTION_DELAY_KEY_WRITE) && + maria_delay_key_write) + share->delay_key_write=1; + + info.state= &share->state.state; /* Change global values by default */ + info.trn= &dummy_transaction_object; + pthread_mutex_unlock(&share->intern_lock); + + /* Allocate buffer for one record */ + /* prerequisites: info->rec_buffer == 0 && info->rec_buff_size == 0 */ + if (_ma_alloc_buffer(&info.rec_buff, &info.rec_buff_size, + share->base.default_rec_buff_size)) + goto err; + + bzero(info.rec_buff, share->base.default_rec_buff_size); + + *m_info=info; +#ifdef THREAD + thr_lock_data_init(&share->lock,&m_info->lock,(void*) m_info); +#endif + m_info->open_list.data=(void*) m_info; + maria_open_list=list_add(maria_open_list,&m_info->open_list); + + DBUG_RETURN(m_info); + +err: + save_errno=my_errno ? my_errno : HA_ERR_END_OF_FILE; + if ((save_errno == HA_ERR_CRASHED) || + (save_errno == HA_ERR_CRASHED_ON_USAGE) || + (save_errno == HA_ERR_CRASHED_ON_REPAIR)) + _ma_report_error(save_errno, share->open_file_name); + switch (errpos) { + case 6: + (*share->end)(&info); + my_free((gptr) m_info,MYF(0)); + /* fall through */ + case 5: + if (data_file < 0) + VOID(my_close(info.dfile.file, MYF(0))); + break; + } + my_errno=save_errno; + DBUG_RETURN (NULL); +} /* maria_clone_internal */ + + +/* Make a clone of a maria table */ + +MARIA_HA *maria_clone(MARIA_SHARE *share, int mode) +{ + MARIA_HA *new_info; + pthread_mutex_lock(&THR_LOCK_maria); + new_info= maria_clone_internal(share, mode, + share->data_file_type == BLOCK_RECORD ? + share->bitmap.file.file : -1); + pthread_mutex_unlock(&THR_LOCK_maria); + return new_info; +} + + /****************************************************************************** - open a MARIA database. + open a MARIA table + See my_base.h for the handle_locking argument if handle_locking and HA_OPEN_ABORT_IF_CRASHED then abort if the table is marked crashed or if we are not using locking and the table doesn't @@ -82,7 +237,7 @@ MARIA_HA *_ma_test_if_reopen(char *filename) MARIA_HA *maria_open(const char *name, int mode, uint open_flags) { - int kfile,open_mode,save_errno,have_rtree=0; + int kfile,open_mode,save_errno; uint i,j,len,errpos,head_length,base_pos,info_length,keys, key_parts,unique_key_parts,fulltext_keys,uniques; char name_buff[FN_REFLEN], org_name[FN_REFLEN], index_name[FN_REFLEN], @@ -93,6 +248,7 @@ MARIA_HA *maria_open(const char *name, int mode, uint open_flags) ulong rec_per_key_part[HA_MAX_POSSIBLE_KEY*HA_MAX_KEY_SEG]; my_off_t key_root[HA_MAX_POSSIBLE_KEY]; ulonglong max_key_file_length, max_data_file_length; + File data_file= -1; DBUG_ENTER("maria_open"); LINT_INIT(m_info); @@ -288,6 +444,7 @@ MARIA_HA *maria_open(const char *name, int mode, uint open_flags) &share->unique_file_name,strlen(name_buff)+1, &share->index_file_name,strlen(index_name)+1, &share->data_file_name,strlen(data_name)+1, + &share->open_file_name,strlen(name)+1, &share->state.key_root,keys*sizeof(my_off_t), #ifdef THREAD &share->key_root_lock,sizeof(rw_lock_t)*keys, @@ -306,6 +463,7 @@ MARIA_HA *maria_open(const char *name, int mode, uint open_flags) share->unique_name_length= strlen(name_buff); strmov(share->index_file_name, index_name); strmov(share->data_file_name, data_name); + strmov(share->open_file_name, name); share->block_size= share->base.block_size; { @@ -317,7 +475,7 @@ MARIA_HA *maria_open(const char *name, int mode, uint open_flags) disk_pos_assert(disk_pos + share->keyinfo[i].keysegs * HA_KEYSEG_SIZE, end_pos); if (share->keyinfo[i].key_alg == HA_KEY_ALG_RTREE) - have_rtree=1; + share->have_rtree= 1; share->keyinfo[i].seg=pos; for (j=0 ; j < share->keyinfo[i].keysegs; j++,pos++) { @@ -458,35 +616,14 @@ MARIA_HA *maria_open(const char *name, int mode, uint open_flags) } } share->columndef[i].type=(int) FIELD_LAST; /* End marker */ -#ifdef ASKMONTY - /* - This code was added to mi_open.c in this cset: - "ChangeSet 1.1616.2941.5 2007/01/22 16:34:58 svoj@mysql.com - BUG#24401 - MySQL server crashes if you try to retrieve data from - corrupted table - Accessing a table with corrupted column definition results in server - crash. - This is fixed by refusing to open such tables. Affects MyISAM only. - No test case, since it requires crashed table. - storage/myisam/mi_open.c 1.80.2.10 2007/01/22 16:34:57 svoj@mysql.com - Refuse to open MyISAM table with summary columns length bigger than - length of the record." - - The problem is that the "offset" variable was removed (by Monty in the - rows-in-block patch). Monty will know how to merge that. - Guilhem will make sure to notify him. - */ - if (offset > share->base.reclength) + + if ((share->data_file_type == BLOCK_RECORD || + share->data_file_type == COMPRESSED_RECORD)) { - /* purecov: begin inspected */ - my_errno= HA_ERR_CRASHED; - goto err; - /* purecov: end */ + if (_ma_open_datafile(&info, share, -1)) + goto err; + data_file= info.dfile.file; } -#endif /* ASKMONTY */ - - if (_ma_open_datafile(&info, share, -1)) - goto err; errpos= 5; share->kfile.file= kfile; @@ -522,6 +659,13 @@ MARIA_HA *maria_open(const char *name, int mode, uint open_flags) } } share->is_log_table= FALSE; + if (open_flags & HA_OPEN_TMP_TABLE) + share->options|= HA_OPTION_TMP_TABLE; + if (open_flags & HA_OPEN_DELAY_KEY_WRITE) + share->options|= HA_OPTION_DELAY_KEY_WRITE; + if (mode == O_RDONLY) + share->options|= HA_OPTION_READ_ONLY_DATA; + #ifdef THREAD thr_lock_init(&share->lock); VOID(pthread_mutex_init(&share->intern_lock,MY_MUTEX_INIT_FAST)); @@ -541,7 +685,7 @@ MARIA_HA *maria_open(const char *name, int mode, uint open_flags) HA_OPTION_TEMP_COMPRESS_RECORD)) || (open_flags & HA_OPEN_TMP_TABLE) || share->data_file_type == BLOCK_RECORD || - have_rtree) ? 0 : 1; + share->have_rtree) ? 0 : 1; if (share->concurrent_insert) { share->lock.get_status=_ma_get_status; @@ -556,106 +700,12 @@ MARIA_HA *maria_open(const char *name, int mode, uint open_flags) else { share= old_info->s; - if (mode == O_RDWR && share->mode == O_RDONLY) - { - my_errno=EACCES; /* Can't open in write mode */ - goto err; - } if (share->data_file_type == BLOCK_RECORD) - info.dfile= share->bitmap.file; - else if (_ma_open_datafile(&info, share, old_info->dfile.file)) - goto err; - errpos= 5; - have_rtree= old_info->maria_rtree_recursion_state != NULL; - } - - /* alloc and set up private structure parts */ - if (!my_multi_malloc(MY_WME, - &m_info,sizeof(MARIA_HA), - &info.blobs,sizeof(MARIA_BLOB)*share->base.blobs, - &info.buff,(share->base.max_key_block_length*2+ - share->base.max_key_length), - &info.lastkey,share->base.max_key_length*3+1, - &info.first_mbr_key, share->base.max_key_length, - &info.filename,strlen(name)+1, - &info.maria_rtree_recursion_state,have_rtree ? 1024 : 0, - NullS)) - goto err; - errpos= 6; - - if (!have_rtree) - info.maria_rtree_recursion_state= NULL; - - strmov(info.filename,name); - memcpy(info.blobs,share->blobs,sizeof(MARIA_BLOB)*share->base.blobs); - info.lastkey2=info.lastkey+share->base.max_key_length; - - info.s=share; - info.cur_row.lastpos= HA_OFFSET_ERROR; - info.update= (short) (HA_STATE_NEXT_FOUND+HA_STATE_PREV_FOUND); - info.opt_flag=READ_CHECK_USED; - info.this_unique= (ulong) info.dfile.file; /* Uniq number in process */ - if (share->data_file_type == COMPRESSED_RECORD) - info.this_unique= share->state.unique; - info.this_loop=0; /* Update counter */ - info.last_unique= share->state.unique; - info.last_loop= share->state.update_count; - if (mode == O_RDONLY) - share->options|=HA_OPTION_READ_ONLY_DATA; - info.lock_type=F_UNLCK; - info.quick_mode=0; - info.bulk_insert=0; - info.ft1_to_ft2=0; - info.errkey= -1; - info.page_changed=1; - info.keyread_buff= info.buff + share->base.max_key_block_length; - if ((*share->init)(&info)) - goto err; - - pthread_mutex_lock(&share->intern_lock); - info.read_record= share->read_record; - share->reopen++; - share->write_flag=MYF(MY_NABP | MY_WAIT_IF_FULL); - if (share->options & HA_OPTION_READ_ONLY_DATA) - { - info.lock_type=F_RDLCK; - share->r_locks++; - share->tot_locks++; + data_file= share->bitmap.file.file; /* Only opened once */ } - if ((open_flags & HA_OPEN_TMP_TABLE) || - (share->options & HA_OPTION_TMP_TABLE)) - { - share->temporary= share->delay_key_write= 1; - share->write_flag=MYF(MY_NABP); - share->w_locks++; /* We don't have to update status */ - share->tot_locks++; - info.lock_type=F_WRLCK; - } - if (((open_flags & HA_OPEN_DELAY_KEY_WRITE) || - (share->options & HA_OPTION_DELAY_KEY_WRITE)) && - maria_delay_key_write) - share->delay_key_write=1; - - info.state= &share->state.state; /* Change global values by default */ - info.trn= &dummy_transaction_object; - pthread_mutex_unlock(&share->intern_lock); - - /* Allocate buffer for one record */ - /* prerequisites: info->rec_buffer == 0 && info->rec_buff_size == 0 */ - if (_ma_alloc_buffer(&info.rec_buff, &info.rec_buff_size, - share->base.default_rec_buff_size)) + if (!(m_info= maria_clone_internal(share, mode, data_file))) goto err; - - bzero(info.rec_buff, share->base.default_rec_buff_size); - - *m_info=info; -#ifdef THREAD - thr_lock_data_init(&share->lock,&m_info->lock,(void*) m_info); -#endif - m_info->open_list.data=(void*) m_info; - maria_open_list=list_add(maria_open_list,&m_info->open_list); - pthread_mutex_unlock(&THR_LOCK_maria); DBUG_RETURN(m_info); @@ -666,13 +716,9 @@ err: (save_errno == HA_ERR_CRASHED_ON_REPAIR)) _ma_report_error(save_errno, name); switch (errpos) { - case 6: - (*share->end)(&info); - my_free((gptr) m_info,MYF(0)); - /* fall through */ case 5: - if (share->data_file_type != BLOCK_RECORD) - VOID(my_close(info.dfile.file, MYF(0))); + if (data_file >= 0) + VOID(my_close(data_file, MYF(0))); if (old_info) break; /* Don't remove open table */ (*share->once_end)(share); @@ -693,7 +739,7 @@ err: break; } pthread_mutex_unlock(&THR_LOCK_maria); - my_errno=save_errno; + my_errno= save_errno; DBUG_RETURN (NULL); } /* maria_open */ diff --git a/storage/maria/ma_packrec.c b/storage/maria/ma_packrec.c index acd7db1df95..f3187314f0e 100644 --- a/storage/maria/ma_packrec.c +++ b/storage/maria/ma_packrec.c @@ -1411,7 +1411,7 @@ uint _ma_pack_get_block_info(MARIA_HA *maria, MARIA_BIT_BUFF *bit_buff, { ref_length=maria->s->pack.ref_length; /* - We can't use my_pread() here because maria_read_rnd_pack_record assumes + We can't use my_pread() here because _ma_read_rnd_pack_record assumes position is ok */ VOID(my_seek(file,filepos,MY_SEEK_SET,MYF(0))); diff --git a/storage/maria/ma_pagecache.c b/storage/maria/ma_pagecache.c index 5b713e6797f..839bd4b7758 100755 --- a/storage/maria/ma_pagecache.c +++ b/storage/maria/ma_pagecache.c @@ -2677,7 +2677,13 @@ void pagecache_unlock_by_link(PAGECACHE *pagecache, inc_counter_for_resize_op(pagecache); if (first_REDO_LSN_for_page) { - DBUG_ASSERT(lock == PAGECACHE_LOCK_WRITE_UNLOCK); + /* + LOCK_READ_UNLOCK is ok here as the page may have first locked + with WRITE lock that was temporarly converted to READ lock before + it's unpinned + */ + DBUG_ASSERT(lock == PAGECACHE_LOCK_WRITE_UNLOCK || + lock == PAGECACHE_LOCK_READ_UNLOCK); DBUG_ASSERT(pin == PAGECACHE_UNPIN); set_if_bigger(block->rec_lsn, first_REDO_LSN_for_page); } diff --git a/storage/maria/ma_update.c b/storage/maria/ma_update.c index db0f5641124..737c7c909b4 100644 --- a/storage/maria/ma_update.c +++ b/storage/maria/ma_update.c @@ -196,8 +196,8 @@ int maria_update(register MARIA_HA *info, const byte *oldrec, byte *newrec) allow_break(); /* Allow SIGHUP & SIGINT */ if (info->invalidator != 0) { - DBUG_PRINT("info", ("invalidator... '%s' (update)", info->filename)); - (*info->invalidator)(info->filename); + DBUG_PRINT("info", ("invalidator... '%s' (update)", info->s->open_file_name)); + (*info->invalidator)(info->s->open_file_name); info->invalidator=0; } DBUG_RETURN(0); diff --git a/storage/maria/ma_write.c b/storage/maria/ma_write.c index 491cba5d187..c16795c05f0 100644 --- a/storage/maria/ma_write.c +++ b/storage/maria/ma_write.c @@ -180,8 +180,8 @@ int maria_write(MARIA_HA *info, byte *record) VOID(_ma_writeinfo(info, WRITEINFO_UPDATE_KEYFILE)); if (info->invalidator != 0) { - DBUG_PRINT("info", ("invalidator... '%s' (update)", info->filename)); - (*info->invalidator)(info->filename); + DBUG_PRINT("info", ("invalidator... '%s' (update)", info->s->open_file_name)); + (*info->invalidator)(info->s->open_file_name); info->invalidator=0; } diff --git a/storage/maria/maria_def.h b/storage/maria/maria_def.h index 25e27c8d296..7a7261d0d6a 100644 --- a/storage/maria/maria_def.h +++ b/storage/maria/maria_def.h @@ -222,9 +222,9 @@ typedef struct st_maria_share MARIA_PACK pack; /* Data about packed records */ MARIA_BLOB *blobs; /* Pointer to blobs */ char *unique_file_name; /* realpath() of index file */ - char *data_file_name, /* Resolved path names from - symlinks */ - *index_file_name; + char *data_file_name; /* Resolved path names from symlinks */ + char *index_file_name; + char *open_file_name; /* parameter to open filename */ byte *file_map; /* mem-map of file if possible */ PAGECACHE *pagecache; /* ref to the current key cache */ MARIA_DECODE_TREE *decode_trees; @@ -299,6 +299,7 @@ typedef struct st_maria_share global_changed, /* If changed since open */ not_flushed, concurrent_insert; my_bool delay_key_write; + my_bool have_rtree; #ifdef THREAD THR_LOCK lock; pthread_mutex_t intern_lock; /* Locking for use with _locking */ @@ -388,7 +389,6 @@ struct st_maria_info DYNAMIC_ARRAY *ft1_to_ft2; /* used only in ft1->ft2 conversion */ MEM_ROOT ft_memroot; /* used by the parser */ MYSQL_FTPARSER_PARAM *ftparser_param; /* share info between init/deinit */ - char *filename; /* parameter to open filename */ byte *buff; /* page buffer */ byte *keyread_buff; /* Buffer for last key read */ byte *lastkey, *lastkey2; /* Last used search key */ diff --git a/storage/maria/maria_ftdump.c b/storage/maria/maria_ftdump.c index a773602b451..8b0256344cb 100644 --- a/storage/maria/maria_ftdump.c +++ b/storage/maria/maria_ftdump.c @@ -100,7 +100,7 @@ int main(int argc,char *argv[]) if ((inx >= info->s->base.keys) || !(info->s->keyinfo[inx].flag & HA_FULLTEXT)) { - printf("Key %d in table %s is not a FULLTEXT key\n", inx, info->filename); + printf("Key %d in table %s is not a FULLTEXT key\n", inx, info->s->open_file_name); goto err; } diff --git a/storage/maria/maria_pack.c b/storage/maria/maria_pack.c index f1b3903c944..f6e962191c2 100644 --- a/storage/maria/maria_pack.c +++ b/storage/maria/maria_pack.c @@ -509,9 +509,11 @@ static int compress(PACK_MRG_INFO *mrg,char *result_table) /* Create temporary or join file */ if (backup) - VOID(fn_format(org_name,isam_file->filename,"",MARIA_NAME_DEXT,2)); + VOID(fn_format(org_name,isam_file->s->open_file_name,"",MARIA_NAME_DEXT, + 2)); else - VOID(fn_format(org_name,isam_file->filename,"",MARIA_NAME_DEXT,2+4+16)); + VOID(fn_format(org_name,isam_file->s->open_file_name,"",MARIA_NAME_DEXT, + 2+4+16)); if (init_pagecache(maria_pagecache, MARIA_MIN_PAGE_CACHE_SIZE, 0, 0, maria_block_size) == 0) @@ -705,7 +707,8 @@ static int compress(PACK_MRG_INFO *mrg,char *result_table) { if (backup) { - if (my_rename(org_name,make_old_name(temp_name,isam_file->filename), + if (my_rename(org_name,make_old_name(temp_name, + isam_file->s->open_file_name), MYF(MY_WME))) error=1; else diff --git a/storage/maria/trnman.h b/storage/maria/trnman.h index 24936253935..cfdd214dda7 100644 --- a/storage/maria/trnman.h +++ b/storage/maria/trnman.h @@ -45,7 +45,7 @@ struct st_transaction LF_PINS *pins; TrID trid, min_read_from, commit_trid; TRN *next, *prev; - LSN undo_lsn; + LSN rec_lsn, undo_lsn; uint locked_tables; /* Note! if locks.loid is 0, trn is NOT initialized */ }; diff --git a/storage/maria/unittest/ma_pagecache_consist.c b/storage/maria/unittest/ma_pagecache_consist.c old mode 100755 new mode 100644 -- cgit v1.2.1 From 8410950316f7abdbecee15f63335ca5c13974163 Mon Sep 17 00:00:00 2001 From: unknown Date: Thu, 7 Jun 2007 15:10:17 +0200 Subject: fix for 2 memory leaks in ps_maria.test (just need to destroy objects allocated at startup). storage/maria/ma_init.c: destroy transaction manager and log's pagecache when Maria shuts down storage/maria/trnman.c: short_trid_to_active_trn!=NULL now serves to detect if trnman has initialized some objects and so if trnman_destroy() has objects to destroy. In other words, short_trid_to_active_trn serves as "trnman_inited" variable. trnman_destroy() is always called by maria_end(), but trnman_init() is not always called (for example in ma_test1), that's why trnman_destroy() cannot blindly destroy. --- storage/maria/ma_init.c | 2 ++ storage/maria/trnman.c | 16 +++++++++++----- 2 files changed, 13 insertions(+), 5 deletions(-) diff --git a/storage/maria/ma_init.c b/storage/maria/ma_init.c index 9d72408ad4b..c56e6704729 100644 --- a/storage/maria/ma_init.c +++ b/storage/maria/ma_init.c @@ -55,7 +55,9 @@ void maria_end(void) { maria_inited= FALSE; ft_free_stopwords(); + trnman_destroy(); translog_destroy(); + end_pagecache(maria_log_pagecache, TRUE); ma_control_file_end(); pthread_mutex_destroy(&THR_LOCK_maria); } diff --git a/storage/maria/trnman.c b/storage/maria/trnman.c index 8f52f1e2825..702b3b20f6c 100644 --- a/storage/maria/trnman.c +++ b/storage/maria/trnman.c @@ -103,6 +103,13 @@ static byte *trn_get_hash_key(const byte *trn, uint* len, int trnman_init() { DBUG_ENTER("trnman_init"); + + short_trid_to_active_trn= (TRN **)my_malloc(SHORT_TRID_MAX*sizeof(TRN*), + MYF(MY_WME|MY_ZEROFILL)); + if (unlikely(!short_trid_to_active_trn)) + DBUG_RETURN(1); + short_trid_to_active_trn--; /* min short_trid is 1 */ + /* Initialize lists. active_list_max.min_read_from must be larger than any trid, @@ -136,11 +143,6 @@ int trnman_init() pthread_mutex_init(&LOCK_trn_list, MY_MUTEX_INIT_FAST); my_atomic_rwlock_init(&LOCK_short_trid_to_trn); my_atomic_rwlock_init(&LOCK_pool); - short_trid_to_active_trn= (TRN **)my_malloc(SHORT_TRID_MAX*sizeof(TRN*), - MYF(MY_WME|MY_ZEROFILL)); - if (unlikely(!short_trid_to_active_trn)) - DBUG_RETURN(1); - short_trid_to_active_trn--; /* min short_trid is 1 */ #ifdef NOT_USED lockman_init(&maria_lockman, (loid_to_lo_func *)&short_trid_to_TRN, 10000); @@ -157,6 +159,9 @@ int trnman_init() void trnman_destroy() { DBUG_ENTER("trnman_destroy"); + + if (short_trid_to_active_trn == NULL) /* trnman already destroyed */ + DBUG_VOID_RETURN; DBUG_ASSERT(trid_to_committed_trn.count == 0); DBUG_ASSERT(trnman_active_transactions == 0); DBUG_ASSERT(trnman_committed_transactions == 0); @@ -178,6 +183,7 @@ void trnman_destroy() my_atomic_rwlock_destroy(&LOCK_short_trid_to_trn); my_atomic_rwlock_destroy(&LOCK_pool); my_free((void *)(short_trid_to_active_trn+1), MYF(0)); + short_trid_to_active_trn= NULL; #ifdef NOT_USED lockman_destroy(&maria_lockman); #endif -- cgit v1.2.1 From bb4a7ad5a08400384428a8981165f72a6764c57f Mon Sep 17 00:00:00 2001 From: unknown Date: Thu, 7 Jun 2007 21:51:11 +0200 Subject: Maria: fix for "use of uninitialized value" Valgrind error. storage/maria/ha_maria.cc: initialize ha_maria::data_file_type when opening the table (it was not initialized, causing a Valgrind error) storage/maria/ma_locking.c: if it's impossible, let's test it --- storage/maria/ha_maria.cc | 1 + storage/maria/ma_locking.c | 1 + 2 files changed, 2 insertions(+) diff --git a/storage/maria/ha_maria.cc b/storage/maria/ha_maria.cc index 812fc69c2c2..c2f4d621305 100644 --- a/storage/maria/ha_maria.cc +++ b/storage/maria/ha_maria.cc @@ -1806,6 +1806,7 @@ int ha_maria::info(uint flag) MY_APPEND_EXT | MY_UNPACK_FILENAME); if (strcmp(name_buff, maria_info.index_file_name)) index_file_name=maria_info.index_file_name; + data_file_type= maria_info.data_file_type; } if (flag & HA_STATUS_ERRKEY) { diff --git a/storage/maria/ma_locking.c b/storage/maria/ma_locking.c index 6865c653d7e..abb095d47c2 100644 --- a/storage/maria/ma_locking.c +++ b/storage/maria/ma_locking.c @@ -190,6 +190,7 @@ int maria_lock_database(MARIA_HA *info, int lock_type) share->tot_locks++; break; default: + DBUG_ASSERT(0); break; /* Impossible */ } } -- cgit v1.2.1 From f5f2a8a11223e87c28e1386bdf840af088fc7248 Mon Sep 17 00:00:00 2001 From: unknown Date: Thu, 7 Jun 2007 22:07:44 +0200 Subject: fix for memory leak in _ma_scan_init_block_record() (Valgrind error): make Maria support multiple calls to rnd_init() without an rnd_end() call in between. storage/maria/ma_blockrec.c: as explained in sql/handler.h, multiple calls to rnd_init() without a rnd_end() in between, are possible, and engine must be prepared to that. So in _ma_scan_init_block_record(), we allocate a buffer only if we have not yet one. --- storage/maria/ma_blockrec.c | 12 ++++++++---- 1 file changed, 8 insertions(+), 4 deletions(-) diff --git a/storage/maria/ma_blockrec.c b/storage/maria/ma_blockrec.c index 27190f4e04c..1d7f96f6557 100644 --- a/storage/maria/ma_blockrec.c +++ b/storage/maria/ma_blockrec.c @@ -3315,12 +3315,16 @@ my_bool _ma_cmp_block_unique(MARIA_HA *info, MARIA_UNIQUEDEF *def, my_bool _ma_scan_init_block_record(MARIA_HA *info) { - byte *ptr; DBUG_ENTER("_ma_scan_init_block_record"); - if (!(ptr= (byte *) my_malloc(info->s->block_size * 2, MYF(MY_WME)))) + /* + bitmap_buff may already be allocated if this is the second call to + rnd_init() without a rnd_end() in between, see sql/handler.h + */ + if (!(info->scan.bitmap_buff || + ((info->scan.bitmap_buff= + (byte *) my_malloc(info->s->block_size * 2, MYF(MY_WME)))))) DBUG_RETURN(1); - info->scan.bitmap_buff= ptr; - info->scan.page_buff= ptr + info->s->block_size; + info->scan.page_buff= info->scan.bitmap_buff + info->s->block_size; info->scan.bitmap_end= info->scan.bitmap_buff + info->s->bitmap.total_size; /* Set scan variables to get _ma_scan_block() to start with reading bitmap */ -- cgit v1.2.1 From e30e21f070b998a0a932f2dbc4da91e05108045a Mon Sep 17 00:00:00 2001 From: unknown Date: Fri, 8 Jun 2007 16:03:39 +0200 Subject: - if table is not transactional, we don't create a transaction - if table is temporary it's not crash-safe so we declare it non-transactional (saves trnman calls, REDO/UNDO log writing, and fixes the assertion failure at the first line of trnman_destroy()). storage/maria/ha_maria.cc: if table is not transactional, no need to create a transaction: - it saves trnman calls (mutex locks etc) - it saves REDO and UNDO log writing - it closes a bug: if this is a temporary table, external_lock(F_RD|WRLCK) is not always paired with external_lock(F_UNLCK), which confuses the transaction logic in external_lock. As temp tables are not crash-safe and so not transactional in this Maria version, we skip transactions and de-confuse. Note that maria_lock_database(F_UNLCK) is properly called, so if the transaction logic moves from external_lock() to maria_lock_database() (probably TODO), transactional temp tables will be possible. storage/maria/ma_create.c: temporary tables cannot be crash-safe as they are dropped at restart storage/maria/maria_def.h: comment --- storage/maria/ha_maria.cc | 13 ++++++++----- storage/maria/ma_create.c | 13 ++++++++----- storage/maria/maria_def.h | 2 +- 3 files changed, 17 insertions(+), 11 deletions(-) diff --git a/storage/maria/ha_maria.cc b/storage/maria/ha_maria.cc index c2f4d621305..f7fd417836a 100644 --- a/storage/maria/ha_maria.cc +++ b/storage/maria/ha_maria.cc @@ -1860,6 +1860,8 @@ int ha_maria::external_lock(THD *thd, int lock_type) { TRN *trn= THD_TRN; DBUG_ENTER("ha_maria::external_lock"); + if (!file->s->base.transactional) + goto skip_transaction; if (!trn && lock_type != F_UNLCK) /* no transaction yet - open it now */ { trn= trnman_new_trn(& thd->mysys_var->mutex, @@ -1872,7 +1874,7 @@ int ha_maria::external_lock(THD *thd, int lock_type) DBUG_PRINT("info", ("THD_TRN set to 0x%lx", (ulong)trn)); THD_TRN= trn; if (thd->options & (OPTION_NOT_AUTOCOMMIT | OPTION_BEGIN)) - trans_register_ha(thd, true, maria_hton); + trans_register_ha(thd, TRUE, maria_hton); } if (lock_type != F_UNLCK) { @@ -1905,6 +1907,7 @@ int ha_maria::external_lock(THD *thd, int lock_type) } } } +skip_transaction: DBUG_RETURN(maria_lock_database(file, !table->s->tmp_table ? lock_type : ((lock_type == F_UNLCK) ? F_UNLCK : F_EXTRA_LCK))); @@ -1913,11 +1916,11 @@ int ha_maria::external_lock(THD *thd, int lock_type) int ha_maria::start_stmt(THD *thd, thr_lock_type lock_type) { TRN *trn= THD_TRN; - DBUG_ASSERT(trn); // this may be called only after external_lock() - DBUG_ASSERT(lock_type != F_UNLCK); - if (!trnman_increment_locked_tables(trn)) + if (file->s->base.transactional) { - trans_register_ha(thd, false, maria_hton); + DBUG_ASSERT(trn); // this may be called only after external_lock() + DBUG_ASSERT(lock_type != F_UNLCK); + /* As external_lock() was already called, don't increment locked_tables */ trnman_new_statement(trn); } return 0; diff --git a/storage/maria/ma_create.c b/storage/maria/ma_create.c index 5dd6e0e1f93..d8660dd41cb 100644 --- a/storage/maria/ma_create.c +++ b/storage/maria/ma_create.c @@ -246,6 +246,14 @@ int maria_create(const char *name, enum data_file_type datafile_type, } } } + + if (flags & HA_CREATE_TMP_TABLE) + { + options|= HA_OPTION_TMP_TABLE; + create_mode|= O_EXCL | O_NOFOLLOW; + /* temp tables are not crash-safe (dropped at restart) */ + ci->transactional= FALSE; + } share.base.null_bytes= ci->null_bytes; share.base.original_null_bytes= ci->null_bytes; share.base.transactional= ci->transactional; @@ -255,11 +263,6 @@ int maria_create(const char *name, enum data_file_type datafile_type, if (pack_reclength != INT_MAX32) pack_reclength+= max_field_lengths + long_varchar_count; - if (flags & HA_CREATE_TMP_TABLE) - { - options|= HA_OPTION_TMP_TABLE; - create_mode|= O_EXCL | O_NOFOLLOW; - } if (flags & HA_CREATE_CHECKSUM || (options & HA_OPTION_CHECKSUM)) { options|= HA_OPTION_CHECKSUM; diff --git a/storage/maria/maria_def.h b/storage/maria/maria_def.h index 7a7261d0d6a..46ffac7cbc2 100644 --- a/storage/maria/maria_def.h +++ b/storage/maria/maria_def.h @@ -168,7 +168,7 @@ typedef struct st_ma_base_info /* The following are from the header */ uint key_parts, all_key_parts; - /* If false, we disable logging, versioning etc */ + /* If false, we disable logging, versioning, transaction etc */ my_bool transactional; } MARIA_BASE_INFO; -- cgit v1.2.1 From fdfb51484c9b1e239fd9eb738051020967c99c7f Mon Sep 17 00:00:00 2001 From: unknown Date: Sat, 9 Jun 2007 14:52:17 +0300 Subject: Fixed compiler warnings Fixed bug in ma_dbug.c that gave valgrind warning (only relevant when using --debug) Fixed bug in blob logging (Fixes valgrind warning) maria_getint() -> maria_data_on_page() mysys/safemalloc.c: Added debug function to print out where a piece of memory was allocated sql/opt_range.cc: Remove DBUG_PRINT of unitailized memory storage/maria/ma_blockrec.c: Fixed bug in blob logging storage/maria/ma_check.c: Fixed compiler warning storage/maria/ma_dbug.c: Added missed end++; Caused usage of unitialized memory for nullable keys that was not NULL storage/maria/ma_delete.c: maria_getint() -> maria_data_on_page() storage/maria/ma_init.c: Added header file to get rid of warning storage/maria/ma_key.c: More debugging storage/maria/ma_loghandler.c: Removed some wrong ';' to get rid of compiler errors when compiling without debugging Indentation fixes Removed not needed 'break's Fixed some compiler warnings Added code to detect logging of unitialized memory storage/maria/ma_page.c: maria_getint() -> maria_data_on_page() Clear rest of index page before writing when used with valgrind (Fixes warning of writing pages with unitialized data) storage/maria/ma_range.c: maria_getint() -> maria_data_on_page() storage/maria/ma_rt_index.c: maria_getint() -> maria_data_on_page() storage/maria/ma_rt_index.h: maria_getint() -> maria_data_on_page() storage/maria/ma_rt_key.c: maria_getint() -> maria_data_on_page() storage/maria/ma_rt_split.c: maria_getint() -> maria_data_on_page() storage/maria/ma_search.c: maria_getint() -> maria_data_on_page() storage/maria/ma_test1.c: Fixed compiler warning storage/maria/ma_write.c: maria_getint() -> maria_data_on_page() storage/maria/maria_chk.c: maria_getint() -> maria_data_on_page() storage/maria/maria_def.h: maria_getint() -> maria_data_on_page() storage/maria/unittest/ma_pagecache_consist.c: Fixed compiler warning storage/maria/unittest/ma_pagecache_single.c: Fixed compiler warning storage/maria/unittest/ma_test_loghandler-t.c: Fixed compiler warning storage/maria/unittest/ma_test_loghandler_multigroup-t.c: Fixed compiler warning storage/maria/unittest/ma_test_loghandler_multithread-t.c: Fixed compiler warning storage/maria/unittest/ma_test_loghandler_pagecache-t.c: Fixed compiler warning storage/myisam/mi_dbug.c: Added missed end++; Caused usage of unitialized memory for nullable keys that was not NULL --- mysys/safemalloc.c | 23 +++++++++++ sql/opt_range.cc | 1 - storage/maria/ma_blockrec.c | 4 +- storage/maria/ma_check.c | 14 +++---- storage/maria/ma_dbug.c | 1 + storage/maria/ma_delete.c | 38 ++++++++--------- storage/maria/ma_init.c | 1 + storage/maria/ma_key.c | 1 + storage/maria/ma_loghandler.c | 48 +++++++++++++--------- storage/maria/ma_page.c | 15 +++++-- storage/maria/ma_range.c | 2 +- storage/maria/ma_rt_index.c | 8 ++-- storage/maria/ma_rt_index.h | 2 +- storage/maria/ma_rt_key.c | 4 +- storage/maria/ma_rt_split.c | 4 +- storage/maria/ma_search.c | 16 ++++---- storage/maria/ma_test1.c | 2 +- storage/maria/ma_write.c | 18 ++++---- storage/maria/maria_chk.c | 2 +- storage/maria/maria_def.h | 2 +- storage/maria/unittest/ma_pagecache_consist.c | 3 +- storage/maria/unittest/ma_pagecache_single.c | 7 +++- storage/maria/unittest/ma_test_loghandler-t.c | 2 +- .../unittest/ma_test_loghandler_multigroup-t.c | 2 +- .../unittest/ma_test_loghandler_multithread-t.c | 3 +- .../unittest/ma_test_loghandler_pagecache-t.c | 2 +- storage/myisam/mi_dbug.c | 1 + 27 files changed, 138 insertions(+), 88 deletions(-) diff --git a/mysys/safemalloc.c b/mysys/safemalloc.c index da15b02345b..5ab6d0dda0d 100644 --- a/mysys/safemalloc.c +++ b/mysys/safemalloc.c @@ -428,6 +428,29 @@ void TERMINATE(FILE *file) } +/* + Report where a piece of memory was allocated + + This is usefull to call from withing a debugger +*/ + + +void sf_malloc_report_allocated(void *memory) +{ + struct st_irem *irem; + for (irem= sf_malloc_root ; irem ; irem=irem->next) + { + char *data= (((char*) irem) + ALIGN_SIZE(sizeof(struct st_irem)) + + sf_malloc_prehunc); + if (data <= (char*) memory && (char*) memory <= data + irem->datasize) + { + printf("%u bytes at 0x%lx, allocated at line %u in '%s'\n", + irem->datasize, (long) data, irem->linenum, irem->filename); + break; + } + } +} + /* Returns 0 if chunk is ok */ static int _checkchunk(register struct st_irem *irem, const char *filename, diff --git a/sql/opt_range.cc b/sql/opt_range.cc index 5789c1a4ae3..76618c9e373 100644 --- a/sql/opt_range.cc +++ b/sql/opt_range.cc @@ -4183,7 +4183,6 @@ static bool ror_intersect_add(ROR_INTERSECT_INFO *info, } info->out_rows *= selectivity_mult; - DBUG_PRINT("info", ("info->total_cost= %g", info->total_cost)); if (is_cpk_scan) { diff --git a/storage/maria/ma_blockrec.c b/storage/maria/ma_blockrec.c index 1d7f96f6557..c747aaeb6cb 100644 --- a/storage/maria/ma_blockrec.c +++ b/storage/maria/ma_blockrec.c @@ -2056,7 +2056,9 @@ static my_bool write_block_record(MARIA_HA *info, blob_length-= (blob_length % FULL_PAGE_SIZE(block_size)); if (blob_length) { - log_array_pos->str= (char*) record + column->offset + length; + memcpy_fixed((byte*) &log_array_pos->str, + record + column->offset + length, + sizeof(byte*)); log_array_pos->length= blob_length; log_entry_length+= blob_length; log_array_pos++; diff --git a/storage/maria/ma_check.c b/storage/maria/ma_check.c index 8755caf2445..8f10c98d0ee 100644 --- a/storage/maria/ma_check.c +++ b/storage/maria/ma_check.c @@ -742,7 +742,7 @@ static int chk_index(HA_CHECK *param, MARIA_HA *info, MARIA_KEYDEF *keyinfo, char llbuff[22]; uint diff_pos[2]; DBUG_ENTER("chk_index"); - DBUG_DUMP("buff",(byte*) buff,maria_getint(buff)); + DBUG_DUMP("buff",(byte*) buff,maria_data_on_page(buff)); /* TODO: implement appropriate check for RTree keys */ if (keyinfo->flag & HA_SPATIAL) @@ -759,7 +759,7 @@ static int chk_index(HA_CHECK *param, MARIA_HA *info, MARIA_KEYDEF *keyinfo, else comp_flag=SEARCH_SAME; /* Keys in positionorder */ nod_flag=_ma_test_if_nod(buff); - used_length=maria_getint(buff); + used_length= maria_data_on_page(buff); keypos=buff+2+nod_flag; endpos=buff+used_length; @@ -2447,7 +2447,7 @@ static int sort_one_index(HA_CHECK *param, MARIA_HA *info, } if ((nod_flag=_ma_test_if_nod(buff)) || keyinfo->flag & HA_FULLTEXT) { - used_length=maria_getint(buff); + used_length= maria_data_on_page(buff); keypos=buff+2+nod_flag; endpos=buff+used_length; for ( ;; ) @@ -2491,7 +2491,7 @@ static int sort_one_index(HA_CHECK *param, MARIA_HA *info, } /* Fill block with zero and write it to the new index file */ - length=maria_getint(buff); + length= maria_data_on_page(buff); bzero((byte*) buff+length,keyinfo->block_length-length); if (my_pwrite(new_file,(byte*) buff,(uint) keyinfo->block_length, new_page_pos,MYF(MY_NABP | MY_WAIT_IF_FULL))) @@ -4403,7 +4403,7 @@ static int sort_insert_key(MARIA_SORT_PARAM *sort_param, lastkey=0; /* No previous key in block */ } else - a_length=maria_getint(anc_buff); + a_length= maria_data_on_page(anc_buff); /* Save pointer to previous block */ if (nod_flag) @@ -4440,7 +4440,7 @@ static int sort_insert_key(MARIA_SORT_PARAM *sort_param, else if (my_pwrite(info->s->kfile.file, anc_buff, (uint) keyinfo->block_length,filepos, param->myf_rw)) DBUG_RETURN(1); - DBUG_DUMP("buff",anc_buff,maria_getint(anc_buff)); + DBUG_DUMP("buff",anc_buff,maria_data_on_page(anc_buff)); /* Write separator-key to block in next level */ if (sort_insert_key(sort_param,key_block+1,key_block->lastkey,filepos)) @@ -4532,7 +4532,7 @@ int _ma_flush_pending_blocks(MARIA_SORT_PARAM *sort_param) for (key_block=sort_info->key_block ; key_block->inited ; key_block++) { key_block->inited=0; - length=maria_getint(key_block->buff); + length= maria_data_on_page(key_block->buff); if (nod_flag) _ma_kpointer(info,key_block->end_pos,filepos); key_file_length=info->state->key_file_length; diff --git a/storage/maria/ma_dbug.c b/storage/maria/ma_dbug.c index 10c570c5794..150385607b6 100644 --- a/storage/maria/ma_dbug.c +++ b/storage/maria/ma_dbug.c @@ -45,6 +45,7 @@ void _ma_print_key(FILE *stream, register HA_KEYSEG *keyseg, fprintf(stream,"NULL"); continue; } + end++; } switch (keyseg->type) { diff --git a/storage/maria/ma_delete.c b/storage/maria/ma_delete.c index 436a65a52ce..54c6b7aaefc 100644 --- a/storage/maria/ma_delete.c +++ b/storage/maria/ma_delete.c @@ -187,7 +187,7 @@ static int _ma_ck_real_delete(register MARIA_HA *info, MARIA_KEYDEF *keyinfo, } else /* error == 1 */ { - if (maria_getint(root_buff) <= (nod_flag=_ma_test_if_nod(root_buff))+3) + if (maria_data_on_page(root_buff) <= (nod_flag=_ma_test_if_nod(root_buff))+3) { error=0; if (nod_flag) @@ -228,7 +228,7 @@ static int d_search(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, my_off_t leaf_page,next_block; byte lastkey[HA_MAX_KEY_BUFF]; DBUG_ENTER("d_search"); - DBUG_DUMP("page",anc_buff,maria_getint(anc_buff)); + DBUG_DUMP("page",anc_buff,maria_data_on_page(anc_buff)); search_key_length= (comp_flag & SEARCH_FIND) ? key_length : USE_WHOLE_KEY; flag=(*keyinfo->bin_search)(info,keyinfo,anc_buff,key, search_key_length, @@ -338,7 +338,7 @@ static int d_search(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, else { /* Found key */ uint tmp; - length=maria_getint(anc_buff); + length= maria_data_on_page(anc_buff); if (!(tmp= remove_key(keyinfo,nod_flag,keypos,lastkey,anc_buff+length, &next_block))) goto err; @@ -375,7 +375,7 @@ static int d_search(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, (byte*) 0,(byte*) 0,(my_off_t) 0,(my_bool) 0); } } - if (ret_value == 0 && maria_getint(anc_buff) > keyinfo->block_length) + if (ret_value == 0 && maria_data_on_page(anc_buff) > keyinfo->block_length) { save_flag=1; ret_value= _ma_split_page(info,keyinfo,key,anc_buff,lastkey,0) | 2; @@ -384,7 +384,7 @@ static int d_search(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, ret_value|= _ma_write_keypage(info,keyinfo,page,DFLT_INIT_HITS,anc_buff); else { - DBUG_DUMP("page",anc_buff,maria_getint(anc_buff)); + DBUG_DUMP("page",anc_buff,maria_data_on_page(anc_buff)); } my_afree(leaf_buff); DBUG_PRINT("exit",("Return: %d",ret_value)); @@ -415,9 +415,9 @@ static int del(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, DBUG_ENTER("del"); DBUG_PRINT("enter",("leaf_page: %ld keypos: 0x%lx", (long) leaf_page, (ulong) keypos)); - DBUG_DUMP("leaf_buff",leaf_buff,maria_getint(leaf_buff)); + DBUG_DUMP("leaf_buff",leaf_buff,maria_data_on_page(leaf_buff)); - endpos= leaf_buff+ maria_getint(leaf_buff); + endpos= leaf_buff+ maria_data_on_page(leaf_buff); if (!(key_start= _ma_get_last_key(info,keyinfo,leaf_buff,keybuff,endpos, &tmp))) DBUG_RETURN(-1); @@ -432,16 +432,16 @@ static int del(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, ret_value= -1; else { - DBUG_DUMP("next_page",next_buff,maria_getint(next_buff)); + DBUG_DUMP("next_page",next_buff,maria_data_on_page(next_buff)); if ((ret_value=del(info,keyinfo,key,anc_buff,next_page,next_buff, keypos,next_block,ret_key)) >0) { - endpos=leaf_buff+maria_getint(leaf_buff); + endpos=leaf_buff+maria_data_on_page(leaf_buff); if (ret_value == 1) { ret_value=underflow(info,keyinfo,leaf_buff,next_page, next_buff,endpos); - if (ret_value == 0 && maria_getint(leaf_buff) > keyinfo->block_length) + if (ret_value == 0 && maria_data_on_page(leaf_buff) > keyinfo->block_length) { ret_value= _ma_split_page(info,keyinfo,key,leaf_buff,ret_key,0) | 2; } @@ -471,7 +471,7 @@ static int del(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, /* Place last key in ancestor page on deleted key position */ - a_length=maria_getint(anc_buff); + a_length= maria_data_on_page(anc_buff); endpos=anc_buff+a_length; if (keypos != anc_buff+2+share->base.key_reflength && !_ma_get_last_key(info,keyinfo,anc_buff,ret_key,keypos,&tmp)) @@ -493,7 +493,7 @@ static int del(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, _ma_kpointer(info,keypos - share->base.key_reflength,next_block); maria_putint(anc_buff,a_length+length,share->base.key_reflength); - DBUG_RETURN( maria_getint(leaf_buff) <= + DBUG_RETURN( maria_data_on_page(leaf_buff) <= (info->quick_mode ? MARIA_MIN_KEYBLOCK_LENGTH : (uint) keyinfo->underflow_block_length)); err: @@ -521,16 +521,16 @@ static int underflow(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, DBUG_ENTER("underflow"); DBUG_PRINT("enter",("leaf_page: %ld keypos: 0x%lx",(long) leaf_page, (ulong) keypos)); - DBUG_DUMP("anc_buff",anc_buff,maria_getint(anc_buff)); - DBUG_DUMP("leaf_buff",leaf_buff,maria_getint(leaf_buff)); + DBUG_DUMP("anc_buff",anc_buff,maria_data_on_page(anc_buff)); + DBUG_DUMP("leaf_buff",leaf_buff,maria_data_on_page(leaf_buff)); buff=info->buff; info->keyread_buff_used=1; next_keypos=keypos; nod_flag=_ma_test_if_nod(leaf_buff); p_length=nod_flag+2; - anc_length=maria_getint(anc_buff); - leaf_length=maria_getint(leaf_buff); + anc_length= maria_data_on_page(anc_buff); + leaf_length= maria_data_on_page(leaf_buff); key_reflength=share->base.key_reflength; if (info->s->keyinfo+info->lastinx == keyinfo) info->page_changed=1; @@ -557,7 +557,7 @@ static int underflow(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, next_page= _ma_kpos(key_reflength,next_keypos); if (!_ma_fetch_keypage(info,keyinfo,next_page,DFLT_INIT_HITS,buff,0)) goto err; - buff_length=maria_getint(buff); + buff_length= maria_data_on_page(buff); DBUG_DUMP("next",buff,buff_length); /* find keys to make a big key-page */ @@ -637,7 +637,7 @@ static int underflow(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, (byte*) 0, (byte*) 0, leaf_key, &s_temp); /* t_length will always be > 0 for a new page !*/ - length=(uint) ((buff+maria_getint(buff))-half_pos); + length=(uint) ((buff+maria_data_on_page(buff))-half_pos); bmove(buff+p_length+t_length, half_pos, (size_t) length); (*keyinfo->store_key)(keyinfo,buff+p_length,&s_temp); maria_putint(buff,length+t_length+p_length,nod_flag); @@ -659,7 +659,7 @@ static int underflow(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, next_page= _ma_kpos(key_reflength,keypos); if (!_ma_fetch_keypage(info,keyinfo,next_page,DFLT_INIT_HITS,buff,0)) goto err; - buff_length=maria_getint(buff); + buff_length= maria_data_on_page(buff); endpos=buff+buff_length; DBUG_DUMP("prev",buff,buff_length); diff --git a/storage/maria/ma_init.c b/storage/maria/ma_init.c index c56e6704729..19b835a837f 100644 --- a/storage/maria/ma_init.c +++ b/storage/maria/ma_init.c @@ -18,6 +18,7 @@ #include "maria_def.h" #include #include "ma_blockrec.h" +#include "trnman_public.h" my_bool maria_inited= FALSE; pthread_mutex_t THR_LOCK_maria; diff --git a/storage/maria/ma_key.c b/storage/maria/ma_key.c index 920b59b5b54..941d5d0665e 100644 --- a/storage/maria/ma_key.c +++ b/storage/maria/ma_key.c @@ -324,6 +324,7 @@ uint _ma_pack_key(register MARIA_HA *info, uint keynr, byte *key, key+=length; } #endif + DBUG_PRINT("exit", ("length: %u", (uint) (key-start_key))); DBUG_RETURN((uint) (key-start_key)); } /* _ma_pack_key */ diff --git a/storage/maria/ma_loghandler.c b/storage/maria/ma_loghandler.c index 6d19f46310d..f398ec90897 100644 --- a/storage/maria/ma_loghandler.c +++ b/storage/maria/ma_loghandler.c @@ -700,7 +700,7 @@ static my_bool translog_buffer_lock(struct st_translog_buffer *buffer) } #else #define translog_buffer_lock(B) \ - pthread_mutex_lock(&B->mutex); + pthread_mutex_lock(&B->mutex) #endif @@ -734,7 +734,7 @@ static my_bool translog_buffer_unlock(struct st_translog_buffer *buffer) } #else #define translog_buffer_unlock(B) \ - pthread_mutex_unlock(&B->mutex); + pthread_mutex_unlock(&B->mutex) #endif @@ -1352,7 +1352,6 @@ static uint16 translog_get_total_chunk_length(byte *page, uint16 offset) if (rec_len + header_len < page_rest) DBUG_RETURN(rec_len + header_len); DBUG_RETURN(page_rest); - break; } case TRANSLOG_CHUNK_FIXED: { @@ -1373,36 +1372,33 @@ static uint16 translog_get_total_chunk_length(byte *page, uint16 offset) (uint) (log_record_type_descriptor[type].fixed_length + 3))); DBUG_RETURN(log_record_type_descriptor[type].fixed_length + 3); } + + ptr= page + offset + 3; /* first compressed LSN */ + length= log_record_type_descriptor[type].fixed_length + 3; + for (i= 0; i < log_record_type_descriptor[type].compressed_LSN; i++) { - ptr= page + offset + 3; /* first compressed LSN */ - length= log_record_type_descriptor[type].fixed_length + 3; - for (i= 0; i < log_record_type_descriptor[type].compressed_LSN; i++) - { - /* first 2 bits is length - 2 */ - uint len= ((((uint8) (*ptr)) & TRANSLOG_CLSN_LEN_BITS) >> 6) + 2; - ptr+= len; - /* subtract economized bytes */ - length-= (TRANSLOG_CLSN_MAX_LEN - len); - } - DBUG_PRINT("info", ("Pseudo-fixed length: %u", length)); - DBUG_RETURN(length); + /* first 2 bits is length - 2 */ + uint len= ((((uint8) (*ptr)) & TRANSLOG_CLSN_LEN_BITS) >> 6) + 2; + ptr+= len; + /* subtract economized bytes */ + length-= (TRANSLOG_CLSN_MAX_LEN - len); } - break; + DBUG_PRINT("info", ("Pseudo-fixed length: %u", length)); + DBUG_RETURN(length); } case TRANSLOG_CHUNK_NOHDR: /* 2 no header chunk (till page end) */ DBUG_PRINT("info", ("TRANSLOG_CHUNK_NOHDR length: %u", (uint) (TRANSLOG_PAGE_SIZE - offset))); DBUG_RETURN(TRANSLOG_PAGE_SIZE - offset); - break; case TRANSLOG_CHUNK_LNGTH: /* 3 chunk with chunk length */ DBUG_PRINT("info", ("TRANSLOG_CHUNK_LNGTH")); DBUG_ASSERT(TRANSLOG_PAGE_SIZE - offset >= 3); DBUG_PRINT("info", ("length: %u", uint2korr(page + offset + 1) + 3)); DBUG_RETURN(uint2korr(page + offset + 1) + 3); - break; default: DBUG_ASSERT(0); + DBUG_RETURN(0); } } @@ -1839,9 +1835,9 @@ static uint16 translog_get_chunk_header_length(byte *page, uint16 offset) { /* TODO: fine header end */ DBUG_ASSERT(0); + DBUG_RETURN(0); /* Keep compiler happy */ } DBUG_RETURN(header_len); - break; } case TRANSLOG_CHUNK_FIXED: { @@ -1861,6 +1857,7 @@ static uint16 translog_get_chunk_header_length(byte *page, uint16 offset) break; default: DBUG_ASSERT(0); + DBUG_RETURN(0); /* Keep compiler happy */ } } @@ -2628,6 +2625,7 @@ translog_write_variable_record_chunk2_page(struct st_translog_parts *parts, DBUG_ENTER("translog_write_variable_record_chunk2_page"); chunk2_header[0]= TRANSLOG_CHUNK_NOHDR; + LINT_INIT(buffer_to_flush); rc= translog_page_next(horizon, cursor, &buffer_to_flush); if (buffer_to_flush != NULL) { @@ -2676,6 +2674,7 @@ translog_write_variable_record_chunk3_page(struct st_translog_parts *parts, byte chunk3_header[1 + 2]; DBUG_ENTER("translog_write_variable_record_chunk3_page"); + LINT_INIT(buffer_to_flush); rc= translog_page_next(horizon, cursor, &buffer_to_flush); if (buffer_to_flush != NULL) { @@ -4144,8 +4143,18 @@ my_bool translog_write_record(LSN *lsn, { uint i; uint len= 0; +#ifdef HAVE_PURIFY + ha_checksum checksum= 0; +#endif for (i= TRANSLOG_INTERNAL_PARTS; i < part_no; i++) + { +#ifdef HAVE_PURIFY + /* Find unitialized bytes early */ + checksum+= my_checksum(checksum, parts_data[i].str, + parts_data[i].length); +#endif len+= parts_data[i].length; + } DBUG_ASSERT(len == rec_len); } #endif @@ -5219,7 +5228,6 @@ static void translog_force_current_buffer_to_finish() } else { - left= 0; log_descriptor.bc.current_page_fill= 0; } diff --git a/storage/maria/ma_page.c b/storage/maria/ma_page.c index 2aaabb1257d..d6b8d5ecd7d 100644 --- a/storage/maria/ma_page.c +++ b/storage/maria/ma_page.c @@ -48,7 +48,7 @@ byte *_ma_fetch_keypage(register MARIA_HA *info, MARIA_KEYDEF *keyinfo, DBUG_RETURN(0); } info->last_keypage=page; - page_size=maria_getint(tmp); + page_size= maria_data_on_page(tmp); if (page_size < 4 || page_size > keyinfo->block_length) { DBUG_PRINT("error",("page %lu had wrong page length: %u", @@ -70,7 +70,7 @@ int _ma_write_keypage(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, { DBUG_ENTER("_ma_write_keypage"); -#ifndef FAST /* Safety check */ +#ifdef EXTRA_DEBUG /* Safety check */ if (page < info->s->base.keystart || page+keyinfo->block_length > info->state->key_file_length || (page & (MARIA_MIN_KEY_BLOCK_LENGTH-1))) @@ -84,7 +84,16 @@ int _ma_write_keypage(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, DBUG_RETURN((-1)); } DBUG_PRINT("page",("write page at: %lu",(long) page)); - DBUG_DUMP("buff",(byte*) buff,maria_getint(buff)); + DBUG_DUMP("buff",(byte*) buff,maria_data_on_page(buff)); +#endif + +#ifdef HAVE_purify + { + /* Clear unitialized part of page to avoid valgrind/purify warnings */ + uint length= maria_data_on_page(buff); + bzero((byte*) buff+length,keyinfo->block_length-length); + length=keyinfo->block_length; + } #endif DBUG_ASSERT(info->s->pagecache->block_size == keyinfo->block_length); diff --git a/storage/maria/ma_range.c b/storage/maria/ma_range.c index 798ca348f92..f91a61259d7 100644 --- a/storage/maria/ma_range.c +++ b/storage/maria/ma_range.c @@ -233,7 +233,7 @@ static uint _ma_keynr(MARIA_HA *info, register MARIA_KEYDEF *keyinfo, uint nod_flag,keynr,max_key; byte t_buff[HA_MAX_KEY_BUFF],*end; - end= page+maria_getint(page); + end= page+maria_data_on_page(page); nod_flag=_ma_test_if_nod(page); page+=2+nod_flag; diff --git a/storage/maria/ma_rt_index.c b/storage/maria/ma_rt_index.c index b61e7ed49a8..27a83e433a4 100644 --- a/storage/maria/ma_rt_index.c +++ b/storage/maria/ma_rt_index.c @@ -824,7 +824,7 @@ static int maria_rtree_delete_req(MARIA_HA *info, MARIA_KEYDEF *keyinfo, if (_ma_write_keypage(info, keyinfo, page, DFLT_INIT_HITS, page_buf)) goto err1; - *page_size = maria_getint(page_buf); + *page_size = maria_data_on_page(page_buf); } goto ok; @@ -839,7 +839,7 @@ static int maria_rtree_delete_req(MARIA_HA *info, MARIA_KEYDEF *keyinfo, if (_ma_write_keypage(info, keyinfo, page, DFLT_INIT_HITS, page_buf)) goto err1; - *page_size = maria_getint(page_buf); + *page_size = maria_data_on_page(page_buf); res = 0; goto ok; } @@ -857,7 +857,7 @@ static int maria_rtree_delete_req(MARIA_HA *info, MARIA_KEYDEF *keyinfo, if (!maria_rtree_key_cmp(keyinfo->seg, key, k, key_length, MBR_EQUAL | MBR_DATA)) { maria_rtree_delete_key(info, page_buf, k, key_length, nod_flag); - *page_size = maria_getint(page_buf); + *page_size = maria_data_on_page(page_buf); if (*page_size == 2) { /* last key in the leaf */ @@ -963,7 +963,7 @@ int maria_rtree_delete(MARIA_HA *info, uint keynr, byte *key, uint key_length) info->buff, 0)) goto err1; nod_flag = _ma_test_if_nod(info->buff); - page_size = maria_getint(info->buff); + page_size = maria_data_on_page(info->buff); if (nod_flag && (page_size == 2 + key_length + nod_flag)) { my_off_t new_root = _ma_kpos(nod_flag, diff --git a/storage/maria/ma_rt_index.h b/storage/maria/ma_rt_index.h index c98422144e2..eae43966aa0 100644 --- a/storage/maria/ma_rt_index.h +++ b/storage/maria/ma_rt_index.h @@ -22,7 +22,7 @@ #define rt_PAGE_FIRST_KEY(page, nod_flag) (page + 2 + nod_flag) #define rt_PAGE_NEXT_KEY(key, key_length, nod_flag) (key + key_length + \ (nod_flag ? nod_flag : info->s->base.rec_reflength)) -#define rt_PAGE_END(page) (page + maria_getint(page)) +#define rt_PAGE_END(page) (page + maria_data_on_page(page)) #define rt_PAGE_MIN_SIZE(block_length) ((uint)(block_length) / 3) diff --git a/storage/maria/ma_rt_key.c b/storage/maria/ma_rt_key.c index d88b2582be4..a27ff23c006 100644 --- a/storage/maria/ma_rt_key.c +++ b/storage/maria/ma_rt_key.c @@ -32,7 +32,7 @@ int maria_rtree_add_key(MARIA_HA *info, MARIA_KEYDEF *keyinfo, byte *key, uint key_length, byte *page_buf, my_off_t *new_page) { - uint page_size = maria_getint(page_buf); + uint page_size = maria_data_on_page(page_buf); uint nod_flag = _ma_test_if_nod(page_buf); if (page_size + key_length + info->s->base.rec_reflength <= @@ -68,7 +68,7 @@ int maria_rtree_add_key(MARIA_HA *info, MARIA_KEYDEF *keyinfo, byte *key, int maria_rtree_delete_key(MARIA_HA *info, byte *page_buf, byte *key, uint key_length, uint nod_flag) { - uint16 page_size = maria_getint(page_buf); + uint16 page_size = maria_data_on_page(page_buf); byte *key_start; key_start= key - nod_flag; diff --git a/storage/maria/ma_rt_split.c b/storage/maria/ma_rt_split.c index 4e0abdcdb6d..6a66c4424eb 100644 --- a/storage/maria/ma_rt_split.c +++ b/storage/maria/ma_rt_split.c @@ -265,7 +265,7 @@ int maria_rtree_split_page(MARIA_HA *info, MARIA_KEYDEF *keyinfo, uint nod_flag= _ma_test_if_nod(page); uint full_length= key_length + (nod_flag ? nod_flag : info->s->base.rec_reflength); - int max_keys= (maria_getint(page)-2) / (full_length); + int max_keys= (maria_data_on_page(page)-2) / (full_length); n_dim = keyinfo->keysegs / 2; @@ -296,7 +296,7 @@ int maria_rtree_split_page(MARIA_HA *info, MARIA_KEYDEF *keyinfo, old_coord = next_coord; if (split_maria_rtree_node(task, max_keys + 1, - maria_getint(page) + full_length + 2, full_length, + maria_data_on_page(page) + full_length + 2, full_length, rt_PAGE_MIN_SIZE(keyinfo->block_length), 2, 2, &next_coord, n_dim)) { diff --git a/storage/maria/ma_search.c b/storage/maria/ma_search.c index 0ec36db59c5..f3e7a0d542a 100644 --- a/storage/maria/ma_search.c +++ b/storage/maria/ma_search.c @@ -80,14 +80,14 @@ int _ma_search(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, info->keyread_buff, test(!(nextflag & SEARCH_SAVE_BUFF))))) goto err; - DBUG_DUMP("page", buff, maria_getint(buff)); + DBUG_DUMP("page", buff, maria_data_on_page(buff)); flag=(*keyinfo->bin_search)(info,keyinfo,buff,key,key_len,nextflag, &keypos,lastkey, &last_key); if (flag == MARIA_FOUND_WRONG_KEY) DBUG_RETURN(-1); nod_flag=_ma_test_if_nod(buff); - maxpos=buff+maria_getint(buff)-1; + maxpos=buff+maria_data_on_page(buff)-1; if (flag) { @@ -187,8 +187,8 @@ int _ma_bin_search(MARIA_HA *info, register MARIA_KEYDEF *keyinfo, byte *page, LINT_INIT(flag); totlength=keyinfo->keylength+(nod_flag=_ma_test_if_nod(page)); start=0; mid=1; - save_end=end=(int) ((maria_getint(page)-2-nod_flag)/totlength-1); - DBUG_PRINT("test",("page_length: %d end: %d",maria_getint(page),end)); + save_end=end=(int) ((maria_data_on_page(page)-2-nod_flag)/totlength-1); + DBUG_PRINT("test",("page_length: %d end: %d",maria_data_on_page(page),end)); page+=2+nod_flag; while (start != end) @@ -249,7 +249,7 @@ int _ma_seq_search(MARIA_HA *info, register MARIA_KEYDEF *keyinfo, byte *page, DBUG_ENTER("_ma_seq_search"); LINT_INIT(flag); LINT_INIT(length); - end= page+maria_getint(page); + end= page+maria_data_on_page(page); nod_flag=_ma_test_if_nod(page); page+=2+nod_flag; *ret_pos=page; @@ -314,7 +314,7 @@ int _ma_prefix_search(MARIA_HA *info, register MARIA_KEYDEF *keyinfo, LINT_INIT(saved_vseg); t_buff[0]=0; /* Avoid bugs */ - end= page+maria_getint(page); + end= page+maria_data_on_page(page); nod_flag=_ma_test_if_nod(page); page+=2+nod_flag; *ret_pos=page; @@ -1324,7 +1324,7 @@ int _ma_search_first(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, info->lastkey))) DBUG_RETURN(-1); /* Crashed */ - info->int_keypos=page; info->int_maxpos=info->keyread_buff+maria_getint(info->keyread_buff)-1; + info->int_keypos=page; info->int_maxpos=info->keyread_buff+maria_data_on_page(info->keyread_buff)-1; info->int_nod_flag=nod_flag; info->int_keytree_version=keyinfo->version; info->last_search_keypage=info->last_keypage; @@ -1361,7 +1361,7 @@ int _ma_search_last(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, info->cur_row.lastpos= HA_OFFSET_ERROR; DBUG_RETURN(-1); } - page= buff+maria_getint(buff); + page= buff+maria_data_on_page(buff); nod_flag=_ma_test_if_nod(buff); } while ((pos= _ma_kpos(nod_flag,page)) != HA_OFFSET_ERROR); diff --git a/storage/maria/ma_test1.c b/storage/maria/ma_test1.c index 3546521e0d1..028e02ab9d1 100644 --- a/storage/maria/ma_test1.c +++ b/storage/maria/ma_test1.c @@ -631,7 +631,7 @@ static struct my_option my_long_options[] = static my_bool get_one_option(int optid, const struct my_option *opt __attribute__((unused)), - char *argument) + char *argument __attribute__((unused))) { switch(optid) { case 'a': diff --git a/storage/maria/ma_write.c b/storage/maria/ma_write.c index c16795c05f0..a87e2d76fc7 100644 --- a/storage/maria/ma_write.c +++ b/storage/maria/ma_write.c @@ -511,7 +511,7 @@ int _ma_insert(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, USE_WHOLE_KEY);); nod_flag=_ma_test_if_nod(anc_buff); - a_length=maria_getint(anc_buff); + a_length= maria_data_on_page(anc_buff); endpos= anc_buff+ a_length; prev_key=(key_pos == anc_buff+2+nod_flag ? (byte*) 0 : key_buff); t_length=(*keyinfo->pack_key)(keyinfo,nod_flag, @@ -630,7 +630,7 @@ int _ma_split_page(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, MARIA_KEY_PARAM s_temp; DBUG_ENTER("maria_split_page"); LINT_INIT(after_key); - DBUG_DUMP("buff",(byte*) buff,maria_getint(buff)); + DBUG_DUMP("buff",(byte*) buff,maria_data_on_page(buff)); if (info->s->keyinfo+info->lastinx == keyinfo) info->page_changed=1; /* Info->buff is used */ @@ -646,7 +646,7 @@ int _ma_split_page(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, DBUG_RETURN(-1); length=(uint) (key_pos-buff); - a_length=maria_getint(buff); + a_length= maria_data_on_page(buff); maria_putint(buff,length,nod_flag); key_pos=after_key; @@ -699,7 +699,7 @@ byte *_ma_find_half_pos(uint nod_flag, MARIA_KEYDEF *keyinfo, byte *page, DBUG_ENTER("_ma_find_half_pos"); key_ref_length=2+nod_flag; - length=maria_getint(page)-key_ref_length; + length= maria_data_on_page(page)-key_ref_length; page+=key_ref_length; if (!(keyinfo->flag & (HA_PACK_KEY | HA_SPACE_PACK_USED | HA_VAR_LENGTH_KEY | @@ -746,7 +746,7 @@ static byte *_ma_find_last_pos(MARIA_KEYDEF *keyinfo, byte *page, DBUG_ENTER("_ma_find_last_pos"); key_ref_length=2; - length=maria_getint(page)-key_ref_length; + length= maria_data_on_page(page)-key_ref_length; page+=key_ref_length; if (!(keyinfo->flag & (HA_PACK_KEY | HA_SPACE_PACK_USED | HA_VAR_LENGTH_KEY | @@ -803,7 +803,7 @@ static int _ma_balance_page(register MARIA_HA *info, MARIA_KEYDEF *keyinfo, DBUG_ENTER("_ma_balance_page"); k_length=keyinfo->keylength; - father_length=maria_getint(father_buff); + father_length= maria_data_on_page(father_buff); father_keylength=k_length+info->s->base.key_reflength; nod_flag=_ma_test_if_nod(curr_buff); curr_keylength=k_length+nod_flag; @@ -831,12 +831,12 @@ static int _ma_balance_page(register MARIA_HA *info, MARIA_KEYDEF *keyinfo, if (!_ma_fetch_keypage(info,keyinfo,next_page,DFLT_INIT_HITS,info->buff,0)) goto err; - DBUG_DUMP("next",(byte*) info->buff,maria_getint(info->buff)); + DBUG_DUMP("next",(byte*) info->buff,maria_data_on_page(info->buff)); /* Test if there is room to share keys */ - left_length=maria_getint(curr_buff); - right_length=maria_getint(buff); + left_length= maria_data_on_page(curr_buff); + right_length= maria_data_on_page(buff); keys=(left_length+right_length-4-nod_flag*2)/curr_keylength; if ((right ? right_length : left_length) + curr_keylength <= diff --git a/storage/maria/maria_chk.c b/storage/maria/maria_chk.c index 36101e7d002..0b82a71f736 100644 --- a/storage/maria/maria_chk.c +++ b/storage/maria/maria_chk.c @@ -1670,7 +1670,7 @@ static int sort_record_index(MARIA_SORT_PARAM *sort_param,MARIA_HA *info, DBUG_RETURN(-1); } } - used_length=maria_getint(buff); + used_length= maria_data_on_page(buff); keypos=buff+2+nod_flag; endpos=buff+used_length; for ( ;; ) diff --git a/storage/maria/maria_def.h b/storage/maria/maria_def.h index 46ffac7cbc2..d9e31e800c4 100644 --- a/storage/maria/maria_def.h +++ b/storage/maria/maria_def.h @@ -485,7 +485,7 @@ struct st_maria_info #define READING_NEXT 1 #define READING_HEADER 2 -#define maria_getint(x) ((uint) mi_uint2korr(x) & 32767) +#define maria_data_on_page(x) ((uint) mi_uint2korr(x) & 32767) #define maria_putint(x,y,nod) { uint16 boh=(nod ? (uint16) 32768 : 0) + (uint16) (y);\ mi_int2store(x,boh); } #define _ma_test_if_nod(x) (x[0] & 128 ? info->s->base.key_reflength : 0) diff --git a/storage/maria/unittest/ma_pagecache_consist.c b/storage/maria/unittest/ma_pagecache_consist.c index 99eea9180a3..39170693573 100644 --- a/storage/maria/unittest/ma_pagecache_consist.c +++ b/storage/maria/unittest/ma_pagecache_consist.c @@ -289,7 +289,8 @@ static void *test_thread_writer(void *arg) DBUG_RETURN(0); } -int main(int argc, char **argv __attribute__((unused))) +int main(int argc __attribute__((unused)), + char **argv __attribute__((unused))) { pthread_t tid; pthread_attr_t thr_attr; diff --git a/storage/maria/unittest/ma_pagecache_single.c b/storage/maria/unittest/ma_pagecache_single.c index 211f080e61d..7b77315e18c 100644 --- a/storage/maria/unittest/ma_pagecache_single.c +++ b/storage/maria/unittest/ma_pagecache_single.c @@ -421,7 +421,9 @@ int simple_big_test() static void *test_thread(void *arg) { - int param=*((int*) arg); +#ifndef DBUG_OFF + int param= *((int*) arg); +#endif my_thread_init(); DBUG_ENTER("test_thread"); @@ -452,7 +454,8 @@ static void *test_thread(void *arg) } -int main(int argc, char **argv __attribute__((unused))) +int main(int argc __attribute__((unused)), + char **argv __attribute__((unused))) { pthread_t tid; pthread_attr_t thr_attr; diff --git a/storage/maria/unittest/ma_test_loghandler-t.c b/storage/maria/unittest/ma_test_loghandler-t.c index 047e9c12bfc..bff5864a5c0 100644 --- a/storage/maria/unittest/ma_test_loghandler-t.c +++ b/storage/maria/unittest/ma_test_loghandler-t.c @@ -92,7 +92,7 @@ static my_bool read_and_check_content(TRANSLOG_HEADER_BUFFER *rec, return check_content(buffer + skip, rec->record_length - skip); } -int main(int argc, char *argv[]) +int main(int argc __attribute__((unused)), char *argv[]) { uint32 i; uint32 rec_len; diff --git a/storage/maria/unittest/ma_test_loghandler_multigroup-t.c b/storage/maria/unittest/ma_test_loghandler_multigroup-t.c index f07ceab1a49..110c35b786a 100644 --- a/storage/maria/unittest/ma_test_loghandler_multigroup-t.c +++ b/storage/maria/unittest/ma_test_loghandler_multigroup-t.c @@ -108,7 +108,7 @@ static uint32 get_len() return rec_len; } -int main(int argc, char *argv[]) +int main(int argc __attribute__((unused)), char *argv[]) { uint32 i; uint32 rec_len; diff --git a/storage/maria/unittest/ma_test_loghandler_multithread-t.c b/storage/maria/unittest/ma_test_loghandler_multithread-t.c index 2651258e290..4afd2b23074 100644 --- a/storage/maria/unittest/ma_test_loghandler_multithread-t.c +++ b/storage/maria/unittest/ma_test_loghandler_multithread-t.c @@ -188,7 +188,8 @@ static void *test_thread_writer(void *arg) } -int main(int argc, char **argv __attribute__ ((unused))) +int main(int argc __attribute__((unused)), + char **argv __attribute__ ((unused))) { uint32 i; uint pagen; diff --git a/storage/maria/unittest/ma_test_loghandler_pagecache-t.c b/storage/maria/unittest/ma_test_loghandler_pagecache-t.c index 13b5afe7444..a56f3f875c6 100644 --- a/storage/maria/unittest/ma_test_loghandler_pagecache-t.c +++ b/storage/maria/unittest/ma_test_loghandler_pagecache-t.c @@ -18,7 +18,7 @@ static char *first_translog_file= (char*)"maria_log.00000001"; static char *file1_name= (char*)"page_cache_test_file_1"; static PAGECACHE_FILE file1; -int main(int argc, char *argv[]) +int main(int argc __attribute__((unused)), char *argv[]) { uint pagen; byte long_tr_id[6]; diff --git a/storage/myisam/mi_dbug.c b/storage/myisam/mi_dbug.c index 07c314c43e6..0808a7e85dd 100644 --- a/storage/myisam/mi_dbug.c +++ b/storage/myisam/mi_dbug.c @@ -45,6 +45,7 @@ void _mi_print_key(FILE *stream, register HA_KEYSEG *keyseg, fprintf(stream,"NULL"); continue; } + end++; } switch (keyseg->type) { -- cgit v1.2.1 From 2170c73ebe61705edd515b0a055d9892bf77642a Mon Sep 17 00:00:00 2001 From: unknown Date: Mon, 11 Jun 2007 12:52:40 +0300 Subject: Small fixes of loghandler code found during a bug investigation. storage/maria/ma_loghandler.c: Variable length record descriptor fixed. Assignment of file number moved out of the loop. --- storage/maria/ma_loghandler.c | 8 ++++---- 1 file changed, 4 insertions(+), 4 deletions(-) diff --git a/storage/maria/ma_loghandler.c b/storage/maria/ma_loghandler.c index f398ec90897..43c5ea666ec 100644 --- a/storage/maria/ma_loghandler.c +++ b/storage/maria/ma_loghandler.c @@ -267,8 +267,7 @@ static LOG_DESC INIT_LOGREC_REDO_PURGE_ROW_TAIL= /* QQQ: TODO: variable and fixed size??? */ static LOG_DESC INIT_LOGREC_REDO_PURGE_BLOCKS= {LOGRECTYPE_VARIABLE_LENGTH, - FILEID_STORE_SIZE + PAGERANGE_STORE_SIZE + PAGE_STORE_SIZE + - PAGERANGE_STORE_SIZE, + 0, FILEID_STORE_SIZE + PAGERANGE_STORE_SIZE + PAGE_STORE_SIZE + PAGERANGE_STORE_SIZE, NULL, NULL, NULL, 0}; @@ -1418,6 +1417,7 @@ static uint16 translog_get_total_chunk_length(byte *page, uint16 offset) static my_bool translog_buffer_flush(struct st_translog_buffer *buffer) { uint32 i; + PAGECACHE_FILE file; DBUG_ENTER("translog_buffer_flush"); DBUG_PRINT("enter", ("Buffer: #%u 0x%lx: " @@ -1441,11 +1441,11 @@ static my_bool translog_buffer_flush(struct st_translog_buffer *buffer) translog_buffer_lock(buffer); } + file.file= buffer->file; for (i= 0; i < buffer->size; i+= TRANSLOG_PAGE_SIZE) { - PAGECACHE_FILE file; - file.file= buffer->file; DBUG_ASSERT(log_descriptor.pagecache->block_size == TRANSLOG_PAGE_SIZE); + DBUG_ASSERT(i + TRANSLOG_PAGE_SIZE <= buffer->size); if (pagecache_write(log_descriptor.pagecache, &file, (LSN_OFFSET(buffer->offset) + i) / TRANSLOG_PAGE_SIZE, -- cgit v1.2.1 From 704b39a158ab3271c19f3e0b921a3136230709ff Mon Sep 17 00:00:00 2001 From: unknown Date: Mon, 11 Jun 2007 16:29:53 +0200 Subject: - moving pagecache.h from include/ to storage/maria as it is Maria- specific - adding TRN::first_undo_lsn, needed to know when a log can be deleted; this variable must be set under log's mutex and that leads to setting TRN::rec_lsn, TRN::undo_lsn and TRN::first_undo_lsn in a inwrite_rec_hook; adding implementation of one hook for REDOs and one for UNDOs. Thus translog_write_record() always uses TRN and so does not need a short_id argument, can find it from TRN. - Monty's patch for the last Valgrind error in the tree. - Log handler's unit tests fail but Sanja says it's known include/Makefile.am: pagecache.h moved and renamed include/maria.h: pagecache.h moved and renamed sql/handler.h: pagecache.h moved and renamed storage/maria/Makefile.am: pagecache.h moved and renamed storage/maria/ha_maria.cc: adding an assertion which sounds logical storage/maria/ma_blockrec.c: trn->rec_lsn and trn->undo_lsn are now set via hooks inside the log record's writing; this allows to also set trn->first_undo_lsn needed to compute the log's low-water mark. The PAGERANGE_STORE_SIZE -> PAGE_STORE_SIZE is Monty's fix to a Valgrind error. storage/maria/ma_loghandler.c: "tcb" renamed to "trn". Log handler now knows what is a transaction, and finds short_id from trn. trn's rec_lsn, undo_lsn, first_undo_lsn are now set by some inwrite_rec_hookS (one for REDOs, one for UNDOs). The HAVE_purify blocks are Monty's fix to a Valgrind error. storage/maria/ma_loghandler.h: Log handler functions use TRN, that needs a forward declaration storage/maria/ma_pagecache.c: pagecache.h was moved and renamed storage/maria/ma_pagecache.h: pagecache.h was moved and renamed storage/maria/ma_pagecaches.c: pagecache.h was moved and renamed storage/maria/trnman.c: initializing some members of TRN. storage/maria/trnman.h: TRN::first_undo_lsn needed for log's low-water mark calculation (which will serve to know which logs can be deleted) storage/maria/unittest/ma_test_loghandler-t.c: translog_write_record() now needs a valid TRN storage/maria/unittest/ma_test_loghandler_multigroup-t.c: translog_write_record() now needs a valid TRN storage/maria/unittest/ma_test_loghandler_multithread-t.c: translog_write_record() now needs a valid TRN storage/maria/unittest/ma_test_loghandler_pagecache-t.c: translog_write_record() now needs a valid TRN storage/maria/unittest/test_file.h: pagecache.h was moved and renamed --- include/Makefile.am | 2 +- include/maria.h | 4 +- include/pagecache.h | 259 --------------------- sql/handler.h | 2 +- storage/maria/Makefile.am | 2 +- storage/maria/ha_maria.cc | 9 +- storage/maria/ma_blockrec.c | 83 +++---- storage/maria/ma_loghandler.c | 151 +++++++----- storage/maria/ma_loghandler.h | 5 +- storage/maria/ma_pagecache.c | 2 +- storage/maria/ma_pagecache.h | 259 +++++++++++++++++++++ storage/maria/ma_pagecaches.c | 2 +- storage/maria/trnman.c | 2 +- storage/maria/trnman.h | 2 +- storage/maria/unittest/ma_test_loghandler-t.c | 20 +- .../unittest/ma_test_loghandler_multigroup-t.c | 25 +- .../unittest/ma_test_loghandler_multithread-t.c | 11 +- .../unittest/ma_test_loghandler_pagecache-t.c | 4 +- storage/maria/unittest/test_file.h | 2 +- 19 files changed, 445 insertions(+), 401 deletions(-) delete mode 100644 include/pagecache.h create mode 100644 storage/maria/ma_pagecache.h diff --git a/include/Makefile.am b/include/Makefile.am index 59f671dd04a..2b590f3d949 100644 --- a/include/Makefile.am +++ b/include/Makefile.am @@ -37,7 +37,7 @@ noinst_HEADERS = config-win.h config-netware.h lf.h my_bit.h \ mysql_version.h.in my_handler.h my_time.h decimal.h \ my_vle.h my_user.h my_atomic.h atomic/nolock.h \ atomic/rwlock.h atomic/x86-gcc.h atomic/x86-msvc.h \ - my_libwrap.h pagecache.h wqueue.h + my_libwrap.h wqueue.h # Remove built files and the symlinked directories CLEANFILES = $(BUILT_SOURCES) readline openssl diff --git a/include/maria.h b/include/maria.h index fbf4bc68c29..5f8e3dcdd41 100644 --- a/include/maria.h +++ b/include/maria.h @@ -27,9 +27,7 @@ extern "C" { #ifndef _m_ctype_h #include #endif -#ifndef _pagecache_h -#include "pagecache.h" -#endif +#include "../storage/maria/ma_pagecache.h" #include "my_handler.h" #include "ft_global.h" #include diff --git a/include/pagecache.h b/include/pagecache.h deleted file mode 100644 index 12906676959..00000000000 --- a/include/pagecache.h +++ /dev/null @@ -1,259 +0,0 @@ -/* Copyright (C) 2006 MySQL AB - - 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; either version 2 of the License, or - (at your option) any later version. - - 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ - -/* Page cache variable structures */ - -#ifndef _pagecache_h -#define _pagecache_h -C_MODE_START - -#include "../storage/maria/ma_loghandler_lsn.h" -#include - -/* Type of the page */ -enum pagecache_page_type -{ - /* - Used only for control page type changing during debugging. This define - should only be using when using DBUG. - */ - PAGECACHE_EMPTY_PAGE, - /* the page does not contain LSN */ - PAGECACHE_PLAIN_PAGE, - /* the page contain LSN (maria tablespace page) */ - PAGECACHE_LSN_PAGE -}; - -/* - This enum describe lock status changing. every type of page cache will - interpret WRITE/READ lock as it need. -*/ -enum pagecache_page_lock -{ - PAGECACHE_LOCK_LEFT_UNLOCKED, /* free -> free */ - PAGECACHE_LOCK_LEFT_READLOCKED, /* read -> read */ - PAGECACHE_LOCK_LEFT_WRITELOCKED, /* write -> write */ - PAGECACHE_LOCK_READ, /* free -> read */ - PAGECACHE_LOCK_WRITE, /* free -> write */ - PAGECACHE_LOCK_READ_UNLOCK, /* read -> free */ - PAGECACHE_LOCK_WRITE_UNLOCK, /* write -> free */ - PAGECACHE_LOCK_WRITE_TO_READ /* write -> read */ -}; -/* - This enum describe pin status changing -*/ -enum pagecache_page_pin -{ - PAGECACHE_PIN_LEFT_PINNED, /* pinned -> pinned */ - PAGECACHE_PIN_LEFT_UNPINNED, /* unpinned -> unpinned */ - PAGECACHE_PIN, /* unpinned -> pinned */ - PAGECACHE_UNPIN /* pinned -> unpinned */ -}; -/* How to write the page */ -enum pagecache_write_mode -{ - /* do not write immediately, i.e. it will be dirty page */ - PAGECACHE_WRITE_DELAY, - /* page already is in the file. (key cache insert analogue) */ - PAGECACHE_WRITE_DONE -}; - -typedef void *PAGECACHE_PAGE_LINK; - -/* file descriptor for Maria */ -typedef struct st_pagecache_file -{ - File file; -} PAGECACHE_FILE; - -/* page number for maria */ -typedef uint32 pgcache_page_no_t; - -/* declare structures that is used by st_pagecache */ - -struct st_pagecache_block_link; -typedef struct st_pagecache_block_link PAGECACHE_BLOCK_LINK; -struct st_pagecache_page; -typedef struct st_pagecache_page PAGECACHE_PAGE; -struct st_pagecache_hash_link; -typedef struct st_pagecache_hash_link PAGECACHE_HASH_LINK; - -#include - -typedef my_bool (*pagecache_disk_read_validator)(byte *page, gptr data); - -#define PAGECACHE_CHANGED_BLOCKS_HASH 128 /* must be power of 2 */ - -/* - The page cache structure - It also contains read-only statistics parameters. -*/ - -typedef struct st_pagecache -{ - my_bool inited; - my_bool resize_in_flush; /* true during flush of resize operation */ - my_bool can_be_used; /* usage of cache for read/write is allowed */ - uint shift; /* block size = 2 ^ shift */ - my_size_t mem_size; /* specified size of the cache memory */ - uint32 block_size; /* size of the page buffer of a cache block */ - ulong min_warm_blocks; /* min number of warm blocks; */ - ulong age_threshold; /* age threshold for hot blocks */ - ulonglong time; /* total number of block link operations */ - uint hash_entries; /* max number of entries in the hash table */ - int hash_links; /* max number of hash links */ - int hash_links_used; /* number of hash links taken from free links pool */ - int disk_blocks; /* max number of blocks in the cache */ - ulong blocks_used; /* maximum number of concurrently used blocks */ - ulong blocks_unused; /* number of currently unused blocks */ - ulong blocks_changed; /* number of currently dirty blocks */ - ulong warm_blocks; /* number of blocks in warm sub-chain */ - ulong cnt_for_resize_op; /* counter to block resize operation */ - ulong blocks_available; /* number of blocks available in the LRU chain */ - PAGECACHE_HASH_LINK **hash_root;/* arr. of entries into hash table buckets */ - PAGECACHE_HASH_LINK *hash_link_root;/* memory for hash table links */ - PAGECACHE_HASH_LINK *free_hash_list;/* list of free hash links */ - PAGECACHE_BLOCK_LINK *free_block_list;/* list of free blocks */ - PAGECACHE_BLOCK_LINK *block_root;/* memory for block links */ - byte HUGE_PTR *block_mem; /* memory for block buffers */ - PAGECACHE_BLOCK_LINK *used_last;/* ptr to the last block of the LRU chain */ - PAGECACHE_BLOCK_LINK *used_ins;/* ptr to the insertion block in LRU chain */ - pthread_mutex_t cache_lock; /* to lock access to the cache structure */ - WQUEUE resize_queue; /* threads waiting during resize operation */ - WQUEUE waiting_for_hash_link;/* waiting for a free hash link */ - WQUEUE waiting_for_block; /* requests waiting for a free block */ - /* hash for dirty file bl.*/ - PAGECACHE_BLOCK_LINK *changed_blocks[PAGECACHE_CHANGED_BLOCKS_HASH]; - /* hash for other file bl.*/ - PAGECACHE_BLOCK_LINK *file_blocks[PAGECACHE_CHANGED_BLOCKS_HASH]; - - /* - The following variables are and variables used to hold parameters for - initializing the key cache. - */ - - ulonglong param_buff_size; /* size the memory allocated for the cache */ - ulong param_block_size; /* size of the blocks in the key cache */ - ulong param_division_limit; /* min. percentage of warm blocks */ - ulong param_age_threshold; /* determines when hot block is downgraded */ - - /* Statistics variables. These are reset in reset_pagecache_counters(). */ - ulong global_blocks_changed; /* number of currently dirty blocks */ - ulonglong global_cache_w_requests;/* number of write requests (write hits) */ - ulonglong global_cache_write; /* number of writes from cache to files */ - ulonglong global_cache_r_requests;/* number of read requests (read hits) */ - ulonglong global_cache_read; /* number of reads from files to cache */ - - int blocks; /* max number of blocks in the cache */ - my_bool in_init; /* Set to 1 in MySQL during init/resize */ -} PAGECACHE; - -/* The default key cache */ -extern PAGECACHE dflt_pagecache_var, *dflt_pagecache; - -extern int init_pagecache(PAGECACHE *pagecache, my_size_t use_mem, - uint division_limit, uint age_threshold, - uint block_size); -extern int resize_pagecache(PAGECACHE *pagecache, - my_size_t use_mem, uint division_limit, - uint age_threshold); -extern void change_pagecache_param(PAGECACHE *pagecache, uint division_limit, - uint age_threshold); - -#define pagecache_read(P,F,N,L,B,T,K,I) \ - pagecache_valid_read(P,F,N,L,B,T,K,I,0,0) - -extern byte *pagecache_valid_read(PAGECACHE *pagecache, - PAGECACHE_FILE *file, - pgcache_page_no_t pageno, - uint level, - byte *buff, - enum pagecache_page_type type, - enum pagecache_page_lock lock, - PAGECACHE_PAGE_LINK *link, - pagecache_disk_read_validator validator, - gptr validator_data); - -#define pagecache_write(P,F,N,L,B,T,O,I,M,K) \ - pagecache_write_part(P,F,N,L,B,T,O,I,M,K,0,(P)->block_size) - -extern my_bool pagecache_write_part(PAGECACHE *pagecache, - PAGECACHE_FILE *file, - pgcache_page_no_t pageno, - uint level, - byte *buff, - enum pagecache_page_type type, - enum pagecache_page_lock lock, - enum pagecache_page_pin pin, - enum pagecache_write_mode write_mode, - PAGECACHE_PAGE_LINK *link, - uint offset, - uint size); -extern void pagecache_unlock(PAGECACHE *pagecache, - PAGECACHE_FILE *file, - pgcache_page_no_t pageno, - enum pagecache_page_lock lock, - enum pagecache_page_pin pin, - LSN first_REDO_LSN_for_page, - LSN lsn); -extern void pagecache_unlock_by_link(PAGECACHE *pagecache, - PAGECACHE_PAGE_LINK *link, - enum pagecache_page_lock lock, - enum pagecache_page_pin pin, - LSN first_REDO_LSN_for_page, - LSN lsn); -extern void pagecache_unpin(PAGECACHE *pagecache, - PAGECACHE_FILE *file, - pgcache_page_no_t pageno, - LSN lsn); -extern void pagecache_unpin_by_link(PAGECACHE *pagecache, - PAGECACHE_PAGE_LINK *link, - LSN lsn); -extern int flush_pagecache_blocks(PAGECACHE *keycache, - PAGECACHE_FILE *file, - enum flush_type type); -extern my_bool pagecache_delete(PAGECACHE *pagecache, - PAGECACHE_FILE *file, - pgcache_page_no_t pageno, - enum pagecache_page_lock lock, - my_bool flush); -extern my_bool pagecache_delete_pages(PAGECACHE *pagecache, - PAGECACHE_FILE *file, - pgcache_page_no_t pageno, - uint page_count, - enum pagecache_page_lock lock, - my_bool flush); -extern void end_pagecache(PAGECACHE *keycache, my_bool cleanup); -extern my_bool pagecache_collect_changed_blocks_with_lsn(PAGECACHE *pagecache, - LEX_STRING *str, - LSN *max_lsn); -extern int reset_pagecache_counters(const char *name, PAGECACHE *pagecache); - - -/* Functions to handle multiple key caches */ -extern my_bool multi_pagecache_init(void); -extern void multi_pagecache_free(void); -extern PAGECACHE *multi_pagecache_search(byte *key, uint length, - PAGECACHE *def); -extern my_bool multi_pagecache_set(const byte *key, uint length, - PAGECACHE *pagecache); -extern void multi_pagecache_change(PAGECACHE *old_data, - PAGECACHE *new_data); -extern int reset_pagecache_counters(const char *name, - PAGECACHE *pagecache); - -C_MODE_END -#endif /* _keycache_h */ diff --git a/sql/handler.h b/sql/handler.h index d488ca5f07e..94d2997a545 100644 --- a/sql/handler.h +++ b/sql/handler.h @@ -23,7 +23,7 @@ #include #include #include -#include +#include "../storage/maria/ma_pagecache.h" #ifndef NO_HASH #define NO_HASH /* Not yet implemented */ diff --git a/storage/maria/Makefile.am b/storage/maria/Makefile.am index 39160106a2c..9d8ab704541 100644 --- a/storage/maria/Makefile.am +++ b/storage/maria/Makefile.am @@ -54,7 +54,7 @@ noinst_HEADERS = maria_def.h ma_rt_index.h ma_rt_key.h ma_rt_mbr.h \ ma_sp_defs.h ma_fulltext.h ma_ftdefs.h ma_ft_test1.h \ ma_ft_eval.h trnman.h lockman.h tablockman.h \ ma_control_file.h ha_maria.h ma_blockrec.h \ - ma_loghandler.h ma_loghandler_lsn.h + ma_loghandler.h ma_loghandler_lsn.h ma_pagecache.h ma_test1_DEPENDENCIES= $(LIBRARIES) ma_test1_LDADD= @CLIENT_EXTRA_LDFLAGS@ libmaria.a \ $(top_builddir)/storage/myisam/libmyisam.a \ diff --git a/storage/maria/ha_maria.cc b/storage/maria/ha_maria.cc index f7fd417836a..288366675a7 100644 --- a/storage/maria/ha_maria.cc +++ b/storage/maria/ha_maria.cc @@ -1919,8 +1919,15 @@ int ha_maria::start_stmt(THD *thd, thr_lock_type lock_type) if (file->s->base.transactional) { DBUG_ASSERT(trn); // this may be called only after external_lock() + DBUG_ASSERT(trnman_has_locked_tables(trn)); DBUG_ASSERT(lock_type != F_UNLCK); - /* As external_lock() was already called, don't increment locked_tables */ + /* + As external_lock() was already called, don't increment locked_tables. + Note that we call the function below possibly several times when + statement starts (once per table). This is ok as long as that function + does cheap operations. Otherwise, we will need to do it only on first + call to start_stmt(). + */ trnman_new_statement(trn); } return 0; diff --git a/storage/maria/ma_blockrec.c b/storage/maria/ma_blockrec.c index c747aaeb6cb..800ed8f14ac 100644 --- a/storage/maria/ma_blockrec.c +++ b/storage/maria/ma_blockrec.c @@ -1141,12 +1141,9 @@ static my_bool write_tail(MARIA_HA *info, log_array[TRANSLOG_INTERNAL_PARTS + 0].length= sizeof(log_data); log_array[TRANSLOG_INTERNAL_PARTS + 1].str= (char*) row_pos.data; log_array[TRANSLOG_INTERNAL_PARTS + 1].length= length; - if (translog_write_record(!info->trn->rec_lsn ? &info->trn->rec_lsn : &lsn, - LOGREC_REDO_INSERT_ROW_TAIL, - info->trn->short_id, NULL, share, - sizeof(log_data) + length, - TRANSLOG_INTERNAL_PARTS + 2, - log_array)) + if (translog_write_record(&lsn, LOGREC_REDO_INSERT_ROW_TAIL, + info->trn, share, sizeof(log_data) + length, + TRANSLOG_INTERNAL_PARTS + 2, log_array)) DBUG_RETURN(1); } @@ -1398,10 +1395,8 @@ static my_bool free_full_pages(MARIA_HA *info, MARIA_ROW *row) log_array[TRANSLOG_INTERNAL_PARTS + 0].length= sizeof(log_data); log_array[TRANSLOG_INTERNAL_PARTS + 1].str= row->extents; log_array[TRANSLOG_INTERNAL_PARTS + 1].length= extents_length; - if (translog_write_record(!info->trn->rec_lsn ? &info->trn->rec_lsn : &lsn, - LOGREC_REDO_PURGE_BLOCKS, - info->trn->short_id, NULL, info->s, - sizeof(log_data) + extents_length, + if (translog_write_record(&lsn, LOGREC_REDO_PURGE_BLOCKS, info->trn, + info->s, sizeof(log_data) + extents_length, TRANSLOG_INTERNAL_PARTS + 2, log_array)) DBUG_RETURN(1); @@ -1416,9 +1411,6 @@ static my_bool free_full_pages(MARIA_HA *info, MARIA_ROW *row) NOTES This is very similar to free_full_pages() - We don't have to update trn->rec_lsn here as before calling this function - we have already generated REDO's for deleting the HEAD block. - RETURN 0 ok 1 error @@ -1449,8 +1441,7 @@ static my_bool free_full_page_range(MARIA_HA *info, ulonglong page, uint count) log_array[TRANSLOG_INTERNAL_PARTS + 0].length= sizeof(log_data); if (translog_write_record(&lsn, LOGREC_REDO_PURGE_BLOCKS, - info->trn->short_id, NULL, info->s, - sizeof(log_data), + info->trn, info->s, sizeof(log_data), TRANSLOG_INTERNAL_PARTS + 1, log_array)) res= 1; @@ -1957,10 +1948,8 @@ static my_bool write_block_record(MARIA_HA *info, log_array[TRANSLOG_INTERNAL_PARTS + 0].length= sizeof(log_data); log_array[TRANSLOG_INTERNAL_PARTS + 1].str= (char*) row_pos->data; log_array[TRANSLOG_INTERNAL_PARTS + 1].length= data_length; - if (translog_write_record(!info->trn->rec_lsn ? &info->trn->rec_lsn : &lsn, - LOGREC_REDO_INSERT_ROW_HEAD, - info->trn->short_id, NULL, share, - sizeof(log_data) + data_length, + if (translog_write_record(&lsn, LOGREC_REDO_INSERT_ROW_HEAD, info->trn, + share, sizeof(log_data) + data_length, TRANSLOG_INTERNAL_PARTS + 2, log_array)) goto disk_err; } @@ -2077,9 +2066,8 @@ static my_bool write_block_record(MARIA_HA *info, /* trn->rec_lsn is already set earlier in this function */ error= translog_write_record(&lsn, LOGREC_REDO_INSERT_ROW_BLOBS, - info->trn->short_id, NULL, share, - log_entry_length, (uint) (log_array_pos - - log_array), + info->trn, share, log_entry_length, + (uint) (log_array_pos - log_array), log_array); if (log_array != tmp_log_array) my_free((gptr) log_array, MYF(0)); @@ -2109,11 +2097,9 @@ static my_bool write_block_record(MARIA_HA *info, if (!old_record) { /* Write UNDO log record for the INSERT */ - if (translog_write_record(&info->trn->undo_lsn, LOGREC_UNDO_ROW_INSERT, - info->trn->short_id, NULL, share, - sizeof(log_data), - TRANSLOG_INTERNAL_PARTS + 1, - log_array)) + if (translog_write_record(&lsn, LOGREC_UNDO_ROW_INSERT, + info->trn, share, sizeof(log_data), + TRANSLOG_INTERNAL_PARTS + 1, log_array)) goto disk_err; } else @@ -2125,9 +2111,8 @@ static my_bool write_block_record(MARIA_HA *info, info->log_row_parts + TRANSLOG_INTERNAL_PARTS + 1, &row_parts_count); - if (translog_write_record(&info->trn->undo_lsn, LOGREC_UNDO_ROW_UPDATE, - info->trn->short_id, NULL, share, - sizeof(log_data) + row_length, + if (translog_write_record(&lsn, LOGREC_UNDO_ROW_UPDATE, info->trn, + share, sizeof(log_data) + row_length, TRANSLOG_INTERNAL_PARTS + 1 + row_parts_count, log_array)) goto disk_err; @@ -2293,6 +2278,7 @@ my_bool _ma_write_abort_block_record(MARIA_HA *info) if (info->s->base.transactional) { + LSN lsn; LEX_STRING log_array[TRANSLOG_INTERNAL_PARTS + 1]; uchar log_data[LSN_STORE_SIZE]; @@ -2302,15 +2288,16 @@ my_bool _ma_write_abort_block_record(MARIA_HA *info) really undo a failed insert. Note that this UNDO will cause recover to ignore the LOGREC_UNDO_ROW_INSERT that is the previous entry in the UNDO chain. + We will soon change that: we will here execute the UNDO records + generated while we were trying to write the row; this will log some CLRs + which will replace this LOGREC_UNDO_PURGE. RECOVERY TODO BUG. */ lsn_store(log_data, info->trn->undo_lsn); log_array[TRANSLOG_INTERNAL_PARTS + 0].str= (char*) log_data; log_array[TRANSLOG_INTERNAL_PARTS + 0].length= sizeof(log_data); - if (translog_write_record(&info->trn->undo_lsn, LOGREC_UNDO_ROW_PURGE, - info->trn->short_id, NULL, info->s, - sizeof(log_data), - TRANSLOG_INTERNAL_PARTS + 1, - log_array)) + if (translog_write_record(&lsn, LOGREC_UNDO_ROW_PURGE, + info->trn, info->s, sizeof(log_data), + TRANSLOG_INTERNAL_PARTS + 1, log_array)) res= 1; } _ma_unpin_all_pages(info, info->trn->undo_lsn); @@ -2534,12 +2521,10 @@ static my_bool delete_head_or_tail(MARIA_HA *info, log_array[TRANSLOG_INTERNAL_PARTS + 0].str= (char*) log_data; log_array[TRANSLOG_INTERNAL_PARTS + 0].length= sizeof(log_data); - if (translog_write_record(!info->trn->rec_lsn ? &info->trn->rec_lsn : &lsn, - (head ? LOGREC_REDO_PURGE_ROW_HEAD : - LOGREC_REDO_PURGE_ROW_TAIL), - info->trn->short_id, NULL, share, - sizeof(log_data), TRANSLOG_INTERNAL_PARTS + 1, - log_array)) + if (translog_write_record(&lsn, (head ? LOGREC_REDO_PURGE_ROW_HEAD : + LOGREC_REDO_PURGE_ROW_TAIL), + info->trn, share, sizeof(log_data), + TRANSLOG_INTERNAL_PARTS + 1, log_array)) DBUG_RETURN(1); if (pagecache_write(share->pagecache, &info->dfile, page, 0, @@ -2564,14 +2549,12 @@ static my_bool delete_head_or_tail(MARIA_HA *info, pagerange_store(log_data + FILEID_STORE_SIZE, 1); page_store(log_data+ FILEID_STORE_SIZE + PAGERANGE_STORE_SIZE, page); pagerange_store(log_data + FILEID_STORE_SIZE + PAGERANGE_STORE_SIZE + - PAGERANGE_STORE_SIZE, 1); + PAGE_STORE_SIZE, 1); log_array[TRANSLOG_INTERNAL_PARTS + 0].str= (char*) log_data; log_array[TRANSLOG_INTERNAL_PARTS + 0].length= sizeof(log_data); - if (translog_write_record(!info->trn->rec_lsn ? &info->trn->rec_lsn : &lsn, - LOGREC_REDO_PURGE_BLOCKS, - info->trn->short_id, NULL, share, - sizeof(log_data), TRANSLOG_INTERNAL_PARTS + 1, - log_array)) + if (translog_write_record(&lsn, LOGREC_REDO_PURGE_BLOCKS, + info->trn, share, sizeof(log_data), + TRANSLOG_INTERNAL_PARTS + 1, log_array)) DBUG_RETURN(1); DBUG_ASSERT(empty_space >= info->s->bitmap.sizes[0]); } @@ -2640,6 +2623,7 @@ my_bool _ma_delete_block_record(MARIA_HA *info, const byte *record) if (info->s->base.transactional) { + LSN lsn; uchar log_data[LSN_STORE_SIZE + FILEID_STORE_SIZE + PAGE_STORE_SIZE + DIR_COUNT_SIZE]; size_t row_length; @@ -2658,9 +2642,8 @@ my_bool _ma_delete_block_record(MARIA_HA *info, const byte *record) TRANSLOG_INTERNAL_PARTS + 1, &row_parts_count); - if (translog_write_record(&info->trn->undo_lsn, LOGREC_UNDO_ROW_DELETE, - info->trn->short_id, NULL, info->s, - sizeof(log_data) + row_length, + if (translog_write_record(&lsn, LOGREC_UNDO_ROW_DELETE, info->trn, + info->s, sizeof(log_data) + row_length, TRANSLOG_INTERNAL_PARTS + 1 + row_parts_count, info->log_row_parts)) goto err; diff --git a/storage/maria/ma_loghandler.c b/storage/maria/ma_loghandler.c index f398ec90897..6551e8ee21f 100644 --- a/storage/maria/ma_loghandler.c +++ b/storage/maria/ma_loghandler.c @@ -15,6 +15,7 @@ #include "maria_def.h" #include "ma_blockrec.h" +#include "trnman.h" /* number of opened log files in the pagecache (should be at least 2) */ #define OPENED_FILES_NUM 3 @@ -187,11 +188,11 @@ enum record_class #define TRANSLOG_CLSN_MAX_LEN 5 /* Maximum length of compressed LSN */ typedef my_bool(*prewrite_rec_hook) (enum translog_record_type type, - void *tcb, struct st_maria_share *share, + TRN *trn, struct st_maria_share *share, struct st_translog_parts *parts); typedef my_bool(*inwrite_rec_hook) (enum translog_record_type type, - void *tcb, + TRN *trn, LSN *lsn, struct st_translog_parts *parts); @@ -225,6 +226,13 @@ struct st_log_record_type_descriptor }; +static my_bool write_hook_for_redo(enum translog_record_type type, + TRN *trn, LSN *lsn, + struct st_translog_parts *parts); +static my_bool write_hook_for_undo(enum translog_record_type type, + TRN *trn, LSN *lsn, + struct st_translog_parts *parts); + /* Initialize log_record_type_descriptors @@ -240,29 +248,31 @@ static LOG_DESC INIT_LOGREC_RESERVED_FOR_CHUNKS23= static LOG_DESC INIT_LOGREC_REDO_INSERT_ROW_HEAD= {LOGRECTYPE_VARIABLE_LENGTH, 0, - FILEID_STORE_SIZE + PAGE_STORE_SIZE + DIRPOS_STORE_SIZE, NULL, NULL, NULL, 0}; + FILEID_STORE_SIZE + PAGE_STORE_SIZE + DIRPOS_STORE_SIZE, NULL, + write_hook_for_redo, NULL, 0}; static LOG_DESC INIT_LOGREC_REDO_INSERT_ROW_TAIL= -{LOGRECTYPE_VARIABLE_LENGTH, 0, 9, NULL, NULL, NULL, 0}; +/* QQ shouldn't this 9 be 8? */ +{LOGRECTYPE_VARIABLE_LENGTH, 0, 9, NULL, write_hook_for_redo, NULL, 0}; static LOG_DESC INIT_LOGREC_REDO_INSERT_ROW_BLOB= -{LOGRECTYPE_VARIABLE_LENGTH, 0, 8, NULL, NULL, NULL, 0}; +{LOGRECTYPE_VARIABLE_LENGTH, 0, 8, NULL, write_hook_for_redo, NULL, 0}; /*QQQ:TODO:header???*/ static LOG_DESC INIT_LOGREC_REDO_INSERT_ROW_BLOBS= -{LOGRECTYPE_VARIABLE_LENGTH, 0, 0, NULL, NULL, NULL, 0}; +{LOGRECTYPE_VARIABLE_LENGTH, 0, 0, NULL, write_hook_for_redo, NULL, 0}; static LOG_DESC INIT_LOGREC_REDO_PURGE_ROW_HEAD= {LOGRECTYPE_FIXEDLENGTH, FILEID_STORE_SIZE + PAGE_STORE_SIZE + DIRPOS_STORE_SIZE, FILEID_STORE_SIZE + PAGE_STORE_SIZE + DIRPOS_STORE_SIZE, - NULL, NULL, NULL, 0}; + NULL, write_hook_for_redo, NULL, 0}; static LOG_DESC INIT_LOGREC_REDO_PURGE_ROW_TAIL= {LOGRECTYPE_FIXEDLENGTH, FILEID_STORE_SIZE + PAGE_STORE_SIZE + DIRPOS_STORE_SIZE, FILEID_STORE_SIZE + PAGE_STORE_SIZE + DIRPOS_STORE_SIZE, - NULL, NULL, NULL, 0}; + NULL, write_hook_for_redo, NULL, 0}; /* QQQ: TODO: variable and fixed size??? */ static LOG_DESC INIT_LOGREC_REDO_PURGE_BLOCKS= @@ -271,22 +281,22 @@ static LOG_DESC INIT_LOGREC_REDO_PURGE_BLOCKS= PAGERANGE_STORE_SIZE, FILEID_STORE_SIZE + PAGERANGE_STORE_SIZE + PAGE_STORE_SIZE + PAGERANGE_STORE_SIZE, - NULL, NULL, NULL, 0}; + NULL, write_hook_for_redo, NULL, 0}; static LOG_DESC INIT_LOGREC_REDO_DELETE_ROW= -{LOGRECTYPE_FIXEDLENGTH, 16, 16, NULL, NULL, NULL, 0}; +{LOGRECTYPE_FIXEDLENGTH, 16, 16, NULL, write_hook_for_redo, NULL, 0}; static LOG_DESC INIT_LOGREC_REDO_UPDATE_ROW_HEAD= -{LOGRECTYPE_VARIABLE_LENGTH, 0, 9, NULL, NULL, NULL, 0}; +{LOGRECTYPE_VARIABLE_LENGTH, 0, 9, NULL, write_hook_for_redo, NULL, 0}; static LOG_DESC INIT_LOGREC_REDO_INDEX= -{LOGRECTYPE_VARIABLE_LENGTH, 0, 9, NULL, NULL, NULL, 0}; +{LOGRECTYPE_VARIABLE_LENGTH, 0, 9, NULL, write_hook_for_redo, NULL, 0}; static LOG_DESC INIT_LOGREC_REDO_UNDELETE_ROW= -{LOGRECTYPE_FIXEDLENGTH, 16, 16, NULL, NULL, NULL, 0}; +{LOGRECTYPE_FIXEDLENGTH, 16, 16, NULL, write_hook_for_redo, NULL, 0}; static LOG_DESC INIT_LOGREC_CLR_END= -{LOGRECTYPE_PSEUDOFIXEDLENGTH, 5, 5, NULL, NULL, NULL, 1}; +{LOGRECTYPE_PSEUDOFIXEDLENGTH, 5, 5, NULL, write_hook_for_redo, NULL, 1}; static LOG_DESC INIT_LOGREC_PURGE_END= {LOGRECTYPE_PSEUDOFIXEDLENGTH, 5, 5, NULL, NULL, NULL, 1}; @@ -295,27 +305,27 @@ static LOG_DESC INIT_LOGREC_UNDO_ROW_INSERT= {LOGRECTYPE_FIXEDLENGTH, LSN_STORE_SIZE + FILEID_STORE_SIZE + PAGE_STORE_SIZE + DIRPOS_STORE_SIZE, LSN_STORE_SIZE + FILEID_STORE_SIZE + PAGE_STORE_SIZE + DIRPOS_STORE_SIZE, - NULL, NULL, NULL, 0}; + NULL, write_hook_for_undo, NULL, 0}; static LOG_DESC INIT_LOGREC_UNDO_ROW_DELETE= {LOGRECTYPE_VARIABLE_LENGTH, 0, LSN_STORE_SIZE + FILEID_STORE_SIZE + PAGE_STORE_SIZE + DIRPOS_STORE_SIZE, - NULL, NULL, NULL, 0}; + NULL, write_hook_for_undo, NULL, 0}; static LOG_DESC INIT_LOGREC_UNDO_ROW_UPDATE= {LOGRECTYPE_VARIABLE_LENGTH, 0, LSN_STORE_SIZE + FILEID_STORE_SIZE + PAGE_STORE_SIZE + DIRPOS_STORE_SIZE, - NULL, NULL, NULL, 1}; + NULL, write_hook_for_undo, NULL, 1}; static LOG_DESC INIT_LOGREC_UNDO_ROW_PURGE= {LOGRECTYPE_PSEUDOFIXEDLENGTH, LSN_STORE_SIZE, LSN_STORE_SIZE, NULL, NULL, NULL, 1}; static LOG_DESC INIT_LOGREC_UNDO_KEY_INSERT= -{LOGRECTYPE_VARIABLE_LENGTH, 0, 10, NULL, NULL, NULL, 1}; +{LOGRECTYPE_VARIABLE_LENGTH, 0, 10, NULL, write_hook_for_undo, NULL, 1}; static LOG_DESC INIT_LOGREC_UNDO_KEY_DELETE= -{LOGRECTYPE_VARIABLE_LENGTH, 0, 15, NULL, NULL, NULL, 0}; +{LOGRECTYPE_VARIABLE_LENGTH, 0, 15, NULL, write_hook_for_undo, NULL, 0}; static LOG_DESC INIT_LOGREC_PREPARE= {LOGRECTYPE_VARIABLE_LENGTH, 0, 0, NULL, NULL, NULL, 0}; @@ -2528,7 +2538,7 @@ static my_bool translog_write_parts_on_page(TRANSLOG_ADDRESS *horizon, translog_write_variable_record_1group_header() parts Descriptor of record source parts type The log record type - short_trid Sort transaction ID or 0 if it has no sense + short_trid Short transaction ID or 0 if it has no sense header_length Calculated header length of chunk type 0 chunk0_header Buffer for the chunk header writing */ @@ -2915,12 +2925,12 @@ static translog_size_t translog_get_current_group_size() translog_write_variable_record_1group() lsn LSN of the record will be written here type the log record type - short_trid Sort transaction ID or 0 if it has no sense + short_trid Short transaction ID or 0 if it has no sense parts Descriptor of record source parts buffer_to_flush Buffer which have to be flushed if it is not 0 header_length Calculated header length of chunk type 0 - tcb Transaction control block pointer for hooks by - record log type + trn Transaction structure pointer for hooks by + record log type, for short_id RETURN 0 OK @@ -2934,7 +2944,7 @@ translog_write_variable_record_1group(LSN *lsn, struct st_translog_parts *parts, struct st_translog_buffer *buffer_to_flush, uint16 header_length, - void *tcb) + TRN *trn) { TRANSLOG_ADDRESS horizon; struct st_buffer_cursor cursor; @@ -2947,7 +2957,7 @@ translog_write_variable_record_1group(LSN *lsn, *lsn= horizon= log_descriptor.horizon; if (log_record_type_descriptor[type].inwrite_hook && - (*log_record_type_descriptor[type].inwrite_hook)(type, tcb, lsn, parts)) + (*log_record_type_descriptor[type].inwrite_hook)(type, trn, lsn, parts)) { translog_unlock(); DBUG_RETURN(1); @@ -3069,12 +3079,12 @@ translog_write_variable_record_1group(LSN *lsn, translog_write_variable_record_1chunk() lsn LSN of the record will be written here type the log record type - short_trid Sort transaction ID or 0 if it has no sense + short_trid Short transaction ID or 0 if it has no sense parts Descriptor of record source parts buffer_to_flush Buffer which have to be flushed if it is not 0 header_length Calculated header length of chunk type 0 - tcb Transaction control block pointer for hooks by - record log type + trn Transaction structure pointer for hooks by + record log type, for short_id RETURN 0 OK @@ -3088,7 +3098,7 @@ translog_write_variable_record_1chunk(LSN *lsn, struct st_translog_parts *parts, struct st_translog_buffer *buffer_to_flush, uint16 header_length, - void *tcb) + TRN *trn) { int rc; byte chunk0_header[1 + 2 + 5 + 2]; @@ -3099,7 +3109,7 @@ translog_write_variable_record_1chunk(LSN *lsn, *lsn= log_descriptor.horizon; if (log_record_type_descriptor[type].inwrite_hook && - (*log_record_type_descriptor[type].inwrite_hook)(type, tcb, + (*log_record_type_descriptor[type].inwrite_hook)(type, trn, lsn, parts)) { translog_unlock(); @@ -3399,13 +3409,13 @@ static my_bool translog_relative_LSN_encode(struct st_translog_parts *parts, translog_write_variable_record_mgroup() lsn LSN of the record will be written here type the log record type - short_trid Sort transaction ID or 0 if it has no sense + short_trid Short transaction ID or 0 if it has no sense parts Descriptor of record source parts buffer_to_flush Buffer which have to be flushed if it is not 0 header_length Header length calculated for 1 group buffer_rest Beginning from which we plan to write in full pages - tcb Transaction control block pointer for hooks by - record log type + trn Transaction structure pointer for hooks by + record log type, for short_id RETURN 0 OK @@ -3421,7 +3431,7 @@ translog_write_variable_record_mgroup(LSN *lsn, *buffer_to_flush, uint16 header_length, translog_size_t buffer_rest, - void *tcb) + TRN *trn) { TRANSLOG_ADDRESS horizon; struct st_buffer_cursor cursor; @@ -3757,7 +3767,7 @@ translog_write_variable_record_mgroup(LSN *lsn, first_chunk0= 0; *lsn= horizon; if (log_record_type_descriptor[type].inwrite_hook && - (*log_record_type_descriptor[type].inwrite_hook) (type, tcb, + (*log_record_type_descriptor[type].inwrite_hook) (type, trn, lsn, parts)) goto err; } @@ -3831,10 +3841,10 @@ err: translog_write_variable_record() lsn LSN of the record will be written here type the log record type - short_trid Sort transaction ID or 0 if it has no sense + short_trid Short transaction ID or 0 if it has no sense parts Descriptor of record source parts - tcb Transaction control block pointer for hooks by - record log type + trn Transaction structure pointer for hooks by + record log type, for short_id RETURN 0 OK @@ -3845,7 +3855,7 @@ static my_bool translog_write_variable_record(LSN *lsn, enum translog_record_type type, SHORT_TRANSACTION_ID short_trid, struct st_translog_parts *parts, - void *tcb) + TRN *trn) { struct st_translog_buffer *buffer_to_flush= NULL; uint header_length1= 1 + 2 + 2 + @@ -3920,7 +3930,7 @@ static my_bool translog_write_variable_record(LSN *lsn, /* following function makes translog_unlock(); */ DBUG_RETURN(translog_write_variable_record_1chunk(lsn, type, short_trid, parts, buffer_to_flush, - header_length1, tcb)); + header_length1, trn)); } buffer_rest= translog_get_current_group_size(); @@ -3930,13 +3940,13 @@ static my_bool translog_write_variable_record(LSN *lsn, /* following function makes translog_unlock(); */ DBUG_RETURN(translog_write_variable_record_1group(lsn, type, short_trid, parts, buffer_to_flush, - header_length1, tcb)); + header_length1, trn)); } /* following function makes translog_unlock(); */ DBUG_RETURN(translog_write_variable_record_mgroup(lsn, type, short_trid, parts, buffer_to_flush, header_length1, - buffer_rest, tcb)); + buffer_rest, trn)); } @@ -3947,10 +3957,10 @@ static my_bool translog_write_variable_record(LSN *lsn, translog_write_fixed_record() lsn LSN of the record will be written here type the log record type - short_trid Sort transaction ID or 0 if it has no sense + short_trid Short transaction ID or 0 if it has no sense parts Descriptor of record source parts - tcb Transaction control block pointer for hooks by - record log type + trn Transaction structure pointer for hooks by + record log type, for short_id RETURN 0 OK @@ -3961,7 +3971,7 @@ static my_bool translog_write_fixed_record(LSN *lsn, enum translog_record_type type, SHORT_TRANSACTION_ID short_trid, struct st_translog_parts *parts, - void *tcb) + TRN *trn) { struct st_translog_buffer *buffer_to_flush= NULL; byte chunk1_header[1 + 2]; @@ -4012,7 +4022,7 @@ static my_bool translog_write_fixed_record(LSN *lsn, *lsn= log_descriptor.horizon; if (log_record_type_descriptor[type].inwrite_hook && - (*log_record_type_descriptor[type].inwrite_hook) (type, tcb, + (*log_record_type_descriptor[type].inwrite_hook) (type, trn, lsn, parts)) { rc= 1; @@ -4074,9 +4084,9 @@ err: translog_write_record() lsn LSN of the record will be written here type the log record type - short_trid Sort transaction ID or 0 if it has no sense - tcb Transaction control block pointer for hooks by - record log type + trn Transaction structure pointer for hooks by + record log type, for short_id + share MARIA_SHARE of table or NULL rec_len record length or 0 (count it) part_no number of parts or 0 (count it) parts_data zero ended (in case of number of parts is 0) @@ -4091,8 +4101,7 @@ err: my_bool translog_write_record(LSN *lsn, enum translog_record_type type, - SHORT_TRANSACTION_ID short_trid, - void *tcb, struct st_maria_share *share, + TRN *trn, struct st_maria_share *share, translog_size_t rec_len, uint part_no, LEX_STRING *parts_data) @@ -4100,6 +4109,7 @@ my_bool translog_write_record(LSN *lsn, struct st_translog_parts parts; LEX_STRING *part; int rc; + uint short_trid= trn->short_id; DBUG_ENTER("translog_write_record"); DBUG_PRINT("enter", ("type: %u ShortTrID: %u", (uint) type, (uint)short_trid)); @@ -4169,17 +4179,17 @@ my_bool translog_write_record(LSN *lsn, /* process this parts */ if (!(rc= (log_record_type_descriptor[type].prewrite_hook && - (*log_record_type_descriptor[type].prewrite_hook) (type, tcb, + (*log_record_type_descriptor[type].prewrite_hook) (type, trn, share, &parts)))) { switch (log_record_type_descriptor[type].class) { case LOGRECTYPE_VARIABLE_LENGTH: - rc= translog_write_variable_record(lsn, type, short_trid, &parts, tcb); + rc= translog_write_variable_record(lsn, type, short_trid, &parts, trn); break; case LOGRECTYPE_PSEUDOFIXEDLENGTH: case LOGRECTYPE_FIXEDLENGTH: - rc= translog_write_fixed_record(lsn, type, short_trid, &parts, tcb); + rc= translog_write_fixed_record(lsn, type, short_trid, &parts, trn); break; case LOGRECTYPE_NOT_ALLOWED: default: @@ -4745,7 +4755,7 @@ translog_read_record_header_from_buffer(byte *page, TRANSLOG_CHUNK_FIXED); buff->type= (page[page_offset] & TRANSLOG_REC_TYPE); buff->short_trid= uint2korr(page + page_offset + 1); - DBUG_PRINT("info", ("Type %u, Sort TrID %u, LSN (%lu,0x%lx)", + DBUG_PRINT("info", ("Type %u, Short TrID %u, LSN (%lu,0x%lx)", (uint) buff->type, (uint)buff->short_trid, (ulong) LSN_FILE_NO(buff->lsn), (ulong) LSN_OFFSET(buff->lsn))); @@ -5386,3 +5396,32 @@ my_bool translog_flush(LSN lsn) translog_unlock(); DBUG_RETURN(rc); } + + +static my_bool write_hook_for_redo(enum translog_record_type type + __attribute__ ((unused)), + TRN *trn, LSN *lsn, + struct st_translog_parts *parts + __attribute__ ((unused))) +{ + if (trn->rec_lsn == 0) + trn->rec_lsn= *lsn; + return 0; +} + + +static my_bool write_hook_for_undo(enum translog_record_type type + __attribute__ ((unused)), + TRN *trn, LSN *lsn, + struct st_translog_parts *parts + __attribute__ ((unused))) +{ + trn->undo_lsn= *lsn; + if (trn->first_undo_lsn == 0) + trn->first_undo_lsn= *lsn; + return 0; + /* + when we implement purging, we will specialize this hook: UNDO_PURGE + records will additionally set trn->undo_purge_lsn + */ +} diff --git a/storage/maria/ma_loghandler.h b/storage/maria/ma_loghandler.h index 3ccb3bf9af2..bad9531df70 100644 --- a/storage/maria/ma_loghandler.h +++ b/storage/maria/ma_loghandler.h @@ -180,6 +180,7 @@ struct st_translog_reader_data my_bool eor; /* end of the record */ }; +struct st_transaction; #ifdef __cplusplus extern "C" { #endif @@ -191,8 +192,8 @@ extern my_bool translog_init(const char *directory, uint32 log_file_max_size, extern my_bool translog_write_record(LSN *lsn, enum translog_record_type type, - SHORT_TRANSACTION_ID short_trid, - void *tcb, struct st_maria_share *share, + struct st_transaction *trn, + struct st_maria_share *share, translog_size_t rec_len, uint part_no, LEX_STRING *parts_data); diff --git a/storage/maria/ma_pagecache.c b/storage/maria/ma_pagecache.c index 53a24e36861..18c36fcfbd1 100755 --- a/storage/maria/ma_pagecache.c +++ b/storage/maria/ma_pagecache.c @@ -41,7 +41,7 @@ #include "maria_def.h" #include -#include +#include "ma_pagecache.h" #include #include #include diff --git a/storage/maria/ma_pagecache.h b/storage/maria/ma_pagecache.h new file mode 100644 index 00000000000..ef14cd48cef --- /dev/null +++ b/storage/maria/ma_pagecache.h @@ -0,0 +1,259 @@ +/* Copyright (C) 2006 MySQL AB + + 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; either version 2 of the License, or + (at your option) any later version. + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ + +/* Page cache variable structures */ + +#ifndef _ma_pagecache_h +#define _ma_pagecache_h +C_MODE_START + +#include "ma_loghandler_lsn.h" +#include + +/* Type of the page */ +enum pagecache_page_type +{ + /* + Used only for control page type changing during debugging. This define + should only be using when using DBUG. + */ + PAGECACHE_EMPTY_PAGE, + /* the page does not contain LSN */ + PAGECACHE_PLAIN_PAGE, + /* the page contain LSN (maria tablespace page) */ + PAGECACHE_LSN_PAGE +}; + +/* + This enum describe lock status changing. every type of page cache will + interpret WRITE/READ lock as it need. +*/ +enum pagecache_page_lock +{ + PAGECACHE_LOCK_LEFT_UNLOCKED, /* free -> free */ + PAGECACHE_LOCK_LEFT_READLOCKED, /* read -> read */ + PAGECACHE_LOCK_LEFT_WRITELOCKED, /* write -> write */ + PAGECACHE_LOCK_READ, /* free -> read */ + PAGECACHE_LOCK_WRITE, /* free -> write */ + PAGECACHE_LOCK_READ_UNLOCK, /* read -> free */ + PAGECACHE_LOCK_WRITE_UNLOCK, /* write -> free */ + PAGECACHE_LOCK_WRITE_TO_READ /* write -> read */ +}; +/* + This enum describe pin status changing +*/ +enum pagecache_page_pin +{ + PAGECACHE_PIN_LEFT_PINNED, /* pinned -> pinned */ + PAGECACHE_PIN_LEFT_UNPINNED, /* unpinned -> unpinned */ + PAGECACHE_PIN, /* unpinned -> pinned */ + PAGECACHE_UNPIN /* pinned -> unpinned */ +}; +/* How to write the page */ +enum pagecache_write_mode +{ + /* do not write immediately, i.e. it will be dirty page */ + PAGECACHE_WRITE_DELAY, + /* page already is in the file. (key cache insert analogue) */ + PAGECACHE_WRITE_DONE +}; + +typedef void *PAGECACHE_PAGE_LINK; + +/* file descriptor for Maria */ +typedef struct st_pagecache_file +{ + File file; +} PAGECACHE_FILE; + +/* page number for maria */ +typedef uint32 pgcache_page_no_t; + +/* declare structures that is used by st_pagecache */ + +struct st_pagecache_block_link; +typedef struct st_pagecache_block_link PAGECACHE_BLOCK_LINK; +struct st_pagecache_page; +typedef struct st_pagecache_page PAGECACHE_PAGE; +struct st_pagecache_hash_link; +typedef struct st_pagecache_hash_link PAGECACHE_HASH_LINK; + +#include + +typedef my_bool (*pagecache_disk_read_validator)(byte *page, gptr data); + +#define PAGECACHE_CHANGED_BLOCKS_HASH 128 /* must be power of 2 */ + +/* + The page cache structure + It also contains read-only statistics parameters. +*/ + +typedef struct st_pagecache +{ + my_bool inited; + my_bool resize_in_flush; /* true during flush of resize operation */ + my_bool can_be_used; /* usage of cache for read/write is allowed */ + uint shift; /* block size = 2 ^ shift */ + my_size_t mem_size; /* specified size of the cache memory */ + uint32 block_size; /* size of the page buffer of a cache block */ + ulong min_warm_blocks; /* min number of warm blocks; */ + ulong age_threshold; /* age threshold for hot blocks */ + ulonglong time; /* total number of block link operations */ + uint hash_entries; /* max number of entries in the hash table */ + int hash_links; /* max number of hash links */ + int hash_links_used; /* number of hash links taken from free links pool */ + int disk_blocks; /* max number of blocks in the cache */ + ulong blocks_used; /* maximum number of concurrently used blocks */ + ulong blocks_unused; /* number of currently unused blocks */ + ulong blocks_changed; /* number of currently dirty blocks */ + ulong warm_blocks; /* number of blocks in warm sub-chain */ + ulong cnt_for_resize_op; /* counter to block resize operation */ + ulong blocks_available; /* number of blocks available in the LRU chain */ + PAGECACHE_HASH_LINK **hash_root;/* arr. of entries into hash table buckets */ + PAGECACHE_HASH_LINK *hash_link_root;/* memory for hash table links */ + PAGECACHE_HASH_LINK *free_hash_list;/* list of free hash links */ + PAGECACHE_BLOCK_LINK *free_block_list;/* list of free blocks */ + PAGECACHE_BLOCK_LINK *block_root;/* memory for block links */ + byte HUGE_PTR *block_mem; /* memory for block buffers */ + PAGECACHE_BLOCK_LINK *used_last;/* ptr to the last block of the LRU chain */ + PAGECACHE_BLOCK_LINK *used_ins;/* ptr to the insertion block in LRU chain */ + pthread_mutex_t cache_lock; /* to lock access to the cache structure */ + WQUEUE resize_queue; /* threads waiting during resize operation */ + WQUEUE waiting_for_hash_link;/* waiting for a free hash link */ + WQUEUE waiting_for_block; /* requests waiting for a free block */ + /* hash for dirty file bl.*/ + PAGECACHE_BLOCK_LINK *changed_blocks[PAGECACHE_CHANGED_BLOCKS_HASH]; + /* hash for other file bl.*/ + PAGECACHE_BLOCK_LINK *file_blocks[PAGECACHE_CHANGED_BLOCKS_HASH]; + + /* + The following variables are and variables used to hold parameters for + initializing the key cache. + */ + + ulonglong param_buff_size; /* size the memory allocated for the cache */ + ulong param_block_size; /* size of the blocks in the key cache */ + ulong param_division_limit; /* min. percentage of warm blocks */ + ulong param_age_threshold; /* determines when hot block is downgraded */ + + /* Statistics variables. These are reset in reset_pagecache_counters(). */ + ulong global_blocks_changed; /* number of currently dirty blocks */ + ulonglong global_cache_w_requests;/* number of write requests (write hits) */ + ulonglong global_cache_write; /* number of writes from cache to files */ + ulonglong global_cache_r_requests;/* number of read requests (read hits) */ + ulonglong global_cache_read; /* number of reads from files to cache */ + + int blocks; /* max number of blocks in the cache */ + my_bool in_init; /* Set to 1 in MySQL during init/resize */ +} PAGECACHE; + +/* The default key cache */ +extern PAGECACHE dflt_pagecache_var, *dflt_pagecache; + +extern int init_pagecache(PAGECACHE *pagecache, my_size_t use_mem, + uint division_limit, uint age_threshold, + uint block_size); +extern int resize_pagecache(PAGECACHE *pagecache, + my_size_t use_mem, uint division_limit, + uint age_threshold); +extern void change_pagecache_param(PAGECACHE *pagecache, uint division_limit, + uint age_threshold); + +#define pagecache_read(P,F,N,L,B,T,K,I) \ + pagecache_valid_read(P,F,N,L,B,T,K,I,0,0) + +extern byte *pagecache_valid_read(PAGECACHE *pagecache, + PAGECACHE_FILE *file, + pgcache_page_no_t pageno, + uint level, + byte *buff, + enum pagecache_page_type type, + enum pagecache_page_lock lock, + PAGECACHE_PAGE_LINK *link, + pagecache_disk_read_validator validator, + gptr validator_data); + +#define pagecache_write(P,F,N,L,B,T,O,I,M,K) \ + pagecache_write_part(P,F,N,L,B,T,O,I,M,K,0,(P)->block_size) + +extern my_bool pagecache_write_part(PAGECACHE *pagecache, + PAGECACHE_FILE *file, + pgcache_page_no_t pageno, + uint level, + byte *buff, + enum pagecache_page_type type, + enum pagecache_page_lock lock, + enum pagecache_page_pin pin, + enum pagecache_write_mode write_mode, + PAGECACHE_PAGE_LINK *link, + uint offset, + uint size); +extern void pagecache_unlock(PAGECACHE *pagecache, + PAGECACHE_FILE *file, + pgcache_page_no_t pageno, + enum pagecache_page_lock lock, + enum pagecache_page_pin pin, + LSN first_REDO_LSN_for_page, + LSN lsn); +extern void pagecache_unlock_by_link(PAGECACHE *pagecache, + PAGECACHE_PAGE_LINK *link, + enum pagecache_page_lock lock, + enum pagecache_page_pin pin, + LSN first_REDO_LSN_for_page, + LSN lsn); +extern void pagecache_unpin(PAGECACHE *pagecache, + PAGECACHE_FILE *file, + pgcache_page_no_t pageno, + LSN lsn); +extern void pagecache_unpin_by_link(PAGECACHE *pagecache, + PAGECACHE_PAGE_LINK *link, + LSN lsn); +extern int flush_pagecache_blocks(PAGECACHE *keycache, + PAGECACHE_FILE *file, + enum flush_type type); +extern my_bool pagecache_delete(PAGECACHE *pagecache, + PAGECACHE_FILE *file, + pgcache_page_no_t pageno, + enum pagecache_page_lock lock, + my_bool flush); +extern my_bool pagecache_delete_pages(PAGECACHE *pagecache, + PAGECACHE_FILE *file, + pgcache_page_no_t pageno, + uint page_count, + enum pagecache_page_lock lock, + my_bool flush); +extern void end_pagecache(PAGECACHE *keycache, my_bool cleanup); +extern my_bool pagecache_collect_changed_blocks_with_lsn(PAGECACHE *pagecache, + LEX_STRING *str, + LSN *max_lsn); +extern int reset_pagecache_counters(const char *name, PAGECACHE *pagecache); + + +/* Functions to handle multiple key caches */ +extern my_bool multi_pagecache_init(void); +extern void multi_pagecache_free(void); +extern PAGECACHE *multi_pagecache_search(byte *key, uint length, + PAGECACHE *def); +extern my_bool multi_pagecache_set(const byte *key, uint length, + PAGECACHE *pagecache); +extern void multi_pagecache_change(PAGECACHE *old_data, + PAGECACHE *new_data); +extern int reset_pagecache_counters(const char *name, + PAGECACHE *pagecache); + +C_MODE_END +#endif /* _keycache_h */ diff --git a/storage/maria/ma_pagecaches.c b/storage/maria/ma_pagecaches.c index 1a120131016..d2ed4edca31 100644 --- a/storage/maria/ma_pagecaches.c +++ b/storage/maria/ma_pagecaches.c @@ -23,7 +23,7 @@ */ #include "maria_def.h" -#include +#include "ma_pagecache.h" #include #include #include "../../mysys/my_safehash.h" diff --git a/storage/maria/trnman.c b/storage/maria/trnman.c index 702b3b20f6c..d6b35f071ea 100644 --- a/storage/maria/trnman.c +++ b/storage/maria/trnman.c @@ -298,7 +298,7 @@ TRN *trnman_new_trn(pthread_mutex_t *mutex, pthread_cond_t *cond, trn->min_read_from= trn->trid; trn->commit_trid= 0; - trn->undo_lsn= 0; + trn->rec_lsn= trn->undo_lsn= trn->first_undo_lsn= 0; trn->locks.mutex= mutex; trn->locks.cond= cond; diff --git a/storage/maria/trnman.h b/storage/maria/trnman.h index cfdd214dda7..1e1550efb46 100644 --- a/storage/maria/trnman.h +++ b/storage/maria/trnman.h @@ -45,7 +45,7 @@ struct st_transaction LF_PINS *pins; TrID trid, min_read_from, commit_trid; TRN *next, *prev; - LSN rec_lsn, undo_lsn; + LSN rec_lsn, undo_lsn, first_undo_lsn; uint locked_tables; /* Note! if locks.loid is 0, trn is NOT initialized */ }; diff --git a/storage/maria/unittest/ma_test_loghandler-t.c b/storage/maria/unittest/ma_test_loghandler-t.c index bff5864a5c0..370a7082a8c 100644 --- a/storage/maria/unittest/ma_test_loghandler-t.c +++ b/storage/maria/unittest/ma_test_loghandler-t.c @@ -2,12 +2,14 @@ #include #include #include +#include "../trnman.h" extern my_bool maria_log_remove(); #ifndef DBUG_OFF static const char *default_dbug_option; #endif +static TRN *trn= &dummy_transaction_object; #define PCACHE_SIZE (1024*1024*10) @@ -166,9 +168,10 @@ int main(int argc __attribute__((unused)), char *argv[]) int4store(long_tr_id, 0); parts[TRANSLOG_INTERNAL_PARTS + 0].str= (char*)long_tr_id; parts[TRANSLOG_INTERNAL_PARTS + 0].length= 6; + trn->short_id= 0; if (translog_write_record(&lsn, LOGREC_LONG_TRANSACTION_ID, - 0, NULL, NULL, + trn, NULL, 6, TRANSLOG_INTERNAL_PARTS + 1, parts)) { fprintf(stderr, "Can't write record #%lu\n", (ulong) 0); @@ -181,6 +184,7 @@ int main(int argc __attribute__((unused)), char *argv[]) for (i= 1; i < ITERATIONS; i++) { + trn->short_id= i % 0xFFFF; if (i % 2) { lsn_store(lsn_buff, lsn_base); @@ -189,7 +193,7 @@ int main(int argc __attribute__((unused)), char *argv[]) /* check auto-count feature */ parts[TRANSLOG_INTERNAL_PARTS + 1].str= NULL; parts[TRANSLOG_INTERNAL_PARTS + 1].length= 0; - if (translog_write_record(&lsn, LOGREC_CLR_END, (i % 0xFFFF), NULL, + if (translog_write_record(&lsn, LOGREC_CLR_END, trn, NULL, LSN_STORE_SIZE, 0, parts)) { fprintf(stderr, "1 Can't write reference defore record #%lu\n", @@ -209,8 +213,7 @@ int main(int argc __attribute__((unused)), char *argv[]) /* check record length auto-counting */ if (translog_write_record(&lsn, LOGREC_UNDO_KEY_INSERT, - (i % 0xFFFF), - NULL, NULL, 0, TRANSLOG_INTERNAL_PARTS + 2, + trn, NULL, 0, TRANSLOG_INTERNAL_PARTS + 2, parts)) { fprintf(stderr, "1 Can't write var reference defore record #%lu\n", @@ -229,7 +232,7 @@ int main(int argc __attribute__((unused)), char *argv[]) parts[TRANSLOG_INTERNAL_PARTS + 0].length= 23; if (translog_write_record(&lsn, LOGREC_UNDO_ROW_DELETE, - (i % 0xFFFF), NULL, NULL, + trn, NULL, 23, TRANSLOG_INTERNAL_PARTS + 1, parts)) { fprintf(stderr, "0 Can't write reference defore record #%lu\n", @@ -249,8 +252,7 @@ int main(int argc __attribute__((unused)), char *argv[]) parts[TRANSLOG_INTERNAL_PARTS + 1].length= rec_len; if (translog_write_record(&lsn, LOGREC_UNDO_KEY_DELETE, - (i % 0xFFFF), - NULL, NULL, 14 + rec_len, + trn, NULL, 14 + rec_len, TRANSLOG_INTERNAL_PARTS + 2, parts)) { fprintf(stderr, "0 Can't write var reference defore record #%lu\n", @@ -266,7 +268,7 @@ int main(int argc __attribute__((unused)), char *argv[]) parts[TRANSLOG_INTERNAL_PARTS + 0].length= 6; if (translog_write_record(&lsn, LOGREC_LONG_TRANSACTION_ID, - (i % 0xFFFF), NULL, NULL, 6, + trn, NULL, 6, TRANSLOG_INTERNAL_PARTS + 1, parts)) { @@ -285,7 +287,7 @@ int main(int argc __attribute__((unused)), char *argv[]) parts[TRANSLOG_INTERNAL_PARTS + 0].length= rec_len; if (translog_write_record(&lsn, LOGREC_REDO_INSERT_ROW_HEAD, - (i % 0xFFFF), NULL, NULL, rec_len, + trn, NULL, rec_len, TRANSLOG_INTERNAL_PARTS + 1, parts)) { diff --git a/storage/maria/unittest/ma_test_loghandler_multigroup-t.c b/storage/maria/unittest/ma_test_loghandler_multigroup-t.c index 110c35b786a..6d35444c656 100644 --- a/storage/maria/unittest/ma_test_loghandler_multigroup-t.c +++ b/storage/maria/unittest/ma_test_loghandler_multigroup-t.c @@ -2,12 +2,14 @@ #include #include #include +#include "../trnman.h" extern my_bool maria_log_remove(); #ifndef DBUG_OFF static const char *default_dbug_option; #endif +static TRN *trn= &dummy_transaction_object; #define PCACHE_SIZE (1024*1024*10) @@ -186,7 +188,8 @@ int main(int argc __attribute__((unused)), char *argv[]) int4store(long_tr_id, 0); parts[TRANSLOG_INTERNAL_PARTS + 0].str= (char*)long_tr_id; parts[TRANSLOG_INTERNAL_PARTS + 0].length= 6; - if (translog_write_record(&lsn, LOGREC_LONG_TRANSACTION_ID, 0, NULL, NULL, + trn->short_id= 0; + if (translog_write_record(&lsn, LOGREC_LONG_TRANSACTION_ID, trn, NULL, 6, TRANSLOG_INTERNAL_PARTS + 1, parts)) { fprintf(stderr, "Can't write record #%lu\n", (ulong) 0); @@ -204,9 +207,10 @@ int main(int argc __attribute__((unused)), char *argv[]) lsn_store(lsn_buff, lsn_base); parts[TRANSLOG_INTERNAL_PARTS + 0].str= (char*)lsn_buff; parts[TRANSLOG_INTERNAL_PARTS + 0].length= LSN_STORE_SIZE; + trn->short_id= i % 0xFFFF; if (translog_write_record(&lsn, LOGREC_CLR_END, - (i % 0xFFFF), NULL, NULL, + trn, NULL, LSN_STORE_SIZE, TRANSLOG_INTERNAL_PARTS + 1, parts)) { @@ -223,10 +227,10 @@ int main(int argc __attribute__((unused)), char *argv[]) parts[TRANSLOG_INTERNAL_PARTS + 0].length= LSN_STORE_SIZE; parts[TRANSLOG_INTERNAL_PARTS + 1].str= (char*)long_buffer; parts[TRANSLOG_INTERNAL_PARTS + 1].length= rec_len; + trn->short_id= i % 0xFFFF; if (translog_write_record(&lsn, LOGREC_UNDO_KEY_INSERT, - (i % 0xFFFF), - NULL, NULL, LSN_STORE_SIZE + rec_len, + trn, NULL, LSN_STORE_SIZE + rec_len, TRANSLOG_INTERNAL_PARTS + 2, parts)) { @@ -244,9 +248,10 @@ int main(int argc __attribute__((unused)), char *argv[]) lsn_store(lsn_buff + LSN_STORE_SIZE, first_lsn); parts[TRANSLOG_INTERNAL_PARTS + 1].str= (char*)lsn_buff; parts[TRANSLOG_INTERNAL_PARTS + 1].length= 23; + trn->short_id= i % 0xFFFF; if (translog_write_record(&lsn, LOGREC_UNDO_ROW_DELETE, - (i % 0xFFFF), NULL, NULL, 23, + trn, NULL, 23, TRANSLOG_INTERNAL_PARTS + 1, parts)) { @@ -264,10 +269,10 @@ int main(int argc __attribute__((unused)), char *argv[]) parts[TRANSLOG_INTERNAL_PARTS + 0].length= LSN_STORE_SIZE * 2; parts[TRANSLOG_INTERNAL_PARTS + 1].str= (char*)long_buffer; parts[TRANSLOG_INTERNAL_PARTS + 1].length= rec_len; + trn->short_id= i % 0xFFFF; if (translog_write_record(&lsn, LOGREC_UNDO_KEY_DELETE, - (i % 0xFFFF), - NULL, NULL, LSN_STORE_SIZE * 2 + rec_len, + trn, NULL, LSN_STORE_SIZE * 2 + rec_len, TRANSLOG_INTERNAL_PARTS + 2, parts)) { @@ -282,9 +287,10 @@ int main(int argc __attribute__((unused)), char *argv[]) int4store(long_tr_id, i); parts[TRANSLOG_INTERNAL_PARTS + 0].str= (char*)long_tr_id; parts[TRANSLOG_INTERNAL_PARTS + 0].length= 6; + trn->short_id= i % 0xFFFF; if (translog_write_record(&lsn, LOGREC_LONG_TRANSACTION_ID, - (i % 0xFFFF), NULL, NULL, 6, + trn, NULL, 6, TRANSLOG_INTERNAL_PARTS + 1, parts)) { fprintf(stderr, "Can't write record #%lu\n", (ulong) i); @@ -299,9 +305,10 @@ int main(int argc __attribute__((unused)), char *argv[]) rec_len= get_len(); parts[TRANSLOG_INTERNAL_PARTS + 0].str= (char*)long_buffer; parts[TRANSLOG_INTERNAL_PARTS + 0].length= rec_len; + trn->short_id= i % 0xFFFF; if (translog_write_record(&lsn, LOGREC_REDO_INSERT_ROW_HEAD, - (i % 0xFFFF), NULL, NULL, rec_len, + trn, NULL, rec_len, TRANSLOG_INTERNAL_PARTS + 1, parts)) { fprintf(stderr, "Can't write variable record #%lu\n", (ulong) i); diff --git a/storage/maria/unittest/ma_test_loghandler_multithread-t.c b/storage/maria/unittest/ma_test_loghandler_multithread-t.c index 4afd2b23074..8b7a9a5fa83 100644 --- a/storage/maria/unittest/ma_test_loghandler_multithread-t.c +++ b/storage/maria/unittest/ma_test_loghandler_multithread-t.c @@ -2,6 +2,8 @@ #include #include #include +#include "../trnman.h" + extern my_bool maria_log_remove(); #ifndef DBUG_OFF @@ -117,9 +119,11 @@ static my_bool read_and_check_content(TRANSLOG_HEADER_BUFFER *rec, void writer(int num) { LSN lsn; + TRN trn; byte long_tr_id[6]; uint i; + trn.short_id= num; for (i= 0; i < ITERATIONS; i++) { uint len= get_len(); @@ -132,7 +136,7 @@ void writer(int num) parts[TRANSLOG_INTERNAL_PARTS + 0].length= 6; if (translog_write_record(&lsn, LOGREC_LONG_TRANSACTION_ID, - num, NULL, NULL, 6, TRANSLOG_INTERNAL_PARTS + 1, + &trn, NULL, 6, TRANSLOG_INTERNAL_PARTS + 1, parts)) { fprintf(stderr, "Can't write LOGREC_LONG_TRANSACTION_ID record #%lu " @@ -148,7 +152,7 @@ void writer(int num) parts[TRANSLOG_INTERNAL_PARTS + 0].length= len; if (translog_write_record(&lsn, LOGREC_REDO_INSERT_ROW_HEAD, - num, NULL, NULL, + &trn, NULL, len, TRANSLOG_INTERNAL_PARTS + 1, parts)) { @@ -296,7 +300,8 @@ int main(int argc __attribute__((unused)), parts[TRANSLOG_INTERNAL_PARTS + 0].length= 6; if (translog_write_record(&first_lsn, LOGREC_LONG_TRANSACTION_ID, - 0, NULL, NULL, 6, TRANSLOG_INTERNAL_PARTS + 1, + &dummy_transaction_object, NULL, 6, + TRANSLOG_INTERNAL_PARTS + 1, parts)) { fprintf(stderr, "Can't write the first record\n"); diff --git a/storage/maria/unittest/ma_test_loghandler_pagecache-t.c b/storage/maria/unittest/ma_test_loghandler_pagecache-t.c index a56f3f875c6..e59f910f549 100644 --- a/storage/maria/unittest/ma_test_loghandler_pagecache-t.c +++ b/storage/maria/unittest/ma_test_loghandler_pagecache-t.c @@ -2,6 +2,7 @@ #include #include #include +#include "../trnman.h" extern my_bool maria_log_remove(); @@ -90,7 +91,8 @@ int main(int argc __attribute__((unused)), char *argv[]) parts[TRANSLOG_INTERNAL_PARTS + 0].length= 6; if (translog_write_record(&lsn, LOGREC_LONG_TRANSACTION_ID, - 0, NULL, NULL, 6, TRANSLOG_INTERNAL_PARTS + 1, + &dummy_transaction_object, NULL, 6, + TRANSLOG_INTERNAL_PARTS + 1, parts)) { fprintf(stderr, "Can't write record #%lu\n", (ulong) 0); diff --git a/storage/maria/unittest/test_file.h b/storage/maria/unittest/test_file.h index bfc660b13d0..293c692717e 100644 --- a/storage/maria/unittest/test_file.h +++ b/storage/maria/unittest/test_file.h @@ -1,5 +1,5 @@ #include -#include +#include "../ma_pagecache.h" /* File content descriptor -- cgit v1.2.1 From c548715c46a351a8c859da873db488c8f1b9c224 Mon Sep 17 00:00:00 2001 From: unknown Date: Fri, 15 Jun 2007 01:45:55 +0300 Subject: Fixed LSN codding to allow code all relative LSN independed on "distance". Added support for test record descriptors to avoid interfere woth real record descriptors. Fixed descriptor of pseudofixed length record length, now it is length of record passed from/to client of the loghandler. BitKeeper/etc/ignore: Added storage/maria/unittest/ma_test_loghandler_long-t-big to the ignore list storage/maria/ma_init.c: Removed loghandler_init call because it is present in translog_init() storage/maria/ma_loghandler.c: Fixed LSN codding to allow code all relative LSN independed on "distance". Added support for test record descriptors to avoid interfere woth real record descriptors. Fixed length of LOGREC_REDO_INSERT_ROW_TAIL. Fixed descriptor of pseudofixed length record length, now it is length of record passed from/to client of the loghandler. storage/maria/ma_loghandler.h: Added support for test record descriptors to avoid interfere woth real record descriptors. storage/maria/unittest/Makefile.am: Made new test for log with reference over 63 files. Layout fixed. storage/maria/unittest/ma_test_loghandler-t.c: Added support for test record descriptors to avoid interfere woth real record descriptors. storage/maria/unittest/ma_test_loghandler_multigroup-t.c: Added support for test record descriptors to avoid interfere woth real record descriptors. storage/maria/unittest/ma_test_loghandler_multithread-t.c: Added support for test record descriptors to avoid interfere woth real record descriptors. storage/maria/unittest/ma_test_loghandler_pagecache-t.c: Added support for test record descriptors to avoid interfere woth real record descriptors. --- .bzrignore | 1 + storage/maria/ma_init.c | 1 - storage/maria/ma_loghandler.c | 172 +++++++++++++++------ storage/maria/ma_loghandler.h | 15 +- storage/maria/unittest/Makefile.am | 13 +- storage/maria/unittest/ma_test_loghandler-t.c | 132 ++++++++++------ .../unittest/ma_test_loghandler_multigroup-t.c | 95 +++++++----- .../unittest/ma_test_loghandler_multithread-t.c | 28 ++-- .../unittest/ma_test_loghandler_pagecache-t.c | 3 +- 9 files changed, 300 insertions(+), 160 deletions(-) diff --git a/.bzrignore b/.bzrignore index 40623790bc6..e4b60deec56 100644 --- a/.bzrignore +++ b/.bzrignore @@ -3000,3 +3000,4 @@ storage/maria/unittest/ma_pagecache_consist_64kWR-t-big storage/maria/unittest/ma_pagecache_single_64k-t-big storage/maria/maria_control storage/maria/maria_log.* +storage/maria/unittest/ma_test_loghandler_long-t-big diff --git a/storage/maria/ma_init.c b/storage/maria/ma_init.c index 19b835a837f..ac4826a721d 100644 --- a/storage/maria/ma_init.c +++ b/storage/maria/ma_init.c @@ -44,7 +44,6 @@ int maria_init(void) maria_inited= TRUE; pthread_mutex_init(&THR_LOCK_maria,MY_MUTEX_INIT_SLOW); _ma_init_block_record_data(); - loghandler_init(); } return 0; } diff --git a/storage/maria/ma_loghandler.c b/storage/maria/ma_loghandler.c index 43c5ea666ec..a74927b25fa 100644 --- a/storage/maria/ma_loghandler.c +++ b/storage/maria/ma_loghandler.c @@ -48,7 +48,9 @@ putchar('\n'); \ } while(0); - +/* Maximum length of compressed LSNs (the worst case of whole LSN storing) */ +#define COMPRESSED_LSN_MAX_STORE_SIZE (2 + LSN_STORE_SIZE) +#define MAX_NUMBER_OF_LSNS_PER_RECORD 2 /* record parts descriptor */ struct st_translog_parts @@ -184,7 +186,6 @@ enum record_class /* compressed (relative) LSN constants */ #define TRANSLOG_CLSN_LEN_BITS 0xC0 /* Mask to get compressed LSN length */ -#define TRANSLOG_CLSN_MAX_LEN 5 /* Maximum length of compressed LSN */ typedef my_bool(*prewrite_rec_hook) (enum translog_record_type type, void *tcb, struct st_maria_share *share, @@ -207,7 +208,10 @@ struct st_log_record_type_descriptor { /* internal class of the record */ enum record_class class; - /* length for fixed-size record, or maximum length of pseudo-fixed */ + /* + length for fixed-size record, pseudo-fixed record + length with uncompressed LSNs + */ uint16 fixed_length; /* how much record body (belonged to headers too) read with headers */ uint16 read_header_len; @@ -230,20 +234,55 @@ struct st_log_record_type_descriptor NOTE that after first public Maria release, these can NOT be changed */ - typedef struct st_log_record_type_descriptor LOG_DESC; static LOG_DESC log_record_type_descriptor[LOGREC_NUMBER_OF_TYPES]; +static LOG_DESC INIT_LOGREC_FIXED_RECORD_0LSN_EXAMPLE= +{LOGRECTYPE_FIXEDLENGTH, 6, 6, NULL, NULL, NULL, 0}; + +static LOG_DESC INIT_LOGREC_VARIABLE_RECORD_0LSN_EXAMPLE= +{LOGRECTYPE_VARIABLE_LENGTH, 0, 9, NULL, NULL, NULL, 0}; + +static LOG_DESC INIT_LOGREC_FIXED_RECORD_1LSN_EXAMPLE= +{LOGRECTYPE_PSEUDOFIXEDLENGTH, 7, 7, NULL, NULL, NULL, 1}; + +static LOG_DESC INIT_LOGREC_VARIABLE_RECORD_1LSN_EXAMPLE= +{LOGRECTYPE_VARIABLE_LENGTH, 0, 12, NULL, NULL, NULL, 1}; + +static LOG_DESC INIT_LOGREC_FIXED_RECORD_2LSN_EXAMPLE= +{LOGRECTYPE_PSEUDOFIXEDLENGTH, 23, 23, NULL, NULL, NULL, 2}; + +static LOG_DESC INIT_LOGREC_VARIABLE_RECORD_2LSN_EXAMPLE= +{LOGRECTYPE_VARIABLE_LENGTH, 0, 19, NULL, NULL, NULL, 2}; + + +void example_loghandler_init() +{ + log_record_type_descriptor[LOGREC_FIXED_RECORD_0LSN_EXAMPLE]= + INIT_LOGREC_FIXED_RECORD_0LSN_EXAMPLE; + log_record_type_descriptor[LOGREC_VARIABLE_RECORD_0LSN_EXAMPLE]= + INIT_LOGREC_VARIABLE_RECORD_0LSN_EXAMPLE; + log_record_type_descriptor[LOGREC_FIXED_RECORD_1LSN_EXAMPLE]= + INIT_LOGREC_FIXED_RECORD_1LSN_EXAMPLE; + log_record_type_descriptor[LOGREC_VARIABLE_RECORD_1LSN_EXAMPLE]= + INIT_LOGREC_VARIABLE_RECORD_1LSN_EXAMPLE; + log_record_type_descriptor[LOGREC_FIXED_RECORD_2LSN_EXAMPLE]= + INIT_LOGREC_FIXED_RECORD_2LSN_EXAMPLE; + log_record_type_descriptor[LOGREC_VARIABLE_RECORD_2LSN_EXAMPLE]= + INIT_LOGREC_VARIABLE_RECORD_2LSN_EXAMPLE; +} + + static LOG_DESC INIT_LOGREC_RESERVED_FOR_CHUNKS23= -{ LOGRECTYPE_NOT_ALLOWED, 0, 0, NULL, NULL, NULL, 0 }; +{LOGRECTYPE_NOT_ALLOWED, 0, 0, NULL, NULL, NULL, 0 }; static LOG_DESC INIT_LOGREC_REDO_INSERT_ROW_HEAD= {LOGRECTYPE_VARIABLE_LENGTH, 0, FILEID_STORE_SIZE + PAGE_STORE_SIZE + DIRPOS_STORE_SIZE, NULL, NULL, NULL, 0}; static LOG_DESC INIT_LOGREC_REDO_INSERT_ROW_TAIL= -{LOGRECTYPE_VARIABLE_LENGTH, 0, 9, NULL, NULL, NULL, 0}; +{LOGRECTYPE_VARIABLE_LENGTH, 0, 8, NULL, NULL, NULL, 0}; static LOG_DESC INIT_LOGREC_REDO_INSERT_ROW_BLOB= {LOGRECTYPE_VARIABLE_LENGTH, 0, 8, NULL, NULL, NULL, 0}; @@ -356,7 +395,7 @@ static LOG_DESC INIT_LOGREC_LONG_TRANSACTION_ID= {LOGRECTYPE_FIXEDLENGTH, 6, 6, NULL, NULL, NULL, 0}; -void loghandler_init() +static void loghandler_init() { log_record_type_descriptor[LOGREC_RESERVED_FOR_CHUNKS23]= INIT_LOGREC_RESERVED_FOR_CHUNKS23; @@ -1378,9 +1417,11 @@ static uint16 translog_get_total_chunk_length(byte *page, uint16 offset) { /* first 2 bits is length - 2 */ uint len= ((((uint8) (*ptr)) & TRANSLOG_CLSN_LEN_BITS) >> 6) + 2; + if (ptr[0] == 0 && ((uint8) ptr[1]) == 1) + len+= LSN_STORE_SIZE; /* case of full LSN storing */ ptr+= len; /* subtract economized bytes */ - length-= (TRANSLOG_CLSN_MAX_LEN - len); + length-= (LSN_STORE_SIZE - len); } DBUG_PRINT("info", ("Pseudo-fixed length: %u", length)); DBUG_RETURN(length); @@ -3215,13 +3256,21 @@ static byte *translog_put_LSN_diff(LSN base_lsn, LSN lsn, byte *dst) offset_diff= base_offset - LSN_OFFSET(lsn); if (diff > 0x3f) { - /*TODO: error - too long transaction - panic!!! */ - UNRECOVERABLE_ERROR(("Too big file diff: %lu", (ulong) diff)); - DBUG_RETURN(NULL); + /* + It is full LSN after special 1 diff (which is impossible + in real life) + */ + dst-= 2 + LSN_STORE_SIZE; + dst[0]= 0; + dst[1]= 1; + lsn_store(dst + 2, lsn); + } + else + { + dst-= 5; + *dst= (0xC0 | diff); + int4store(dst + 1, offset_diff); } - dst-= 5; - *dst= (0xC0 | diff); - int4store(dst + 1, offset_diff); } DBUG_PRINT("info", ("new dst: 0x%lx", (ulong) dst)); DBUG_RETURN(dst); @@ -3275,6 +3324,17 @@ static byte *translog_get_LSN_from_diff(LSN base_lsn, byte *src, byte *dst) (uint) code, (ulong) first_byte)); switch (code) { case 0: + if (first_byte == 0 && *((uint8*)src) == 1) + { + /* + It is full LSN after special 1 diff (which is impossible + in real life) + */ + memcpy(dst, src + 1, LSN_STORE_SIZE); + DBUG_PRINT("info", ("Special case of full LSN, new src: 0x%lx", + (ulong) (src + 1 + LSN_STORE_SIZE))); + DBUG_RETURN(src + 1 + LSN_STORE_SIZE); + } rec_offset= LSN_OFFSET(base_lsn) - ((first_byte << 8) + *((uint8*)src)); break; case 1: @@ -3306,7 +3366,7 @@ static byte *translog_get_LSN_from_diff(LSN base_lsn, byte *src, byte *dst) lsn= MAKE_LSN(file_no, rec_offset); src+= code + 1; lsn_store(dst, lsn); - DBUG_PRINT("info", ("new src: 0x%lx", (ulong) dst)); + DBUG_PRINT("info", ("new src: 0x%lx", (ulong) src)); DBUG_RETURN(src); } @@ -3332,59 +3392,77 @@ static my_bool translog_relative_LSN_encode(struct st_translog_parts *parts, { LEX_STRING *part; uint lsns_len= lsns * LSN_STORE_SIZE; + char buffer_src[MAX_NUMBER_OF_LSNS_PER_RECORD * LSN_STORE_SIZE]; + char *buffer= buffer_src; DBUG_ENTER("translog_relative_LSN_encode"); + DBUG_ASSERT(parts->current != 0); part= parts->parts + parts->current; + /* collect all LSN(s) in one chunk if it (they) is (are) divided */ if (part->length < lsns_len) { uint copied= part->length; LEX_STRING *next_part; DBUG_PRINT("info", ("Using buffer: 0x%lx", (ulong) compressed_LSNs)); - memcpy(compressed_LSNs, (byte*)part->str, part->length); + memcpy(buffer, (byte*)part->str, part->length); next_part= parts->parts + parts->current + 1; do { DBUG_ASSERT(next_part < parts->parts + parts->elements); if ((next_part->length + copied) < lsns_len) { - memcpy(compressed_LSNs + copied, (byte*)next_part->str, + memcpy(buffer + copied, (byte*)next_part->str, next_part->length); copied+= next_part->length; next_part->length= 0; next_part->str= 0; /* delete_dynamic_element(&parts->parts, parts->current + 1); */ next_part++; + parts->current++; + part= parts->parts + parts->current; } else { uint len= lsns_len - copied; - memcpy(compressed_LSNs + copied, (byte*)next_part->str, len); + memcpy(buffer + copied, (byte*)next_part->str, len); copied= lsns_len; next_part->str+= len; next_part->length-= len; } } while (copied < lsns_len); - part->length= lsns_len; - part->str= (char*)compressed_LSNs; } + else + { + buffer= part->str; + part->str+= lsns_len; + part->length-= lsns_len; + parts->current--; + part= parts->parts + parts->current; + } + { /* Compress */ LSN ref; - uint economy; - byte *ref_ptr= (byte*)part->str + lsns_len - LSN_STORE_SIZE; - byte *dst_ptr= (byte*)part->str + lsns_len; - for (; ref_ptr >= (byte*)part->str ; ref_ptr-= LSN_STORE_SIZE) + int economy; + byte *src_ptr; + byte *dst_ptr= compressed_LSNs + (MAX_NUMBER_OF_LSNS_PER_RECORD * + COMPRESSED_LSN_MAX_STORE_SIZE); + for (src_ptr= buffer + lsns_len - LSN_STORE_SIZE; + src_ptr >= buffer; + src_ptr-= LSN_STORE_SIZE) { - ref= lsn_korr(ref_ptr); + ref= lsn_korr(src_ptr); if ((dst_ptr= translog_put_LSN_diff(base_lsn, ref, dst_ptr)) == NULL) DBUG_RETURN(1); } - /* Note that dst_ptr did grow downward ! */ - economy= (uint) (dst_ptr - (byte*)part->str); - DBUG_PRINT("info", ("Economy: %u", economy)); - part->length-= economy; - parts->record_length-= economy; + part->length= (uint)((compressed_LSNs + + (MAX_NUMBER_OF_LSNS_PER_RECORD * + COMPRESSED_LSN_MAX_STORE_SIZE)) - + dst_ptr); + parts->record_length-= (economy= lsns_len - part->length); + DBUG_PRINT("info", ("new length of LSNs: %u economy: %d", + part->length, economy)); parts->total_record_length-= economy; part->str= (char*)dst_ptr; } @@ -3853,7 +3931,8 @@ static my_bool translog_write_variable_record(LSN *lsn, ulong buffer_rest; uint page_rest; /* Max number of such LSNs per record is 2 */ - byte compressed_LSNs[2 * LSN_STORE_SIZE]; + byte compressed_LSNs[MAX_NUMBER_OF_LSNS_PER_RECORD * + COMPRESSED_LSN_MAX_STORE_SIZE]; DBUG_ENTER("translog_write_variable_record"); translog_lock(); @@ -3966,7 +4045,8 @@ static my_bool translog_write_fixed_record(LSN *lsn, struct st_translog_buffer *buffer_to_flush= NULL; byte chunk1_header[1 + 2]; /* Max number of such LSNs per record is 2 */ - byte compressed_LSNs[2 * LSN_STORE_SIZE]; + byte compressed_LSNs[MAX_NUMBER_OF_LSNS_PER_RECORD * + COMPRESSED_LSN_MAX_STORE_SIZE]; LEX_STRING *part; int rc; DBUG_ENTER("translog_write_fixed_record"); @@ -3976,8 +4056,7 @@ static my_bool translog_write_fixed_record(LSN *lsn, log_record_type_descriptor[type].fixed_length) || (log_record_type_descriptor[type].class == LOGRECTYPE_PSEUDOFIXEDLENGTH && - (parts->record_length - - log_record_type_descriptor[type].compressed_LSN * 2) <= + parts->record_length == log_record_type_descriptor[type].fixed_length)); translog_lock(); @@ -3989,19 +4068,18 @@ static my_bool translog_write_fixed_record(LSN *lsn, DBUG_PRINT("info", ("Page size: %u record: %u next cond: %d", log_descriptor.bc.current_page_fill, - (parts->record_length - + (parts->record_length + log_record_type_descriptor[type].compressed_LSN * 2 + 3), ((((uint) log_descriptor.bc.current_page_fill) + - (parts->record_length - + (parts->record_length + log_record_type_descriptor[type].compressed_LSN * 2 + 3)) > TRANSLOG_PAGE_SIZE))); /* - check that there is enough place on current page: - (log_record_type_descriptor[type].fixed_length - economized on compressed - LSNs) bytes + check that there is enough place on current page. + NOTE: compressing may increase page LSN size on two bytes for every LSN */ if ((((uint) log_descriptor.bc.current_page_fill) + - (parts->record_length - + (parts->record_length + log_record_type_descriptor[type].compressed_LSN * 2 + 3)) > TRANSLOG_PAGE_SIZE) { @@ -4123,7 +4201,7 @@ my_bool translog_write_record(LSN *lsn, parts.current= TRANSLOG_INTERNAL_PARTS; /* clear TRANSLOG_INTERNAL_PARTS */ - DBUG_ASSERT(TRANSLOG_INTERNAL_PARTS == 1); + DBUG_ASSERT(TRANSLOG_INTERNAL_PARTS != 0); parts_data[0].str= 0; parts_data[0].length= 0; @@ -4243,7 +4321,7 @@ translog_size_t translog_fixed_length_header(byte *page, byte *dst= buff->header; byte *start= src; uint lsns= desc->compressed_LSN; - uint length= desc->fixed_length + (lsns * 2); + uint length= desc->fixed_length; DBUG_ENTER("translog_fixed_length_header"); @@ -4256,7 +4334,7 @@ translog_size_t translog_fixed_length_header(byte *page, lsns*= LSN_STORE_SIZE; dst+= lsns; length-= lsns; - buff->compressed_LSN_economy= (uint16) (lsns - (src - start)); + buff->compressed_LSN_economy= (lsns - (src - start)); } else buff->compressed_LSN_economy= 0; @@ -4575,7 +4653,7 @@ translog_size_t translog_variable_length_header(byte *page, LSN base_lsn; uint lsns= desc->compressed_LSN; uint16 chunk_len; - uint16 length= desc->read_header_len + (lsns * 2); + uint16 length= desc->read_header_len; uint16 buffer_length= length; uint16 body_len; TRANSLOG_SCANNER_DATA internal_scanner; @@ -4697,10 +4775,10 @@ translog_size_t translog_variable_length_header(byte *page, dst+= lsns; length-= lsns; buff->record_length+= (buff->compressed_LSN_economy= - (uint16) (lsns - (src - start))); - DBUG_PRINT("info", ("lsns: %u length: %u economy: %u new length: %lu", + (lsns - (src - start))); + DBUG_PRINT("info", ("lsns: %u length: %u economy: %d new length: %lu", lsns / LSN_STORE_SIZE, (uint) length, - (uint) buff->compressed_LSN_economy, + (int) buff->compressed_LSN_economy, (ulong) buff->record_length)); body_len-= (src - start); } diff --git a/storage/maria/ma_loghandler.h b/storage/maria/ma_loghandler.h index 3ccb3bf9af2..967d75a53cb 100644 --- a/storage/maria/ma_loghandler.h +++ b/storage/maria/ma_loghandler.h @@ -55,7 +55,7 @@ struct st_maria_share; #define LOG_INTERNAL_PARTS 1 /* position reserved in an array of parts of a log record */ -#define TRANSLOG_INTERNAL_PARTS 1 +#define TRANSLOG_INTERNAL_PARTS 2 /* types of records in the transaction log */ /* Todo: Set numbers for these when we have all entries figured out */ @@ -140,7 +140,7 @@ typedef struct st_translog_header_buffer /* Real compressed LSN(s) size economy (*7 - ) */ - uint16 compressed_LSN_economy; + int16 compressed_LSN_economy; /* short transaction ID or 0 if it has no sense for the record */ uint16 non_header_data_start_offset; /* non read body data length in this first chunk */ @@ -184,7 +184,16 @@ struct st_translog_reader_data extern "C" { #endif -extern void loghandler_init(); +/* Records types for unittests */ +#define LOGREC_FIXED_RECORD_0LSN_EXAMPLE 1 +#define LOGREC_VARIABLE_RECORD_0LSN_EXAMPLE 2 +#define LOGREC_FIXED_RECORD_1LSN_EXAMPLE 3 +#define LOGREC_VARIABLE_RECORD_1LSN_EXAMPLE 4 +#define LOGREC_FIXED_RECORD_2LSN_EXAMPLE 5 +#define LOGREC_VARIABLE_RECORD_2LSN_EXAMPLE 6 + +extern void example_loghandler_init(); + extern my_bool translog_init(const char *directory, uint32 log_file_max_size, uint32 server_version, uint32 server_id, PAGECACHE *pagecache, uint flags); diff --git a/storage/maria/unittest/Makefile.am b/storage/maria/unittest/Makefile.am index 76c8aa24779..28264d5d903 100644 --- a/storage/maria/unittest/Makefile.am +++ b/storage/maria/unittest/Makefile.am @@ -41,12 +41,15 @@ noinst_PROGRAMS = ma_control_file-t trnman-t lockman2-t \ ma_test_loghandler-t \ ma_test_loghandler_multigroup-t \ ma_test_loghandler_multithread-t \ - ma_test_loghandler_pagecache-t + ma_test_loghandler_pagecache-t \ + ma_test_loghandler_long-t-big -ma_test_loghandler_t_SOURCES= ma_test_loghandler-t.c ma_maria_log_cleanup.c -ma_test_loghandler_multigroup_t_SOURCES= ma_test_loghandler_multigroup-t.c ma_maria_log_cleanup.c -ma_test_loghandler_multithread_t_SOURCES= ma_test_loghandler_multithread-t.c ma_maria_log_cleanup.c -ma_test_loghandler_pagecache_t_SOURCES= ma_test_loghandler_pagecache-t.c ma_maria_log_cleanup.c +ma_test_loghandler_t_SOURCES = ma_test_loghandler-t.c ma_maria_log_cleanup.c +ma_test_loghandler_multigroup_t_SOURCES = ma_test_loghandler_multigroup-t.c ma_maria_log_cleanup.c +ma_test_loghandler_multithread_t_SOURCES = ma_test_loghandler_multithread-t.c ma_maria_log_cleanup.c +ma_test_loghandler_pagecache_t_SOURCES = ma_test_loghandler_pagecache-t.c ma_maria_log_cleanup.c +ma_test_loghandler_long_t_big_SOURCES = ma_test_loghandler-t.c ma_maria_log_cleanup.c +ma_test_loghandler_long_t_big_CPPFLAGS = -DLONG_LOG_TEST ma_pagecache_single_src = ma_pagecache_single.c test_file.c ma_pagecache_consist_src = ma_pagecache_consist.c test_file.c diff --git a/storage/maria/unittest/ma_test_loghandler-t.c b/storage/maria/unittest/ma_test_loghandler-t.c index bff5864a5c0..a19a32231ea 100644 --- a/storage/maria/unittest/ma_test_loghandler-t.c +++ b/storage/maria/unittest/ma_test_loghandler-t.c @@ -13,10 +13,15 @@ static const char *default_dbug_option; #define LONG_BUFFER_SIZE (100 * 1024) - +#ifdef LONG_LOG_TEST +#define LOG_FLAGS 0 +#define LOG_FILE_SIZE (1024L*1024L) +#define ITERATIONS (1600*4) +#else #define LOG_FLAGS TRANSLOG_SECTOR_PROTECTION | TRANSLOG_PAGE_CRC -#define LOG_FILE_SIZE 1024L*1024L*3L +#define LOG_FILE_SIZE (1024L*1024L*3L) #define ITERATIONS 1600 +#endif /* #define LOG_FLAGS 0 @@ -68,6 +73,23 @@ static my_bool check_content(byte *ptr, ulong length) } +/* + Report OK for read operation + + SYNOPSIS + read_ok() + rec the record header +*/ + +void read_ok(TRANSLOG_HEADER_BUFFER *rec) +{ + char buff[80]; + snprintf(buff, sizeof(buff), "read record type: %u LSN: (%lu,0x%lx)", + rec->type, (ulong) LSN_FILE_NO(rec->lsn), + (ulong) LSN_OFFSET(rec->lsn)); + ok(1, buff); +} + /* Read whole record content, and check content (put with offset) @@ -92,6 +114,7 @@ static my_bool read_and_check_content(TRANSLOG_HEADER_BUFFER *rec, return check_content(buffer + skip, rec->record_length - skip); } + int main(int argc __attribute__((unused)), char *argv[]) { uint32 i; @@ -156,6 +179,7 @@ int main(int argc __attribute__((unused)), char *argv[]) translog_destroy(); exit(1); } + example_loghandler_init(); plan(((ITERATIONS - 1) * 4 + 1)*2 + ITERATIONS - 1); @@ -167,16 +191,16 @@ int main(int argc __attribute__((unused)), char *argv[]) parts[TRANSLOG_INTERNAL_PARTS + 0].str= (char*)long_tr_id; parts[TRANSLOG_INTERNAL_PARTS + 0].length= 6; if (translog_write_record(&lsn, - LOGREC_LONG_TRANSACTION_ID, + LOGREC_FIXED_RECORD_0LSN_EXAMPLE, 0, NULL, NULL, 6, TRANSLOG_INTERNAL_PARTS + 1, parts)) { fprintf(stderr, "Can't write record #%lu\n", (ulong) 0); translog_destroy(); - ok(0, "write LOGREC_LONG_TRANSACTION_ID"); + ok(0, "write LOGREC_FIXED_RECORD_0LSN_EXAMPLE"); exit(1); } - ok(1, "write LOGREC_LONG_TRANSACTION_ID"); + ok(1, "write LOGREC_FIXED_RECORD_0LSN_EXAMPLE"); lsn_base= first_lsn= lsn; for (i= 1; i < ITERATIONS; i++) @@ -189,16 +213,17 @@ int main(int argc __attribute__((unused)), char *argv[]) /* check auto-count feature */ parts[TRANSLOG_INTERNAL_PARTS + 1].str= NULL; parts[TRANSLOG_INTERNAL_PARTS + 1].length= 0; - if (translog_write_record(&lsn, LOGREC_CLR_END, (i % 0xFFFF), NULL, + if (translog_write_record(&lsn, LOGREC_FIXED_RECORD_1LSN_EXAMPLE, + (i % 0xFFFF), NULL, NULL, LSN_STORE_SIZE, 0, parts)) { fprintf(stderr, "1 Can't write reference defore record #%lu\n", (ulong) i); translog_destroy(); - ok(0, "write LOGREC_CLR_END"); + ok(0, "write LOGREC_FIXED_RECORD_1LSN_EXAMPLE"); exit(1); } - ok(1, "write LOGREC_CLR_END"); + ok(1, "write LOGREC_FIXED_RECORD_1LSN_EXAMPLE"); lsn_store(lsn_buff, lsn_base); if ((rec_len= random() / (RAND_MAX / (LONG_BUFFER_SIZE + 1))) < 12) rec_len= 12; @@ -208,7 +233,7 @@ int main(int argc __attribute__((unused)), char *argv[]) parts[TRANSLOG_INTERNAL_PARTS + 1].length= rec_len; /* check record length auto-counting */ if (translog_write_record(&lsn, - LOGREC_UNDO_KEY_INSERT, + LOGREC_VARIABLE_RECORD_1LSN_EXAMPLE, (i % 0xFFFF), NULL, NULL, 0, TRANSLOG_INTERNAL_PARTS + 2, parts)) @@ -216,10 +241,10 @@ int main(int argc __attribute__((unused)), char *argv[]) fprintf(stderr, "1 Can't write var reference defore record #%lu\n", (ulong) i); translog_destroy(); - ok(0, "write LOGREC_UNDO_KEY_INSERT"); + ok(0, "write LOGREC_VARIABLE_RECORD_1LSN_EXAMPLE"); exit(1); } - ok(1, "write LOGREC_UNDO_KEY_INSERT"); + ok(1, "write LOGREC_VARIABLE_RECORD_1LSN_EXAMPLE"); } else { @@ -228,17 +253,17 @@ int main(int argc __attribute__((unused)), char *argv[]) parts[TRANSLOG_INTERNAL_PARTS + 0].str= (char*)lsn_buff; parts[TRANSLOG_INTERNAL_PARTS + 0].length= 23; if (translog_write_record(&lsn, - LOGREC_UNDO_ROW_DELETE, + LOGREC_FIXED_RECORD_2LSN_EXAMPLE, (i % 0xFFFF), NULL, NULL, 23, TRANSLOG_INTERNAL_PARTS + 1, parts)) { fprintf(stderr, "0 Can't write reference defore record #%lu\n", (ulong) i); translog_destroy(); - ok(0, "write LOGREC_UNDO_ROW_DELETE"); + ok(0, "write LOGREC_FIXED_RECORD_2LSN_EXAMPLE"); exit(1); } - ok(1, "write LOGREC_UNDO_ROW_DELETE"); + ok(1, "write LOGREC_FIXED_RECORD_2LSN_EXAMPLE"); lsn_store(lsn_buff, lsn_base); lsn_store(lsn_buff + LSN_STORE_SIZE, first_lsn); if ((rec_len= random() / (RAND_MAX / (LONG_BUFFER_SIZE + 1))) < 19) @@ -248,7 +273,7 @@ int main(int argc __attribute__((unused)), char *argv[]) parts[TRANSLOG_INTERNAL_PARTS + 1].str= (char*)long_buffer; parts[TRANSLOG_INTERNAL_PARTS + 1].length= rec_len; if (translog_write_record(&lsn, - LOGREC_UNDO_KEY_DELETE, + LOGREC_VARIABLE_RECORD_2LSN_EXAMPLE, (i % 0xFFFF), NULL, NULL, 14 + rec_len, TRANSLOG_INTERNAL_PARTS + 2, parts)) @@ -256,26 +281,26 @@ int main(int argc __attribute__((unused)), char *argv[]) fprintf(stderr, "0 Can't write var reference defore record #%lu\n", (ulong) i); translog_destroy(); - ok(0, "write LOGREC_UNDO_KEY_DELETE"); + ok(0, "write LOGREC_VARIABLE_RECORD_2LSN_EXAMPLE"); exit(1); } - ok(1, "write LOGREC_UNDO_KEY_DELETE"); + ok(1, "write LOGREC_VARIABLE_RECORD_2LSN_EXAMPLE"); } int4store(long_tr_id, i); parts[TRANSLOG_INTERNAL_PARTS + 0].str= (char*)long_tr_id; parts[TRANSLOG_INTERNAL_PARTS + 0].length= 6; if (translog_write_record(&lsn, - LOGREC_LONG_TRANSACTION_ID, + LOGREC_FIXED_RECORD_0LSN_EXAMPLE, (i % 0xFFFF), NULL, NULL, 6, TRANSLOG_INTERNAL_PARTS + 1, parts)) { fprintf(stderr, "Can't write record #%lu\n", (ulong) i); translog_destroy(); - ok(0, "write LOGREC_LONG_TRANSACTION_ID"); + ok(0, "write LOGREC_FIXED_RECORD_0LSN_EXAMPLE"); exit(1); } - ok(1, "write LOGREC_LONG_TRANSACTION_ID"); + ok(1, "write LOGREC_FIXED_RECORD_0LSN_EXAMPLE"); lsn_base= lsn; @@ -284,17 +309,17 @@ int main(int argc __attribute__((unused)), char *argv[]) parts[TRANSLOG_INTERNAL_PARTS + 0].str= (char*)long_buffer; parts[TRANSLOG_INTERNAL_PARTS + 0].length= rec_len; if (translog_write_record(&lsn, - LOGREC_REDO_INSERT_ROW_HEAD, + LOGREC_VARIABLE_RECORD_0LSN_EXAMPLE, (i % 0xFFFF), NULL, NULL, rec_len, TRANSLOG_INTERNAL_PARTS + 1, parts)) { fprintf(stderr, "Can't write variable record #%lu\n", (ulong) i); translog_destroy(); - ok(0, "write LOGREC_REDO_INSERT_ROW_HEAD"); + ok(0, "write LOGREC_VARIABLE_RECORD_0LSN_EXAMPLE"); exit(1); } - ok(1, "write LOGREC_REDO_INSERT_ROW_HEAD"); + ok(1, "write LOGREC_VARIABLE_RECORD_0LSN_EXAMPLE"); if (translog_flush(lsn)) { fprintf(stderr, "Can't flush #%lu\n", (ulong) i); @@ -327,6 +352,7 @@ int main(int argc __attribute__((unused)), char *argv[]) translog_destroy(); exit(1); } + example_loghandler_init(); srandom(122334817L); @@ -339,12 +365,13 @@ int main(int argc __attribute__((unused)), char *argv[]) fprintf(stderr, "translog_read_record_header failed (%d)\n", errno); goto err; } - if (rec.type !=LOGREC_LONG_TRANSACTION_ID || rec.short_trid != 0 || + if (rec.type !=LOGREC_FIXED_RECORD_0LSN_EXAMPLE || rec.short_trid != 0 || rec.record_length != 6 || uint4korr(rec.header) != 0 || ((uchar)rec.header[4]) != 0 || ((uchar)rec.header[5]) != 0xFF || first_lsn != rec.lsn) { - fprintf(stderr, "Incorrect LOGREC_LONG_TRANSACTION_ID data read(0)\n" + fprintf(stderr, "Incorrect LOGREC_FIXED_RECORD_0LSN_EXAMPLE " + "data read(0)\n" "type %u, strid %u, len %u, i: %u, 4: %u 5: %u, " "lsn(%lu,0x%lx)\n", (uint) rec.type, (uint) rec.short_trid, (uint) rec.record_length, @@ -353,7 +380,7 @@ int main(int argc __attribute__((unused)), char *argv[]) (ulong) LSN_FILE_NO(rec.lsn), (ulong) LSN_OFFSET(rec.lsn)); goto err; } - ok(1, "read record"); + read_ok(&rec); translog_free_record_header(&rec); lsn= first_lsn; if (translog_init_scanner(first_lsn, 1, &scanner)) @@ -384,10 +411,12 @@ int main(int argc __attribute__((unused)), char *argv[]) { LSN ref; ref= lsn_korr(rec.header); - if (rec.type !=LOGREC_CLR_END || rec.short_trid != (i % 0xFFFF) || + if (rec.type != LOGREC_FIXED_RECORD_1LSN_EXAMPLE || + rec.short_trid != (i % 0xFFFF) || rec.record_length != 7 || ref != lsn) { - fprintf(stderr, "Incorrect LOGREC_CLR_END data read(%d) " + fprintf(stderr, "Incorrect LOGREC_FIXED_RECORD_1LSN_EXAMPLE " + "data read(%d) " "type: %u strid: %u len: %u" "ref: (%lu,0x%lx) (%lu,0x%lx) " "lsn(%lu,0x%lx)\n", @@ -404,7 +433,7 @@ int main(int argc __attribute__((unused)), char *argv[]) LSN ref1, ref2; ref1= lsn_korr(rec.header); ref2= lsn_korr(rec.header + LSN_STORE_SIZE); - if (rec.type != LOGREC_UNDO_ROW_DELETE || + if (rec.type != LOGREC_FIXED_RECORD_2LSN_EXAMPLE || rec.short_trid != (i % 0xFFFF) || rec.record_length != 23 || ref1 != lsn || @@ -419,7 +448,8 @@ int main(int argc __attribute__((unused)), char *argv[]) ((uchar)rec.header[15]) != 0xAA || ((uchar)rec.header[14]) != 0x55) { - fprintf(stderr, "Incorrect LOGREC_UNDO_ROW_DELETE data read(%d)" + fprintf(stderr, "Incorrect LOGREC_FIXED_RECORD_2LSN_EXAMPLE " + "data read(%d) " "type %u, strid %u, len %u, ref1(%lu,0x%lx), " "ref2(%lu,0x%lx) %x%x%x%x%x%x%x%x%x " "lsn(%lu,0x%lx)\n", @@ -436,7 +466,7 @@ int main(int argc __attribute__((unused)), char *argv[]) goto err; } } - ok(1, "read record"); + read_ok(&rec); translog_free_record_header(&rec); len= translog_read_next_record_header(&scanner, &rec); @@ -458,18 +488,19 @@ int main(int argc __attribute__((unused)), char *argv[]) ref= lsn_korr(rec.header); if ((rec_len= random() / (RAND_MAX / (LONG_BUFFER_SIZE + 1))) < 12) rec_len= 12; - if (rec.type !=LOGREC_UNDO_KEY_INSERT || + if (rec.type != LOGREC_VARIABLE_RECORD_1LSN_EXAMPLE || rec.short_trid != (i % 0xFFFF) || rec.record_length != rec_len + LSN_STORE_SIZE || len != 12 || ref != lsn || check_content(rec.header + LSN_STORE_SIZE, len - LSN_STORE_SIZE)) { - fprintf(stderr, "Incorrect LOGREC_UNDO_KEY_INSERT data read(%d)" + fprintf(stderr, "Incorrect LOGREC_VARIABLE_RECORD_1LSN_EXAMPLE " + "data read(%d)" "type %u (%d), strid %u (%d), len %lu, %lu + 7 (%d), " "hdr len: %u (%d), " "ref(%lu,0x%lx), lsn(%lu,0x%lx) (%d), content: %d\n", i, (uint) rec.type, - rec.type !=LOGREC_UNDO_KEY_INSERT, + rec.type != LOGREC_VARIABLE_RECORD_1LSN_EXAMPLE, (uint) rec.short_trid, rec.short_trid != (i % 0xFFFF), (ulong) rec.record_length, (ulong) rec_len, @@ -486,8 +517,8 @@ int main(int argc __attribute__((unused)), char *argv[]) if (read_and_check_content(&rec, long_buffer, LSN_STORE_SIZE)) { fprintf(stderr, - "Incorrect LOGREC_UNDO_KEY_INSERT in whole rec read " - "lsn(%lu,0x%lx)\n", + "Incorrect LOGREC_VARIABLE_RECORD_1LSN_EXAMPLE " + "in whole rec read lsn(%lu,0x%lx)\n", (ulong) LSN_FILE_NO(rec.lsn), (ulong) LSN_OFFSET(rec.lsn)); goto err; } @@ -499,7 +530,7 @@ int main(int argc __attribute__((unused)), char *argv[]) ref2= lsn_korr(rec.header + LSN_STORE_SIZE); if ((rec_len= random() / (RAND_MAX / (LONG_BUFFER_SIZE + 1))) < 19) rec_len= 19; - if (rec.type !=LOGREC_UNDO_KEY_DELETE || + if (rec.type != LOGREC_VARIABLE_RECORD_2LSN_EXAMPLE || rec.short_trid != (i % 0xFFFF) || rec.record_length != rec_len + LSN_STORE_SIZE * 2 || len != 19 || @@ -508,7 +539,8 @@ int main(int argc __attribute__((unused)), char *argv[]) check_content(rec.header + LSN_STORE_SIZE * 2, len - LSN_STORE_SIZE * 2)) { - fprintf(stderr, "Incorrect LOGREC_UNDO_KEY_DELETE data read(%d)" + fprintf(stderr, "Incorrect LOGREC_VARIABLE_RECORD_2LSN_EXAMPLE " + "data read(%d) " "type %u, strid %u, len %lu != %lu + 14, hdr len: %u, " "ref1(%lu,0x%lx), ref2(%lu,0x%lx), " "lsn(%lu,0x%lx)\n", @@ -523,13 +555,13 @@ int main(int argc __attribute__((unused)), char *argv[]) if (read_and_check_content(&rec, long_buffer, LSN_STORE_SIZE * 2)) { fprintf(stderr, - "Incorrect LOGREC_UNDO_KEY_DELETE in whole rec read " - "lsn(%lu,0x%lx)\n", + "Incorrect LOGREC_VARIABLE_RECORD_2LSN_EXAMPLE " + "in whole rec read lsn(%lu,0x%lx)\n", (ulong) LSN_FILE_NO(rec.lsn), (ulong) LSN_OFFSET(rec.lsn)); goto err; } } - ok(1, "read record"); + read_ok(&rec); translog_free_record_header(&rec); len= translog_read_next_record_header(&scanner, &rec); @@ -545,12 +577,13 @@ int main(int argc __attribute__((unused)), char *argv[]) "instead of beginning of %u\n", i, ITERATIONS); goto err; } - if (rec.type !=LOGREC_LONG_TRANSACTION_ID || + if (rec.type != LOGREC_FIXED_RECORD_0LSN_EXAMPLE || rec.short_trid != (i % 0xFFFF) || rec.record_length != 6 || uint4korr(rec.header) != i || ((uchar)rec.header[4]) != 0 || ((uchar)rec.header[5]) != 0xFF) { - fprintf(stderr, "Incorrect LOGREC_LONG_TRANSACTION_ID data read(%d)\n" + fprintf(stderr, "Incorrect LOGREC_FIXED_RECORD_0LSN_EXAMPLE " + "data read(%d)\n" "type %u, strid %u, len %u, i: %u, 4: %u 5: %u " "lsn(%lu,0x%lx)\n", i, (uint) rec.type, (uint) rec.short_trid, @@ -561,18 +594,19 @@ int main(int argc __attribute__((unused)), char *argv[]) goto err; } lsn= rec.lsn; - ok(1, "read record"); + read_ok(&rec); translog_free_record_header(&rec); len= translog_read_next_record_header(&scanner, &rec); if ((rec_len= random() / (RAND_MAX / (LONG_BUFFER_SIZE + 1))) < 9) rec_len= 9; - if (rec.type !=LOGREC_REDO_INSERT_ROW_HEAD || + if (rec.type != LOGREC_VARIABLE_RECORD_0LSN_EXAMPLE || rec.short_trid != (i % 0xFFFF) || rec.record_length != rec_len || len != 9 || check_content(rec.header, len)) { - fprintf(stderr, "Incorrect LOGREC_REDO_INSERT_ROW_HEAD data read(%d)" + fprintf(stderr, "Incorrect LOGREC_VARIABLE_RECORD_0LSN_EXAMPLE " + "data read(%d) " "type %u, strid %u, len %lu != %lu, hdr len: %u, " "lsn(%lu,0x%lx)\n", i, (uint) rec.type, (uint) rec.short_trid, @@ -584,12 +618,12 @@ int main(int argc __attribute__((unused)), char *argv[]) if (read_and_check_content(&rec, long_buffer, 0)) { fprintf(stderr, - "Incorrect LOGREC_UNDO_KEY_DELETE in whole rec read " - "lsn(%lu,0x%lx)\n", + "Incorrect LOGREC_VARIABLE_RECORD_2LSN_EXAMPLE " + "in whole rec read lsn(%lu,0x%lx)\n", (ulong) LSN_FILE_NO(rec.lsn), (ulong) LSN_OFFSET(rec.lsn)); goto err; } - ok(1, "read record"); + read_ok(&rec); translog_free_record_header(&rec); } } diff --git a/storage/maria/unittest/ma_test_loghandler_multigroup-t.c b/storage/maria/unittest/ma_test_loghandler_multigroup-t.c index 110c35b786a..db2f5dfef5f 100644 --- a/storage/maria/unittest/ma_test_loghandler_multigroup-t.c +++ b/storage/maria/unittest/ma_test_loghandler_multigroup-t.c @@ -176,6 +176,7 @@ int main(int argc __attribute__((unused)), char *argv[]) translog_destroy(); exit(1); } + example_loghandler_init(); plan(((ITERATIONS - 1) * 4 + 1) * 2); @@ -186,15 +187,16 @@ int main(int argc __attribute__((unused)), char *argv[]) int4store(long_tr_id, 0); parts[TRANSLOG_INTERNAL_PARTS + 0].str= (char*)long_tr_id; parts[TRANSLOG_INTERNAL_PARTS + 0].length= 6; - if (translog_write_record(&lsn, LOGREC_LONG_TRANSACTION_ID, 0, NULL, NULL, + if (translog_write_record(&lsn, LOGREC_FIXED_RECORD_0LSN_EXAMPLE, + 0, NULL, NULL, 6, TRANSLOG_INTERNAL_PARTS + 1, parts)) { fprintf(stderr, "Can't write record #%lu\n", (ulong) 0); translog_destroy(); - ok(0, "write LOGREC_LONG_TRANSACTION_ID"); + ok(0, "write LOGREC_FIXED_RECORD_0LSN_EXAMPLE"); exit(1); } - ok(1, "write LOGREC_LONG_TRANSACTION_ID"); + ok(1, "write LOGREC_FIXED_RECORD_0LSN_EXAMPLE"); lsn_base= first_lsn= lsn; for (i= 1; i < ITERATIONS; i++) @@ -205,7 +207,7 @@ int main(int argc __attribute__((unused)), char *argv[]) parts[TRANSLOG_INTERNAL_PARTS + 0].str= (char*)lsn_buff; parts[TRANSLOG_INTERNAL_PARTS + 0].length= LSN_STORE_SIZE; if (translog_write_record(&lsn, - LOGREC_CLR_END, + LOGREC_FIXED_RECORD_1LSN_EXAMPLE, (i % 0xFFFF), NULL, NULL, LSN_STORE_SIZE, TRANSLOG_INTERNAL_PARTS + 1, parts)) @@ -213,10 +215,10 @@ int main(int argc __attribute__((unused)), char *argv[]) fprintf(stderr, "1 Can't write reference before record #%lu\n", (ulong) i); translog_destroy(); - ok(0, "write LOGREC_CLR_END"); + ok(0, "write LOGREC_FIXED_RECORD_1LSN_EXAMPLE"); exit(1); } - ok(1, "write LOGREC_CLR_END"); + ok(1, "write LOGREC_FIXED_RECORD_1LSN_EXAMPLE"); lsn_store(lsn_buff, lsn_base); rec_len= get_len(); parts[TRANSLOG_INTERNAL_PARTS + 0].str= (char*)lsn_buff; @@ -224,7 +226,7 @@ int main(int argc __attribute__((unused)), char *argv[]) parts[TRANSLOG_INTERNAL_PARTS + 1].str= (char*)long_buffer; parts[TRANSLOG_INTERNAL_PARTS + 1].length= rec_len; if (translog_write_record(&lsn, - LOGREC_UNDO_KEY_INSERT, + LOGREC_VARIABLE_RECORD_1LSN_EXAMPLE, (i % 0xFFFF), NULL, NULL, LSN_STORE_SIZE + rec_len, TRANSLOG_INTERNAL_PARTS + 2, @@ -233,10 +235,10 @@ int main(int argc __attribute__((unused)), char *argv[]) fprintf(stderr, "1 Can't write var reference before record #%lu\n", (ulong) i); translog_destroy(); - ok(0, "write LOGREC_UNDO_KEY_INSERT"); + ok(0, "write LOGREC_VARIABLE_RECORD_1LSN_EXAMPLE"); exit(1); } - ok(1, "write LOGREC_UNDO_KEY_INSERT"); + ok(1, "write LOGREC_VARIABLE_RECORD_1LSN_EXAMPLE"); } else { @@ -245,7 +247,7 @@ int main(int argc __attribute__((unused)), char *argv[]) parts[TRANSLOG_INTERNAL_PARTS + 1].str= (char*)lsn_buff; parts[TRANSLOG_INTERNAL_PARTS + 1].length= 23; if (translog_write_record(&lsn, - LOGREC_UNDO_ROW_DELETE, + LOGREC_FIXED_RECORD_2LSN_EXAMPLE, (i % 0xFFFF), NULL, NULL, 23, TRANSLOG_INTERNAL_PARTS + 1, parts)) @@ -253,10 +255,10 @@ int main(int argc __attribute__((unused)), char *argv[]) fprintf(stderr, "0 Can't write reference before record #%lu\n", (ulong) i); translog_destroy(); - ok(0, "write LOGREC_UNDO_ROW_DELETE"); + ok(0, "write LOGREC_FIXED_RECORD_2LSN_EXAMPLE"); exit(1); } - ok(1, "write LOGREC_UNDO_ROW_DELETE"); + ok(1, "write LOGREC_FIXED_RECORD_2LSN_EXAMPLE"); lsn_store(lsn_buff, lsn_base); lsn_store(lsn_buff + LSN_STORE_SIZE, first_lsn); rec_len= get_len(); @@ -265,7 +267,7 @@ int main(int argc __attribute__((unused)), char *argv[]) parts[TRANSLOG_INTERNAL_PARTS + 1].str= (char*)long_buffer; parts[TRANSLOG_INTERNAL_PARTS + 1].length= rec_len; if (translog_write_record(&lsn, - LOGREC_UNDO_KEY_DELETE, + LOGREC_VARIABLE_RECORD_2LSN_EXAMPLE, (i % 0xFFFF), NULL, NULL, LSN_STORE_SIZE * 2 + rec_len, TRANSLOG_INTERNAL_PARTS + 2, @@ -274,25 +276,25 @@ int main(int argc __attribute__((unused)), char *argv[]) fprintf(stderr, "0 Can't write var reference before record #%lu\n", (ulong) i); translog_destroy(); - ok(0, "write LOGREC_UNDO_KEY_DELETE"); + ok(0, "write LOGREC_VARIABLE_RECORD_2LSN_EXAMPLE"); exit(1); } - ok(1, "write LOGREC_UNDO_KEY_DELETE"); + ok(1, "write LOGREC_VARIABLE_RECORD_2LSN_EXAMPLE"); } int4store(long_tr_id, i); parts[TRANSLOG_INTERNAL_PARTS + 0].str= (char*)long_tr_id; parts[TRANSLOG_INTERNAL_PARTS + 0].length= 6; if (translog_write_record(&lsn, - LOGREC_LONG_TRANSACTION_ID, + LOGREC_FIXED_RECORD_0LSN_EXAMPLE, (i % 0xFFFF), NULL, NULL, 6, TRANSLOG_INTERNAL_PARTS + 1, parts)) { fprintf(stderr, "Can't write record #%lu\n", (ulong) i); translog_destroy(); - ok(0, "write LOGREC_LONG_TRANSACTION_ID"); + ok(0, "write LOGREC_FIXED_RECORD_0LSN_EXAMPLE"); exit(1); } - ok(1, "write LOGREC_LONG_TRANSACTION_ID"); + ok(1, "write LOGREC_FIXED_RECORD_0LSN_EXAMPLE"); lsn_base= lsn; @@ -300,16 +302,16 @@ int main(int argc __attribute__((unused)), char *argv[]) parts[TRANSLOG_INTERNAL_PARTS + 0].str= (char*)long_buffer; parts[TRANSLOG_INTERNAL_PARTS + 0].length= rec_len; if (translog_write_record(&lsn, - LOGREC_REDO_INSERT_ROW_HEAD, + LOGREC_VARIABLE_RECORD_0LSN_EXAMPLE, (i % 0xFFFF), NULL, NULL, rec_len, TRANSLOG_INTERNAL_PARTS + 1, parts)) { fprintf(stderr, "Can't write variable record #%lu\n", (ulong) i); translog_destroy(); - ok(0, "write LOGREC_REDO_INSERT_ROW_HEAD"); + ok(0, "write LOGREC_VARIABLE_RECORD_0LSN_EXAMPLE"); exit(1); } - ok(1, "write LOGREC_REDO_INSERT_ROW_HEAD"); + ok(1, "write LOGREC_VARIABLE_RECORD_0LSN_EXAMPLE"); } translog_destroy(); @@ -333,6 +335,7 @@ int main(int argc __attribute__((unused)), char *argv[]) translog_destroy(); exit(1); } + example_loghandler_init(); srandom(122334817L); @@ -346,12 +349,13 @@ int main(int argc __attribute__((unused)), char *argv[]) translog_free_record_header(&rec); goto err; } - if (rec.type !=LOGREC_LONG_TRANSACTION_ID || rec.short_trid != 0 || + if (rec.type !=LOGREC_FIXED_RECORD_0LSN_EXAMPLE || rec.short_trid != 0 || rec.record_length != 6 || uint4korr(rec.header) != 0 || ((uchar)rec.header[4]) != 0 || ((uchar)rec.header[5]) != 0xFF || first_lsn != rec.lsn) { - fprintf(stderr, "Incorrect LOGREC_LONG_TRANSACTION_ID data read(0)\n" + fprintf(stderr, "Incorrect LOGREC_FIXED_RECORD_0LSN_EXAMPLE " + "data read(0)\n" "type %u, strid %u, len %u, i: %u, 4: %u 5: %u, " "lsn(0x%lu,0x%lx)\n", (uint) rec.type, (uint) rec.short_trid, (uint) rec.record_length, @@ -395,10 +399,12 @@ int main(int argc __attribute__((unused)), char *argv[]) { LSN ref; ref= lsn_korr(rec.header); - if (rec.type != LOGREC_CLR_END || rec.short_trid != (i % 0xFFFF) || + if (rec.type != LOGREC_FIXED_RECORD_1LSN_EXAMPLE || + rec.short_trid != (i % 0xFFFF) || rec.record_length != LSN_STORE_SIZE || ref != lsn) { - fprintf(stderr, "Incorrect LOGREC_CLR_END data read(%d)" + fprintf(stderr, "Incorrect LOGREC_FIXED_RECORD_1LSN_EXAMPLE " + "data read(%d)" "type %u, strid %u, len %u, ref(%lu,0x%lx), lsn(%lu,0x%lx)\n", i, (uint) rec.type, (uint) rec.short_trid, (uint) rec.record_length, @@ -413,7 +419,7 @@ int main(int argc __attribute__((unused)), char *argv[]) LSN ref1, ref2; ref1= lsn_korr(rec.header); ref2= lsn_korr(rec.header + LSN_STORE_SIZE); - if (rec.type !=LOGREC_UNDO_ROW_DELETE || + if (rec.type != LOGREC_FIXED_RECORD_2LSN_EXAMPLE || rec.short_trid != (i % 0xFFFF) || rec.record_length != 23 || ref1 != lsn || @@ -428,7 +434,8 @@ int main(int argc __attribute__((unused)), char *argv[]) ((uchar)rec.header[15]) != 0xAA || ((uchar)rec.header[14]) != 0x55) { - fprintf(stderr, "Incorrect LOGREC_UNDO_ROW_DELETE data read(%d)" + fprintf(stderr, "Incorrect LOGREC_FIXED_RECORD_2LSN_EXAMPLE " + "data read(%d) " "type %u, strid %u, len %u, ref1(%lu,0x%lx), " "ref2(%lu,0x%lx) %x%x%x%x%x%x%x%x%x " "lsn(%lu,0x%lx)\n", @@ -467,18 +474,19 @@ int main(int argc __attribute__((unused)), char *argv[]) LSN ref; ref= lsn_korr(rec.header); rec_len= get_len(); - if (rec.type !=LOGREC_UNDO_KEY_INSERT || + if (rec.type !=LOGREC_VARIABLE_RECORD_1LSN_EXAMPLE || rec.short_trid != (i % 0xFFFF) || rec.record_length != rec_len + LSN_STORE_SIZE || len != 12 || ref != lsn || check_content(rec.header + LSN_STORE_SIZE, len - LSN_STORE_SIZE)) { - fprintf(stderr, "Incorrect LOGREC_UNDO_KEY_INSERT data read(%d)" + fprintf(stderr, "Incorrect LOGREC_VARIABLE_RECORD_1LSN_EXAMPLE " + "data read(%d)" "type %u (%d), strid %u (%d), len %lu, %lu + 7 (%d), " "hdr len: %u (%d), " "ref(%lu,0x%lx), lsn(%lu,0x%lx) (%d), content: %d\n", i, (uint) rec.type, - rec.type !=LOGREC_UNDO_KEY_INSERT, + rec.type !=LOGREC_VARIABLE_RECORD_1LSN_EXAMPLE, (uint) rec.short_trid, rec.short_trid != (i % 0xFFFF), (ulong) rec.record_length, (ulong) rec_len, @@ -496,8 +504,8 @@ int main(int argc __attribute__((unused)), char *argv[]) if (read_and_check_content(&rec, long_buffer, LSN_STORE_SIZE)) { fprintf(stderr, - "Incorrect LOGREC_UNDO_KEY_INSERT in whole rec read " - "lsn(%lu,0x%lx)\n", + "Incorrect LOGREC_VARIABLE_RECORD_1LSN_EXAMPLE " + "in whole rec read lsn(%lu,0x%lx)\n", (ulong) LSN_FILE_NO(rec.lsn), (ulong) LSN_OFFSET(rec.lsn)); translog_free_record_header(&rec); goto err; @@ -509,7 +517,7 @@ int main(int argc __attribute__((unused)), char *argv[]) ref1= lsn_korr(rec.header); ref2= lsn_korr(rec.header + LSN_STORE_SIZE); rec_len= get_len(); - if (rec.type !=LOGREC_UNDO_KEY_DELETE || + if (rec.type != LOGREC_VARIABLE_RECORD_2LSN_EXAMPLE || rec.short_trid != (i % 0xFFFF) || rec.record_length != rec_len + LSN_STORE_SIZE * 2 || len != 19 || @@ -518,7 +526,8 @@ int main(int argc __attribute__((unused)), char *argv[]) check_content(rec.header + LSN_STORE_SIZE * 2, len - LSN_STORE_SIZE * 2)) { - fprintf(stderr, "Incorrect LOGREC_UNDO_KEY_DELETE data read(%d)" + fprintf(stderr, "Incorrect LOGREC_VARIABLE_RECORD_2LSN_EXAMPLE " + " data read(%d) " "type %u, strid %u, len %lu != %lu + 14, hdr len: %u, " "ref1(%lu,0x%lx), ref2(%lu,0x%lx), " "lsn(%lu,0x%lx)\n", @@ -534,8 +543,8 @@ int main(int argc __attribute__((unused)), char *argv[]) if (read_and_check_content(&rec, long_buffer, LSN_STORE_SIZE * 2)) { fprintf(stderr, - "Incorrect LOGREC_UNDO_KEY_DELETE in whole rec read " - "lsn(%lu,0x%lx)\n", + "Incorrect LOGREC_VARIABLE_RECORD_2LSN_EXAMPLE " + "in whole rec read lsn(%lu,0x%lx)\n", (ulong) LSN_FILE_NO(rec.lsn), (ulong) LSN_OFFSET(rec.lsn)); translog_free_record_header(&rec); goto err; @@ -559,12 +568,13 @@ int main(int argc __attribute__((unused)), char *argv[]) translog_free_record_header(&rec); goto err; } - if (rec.type !=LOGREC_LONG_TRANSACTION_ID || + if (rec.type != LOGREC_FIXED_RECORD_0LSN_EXAMPLE || rec.short_trid != (i % 0xFFFF) || rec.record_length != 6 || uint4korr(rec.header) != i || ((uchar)rec.header[4]) != 0 || ((uchar)rec.header[5]) != 0xFF) { - fprintf(stderr, "Incorrect LOGREC_LONG_TRANSACTION_ID data read(%d)\n" + fprintf(stderr, "Incorrect LOGREC_FIXED_RECORD_0LSN_EXAMPLE " + "data read(%d)\n" "type %u, strid %u, len %u, i: %u, 4: %u 5: %u " "lsn(%lu,0x%lx)\n", i, (uint) rec.type, (uint) rec.short_trid, @@ -582,12 +592,13 @@ int main(int argc __attribute__((unused)), char *argv[]) len= translog_read_next_record_header(&scanner, &rec); rec_len= get_len(); - if (rec.type !=LOGREC_REDO_INSERT_ROW_HEAD || + if (rec.type != LOGREC_VARIABLE_RECORD_0LSN_EXAMPLE || rec.short_trid != (i % 0xFFFF) || rec.record_length != rec_len || len != 9 || check_content(rec.header, len)) { - fprintf(stderr, "Incorrect LOGREC_REDO_INSERT_ROW_HEAD data read(%d)" + fprintf(stderr, "Incorrect LOGREC_VARIABLE_RECORD_0LSN_EXAMPLE " + "data read(%d) " "type %u, strid %u, len %lu != %lu, hdr len: %u, " "lsn(%lu,0x%lx)\n", i, (uint) rec.type, (uint) rec.short_trid, @@ -600,8 +611,8 @@ int main(int argc __attribute__((unused)), char *argv[]) if (read_and_check_content(&rec, long_buffer, 0)) { fprintf(stderr, - "Incorrect LOGREC_UNDO_KEY_DELETE in whole rec read " - "lsn(%lu,0x%lx)\n", + "Incorrect LOGREC_VARIABLE_RECORD_2LSN_EXAMPLE " + "in whole rec read lsn(%lu,0x%lx)\n", (ulong) LSN_FILE_NO(rec.lsn), (ulong) LSN_OFFSET(rec.lsn)); translog_free_record_header(&rec); goto err; diff --git a/storage/maria/unittest/ma_test_loghandler_multithread-t.c b/storage/maria/unittest/ma_test_loghandler_multithread-t.c index 4afd2b23074..57aef7e3ec7 100644 --- a/storage/maria/unittest/ma_test_loghandler_multithread-t.c +++ b/storage/maria/unittest/ma_test_loghandler_multithread-t.c @@ -131,11 +131,11 @@ void writer(int num) parts[TRANSLOG_INTERNAL_PARTS + 0].str= (char*)long_tr_id; parts[TRANSLOG_INTERNAL_PARTS + 0].length= 6; if (translog_write_record(&lsn, - LOGREC_LONG_TRANSACTION_ID, + LOGREC_FIXED_RECORD_0LSN_EXAMPLE, num, NULL, NULL, 6, TRANSLOG_INTERNAL_PARTS + 1, parts)) { - fprintf(stderr, "Can't write LOGREC_LONG_TRANSACTION_ID record #%lu " + fprintf(stderr, "Can't write LOGREC_FIXED_RECORD_0LSN_EXAMPLE record #%lu " "thread %i\n", (ulong) i, num); translog_destroy(); pthread_mutex_lock(&LOCK_thread_count); @@ -147,7 +147,7 @@ void writer(int num) parts[TRANSLOG_INTERNAL_PARTS + 0].str= (char*)long_buffer; parts[TRANSLOG_INTERNAL_PARTS + 0].length= len; if (translog_write_record(&lsn, - LOGREC_REDO_INSERT_ROW_HEAD, + LOGREC_VARIABLE_RECORD_0LSN_EXAMPLE, num, NULL, NULL, len, TRANSLOG_INTERNAL_PARTS + 1, parts)) @@ -283,6 +283,7 @@ int main(int argc __attribute__((unused)), translog_destroy(); exit(1); } + example_loghandler_init(); srandom(122334817L); { @@ -295,7 +296,7 @@ int main(int argc __attribute__((unused)), parts[TRANSLOG_INTERNAL_PARTS + 0].str= (char*)long_tr_id; parts[TRANSLOG_INTERNAL_PARTS + 0].length= 6; if (translog_write_record(&first_lsn, - LOGREC_LONG_TRANSACTION_ID, + LOGREC_FIXED_RECORD_0LSN_EXAMPLE, 0, NULL, NULL, 6, TRANSLOG_INTERNAL_PARTS + 1, parts)) { @@ -393,13 +394,14 @@ int main(int argc __attribute__((unused)), stage= indeces[rec.short_trid] % 2; if (stage == 0) { - if (rec.type !=LOGREC_LONG_TRANSACTION_ID || + if (rec.type !=LOGREC_FIXED_RECORD_0LSN_EXAMPLE || rec.record_length != 6 || uint2korr(rec.header) != rec.short_trid || index != uint4korr(rec.header + 2) || cmp_translog_addr(lsns1[rec.short_trid][index], rec.lsn) != 0) { - fprintf(stderr, "Incorrect LOGREC_LONG_TRANSACTION_ID data read(%d)\n" + fprintf(stderr, "Incorrect LOGREC_FIXED_RECORD_0LSN_EXAMPLE " + "data read(%d)\n" "type %u, strid %u %u, len %u, i: %u %u, " "lsn(%lu,0x%lx) (%lu,0x%lx)\n", i, (uint) rec.type, @@ -416,19 +418,21 @@ int main(int argc __attribute__((unused)), } else { - if (rec.type !=LOGREC_REDO_INSERT_ROW_HEAD || + if (rec.type != LOGREC_VARIABLE_RECORD_0LSN_EXAMPLE || len != 9 || rec.record_length != lens[rec.short_trid][index] || cmp_translog_addr(lsns2[rec.short_trid][index], rec.lsn) != 0 || check_content(rec.header, len)) { fprintf(stderr, - "Incorrect LOGREC_REDO_INSERT_ROW_HEAD data read(%d) " - " thread: %d, iteration %d, stage %d\n" + "Incorrect LOGREC_VARIABLE_RECORD_0LSN_EXAMPLE " + "data read(%d) " + "thread: %d, iteration %d, stage %d\n" "type %u (%d), len %u, length %lu %lu (%d) " "lsn(%lu,0x%lx) (%lu,0x%lx)\n", i, (uint) rec.short_trid, index, stage, - (uint) rec.type, (rec.type !=LOGREC_REDO_INSERT_ROW_HEAD), + (uint) rec.type, (rec.type != + LOGREC_VARIABLE_RECORD_0LSN_EXAMPLE), (uint) len, (ulong) rec.record_length, lens[rec.short_trid][index], (rec.record_length != lens[rec.short_trid][index]), @@ -442,8 +446,8 @@ int main(int argc __attribute__((unused)), if (read_and_check_content(&rec, long_buffer, 0)) { fprintf(stderr, - "Incorrect LOGREC_REDO_INSERT_ROW_HEAD in whole rec read " - "lsn(%lu,0x%lx)\n", + "Incorrect LOGREC_VARIABLE_RECORD_0LSN_EXAMPLE " + "in whole rec read lsn(%lu,0x%lx)\n", (ulong) LSN_FILE_NO(rec.lsn), (ulong) LSN_OFFSET(rec.lsn)); translog_free_record_header(&rec); diff --git a/storage/maria/unittest/ma_test_loghandler_pagecache-t.c b/storage/maria/unittest/ma_test_loghandler_pagecache-t.c index a56f3f875c6..e378ee3acf0 100644 --- a/storage/maria/unittest/ma_test_loghandler_pagecache-t.c +++ b/storage/maria/unittest/ma_test_loghandler_pagecache-t.c @@ -72,6 +72,7 @@ int main(int argc __attribute__((unused)), char *argv[]) translog_destroy(); exit(1); } + example_loghandler_init(); if ((stat= my_stat(first_translog_file, &st, MYF(0))) == 0) { @@ -89,7 +90,7 @@ int main(int argc __attribute__((unused)), char *argv[]) parts[TRANSLOG_INTERNAL_PARTS + 0].str= (char*)long_tr_id; parts[TRANSLOG_INTERNAL_PARTS + 0].length= 6; if (translog_write_record(&lsn, - LOGREC_LONG_TRANSACTION_ID, + LOGREC_FIXED_RECORD_0LSN_EXAMPLE, 0, NULL, NULL, 6, TRANSLOG_INTERNAL_PARTS + 1, parts)) { -- cgit v1.2.1 From 9f903637d98deb20ed12f35321178e205de9a605 Mon Sep 17 00:00:00 2001 From: unknown Date: Fri, 15 Jun 2007 08:31:12 +0300 Subject: Postmerge changes storage/maria/unittest/ma_test_loghandler-t.c: Spaces at the end of the line removed. Parameters of translog_write_record() fixed. storage/maria/unittest/ma_test_loghandler_multigroup-t.c: Parameters of translog_write_record() fixed. --- storage/maria/unittest/ma_test_loghandler-t.c | 5 ++--- storage/maria/unittest/ma_test_loghandler_multigroup-t.c | 2 +- 2 files changed, 3 insertions(+), 4 deletions(-) diff --git a/storage/maria/unittest/ma_test_loghandler-t.c b/storage/maria/unittest/ma_test_loghandler-t.c index db18ecc45d5..f05d58a784f 100644 --- a/storage/maria/unittest/ma_test_loghandler-t.c +++ b/storage/maria/unittest/ma_test_loghandler-t.c @@ -208,7 +208,7 @@ int main(int argc __attribute__((unused)), char *argv[]) for (i= 1; i < ITERATIONS; i++) { - trn->short_id= i % 0xFFFF; + trn->short_id= i % 0xFFFF; if (i % 2) { lsn_store(lsn_buff, lsn_base); @@ -218,8 +218,7 @@ int main(int argc __attribute__((unused)), char *argv[]) parts[TRANSLOG_INTERNAL_PARTS + 1].str= NULL; parts[TRANSLOG_INTERNAL_PARTS + 1].length= 0; if (translog_write_record(&lsn, LOGREC_FIXED_RECORD_1LSN_EXAMPLE, - trn, NULL, - NULL, LSN_STORE_SIZE, 0, parts)) + trn, NULL, LSN_STORE_SIZE, 0, parts)) { fprintf(stderr, "1 Can't write reference defore record #%lu\n", (ulong) i); diff --git a/storage/maria/unittest/ma_test_loghandler_multigroup-t.c b/storage/maria/unittest/ma_test_loghandler_multigroup-t.c index c7ab14bb0b7..9ed57da8fec 100644 --- a/storage/maria/unittest/ma_test_loghandler_multigroup-t.c +++ b/storage/maria/unittest/ma_test_loghandler_multigroup-t.c @@ -191,7 +191,7 @@ int main(int argc __attribute__((unused)), char *argv[]) parts[TRANSLOG_INTERNAL_PARTS + 0].length= 6; trn->short_id= 0; if (translog_write_record(&lsn, LOGREC_FIXED_RECORD_0LSN_EXAMPLE, - trn, NULL, NULL, + trn, NULL, 6, TRANSLOG_INTERNAL_PARTS + 1, parts)) { fprintf(stderr, "Can't write record #%lu\n", (ulong) 0); -- cgit v1.2.1 From fd9bd5802932b08da7484c54445ae14ee4e25385 Mon Sep 17 00:00:00 2001 From: unknown Date: Tue, 19 Jun 2007 16:01:04 +0200 Subject: Monty's fix for Maria's table scan sometimes not seeing all rows; without the fix, only 896 rows were inserted into t2 in maria-big.test. storage/maria/ma_blockrec.c: due to wrong test we were skipping some rows when scanning mysql-test/r/maria-big.result: result for new test mysql-test/t/maria-big.test: test for a bug where we missed some rows when scanning --- mysql-test/r/maria-big.result | 61 +++++++++++++++++++++++++++++++++++++++++++ mysql-test/t/maria-big.test | 26 ++++++++++++++++++ storage/maria/ma_blockrec.c | 3 ++- 3 files changed, 89 insertions(+), 1 deletion(-) create mode 100644 mysql-test/r/maria-big.result create mode 100644 mysql-test/t/maria-big.test diff --git a/mysql-test/r/maria-big.result b/mysql-test/r/maria-big.result new file mode 100644 index 00000000000..ca18d992187 --- /dev/null +++ b/mysql-test/r/maria-big.result @@ -0,0 +1,61 @@ +set storage_engine=maria; +affected rows: 0 +drop table if exists t1, t2; +affected rows: 0 +create table t1(a char(3)); +affected rows: 0 +insert into t1 values("abc"); +affected rows: 1 +insert into t1 select "def" from t1; +affected rows: 1 +info: Records: 1 Duplicates: 0 Warnings: 0 +insert into t1 select "ghi" from t1; +affected rows: 2 +info: Records: 2 Duplicates: 0 Warnings: 0 +insert into t1 select "jkl" from t1; +affected rows: 4 +info: Records: 4 Duplicates: 0 Warnings: 0 +insert into t1 select "mno" from t1; +affected rows: 8 +info: Records: 8 Duplicates: 0 Warnings: 0 +insert into t1 select "pqr" from t1; +affected rows: 16 +info: Records: 16 Duplicates: 0 Warnings: 0 +insert into t1 select "stu" from t1; +affected rows: 32 +info: Records: 32 Duplicates: 0 Warnings: 0 +insert into t1 select "vwx" from t1; +affected rows: 64 +info: Records: 64 Duplicates: 0 Warnings: 0 +insert into t1 select "yza" from t1; +affected rows: 128 +info: Records: 128 Duplicates: 0 Warnings: 0 +insert into t1 select "ceg" from t1; +affected rows: 256 +info: Records: 256 Duplicates: 0 Warnings: 0 +insert into t1 select "ikm" from t1; +affected rows: 512 +info: Records: 512 Duplicates: 0 Warnings: 0 +insert into t1 select "oqs" from t1; +affected rows: 1024 +info: Records: 1024 Duplicates: 0 Warnings: 0 +select count(*) from t1; +count(*) +2048 +affected rows: 1 +insert into t1 select "uwy" from t1; +affected rows: 2048 +info: Records: 2048 Duplicates: 0 Warnings: 0 +create table t2 select * from t1; +affected rows: 4096 +info: Records: 4096 Duplicates: 0 Warnings: 0 +select count(*) from t1; +count(*) +4096 +affected rows: 1 +select count(*) from t2; +count(*) +4096 +affected rows: 1 +drop table t1, t2; +affected rows: 0 diff --git a/mysql-test/t/maria-big.test b/mysql-test/t/maria-big.test new file mode 100644 index 00000000000..f68531c3894 --- /dev/null +++ b/mysql-test/t/maria-big.test @@ -0,0 +1,26 @@ +# Test of scenarios potentially too big for --valgrind or --mem +enable_info; +set storage_engine=maria; +disable_warnings; +drop table if exists t1, t2; +enable_warnings; +create table t1(a char(3)); +insert into t1 values("abc"); +insert into t1 select "def" from t1; +insert into t1 select "ghi" from t1; +insert into t1 select "jkl" from t1; +insert into t1 select "mno" from t1; +insert into t1 select "pqr" from t1; +insert into t1 select "stu" from t1; +insert into t1 select "vwx" from t1; +insert into t1 select "yza" from t1; +insert into t1 select "ceg" from t1; +insert into t1 select "ikm" from t1; +insert into t1 select "oqs" from t1; +select count(*) from t1; +insert into t1 select "uwy" from t1; +create table t2 select * from t1; +select count(*) from t1; +select count(*) from t2; +drop table t1, t2; +disable_info; diff --git a/storage/maria/ma_blockrec.c b/storage/maria/ma_blockrec.c index 800ed8f14ac..39769507887 100644 --- a/storage/maria/ma_blockrec.c +++ b/storage/maria/ma_blockrec.c @@ -3454,7 +3454,8 @@ restart_bitmap_scan: for (data+= 6; data < info->scan.bitmap_end; data+= 6) { bits= uint6korr(data); - if (bits && ((bits & LL(04444444444444444)) != LL(04444444444444444))) + /* Skip not allocated pages and blob / full tail pages */ + if (bits && bits != LL(07777777777777777)) break; } bit_pos= 0; -- cgit v1.2.1 From 1a96259191b193b353387cbb70d7567009e3b247 Mon Sep 17 00:00:00 2001 From: unknown Date: Fri, 22 Jun 2007 14:49:37 +0200 Subject: - WL#3239 "log CREATE TABLE in Maria" - WL#3240 "log DROP TABLE in Maria" - similarly, log RENAME TABLE, REPAIR/OPTIMIZE TABLE, and DELETE no_WHERE_clause (== the DELETE which just truncates the files) - create_rename_lsn added to MARIA_SHARE's state - all these operations (except DROP TABLE) also update the table's create_rename_lsn, which is needed for the correctness of Recovery (see function comment of _ma_repair_write_log_record() in ma_check.c) - write a COMMIT record when transaction commits. - don't log REDOs/UNDOs if this is an internal temporary table like inside ALTER TABLE (I expect this to be a big win). There was already no logging for user-created "CREATE TEMPORARY" tables. - don't fsync files/directories if the table is not transactional - in translog_write_record(), autogenerate a 2-byte-id for the table and log the "id->name" pair (LOGREC_FILE_ID); log LOGREC_LONG_TRANSACTION_ID; automatically store the table's 2-byte-id in any log record. - preparations for Checkpoint: translog_get_horizon(); pausing Checkpoint when some dirty pages are unknown; capturing trn->rec_lsn, trn->first_undo_lsn for Checkpoint and log's low-water-mark computing. - assertions, comments. storage/maria/Makefile.am: more files to build storage/maria/ha_maria.cc: - logging a REPAIR log record if REPAIR/OPTIMIZE was successful. - ha_maria::data_file_type does not have to be set in every info() call, just do it once in open(). - if caller said that transactionality can be disabled (like if caller is ALTER TABLE) i.e. thd->transaction.on==FALSE, then we temporarily disable transactionality of the table in external_lock(); that will ensure that no REDOs/UNDOs are logged for this possibly massive write operation (they are not needed, as if any write fails, the table will be dropped). We re-enable in external_lock(F_UNLCK), which in ALTER TABLE happens before the tmp table replaces the original one (which is good, as thus the final table will have a REDO RENAME and a correct create_rename_lsn). - when we commit we also have to write a log record, so trnman_commit_trn() calls become ma_commit() calls - at end of engine's initialization, we are potentially entering a multi-threaded dangerous world (clients are going to be accepted) and so some assertions of mutex-owning become enforceable, for that we set maria_multi_threaded=TRUE (see ma_control_file.c) storage/maria/ha_maria.h: new member ha_maria::save_transactional (see also ha_maria.cc) storage/maria/ma_blockrec.c: - fixing comments according to discussion with Monty - if a table is transactional but temporarily non-transactional (like in ALTER TABLE), we need to give a sensible LSN to the pages (and, if we give 0, pagecache asserts). - translog_write_record() now takes care of storing the share's 2-byte-id in the log record storage/maria/ma_blockrec.h: fixing comment according to discussion with Monty storage/maria/ma_check.c: When REPAIR/OPTIMIZE modify the data/index file, if this is a transactional table, they must sync it; if they remove files or rename files, they must sync the directory, so that everything is durable. This is just applying to REPAIR/OPTIMIZE the logic already implemented in CREATE/DROP/RENAME a few months ago. Adding a function to write a LOGREC_REPAIR_TABLE at end of REPAIR/OPTIMIZE (called only by ha_maria, not by maria_chk), and to update the table's create_rename_lsn. storage/maria/ma_close.c: fix for a future bug storage/maria/ma_control_file.c: ensuring that if Maria is running in multi-threaded mode, anybody wanting to write to the control file and update last_checkpoint_lsn/last_logno owns the log's lock. storage/maria/ma_control_file.h: see ma_control_file.c storage/maria/ma_create.c: when creating a table: - sync it and its directory only if this is a transactional table and there is a log (no point in syncing in maria_chk) - decouple the two uses of linkname/linkname_ptr (for index file and for data file) into more variables, as we need to know all links until the moment we write the LOGREC_CREATE_TABLE. - set share.data_file_type early so that _ma_initialize_data_file() knows it (Monty's bugfix so that a table always has at least a bitmap page when it is created; so data-file is not 0 bytes anymore). - log a LOGREC_CREATE_TABLE; it contains the bytes which we have just written to the index file's header. Update table's create_rename_lsn. - syncing of kfile had been bugified in a previous merge, correcting - syncing of dfile is now needed as it's not empty anymore - in _ma_initialize_data_file(), use share's block_size and not the global one. This is a gratuitous change, both variables are equal, just that I find it more future-proof to use share-bound variable rather than global one. storage/maria/ma_delete_all.c: log a LOGREC_DELETE_ALL record when doing ma_delete_all_rows(); update create_rename_lsn then. storage/maria/ma_delete_table.c: - logging LOGREC_DROP_TABLE; knowing if this is needed, requires knowing if the table is transactional, which requires opening the table. - we need to sync directories only if the table is transactional storage/maria/ma_extra.c: questions storage/maria/ma_init.c: when maria_end() is called, engine is not multithreaded storage/maria/ma_loghandler.c: - translog_inited has to be visible to ma_create() (see how it is used in ma_create()) - checkpoint record will be a single record, not three - no REDO for TRUNCATE (TRUNCATE calls ma_create() internally so will log a REDO_CREATE) - adding REDO for DELETE no_WHERE_clause (fast DELETE of all rows by truncating the files), REPAIR. - MY_WAIT_IF_FULL to wait&retry if a log write hits a full disk - in translog_write_record(), if MARIA_SHARE does not yet have a 2-byte-id, generate one for it and log LOGREC_FILE_ID; automatically store this short id into log records. - in translog_write_record(), if transaction has not logged its long trid, log LOGREC_LONG_TRANSACTION_ID. - For Checkpoint, we need to know the current end-of-log: adding translog_get_horizon(). - For Control File, adding an assertion that the thread owns the log's lock (control file is protected by this lock) storage/maria/ma_loghandler.h: Changes in log records (see ma_loghandler.c). new prototypes, new functions. storage/maria/ma_loghandler_lsn.h: adding a type LSN_WITH_FLAGS especially for TRN::first_undo_lsn, where the most significant byte is used for flags. storage/maria/ma_open.c: storing the create_rename_lsn in the index file's header (in the state, precisely) and retrieving it from there. storage/maria/ma_pagecache.c: - my set_if_bigger was wrong, correcting it - if the first_in_switch list is not empty, it means that changed_blocks misses some dirty pages, so Checkpoint cannot run and needs to wait. A variable missing_blocks_in_changed_list is added to tell that (should it be named missing_blocks_in_changed_blocks?) - pagecache_collect_changed_blocks_with_lsn() now also tells the minimum rec_lsn (needed for low-water mark computation). storage/maria/ma_pagecache.h: see ma_pagecache.c storage/maria/ma_panic.c: comment storage/maria/ma_range.c: comment storage/maria/ma_rename.c: - logging LOGREC_RENAME_TABLE; knowing if this is needed, requires knowing if the table is transactional, which requires opening the table. - update create_rename_lsn - we need to sync directories only if the table is transactional storage/maria/ma_static.c: comment storage/maria/ma_test_all.sh: - tip for Valgrind-ing ma_test_all - do "export maria_path=somepath" before calling ma_test_all, if you want to run ma_test_all out of storage/maria (useful to have parallel runs, like one normal and one Valgrind, they must not use the same tables so need to run in different directories) storage/maria/maria_def.h: - state now contains, in memory and on disk, the create_rename_lsn - share now contains a 2-byte-id storage/maria/trnman.c: preparations for Checkpoint: capture trn->rec_lsn, trn->first_undo_lsn; minimum first_undo_lsn needed to know log's low-water-mark storage/maria/trnman.h: using most significant byte of first_undo_lsn to hold miscellaneous flags, for now TRANSACTION_LOGGED_LONG_ID. dummy_transaction_object is already declared in ma_static.c. storage/maria/trnman_public.h: dummy_transaction_object was declared in all files including trnman_public.h, while in fact it's a single object. new prototype storage/maria/unittest/ma_test_loghandler-t.c: update for new prototype storage/maria/unittest/ma_test_loghandler_multigroup-t.c: update for new prototype storage/maria/unittest/ma_test_loghandler_multithread-t.c: update for new prototype storage/maria/unittest/ma_test_loghandler_pagecache-t.c: update for new prototype storage/maria/ma_commit.c: function which wraps: - writing a LOGREC_COMMIT record (==commit on disk) - calling trnman_commit_trn() (=commit in memory) storage/maria/ma_commit.h: new header file .tree-is-private: this file is now needed to keep our tree private (don't push it to public trees). When 5.1 is merged into mysql-maria, we can abandon our maria-specific post-commit trigger; .tree_is_private will take care of keeping commit mails private. Don't push this file to public trees. --- .tree-is-private | 0 storage/maria/Makefile.am | 6 +- storage/maria/ha_maria.cc | 50 ++- storage/maria/ha_maria.h | 5 + storage/maria/ma_blockrec.c | 140 ++++---- storage/maria/ma_blockrec.h | 2 +- storage/maria/ma_check.c | 88 ++++- storage/maria/ma_close.c | 15 +- storage/maria/ma_commit.c | 71 +++++ storage/maria/ma_commit.h | 18 ++ storage/maria/ma_control_file.c | 16 +- storage/maria/ma_control_file.h | 2 + storage/maria/ma_create.c | 148 ++++++--- storage/maria/ma_delete_all.c | 79 +++-- storage/maria/ma_delete_table.c | 99 ++++-- storage/maria/ma_extra.c | 48 ++- storage/maria/ma_init.c | 2 +- storage/maria/ma_loghandler.c | 355 ++++++++++++++++----- storage/maria/ma_loghandler.h | 35 +- storage/maria/ma_loghandler_lsn.h | 10 +- storage/maria/ma_open.c | 22 +- storage/maria/ma_pagecache.c | 179 ++++++----- storage/maria/ma_pagecache.h | 1 + storage/maria/ma_panic.c | 7 +- storage/maria/ma_range.c | 32 +- storage/maria/ma_rename.c | 96 ++++-- storage/maria/ma_static.c | 8 +- storage/maria/ma_test_all.sh | 278 ++++++++-------- storage/maria/maria_def.h | 6 +- storage/maria/trnman.c | 173 +++++++--- storage/maria/trnman.h | 5 +- storage/maria/trnman_public.h | 7 +- storage/maria/unittest/ma_test_loghandler-t.c | 14 +- .../unittest/ma_test_loghandler_multigroup-t.c | 14 +- .../unittest/ma_test_loghandler_multithread-t.c | 6 +- .../unittest/ma_test_loghandler_pagecache-t.c | 2 +- 36 files changed, 1407 insertions(+), 632 deletions(-) create mode 100644 .tree-is-private create mode 100644 storage/maria/ma_commit.c create mode 100644 storage/maria/ma_commit.h diff --git a/.tree-is-private b/.tree-is-private new file mode 100644 index 00000000000..e69de29bb2d diff --git a/storage/maria/Makefile.am b/storage/maria/Makefile.am index 9d8ab704541..fbb25584910 100644 --- a/storage/maria/Makefile.am +++ b/storage/maria/Makefile.am @@ -54,7 +54,8 @@ noinst_HEADERS = maria_def.h ma_rt_index.h ma_rt_key.h ma_rt_mbr.h \ ma_sp_defs.h ma_fulltext.h ma_ftdefs.h ma_ft_test1.h \ ma_ft_eval.h trnman.h lockman.h tablockman.h \ ma_control_file.h ha_maria.h ma_blockrec.h \ - ma_loghandler.h ma_loghandler_lsn.h ma_pagecache.h + ma_loghandler.h ma_loghandler_lsn.h ma_pagecache.h \ + ma_commit.h ma_test1_DEPENDENCIES= $(LIBRARIES) ma_test1_LDADD= @CLIENT_EXTRA_LDFLAGS@ libmaria.a \ $(top_builddir)/storage/myisam/libmyisam.a \ @@ -112,7 +113,8 @@ libmaria_a_SOURCES = ma_init.c ma_open.c ma_extra.c ma_info.c ma_rkey.c \ ha_maria.cc trnman.c lockman.c tablockman.c \ ma_rt_index.c ma_rt_key.c ma_rt_mbr.c ma_rt_split.c \ ma_sp_key.c ma_control_file.c ma_loghandler.c \ - ma_pagecache.c ma_pagecaches.c + ma_pagecache.c ma_pagecaches.c \ + ma_commit.c CLEANFILES = test?.MA? FT?.MA? isam.log ma_test_all ma_rt_test.MA? sp_test.MA? SUFFIXES = .sh diff --git a/storage/maria/ha_maria.cc b/storage/maria/ha_maria.cc index 288366675a7..e05f97a384d 100644 --- a/storage/maria/ha_maria.cc +++ b/storage/maria/ha_maria.cc @@ -30,6 +30,7 @@ #include "maria_def.h" #include "ma_rt_index.h" #include "ma_blockrec.h" +#include "ma_commit.h" #define MARIA_CANNOT_ROLLBACK HA_NO_TRANSACTIONS #ifdef MARIA_CANNOT_ROLLBACK @@ -690,7 +691,8 @@ int ha_maria::open(const char *name, int mode, uint test_if_locked) info(HA_STATUS_NO_LOCK | HA_STATUS_VARIABLE | HA_STATUS_CONST); if (!(test_if_locked & HA_OPEN_WAIT_IF_LOCKED)) VOID(maria_extra(file, HA_EXTRA_WAIT_LOCK, 0)); - if (file->s->data_file_type != STATIC_RECORD) + save_transactional= file->s->base.transactional; + if ((data_file_type= file->s->data_file_type) != STATIC_RECORD) int_table_flags |= HA_REC_NOT_IN_SEQ; if (file->s->options & (HA_OPTION_CHECKSUM | HA_OPTION_COMPRESS_RECORD)) int_table_flags |= HA_HAS_CHECKSUM; @@ -1178,6 +1180,8 @@ int ha_maria::repair(THD *thd, HA_CHECK ¶m, bool do_optimize) llstr(rows, llbuff), llstr(file->state->records, llbuff2)); } + if (!error) + error= _ma_repair_write_log_record(¶m, file); } else { @@ -1806,7 +1810,6 @@ int ha_maria::info(uint flag) MY_APPEND_EXT | MY_UNPACK_FILENAME); if (strcmp(name_buff, maria_info.index_file_name)) index_file_name=maria_info.index_file_name; - data_file_type= maria_info.data_file_type; } if (flag & HA_STATUS_ERRKEY) { @@ -1860,7 +1863,7 @@ int ha_maria::external_lock(THD *thd, int lock_type) { TRN *trn= THD_TRN; DBUG_ENTER("ha_maria::external_lock"); - if (!file->s->base.transactional) + if (!save_transactional) goto skip_transaction; if (!trn && lock_type != F_UNLCK) /* no transaction yet - open it now */ { @@ -1884,6 +1887,19 @@ int ha_maria::external_lock(THD *thd, int lock_type) trans_register_ha(thd, FALSE, maria_hton); trnman_new_statement(trn); } + if (!thd->transaction.on) + { + /* + No need to log REDOs/UNDOs. If this is an internal temporary table + which will be renamed to a permanent table (like in ALTER TABLE), + the rename happens after unlocking so will be durable (and the table + will get its create_rename_lsn). + Note: if we wanted to enable users to have an old backup and apply + tons of archived logs to roll-forward, we could then not disable + REDOs/UNDOs in this case. + */ + file->s->base.transactional= FALSE; + } } else { @@ -1894,7 +1910,8 @@ int ha_maria::external_lock(THD *thd, int lock_type) { /* autocommit ? rollback a transaction */ #ifdef MARIA_CANNOT_ROLLBACK - trnman_commit_trn(trn); + if (ma_commit(trn)) + DBUG_RETURN(1); THD_TRN= 0; #else if (!(thd->options & (OPTION_NOT_AUTOCOMMIT | OPTION_BEGIN))) @@ -1906,6 +1923,7 @@ int ha_maria::external_lock(THD *thd, int lock_type) #endif } } + file->s->base.transactional= save_transactional; } skip_transaction: DBUG_RETURN(maria_lock_database(file, !table->s->tmp_table ? @@ -1916,7 +1934,7 @@ skip_transaction: int ha_maria::start_stmt(THD *thd, thr_lock_type lock_type) { TRN *trn= THD_TRN; - if (file->s->base.transactional) + if (save_transactional) { DBUG_ASSERT(trn); // this may be called only after external_lock() DBUG_ASSERT(trnman_has_locked_tables(trn)); @@ -2186,8 +2204,7 @@ static int maria_commit(handlerton *hton __attribute__ ((unused)), DBUG_RETURN(0); // end of statement DBUG_PRINT("info", ("THD_TRN set to 0x0")); THD_TRN= 0; - DBUG_RETURN(trnman_commit_trn(trn) ? - HA_ERR_OUT_OF_MEM : 0); // end of transaction + DBUG_RETURN(ma_commit(trn)); // end of transaction } @@ -2212,6 +2229,7 @@ static int maria_rollback(handlerton *hton __attribute__ ((unused)), static int ha_maria_init(void *p) { + int res; maria_hton= (handlerton *)p; maria_hton->state= SHOW_OPTION_YES; maria_hton->db_type= DB_TYPE_MARIA; @@ -2223,14 +2241,16 @@ static int ha_maria_init(void *p) maria_hton->flags= HTON_CAN_RECREATE | HTON_SUPPORT_LOG_TABLES; bzero(maria_log_pagecache, sizeof(*maria_log_pagecache)); maria_data_root= mysql_real_data_home; - return (test(maria_init() || ma_control_file_create_or_open() || - (init_pagecache(maria_log_pagecache, - TRANSLOG_PAGECACHE_SIZE, 0, 0, - TRANSLOG_PAGE_SIZE) == 0) || - translog_init(maria_data_root, TRANSLOG_FILE_SIZE, - MYSQL_VERSION_ID, server_id, maria_log_pagecache, - TRANSLOG_DEFAULT_FLAGS) || - trnman_init())); + res= maria_init() || ma_control_file_create_or_open() || + (init_pagecache(maria_log_pagecache, + TRANSLOG_PAGECACHE_SIZE, 0, 0, + TRANSLOG_PAGE_SIZE) == 0) || + translog_init(maria_data_root, TRANSLOG_FILE_SIZE, + MYSQL_VERSION_ID, server_id, maria_log_pagecache, + TRANSLOG_DEFAULT_FLAGS) || + trnman_init(); + maria_multi_threaded= TRUE; + return res; } diff --git a/storage/maria/ha_maria.h b/storage/maria/ha_maria.h index dd0a9594ef3..a2f6b190657 100644 --- a/storage/maria/ha_maria.h +++ b/storage/maria/ha_maria.h @@ -39,6 +39,11 @@ class ha_maria :public handler char *data_file_name, *index_file_name; enum data_file_type data_file_type; bool can_enable_indexes; + /** + @brief for temporarily disabling table's transactionality + (if THD::transaction::on is false), remember the original value here + */ + bool save_transactional; int repair(THD * thd, HA_CHECK ¶m, bool optimize); public: diff --git a/storage/maria/ma_blockrec.c b/storage/maria/ma_blockrec.c index 39769507887..d2512f1e025 100644 --- a/storage/maria/ma_blockrec.c +++ b/storage/maria/ma_blockrec.c @@ -171,11 +171,14 @@ started and we can then delete TRANSID and VER_PTR from the row to gain more space. - If a row is deleted in Maria, we change TRANSID to current transid and - change VER_PTR to point to the undo record for the delete. The undo - record must contain the original TRANSID, so that another transaction - can use this to check if they should use the found row or go to the - previous row pointed to by the VER_PTR in the undo row. + If a row is deleted in Maria, we change TRANSID to the deleting + transaction's id, change VER_PTR to point to the undo record for the delete, + and add DELETE_TRANSID (the id of the transaction which last + inserted/updated the row before its deletion). DELETE_TRANSID allows an old + transaction to avoid reading the log to know if it can see the last version + before delete (in other words it reduces the probability of having to follow + VER_PTR). TODO: depending on a compilation option, evaluate the performance + impact of not storing DELETE_TRANSID (which would make the row smaller). Description of the different parts: @@ -391,7 +394,12 @@ my_bool _ma_once_end_block_record(MARIA_SHARE *share) share->temporary ? FLUSH_IGNORE_CHANGED : FLUSH_RELEASE)) res= 1; - if (my_close(share->bitmap.file.file, MYF(MY_WME))) + /* + File must be synced as it is going out of the maria_open_list and so + becoming unknown to Checkpoint. + */ + if (my_sync(share->bitmap.file.file, MYF(MY_WME)) || + my_close(share->bitmap.file.file, MYF(MY_WME))) res= 1; /* Trivial assignment to guard against multiple invocations @@ -400,6 +408,8 @@ my_bool _ma_once_end_block_record(MARIA_SHARE *share) */ share->bitmap.file.file= -1; } + if (share->id != 0) + translog_deassign_id_from_share(share); return res; } @@ -573,7 +583,14 @@ void _ma_unpin_all_pages(MARIA_HA *info, LSN undo_lsn) DBUG_ASSERT(undo_lsn != 0 || !info->s->base.transactional); if (!info->s->base.transactional) - undo_lsn= 0; /* Avoid assert in key cache */ + { + /* + If this is a transactional table but with transactionality temporarily + disabled (like in ALTER TABLE) we need to give a sensible LSN to pages + and not 0. If this is not a transactional table it will reduce to 0. + */ + undo_lsn= info->s->state.create_rename_lsn; + } while (pinned_page-- != page_link) pagecache_unlock_by_link(info->s->pagecache, pinned_page->link, @@ -1133,7 +1150,6 @@ static my_bool write_tail(MARIA_HA *info, LSN lsn; /* Log REDO changes of tail page */ - fileid_store(log_data, info->dfile.file); page_store(log_data+ FILEID_STORE_SIZE, block->page); dirpos_store(log_data+ FILEID_STORE_SIZE + PAGE_STORE_SIZE, row_pos.rownr); @@ -1143,7 +1159,8 @@ static my_bool write_tail(MARIA_HA *info, log_array[TRANSLOG_INTERNAL_PARTS + 1].length= length; if (translog_write_record(&lsn, LOGREC_REDO_INSERT_ROW_TAIL, info->trn, share, sizeof(log_data) + length, - TRANSLOG_INTERNAL_PARTS + 2, log_array)) + TRANSLOG_INTERNAL_PARTS + 2, log_array, + log_data)) DBUG_RETURN(1); } @@ -1388,7 +1405,6 @@ static my_bool free_full_pages(MARIA_HA *info, MARIA_ROW *row) size_t extents_length= row->extents_count * ROW_EXTENT_SIZE; DBUG_ENTER("free_full_pages"); - fileid_store(log_data, info->dfile.file); pagerange_store(log_data + FILEID_STORE_SIZE, row->extents_count); log_array[TRANSLOG_INTERNAL_PARTS + 0].str= (char*) log_data; @@ -1397,7 +1413,8 @@ static my_bool free_full_pages(MARIA_HA *info, MARIA_ROW *row) log_array[TRANSLOG_INTERNAL_PARTS + 1].length= extents_length; if (translog_write_record(&lsn, LOGREC_REDO_PURGE_BLOCKS, info->trn, info->s, sizeof(log_data) + extents_length, - TRANSLOG_INTERNAL_PARTS + 2, log_array)) + TRANSLOG_INTERNAL_PARTS + 2, log_array, + log_data)) DBUG_RETURN(1); DBUG_RETURN (_ma_bitmap_free_full_pages(info, row->extents, @@ -1431,7 +1448,6 @@ static my_bool free_full_page_range(MARIA_HA *info, ulonglong page, uint count) { LSN lsn; DBUG_ASSERT(info->trn->rec_lsn); - fileid_store(log_data, info->dfile.file); pagerange_store(log_data + FILEID_STORE_SIZE, 1); int5store(log_data + FILEID_STORE_SIZE + PAGERANGE_STORE_SIZE, page); @@ -1442,7 +1458,8 @@ static my_bool free_full_page_range(MARIA_HA *info, ulonglong page, uint count) if (translog_write_record(&lsn, LOGREC_REDO_PURGE_BLOCKS, info->trn, info->s, sizeof(log_data), - TRANSLOG_INTERNAL_PARTS + 1, log_array)) + TRANSLOG_INTERNAL_PARTS + 1, log_array, + log_data)) res= 1; } @@ -1455,24 +1472,25 @@ static my_bool free_full_page_range(MARIA_HA *info, ulonglong page, uint count) } -/* - Write a record to a (set of) pages +/** + @brief Write a record to a (set of) pages - SYNOPSIS - write_block_record() - info Maria handler - old_record Orignal record in case of update; NULL in case of insert - record Record we should write - row Statistics about record (calculated by calc_record_size()) - map_blocks On which pages the record should be stored - row_pos Position on head page where to put head part of record + @param info Maria handler + @param old_record Original record in case of update; NULL in case of + insert + @param record Record we should write + @param row Statistics about record (calculated by + calc_record_size()) + @param map_blocks On which pages the record should be stored + @param row_pos Position on head page where to put head part of + record - NOTES - On return all pinned pages are released. + @note + On return all pinned pages are released. - RETURN - 0 ok - 1 error + @return Operation status + @retval 0 OK + @retval 1 Error */ static my_bool write_block_record(MARIA_HA *info, @@ -1940,7 +1958,6 @@ static my_bool write_block_record(MARIA_HA *info, size_t data_length= (size_t) (data - row_pos->data); /* Log REDO changes of head page */ - fileid_store(log_data, info->dfile.file); page_store(log_data+ FILEID_STORE_SIZE, head_block->page); dirpos_store(log_data+ FILEID_STORE_SIZE + PAGE_STORE_SIZE, row_pos->rownr); @@ -1950,7 +1967,8 @@ static my_bool write_block_record(MARIA_HA *info, log_array[TRANSLOG_INTERNAL_PARTS + 1].length= data_length; if (translog_write_record(&lsn, LOGREC_REDO_INSERT_ROW_HEAD, info->trn, share, sizeof(log_data) + data_length, - TRANSLOG_INTERNAL_PARTS + 2, log_array)) + TRANSLOG_INTERNAL_PARTS + 2, log_array, + log_data)) goto disk_err; } @@ -2010,7 +2028,6 @@ static my_bool write_block_record(MARIA_HA *info, NullS)) goto disk_err; } - fileid_store(log_data, info->dfile.file); log_pos= log_data + FILEID_STORE_SIZE; log_array_pos= log_array+ TRANSLOG_INTERNAL_PARTS+1; @@ -2068,7 +2085,7 @@ static my_bool write_block_record(MARIA_HA *info, error= translog_write_record(&lsn, LOGREC_REDO_INSERT_ROW_BLOBS, info->trn, share, log_entry_length, (uint) (log_array_pos - log_array), - log_array); + log_array, log_data); if (log_array != tmp_log_array) my_free((gptr) log_array, MYF(0)); if (error) @@ -2084,7 +2101,6 @@ static my_bool write_block_record(MARIA_HA *info, /* LOGREC_UNDO_ROW_INSERT & LOGREC_UNDO_ROW_INSERT share same header */ lsn_store(log_data, info->trn->undo_lsn); - fileid_store(log_data + LSN_STORE_SIZE, info->dfile.file); page_store(log_data+ LSN_STORE_SIZE + FILEID_STORE_SIZE, head_block->page); dirpos_store(log_data+ LSN_STORE_SIZE + FILEID_STORE_SIZE + @@ -2099,7 +2115,8 @@ static my_bool write_block_record(MARIA_HA *info, /* Write UNDO log record for the INSERT */ if (translog_write_record(&lsn, LOGREC_UNDO_ROW_INSERT, info->trn, share, sizeof(log_data), - TRANSLOG_INTERNAL_PARTS + 1, log_array)) + TRANSLOG_INTERNAL_PARTS + 1, log_array, + log_data + LSN_STORE_SIZE)) goto disk_err; } else @@ -2114,7 +2131,7 @@ static my_bool write_block_record(MARIA_HA *info, if (translog_write_record(&lsn, LOGREC_UNDO_ROW_UPDATE, info->trn, share, sizeof(log_data) + row_length, TRANSLOG_INTERNAL_PARTS + 1 + row_parts_count, - log_array)) + log_array, log_data + LSN_STORE_SIZE)) goto disk_err; } } @@ -2164,6 +2181,15 @@ crashed: my_errno= HA_ERR_WRONG_IN_RECORD; disk_err: + /** + @todo RECOVERY we are going to let dirty pages go to disk while we have + logged UNDO, this violates WAL. If we have not written any full pages, + all dirty pages are pinned so we could just delete them from the + pagecache. Moreover, we have written some REDOs without a closing UNDO, + it's possible that a next operation by this transaction succeeds and then + Recovery would glue the "orphan REDOs" to the succeeded operation and + execute the failed REDOs. + */ /* Unpin all pinned pages to not cause problems for disk cache */ _ma_unpin_all_pages(info, 0); @@ -2229,20 +2255,18 @@ my_bool _ma_write_block_record(MARIA_HA *info __attribute__ ((unused)), } -/* - Remove row written by _ma_write_block_record +/** + @brief Remove row written by _ma_write_block_record() - SYNOPSIS - _ma_abort_write_block_record() - info Maria handler + @param info Maria handler - INFORMATION - This is called in case we got a duplicate unique key while - writing keys. + @note + This is called in case we got a duplicate unique key while + writing keys. - RETURN - 0 ok - 1 error + @return Operation status + @retval 0 OK + @retval 1 Error */ my_bool _ma_write_abort_block_record(MARIA_HA *info) @@ -2288,16 +2312,19 @@ my_bool _ma_write_abort_block_record(MARIA_HA *info) really undo a failed insert. Note that this UNDO will cause recover to ignore the LOGREC_UNDO_ROW_INSERT that is the previous entry in the UNDO chain. - We will soon change that: we will here execute the UNDO records - generated while we were trying to write the row; this will log some CLRs - which will replace this LOGREC_UNDO_PURGE. RECOVERY TODO BUG. + */ + /** + @todo RECOVERY BUG + We will soon change that: we will here execute the UNDO records + generated while we were trying to write the row; this will log some + CLRs which will replace this LOGREC_UNDO_PURGE. */ lsn_store(log_data, info->trn->undo_lsn); log_array[TRANSLOG_INTERNAL_PARTS + 0].str= (char*) log_data; log_array[TRANSLOG_INTERNAL_PARTS + 0].length= sizeof(log_data); if (translog_write_record(&lsn, LOGREC_UNDO_ROW_PURGE, - info->trn, info->s, sizeof(log_data), - TRANSLOG_INTERNAL_PARTS + 1, log_array)) + info->trn, NULL, sizeof(log_data), + TRANSLOG_INTERNAL_PARTS + 1, log_array, NULL)) res= 1; } _ma_unpin_all_pages(info, info->trn->undo_lsn); @@ -2514,7 +2541,6 @@ static my_bool delete_head_or_tail(MARIA_HA *info, DBUG_ASSERT(share->pagecache->block_size == block_size); /* Log REDO data */ - fileid_store(log_data, info->dfile.file); page_store(log_data+ FILEID_STORE_SIZE, page); dirpos_store(log_data+ FILEID_STORE_SIZE + PAGE_STORE_SIZE, record_number); @@ -2524,7 +2550,8 @@ static my_bool delete_head_or_tail(MARIA_HA *info, if (translog_write_record(&lsn, (head ? LOGREC_REDO_PURGE_ROW_HEAD : LOGREC_REDO_PURGE_ROW_TAIL), info->trn, share, sizeof(log_data), - TRANSLOG_INTERNAL_PARTS + 1, log_array)) + TRANSLOG_INTERNAL_PARTS + 1, log_array, + log_data)) DBUG_RETURN(1); if (pagecache_write(share->pagecache, &info->dfile, page, 0, @@ -2545,7 +2572,6 @@ static my_bool delete_head_or_tail(MARIA_HA *info, PAGE_STORE_SIZE + PAGERANGE_STORE_SIZE]; LEX_STRING log_array[TRANSLOG_INTERNAL_PARTS + 1]; - fileid_store(log_data, info->dfile.file); pagerange_store(log_data + FILEID_STORE_SIZE, 1); page_store(log_data+ FILEID_STORE_SIZE + PAGERANGE_STORE_SIZE, page); pagerange_store(log_data + FILEID_STORE_SIZE + PAGERANGE_STORE_SIZE + @@ -2554,7 +2580,8 @@ static my_bool delete_head_or_tail(MARIA_HA *info, log_array[TRANSLOG_INTERNAL_PARTS + 0].length= sizeof(log_data); if (translog_write_record(&lsn, LOGREC_REDO_PURGE_BLOCKS, info->trn, share, sizeof(log_data), - TRANSLOG_INTERNAL_PARTS + 1, log_array)) + TRANSLOG_INTERNAL_PARTS + 1, log_array, + log_data)) DBUG_RETURN(1); DBUG_ASSERT(empty_space >= info->s->bitmap.sizes[0]); } @@ -2631,7 +2658,6 @@ my_bool _ma_delete_block_record(MARIA_HA *info, const byte *record) /* Write UNDO record */ lsn_store(log_data, info->trn->undo_lsn); - fileid_store(log_data+ LSN_STORE_SIZE, info->dfile.file); page_store(log_data+ LSN_STORE_SIZE + FILEID_STORE_SIZE, page); dirpos_store(log_data+ LSN_STORE_SIZE + FILEID_STORE_SIZE + PAGE_STORE_SIZE, record_number); @@ -2645,7 +2671,7 @@ my_bool _ma_delete_block_record(MARIA_HA *info, const byte *record) if (translog_write_record(&lsn, LOGREC_UNDO_ROW_DELETE, info->trn, info->s, sizeof(log_data) + row_length, TRANSLOG_INTERNAL_PARTS + 1 + row_parts_count, - info->log_row_parts)) + info->log_row_parts, log_data + LSN_STORE_SIZE)) goto err; } diff --git a/storage/maria/ma_blockrec.h b/storage/maria/ma_blockrec.h index f45250ff39c..819d1c2e4d2 100644 --- a/storage/maria/ma_blockrec.h +++ b/storage/maria/ma_blockrec.h @@ -96,7 +96,7 @@ enum en_page_type { UNALLOCATED_PAGE, HEAD_PAGE, TAIL_PAGE, BLOB_PAGE, MAX_PAGE_ /******* defines that affects allocation (density) of data *******/ /* - If the tail part (from the main block or a blob) uses more than 75 % of + If the tail part (from the main block or a blob) would use more than 75 % of the size of page, store the tail on a full page instead of a shared tail page. */ diff --git a/storage/maria/ma_check.c b/storage/maria/ma_check.c index 8f10c98d0ee..0fc2b77304d 100644 --- a/storage/maria/ma_check.c +++ b/storage/maria/ma_check.c @@ -53,6 +53,7 @@ #endif #include "ma_rt_index.h" #include "ma_blockrec.h" +#include "trnman_public.h" /* Functions defined in this file */ @@ -2132,11 +2133,15 @@ err: /* Replace the actual file with the temporary file */ if (new_file >= 0) { + myf sync_dir= (share->base.transactional && !share->temporary) ? + MY_SYNC_DIR : 0; my_close(new_file,MYF(0)); info->dfile.file= new_file= -1; if (maria_change_to_newfile(share->data_file_name,MARIA_NAME_DEXT, - DATA_TMP_EXT, (param->testflag & T_BACKUP_DATA ? - MYF(MY_REDEL_MAKE_BACKUP): MYF(0))) || + DATA_TMP_EXT, + MYF((param->testflag & T_BACKUP_DATA ? + MY_REDEL_MAKE_BACKUP : 0) | + sync_dir)) || _ma_open_datafile(info,share,-1)) got_error=1; } @@ -2328,6 +2333,8 @@ int maria_sort_index(HA_CHECK *param, register MARIA_HA *info, my_string name) int old_lock; MARIA_SHARE *share=info->s; MARIA_STATE_INFO old_state; + myf sync_dir= (share->base.transactional && !share->temporary) ? + MY_SYNC_DIR : 0; DBUG_ENTER("maria_sort_index"); /* cannot sort index files with R-tree indexes */ @@ -2388,7 +2395,7 @@ int maria_sort_index(HA_CHECK *param, register MARIA_HA *info, my_string name) share->kfile.file = -1; VOID(my_close(new_file,MYF(MY_WME))); if (maria_change_to_newfile(share->index_file_name, MARIA_NAME_IEXT, - INDEX_TMP_EXT, MYF(0)) || + INDEX_TMP_EXT, sync_dir) || _ma_open_keyfile(share)) goto err2; info->lock_type= F_UNLCK; /* Force maria_readinfo to lock */ @@ -2604,6 +2611,8 @@ int maria_repair_by_sort(HA_CHECK *param, register MARIA_HA *info, char llbuff[22]; MARIA_SORT_INFO sort_info; ulonglong key_map=share->state.key_map; + myf sync_dir= (share->base.transactional && !share->temporary) ? + MY_SYNC_DIR : 0; DBUG_ENTER("maria_repair_by_sort"); start_records=info->state->records; @@ -2922,8 +2931,9 @@ err: info->dfile.file= new_file= -1; if (maria_change_to_newfile(share->data_file_name,MARIA_NAME_DEXT, DATA_TMP_EXT, - (param->testflag & T_BACKUP_DATA ? - MYF(MY_REDEL_MAKE_BACKUP): MYF(0))) || + MYF((param->testflag & T_BACKUP_DATA ? + MY_REDEL_MAKE_BACKUP : 0) | + sync_dir)) || _ma_open_datafile(info,share,-1)) got_error=1; } @@ -3022,6 +3032,8 @@ int maria_repair_parallel(HA_CHECK *param, register MARIA_HA *info, MARIA_SORT_INFO sort_info; ulonglong key_map=share->state.key_map; pthread_attr_t thr_attr; + myf sync_dir= (share->base.transactional && !share->temporary) ? + MY_SYNC_DIR : 0; DBUG_ENTER("maria_repair_parallel"); start_records=info->state->records; @@ -3445,8 +3457,9 @@ err: info->dfile.file= new_file= -1; if (maria_change_to_newfile(share->data_file_name,MARIA_NAME_DEXT, DATA_TMP_EXT, - (param->testflag & T_BACKUP_DATA ? - MYF(MY_REDEL_MAKE_BACKUP): MYF(0))) || + MYF((param->testflag & T_BACKUP_DATA ? + MY_REDEL_MAKE_BACKUP : 0) | + sync_dir)) || _ma_open_datafile(info,share,-1)) got_error=1; } @@ -5135,3 +5148,64 @@ static void restore_data_file_type(MARIA_SHARE *share) share->data_file_type= share->state.header.data_file_type= share->pack.header_length= 0; } + + +/** + @brief Writes a LOGREC_REPAIR_TABLE record and updates create_rename_lsn + + REPAIR/OPTIMIZE have replaced the data/index file with a new file + and so, in this scenario: + @verbatim + CHECKPOINT - REDO_INSERT - COMMIT - ... - REPAIR - ... - crash + @endverbatim + we do not want Recovery to apply the REDO_INSERT to the table, as it would + then possibly wrongly extend the table. By updating create_rename_lsn at + the end of REPAIR, we know that REDO_INSERT will be skipped. + + @param param description of the REPAIR operation + @param info table + + @return Operation status + @retval 0 ok + @retval 1 error (disk problem) +*/ + +int _ma_repair_write_log_record(const HA_CHECK *param, MARIA_HA *info) +{ + MARIA_SHARE *share= info->s; + /* Only called from ha_maria.cc, not maria_check, so translog is inited */ + if (share->base.transactional && !share->temporary) + { + /* For now this record is only informative */ + LEX_STRING log_array[TRANSLOG_INTERNAL_PARTS + 1]; + uchar log_data[LSN_STORE_SIZE]; + compile_time_assert(LSN_STORE_SIZE >= (FILEID_STORE_SIZE + 4)); + log_array[TRANSLOG_INTERNAL_PARTS + 0].str= (char*) log_data; + log_array[TRANSLOG_INTERNAL_PARTS + 0].length= FILEID_STORE_SIZE + 4; + /* + testflag gives an idea of what REPAIR did (in particular T_QUICK + or not: did it touch the data file or not?). + */ + int4store(log_data + FILEID_STORE_SIZE, param->testflag); + if (unlikely(translog_write_record(&share->state.create_rename_lsn, + LOGREC_REDO_REPAIR_TABLE, + &dummy_transaction_object, share, + log_array[TRANSLOG_INTERNAL_PARTS + + 0].length, + sizeof(log_array)/sizeof(log_array[0]), + log_array, log_data))) + return 1; + /* + But this piece is really needed, to have the new table's content durable + and to not apply old REDOs to the new table. The table's existence was + made durable earlier (MY_SYNC_DIR passed to maria_change_to_newfile()). + */ + lsn_store(log_data, share->state.create_rename_lsn); + DBUG_ASSERT(info->dfile.file >= 0); + DBUG_ASSERT(share->kfile.file >= 0); + return (my_pwrite(share->kfile.file, log_data, sizeof(log_data), + sizeof(share->state.header) + 2, MYF(MY_NABP)) || + _ma_sync_table_files(info)); + } + return 0; +} diff --git a/storage/maria/ma_close.c b/storage/maria/ma_close.c index dc60ce8aa83..34c1bfb4d6d 100644 --- a/storage/maria/ma_close.c +++ b/storage/maria/ma_close.c @@ -57,14 +57,6 @@ int maria_close(register MARIA_HA *info) info->opt_flag&= ~(READ_CACHE_USED | WRITE_CACHE_USED); } flag= !--share->reopen; - /* - RECOVERY TODO: - If "flag" is TRUE, in the line below we are going to make the table - unknown to future checkpoints, so it needs to have fsync'ed itself - entirely (bitmap, pages, etc) at this point. - The flushing is currently done a few lines further (which is ok, as we - still hold THR_LOCK_maria), but syncing is missing. - */ maria_open_list=list_delete(maria_open_list,&info->open_list); pthread_mutex_unlock(&share->intern_lock); @@ -82,7 +74,12 @@ int maria_close(register MARIA_HA *info) FLUSH_IGNORE_CHANGED : FLUSH_RELEASE))) error= my_errno; - + /* + File must be synced as it is going out of the maria_open_list and so + becoming unknown to Checkpoint. + */ + if (my_sync(share->kfile.file, MYF(MY_WME))) + error= my_errno; /* If we are crashed, we can safely flush the current state as it will not change the crashed state. diff --git a/storage/maria/ma_commit.c b/storage/maria/ma_commit.c new file mode 100644 index 00000000000..88aaee0509f --- /dev/null +++ b/storage/maria/ma_commit.c @@ -0,0 +1,71 @@ +/* Copyright (C) 2007 MySQL AB + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ + +#include "maria_def.h" +#include "trnman.h" + +/** + @brief writes a COMMIT record to log and commits transaction in memory + + @param trn transaction + + @return Operation status + @retval 0 ok + @retval 1 error (disk error or out of memory) +*/ + +int ma_commit(TRN *trn) +{ + if (trn->undo_lsn == 0) /* no work done, rollback (cheaper than commit) */ + return trnman_rollback_trn(trn); + /* + - if COMMIT record is written before trnman_commit_trn(): + if Checkpoint comes in the middle it will see trn is not committed, + then if crash, Recovery might roll back trn (if min(rec_lsn) is after + COMMIT record) and this is not an issue as + * transaction's updates were not made visible to other transactions + * "commit ok" was not sent to client + Alternatively, Recovery might commit trn (if min(rec_lsn) is before COMMIT + record), which is ok too. All in all it means that "trn committed" is not + 100% equal to "COMMIT record written". + - if COMMIT record is written after trnman_commit_trn(): + if crash happens between the two, trn will be rolled back which is an + issue (transaction's updates were made visible to other transactions). + So we need to go the first way. + */ + /** + @todo RECOVERY share's state is written to disk only in + maria_lock_database(), so COMMIT record is not the last record of the + transaction! It is probably an issue. Recovery of the state is a problem + not yet solved. + */ + LSN commit_lsn; + LEX_STRING log_array[TRANSLOG_INTERNAL_PARTS]; + /* + We do not store "thd->transaction.xid_state.xid" for now, it will be + needed only when we support XA. + */ + return + translog_write_record(&commit_lsn, LOGREC_COMMIT, + trn, NULL, 0, + sizeof(log_array)/sizeof(log_array[0]), + log_array, NULL) || + translog_flush(commit_lsn) || trnman_commit_trn(trn); + /* + Note: if trnman_commit_trn() fails above, we have already + written the COMMIT record, so Checkpoint and Recovery will see the + transaction as committed. + */ +} diff --git a/storage/maria/ma_commit.h b/storage/maria/ma_commit.h new file mode 100644 index 00000000000..2c57c73fd7a --- /dev/null +++ b/storage/maria/ma_commit.h @@ -0,0 +1,18 @@ +/* Copyright (C) 2007 MySQL AB + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ + +C_MODE_START +int ma_commit(TRN *trn); +C_MODE_END diff --git a/storage/maria/ma_control_file.c b/storage/maria/ma_control_file.c index f53da8a5881..db5440dc873 100644 --- a/storage/maria/ma_control_file.c +++ b/storage/maria/ma_control_file.c @@ -50,6 +50,13 @@ LSN last_checkpoint_lsn; uint32 last_logno; +/** + @brief If log's lock should be asserted when writing to control file. + + Can be re-used by any function which needs to be thread-safe except when + it is called at startup. +*/ +my_bool maria_multi_threaded= FALSE; /* Control file is less then 512 bytes (a disk sector), @@ -203,6 +210,8 @@ err: the last_checkpoint_lsn and last_logno global variables. Called when we have created a new log (after syncing this log's creation) and when we have written a checkpoint (after syncing this log record). + Variables last_checkpoint_lsn and last_logno must be protected by caller + using log's lock, unless this function is called at startup. SYNOPSIS ma_control_file_write_and_force() @@ -233,12 +242,14 @@ int ma_control_file_write_and_force(const LSN checkpoint_lsn, uint32 logno, DBUG_ENTER("ma_control_file_write_and_force"); DBUG_ASSERT(control_file_fd >= 0); /* must be open */ +#ifndef DBUG_OFF + if (maria_multi_threaded) + translog_lock_assert_owner(); +#endif memcpy(buffer + CONTROL_FILE_MAGIC_STRING_OFFSET, CONTROL_FILE_MAGIC_STRING, CONTROL_FILE_MAGIC_STRING_SIZE); - /* TODO: you need some protection to be able to read last_* global vars */ - if (objs_to_write == CONTROL_FILE_UPDATE_ONLY_LSN) update_checkpoint_lsn= TRUE; else if (objs_to_write == CONTROL_FILE_UPDATE_ONLY_LOGNO) @@ -270,7 +281,6 @@ int ma_control_file_write_and_force(const LSN checkpoint_lsn, uint32 logno, my_sync(control_file_fd, MYF(MY_WME))) DBUG_RETURN(1); - /* TODO: you need some protection to be able to write last_* global vars */ if (update_checkpoint_lsn) last_checkpoint_lsn= checkpoint_lsn; if (update_logno) diff --git a/storage/maria/ma_control_file.h b/storage/maria/ma_control_file.h index 4728d719b2f..c974838684b 100644 --- a/storage/maria/ma_control_file.h +++ b/storage/maria/ma_control_file.h @@ -43,6 +43,8 @@ extern LSN last_checkpoint_lsn; */ extern uint32 last_logno; +extern my_bool maria_multi_threaded; + typedef enum enum_control_file_error { CONTROL_FILE_OK= 0, CONTROL_FILE_TOO_SMALL, diff --git a/storage/maria/ma_create.c b/storage/maria/ma_create.c index d8660dd41cb..53e15deb74b 100644 --- a/storage/maria/ma_create.c +++ b/storage/maria/ma_create.c @@ -19,6 +19,7 @@ #include "ma_sp_defs.h" #include #include "ma_blockrec.h" +#include "trnman_public.h" #if defined(MSDOS) || defined(__WIN__) #ifdef __WIN__ @@ -51,7 +52,8 @@ int maria_create(const char *name, enum data_file_type datafile_type, unique_key_parts,fulltext_keys,offset, not_block_record_extra_length; uint max_field_lengths, extra_header_size; ulong reclength, real_reclength,min_pack_length; - char filename[FN_REFLEN],linkname[FN_REFLEN], *linkname_ptr; + char filename[FN_REFLEN], dlinkname[FN_REFLEN], *dlinkname_ptr= NULL, + klinkname[FN_REFLEN], *klinkname_ptr= NULL; ulong pack_reclength; ulonglong tot_length,max_rows, tmp; enum en_fieldtype type; @@ -62,11 +64,12 @@ int maria_create(const char *name, enum data_file_type datafile_type, HA_KEYSEG *keyseg,tmp_keyseg; MARIA_COLUMNDEF *column, *end_column; ulong *rec_per_key_part; - my_off_t key_root[HA_MAX_POSSIBLE_KEY]; + my_off_t key_root[HA_MAX_POSSIBLE_KEY], kfile_size_before_extension; MARIA_CREATE_INFO tmp_create_info; my_bool tmp_table= FALSE; /* cache for presence of HA_OPTION_TMP_TABLE */ my_bool forced_packed; - myf sync_dir= MY_SYNC_DIR; + myf sync_dir= 0; + uchar *log_data= NULL; DBUG_ENTER("maria_create"); DBUG_PRINT("enter", ("keys: %u columns: %u uniques: %u flags: %u", keys, columns, uniques, flags)); @@ -250,8 +253,9 @@ int maria_create(const char *name, enum data_file_type datafile_type, if (flags & HA_CREATE_TMP_TABLE) { options|= HA_OPTION_TMP_TABLE; + tmp_table= TRUE; create_mode|= O_EXCL | O_NOFOLLOW; - /* temp tables are not crash-safe (dropped at restart) */ + /* "CREATE TEMPORARY" tables are not crash-safe (dropped at restart) */ ci->transactional= FALSE; } share.base.null_bytes= ci->null_bytes; @@ -624,6 +628,7 @@ int maria_create(const char *name, enum data_file_type datafile_type, share.state.dellink = HA_OFFSET_ERROR; share.state.first_bitmap_with_space= 0; + share.state.create_rename_lsn= 0; share.state.process= (ulong) getpid(); share.state.unique= (ulong) 0; share.state.update_count=(ulong) 0; @@ -671,11 +676,15 @@ int maria_create(const char *name, enum data_file_type datafile_type, #endif /* max_data_file_length and max_key_file_length are recalculated on open */ - if (options & HA_OPTION_TMP_TABLE) - { - tmp_table= TRUE; - sync_dir= 0; + if (tmp_table) share.base.max_data_file_length= (my_off_t) ci->data_file_length; + else if (ci->transactional && translog_inited) + { + /* + we have checked translog_inited above, because maria_chk may call us + (via maria_recreate_table()) and it does not have a log. + */ + sync_dir= MY_SYNC_DIR; } if (datafile_type == BLOCK_RECORD) @@ -712,9 +721,9 @@ int maria_create(const char *name, enum data_file_type datafile_type, MY_UNPACK_FILENAME | (have_iext ? MY_REPLACE_EXT : MY_APPEND_EXT)); } - fn_format(linkname, name, "", MARIA_NAME_IEXT, + fn_format(klinkname, name, "", MARIA_NAME_IEXT, MY_UNPACK_FILENAME|MY_APPEND_EXT); - linkname_ptr=linkname; + klinkname_ptr= klinkname; /* Don't create the table if the link or file exists to ensure that one doesn't accidently destroy another table. @@ -730,7 +739,6 @@ int maria_create(const char *name, enum data_file_type datafile_type, (MY_UNPACK_FILENAME | (flags & HA_DONT_TOUCH_DATA) ? MY_RETURN_REAL_PATH : 0) | MY_APPEND_EXT); - linkname_ptr=0; /* Replace the current file. Don't sync dir now if the data file has the same path. @@ -753,7 +761,7 @@ int maria_create(const char *name, enum data_file_type datafile_type, goto err; } - if ((file= my_create_with_symlink(linkname_ptr, filename, 0, create_mode, + if ((file= my_create_with_symlink(klinkname_ptr, filename, 0, create_mode, MYF(MY_WME|create_flag))) < 0) goto err; errpos=1; @@ -780,24 +788,24 @@ int maria_create(const char *name, enum data_file_type datafile_type, MY_UNPACK_FILENAME | (have_dext ? MY_REPLACE_EXT : MY_APPEND_EXT)); } - fn_format(linkname, name, "",MARIA_NAME_DEXT, + fn_format(dlinkname, name, "",MARIA_NAME_DEXT, MY_UNPACK_FILENAME | MY_APPEND_EXT); - linkname_ptr=linkname; + dlinkname_ptr= dlinkname; create_flag=0; } else { fn_format(filename,name,"", MARIA_NAME_DEXT, MY_UNPACK_FILENAME | MY_APPEND_EXT); - linkname_ptr=0; create_flag=MY_DELETE_OLD; } if ((dfile= - my_create_with_symlink(linkname_ptr, filename, 0, create_mode, + my_create_with_symlink(dlinkname_ptr, filename, 0, create_mode, MYF(MY_WME | create_flag | sync_dir))) < 0) goto err; errpos=3; + share.data_file_type= datafile_type; if (_ma_initialize_data_file(dfile, &share)) goto err; } @@ -925,14 +933,82 @@ int maria_create(const char *name, enum data_file_type datafile_type, goto err; } + if ((kfile_size_before_extension= my_tell(file,MYF(0))) == MY_FILEPOS_ERROR) + goto err; #ifndef DBUG_OFF - if ((uint) my_tell(file,MYF(0)) != info_length) + if (kfile_size_before_extension != info_length) + DBUG_PRINT("warning",("info_length: %u != used_length: %u", + info_length, (uint)kfile_size_before_extension)); +#endif + + if (sync_dir) { - uint pos= (uint) my_tell(file,MYF(0)); - DBUG_PRINT("warning",("info_length: %d != used_length: %d", - info_length, pos)); + /* + we log the first bytes and then the size to which we extend; this is + not log 1 KB of mostly zeroes if this is a small table. + */ + char empty_string[]= ""; + LEX_STRING log_array[TRANSLOG_INTERNAL_PARTS + 3]; + uint total_rec_length= 0; + uint i; + log_array[TRANSLOG_INTERNAL_PARTS + 0].length= 1 + 2 + + kfile_size_before_extension; + /* we are needing maybe 64 kB, so don't use the stack */ + log_data= my_malloc(log_array[TRANSLOG_INTERNAL_PARTS + 0].length, MYF(0)); + if ((log_data == NULL) || + my_pread(file, 1 + 2 + log_data, kfile_size_before_extension, + 0, MYF(MY_NABP))) + goto err_no_lock; + /* + remember if the data file was created or not, to know if Recovery can + do it or not, in the future + */ + log_data[0]= test(flags & HA_DONT_TOUCH_DATA); + int2store(log_data + 1, kfile_size_before_extension); + log_array[TRANSLOG_INTERNAL_PARTS + 0].str= log_data; + /* symlink description is also needed for re-creation by Recovery: */ + log_array[TRANSLOG_INTERNAL_PARTS + 1].str= + dlinkname_ptr ? dlinkname : empty_string; + log_array[TRANSLOG_INTERNAL_PARTS + 1].length= + strlen(log_array[TRANSLOG_INTERNAL_PARTS + 1].str); + log_array[TRANSLOG_INTERNAL_PARTS + 2].str= + klinkname_ptr ? klinkname : empty_string; + log_array[TRANSLOG_INTERNAL_PARTS + 2].length= + strlen(log_array[TRANSLOG_INTERNAL_PARTS + 2].str); + for (i= TRANSLOG_INTERNAL_PARTS; + i < (sizeof(log_array)/sizeof(log_array[0])); i++) + total_rec_length+= log_array[i].length; + /* + For this record to be of any use for Recovery, we need the upper + MySQL layer to be crash-safe, which it is not now (that would require + work using the ddl_log of sql/sql_table.cc); when it is, we should + reconsider the moment of writing this log record (before or after op, + under THR_LOCK_maria or not...), how to use it in Recovery, and force + the log. For now this record is just informative. + Note that in case of TRUNCATE TABLE we also come here. + When in CREATE/TRUNCATE (or DROP or RENAME or REPAIR) we have not called + external_lock(), so have no TRN. It does not matter, as all these + operations are non-transactional and sync their files. + */ + if (unlikely(translog_write_record(&share.state.create_rename_lsn, + LOGREC_REDO_CREATE_TABLE, + &dummy_transaction_object, NULL, + total_rec_length, + sizeof(log_array)/sizeof(log_array[0]), + log_array, NULL))) + goto err_no_lock; + /* + store LSN into file, needed for Recovery to not be confused if a + DROP+CREATE happened (applying REDOs to the wrong table). + If such direct my_pwrite() to a fixed offset is too "hackish", I can + call ma_state_info_write() again but it will be less efficient. + */ + lsn_store(log_data, share.state.create_rename_lsn); + if (my_pwrite(file, log_data, LSN_STORE_SIZE, + sizeof(share.state.header) + 2, MYF(MY_NABP))) + goto err_no_lock; + my_free(log_data, MYF(0)); } -#endif /* Enlarge files */ DBUG_PRINT("info", ("enlarge to keystart: %lu", @@ -940,38 +1016,25 @@ int maria_create(const char *name, enum data_file_type datafile_type, if (my_chsize(file,(ulong) share.base.keystart,0,MYF(0))) goto err; + if (sync_dir && my_sync(file, MYF(0))) + goto err; + if (! (flags & HA_DONT_TOUCH_DATA)) { #ifdef USE_RELOC if (my_chsize(dfile,share.base.min_pack_length*ci->reloc_rows,0,MYF(0))) goto err; - if (!tmp_table && my_sync(file, MYF(0))) - goto err; #endif - /* if !USE_RELOC, there was no write to the file, no need to sync it */ errpos=2; - if (my_close(dfile,MYF(0))) + if ((sync_dir && my_sync(dfile, MYF(0))) || my_close(dfile,MYF(0))) goto err; } - errpos=0; pthread_mutex_unlock(&THR_LOCK_maria); res= 0; + my_free((char*) rec_per_key_part,MYF(0)); + errpos=0; if (my_close(file,MYF(0))) res= my_errno; - /* - RECOVERY TODO - Write a log record describing the CREATE operation (just the file - names, link names, and the full header's content). - For this record to be of any use for Recovery, we need the upper - MySQL layer to be crash-safe, which it is not now (that would require work - using the ddl_log of sql/sql_table.cc); when is is, we should reconsider - the moment of writing this log record (before or after op, under - THR_LOCK_maria or not...), how to use it in Recovery, and force the log. - For now this record is just informative. - If operation failed earlier, we clean up in "err:" and the MySQL layer - will clean up the frm, so we needn't write anything to the log. - */ - my_free((char*) rec_per_key_part,MYF(0)); DBUG_RETURN(res); err: @@ -996,6 +1059,7 @@ err_no_lock: MY_UNPACK_FILENAME | MY_APPEND_EXT), sync_dir); } + my_free(log_data, MYF(MY_ALLOW_ZERO_PTR)); my_free((char*) rec_per_key_part, MYF(0)); DBUG_RETURN(my_errno=save_errno); /* return the fatal errno */ } @@ -1086,9 +1150,9 @@ int _ma_initialize_data_file(File dfile, MARIA_SHARE *share) { if (share->data_file_type == BLOCK_RECORD) { - if (my_chsize(dfile, maria_block_size, 0, MYF(MY_WME))) + if (my_chsize(dfile, share->base.block_size, 0, MYF(MY_WME))) return 1; - share->state.state.data_file_length= maria_block_size; + share->state.state.data_file_length= share->base.block_size; _ma_bitmap_delete_all(share); } return 0; diff --git a/storage/maria/ma_delete_all.c b/storage/maria/ma_delete_all.c index 2d85b347662..7286f540aa1 100644 --- a/storage/maria/ma_delete_all.c +++ b/storage/maria/ma_delete_all.c @@ -17,21 +17,38 @@ /* This clears the status information and truncates files */ #include "maria_def.h" +#include "trnman_public.h" + +/** + @brief deletes all rows from a table + + @param info Maria handler + + @return Operation status + @retval 0 ok + @retval 1 error +*/ int maria_delete_all_rows(MARIA_HA *info) { uint i; MARIA_SHARE *share=info->s; MARIA_STATE_INFO *state=&share->state; + my_bool log_record; DBUG_ENTER("maria_delete_all_rows"); if (share->options & HA_OPTION_READ_ONLY_DATA) { DBUG_RETURN(my_errno=EACCES); } - /* LOCK TODO take X-lock on table here */ + /** + @todo LOCK take X-lock on table here. + When we have versioning, if some other thread is looking at this table, + we cannot shrink the file like this. + */ if (_ma_readinfo(info,F_WRLCK,1)) DBUG_RETURN(my_errno); + log_record= share->base.transactional && !share->temporary; if (_ma_mark_file_changed(info)) goto err; @@ -54,27 +71,13 @@ int maria_delete_all_rows(MARIA_HA *info) */ flush_pagecache_blocks(share->pagecache, &share->kfile, FLUSH_IGNORE_CHANGED); - /* - RECOVERY TODO Log the two chsize and header modifications and force the - log. So that if crash between the two chsize, we finish the work at - Recovery. For this scenario: - "TRUNCATE TABLE t1; DROP TABLE t1; RENAME TABLE t2 to t1; crash;" - Recovery mustn't truncate the new t1, so the log records of TRUNCATE - should be applied only if t1 exists and its ZeroDirtyPagesLSN is smaller - than the records'. See more comments below. - */ if (my_chsize(info->dfile.file, 0, 0, MYF(MY_WME)) || my_chsize(share->kfile.file, share->base.keystart, 0, MYF(MY_WME)) ) goto err; - if (_ma_initialize_data_file(info->dfile.file, info->s)) + if (_ma_initialize_data_file(info->dfile.file, share)) goto err; - /* - RECOVERY TODO Consider updating ZeroDirtyPagesLSN here. It is - not a necessity (it is one only in RENAME commands) but an optional - optimization which will allow some REDO skipping at Recovery. - */ VOID(_ma_writeinfo(info,WRITEINFO_UPDATE_KEYFILE)); #ifdef HAVE_MMAP /* Resize mmaped area */ @@ -82,24 +85,48 @@ int maria_delete_all_rows(MARIA_HA *info) _ma_remap_file(info, (my_off_t)0); rw_unlock(&info->s->mmap_lock); #endif - /* - RECOVERY TODO Until we have the TRUNCATE log record and take it into - account for log-low-water-mark calculation and use it in Recovery, we need - to sync. - */ - if (_ma_sync_table_files(info)) - goto err; + if (log_record) + { + /* For now this record is only informative */ + LEX_STRING log_array[TRANSLOG_INTERNAL_PARTS + 1]; + uchar log_data[LSN_STORE_SIZE]; + log_array[TRANSLOG_INTERNAL_PARTS + 0].str= (char*) log_data; + log_array[TRANSLOG_INTERNAL_PARTS + 0].length= FILEID_STORE_SIZE; + if (unlikely(translog_write_record(&share->state.create_rename_lsn, + LOGREC_REDO_DELETE_ALL, + info->trn, share, 0, + sizeof(log_array)/sizeof(log_array[0]), + log_array, log_data))) + goto err; + /* + store LSN into file. It is an optimization so that all old REDOs for + this table are ignored (scenario: checkpoint, INSERT1s, DELETE ALL; + INSERT2s, crash: then Recovery can skip INSERT1s). It also allows us to + ignore the present record at Recovery. + Note that storing the LSN could not be done by _ma_writeinfo() above as + the table is locked at this moment. So we need to do it by ourselves. + */ + lsn_store(log_data, share->state.create_rename_lsn); + if (my_pwrite(share->kfile.file, log_data, sizeof(log_data), + sizeof(share->state.header) + 2, MYF(MY_NABP)) || + _ma_sync_table_files(info)) + goto err; + /** + @todo RECOVERY Until we take into account the log record above + for log-low-water-mark calculation and use it in Recovery, we need + to sync above. + */ + } allow_break(); /* Allow SIGHUP & SIGINT */ DBUG_RETURN(0); err: { int save_errno=my_errno; - /* RECOVERY TODO log the header modifications */ VOID(_ma_writeinfo(info,WRITEINFO_UPDATE_KEYFILE)); info->update|=HA_STATE_WRITTEN; /* Buffer changed */ - /* RECOVERY TODO until we log above we have to sync */ - if (_ma_sync_table_files(info) && !save_errno) + /** @todo RECOVERY until we use the log record above we have to sync */ + if (log_record &&_ma_sync_table_files(info) && !save_errno) save_errno= my_errno; allow_break(); /* Allow SIGHUP & SIGINT */ DBUG_RETURN(my_errno=save_errno); diff --git a/storage/maria/ma_delete_table.c b/storage/maria/ma_delete_table.c index aafe7a1dee9..990714043bf 100644 --- a/storage/maria/ma_delete_table.c +++ b/storage/maria/ma_delete_table.c @@ -13,11 +13,18 @@ along with this program; if not, write to the Free Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ -/* - deletes a table -*/ - #include "ma_fulltext.h" +#include "trnman_public.h" + +/** + @brief drops (deletes) a table + + @param name table's name + + @return Operation status + @retval 0 ok + @retval 1 error +*/ int maria_delete_table(const char *name) { @@ -25,56 +32,78 @@ int maria_delete_table(const char *name) #ifdef USE_RAID uint raid_type=0,raid_chunks=0; #endif + MARIA_HA *info; + myf sync_dir; DBUG_ENTER("maria_delete_table"); #ifdef EXTRA_DEBUG _ma_check_table_is_closed(name,"delete"); #endif - /* LOCK TODO take X-lock on table here */ + /** @todo LOCK take X-lock on table */ + /* + We need to know if this table is transactional. + When built with RAID support, we also need to determine if this table + makes use of the raid feature. If yes, we need to remove all raid + chunks. This is done with my_raid_delete(). Unfortunately it is + necessary to open the table just to check this. We use + 'open_for_repair' to be able to open even a crashed table. If even + this open fails, we assume no raid configuration for this table + and try to remove the normal data file only. This may however + leave the raid chunks behind. + */ + if (!(info= maria_open(name, O_RDONLY, HA_OPEN_FOR_REPAIR))) + { #ifdef USE_RAID + raid_type= 0; +#endif + sync_dir= 0; + } + else { - MARIA_HA *info; - /* - When built with RAID support, we need to determine if this table - makes use of the raid feature. If yes, we need to remove all raid - chunks. This is done with my_raid_delete(). Unfortunately it is - necessary to open the table just to check this. We use - 'open_for_repair' to be able to open even a crashed table. If even - this open fails, we assume no raid configuration for this table - and try to remove the normal data file only. This may however - leave the raid chunks behind. - */ - if (!(info= maria_open(name, O_RDONLY, HA_OPEN_FOR_REPAIR))) - raid_type= 0; - else - { - raid_type= info->s->base.raid_type; - raid_chunks= info->s->base.raid_chunks; - maria_close(info); - } +#ifdef USE_RAID + raid_type= info->s->base.raid_type; + raid_chunks= info->s->base.raid_chunks; +#endif + sync_dir= (info->s->base.transactional && !info->s->temporary) ? + MY_SYNC_DIR : 0; + maria_close(info); } +#ifdef USE_RAID #ifdef EXTRA_DEBUG _ma_check_table_is_closed(name,"delete"); #endif #endif /* USE_RAID */ + if (sync_dir) + { + /* + For this log record to be of any use for Recovery, we need the upper + MySQL layer to be crash-safe in DDLs; when it is we should reconsider + the moment of writing this log record, how to use it in Recovery, and + force the log. For now this record is only informative. + */ + LSN lsn; + LEX_STRING log_array[TRANSLOG_INTERNAL_PARTS + 1]; + log_array[TRANSLOG_INTERNAL_PARTS + 0].str= (char *)name; + log_array[TRANSLOG_INTERNAL_PARTS + 0].length= strlen(name); + if (unlikely(translog_write_record(&lsn, LOGREC_REDO_DROP_TABLE, + &dummy_transaction_object, NULL, + log_array[TRANSLOG_INTERNAL_PARTS + + 0].length, + sizeof(log_array)/sizeof(log_array[0]), + log_array, NULL))) + DBUG_RETURN(1); + } + fn_format(from,name,"",MARIA_NAME_IEXT,MY_UNPACK_FILENAME|MY_APPEND_EXT); - /* - RECOVERY TODO log the two deletes below. - Then do the file deletions. - For this log record to be of any use for Recovery, we need the upper MySQL - layer to be crash-safe in DDLs; when it is we should reconsider the moment - of writing this log record, how to use it in Recovery, and force the log. - For now this record is only informative. - */ - if (my_delete_with_symlink(from, MYF(MY_WME | MY_SYNC_DIR))) + if (my_delete_with_symlink(from, MYF(MY_WME | sync_dir))) DBUG_RETURN(my_errno); fn_format(from,name,"",MARIA_NAME_DEXT,MY_UNPACK_FILENAME|MY_APPEND_EXT); #ifdef USE_RAID if (raid_type) - DBUG_RETURN(my_raid_delete(from, raid_chunks, MYF(MY_WME | MY_SYNC_DIR)) ? + DBUG_RETURN(my_raid_delete(from, raid_chunks, MYF(MY_WME | sync_dir)) ? my_errno : 0); #endif - DBUG_RETURN(my_delete_with_symlink(from, MYF(MY_WME | MY_SYNC_DIR)) ? + DBUG_RETURN(my_delete_with_symlink(from, MYF(MY_WME | sync_dir)) ? my_errno : 0); } diff --git a/storage/maria/ma_extra.c b/storage/maria/ma_extra.c index d6a0d2f4441..61eba165412 100644 --- a/storage/maria/ma_extra.c +++ b/storage/maria/ma_extra.c @@ -21,21 +21,20 @@ static void maria_extra_keyflag(MARIA_HA *info, enum ha_extra_function function); +/** + @brief Set options and buffers to optimize table handling -/* - Set options and buffers to optimize table handling + @param name table's name + @param info open table + @param function operation + @param extra_arg Pointer to extra argument (normally pointer to + ulong); used when function is one of: + HA_EXTRA_WRITE_CACHE + HA_EXTRA_CACHE - SYNOPSIS - maria_extra() - info open table - function operation - extra_arg Pointer to extra argument (normally pointer to ulong) - Used when function is one of: - HA_EXTRA_WRITE_CACHE - HA_EXTRA_CACHE - RETURN VALUES - 0 ok - # error + @return Operation status + @retval 0 ok + @retval !=0 error */ int maria_extra(MARIA_HA *info, enum ha_extra_function function, @@ -265,14 +264,24 @@ int maria_extra(MARIA_HA *info, enum ha_extra_function function, pthread_mutex_unlock(&THR_LOCK_maria); break; case HA_EXTRA_PREPARE_FOR_DELETE: + /* QQ: suggest to rename it to "PREPARE_FOR_DROP" */ pthread_mutex_lock(&THR_LOCK_maria); share->last_version= 0L; /* Impossible version */ #ifdef __WIN__ /* Close the isam and data files as Win32 can't drop an open table */ pthread_mutex_lock(&share->intern_lock); + /* + If this is Windows we remove blocks from pagecache. If not Windows we + don't do it, so these pages stay in the pagecache? So they may later be + flushed to a wrong file? + Or is it that this flush_pagecache_blocks() never finds any blocks? Then + why do we do it on Windows? + Don't we wait for all instances to be closed before dropping the table? + Do we ever do something useful here? + BUG? + */ if (flush_pagecache_blocks(share->pagecache, &share->kfile, - (function == HA_EXTRA_FORCE_REOPEN ? - FLUSH_RELEASE : FLUSH_IGNORE_CHANGED))) + FLUSH_IGNORE_CHANGED)) { error=my_errno; share->changed=1; @@ -292,9 +301,11 @@ int maria_extra(MARIA_HA *info, enum ha_extra_function function, info->lock_type = F_UNLCK; } if (share->kfile.file >= 0) + { _ma_decrement_open_count(info); - if (share->kfile.file >= 0 && my_close(share->kfile,MYF(0))) - error=my_errno; + if (my_close(share->kfile,MYF(0))) + error=my_errno; + } { LIST *list_element ; for (list_element=maria_open_list ; @@ -304,6 +315,9 @@ int maria_extra(MARIA_HA *info, enum ha_extra_function function, MARIA_HA *tmpinfo=(MARIA_HA*) list_element->data; if (tmpinfo->s == info->s) { + /** + @todo RECOVERY BUG: flush of bitmap and sync of dfile are missing + */ if (tmpinfo->dfile.file >= 0 && my_close(tmpinfo->dfile.file, MYF(0))) error = my_errno; diff --git a/storage/maria/ma_init.c b/storage/maria/ma_init.c index ac4826a721d..8042c6d9873 100644 --- a/storage/maria/ma_init.c +++ b/storage/maria/ma_init.c @@ -53,7 +53,7 @@ void maria_end(void) { if (maria_inited) { - maria_inited= FALSE; + maria_inited= maria_multi_threaded= FALSE; ft_free_stopwords(); trnman_destroy(); translog_destroy(); diff --git a/storage/maria/ma_loghandler.c b/storage/maria/ma_loghandler.c index 474f50e1e2c..9ed1d4b9d93 100644 --- a/storage/maria/ma_loghandler.c +++ b/storage/maria/ma_loghandler.c @@ -17,6 +17,14 @@ #include "ma_blockrec.h" #include "trnman.h" +/** + @file + @brief Module which writes and reads to a transaction log + + @todo LOG: in functions where the log's lock is required, a + translog_assert_owner() could be added. +*/ + /* number of opened log files in the pagecache (should be at least 2) */ #define OPENED_FILES_NUM 3 @@ -166,7 +174,7 @@ static struct st_translog_descriptor log_descriptor; /* Marker for end of log */ static byte end_of_log= 0; -static my_bool translog_inited; +my_bool translog_inited= 0; /* record classes */ enum record_class @@ -218,7 +226,7 @@ struct st_log_record_type_descriptor uint16 read_header_len; /* HOOK for writing the record called before lock */ prewrite_rec_hook prewrite_hook; - /* HOOK for writing the record called when LSN is known */ + /* HOOK for writing the record called when LSN is known, inside lock */ inwrite_rec_hook inwrite_hook; /* HOOK for reading headers */ read_rec_hook read_hook; @@ -230,6 +238,13 @@ struct st_log_record_type_descriptor }; +#include +/* an array that maps id of a MARIA_SHARE to this MARIA_SHARE */ +static MARIA_SHARE **id_to_share= NULL; +#define SHARE_ID_MAX 65535 /* array's size */ +/* lock for id_to_share */ +static my_atomic_rwlock_t LOCK_id_to_share; + static my_bool write_hook_for_redo(enum translog_record_type type, TRN *trn, LSN *lsn, struct st_translog_parts *parts); @@ -291,7 +306,9 @@ static LOG_DESC INIT_LOGREC_REDO_INSERT_ROW_HEAD= write_hook_for_redo, NULL, 0}; static LOG_DESC INIT_LOGREC_REDO_INSERT_ROW_TAIL= -{LOGRECTYPE_VARIABLE_LENGTH, 0, 8, NULL, NULL, NULL, 0}; +{LOGRECTYPE_VARIABLE_LENGTH, 0, + FILEID_STORE_SIZE + PAGE_STORE_SIZE + DIRPOS_STORE_SIZE, NULL, + write_hook_for_redo, NULL, 0}; static LOG_DESC INIT_LOGREC_REDO_INSERT_ROW_BLOB= {LOGRECTYPE_VARIABLE_LENGTH, 0, 8, NULL, write_hook_for_redo, NULL, 0}; @@ -376,15 +393,9 @@ static LOG_DESC INIT_LOGREC_COMMIT= static LOG_DESC INIT_LOGREC_COMMIT_WITH_UNDO_PURGE= {LOGRECTYPE_PSEUDOFIXEDLENGTH, 5, 5, NULL, NULL, NULL, 1}; -static LOG_DESC INIT_LOGREC_CHECKPOINT_PAGE= -{LOGRECTYPE_VARIABLE_LENGTH, 0, 6, NULL, NULL, NULL, 0}; - -static LOG_DESC INIT_LOGREC_CHECKPOINT_TRAN= +static LOG_DESC INIT_LOGREC_CHECKPOINT= {LOGRECTYPE_VARIABLE_LENGTH, 0, 0, NULL, NULL, NULL, 0}; -static LOG_DESC INIT_LOGREC_CHECKPOINT_TABL= -{LOGRECTYPE_VARIABLE_LENGTH, 0, 8, NULL, NULL, NULL, 0}; - static LOG_DESC INIT_LOGREC_REDO_CREATE_TABLE= {LOGRECTYPE_VARIABLE_LENGTH, 0, 0, NULL, NULL, NULL, 0}; @@ -394,8 +405,13 @@ static LOG_DESC INIT_LOGREC_REDO_RENAME_TABLE= static LOG_DESC INIT_LOGREC_REDO_DROP_TABLE= {LOGRECTYPE_VARIABLE_LENGTH, 0, 0, NULL, NULL, NULL, 0}; -static LOG_DESC INIT_LOGREC_REDO_TRUNCATE_TABLE= -{LOGRECTYPE_VARIABLE_LENGTH, 0, 0, NULL, NULL, NULL, 0}; +static LOG_DESC INIT_LOGREC_REDO_DELETE_ALL= +{LOGRECTYPE_FIXEDLENGTH, FILEID_STORE_SIZE, FILEID_STORE_SIZE, + NULL, NULL, NULL, 0}; + +static LOG_DESC INIT_LOGREC_REDO_REPAIR_TABLE= +{LOGRECTYPE_FIXEDLENGTH, FILEID_STORE_SIZE + 4, FILEID_STORE_SIZE + 4, + NULL, NULL, NULL, 0}; static LOG_DESC INIT_LOGREC_FILE_ID= {LOGRECTYPE_VARIABLE_LENGTH, 0, 4, NULL, NULL, NULL, 0}; @@ -403,6 +419,7 @@ static LOG_DESC INIT_LOGREC_FILE_ID= static LOG_DESC INIT_LOGREC_LONG_TRANSACTION_ID= {LOGRECTYPE_FIXEDLENGTH, 6, 6, NULL, NULL, NULL, 0}; +const myf log_write_flags= MY_WME | MY_NABP | MY_WAIT_IF_FULL; static void loghandler_init() { @@ -454,20 +471,18 @@ static void loghandler_init() INIT_LOGREC_COMMIT; log_record_type_descriptor[LOGREC_COMMIT_WITH_UNDO_PURGE]= INIT_LOGREC_COMMIT_WITH_UNDO_PURGE; - log_record_type_descriptor[LOGREC_CHECKPOINT_PAGE]= - INIT_LOGREC_CHECKPOINT_PAGE; - log_record_type_descriptor[LOGREC_CHECKPOINT_TRAN]= - INIT_LOGREC_CHECKPOINT_TRAN; - log_record_type_descriptor[LOGREC_CHECKPOINT_TABL]= - INIT_LOGREC_CHECKPOINT_TABL; + log_record_type_descriptor[LOGREC_CHECKPOINT]= + INIT_LOGREC_CHECKPOINT; log_record_type_descriptor[LOGREC_REDO_CREATE_TABLE]= INIT_LOGREC_REDO_CREATE_TABLE; log_record_type_descriptor[LOGREC_REDO_RENAME_TABLE]= INIT_LOGREC_REDO_RENAME_TABLE; log_record_type_descriptor[LOGREC_REDO_DROP_TABLE]= INIT_LOGREC_REDO_DROP_TABLE; - log_record_type_descriptor[LOGREC_REDO_TRUNCATE_TABLE]= - INIT_LOGREC_REDO_TRUNCATE_TABLE; + log_record_type_descriptor[LOGREC_REDO_DELETE_ALL]= + INIT_LOGREC_REDO_DELETE_ALL; + log_record_type_descriptor[LOGREC_REDO_REPAIR_TABLE]= + INIT_LOGREC_REDO_REPAIR_TABLE; log_record_type_descriptor[LOGREC_FILE_ID]= INIT_LOGREC_FILE_ID; log_record_type_descriptor[LOGREC_LONG_TRANSACTION_ID]= @@ -554,6 +569,7 @@ static File open_logfile_by_number_no_cache(uint32 file_no) DBUG_ENTER("open_logfile_by_number_no_cache"); /* TODO: add O_DIRECT to open flags (when buffer is aligned) */ + /* TODO: use my_create() */ if ((file= my_open(translog_filename_by_fileno(file_no, path), O_CREAT | O_BINARY | O_RDWR, MYF(MY_WME))) < 0) @@ -615,7 +631,7 @@ static my_bool translog_write_file_header() bzero(page, sizeof(page_buff) - (page- page_buff)); DBUG_RETURN(my_pwrite(log_descriptor.log_file_num[0], page_buff, - sizeof(page_buff), 0, MYF(MY_WME | MY_NABP)) != 0); + sizeof(page_buff), 0, log_write_flags) != 0); } @@ -1222,7 +1238,7 @@ static my_bool translog_buffer_next(TRANSLOG_ADDRESS *horizon, /* - Set max LSN send to file + Set max LSN sent to file SYNOPSIS translog_set_sent_to_file() @@ -1512,7 +1528,7 @@ static my_bool translog_buffer_flush(struct st_translog_buffer *buffer) } if (my_pwrite(buffer->file, (char*) buffer->buffer, buffer->size, LSN_OFFSET(buffer->offset), - MYF(MY_WME | MY_NABP))) + log_write_flags)) { UNRECOVERABLE_ERROR(("Can't write buffer (%lu,0x%lx) size %lu " "to the disk (%d)", @@ -2230,7 +2246,16 @@ my_bool translog_init(const char *directory, */ log_descriptor.flushed--; /* offset decreased */ log_descriptor.sent_to_file--; /* offset decreased */ - + /* + Log records will refer to a MARIA_SHARE by a unique 2-byte id; set up + structures for generating 2-byte ids: + */ + my_atomic_rwlock_init(&LOCK_id_to_share); + id_to_share= (MARIA_SHARE **) my_malloc(SHARE_ID_MAX*sizeof(MARIA_SHARE*), + MYF(MY_WME|MY_ZEROFILL)); + if (unlikely(!id_to_share)) + DBUG_RETURN(1); + id_to_share--; /* min id is 1 */ translog_inited= 1; DBUG_RETURN(0); } @@ -2303,6 +2328,8 @@ void translog_destroy() } pthread_mutex_destroy(&log_descriptor.sent_to_file_lock); my_close(log_descriptor.directory_fd, MYF(MY_WME)); + my_atomic_rwlock_destroy(&LOCK_id_to_share); + my_free((gptr)(id_to_share + 1), MYF(MY_ALLOW_ZERO_PTR)); translog_inited= 0; } DBUG_VOID_RETURN; @@ -2362,6 +2389,14 @@ static inline my_bool translog_unlock() } +#define translog_buffer_lock_assert_owner(B) \ + safe_mutex_assert_owner(&B->mutex); +void translog_lock_assert_owner() +{ + translog_buffer_lock_assert_owner(log_descriptor.bc.buffer); +} + + /* Start new page @@ -4154,26 +4189,30 @@ err: } -/* - Write the log record - - SYNOPSIS - translog_write_record() - lsn LSN of the record will be written here - type the log record type - trn Transaction structure pointer for hooks by - record log type, for short_id - share MARIA_SHARE of table or NULL - rec_len record length or 0 (count it) - part_no number of parts or 0 (count it) - parts_data zero ended (in case of number of parts is 0) - array of LEX_STRINGs (parts), first - TRANSLOG_INTERNAL_PARTS positions in the log - should be unused (need for loghandler) - - RETURN - 0 OK - 1 Error +/** + @brief Writes the log record + + If share has no 2-byte-id yet, gives an id to the share and logs + LOGREC_FILE_ID. If transaction has not logged LOGREC_LONG_TRANSACTION_ID + yet, logs it. + + @param lsn LSN of the record will be written here + @param type the log record type + @param trn Transaction structure pointer for hooks by + record log type, for short_id + @param share MARIA_SHARE of table or NULL + @param rec_len record length or 0 (count it) + @param part_no number of parts or 0 (count it) + @param parts_data zero ended (in case of number of parts is 0) + array of LEX_STRINGs (parts), first + TRANSLOG_INTERNAL_PARTS positions in the log + should be unused (need for loghandler) + @param store_share_id if share!=NULL then share's id will automatically + be stored in the two first bytes pointed (so + pointer is assumed to be !=NULL) + @return Operation status + @retval 0 OK + @retval 1 Error */ my_bool translog_write_record(LSN *lsn, @@ -4181,7 +4220,8 @@ my_bool translog_write_record(LSN *lsn, TRN *trn, struct st_maria_share *share, translog_size_t rec_len, uint part_no, - LEX_STRING *parts_data) + LEX_STRING *parts_data, + uchar *store_share_id) { struct st_translog_parts parts; LEX_STRING *part; @@ -4191,10 +4231,41 @@ my_bool translog_write_record(LSN *lsn, DBUG_PRINT("enter", ("type: %u ShortTrID: %u", (uint) type, (uint)short_trid)); - if (share && !share->base.transactional) + if (share) { - DBUG_PRINT("info", ("It is not transactional table")); - DBUG_RETURN(0); + if (!share->base.transactional) + { + DBUG_PRINT("info", ("It is not transactional table")); + DBUG_RETURN(0); + } + if (unlikely(share->id == 0)) + { + /* + First log write for this MARIA_SHARE; give it a short id. + When the lock manager is enabled and needs a short id, it should be + assigned in the lock manager (because row locks will be taken before + log records are written; for example SELECT FOR UPDATE takes locks but + writes no log record. + */ + if (unlikely(translog_assign_id_to_share(share, trn))) + DBUG_RETURN(1); + } + fileid_store(store_share_id, share->id); + } + if (unlikely(!(trn->first_undo_lsn & TRANSACTION_LOGGED_LONG_ID))) + { + LSN lsn; + LEX_STRING log_array[TRANSLOG_INTERNAL_PARTS + 1]; + uchar log_data[6]; + int6store(log_data, trn->trid); + log_array[TRANSLOG_INTERNAL_PARTS + 0].str= (char*) log_data; + log_array[TRANSLOG_INTERNAL_PARTS + 0].length= sizeof(log_data); + trn->first_undo_lsn|= TRANSACTION_LOGGED_LONG_ID; /* no recursion */ + if (unlikely(translog_write_record(&lsn, LOGREC_LONG_TRANSACTION_ID, + trn, NULL, sizeof(log_data), + sizeof(log_array)/sizeof(log_array[0]), + log_array, NULL))) + DBUG_RETURN(1); } parts.parts= parts_data; @@ -4375,20 +4446,19 @@ void translog_free_record_header(TRANSLOG_HEADER_BUFFER *buff) } -/* - Set current horizon in the scanner data structure +/** + @brief Returns the current horizon at the end of the current log - SYNOPSIS - translog_scanner_set_horizon() - scanner Information about current chunk during scanning + @return Horizon */ -static void translog_scanner_set_horizon(struct st_translog_scanner_data - *scanner) +TRANSLOG_ADDRESS translog_get_horizon() { + TRANSLOG_ADDRESS res; translog_lock(); - scanner->horizon= log_descriptor.horizon; + res= log_descriptor.horizon; translog_unlock(); + return res; } @@ -4446,7 +4516,7 @@ my_bool translog_init_scanner(LSN lsn, scanner->fixed_horizon= fixed_horizon; - translog_scanner_set_horizon(scanner); + scanner->horizon= translog_get_horizon(); DBUG_PRINT("info", ("horizon: (0x%lu,0x%lx)", (ulong) LSN_FILE_NO(scanner->horizon), (ulong) LSN_OFFSET(scanner->horizon))); @@ -4499,7 +4569,7 @@ static my_bool translog_scanner_eol(TRANSLOG_SCANNER_DATA *scanner) DBUG_PRINT("info", ("Horizon is fixed and reached")); DBUG_RETURN(1); } - translog_scanner_set_horizon(scanner); + scanner->horizon= translog_get_horizon(); DBUG_PRINT("info", ("Horizon is re-read, EOL: %d", scanner->horizon <= (scanner->page_addr + @@ -5368,17 +5438,31 @@ static void translog_force_current_buffer_to_finish() } -/* - Flush the log up to given LSN (included) - - SYNOPSIS - translog_flush() - lsn log record serial number up to which (inclusive) - the log have to be flushed - - RETURN - 0 OK - 1 Error +/** + @brief Flush the log up to given LSN (included) + + @param lsn log record serial number up to which (inclusive) + the log has to be flushed + + @return Operation status + @retval 0 OK + @retval 1 Error + + @todo LOG: when a log write fails, we should not write to this log anymore + (if we add more log records to this log they will be unreadable: we will hit + the broken log record): all translog_flush() should be made to fail (because + translog_flush() is when a a transaction wants something durable and we + cannot make anything durable as log is corrupted). For that, a "my_bool + st_translog_descriptor::write_error" could be set to 1 when a + translog_write_record() or translog_flush() fails, and translog_flush() + would test this var (and translog_write_record() could also test this var if + it wants, though it's not absolutely needed). + Then, either shut Maria down immediately, or switch to a new log (but if we + get write error after write error, that would create too many logs). + A popular open-source transactional engine intentionally crashes as soon as + a log flush fails (we however don't want to crash the entire mysqld, but + stopping all engine's operations immediately would make sense). + Same applies to translog_write_record(). */ my_bool translog_flush(LSN lsn) @@ -5469,24 +5553,55 @@ my_bool translog_flush(LSN lsn) /* We sync file when we are closing it => do nothing if file closed */ } log_descriptor.flushed= sent_to_file; + /** @todo LOG decide if syncing of directory is needed */ rc|= my_sync(log_descriptor.directory_fd, MYF(MY_WME)); translog_unlock(); DBUG_RETURN(rc); } +/** + @brief Sets transaction's rec_lsn if needed + + A transaction sometimes writes a REDO even before the page is in the + pagecache (example: brand new head or tail pages; full pages). So, if + Checkpoint happens just after the REDO write, it needs to know that the + REDO phase must start before this REDO. Scanning the pagecache cannot + tell that as the page is not in the cache. So, transaction sets its rec_lsn + to the REDO's LSN or somewhere before, and Checkpoint reads the + transaction's rec_lsn. + + @todo move it to a separate file + + @return Operation status, always 0 (success) +*/ + static my_bool write_hook_for_redo(enum translog_record_type type __attribute__ ((unused)), TRN *trn, LSN *lsn, struct st_translog_parts *parts __attribute__ ((unused))) { + /* + If the hook stays so simple, it would be faster to pass + !trn->rec_lsn ? trn->rec_lsn : some_dummy_lsn + to translog_write_record(), like Monty did in his original code, and not + have a hook. For now we keep it like this. + */ if (trn->rec_lsn == 0) trn->rec_lsn= *lsn; return 0; } +/** + @brief Sets transaction's undo_lsn, first_undo_lsn if needed + + @todo move it to a separate file + + @return Operation status, always 0 (success) +*/ + static my_bool write_hook_for_undo(enum translog_record_type type __attribute__ ((unused)), TRN *trn, LSN *lsn, @@ -5494,11 +5609,109 @@ static my_bool write_hook_for_undo(enum translog_record_type type __attribute__ ((unused))) { trn->undo_lsn= *lsn; - if (trn->first_undo_lsn == 0) - trn->first_undo_lsn= *lsn; + if (unlikely(LSN_WITH_FLAGS_TO_LSN(trn->first_undo_lsn) == 0)) + trn->first_undo_lsn= + trn->undo_lsn | LSN_WITH_FLAGS_TO_FLAGS(trn->first_undo_lsn); return 0; /* when we implement purging, we will specialize this hook: UNDO_PURGE records will additionally set trn->undo_purge_lsn */ } + + +/** + @brief Gives a 2-byte-id to MARIA_SHARE and logs this fact + + If a MARIA_SHARE does not yet have a 2-byte-id (unique over all currently + open MARIA_SHAREs), give it one and record this assignment in the log + (LOGREC_FILE_ID log record). + + @param share table + @param trn calling transaction + + @return Operation status + @retval 0 OK + @retval 1 Error + + @note Can be called even if share already has an id (then will do nothing) +*/ + +int translog_assign_id_to_share(MARIA_SHARE *share, TRN *trn) +{ + /* + If you give an id to a non-BLOCK_RECORD table, you also need to release + this id somewhere. Then you can change the assertion. + */ + DBUG_ASSERT(share->data_file_type == BLOCK_RECORD); + /* re-check under mutex to avoid having 2 ids for the same share */ + pthread_mutex_lock(&share->intern_lock); + if (likely(share->id == 0)) + { + /* Inspired by set_short_trid() of trnman.c */ + int i= share->kfile.file % SHARE_ID_MAX + 1; + my_atomic_rwlock_wrlock(&LOCK_id_to_share); + /** + @todo RECOVERY BUG: if all slots are used, and we're using rwlocks + above, we will never exit the loop. To be discussed with Serg. + */ + for ( ; ; i= i % SHARE_ID_MAX + 1) /* the range is [1..SHARE_ID_MAX] */ + { + void *tmp= NULL; + if (id_to_share[i] == NULL && + my_atomic_casptr((void **)&id_to_share[i], &tmp, share)) + break; + } + my_atomic_rwlock_wrunlock(&LOCK_id_to_share); + share->id= (uint16)i; + DBUG_PRINT("info", ("id_to_share: 0x%lx -> %u", (ulong)share, i)); + LSN lsn; + LEX_STRING log_array[TRANSLOG_INTERNAL_PARTS + 2]; + uchar log_data[FILEID_STORE_SIZE]; + log_array[TRANSLOG_INTERNAL_PARTS + 0].str= (char*) log_data; + log_array[TRANSLOG_INTERNAL_PARTS + 0].length= sizeof(log_data); + /* + open_file_name is an unresolved name (symlinks are not resolved, datadir + is not realpath-ed, etc) which is good: the log can be moved to another + directory and continue working. + */ + log_array[TRANSLOG_INTERNAL_PARTS + 1].str= share->open_file_name; + /** + @todo if we had the name's length in MARIA_SHARE we could avoid this + strlen() + */ + log_array[TRANSLOG_INTERNAL_PARTS + 1].length= + strlen(share->open_file_name); + if (unlikely(translog_write_record(&lsn, LOGREC_FILE_ID, trn, share, + sizeof(log_data) + + log_array[TRANSLOG_INTERNAL_PARTS + + 1].length, + sizeof(log_array)/sizeof(log_array[0]), + log_array, log_data))) + return 1; + } + pthread_mutex_unlock(&share->intern_lock); + return 0; +} + + +/** + @brief Recycles a MARIA_SHARE's short id. + + @param share table + + @note Must be called only if share has an id (i.e. id != 0) +*/ + +void translog_deassign_id_from_share(MARIA_SHARE *share) +{ + DBUG_PRINT("info", ("id_to_share: 0x%lx id %u -> 0", + (ulong)share, share->id)); + /* + We don't need any mutex as we are called only when closing the last + instance of the table: no writes can be happening. + */ + my_atomic_rwlock_rdlock(&LOCK_id_to_share); + my_atomic_storeptr((void **)&id_to_share[share->id], 0); + my_atomic_rwlock_rdunlock(&LOCK_id_to_share); +} diff --git a/storage/maria/ma_loghandler.h b/storage/maria/ma_loghandler.h index e9872e7bfb7..0a160a9bc53 100644 --- a/storage/maria/ma_loghandler.h +++ b/storage/maria/ma_loghandler.h @@ -86,13 +86,12 @@ enum translog_record_type LOGREC_PREPARE_WITH_UNDO_PURGE, LOGREC_COMMIT, LOGREC_COMMIT_WITH_UNDO_PURGE, - LOGREC_CHECKPOINT_PAGE, - LOGREC_CHECKPOINT_TRAN, - LOGREC_CHECKPOINT_TABL, + LOGREC_CHECKPOINT, LOGREC_REDO_CREATE_TABLE, LOGREC_REDO_RENAME_TABLE, LOGREC_REDO_DROP_TABLE, - LOGREC_REDO_TRUNCATE_TABLE, + LOGREC_REDO_DELETE_ALL, + LOGREC_REDO_REPAIR_TABLE, LOGREC_FILE_ID, LOGREC_LONG_TRANSACTION_ID, LOGREC_RESERVED_FUTURE_EXTENSION= 63 @@ -181,9 +180,7 @@ struct st_translog_reader_data }; struct st_transaction; -#ifdef __cplusplus -extern "C" { -#endif +C_MODE_START /* Records types for unittests */ #define LOGREC_FIXED_RECORD_0LSN_EXAMPLE 1 @@ -199,13 +196,12 @@ extern my_bool translog_init(const char *directory, uint32 log_file_max_size, uint32 server_version, uint32 server_id, PAGECACHE *pagecache, uint flags); -extern my_bool translog_write_record(LSN *lsn, - enum translog_record_type type, - struct st_transaction *trn, - struct st_maria_share *share, - translog_size_t rec_len, - uint part_no, - LEX_STRING *parts_data); +extern my_bool +translog_write_record(LSN *lsn, enum translog_record_type type, + struct st_transaction *trn, + struct st_maria_share *share, + translog_size_t rec_len, uint part_no, + LEX_STRING *parts_data, uchar *store_share_id); extern void translog_destroy(); @@ -232,7 +228,10 @@ extern translog_size_t translog_read_next_record_header(TRANSLOG_SCANNER_DATA *scanner, TRANSLOG_HEADER_BUFFER *buff); -#ifdef __cplusplus -} -#endif - +extern void translog_lock_assert_owner(); +extern TRANSLOG_ADDRESS translog_get_horizon(); +extern int translog_assign_id_to_share(struct st_maria_share *share, + struct st_transaction *trn); +extern void translog_deassign_id_from_share(struct st_maria_share *share); +extern my_bool translog_inited; +C_MODE_END diff --git a/storage/maria/ma_loghandler_lsn.h b/storage/maria/ma_loghandler_lsn.h index 1789d3ce61b..c641337e8ba 100644 --- a/storage/maria/ma_loghandler_lsn.h +++ b/storage/maria/ma_loghandler_lsn.h @@ -35,7 +35,7 @@ typedef TRANSLOG_ADDRESS LSN; /* checks LSN */ #define LSN_VALID(L) DBUG_ASSERT((L) >= 0 && (L) < (uint64)0xFFFFFFFFFFFFFFLL) -/* size of stored LSN on a disk */ +/* size of stored LSN on a disk, don't change it! */ #define LSN_STORE_SIZE 7 /* Puts LSN into buffer (dst) */ @@ -53,4 +53,12 @@ typedef TRANSLOG_ADDRESS LSN; #define LSN_REPLACE_OFFSET(L, S) (LSN_FINE_NO_PART(L) | (S)) +/* + an 8-byte type whose most significant byte is used for "flags"; 7 + other bytes are a LSN. +*/ +typedef LSN LSN_WITH_FLAGS; +#define LSN_WITH_FLAGS_TO_LSN(x) (x & ULL(0x00FFFFFFFFFFFFFF)) +#define LSN_WITH_FLAGS_TO_FLAGS(x) (x & ULL(0xFF00000000000000)) + #endif diff --git a/storage/maria/ma_open.c b/storage/maria/ma_open.c index b8ce6d123e7..4e72adf3b7e 100644 --- a/storage/maria/ma_open.c +++ b/storage/maria/ma_open.c @@ -919,12 +919,23 @@ static void setup_key_functions(register MARIA_KEYDEF *keyinfo) } -/* - Function to save and store the header in the index file (.MYI) +/** + @brief Function to save and store the header in the index file (.MYI) + + @param file descriptor of the index file to write + @param state state information to write to the file + @param pWrite bitmap (determines the amount of information to + write, and if my_write() or my_pwrite() should be + used) + + @return Operation status + @retval 0 OK + @retval 1 Error */ uint _ma_state_info_write(File file, MARIA_STATE_INFO *state, uint pWrite) { + /** @todo RECOVERY write it only at checkpoint time */ uchar buff[MARIA_STATE_INFO_SIZE + MARIA_STATE_EXTRA_SIZE]; uchar *ptr=buff; uint i, keys= (uint) state->header.keys; @@ -935,6 +946,11 @@ uint _ma_state_info_write(File file, MARIA_STATE_INFO *state, uint pWrite) /* open_count must be first because of _ma_mark_file_changed ! */ mi_int2store(ptr,state->open_count); ptr+= 2; + /* + if you change the offset of this LSN inside the file, fix + ma_create + ma_rename + ma_delete_all + backward-compatibility. + */ + lsn_store(ptr, state->create_rename_lsn); ptr+= LSN_STORE_SIZE; *ptr++= (uchar)state->changed; *ptr++= state->sortkey; mi_rowstore(ptr,state->state.records); ptr+= 8; @@ -959,6 +975,7 @@ uint _ma_state_info_write(File file, MARIA_STATE_INFO *state, uint pWrite) { mi_sizestore(ptr,state->key_root[i]); ptr+= 8; } + /** @todo RECOVERY key_del is a problem for recovery */ mi_sizestore(ptr,state->key_del); ptr+= 8; if (pWrite & 2) /* From maria_chk */ { @@ -994,6 +1011,7 @@ byte *_ma_state_info_read(byte *ptr, MARIA_STATE_INFO *state) key_parts= mi_uint2korr(state->header.key_parts); state->open_count = mi_uint2korr(ptr); ptr+= 2; + state->create_rename_lsn= lsn_korr(ptr); ptr+= LSN_STORE_SIZE; state->changed= (my_bool) *ptr++; state->sortkey= (uint) *ptr++; state->state.records= mi_rowkorr(ptr); ptr+= 8; diff --git a/storage/maria/ma_pagecache.c b/storage/maria/ma_pagecache.c index 18c36fcfbd1..ae42f702b0a 100755 --- a/storage/maria/ma_pagecache.c +++ b/storage/maria/ma_pagecache.c @@ -114,6 +114,11 @@ /* TODO: put it to my_static.c */ my_bool my_disable_flush_pagecache_blocks= 0; +/** + when flushing pages of a file, it can happen that we take some dirty blocks + out of changed_blocks[]; Checkpoint must not run at this moment. +*/ +uint changed_blocks_is_incomplete= 0; #define STRUCT_PTR(TYPE, MEMBER, a) \ (TYPE *) ((char *) (a) - offsetof(TYPE, MEMBER)) @@ -308,7 +313,7 @@ struct st_pagecache_block_link enum pagecache_page_type type; /* type of the block */ uint hits_left; /* number of hits left until promotion */ ulonglong last_hit_time; /* timestamp of the last hit */ - LSN rec_lsn; /* LSN when first became dirty */ + LSN rec_lsn; /**< LSN when first became dirty */ KEYCACHE_CONDVAR *condvar; /* condition variable for 'no readers' event */ }; @@ -2523,7 +2528,8 @@ void pagecache_unlock(PAGECACHE *pagecache, { DBUG_ASSERT(lock == PAGECACHE_LOCK_WRITE_UNLOCK); DBUG_ASSERT(pin == PAGECACHE_UNPIN); - set_if_bigger(block->rec_lsn, first_REDO_LSN_for_page); + if (block->rec_lsn == 0) + block->rec_lsn= first_REDO_LSN_for_page; } if (lsn != 0) { @@ -2685,7 +2691,8 @@ void pagecache_unlock_by_link(PAGECACHE *pagecache, DBUG_ASSERT(lock == PAGECACHE_LOCK_WRITE_UNLOCK || lock == PAGECACHE_LOCK_READ_UNLOCK); DBUG_ASSERT(pin == PAGECACHE_UNPIN); - set_if_bigger(block->rec_lsn, first_REDO_LSN_for_page); + if (block->rec_lsn == 0) + block->rec_lsn= first_REDO_LSN_for_page; } if (lsn != 0) { @@ -3279,8 +3286,8 @@ restart: if (need_lock_change) { /* - RECOVERY TODO BUG We are doing an unlock here, so need to give the - page its rec_lsn + We don't set rec_lsn of the block; this is ok as for the + Maria-block-record's pages, we always keep pages pinned here. */ if (make_lock_and_pin(pagecache, block, write_lock_change_table[lock].unlock_lock, @@ -3500,22 +3507,21 @@ static int flush_cached_blocks(PAGECACHE *pagecache, } -/* - flush all key blocks for a file to disk, but don't do any mutex locks +/** + @brief flush all key blocks for a file to disk but don't do any mutex locks - flush_pagecache_blocks_int() - pagecache pointer to a key cache data structure - file handler for the file to flush to - flush_type type of the flush + @param pagecache pointer to a pagecache data structure + @param file handler for the file to flush to + @param flush_type type of the flush - NOTES - This function doesn't do any mutex locks because it needs to be called - both from flush_pagecache_blocks and flush_all_key_blocks (the later one - does the mutex lock in the resize_pagecache() function). + @note + This function doesn't do any mutex locks because it needs to be called + both from flush_pagecache_blocks and flush_all_key_blocks (the later one + does the mutex lock in the resize_pagecache() function). - RETURN - 0 ok - 1 error + @return Operation status + @retval 0 OK + @retval 1 Error */ static int flush_pagecache_blocks_int(PAGECACHE *pagecache, @@ -3547,6 +3553,7 @@ static int flush_pagecache_blocks_int(PAGECACHE *pagecache, #if defined(PAGECACHE_DEBUG) uint cnt= 0; #endif + uint8 changed_blocks_is_incomplete_incremented= 0; if (type != FLUSH_IGNORE_CHANGED) { @@ -3636,16 +3643,23 @@ restart: else { /* Link the block into a list of blocks 'in switch' */ - /* - RECOVERY TODO BUG this unlink_changed() is a serious problem for - Maria's Checkpoint: it removes a page from the list of dirty - pages, while it's still dirty. A solution is to abandon - first_in_switch, just wait for this page to be - flushed by somebody else, and loop. TODO: check all places - where we remove a page from the list of dirty pages - */ unlink_changed(block); link_changed(block, &first_in_switch); + /* + We have just removed a page from the list of dirty pages + ("changed_blocks") though it's still dirty (the flush by another + thread has not yet happened). Checkpoint will miss the page and so + must be blocked until that flush has happened. + */ + /** + @todo RECOVERY: check all places where we remove a page from the + list of dirty pages + */ + if (unlikely(!changed_blocks_is_incomplete_incremented)) + { + changed_blocks_is_incomplete_incremented= 1; + changed_blocks_is_incomplete++; + } } } } @@ -3683,6 +3697,8 @@ restart: KEYCACHE_DBUG_ASSERT(cnt <= pagecache->blocks_used); #endif } + changed_blocks_is_incomplete-= + changed_blocks_is_incomplete_incremented; /* The following happens very seldom */ if (! (type == FLUSH_KEEP || type == FLUSH_FORCE_WRITE)) { @@ -3789,51 +3805,56 @@ int reset_pagecache_counters(const char *name, PAGECACHE *pagecache) } -/* - Allocates a buffer and stores in it some information about all dirty pages - of type PAGECACHE_LSN_PAGE. - - SYNOPSIS - pagecache_collect_changed_blocks_with_lsn() - pagecache pointer to the page cache - str (OUT) pointer to a LEX_STRING where the allocated buffer, and - its size, will be put - max_lsn (OUT) pointer to a LSN where the maximum rec_lsn of all - relevant dirty pages will be put - - DESCRIPTION - Does the allocation because the caller cannot know the size itself. - Memory freeing is to be done by the caller (if the "str" member of the - LEX_STRING is not NULL). - Ignores all pages of another type than PAGECACHE_LSN_PAGE, because they - are not interesting for a checkpoint record. - The caller has the intention of doing checkpoints. - - RETURN - 0 on success - 1 on error +/** + @brief Allocates a buffer and stores in it some info about all dirty pages + + Does the allocation because the caller cannot know the size itself. + Memory freeing is to be done by the caller (if the "str" member of the + LEX_STRING is not NULL). + Ignores all pages of another type than PAGECACHE_LSN_PAGE, because they + are not interesting for a checkpoint record. + The caller has the intention of doing checkpoints. + + @param pagecache pointer to the page cache + @param[out] str pointer to where the allocated buffer, and + its size, will be put + @param[out] min_rec_lsn pointer to where the minimum rec_lsn of all + relevant dirty pages will be put + @param[out] max_rec_lsn pointer to where the maximum rec_lsn of all + relevant dirty pages will be put + @return Operation status + @retval 0 OK + @retval 1 Error */ + my_bool pagecache_collect_changed_blocks_with_lsn(PAGECACHE *pagecache, LEX_STRING *str, - LSN *max_lsn) + LSN *min_rec_lsn, + LSN *max_rec_lsn) { my_bool error= 0; ulong stored_list_size= 0; uint file_hash; char *ptr; + LSN minimum_rec_lsn= ULONGLONG_MAX, maximum_rec_lsn= 0; DBUG_ENTER("pagecache_collect_changed_blocks_with_LSN"); - *max_lsn= 0; DBUG_ASSERT(NULL == str->str); /* We lock the entire cache but will be quick, just reading/writing a few MBs of memory at most. - When we enter here, we must be sure that no "first_in_switch" situation - is happening or will happen (either we have to get rid of - first_in_switch in the code or, first_in_switch has to increment a - "danger" counter for this function to know it has to wait). TODO. */ pagecache_pthread_mutex_lock(&pagecache->cache_lock); + while (changed_blocks_is_incomplete > 0) + { + /* + Some pages are more recent in memory than on disk (=dirty) and are not + in "changed_blocks" so we cannot know them. Wait. + */ + pagecache_pthread_mutex_unlock(&pagecache->cache_lock); + sleep(1); + pagecache_pthread_mutex_lock(&pagecache->cache_lock); + } /* Count how many dirty pages are interesting */ for (file_hash= 0; file_hash < PAGECACHE_CHANGED_BLOCKS_HASH; file_hash++) @@ -3851,35 +3872,15 @@ my_bool pagecache_collect_changed_blocks_with_lsn(PAGECACHE *pagecache, DBUG_ASSERT(block->status & PCBLOCK_CHANGED); if (block->type != PAGECACHE_LSN_PAGE) continue; /* no need to store it */ - /* - In the current pagecache, rec_lsn is not set correctly: - 1) it is set on pagecache_unlock(), too late (a page is dirty - (PCBLOCK_CHANGED) since the first pagecache_write()). So in this - scenario: - thread1: thread2: - write_REDO - pagecache_write() checkpoint : reclsn not known - pagecache_unlock(sets rec_lsn) - commit - crash, - at recovery we will wrongly skip the REDO. It also affects the - low-water mark's computation. - 2) sometimes the unlocking can be an implicit action of - pagecache_write(), without any call to pagecache_unlock(), then - rec_lsn is not set. - 1) and 2) are critical problems. - TODO: fix this when Monty has explained how he writes BLOB pages. - */ - if (block->rec_lsn == 0) - { - DBUG_ASSERT(0); - goto err; - } stored_list_size++; } } - str->length= 8+(4+4+8)*stored_list_size; + str->length= 8 + /* number of dirty pages */ + (4 + /* file */ + 4 + /* pageno */ + LSN_STORE_SIZE /* rec_lsn */ + ) * stored_list_size; if (NULL == (str->str= my_malloc(str->length, MYF(MY_WME)))) goto err; ptr= str->str; @@ -3896,19 +3897,27 @@ my_bool pagecache_collect_changed_blocks_with_lsn(PAGECACHE *pagecache, { if (block->type != PAGECACHE_LSN_PAGE) continue; /* no need to store it in the checkpoint record */ - DBUG_ASSERT((4 == sizeof(block->hash_link->file.file))); - DBUG_ASSERT((4 == sizeof(block->hash_link->pageno))); + compile_time_assert((4 == sizeof(block->hash_link->file.file))); + compile_time_assert((4 == sizeof(block->hash_link->pageno))); int4store(ptr, block->hash_link->file.file); ptr+= 4; int4store(ptr, block->hash_link->pageno); ptr+= 4; - int8store(ptr, (ulonglong) block->rec_lsn); - ptr+= 8; - set_if_bigger(*max_lsn, block->rec_lsn); + lsn_store(ptr, block->rec_lsn); + ptr+= LSN_STORE_SIZE; + if (block->rec_lsn != 0) + { + if (cmp_translog_addr(block->rec_lsn, minimum_rec_lsn) < 0) + minimum_rec_lsn= block->rec_lsn; + if (cmp_translog_addr(block->rec_lsn, maximum_rec_lsn) > 0) + maximum_rec_lsn= block->rec_lsn; + } /* otherwise, some trn->rec_lsn should hold the info */ } } end: pagecache_pthread_mutex_unlock(&pagecache->cache_lock); + *min_rec_lsn= minimum_rec_lsn; + *max_rec_lsn= maximum_rec_lsn; DBUG_RETURN(error); err: diff --git a/storage/maria/ma_pagecache.h b/storage/maria/ma_pagecache.h index ef14cd48cef..478f71161eb 100644 --- a/storage/maria/ma_pagecache.h +++ b/storage/maria/ma_pagecache.h @@ -239,6 +239,7 @@ extern my_bool pagecache_delete_pages(PAGECACHE *pagecache, extern void end_pagecache(PAGECACHE *keycache, my_bool cleanup); extern my_bool pagecache_collect_changed_blocks_with_lsn(PAGECACHE *pagecache, LEX_STRING *str, + LSN *min_lsn, LSN *max_lsn); extern int reset_pagecache_counters(const char *name, PAGECACHE *pagecache); diff --git a/storage/maria/ma_panic.c b/storage/maria/ma_panic.c index b74403e6eb2..0394f630343 100644 --- a/storage/maria/ma_panic.c +++ b/storage/maria/ma_panic.c @@ -52,7 +52,12 @@ int maria_panic(enum ha_panic_function flag) info=(MARIA_HA*) list_element->data; switch (flag) { case HA_PANIC_CLOSE: - pthread_mutex_unlock(&THR_LOCK_maria); /* Not exactly right... */ + /* + If bad luck (if some tables would be used now, which normally does not + happen in MySQL), as we release the mutex, the list may change and so + we may crash. + */ + pthread_mutex_unlock(&THR_LOCK_maria); if (maria_close(info)) error=my_errno; pthread_mutex_lock(&THR_LOCK_maria); diff --git a/storage/maria/ma_range.c b/storage/maria/ma_range.c index f91a61259d7..b359868e8e4 100644 --- a/storage/maria/ma_range.c +++ b/storage/maria/ma_range.c @@ -29,25 +29,22 @@ static uint _ma_keynr(MARIA_HA *info, MARIA_KEYDEF *keyinfo, byte *page, byte *keypos, uint *ret_max_key); -/* - Estimate how many records there is in a given range +/** + @brief Estimate how many records there is in a given range - SYNOPSIS - maria_records_in_range() - info MARIA handler - inx Index to use - min_key Min key. Is = 0 if no min range - max_key Max key. Is = 0 if no max range + @param info MARIA handler + @param inx Index to use + @param min_key Min key. Is = 0 if no min range + @param max_key Max key. Is = 0 if no max range - NOTES - We should ONLY return 0 if there is no rows in range + @note + We should ONLY return 0 if there is no rows in range - RETURN - HA_POS_ERROR error (or we can't estimate number of rows) - number Estimated number of rows + @return Estimated number of rows or error + @retval HA_POS_ERROR error (or we can't estimate number of rows) + @retval number Estimated number of rows */ - ha_rows maria_records_in_range(MARIA_HA *info, int inx, key_range *min_key, key_range *max_key) { @@ -115,6 +112,13 @@ ha_rows maria_records_in_range(MARIA_HA *info, int inx, key_range *min_key, rw_unlock(&info->s->key_root_lock[inx]); fast_ma_writeinfo(info); + /** + @todo LOCK + If res==0 (no rows), if we need to guarantee repeatability of the search, + we will need to set a next-key lock in this statement. + Also SELECT COUNT(*)... + */ + DBUG_PRINT("info",("records: %ld",(ulong) (res))); DBUG_RETURN(res); } diff --git a/storage/maria/ma_rename.c b/storage/maria/ma_rename.c index a80bbcd398f..5224698c614 100644 --- a/storage/maria/ma_rename.c +++ b/storage/maria/ma_rename.c @@ -18,6 +18,18 @@ */ #include "ma_fulltext.h" +#include "trnman_public.h" + +/** + @brief renames a table + + @param old_name current name of table + @param new_name table should be renamed to this name + + @return Operation status + @retval 0 OK + @retval !=0 Error +*/ int maria_rename(const char *old_name, const char *new_name) { @@ -26,22 +38,73 @@ int maria_rename(const char *old_name, const char *new_name) #ifdef USE_RAID uint raid_type=0,raid_chunks=0; #endif + MARIA_HA *info; + MARIA_SHARE *share; + myf sync_dir; DBUG_ENTER("maria_rename"); #ifdef EXTRA_DEBUG _ma_check_table_is_closed(old_name,"rename old_table"); _ma_check_table_is_closed(new_name,"rename new table2"); #endif - /* LOCK TODO take X-lock on table here */ + /** @todo LOCK take X-lock on table */ + if (!(info= maria_open(old_name, O_RDWR, HA_OPEN_FOR_REPAIR))) + DBUG_RETURN(my_errno); + share= info->s; #ifdef USE_RAID + raid_type = share->base.raid_type; + raid_chunks = share->base.raid_chunks; +#endif + + sync_dir= (share->base.transactional && !share->temporary) ? + MY_SYNC_DIR : 0; + if (sync_dir) { - MARIA_HA *info; - if (!(info=maria_open(old_name, O_RDONLY, 0))) - DBUG_RETURN(my_errno); - raid_type = info->s->base.raid_type; - raid_chunks = info->s->base.raid_chunks; - maria_close(info); + uchar log_data[LSN_STORE_SIZE]; + LEX_STRING log_array[TRANSLOG_INTERNAL_PARTS + 3]; + uint old_name_len= strlen(old_name), new_name_len= strlen(new_name); + int2store(log_data, old_name_len); + int2store(log_data + 2, new_name_len); + log_array[TRANSLOG_INTERNAL_PARTS + 0].str= log_data; + log_array[TRANSLOG_INTERNAL_PARTS + 0].length= 2 + 2; + log_array[TRANSLOG_INTERNAL_PARTS + 1].str= (char *)old_name; + log_array[TRANSLOG_INTERNAL_PARTS + 1].length= old_name_len; + log_array[TRANSLOG_INTERNAL_PARTS + 2].str= (char *)new_name; + log_array[TRANSLOG_INTERNAL_PARTS + 2].length= new_name_len; + /* + For this record to be of any use for Recovery, we need the upper + MySQL layer to be crash-safe, which it is not now (that would require + work using the ddl_log of sql/sql_table.cc); when it is, we should + reconsider the moment of writing this log record (before or after op, + under THR_LOCK_maria or not...), how to use it in Recovery, and force + the log. For now this record is just informative. + */ + if (unlikely(translog_write_record(&share->state.create_rename_lsn, + LOGREC_REDO_RENAME_TABLE, + &dummy_transaction_object, NULL, + 2 + 2 + old_name_len + new_name_len, + sizeof(log_array)/sizeof(log_array[0]), + log_array, NULL))) + { + maria_close(info); + DBUG_RETURN(1); + } + /* + store LSN into file, needed for Recovery to not be confused if a + RENAME happened (applying REDOs to the wrong table). + */ + lsn_store(log_data, share->state.create_rename_lsn); + if (my_pwrite(share->kfile.file, log_data, sizeof(log_data), + sizeof(share->state.header) + 2, MYF(MY_NABP)) || + my_sync(share->kfile.file, MYF(MY_WME))) + { + maria_close(info); + DBUG_RETURN(1); + } } + + maria_close(info); +#ifdef USE_RAID #ifdef EXTRA_DEBUG _ma_check_table_is_closed(old_name,"rename raidcheck"); #endif @@ -49,29 +112,18 @@ int maria_rename(const char *old_name, const char *new_name) fn_format(from,old_name,"",MARIA_NAME_IEXT,MY_UNPACK_FILENAME|MY_APPEND_EXT); fn_format(to,new_name,"",MARIA_NAME_IEXT,MY_UNPACK_FILENAME|MY_APPEND_EXT); - /* - RECOVERY TODO log the two renames below. Update - ZeroDirtyPagesLSN of the table on disk (=> sync the files), this is - needed so that Recovery does not pick a wrong table. - Then do the file renames. - For this log record to be of any use for Recovery, we need the upper MySQL - layer to be crash-safe in DDLs; when it is we should reconsider the moment - of writing this log record, how to use it in Recovery, and force the log. - For now this record is only informative. But ZeroDirtyPagesLSN is - critically needed! - */ - if (my_rename_with_symlink(from, to, MYF(MY_WME | MY_SYNC_DIR))) + if (my_rename_with_symlink(from, to, MYF(MY_WME | sync_dir))) DBUG_RETURN(my_errno); fn_format(from,old_name,"",MARIA_NAME_DEXT,MY_UNPACK_FILENAME|MY_APPEND_EXT); fn_format(to,new_name,"",MARIA_NAME_DEXT,MY_UNPACK_FILENAME|MY_APPEND_EXT); #ifdef USE_RAID if (raid_type) data_file_rename_error= my_raid_rename(from, to, raid_chunks, - MYF(MY_WME | MY_SYNC_DIR)); + MYF(MY_WME | sync_dir)); else #endif data_file_rename_error= - my_rename_with_symlink(from, to, MYF(MY_WME | MY_SYNC_DIR)); + my_rename_with_symlink(from, to, MYF(MY_WME | sync_dir)); if (data_file_rename_error) { /* @@ -81,7 +133,7 @@ int maria_rename(const char *old_name, const char *new_name) data_file_rename_error= my_errno; fn_format(from, old_name, "", MARIA_NAME_IEXT, MYF(MY_UNPACK_FILENAME|MY_APPEND_EXT)); fn_format(to, new_name, "", MARIA_NAME_IEXT, MYF(MY_UNPACK_FILENAME|MY_APPEND_EXT)); - my_rename_with_symlink(to, from, MYF(MY_WME | MY_SYNC_DIR)); + my_rename_with_symlink(to, from, MYF(MY_WME | sync_dir)); } DBUG_RETURN(data_file_rename_error); diff --git a/storage/maria/ma_static.c b/storage/maria/ma_static.c index c77f3f512fd..16bf0eca935 100644 --- a/storage/maria/ma_static.c +++ b/storage/maria/ma_static.c @@ -47,7 +47,13 @@ PAGECACHE *maria_pagecache= &maria_pagecache_var; PAGECACHE maria_log_pagecache_var; PAGECACHE *maria_log_pagecache= &maria_log_pagecache_var; -/* For using maria externally */ +/** + @brief when transactionality does not matter we can use this transaction + + Used in external programs like ma_test*, and also internally inside + libmaria when there is no transaction around and the operation isn't + transactional (CREATE/DROP/RENAME/OPTIMIZE/REPAIR). +*/ TRN dummy_transaction_object; /* Enough for comparing if number is zero */ diff --git a/storage/maria/ma_test_all.sh b/storage/maria/ma_test_all.sh index 8ee326a9c69..76b6c32913f 100755 --- a/storage/maria/ma_test_all.sh +++ b/storage/maria/ma_test_all.sh @@ -3,10 +3,16 @@ # Execute some simple basic test on MyISAM libary to check if things # works at all. +# If you want to run this in Valgrind, you should use --trace-children=yes, +# so that it detects problems in ma_test* and not in the shell script valgrind="valgrind --alignment=8 --leak-check=yes" silent="-s" suffix="" #set -x -v -e +if [ -z "$maria_path" ] +then + maria_path="." +fi run_tests() { @@ -14,139 +20,139 @@ run_tests() # # First some simple tests # - ./ma_test1$suffix $silent $row_type - ./maria_chk$suffix -se test1 - ./ma_test1$suffix $silent -N $row_type - ./maria_chk$suffix -se test1 - ./ma_test1$suffix $silent -P --checksum $row_type - ./maria_chk$suffix -se test1 - ./ma_test1$suffix $silent -P -N $row_type - ./maria_chk$suffix -se test1 - ./ma_test1$suffix $silent -B -N -R2 $row_type - ./maria_chk$suffix -sm test1 - ./ma_test1$suffix $silent -a -k 480 --unique $row_type - ./maria_chk$suffix -sm test1 - ./ma_test1$suffix $silent -a -N -R1 $row_type - ./maria_chk$suffix -sm test1 - ./ma_test1$suffix $silent -p $row_type - ./maria_chk$suffix -sm test1 - ./ma_test1$suffix $silent -p -N --unique $row_type - ./maria_chk$suffix -sm test1 - ./ma_test1$suffix $silent -p -N --key_length=127 --checksum $row_type - ./maria_chk$suffix -sm test1 - ./ma_test1$suffix $silent -p -N --key_length=128 $row_type - ./maria_chk$suffix -sm test1 - ./ma_test1$suffix $silent -p --key_length=480 $row_type - ./maria_chk$suffix -sm test1 - ./ma_test1$suffix $silent -a -B $row_type - ./maria_chk$suffix -sm test1 - ./ma_test1$suffix $silent -a -B --key_length=64 --unique $row_type - ./maria_chk$suffix -sm test1 - ./ma_test1$suffix $silent -a -B -k 480 --checksum $row_type - ./maria_chk$suffix -sm test1 - ./ma_test1$suffix $silent -a -B -k 480 -N --unique --checksum $row_type - ./maria_chk$suffix -sm test1 - ./ma_test1$suffix $silent -a -m $row_type - ./maria_chk$suffix -sm test1 - ./ma_test1$suffix $silent -a -m -P --unique --checksum $row_type - ./maria_chk$suffix -sm test1 - ./ma_test1$suffix $silent -a -m -P --key_length=480 --key_cache $row_type - ./maria_chk$suffix -sm test1 - ./ma_test1$suffix $silent -m -p $row_type - ./maria_chk$suffix -sm test1 - ./ma_test1$suffix $silent -w --unique $row_type - ./maria_chk$suffix -sm test1 - ./ma_test1$suffix $silent -a -w --key_length=64 --checksum $row_type - ./maria_chk$suffix -sm test1 - ./ma_test1$suffix $silent -a -w -N --key_length=480 $row_type - ./maria_chk$suffix -sm test1 - ./ma_test1$suffix $silent -a -w --key_length=480 --checksum $row_type - ./maria_chk$suffix -sm test1 - ./ma_test1$suffix $silent -a -b -N $row_type - ./maria_chk$suffix -sm test1 - ./ma_test1$suffix $silent -a -b --key_length=480 $row_type - ./maria_chk$suffix -sm test1 - ./ma_test1$suffix $silent -p -B --key_length=480 $row_type - ./maria_chk$suffix -sm test1 - ./ma_test1$suffix $silent --checksum --unique $row_type - ./maria_chk$suffix -se test1 - ./ma_test1$suffix $silent --unique $row_type - ./maria_chk$suffix -se test1 + $maria_path/ma_test1$suffix $silent $row_type + $maria_path/maria_chk$suffix -se test1 + $maria_path/ma_test1$suffix $silent -N $row_type + $maria_path/maria_chk$suffix -se test1 + $maria_path/ma_test1$suffix $silent -P --checksum $row_type + $maria_path/maria_chk$suffix -se test1 + $maria_path/ma_test1$suffix $silent -P -N $row_type + $maria_path/maria_chk$suffix -se test1 + $maria_path/ma_test1$suffix $silent -B -N -R2 $row_type + $maria_path/maria_chk$suffix -sm test1 + $maria_path/ma_test1$suffix $silent -a -k 480 --unique $row_type + $maria_path/maria_chk$suffix -sm test1 + $maria_path/ma_test1$suffix $silent -a -N -R1 $row_type + $maria_path/maria_chk$suffix -sm test1 + $maria_path/ma_test1$suffix $silent -p $row_type + $maria_path/maria_chk$suffix -sm test1 + $maria_path/ma_test1$suffix $silent -p -N --unique $row_type + $maria_path/maria_chk$suffix -sm test1 + $maria_path/ma_test1$suffix $silent -p -N --key_length=127 --checksum $row_type + $maria_path/maria_chk$suffix -sm test1 + $maria_path/ma_test1$suffix $silent -p -N --key_length=128 $row_type + $maria_path/maria_chk$suffix -sm test1 + $maria_path/ma_test1$suffix $silent -p --key_length=480 $row_type + $maria_path/maria_chk$suffix -sm test1 + $maria_path/ma_test1$suffix $silent -a -B $row_type + $maria_path/maria_chk$suffix -sm test1 + $maria_path/ma_test1$suffix $silent -a -B --key_length=64 --unique $row_type + $maria_path/maria_chk$suffix -sm test1 + $maria_path/ma_test1$suffix $silent -a -B -k 480 --checksum $row_type + $maria_path/maria_chk$suffix -sm test1 + $maria_path/ma_test1$suffix $silent -a -B -k 480 -N --unique --checksum $row_type + $maria_path/maria_chk$suffix -sm test1 + $maria_path/ma_test1$suffix $silent -a -m $row_type + $maria_path/maria_chk$suffix -sm test1 + $maria_path/ma_test1$suffix $silent -a -m -P --unique --checksum $row_type + $maria_path/maria_chk$suffix -sm test1 + $maria_path/ma_test1$suffix $silent -a -m -P --key_length=480 --key_cache $row_type + $maria_path/maria_chk$suffix -sm test1 + $maria_path/ma_test1$suffix $silent -m -p $row_type + $maria_path/maria_chk$suffix -sm test1 + $maria_path/ma_test1$suffix $silent -w --unique $row_type + $maria_path/maria_chk$suffix -sm test1 + $maria_path/ma_test1$suffix $silent -a -w --key_length=64 --checksum $row_type + $maria_path/maria_chk$suffix -sm test1 + $maria_path/ma_test1$suffix $silent -a -w -N --key_length=480 $row_type + $maria_path/maria_chk$suffix -sm test1 + $maria_path/ma_test1$suffix $silent -a -w --key_length=480 --checksum $row_type + $maria_path/maria_chk$suffix -sm test1 + $maria_path/ma_test1$suffix $silent -a -b -N $row_type + $maria_path/maria_chk$suffix -sm test1 + $maria_path/ma_test1$suffix $silent -a -b --key_length=480 $row_type + $maria_path/maria_chk$suffix -sm test1 + $maria_path/ma_test1$suffix $silent -p -B --key_length=480 $row_type + $maria_path/maria_chk$suffix -sm test1 + $maria_path/ma_test1$suffix $silent --checksum --unique $row_type + $maria_path/maria_chk$suffix -se test1 + $maria_path/ma_test1$suffix $silent --unique $row_type + $maria_path/maria_chk$suffix -se test1 - ./ma_test1$suffix $silent --key_multiple -N -S $row_type - ./maria_chk$suffix -sm test1 - ./ma_test1$suffix $silent --key_multiple -a -p --key_length=480 $row_type - ./maria_chk$suffix -sm test1 - ./ma_test1$suffix $silent --key_multiple -a -B --key_length=480 $row_type - ./maria_chk$suffix -sm test1 - ./ma_test1$suffix $silent --key_multiple -P -S $row_type - ./maria_chk$suffix -sm test1 + $maria_path/ma_test1$suffix $silent --key_multiple -N -S $row_type + $maria_path/maria_chk$suffix -sm test1 + $maria_path/ma_test1$suffix $silent --key_multiple -a -p --key_length=480 $row_type + $maria_path/maria_chk$suffix -sm test1 + $maria_path/ma_test1$suffix $silent --key_multiple -a -B --key_length=480 $row_type + $maria_path/maria_chk$suffix -sm test1 + $maria_path/ma_test1$suffix $silent --key_multiple -P -S $row_type + $maria_path/maria_chk$suffix -sm test1 - ./maria_pack$suffix --force -s test1 - ./maria_chk$suffix -ess test1 + $maria_path/maria_pack$suffix --force -s test1 + $maria_path/maria_chk$suffix -ess test1 - ./ma_test2$suffix $silent -L -K -W -P $row_type - ./maria_chk$suffix -sm test2 - ./ma_test2$suffix $silent -L -K -W -P -A $row_type - ./maria_chk$suffix -sm test2 - ./ma_test2$suffix $silent -L -K -P -R3 -m50 -b1000000 $row_type - ./maria_chk$suffix -sm test2 - ./ma_test2$suffix $silent -L -B $row_type - ./maria_chk$suffix -sm test2 - ./ma_test2$suffix $silent -D -B -c $row_type - ./maria_chk$suffix -sm test2 - ./ma_test2$suffix $silent -m10000 -e4096 -K $row_type - ./maria_chk$suffix -sm test2 - ./ma_test2$suffix $silent -m10000 -e8192 -K $row_type - ./maria_chk$suffix -sm test2 - ./ma_test2$suffix $silent -m10000 -e16384 -E16384 -K -L $row_type - ./maria_chk$suffix -sm test2 + $maria_path/ma_test2$suffix $silent -L -K -W -P $row_type + $maria_path/maria_chk$suffix -sm test2 + $maria_path/ma_test2$suffix $silent -L -K -W -P -A $row_type + $maria_path/maria_chk$suffix -sm test2 + $maria_path/ma_test2$suffix $silent -L -K -P -R3 -m50 -b1000000 $row_type + $maria_path/maria_chk$suffix -sm test2 + $maria_path/ma_test2$suffix $silent -L -B $row_type + $maria_path/maria_chk$suffix -sm test2 + $maria_path/ma_test2$suffix $silent -D -B -c $row_type + $maria_path/maria_chk$suffix -sm test2 + $maria_path/ma_test2$suffix $silent -m10000 -e4096 -K $row_type + $maria_path/maria_chk$suffix -sm test2 + $maria_path/ma_test2$suffix $silent -m10000 -e8192 -K $row_type + $maria_path/maria_chk$suffix -sm test2 + $maria_path/ma_test2$suffix $silent -m10000 -e16384 -E16384 -K -L $row_type + $maria_path/maria_chk$suffix -sm test2 } run_repair_tests() { row_type=$1 - ./ma_test1$suffix $silent --checksum $row_type - ./maria_chk$suffix -se test1 - ./maria_chk$suffix -rs test1 - ./maria_chk$suffix -se test1 - ./maria_chk$suffix -rqs test1 - ./maria_chk$suffix -se test1 - ./maria_chk$suffix -rs --correct-checksum test1 - ./maria_chk$suffix -se test1 - ./maria_chk$suffix -rqs --correct-checksum test1 - ./maria_chk$suffix -se test1 - ./maria_chk$suffix -ros --correct-checksum test1 - ./maria_chk$suffix -se test1 - ./maria_chk$suffix -rqos --correct-checksum test1 - ./maria_chk$suffix -se test1 + $maria_path/ma_test1$suffix $silent --checksum $row_type + $maria_path/maria_chk$suffix -se test1 + $maria_path/maria_chk$suffix -rs test1 + $maria_path/maria_chk$suffix -se test1 + $maria_path/maria_chk$suffix -rqs test1 + $maria_path/maria_chk$suffix -se test1 + $maria_path/maria_chk$suffix -rs --correct-checksum test1 + $maria_path/maria_chk$suffix -se test1 + $maria_path/maria_chk$suffix -rqs --correct-checksum test1 + $maria_path/maria_chk$suffix -se test1 + $maria_path/maria_chk$suffix -ros --correct-checksum test1 + $maria_path/maria_chk$suffix -se test1 + $maria_path/maria_chk$suffix -rqos --correct-checksum test1 + $maria_path/maria_chk$suffix -se test1 } run_pack_tests() { row_type=$1 # check of maria_pack / maria_chk - ./ma_test1$suffix $silent --checksum $row_type - ./maria_pack$suffix --force -s test1 - ./maria_chk$suffix -ess test1 - ./maria_chk$suffix -rqs test1 - ./maria_chk$suffix -es test1 - ./maria_chk$suffix -rs test1 - ./maria_chk$suffix -es test1 - ./maria_chk$suffix -rus test1 - ./maria_chk$suffix -es test1 + $maria_path/ma_test1$suffix $silent --checksum $row_type + $maria_path/maria_pack$suffix --force -s test1 + $maria_path/maria_chk$suffix -ess test1 + $maria_path/maria_chk$suffix -rqs test1 + $maria_path/maria_chk$suffix -es test1 + $maria_path/maria_chk$suffix -rs test1 + $maria_path/maria_chk$suffix -es test1 + $maria_path/maria_chk$suffix -rus test1 + $maria_path/maria_chk$suffix -es test1 - ./ma_test1$suffix $silent --checksum -S $row_type - ./maria_chk$suffix -se test1 - ./maria_chk$suffix -ros test1 - ./maria_chk$suffix -rqs test1 - ./maria_chk$suffix -se test1 + $maria_path/ma_test1$suffix $silent --checksum -S $row_type + $maria_path/maria_chk$suffix -se test1 + $maria_path/maria_chk$suffix -ros test1 + $maria_path/maria_chk$suffix -rqs test1 + $maria_path/maria_chk$suffix -se test1 - ./maria_pack$suffix --force -s test1 - ./maria_chk$suffix -rqs test1 - ./maria_chk$suffix -es test1 - ./maria_chk$suffix -rus test1 - ./maria_chk$suffix -es test1 + $maria_path/maria_pack$suffix --force -s test1 + $maria_path/maria_chk$suffix -rqs test1 + $maria_path/maria_chk$suffix -es test1 + $maria_path/maria_chk$suffix -rus test1 + $maria_path/maria_chk$suffix -es test1 } echo "Running tests with dynamic row format" @@ -169,27 +175,27 @@ run_tests "-M -T" # Tests that gives warnings # -./ma_test2$suffix $silent -L -K -W -P -S -R1 -m500 -./maria_chk$suffix -sm test2 +$maria_path/ma_test2$suffix $silent -L -K -W -P -S -R1 -m500 +$maria_path/maria_chk$suffix -sm test2 echo "ma_test2$suffix $silent -L -K -R1 -m2000 ; Should give error 135" -./ma_test2$suffix $silent -L -K -R1 -m2000 -echo "./maria_chk$suffix -sm test2 will warn that 'Datafile is almost full'" -./maria_chk$suffix -sm test2 -./maria_chk$suffix -ssm test2 +$maria_path/ma_test2$suffix $silent -L -K -R1 -m2000 +echo "$maria_path/maria_chk$suffix -sm test2 will warn that 'Datafile is almost full'" +$maria_path/maria_chk$suffix -sm test2 +$maria_path/maria_chk$suffix -ssm test2 # # Some timing tests # -time ./ma_test2$suffix $silent -time ./ma_test2$suffix $silent -S -time ./ma_test2$suffix $silent -M -time ./ma_test2$suffix $silent -B -time ./ma_test2$suffix $silent -L -time ./ma_test2$suffix $silent -K -time ./ma_test2$suffix $silent -K -B -time ./ma_test2$suffix $silent -L -B -time ./ma_test2$suffix $silent -L -K -B -time ./ma_test2$suffix $silent -L -K -W -B -time ./ma_test2$suffix $silent -L -K -W -B -S -time ./ma_test2$suffix $silent -L -K -W -B -M -time ./ma_test2$suffix $silent -D -K -W -B -S +time $maria_path/ma_test2$suffix $silent +time $maria_path/ma_test2$suffix $silent -S +time $maria_path/ma_test2$suffix $silent -M +time $maria_path/ma_test2$suffix $silent -B +time $maria_path/ma_test2$suffix $silent -L +time $maria_path/ma_test2$suffix $silent -K +time $maria_path/ma_test2$suffix $silent -K -B +time $maria_path/ma_test2$suffix $silent -L -B +time $maria_path/ma_test2$suffix $silent -L -K -B +time $maria_path/ma_test2$suffix $silent -L -K -W -B +time $maria_path/ma_test2$suffix $silent -L -K -W -B -S +time $maria_path/ma_test2$suffix $silent -L -K -W -B -M +time $maria_path/ma_test2$suffix $silent -D -K -W -B -S diff --git a/storage/maria/maria_def.h b/storage/maria/maria_def.h index d9e31e800c4..740808c7bbe 100644 --- a/storage/maria/maria_def.h +++ b/storage/maria/maria_def.h @@ -93,6 +93,7 @@ typedef struct st_maria_state_info uint sortkey; /* sorted by this key (not used) */ uint open_count; uint8 changed; /* Changed since mariachk */ + LSN create_rename_lsn; /**< LSN when table was last created/renamed */ /* the following isn't saved on disk */ uint state_diff_length; /* Should be 0 */ @@ -101,7 +102,8 @@ typedef struct st_maria_state_info } MARIA_STATE_INFO; -#define MARIA_STATE_INFO_SIZE (24 + 4 + 11*8 + 4*4 + 8 + 3*4 + 5*8) +#define MARIA_STATE_INFO_SIZE \ + (24 + LSN_STORE_SIZE + 4 + 11*8 + 4*4 + 8 + 3*4 + 5*8) #define MARIA_STATE_KEY_SIZE 8 #define MARIA_STATE_KEYBLOCK_SIZE 8 #define MARIA_STATE_KEYSEG_SIZE 4 @@ -229,6 +231,7 @@ typedef struct st_maria_share PAGECACHE *pagecache; /* ref to the current key cache */ MARIA_DECODE_TREE *decode_trees; uint16 *decode_tables; + uint16 id; /**< 2-byte id by which log records refer to the table */ /* Called the first time the table instance is opened */ my_bool (*once_init)(struct st_maria_share *, File); /* Called when the last instance of the table is closed */ @@ -889,6 +892,7 @@ volatile int *_ma_killed_ptr(HA_CHECK *param); void _ma_check_print_error _VARARGS((HA_CHECK *param, const char *fmt, ...)); void _ma_check_print_warning _VARARGS((HA_CHECK *param, const char *fmt, ...)); void _ma_check_print_info _VARARGS((HA_CHECK *param, const char *fmt, ...)); +int _ma_repair_write_log_record(const HA_CHECK *param, MARIA_HA *info); C_MODE_END int _ma_flush_pending_blocks(MARIA_SORT_PARAM *param); diff --git a/storage/maria/trnman.c b/storage/maria/trnman.c index d6b35f071ea..83249ab328f 100644 --- a/storage/maria/trnman.c +++ b/storage/maria/trnman.c @@ -52,6 +52,7 @@ static my_atomic_rwlock_t LOCK_short_trid_to_trn, LOCK_pool; /* Simple interface functions + QQ: if they stay so simple, should we make them inline? */ uint trnman_increment_locked_tables(TRN *trn) @@ -343,6 +344,9 @@ int trnman_end_trn(TRN *trn, my_bool commit) LF_PINS *pins= trn->pins; DBUG_ENTER("trnman_end_trn"); + DBUG_ASSERT(trn->rec_lsn == 0); + /* if a rollback, all UNDO records should have been executed */ + DBUG_ASSERT(commit || trn->undo_lsn == 0); DBUG_PRINT("info", ("pthread_mutex_lock LOCK_trn_list")); pthread_mutex_lock(&LOCK_trn_list); @@ -379,8 +383,6 @@ int trnman_end_trn(TRN *trn, my_bool commit) /* if transaction is committed and it was not the only active transaction - add it to the committed list (which is used for read-from relation) - TODO check in the condition below that a transaction have made some - changes, was not read-only. Something like '&& UndoLSN != 0' */ if (commit && active_list_min.next != &active_list_max) { @@ -390,6 +392,19 @@ int trnman_end_trn(TRN *trn, my_bool commit) trnman_committed_transactions++; res= lf_hash_insert(&trid_to_committed_trn, pins, &trn); + /* + By going on with life is res<0, we let other threads block on + our rows (because they will never see us committed in + trid_to_committed_trn) until they timeout. Though correct, this is not a + good situation: + - if connection reconnects and wants to check if its rows have been + committed, it will not be able to do that (it will just lock on them) so + connection stays permanently in doubt + - internal structures trid_to_committed_trn and committed_list are + desynchronized. + So we should take Maria down immediately, the two problems being + automatically solved at restart. + */ DBUG_ASSERT(res <= 0); } if (res) @@ -526,71 +541,133 @@ void trnman_rollback_statement(TRN *trn __attribute__ ((unused))) } -/* - Allocates two buffers and stores in them some information about transactions - of the active list (into the first buffer) and of the committed list (into - the second buffer). - - SYNOPSIS - trnman_collect_transactions() - str_act (OUT) pointer to a LEX_STRING where the allocated buffer, and - its size, will be put - str_com (OUT) pointer to a LEX_STRING where the allocated buffer, and - its size, will be put +/** + @brief Allocates buffers and stores in them some info about transactions + Does the allocation because the caller cannot know the size itself. + Memory freeing is to be done by the caller (if the "str" member of the + LEX_STRING is not NULL). + The caller has the intention of doing checkpoints. - DESCRIPTION - Does the allocation because the caller cannot know the size itself. - Memory freeing is to be done by the caller (if the "str" member of the - LEX_STRING is not NULL). - The caller has the intention of doing checkpoints. + @param[out] str_act pointer to where the allocated buffer, + and its size, will be put; buffer will be filled + with info about active transactions + @param[out] str_com pointer to where the allocated buffer, + and its size, will be put; buffer will be filled + with info about committed transactions + @param[out] min_first_undo_lsn pointer to where the minimum + first_undo_lsn of all transactions will be put - RETURN - 0 on success - 1 on error + @return Operation status + @retval 0 OK + @retval 1 Error */ -my_bool trnman_collect_transactions(LEX_STRING *str_act, LEX_STRING *str_com) + +my_bool trnman_collect_transactions(LEX_STRING *str_act, LEX_STRING *str_com, + LSN *min_rec_lsn, LSN *min_first_undo_lsn) { my_bool error; TRN *trn; char *ptr; + uint stored_transactions= 0; + LSN minimum_rec_lsn= ULONGLONG_MAX, minimum_first_undo_lsn= ULONGLONG_MAX; DBUG_ENTER("trnman_collect_transactions"); DBUG_ASSERT((NULL == str_act->str) && (NULL == str_com->str)); + /* validate the use of read_non_atomic() in general: */ + compile_time_assert((sizeof(LSN) == 8) && (sizeof(LSN_WITH_FLAGS) == 8)); + DBUG_PRINT("info", ("pthread_mutex_lock LOCK_trn_list")); pthread_mutex_lock(&LOCK_trn_list); - str_act->length= 8+(6+2+7+7+7)*trnman_active_transactions; - str_com->length= 8+(6+7+7)*trnman_committed_transactions; + str_act->length= 2 + /* number of active transactions */ + LSN_STORE_SIZE + /* minimum of their rec_lsn */ + (6 + /* long id */ + 2 + /* short id */ + LSN_STORE_SIZE + /* undo_lsn */ +#ifdef MARIA_VERSIONING /* not enabled yet */ + LSN_STORE_SIZE + /* undo_purge_lsn */ +#endif + LSN_STORE_SIZE /* first_undo_lsn */ + ) * trnman_active_transactions; + str_com->length= 8 + /* number of committed transactions */ + (6 + /* long id */ +#ifdef MARIA_VERSIONING /* not enabled yet */ + LSN_STORE_SIZE + /* undo_purge_lsn */ +#endif + LSN_STORE_SIZE /* first_undo_lsn */ + ) * trnman_committed_transactions; if ((NULL == (str_act->str= my_malloc(str_act->length, MYF(MY_WME)))) || (NULL == (str_com->str= my_malloc(str_com->length, MYF(MY_WME))))) goto err; /* First, the active transactions */ - ptr= str_act->str; - int8store(ptr, (ulonglong)trnman_active_transactions); - ptr+= 8; + ptr= str_act->str + 2 + LSN_STORE_SIZE; for (trn= active_list_min.next; trn != &active_list_max; trn= trn->next) { /* - trns with a short trid of 0 are not initialized; Recovery will recognize - this and ignore them. - State is not needed for now (only when we supported prepared trns). - For LSNs, Sanja will soon push lsn7store. + trns with a short trid of 0 are not even initialized, we can ignore + them. trns with undo_lsn==0 have done no writes, we can ignore them + too. XID not needed now. */ + uint sid; + LSN rec_lsn, undo_lsn, first_undo_lsn; + if ((sid= trn->short_id) == 0) + { + /* + Not even inited, has done nothing. Or it is the + dummy_transaction_object, which does only non-transactional + immediate-sync operations (CREATE/DROP/RENAME/REPAIR TABLE), and so + can be forgotten for Checkpoint. + */ + continue; + } +#ifndef MARIA_CHECKPOINT +/* + in the checkpoint patch (not yet ready) we will have a real implementation + of lsn_read_non_atomic(); for now it's not needed +*/ +#define lsn_read_non_atomic(A) (A) +#endif + /* needed for low-water mark calculation */ + if (((rec_lsn= lsn_read_non_atomic(trn->rec_lsn)) > 0) && + (cmp_translog_addr(rec_lsn, minimum_rec_lsn) < 0)) + minimum_rec_lsn= rec_lsn; + /* + trn may have logged REDOs but not yet UNDO, that's why we read rec_lsn + before deciding to ignore if undo_lsn==0. + */ + if ((undo_lsn= trn->undo_lsn) == 0) /* trn can be forgotten */ + continue; + stored_transactions++; int6store(ptr, trn->trid); ptr+= 6; - int2store(ptr, trn->short_id); + int2store(ptr, sid); ptr+= 2; - /* needed for rollback */ - /* lsn7store(ptr, trn->undo_lsn); */ - ptr+= 7; - /* needed for purge */ - /* lsn7store(ptr, trn->undo_purge_lsn); */ - ptr+= 7; + lsn_store(ptr, undo_lsn); /* needed for rollback */ + ptr+= LSN_STORE_SIZE; +#ifdef MARIA_VERSIONING /* not enabled yet */ + /* to know where purging should start (last delete of this trn) */ + lsn_store(ptr, trn->undo_purge_lsn); + ptr+= LSN_STORE_SIZE; +#endif /* needed for low-water mark calculation */ - /* lsn7store(ptr, read_non_atomic(&trn->first_undo_lsn)); */ - ptr+= 7; + if (((first_undo_lsn= lsn_read_non_atomic(trn->first_undo_lsn)) > 0) && + (cmp_translog_addr(first_undo_lsn, minimum_first_undo_lsn) < 0)) + minimum_first_undo_lsn= first_undo_lsn; + lsn_store(ptr, first_undo_lsn); + ptr+= LSN_STORE_SIZE; + /** + @todo RECOVERY: add a comment explaining why we can dirtily read some + vars, inspired by the text of "assumption 8" in WL#3072 + */ } + str_act->length= ptr - str_act->str; /* as we maybe over-estimated */ + ptr= str_act->str; + int2store(ptr, stored_transactions); + ptr+= 2; + /* this LSN influences how REDOs for any page can be ignored by Recovery */ + lsn_store(ptr, minimum_rec_lsn); + /* one day there will also be a list of prepared transactions */ /* do the same for committed ones */ ptr= str_com->str; int8store(ptr, (ulonglong)trnman_committed_transactions); @@ -598,18 +675,26 @@ my_bool trnman_collect_transactions(LEX_STRING *str_act, LEX_STRING *str_com) for (trn= committed_list_min.next; trn != &committed_list_max; trn= trn->next) { + LSN first_undo_lsn; int6store(ptr, trn->trid); ptr+= 6; - /* mi_int7store(ptr, trn->undo_purge_lsn); */ - ptr+= 7; - /* mi_int7store(ptr, read_non_atomic(&trn->first_undo_lsn)); */ - ptr+= 7; +#ifdef MARIA_VERSIONING /* not enabled yet */ + lsn_store(ptr, trn->undo_purge_lsn); + ptr+= LSN_STORE_SIZE; +#endif + first_undo_lsn= LSN_WITH_FLAGS_TO_LSN(trn->first_undo_lsn); + if (cmp_translog_addr(first_undo_lsn, minimum_first_undo_lsn) < 0) + minimum_first_undo_lsn= first_undo_lsn; + lsn_store(ptr, first_undo_lsn); + ptr+= LSN_STORE_SIZE; } /* TODO: if we see there exists no transaction (active and committed) we can tell the lock-free structures to do some freeing (my_free()). */ error= 0; + *min_rec_lsn= minimum_rec_lsn; + *min_first_undo_lsn= minimum_first_undo_lsn; goto end; err: error= 1; diff --git a/storage/maria/trnman.h b/storage/maria/trnman.h index 1e1550efb46..1a4423f2a11 100644 --- a/storage/maria/trnman.h +++ b/storage/maria/trnman.h @@ -45,12 +45,13 @@ struct st_transaction LF_PINS *pins; TrID trid, min_read_from, commit_trid; TRN *next, *prev; - LSN rec_lsn, undo_lsn, first_undo_lsn; + LSN rec_lsn, undo_lsn; + LSN_WITH_FLAGS first_undo_lsn; uint locked_tables; /* Note! if locks.loid is 0, trn is NOT initialized */ }; -TRN dummy_transaction_object; +#define TRANSACTION_LOGGED_LONG_ID ULL(0x8000000000000000) C_MODE_END diff --git a/storage/maria/trnman_public.h b/storage/maria/trnman_public.h index 4b3f8acb4b3..3e0a21c26a6 100644 --- a/storage/maria/trnman_public.h +++ b/storage/maria/trnman_public.h @@ -20,6 +20,8 @@ to include my_atomic.h in C++ code. */ +#include "ma_loghandler_lsn.h" + C_MODE_START typedef uint64 TrID; /* our TrID is 6 bytes */ typedef struct st_transaction TRN; @@ -27,6 +29,7 @@ typedef struct st_transaction TRN; #define SHORT_TRID_MAX 65535 extern uint trnman_active_transactions, trnman_allocated_transactions; +extern TRN dummy_transaction_object; int trnman_init(void); void trnman_destroy(void); @@ -39,7 +42,9 @@ void trnman_free_trn(TRN *trn); int trnman_can_read_from(TRN *trn, TrID trid); void trnman_new_statement(TRN *trn); void trnman_rollback_statement(TRN *trn); -my_bool trnman_collect_transactions(LEX_STRING *str_act, LEX_STRING *str_com); +my_bool trnman_collect_transactions(LEX_STRING *str_act, LEX_STRING *str_com, + LSN *min_rec_lsn, + LSN *min_first_undo_lsn); uint trnman_increment_locked_tables(TRN *trn); uint trnman_decrement_locked_tables(TRN *trn); diff --git a/storage/maria/unittest/ma_test_loghandler-t.c b/storage/maria/unittest/ma_test_loghandler-t.c index f05d58a784f..e31136d52ec 100644 --- a/storage/maria/unittest/ma_test_loghandler-t.c +++ b/storage/maria/unittest/ma_test_loghandler-t.c @@ -196,7 +196,7 @@ int main(int argc __attribute__((unused)), char *argv[]) if (translog_write_record(&lsn, LOGREC_FIXED_RECORD_0LSN_EXAMPLE, trn, NULL, - 6, TRANSLOG_INTERNAL_PARTS + 1, parts)) + 6, TRANSLOG_INTERNAL_PARTS + 1, parts, NULL)) { fprintf(stderr, "Can't write record #%lu\n", (ulong) 0); translog_destroy(); @@ -218,7 +218,7 @@ int main(int argc __attribute__((unused)), char *argv[]) parts[TRANSLOG_INTERNAL_PARTS + 1].str= NULL; parts[TRANSLOG_INTERNAL_PARTS + 1].length= 0; if (translog_write_record(&lsn, LOGREC_FIXED_RECORD_1LSN_EXAMPLE, - trn, NULL, LSN_STORE_SIZE, 0, parts)) + trn, NULL, LSN_STORE_SIZE, 0, parts, NULL)) { fprintf(stderr, "1 Can't write reference defore record #%lu\n", (ulong) i); @@ -238,7 +238,7 @@ int main(int argc __attribute__((unused)), char *argv[]) if (translog_write_record(&lsn, LOGREC_VARIABLE_RECORD_1LSN_EXAMPLE, trn, NULL, 0, TRANSLOG_INTERNAL_PARTS + 2, - parts)) + parts, NULL)) { fprintf(stderr, "1 Can't write var reference defore record #%lu\n", (ulong) i); @@ -257,7 +257,7 @@ int main(int argc __attribute__((unused)), char *argv[]) if (translog_write_record(&lsn, LOGREC_FIXED_RECORD_2LSN_EXAMPLE, trn, NULL, - 23, TRANSLOG_INTERNAL_PARTS + 1, parts)) + 23, TRANSLOG_INTERNAL_PARTS + 1, parts, NULL)) { fprintf(stderr, "0 Can't write reference defore record #%lu\n", (ulong) i); @@ -277,7 +277,7 @@ int main(int argc __attribute__((unused)), char *argv[]) if (translog_write_record(&lsn, LOGREC_VARIABLE_RECORD_2LSN_EXAMPLE, trn, NULL, 14 + rec_len, - TRANSLOG_INTERNAL_PARTS + 2, parts)) + TRANSLOG_INTERNAL_PARTS + 2, parts, NULL)) { fprintf(stderr, "0 Can't write var reference defore record #%lu\n", (ulong) i); @@ -294,7 +294,7 @@ int main(int argc __attribute__((unused)), char *argv[]) LOGREC_FIXED_RECORD_0LSN_EXAMPLE, trn, NULL, 6, TRANSLOG_INTERNAL_PARTS + 1, - parts)) + parts, NULL)) { fprintf(stderr, "Can't write record #%lu\n", (ulong) i); translog_destroy(); @@ -313,7 +313,7 @@ int main(int argc __attribute__((unused)), char *argv[]) LOGREC_VARIABLE_RECORD_0LSN_EXAMPLE, trn, NULL, rec_len, TRANSLOG_INTERNAL_PARTS + 1, - parts)) + parts, NULL)) { fprintf(stderr, "Can't write variable record #%lu\n", (ulong) i); translog_destroy(); diff --git a/storage/maria/unittest/ma_test_loghandler_multigroup-t.c b/storage/maria/unittest/ma_test_loghandler_multigroup-t.c index 9ed57da8fec..1281ee425d8 100644 --- a/storage/maria/unittest/ma_test_loghandler_multigroup-t.c +++ b/storage/maria/unittest/ma_test_loghandler_multigroup-t.c @@ -192,7 +192,7 @@ int main(int argc __attribute__((unused)), char *argv[]) trn->short_id= 0; if (translog_write_record(&lsn, LOGREC_FIXED_RECORD_0LSN_EXAMPLE, trn, NULL, - 6, TRANSLOG_INTERNAL_PARTS + 1, parts)) + 6, TRANSLOG_INTERNAL_PARTS + 1, parts, NULL)) { fprintf(stderr, "Can't write record #%lu\n", (ulong) 0); translog_destroy(); @@ -214,7 +214,7 @@ int main(int argc __attribute__((unused)), char *argv[]) LOGREC_FIXED_RECORD_1LSN_EXAMPLE, trn, NULL, LSN_STORE_SIZE, - TRANSLOG_INTERNAL_PARTS + 1, parts)) + TRANSLOG_INTERNAL_PARTS + 1, parts, NULL)) { fprintf(stderr, "1 Can't write reference before record #%lu\n", (ulong) i); @@ -234,7 +234,7 @@ int main(int argc __attribute__((unused)), char *argv[]) LOGREC_VARIABLE_RECORD_1LSN_EXAMPLE, trn, NULL, LSN_STORE_SIZE + rec_len, TRANSLOG_INTERNAL_PARTS + 2, - parts)) + parts, NULL)) { fprintf(stderr, "1 Can't write var reference before record #%lu\n", (ulong) i); @@ -255,7 +255,7 @@ int main(int argc __attribute__((unused)), char *argv[]) LOGREC_FIXED_RECORD_2LSN_EXAMPLE, trn, NULL, 23, TRANSLOG_INTERNAL_PARTS + 1, - parts)) + parts, NULL)) { fprintf(stderr, "0 Can't write reference before record #%lu\n", (ulong) i); @@ -276,7 +276,7 @@ int main(int argc __attribute__((unused)), char *argv[]) LOGREC_VARIABLE_RECORD_2LSN_EXAMPLE, trn, NULL, LSN_STORE_SIZE * 2 + rec_len, TRANSLOG_INTERNAL_PARTS + 2, - parts)) + parts, NULL)) { fprintf(stderr, "0 Can't write var reference before record #%lu\n", (ulong) i); @@ -293,7 +293,7 @@ int main(int argc __attribute__((unused)), char *argv[]) if (translog_write_record(&lsn, LOGREC_FIXED_RECORD_0LSN_EXAMPLE, trn, NULL, 6, - TRANSLOG_INTERNAL_PARTS + 1, parts)) + TRANSLOG_INTERNAL_PARTS + 1, parts, NULL)) { fprintf(stderr, "Can't write record #%lu\n", (ulong) i); translog_destroy(); @@ -311,7 +311,7 @@ int main(int argc __attribute__((unused)), char *argv[]) if (translog_write_record(&lsn, LOGREC_VARIABLE_RECORD_0LSN_EXAMPLE, trn, NULL, rec_len, - TRANSLOG_INTERNAL_PARTS + 1, parts)) + TRANSLOG_INTERNAL_PARTS + 1, parts, NULL)) { fprintf(stderr, "Can't write variable record #%lu\n", (ulong) i); translog_destroy(); diff --git a/storage/maria/unittest/ma_test_loghandler_multithread-t.c b/storage/maria/unittest/ma_test_loghandler_multithread-t.c index 688c1ec33be..ff966160acc 100644 --- a/storage/maria/unittest/ma_test_loghandler_multithread-t.c +++ b/storage/maria/unittest/ma_test_loghandler_multithread-t.c @@ -137,7 +137,7 @@ void writer(int num) if (translog_write_record(&lsn, LOGREC_FIXED_RECORD_0LSN_EXAMPLE, &trn, NULL, 6, TRANSLOG_INTERNAL_PARTS + 1, - parts)) + parts, NULL)) { fprintf(stderr, "Can't write LOGREC_FIXED_RECORD_0LSN_EXAMPLE record #%lu " "thread %i\n", (ulong) i, num); @@ -154,7 +154,7 @@ void writer(int num) LOGREC_VARIABLE_RECORD_0LSN_EXAMPLE, &trn, NULL, len, TRANSLOG_INTERNAL_PARTS + 1, - parts)) + parts, NULL)) { fprintf(stderr, "Can't write variable record #%lu\n", (ulong) i); translog_destroy(); @@ -303,7 +303,7 @@ int main(int argc __attribute__((unused)), LOGREC_FIXED_RECORD_0LSN_EXAMPLE, &dummy_transaction_object, NULL, 6, TRANSLOG_INTERNAL_PARTS + 1, - parts)) + parts, NULL)) { fprintf(stderr, "Can't write the first record\n"); translog_destroy(); diff --git a/storage/maria/unittest/ma_test_loghandler_pagecache-t.c b/storage/maria/unittest/ma_test_loghandler_pagecache-t.c index b43f0cfa98c..35e05f9c997 100644 --- a/storage/maria/unittest/ma_test_loghandler_pagecache-t.c +++ b/storage/maria/unittest/ma_test_loghandler_pagecache-t.c @@ -94,7 +94,7 @@ int main(int argc __attribute__((unused)), char *argv[]) LOGREC_FIXED_RECORD_0LSN_EXAMPLE, &dummy_transaction_object, NULL, 6, TRANSLOG_INTERNAL_PARTS + 1, - parts)) + parts, NULL)) { fprintf(stderr, "Can't write record #%lu\n", (ulong) 0); translog_destroy(); -- cgit v1.2.1 From bb8bde8f931ec934b1926489da5b2fb087105f6e Mon Sep 17 00:00:00 2001 From: unknown Date: Mon, 25 Jun 2007 10:07:46 +0300 Subject: Creation of new log when maria change version added. storage/maria/ma_loghandler.c: Structure and function to read loghandler file data added. Creation of new log when maria change version added. --- storage/maria/ma_loghandler.c | 60 ++++++++++++++++++++++++++++++++++++++++++- 1 file changed, 59 insertions(+), 1 deletion(-) diff --git a/storage/maria/ma_loghandler.c b/storage/maria/ma_loghandler.c index 9ed1d4b9d93..44be624bed0 100644 --- a/storage/maria/ma_loghandler.c +++ b/storage/maria/ma_loghandler.c @@ -635,6 +635,56 @@ static my_bool translog_write_file_header() } +/* + Information from transaction log file header +*/ + +typedef struct st_loghandler_file_info +{ + ulonglong timestamp; /* Time stamp */ + ulong maria_version; /* Version of maria loghandler */ + ulong mysql_versiob; /* Version of mysql server */ + ulong server_id; /* Server ID */ + uint page_size; /* Loghandler page size */ + uint file_number; /* Number of the file (from the file header) */ +} LOGHANDLER_FILE_INFO; + +/* + @brief Read hander file information from last opened loghandler file + + @param desc header information descriptor to be filled with information + + @retval 0 OK + @retval 1 Error +*/ + +my_bool translog_read_file_header(LOGHANDLER_FILE_INFO *desc) +{ + byte page_buff[TRANSLOG_PAGE_SIZE], *ptr; + DBUG_ENTER("translog_read_file_header"); + + if (my_pread(log_descriptor.log_file_num[0], page_buff, + sizeof(page_buff), 0, MYF(MY_FNABP | MY_WME))) + { + DBUG_PRINT("info", ("log read fail error: %d", my_errno)); + DBUG_RETURN(1); + } + ptr= page_buff + sizeof(maria_trans_file_magic); + desc->timestamp= uint8korr(ptr); + ptr+= 8; + desc->maria_version= uint4korr(ptr); + ptr+= 4; + desc->mysql_versiob= uint4korr(ptr); + ptr+= 4; + desc->server_id= uint4korr(ptr); + ptr+= 2; + desc->page_size= uint2korr(ptr); + ptr+= 2; + desc->file_number= uint3korr(ptr); + DBUG_RETURN(0); +} + + /* Initialize transaction log file buffer @@ -1958,6 +2008,7 @@ my_bool translog_init(const char *directory, int old_log_was_recovered= 0, logs_found= 0; uint old_flags= flags; TRANSLOG_ADDRESS sure_page, last_page, last_valid_page; + my_bool version_changed= 0; DBUG_ENTER("translog_init"); loghandler_init(); /* Safe to do many times */ @@ -2201,6 +2252,13 @@ my_bool translog_init(const char *directory, buffer->buffer))); DBUG_EXECUTE("info", translog_check_cursor(&log_descriptor.bc);); } + if (!old_log_was_recovered && old_flags == flags) + { + LOGHANDLER_FILE_INFO info; + if (translog_read_file_header(&info)) + DBUG_RETURN(1); + version_changed= (info.maria_version != TRANSLOG_VERSION_ID); + } } DBUG_PRINT("info", ("Logs found: %d was recovered: %d", logs_found, old_log_was_recovered)); @@ -2221,7 +2279,7 @@ my_bool translog_init(const char *directory, translog_start_buffer(log_descriptor.buffers, &log_descriptor.bc, 0); translog_new_page_header(&log_descriptor.horizon, &log_descriptor.bc); } - else if (old_log_was_recovered || old_flags != flags) + else if (old_log_was_recovered || old_flags != flags || version_changed) { /* leave the damaged file untouched */ log_descriptor.horizon+= LSN_ONE_FILE; -- cgit v1.2.1 From 5ef7e0377f003427a8c678c87bc345f4d38e3b12 Mon Sep 17 00:00:00 2001 From: unknown Date: Tue, 26 Jun 2007 16:14:59 +0200 Subject: storage/maria/trnman.c fix for architectures not supported by my_atomic.h we cannot iterate the array over and over without releasing a lock storage/maria/trnman.c: fix for architectures not supported by my_atomic.h we cannot iterate the array over and over without releasing a lock --- storage/maria/trnman.c | 21 +++++++++++++-------- 1 file changed, 13 insertions(+), 8 deletions(-) diff --git a/storage/maria/trnman.c b/storage/maria/trnman.c index 83249ab328f..9c300a2ba3f 100644 --- a/storage/maria/trnman.c +++ b/storage/maria/trnman.c @@ -209,16 +209,21 @@ static TrID new_trid() static void set_short_trid(TRN *trn) { int i= (global_trid_generator + (intptr)trn) * 312089 % SHORT_TRID_MAX + 1; - my_atomic_rwlock_wrlock(&LOCK_short_trid_to_trn); - for ( ; ; i= i % SHORT_TRID_MAX + 1) /* the range is [1..SHORT_TRID_MAX] */ + for ( ; !trn->short_id ; i= 1) { - void *tmp= NULL; - if (short_trid_to_active_trn[i] == NULL && - my_atomic_casptr((void **)&short_trid_to_active_trn[i], &tmp, trn)) - break; + my_atomic_rwlock_wrlock(&LOCK_short_trid_to_trn); + for ( ; i <= SHORT_TRID_MAX; i++) /* the range is [1..SHORT_TRID_MAX] */ + { + void *tmp= NULL; + if (short_trid_to_active_trn[i] == NULL && + my_atomic_casptr((void **)&short_trid_to_active_trn[i], &tmp, trn)) + { + trn->short_id= i; + break; + } + } + my_atomic_rwlock_wrunlock(&LOCK_short_trid_to_trn); } - my_atomic_rwlock_wrunlock(&LOCK_short_trid_to_trn); - trn->short_id= i; } /* -- cgit v1.2.1 From adac9798bff81a682b346f438eb7b58264b1b541 Mon Sep 17 00:00:00 2001 From: unknown Date: Tue, 26 Jun 2007 16:49:23 +0200 Subject: WL#3072 Maria Recovery - new program maria_read_log to display and apply log records found in a Maria log (see file's revision comment) - minor, misc fixes storage/maria/Makefile.am: new program maria_read_log storage/maria/ha_maria.cc: create control file if missing storage/maria/ma_blockrec.c: 0 -> LSN_IMPOSSIBLE; comments storage/maria/ma_checkpoint.h: preparations for Checkpoint module storage/maria/ma_close.c: comment storage/maria/ma_control_file.c: renaming constants. Possibility to say "open control file but don't create it if it's missing" (used by maria_read_log which does not want to create anything) storage/maria/ma_control_file.h: renaming constants storage/maria/ma_create.c: I had duplicated "linkname" and "linkname_ptr", now I see it's not needed, reverting. Indeed those variables don't contain interesting information; fixing log record accordingly (the links are in ci->data/index_file_name). Storing keystart in log record is needed, to know at which size we must extend the file if we replay LOGREC_CREATE_TABLE. storage/maria/ma_loghandler.c: some structures need to be known to maria_read_log.c, taking them to ma_loghandler.h storage/maria/ma_loghandler.h: we have page_store, adding page_korr. translog_lock() made public, because Checkpoint will need it (to write to control file). Some structures moved from ma_loghandler.c because maria_read_log.c needs them (needs to know the execute-in-REDO-phase hooks of each record). storage/maria/ma_loghandler_lsn.h: constants defined in ma_control_file.h serve everywhere, and they relate to LSNs, so putting them in ma_loghandler_lsn.h. Stronger constraints in LSN_VALID(). storage/maria/ma_pagecache.c: renaming constants storage/maria/ma_recovery.h: copyright storage/maria/ma_test1.c: new prototype storage/maria/ma_test2.c: new prototype storage/maria/trnman_public.h: double-inclusion safe storage/maria/unittest/ma_control_file-t.c: constants renamed, new prototype storage/maria/unittest/ma_test_loghandler-t.c: constants renamed, new prototype storage/maria/unittest/ma_test_loghandler_multigroup-t.c: constants renamed, new prototype storage/maria/unittest/ma_test_loghandler_multithread-t.c: constants renamed, new prototype storage/maria/unittest/ma_test_loghandler_pagecache-t.c: constants renamed, new prototype storage/myisam/mi_close.c: comment storage/maria/maria_read_log.c: program to read and print log records from a Maria transaction log, and optionally apply them to tables. Very basic, early version. Should serve as a base for Recovery's code. Designed to be idempotent. Create a log by running maria.test, then cd to var/master-data and run "maria_read_log --only-display" to see info about records; run "maria_read_log --display-and-apply" to also apply the records to tables (it's more interesting if you first wipe out the tables in var/master-data/test, to see how they get re-created). Only a few records are handled by now: LONG_TRANSACTION_ID, COMMIT, FILE_ID, REDO_CREATE_TABLE; place is ready for REDO_INSERT_ROW_HEAD where I could use Monty's help (search for "Monty" in the file). Note: changes to the index pages, index's header and bitmap pages are not properly logged yet, so don't expect the program to work with that. --- storage/maria/Makefile.am | 8 +- storage/maria/ha_maria.cc | 2 +- storage/maria/ma_blockrec.c | 30 +- storage/maria/ma_checkpoint.h | 67 +- storage/maria/ma_close.c | 1 + storage/maria/ma_control_file.c | 40 +- storage/maria/ma_control_file.h | 23 +- storage/maria/ma_create.c | 48 +- storage/maria/ma_loghandler.c | 231 ++++--- storage/maria/ma_loghandler.h | 99 +++ storage/maria/ma_loghandler_lsn.h | 11 +- storage/maria/ma_pagecache.c | 25 +- storage/maria/ma_recovery.h | 2 +- storage/maria/ma_test1.c | 2 +- storage/maria/ma_test2.c | 2 +- storage/maria/maria_read_log.c | 696 +++++++++++++++++++++ storage/maria/trnman_public.h | 4 + storage/maria/unittest/ma_control_file-t.c | 23 +- storage/maria/unittest/ma_test_loghandler-t.c | 10 +- .../unittest/ma_test_loghandler_multigroup-t.c | 10 +- .../unittest/ma_test_loghandler_multithread-t.c | 4 +- .../unittest/ma_test_loghandler_pagecache-t.c | 2 +- storage/myisam/mi_close.c | 1 + 23 files changed, 1093 insertions(+), 248 deletions(-) create mode 100644 storage/maria/maria_read_log.c diff --git a/storage/maria/Makefile.am b/storage/maria/Makefile.am index fbb25584910..2d11d2f470b 100644 --- a/storage/maria/Makefile.am +++ b/storage/maria/Makefile.am @@ -33,7 +33,7 @@ SUBDIRS = . unittest EXTRA_DIST = ma_test_all.sh ma_test_all.res ma_ft_stem.c CMakeLists.txt plug.in pkgdata_DATA = ma_test_all ma_test_all.res pkglib_LIBRARIES = libmaria.a -bin_PROGRAMS = maria_chk maria_pack maria_ftdump +bin_PROGRAMS = maria_chk maria_pack maria_ftdump maria_read_log maria_chk_DEPENDENCIES= $(LIBRARIES) # Only reason to link with libmyisam.a here is that it's where some fulltext # pieces are (but soon we'll remove fulltext dependencies from Maria). @@ -49,6 +49,12 @@ maria_pack_LDADD= @CLIENT_EXTRA_LDFLAGS@ libmaria.a \ $(top_builddir)/mysys/libmysys.a \ $(top_builddir)/dbug/libdbug.a \ $(top_builddir)/strings/libmystrings.a @ZLIB_LIBS@ +maria_read_log_DEPENDENCIES=$(LIBRARIES) +maria_read_log_LDADD= @CLIENT_EXTRA_LDFLAGS@ libmaria.a \ + $(top_builddir)/storage/myisam/libmyisam.a \ + $(top_builddir)/mysys/libmysys.a \ + $(top_builddir)/dbug/libdbug.a \ + $(top_builddir)/strings/libmystrings.a @ZLIB_LIBS@ noinst_PROGRAMS = ma_test1 ma_test2 ma_test3 ma_rt_test ma_sp_test noinst_HEADERS = maria_def.h ma_rt_index.h ma_rt_key.h ma_rt_mbr.h \ ma_sp_defs.h ma_fulltext.h ma_ftdefs.h ma_ft_test1.h \ diff --git a/storage/maria/ha_maria.cc b/storage/maria/ha_maria.cc index e05f97a384d..24cc6dfb915 100644 --- a/storage/maria/ha_maria.cc +++ b/storage/maria/ha_maria.cc @@ -2241,7 +2241,7 @@ static int ha_maria_init(void *p) maria_hton->flags= HTON_CAN_RECREATE | HTON_SUPPORT_LOG_TABLES; bzero(maria_log_pagecache, sizeof(*maria_log_pagecache)); maria_data_root= mysql_real_data_home; - res= maria_init() || ma_control_file_create_or_open() || + res= maria_init() || ma_control_file_create_or_open(TRUE) || (init_pagecache(maria_log_pagecache, TRANSLOG_PAGECACHE_SIZE, 0, 0, TRANSLOG_PAGE_SIZE) == 0) || diff --git a/storage/maria/ma_blockrec.c b/storage/maria/ma_blockrec.c index d2512f1e025..17ca22390f4 100644 --- a/storage/maria/ma_blockrec.c +++ b/storage/maria/ma_blockrec.c @@ -557,7 +557,8 @@ static my_bool check_if_zero(byte *pos, uint length) SYNOPSIS _ma_unpin_all_pages() info Maria handler - undo_lsn LSN for undo pages. 0 if we shouldn't write undo (error) + undo_lsn LSN for undo pages. LSN_IMPOSSIBLE if we shouldn't write undo + (error) NOTE We unpin pages in the reverse order as they where pinned; This may not @@ -580,14 +581,15 @@ void _ma_unpin_all_pages(MARIA_HA *info, LSN undo_lsn) DBUG_PRINT("info", ("undo_lsn: %lu", (ulong) undo_lsn)); /* True if not disk error */ - DBUG_ASSERT(undo_lsn != 0 || !info->s->base.transactional); + DBUG_ASSERT((undo_lsn != LSN_IMPOSSIBLE) || !info->s->base.transactional); if (!info->s->base.transactional) { /* If this is a transactional table but with transactionality temporarily disabled (like in ALTER TABLE) we need to give a sensible LSN to pages - and not 0. If this is not a transactional table it will reduce to 0. + and not LSN_IMPOSSIBLE. If this is not a transactional table it will + reduce to LSN_IMPOSSIBLE. */ undo_lsn= info->s->state.create_rename_lsn; } @@ -1958,8 +1960,8 @@ static my_bool write_block_record(MARIA_HA *info, size_t data_length= (size_t) (data - row_pos->data); /* Log REDO changes of head page */ - page_store(log_data+ FILEID_STORE_SIZE, head_block->page); - dirpos_store(log_data+ FILEID_STORE_SIZE + PAGE_STORE_SIZE, + page_store(log_data + FILEID_STORE_SIZE, head_block->page); + dirpos_store(log_data + FILEID_STORE_SIZE + PAGE_STORE_SIZE, row_pos->rownr); log_array[TRANSLOG_INTERNAL_PARTS + 0].str= (char*) log_data; log_array[TRANSLOG_INTERNAL_PARTS + 0].length= sizeof(log_data); @@ -2183,12 +2185,22 @@ crashed: disk_err: /** @todo RECOVERY we are going to let dirty pages go to disk while we have - logged UNDO, this violates WAL. If we have not written any full pages, - all dirty pages are pinned so we could just delete them from the - pagecache. Moreover, we have written some REDOs without a closing UNDO, + logged UNDO, this violates WAL. We must mark the table corrupted! + + @todo RECOVERY we have written some REDOs without a closing UNDO, it's possible that a next operation by this transaction succeeds and then Recovery would glue the "orphan REDOs" to the succeeded operation and - execute the failed REDOs. + execute the failed REDOs. We need some mark "abort this group" in the + log, or mark the table corrupted (then user will repair it and thus REDOs + will be skipped). + + @todo RECOVERY to not let write errors go unnoticed, pagecache_write() + should take a MARIA_HA* in argument, and it it + fails when flushing a page to disk it should call + (*the_maria_ha->write_error_func)(the_maria_ha) + and this hook will mark the table corrupted. + Maybe hook should be stored in the pagecache's block structure, or in a + hash "file->maria_ha*". */ /* Unpin all pinned pages to not cause problems for disk cache */ _ma_unpin_all_pages(info, 0); diff --git a/storage/maria/ma_checkpoint.h b/storage/maria/ma_checkpoint.h index 1ce2ccb7012..c011c8234b7 100644 --- a/storage/maria/ma_checkpoint.h +++ b/storage/maria/ma_checkpoint.h @@ -1,4 +1,4 @@ -/* Copyright (C) 2006 MySQL AB & MySQL Finland AB & TCX DataKonsult AB +/* Copyright (C) 2006,2007 MySQL AB 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 @@ -21,14 +21,61 @@ /* This is the interface of this module. */ -typedef enum enum_checkpoint_level { - NONE=-1, - INDIRECT, /* just write dirty_pages, transactions table and sync files */ - MEDIUM, /* also flush all dirty pages which were already dirty at prev checkpoint*/ - FULL /* also flush all dirty pages */ +typedef enum enum_ma_checkpoint_level { + CHECKPOINT_NONE= 0, + /* just write dirty_pages, transactions table and sync files */ + CHECKPOINT_INDIRECT, + /* also flush all dirty pages which were already dirty at prev checkpoint */ + CHECKPOINT_MEDIUM, + /* also flush all dirty pages */ + CHECKPOINT_FULL } CHECKPOINT_LEVEL; -void request_asynchronous_checkpoint(CHECKPOINT_LEVEL level); -my_bool execute_synchronous_checkpoint(CHECKPOINT_LEVEL level); -my_bool execute_asynchronous_checkpoint_if_any(); -/* that's all that's needed in the interface */ +C_MODE_START +int ma_checkpoint_init(); +void ma_checkpoint_end(); +int ma_checkpoint_execute(CHECKPOINT_LEVEL level, my_bool no_wait); +C_MODE_END + +/** + @brief reads some LSNs with special trickery + + If a 64-bit variable transitions between both halves being zero to both + halves being non-zero, and back, this function can be used to do a read of + it (without mutex, without atomic load) which always produces a correct + (though maybe slightly old) value (even on 32-bit CPUs). The value is at + least as new as the latest mutex unlock done by the calling thread. + The assumption is that the system sets both 4-byte halves either at the + same time, or one after the other (in any order), but NOT some bytes of the + first half then some bytes of the second half then the rest of bytes of the + first half. With this assumption, the function can detect when it is + seeing an inconsistent value. + + @param LSN pointer to the LSN variable to read + + @return LSN part (most significant byte always 0) +*/ +#if ( SIZEOF_CHARP >= 8 ) +/* 64-bit CPU, 64-bit reads are atomic */ +#define lsn_read_non_atomic LSN_WITH_FLAGS_TO_LSN +#else +static inline LSN lsn_read_non_atomic_32(const volatile LSN *x) +{ + /* + 32-bit CPU, 64-bit reads may give a mixed of old half and new half (old + low bits and new high bits, or the contrary). + */ + for (;;) /* loop until no atomicity problems */ + { + /* + Remove most significant byte in case this is a LSN_WITH_FLAGS object. + Those flags in TRN::first_undo_lsn break the condition on transitions so + they must be removed below. + */ + LSN y= LSN_WITH_FLAGS_TO_LSN(*x); + if (likely((y == LSN_IMPOSSIBLE) || LSN_VALID(y))) + return y; + } +} +#define lsn_read_non_atomic(x) lsn_read_non_atomic_32(&x) +#endif diff --git a/storage/maria/ma_close.c b/storage/maria/ma_close.c index 34c1bfb4d6d..fdee50f6fde 100644 --- a/storage/maria/ma_close.c +++ b/storage/maria/ma_close.c @@ -85,6 +85,7 @@ int maria_close(register MARIA_HA *info) not change the crashed state. We can NOT write the state in other cases as other threads may be using the file at this point + IF using --external-locking, which does not apply to Maria. */ if (share->mode != O_RDONLY && maria_is_crashed(info)) _ma_state_info_write(share->kfile.file, &share->state, 1); diff --git a/storage/maria/ma_control_file.c b/storage/maria/ma_control_file.c index db5440dc873..66f0c37f4a3 100644 --- a/storage/maria/ma_control_file.c +++ b/storage/maria/ma_control_file.c @@ -40,15 +40,9 @@ #define CONTROL_FILE_FILENO_SIZE 4 #define CONTROL_FILE_SIZE (CONTROL_FILE_FILENO_OFFSET + CONTROL_FILE_FILENO_SIZE) -/* - This module owns these two vars. - uint32 is always atomically updated, but LSN is 8 bytes, we will need - provisions to ensure that it's updated atomically in - ma_control_file_write_and_force(). Probably the log mutex could be - used. TODO. -*/ -LSN last_checkpoint_lsn; -uint32 last_logno; +/* This module owns these two vars. */ +LSN last_checkpoint_lsn= LSN_IMPOSSIBLE; +uint32 last_logno= FILENO_IMPOSSIBLE; /** @brief If log's lock should be asserted when writing to control file. @@ -65,16 +59,16 @@ my_bool maria_multi_threaded= FALSE; static int control_file_fd= -1; /* - Initialize control file subsystem - - SYNOPSIS - ma_control_file_create_or_open() + @brief Initialize control file subsystem - Looks for the control file. If absent, it's a fresh start, creates file. + Looks for the control file. If none and creation is requested, creates file. If present, reads it to find out last checkpoint's LSN and last log, updates the last_checkpoint_lsn and last_logno global variables. Called at engine's start. + @param create_if_missing + + @note The format of the control file is: 4 bytes: magic string 4 bytes: checksum of the following bytes @@ -82,11 +76,11 @@ static int control_file_fd= -1; 4 bytes: offset in log where last checkpoint is 4 bytes: number of last log - RETURN - 0 - OK - 1 - Error (in which case the file is left closed) + @return Operation status + @retval 0 OK + @retval 1 Error (in which case the file is left closed) */ -CONTROL_FILE_ERROR ma_control_file_create_or_open() +CONTROL_FILE_ERROR ma_control_file_create_or_open(my_bool create_if_missing) { char buffer[CONTROL_FILE_SIZE]; char name[FN_REFLEN]; @@ -115,6 +109,8 @@ CONTROL_FILE_ERROR ma_control_file_create_or_open() if (create_file) { + if (!create_if_missing) + DBUG_RETURN(CONTROL_FILE_MISSING); if ((control_file_fd= my_create(name, 0, open_flags, MYF(MY_SYNC_DIR))) < 0) DBUG_RETURN(CONTROL_FILE_UNKNOWN_ERROR); @@ -136,8 +132,8 @@ CONTROL_FILE_ERROR ma_control_file_create_or_open() */ /* init the file with these "undefined" values */ - DBUG_RETURN(ma_control_file_write_and_force(CONTROL_FILE_IMPOSSIBLE_LSN, - CONTROL_FILE_IMPOSSIBLE_FILENO, + DBUG_RETURN(ma_control_file_write_and_force(LSN_IMPOSSIBLE, + FILENO_IMPOSSIBLE, CONTROL_FILE_UPDATE_ALL)); } @@ -315,8 +311,8 @@ int ma_control_file_end() As this module owns these variables, closing the module forbids access to them (just a safety): */ - last_checkpoint_lsn= CONTROL_FILE_IMPOSSIBLE_LSN; - last_logno= CONTROL_FILE_IMPOSSIBLE_FILENO; + last_checkpoint_lsn= LSN_IMPOSSIBLE; + last_logno= FILENO_IMPOSSIBLE; DBUG_RETURN(close_error); } diff --git a/storage/maria/ma_control_file.h b/storage/maria/ma_control_file.h index c974838684b..fa4ec442e41 100644 --- a/storage/maria/ma_control_file.h +++ b/storage/maria/ma_control_file.h @@ -19,27 +19,17 @@ */ #define CONTROL_FILE_BASE_NAME "maria_control" -/* - indicate absence of the log file number; first log is always number 1, 0 is - impossible. -*/ -#define CONTROL_FILE_IMPOSSIBLE_FILENO 0 -/* logs always have a header */ -#define CONTROL_FILE_IMPOSSIBLE_LOG_OFFSET 0 -/* indicate absence of LSN. */ -#define CONTROL_FILE_IMPOSSIBLE_LSN ((LSN)0) /* Here is the interface of this module */ /* LSN of the last checkoint - (if last_checkpoint_lsn == CONTROL_FILE_IMPOSSIBLE_LSN - then there was never a checkpoint) + (if last_checkpoint_lsn == LSN_IMPOSSIBLE then there was never a checkpoint) */ extern LSN last_checkpoint_lsn; /* - Last log number (if last_logno == - CONTROL_FILE_IMPOSSIBLE_FILENO then there is no log file yet) + Last log number (if last_logno == FILENO_IMPOSSIBLE then there is no log + file yet) */ extern uint32 last_logno; @@ -51,6 +41,7 @@ typedef enum enum_control_file_error { CONTROL_FILE_TOO_BIG, CONTROL_FILE_BAD_MAGIC_STRING, CONTROL_FILE_BAD_CHECKSUM, + CONTROL_FILE_MISSING, CONTROL_FILE_UNKNOWN_ERROR /* any other error */ } CONTROL_FILE_ERROR; @@ -63,11 +54,11 @@ extern "C" { #endif /* - Looks for the control file. If absent, it's a fresh start, create file. - If present, read it to find out last checkpoint's LSN and last log. + Looks for the control file. If none and creation was requested, creates file. + If present, reads it to find out last checkpoint's LSN and last log. Called at engine's start. */ -CONTROL_FILE_ERROR ma_control_file_create_or_open(); +CONTROL_FILE_ERROR ma_control_file_create_or_open(my_bool); /* Write information durably to the control file. Called when we have created a new log (after syncing this log's creation) diff --git a/storage/maria/ma_create.c b/storage/maria/ma_create.c index 53e15deb74b..22b490c907c 100644 --- a/storage/maria/ma_create.c +++ b/storage/maria/ma_create.c @@ -52,8 +52,7 @@ int maria_create(const char *name, enum data_file_type datafile_type, unique_key_parts,fulltext_keys,offset, not_block_record_extra_length; uint max_field_lengths, extra_header_size; ulong reclength, real_reclength,min_pack_length; - char filename[FN_REFLEN], dlinkname[FN_REFLEN], *dlinkname_ptr= NULL, - klinkname[FN_REFLEN], *klinkname_ptr= NULL; + char filename[FN_REFLEN], linkname[FN_REFLEN], *linkname_ptr; ulong pack_reclength; ulonglong tot_length,max_rows, tmp; enum en_fieldtype type; @@ -628,7 +627,7 @@ int maria_create(const char *name, enum data_file_type datafile_type, share.state.dellink = HA_OFFSET_ERROR; share.state.first_bitmap_with_space= 0; - share.state.create_rename_lsn= 0; + share.state.create_rename_lsn= LSN_IMPOSSIBLE; share.state.process= (ulong) getpid(); share.state.unique= (ulong) 0; share.state.update_count=(ulong) 0; @@ -721,9 +720,9 @@ int maria_create(const char *name, enum data_file_type datafile_type, MY_UNPACK_FILENAME | (have_iext ? MY_REPLACE_EXT : MY_APPEND_EXT)); } - fn_format(klinkname, name, "", MARIA_NAME_IEXT, + fn_format(linkname, name, "", MARIA_NAME_IEXT, MY_UNPACK_FILENAME|MY_APPEND_EXT); - klinkname_ptr= klinkname; + linkname_ptr= linkname; /* Don't create the table if the link or file exists to ensure that one doesn't accidently destroy another table. @@ -739,6 +738,7 @@ int maria_create(const char *name, enum data_file_type datafile_type, (MY_UNPACK_FILENAME | (flags & HA_DONT_TOUCH_DATA) ? MY_RETURN_REAL_PATH : 0) | MY_APPEND_EXT); + linkname_ptr= NULL; /* Replace the current file. Don't sync dir now if the data file has the same path. @@ -761,7 +761,7 @@ int maria_create(const char *name, enum data_file_type datafile_type, goto err; } - if ((file= my_create_with_symlink(klinkname_ptr, filename, 0, create_mode, + if ((file= my_create_with_symlink(linkname_ptr, filename, 0, create_mode, MYF(MY_WME|create_flag))) < 0) goto err; errpos=1; @@ -788,19 +788,20 @@ int maria_create(const char *name, enum data_file_type datafile_type, MY_UNPACK_FILENAME | (have_dext ? MY_REPLACE_EXT : MY_APPEND_EXT)); } - fn_format(dlinkname, name, "",MARIA_NAME_DEXT, + fn_format(linkname, name, "",MARIA_NAME_DEXT, MY_UNPACK_FILENAME | MY_APPEND_EXT); - dlinkname_ptr= dlinkname; + linkname_ptr= linkname; create_flag=0; } else { fn_format(filename,name,"", MARIA_NAME_DEXT, MY_UNPACK_FILENAME | MY_APPEND_EXT); + linkname_ptr= NULL; create_flag=MY_DELETE_OLD; } if ((dfile= - my_create_with_symlink(dlinkname_ptr, filename, 0, create_mode, + my_create_with_symlink(linkname_ptr, filename, 0, create_mode, MYF(MY_WME | create_flag | sync_dir))) < 0) goto err; errpos=3; @@ -948,15 +949,15 @@ int maria_create(const char *name, enum data_file_type datafile_type, not log 1 KB of mostly zeroes if this is a small table. */ char empty_string[]= ""; - LEX_STRING log_array[TRANSLOG_INTERNAL_PARTS + 3]; + LEX_STRING log_array[TRANSLOG_INTERNAL_PARTS + 4]; uint total_rec_length= 0; uint i; - log_array[TRANSLOG_INTERNAL_PARTS + 0].length= 1 + 2 + + log_array[TRANSLOG_INTERNAL_PARTS + 1].length= 1 + 2 + 2 + kfile_size_before_extension; /* we are needing maybe 64 kB, so don't use the stack */ - log_data= my_malloc(log_array[TRANSLOG_INTERNAL_PARTS + 0].length, MYF(0)); + log_data= my_malloc(log_array[TRANSLOG_INTERNAL_PARTS + 1].length, MYF(0)); if ((log_data == NULL) || - my_pread(file, 1 + 2 + log_data, kfile_size_before_extension, + my_pread(file, 1 + 2 + 2 + log_data, kfile_size_before_extension, 0, MYF(MY_NABP))) goto err_no_lock; /* @@ -965,16 +966,21 @@ int maria_create(const char *name, enum data_file_type datafile_type, */ log_data[0]= test(flags & HA_DONT_TOUCH_DATA); int2store(log_data + 1, kfile_size_before_extension); - log_array[TRANSLOG_INTERNAL_PARTS + 0].str= log_data; + int2store(log_data + 1 + 2, share.base.keystart); + log_array[TRANSLOG_INTERNAL_PARTS + 0].str= (char *)name; + /* we store the end-zero, for Recovery to just pass it to my_create() */ + log_array[TRANSLOG_INTERNAL_PARTS + 0].length= + strlen(log_array[TRANSLOG_INTERNAL_PARTS + 0].str) + 1; + log_array[TRANSLOG_INTERNAL_PARTS + 1].str= log_data; /* symlink description is also needed for re-creation by Recovery: */ - log_array[TRANSLOG_INTERNAL_PARTS + 1].str= - dlinkname_ptr ? dlinkname : empty_string; - log_array[TRANSLOG_INTERNAL_PARTS + 1].length= - strlen(log_array[TRANSLOG_INTERNAL_PARTS + 1].str); - log_array[TRANSLOG_INTERNAL_PARTS + 2].str= - klinkname_ptr ? klinkname : empty_string; + log_array[TRANSLOG_INTERNAL_PARTS + 2].str= (char *) + (ci->data_file_name ? ci->data_file_name : empty_string); log_array[TRANSLOG_INTERNAL_PARTS + 2].length= - strlen(log_array[TRANSLOG_INTERNAL_PARTS + 2].str); + strlen(log_array[TRANSLOG_INTERNAL_PARTS + 2].str) + 1; + log_array[TRANSLOG_INTERNAL_PARTS + 3].str= (char *) + (ci->index_file_name ? ci->index_file_name : empty_string); + log_array[TRANSLOG_INTERNAL_PARTS + 3].length= + strlen(log_array[TRANSLOG_INTERNAL_PARTS + 3].str) + 1; for (i= TRANSLOG_INTERNAL_PARTS; i < (sizeof(log_array)/sizeof(log_array[0])); i++) total_rec_length+= log_array[i].length; diff --git a/storage/maria/ma_loghandler.c b/storage/maria/ma_loghandler.c index 44be624bed0..6f238ef4055 100644 --- a/storage/maria/ma_loghandler.c +++ b/storage/maria/ma_loghandler.c @@ -61,21 +61,6 @@ #define COMPRESSED_LSN_MAX_STORE_SIZE (2 + LSN_STORE_SIZE) #define MAX_NUMBER_OF_LSNS_PER_RECORD 2 -/* record parts descriptor */ -struct st_translog_parts -{ - /* full record length */ - translog_size_t record_length; - /* full record length with chunk headers */ - translog_size_t total_record_length; - /* current part index */ - uint current; - /* total number of elements in parts */ - uint elements; - /* array of parts (LEX_STRING) */ - LEX_STRING *parts; -}; - /* log write buffer descriptor */ struct st_translog_buffer { @@ -176,15 +161,6 @@ static byte end_of_log= 0; my_bool translog_inited= 0; -/* record classes */ -enum record_class -{ - LOGRECTYPE_NOT_ALLOWED, - LOGRECTYPE_VARIABLE_LENGTH, - LOGRECTYPE_PSEUDOFIXEDLENGTH, - LOGRECTYPE_FIXEDLENGTH -}; - /* chunk types */ #define TRANSLOG_CHUNK_LSN 0x00 /* 0 chunk refer as LSN (head or tail */ #define TRANSLOG_CHUNK_FIXED (1 << 6) /* 1 (pseudo)fixed record (also LSN) */ @@ -196,46 +172,6 @@ enum record_class /* compressed (relative) LSN constants */ #define TRANSLOG_CLSN_LEN_BITS 0xC0 /* Mask to get compressed LSN length */ -typedef my_bool(*prewrite_rec_hook) (enum translog_record_type type, - TRN *trn, struct st_maria_share *share, - struct st_translog_parts *parts); - -typedef my_bool(*inwrite_rec_hook) (enum translog_record_type type, - TRN *trn, - LSN *lsn, - struct st_translog_parts *parts); - -typedef uint16(*read_rec_hook) (enum translog_record_type type, - uint16 read_length, uchar *read_buff, - byte *decoded_buff); - -/* - Descriptor of log record type - Note: Don't reorder because of constructs later... -*/ -struct st_log_record_type_descriptor -{ - /* internal class of the record */ - enum record_class class; - /* - length for fixed-size record, pseudo-fixed record - length with uncompressed LSNs - */ - uint16 fixed_length; - /* how much record body (belonged to headers too) read with headers */ - uint16 read_header_len; - /* HOOK for writing the record called before lock */ - prewrite_rec_hook prewrite_hook; - /* HOOK for writing the record called when LSN is known, inside lock */ - inwrite_rec_hook inwrite_hook; - /* HOOK for reading headers */ - read_rec_hook read_hook; - /* - For pseudo fixed records number of compressed LSNs followed by - system header - */ - int16 compressed_LSN; -}; #include @@ -257,27 +193,32 @@ static my_bool write_hook_for_undo(enum translog_record_type type, NOTE that after first public Maria release, these can NOT be changed */ -typedef struct st_log_record_type_descriptor LOG_DESC; -static LOG_DESC log_record_type_descriptor[LOGREC_NUMBER_OF_TYPES]; +LOG_DESC log_record_type_descriptor[LOGREC_NUMBER_OF_TYPES]; static LOG_DESC INIT_LOGREC_FIXED_RECORD_0LSN_EXAMPLE= -{LOGRECTYPE_FIXEDLENGTH, 6, 6, NULL, NULL, NULL, 0}; +{LOGRECTYPE_FIXEDLENGTH, 6, 6, NULL, NULL, NULL, 0, + "fixed0example", FALSE, NULL, NULL}; static LOG_DESC INIT_LOGREC_VARIABLE_RECORD_0LSN_EXAMPLE= -{LOGRECTYPE_VARIABLE_LENGTH, 0, 9, NULL, NULL, NULL, 0}; +{LOGRECTYPE_VARIABLE_LENGTH, 0, 9, NULL, NULL, NULL, 0, +"variable0example", FALSE, NULL, NULL}; static LOG_DESC INIT_LOGREC_FIXED_RECORD_1LSN_EXAMPLE= -{LOGRECTYPE_PSEUDOFIXEDLENGTH, 7, 7, NULL, NULL, NULL, 1}; +{LOGRECTYPE_PSEUDOFIXEDLENGTH, 7, 7, NULL, NULL, NULL, 1, +"fixed1example", FALSE, NULL, NULL}; static LOG_DESC INIT_LOGREC_VARIABLE_RECORD_1LSN_EXAMPLE= -{LOGRECTYPE_VARIABLE_LENGTH, 0, 12, NULL, NULL, NULL, 1}; +{LOGRECTYPE_VARIABLE_LENGTH, 0, 12, NULL, NULL, NULL, 1, +"variable1example", FALSE, NULL, NULL}; static LOG_DESC INIT_LOGREC_FIXED_RECORD_2LSN_EXAMPLE= -{LOGRECTYPE_PSEUDOFIXEDLENGTH, 23, 23, NULL, NULL, NULL, 2}; +{LOGRECTYPE_PSEUDOFIXEDLENGTH, 23, 23, NULL, NULL, NULL, 2, +"fixed2example", FALSE, NULL, NULL}; static LOG_DESC INIT_LOGREC_VARIABLE_RECORD_2LSN_EXAMPLE= -{LOGRECTYPE_VARIABLE_LENGTH, 0, 19, NULL, NULL, NULL, 2}; +{LOGRECTYPE_VARIABLE_LENGTH, 0, 19, NULL, NULL, NULL, 2, +"variable2example", FALSE, NULL, NULL}; void example_loghandler_init() @@ -298,126 +239,158 @@ void example_loghandler_init() static LOG_DESC INIT_LOGREC_RESERVED_FOR_CHUNKS23= -{LOGRECTYPE_NOT_ALLOWED, 0, 0, NULL, NULL, NULL, 0 }; +{LOGRECTYPE_NOT_ALLOWED, 0, 0, NULL, NULL, NULL, 0, + "reserved", FALSE, NULL, NULL }; static LOG_DESC INIT_LOGREC_REDO_INSERT_ROW_HEAD= {LOGRECTYPE_VARIABLE_LENGTH, 0, FILEID_STORE_SIZE + PAGE_STORE_SIZE + DIRPOS_STORE_SIZE, NULL, - write_hook_for_redo, NULL, 0}; + write_hook_for_redo, NULL, 0, + "redo_insert_row_head", FALSE, NULL, NULL}; static LOG_DESC INIT_LOGREC_REDO_INSERT_ROW_TAIL= {LOGRECTYPE_VARIABLE_LENGTH, 0, FILEID_STORE_SIZE + PAGE_STORE_SIZE + DIRPOS_STORE_SIZE, NULL, - write_hook_for_redo, NULL, 0}; + write_hook_for_redo, NULL, 0, + "redo_insert_row_tail", FALSE, NULL, NULL}; static LOG_DESC INIT_LOGREC_REDO_INSERT_ROW_BLOB= -{LOGRECTYPE_VARIABLE_LENGTH, 0, 8, NULL, write_hook_for_redo, NULL, 0}; +{LOGRECTYPE_VARIABLE_LENGTH, 0, 8, NULL, write_hook_for_redo, NULL, 0, + "redo_insert_row_blob", FALSE, NULL, NULL}; /*QQQ:TODO:header???*/ static LOG_DESC INIT_LOGREC_REDO_INSERT_ROW_BLOBS= -{LOGRECTYPE_VARIABLE_LENGTH, 0, 0, NULL, write_hook_for_redo, NULL, 0}; +{LOGRECTYPE_VARIABLE_LENGTH, 0, FILEID_STORE_SIZE, NULL, + write_hook_for_redo, NULL, 0, + "redo_insert_row_blobs", FALSE, NULL, NULL}; static LOG_DESC INIT_LOGREC_REDO_PURGE_ROW_HEAD= {LOGRECTYPE_FIXEDLENGTH, FILEID_STORE_SIZE + PAGE_STORE_SIZE + DIRPOS_STORE_SIZE, FILEID_STORE_SIZE + PAGE_STORE_SIZE + DIRPOS_STORE_SIZE, - NULL, write_hook_for_redo, NULL, 0}; + NULL, write_hook_for_redo, NULL, 0, + "redo_purge_row_head", FALSE, NULL, NULL}; static LOG_DESC INIT_LOGREC_REDO_PURGE_ROW_TAIL= {LOGRECTYPE_FIXEDLENGTH, FILEID_STORE_SIZE + PAGE_STORE_SIZE + DIRPOS_STORE_SIZE, FILEID_STORE_SIZE + PAGE_STORE_SIZE + DIRPOS_STORE_SIZE, - NULL, write_hook_for_redo, NULL, 0}; + NULL, write_hook_for_redo, NULL, 0, + "redo_purge_row_tail", FALSE, NULL, NULL}; /* QQQ: TODO: variable and fixed size??? */ static LOG_DESC INIT_LOGREC_REDO_PURGE_BLOCKS= {LOGRECTYPE_VARIABLE_LENGTH, 0, - FILEID_STORE_SIZE + PAGERANGE_STORE_SIZE + PAGE_STORE_SIZE + - PAGERANGE_STORE_SIZE, - NULL, write_hook_for_redo, NULL, 0}; + FILEID_STORE_SIZE + PAGERANGE_STORE_SIZE, + NULL, write_hook_for_redo, NULL, 0, + "redo_purge_blocks", FALSE, NULL, NULL}; static LOG_DESC INIT_LOGREC_REDO_DELETE_ROW= -{LOGRECTYPE_FIXEDLENGTH, 16, 16, NULL, write_hook_for_redo, NULL, 0}; +{LOGRECTYPE_FIXEDLENGTH, 16, 16, NULL, write_hook_for_redo, NULL, 0, + "redo_delete_row", FALSE, NULL, NULL}; static LOG_DESC INIT_LOGREC_REDO_UPDATE_ROW_HEAD= -{LOGRECTYPE_VARIABLE_LENGTH, 0, 9, NULL, write_hook_for_redo, NULL, 0}; +{LOGRECTYPE_VARIABLE_LENGTH, 0, 9, NULL, write_hook_for_redo, NULL, 0, + "redo_update_row_head", FALSE, NULL, NULL}; static LOG_DESC INIT_LOGREC_REDO_INDEX= -{LOGRECTYPE_VARIABLE_LENGTH, 0, 9, NULL, write_hook_for_redo, NULL, 0}; +{LOGRECTYPE_VARIABLE_LENGTH, 0, 9, NULL, write_hook_for_redo, NULL, 0, + "redo_index", FALSE, NULL, NULL}; static LOG_DESC INIT_LOGREC_REDO_UNDELETE_ROW= -{LOGRECTYPE_FIXEDLENGTH, 16, 16, NULL, write_hook_for_redo, NULL, 0}; +{LOGRECTYPE_FIXEDLENGTH, 16, 16, NULL, write_hook_for_redo, NULL, 0, + "redo_undelete_row", FALSE, NULL, NULL}; static LOG_DESC INIT_LOGREC_CLR_END= -{LOGRECTYPE_PSEUDOFIXEDLENGTH, 5, 5, NULL, write_hook_for_redo, NULL, 1}; +{LOGRECTYPE_PSEUDOFIXEDLENGTH, 5, 5, NULL, write_hook_for_redo, NULL, 1, + "clr_end", TRUE, NULL, NULL}; static LOG_DESC INIT_LOGREC_PURGE_END= -{LOGRECTYPE_PSEUDOFIXEDLENGTH, 5, 5, NULL, NULL, NULL, 1}; +{LOGRECTYPE_PSEUDOFIXEDLENGTH, 5, 5, NULL, NULL, NULL, 1, + "purge_end", TRUE, NULL, NULL}; static LOG_DESC INIT_LOGREC_UNDO_ROW_INSERT= {LOGRECTYPE_FIXEDLENGTH, LSN_STORE_SIZE + FILEID_STORE_SIZE + PAGE_STORE_SIZE + DIRPOS_STORE_SIZE, LSN_STORE_SIZE + FILEID_STORE_SIZE + PAGE_STORE_SIZE + DIRPOS_STORE_SIZE, - NULL, write_hook_for_undo, NULL, 0}; + NULL, write_hook_for_undo, NULL, 0, + "undo_row_insert", TRUE, NULL, NULL}; static LOG_DESC INIT_LOGREC_UNDO_ROW_DELETE= {LOGRECTYPE_VARIABLE_LENGTH, 0, LSN_STORE_SIZE + FILEID_STORE_SIZE + PAGE_STORE_SIZE + DIRPOS_STORE_SIZE, - NULL, write_hook_for_undo, NULL, 0}; + NULL, write_hook_for_undo, NULL, 0, + "undo_row_delete", TRUE, NULL, NULL}; static LOG_DESC INIT_LOGREC_UNDO_ROW_UPDATE= {LOGRECTYPE_VARIABLE_LENGTH, 0, LSN_STORE_SIZE + FILEID_STORE_SIZE + PAGE_STORE_SIZE + DIRPOS_STORE_SIZE, - NULL, write_hook_for_undo, NULL, 1}; + NULL, write_hook_for_undo, NULL, 1, + "undo_row_update", TRUE, NULL, NULL}; static LOG_DESC INIT_LOGREC_UNDO_ROW_PURGE= {LOGRECTYPE_PSEUDOFIXEDLENGTH, LSN_STORE_SIZE, LSN_STORE_SIZE, - NULL, NULL, NULL, 1}; + NULL, NULL, NULL, 1, + "undo_row_purge", TRUE, NULL, NULL}; static LOG_DESC INIT_LOGREC_UNDO_KEY_INSERT= -{LOGRECTYPE_VARIABLE_LENGTH, 0, 10, NULL, write_hook_for_undo, NULL, 1}; +{LOGRECTYPE_VARIABLE_LENGTH, 0, 10, NULL, write_hook_for_undo, NULL, 1, + "undo_key_insert", TRUE, NULL, NULL}; static LOG_DESC INIT_LOGREC_UNDO_KEY_DELETE= -{LOGRECTYPE_VARIABLE_LENGTH, 0, 15, NULL, write_hook_for_undo, NULL, 0}; +{LOGRECTYPE_VARIABLE_LENGTH, 0, 15, NULL, write_hook_for_undo, NULL, 0, + "undo_key_delete", TRUE, NULL, NULL}; // QQ: why not compressed? static LOG_DESC INIT_LOGREC_PREPARE= -{LOGRECTYPE_VARIABLE_LENGTH, 0, 0, NULL, NULL, NULL, 0}; +{LOGRECTYPE_VARIABLE_LENGTH, 0, 0, NULL, NULL, NULL, 0, + "prepare", TRUE, NULL, NULL}; static LOG_DESC INIT_LOGREC_PREPARE_WITH_UNDO_PURGE= -{LOGRECTYPE_VARIABLE_LENGTH, 0, 5, NULL, NULL, NULL, 1}; +{LOGRECTYPE_VARIABLE_LENGTH, 0, 5, NULL, NULL, NULL, 1, + "prepare_with_undo_purge", TRUE, NULL, NULL}; static LOG_DESC INIT_LOGREC_COMMIT= -{LOGRECTYPE_FIXEDLENGTH, 0, 0, NULL, NULL, NULL, 0}; +{LOGRECTYPE_FIXEDLENGTH, 0, 0, NULL, NULL, NULL, 0, + "commit", TRUE, NULL, NULL}; static LOG_DESC INIT_LOGREC_COMMIT_WITH_UNDO_PURGE= -{LOGRECTYPE_PSEUDOFIXEDLENGTH, 5, 5, NULL, NULL, NULL, 1}; +{LOGRECTYPE_PSEUDOFIXEDLENGTH, 5, 5, NULL, NULL, NULL, 1, + "commit_with_undo_purge", TRUE, NULL, NULL}; static LOG_DESC INIT_LOGREC_CHECKPOINT= -{LOGRECTYPE_VARIABLE_LENGTH, 0, 0, NULL, NULL, NULL, 0}; +{LOGRECTYPE_VARIABLE_LENGTH, 0, 0, NULL, NULL, NULL, 0, + "checkpoint", TRUE, NULL, NULL}; static LOG_DESC INIT_LOGREC_REDO_CREATE_TABLE= -{LOGRECTYPE_VARIABLE_LENGTH, 0, 0, NULL, NULL, NULL, 0}; +{LOGRECTYPE_VARIABLE_LENGTH, 0, 1 + 2, NULL, NULL, NULL, 0, +"redo_create_table", TRUE, NULL, NULL}; static LOG_DESC INIT_LOGREC_REDO_RENAME_TABLE= -{LOGRECTYPE_VARIABLE_LENGTH, 0, 0, NULL, NULL, NULL, 0}; +{LOGRECTYPE_VARIABLE_LENGTH, 0, 0, NULL, NULL, NULL, 0, + "redo_rename_table", TRUE, NULL, NULL}; static LOG_DESC INIT_LOGREC_REDO_DROP_TABLE= -{LOGRECTYPE_VARIABLE_LENGTH, 0, 0, NULL, NULL, NULL, 0}; +{LOGRECTYPE_VARIABLE_LENGTH, 0, 0, NULL, NULL, NULL, 0, + "redo_drop_table", TRUE, NULL, NULL}; static LOG_DESC INIT_LOGREC_REDO_DELETE_ALL= {LOGRECTYPE_FIXEDLENGTH, FILEID_STORE_SIZE, FILEID_STORE_SIZE, - NULL, NULL, NULL, 0}; + NULL, NULL, NULL, 0, + "redo_delete_all", TRUE, NULL, NULL}; static LOG_DESC INIT_LOGREC_REDO_REPAIR_TABLE= {LOGRECTYPE_FIXEDLENGTH, FILEID_STORE_SIZE + 4, FILEID_STORE_SIZE + 4, - NULL, NULL, NULL, 0}; + NULL, NULL, NULL, 0, + "redo_repair_table", TRUE, NULL, NULL}; static LOG_DESC INIT_LOGREC_FILE_ID= -{LOGRECTYPE_VARIABLE_LENGTH, 0, 4, NULL, NULL, NULL, 0}; +{LOGRECTYPE_VARIABLE_LENGTH, 0, 2, NULL, NULL, NULL, 0, + "file_id", TRUE, NULL, NULL}; static LOG_DESC INIT_LOGREC_LONG_TRANSACTION_ID= -{LOGRECTYPE_FIXEDLENGTH, 6, 6, NULL, NULL, NULL, 0}; +{LOGRECTYPE_FIXEDLENGTH, 6, 6, NULL, NULL, NULL, 0, + "long_transaction_id", TRUE, NULL, NULL}; const myf log_write_flags= MY_WME | MY_NABP | MY_WAIT_IF_FULL; @@ -701,7 +674,7 @@ static my_bool translog_buffer_init(struct st_translog_buffer *buffer) { DBUG_ENTER("translog_buffer_init"); /* This buffer offset */ - buffer->last_lsn= CONTROL_FILE_IMPOSSIBLE_LSN; + buffer->last_lsn= LSN_IMPOSSIBLE; /* This Buffer File */ buffer->file= -1; buffer->overlay= 0; @@ -779,7 +752,7 @@ static my_bool translog_create_new_file() translog_write_file_header()) DBUG_RETURN(1); - if (ma_control_file_write_and_force(CONTROL_FILE_IMPOSSIBLE_LSN, file_no, + if (ma_control_file_write_and_force(LSN_IMPOSSIBLE, file_no, CONTROL_FILE_UPDATE_ONLY_LOGNO)) DBUG_RETURN(1); @@ -1206,7 +1179,7 @@ static void translog_start_buffer(struct st_translog_buffer *buffer, (ulong) LSN_OFFSET(log_descriptor.horizon), (ulong) LSN_OFFSET(log_descriptor.horizon))); DBUG_ASSERT(buffer_no == buffer->buffer_no); - buffer->last_lsn= CONTROL_FILE_IMPOSSIBLE_LSN; + buffer->last_lsn= LSN_IMPOSSIBLE; buffer->offset= log_descriptor.horizon; buffer->file= log_descriptor.log_file_num[0]; buffer->overlay= 0; @@ -2088,7 +2061,7 @@ my_bool translog_init(const char *directory, i, (ulong) log_descriptor.buffers + i)); } - logs_found= (last_logno != CONTROL_FILE_IMPOSSIBLE_FILENO); + logs_found= (last_logno != FILENO_IMPOSSIBLE); if (logs_found) { @@ -2100,7 +2073,7 @@ my_bool translog_init(const char *directory, find the log end */ - if (LSN_FILE_NO(last_checkpoint_lsn) == CONTROL_FILE_IMPOSSIBLE_FILENO) + if (LSN_FILE_NO(last_checkpoint_lsn) == FILENO_IMPOSSIBLE) { DBUG_ASSERT(LSN_OFFSET(last_checkpoint_lsn) == 0); /* there was no checkpoints we will read from the beginning */ @@ -2138,7 +2111,7 @@ my_bool translog_init(const char *directory, /* TODO: check page size */ - last_valid_page= CONTROL_FILE_IMPOSSIBLE_LSN; + last_valid_page= LSN_IMPOSSIBLE; /* scan and validate pages */ do { @@ -2186,7 +2159,7 @@ my_bool translog_init(const char *directory, current_page= LSN_REPLACE_OFFSET(current_page, TRANSLOG_PAGE_SIZE); } while (LSN_FILE_NO(current_page) <= LSN_FILE_NO(last_page) && !old_log_was_recovered); - if (last_valid_page == CONTROL_FILE_IMPOSSIBLE_LSN) + if (last_valid_page == LSN_IMPOSSIBLE) { /* Panic!!! Even page which should be valid is invalid */ /* TODO: issue error */ @@ -2272,7 +2245,7 @@ my_bool translog_init(const char *directory, open_logfile_by_number_no_cache(1)) == -1 || translog_write_file_header()) DBUG_RETURN(1); - if (ma_control_file_write_and_force(CONTROL_FILE_IMPOSSIBLE_LSN, 1, + if (ma_control_file_write_and_force(LSN_IMPOSSIBLE, 1, CONTROL_FILE_UPDATE_ONLY_LOGNO)) DBUG_RETURN(1); /* assign buffer 0 */ @@ -2405,7 +2378,7 @@ void translog_destroy() 1 Error */ -static my_bool translog_lock() +my_bool translog_lock() { struct st_translog_buffer *current_buffer; DBUG_ENTER("translog_lock"); @@ -2438,7 +2411,7 @@ static my_bool translog_lock() 1 Error */ -static inline my_bool translog_unlock() +my_bool translog_unlock() { DBUG_ENTER("translog_unlock"); translog_buffer_unlock(log_descriptor.bc.buffer); @@ -4312,14 +4285,14 @@ my_bool translog_write_record(LSN *lsn, } if (unlikely(!(trn->first_undo_lsn & TRANSACTION_LOGGED_LONG_ID))) { - LSN lsn; + LSN dummy_lsn; LEX_STRING log_array[TRANSLOG_INTERNAL_PARTS + 1]; uchar log_data[6]; int6store(log_data, trn->trid); log_array[TRANSLOG_INTERNAL_PARTS + 0].str= (char*) log_data; log_array[TRANSLOG_INTERNAL_PARTS + 0].length= sizeof(log_data); trn->first_undo_lsn|= TRANSACTION_LOGGED_LONG_ID; /* no recursion */ - if (unlikely(translog_write_record(&lsn, LOGREC_LONG_TRANSACTION_ID, + if (unlikely(translog_write_record(&dummy_lsn, LOGREC_LONG_TRANSACTION_ID, trn, NULL, sizeof(log_data), sizeof(log_array)/sizeof(log_array[0]), log_array, NULL))) @@ -4404,6 +4377,8 @@ my_bool translog_write_record(LSN *lsn, } } + DBUG_PRINT("info", ("LSN: (%lu,0x%lx)", (ulong) LSN_FILE_NO(*lsn), + (ulong) LSN_OFFSET(*lsn))); DBUG_RETURN(rc); } @@ -5093,7 +5068,7 @@ translog_read_record_header_scan(TRANSLOG_SCANNER_DATA - it is like translog_read_record_header, but read next record, so see its NOTES. - in case of end of the log buff->lsn will be set to - (CONTROL_FILE_IMPOSSIBLE_LSN) + (LSN_IMPOSSIBLE) RETURN 0 error @@ -5138,7 +5113,7 @@ translog_size_t translog_read_next_record_header(TRANSLOG_SCANNER_DATA if (scanner->page[scanner->page_offset] == 0) { /* Last record was read */ - buff->lsn= CONTROL_FILE_IMPOSSIBLE_LSN; + buff->lsn= LSN_IMPOSSIBLE; /* Return 'end of log' marker */ DBUG_RETURN(TRANSLOG_RECORD_HEADER_MAX_SIZE + 1); } @@ -5300,7 +5275,7 @@ translog_size_t translog_read_record(LSN lsn, if (data == NULL) { - DBUG_ASSERT(lsn != CONTROL_FILE_IMPOSSIBLE_LSN); + DBUG_ASSERT(lsn != LSN_IMPOSSIBLE); data= &internal_data; } if (lsn || @@ -5739,7 +5714,7 @@ int translog_assign_id_to_share(MARIA_SHARE *share, TRN *trn) strlen() */ log_array[TRANSLOG_INTERNAL_PARTS + 1].length= - strlen(share->open_file_name); + strlen(share->open_file_name) + 1; if (unlikely(translog_write_record(&lsn, LOGREC_FILE_ID, trn, share, sizeof(log_data) + log_array[TRANSLOG_INTERNAL_PARTS + @@ -5773,3 +5748,15 @@ void translog_deassign_id_from_share(MARIA_SHARE *share) my_atomic_storeptr((void **)&id_to_share[share->id], 0); my_atomic_rwlock_rdunlock(&LOCK_id_to_share); } + + +/** + @brief returns the LSN of the first record starting in this log + + @note so far works only for the very first log created on this system +*/ + +LSN first_lsn_in_log() +{ + return MAKE_LSN(1, TRANSLOG_PAGE_SIZE + log_descriptor.page_overhead); +} diff --git a/storage/maria/ma_loghandler.h b/storage/maria/ma_loghandler.h index 0a160a9bc53..22b8cca3a08 100644 --- a/storage/maria/ma_loghandler.h +++ b/storage/maria/ma_loghandler.h @@ -1,3 +1,8 @@ +// TODO copyright + +#ifndef _ma_loghandler_h +#define _ma_loghandler_h + /* transaction log default cache size (TODO: make it global variable) */ #define TRANSLOG_PAGECACHE_SIZE 1024*1024*2 /* transaction log default file size (TODO: make it global variable) */ @@ -20,6 +25,7 @@ #define TRANSLOG_PAGE_SIZE (8*1024) #include "ma_loghandler_lsn.h" +#include "trnman_public.h" /* short transaction ID type */ typedef uint16 SHORT_TRANSACTION_ID; @@ -41,6 +47,10 @@ struct st_maria_share; #define page_store(T,A) int5store(T,A) #define dirpos_store(T,A) ((*(uchar*) (T)) = A) #define pagerange_store(T,A) int2store(T,A) +#define fileid_korr(P) uint2korr(P) +#define page_korr(P) uint5korr(P) +#define dirpos_korr(P) (P[0]) +#define pagerange_korr(P) uint2korr(P) /* Length of disk drive sector size (we assume that writing it @@ -228,10 +238,99 @@ extern translog_size_t translog_read_next_record_header(TRANSLOG_SCANNER_DATA *scanner, TRANSLOG_HEADER_BUFFER *buff); +extern my_bool translog_lock(); +extern my_bool translog_unlock(); extern void translog_lock_assert_owner(); extern TRANSLOG_ADDRESS translog_get_horizon(); extern int translog_assign_id_to_share(struct st_maria_share *share, struct st_transaction *trn); extern void translog_deassign_id_from_share(struct st_maria_share *share); extern my_bool translog_inited; + +/* + all the rest added because of recovery; should we make + ma_loghandler_for_recovery.h ? +*/ +extern LSN first_lsn_in_log(); + +/* record parts descriptor */ +struct st_translog_parts +{ + /* full record length */ + translog_size_t record_length; + /* full record length with chunk headers */ + translog_size_t total_record_length; + /* current part index */ + uint current; + /* total number of elements in parts */ + uint elements; + /* array of parts (LEX_STRING) */ + LEX_STRING *parts; +}; + +typedef my_bool(*prewrite_rec_hook) (enum translog_record_type type, + TRN *trn, struct st_maria_share *share, + struct st_translog_parts *parts); + +typedef my_bool(*inwrite_rec_hook) (enum translog_record_type type, + TRN *trn, + LSN *lsn, + struct st_translog_parts *parts); + +typedef uint16(*read_rec_hook) (enum translog_record_type type, + uint16 read_length, uchar *read_buff, + byte *decoded_buff); + + +/* record classes */ +enum record_class +{ + LOGRECTYPE_NOT_ALLOWED, + LOGRECTYPE_VARIABLE_LENGTH, + LOGRECTYPE_PSEUDOFIXEDLENGTH, + LOGRECTYPE_FIXEDLENGTH +}; + +/* C++ can't bear that a variable's name is "class" */ +#ifndef __cplusplus +/* + Descriptor of log record type + Note: Don't reorder because of constructs later... +*/ +typedef struct st_log_record_type_descriptor +{ + /* internal class of the record */ + enum record_class class; + /* + length for fixed-size record, pseudo-fixed record + length with uncompressed LSNs + */ + uint16 fixed_length; + /* how much record body (belonged to headers too) read with headers */ + uint16 read_header_len; + /* HOOK for writing the record called before lock */ + prewrite_rec_hook prewrite_hook; + /* HOOK for writing the record called when LSN is known, inside lock */ + inwrite_rec_hook inwrite_hook; + /* HOOK for reading headers */ + read_rec_hook read_hook; + /* + For pseudo fixed records number of compressed LSNs followed by + system header + */ + int16 compressed_LSN; + /* the rest is for maria_read_log & Recovery */ + /** @brief for debug error messages or "maria_read_log" command-line tool */ + const char *name; + my_bool record_ends_group; + /* a function to execute when we see the record during the REDO phase */ + int (*record_execute_in_redo_phase)(const TRANSLOG_HEADER_BUFFER *); + /* a function to execute when we see the record during the UNDO phase */ + int (*record_execute_in_undo_phase)(const TRANSLOG_HEADER_BUFFER *); +} LOG_DESC; + +extern LOG_DESC log_record_type_descriptor[LOGREC_NUMBER_OF_TYPES]; +#endif + C_MODE_END +#endif diff --git a/storage/maria/ma_loghandler_lsn.h b/storage/maria/ma_loghandler_lsn.h index c641337e8ba..af7594e3b00 100644 --- a/storage/maria/ma_loghandler_lsn.h +++ b/storage/maria/ma_loghandler_lsn.h @@ -24,7 +24,7 @@ typedef TRANSLOG_ADDRESS LSN; #define LSN_FILE_NO(L) ((L) >> 32) /* Gets raw file number part of a LSN/log address */ -#define LSN_FINE_NO_PART(L) ((L) & ((int64)0xFFFFFF00000000LL)) +#define LSN_FILE_NO_PART(L) ((L) & ((int64)0xFFFFFF00000000LL)) /* Gets record offset of a LSN/log address */ #define LSN_OFFSET(L) ((L) & 0xFFFFFFFFL) @@ -33,7 +33,9 @@ typedef TRANSLOG_ADDRESS LSN; #define MAKE_LSN(F,S) ((((uint64)(F)) << 32) | (S)) /* checks LSN */ -#define LSN_VALID(L) DBUG_ASSERT((L) >= 0 && (L) < (uint64)0xFFFFFFFFFFFFFFLL) +#define LSN_VALID(L) \ + ((LSN_FILE_NO_PART(L) != FILENO_IMPOSSIBLE) && \ + (LSN_OFFSET(L) != LOG_OFFSET_IMPOSSIBLE)) /* size of stored LSN on a disk, don't change it! */ #define LSN_STORE_SIZE 7 @@ -51,7 +53,7 @@ typedef TRANSLOG_ADDRESS LSN; /* what we need to add to LSN to increase it on one file */ #define LSN_ONE_FILE ((int64)0x100000000LL) -#define LSN_REPLACE_OFFSET(L, S) (LSN_FINE_NO_PART(L) | (S)) +#define LSN_REPLACE_OFFSET(L, S) (LSN_FILE_NO_PART(L) | (S)) /* an 8-byte type whose most significant byte is used for "flags"; 7 @@ -61,4 +63,7 @@ typedef LSN LSN_WITH_FLAGS; #define LSN_WITH_FLAGS_TO_LSN(x) (x & ULL(0x00FFFFFFFFFFFFFF)) #define LSN_WITH_FLAGS_TO_FLAGS(x) (x & ULL(0xFF00000000000000)) +#define FILENO_IMPOSSIBLE 0 /**< log file's numbering starts at 1 */ +#define LOG_OFFSET_IMPOSSIBLE 0 /**< log always has a header */ +#define LSN_IMPOSSIBLE 0 #endif diff --git a/storage/maria/ma_pagecache.c b/storage/maria/ma_pagecache.c index ae42f702b0a..b1ebfbbe7c6 100755 --- a/storage/maria/ma_pagecache.c +++ b/storage/maria/ma_pagecache.c @@ -587,11 +587,7 @@ static uint pagecache_fwrite(PAGECACHE *pagecache, DBUG_PRINT("info", ("Log handler call")); /* TODO: integrate with page format */ lsn= lsn_korr(buffer + PAGE_LSN_OFFSET); - /* - check CONTROL_FILE_IMPOSSIBLE_FILENO & - CONTROL_FILE_IMPOSSIBLE_LOG_OFFSET - */ - DBUG_ASSERT(lsn != 0); + DBUG_ASSERT(LSN_VALID(lsn)); translog_flush(lsn); } DBUG_RETURN(my_pwrite(filedesc->file, buffer, pagecache->block_size, @@ -2474,7 +2470,7 @@ static void check_and_set_lsn(LSN lsn, PAGECACHE_BLOCK_LINK *block) lock lock change pin pin page first_REDO_LSN_for_page do not set it if it is zero - lsn if it is not CONTROL_FILE_IMPOSSIBLE_LSN (0) and it + lsn if it is not LSN_IMPOSSIBLE (0) and it is bigger then LSN on the page it will be written on the page @@ -2566,7 +2562,7 @@ void pagecache_unlock(PAGECACHE *pagecache, pagecache pointer to a page cache data structure file handler for the file for the block of data to be read pageno number of the block of data in the file - lsn if it is not CONTROL_FILE_IMPOSSIBLE_LSN (0) and it + lsn if it is not LSN_IMPOSSIBLE (0) and it is bigger then LSN on the page it will be written on the page */ @@ -2635,10 +2631,9 @@ void pagecache_unpin(PAGECACHE *pagecache, link direct link to page (returned by read or write) lock lock change pin pin page - first_REDO_LSN_for_page do not set it if it is zero - lsn if it is not CONTROL_FILE_IMPOSSIBLE_LSN (0) and it - is bigger then LSN on the page it will be written on - the page + first_REDO_LSN_for_page do not set it if it is LSN_IMPOSSIBLE (0) + lsn if it is not LSN_IMPOSSIBLE and it is bigger then + LSN on the page it will be written on the page */ void pagecache_unlock_by_link(PAGECACHE *pagecache, @@ -2681,7 +2676,7 @@ void pagecache_unlock_by_link(PAGECACHE *pagecache, DBUG_ASSERT(pagecache->can_be_used); inc_counter_for_resize_op(pagecache); - if (first_REDO_LSN_for_page) + if (first_REDO_LSN_for_page != LSN_IMPOSSIBLE) { /* LOCK_READ_UNLOCK is ok here as the page may have first locked @@ -2694,10 +2689,8 @@ void pagecache_unlock_by_link(PAGECACHE *pagecache, if (block->rec_lsn == 0) block->rec_lsn= first_REDO_LSN_for_page; } - if (lsn != 0) - { + if (lsn != LSN_IMPOSSIBLE) check_and_set_lsn(lsn, block); - } if (make_lock_and_pin(pagecache, block, lock, pin)) DBUG_ASSERT(0); /* should not happend */ @@ -2726,7 +2719,7 @@ void pagecache_unlock_by_link(PAGECACHE *pagecache, pagecache_unpin_by_link() pagecache pointer to a page cache data structure link direct link to page (returned by read or write) - lsn if it is not CONTROL_FILE_IMPOSSIBLE_LSN (0) and it + lsn if it is not LSN_IMPOSSIBLE (0) and it is bigger then LSN on the page it will be written on the page */ diff --git a/storage/maria/ma_recovery.h b/storage/maria/ma_recovery.h index d2901f5724c..42c5071babd 100644 --- a/storage/maria/ma_recovery.h +++ b/storage/maria/ma_recovery.h @@ -1,4 +1,4 @@ -/* Copyright (C) 2006 MySQL AB & MySQL Finland AB & TCX DataKonsult AB +/* Copyright (C) 2006,2007 MySQL AB 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 diff --git a/storage/maria/ma_test1.c b/storage/maria/ma_test1.c index 028e02ab9d1..a87e3445082 100644 --- a/storage/maria/ma_test1.c +++ b/storage/maria/ma_test1.c @@ -60,7 +60,7 @@ int main(int argc,char *argv[]) if (maria_init() || (init_pagecache(maria_pagecache, IO_SIZE*16, 0, 0, maria_block_size) == 0) || - ma_control_file_create_or_open() || + ma_control_file_create_or_open(TRUE) || (init_pagecache(maria_log_pagecache, TRANSLOG_PAGECACHE_SIZE, 0, 0, TRANSLOG_PAGE_SIZE) == 0) || diff --git a/storage/maria/ma_test2.c b/storage/maria/ma_test2.c index bbbb4fca1bf..1839efd0813 100644 --- a/storage/maria/ma_test2.c +++ b/storage/maria/ma_test2.c @@ -224,7 +224,7 @@ int main(int argc, char *argv[]) /* Maria requires that we always have a page cache */ if ((init_pagecache(maria_pagecache, pagecache_size, 0, 0, maria_block_size) == 0) || - ma_control_file_create_or_open() || + ma_control_file_create_or_open(TRUE) || (init_pagecache(maria_log_pagecache, TRANSLOG_PAGECACHE_SIZE, 0, 0, TRANSLOG_PAGE_SIZE) == 0) || diff --git a/storage/maria/maria_read_log.c b/storage/maria/maria_read_log.c new file mode 100644 index 00000000000..7bb15e27f7a --- /dev/null +++ b/storage/maria/maria_read_log.c @@ -0,0 +1,696 @@ +/* Copyright (C) 2007 MySQL AB + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ + +#include "maria_def.h" +#include + +#define PCACHE_SIZE (1024*1024*10) +#define LOG_FLAGS 0 +#define LOG_FILE_SIZE (1024L*1024L) + + +static PAGECACHE pagecache; + +static const char *load_default_groups[]= { "maria_read_log",0 }; +static void get_options(int *argc,char * * *argv); +#ifndef DBUG_OFF +static const char *default_dbug_option; +#endif +static my_bool opt_only_display, opt_display_and_apply; + +struct TRN_FOR_RECOVERY +{ + LSN group_start_lsn, undo_lsn; + TrID long_trid; +}; + +struct TRN_FOR_RECOVERY all_active_trans[SHORT_TRID_MAX + 1]; +MARIA_HA *all_tables[SHORT_TRID_MAX + 1]; + +static void end_of_redo_phase(); +static void display_record_position(const LOG_DESC *log_desc, + const TRANSLOG_HEADER_BUFFER *rec, + uint number); +static int display_and_apply_record(const LOG_DESC *log_desc, + const TRANSLOG_HEADER_BUFFER *rec); +#define prototype_exec_hook(R) \ +static int exec_LOGREC_ ## R(const TRANSLOG_HEADER_BUFFER *rec) +prototype_exec_hook(LONG_TRANSACTION_ID); +prototype_exec_hook(CHECKPOINT); +prototype_exec_hook(REDO_CREATE_TABLE); +prototype_exec_hook(FILE_ID); +prototype_exec_hook(REDO_INSERT_ROW_HEAD); +prototype_exec_hook(COMMIT); +/* + To implement REDO_DROP_TABLE and REDO_RENAME_TABLE, we would need to go + through the all_tables[] array, find all open instances of the + table-to-drop-or-rename, and remove them from the array. + We however know that in real Recovery, we don't have to handle those log + records at all, same for REDO_CREATE_TABLE. + So for now, we can use this program to replay/debug a sequence of CREATE + + DMLs, but not DROP/RENAME; it is probably enough for a start. +*/ + +int main(int argc, char **argv) +{ + LSN lsn; + char **default_argv; + MY_INIT(argv[0]); + + load_defaults("my", load_default_groups, &argc, &argv); + default_argv= argv; + get_options(&argc, &argv); + + maria_data_root= "."; + +#ifndef DBUG_OFF +#if defined(__WIN__) + default_dbug_option= "d:t:i:O,\\maria_read_log.trace"; +#else + default_dbug_option= "d:t:i:o,/tmp/maria_read_log.trace"; +#endif + if (argc > 1) + { + DBUG_SET(default_dbug_option); + DBUG_SET_INITIAL(default_dbug_option); + } +#endif + + if (maria_init()) + { + fprintf(stderr, "Can't init Maria engine (%d)\n", errno); + goto err; + } + /* we don't want to create a control file, it MUST exist */ + if (ma_control_file_create_or_open(FALSE)) + { + fprintf(stderr, "Can't open control file (%d)\n", errno); + goto err; + } + if (last_logno == FILENO_IMPOSSIBLE) + { + fprintf(stderr, "Can't find any log\n"); + goto err; + } + if (init_pagecache(&pagecache, PCACHE_SIZE, 0, 0, + TRANSLOG_PAGE_SIZE) == 0) + { + fprintf(stderr, "Got error in init_pagecache() (errno: %d)\n", errno); + goto err; + } + /* + If log handler does not find the "last_logno" log it will return error, + which is good. + But if it finds a log and this log was crashed, it will create a new log, + which is useless. TODO: start log handler in read-only mode. + */ + if (translog_init(".", LOG_FILE_SIZE, 50112, 0, &pagecache, + TRANSLOG_DEFAULT_FLAGS)) + { + fprintf(stderr, "Can't init loghandler (%d)\n", errno); + goto err; + } + + /* install hooks for execution */ +#define install_exec_hook(R) \ + log_record_type_descriptor[LOGREC_ ## R].record_execute_in_redo_phase= \ + exec_LOGREC_ ## R; + install_exec_hook(LONG_TRANSACTION_ID); + install_exec_hook(CHECKPOINT); + install_exec_hook(REDO_CREATE_TABLE); + install_exec_hook(FILE_ID); + install_exec_hook(REDO_INSERT_ROW_HEAD); + install_exec_hook(COMMIT); + + if (opt_only_display) + printf("You are using --only-display, NOTHING will be written to disk\n"); + + lsn= first_lsn_in_log(); /*could also be last_checkpoint_lsn */ + + TRANSLOG_HEADER_BUFFER rec; + struct st_translog_scanner_data scanner; + uint i= 1; + + translog_size_t len= translog_read_record_header(lsn, &rec); + + if (len == (TRANSLOG_RECORD_HEADER_MAX_SIZE + 1)) + { + printf("EOF on the log\n"); + goto end; + } + + if (translog_init_scanner(lsn, 1, &scanner)) + { + fprintf(stderr, "Scanner init failed\n"); + goto err; + } + for (;;i++) + { + uint16 sid= rec.short_trid; + const LOG_DESC *log_desc= &log_record_type_descriptor[rec.type]; + display_record_position(log_desc, &rec, i); + + /* + A complete group is a set of log records with an "end mark" record + (e.g. a set of REDOs for an operation, terminated by an UNDO for this + operation); if there is no "end mark" record the group is incomplete + and won't be executed. + */ + if (log_desc->record_ends_group) + { + if (all_active_trans[sid].group_start_lsn != LSN_IMPOSSIBLE) + { + /* + There is a complete group for this transaction, containing more than + this event. + */ + printf(" ends a group:\n"); + struct st_translog_scanner_data scanner2; + TRANSLOG_HEADER_BUFFER rec2; + len= + translog_read_record_header(all_active_trans[sid].group_start_lsn, &rec2); + if (len == (TRANSLOG_RECORD_HEADER_MAX_SIZE + 1)) + { + fprintf(stderr, "Cannot find record where it should be\n"); + goto err; + } + if (translog_init_scanner(rec2.lsn, 1, &scanner2)) + { + fprintf(stderr, "Scanner2 init failed\n"); + goto err; + } + do + { + if (rec2.short_trid == sid) /* it's in our group */ + { + const LOG_DESC *log_desc2= &log_record_type_descriptor[rec2.type]; + display_record_position(log_desc2, &rec2, 0); + if (display_and_apply_record(log_desc2, &rec2)) + goto err; + } + len= translog_read_next_record_header(&scanner2, &rec2); + if (len == (TRANSLOG_RECORD_HEADER_MAX_SIZE + 1)) + { + fprintf(stderr, "Cannot find record where it should be\n"); + goto err; + } + } + while (rec2.lsn < rec.lsn); + translog_free_record_header(&rec2); + /* group finished */ + all_active_trans[sid].group_start_lsn= LSN_IMPOSSIBLE; + } + if (display_and_apply_record(log_desc, &rec)) + goto err; + } + else /* record does not end group */ + { + /* just record the fact, can't know if can execute yet */ + if (all_active_trans[sid].group_start_lsn == LSN_IMPOSSIBLE) + { + /* group not yet started */ + all_active_trans[sid].group_start_lsn= rec.lsn; + } + } + len= translog_read_next_record_header(&scanner, &rec); + if (len == (TRANSLOG_RECORD_HEADER_MAX_SIZE + 1)) + { + printf("EOF on the log\n"); + goto end; + } + } + translog_free_record_header(&rec); + + /* + So we have applied all REDOs. + We may now have unfinished transactions. + I don't think it's this program's job to roll them back: + to roll back and at the same time stay idempotent, it needs to write log + records (without CLRs, 2nd rollback would hit the effects of first + rollback and fail). But this standalone tool is not allowed to write to + the server's transaction log. So we do not roll back anything. + In the real Recovery code, or the code to do "recover after online + backup", yes we will roll back. + */ + end_of_redo_phase(); + goto end; +err: + /* don't touch anything more, in case we hit a bug */ + exit(1); +end: + maria_end(); + free_defaults(default_argv); + my_end(0); + exit(0); + return 0; /* No compiler warning */ +} + + +static struct my_option my_long_options[] = +{ + {"only-display", 'o', "display brief info about records's header", + (gptr*) &opt_only_display, (gptr*) &opt_only_display, 0, GET_BOOL, NO_ARG, + 0, 0, 0, 0, 0, 0}, + {"display-and-apply", 'a', + "like --only-display but displays more info and modifies tables", + (gptr*) &opt_display_and_apply, (gptr*) &opt_display_and_apply, 0, + GET_BOOL, NO_ARG, 0, 0, 0, 0, 0, 0}, +#ifndef DBUG_OFF + {"debug", '#', "Output debug log. Often this is 'd:t:o,filename'.", + 0, 0, 0, GET_STR, OPT_ARG, 0, 0, 0, 0, 0, 0}, +#endif + { 0, 0, 0, 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0} +}; + +#include + +static void print_version(void) +{ + VOID(printf("%s Ver 1.0 for %s on %s\n", + my_progname, SYSTEM_TYPE, MACHINE_TYPE)); + NETWARE_SET_SCREEN_MODE(1); +} + + +static void usage(void) +{ + print_version(); + puts("Copyright (C) 2007 MySQL AB"); + puts("This software comes with ABSOLUTELY NO WARRANTY. This is free software,"); + puts("and you are welcome to modify and redistribute it under the GPL license\n"); + + puts("Display and apply log records from a MARIA transaction log"); + puts("found in the current directory (for now)"); + VOID(printf("\nUsage: %s OPTIONS\n", my_progname)); + puts("You need to use one of -o or -a"); + my_print_help(my_long_options); + print_defaults("my", load_default_groups); + my_print_variables(my_long_options); +} + +#include + +static my_bool +get_one_option(int optid __attribute__((unused)), + const struct my_option *opt __attribute__((unused)), + char *argument __attribute__((unused))) +{ + /* for now there is nothing special with our options */ + return 0; +} + +static void get_options(int *argc,char ***argv) +{ + int ho_error; + + my_progname= argv[0][0]; + + if ((ho_error=handle_options(argc, argv, my_long_options, get_one_option))) + exit(ho_error); + + if ((opt_only_display + opt_display_and_apply) != 1) + { + usage(); + exit(1); + } +} + + +/* very basic info about the record's header */ +static void display_record_position(const LOG_DESC *log_desc, + const TRANSLOG_HEADER_BUFFER *rec, + uint number) +{ + /* + if number==0, we're going over records which we had already seen and which + form a group, so we indent below the group's end record + */ + printf("%sRecord #%u LSN (%lu,0x%lx) short_trid %u %s(num_type:%u)\n", + number ? "" : " ", number, + (ulong) LSN_FILE_NO(rec->lsn), (ulong) LSN_OFFSET(rec->lsn), + rec->short_trid, log_desc->name, rec->type); +} + + +static int display_and_apply_record(const LOG_DESC *log_desc, + const TRANSLOG_HEADER_BUFFER *rec) +{ + int error; + if (opt_only_display) + return 0; + if (log_desc->record_execute_in_redo_phase == NULL) + { + /* die on all not-yet-handled records :) */ + DBUG_ASSERT("one more hook" == "to write"); + } + if ((error= (*log_desc->record_execute_in_redo_phase)(rec))) + fprintf(stderr, "Got error when executing record\n"); + return error; +} + + +prototype_exec_hook(LONG_TRANSACTION_ID) +{ + uint16 sid= rec->short_trid; + TrID long_trid= all_active_trans[sid].long_trid; + /* abort group of this trn (must be of before a crash) */ + LSN gslsn= all_active_trans[sid].group_start_lsn; + if (gslsn != LSN_IMPOSSIBLE) + { + printf("Group at LSN (%lu,0x%lx) short_trid %u aborted\n", + (ulong) LSN_FILE_NO(gslsn), (ulong) LSN_OFFSET(gslsn), sid); + all_active_trans[sid].group_start_lsn= LSN_IMPOSSIBLE; + } + if (long_trid != 0) + { + LSN ulsn= all_active_trans[sid].undo_lsn; + if (ulsn != LSN_IMPOSSIBLE) + { + fprintf(stderr, "Found an old transaction long_trid %llu short_trid %u" + " with same short id as this new transaction, and has neither" + " committed nor rollback (undo_lsn: (%lu,0x%lx))\n", long_trid, + sid, (ulong) LSN_FILE_NO(ulsn), (ulong) LSN_OFFSET(ulsn)); + goto err; + } + } + long_trid= uint6korr(rec->header); + all_active_trans[sid].long_trid= long_trid; + printf("Transaction long_trid %lu short_trid %u starts\n", long_trid, sid); + goto end; +err: + DBUG_ASSERT(0); + return 1; +end: + return 0; +} + +prototype_exec_hook(CHECKPOINT) +{ + /* the only checkpoint we care about was found via control file, ignore */ + return 0; +} + + +prototype_exec_hook(REDO_CREATE_TABLE) +{ + File dfile= -1, kfile= -1; + char *linkname_ptr, filename[FN_REFLEN]; + char *name, *ptr; + myf create_flag; + uint flags; + int error, create_mode= O_RDWR | O_TRUNC; + MARIA_HA *info= NULL; + if (((name= my_malloc(rec->record_length, MYF(MY_WME))) == NULL) || + (translog_read_record(rec->lsn, 0, rec->record_length, name, NULL) != + rec->record_length)) + { + fprintf(stderr, "Failed to read record\n"); + goto err; + } + printf("Table '%s'", name); + /* we try hard to get create_rename_lsn, to avoid mistakes if possible */ + info= maria_open(name, O_RDONLY, HA_OPEN_FOR_REPAIR); + if (info) + { + if (!info->s->base.transactional) + { + /* + could be that transactional table was later dropped, and a non-trans + one was renamed to its name, thus create_rename_lsn is 0 and should + not be trusted. + */ + printf(", is not transactional\n"); + DBUG_ASSERT(0); /* I want to know this */ + goto end; + } + if (cmp_translog_addr(info->s->state.create_rename_lsn, rec->lsn) >= 0) + { + printf(", has create_rename_lsn (%lu,0x%lx) is more recent than log" + " record\n", + (ulong) LSN_FILE_NO(rec->lsn), + (ulong) LSN_OFFSET(rec->lsn)); + goto end; + } + if (maria_is_crashed(info)) + { + printf(", is crashed, overwriting it"); + DBUG_ASSERT(0); /* I want to know this */ + } + maria_close(info); + info= NULL; + } + /* if does not exist, is older, or its header is corrupted, overwrite it */ + // TODO symlinks + ptr= name + strlen(name) + 1; + if ((flags= ptr[0] ? HA_DONT_TOUCH_DATA : 0)) + printf(", we will only touch index file"); + fn_format(filename, name, "", MARIA_NAME_IEXT, + (MY_UNPACK_FILENAME | + (flags & HA_DONT_TOUCH_DATA) ? MY_RETURN_REAL_PATH : 0) | + MY_APPEND_EXT); + linkname_ptr= NULL; + create_flag= MY_DELETE_OLD; + printf(", creating as '%s'", filename); + if ((kfile= my_create_with_symlink(linkname_ptr, filename, 0, create_mode, + MYF(MY_WME|create_flag))) < 0) + { + fprintf(stderr, "Failed to create index file\n"); + goto err; + } + ptr++; + uint kfile_size_before_extension= uint2korr(ptr); + ptr+= 2; + uint keystart= uint2korr(ptr); + ptr+= 2; + /* set create_rename_lsn (for maria_read_log to be idempotent) */ + lsn_store(ptr + sizeof(info->s->state.header) + 2, rec->lsn); + if (my_pwrite(kfile, ptr, + kfile_size_before_extension, 0, MYF(MY_NABP|MY_WME)) || + my_chsize(kfile, keystart, 0, MYF(MY_WME))) + { + fprintf(stderr, "Failed to write to index file\n"); + goto err; + } + if (!(flags & HA_DONT_TOUCH_DATA)) + { + fn_format(filename,name,"", MARIA_NAME_DEXT, + MY_UNPACK_FILENAME | MY_APPEND_EXT); + linkname_ptr= NULL; + create_flag=MY_DELETE_OLD; + if ((dfile= + my_create_with_symlink(linkname_ptr, filename, 0, create_mode, + MYF(MY_WME | create_flag))) < 0) + { + fprintf(stderr, "Failed to create data file\n"); + goto err; + } + /* + we now have an empty data file. To be able to + _ma_initialize_data_file() we need some pieces of the share to be + correctly filled. So we just open the table (fortunately, an empty + data file does not preclude this). + */ + if (((info= maria_open(name, O_RDONLY, 0)) == NULL) || + _ma_initialize_data_file(dfile, info->s)) + { + fprintf(stderr, "Failed to open new table or write to data file\n"); + goto err; + } + } + error= 0; + goto end; +err: + DBUG_ASSERT(0); + error= 1; +end: + printf("\n"); + if (kfile >= 0) + error|= my_close(kfile, MYF(MY_WME)); + if (dfile >= 0) + error|= my_close(dfile, MYF(MY_WME)); + if (info != NULL) + error|= maria_close(info); + my_free(name, MYF(MY_ALLOW_ZERO_PTR)); + return 0; +} + + +prototype_exec_hook(FILE_ID) +{ + uint16 sid; + int error; + char *name, *buff; + MARIA_HA *info= NULL; + if (((buff= my_malloc(rec->record_length, MYF(MY_WME))) == NULL) || + (translog_read_record(rec->lsn, 0, rec->record_length, buff, NULL) != + rec->record_length)) + { + fprintf(stderr, "Failed to read record\n"); + goto err; + } + sid= fileid_korr(buff); + name= buff + FILEID_STORE_SIZE; + printf("Table '%s', id %u", name, sid); + info= all_tables[sid]; + if (info != NULL) + { + printf(", closing table '%s'", info->s->open_file_name); + all_tables[sid]= NULL; + info->s->base.transactional= TRUE; /* put back the truth */ + if (maria_close(info)) + { + fprintf(stderr, "Failed to close table\n"); + goto err; + } + } + info= maria_open(name, O_RDWR, HA_OPEN_FOR_REPAIR); + if (info == NULL) + { + printf(", is absent (must have been dropped later?)" + " or its header is so corrupted that we cannot open it;" + " we skip it\n"); + goto end; + } + if (maria_is_crashed(info)) + { + fprintf(stderr, "Table is crashed, can't apply log records to it\n"); + goto err; + } + if (!info->s->base.transactional) + { + printf(", is not transactional\n"); + DBUG_ASSERT(0); /* I want to know this */ + goto end; + } + all_tables[sid]= info; + /* + don't log any records for this work. TODO make sure this variable does not + go to disk before we restore it to its true value. + */ + info->s->base.transactional= FALSE; + printf(", opened\n"); + error= 0; + goto end; +err: + DBUG_ASSERT(0); + error= 1; + if (info != NULL) + error|= maria_close(info); +end: + my_free(buff, MYF(MY_ALLOW_ZERO_PTR)); + return 0; +} + + +prototype_exec_hook(REDO_INSERT_ROW_HEAD) +{ + uint16 sid; + ulonglong page; + MARIA_HA *info; + sid= fileid_korr(rec->header); + page= page_korr(rec->header + FILEID_STORE_SIZE); + printf("For page %llu of table of short id %u", page, sid); + info= all_tables[sid]; + if (info == NULL) + { + printf(", table skipped, so skipping record\n"); + goto end; + } + printf(", '%s'", info->s->open_file_name); + if (cmp_translog_addr(info->s->state.create_rename_lsn, rec->lsn) >= 0) + { + printf(", has create_rename_lsn (%lu,0x%lx) is more recent than log" + " record\n", + (ulong) LSN_FILE_NO(rec->lsn), (ulong) LSN_OFFSET(rec->lsn)); + goto end; + } + /* + Soon we will also skip the page depending on the rec_lsn for this page in + the checkpoint record, but this is not absolutely needed for now (just + assume we have made no checkpoint). + */ + printf(", applying record\n"); + DBUG_ASSERT("Monty" == "this is the place"); +end: + /* as we don't have apply working: */ + return 1; +} + + +prototype_exec_hook(COMMIT) +{ + uint16 sid= rec->short_trid; + TrID long_trid= all_active_trans[sid].long_trid; + LSN gslsn= all_active_trans[sid].group_start_lsn; + + if (long_trid == 0) + { + printf("We don't know about transaction short_trid %u;" + "it probably committed long ago, forget it\n", sid); + return 0; + } + printf("Transaction long_trid %lu short_trid %u committed", long_trid, sid); + if (gslsn != LSN_IMPOSSIBLE) + { + /* + It's not an error, it may be that trn got a disk error when writing to a + table, so an unfinished group staid in the log. + */ + printf(", with group at LSN (%lu,0x%lx) short_trid %u aborted\n", + (ulong) LSN_FILE_NO(gslsn), (ulong) LSN_OFFSET(gslsn), sid); + all_active_trans[sid].group_start_lsn= LSN_IMPOSSIBLE; + } + else + printf("\n"); + all_active_trans[sid].long_trid= 0; +#ifdef MARIA_VERSIONING + /* + if real recovery: + transaction was committed, move it to some separate list for later + purging (but don't purge now! purging may have been started before, we + may find REDO_PURGE records soon). + */ +#endif + return 0; +} + + +/* Just to inform about any aborted groups or unfinished transactions */ +static void end_of_redo_phase() +{ + uint sid; + for (sid= 0; sid <= SHORT_TRID_MAX; sid++) + { + TrID long_trid= all_active_trans[sid].long_trid; + LSN gslsn= all_active_trans[sid].group_start_lsn; + if (long_trid == 0) + continue; + if (all_active_trans[sid].undo_lsn != LSN_IMPOSSIBLE) + printf("Transaction long_trid %lu short_trid %u unfinished\n", + long_trid, sid); + if (gslsn != LSN_IMPOSSIBLE) + { + printf("Group at LSN (%lu,0x%lx) short_trid %u aborted\n", + (ulong) LSN_FILE_NO(gslsn), (ulong) LSN_OFFSET(gslsn), sid); + } + /* If real recovery: roll back unfinished transaction */ +#ifdef MARIA_VERSIONING + /* + If real recovery: transaction was committed, move it to some separate + list for soon purging. + */ +#endif + } +} diff --git a/storage/maria/trnman_public.h b/storage/maria/trnman_public.h index 3e0a21c26a6..e1891466c4d 100644 --- a/storage/maria/trnman_public.h +++ b/storage/maria/trnman_public.h @@ -20,6 +20,9 @@ to include my_atomic.h in C++ code. */ +#ifndef _trnman_public_h +#define _trnman_public_h + #include "ma_loghandler_lsn.h" C_MODE_START @@ -52,3 +55,4 @@ my_bool trnman_has_locked_tables(TRN *trn); void trnman_reset_locked_tables(TRN *trn); C_MODE_END +#endif diff --git a/storage/maria/unittest/ma_control_file-t.c b/storage/maria/unittest/ma_control_file-t.c index 71a1157f1ba..a7472361dad 100644 --- a/storage/maria/unittest/ma_control_file-t.c +++ b/storage/maria/unittest/ma_control_file-t.c @@ -121,8 +121,8 @@ static int delete_file(myf my_flags) The error will however be printed on stderr. */ my_delete(file_name, my_flags); - expect_checkpoint_lsn= CONTROL_FILE_IMPOSSIBLE_LSN; - expect_logno= CONTROL_FILE_IMPOSSIBLE_FILENO; + expect_checkpoint_lsn= LSN_IMPOSSIBLE; + expect_logno= FILENO_IMPOSSIBLE; return 0; } @@ -146,9 +146,9 @@ static int verify_module_values_match_expected() */ static int verify_module_values_are_impossible() { - RET_ERR_UNLESS(last_logno == CONTROL_FILE_IMPOSSIBLE_FILENO); + RET_ERR_UNLESS(last_logno == FILENO_IMPOSSIBLE); RET_ERR_UNLESS(last_checkpoint_lsn == - CONTROL_FILE_IMPOSSIBLE_LSN); + LSN_IMPOSSIBLE); return 0; } @@ -164,7 +164,7 @@ static int close_file() static int create_or_open_file() { - RET_ERR_UNLESS(ma_control_file_create_or_open() == CONTROL_FILE_OK); + RET_ERR_UNLESS(ma_control_file_create_or_open(TRUE) == CONTROL_FILE_OK); /* Check that the module reports expected information */ RET_ERR_UNLESS(verify_module_values_match_expected() == 0); return 0; @@ -188,7 +188,7 @@ static int test_one_log() RET_ERR_UNLESS(create_or_open_file() == CONTROL_FILE_OK); objs_to_write= CONTROL_FILE_UPDATE_ONLY_LOGNO; expect_logno= 123; - RET_ERR_UNLESS(write_file(CONTROL_FILE_IMPOSSIBLE_LSN, + RET_ERR_UNLESS(write_file(LSN_IMPOSSIBLE, expect_logno, objs_to_write) == 0); RET_ERR_UNLESS(close_file() == 0); @@ -206,7 +206,7 @@ static int test_five_logs() for (i= 0; i<5; i++) { expect_logno*= 3; - RET_ERR_UNLESS(write_file(CONTROL_FILE_IMPOSSIBLE_LSN, expect_logno, + RET_ERR_UNLESS(write_file(LSN_IMPOSSIBLE, expect_logno, objs_to_write) == 0); } RET_ERR_UNLESS(close_file() == 0); @@ -320,7 +320,7 @@ static int test_bad_magic_string() RET_ERR_UNLESS(my_pwrite(fd, "papa", 4, 0, MYF(MY_FNABP | MY_WME)) == 0); /* Check that control file module sees the problem */ - RET_ERR_UNLESS(ma_control_file_create_or_open() == + RET_ERR_UNLESS(ma_control_file_create_or_open(TRUE) == CONTROL_FILE_BAD_MAGIC_STRING); /* Restore magic string */ RET_ERR_UNLESS(my_pwrite(fd, buffer, 4, 0, MYF(MY_FNABP | MY_WME)) == 0); @@ -346,7 +346,7 @@ static int test_bad_checksum() buffer[0]+= 3; /* mangle checksum */ RET_ERR_UNLESS(my_pwrite(fd, buffer, 1, 8, MYF(MY_FNABP | MY_WME)) == 0); /* Check that control file module sees the problem */ - RET_ERR_UNLESS(ma_control_file_create_or_open() == + RET_ERR_UNLESS(ma_control_file_create_or_open(TRUE) == CONTROL_FILE_BAD_CHECKSUM); /* Restore checksum */ buffer[0]-= 3; @@ -369,10 +369,11 @@ static int test_bad_size() MYF(MY_WME))) >= 0); RET_ERR_UNLESS(my_write(fd, buffer, 10, MYF(MY_FNABP | MY_WME)) == 0); /* Check that control file module sees the problem */ - RET_ERR_UNLESS(ma_control_file_create_or_open() == CONTROL_FILE_TOO_SMALL); + RET_ERR_UNLESS(ma_control_file_create_or_open(TRUE) == + CONTROL_FILE_TOO_SMALL); RET_ERR_UNLESS(my_write(fd, buffer, 30, MYF(MY_FNABP | MY_WME)) == 0); /* Check that control file module sees the problem */ - RET_ERR_UNLESS(ma_control_file_create_or_open() == CONTROL_FILE_TOO_BIG); + RET_ERR_UNLESS(ma_control_file_create_or_open(TRUE) == CONTROL_FILE_TOO_BIG); RET_ERR_UNLESS(my_close(fd, MYF(MY_WME)) == 0); /* Leave a correct control file */ diff --git a/storage/maria/unittest/ma_test_loghandler-t.c b/storage/maria/unittest/ma_test_loghandler-t.c index e31136d52ec..19e6704dc5a 100644 --- a/storage/maria/unittest/ma_test_loghandler-t.c +++ b/storage/maria/unittest/ma_test_loghandler-t.c @@ -164,7 +164,7 @@ int main(int argc __attribute__((unused)), char *argv[]) } #endif - if (ma_control_file_create_or_open()) + if (ma_control_file_create_or_open(TRUE)) { fprintf(stderr, "Can't init control file (%d)\n", errno); exit(1); @@ -336,7 +336,7 @@ int main(int argc __attribute__((unused)), char *argv[]) ma_control_file_end(); - if (ma_control_file_create_or_open()) + if (ma_control_file_create_or_open(TRUE)) { fprintf(stderr, "pass2: Can't init control file (%d)\n", errno); exit(1); @@ -398,7 +398,7 @@ int main(int argc __attribute__((unused)), char *argv[]) i, errno); goto err; } - if (rec.lsn == CONTROL_FILE_IMPOSSIBLE_LSN) + if (rec.lsn == LSN_IMPOSSIBLE) { if (i != ITERATIONS) { @@ -477,7 +477,7 @@ int main(int argc __attribute__((unused)), char *argv[]) "failed (%d)\n", i, errno); goto err; } - if (rec.lsn == CONTROL_FILE_IMPOSSIBLE_LSN) + if (rec.lsn == LSN_IMPOSSIBLE) { fprintf(stderr, "EOL met at the middle of iteration (first var) %u " "instead of beginning of %u\n", i, ITERATIONS); @@ -572,7 +572,7 @@ int main(int argc __attribute__((unused)), char *argv[]) i, errno); goto err; } - if (rec.lsn == CONTROL_FILE_IMPOSSIBLE_LSN) + if (rec.lsn == LSN_IMPOSSIBLE) { fprintf(stderr, "EOL met at the middle of iteration %u " "instead of beginning of %u\n", i, ITERATIONS); diff --git a/storage/maria/unittest/ma_test_loghandler_multigroup-t.c b/storage/maria/unittest/ma_test_loghandler_multigroup-t.c index 1281ee425d8..5fe24be597d 100644 --- a/storage/maria/unittest/ma_test_loghandler_multigroup-t.c +++ b/storage/maria/unittest/ma_test_loghandler_multigroup-t.c @@ -161,7 +161,7 @@ int main(int argc __attribute__((unused)), char *argv[]) } #endif - if (ma_control_file_create_or_open()) + if (ma_control_file_create_or_open(TRUE)) { fprintf(stderr, "Can't init control file (%d)\n", errno); exit(1); @@ -325,7 +325,7 @@ int main(int argc __attribute__((unused)), char *argv[]) end_pagecache(&pagecache, 1); ma_control_file_end(); - if (ma_control_file_create_or_open()) + if (ma_control_file_create_or_open(TRUE)) { fprintf(stderr, "pass2: Can't init control file (%d)\n", errno); exit(1); @@ -390,7 +390,7 @@ int main(int argc __attribute__((unused)), char *argv[]) translog_free_record_header(&rec); goto err; } - if (rec.lsn == CONTROL_FILE_IMPOSSIBLE_LSN) + if (rec.lsn == LSN_IMPOSSIBLE) { if (i != ITERATIONS) { @@ -470,7 +470,7 @@ int main(int argc __attribute__((unused)), char *argv[]) "failed (%d)\n", i, errno); goto err; } - if (rec.lsn == CONTROL_FILE_IMPOSSIBLE_LSN) + if (rec.lsn == LSN_IMPOSSIBLE) { fprintf(stderr, "EOL met at the middle of iteration (first var) %u " "instead of beginning of %u\n", i, ITERATIONS); @@ -568,7 +568,7 @@ int main(int argc __attribute__((unused)), char *argv[]) translog_free_record_header(&rec); goto err; } - if (rec.lsn == CONTROL_FILE_IMPOSSIBLE_LSN) + if (rec.lsn == LSN_IMPOSSIBLE) { fprintf(stderr, "EOL met at the middle of iteration %u " "instead of beginning of %u\n", i, ITERATIONS); diff --git a/storage/maria/unittest/ma_test_loghandler_multithread-t.c b/storage/maria/unittest/ma_test_loghandler_multithread-t.c index ff966160acc..ba5d217a45a 100644 --- a/storage/maria/unittest/ma_test_loghandler_multithread-t.c +++ b/storage/maria/unittest/ma_test_loghandler_multithread-t.c @@ -270,7 +270,7 @@ int main(int argc __attribute__((unused)), my_thread_global_init(); - if (ma_control_file_create_or_open()) + if (ma_control_file_create_or_open(TRUE)) { fprintf(stderr, "Can't init control file (%d)\n", errno); exit(1); @@ -384,7 +384,7 @@ int main(int argc __attribute__((unused)), translog_free_record_header(&rec); goto err; } - if (rec.lsn == CONTROL_FILE_IMPOSSIBLE_LSN) + if (rec.lsn == LSN_IMPOSSIBLE) { if (i != WRITERS * ITERATIONS * 2) { diff --git a/storage/maria/unittest/ma_test_loghandler_pagecache-t.c b/storage/maria/unittest/ma_test_loghandler_pagecache-t.c index 35e05f9c997..4ac500ce8b2 100644 --- a/storage/maria/unittest/ma_test_loghandler_pagecache-t.c +++ b/storage/maria/unittest/ma_test_loghandler_pagecache-t.c @@ -56,7 +56,7 @@ int main(int argc __attribute__((unused)), char *argv[]) } #endif - if (ma_control_file_create_or_open()) + if (ma_control_file_create_or_open(TRUE)) { fprintf(stderr, "Can't init control file (%d)\n", errno); exit(1); diff --git a/storage/myisam/mi_close.c b/storage/myisam/mi_close.c index 47b7ba855c0..270a5dff056 100644 --- a/storage/myisam/mi_close.c +++ b/storage/myisam/mi_close.c @@ -75,6 +75,7 @@ int mi_close(register MI_INFO *info) not change the crashed state. We can NOT write the state in other cases as other threads may be using the file at this point + IF using --external-locking. */ if (share->mode != O_RDONLY && mi_is_crashed(info)) mi_state_info_write(share->kfile, &share->state, 1); -- cgit v1.2.1 From 79672e8c44804fd9b877f51c6e9e054d90ed3d04 Mon Sep 17 00:00:00 2001 From: unknown Date: Tue, 26 Jun 2007 18:29:17 +0200 Subject: WL#3072 - Maria recovery: safety assertions. storage/maria/maria_read_log.c: assertions to protect against future bugs (especially, to ensure that replaying DROP TABLE, if implemented, wouldn't leave open tables behind it) --- storage/maria/maria_read_log.c | 5 +++-- 1 file changed, 3 insertions(+), 2 deletions(-) diff --git a/storage/maria/maria_read_log.c b/storage/maria/maria_read_log.c index 7bb15e27f7a..e654b8ea2ac 100644 --- a/storage/maria/maria_read_log.c +++ b/storage/maria/maria_read_log.c @@ -424,6 +424,7 @@ prototype_exec_hook(REDO_CREATE_TABLE) info= maria_open(name, O_RDONLY, HA_OPEN_FOR_REPAIR); if (info) { + DBUG_ASSERT(info->s->reopen == 1); /* check that we're not using it */ if (!info->s->base.transactional) { /* @@ -437,8 +438,7 @@ prototype_exec_hook(REDO_CREATE_TABLE) } if (cmp_translog_addr(info->s->state.create_rename_lsn, rec->lsn) >= 0) { - printf(", has create_rename_lsn (%lu,0x%lx) is more recent than log" - " record\n", + printf(", has create_rename_lsn (%lu,0x%lx) is more recent than record", (ulong) LSN_FILE_NO(rec->lsn), (ulong) LSN_OFFSET(rec->lsn)); goto end; @@ -568,6 +568,7 @@ prototype_exec_hook(FILE_ID) fprintf(stderr, "Table is crashed, can't apply log records to it\n"); goto err; } + DBUG_ASSERT(info->s->reopen == 1); /* should always be only one instance */ if (!info->s->base.transactional) { printf(", is not transactional\n"); -- cgit v1.2.1 From 1e73169a82f86fa2fdaf43e7601705eb9a81cb85 Mon Sep 17 00:00:00 2001 From: unknown Date: Tue, 26 Jun 2007 22:30:09 +0200 Subject: WL#3072 - Maria recovery fixes for build failures; copyrights; small bugfixes and comments mysys/Makefile.am: missing .h breaks building from tarball storage/maria/ma_loghandler.c: applying Serg's bugfix of trnman_new_trid() to translog_assign_id_to_share() storage/maria/ma_loghandler.h: copyright storage/maria/ma_loghandler_lsn.h: copyright storage/maria/maria_read_log.c: fix for compiler warnings. Comments. Close tables when program ends. --- mysys/Makefile.am | 2 +- storage/maria/ma_loghandler.c | 35 +++++++++++---------- storage/maria/ma_loghandler.h | 18 ++++++++++- storage/maria/ma_loghandler_lsn.h | 15 +++++++++ storage/maria/maria_read_log.c | 66 ++++++++++++++++++++++++++++++++------- 5 files changed, 106 insertions(+), 30 deletions(-) diff --git a/mysys/Makefile.am b/mysys/Makefile.am index cd84e09a60e..60aa59fd3eb 100644 --- a/mysys/Makefile.am +++ b/mysys/Makefile.am @@ -20,7 +20,7 @@ INCLUDES = @ZLIB_INCLUDES@ -I$(top_builddir)/include \ -I$(top_srcdir)/include -I$(srcdir) pkglib_LIBRARIES = libmysys.a LDADD = libmysys.a $(top_builddir)/strings/libmystrings.a $(top_builddir)/dbug/libdbug.a -noinst_HEADERS = mysys_priv.h my_static.h +noinst_HEADERS = mysys_priv.h my_static.h my_safehash.h libmysys_a_SOURCES = my_init.c my_getwd.c mf_getdate.c my_mmap.c \ mf_path.c mf_loadpath.c my_file.c \ my_open.c my_create.c my_dup.c my_seek.c my_read.c \ diff --git a/storage/maria/ma_loghandler.c b/storage/maria/ma_loghandler.c index 6f238ef4055..79bf44046b1 100644 --- a/storage/maria/ma_loghandler.c +++ b/storage/maria/ma_loghandler.c @@ -177,7 +177,6 @@ my_bool translog_inited= 0; #include /* an array that maps id of a MARIA_SHARE to this MARIA_SHARE */ static MARIA_SHARE **id_to_share= NULL; -#define SHARE_ID_MAX 65535 /* array's size */ /* lock for id_to_share */ static my_atomic_rwlock_t LOCK_id_to_share; @@ -2282,8 +2281,8 @@ my_bool translog_init(const char *directory, structures for generating 2-byte ids: */ my_atomic_rwlock_init(&LOCK_id_to_share); - id_to_share= (MARIA_SHARE **) my_malloc(SHARE_ID_MAX*sizeof(MARIA_SHARE*), - MYF(MY_WME|MY_ZEROFILL)); + id_to_share= (MARIA_SHARE **) my_malloc(SHARE_ID_MAX * sizeof(MARIA_SHARE*), + MYF(MY_WME | MY_ZEROFILL)); if (unlikely(!id_to_share)) DBUG_RETURN(1); id_to_share--; /* min id is 1 */ @@ -5682,21 +5681,23 @@ int translog_assign_id_to_share(MARIA_SHARE *share, TRN *trn) if (likely(share->id == 0)) { /* Inspired by set_short_trid() of trnman.c */ - int i= share->kfile.file % SHARE_ID_MAX + 1; - my_atomic_rwlock_wrlock(&LOCK_id_to_share); - /** - @todo RECOVERY BUG: if all slots are used, and we're using rwlocks - above, we will never exit the loop. To be discussed with Serg. - */ - for ( ; ; i= i % SHARE_ID_MAX + 1) /* the range is [1..SHARE_ID_MAX] */ + uint i= share->kfile.file % SHARE_ID_MAX + 1; + do { - void *tmp= NULL; - if (id_to_share[i] == NULL && - my_atomic_casptr((void **)&id_to_share[i], &tmp, share)) - break; - } - my_atomic_rwlock_wrunlock(&LOCK_id_to_share); - share->id= (uint16)i; + my_atomic_rwlock_wrlock(&LOCK_id_to_share); + for ( ; i <= SHARE_ID_MAX ; i++) /* the range is [1..SHARE_ID_MAX] */ + { + void *tmp= NULL; + if (id_to_share[i] == NULL && + my_atomic_casptr((void **)&id_to_share[i], &tmp, share)) + { + share->id= (uint16)i; + break; + } + } + my_atomic_rwlock_wrunlock(&LOCK_id_to_share); + i= 1; /* scan the whole array */ + } while (share->id == 0); DBUG_PRINT("info", ("id_to_share: 0x%lx -> %u", (ulong)share, i)); LSN lsn; LEX_STRING log_array[TRANSLOG_INTERNAL_PARTS + 2]; diff --git a/storage/maria/ma_loghandler.h b/storage/maria/ma_loghandler.h index 22b8cca3a08..f2bfd2c9d7e 100644 --- a/storage/maria/ma_loghandler.h +++ b/storage/maria/ma_loghandler.h @@ -1,4 +1,17 @@ -// TODO copyright +/* Copyright (C) 2007 MySQL AB & Sanja Belkin + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ #ifndef _ma_loghandler_h #define _ma_loghandler_h @@ -251,6 +264,9 @@ extern my_bool translog_inited; all the rest added because of recovery; should we make ma_loghandler_for_recovery.h ? */ + +#define SHARE_ID_MAX 65535 /* array's size */ + extern LSN first_lsn_in_log(); /* record parts descriptor */ diff --git a/storage/maria/ma_loghandler_lsn.h b/storage/maria/ma_loghandler_lsn.h index af7594e3b00..34cb7616b74 100644 --- a/storage/maria/ma_loghandler_lsn.h +++ b/storage/maria/ma_loghandler_lsn.h @@ -1,3 +1,18 @@ +/* Copyright (C) 2007 MySQL AB & Sanja Belkin + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ + #ifndef _ma_loghandler_lsn_h #define _ma_loghandler_lsn_h diff --git a/storage/maria/maria_read_log.c b/storage/maria/maria_read_log.c index e654b8ea2ac..c8263495fbc 100644 --- a/storage/maria/maria_read_log.c +++ b/storage/maria/maria_read_log.c @@ -48,7 +48,9 @@ static int display_and_apply_record(const LOG_DESC *log_desc, #define prototype_exec_hook(R) \ static int exec_LOGREC_ ## R(const TRANSLOG_HEADER_BUFFER *rec) prototype_exec_hook(LONG_TRANSACTION_ID); +#ifdef MARIA_CHECKPOINT prototype_exec_hook(CHECKPOINT); +#endif prototype_exec_hook(REDO_CREATE_TABLE); prototype_exec_hook(FILE_ID); prototype_exec_hook(REDO_INSERT_ROW_HEAD); @@ -128,7 +130,9 @@ int main(int argc, char **argv) log_record_type_descriptor[LOGREC_ ## R].record_execute_in_redo_phase= \ exec_LOGREC_ ## R; install_exec_hook(LONG_TRANSACTION_ID); +#ifdef MARIA_CHECKPOINT install_exec_hook(CHECKPOINT); +#endif install_exec_hook(REDO_CREATE_TABLE); install_exec_hook(FILE_ID); install_exec_hook(REDO_INSERT_ROW_HEAD); @@ -337,10 +341,11 @@ static void display_record_position(const LOG_DESC *log_desc, if number==0, we're going over records which we had already seen and which form a group, so we indent below the group's end record */ - printf("%sRecord #%u LSN (%lu,0x%lx) short_trid %u %s(num_type:%u)\n", + printf("%sRec#%u LSN (%lu,0x%lx) short_trid %u %s(num_type:%u) len %lu\n", number ? "" : " ", number, (ulong) LSN_FILE_NO(rec->lsn), (ulong) LSN_OFFSET(rec->lsn), - rec->short_trid, log_desc->name, rec->type); + rec->short_trid, log_desc->name, rec->type, + (ulong)rec->record_length); } @@ -367,6 +372,7 @@ prototype_exec_hook(LONG_TRANSACTION_ID) TrID long_trid= all_active_trans[sid].long_trid; /* abort group of this trn (must be of before a crash) */ LSN gslsn= all_active_trans[sid].group_start_lsn; + char llbuf[22]; if (gslsn != LSN_IMPOSSIBLE) { printf("Group at LSN (%lu,0x%lx) short_trid %u aborted\n", @@ -378,16 +384,18 @@ prototype_exec_hook(LONG_TRANSACTION_ID) LSN ulsn= all_active_trans[sid].undo_lsn; if (ulsn != LSN_IMPOSSIBLE) { - fprintf(stderr, "Found an old transaction long_trid %llu short_trid %u" + llstr(long_trid, llbuf); + fprintf(stderr, "Found an old transaction long_trid %s short_trid %u" " with same short id as this new transaction, and has neither" - " committed nor rollback (undo_lsn: (%lu,0x%lx))\n", long_trid, + " committed nor rollback (undo_lsn: (%lu,0x%lx))\n", llbuf, sid, (ulong) LSN_FILE_NO(ulsn), (ulong) LSN_OFFSET(ulsn)); goto err; } } long_trid= uint6korr(rec->header); all_active_trans[sid].long_trid= long_trid; - printf("Transaction long_trid %lu short_trid %u starts\n", long_trid, sid); + llstr(long_trid, llbuf); + printf("Transaction long_trid %s short_trid %u starts\n", llbuf, sid); goto end; err: DBUG_ASSERT(0); @@ -396,11 +404,14 @@ end: return 0; } + +#ifdef MARIA_CHECKPOINT prototype_exec_hook(CHECKPOINT) { /* the only checkpoint we care about was found via control file, ignore */ return 0; } +#endif prototype_exec_hook(REDO_CREATE_TABLE) @@ -600,9 +611,11 @@ prototype_exec_hook(REDO_INSERT_ROW_HEAD) uint16 sid; ulonglong page; MARIA_HA *info; + char llbuf[22]; sid= fileid_korr(rec->header); page= page_korr(rec->header + FILEID_STORE_SIZE); - printf("For page %llu of table of short id %u", page, sid); + llstr(page, llbuf); + printf("For page %s of table of short id %u", llbuf, sid); info= all_tables[sid]; if (info == NULL) { @@ -623,6 +636,16 @@ prototype_exec_hook(REDO_INSERT_ROW_HEAD) assume we have made no checkpoint). */ printf(", applying record\n"); + /* + If REDO's LSN is > page's LSN (read from disk), we are going to modify the + page and change its LSN. The normal runtime code stores the UNDO's LSN + into the page; but here storing the REDO's LSN (rec->lsn) is more + straightforward and should not cause any problem (we are not writing to + the log here, so don't have to "flush up to UNDO's LSN"). + If the UNDO's LSN is desired, it can be found, as we saw the UNDO record + before deciding to execute this REDO; UNDO's LSN could simply be stored in + all_trans[rec->short_trid].group_end_lsn for this. + */ DBUG_ASSERT("Monty" == "this is the place"); end: /* as we don't have apply working: */ @@ -635,14 +658,15 @@ prototype_exec_hook(COMMIT) uint16 sid= rec->short_trid; TrID long_trid= all_active_trans[sid].long_trid; LSN gslsn= all_active_trans[sid].group_start_lsn; - + char llbuf[22]; if (long_trid == 0) { printf("We don't know about transaction short_trid %u;" "it probably committed long ago, forget it\n", sid); return 0; } - printf("Transaction long_trid %lu short_trid %u committed", long_trid, sid); + llstr(long_trid, llbuf); + printf("Transaction long_trid %s short_trid %u committed", llbuf, sid); if (gslsn != LSN_IMPOSSIBLE) { /* @@ -671,7 +695,7 @@ prototype_exec_hook(COMMIT) /* Just to inform about any aborted groups or unfinished transactions */ static void end_of_redo_phase() { - uint sid; + uint sid, unfinished= 0; for (sid= 0; sid <= SHORT_TRID_MAX; sid++) { TrID long_trid= all_active_trans[sid].long_trid; @@ -679,8 +703,12 @@ static void end_of_redo_phase() if (long_trid == 0) continue; if (all_active_trans[sid].undo_lsn != LSN_IMPOSSIBLE) - printf("Transaction long_trid %lu short_trid %u unfinished\n", - long_trid, sid); + { + char llbuf[22]; + llstr(long_trid, llbuf); + printf("Transaction long_trid %s short_trid %u unfinished\n", + llbuf, sid); + } if (gslsn != LSN_IMPOSSIBLE) { printf("Group at LSN (%lu,0x%lx) short_trid %u aborted\n", @@ -694,4 +722,20 @@ static void end_of_redo_phase() */ #endif } + /* + We don't close tables if there are some unfinished transactions, because + closing tables normally requires that all unfinished transactions on them + be rolled back. + For example, closing will soon write the state to disk and when doing that + it will think this is a committed state, but it may not be. + */ + if (unfinished == 0) + { + for (sid= 0; sid <= SHORT_TRID_MAX; sid++) + { + MARIA_HA *info= all_tables[sid]; + if (info != NULL) + maria_close(info); + } + } } -- cgit v1.2.1 From 10bce560f6e5d2558699ce389ab534a44c94b5fc Mon Sep 17 00:00:00 2001 From: unknown Date: Tue, 26 Jun 2007 22:53:35 +0200 Subject: WL#3072 - Maria recovery comments; remember the UNDO's LSN for storing it in pages when executing REDO's (to imitate what the runtime code does) storage/maria/maria_read_log.c: comments; remember the UNDO's LSN for storing it in pages when executing REDO's (to imitate what the runtime code does) --- storage/maria/maria_read_log.c | 19 +++++++++++++------ 1 file changed, 13 insertions(+), 6 deletions(-) diff --git a/storage/maria/maria_read_log.c b/storage/maria/maria_read_log.c index c8263495fbc..568814f6f8a 100644 --- a/storage/maria/maria_read_log.c +++ b/storage/maria/maria_read_log.c @@ -38,6 +38,7 @@ struct TRN_FOR_RECOVERY struct TRN_FOR_RECOVERY all_active_trans[SHORT_TRID_MAX + 1]; MARIA_HA *all_tables[SHORT_TRID_MAX + 1]; +LSN current_group_end_lsn= LSN_IMPOSSIBLE; static void end_of_redo_phase(); static void display_record_position(const LOG_DESC *log_desc, @@ -171,6 +172,10 @@ int main(int argc, char **argv) (e.g. a set of REDOs for an operation, terminated by an UNDO for this operation); if there is no "end mark" record the group is incomplete and won't be executed. + There are pitfalls: if a table write failed, the transaction may have + put an incomplete group in the log and then a COMMIT record, that will + make a complete group which is wrong. We say that we should mark the + table corrupted if such error happens (what if it cannot be marked?). */ if (log_desc->record_ends_group) { @@ -195,6 +200,7 @@ int main(int argc, char **argv) fprintf(stderr, "Scanner2 init failed\n"); goto err; } + current_group_end_lsn= rec.lsn; do { if (rec2.short_trid == sid) /* it's in our group */ @@ -215,6 +221,7 @@ int main(int argc, char **argv) translog_free_record_header(&rec2); /* group finished */ all_active_trans[sid].group_start_lsn= LSN_IMPOSSIBLE; + current_group_end_lsn= LSN_IMPOSSIBLE; /* for debugging */ } if (display_and_apply_record(log_desc, &rec)) goto err; @@ -639,12 +646,12 @@ prototype_exec_hook(REDO_INSERT_ROW_HEAD) /* If REDO's LSN is > page's LSN (read from disk), we are going to modify the page and change its LSN. The normal runtime code stores the UNDO's LSN - into the page; but here storing the REDO's LSN (rec->lsn) is more - straightforward and should not cause any problem (we are not writing to - the log here, so don't have to "flush up to UNDO's LSN"). - If the UNDO's LSN is desired, it can be found, as we saw the UNDO record - before deciding to execute this REDO; UNDO's LSN could simply be stored in - all_trans[rec->short_trid].group_end_lsn for this. + into the page. Here storing the REDO's LSN (rec->lsn) would work + (we are not writing to the log here, so don't have to "flush up to UNDO's + LSN"). But in a test scenario where we do updates at runtime, then remove + tables, apply the log and check that this results in the same table as at + runtime, putting the same LSN as runtime had done will decrease + differences. So we use the UNDO's LSN which is current_group_end_lsn. */ DBUG_ASSERT("Monty" == "this is the place"); end: -- cgit v1.2.1 From 0cf96a32061979a961b1e137141a53ff4a9513bb Mon Sep 17 00:00:00 2001 From: unknown Date: Wed, 27 Jun 2007 12:58:08 +0200 Subject: WL#3072 - Maria recovery For this scenario: server crashes (could be because a table is corrupted) and Recovery repeatedly crashes on this table. User repairs it with maria_chk (as REPAIR TABLE is not possible), restarts the server, Recovery runs: for Recovery to not apply old REDOs to this repaired table (which would fail: rows have moved), maria_chk sets create_rename_lsn to the max value. Later when the server opens the table via ha_maria, it sets the LSN to the correct current value. storage/maria/ma_check.c: using helper function storage/maria/ma_create.c: A new helper function which stores the create_rename_lsn into the table's header on disk when we cannot wait for this to happen naturally at a later _ma_state_info_write(). storage/maria/ma_delete_all.c: using helper function; so log_data now can be FILEID_STORE_SIZE. storage/maria/ma_open.c: When opening a transactional table in the server, we discover if it has been repaired with maria_chk and if yes, give it a correct create_rename_lsn. storage/maria/ma_rename.c: using helper function storage/maria/maria_chk.c: By setting create_rename_lsn to the maximum possible LSN, maria_chk ensures that old REDOs are not applied to the new table it is going to produce. storage/maria/maria_def.h: new helper function --- storage/maria/ma_check.c | 7 ++----- storage/maria/ma_create.c | 34 +++++++++++++++++++++++++++++++--- storage/maria/ma_delete_all.c | 8 +++----- storage/maria/ma_open.c | 12 ++++++++++++ storage/maria/ma_rename.c | 9 +++------ storage/maria/maria_chk.c | 7 +++++++ storage/maria/maria_def.h | 5 +++-- 7 files changed, 61 insertions(+), 21 deletions(-) diff --git a/storage/maria/ma_check.c b/storage/maria/ma_check.c index 0fc2b77304d..72054ffe92a 100644 --- a/storage/maria/ma_check.c +++ b/storage/maria/ma_check.c @@ -5200,12 +5200,9 @@ int _ma_repair_write_log_record(const HA_CHECK *param, MARIA_HA *info) and to not apply old REDOs to the new table. The table's existence was made durable earlier (MY_SYNC_DIR passed to maria_change_to_newfile()). */ - lsn_store(log_data, share->state.create_rename_lsn); DBUG_ASSERT(info->dfile.file >= 0); - DBUG_ASSERT(share->kfile.file >= 0); - return (my_pwrite(share->kfile.file, log_data, sizeof(log_data), - sizeof(share->state.header) + 2, MYF(MY_NABP)) || - _ma_sync_table_files(info)); + return _ma_update_create_rename_lsn_on_disk(share, FALSE) || + _ma_sync_table_files(info); } return 0; } diff --git a/storage/maria/ma_create.c b/storage/maria/ma_create.c index 22b490c907c..b439d7760e7 100644 --- a/storage/maria/ma_create.c +++ b/storage/maria/ma_create.c @@ -1009,9 +1009,8 @@ int maria_create(const char *name, enum data_file_type datafile_type, If such direct my_pwrite() to a fixed offset is too "hackish", I can call ma_state_info_write() again but it will be less efficient. */ - lsn_store(log_data, share.state.create_rename_lsn); - if (my_pwrite(file, log_data, LSN_STORE_SIZE, - sizeof(share.state.header) + 2, MYF(MY_NABP))) + share.kfile.file= file; + if (_ma_update_create_rename_lsn_on_disk(&share, FALSE)) goto err_no_lock; my_free(log_data, MYF(0)); } @@ -1163,3 +1162,32 @@ int _ma_initialize_data_file(File dfile, MARIA_SHARE *share) } return 0; } + + +/** + @brief Writes create_rename_lsn to disk, optionally forces + + This is for special cases where: + - we don't want to write the full state to disk (so, not call + _ma_state_info_write()) because some parts of the state may be + currently inconsistent, or because it would be overkill + - we must sync this LSN immediately for correctness. + + @param share table's share + @param do_sync if the write should be forced to disk + + @return Operation status + @retval 0 ok + @retval 1 error (disk problem) +*/ + +int _ma_update_create_rename_lsn_on_disk(MARIA_SHARE *share, my_bool do_sync) +{ + char buf[LSN_STORE_SIZE]; + File file= share->kfile.file; + DBUG_ASSERT(file >= 0); + lsn_store(buf, share->state.create_rename_lsn); + return (my_pwrite(file, buf, sizeof(buf), + sizeof(share->state.header) + 2, MYF(MY_NABP)) || + (do_sync && my_sync(file, MYF(0)))); +} diff --git a/storage/maria/ma_delete_all.c b/storage/maria/ma_delete_all.c index 7286f540aa1..a08e259d09b 100644 --- a/storage/maria/ma_delete_all.c +++ b/storage/maria/ma_delete_all.c @@ -89,9 +89,9 @@ int maria_delete_all_rows(MARIA_HA *info) { /* For now this record is only informative */ LEX_STRING log_array[TRANSLOG_INTERNAL_PARTS + 1]; - uchar log_data[LSN_STORE_SIZE]; + uchar log_data[FILEID_STORE_SIZE]; log_array[TRANSLOG_INTERNAL_PARTS + 0].str= (char*) log_data; - log_array[TRANSLOG_INTERNAL_PARTS + 0].length= FILEID_STORE_SIZE; + log_array[TRANSLOG_INTERNAL_PARTS + 0].length= sizeof(log_data); if (unlikely(translog_write_record(&share->state.create_rename_lsn, LOGREC_REDO_DELETE_ALL, info->trn, share, 0, @@ -106,9 +106,7 @@ int maria_delete_all_rows(MARIA_HA *info) Note that storing the LSN could not be done by _ma_writeinfo() above as the table is locked at this moment. So we need to do it by ourselves. */ - lsn_store(log_data, share->state.create_rename_lsn); - if (my_pwrite(share->kfile.file, log_data, sizeof(log_data), - sizeof(share->state.header) + 2, MYF(MY_NABP)) || + if (_ma_update_create_rename_lsn_on_disk(share, FALSE) || _ma_sync_table_files(info)) goto err; /** diff --git a/storage/maria/ma_open.c b/storage/maria/ma_open.c index 4e72adf3b7e..5cd2bfbb838 100644 --- a/storage/maria/ma_open.c +++ b/storage/maria/ma_open.c @@ -587,7 +587,19 @@ MARIA_HA *maria_open(const char *name, int mode, uint open_flags) share->base.pack_bytes + test(share->options & HA_OPTION_CHECKSUM)); if (share->base.transactional) + { share->base_length+= TRANS_ROW_EXTRA_HEADER_SIZE; + if (unlikely((share->state.create_rename_lsn == (LSN)ULONGLONG_MAX) && + (open_flags & HA_OPEN_FROM_SQL_LAYER))) + { + /* + This table was repaired with maria_chk. Past log records should be + ignored, future log records should not: we define the present. + */ + share->state.create_rename_lsn= translog_get_horizon(); + _ma_update_create_rename_lsn_on_disk(share, TRUE); + } + } share->base.default_rec_buff_size= max(share->base.pack_reclength, share->base.max_key_length); share->page_type= (share->base.transactional ? PAGECACHE_LSN_PAGE : diff --git a/storage/maria/ma_rename.c b/storage/maria/ma_rename.c index 5224698c614..3f2a0a9002c 100644 --- a/storage/maria/ma_rename.c +++ b/storage/maria/ma_rename.c @@ -60,13 +60,13 @@ int maria_rename(const char *old_name, const char *new_name) MY_SYNC_DIR : 0; if (sync_dir) { - uchar log_data[LSN_STORE_SIZE]; + uchar log_data[2 + 2]; LEX_STRING log_array[TRANSLOG_INTERNAL_PARTS + 3]; uint old_name_len= strlen(old_name), new_name_len= strlen(new_name); int2store(log_data, old_name_len); int2store(log_data + 2, new_name_len); log_array[TRANSLOG_INTERNAL_PARTS + 0].str= log_data; - log_array[TRANSLOG_INTERNAL_PARTS + 0].length= 2 + 2; + log_array[TRANSLOG_INTERNAL_PARTS + 0].length= sizeof(log_data); log_array[TRANSLOG_INTERNAL_PARTS + 1].str= (char *)old_name; log_array[TRANSLOG_INTERNAL_PARTS + 1].length= old_name_len; log_array[TRANSLOG_INTERNAL_PARTS + 2].str= (char *)new_name; @@ -93,10 +93,7 @@ int maria_rename(const char *old_name, const char *new_name) store LSN into file, needed for Recovery to not be confused if a RENAME happened (applying REDOs to the wrong table). */ - lsn_store(log_data, share->state.create_rename_lsn); - if (my_pwrite(share->kfile.file, log_data, sizeof(log_data), - sizeof(share->state.header) + 2, MYF(MY_NABP)) || - my_sync(share->kfile.file, MYF(MY_WME))) + if (_ma_update_create_rename_lsn_on_disk(share, TRUE)) { maria_close(info); DBUG_RETURN(1); diff --git a/storage/maria/maria_chk.c b/storage/maria/maria_chk.c index 0b82a71f736..9019cc33295 100644 --- a/storage/maria/maria_chk.c +++ b/storage/maria/maria_chk.c @@ -1026,6 +1026,13 @@ static int maria_chk(HA_CHECK *param, my_string filename) } if (!error) { + /* + Tell the server's Recovery to ignore old REDOs on this table; we don't + know what the log's end LSN is now, so we just let the server know + that it will have to find and store it. + */ + if (share->base.transactional) + share->state.create_rename_lsn= (LSN)ULONGLONG_MAX; if ((param->testflag & (T_REP_BY_SORT | T_REP_PARALLEL)) && (maria_is_any_key_active(share->state.key_map) || (rep_quick && !param->keys_in_use && !recreate)) && diff --git a/storage/maria/maria_def.h b/storage/maria/maria_def.h index 740808c7bbe..39b8ba2292c 100644 --- a/storage/maria/maria_def.h +++ b/storage/maria/maria_def.h @@ -886,13 +886,13 @@ void _ma_remap_file(MARIA_HA *info, my_off_t size); MARIA_RECORD_POS _ma_write_init_default(MARIA_HA *info, const byte *record); my_bool _ma_write_abort_default(MARIA_HA *info); -/* Functions needed by _ma_check (are overrided in MySQL) */ C_MODE_START +int _ma_repair_write_log_record(const HA_CHECK *param, MARIA_HA *info); +/* Functions needed by _ma_check (are overrided in MySQL) */ volatile int *_ma_killed_ptr(HA_CHECK *param); void _ma_check_print_error _VARARGS((HA_CHECK *param, const char *fmt, ...)); void _ma_check_print_warning _VARARGS((HA_CHECK *param, const char *fmt, ...)); void _ma_check_print_info _VARARGS((HA_CHECK *param, const char *fmt, ...)); -int _ma_repair_write_log_record(const HA_CHECK *param, MARIA_HA *info); C_MODE_END int _ma_flush_pending_blocks(MARIA_SORT_PARAM *param); @@ -909,6 +909,7 @@ int _ma_create_index_by_sort(MARIA_SORT_PARAM *info, my_bool no_messages, ulong); int _ma_sync_table_files(const MARIA_HA *info); int _ma_initialize_data_file(File dfile, MARIA_SHARE *share); +int _ma_update_create_rename_lsn_on_disk(MARIA_SHARE *share, my_bool do_sync); void _ma_unpin_all_pages(MARIA_HA *info, LSN undo_lsn); -- cgit v1.2.1 From 4b1fe65b5fd169ed6d870e38f126780ed2184536 Mon Sep 17 00:00:00 2001 From: unknown Date: Thu, 28 Jun 2007 14:01:57 +0200 Subject: WL#3239 "log CREATE TABLE in Maria": write the log record _before_ creating the data file, and sync this log, so that the table cannot be used if log record didn't reach disk. The same way, we force the log in DROP/RENAME TABLE. Also in REPAIR TABLE though logging in this case is not polished. Making DELETE FROM t atomic: we log the record before starting the operation, and will finish this op at Recovery if needed. storage/maria/ma_check.c: comment. Force the log record for the log to have a complete history. storage/maria/ma_create.c: better conformance to the text of WL#3239 "log CREATE TABLE in Maria": write the log record before creating the data file. This ensures that the log can be applied to an old backup in all circumstances. errpos=2 was wrong. storage/maria/ma_delete_all.c: making DELETE FROM t atomic: we log the record before starting the operation, and will finish the operation at Recovery if needed. Thus there is no need to force files to disk. storage/maria/ma_delete_table.c: forcing the log before dropping a table, so that the log has the entire history. storage/maria/ma_loghandler.c: LOGREC_REDO_DELETE_ALL needs to set trn's rec_lsn so that the log's low-water mark and Checkpoint retain this record until the delete operation has finished. storage/maria/ma_rename.c: force the log before renaming a table, so that the log has a complete history. --- storage/maria/ma_check.c | 21 +++++++- storage/maria/ma_create.c | 112 +++++++++++++++++++++------------------- storage/maria/ma_delete_all.c | 65 +++++++++++------------ storage/maria/ma_delete_table.c | 9 ++-- storage/maria/ma_loghandler.c | 2 +- storage/maria/ma_rename.c | 7 +-- 6 files changed, 118 insertions(+), 98 deletions(-) diff --git a/storage/maria/ma_check.c b/storage/maria/ma_check.c index 72054ffe92a..cd10e87325c 100644 --- a/storage/maria/ma_check.c +++ b/storage/maria/ma_check.c @@ -5176,7 +5176,23 @@ int _ma_repair_write_log_record(const HA_CHECK *param, MARIA_HA *info) /* Only called from ha_maria.cc, not maria_check, so translog is inited */ if (share->base.transactional && !share->temporary) { - /* For now this record is only informative */ + /* + For now this record is only informative. It could serve when applying + logs to a backup, but that needs more thought. Assume table became + corrupted. It is repaired, then some writes happen to it. + Later we restore an old backup, and want to apply this REDO_REPAIR_TABLE + record. For it to give the same result as originally, the table should + be corrupted the same way, so applying previous REDOs should produce the + same corruption; that's really not guaranteed (different execution paths + in execution of REDOs vs runtime code so not same bugs hit, temporary + hardware issues not repeatable etc). Corruption may not be repeatable. + A reasonable solution is to execute the REDO_REPAIR_TABLE record and + check if the checksum of the resulting table matches what it was at the + end of the original repair (should be stored in log record); or execute + the REDO_REPAIR_TABLE if the checksum of the table-before-repair matches + was it was at the start of the original repair (should be stored in log + record). + */ LEX_STRING log_array[TRANSLOG_INTERNAL_PARTS + 1]; uchar log_data[LSN_STORE_SIZE]; compile_time_assert(LSN_STORE_SIZE >= (FILEID_STORE_SIZE + 4)); @@ -5193,7 +5209,8 @@ int _ma_repair_write_log_record(const HA_CHECK *param, MARIA_HA *info) log_array[TRANSLOG_INTERNAL_PARTS + 0].length, sizeof(log_array)/sizeof(log_array[0]), - log_array, log_data))) + log_array, log_data) || + translog_flush(share->state.create_rename_lsn))) return 1; /* But this piece is really needed, to have the new table's content durable diff --git a/storage/maria/ma_create.c b/storage/maria/ma_create.c index b439d7760e7..8ad8f0564d7 100644 --- a/storage/maria/ma_create.c +++ b/storage/maria/ma_create.c @@ -620,7 +620,7 @@ int maria_create(const char *name, enum data_file_type datafile_type, mi_int2store(share.state.header.state_info_length,MARIA_STATE_INFO_SIZE); mi_int2store(share.state.header.base_info_length,MARIA_BASE_INFO_SIZE); mi_int2store(share.state.header.base_pos,base_pos); - share.state.header.data_file_type= datafile_type; + share.state.header.data_file_type= share.data_file_type= datafile_type; share.state.header.org_data_file_type= org_datafile_type; share.state.header.language= (ci->language ? ci->language : default_charset_info->number); @@ -766,50 +766,6 @@ int maria_create(const char *name, enum data_file_type datafile_type, goto err; errpos=1; - if (!(flags & HA_DONT_TOUCH_DATA)) - { - if (ci->data_file_name) - { - char *dext= strrchr(ci->data_file_name, '.'); - int have_dext= dext && !strcmp(dext, MARIA_NAME_DEXT); - - if (tmp_table) - { - char *path; - /* chop off the table name, tempory tables use generated name */ - if ((path= strrchr(ci->data_file_name, FN_LIBCHAR))) - *path= '\0'; - fn_format(filename, name, ci->data_file_name, MARIA_NAME_DEXT, - MY_REPLACE_DIR | MY_UNPACK_FILENAME | MY_APPEND_EXT); - } - else - { - fn_format(filename, ci->data_file_name, "", MARIA_NAME_DEXT, - MY_UNPACK_FILENAME | - (have_dext ? MY_REPLACE_EXT : MY_APPEND_EXT)); - } - fn_format(linkname, name, "",MARIA_NAME_DEXT, - MY_UNPACK_FILENAME | MY_APPEND_EXT); - linkname_ptr= linkname; - create_flag=0; - } - else - { - fn_format(filename,name,"", MARIA_NAME_DEXT, - MY_UNPACK_FILENAME | MY_APPEND_EXT); - linkname_ptr= NULL; - create_flag=MY_DELETE_OLD; - } - if ((dfile= - my_create_with_symlink(linkname_ptr, filename, 0, create_mode, - MYF(MY_WME | create_flag | sync_dir))) < 0) - goto err; - errpos=3; - - share.data_file_type= datafile_type; - if (_ma_initialize_data_file(dfile, &share)) - goto err; - } DBUG_PRINT("info", ("write state info and base info")); if (_ma_state_info_write(file, &share.state, 2) || _ma_base_info_write(file, &share.base)) @@ -959,7 +915,7 @@ int maria_create(const char *name, enum data_file_type datafile_type, if ((log_data == NULL) || my_pread(file, 1 + 2 + 2 + log_data, kfile_size_before_extension, 0, MYF(MY_NABP))) - goto err_no_lock; + goto err; /* remember if the data file was created or not, to know if Recovery can do it or not, in the future @@ -989,8 +945,14 @@ int maria_create(const char *name, enum data_file_type datafile_type, MySQL layer to be crash-safe, which it is not now (that would require work using the ddl_log of sql/sql_table.cc); when it is, we should reconsider the moment of writing this log record (before or after op, - under THR_LOCK_maria or not...), how to use it in Recovery, and force - the log. For now this record is just informative. + under THR_LOCK_maria or not...), how to use it in Recovery. + For now this record can serve when we apply logs to a backup, + so we sync it. This happens before the data file is created. If the data + file was created before, and we crashed before writing the log record, + at restart the table may be used, so we would not have a trustable + history in the log (impossible to apply this log to a backup). The way + we do it, if we crash before writing the log record then there is no + data file and the table cannot be used. Note that in case of TRUNCATE TABLE we also come here. When in CREATE/TRUNCATE (or DROP or RENAME or REPAIR) we have not called external_lock(), so have no TRN. It does not matter, as all these @@ -1001,20 +963,63 @@ int maria_create(const char *name, enum data_file_type datafile_type, &dummy_transaction_object, NULL, total_rec_length, sizeof(log_array)/sizeof(log_array[0]), - log_array, NULL))) - goto err_no_lock; + log_array, NULL) || + translog_flush(share.state.create_rename_lsn))) + goto err; /* store LSN into file, needed for Recovery to not be confused if a DROP+CREATE happened (applying REDOs to the wrong table). - If such direct my_pwrite() to a fixed offset is too "hackish", I can - call ma_state_info_write() again but it will be less efficient. */ share.kfile.file= file; if (_ma_update_create_rename_lsn_on_disk(&share, FALSE)) - goto err_no_lock; + goto err; my_free(log_data, MYF(0)); } + if (!(flags & HA_DONT_TOUCH_DATA)) + { + if (ci->data_file_name) + { + char *dext= strrchr(ci->data_file_name, '.'); + int have_dext= dext && !strcmp(dext, MARIA_NAME_DEXT); + + if (tmp_table) + { + char *path; + /* chop off the table name, tempory tables use generated name */ + if ((path= strrchr(ci->data_file_name, FN_LIBCHAR))) + *path= '\0'; + fn_format(filename, name, ci->data_file_name, MARIA_NAME_DEXT, + MY_REPLACE_DIR | MY_UNPACK_FILENAME | MY_APPEND_EXT); + } + else + { + fn_format(filename, ci->data_file_name, "", MARIA_NAME_DEXT, + MY_UNPACK_FILENAME | + (have_dext ? MY_REPLACE_EXT : MY_APPEND_EXT)); + } + fn_format(linkname, name, "",MARIA_NAME_DEXT, + MY_UNPACK_FILENAME | MY_APPEND_EXT); + linkname_ptr= linkname; + create_flag=0; + } + else + { + fn_format(filename,name,"", MARIA_NAME_DEXT, + MY_UNPACK_FILENAME | MY_APPEND_EXT); + linkname_ptr= NULL; + create_flag=MY_DELETE_OLD; + } + if ((dfile= + my_create_with_symlink(linkname_ptr, filename, 0, create_mode, + MYF(MY_WME | create_flag | sync_dir))) < 0) + goto err; + errpos=3; + + if (_ma_initialize_data_file(dfile, &share)) + goto err; + } + /* Enlarge files */ DBUG_PRINT("info", ("enlarge to keystart: %lu", (ulong) share.base.keystart)); @@ -1030,7 +1035,6 @@ int maria_create(const char *name, enum data_file_type datafile_type, if (my_chsize(dfile,share.base.min_pack_length*ci->reloc_rows,0,MYF(0))) goto err; #endif - errpos=2; if ((sync_dir && my_sync(dfile, MYF(0))) || my_close(dfile,MYF(0))) goto err; } diff --git a/storage/maria/ma_delete_all.c b/storage/maria/ma_delete_all.c index a08e259d09b..3e531b518f8 100644 --- a/storage/maria/ma_delete_all.c +++ b/storage/maria/ma_delete_all.c @@ -17,7 +17,7 @@ /* This clears the status information and truncates files */ #include "maria_def.h" -#include "trnman_public.h" +#include "trnman.h" /** @brief deletes all rows from a table @@ -52,6 +52,25 @@ int maria_delete_all_rows(MARIA_HA *info) if (_ma_mark_file_changed(info)) goto err; + if (log_record) + { + /* + This record will be used by Recovery to finish the deletion if it + crashed. We force it because it's a non-undoable operation. + */ + LSN lsn; + LEX_STRING log_array[TRANSLOG_INTERNAL_PARTS + 1]; + uchar log_data[FILEID_STORE_SIZE]; + log_array[TRANSLOG_INTERNAL_PARTS + 0].str= (char*) log_data; + log_array[TRANSLOG_INTERNAL_PARTS + 0].length= sizeof(log_data); + if (unlikely(translog_write_record(&lsn, LOGREC_REDO_DELETE_ALL, + info->trn, share, 0, + sizeof(log_array)/sizeof(log_array[0]), + log_array, log_data) || + translog_flush(lsn))) + goto err; + } + info->state->records=info->state->del=state->split=0; state->changed= 0; /* File is optimized */ state->dellink = HA_OFFSET_ERROR; @@ -78,6 +97,12 @@ int maria_delete_all_rows(MARIA_HA *info) if (_ma_initialize_data_file(info->dfile.file, share)) goto err; + /* + The operations above on the index/data file will be forced to disk at + Checkpoint or maria_close() time. So we can reset: + */ + info->trn->rec_lsn= LSN_IMPOSSIBLE; + VOID(_ma_writeinfo(info,WRITEINFO_UPDATE_KEYFILE)); #ifdef HAVE_MMAP /* Resize mmaped area */ @@ -85,36 +110,6 @@ int maria_delete_all_rows(MARIA_HA *info) _ma_remap_file(info, (my_off_t)0); rw_unlock(&info->s->mmap_lock); #endif - if (log_record) - { - /* For now this record is only informative */ - LEX_STRING log_array[TRANSLOG_INTERNAL_PARTS + 1]; - uchar log_data[FILEID_STORE_SIZE]; - log_array[TRANSLOG_INTERNAL_PARTS + 0].str= (char*) log_data; - log_array[TRANSLOG_INTERNAL_PARTS + 0].length= sizeof(log_data); - if (unlikely(translog_write_record(&share->state.create_rename_lsn, - LOGREC_REDO_DELETE_ALL, - info->trn, share, 0, - sizeof(log_array)/sizeof(log_array[0]), - log_array, log_data))) - goto err; - /* - store LSN into file. It is an optimization so that all old REDOs for - this table are ignored (scenario: checkpoint, INSERT1s, DELETE ALL; - INSERT2s, crash: then Recovery can skip INSERT1s). It also allows us to - ignore the present record at Recovery. - Note that storing the LSN could not be done by _ma_writeinfo() above as - the table is locked at this moment. So we need to do it by ourselves. - */ - if (_ma_update_create_rename_lsn_on_disk(share, FALSE) || - _ma_sync_table_files(info)) - goto err; - /** - @todo RECOVERY Until we take into account the log record above - for log-low-water-mark calculation and use it in Recovery, we need - to sync above. - */ - } allow_break(); /* Allow SIGHUP & SIGINT */ DBUG_RETURN(0); @@ -123,9 +118,11 @@ err: int save_errno=my_errno; VOID(_ma_writeinfo(info,WRITEINFO_UPDATE_KEYFILE)); info->update|=HA_STATE_WRITTEN; /* Buffer changed */ - /** @todo RECOVERY until we use the log record above we have to sync */ - if (log_record &&_ma_sync_table_files(info) && !save_errno) - save_errno= my_errno; + /** + @todo RECOVERY if we come here, Recovery may later apply the REDO above, + which may be wrong. Not fixing it now, as anyway this way of deleting + rows will have to be re-examined when we have versioning. + */ allow_break(); /* Allow SIGHUP & SIGINT */ DBUG_RETURN(my_errno=save_errno); } diff --git a/storage/maria/ma_delete_table.c b/storage/maria/ma_delete_table.c index 990714043bf..39a286ad1f7 100644 --- a/storage/maria/ma_delete_table.c +++ b/storage/maria/ma_delete_table.c @@ -78,9 +78,9 @@ int maria_delete_table(const char *name) { /* For this log record to be of any use for Recovery, we need the upper - MySQL layer to be crash-safe in DDLs; when it is we should reconsider - the moment of writing this log record, how to use it in Recovery, and - force the log. For now this record is only informative. + MySQL layer to be crash-safe in DDLs. + For now this record can serve when we apply logs to a backup, so we sync + it. */ LSN lsn; LEX_STRING log_array[TRANSLOG_INTERNAL_PARTS + 1]; @@ -91,7 +91,8 @@ int maria_delete_table(const char *name) log_array[TRANSLOG_INTERNAL_PARTS + 0].length, sizeof(log_array)/sizeof(log_array[0]), - log_array, NULL))) + log_array, NULL) || + translog_flush(lsn))) DBUG_RETURN(1); } diff --git a/storage/maria/ma_loghandler.c b/storage/maria/ma_loghandler.c index 79bf44046b1..3a8e01da09a 100644 --- a/storage/maria/ma_loghandler.c +++ b/storage/maria/ma_loghandler.c @@ -375,7 +375,7 @@ static LOG_DESC INIT_LOGREC_REDO_DROP_TABLE= static LOG_DESC INIT_LOGREC_REDO_DELETE_ALL= {LOGRECTYPE_FIXEDLENGTH, FILEID_STORE_SIZE, FILEID_STORE_SIZE, - NULL, NULL, NULL, 0, + NULL, write_hook_for_redo, NULL, 0, "redo_delete_all", TRUE, NULL, NULL}; static LOG_DESC INIT_LOGREC_REDO_REPAIR_TABLE= diff --git a/storage/maria/ma_rename.c b/storage/maria/ma_rename.c index 3f2a0a9002c..8f42a5b931a 100644 --- a/storage/maria/ma_rename.c +++ b/storage/maria/ma_rename.c @@ -76,15 +76,16 @@ int maria_rename(const char *old_name, const char *new_name) MySQL layer to be crash-safe, which it is not now (that would require work using the ddl_log of sql/sql_table.cc); when it is, we should reconsider the moment of writing this log record (before or after op, - under THR_LOCK_maria or not...), how to use it in Recovery, and force - the log. For now this record is just informative. + under THR_LOCK_maria or not...), how to use it in Recovery. + For now it can serve to apply logs to a backup so we sync it. */ if (unlikely(translog_write_record(&share->state.create_rename_lsn, LOGREC_REDO_RENAME_TABLE, &dummy_transaction_object, NULL, 2 + 2 + old_name_len + new_name_len, sizeof(log_array)/sizeof(log_array[0]), - log_array, NULL))) + log_array, NULL) || + translog_flush(share->state.create_rename_lsn))) { maria_close(info); DBUG_RETURN(1); -- cgit v1.2.1 From d6f2fda680ec2be373ff5694e91cfccf792f4eb0 Mon Sep 17 00:00:00 2001 From: unknown Date: Sun, 1 Jul 2007 16:20:57 +0300 Subject: Fixed REPAIR/CHECK/ANALYZE TABLE for tables with new BLOCK-ROW format. Fixed maria_chk to repair BLOCK-ROW tables. Added CREATE options ROW_FORMAT=PAGE & TRANSACTIONAL= 0|1 More DBUG information in a lot of functions Some minor code cleanups Enable handler errors earlier for better clear text error messages at handler startup / standalone usage. Don't print NULL strings in my_create_with_symlink(); Fixes core dump when used with --debug include/maria.h: Added extra variables needed for REPAIR with BLOCK records include/my_base.h: Added argument for opening copy of maria table without a shared object include/my_handler.h: Prototypes for my_handler_error_register() & my_handler_error_unregister() include/pagecache.h: Added PAGECACHE_READ_UNKNOWN_PAGE mysql-test/include/ps_conv.inc: Enforce creation of table as MyISAM (to allow one to use --default-storage-engine) mysql-test/r/maria.result: Moved some things to maria-connect.test Updared results as REPAIR now works Added tests for creation option TRANSACTIONAL mysql-test/r/ps_2myisam.result: Enforce creation of table as MyISAM (to allow one to use --default-storage-engine) mysql-test/r/ps_3innodb.result: Enforce creation of table as MyISAM (to allow one to use --default-storage-engine) mysql-test/r/ps_4heap.result: Enforce creation of table as MyISAM (to allow one to use --default-storage-engine) mysql-test/r/ps_5merge.result: Enforce creation of table as MyISAM (to allow one to use --default-storage-engine) mysql-test/r/ps_7ndb.result: Enforce creation of table as MyISAM (to allow one to use --default-storage-engine) mysql-test/r/ps_maria.result: Enforce creation of table as MyISAM (to allow one to use --default-storage-engine) mysql-test/t/maria.test: Moved some things to maria-connect.test Updared results as REPAIR now works Added tests for creation option TRANSACTIONAL mysys/mf_iocache.c: More debugging mysys/mf_tempfile.c: Added missing close() mysys/my_error.c: init_glob_errs() is now done in my_init() mysys/my_handler.c: Added functions to initialize handler error messages mysys/my_init.c: Moevd init_glob_errs() here. mysys/my_open.c: More comments More debugging Code cleanup (join multiple code paths) and indentation fixes. No change in logic. mysys/my_symlink2.c: Don't print NULL strings sql/handler.cc: Added printing of PAGE row type Moved out initializing of handler errors to allow handler to give better error messages at startup sql/handler.h: ROW_TYPE_PAGES -> ROW_TYPE_PAGE sql/lex.h: Added 'PAGE' and 'TRANSACTIONAL' sql/mysqld.cc: Initialize handler error messages early to get better error messages from handler startup sql/sql_show.cc: ROW_TYPE_PAGES -> ROW_TYPE_PAGE sql/sql_table.cc: Removed not needed initializer sql/sql_yacc.yy: Added CREATE options ROW_FORMAT=PAGE and TRANSACTIONAL=[0|1] sql/table.cc: Store transactional flag in .frm More comments sql-bench/example: Better example sql/table.h: Added transactional table option storage/maria/ha_maria.cc: More debug information Enable REPAIR Detect usage of TRANSACTIONAL table option storage/maria/ma_bitmap.c: More comments (from Guilhem) storage/maria/ma_blockrec.c: SANITY_CHECK -> SANITY_CHECKS (fixed typo) Write out pages on delete even if there is no rows. (Fixed problem with REPAIR) Removed some ASSERTS to runtime checks (for better REPAIR) Fixed bug when scanning rows More DBUG information storage/maria/ma_check.c: Partial rewrite to allow REPAIR of BLOCK/PAGE format. Repair of BLOCK format rows is for now only done with 'maria_repair()' (= repair through key cache) The new logic to repair rows with BLOCK format is: - Create new, unrelated MARIA_HA of the table - Create new datafile and associate it with new handler - Reset all statistic information in new handler - Copy all data to new handler with normal write operations - Move state of new handler to old handler - Close new handler - Close data file in old handler - Rename old data file to new data file. - Reopen data file in old handler storage/maria/ma_close.c: REmoved not needed block storage/maria/ma_create.c: Swap arguments to _ma_initialize_data_file() storage/maria/ma_delete_all.c: Split maria_delete_all_rows() to two functions to allow REPAIR to easily reset all status information. storage/maria/ma_dynrec.c: Added checksum argument to _ma_rec_check (multi-thread fix) storage/maria/ma_info.c: Indentation fix storage/maria/ma_init.c: Register error message to get better error message on init and when using as standalone module. storage/maria/ma_loghandler.c: Fixed typo that disabled some error detection by valgrind storage/maria/ma_open.c: Added 'calc_check_checksum()' Don't log things during repair Added option HA_OPEN_COPY to allow one to open a Maria table with an independent share (required by REPAIR) storage/maria/ma_pagecache.c: Fixed some compiler warnings Added support for PAGECACHE_READ_UNKNOWN_PAGE (used for scanning file without knowing page types) storage/maria/ma_test_all.sh: More test of REPAIR storage/maria/ma_update.c: Optimized checksum code storage/maria/maria_chk.c: Use DBUG_SET_INITIAL() to get DBUG to work with --parallel-repair Ensure we always use maria_repair() for BLOCK format (for now) More DBUG information storage/maria/maria_def.h: For now, always run with more checkings (SANITY_CHECKS) Added share->calc_check_checksum to be used with REPAIR / CHECK table. Swaped arguments to _ma_initialize_data_file() storage/myisam/ft_stopwords.c: Added DBUG information mysql-test/r/maria-connect.result: New BitKeeper file ``mysql-test/r/maria-connect.result'' mysql-test/t/maria-connect.test: New BitKeeper file ``mysql-test/t/maria-connect.test'' --- include/maria.h | 5 +- include/my_base.h | 1 + include/my_handler.h | 3 +- include/pagecache.h | 4 +- mysql-test/include/ps_conv.inc | 2 +- mysql-test/r/maria-connect.result | 23 ++ mysql-test/r/maria.result | 73 +++-- mysql-test/r/ps_2myisam.result | 2 +- mysql-test/r/ps_3innodb.result | 2 +- mysql-test/r/ps_4heap.result | 2 +- mysql-test/r/ps_5merge.result | 4 +- mysql-test/r/ps_7ndb.result | 2 +- mysql-test/r/ps_maria.result | 2 +- mysql-test/t/maria-connect.test | 39 +++ mysql-test/t/maria.test | 42 +-- mysys/mf_iocache.c | 1 + mysys/mf_tempfile.c | 1 + mysys/my_error.c | 5 - mysys/my_handler.c | 66 +++++ mysys/my_init.c | 1 + mysys/my_open.c | 73 ++--- mysys/my_symlink2.c | 4 +- sql-bench/example | 9 +- sql/handler.cc | 8 +- sql/handler.h | 7 +- sql/lex.h | 2 + sql/mysqld.cc | 4 + sql/sql_show.cc | 9 +- sql/sql_table.cc | 3 +- sql/sql_yacc.yy | 11 + sql/table.cc | 8 +- sql/table.h | 1 + storage/maria/ha_maria.cc | 31 ++- storage/maria/ma_bitmap.c | 13 + storage/maria/ma_blockrec.c | 54 ++-- storage/maria/ma_check.c | 570 +++++++++++++++++++++++++++++--------- storage/maria/ma_close.c | 4 +- storage/maria/ma_create.c | 4 +- storage/maria/ma_delete_all.c | 54 ++-- storage/maria/ma_dynrec.c | 5 +- storage/maria/ma_info.c | 1 + storage/maria/ma_init.c | 1 + storage/maria/ma_loghandler.c | 4 +- storage/maria/ma_open.c | 8 +- storage/maria/ma_pagecache.c | 14 +- storage/maria/ma_test_all.sh | 21 ++ storage/maria/ma_update.c | 3 +- storage/maria/maria_chk.c | 42 +-- storage/maria/maria_def.h | 11 +- storage/myisam/ft_stopwords.c | 17 +- 50 files changed, 944 insertions(+), 332 deletions(-) create mode 100644 mysql-test/r/maria-connect.result create mode 100644 mysql-test/t/maria-connect.test diff --git a/include/maria.h b/include/maria.h index fbf4bc68c29..c7c32ff0e78 100644 --- a/include/maria.h +++ b/include/maria.h @@ -327,17 +327,18 @@ typedef struct st_maria_sort_info pthread_mutex_t mutex; pthread_cond_t cond; #endif - MARIA_HA *info; + MARIA_HA *info, *new_info; HA_CHECK *param; char *buff; SORT_KEY_BLOCKS *key_block, *key_block_end; SORT_FT_BUF *ft_buf; my_off_t filelength, dupp, buff_length; + ulonglong page; ha_rows max_records; uint current_key, total_keys; uint got_error, threads_running; myf myf_rw; - enum data_file_type new_data_file_type; + enum data_file_type new_data_file_type, org_data_file_type; } MARIA_SORT_INFO; typedef struct st_maria_sort_param diff --git a/include/my_base.h b/include/my_base.h index 38376adfe85..952c325f911 100644 --- a/include/my_base.h +++ b/include/my_base.h @@ -47,6 +47,7 @@ #define HA_OPEN_FOR_REPAIR 32 /* open even if crashed */ #define HA_OPEN_FROM_SQL_LAYER 64 #define HA_OPEN_MMAP 128 /* open memory mapped */ +#define HA_OPEN_COPY 256 /* Open copy (for repair) */ /* The following is parameter to ha_rkey() how to use key */ diff --git a/include/my_handler.h b/include/my_handler.h index 13dcd01a332..1a1235d0588 100644 --- a/include/my_handler.h +++ b/include/my_handler.h @@ -110,7 +110,8 @@ extern int ha_key_cmp(register HA_KEYSEG *keyseg, register uchar *a, uint *diff_pos); extern HA_KEYSEG *ha_find_null(HA_KEYSEG *keyseg, uchar *a); - +extern void my_handler_error_register(void); +extern void my_handler_error_unregister(void); /* Inside an in-memory data record, memory pointers to pieces of the record (like BLOBs) are stored in their native byte order and in diff --git a/include/pagecache.h b/include/pagecache.h index 12906676959..f505dec0fe9 100644 --- a/include/pagecache.h +++ b/include/pagecache.h @@ -34,7 +34,9 @@ enum pagecache_page_type /* the page does not contain LSN */ PAGECACHE_PLAIN_PAGE, /* the page contain LSN (maria tablespace page) */ - PAGECACHE_LSN_PAGE + PAGECACHE_LSN_PAGE, + /* Page type used when scanning file and we don't care about the type */ + PAGECACHE_READ_UNKNOWN_PAGE }; /* diff --git a/mysql-test/include/ps_conv.inc b/mysql-test/include/ps_conv.inc index 0dd819f6e62..b0c0f9bd9e0 100644 --- a/mysql-test/include/ps_conv.inc +++ b/mysql-test/include/ps_conv.inc @@ -52,7 +52,7 @@ set @arg14= 'abc'; set @arg14= NULL ; set @arg15= CAST('abc' as binary) ; set @arg15= NULL ; -create table t5 as select +eval create table t5 engine = MyISAM as select 8 as const01, @arg01 as param01, 8.0 as const02, @arg02 as param02, 80.00000000000e-1 as const03, @arg03 as param03, diff --git a/mysql-test/r/maria-connect.result b/mysql-test/r/maria-connect.result new file mode 100644 index 00000000000..e232f564d10 --- /dev/null +++ b/mysql-test/r/maria-connect.result @@ -0,0 +1,23 @@ +set global storage_engine=maria; +set session storage_engine=maria; +drop table if exists t1; +SET SQL_WARNINGS=1; +RESET MASTER; +set binlog_format=statement; +CREATE TABLE t1 (a int primary key); +insert t1 values (1),(2),(3); +insert t1 values (4),(2),(5); +ERROR 23000: Duplicate entry '2' for key 'PRIMARY' +select * from t1; +a +1 +2 +3 +4 +SHOW BINLOG EVENTS FROM 102; +Log_name Pos Event_type Server_id End_log_pos Info +master-bin.000001 102 Query 1 200 use `test`; CREATE TABLE t1 (a int primary key) +master-bin.000001 200 Query 1 291 use `test`; insert t1 values (1),(2),(3) +master-bin.000001 291 Query 1 382 use `test`; insert t1 values (4),(2),(5) +drop table t1; +set binlog_format=default; diff --git a/mysql-test/r/maria.result b/mysql-test/r/maria.result index 3ec9af0fffa..a1c49fdfe4f 100644 --- a/mysql-test/r/maria.result +++ b/mysql-test/r/maria.result @@ -2,25 +2,6 @@ set global storage_engine=maria; set session storage_engine=maria; drop table if exists t1,t2; SET SQL_WARNINGS=1; -RESET MASTER; -set binlog_format=statement; -CREATE TABLE t1 (a int primary key); -insert t1 values (1),(2),(3); -insert t1 values (4),(2),(5); -ERROR 23000: Duplicate entry '2' for key 'PRIMARY' -select * from t1; -a -1 -2 -3 -4 -SHOW BINLOG EVENTS FROM 102; -Log_name Pos Event_type Server_id End_log_pos Info -master-bin.000001 102 Query 1 200 use `test`; CREATE TABLE t1 (a int primary key) -master-bin.000001 200 Query 1 291 use `test`; insert t1 values (1),(2),(3) -master-bin.000001 291 Query 1 382 use `test`; insert t1 values (4),(2),(5) -drop table t1; -set binlog_format=default; CREATE TABLE t1 ( STRING_DATA char(255) default NULL, KEY string_data (STRING_DATA) @@ -618,7 +599,7 @@ t1 1 a 1 a A NULL NULL NULL YES BTREE disabled alter table t1 enable keys; show keys from t1; Table Non_unique Key_name Seq_in_index Column_name Collation Cardinality Sub_part Packed Null Index_type Comment -t1 1 a 1 a A NULL NULL NULL YES BTREE disabled +t1 1 a 1 a A NULL NULL NULL YES BTREE alter table t1 engine=heap; alter table t1 disable keys; Warnings: @@ -853,19 +834,19 @@ _id DELETE FROM t1 WHERE _id < 8; SHOW TABLE STATUS LIKE 't1'; Name Engine Version Row_format Rows Avg_row_length Data_length Max_data_length Index_length Data_free Auto_increment Create_time Update_time Check_time Collation Checksum Create_options Comment -t1 MARIA 10 Paged 2 # # # # 0 # # # # # # +t1 MARIA 10 Page 2 # # # # 0 # # # # # # CHECK TABLE t1 EXTENDED; Table Op Msg_type Msg_text test.t1 check status OK OPTIMIZE TABLE t1; Table Op Msg_type Msg_text -test.t1 optimize status Table is already up to date +test.t1 optimize status OK CHECK TABLE t1 EXTENDED; Table Op Msg_type Msg_text test.t1 check status OK SHOW TABLE STATUS LIKE 't1'; Name Engine Version Row_format Rows Avg_row_length Data_length Max_data_length Index_length Data_free Auto_increment Create_time Update_time Check_time Collation Checksum Create_options Comment -t1 MARIA 10 Paged 2 # # # # 0 # # # # # # +t1 MARIA 10 Page 2 # # # # 0 # # # # # # SELECT _id FROM t1; _id 8 @@ -912,7 +893,7 @@ _id DELETE FROM t1 WHERE _id < 8; SHOW TABLE STATUS LIKE 't1'; Name Engine Version Row_format Rows Avg_row_length Data_length Max_data_length Index_length Data_free Auto_increment Create_time Update_time Check_time Collation Checksum Create_options Comment -t1 MARIA 10 Paged 2 # # # # 0 # # # # # # +t1 MARIA 10 Page 2 # # # # 0 # # # # # # CHECK TABLE t1 EXTENDED; Table Op Msg_type Msg_text test.t1 check status OK @@ -924,7 +905,7 @@ Table Op Msg_type Msg_text test.t1 check status OK SHOW TABLE STATUS LIKE 't1'; Name Engine Version Row_format Rows Avg_row_length Data_length Max_data_length Index_length Data_free Auto_increment Create_time Update_time Check_time Collation Checksum Create_options Comment -t1 MARIA 10 Paged 2 # # # # 0 # # # # # # +t1 MARIA 10 Page 2 # # # # 0 # # # # # # SELECT _id FROM t1; _id 8 @@ -1598,7 +1579,7 @@ alter table t1 disable keys; alter table t1 enable keys; show keys from t1; Table Non_unique Key_name Seq_in_index Column_name Collation Cardinality Sub_part Packed Null Index_type Comment -t1 1 a 1 a A 8 NULL NULL YES BTREE disabled +t1 1 a 1 a A 8 NULL NULL YES BTREE drop table t1; show create table t1; show create table t1; @@ -1811,3 +1792,43 @@ CHECK TABLE t1; Table Op Msg_type Msg_text test.t1 check status OK DROP TABLE t1; +create table t1 (a int) transactional=0; +show create table t1; +Table Create Table +t1 CREATE TABLE `t1` ( + `a` int(11) DEFAULT NULL +) ENGINE=MARIA DEFAULT CHARSET=latin1 TRANSACTIONAL=0 +drop table t1; +create table t1 (a int) row_format=dynamic transactional=0; +show create table t1; +Table Create Table +t1 CREATE TABLE `t1` ( + `a` int(11) DEFAULT NULL +) ENGINE=MARIA DEFAULT CHARSET=latin1 ROW_FORMAT=DYNAMIC TRANSACTIONAL=0 +drop table t1; +create table t1 (a int) row_format=dynamic transactional=1; +show create table t1; +Table Create Table +t1 CREATE TABLE `t1` ( + `a` int(11) DEFAULT NULL +) ENGINE=MARIA DEFAULT CHARSET=latin1 ROW_FORMAT=DYNAMIC TRANSACTIONAL=1 +alter table t1 row_format=PAGE; +show create table t1; +Table Create Table +t1 CREATE TABLE `t1` ( + `a` int(11) DEFAULT NULL +) ENGINE=MARIA DEFAULT CHARSET=latin1 ROW_FORMAT=PAGE TRANSACTIONAL=1 +alter table t1 transactional=0; +show create table t1; +Table Create Table +t1 CREATE TABLE `t1` ( + `a` int(11) DEFAULT NULL +) ENGINE=MARIA DEFAULT CHARSET=latin1 ROW_FORMAT=PAGE TRANSACTIONAL=0 +drop table t1; +create table t1 (a int) row_format=PAGE; +show create table t1; +Table Create Table +t1 CREATE TABLE `t1` ( + `a` int(11) DEFAULT NULL +) ENGINE=MARIA DEFAULT CHARSET=latin1 ROW_FORMAT=PAGE +drop table t1; diff --git a/mysql-test/r/ps_2myisam.result b/mysql-test/r/ps_2myisam.result index 2bfd6d31ac9..9330ac2853e 100644 --- a/mysql-test/r/ps_2myisam.result +++ b/mysql-test/r/ps_2myisam.result @@ -1756,7 +1756,7 @@ set @arg14= 'abc'; set @arg14= NULL ; set @arg15= CAST('abc' as binary) ; set @arg15= NULL ; -create table t5 as select +create table t5 engine = MyISAM as select 8 as const01, @arg01 as param01, 8.0 as const02, @arg02 as param02, 80.00000000000e-1 as const03, @arg03 as param03, diff --git a/mysql-test/r/ps_3innodb.result b/mysql-test/r/ps_3innodb.result index 607a0426bd7..4972942e6f3 100644 --- a/mysql-test/r/ps_3innodb.result +++ b/mysql-test/r/ps_3innodb.result @@ -1739,7 +1739,7 @@ set @arg14= 'abc'; set @arg14= NULL ; set @arg15= CAST('abc' as binary) ; set @arg15= NULL ; -create table t5 as select +create table t5 engine = MyISAM as select 8 as const01, @arg01 as param01, 8.0 as const02, @arg02 as param02, 80.00000000000e-1 as const03, @arg03 as param03, diff --git a/mysql-test/r/ps_4heap.result b/mysql-test/r/ps_4heap.result index f4eec0c610c..bb17d0d161c 100644 --- a/mysql-test/r/ps_4heap.result +++ b/mysql-test/r/ps_4heap.result @@ -1740,7 +1740,7 @@ set @arg14= 'abc'; set @arg14= NULL ; set @arg15= CAST('abc' as binary) ; set @arg15= NULL ; -create table t5 as select +create table t5 engine = MyISAM as select 8 as const01, @arg01 as param01, 8.0 as const02, @arg02 as param02, 80.00000000000e-1 as const03, @arg03 as param03, diff --git a/mysql-test/r/ps_5merge.result b/mysql-test/r/ps_5merge.result index 38e4626d59c..1ed2136151b 100644 --- a/mysql-test/r/ps_5merge.result +++ b/mysql-test/r/ps_5merge.result @@ -1676,7 +1676,7 @@ set @arg14= 'abc'; set @arg14= NULL ; set @arg15= CAST('abc' as binary) ; set @arg15= NULL ; -create table t5 as select +create table t5 engine = MyISAM as select 8 as const01, @arg01 as param01, 8.0 as const02, @arg02 as param02, 80.00000000000e-1 as const03, @arg03 as param03, @@ -4690,7 +4690,7 @@ set @arg14= 'abc'; set @arg14= NULL ; set @arg15= CAST('abc' as binary) ; set @arg15= NULL ; -create table t5 as select +create table t5 engine = MyISAM as select 8 as const01, @arg01 as param01, 8.0 as const02, @arg02 as param02, 80.00000000000e-1 as const03, @arg03 as param03, diff --git a/mysql-test/r/ps_7ndb.result b/mysql-test/r/ps_7ndb.result index 432a07df9d0..af693de986b 100644 --- a/mysql-test/r/ps_7ndb.result +++ b/mysql-test/r/ps_7ndb.result @@ -1739,7 +1739,7 @@ set @arg14= 'abc'; set @arg14= NULL ; set @arg15= CAST('abc' as binary) ; set @arg15= NULL ; -create table t5 as select +create table t5 engine = MyISAM as select 8 as const01, @arg01 as param01, 8.0 as const02, @arg02 as param02, 80.00000000000e-1 as const03, @arg03 as param03, diff --git a/mysql-test/r/ps_maria.result b/mysql-test/r/ps_maria.result index 9268c44eecd..b1ea905c406 100644 --- a/mysql-test/r/ps_maria.result +++ b/mysql-test/r/ps_maria.result @@ -1756,7 +1756,7 @@ set @arg14= 'abc'; set @arg14= NULL ; set @arg15= CAST('abc' as binary) ; set @arg15= NULL ; -create table t5 as select +create table t5 engine = MyISAM as select 8 as const01, @arg01 as param01, 8.0 as const02, @arg02 as param02, 80.00000000000e-1 as const03, @arg03 as param03, diff --git a/mysql-test/t/maria-connect.test b/mysql-test/t/maria-connect.test new file mode 100644 index 00000000000..aedfa92e278 --- /dev/null +++ b/mysql-test/t/maria-connect.test @@ -0,0 +1,39 @@ +# +# Test that can't be run with --extern +# + +-- source include/have_maria.inc + +let $default=`select @@global.storage_engine`; +set global storage_engine=maria; +set session storage_engine=maria; + +# Initialise +--disable_warnings +drop table if exists t1; +--enable_warnings +SET SQL_WARNINGS=1; + +# +# UNIQUE key test +# +# as long as maria cannot rollback, binlog should contain both inserts +# +RESET MASTER; +set binlog_format=statement; +CREATE TABLE t1 (a int primary key); +insert t1 values (1),(2),(3); +--error 1582 +insert t1 values (4),(2),(5); +select * from t1; +SHOW BINLOG EVENTS FROM 102; +drop table t1; +set binlog_format=default; + +# End of 5.2 tests + +--disable_result_log +--disable_query_log +eval set global storage_engine=$default; +--enable_result_log +--enable_query_log diff --git a/mysql-test/t/maria.test b/mysql-test/t/maria.test index 763abbd9d25..25000abd426 100644 --- a/mysql-test/t/maria.test +++ b/mysql-test/t/maria.test @@ -15,22 +15,6 @@ drop table if exists t1,t2; --enable_warnings SET SQL_WARNINGS=1; -# -# UNIQUE key test -# -# as long as maria cannot rollback, binlog should contain both inserts -# -RESET MASTER; -set binlog_format=statement; -CREATE TABLE t1 (a int primary key); -insert t1 values (1),(2),(3); ---error 1582 -insert t1 values (4),(2),(5); -select * from t1; -SHOW BINLOG EVENTS FROM 102; -drop table t1; -set binlog_format=default; - # # Test problem with CHECK TABLE; # @@ -597,10 +581,7 @@ insert t1 select * from t2; show keys from t1; alter table t1 enable keys; show keys from t1; -#TODO after we have repair: delete the following --disable-warnings ---disable_warnings alter table t1 engine=heap; ---enable_warnings alter table t1 disable keys; show keys from t1; drop table t1,t2; @@ -1072,10 +1053,10 @@ create table t1 (a int not null, key key_block_size=1024 (a)); --error 1064 create table t1 (a int not null, key `a` key_block_size=1024 (a)); - # # Test of changing MI_KEY_BLOCK_LENGTH # + CREATE TABLE t1 ( c1 INT, c2 VARCHAR(300), @@ -1116,6 +1097,27 @@ DELETE FROM t1 WHERE c1 >= 10; CHECK TABLE t1; DROP TABLE t1; +# +# Test that TRANSACTIONAL is preserved +# + +create table t1 (a int) transactional=0; +show create table t1; +drop table t1; +create table t1 (a int) row_format=dynamic transactional=0; +show create table t1; +drop table t1; +create table t1 (a int) row_format=dynamic transactional=1; +show create table t1; +alter table t1 row_format=PAGE; +show create table t1; +alter table t1 transactional=0; +show create table t1; +drop table t1; +create table t1 (a int) row_format=PAGE; +show create table t1; +drop table t1; + # End of 5.2 tests --disable_result_log diff --git a/mysys/mf_iocache.c b/mysys/mf_iocache.c index e40490776f8..8b8ba540a4e 100644 --- a/mysys/mf_iocache.c +++ b/mysys/mf_iocache.c @@ -1696,6 +1696,7 @@ int my_b_flush_io_cache(IO_CACHE *info, int need_append_buffer_lock) my_bool append_cache; my_off_t pos_in_file; DBUG_ENTER("my_b_flush_io_cache"); + DBUG_PRINT("enter", ("cache: 0x%lx", (long) info)); if (!(append_cache = (info->type == SEQ_READ_APPEND))) need_append_buffer_lock=0; diff --git a/mysys/mf_tempfile.c b/mysys/mf_tempfile.c index 6c412157937..a820d09a2c6 100644 --- a/mysys/mf_tempfile.c +++ b/mysys/mf_tempfile.c @@ -107,6 +107,7 @@ File create_temp_file(char *to, const char *dir, const char *prefix, if (org_file >= 0 && file < 0) { int tmp=my_errno; + close(org_file); (void) my_delete(to, MYF(MY_WME | ME_NOINPUT)); my_errno=tmp; } diff --git a/mysys/my_error.c b/mysys/my_error.c index 48392fe84c3..00c78b64e0e 100644 --- a/mysys/my_error.c +++ b/mysys/my_error.c @@ -84,11 +84,6 @@ int my_error(int nr, myf MyFlags, ...) if (nr <= meh_p->meh_last) break; -#ifdef SHARED_LIBRARY - if ((meh_p == &my_errmsgs_globerrs) && ! globerrs[0]) - init_glob_errs(); -#endif - /* get the error message string. Default, if NULL or empty string (""). */ if (! (format= (meh_p && (nr >= meh_p->meh_first)) ? meh_p->meh_errmsgs[nr - meh_p->meh_first] : NULL) || ! *format) diff --git a/mysys/my_handler.c b/mysys/my_handler.c index 757cbe490f8..bf75d992f9d 100644 --- a/mysys/my_handler.c +++ b/mysys/my_handler.c @@ -19,6 +19,7 @@ #include #include #include +#include int ha_compare_text(CHARSET_INFO *charset_info, uchar *a, uint a_length, uchar *b, uint b_length, my_bool part_key, @@ -563,3 +564,68 @@ HA_KEYSEG *ha_find_null(HA_KEYSEG *keyseg, uchar *a) return keyseg; } + +/* + Errors a handler can give you +*/ + +static const char *handler_error_messages[]= +{ + "Didn't find key on read or update", + "Duplicate key on write or update", + "Undefined handler error 122", + "Someone has changed the row since it was read (while the table was locked to prevent it)", + "Wrong index given to function", + "Undefined handler error 125", + "Index file is crashed", + "Record file is crashed", + "Out of memory in engine", + "Undefined handler error 129", + "Incorrect file format", + "Command not supported by database", + "Old database file", + "No record read before update", + "Record was already deleted (or record file crashed)", + "No more room in record file", + "No more room in index file", + "No more records (read after end of file)", + "Unsupported extension used for table", + "Too big row", + "Wrong create options", + "Duplicate unique key or constraint on write or update", + "Unknown character set used in table", + "Conflicting table definitions in sub-tables of MERGE table", + "Table is crashed and last repair failed", + "Table was marked as crashed and should be repaired", + "Lock timed out; Retry transaction", + "Lock table is full; Restart program with a larger locktable", + "Updates are not allowed under a read only transactions", + "Lock deadlock; Retry transaction", + "Foreign key constraint is incorrectly formed", + "Cannot add a child row", + "Cannot delete a parent row", + "Unknown handler error" +}; + + +/* + Register handler error messages for usage with my_error() + + NOTES + This is safe to call multiple times as my_error_register() + will ignore calls to register already registered error numbers. +*/ + + +void my_handler_error_register(void) +{ + my_error_register(handler_error_messages, HA_ERR_FIRST, + HA_ERR_FIRST+ array_elements(handler_error_messages)-1); +} + + +void my_handler_error_unregister(void) +{ + my_error_unregister(HA_ERR_FIRST, + HA_ERR_FIRST+ array_elements(handler_error_messages)-1); +} diff --git a/mysys/my_init.c b/mysys/my_init.c index e8a55fdc1e6..2023a7da223 100644 --- a/mysys/my_init.c +++ b/mysys/my_init.c @@ -77,6 +77,7 @@ my_bool my_init(void) mysys_usage_id++; my_umask= 0660; /* Default umask for new files */ my_umask_dir= 0700; /* Default umask for new directories */ + init_glob_errs(); #if defined(THREAD) && defined(SAFE_MUTEX) safe_mutex_global_init(); /* Must be called early */ #endif diff --git a/mysys/my_open.c b/mysys/my_open.c index 6fe7883b99b..b4bb7e25810 100644 --- a/mysys/my_open.c +++ b/mysys/my_open.c @@ -71,6 +71,7 @@ File my_open(const char *FileName, int Flags, myf MyFlags) #else fd = open((my_string) FileName, Flags); #endif + DBUG_RETURN(my_register_filename(fd, FileName, FILE_BY_OPEN, EE_FILENOTFOUND, MyFlags)); } /* my_open */ @@ -124,61 +125,65 @@ int my_close(File fd, myf MyFlags) SYNOPSIS my_register_filename() - fd - FileName - type_file_type + fd File number opened, -1 if error on open + FileName File name + type_file_type How file was created + error_message_number Error message number if caller got error (fd == -1) + MyFlags Flags for my_close() + + RETURN + -1 error + # Filenumber + */ File my_register_filename(File fd, const char *FileName, enum file_type type_of_file, uint error_message_number, myf MyFlags) { + DBUG_ENTER("my_register_filename"); if ((int) fd >= 0) { if ((uint) fd >= my_file_limit) { #if defined(THREAD) && !defined(HAVE_PREAD) - (void) my_close(fd,MyFlags); - my_errno=EMFILE; - if (MyFlags & (MY_FFNF | MY_FAE | MY_WME)) - my_error(EE_OUT_OF_FILERESOURCES, MYF(ME_BELL+ME_WAITTANG), - FileName, my_errno); - return(-1); -#endif + my_errno= EMFILE; +#else thread_safe_increment(my_file_opened,&THR_LOCK_open); - return(fd); /* safeguard */ + DBUG_RETURN(fd); /* safeguard */ +#endif } - pthread_mutex_lock(&THR_LOCK_open); - if ((my_file_info[fd].name = (char*) my_strdup(FileName,MyFlags))) + else { - my_file_opened++; - my_file_info[fd].type = type_of_file; + pthread_mutex_lock(&THR_LOCK_open); + if ((my_file_info[fd].name = (char*) my_strdup(FileName,MyFlags))) + { + my_file_opened++; + my_file_info[fd].type = type_of_file; #if defined(THREAD) && !defined(HAVE_PREAD) - pthread_mutex_init(&my_file_info[fd].mutex,MY_MUTEX_INIT_FAST); + pthread_mutex_init(&my_file_info[fd].mutex,MY_MUTEX_INIT_FAST); #endif + pthread_mutex_unlock(&THR_LOCK_open); + DBUG_PRINT("exit",("fd: %d",fd)); + DBUG_RETURN(fd); + } pthread_mutex_unlock(&THR_LOCK_open); - DBUG_PRINT("exit",("fd: %d",fd)); - return(fd); + my_errno= ENOMEM; } - pthread_mutex_unlock(&THR_LOCK_open); (void) my_close(fd, MyFlags); - fd= -1; - my_errno=ENOMEM; } else - my_errno=errno; - DBUG_PRINT("error",("Got error %d on open",my_errno)); - if (MyFlags & (MY_FFNF | MY_FAE | MY_WME)) { - if (my_errno == EMFILE) { - DBUG_PRINT("error",("print err: %d",EE_OUT_OF_FILERESOURCES)); - my_error(EE_OUT_OF_FILERESOURCES, MYF(ME_BELL+ME_WAITTANG), - FileName, my_errno); - } else { - DBUG_PRINT("error",("print err: %d",error_message_number)); - my_error(error_message_number, MYF(ME_BELL+ME_WAITTANG), - FileName, my_errno); - } + my_errno= errno; + + DBUG_PRINT("error",("Got error %d on open", my_errno)); + if (MyFlags & (MY_FFNF | MY_FAE | MY_WME)) + { + if (my_errno == EMFILE) + error_message_number= EE_OUT_OF_FILERESOURCES; + DBUG_PRINT("error",("print err: %d",error_message_number)); + my_error(error_message_number, MYF(ME_BELL+ME_WAITTANG), + FileName, my_errno); } - return(fd); + DBUG_RETURN(-1); } #ifdef __WIN__ diff --git a/mysys/my_symlink2.c b/mysys/my_symlink2.c index 279672be11c..932f2b6424f 100644 --- a/mysys/my_symlink2.c +++ b/mysys/my_symlink2.c @@ -33,7 +33,9 @@ File my_create_with_symlink(const char *linkname, const char *filename, int create_link; char abs_linkname[FN_REFLEN]; DBUG_ENTER("my_create_with_symlink"); - DBUG_PRINT("enter", ("linkname: %s filename: %s", linkname, filename)); + DBUG_PRINT("enter", + ("linkname: %s filename: %s", linkname ? linkname : "NULL", + filename)); if (my_disable_symlinks) { diff --git a/sql-bench/example b/sql-bench/example index df2a9b8be69..cb39fad819e 100644 --- a/sql-bench/example +++ b/sql-bench/example @@ -6,15 +6,14 @@ machine="Linux-x64" # InnoDB tests -./run-all-tests --suffix=-innodb --comments="Engine=InnoDB --innodb_log_file_size=100M" --create-options="ENGINE=InnoDB" --hw="$hw" --optimization="$optimization" --machine="$machine" --log - -./run-all-tests --suffix=_fast-innodb --comments="Engine=InnoDB --innodb_log_file_size=100M" --create-options="ENGINE=InnoDB" --hw="$hw" --optimization="$optimization" --machine="$machine" --fast --log +./run-all-tests --suffix=-innodb --comments="Engine=InnoDB --innodb_buffer_pool_size=256M --innodb_additional_mem_pool_size=20M --innodb_log_file_size=1000M --innodb_log_buffer_size=16M --innodb_lock_wait_timeout=50 --innodb_flush_log_at_trx_commit=1 --innodb_flush_method=O_DIRECT --innodb_log_files_in_group=2 --skip-innodb-doblewrite" --create-options="ENGINE=InnoDB" --hw="$hw" --optimization="$optimization" --machine="$machine" --log +./run-all-tests --suffix=_fast-innodb --comments="Engine=InnoDB --innodb_buffer_pool_size=256M --innodb_additional_mem_pool_size=20M --innodb_log_file_size=1000M --innodb_log_buffer_size=16M --innodb_lock_wait_timeout=50 --innodb_flush_log_at_trx_commit=1 --innodb_flush_method=O_DIRECT --innodb_log_files_in_group=2 --skip-innodb-doblewrite" --create-options="ENGINE=InnoDB" --hw="$hw" --optimization="$optimization" --machine="$machine" --fast --log # MyISAM tests -./run-all-tests --suffix=-myisam --comments="Engine=MyISAM key_buffer_size=16M" --create-options="ENGINE=myisam" --hw="$hw" --optimization="$optimization" --machine="$machine" --log +./run-all-tests --suffix=-myisam --comments="Engine=MyISAM key_buffer_size=256M" --create-options="ENGINE=myisam" --hw="$hw" --optimization="$optimization" --machine="$machine" --log -./run-all-tests --suffix=_fast-myisam --comments="Engine=MyISAM key_buffer_size=16M" --create-options="ENGINE=myisam" --hw="$hw" --optimization="$optimization" --machine="$machine" --fast --log +./run-all-tests --suffix=_fast-myisam --comments="Engine=MyISAM key_buffer_size=256M" --create-options="ENGINE=myisam" --hw="$hw" --optimization="$optimization" --machine="$machine" --fast --log compare-results --relative output/RUN-mysql-myisam-* output/RUN-mysql_fast-myisam* output/RUN-mysql* diff --git a/sql/handler.cc b/sql/handler.cc index b32098bfc78..f3c71887e9a 100644 --- a/sql/handler.cc +++ b/sql/handler.cc @@ -68,7 +68,7 @@ static const LEX_STRING sys_table_aliases[]= }; const char *ha_row_type[] = { - "", "FIXED", "DYNAMIC", "COMPRESSED", "REDUNDANT", "COMPACT", "?","?","?" + "", "FIXED", "DYNAMIC", "COMPRESSED", "REDUNDANT", "COMPACT", "PAGE","?","?","?" }; const char *tx_isolation_names[] = @@ -281,7 +281,8 @@ handler *get_ha_partition(partition_info *part_info) 0 OK != 0 Error */ -static int ha_init_errors(void) + +int ha_init_errors(void) { #define SETMSG(nr, msg) errmsgs[(nr) - HA_ERR_FIRST]= (msg) const char **errmsgs; @@ -495,9 +496,6 @@ int ha_init() int error= 0; DBUG_ENTER("ha_init"); - if (ha_init_errors()) - DBUG_RETURN(1); - DBUG_ASSERT(total_ha < MAX_HA); /* Check if there is a transaction-capable storage engine besides the diff --git a/sql/handler.h b/sql/handler.h index d488ca5f07e..b5679147d8a 100644 --- a/sql/handler.h +++ b/sql/handler.h @@ -268,7 +268,7 @@ enum legacy_db_type enum row_type { ROW_TYPE_NOT_USED=-1, ROW_TYPE_DEFAULT, ROW_TYPE_FIXED, ROW_TYPE_DYNAMIC, ROW_TYPE_COMPRESSED, - ROW_TYPE_REDUNDANT, ROW_TYPE_COMPACT, ROW_TYPE_PAGES }; + ROW_TYPE_REDUNDANT, ROW_TYPE_COMPACT, ROW_TYPE_PAGE }; enum enum_binlog_func { BFN_RESET_LOGS= 1, @@ -311,6 +311,7 @@ enum enum_binlog_command { #define HA_CREATE_USED_PASSWORD (1L << 17) #define HA_CREATE_USED_CONNECTION (1L << 18) #define HA_CREATE_USED_KEY_BLOCK_SIZE (1L << 19) +#define HA_CREATE_USED_TRANSACTIONAL (1L << 20) typedef ulonglong my_xid; // this line is the same as in log_event.h #define MYSQL_XID_PREFIX "MySQLXid" @@ -741,6 +742,7 @@ class partition_info; struct st_partition_iter; #define NOT_A_PARTITION_ID ((uint32)-1) +enum ha_choice { HA_CHOICE_UNDEF, HA_CHOICE_NO, HA_CHOICE_YES }; typedef struct st_ha_create_information { @@ -763,6 +765,8 @@ typedef struct st_ha_create_information uint options; /* OR of HA_CREATE_ options */ uint merge_insert_method; uint extra_size; /* length of extra data segment */ + /* 0 not used, 1 if not transactional, 2 if transactional */ + enum ha_choice transactional; bool table_existed; /* 1 in create if table existed */ bool frm_only; /* 1 if no ha_create_table() */ bool varchar; /* 1 if table has a VARCHAR */ @@ -1661,6 +1665,7 @@ static inline bool ha_storage_engine_is_enabled(const handlerton *db_type) } /* basic stuff */ +int ha_init_errors(void); int ha_init(void); int ha_end(void); int ha_initialize_handlerton(st_plugin_int *plugin); diff --git a/sql/lex.h b/sql/lex.h index 45155da7692..28271bf46d9 100644 --- a/sql/lex.h +++ b/sql/lex.h @@ -379,6 +379,7 @@ static SYMBOL symbols[] = { { "OWNER", SYM(OWNER_SYM)}, { "PACK_KEYS", SYM(PACK_KEYS_SYM)}, { "PARSER", SYM(PARSER_SYM)}, + { "PAGE", SYM(PAGE_SYM)}, { "PARTIAL", SYM(PARTIAL)}, { "PARTITION", SYM(PARTITION_SYM)}, { "PARTITIONING", SYM(PARTITIONING_SYM)}, @@ -528,6 +529,7 @@ static SYMBOL symbols[] = { { "TO", SYM(TO_SYM)}, { "TRAILING", SYM(TRAILING)}, { "TRANSACTION", SYM(TRANSACTION_SYM)}, + { "TRANSACTIONAL", SYM(TRANSACTIONAL_SYM)}, { "TRIGGER", SYM(TRIGGER_SYM)}, { "TRIGGERS", SYM(TRIGGERS_SYM)}, { "TRUE", SYM(TRUE_SYM)}, diff --git a/sql/mysqld.cc b/sql/mysqld.cc index a49e4005c67..fd77317509b 100644 --- a/sql/mysqld.cc +++ b/sql/mysqld.cc @@ -3350,6 +3350,10 @@ server."); using_update_log=1; } + /* Allow storage engine to give real error messages */ + if (ha_init_errors()) + DBUG_RETURN(1); + if (plugin_init(opt_bootstrap)) { sql_print_error("Failed to init plugins."); diff --git a/sql/sql_show.cc b/sql/sql_show.cc index 250d9d917eb..9ae38d5dcec 100644 --- a/sql/sql_show.cc +++ b/sql/sql_show.cc @@ -1371,6 +1371,11 @@ int store_create_info(THD *thd, TABLE_LIST *table_list, String *packet, packet->append(STRING_WITH_LEN(" ROW_FORMAT=")); packet->append(ha_row_type[(uint) share->row_type]); } + if (share->transactional != HA_CHOICE_UNDEF) + { + packet->append(STRING_WITH_LEN(" TRANSACTIONAL=")); + packet->append(share->transactional == HA_CHOICE_YES ? "1" : "0", 1); + } if (table->s->key_block_size) { char *end; @@ -2910,8 +2915,8 @@ static int get_schema_tables_record(THD *thd, struct st_table_list *tables, case ROW_TYPE_COMPACT: tmp_buff= "Compact"; break; - case ROW_TYPE_PAGES: - tmp_buff= "Paged"; + case ROW_TYPE_PAGE: + tmp_buff= "Page"; break; } table->field[6]->store(tmp_buff, strlen(tmp_buff), cs); diff --git a/sql/sql_table.cc b/sql/sql_table.cc index 0697fdd79b4..a037fc6f727 100644 --- a/sql/sql_table.cc +++ b/sql/sql_table.cc @@ -5653,6 +5653,8 @@ view_err: create_info->default_table_charset= table->s->table_charset; if (!(used_fields & HA_CREATE_USED_KEY_BLOCK_SIZE)) create_info->key_block_size= table->s->key_block_size; + if (!(used_fields & HA_CREATE_USED_TRANSACTIONAL)) + create_info->transactional= table->s->transactional; if (!create_info->tablespace && create_info->storage_media != HA_SM_MEMORY) { @@ -6916,7 +6918,6 @@ bool mysql_recreate_table(THD *thd, TABLE_LIST *table_list, lex->col_list.empty(); lex->alter_info.reset(); bzero((char*) &create_info,sizeof(create_info)); - create_info.db_type= 0; create_info.row_type=ROW_TYPE_NOT_USED; create_info.default_table_charset=default_charset_info; /* Force alter table to recreate table */ diff --git a/sql/sql_yacc.yy b/sql/sql_yacc.yy index 9c062407921..0653863cc73 100644 --- a/sql/sql_yacc.yy +++ b/sql/sql_yacc.yy @@ -720,6 +720,7 @@ bool my_yyoverflow(short **a, YYSTYPE **b, ulong *yystacksize); %token OUT_SYM /* SQL-2003-R */ %token OWNER_SYM %token PACK_KEYS_SYM +%token PAGE_SYM %token PARAM_MARKER %token PARSER_SYM %token PARTIAL /* SQL-2003-N */ @@ -872,6 +873,7 @@ bool my_yyoverflow(short **a, YYSTYPE **b, ulong *yystacksize); %token TO_SYM /* SQL-2003-R */ %token TRAILING /* SQL-2003-R */ %token TRANSACTION_SYM +%token TRANSACTIONAL_SYM %token TRIGGERS_SYM %token TRIGGER_SYM /* SQL-2003-R */ %token TRIM /* SQL-2003-N */ @@ -4213,6 +4215,12 @@ create_table_option: Lex->create_info.used_fields|= HA_CREATE_USED_KEY_BLOCK_SIZE; Lex->create_info.key_block_size= $3; } + | TRANSACTIONAL_SYM opt_equal ulong_num + { + Lex->create_info.used_fields|= HA_CREATE_USED_TRANSACTIONAL; + Lex->create_info.transactional= ($3 != 0 ? HA_CHOICE_YES : + HA_CHOICE_NO); + } ; default_charset: @@ -4273,6 +4281,7 @@ row_types: | COMPRESSED_SYM { $$= ROW_TYPE_COMPRESSED; } | REDUNDANT_SYM { $$= ROW_TYPE_REDUNDANT; } | COMPACT_SYM { $$= ROW_TYPE_COMPACT; }; + | PAGE_SYM { $$= ROW_TYPE_PAGE; }; merge_insert_types: NO_SYM { $$= MERGE_INSERT_DISABLED; } @@ -9786,6 +9795,7 @@ keyword_sp: | ONE_SHOT_SYM {} | ONE_SYM {} | PACK_KEYS_SYM {} + | PAGE_SYM {} | PARTIAL {} | PARTITIONING_SYM {} | PARTITIONS_SYM {} @@ -9855,6 +9865,7 @@ keyword_sp: | TEXT_SYM {} | THAN_SYM {} | TRANSACTION_SYM {} + | TRANSACTIONAL_SYM {} | TRIGGERS_SYM {} | TIMESTAMP {} | TIMESTAMP_ADD {} diff --git a/sql/table.cc b/sql/table.cc index ed3cac85214..316d99a85b5 100644 --- a/sql/table.cc +++ b/sql/table.cc @@ -460,7 +460,8 @@ static int open_binary_frm(THD *thd, TABLE_SHARE *share, uchar *head, if (!head[32]) // New frm file in 3.23 { share->avg_row_length= uint4korr(head+34); - share-> row_type= (row_type) head[40]; + share->transactional= (ha_choice) head[39]; + share->row_type= (row_type) head[40]; share->table_charset= get_charset((uint) head[38],MYF(0)); share->null_field_first= 1; } @@ -2111,7 +2112,9 @@ File create_frm(THD *thd, const char *name, const char *db, int2store(fileinfo+16,reclength); int4store(fileinfo+18,create_info->max_rows); int4store(fileinfo+22,create_info->min_rows); + /* fileinfo[26] is set in mysql_create_frm() */ fileinfo[27]=2; // Use long pack-fields + /* fileinfo[28 & 29] is set to key_info_length in mysql_create_frm() */ create_info->table_options|=HA_OPTION_LONG_BLOB_PTR; // Use portable blob pointers int2store(fileinfo+30,create_info->table_options); fileinfo[32]=0; // No filename anymore @@ -2119,8 +2122,9 @@ File create_frm(THD *thd, const char *name, const char *db, int4store(fileinfo+34,create_info->avg_row_length); fileinfo[38]= (create_info->default_table_charset ? create_info->default_table_charset->number : 0); + fileinfo[39]= (uchar) create_info->transactional; fileinfo[40]= (uchar) create_info->row_type; - /* Next few bytes were for RAID support */ + /* Next few bytes where for RAID support */ fileinfo[41]= 0; fileinfo[42]= 0; fileinfo[43]= 0; diff --git a/sql/table.h b/sql/table.h index fc2f25f3aa8..fc9f1b7caa4 100644 --- a/sql/table.h +++ b/sql/table.h @@ -175,6 +175,7 @@ typedef struct st_table_share handlerton *db_type; /* table_type for handler */ enum row_type row_type; /* How rows are stored */ enum tmp_table_type tmp_table; + enum ha_choice transactional; uint ref_count; /* How many TABLE objects uses this */ uint open_count; /* Number of tables in open list */ diff --git a/storage/maria/ha_maria.cc b/storage/maria/ha_maria.cc index f7fd417836a..0c83d73c3ef 100644 --- a/storage/maria/ha_maria.cc +++ b/storage/maria/ha_maria.cc @@ -436,32 +436,38 @@ volatile int *_ma_killed_ptr(HA_CHECK *param) void _ma_check_print_error(HA_CHECK *param, const char *fmt, ...) { + va_list args; + DBUG_ENTER("_ma_check_print_error"); param->error_printed |= 1; param->out_flag |= O_DATA_LOST; - va_list args; va_start(args, fmt); _ma_check_print_msg(param, "error", fmt, args); va_end(args); + DBUG_VOID_RETURN; } void _ma_check_print_info(HA_CHECK *param, const char *fmt, ...) { va_list args; + DBUG_ENTER("_ma_check_print_info"); va_start(args, fmt); _ma_check_print_msg(param, "info", fmt, args); va_end(args); + DBUG_VOID_RETURN; } void _ma_check_print_warning(HA_CHECK *param, const char *fmt, ...) { + va_list args; + DBUG_ENTER("_ma_check_print_warning"); param->warning_printed= 1; param->out_flag |= O_DATA_LOST; - va_list args; va_start(args, fmt); _ma_check_print_msg(param, "warning", fmt, args); va_end(args); + DBUG_VOID_RETURN; } } @@ -1065,16 +1071,6 @@ int ha_maria::repair(THD *thd, HA_CHECK ¶m, bool do_optimize) param.out_flag= 0; strmov(fixed_name, file->s->open_file_name); -#ifndef TO_BE_FIXED - /* QQ: Until we have repair for block format, lie that it succeded */ - if (file->s->data_file_type == BLOCK_RECORD) - { - if (do_optimize) - DBUG_RETURN(analyze(thd, (HA_CHECK_OPT*) 0)); - DBUG_RETURN(HA_ADMIN_OK); - } -#endif - // Don't lock tables if we have used LOCK TABLE if (!thd->locked_tables && maria_lock_database(file, table->s->tmp_table ? F_EXTRA_LCK : F_WRLCK)) @@ -1099,7 +1095,9 @@ int ha_maria::repair(THD *thd, HA_CHECK ¶m, bool do_optimize) local_testflag |= T_STATISTICS; param.testflag |= T_STATISTICS; // We get this for free statistics_done= 1; - if (thd->variables.maria_repair_threads > 1) + /* TODO: Remove BLOCK_RECORD test when parallel works with blocks */ + if (thd->variables.maria_repair_threads > 1 && + file->s->data_file_type != BLOCK_RECORD) { char buf[40]; /* TODO: respect maria_repair_threads variable */ @@ -1954,7 +1952,7 @@ enum row_type ha_maria::get_row_type() const switch (file->s->data_file_type) { case STATIC_RECORD: return ROW_TYPE_FIXED; case DYNAMIC_RECORD: return ROW_TYPE_DYNAMIC; - case BLOCK_RECORD: return ROW_TYPE_PAGES; + case BLOCK_RECORD: return ROW_TYPE_PAGE; case COMPRESSED_RECORD: return ROW_TYPE_COMPRESSED; default: return ROW_TYPE_NOT_USED; } @@ -1963,6 +1961,8 @@ enum row_type ha_maria::get_row_type() const static enum data_file_type maria_row_type(HA_CREATE_INFO *info) { + if (info->transactional == HA_CHOICE_YES) + return BLOCK_RECORD; switch (info->row_type) { case ROW_TYPE_FIXED: return STATIC_RECORD; case ROW_TYPE_DYNAMIC: return DYNAMIC_RECORD; @@ -2007,7 +2007,8 @@ int ha_maria::create(const char *name, register TABLE *table_arg, share->avg_row_length); create_info.data_file_name= ha_create_info->data_file_name; create_info.index_file_name= ha_create_info->index_file_name; - create_info.transactional= row_type == BLOCK_RECORD; + create_info.transactional= (row_type == BLOCK_RECORD && + ha_create_info->transactional != HA_CHOICE_NO); if (ha_create_info->options & HA_LEX_CREATE_TMP_TABLE) create_flags|= HA_CREATE_TMP_TABLE; diff --git a/storage/maria/ma_bitmap.c b/storage/maria/ma_bitmap.c index e1308bce487..f6a8172935f 100644 --- a/storage/maria/ma_bitmap.c +++ b/storage/maria/ma_bitmap.c @@ -106,6 +106,19 @@ put on disk even if they are not in the page cache). - When explicitely requested (for example on backup or after recvoery, to simplify things) + + The flow of writing a row is that: + - Lock the bitmap + - Decide which data pages we will write to + - Mark them full in the bitmap page so that other threads do not try to + use the same data pages as us + - We unlock the bitmap + - Write the data pages + - Lock the bitmap + - Correct the bitmap page with the true final occupation of the data + pages (that is, we marked pages full but when we are done we realize + we didn't fill them) + - Unlock the bitmap. */ #include "maria_def.h" diff --git a/storage/maria/ma_blockrec.c b/storage/maria/ma_blockrec.c index c747aaeb6cb..0ed8c1f7232 100644 --- a/storage/maria/ma_blockrec.c +++ b/storage/maria/ma_blockrec.c @@ -1778,7 +1778,7 @@ static my_bool write_block_record(MARIA_HA *info, ulong length; ulong data_length= (tmp_data - info->rec_buff); -#ifdef SANITY_CHECK +#ifdef SANITY_CHECKS if (cur_block->sub_blocks == 1) goto crashed; /* no reserved full or tails */ #endif @@ -1814,8 +1814,8 @@ static my_bool write_block_record(MARIA_HA *info, FULL_PAGE_SIZE(block_size))) && cur_block->page_count) { -#ifdef SANITY_CHECK - if ((cur_block == end_block) || (cur_block->used & BLOCKUSED_BIT)) +#ifdef SANITY_CHECKS + if ((cur_block == end_block) || (cur_block->used & BLOCKUSED_USED)) goto crashed; #endif data_length-= length; @@ -1829,7 +1829,7 @@ static my_bool write_block_record(MARIA_HA *info, /* Skip empty filler block */ cur_block++; } -#ifdef SANITY_CHECK +#ifdef SANITY_CHECKS if ((cur_block >= end_block)) goto crashed; #endif @@ -2548,11 +2548,6 @@ static my_bool delete_head_or_tail(MARIA_HA *info, PAGECACHE_PIN_LEFT_PINNED, PAGECACHE_WRITE_DELAY, &page_link.link)) DBUG_RETURN(1); - - /* Change the lock used when we read the page */ - page_link.unlock= PAGECACHE_LOCK_READ_UNLOCK; - set_dynamic(&info->pinned_pages, (void*) &page_link, - info->pinned_pages.elements-1); } else { @@ -2564,7 +2559,7 @@ static my_bool delete_head_or_tail(MARIA_HA *info, pagerange_store(log_data + FILEID_STORE_SIZE, 1); page_store(log_data+ FILEID_STORE_SIZE + PAGERANGE_STORE_SIZE, page); pagerange_store(log_data + FILEID_STORE_SIZE + PAGERANGE_STORE_SIZE + - PAGERANGE_STORE_SIZE, 1); + PAGE_STORE_SIZE, 1); log_array[TRANSLOG_INTERNAL_PARTS + 0].str= (char*) log_data; log_array[TRANSLOG_INTERNAL_PARTS + 0].length= sizeof(log_data); if (translog_write_record(!info->trn->rec_lsn ? &info->trn->rec_lsn : &lsn, @@ -2573,8 +2568,24 @@ static my_bool delete_head_or_tail(MARIA_HA *info, sizeof(log_data), TRANSLOG_INTERNAL_PARTS + 1, log_array)) DBUG_RETURN(1); + + /* Write the empty page (needed only for REPAIR to work) */ + buff[PAGE_TYPE_OFFSET]= UNALLOCATED_PAGE; + if (pagecache_write(share->pagecache, + &info->dfile, page, 0, + buff, share->page_type, + PAGECACHE_LOCK_WRITE_TO_READ, + PAGECACHE_PIN_LEFT_PINNED, + PAGECACHE_WRITE_DELAY, &page_link.link)) + DBUG_RETURN(1); + DBUG_ASSERT(empty_space >= info->s->bitmap.sizes[0]); } + /* Change the lock used when we read the page */ + page_link.unlock= PAGECACHE_LOCK_READ_UNLOCK; + set_dynamic(&info->pinned_pages, (void*) &page_link, + info->pinned_pages.elements-1); + DBUG_PRINT("info", ("empty_space: %u", empty_space)); DBUG_RETURN(_ma_bitmap_set(info, page, head, empty_space)); } @@ -2794,7 +2805,8 @@ static byte *read_next_extent(MARIA_HA *info, MARIA_EXTENT_CURSOR *extent, extent->extent+= ROW_EXTENT_SIZE; extent->page= uint5korr(extent->extent); page_count= uint2korr(extent->extent+ROW_EXTENT_PAGE_SIZE); - DBUG_ASSERT(page_count != 0); + if (!page_count) + goto crashed; extent->tail= page_count & TAIL_BIT; extent->page_count= (page_count & ~TAIL_BIT); extent->first_extent= 0; @@ -2817,7 +2829,8 @@ static byte *read_next_extent(MARIA_HA *info, MARIA_EXTENT_CURSOR *extent, if (!extent->tail) { /* Full data page */ - DBUG_ASSERT((buff[PAGE_TYPE_OFFSET] & PAGE_TYPE_MASK) == BLOB_PAGE); + if ((buff[PAGE_TYPE_OFFSET] & PAGE_TYPE_MASK) != BLOB_PAGE) + goto crashed; extent->page++; /* point to next page */ extent->page_count--; *end_of_data= buff + share->block_size; @@ -2826,7 +2839,8 @@ static byte *read_next_extent(MARIA_HA *info, MARIA_EXTENT_CURSOR *extent, } /* Found tail. page_count is in this case the position in the tail page */ - DBUG_ASSERT((buff[PAGE_TYPE_OFFSET] & PAGE_TYPE_MASK) == TAIL_PAGE); + if ((buff[PAGE_TYPE_OFFSET] & PAGE_TYPE_MASK) != TAIL_PAGE) + goto crashed; *(extent->tail_positions++)= ma_recordpos(extent->page, extent->page_count); info->cur_row.tail_count++; /* For maria_chk */ @@ -2948,7 +2962,6 @@ int _ma_read_block_record2(MARIA_HA *info, byte *record, MARIA_COLUMNDEF *column, *end_column; DBUG_ENTER("_ma_read_block_record2"); - LINT_INIT(field_lengths); LINT_INIT(field_length_data); LINT_INIT(blob_buffer); @@ -2994,6 +3007,7 @@ int _ma_read_block_record2(MARIA_HA *info, byte *record, } extent.first_extent= 1; + field_lengths= 0; if (share->base.max_field_lengths) { get_key_length(field_lengths, data); @@ -3028,7 +3042,7 @@ int _ma_read_block_record2(MARIA_HA *info, byte *record, Read row extents (note that first extent was already read into info->cur_row.extents above) */ - if (row_extents) + if (row_extents > 1) { if (read_long_data(info, info->cur_row.extents + ROW_EXTENT_SIZE, (row_extents - 1) * ROW_EXTENT_SIZE, @@ -3053,7 +3067,7 @@ int _ma_read_block_record2(MARIA_HA *info, byte *record, } /* Read array of field lengths. This may be stored in several extents */ - if (share->base.max_field_lengths) + if (field_lengths) { field_length_data= info->cur_row.field_lengths; if (read_long_data(info, field_length_data, field_lengths, &extent, @@ -3459,6 +3473,8 @@ restart_bitmap_scan: DBUG_PRINT("error", ("Wrong page header")); DBUG_RETURN((my_errno= HA_ERR_WRONG_IN_RECORD)); } + DBUG_PRINT("info", ("Page %lu has %u rows", + (ulong) page, info->scan.number_of_rows)); info->scan.dir= (info->scan.page_buff + block_size - PAGE_SUFFIX_SIZE - DIR_ENTRY_SIZE); info->scan.dir_end= (info->scan.dir - @@ -3471,7 +3487,8 @@ restart_bitmap_scan: for (data+= 6; data < info->scan.bitmap_end; data+= 6) { bits= uint6korr(data); - if (bits && ((bits & LL(04444444444444444)) != LL(04444444444444444))) + /* Skip not allocated pages and blob / full tail pages */ + if (bits && bits != LL(07777777777777777)) break; } bit_pos= 0; @@ -3483,8 +3500,11 @@ restart_bitmap_scan: filepos= (my_off_t) info->scan.bitmap_page * block_size; if (unlikely(filepos >= info->state->data_file_length)) { + DBUG_PRINT("info", ("Found end of file")); DBUG_RETURN((my_errno= HA_ERR_END_OF_FILE)); } + DBUG_PRINT("info", ("Reading bitmap at %lu", + (ulong) info->scan.bitmap_page)); if (!(pagecache_read(share->pagecache, &info->dfile, info->scan.bitmap_page, 0, info->scan.bitmap_buff, PAGECACHE_PLAIN_PAGE, diff --git a/storage/maria/ma_check.c b/storage/maria/ma_check.c index 8f10c98d0ee..9d017bc6ad5 100644 --- a/storage/maria/ma_check.c +++ b/storage/maria/ma_check.c @@ -42,7 +42,6 @@ #include "ma_ftdefs.h" #include -#include #include #include #ifdef HAVE_SYS_VADVISE_H @@ -86,6 +85,12 @@ static SORT_KEY_BLOCKS *alloc_key_blocks(HA_CHECK *param, uint blocks, static ha_checksum maria_byte_checksum(const byte *buf, uint length); static void set_data_file_type(MARIA_SORT_INFO *sort_info, MARIA_SHARE *share); static void restore_data_file_type(MARIA_SHARE *share); +static void change_data_file_descriptor(MARIA_HA *info, File new_file); +static int _ma_safe_scan_block_record(MARIA_SORT_INFO *sort_info, + MARIA_HA *info, byte *record); +static void copy_data_file_state(MARIA_STATE_INFO *to, + MARIA_STATE_INFO *from); + void maria_chk_init(HA_CHECK *param) { @@ -837,7 +842,7 @@ static int chk_index(HA_CHECK *param, MARIA_HA *info, MARIA_KEYDEF *keyinfo, } } (*key_checksum)+= maria_byte_checksum((byte*) key, - key_length- info->s->rec_reflength); + key_length- info->s->rec_reflength); record= _ma_dpos(info,0,key+key_length); if (keyinfo->flag & HA_FULLTEXT) /* special handling for ft2 */ { @@ -1262,18 +1267,21 @@ static int check_dynamic_record(HA_CHECK *param, MARIA_HA *info, int extend, } else { - info->cur_row.checksum= _ma_checksum(info,record); + ha_checksum checksum= 0; + if (info->s->calc_checksum) + checksum= (*info->s->calc_checksum)(info, record); + if (param->testflag & (T_EXTEND | T_MEDIUM | T_VERBOSE)) { if (_ma_rec_check(info,record, info->rec_buff,block_info.rec_len, - test(info->s->calc_checksum))) + test(info->s->calc_checksum), checksum)) { _ma_check_print_error(param,"Found wrong packed record at %s", llstr(start_recpos,llbuff)); got_error= 1; } } - param->glob_crc+= info->cur_row.checksum; + param->glob_crc+= checksum; } if (! got_error) @@ -1506,8 +1514,11 @@ static my_bool check_head_page(HA_CHECK *param, MARIA_HA *info, byte *record, } if (info->s->calc_checksum) { - info->cur_row.checksum= _ma_checksum(info, record); - param->glob_crc+= info->cur_row.checksum; + ha_checksum checksum= (*info->s->calc_checksum)(info, record); + if (info->cur_row.checksum != (checksum & 255)) + _ma_check_print_error(param, "Page %9s: Row %3d has wrong checksum", + llstr(page_pos, llbuff), row); + param->glob_crc+= checksum; } if (info->cur_row.extents_count) { @@ -1571,6 +1582,8 @@ static int check_block_record(HA_CHECK *param, MARIA_HA *info, int extend, my_bool full_dir; uint offset_page, offset; + LINT_INIT(full_dir); + if (_ma_scan_init_block_record(info)) { _ma_check_print_error(param, "got error %d when initializing scan", @@ -1648,13 +1661,12 @@ static int check_block_record(HA_CHECK *param, MARIA_HA *info, int extend, llstr(pos, llbuff), page_type); if (param->err_count++ > MAXERR || !(param->testflag & T_VERBOSE)) goto err; + continue; } switch ((enum en_page_type) page_type) { case UNALLOCATED_PAGE: case MAX_PAGE_TYPE: - DBUG_PRINT("warning", - ("Found page with wrong page type: %d", page_type)); - DBUG_ASSERT(0); + DBUG_ASSERT(0); /* Impossible */ break; case HEAD_PAGE: row_count= ((uchar*) page_buff)[DIR_COUNT_OFFSET]; @@ -1907,13 +1919,28 @@ int maria_chk_data_link(HA_CHECK *param, MARIA_HA *info,int extend) } /* maria_chk_data_link */ - /* Recover old table by reading each record and writing all keys */ - /* Save new datafile-name in temp_filename */ +/* + Recover old table by reading each record and writing all keys + + NOTES + Save new datafile-name in temp_filename + + IMPLEMENTATION (for hard repair with block format) + - Create new, unrelated MARIA_HA of the table + - Create new datafile and associate it with new handler + - Reset all statistic information in new handler + - Copy all data to new handler with normal write operations + - Move state of new handler to old handler + - Close new handler + - Close data file in old handler + - Rename old data file to new data file. + - Reopen data file in old handler +*/ int maria_repair(HA_CHECK *param, register MARIA_HA *info, my_string name, int rep_quick) { - int error,got_error; + int error, got_error= 1; uint i; ha_rows start_records,new_header_length; my_off_t del; @@ -1922,6 +1949,8 @@ int maria_repair(HA_CHECK *param, register MARIA_HA *info, char llbuff[22],llbuff2[22]; MARIA_SORT_INFO sort_info; MARIA_SORT_PARAM sort_param; + my_bool block_record, scan_inited= 0; + enum data_file_type org_data_file_type= info->s->data_file_type; DBUG_ENTER("maria_repair"); bzero((char *)&sort_info, sizeof(sort_info)); @@ -1929,9 +1958,11 @@ int maria_repair(HA_CHECK *param, register MARIA_HA *info, start_records=info->state->records; new_header_length=(param->testflag & T_UNPACK) ? 0L : share->pack.header_length; - got_error=1; new_file= -1; sort_param.sort_info=&sort_info; + block_record= org_data_file_type == BLOCK_RECORD; + sort_info.info= sort_info.new_info= info; + bzero(&info->rec_cache,sizeof(info->rec_cache)); if (!(param->testflag & T_SILENT)) { @@ -1943,28 +1974,6 @@ int maria_repair(HA_CHECK *param, register MARIA_HA *info, if (info->s->options & (HA_OPTION_CHECKSUM | HA_OPTION_COMPRESS_RECORD)) param->testflag|=T_CALC_CHECKSUM; - if (init_io_cache(¶m->read_cache, info->dfile.file, - (uint) param->read_buffer_length, - READ_CACHE,share->pack.header_length,1,MYF(MY_WME))) - { - bzero(&info->rec_cache,sizeof(info->rec_cache)); - goto err; - } - if (!rep_quick) - if (init_io_cache(&info->rec_cache,-1,(uint) param->write_buffer_length, - WRITE_CACHE, new_header_length, 1, - MYF(MY_WME | MY_WAIT_IF_FULL))) - goto err; - info->opt_flag|=WRITE_CACHE_USED; - if (!(sort_param.record=(byte*) my_malloc((uint) share->base.pack_reclength, - MYF(0))) || - _ma_alloc_buffer(&sort_param.rec_buff, &sort_param.rec_buff_size, - info->s->base.default_rec_buff_size)) - { - _ma_check_print_error(param, "Not enough memory for extra record"); - goto err; - } - if (!rep_quick) { /* Get real path for data file */ @@ -1983,11 +1992,71 @@ int maria_repair(HA_CHECK *param, register MARIA_HA *info, new_header_length, "datafile-header")) goto err; info->s->state.dellink= HA_OFFSET_ERROR; - info->rec_cache.file=new_file; - if (param->testflag & T_UNPACK) - restore_data_file_type(share); + info->rec_cache.file= new_file; + if (share->data_file_type == BLOCK_RECORD || + ((param->testflag & T_UNPACK) && + share->state.header.org_data_file_type == BLOCK_RECORD)) + { + MARIA_HA *new_info; + if (!(sort_info.new_info= maria_open(info->s->unique_file_name, O_RDWR, + HA_OPEN_COPY | HA_OPEN_FOR_REPAIR))) + goto err; + new_info= sort_info.new_info; + change_data_file_descriptor(new_info, new_file); + maria_lock_database(new_info, F_EXTRA_LCK); + if ((param->testflag & T_UNPACK) && + share->data_file_type == COMPRESSED_RECORD) + { + (*new_info->s->once_end)(new_info->s); + (*new_info->s->end)(new_info); + restore_data_file_type(new_info->s); + _ma_setup_functions(new_info->s); + if ((*new_info->s->once_init)(new_info->s, new_file) || + (*new_info->s->init)(new_info)) + goto err; + } + _ma_reset_status(sort_info.new_info); + if (_ma_initialize_data_file(sort_info.new_info->s, new_file)) + goto err; + block_record= 1; + } + } + + if (org_data_file_type != BLOCK_RECORD) + { + /* We need a read buffer to read rows in big blocks */ + if (init_io_cache(¶m->read_cache, info->dfile.file, + (uint) param->read_buffer_length, + READ_CACHE, share->pack.header_length, 1, MYF(MY_WME))) + goto err; } - sort_info.info=info; + if (sort_info.new_info->s->data_file_type != BLOCK_RECORD) + { + /* When writing to not block records, we need a write buffer */ + if (!rep_quick) + if (init_io_cache(&info->rec_cache, new_file, + (uint) param->write_buffer_length, + WRITE_CACHE, new_header_length, 1, + MYF(MY_WME | MY_WAIT_IF_FULL))) + goto err; + info->opt_flag|=WRITE_CACHE_USED; + } + else + { + scan_inited= 1; + if (maria_scan_init(sort_info.info)) + goto err; + } + + if (!(sort_param.record=(byte*) my_malloc((uint) share->base.pack_reclength, + MYF(0))) || + _ma_alloc_buffer(&sort_param.rec_buff, &sort_param.rec_buff_size, + info->s->base.default_rec_buff_size)) + { + _ma_check_print_error(param, "Not enough memory for extra record"); + goto err; + } + sort_info.param = param; sort_param.read_cache=param->read_cache; sort_param.pos=sort_param.max_pos=share->pack.header_length; @@ -2030,9 +2099,14 @@ int maria_repair(HA_CHECK *param, register MARIA_HA *info, maria_lock_memory(param); /* Everything is alloced */ + sort_info.org_data_file_type= info->s->data_file_type; + /* Re-create all keys, which are set in key_map. */ while (!(error=sort_get_next_record(&sort_param))) { + if (block_record && _ma_sort_write_record(&sort_param)) + goto err; + if (writekeys(&sort_param)) { if (my_errno != HA_ERR_FOUND_DUPP_KEY) @@ -2058,7 +2132,8 @@ int maria_repair(HA_CHECK *param, register MARIA_HA *info, } continue; } - if (_ma_sort_write_record(&sort_param)) + + if (!block_record && _ma_sort_write_record(&sort_param)) goto err; } if (error > 0 || maria_write_data_suffix(&sort_info, (my_bool)!rep_quick) || @@ -2081,35 +2156,58 @@ int maria_repair(HA_CHECK *param, register MARIA_HA *info, { _ma_check_print_error(param,"Couldn't fix table with quick recovery: Found wrong number of deleted records"); _ma_check_print_error(param,"Run recovery again without -q"); - got_error=1; param->retry_repair=1; param->testflag|=T_RETRY_WITHOUT_QUICK; goto err; } + if (param->testflag & T_SAFE_REPAIR) { /* Don't repair if we loosed more than one row */ - if (info->state->records+1 < start_records) + if (sort_info.new_info->state->records+1 < start_records) { info->state->records=start_records; - got_error=1; goto err; } } if (!rep_quick) { - my_close(info->dfile.file, MYF(0)); - info->dfile.file= new_file; - info->state->data_file_length=sort_param.filepos; + if (sort_info.new_info != sort_info.info) + { + MARIA_STATE_INFO save_state= sort_info.new_info->s->state; + if (maria_close(sort_info.new_info)) + { + _ma_check_print_error(param, "Got error %d on close", my_errno); + goto err; + } + copy_data_file_state(&info->s->state, &save_state); + new_file= -1; + } + else + info->state->data_file_length= sort_param.filepos; share->state.version=(ulong) time((time_t*) 0); /* Force reopen */ + + /* Replace the actual file with the temporary file */ + if (new_file >= 0) + my_close(new_file, MYF(MY_WME)); + my_close(info->dfile.file, MYF(MY_WME)); + info->dfile.file= new_file= -1; + if (maria_change_to_newfile(share->data_file_name,MARIA_NAME_DEXT, + DATA_TMP_EXT, + (param->testflag & T_BACKUP_DATA ? + MYF(MY_REDEL_MAKE_BACKUP): MYF(0))) || + _ma_open_datafile(info, share, -1)) + { + goto err; + } } else { - info->state->data_file_length=sort_param.max_pos; + info->state->data_file_length= sort_param.max_pos; } if (param->testflag & T_CALC_CHECKSUM) - info->state->checksum=param->glob_crc; + info->state->checksum= param->glob_crc; if (!(param->testflag & T_SILENT)) { @@ -2127,25 +2225,19 @@ int maria_repair(HA_CHECK *param, register MARIA_HA *info, memcpy( &share->state.state, info->state, sizeof(*info->state)); err: - if (!got_error) - { - /* Replace the actual file with the temporary file */ - if (new_file >= 0) - { - my_close(new_file,MYF(0)); - info->dfile.file= new_file= -1; - if (maria_change_to_newfile(share->data_file_name,MARIA_NAME_DEXT, - DATA_TMP_EXT, (param->testflag & T_BACKUP_DATA ? - MYF(MY_REDEL_MAKE_BACKUP): MYF(0))) || - _ma_open_datafile(info,share,-1)) - got_error=1; - } - } + if (scan_inited) + maria_scan_end(sort_info.info); + if (got_error) { if (! param->error_printed) _ma_check_print_error(param,"%d for record at pos %s",my_errno, llstr(sort_param.start_recpos,llbuff)); + if (sort_info.new_info && sort_info.new_info != sort_info.info) + { + sort_info.new_info->dfile.file= -1; + maria_close(sort_info.new_info); + } if (new_file >= 0) { VOID(my_close(new_file,MYF(0))); @@ -2595,7 +2687,7 @@ int maria_repair_by_sort(HA_CHECK *param, register MARIA_HA *info, uint i; ulong length; ha_rows start_records; - my_off_t new_header_length,del; + my_off_t new_header_length, org_header_length, del; File new_file; MARIA_SORT_PARAM sort_param; MARIA_SHARE *share=info->s; @@ -2606,11 +2698,15 @@ int maria_repair_by_sort(HA_CHECK *param, register MARIA_HA *info, ulonglong key_map=share->state.key_map; DBUG_ENTER("maria_repair_by_sort"); + bzero((char*)&sort_info,sizeof(sort_info)); + bzero((char *)&sort_param, sizeof(sort_param)); + start_records=info->state->records; got_error=1; new_file= -1; - new_header_length=(param->testflag & T_UNPACK) ? 0 : - share->pack.header_length; + org_header_length= share->pack.header_length; + new_header_length= (param->testflag & T_UNPACK) ? 0 : org_header_length; + if (!(param->testflag & T_SILENT)) { printf("- recovering (with sort) MARIA-table '%s'\n",name); @@ -2621,15 +2717,13 @@ int maria_repair_by_sort(HA_CHECK *param, register MARIA_HA *info, if (info->s->options & (HA_OPTION_CHECKSUM | HA_OPTION_COMPRESS_RECORD)) param->testflag|=T_CALC_CHECKSUM; - bzero((char*)&sort_info,sizeof(sort_info)); - bzero((char *)&sort_param, sizeof(sort_param)); if (!(sort_info.key_block= alloc_key_blocks(param, (uint) param->sort_key_blocks, - share->base.max_key_block_length)) - || init_io_cache(¶m->read_cache, info->dfile.file, - (uint) param->read_buffer_length, - READ_CACHE,share->pack.header_length,1,MYF(MY_WME)) || + share->base.max_key_block_length)) || + init_io_cache(¶m->read_cache, info->dfile.file, + (uint) param->read_buffer_length, + READ_CACHE, org_header_length, 1, MYF(MY_WME)) || (! rep_quick && init_io_cache(&info->rec_cache, info->dfile.file, (uint) param->write_buffer_length, @@ -2639,6 +2733,7 @@ int maria_repair_by_sort(HA_CHECK *param, register MARIA_HA *info, sort_info.key_block_end=sort_info.key_block+param->sort_key_blocks; info->opt_flag|=WRITE_CACHE_USED; info->rec_cache.file= info->dfile.file; /* for sort_delete_record */ + sort_info.org_data_file_type= info->s->data_file_type; if (!(sort_param.record=(byte*) my_malloc((uint) share->base.pack_reclength, MYF(0))) || @@ -2694,8 +2789,8 @@ int maria_repair_by_sort(HA_CHECK *param, register MARIA_HA *info, key_map= ~key_map; /* Create the missing keys */ } - sort_info.info=info; - sort_info.param = param; + sort_info.info= sort_info.new_info= info; + sort_info.param= param; set_data_file_type(&sort_info, share); sort_param.filepos=new_header_length; @@ -2707,9 +2802,9 @@ int maria_repair_by_sort(HA_CHECK *param, register MARIA_HA *info, sort_param.wordlist=NULL; init_alloc_root(&sort_param.wordroot, FTPARSER_MEMROOT_ALLOC_SIZE, 0); - if (share->data_file_type == DYNAMIC_RECORD) + if (sort_info.org_data_file_type == DYNAMIC_RECORD) length=max(share->base.min_pack_length+1,share->base.min_block_length); - else if (share->data_file_type == COMPRESSED_RECORD) + else if (sort_info.org_data_file_type == COMPRESSED_RECORD) length=share->base.min_block_length; else length=share->base.pack_reclength; @@ -2747,7 +2842,7 @@ int maria_repair_by_sort(HA_CHECK *param, register MARIA_HA *info, if ((!(param->testflag & T_SILENT))) printf ("- Fixing index %d\n",sort_param.key+1); - sort_param.max_pos=sort_param.pos=share->pack.header_length; + sort_param.max_pos= sort_param.pos= org_header_length; keyseg=sort_param.seg; bzero((char*) sort_param.unique,sizeof(sort_param.unique)); sort_param.key_length=share->rec_reflength; @@ -2845,8 +2940,9 @@ int maria_repair_by_sort(HA_CHECK *param, register MARIA_HA *info, share->state.version=(ulong) time((time_t*) 0); my_close(info->dfile.file, MYF(0)); info->dfile.file= new_file; - share->data_file_type=sort_info.new_data_file_type; - share->pack.header_length=(ulong) new_header_length; + share->data_file_type= sort_info.new_data_file_type; + org_header_length= (ulong) new_header_length; + sort_info.org_data_file_type= info->s->data_file_type; sort_param.fix_datafile=0; } else @@ -2874,11 +2970,11 @@ int maria_repair_by_sort(HA_CHECK *param, register MARIA_HA *info, if (rep_quick & T_FORCE_UNIQUENESS) { - my_off_t skr=info->state->data_file_length+ - (share->options & HA_OPTION_COMPRESS_RECORD ? - MEMMAP_EXTRA_MARGIN : 0); + my_off_t skr= (info->state->data_file_length + + (sort_info.org_data_file_type == COMPRESSED_RECORD) ? + MEMMAP_EXTRA_MARGIN : 0); #ifdef USE_RELOC - if (share->data_file_type == STATIC_RECORD && + if (sort_info.org_data_file_type == STATIC_RECORD && skr < share->base.reloc*share->base.min_pack_length) skr=share->base.reloc*share->base.min_pack_length; #endif @@ -3073,6 +3169,8 @@ int maria_repair_parallel(HA_CHECK *param, register MARIA_HA *info, pthread_mutex_init(&sort_info.mutex, MY_MUTEX_INIT_FAST); pthread_cond_init(&sort_info.cond, 0); + sort_info.org_data_file_type= info->s->data_file_type; + if (!(sort_info.key_block= alloc_key_blocks(param, (uint) param->sort_key_blocks, share->base.max_key_block_length)) || @@ -3140,8 +3238,8 @@ int maria_repair_parallel(HA_CHECK *param, register MARIA_HA *info, key_map= ~key_map; /* Create the missing keys */ } - sort_info.info=info; - sort_info.param = param; + sort_info.info= sort_info.new_info= info; + sort_info.param= param; set_data_file_type(&sort_info, share); sort_info.dupp=0; @@ -3149,9 +3247,9 @@ int maria_repair_parallel(HA_CHECK *param, register MARIA_HA *info, param->read_cache.end_of_file=sort_info.filelength= my_seek(param->read_cache.file,0L,MY_SEEK_END,MYF(0)); - if (share->data_file_type == DYNAMIC_RECORD) + if (sort_info.org_data_file_type == DYNAMIC_RECORD) rec_length=max(share->base.min_pack_length+1,share->base.min_block_length); - else if (share->data_file_type == COMPRESSED_RECORD) + else if (sort_info.org_data_file_type == COMPRESSED_RECORD) rec_length=share->base.min_block_length; else rec_length=share->base.pack_reclength; @@ -3367,8 +3465,6 @@ int maria_repair_parallel(HA_CHECK *param, register MARIA_HA *info, */ my_close(info->dfile.file, MYF(0)); info->dfile.file= new_file; - - share->data_file_type=sort_info.new_data_file_type; share->pack.header_length=(ulong) new_header_length; } else @@ -3385,11 +3481,11 @@ int maria_repair_parallel(HA_CHECK *param, register MARIA_HA *info, if (rep_quick & T_FORCE_UNIQUENESS) { - my_off_t skr=info->state->data_file_length+ - (share->options & HA_OPTION_COMPRESS_RECORD ? - MEMMAP_EXTRA_MARGIN : 0); + my_off_t skr= (info->state->data_file_length + + (sort_info.org_data_file_type == COMPRESSED_RECORD) ? + MEMMAP_EXTRA_MARGIN : 0); #ifdef USE_RELOC - if (share->data_file_type == STATIC_RECORD && + if (sort_info.org_data_file_type == STATIC_RECORD && skr < share->base.reloc*share->base.min_pack_length) skr=share->base.reloc*share->base.min_pack_length; #endif @@ -3574,27 +3670,28 @@ static int sort_maria_ft_key_read(MARIA_SORT_PARAM *sort_param, byte *key) sort_get_next_record() sort_param Information about and for the sort process - NOTE - + NOTES Dynamic Records With Non-Quick Parallel Repair - For non-quick parallel repair we use a synchronized read/write - cache. This means that one thread is the master who fixes the data - file by reading each record from the old data file and writing it - to the new data file. By doing this the records in the new data - file are written contiguously. Whenever the write buffer is full, - it is copied to the read buffer. The slaves read from the read - buffer, which is not associated with a file. Thus read_cache.file - is -1. When using _mi_read_cache(), the slaves must always set - flag to READING_NEXT so that the function never tries to read from - file. This is safe because the records are contiguous. There is no - need to read outside the cache. This condition is evaluated in the - variable 'parallel_flag' for quick reference. read_cache.file must - be >= 0 in every other case. + For non-quick parallel repair we use a synchronized read/write + cache. This means that one thread is the master who fixes the data + file by reading each record from the old data file and writing it + to the new data file. By doing this the records in the new data + file are written contiguously. Whenever the write buffer is full, + it is copied to the read buffer. The slaves read from the read + buffer, which is not associated with a file. Thus read_cache.file + is -1. When using _mi_read_cache(), the slaves must always set + flag to READING_NEXT so that the function never tries to read from + file. This is safe because the records are contiguous. There is no + need to read outside the cache. This condition is evaluated in the + variable 'parallel_flag' for quick reference. read_cache.file must + be >= 0 in every other case. RETURN -1 end of file 0 ok + sort_param->filepos points to record position. + sort_param->record contains record > 0 error */ @@ -3615,10 +3712,61 @@ static int sort_get_next_record(MARIA_SORT_PARAM *sort_param) if (*_ma_killed_ptr(param)) DBUG_RETURN(1); - switch (share->data_file_type) { + switch (sort_info->org_data_file_type) { case BLOCK_RECORD: - DBUG_ASSERT(0); + { + for (;;) + { + int flag; + + if (info != sort_info->new_info) + { + /* Safe scanning */ + flag= _ma_safe_scan_block_record(sort_info, info, + sort_param->record); + } + else + { + /* Scan on clean table */ + flag= _ma_scan_block_record(info, sort_param->record, + info->cur_row.nextpos, 1); + } + if (!flag) + { + if (sort_param->calc_checksum) + { + ha_checksum checksum; + checksum= (*info->s->calc_check_checksum)(info, sort_param->record); + if (info->s->calc_checksum && + info->cur_row.checksum != (checksum & 255)) + { + if (param->testflag & T_VERBOSE) + { + char llbuff[22]; + record_pos_to_txt(info, sort_param->filepos, llbuff); + _ma_check_print_info(param, + "Found record with wrong checksum at %s", + llbuff); + } + continue; + } + info->cur_row.checksum= checksum; + param->glob_crc+= checksum; + } + sort_param->filepos= info->cur_row.lastpos; + DBUG_RETURN(0); + } + if (flag == HA_ERR_END_OF_FILE) + { + sort_param->max_pos= sort_info->filelength; + DBUG_RETURN(-1); + } + /* Retry only if wrong record, not if disk error */ + if (flag != HA_ERR_WRONG_IN_RECORD) + DBUG_RETURN(flag); + } break; + } case STATIC_RECORD: for (;;) { @@ -3656,6 +3804,8 @@ static int sort_get_next_record(MARIA_SORT_PARAM *sort_param) { byte *to; LINT_INIT(to); + ha_checksum checksum= 0; + pos=sort_param->pos; searching=(sort_param->fix_datafile && (param->testflag & T_EXTEND)); parallel_flag= (sort_param->read_cache.file < 0) ? READING_NEXT : 0; @@ -3925,14 +4075,14 @@ static int sort_get_next_record(MARIA_SORT_PARAM *sort_param) if (sort_param->read_cache.error < 0) DBUG_RETURN(1); if (sort_param->calc_checksum) - info->cur_row.checksum= _ma_checksum(info, sort_param->record); + checksum= (info->s->calc_check_checksum)(info, sort_param->record); if ((param->testflag & (T_EXTEND | T_REP)) || searching) { if (_ma_rec_check(info, sort_param->record, sort_param->rec_buff, sort_param->find_length, (param->testflag & T_QUICK) && sort_param->calc_checksum && - test(info->s->calc_checksum))) + test(info->s->calc_checksum), checksum)) { _ma_check_print_info(param,"Found wrong packed record at %s", llstr(sort_param->start_recpos,llbuff)); @@ -3940,7 +4090,7 @@ static int sort_get_next_record(MARIA_SORT_PARAM *sort_param) } } if (sort_param->calc_checksum) - param->glob_crc+= info->cur_row.checksum; + param->glob_crc+= checksum; DBUG_RETURN(0); } if (!searching) @@ -4014,8 +4164,9 @@ static int sort_get_next_record(MARIA_SORT_PARAM *sort_param) if (sort_param->calc_checksum) { - info->cur_row.checksum= (*info->s->calc_checksum)(info, - sort_param->record); + info->cur_row.checksum= (*info->s->calc_check_checksum)(info, + sort_param-> + record); param->glob_crc+= info->cur_row.checksum; } DBUG_RETURN(0); @@ -4048,8 +4199,8 @@ int _ma_sort_write_record(MARIA_SORT_PARAM *sort_param) byte *from; byte block_buff[8]; MARIA_SORT_INFO *sort_info=sort_param->sort_info; - HA_CHECK *param=sort_info->param; - MARIA_HA *info=sort_info->info; + HA_CHECK *param= sort_info->param; + MARIA_HA *info= sort_info->new_info; MARIA_SHARE *share=info->s; DBUG_ENTER("_ma_sort_write_record"); @@ -4057,7 +4208,11 @@ int _ma_sort_write_record(MARIA_SORT_PARAM *sort_param) { switch (sort_info->new_data_file_type) { case BLOCK_RECORD: - DBUG_ASSERT(0); + if ((sort_param->filepos= (*share->write_record_init)(info, + sort_param-> + record)) == + HA_OFFSET_ERROR) + DBUG_RETURN(1); break; case STATIC_RECORD: if (my_b_write(&info->rec_cache,sort_param->record, @@ -4090,7 +4245,9 @@ int _ma_sort_write_record(MARIA_SORT_PARAM *sort_param) from=sort_info->buff+ALIGN_SIZE(MARIA_MAX_DYN_BLOCK_HEADER); } /* We can use info->checksum here as only one thread calls this */ - info->cur_row.checksum= _ma_checksum(info,sort_param->record); + info->cur_row.checksum= (*info->s->calc_check_checksum)(info, + sort_param-> + record); reclength= _ma_rec_pack(info,from,sort_param->record); flag=0; @@ -4147,7 +4304,7 @@ int _ma_sort_write_record(MARIA_SORT_PARAM *sort_param) } /* _ma_sort_write_record */ - /* Compare two keys from _ma_create_index_by_sort */ +/* Compare two keys from _ma_create_index_by_sort */ static int sort_key_cmp(MARIA_SORT_PARAM *sort_param, const void *a, const void *b) @@ -4505,7 +4662,8 @@ static int sort_delete_record(MARIA_SORT_PARAM *sort_param) } } if (sort_param->calc_checksum) - param->glob_crc-=(*info->s->calc_checksum)(info, sort_param->record); + param->glob_crc-=(*info->s->calc_check_checksum)(info, + sort_param->record); } error= (flush_io_cache(&info->rec_cache) || (*info->s->delete_record)(info, sort_param->record)); @@ -4514,7 +4672,8 @@ static int sort_delete_record(MARIA_SORT_PARAM *sort_param) DBUG_RETURN(error); } /* sort_delete_record */ - /* Fix all pending blocks and flush everything to disk */ + +/* Fix all pending blocks and flush everything to disk */ int _ma_flush_pending_blocks(MARIA_SORT_PARAM *sort_param) { @@ -4786,9 +4945,9 @@ end: int maria_write_data_suffix(MARIA_SORT_INFO *sort_info, my_bool fix_datafile) { - MARIA_HA *info=sort_info->info; + MARIA_HA *info=sort_info->new_info; - if (info->s->options & HA_OPTION_COMPRESS_RECORD && fix_datafile) + if (info->s->data_file_type == COMPRESSED_RECORD && fix_datafile) { char buff[MEMMAP_EXTRA_MARGIN]; bzero(buff,sizeof(buff)); @@ -5101,6 +5260,9 @@ my_bool maria_test_if_sort_rep(MARIA_HA *info, ha_rows rows, */ if (! maria_is_any_key_active(key_map)) return FALSE; /* Can't use sort */ + /* QQ: Remove this when maria_repair_by_sort() works with block format */ + if (info->s->data_file_type == BLOCK_RECORD) + return FALSE; for (i=0 ; i < share->base.keys ; i++,key++) { if (!force && maria_too_big_key_for_sort(key,rows)) @@ -5119,7 +5281,8 @@ set_data_file_type(MARIA_SORT_INFO *sort_info, MARIA_SHARE *share) MARIA_SHARE tmp; sort_info->new_data_file_type= share->state.header.org_data_file_type; /* Set delete_function for sort_delete_record() */ - memcpy((char*) &tmp, share, sizeof(*share)); + tmp= *share; + tmp.state.header.data_file_type= tmp.state.header.org_data_file_type; tmp.options= ~HA_OPTION_COMPRESS_RECORD; _ma_setup_functions(&tmp); share->delete_record=tmp.delete_record; @@ -5132,6 +5295,161 @@ static void restore_data_file_type(MARIA_SHARE *share) mi_int2store(share->state.header.options,share->options); share->state.header.data_file_type= share->state.header.org_data_file_type; - share->data_file_type= share->state.header.data_file_type= + share->data_file_type= share->state.header.data_file_type; share->pack.header_length= 0; } + + +static void change_data_file_descriptor(MARIA_HA *info, File new_file) +{ + my_close(info->dfile.file, MYF(0)); + info->dfile.file= info->s->bitmap.file.file= new_file; +} + + +/* + Copy all states that has to do with the data file + + NOTES + This is done to copy the state from the data file generated from + repair to the original handler +*/ + +static void copy_data_file_state(MARIA_STATE_INFO *to, + MARIA_STATE_INFO *from) +{ + to->state.records= from->state.records; + to->state.del= from->state.del; + to->state.empty= from->state.empty; + to->state.data_file_length= from->state.data_file_length; + to->split= from->split; + to->dellink= from->dellink; + to->first_bitmap_with_space= from->first_bitmap_with_space; +} + + +/* + Read 'safely' next record while scanning table. + + SYNOPSIS + _ma_safe_scan_block_record() + info Maria handler + record Store found here + + NOTES + - One must have called mi_scan() before this + + Differences compared to _ma_scan_block_records() are: + - We read all blocks, not only blocks marked by the bitmap to be safe + - In case of errors, next read will read next record. + - More sanity checks + + RETURN + 0 ok + HA_ERR_END_OF_FILE End of file + # error number +*/ + + +static int _ma_safe_scan_block_record(MARIA_SORT_INFO *sort_info, + MARIA_HA *info, byte *record) +{ + uint record_pos= info->cur_row.nextpos; + ulonglong page= sort_info->page; + DBUG_ENTER("_ma_safe_scan_block_record"); + + for (;;) + { + /* Find next row in current page */ + if (likely(record_pos < info->scan.number_of_rows)) + { + uint length, offset; + byte *data, *end_of_data; + char llbuff[22]; + + while (!(offset= uint2korr(info->scan.dir))) + { + info->scan.dir-= DIR_ENTRY_SIZE; + record_pos++; + if (info->scan.dir < info->scan.dir_end) + { + _ma_check_print_info(sort_info->param, + "Wrong directory on page: %s", + llstr(page, llbuff)); + goto read_next_page; + } + } + /* found row */ + info->cur_row.lastpos= info->scan.row_base_page + record_pos; + info->cur_row.nextpos= record_pos + 1; + data= info->scan.page_buff + offset; + length= uint2korr(info->scan.dir + 2); + end_of_data= data + length; + info->scan.dir-= DIR_ENTRY_SIZE; /* Point to previous row */ + + if (end_of_data > info->scan.dir_end || + offset < PAGE_HEADER_SIZE || length < info->s->base.min_block_length) + { + _ma_check_print_info(sort_info->param, + "Wrong directory entry %3u at page %s", + record_pos, llstr(page, llbuff)); + record_pos++; + continue; + } + else + { + DBUG_PRINT("info", ("rowid: %lu", (ulong) info->cur_row.lastpos)); + DBUG_RETURN(_ma_read_block_record2(info, record, data, end_of_data)); + } + } + +read_next_page: + /* Read until we find next head page */ + for (;;) + { + uint page_type; + char llbuff[22]; + + sort_info->page++; /* In case of errors */ + page++; + if (!(page % info->s->bitmap.pages_covered)) + page++; /* Skip bitmap */ + if ((page + 1) * info->s->block_size > sort_info->filelength) + DBUG_RETURN(HA_ERR_END_OF_FILE); + if (!(pagecache_read(info->s->pagecache, + &info->dfile, + page, 0, info->scan.page_buff, + PAGECACHE_READ_UNKNOWN_PAGE, + PAGECACHE_LOCK_LEFT_UNLOCKED, 0))) + DBUG_RETURN(my_errno); + + page_type= (info->scan.page_buff[PAGE_TYPE_OFFSET] & + PAGE_TYPE_MASK); + if (page_type == HEAD_PAGE) + { + if ((info->scan.number_of_rows= + (uint) (uchar) info->scan.page_buff[DIR_COUNT_OFFSET]) != 0) + break; + _ma_check_print_info(sort_info->param, + "Wrong head page at %s", + llstr(page * info->s->block_size, llbuff)); + } + else if (page_type >= MAX_PAGE_TYPE) + { + _ma_check_print_info(sort_info->param, + "Found wrong page type: %d at %s", + page_type, llstr(page * info->s->block_size, + llbuff)); + } + } + + /* New head page */ + info->scan.dir= (info->scan.page_buff + info->s->block_size - + PAGE_SUFFIX_SIZE - DIR_ENTRY_SIZE); + info->scan.dir_end= (info->scan.dir - + (info->scan.number_of_rows - 1) * + DIR_ENTRY_SIZE); + info->scan.row_base_page= ma_recordpos(page, 0); + record_pos= 0; + } +} diff --git a/storage/maria/ma_close.c b/storage/maria/ma_close.c index dc60ce8aa83..e73629c3c87 100644 --- a/storage/maria/ma_close.c +++ b/storage/maria/ma_close.c @@ -124,8 +124,6 @@ int maria_close(register MARIA_HA *info) my_free((gptr) info,MYF(0)); if (error) - { - DBUG_RETURN(my_errno=error); - } + DBUG_RETURN(my_errno= error); DBUG_RETURN(0); } /* maria_close */ diff --git a/storage/maria/ma_create.c b/storage/maria/ma_create.c index d8660dd41cb..280321d40ec 100644 --- a/storage/maria/ma_create.c +++ b/storage/maria/ma_create.c @@ -798,7 +798,7 @@ int maria_create(const char *name, enum data_file_type datafile_type, goto err; errpos=3; - if (_ma_initialize_data_file(dfile, &share)) + if (_ma_initialize_data_file(&share, dfile)) goto err; } DBUG_PRINT("info", ("write state info and base info")); @@ -1082,7 +1082,7 @@ static int compare_columns(MARIA_COLUMNDEF **a_ptr, MARIA_COLUMNDEF **b_ptr) /* Initialize data file */ -int _ma_initialize_data_file(File dfile, MARIA_SHARE *share) +int _ma_initialize_data_file(MARIA_SHARE *share, File dfile) { if (share->data_file_type == BLOCK_RECORD) { diff --git a/storage/maria/ma_delete_all.c b/storage/maria/ma_delete_all.c index 2d85b347662..b18b1105391 100644 --- a/storage/maria/ma_delete_all.c +++ b/storage/maria/ma_delete_all.c @@ -20,9 +20,7 @@ int maria_delete_all_rows(MARIA_HA *info) { - uint i; MARIA_SHARE *share=info->s; - MARIA_STATE_INFO *state=&share->state; DBUG_ENTER("maria_delete_all_rows"); if (share->options & HA_OPTION_READ_ONLY_DATA) @@ -35,18 +33,7 @@ int maria_delete_all_rows(MARIA_HA *info) if (_ma_mark_file_changed(info)) goto err; - info->state->records=info->state->del=state->split=0; - state->changed= 0; /* File is optimized */ - state->dellink = HA_OFFSET_ERROR; - state->sortkey= (ushort) ~0; - info->state->key_file_length=share->base.keystart; - info->state->data_file_length=0; - info->state->empty=info->state->key_empty=0; - info->state->checksum=0; - - state->key_del= HA_OFFSET_ERROR; - for (i=0 ; i < share->base.keys ; i++) - state->key_root[i]= HA_OFFSET_ERROR; + _ma_reset_status(info); /* If we are using delayed keys or if the user has done changes to the tables @@ -67,7 +54,7 @@ int maria_delete_all_rows(MARIA_HA *info) my_chsize(share->kfile.file, share->base.keystart, 0, MYF(MY_WME)) ) goto err; - if (_ma_initialize_data_file(info->dfile.file, info->s)) + if (_ma_initialize_data_file(info->s, info->dfile.file)) goto err; /* @@ -104,4 +91,39 @@ err: allow_break(); /* Allow SIGHUP & SIGINT */ DBUG_RETURN(my_errno=save_errno); } -} /* maria_delete */ +} /* maria_delete_all_rows */ + + +/* + Reset status information + + SYNOPSIS + _ma_reset_status() + maria Maria handler + + DESCRIPTION + Resets data and index file information as if the file would be empty + Files are not touched. +*/ + +void _ma_reset_status(MARIA_HA *info) +{ + MARIA_SHARE *share= info->s; + MARIA_STATE_INFO *state= &share->state; + uint i; + + info->state->records= info->state->del= state->split= 0; + state->changed= 0; /* File is optimized */ + state->dellink= HA_OFFSET_ERROR; + state->sortkey= (ushort) ~0; + info->state->key_file_length= share->base.keystart; + info->state->data_file_length= 0; + info->state->empty= info->state->key_empty= 0; + info->state->checksum= 0; + + /* Drop the delete key chain. */ + state->key_del= HA_OFFSET_ERROR; + /* Clear all keys */ + for (i=0 ; i < share->base.keys ; i++) + state->key_root[i]= HA_OFFSET_ERROR; +} diff --git a/storage/maria/ma_dynrec.c b/storage/maria/ma_dynrec.c index ebf84032106..9281378fd33 100644 --- a/storage/maria/ma_dynrec.c +++ b/storage/maria/ma_dynrec.c @@ -1018,7 +1018,8 @@ uint _ma_rec_pack(MARIA_HA *info, register byte *to, register const byte *from) */ my_bool _ma_rec_check(MARIA_HA *info,const char *record, byte *rec_buff, - ulong packed_length, my_bool with_checksum) + ulong packed_length, my_bool with_checksum, + ha_checksum checksum) { uint length,new_length,flag,bit,i; char *pos,*end,*packpos,*to; @@ -1124,7 +1125,7 @@ my_bool _ma_rec_check(MARIA_HA *info,const char *record, byte *rec_buff, if (packed_length != (uint) (to - rec_buff) + test(info->s->calc_checksum) || (bit != 1 && (flag & ~(bit - 1)))) goto err; - if (with_checksum && ((uchar) info->cur_row.checksum != (uchar) *to)) + if (with_checksum && ((uchar) checksum != (uchar) *to)) { DBUG_PRINT("error",("wrong checksum for row")); goto err; diff --git a/storage/maria/ma_info.c b/storage/maria/ma_info.c index a04fba4e0d8..cfb4580a72f 100644 --- a/storage/maria/ma_info.c +++ b/storage/maria/ma_info.c @@ -135,6 +135,7 @@ void _ma_report_error(int errcode, const char *file_name) file_name+= length - 64; } } + my_error(errcode, MYF(ME_NOREFRESH), file_name); DBUG_VOID_RETURN; } diff --git a/storage/maria/ma_init.c b/storage/maria/ma_init.c index 19b835a837f..ab62d1bfaa0 100644 --- a/storage/maria/ma_init.c +++ b/storage/maria/ma_init.c @@ -45,6 +45,7 @@ int maria_init(void) pthread_mutex_init(&THR_LOCK_maria,MY_MUTEX_INIT_SLOW); _ma_init_block_record_data(); loghandler_init(); + my_handler_error_register(); } return 0; } diff --git a/storage/maria/ma_loghandler.c b/storage/maria/ma_loghandler.c index f398ec90897..b029297d2d0 100644 --- a/storage/maria/ma_loghandler.c +++ b/storage/maria/ma_loghandler.c @@ -4143,12 +4143,12 @@ my_bool translog_write_record(LSN *lsn, { uint i; uint len= 0; -#ifdef HAVE_PURIFY +#ifdef HAVE_purify ha_checksum checksum= 0; #endif for (i= TRANSLOG_INTERNAL_PARTS; i < part_no; i++) { -#ifdef HAVE_PURIFY +#ifdef HAVE_purify /* Find unitialized bytes early */ checksum+= my_checksum(checksum, parts_data[i].str, parts_data[i].length); diff --git a/storage/maria/ma_open.c b/storage/maria/ma_open.c index b8ce6d123e7..79ff25e3c2f 100644 --- a/storage/maria/ma_open.c +++ b/storage/maria/ma_open.c @@ -260,7 +260,9 @@ MARIA_HA *maria_open(const char *name, int mode, uint open_flags) my_realpath(name_buff, fn_format(org_name,name,"",MARIA_NAME_IEXT, MY_UNPACK_FILENAME),MYF(0)); pthread_mutex_lock(&THR_LOCK_maria); - if (!(old_info=_ma_test_if_reopen(name_buff))) + old_info= 0; + if ((open_flags & HA_OPEN_COPY) || + !(old_info=_ma_test_if_reopen(name_buff))) { share= &share_buff; bzero((gptr) &share_buff,sizeof(share_buff)); @@ -586,6 +588,8 @@ MARIA_HA *maria_open(const char *name, int mode, uint open_flags) share->base.null_bytes + share->base.pack_bytes + test(share->options & HA_OPTION_CHECKSUM)); + if (open_flags & HA_OPEN_COPY) + share->base.transactional= 0; /* Repair: no logging */ if (share->base.transactional) share->base_length+= TRANS_ROW_EXTRA_HEADER_SIZE; share->base.default_rec_buff_size= max(share->base.pack_reclength, @@ -858,6 +862,8 @@ void _ma_setup_functions(register MARIA_SHARE *share) } share->file_read= _ma_nommap_pread; share->file_write= _ma_nommap_pwrite; + share->calc_check_checksum= share->calc_checksum; + if (!(share->options & HA_OPTION_CHECKSUM) && share->data_file_type != COMPRESSED_RECORD) share->calc_checksum= share->calc_write_checksum= 0; diff --git a/storage/maria/ma_pagecache.c b/storage/maria/ma_pagecache.c index 53a24e36861..ca47230cfbd 100755 --- a/storage/maria/ma_pagecache.c +++ b/storage/maria/ma_pagecache.c @@ -315,7 +315,8 @@ struct st_pagecache_block_link #ifndef DBUG_OFF /* debug checks */ static my_bool info_check_pin(PAGECACHE_BLOCK_LINK *block, - enum pagecache_page_pin mode) + enum pagecache_page_pin mode + __attribute__((unused))) { struct st_my_thread_var *thread= my_thread_var; PAGECACHE_PIN_INFO *info= info_find(block->pin_list, thread); @@ -373,6 +374,7 @@ static my_bool info_check_pin(PAGECACHE_BLOCK_LINK *block, 1 - Error */ +#ifdef NOT_USED static my_bool info_check_lock(PAGECACHE_BLOCK_LINK *block, enum pagecache_page_lock lock, enum pagecache_page_pin pin) @@ -440,7 +442,8 @@ error: page_cache_page_pin_str[pin])); DBUG_RETURN(1); } -#endif +#endif /* NOT_USED */ +#endif /* !DBUG_OFF */ #define FLUSH_CACHE 2000 /* sort this many blocks at once */ @@ -2858,8 +2861,10 @@ restart: (pin == PAGECACHE_PIN)), &page_st); DBUG_ASSERT(block->type == PAGECACHE_EMPTY_PAGE || - block->type == type); - block->type= type; + block->type == type || type == PAGECACHE_READ_UNKNOWN_PAGE); + if (type != PAGECACHE_READ_UNKNOWN_PAGE || + block->type == PAGECACHE_EMPTY_PAGE) + block->type= type; if (((block->status & PCBLOCK_ERROR) == 0) && (page_st != PAGE_READ)) { DBUG_PRINT("info", ("read block 0x%lx", (ulong)block)); @@ -3223,6 +3228,7 @@ restart: } DBUG_ASSERT(block->type == PAGECACHE_EMPTY_PAGE || + block->type == PAGECACHE_READ_UNKNOWN_PAGE || block->type == type); block->type= type; diff --git a/storage/maria/ma_test_all.sh b/storage/maria/ma_test_all.sh index 8ee326a9c69..a7c8df827ff 100755 --- a/storage/maria/ma_test_all.sh +++ b/storage/maria/ma_test_all.sh @@ -8,6 +8,9 @@ silent="-s" suffix="" #set -x -v -e +# Delete temporary files +rm -f *.TMD + run_tests() { row_type=$1 @@ -120,6 +123,11 @@ run_repair_tests() ./maria_chk$suffix -se test1 ./maria_chk$suffix -rqos --correct-checksum test1 ./maria_chk$suffix -se test1 + ./ma_test2$suffix $silent -c -d1 $row_type + ./maria_chk$suffix -s --parallel-recover test2 + ./maria_chk$suffix -se test2 + ./maria_chk$suffix -s --parallel-recover --quick test2 + ./maria_chk$suffix -se test2 } run_pack_tests() @@ -147,6 +155,15 @@ run_pack_tests() ./maria_chk$suffix -es test1 ./maria_chk$suffix -rus test1 ./maria_chk$suffix -es test1 + + ./ma_test2$suffix $silent -c -d1 $row_type + ./maria_chk$suffix -s --parallel-recover test2 + ./maria_chk$suffix -se test2 + ./maria_chk$suffix -s --parallel-recover --unpack test2 + ./maria_chk$suffix -se test2 + ./maria_pack$suffix --force -s test1 + ./maria_chk$suffix -s --parallel-recover --unpack test2 + ./maria_chk$suffix -se test2 } echo "Running tests with dynamic row format" @@ -161,9 +178,13 @@ run_pack_tests -S echo "Running tests with block row format" run_tests -M +run_repair_tests -M +run_pack_tests -M echo "Running tests with block row format and transactions" run_tests "-M -T" +run_repair_tests "-M -T" +run_pack_tests "-M -T" # # Tests that gives warnings diff --git a/storage/maria/ma_update.c b/storage/maria/ma_update.c index 737c7c909b4..913959717fc 100644 --- a/storage/maria/ma_update.c +++ b/storage/maria/ma_update.c @@ -147,6 +147,7 @@ int maria_update(register MARIA_HA *info, const byte *oldrec, byte *newrec) if (share->calc_checksum) { info->cur_row.checksum= (*share->calc_checksum)(info,newrec); + info->state->checksum+= (info->cur_row.checksum - old_checksum); /* Store new checksum in index file header */ key_changed|= HA_STATE_CHANGED; } @@ -173,8 +174,6 @@ int maria_update(register MARIA_HA *info, const byte *oldrec, byte *newrec) if (auto_key_changed) set_if_bigger(info->s->state.auto_increment, ma_retrieve_auto_increment(info, newrec)); - if (share->calc_checksum) - info->state->checksum+= (info->cur_row.checksum - old_checksum); /* We can't yet have HA_STATE_AKTIV here, as block_record dosn't support diff --git a/storage/maria/maria_chk.c b/storage/maria/maria_chk.c index 0b82a71f736..e7e0f5d40b5 100644 --- a/storage/maria/maria_chk.c +++ b/storage/maria/maria_chk.c @@ -676,14 +676,7 @@ get_one_option(int optid, check_param.testflag|= T_UPDATE_STATE; break; case '#': - if (argument == disabled_my_option) - { - DBUG_POP(); - } - else - { - DBUG_PUSH(argument ? argument : "d:t:o,/tmp/maria_chk.trace"); - } + DBUG_SET_INITIAL(argument ? argument : "d:t:o,/tmp/maria_chk.trace"); break; case 'V': print_version(); @@ -862,16 +855,25 @@ static int maria_chk(HA_CHECK *param, my_string filename) share->r_locks=0; maria_block_size= share->base.block_size; - if (share->data_file_type == BLOCK_RECORD && - (param->testflag & (T_REP_ANY | T_SORT_RECORDS | T_FAST | T_STATISTICS | - T_CHECK | T_CHECK_ONLY_CHANGED))) + if (share->data_file_type == BLOCK_RECORD || + ((param->testflag & T_UNPACK) && + share->state.header.org_data_file_type == BLOCK_RECORD)) { - _ma_check_print_error(param, - "Record format used by '%s' is is not yet supported with repair/check", - filename); - param->error_printed= 0; - error= 1; - goto end2; + if (param->testflag & T_SORT_RECORDS) + { + _ma_check_print_error(param, + "Record format used by '%s' is is not yet supported with repair/check", + filename); + param->error_printed= 0; + error= 1; + goto end2; + } + /* We can't do parallell repair with BLOCK_RECORD yet */ + if (param->testflag & (T_REP_BY_SORT | T_REP_PARALLEL)) + { + param->testflag&= ~(T_REP_BY_SORT | T_REP_PARALLEL); + param->testflag|= T_REP; + } } /* @@ -1757,11 +1759,14 @@ void _ma_check_print_info(HA_CHECK *param __attribute__((unused)), const char *fmt,...) { va_list args; + DBUG_ENTER("_ma_check_print_info"); + DBUG_PRINT("enter", ("format: %s", fmt)); va_start(args,fmt); VOID(vfprintf(stdout, fmt, args)); VOID(fputc('\n',stdout)); va_end(args); + DBUG_VOID_RETURN; } /* VARARGS */ @@ -1770,6 +1775,7 @@ void _ma_check_print_warning(HA_CHECK *param, const char *fmt,...) { va_list args; DBUG_ENTER("_ma_check_print_warning"); + DBUG_PRINT("enter", ("format: %s", fmt)); fflush(stdout); if (!param->warning_printed && !param->error_printed) @@ -1795,7 +1801,7 @@ void _ma_check_print_error(HA_CHECK *param, const char *fmt,...) { va_list args; DBUG_ENTER("_ma_check_print_error"); - DBUG_PRINT("enter",("format: %s",fmt)); + DBUG_PRINT("enter", ("format: %s", fmt)); fflush(stdout); if (!param->warning_printed && !param->error_printed) diff --git a/storage/maria/maria_def.h b/storage/maria/maria_def.h index d9e31e800c4..bd48a5288d5 100644 --- a/storage/maria/maria_def.h +++ b/storage/maria/maria_def.h @@ -30,6 +30,7 @@ #define MAX_NONMAPPED_INSERTS 1000 #define MARIA_MAX_TREE_LEVELS 32 +#define SANITY_CHECKS struct st_transaction; @@ -261,7 +262,9 @@ typedef struct st_maria_share Calculate checksum for a row during write. May be 0 if we calculate the checksum in write_record_init() */ - ha_checksum(*calc_write_checksum) (struct st_maria_info *, const byte *); + ha_checksum(*calc_write_checksum)(struct st_maria_info *, const byte *); + /* calculate checksum for a row during check table */ + ha_checksum(*calc_check_checksum)(struct st_maria_info *, const byte *); /* Compare a row in memory with a row on disk */ my_bool (*compare_unique)(struct st_maria_info *, MARIA_UNIQUEDEF *, const byte *record, MARIA_RECORD_POS pos); @@ -746,7 +749,7 @@ extern ulong _ma_rec_unpack(MARIA_HA *info, byte *to, byte *from, ulong reclength); extern my_bool _ma_rec_check(MARIA_HA *info, const char *record, byte *packpos, ulong packed_length, - my_bool with_checkum); + my_bool with_checkum, ha_checksum checksum); extern int _ma_write_part_record(MARIA_HA *info, my_off_t filepos, ulong length, my_off_t next_filepos, byte ** record, ulong *reclength, @@ -871,6 +874,7 @@ void _ma_update_status(void *param); void _ma_restore_status(void *param); void _ma_copy_status(void *to, void *from); my_bool _ma_check_status(void *param); +void _ma_reset_status(MARIA_HA *maria); extern MARIA_HA *_ma_test_if_reopen(char *filename); my_bool _ma_check_table_is_closed(const char *name, const char *where); @@ -904,9 +908,8 @@ int _ma_sort_write_record(MARIA_SORT_PARAM *sort_param); int _ma_create_index_by_sort(MARIA_SORT_PARAM *info, my_bool no_messages, ulong); int _ma_sync_table_files(const MARIA_HA *info); -int _ma_initialize_data_file(File dfile, MARIA_SHARE *share); +int _ma_initialize_data_file(MARIA_SHARE *share, File dfile); void _ma_unpin_all_pages(MARIA_HA *info, LSN undo_lsn); extern PAGECACHE *maria_log_pagecache; - diff --git a/storage/myisam/ft_stopwords.c b/storage/myisam/ft_stopwords.c index 1b6cff5e903..b95e0f4d857 100644 --- a/storage/myisam/ft_stopwords.c +++ b/storage/myisam/ft_stopwords.c @@ -51,10 +51,11 @@ static int ft_add_stopword(const char *w) int ft_init_stopwords() { + DBUG_ENTER("ft_init_stopwords"); if (!stopwords3) { if (!(stopwords3=(TREE *)my_malloc(sizeof(TREE),MYF(0)))) - return -1; + DBUG_RETURN(-1); init_tree(stopwords3,0,0,sizeof(FT_STOPWORD),(qsort_cmp2)&FT_STOPWORD_cmp, 0, (ft_stopword_file ? (tree_element_free)&FT_STOPWORD_free : 0), @@ -70,10 +71,10 @@ int ft_init_stopwords() int error=-1; if (!*ft_stopword_file) - return 0; + DBUG_RETURN(0); if ((fd=my_open(ft_stopword_file, O_RDONLY, MYF(MY_WME))) == -1) - return -1; + DBUG_RETURN(-1); len=(uint)my_seek(fd, 0L, MY_SEEK_END, MYF(0)); my_seek(fd, 0L, MY_SEEK_SET, MYF(0)); if (!(start=buffer=my_malloc(len+1, MYF(MY_WME)))) @@ -90,7 +91,7 @@ err1: my_free(buffer, MYF(0)); err0: my_close(fd, MYF(MY_WME)); - return error; + DBUG_RETURN(error); } else { @@ -100,13 +101,14 @@ err0: for (;*sws;sws++) { if (ft_add_stopword(*sws)) - return -1; + DBUG_RETURN(-1); } ft_stopword_file="(built-in)"; /* for SHOW VARIABLES */ } - return 0; + DBUG_RETURN(0); } + int is_stopword(char *word, uint len) { FT_STOPWORD sw; @@ -118,6 +120,8 @@ int is_stopword(char *word, uint len) void ft_free_stopwords() { + DBUG_ENTER("ft_free_stopwords"); + if (stopwords3) { delete_tree(stopwords3); /* purecov: inspected */ @@ -125,4 +129,5 @@ void ft_free_stopwords() stopwords3=0; } ft_stopword_file= 0; + DBUG_VOID_RETURN; } -- cgit v1.2.1 From ef7a757b7c09f65207e6f30619a32533c27f400f Mon Sep 17 00:00:00 2001 From: unknown Date: Sun, 1 Jul 2007 20:45:01 +0300 Subject: After merge fixes BitKeeper/etc/ignore: added storage/maria/maria_read_log support-files/compiler_warnings.supp: Ignore function used when debugging (can be called from gdb) --- .bzrignore | 1 + storage/maria/ma_delete_all.c | 1 + storage/maria/maria_read_log.c | 2 +- support-files/compiler_warnings.supp | 5 +++++ 4 files changed, 8 insertions(+), 1 deletion(-) diff --git a/.bzrignore b/.bzrignore index e4b60deec56..cb4056778cf 100644 --- a/.bzrignore +++ b/.bzrignore @@ -3001,3 +3001,4 @@ storage/maria/unittest/ma_pagecache_single_64k-t-big storage/maria/maria_control storage/maria/maria_log.* storage/maria/unittest/ma_test_loghandler_long-t-big +storage/maria/maria_read_log diff --git a/storage/maria/ma_delete_all.c b/storage/maria/ma_delete_all.c index 846006787fe..c3bdcdf365c 100644 --- a/storage/maria/ma_delete_all.c +++ b/storage/maria/ma_delete_all.c @@ -142,6 +142,7 @@ void _ma_reset_status(MARIA_HA *info) info->state->data_file_length= 0; info->state->empty= info->state->key_empty= 0; info->state->checksum= 0; + share->state.create_rename_lsn= LSN_IMPOSSIBLE; /* Drop the delete key chain. */ state->key_del= HA_OFFSET_ERROR; diff --git a/storage/maria/maria_read_log.c b/storage/maria/maria_read_log.c index 568814f6f8a..5b2d5b057c2 100644 --- a/storage/maria/maria_read_log.c +++ b/storage/maria/maria_read_log.c @@ -521,7 +521,7 @@ prototype_exec_hook(REDO_CREATE_TABLE) data file does not preclude this). */ if (((info= maria_open(name, O_RDONLY, 0)) == NULL) || - _ma_initialize_data_file(dfile, info->s)) + _ma_initialize_data_file(info->s, dfile)) { fprintf(stderr, "Failed to open new table or write to data file\n"); goto err; diff --git a/support-files/compiler_warnings.supp b/support-files/compiler_warnings.supp index babc482976d..1d73e7a55cc 100644 --- a/support-files/compiler_warnings.supp +++ b/support-files/compiler_warnings.supp @@ -54,6 +54,11 @@ db_vrfy.c : .*comparison is always false due to limited range of data type.* .*/ndb/.* : .*unused variable.* .*/ndb/.* : .*defined but not used.* +# +# Maria warning that is ok in debug builds +# +storage/maria/ma_pagecache.c: .*'info_check_pin' defined but not used + # # Unexplanable (?) stuff # -- cgit v1.2.1 From 631ecaabea7336a8f28367c0d1c291f0433f7e88 Mon Sep 17 00:00:00 2001 From: unknown Date: Mon, 2 Jul 2007 20:45:15 +0300 Subject: Merged with mysql-5.1 main tree. BUILD/compile-pentium-debug-max: Added definition after macro was removed from main tree. This will be fixed back in main tree later. --- BUILD/compile-pentium-debug-max | 2 +- include/lf.h | 2 +- include/maria.h | 74 +++--- include/my_sys.h | 8 +- include/myisam.h | 2 +- include/myisamchk.h | 10 +- mysys/lf_hash.c | 28 +-- mysys/my_compress.c | 3 +- mysys/my_rename.c | 5 +- mysys/my_safehash.c | 30 +-- mysys/my_safehash.h | 18 +- mysys/my_sync.c | 3 +- sql/filesort.cc | 6 +- sql/item_xmlfunc.cc | 2 +- sql/mysqld.cc | 22 +- sql/set_var.cc | 14 +- sql/set_var.h | 2 +- sql/uniques.cc | 2 +- storage/maria/ft_maria.c | 4 +- storage/maria/ha_maria.cc | 98 ++++---- storage/maria/ha_maria.h | 34 +-- storage/maria/ma_bitmap.c | 12 +- storage/maria/ma_blockrec.c | 234 +++++++++--------- storage/maria/ma_blockrec.h | 26 +- storage/maria/ma_cache.c | 4 +- storage/maria/ma_check.c | 222 ++++++++--------- storage/maria/ma_checkpoint.c | 2 +- storage/maria/ma_checksum.c | 6 +- storage/maria/ma_close.c | 6 +- storage/maria/ma_create.c | 18 +- storage/maria/ma_dbug.c | 6 +- storage/maria/ma_delete.c | 104 ++++---- storage/maria/ma_dynrec.c | 183 +++++++------- storage/maria/ma_extra.c | 8 +- storage/maria/ma_ft_boolean_search.c | 35 +-- storage/maria/ma_ft_nlq_search.c | 12 +- storage/maria/ma_ft_parser.c | 26 +- storage/maria/ma_ft_update.c | 34 +-- storage/maria/ma_ftdefs.h | 28 +-- storage/maria/ma_fulltext.h | 8 +- storage/maria/ma_key.c | 42 ++-- storage/maria/ma_loghandler.c | 156 ++++++------ storage/maria/ma_loghandler.h | 8 +- storage/maria/ma_loghandler_lsn.h | 2 +- storage/maria/ma_open.c | 34 +-- storage/maria/ma_packrec.c | 230 +++++++++--------- storage/maria/ma_page.c | 14 +- storage/maria/ma_pagecache.c | 54 ++--- storage/maria/ma_pagecache.h | 22 +- storage/maria/ma_pagecaches.c | 10 +- storage/maria/ma_preload.c | 6 +- storage/maria/ma_range.c | 22 +- storage/maria/ma_rfirst.c | 2 +- storage/maria/ma_rkey.c | 4 +- storage/maria/ma_rlast.c | 2 +- storage/maria/ma_rnext.c | 2 +- storage/maria/ma_rnext_same.c | 2 +- storage/maria/ma_rprev.c | 2 +- storage/maria/ma_rrnd.c | 2 +- storage/maria/ma_rsame.c | 2 +- storage/maria/ma_rsamepos.c | 2 +- storage/maria/ma_rt_index.c | 94 +++---- storage/maria/ma_rt_index.h | 14 +- storage/maria/ma_rt_key.c | 10 +- storage/maria/ma_rt_key.h | 8 +- storage/maria/ma_rt_mbr.c | 24 +- storage/maria/ma_rt_mbr.h | 18 +- storage/maria/ma_rt_split.c | 16 +- storage/maria/ma_scan.c | 2 +- storage/maria/ma_search.c | 162 ++++++------- storage/maria/ma_sort.c | 132 +++++----- storage/maria/ma_sp_defs.h | 4 +- storage/maria/ma_sp_key.c | 16 +- storage/maria/ma_static.c | 2 +- storage/maria/ma_statrec.c | 34 +-- storage/maria/ma_test1.c | 50 ++-- storage/maria/ma_test2.c | 8 +- storage/maria/ma_test3.c | 4 +- storage/maria/ma_unique.c | 18 +- storage/maria/ma_update.c | 4 +- storage/maria/ma_write.c | 158 ++++++------ storage/maria/maria_chk.c | 98 ++++---- storage/maria/maria_def.h | 270 ++++++++++----------- storage/maria/maria_ftdump.c | 2 +- storage/maria/maria_pack.c | 162 ++++++------- storage/maria/tablockman.c | 8 +- storage/maria/trnman.c | 4 +- storage/maria/trnman.h | 2 +- storage/maria/unittest/ma_pagecache_consist.c | 6 +- storage/maria/unittest/ma_pagecache_single.c | 2 +- storage/maria/unittest/ma_test_loghandler-t.c | 12 +- .../unittest/ma_test_loghandler_multigroup-t.c | 14 +- .../unittest/ma_test_loghandler_multithread-t.c | 12 +- .../unittest/ma_test_loghandler_pagecache-t.c | 2 +- storage/myisam/ft_myisam.c | 4 +- storage/myisam/ha_myisam.cc | 9 +- storage/myisam/myisamdef.h | 7 +- storage/myisam/sort.c | 8 +- storage/myisammrg/ha_myisammrg.cc | 20 +- 99 files changed, 1692 insertions(+), 1691 deletions(-) diff --git a/BUILD/compile-pentium-debug-max b/BUILD/compile-pentium-debug-max index 941a63a209f..39d12a8320a 100755 --- a/BUILD/compile-pentium-debug-max +++ b/BUILD/compile-pentium-debug-max @@ -4,7 +4,7 @@ path=`dirname $0` set -- "$@" --with-debug=full . "$path/SETUP.sh" -extra_flags="$pentium_cflags $debug_cflags" +extra_flags="$pentium_cflags $debug_cflags -DSTACK_DIRECTION=-1" extra_configs="$pentium_configs $debug_configs $max_configs $error_inject --with-experimental-collations" . "$path/FINISH.sh" diff --git a/include/lf.h b/include/lf.h index 4712f9c4862..8d26efd1232 100644 --- a/include/lf.h +++ b/include/lf.h @@ -206,7 +206,7 @@ uint lf_alloc_pool_count(LF_ALLOCATOR *allocator); #define lf_alloc_get_pins(A, ST) lf_pinbox_get_pins(&(A)->pinbox, (ST)) #define _lf_alloc_put_pins(PINS) _lf_pinbox_put_pins(PINS) #define lf_alloc_put_pins(PINS) lf_pinbox_put_pins(PINS) -#define lf_alloc_direct_free(ALLOC, ADDR) my_free((gptr)(ADDR), MYF(0)) +#define lf_alloc_direct_free(ALLOC, ADDR) my_free((uchar*)(ADDR), MYF(0)) lock_wrap(lf_alloc_new, void *, (LF_PINS *pins), diff --git a/include/maria.h b/include/maria.h index 5f8e3dcdd41..9dd6e077666 100644 --- a/include/maria.h +++ b/include/maria.h @@ -179,18 +179,18 @@ typedef struct st_maria_keydef /* Key definition with open & info */ HA_KEYSEG *seg, *end; struct st_mysql_ftparser *parser; /* Fulltext [pre]parser */ int (*bin_search)(struct st_maria_info *info, - struct st_maria_keydef *keyinfo, byte *page, byte *key, - uint key_len, uint comp_flag, byte **ret_pos, - byte *buff, my_bool *was_last_key); + struct st_maria_keydef *keyinfo, uchar *page, uchar *key, + uint key_len, uint comp_flag, uchar **ret_pos, + uchar *buff, my_bool *was_last_key); uint(*get_key)(struct st_maria_keydef *keyinfo, uint nod_flag, - byte **page, byte *key); + uchar **page, uchar *key); int (*pack_key)(struct st_maria_keydef *keyinfo, uint nod_flag, - byte *next_key, byte *org_key, byte *prev_key, - const byte *key, struct st_maria_s_param *s_temp); - void (*store_key)(struct st_maria_keydef *keyinfo, byte *key_pos, + uchar *next_key, uchar *org_key, uchar *prev_key, + const uchar *key, struct st_maria_s_param *s_temp); + void (*store_key)(struct st_maria_keydef *keyinfo, uchar *key_pos, struct st_maria_s_param *s_temp); - int (*ck_insert)(struct st_maria_info *inf, uint k_nr, byte *k, uint klen); - int (*ck_delete)(struct st_maria_info *inf, uint k_nr, byte *k, uint klen); + int (*ck_insert)(struct st_maria_info *inf, uint k_nr, uchar *k, uint klen); + int (*ck_delete)(struct st_maria_info *inf, uint k_nr, uchar *k, uint klen); } MARIA_KEYDEF; @@ -208,7 +208,7 @@ typedef struct st_maria_decode_tree /* Decode huff-table */ { uint16 *table; uint quick_table_bits; - byte *intervalls; + uchar *intervalls; } MARIA_DECODE_TREE; @@ -236,7 +236,7 @@ typedef struct st_maria_columndef /* column information */ #ifndef NOT_PACKED_DATABASES void(*unpack)(struct st_maria_columndef *rec, struct st_maria_bit_buff *buff, - byte *start, byte *end); + uchar *start, uchar *end); enum en_fieldtype base_type; uint space_length_bits, pack_type; MARIA_DECODE_TREE *huff_tree; @@ -258,30 +258,30 @@ extern PAGECACHE maria_pagecache_var, *maria_pagecache; extern int maria_init(void); extern void maria_end(void); extern int maria_close(struct st_maria_info *file); -extern int maria_delete(struct st_maria_info *file, const byte *buff); +extern int maria_delete(struct st_maria_info *file, const uchar *buff); extern struct st_maria_info *maria_open(const char *name, int mode, uint wait_if_locked); extern struct st_maria_info *maria_clone(struct st_maria_share *share, int mode); extern int maria_panic(enum ha_panic_function function); -extern int maria_rfirst(struct st_maria_info *file, byte *buf, int inx); -extern int maria_rkey(struct st_maria_info *file, byte *buf, int inx, - const byte *key, +extern int maria_rfirst(struct st_maria_info *file, uchar *buf, int inx); +extern int maria_rkey(struct st_maria_info *file, uchar *buf, int inx, + const uchar *key, uint key_len, enum ha_rkey_function search_flag); -extern int maria_rlast(struct st_maria_info *file, byte *buf, int inx); -extern int maria_rnext(struct st_maria_info *file, byte *buf, int inx); -extern int maria_rnext_same(struct st_maria_info *info, byte *buf); -extern int maria_rprev(struct st_maria_info *file, byte *buf, int inx); -extern int maria_rrnd(struct st_maria_info *file, byte *buf, +extern int maria_rlast(struct st_maria_info *file, uchar *buf, int inx); +extern int maria_rnext(struct st_maria_info *file, uchar *buf, int inx); +extern int maria_rnext_same(struct st_maria_info *info, uchar *buf); +extern int maria_rprev(struct st_maria_info *file, uchar *buf, int inx); +extern int maria_rrnd(struct st_maria_info *file, uchar *buf, MARIA_RECORD_POS pos); extern int maria_scan_init(struct st_maria_info *file); -extern int maria_scan(struct st_maria_info *file, byte *buf); +extern int maria_scan(struct st_maria_info *file, uchar *buf); extern void maria_scan_end(struct st_maria_info *file); -extern int maria_rsame(struct st_maria_info *file, byte *record, int inx); -extern int maria_rsame_with_pos(struct st_maria_info *file, byte *record, +extern int maria_rsame(struct st_maria_info *file, uchar *record, int inx); +extern int maria_rsame_with_pos(struct st_maria_info *file, uchar *record, int inx, MARIA_RECORD_POS pos); -extern int maria_update(struct st_maria_info *file, const byte *old, - byte *new_record); -extern int maria_write(struct st_maria_info *file, byte *buff); +extern int maria_update(struct st_maria_info *file, const uchar *old, + uchar *new_record); +extern int maria_write(struct st_maria_info *file, uchar *buff); extern MARIA_RECORD_POS maria_position(struct st_maria_info *file); extern int maria_status(struct st_maria_info *info, MARIA_INFO *x, uint flag); extern int maria_lock_database(struct st_maria_info *file, int lock_type); @@ -348,8 +348,8 @@ typedef struct st_maria_sort_param MARIA_KEYDEF *keyinfo; MARIA_SORT_INFO *sort_info; HA_KEYSEG *seg; - byte **sort_keys; - byte *rec_buff; + uchar **sort_keys; + uchar *rec_buff; void *wordlist, *wordptr; MEM_ROOT wordroot; char *record; @@ -367,13 +367,13 @@ typedef struct st_maria_sort_param uint maxbuffers, keys, find_length, sort_keys_length; my_bool fix_datafile, master; my_bool calc_checksum; /* calculate table checksum */ - my_size_t rec_buff_size; + size_t rec_buff_size; int (*key_cmp)(struct st_maria_sort_param *, const void *, const void *); - int (*key_read)(struct st_maria_sort_param *, byte *); - int (*key_write)(struct st_maria_sort_param *, const byte *); + int (*key_read)(struct st_maria_sort_param *, uchar *); + int (*key_write)(struct st_maria_sort_param *, const uchar *); void (*lock_in_memory)(HA_CHECK *); - NEAR int (*write_keys)(struct st_maria_sort_param *, register byte **, + NEAR int (*write_keys)(struct st_maria_sort_param *, register uchar **, uint , struct st_buffpek *, IO_CACHE *); NEAR uint (*read_to_buffer)(IO_CACHE *,struct st_buffpek *, uint); NEAR int (*write_key)(struct st_maria_sort_param *, IO_CACHE *,char *, @@ -389,9 +389,9 @@ int maria_chk_size(HA_CHECK *param, MARIA_HA *info); int maria_chk_key(HA_CHECK *param, MARIA_HA *info); int maria_chk_data_link(HA_CHECK *param, MARIA_HA *info, int extend); int maria_repair(HA_CHECK *param, register MARIA_HA *info, - my_string name, int rep_quick); + char * name, int rep_quick); int maria_sort_index(HA_CHECK *param, register MARIA_HA *info, - my_string name); + char * name); int maria_repair_by_sort(HA_CHECK *param, register MARIA_HA *info, const char *name, int rep_quick); int maria_repair_parallel(HA_CHECK *param, register MARIA_HA *info, @@ -405,7 +405,7 @@ void maria_update_key_parts(MARIA_KEYDEF *keyinfo, ulong *rec_per_key_part, ulonglong records); int maria_filecopy(HA_CHECK *param, File to, File from, my_off_t start, my_off_t length, const char *type); -int maria_movepoint(MARIA_HA *info, byte *record, my_off_t oldpos, +int maria_movepoint(MARIA_HA *info, uchar *record, my_off_t oldpos, my_off_t newpos, uint prot_key); int maria_write_data_suffix(MARIA_SORT_INFO *sort_info, my_bool fix_datafile); int maria_test_if_almost_full(MARIA_HA *info); @@ -427,8 +427,8 @@ void maria_change_pagecache(PAGECACHE *old_key_cache, int maria_preload(MARIA_HA *info, ulonglong key_map, my_bool ignore_leaves); /* fulltext functions */ -FT_INFO *maria_ft_init_search(uint,void *, uint, byte *, uint, - CHARSET_INFO *, byte *); +FT_INFO *maria_ft_init_search(uint,void *, uint, uchar *, uint, + CHARSET_INFO *, uchar *); /* 'Almost-internal' Maria functions */ diff --git a/include/my_sys.h b/include/my_sys.h index a9c7b1dd549..9148d0cf29b 100644 --- a/include/my_sys.h +++ b/include/my_sys.h @@ -717,12 +717,12 @@ extern void handle_recived_signals(void); extern sig_handler my_set_alarm_variable(int signo); extern void my_string_ptr_sort(uchar *base,uint items,size_t size); extern void radixsort_for_str_ptr(uchar* base[], uint number_of_elements, - size_s size_of_element,uchar *buffer[]); + size_t size_of_element,uchar *buffer[]); extern qsort_t qsort2(void *base_ptr, size_t total_elems, size_t size, qsort2_cmp cmp, void *cmp_argument); extern qsort2_cmp get_ptr_compare(size_t); void my_store_ptr(uchar *buff, size_t pack_length, my_off_t pos); -my_off_t my_get_ptr(cuhar *ptr, size_t pack_length); +my_off_t my_get_ptr(uchar *ptr, size_t pack_length); extern int init_io_cache(IO_CACHE *info,File file,size_t cachesize, enum cache_type type,my_off_t seek_offset, pbool use_async_io, myf cache_myflags); @@ -840,11 +840,11 @@ extern void free_defaults(char **argv); extern void my_print_default_files(const char *conf_file); extern void print_defaults(const char *conf_file, const char **groups); extern my_bool my_compress(uchar *, size_t *, size_t *); -extern my_bool my_uncompress(uchar *, size_t *, size_t *); +extern my_bool my_uncompress(uchar *, size_t , size_t *); extern uchar *my_compress_alloc(const uchar *packet, size_t *len, size_t *complen); extern int packfrm(const uchar *, size_t, uchar **, size_t *); -extern int unpackfrm(const uchar **, size_t *, const uchar *); +extern int unpackfrm(uchar **, size_t *, const uchar *); extern ha_checksum my_checksum(ha_checksum crc, const uchar *mem, size_t count); diff --git a/include/myisam.h b/include/myisam.h index 2ec083c3e12..154a3d95b41 100644 --- a/include/myisam.h +++ b/include/myisam.h @@ -338,7 +338,7 @@ typedef struct st_mi_sort_param MI_SORT_INFO *sort_info; HA_KEYSEG *seg; uchar **sort_keys; - byte *rec_buff; + uchar *rec_buff; void *wordlist, *wordptr; MEM_ROOT wordroot; char *record; diff --git a/include/myisamchk.h b/include/myisamchk.h index 887cf835b87..78350b60bb0 100644 --- a/include/myisamchk.h +++ b/include/myisamchk.h @@ -76,8 +76,8 @@ typedef struct st_sort_key_blocks /* Used when sorting */ { - byte *buff, *end_pos; - byte lastkey[HA_MAX_POSSIBLE_KEY_BUFF]; + uchar *buff, *end_pos; + uchar lastkey[HA_MAX_POSSIBLE_KEY_BUFF]; uint last_length; int inited; } SORT_KEY_BLOCKS; @@ -147,15 +147,15 @@ typedef struct st_handler_check_param typedef struct st_sort_ftbuf { - byte *buf, *end; + uchar *buf, *end; int count; - byte lastkey[HA_MAX_KEY_BUFF]; + uchar lastkey[HA_MAX_KEY_BUFF]; } SORT_FT_BUF; typedef struct st_buffpek { my_off_t file_pos; /* Where we are in the sort file */ - byte *base, *key; /* Key pointers */ + uchar *base, *key; /* Key pointers */ ha_rows count; /* Number of rows in table */ ulong mem_count; /* numbers of keys in memory */ ulong max_keys; /* Max keys in buffert */ diff --git a/mysys/lf_hash.c b/mysys/lf_hash.c index 832f0eb5852..3f6b9082ab9 100644 --- a/mysys/lf_hash.c +++ b/mysys/lf_hash.c @@ -34,7 +34,7 @@ LF_REQUIRE_PINS(3); typedef struct { intptr volatile link; /* a pointer to the next element in a listand a flag */ uint32 hashnr; /* reversed hash number, for sorting */ - const byte *key; + const uchar *key; uint keylen; /* data is stored here, directly after the keylen. @@ -72,10 +72,10 @@ typedef struct { pins[0..2] are used, they are NOT removed on return */ static int lfind(LF_SLIST * volatile *head, CHARSET_INFO *cs, uint32 hashnr, - const byte *key, uint keylen, CURSOR *cursor, LF_PINS *pins) + const uchar *key, uint keylen, CURSOR *cursor, LF_PINS *pins) { uint32 cur_hashnr; - const byte *cur_key; + const uchar *cur_key; uint cur_keylen; intptr link; @@ -201,7 +201,7 @@ static LF_SLIST *linsert(LF_SLIST * volatile *head, CHARSET_INFO *cs, it uses pins[0..2], on return all pins are removed. */ static int ldelete(LF_SLIST * volatile *head, CHARSET_INFO *cs, uint32 hashnr, - const byte *key, uint keylen, LF_PINS *pins) + const uchar *key, uint keylen, LF_PINS *pins) { CURSOR cursor; int res; @@ -259,7 +259,7 @@ static int ldelete(LF_SLIST * volatile *head, CHARSET_INFO *cs, uint32 hashnr, all other pins are removed. */ static LF_SLIST *lsearch(LF_SLIST * volatile *head, CHARSET_INFO *cs, - uint32 hashnr, const byte *key, uint keylen, + uint32 hashnr, const uchar *key, uint keylen, LF_PINS *pins) { CURSOR cursor; @@ -271,8 +271,8 @@ static LF_SLIST *lsearch(LF_SLIST * volatile *head, CHARSET_INFO *cs, return res ? cursor.curr : 0; } -static inline const byte* hash_key(const LF_HASH *hash, - const byte *record, uint *length) +static inline const uchar* hash_key(const LF_HASH *hash, + const uchar *record, uint *length) { if (hash->get_key) return (*hash->get_key)(record, length, 0); @@ -285,7 +285,7 @@ static inline const byte* hash_key(const LF_HASH *hash, note, that the hash value is limited to 2^31, because we need one bit to distinguish between normal and dummy nodes. */ -static inline uint calc_hash(LF_HASH *hash, const byte *key, uint keylen) +static inline uint calc_hash(LF_HASH *hash, const uchar *key, uint keylen) { ulong nr1= 1, nr2= 4; hash->charset->coll->hash_sort(hash->charset, (uchar*) key, keylen, @@ -362,7 +362,7 @@ int lf_hash_insert(LF_HASH *hash, LF_PINS *pins, const void *data) if (unlikely(!node)) return -1; memcpy(node+1, data, hash->element_size); - node->key= hash_key(hash, (byte *)(node+1), &node->keylen); + node->key= hash_key(hash, (uchar *)(node+1), &node->keylen); hashnr= calc_hash(hash, node->key, node->keylen); bucket= hashnr % hash->size; el= _lf_dynarray_lvalue(&hash->array, bucket); @@ -399,7 +399,7 @@ int lf_hash_insert(LF_HASH *hash, LF_PINS *pins, const void *data) int lf_hash_delete(LF_HASH *hash, LF_PINS *pins, const void *key, uint keylen) { LF_SLIST * volatile *el; - uint bucket, hashnr= calc_hash(hash, (byte *)key, keylen); + uint bucket, hashnr= calc_hash(hash, (uchar *)key, keylen); bucket= hashnr % hash->size; lf_rwlock_by_pins(pins); @@ -415,7 +415,7 @@ int lf_hash_delete(LF_HASH *hash, LF_PINS *pins, const void *key, uint keylen) if (*el == NULL && unlikely(initialize_bucket(hash, el, bucket, pins))) return -1; if (ldelete(el, hash->charset, my_reverse_bits(hashnr) | 1, - (byte *)key, keylen, pins)) + (uchar *)key, keylen, pins)) { lf_rwunlock_by_pins(pins); return 1; @@ -438,7 +438,7 @@ int lf_hash_delete(LF_HASH *hash, LF_PINS *pins, const void *key, uint keylen) void *lf_hash_search(LF_HASH *hash, LF_PINS *pins, const void *key, uint keylen) { LF_SLIST * volatile *el, *found; - uint bucket, hashnr= calc_hash(hash, (byte *)key, keylen); + uint bucket, hashnr= calc_hash(hash, (uchar *)key, keylen); bucket= hashnr % hash->size; lf_rwlock_by_pins(pins); @@ -448,12 +448,12 @@ void *lf_hash_search(LF_HASH *hash, LF_PINS *pins, const void *key, uint keylen) if (*el == NULL && unlikely(initialize_bucket(hash, el, bucket, pins))) return MY_ERRPTR; found= lsearch(el, hash->charset, my_reverse_bits(hashnr) | 1, - (byte *)key, keylen, pins); + (uchar *)key, keylen, pins); lf_rwunlock_by_pins(pins); return found ? found+1 : 0; } -static const byte *dummy_key= ""; +static const uchar *dummy_key= ""; /* RETURN diff --git a/mysys/my_compress.c b/mysys/my_compress.c index d495a1c1c6d..5e770959105 100644 --- a/mysys/my_compress.c +++ b/mysys/my_compress.c @@ -178,7 +178,8 @@ int packfrm(const uchar *data, size_t len, if (my_compress((uchar*)data, &org_len, &comp_len)) goto err; - DBUG_PRINT("info", ("org_len: %lu comp_len: %lu", org_len, comp_len)); + DBUG_PRINT("info", ("org_len: %lu comp_len: %lu", (ulong) org_len, + (ulong) comp_len)); DBUG_DUMP("compressed", (char*)data, org_len); error= 2; diff --git a/mysys/my_rename.c b/mysys/my_rename.c index 64dbac955ea..39e6056a9e4 100644 --- a/mysys/my_rename.c +++ b/mysys/my_rename.c @@ -66,8 +66,9 @@ int my_rename(const char *from, const char *to, myf MyFlags) #ifdef NEED_EXPLICIT_SYNC_DIR /* do only the needed amount of syncs: */ char dir_from[FN_REFLEN], dir_to[FN_REFLEN]; - dirname_part(dir_from, from); - dirname_part(dir_to, to); + size_t dir_from_length, dir_to_length; + dirname_part(dir_from, from, &dir_from_length); + dirname_part(dir_to, to, &dir_to_length); if (my_sync_dir(dir_from, MyFlags) || (strcmp(dir_from, dir_to) && my_sync_dir(dir_to, MyFlags))) diff --git a/mysys/my_safehash.c b/mysys/my_safehash.c index 57f408942bf..b34ad5f16ff 100644 --- a/mysys/my_safehash.c +++ b/mysys/my_safehash.c @@ -53,7 +53,7 @@ static void safe_hash_entry_free(SAFE_HASH_ENTRY *entry) { DBUG_ENTER("safe_hash_entry_free"); - my_free((gptr) entry, MYF(0)); + my_free((uchar*) entry, MYF(0)); DBUG_VOID_RETURN; } @@ -70,11 +70,11 @@ static void safe_hash_entry_free(SAFE_HASH_ENTRY *entry) # reference on the key */ -static byte *safe_hash_entry_get(SAFE_HASH_ENTRY *entry, uint *length, - my_bool not_used __attribute__((unused))) +static uchar *safe_hash_entry_get(SAFE_HASH_ENTRY *entry, uint *length, + my_bool not_used __attribute__((unused))) { *length= entry->length; - return (byte*) entry->key; + return (uchar*) entry->key; } @@ -97,7 +97,7 @@ static byte *safe_hash_entry_get(SAFE_HASH_ENTRY *entry, uint *length, */ my_bool safe_hash_init(SAFE_HASH *hash, uint elements, - byte *default_value) + uchar *default_value) { DBUG_ENTER("safe_hash_init"); if (hash_init(&hash->hash, &my_charset_bin, elements, @@ -154,10 +154,10 @@ void safe_hash_free(SAFE_HASH *hash) # data associated with the key of default value if data was not found */ -byte *safe_hash_search(SAFE_HASH *hash, const byte *key, uint length, - byte *def) +uchar *safe_hash_search(SAFE_HASH *hash, const uchar *key, uint length, + uchar *def) { - byte *result; + uchar *result; DBUG_ENTER("safe_hash_search"); rw_rdlock(&hash->mutex); result= hash_search(&hash->hash, key, length); @@ -191,8 +191,8 @@ byte *safe_hash_search(SAFE_HASH *hash, const byte *key, uint length, 1 error (Can only be EOM). In this case my_message() is called. */ -my_bool safe_hash_set(SAFE_HASH *hash, const byte *key, uint length, - byte *data) +my_bool safe_hash_set(SAFE_HASH *hash, const uchar *key, uint length, + uchar *data) { SAFE_HASH_ENTRY *entry; my_bool error= 0; @@ -214,7 +214,7 @@ my_bool safe_hash_set(SAFE_HASH *hash, const byte *key, uint length, /* unlink entry from list */ if ((*entry->prev= entry->next)) entry->next->prev= entry->prev; - hash_delete(&hash->hash, (byte*) entry); + hash_delete(&hash->hash, (uchar*) entry); goto end; } if (entry) @@ -230,7 +230,7 @@ my_bool safe_hash_set(SAFE_HASH *hash, const byte *key, uint length, error= 1; goto end; } - entry->key= (byte*) (entry +1); + entry->key= (uchar*) (entry +1); memcpy((char*) entry->key, (char*) key, length); entry->length= length; entry->data= data; @@ -239,7 +239,7 @@ my_bool safe_hash_set(SAFE_HASH *hash, const byte *key, uint length, entry->next->prev= &entry->next; entry->prev= &hash->root; hash->root= entry; - if (my_hash_insert(&hash->hash, (byte*) entry)) + if (my_hash_insert(&hash->hash, (uchar*) entry)) { /* This can only happen if hash got out of memory */ my_free((char*) entry, MYF(0)); @@ -269,7 +269,7 @@ end: default value. */ -void safe_hash_change(SAFE_HASH *hash, byte *old_data, byte *new_data) +void safe_hash_change(SAFE_HASH *hash, uchar *old_data, uchar *new_data) { SAFE_HASH_ENTRY *entry, *next; DBUG_ENTER("safe_hash_change"); @@ -285,7 +285,7 @@ void safe_hash_change(SAFE_HASH *hash, byte *old_data, byte *new_data) { if ((*entry->prev= entry->next)) entry->next->prev= entry->prev; - hash_delete(&hash->hash, (byte*) entry); + hash_delete(&hash->hash, (uchar*) entry); } else entry->data= new_data; diff --git a/mysys/my_safehash.h b/mysys/my_safehash.h index 53845a5fec7..8a5856b6763 100644 --- a/mysys/my_safehash.h +++ b/mysys/my_safehash.h @@ -30,9 +30,9 @@ typedef struct st_safe_hash_entry { - byte *key; + uchar *key; uint length; - byte *data; + uchar *data; struct st_safe_hash_entry *next, **prev; } SAFE_HASH_ENTRY; @@ -43,16 +43,16 @@ typedef struct st_safe_hash_with_default rw_lock_t mutex; #endif HASH hash; - byte *default_value; + uchar *default_value; SAFE_HASH_ENTRY *root; } SAFE_HASH; my_bool safe_hash_init(SAFE_HASH *hash, uint elements, - byte *default_value); + uchar *default_value); void safe_hash_free(SAFE_HASH *hash); -byte *safe_hash_search(SAFE_HASH *hash, const byte *key, uint length, - byte *def); -my_bool safe_hash_set(SAFE_HASH *hash, const byte *key, uint length, - byte *data); -void safe_hash_change(SAFE_HASH *hash, byte *old_data, byte *new_data); +uchar *safe_hash_search(SAFE_HASH *hash, const uchar *key, uint length, + uchar *def); +my_bool safe_hash_set(SAFE_HASH *hash, const uchar *key, uint length, + uchar *data); +void safe_hash_change(SAFE_HASH *hash, uchar *old_data, uchar *new_data); diff --git a/mysys/my_sync.c b/mysys/my_sync.c index ab3fc89e0d3..ba6964b00d6 100644 --- a/mysys/my_sync.c +++ b/mysys/my_sync.c @@ -145,7 +145,8 @@ int my_sync_dir_by_file(const char *file_name, myf my_flags) { #ifdef NEED_EXPLICIT_SYNC_DIR char dir_name[FN_REFLEN]; - dirname_part(dir_name, file_name); + size_t dir_name_length; + dirname_part(dir_name, file_name, &dir_name_length); return my_sync_dir(dir_name, my_flags); #else return 0; diff --git a/sql/filesort.cc b/sql/filesort.cc index 6c72a0a2856..0012bebf688 100644 --- a/sql/filesort.cc +++ b/sql/filesort.cc @@ -1076,7 +1076,7 @@ uint read_to_buffer(IO_CACHE *fromfile, BUFFPEK *buffpek, void reuse_freed_buff(QUEUE *queue, BUFFPEK *reuse, uint key_length) { - byte *reuse_end= reuse->base + reuse->max_keys * key_length; + uchar *reuse_end= reuse->base + reuse->max_keys * key_length; for (uint i= 0; i < queue->elements; ++i) { BUFFPEK *bp= (BUFFPEK *) queue_element(queue, i); @@ -1158,7 +1158,7 @@ int merge_buffers(SORTPARAM *param, IO_CACHE *from_file, DBUG_RETURN(1); /* purecov: inspected */ for (buffpek= Fb ; buffpek <= Tb ; buffpek++) { - buffpek->base= (byte*) strpos; + buffpek->base= (uchar*) strpos; buffpek->max_keys= maxcount; strpos+= (uint) (error= (int) read_to_buffer(from_file, buffpek, rec_length)); @@ -1250,7 +1250,7 @@ int merge_buffers(SORTPARAM *param, IO_CACHE *from_file, } } buffpek= (BUFFPEK*) queue_top(&queue); - buffpek->base= (byte*) sort_buffer; + buffpek->base= (uchar*) sort_buffer; buffpek->max_keys= param->keys; /* diff --git a/sql/item_xmlfunc.cc b/sql/item_xmlfunc.cc index f8457a1ae50..6bc73f49a5a 100644 --- a/sql/item_xmlfunc.cc +++ b/sql/item_xmlfunc.cc @@ -2767,7 +2767,7 @@ String *Item_xml_str_func::parse_xml(String *raw_xml, String *parsed_xml_buf) if ((rc= my_xml_parse(&p, raw_xml->ptr(), raw_xml->length())) != MY_XML_OK) { char buf[128]; - my_snprintf(buf, sizeof(buf)-1, "parse error at line %d pos %lu: %s", + my_snprintf(buf, sizeof(buf)-1, "parse error at line %d pos %u: %s", my_xml_error_lineno(&p) + 1, my_xml_error_pos(&p) + 1, my_xml_error_string(&p)); diff --git a/sql/mysqld.cc b/sql/mysqld.cc index ddaf0c360bb..5c915f5474b 100644 --- a/sql/mysqld.cc +++ b/sql/mysqld.cc @@ -548,7 +548,7 @@ MY_LOCALE *my_default_lc_time_names; SHOW_COMP_OPTION have_ssl, have_symlink, have_dlopen, have_query_cache; SHOW_COMP_OPTION have_geometry, have_rtree_keys; -SHOW_COMP_OPTION have_crypt, have_compress; +SHOW_COMP_OPTION have_crypt, have_compress, have_maria_db; /* Thread specific variables */ @@ -1210,7 +1210,7 @@ void clean_up(bool print_message) xid_cache_free(); delete_elements(&key_caches, (void (*)(const char*, uchar*)) free_key_cache); #ifdef WITH_MARIA_STORAGE_ENGINE - delete_elements(&pagecaches, (void (*)(const char*, gptr)) free_pagecache); + delete_elements(&pagecaches, (void (*)(const char*, uchar*)) free_pagecache); #endif /* WITH_MARIA_STORAGE_ENGINE */ multi_keycache_free(); #ifdef WITH_MARIA_STORAGE_ENGINE @@ -6227,16 +6227,13 @@ The minimum value for this variable is 4096.", {"pagecache_age_threshold", OPT_KEY_CACHE_AGE_THRESHOLD, "This characterizes the number of hits a hot block has to be untouched until it is considered aged enough to be downgraded to a warm block. This specifies the percentage ratio of that number of hits to the total number of blocks in key cache", (uchar**) &maria_pagecache_var.param_age_threshold, - (uchar**) 0, - 0, (GET_ULONG | GET_ASK_ADDR), REQUIRED_ARG, + (uchar**) 0, 0, (GET_ULONG | GET_ASK_ADDR), REQUIRED_ARG, 300, 100, ~0L, 0, 100, 0}, {"pagecache_buffer_size", OPT_KEY_BUFFER_SIZE, "The size of the buffer used for index blocks for MyISAM tables. Increase this to get better index handling (for all reads and multiple writes) to as much as you can afford; 64M on a 256M machine that mainly runs MySQL is quite common.", (uchar**) &maria_pagecache_var.param_buff_size, - (uchar**) 0, - 0, (GET_ULL | GET_ASK_ADDR), - REQUIRED_ARG, KEY_CACHE_SIZE, MALLOC_OVERHEAD, ~(ulong) 0, MALLOC_OVERHEAD, - IO_SIZE, KEY_CACHE_SIZE}, + (uchar**) 0, 0, (GET_ULL | GET_ASK_ADDR), REQUIRED_ARG, + KEY_CACHE_SIZE, MALLOC_OVERHEAD, ~(ulong) 0, MALLOC_OVERHEAD, IO_SIZE, 0}, {"pagecache_division_limit", OPT_KEY_CACHE_DIVISION_LIMIT, "The minimum percentage of warm blocks in key cache", (uchar**) &maria_pagecache_var.param_division_limit, @@ -6252,11 +6249,12 @@ The minimum value for this variable is 4096.", "Optional colon separated list of plugins to load, where each plugin is " "identified by name and path to library seperated by an equals.", (uchar**) &opt_plugin_load, (uchar**) &opt_plugin_load, 0, + GET_STR, REQUIRED_ARG, 0, 0, 0, 0, 0, 0}, {"preload_buffer_size", OPT_PRELOAD_BUFFER_SIZE, - "The size of the buffer that is allocated when preloading indexes", - (uchar**) &global_system_variables.preload_buff_size, - (uchar**) &max_system_variables.preload_buff_size, 0, GET_ULONG, - REQUIRED_ARG, 32*1024L, 1024, 1024*1024*1024L, 0, 1, 0}, + "The size of the buffer that is allocated when preloading indexes", + (uchar**) &global_system_variables.preload_buff_size, + (uchar**) &max_system_variables.preload_buff_size, 0, GET_ULONG, + REQUIRED_ARG, 32*1024L, 1024, 1024*1024*1024L, 0, 1, 0}, {"query_alloc_block_size", OPT_QUERY_ALLOC_BLOCK_SIZE, "Allocation block size for query parsing and execution", (uchar**) &global_system_variables.query_alloc_block_size, diff --git a/sql/set_var.cc b/sql/set_var.cc index 5b28e257f5a..ea47f2f557b 100644 --- a/sql/set_var.cc +++ b/sql/set_var.cc @@ -345,10 +345,10 @@ static sys_var_thd_enum sys_myisam_stats_method(&vars, "myisam_stats_met &myisam_stats_method_typelib, NULL); -sys_var_thd_ulonglong sys_maria_max_sort_file_size("maria_max_sort_file_size", &SV::maria_max_sort_file_size, fix_maria_max_sort_file_size, 1); -sys_var_thd_ulong sys_maria_repair_threads("maria_repair_threads", &SV::maria_repair_threads); -sys_var_thd_ulong sys_maria_sort_buffer_size("maria_sort_buffer_size", &SV::maria_sort_buff_size); -sys_var_thd_enum sys_maria_stats_method("maria_stats_method", +static sys_var_thd_ulonglong sys_maria_max_sort_file_size(&vars, "maria_max_sort_file_size", &SV::maria_max_sort_file_size, fix_maria_max_sort_file_size, 1); +static sys_var_thd_ulong sys_maria_repair_threads(&vars, "maria_repair_threads", &SV::maria_repair_threads); +static sys_var_thd_ulong sys_maria_sort_buffer_size(&vars, "maria_sort_buffer_size", &SV::maria_sort_buff_size); +static sys_var_thd_enum sys_maria_stats_method(&vars, "maria_stats_method", &SV::maria_stats_method, &myisam_stats_method_typelib, NULL); @@ -1963,13 +1963,13 @@ uchar *sys_var_key_cache_param::value_ptr(THD *thd, enum_var_type type, #ifdef WITH_MARIA_STORAGE_ENGINE -byte *sys_var_pagecache_param::value_ptr(THD *thd, enum_var_type type, +uchar *sys_var_pagecache_param::value_ptr(THD *thd, enum_var_type type, LEX_STRING *base) { PAGECACHE *pagecache= get_pagecache(base); if (!pagecache) pagecache= &zero_pagecache; - return (byte*) pagecache + offset ; + return (uchar*) pagecache + offset ; } #endif /* WITH_MARIA_STORAGE_ENGINE */ @@ -3664,7 +3664,7 @@ static PAGECACHE *create_pagecache(const char *name, uint length) if ((pagecache= (PAGECACHE*) my_malloc(sizeof(PAGECACHE), MYF(MY_ZEROFILL | MY_WME)))) { - if (!new NAMED_LIST(&pagecaches, name, length, (gptr) pagecache)) + if (!new NAMED_LIST(&pagecaches, name, length, (uchar*) pagecache)) { my_free((char*) pagecache, MYF(0)); pagecache= 0; diff --git a/sql/set_var.h b/sql/set_var.h index ab865f312b3..3c8c06b34df 100644 --- a/sql/set_var.h +++ b/sql/set_var.h @@ -742,7 +742,7 @@ public: sys_var_pagecache_param(const char *name_arg, size_t offset_arg) :sys_var(name_arg), offset(offset_arg) {} - byte *value_ptr(THD *thd, enum_var_type type, LEX_STRING *base); + uchar *value_ptr(THD *thd, enum_var_type type, LEX_STRING *base); bool check_default(enum_var_type type) { return 1; } bool is_struct() { return 1; } }; diff --git a/sql/uniques.cc b/sql/uniques.cc index 619a516e54a..6bcffbd53de 100644 --- a/sql/uniques.cc +++ b/sql/uniques.cc @@ -448,7 +448,7 @@ static bool merge_walk(uchar *merge_buffer, ulong merge_buffer_size, */ for (top= begin; top != end; ++top) { - top->base= (byte*) (merge_buffer + (top - begin) * piece_size); + top->base= (uchar*) (merge_buffer + (top - begin) * piece_size); top->max_keys= max_key_count_per_piece; bytes_read= read_to_buffer(file, top, key_length); if (bytes_read == (uint) (-1)) diff --git a/storage/maria/ft_maria.c b/storage/maria/ft_maria.c index 06e7c4bd59c..1b082f904d0 100644 --- a/storage/maria/ft_maria.c +++ b/storage/maria/ft_maria.c @@ -22,8 +22,8 @@ #include "ma_ftdefs.h" FT_INFO *maria_ft_init_search(uint flags, void *info, uint keynr, - byte *query, uint query_len, CHARSET_INFO *cs, - byte *record) + uchar *query, uint query_len, CHARSET_INFO *cs, + uchar *record) { FT_INFO *res; if (flags & FT_BOOL) diff --git a/storage/maria/ha_maria.cc b/storage/maria/ha_maria.cc index e05f97a384d..0e629d6638c 100644 --- a/storage/maria/ha_maria.cc +++ b/storage/maria/ha_maria.cc @@ -142,7 +142,7 @@ int table2maria(TABLE *table_arg, MARIA_KEYDEF **keydef_out, { uint i, j, recpos, minpos, fieldpos, temp_length, length; enum ha_base_keytype type= HA_KEYTYPE_BINARY; - byte *record; + uchar *record; KEY *pos; MARIA_KEYDEF *keydef; MARIA_COLUMNDEF *recinfo, *recinfo_pos; @@ -559,7 +559,7 @@ int ha_maria::net_read_dump(NET * net) error= -1; goto err; } - if (my_write(data_fd, (byte *) net->read_pos, (uint) packet_len, + if (my_write(data_fd, (uchar *) net->read_pos, (uint) packet_len, MYF(MY_WME | MY_FNABP))) { error= errno; @@ -578,7 +578,7 @@ int ha_maria::dump(THD * thd, int fd) uint block_size= share->block_size; my_off_t bytes_to_read= share->state.state.data_file_length; int data_fd= file->dfile.file; - byte *buf= (byte *) my_malloc(block_size, MYF(MY_WME)); + uchar *buf= (uchar *) my_malloc(block_size, MYF(MY_WME)); if (!buf) return ENOMEM; @@ -603,7 +603,7 @@ int ha_maria::dump(THD * thd, int fd) } else { - if (my_net_write(net, (char*) buf, bytes)) + if (my_net_write(net, buf, bytes)) { error= errno ? errno : EPIPE; goto err; @@ -614,13 +614,13 @@ int ha_maria::dump(THD * thd, int fd) if (fd < 0) { - if (my_net_write(net, "", 0)) + if (my_net_write(net, (uchar*) "", 0)) error= errno ? errno : EPIPE; net_flush(net); } err: - my_free((gptr) buf, MYF(0)); + my_free((uchar*) buf, MYF(0)); return error; } #endif /* HAVE_REPLICATION */ @@ -699,10 +699,10 @@ int ha_maria::open(const char *name, int mode, uint test_if_locked) for (i= 0; i < table->s->keys; i++) { - struct st_plugin_int *parser= table->key_info[i].parser; + plugin_ref parser= table->key_info[i].parser; if (table->key_info[i].flags & HA_USES_PARSER) file->s->keyinfo[i].parser= - (struct st_mysql_ftparser *) parser->plugin->info; + (struct st_mysql_ftparser *)plugin_decl(parser)->info; table->key_info[i].block_size= file->s->keyinfo[i].block_length; } return (0); @@ -717,7 +717,7 @@ int ha_maria::close(void) } -int ha_maria::write_row(byte * buf) +int ha_maria::write_row(uchar * buf) { statistic_increment(table->in_use->status_var.ha_write_count, &LOCK_status); @@ -1210,19 +1210,14 @@ int ha_maria::assign_to_keycache(THD * thd, HA_CHECK_OPT *check_opt) TABLE_LIST *table_list= table->pos_in_table_list; DBUG_ENTER("ha_maria::assign_to_keycache"); - /* Check validity of the index references */ - if (table_list->use_index) - { - /* We only come here when the user did specify an index map */ - key_map kmap; - if (get_key_map_from_key_list(&kmap, table, table_list->use_index)) - { - errmsg= thd->net.last_error; - error= HA_ADMIN_FAILED; - goto err; - } - map= kmap.to_ulonglong(); - } + table->keys_in_use_for_query.clear_all(); + + if (table_list->process_index_hints(table)) + DBUG_RETURN(HA_ADMIN_FAILED); + map= ~(ulonglong) 0; + if (!table->keys_in_use_for_query.is_clear_all()) + /* use all keys if there's no list specified by the user through hints */ + map= table->keys_in_use_for_query.to_ulonglong(); if ((error= maria_assign_to_pagecache(file, map, new_pagecache))) { @@ -1233,7 +1228,6 @@ int ha_maria::assign_to_keycache(THD * thd, HA_CHECK_OPT *check_opt) error= HA_ADMIN_CORRUPT; } -err: if (error != HA_ADMIN_OK) { /* Send error to user */ @@ -1264,20 +1258,16 @@ int ha_maria::preload_keys(THD * thd, HA_CHECK_OPT *check_opt) DBUG_ENTER("ha_maria::preload_keys"); + table->keys_in_use_for_query.clear_all(); + + if (table_list->process_index_hints(table)) + DBUG_RETURN(HA_ADMIN_FAILED); + + map= ~(ulonglong) 0; /* Check validity of the index references */ - if (table_list->use_index) - { - key_map kmap; - get_key_map_from_key_list(&kmap, table, table_list->use_index); - if (kmap.is_set_all()) - { - errmsg= thd->net.last_error; - error= HA_ADMIN_FAILED; - goto err; - } - if (!kmap.is_clear_all()) - map= kmap.to_ulonglong(); - } + if (!table->keys_in_use_for_query.is_clear_all()) + /* use all keys if there's no list specified by the user through hints */ + map= table->keys_in_use_for_query.to_ulonglong(); maria_extra(file, HA_EXTRA_PRELOAD_BUFFER_SIZE, (void*) &thd->variables.preload_buff_size); @@ -1599,7 +1589,7 @@ bool ha_maria::is_crashed() const } -int ha_maria::update_row(const byte * old_data, byte * new_data) +int ha_maria::update_row(const uchar * old_data, uchar * new_data) { statistic_increment(table->in_use->status_var.ha_update_count, &LOCK_status); if (table->timestamp_field_type & TIMESTAMP_AUTO_SET_ON_UPDATE) @@ -1608,14 +1598,14 @@ int ha_maria::update_row(const byte * old_data, byte * new_data) } -int ha_maria::delete_row(const byte * buf) +int ha_maria::delete_row(const uchar * buf) { statistic_increment(table->in_use->status_var.ha_delete_count, &LOCK_status); return maria_delete(file, buf); } -int ha_maria::index_read(byte * buf, const byte * key, +int ha_maria::index_read(uchar * buf, const uchar * key, uint key_len, enum ha_rkey_function find_flag) { DBUG_ASSERT(inited == INDEX); @@ -1627,7 +1617,7 @@ int ha_maria::index_read(byte * buf, const byte * key, } -int ha_maria::index_read_idx(byte * buf, uint index, const byte * key, +int ha_maria::index_read_idx(uchar * buf, uint index, const uchar * key, uint key_len, enum ha_rkey_function find_flag) { statistic_increment(table->in_use->status_var.ha_read_key_count, @@ -1638,7 +1628,7 @@ int ha_maria::index_read_idx(byte * buf, uint index, const byte * key, } -int ha_maria::index_read_last(byte * buf, const byte * key, uint key_len) +int ha_maria::index_read_last(uchar * buf, const uchar * key, uint key_len) { DBUG_ENTER("ha_maria::index_read_last"); DBUG_ASSERT(inited == INDEX); @@ -1651,7 +1641,7 @@ int ha_maria::index_read_last(byte * buf, const byte * key, uint key_len) } -int ha_maria::index_next(byte * buf) +int ha_maria::index_next(uchar * buf) { DBUG_ASSERT(inited == INDEX); statistic_increment(table->in_use->status_var.ha_read_next_count, @@ -1662,7 +1652,7 @@ int ha_maria::index_next(byte * buf) } -int ha_maria::index_prev(byte * buf) +int ha_maria::index_prev(uchar * buf) { DBUG_ASSERT(inited == INDEX); statistic_increment(table->in_use->status_var.ha_read_prev_count, @@ -1673,7 +1663,7 @@ int ha_maria::index_prev(byte * buf) } -int ha_maria::index_first(byte * buf) +int ha_maria::index_first(uchar * buf) { DBUG_ASSERT(inited == INDEX); statistic_increment(table->in_use->status_var.ha_read_first_count, @@ -1684,7 +1674,7 @@ int ha_maria::index_first(byte * buf) } -int ha_maria::index_last(byte * buf) +int ha_maria::index_last(uchar * buf) { DBUG_ASSERT(inited == INDEX); statistic_increment(table->in_use->status_var.ha_read_last_count, @@ -1695,8 +1685,8 @@ int ha_maria::index_last(byte * buf) } -int ha_maria::index_next_same(byte * buf, - const byte *key __attribute__ ((unused)), +int ha_maria::index_next_same(uchar * buf, + const uchar *key __attribute__ ((unused)), uint length __attribute__ ((unused))) { DBUG_ASSERT(inited == INDEX); @@ -1724,7 +1714,7 @@ int ha_maria::rnd_end() } -int ha_maria::rnd_next(byte *buf) +int ha_maria::rnd_next(uchar *buf) { statistic_increment(table->in_use->status_var.ha_read_rnd_next_count, &LOCK_status); @@ -1734,13 +1724,13 @@ int ha_maria::rnd_next(byte *buf) } -int ha_maria::restart_rnd_next(byte *buf, byte *pos) +int ha_maria::restart_rnd_next(uchar *buf, uchar *pos) { return rnd_pos(buf, pos); } -int ha_maria::rnd_pos(byte * buf, byte *pos) +int ha_maria::rnd_pos(uchar * buf, uchar *pos) { statistic_increment(table->in_use->status_var.ha_read_rnd_count, &LOCK_status); @@ -1750,7 +1740,7 @@ int ha_maria::rnd_pos(byte * buf, byte *pos) } -void ha_maria::position(const byte * record) +void ha_maria::position(const uchar * record) { my_off_t row_position= maria_position(file); my_store_ptr(ref, ref_length, row_position); @@ -2052,7 +2042,7 @@ int ha_maria::create(const char *name, register TABLE *table_arg, 0, (MARIA_UNIQUEDEF *) 0, &create_info, create_flags); - my_free((gptr) recinfo, MYF(0)); + my_free((uchar*) recinfo, MYF(0)); DBUG_RETURN(error); } @@ -2070,7 +2060,7 @@ void ha_maria::get_auto_increment(ulonglong offset, ulonglong increment, { ulonglong nr; int error; - byte key[HA_MAX_KEY_LENGTH]; + uchar key[HA_MAX_KEY_LENGTH]; if (!table->s->next_number_key_offset) { // Autoincrement at key-start @@ -2142,7 +2132,7 @@ ha_rows ha_maria::records_in_range(uint inx, key_range *min_key, } -int ha_maria::ft_read(byte * buf) +int ha_maria::ft_read(uchar * buf) { int error; diff --git a/storage/maria/ha_maria.h b/storage/maria/ha_maria.h index a2f6b190657..22741ddb24d 100644 --- a/storage/maria/ha_maria.h +++ b/storage/maria/ha_maria.h @@ -77,19 +77,19 @@ public: bool called_by_logger_thread); int open(const char *name, int mode, uint test_if_locked); int close(void); - int write_row(byte * buf); - int update_row(const byte * old_data, byte * new_data); - int delete_row(const byte * buf); - int index_read(byte * buf, const byte * key, + int write_row(uchar * buf); + int update_row(const uchar * old_data, uchar * new_data); + int delete_row(const uchar * buf); + int index_read(uchar * buf, const uchar * key, uint key_len, enum ha_rkey_function find_flag); - int index_read_idx(byte * buf, uint idx, const byte * key, + int index_read_idx(uchar * buf, uint idx, const uchar * key, uint key_len, enum ha_rkey_function find_flag); - int index_read_last(byte * buf, const byte * key, uint key_len); - int index_next(byte * buf); - int index_prev(byte * buf); - int index_first(byte * buf); - int index_last(byte * buf); - int index_next_same(byte * buf, const byte * key, uint keylen); + int index_read_last(uchar * buf, const uchar * key, uint key_len); + int index_next(uchar * buf); + int index_prev(uchar * buf); + int index_first(uchar * buf); + int index_last(uchar * buf); + int index_next_same(uchar * buf, const uchar * key, uint keylen); int ft_init() { if (!ft_handler) @@ -100,16 +100,16 @@ public: FT_INFO *ft_init_ext(uint flags, uint inx, String * key) { return maria_ft_init_search(flags, file, inx, - (byte *) key->ptr(), key->length(), + (uchar *) key->ptr(), key->length(), key->charset(), table->record[0]); } - int ft_read(byte * buf); + int ft_read(uchar * buf); int rnd_init(bool scan); int rnd_end(void); - int rnd_next(byte * buf); - int rnd_pos(byte * buf, byte * pos); - int restart_rnd_next(byte * buf, byte * pos); - void position(const byte * record); + int rnd_next(uchar * buf); + int rnd_pos(uchar * buf, uchar * pos); + int restart_rnd_next(uchar * buf, uchar * pos); + void position(const uchar * record); int info(uint); int extra(enum ha_extra_function operation); int extra_opt(enum ha_extra_function operation, ulong cache_size); diff --git a/storage/maria/ma_bitmap.c b/storage/maria/ma_bitmap.c index e1308bce487..e781c47c241 100644 --- a/storage/maria/ma_bitmap.c +++ b/storage/maria/ma_bitmap.c @@ -132,7 +132,7 @@ static inline my_bool write_changed_bitmap(MARIA_SHARE *share, DBUG_ASSERT(share->pagecache->block_size == bitmap->block_size); return (pagecache_write(share->pagecache, &bitmap->file, bitmap->page, 0, - (byte*) bitmap->map, PAGECACHE_PLAIN_PAGE, + (uchar*) bitmap->map, PAGECACHE_PLAIN_PAGE, PAGECACHE_LOCK_LEFT_UNLOCKED, PAGECACHE_PIN_LEFT_UNPINNED, PAGECACHE_WRITE_DELAY, 0)); @@ -224,7 +224,7 @@ my_bool _ma_bitmap_end(MARIA_SHARE *share) { my_bool res= _ma_flush_bitmap(share); pthread_mutex_destroy(&share->bitmap.bitmap_lock); - my_free((byte*) share->bitmap.map, MYF(MY_ALLOW_ZERO_PTR)); + my_free((uchar*) share->bitmap.map, MYF(MY_ALLOW_ZERO_PTR)); share->bitmap.map= 0; return res; } @@ -514,7 +514,7 @@ static my_bool _ma_read_bitmap_page(MARIA_SHARE *share, DBUG_ASSERT(share->pagecache->block_size == bitmap->block_size); res= pagecache_read(share->pagecache, (PAGECACHE_FILE*)&bitmap->file, page, 0, - (byte*) bitmap->map, + (uchar*) bitmap->map, PAGECACHE_PLAIN_PAGE, PAGECACHE_LOCK_LEFT_UNLOCKED, 0) == 0; #ifndef DBUG_OFF @@ -607,7 +607,7 @@ static my_bool move_to_next_bitmap(MARIA_HA *info, MARIA_FILE_BITMAP *bitmap) fill_block() bitmap Bitmap handle block Store data about what we found - best_data Pointer to best 6 byte aligned area in bitmap->map + best_data Pointer to best 6 uchar aligned area in bitmap->map best_pos Which bit in *best_data the area starts 0 = first bit pattern, 1 second bit pattern etc best_bits The original value of the bits at best_pos @@ -997,7 +997,7 @@ static ulong allocate_full_pages(MARIA_FILE_BITMAP *bitmap, best_data+= size; if ((best_area_size-= size * 8)) { - /* fill last byte */ + /* fill last uchar */ *best_data|= (uchar) ((1 << best_area_size) -1); best_data++; } @@ -1857,7 +1857,7 @@ err: 1 error (Couldn't write or read bitmap page) */ -my_bool _ma_bitmap_free_full_pages(MARIA_HA *info, const byte *extents, +my_bool _ma_bitmap_free_full_pages(MARIA_HA *info, const uchar *extents, uint count) { DBUG_ENTER("_ma_bitmap_free_full_pages"); diff --git a/storage/maria/ma_blockrec.c b/storage/maria/ma_blockrec.c index d2512f1e025..0c502641d9b 100644 --- a/storage/maria/ma_blockrec.c +++ b/storage/maria/ma_blockrec.c @@ -53,8 +53,8 @@ Page header: LSN 7 bytes Log position for last page change - PAGE_TYPE 1 byte 1 for head / 2 for tail / 3 for blob - NO 1 byte Number of row/tail entries on page + PAGE_TYPE 1 uchar 1 for head / 2 for tail / 3 for blob + NO 1 uchar Number of row/tail entries on page empty space 2 bytes Empty space on page The most significant bit in PAGE_TYPE is set to 1 if the data on the page @@ -72,7 +72,7 @@ the 1 most significant bit of the length could be used for some states of the row (in other words, we should try to keep these reserved) - eof flag 1 byte Reserved for full page read testing. (Ie, did the + eof flag 1 uchar Reserved for full page read testing. (Ie, did the previous write get the whole block on disk. ---------------- @@ -80,7 +80,7 @@ Structure of blob pages: LSN 7 bytes Log position for last page change - PAGE_TYPE 1 byte 3 + PAGE_TYPE 1 uchar 3 data @@ -88,7 +88,7 @@ Row data structure: - Flag 1 byte Marker of which header field exists + Flag 1 uchar Marker of which header field exists TRANSID 6 bytes TRANSID of changing transaction (optional, added on insert and first update/delete) @@ -98,17 +98,17 @@ update/delete) DELETE_TRANSID 6 bytes (optional). TRANSID of original row. Added on delete. - Nulls_extended 1 byte To allow us to add new DEFAULT NULL + Nulls_extended 1 uchar To allow us to add new DEFAULT NULL fields (optional, added after first change of row after alter table) - Number of ROW_EXTENT's 1-3 byte Length encoded, optional + Number of ROW_EXTENT's 1-3 uchar Length encoded, optional This is the number of extents the row is split into - First row_extent 7 byte Pointer to first row extent (optional) + First row_extent 7 uchar Pointer to first row extent (optional) - Total length of length array 1-3 byte Only used if we have + Total length of length array 1-3 uchar Only used if we have char/varchar/blob fields. - Row checksum 1 byte Only if table created with checksums + Row checksum 1 uchar Only if table created with checksums Null_bits .. One bit for each NULL field (a field that may have the value NULL) Empty_bits .. One bit for each field that may be 'empty'. @@ -137,7 +137,7 @@ Critical fixed length, not null, fields. (Note, these can't be dropped) Fixed length, null fields - Length array, 1-4 byte per field for all CHAR/VARCHAR/BLOB fields. + Length array, 1-4 uchar per field for all CHAR/VARCHAR/BLOB fields. Number of bytes used in length array per entry is depending on max length for field. @@ -272,12 +272,12 @@ typedef struct st_maria_extent_cursor { /* - Pointer to packed byte array of extents for the row. + Pointer to packed uchar array of extents for the row. Format is described above in the header */ - byte *extent; + uchar *extent; /* Where data starts on page; Only for debugging */ - byte *data_start; + uchar *data_start; /* Position to all tails in the row. Updated when reading a row */ MARIA_RECORD_POS *tail_positions; /* Current page */ @@ -300,16 +300,16 @@ static my_bool delete_tails(MARIA_HA *info, MARIA_RECORD_POS *tails); static my_bool delete_head_or_tail(MARIA_HA *info, ulonglong page, uint record_number, my_bool head); -static void _ma_print_directory(byte *buff, uint block_size); -static void compact_page(byte *buff, uint block_size, uint rownr, +static void _ma_print_directory(uchar *buff, uint block_size); +static void compact_page(uchar *buff, uint block_size, uint rownr, my_bool extend_block); static uchar *store_page_range(uchar *to, MARIA_BITMAP_BLOCK *block, uint block_size, ulong length); -static size_t fill_insert_undo_parts(MARIA_HA *info, const byte *record, +static size_t fill_insert_undo_parts(MARIA_HA *info, const uchar *record, LEX_STRING *log_parts, uint *log_parts_count); -static size_t fill_update_undo_parts(MARIA_HA *info, const byte *oldrec, - const byte *newrec, +static size_t fill_update_undo_parts(MARIA_HA *info, const uchar *oldrec, + const uchar *newrec, LEX_STRING *log_parts, uint *log_parts_count); @@ -358,7 +358,7 @@ void _ma_init_block_record_data(void) { uint i; bzero(total_header_size, sizeof(total_header_size)); - total_header_size[0]= FLAG_SIZE; /* Flag byte */ + total_header_size[0]= FLAG_SIZE; /* Flag uchar */ for (i= 1; i < array_elements(total_header_size); i++) { uint size= FLAG_SIZE, j, bit; @@ -484,10 +484,10 @@ err: void _ma_end_block_record(MARIA_HA *info) { DBUG_ENTER("_ma_end_block_record"); - my_free((gptr) info->cur_row.empty_bits, MYF(MY_ALLOW_ZERO_PTR)); + my_free((uchar*) info->cur_row.empty_bits, MYF(MY_ALLOW_ZERO_PTR)); delete_dynamic(&info->bitmap_blocks); delete_dynamic(&info->pinned_pages); - my_free((gptr) info->cur_row.extents, MYF(MY_ALLOW_ZERO_PTR)); + my_free((uchar*) info->cur_row.extents, MYF(MY_ALLOW_ZERO_PTR)); /* The data file is closed, when needed, in ma_once_end_block_record(). The following protects us from doing an extra, not allowed, close @@ -503,7 +503,7 @@ void _ma_end_block_record(MARIA_HA *info) ****************************************************************************/ /* - Return the next used byte on the page after a directory entry. + Return the next used uchar on the page after a directory entry. SYNOPSIS start_of_next_entry() @@ -514,9 +514,9 @@ void _ma_end_block_record(MARIA_HA *info) Everything between the '*dir' and this are free to be used. */ -static inline uint start_of_next_entry(byte *dir) +static inline uint start_of_next_entry(uchar *dir) { - byte *prev; + uchar *prev; /* Find previous used entry. (There is always a previous entry as the directory never starts with a deleted entry) @@ -541,9 +541,9 @@ static inline uint start_of_next_entry(byte *dir) Used mainly to detect rows with wrong extent information */ -static my_bool check_if_zero(byte *pos, uint length) +static my_bool check_if_zero(uchar *pos, uint length) { - byte *end; + uchar *end; for (end= pos+ length; pos != end ; pos++) if (pos[0] != 0) return 1; @@ -636,12 +636,12 @@ void _ma_unpin_all_pages(MARIA_HA *info, LSN undo_lsn) # Pointer to directory entry on page */ -static byte *find_free_position(byte *buff, uint block_size, uint *res_rownr, +static uchar *find_free_position(uchar *buff, uint block_size, uint *res_rownr, uint *res_length, uint *empty_space) { uint max_entry= (uint) ((uchar*) buff)[DIR_COUNT_OFFSET]; uint entry, length, first_pos; - byte *dir, *end; + uchar *dir, *end; DBUG_ENTER("find_free_position"); DBUG_PRINT("info", ("max_entry: %u", max_entry)); @@ -680,7 +680,7 @@ static byte *find_free_position(byte *buff, uint block_size, uint *res_rownr, uint2korr(end + DIR_ENTRY_SIZE+ 2)); *empty_space= uint2korr(buff + EMPTY_SPACE_OFFSET); } - buff[DIR_COUNT_OFFSET]= (byte) (uchar) max_entry+1; + buff[DIR_COUNT_OFFSET]= (uchar) (uchar) max_entry+1; length= (uint) (dir - buff - first_pos); DBUG_ASSERT(length <= *empty_space - DIR_ENTRY_SIZE); int2store(dir, first_pos); @@ -713,11 +713,11 @@ static byte *find_free_position(byte *buff, uint block_size, uint *res_rownr, extents. */ -static void calc_record_size(MARIA_HA *info, const byte *record, +static void calc_record_size(MARIA_HA *info, const uchar *record, MARIA_ROW *row) { MARIA_SHARE *share= info->s; - byte *field_length_data; + uchar *field_length_data; MARIA_COLUMNDEF *column, *end_column; uint *null_field_lengths= row->null_field_lengths; ulong *blob_lengths= row->blob_lengths; @@ -778,7 +778,7 @@ static void calc_record_size(MARIA_HA *info, const byte *record, { uint length= (end - pos); if (column->length <= 255) - *field_length_data++= (byte) (uchar) length; + *field_length_data++= (uchar) length; else { int2store(field_length_data, length); @@ -792,8 +792,8 @@ static void calc_record_size(MARIA_HA *info, const byte *record, case FIELD_VARCHAR: { uint length, field_length_data_length; - const byte *field_pos= record + column->offset; - /* 256 is correct as this includes the length byte */ + const uchar *field_pos= record + column->offset; + /* 256 is correct as this includes the length uchar */ field_length_data[0]= field_pos[0]; if (column->length <= 256) @@ -820,7 +820,7 @@ static void calc_record_size(MARIA_HA *info, const byte *record, } case FIELD_BLOB: { - const byte *field_pos= record + column->offset; + const uchar *field_pos= record + column->offset; uint size_length= column->length - portable_sizeof_char_ptr; ulong blob_length= _ma_calc_blob_length(size_length, field_pos); @@ -872,12 +872,12 @@ static void calc_record_size(MARIA_HA *info, const byte *record, */ -static void compact_page(byte *buff, uint block_size, uint rownr, +static void compact_page(uchar *buff, uint block_size, uint rownr, my_bool extend_block) { uint max_entry= (uint) ((uchar *) buff)[DIR_COUNT_OFFSET]; uint page_pos, next_free_pos, start_of_found_block, diff, end_of_found_block; - byte *dir, *end; + uchar *dir, *end; DBUG_ENTER("compact_page"); DBUG_PRINT("enter", ("rownr: %u", rownr)); DBUG_ASSERT(max_entry > 0 && @@ -978,7 +978,7 @@ static void compact_page(byte *buff, uint block_size, uint rownr, uint length= (uint) (dir - buff) - start_of_found_block; int2store(dir+2, length); } - buff[PAGE_TYPE_OFFSET]&= ~(byte) PAGE_CAN_BE_COMPACTED; + buff[PAGE_TYPE_OFFSET]&= ~(uchar) PAGE_CAN_BE_COMPACTED; } DBUG_EXECUTE("directory", _ma_print_directory(buff, block_size);); DBUG_VOID_RETURN; @@ -1009,9 +1009,9 @@ static void compact_page(byte *buff, uint block_size, uint rownr, struct st_row_pos_info { - byte *buff; /* page buffer */ - byte *data; /* Place for data */ - byte *dir; /* Directory */ + uchar *buff; /* page buffer */ + uchar *data; /* Place for data */ + uchar *dir; /* Directory */ uint length; /* Length for data */ uint rownr; /* Offset in directory */ uint empty_space; /* Space left on page */ @@ -1019,7 +1019,7 @@ struct st_row_pos_info static my_bool get_head_or_tail_page(MARIA_HA *info, MARIA_BITMAP_BLOCK *block, - byte *buff, uint length, uint page_type, + uchar *buff, uint length, uint page_type, enum pagecache_page_lock lock, struct st_row_pos_info *res) { @@ -1043,7 +1043,7 @@ static my_bool get_head_or_tail_page(MARIA_HA *info, */ bzero(buff+ PAGE_HEADER_SIZE, block_size - PAGE_HEADER_SIZE); - buff[PAGE_TYPE_OFFSET]= (byte) page_type; + buff[PAGE_TYPE_OFFSET]= (uchar) page_type; buff[DIR_COUNT_OFFSET]= 1; res->buff= buff; res->empty_space= res->length= (block_size - PAGE_OVERHEAD_SIZE); @@ -1056,7 +1056,7 @@ static my_bool get_head_or_tail_page(MARIA_HA *info, } else { - byte *dir; + uchar *dir; /* Read old page */ DBUG_ASSERT(share->pagecache->block_size == block_size); if (!(res->buff= pagecache_read(share->pagecache, @@ -1120,7 +1120,7 @@ crashed: static my_bool write_tail(MARIA_HA *info, MARIA_BITMAP_BLOCK *block, - byte *row_part, uint length) + uchar *row_part, uint length) { MARIA_SHARE *share= share= info->s; MARIA_PINNED_PAGE page_link; @@ -1237,13 +1237,13 @@ static my_bool write_tail(MARIA_HA *info, static my_bool write_full_pages(MARIA_HA *info, LSN lsn, MARIA_BITMAP_BLOCK *block, - byte *data, ulong length) + uchar *data, ulong length) { my_off_t page; MARIA_SHARE *share= share= info->s; uint block_size= share->block_size; uint data_size= FULL_PAGE_SIZE(block_size); - byte *buff= info->keyread_buff; + uchar *buff= info->keyread_buff; uint page_count; my_off_t position; DBUG_ENTER("write_full_pages"); @@ -1277,7 +1277,7 @@ static my_bool write_full_pages(MARIA_HA *info, info->state->data_file_length= position; } lsn_store(buff, lsn); - buff[PAGE_TYPE_OFFSET]= (byte) BLOB_PAGE; + buff[PAGE_TYPE_OFFSET]= (uchar) BLOB_PAGE; copy_length= min(data_size, length); memcpy(buff + LSN_SIZE + PAGE_TYPE_SIZE, data, copy_length); length-= copy_length; @@ -1354,8 +1354,8 @@ static uchar *store_page_range(uchar *to, MARIA_BITMAP_BLOCK *block, We don't have to store the position for the head block */ -static void store_extent_info(byte *to, - byte *row_extents_second_part, +static void store_extent_info(uchar *to, + uchar *row_extents_second_part, MARIA_BITMAP_BLOCK *first_block, uint count) { @@ -1385,7 +1385,7 @@ static void store_extent_info(byte *to, In some unlikely cases we have allocated to many blocks. Clear this data. */ - bzero(to, (my_size_t) (row_extents_second_part + copy_length - to)); + bzero(to, (size_t) (row_extents_second_part + copy_length - to)); } @@ -1494,16 +1494,16 @@ static my_bool free_full_page_range(MARIA_HA *info, ulonglong page, uint count) */ static my_bool write_block_record(MARIA_HA *info, - const byte *old_record, const byte *record, + const uchar *old_record, const uchar *record, MARIA_ROW *row, MARIA_BITMAP_BLOCKS *bitmap_blocks, my_bool head_block_is_read, struct st_row_pos_info *row_pos) { - byte *data, *end_of_data, *tmp_data_used, *tmp_data; - byte *row_extents_first_part, *row_extents_second_part; - byte *field_length_data; - byte *page_buff; + uchar *data, *end_of_data, *tmp_data_used, *tmp_data; + uchar *row_extents_first_part, *row_extents_second_part; + uchar *field_length_data; + uchar *page_buff; MARIA_BITMAP_BLOCK *block, *head_block; MARIA_SHARE *share= info->s; MARIA_COLUMNDEF *column, *end_column; @@ -1552,7 +1552,7 @@ static my_bool write_block_record(MARIA_HA *info, if (share->base.pack_fields) store_key_length_inc(data, row->field_lengths_length); if (share->calc_checksum) - *(data++)= (byte) info->cur_row.checksum; + *(data++)= (uchar) info->cur_row.checksum; memcpy(data, record, share->base.null_bytes); data+= share->base.null_bytes; memcpy(data, row->empty_bits, share->base.pack_bytes); @@ -1572,7 +1572,7 @@ static my_bool write_block_record(MARIA_HA *info, row->head_length)) DBUG_RETURN(1); - tmp_data_used= 0; /* Either 0 or last used byte in 'data' */ + tmp_data_used= 0; /* Either 0 or last used uchar in 'data' */ tmp_data= data; if (row_extents_in_use) @@ -1620,7 +1620,7 @@ static my_bool write_block_record(MARIA_HA *info, column < end_column ; column++) { - const byte *field_pos; + const uchar *field_pos; ulong length; if ((record[column->null_pos] & column->null_bit) || (row->empty_bits[column->empty_pos] & column->empty_bit)) @@ -1647,7 +1647,7 @@ static my_bool write_block_record(MARIA_HA *info, if (column->length <= 256) { length= (uint) (uchar) *field_length_data++; - field_pos++; /* Skip length byte */ + field_pos++; /* Skip length uchar */ } else { @@ -1681,12 +1681,12 @@ static my_bool write_block_record(MARIA_HA *info, for (; column < end_column && *blob_lengths < (ulong) (end_of_data - data); column++, blob_lengths++) { - byte *tmp_pos; + uchar *tmp_pos; uint length; if (!*blob_lengths) /* Null or "" */ continue; length= column->length - portable_sizeof_char_ptr; - memcpy_fixed((byte*) &tmp_pos, record + column->offset + length, + memcpy_fixed((uchar*) &tmp_pos, record + column->offset + length, sizeof(char*)); memcpy(data, tmp_pos, *blob_lengths); data+= *blob_lengths; @@ -1746,14 +1746,14 @@ static my_bool write_block_record(MARIA_HA *info, for (; column < end_column; column++, blob_lengths++) { - byte *blob_pos; + uchar *blob_pos; if (!*blob_lengths) /* Null or "" */ continue; if (block[block->sub_blocks - 1].used & BLOCKUSED_TAIL) { uint length; length= column->length - portable_sizeof_char_ptr; - memcpy_fixed((byte *) &blob_pos, record + column->offset + length, + memcpy_fixed((uchar *) &blob_pos, record + column->offset + length, sizeof(char*)); length= *blob_lengths % FULL_PAGE_SIZE(block_size); /* tail size */ if (length != *blob_lengths) @@ -1912,7 +1912,7 @@ static my_bool write_block_record(MARIA_HA *info, { ulong data_length= (tmp_data - info->rec_buff); uint length; - byte *extent_data; + uchar *extent_data; length= (uint) (data_length % FULL_PAGE_SIZE(block_size)); if (write_tail(info, head_tail_block, @@ -2062,9 +2062,9 @@ static my_bool write_block_record(MARIA_HA *info, blob_length-= (blob_length % FULL_PAGE_SIZE(block_size)); if (blob_length) { - memcpy_fixed((byte*) &log_array_pos->str, + memcpy_fixed((uchar*) &log_array_pos->str, record + column->offset + length, - sizeof(byte*)); + sizeof(uchar*)); log_array_pos->length= blob_length; log_entry_length+= blob_length; log_array_pos++; @@ -2087,7 +2087,7 @@ static my_bool write_block_record(MARIA_HA *info, (uint) (log_array_pos - log_array), log_array, log_data); if (log_array != tmp_log_array) - my_free((gptr) log_array, MYF(0)); + my_free((uchar*) log_array, MYF(0)); if (error) goto disk_err; } @@ -2153,13 +2153,13 @@ static my_bool write_block_record(MARIA_HA *info, /* Write rest of blobs (data, but no tails as they are already written) */ for (; column < end_column; column++, blob_lengths++) { - byte *blob_pos; + uchar *blob_pos; uint length; ulong blob_length; if (!*blob_lengths) /* Null or "" */ continue; length= column->length - portable_sizeof_char_ptr; - memcpy_fixed((byte*) &blob_pos, record + column->offset + length, + memcpy_fixed((uchar*) &blob_pos, record + column->offset + length, sizeof(char*)); /* remove tail part */ blob_length= *blob_lengths; @@ -2214,7 +2214,7 @@ disk_err: */ MARIA_RECORD_POS _ma_write_init_block_record(MARIA_HA *info, - const byte *record) + const uchar *record) { MARIA_BITMAP_BLOCKS *blocks= &info->cur_row.insert_blocks; struct st_row_pos_info row_pos; @@ -2231,7 +2231,7 @@ MARIA_RECORD_POS _ma_write_init_block_record(MARIA_HA *info, info->cur_row.lastpos= ma_recordpos(blocks->block->page, row_pos.rownr); if (info->s->calc_checksum) info->cur_row.checksum= (info->s->calc_checksum)(info,record); - if (write_block_record(info, (byte*) 0, record, &info->cur_row, + if (write_block_record(info, (uchar*) 0, record, &info->cur_row, blocks, blocks->block->org_bitmap_value != 0, &row_pos)) DBUG_RETURN(HA_OFFSET_ERROR); /* Error reading bitmap */ @@ -2249,7 +2249,7 @@ MARIA_RECORD_POS _ma_write_init_block_record(MARIA_HA *info, */ my_bool _ma_write_block_record(MARIA_HA *info __attribute__ ((unused)), - const byte *record __attribute__ ((unused))) + const uchar *record __attribute__ ((unused))) { return 0; /* Row already written */ } @@ -2342,15 +2342,15 @@ my_bool _ma_write_abort_block_record(MARIA_HA *info) */ my_bool _ma_update_block_record(MARIA_HA *info, MARIA_RECORD_POS record_pos, - const byte *oldrec, const byte *record) + const uchar *oldrec, const uchar *record) { MARIA_BITMAP_BLOCKS *blocks= &info->cur_row.insert_blocks; - byte *buff; + uchar *buff; MARIA_ROW *cur_row= &info->cur_row, *new_row= &info->new_row; MARIA_PINNED_PAGE page_link; uint rownr, org_empty_size, head_length; uint block_size= info->s->block_size; - byte *dir; + uchar *dir; ulonglong page; struct st_row_pos_info row_pos; MARIA_SHARE *share= info->s; @@ -2483,7 +2483,7 @@ static my_bool delete_head_or_tail(MARIA_HA *info, MARIA_SHARE *share= info->s; uint number_of_records, empty_space, length; uint block_size= share->block_size; - byte *buff, *dir; + uchar *buff, *dir; LSN lsn; MARIA_PINNED_PAGE page_link; DBUG_ENTER("delete_head_or_tail"); @@ -2520,14 +2520,14 @@ static my_bool delete_head_or_tail(MARIA_HA *info, if (record_number == number_of_records - 1) { /* Delete this entry and all following empty directory entries */ - byte *end= buff + block_size - PAGE_SUFFIX_SIZE; + uchar *end= buff + block_size - PAGE_SUFFIX_SIZE; do { number_of_records--; dir+= DIR_ENTRY_SIZE; empty_space+= DIR_ENTRY_SIZE; } while (dir < end && dir[0] == 0 && dir[1] == 0); - buff[DIR_COUNT_OFFSET]= (byte) (uchar) number_of_records; + buff[DIR_COUNT_OFFSET]= (uchar) (uchar) number_of_records; } empty_space+= length; if (number_of_records != 0) @@ -2537,7 +2537,7 @@ static my_bool delete_head_or_tail(MARIA_HA *info, /* Update directory */ int2store(buff + EMPTY_SPACE_OFFSET, empty_space); - buff[PAGE_TYPE_OFFSET]|= (byte) PAGE_CAN_BE_COMPACTED; + buff[PAGE_TYPE_OFFSET]|= (uchar) PAGE_CAN_BE_COMPACTED; DBUG_ASSERT(share->pagecache->block_size == block_size); /* Log REDO data */ @@ -2630,7 +2630,7 @@ static my_bool delete_tails(MARIA_HA *info, MARIA_RECORD_POS *tails) for rows with many splits. */ -my_bool _ma_delete_block_record(MARIA_HA *info, const byte *record) +my_bool _ma_delete_block_record(MARIA_HA *info, const uchar *record) { ulonglong page; uint record_number; @@ -2705,12 +2705,12 @@ err: In this case *end_of_data is set. */ -static byte *get_record_position(byte *buff, uint block_size, - uint record_number, byte **end_of_data) +static uchar *get_record_position(uchar *buff, uint block_size, + uint record_number, uchar **end_of_data) { uint number_of_records= (uint) ((uchar *) buff)[DIR_COUNT_OFFSET]; - byte *dir; - byte *data; + uchar *dir; + uchar *data; uint offset, length; #ifdef SANITY_CHECKS @@ -2755,7 +2755,7 @@ static byte *get_record_position(byte *buff, uint block_size, extent is a cursor over which pages to read */ -static void init_extent(MARIA_EXTENT_CURSOR *extent, byte *extent_info, +static void init_extent(MARIA_EXTENT_CURSOR *extent, uchar *extent_info, uint extents, MARIA_RECORD_POS *tail_positions) { uint page_count; @@ -2788,11 +2788,11 @@ static void init_extent(MARIA_EXTENT_CURSOR *extent, byte *extent_info, In this case end_of_data is updated to point to end of data. */ -static byte *read_next_extent(MARIA_HA *info, MARIA_EXTENT_CURSOR *extent, - byte **end_of_data) +static uchar *read_next_extent(MARIA_HA *info, MARIA_EXTENT_CURSOR *extent, + uchar **end_of_data) { MARIA_SHARE *share= info->s; - byte *buff, *data; + uchar *buff, *data; DBUG_ENTER("read_next_extent"); if (!extent->page_count) @@ -2879,9 +2879,9 @@ crashed: 1 error */ -static my_bool read_long_data(MARIA_HA *info, byte *to, ulong length, +static my_bool read_long_data(MARIA_HA *info, uchar *to, ulong length, MARIA_EXTENT_CURSOR *extent, - byte **data, byte **end_of_data) + uchar **data, uchar **end_of_data) { DBUG_ENTER("read_long_data"); DBUG_PRINT("enter", ("length: %lu", length)); @@ -2946,11 +2946,11 @@ static my_bool read_long_data(MARIA_HA *info, byte *to, ulong length, # Error code */ -int _ma_read_block_record2(MARIA_HA *info, byte *record, - byte *data, byte *end_of_data) +int _ma_read_block_record2(MARIA_HA *info, uchar *record, + uchar *data, uchar *end_of_data) { MARIA_SHARE *share= info->s; - byte *field_length_data, *blob_buffer, *start_of_data; + uchar *field_length_data, *blob_buffer, *start_of_data; uint flag, null_bytes, cur_null_bytes, row_extents, field_lengths; my_bool found_blob= 0; MARIA_EXTENT_CURSOR extent; @@ -3075,7 +3075,7 @@ int _ma_read_block_record2(MARIA_HA *info, byte *record, column < end_column; column++) { enum en_fieldtype type= (enum en_fieldtype) column->type; - byte *field_pos= record + column->offset; + uchar *field_pos= record + column->offset; /* First check if field is present in record */ if ((record[column->null_pos] & column->null_bit) || (info->cur_row.empty_bits[column->empty_pos] & column->empty_bit)) @@ -3146,7 +3146,7 @@ int _ma_read_block_record2(MARIA_HA *info, byte *record, { /* Calculate total length for all blobs */ ulong blob_lengths= 0; - byte *length_data= field_length_data; + uchar *length_data= field_length_data; MARIA_COLUMNDEF *blob_field= column; found_blob= 1; @@ -3169,7 +3169,7 @@ int _ma_read_block_record2(MARIA_HA *info, byte *record, } memcpy(field_pos, field_length_data, size_length); - memcpy_fixed(field_pos + size_length, (byte *) & blob_buffer, + memcpy_fixed(field_pos + size_length, (uchar *) & blob_buffer, sizeof(char*)); field_length_data+= size_length; @@ -3239,10 +3239,10 @@ err: record_pos Record position */ -int _ma_read_block_record(MARIA_HA *info, byte *record, +int _ma_read_block_record(MARIA_HA *info, uchar *record, MARIA_RECORD_POS record_pos) { - byte *data, *end_of_data, *buff; + uchar *data, *end_of_data, *buff; uint offset; uint block_size= info->s->block_size; DBUG_ENTER("_ma_read_block_record"); @@ -3271,10 +3271,10 @@ int _ma_read_block_record(MARIA_HA *info, byte *record, /* compare unique constraint between stored rows */ my_bool _ma_cmp_block_unique(MARIA_HA *info, MARIA_UNIQUEDEF *def, - const byte *record, MARIA_RECORD_POS pos) + const uchar *record, MARIA_RECORD_POS pos) { - byte *org_rec_buff, *old_record; - my_size_t org_rec_buff_size; + uchar *org_rec_buff, *old_record; + size_t org_rec_buff_size; int error; DBUG_ENTER("_ma_cmp_block_unique"); @@ -3333,7 +3333,7 @@ my_bool _ma_scan_init_block_record(MARIA_HA *info) */ if (!(info->scan.bitmap_buff || ((info->scan.bitmap_buff= - (byte *) my_malloc(info->s->block_size * 2, MYF(MY_WME)))))) + (uchar *) my_malloc(info->s->block_size * 2, MYF(MY_WME)))))) DBUG_RETURN(1); info->scan.page_buff= info->scan.bitmap_buff + info->s->block_size; info->scan.bitmap_end= info->scan.bitmap_buff + info->s->bitmap.total_size; @@ -3387,7 +3387,7 @@ void _ma_scan_end_block_record(MARIA_HA *info) # Error code */ -int _ma_scan_block_record(MARIA_HA *info, byte *record, +int _ma_scan_block_record(MARIA_HA *info, uchar *record, MARIA_RECORD_POS record_pos, my_bool skip_deleted __attribute__ ((unused))) { @@ -3401,7 +3401,7 @@ restart_record_read: if (likely(record_pos < info->scan.number_of_rows)) { uint length, offset; - byte *data, *end_of_data; + uchar *data, *end_of_data; while (!(offset= uint2korr(info->scan.dir))) { @@ -3433,7 +3433,7 @@ restart_bitmap_scan: block_size= share->block_size; if (likely(info->scan.bitmap_pos < info->scan.bitmap_end)) { - byte *data= info->scan.bitmap_pos; + uchar *data= info->scan.bitmap_pos; longlong bits= info->scan.bits; uint bit_pos= info->scan.bit_pos; @@ -3520,7 +3520,7 @@ err: */ my_bool _ma_compare_block_record(MARIA_HA *info __attribute__ ((unused)), - const byte *record __attribute__ ((unused))) + const uchar *record __attribute__ ((unused))) { return 0; } @@ -3528,11 +3528,11 @@ my_bool _ma_compare_block_record(MARIA_HA *info __attribute__ ((unused)), #ifndef DBUG_OFF -static void _ma_print_directory(byte *buff, uint block_size) +static void _ma_print_directory(uchar *buff, uint block_size) { uint max_entry= (uint) ((uchar *) buff)[DIR_COUNT_OFFSET], row= 0; uint end_of_prev_row= PAGE_HEADER_SIZE; - byte *dir, *end; + uchar *dir, *end; dir= buff + block_size - DIR_ENTRY_SIZE * max_entry - PAGE_SUFFIX_SIZE; end= buff + block_size - DIR_ENTRY_SIZE - PAGE_SUFFIX_SIZE; @@ -3649,7 +3649,7 @@ uint ma_calc_length_for_store_length(ulong nr) log_parts_count contains number of used log_parts */ -static size_t fill_insert_undo_parts(MARIA_HA *info, const byte *record, +static size_t fill_insert_undo_parts(MARIA_HA *info, const uchar *record, LEX_STRING *log_parts, uint *log_parts_count) { @@ -3755,7 +3755,7 @@ static size_t fill_insert_undo_parts(MARIA_HA *info, const byte *record, /* Add blobs */ for (end_column+= share->base.blobs; column < end_column; column++) { - const byte *field_pos= record + column->offset; + const uchar *field_pos= record + column->offset; uint size_length= column->length - portable_sizeof_char_ptr; ulong blob_length= _ma_calc_blob_length(size_length, field_pos); @@ -3766,7 +3766,7 @@ static size_t fill_insert_undo_parts(MARIA_HA *info, const byte *record, if (blob_length) { char *blob_pos; - memcpy_fixed((byte*) &blob_pos, record + column->offset + size_length, + memcpy_fixed((uchar*) &blob_pos, record + column->offset + size_length, sizeof(blob_pos)); log_parts->str= blob_pos; log_parts->length= blob_length; @@ -3818,8 +3818,8 @@ static size_t fill_insert_undo_parts(MARIA_HA *info, const byte *record, log_parts_count contains number of used log_parts */ -static size_t fill_update_undo_parts(MARIA_HA *info, const byte *oldrec, - const byte *newrec, +static size_t fill_update_undo_parts(MARIA_HA *info, const uchar *oldrec, + const uchar *newrec, LEX_STRING *log_parts, uint *log_parts_count) { @@ -3950,13 +3950,13 @@ static size_t fill_update_undo_parts(MARIA_HA *info, const byte *oldrec, { uint size_length= column->length - portable_sizeof_char_ptr; old_column_length= _ma_calc_blob_length(size_length, old_column_pos); - memcpy_fixed((byte*) &old_column_pos, + memcpy_fixed((uchar*) &old_column_pos, oldrec + column->offset + size_length, sizeof(old_column_pos)); if (!new_column_is_empty) { new_column_length= _ma_calc_blob_length(size_length, new_column_pos); - memcpy_fixed((byte*) &new_column_pos, + memcpy_fixed((uchar*) &new_column_pos, newrec + column->offset + size_length, sizeof(old_column_pos)); } diff --git a/storage/maria/ma_blockrec.h b/storage/maria/ma_blockrec.h index 819d1c2e4d2..e9364f71069 100644 --- a/storage/maria/ma_blockrec.h +++ b/storage/maria/ma_blockrec.h @@ -60,7 +60,7 @@ enum en_page_type { UNALLOCATED_PAGE, HEAD_PAGE, TAIL_PAGE, BLOB_PAGE, MAX_PAGE_ #define PAGE_CAN_BE_COMPACTED 128 /* Bit in PAGE_TYPE */ -/* Bits used for flag byte (one byte, first in record) */ +/* Bits used for flag uchar (one byte, first in record) */ #define ROW_FLAG_TRANSID 1 #define ROW_FLAG_VER_PTR 2 #define ROW_FLAG_DELETE_TRANSID 4 @@ -82,7 +82,7 @@ enum en_page_type { UNALLOCATED_PAGE, HEAD_PAGE, TAIL_PAGE, BLOB_PAGE, MAX_PAGE_ TRANSID_SIZE + VERPTR_SIZE + \ TRANSID_SIZE) -/* We use 1 byte in record header to store number of directory entries */ +/* We use 1 uchar in record header to store number of directory entries */ #define MAX_ROWS_PER_PAGE 255 /* Bits for MARIA_BITMAP_BLOCKS->used */ @@ -133,25 +133,25 @@ my_bool _ma_init_block_record(MARIA_HA *info); void _ma_end_block_record(MARIA_HA *info); my_bool _ma_update_block_record(MARIA_HA *info, MARIA_RECORD_POS pos, - const byte *oldrec, const byte *newrec); -my_bool _ma_delete_block_record(MARIA_HA *info, const byte *record); -int _ma_read_block_record(MARIA_HA *info, byte *record, + const uchar *oldrec, const uchar *newrec); +my_bool _ma_delete_block_record(MARIA_HA *info, const uchar *record); +int _ma_read_block_record(MARIA_HA *info, uchar *record, MARIA_RECORD_POS record_pos); -int _ma_read_block_record2(MARIA_HA *info, byte *record, - byte *data, byte *end_of_data); -int _ma_scan_block_record(MARIA_HA *info, byte *record, +int _ma_read_block_record2(MARIA_HA *info, uchar *record, + uchar *data, uchar *end_of_data); +int _ma_scan_block_record(MARIA_HA *info, uchar *record, MARIA_RECORD_POS, my_bool); my_bool _ma_cmp_block_unique(MARIA_HA *info, MARIA_UNIQUEDEF *def, - const byte *record, MARIA_RECORD_POS pos); + const uchar *record, MARIA_RECORD_POS pos); my_bool _ma_scan_init_block_record(MARIA_HA *info); void _ma_scan_end_block_record(MARIA_HA *info); MARIA_RECORD_POS _ma_write_init_block_record(MARIA_HA *info, - const byte *record); -my_bool _ma_write_block_record(MARIA_HA *info, const byte *record); + const uchar *record); +my_bool _ma_write_block_record(MARIA_HA *info, const uchar *record); my_bool _ma_write_abort_block_record(MARIA_HA *info); my_bool _ma_compare_block_record(register MARIA_HA *info, - register const byte *record); + register const uchar *record); /* ma_bitmap.c */ my_bool _ma_bitmap_init(MARIA_SHARE *share, File file); @@ -160,7 +160,7 @@ my_bool _ma_flush_bitmap(MARIA_SHARE *share); my_bool _ma_bitmap_find_place(MARIA_HA *info, MARIA_ROW *row, MARIA_BITMAP_BLOCKS *result_blocks); my_bool _ma_bitmap_release_unused(MARIA_HA *info, MARIA_BITMAP_BLOCKS *blocks); -my_bool _ma_bitmap_free_full_pages(MARIA_HA *info, const byte *extents, +my_bool _ma_bitmap_free_full_pages(MARIA_HA *info, const uchar *extents, uint count); my_bool _ma_bitmap_set(MARIA_HA *info, ulonglong pos, my_bool head, uint empty_space); diff --git a/storage/maria/ma_cache.c b/storage/maria/ma_cache.c index e8a4b20571b..44aa7a15058 100644 --- a/storage/maria/ma_cache.c +++ b/storage/maria/ma_cache.c @@ -35,7 +35,7 @@ #include "maria_def.h" -int _ma_read_cache(IO_CACHE *info, byte *buff, my_off_t pos, uint length, +int _ma_read_cache(IO_CACHE *info, uchar *buff, my_off_t pos, uint length, int flag) { uint read_length,in_buff_length; @@ -61,7 +61,7 @@ int _ma_read_cache(IO_CACHE *info, byte *buff, my_off_t pos, uint length, (my_off_t) (info->read_end - info->request_pos)) { in_buff_pos=info->request_pos+(uint) offset; - in_buff_length= min(length,(uint) (info->read_end-in_buff_pos)); + in_buff_length= min(length,(uint) ((char*)(info->read_end)-in_buff_pos)); memcpy(buff,info->request_pos+(uint) offset,(size_t) in_buff_length); if (!(length-=in_buff_length)) DBUG_RETURN(0); diff --git a/storage/maria/ma_check.c b/storage/maria/ma_check.c index 0fc2b77304d..98c686d724b 100644 --- a/storage/maria/ma_check.c +++ b/storage/maria/ma_check.c @@ -20,7 +20,7 @@ There are two types of checksums. Table checksum and row checksum. - Row checksum is an additional byte at the end of dynamic length + Row checksum is an additional uchar at the end of dynamic length records. It must be calculated if the table is configured for them. Otherwise they must not be used. The variable MYISAM_SHARE::calc_checksum determines if row checksums are used. @@ -31,7 +31,7 @@ wrong. But since all threads read through a common read buffer, it is sufficient if only one thread checks it. - Table checksum is an eight byte value in the header of the index file. + Table checksum is an eight uchar value in the header of the index file. It can be calculated even if row checksums are not used. The variable MI_CHECK::glob_crc is calculated over all records. MI_SORT_PARAM::calc_checksum determines if this should be done. This @@ -59,7 +59,7 @@ static int check_k_link(HA_CHECK *param, MARIA_HA *info, my_off_t next_link); static int chk_index(HA_CHECK *param, MARIA_HA *info,MARIA_KEYDEF *keyinfo, - my_off_t page, byte *buff, ha_rows *keys, + my_off_t page, uchar *buff, ha_rows *keys, ha_checksum *key_checksum, uint level); static uint isam_key_length(MARIA_HA *info,MARIA_KEYDEF *keyinfo); static ha_checksum calc_checksum(ha_rows count); @@ -67,30 +67,30 @@ static int writekeys(MARIA_SORT_PARAM *sort_param); static int sort_one_index(HA_CHECK *param, MARIA_HA *info, MARIA_KEYDEF *keyinfo, my_off_t pagepos, File new_file); -static int sort_key_read(MARIA_SORT_PARAM *sort_param, byte *key); -static int sort_maria_ft_key_read(MARIA_SORT_PARAM *sort_param, byte *key); +static int sort_key_read(MARIA_SORT_PARAM *sort_param, uchar *key); +static int sort_maria_ft_key_read(MARIA_SORT_PARAM *sort_param, uchar *key); static int sort_get_next_record(MARIA_SORT_PARAM *sort_param); static int sort_key_cmp(MARIA_SORT_PARAM *sort_param, const void *a, const void *b); static int sort_maria_ft_key_write(MARIA_SORT_PARAM *sort_param, - const byte *a); -static int sort_key_write(MARIA_SORT_PARAM *sort_param, const byte *a); + const uchar *a); +static int sort_key_write(MARIA_SORT_PARAM *sort_param, const uchar *a); static my_off_t get_record_for_key(MARIA_HA *info,MARIA_KEYDEF *keyinfo, - const byte *key); + const uchar *key); static int sort_insert_key(MARIA_SORT_PARAM *sort_param, reg1 SORT_KEY_BLOCKS *key_block, - const byte *key, my_off_t prev_block); + const uchar *key, my_off_t prev_block); static int sort_delete_record(MARIA_SORT_PARAM *sort_param); /*static int _ma_flush_pending_blocks(HA_CHECK *param);*/ static SORT_KEY_BLOCKS *alloc_key_blocks(HA_CHECK *param, uint blocks, uint buffer_length); -static ha_checksum maria_byte_checksum(const byte *buf, uint length); +static ha_checksum maria_byte_checksum(const uchar *buf, uint length); static void set_data_file_type(MARIA_SORT_INFO *sort_info, MARIA_SHARE *share); static void restore_data_file_type(MARIA_SHARE *share); void maria_chk_init(HA_CHECK *param) { - bzero((gptr) param,sizeof(*param)); + bzero((uchar*) param,sizeof(*param)); param->opt_follow_links=1; param->keys_in_use= ~(ulonglong) 0; param->search_after_block=HA_OFFSET_ERROR; @@ -297,7 +297,7 @@ static int check_k_link(HA_CHECK *param, register MARIA_HA *info, if (!(buff= pagecache_read(info->s->pagecache, &info->s->kfile, next_link/block_size, DFLT_INIT_HITS, - (byte*) info->buff, + (uchar*) info->buff, PAGECACHE_PLAIN_PAGE, PAGECACHE_LOCK_LEFT_UNLOCKED, 0))) { @@ -532,7 +532,7 @@ int maria_chk_key(HA_CHECK *param, register MARIA_HA *info) /* Check that there isn't a row with auto_increment = 0 in the table */ maria_extra(info,HA_EXTRA_KEYREAD,0); bzero(info->lastkey,keyinfo->seg->length); - if (!maria_rkey(info, info->rec_buff, key, (const byte*) info->lastkey, + if (!maria_rkey(info, info->rec_buff, key, (const uchar*) info->lastkey, keyinfo->seg->length, HA_READ_KEY_EXACT)) { /* Don't count this as a real warning, as maria_chk can't correct it */ @@ -586,7 +586,7 @@ do_stat: static int chk_index_down(HA_CHECK *param, MARIA_HA *info, MARIA_KEYDEF *keyinfo, - my_off_t page, byte *buff, ha_rows *keys, + my_off_t page, uchar *buff, ha_rows *keys, ha_checksum *key_checksum, uint level) { char llbuff[22],llbuff2[22]; @@ -657,7 +657,7 @@ err: static void maria_collect_stats_nonulls_first(HA_KEYSEG *keyseg, ulonglong *notnull, - const byte *key) + const uchar *key) { uint first_null, kp; first_null= ha_find_null(keyseg, (uchar*) key) - keyseg; @@ -697,8 +697,8 @@ void maria_collect_stats_nonulls_first(HA_KEYSEG *keyseg, ulonglong *notnull, static int maria_collect_stats_nonulls_next(HA_KEYSEG *keyseg, ulonglong *notnull, - const byte *prev_key, - const byte *last_key) + const uchar *prev_key, + const uchar *last_key) { uint diffs[2]; uint first_null_seg, kp; @@ -733,23 +733,23 @@ int maria_collect_stats_nonulls_next(HA_KEYSEG *keyseg, ulonglong *notnull, /* Check if index is ok */ static int chk_index(HA_CHECK *param, MARIA_HA *info, MARIA_KEYDEF *keyinfo, - my_off_t page, byte *buff, ha_rows *keys, + my_off_t page, uchar *buff, ha_rows *keys, ha_checksum *key_checksum, uint level) { int flag; uint used_length,comp_flag,nod_flag,key_length=0; - byte key[HA_MAX_POSSIBLE_KEY_BUFF],*temp_buff,*keypos,*old_keypos,*endpos; + uchar key[HA_MAX_POSSIBLE_KEY_BUFF],*temp_buff,*keypos,*old_keypos,*endpos; my_off_t next_page,record; char llbuff[22]; uint diff_pos[2]; DBUG_ENTER("chk_index"); - DBUG_DUMP("buff",(byte*) buff,maria_data_on_page(buff)); + DBUG_DUMP("buff",(uchar*) buff,maria_data_on_page(buff)); /* TODO: implement appropriate check for RTree keys */ if (keyinfo->flag & HA_SPATIAL) DBUG_RETURN(0); - if (!(temp_buff=(byte*) my_alloca((uint) keyinfo->block_length))) + if (!(temp_buff=(uchar*) my_alloca((uint) keyinfo->block_length))) { _ma_check_print_error(param,"Not enough memory for keyblock"); DBUG_RETURN(-1); @@ -837,7 +837,7 @@ static int chk_index(HA_CHECK *param, MARIA_HA *info, MARIA_KEYDEF *keyinfo, key); } } - (*key_checksum)+= maria_byte_checksum((byte*) key, + (*key_checksum)+= maria_byte_checksum((uchar*) key, key_length- info->s->rec_reflength); record= _ma_dpos(info,0,key+key_length); if (keyinfo->flag & HA_FULLTEXT) /* special handling for ft2 */ @@ -875,7 +875,7 @@ static int chk_index(HA_CHECK *param, MARIA_HA *info, MARIA_KEYDEF *keyinfo, DBUG_PRINT("test",("page: %s record: %s filelength: %s", llstr(page,llbuff),llstr(record,llbuff2), llstr(info->state->data_file_length,llbuff3))); - DBUG_DUMP("key",(byte*) key,key_length); + DBUG_DUMP("key",(uchar*) key,key_length); DBUG_DUMP("new_in_page",(char*) old_keypos,(uint) (keypos-old_keypos)); goto err; } @@ -887,10 +887,10 @@ static int chk_index(HA_CHECK *param, MARIA_HA *info, MARIA_KEYDEF *keyinfo, llstr(page,llbuff), used_length, (keypos - buff)); goto err; } - my_afree((byte*) temp_buff); + my_afree((uchar*) temp_buff); DBUG_RETURN(0); err: - my_afree((byte*) temp_buff); + my_afree((uchar*) temp_buff); DBUG_RETURN(1); } /* chk_index */ @@ -969,7 +969,7 @@ static void record_pos_to_txt(MARIA_HA *info, my_off_t recpos, */ static int check_keys_in_record(HA_CHECK *param, MARIA_HA *info, int extend, - my_off_t start_recpos, byte *record) + my_off_t start_recpos, uchar *record) { MARIA_KEYDEF *keyinfo; char llbuff[22+4]; @@ -1019,7 +1019,7 @@ static int check_keys_in_record(HA_CHECK *param, MARIA_HA *info, int extend, } else param->tmp_key_crc[key]+= - maria_byte_checksum((byte*) info->lastkey, key_length); + maria_byte_checksum((uchar*) info->lastkey, key_length); } } } @@ -1040,7 +1040,7 @@ static int check_keys_in_record(HA_CHECK *param, MARIA_HA *info, int extend, */ static int check_static_record(HA_CHECK *param, MARIA_HA *info, int extend, - byte *record) + uchar *record) { my_off_t start_recpos, pos; char llbuff[22]; @@ -1050,7 +1050,7 @@ static int check_static_record(HA_CHECK *param, MARIA_HA *info, int extend, { if (*_ma_killed_ptr(param)) return -1; - if (my_b_read(¶m->read_cache,(byte*) record, + if (my_b_read(¶m->read_cache,(uchar*) record, info->s->base.pack_reclength)) { _ma_check_print_error(param, @@ -1077,11 +1077,11 @@ static int check_static_record(HA_CHECK *param, MARIA_HA *info, int extend, static int check_dynamic_record(HA_CHECK *param, MARIA_HA *info, int extend, - byte *record) + uchar *record) { MARIA_BLOCK_INFO block_info; my_off_t start_recpos, start_block, pos; - byte *to; + uchar *to; ulong left_length; uint b_type; char llbuff[22],llbuff2[22],llbuff3[22]; @@ -1103,7 +1103,7 @@ static int check_dynamic_record(HA_CHECK *param, MARIA_HA *info, int extend, block_info.next_filepos=pos; do { - if (_ma_read_cache(¶m->read_cache,(byte*) block_info.header, + if (_ma_read_cache(¶m->read_cache,(uchar*) block_info.header, (start_block=block_info.next_filepos), sizeof(block_info.header), (flag ? 0 : READING_NEXT) | READING_HEADER)) @@ -1215,7 +1215,7 @@ static int check_dynamic_record(HA_CHECK *param, MARIA_HA *info, int extend, got_error=1; break; } - if (_ma_read_cache(¶m->read_cache,(byte*) to,block_info.filepos, + if (_ma_read_cache(¶m->read_cache,(uchar*) to,block_info.filepos, (uint) block_info.data_len, flag == 1 ? READING_NEXT : 0)) { @@ -1297,7 +1297,7 @@ next:; static int check_compressed_record(HA_CHECK *param, MARIA_HA *info, int extend, - byte *record) + uchar *record) { my_off_t start_recpos, pos; char llbuff[22]; @@ -1311,7 +1311,7 @@ static int check_compressed_record(HA_CHECK *param, MARIA_HA *info, int extend, if (*_ma_killed_ptr(param)) DBUG_RETURN(-1); - if (_ma_read_cache(¶m->read_cache,(byte*) block_info.header, pos, + if (_ma_read_cache(¶m->read_cache,(uchar*) block_info.header, pos, info->s->pack.ref_length, READING_NEXT)) { _ma_check_print_error(param, @@ -1335,7 +1335,7 @@ static int check_compressed_record(HA_CHECK *param, MARIA_HA *info, int extend, got_error=1; goto end; } - if (_ma_read_cache(¶m->read_cache,(byte*) info->rec_buff, + if (_ma_read_cache(¶m->read_cache,(uchar*) info->rec_buff, block_info.filepos, block_info.rec_len, READING_NEXT)) { _ma_check_print_error(param, @@ -1380,12 +1380,12 @@ end: */ static int check_page_layout(HA_CHECK *param, MARIA_HA *info, - my_off_t page_pos, byte *page, + my_off_t page_pos, uchar *page, uint row_count, uint head_empty, uint *real_rows_found) { uint empty, last_row_end, row, first_dir_entry; - byte *dir_entry; + uchar *dir_entry; char llbuff[22]; DBUG_ENTER("check_page_layout"); @@ -1460,11 +1460,11 @@ static int check_page_layout(HA_CHECK *param, MARIA_HA *info, */ -static my_bool check_head_page(HA_CHECK *param, MARIA_HA *info, byte *record, - int extend, my_off_t page_pos, byte *page_buff, +static my_bool check_head_page(HA_CHECK *param, MARIA_HA *info, uchar *record, + int extend, my_off_t page_pos, uchar *page_buff, uint row_count) { - byte *dir_entry; + uchar *dir_entry; uint row; char llbuff[22], llbuff2[22]; DBUG_ENTER("check_head_page"); @@ -1512,7 +1512,7 @@ static my_bool check_head_page(HA_CHECK *param, MARIA_HA *info, byte *record, } if (info->cur_row.extents_count) { - byte *extents= info->cur_row.extents; + uchar *extents= info->cur_row.extents; uint i; /* Check that bitmap has the right marker for the found extents */ for (i= 0 ; i < info->cur_row.extents_count ; i++) @@ -1562,10 +1562,10 @@ static my_bool check_head_page(HA_CHECK *param, MARIA_HA *info, byte *record, */ static int check_block_record(HA_CHECK *param, MARIA_HA *info, int extend, - byte *record) + uchar *record) { my_off_t pos; - byte *page_buff, *bitmap_buff, *data; + uchar *page_buff, *bitmap_buff, *data; char llbuff[22], llbuff2[22]; uint block_size= info->s->block_size; ha_rows full_page_count, tail_count; @@ -1737,7 +1737,7 @@ err: int maria_chk_data_link(HA_CHECK *param, MARIA_HA *info,int extend) { int error; - byte *record; + uchar *record; char llbuff[22],llbuff2[22],llbuff3[22]; DBUG_ENTER("maria_chk_data_link"); @@ -1749,7 +1749,7 @@ int maria_chk_data_link(HA_CHECK *param, MARIA_HA *info,int extend) puts("- check record links"); } - if (!(record= (byte*) my_malloc(info->s->base.pack_reclength,MYF(0)))) + if (!(record= (uchar*) my_malloc(info->s->base.pack_reclength,MYF(0)))) { _ma_check_print_error(param,"Not enough memory for record"); DBUG_RETURN(-1); @@ -1898,11 +1898,11 @@ int maria_chk_data_link(HA_CHECK *param, MARIA_HA *info,int extend) printf("Lost space: %12s Linkdata: %10s\n", llstr(param->empty, llbuff),llstr(param->link_used, llbuff2)); } - my_free((gptr) record,MYF(0)); + my_free((uchar*) record,MYF(0)); DBUG_RETURN (error); err: - my_free((gptr) record,MYF(0)); + my_free((uchar*) record,MYF(0)); param->testflag|=T_RETRY_WITHOUT_QUICK; DBUG_RETURN(1); } /* maria_chk_data_link */ @@ -1912,7 +1912,7 @@ int maria_chk_data_link(HA_CHECK *param, MARIA_HA *info,int extend) /* Save new datafile-name in temp_filename */ int maria_repair(HA_CHECK *param, register MARIA_HA *info, - my_string name, int rep_quick) + char *name, int rep_quick) { int error,got_error; uint i; @@ -1957,7 +1957,7 @@ int maria_repair(HA_CHECK *param, register MARIA_HA *info, MYF(MY_WME | MY_WAIT_IF_FULL))) goto err; info->opt_flag|=WRITE_CACHE_USED; - if (!(sort_param.record=(byte*) my_malloc((uint) share->base.pack_reclength, + if (!(sort_param.record=(uchar*) my_malloc((uint) share->base.pack_reclength, MYF(0))) || _ma_alloc_buffer(&sort_param.rec_buff, &sort_param.rec_buff_size, info->s->base.default_rec_buff_size)) @@ -2038,7 +2038,7 @@ int maria_repair(HA_CHECK *param, register MARIA_HA *info, { if (my_errno != HA_ERR_FOUND_DUPP_KEY) goto err; - DBUG_DUMP("record",(byte*) sort_param.record,share->base.pack_reclength); + DBUG_DUMP("record",(uchar*) sort_param.record,share->base.pack_reclength); _ma_check_print_info(param,"Duplicate key %2d for record at %10s against new record at %10s", info->errkey+1, llstr(sort_param.start_recpos,llbuff), @@ -2180,9 +2180,9 @@ err: static int writekeys(MARIA_SORT_PARAM *sort_param) { register uint i; - byte *key; + uchar *key; MARIA_HA *info= sort_param->sort_info->info; - byte *buff= sort_param->record; + uchar *buff= sort_param->record; my_off_t filepos= sort_param->filepos; DBUG_ENTER("writekeys"); @@ -2246,12 +2246,12 @@ static int writekeys(MARIA_SORT_PARAM *sort_param) /* Change all key-pointers that points to a records */ -int maria_movepoint(register MARIA_HA *info, byte *record, +int maria_movepoint(register MARIA_HA *info, uchar *record, MARIA_RECORD_POS oldpos, MARIA_RECORD_POS newpos, uint prot_key) { register uint i; - byte *key; + uchar *key; uint key_length; DBUG_ENTER("maria_movepoint"); @@ -2323,7 +2323,7 @@ int _ma_flush_blocks(HA_CHECK *param, PAGECACHE *pagecache, /* Sort index for more efficent reads */ -int maria_sort_index(HA_CHECK *param, register MARIA_HA *info, my_string name) +int maria_sort_index(HA_CHECK *param, register MARIA_HA *info, char *name) { reg2 uint key; reg1 MARIA_KEYDEF *keyinfo; @@ -2430,8 +2430,8 @@ static int sort_one_index(HA_CHECK *param, MARIA_HA *info, my_off_t pagepos, File new_file) { uint length,nod_flag,used_length, key_length; - byte *buff,*keypos,*endpos; - byte key[HA_MAX_POSSIBLE_KEY_BUFF]; + uchar *buff,*keypos,*endpos; + uchar key[HA_MAX_POSSIBLE_KEY_BUFF]; my_off_t new_page_pos,next_page; char llbuff[22]; DBUG_ENTER("sort_one_index"); @@ -2441,7 +2441,7 @@ static int sort_one_index(HA_CHECK *param, MARIA_HA *info, new_page_pos=param->new_file_pos; param->new_file_pos+=keyinfo->block_length; - if (!(buff= (byte*) my_alloca((uint) keyinfo->block_length))) + if (!(buff= (uchar*) my_alloca((uint) keyinfo->block_length))) { _ma_check_print_error(param,"Not enough memory for key block"); DBUG_RETURN(-1); @@ -2470,7 +2470,7 @@ static int sort_one_index(HA_CHECK *param, MARIA_HA *info, ("From page: %ld, keyoffset: %lu used_length: %d", (ulong) pagepos, (ulong) (keypos - buff), (int) used_length)); - DBUG_DUMP("buff",(byte*) buff,used_length); + DBUG_DUMP("buff",(uchar*) buff,used_length); goto err; } } @@ -2499,17 +2499,17 @@ static int sort_one_index(HA_CHECK *param, MARIA_HA *info, /* Fill block with zero and write it to the new index file */ length= maria_data_on_page(buff); - bzero((byte*) buff+length,keyinfo->block_length-length); - if (my_pwrite(new_file,(byte*) buff,(uint) keyinfo->block_length, + bzero((uchar*) buff+length,keyinfo->block_length-length); + if (my_pwrite(new_file,(uchar*) buff,(uint) keyinfo->block_length, new_page_pos,MYF(MY_NABP | MY_WAIT_IF_FULL))) { _ma_check_print_error(param,"Can't write indexblock, error: %d",my_errno); goto err; } - my_afree((gptr) buff); + my_afree((uchar*) buff); DBUG_RETURN(0); err: - my_afree((gptr) buff); + my_afree((uchar*) buff); DBUG_RETURN(1); } /* sort_one_index */ @@ -2560,13 +2560,13 @@ int maria_filecopy(HA_CHECK *param, File to,File from,my_off_t start, VOID(my_seek(from,start,MY_SEEK_SET,MYF(0))); while (length > buff_length) { - if (my_read(from,(byte*) buff,buff_length,MYF(MY_NABP)) || - my_write(to,(byte*) buff,buff_length,param->myf_rw)) + if (my_read(from,(uchar*) buff,buff_length,MYF(MY_NABP)) || + my_write(to,(uchar*) buff,buff_length,param->myf_rw)) goto err; length-= buff_length; } - if (my_read(from,(byte*) buff,(uint) length,MYF(MY_NABP)) || - my_write(to,(byte*) buff,(uint) length,param->myf_rw)) + if (my_read(from,(uchar*) buff,(uint) length,MYF(MY_NABP)) || + my_write(to,(uchar*) buff,(uint) length,param->myf_rw)) goto err; if (buff != tmp_buff) my_free(buff,MYF(0)); @@ -2649,7 +2649,7 @@ int maria_repair_by_sort(HA_CHECK *param, register MARIA_HA *info, info->opt_flag|=WRITE_CACHE_USED; info->rec_cache.file= info->dfile.file; /* for sort_delete_record */ - if (!(sort_param.record=(byte*) my_malloc((uint) share->base.pack_reclength, + if (!(sort_param.record=(uchar*) my_malloc((uint) share->base.pack_reclength, MYF(0))) || _ma_alloc_buffer(&sort_param.rec_buff, &sort_param.rec_buff_size, info->s->base.default_rec_buff_size)) @@ -2958,8 +2958,8 @@ err: my_free(sort_param.rec_buff, MYF(MY_ALLOW_ZERO_PTR)); my_free(sort_param.record,MYF(MY_ALLOW_ZERO_PTR)); - my_free((gptr) sort_info.key_block,MYF(MY_ALLOW_ZERO_PTR)); - my_free((gptr) sort_info.ft_buf, MYF(MY_ALLOW_ZERO_PTR)); + my_free((uchar*) sort_info.key_block,MYF(MY_ALLOW_ZERO_PTR)); + my_free((uchar*) sort_info.ft_buf, MYF(MY_ALLOW_ZERO_PTR)); my_free(sort_info.buff,MYF(MY_ALLOW_ZERO_PTR)); VOID(end_io_cache(¶m->read_cache)); info->opt_flag&= ~(READ_CACHE_USED | WRITE_CACHE_USED); @@ -3485,9 +3485,9 @@ err: pthread_cond_destroy (&sort_info.cond); pthread_mutex_destroy(&sort_info.mutex); - my_free((gptr) sort_info.ft_buf, MYF(MY_ALLOW_ZERO_PTR)); - my_free((gptr) sort_info.key_block,MYF(MY_ALLOW_ZERO_PTR)); - my_free((gptr) sort_param,MYF(MY_ALLOW_ZERO_PTR)); + my_free((uchar*) sort_info.ft_buf, MYF(MY_ALLOW_ZERO_PTR)); + my_free((uchar*) sort_info.key_block,MYF(MY_ALLOW_ZERO_PTR)); + my_free((uchar*) sort_param,MYF(MY_ALLOW_ZERO_PTR)); my_free(sort_info.buff,MYF(MY_ALLOW_ZERO_PTR)); VOID(end_io_cache(¶m->read_cache)); info->opt_flag&= ~(READ_CACHE_USED | WRITE_CACHE_USED); @@ -3499,7 +3499,7 @@ err: /* Read next record and return next key */ -static int sort_key_read(MARIA_SORT_PARAM *sort_param, byte *key) +static int sort_key_read(MARIA_SORT_PARAM *sort_param, uchar *key) { int error; MARIA_SORT_INFO *sort_info= sort_param->sort_info; @@ -3527,7 +3527,7 @@ static int sort_key_read(MARIA_SORT_PARAM *sort_param, byte *key) } /* sort_key_read */ -static int sort_maria_ft_key_read(MARIA_SORT_PARAM *sort_param, byte *key) +static int sort_maria_ft_key_read(MARIA_SORT_PARAM *sort_param, uchar *key) { int error; MARIA_SORT_INFO *sort_info=sort_param->sort_info; @@ -3667,7 +3667,7 @@ static int sort_get_next_record(MARIA_SORT_PARAM *sort_param) } case DYNAMIC_RECORD: { - byte *to; + uchar *to; LINT_INIT(to); pos=sort_param->pos; searching=(sort_param->fix_datafile && (param->testflag & T_EXTEND)); @@ -3699,7 +3699,7 @@ static int sort_get_next_record(MARIA_SORT_PARAM *sort_param) llstr(param->search_after_block,llbuff), llstr(sort_param->start_recpos,llbuff2)); if (_ma_read_cache(&sort_param->read_cache, - (byte*) block_info.header,pos, + (uchar*) block_info.header,pos, MARIA_BLOCK_INFO_HEADER_LENGTH, (! found_record ? READING_NEXT : 0) | parallel_flag | READING_HEADER)) @@ -3968,7 +3968,7 @@ static int sort_get_next_record(MARIA_SORT_PARAM *sort_param) case COMPRESSED_RECORD: for (searching=0 ;; searching=1, sort_param->pos++) { - if (_ma_read_cache(&sort_param->read_cache,(byte*) block_info.header, + if (_ma_read_cache(&sort_param->read_cache,(uchar*) block_info.header, sort_param->pos, share->pack.ref_length,READING_NEXT)) DBUG_RETURN(-1); @@ -3998,7 +3998,7 @@ static int sort_get_next_record(MARIA_SORT_PARAM *sort_param) llstr(sort_param->pos,llbuff)); continue; } - if (_ma_read_cache(&sort_param->read_cache,(byte*) sort_param->rec_buff, + if (_ma_read_cache(&sort_param->read_cache,(uchar*) sort_param->rec_buff, block_info.filepos, block_info.rec_len, READING_NEXT)) { @@ -4058,8 +4058,8 @@ int _ma_sort_write_record(MARIA_SORT_PARAM *sort_param) int flag; uint length; ulong block_length,reclength; - byte *from; - byte block_buff[8]; + uchar *from; + uchar block_buff[8]; MARIA_SORT_INFO *sort_info=sort_param->sort_info; HA_CHECK *param=sort_info->param; MARIA_HA *info=sort_info->info; @@ -4135,7 +4135,7 @@ int _ma_sort_write_record(MARIA_SORT_PARAM *sort_param) length+= _ma_save_pack_length((uint) share->pack.version, block_buff + length, info->blob_length); if (my_b_write(&info->rec_cache,block_buff,length) || - my_b_write(&info->rec_cache,(byte*) sort_param->rec_buff,reclength)) + my_b_write(&info->rec_cache,(uchar*) sort_param->rec_buff,reclength)) { _ma_check_print_error(param,"%d when writing to datafile",my_errno); DBUG_RETURN(1); @@ -4171,7 +4171,7 @@ static int sort_key_cmp(MARIA_SORT_PARAM *sort_param, const void *a, } /* sort_key_cmp */ -static int sort_key_write(MARIA_SORT_PARAM *sort_param, const byte *a) +static int sort_key_write(MARIA_SORT_PARAM *sort_param, const uchar *a) { uint diff_pos[2]; char llbuff[22],llbuff2[22]; @@ -4244,7 +4244,7 @@ int _ma_sort_ft_buf_flush(MARIA_SORT_PARAM *sort_param) uint val_off, val_len; int error; SORT_FT_BUF *maria_ft_buf=sort_info->ft_buf; - byte *from, *to; + uchar *from, *to; val_len=share->ft2_keyinfo.keylength; get_key_full_length_rdonly(val_off, maria_ft_buf->lastkey); @@ -4283,7 +4283,7 @@ int _ma_sort_ft_buf_flush(MARIA_SORT_PARAM *sort_param) static int sort_maria_ft_key_write(MARIA_SORT_PARAM *sort_param, - const byte *a) + const uchar *a) { uint a_len, val_off, val_len, error; MARIA_SORT_INFO *sort_info= sort_param->sort_info; @@ -4320,7 +4320,7 @@ static int sort_maria_ft_key_write(MARIA_SORT_PARAM *sort_param, ((uchar *)a)+1,a_len-1, (uchar*) ft_buf->lastkey+1,val_off-1, 0, 0)==0) { - byte *p; + uchar *p; if (!ft_buf->buf) /* store in second-level tree */ { ft_buf->count++; @@ -4374,7 +4374,7 @@ word_init_ft_buf: /* get pointer to record from a key */ static my_off_t get_record_for_key(MARIA_HA *info, MARIA_KEYDEF *keyinfo, - const byte *key) + const uchar *key) { return _ma_dpos(info,0, key + _ma_keylength(keyinfo, key)); } /* get_record_for_key */ @@ -4384,12 +4384,12 @@ static my_off_t get_record_for_key(MARIA_HA *info, MARIA_KEYDEF *keyinfo, static int sort_insert_key(MARIA_SORT_PARAM *sort_param, register SORT_KEY_BLOCKS *key_block, - const byte *key, + const uchar *key, my_off_t prev_block) { uint a_length,t_length,nod_flag; my_off_t filepos,key_file_length; - byte *anc_buff,*lastkey; + uchar *anc_buff,*lastkey; MARIA_KEY_PARAM s_temp; MARIA_HA *info; MARIA_KEYDEF *keyinfo=sort_param->keyinfo; @@ -4423,7 +4423,7 @@ static int sort_insert_key(MARIA_SORT_PARAM *sort_param, _ma_kpointer(info,key_block->end_pos,prev_block); t_length=(*keyinfo->pack_key)(keyinfo,nod_flag, - (byte*) 0,lastkey,lastkey,key, + (uchar*) 0,lastkey,lastkey,key, &s_temp); (*keyinfo->store_key)(keyinfo, key_block->end_pos+nod_flag,&s_temp); a_length+=t_length; @@ -4471,7 +4471,7 @@ static int sort_delete_record(MARIA_SORT_PARAM *sort_param) { uint i; int old_file,error; - byte *key; + uchar *key; MARIA_SORT_INFO *sort_info=sort_param->sort_info; HA_CHECK *param=sort_info->param; MARIA_HA *info=sort_info->info; @@ -4589,7 +4589,7 @@ static SORT_KEY_BLOCKS *alloc_key_blocks(HA_CHECK *param, uint blocks, for (i=0 ; i < blocks ; i++) { block[i].inited=0; - block[i].buff= (byte*) (block+blocks)+(buffer_length+IO_SIZE)*i; + block[i].buff= (uchar*) (block+blocks)+(buffer_length+IO_SIZE)*i; } DBUG_RETURN(block); } /* alloc_key_blocks */ @@ -4635,34 +4635,34 @@ int maria_recreate_table(HA_CHECK *param, MARIA_HA **org_info, char *filename) if (!(keyinfo=(MARIA_KEYDEF*) my_alloca(sizeof(MARIA_KEYDEF) * share.base.keys))) DBUG_RETURN(0); - memcpy((byte*) keyinfo,(byte*) share.keyinfo, + memcpy((uchar*) keyinfo,(uchar*) share.keyinfo, (size_t) (sizeof(MARIA_KEYDEF)*share.base.keys)); key_parts= share.base.all_key_parts; if (!(keysegs=(HA_KEYSEG*) my_alloca(sizeof(HA_KEYSEG)* (key_parts+share.base.keys)))) { - my_afree((gptr) keyinfo); + my_afree((uchar*) keyinfo); DBUG_RETURN(1); } if (!(columndef=(MARIA_COLUMNDEF*) my_alloca(sizeof(MARIA_COLUMNDEF)*(share.base.fields+1)))) { - my_afree((gptr) keyinfo); - my_afree((gptr) keysegs); + my_afree((uchar*) keyinfo); + my_afree((uchar*) keysegs); DBUG_RETURN(1); } if (!(uniquedef=(MARIA_UNIQUEDEF*) my_alloca(sizeof(MARIA_UNIQUEDEF)*(share.state.header.uniques+1)))) { - my_afree((gptr) columndef); - my_afree((gptr) keyinfo); - my_afree((gptr) keysegs); + my_afree((uchar*) columndef); + my_afree((uchar*) keyinfo); + my_afree((uchar*) keysegs); DBUG_RETURN(1); } /* Copy the column definitions */ - memcpy((byte*) columndef,(byte*) share.columndef, + memcpy((uchar*) columndef,(uchar*) share.columndef, (size_t) (sizeof(MARIA_COLUMNDEF)*(share.base.fields+1))); for (column=columndef, end= columndef+share.base.fields; column != end ; @@ -4676,7 +4676,7 @@ int maria_recreate_table(HA_CHECK *param, MARIA_HA **org_info, char *filename) } /* Change the new key to point at the saved key segments */ - memcpy((byte*) keysegs,(byte*) share.keyparts, + memcpy((uchar*) keysegs,(uchar*) share.keyparts, (size_t) (sizeof(HA_KEYSEG)*(key_parts+share.base.keys+ share.state.header.uniques))); keyseg=keysegs; @@ -4695,7 +4695,7 @@ int maria_recreate_table(HA_CHECK *param, MARIA_HA **org_info, char *filename) Copy the unique definitions and change them to point at the new key segments */ - memcpy((byte*) uniquedef,(byte*) share.uniqueinfo, + memcpy((uchar*) uniquedef,(uchar*) share.uniqueinfo, (size_t) (sizeof(MARIA_UNIQUEDEF)*(share.state.header.uniques))); for (u_ptr=uniquedef,u_end=uniquedef+share.state.header.uniques; u_ptr != u_end ; u_ptr++) @@ -4787,10 +4787,10 @@ int maria_recreate_table(HA_CHECK *param, MARIA_HA **org_info, char *filename) goto end; error=0; end: - my_afree((gptr) uniquedef); - my_afree((gptr) keyinfo); - my_afree((gptr) columndef); - my_afree((gptr) keysegs); + my_afree((uchar*) uniquedef); + my_afree((uchar*) keyinfo); + my_afree((uchar*) columndef); + my_afree((uchar*) keysegs); DBUG_RETURN(error); } @@ -4894,7 +4894,7 @@ err: void _ma_update_auto_increment_key(HA_CHECK *param, MARIA_HA *info, my_bool repair_only) { - byte *record; + uchar *record; DBUG_ENTER("update_auto_increment_key"); if (!info->s->base.auto_key || @@ -4913,7 +4913,7 @@ void _ma_update_auto_increment_key(HA_CHECK *param, MARIA_HA *info, We have to use an allocated buffer instead of info->rec_buff as _ma_put_key_in_record() may use info->rec_buff */ - if (!(record= (byte*) my_malloc((uint) info->s->base.pack_reclength, + if (!(record= (uchar*) my_malloc((uint) info->s->base.pack_reclength, MYF(0)))) { _ma_check_print_error(param,"Not enough memory for extra record"); @@ -5041,10 +5041,10 @@ void maria_update_key_parts(MARIA_KEYDEF *keyinfo, ulong *rec_per_key_part, } -static ha_checksum maria_byte_checksum(const byte *buf, uint length) +static ha_checksum maria_byte_checksum(const uchar *buf, uint length) { ha_checksum crc; - const byte *end=buf+length; + const uchar *end=buf+length; for (crc=0; buf != end; buf++) crc=((crc << 1) + *((uchar*) buf)) + test(crc & (((ha_checksum) 1) << (8*sizeof(ha_checksum)-1))); diff --git a/storage/maria/ma_checkpoint.c b/storage/maria/ma_checkpoint.c index ed5520f66bf..b23d077897f 100644 --- a/storage/maria/ma_checkpoint.c +++ b/storage/maria/ma_checkpoint.c @@ -182,7 +182,7 @@ my_bool execute_checkpoint_indirect() LSN candidate_max_rec_lsn_at_last_checkpoint; DBUG_ENTER("execute_checkpoint_indirect"); - DBUG_ASSERT(sizeof(byte *) <= 8); + DBUG_ASSERT(sizeof(uchar *) <= 8); DBUG_ASSERT(sizeof(LSN) <= 8); safemutex_assert_owner(log_mutex); diff --git a/storage/maria/ma_checksum.c b/storage/maria/ma_checksum.c index 95555aa3129..30921ad8213 100644 --- a/storage/maria/ma_checksum.c +++ b/storage/maria/ma_checksum.c @@ -17,7 +17,7 @@ #include "maria_def.h" -ha_checksum _ma_checksum(MARIA_HA *info, const byte *record) +ha_checksum _ma_checksum(MARIA_HA *info, const uchar *record) { ha_checksum crc=0; MARIA_COLUMNDEF *column= info->s->columndef; @@ -28,7 +28,7 @@ ha_checksum _ma_checksum(MARIA_HA *info, const byte *record) for ( ; column != column_end ; column++) { - const byte *pos= record + column->offset; + const uchar *pos= record + column->offset; ulong length; switch (column->type) { @@ -63,7 +63,7 @@ ha_checksum _ma_checksum(MARIA_HA *info, const byte *record) } -ha_checksum _ma_static_checksum(MARIA_HA *info, const byte *pos) +ha_checksum _ma_static_checksum(MARIA_HA *info, const uchar *pos) { return my_checksum(0, pos, info->s->base.reclength); } diff --git a/storage/maria/ma_close.c b/storage/maria/ma_close.c index 34c1bfb4d6d..b760d537670 100644 --- a/storage/maria/ma_close.c +++ b/storage/maria/ma_close.c @@ -107,18 +107,18 @@ int maria_close(register MARIA_HA *info) } } #endif - my_free((gptr) info->s,MYF(0)); + my_free((uchar*) info->s,MYF(0)); } pthread_mutex_unlock(&THR_LOCK_maria); if (info->ftparser_param) { - my_free((gptr)info->ftparser_param, MYF(0)); + my_free((uchar*)info->ftparser_param, MYF(0)); info->ftparser_param= 0; } if (info->dfile.file >= 0 && my_close(info->dfile.file, MYF(0))) error = my_errno; - my_free((gptr) info,MYF(0)); + my_free((uchar*) info,MYF(0)); if (error) { diff --git a/storage/maria/ma_create.c b/storage/maria/ma_create.c index 53e15deb74b..e26b49a1d37 100644 --- a/storage/maria/ma_create.c +++ b/storage/maria/ma_create.c @@ -90,7 +90,7 @@ int maria_create(const char *name, enum data_file_type datafile_type, } errpos=0; options=0; - bzero((byte*) &share,sizeof(share)); + bzero((uchar*) &share,sizeof(share)); if (flags & HA_DONT_TOUCH_DATA) { @@ -232,7 +232,7 @@ int maria_create(const char *name, enum data_file_type datafile_type, if ((packed & 7) == 1) { /* - Not optimal packing, try to remove a 1 byte length zero-field as + Not optimal packing, try to remove a 1 uchar length zero-field as this will get same record length, but smaller pack overhead */ while (column != columndef) @@ -401,7 +401,7 @@ int maria_create(const char *name, enum data_file_type datafile_type, } keydef->keysegs+=sp_segs; key_length+=SPLEN*sp_segs; - length++; /* At least one length byte */ + length++; /* At least one length uchar */ min_key_length_skip+=SPLEN*2*SPDIMS; #else my_errno= HA_ERR_UNSUPPORTED; @@ -437,7 +437,7 @@ int maria_create(const char *name, enum data_file_type datafile_type, fulltext_keys++; key_length+= HA_FT_MAXBYTELEN+HA_FT_WLEN; - length++; /* At least one length byte */ + length++; /* At least one length uchar */ min_key_length_skip+=HA_FT_MAXBYTELEN; real_length_diff=HA_FT_MAXBYTELEN-FT_MAX_WORD_LEN_FOR_SORT; } @@ -510,7 +510,7 @@ int maria_create(const char *name, enum data_file_type datafile_type, DBUG_ASSERT(!(keyseg->flag & HA_VAR_LENGTH_PART)); keydef->flag |= HA_SPACE_PACK_USED | HA_VAR_LENGTH_KEY; options|=HA_OPTION_PACK_KEYS; /* Using packed keys */ - length++; /* At least one length byte */ + length++; /* At least one length uchar */ min_key_length_skip+=keyseg->length; if (keyseg->length >= 255) { /* prefix may be 3 bytes */ @@ -523,7 +523,7 @@ int maria_create(const char *name, enum data_file_type datafile_type, DBUG_ASSERT(!test_all_bits(keyseg->flag, (HA_VAR_LENGTH_PART | HA_BLOB_PART))); keydef->flag|=HA_VAR_LENGTH_KEY; - length++; /* At least one length byte */ + length++; /* At least one length uchar */ options|=HA_OPTION_PACK_KEYS; /* Using packed keys */ min_key_length_skip+=keyseg->length; if (keyseg->length >= 255) @@ -612,7 +612,7 @@ int maria_create(const char *name, enum data_file_type datafile_type, goto err_no_lock; } - bmove(share.state.header.file_version,(byte*) maria_file_magic,4); + bmove(share.state.header.file_version,(uchar*) maria_file_magic,4); ci->old_options=options| (ci->old_options & HA_OPTION_TEMP_COMPRESS_RECORD ? HA_OPTION_COMPRESS_RECORD | HA_OPTION_TEMP_COMPRESS_RECORD: 0); @@ -920,11 +920,11 @@ int maria_create(const char *name, enum data_file_type datafile_type, { if (_ma_columndef_write(file, col_order[i])) { - my_free((gptr) col_order, MYF(0)); + my_free((uchar*) col_order, MYF(0)); goto err; } } - my_free((gptr) col_order, MYF(0)); + my_free((uchar*) col_order, MYF(0)); } else { diff --git a/storage/maria/ma_dbug.c b/storage/maria/ma_dbug.c index 150385607b6..a23e7248029 100644 --- a/storage/maria/ma_dbug.c +++ b/storage/maria/ma_dbug.c @@ -20,15 +20,15 @@ /* Print a key in user understandable format */ void _ma_print_key(FILE *stream, register HA_KEYSEG *keyseg, - const byte *key, uint length) + const uchar *key, uint length) { int flag; short int s_1; long int l_1; float f_1; double d_1; - const byte *end; - const byte *key_end= key + length; + const uchar *end; + const uchar *key_end= key + length; VOID(fputs("Key: \"",stream)); flag=0; diff --git a/storage/maria/ma_delete.c b/storage/maria/ma_delete.c index 54c6b7aaefc..8dafd1c4f17 100644 --- a/storage/maria/ma_delete.c +++ b/storage/maria/ma_delete.c @@ -19,23 +19,23 @@ #include "ma_rt_index.h" static int d_search(MARIA_HA *info,MARIA_KEYDEF *keyinfo,uint comp_flag, - byte *key,uint key_length,my_off_t page,byte *anc_buff); -static int del(MARIA_HA *info,MARIA_KEYDEF *keyinfo,byte *key,byte *anc_buff, - my_off_t leaf_page,byte *leaf_buff,byte *keypos, - my_off_t next_block,byte *ret_key); -static int underflow(MARIA_HA *info,MARIA_KEYDEF *keyinfo,byte *anc_buff, - my_off_t leaf_page,byte *leaf_buff,byte *keypos); -static uint remove_key(MARIA_KEYDEF *keyinfo,uint nod_flag,byte *keypos, - byte *lastkey,byte *page_end, + uchar *key,uint key_length,my_off_t page,uchar *anc_buff); +static int del(MARIA_HA *info,MARIA_KEYDEF *keyinfo,uchar *key,uchar *anc_buff, + my_off_t leaf_page,uchar *leaf_buff,uchar *keypos, + my_off_t next_block,uchar *ret_key); +static int underflow(MARIA_HA *info,MARIA_KEYDEF *keyinfo,uchar *anc_buff, + my_off_t leaf_page,uchar *leaf_buff,uchar *keypos); +static uint remove_key(MARIA_KEYDEF *keyinfo,uint nod_flag,uchar *keypos, + uchar *lastkey,uchar *page_end, my_off_t *next_block); static int _ma_ck_real_delete(register MARIA_HA *info,MARIA_KEYDEF *keyinfo, - byte *key, uint key_length, my_off_t *root); + uchar *key, uint key_length, my_off_t *root); -int maria_delete(MARIA_HA *info,const byte *record) +int maria_delete(MARIA_HA *info,const uchar *record) { uint i; - byte *old_key; + uchar *old_key; int save_errno; char lastpos[8]; MARIA_SHARE *share=info->s; @@ -141,7 +141,7 @@ err: /* Remove a key from the btree index */ -int _ma_ck_delete(register MARIA_HA *info, uint keynr, byte *key, +int _ma_ck_delete(register MARIA_HA *info, uint keynr, uchar *key, uint key_length) { return _ma_ck_real_delete(info, info->s->keyinfo+keynr, key, key_length, @@ -150,12 +150,12 @@ int _ma_ck_delete(register MARIA_HA *info, uint keynr, byte *key, static int _ma_ck_real_delete(register MARIA_HA *info, MARIA_KEYDEF *keyinfo, - byte *key, uint key_length, my_off_t *root) + uchar *key, uint key_length, my_off_t *root) { int error; uint nod_flag; my_off_t old_root; - byte *root_buff; + uchar *root_buff; DBUG_ENTER("_ma_ck_real_delete"); if ((old_root=*root) == HA_OFFSET_ERROR) @@ -163,7 +163,7 @@ static int _ma_ck_real_delete(register MARIA_HA *info, MARIA_KEYDEF *keyinfo, maria_print_error(info->s, HA_ERR_CRASHED); DBUG_RETURN(my_errno=HA_ERR_CRASHED); } - if (!(root_buff= (byte*) my_alloca((uint) keyinfo->block_length+ + if (!(root_buff= (uchar*) my_alloca((uint) keyinfo->block_length+ HA_MAX_KEY_BUFF*2))) { DBUG_PRINT("error",("Couldn't allocate memory")); @@ -203,7 +203,7 @@ static int _ma_ck_real_delete(register MARIA_HA *info, MARIA_KEYDEF *keyinfo, } } err: - my_afree((gptr) root_buff); + my_afree((uchar*) root_buff); DBUG_PRINT("exit",("Return: %d",error)); DBUG_RETURN(error); } /* _ma_ck_real_delete */ @@ -218,15 +218,15 @@ err: */ static int d_search(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, - uint comp_flag, byte *key, uint key_length, - my_off_t page, byte *anc_buff) + uint comp_flag, uchar *key, uint key_length, + my_off_t page, uchar *anc_buff) { int flag,ret_value,save_flag; uint length,nod_flag,search_key_length; my_bool last_key; - byte *leaf_buff,*keypos; + uchar *leaf_buff,*keypos; my_off_t leaf_page,next_block; - byte lastkey[HA_MAX_KEY_BUFF]; + uchar lastkey[HA_MAX_KEY_BUFF]; DBUG_ENTER("d_search"); DBUG_DUMP("page",anc_buff,maria_data_on_page(anc_buff)); @@ -270,7 +270,7 @@ static int d_search(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, /* popular word. two-level tree. going down */ uint tmp_key_length; my_off_t root; - byte *kpos=keypos; + uchar *kpos=keypos; if (!(tmp_key_length=(*keyinfo->get_key)(keyinfo,nod_flag,&kpos,lastkey))) { @@ -310,7 +310,7 @@ static int d_search(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, if (nod_flag) { leaf_page= _ma_kpos(nod_flag,keypos); - if (!(leaf_buff= (byte*) my_alloca((uint) keyinfo->block_length+ + if (!(leaf_buff= (uchar*) my_alloca((uint) keyinfo->block_length+ HA_MAX_KEY_BUFF*2))) { DBUG_PRINT("error",("Couldn't allocate memory")); @@ -372,7 +372,7 @@ static int d_search(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, if (!_ma_get_last_key(info,keyinfo,anc_buff,lastkey,keypos,&length)) goto err; ret_value= _ma_insert(info,keyinfo,key,anc_buff,keypos,lastkey, - (byte*) 0,(byte*) 0,(my_off_t) 0,(my_bool) 0); + (uchar*) 0,(uchar*) 0,(my_off_t) 0,(my_bool) 0); } } if (ret_value == 0 && maria_data_on_page(anc_buff) > keyinfo->block_length) @@ -400,16 +400,16 @@ err: /* Remove a key that has a page-reference */ static int del(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, - byte *key, byte *anc_buff, my_off_t leaf_page, - byte *leaf_buff, - byte *keypos, /* Pos to where deleted key was */ + uchar *key, uchar *anc_buff, my_off_t leaf_page, + uchar *leaf_buff, + uchar *keypos, /* Pos to where deleted key was */ my_off_t next_block, - byte *ret_key) /* key before keypos in anc_buff */ + uchar *ret_key) /* key before keypos in anc_buff */ { int ret_value,length; uint a_length,nod_flag,tmp; my_off_t next_page; - byte keybuff[HA_MAX_KEY_BUFF],*endpos,*next_buff,*key_start, *prev_key; + uchar keybuff[HA_MAX_KEY_BUFF],*endpos,*next_buff,*key_start, *prev_key; MARIA_SHARE *share=info->s; MARIA_KEY_PARAM s_temp; DBUG_ENTER("del"); @@ -425,7 +425,7 @@ static int del(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, if ((nod_flag=_ma_test_if_nod(leaf_buff))) { next_page= _ma_kpos(nod_flag,endpos); - if (!(next_buff= (byte*) my_alloca((uint) keyinfo->block_length+ + if (!(next_buff= (uchar*) my_alloca((uint) keyinfo->block_length+ HA_MAX_KEY_BUFF*2))) DBUG_RETURN(-1); if (!_ma_fetch_keypage(info,keyinfo,next_page,DFLT_INIT_HITS,next_buff,0)) @@ -453,7 +453,7 @@ static int del(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, &tmp)) goto err; ret_value= _ma_insert(info,keyinfo,key,leaf_buff,endpos,keybuff, - (byte*) 0,(byte*) 0,(my_off_t) 0,0); + (uchar*) 0,(uchar*) 0,(my_off_t) 0,0); } } if (_ma_write_keypage(info,keyinfo,leaf_page,DFLT_INIT_HITS,leaf_buff)) @@ -479,7 +479,7 @@ static int del(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, prev_key=(keypos == anc_buff+2+share->base.key_reflength ? 0 : ret_key); length=(*keyinfo->pack_key)(keyinfo,share->base.key_reflength, - keypos == endpos ? (byte*) 0 : keypos, + keypos == endpos ? (uchar*) 0 : keypos, prev_key, prev_key, keybuff,&s_temp); if (length > 0) @@ -504,18 +504,18 @@ err: /* Balances adjacent pages if underflow occours */ static int underflow(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, - byte *anc_buff, + uchar *anc_buff, my_off_t leaf_page,/* Ancestor page and underflow page */ - byte *leaf_buff, - byte *keypos) /* Position to pos after key */ + uchar *leaf_buff, + uchar *keypos) /* Position to pos after key */ { int t_length; uint length,anc_length,buff_length,leaf_length,p_length,s_length,nod_flag, key_reflength,key_length; my_off_t next_page; - byte anc_key[HA_MAX_KEY_BUFF],leaf_key[HA_MAX_KEY_BUFF]; - byte *buff,*endpos,*next_keypos,*anc_pos,*half_pos,*temp_pos,*prev_key; - byte *after_key; + uchar anc_key[HA_MAX_KEY_BUFF],leaf_key[HA_MAX_KEY_BUFF]; + uchar *buff,*endpos,*next_keypos,*anc_pos,*half_pos,*temp_pos,*prev_key; + uchar *after_key; MARIA_KEY_PARAM s_temp; MARIA_SHARE *share=info->s; DBUG_ENTER("underflow"); @@ -568,7 +568,7 @@ static int underflow(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, goto err; /* merge pages and put parting key from anc_buff between */ - prev_key=(leaf_length == p_length ? (byte*) 0 : leaf_key); + prev_key=(leaf_length == p_length ? (uchar*) 0 : leaf_key); t_length=(*keyinfo->pack_key)(keyinfo,nod_flag,buff+p_length, prev_key, prev_key, anc_key, &s_temp); @@ -615,9 +615,9 @@ static int underflow(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, half_pos=after_key; _ma_kpointer(info,leaf_key+key_length,next_page); /* Save key in anc_buff */ - prev_key=(keypos == anc_buff+2+key_reflength ? (byte*) 0 : anc_key), + prev_key=(keypos == anc_buff+2+key_reflength ? (uchar*) 0 : anc_key), t_length=(*keyinfo->pack_key)(keyinfo,key_reflength, - (keypos == endpos ? (byte*) 0 : + (keypos == endpos ? (uchar*) 0 : keypos), prev_key, prev_key, leaf_key, &s_temp); @@ -633,8 +633,8 @@ static int underflow(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, bmove(buff+2,half_pos-nod_flag,(size_t) nod_flag); if (!(*keyinfo->get_key)(keyinfo,nod_flag,&half_pos,leaf_key)) goto err; - t_length=(int) (*keyinfo->pack_key)(keyinfo, nod_flag, (byte*) 0, - (byte*) 0, (byte*) 0, + t_length=(int) (*keyinfo->pack_key)(keyinfo, nod_flag, (uchar*) 0, + (uchar*) 0, (uchar*) 0, leaf_key, &s_temp); /* t_length will always be > 0 for a new page !*/ length=(uint) ((buff+maria_data_on_page(buff))-half_pos); @@ -673,10 +673,10 @@ static int underflow(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, goto err; /* merge pages and put parting key from anc_buff between */ - prev_key=(leaf_length == p_length ? (byte*) 0 : leaf_key); + prev_key=(leaf_length == p_length ? (uchar*) 0 : leaf_key); t_length=(*keyinfo->pack_key)(keyinfo,nod_flag, (leaf_length == p_length ? - (byte*) 0 : leaf_buff+p_length), + (uchar*) 0 : leaf_buff+p_length), prev_key, prev_key, anc_key, &s_temp); if (t_length >= 0) @@ -721,7 +721,7 @@ static int underflow(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, temp_pos=anc_buff+anc_length; t_length=(*keyinfo->pack_key)(keyinfo,key_reflength, - keypos == temp_pos ? (byte*) 0 + keypos == temp_pos ? (uchar*) 0 : keypos, anc_pos, anc_pos, leaf_key,&s_temp); @@ -738,8 +738,8 @@ static int underflow(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, if (!(length=(*keyinfo->get_key)(keyinfo,nod_flag,&half_pos,leaf_key))) goto err; DBUG_DUMP("key_to_leaf",leaf_key,length); - t_length=(*keyinfo->pack_key)(keyinfo,nod_flag, (byte*) 0, - (byte*) 0, (byte*) 0, leaf_key, &s_temp); + t_length=(*keyinfo->pack_key)(keyinfo,nod_flag, (uchar*) 0, + (uchar*) 0, (uchar*) 0, leaf_key, &s_temp); length=(uint) ((buff+buff_length)-half_pos); DBUG_PRINT("info",("t_length: %d length: %d",t_length,(int) length)); bmove(leaf_buff+p_length+t_length,half_pos, @@ -767,13 +767,13 @@ err: */ static uint remove_key(MARIA_KEYDEF *keyinfo, uint nod_flag, - byte *keypos, /* Where key starts */ - byte *lastkey, /* key to be removed */ - byte *page_end, /* End of page */ + uchar *keypos, /* Where key starts */ + uchar *lastkey, /* key to be removed */ + uchar *page_end, /* End of page */ my_off_t *next_block) /* ptr to next block */ { int s_length; - byte *start; + uchar *start; DBUG_ENTER("remove_key"); DBUG_PRINT("enter",("keypos: 0x%lx page_end: 0x%lx",(long) keypos, (long) page_end)); @@ -799,7 +799,7 @@ static uint remove_key(MARIA_KEYDEF *keyinfo, uint nod_flag, { if (keyinfo->flag & HA_BINARY_PACK_KEY) { - byte *old_key=start; + uchar *old_key=start; uint next_length,prev_length,prev_pack_length; get_key_length(next_length,keypos); get_key_pack_length(prev_length,prev_pack_length,old_key); diff --git a/storage/maria/ma_dynrec.c b/storage/maria/ma_dynrec.c index ebf84032106..e1968811ba2 100644 --- a/storage/maria/ma_dynrec.c +++ b/storage/maria/ma_dynrec.c @@ -25,15 +25,15 @@ #include "maria_def.h" -static my_bool write_dynamic_record(MARIA_HA *info,const byte *record, +static my_bool write_dynamic_record(MARIA_HA *info,const uchar *record, ulong reclength); static int _ma_find_writepos(MARIA_HA *info,ulong reclength,my_off_t *filepos, ulong *length); static my_bool update_dynamic_record(MARIA_HA *info, MARIA_RECORD_POS filepos, - byte *record, ulong reclength); + uchar *record, ulong reclength); static my_bool delete_dynamic_record(MARIA_HA *info,MARIA_RECORD_POS filepos, uint second_read); -static my_bool _ma_cmp_buffer(File file, const byte *buff, my_off_t filepos, +static my_bool _ma_cmp_buffer(File file, const uchar *buff, my_off_t filepos, uint length); #ifdef THREAD @@ -76,13 +76,13 @@ my_bool _ma_dynmap_file(MARIA_HA *info, my_off_t size) mapping. When swap space is not reserved one might get SIGSEGV upon a write if no physical memory is available. */ - info->s->file_map= (byte*) + info->s->file_map= (uchar*) my_mmap(0, (size_t)(size + MEMMAP_EXTRA_MARGIN), info->s->mode==O_RDONLY ? PROT_READ : PROT_READ | PROT_WRITE, MAP_SHARED | MAP_NORESERVE, info->dfile.file, 0L); - if (info->s->file_map == (byte*) MAP_FAILED) + if (info->s->file_map == (uchar*) MAP_FAILED) { info->s->file_map= NULL; DBUG_RETURN(1); @@ -132,7 +132,7 @@ void _ma_remap_file(MARIA_HA *info, my_off_t size) 0 ok */ -uint _ma_mmap_pread(MARIA_HA *info, byte *Buffer, +uint _ma_mmap_pread(MARIA_HA *info, uchar *Buffer, uint Count, my_off_t offset, myf MyFlags) { DBUG_PRINT("info", ("maria_read with mmap %d\n", info->dfile.file)); @@ -164,7 +164,7 @@ uint _ma_mmap_pread(MARIA_HA *info, byte *Buffer, /* wrapper for my_pread in case if mmap isn't used */ -uint _ma_nommap_pread(MARIA_HA *info, byte *Buffer, +uint _ma_nommap_pread(MARIA_HA *info, uchar *Buffer, uint Count, my_off_t offset, myf MyFlags) { return my_pread(info->dfile.file, Buffer, Count, offset, MyFlags); @@ -187,7 +187,7 @@ uint _ma_nommap_pread(MARIA_HA *info, byte *Buffer, !=0 error. In this case return error from pwrite */ -uint _ma_mmap_pwrite(MARIA_HA *info, byte *Buffer, +uint _ma_mmap_pwrite(MARIA_HA *info, uchar *Buffer, uint Count, my_off_t offset, myf MyFlags) { DBUG_PRINT("info", ("maria_write with mmap %d\n", info->dfile.file)); @@ -221,14 +221,14 @@ uint _ma_mmap_pwrite(MARIA_HA *info, byte *Buffer, /* wrapper for my_pwrite in case if mmap isn't used */ -uint _ma_nommap_pwrite(MARIA_HA *info, byte *Buffer, +uint _ma_nommap_pwrite(MARIA_HA *info, uchar *Buffer, uint Count, my_off_t offset, myf MyFlags) { return my_pwrite(info->dfile.file, Buffer, Count, offset, MyFlags); } -my_bool _ma_write_dynamic_record(MARIA_HA *info, const byte *record) +my_bool _ma_write_dynamic_record(MARIA_HA *info, const uchar *record) { ulong reclength= _ma_rec_pack(info,info->rec_buff + MARIA_REC_BUFF_OFFSET, record); @@ -237,8 +237,8 @@ my_bool _ma_write_dynamic_record(MARIA_HA *info, const byte *record) } my_bool _ma_update_dynamic_record(MARIA_HA *info, MARIA_RECORD_POS pos, - const byte *oldrec __attribute__ ((unused)), - const byte *record) + const uchar *oldrec __attribute__ ((unused)), + const uchar *record) { uint length= _ma_rec_pack(info, info->rec_buff + MARIA_REC_BUFF_OFFSET, record); @@ -248,9 +248,9 @@ my_bool _ma_update_dynamic_record(MARIA_HA *info, MARIA_RECORD_POS pos, } -my_bool _ma_write_blob_record(MARIA_HA *info, const byte *record) +my_bool _ma_write_blob_record(MARIA_HA *info, const uchar *record) { - byte *rec_buff; + uchar *rec_buff; int error; ulong reclength,reclength2,extra; @@ -258,7 +258,7 @@ my_bool _ma_write_blob_record(MARIA_HA *info, const byte *record) MARIA_DYN_DELETE_BLOCK_HEADER+1); reclength= (info->s->base.pack_reclength + _ma_calc_total_blob_length(info,record)+ extra); - if (!(rec_buff=(byte*) my_alloca(reclength))) + if (!(rec_buff=(uchar*) my_alloca(reclength))) { my_errno= HA_ERR_OUT_OF_MEM; /* purecov: inspected */ return(1); @@ -278,10 +278,10 @@ my_bool _ma_write_blob_record(MARIA_HA *info, const byte *record) my_bool _ma_update_blob_record(MARIA_HA *info, MARIA_RECORD_POS pos, - const byte *oldrec __attribute__ ((unused)), - const byte *record) + const uchar *oldrec __attribute__ ((unused)), + const uchar *record) { - byte *rec_buff; + uchar *rec_buff; int error; ulong reclength,extra; @@ -296,7 +296,7 @@ my_bool _ma_update_blob_record(MARIA_HA *info, MARIA_RECORD_POS pos, return 1; } #endif - if (!(rec_buff=(byte*) my_alloca(reclength))) + if (!(rec_buff=(uchar*) my_alloca(reclength))) { my_errno= HA_ERR_OUT_OF_MEM; /* purecov: inspected */ return(1); @@ -312,7 +312,7 @@ my_bool _ma_update_blob_record(MARIA_HA *info, MARIA_RECORD_POS pos, my_bool _ma_delete_dynamic_record(MARIA_HA *info, - const byte *record __attribute__ ((unused))) + const uchar *record __attribute__ ((unused))) { return delete_dynamic_record(info, info->cur_row.lastpos, 0); } @@ -320,7 +320,7 @@ my_bool _ma_delete_dynamic_record(MARIA_HA *info, /* Write record to data-file */ -static my_bool write_dynamic_record(MARIA_HA *info, const byte *record, +static my_bool write_dynamic_record(MARIA_HA *info, const uchar *record, ulong reclength) { int flag; @@ -336,7 +336,7 @@ static my_bool write_dynamic_record(MARIA_HA *info, const byte *record, if (_ma_write_part_record(info,filepos,length, (info->append_insert_at_end ? HA_OFFSET_ERROR : info->s->state.dellink), - (byte**) &record,&reclength,&flag)) + (uchar**) &record,&reclength,&flag)) goto err; } while (reclength); @@ -550,7 +550,7 @@ static my_bool delete_dynamic_record(MARIA_HA *info, MARIA_RECORD_POS filepos, bfill(block_info.header+12,8,255); else mi_sizestore(block_info.header+12,block_info.next_filepos); - if (info->s->file_write(info,(byte*) block_info.header,20,filepos, + if (info->s->file_write(info,(uchar*) block_info.header,20,filepos, MYF(MY_NABP))) DBUG_RETURN(1); info->s->state.dellink = filepos; @@ -573,12 +573,12 @@ int _ma_write_part_record(MARIA_HA *info, my_off_t filepos, /* points at empty block */ ulong length, /* length of block */ my_off_t next_filepos,/* Next empty block */ - byte **record, /* pointer to record ptr */ + uchar **record, /* pointer to record ptr */ ulong *reclength, /* length of *record */ int *flag) /* *flag == 0 if header */ { ulong head_length,res_length,extra_length,long_block,del_length; - byte *pos,*record_end; + uchar *pos,*record_end; my_off_t next_delete_block; uchar temp[MARIA_SPLIT_LENGTH+MARIA_DYN_DELETE_BLOCK_HEADER]; DBUG_ENTER("_ma_write_part_record"); @@ -622,7 +622,7 @@ int _ma_write_part_record(MARIA_HA *info, temp[0]=13; mi_int4store(temp+1,*reclength); mi_int3store(temp+5,length-head_length); - mi_sizestore((byte*) temp+8,next_filepos); + mi_sizestore((uchar*) temp+8,next_filepos); } else { @@ -632,13 +632,13 @@ int _ma_write_part_record(MARIA_HA *info, { mi_int3store(temp+1,*reclength); mi_int3store(temp+4,length-head_length); - mi_sizestore((byte*) temp+7,next_filepos); + mi_sizestore((uchar*) temp+7,next_filepos); } else { mi_int2store(temp+1,*reclength); mi_int2store(temp+3,length-head_length); - mi_sizestore((byte*) temp+5,next_filepos); + mi_sizestore((uchar*) temp+5,next_filepos); } } } @@ -649,12 +649,12 @@ int _ma_write_part_record(MARIA_HA *info, if (long_block) { mi_int3store(temp+1,length-head_length); - mi_sizestore((byte*) temp+4,next_filepos); + mi_sizestore((uchar*) temp+4,next_filepos); } else { mi_int2store(temp+1,length-head_length); - mi_sizestore((byte*) temp+3,next_filepos); + mi_sizestore((uchar*) temp+3,next_filepos); } } } @@ -675,14 +675,14 @@ int _ma_write_part_record(MARIA_HA *info, } length= *reclength+head_length; /* Write only what is needed */ } - DBUG_DUMP("header",(byte*) temp,head_length); + DBUG_DUMP("header",(uchar*) temp,head_length); /* Make a long block for one write */ record_end= *record+length-head_length; del_length=(res_length ? MARIA_DYN_DELETE_BLOCK_HEADER : 0); - bmove((byte*) (*record-head_length),(byte*) temp,head_length); + bmove((uchar*) (*record-head_length),(uchar*) temp,head_length); memcpy(temp,record_end,(size_t) (extra_length+del_length)); - bzero((byte*) record_end,extra_length); + bzero((uchar*) record_end,extra_length); if (res_length) { @@ -722,18 +722,18 @@ int _ma_write_part_record(MARIA_HA *info, if (info->update & HA_STATE_EXTEND_BLOCK) { info->update&= ~HA_STATE_EXTEND_BLOCK; - if (my_block_write(&info->rec_cache,(byte*) *record-head_length, + if (my_block_write(&info->rec_cache,(uchar*) *record-head_length, length+extra_length+del_length,filepos)) goto err; } - else if (my_b_write(&info->rec_cache,(byte*) *record-head_length, + else if (my_b_write(&info->rec_cache,(uchar*) *record-head_length, length+extra_length+del_length)) goto err; } else { info->rec_cache.seek_not_done=1; - if (info->s->file_write(info,(byte*) *record-head_length, + if (info->s->file_write(info,(uchar*) *record-head_length, length+extra_length+ del_length,filepos,info->s->write_flag)) goto err; @@ -761,7 +761,7 @@ err: /* update record from datafile */ static my_bool update_dynamic_record(MARIA_HA *info, MARIA_RECORD_POS filepos, - byte *record, ulong reclength) + uchar *record, ulong reclength) { int flag; uint error; @@ -844,7 +844,7 @@ static my_bool update_dynamic_record(MARIA_HA *info, MARIA_RECORD_POS filepos, mi_int3store(del_block.header+1, rest_length); mi_sizestore(del_block.header+4,info->s->state.dellink); bfill(del_block.header+12,8,255); - if (info->s->file_write(info,(byte*) del_block.header, 20, + if (info->s->file_write(info,(uchar*) del_block.header, 20, next_pos, MYF(MY_NABP))) DBUG_RETURN(1); info->s->state.dellink= next_pos; @@ -883,7 +883,8 @@ err: /* Pack a record. Return new reclength */ -uint _ma_rec_pack(MARIA_HA *info, register byte *to, register const byte *from) +uint _ma_rec_pack(MARIA_HA *info, register uchar *to, + register const uchar *from) { uint length,new_length,flag,bit,i; char *pos,*end,*startpos,*packpos; @@ -918,7 +919,7 @@ uint _ma_rec_pack(MARIA_HA *info, register byte *to, register const byte *from) { char *temp_pos; size_t tmp_length=length-portable_sizeof_char_ptr; - memcpy((byte*) to,from,tmp_length); + memcpy((uchar*) to,from,tmp_length); memcpy_fixed(&temp_pos,from+tmp_length,sizeof(char*)); memcpy(to+tmp_length,temp_pos,(size_t) blob->length); to+=tmp_length+blob->length; @@ -927,20 +928,20 @@ uint _ma_rec_pack(MARIA_HA *info, register byte *to, register const byte *from) } else if (type == FIELD_SKIP_ZERO) { - if (memcmp((byte*) from, maria_zero_string, length) == 0) + if (memcmp((uchar*) from, maria_zero_string, length) == 0) flag|=bit; else { - memcpy((byte*) to,from,(size_t) length); to+=length; + memcpy((uchar*) to,from,(size_t) length); to+=length; } } else if (type == FIELD_SKIP_ENDSPACE || type == FIELD_SKIP_PRESPACE) { - pos= (byte*) from; end= (byte*) from + length; + pos= (uchar*) from; end= (uchar*) from + length; if (type == FIELD_SKIP_ENDSPACE) { /* Pack trailing spaces */ - while (end > from && *(end-1) == ' ') + while (end > (char*) from && *(end-1) == ' ') end--; } else @@ -960,7 +961,7 @@ uint _ma_rec_pack(MARIA_HA *info, register byte *to, register const byte *from) } else *to++= (char) new_length; - memcpy((byte*) to,pos,(size_t) new_length); to+=new_length; + memcpy((uchar*) to,pos,(size_t) new_length); to+=new_length; flag|=bit; } else @@ -1005,9 +1006,9 @@ uint _ma_rec_pack(MARIA_HA *info, register byte *to, register const byte *from) if (bit != 1) *packpos= (char) (uchar) flag; if (info->s->calc_checksum) - *to++= (byte) info->cur_row.checksum; - DBUG_PRINT("exit",("packed length: %d",(int) (to-startpos))); - DBUG_RETURN((uint) (to-startpos)); + *to++= (uchar) info->cur_row.checksum; + DBUG_PRINT("exit",("packed length: %d",(int) ((char*)to-startpos))); + DBUG_RETURN((uint) ((char*)to-startpos)); } /* _ma_rec_pack */ @@ -1017,7 +1018,7 @@ uint _ma_rec_pack(MARIA_HA *info, register byte *to, register const byte *from) Returns 0 if record is ok. */ -my_bool _ma_rec_check(MARIA_HA *info,const char *record, byte *rec_buff, +my_bool _ma_rec_check(MARIA_HA *info,const char *record, uchar *rec_buff, ulong packed_length, my_bool with_checksum) { uint length,new_length,flag,bit,i; @@ -1048,7 +1049,7 @@ my_bool _ma_rec_check(MARIA_HA *info,const char *record, byte *rec_buff, } else if (type == FIELD_SKIP_ZERO) { - if (memcmp((byte*) record, maria_zero_string, length) == 0) + if (memcmp((uchar*) record, maria_zero_string, length) == 0) { if (!(flag & bit)) goto err; @@ -1059,7 +1060,7 @@ my_bool _ma_rec_check(MARIA_HA *info,const char *record, byte *rec_buff, else if (type == FIELD_SKIP_ENDSPACE || type == FIELD_SKIP_PRESPACE) { - pos= (byte*) record; end= (byte*) record + length; + pos= (uchar*) record; end= (uchar*) record + length; if (type == FIELD_SKIP_ENDSPACE) { /* Pack trailing spaces */ while (end > record && *(end-1) == ' ') @@ -1121,8 +1122,8 @@ my_bool _ma_rec_check(MARIA_HA *info,const char *record, byte *rec_buff, else to+= length; } - if (packed_length != (uint) (to - rec_buff) + test(info->s->calc_checksum) || - (bit != 1 && (flag & ~(bit - 1)))) + if (packed_length != (uint) (to - (char*) rec_buff) + + test(info->s->calc_checksum) || (bit != 1 && (flag & ~(bit - 1)))) goto err; if (with_checksum && ((uchar) info->cur_row.checksum != (uchar) *to)) { @@ -1141,12 +1142,12 @@ err: /* Returns -1 and my_errno =HA_ERR_RECORD_DELETED if reclength isn't */ /* right. Returns reclength (>0) if ok */ -ulong _ma_rec_unpack(register MARIA_HA *info, register byte *to, byte *from, +ulong _ma_rec_unpack(register MARIA_HA *info, register uchar *to, uchar *from, ulong found_length) { uint flag,bit,length,min_pack_length, column_length; enum en_fieldtype type; - byte *from_end,*to_end,*packpos; + uchar *from_end,*to_end,*packpos; reg3 MARIA_COLUMNDEF *column, *end_column; DBUG_ENTER("_ma_rec_unpack"); @@ -1200,7 +1201,7 @@ ulong _ma_rec_unpack(register MARIA_HA *info, register byte *to, byte *from, if (flag & bit) { if (type == FIELD_BLOB || type == FIELD_SKIP_ZERO) - bzero((byte*) to,column_length); + bzero((uchar*) to,column_length); else if (type == FIELD_SKIP_ENDSPACE || type == FIELD_SKIP_PRESPACE) { @@ -1222,13 +1223,13 @@ ulong _ma_rec_unpack(register MARIA_HA *info, register byte *to, byte *from, goto err; if (type == FIELD_SKIP_ENDSPACE) { - memcpy(to,(byte*) from,(size_t) length); - bfill((byte*) to+length,column_length-length,' '); + memcpy(to,(uchar*) from,(size_t) length); + bfill((uchar*) to+length,column_length-length,' '); } else { - bfill((byte*) to,column_length-length,' '); - memcpy(to+column_length-length,(byte*) from,(size_t) length); + bfill((uchar*) to,column_length-length,' '); + memcpy(to+column_length-length,(uchar*) from,(size_t) length); } from+=length; } @@ -1242,9 +1243,9 @@ ulong _ma_rec_unpack(register MARIA_HA *info, register byte *to, byte *from, from_left - size_length < blob_length || from_left - size_length - blob_length < min_pack_length) goto err; - memcpy((byte*) to,(byte*) from,(size_t) size_length); + memcpy((uchar*) to,(uchar*) from,(size_t) size_length); from+=size_length; - memcpy_fixed((byte*) to+size_length,(byte*) &from,sizeof(char*)); + memcpy_fixed((uchar*) to+size_length,(uchar*) &from,sizeof(char*)); from+=blob_length; } else @@ -1253,7 +1254,7 @@ ulong _ma_rec_unpack(register MARIA_HA *info, register byte *to, byte *from, min_pack_length--; if (min_pack_length + column_length > (uint) (from_end - from)) goto err; - memcpy(to,(byte*) from,(size_t) column_length); from+=column_length; + memcpy(to,(uchar*) from,(size_t) column_length); from+=column_length; } if ((bit= bit << 1) >= 256) { @@ -1265,7 +1266,7 @@ ulong _ma_rec_unpack(register MARIA_HA *info, register byte *to, byte *from, if (min_pack_length > (uint) (from_end - from)) goto err; min_pack_length-=column_length; - memcpy(to, (byte*) from, (size_t) column_length); + memcpy(to, (uchar*) from, (size_t) column_length); from+=column_length; } } @@ -1278,14 +1279,14 @@ err: my_errno= HA_ERR_WRONG_IN_RECORD; DBUG_PRINT("error",("to_end: 0x%lx -> 0x%lx from_end: 0x%lx -> 0x%lx", (long) to, (long) to_end, (long) from, (long) from_end)); - DBUG_DUMP("from",(byte*) info->rec_buff,info->s->base.min_pack_length); + DBUG_DUMP("from",(uchar*) info->rec_buff,info->s->base.min_pack_length); DBUG_RETURN(MY_FILE_ERROR); } /* _ma_rec_unpack */ /* Calc length of blob. Update info in blobs->length */ -ulong _ma_calc_total_blob_length(MARIA_HA *info, const byte *record) +ulong _ma_calc_total_blob_length(MARIA_HA *info, const uchar *record) { ulong length; MARIA_BLOB *blob,*end; @@ -1301,7 +1302,7 @@ ulong _ma_calc_total_blob_length(MARIA_HA *info, const byte *record) } -ulong _ma_calc_blob_length(uint length, const byte *pos) +ulong _ma_calc_blob_length(uint length, const uchar *pos) { switch (length) { case 1: @@ -1319,7 +1320,7 @@ ulong _ma_calc_blob_length(uint length, const byte *pos) } -void _ma_store_blob_length(byte *pos,uint pack_length,uint length) +void _ma_store_blob_length(uchar *pos,uint pack_length,uint length) { switch (pack_length) { case 1: @@ -1352,11 +1353,11 @@ void _ma_store_blob_length(byte *pos,uint pack_length,uint length) NOTE If a write buffer is active, it needs to be flushed if its contents intersects with the record to read. We always check if the position - of the first byte of the write buffer is lower than the position - past the last byte to read. In theory this is also true if the write + of the first uchar of the write buffer is lower than the position + past the last uchar to read. In theory this is also true if the write buffer is completely below the read segment. That is, if there is no intersection. But this case is unusual. We flush anyway. Only if the - first byte in the write buffer is above the last byte to read, we do + first uchar in the write buffer is above the last uchar to read, we do not flush. A dynamic record may need several reads. So this check must be done @@ -1370,7 +1371,7 @@ void _ma_store_blob_length(byte *pos,uint pack_length,uint length) 0 OK 1 Error */ -int _ma_read_dynamic_record(MARIA_HA *info, byte *buf, +int _ma_read_dynamic_record(MARIA_HA *info, uchar *buf, MARIA_RECORD_POS filepos) { int block_of_record; @@ -1381,7 +1382,7 @@ int _ma_read_dynamic_record(MARIA_HA *info, byte *buf, if (filepos != HA_OFFSET_ERROR) { - byte *to; + uchar *to; uint left_length; LINT_INIT(to); @@ -1434,7 +1435,7 @@ int _ma_read_dynamic_record(MARIA_HA *info, byte *buf, prefetch_len= block_info.data_len; if (prefetch_len) { - memcpy((byte*) to, block_info.header + offset, prefetch_len); + memcpy((uchar*) to, block_info.header + offset, prefetch_len); block_info.data_len-= prefetch_len; left_length-= prefetch_len; to+= prefetch_len; @@ -1452,7 +1453,7 @@ int _ma_read_dynamic_record(MARIA_HA *info, byte *buf, there is no equivalent without seeking. We are at the right position already. :( */ - if (info->s->file_read(info, (byte*) to, block_info.data_len, + if (info->s->file_read(info, (uchar*) to, block_info.data_len, filepos, MYF(MY_NABP))) goto panic; left_length-=block_info.data_len; @@ -1479,9 +1480,9 @@ err: /* compare unique constraint between stored rows */ my_bool _ma_cmp_dynamic_unique(MARIA_HA *info, MARIA_UNIQUEDEF *def, - const byte *record, MARIA_RECORD_POS pos) + const uchar *record, MARIA_RECORD_POS pos) { - byte *old_rec_buff,*old_record; + uchar *old_rec_buff,*old_record; my_off_t old_rec_buff_size; my_bool error; DBUG_ENTER("_ma_cmp_dynamic_unique"); @@ -1512,11 +1513,11 @@ my_bool _ma_cmp_dynamic_unique(MARIA_HA *info, MARIA_UNIQUEDEF *def, /* Compare of record on disk with packed record in memory */ my_bool _ma_cmp_dynamic_record(register MARIA_HA *info, - register const byte *record) + register const uchar *record) { uint flag, reclength, b_type,cmp_length; my_off_t filepos; - byte *buffer; + uchar *buffer; MARIA_BLOCK_INFO block_info; my_bool error= 1; DBUG_ENTER("_ma_cmp_dynamic_record"); @@ -1539,7 +1540,7 @@ my_bool _ma_cmp_dynamic_record(register MARIA_HA *info, { /* If check isn't disabled */ if (info->s->base.blobs) { - if (!(buffer=(byte*) my_alloca(info->s->base.pack_reclength+ + if (!(buffer=(uchar*) my_alloca(info->s->base.pack_reclength+ _ma_calc_total_blob_length(info,record)))) DBUG_RETURN(1); } @@ -1592,7 +1593,7 @@ my_bool _ma_cmp_dynamic_record(register MARIA_HA *info, error= 0; err: if (buffer != info->rec_buff) - my_afree((gptr) buffer); + my_afree((uchar*) buffer); DBUG_PRINT("exit", ("result: %d", error)); DBUG_RETURN(error); } @@ -1600,7 +1601,7 @@ err: /* Compare file to buffert */ -static my_bool _ma_cmp_buffer(File file, const byte *buff, my_off_t filepos, +static my_bool _ma_cmp_buffer(File file, const uchar *buff, my_off_t filepos, uint length) { uint next_length; @@ -1612,7 +1613,7 @@ static my_bool _ma_cmp_buffer(File file, const byte *buff, my_off_t filepos, while (length > IO_SIZE*2) { if (my_pread(file,temp_buff,next_length,filepos, MYF(MY_NABP)) || - memcmp((byte*) buff,temp_buff,next_length)) + memcmp((uchar*) buff,temp_buff,next_length)) goto err; filepos+=next_length; buff+=next_length; @@ -1621,7 +1622,7 @@ static my_bool _ma_cmp_buffer(File file, const byte *buff, my_off_t filepos, } if (my_pread(file,temp_buff,length,filepos,MYF(MY_NABP))) goto err; - DBUG_RETURN(memcmp((byte*) buff,temp_buff,length) != 0); + DBUG_RETURN(memcmp((uchar*) buff,temp_buff,length) != 0); err: DBUG_RETURN(1); } @@ -1657,13 +1658,13 @@ err: */ int _ma_read_rnd_dynamic_record(MARIA_HA *info, - byte *buf, + uchar *buf, MARIA_RECORD_POS filepos, my_bool skip_deleted_blocks) { int block_of_record, info_read, save_errno; uint left_len,b_type; - byte *to; + uchar *to; MARIA_BLOCK_INFO block_info; MARIA_SHARE *share=info->s; DBUG_ENTER("_ma_read_rnd_dynamic_record"); @@ -1703,7 +1704,7 @@ int _ma_read_rnd_dynamic_record(MARIA_HA *info, } if (info->opt_flag & READ_CACHE_USED) { - if (_ma_read_cache(&info->rec_cache,(byte*) block_info.header,filepos, + if (_ma_read_cache(&info->rec_cache,(uchar*) block_info.header,filepos, sizeof(block_info.header), (!block_of_record && skip_deleted_blocks ? READING_NEXT : 0) | READING_HEADER)) @@ -1766,7 +1767,7 @@ int _ma_read_rnd_dynamic_record(MARIA_HA *info, tmp_length= block_info.data_len; if (tmp_length) { - memcpy((byte*) to, block_info.header+offset,tmp_length); + memcpy((uchar*) to, block_info.header+offset,tmp_length); block_info.data_len-=tmp_length; left_len-=tmp_length; to+=tmp_length; @@ -1778,7 +1779,7 @@ int _ma_read_rnd_dynamic_record(MARIA_HA *info, { if (info->opt_flag & READ_CACHE_USED) { - if (_ma_read_cache(&info->rec_cache,(byte*) to,filepos, + if (_ma_read_cache(&info->rec_cache,(uchar*) to,filepos, block_info.data_len, (!block_of_record && skip_deleted_blocks) ? READING_NEXT : 0)) @@ -1792,7 +1793,7 @@ int _ma_read_rnd_dynamic_record(MARIA_HA *info, flush_io_cache(&info->rec_cache)) goto err; /* VOID(my_seek(info->dfile.file, filepos, MY_SEEK_SET, MYF(0))); */ - if (my_read(info->dfile.file, (byte*)to, block_info.data_len, + if (my_read(info->dfile.file, (uchar*)to, block_info.data_len, MYF(MY_NABP))) { if (my_errno == -1) @@ -1850,7 +1851,7 @@ uint _ma_get_block_info(MARIA_BLOCK_INFO *info, File file, my_off_t filepos) sizeof(info->header)) goto err; } - DBUG_DUMP("header",(byte*) header,MARIA_BLOCK_INFO_HEADER_LENGTH); + DBUG_DUMP("header",(uchar*) header,MARIA_BLOCK_INFO_HEADER_LENGTH); if (info->second_read) { if (info->header[0] <= 6 || info->header[0] == 13) diff --git a/storage/maria/ma_extra.c b/storage/maria/ma_extra.c index 61eba165412..2fc4873d535 100644 --- a/storage/maria/ma_extra.c +++ b/storage/maria/ma_extra.c @@ -188,8 +188,8 @@ int maria_extra(MARIA_HA *info, enum ha_extra_function function, case HA_EXTRA_KEYREAD: /* Read only keys to record */ case HA_EXTRA_REMEMBER_POS: info->opt_flag |= REMEMBER_OLD_POS; - bmove((byte*) info->lastkey+share->base.max_key_length*2, - (byte*) info->lastkey,info->lastkey_length); + bmove((uchar*) info->lastkey+share->base.max_key_length*2, + (uchar*) info->lastkey,info->lastkey_length); info->save_update= info->update; info->save_lastinx= info->lastinx; info->save_lastpos= info->cur_row.lastpos; @@ -205,8 +205,8 @@ int maria_extra(MARIA_HA *info, enum ha_extra_function function, case HA_EXTRA_RESTORE_POS: if (info->opt_flag & REMEMBER_OLD_POS) { - bmove((byte*) info->lastkey, - (byte*) info->lastkey+share->base.max_key_length*2, + bmove((uchar*) info->lastkey, + (uchar*) info->lastkey+share->base.max_key_length*2, info->save_lastkey_length); info->update= info->save_update | HA_STATE_WRITTEN; info->lastinx= info->save_lastinx; diff --git a/storage/maria/ma_ft_boolean_search.c b/storage/maria/ma_ft_boolean_search.c index 6e95262fe84..41661d1c288 100644 --- a/storage/maria/ma_ft_boolean_search.c +++ b/storage/maria/ma_ft_boolean_search.c @@ -111,7 +111,7 @@ typedef struct st_ftb_word uint ndepth; uint len; uchar off; - byte word[1]; + uchar word[1]; } FTB_WORD; typedef struct st_ft_info @@ -161,7 +161,7 @@ typedef struct st_my_ftb_param { FTB *ftb; FTB_EXPR *ftbe; - byte *up_quot; + uchar *up_quot; uint depth; } MY_FTB_PARAM; @@ -274,7 +274,7 @@ static int ftb_parse_query_internal(MYSQL_FTPARSER_PARAM *param, MY_FTB_PARAM *ftb_param= param->mysql_ftparam; MYSQL_FTPARSER_BOOLEAN_INFO info; CHARSET_INFO *cs= ftb_param->ftb->charset; - char **start= &query; + uchar **start= (uchar**) &query; char *end= query + len; FT_WORD w; @@ -286,7 +286,7 @@ static int ftb_parse_query_internal(MYSQL_FTPARSER_PARAM *param, } -static void _ftb_parse_query(FTB *ftb, byte *query, uint len, +static void _ftb_parse_query(FTB *ftb, uchar *query, uint len, struct st_mysql_ftparser *parser) { MYSQL_FTPARSER_PARAM *param; @@ -331,7 +331,7 @@ static int _ft2_search(FTB *ftb, FTB_WORD *ftbw, my_bool init_search) my_bool can_go_down; MARIA_HA *info=ftb->info; uint off= 0, extra=HA_FT_WLEN+info->s->base.rec_reflength; - byte *lastkey_buf= ftbw->word+ftbw->off; + uchar *lastkey_buf= ftbw->word+ftbw->off; if (ftbw->flags & FTB_FLAG_TRUNC) lastkey_buf+=ftbw->len; @@ -504,7 +504,7 @@ static void _ftb_init_index_search(FT_INFO *ftb) } -FT_INFO * maria_ft_init_boolean_search(MARIA_HA *info, uint keynr, byte *query, +FT_INFO * maria_ft_init_boolean_search(MARIA_HA *info, uint keynr, uchar *query, uint query_len, CHARSET_INFO *cs) { FTB *ftb; @@ -544,14 +544,14 @@ FT_INFO * maria_ft_init_boolean_search(MARIA_HA *info, uint keynr, byte *query, Hack: instead of init_queue, we'll use reinit queue to be able to alloc queue with alloc_root() */ - if (! (ftb->queue.root= (byte **)alloc_root(&ftb->mem_root, + if (! (ftb->queue.root= (uchar **)alloc_root(&ftb->mem_root, (ftb->queue.max_elements + 1) * sizeof(void *)))) goto err; reinit_queue(&ftb->queue, ftb->queue.max_elements, 0, 0, - (int (*)(void*, byte*, byte*))FTB_WORD_cmp, 0); + (int (*)(void*, uchar*, uchar*))FTB_WORD_cmp, 0); for (ftbw= ftb->last_word; ftbw; ftbw= ftbw->prev) - queue_insert(&ftb->queue, (byte *)ftbw); + queue_insert(&ftb->queue, (uchar *)ftbw); ftb->list=(FTB_WORD **)alloc_root(&ftb->mem_root, sizeof(FTB_WORD *)*ftb->queue.elements); memcpy(ftb->list, ftb->queue.root+1, sizeof(FTB_WORD *)*ftb->queue.elements); @@ -562,7 +562,7 @@ FT_INFO * maria_ft_init_boolean_search(MARIA_HA *info, uint keynr, byte *query, return ftb; err: free_root(& ftb->mem_root, MYF(0)); - my_free((gptr)ftb,MYF(0)); + my_free((uchar*)ftb,MYF(0)); return 0; } @@ -616,7 +616,8 @@ static int ftb_check_phrase_internal(MYSQL_FTPARSER_PARAM *param, FT_WORD word; MY_FTB_PHRASE_PARAM *phrase_param= param->mysql_ftparam; const char *docend= document + len; - while (maria_ft_simple_get_word(phrase_param->cs, &document, docend, &word, FALSE)) + while (maria_ft_simple_get_word(phrase_param->cs, (uchar**) &document, + docend, &word, FALSE)) { param->mysql_add_word(param, word.pos, word.len, 0); if (phrase_param->match) @@ -640,7 +641,7 @@ static int ftb_check_phrase_internal(MYSQL_FTPARSER_PARAM *param, 1 is returned if phrase found, 0 else. */ -static int _ftb_check_phrase(FTB *ftb, const byte *document, uint len, +static int _ftb_check_phrase(FTB *ftb, const uchar *document, uint len, FTB_EXPR *ftbe, struct st_mysql_ftparser *parser) { MY_FTB_PHRASE_PARAM ftb_param; @@ -661,7 +662,7 @@ static int _ftb_check_phrase(FTB *ftb, const byte *document, uint len, param->mysql_add_word= ftb_phrase_add_word; param->mysql_ftparam= (void *)&ftb_param; param->cs= ftb->charset; - param->doc= (byte *)document; + param->doc= (uchar *)document; param->length= len; param->flags= 0; param->mode= MYSQL_FTPARSER_WITH_STOPWORDS; @@ -865,13 +866,13 @@ static int ftb_find_relevance_parse(MYSQL_FTPARSER_PARAM *param, FT_INFO *ftb= ftb_param->ftb; char *end= doc + len; FT_WORD w; - while (maria_ft_simple_get_word(ftb->charset, &doc, end, &w, TRUE)) + while (maria_ft_simple_get_word(ftb->charset, (uchar**) &doc, end, &w, TRUE)) param->mysql_add_word(param, w.pos, w.len, 0); return(0); } -float maria_ft_boolean_find_relevance(FT_INFO *ftb, byte *record, uint length) +float maria_ft_boolean_find_relevance(FT_INFO *ftb, uchar *record, uint length) { FTB_EXPR *ftbe; FT_SEG_ITERATOR ftsi, ftsi2; @@ -923,7 +924,7 @@ float maria_ft_boolean_find_relevance(FT_INFO *ftb, byte *record, uint length) { if (!ftsi.pos) continue; - param->doc= (byte *)ftsi.pos; + param->doc= (uchar *)ftsi.pos; param->length= ftsi.len; parser->parse(param); } @@ -947,7 +948,7 @@ void maria_ft_boolean_close_search(FT_INFO *ftb) delete_tree(& ftb->no_dupes); } free_root(& ftb->mem_root, MYF(0)); - my_free((gptr)ftb,MYF(0)); + my_free((uchar*)ftb,MYF(0)); } diff --git a/storage/maria/ma_ft_nlq_search.c b/storage/maria/ma_ft_nlq_search.c index 145b7891dd1..cad5238d4a5 100644 --- a/storage/maria/ma_ft_nlq_search.c +++ b/storage/maria/ma_ft_nlq_search.c @@ -69,7 +69,7 @@ static int walk_and_match(FT_WORD *word, uint32 count, ALL_IN_ONE *aio) TREE_ELEMENT *selem; double gweight=1; MARIA_HA *info= aio->info; - byte *keybuff= (byte*) aio->keybuff; + uchar *keybuff= (uchar*) aio->keybuff; MARIA_KEYDEF *keyinfo=info->s->keyinfo+aio->keynr; my_off_t key_root=info->s->state.key_root[aio->keynr]; uint extra=HA_FT_WLEN+info->s->base.rec_reflength; @@ -190,7 +190,7 @@ static int walk_and_push(FT_SUPERDOC *from, DBUG_ENTER("walk_and_copy"); from->doc.weight+=from->tmp_weight*from->word_ptr->weight; set_if_smaller(best->elements, ft_query_expansion_limit-1); - queue_insert(best, (byte *)& from->doc); + queue_insert(best, (uchar *)& from->doc); DBUG_RETURN(0); } @@ -202,8 +202,8 @@ static int FT_DOC_cmp(void *unused __attribute__((unused)), } -FT_INFO *maria_ft_init_nlq_search(MARIA_HA *info, uint keynr, byte *query, - uint query_len, uint flags, byte *record) +FT_INFO *maria_ft_init_nlq_search(MARIA_HA *info, uint keynr, uchar *query, + uint query_len, uint flags, uchar *record) { TREE wtree; ALL_IN_ONE aio; @@ -324,7 +324,7 @@ int maria_ft_nlq_read_next(FT_INFO *handler, char *record) float maria_ft_nlq_find_relevance(FT_INFO *handler, - byte *record __attribute__((unused)), + uchar *record __attribute__((unused)), uint length __attribute__((unused))) { int a,b,c; @@ -353,7 +353,7 @@ float maria_ft_nlq_find_relevance(FT_INFO *handler, void maria_ft_nlq_close_search(FT_INFO *handler) { - my_free((gptr)handler,MYF(0)); + my_free((uchar*)handler,MYF(0)); } diff --git a/storage/maria/ma_ft_parser.c b/storage/maria/ma_ft_parser.c index f41b53bf3f7..2cbbb2dc5f7 100644 --- a/storage/maria/ma_ft_parser.c +++ b/storage/maria/ma_ft_parser.c @@ -80,7 +80,7 @@ FT_WORD * maria_ft_linearize(TREE *wtree, MEM_ROOT *mem_root) DBUG_RETURN(wlist); } -my_bool maria_ft_boolean_check_syntax_string(const byte *str) +my_bool maria_ft_boolean_check_syntax_string(const uchar *str) { uint i, j; @@ -109,10 +109,10 @@ my_bool maria_ft_boolean_check_syntax_string(const byte *str) 3 - right bracket 4 - stopword found */ -byte maria_ft_get_word(CHARSET_INFO *cs, byte **start, byte *end, - FT_WORD *word, MYSQL_FTPARSER_BOOLEAN_INFO *param) +uchar maria_ft_get_word(CHARSET_INFO *cs, uchar **start, uchar *end, + FT_WORD *word, MYSQL_FTPARSER_BOOLEAN_INFO *param) { - byte *doc=*start; + uchar *doc=*start; int ctype; uint mwc, length, mbl; @@ -199,10 +199,11 @@ ret: return param->type; } -byte maria_ft_simple_get_word(CHARSET_INFO *cs, byte **start, const byte *end, - FT_WORD *word, my_bool skip_stopwords) +uchar maria_ft_simple_get_word(CHARSET_INFO *cs, uchar **start, + const uchar *end, FT_WORD *word, + my_bool skip_stopwords) { - byte *doc= *start; + uchar *doc= *start; uint mwc, length, mbl; int ctype; DBUG_ENTER("maria_ft_simple_get_word"); @@ -263,9 +264,9 @@ static int maria_ft_add_word(MYSQL_FTPARSER_PARAM *param, wtree= ft_param->wtree; if (param->flags & MYSQL_FTFLAGS_NEED_COPY) { - byte *ptr; + uchar *ptr; DBUG_ASSERT(wtree->with_delete == 0); - ptr= (byte *)alloc_root(ft_param->mem_root, word_len); + ptr= (uchar *)alloc_root(ft_param->mem_root, word_len); memcpy(ptr, word, word_len); w.pos= ptr; } @@ -282,9 +283,10 @@ static int maria_ft_add_word(MYSQL_FTPARSER_PARAM *param, static int maria_ft_parse_internal(MYSQL_FTPARSER_PARAM *param, - char *doc, int doc_len) + char *doc_arg, int doc_len) { - byte *end=doc+doc_len; + uchar *doc= (uchar*) doc_arg; + uchar *end= doc + doc_len; MY_FT_PARSER_PARAM *ft_param=param->mysql_ftparam; TREE *wtree= ft_param->wtree; FT_WORD w; @@ -297,7 +299,7 @@ static int maria_ft_parse_internal(MYSQL_FTPARSER_PARAM *param, } -int maria_ft_parse(TREE *wtree, byte *doc, int doclen, +int maria_ft_parse(TREE *wtree, uchar *doc, int doclen, struct st_mysql_ftparser *parser, MYSQL_FTPARSER_PARAM *param, MEM_ROOT *mem_root) { diff --git a/storage/maria/ma_ft_update.c b/storage/maria/ma_ft_update.c index 97ebdb05b42..f36147ccde2 100644 --- a/storage/maria/ma_ft_update.c +++ b/storage/maria/ma_ft_update.c @@ -20,7 +20,7 @@ #include "ma_ftdefs.h" #include -void _ma_ft_segiterator_init(MARIA_HA *info, uint keynr, const byte *record, +void _ma_ft_segiterator_init(MARIA_HA *info, uint keynr, const uchar *record, FT_SEG_ITERATOR *ftsi) { DBUG_ENTER("_ma_ft_segiterator_init"); @@ -31,7 +31,7 @@ void _ma_ft_segiterator_init(MARIA_HA *info, uint keynr, const byte *record, DBUG_VOID_RETURN; } -void _ma_ft_segiterator_dummy_init(const byte *record, uint len, +void _ma_ft_segiterator_dummy_init(const uchar *record, uint len, FT_SEG_ITERATOR *ftsi) { DBUG_ENTER("_ma_ft_segiterator_dummy_init"); @@ -94,7 +94,7 @@ uint _ma_ft_segiterator(register FT_SEG_ITERATOR *ftsi) /* parses a document i.e. calls maria_ft_parse for every keyseg */ -uint _ma_ft_parse(TREE *parsed, MARIA_HA *info, uint keynr, const byte *record, +uint _ma_ft_parse(TREE *parsed, MARIA_HA *info, uint keynr, const uchar *record, MYSQL_FTPARSER_PARAM *param, MEM_ROOT *mem_root) { FT_SEG_ITERATOR ftsi; @@ -108,14 +108,14 @@ uint _ma_ft_parse(TREE *parsed, MARIA_HA *info, uint keynr, const byte *record, while (_ma_ft_segiterator(&ftsi)) { if (ftsi.pos) - if (maria_ft_parse(parsed, (byte *)ftsi.pos, ftsi.len, parser, param, + if (maria_ft_parse(parsed, (uchar *)ftsi.pos, ftsi.len, parser, param, mem_root)) DBUG_RETURN(1); } DBUG_RETURN(0); } -FT_WORD * _ma_ft_parserecord(MARIA_HA *info, uint keynr, const byte *record, +FT_WORD * _ma_ft_parserecord(MARIA_HA *info, uint keynr, const uchar *record, MEM_ROOT *mem_root) { TREE ptree; @@ -131,7 +131,7 @@ FT_WORD * _ma_ft_parserecord(MARIA_HA *info, uint keynr, const byte *record, DBUG_RETURN(maria_ft_linearize(&ptree, mem_root)); } -static int _ma_ft_store(MARIA_HA *info, uint keynr, byte *keybuf, +static int _ma_ft_store(MARIA_HA *info, uint keynr, uchar *keybuf, FT_WORD *wlist, my_off_t filepos) { uint key_length; @@ -146,7 +146,7 @@ static int _ma_ft_store(MARIA_HA *info, uint keynr, byte *keybuf, DBUG_RETURN(0); } -static int _ma_ft_erase(MARIA_HA *info, uint keynr, byte *keybuf, +static int _ma_ft_erase(MARIA_HA *info, uint keynr, uchar *keybuf, FT_WORD *wlist, my_off_t filepos) { uint key_length, err=0; @@ -169,7 +169,7 @@ static int _ma_ft_erase(MARIA_HA *info, uint keynr, byte *keybuf, #define THOSE_TWO_DAMN_KEYS_ARE_REALLY_DIFFERENT 1 #define GEE_THEY_ARE_ABSOLUTELY_IDENTICAL 0 -int _ma_ft_cmp(MARIA_HA *info, uint keynr, const byte *rec1, const byte *rec2) +int _ma_ft_cmp(MARIA_HA *info, uint keynr, const uchar *rec1, const uchar *rec2) { FT_SEG_ITERATOR ftsi1, ftsi2; CHARSET_INFO *cs=info->s->keyinfo[keynr].seg->charset; @@ -192,8 +192,8 @@ int _ma_ft_cmp(MARIA_HA *info, uint keynr, const byte *rec1, const byte *rec2) /* update a document entry */ -int _ma_ft_update(MARIA_HA *info, uint keynr, byte *keybuf, - const byte *oldrec, const byte *newrec, my_off_t pos) +int _ma_ft_update(MARIA_HA *info, uint keynr, uchar *keybuf, + const uchar *oldrec, const uchar *newrec, my_off_t pos) { int error= -1; FT_WORD *oldlist,*newlist, *old_word, *new_word; @@ -243,7 +243,7 @@ err: /* adds a document to the collection */ -int _ma_ft_add(MARIA_HA *info, uint keynr, byte *keybuf, const byte *record, +int _ma_ft_add(MARIA_HA *info, uint keynr, uchar *keybuf, const uchar *record, my_off_t pos) { int error= -1; @@ -261,7 +261,7 @@ int _ma_ft_add(MARIA_HA *info, uint keynr, byte *keybuf, const byte *record, /* removes a document from the collection */ -int _ma_ft_del(MARIA_HA *info, uint keynr, byte *keybuf, const byte *record, +int _ma_ft_del(MARIA_HA *info, uint keynr, uchar *keybuf, const uchar *record, my_off_t pos) { int error= -1; @@ -277,10 +277,10 @@ int _ma_ft_del(MARIA_HA *info, uint keynr, byte *keybuf, const byte *record, } -uint _ma_ft_make_key(MARIA_HA *info, uint keynr, byte *keybuf, FT_WORD *wptr, +uint _ma_ft_make_key(MARIA_HA *info, uint keynr, uchar *keybuf, FT_WORD *wptr, my_off_t filepos) { - byte buf[HA_FT_MAXBYTELEN+16]; + uchar buf[HA_FT_MAXBYTELEN+16]; DBUG_ENTER("_ma_ft_make_key"); #if HA_FT_WTYPE == HA_KEYTYPE_FLOAT @@ -302,12 +302,12 @@ uint _ma_ft_make_key(MARIA_HA *info, uint keynr, byte *keybuf, FT_WORD *wptr, convert key value to ft2 */ -uint _ma_ft_convert_to_ft2(MARIA_HA *info, uint keynr, byte *key) +uint _ma_ft_convert_to_ft2(MARIA_HA *info, uint keynr, uchar *key) { my_off_t root; DYNAMIC_ARRAY *da=info->ft1_to_ft2; MARIA_KEYDEF *keyinfo=&info->s->ft2_keyinfo; - byte *key_ptr= (byte*) dynamic_array_ptr(da, 0), *end; + uchar *key_ptr= (uchar*) dynamic_array_ptr(da, 0), *end; uint length, key_length; DBUG_ENTER("_ma_ft_convert_to_ft2"); @@ -335,7 +335,7 @@ uint _ma_ft_convert_to_ft2(MARIA_HA *info, uint keynr, byte *key) DBUG_RETURN(-1); /* inserting the rest of key values */ - end= (byte*) dynamic_array_ptr(da, da->elements); + end= (uchar*) dynamic_array_ptr(da, da->elements); for (key_ptr+=length; key_ptr < end; key_ptr+=keyinfo->keylength) if(_ma_ck_real_write_btree(info, keyinfo, key_ptr, 0, &root, SEARCH_SAME)) DBUG_RETURN(-1); diff --git a/storage/maria/ma_ftdefs.h b/storage/maria/ma_ftdefs.h index e25687d54b9..5a7357e451c 100644 --- a/storage/maria/ma_ftdefs.h +++ b/storage/maria/ma_ftdefs.h @@ -96,44 +96,44 @@ #define FTB_RQUOT (ft_boolean_syntax[11]) typedef struct st_maria_ft_word { - byte * pos; + uchar * pos; uint len; double weight; } FT_WORD; int is_stopword(char *word, uint len); -uint _ma_ft_make_key(MARIA_HA *, uint , byte *, FT_WORD *, my_off_t); +uint _ma_ft_make_key(MARIA_HA *, uint , uchar *, FT_WORD *, my_off_t); -byte maria_ft_get_word(CHARSET_INFO *, byte **, byte *, FT_WORD *, +uchar maria_ft_get_word(CHARSET_INFO *, uchar **, uchar *, FT_WORD *, MYSQL_FTPARSER_BOOLEAN_INFO *); -byte maria_ft_simple_get_word(CHARSET_INFO *, byte **, const byte *, +uchar maria_ft_simple_get_word(CHARSET_INFO *, uchar **, const uchar *, FT_WORD *, my_bool); typedef struct _st_maria_ft_seg_iterator { uint num, len; HA_KEYSEG *seg; - const byte *rec, *pos; + const uchar *rec, *pos; } FT_SEG_ITERATOR; -void _ma_ft_segiterator_init(MARIA_HA *, uint, const byte *, FT_SEG_ITERATOR *); -void _ma_ft_segiterator_dummy_init(const byte *, uint, FT_SEG_ITERATOR *); +void _ma_ft_segiterator_init(MARIA_HA *, uint, const uchar *, FT_SEG_ITERATOR *); +void _ma_ft_segiterator_dummy_init(const uchar *, uint, FT_SEG_ITERATOR *); uint _ma_ft_segiterator(FT_SEG_ITERATOR *); void maria_ft_parse_init(TREE *, CHARSET_INFO *); -int maria_ft_parse(TREE *, byte *, int, struct st_mysql_ftparser *parser, +int maria_ft_parse(TREE *, uchar *, int, struct st_mysql_ftparser *parser, MYSQL_FTPARSER_PARAM *, MEM_ROOT *); FT_WORD * maria_ft_linearize(TREE *, MEM_ROOT *); -FT_WORD * _ma_ft_parserecord(MARIA_HA *, uint, const byte *, MEM_ROOT *); -uint _ma_ft_parse(TREE *, MARIA_HA *, uint, const byte *, +FT_WORD * _ma_ft_parserecord(MARIA_HA *, uint, const uchar *, MEM_ROOT *); +uint _ma_ft_parse(TREE *, MARIA_HA *, uint, const uchar *, MYSQL_FTPARSER_PARAM *, MEM_ROOT *); -FT_INFO *maria_ft_init_nlq_search(MARIA_HA *, uint, byte *, uint, uint, byte *); -FT_INFO *maria_ft_init_boolean_search(MARIA_HA *, uint, byte *, uint, CHARSET_INFO *); +FT_INFO *maria_ft_init_nlq_search(MARIA_HA *, uint, uchar *, uint, uint, uchar *); +FT_INFO *maria_ft_init_boolean_search(MARIA_HA *, uint, uchar *, uint, CHARSET_INFO *); extern const struct _ft_vft _ma_ft_vft_nlq; int maria_ft_nlq_read_next(FT_INFO *, char *); -float maria_ft_nlq_find_relevance(FT_INFO *, byte *, uint); +float maria_ft_nlq_find_relevance(FT_INFO *, uchar *, uint); void maria_ft_nlq_close_search(FT_INFO *); float maria_ft_nlq_get_relevance(FT_INFO *); my_off_t maria_ft_nlq_get_docid(FT_INFO *); @@ -141,7 +141,7 @@ void maria_ft_nlq_reinit_search(FT_INFO *); extern const struct _ft_vft _ma_ft_vft_boolean; int maria_ft_boolean_read_next(FT_INFO *, char *); -float maria_ft_boolean_find_relevance(FT_INFO *, byte *, uint); +float maria_ft_boolean_find_relevance(FT_INFO *, uchar *, uint); void maria_ft_boolean_close_search(FT_INFO *); float maria_ft_boolean_get_relevance(FT_INFO *); my_off_t maria_ft_boolean_get_docid(FT_INFO *); diff --git a/storage/maria/ma_fulltext.h b/storage/maria/ma_fulltext.h index 778ffd49196..dc6cf9d1204 100644 --- a/storage/maria/ma_fulltext.h +++ b/storage/maria/ma_fulltext.h @@ -20,8 +20,8 @@ #include "maria_def.h" #include "ft_global.h" -int _ma_ft_cmp(MARIA_HA *, uint, const byte *, const byte *); -int _ma_ft_add(MARIA_HA *, uint, byte *, const byte *, my_off_t); -int _ma_ft_del(MARIA_HA *, uint, byte *, const byte *, my_off_t); +int _ma_ft_cmp(MARIA_HA *, uint, const uchar *, const uchar *); +int _ma_ft_add(MARIA_HA *, uint, uchar *, const uchar *, my_off_t); +int _ma_ft_del(MARIA_HA *, uint, uchar *, const uchar *, my_off_t); -uint _ma_ft_convert_to_ft2(MARIA_HA *, uint, byte *); +uint _ma_ft_convert_to_ft2(MARIA_HA *, uint, uchar *); diff --git a/storage/maria/ma_key.c b/storage/maria/ma_key.c index 941d5d0665e..83ba6853330 100644 --- a/storage/maria/ma_key.c +++ b/storage/maria/ma_key.c @@ -31,7 +31,7 @@ set_if_smaller(char_length,length); \ } while(0) -static int _ma_put_key_in_record(MARIA_HA *info,uint keynr,byte *record); +static int _ma_put_key_in_record(MARIA_HA *info,uint keynr,uchar *record); /* Make a intern key from a record @@ -48,11 +48,11 @@ static int _ma_put_key_in_record(MARIA_HA *info,uint keynr,byte *record); Length of key */ -uint _ma_make_key(register MARIA_HA *info, uint keynr, byte *key, - const byte *record, MARIA_RECORD_POS filepos) +uint _ma_make_key(register MARIA_HA *info, uint keynr, uchar *key, + const uchar *record, MARIA_RECORD_POS filepos) { - const byte *pos; - byte *start; + const uchar *pos; + uchar *start; reg1 HA_KEYSEG *keyseg; my_bool is_ft= info->s->keyinfo[keynr].flag & HA_FULLTEXT; DBUG_ENTER("_ma_make_key"); @@ -112,7 +112,7 @@ uint _ma_make_key(register MARIA_HA *info, uint keynr, byte *key, } else { - const byte *end= pos + length; + const uchar *end= pos + length; while (pos < end && pos[0] == ' ') pos++; length= (uint) (end-pos); @@ -215,10 +215,10 @@ uint _ma_make_key(register MARIA_HA *info, uint keynr, byte *key, last_use_keyseg Store pointer to the keyseg after the last used one */ -uint _ma_pack_key(register MARIA_HA *info, uint keynr, byte *key, - const byte *old, uint k_length, HA_KEYSEG **last_used_keyseg) +uint _ma_pack_key(register MARIA_HA *info, uint keynr, uchar *key, + const uchar *old, uint k_length, HA_KEYSEG **last_used_keyseg) { - byte *start_key=key; + uchar *start_key=key; HA_KEYSEG *keyseg; my_bool is_ft= info->s->keyinfo[keynr].flag & HA_FULLTEXT; DBUG_ENTER("_ma_pack_key"); @@ -230,7 +230,7 @@ uint _ma_pack_key(register MARIA_HA *info, uint keynr, byte *key, enum ha_base_keytype type=(enum ha_base_keytype) keyseg->type; uint length=min((uint) keyseg->length,(uint) k_length); uint char_length; - const byte *pos; + const uchar *pos; CHARSET_INFO *cs=keyseg->charset; if (keyseg->null_bit) @@ -252,7 +252,7 @@ uint _ma_pack_key(register MARIA_HA *info, uint keynr, byte *key, pos= old; if (keyseg->flag & HA_SPACE_PACK) { - const byte *end= pos + length; + const uchar *end= pos + length; if (type != HA_KEYTYPE_NUM) { while (end > pos && end[-1] == ' ') @@ -350,12 +350,12 @@ uint _ma_pack_key(register MARIA_HA *info, uint keynr, byte *key, */ static int _ma_put_key_in_record(register MARIA_HA *info, uint keynr, - byte *record) + uchar *record) { - reg2 byte *key; - byte *pos,*key_end; + reg2 uchar *key; + uchar *pos,*key_end; reg1 HA_KEYSEG *keyseg; - byte *blob_ptr; + uchar *blob_ptr; DBUG_ENTER("_ma_put_key_in_record"); blob_ptr= info->lastkey2; /* Place to put blob parts */ @@ -378,7 +378,7 @@ static int _ma_put_key_in_record(register MARIA_HA *info, uint keynr, if (keyseg->bit_length) { - byte bits= *key++; + uchar bits= *key++; set_rec_bits(bits, record + keyseg->bit_pos, keyseg->bit_start, keyseg->bit_length); length--; @@ -456,8 +456,8 @@ static int _ma_put_key_in_record(register MARIA_HA *info, uint keynr, } else if (keyseg->flag & HA_SWAP_KEY) { - byte *to= record+keyseg->start+keyseg->length; - byte *end= key+keyseg->length; + uchar *to= record+keyseg->start+keyseg->length; + uchar *end= key+keyseg->length; #ifdef CHECK_KEYS if (end > key_end) goto err; @@ -487,7 +487,7 @@ err: /* Here when key reads are used */ -int _ma_read_key_record(MARIA_HA *info, byte *buf, MARIA_RECORD_POS filepos) +int _ma_read_key_record(MARIA_HA *info, uchar *buf, MARIA_RECORD_POS filepos) { fast_ma_writeinfo(info); if (filepos != HA_OFFSET_ERROR) @@ -522,12 +522,12 @@ int _ma_read_key_record(MARIA_HA *info, byte *buf, MARIA_RECORD_POS filepos) less than zero. */ -ulonglong ma_retrieve_auto_increment(MARIA_HA *info,const byte *record) +ulonglong ma_retrieve_auto_increment(MARIA_HA *info,const uchar *record) { ulonglong value= 0; /* Store unsigned values here */ longlong s_value= 0; /* Store signed values here */ HA_KEYSEG *keyseg= info->s->keyinfo[info->s->base.auto_key-1].seg; - const byte *key= record + keyseg->start; + const uchar *key= record + keyseg->start; switch (keyseg->type) { case HA_KEYTYPE_INT8: diff --git a/storage/maria/ma_loghandler.c b/storage/maria/ma_loghandler.c index 9ed1d4b9d93..9364fe6a5c8 100644 --- a/storage/maria/ma_loghandler.c +++ b/storage/maria/ma_loghandler.c @@ -102,14 +102,14 @@ struct st_translog_buffer /* lock for the buffer. Current buffer also lock the handler */ pthread_mutex_t mutex; /* IO cache for current log */ - byte buffer[TRANSLOG_WRITE_BUFFER]; + uchar buffer[TRANSLOG_WRITE_BUFFER]; }; struct st_buffer_cursor { /* pointer on the buffer */ - byte *ptr; + uchar *ptr; /* current buffer */ struct st_translog_buffer *buffer; /* current page fill */ @@ -172,7 +172,7 @@ struct st_translog_descriptor static struct st_translog_descriptor log_descriptor; /* Marker for end of log */ -static byte end_of_log= 0; +static uchar end_of_log= 0; my_bool translog_inited= 0; @@ -207,7 +207,7 @@ typedef my_bool(*inwrite_rec_hook) (enum translog_record_type type, typedef uint16(*read_rec_hook) (enum translog_record_type type, uint16 read_length, uchar *read_buff, - byte *decoded_buff); + uchar *decoded_buff); /* Descriptor of log record type @@ -603,7 +603,7 @@ uchar NEAR maria_trans_file_magic[]= static my_bool translog_write_file_header() { ulonglong timestamp; - byte page_buff[TRANSLOG_PAGE_SIZE], *page= page_buff; + uchar page_buff[TRANSLOG_PAGE_SIZE], *page= page_buff; DBUG_ENTER("translog_write_file_header"); /* file tag */ @@ -816,7 +816,7 @@ static my_bool translog_buffer_unlock(struct st_translog_buffer *buffer) static void translog_new_page_header(TRANSLOG_ADDRESS *horizon, struct st_buffer_cursor *cursor) { - byte *ptr; + uchar *ptr; DBUG_ENTER("translog_new_page_header"); DBUG_ASSERT(cursor->ptr); @@ -830,7 +830,7 @@ static void translog_new_page_header(TRANSLOG_ADDRESS *horizon, /* File number */ int3store(ptr, LSN_FILE_NO(*horizon)); ptr+= 3; - *(ptr++)= (byte) log_descriptor.flags; + *(ptr++)= (uchar) log_descriptor.flags; if (log_descriptor.flags & TRANSLOG_PAGE_CRC) { #ifndef DBUG_OFF @@ -880,10 +880,10 @@ static void translog_new_page_header(TRANSLOG_ADDRESS *horizon, except the first sector that is protected by page header. */ -static void translog_put_sector_protection(byte *page, +static void translog_put_sector_protection(uchar *page, struct st_buffer_cursor *cursor) { - byte *table= page + log_descriptor.page_overhead - + uchar *table= page + log_descriptor.page_overhead - (TRANSLOG_PAGE_SIZE / DISK_DRIVE_SECTOR_SIZE) * 2; uint16 value= uint2korr(table) + cursor->write_counter; uint16 last_protected_sector= ((cursor->previous_offset - 1) / @@ -942,7 +942,7 @@ static void translog_put_sector_protection(byte *page, CRC32 */ -static uint32 translog_crc(byte *area, uint length) +static uint32 translog_crc(uchar *area, uint length) { return crc32(0L, (unsigned char*) area, length); } @@ -961,7 +961,7 @@ static void translog_finish_page(TRANSLOG_ADDRESS *horizon, struct st_buffer_cursor *cursor) { uint16 left= TRANSLOG_PAGE_SIZE - cursor->current_page_fill; - byte *page= cursor->ptr -cursor->current_page_fill; + uchar *page= cursor->ptr -cursor->current_page_fill; DBUG_ENTER("translog_finish_page"); DBUG_PRINT("enter", ("Buffer: #%u 0x%lx " "Buffer addr: (%lu,0x%lx) " @@ -1285,7 +1285,7 @@ static void translog_get_sent_to_file(LSN *lsn) first chunk offset */ -static my_bool translog_get_first_chunk_offset(byte *page) +static my_bool translog_get_first_chunk_offset(uchar *page) { uint16 page_header= 7; DBUG_ENTER("translog_get_first_chunk_offset"); @@ -1309,7 +1309,7 @@ static my_bool translog_get_first_chunk_offset(byte *page) */ static void -translog_write_variable_record_1group_code_len(byte *dst, +translog_write_variable_record_1group_code_len(uchar *dst, translog_size_t length, uint16 header_len) { @@ -1350,7 +1350,7 @@ translog_write_variable_record_1group_code_len(byte *dst, decoded length */ -static translog_size_t translog_variable_record_1group_decode_len(byte **src) +static translog_size_t translog_variable_record_1group_decode_len(uchar **src) { uint8 first= (uint8) (**src); switch (first) { @@ -1386,7 +1386,7 @@ static translog_size_t translog_variable_record_1group_decode_len(byte **src) total length of the chunk */ -static uint16 translog_get_total_chunk_length(byte *page, uint16 offset) +static uint16 translog_get_total_chunk_length(uchar *page, uint16 offset) { DBUG_ENTER("translog_get_total_chunk_length"); switch (page[offset] & TRANSLOG_CHUNK_TYPE) { @@ -1394,8 +1394,8 @@ static uint16 translog_get_total_chunk_length(byte *page, uint16 offset) { /* 0 chunk referred as LSN (head or tail) */ translog_size_t rec_len; - byte *start= page + offset; - byte *ptr= start + 1 + 2; + uchar *start= page + offset; + uchar *ptr= start + 1 + 2; uint16 chunk_len, header_len, page_rest; DBUG_PRINT("info", ("TRANSLOG_CHUNK_LSN")); rec_len= translog_variable_record_1group_decode_len(&ptr); @@ -1418,7 +1418,7 @@ static uint16 translog_get_total_chunk_length(byte *page, uint16 offset) } case TRANSLOG_CHUNK_FIXED: { - byte *ptr; + uchar *ptr; uint type= page[offset] & TRANSLOG_REC_TYPE; uint length; int i; @@ -1564,7 +1564,7 @@ static my_bool translog_buffer_flush(struct st_translog_buffer *buffer) 1 Error */ -static my_bool translog_recover_page_up_to_sector(byte *page, uint16 offset) +static my_bool translog_recover_page_up_to_sector(uchar *page, uint16 offset) { uint16 chunk_offset= translog_get_first_chunk_offset(page), valid_chunk_end; DBUG_ENTER("translog_recover_page_up_to_sector"); @@ -1630,11 +1630,11 @@ static my_bool translog_recover_page_up_to_sector(byte *page, uint16 offset) 0 OK 1 Error */ -static my_bool translog_page_validator(byte *page_addr, gptr data_ptr) +static my_bool translog_page_validator(uchar *page_addr, uchar* data_ptr) { uint this_page_page_overhead; uint flags; - byte *page= (byte*) page_addr, *page_pos; + uchar *page= (uchar*) page_addr, *page_pos; TRANSLOG_VALIDATOR_DATA *data= (TRANSLOG_VALIDATOR_DATA *) data_ptr; TRANSLOG_ADDRESS addr= *(data->addr); DBUG_ENTER("translog_page_validator"); @@ -1683,7 +1683,7 @@ static my_bool translog_page_validator(byte *page_addr, gptr data_ptr) if (flags & TRANSLOG_SECTOR_PROTECTION) { uint i, offset; - byte *table= page_pos; + uchar *table= page_pos; uint16 current= uint2korr(table); for (i= 2, offset= DISK_DRIVE_SECTOR_SIZE; i < (TRANSLOG_PAGE_SIZE / DISK_DRIVE_SECTOR_SIZE) * 2; @@ -1739,7 +1739,7 @@ static my_bool translog_page_validator(byte *page_addr, gptr data_ptr) # pointer to the page cache which should be used to read this page */ -static byte *translog_get_page(TRANSLOG_VALIDATOR_DATA *data, byte *buffer) +static uchar *translog_get_page(TRANSLOG_VALIDATOR_DATA *data, uchar *buffer) { TRANSLOG_ADDRESS addr= *(data->addr); uint cache_index; @@ -1766,13 +1766,13 @@ static byte *translog_get_page(TRANSLOG_VALIDATOR_DATA *data, byte *buffer) } file.file= log_descriptor.log_file_num[cache_index]; - buffer= (byte*) + buffer= (uchar*) pagecache_valid_read(log_descriptor.pagecache, &file, LSN_OFFSET(addr) / TRANSLOG_PAGE_SIZE, 3, (char*) buffer, PAGECACHE_PLAIN_PAGE, PAGECACHE_LOCK_LEFT_UNLOCKED, 0, - &translog_page_validator, (gptr) data); + &translog_page_validator, (uchar*) data); } else { @@ -1790,7 +1790,7 @@ static byte *translog_get_page(TRANSLOG_VALIDATOR_DATA *data, byte *buffer) if (my_pread(file, (char*) buffer, TRANSLOG_PAGE_SIZE, LSN_OFFSET(addr), MYF(MY_FNABP | MY_WME))) buffer= NULL; - else if (translog_page_validator((byte*) buffer, (gptr) data)) + else if (translog_page_validator((uchar*) buffer, (uchar*) data)) buffer= NULL; my_close(file, MYF(MY_WME)); } @@ -1879,7 +1879,7 @@ static uint translog_variable_record_length_bytes(translog_size_t length) 0 Error */ -static uint16 translog_get_chunk_header_length(byte *page, uint16 offset) +static uint16 translog_get_chunk_header_length(uchar *page, uint16 offset) { DBUG_ENTER("translog_get_chunk_header_length"); page+= offset; @@ -1888,8 +1888,8 @@ static uint16 translog_get_chunk_header_length(byte *page, uint16 offset) { /* 0 chunk referred as LSN (head or tail) */ translog_size_t rec_len; - byte *start= page; - byte *ptr= start + 1 + 2; + uchar *start= page; + uchar *ptr= start + 1 + 2; uint16 chunk_len, header_len; DBUG_PRINT("info", ("TRANSLOG_CHUNK_LSN")); rec_len= translog_variable_record_1group_decode_len(&ptr); @@ -2107,7 +2107,7 @@ my_bool translog_init(const char *directory, do { TRANSLOG_VALIDATOR_DATA data; - byte buffer[TRANSLOG_PAGE_SIZE], *page; + uchar buffer[TRANSLOG_PAGE_SIZE], *page; data.addr= ¤t_page; if ((page= translog_get_page(&data, buffer)) == NULL) DBUG_RETURN(1); @@ -2155,7 +2155,7 @@ my_bool translog_init(const char *directory, if (logs_found && !old_log_was_recovered && old_flags == flags) { TRANSLOG_VALIDATOR_DATA data; - byte buffer[TRANSLOG_PAGE_SIZE], *page; + uchar buffer[TRANSLOG_PAGE_SIZE], *page; uint16 chunk_offset; data.addr= &last_valid_page; /* continue old log */ @@ -2169,7 +2169,7 @@ my_bool translog_init(const char *directory, log_descriptor.horizon= last_valid_page; translog_start_buffer(log_descriptor.buffers, &log_descriptor.bc, 0); /* - Free space if filled with 0 and first byte of + Free space if filled with 0 and first uchar of real chunk can't be 0 */ while (chunk_offset < TRANSLOG_PAGE_SIZE && page[chunk_offset] != '\0') @@ -2329,7 +2329,7 @@ void translog_destroy() pthread_mutex_destroy(&log_descriptor.sent_to_file_lock); my_close(log_descriptor.directory_fd, MYF(MY_WME)); my_atomic_rwlock_destroy(&LOCK_id_to_share); - my_free((gptr)(id_to_share + 1), MYF(MY_ALLOW_ZERO_PTR)); + my_free((uchar*)(id_to_share + 1), MYF(MY_ALLOW_ZERO_PTR)); translog_inited= 0; } DBUG_VOID_RETURN; @@ -2481,7 +2481,7 @@ static my_bool translog_page_next(TRANSLOG_ADDRESS *horizon, static my_bool translog_write_data_on_page(TRANSLOG_ADDRESS *horizon, struct st_buffer_cursor *cursor, translog_size_t length, - byte *buffer) + uchar *buffer) { DBUG_ENTER("translog_write_data_on_page"); DBUG_PRINT("enter", ("Chunk length: %lu Page size %u", @@ -2547,11 +2547,11 @@ static my_bool translog_write_parts_on_page(TRANSLOG_ADDRESS *horizon, { translog_size_t len; LEX_STRING *part; - byte *buff; + uchar *buff; DBUG_ASSERT(cur < parts->elements); part= parts->parts + cur; - buff= (byte*) part->str; + buff= (uchar*) part->str; DBUG_PRINT("info", ("Part: %u Length: %lu left: %lu buff: 0x%lx", (uint) (cur + 1), (ulong) part->length, (ulong) left, (ulong) buff)); @@ -2623,7 +2623,7 @@ translog_write_variable_record_1group_header(struct st_translog_parts *parts, enum translog_record_type type, SHORT_TRANSACTION_ID short_trid, uint16 header_length, - byte *chunk0_header) + uchar *chunk0_header) { LEX_STRING *part; DBUG_ASSERT(parts->current != 0); /* first part is left for header */ @@ -2631,7 +2631,7 @@ translog_write_variable_record_1group_header(struct st_translog_parts *parts, parts->total_record_length+= (part->length= header_length); part->str= (char*)chunk0_header; /* puts chunk type */ - *chunk0_header= (byte) (type | TRANSLOG_CHUNK_LSN); + *chunk0_header= (uchar) (type | TRANSLOG_CHUNK_LSN); int2store(chunk0_header + 1, short_trid); /* puts record length */ translog_write_variable_record_1group_code_len(chunk0_header + 3, @@ -2706,7 +2706,7 @@ translog_write_variable_record_chunk2_page(struct st_translog_parts *parts, { struct st_translog_buffer *buffer_to_flush; int rc; - byte chunk2_header[1]; + uchar chunk2_header[1]; DBUG_ENTER("translog_write_variable_record_chunk2_page"); chunk2_header[0]= TRANSLOG_CHUNK_NOHDR; @@ -2756,7 +2756,7 @@ translog_write_variable_record_chunk3_page(struct st_translog_parts *parts, struct st_translog_buffer *buffer_to_flush; LEX_STRING *part; int rc; - byte chunk3_header[1 + 2]; + uchar chunk3_header[1 + 2]; DBUG_ENTER("translog_write_variable_record_chunk3_page"); LINT_INIT(buffer_to_flush); @@ -2783,7 +2783,7 @@ translog_write_variable_record_chunk3_page(struct st_translog_parts *parts, parts->total_record_length+= (part->length= 1 + 2); part->str= (char*)chunk3_header; /* Puts chunk type */ - *chunk3_header= (byte) (TRANSLOG_CHUNK_LNGTH); + *chunk3_header= (uchar) (TRANSLOG_CHUNK_LNGTH); /* Puts chunk length */ int2store(chunk3_header + 1, length); @@ -3027,7 +3027,7 @@ translog_write_variable_record_1group(LSN *lsn, uint i; translog_size_t record_rest, full_pages, first_page; uint additional_chunk3_page= 0; - byte chunk0_header[1 + 2 + 5 + 2]; + uchar chunk0_header[1 + 2 + 5 + 2]; DBUG_ENTER("translog_write_variable_record_1group"); *lsn= horizon= log_descriptor.horizon; @@ -3176,7 +3176,7 @@ translog_write_variable_record_1chunk(LSN *lsn, TRN *trn) { int rc; - byte chunk0_header[1 + 2 + 5 + 2]; + uchar chunk0_header[1 + 2 + 5 + 2]; DBUG_ENTER("translog_write_variable_record_1chunk"); translog_write_variable_record_1group_header(parts, type, short_trid, @@ -3239,7 +3239,7 @@ translog_write_variable_record_1chunk(LSN *lsn, NULL Error */ -static byte *translog_put_LSN_diff(LSN base_lsn, LSN lsn, byte *dst) +static uchar *translog_put_LSN_diff(LSN base_lsn, LSN lsn, uchar *dst) { DBUG_ENTER("translog_put_LSN_diff"); DBUG_PRINT("enter", ("Base: (0x%lu,0x%lx) val: (0x%lu,0x%lx) dst: 0x%lx", @@ -3257,7 +3257,7 @@ static byte *translog_put_LSN_diff(LSN base_lsn, LSN lsn, byte *dst) { dst-= 2; /* - Note we store this high byte first to ensure that first byte has + Note we store this high uchar first to ensure that first uchar has 0 in the 3 upper bits. */ dst[0]= diff >> 8; @@ -3347,7 +3347,7 @@ static byte *translog_put_LSN_diff(LSN base_lsn, LSN lsn, byte *dst) pointer to buffer after decoded LSN */ -static byte *translog_get_LSN_from_diff(LSN base_lsn, byte *src, byte *dst) +static uchar *translog_get_LSN_from_diff(LSN base_lsn, uchar *src, uchar *dst) { LSN lsn; uint32 diff; @@ -3432,7 +3432,7 @@ static byte *translog_get_LSN_from_diff(LSN base_lsn, byte *src, byte *dst) static my_bool translog_relative_LSN_encode(struct st_translog_parts *parts, LSN base_lsn, - uint lsns, byte *compressed_LSNs) + uint lsns, uchar *compressed_LSNs) { LEX_STRING *part; uint lsns_len= lsns * LSN_STORE_SIZE; @@ -3450,14 +3450,14 @@ static my_bool translog_relative_LSN_encode(struct st_translog_parts *parts, uint copied= part->length; LEX_STRING *next_part; DBUG_PRINT("info", ("Using buffer: 0x%lx", (ulong) compressed_LSNs)); - memcpy(buffer, (byte*)part->str, part->length); + memcpy(buffer, (uchar*)part->str, part->length); next_part= parts->parts + parts->current + 1; do { DBUG_ASSERT(next_part < parts->parts + parts->elements); if ((next_part->length + copied) < lsns_len) { - memcpy(buffer + copied, (byte*)next_part->str, + memcpy(buffer + copied, (uchar*)next_part->str, next_part->length); copied+= next_part->length; next_part->length= 0; next_part->str= 0; @@ -3469,7 +3469,7 @@ static my_bool translog_relative_LSN_encode(struct st_translog_parts *parts, else { uint len= lsns_len - copied; - memcpy(buffer + copied, (byte*)next_part->str, len); + memcpy(buffer + copied, (uchar*)next_part->str, len); copied= lsns_len; next_part->str+= len; next_part->length-= len; @@ -3489,8 +3489,8 @@ static my_bool translog_relative_LSN_encode(struct st_translog_parts *parts, /* Compress */ LSN ref; int economy; - byte *src_ptr; - byte *dst_ptr= compressed_LSNs + (MAX_NUMBER_OF_LSNS_PER_RECORD * + uchar *src_ptr; + uchar *dst_ptr= compressed_LSNs + (MAX_NUMBER_OF_LSNS_PER_RECORD * COMPRESSED_LSN_MAX_STORE_SIZE); for (src_ptr= buffer + lsns_len - LSN_STORE_SIZE; src_ptr >= buffer; @@ -3558,8 +3558,8 @@ translog_write_variable_record_mgroup(LSN *lsn, uint16 page_capacity= log_descriptor.page_capacity_chunk_2 + 1; uint16 last_page_capacity; my_bool new_page_before_chunk0= 1, first_chunk0= 1; - byte chunk0_header[1 + 2 + 5 + 2 + 2], group_desc[7 + 1]; - byte chunk2_header[1]; + uchar chunk0_header[1 + 2 + 5 + 2 + 2], group_desc[7 + 1]; + uchar chunk2_header[1]; uint header_fixed_part= header_length + 2; uint groups_per_page= (page_capacity - header_fixed_part) / (7 + 1); DBUG_ENTER("translog_write_variable_record_mgroup"); @@ -3602,7 +3602,7 @@ translog_write_variable_record_mgroup(LSN *lsn, But here we assign number of chunks - 1 */ group.num= full_pages; - if (insert_dynamic(&groups, (gptr) &group)) + if (insert_dynamic(&groups, (uchar*) &group)) { UNRECOVERABLE_ERROR(("insert into array failed")); goto err_unlock; @@ -3701,7 +3701,7 @@ translog_write_variable_record_mgroup(LSN *lsn, cursor= log_descriptor.bc; cursor.chaser= 1; group.num= 0; /* 0 because it does not matter */ - if (insert_dynamic(&groups, (gptr) &group)) + if (insert_dynamic(&groups, (uchar*) &group)) { UNRECOVERABLE_ERROR(("insert into array failed")); goto err_unlock; @@ -3797,7 +3797,7 @@ translog_write_variable_record_mgroup(LSN *lsn, { DBUG_PRINT("info", ("chunk 3")); DBUG_ASSERT(full_pages == 0); - byte chunk3_header[3]; + uchar chunk3_header[3]; chunk3_pages= 0; chunk3_header[0]= TRANSLOG_CHUNK_LNGTH; int2store(chunk3_header + 1, chunk3_size); @@ -3846,7 +3846,7 @@ translog_write_variable_record_mgroup(LSN *lsn, (ulong) LSN_OFFSET(horizon))); - *chunk0_header= (byte) (type |TRANSLOG_CHUNK_LSN); + *chunk0_header= (uchar) (type |TRANSLOG_CHUNK_LSN); int2store(chunk0_header + 1, short_trid); translog_write_variable_record_1group_code_len(chunk0_header + 3, parts->record_length, @@ -3975,7 +3975,7 @@ static my_bool translog_write_variable_record(LSN *lsn, ulong buffer_rest; uint page_rest; /* Max number of such LSNs per record is 2 */ - byte compressed_LSNs[MAX_NUMBER_OF_LSNS_PER_RECORD * + uchar compressed_LSNs[MAX_NUMBER_OF_LSNS_PER_RECORD * COMPRESSED_LSN_MAX_STORE_SIZE]; DBUG_ENTER("translog_write_variable_record"); @@ -4001,7 +4001,7 @@ static my_bool translog_write_variable_record(LSN *lsn, log_record_type_descriptor[type].read_header_len)); translog_page_next(&log_descriptor.horizon, &log_descriptor.bc, &buffer_to_flush); - /* Chunk 2 header is 1 byte, so full page capacity will be one byte more */ + /* Chunk 2 header is 1 byte, so full page capacity will be one uchar more */ page_rest= log_descriptor.page_capacity_chunk_2 + 1; DBUG_PRINT("info", ("page_rest: %u", page_rest)); } @@ -4087,9 +4087,9 @@ static my_bool translog_write_fixed_record(LSN *lsn, TRN *trn) { struct st_translog_buffer *buffer_to_flush= NULL; - byte chunk1_header[1 + 2]; + uchar chunk1_header[1 + 2]; /* Max number of such LSNs per record is 2 */ - byte compressed_LSNs[MAX_NUMBER_OF_LSNS_PER_RECORD * + uchar compressed_LSNs[MAX_NUMBER_OF_LSNS_PER_RECORD * COMPRESSED_LSN_MAX_STORE_SIZE]; LEX_STRING *part; int rc; @@ -4162,7 +4162,7 @@ static my_bool translog_write_fixed_record(LSN *lsn, part= parts->parts + (--parts->current); parts->total_record_length+= (part->length= 1 + 2); part->str= (char*)chunk1_header; - *chunk1_header= (byte) (type | TRANSLOG_CHUNK_FIXED); + *chunk1_header= (uchar) (type | TRANSLOG_CHUNK_FIXED); int2store(chunk1_header + 1, short_trid); rc= translog_write_parts_on_page(&log_descriptor.horizon, @@ -4364,8 +4364,8 @@ my_bool translog_write_record(LSN *lsn, position in sources after decoded LSN(s) */ -static byte *translog_relative_LSN_decode(LSN base_lsn, - byte *src, byte *dst, uint lsns) +static uchar *translog_relative_LSN_decode(LSN base_lsn, + uchar *src, uchar *dst, uint lsns) { uint i; for (i= 0; i < lsns; i++, dst+= LSN_STORE_SIZE) @@ -4391,15 +4391,15 @@ static byte *translog_relative_LSN_decode(LSN base_lsn, part of the header */ -translog_size_t translog_fixed_length_header(byte *page, +translog_size_t translog_fixed_length_header(uchar *page, translog_size_t page_offset, TRANSLOG_HEADER_BUFFER *buff) { struct st_log_record_type_descriptor *desc= log_record_type_descriptor + buff->type; - byte *src= page + page_offset + 3; - byte *dst= buff->header; - byte *start= src; + uchar *src= page + page_offset + 3; + uchar *dst= buff->header; + uchar *start= src; uint lsns= desc->compressed_LSN; uint length= desc->fixed_length; @@ -4439,7 +4439,7 @@ void translog_free_record_header(TRANSLOG_HEADER_BUFFER *buff) DBUG_ENTER("translog_free_record_header"); if (buff->groups_no != 0) { - my_free((gptr) buff->groups, MYF(0)); + my_free((uchar*) buff->groups, MYF(0)); buff->groups_no= 0; } DBUG_VOID_RETURN; @@ -4719,7 +4719,7 @@ translog_get_next_chunk(TRANSLOG_SCANNER_DATA *scanner) part of the header */ -translog_size_t translog_variable_length_header(byte *page, +translog_size_t translog_variable_length_header(uchar *page, translog_size_t page_offset, TRANSLOG_HEADER_BUFFER *buff, TRANSLOG_SCANNER_DATA @@ -4727,8 +4727,8 @@ translog_size_t translog_variable_length_header(byte *page, { struct st_log_record_type_descriptor *desc= (log_record_type_descriptor + buff->type); - byte *src= page + page_offset + 1 + 2; - byte *dst= buff->header; + uchar *src= page + page_offset + 1 + 2; + uchar *dst= buff->header; LSN base_lsn; uint lsns= desc->compressed_LSN; uint16 chunk_len; @@ -4848,7 +4848,7 @@ translog_size_t translog_variable_length_header(byte *page, } if (lsns) { - byte *start= src; + uchar *start= src; src= translog_relative_LSN_decode(base_lsn, src, dst, lsns); lsns*= LSN_STORE_SIZE; dst+= lsns; @@ -4890,7 +4890,7 @@ translog_size_t translog_variable_length_header(byte *page, */ translog_size_t -translog_read_record_header_from_buffer(byte *page, +translog_read_record_header_from_buffer(uchar *page, uint16 page_offset, TRANSLOG_HEADER_BUFFER *buff, TRANSLOG_SCANNER_DATA *scanner) @@ -4946,7 +4946,7 @@ translog_read_record_header_from_buffer(byte *page, translog_size_t translog_read_record_header(LSN lsn, TRANSLOG_HEADER_BUFFER *buff) { - byte buffer[TRANSLOG_PAGE_SIZE], *page; + uchar buffer[TRANSLOG_PAGE_SIZE], *page; translog_size_t page_offset= LSN_OFFSET(lsn) % TRANSLOG_PAGE_SIZE; TRANSLOG_ADDRESS addr; TRANSLOG_VALIDATOR_DATA data; @@ -5232,7 +5232,7 @@ static my_bool translog_init_reader_data(LSN lsn, translog_size_t translog_read_record(LSN lsn, translog_size_t offset, translog_size_t length, - byte *buffer, + uchar *buffer, struct st_translog_reader_data *data) { translog_size_t requested_length= length; @@ -5331,7 +5331,7 @@ static void translog_force_current_buffer_to_finish() struct st_translog_buffer *new_buffer= (log_descriptor.buffers + new_buffer_no); struct st_translog_buffer *old_buffer= log_descriptor.bc.buffer; - byte *data= log_descriptor.bc.ptr -log_descriptor.bc.current_page_fill; + uchar *data= log_descriptor.bc.ptr -log_descriptor.bc.current_page_fill; uint16 left= TRANSLOG_PAGE_SIZE - log_descriptor.bc.current_page_fill; uint16 current_page_fill, write_counter, previous_offset; DBUG_ENTER("translog_force_current_buffer_to_finish"); diff --git a/storage/maria/ma_loghandler.h b/storage/maria/ma_loghandler.h index 0a160a9bc53..a831f088e9b 100644 --- a/storage/maria/ma_loghandler.h +++ b/storage/maria/ma_loghandler.h @@ -127,7 +127,7 @@ typedef struct st_translog_header_buffer Buffer for write decoded header of the record (depend on the record type) */ - byte header[TRANSLOG_RECORD_HEADER_MAX_SIZE]; + uchar header[TRANSLOG_RECORD_HEADER_MAX_SIZE]; /* number of groups listed in */ uint groups_no; /* in multi-group number of chunk0 pages (valid only if groups_no > 0) */ @@ -151,12 +151,12 @@ typedef struct st_translog_header_buffer typedef struct st_translog_scanner_data { - byte buffer[TRANSLOG_PAGE_SIZE]; /* buffer for page content */ + uchar buffer[TRANSLOG_PAGE_SIZE]; /* buffer for page content */ TRANSLOG_ADDRESS page_addr; /* current page address */ /* end of the log which we saw last time */ TRANSLOG_ADDRESS horizon; TRANSLOG_ADDRESS last_file_page; /* Last page on in this file */ - byte *page; /* page content pointer */ + uchar *page; /* page content pointer */ /* offset of the chunk in the page */ translog_size_t page_offset; /* set horizon only once at init */ @@ -214,7 +214,7 @@ extern void translog_free_record_header(TRANSLOG_HEADER_BUFFER *buff); extern translog_size_t translog_read_record(LSN lsn, translog_size_t offset, translog_size_t length, - byte *buffer, + uchar *buffer, struct st_translog_reader_data *data); diff --git a/storage/maria/ma_loghandler_lsn.h b/storage/maria/ma_loghandler_lsn.h index c641337e8ba..0c02fe2c489 100644 --- a/storage/maria/ma_loghandler_lsn.h +++ b/storage/maria/ma_loghandler_lsn.h @@ -54,7 +54,7 @@ typedef TRANSLOG_ADDRESS LSN; #define LSN_REPLACE_OFFSET(L, S) (LSN_FINE_NO_PART(L) | (S)) /* - an 8-byte type whose most significant byte is used for "flags"; 7 + an 8-byte type whose most significant uchar is used for "flags"; 7 other bytes are a LSN. */ typedef LSN LSN_WITH_FLAGS; diff --git a/storage/maria/ma_open.c b/storage/maria/ma_open.c index 4e72adf3b7e..15f8dcf4e51 100644 --- a/storage/maria/ma_open.c +++ b/storage/maria/ma_open.c @@ -38,7 +38,7 @@ static my_bool maria_scan_init_dummy(MARIA_HA *info); static void maria_scan_end_dummy(MARIA_HA *info); static my_bool maria_once_init_dummy(MARIA_SHARE *, File); static my_bool maria_once_end_dummy(MARIA_SHARE *); -static byte *_ma_base_info_read(byte *ptr, MARIA_BASE_INFO *base); +static uchar *_ma_base_info_read(uchar *ptr, MARIA_BASE_INFO *base); #define get_next_element(to,pos,size) { memcpy((char*) to,pos,(size_t) size); \ pos+=size;} @@ -97,7 +97,7 @@ static MARIA_HA *maria_clone_internal(MARIA_SHARE *share, int mode, DBUG_ENTER("maria_clone_internal"); errpos= 0; - bzero((byte*) &info,sizeof(info)); + bzero((uchar*) &info,sizeof(info)); if (mode == O_RDWR && share->mode == O_RDONLY) { @@ -200,7 +200,7 @@ err: switch (errpos) { case 6: (*share->end)(&info); - my_free((gptr) m_info,MYF(0)); + my_free((uchar*) m_info,MYF(0)); /* fall through */ case 5: if (data_file < 0) @@ -255,7 +255,7 @@ MARIA_HA *maria_open(const char *name, int mode, uint open_flags) kfile= -1; errpos= 0; head_length=sizeof(share_buff.state.header); - bzero((byte*) &info,sizeof(info)); + bzero((uchar*) &info,sizeof(info)); my_realpath(name_buff, fn_format(org_name,name,"",MARIA_NAME_IEXT, MY_UNPACK_FILENAME),MYF(0)); @@ -263,7 +263,7 @@ MARIA_HA *maria_open(const char *name, int mode, uint open_flags) if (!(old_info=_ma_test_if_reopen(name_buff))) { share= &share_buff; - bzero((gptr) &share_buff,sizeof(share_buff)); + bzero((uchar*) &share_buff,sizeof(share_buff)); share_buff.state.rec_per_key_part=rec_per_key_part; share_buff.state.key_root=key_root; share_buff.pagecache= multi_pagecache_search(name_buff, strlen(name_buff), @@ -290,8 +290,8 @@ MARIA_HA *maria_open(const char *name, int mode, uint open_flags) my_errno= HA_ERR_NOT_A_TABLE; goto err; } - if (memcmp((byte*) share->state.header.file_version, - (byte*) maria_file_magic, 4)) + if (memcmp((uchar*) share->state.header.file_version, + (uchar*) maria_file_magic, 4)) { DBUG_PRINT("error",("Wrong header in %s",name_buff)); DBUG_DUMP("error_dump",(char*) share->state.header.file_version, @@ -638,7 +638,7 @@ MARIA_HA *maria_open(const char *name, int mode, uint open_flags) (keys ? MARIA_INDEX_BLOCK_MARGIN * share->block_size * keys : 0)); share->block_size= share->base.block_size; - my_afree((gptr) disk_cache); + my_afree((uchar*) disk_cache); _ma_setup_functions(share); if ((*share->once_init)(share, info.dfile.file)) goto err; @@ -724,12 +724,12 @@ err: (*share->once_end)(share); /* fall through */ case 4: - my_free((gptr) share,MYF(0)); + my_free((uchar*) share,MYF(0)); /* fall through */ case 3: /* fall through */ case 2: - my_afree((gptr) disk_cache); + my_afree((uchar*) disk_cache); /* fall through */ case 1: VOID(my_close(kfile,MYF(0))); @@ -748,13 +748,13 @@ err: Reallocate a buffer, if the current buffer is not large enough */ -my_bool _ma_alloc_buffer(byte **old_addr, my_size_t *old_size, - my_size_t new_size) +my_bool _ma_alloc_buffer(uchar **old_addr, size_t *old_size, + size_t new_size) { if (*old_size < new_size) { - byte *addr; - if (!(addr= (byte*) my_realloc((gptr) *old_addr, new_size, + uchar *addr; + if (!(addr= (uchar*) my_realloc((uchar*) *old_addr, new_size, MYF(MY_ALLOW_ZERO_PTR)))) return 1; *old_addr= addr; @@ -1002,7 +1002,7 @@ uint _ma_state_info_write(File file, MARIA_STATE_INFO *state, uint pWrite) } -byte *_ma_state_info_read(byte *ptr, MARIA_STATE_INFO *state) +uchar *_ma_state_info_read(uchar *ptr, MARIA_STATE_INFO *state) { uint i,keys,key_parts; memcpy_fixed(&state->header,ptr, sizeof(state->header)); @@ -1120,7 +1120,7 @@ uint _ma_base_info_write(File file, MARIA_BASE_INFO *base) } -static byte *_ma_base_info_read(byte *ptr, MARIA_BASE_INFO *base) +static uchar *_ma_base_info_read(uchar *ptr, MARIA_BASE_INFO *base) { base->keystart= mi_sizekorr(ptr); ptr+= 8; base->max_data_file_length= mi_sizekorr(ptr); ptr+= 8; @@ -1268,7 +1268,7 @@ char *_ma_uniquedef_read(char *ptr, MARIA_UNIQUEDEF *def) def->keysegs = mi_uint2korr(ptr); def->key = ptr[2]; def->null_are_equal=ptr[3]; - return ptr+4; /* 1 extra byte */ + return ptr+4; /* 1 extra uchar */ } /*************************************************************************** diff --git a/storage/maria/ma_packrec.c b/storage/maria/ma_packrec.c index f3187314f0e..5d54e42ac4f 100644 --- a/storage/maria/ma_packrec.c +++ b/storage/maria/ma_packrec.c @@ -44,14 +44,14 @@ { bits-=(bit+1); break; } \ pos+= *pos -/* Size in uint16 of a Huffman tree for byte compression of 256 byte values. */ +/* Size in uint16 of a Huffman tree for uchar compression of 256 uchar values. */ #define OFFSET_TABLE_SIZE 512 static my_bool _ma_read_pack_info(MARIA_SHARE *share, File file, pbool fix_keys); static uint read_huff_table(MARIA_BIT_BUFF *bit_buff, MARIA_DECODE_TREE *decode_tree, - uint16 **decode_table,byte **intervall_buff, + uint16 **decode_table,uchar **intervall_buff, uint16 *tmp_buff); static void make_quick_table(uint16 *to_table,uint16 *decode_table, uint *next_free,uint value,uint bits, @@ -63,49 +63,49 @@ static uint copy_decode_table(uint16 *to_pos,uint offset, static uint find_longest_bitstream(uint16 *table, uint16 *end); static void (*get_unpack_function(MARIA_COLUMNDEF *rec))(MARIA_COLUMNDEF *field, MARIA_BIT_BUFF *buff, - byte *to, - byte *end); + uchar *to, + uchar *end); static void uf_zerofill_skip_zero(MARIA_COLUMNDEF *rec, MARIA_BIT_BUFF *bit_buff, - byte *to,byte *end); + uchar *to,uchar *end); static void uf_skip_zero(MARIA_COLUMNDEF *rec,MARIA_BIT_BUFF *bit_buff, - byte *to,byte *end); + uchar *to,uchar *end); static void uf_space_normal(MARIA_COLUMNDEF *rec,MARIA_BIT_BUFF *bit_buff, - byte *to,byte *end); + uchar *to,uchar *end); static void uf_space_endspace_selected(MARIA_COLUMNDEF *rec, MARIA_BIT_BUFF *bit_buff, - byte *to, byte *end); + uchar *to, uchar *end); static void uf_endspace_selected(MARIA_COLUMNDEF *rec,MARIA_BIT_BUFF *bit_buff, - byte *to,byte *end); + uchar *to,uchar *end); static void uf_space_endspace(MARIA_COLUMNDEF *rec,MARIA_BIT_BUFF *bit_buff, - byte *to,byte *end); + uchar *to,uchar *end); static void uf_endspace(MARIA_COLUMNDEF *rec,MARIA_BIT_BUFF *bit_buff, - byte *to,byte *end); + uchar *to,uchar *end); static void uf_space_prespace_selected(MARIA_COLUMNDEF *rec, MARIA_BIT_BUFF *bit_buff, - byte *to, byte *end); + uchar *to, uchar *end); static void uf_prespace_selected(MARIA_COLUMNDEF *rec,MARIA_BIT_BUFF *bit_buff, - byte *to,byte *end); + uchar *to,uchar *end); static void uf_space_prespace(MARIA_COLUMNDEF *rec,MARIA_BIT_BUFF *bit_buff, - byte *to,byte *end); + uchar *to,uchar *end); static void uf_prespace(MARIA_COLUMNDEF *rec,MARIA_BIT_BUFF *bit_buff, - byte *to,byte *end); + uchar *to,uchar *end); static void uf_zerofill_normal(MARIA_COLUMNDEF *rec,MARIA_BIT_BUFF *bit_buff, - byte *to,byte *end); + uchar *to,uchar *end); static void uf_constant(MARIA_COLUMNDEF *rec,MARIA_BIT_BUFF *bit_buff, - byte *to,byte *end); + uchar *to,uchar *end); static void uf_intervall(MARIA_COLUMNDEF *rec,MARIA_BIT_BUFF *bit_buff, - byte *to,byte *end); + uchar *to,uchar *end); static void uf_zero(MARIA_COLUMNDEF *rec,MARIA_BIT_BUFF *bit_buff, - byte *to,byte *end); + uchar *to,uchar *end); static void uf_blob(MARIA_COLUMNDEF *rec, MARIA_BIT_BUFF *bit_buff, - byte *to, byte *end); + uchar *to, uchar *end); static void uf_varchar1(MARIA_COLUMNDEF *rec, MARIA_BIT_BUFF *bit_buff, - byte *to, byte *end); + uchar *to, uchar *end); static void uf_varchar2(MARIA_COLUMNDEF *rec, MARIA_BIT_BUFF *bit_buff, - byte *to, byte *end); + uchar *to, uchar *end); static void decode_bytes(MARIA_COLUMNDEF *rec,MARIA_BIT_BUFF *bit_buff, - byte *to,byte *end); + uchar *to,uchar *end); static uint decode_pos(MARIA_BIT_BUFF *bit_buff, MARIA_DECODE_TREE *decode_tree); static void init_bit_buffer(MARIA_BIT_BUFF *bit_buff,uchar *buffer, @@ -118,8 +118,8 @@ static uint read_pack_length(uint version, const uchar *buf, ulong *length); static uchar *_ma_mempack_get_block_info(MARIA_HA *maria, MARIA_BIT_BUFF *bit_buff, MARIA_BLOCK_INFO *info, - byte **rec_buff_p, - my_size_t *rec_buff_size_p, + uchar **rec_buff_p, + size_t *rec_buff_size_p, uchar *header); #endif @@ -154,8 +154,8 @@ my_bool _ma_once_end_pack_row(MARIA_SHARE *share) { if (share->decode_trees) { - my_free((gptr) share->decode_trees,MYF(0)); - my_free((gptr) share->decode_tables,MYF(0)); + my_free((uchar*) share->decode_trees,MYF(0)); + my_free((uchar*) share->decode_tables,MYF(0)); } return 0; } @@ -181,19 +181,19 @@ static my_bool _ma_read_pack_info(MARIA_SHARE *share, File file, maria_quick_table_bits=MAX_QUICK_TABLE_BITS; my_errno=0; - if (my_read(file,(byte*) header,sizeof(header),MYF(MY_NABP))) + if (my_read(file,(uchar*) header,sizeof(header),MYF(MY_NABP))) { if (!my_errno) my_errno=HA_ERR_END_OF_FILE; goto err0; } /* Only the first three bytes of magic number are independent of version. */ - if (memcmp((byte*) header, (byte*) maria_pack_file_magic, 3)) + if (memcmp((uchar*) header, (uchar*) maria_pack_file_magic, 3)) { my_errno=HA_ERR_WRONG_IN_RECORD; goto err0; } - share->pack.version= header[3]; /* fourth byte of magic number */ + share->pack.version= header[3]; /* fourth uchar of magic number */ share->pack.header_length= uint4korr(header+4); share->min_pack_length=(uint) uint4korr(header+8); share->max_pack_length=(uint) uint4korr(header+12); @@ -228,10 +228,10 @@ static my_bool _ma_read_pack_info(MARIA_SHARE *share, File file, */ if (!(share->decode_trees=(MARIA_DECODE_TREE*) my_malloc((uint) (trees*sizeof(MARIA_DECODE_TREE)+ - intervall_length*sizeof(byte)), + intervall_length*sizeof(uchar)), MYF(MY_WME)))) goto err0; - intervall_buff=(byte*) (share->decode_trees+trees); + intervall_buff=(uchar*) (share->decode_trees+trees); /* Memory segment #2: @@ -248,7 +248,7 @@ static my_bool _ma_read_pack_info(MARIA_SHARE *share, File file, MYF(MY_WME | MY_ZEROFILL)))) goto err1; tmp_buff=share->decode_tables+length; - disk_cache=(byte*) (tmp_buff+OFFSET_TABLE_SIZE); + disk_cache=(uchar*) (tmp_buff+OFFSET_TABLE_SIZE); if (my_read(file,disk_cache, (uint) (share->pack.header_length-sizeof(header)), @@ -284,8 +284,8 @@ static my_bool _ma_read_pack_info(MARIA_SHARE *share, File file, goto err3; /* Reallocate the decoding tables to the used size. */ decode_table=(uint16*) - my_realloc((gptr) share->decode_tables, - (uint) ((byte*) decode_table - (byte*) share->decode_tables), + my_realloc((uchar*) share->decode_tables, + (uint) ((uchar*) decode_table - (uchar*) share->decode_tables), MYF(MY_HOLD_ON_ERROR)); /* Fix the table addresses in the tree heads. */ { @@ -325,9 +325,9 @@ static my_bool _ma_read_pack_info(MARIA_SHARE *share, File file, err3: my_errno=HA_ERR_WRONG_IN_RECORD; err2: - my_free((gptr) share->decode_tables,MYF(0)); + my_free((uchar*) share->decode_tables,MYF(0)); err1: - my_free((gptr) share->decode_trees,MYF(0)); + my_free((uchar*) share->decode_trees,MYF(0)); err0: DBUG_RETURN(1); } @@ -352,7 +352,7 @@ err0: */ static uint read_huff_table(MARIA_BIT_BUFF *bit_buff, MARIA_DECODE_TREE *decode_tree, - uint16 **decode_table, byte **intervall_buff, + uint16 **decode_table, uchar **intervall_buff, uint16 *tmp_buff) { uint min_chr,elements,char_bits,offset_bits,size,intervall_length,table_bits, @@ -371,7 +371,7 @@ static uint read_huff_table(MARIA_BIT_BUFF *bit_buff, ptr=tmp_buff; ptr=tmp_buff; DBUG_PRINT("info", ("byte value compression")); - DBUG_PRINT("info", ("minimum byte value: %u", min_chr)); + DBUG_PRINT("info", ("minimum uchar value: %u", min_chr)); DBUG_PRINT("info", ("number of tree nodes: %u", elements)); DBUG_PRINT("info", ("bits for values: %u", char_bits)); DBUG_PRINT("info", ("bits for tree offsets: %u", offset_bits)); @@ -477,13 +477,13 @@ static uint read_huff_table(MARIA_BIT_BUFF *bit_buff, In most cases table_bits is 9. So there are 512 16-bit values. If the high-order bit (16) is set (IS_CHAR) then the array slot for - this value is a valid Huffman code for a resulting byte value. + this value is a valid Huffman code for a resulting uchar value. - The low-order 8 bits (1..8) are the resulting byte value. + The low-order 8 bits (1..8) are the resulting uchar value. - Bits 9..14 are the length of the Huffman code for this byte value. + Bits 9..14 are the length of the Huffman code for this uchar value. This means so many bits from the input stream were needed to - represent this byte value. The remaining bits belong to later + represent this uchar value. The remaining bits belong to later Huffman codes. This also means that for every Huffman code shorter than table_bits there are multiple entires in the array, which differ just in the unused bits. @@ -570,7 +570,7 @@ static void make_quick_table(uint16 *to_table, uint16 *decode_table, table Target quick_table position. bits Unused bits from max_bits. max_bits Total number of bits to collect (table_bits). - value The byte encoded by the found Huffman code. + value The uchar encoded by the found Huffman code. DESCRIPTION @@ -593,8 +593,8 @@ static void fill_quick_table(uint16 *table, uint bits, uint max_bits, DBUG_ENTER("fill_quick_table"); /* - Bits 1..8 of value represent the decoded byte value. - Bits 9..14 become the length of the Huffman code for this byte value. + Bits 1..8 of value represent the decoded uchar value. + Bits 9..14 become the length of the Huffman code for this uchar value. Bit 16 flags a valid code (IS_CHAR). */ value|= (max_bits - bits) << 8 | IS_CHAR; @@ -639,7 +639,7 @@ static uint copy_decode_table(uint16 *to_pos, uint offset, } else { - /* Copy the byte value. */ + /* Copy the uchar value. */ to_pos[offset]= *decode_table; /* Step behind this node. */ offset+=2; @@ -656,7 +656,7 @@ static uint copy_decode_table(uint16 *to_pos, uint offset, } else { - /* Copy the byte value. */ + /* Copy the uchar value. */ to_pos[prev_offset+1]= *decode_table; } DBUG_RETURN(offset); @@ -674,7 +674,7 @@ static uint copy_decode_table(uint16 *to_pos, uint offset, IMPLEMENTATION Recursively follow the branch(es) of the code pair on every level of - the tree until two byte values (and no branch) are found. Add one to + the tree until two uchar values (and no branch) are found. Add one to each level when returning back from each recursion stage. 'end' is used for error checking only. A clean tree terminates @@ -731,7 +731,7 @@ static uint find_longest_bitstream(uint16 *table, uint16 *end) HA_ERR_WRONG_IN_RECORD or -1 on error */ -int _ma_read_pack_record(MARIA_HA *info, byte *buf, MARIA_RECORD_POS filepos) +int _ma_read_pack_record(MARIA_HA *info, uchar *buf, MARIA_RECORD_POS filepos) { MARIA_BLOCK_INFO block_info; File file; @@ -745,7 +745,7 @@ int _ma_read_pack_record(MARIA_HA *info, byte *buf, MARIA_RECORD_POS filepos) &info->rec_buff, &info->rec_buff_size, file, filepos)) goto err; - if (my_read(file,(byte*) info->rec_buff + block_info.offset , + if (my_read(file,(uchar*) info->rec_buff + block_info.offset , block_info.rec_len - block_info.offset, MYF(MY_NABP))) goto panic; info->update|= HA_STATE_AKTIV; @@ -760,9 +760,9 @@ err: int _ma_pack_rec_unpack(register MARIA_HA *info, MARIA_BIT_BUFF *bit_buff, - register byte *to, byte *from, ulong reclength) + register uchar *to, uchar *from, ulong reclength) { - byte *end_field; + uchar *end_field; reg3 MARIA_COLUMNDEF *end; MARIA_COLUMNDEF *current_field; MARIA_SHARE *share=info->s; @@ -794,7 +794,7 @@ int _ma_pack_rec_unpack(register MARIA_HA *info, MARIA_BIT_BUFF *bit_buff, /* Return function to unpack field */ static void (*get_unpack_function(MARIA_COLUMNDEF *rec)) - (MARIA_COLUMNDEF *, MARIA_BIT_BUFF *, byte *, byte *) + (MARIA_COLUMNDEF *, MARIA_BIT_BUFF *, uchar *, uchar *) { switch (rec->base_type) { case FIELD_SKIP_ZERO: @@ -837,7 +837,7 @@ static void (*get_unpack_function(MARIA_COLUMNDEF *rec)) case FIELD_BLOB: return &uf_blob; case FIELD_VARCHAR: - if (rec->length <= 256) /* 255 + 1 byte length */ + if (rec->length <= 256) /* 255 + 1 uchar length */ return &uf_varchar1; return &uf_varchar2; case FIELD_LAST: @@ -850,7 +850,7 @@ static void (*get_unpack_function(MARIA_COLUMNDEF *rec)) static void uf_zerofill_skip_zero(MARIA_COLUMNDEF *rec, MARIA_BIT_BUFF *bit_buff, - byte *to, byte *end) + uchar *to, uchar *end) { if (get_bit(bit_buff)) bzero((char*) to,(uint) (end-to)); @@ -863,7 +863,7 @@ static void uf_zerofill_skip_zero(MARIA_COLUMNDEF *rec, } static void uf_skip_zero(MARIA_COLUMNDEF *rec, MARIA_BIT_BUFF *bit_buff, - byte *to, byte *end) + uchar *to, uchar *end) { if (get_bit(bit_buff)) bzero((char*) to,(uint) (end-to)); @@ -872,21 +872,21 @@ static void uf_skip_zero(MARIA_COLUMNDEF *rec, MARIA_BIT_BUFF *bit_buff, } static void uf_space_normal(MARIA_COLUMNDEF *rec, MARIA_BIT_BUFF *bit_buff, - byte *to, byte *end) + uchar *to, uchar *end) { if (get_bit(bit_buff)) - bfill((byte*) to,(end-to),' '); + bfill((uchar*) to,(end-to),' '); else decode_bytes(rec,bit_buff,to,end); } static void uf_space_endspace_selected(MARIA_COLUMNDEF *rec, MARIA_BIT_BUFF *bit_buff, - byte *to, byte *end) + uchar *to, uchar *end) { uint spaces; if (get_bit(bit_buff)) - bfill((byte*) to,(end-to),' '); + bfill((uchar*) to,(end-to),' '); else { if (get_bit(bit_buff)) @@ -898,7 +898,7 @@ static void uf_space_endspace_selected(MARIA_COLUMNDEF *rec, } if (to+spaces != end) decode_bytes(rec,bit_buff,to,end-spaces); - bfill((byte*) end-spaces,spaces,' '); + bfill((uchar*) end-spaces,spaces,' '); } else decode_bytes(rec,bit_buff,to,end); @@ -907,7 +907,7 @@ static void uf_space_endspace_selected(MARIA_COLUMNDEF *rec, static void uf_endspace_selected(MARIA_COLUMNDEF *rec, MARIA_BIT_BUFF *bit_buff, - byte *to, byte *end) + uchar *to, uchar *end) { uint spaces; if (get_bit(bit_buff)) @@ -919,18 +919,18 @@ static void uf_endspace_selected(MARIA_COLUMNDEF *rec, } if (to+spaces != end) decode_bytes(rec,bit_buff,to,end-spaces); - bfill((byte*) end-spaces,spaces,' '); + bfill((uchar*) end-spaces,spaces,' '); } else decode_bytes(rec,bit_buff,to,end); } static void uf_space_endspace(MARIA_COLUMNDEF *rec, MARIA_BIT_BUFF *bit_buff, - byte *to, byte *end) + uchar *to, uchar *end) { uint spaces; if (get_bit(bit_buff)) - bfill((byte*) to,(end-to),' '); + bfill((uchar*) to,(end-to),' '); else { if ((spaces=get_bits(bit_buff,rec->space_length_bits))+to > end) @@ -940,12 +940,12 @@ static void uf_space_endspace(MARIA_COLUMNDEF *rec, MARIA_BIT_BUFF *bit_buff, } if (to+spaces != end) decode_bytes(rec,bit_buff,to,end-spaces); - bfill((byte*) end-spaces,spaces,' '); + bfill((uchar*) end-spaces,spaces,' '); } } static void uf_endspace(MARIA_COLUMNDEF *rec, MARIA_BIT_BUFF *bit_buff, - byte *to, byte *end) + uchar *to, uchar *end) { uint spaces; if ((spaces=get_bits(bit_buff,rec->space_length_bits))+to > end) @@ -955,16 +955,16 @@ static void uf_endspace(MARIA_COLUMNDEF *rec, MARIA_BIT_BUFF *bit_buff, } if (to+spaces != end) decode_bytes(rec,bit_buff,to,end-spaces); - bfill((byte*) end-spaces,spaces,' '); + bfill((uchar*) end-spaces,spaces,' '); } static void uf_space_prespace_selected(MARIA_COLUMNDEF *rec, MARIA_BIT_BUFF *bit_buff, - byte *to, byte *end) + uchar *to, uchar *end) { uint spaces; if (get_bit(bit_buff)) - bfill((byte*) to,(end-to),' '); + bfill((uchar*) to,(end-to),' '); else { if (get_bit(bit_buff)) @@ -974,7 +974,7 @@ static void uf_space_prespace_selected(MARIA_COLUMNDEF *rec, bit_buff->error=1; return; } - bfill((byte*) to,spaces,' '); + bfill((uchar*) to,spaces,' '); if (to+spaces != end) decode_bytes(rec,bit_buff,to+spaces,end); } @@ -986,7 +986,7 @@ static void uf_space_prespace_selected(MARIA_COLUMNDEF *rec, static void uf_prespace_selected(MARIA_COLUMNDEF *rec, MARIA_BIT_BUFF *bit_buff, - byte *to, byte *end) + uchar *to, uchar *end) { uint spaces; if (get_bit(bit_buff)) @@ -996,7 +996,7 @@ static void uf_prespace_selected(MARIA_COLUMNDEF *rec, bit_buff->error=1; return; } - bfill((byte*) to,spaces,' '); + bfill((uchar*) to,spaces,' '); if (to+spaces != end) decode_bytes(rec,bit_buff,to+spaces,end); } @@ -1006,11 +1006,11 @@ static void uf_prespace_selected(MARIA_COLUMNDEF *rec, static void uf_space_prespace(MARIA_COLUMNDEF *rec, MARIA_BIT_BUFF *bit_buff, - byte *to, byte *end) + uchar *to, uchar *end) { uint spaces; if (get_bit(bit_buff)) - bfill((byte*) to,(end-to),' '); + bfill((uchar*) to,(end-to),' '); else { if ((spaces=get_bits(bit_buff,rec->space_length_bits))+to > end) @@ -1018,14 +1018,14 @@ static void uf_space_prespace(MARIA_COLUMNDEF *rec, MARIA_BIT_BUFF *bit_buff, bit_buff->error=1; return; } - bfill((byte*) to,spaces,' '); + bfill((uchar*) to,spaces,' '); if (to+spaces != end) decode_bytes(rec,bit_buff,to+spaces,end); } } static void uf_prespace(MARIA_COLUMNDEF *rec, MARIA_BIT_BUFF *bit_buff, - byte *to, byte *end) + uchar *to, uchar *end) { uint spaces; if ((spaces=get_bits(bit_buff,rec->space_length_bits))+to > end) @@ -1033,13 +1033,13 @@ static void uf_prespace(MARIA_COLUMNDEF *rec, MARIA_BIT_BUFF *bit_buff, bit_buff->error=1; return; } - bfill((byte*) to,spaces,' '); + bfill((uchar*) to,spaces,' '); if (to+spaces != end) decode_bytes(rec,bit_buff,to+spaces,end); } static void uf_zerofill_normal(MARIA_COLUMNDEF *rec, MARIA_BIT_BUFF *bit_buff, - byte *to, byte *end) + uchar *to, uchar *end) { end-=rec->space_length_bits; decode_bytes(rec,bit_buff, to, end); @@ -1048,14 +1048,14 @@ static void uf_zerofill_normal(MARIA_COLUMNDEF *rec, MARIA_BIT_BUFF *bit_buff, static void uf_constant(MARIA_COLUMNDEF *rec, MARIA_BIT_BUFF *bit_buff __attribute__((unused)), - byte *to, byte *end) + uchar *to, uchar *end) { memcpy(to,rec->huff_tree->intervalls,(size_t) (end-to)); } static void uf_intervall(MARIA_COLUMNDEF *rec, MARIA_BIT_BUFF *bit_buff, - byte *to, - byte *end) + uchar *to, + uchar *end) { reg1 uint field_length=(uint) (end-to); memcpy(to,rec->huff_tree->intervalls+field_length*decode_pos(bit_buff, @@ -1067,13 +1067,13 @@ static void uf_intervall(MARIA_COLUMNDEF *rec, MARIA_BIT_BUFF *bit_buff, /*ARGSUSED*/ static void uf_zero(MARIA_COLUMNDEF *rec __attribute__((unused)), MARIA_BIT_BUFF *bit_buff __attribute__((unused)), - byte *to, byte *end) + uchar *to, uchar *end) { bzero(to, (uint) (end-to)); } static void uf_blob(MARIA_COLUMNDEF *rec, MARIA_BIT_BUFF *bit_buff, - byte *to, byte *end) + uchar *to, uchar *end) { if (get_bit(bit_buff)) bzero(to, (uint) (end-to)); @@ -1084,12 +1084,12 @@ static void uf_blob(MARIA_COLUMNDEF *rec, MARIA_BIT_BUFF *bit_buff, if (bit_buff->blob_pos+length > bit_buff->blob_end) { bit_buff->error=1; - bzero((byte*) to,(end-to)); + bzero((uchar*) to,(end-to)); return; } - decode_bytes(rec,bit_buff,(byte*) bit_buff->blob_pos, - (byte*) bit_buff->blob_pos+length); - _ma_store_blob_length((byte*) to,pack_length,length); + decode_bytes(rec,bit_buff,(uchar*) bit_buff->blob_pos, + (uchar*) bit_buff->blob_pos+length); + _ma_store_blob_length((uchar*) to,pack_length,length); memcpy_fixed((char*) to+pack_length,(char*) &bit_buff->blob_pos, sizeof(char*)); bit_buff->blob_pos+=length; @@ -1098,7 +1098,7 @@ static void uf_blob(MARIA_COLUMNDEF *rec, MARIA_BIT_BUFF *bit_buff, static void uf_varchar1(MARIA_COLUMNDEF *rec, MARIA_BIT_BUFF *bit_buff, - byte *to, byte *end __attribute__((unused))) + uchar *to, uchar *end __attribute__((unused))) { if (get_bit(bit_buff)) to[0]= 0; /* Zero lengths */ @@ -1112,7 +1112,7 @@ static void uf_varchar1(MARIA_COLUMNDEF *rec, MARIA_BIT_BUFF *bit_buff, static void uf_varchar2(MARIA_COLUMNDEF *rec, MARIA_BIT_BUFF *bit_buff, - byte *to, byte *end __attribute__((unused))) + uchar *to, uchar *end __attribute__((unused))) { if (get_bit(bit_buff)) to[0]=to[1]=0; /* Zero lengths */ @@ -1129,7 +1129,7 @@ static void uf_varchar2(MARIA_COLUMNDEF *rec, MARIA_BIT_BUFF *bit_buff, #if BITS_SAVED == 64 static void decode_bytes(MARIA_COLUMNDEF *rec,MARIA_BIT_BUFF *bit_buff, - byte *to, byte *end) + uchar *to, uchar *end) { reg1 uint bits,low_byte; reg3 uint16 *pos; @@ -1166,13 +1166,13 @@ static void decode_bytes(MARIA_COLUMNDEF *rec,MARIA_BIT_BUFF *bit_buff, In most cases table_bits is 9. So there are 512 16-bit values. If the high-order bit (16) is set (IS_CHAR) then the array slot - for this value is a valid Huffman code for a resulting byte value. + for this value is a valid Huffman code for a resulting uchar value. - The low-order 8 bits (1..8) are the resulting byte value. + The low-order 8 bits (1..8) are the resulting uchar value. - Bits 9..14 are the length of the Huffman code for this byte value. + Bits 9..14 are the length of the Huffman code for this uchar value. This means so many bits from the input stream were needed to - represent this byte value. The remaining bits belong to later + represent this uchar value. The remaining bits belong to later Huffman codes. This also means that for every Huffman code shorter than table_bits there are multiple entires in the array, which differ just in the unused bits. @@ -1222,7 +1222,7 @@ static void decode_bytes(MARIA_COLUMNDEF *rec,MARIA_BIT_BUFF *bit_buff, #else static void decode_bytes(MARIA_COLUMNDEF *rec, MARIA_BIT_BUFF *bit_buff, - byte *to, byte *end) + uchar *to, uchar *end) { reg1 uint bits,low_byte; reg3 uint16 *pos; @@ -1334,7 +1334,7 @@ static uint decode_pos(MARIA_BIT_BUFF *bit_buff, int _ma_read_rnd_pack_record(MARIA_HA *info, - byte *buf, + uchar *buf, register MARIA_RECORD_POS filepos, my_bool skip_deleted_blocks) { @@ -1352,7 +1352,7 @@ int _ma_read_rnd_pack_record(MARIA_HA *info, file= info->dfile.file; if (info->opt_flag & READ_CACHE_USED) { - if (_ma_read_cache(&info->rec_cache, (byte*) block_info.header, + if (_ma_read_cache(&info->rec_cache, (uchar*) block_info.header, filepos, share->pack.ref_length, skip_deleted_blocks ? READING_NEXT : 0)) goto err; @@ -1372,14 +1372,14 @@ int _ma_read_rnd_pack_record(MARIA_HA *info, if (info->opt_flag & READ_CACHE_USED) { - if (_ma_read_cache(&info->rec_cache, (byte*) info->rec_buff, + if (_ma_read_cache(&info->rec_cache, (uchar*) info->rec_buff, block_info.filepos, block_info.rec_len, skip_deleted_blocks ? READING_NEXT : 0)) goto err; } else { - if (my_read(info->dfile.file, (byte*)info->rec_buff + block_info.offset, + if (my_read(info->dfile.file, (uchar*)info->rec_buff + block_info.offset, block_info.rec_len-block_info.offset, MYF(MY_NABP))) goto err; @@ -1400,7 +1400,7 @@ int _ma_read_rnd_pack_record(MARIA_HA *info, uint _ma_pack_get_block_info(MARIA_HA *maria, MARIA_BIT_BUFF *bit_buff, MARIA_BLOCK_INFO *info, - byte **rec_buff_p, my_size_t *rec_buff_size_p, + uchar **rec_buff_p, size_t *rec_buff_size_p, File file, my_off_t filepos) { uchar *header= info->header; @@ -1417,7 +1417,7 @@ uint _ma_pack_get_block_info(MARIA_HA *maria, MARIA_BIT_BUFF *bit_buff, VOID(my_seek(file,filepos,MY_SEEK_SET,MYF(0))); if (my_read(file,(char*) header,ref_length,MYF(MY_NABP))) return BLOCK_FATAL_ERROR; - DBUG_DUMP("header",(byte*) header,ref_length); + DBUG_DUMP("header",(uchar*) header,ref_length); } head_length= read_pack_length((uint) maria->s->pack.version, header, &info->rec_len); @@ -1449,7 +1449,7 @@ uint _ma_pack_get_block_info(MARIA_HA *maria, MARIA_BIT_BUFF *bit_buff, /* rutines for bit buffer */ - /* Note buffer must be 6 byte bigger than longest row */ + /* Note buffer must be 6 uchar bigger than longest row */ static void init_bit_buffer(MARIA_BIT_BUFF *bit_buff, uchar *buffer, uint length) @@ -1528,9 +1528,9 @@ static uint max_bit(register uint value) #ifdef HAVE_MMAP -static int _ma_read_mempack_record(MARIA_HA *info, byte *buf, +static int _ma_read_mempack_record(MARIA_HA *info, uchar *buf, MARIA_RECORD_POS filepos); -static int _ma_read_rnd_mempack_record(MARIA_HA*, byte *, MARIA_RECORD_POS, +static int _ma_read_rnd_mempack_record(MARIA_HA*, uchar *, MARIA_RECORD_POS, my_bool); my_bool _ma_memmap_file(MARIA_HA *info) @@ -1567,8 +1567,8 @@ static uchar * _ma_mempack_get_block_info(MARIA_HA *maria, MARIA_BIT_BUFF *bit_buff, MARIA_BLOCK_INFO *info, - byte **rec_buff_p, - my_size_t *rec_buff_size_p, + uchar **rec_buff_p, + size_t *rec_buff_size_p, uchar *header) { header+= read_pack_length((uint) maria->s->pack.version, header, @@ -1588,18 +1588,18 @@ _ma_mempack_get_block_info(MARIA_HA *maria, } -static int _ma_read_mempack_record(MARIA_HA *info, byte *buf, +static int _ma_read_mempack_record(MARIA_HA *info, uchar *buf, MARIA_RECORD_POS filepos) { MARIA_BLOCK_INFO block_info; MARIA_SHARE *share=info->s; - byte *pos; + uchar *pos; DBUG_ENTER("maria_read_mempack_record"); if (filepos == HA_OFFSET_ERROR) DBUG_RETURN(-1); /* _search() didn't find record */ - if (!(pos= (byte*) _ma_mempack_get_block_info(info, &info->bit_buff, + if (!(pos= (uchar*) _ma_mempack_get_block_info(info, &info->bit_buff, &block_info, &info->rec_buff, &info->rec_buff_size, (uchar*) share->file_map+ @@ -1612,14 +1612,14 @@ static int _ma_read_mempack_record(MARIA_HA *info, byte *buf, /*ARGSUSED*/ static int _ma_read_rnd_mempack_record(MARIA_HA *info, - byte *buf, + uchar *buf, register MARIA_RECORD_POS filepos, my_bool skip_deleted_blocks __attribute__((unused))) { MARIA_BLOCK_INFO block_info; MARIA_SHARE *share=info->s; - byte *pos,*start; + uchar *pos,*start; DBUG_ENTER("_ma_read_rnd_mempack_record"); if (filepos >= share->state.state.data_file_length) @@ -1627,7 +1627,7 @@ static int _ma_read_rnd_mempack_record(MARIA_HA *info, my_errno=HA_ERR_END_OF_FILE; goto err; } - if (!(pos= (byte*) _ma_mempack_get_block_info(info, &info->bit_buff, + if (!(pos= (uchar*) _ma_mempack_get_block_info(info, &info->bit_buff, &block_info, &info->rec_buff, &info->rec_buff_size, @@ -1657,7 +1657,7 @@ static int _ma_read_rnd_mempack_record(MARIA_HA *info, /* Save length of row */ -uint _ma_save_pack_length(uint version, byte *block_buff, ulong length) +uint _ma_save_pack_length(uint version, uchar *block_buff, ulong length) { if (length < 254) { diff --git a/storage/maria/ma_page.c b/storage/maria/ma_page.c index d6b8d5ecd7d..9e57898f4a1 100644 --- a/storage/maria/ma_page.c +++ b/storage/maria/ma_page.c @@ -19,12 +19,12 @@ /* Fetch a key-page in memory */ -byte *_ma_fetch_keypage(register MARIA_HA *info, MARIA_KEYDEF *keyinfo, +uchar *_ma_fetch_keypage(register MARIA_HA *info, MARIA_KEYDEF *keyinfo, my_off_t page, int level, - byte *buff, + uchar *buff, int return_buffer __attribute__ ((unused))) { - byte *tmp; + uchar *tmp; uint page_size; DBUG_ENTER("_ma_fetch_keypage"); DBUG_PRINT("enter",("page: %ld", (long) page)); @@ -66,7 +66,7 @@ byte *_ma_fetch_keypage(register MARIA_HA *info, MARIA_KEYDEF *keyinfo, /* Write a key-page on disk */ int _ma_write_keypage(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, - my_off_t page, int level, byte *buff) + my_off_t page, int level, uchar *buff) { DBUG_ENTER("_ma_write_keypage"); @@ -84,14 +84,14 @@ int _ma_write_keypage(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, DBUG_RETURN((-1)); } DBUG_PRINT("page",("write page at: %lu",(long) page)); - DBUG_DUMP("buff",(byte*) buff,maria_data_on_page(buff)); + DBUG_DUMP("buff",(uchar*) buff,maria_data_on_page(buff)); #endif #ifdef HAVE_purify { /* Clear unitialized part of page to avoid valgrind/purify warnings */ uint length= maria_data_on_page(buff); - bzero((byte*) buff+length,keyinfo->block_length-length); + bzero((uchar*) buff+length,keyinfo->block_length-length); length=keyinfo->block_length; } #endif @@ -150,7 +150,7 @@ int _ma_dispose(register MARIA_HA *info, MARIA_KEYDEF *keyinfo, my_off_t pos, my_off_t _ma_new(register MARIA_HA *info, MARIA_KEYDEF *keyinfo, int level) { my_off_t pos; - byte *buff; + uchar *buff; DBUG_ENTER("_ma_new"); if ((pos= info->s->state.key_del) == HA_OFFSET_ERROR) diff --git a/storage/maria/ma_pagecache.c b/storage/maria/ma_pagecache.c index ae42f702b0a..8af4532ff97 100755 --- a/storage/maria/ma_pagecache.c +++ b/storage/maria/ma_pagecache.c @@ -302,7 +302,7 @@ struct st_pagecache_block_link WQUEUE wqueue[COND_SIZE]; /* queues on waiting requests for new/old pages */ uint requests; /* number of requests for the block */ - byte *buffer; /* buffer for the block page */ + uchar *buffer; /* buffer for the block page */ uint status; /* state of the block */ uint pins; /* pin counter */ #ifndef DBUG_OFF @@ -575,7 +575,7 @@ extern my_bool translog_flush(LSN lsn); static uint pagecache_fwrite(PAGECACHE *pagecache, PAGECACHE_FILE *filedesc, - byte *buffer, + uchar *buffer, pgcache_page_no_t pageno, enum pagecache_page_type type, myf flags) @@ -651,7 +651,7 @@ static inline uint next_power(uint value) */ -int init_pagecache(PAGECACHE *pagecache, my_size_t use_mem, +int init_pagecache(PAGECACHE *pagecache, size_t use_mem, uint division_limit, uint age_threshold, uint block_size) { @@ -745,11 +745,11 @@ int init_pagecache(PAGECACHE *pagecache, my_size_t use_mem, (PAGECACHE_HASH_LINK*) ((char*) pagecache->hash_root + ALIGN_SIZE((sizeof(PAGECACHE_HASH_LINK*) * pagecache->hash_entries))); - bzero((byte*) pagecache->block_root, + bzero((uchar*) pagecache->block_root, pagecache->disk_blocks * sizeof(PAGECACHE_BLOCK_LINK)); - bzero((byte*) pagecache->hash_root, + bzero((uchar*) pagecache->hash_root, pagecache->hash_entries * sizeof(PAGECACHE_HASH_LINK*)); - bzero((byte*) pagecache->hash_link_root, + bzero((uchar*) pagecache->hash_link_root, pagecache->hash_links * sizeof(PAGECACHE_HASH_LINK)); pagecache->hash_links_used= 0; pagecache->free_hash_list= NULL; @@ -783,10 +783,10 @@ int init_pagecache(PAGECACHE *pagecache, my_size_t use_mem, pagecache->disk_blocks, (long) pagecache->block_root, pagecache->hash_entries, (long) pagecache->hash_root, pagecache->hash_links, (long) pagecache->hash_link_root)); - bzero((gptr) pagecache->changed_blocks, + bzero((uchar*) pagecache->changed_blocks, sizeof(pagecache->changed_blocks[0]) * PAGECACHE_CHANGED_BLOCKS_HASH); - bzero((gptr) pagecache->file_blocks, + bzero((uchar*) pagecache->file_blocks, sizeof(pagecache->file_blocks[0]) * PAGECACHE_CHANGED_BLOCKS_HASH); } @@ -800,12 +800,12 @@ err: pagecache->blocks= 0; if (pagecache->block_mem) { - my_large_free((gptr) pagecache->block_mem, MYF(0)); + my_large_free((uchar*) pagecache->block_mem, MYF(0)); pagecache->block_mem= NULL; } if (pagecache->block_root) { - my_free((gptr) pagecache->block_root, MYF(0)); + my_free((uchar*) pagecache->block_root, MYF(0)); pagecache->block_root= NULL; } my_errno= error; @@ -884,7 +884,7 @@ static int flush_all_key_blocks(PAGECACHE *pagecache) */ #if NOT_USED /* keep disabled until code is fixed see above !! */ int resize_pagecache(PAGECACHE *pagecache, - my_size_t use_mem, uint division_limit, + size_t use_mem, uint division_limit, uint age_threshold) { int blocks; @@ -1049,9 +1049,9 @@ void end_pagecache(PAGECACHE *pagecache, my_bool cleanup) { if (pagecache->block_mem) { - my_large_free((gptr) pagecache->block_mem, MYF(0)); + my_large_free((uchar*) pagecache->block_mem, MYF(0)); pagecache->block_mem= NULL; - my_free((gptr) pagecache->block_root, MYF(0)); + my_free((uchar*) pagecache->block_root, MYF(0)); pagecache->block_root= NULL; } pagecache->disk_blocks= -1; @@ -1876,7 +1876,7 @@ restart: block->buffer= ADD_TO_PTR(pagecache->block_mem, ((ulong) pagecache->blocks_used* pagecache->block_size), - byte*); + uchar*); pagecache->blocks_used++; } pagecache->blocks_unused--; @@ -2097,7 +2097,7 @@ static void remove_pin(PAGECACHE_BLOCK_LINK *block) PAGECACHE_PIN_INFO *info= info_find(block->pin_list, my_thread_var); DBUG_ASSERT(info != 0); info_unlink(info); - my_free((gptr) info, MYF(0)); + my_free((uchar*) info, MYF(0)); } #endif DBUG_VOID_RETURN; @@ -2119,7 +2119,7 @@ static void info_remove_lock(PAGECACHE_BLOCK_LINK *block) my_thread_var); DBUG_ASSERT(info != 0); info_unlink((PAGECACHE_PIN_INFO *)info); - my_free((gptr)info, MYF(0)); + my_free((uchar*)info, MYF(0)); } static void info_change_lock(PAGECACHE_BLOCK_LINK *block, my_bool wl) { @@ -2362,7 +2362,7 @@ static void read_block(PAGECACHE *pagecache, PAGECACHE_BLOCK_LINK *block, my_bool primary, pagecache_disk_read_validator validator, - gptr validator_data) + uchar* validator_data) { uint got_length; @@ -2814,16 +2814,16 @@ static enum pagecache_page_pin lock_to_pin[]= PAGECACHE_UNPIN /*PAGECACHE_LOCK_WRITE_TO_READ*/ }; -byte *pagecache_valid_read(PAGECACHE *pagecache, +uchar *pagecache_valid_read(PAGECACHE *pagecache, PAGECACHE_FILE *file, pgcache_page_no_t pageno, uint level, - byte *buff, + uchar *buff, enum pagecache_page_type type, enum pagecache_page_lock lock, PAGECACHE_PAGE_LINK *link, pagecache_disk_read_validator validator, - gptr validator_data) + uchar* validator_data) { int error= 0; enum pagecache_page_pin pin= lock_to_pin[lock]; @@ -2918,7 +2918,7 @@ restart: pagecache_pthread_mutex_unlock(&pagecache->cache_lock); if (status & PCBLOCK_ERROR) - DBUG_RETURN((byte *) 0); + DBUG_RETURN((uchar *) 0); DBUG_RETURN(buff); } @@ -2928,9 +2928,9 @@ no_key_cache: /* Key cache is not used */ /* We can't use mutex here as the key cache may not be initialized */ pagecache->global_cache_r_requests++; pagecache->global_cache_read++; - if (pagecache_fread(pagecache, file, (byte*) buff, pageno, MYF(MY_NABP))) + if (pagecache_fread(pagecache, file, (uchar*) buff, pageno, MYF(MY_NABP))) error= 1; - DBUG_RETURN(error ? (byte*) 0 : buff); + DBUG_RETURN(error ? (uchar*) 0 : buff); } @@ -3160,7 +3160,7 @@ my_bool pagecache_write_part(PAGECACHE *pagecache, PAGECACHE_FILE *file, pgcache_page_no_t pageno, uint level, - byte *buff, + uchar *buff, enum pagecache_page_type type, enum pagecache_page_lock lock, enum pagecache_page_pin pin, @@ -3320,7 +3320,7 @@ no_key_cache: { pagecache->global_cache_w_requests++; pagecache->global_cache_write++; - if (pagecache_fwrite(pagecache, file, (byte*) buff, pageno, type, + if (pagecache_fwrite(pagecache, file, (uchar*) buff, pageno, type, MYF(MY_NABP | MY_WAIT_IF_FULL))) error=1; } @@ -3426,7 +3426,7 @@ static int flush_cached_blocks(PAGECACHE *pagecache, As all blocks referred in 'cache' are marked by PCBLOCK_IN_FLUSH we are guarunteed no thread will change them */ - qsort((byte*) cache, count, sizeof(*cache), (qsort_cmp) cmp_sec_link); + qsort((uchar*) cache, count, sizeof(*cache), (qsort_cmp) cmp_sec_link); pagecache_pthread_mutex_lock(&pagecache->cache_lock); for (; cache != end; cache++) @@ -3730,7 +3730,7 @@ restart: test_key_cache(pagecache, "end of flush_pagecache_blocks", 0);); #endif if (cache != cache_buff) - my_free((gptr) cache, MYF(0)); + my_free((uchar*) cache, MYF(0)); if (last_errno) errno=last_errno; /* Return first error */ DBUG_RETURN(last_errno != 0); diff --git a/storage/maria/ma_pagecache.h b/storage/maria/ma_pagecache.h index 478f71161eb..5030c2a2d7b 100644 --- a/storage/maria/ma_pagecache.h +++ b/storage/maria/ma_pagecache.h @@ -93,7 +93,7 @@ typedef struct st_pagecache_hash_link PAGECACHE_HASH_LINK; #include -typedef my_bool (*pagecache_disk_read_validator)(byte *page, gptr data); +typedef my_bool (*pagecache_disk_read_validator)(uchar *page, uchar** data); #define PAGECACHE_CHANGED_BLOCKS_HASH 128 /* must be power of 2 */ @@ -108,7 +108,7 @@ typedef struct st_pagecache my_bool resize_in_flush; /* true during flush of resize operation */ my_bool can_be_used; /* usage of cache for read/write is allowed */ uint shift; /* block size = 2 ^ shift */ - my_size_t mem_size; /* specified size of the cache memory */ + size_t mem_size; /* specified size of the cache memory */ uint32 block_size; /* size of the page buffer of a cache block */ ulong min_warm_blocks; /* min number of warm blocks; */ ulong age_threshold; /* age threshold for hot blocks */ @@ -128,7 +128,7 @@ typedef struct st_pagecache PAGECACHE_HASH_LINK *free_hash_list;/* list of free hash links */ PAGECACHE_BLOCK_LINK *free_block_list;/* list of free blocks */ PAGECACHE_BLOCK_LINK *block_root;/* memory for block links */ - byte HUGE_PTR *block_mem; /* memory for block buffers */ + uchar HUGE_PTR *block_mem; /* memory for block buffers */ PAGECACHE_BLOCK_LINK *used_last;/* ptr to the last block of the LRU chain */ PAGECACHE_BLOCK_LINK *used_ins;/* ptr to the insertion block in LRU chain */ pthread_mutex_t cache_lock; /* to lock access to the cache structure */ @@ -164,11 +164,11 @@ typedef struct st_pagecache /* The default key cache */ extern PAGECACHE dflt_pagecache_var, *dflt_pagecache; -extern int init_pagecache(PAGECACHE *pagecache, my_size_t use_mem, +extern int init_pagecache(PAGECACHE *pagecache, size_t use_mem, uint division_limit, uint age_threshold, uint block_size); extern int resize_pagecache(PAGECACHE *pagecache, - my_size_t use_mem, uint division_limit, + size_t use_mem, uint division_limit, uint age_threshold); extern void change_pagecache_param(PAGECACHE *pagecache, uint division_limit, uint age_threshold); @@ -176,16 +176,16 @@ extern void change_pagecache_param(PAGECACHE *pagecache, uint division_limit, #define pagecache_read(P,F,N,L,B,T,K,I) \ pagecache_valid_read(P,F,N,L,B,T,K,I,0,0) -extern byte *pagecache_valid_read(PAGECACHE *pagecache, +extern uchar *pagecache_valid_read(PAGECACHE *pagecache, PAGECACHE_FILE *file, pgcache_page_no_t pageno, uint level, - byte *buff, + uchar *buff, enum pagecache_page_type type, enum pagecache_page_lock lock, PAGECACHE_PAGE_LINK *link, pagecache_disk_read_validator validator, - gptr validator_data); + uchar* validator_data); #define pagecache_write(P,F,N,L,B,T,O,I,M,K) \ pagecache_write_part(P,F,N,L,B,T,O,I,M,K,0,(P)->block_size) @@ -194,7 +194,7 @@ extern my_bool pagecache_write_part(PAGECACHE *pagecache, PAGECACHE_FILE *file, pgcache_page_no_t pageno, uint level, - byte *buff, + uchar *buff, enum pagecache_page_type type, enum pagecache_page_lock lock, enum pagecache_page_pin pin, @@ -247,9 +247,9 @@ extern int reset_pagecache_counters(const char *name, PAGECACHE *pagecache); /* Functions to handle multiple key caches */ extern my_bool multi_pagecache_init(void); extern void multi_pagecache_free(void); -extern PAGECACHE *multi_pagecache_search(byte *key, uint length, +extern PAGECACHE *multi_pagecache_search(uchar *key, uint length, PAGECACHE *def); -extern my_bool multi_pagecache_set(const byte *key, uint length, +extern my_bool multi_pagecache_set(const uchar *key, uint length, PAGECACHE *pagecache); extern void multi_pagecache_change(PAGECACHE *old_data, PAGECACHE *new_data); diff --git a/storage/maria/ma_pagecaches.c b/storage/maria/ma_pagecaches.c index d2ed4edca31..a9460be10c5 100644 --- a/storage/maria/ma_pagecaches.c +++ b/storage/maria/ma_pagecaches.c @@ -38,7 +38,7 @@ static SAFE_HASH pagecache_hash; my_bool multi_pagecache_init(void) { - return safe_hash_init(&pagecache_hash, 16, (byte*) maria_pagecache); + return safe_hash_init(&pagecache_hash, 16, (uchar*) maria_pagecache); } @@ -65,7 +65,7 @@ void multi_pagecache_free(void) key cache to use */ -PAGECACHE *multi_pagecache_search(byte *key, uint length, +PAGECACHE *multi_pagecache_search(uchar *key, uint length, PAGECACHE *def) { if (!pagecache_hash.hash.records) @@ -91,15 +91,15 @@ PAGECACHE *multi_pagecache_search(byte *key, uint length, */ -my_bool multi_pagecache_set(const byte *key, uint length, +my_bool multi_pagecache_set(const uchar *key, uint length, PAGECACHE *pagecache) { - return safe_hash_set(&pagecache_hash, key, length, (byte*) pagecache); + return safe_hash_set(&pagecache_hash, key, length, (uchar*) pagecache); } void multi_pagecache_change(PAGECACHE *old_data, PAGECACHE *new_data) { - safe_hash_change(&pagecache_hash, (byte*) old_data, (byte*) new_data); + safe_hash_change(&pagecache_hash, (uchar*) old_data, (uchar*) new_data); } diff --git a/storage/maria/ma_preload.c b/storage/maria/ma_preload.c index 44fc12f8571..35ae8868ee7 100644 --- a/storage/maria/ma_preload.c +++ b/storage/maria/ma_preload.c @@ -76,7 +76,7 @@ int maria_preload(MARIA_HA *info, ulonglong key_map, my_bool ignore_leaves) /* Read the next block of index file into the preload buffer */ if ((my_off_t) length > (key_file_length-pos)) length= (ulong) (key_file_length-pos); - if (my_pread(share->kfile.file, (byte*) buff, length, pos, + if (my_pread(share->kfile.file, (uchar*) buff, length, pos, MYF(MY_FAE|MY_FNABP))) goto err; @@ -91,7 +91,7 @@ int maria_preload(MARIA_HA *info, ulonglong key_map, my_bool ignore_leaves) if (pagecache_write(share->pagecache, &share->kfile, pos / block_length, DFLT_INIT_HITS, - (byte*) buff, + (uchar*) buff, PAGECACHE_PLAIN_PAGE, PAGECACHE_LOCK_LEFT_UNLOCKED, PAGECACHE_PIN_LEFT_UNPINNED, @@ -108,7 +108,7 @@ int maria_preload(MARIA_HA *info, ulonglong key_map, my_bool ignore_leaves) if (pagecache_write(share->pagecache, &share->kfile, pos / block_length, DFLT_INIT_HITS, - (byte*) buff, + (uchar*) buff, PAGECACHE_PLAIN_PAGE, PAGECACHE_LOCK_LEFT_UNLOCKED, PAGECACHE_PIN_LEFT_UNPINNED, diff --git a/storage/maria/ma_range.c b/storage/maria/ma_range.c index b359868e8e4..7250e6796d2 100644 --- a/storage/maria/ma_range.c +++ b/storage/maria/ma_range.c @@ -21,12 +21,12 @@ #include "maria_def.h" #include "ma_rt_index.h" -static ha_rows _ma_record_pos(MARIA_HA *info,const byte *key,uint key_len, +static ha_rows _ma_record_pos(MARIA_HA *info,const uchar *key,uint key_len, enum ha_rkey_function search_flag); -static double _ma_search_pos(MARIA_HA *info,MARIA_KEYDEF *keyinfo, byte *key, +static double _ma_search_pos(MARIA_HA *info,MARIA_KEYDEF *keyinfo, uchar *key, uint key_len,uint nextflag, my_off_t pos); -static uint _ma_keynr(MARIA_HA *info, MARIA_KEYDEF *keyinfo, byte *page, - byte *keypos, uint *ret_max_key); +static uint _ma_keynr(MARIA_HA *info, MARIA_KEYDEF *keyinfo, uchar *page, + uchar *keypos, uint *ret_max_key); /** @@ -64,7 +64,7 @@ ha_rows maria_records_in_range(MARIA_HA *info, int inx, key_range *min_key, #ifdef HAVE_RTREE_KEYS case HA_KEY_ALG_RTREE: { - byte *key_buff; + uchar *key_buff; uint start_key_len; /* @@ -126,12 +126,12 @@ ha_rows maria_records_in_range(MARIA_HA *info, int inx, key_range *min_key, /* Find relative position (in records) for key in index-tree */ -static ha_rows _ma_record_pos(MARIA_HA *info, const byte *key, uint key_len, +static ha_rows _ma_record_pos(MARIA_HA *info, const uchar *key, uint key_len, enum ha_rkey_function search_flag) { uint inx=(uint) info->lastinx, nextflag; MARIA_KEYDEF *keyinfo=info->s->keyinfo+inx; - byte *key_buff; + uchar *key_buff; double pos; DBUG_ENTER("_ma_record_pos"); DBUG_PRINT("enter",("search_flag: %d",search_flag)); @@ -164,13 +164,13 @@ static ha_rows _ma_record_pos(MARIA_HA *info, const byte *key, uint key_len, static double _ma_search_pos(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, - byte *key, uint key_len, uint nextflag, + uchar *key, uint key_len, uint nextflag, register my_off_t pos) { int flag; uint nod_flag,keynr,max_keynr; my_bool after_key; - byte *keypos, *buff; + uchar *keypos, *buff; double offset; DBUG_ENTER("_ma_search_pos"); LINT_INIT(max_keynr); @@ -232,10 +232,10 @@ err: /* Get keynummer of current key and max number of keys in nod */ static uint _ma_keynr(MARIA_HA *info, register MARIA_KEYDEF *keyinfo, - byte *page, byte *keypos, uint *ret_max_key) + uchar *page, uchar *keypos, uint *ret_max_key) { uint nod_flag,keynr,max_key; - byte t_buff[HA_MAX_KEY_BUFF],*end; + uchar t_buff[HA_MAX_KEY_BUFF],*end; end= page+maria_data_on_page(page); nod_flag=_ma_test_if_nod(page); diff --git a/storage/maria/ma_rfirst.c b/storage/maria/ma_rfirst.c index 04c496d9c56..226aaa551f0 100644 --- a/storage/maria/ma_rfirst.c +++ b/storage/maria/ma_rfirst.c @@ -17,7 +17,7 @@ /* Read first row through a specfic key */ -int maria_rfirst(MARIA_HA *info, byte *buf, int inx) +int maria_rfirst(MARIA_HA *info, uchar *buf, int inx) { DBUG_ENTER("maria_rfirst"); info->cur_row.lastpos= HA_OFFSET_ERROR; diff --git a/storage/maria/ma_rkey.c b/storage/maria/ma_rkey.c index 6158935472b..ef8b8468f1f 100644 --- a/storage/maria/ma_rkey.c +++ b/storage/maria/ma_rkey.c @@ -21,10 +21,10 @@ /* Read a record using key */ /* Ordinary search_flag is 0 ; Give error if no record with key */ -int maria_rkey(MARIA_HA *info, byte *buf, int inx, const byte *key, +int maria_rkey(MARIA_HA *info, uchar *buf, int inx, const uchar *key, uint key_len, enum ha_rkey_function search_flag) { - byte *key_buff; + uchar *key_buff; MARIA_SHARE *share=info->s; MARIA_KEYDEF *keyinfo; HA_KEYSEG *last_used_keyseg; diff --git a/storage/maria/ma_rlast.c b/storage/maria/ma_rlast.c index ebd039843c8..a9a470d37d9 100644 --- a/storage/maria/ma_rlast.c +++ b/storage/maria/ma_rlast.c @@ -17,7 +17,7 @@ /* Read last row with the same key as the previous read. */ -int maria_rlast(MARIA_HA *info, byte *buf, int inx) +int maria_rlast(MARIA_HA *info, uchar *buf, int inx) { DBUG_ENTER("maria_rlast"); info->cur_row.lastpos= HA_OFFSET_ERROR; diff --git a/storage/maria/ma_rnext.c b/storage/maria/ma_rnext.c index ccca05ff3ad..fcc0f1f6a90 100644 --- a/storage/maria/ma_rnext.c +++ b/storage/maria/ma_rnext.c @@ -24,7 +24,7 @@ based on the position of the last used key! */ -int maria_rnext(MARIA_HA *info, byte *buf, int inx) +int maria_rnext(MARIA_HA *info, uchar *buf, int inx) { int error,changed; uint flag; diff --git a/storage/maria/ma_rnext_same.c b/storage/maria/ma_rnext_same.c index 207a438e10b..6782cf5b8cf 100644 --- a/storage/maria/ma_rnext_same.c +++ b/storage/maria/ma_rnext_same.c @@ -25,7 +25,7 @@ based on the position of the last used key! */ -int maria_rnext_same(MARIA_HA *info, byte *buf) +int maria_rnext_same(MARIA_HA *info, uchar *buf) { int error; uint inx,not_used[2]; diff --git a/storage/maria/ma_rprev.c b/storage/maria/ma_rprev.c index 5e7cfc9f41a..753ff604975 100644 --- a/storage/maria/ma_rprev.c +++ b/storage/maria/ma_rprev.c @@ -22,7 +22,7 @@ based on the position of the last used key! */ -int maria_rprev(MARIA_HA *info, byte *buf, int inx) +int maria_rprev(MARIA_HA *info, uchar *buf, int inx) { int error,changed; register uint flag; diff --git a/storage/maria/ma_rrnd.c b/storage/maria/ma_rrnd.c index 8e2b12dc60d..4f5c2fb06cf 100644 --- a/storage/maria/ma_rrnd.c +++ b/storage/maria/ma_rrnd.c @@ -28,7 +28,7 @@ HA_ERR_END_OF_FILE EOF. */ -int maria_rrnd(MARIA_HA *info, byte *buf, MARIA_RECORD_POS filepos) +int maria_rrnd(MARIA_HA *info, uchar *buf, MARIA_RECORD_POS filepos) { DBUG_ENTER("maria_rrnd"); diff --git a/storage/maria/ma_rsame.c b/storage/maria/ma_rsame.c index 052fe79af58..9c9acac013a 100644 --- a/storage/maria/ma_rsame.c +++ b/storage/maria/ma_rsame.c @@ -28,7 +28,7 @@ */ -int maria_rsame(MARIA_HA *info, byte *record, int inx) +int maria_rsame(MARIA_HA *info, uchar *record, int inx) { DBUG_ENTER("maria_rsame"); diff --git a/storage/maria/ma_rsamepos.c b/storage/maria/ma_rsamepos.c index 1e09bdb8db4..186bc80c06d 100644 --- a/storage/maria/ma_rsamepos.c +++ b/storage/maria/ma_rsamepos.c @@ -27,7 +27,7 @@ ** HA_ERR_END_OF_FILE = End of file */ -int maria_rsame_with_pos(MARIA_HA *info, byte *record, int inx, +int maria_rsame_with_pos(MARIA_HA *info, uchar *record, int inx, MARIA_RECORD_POS filepos) { DBUG_ENTER("maria_rsame_with_pos"); diff --git a/storage/maria/ma_rt_index.c b/storage/maria/ma_rt_index.c index 27a83e433a4..4d99eade9b5 100644 --- a/storage/maria/ma_rt_index.c +++ b/storage/maria/ma_rt_index.c @@ -58,11 +58,11 @@ static int maria_rtree_find_req(MARIA_HA *info, MARIA_KEYDEF *keyinfo, { uint nod_flag; int res; - byte *page_buf, *k, *last; + uchar *page_buf, *k, *last; int k_len; uint *saved_key = (uint*) (info->maria_rtree_recursion_state) + level; - if (!(page_buf = (byte*) my_alloca((uint)keyinfo->block_length))) + if (!(page_buf = (uchar*) my_alloca((uint)keyinfo->block_length))) { my_errno = HA_ERR_OUT_OF_MEM; return -1; @@ -115,7 +115,7 @@ static int maria_rtree_find_req(MARIA_HA *info, MARIA_KEYDEF *keyinfo, if (!maria_rtree_key_cmp(keyinfo->seg, info->first_mbr_key, k, info->last_rkey_length, search_flag)) { - byte *after_key = (byte*) rt_PAGE_NEXT_KEY(k, k_len, nod_flag); + uchar *after_key = (uchar*) rt_PAGE_NEXT_KEY(k, k_len, nod_flag); info->cur_row.lastpos = _ma_dpos(info, 0, after_key); info->lastkey_length = k_len + info->s->base.rec_reflength; memcpy(info->lastkey, k, info->lastkey_length); @@ -144,11 +144,11 @@ static int maria_rtree_find_req(MARIA_HA *info, MARIA_KEYDEF *keyinfo, res = 1; ok: - my_afree((byte*)page_buf); + my_afree((uchar*)page_buf); return res; err1: - my_afree((byte*)page_buf); + my_afree((uchar*)page_buf); info->cur_row.lastpos = HA_OFFSET_ERROR; return -1; } @@ -171,7 +171,7 @@ err1: 1 Not found */ -int maria_rtree_find_first(MARIA_HA *info, uint keynr, byte *key, +int maria_rtree_find_first(MARIA_HA *info, uint keynr, uchar *key, uint key_length, uint search_flag) { my_off_t root; @@ -229,7 +229,7 @@ int maria_rtree_find_next(MARIA_HA *info, uint keynr, uint search_flag) if (!info->keyread_buff_used) { - byte *key= info->int_keypos; + uchar *key= info->int_keypos; while (key < info->int_maxpos) { @@ -237,7 +237,7 @@ int maria_rtree_find_next(MARIA_HA *info, uint keynr, uint search_flag) info->first_mbr_key, key, info->last_rkey_length, search_flag)) { - byte *after_key= key + keyinfo->keylength; + uchar *after_key= key + keyinfo->keylength; info->cur_row.lastpos= _ma_dpos(info, 0, after_key); memcpy(info->lastkey, key, info->lastkey_length); @@ -278,12 +278,12 @@ int maria_rtree_find_next(MARIA_HA *info, uint keynr, uint search_flag) static int maria_rtree_get_req(MARIA_HA *info, MARIA_KEYDEF *keyinfo, uint key_length, my_off_t page, int level) { - byte *page_buf, *last, *k; + uchar *page_buf, *last, *k; uint nod_flag, k_len; int res; uint *saved_key= (uint*) (info->maria_rtree_recursion_state) + level; - if (!(page_buf= (byte*) my_alloca((uint)keyinfo->block_length))) + if (!(page_buf= (uchar*) my_alloca((uint)keyinfo->block_length))) return -1; if (!_ma_fetch_keypage(info, keyinfo, page, DFLT_INIT_HITS, page_buf, 0)) goto err1; @@ -329,7 +329,7 @@ static int maria_rtree_get_req(MARIA_HA *info, MARIA_KEYDEF *keyinfo, uint key_l else { /* this is a leaf */ - byte *after_key = rt_PAGE_NEXT_KEY(k, k_len, nod_flag); + uchar *after_key = rt_PAGE_NEXT_KEY(k, k_len, nod_flag); info->cur_row.lastpos = _ma_dpos(info, 0, after_key); info->lastkey_length = k_len + info->s->base.rec_reflength; memcpy(info->lastkey, k, info->lastkey_length); @@ -339,7 +339,7 @@ static int maria_rtree_get_req(MARIA_HA *info, MARIA_KEYDEF *keyinfo, uint key_l if (after_key < last) { - info->int_keypos = (byte*) saved_key; + info->int_keypos = (uchar*) saved_key; memcpy(info->buff, page_buf, keyinfo->block_length); info->int_maxpos = rt_PAGE_END(info->buff); info->keyread_buff_used = 0; @@ -358,11 +358,11 @@ static int maria_rtree_get_req(MARIA_HA *info, MARIA_KEYDEF *keyinfo, uint key_l res = 1; ok: - my_afree((byte*)page_buf); + my_afree((uchar*)page_buf); return res; err1: - my_afree((byte*)page_buf); + my_afree((uchar*)page_buf); info->cur_row.lastpos = HA_OFFSET_ERROR; return -1; } @@ -413,10 +413,10 @@ int maria_rtree_get_next(MARIA_HA *info, uint keynr, uint key_length) { uint k_len = keyinfo->keylength - info->s->base.rec_reflength; /* rt_PAGE_NEXT_KEY(info->int_keypos) */ - byte *key = info->buff + *(int*)info->int_keypos + k_len + + uchar *key = info->buff + *(int*)info->int_keypos + k_len + info->s->base.rec_reflength; /* rt_PAGE_NEXT_KEY(key) */ - byte *after_key = key + k_len + info->s->base.rec_reflength; + uchar *after_key = key + k_len + info->s->base.rec_reflength; info->cur_row.lastpos = _ma_dpos(info, 0, after_key); info->lastkey_length = k_len + info->s->base.rec_reflength; @@ -450,7 +450,7 @@ int maria_rtree_get_next(MARIA_HA *info, uint keynr, uint key_length) #ifdef PICK_BY_PERIMETER static uchar *maria_rtree_pick_key(MARIA_HA *info, MARIA_KEYDEF *keyinfo, uchar *key, - uint key_length, byte *page_buf, + uint key_length, uchar *page_buf, uint nod_flag) { double increase; @@ -483,18 +483,18 @@ static uchar *maria_rtree_pick_key(MARIA_HA *info, MARIA_KEYDEF *keyinfo, #endif /*PICK_BY_PERIMETER*/ #ifdef PICK_BY_AREA -static byte *maria_rtree_pick_key(MARIA_HA *info, MARIA_KEYDEF *keyinfo, - byte *key, - uint key_length, byte *page_buf, +static uchar *maria_rtree_pick_key(MARIA_HA *info, MARIA_KEYDEF *keyinfo, + uchar *key, + uint key_length, uchar *page_buf, uint nod_flag) { double increase; double best_incr = DBL_MAX; double area; double best_area; - byte *best_key; - byte *k = rt_PAGE_FIRST_KEY(page_buf, nod_flag); - byte *last = rt_PAGE_END(page_buf); + uchar *best_key; + uchar *k = rt_PAGE_FIRST_KEY(page_buf, nod_flag); + uchar *last = rt_PAGE_END(page_buf); LINT_INIT(best_area); LINT_INIT(best_key); @@ -538,16 +538,16 @@ static byte *maria_rtree_pick_key(MARIA_HA *info, MARIA_KEYDEF *keyinfo, */ static int maria_rtree_insert_req(MARIA_HA *info, MARIA_KEYDEF *keyinfo, - byte *key, + uchar *key, uint key_length, my_off_t page, my_off_t *new_page, int ins_level, int level) { uint nod_flag; int res; - byte *page_buf, *k; + uchar *page_buf, *k; - if (!(page_buf= (byte*) my_alloca((uint)keyinfo->block_length + + if (!(page_buf= (uchar*) my_alloca((uint)keyinfo->block_length + HA_MAX_KEY_BUFF))) { my_errno = HA_ERR_OUT_OF_MEM; @@ -576,7 +576,7 @@ static int maria_rtree_insert_req(MARIA_HA *info, MARIA_KEYDEF *keyinfo, } case 1: /* child was split */ { - byte *new_key = page_buf + keyinfo->block_length + nod_flag; + uchar *new_key = page_buf + keyinfo->block_length + nod_flag; /* set proper MBR for key */ if (maria_rtree_set_key_mbr(info, keyinfo, k, key_length, _ma_kpos(nod_flag, k))) @@ -626,7 +626,7 @@ err1: 1 Root was split */ -static int maria_rtree_insert_level(MARIA_HA *info, uint keynr, byte *key, +static int maria_rtree_insert_level(MARIA_HA *info, uint keynr, uchar *key, uint key_length, int ins_level) { my_off_t old_root; @@ -656,11 +656,11 @@ static int maria_rtree_insert_level(MARIA_HA *info, uint keynr, byte *key, } case 1: /* root was split, grow a new root */ { - byte *new_root_buf, *new_key; + uchar *new_root_buf, *new_key; my_off_t new_root; uint nod_flag = info->s->base.key_reflength; - if (!(new_root_buf= (byte*) my_alloca((uint)keyinfo->block_length + + if (!(new_root_buf= (uchar*) my_alloca((uint)keyinfo->block_length + HA_MAX_KEY_BUFF))) { my_errno = HA_ERR_OUT_OF_MEM; @@ -695,10 +695,10 @@ static int maria_rtree_insert_level(MARIA_HA *info, uint keynr, byte *key, goto err1; info->s->state.key_root[keynr] = new_root; - my_afree((byte*)new_root_buf); + my_afree((uchar*)new_root_buf); break; err1: - my_afree((byte*)new_root_buf); + my_afree((uchar*)new_root_buf); return -1; } default: @@ -719,7 +719,7 @@ err1: 0 OK */ -int maria_rtree_insert(MARIA_HA *info, uint keynr, byte *key, uint key_length) +int maria_rtree_insert(MARIA_HA *info, uint keynr, uchar *key, uint key_length) { return (!key_length || (maria_rtree_insert_level(info, keynr, key, key_length, -1) == -1)) ? @@ -741,7 +741,7 @@ static int maria_rtree_fill_reinsert_list(stPageList *ReinsertList, my_off_t pag if (ReinsertList->n_pages == ReinsertList->m_pages) { ReinsertList->m_pages += REINSERT_BUFFER_INC; - if (!(ReinsertList->pages = (stPageLevel*)my_realloc((gptr)ReinsertList->pages, + if (!(ReinsertList->pages = (stPageLevel*)my_realloc((uchar*)ReinsertList->pages, ReinsertList->m_pages * sizeof(stPageLevel), MYF(MY_ALLOW_ZERO_PTR)))) goto err1; } @@ -767,7 +767,7 @@ err1: */ static int maria_rtree_delete_req(MARIA_HA *info, MARIA_KEYDEF *keyinfo, - byte *key, + uchar *key, uint key_length, my_off_t page, uint *page_size, stPageList *ReinsertList, int level) @@ -775,9 +775,9 @@ static int maria_rtree_delete_req(MARIA_HA *info, MARIA_KEYDEF *keyinfo, ulong i; uint nod_flag; int res; - byte *page_buf, *last, *k; + uchar *page_buf, *last, *k; - if (!(page_buf = (byte*) my_alloca((uint)keyinfo->block_length))) + if (!(page_buf = (uchar*) my_alloca((uint)keyinfo->block_length))) { my_errno = HA_ERR_OUT_OF_MEM; return -1; @@ -878,11 +878,11 @@ static int maria_rtree_delete_req(MARIA_HA *info, MARIA_KEYDEF *keyinfo, res = 1; ok: - my_afree((byte*)page_buf); + my_afree((uchar*)page_buf); return res; err1: - my_afree((byte*)page_buf); + my_afree((uchar*)page_buf); return -1; } @@ -895,7 +895,7 @@ err1: 0 Deleted */ -int maria_rtree_delete(MARIA_HA *info, uint keynr, byte *key, uint key_length) +int maria_rtree_delete(MARIA_HA *info, uint keynr, uchar *key, uint key_length) { uint page_size; stPageList ReinsertList; @@ -926,9 +926,9 @@ int maria_rtree_delete(MARIA_HA *info, uint keynr, byte *key, uint key_length) ulong i; for (i = 0; i < ReinsertList.n_pages; ++i) { - byte *page_buf, *k, *last; + uchar *page_buf, *k, *last; - if (!(page_buf = (byte*) my_alloca((uint)keyinfo->block_length))) + if (!(page_buf = (uchar*) my_alloca((uint)keyinfo->block_length))) { my_errno = HA_ERR_OUT_OF_MEM; goto err1; @@ -954,7 +954,7 @@ int maria_rtree_delete(MARIA_HA *info, uint keynr, byte *key, uint key_length) goto err1; } if (ReinsertList.pages) - my_free((byte*) ReinsertList.pages, MYF(0)); + my_free((uchar*) ReinsertList.pages, MYF(0)); /* check for redundant root (not leaf, 1 child) and eliminate */ if ((old_root = info->s->state.key_root[keynr]) == HA_OFFSET_ERROR) @@ -997,14 +997,14 @@ err1: estimated value */ -ha_rows maria_rtree_estimate(MARIA_HA *info, uint keynr, byte *key, +ha_rows maria_rtree_estimate(MARIA_HA *info, uint keynr, uchar *key, uint key_length, uint flag) { MARIA_KEYDEF *keyinfo = info->s->keyinfo + keynr; my_off_t root; uint i = 0; uint nod_flag, k_len; - byte *page_buf, *k, *last; + uchar *page_buf, *k, *last; double area = 0; ha_rows res = 0; @@ -1013,7 +1013,7 @@ ha_rows maria_rtree_estimate(MARIA_HA *info, uint keynr, byte *key, if ((root = info->s->state.key_root[keynr]) == HA_OFFSET_ERROR) return HA_POS_ERROR; - if (!(page_buf= (byte*) my_alloca((uint)keyinfo->block_length))) + if (!(page_buf= (uchar*) my_alloca((uint)keyinfo->block_length))) return HA_POS_ERROR; if (!_ma_fetch_keypage(info, keyinfo, root, DFLT_INIT_HITS, page_buf, 0)) goto err1; @@ -1078,7 +1078,7 @@ ha_rows maria_rtree_estimate(MARIA_HA *info, uint keynr, byte *key, res = HA_POS_ERROR; } - my_afree((byte*)page_buf); + my_afree((uchar*)page_buf); return res; err1: diff --git a/storage/maria/ma_rt_index.h b/storage/maria/ma_rt_index.h index eae43966aa0..fe2f62b662c 100644 --- a/storage/maria/ma_rt_index.h +++ b/storage/maria/ma_rt_index.h @@ -26,21 +26,23 @@ #define rt_PAGE_MIN_SIZE(block_length) ((uint)(block_length) / 3) -int maria_rtree_insert(MARIA_HA *info, uint keynr, byte *key, uint key_length); -int maria_rtree_delete(MARIA_HA *info, uint keynr, byte *key, uint key_length); +int maria_rtree_insert(MARIA_HA *info, uint keynr, uchar *key, + uint key_length); +int maria_rtree_delete(MARIA_HA *info, uint keynr, uchar *key, + uint key_length); -int maria_rtree_find_first(MARIA_HA *info, uint keynr, byte *key, +int maria_rtree_find_first(MARIA_HA *info, uint keynr, uchar *key, uint key_length, uint search_flag); int maria_rtree_find_next(MARIA_HA *info, uint keynr, uint search_flag); int maria_rtree_get_first(MARIA_HA *info, uint keynr, uint key_length); int maria_rtree_get_next(MARIA_HA *info, uint keynr, uint key_length); -ha_rows maria_rtree_estimate(MARIA_HA *info, uint keynr, byte *key, +ha_rows maria_rtree_estimate(MARIA_HA *info, uint keynr, uchar *key, uint key_length, uint flag); -int maria_rtree_split_page(MARIA_HA *info, MARIA_KEYDEF *keyinfo, byte *page, - byte *key, uint key_length, +int maria_rtree_split_page(MARIA_HA *info, MARIA_KEYDEF *keyinfo, uchar *page, + uchar *key, uint key_length, my_off_t *new_page_offs); #endif /*HAVE_RTREE_KEYS*/ diff --git a/storage/maria/ma_rt_key.c b/storage/maria/ma_rt_key.c index a27ff23c006..1b9f246081d 100644 --- a/storage/maria/ma_rt_key.c +++ b/storage/maria/ma_rt_key.c @@ -29,8 +29,8 @@ 1 Split */ -int maria_rtree_add_key(MARIA_HA *info, MARIA_KEYDEF *keyinfo, byte *key, - uint key_length, byte *page_buf, my_off_t *new_page) +int maria_rtree_add_key(MARIA_HA *info, MARIA_KEYDEF *keyinfo, uchar *key, + uint key_length, uchar *page_buf, my_off_t *new_page) { uint page_size = maria_data_on_page(page_buf); uint nod_flag = _ma_test_if_nod(page_buf); @@ -65,11 +65,11 @@ int maria_rtree_add_key(MARIA_HA *info, MARIA_KEYDEF *keyinfo, byte *key, Delete key from the page */ -int maria_rtree_delete_key(MARIA_HA *info, byte *page_buf, byte *key, +int maria_rtree_delete_key(MARIA_HA *info, uchar *page_buf, uchar *key, uint key_length, uint nod_flag) { uint16 page_size = maria_data_on_page(page_buf); - byte *key_start; + uchar *key_start; key_start= key - nod_flag; if (!nod_flag) @@ -88,7 +88,7 @@ int maria_rtree_delete_key(MARIA_HA *info, byte *page_buf, byte *key, Calculate and store key MBR */ -int maria_rtree_set_key_mbr(MARIA_HA *info, MARIA_KEYDEF *keyinfo, byte *key, +int maria_rtree_set_key_mbr(MARIA_HA *info, MARIA_KEYDEF *keyinfo, uchar *key, uint key_length, my_off_t child_page) { if (!_ma_fetch_keypage(info, keyinfo, child_page, diff --git a/storage/maria/ma_rt_key.h b/storage/maria/ma_rt_key.h index 03c9ef46438..3f95d3d3e67 100644 --- a/storage/maria/ma_rt_key.h +++ b/storage/maria/ma_rt_key.h @@ -21,11 +21,11 @@ #ifdef HAVE_RTREE_KEYS -int maria_rtree_add_key(MARIA_HA *info, MARIA_KEYDEF *keyinfo, byte *key, - uint key_length, byte *page_buf, my_off_t *new_page); -int maria_rtree_delete_key(MARIA_HA *info, byte *page, byte *key, +int maria_rtree_add_key(MARIA_HA *info, MARIA_KEYDEF *keyinfo, uchar *key, + uint key_length, uchar *page_buf, my_off_t *new_page); +int maria_rtree_delete_key(MARIA_HA *info, uchar *page, uchar *key, uint key_length, uint nod_flag); -int maria_rtree_set_key_mbr(MARIA_HA *info, MARIA_KEYDEF *keyinfo, byte *key, +int maria_rtree_set_key_mbr(MARIA_HA *info, MARIA_KEYDEF *keyinfo, uchar *key, uint key_length, my_off_t child_page); #endif /*HAVE_RTREE_KEYS*/ diff --git a/storage/maria/ma_rt_mbr.c b/storage/maria/ma_rt_mbr.c index 2da4ffea7a4..a224cefac12 100644 --- a/storage/maria/ma_rt_mbr.c +++ b/storage/maria/ma_rt_mbr.c @@ -93,7 +93,7 @@ Returns 0 on success. */ -int maria_rtree_key_cmp(HA_KEYSEG *keyseg, byte *b, byte *a, uint key_length, +int maria_rtree_key_cmp(HA_KEYSEG *keyseg, uchar *b, uchar *a, uint key_length, uint nextflag) { for (; (int) key_length > 0; keyseg += 2 ) @@ -153,7 +153,7 @@ int maria_rtree_key_cmp(HA_KEYSEG *keyseg, byte *b, byte *a, uint key_length, end: if (nextflag & MBR_DATA) { - byte *end = a + keyseg->length; + uchar *end = a + keyseg->length; do { if (*a++ != *b++) @@ -182,7 +182,7 @@ end: /* Calculates rectangle volume */ -double maria_rtree_rect_volume(HA_KEYSEG *keyseg, byte *a, uint key_length) +double maria_rtree_rect_volume(HA_KEYSEG *keyseg, uchar *a, uint key_length) { double res = 1; for (; (int)key_length > 0; keyseg += 2) @@ -263,7 +263,7 @@ double maria_rtree_rect_volume(HA_KEYSEG *keyseg, byte *a, uint key_length) Creates an MBR as an array of doubles. */ -int maria_rtree_d_mbr(HA_KEYSEG *keyseg, byte *a, uint key_length, double *res) +int maria_rtree_d_mbr(HA_KEYSEG *keyseg, uchar *a, uint key_length, double *res) { for (; (int)key_length > 0; keyseg += 2) { @@ -352,7 +352,7 @@ int maria_rtree_d_mbr(HA_KEYSEG *keyseg, byte *a, uint key_length, double *res) Result is written to c */ -int maria_rtree_combine_rect(HA_KEYSEG *keyseg, byte* a, byte* b, byte* c, +int maria_rtree_combine_rect(HA_KEYSEG *keyseg, uchar* a, uchar* b, uchar* c, uint key_length) { for ( ; (int) key_length > 0 ; keyseg += 2) @@ -443,7 +443,7 @@ int maria_rtree_combine_rect(HA_KEYSEG *keyseg, byte* a, byte* b, byte* c, /* Calculates overlapping area of two MBRs a & b */ -double maria_rtree_overlapping_area(HA_KEYSEG *keyseg, byte* a, byte* b, +double maria_rtree_overlapping_area(HA_KEYSEG *keyseg, uchar* a, uchar* b, uint key_length) { double res = 1; @@ -528,7 +528,7 @@ double maria_rtree_overlapping_area(HA_KEYSEG *keyseg, byte* a, byte* b, Calculates MBR_AREA(a+b) - MBR_AREA(a) */ -double maria_rtree_area_increase(HA_KEYSEG *keyseg, byte *a, byte *b, +double maria_rtree_area_increase(HA_KEYSEG *keyseg, uchar *a, uchar *b, uint key_length, double *ab_area) { double a_area= 1.0; @@ -621,7 +621,7 @@ safe_end: /* Calculates MBR_PERIMETER(a+b) - MBR_PERIMETER(a) */ -double maria_rtree_perimeter_increase(HA_KEYSEG *keyseg, byte* a, byte* b, +double maria_rtree_perimeter_increase(HA_KEYSEG *keyseg, uchar* a, uchar* b, uint key_length, double *ab_perim) { double a_perim = 0.0; @@ -734,14 +734,14 @@ double maria_rtree_perimeter_increase(HA_KEYSEG *keyseg, byte* a, byte* b, /* Calculates key page total MBR = MBR(key1) + MBR(key2) + ... */ -int maria_rtree_page_mbr(MARIA_HA *info, HA_KEYSEG *keyseg, byte *page_buf, - byte *c, uint key_length) +int maria_rtree_page_mbr(MARIA_HA *info, HA_KEYSEG *keyseg, uchar *page_buf, + uchar *c, uint key_length) { uint inc = 0; uint k_len = key_length; uint nod_flag = _ma_test_if_nod(page_buf); - byte *k; - byte *last = rt_PAGE_END(page_buf); + uchar *k; + uchar *last = rt_PAGE_END(page_buf); for (; (int)key_length > 0; keyseg += 2) { diff --git a/storage/maria/ma_rt_mbr.h b/storage/maria/ma_rt_mbr.h index 01da74418a6..ad855518e62 100644 --- a/storage/maria/ma_rt_mbr.h +++ b/storage/maria/ma_rt_mbr.h @@ -19,20 +19,20 @@ #ifdef HAVE_RTREE_KEYS -int maria_rtree_key_cmp(HA_KEYSEG *keyseg, byte *a, byte *b, uint key_length, +int maria_rtree_key_cmp(HA_KEYSEG *keyseg, uchar *a, uchar *b, uint key_length, uint nextflag); -int maria_rtree_combine_rect(HA_KEYSEG *keyseg,byte *, byte *, byte*, +int maria_rtree_combine_rect(HA_KEYSEG *keyseg,uchar *, uchar *, uchar*, uint key_length); -double maria_rtree_rect_volume(HA_KEYSEG *keyseg, byte*, uint key_length); -int maria_rtree_d_mbr(HA_KEYSEG *keyseg, byte *a, uint key_length, +double maria_rtree_rect_volume(HA_KEYSEG *keyseg, uchar*, uint key_length); +int maria_rtree_d_mbr(HA_KEYSEG *keyseg, uchar *a, uint key_length, double *res); -double maria_rtree_overlapping_area(HA_KEYSEG *keyseg, byte *a, byte *b, +double maria_rtree_overlapping_area(HA_KEYSEG *keyseg, uchar *a, uchar *b, uint key_length); -double maria_rtree_area_increase(HA_KEYSEG *keyseg, byte *a, byte *b, +double maria_rtree_area_increase(HA_KEYSEG *keyseg, uchar *a, uchar *b, uint key_length, double *ab_area); -double maria_rtree_perimeter_increase(HA_KEYSEG *keyseg, byte* a, byte* b, +double maria_rtree_perimeter_increase(HA_KEYSEG *keyseg, uchar* a, uchar* b, uint key_length, double *ab_perim); -int maria_rtree_page_mbr(MARIA_HA *info, HA_KEYSEG *keyseg, byte *page_buf, - byte* c, uint key_length); +int maria_rtree_page_mbr(MARIA_HA *info, HA_KEYSEG *keyseg, uchar *page_buf, + uchar* c, uint key_length); #endif /*HAVE_RTREE_KEYS*/ #endif /* _rt_mbr_h */ diff --git a/storage/maria/ma_rt_split.c b/storage/maria/ma_rt_split.c index 6a66c4424eb..9d195d802c1 100644 --- a/storage/maria/ma_rt_split.c +++ b/storage/maria/ma_rt_split.c @@ -26,7 +26,7 @@ typedef struct { double square; int n_node; - byte *key; + uchar *key; double *coords; } SplitStruct; @@ -247,7 +247,7 @@ static int split_maria_rtree_node(SplitStruct *node, int n_entries, } int maria_rtree_split_page(MARIA_HA *info, MARIA_KEYDEF *keyinfo, - byte *page, byte *key, + uchar *page, uchar *key, uint key_length, my_off_t *new_page_offs) { int n1, n2; /* Number of items in groups */ @@ -259,8 +259,8 @@ int maria_rtree_split_page(MARIA_HA *info, MARIA_KEYDEF *keyinfo, double *next_coord; double *old_coord; int n_dim; - byte *source_cur, *cur1, *cur2; - byte *new_page; + uchar *source_cur, *cur1, *cur2; + uchar *new_page; int err_code= 0; uint nod_flag= _ma_test_if_nod(page); uint full_length= key_length + (nod_flag ? nod_flag : @@ -304,7 +304,7 @@ int maria_rtree_split_page(MARIA_HA *info, MARIA_KEYDEF *keyinfo, goto split_err; } - if (!(new_page = (byte*) my_alloca((uint)keyinfo->block_length))) + if (!(new_page = (uchar*) my_alloca((uint)keyinfo->block_length))) { err_code= -1; goto split_err; @@ -317,7 +317,7 @@ int maria_rtree_split_page(MARIA_HA *info, MARIA_KEYDEF *keyinfo, n1= n2 = 0; for (cur = task; cur < stop; ++cur) { - byte *to; + uchar *to; if (cur->n_node == 1) { to = cur1; @@ -344,10 +344,10 @@ int maria_rtree_split_page(MARIA_HA *info, MARIA_KEYDEF *keyinfo, err_code= _ma_write_keypage(info, keyinfo, *new_page_offs, DFLT_INIT_HITS, new_page); - my_afree((byte*)new_page); + my_afree((uchar*)new_page); split_err: - my_afree((byte*) coord_buf); + my_afree((uchar*) coord_buf); return err_code; } diff --git a/storage/maria/ma_scan.c b/storage/maria/ma_scan.c index 4ed4027378e..f9657833fdd 100644 --- a/storage/maria/ma_scan.c +++ b/storage/maria/ma_scan.c @@ -45,7 +45,7 @@ int maria_scan_init(register MARIA_HA *info) # Error code */ -int maria_scan(MARIA_HA *info, byte *record) +int maria_scan(MARIA_HA *info, uchar *record) { DBUG_ENTER("maria_scan"); /* Init all but update-flag */ diff --git a/storage/maria/ma_search.c b/storage/maria/ma_search.c index f3e7a0d542a..1a7263e6a5a 100644 --- a/storage/maria/ma_search.c +++ b/storage/maria/ma_search.c @@ -19,8 +19,8 @@ #include "m_ctype.h" static my_bool _ma_get_prev_key(MARIA_HA *info, MARIA_KEYDEF *keyinfo, - byte *page, - byte *key, byte *keypos, + uchar *page, + uchar *key, uchar *keypos, uint *return_key_length); /* Check index */ @@ -55,13 +55,13 @@ int _ma_check_index(MARIA_HA *info, int inx) */ int _ma_search(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, - byte *key, uint key_len, uint nextflag, register my_off_t pos) + uchar *key, uint key_len, uint nextflag, register my_off_t pos) { my_bool last_key; int error,flag; uint nod_flag; - byte *keypos,*maxpos; - byte lastkey[HA_MAX_KEY_BUFF],*buff; + uchar *keypos,*maxpos; + uchar lastkey[HA_MAX_KEY_BUFF],*buff; DBUG_ENTER("_ma_search"); DBUG_PRINT("enter",("pos: %lu nextflag: %u lastpos: %lu", (ulong) pos, nextflag, (ulong) info->cur_row.lastpos)); @@ -119,7 +119,7 @@ int _ma_search(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, } if (pos != info->last_keypage) { - byte *old_buff=buff; + uchar *old_buff=buff; if (!(buff= _ma_fetch_keypage(info,keyinfo,pos,DFLT_INIT_HITS, info->keyread_buff, test(!(nextflag & SEARCH_SAVE_BUFF))))) @@ -175,9 +175,9 @@ err: /* ret_pos point to where find or bigger key starts */ /* ARGSUSED */ -int _ma_bin_search(MARIA_HA *info, register MARIA_KEYDEF *keyinfo, byte *page, - byte *key, uint key_len, uint comp_flag, byte **ret_pos, - byte *buff __attribute__((unused)), my_bool *last_key) +int _ma_bin_search(MARIA_HA *info, register MARIA_KEYDEF *keyinfo, uchar *page, + uchar *key, uint key_len, uint comp_flag, uchar **ret_pos, + uchar *buff __attribute__((unused)), my_bool *last_key) { reg4 int start,mid,end,save_end; int flag; @@ -239,13 +239,13 @@ int _ma_bin_search(MARIA_HA *info, register MARIA_KEYDEF *keyinfo, byte *page, < 0 Not found. */ -int _ma_seq_search(MARIA_HA *info, register MARIA_KEYDEF *keyinfo, byte *page, - byte *key, uint key_len, uint comp_flag, byte **ret_pos, - byte *buff, my_bool *last_key) +int _ma_seq_search(MARIA_HA *info, register MARIA_KEYDEF *keyinfo, uchar *page, + uchar *key, uint key_len, uint comp_flag, uchar **ret_pos, + uchar *buff, my_bool *last_key) { int flag; uint nod_flag,length,not_used[2]; - byte t_buff[HA_MAX_KEY_BUFF],*end; + uchar t_buff[HA_MAX_KEY_BUFF],*end; DBUG_ENTER("_ma_seq_search"); LINT_INIT(flag); LINT_INIT(length); @@ -285,8 +285,8 @@ int _ma_seq_search(MARIA_HA *info, register MARIA_KEYDEF *keyinfo, byte *page, int _ma_prefix_search(MARIA_HA *info, register MARIA_KEYDEF *keyinfo, - byte *page, byte *key, uint key_len, uint nextflag, - byte **ret_pos, byte *buff, my_bool *last_key) + uchar *page, uchar *key, uint key_len, uint nextflag, + uchar **ret_pos, uchar *buff, my_bool *last_key) { /* my_flag is raw comparison result to be changed according to @@ -297,11 +297,11 @@ int _ma_prefix_search(MARIA_HA *info, register MARIA_KEYDEF *keyinfo, uint nod_flag, length, len, matched, cmplen, kseg_len; uint prefix_len,suffix_len; int key_len_skip, seg_len_pack, key_len_left; - byte *end; + uchar *end; uchar *kseg, *vseg, *saved_vseg, *saved_from; uchar *sort_order= keyinfo->seg->charset->sort_order; - byte tt_buff[HA_MAX_KEY_BUFF+2], *t_buff=tt_buff+2; - byte *saved_to; + uchar tt_buff[HA_MAX_KEY_BUFF+2], *t_buff=tt_buff+2; + uchar *saved_to; uint saved_length=0, saved_prefix_len=0; uint length_pack; DBUG_ENTER("_ma_prefix_search"); @@ -331,7 +331,7 @@ int _ma_prefix_search(MARIA_HA *info, register MARIA_KEYDEF *keyinfo, Keys are compressed the following way: If the max length of first key segment <= 127 bytes the prefix is - 1 byte else it's 2 byte + 1 uchar else it's 2 byte (prefix) length The high bit is set if this is a prefix for the prev key. [suffix length] Packed length of suffix if the previous was a prefix. @@ -412,7 +412,7 @@ int _ma_prefix_search(MARIA_HA *info, register MARIA_KEYDEF *keyinfo, from+=l; } from+= keyseg->length; - page= (byte*) from+nod_flag; + page= (uchar*) from+nod_flag; length= (uint) (from-vseg); } @@ -496,7 +496,7 @@ int _ma_prefix_search(MARIA_HA *info, register MARIA_KEYDEF *keyinfo, /* We have to compare k and vseg as if they were space extended */ for (vseg_end= vseg + (len-cmplen) ; - vseg < vseg_end && *vseg == (byte) ' '; + vseg < vseg_end && *vseg == (uchar) ' '; vseg++, matched++) ; DBUG_ASSERT(vseg < vseg_end); @@ -552,7 +552,7 @@ int _ma_prefix_search(MARIA_HA *info, register MARIA_KEYDEF *keyinfo, saved_length=length; } if (saved_length) - memcpy(saved_to, (byte*) saved_from, saved_length); + memcpy(saved_to, (uchar*) saved_from, saved_length); *last_key= page == end; @@ -563,7 +563,7 @@ int _ma_prefix_search(MARIA_HA *info, register MARIA_KEYDEF *keyinfo, /* Get pos to a key_block */ -my_off_t _ma_kpos(uint nod_flag, byte *after_key) +my_off_t _ma_kpos(uint nod_flag, uchar *after_key) { after_key-=nod_flag; switch (nod_flag) { @@ -599,7 +599,7 @@ my_off_t _ma_kpos(uint nod_flag, byte *after_key) /* Save pos to a key_block */ -void _ma_kpointer(register MARIA_HA *info, register byte *buff, my_off_t pos) +void _ma_kpointer(register MARIA_HA *info, register uchar *buff, my_off_t pos) { pos/=MARIA_MIN_KEY_BLOCK_LENGTH; switch (info->s->base.key_reflength) { @@ -618,7 +618,7 @@ void _ma_kpointer(register MARIA_HA *info, register byte *buff, my_off_t pos) case 4: mi_int4store(buff,pos); break; case 3: mi_int3store(buff,pos); break; case 2: mi_int2store(buff,(uint) pos); break; - case 1: buff[0]= (byte) pos; break; + case 1: buff[0]= (uchar) pos; break; default: abort(); /* impossible */ } } /* _ma_kpointer */ @@ -627,7 +627,7 @@ void _ma_kpointer(register MARIA_HA *info, register byte *buff, my_off_t pos) /* Calc pos to a data-record from a key */ -my_off_t _ma_dpos(MARIA_HA *info, uint nod_flag, const byte *after_key) +my_off_t _ma_dpos(MARIA_HA *info, uint nod_flag, const uchar *after_key) { my_off_t pos; after_key-=(nod_flag + info->s->rec_reflength); @@ -656,7 +656,7 @@ my_off_t _ma_dpos(MARIA_HA *info, uint nod_flag, const byte *after_key) /* Calc position from a record pointer ( in delete link chain ) */ -my_off_t _ma_rec_pos(MARIA_SHARE *s, byte *ptr) +my_off_t _ma_rec_pos(MARIA_SHARE *s, uchar *ptr) { my_off_t pos; switch (s->rec_reflength) { @@ -713,7 +713,7 @@ my_off_t _ma_rec_pos(MARIA_SHARE *s, byte *ptr) /* save position to record */ -void _ma_dpointer(MARIA_HA *info, byte *buff, my_off_t pos) +void _ma_dpointer(MARIA_HA *info, uchar *buff, my_off_t pos) { if (info->s->data_file_type == STATIC_RECORD && pos != HA_OFFSET_ERROR) @@ -752,9 +752,9 @@ void _ma_dpointer(MARIA_HA *info, byte *buff, my_off_t pos) /* same as _ma_get_key but used with fixed length keys */ uint _ma_get_static_key(register MARIA_KEYDEF *keyinfo, uint nod_flag, - register byte **page, register byte *key) + register uchar **page, register uchar *key) { - memcpy((byte*) key,(byte*) *page, + memcpy((uchar*) key,(uchar*) *page, (size_t) (keyinfo->keylength+nod_flag)); *page+=keyinfo->keylength+nod_flag; return(keyinfo->keylength); @@ -776,10 +776,10 @@ uint _ma_get_static_key(register MARIA_KEYDEF *keyinfo, uint nod_flag, */ uint _ma_get_pack_key(register MARIA_KEYDEF *keyinfo, uint nod_flag, - register byte **page_pos, register byte *key) + register uchar **page_pos, register uchar *key) { reg1 HA_KEYSEG *keyseg; - byte *start_key,*page=*page_pos; + uchar *start_key,*page=*page_pos; uint length; start_key=key; @@ -788,7 +788,7 @@ uint _ma_get_pack_key(register MARIA_KEYDEF *keyinfo, uint nod_flag, if (keyseg->flag & HA_PACK_KEY) { /* key with length, packed to previous key */ - byte *start= key; + uchar *start= key; uint packed= *page & 128,tot_length,rest_length; if (keyseg->length >= 127) { @@ -891,19 +891,19 @@ uint _ma_get_pack_key(register MARIA_KEYDEF *keyinfo, uint nod_flag, if (keyseg->flag & (HA_VAR_LENGTH_PART | HA_BLOB_PART | HA_SPACE_PACK)) { - byte *tmp=page; + uchar *tmp=page; get_key_length(length,tmp); length+=(uint) (tmp-page); } else length=keyseg->length; } - memcpy((byte*) key,(byte*) page,(size_t) length); + memcpy((uchar*) key,(uchar*) page,(size_t) length); key+=length; page+=length; } length=keyseg->length+nod_flag; - bmove((byte*) key,(byte*) page,length); + bmove((uchar*) key,(uchar*) page,length); *page_pos= page+length; return ((uint) (key-start_key)+keyseg->length); } /* _ma_get_pack_key */ @@ -913,10 +913,10 @@ uint _ma_get_pack_key(register MARIA_KEYDEF *keyinfo, uint nod_flag, /* key that is packed relatively to previous */ uint _ma_get_binary_pack_key(register MARIA_KEYDEF *keyinfo, uint nod_flag, - register byte **page_pos, register byte *key) + register uchar **page_pos, register uchar *key) { reg1 HA_KEYSEG *keyseg; - byte *start_key,*page,*page_end,*from,*from_end; + uchar *start_key,*page,*page_end,*from,*from_end; uint length,tmp; DBUG_ENTER("_ma_get_binary_pack_key"); @@ -990,7 +990,7 @@ uint _ma_get_binary_pack_key(register MARIA_KEYDEF *keyinfo, uint nod_flag, DBUG_ASSERT((int) length >= 0); DBUG_PRINT("info",("key: 0x%lx from: 0x%lx length: %u", (long) key, (long) from, length)); - memmove((byte*) key, (byte*) from, (size_t) length); + memmove((uchar*) key, (uchar*) from, (size_t) length); key+=length; from+=length; } @@ -1009,7 +1009,7 @@ uint _ma_get_binary_pack_key(register MARIA_KEYDEF *keyinfo, uint nod_flag, my_errno=HA_ERR_CRASHED; DBUG_RETURN(0); /* Error */ } - memcpy((byte*) key,(byte*) from,(size_t) length); + memcpy((uchar*) key,(uchar*) from,(size_t) length); *page_pos= from+length; } DBUG_RETURN((uint) (key-start_key)+keyseg->length); @@ -1019,8 +1019,8 @@ uint _ma_get_binary_pack_key(register MARIA_KEYDEF *keyinfo, uint nod_flag, /* Get key at position without knowledge of previous key */ /* Returns pointer to next key */ -byte *_ma_get_key(MARIA_HA *info, MARIA_KEYDEF *keyinfo, byte *page, - byte *key, byte *keypos, uint *return_key_length) +uchar *_ma_get_key(MARIA_HA *info, MARIA_KEYDEF *keyinfo, uchar *page, + uchar *key, uchar *keypos, uint *return_key_length) { uint nod_flag; DBUG_ENTER("_ma_get_key"); @@ -1028,7 +1028,7 @@ byte *_ma_get_key(MARIA_HA *info, MARIA_KEYDEF *keyinfo, byte *page, nod_flag=_ma_test_if_nod(page); if (! (keyinfo->flag & (HA_VAR_LENGTH_KEY | HA_BINARY_PACK_KEY))) { - bmove((byte*) key,(byte*) keypos,keyinfo->keylength+nod_flag); + bmove((uchar*) key,(uchar*) keypos,keyinfo->keylength+nod_flag); DBUG_RETURN(keypos+keyinfo->keylength+nod_flag); } else @@ -1056,7 +1056,7 @@ byte *_ma_get_key(MARIA_HA *info, MARIA_KEYDEF *keyinfo, byte *page, /* Returns 0 if ok */ static my_bool _ma_get_prev_key(MARIA_HA *info, MARIA_KEYDEF *keyinfo, - byte *page, byte *key, byte *keypos, + uchar *page, uchar *key, uchar *keypos, uint *return_key_length) { uint nod_flag; @@ -1066,7 +1066,7 @@ static my_bool _ma_get_prev_key(MARIA_HA *info, MARIA_KEYDEF *keyinfo, if (! (keyinfo->flag & (HA_VAR_LENGTH_KEY | HA_BINARY_PACK_KEY))) { *return_key_length=keyinfo->keylength; - bmove((byte*) key,(byte*) keypos- *return_key_length-nod_flag, + bmove((uchar*) key,(uchar*) keypos- *return_key_length-nod_flag, *return_key_length); DBUG_RETURN(0); } @@ -1093,11 +1093,11 @@ static my_bool _ma_get_prev_key(MARIA_HA *info, MARIA_KEYDEF *keyinfo, /* Get last key from key-page */ /* Return pointer to where key starts */ -byte *_ma_get_last_key(MARIA_HA *info, MARIA_KEYDEF *keyinfo, byte *page, - byte *lastkey, byte *endpos, uint *return_key_length) +uchar *_ma_get_last_key(MARIA_HA *info, MARIA_KEYDEF *keyinfo, uchar *page, + uchar *lastkey, uchar *endpos, uint *return_key_length) { uint nod_flag; - byte *lastpos; + uchar *lastpos; DBUG_ENTER("_ma_get_last_key"); DBUG_PRINT("enter",("page: 0x%lx endpos: 0x%lx", (long) page, (long) endpos)); @@ -1108,7 +1108,7 @@ byte *_ma_get_last_key(MARIA_HA *info, MARIA_KEYDEF *keyinfo, byte *page, lastpos=endpos-keyinfo->keylength-nod_flag; *return_key_length=keyinfo->keylength; if (lastpos > page) - bmove((byte*) lastkey,(byte*) lastpos,keyinfo->keylength+nod_flag); + bmove((uchar*) lastkey,(uchar*) lastpos,keyinfo->keylength+nod_flag); } else { @@ -1136,10 +1136,10 @@ byte *_ma_get_last_key(MARIA_HA *info, MARIA_KEYDEF *keyinfo, byte *page, /* Calculate length of key */ -uint _ma_keylength(MARIA_KEYDEF *keyinfo, register const byte *key) +uint _ma_keylength(MARIA_KEYDEF *keyinfo, register const uchar *key) { reg1 HA_KEYSEG *keyseg; - const byte *start; + const uchar *start; if (! (keyinfo->flag & (HA_VAR_LENGTH_KEY | HA_BINARY_PACK_KEY))) return (keyinfo->keylength); @@ -1171,11 +1171,11 @@ uint _ma_keylength(MARIA_KEYDEF *keyinfo, register const byte *key) after '0xDF' but find 'ss' */ -uint _ma_keylength_part(MARIA_KEYDEF *keyinfo, register const byte *key, +uint _ma_keylength_part(MARIA_KEYDEF *keyinfo, register const uchar *key, HA_KEYSEG *end) { reg1 HA_KEYSEG *keyseg; - const byte *start= key; + const uchar *start= key; for (keyseg=keyinfo->seg ; keyseg != end ; keyseg++) { @@ -1197,7 +1197,7 @@ uint _ma_keylength_part(MARIA_KEYDEF *keyinfo, register const byte *key, /* Move a key */ -byte *_ma_move_key(MARIA_KEYDEF *keyinfo, byte *to, const byte *from) +uchar *_ma_move_key(MARIA_KEYDEF *keyinfo, uchar *to, const uchar *from) { reg1 uint length; memcpy(to, from, (size_t) (length= _ma_keylength(keyinfo, from))); @@ -1213,11 +1213,11 @@ byte *_ma_move_key(MARIA_KEYDEF *keyinfo, byte *to, const byte *from) */ int _ma_search_next(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, - byte *key, uint key_length, uint nextflag, my_off_t pos) + uchar *key, uint key_length, uint nextflag, my_off_t pos) { int error; uint nod_flag; - byte lastkey[HA_MAX_KEY_BUFF]; + uchar lastkey[HA_MAX_KEY_BUFF]; DBUG_ENTER("_ma_search_next"); DBUG_PRINT("enter",("nextflag: %u lastpos: %lu int_keypos: %lu page_changed %d keyread_buff_used: %d", nextflag, (ulong) info->cur_row.lastpos, @@ -1299,7 +1299,7 @@ int _ma_search_first(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, register my_off_t pos) { uint nod_flag; - byte *page; + uchar *page; DBUG_ENTER("_ma_search_first"); if (pos == HA_OFFSET_ERROR) @@ -1343,7 +1343,7 @@ int _ma_search_last(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, register my_off_t pos) { uint nod_flag; - byte *buff,*page; + uchar *buff,*page; DBUG_ENTER("_ma_search_last"); if (pos == HA_OFFSET_ERROR) @@ -1398,10 +1398,10 @@ int _ma_search_last(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, int _ma_calc_static_key_length(MARIA_KEYDEF *keyinfo,uint nod_flag, - byte *next_pos __attribute__((unused)), - byte *org_key __attribute__((unused)), - byte *prev_key __attribute__((unused)), - const byte *key, MARIA_KEY_PARAM *s_temp) + uchar *next_pos __attribute__((unused)), + uchar *org_key __attribute__((unused)), + uchar *prev_key __attribute__((unused)), + const uchar *key, MARIA_KEY_PARAM *s_temp) { s_temp->key= key; return (int) (s_temp->totlength=keyinfo->keylength+nod_flag); @@ -1411,10 +1411,10 @@ _ma_calc_static_key_length(MARIA_KEYDEF *keyinfo,uint nod_flag, int _ma_calc_var_key_length(MARIA_KEYDEF *keyinfo,uint nod_flag, - byte *next_pos __attribute__((unused)), - byte *org_key __attribute__((unused)), - byte *prev_key __attribute__((unused)), - const byte *key, MARIA_KEY_PARAM *s_temp) + uchar *next_pos __attribute__((unused)), + uchar *org_key __attribute__((unused)), + uchar *prev_key __attribute__((unused)), + const uchar *key, MARIA_KEY_PARAM *s_temp) { s_temp->key= key; return (int) (s_temp->totlength= _ma_keylength(keyinfo,key)+nod_flag); @@ -1427,7 +1427,7 @@ _ma_calc_var_key_length(MARIA_KEYDEF *keyinfo,uint nod_flag, Keys are compressed the following way: If the max length of first key segment <= 127 bytes the prefix is - 1 byte else it's 2 byte + 1 uchar else it's 2 byte prefix byte(s) The high bit is set if this is a prefix for the prev key length Packed length if the previous was a prefix byte @@ -1441,15 +1441,15 @@ _ma_calc_var_key_length(MARIA_KEYDEF *keyinfo,uint nod_flag, int _ma_calc_var_pack_key_length(MARIA_KEYDEF *keyinfo, uint nod_flag, - byte *next_key, - byte *org_key, byte *prev_key, const byte *key, + uchar *next_key, + uchar *org_key, uchar *prev_key, const uchar *key, MARIA_KEY_PARAM *s_temp) { reg1 HA_KEYSEG *keyseg; int length; uint key_length,ref_length,org_key_length=0, length_pack,new_key_length,diff_flag,pack_marker; - const byte *start,*end,*key_end; + const uchar *start,*end,*key_end; uchar *sort_order; bool same_length; @@ -1726,9 +1726,9 @@ _ma_calc_var_pack_key_length(MARIA_KEYDEF *keyinfo, uint nod_flag, /* Length of key which is prefix compressed */ int _ma_calc_bin_pack_key_length(MARIA_KEYDEF *keyinfo, uint nod_flag, - byte *next_key, - byte *org_key, byte *prev_key, - const byte *key, + uchar *next_key, + uchar *org_key, uchar *prev_key, + const uchar *key, MARIA_KEY_PARAM *s_temp) { uint length,key_length,ref_length; @@ -1746,7 +1746,7 @@ int _ma_calc_bin_pack_key_length(MARIA_KEYDEF *keyinfo, uint nod_flag, As keys may be identical when running a sort in maria_chk, we have to guard against the case where keys may be identical */ - const byte *end; + const uchar *end; end=key+key_length; for ( ; *key == *prev_key && key < end; key++,prev_key++) ; s_temp->ref_length= ref_length=(uint) (key-s_temp->key); @@ -1767,7 +1767,7 @@ int _ma_calc_bin_pack_key_length(MARIA_KEYDEF *keyinfo, uint nod_flag, /* If first key and next key is packed (only on delete) */ if (!prev_key && org_key && next_length) { - const byte *end; + const uchar *end; for (key= s_temp->key, end=key+next_length ; *key == *org_key && key < end; key++,org_key++) ; @@ -1809,10 +1809,10 @@ int _ma_calc_bin_pack_key_length(MARIA_KEYDEF *keyinfo, uint nod_flag, /* store key without compression */ void _ma_store_static_key(MARIA_KEYDEF *keyinfo __attribute__((unused)), - register byte *key_pos, + register uchar *key_pos, register MARIA_KEY_PARAM *s_temp) { - memcpy((byte*) key_pos,(byte*) s_temp->key,(size_t) s_temp->totlength); + memcpy((uchar*) key_pos,(uchar*) s_temp->key,(size_t) s_temp->totlength); } @@ -1824,11 +1824,11 @@ void _ma_store_static_key(MARIA_KEYDEF *keyinfo __attribute__((unused)), void _ma_store_var_pack_key(MARIA_KEYDEF *keyinfo __attribute__((unused)), - register byte *key_pos, + register uchar *key_pos, register MARIA_KEY_PARAM *s_temp) { uint length; - byte *start; + uchar *start; start=key_pos; @@ -1845,7 +1845,7 @@ void _ma_store_var_pack_key(MARIA_KEYDEF *keyinfo __attribute__((unused)), /* Not packed against previous key */ store_pack_length(s_temp->pack_marker == 128,key_pos,s_temp->key_length); } - bmove((byte*) key_pos,(byte*) s_temp->key, + bmove((uchar*) key_pos,(uchar*) s_temp->key, (length=s_temp->totlength-(uint) (key_pos-start))); if (!s_temp->next_key_pos) /* No following key */ @@ -1887,7 +1887,7 @@ void _ma_store_var_pack_key(MARIA_KEYDEF *keyinfo __attribute__((unused)), /* variable length key with prefix compression */ void _ma_store_bin_pack_key(MARIA_KEYDEF *keyinfo __attribute__((unused)), - register byte *key_pos, + register uchar *key_pos, register MARIA_KEY_PARAM *s_temp) { store_key_length_inc(key_pos,s_temp->ref_length); diff --git a/storage/maria/ma_sort.c b/storage/maria/ma_sort.c index d6256deb39c..bc2b75807d9 100644 --- a/storage/maria/ma_sort.c +++ b/storage/maria/ma_sort.c @@ -48,31 +48,31 @@ extern void print_error _VARARGS((const char *fmt,...)); /* Functions defined in this file */ static ha_rows NEAR_F find_all_keys(MARIA_SORT_PARAM *info,uint keys, - byte **sort_keys, + uchar **sort_keys, DYNAMIC_ARRAY *buffpek,int *maxbuffer, IO_CACHE *tempfile, IO_CACHE *tempfile_for_exceptions); -static int NEAR_F write_keys(MARIA_SORT_PARAM *info, byte **sort_keys, +static int NEAR_F write_keys(MARIA_SORT_PARAM *info, uchar **sort_keys, uint count, BUFFPEK *buffpek,IO_CACHE *tempfile); -static int NEAR_F write_key(MARIA_SORT_PARAM *info, byte *key, +static int NEAR_F write_key(MARIA_SORT_PARAM *info, uchar *key, IO_CACHE *tempfile); -static int NEAR_F write_index(MARIA_SORT_PARAM *info, byte **sort_keys, +static int NEAR_F write_index(MARIA_SORT_PARAM *info, uchar **sort_keys, uint count); static int NEAR_F merge_many_buff(MARIA_SORT_PARAM *info,uint keys, - byte **sort_keys, + uchar **sort_keys, BUFFPEK *buffpek,int *maxbuffer, IO_CACHE *t_file); static uint NEAR_F read_to_buffer(IO_CACHE *fromfile,BUFFPEK *buffpek, uint sort_length); static int NEAR_F merge_buffers(MARIA_SORT_PARAM *info,uint keys, IO_CACHE *from_file, IO_CACHE *to_file, - byte **sort_keys, BUFFPEK *lastbuff, + uchar **sort_keys, BUFFPEK *lastbuff, BUFFPEK *Fb, BUFFPEK *Tb); -static int NEAR_F merge_index(MARIA_SORT_PARAM *,uint, byte **,BUFFPEK *, int, +static int NEAR_F merge_index(MARIA_SORT_PARAM *,uint, uchar **,BUFFPEK *, int, IO_CACHE *); static int flush_maria_ft_buf(MARIA_SORT_PARAM *info); -static int NEAR_F write_keys_varlen(MARIA_SORT_PARAM *info, byte **sort_keys, +static int NEAR_F write_keys_varlen(MARIA_SORT_PARAM *info, uchar **sort_keys, uint count, BUFFPEK *buffpek, IO_CACHE *tempfile); static uint NEAR_F read_to_buffer_varlen(IO_CACHE *fromfile,BUFFPEK *buffpek, @@ -84,7 +84,7 @@ static int NEAR_F write_merge_key_varlen(MARIA_SORT_PARAM *info, char* key, uint sort_length, uint count); static inline int -my_var_write(MARIA_SORT_PARAM *info, IO_CACHE *to_file, byte *bufs); +my_var_write(MARIA_SORT_PARAM *info, IO_CACHE *to_file, uchar *bufs); /* Creates a index of sorted keys @@ -107,7 +107,7 @@ int _ma_create_index_by_sort(MARIA_SORT_PARAM *info, my_bool no_messages, uint memavl,old_memavl,keys,sort_length; DYNAMIC_ARRAY buffpek; ha_rows records; - byte **sort_keys; + uchar **sort_keys; IO_CACHE tempfile, tempfile_for_exceptions; DBUG_ENTER("_ma_create_index_by_sort"); DBUG_PRINT("enter",("sort_length: %d", info->key_length)); @@ -128,7 +128,7 @@ int _ma_create_index_by_sort(MARIA_SORT_PARAM *info, my_bool no_messages, my_b_clear(&tempfile); my_b_clear(&tempfile_for_exceptions); bzero((char*) &buffpek,sizeof(buffpek)); - sort_keys= (byte **) NULL; error= 1; + sort_keys= (uchar **) NULL; error= 1; maxbuffer=1; memavl=max(sortbuff_size,MIN_SORT_MEMORY); @@ -157,13 +157,13 @@ int _ma_create_index_by_sort(MARIA_SORT_PARAM *info, my_bool no_messages, } while ((maxbuffer= (int) (records/(keys-1)+1)) != skr); - if ((sort_keys=(byte**) my_malloc(keys*(sort_length+sizeof(char*))+ + if ((sort_keys=(uchar**) my_malloc(keys*(sort_length+sizeof(char*))+ HA_FT_MAXBYTELEN, MYF(0)))) { if (my_init_dynamic_array(&buffpek, sizeof(BUFFPEK), maxbuffer, maxbuffer/2)) { - my_free((gptr) sort_keys,MYF(0)); + my_free((uchar*) sort_keys,MYF(0)); sort_keys= 0; } else @@ -230,12 +230,12 @@ int _ma_create_index_by_sort(MARIA_SORT_PARAM *info, my_bool no_messages, reinit_io_cache(&tempfile_for_exceptions,READ_CACHE,0L,0,0)) goto err; - while (!my_b_read(&tempfile_for_exceptions,(byte*)&key_length, + while (!my_b_read(&tempfile_for_exceptions,(uchar*)&key_length, sizeof(key_length)) - && !my_b_read(&tempfile_for_exceptions,(byte*)sort_keys, + && !my_b_read(&tempfile_for_exceptions,(uchar*)sort_keys, (uint) key_length)) { - if (_ma_ck_write(idx,keyno,(byte*) sort_keys,key_length-ref_length)) + if (_ma_ck_write(idx,keyno,(uchar*) sort_keys,key_length-ref_length)) goto err; } } @@ -244,7 +244,7 @@ int _ma_create_index_by_sort(MARIA_SORT_PARAM *info, my_bool no_messages, err: if (sort_keys) - my_free((gptr) sort_keys,MYF(0)); + my_free((uchar*) sort_keys,MYF(0)); delete_dynamic(&buffpek); close_cached_file(&tempfile); close_cached_file(&tempfile_for_exceptions); @@ -256,7 +256,7 @@ err: /* Search after all keys and place them in a temp. file */ static ha_rows NEAR_F find_all_keys(MARIA_SORT_PARAM *info, uint keys, - byte **sort_keys, DYNAMIC_ARRAY *buffpek, + uchar **sort_keys, DYNAMIC_ARRAY *buffpek, int *maxbuffer, IO_CACHE *tempfile, IO_CACHE *tempfile_for_exceptions) { @@ -265,7 +265,7 @@ static ha_rows NEAR_F find_all_keys(MARIA_SORT_PARAM *info, uint keys, DBUG_ENTER("find_all_keys"); idx=error=0; - sort_keys[0]= (byte*) (sort_keys+keys); + sort_keys[0]= (uchar*) (sort_keys+keys); while (!(error=(*info->key_read)(info,sort_keys[idx]))) { @@ -283,7 +283,7 @@ static ha_rows NEAR_F find_all_keys(MARIA_SORT_PARAM *info, uint keys, tempfile)) DBUG_RETURN(HA_POS_ERROR); /* purecov: inspected */ - sort_keys[0]=(byte*) (sort_keys+keys); + sort_keys[0]=(uchar*) (sort_keys+keys); memcpy(sort_keys[0],sort_keys[idx-1],(size_t) info->key_length); idx=1; } @@ -314,7 +314,7 @@ pthread_handler_t _ma_thr_find_all_keys(void *arg) int error; uint memavl,old_memavl,keys,sort_length; uint idx, maxbuffer; - byte **sort_keys=0; + uchar **sort_keys=0; LINT_INIT(keys); @@ -375,7 +375,7 @@ pthread_handler_t _ma_thr_find_all_keys(void *arg) } while ((maxbuffer= (int) (idx/(keys-1)+1)) != skr); } - if ((sort_keys= (byte **) + if ((sort_keys= (uchar **) my_malloc(keys*(sort_length+sizeof(char*))+ ((sort_param->keyinfo->flag & HA_FULLTEXT) ? HA_FT_MAXBYTELEN : 0), MYF(0)))) @@ -383,8 +383,8 @@ pthread_handler_t _ma_thr_find_all_keys(void *arg) if (my_init_dynamic_array(&sort_param->buffpek, sizeof(BUFFPEK), maxbuffer, maxbuffer/2)) { - my_free((gptr) sort_keys,MYF(0)); - sort_keys= (byte **) NULL; /* for err: label */ + my_free((uchar*) sort_keys,MYF(0)); + sort_keys= (uchar **) NULL; /* for err: label */ } else break; @@ -407,7 +407,7 @@ pthread_handler_t _ma_thr_find_all_keys(void *arg) sort_param->sort_keys= sort_keys; idx= error= 0; - sort_keys[0]= (byte*) (sort_keys+keys); + sort_keys[0]= (uchar*) (sort_keys+keys); DBUG_PRINT("info", ("reading keys")); while (!(error= sort_param->sort_info->got_error) && @@ -428,7 +428,7 @@ pthread_handler_t _ma_thr_find_all_keys(void *arg) buffpek), &sort_param->tempfile)) goto err; - sort_keys[0]= (byte*) (sort_keys+keys); + sort_keys[0]= (uchar*) (sort_keys+keys); memcpy(sort_keys[0], sort_keys[idx - 1], (size_t) sort_param->key_length); idx= 1; @@ -455,7 +455,7 @@ pthread_handler_t _ma_thr_find_all_keys(void *arg) err: DBUG_PRINT("error", ("got some error")); sort_param->sort_info->got_error= 1; /* no need to protect with a mutex */ - my_free((gptr) sort_keys,MYF(MY_ALLOW_ZERO_PTR)); + my_free((uchar*) sort_keys,MYF(MY_ALLOW_ZERO_PTR)); sort_param->sort_keys=0; delete_dynamic(& sort_param->buffpek); close_cached_file(&sort_param->tempfile); @@ -497,7 +497,7 @@ int _ma_thr_write_keys(MARIA_SORT_PARAM *sort_param) MARIA_HA *info=sort_info->info; MARIA_SHARE *share=info->s; MARIA_SORT_PARAM *sinfo; - byte *mergebuf=0; + uchar *mergebuf=0; DBUG_ENTER("_ma_thr_write_keys"); LINT_INIT(length); @@ -532,7 +532,7 @@ int _ma_thr_write_keys(MARIA_SORT_PARAM *sort_param) sinfo->notnull: NULL, (ulonglong) info->state->records); } - my_free((gptr) sinfo->sort_keys,MYF(0)); + my_free((uchar*) sinfo->sort_keys,MYF(0)); my_free(sinfo->rec_buff, MYF(MY_ALLOW_ZERO_PTR)); sinfo->sort_keys=0; } @@ -581,7 +581,7 @@ int _ma_thr_write_keys(MARIA_SORT_PARAM *sort_param) { if (param->testflag & T_VERBOSE) printf("Key %d - Merging %u keys\n",sinfo->key+1, sinfo->keys); - if (merge_many_buff(sinfo, keys, (byte **) mergebuf, + if (merge_many_buff(sinfo, keys, (uchar **) mergebuf, dynamic_element(&sinfo->buffpek, 0, BUFFPEK *), (int*) &maxbuffer, &sinfo->tempfile)) { @@ -597,7 +597,7 @@ int _ma_thr_write_keys(MARIA_SORT_PARAM *sort_param) } if (param->testflag & T_VERBOSE) printf("Key %d - Last merge and dumping keys\n", sinfo->key+1); - if (merge_index(sinfo, keys, (byte**) mergebuf, + if (merge_index(sinfo, keys, (uchar**) mergebuf, dynamic_element(&sinfo->buffpek,0,BUFFPEK *), maxbuffer,&sinfo->tempfile) || flush_maria_ft_buf(sinfo) || @@ -622,12 +622,12 @@ int _ma_thr_write_keys(MARIA_SORT_PARAM *sort_param) } while (!got_error && - !my_b_read(&sinfo->tempfile_for_exceptions,(byte*)&key_length, + !my_b_read(&sinfo->tempfile_for_exceptions,(uchar*)&key_length, sizeof(key_length))) { - byte maria_ft_buf[HA_FT_MAXBYTELEN + HA_FT_WLEN + 10]; + uchar maria_ft_buf[HA_FT_MAXBYTELEN + HA_FT_WLEN + 10]; if (key_length > sizeof(maria_ft_buf) || - my_b_read(&sinfo->tempfile_for_exceptions, (byte*)maria_ft_buf, + my_b_read(&sinfo->tempfile_for_exceptions, (uchar*)maria_ft_buf, (uint)key_length) || _ma_ck_write(info, sinfo->key, maria_ft_buf, key_length - info->s->rec_reflength)) @@ -635,21 +635,21 @@ int _ma_thr_write_keys(MARIA_SORT_PARAM *sort_param) } } } - my_free((gptr) mergebuf,MYF(MY_ALLOW_ZERO_PTR)); + my_free((uchar*) mergebuf,MYF(MY_ALLOW_ZERO_PTR)); DBUG_RETURN(got_error); } #endif /* THREAD */ /* Write all keys in memory to file for later merge */ -static int write_keys(MARIA_SORT_PARAM *info, register byte **sort_keys, +static int write_keys(MARIA_SORT_PARAM *info, register uchar **sort_keys, uint count, BUFFPEK *buffpek, IO_CACHE *tempfile) { - byte **end; + uchar **end; uint sort_length=info->key_length; DBUG_ENTER("write_keys"); - qsort2((byte*) sort_keys,count,sizeof(byte*),(qsort2_cmp) info->key_cmp, + qsort2((uchar*) sort_keys,count,sizeof(uchar*),(qsort2_cmp) info->key_cmp, info); if (!my_b_inited(tempfile) && open_cached_file(tempfile, my_tmpdir(info->tmpdir), "ST", @@ -669,13 +669,13 @@ static int write_keys(MARIA_SORT_PARAM *info, register byte **sort_keys, static inline int -my_var_write(MARIA_SORT_PARAM *info, IO_CACHE *to_file, byte *bufs) +my_var_write(MARIA_SORT_PARAM *info, IO_CACHE *to_file, uchar *bufs) { int err; uint16 len= _ma_keylength(info->keyinfo, bufs); /* The following is safe as this is a local file */ - if ((err= my_b_write(to_file, (byte*)&len, sizeof(len)))) + if ((err= my_b_write(to_file, (uchar*)&len, sizeof(len)))) return (err); if ((err= my_b_write(to_file,bufs, (uint) len))) return (err); @@ -684,15 +684,15 @@ my_var_write(MARIA_SORT_PARAM *info, IO_CACHE *to_file, byte *bufs) static int NEAR_F write_keys_varlen(MARIA_SORT_PARAM *info, - register byte **sort_keys, + register uchar **sort_keys, uint count, BUFFPEK *buffpek, IO_CACHE *tempfile) { - byte **end; + uchar **end; int err; DBUG_ENTER("write_keys_varlen"); - qsort2((byte*) sort_keys,count,sizeof(byte*),(qsort2_cmp) info->key_cmp, + qsort2((uchar*) sort_keys,count,sizeof(uchar*),(qsort2_cmp) info->key_cmp, info); if (!my_b_inited(tempfile) && open_cached_file(tempfile, my_tmpdir(info->tmpdir), "ST", @@ -710,7 +710,7 @@ static int NEAR_F write_keys_varlen(MARIA_SORT_PARAM *info, } /* write_keys_varlen */ -static int NEAR_F write_key(MARIA_SORT_PARAM *info, byte *key, +static int NEAR_F write_key(MARIA_SORT_PARAM *info, uchar *key, IO_CACHE *tempfile) { uint key_length=info->real_key_length; @@ -721,7 +721,7 @@ static int NEAR_F write_key(MARIA_SORT_PARAM *info, byte *key, DISK_BUFFER_SIZE, info->sort_info->param->myf_rw)) DBUG_RETURN(1); - if (my_b_write(tempfile, (byte*)&key_length,sizeof(key_length)) || + if (my_b_write(tempfile, (uchar*)&key_length,sizeof(key_length)) || my_b_write(tempfile, key, (uint) key_length)) DBUG_RETURN(1); DBUG_RETURN(0); @@ -731,12 +731,12 @@ static int NEAR_F write_key(MARIA_SORT_PARAM *info, byte *key, /* Write index */ static int NEAR_F write_index(MARIA_SORT_PARAM *info, - register byte **sort_keys, + register uchar **sort_keys, register uint count) { DBUG_ENTER("write_index"); - qsort2((gptr) sort_keys,(size_t) count,sizeof(byte*), + qsort2((uchar*) sort_keys,(size_t) count,sizeof(uchar*), (qsort2_cmp) info->key_cmp,info); while (count--) { @@ -750,7 +750,7 @@ static int NEAR_F write_index(MARIA_SORT_PARAM *info, /* Merge buffers to make < MERGEBUFF2 buffers */ static int NEAR_F merge_many_buff(MARIA_SORT_PARAM *info, uint keys, - byte **sort_keys, BUFFPEK *buffpek, + uchar **sort_keys, BUFFPEK *buffpek, int *maxbuffer, IO_CACHE *t_file) { register int i; @@ -814,7 +814,7 @@ static uint NEAR_F read_to_buffer(IO_CACHE *fromfile, BUFFPEK *buffpek, if ((count=(uint) min((ha_rows) buffpek->max_keys,buffpek->count))) { - if (my_pread(fromfile->file,(byte*) buffpek->base, + if (my_pread(fromfile->file,(uchar*) buffpek->base, (length= sort_length*count),buffpek->file_pos,MYF_RW)) return((uint) -1); /* purecov: inspected */ buffpek->key=buffpek->base; @@ -831,7 +831,7 @@ static uint NEAR_F read_to_buffer_varlen(IO_CACHE *fromfile, BUFFPEK *buffpek, register uint count; uint16 length_of_key = 0; uint idx; - byte *buffp; + uchar *buffp; if ((count=(uint) min((ha_rows) buffpek->max_keys,buffpek->count))) { @@ -839,11 +839,11 @@ static uint NEAR_F read_to_buffer_varlen(IO_CACHE *fromfile, BUFFPEK *buffpek, for (idx=1;idx<=count;idx++) { - if (my_pread(fromfile->file,(byte*)&length_of_key,sizeof(length_of_key), + if (my_pread(fromfile->file,(uchar*)&length_of_key,sizeof(length_of_key), buffpek->file_pos,MYF_RW)) return((uint) -1); buffpek->file_pos+=sizeof(length_of_key); - if (my_pread(fromfile->file,(byte*) buffp,length_of_key, + if (my_pread(fromfile->file,(uchar*) buffp,length_of_key, buffpek->file_pos,MYF_RW)) return((uint) -1); buffpek->file_pos+=length_of_key; @@ -867,7 +867,7 @@ static int NEAR_F write_merge_key_varlen(MARIA_SORT_PARAM *info, for (idx=1;idx<=count;idx++) { int err; - if ((err= my_var_write(info,to_file, (byte*) bufs))) + if ((err= my_var_write(info,to_file, (uchar*) bufs))) return (err); bufs=bufs+sort_length; } @@ -879,7 +879,7 @@ static int NEAR_F write_merge_key(MARIA_SORT_PARAM *info __attribute__((unused)) IO_CACHE *to_file, char* key, uint sort_length, uint count) { - return my_b_write(to_file,(byte*) key,(uint) sort_length*count); + return my_b_write(to_file,(uchar*) key,(uint) sort_length*count); } /* @@ -889,14 +889,14 @@ static int NEAR_F write_merge_key(MARIA_SORT_PARAM *info __attribute__((unused)) static int NEAR_F merge_buffers(MARIA_SORT_PARAM *info, uint keys, IO_CACHE *from_file, - IO_CACHE *to_file, byte **sort_keys, BUFFPEK *lastbuff, + IO_CACHE *to_file, uchar **sort_keys, BUFFPEK *lastbuff, BUFFPEK *Fb, BUFFPEK *Tb) { int error; uint sort_length,maxcount; ha_rows count; my_off_t to_start_filepos; - byte *strpos; + uchar *strpos; BUFFPEK *buffpek,**refpek; QUEUE queue; volatile int *killed= _ma_killed_ptr(info->sort_info->param); @@ -907,11 +907,11 @@ merge_buffers(MARIA_SORT_PARAM *info, uint keys, IO_CACHE *from_file, LINT_INIT(to_start_filepos); if (to_file) to_start_filepos=my_b_tell(to_file); - strpos= (byte*) sort_keys; + strpos= (uchar*) sort_keys; sort_length=info->key_length; if (init_queue(&queue,(uint) (Tb-Fb)+1,offsetof(BUFFPEK,key),0, - (int (*)(void*, byte *,byte*)) info->key_cmp, + (int (*)(void*, uchar *,uchar*)) info->key_cmp, (void*) info)) DBUG_RETURN(1); /* purecov: inspected */ @@ -938,7 +938,7 @@ merge_buffers(MARIA_SORT_PARAM *info, uint keys, IO_CACHE *from_file, buffpek=(BUFFPEK*) queue_top(&queue); if (to_file) { - if (info->write_key(info,to_file,(byte*) buffpek->key, + if (info->write_key(info,to_file,(uchar*) buffpek->key, (uint) sort_length,1)) { error=1; goto err; /* purecov: inspected */ @@ -956,7 +956,7 @@ merge_buffers(MARIA_SORT_PARAM *info, uint keys, IO_CACHE *from_file, { if (!(error=(int) info->read_to_buffer(from_file,buffpek,sort_length))) { - byte *base= buffpek->base; + uchar *base= buffpek->base; uint max_keys=buffpek->max_keys; VOID(queue_remove(&queue,0)); @@ -988,13 +988,13 @@ merge_buffers(MARIA_SORT_PARAM *info, uint keys, IO_CACHE *from_file, } } buffpek=(BUFFPEK*) queue_top(&queue); - buffpek->base= (byte*) sort_keys; + buffpek->base= (uchar*) sort_keys; buffpek->max_keys=keys; do { if (to_file) { - if (info->write_key(info,to_file,(byte*) buffpek->key, + if (info->write_key(info,to_file,(uchar*) buffpek->key, sort_length,buffpek->mem_count)) { error=1; goto err; /* purecov: inspected */ @@ -1002,13 +1002,13 @@ merge_buffers(MARIA_SORT_PARAM *info, uint keys, IO_CACHE *from_file, } else { - register byte *end; + register uchar *end; strpos= buffpek->key; for (end= strpos+buffpek->mem_count*sort_length; strpos != end ; strpos+=sort_length) { - if ((*info->key_write)(info, (byte*) strpos)) + if ((*info->key_write)(info, (uchar*) strpos)) { error=1; goto err; /* purecov: inspected */ } @@ -1030,7 +1030,7 @@ err: /* Do a merge to output-file (save only positions) */ static int NEAR_F -merge_index(MARIA_SORT_PARAM *info, uint keys, byte **sort_keys, +merge_index(MARIA_SORT_PARAM *info, uint keys, uchar **sort_keys, BUFFPEK *buffpek, int maxbuffer, IO_CACHE *tempfile) { DBUG_ENTER("merge_index"); @@ -1047,7 +1047,7 @@ static int flush_maria_ft_buf(MARIA_SORT_PARAM *info) if (info->sort_info->ft_buf) { err=_ma_sort_ft_buf_flush(info); - my_free((gptr)info->sort_info->ft_buf, MYF(0)); + my_free((uchar*)info->sort_info->ft_buf, MYF(0)); info->sort_info->ft_buf=0; } return err; diff --git a/storage/maria/ma_sp_defs.h b/storage/maria/ma_sp_defs.h index a2870bfa062..a70695bea3a 100644 --- a/storage/maria/ma_sp_defs.h +++ b/storage/maria/ma_sp_defs.h @@ -40,8 +40,8 @@ enum wkbByteOrder wkbNDR = 1 /* Little Endian */ }; -uint _ma_sp_make_key(register MARIA_HA *info, uint keynr, byte *key, - const byte *record, my_off_t filepos); +uint _ma_sp_make_key(register MARIA_HA *info, uint keynr, uchar *key, + const uchar *record, my_off_t filepos); #endif /*HAVE_SPATIAL*/ #endif /* _SP_DEFS_H */ diff --git a/storage/maria/ma_sp_key.c b/storage/maria/ma_sp_key.c index 06769e97d30..1ea9b410ab6 100644 --- a/storage/maria/ma_sp_key.c +++ b/storage/maria/ma_sp_key.c @@ -31,25 +31,25 @@ static int sp_get_geometry_mbr(uchar *(*wkb), uchar *end, uint n_dims, double *mbr, int top); static int sp_mbr_from_wkb(uchar (*wkb), uint size, uint n_dims, double *mbr); -static void get_double(double *d, const byte *pos) +static void get_double(double *d, const uchar *pos) { float8get(*d, pos); } -uint _ma_sp_make_key(register MARIA_HA *info, uint keynr, byte *key, - const byte *record, my_off_t filepos) +uint _ma_sp_make_key(register MARIA_HA *info, uint keynr, uchar *key, + const uchar *record, my_off_t filepos) { HA_KEYSEG *keyseg; MARIA_KEYDEF *keyinfo = &info->s->keyinfo[keynr]; uint len = 0; - byte *pos; + uchar *pos; uint dlen; uchar *dptr; double mbr[SPDIMS * 2]; uint i; keyseg = &keyinfo->seg[-1]; - pos = (byte*)record + keyseg->start; + pos = (uchar*)record + keyseg->start; dlen = _ma_calc_blob_length(keyseg->bit_start, pos); memcpy_fixed(&dptr, pos + keyseg->bit_start, sizeof(char*)); @@ -64,7 +64,7 @@ uint _ma_sp_make_key(register MARIA_HA *info, uint keynr, byte *key, { uint length = keyseg->length; - pos = ((byte*)mbr) + keyseg->start; + pos = ((uchar*)mbr) + keyseg->start; if (keyseg->flag & HA_SWAP_KEY) { #ifdef HAVE_ISNAN @@ -100,7 +100,7 @@ uint _ma_sp_make_key(register MARIA_HA *info, uint keynr, byte *key, } else { - memcpy((byte*)key, pos, length); + memcpy((uchar*)key, pos, length); key += keyseg->length; } len += keyseg->length; @@ -141,7 +141,7 @@ static int sp_add_point_to_mbr(uchar *(*wkb), uchar *end, uint n_dims, { if ((*wkb) > end - 8) return -1; - get_double(&ord, (const byte*) *wkb); + get_double(&ord, (const uchar*) *wkb); (*wkb)+= 8; if (ord < *mbr) float8store((char*) mbr, ord); diff --git a/storage/maria/ma_static.c b/storage/maria/ma_static.c index 16bf0eca935..41b202491a7 100644 --- a/storage/maria/ma_static.c +++ b/storage/maria/ma_static.c @@ -57,7 +57,7 @@ PAGECACHE *maria_log_pagecache= &maria_log_pagecache_var; TRN dummy_transaction_object; /* Enough for comparing if number is zero */ -byte maria_zero_string[]= {0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0}; +uchar maria_zero_string[]= {0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0}; /* read_vec[] is used for converting between P_READ_KEY.. and SEARCH_ diff --git a/storage/maria/ma_statrec.c b/storage/maria/ma_statrec.c index 8ca3a5e989d..b04b858c685 100644 --- a/storage/maria/ma_statrec.c +++ b/storage/maria/ma_statrec.c @@ -18,9 +18,9 @@ #include "maria_def.h" -my_bool _ma_write_static_record(MARIA_HA *info, const byte *record) +my_bool _ma_write_static_record(MARIA_HA *info, const uchar *record) { - byte temp[8]; /* max pointer length */ + uchar temp[8]; /* max pointer length */ if (info->s->state.dellink != HA_OFFSET_ERROR && !info->append_insert_at_end) { @@ -48,14 +48,14 @@ my_bool _ma_write_static_record(MARIA_HA *info, const byte *record) } if (info->opt_flag & WRITE_CACHE_USED) { /* Cash in use */ - if (my_b_write(&info->rec_cache, (byte*) record, + if (my_b_write(&info->rec_cache, (uchar*) record, info->s->base.reclength)) goto err; if (info->s->base.pack_reclength != info->s->base.reclength) { uint length=info->s->base.pack_reclength - info->s->base.reclength; bzero((char*) temp,length); - if (my_b_write(&info->rec_cache, (byte*) temp,length)) + if (my_b_write(&info->rec_cache, (uchar*) temp,length)) goto err; } } @@ -70,7 +70,7 @@ my_bool _ma_write_static_record(MARIA_HA *info, const byte *record) { uint length=info->s->base.pack_reclength - info->s->base.reclength; bzero((char*) temp,length); - if (info->s->file_write(info, (byte*) temp,length, + if (info->s->file_write(info, (uchar*) temp,length, info->state->data_file_length+ info->s->base.reclength, info->s->write_flag)) @@ -86,8 +86,8 @@ my_bool _ma_write_static_record(MARIA_HA *info, const byte *record) } my_bool _ma_update_static_record(MARIA_HA *info, MARIA_RECORD_POS pos, - const byte *oldrec __attribute__ ((unused)), - const byte *record) + const uchar *oldrec __attribute__ ((unused)), + const uchar *record) { info->rec_cache.seek_not_done=1; /* We have done a seek */ return (info->s->file_write(info, @@ -98,9 +98,9 @@ my_bool _ma_update_static_record(MARIA_HA *info, MARIA_RECORD_POS pos, my_bool _ma_delete_static_record(MARIA_HA *info, - const byte *record __attribute__ ((unused))) + const uchar *record __attribute__ ((unused))) { - byte temp[9]; /* 1+sizeof(uint32) */ + uchar temp[9]; /* 1+sizeof(uint32) */ info->state->del++; info->state->empty+=info->s->base.pack_reclength; temp[0]= '\0'; /* Mark that record is deleted */ @@ -113,7 +113,7 @@ my_bool _ma_delete_static_record(MARIA_HA *info, my_bool _ma_cmp_static_record(register MARIA_HA *info, - register const byte *old) + register const uchar *old) { DBUG_ENTER("_ma_cmp_static_record"); @@ -137,7 +137,7 @@ my_bool _ma_cmp_static_record(register MARIA_HA *info, info->cur_row.lastpos, MYF(MY_NABP))) DBUG_RETURN(1); - if (memcmp((byte*) info->rec_buff, (byte*) old, + if (memcmp((uchar*) info->rec_buff, (uchar*) old, (uint) info->s->base.reclength)) { DBUG_DUMP("read",old,info->s->base.reclength); @@ -151,7 +151,7 @@ my_bool _ma_cmp_static_record(register MARIA_HA *info, my_bool _ma_cmp_static_unique(MARIA_HA *info, MARIA_UNIQUEDEF *def, - const byte *record, MARIA_RECORD_POS pos) + const uchar *record, MARIA_RECORD_POS pos) { DBUG_ENTER("_ma_cmp_static_unique"); @@ -159,7 +159,7 @@ my_bool _ma_cmp_static_unique(MARIA_HA *info, MARIA_UNIQUEDEF *def, if (info->s->file_read(info, (char*) info->rec_buff, info->s->base.reclength, pos, MYF(MY_NABP))) DBUG_RETURN(1); - DBUG_RETURN(_ma_unique_comp(def, record, (byte*) info->rec_buff, + DBUG_RETURN(_ma_unique_comp(def, record, (uchar*) info->rec_buff, def->null_are_equal)); } @@ -173,7 +173,7 @@ my_bool _ma_cmp_static_unique(MARIA_HA *info, MARIA_UNIQUEDEF *def, -1 on read-error or locking-error */ -int _ma_read_static_record(register MARIA_HA *info, register byte *record, +int _ma_read_static_record(register MARIA_HA *info, register uchar *record, MARIA_RECORD_POS pos) { int error; @@ -207,7 +207,7 @@ int _ma_read_static_record(register MARIA_HA *info, register byte *record, -int _ma_read_rnd_static_record(MARIA_HA *info, byte *buf, +int _ma_read_rnd_static_record(MARIA_HA *info, uchar *buf, MARIA_RECORD_POS filepos, my_bool skip_deleted_blocks) { @@ -275,11 +275,11 @@ int _ma_read_rnd_static_record(MARIA_HA *info, byte *buf, } /* Read record with cacheing */ - error=my_b_read(&info->rec_cache,(byte*) buf,share->base.reclength); + error=my_b_read(&info->rec_cache,(uchar*) buf,share->base.reclength); if (info->s->base.pack_reclength != info->s->base.reclength && !error) { char tmp[8]; /* Skill fill bytes */ - error=my_b_read(&info->rec_cache,(byte*) tmp, + error=my_b_read(&info->rec_cache,(uchar*) tmp, info->s->base.pack_reclength - info->s->base.reclength); } if (locked) diff --git a/storage/maria/ma_test1.c b/storage/maria/ma_test1.c index 028e02ab9d1..35d654bbb45 100644 --- a/storage/maria/ma_test1.c +++ b/storage/maria/ma_test1.c @@ -128,7 +128,7 @@ static int run_test(const char *filename) } keyinfo[0].flag = (uint8) (pack_keys | unique_key); - bzero((byte*) flags,sizeof(flags)); + bzero((uchar*) flags,sizeof(flags)); if (opt_unique) { uint start; @@ -575,22 +575,22 @@ static struct my_option my_long_options[] = 0, 0, 0, GET_STR, REQUIRED_ARG, 0, 0, 0, 0, 0, 0}, #endif {"delete-rows", 'd', "Abort after this many rows has been deleted", - (gptr*) &remove_count, (gptr*) &remove_count, 0, GET_UINT, REQUIRED_ARG, + (uchar**) &remove_count, (uchar**) &remove_count, 0, GET_UINT, REQUIRED_ARG, 1000, 0, 0, 0, 0, 0}, {"help", '?', "Display help and exit", 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, - {"insert-rows", 'i', "Undocumented", (gptr*) &insert_count, - (gptr*) &insert_count, 0, GET_UINT, REQUIRED_ARG, 1000, 0, 0, 0, 0, 0}, + {"insert-rows", 'i', "Undocumented", (uchar**) &insert_count, + (uchar**) &insert_count, 0, GET_UINT, REQUIRED_ARG, 1000, 0, 0, 0, 0, 0}, {"key-alpha", 'a', "Use a key of type HA_KEYTYPE_TEXT", 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, {"key-binary-pack", 'B', "Undocumented", 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, {"key-blob", 'b', "Undocumented", 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, - {"key-cache", 'K', "Undocumented", (gptr*) &pagecacheing, - (gptr*) &pagecacheing, 0, GET_BOOL, NO_ARG, 0, 0, 0, 0, 0, 0}, - {"key-length", 'k', "Undocumented", (gptr*) &key_length, (gptr*) &key_length, - 0, GET_UINT, REQUIRED_ARG, 6, 0, 0, 0, 0, 0}, + {"key-cache", 'K', "Undocumented", (uchar**) &pagecacheing, + (uchar**) &pagecacheing, 0, GET_BOOL, NO_ARG, 0, 0, 0, 0, 0, 0}, + {"key-length", 'k', "Undocumented", (uchar**) &key_length, + (uchar**) &key_length, 0, GET_UINT, REQUIRED_ARG, 6, 0, 0, 0, 0, 0}, {"key-multiple", 'm', "Undocumented", 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, {"key-prefix_pack", 'P', "Undocumented", @@ -600,29 +600,31 @@ static struct my_option my_long_options[] = {"key-varchar", 'w', "Test VARCHAR keys", 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, {"null-fields", 'N', "Define fields with NULL", - (gptr*) &null_fields, (gptr*) &null_fields, 0, GET_BOOL, NO_ARG, + (uchar**) &null_fields, (uchar**) &null_fields, 0, GET_BOOL, NO_ARG, 0, 0, 0, 0, 0, 0}, {"row-fixed-size", 'S', "Fixed size records", 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, {"rows-in-block", 'M', "Store rows in block format", 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, - {"row-pointer-size", 'R', "Undocumented", (gptr*) &rec_pointer_size, - (gptr*) &rec_pointer_size, 0, GET_INT, REQUIRED_ARG, 0, 0, 0, 0, 0, 0}, + {"row-pointer-size", 'R', "Undocumented", (uchar**) &rec_pointer_size, + (uchar**) &rec_pointer_size, 0, GET_INT, REQUIRED_ARG, 0, 0, 0, 0, 0, 0}, {"silent", 's', "Undocumented", - (gptr*) &silent, (gptr*) &silent, 0, GET_BOOL, NO_ARG, 0, 0, 0, 0, 0, 0}, - {"skip-delete", 'U', "Don't test deletes", (gptr*) &skip_delete, - (gptr*) &skip_delete, 0, GET_BOOL, NO_ARG, 0, 0, 0, 0, 0, 0}, - {"skip-update", 'D', "Don't test updates", (gptr*) &skip_update, - (gptr*) &skip_update, 0, GET_BOOL, NO_ARG, 0, 0, 0, 0, 0, 0}, - {"transactional", 'T', "Test in transactional mode. (Only works with block format)", - (gptr*) &transactional, (gptr*) &transactional, 0, GET_BOOL, NO_ARG, + (uchar**) &silent, (uchar**) &silent, 0, GET_BOOL, NO_ARG, 0, 0, 0, 0, + 0, 0}, + {"skip-delete", 'U', "Don't test deletes", (uchar**) &skip_delete, + (uchar**) &skip_delete, 0, GET_BOOL, NO_ARG, 0, 0, 0, 0, 0, 0}, + {"skip-update", 'D', "Don't test updates", (uchar**) &skip_update, + (uchar**) &skip_update, 0, GET_BOOL, NO_ARG, 0, 0, 0, 0, 0, 0}, + {"transactional", 'T', + "Test in transactional mode. (Only works with block format)", + (uchar**) &transactional, (uchar**) &transactional, 0, GET_BOOL, NO_ARG, 0, 0, 0, 0, 0, 0}, - {"unique", 'C', "Undocumented", (gptr*) &opt_unique, (gptr*) &opt_unique, 0, - GET_BOOL, NO_ARG, 0, 0, 0, 0, 0, 0}, - {"update-rows", 'u', "Undocumented", (gptr*) &update_count, - (gptr*) &update_count, 0, GET_UINT, REQUIRED_ARG, 1000, 0, 0, 0, 0, 0}, - {"verbose", 'v', "Be more verbose", (gptr*) &verbose, (gptr*) &verbose, 0, - GET_BOOL, NO_ARG, 0, 0, 0, 0, 0, 0}, + {"unique", 'C', "Undocumented", (uchar**) &opt_unique, + (uchar**) &opt_unique, 0, GET_BOOL, NO_ARG, 0, 0, 0, 0, 0, 0}, + {"update-rows", 'u', "Undocumented", (uchar**) &update_count, + (uchar**) &update_count, 0, GET_UINT, REQUIRED_ARG, 1000, 0, 0, 0, 0, 0}, + {"verbose", 'v', "Be more verbose", (uchar**) &verbose, + (uchar**) &verbose, 0, GET_BOOL, NO_ARG, 0, 0, 0, 0, 0, 0}, {"version", 'V', "Print version number and exit", 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, { 0, 0, 0, 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0} diff --git a/storage/maria/ma_test2.c b/storage/maria/ma_test2.c index bbbb4fca1bf..dd5596c4d5c 100644 --- a/storage/maria/ma_test2.c +++ b/storage/maria/ma_test2.c @@ -38,7 +38,7 @@ static void get_options(int argc, char *argv[]); static uint rnd(uint max_value); -static void fix_length(byte *record,uint length); +static void fix_length(uchar *record,uint length); static void put_blob_in_record(char *blob_pos,char **blob_buffer, ulong *length); static void copy_key(struct st_maria_info *info,uint inx, @@ -496,8 +496,8 @@ int main(int argc, char *argv[]) bcmp(read_record2,read_record3,reclength)) { printf("Can't find last record\n"); - DBUG_DUMP("record2",(byte*) read_record2,reclength); - DBUG_DUMP("record3",(byte*) read_record3,reclength); + DBUG_DUMP("record2",(uchar*) read_record2,reclength); + DBUG_DUMP("record3",(uchar*) read_record3,reclength); goto end; } ant=1; @@ -1050,7 +1050,7 @@ static uint rnd(uint max_value) /* Create a variable length record */ -static void fix_length(byte *rec, uint length) +static void fix_length(uchar *rec, uint length) { bmove(rec+STANDARD_LENGTH, "0123456789012345678901234567890123456789012345678901234567890", diff --git a/storage/maria/ma_test3.c b/storage/maria/ma_test3.c index dcc0ac2e013..948ba09aa24 100644 --- a/storage/maria/ma_test3.c +++ b/storage/maria/ma_test3.c @@ -242,7 +242,7 @@ int test_read(MARIA_HA *file,int id) for (i=0 ; i < 100 ; i++) { find=rnd(100000); - if (!maria_rkey(file,record.id,1,(byte*) &find, + if (!maria_rkey(file,record.id,1,(uchar*) &find, sizeof(find),HA_READ_KEY_EXACT)) found++; else @@ -425,7 +425,7 @@ int test_update(MARIA_HA *file,int id,int lock_type) { tmp=rnd(100000); int4store(find,tmp); - if (!maria_rkey(file,record.id,1,(byte*) find, + if (!maria_rkey(file,record.id,1,(uchar*) find, sizeof(find),HA_READ_KEY_EXACT)) found++; else diff --git a/storage/maria/ma_unique.c b/storage/maria/ma_unique.c index 06d29b9b037..3ab717887c7 100644 --- a/storage/maria/ma_unique.c +++ b/storage/maria/ma_unique.c @@ -18,12 +18,12 @@ #include "maria_def.h" #include -my_bool _ma_check_unique(MARIA_HA *info, MARIA_UNIQUEDEF *def, byte *record, +my_bool _ma_check_unique(MARIA_HA *info, MARIA_UNIQUEDEF *def, uchar *record, ha_checksum unique_hash, my_off_t disk_pos) { my_off_t lastpos=info->cur_row.lastpos; MARIA_KEYDEF *key= &info->s->keyinfo[def->key]; - byte *key_buff= info->lastkey2; + uchar *key_buff= info->lastkey2; DBUG_ENTER("_ma_check_unique"); DBUG_PRINT("enter",("unique_hash: %lu", (ulong) unique_hash)); @@ -76,9 +76,9 @@ my_bool _ma_check_unique(MARIA_HA *info, MARIA_UNIQUEDEF *def, byte *record, Add support for bit fields */ -ha_checksum _ma_unique_hash(MARIA_UNIQUEDEF *def, const byte *record) +ha_checksum _ma_unique_hash(MARIA_UNIQUEDEF *def, const uchar *record) { - const byte *pos, *end; + const uchar *pos, *end; ha_checksum crc= 0; ulong seed1=0, seed2= 4; HA_KEYSEG *keyseg; @@ -114,7 +114,7 @@ ha_checksum _ma_unique_hash(MARIA_UNIQUEDEF *def, const byte *record) else if (keyseg->flag & HA_BLOB_PART) { uint tmp_length= _ma_calc_blob_length(keyseg->bit_start,pos); - memcpy_fixed((byte*) &pos,pos+keyseg->bit_start,sizeof(char*)); + memcpy_fixed((uchar*) &pos,pos+keyseg->bit_start,sizeof(char*)); if (!length || length > tmp_length) length=tmp_length; /* The whole blob */ } @@ -148,10 +148,10 @@ ha_checksum _ma_unique_hash(MARIA_UNIQUEDEF *def, const byte *record) 1 Rows are different */ -my_bool _ma_unique_comp(MARIA_UNIQUEDEF *def, const byte *a, const byte *b, +my_bool _ma_unique_comp(MARIA_UNIQUEDEF *def, const uchar *a, const uchar *b, my_bool null_are_equal) { - const byte *pos_a, *pos_b, *end; + const uchar *pos_a, *pos_b, *end; HA_KEYSEG *keyseg; for (keyseg=def->seg ; keyseg < def->end ; keyseg++) @@ -209,8 +209,8 @@ my_bool _ma_unique_comp(MARIA_UNIQUEDEF *def, const byte *a, const byte *b, set_if_smaller(a_length, keyseg->length); set_if_smaller(b_length, keyseg->length); } - memcpy_fixed((byte*) &pos_a,pos_a+keyseg->bit_start,sizeof(char*)); - memcpy_fixed((byte*) &pos_b,pos_b+keyseg->bit_start,sizeof(char*)); + memcpy_fixed((uchar*) &pos_a,pos_a+keyseg->bit_start,sizeof(char*)); + memcpy_fixed((uchar*) &pos_b,pos_b+keyseg->bit_start,sizeof(char*)); } if (type == HA_KEYTYPE_TEXT || type == HA_KEYTYPE_VARTEXT1 || type == HA_KEYTYPE_VARTEXT2) diff --git a/storage/maria/ma_update.c b/storage/maria/ma_update.c index 737c7c909b4..4d21167535d 100644 --- a/storage/maria/ma_update.c +++ b/storage/maria/ma_update.c @@ -18,12 +18,12 @@ #include "ma_fulltext.h" #include "ma_rt_index.h" -int maria_update(register MARIA_HA *info, const byte *oldrec, byte *newrec) +int maria_update(register MARIA_HA *info, const uchar *oldrec, uchar *newrec) { int flag,key_changed,save_errno; reg3 my_off_t pos; uint i; - byte old_key[HA_MAX_KEY_BUFF],*new_key; + uchar old_key[HA_MAX_KEY_BUFF],*new_key; bool auto_key_changed=0; ulonglong changed; MARIA_SHARE *share=info->s; diff --git a/storage/maria/ma_write.c b/storage/maria/ma_write.c index a87e2d76fc7..cb15280fc6e 100644 --- a/storage/maria/ma_write.c +++ b/storage/maria/ma_write.c @@ -23,24 +23,24 @@ /* Functions declared in this file */ static int w_search(MARIA_HA *info,MARIA_KEYDEF *keyinfo, - uint comp_flag, byte *key, - uint key_length, my_off_t pos, byte *father_buff, - byte *father_keypos, my_off_t father_page, + uint comp_flag, uchar *key, + uint key_length, my_off_t pos, uchar *father_buff, + uchar *father_keypos, my_off_t father_page, my_bool insert_last); -static int _ma_balance_page(MARIA_HA *info,MARIA_KEYDEF *keyinfo,byte *key, - byte *curr_buff,byte *father_buff, - byte *father_keypos,my_off_t father_page); -static byte *_ma_find_last_pos(MARIA_KEYDEF *keyinfo, byte *page, - byte *key, uint *return_key_length, - byte **after_key); -int _ma_ck_write_tree(register MARIA_HA *info, uint keynr,byte *key, +static int _ma_balance_page(MARIA_HA *info,MARIA_KEYDEF *keyinfo,uchar *key, + uchar *curr_buff,uchar *father_buff, + uchar *father_keypos,my_off_t father_page); +static uchar *_ma_find_last_pos(MARIA_KEYDEF *keyinfo, uchar *page, + uchar *key, uint *return_key_length, + uchar **after_key); +int _ma_ck_write_tree(register MARIA_HA *info, uint keynr,uchar *key, uint key_length); -int _ma_ck_write_btree(register MARIA_HA *info, uint keynr,byte *key, +int _ma_ck_write_btree(register MARIA_HA *info, uint keynr,uchar *key, uint key_length); MARIA_RECORD_POS _ma_write_init_default(MARIA_HA *info, - const byte *record + const uchar *record __attribute__((unused))) { return ((info->s->state.dellink != HA_OFFSET_ERROR && @@ -57,13 +57,13 @@ my_bool _ma_write_abort_default(MARIA_HA *info __attribute__((unused))) /* Write new record to a table */ -int maria_write(MARIA_HA *info, byte *record) +int maria_write(MARIA_HA *info, uchar *record) { MARIA_SHARE *share=info->s; uint i; int save_errno; MARIA_RECORD_POS filepos; - byte *buff; + uchar *buff; my_bool lock_tree= share->concurrent_insert; my_bool fatal_error; DBUG_ENTER("maria_write"); @@ -271,7 +271,7 @@ err2: /* Write one key to btree */ -int _ma_ck_write(MARIA_HA *info, uint keynr, byte *key, uint key_length) +int _ma_ck_write(MARIA_HA *info, uint keynr, uchar *key, uint key_length) { DBUG_ENTER("_ma_ck_write"); @@ -290,7 +290,7 @@ int _ma_ck_write(MARIA_HA *info, uint keynr, byte *key, uint key_length) * Normal insert code * **********************************************************************/ -int _ma_ck_write_btree(register MARIA_HA *info, uint keynr, byte *key, +int _ma_ck_write_btree(register MARIA_HA *info, uint keynr, uchar *key, uint key_length) { int error; @@ -317,7 +317,7 @@ int _ma_ck_write_btree(register MARIA_HA *info, uint keynr, byte *key, if (!error) error= _ma_ft_convert_to_ft2(info, keynr, key); delete_dynamic(info->ft1_to_ft2); - my_free((gptr)info->ft1_to_ft2, MYF(0)); + my_free((uchar*)info->ft1_to_ft2, MYF(0)); info->ft1_to_ft2=0; } DBUG_RETURN(error); @@ -325,7 +325,7 @@ int _ma_ck_write_btree(register MARIA_HA *info, uint keynr, byte *key, int _ma_ck_real_write_btree(MARIA_HA *info, MARIA_KEYDEF *keyinfo, - byte *key, uint key_length, my_off_t *root, + uchar *key, uint key_length, my_off_t *root, uint comp_flag) { int error; @@ -333,7 +333,7 @@ int _ma_ck_real_write_btree(MARIA_HA *info, MARIA_KEYDEF *keyinfo, /* key_length parameter is used only if comp_flag is SEARCH_FIND */ if (*root == HA_OFFSET_ERROR || (error=w_search(info, keyinfo, comp_flag, key, key_length, - *root, (byte*) 0, (byte*) 0, + *root, (uchar*) 0, (uchar*) 0, (my_off_t) 0, 1)) > 0) error= _ma_enlarge_root(info,keyinfo,key,root); DBUG_RETURN(error); @@ -342,7 +342,7 @@ int _ma_ck_real_write_btree(MARIA_HA *info, MARIA_KEYDEF *keyinfo, /* Make a new root with key as only pointer */ -int _ma_enlarge_root(MARIA_HA *info, MARIA_KEYDEF *keyinfo, byte *key, +int _ma_enlarge_root(MARIA_HA *info, MARIA_KEYDEF *keyinfo, uchar *key, my_off_t *root) { uint t_length,nod_flag; @@ -352,8 +352,8 @@ int _ma_enlarge_root(MARIA_HA *info, MARIA_KEYDEF *keyinfo, byte *key, nod_flag= (*root != HA_OFFSET_ERROR) ? share->base.key_reflength : 0; _ma_kpointer(info,info->buff+2,*root); /* if nod */ - t_length=(*keyinfo->pack_key)(keyinfo,nod_flag,(byte*) 0, - (byte*) 0, (byte*) 0, key,&s_temp); + t_length=(*keyinfo->pack_key)(keyinfo,nod_flag,(uchar*) 0, + (uchar*) 0, (uchar*) 0, key,&s_temp); maria_putint(info->buff,t_length+2+nod_flag,nod_flag); (*keyinfo->store_key)(keyinfo,info->buff+2+nod_flag,&s_temp); info->keyread_buff_used=info->page_changed=1; /* info->buff is used */ @@ -372,21 +372,21 @@ int _ma_enlarge_root(MARIA_HA *info, MARIA_KEYDEF *keyinfo, byte *key, */ static int w_search(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, - uint comp_flag, byte *key, uint key_length, my_off_t page, - byte *father_buff, byte *father_keypos, + uint comp_flag, uchar *key, uint key_length, my_off_t page, + uchar *father_buff, uchar *father_keypos, my_off_t father_page, my_bool insert_last) { int error,flag; uint nod_flag, search_key_length; - byte *temp_buff,*keypos; - byte keybuff[HA_MAX_KEY_BUFF]; + uchar *temp_buff,*keypos; + uchar keybuff[HA_MAX_KEY_BUFF]; my_bool was_last_key; my_off_t next_page, dup_key_pos; DBUG_ENTER("w_search"); DBUG_PRINT("enter",("page: %ld", (long) page)); search_key_length= (comp_flag & SEARCH_FIND) ? key_length : USE_WHOLE_KEY; - if (!(temp_buff= (byte*) my_alloca((uint) keyinfo->block_length+ + if (!(temp_buff= (uchar*) my_alloca((uint) keyinfo->block_length+ HA_MAX_KEY_BUFF*2))) DBUG_RETURN(-1); if (!_ma_fetch_keypage(info,keyinfo,page,DFLT_INIT_HITS,temp_buff,0)) @@ -436,14 +436,14 @@ static int w_search(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, ft_intXstore(keypos, subkeys); if (!error) error= _ma_write_keypage(info,keyinfo,page,DFLT_INIT_HITS,temp_buff); - my_afree((byte*) temp_buff); + my_afree((uchar*) temp_buff); DBUG_RETURN(error); } } else /* not HA_FULLTEXT, normal HA_NOSAME key */ { info->dup_key_pos= dup_key_pos; - my_afree((byte*) temp_buff); + my_afree((uchar*) temp_buff); my_errno=HA_ERR_FOUND_DUPP_KEY; DBUG_RETURN(-1); } @@ -462,10 +462,10 @@ static int w_search(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, if (_ma_write_keypage(info,keyinfo,page,DFLT_INIT_HITS,temp_buff)) goto err; } - my_afree((byte*) temp_buff); + my_afree((uchar*) temp_buff); DBUG_RETURN(error); err: - my_afree((byte*) temp_buff); + my_afree((uchar*) temp_buff); DBUG_PRINT("exit",("Error: %d",my_errno)); DBUG_RETURN (-1); } /* w_search */ @@ -497,13 +497,13 @@ err: */ int _ma_insert(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, - byte *key, byte *anc_buff, byte *key_pos, byte *key_buff, - byte *father_buff, byte *father_key_pos, my_off_t father_page, + uchar *key, uchar *anc_buff, uchar *key_pos, uchar *key_buff, + uchar *father_buff, uchar *father_key_pos, my_off_t father_page, my_bool insert_last) { uint a_length,nod_flag; int t_length; - byte *endpos, *prev_key; + uchar *endpos, *prev_key; MARIA_KEY_PARAM s_temp; DBUG_ENTER("_ma_insert"); DBUG_PRINT("enter",("key_pos: 0x%lx", (ulong) key_pos)); @@ -513,16 +513,16 @@ int _ma_insert(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, nod_flag=_ma_test_if_nod(anc_buff); a_length= maria_data_on_page(anc_buff); endpos= anc_buff+ a_length; - prev_key=(key_pos == anc_buff+2+nod_flag ? (byte*) 0 : key_buff); + prev_key=(key_pos == anc_buff+2+nod_flag ? (uchar*) 0 : key_buff); t_length=(*keyinfo->pack_key)(keyinfo,nod_flag, - (key_pos == endpos ? (byte*) 0 : key_pos), + (key_pos == endpos ? (uchar*) 0 : key_pos), prev_key, prev_key, key,&s_temp); #ifndef DBUG_OFF if (key_pos != anc_buff+2+nod_flag && (keyinfo->flag & (HA_BINARY_PACK_KEY | HA_PACK_KEY))) { - DBUG_DUMP("prev_key",(byte*) key_buff, _ma_keylength(keyinfo,key_buff)); + DBUG_DUMP("prev_key",(uchar*) key_buff, _ma_keylength(keyinfo,key_buff)); } if (keyinfo->flag & HA_PACK_KEY) { @@ -540,7 +540,7 @@ int _ma_insert(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, my_errno=HA_ERR_CRASHED; DBUG_RETURN(-1); } - bmove_upp((byte*) endpos+t_length,(byte*) endpos,(uint) (endpos-key_pos)); + bmove_upp((uchar*) endpos+t_length,(uchar*) endpos,(uint) (endpos-key_pos)); } else { @@ -567,7 +567,7 @@ int _ma_insert(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, Let's consider converting. We'll compare 'key' and the first key at anc_buff */ - byte *a=key, *b=anc_buff+2+nod_flag; + uchar *a=key, *b=anc_buff+2+nod_flag; uint alen, blen, ft2len=info->s->ft2_keyinfo.keylength; /* the very first key on the page is always unpacked */ DBUG_ASSERT((*b & 128) == 0); @@ -621,16 +621,16 @@ int _ma_insert(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, /* split a full page in two and assign emerging item to key */ int _ma_split_page(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, - byte *key, byte *buff, byte *key_buff, + uchar *key, uchar *buff, uchar *key_buff, my_bool insert_last_key) { uint length,a_length,key_ref_length,t_length,nod_flag,key_length; - byte *key_pos,*pos, *after_key; + uchar *key_pos,*pos, *after_key; my_off_t new_pos; MARIA_KEY_PARAM s_temp; DBUG_ENTER("maria_split_page"); LINT_INIT(after_key); - DBUG_DUMP("buff",(byte*) buff,maria_data_on_page(buff)); + DBUG_DUMP("buff",(uchar*) buff,maria_data_on_page(buff)); if (info->s->keyinfo+info->lastinx == keyinfo) info->page_changed=1; /* Info->buff is used */ @@ -654,7 +654,7 @@ int _ma_split_page(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, { DBUG_PRINT("test",("Splitting nod")); pos=key_pos-nod_flag; - memcpy((byte*) info->buff+2,(byte*) pos,(size_t) nod_flag); + memcpy((uchar*) info->buff+2,(uchar*) pos,(size_t) nod_flag); } /* Move middle item to key and pointer to new page */ @@ -666,18 +666,18 @@ int _ma_split_page(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, if (!(*keyinfo->get_key)(keyinfo,nod_flag,&key_pos,key_buff)) DBUG_RETURN(-1); - t_length=(*keyinfo->pack_key)(keyinfo,nod_flag,(byte *) 0, - (byte*) 0, (byte*) 0, + t_length=(*keyinfo->pack_key)(keyinfo,nod_flag,(uchar *) 0, + (uchar*) 0, (uchar*) 0, key_buff, &s_temp); length=(uint) ((buff+a_length)-key_pos); - memcpy((byte*) info->buff+key_ref_length+t_length,(byte*) key_pos, + memcpy((uchar*) info->buff+key_ref_length+t_length,(uchar*) key_pos, (size_t) length); (*keyinfo->store_key)(keyinfo,info->buff+key_ref_length,&s_temp); maria_putint(info->buff,length+t_length+key_ref_length,nod_flag); if (_ma_write_keypage(info,keyinfo,new_pos,DFLT_INIT_HITS,info->buff)) DBUG_RETURN(-1); - DBUG_DUMP("key",(byte*) key, _ma_keylength(keyinfo,key)); + DBUG_DUMP("key",(uchar*) key, _ma_keylength(keyinfo,key)); DBUG_RETURN(2); /* Middle key up */ } /* _ma_split_page */ @@ -690,12 +690,12 @@ int _ma_split_page(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, after_key will contain the position to where the next key starts */ -byte *_ma_find_half_pos(uint nod_flag, MARIA_KEYDEF *keyinfo, byte *page, - byte *key, uint *return_key_length, - byte **after_key) +uchar *_ma_find_half_pos(uint nod_flag, MARIA_KEYDEF *keyinfo, uchar *page, + uchar *key, uint *return_key_length, + uchar **after_key) { uint keys,length,key_ref_length; - byte *end,*lastpos; + uchar *end,*lastpos; DBUG_ENTER("_ma_find_half_pos"); key_ref_length=2+nod_flag; @@ -736,13 +736,13 @@ byte *_ma_find_half_pos(uint nod_flag, MARIA_KEYDEF *keyinfo, byte *page, key will contain the last key */ -static byte *_ma_find_last_pos(MARIA_KEYDEF *keyinfo, byte *page, - byte *key, uint *return_key_length, - byte **after_key) +static uchar *_ma_find_last_pos(MARIA_KEYDEF *keyinfo, uchar *page, + uchar *key, uint *return_key_length, + uchar **after_key) { uint keys,length,last_length,key_ref_length; - byte *end,*lastpos,*prevpos; - byte key_buff[HA_MAX_KEY_BUFF]; + uchar *end,*lastpos,*prevpos; + uchar key_buff[HA_MAX_KEY_BUFF]; DBUG_ENTER("_ma_find_last_pos"); key_ref_length=2; @@ -790,16 +790,16 @@ static byte *_ma_find_last_pos(MARIA_KEYDEF *keyinfo, byte *page, /* returns 0 if balance was done */ static int _ma_balance_page(register MARIA_HA *info, MARIA_KEYDEF *keyinfo, - byte *key, byte *curr_buff, byte *father_buff, - byte *father_key_pos, my_off_t father_page) + uchar *key, uchar *curr_buff, uchar *father_buff, + uchar *father_key_pos, my_off_t father_page) { my_bool right; uint k_length,father_length,father_keylength,nod_flag,curr_keylength, right_length,left_length,new_right_length,new_left_length,extra_length, length,keys; - byte *pos,*buff,*extra_buff; + uchar *pos,*buff,*extra_buff; my_off_t next_page,new_pos; - byte tmp_part_key[HA_MAX_KEY_BUFF]; + uchar tmp_part_key[HA_MAX_KEY_BUFF]; DBUG_ENTER("_ma_balance_page"); k_length=keyinfo->keylength; @@ -831,7 +831,7 @@ static int _ma_balance_page(register MARIA_HA *info, MARIA_KEYDEF *keyinfo, if (!_ma_fetch_keypage(info,keyinfo,next_page,DFLT_INIT_HITS,info->buff,0)) goto err; - DBUG_DUMP("next",(byte*) info->buff,maria_data_on_page(info->buff)); + DBUG_DUMP("next",(uchar*) info->buff,maria_data_on_page(info->buff)); /* Test if there is room to share keys */ @@ -850,23 +850,23 @@ static int _ma_balance_page(register MARIA_HA *info, MARIA_KEYDEF *keyinfo, if (left_length < new_left_length) { /* Move keys buff -> leaf */ pos=curr_buff+left_length; - memcpy((byte*) pos,(byte*) father_key_pos, (size_t) k_length); - memcpy((byte*) pos+k_length, (byte*) buff+2, + memcpy((uchar*) pos,(uchar*) father_key_pos, (size_t) k_length); + memcpy((uchar*) pos+k_length, (uchar*) buff+2, (size_t) (length=new_left_length - left_length - k_length)); pos=buff+2+length; - memcpy((byte*) father_key_pos,(byte*) pos,(size_t) k_length); - bmove((byte*) buff+2,(byte*) pos+k_length,new_right_length); + memcpy((uchar*) father_key_pos,(uchar*) pos,(size_t) k_length); + bmove((uchar*) buff+2,(uchar*) pos+k_length,new_right_length); } else { /* Move keys -> buff */ - bmove_upp((byte*) buff+new_right_length,(byte*) buff+right_length, + bmove_upp((uchar*) buff+new_right_length,(uchar*) buff+right_length, right_length-2); length=new_right_length-right_length-k_length; - memcpy((byte*) buff+2+length,father_key_pos,(size_t) k_length); + memcpy((uchar*) buff+2+length,father_key_pos,(size_t) k_length); pos=curr_buff+new_left_length; - memcpy((byte*) father_key_pos,(byte*) pos,(size_t) k_length); - memcpy((byte*) buff+2,(byte*) pos+k_length,(size_t) length); + memcpy((uchar*) father_key_pos,(uchar*) pos,(size_t) k_length); + memcpy((uchar*) buff+2,(uchar*) pos+k_length,(size_t) length); } if (_ma_write_keypage(info,keyinfo,next_page,DFLT_INIT_HITS,info->buff) || @@ -893,22 +893,22 @@ static int _ma_balance_page(register MARIA_HA *info, MARIA_KEYDEF *keyinfo, /* move first largest keys to new page */ pos=buff+right_length-extra_length; - memcpy((byte*) extra_buff+2,pos,(size_t) extra_length); + memcpy((uchar*) extra_buff+2,pos,(size_t) extra_length); /* Save new parting key */ memcpy(tmp_part_key, pos-k_length,k_length); /* Make place for new keys */ - bmove_upp((byte*) buff+new_right_length,(byte*) pos-k_length, + bmove_upp((uchar*) buff+new_right_length,(uchar*) pos-k_length, right_length-extra_length-k_length-2); /* Copy keys from left page */ pos= curr_buff+new_left_length; - memcpy((byte*) buff+2,(byte*) pos+k_length, + memcpy((uchar*) buff+2,(uchar*) pos+k_length, (size_t) (length=left_length-new_left_length-k_length)); /* Copy old parting key */ - memcpy((byte*) buff+2+length,father_key_pos,(size_t) k_length); + memcpy((uchar*) buff+2+length,father_key_pos,(size_t) k_length); /* Move new parting keys up to caller */ - memcpy((byte*) (right ? key : father_key_pos),pos,(size_t) k_length); - memcpy((byte*) (right ? father_key_pos : key),tmp_part_key, k_length); + memcpy((uchar*) (right ? key : father_key_pos),pos,(size_t) k_length); + memcpy((uchar*) (right ? father_key_pos : key),tmp_part_key, k_length); if ((new_pos= _ma_new(info,keyinfo,DFLT_INIT_HITS)) == HA_OFFSET_ERROR) goto err; @@ -935,7 +935,7 @@ typedef struct { } bulk_insert_param; -int _ma_ck_write_tree(register MARIA_HA *info, uint keynr, byte *key, +int _ma_ck_write_tree(register MARIA_HA *info, uint keynr, uchar *key, uint key_length) { int error; @@ -951,7 +951,7 @@ int _ma_ck_write_tree(register MARIA_HA *info, uint keynr, byte *key, /* typeof(_ma_keys_compare)=qsort_cmp2 */ -static int keys_compare(bulk_insert_param *param, byte *key1, byte *key2) +static int keys_compare(bulk_insert_param *param, uchar *key1, uchar *key2) { uint not_used[2]; return ha_key_cmp(param->info->s->keyinfo[param->keynr].seg, @@ -960,13 +960,13 @@ static int keys_compare(bulk_insert_param *param, byte *key1, byte *key2) } -static int keys_free(byte *key, TREE_FREE mode, bulk_insert_param *param) +static int keys_free(uchar *key, TREE_FREE mode, bulk_insert_param *param) { /* Probably I can use info->lastkey here, but I'm not sure, and to be safe I'd better use local lastkey. */ - byte lastkey[HA_MAX_KEY_BUFF]; + uchar lastkey[HA_MAX_KEY_BUFF]; uint keylen; MARIA_KEYDEF *keyinfo; diff --git a/storage/maria/maria_chk.c b/storage/maria/maria_chk.c index 0b82a71f736..f8a51507624 100644 --- a/storage/maria/maria_chk.c +++ b/storage/maria/maria_chk.c @@ -76,13 +76,13 @@ static void get_options(int *argc,char * * *argv); static void print_version(void); static void usage(void); static int maria_chk(HA_CHECK *param, char *filename); -static void descript(HA_CHECK *param, register MARIA_HA *info, my_string name); +static void descript(HA_CHECK *param, register MARIA_HA *info, char *name); static int maria_sort_records(HA_CHECK *param, register MARIA_HA *info, - my_string name, uint sort_key, + char *name, uint sort_key, my_bool write_info, my_bool update_index); static int sort_record_index(MARIA_SORT_PARAM *sort_param, MARIA_HA *info, MARIA_KEYDEF *keyinfo, - my_off_t page, byte *buff,uint sortkey, + my_off_t page, uchar *buff,uint sortkey, File new_file, my_bool update_index); HA_CHECK check_param; @@ -178,7 +178,7 @@ static struct my_option my_long_options[] = 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, {"character-sets-dir", OPT_CHARSETS_DIR, "Directory where character sets are.", - (gptr*) &charsets_dir, 0, 0, GET_STR, REQUIRED_ARG, 0, 0, 0, 0, 0, 0}, + (uchar**) &charsets_dir, 0, 0, GET_STR, REQUIRED_ARG, 0, 0, 0, 0, 0, 0}, {"check", 'c', "Check table for errors.", 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, @@ -198,8 +198,8 @@ static struct my_option my_long_options[] = 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, {"data-file-length", 'D', "Max length of data file (when recreating data-file when it's full).", - (gptr*) &check_param.max_data_file_length, - (gptr*) &check_param.max_data_file_length, + (uchar**) &check_param.max_data_file_length, + (uchar**) &check_param.max_data_file_length, 0, GET_LL, REQUIRED_ARG, 0, 0, 0, 0, 0, 0}, {"extend-check", 'e', "If used when checking a table, ensure that the table is 100 percent consistent, which will take a long time. If used when repairing a table, try to recover every possible row from the data file. Normally this will also find a lot of garbage rows; Don't use this option with repair if you are not totally desperate.", @@ -221,13 +221,13 @@ static struct my_option my_long_options[] = 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, {"keys-used", 'k', "Tell MARIA to update only some specific keys. # is a bit mask of which keys to use. This can be used to get faster inserts.", - (gptr*) &check_param.keys_in_use, - (gptr*) &check_param.keys_in_use, + (uchar**) &check_param.keys_in_use, + (uchar**) &check_param.keys_in_use, 0, GET_ULL, REQUIRED_ARG, -1, 0, 0, 0, 0, 0}, {"max-record-length", OPT_MAX_RECORD_LENGTH, "Skip rows bigger than this if maria_chk can't allocate memory to hold it", - (gptr*) &check_param.max_record_length, - (gptr*) &check_param.max_record_length, + (uchar**) &check_param.max_record_length, + (uchar**) &check_param.max_record_length, 0, GET_ULL, REQUIRED_ARG, LONGLONG_MAX, 0, LONGLONG_MAX, 0, 0, 0}, {"medium-check", 'm', "Faster than extend-check, but only finds 99.99% of all errors. Should be good enough for most cases.", @@ -256,12 +256,12 @@ static struct my_option my_long_options[] = #endif {"set-auto-increment", 'A', "Force auto_increment to start at this or higher value. If no value is given, then sets the next auto_increment value to the highest used value for the auto key + 1.", - (gptr*) &check_param.auto_increment_value, - (gptr*) &check_param.auto_increment_value, + (uchar**) &check_param.auto_increment_value, + (uchar**) &check_param.auto_increment_value, 0, GET_ULL, OPT_ARG, 0, 0, 0, 0, 0, 0}, {"set-collation", OPT_SET_COLLATION, "Change the collation used by the index", - (gptr*) &set_collation_name, 0, 0, GET_STR, REQUIRED_ARG, 0, 0, 0, 0, 0, 0}, + (uchar**) &set_collation_name, 0, 0, GET_STR, REQUIRED_ARG, 0, 0, 0, 0, 0, 0}, {"set-variable", 'O', "Change the value of a variable. Please note that this option is deprecated; you can set variables directly with --variable-name=value.", 0, 0, 0, GET_STR, REQUIRED_ARG, 0, 0, 0, 0, 0, 0}, @@ -273,12 +273,12 @@ static struct my_option my_long_options[] = 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, {"sort-records", 'R', "Sort records according to an index. This makes your data much more localized and may speed up things. (It may be VERY slow to do a sort the first time!)", - (gptr*) &check_param.opt_sort_key, - (gptr*) &check_param.opt_sort_key, + (uchar**) &check_param.opt_sort_key, + (uchar**) &check_param.opt_sort_key, 0, GET_UINT, REQUIRED_ARG, 0, 0, 0, 0, 0, 0}, {"tmpdir", 't', "Path for temporary files.", - (gptr*) &opt_tmpdir, + (uchar**) &opt_tmpdir, 0, 0, GET_STR, REQUIRED_ARG, 0, 0, 0, 0, 0, 0}, {"update-state", 'U', "Mark tables as crashed if any errors were found.", @@ -296,45 +296,45 @@ static struct my_option my_long_options[] = "Wait if table is locked.", 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, { "key_buffer_size", OPT_KEY_BUFFER_SIZE, "", - (gptr*) &check_param.use_buffers, (gptr*) &check_param.use_buffers, 0, + (uchar**) &check_param.use_buffers, (uchar**) &check_param.use_buffers, 0, GET_ULONG, REQUIRED_ARG, (long) USE_BUFFER_INIT, (long) MALLOC_OVERHEAD, (long) ~0L, (long) MALLOC_OVERHEAD, (long) IO_SIZE, 0}, { "read_buffer_size", OPT_READ_BUFFER_SIZE, "", - (gptr*) &check_param.read_buffer_length, - (gptr*) &check_param.read_buffer_length, 0, GET_ULONG, REQUIRED_ARG, + (uchar**) &check_param.read_buffer_length, + (uchar**) &check_param.read_buffer_length, 0, GET_ULONG, REQUIRED_ARG, (long) READ_BUFFER_INIT, (long) MALLOC_OVERHEAD, (long) ~0L, (long) MALLOC_OVERHEAD, (long) 1L, 0}, { "write_buffer_size", OPT_WRITE_BUFFER_SIZE, "", - (gptr*) &check_param.write_buffer_length, - (gptr*) &check_param.write_buffer_length, 0, GET_ULONG, REQUIRED_ARG, + (uchar**) &check_param.write_buffer_length, + (uchar**) &check_param.write_buffer_length, 0, GET_ULONG, REQUIRED_ARG, (long) READ_BUFFER_INIT, (long) MALLOC_OVERHEAD, (long) ~0L, (long) MALLOC_OVERHEAD, (long) 1L, 0}, { "sort_buffer_size", OPT_SORT_BUFFER_SIZE, "", - (gptr*) &check_param.sort_buffer_length, - (gptr*) &check_param.sort_buffer_length, 0, GET_ULONG, REQUIRED_ARG, + (uchar**) &check_param.sort_buffer_length, + (uchar**) &check_param.sort_buffer_length, 0, GET_ULONG, REQUIRED_ARG, (long) SORT_BUFFER_INIT, (long) (MIN_SORT_BUFFER + MALLOC_OVERHEAD), (long) ~0L, (long) MALLOC_OVERHEAD, (long) 1L, 0}, { "sort_key_blocks", OPT_SORT_KEY_BLOCKS, "", - (gptr*) &check_param.sort_key_blocks, - (gptr*) &check_param.sort_key_blocks, 0, GET_ULONG, REQUIRED_ARG, + (uchar**) &check_param.sort_key_blocks, + (uchar**) &check_param.sort_key_blocks, 0, GET_ULONG, REQUIRED_ARG, BUFFERS_WHEN_SORTING, 4L, 100L, 0L, 1L, 0}, - { "decode_bits", OPT_DECODE_BITS, "", (gptr*) &decode_bits, - (gptr*) &decode_bits, 0, GET_UINT, REQUIRED_ARG, 9L, 4L, 17L, 0L, 1L, 0}, - { "ft_min_word_len", OPT_FT_MIN_WORD_LEN, "", (gptr*) &ft_min_word_len, - (gptr*) &ft_min_word_len, 0, GET_ULONG, REQUIRED_ARG, 4, 1, HA_FT_MAXCHARLEN, + { "decode_bits", OPT_DECODE_BITS, "", (uchar**) &decode_bits, + (uchar**) &decode_bits, 0, GET_UINT, REQUIRED_ARG, 9L, 4L, 17L, 0L, 1L, 0}, + { "ft_min_word_len", OPT_FT_MIN_WORD_LEN, "", (uchar**) &ft_min_word_len, + (uchar**) &ft_min_word_len, 0, GET_ULONG, REQUIRED_ARG, 4, 1, HA_FT_MAXCHARLEN, 0, 1, 0}, - { "ft_max_word_len", OPT_FT_MAX_WORD_LEN, "", (gptr*) &ft_max_word_len, - (gptr*) &ft_max_word_len, 0, GET_ULONG, REQUIRED_ARG, HA_FT_MAXCHARLEN, 10, + { "ft_max_word_len", OPT_FT_MAX_WORD_LEN, "", (uchar**) &ft_max_word_len, + (uchar**) &ft_max_word_len, 0, GET_ULONG, REQUIRED_ARG, HA_FT_MAXCHARLEN, 10, HA_FT_MAXCHARLEN, 0, 1, 0}, { "maria_ft_stopword_file", OPT_FT_STOPWORD_FILE, "Use stopwords from this file instead of built-in list.", - (gptr*) &ft_stopword_file, (gptr*) &ft_stopword_file, 0, GET_STR, + (uchar**) &ft_stopword_file, (uchar**) &ft_stopword_file, 0, GET_STR, REQUIRED_ARG, 0, 0, 0, 0, 0, 0}, {"stats_method", OPT_STATS_METHOD, "Specifies how index statistics collection code should threat NULLs. " "Possible values of name are \"nulls_unequal\" (default behavior for 4.1/5.0), " "\"nulls_equal\" (emulate 4.0 behavior), and \"nulls_ignored\".", - (gptr*) &maria_stats_method_str, (gptr*) &maria_stats_method_str, 0, + (uchar**) &maria_stats_method_str, (uchar**) &maria_stats_method_str, 0, GET_STR, REQUIRED_ARG, 0, 0, 0, 0, 0, 0}, { 0, 0, 0, 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0} }; @@ -794,7 +794,7 @@ static void get_options(register int *argc,register char ***argv) /* Check table */ -static int maria_chk(HA_CHECK *param, my_string filename) +static int maria_chk(HA_CHECK *param, char *filename) { int error,lock_type,recreate; int rep_quick= param->testflag & (T_QUICK | T_FORCE_UNIQUENESS); @@ -1215,7 +1215,7 @@ end2: /* Write info about table */ -static void descript(HA_CHECK *param, register MARIA_HA *info, my_string name) +static void descript(HA_CHECK *param, register MARIA_HA *info, char *name) { uint key,keyseg_nr,field; reg3 MARIA_KEYDEF *keyinfo; @@ -1468,7 +1468,7 @@ static void descript(HA_CHECK *param, register MARIA_HA *info, my_string name) /* Sort records according to one key */ static int maria_sort_records(HA_CHECK *param, - register MARIA_HA *info, my_string name, + register MARIA_HA *info, char *name, uint sort_key, my_bool write_info, my_bool update_index) @@ -1477,7 +1477,7 @@ static int maria_sort_records(HA_CHECK *param, uint key; MARIA_KEYDEF *keyinfo; File new_file; - byte *temp_buff; + uchar *temp_buff; ha_rows old_record_count; MARIA_SHARE *share=info->s; char llbuff[22],llbuff2[22]; @@ -1532,12 +1532,12 @@ static int maria_sort_records(HA_CHECK *param, goto err; info->opt_flag|=WRITE_CACHE_USED; - if (!(temp_buff=(byte*) my_alloca((uint) keyinfo->block_length))) + if (!(temp_buff=(uchar*) my_alloca((uint) keyinfo->block_length))) { _ma_check_print_error(param,"Not enough memory for key block"); goto err; } - if (!(sort_param.record=(byte*) my_malloc((uint) share->base.pack_reclength, + if (!(sort_param.record=(uchar*) my_malloc((uint) share->base.pack_reclength, MYF(0)))) { _ma_check_print_error(param,"Not enough memory for record"); @@ -1630,7 +1630,7 @@ err: } if (temp_buff) { - my_afree((gptr) temp_buff); + my_afree((uchar*) temp_buff); } my_free(sort_param.record,MYF(MY_ALLOW_ZERO_PTR)); info->opt_flag&= ~(READ_CACHE_USED | WRITE_CACHE_USED); @@ -1647,13 +1647,13 @@ err: static int sort_record_index(MARIA_SORT_PARAM *sort_param,MARIA_HA *info, MARIA_KEYDEF *keyinfo, - my_off_t page, byte *buff, uint sort_key, + my_off_t page, uchar *buff, uint sort_key, File new_file,my_bool update_index) { uint nod_flag,used_length,key_length; - byte *temp_buff,*keypos,*endpos; + uchar *temp_buff,*keypos,*endpos; my_off_t next_page,rec_pos; - byte lastkey[HA_MAX_KEY_BUFF]; + uchar lastkey[HA_MAX_KEY_BUFF]; char llbuff[22]; MARIA_SORT_INFO *sort_info= sort_param->sort_info; HA_CHECK *param=sort_info->param; @@ -1664,7 +1664,7 @@ static int sort_record_index(MARIA_SORT_PARAM *sort_param,MARIA_HA *info, if (nod_flag) { - if (!(temp_buff= (byte*) my_alloca((uint) keyinfo->block_length))) + if (!(temp_buff= (uchar*) my_alloca((uint) keyinfo->block_length))) { _ma_check_print_error(param,"Not Enough memory"); DBUG_RETURN(-1); @@ -1679,7 +1679,7 @@ static int sort_record_index(MARIA_SORT_PARAM *sort_param,MARIA_HA *info, if (nod_flag) { next_page= _ma_kpos(nod_flag, keypos); - if (my_pread(info->s->kfile.file, (byte*)temp_buff, + if (my_pread(info->s->kfile.file, (uchar*)temp_buff, (uint) keyinfo->block_length, next_page, MYF(MY_NABP+MY_WME))) { @@ -1719,19 +1719,19 @@ static int sort_record_index(MARIA_SORT_PARAM *sort_param,MARIA_HA *info, goto err; } /* Clear end of block to get better compression if the table is backuped */ - bzero((byte*) buff+used_length,keyinfo->block_length-used_length); - if (my_pwrite(info->s->kfile.file, (byte*)buff, (uint)keyinfo->block_length, + bzero((uchar*) buff+used_length,keyinfo->block_length-used_length); + if (my_pwrite(info->s->kfile.file, (uchar*)buff, (uint)keyinfo->block_length, page,param->myf_rw)) { _ma_check_print_error(param,"%d when updating keyblock",my_errno); goto err; } if (temp_buff) - my_afree((gptr) temp_buff); + my_afree((uchar*) temp_buff); DBUG_RETURN(0); err: if (temp_buff) - my_afree((gptr) temp_buff); + my_afree((uchar*) temp_buff); DBUG_RETURN(1); } /* sort_record_index */ diff --git a/storage/maria/maria_def.h b/storage/maria/maria_def.h index 740808c7bbe..6c735b745ea 100644 --- a/storage/maria/maria_def.h +++ b/storage/maria/maria_def.h @@ -227,7 +227,7 @@ typedef struct st_maria_share char *data_file_name; /* Resolved path names from symlinks */ char *index_file_name; char *open_file_name; /* parameter to open filename */ - byte *file_map; /* mem-map of file if possible */ + uchar *file_map; /* mem-map of file if possible */ PAGECACHE *pagecache; /* ref to the current key cache */ MARIA_DECODE_TREE *decode_trees; uint16 *decode_tables; @@ -241,36 +241,36 @@ typedef struct st_maria_share /* Is called for every close of the table */ void (*end)(struct st_maria_info *); /* Called when we want to read a record from a specific position */ - int (*read_record)(struct st_maria_info *, byte *, MARIA_RECORD_POS); + int (*read_record)(struct st_maria_info *, uchar *, MARIA_RECORD_POS); /* Initialize a scan */ my_bool (*scan_init)(struct st_maria_info *); /* Read next record while scanning */ - int (*scan)(struct st_maria_info *, byte *, MARIA_RECORD_POS, my_bool); + int (*scan)(struct st_maria_info *, uchar *, MARIA_RECORD_POS, my_bool); /* End scan */ void (*scan_end)(struct st_maria_info *); /* Pre-write of row (some handlers may do the actual write here) */ - MARIA_RECORD_POS (*write_record_init)(struct st_maria_info *, const byte *); + MARIA_RECORD_POS (*write_record_init)(struct st_maria_info *, const uchar *); /* Write record (or accept write_record_init) */ - my_bool (*write_record)(struct st_maria_info *, const byte *); + my_bool (*write_record)(struct st_maria_info *, const uchar *); /* Called when write failed */ my_bool (*write_record_abort)(struct st_maria_info *); my_bool (*update_record)(struct st_maria_info *, MARIA_RECORD_POS, - const byte *, const byte *); - my_bool (*delete_record)(struct st_maria_info *, const byte *record); - my_bool (*compare_record)(struct st_maria_info *, const byte *); + const uchar *, const uchar *); + my_bool (*delete_record)(struct st_maria_info *, const uchar *record); + my_bool (*compare_record)(struct st_maria_info *, const uchar *); /* calculate checksum for a row */ - ha_checksum(*calc_checksum)(struct st_maria_info *, const byte *); + ha_checksum(*calc_checksum)(struct st_maria_info *, const uchar *); /* Calculate checksum for a row during write. May be 0 if we calculate the checksum in write_record_init() */ - ha_checksum(*calc_write_checksum) (struct st_maria_info *, const byte *); + ha_checksum(*calc_write_checksum) (struct st_maria_info *, const uchar *); /* Compare a row in memory with a row on disk */ my_bool (*compare_unique)(struct st_maria_info *, MARIA_UNIQUEDEF *, - const byte *record, MARIA_RECORD_POS pos); + const uchar *record, MARIA_RECORD_POS pos); /* Mapings to read/write the data file */ - uint (*file_read)(MARIA_HA *, byte *, uint, my_off_t, myf); - uint (*file_write)(MARIA_HA *, byte *, uint, my_off_t, myf); + uint (*file_read)(MARIA_HA *, uchar *, uint, my_off_t, myf); + uint (*file_write)(MARIA_HA *, uchar *, uint, my_off_t, myf); invalidator_by_filename invalidator; /* query cache invalidator */ ulong this_process; /* processid */ ulong last_process; /* For table-change-check */ @@ -316,7 +316,7 @@ typedef struct st_maria_share } MARIA_SHARE; -typedef byte MARIA_BITMAP_BUFFER; +typedef uchar MARIA_BITMAP_BUFFER; typedef struct st_maria_bitmap_block { @@ -351,12 +351,12 @@ typedef struct st_maria_row MARIA_RECORD_POS lastpos, nextpos; MARIA_RECORD_POS *tail_positions; ha_checksum checksum; - byte *empty_bits, *field_lengths; + uchar *empty_bits, *field_lengths; uint *null_field_lengths; /* All null field lengths */ ulong *blob_lengths; /* Length for each blob */ ulong base_length, normal_length, char_length, varchar_length, blob_length; ulong head_length, total_length; - my_size_t extents_buffer_length; /* Size of 'extents' buffer */ + size_t extents_buffer_length; /* Size of 'extents' buffer */ uint field_lengths_length; /* Length of data in field_lengths */ uint extents_count; /* number of extents in 'extents' */ uint full_page_count, tail_count; /* For maria_chk */ @@ -365,8 +365,8 @@ typedef struct st_maria_row /* Data to scan row in blocked format */ typedef struct st_maria_block_scan { - byte *bitmap_buff, *bitmap_pos, *bitmap_end, *page_buff; - byte *dir, *dir_end; + uchar *bitmap_buff, *bitmap_pos, *bitmap_end, *page_buff; + uchar *dir, *dir_end; ulong bitmap_page; ulonglong bits; uint number_of_rows, bit_pos; @@ -392,17 +392,17 @@ struct st_maria_info DYNAMIC_ARRAY *ft1_to_ft2; /* used only in ft1->ft2 conversion */ MEM_ROOT ft_memroot; /* used by the parser */ MYSQL_FTPARSER_PARAM *ftparser_param; /* share info between init/deinit */ - byte *buff; /* page buffer */ - byte *keyread_buff; /* Buffer for last key read */ - byte *lastkey, *lastkey2; /* Last used search key */ - byte *first_mbr_key; /* Searhed spatial key */ - byte *rec_buff; /* Temp buffer for recordpack */ - byte *int_keypos, /* Save position for next/previous */ + uchar *buff; /* page buffer */ + uchar *keyread_buff; /* Buffer for last key read */ + uchar *lastkey, *lastkey2; /* Last used search key */ + uchar *first_mbr_key; /* Searhed spatial key */ + uchar *rec_buff; /* Temp buffer for recordpack */ + uchar *int_keypos, /* Save position for next/previous */ *int_maxpos; /* -""- */ - byte *update_field_data; /* Used by update in rows-in-block */ + uchar *update_field_data; /* Used by update in rows-in-block */ uint int_nod_flag; /* -""- */ uint32 int_keytree_version; /* -""- */ - int (*read_record) (struct st_maria_info *, byte*, MARIA_RECORD_POS); + int (*read_record) (struct st_maria_info *, uchar*, MARIA_RECORD_POS); invalidator_by_filename invalidator; /* query cache invalidator */ ulong this_unique; /* uniq filenumber or thread */ ulong last_unique; /* last unique number */ @@ -419,7 +419,7 @@ struct st_maria_info as they are not compatible with parallel repair */ ulong packed_length, blob_length; /* Length of found, packed record */ - my_size_t rec_buff_size; + size_t rec_buff_size; PAGECACHE_FILE dfile; /* The datafile */ uint opt_flag; /* Optim. for space/speed */ uint update; /* If file changed since open */ @@ -543,7 +543,7 @@ struct st_maria_info #define MARIA_DYN_MAX_BLOCK_LENGTH ((1L << 24)-4L) #define MARIA_DYN_MAX_ROW_LENGTH (MARIA_DYN_MAX_BLOCK_LENGTH - MARIA_SPLIT_LENGTH) #define MARIA_DYN_ALIGN_SIZE 4 /* Align blocks on this */ -#define MARIA_MAX_DYN_HEADER_BYTE 13 /* max header byte for dynamic rows */ +#define MARIA_MAX_DYN_HEADER_BYTE 13 /* max header uchar for dynamic rows */ #define MARIA_MAX_BLOCK_LENGTH ((((ulong) 1 << 24)-1) & (~ (ulong) (MARIA_DYN_ALIGN_SIZE-1))) #define MARIA_REC_BUFF_OFFSET ALIGN_SIZE(MARIA_DYN_DELETE_BLOCK_HEADER+sizeof(uint32)) @@ -584,7 +584,7 @@ extern uchar NEAR maria_file_magic[], NEAR maria_pack_file_magic[]; extern uint NEAR maria_read_vec[], NEAR maria_readnext_vec[]; extern uint maria_quick_table_bits; extern const char *maria_data_root; -extern byte maria_zero_string[]; +extern uchar maria_zero_string[]; extern my_bool maria_inited; @@ -593,8 +593,8 @@ typedef struct st_maria_s_param { uint ref_length, key_length, n_ref_length; uint n_length, totlength, part_of_prev_key, prev_length, pack_marker; - const byte *key; - byte *prev_key, *next_key_pos; + const uchar *key; + uchar *prev_key, *next_key_pos; bool store_not_null; } MARIA_KEY_PARAM; @@ -608,73 +608,73 @@ typedef struct st_pinned_page /* Prototypes for intern functions */ -extern int _ma_read_dynamic_record(MARIA_HA *, byte *, MARIA_RECORD_POS); -extern int _ma_read_rnd_dynamic_record(MARIA_HA *, byte *, MARIA_RECORD_POS, +extern int _ma_read_dynamic_record(MARIA_HA *, uchar *, MARIA_RECORD_POS); +extern int _ma_read_rnd_dynamic_record(MARIA_HA *, uchar *, MARIA_RECORD_POS, my_bool); -extern my_bool _ma_write_dynamic_record(MARIA_HA *, const byte *); +extern my_bool _ma_write_dynamic_record(MARIA_HA *, const uchar *); extern my_bool _ma_update_dynamic_record(MARIA_HA *, MARIA_RECORD_POS, - const byte *, const byte *); -extern my_bool _ma_delete_dynamic_record(MARIA_HA *info, const byte *record); -extern my_bool _ma_cmp_dynamic_record(MARIA_HA *info, const byte *record); -extern my_bool _ma_write_blob_record(MARIA_HA *, const byte *); + const uchar *, const uchar *); +extern my_bool _ma_delete_dynamic_record(MARIA_HA *info, const uchar *record); +extern my_bool _ma_cmp_dynamic_record(MARIA_HA *info, const uchar *record); +extern my_bool _ma_write_blob_record(MARIA_HA *, const uchar *); extern my_bool _ma_update_blob_record(MARIA_HA *, MARIA_RECORD_POS, - const byte *, const byte *); -extern int _ma_read_static_record(MARIA_HA *info, byte *, MARIA_RECORD_POS); -extern int _ma_read_rnd_static_record(MARIA_HA *, byte *, MARIA_RECORD_POS, + const uchar *, const uchar *); +extern int _ma_read_static_record(MARIA_HA *info, uchar *, MARIA_RECORD_POS); +extern int _ma_read_rnd_static_record(MARIA_HA *, uchar *, MARIA_RECORD_POS, my_bool); -extern my_bool _ma_write_static_record(MARIA_HA *, const byte *); +extern my_bool _ma_write_static_record(MARIA_HA *, const uchar *); extern my_bool _ma_update_static_record(MARIA_HA *, MARIA_RECORD_POS, - const byte *, const byte *); -extern my_bool _ma_delete_static_record(MARIA_HA *info, const byte *record); -extern my_bool _ma_cmp_static_record(MARIA_HA *info, const byte *record); -extern int _ma_ck_write(MARIA_HA *info, uint keynr, byte *key, + const uchar *, const uchar *); +extern my_bool _ma_delete_static_record(MARIA_HA *info, const uchar *record); +extern my_bool _ma_cmp_static_record(MARIA_HA *info, const uchar *record); +extern int _ma_ck_write(MARIA_HA *info, uint keynr, uchar *key, uint length); extern int _ma_ck_real_write_btree(MARIA_HA *info, MARIA_KEYDEF *keyinfo, - byte *key, uint key_length, + uchar *key, uint key_length, MARIA_RECORD_POS *root, uint comp_flag); extern int _ma_enlarge_root(MARIA_HA *info, MARIA_KEYDEF *keyinfo, - byte *key, MARIA_RECORD_POS *root); -extern int _ma_insert(MARIA_HA *info, MARIA_KEYDEF *keyinfo, byte *key, - byte *anc_buff, byte *key_pos, byte *key_buff, - byte *father_buff, byte *father_keypos, + uchar *key, MARIA_RECORD_POS *root); +extern int _ma_insert(MARIA_HA *info, MARIA_KEYDEF *keyinfo, uchar *key, + uchar *anc_buff, uchar *key_pos, uchar *key_buff, + uchar *father_buff, uchar *father_keypos, my_off_t father_page, my_bool insert_last); extern int _ma_split_page(MARIA_HA *info, MARIA_KEYDEF *keyinfo, - byte *key, byte *buff, byte *key_buff, + uchar *key, uchar *buff, uchar *key_buff, my_bool insert_last); -extern byte *_ma_find_half_pos(uint nod_flag, MARIA_KEYDEF *keyinfo, - byte *page, byte *key, +extern uchar *_ma_find_half_pos(uint nod_flag, MARIA_KEYDEF *keyinfo, + uchar *page, uchar *key, uint *return_key_length, - byte ** after_key); + uchar ** after_key); extern int _ma_calc_static_key_length(MARIA_KEYDEF *keyinfo, uint nod_flag, - byte *key_pos, byte *org_key, - byte *key_buff, const byte *key, + uchar *key_pos, uchar *org_key, + uchar *key_buff, const uchar *key, MARIA_KEY_PARAM *s_temp); extern int _ma_calc_var_key_length(MARIA_KEYDEF *keyinfo, uint nod_flag, - byte *key_pos, byte *org_key, - byte *key_buff, const byte *key, + uchar *key_pos, uchar *org_key, + uchar *key_buff, const uchar *key, MARIA_KEY_PARAM *s_temp); extern int _ma_calc_var_pack_key_length(MARIA_KEYDEF *keyinfo, - uint nod_flag, byte *key_pos, - byte *org_key, byte *prev_key, - const byte *key, + uint nod_flag, uchar *key_pos, + uchar *org_key, uchar *prev_key, + const uchar *key, MARIA_KEY_PARAM *s_temp); extern int _ma_calc_bin_pack_key_length(MARIA_KEYDEF *keyinfo, - uint nod_flag, byte *key_pos, - byte *org_key, byte *prev_key, - const byte *key, + uint nod_flag, uchar *key_pos, + uchar *org_key, uchar *prev_key, + const uchar *key, MARIA_KEY_PARAM *s_temp); -void _ma_store_static_key(MARIA_KEYDEF *keyinfo, byte *key_pos, +void _ma_store_static_key(MARIA_KEYDEF *keyinfo, uchar *key_pos, MARIA_KEY_PARAM *s_temp); -void _ma_store_var_pack_key(MARIA_KEYDEF *keyinfo, byte *key_pos, +void _ma_store_var_pack_key(MARIA_KEYDEF *keyinfo, uchar *key_pos, MARIA_KEY_PARAM *s_temp); #ifdef NOT_USED -void _ma_store_pack_key(MARIA_KEYDEF *keyinfo, byte *key_pos, +void _ma_store_pack_key(MARIA_KEYDEF *keyinfo, uchar *key_pos, MARIA_KEY_PARAM *s_temp); #endif -void _ma_store_bin_pack_key(MARIA_KEYDEF *keyinfo, byte *key_pos, +void _ma_store_bin_pack_key(MARIA_KEYDEF *keyinfo, uchar *key_pos, MARIA_KEY_PARAM *s_temp); -extern int _ma_ck_delete(MARIA_HA *info, uint keynr, byte *key, +extern int _ma_ck_delete(MARIA_HA *info, uint keynr, uchar *key, uint key_length); extern int _ma_readinfo(MARIA_HA *info, int lock_flag, int check_keybuffer); extern int _ma_writeinfo(MARIA_HA *info, uint options); @@ -682,91 +682,91 @@ extern int _ma_test_if_changed(MARIA_HA *info); extern int _ma_mark_file_changed(MARIA_HA *info); extern int _ma_decrement_open_count(MARIA_HA *info); extern int _ma_check_index(MARIA_HA *info, int inx); -extern int _ma_search(MARIA_HA *info, MARIA_KEYDEF *keyinfo, byte *key, +extern int _ma_search(MARIA_HA *info, MARIA_KEYDEF *keyinfo, uchar *key, uint key_len, uint nextflag, my_off_t pos); extern int _ma_bin_search(struct st_maria_info *info, MARIA_KEYDEF *keyinfo, - byte *page, byte *key, uint key_len, - uint comp_flag, byte **ret_pos, byte *buff, + uchar *page, uchar *key, uint key_len, + uint comp_flag, uchar **ret_pos, uchar *buff, my_bool *was_last_key); extern int _ma_seq_search(MARIA_HA *info, MARIA_KEYDEF *keyinfo, - byte *page, byte *key, uint key_len, - uint comp_flag, byte ** ret_pos, byte *buff, + uchar *page, uchar *key, uint key_len, + uint comp_flag, uchar ** ret_pos, uchar *buff, my_bool *was_last_key); extern int _ma_prefix_search(MARIA_HA *info, MARIA_KEYDEF *keyinfo, - byte *page, byte *key, uint key_len, - uint comp_flag, byte ** ret_pos, byte *buff, + uchar *page, uchar *key, uint key_len, + uint comp_flag, uchar ** ret_pos, uchar *buff, my_bool *was_last_key); -extern my_off_t _ma_kpos(uint nod_flag, byte *after_key); -extern void _ma_kpointer(MARIA_HA *info, byte *buff, my_off_t pos); +extern my_off_t _ma_kpos(uint nod_flag, uchar *after_key); +extern void _ma_kpointer(MARIA_HA *info, uchar *buff, my_off_t pos); extern MARIA_RECORD_POS _ma_dpos(MARIA_HA *info, uint nod_flag, - const byte *after_key); -extern MARIA_RECORD_POS _ma_rec_pos(MARIA_SHARE *info, byte *ptr); -extern void _ma_dpointer(MARIA_HA *info, byte *buff, MARIA_RECORD_POS pos); + const uchar *after_key); +extern MARIA_RECORD_POS _ma_rec_pos(MARIA_SHARE *info, uchar *ptr); +extern void _ma_dpointer(MARIA_HA *info, uchar *buff, MARIA_RECORD_POS pos); extern uint _ma_get_static_key(MARIA_KEYDEF *keyinfo, uint nod_flag, - byte **page, byte *key); + uchar **page, uchar *key); extern uint _ma_get_pack_key(MARIA_KEYDEF *keyinfo, uint nod_flag, - byte **page, byte *key); + uchar **page, uchar *key); extern uint _ma_get_binary_pack_key(MARIA_KEYDEF *keyinfo, uint nod_flag, - byte ** page_pos, byte *key); -extern byte *_ma_get_last_key(MARIA_HA *info, MARIA_KEYDEF *keyinfo, - byte *keypos, byte *lastkey, - byte *endpos, uint *return_key_length); -extern byte *_ma_get_key(MARIA_HA *info, MARIA_KEYDEF *keyinfo, - byte *page, byte *key, byte *keypos, + uchar ** page_pos, uchar *key); +extern uchar *_ma_get_last_key(MARIA_HA *info, MARIA_KEYDEF *keyinfo, + uchar *keypos, uchar *lastkey, + uchar *endpos, uint *return_key_length); +extern uchar *_ma_get_key(MARIA_HA *info, MARIA_KEYDEF *keyinfo, + uchar *page, uchar *key, uchar *keypos, uint *return_key_length); -extern uint _ma_keylength(MARIA_KEYDEF *keyinfo, const byte *key); -extern uint _ma_keylength_part(MARIA_KEYDEF *keyinfo, register const byte *key, +extern uint _ma_keylength(MARIA_KEYDEF *keyinfo, const uchar *key); +extern uint _ma_keylength_part(MARIA_KEYDEF *keyinfo, register const uchar *key, HA_KEYSEG *end); -extern byte *_ma_move_key(MARIA_KEYDEF *keyinfo, byte *to, const byte *from); +extern uchar *_ma_move_key(MARIA_KEYDEF *keyinfo, uchar *to, const uchar *from); extern int _ma_search_next(MARIA_HA *info, MARIA_KEYDEF *keyinfo, - byte *key, uint key_length, uint nextflag, + uchar *key, uint key_length, uint nextflag, my_off_t pos); extern int _ma_search_first(MARIA_HA *info, MARIA_KEYDEF *keyinfo, my_off_t pos); extern int _ma_search_last(MARIA_HA *info, MARIA_KEYDEF *keyinfo, my_off_t pos); -extern byte *_ma_fetch_keypage(MARIA_HA *info, MARIA_KEYDEF *keyinfo, - my_off_t page, int level, byte *buff, +extern uchar *_ma_fetch_keypage(MARIA_HA *info, MARIA_KEYDEF *keyinfo, + my_off_t page, int level, uchar *buff, int return_buffer); extern int _ma_write_keypage(MARIA_HA *info, MARIA_KEYDEF *keyinfo, - my_off_t page, int level, byte *buff); + my_off_t page, int level, uchar *buff); extern int _ma_dispose(MARIA_HA *info, MARIA_KEYDEF *keyinfo, my_off_t pos, int level); extern my_off_t _ma_new(MARIA_HA *info, MARIA_KEYDEF *keyinfo, int level); -extern uint _ma_make_key(MARIA_HA *info, uint keynr, byte *key, - const byte *record, MARIA_RECORD_POS filepos); -extern uint _ma_pack_key(MARIA_HA *info, uint keynr, byte *key, - const byte *old, uint key_length, +extern uint _ma_make_key(MARIA_HA *info, uint keynr, uchar *key, + const uchar *record, MARIA_RECORD_POS filepos); +extern uint _ma_pack_key(MARIA_HA *info, uint keynr, uchar *key, + const uchar *old, uint key_length, HA_KEYSEG ** last_used_keyseg); -extern int _ma_read_key_record(MARIA_HA *info, byte *buf, MARIA_RECORD_POS); -extern int _ma_read_cache(IO_CACHE *info, byte *buff, MARIA_RECORD_POS pos, +extern int _ma_read_key_record(MARIA_HA *info, uchar *buf, MARIA_RECORD_POS); +extern int _ma_read_cache(IO_CACHE *info, uchar *buff, MARIA_RECORD_POS pos, uint length, int re_read_if_possibly); -extern ulonglong ma_retrieve_auto_increment(MARIA_HA *info, const byte *record); +extern ulonglong ma_retrieve_auto_increment(MARIA_HA *info, const uchar *record); -extern my_bool _ma_alloc_buffer(byte **old_addr, my_size_t *old_size, - my_size_t new_size); -extern ulong _ma_rec_unpack(MARIA_HA *info, byte *to, byte *from, +extern my_bool _ma_alloc_buffer(uchar **old_addr, size_t *old_size, + size_t new_size); +extern ulong _ma_rec_unpack(MARIA_HA *info, uchar *to, uchar *from, ulong reclength); extern my_bool _ma_rec_check(MARIA_HA *info, const char *record, - byte *packpos, ulong packed_length, + uchar *packpos, ulong packed_length, my_bool with_checkum); extern int _ma_write_part_record(MARIA_HA *info, my_off_t filepos, ulong length, my_off_t next_filepos, - byte ** record, ulong *reclength, + uchar ** record, ulong *reclength, int *flag); extern void _ma_print_key(FILE *stream, HA_KEYSEG *keyseg, - const byte *key, uint length); + const uchar *key, uint length); extern my_bool _ma_once_init_pack_row(MARIA_SHARE *share, File dfile); extern my_bool _ma_once_end_pack_row(MARIA_SHARE *share); -extern int _ma_read_pack_record(MARIA_HA *info, byte *buf, +extern int _ma_read_pack_record(MARIA_HA *info, uchar *buf, MARIA_RECORD_POS filepos); -extern int _ma_read_rnd_pack_record(MARIA_HA *, byte *, MARIA_RECORD_POS, +extern int _ma_read_rnd_pack_record(MARIA_HA *, uchar *, MARIA_RECORD_POS, my_bool); extern int _ma_pack_rec_unpack(MARIA_HA *info, MARIA_BIT_BUFF *bit_buff, - byte *to, byte *from, ulong reclength); + uchar *to, uchar *from, ulong reclength); extern ulonglong _ma_safe_mul(ulonglong a, ulonglong b); -extern int _ma_ft_update(MARIA_HA *info, uint keynr, byte *keybuf, - const byte *oldrec, const byte *newrec, +extern int _ma_ft_update(MARIA_HA *info, uint keynr, uchar *keybuf, + const uchar *oldrec, const uchar *newrec, my_off_t pos); /* @@ -821,30 +821,30 @@ typedef struct st_maria_block_info #define fast_ma_readinfo(INFO) ((INFO)->lock_type == F_UNLCK) && _ma_readinfo((INFO),F_RDLCK,1) extern uint _ma_get_block_info(MARIA_BLOCK_INFO *, File, my_off_t); -extern uint _ma_rec_pack(MARIA_HA *info, byte *to, const byte *from); +extern uint _ma_rec_pack(MARIA_HA *info, uchar *to, const uchar *from); extern uint _ma_pack_get_block_info(MARIA_HA *maria, MARIA_BIT_BUFF *bit_buff, - MARIA_BLOCK_INFO *info, byte **rec_buff_p, - my_size_t *rec_buff_size, + MARIA_BLOCK_INFO *info, uchar **rec_buff_p, + size_t *rec_buff_size, File file, my_off_t filepos); -extern void _ma_store_blob_length(byte *pos, uint pack_length, uint length); +extern void _ma_store_blob_length(uchar *pos, uint pack_length, uint length); extern void _ma_report_error(int errcode, const char *file_name); extern my_bool _ma_memmap_file(MARIA_HA *info); extern void _ma_unmap_file(MARIA_HA *info); -extern uint _ma_save_pack_length(uint version, byte * block_buff, +extern uint _ma_save_pack_length(uint version, uchar * block_buff, ulong length); extern uint _ma_calc_pack_length(uint version, ulong length); -extern ulong _ma_calc_blob_length(uint length, const byte *pos); -extern uint _ma_mmap_pread(MARIA_HA *info, byte *Buffer, +extern ulong _ma_calc_blob_length(uint length, const uchar *pos); +extern uint _ma_mmap_pread(MARIA_HA *info, uchar *Buffer, uint Count, my_off_t offset, myf MyFlags); -extern uint _ma_mmap_pwrite(MARIA_HA *info, byte *Buffer, +extern uint _ma_mmap_pwrite(MARIA_HA *info, uchar *Buffer, uint Count, my_off_t offset, myf MyFlags); -extern uint _ma_nommap_pread(MARIA_HA *info, byte *Buffer, +extern uint _ma_nommap_pread(MARIA_HA *info, uchar *Buffer, uint Count, my_off_t offset, myf MyFlags); -extern uint _ma_nommap_pwrite(MARIA_HA *info, byte *Buffer, +extern uint _ma_nommap_pwrite(MARIA_HA *info, uchar *Buffer, uint Count, my_off_t offset, myf MyFlags); uint _ma_state_info_write(File file, MARIA_STATE_INFO *state, uint pWrite); -byte *_ma_state_info_read(byte *ptr, MARIA_STATE_INFO *state); +uchar *_ma_state_info_read(uchar *ptr, MARIA_STATE_INFO *state); uint _ma_state_info_read_dsk(File file, MARIA_STATE_INFO *state, my_bool pRead); uint _ma_base_info_write(File file, MARIA_BASE_INFO *base); @@ -856,18 +856,18 @@ uint _ma_uniquedef_write(File file, MARIA_UNIQUEDEF *keydef); char *_ma_uniquedef_read(char *ptr, MARIA_UNIQUEDEF *keydef); uint _ma_columndef_write(File file, MARIA_COLUMNDEF *columndef); char *_ma_columndef_read(char *ptr, MARIA_COLUMNDEF *columndef); -ulong _ma_calc_total_blob_length(MARIA_HA *info, const byte *record); -ha_checksum _ma_checksum(MARIA_HA *info, const byte *buf); -ha_checksum _ma_static_checksum(MARIA_HA *info, const byte *buf); +ulong _ma_calc_total_blob_length(MARIA_HA *info, const uchar *record); +ha_checksum _ma_checksum(MARIA_HA *info, const uchar *buf); +ha_checksum _ma_static_checksum(MARIA_HA *info, const uchar *buf); my_bool _ma_check_unique(MARIA_HA *info, MARIA_UNIQUEDEF *def, - byte *record, ha_checksum unique_hash, + uchar *record, ha_checksum unique_hash, MARIA_RECORD_POS pos); -ha_checksum _ma_unique_hash(MARIA_UNIQUEDEF *def, const byte *buf); +ha_checksum _ma_unique_hash(MARIA_UNIQUEDEF *def, const uchar *buf); my_bool _ma_cmp_static_unique(MARIA_HA *info, MARIA_UNIQUEDEF *def, - const byte *record, MARIA_RECORD_POS pos); + const uchar *record, MARIA_RECORD_POS pos); my_bool _ma_cmp_dynamic_unique(MARIA_HA *info, MARIA_UNIQUEDEF *def, - const byte *record, MARIA_RECORD_POS pos); -my_bool _ma_unique_comp(MARIA_UNIQUEDEF *def, const byte *a, const byte *b, + const uchar *record, MARIA_RECORD_POS pos); +my_bool _ma_unique_comp(MARIA_UNIQUEDEF *def, const uchar *a, const uchar *b, my_bool null_are_equal); void _ma_get_status(void *param, int concurrent_insert); void _ma_update_status(void *param); @@ -883,7 +883,7 @@ void _ma_setup_functions(register MARIA_SHARE *share); my_bool _ma_dynmap_file(MARIA_HA *info, my_off_t size); void _ma_remap_file(MARIA_HA *info, my_off_t size); -MARIA_RECORD_POS _ma_write_init_default(MARIA_HA *info, const byte *record); +MARIA_RECORD_POS _ma_write_init_default(MARIA_HA *info, const uchar *record); my_bool _ma_write_abort_default(MARIA_HA *info); /* Functions needed by _ma_check (are overrided in MySQL) */ diff --git a/storage/maria/maria_ftdump.c b/storage/maria/maria_ftdump.c index 8b0256344cb..9df86b50474 100644 --- a/storage/maria/maria_ftdump.c +++ b/storage/maria/maria_ftdump.c @@ -46,7 +46,7 @@ static struct my_option my_long_options[] = {"stats", 's', "Report global stats.", 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, {"verbose", 'v', "Be verbose.", - (gptr*) &verbose, (gptr*) &verbose, 0, GET_BOOL, NO_ARG, 0, 0, 0, 0, 0, 0}, + (uchar**) &verbose, (uchar**) &verbose, 0, GET_BOOL, NO_ARG, 0, 0, 0, 0, 0, 0}, { 0, 0, 0, 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0} }; diff --git a/storage/maria/maria_pack.c b/storage/maria/maria_pack.c index f6e962191c2..987711a270d 100644 --- a/storage/maria/maria_pack.c +++ b/storage/maria/maria_pack.c @@ -69,8 +69,8 @@ typedef struct st_huff_counts { my_off_t pre_space[8]; my_off_t tot_end_space,tot_pre_space,zero_fields,empty_fields,bytes_packed; TREE int_tree; /* Tree for detecting distinct column values. */ - byte *tree_buff; /* Column values, 'field_length' each. */ - byte *tree_pos; /* Points to end of column values in 'tree_buff'. */ + uchar *tree_buff; /* Column values, 'field_length' each. */ + uchar *tree_pos; /* Points to end of column values in 'tree_buff'. */ } HUFF_COUNTS; typedef struct st_huff_element HUFF_ELEMENT; @@ -141,8 +141,8 @@ static int test_space_compress(HUFF_COUNTS *huff_counts,my_off_t records, enum en_fieldtype field_type); static HUFF_TREE* make_huff_trees(HUFF_COUNTS *huff_counts,uint trees); static int make_huff_tree(HUFF_TREE *tree,HUFF_COUNTS *huff_counts); -static int compare_huff_elements(void *not_used, byte *a,byte *b); -static int save_counts_in_queue(byte *key,element_count count, +static int compare_huff_elements(void *not_used, uchar *a,uchar *b); +static int save_counts_in_queue(uchar *key,element_count count, HUFF_TREE *tree); static my_off_t calc_packed_length(HUFF_COUNTS *huff_counts,uint flag); static uint join_same_trees(HUFF_COUNTS *huff_counts,uint trees); @@ -171,7 +171,7 @@ static int save_state(MARIA_HA *isam_file,PACK_MRG_INFO *mrg,my_off_t new_length static int save_state_mrg(File file,PACK_MRG_INFO *isam_file,my_off_t new_length, ha_checksum crc); static int mrg_close(PACK_MRG_INFO *mrg); -static int mrg_rrnd(PACK_MRG_INFO *info,byte *buf); +static int mrg_rrnd(PACK_MRG_INFO *info,uchar *buf); static void mrg_reset(PACK_MRG_INFO *mrg); #if !defined(DBUG_OFF) static void fakebigcodes(HUFF_COUNTS *huff_counts, HUFF_COUNTS *end_count); @@ -259,10 +259,10 @@ static struct my_option my_long_options[] = 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, #endif {"backup", 'b', "Make a backup of the table as table_name.OLD.", - (gptr*) &backup, (gptr*) &backup, 0, GET_BOOL, NO_ARG, 0, 0, 0, 0, 0, 0}, + (uchar**) &backup, (uchar**) &backup, 0, GET_BOOL, NO_ARG, 0, 0, 0, 0, 0, 0}, {"character-sets-dir", OPT_CHARSETS_DIR_MP, - "Directory where character sets are.", (gptr*) &charsets_dir, - (gptr*) &charsets_dir, 0, GET_STR, REQUIRED_ARG, 0, 0, 0, 0, 0, 0}, + "Directory where character sets are.", (uchar**) &charsets_dir, + (uchar**) &charsets_dir, 0, GET_STR, REQUIRED_ARG, 0, 0, 0, 0, 0, 0}, {"debug", '#', "Output debug log. Often this is 'd:t:o,filename'.", 0, 0, 0, GET_STR, OPT_ARG, 0, 0, 0, 0, 0, 0}, {"force", 'f', @@ -270,7 +270,7 @@ static struct my_option my_long_options[] = 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, {"join", 'j', "Join all given tables into 'new_table_name'. All tables MUST have identical layouts.", - (gptr*) &join_table, (gptr*) &join_table, 0, GET_STR, REQUIRED_ARG, 0, 0, 0, + (uchar**) &join_table, (uchar**) &join_table, 0, GET_STR, REQUIRED_ARG, 0, 0, 0, 0, 0, 0}, {"help", '?', "Display this help and exit.", 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, @@ -284,8 +284,8 @@ static struct my_option my_long_options[] = 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, {"version", 'V', "Output version information and exit.", 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, - {"wait", 'w', "Wait and retry if table is in use.", (gptr*) &opt_wait, - (gptr*) &opt_wait, 0, GET_BOOL, NO_ARG, 0, 0, 0, 0, 0, 0}, + {"wait", 'w', "Wait and retry if table is in use.", (uchar**) &opt_wait, + (uchar**) &opt_wait, 0, GET_BOOL, NO_ARG, 0, 0, 0, 0, 0, 0}, { 0, 0, 0, 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0} }; @@ -481,7 +481,7 @@ static bool open_isam_files(PACK_MRG_INFO *mrg,char **names,uint count) error: while (i--) maria_close(mrg->file[i]); - my_free((gptr) mrg->file,MYF(0)); + my_free((uchar*) mrg->file,MYF(0)); return 1; } @@ -606,14 +606,14 @@ static int compress(PACK_MRG_INFO *mrg,char *result_table) /* If the packed lengths of combined columns is less then the sum of the non-combined columns, then create common Huffman trees for them. - We do this only for byte compressed columns, not for distinct values + We do this only for uchar compressed columns, not for distinct values compressed columns. */ if ((int) (used_trees=join_same_trees(huff_counts,trees)) < 0) goto err; /* - Assign codes to all byte or column values. + Assign codes to all uchar or column values. */ if (make_huff_decode_table(huff_trees,fields)) goto err; @@ -827,11 +827,11 @@ static void free_counts_and_tree_and_queue(HUFF_TREE *huff_trees, uint trees, for (i=0 ; i < trees ; i++) { if (huff_trees[i].element_buffer) - my_free((gptr) huff_trees[i].element_buffer,MYF(0)); + my_free((uchar*) huff_trees[i].element_buffer,MYF(0)); if (huff_trees[i].code) - my_free((gptr) huff_trees[i].code,MYF(0)); + my_free((uchar*) huff_trees[i].code,MYF(0)); } - my_free((gptr) huff_trees,MYF(0)); + my_free((uchar*) huff_trees,MYF(0)); } if (huff_counts) { @@ -839,11 +839,11 @@ static void free_counts_and_tree_and_queue(HUFF_TREE *huff_trees, uint trees, { if (huff_counts[i].tree_buff) { - my_free((gptr) huff_counts[i].tree_buff,MYF(0)); + my_free((uchar*) huff_counts[i].tree_buff,MYF(0)); delete_tree(&huff_counts[i].int_tree); } } - my_free((gptr) huff_counts,MYF(0)); + my_free((uchar*) huff_counts,MYF(0)); } delete_queue(&queue); /* This is safe to free */ return; @@ -856,16 +856,16 @@ static int get_statistic(PACK_MRG_INFO *mrg,HUFF_COUNTS *huff_counts) int error; uint length, null_bytes; ulong reclength,max_blob_length; - byte *record,*pos,*next_pos,*end_pos,*start_pos; + uchar *record,*pos,*next_pos,*end_pos,*start_pos; ha_rows record_count; HUFF_COUNTS *count,*end_count; TREE_ELEMENT *element; - ha_checksum(*calc_checksum) (struct st_maria_info *, const byte *); + ha_checksum(*calc_checksum) (struct st_maria_info *, const uchar *); DBUG_ENTER("get_statistic"); reclength= mrg->file[0]->s->base.reclength; null_bytes= mrg->file[0]->s->base.null_bytes; - record=(byte*) my_alloca(reclength); + record=(uchar*) my_alloca(reclength); end_count=huff_counts+mrg->file[0]->s->base.fields; record_count=0; glob_crc=0; max_blob_length=0; @@ -1040,7 +1040,7 @@ static int get_statistic(PACK_MRG_INFO *mrg,HUFF_COUNTS *huff_counts) { uint i; /* Zero fields are just counted. Go to the next record. */ - if (!memcmp((byte*) start_pos,zero_string,count->field_length)) + if (!memcmp((uchar*) start_pos,zero_string,count->field_length)) { count->zero_fields++; continue; @@ -1063,7 +1063,7 @@ static int get_statistic(PACK_MRG_INFO *mrg,HUFF_COUNTS *huff_counts) continue; /* - Count the incidence of every byte value in the + Count the incidence of every uchar value in the significant field value. */ for ( ; pos < end_pos ; pos++) @@ -1102,10 +1102,10 @@ static int get_statistic(PACK_MRG_INFO *mrg,HUFF_COUNTS *huff_counts) DBUG_EXECUTE_IF("fakebigcodes", fakebigcodes(huff_counts, end_count);); DBUG_PRINT("info", ("Found the following number of incidents " - "of the byte codes:")); + "of the uchar codes:")); if (verbose >= 2) VOID(printf("Found the following number of incidents " - "of the byte codes:\n")); + "of the uchar codes:\n")); for (count= huff_counts ; count < end_count; count++) { uint idx; @@ -1149,12 +1149,12 @@ static int get_statistic(PACK_MRG_INFO *mrg,HUFF_COUNTS *huff_counts) mrg->records=record_count; mrg->max_blob_length=max_blob_length; - my_afree((gptr) record); + my_afree((uchar*) record); DBUG_RETURN(error != HA_ERR_END_OF_FILE); } static int compare_huff_elements(void *not_used __attribute__((unused)), - byte *a, byte *b) + uchar *a, uchar *b) { return *((my_off_t*) a) < *((my_off_t*) b) ? -1 : (*((my_off_t*) a) == *((my_off_t*) b) ? 0 : 1); @@ -1170,7 +1170,7 @@ static void check_counts(HUFF_COUNTS *huff_counts, uint trees, my_off_t old_length,new_length,length; DBUG_ENTER("check_counts"); - bzero((gptr) field_count,sizeof(field_count)); + bzero((uchar*) field_count,sizeof(field_count)); space_fields=fill_zero_fields=0; for (; trees-- ; huff_counts++) @@ -1336,12 +1336,12 @@ static void check_counts(HUFF_COUNTS *huff_counts, uint trees, } else { - my_free((gptr) huff_counts->tree_buff,MYF(0)); + my_free((uchar*) huff_counts->tree_buff,MYF(0)); delete_tree(&huff_counts->int_tree); huff_counts->tree_buff=0; } if (tree.element_buffer) - my_free((gptr) tree.element_buffer,MYF(0)); + my_free((uchar*) tree.element_buffer,MYF(0)); } if (huff_counts->pack_type & PACK_TYPE_SPACE_FIELDS) space_fields++; @@ -1459,8 +1459,8 @@ static HUFF_TREE* make_huff_trees(HUFF_COUNTS *huff_counts, uint trees) if (make_huff_tree(huff_tree+tree,huff_counts+tree)) { while (tree--) - my_free((gptr) huff_tree[tree].element_buffer,MYF(0)); - my_free((gptr) huff_tree,MYF(0)); + my_free((uchar*) huff_tree[tree].element_buffer,MYF(0)); + my_free((uchar*) huff_tree,MYF(0)); DBUG_RETURN(0); } } @@ -1502,7 +1502,7 @@ static int make_huff_tree(HUFF_TREE *huff_tree, HUFF_COUNTS *huff_counts) } else { - /* Count the number of byte codes found in the column. */ + /* Count the number of uchar codes found in the column. */ for (i=found=0 ; i < 256 ; i++) { if (huff_counts->counts[i]) @@ -1535,7 +1535,7 @@ static int make_huff_tree(HUFF_TREE *huff_tree, HUFF_COUNTS *huff_counts) { HUFF_ELEMENT *temp; if (!(temp= - (HUFF_ELEMENT*) my_realloc((gptr) huff_tree->element_buffer, + (HUFF_ELEMENT*) my_realloc((uchar*) huff_tree->element_buffer, found*2*sizeof(HUFF_ELEMENT), MYF(MY_WME)))) return 1; @@ -1570,7 +1570,7 @@ static int make_huff_tree(HUFF_TREE *huff_tree, HUFF_COUNTS *huff_counts) */ tree_walk(&huff_counts->int_tree, (int (*)(void*, element_count,void*)) save_counts_in_queue, - (gptr) huff_tree, left_root_right); + (uchar*) huff_tree, left_root_right); } else { @@ -1580,7 +1580,7 @@ static int make_huff_tree(HUFF_TREE *huff_tree, HUFF_COUNTS *huff_counts) (huff_tree->offset_bits+1)* (found-2)+7)/8; /* - Put a HUFF_ELEMENT into the queue for every byte code found in the column. + Put a HUFF_ELEMENT into the queue for every uchar code found in the column. The elements are taken from the target trees element buffer. Instead of using queue_insert(), we just place references to the @@ -1596,11 +1596,11 @@ static int make_huff_tree(HUFF_TREE *huff_tree, HUFF_COUNTS *huff_counts) new_huff_el->count=huff_counts->counts[i]; new_huff_el->a.leaf.null=0; new_huff_el->a.leaf.element_nr=i; - queue.root[found]=(byte*) new_huff_el; + queue.root[found]=(uchar*) new_huff_el; } } /* - If there is only a single byte value in this field in all records, + If there is only a single uchar value in this field in all records, add a second element with zero incidence. This is required to enter the loop, which builds the Huffman tree. */ @@ -1613,7 +1613,7 @@ static int make_huff_tree(HUFF_TREE *huff_tree, HUFF_COUNTS *huff_counts) new_huff_el->a.leaf.element_nr=huff_tree->min_chr=last-1; else new_huff_el->a.leaf.element_nr=huff_tree->max_chr=last+1; - queue.root[found]=(byte*) new_huff_el; + queue.root[found]=(uchar*) new_huff_el; } } @@ -1650,7 +1650,7 @@ static int make_huff_tree(HUFF_TREE *huff_tree, HUFF_COUNTS *huff_counts) The Huffman algorithm assigns another bit to the code for a byte every time that bytes incidence is combined (directly or indirectly) to a new element as one of the two least incidence elements. - This means that one more bit per incidence of that byte is required + This means that one more bit per incidence of that uchar is required in the resulting file. So we add the new combined incidence as the number of bits by which the result grows. */ @@ -1663,7 +1663,7 @@ static int make_huff_tree(HUFF_TREE *huff_tree, HUFF_COUNTS *huff_counts) Replace the copied top element by the new element and re-order the queue. */ - queue.root[1]=(byte*) new_huff_el; + queue.root[1]=(uchar*) new_huff_el; queue_replaced(&queue); } huff_tree->root=(HUFF_ELEMENT*) queue.root[1]; @@ -1702,7 +1702,7 @@ static int compare_tree(void* cmp_arg __attribute__((unused)), 0 */ -static int save_counts_in_queue(byte *key, element_count count, +static int save_counts_in_queue(uchar *key, element_count count, HUFF_TREE *tree) { HUFF_ELEMENT *new_huff_el; @@ -1712,7 +1712,7 @@ static int save_counts_in_queue(byte *key, element_count count, new_huff_el->a.leaf.null=0; new_huff_el->a.leaf.element_nr= (uint) (key- tree->counts->tree_buff) / tree->counts->field_length; - queue.root[tree->elements]=(byte*) new_huff_el; + queue.root[tree->elements]=(uchar*) new_huff_el; return 0; } @@ -1727,7 +1727,7 @@ static int save_counts_in_queue(byte *key, element_count count, DESCRIPTION We need to follow the Huffman algorithm until we know, how many bits - are required for each byte code. But we do not need the resulting + are required for each uchar code. But we do not need the resulting Huffman tree. Hence, we can leave out some steps which are essential in make_huff_tree(). @@ -1746,7 +1746,7 @@ static my_off_t calc_packed_length(HUFF_COUNTS *huff_counts, /* WARNING: We use a small hack for efficiency: Instead of placing references to HUFF_ELEMENTs into the queue, we just insert - references to the counts of the byte codes which appeared in this + references to the counts of the uchar codes which appeared in this table column. During the Huffman algorithm they are successively replaced by references to HUFF_ELEMENTs. This works, because HUFF_ELEMENTs have the incidence count at their beginning. @@ -1756,7 +1756,7 @@ static my_off_t calc_packed_length(HUFF_COUNTS *huff_counts, same type. Instead of using queue_insert(), we just copy the references into - the buffer of the priority queue. We insert in byte value order, but + the buffer of the priority queue. We insert in uchar value order, but the order is in fact irrelevant here. We will establish the correct order later. */ @@ -1769,18 +1769,18 @@ static my_off_t calc_packed_length(HUFF_COUNTS *huff_counts, first=i; last=i; /* We start with root[1], which is the queues top element. */ - queue.root[found]=(byte*) &huff_counts->counts[i]; + queue.root[found]=(uchar*) &huff_counts->counts[i]; } } if (!found) DBUG_RETURN(0); /* Empty tree */ /* - If there is only a single byte value in this field in all records, + If there is only a single uchar value in this field in all records, add a second element with zero incidence. This is required to enter the loop, which follows the Huffman algorithm. */ if (found < 2) - queue.root[++found]=(byte*) &huff_counts->counts[last ? 0 : 1]; + queue.root[++found]=(uchar*) &huff_counts->counts[last ? 0 : 1]; /* Make a queue from the queue buffer. */ queue.elements=found; @@ -1824,7 +1824,7 @@ static my_off_t calc_packed_length(HUFF_COUNTS *huff_counts, The Huffman algorithm assigns another bit to the code for a byte every time that bytes incidence is combined (directly or indirectly) to a new element as one of the two least incidence elements. - This means that one more bit per incidence of that byte is required + This means that one more bit per incidence of that uchar is required in the resulting file. So we add the new combined incidence as the number of bits by which the result grows. */ @@ -1835,7 +1835,7 @@ static my_off_t calc_packed_length(HUFF_COUNTS *huff_counts, queue. This successively replaces the references to counts by references to HUFF_ELEMENTs. */ - queue.root[1]=(byte*) new_huff_el; + queue.root[1]=(uchar*) new_huff_el; queue_replaced(&queue); } DBUG_RETURN(bytes_packed+(bits_packed+7)/8); @@ -1868,12 +1868,12 @@ static uint join_same_trees(HUFF_COUNTS *huff_counts, uint trees) i->tree->tree_pack_length+j->tree->tree_pack_length+ ALLOWED_JOIN_DIFF) { - memcpy_fixed((byte*) i->counts,(byte*) count.counts, + memcpy_fixed((uchar*) i->counts,(uchar*) count.counts, sizeof(count.counts[0])*256); - my_free((gptr) j->tree->element_buffer,MYF(0)); + my_free((uchar*) j->tree->element_buffer,MYF(0)); j->tree->element_buffer=0; j->tree=i->tree; - bmove((byte*) i->counts,(byte*) count.counts, + bmove((uchar*) i->counts,(uchar*) count.counts, sizeof(count.counts[0])*256); if (make_huff_tree(i->tree,i)) return (uint) -1; @@ -2016,7 +2016,7 @@ static char *hexdigits(ulonglong value) static int write_header(PACK_MRG_INFO *mrg,uint head_length,uint trees, my_off_t tot_elements,my_off_t filelength) { - byte *buff= (byte*) file_buffer.pos; + uchar *buff= (uchar*) file_buffer.pos; bzero(buff,HEAD_LENGTH); memcpy_fixed(buff,maria_pack_file_magic,4); @@ -2032,7 +2032,7 @@ static int write_header(PACK_MRG_INFO *mrg,uint head_length,uint trees, if (test_only) return 0; VOID(my_seek(file_buffer.file,0L,MY_SEEK_SET,MYF(0))); - return my_write(file_buffer.file,(const byte *) file_buffer.pos,HEAD_LENGTH, + return my_write(file_buffer.file,(const uchar *) file_buffer.pos,HEAD_LENGTH, MYF(MY_WME | MY_NABP | MY_WAIT_IF_FULL)) != 0; } @@ -2168,7 +2168,7 @@ static my_off_t write_huff_tree(HUFF_TREE *huff_tree, uint trees) { /* This should be impossible */ VOID(fprintf(stderr, "Tree offset got too big: %d, aborted\n", huff_tree->max_offset)); - my_afree((gptr) packed_tree); + my_afree((uchar*) packed_tree); return 0; } @@ -2179,7 +2179,7 @@ static my_off_t write_huff_tree(HUFF_TREE *huff_tree, uint trees) huff_tree->char_bits)); if (!huff_tree->counts->tree_buff) { - /* We do a byte compression on this column. Mark with bit 0. */ + /* We do a uchar compression on this column. Mark with bit 0. */ write_bits(0,1); write_bits(huff_tree->min_chr,8); write_bits(huff_tree->elements,9); @@ -2340,7 +2340,7 @@ static my_off_t write_huff_tree(HUFF_TREE *huff_tree, uint trees) DBUG_PRINT("info", (" ")); if (verbose >= 2) VOID(printf("\n")); - my_afree((gptr) packed_tree); + my_afree((uchar*) packed_tree); if (errors) { VOID(fprintf(stderr, "Error: Generated decode trees are corrupt. Stop.\n")); @@ -2366,7 +2366,7 @@ static uint *make_offset_code_tree(HUFF_TREE *huff_tree, HUFF_ELEMENT *element, */ if (!element->a.nod.left->a.leaf.null) { - /* Store the byte code or the index of the column value. */ + /* Store the uchar code or the index of the column value. */ prev_offset[0] =(uint) element->a.nod.left->a.leaf.element_nr; offset+=2; } @@ -2374,7 +2374,7 @@ static uint *make_offset_code_tree(HUFF_TREE *huff_tree, HUFF_ELEMENT *element, { /* Recursively traverse the tree to the left. Mark it as an offset to - another tree node (in contrast to a byte code or column value index). + another tree node (in contrast to a uchar code or column value index). */ prev_offset[0]= IS_OFFSET+2; offset=make_offset_code_tree(huff_tree,element->a.nod.left,offset+2); @@ -2383,7 +2383,7 @@ static uint *make_offset_code_tree(HUFF_TREE *huff_tree, HUFF_ELEMENT *element, /* Now, check the right child. */ if (!element->a.nod.right->a.leaf.null) { - /* Store the byte code or the index of the column value. */ + /* Store the uchar code or the index of the column value. */ prev_offset[1]=element->a.nod.right->a.leaf.element_nr; return offset; } @@ -2391,7 +2391,7 @@ static uint *make_offset_code_tree(HUFF_TREE *huff_tree, HUFF_ELEMENT *element, { /* Recursively traverse the tree to the right. Mark it as an offset to - another tree node (in contrast to a byte code or column value index). + another tree node (in contrast to a uchar code or column value index). */ uint temp=(uint) (offset-prev_offset-1); prev_offset[1]= IS_OFFSET+ temp; @@ -2421,7 +2421,7 @@ static int compress_isam_file(PACK_MRG_INFO *mrg, HUFF_COUNTS *huff_counts) my_off_t record_count; char llbuf[32]; ulong length,pack_length; - byte *record,*pos,*end_pos,*record_pos,*start_pos; + uchar *record,*pos,*end_pos,*record_pos,*start_pos; HUFF_COUNTS *count,*end_count; HUFF_TREE *tree; MARIA_HA *isam_file=mrg->file[0]; @@ -2429,7 +2429,7 @@ static int compress_isam_file(PACK_MRG_INFO *mrg, HUFF_COUNTS *huff_counts) DBUG_ENTER("compress_isam_file"); /* Allocate a buffer for the records (excluding blobs). */ - if (!(record=(byte*) my_alloca(isam_file->s->base.reclength))) + if (!(record=(uchar*) my_alloca(isam_file->s->base.reclength))) return -1; end_count=huff_counts+isam_file->s->base.fields; @@ -2482,7 +2482,7 @@ static int compress_isam_file(PACK_MRG_INFO *mrg, HUFF_COUNTS *huff_counts) { if (flush_buffer((ulong) max_calc_length + (ulong) max_pack_length)) break; - record_pos= (byte*) file_buffer.pos; + record_pos= (uchar*) file_buffer.pos; file_buffer.pos+= max_pack_length; if (null_bytes) { @@ -2527,7 +2527,7 @@ static int compress_isam_file(PACK_MRG_INFO *mrg, HUFF_COUNTS *huff_counts) switch (count->field_type) { case FIELD_SKIP_ZERO: - if (!memcmp((byte*) start_pos,zero_string,field_length)) + if (!memcmp((uchar*) start_pos,zero_string,field_length)) { DBUG_PRINT("fields", ("FIELD_SKIP_ZERO zeroes only, bits: 1")); write_bits(1,1); @@ -2656,7 +2656,7 @@ static int compress_isam_file(PACK_MRG_INFO *mrg, HUFF_COUNTS *huff_counts) break; case FIELD_INTERVALL: global_count=count; - pos=(byte*) tree_search(&count->int_tree, start_pos, + pos=(uchar*) tree_search(&count->int_tree, start_pos, count->int_tree.custom_arg); intervall=(uint) (pos - count->tree_buff)/field_length; DBUG_PRINT("fields", ("FIELD_INTERVALL")); @@ -2679,7 +2679,7 @@ static int compress_isam_file(PACK_MRG_INFO *mrg, HUFF_COUNTS *huff_counts) } else { - byte *blob,*blob_end; + uchar *blob,*blob_end; DBUG_PRINT("fields", ("FIELD_BLOB not empty, bits: 1")); write_bits(0,1); /* Write the blob length. */ @@ -2720,7 +2720,7 @@ static int compress_isam_file(PACK_MRG_INFO *mrg, HUFF_COUNTS *huff_counts) } else { - byte *end= start_pos + var_pack_length + col_length; + uchar *end= start_pos + var_pack_length + col_length; DBUG_PRINT("fields", ("FIELD_VARCHAR not empty, bits: 1")); write_bits(0,1); /* Write the varchar length. */ @@ -2752,7 +2752,7 @@ static int compress_isam_file(PACK_MRG_INFO *mrg, HUFF_COUNTS *huff_counts) DBUG_PRINT("fields", ("---")); } flush_bits(); - length=(ulong) ((byte*) file_buffer.pos - record_pos) - max_pack_length; + length=(ulong) ((uchar*) file_buffer.pos - record_pos) - max_pack_length; pack_length= _ma_save_pack_length(pack_version, record_pos, length); if (pack_blob_length) pack_length+= _ma_save_pack_length(pack_version, @@ -2793,7 +2793,7 @@ static int compress_isam_file(PACK_MRG_INFO *mrg, HUFF_COUNTS *huff_counts) if (verbose >= 2) VOID(printf("wrote %s records.\n", llstr((longlong) record_count, llbuf))); - my_afree((gptr) record); + my_afree((uchar*) record); mrg->ref_length=max_pack_length; mrg->min_pack_length=max_record_length ? min_record_length : 0; mrg->max_pack_length=max_record_length; @@ -2843,7 +2843,7 @@ static int flush_buffer(ulong neaded_length) /* file_buffer.end is 8 bytes lower than the real end of the buffer. This is done so that the end-of-buffer condition does not need to be - checked for every byte (see write_bits()). Consequently, + checked for every uchar (see write_bits()). Consequently, file_buffer.pos can become greater than file_buffer.end. The algorithms in the other functions ensure that there will never be more than 8 bytes written to the buffer without an end-of-buffer @@ -2860,7 +2860,7 @@ static int flush_buffer(ulong neaded_length) if (test_only) return 0; if (error_on_write|| my_write(file_buffer.file, - (const byte*) file_buffer.buffer, + (const uchar*) file_buffer.buffer, length, MYF(MY_WME | MY_NABP | MY_WAIT_IF_FULL))) { @@ -2887,7 +2887,7 @@ static int flush_buffer(ulong neaded_length) static void end_file_buffer(void) { - my_free((gptr) file_buffer.buffer,MYF(0)); + my_free((uchar*) file_buffer.buffer,MYF(0)); } /* output `bits` low bits of `value' */ @@ -3048,7 +3048,7 @@ static void mrg_reset(PACK_MRG_INFO *mrg) } } -static int mrg_rrnd(PACK_MRG_INFO *info,byte *buf) +static int mrg_rrnd(PACK_MRG_INFO *info,uchar *buf) { int error; MARIA_HA *isam_info; @@ -3095,7 +3095,7 @@ static int mrg_close(PACK_MRG_INFO *mrg) for (i=0 ; i < mrg->count ; i++) error|=maria_close(mrg->file[i]); if (mrg->free_file) - my_free((gptr) mrg->file,MYF(0)); + my_free((uchar*) mrg->file,MYF(0)); DBUG_RETURN(error); } @@ -3127,8 +3127,8 @@ static int mrg_close(PACK_MRG_INFO *mrg) To get 64(32)-bit codes, I sort the counts by decreasing incidence. I assign counts of 1 to the two most frequent values, a count of 2 for the next one, then 4, 8, and so on until 2**64-1(2**30-1). All - the remaining values get 1. That way every possible byte has an - assigned code, though not all codes are used if not all byte values + the remaining values get 1. That way every possible uchar has an + assigned code, though not all codes are used if not all uchar values are present in the column. This strategy would work with distinct column values too, but @@ -3158,7 +3158,7 @@ static void fakebigcodes(HUFF_COUNTS *huff_counts, HUFF_COUNTS *end_count) */ if (huff_counts->tree_buff) { - my_free((gptr) huff_counts->tree_buff, MYF(0)); + my_free((uchar*) huff_counts->tree_buff, MYF(0)); delete_tree(&huff_counts->int_tree); huff_counts->tree_buff= NULL; DBUG_PRINT("fakebigcodes", ("freed distinct column values")); diff --git a/storage/maria/tablockman.c b/storage/maria/tablockman.c index 4634f60a085..b7e2d62e1ab 100644 --- a/storage/maria/tablockman.c +++ b/storage/maria/tablockman.c @@ -231,7 +231,7 @@ static inline TABLE_LOCK *find_by_loid(LOCKED_TABLE *table, uint16 loid) { return (TABLE_LOCK *)hash_search(& table->latest_locks, - (byte *)& loid, sizeof(loid)); + (uchar *)& loid, sizeof(loid)); } static inline @@ -487,8 +487,8 @@ tablockman_getlock(TABLOCKMAN *lm, TABLE_LOCK_OWNER *lo, /* update the latest_locks hash */ if (old) - hash_delete(& table->latest_locks, (byte *)old); - hash_insert(& table->latest_locks, (byte *)new); + hash_delete(& table->latest_locks, (uchar *)old); + hash_insert(& table->latest_locks, (uchar *)new); new->upgraded_from= old; @@ -571,7 +571,7 @@ void tablockman_release_locks(TABLOCKMAN *lm, TABLE_LOCK_OWNER *lo) /* TODO ? group locks by table to reduce the number of mutex locks */ pthread_mutex_lock(mutex); - hash_delete(& cur->table->latest_locks, (byte *)cur); + hash_delete(& cur->table->latest_locks, (uchar *)cur); if (cur->prev) cur->prev->next= cur->next; diff --git a/storage/maria/trnman.c b/storage/maria/trnman.c index 83249ab328f..4f009a7d5a8 100644 --- a/storage/maria/trnman.c +++ b/storage/maria/trnman.c @@ -94,11 +94,11 @@ static TRN *short_trid_to_TRN(uint16 short_trid) } #endif -static byte *trn_get_hash_key(const byte *trn, uint* len, +static uchar *trn_get_hash_key(const uchar *trn, uint* len, my_bool unused __attribute__ ((unused))) { *len= sizeof(TrID); - return (byte *) & ((*((TRN **)trn))->trid); + return (uchar *) & ((*((TRN **)trn))->trid); } int trnman_init() diff --git a/storage/maria/trnman.h b/storage/maria/trnman.h index 1a4423f2a11..fce02d9ab89 100644 --- a/storage/maria/trnman.h +++ b/storage/maria/trnman.h @@ -24,7 +24,7 @@ C_MODE_START #include "ma_loghandler_lsn.h" /* - trid - 6 byte transaction identifier. Assigned when a transaction + trid - 6 uchar transaction identifier. Assigned when a transaction is created. Transaction can always be identified by its trid, even after transaction has ended. diff --git a/storage/maria/unittest/ma_pagecache_consist.c b/storage/maria/unittest/ma_pagecache_consist.c index 39170693573..54491a09c3b 100644 --- a/storage/maria/unittest/ma_pagecache_consist.c +++ b/storage/maria/unittest/ma_pagecache_consist.c @@ -125,7 +125,7 @@ uint check_page(uchar *buff, ulong offset, int page_locked, int page_no, (page_locked ? "locked" : "unlocked"), end, num, tag); h= my_open("wrong_page", O_CREAT | O_TRUNC | O_RDWR, MYF(0)); - my_pwrite(h, (byte*) buff, PAGE_SIZE, 0, MYF(0)); + my_pwrite(h, (uchar*) buff, PAGE_SIZE, 0, MYF(0)); my_close(h, MYF(0)); goto err; } @@ -264,7 +264,7 @@ static void *test_thread_reader(void *arg) thread_count--; VOID(pthread_cond_signal(&COND_thread_count)); /* Tell main we are ready */ pthread_mutex_unlock(&LOCK_thread_count); - free((gptr) arg); + free((uchar*) arg); my_thread_end(); DBUG_RETURN(0); } @@ -284,7 +284,7 @@ static void *test_thread_writer(void *arg) thread_count--; VOID(pthread_cond_signal(&COND_thread_count)); /* Tell main we are ready */ pthread_mutex_unlock(&LOCK_thread_count); - free((gptr) arg); + free((uchar*) arg); my_thread_end(); DBUG_RETURN(0); } diff --git a/storage/maria/unittest/ma_pagecache_single.c b/storage/maria/unittest/ma_pagecache_single.c index 7b77315e18c..8add95e8a36 100644 --- a/storage/maria/unittest/ma_pagecache_single.c +++ b/storage/maria/unittest/ma_pagecache_single.c @@ -448,7 +448,7 @@ static void *test_thread(void *arg) thread_count--; VOID(pthread_cond_signal(&COND_thread_count)); /* Tell main we are ready */ pthread_mutex_unlock(&LOCK_thread_count); - free((gptr) arg); + free((uchar*) arg); my_thread_end(); DBUG_RETURN(0); } diff --git a/storage/maria/unittest/ma_test_loghandler-t.c b/storage/maria/unittest/ma_test_loghandler-t.c index e31136d52ec..40f9e72c3b2 100644 --- a/storage/maria/unittest/ma_test_loghandler-t.c +++ b/storage/maria/unittest/ma_test_loghandler-t.c @@ -56,10 +56,10 @@ static TRN *trn= &dummy_transaction_object; 1 - Error */ -static my_bool check_content(byte *ptr, ulong length) +static my_bool check_content(uchar *ptr, ulong length) { ulong i; - byte buff[2]; + uchar buff[2]; for (i= 0; i < length; i++) { if (i % 2 == 0) @@ -107,7 +107,7 @@ void read_ok(TRANSLOG_HEADER_BUFFER *rec) */ static my_bool read_and_check_content(TRANSLOG_HEADER_BUFFER *rec, - byte *buffer, uint skip) + uchar *buffer, uint skip) { DBUG_ASSERT(rec->record_length < LONG_BUFFER_SIZE * 2 + 7 * 2 + 2); if (translog_read_record(rec->lsn, 0, rec->record_length, buffer, NULL) != @@ -122,14 +122,14 @@ int main(int argc __attribute__((unused)), char *argv[]) uint32 i; uint32 rec_len; uint pagen; - byte long_tr_id[6]; - byte lsn_buff[23]= + uchar long_tr_id[6]; + uchar lsn_buff[23]= { 0x55, 0xAA, 0x55, 0xAA, 0x55, 0xAA, 0x55, 0xAA, 0x55, 0xAA, 0x55, 0xAA, 0x55, 0xAA, 0x55, 0xAA, 0x55, 0xAA, 0x55, 0xAA, 0x55, 0xAA, 0x55 }; - byte long_buffer[LONG_BUFFER_SIZE * 2 + LSN_STORE_SIZE * 2 + 2]; + uchar long_buffer[LONG_BUFFER_SIZE * 2 + LSN_STORE_SIZE * 2 + 2]; PAGECACHE pagecache; LSN lsn, lsn_base, first_lsn; TRANSLOG_HEADER_BUFFER rec; diff --git a/storage/maria/unittest/ma_test_loghandler_multigroup-t.c b/storage/maria/unittest/ma_test_loghandler_multigroup-t.c index 1281ee425d8..6a27321ec98 100644 --- a/storage/maria/unittest/ma_test_loghandler_multigroup-t.c +++ b/storage/maria/unittest/ma_test_loghandler_multigroup-t.c @@ -44,10 +44,10 @@ static TRN *trn= &dummy_transaction_object; 1 - Error */ -static my_bool check_content(byte *ptr, ulong length) +static my_bool check_content(uchar *ptr, ulong length) { ulong i; - byte buff[4]; + uchar buff[4]; DBUG_ENTER("check_content"); for (i= 0; i < length; i++) { @@ -81,7 +81,7 @@ static my_bool check_content(byte *ptr, ulong length) */ static my_bool read_and_check_content(TRANSLOG_HEADER_BUFFER *rec, - byte *buffer, uint skip) + uchar *buffer, uint skip) { int res= 0; translog_size_t len; @@ -115,14 +115,14 @@ int main(int argc __attribute__((unused)), char *argv[]) uint32 i; uint32 rec_len; uint pagen; - byte long_tr_id[6]; - byte lsn_buff[23]= + uchar long_tr_id[6]; + uchar lsn_buff[23]= { 0x55, 0xAA, 0x55, 0xAA, 0x55, 0xAA, 0x55, 0xAA, 0x55, 0xAA, 0x55, 0xAA, 0x55, 0xAA, 0x55, 0xAA, 0x55, 0xAA, 0x55, 0xAA, 0x55, 0xAA, 0x55 }; - byte *long_buffer= malloc(LONG_BUFFER_SIZE + LSN_STORE_SIZE * 2 + 2); + uchar *long_buffer= malloc(LONG_BUFFER_SIZE + LSN_STORE_SIZE * 2 + 2); PAGECACHE pagecache; LSN lsn, lsn_base, first_lsn; TRANSLOG_HEADER_BUFFER rec; @@ -138,7 +138,7 @@ int main(int argc __attribute__((unused)), char *argv[]) exit(1); { - byte buff[4]; + uchar buff[4]; for (i= 0; i < (LONG_BUFFER_SIZE + LSN_STORE_SIZE * 2 + 2); i++) { if (i % 4 == 0) diff --git a/storage/maria/unittest/ma_test_loghandler_multithread-t.c b/storage/maria/unittest/ma_test_loghandler_multithread-t.c index ff966160acc..bf3ede113c0 100644 --- a/storage/maria/unittest/ma_test_loghandler_multithread-t.c +++ b/storage/maria/unittest/ma_test_loghandler_multithread-t.c @@ -30,7 +30,7 @@ static uint thread_count; static ulong lens[WRITERS][ITERATIONS]; static LSN lsns1[WRITERS][ITERATIONS]; static LSN lsns2[WRITERS][ITERATIONS]; -static byte *long_buffer; +static uchar *long_buffer; /* Get pseudo-random length of the field in @@ -68,7 +68,7 @@ static uint32 get_len() 1 - Error */ -static my_bool check_content(byte *ptr, ulong length) +static my_bool check_content(uchar *ptr, ulong length) { ulong i; for (i= 0; i < length; i++) @@ -100,7 +100,7 @@ static my_bool check_content(byte *ptr, ulong length) static my_bool read_and_check_content(TRANSLOG_HEADER_BUFFER *rec, - byte *buffer, uint skip) + uchar *buffer, uint skip) { int res= 0; translog_size_t len; @@ -120,7 +120,7 @@ void writer(int num) { LSN lsn; TRN trn; - byte long_tr_id[6]; + uchar long_tr_id[6]; uint i; trn.short_id= num; @@ -186,7 +186,7 @@ static void *test_thread_writer(void *arg) VOID(pthread_cond_signal(&COND_thread_count)); /* Tell main we are ready */ pthread_mutex_unlock(&LOCK_thread_count); - free((gptr) arg); + free((uchar*) arg); my_thread_end(); return(0); } @@ -292,7 +292,7 @@ int main(int argc __attribute__((unused)), srandom(122334817L); { LEX_STRING parts[TRANSLOG_INTERNAL_PARTS + 1]; - byte long_tr_id[6]= + uchar long_tr_id[6]= { 0x11, 0x22, 0x33, 0x44, 0x55, 0x66 }; diff --git a/storage/maria/unittest/ma_test_loghandler_pagecache-t.c b/storage/maria/unittest/ma_test_loghandler_pagecache-t.c index 35e05f9c997..327b8300fbb 100644 --- a/storage/maria/unittest/ma_test_loghandler_pagecache-t.c +++ b/storage/maria/unittest/ma_test_loghandler_pagecache-t.c @@ -22,7 +22,7 @@ static PAGECACHE_FILE file1; int main(int argc __attribute__((unused)), char *argv[]) { uint pagen; - byte long_tr_id[6]; + uchar long_tr_id[6]; PAGECACHE pagecache; LSN lsn; MY_STAT st, *stat; diff --git a/storage/myisam/ft_myisam.c b/storage/myisam/ft_myisam.c index 76c04ba4c0b..bef3fbfd5f5 100644 --- a/storage/myisam/ft_myisam.c +++ b/storage/myisam/ft_myisam.c @@ -23,8 +23,8 @@ #include "ftdefs.h" FT_INFO *ft_init_search(uint flags, void *info, uint keynr, - byte *query, uint query_len, CHARSET_INFO *cs, - byte *record) + uchar *query, uint query_len, CHARSET_INFO *cs, + uchar *record) { FT_INFO *res; if (flags & FT_BOOL) diff --git a/storage/myisam/ha_myisam.cc b/storage/myisam/ha_myisam.cc index 3472bbcabd5..4f02705c18c 100644 --- a/storage/myisam/ha_myisam.cc +++ b/storage/myisam/ha_myisam.cc @@ -1244,7 +1244,7 @@ int ha_myisam::assign_to_keycache(THD* thd, HA_CHECK_OPT *check_opt) if (error != HA_ADMIN_OK) { /* Send error to user */ - MI_CHECK param; + HA_CHECK param; myisamchk_init(¶m); param.thd= thd; param.op_name= "assign_to_keycache"; @@ -1415,7 +1415,7 @@ int ha_myisam::enable_indexes(uint mode) else if (mode == HA_KEY_SWITCH_NONUNIQ_SAVE) { THD *thd=current_thd; - MI_CHECK param; + HA_CHECK param; const char *save_proc_info=thd->proc_info; thd->proc_info="Creating index"; myisamchk_init(¶m); @@ -1424,7 +1424,8 @@ int ha_myisam::enable_indexes(uint mode) T_CREATE_MISSING_KEYS); param.myf_rw&= ~MY_WAIT_IF_FULL; param.sort_buffer_length= thd->variables.myisam_sort_buff_size; - param.stats_method= (enum_mi_stats_method)thd->variables.myisam_stats_method; + param.stats_method= + (enum_handler_stats_method)thd->variables.myisam_stats_method; param.tmpdir=&mysql_tmpdir_list; if ((error= (repair(thd,param,0) != HA_ADMIN_OK)) && param.retry_repair) { @@ -1922,7 +1923,7 @@ void ha_myisam::get_auto_increment(ulonglong offset, ulonglong increment, { ulonglong nr; int error; - uchar key[MI_MAX_KEY_LENGTH]; + uchar key[HA_MAX_KEY_LENGTH]; if (!table->s->next_number_key_offset) { // Autoincrement at key-start diff --git a/storage/myisam/myisamdef.h b/storage/myisam/myisamdef.h index 0daa6a7fc72..004e8cce8bb 100644 --- a/storage/myisam/myisamdef.h +++ b/storage/myisam/myisamdef.h @@ -550,8 +550,9 @@ extern int _mi_dispose(MI_INFO *info, MI_KEYDEF *keyinfo, my_off_t pos, extern my_off_t _mi_new(MI_INFO *info, MI_KEYDEF *keyinfo, int level); extern uint _mi_make_key(MI_INFO *info, uint keynr, uchar *key, const uchar *record, my_off_t filepos); -extern uint _mi_pack_key(MI_INFO *info, uint keynr, uchar *key, uchar *old, - uint key_length, HA_KEYSEG ** last_used_keyseg); +extern uint _mi_pack_key(MI_INFO *info, uint keynr, uchar *key, + uchar *old, key_part_map keypart_map, + HA_KEYSEG ** last_used_keyseg); extern int _mi_read_key_record(MI_INFO *info, my_off_t filepos, uchar *buf); extern int _mi_read_cache(IO_CACHE *info, uchar *buff, my_off_t pos, uint length, int re_read_if_possibly); @@ -566,7 +567,7 @@ extern uchar *mi_alloc_rec_buff(MI_INFO *, ulong, uchar **); extern ulong _mi_rec_unpack(MI_INFO *info, uchar *to, uchar *from, ulong reclength); -extern my_bool _mi_rec_check(MI_INFO *info, const char *record, uchar *packpos, +extern my_bool _mi_rec_check(MI_INFO *info,const uchar *record, uchar *packpos, ulong packed_length, my_bool with_checkum); extern int _mi_write_part_record(MI_INFO *info, my_off_t filepos, ulong length, my_off_t next_filepos, uchar ** record, diff --git a/storage/myisam/sort.c b/storage/myisam/sort.c index 2e2684230e7..3ab478682c6 100644 --- a/storage/myisam/sort.c +++ b/storage/myisam/sort.c @@ -831,7 +831,7 @@ static uint NEAR_F read_to_buffer_varlen(IO_CACHE *fromfile, BUFFPEK *buffpek, register uint count; uint16 length_of_key = 0; uint idx; - byte *buffp; + uchar *buffp; if ((count=(uint) min((ha_rows) buffpek->max_keys,buffpek->count))) { @@ -918,7 +918,7 @@ merge_buffers(MI_SORT_PARAM *info, uint keys, IO_CACHE *from_file, for (buffpek= Fb ; buffpek <= Tb ; buffpek++) { count+= buffpek->count; - buffpek->base= (byte*) strpos; + buffpek->base= (uchar*) strpos; buffpek->max_keys=maxcount; strpos+= (uint) (error=(int) info->read_to_buffer(from_file,buffpek, sort_length)); @@ -956,7 +956,7 @@ merge_buffers(MI_SORT_PARAM *info, uint keys, IO_CACHE *from_file, { if (!(error=(int) info->read_to_buffer(from_file,buffpek,sort_length))) { - byte *base= buffpek->base; + uchar *base= buffpek->base; uint max_keys=buffpek->max_keys; VOID(queue_remove(&queue,0)); @@ -988,7 +988,7 @@ merge_buffers(MI_SORT_PARAM *info, uint keys, IO_CACHE *from_file, } } buffpek=(BUFFPEK*) queue_top(&queue); - buffpek->base= (byte*) sort_keys; + buffpek->base= (uchar*) sort_keys; buffpek->max_keys=keys; do { diff --git a/storage/myisammrg/ha_myisammrg.cc b/storage/myisammrg/ha_myisammrg.cc index 551f12e4a8c..f3df1e82c4b 100644 --- a/storage/myisammrg/ha_myisammrg.cc +++ b/storage/myisammrg/ha_myisammrg.cc @@ -48,12 +48,12 @@ static const char *ha_myisammrg_exts[] = { }; extern int table2myisam(TABLE *table_arg, MI_KEYDEF **keydef_out, MI_COLUMNDEF **recinfo_out, uint *records_out); -extern int myisam_check_definition(MI_KEYDEF *t1_keyinfo, - MI_COLUMNDEF *t1_recinfo, - uint t1_keys, uint t1_recs, - MI_KEYDEF *t2_keyinfo, - MI_COLUMNDEF *t2_recinfo, - uint t2_keys, uint t2_recs, bool strict); +extern int check_definition(MI_KEYDEF *t1_keyinfo, + MI_COLUMNDEF *t1_recinfo, + uint t1_keys, uint t1_recs, + MI_KEYDEF *t2_keyinfo, + MI_COLUMNDEF *t2_recinfo, + uint t2_keys, uint t2_recs, bool strict); static void split_file_name(const char *file_name, LEX_STRING *db, LEX_STRING *name); @@ -137,10 +137,10 @@ int ha_myisammrg::open(const char *name, int mode, uint test_if_locked) } for (u_table= file->open_tables; u_table < file->end_table; u_table++) { - if (myisam_check_definition(keyinfo, recinfo, keys, recs, - u_table->table->s->keyinfo, u_table->table->s->rec, - u_table->table->s->base.keys, - u_table->table->s->base.fields, false)) + if (check_definition(keyinfo, recinfo, keys, recs, + u_table->table->s->keyinfo, u_table->table->s->rec, + u_table->table->s->base.keys, + u_table->table->s->base.fields, false)) { error= HA_ERR_WRONG_MRG_TABLE_DEF; if (test_if_locked & HA_OPEN_FOR_REPAIR) -- cgit v1.2.1 From bca70f9f6ec10290a414c683eeecb612dda144f9 Mon Sep 17 00:00:00 2001 From: unknown Date: Tue, 3 Jul 2007 01:19:26 +0300 Subject: Mark the page dirty if we store LSN on it. New type of page in the page cache fixes. storage/maria/ma_pagecache.c: Mark the page dirty if we store LSN on it. Symbolic representation of new page type added (for debugging output). Asserts added (unknown type can't be used if we write the page). --- storage/maria/ma_pagecache.c | 10 +++++++++- 1 file changed, 9 insertions(+), 1 deletion(-) diff --git a/storage/maria/ma_pagecache.c b/storage/maria/ma_pagecache.c index 05173eddf46..bf66a8fd088 100755 --- a/storage/maria/ma_pagecache.c +++ b/storage/maria/ma_pagecache.c @@ -177,7 +177,8 @@ static const char *page_cache_page_type_str[]= /* used only for control page type changing during debugging */ "EMPTY", "PLAIN", - "LSN" + "LSN", + "READ_UNKNOWN" }; static const char *page_cache_page_write_mode_str[]= @@ -584,6 +585,7 @@ static uint pagecache_fwrite(PAGECACHE *pagecache, myf flags) { DBUG_ENTER("pagecache_fwrite"); + DBUG_ASSERT(type != PAGECACHE_READ_UNKNOWN_PAGE); if (type == PAGECACHE_LSN_PAGE) { LSN lsn; @@ -2457,7 +2459,12 @@ static void check_and_set_lsn(LSN lsn, PAGECACHE_BLOCK_LINK *block) (ulong)LSN_FILE_NO(old), (ulong)LSN_OFFSET(old), (ulong)LSN_FILE_NO(lsn), (ulong)LSN_OFFSET(lsn))); if (cmp_translog_addr(lsn, old) > 0) + { + + DBUG_ASSERT(block->type != PAGECACHE_READ_UNKNOWN_PAGE); lsn_store(block->buffer + PAGE_LSN_OFFSET, lsn); + block->status|= PCBLOCK_CHANGED; + } DBUG_VOID_RETURN; } @@ -3179,6 +3186,7 @@ my_bool pagecache_write_part(PAGECACHE *pagecache, page_cache_page_pin_str[pin], page_cache_page_write_mode_str[write_mode], offset, size)); + DBUG_ASSERT(type != PAGECACHE_READ_UNKNOWN_PAGE); DBUG_ASSERT(lock != PAGECACHE_LOCK_LEFT_READLOCKED); DBUG_ASSERT(lock != PAGECACHE_LOCK_READ_UNLOCK); DBUG_ASSERT(offset + size <= pagecache->block_size); -- cgit v1.2.1 From 388122558c83643e320c08d93faa45c7c6d1245e Mon Sep 17 00:00:00 2001 From: unknown Date: Tue, 3 Jul 2007 15:20:41 +0200 Subject: Maria: * Don't modify share->base.born_transactional; now it is a value carved in stone at creation time. share->now_transactional is what can be modified: it starts at born_transactional, can become false during ALTER TABLE (when we want no logging), and restored later. * Not resetting create_rename_lsn to 0 during delete_all or repair. * when we temporarily disable transactionality, we also change the page type to PAGECACHE_PLAIN_PAGE: it bypasses some work in the page cache (optimization), and avoids assertions related to LSNs. * Disable INSERT DELAYED for transactional tables, because durability could not be guaranteed (insertion may even not happen) mysys/mf_keycache.c: comment storage/maria/ha_maria.cc: * a transactional table cannot do INSERT DELAYED * ha_maria::save_transactional not needed anymore, as now instead we don't modify MARIA_SHARE::MARIA_BASE_INFO::born_transactional (born_transactional plays the role of save_transactional), and modify MARIA_SHARE::now_transactional. * REPAIR_TABLE log record is now logged by maria_repair() * comment why we rely on born_transactional to know if we should skipping a transaction. * putting together two if()s which test for F_UNLCK storage/maria/ha_maria.h: ha_maria::save_transactional not needed anymore (moved to the C layer) storage/maria/ma_blockrec.c: * For the block record's code (writing/updating/deleting records), all that counts is now_transactional, not born_transactional. * As we now set the page type to PAGECACHE_PLAIN_PAGE for tables which have now_transactional==FALSE, pagecache will not expect a meaningful LSN for them in pagecache_unlock_by_link(), so we can pass it LSN_IMPOSSIBLE. storage/maria/ma_check.c: * writing LOGREC_REPAIR_TABLE moves from ha_maria::repair() to maria_repair(), sounds cleaner (less functions to export). * when opening a table during REPAIR, don't use the realpath-ed name, as this may fail if the table has symlinked files (maria_open() would try to find the data and index file in the directory of unique_file_name, it would fail if data and index files are in different dirs); use the unresolved name, open_file_name, which is the argument which was passed to the maria_open() which created 'info'. storage/maria/ma_close.c: assert that when a statement is done with a table, it cleans up storage/maria/ma_create.c: new name storage/maria/ma_delete_all.c: * using now_transactional * no reason to reset create_rename_lsn during delete_all (a bug); also no reason to do it during repair: it was put there because a positive create_rename_lsn caused a call to check_and_set_lsn() which asserted in DBUG_ASSERT(block->type == PAGECACHE_LSN_PAGE); first solution was to use LSN_IMPOSSIBLE in _ma_unpin_all_pages() if not transactional; but then in the case of ALTER TABLE, with transactionality temporarily disabled, it asserted in DBUG_ASSERT(LSN_VALID(lsn)) in pagecache_fwrite() (PAGECACHE_LSN_PAGE page with zero LSN - bad). The additional solution is to use PAGECACHE_PLAIN_PAGE when we disable transactionality temporarily: this avoids checks on the LSN, and also bypasses (optimization) the "flush log up to LSN" call when the pagecache flushes our page (in other words, no WAL needed). storage/maria/ma_delete_table.c: use now_transactional storage/maria/ma_locking.c: assert that when a statement is done with a table, it cleans up. storage/maria/ma_loghandler.c: * now_transactional should be used to test if we want a log record. * Assertions to make sure dummy_transaction_object is not spoilt by its many users. storage/maria/ma_open.c: base.transactional -> base.born_transactional storage/maria/ma_pagecache.c: missing name for page's type. Comment for future. storage/maria/ma_rename.c: use now_transactional storage/maria/maria_chk.c: use born_transactional storage/maria/maria_def.h: MARIA_BASE_INFO::transactional renamed to born_transactional. MARIA_SHARE::now_transactional introduced. _ma_repair_write_log_record() is made local to ma_check.c. Macros to temporarily disable, and re-enable, transactionality for a table. storage/maria/maria_read_log.c: assertions and using the new macros. Adding a forgotten resetting when we finally close all tables. --- mysys/mf_keycache.c | 9 ++++- storage/maria/ha_maria.cc | 73 +++++++++++++++++++++++++---------------- storage/maria/ha_maria.h | 5 --- storage/maria/ma_blockrec.c | 26 +++++---------- storage/maria/ma_check.c | 28 ++++++++++------ storage/maria/ma_close.c | 3 +- storage/maria/ma_create.c | 2 +- storage/maria/ma_delete_all.c | 3 +- storage/maria/ma_delete_table.c | 2 +- storage/maria/ma_locking.c | 1 + storage/maria/ma_loghandler.c | 13 +++++++- storage/maria/ma_open.c | 23 +++++++++---- storage/maria/ma_pagecache.c | 11 ++++++- storage/maria/ma_rename.c | 8 ++++- storage/maria/maria_chk.c | 2 +- storage/maria/maria_def.h | 20 +++++++++-- storage/maria/maria_read_log.c | 29 +++++++++------- 17 files changed, 167 insertions(+), 91 deletions(-) diff --git a/mysys/mf_keycache.c b/mysys/mf_keycache.c index 95a9f08a07a..065c10e3d73 100644 --- a/mysys/mf_keycache.c +++ b/mysys/mf_keycache.c @@ -2456,7 +2456,14 @@ restart: } else { - /* Link the block into a list of blocks 'in switch' */ + /* + Link the block into a list of blocks 'in switch'. + Note that if there could be two concurrent flush_key_blocks_int() + on this file (normally this does not happen, as MyISAM uses + intern_lock for flushing), then the first one may move the block + into its first_in_switch, and the second one would just not see + the block and wrongly consider its job done. + */ unlink_changed(block); link_changed(block, &first_in_switch); } diff --git a/storage/maria/ha_maria.cc b/storage/maria/ha_maria.cc index 938e99375f2..232dd7e695d 100644 --- a/storage/maria/ha_maria.cc +++ b/storage/maria/ha_maria.cc @@ -479,7 +479,7 @@ handler(hton, table_arg), file(0), int_table_flags(HA_NULL_IN_KEY | HA_CAN_FULLTEXT | HA_CAN_SQL_HANDLER | HA_DUPLICATE_POS | HA_CAN_INDEX_BLOBS | HA_AUTO_PART_KEY | HA_FILE_BASED | HA_CAN_GEOMETRY | MARIA_CANNOT_ROLLBACK | - HA_CAN_INSERT_DELAYED | HA_CAN_BIT_FIELD | HA_CAN_RTREEKEYS | + HA_CAN_BIT_FIELD | HA_CAN_RTREEKEYS | HA_HAS_RECORDS | HA_STATS_RECORDS_IS_EXACT), can_enable_indexes(1) {} @@ -697,9 +697,19 @@ int ha_maria::open(const char *name, int mode, uint test_if_locked) info(HA_STATUS_NO_LOCK | HA_STATUS_VARIABLE | HA_STATUS_CONST); if (!(test_if_locked & HA_OPEN_WAIT_IF_LOCKED)) VOID(maria_extra(file, HA_EXTRA_WAIT_LOCK, 0)); - save_transactional= file->s->base.transactional; if ((data_file_type= file->s->data_file_type) != STATIC_RECORD) int_table_flags |= HA_REC_NOT_IN_SEQ; + if (!file->s->base.born_transactional) + { + /* + INSERT DELAYED cannot work with transactional tables (because it cannot + stand up to "when client gets ok the data is safe on disk": the record + may not even be inserted). In the future, we could enable it back (as a + client doing INSERT DELAYED knows the specificities; but we then should + make sure to regularly commit in the delayed_insert thread). + */ + int_table_flags|= HA_CAN_INSERT_DELAYED; + } if (file->s->options & (HA_OPTION_CHECKSUM | HA_OPTION_COMPRESS_RECORD)) int_table_flags |= HA_HAS_CHECKSUM; @@ -1178,8 +1188,6 @@ int ha_maria::repair(THD *thd, HA_CHECK ¶m, bool do_optimize) llstr(rows, llbuff), llstr(file->state->records, llbuff2)); } - if (!error) - error= _ma_repair_write_log_record(¶m, file); } else { @@ -1861,30 +1869,19 @@ int ha_maria::external_lock(THD *thd, int lock_type) { TRN *trn= THD_TRN; DBUG_ENTER("ha_maria::external_lock"); - if (!save_transactional) + /* + We don't test now_transactional because it may vary between lock/unlock + and thus confuse our reference counting. + It is critical to skip non-transactional tables: user-visible temporary + tables get an external_lock() when read/written for the first time, but no + corresponding unlock (they just stay locked and are later dropped while + locked); if a tmp table was transactional, "SELECT FROM non_tmp, tmp" + would never commit as its "locked_tables" count would stay 1. + */ + if (!file->s->base.born_transactional) goto skip_transaction; - if (!trn && lock_type != F_UNLCK) /* no transaction yet - open it now */ - { - trn= trnman_new_trn(& thd->mysys_var->mutex, - & thd->mysys_var->suspend, - thd->thread_stack + STACK_DIRECTION * - (my_thread_stack_size - STACK_MIN_SIZE)); - if (!trn) - DBUG_RETURN(HA_ERR_OUT_OF_MEM); - - DBUG_PRINT("info", ("THD_TRN set to 0x%lx", (ulong)trn)); - THD_TRN= trn; - if (thd->options & (OPTION_NOT_AUTOCOMMIT | OPTION_BEGIN)) - trans_register_ha(thd, TRUE, maria_hton); - } if (lock_type != F_UNLCK) { - this->file->trn= trn; - if (!trnman_increment_locked_tables(trn)) - { - trans_register_ha(thd, FALSE, maria_hton); - trnman_new_statement(trn); - } if (!thd->transaction.on) { /* @@ -1896,11 +1893,32 @@ int ha_maria::external_lock(THD *thd, int lock_type) tons of archived logs to roll-forward, we could then not disable REDOs/UNDOs in this case. */ - file->s->base.transactional= FALSE; + _ma_tmp_disable_logging_for_table(file->s); + } + if (!trn) /* no transaction yet - open it now */ + { + trn= trnman_new_trn(& thd->mysys_var->mutex, + & thd->mysys_var->suspend, + thd->thread_stack + STACK_DIRECTION * + (my_thread_stack_size - STACK_MIN_SIZE)); + if (unlikely(!trn)) + DBUG_RETURN(HA_ERR_OUT_OF_MEM); + + DBUG_PRINT("info", ("THD_TRN set to 0x%lx", (ulong)trn)); + THD_TRN= trn; + if (thd->options & (OPTION_NOT_AUTOCOMMIT | OPTION_BEGIN)) + trans_register_ha(thd, TRUE, maria_hton); + } + this->file->trn= trn; + if (!trnman_increment_locked_tables(trn)) + { + trans_register_ha(thd, FALSE, maria_hton); + trnman_new_statement(trn); } } else { + _ma_reenable_logging_for_table(file->s); this->file->trn= 0; /* TODO: remove it also in commit and rollback */ if (trn && trnman_has_locked_tables(trn)) { @@ -1921,7 +1939,6 @@ int ha_maria::external_lock(THD *thd, int lock_type) #endif } } - file->s->base.transactional= save_transactional; } skip_transaction: DBUG_RETURN(maria_lock_database(file, !table->s->tmp_table ? @@ -1932,7 +1949,7 @@ skip_transaction: int ha_maria::start_stmt(THD *thd, thr_lock_type lock_type) { TRN *trn= THD_TRN; - if (save_transactional) + if (file->s->base.born_transactional) { DBUG_ASSERT(trn); // this may be called only after external_lock() DBUG_ASSERT(trnman_has_locked_tables(trn)); diff --git a/storage/maria/ha_maria.h b/storage/maria/ha_maria.h index a2f6b190657..dd0a9594ef3 100644 --- a/storage/maria/ha_maria.h +++ b/storage/maria/ha_maria.h @@ -39,11 +39,6 @@ class ha_maria :public handler char *data_file_name, *index_file_name; enum data_file_type data_file_type; bool can_enable_indexes; - /** - @brief for temporarily disabling table's transactionality - (if THD::transaction::on is false), remember the original value here - */ - bool save_transactional; int repair(THD * thd, HA_CHECK ¶m, bool optimize); public: diff --git a/storage/maria/ma_blockrec.c b/storage/maria/ma_blockrec.c index cfa9df02102..d8694f50a68 100644 --- a/storage/maria/ma_blockrec.c +++ b/storage/maria/ma_blockrec.c @@ -581,18 +581,10 @@ void _ma_unpin_all_pages(MARIA_HA *info, LSN undo_lsn) DBUG_PRINT("info", ("undo_lsn: %lu", (ulong) undo_lsn)); /* True if not disk error */ - DBUG_ASSERT((undo_lsn != LSN_IMPOSSIBLE) || !info->s->base.transactional); + DBUG_ASSERT((undo_lsn != LSN_IMPOSSIBLE) || !info->s->now_transactional); - if (!info->s->base.transactional) - { - /* - If this is a transactional table but with transactionality temporarily - disabled (like in ALTER TABLE) we need to give a sensible LSN to pages - and not LSN_IMPOSSIBLE. If this is not a transactional table it will - reduce to LSN_IMPOSSIBLE. - */ - undo_lsn= info->s->state.create_rename_lsn; - } + if (!info->s->now_transactional) + undo_lsn= LSN_IMPOSSIBLE; /* don't try to set a LSN on pages */ while (pinned_page-- != page_link) pagecache_unlock_by_link(info->s->pagecache, pinned_page->link, @@ -1446,7 +1438,7 @@ static my_bool free_full_page_range(MARIA_HA *info, ulonglong page, uint count) page, count, PAGECACHE_LOCK_WRITE, 0)) res= 1; - if (info->s->base.transactional) + if (info->s->now_transactional) { LSN lsn; DBUG_ASSERT(info->trn->rec_lsn); @@ -1953,7 +1945,7 @@ static my_bool write_block_record(MARIA_HA *info, head_block+1, bitmap_blocks->count - 1); } - if (share->base.transactional) + if (share->now_transactional) { uchar log_data[FILEID_STORE_SIZE + PAGE_STORE_SIZE + DIRPOS_STORE_SIZE]; LEX_STRING log_array[TRANSLOG_INTERNAL_PARTS + 2]; @@ -1998,7 +1990,7 @@ static my_bool write_block_record(MARIA_HA *info, else push_dynamic(&info->pinned_pages, (void*) &page_link); - if (share->base.transactional && (tmp_data_used || blob_full_pages_exists)) + if (share->now_transactional && (tmp_data_used || blob_full_pages_exists)) { /* Log REDO writes for all full pages (head part and all blobs) @@ -2095,7 +2087,7 @@ static my_bool write_block_record(MARIA_HA *info, } /* Write UNDO record */ - if (share->base.transactional) + if (share->now_transactional) { uchar log_data[LSN_STORE_SIZE + FILEID_STORE_SIZE + PAGE_STORE_SIZE + DIRPOS_STORE_SIZE]; @@ -2312,7 +2304,7 @@ my_bool _ma_write_abort_block_record(MARIA_HA *info) } } - if (info->s->base.transactional) + if (info->s->now_transactional) { LSN lsn; LEX_STRING log_array[TRANSLOG_INTERNAL_PARTS + 1]; @@ -2671,7 +2663,7 @@ my_bool _ma_delete_block_record(MARIA_HA *info, const byte *record) if (info->cur_row.extents && free_full_pages(info, &info->cur_row)) goto err; - if (info->s->base.transactional) + if (info->s->now_transactional) { LSN lsn; uchar log_data[LSN_STORE_SIZE + FILEID_STORE_SIZE + PAGE_STORE_SIZE + diff --git a/storage/maria/ma_check.c b/storage/maria/ma_check.c index bbe7e6a193a..ae23e64575b 100644 --- a/storage/maria/ma_check.c +++ b/storage/maria/ma_check.c @@ -91,6 +91,7 @@ static int _ma_safe_scan_block_record(MARIA_SORT_INFO *sort_info, MARIA_HA *info, byte *record); static void copy_data_file_state(MARIA_STATE_INFO *to, MARIA_STATE_INFO *from); +static int write_log_record_for_repair(const HA_CHECK *param, MARIA_HA *info); void maria_chk_init(HA_CHECK *param) @@ -1952,6 +1953,8 @@ int maria_repair(HA_CHECK *param, register MARIA_HA *info, MARIA_SORT_PARAM sort_param; my_bool block_record, scan_inited= 0; enum data_file_type org_data_file_type= info->s->data_file_type; + myf sync_dir= ((share->now_transactional && !share->temporary) ? + MY_SYNC_DIR : 0); DBUG_ENTER("maria_repair"); bzero((char *)&sort_info, sizeof(sort_info)); @@ -1999,7 +2002,15 @@ int maria_repair(HA_CHECK *param, register MARIA_HA *info, share->state.header.org_data_file_type == BLOCK_RECORD)) { MARIA_HA *new_info; - if (!(sort_info.new_info= maria_open(info->s->unique_file_name, O_RDWR, + /** + @todo RECOVERY it's a bit worrying to have two MARIA_SHARE on the + same index file: + - Checkpoint will see them as two tables + - are we sure that new_info never flushes an in-progress state + to the index file? And how to prevent Checkpoint from doing that? + - in the close future maria_close() will write the state... + */ + if (!(sort_info.new_info= maria_open(info->s->open_file_name, O_RDWR, HA_OPEN_COPY | HA_OPEN_FOR_REPAIR))) goto err; new_info= sort_info.new_info; @@ -2174,8 +2185,6 @@ int maria_repair(HA_CHECK *param, register MARIA_HA *info, if (!rep_quick) { - myf sync_dir= ((share->base.transactional && !share->temporary) ? - MY_SYNC_DIR : 0); if (sort_info.new_info != sort_info.info) { MARIA_STATE_INFO save_state= sort_info.new_info->s->state; @@ -2223,7 +2232,7 @@ int maria_repair(HA_CHECK *param, register MARIA_HA *info, llstr(sort_info.dupp,llbuff)); } - got_error=0; + got_error= sync_dir ? write_log_record_for_repair(param, info) : 0; /* If invoked by external program that uses thr_lock */ if (&share->state.state != info->state) memcpy( &share->state.state, info->state, sizeof(*info->state)); @@ -2424,7 +2433,7 @@ int maria_sort_index(HA_CHECK *param, register MARIA_HA *info, my_string name) int old_lock; MARIA_SHARE *share=info->s; MARIA_STATE_INFO old_state; - myf sync_dir= (share->base.transactional && !share->temporary) ? + myf sync_dir= (share->now_transactional && !share->temporary) ? MY_SYNC_DIR : 0; DBUG_ENTER("maria_sort_index"); @@ -2702,7 +2711,7 @@ int maria_repair_by_sort(HA_CHECK *param, register MARIA_HA *info, char llbuff[22]; MARIA_SORT_INFO sort_info; ulonglong key_map=share->state.key_map; - myf sync_dir= ((share->base.transactional && !share->temporary) ? + myf sync_dir= ((share->now_transactional && !share->temporary) ? MY_SYNC_DIR : 0); DBUG_ENTER("maria_repair_by_sort"); @@ -3127,7 +3136,7 @@ int maria_repair_parallel(HA_CHECK *param, register MARIA_HA *info, MARIA_SORT_INFO sort_info; ulonglong key_map=share->state.key_map; pthread_attr_t thr_attr; - myf sync_dir= (share->base.transactional && !share->temporary) ? + myf sync_dir= (share->now_transactional && !share->temporary) ? MY_SYNC_DIR : 0; DBUG_ENTER("maria_repair_parallel"); @@ -5487,11 +5496,10 @@ read_next_page: @retval 1 error (disk problem) */ -int _ma_repair_write_log_record(const HA_CHECK *param, MARIA_HA *info) +static int write_log_record_for_repair(const HA_CHECK *param, MARIA_HA *info) { MARIA_SHARE *share= info->s; - /* Only called from ha_maria.cc, not maria_check, so translog is inited */ - if (share->base.transactional && !share->temporary) + if (translog_inited) /* test it in case this is maria_chk */ { /* For now this record is only informative. It could serve when applying diff --git a/storage/maria/ma_close.c b/storage/maria/ma_close.c index 19f94aa3b56..4fec7359d66 100644 --- a/storage/maria/ma_close.c +++ b/storage/maria/ma_close.c @@ -108,7 +108,8 @@ int maria_close(register MARIA_HA *info) } } #endif - my_free((gptr) info->s,MYF(0)); + DBUG_ASSERT(share->now_transactional == share->base.born_transactional); + my_free((gptr) share, MYF(0)); } pthread_mutex_unlock(&THR_LOCK_maria); if (info->ftparser_param) diff --git a/storage/maria/ma_create.c b/storage/maria/ma_create.c index 997ce13ca27..2098d7119eb 100644 --- a/storage/maria/ma_create.c +++ b/storage/maria/ma_create.c @@ -259,7 +259,7 @@ int maria_create(const char *name, enum data_file_type datafile_type, } share.base.null_bytes= ci->null_bytes; share.base.original_null_bytes= ci->null_bytes; - share.base.transactional= ci->transactional; + share.base.born_transactional= ci->transactional; share.base.max_field_lengths= max_field_lengths; share.base.field_offsets= 0; /* for future */ diff --git a/storage/maria/ma_delete_all.c b/storage/maria/ma_delete_all.c index c3bdcdf365c..42e7fb3c2f9 100644 --- a/storage/maria/ma_delete_all.c +++ b/storage/maria/ma_delete_all.c @@ -46,7 +46,7 @@ int maria_delete_all_rows(MARIA_HA *info) */ if (_ma_readinfo(info,F_WRLCK,1)) DBUG_RETURN(my_errno); - log_record= share->base.transactional && !share->temporary; + log_record= share->now_transactional && !share->temporary; if (_ma_mark_file_changed(info)) goto err; @@ -142,7 +142,6 @@ void _ma_reset_status(MARIA_HA *info) info->state->data_file_length= 0; info->state->empty= info->state->key_empty= 0; info->state->checksum= 0; - share->state.create_rename_lsn= LSN_IMPOSSIBLE; /* Drop the delete key chain. */ state->key_del= HA_OFFSET_ERROR; diff --git a/storage/maria/ma_delete_table.c b/storage/maria/ma_delete_table.c index 39a286ad1f7..6d6b9d032fd 100644 --- a/storage/maria/ma_delete_table.c +++ b/storage/maria/ma_delete_table.c @@ -64,7 +64,7 @@ int maria_delete_table(const char *name) raid_type= info->s->base.raid_type; raid_chunks= info->s->base.raid_chunks; #endif - sync_dir= (info->s->base.transactional && !info->s->temporary) ? + sync_dir= (info->s->now_transactional && !info->s->temporary) ? MY_SYNC_DIR : 0; maria_close(info); } diff --git a/storage/maria/ma_locking.c b/storage/maria/ma_locking.c index abb095d47c2..1825367c44c 100644 --- a/storage/maria/ma_locking.c +++ b/storage/maria/ma_locking.c @@ -129,6 +129,7 @@ int maria_lock_database(MARIA_HA *info, int lock_type) } info->opt_flag&= ~(READ_CACHE_USED | WRITE_CACHE_USED); info->lock_type= F_UNLCK; + DBUG_ASSERT(share->now_transactional == share->base.born_transactional); break; case F_RDLCK: if (info->lock_type == F_WRLCK) diff --git a/storage/maria/ma_loghandler.c b/storage/maria/ma_loghandler.c index dc524d858e7..6195e552185 100644 --- a/storage/maria/ma_loghandler.c +++ b/storage/maria/ma_loghandler.c @@ -4263,7 +4263,7 @@ my_bool translog_write_record(LSN *lsn, if (share) { - if (!share->base.transactional) + if (!share->now_transactional) { DBUG_PRINT("info", ("It is not transactional table")); DBUG_RETURN(0); @@ -5614,6 +5614,16 @@ static my_bool write_hook_for_redo(enum translog_record_type type struct st_translog_parts *parts __attribute__ ((unused))) { + /* + Users of dummy_transaction_object must keep this TRN clean as it + is used by many threads (like those manipulating non-transactional + tables). It might be dangerous if one user sets rec_lsn or some other + member and it is picked up by another user (like putting this rec_lsn into + a page of a non-transactional table); it's safer if all members stay 0. So + non-transactional log records (REPAIR, CREATE, RENAME, DROP) should not + call this hook; we trust them but verify ;) + */ + DBUG_ASSERT(trn->trid != 0); /* If the hook stays so simple, it would be faster to pass !trn->rec_lsn ? trn->rec_lsn : some_dummy_lsn @@ -5640,6 +5650,7 @@ static my_bool write_hook_for_undo(enum translog_record_type type struct st_translog_parts *parts __attribute__ ((unused))) { + DBUG_ASSERT(trn->trid != 0); /* see write_hook_for_redo() */ trn->undo_lsn= *lsn; if (unlikely(LSN_WITH_FLAGS_TO_LSN(trn->first_undo_lsn) == 0)) trn->first_undo_lsn= diff --git a/storage/maria/ma_open.c b/storage/maria/ma_open.c index cb05ab5b5f0..eb0bba7503f 100644 --- a/storage/maria/ma_open.c +++ b/storage/maria/ma_open.c @@ -589,9 +589,17 @@ MARIA_HA *maria_open(const char *name, int mode, uint open_flags) share->base.pack_bytes + test(share->options & HA_OPTION_CHECKSUM)); if (open_flags & HA_OPEN_COPY) - share->base.transactional= 0; /* Repair: no logging */ - if (share->base.transactional) { + /* + this instance will be a temporary one used just to create a data + file for REPAIR. Don't do logging. This base information will not go + to disk. + */ + share->base.born_transactional= FALSE; + } + if (share->base.born_transactional) + { + share->page_type= PAGECACHE_LSN_PAGE; share->base_length+= TRANS_ROW_EXTRA_HEADER_SIZE; if (unlikely((share->state.create_rename_lsn == (LSN)ULONGLONG_MAX) && (open_flags & HA_OPEN_FROM_SQL_LAYER))) @@ -604,11 +612,12 @@ MARIA_HA *maria_open(const char *name, int mode, uint open_flags) _ma_update_create_rename_lsn_on_disk(share, TRUE); } } + else + share->page_type= PAGECACHE_PLAIN_PAGE; + share->now_transactional= share->base.born_transactional; + share->base.default_rec_buff_size= max(share->base.pack_reclength, share->base.max_key_length); - share->page_type= (share->base.transactional ? PAGECACHE_LSN_PAGE : - PAGECACHE_PLAIN_PAGE); - if (share->data_file_type == DYNAMIC_RECORD) { share->base.extra_rec_buff_size= @@ -1124,7 +1133,7 @@ uint _ma_base_info_write(File file, MARIA_BASE_INFO *base) *ptr++= base->key_reflength; *ptr++= base->keys; *ptr++= base->auto_key; - *ptr++= base->transactional; + *ptr++= base->born_transactional; *ptr++= 0; /* Reserved */ mi_int2store(ptr,base->pack_bytes); ptr+= 2; mi_int2store(ptr,base->blobs); ptr+= 2; @@ -1167,7 +1176,7 @@ static byte *_ma_base_info_read(byte *ptr, MARIA_BASE_INFO *base) base->key_reflength= *ptr++; base->keys= *ptr++; base->auto_key= *ptr++; - base->transactional= *ptr++; + base->born_transactional= *ptr++; ptr++; base->pack_bytes= mi_uint2korr(ptr); ptr+= 2; base->blobs= mi_uint2korr(ptr); ptr+= 2; diff --git a/storage/maria/ma_pagecache.c b/storage/maria/ma_pagecache.c index 05173eddf46..994da92e0e9 100755 --- a/storage/maria/ma_pagecache.c +++ b/storage/maria/ma_pagecache.c @@ -177,7 +177,8 @@ static const char *page_cache_page_type_str[]= /* used only for control page type changing during debugging */ "EMPTY", "PLAIN", - "LSN" + "LSN", + "UNKNOWN" }; static const char *page_cache_page_write_mode_str[]= @@ -3649,6 +3650,14 @@ restart: ("changed_blocks") though it's still dirty (the flush by another thread has not yet happened). Checkpoint will miss the page and so must be blocked until that flush has happened. + Note that if there are two concurrent + flush_pagecache_blocks_int() on this file, then the first one may + move the block into its first_in_switch, and the second one would + just not see the block and wrongly consider its job done. + @todo RECOVERY Maria does protect such flushes with intern_lock, + but Checkpoint does not (Checkpoint makes sure that + changed_blocks_is_incomplete is 0 when it starts, but as + flush_cached_blocks() releases mutex, this may change... */ /** @todo RECOVERY: check all places where we remove a page from the diff --git a/storage/maria/ma_rename.c b/storage/maria/ma_rename.c index 8f42a5b931a..9dd75705229 100644 --- a/storage/maria/ma_rename.c +++ b/storage/maria/ma_rename.c @@ -56,7 +56,13 @@ int maria_rename(const char *old_name, const char *new_name) raid_chunks = share->base.raid_chunks; #endif - sync_dir= (share->base.transactional && !share->temporary) ? + /* + the renaming of an internal table to the final table (like in ALTER TABLE) + is the moment when this table receives its correct create_rename_lsn and + this is important; make sure transactionality has been re-enabled. + */ + DBUG_ASSERT(share->now_transactional == share->base.born_transactional); + sync_dir= (share->now_transactional && !share->temporary) ? MY_SYNC_DIR : 0; if (sync_dir) { diff --git a/storage/maria/maria_chk.c b/storage/maria/maria_chk.c index 3c93b39509f..37f6f1fb49b 100644 --- a/storage/maria/maria_chk.c +++ b/storage/maria/maria_chk.c @@ -1033,7 +1033,7 @@ static int maria_chk(HA_CHECK *param, my_string filename) know what the log's end LSN is now, so we just let the server know that it will have to find and store it. */ - if (share->base.transactional) + if (share->base.born_transactional) share->state.create_rename_lsn= (LSN)ULONGLONG_MAX; if ((param->testflag & (T_REP_BY_SORT | T_REP_PARALLEL)) && (maria_is_any_key_active(share->state.key_map) || diff --git a/storage/maria/maria_def.h b/storage/maria/maria_def.h index 5f508da213d..e46b120bf3f 100644 --- a/storage/maria/maria_def.h +++ b/storage/maria/maria_def.h @@ -171,8 +171,11 @@ typedef struct st_ma_base_info /* The following are from the header */ uint key_parts, all_key_parts; - /* If false, we disable logging, versioning, transaction etc */ - my_bool transactional; + /** + @brief If false, we disable logging, versioning, transaction etc. Observe + difference with MARIA_SHARE::now_transactional + */ + my_bool born_transactional; } MARIA_BASE_INFO; @@ -306,6 +309,13 @@ typedef struct st_maria_share not_flushed, concurrent_insert; my_bool delay_key_write; my_bool have_rtree; + /** + @brief if the table is transactional right now. It may have been created + transactional (base.born_transactional==TRUE) but with transactionality + (logging) temporarily disabled (now_transactional==FALSE). The opposite + (FALSE, TRUE) is impossible. + */ + my_bool now_transactional; #ifdef THREAD THR_LOCK lock; pthread_mutex_t intern_lock; /* Locking for use with _locking */ @@ -891,7 +901,6 @@ MARIA_RECORD_POS _ma_write_init_default(MARIA_HA *info, const byte *record); my_bool _ma_write_abort_default(MARIA_HA *info); C_MODE_START -int _ma_repair_write_log_record(const HA_CHECK *param, MARIA_HA *info); /* Functions needed by _ma_check (are overrided in MySQL) */ volatile int *_ma_killed_ptr(HA_CHECK *param); void _ma_check_print_error _VARARGS((HA_CHECK *param, const char *fmt, ...)); @@ -916,5 +925,10 @@ int _ma_initialize_data_file(MARIA_SHARE *share, File dfile); int _ma_update_create_rename_lsn_on_disk(MARIA_SHARE *share, my_bool do_sync); void _ma_unpin_all_pages(MARIA_HA *info, LSN undo_lsn); +#define _ma_tmp_disable_logging_for_table(S) \ + { (S)->now_transactional= FALSE; (S)->page_type= PAGECACHE_PLAIN_PAGE; } +#define _ma_reenable_logging_for_table(S) \ + { if (((S)->now_transactional= (S)->base.born_transactional)) \ + (S)->page_type= PAGECACHE_LSN_PAGE; } extern PAGECACHE *maria_log_pagecache; diff --git a/storage/maria/maria_read_log.c b/storage/maria/maria_read_log.c index 5b2d5b057c2..55f6c9f0cdf 100644 --- a/storage/maria/maria_read_log.c +++ b/storage/maria/maria_read_log.c @@ -442,8 +442,11 @@ prototype_exec_hook(REDO_CREATE_TABLE) info= maria_open(name, O_RDONLY, HA_OPEN_FOR_REPAIR); if (info) { - DBUG_ASSERT(info->s->reopen == 1); /* check that we're not using it */ - if (!info->s->base.transactional) + MARIA_SHARE *share= info->s; + /* check that we're not already using it */ + DBUG_ASSERT(share->reopen == 1); + DBUG_ASSERT(share->now_transactional == share->base.born_transactional); + if (!share->base.born_transactional) { /* could be that transactional table was later dropped, and a non-trans @@ -454,7 +457,7 @@ prototype_exec_hook(REDO_CREATE_TABLE) DBUG_ASSERT(0); /* I want to know this */ goto end; } - if (cmp_translog_addr(info->s->state.create_rename_lsn, rec->lsn) >= 0) + if (cmp_translog_addr(share->state.create_rename_lsn, rec->lsn) >= 0) { printf(", has create_rename_lsn (%lu,0x%lx) is more recent than record", (ulong) LSN_FILE_NO(rec->lsn), @@ -551,6 +554,7 @@ prototype_exec_hook(FILE_ID) int error; char *name, *buff; MARIA_HA *info= NULL; + MARIA_SHARE *share; if (((buff= my_malloc(rec->record_length, MYF(MY_WME))) == NULL) || (translog_read_record(rec->lsn, 0, rec->record_length, buff, NULL) != rec->record_length)) @@ -566,7 +570,7 @@ prototype_exec_hook(FILE_ID) { printf(", closing table '%s'", info->s->open_file_name); all_tables[sid]= NULL; - info->s->base.transactional= TRUE; /* put back the truth */ + _ma_reenable_logging_for_table(info->s); /* put back the truth */ if (maria_close(info)) { fprintf(stderr, "Failed to close table\n"); @@ -586,19 +590,19 @@ prototype_exec_hook(FILE_ID) fprintf(stderr, "Table is crashed, can't apply log records to it\n"); goto err; } - DBUG_ASSERT(info->s->reopen == 1); /* should always be only one instance */ - if (!info->s->base.transactional) + share= info->s; + /* check that we're not already using it */ + DBUG_ASSERT(share->reopen == 1); + DBUG_ASSERT(share->now_transactional == share->base.born_transactional); + if (!share->base.born_transactional) { printf(", is not transactional\n"); DBUG_ASSERT(0); /* I want to know this */ goto end; } all_tables[sid]= info; - /* - don't log any records for this work. TODO make sure this variable does not - go to disk before we restore it to its true value. - */ - info->s->base.transactional= FALSE; + /* don't log any records for this work */ + _ma_tmp_disable_logging_for_table(share); printf(", opened\n"); error= 0; goto end; @@ -742,7 +746,10 @@ static void end_of_redo_phase() { MARIA_HA *info= all_tables[sid]; if (info != NULL) + { + _ma_reenable_logging_for_table(info->s); /* put back the truth */ maria_close(info); + } } } } -- cgit v1.2.1 From ad3e38f88eca4deb22b030669f32a1fb1db1feeb Mon Sep 17 00:00:00 2001 From: unknown Date: Tue, 3 Jul 2007 16:00:05 +0200 Subject: Marking the block dirty requires linking it into the changed_blocks[] list (for flush_pagecache*() functions and Checkpoint to see it) --- storage/maria/ma_pagecache.c | 40 ++++++++++++++++++---------------------- 1 file changed, 18 insertions(+), 22 deletions(-) diff --git a/storage/maria/ma_pagecache.c b/storage/maria/ma_pagecache.c index 6add7231e6f..50dde101c0d 100755 --- a/storage/maria/ma_pagecache.c +++ b/storage/maria/ma_pagecache.c @@ -2440,16 +2440,16 @@ static void read_block(PAGECACHE *pagecache, } -/* - Set LSN on the page to the given one if the given LSN is bigger +/** + @brief Set LSN on the page to the given one if the given LSN is bigger - SYNOPSIS - check_and_set_lsn() - lsn LSN to set - block block to check and set + @param pagecache pointer to a page cache data structure + @param lsn LSN to set + @param block block to check and set */ -static void check_and_set_lsn(LSN lsn, PAGECACHE_BLOCK_LINK *block) +static void check_and_set_lsn(PAGECACHE *pagecache, + LSN lsn, PAGECACHE_BLOCK_LINK *block) { LSN old; DBUG_ENTER("check_and_set_lsn"); @@ -2463,7 +2463,9 @@ static void check_and_set_lsn(LSN lsn, PAGECACHE_BLOCK_LINK *block) DBUG_ASSERT(block->type != PAGECACHE_READ_UNKNOWN_PAGE); lsn_store(block->buffer + PAGE_LSN_OFFSET, lsn); - block->status|= PCBLOCK_CHANGED; + /* we stored LSN in page so we dirtied it */ + if (!(block->status & PCBLOCK_CHANGED)) + link_to_changed_list(pagecache, block); } DBUG_VOID_RETURN; } @@ -2537,10 +2539,8 @@ void pagecache_unlock(PAGECACHE *pagecache, if (block->rec_lsn == 0) block->rec_lsn= first_REDO_LSN_for_page; } - if (lsn != 0) - { - check_and_set_lsn(lsn, block); - } + if (lsn != LSN_IMPOSSIBLE) + check_and_set_lsn(pagecache, lsn, block); if (make_lock_and_pin(pagecache, block, lock, pin)) { @@ -2600,10 +2600,8 @@ void pagecache_unpin(PAGECACHE *pagecache, DBUG_ASSERT(block != 0); DBUG_ASSERT(page_st == PAGE_READ); - if (lsn != 0) - { - check_and_set_lsn(lsn, block); - } + if (lsn != LSN_IMPOSSIBLE) + check_and_set_lsn(pagecache, lsn, block); /* we can just unpin only with keeping read lock because: @@ -2700,7 +2698,7 @@ void pagecache_unlock_by_link(PAGECACHE *pagecache, block->rec_lsn= first_REDO_LSN_for_page; } if (lsn != LSN_IMPOSSIBLE) - check_and_set_lsn(lsn, block); + check_and_set_lsn(pagecache, lsn, block); if (make_lock_and_pin(pagecache, block, lock, pin)) DBUG_ASSERT(0); /* should not happend */ @@ -2754,10 +2752,8 @@ void pagecache_unpin_by_link(PAGECACHE *pagecache, inc_counter_for_resize_op(pagecache); - if (lsn != 0) - { - check_and_set_lsn(lsn, block); - } + if (lsn != LSN_IMPOSSIBLE) + check_and_set_lsn(pagecache, lsn, block); /* We can just unpin only with keeping read lock because: @@ -3920,7 +3916,7 @@ my_bool pagecache_collect_changed_blocks_with_lsn(PAGECACHE *pagecache, ptr+= 4; lsn_store(ptr, block->rec_lsn); ptr+= LSN_STORE_SIZE; - if (block->rec_lsn != 0) + if (block->rec_lsn != LSN_IMPOSSIBLE) { if (cmp_translog_addr(block->rec_lsn, minimum_rec_lsn) < 0) minimum_rec_lsn= block->rec_lsn; -- cgit v1.2.1 From 29d7d1becdb777d84c4d9948a9791681d67fe3c5 Mon Sep 17 00:00:00 2001 From: unknown Date: Tue, 3 Jul 2007 16:36:13 +0200 Subject: Verify that INSERT DELAYED is disabled only for transactional Maria tables mysql-test/r/maria.result: result update (we see DELAYED is refused only for transactional Maria tables) mysql-test/t/maria.test: verify that INSERT DELAYED is disabled only for transactional Maria tables --- mysql-test/r/maria.result | 24 ++++++++++++++++++++++++ mysql-test/t/maria.test | 18 ++++++++++++++++++ 2 files changed, 42 insertions(+) diff --git a/mysql-test/r/maria.result b/mysql-test/r/maria.result index a1c49fdfe4f..6303f498e36 100644 --- a/mysql-test/r/maria.result +++ b/mysql-test/r/maria.result @@ -1832,3 +1832,27 @@ t1 CREATE TABLE `t1` ( `a` int(11) DEFAULT NULL ) ENGINE=MARIA DEFAULT CHARSET=latin1 ROW_FORMAT=PAGE drop table t1; +create table t1 (a int) row_format=page; +insert delayed into t1 values(1); +ERROR HY000: Table storage engine for 't1' doesn't have this option +drop table t1; +create table t1 (a int) row_format=page transactional=0; +insert delayed into t1 values(1); +flush table t1; +select * from t1; +a +1 +select count(*) from t1; +count(*) +1 +drop table t1; +create table t1 (a int) row_format=dynamic; +insert delayed into t1 values(1); +flush table t1; +select * from t1; +a +1 +select count(*) from t1; +count(*) +1 +drop table t1; diff --git a/mysql-test/t/maria.test b/mysql-test/t/maria.test index 25000abd426..f03d744f850 100644 --- a/mysql-test/t/maria.test +++ b/mysql-test/t/maria.test @@ -1118,6 +1118,24 @@ create table t1 (a int) row_format=PAGE; show create table t1; drop table t1; +# Verify that INSERT DELAYED is disabled only for transactional tables +create table t1 (a int) row_format=page; +--error ER_ILLEGAL_HA +insert delayed into t1 values(1); +drop table t1; +create table t1 (a int) row_format=page transactional=0; +insert delayed into t1 values(1); +flush table t1; +select * from t1; +select count(*) from t1; +drop table t1; +create table t1 (a int) row_format=dynamic; +insert delayed into t1 values(1); +flush table t1; +select * from t1; +select count(*) from t1; +drop table t1; + # End of 5.2 tests --disable_result_log -- cgit v1.2.1 From 85da513341c7b512440958302882c765813cc06d Mon Sep 17 00:00:00 2001 From: unknown Date: Wed, 4 Jul 2007 00:50:17 +0300 Subject: Implement applying of REDO entries for - LOGREC_REDO_INSERT_ROW_HEAD - LOGREC_REDO_INSERT_ROW_TAIL - LOGREC_REDO_PURGE_ROW_HEAD - LOGREC_REDO_PURGE_ROW_TAIL sql/sql_yacc.yy: Fixed typo in previous push storage/maria/ma_bitmap.c: Ensure we flush the new bitmap on close storage/maria/ma_blockrec.c: Implement applying of REDO entries for - LOGREC_REDO_INSERT_ROW_HEAD - LOGREC_REDO_INSERT_ROW_TAIL - LOGREC_REDO_PURGE_ROW_HEAD - LOGREC_REDO_PURGE_ROW_TAIL Split some functions into subfunctions to be able to reuse code storage/maria/ma_blockrec.h: Added prototypes for REDO applying functions storage/maria/ma_loghandler.h: Safety fix storage/maria/ma_loghandler_lsn.h: Avoid compiler warnings storage/maria/maria_read_log.c: Added hocks for: - REDO_INSERT_ROW_HEAD - REDO_INSERT_ROW_TAIL - REDO_PURGE_ROW_HEAD - REDO_PURGE_ROW_TAIL Added dummy hooks for: - UNDO_ROW_INSERT - UNDO_ROW_DELETE Changed to use maria_pagecache instead of own pagecache (fixed problem with unitialized share->pagecache) Use maria_panic() at end to ensure that all files are closed properly. Fixed option handling for --debug --- sql/sql_yacc.yy | 2 +- storage/maria/ma_bitmap.c | 2 +- storage/maria/ma_blockrec.c | 493 ++++++++++++++++++++++++++++++++------ storage/maria/ma_blockrec.h | 8 + storage/maria/ma_loghandler.h | 2 +- storage/maria/ma_loghandler_lsn.h | 2 +- storage/maria/maria_read_log.c | 272 +++++++++++++++++++-- 7 files changed, 677 insertions(+), 104 deletions(-) diff --git a/sql/sql_yacc.yy b/sql/sql_yacc.yy index 0653863cc73..1bf198b5b4a 100644 --- a/sql/sql_yacc.yy +++ b/sql/sql_yacc.yy @@ -4280,7 +4280,7 @@ row_types: | DYNAMIC_SYM { $$= ROW_TYPE_DYNAMIC; } | COMPRESSED_SYM { $$= ROW_TYPE_COMPRESSED; } | REDUNDANT_SYM { $$= ROW_TYPE_REDUNDANT; } - | COMPACT_SYM { $$= ROW_TYPE_COMPACT; }; + | COMPACT_SYM { $$= ROW_TYPE_COMPACT; } | PAGE_SYM { $$= ROW_TYPE_PAGE; }; merge_insert_types: diff --git a/storage/maria/ma_bitmap.c b/storage/maria/ma_bitmap.c index f6a8172935f..3376f4abf2c 100644 --- a/storage/maria/ma_bitmap.c +++ b/storage/maria/ma_bitmap.c @@ -296,7 +296,7 @@ void _ma_bitmap_delete_all(MARIA_SHARE *share) { bzero(bitmap->map, share->block_size); memcpy(bitmap->map + share->block_size - 2, maria_bitmap_marker, 2); - bitmap->changed= 0; + bitmap->changed= 1; bitmap->page= 0; bitmap->used_size= bitmap->total_size; } diff --git a/storage/maria/ma_blockrec.c b/storage/maria/ma_blockrec.c index cfa9df02102..453af37089d 100644 --- a/storage/maria/ma_blockrec.c +++ b/storage/maria/ma_blockrec.c @@ -868,7 +868,7 @@ static void calc_record_size(MARIA_HA *info, const byte *record, compact_page() buff Page to compact block_size Size of page - recnr Put empty data after this row + rownr Put empty data after this row extend_block If 1, extend the block at 'rownr' to cover the whole block. */ @@ -980,6 +980,13 @@ static void compact_page(byte *buff, uint block_size, uint rownr, uint length= (uint) (dir - buff) - start_of_found_block; int2store(dir+2, length); } + else + { + /* + TODO: + Update (buff + EMPTY_SPACE_OFFSET) if we remove transid from rows + */ + } buff[PAGE_TYPE_OFFSET]&= ~(byte) PAGE_CAN_BE_COMPACTED; } DBUG_EXECUTE("directory", _ma_print_directory(buff, block_size);); @@ -987,6 +994,37 @@ static void compact_page(byte *buff, uint block_size, uint rownr, } +/* + Create an empty tail or head page + + SYNOPSIS + make_empty_page() + buff Page buffer + block_size Block size + page_type HEAD_PAGE or TAIL_PAGE + + NOTES + EMPTY_SPACE is not updated +*/ + +static void make_empty_page(byte *buff, uint block_size, uint page_type) +{ + + bzero(buff, PAGE_HEADER_SIZE); + /* + We zero the rest of the block to avoid getting old memory information + to disk and to allow the file to be compressed better if archived. + The rest of the code does not assume the block is zeroed above + PAGE_OVERHEAD_SIZE + */ + bzero(buff+ PAGE_HEADER_SIZE, block_size - PAGE_HEADER_SIZE); + buff[PAGE_TYPE_OFFSET]= (byte) page_type; + buff[DIR_COUNT_OFFSET]= 1; + /* Store position to the first row */ + int2store(buff + block_size - PAGE_SUFFIX_SIZE - DIR_ENTRY_SIZE, + PAGE_HEADER_SIZE); +} + /* Read or initialize new head or tail page @@ -1019,6 +1057,7 @@ struct st_row_pos_info uint empty_space; /* Space left on page */ }; + static my_bool get_head_or_tail_page(MARIA_HA *info, MARIA_BITMAP_BLOCK *block, byte *buff, uint length, uint page_type, @@ -1035,25 +1074,12 @@ static my_bool get_head_or_tail_page(MARIA_HA *info, if (block->org_bitmap_value == 0) /* Empty block */ { /* New page */ - bzero(buff, PAGE_HEADER_SIZE); - - /* - We zero the rest of the block to avoid getting old memory information - to disk and to allow the file to be compressed better if archived. - The rest of the code does not assume the block is zeroed above - PAGE_OVERHEAD_SIZE - */ - bzero(buff+ PAGE_HEADER_SIZE, block_size - PAGE_HEADER_SIZE); - - buff[PAGE_TYPE_OFFSET]= (byte) page_type; - buff[DIR_COUNT_OFFSET]= 1; + make_empty_page(buff, block_size, page_type); res->buff= buff; res->empty_space= res->length= (block_size - PAGE_OVERHEAD_SIZE); res->data= (buff + PAGE_HEADER_SIZE); res->dir= res->data + res->length; res->rownr= 0; - /* Store position to the first row */ - int2store(res->dir, PAGE_HEADER_SIZE); DBUG_ASSERT(length <= res->length); } else @@ -1710,8 +1736,12 @@ static my_bool write_block_record(MARIA_HA *info, uint length= (uint) (data - row_pos->data); DBUG_PRINT("info", ("head length: %u", length)); if (length < info->s->base.min_row_length) + { + uint diff_length= info->s->base.min_row_length - length; + bzero(data, diff_length); + data+= diff_length; length= info->s->base.min_row_length; - + } int2store(row_pos->dir + 2, length); /* update empty space at start of block */ row_pos->empty_space-= length; @@ -2471,6 +2501,76 @@ err: } +/* + Delete a directory entry + + SYNOPSIS + delete_dir_entry() + buff Page buffer + block_size Block size + record_number Record number to delete + empty_space Empty space on page after delete + + RETURN + -1 Error on page + 0 ok + 1 Page is now empty +*/ + +static int delete_dir_entry(byte *buff, uint block_size, uint record_number, + uint *empty_space_res) +{ + uint number_of_records= (uint) ((uchar *) buff)[DIR_COUNT_OFFSET]; + uint length, empty_space; + byte *dir; + DBUG_ENTER("delete_dir_entry"); + +#ifdef SANITY_CHECKS + if (record_number >= number_of_records || + record_number > ((block_size - LSN_SIZE - PAGE_TYPE_SIZE - 1 - + PAGE_SUFFIX_SIZE) / DIR_ENTRY_SIZE)) + { + DBUG_PRINT("error", ("record_number: %u number_of_records: %u", + record_number, number_of_records)); + + DBUG_RETURN(-1); + } +#endif + + empty_space= uint2korr(buff + EMPTY_SPACE_OFFSET); + dir= (buff + block_size - DIR_ENTRY_SIZE * record_number - + DIR_ENTRY_SIZE - PAGE_SUFFIX_SIZE); + dir[0]= dir[1]= 0; /* Delete entry */ + length= uint2korr(dir + 2); + + if (record_number == number_of_records - 1) + { + /* Delete this entry and all following empty directory entries */ + byte *end= buff + block_size - PAGE_SUFFIX_SIZE; + do + { + number_of_records--; + dir+= DIR_ENTRY_SIZE; + empty_space+= DIR_ENTRY_SIZE; + } while (dir < end && dir[0] == 0 && dir[1] == 0); + buff[DIR_COUNT_OFFSET]= (byte) (uchar) number_of_records; + } + empty_space+= length; + if (number_of_records != 0) + { + /* Update directory */ + int2store(buff + EMPTY_SPACE_OFFSET, empty_space); + buff[PAGE_TYPE_OFFSET]|= (byte) PAGE_CAN_BE_COMPACTED; + + *empty_space_res= empty_space; + DBUG_RETURN(0); + } + buff[PAGE_TYPE_OFFSET]= UNALLOCATED_PAGE; + *empty_space_res= block_size; + DBUG_RETURN(1); +} + + /* Delete a head a tail part @@ -2493,11 +2593,12 @@ static my_bool delete_head_or_tail(MARIA_HA *info, my_bool head) { MARIA_SHARE *share= info->s; - uint number_of_records, empty_space, length; + uint empty_space; uint block_size= share->block_size; - byte *buff, *dir; + byte *buff; LSN lsn; MARIA_PINNED_PAGE page_link; + int res; DBUG_ENTER("delete_head_or_tail"); info->keyread_buff_used= 1; @@ -2511,60 +2612,30 @@ static my_bool delete_head_or_tail(MARIA_HA *info, page_link.unlock= PAGECACHE_LOCK_WRITE_UNLOCK; push_dynamic(&info->pinned_pages, (void*) &page_link); - number_of_records= (uint) ((uchar *) buff)[DIR_COUNT_OFFSET]; -#ifdef SANITY_CHECKS - if (record_number >= number_of_records || - record_number > ((block_size - LSN_SIZE - PAGE_TYPE_SIZE - 1 - - PAGE_SUFFIX_SIZE) / DIR_ENTRY_SIZE)) - { - DBUG_PRINT("error", ("record_number: %u number_of_records: %u", - record_number, number_of_records)); + res= delete_dir_entry(buff, block_size, record_number, &empty_space); + if (res < 0) DBUG_RETURN(1); - } -#endif - - dir= (buff + block_size - DIR_ENTRY_SIZE * record_number - - DIR_ENTRY_SIZE - PAGE_SUFFIX_SIZE); - dir[0]= dir[1]= 0; /* Delete entry */ - length= uint2korr(dir + 2); - empty_space= uint2korr(buff + EMPTY_SPACE_OFFSET); - - if (record_number == number_of_records - 1) - { - /* Delete this entry and all following empty directory entries */ - byte *end= buff + block_size - PAGE_SUFFIX_SIZE; - do - { - number_of_records--; - dir+= DIR_ENTRY_SIZE; - empty_space+= DIR_ENTRY_SIZE; - } while (dir < end && dir[0] == 0 && dir[1] == 0); - buff[DIR_COUNT_OFFSET]= (byte) (uchar) number_of_records; - } - empty_space+= length; - if (number_of_records != 0) + if (res == 0) { uchar log_data[FILEID_STORE_SIZE + PAGE_STORE_SIZE + DIRPOS_STORE_SIZE]; LEX_STRING log_array[TRANSLOG_INTERNAL_PARTS + 1]; - /* Update directory */ - int2store(buff + EMPTY_SPACE_OFFSET, empty_space); - buff[PAGE_TYPE_OFFSET]|= (byte) PAGE_CAN_BE_COMPACTED; - DBUG_ASSERT(share->pagecache->block_size == block_size); - - /* Log REDO data */ - page_store(log_data+ FILEID_STORE_SIZE, page); - dirpos_store(log_data+ FILEID_STORE_SIZE + PAGE_STORE_SIZE, + if (info->s->base.transactional) + { + /* Log REDO data */ + page_store(log_data+ FILEID_STORE_SIZE, page); + dirpos_store(log_data+ FILEID_STORE_SIZE + PAGE_STORE_SIZE, record_number); - log_array[TRANSLOG_INTERNAL_PARTS + 0].str= (char*) log_data; - log_array[TRANSLOG_INTERNAL_PARTS + 0].length= sizeof(log_data); - if (translog_write_record(&lsn, (head ? LOGREC_REDO_PURGE_ROW_HEAD : - LOGREC_REDO_PURGE_ROW_TAIL), - info->trn, share, sizeof(log_data), - TRANSLOG_INTERNAL_PARTS + 1, log_array, - log_data)) - DBUG_RETURN(1); + log_array[TRANSLOG_INTERNAL_PARTS + 0].str= (char*) log_data; + log_array[TRANSLOG_INTERNAL_PARTS + 0].length= sizeof(log_data); + if (translog_write_record(&lsn, (head ? LOGREC_REDO_PURGE_ROW_HEAD : + LOGREC_REDO_PURGE_ROW_TAIL), + info->trn, share, sizeof(log_data), + TRANSLOG_INTERNAL_PARTS + 1, log_array, + log_data)) + DBUG_RETURN(1); + } if (pagecache_write(share->pagecache, &info->dfile, page, 0, buff, share->page_type, @@ -2579,20 +2650,21 @@ static my_bool delete_head_or_tail(MARIA_HA *info, PAGE_STORE_SIZE + PAGERANGE_STORE_SIZE]; LEX_STRING log_array[TRANSLOG_INTERNAL_PARTS + 1]; - pagerange_store(log_data + FILEID_STORE_SIZE, 1); - page_store(log_data+ FILEID_STORE_SIZE + PAGERANGE_STORE_SIZE, page); - pagerange_store(log_data + FILEID_STORE_SIZE + PAGERANGE_STORE_SIZE + - PAGE_STORE_SIZE, 1); - log_array[TRANSLOG_INTERNAL_PARTS + 0].str= (char*) log_data; - log_array[TRANSLOG_INTERNAL_PARTS + 0].length= sizeof(log_data); - if (translog_write_record(&lsn, LOGREC_REDO_PURGE_BLOCKS, - info->trn, share, sizeof(log_data), - TRANSLOG_INTERNAL_PARTS + 1, log_array, - log_data)) - DBUG_RETURN(1); - + if (info->s->base.transactional) + { + pagerange_store(log_data + FILEID_STORE_SIZE, 1); + page_store(log_data+ FILEID_STORE_SIZE + PAGERANGE_STORE_SIZE, page); + pagerange_store(log_data + FILEID_STORE_SIZE + PAGERANGE_STORE_SIZE + + PAGE_STORE_SIZE, 1); + log_array[TRANSLOG_INTERNAL_PARTS + 0].str= (char*) log_data; + log_array[TRANSLOG_INTERNAL_PARTS + 0].length= sizeof(log_data); + if (translog_write_record(&lsn, LOGREC_REDO_PURGE_BLOCKS, + info->trn, share, sizeof(log_data), + TRANSLOG_INTERNAL_PARTS + 1, log_array, + log_data)) + DBUG_RETURN(1); + } /* Write the empty page (needed only for REPAIR to work) */ - buff[PAGE_TYPE_OFFSET]= UNALLOCATED_PAGE; if (pagecache_write(share->pagecache, &info->dfile, page, 0, buff, share->page_type, @@ -4024,3 +4096,268 @@ static size_t fill_update_undo_parts(MARIA_HA *info, const byte *oldrec, row_length+= start_log_parts->length; DBUG_RETURN(row_length); } + +/*************************************************************************** + Applying of REDO log records +***************************************************************************/ + +/* + Apply LOGREC_REDO_INSERT_ROW_HEAD & LOGREC_REDO_INSERT_ROW_TAIL + + SYNOPSIS + _ma_apply_redo_insert_row_head_or_tail() + info Maria handler + lsn LSN to put on page + page_type HEAD_PAGE or TAIL_PAGE + header Header (without FILEID) + data Data to be put on page + data_length Length of data + + RETURN + 0 ok + # Error number +*/ + +uint _ma_apply_redo_insert_row_head_or_tail(MARIA_HA *info, LSN lsn, + uint page_type, + const byte *header, + const byte *data, + size_t data_length) +{ + MARIA_SHARE *share= info->s; + ulonglong page; + uint rownr, empty_space; + uint block_size= share->block_size; + uint rec_offset; + byte *buff= info->keyread_buff, *dir; + DBUG_ENTER("_ma_apply_redo_insert_row_head"); + + info->keyread_buff_used= 1; + page= page_korr(header); + rownr= dirpos_korr(header+PAGE_STORE_SIZE); + + if (page * info->s->block_size > info->state->data_file_length) + { + /* New page at end of file */ + DBUG_ASSERT(rownr == 0); + if (rownr != 0) + goto err; + make_empty_page(buff, block_size, page_type); + empty_space= (block_size - PAGE_OVERHEAD_SIZE); + rec_offset= PAGE_HEADER_SIZE; + dir= buff+ block_size - PAGE_SUFFIX_SIZE - DIR_ENTRY_SIZE; + + /* Update that file is extended */ + info->state->data_file_length= page * info->s->block_size; + } + else + { + uint max_entry; + if (!(buff= pagecache_read(share->pagecache, + &info->dfile, + page, 0, + buff, PAGECACHE_PLAIN_PAGE, + PAGECACHE_LOCK_LEFT_UNLOCKED, 0))) + DBUG_RETURN(my_errno); + if (lsn_korr(buff) >= lsn) + { + /* Already applied */ + + /* Fix bitmap, just in case */ + empty_space= uint2korr(buff + EMPTY_SPACE_OFFSET); + if (_ma_bitmap_set(info, page, page_type == HEAD_PAGE, empty_space)) + DBUG_RETURN(my_errno); + DBUG_RETURN(0); + } + + max_entry= (uint) ((uchar*) buff)[DIR_COUNT_OFFSET]; + if (((buff[PAGE_TYPE_OFFSET] & PAGE_TYPE_MASK) != page_type)) + { + /* + This is a page that has been freed before and now should be + changed to new type. + */ + if ((buff[PAGE_TYPE_OFFSET] & PAGE_TYPE_MASK) != BLOB_PAGE && + (buff[PAGE_TYPE_OFFSET] & PAGE_TYPE_MASK) != UNALLOCATED_PAGE) + goto err; + make_empty_page(buff, block_size, page_type); + empty_space= (block_size - PAGE_OVERHEAD_SIZE); + rec_offset= PAGE_HEADER_SIZE; + dir= buff+ block_size - PAGE_SUFFIX_SIZE - DIR_ENTRY_SIZE; + } + else + { + dir= (buff + block_size - DIR_ENTRY_SIZE * (rownr + 1) - + PAGE_SUFFIX_SIZE); + empty_space= uint2korr(buff + EMPTY_SPACE_OFFSET); + + if (max_entry >= rownr) + { + /* Add directory entry first in directory and data last on page */ + DBUG_ASSERT(max_entry == rownr); + if (max_entry != rownr) + goto err; + rec_offset= (uint2korr(dir + DIR_ENTRY_SIZE) + + uint2korr(dir + DIR_ENTRY_SIZE +2)); + if ((uint) (dir - buff) < rec_offset + data_length) + { + /* Create place for directory & data */ + compact_page(buff, block_size, max_entry - 1, 0); + rec_offset= (uint2korr(dir + DIR_ENTRY_SIZE) + + uint2korr(dir + DIR_ENTRY_SIZE +2)); + empty_space= uint2korr(buff + EMPTY_SPACE_OFFSET); + DBUG_ASSERT(!((uint) (dir - buff) < rec_offset + data_length)); + if ((uint) (dir - buff) < rec_offset + data_length) + goto err; + } + buff[DIR_COUNT_OFFSET]= (byte) (uchar) max_entry+1; + int2store(dir, rec_offset); + empty_space-= DIR_ENTRY_SIZE; + } + else + { + /* reuse old empty entry */ + byte *pos, *end, *end_data; + DBUG_ASSERT(uint2korr(dir) == 0); + if (uint2korr(dir)) + goto err; /* Should have been empty */ + + /* Find start of where we can put data */ + end= (buff + block_size - DIR_ENTRY_SIZE * max_entry - + PAGE_SUFFIX_SIZE); + for (pos= dir ; pos >= end ; pos-= DIR_ENTRY_SIZE) + { + if ((rec_offset= uint2korr(pos))) + { + rec_offset+= uint2korr(pos+2); + break; + } + } + DBUG_ASSERT(pos >= end); + if (pos < end) /* Wrong directory */ + goto err; + + /* find end data */ + end_data= end; /* Start of directory */ + end= (buff + block_size - PAGE_SUFFIX_SIZE); + for (pos= dir ; pos < end ; pos+= DIR_ENTRY_SIZE) + { + uint offset; + if ((offset= uint2korr(pos))) + { + end_data= buff + offset; + break; + } + } + if ((uint) (end_data - (buff + rec_offset)) < data_length) + { + uint length; + /* Not enough continues space, compact page to get more */ + int2store(dir, rec_offset); + compact_page(buff, block_size, rownr, 1); + rec_offset= uint2korr(dir); + length= uint2korr(dir+2); + DBUG_ASSERT(length >= data_length); + if (length < data_length) + goto err; + empty_space= length; + } + } + } + } + /* Copy data */ + int2store(dir+2, data_length); + memcpy(buff + rec_offset, data, data_length); + empty_space-= data_length; + int2store(buff + EMPTY_SPACE_OFFSET, empty_space); + + /* Write modified page */ + lsn_store(buff, lsn); + if (pagecache_write(share->pagecache, + &info->dfile, page, 0, + buff, PAGECACHE_PLAIN_PAGE, + PAGECACHE_LOCK_LEFT_UNLOCKED, + PAGECACHE_PIN_LEFT_UNPINNED, + PAGECACHE_WRITE_DELAY, 0)) + DBUG_RETURN(my_errno); + + /* Fix bitmap */ + if (_ma_bitmap_set(info, page, page_type == HEAD_PAGE, empty_space)) + DBUG_RETURN(my_errno); + + DBUG_RETURN(0); + +err: + DBUG_RETURN(HA_ERR_WRONG_IN_RECORD); +} + + +/* + Apply LOGREC_REDO_PURGE_ROW_HEAD & LOGREC_REDO_PURGE_ROW_TAIL + + SYNOPSIS + _ma_apply_redo_purge_row_head_or_tail() + info Maria handler + lsn LSN to put on page + page_type HEAD_PAGE or TAIL_PAGE + header Header (without FILEID) + data Data to be put on page + data_length Length of data + + NOTES + This function is very similar to delete_head_or_tail() + + RETURN + 0 ok + # Error number +*/ + +uint _ma_apply_redo_purge_row_head_or_tail(MARIA_HA *info, LSN lsn, + uint page_type, + const byte *header) +{ + MARIA_SHARE *share= info->s; + ulonglong page; + uint record_number, empty_space; + uint block_size= share->block_size; + byte *buff= info->keyread_buff; + DBUG_ENTER("_ma_apply_redo_purge_row_head_or_tail"); + + info->keyread_buff_used= 1; + page= page_korr(header); + record_number= dirpos_korr(header+PAGE_STORE_SIZE); + + if (!(buff= pagecache_read(share->pagecache, + &info->dfile, + page, 0, + buff, PAGECACHE_PLAIN_PAGE, + PAGECACHE_LOCK_LEFT_UNLOCKED, 0))) + DBUG_RETURN(my_errno); + DBUG_ASSERT((buff[PAGE_TYPE_OFFSET] & PAGE_TYPE_MASK) == (byte) page_type); + + if (lsn_korr(buff) >= lsn) + { + /* Already applied */ + empty_space= uint2korr(buff + EMPTY_SPACE_OFFSET); + if (_ma_bitmap_set(info, page, page_type == HEAD_PAGE, empty_space)) + DBUG_RETURN(my_errno); + DBUG_RETURN(0); + } + + if (delete_dir_entry(buff, block_size, record_number, &empty_space) < 0) + DBUG_RETURN(HA_ERR_WRONG_IN_RECORD); + + if (pagecache_write(share->pagecache, + &info->dfile, page, 0, + buff, PAGECACHE_PLAIN_PAGE, + PAGECACHE_LOCK_LEFT_UNLOCKED, + PAGECACHE_PIN_LEFT_UNPINNED, + PAGECACHE_WRITE_DELAY, 0)) + DBUG_RETURN(my_errno); + + /* This will work even if the page was marked as UNALLOCATED_PAGE */ + if (_ma_bitmap_set(info, page, page_type == HEAD_PAGE, empty_space)) + DBUG_RETURN(my_errno); + + DBUG_RETURN(0); +} diff --git a/storage/maria/ma_blockrec.h b/storage/maria/ma_blockrec.h index 819d1c2e4d2..0ed0898859c 100644 --- a/storage/maria/ma_blockrec.h +++ b/storage/maria/ma_blockrec.h @@ -178,3 +178,11 @@ my_bool _ma_check_if_right_bitmap_type(MARIA_HA *info, ulonglong page, uint *bitmap_pattern); void _ma_bitmap_delete_all(MARIA_SHARE *share); +uint _ma_apply_redo_insert_row_head_or_tail(MARIA_HA *info, LSN lsn, + uint page_type, + const byte *header, + const byte *data, + size_t data_length); +uint _ma_apply_redo_purge_row_head_or_tail(MARIA_HA *info, LSN lsn, + uint page_type, + const byte *header); diff --git a/storage/maria/ma_loghandler.h b/storage/maria/ma_loghandler.h index f2bfd2c9d7e..230f999c19a 100644 --- a/storage/maria/ma_loghandler.h +++ b/storage/maria/ma_loghandler.h @@ -62,7 +62,7 @@ struct st_maria_share; #define pagerange_store(T,A) int2store(T,A) #define fileid_korr(P) uint2korr(P) #define page_korr(P) uint5korr(P) -#define dirpos_korr(P) (P[0]) +#define dirpos_korr(P) ((P)[0]) #define pagerange_korr(P) uint2korr(P) /* diff --git a/storage/maria/ma_loghandler_lsn.h b/storage/maria/ma_loghandler_lsn.h index 34cb7616b74..e034834aa20 100644 --- a/storage/maria/ma_loghandler_lsn.h +++ b/storage/maria/ma_loghandler_lsn.h @@ -45,7 +45,7 @@ typedef TRANSLOG_ADDRESS LSN; #define LSN_OFFSET(L) ((L) & 0xFFFFFFFFL) /* Makes lsn/log address from file number and record offset */ -#define MAKE_LSN(F,S) ((((uint64)(F)) << 32) | (S)) +#define MAKE_LSN(F,S) ((LSN) ((((uint64)(F)) << 32) | (S))) /* checks LSN */ #define LSN_VALID(L) \ diff --git a/storage/maria/maria_read_log.c b/storage/maria/maria_read_log.c index 5b2d5b057c2..e6911007230 100644 --- a/storage/maria/maria_read_log.c +++ b/storage/maria/maria_read_log.c @@ -14,20 +14,22 @@ Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ #include "maria_def.h" +#include #include #define PCACHE_SIZE (1024*1024*10) #define LOG_FLAGS 0 #define LOG_FILE_SIZE (1024L*1024L) - -static PAGECACHE pagecache; - static const char *load_default_groups[]= { "maria_read_log",0 }; static void get_options(int *argc,char * * *argv); #ifndef DBUG_OFF -static const char *default_dbug_option; +#if defined(__WIN__) +const char *default_dbug_option= "d:t:i:O,\\maria_read_log.trace"; +#else +const char *default_dbug_option= "d:t:i:o,/tmp/maria_read_log.trace"; #endif +#endif /* DBUG_OFF */ static my_bool opt_only_display, opt_display_and_apply; struct TRN_FOR_RECOVERY @@ -55,7 +57,25 @@ prototype_exec_hook(CHECKPOINT); prototype_exec_hook(REDO_CREATE_TABLE); prototype_exec_hook(FILE_ID); prototype_exec_hook(REDO_INSERT_ROW_HEAD); +prototype_exec_hook(REDO_INSERT_ROW_TAIL); +prototype_exec_hook(REDO_PURGE_ROW_HEAD); +prototype_exec_hook(REDO_PURGE_ROW_TAIL); +prototype_exec_hook(UNDO_ROW_INSERT); +prototype_exec_hook(UNDO_ROW_DELETE); prototype_exec_hook(COMMIT); + + +/* + TODO: Avoid mallocs in exec. + + Proposed fix: + Add either a context/buffer argument to all exec_hook functions + or add 'record_buffer' and 'record_buffer_length' to + TRANSLOG_HEADER_BUFFER. + With this we could use my_realloc() instead of my_malloc() to + allocate data and save some mallocs. +*/ + /* To implement REDO_DROP_TABLE and REDO_RENAME_TABLE, we would need to go through the all_tables[] array, find all open instances of the @@ -78,19 +98,6 @@ int main(int argc, char **argv) maria_data_root= "."; -#ifndef DBUG_OFF -#if defined(__WIN__) - default_dbug_option= "d:t:i:O,\\maria_read_log.trace"; -#else - default_dbug_option= "d:t:i:o,/tmp/maria_read_log.trace"; -#endif - if (argc > 1) - { - DBUG_SET(default_dbug_option); - DBUG_SET_INITIAL(default_dbug_option); - } -#endif - if (maria_init()) { fprintf(stderr, "Can't init Maria engine (%d)\n", errno); @@ -107,7 +114,7 @@ int main(int argc, char **argv) fprintf(stderr, "Can't find any log\n"); goto err; } - if (init_pagecache(&pagecache, PCACHE_SIZE, 0, 0, + if (init_pagecache(maria_pagecache, PCACHE_SIZE, 0, 0, TRANSLOG_PAGE_SIZE) == 0) { fprintf(stderr, "Got error in init_pagecache() (errno: %d)\n", errno); @@ -119,7 +126,7 @@ int main(int argc, char **argv) But if it finds a log and this log was crashed, it will create a new log, which is useless. TODO: start log handler in read-only mode. */ - if (translog_init(".", LOG_FILE_SIZE, 50112, 0, &pagecache, + if (translog_init(".", LOG_FILE_SIZE, 50112, 0, maria_pagecache, TRANSLOG_DEFAULT_FLAGS)) { fprintf(stderr, "Can't init loghandler (%d)\n", errno); @@ -137,6 +144,11 @@ int main(int argc, char **argv) install_exec_hook(REDO_CREATE_TABLE); install_exec_hook(FILE_ID); install_exec_hook(REDO_INSERT_ROW_HEAD); + install_exec_hook(REDO_INSERT_ROW_TAIL); + install_exec_hook(REDO_PURGE_ROW_HEAD); + install_exec_hook(REDO_PURGE_ROW_TAIL); + install_exec_hook(UNDO_ROW_INSERT); + install_exec_hook(UNDO_ROW_DELETE); install_exec_hook(COMMIT); if (opt_only_display) @@ -261,7 +273,7 @@ err: /* don't touch anything more, in case we hit a bug */ exit(1); end: - maria_end(); + maria_panic(HA_PANIC_CLOSE); free_defaults(default_argv); my_end(0); exit(0); @@ -318,7 +330,13 @@ get_one_option(int optid __attribute__((unused)), const struct my_option *opt __attribute__((unused)), char *argument __attribute__((unused))) { - /* for now there is nothing special with our options */ + switch (optid) { +#ifndef DBUG_OFF + case '#': + DBUG_SET_INITIAL(argument ? argument : default_dbug_option); + break; + } +#endif return 0; } @@ -619,6 +637,140 @@ prototype_exec_hook(REDO_INSERT_ROW_HEAD) ulonglong page; MARIA_HA *info; char llbuf[22]; + byte *buff= 0; + + sid= fileid_korr(rec->header); + page= page_korr(rec->header + FILEID_STORE_SIZE); + llstr(page, llbuf); + printf("For page %s of table of short id %u", llbuf, sid); + info= all_tables[sid]; + if (info == NULL) + { + printf(", table skipped, so skipping record\n"); + goto end; + } + printf(", '%s'", info->s->open_file_name); + if (cmp_translog_addr(info->s->state.create_rename_lsn, rec->lsn) >= 0) + { + printf(", has create_rename_lsn (%lu,0x%lx) is more recent than log" + " record\n", + (ulong) LSN_FILE_NO(rec->lsn), (ulong) LSN_OFFSET(rec->lsn)); + goto end; + } + /* + Soon we will also skip the page depending on the rec_lsn for this page in + the checkpoint record, but this is not absolutely needed for now (just + assume we have made no checkpoint). + */ + printf(", applying record\n"); + /* + If REDO's LSN is > page's LSN (read from disk), we are going to modify the + page and change its LSN. The normal runtime code stores the UNDO's LSN + into the page. Here storing the REDO's LSN (rec->lsn) would work + (we are not writing to the log here, so don't have to "flush up to UNDO's + LSN"). But in a test scenario where we do updates at runtime, then remove + tables, apply the log and check that this results in the same table as at + runtime, putting the same LSN as runtime had done will decrease + differences. So we use the UNDO's LSN which is current_group_end_lsn. + */ + + if ((!(buff= (byte*) my_malloc(rec->record_length, MYF(MY_WME)))) || + (translog_read_record(rec->lsn, 0, rec->record_length, buff, NULL) != + rec->record_length)) + { + fprintf(stderr, "Failed to read record\n"); + goto end; + } + if (_ma_apply_redo_insert_row_head_or_tail(info, rec->lsn, HEAD_PAGE, + rec->header + FILEID_STORE_SIZE, + buff + (rec->record_length - + rec->non_header_data_len), + rec->non_header_data_len)) + goto end; + my_free(buff, MYF(0)); + return 0; + +end: + /* as we don't have apply working: */ + my_free(buff, MYF(MY_ALLOW_ZERO_PTR)); + return 1; +} + + +prototype_exec_hook(REDO_INSERT_ROW_TAIL) +{ + uint16 sid; + ulonglong page; + MARIA_HA *info; + char llbuf[22]; + byte *buff= 0; + + sid= fileid_korr(rec->header); + page= page_korr(rec->header + FILEID_STORE_SIZE); + llstr(page, llbuf); + printf("For page %s of table of short id %u", llbuf, sid); + info= all_tables[sid]; + if (info == NULL) + { + printf(", table skipped, so skipping record\n"); + goto end; + } + printf(", '%s'", info->s->open_file_name); + if (cmp_translog_addr(info->s->state.create_rename_lsn, rec->lsn) >= 0) + { + printf(", has create_rename_lsn (%lu,0x%lx) is more recent than log" + " record\n", + (ulong) LSN_FILE_NO(rec->lsn), (ulong) LSN_OFFSET(rec->lsn)); + goto end; + } + /* + Soon we will also skip the page depending on the rec_lsn for this page in + the checkpoint record, but this is not absolutely needed for now (just + assume we have made no checkpoint). + */ + printf(", applying record\n"); + /* + If REDO's LSN is > page's LSN (read from disk), we are going to modify the + page and change its LSN. The normal runtime code stores the UNDO's LSN + into the page. Here storing the REDO's LSN (rec->lsn) would work + (we are not writing to the log here, so don't have to "flush up to UNDO's + LSN"). But in a test scenario where we do updates at runtime, then remove + tables, apply the log and check that this results in the same table as at + runtime, putting the same LSN as runtime had done will decrease + differences. So we use the UNDO's LSN which is current_group_end_lsn. + */ + + if ((!(buff= (byte*) my_malloc(rec->record_length, MYF(MY_WME)))) || + (translog_read_record(rec->lsn, 0, rec->record_length, buff, NULL) != + rec->record_length)) + { + fprintf(stderr, "Failed to read record\n"); + goto end; + } + if (_ma_apply_redo_insert_row_head_or_tail(info, rec->lsn, TAIL_PAGE, + rec->header + FILEID_STORE_SIZE, + buff + (rec->record_length - + rec->non_header_data_len), + rec->non_header_data_len)) + goto end; + + my_free(buff, MYF(0)); + return 0; + +end: + /* as we don't have apply working: */ + my_free(buff, MYF(MY_ALLOW_ZERO_PTR)); + return 1; +} + + +prototype_exec_hook(REDO_PURGE_ROW_HEAD) +{ + uint16 sid; + ulonglong page; + MARIA_HA *info; + char llbuf[22]; + sid= fileid_korr(rec->header); page= page_korr(rec->header + FILEID_STORE_SIZE); llstr(page, llbuf); @@ -653,13 +805,89 @@ prototype_exec_hook(REDO_INSERT_ROW_HEAD) runtime, putting the same LSN as runtime had done will decrease differences. So we use the UNDO's LSN which is current_group_end_lsn. */ - DBUG_ASSERT("Monty" == "this is the place"); + + if (_ma_apply_redo_purge_row_head_or_tail(info, rec->lsn, HEAD_PAGE, + rec->header + FILEID_STORE_SIZE)) + goto end; + + return 0; + end: /* as we don't have apply working: */ return 1; } +prototype_exec_hook(REDO_PURGE_ROW_TAIL) +{ + uint16 sid; + ulonglong page; + MARIA_HA *info; + char llbuf[22]; + + sid= fileid_korr(rec->header); + page= page_korr(rec->header + FILEID_STORE_SIZE); + llstr(page, llbuf); + printf("For page %s of table of short id %u", llbuf, sid); + info= all_tables[sid]; + if (info == NULL) + { + printf(", table skipped, so skipping record\n"); + goto end; + } + printf(", '%s'", info->s->open_file_name); + if (cmp_translog_addr(info->s->state.create_rename_lsn, rec->lsn) >= 0) + { + printf(", has create_rename_lsn (%lu,0x%lx) is more recent than log" + " record\n", + (ulong) LSN_FILE_NO(rec->lsn), (ulong) LSN_OFFSET(rec->lsn)); + goto end; + } + /* + Soon we will also skip the page depending on the rec_lsn for this page in + the checkpoint record, but this is not absolutely needed for now (just + assume we have made no checkpoint). + */ + printf(", applying record\n"); + /* + If REDO's LSN is > page's LSN (read from disk), we are going to modify the + page and change its LSN. The normal runtime code stores the UNDO's LSN + into the page. Here storing the REDO's LSN (rec->lsn) would work + (we are not writing to the log here, so don't have to "flush up to UNDO's + LSN"). But in a test scenario where we do updates at runtime, then remove + tables, apply the log and check that this results in the same table as at + runtime, putting the same LSN as runtime had done will decrease + differences. So we use the UNDO's LSN which is current_group_end_lsn. + */ + + if (_ma_apply_redo_purge_row_head_or_tail(info, rec->lsn, TAIL_PAGE, + rec->header + FILEID_STORE_SIZE)) + goto end; + + return 0; + +end: + /* as we don't have apply working: */ + return 1; +} + + +static int exec_LOGREC_UNDO_ROW_INSERT(const TRANSLOG_HEADER_BUFFER *rec + __attribute__((unused))) +{ + /* Ignore this during the redo phase */ + return 0; +} + +static int exec_LOGREC_UNDO_ROW_DELETE(const TRANSLOG_HEADER_BUFFER *rec + __attribute__((unused))) +{ + /* Ignore this during the redo phase */ + return 0; +} + + + prototype_exec_hook(COMMIT) { uint16 sid= rec->short_trid; -- cgit v1.2.1 From a898a7b63e65b46da82225ab8823897963dcbb3a Mon Sep 17 00:00:00 2001 From: unknown Date: Wed, 4 Jul 2007 01:04:21 +0300 Subject: After merge fixes Note that ma_test_all doesn't work for the moment. (ma_test1 -s -M -T fails because it uses the dummy_transaction_object) storage/maria/ma_blockrec.c: After merge fixes --- storage/maria/ma_blockrec.c | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/storage/maria/ma_blockrec.c b/storage/maria/ma_blockrec.c index 7b8bc9ea2f0..06c1df16663 100644 --- a/storage/maria/ma_blockrec.c +++ b/storage/maria/ma_blockrec.c @@ -2612,7 +2612,7 @@ static my_bool delete_head_or_tail(MARIA_HA *info, uchar log_data[FILEID_STORE_SIZE + PAGE_STORE_SIZE + DIRPOS_STORE_SIZE]; LEX_STRING log_array[TRANSLOG_INTERNAL_PARTS + 1]; - if (info->s->base.transactional) + if (info->s->now_transactional) { /* Log REDO data */ page_store(log_data+ FILEID_STORE_SIZE, page); @@ -2642,7 +2642,7 @@ static my_bool delete_head_or_tail(MARIA_HA *info, PAGE_STORE_SIZE + PAGERANGE_STORE_SIZE]; LEX_STRING log_array[TRANSLOG_INTERNAL_PARTS + 1]; - if (info->s->base.transactional) + if (info->s->now_transactional) { pagerange_store(log_data + FILEID_STORE_SIZE, 1); page_store(log_data+ FILEID_STORE_SIZE + PAGERANGE_STORE_SIZE, page); -- cgit v1.2.1 From 6ed46be70e0c1886231729458b94fb3321ac1a31 Mon Sep 17 00:00:00 2001 From: unknown Date: Wed, 4 Jul 2007 12:39:19 +0300 Subject: Fixed bytes to uchar and gptr to uchar* --- storage/maria/ma_blockrec.c | 30 +++++++++++++++--------------- storage/maria/ma_blockrec.h | 6 +++--- storage/maria/ma_check.c | 6 +++--- storage/maria/ma_loghandler.c | 2 +- storage/maria/ma_loghandler.h | 2 +- storage/maria/maria_read_log.c | 12 ++++++------ 6 files changed, 29 insertions(+), 29 deletions(-) diff --git a/storage/maria/ma_blockrec.c b/storage/maria/ma_blockrec.c index 6e2dfc4fe15..3292ce65ff1 100644 --- a/storage/maria/ma_blockrec.c +++ b/storage/maria/ma_blockrec.c @@ -999,7 +999,7 @@ static void compact_page(uchar *buff, uint block_size, uint rownr, EMPTY_SPACE is not updated */ -static void make_empty_page(byte *buff, uint block_size, uint page_type) +static void make_empty_page(uchar *buff, uint block_size, uint page_type) { bzero(buff, PAGE_HEADER_SIZE); @@ -1010,7 +1010,7 @@ static void make_empty_page(byte *buff, uint block_size, uint page_type) PAGE_OVERHEAD_SIZE */ bzero(buff+ PAGE_HEADER_SIZE, block_size - PAGE_HEADER_SIZE); - buff[PAGE_TYPE_OFFSET]= (byte) page_type; + buff[PAGE_TYPE_OFFSET]= (uchar) page_type; buff[DIR_COUNT_OFFSET]= 1; /* Store position to the first row */ int2store(buff + block_size - PAGE_SUFFIX_SIZE - DIR_ENTRY_SIZE, @@ -2509,12 +2509,12 @@ err: 1 Page is now empty */ -static int delete_dir_entry(byte *buff, uint block_size, uint record_number, +static int delete_dir_entry(uchar *buff, uint block_size, uint record_number, uint *empty_space_res) { uint number_of_records= (uint) ((uchar *) buff)[DIR_COUNT_OFFSET]; uint length, empty_space; - byte *dir; + uchar *dir; DBUG_ENTER("delete_dir_entry"); #ifdef SANITY_CHECKS @@ -2538,21 +2538,21 @@ static int delete_dir_entry(byte *buff, uint block_size, uint record_number, if (record_number == number_of_records - 1) { /* Delete this entry and all following empty directory entries */ - byte *end= buff + block_size - PAGE_SUFFIX_SIZE; + uchar *end= buff + block_size - PAGE_SUFFIX_SIZE; do { number_of_records--; dir+= DIR_ENTRY_SIZE; empty_space+= DIR_ENTRY_SIZE; } while (dir < end && dir[0] == 0 && dir[1] == 0); - buff[DIR_COUNT_OFFSET]= (byte) (uchar) number_of_records; + buff[DIR_COUNT_OFFSET]= (uchar) number_of_records; } empty_space+= length; if (number_of_records != 0) { /* Update directory */ int2store(buff + EMPTY_SPACE_OFFSET, empty_space); - buff[PAGE_TYPE_OFFSET]|= (byte) PAGE_CAN_BE_COMPACTED; + buff[PAGE_TYPE_OFFSET]|= (uchar) PAGE_CAN_BE_COMPACTED; *empty_space_res= empty_space; DBUG_RETURN(0); @@ -4111,8 +4111,8 @@ static size_t fill_update_undo_parts(MARIA_HA *info, const uchar *oldrec, uint _ma_apply_redo_insert_row_head_or_tail(MARIA_HA *info, LSN lsn, uint page_type, - const byte *header, - const byte *data, + const uchar *header, + const uchar *data, size_t data_length) { MARIA_SHARE *share= info->s; @@ -4120,7 +4120,7 @@ uint _ma_apply_redo_insert_row_head_or_tail(MARIA_HA *info, LSN lsn, uint rownr, empty_space; uint block_size= share->block_size; uint rec_offset; - byte *buff= info->keyread_buff, *dir; + uchar *buff= info->keyread_buff, *dir; DBUG_ENTER("_ma_apply_redo_insert_row_head"); info->keyread_buff_used= 1; @@ -4201,14 +4201,14 @@ uint _ma_apply_redo_insert_row_head_or_tail(MARIA_HA *info, LSN lsn, if ((uint) (dir - buff) < rec_offset + data_length) goto err; } - buff[DIR_COUNT_OFFSET]= (byte) (uchar) max_entry+1; + buff[DIR_COUNT_OFFSET]= (uchar) max_entry+1; int2store(dir, rec_offset); empty_space-= DIR_ENTRY_SIZE; } else { /* reuse old empty entry */ - byte *pos, *end, *end_data; + uchar *pos, *end, *end_data; DBUG_ASSERT(uint2korr(dir) == 0); if (uint2korr(dir)) goto err; /* Should have been empty */ @@ -4305,13 +4305,13 @@ err: uint _ma_apply_redo_purge_row_head_or_tail(MARIA_HA *info, LSN lsn, uint page_type, - const byte *header) + const uchar *header) { MARIA_SHARE *share= info->s; ulonglong page; uint record_number, empty_space; uint block_size= share->block_size; - byte *buff= info->keyread_buff; + uchar *buff= info->keyread_buff; DBUG_ENTER("_ma_apply_redo_purge_row_head_or_tail"); info->keyread_buff_used= 1; @@ -4324,7 +4324,7 @@ uint _ma_apply_redo_purge_row_head_or_tail(MARIA_HA *info, LSN lsn, buff, PAGECACHE_PLAIN_PAGE, PAGECACHE_LOCK_LEFT_UNLOCKED, 0))) DBUG_RETURN(my_errno); - DBUG_ASSERT((buff[PAGE_TYPE_OFFSET] & PAGE_TYPE_MASK) == (byte) page_type); + DBUG_ASSERT((buff[PAGE_TYPE_OFFSET] & PAGE_TYPE_MASK) == (uchar) page_type); if (lsn_korr(buff) >= lsn) { diff --git a/storage/maria/ma_blockrec.h b/storage/maria/ma_blockrec.h index 1eeb3368972..c11c341f782 100644 --- a/storage/maria/ma_blockrec.h +++ b/storage/maria/ma_blockrec.h @@ -180,9 +180,9 @@ my_bool _ma_check_if_right_bitmap_type(MARIA_HA *info, void _ma_bitmap_delete_all(MARIA_SHARE *share); uint _ma_apply_redo_insert_row_head_or_tail(MARIA_HA *info, LSN lsn, uint page_type, - const byte *header, - const byte *data, + const uchar *header, + const uchar *data, size_t data_length); uint _ma_apply_redo_purge_row_head_or_tail(MARIA_HA *info, LSN lsn, uint page_type, - const byte *header); + const uchar *header); diff --git a/storage/maria/ma_check.c b/storage/maria/ma_check.c index e1fd9c0170a..4d23b66421d 100644 --- a/storage/maria/ma_check.c +++ b/storage/maria/ma_check.c @@ -2060,7 +2060,7 @@ int maria_repair(HA_CHECK *param, register MARIA_HA *info, goto err; } - if (!(sort_param.record=(byte*) my_malloc((uint) share->base.pack_reclength, + if (!(sort_param.record=(uchar*) my_malloc((uint) share->base.pack_reclength, MYF(0))) || _ma_alloc_buffer(&sort_param.rec_buff, &sort_param.rec_buff_size, info->s->base.default_rec_buff_size)) @@ -5373,7 +5373,7 @@ static void copy_data_file_state(MARIA_STATE_INFO *to, static int _ma_safe_scan_block_record(MARIA_SORT_INFO *sort_info, - MARIA_HA *info, byte *record) + MARIA_HA *info, uchar *record) { uint record_pos= info->cur_row.nextpos; ulonglong page= sort_info->page; @@ -5385,7 +5385,7 @@ static int _ma_safe_scan_block_record(MARIA_SORT_INFO *sort_info, if (likely(record_pos < info->scan.number_of_rows)) { uint length, offset; - byte *data, *end_of_data; + uchar *data, *end_of_data; char llbuff[22]; while (!(offset= uint2korr(info->scan.dir))) diff --git a/storage/maria/ma_loghandler.c b/storage/maria/ma_loghandler.c index 95c8aacaf09..8e38e374a4d 100644 --- a/storage/maria/ma_loghandler.c +++ b/storage/maria/ma_loghandler.c @@ -632,7 +632,7 @@ typedef struct st_loghandler_file_info my_bool translog_read_file_header(LOGHANDLER_FILE_INFO *desc) { - byte page_buff[TRANSLOG_PAGE_SIZE], *ptr; + uchar page_buff[TRANSLOG_PAGE_SIZE], *ptr; DBUG_ENTER("translog_read_file_header"); if (my_pread(log_descriptor.log_file_num[0], page_buff, diff --git a/storage/maria/ma_loghandler.h b/storage/maria/ma_loghandler.h index 088e0d8ab8b..8382271a07a 100644 --- a/storage/maria/ma_loghandler.h +++ b/storage/maria/ma_loghandler.h @@ -295,7 +295,7 @@ typedef my_bool(*inwrite_rec_hook) (enum translog_record_type type, typedef uint16(*read_rec_hook) (enum translog_record_type type, uint16 read_length, uchar *read_buff, - byte *decoded_buff); + uchar *decoded_buff); /* record classes */ diff --git a/storage/maria/maria_read_log.c b/storage/maria/maria_read_log.c index 2d664e08662..dd130f287f5 100644 --- a/storage/maria/maria_read_log.c +++ b/storage/maria/maria_read_log.c @@ -284,11 +284,11 @@ end: static struct my_option my_long_options[] = { {"only-display", 'o', "display brief info about records's header", - (gptr*) &opt_only_display, (gptr*) &opt_only_display, 0, GET_BOOL, NO_ARG, + (uchar**) &opt_only_display, (uchar**) &opt_only_display, 0, GET_BOOL, NO_ARG, 0, 0, 0, 0, 0, 0}, {"display-and-apply", 'a', "like --only-display but displays more info and modifies tables", - (gptr*) &opt_display_and_apply, (gptr*) &opt_display_and_apply, 0, + (uchar**) &opt_display_and_apply, (uchar**) &opt_display_and_apply, 0, GET_BOOL, NO_ARG, 0, 0, 0, 0, 0, 0}, #ifndef DBUG_OFF {"debug", '#', "Output debug log. Often this is 'd:t:o,filename'.", @@ -641,7 +641,7 @@ prototype_exec_hook(REDO_INSERT_ROW_HEAD) ulonglong page; MARIA_HA *info; char llbuf[22]; - byte *buff= 0; + uchar *buff= 0; sid= fileid_korr(rec->header); page= page_korr(rec->header + FILEID_STORE_SIZE); @@ -678,7 +678,7 @@ prototype_exec_hook(REDO_INSERT_ROW_HEAD) differences. So we use the UNDO's LSN which is current_group_end_lsn. */ - if ((!(buff= (byte*) my_malloc(rec->record_length, MYF(MY_WME)))) || + if ((!(buff= (uchar*) my_malloc(rec->record_length, MYF(MY_WME)))) || (translog_read_record(rec->lsn, 0, rec->record_length, buff, NULL) != rec->record_length)) { @@ -707,7 +707,7 @@ prototype_exec_hook(REDO_INSERT_ROW_TAIL) ulonglong page; MARIA_HA *info; char llbuf[22]; - byte *buff= 0; + uchar *buff= 0; sid= fileid_korr(rec->header); page= page_korr(rec->header + FILEID_STORE_SIZE); @@ -744,7 +744,7 @@ prototype_exec_hook(REDO_INSERT_ROW_TAIL) differences. So we use the UNDO's LSN which is current_group_end_lsn. */ - if ((!(buff= (byte*) my_malloc(rec->record_length, MYF(MY_WME)))) || + if ((!(buff= (uchar*) my_malloc(rec->record_length, MYF(MY_WME)))) || (translog_read_record(rec->lsn, 0, rec->record_length, buff, NULL) != rec->record_length)) { -- cgit v1.2.1 From 3a1c7c914ca08b43094f3eba257798750c77e714 Mon Sep 17 00:00:00 2001 From: unknown Date: Wed, 4 Jul 2007 16:01:00 +0200 Subject: Maria: correcting assertions (should be enforced only in multi-threaded mode) so ma_test_all works again; more error detection in ma_test_all; maria_control renamed to maria_log_control (Monty's suggestion, so that a "rm maria_log*" removes all log-related files). Disabling failing wrong assertion. storage/maria/ma_blockrec.c: disabling assertion which fails because cur_block is a local variable not initialized storage/maria/ma_check.c: comment storage/maria/ma_control_file.h: control file renamed storage/maria/ma_loghandler.c: assertions needed only in multi-threaded mode (ma_test1 and ma_test2 are single-threaded, it's ok for them to use dummy_transaction_object with transactional tables: trn->rec_lsn can be set without interfering with other threads). storage/maria/ma_test_all.sh: got caught by failures in some ma_test1 runs, which I didn't see because ma_test_all returned 0 and I didn't scroll up in the window; now using "set -e" to avoid that. Also testing that we get the errors and warnings we expect. storage/maria/unittest/Makefile.am: maria_control renamed --- storage/maria/ma_blockrec.c | 2 ++ storage/maria/ma_check.c | 6 +++++- storage/maria/ma_control_file.h | 2 +- storage/maria/ma_loghandler.c | 4 ++-- storage/maria/ma_test_all.sh | 12 +++++++++--- storage/maria/unittest/Makefile.am | 2 +- 6 files changed, 20 insertions(+), 8 deletions(-) diff --git a/storage/maria/ma_blockrec.c b/storage/maria/ma_blockrec.c index 06c1df16663..3ce4c9efe42 100644 --- a/storage/maria/ma_blockrec.c +++ b/storage/maria/ma_blockrec.c @@ -1811,9 +1811,11 @@ static my_bool write_block_record(MARIA_HA *info, ulong length; ulong data_length= (tmp_data - info->rec_buff); +#ifdef MONTY_WILL_KNOW #ifdef SANITY_CHECKS if (cur_block->sub_blocks == 1) goto crashed; /* no reserved full or tails */ +#endif #endif /* diff --git a/storage/maria/ma_check.c b/storage/maria/ma_check.c index ae23e64575b..88198892985 100644 --- a/storage/maria/ma_check.c +++ b/storage/maria/ma_check.c @@ -1925,7 +1925,11 @@ int maria_chk_data_link(HA_CHECK *param, MARIA_HA *info,int extend) Recover old table by reading each record and writing all keys NOTES - Save new datafile-name in temp_filename + Save new datafile-name in temp_filename. + We overwrite the index file as we go (writekeys() for example), so if we + crash during this the table is unusable and user (or Recovery in the + future) must repeat the REPAIR/OPTIMIZE operation. We could use a + temporary index file in the future (drawback: more disk space). IMPLEMENTATION (for hard repair with block format) - Create new, unrelated MARIA_HA of the table diff --git a/storage/maria/ma_control_file.h b/storage/maria/ma_control_file.h index fa4ec442e41..d6c121b21be 100644 --- a/storage/maria/ma_control_file.h +++ b/storage/maria/ma_control_file.h @@ -18,7 +18,7 @@ First version written by Guilhem Bichot on 2006-04-27. */ -#define CONTROL_FILE_BASE_NAME "maria_control" +#define CONTROL_FILE_BASE_NAME "maria_log_control" /* Here is the interface of this module */ diff --git a/storage/maria/ma_loghandler.c b/storage/maria/ma_loghandler.c index 6195e552185..cb5e02a1cc0 100644 --- a/storage/maria/ma_loghandler.c +++ b/storage/maria/ma_loghandler.c @@ -5623,7 +5623,7 @@ static my_bool write_hook_for_redo(enum translog_record_type type non-transactional log records (REPAIR, CREATE, RENAME, DROP) should not call this hook; we trust them but verify ;) */ - DBUG_ASSERT(trn->trid != 0); + DBUG_ASSERT(!(maria_multi_threaded && (trn->trid == 0))); /* If the hook stays so simple, it would be faster to pass !trn->rec_lsn ? trn->rec_lsn : some_dummy_lsn @@ -5650,7 +5650,7 @@ static my_bool write_hook_for_undo(enum translog_record_type type struct st_translog_parts *parts __attribute__ ((unused))) { - DBUG_ASSERT(trn->trid != 0); /* see write_hook_for_redo() */ + DBUG_ASSERT(!(maria_multi_threaded && (trn->trid == 0))); trn->undo_lsn= *lsn; if (unlikely(LSN_WITH_FLAGS_TO_LSN(trn->first_undo_lsn) == 0)) trn->first_undo_lsn= diff --git a/storage/maria/ma_test_all.sh b/storage/maria/ma_test_all.sh index 5ea76a7037d..a6786315afe 100755 --- a/storage/maria/ma_test_all.sh +++ b/storage/maria/ma_test_all.sh @@ -9,6 +9,8 @@ # Remove # from following line if you need some more information #set -x -v -e +set -e # abort at first failure + valgrind="valgrind --alignment=8 --leak-check=yes" silent="-s" suffix="" @@ -196,15 +198,19 @@ run_repair_tests "-M -T" run_pack_tests "-M -T" # -# Tests that gives warnings +# Tests that gives warnings or errors # $maria_path/ma_test2$suffix $silent -L -K -W -P -S -R1 -m500 $maria_path/maria_chk$suffix -sm test2 echo "ma_test2$suffix $silent -L -K -R1 -m2000 ; Should give error 135" -$maria_path/ma_test2$suffix $silent -L -K -R1 -m2000 +$maria_path/ma_test2$suffix $silent -L -K -R1 -m2000 >ma_test2_message.txt 2>&1 && false # success is failure +cat ma_test2_message.txt +grep "Error: 135" ma_test2_message.txt > /dev/null echo "$maria_path/maria_chk$suffix -sm test2 will warn that 'Datafile is almost full'" -$maria_path/maria_chk$suffix -sm test2 +$maria_path/maria_chk$suffix -sm test2 >ma_test2_message.txt 2>&1 +cat ma_test2_message.txt +grep "warning: Datafile is almost full" ma_test2_message.txt >/dev/null $maria_path/maria_chk$suffix -ssm test2 # diff --git a/storage/maria/unittest/Makefile.am b/storage/maria/unittest/Makefile.am index 28264d5d903..b63cb60c059 100644 --- a/storage/maria/unittest/Makefile.am +++ b/storage/maria/unittest/Makefile.am @@ -84,6 +84,6 @@ ma_pagecache_consist_64kWR_t_big_CPPFLAGS = $(ma_pagecache_common_cppflags) -DPA # the generic lock manager may not be used in the end and lockman1-t crashes, # so we don't build lockman-t and lockman1-t -CLEANFILES = maria_control page_cache_test_file_1 \ +CLEANFILES = maria_log_control page_cache_test_file_1 \ maria_log.???????? -- cgit v1.2.1 From fcdc76c28952608524d6e5a388bc7b04ad8de09f Mon Sep 17 00:00:00 2001 From: unknown Date: Wed, 4 Jul 2007 22:27:58 +0200 Subject: in mysql_unlock_tables(), do thr_unlock() AFTER external_unlock(). it means, {update,restore}_status() should be called in external_lock, not in thr_unlock. Only affects storage engines that support TL_WRITE_CONCURRENT. --- mysys/thr_lock.c | 20 ++++++----------- sql/lock.cc | 4 ++-- storage/csv/ha_tina.cc | 12 +++++++++-- storage/csv/ha_tina.h | 4 ++++ storage/maria/ma_locking.c | 6 ++++++ storage/myisam/mi_locking.c | 52 +++++++++++++++++++++++++++++++-------------- 6 files changed, 65 insertions(+), 33 deletions(-) diff --git a/mysys/thr_lock.c b/mysys/thr_lock.c index 94c34eef2fa..b70be0d66db 100644 --- a/mysys/thr_lock.c +++ b/mysys/thr_lock.c @@ -24,7 +24,7 @@ Locks are prioritized according to: The current lock types are: -TL_READ # Low priority read +TL_READ # Low priority read TL_READ_WITH_SHARED_LOCKS TL_READ_HIGH_PRIORITY # High priority read TL_READ_NO_INSERT # Read without concurrent inserts @@ -57,8 +57,12 @@ check_status: In MyISAM this is a simple check if the insert can be done at the end of the datafile. update_status: - Before a write lock is released, this function is called. - In MyISAM this functions updates the count and length of the datafile + in thr_reschedule_write_lock(), when an insert delayed thread + downgrades TL_WRITE lock to TL_WRITE_DELAYED, to allow SELECT + threads to proceed. + A storage engine should also call update_status internally + in the ::external_lock(F_UNLCK) method. + In MyISAM and CSV this functions updates the length of the datafile. get_status: When one gets a lock this functions is called. In MyISAM this stores the number of rows and size of the datafile @@ -762,16 +766,6 @@ void thr_unlock(THR_LOCK_DATA *data) } else lock->write.last=data->prev; - if (lock_type >= TL_WRITE_CONCURRENT_INSERT) - { - if (lock->update_status) - (*lock->update_status)(data->status_param); - } - else - { - if (lock->restore_status) - (*lock->restore_status)(data->status_param); - } if (lock_type == TL_READ_NO_INSERT) lock->read_no_write_count--; data->type=TL_UNLOCK; /* Mark unlocked */ diff --git a/sql/lock.cc b/sql/lock.cc index edef3b3b67f..93d88320bb7 100644 --- a/sql/lock.cc +++ b/sql/lock.cc @@ -281,10 +281,10 @@ static int lock_external(THD *thd, TABLE **tables, uint count) void mysql_unlock_tables(THD *thd, MYSQL_LOCK *sql_lock) { DBUG_ENTER("mysql_unlock_tables"); - if (sql_lock->lock_count) - thr_multi_unlock(sql_lock->locks,sql_lock->lock_count); if (sql_lock->table_count) VOID(unlock_external(thd,sql_lock->table,sql_lock->table_count)); + if (sql_lock->lock_count) + thr_multi_unlock(sql_lock->locks,sql_lock->lock_count); my_free((gptr) sql_lock,MYF(0)); DBUG_VOID_RETURN; } diff --git a/storage/csv/ha_tina.cc b/storage/csv/ha_tina.cc index afe8e5f1b27..6dacc8f2663 100644 --- a/storage/csv/ha_tina.cc +++ b/storage/csv/ha_tina.cc @@ -441,7 +441,7 @@ ha_tina::ha_tina(handlerton *hton, TABLE_SHARE *table_arg) */ current_position(0), next_position(0), local_saved_data_file_length(0), file_buff(0), chain_alloced(0), chain_size(DEFAULT_CHAIN_LENGTH), - records_is_known(0) + records_is_known(0), curr_lock_type(F_UNLCK) { /* Set our original buffers from pre-allocated memory */ buffer.set((char*)byte_buffer, IO_SIZE, system_charset_info); @@ -1394,6 +1394,14 @@ int ha_tina::delete_all_rows() DBUG_RETURN(rc); } +int ha_tina::external_lock(THD *thd __attribute__((unused)), int lock_type) +{ + if (lock_type==F_UNLCK && curr_lock_type == F_WRLCK) + update_status(); + curr_lock_type= lock_type; + return 0; +} + /* Called by the database to lock the table. Keep in mind that this is an internal lock. @@ -1408,7 +1416,7 @@ THR_LOCK_DATA **ha_tina::store_lock(THD *thd, return to; } -/* +/* Create a table. You do not want to leave the table open after a call to this (the database will call ::open() if it needs to). */ diff --git a/storage/csv/ha_tina.h b/storage/csv/ha_tina.h index 0c667237c0f..e52e9cd28e5 100644 --- a/storage/csv/ha_tina.h +++ b/storage/csv/ha_tina.h @@ -81,6 +81,8 @@ class ha_tina: public handler bool records_is_known; private: + int curr_lock_type; + bool get_write_pos(off_t *end_pos, tina_set *closest_hole); int open_update_temp_file_if_needed(); int init_tina_writer(); @@ -153,6 +155,8 @@ public: bool check_if_incompatible_data(HA_CREATE_INFO *info, uint table_changes); + int external_lock(THD *thd, int lock_type); + THR_LOCK_DATA **store_lock(THD *thd, THR_LOCK_DATA **to, enum thr_lock_type lock_type); diff --git a/storage/maria/ma_locking.c b/storage/maria/ma_locking.c index abb095d47c2..4f92054dcb4 100644 --- a/storage/maria/ma_locking.c +++ b/storage/maria/ma_locking.c @@ -55,9 +55,15 @@ int maria_lock_database(MARIA_HA *info, int lock_type) case F_UNLCK: maria_ftparser_call_deinitializer(info); if (info->lock_type == F_RDLCK) + { count= --share->r_locks; + _ma_restore_status(info); + } else + { count= --share->w_locks; + _ma_update_status(info); + } --share->tot_locks; if (info->lock_type == F_WRLCK && !share->w_locks) { diff --git a/storage/myisam/mi_locking.c b/storage/myisam/mi_locking.c index e822ea9e6da..d142b5eaa53 100644 --- a/storage/myisam/mi_locking.c +++ b/storage/myisam/mi_locking.c @@ -56,9 +56,15 @@ int mi_lock_database(MI_INFO *info, int lock_type) case F_UNLCK: ftparser_call_deinitializer(info); if (info->lock_type == F_RDLCK) + { count= --share->r_locks; + mi_restore_status(info); + } else + { count= --share->w_locks; + mi_update_status(info); + } --share->tot_locks; if (info->lock_type == F_WRLCK && !share->w_locks && !share->delay_key_write && flush_key_blocks(share->key_cache, @@ -84,16 +90,16 @@ int mi_lock_database(MI_INFO *info, int lock_type) if (share->changed && !share->w_locks) { #ifdef HAVE_MMAP - if ((info->s->mmaped_length != info->s->state.state.data_file_length) && - (info->s->nonmmaped_inserts > MAX_NONMAPPED_INSERTS)) - { - if (info->s->concurrent_insert) - rw_wrlock(&info->s->mmap_lock); - mi_remap_file(info, info->s->state.state.data_file_length); - info->s->nonmmaped_inserts= 0; - if (info->s->concurrent_insert) - rw_unlock(&info->s->mmap_lock); - } + if ((info->s->mmaped_length != info->s->state.state.data_file_length) && + (info->s->nonmmaped_inserts > MAX_NONMAPPED_INSERTS)) + { + if (info->s->concurrent_insert) + rw_wrlock(&info->s->mmap_lock); + mi_remap_file(info, info->s->state.state.data_file_length); + info->s->nonmmaped_inserts= 0; + if (info->s->concurrent_insert) + rw_unlock(&info->s->mmap_lock); + } #endif share->state.process= share->last_process=share->this_process; share->state.unique= info->last_unique= info->this_unique; @@ -300,6 +306,7 @@ void mi_get_status(void* param, int concurrent_insert) void mi_update_status(void* param) { MI_INFO *info=(MI_INFO*) param; + DBUG_ENTER("mi_update_status"); /* Because someone may have closed the table we point at, we only update the state if its our own state. This isn't a problem as @@ -336,20 +343,32 @@ void mi_update_status(void* param) } info->opt_flag&= ~WRITE_CACHE_USED; } + DBUG_VOID_RETURN; } void mi_restore_status(void *param) { MI_INFO *info= (MI_INFO*) param; + DBUG_ENTER("mi_restore_status"); + DBUG_PRINT("info",("key_file: %ld data_file: %ld", + (long) info->s->state.state.key_file_length, + (long) info->s->state.state.data_file_length)); info->state= &info->s->state.state; info->append_insert_at_end= 0; + DBUG_VOID_RETURN; } void mi_copy_status(void* to,void *from) { - ((MI_INFO*) to)->state= &((MI_INFO*) from)->save_state; + MI_INFO *info= (MI_INFO*) to; + DBUG_ENTER("mi_copy_status"); + info->state= &((MI_INFO*) from)->save_state; + DBUG_PRINT("info",("key_file: %ld data_file: %ld", + (long) info->state->key_file_length, + (long) info->state->data_file_length)); + DBUG_VOID_RETURN; } @@ -377,17 +396,18 @@ void mi_copy_status(void* to,void *from) my_bool mi_check_status(void *param) { MI_INFO *info=(MI_INFO*) param; + DBUG_ENTER("mi_check_status"); + DBUG_PRINT("info",("dellink: %ld r_locks: %u w_locks: %u", + (long) info->s->state.dellink, (uint) info->s->r_locks, + (uint) info->s->w_locks)); /* The test for w_locks == 1 is here because this thread has already done an external lock (in other words: w_locks == 1 means no other threads has a write lock) */ - DBUG_PRINT("info",("dellink: %ld r_locks: %u w_locks: %u", - (long) info->s->state.dellink, (uint) info->s->r_locks, - (uint) info->s->w_locks)); - return (my_bool) !(info->s->state.dellink == HA_OFFSET_ERROR || + DBUG_RETURN((my_bool) !(info->s->state.dellink == HA_OFFSET_ERROR || (myisam_concurrent_insert == 2 && info->s->r_locks && - info->s->w_locks == 1)); + info->s->w_locks == 1))); } -- cgit v1.2.1 From 85510db991b5bbaa954bd181e1d02e45ca9f845f Mon Sep 17 00:00:00 2001 From: unknown Date: Sat, 7 Jul 2007 23:17:35 +0300 Subject: Merged from main 5.1 --- mysql-test/r/alter_table.result | 2 ++ mysql-test/r/rpl_events.result | 6 ++++-- 2 files changed, 6 insertions(+), 2 deletions(-) diff --git a/mysql-test/r/alter_table.result b/mysql-test/r/alter_table.result index a9c01f308f7..cc93eab0cd5 100644 --- a/mysql-test/r/alter_table.result +++ b/mysql-test/r/alter_table.result @@ -1030,6 +1030,7 @@ select * from t2; c NULL 1 +Two Three lock table t2 write, t3 read; alter table t2 change c vc varchar(100) default "Four", rename to t1; @@ -1046,6 +1047,7 @@ select * from t1; vc NULL 1 +Two Three Four drop tables t1, t3; diff --git a/mysql-test/r/rpl_events.result b/mysql-test/r/rpl_events.result index 18fe72a9879..2cb9a4ab62f 100644 --- a/mysql-test/r/rpl_events.result +++ b/mysql-test/r/rpl_events.result @@ -122,13 +122,15 @@ SELECT * FROM t1 ORDER BY id; id c ts 1 manually TIMESTAMP 2 from justonce TIMESTAMP -affected rows: 2 +4 from er TIMESTAMP +affected rows: 3 "Checking event data on the slave" SELECT * FROM t1 ORDER BY id; id c ts 1 manually TIMESTAMP 2 from justonce TIMESTAMP -affected rows: 2 +4 from er TIMESTAMP +affected rows: 3 "Checking event is inactive on slave" SELECT db, name, status, originator FROM mysql.event WHERE db = 'test' AND name = 'justonce'; db name status originator -- cgit v1.2.1 From 6bbca54d7dfa5cc3f5a6da46395b73f77d142527 Mon Sep 17 00:00:00 2001 From: unknown Date: Thu, 26 Jul 2007 11:56:21 +0200 Subject: WL#3072 - Maria recovery Unit test for recovery: runs ma_test1 and ma_test2 (both only with INSERTs and DELETEs; UPDATEs disabled as not handled by recovery) then moves the tables elswhere; recreates tables from the log, and compares and fails if there is a difference. Passes now. Most of maria_read_log.c moved to ma_recovery.c, as it will be re-used for recovery-from-ha_maria. Bugfixes of applying of REDO_INSERT, REDO_PURGE_ROW. Applying of REDO_PURGE_BLOCKS, REDO_DELETE_ALL, REDO_DROP_TABLE, UNDO_ROW_INSERT (in REDO phase only, i.e. just doing records++), UNDO_ROW_DELETE, UNDO_ROW_PURGE. Code cleanups. Monty: please look for "QQ". Sanja: please look for "Sanja". Future tasks: recovery of the bitmap (easy), recovery of the state (make it idempotent), more REDOs (Monty to work on REDO_UPDATE?), UNDO phase... Pushing this cset as it looks safe, contains test and bugfixes which will help Monty implement applying of REDO_UPDATE. sql/handler.cc: typo storage/maria/Makefile.am: Adding ma_test_recovery (which ma_test_all invokes, and which can also be run alone). Most of maria_read_log.c moved to ma_recovery.c storage/maria/ha_maria.cc: comments storage/maria/ma_bitmap.c: fixing comments. 2 -> sizeof(maria_bitmap_marker). Bitmap-related part of _ma_initialize_datafile() moves in bitmap module. Now putting the "bm" signature when creating the first bitmap page (it used to happen only at next open, but that caused an annoying difference when testing Recovery if the original run didn't open the table, and it looks more logical like this: it goes to disk only with its signature correct); see the "QQ" comment towards the _ma_initialize_data_file() call in ma_create.c for more). When reading a bitmap page, verify its signature (happens when normally using the table or when CHECKing it; not when REPAIRing it). storage/maria/ma_blockrec.c: * no need to sync the data file if table is not transactional * Comments, code cleanup (log-related data moved to log-related code block, int5store->page_store). * Store the table's short id into LOGREC_UNDO_ROW_PURGE, like we do for other records (though this record will soon be replaced with a CLR). * If "page" is 1 it means the page which extends from byte page*block_size+1 to (page+1)*block_size (byte number 1 being the first byte of the file). The last byte of the file is data_file_length (same convention). A new page needs to be created if the last byte of the page is beyond the last byte of the file, i.e. (page+1)*block_size+1 > data_file_length, so we correct the test (bug found when testing log applying for ma_test1 -M -T --skip-update). * update the page's LSN when removing a row from it during execution of a REDO_PURGE_ROW record (bug found when testing log applying for ma_test1 -M -T --skip-update). * applying of REDO_PURGE_BLOCKs (limited to a one-page range for now). storage/maria/ma_blockrec.h: new functions. maria_bitmap_marker does not need to be exported. storage/maria/ma_close.c: we can always flush the table's state when closing the last instance of the table. And it is needed for maria_read_log (as it does not use maria_lock_database()). storage/maria/ma_control_file.c: when in Recovery, some assertions should not be used. storage/maria/ma_control_file.h: double-inclusion safe storage/maria/ma_create.c: during recovery, don't log records. Comments. Moving the creation of the first bitmap page to ma_bitmap.c storage/maria/ma_delete_table.c: during recovery, don't log records. Log the end-zero of the dropped table's name, so that recovery can use the string in place without extending it to fit an end zero. storage/maria/ma_loghandler.c: * inwrite_rec_hook also needs access to the MARIA_SHARE, like prewrite_rec_hook. This will be needed to update share->records_diff (in the upcoming patch "recovery of the state"). * LOG_DESC::record_ends_group changed to an enum. * LOG_DESC for LOGREC_REDO_PURGE_BLOCKS and LOGREC_UNDO_ROW_PURGE corrected * Sanja please see the @todo LOG BUG * avoiding DBUG_RETURN(func()) as it gives confusing debug traces. storage/maria/ma_loghandler.h: - log write hooks called while the log's lock is held (inwrite_rec_hook) now need the MARIA_SHARE, like prewrite_rec_hook already had - instead of a bool saying if this record's type ends groups or not, we refine: it may not end a group, it may end a group, or it may be a group in itself. Imagine that we had a physical write failure to a table before we log the UNDO, we still end up in external_lock(F_UNLCK) and then we log a COMMIT: we don't want to consider this COMMIT as ending the group of REDOs (don't want to execute those REDOs during Recovery), that's why we say "COMMIT is a group in itself, it aborts any previous group". This also gives one more sanity check in maria_read_log. storage/maria/ma_recovery.c: New Recovery code, replacing the old pseudocode. Most of maria_read_log moved here. Call-able from ha_maria, but not enabled yet. Compared to the previous version of maria_read_log, some bugs have been fixed, debugging output can go to stdout or a disk file (for now it's useful for me, later it can be changed), execution of REDO_DROP_TABLE, REDO_DELETE_ALL, REDO_PURGE_BLOCKS has been added. Duplicate code has been factored into functions. We abort an unfinished group of records if we see a record which is a group in itself (like COMMIT). No need for maria_panic() after a bug (which caused tables to not be closed) was fixed; if there is yet another bug I prefer to see it. When opening a table for Recovery, set data_file_length and key_file_length to their real physical value (these are the easiest state members to restore :). Warn us if the last page was truncated (but Recovery handles it). MARIA_SHARE::state::state::records is now partly recovered (not idempotent, but works if recreating tables from scracth). When applying a REDO to a page, stamp it with the UNDO's LSN (current_group_end_lsn), not with the REDO's LSN; it makes the table more identical to the original table (easier to compare the two tables in the end). Big thing missing: some types of REDOs are not handled, and the UNDO phase does not exist (missing functions to execute UNDOs to actually rollback). So for now tests are only inserting/deleting a few 100 rows, closing the table and seeing if the log is applied ok; it works. UPDATE not handled. storage/maria/ma_recovery.h: new functions: ma_recover() for recovery from inside ha_maria; _ma_apply_log() for maria_read_log (ma_recover() calls _ma_apply_log()). Btw, we need to not use the word "recover" for REPAIR/maria_chk anymore. storage/maria/ma_rename.c: don't write log records during recovery storage/maria/ma_test2.c: - fail if maria_info() or other subtests find some wrong information - new option -g to skip updates. - init the translog before creating the table, so that log applying can work. - in "#if 0" you'll see some fixed bugs (will be removed). storage/maria/ma_test_all.sh: cleanup files. Test log applying. storage/maria/maria_read_log.c: most of the logic moves to ma_recovery.c to be shared between maria_read_log and recovery-from-inside-mysqld. See ma_recovery.c for additional changes made to the moved code. storage/maria/ma_test_recovery: unit test for Recovery. Tests insert and delete, REDO_UPDATE not yet coded. Script is called from ma_test_all. Can run standalone. --- sql/handler.cc | 2 +- storage/maria/Makefile.am | 8 +- storage/maria/ha_maria.cc | 13 + storage/maria/ma_bitmap.c | 71 ++- storage/maria/ma_blockrec.c | 175 ++++-- storage/maria/ma_blockrec.h | 5 +- storage/maria/ma_close.c | 2 +- storage/maria/ma_control_file.c | 2 + storage/maria/ma_control_file.h | 6 +- storage/maria/ma_create.c | 69 ++- storage/maria/ma_delete_table.c | 5 +- storage/maria/ma_loghandler.c | 204 ++++--- storage/maria/ma_loghandler.h | 9 +- storage/maria/ma_recovery.c | 1169 ++++++++++++++++++++++++++++++++++----- storage/maria/ma_recovery.h | 6 +- storage/maria/ma_rename.c | 4 +- storage/maria/ma_test2.c | 110 ++-- storage/maria/ma_test_all.sh | 10 + storage/maria/ma_test_recovery | 37 ++ storage/maria/maria_read_log.c | 820 +-------------------------- 20 files changed, 1580 insertions(+), 1147 deletions(-) create mode 100644 storage/maria/ma_test_recovery diff --git a/sql/handler.cc b/sql/handler.cc index f3c71887e9a..ba143fdd9af 100644 --- a/sql/handler.cc +++ b/sql/handler.cc @@ -2788,7 +2788,7 @@ int ha_change_key_cache(KEY_CACHE *old_key_cache, int ha_init_pagecache(const char *name, PAGECACHE *pagecache) { - DBUG_ENTER("ha_init_key_cache"); + DBUG_ENTER("ha_init_pagecache"); if (!pagecache->inited) { diff --git a/storage/maria/Makefile.am b/storage/maria/Makefile.am index 2d11d2f470b..6e15b1df056 100644 --- a/storage/maria/Makefile.am +++ b/storage/maria/Makefile.am @@ -30,8 +30,8 @@ DEFS = @DEFS@ # "." is needed first because tests in unittest need libmaria SUBDIRS = . unittest -EXTRA_DIST = ma_test_all.sh ma_test_all.res ma_ft_stem.c CMakeLists.txt plug.in -pkgdata_DATA = ma_test_all ma_test_all.res +EXTRA_DIST = ma_test_all.sh ma_test_all.res ma_ft_stem.c CMakeLists.txt plug.in ma_test_recovery +pkgdata_DATA = ma_test_all ma_test_all.res ma_test_recovery pkglib_LIBRARIES = libmaria.a bin_PROGRAMS = maria_chk maria_pack maria_ftdump maria_read_log maria_chk_DEPENDENCIES= $(LIBRARIES) @@ -61,7 +61,7 @@ noinst_HEADERS = maria_def.h ma_rt_index.h ma_rt_key.h ma_rt_mbr.h \ ma_ft_eval.h trnman.h lockman.h tablockman.h \ ma_control_file.h ha_maria.h ma_blockrec.h \ ma_loghandler.h ma_loghandler_lsn.h ma_pagecache.h \ - ma_commit.h + ma_recovery.h ma_commit.h ma_test1_DEPENDENCIES= $(LIBRARIES) ma_test1_LDADD= @CLIENT_EXTRA_LDFLAGS@ libmaria.a \ $(top_builddir)/storage/myisam/libmyisam.a \ @@ -120,7 +120,7 @@ libmaria_a_SOURCES = ma_init.c ma_open.c ma_extra.c ma_info.c ma_rkey.c \ ma_rt_index.c ma_rt_key.c ma_rt_mbr.c ma_rt_split.c \ ma_sp_key.c ma_control_file.c ma_loghandler.c \ ma_pagecache.c ma_pagecaches.c \ - ma_commit.c + ma_recovery.c ma_commit.c CLEANFILES = test?.MA? FT?.MA? isam.log ma_test_all ma_rt_test.MA? sp_test.MA? SUFFIXES = .sh diff --git a/storage/maria/ha_maria.cc b/storage/maria/ha_maria.cc index 232dd7e695d..7352235de35 100644 --- a/storage/maria/ha_maria.cc +++ b/storage/maria/ha_maria.cc @@ -37,6 +37,15 @@ #define trans_register_ha(A, B, C) do { /* nothing */ } while(0) #endif +/** + @todo For now there is no way for a user to set a different value of + maria_recover_options, i.e. auto-check-and-repair is always disabled. + We could enable it. As the auto-repair is initiated when opened from the + SQL layer (open_unireg_entry(), check_and_repair()), it does not happen + when Maria's Recovery internally opens the table to apply log records to + it, which is good. It would happen only after Recovery, if the table is + still corrupted. +*/ ulong maria_recover_options= HA_RECOVER_NONE; static handlerton *maria_hton; @@ -1877,6 +1886,10 @@ int ha_maria::external_lock(THD *thd, int lock_type) corresponding unlock (they just stay locked and are later dropped while locked); if a tmp table was transactional, "SELECT FROM non_tmp, tmp" would never commit as its "locked_tables" count would stay 1. + When Maria has has_transactions()==TRUE, open_temporary_table() + (sql_base.cc) will use TRANSACTIONAL_TMP_TABLE and thus the + external_lock(F_UNLCK) will happen and we can then allow the user to + create transactional temporary tables. */ if (!file->s->base.born_transactional) goto skip_transaction; diff --git a/storage/maria/ma_bitmap.c b/storage/maria/ma_bitmap.c index 3376f4abf2c..857c3ad22b7 100644 --- a/storage/maria/ma_bitmap.c +++ b/storage/maria/ma_bitmap.c @@ -130,6 +130,7 @@ #define FULL_HEAD_PAGE 4 #define FULL_TAIL_PAGE 7 +/** all bitmap pages end with this 2-byte signature */ uchar maria_bitmap_marker[2]= {(uchar) 'b',(uchar) 'm'}; static my_bool _ma_read_bitmap_page(MARIA_SHARE *share, @@ -244,7 +245,7 @@ my_bool _ma_bitmap_end(MARIA_SHARE *share) /* - Flush bitmap to disk + Send updated bitmap to the page cache SYNOPSIS _ma_flush_bitmap() @@ -286,7 +287,7 @@ my_bool _ma_flush_bitmap(MARIA_SHARE *share) share Share handler NOTES - This is called on ma_delete_all (truncate data file). + This is called on maria_delete_all_rows (truncate data file). */ void _ma_bitmap_delete_all(MARIA_SHARE *share) @@ -294,8 +295,9 @@ void _ma_bitmap_delete_all(MARIA_SHARE *share) MARIA_FILE_BITMAP *bitmap= &share->bitmap; if (bitmap->map) /* Not in create */ { - bzero(bitmap->map, share->block_size); - memcpy(bitmap->map + share->block_size - 2, maria_bitmap_marker, 2); + bzero(bitmap->map, bitmap->block_size); + memcpy(bitmap->map + bitmap->block_size - sizeof(maria_bitmap_marker), + maria_bitmap_marker, sizeof(maria_bitmap_marker)); bitmap->changed= 1; bitmap->page= 0; bitmap->used_size= bitmap->total_size; @@ -497,6 +499,10 @@ static void _ma_print_bitmap(MARIA_FILE_BITMAP *bitmap) TODO Update 'bitmap->used_size' to real size of used bitmap + NOTE + We don't always have share->bitmap.bitmap_lock here + (when called from_ma_check_bitmap_data() for example). + RETURN 0 ok 1 error (Error writing old bitmap or reading bitmap page) @@ -516,7 +522,8 @@ static my_bool _ma_read_bitmap_page(MARIA_SHARE *share, { share->state.state.data_file_length= position + bitmap->block_size; bzero(bitmap->map, bitmap->block_size); - memcpy(bitmap->map + share->block_size - 2, maria_bitmap_marker, 2); + memcpy(bitmap->map + bitmap->block_size - sizeof(maria_bitmap_marker), + maria_bitmap_marker, sizeof(maria_bitmap_marker)); bitmap->used_size= 0; #ifndef DBUG_OFF memcpy(bitmap->map + bitmap->block_size, bitmap->map, bitmap->block_size); @@ -525,11 +532,14 @@ static my_bool _ma_read_bitmap_page(MARIA_SHARE *share, } bitmap->used_size= bitmap->total_size; DBUG_ASSERT(share->pagecache->block_size == bitmap->block_size); - res= pagecache_read(share->pagecache, - (PAGECACHE_FILE*)&bitmap->file, page, 0, - (byte*) bitmap->map, - PAGECACHE_PLAIN_PAGE, - PAGECACHE_LOCK_LEFT_UNLOCKED, 0) == 0; + res= (pagecache_read(share->pagecache, + (PAGECACHE_FILE*)&bitmap->file, page, 0, + (byte*) bitmap->map, + PAGECACHE_PLAIN_PAGE, + PAGECACHE_LOCK_LEFT_UNLOCKED, 0) == NULL) | + memcmp(bitmap->map + bitmap->block_size - + sizeof(maria_bitmap_marker), + maria_bitmap_marker, sizeof(maria_bitmap_marker)); #ifndef DBUG_OFF if (!res) memcpy(bitmap->map + bitmap->block_size, bitmap->map, bitmap->block_size); @@ -1630,9 +1640,16 @@ static my_bool set_page_bits(MARIA_HA *info, MARIA_FILE_BITMAP *bitmap, bitmap->changed= 1; DBUG_EXECUTE("bitmap", _ma_print_bitmap(bitmap);); - if (fill_pattern != 3 && fill_pattern != 7 && - bitmap_page < info->s->state.first_bitmap_with_space) - info->s->state.first_bitmap_with_space= bitmap_page; + if (fill_pattern != 3 && fill_pattern != 7) + set_if_smaller(info->s->state.first_bitmap_with_space, bitmap_page); + /* + Note that if the condition above is false (page is full), and all pages of + this bitmap are now full, and that bitmap page was + first_bitmap_with_space, we don't modify first_bitmap_with_space, indeed + its value still tells us where to start our search for a bitmap with space + (which is for sure after this full one). + That does mean that first_bitmap_with_space is only a lower bound. + */ DBUG_RETURN(0); } @@ -1747,8 +1764,7 @@ my_bool _ma_reset_full_page_bits(MARIA_HA *info, MARIA_FILE_BITMAP *bitmap, tmp= (1 << bit_count) - 1; *data&= ~tmp; } - if (bitmap_page < info->s->state.first_bitmap_with_space) - info->s->state.first_bitmap_with_space= bitmap_page; + set_if_smaller(info->s->state.first_bitmap_with_space, bitmap_page); bitmap->changed= 1; DBUG_EXECUTE("bitmap", _ma_print_bitmap(bitmap);); DBUG_RETURN(0); @@ -2014,3 +2030,28 @@ my_bool _ma_check_if_right_bitmap_type(MARIA_HA *info, DBUG_ASSERT(0); return 1; } + + +/** + @brief create the first bitmap page of a freshly created data file + + @param share table's share + + @return Operation status + @retval 0 OK + @retval !=0 Error +*/ + +int _ma_bitmap_create_first(MARIA_SHARE *share) +{ + uint block_size= share->bitmap.block_size; + File file= share->bitmap.file.file; + if (my_chsize(file, block_size, 0, MYF(MY_WME)) || + my_pwrite(file, maria_bitmap_marker, sizeof(maria_bitmap_marker), + block_size - sizeof(maria_bitmap_marker), + MYF(MY_NABP | MY_WME))) + return 1; + share->state.state.data_file_length= block_size; + _ma_bitmap_delete_all(share); + return 0; +} diff --git a/storage/maria/ma_blockrec.c b/storage/maria/ma_blockrec.c index 3ce4c9efe42..a1a72325f84 100644 --- a/storage/maria/ma_blockrec.c +++ b/storage/maria/ma_blockrec.c @@ -398,7 +398,8 @@ my_bool _ma_once_end_block_record(MARIA_SHARE *share) File must be synced as it is going out of the maria_open_list and so becoming unknown to Checkpoint. */ - if (my_sync(share->bitmap.file.file, MYF(MY_WME)) || + if ((share->now_transactional && + my_sync(share->bitmap.file.file, MYF(MY_WME))) || my_close(share->bitmap.file.file, MYF(MY_WME))) res= 1; /* @@ -1455,9 +1456,6 @@ static my_bool free_full_pages(MARIA_HA *info, MARIA_ROW *row) static my_bool free_full_page_range(MARIA_HA *info, ulonglong page, uint count) { - uchar log_data[FILEID_STORE_SIZE + PAGERANGE_STORE_SIZE + - ROW_EXTENT_SIZE]; - LEX_STRING log_array[TRANSLOG_INTERNAL_PARTS + 1]; my_bool res= 0; if (pagecache_delete_pages(info->s->pagecache, &info->dfile, @@ -1467,12 +1465,16 @@ static my_bool free_full_page_range(MARIA_HA *info, ulonglong page, uint count) if (info->s->now_transactional) { LSN lsn; + /** @todo unify log_data's shape with delete_head_or_tail() */ + uchar log_data[FILEID_STORE_SIZE + PAGERANGE_STORE_SIZE + + ROW_EXTENT_SIZE]; + LEX_STRING log_array[TRANSLOG_INTERNAL_PARTS + 1]; DBUG_ASSERT(info->trn->rec_lsn); pagerange_store(log_data + FILEID_STORE_SIZE, 1); - int5store(log_data + FILEID_STORE_SIZE + PAGERANGE_STORE_SIZE, + page_store(log_data + FILEID_STORE_SIZE + PAGERANGE_STORE_SIZE, page); - int2store(log_data + FILEID_STORE_SIZE + PAGERANGE_STORE_SIZE + 5, - count); + int2store(log_data + FILEID_STORE_SIZE + PAGERANGE_STORE_SIZE + + PAGE_STORE_SIZE, count); log_array[TRANSLOG_INTERNAL_PARTS + 0].str= (char*) log_data; log_array[TRANSLOG_INTERNAL_PARTS + 0].length= sizeof(log_data); @@ -1967,8 +1969,8 @@ static my_bool write_block_record(MARIA_HA *info, ((last_head_block - head_block) - 2) * ROW_EXTENT_SIZE; } DBUG_ASSERT(uint2korr(extent_data+5) & TAIL_BIT); - int5store(extent_data, head_tail_block->page); - int2store(extent_data + 5, head_tail_block->page_count); + page_store(extent_data, head_tail_block->page); + int2store(extent_data + PAGE_STORE_SIZE, head_tail_block->page_count); } } else @@ -2225,7 +2227,11 @@ disk_err: and this hook will mark the table corrupted. Maybe hook should be stored in the pagecache's block structure, or in a hash "file->maria_ha*". - */ + + @todo RECOVERY we should distinguish below between log write error and + table write error. The former should stop Maria immediately, the latter + should mark the table corrupted. + */ /* Unpin all pinned pages to not cause problems for disk cache */ _ma_unpin_all_pages(info, 0); @@ -2340,7 +2346,7 @@ my_bool _ma_write_abort_block_record(MARIA_HA *info) { LSN lsn; LEX_STRING log_array[TRANSLOG_INTERNAL_PARTS + 1]; - uchar log_data[LSN_STORE_SIZE]; + uchar log_data[LSN_STORE_SIZE + FILEID_STORE_SIZE]; /* Write UNDO record @@ -2351,16 +2357,28 @@ my_bool _ma_write_abort_block_record(MARIA_HA *info) */ /** @todo RECOVERY BUG - We will soon change that: we will here execute the UNDO records - generated while we were trying to write the row; this will log some - CLRs which will replace this LOGREC_UNDO_PURGE. + We do need the code above (delete_head_or_tail() etc) for + non-transactional tables. + For transactional tables we can either also use it or execute the + UNDO_INSERT. If we crash before this + _ma_write_abort_block_record(), Recovery will do the work of this + function by executing UNDO_INSERT. + For transactional tables, we will remove this LOGREC_UNDO_PURGE and + replace it with a LOGREC_CLR_END: we should go back the UNDO chain + until we reach the UNDO which inserted the row into the data file, and + use its previous_undo_lsn. + Same logic for when we remove inserted keys (in case of error in + maria_write(): we come to the present function only after removing the + inserted keys... as long as we unpin the key pages only after writing + the CLR_END, this would be recovery-safe...). */ lsn_store(log_data, info->trn->undo_lsn); log_array[TRANSLOG_INTERNAL_PARTS + 0].str= (char*) log_data; log_array[TRANSLOG_INTERNAL_PARTS + 0].length= sizeof(log_data); if (translog_write_record(&lsn, LOGREC_UNDO_ROW_PURGE, - info->trn, NULL, sizeof(log_data), - TRANSLOG_INTERNAL_PARTS + 1, log_array, NULL)) + info->trn, info->s, sizeof(log_data), + TRANSLOG_INTERNAL_PARTS + 1, log_array, + log_data + LSN_STORE_SIZE)) res= 1; } _ma_unpin_all_pages(info, info->trn->undo_lsn); @@ -2390,6 +2408,7 @@ my_bool _ma_update_block_record(MARIA_HA *info, MARIA_RECORD_POS record_pos, ulonglong page; struct st_row_pos_info row_pos; MARIA_SHARE *share= info->s; + my_bool res; DBUG_ENTER("_ma_update_block_record"); DBUG_PRINT("enter", ("rowid: %lu", (long) record_pos)); @@ -2486,8 +2505,8 @@ my_bool _ma_update_block_record(MARIA_HA *info, MARIA_RECORD_POS record_pos, row_pos.dir= dir; row_pos.data= buff + uint2korr(dir); row_pos.length= head_length; - DBUG_RETURN(write_block_record(info, oldrec, record, new_row, blocks, 1, - &row_pos)); + res= write_block_record(info, oldrec, record, new_row, blocks, 1, &row_pos); + DBUG_RETURN(res); err: _ma_unpin_all_pages(info, 0); @@ -2609,7 +2628,7 @@ static my_bool delete_head_or_tail(MARIA_HA *info, res= delete_dir_entry(buff, block_size, record_number, &empty_space); if (res < 0) DBUG_RETURN(1); - if (res == 0) + if (res == 0) /* after our deletion, page is still not empty */ { uchar log_data[FILEID_STORE_SIZE + PAGE_STORE_SIZE + DIRPOS_STORE_SIZE]; LEX_STRING log_array[TRANSLOG_INTERNAL_PARTS + 1]; @@ -2638,14 +2657,13 @@ static my_bool delete_head_or_tail(MARIA_HA *info, PAGECACHE_WRITE_DELAY, &page_link.link)) DBUG_RETURN(1); } - else + else /* page is now empty */ { - uchar log_data[FILEID_STORE_SIZE + PAGERANGE_STORE_SIZE + - PAGE_STORE_SIZE + PAGERANGE_STORE_SIZE]; - LEX_STRING log_array[TRANSLOG_INTERNAL_PARTS + 1]; - if (info->s->now_transactional) { + uchar log_data[FILEID_STORE_SIZE + PAGERANGE_STORE_SIZE + + PAGE_STORE_SIZE + PAGERANGE_STORE_SIZE]; + LEX_STRING log_array[TRANSLOG_INTERNAL_PARTS + 1]; pagerange_store(log_data + FILEID_STORE_SIZE, 1); page_store(log_data+ FILEID_STORE_SIZE + PAGERANGE_STORE_SIZE, page); pagerange_store(log_data + FILEID_STORE_SIZE + PAGERANGE_STORE_SIZE + @@ -2850,7 +2868,7 @@ static void init_extent(MARIA_EXTENT_CURSOR *extent, byte *extent_info, uint page_count; extent->extent= extent_info; extent->extent_count= extents; - extent->page= uint5korr(extent_info); /* First extent */ + extent->page= page_korr(extent_info); /* First extent */ page_count= uint2korr(extent_info + ROW_EXTENT_PAGE_SIZE); extent->page_count= page_count & ~TAIL_BIT; extent->tail= page_count & TAIL_BIT; @@ -2890,7 +2908,7 @@ static byte *read_next_extent(MARIA_HA *info, MARIA_EXTENT_CURSOR *extent, if (!--extent->extent_count) goto crashed; extent->extent+= ROW_EXTENT_SIZE; - extent->page= uint5korr(extent->extent); + extent->page= page_korr(extent->extent); page_count= uint2korr(extent->extent+ROW_EXTENT_PAGE_SIZE); if (!page_count) goto crashed; @@ -4124,15 +4142,21 @@ uint _ma_apply_redo_insert_row_head_or_tail(MARIA_HA *info, LSN lsn, uint block_size= share->block_size; uint rec_offset; byte *buff= info->keyread_buff, *dir; - DBUG_ENTER("_ma_apply_redo_insert_row_head"); + DBUG_ENTER("_ma_apply_redo_insert_row_head_or_tail"); info->keyread_buff_used= 1; page= page_korr(header); rownr= dirpos_korr(header+PAGE_STORE_SIZE); - if (page * info->s->block_size > info->state->data_file_length) + if (((page + 1) * info->s->block_size) > info->state->data_file_length) { - /* New page at end of file */ + /* + New page at end of file. Note that the test above is also positive if + data_file_length is not a multiple of block_size (system crashed while + writing the last page): in this case we just extend the last page and + fill it entirely with zeroes, then the REDO will put correct data on + it. + */ DBUG_ASSERT(rownr == 0); if (rownr != 0) goto err; @@ -4142,7 +4166,7 @@ uint _ma_apply_redo_insert_row_head_or_tail(MARIA_HA *info, LSN lsn, dir= buff+ block_size - PAGE_SUFFIX_SIZE - DIR_ENTRY_SIZE; /* Update that file is extended */ - info->state->data_file_length= page * info->s->block_size; + info->state->data_file_length= (page + 1) * info->s->block_size; } else { @@ -4295,8 +4319,6 @@ err: lsn LSN to put on page page_type HEAD_PAGE or TAIL_PAGE header Header (without FILEID) - data Data to be put on page - data_length Length of data NOTES This function is very similar to delete_head_or_tail() @@ -4341,6 +4363,7 @@ uint _ma_apply_redo_purge_row_head_or_tail(MARIA_HA *info, LSN lsn, if (delete_dir_entry(buff, block_size, record_number, &empty_space) < 0) DBUG_RETURN(HA_ERR_WRONG_IN_RECORD); + lsn_store(buff, lsn); if (pagecache_write(share->pagecache, &info->dfile, page, 0, buff, PAGECACHE_PLAIN_PAGE, @@ -4355,3 +4378,91 @@ uint _ma_apply_redo_purge_row_head_or_tail(MARIA_HA *info, LSN lsn, DBUG_RETURN(0); } + + +/** + @brief Apply LOGREC_REDO_PURGE_BLOCKS + + @param info Maria handler + @param header Header (without FILEID) + + @note It marks the page free in the bitmap, and sets the directory's count + to 0. + + @return Operation status + @retval 0 OK + @retval !=0 Error +*/ + +uint _ma_apply_redo_purge_blocks(MARIA_HA *info, + LSN lsn, const byte *header) +{ + MARIA_SHARE *share= info->s; + ulonglong page; + uint page_range; + uint res; + byte *buff= info->keyread_buff; + uint block_size= share->block_size; + DBUG_ENTER("_ma_apply_redo_purge_blocks"); + + info->keyread_buff_used= 1; + page_range= pagerange_korr(header); + /* works only for a one-page range for now */ + DBUG_ASSERT(page_range == 1); // for now + header+= PAGERANGE_STORE_SIZE; + page= page_korr(header); + header+= PAGE_STORE_SIZE; + page_range= pagerange_korr(header); + DBUG_ASSERT(page_range == 1); // for now + + if (!(buff= pagecache_read(share->pagecache, + &info->dfile, + page, 0, + buff, PAGECACHE_PLAIN_PAGE, + PAGECACHE_LOCK_LEFT_UNLOCKED, 0))) + DBUG_RETURN(my_errno); + + if (lsn_korr(buff) >= lsn) + { + /* Already applied */ + goto mark_free_in_bitmap; + } + + buff[PAGE_TYPE_OFFSET]= UNALLOCATED_PAGE; + + /* + Strictly speaking, we don't need to zero the last directory entry of this + page; setting the directory's count to zero is enough (it makes the last + directory entry invisible, irrelevant). + But as the "runtime" code (delete_head_or_tail()) called + delete_dir_entry() which zeroed the entry, if we don't do it here, we get + a difference between runtime and log-applying. Irrelevant, but it's + time-consuming to differentiate irrelevant differences from relevant + ones. So we remove the difference by zeroing the entry. + */ + { + uint rownr= ((uint) ((uchar *) buff)[DIR_COUNT_OFFSET]) - 1; + byte *dir= (buff + block_size - DIR_ENTRY_SIZE * rownr - + DIR_ENTRY_SIZE - PAGE_SUFFIX_SIZE); + dir[0]= dir[1]= 0; /* Delete entry */ + } + + buff[DIR_COUNT_OFFSET]= 0; + + lsn_store(buff, lsn); + if (pagecache_write(share->pagecache, + &info->dfile, page, 0, + buff, PAGECACHE_PLAIN_PAGE, + PAGECACHE_LOCK_LEFT_UNLOCKED, + PAGECACHE_PIN_LEFT_UNPINNED, + PAGECACHE_WRITE_DELAY, 0)) + DBUG_RETURN(my_errno); + +mark_free_in_bitmap: + /** @todo leave bitmap lock to the bitmap code... */ + pthread_mutex_lock(&share->bitmap.bitmap_lock); + res= _ma_reset_full_page_bits(info, &share->bitmap, page, 1); + pthread_mutex_unlock(&share->bitmap.bitmap_lock); + + DBUG_RETURN(res); +} diff --git a/storage/maria/ma_blockrec.h b/storage/maria/ma_blockrec.h index 0ed0898859c..d32f4886682 100644 --- a/storage/maria/ma_blockrec.h +++ b/storage/maria/ma_blockrec.h @@ -105,8 +105,6 @@ enum en_page_type { UNALLOCATED_PAGE, HEAD_PAGE, TAIL_PAGE, BLOB_PAGE, MAX_PAGE_ /* Don't allocate memory for too many row extents on the stack */ #define ROW_EXTENTS_ON_STACK 32 -extern uchar maria_bitmap_marker[2]; - /* Functions to convert MARIA_RECORD_POS to/from page:offset */ static inline MARIA_RECORD_POS ma_recordpos(ulonglong page, uint dir_entry) @@ -178,6 +176,7 @@ my_bool _ma_check_if_right_bitmap_type(MARIA_HA *info, ulonglong page, uint *bitmap_pattern); void _ma_bitmap_delete_all(MARIA_SHARE *share); +int _ma_bitmap_create_first(MARIA_SHARE *share); uint _ma_apply_redo_insert_row_head_or_tail(MARIA_HA *info, LSN lsn, uint page_type, const byte *header, @@ -186,3 +185,5 @@ uint _ma_apply_redo_insert_row_head_or_tail(MARIA_HA *info, LSN lsn, uint _ma_apply_redo_purge_row_head_or_tail(MARIA_HA *info, LSN lsn, uint page_type, const byte *header); +uint _ma_apply_redo_purge_blocks(MARIA_HA *info, LSN lsn, + const byte *header); diff --git a/storage/maria/ma_close.c b/storage/maria/ma_close.c index 4fec7359d66..24887f30de4 100644 --- a/storage/maria/ma_close.c +++ b/storage/maria/ma_close.c @@ -87,7 +87,7 @@ int maria_close(register MARIA_HA *info) may be using the file at this point IF using --external-locking, which does not apply to Maria. */ - if (share->mode != O_RDONLY && maria_is_crashed(info)) + if (share->mode != O_RDONLY) _ma_state_info_write(share->kfile.file, &share->state, 1); if (my_close(share->kfile.file, MYF(0))) error= my_errno; diff --git a/storage/maria/ma_control_file.c b/storage/maria/ma_control_file.c index 66f0c37f4a3..4174a0e797e 100644 --- a/storage/maria/ma_control_file.c +++ b/storage/maria/ma_control_file.c @@ -51,6 +51,8 @@ uint32 last_logno= FILENO_IMPOSSIBLE; it is called at startup. */ my_bool maria_multi_threaded= FALSE; +/** @brief if currently doing a recovery */ +my_bool maria_in_recovery= FALSE; /* Control file is less then 512 bytes (a disk sector), diff --git a/storage/maria/ma_control_file.h b/storage/maria/ma_control_file.h index d6c121b21be..d69f221abb8 100644 --- a/storage/maria/ma_control_file.h +++ b/storage/maria/ma_control_file.h @@ -18,6 +18,9 @@ First version written by Guilhem Bichot on 2006-04-27. */ +#ifndef _ma_control_file_h +#define _ma_control_file_h + #define CONTROL_FILE_BASE_NAME "maria_log_control" /* Here is the interface of this module */ @@ -33,7 +36,7 @@ extern LSN last_checkpoint_lsn; */ extern uint32 last_logno; -extern my_bool maria_multi_threaded; +extern my_bool maria_multi_threaded, maria_in_recovery; typedef enum enum_control_file_error { CONTROL_FILE_OK= 0, @@ -74,3 +77,4 @@ int ma_control_file_end(); #ifdef __cplusplus } #endif +#endif diff --git a/storage/maria/ma_create.c b/storage/maria/ma_create.c index 2098d7119eb..50f1ed5d967 100644 --- a/storage/maria/ma_create.c +++ b/storage/maria/ma_create.c @@ -677,7 +677,7 @@ int maria_create(const char *name, enum data_file_type datafile_type, /* max_data_file_length and max_key_file_length are recalculated on open */ if (tmp_table) share.base.max_data_file_length= (my_off_t) ci->data_file_length; - else if (ci->transactional && translog_inited) + else if (ci->transactional && translog_inited && !maria_in_recovery) { /* we have checked translog_inited above, because maria_chk may call us @@ -940,23 +940,31 @@ int maria_create(const char *name, enum data_file_type datafile_type, for (i= TRANSLOG_INTERNAL_PARTS; i < (sizeof(log_array)/sizeof(log_array[0])); i++) total_rec_length+= log_array[i].length; - /* - For this record to be of any use for Recovery, we need the upper - MySQL layer to be crash-safe, which it is not now (that would require - work using the ddl_log of sql/sql_table.cc); when it is, we should - reconsider the moment of writing this log record (before or after op, - under THR_LOCK_maria or not...), how to use it in Recovery. - For now this record can serve when we apply logs to a backup, - so we sync it. This happens before the data file is created. If the data - file was created before, and we crashed before writing the log record, - at restart the table may be used, so we would not have a trustable - history in the log (impossible to apply this log to a backup). The way - we do it, if we crash before writing the log record then there is no - data file and the table cannot be used. - Note that in case of TRUNCATE TABLE we also come here. - When in CREATE/TRUNCATE (or DROP or RENAME or REPAIR) we have not called - external_lock(), so have no TRN. It does not matter, as all these - operations are non-transactional and sync their files. + /** + For this record to be of any use for Recovery, we need the upper + MySQL layer to be crash-safe, which it is not now (that would require + work using the ddl_log of sql/sql_table.cc); when it is, we should + reconsider the moment of writing this log record (before or after op, + under THR_LOCK_maria or not...), how to use it in Recovery. + For now this record can serve when we apply logs to a backup, + so we sync it. This happens before the data file is created. If the + data file was created before, and we crashed before writing the log + record, at restart the table may be used, so we would not have a + trustable history in the log (impossible to apply this log to a + backup). The way we do it, if we crash before writing the log record + then there is no data file and the table cannot be used. + @todo Note that in case of TRUNCATE TABLE we also come here; for + Recovery to be able to finish TRUNCATE TABLE, instead of leaving a + half-truncated table, we should log the record at start of + maria_create(); for that we shouldn't write to the index file but to a + buffer (DYNAMIC_STRING), put the buffer into the record, then put the + buffer into the index file (so, change _ma_keydef_write() etc). That + would also enable Recovery to finish a CREATE TABLE. The final result + would be that we would be able to finish what the SQL layer has asked + for: it would be atomic. + When in CREATE/TRUNCATE (or DROP or RENAME or REPAIR) we have not + called external_lock(), so have no TRN. It does not matter, as all + these operations are non-transactional and sync their files. */ if (unlikely(translog_write_record(&share.state.create_rename_lsn, LOGREC_REDO_CREATE_TABLE, @@ -1016,6 +1024,20 @@ int maria_create(const char *name, enum data_file_type datafile_type, goto err; errpos=3; + /* + QQ: this sets data_file_length from 0 to 8192, but we wrote the state + already to the index file (because: + - log record is built from index header so state must be written before + log record + - data file must be created after log record, so that "missing log + record" implies "unusable table"). + Thus, we below create a 8192-byte data file, but its recorded size is 0, + so next time we read the bitmap (a maria_write() for example) we'll + overwrite the bitmap we just created below. + It's not very efficient. Though there is no bug. + Why do we absolutely want to create a 8192-byte page for a freshly + created, empty table? Why don't we leave the data file empty? + */ if (_ma_initialize_data_file(&share, dfile)) goto err; } @@ -1159,11 +1181,14 @@ int _ma_initialize_data_file(MARIA_SHARE *share, File dfile) { if (share->data_file_type == BLOCK_RECORD) { - if (my_chsize(dfile, share->base.block_size, 0, MYF(MY_WME))) - return 1; - share->state.state.data_file_length= share->base.block_size; - _ma_bitmap_delete_all(share); + share->bitmap.block_size= share->base.block_size; + share->bitmap.file.file = dfile; + return _ma_bitmap_create_first(share); } + /* + So, in BLOCK_RECORD, a freshly created datafile is one page long; while in + other formats it is 0-byte long. + */ return 0; } diff --git a/storage/maria/ma_delete_table.c b/storage/maria/ma_delete_table.c index 6d6b9d032fd..693c68c7e5f 100644 --- a/storage/maria/ma_delete_table.c +++ b/storage/maria/ma_delete_table.c @@ -64,7 +64,8 @@ int maria_delete_table(const char *name) raid_type= info->s->base.raid_type; raid_chunks= info->s->base.raid_chunks; #endif - sync_dir= (info->s->now_transactional && !info->s->temporary) ? + sync_dir= (info->s->now_transactional && !info->s->temporary && + !maria_in_recovery) ? MY_SYNC_DIR : 0; maria_close(info); } @@ -85,7 +86,7 @@ int maria_delete_table(const char *name) LSN lsn; LEX_STRING log_array[TRANSLOG_INTERNAL_PARTS + 1]; log_array[TRANSLOG_INTERNAL_PARTS + 0].str= (char *)name; - log_array[TRANSLOG_INTERNAL_PARTS + 0].length= strlen(name); + log_array[TRANSLOG_INTERNAL_PARTS + 0].length= strlen(name) + 1; if (unlikely(translog_write_record(&lsn, LOGREC_REDO_DROP_TABLE, &dummy_transaction_object, NULL, log_array[TRANSLOG_INTERNAL_PARTS + diff --git a/storage/maria/ma_loghandler.c b/storage/maria/ma_loghandler.c index cb5e02a1cc0..09f91274ccb 100644 --- a/storage/maria/ma_loghandler.c +++ b/storage/maria/ma_loghandler.c @@ -181,10 +181,10 @@ static MARIA_SHARE **id_to_share= NULL; static my_atomic_rwlock_t LOCK_id_to_share; static my_bool write_hook_for_redo(enum translog_record_type type, - TRN *trn, LSN *lsn, + TRN *trn, MARIA_SHARE *share, LSN *lsn, struct st_translog_parts *parts); static my_bool write_hook_for_undo(enum translog_record_type type, - TRN *trn, LSN *lsn, + TRN *trn, MARIA_SHARE *share, LSN *lsn, struct st_translog_parts *parts); /* @@ -197,27 +197,27 @@ LOG_DESC log_record_type_descriptor[LOGREC_NUMBER_OF_TYPES]; static LOG_DESC INIT_LOGREC_FIXED_RECORD_0LSN_EXAMPLE= {LOGRECTYPE_FIXEDLENGTH, 6, 6, NULL, NULL, NULL, 0, - "fixed0example", FALSE, NULL, NULL}; + "fixed0example", LOGREC_NOT_LAST_IN_GROUP, NULL, NULL}; static LOG_DESC INIT_LOGREC_VARIABLE_RECORD_0LSN_EXAMPLE= {LOGRECTYPE_VARIABLE_LENGTH, 0, 9, NULL, NULL, NULL, 0, -"variable0example", FALSE, NULL, NULL}; +"variable0example", LOGREC_NOT_LAST_IN_GROUP, NULL, NULL}; static LOG_DESC INIT_LOGREC_FIXED_RECORD_1LSN_EXAMPLE= {LOGRECTYPE_PSEUDOFIXEDLENGTH, 7, 7, NULL, NULL, NULL, 1, -"fixed1example", FALSE, NULL, NULL}; +"fixed1example", LOGREC_NOT_LAST_IN_GROUP, NULL, NULL}; static LOG_DESC INIT_LOGREC_VARIABLE_RECORD_1LSN_EXAMPLE= {LOGRECTYPE_VARIABLE_LENGTH, 0, 12, NULL, NULL, NULL, 1, -"variable1example", FALSE, NULL, NULL}; +"variable1example", LOGREC_NOT_LAST_IN_GROUP, NULL, NULL}; static LOG_DESC INIT_LOGREC_FIXED_RECORD_2LSN_EXAMPLE= {LOGRECTYPE_PSEUDOFIXEDLENGTH, 23, 23, NULL, NULL, NULL, 2, -"fixed2example", FALSE, NULL, NULL}; +"fixed2example", LOGREC_NOT_LAST_IN_GROUP, NULL, NULL}; static LOG_DESC INIT_LOGREC_VARIABLE_RECORD_2LSN_EXAMPLE= {LOGRECTYPE_VARIABLE_LENGTH, 0, 19, NULL, NULL, NULL, 2, -"variable2example", FALSE, NULL, NULL}; +"variable2example", LOGREC_NOT_LAST_IN_GROUP, NULL, NULL}; void example_loghandler_init() @@ -239,157 +239,172 @@ void example_loghandler_init() static LOG_DESC INIT_LOGREC_RESERVED_FOR_CHUNKS23= {LOGRECTYPE_NOT_ALLOWED, 0, 0, NULL, NULL, NULL, 0, - "reserved", FALSE, NULL, NULL }; + "reserved", LOGREC_NOT_LAST_IN_GROUP, NULL, NULL }; static LOG_DESC INIT_LOGREC_REDO_INSERT_ROW_HEAD= {LOGRECTYPE_VARIABLE_LENGTH, 0, FILEID_STORE_SIZE + PAGE_STORE_SIZE + DIRPOS_STORE_SIZE, NULL, write_hook_for_redo, NULL, 0, - "redo_insert_row_head", FALSE, NULL, NULL}; + "redo_insert_row_head", LOGREC_NOT_LAST_IN_GROUP, NULL, NULL}; static LOG_DESC INIT_LOGREC_REDO_INSERT_ROW_TAIL= {LOGRECTYPE_VARIABLE_LENGTH, 0, FILEID_STORE_SIZE + PAGE_STORE_SIZE + DIRPOS_STORE_SIZE, NULL, write_hook_for_redo, NULL, 0, - "redo_insert_row_tail", FALSE, NULL, NULL}; + "redo_insert_row_tail", LOGREC_NOT_LAST_IN_GROUP, NULL, NULL}; static LOG_DESC INIT_LOGREC_REDO_INSERT_ROW_BLOB= {LOGRECTYPE_VARIABLE_LENGTH, 0, 8, NULL, write_hook_for_redo, NULL, 0, - "redo_insert_row_blob", FALSE, NULL, NULL}; + "redo_insert_row_blob", LOGREC_NOT_LAST_IN_GROUP, NULL, NULL}; /*QQQ:TODO:header???*/ static LOG_DESC INIT_LOGREC_REDO_INSERT_ROW_BLOBS= {LOGRECTYPE_VARIABLE_LENGTH, 0, FILEID_STORE_SIZE, NULL, write_hook_for_redo, NULL, 0, - "redo_insert_row_blobs", FALSE, NULL, NULL}; + "redo_insert_row_blobs", LOGREC_NOT_LAST_IN_GROUP, NULL, NULL}; static LOG_DESC INIT_LOGREC_REDO_PURGE_ROW_HEAD= {LOGRECTYPE_FIXEDLENGTH, FILEID_STORE_SIZE + PAGE_STORE_SIZE + DIRPOS_STORE_SIZE, FILEID_STORE_SIZE + PAGE_STORE_SIZE + DIRPOS_STORE_SIZE, NULL, write_hook_for_redo, NULL, 0, - "redo_purge_row_head", FALSE, NULL, NULL}; + "redo_purge_row_head", LOGREC_NOT_LAST_IN_GROUP, NULL, NULL}; static LOG_DESC INIT_LOGREC_REDO_PURGE_ROW_TAIL= {LOGRECTYPE_FIXEDLENGTH, FILEID_STORE_SIZE + PAGE_STORE_SIZE + DIRPOS_STORE_SIZE, FILEID_STORE_SIZE + PAGE_STORE_SIZE + DIRPOS_STORE_SIZE, NULL, write_hook_for_redo, NULL, 0, - "redo_purge_row_tail", FALSE, NULL, NULL}; + "redo_purge_row_tail", LOGREC_NOT_LAST_IN_GROUP, NULL, NULL}; /* QQQ: TODO: variable and fixed size??? */ static LOG_DESC INIT_LOGREC_REDO_PURGE_BLOCKS= {LOGRECTYPE_VARIABLE_LENGTH, - 0, - FILEID_STORE_SIZE + PAGERANGE_STORE_SIZE, + FILEID_STORE_SIZE + PAGERANGE_STORE_SIZE + + PAGE_STORE_SIZE + PAGERANGE_STORE_SIZE, + FILEID_STORE_SIZE + PAGERANGE_STORE_SIZE + + PAGE_STORE_SIZE + PAGERANGE_STORE_SIZE, NULL, write_hook_for_redo, NULL, 0, - "redo_purge_blocks", FALSE, NULL, NULL}; + "redo_purge_blocks", LOGREC_NOT_LAST_IN_GROUP, NULL, NULL}; static LOG_DESC INIT_LOGREC_REDO_DELETE_ROW= {LOGRECTYPE_FIXEDLENGTH, 16, 16, NULL, write_hook_for_redo, NULL, 0, - "redo_delete_row", FALSE, NULL, NULL}; + "redo_delete_row", LOGREC_NOT_LAST_IN_GROUP, NULL, NULL}; static LOG_DESC INIT_LOGREC_REDO_UPDATE_ROW_HEAD= {LOGRECTYPE_VARIABLE_LENGTH, 0, 9, NULL, write_hook_for_redo, NULL, 0, - "redo_update_row_head", FALSE, NULL, NULL}; + "redo_update_row_head", LOGREC_NOT_LAST_IN_GROUP, NULL, NULL}; static LOG_DESC INIT_LOGREC_REDO_INDEX= {LOGRECTYPE_VARIABLE_LENGTH, 0, 9, NULL, write_hook_for_redo, NULL, 0, - "redo_index", FALSE, NULL, NULL}; + "redo_index", LOGREC_NOT_LAST_IN_GROUP, NULL, NULL}; static LOG_DESC INIT_LOGREC_REDO_UNDELETE_ROW= {LOGRECTYPE_FIXEDLENGTH, 16, 16, NULL, write_hook_for_redo, NULL, 0, - "redo_undelete_row", FALSE, NULL, NULL}; + "redo_undelete_row", LOGREC_NOT_LAST_IN_GROUP, NULL, NULL}; static LOG_DESC INIT_LOGREC_CLR_END= {LOGRECTYPE_PSEUDOFIXEDLENGTH, 5, 5, NULL, write_hook_for_redo, NULL, 1, - "clr_end", TRUE, NULL, NULL}; + "clr_end", LOGREC_LAST_IN_GROUP, NULL, NULL}; static LOG_DESC INIT_LOGREC_PURGE_END= {LOGRECTYPE_PSEUDOFIXEDLENGTH, 5, 5, NULL, NULL, NULL, 1, - "purge_end", TRUE, NULL, NULL}; + "purge_end", LOGREC_LAST_IN_GROUP, NULL, NULL}; static LOG_DESC INIT_LOGREC_UNDO_ROW_INSERT= {LOGRECTYPE_FIXEDLENGTH, LSN_STORE_SIZE + FILEID_STORE_SIZE + PAGE_STORE_SIZE + DIRPOS_STORE_SIZE, LSN_STORE_SIZE + FILEID_STORE_SIZE + PAGE_STORE_SIZE + DIRPOS_STORE_SIZE, NULL, write_hook_for_undo, NULL, 0, - "undo_row_insert", TRUE, NULL, NULL}; + "undo_row_insert", LOGREC_LAST_IN_GROUP, NULL, NULL}; static LOG_DESC INIT_LOGREC_UNDO_ROW_DELETE= {LOGRECTYPE_VARIABLE_LENGTH, 0, LSN_STORE_SIZE + FILEID_STORE_SIZE + PAGE_STORE_SIZE + DIRPOS_STORE_SIZE, NULL, write_hook_for_undo, NULL, 0, - "undo_row_delete", TRUE, NULL, NULL}; + "undo_row_delete", LOGREC_LAST_IN_GROUP, NULL, NULL}; static LOG_DESC INIT_LOGREC_UNDO_ROW_UPDATE= {LOGRECTYPE_VARIABLE_LENGTH, 0, LSN_STORE_SIZE + FILEID_STORE_SIZE + PAGE_STORE_SIZE + DIRPOS_STORE_SIZE, NULL, write_hook_for_undo, NULL, 1, - "undo_row_update", TRUE, NULL, NULL}; + "undo_row_update", LOGREC_LAST_IN_GROUP, NULL, NULL}; static LOG_DESC INIT_LOGREC_UNDO_ROW_PURGE= -{LOGRECTYPE_PSEUDOFIXEDLENGTH, LSN_STORE_SIZE, LSN_STORE_SIZE, - NULL, NULL, NULL, 1, - "undo_row_purge", TRUE, NULL, NULL}; +{LOGRECTYPE_PSEUDOFIXEDLENGTH, LSN_STORE_SIZE + FILEID_STORE_SIZE, + LSN_STORE_SIZE + FILEID_STORE_SIZE, + NULL, write_hook_for_undo, NULL, 1, + "undo_row_purge", LOGREC_LAST_IN_GROUP, NULL, NULL}; static LOG_DESC INIT_LOGREC_UNDO_KEY_INSERT= {LOGRECTYPE_VARIABLE_LENGTH, 0, 10, NULL, write_hook_for_undo, NULL, 1, - "undo_key_insert", TRUE, NULL, NULL}; + "undo_key_insert", LOGREC_LAST_IN_GROUP, NULL, NULL}; static LOG_DESC INIT_LOGREC_UNDO_KEY_DELETE= {LOGRECTYPE_VARIABLE_LENGTH, 0, 15, NULL, write_hook_for_undo, NULL, 0, - "undo_key_delete", TRUE, NULL, NULL}; // QQ: why not compressed? + "undo_key_delete", LOGREC_LAST_IN_GROUP, NULL, NULL}; // QQ: why not compressed? static LOG_DESC INIT_LOGREC_PREPARE= {LOGRECTYPE_VARIABLE_LENGTH, 0, 0, NULL, NULL, NULL, 0, - "prepare", TRUE, NULL, NULL}; + "prepare", LOGREC_IS_GROUP_ITSELF, NULL, NULL}; static LOG_DESC INIT_LOGREC_PREPARE_WITH_UNDO_PURGE= {LOGRECTYPE_VARIABLE_LENGTH, 0, 5, NULL, NULL, NULL, 1, - "prepare_with_undo_purge", TRUE, NULL, NULL}; + "prepare_with_undo_purge", LOGREC_IS_GROUP_ITSELF, NULL, NULL}; static LOG_DESC INIT_LOGREC_COMMIT= -{LOGRECTYPE_FIXEDLENGTH, 0, 0, NULL, NULL, NULL, 0, - "commit", TRUE, NULL, NULL}; +{LOGRECTYPE_FIXEDLENGTH, 0, 0, NULL, + NULL, NULL, 0, "commit", LOGREC_IS_GROUP_ITSELF, NULL, + NULL}; static LOG_DESC INIT_LOGREC_COMMIT_WITH_UNDO_PURGE= {LOGRECTYPE_PSEUDOFIXEDLENGTH, 5, 5, NULL, NULL, NULL, 1, - "commit_with_undo_purge", TRUE, NULL, NULL}; + "commit_with_undo_purge", LOGREC_IS_GROUP_ITSELF, NULL, NULL}; static LOG_DESC INIT_LOGREC_CHECKPOINT= {LOGRECTYPE_VARIABLE_LENGTH, 0, 0, NULL, NULL, NULL, 0, - "checkpoint", TRUE, NULL, NULL}; + "checkpoint", LOGREC_IS_GROUP_ITSELF, NULL, NULL}; static LOG_DESC INIT_LOGREC_REDO_CREATE_TABLE= {LOGRECTYPE_VARIABLE_LENGTH, 0, 1 + 2, NULL, NULL, NULL, 0, -"redo_create_table", TRUE, NULL, NULL}; +"redo_create_table", LOGREC_IS_GROUP_ITSELF, NULL, NULL}; static LOG_DESC INIT_LOGREC_REDO_RENAME_TABLE= {LOGRECTYPE_VARIABLE_LENGTH, 0, 0, NULL, NULL, NULL, 0, - "redo_rename_table", TRUE, NULL, NULL}; + "redo_rename_table", LOGREC_IS_GROUP_ITSELF, NULL, NULL}; +/** + @todo LOG BUG + the "1" below is a hack to overcome a bug in the log handler where a 0-byte + header is considered a read failure: + translog_read_record() calls translog_init_reader_data() which calls + translog_read_record_header_scan() which calls + translog_read_record_header_from_buffer() which calls + translog_variable_length_header() which returns 0 (normal); + translog_init_reader_data() considers this 0 as a problem, + and thus translog_read_record() fails. +*/ static LOG_DESC INIT_LOGREC_REDO_DROP_TABLE= -{LOGRECTYPE_VARIABLE_LENGTH, 0, 0, NULL, NULL, NULL, 0, - "redo_drop_table", TRUE, NULL, NULL}; +{LOGRECTYPE_VARIABLE_LENGTH, 0, 1, NULL, NULL, NULL, 0, + "redo_drop_table", LOGREC_IS_GROUP_ITSELF, NULL, NULL}; static LOG_DESC INIT_LOGREC_REDO_DELETE_ALL= {LOGRECTYPE_FIXEDLENGTH, FILEID_STORE_SIZE, FILEID_STORE_SIZE, NULL, write_hook_for_redo, NULL, 0, - "redo_delete_all", TRUE, NULL, NULL}; + "redo_delete_all", LOGREC_IS_GROUP_ITSELF, NULL, NULL}; static LOG_DESC INIT_LOGREC_REDO_REPAIR_TABLE= {LOGRECTYPE_FIXEDLENGTH, FILEID_STORE_SIZE + 4, FILEID_STORE_SIZE + 4, NULL, NULL, NULL, 0, - "redo_repair_table", TRUE, NULL, NULL}; + "redo_repair_table", LOGREC_IS_GROUP_ITSELF, NULL, NULL}; static LOG_DESC INIT_LOGREC_FILE_ID= {LOGRECTYPE_VARIABLE_LENGTH, 0, 2, NULL, NULL, NULL, 0, - "file_id", TRUE, NULL, NULL}; + "file_id", LOGREC_IS_GROUP_ITSELF, NULL, NULL}; static LOG_DESC INIT_LOGREC_LONG_TRANSACTION_ID= {LOGRECTYPE_FIXEDLENGTH, 6, 6, NULL, NULL, NULL, 0, - "long_transaction_id", TRUE, NULL, NULL}; + "long_transaction_id", LOGREC_IS_GROUP_ITSELF, NULL, NULL}; const myf log_write_flags= MY_WME | MY_NABP | MY_WAIT_IF_FULL; @@ -3045,6 +3060,7 @@ static translog_size_t translog_get_current_group_size() static my_bool translog_write_variable_record_1group(LSN *lsn, enum translog_record_type type, + MARIA_SHARE *share, SHORT_TRANSACTION_ID short_trid, struct st_translog_parts *parts, struct st_translog_buffer @@ -3062,7 +3078,8 @@ translog_write_variable_record_1group(LSN *lsn, *lsn= horizon= log_descriptor.horizon; if (log_record_type_descriptor[type].inwrite_hook && - (*log_record_type_descriptor[type].inwrite_hook)(type, trn, lsn, parts)) + (*log_record_type_descriptor[type].inwrite_hook)(type, trn, share, + lsn, parts)) { translog_unlock(); DBUG_RETURN(1); @@ -3199,6 +3216,7 @@ translog_write_variable_record_1group(LSN *lsn, static my_bool translog_write_variable_record_1chunk(LSN *lsn, enum translog_record_type type, + MARIA_SHARE *share, SHORT_TRANSACTION_ID short_trid, struct st_translog_parts *parts, struct st_translog_buffer @@ -3214,7 +3232,7 @@ translog_write_variable_record_1chunk(LSN *lsn, *lsn= log_descriptor.horizon; if (log_record_type_descriptor[type].inwrite_hook && - (*log_record_type_descriptor[type].inwrite_hook)(type, trn, + (*log_record_type_descriptor[type].inwrite_hook)(type, trn, share, lsn, parts)) { translog_unlock(); @@ -3567,6 +3585,7 @@ static my_bool translog_relative_LSN_encode(struct st_translog_parts *parts, static my_bool translog_write_variable_record_mgroup(LSN *lsn, enum translog_record_type type, + MARIA_SHARE *share, SHORT_TRANSACTION_ID short_trid, struct st_translog_parts *parts, struct st_translog_buffer @@ -3909,7 +3928,7 @@ translog_write_variable_record_mgroup(LSN *lsn, first_chunk0= 0; *lsn= horizon; if (log_record_type_descriptor[type].inwrite_hook && - (*log_record_type_descriptor[type].inwrite_hook) (type, trn, + (*log_record_type_descriptor[type].inwrite_hook) (type, trn, share, lsn, parts)) goto err; } @@ -3995,6 +4014,7 @@ err: static my_bool translog_write_variable_record(LSN *lsn, enum translog_record_type type, + MARIA_SHARE *share, SHORT_TRANSACTION_ID short_trid, struct st_translog_parts *parts, TRN *trn) @@ -4007,6 +4027,7 @@ static my_bool translog_write_variable_record(LSN *lsn, /* Max number of such LSNs per record is 2 */ byte compressed_LSNs[MAX_NUMBER_OF_LSNS_PER_RECORD * COMPRESSED_LSN_MAX_STORE_SIZE]; + my_bool res; DBUG_ENTER("translog_write_variable_record"); translog_lock(); @@ -4071,9 +4092,11 @@ static my_bool translog_write_variable_record(LSN *lsn, if (page_rest >= parts->record_length + header_length1) { /* following function makes translog_unlock(); */ - DBUG_RETURN(translog_write_variable_record_1chunk(lsn, type, short_trid, - parts, buffer_to_flush, - header_length1, trn)); + res= translog_write_variable_record_1chunk(lsn, type, share, + short_trid, + parts, buffer_to_flush, + header_length1, trn); + DBUG_RETURN(res); } buffer_rest= translog_get_current_group_size(); @@ -4081,15 +4104,19 @@ static my_bool translog_write_variable_record(LSN *lsn, if (buffer_rest >= parts->record_length + header_length1 - page_rest) { /* following function makes translog_unlock(); */ - DBUG_RETURN(translog_write_variable_record_1group(lsn, type, short_trid, - parts, buffer_to_flush, - header_length1, trn)); + res= translog_write_variable_record_1group(lsn, type, share, + short_trid, + parts, buffer_to_flush, + header_length1, trn); + DBUG_RETURN(res); } /* following function makes translog_unlock(); */ - DBUG_RETURN(translog_write_variable_record_mgroup(lsn, type, short_trid, - parts, buffer_to_flush, - header_length1, - buffer_rest, trn)); + res= translog_write_variable_record_mgroup(lsn, type, share, + short_trid, + parts, buffer_to_flush, + header_length1, + buffer_rest, trn); + DBUG_RETURN(res); } @@ -4112,6 +4139,7 @@ static my_bool translog_write_variable_record(LSN *lsn, static my_bool translog_write_fixed_record(LSN *lsn, enum translog_record_type type, + MARIA_SHARE *share, SHORT_TRANSACTION_ID short_trid, struct st_translog_parts *parts, TRN *trn) @@ -4164,7 +4192,7 @@ static my_bool translog_write_fixed_record(LSN *lsn, *lsn= log_descriptor.horizon; if (log_record_type_descriptor[type].inwrite_hook && - (*log_record_type_descriptor[type].inwrite_hook) (type, trn, + (*log_record_type_descriptor[type].inwrite_hook) (type, trn, share, lsn, parts)) { rc= 1; @@ -4363,11 +4391,13 @@ my_bool translog_write_record(LSN *lsn, { switch (log_record_type_descriptor[type].class) { case LOGRECTYPE_VARIABLE_LENGTH: - rc= translog_write_variable_record(lsn, type, short_trid, &parts, trn); + rc= translog_write_variable_record(lsn, type, share, + short_trid, &parts, trn); break; case LOGRECTYPE_PSEUDOFIXEDLENGTH: case LOGRECTYPE_FIXEDLENGTH: - rc= translog_write_fixed_record(lsn, type, short_trid, &parts, trn); + rc= translog_write_fixed_record(lsn, type, share, + short_trid, &parts, trn); break; case LOGRECTYPE_NOT_ALLOWED: default: @@ -4927,6 +4957,7 @@ translog_read_record_header_from_buffer(byte *page, TRANSLOG_HEADER_BUFFER *buff, TRANSLOG_SCANNER_DATA *scanner) { + translog_size_t res; DBUG_ENTER("translog_read_record_header_from_buffer"); DBUG_ASSERT((page[page_offset] & TRANSLOG_CHUNK_TYPE) == TRANSLOG_CHUNK_LSN || @@ -4941,15 +4972,18 @@ translog_read_record_header_from_buffer(byte *page, /* Read required bytes from the header and call hook */ switch (log_record_type_descriptor[buff->type].class) { case LOGRECTYPE_VARIABLE_LENGTH: - DBUG_RETURN(translog_variable_length_header(page, page_offset, buff, - scanner)); + res= translog_variable_length_header(page, page_offset, buff, + scanner); + break; case LOGRECTYPE_PSEUDOFIXEDLENGTH: case LOGRECTYPE_FIXEDLENGTH: - DBUG_RETURN(translog_fixed_length_header(page, page_offset, buff)); + res= translog_fixed_length_header(page, page_offset, buff); + break; default: DBUG_ASSERT(0); + res= 0; } - DBUG_RETURN(0); /* purecov: deadcode */ + DBUG_RETURN(res); } @@ -4979,7 +5013,7 @@ translog_size_t translog_read_record_header(LSN lsn, TRANSLOG_HEADER_BUFFER *buff) { byte buffer[TRANSLOG_PAGE_SIZE], *page; - translog_size_t page_offset= LSN_OFFSET(lsn) % TRANSLOG_PAGE_SIZE; + translog_size_t res, page_offset= LSN_OFFSET(lsn) % TRANSLOG_PAGE_SIZE; TRANSLOG_ADDRESS addr; TRANSLOG_VALIDATOR_DATA data; DBUG_ENTER("translog_read_record_header"); @@ -4993,11 +5027,9 @@ translog_size_t translog_read_record_header(LSN lsn, data.was_recovered= 0; addr= lsn; addr-= page_offset; /* offset decreasing */ - if (!(page= translog_get_page(&data, buffer))) - DBUG_RETURN(0); - - DBUG_RETURN(translog_read_record_header_from_buffer(page, page_offset, - buff, 0)); + res= (!(page= translog_get_page(&data, buffer))) ? 0 : + translog_read_record_header_from_buffer(page, page_offset, buff, 0); + DBUG_RETURN(res); } @@ -5030,6 +5062,7 @@ translog_read_record_header_scan(TRANSLOG_SCANNER_DATA TRANSLOG_HEADER_BUFFER *buff, my_bool move_scanner) { + translog_size_t res; DBUG_ENTER("translog_read_record_header_scan"); DBUG_PRINT("enter", ("Scanner: Cur: (%lu,0x%lx) Hrz: (%lu,0x%lx) " "Lst: (%lu,0x%lx) Offset: %u(%x) fixed %d", @@ -5044,11 +5077,12 @@ translog_read_record_header_scan(TRANSLOG_SCANNER_DATA buff->groups_no= 0; buff->lsn= scanner->page_addr; buff->lsn+= scanner->page_offset; /* offset increasing */ - DBUG_RETURN(translog_read_record_header_from_buffer(scanner->page, - scanner->page_offset, - buff, - (move_scanner ? - scanner : 0))); + res= translog_read_record_header_from_buffer(scanner->page, + scanner->page_offset, + buff, + (move_scanner ? + scanner : 0)); + DBUG_RETURN(res); } @@ -5083,7 +5117,7 @@ translog_size_t translog_read_next_record_header(TRANSLOG_SCANNER_DATA TRANSLOG_HEADER_BUFFER *buff) { uint8 chunk_type; - + translog_size_t res; buff->groups_no= 0; /* to be sure that we will free it right */ DBUG_ENTER("translog_read_next_record_header"); @@ -5114,9 +5148,11 @@ translog_size_t translog_read_next_record_header(TRANSLOG_SCANNER_DATA /* Last record was read */ buff->lsn= LSN_IMPOSSIBLE; /* Return 'end of log' marker */ - DBUG_RETURN(TRANSLOG_RECORD_HEADER_MAX_SIZE + 1); + res= TRANSLOG_RECORD_HEADER_MAX_SIZE + 1; } - DBUG_RETURN(translog_read_record_header_scan(scanner, buff, 0)); + else + res= translog_read_record_header_scan(scanner, buff, 0); + DBUG_RETURN(res); } @@ -5610,7 +5646,9 @@ my_bool translog_flush(LSN lsn) static my_bool write_hook_for_redo(enum translog_record_type type __attribute__ ((unused)), - TRN *trn, LSN *lsn, + TRN *trn, MARIA_SHARE *share + __attribute__ ((unused)), + LSN *lsn, struct st_translog_parts *parts __attribute__ ((unused))) { @@ -5646,7 +5684,9 @@ static my_bool write_hook_for_redo(enum translog_record_type type static my_bool write_hook_for_undo(enum translog_record_type type __attribute__ ((unused)), - TRN *trn, LSN *lsn, + TRN *trn, MARIA_SHARE *share + __attribute__ ((unused)), + LSN *lsn, struct st_translog_parts *parts __attribute__ ((unused))) { diff --git a/storage/maria/ma_loghandler.h b/storage/maria/ma_loghandler.h index 230f999c19a..efff32c9b27 100644 --- a/storage/maria/ma_loghandler.h +++ b/storage/maria/ma_loghandler.h @@ -289,7 +289,7 @@ typedef my_bool(*prewrite_rec_hook) (enum translog_record_type type, struct st_translog_parts *parts); typedef my_bool(*inwrite_rec_hook) (enum translog_record_type type, - TRN *trn, + TRN *trn, struct st_maria_share *share, LSN *lsn, struct st_translog_parts *parts); @@ -309,6 +309,11 @@ enum record_class /* C++ can't bear that a variable's name is "class" */ #ifndef __cplusplus + +enum enum_record_in_group { + LOGREC_NOT_LAST_IN_GROUP= 0, LOGREC_LAST_IN_GROUP, LOGREC_IS_GROUP_ITSELF +}; + /* Descriptor of log record type Note: Don't reorder because of constructs later... @@ -338,7 +343,7 @@ typedef struct st_log_record_type_descriptor /* the rest is for maria_read_log & Recovery */ /** @brief for debug error messages or "maria_read_log" command-line tool */ const char *name; - my_bool record_ends_group; + enum enum_record_in_group record_in_group; /* a function to execute when we see the record during the REDO phase */ int (*record_execute_in_redo_phase)(const TRANSLOG_HEADER_BUFFER *); /* a function to execute when we see the record during the UNDO phase */ diff --git a/storage/maria/ma_recovery.c b/storage/maria/ma_recovery.c index a42fbdf0458..eb802969bce 100644 --- a/storage/maria/ma_recovery.c +++ b/storage/maria/ma_recovery.c @@ -1,4 +1,4 @@ -/* Copyright (C) 2006 MySQL AB & MySQL Finland AB & TCX DataKonsult AB +/* Copyright (C) 2006, 2007 MySQL AB 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 @@ -16,180 +16,1097 @@ /* WL#3072 Maria recovery First version written by Guilhem Bichot on 2006-04-27. - Does not compile yet. */ /* Here is the implementation of this module */ -#include "page_cache.h" -#include "least_recently_dirtied.h" -#include "transaction.h" -#include "share.h" -#include "log.h" - -typedef struct st_record_type_properties { - /* used for debug error messages or "maria_read_log" command-line tool: */ - char *name, - my_bool record_ends_group; - /* a function to execute when we see the record during the REDO phase */ - int (*record_execute_in_redo_phase)(RECORD *); /* param will be record header instead later */ - /* a function to execute when we see the record during the UNDO phase */ - int (*record_execute_in_undo_phase)(RECORD *); /* param will be record header instead later */ -} RECORD_TYPE_PROPERTIES; - -int no_op(RECORD *) {return 0}; - -RECORD_TYPE_PROPERTIES all_record_type_properties[]= +#include "maria_def.h" +#include "ma_recovery.h" +#include "ma_blockrec.h" + +struct TRN_FOR_RECOVERY { - /* listed here in the order of the "log records type" enumeration */ - {"REDO_INSERT_HEAD", FALSE, redo_insert_head_execute_in_redo_phase, no_op}, - ..., - {"UNDO_INSERT" , TRUE , undo_insert_execute_in_redo_phase, undo_insert_execute_in_undo_phase}, - {"COMMIT", , TRUE , commit_execute_in_redo_phase, no_op}, - ... + LSN group_start_lsn, undo_lsn; + TrID long_trid; }; -int redo_insert_head_execute_in_redo_phase(RECORD *record) -{ - /* write the data to the proper page */ -} +/* Variables used by all functions of this module. Ok as single-threaded */ +static struct TRN_FOR_RECOVERY *all_active_trans; +static MARIA_HA **all_tables; +static LSN current_group_end_lsn; +FILE *tracef; /**< trace file for debugging */ -int undo_insert_execute_in_redo_phase(RECORD *record) -{ - trans_table[short_trans_id].undo_lsn= record.lsn; - /* don't restore the old version of the row */ -} +#define prototype_exec_hook(R) \ +static int exec_LOGREC_ ## R(const TRANSLOG_HEADER_BUFFER *rec) +prototype_exec_hook(LONG_TRANSACTION_ID); +#ifdef MARIA_CHECKPOINT +prototype_exec_hook(CHECKPOINT); +#endif +prototype_exec_hook(REDO_CREATE_TABLE); +prototype_exec_hook(REDO_DROP_TABLE); +prototype_exec_hook(FILE_ID); +prototype_exec_hook(REDO_INSERT_ROW_HEAD); +prototype_exec_hook(REDO_INSERT_ROW_TAIL); +prototype_exec_hook(REDO_PURGE_ROW_HEAD); +prototype_exec_hook(REDO_PURGE_ROW_TAIL); +prototype_exec_hook(REDO_PURGE_BLOCKS); +prototype_exec_hook(REDO_DELETE_ALL); +prototype_exec_hook(UNDO_ROW_INSERT); +prototype_exec_hook(UNDO_ROW_DELETE); +prototype_exec_hook(UNDO_ROW_PURGE); +prototype_exec_hook(COMMIT); +static int end_of_redo_phase(); +static void display_record_position(const LOG_DESC *log_desc, + const TRANSLOG_HEADER_BUFFER *rec, + uint number); +static int display_and_apply_record(const LOG_DESC *log_desc, + const TRANSLOG_HEADER_BUFFER *rec); +static MARIA_HA *get_MARIA_HA_from_REDO_record(const + TRANSLOG_HEADER_BUFFER *rec); +static MARIA_HA *get_MARIA_HA_from_UNDO_record(const + TRANSLOG_HEADER_BUFFER *rec); +static int close_recovered_table(MARIA_HA *info); -int undo_insert_execute_in_undo_phase(RECORD *record) -{ - /* restore the old version of the row */ - trans_table[short_trans_id].undo_lsn= record.prev_undo_lsn; -} -int commit_execute_in_redo_phase(RECORD *record) +/** @brief global [out] buffer for translog_read_record(); never shrinks */ +static LEX_STRING log_record_buffer; +#define enlarge_buffer(rec) \ + if (log_record_buffer.length < rec->record_length) \ + { \ + log_record_buffer.length= rec->record_length; \ + log_record_buffer.str= my_realloc(log_record_buffer.str, \ + rec->record_length, MYF(MY_WME)); \ + } + +#define ALERT_USER() DBUG_ASSERT(0) + + +/** + @brief Recovers from the last checkpoint +*/ + +int maria_recover() { - trans_table[short_trans_id].state= COMMITTED; + my_bool res= TRUE; + LSN from_lsn; + FILE *trace_file; + DBUG_ENTER("maria_recover"); + + DBUG_ASSERT(!maria_in_recovery); + maria_in_recovery= TRUE; + + if (last_checkpoint_lsn == LSN_IMPOSSIBLE) + from_lsn= first_lsn_in_log(); + else + { + DBUG_ASSERT(0); /* not yet implemented */ + /** + @todo read the checkpoint record, fill structures + and use the minimum of checkpoint_start_lsn, rec_lsn of trns, rec_lsn + of dirty pages. + */ + //from_lsn= something; + } + /* - and that's all: the delete/update handler should not be woken up! as there - may be REDO for purge further in the log. + mysqld has not yet initialized any page cache. Let's create a dedicated + one for recovery. + */ + if ((trace_file= fopen("maria_recovery.trace", "w"))) + { + fprintf(trace_file, "TRACE of the last MARIA recovery from mysqld\n"); + res= (init_pagecache(maria_pagecache, + /** @todo what size? */ + 1024*1024, + 0, 0, + maria_block_size) == 0) || + maria_apply_log(from_lsn, TRUE, trace_file); + end_pagecache(maria_pagecache, TRUE); + if (!res) + fprintf(trace_file, "SUCCESS\n"); + fclose(trace_file); + } + /** + @todo take checkpoint if log applying did some work. + Be sure to not checkpoint if no work. */ + maria_in_recovery= FALSE; + DBUG_RETURN(res); } -#define record_ends_group(R) \ - all_record_type_properties[(R)->type].record_ends_group) -#define execute_log_record_in_redo_phase(R) \ - all_record_type_properties[(R).type].record_execute_in_redo_phase(R) +/** + @brief Displays and/or applies the log + + @param lsn LSN from which log reading/applying should start + @param apply if log records should be applied or not + @param trace_file trace file where progress/debug messages will go + + @todo This trace_file thing is primitive; soon we will make it similar to + ma_check_print_warning() etc, and a successful recovery does not need to + create a trace file. But for debugging now it is useful. + @return Operation status + @retval 0 OK + @retval !=0 Error +*/ -int recovery() +int maria_apply_log(LSN lsn, my_bool apply, FILE *trace_file) { - control_file_create_or_open(); - /* - init log handler: tell it that we are going to do large reads of the - log, sequential and backward. Log handler could decide to alloc a big - read-only IO_CACHE for this, or use its usual page cache. - */ + int error= 0; + DBUG_ENTER("maria_apply_log"); - /* read checkpoint log record from log handler */ - RECORD *checkpoint_record= log_read_record(last_checkpoint_lsn_at_start); + DBUG_ASSERT(!maria_multi_threaded); + all_active_trans= (struct TRN_FOR_RECOVERY *) + my_malloc((SHORT_TRID_MAX + 1) * sizeof(struct TRN_FOR_RECOVERY), + MYF(MY_ZEROFILL)); + all_tables= (MARIA_HA **)my_malloc((SHARE_ID_MAX + 1) * sizeof(MARIA_HA *), + MYF(MY_ZEROFILL)); + if (!all_active_trans || !all_tables) + goto err; - /* parse this record, build structs (dirty_pages, transactions table, file_map) */ - /* - read log records (note: sometimes only the header is needed, for ex during - REDO phase only the header of UNDO is needed, not the 4G blob in the - variable-length part, so I could use that; however for PREPARE (which is a - variable-length record) I'll need to read the full record in the REDO - phase): - */ + tracef= trace_file; + /* install hooks for execution */ +#define install_exec_hook(R) \ + log_record_type_descriptor[LOGREC_ ## R].record_execute_in_redo_phase= \ + exec_LOGREC_ ## R; + install_exec_hook(LONG_TRANSACTION_ID); +#ifdef MARIA_CHECKPOINT + install_exec_hook(CHECKPOINT); +#endif + install_exec_hook(REDO_CREATE_TABLE); + install_exec_hook(REDO_DROP_TABLE); + install_exec_hook(FILE_ID); + install_exec_hook(REDO_INSERT_ROW_HEAD); + install_exec_hook(REDO_INSERT_ROW_TAIL); + install_exec_hook(REDO_PURGE_ROW_HEAD); + install_exec_hook(REDO_PURGE_ROW_TAIL); + install_exec_hook(REDO_PURGE_BLOCKS); + install_exec_hook(REDO_DELETE_ALL); + install_exec_hook(UNDO_ROW_INSERT); + install_exec_hook(UNDO_ROW_DELETE); + install_exec_hook(UNDO_ROW_PURGE); + install_exec_hook(COMMIT); - /**** REDO PHASE *****/ + current_group_end_lsn= LSN_IMPOSSIBLE; - record= log_read_record(min(rec_lsn, ...)); /* later, read only header */ + TRANSLOG_HEADER_BUFFER rec; + struct st_translog_scanner_data scanner; + uint i= 1; - /* - if log handler knows the end LSN of the log, we could print here how many - MB of log we have to read (to give an idea of the time), and print - progress notes. - */ + translog_size_t len= translog_read_record_header(lsn, &rec); + + /** @todo translog_read_record_header() should be fixed for 0-byte headers */ + if (len == 0) /* means error, but apparently EOF too */ + { + fprintf(tracef, "empty log\n"); + goto end; + } - while (record != NULL) + if (translog_init_scanner(lsn, 1, &scanner)) + { + fprintf(tracef, "Scanner init failed\n"); + goto err; + } + for (;;i++) { + uint16 sid= rec.short_trid; + const LOG_DESC *log_desc= &log_record_type_descriptor[rec.type]; + display_record_position(log_desc, &rec, i); + /* A complete group is a set of log records with an "end mark" record (e.g. a set of REDOs for an operation, terminated by an UNDO for this operation); if there is no "end mark" record the group is incomplete and won't be executed. */ - if (record_ends_group(record) + if ((log_desc->record_in_group == LOGREC_IS_GROUP_ITSELF) || + (log_desc->record_in_group == LOGREC_LAST_IN_GROUP)) { - if (trans_table[record.short_trans_id].group_start_lsn != 0) + if (all_active_trans[sid].group_start_lsn != LSN_IMPOSSIBLE) { - /* - There is a complete group for this transaction, containing more than - this event. - We're going to read recently read log records: - for this log_read_record() to be efficient (not touch the disk), - log handler could cache recently read pages - (can just use an IO_CACHE of 10 MB to read the log, or the normal - log handler page cache). - Without it only OS file cache will help. - */ - record2= - log_read_record(trans_table[record.short_trans_id].group_start_lsn); - - do + if (log_desc->record_in_group == LOGREC_IS_GROUP_ITSELF) + { + /* + can happen if the transaction got a table write error, then + unlocked tables thus wrote a COMMIT record. + */ + fprintf(tracef, "\nDiscarding unfinished group before this record\n"); + ALERT_USER(); + all_active_trans[sid].group_start_lsn= LSN_IMPOSSIBLE; + } + else { - if (record2.short_trans_id == record.short_trans_id) - execute_log_record_in_redo_phase(record2); /* it's in our group */ - record2= log_read_next_record(); + /* + There is a complete group for this transaction, containing more + than this event. + */ + fprintf(tracef, " ends a group:\n"); + struct st_translog_scanner_data scanner2; + TRANSLOG_HEADER_BUFFER rec2; + len= + translog_read_record_header(all_active_trans[sid].group_start_lsn, &rec2); + if (len == (TRANSLOG_RECORD_HEADER_MAX_SIZE + 1)) + { + fprintf(tracef, "Cannot find record where it should be\n"); + goto err; + } + if (translog_init_scanner(rec2.lsn, 1, &scanner2)) + { + fprintf(tracef, "Scanner2 init failed\n"); + goto err; + } + current_group_end_lsn= rec.lsn; + do + { + if (rec2.short_trid == sid) /* it's in our group */ + { + const LOG_DESC *log_desc2= &log_record_type_descriptor[rec2.type]; + display_record_position(log_desc2, &rec2, 0); + if (apply && display_and_apply_record(log_desc2, &rec2)) + goto err; + } + len= translog_read_next_record_header(&scanner2, &rec2); + if (len == (TRANSLOG_RECORD_HEADER_MAX_SIZE + 1)) + { + fprintf(tracef, "Cannot find record where it should be\n"); + goto err; + } + } + while (rec2.lsn < rec.lsn); + translog_free_record_header(&rec2); + /* group finished */ + all_active_trans[sid].group_start_lsn= LSN_IMPOSSIBLE; + current_group_end_lsn= LSN_IMPOSSIBLE; /* for debugging */ + display_record_position(log_desc, &rec, 0); } - while (record2.lsn < record.lsn); - trans_table[record.short_trans_id].group_start_lsn= 0; /* group finished */ } - execute_log_record_in_redo_phase(record); + if (apply && display_and_apply_record(log_desc, &rec)) + goto err; } else /* record does not end group */ { /* just record the fact, can't know if can execute yet */ - if (trans_table[short_trans_id].group_start_lsn == 0) /* group not yet started */ - trans_table[short_trans_id].group_start_lsn= record.lsn; + if (all_active_trans[sid].group_start_lsn == LSN_IMPOSSIBLE) + { + /* group not yet started */ + all_active_trans[sid].group_start_lsn= rec.lsn; + } + } + len= translog_read_next_record_header(&scanner, &rec); + if (len == (TRANSLOG_RECORD_HEADER_MAX_SIZE + 1)) + { + fprintf(tracef, "EOF on the log\n"); + break; } + } + translog_free_record_header(&rec); + + /* + So we have applied all REDOs. + We may now have unfinished transactions. + I don't think it's this program's job to roll them back: + to roll back and at the same time stay idempotent, it needs to write log + records (without CLRs, 2nd rollback would hit the effects of first + rollback and fail). But this standalone tool is not allowed to write to + the server's transaction log. So we do not roll back anything. + In the real Recovery code, or the code to do "recover after online + backup", yes we will roll back. + */ + if (end_of_redo_phase()) + goto err; + + goto end; +err: + error= 1; + fprintf(tracef, "Recovery of tables with transaction logs FAILED\n"); +end: + my_free((gptr)all_tables, MYF(MY_ALLOW_ZERO_PTR)); + my_free((gptr)all_active_trans, MYF(MY_ALLOW_ZERO_PTR)); + my_free(log_record_buffer.str, MYF(MY_ALLOW_ZERO_PTR)); + log_record_buffer.str= NULL; + log_record_buffer.length= 0; + DBUG_RETURN(error); +} + + +/* very basic info about the record's header */ +static void display_record_position(const LOG_DESC *log_desc, + const TRANSLOG_HEADER_BUFFER *rec, + uint number) +{ + /* + if number==0, we're going over records which we had already seen and which + form a group, so we indent below the group's end record + */ + fprintf(tracef, "%sRec#%u LSN (%lu,0x%lx) short_trid %u %s(num_type:%u) len %lu\n", + number ? "" : " ", number, + (ulong) LSN_FILE_NO(rec->lsn), (ulong) LSN_OFFSET(rec->lsn), + rec->short_trid, log_desc->name, rec->type, + (ulong)rec->record_length); +} + +static int display_and_apply_record(const LOG_DESC *log_desc, + const TRANSLOG_HEADER_BUFFER *rec) +{ + int error; + if (log_desc->record_execute_in_redo_phase == NULL) + { + /* die on all not-yet-handled records :) */ + DBUG_ASSERT("one more hook" == "to write"); + return 1; + } + if ((error= (*log_desc->record_execute_in_redo_phase)(rec))) + fprintf(tracef, "Got error when executing record\n"); + return error; +} + + +prototype_exec_hook(LONG_TRANSACTION_ID) +{ + uint16 sid= rec->short_trid; + TrID long_trid= all_active_trans[sid].long_trid; + /* abort group of this trn (must be of before a crash) */ + LSN gslsn= all_active_trans[sid].group_start_lsn; + char llbuf[22]; + if (gslsn != LSN_IMPOSSIBLE) + { + fprintf(tracef, "Group at LSN (%lu,0x%lx) short_trid %u aborted\n", + (ulong) LSN_FILE_NO(gslsn), (ulong) LSN_OFFSET(gslsn), sid); + all_active_trans[sid].group_start_lsn= LSN_IMPOSSIBLE; + } + if (long_trid != 0) + { + LSN ulsn= all_active_trans[sid].undo_lsn; + if (ulsn != LSN_IMPOSSIBLE) + { + llstr(long_trid, llbuf); + fprintf(tracef, "Found an old transaction long_trid %s short_trid %u" + " with same short id as this new transaction, and has neither" + " committed nor rollback (undo_lsn: (%lu,0x%lx))\n", llbuf, + sid, (ulong) LSN_FILE_NO(ulsn), (ulong) LSN_OFFSET(ulsn)); + goto err; + } + } + long_trid= uint6korr(rec->header); + all_active_trans[sid].long_trid= long_trid; + llstr(long_trid, llbuf); + fprintf(tracef, "Transaction long_trid %s short_trid %u starts\n", llbuf, sid); + goto end; +err: + ALERT_USER(); + return 1; +end: + return 0; +} + + +#ifdef MARIA_CHECKPOINT +prototype_exec_hook(CHECKPOINT) +{ + /* the only checkpoint we care about was found via control file, ignore */ + return 0; +} +#endif + + +prototype_exec_hook(REDO_CREATE_TABLE) +{ + File dfile= -1, kfile= -1; + char *linkname_ptr, filename[FN_REFLEN]; + char *name, *ptr; + myf create_flag; + uint flags; + int error= 1, create_mode= O_RDWR | O_TRUNC; + MARIA_HA *info= NULL; + enlarge_buffer(rec); + if (log_record_buffer.str == NULL || + translog_read_record(rec->lsn, 0, rec->record_length, + log_record_buffer.str, NULL) != + rec->record_length) + { + fprintf(tracef, "Failed to read record\n"); + goto end; + } + name= log_record_buffer.str; + fprintf(tracef, "Table '%s'", name); + /* we try hard to get create_rename_lsn, to avoid mistakes if possible */ + info= maria_open(name, O_RDONLY, HA_OPEN_FOR_REPAIR); + if (info) + { + MARIA_SHARE *share= info->s; + /* check that we're not already using it */ + DBUG_ASSERT(share->reopen == 1); + DBUG_ASSERT(share->now_transactional == share->base.born_transactional); + if (!share->base.born_transactional) + { + /* + could be that transactional table was later dropped, and a non-trans + one was renamed to its name, thus create_rename_lsn is 0 and should + not be trusted. + */ + fprintf(tracef, ", is not transactional\n"); + ALERT_USER(); + error= 0; + goto end; + } + if (cmp_translog_addr(share->state.create_rename_lsn, rec->lsn) >= 0) + { + fprintf(tracef, ", has create_rename_lsn (%lu,0x%lx) more recent than record", + (ulong) LSN_FILE_NO(rec->lsn), + (ulong) LSN_OFFSET(rec->lsn)); + error= 0; + goto end; + } + if (maria_is_crashed(info)) + { + fprintf(tracef, ", is crashed, overwriting it"); + ALERT_USER(); + } + maria_close(info); + info= NULL; + } + /* if does not exist, is older, or its header is corrupted, overwrite it */ + // TODO symlinks + ptr= name + strlen(name) + 1; + if ((flags= ptr[0] ? HA_DONT_TOUCH_DATA : 0)) + fprintf(tracef, ", we will only touch index file"); + fn_format(filename, name, "", MARIA_NAME_IEXT, + (MY_UNPACK_FILENAME | + (flags & HA_DONT_TOUCH_DATA) ? MY_RETURN_REAL_PATH : 0) | + MY_APPEND_EXT); + linkname_ptr= NULL; + create_flag= MY_DELETE_OLD; + fprintf(tracef, ", creating as '%s'", filename); + if ((kfile= my_create_with_symlink(linkname_ptr, filename, 0, create_mode, + MYF(MY_WME|create_flag))) < 0) + { + fprintf(tracef, "Failed to create index file\n"); + goto end; + } + ptr++; + uint kfile_size_before_extension= uint2korr(ptr); + ptr+= 2; + uint keystart= uint2korr(ptr); + ptr+= 2; + /* set create_rename_lsn (for maria_read_log to be idempotent) */ + lsn_store(ptr + sizeof(info->s->state.header) + 2, rec->lsn); + /* we also set is_of_lsn, like maria_create() does */ + lsn_store(ptr + sizeof(info->s->state.header) + 2 + LSN_STORE_SIZE, + rec->lsn); + if (my_pwrite(kfile, ptr, + kfile_size_before_extension, 0, MYF(MY_NABP|MY_WME)) || + my_chsize(kfile, keystart, 0, MYF(MY_WME))) + { + fprintf(tracef, "Failed to write to index file\n"); + goto end; + } + if (!(flags & HA_DONT_TOUCH_DATA)) + { + fn_format(filename,name,"", MARIA_NAME_DEXT, + MY_UNPACK_FILENAME | MY_APPEND_EXT); + linkname_ptr= NULL; + create_flag=MY_DELETE_OLD; + if (((dfile= + my_create_with_symlink(linkname_ptr, filename, 0, create_mode, + MYF(MY_WME | create_flag))) < 0) || + my_close(dfile, MYF(MY_WME))) + { + fprintf(tracef, "Failed to create data file\n"); + goto end; + } /* - Later we can optimize: instead of "execute_log_record(record2)", do - copy_record_into_exec_buffer(record2): - this will just copy record into a multi-record (10 MB?) memory buffer, - and when buffer is full, will do sorting of REDOs per - page id and execute them. - This sorting will enable us to do more sequential reads of the - data/index pages. - Note that updating bitmap pages (when we have executed a REDO for a page - we update its bitmap page) may break the sequential read of pages, - so maybe we should read and cache bitmap pages in the beginning. - Or ok the sequence will be broken, but quickly all bitmap pages will be - in memory and so the sequence will not be broken anymore. - Sorting could even determine, based on physical device of files - ("st_dev" in stat()), that some files should be should be taken by - different threads, if we want to do parallism. + we now have an empty data file. To be able to + _ma_initialize_data_file() we need some pieces of the share to be + correctly filled. So we just open the table (fortunately, an empty + data file does not preclude this). */ + if (((info= maria_open(name, O_RDONLY, 0)) == NULL) || + _ma_initialize_data_file(info->s, info->dfile.file)) + { + fprintf(tracef, "Failed to open new table or write to data file\n"); + goto end; + } + } + error= 0; +end: + fprintf(tracef, "\n"); + if (kfile >= 0) + error|= my_close(kfile, MYF(MY_WME)); + if (info != NULL) + error|= maria_close(info); + return error; +} + + +prototype_exec_hook(REDO_DROP_TABLE) +{ + char *name; + int error= 1; + MARIA_HA *info= NULL; + enlarge_buffer(rec); + if (log_record_buffer.str == NULL || + translog_read_record(rec->lsn, 0, rec->record_length, + log_record_buffer.str, NULL) != + rec->record_length) + { + fprintf(tracef, "Failed to read record\n"); + goto end; + } + name= log_record_buffer.str; + fprintf(tracef, "Table '%s'", name); + info= maria_open(name, O_RDONLY, HA_OPEN_FOR_REPAIR); + if (info) + { + MARIA_SHARE *share= info->s; + /* + We may have open instances on this table. But it does not matter, the + maria_extra() below will take care of them. + */ + if (!share->base.born_transactional) + { + fprintf(tracef, ", is not transactional\n"); + ALERT_USER(); + error= 0; + goto end; + } + if (cmp_translog_addr(share->state.create_rename_lsn, rec->lsn) >= 0) + { + fprintf(tracef, ", has create_rename_lsn (%lu,0x%lx) more recent than record", + (ulong) LSN_FILE_NO(rec->lsn), + (ulong) LSN_OFFSET(rec->lsn)); + error= 0; + goto end; + } + if (maria_is_crashed(info)) + { + fprintf(tracef, ", is crashed, dropping it"); + ALERT_USER(); + } /* - Here's how to read a complete variable-length record if needed: - read the header, allocate buffer of record length, read whole - record. + This maria_extra() call serves to signal that old open instances of + this table should not be used anymore, and (only on Windows) to close + open files so they can be deleted */ - record= log_read_next_record(); + if (maria_extra(info, HA_EXTRA_PREPARE_FOR_DELETE, NULL) || + maria_close(info)) + goto end; + info= NULL; } + /* if does not exist, is older, or its header is corrupted, drop it */ + fprintf(tracef, ", dropping '%s'", name); + if (maria_delete_table(name)) + { + fprintf(tracef, "Failed to drop table\n"); + goto end; + } + error= 0; +end: + fprintf(tracef, "\n"); + if (info != NULL) + error|= maria_close(info); + return error; +} + +prototype_exec_hook(FILE_ID) +{ + uint16 sid; + int error= 1; + char *name, *buff; + MARIA_HA *info= NULL; + MARIA_SHARE *share; + enlarge_buffer(rec); + if (log_record_buffer.str == NULL || + translog_read_record(rec->lsn, 0, rec->record_length, + log_record_buffer.str, NULL) != + rec->record_length) + { + fprintf(tracef, "Failed to read record\n"); + goto end; + } + buff= log_record_buffer.str; + sid= fileid_korr(buff); + name= buff + FILEID_STORE_SIZE; + info= all_tables[sid]; + if (info != NULL) + { + all_tables[sid]= NULL; + if (close_recovered_table(info)) + { + fprintf(tracef, "Failed to close table\n"); + goto end; + } + } + fprintf(tracef, "Table '%s', id %u", name, sid); + info= maria_open(name, O_RDWR, HA_OPEN_FOR_REPAIR); + if (info == NULL) + { + fprintf(tracef, ", is absent (must have been dropped later?)" + " or its header is so corrupted that we cannot open it;" + " we skip it\n"); + error= 0; + goto end; + } + if (maria_is_crashed(info)) + { + fprintf(tracef, "Table is crashed, can't apply log records to it\n"); + goto end; + /* + we should make an exception for REDO_REPAIR_TABLE records: if we want to + execute them, we should not reject the crashed table here. + */ + } + share= info->s; + /* check that we're not already using it */ + DBUG_ASSERT(share->reopen == 1); + DBUG_ASSERT(share->now_transactional == share->base.born_transactional); + if (!share->base.born_transactional) + { + fprintf(tracef, ", is not transactional\n"); + ALERT_USER(); + error= 0; + goto end; + } + all_tables[sid]= info; + /* don't log any records for this work */ + _ma_tmp_disable_logging_for_table(share); + /* execution of some REDO records relies on data_file_length */ + my_off_t dfile_len= my_seek(info->dfile.file, 0, SEEK_END, MYF(MY_WME)); + my_off_t kfile_len= my_seek(info->s->kfile.file, 0, SEEK_END, MYF(MY_WME)); + if ((dfile_len == MY_FILEPOS_ERROR) || + (kfile_len == MY_FILEPOS_ERROR)) + { + fprintf(tracef, ", length unknown\n"); + goto end; + } + share->state.state.data_file_length= dfile_len; + share->state.state.key_file_length= kfile_len; + if ((dfile_len == 0) || ((dfile_len % share->block_size) > 0)) + { + fprintf(tracef, ", has too short last page\n"); + /* Recovery will fix this, no error */ + ALERT_USER(); + } + fprintf(tracef, ", opened\n"); + error= 0; +end: + if (error && info != NULL) + error|= maria_close(info); + return error; +} + + +prototype_exec_hook(REDO_INSERT_ROW_HEAD) +{ + int error= 1; + byte *buff= NULL; + MARIA_HA *info= get_MARIA_HA_from_REDO_record(rec); + if (info == NULL) + goto end; + /* + If REDO's LSN is > page's LSN (read from disk), we are going to modify the + page and change its LSN. The normal runtime code stores the UNDO's LSN + into the page. Here storing the REDO's LSN (rec->lsn) would work + (we are not writing to the log here, so don't have to "flush up to UNDO's + LSN"). But in a test scenario where we do updates at runtime, then remove + tables, apply the log and check that this results in the same table as at + runtime, putting the same LSN as runtime had done will decrease + differences. So we use the UNDO's LSN which is current_group_end_lsn. + */ + enlarge_buffer(rec); + if (log_record_buffer.str == NULL || + translog_read_record(rec->lsn, 0, rec->record_length, + log_record_buffer.str, NULL) != + rec->record_length) + { + fprintf(tracef, "Failed to read record\n"); + goto end; + } + buff= log_record_buffer.str; + if (_ma_apply_redo_insert_row_head_or_tail(info, current_group_end_lsn, + HEAD_PAGE, + buff + FILEID_STORE_SIZE, + buff + + FILEID_STORE_SIZE + + PAGE_STORE_SIZE + + DIRPOS_STORE_SIZE, + rec->record_length - + (FILEID_STORE_SIZE + + PAGE_STORE_SIZE + + DIRPOS_STORE_SIZE))) + goto end; + error= 0; +end: + return error; +} + + +prototype_exec_hook(REDO_INSERT_ROW_TAIL) +{ + int error= 1; + byte *buff= NULL; + MARIA_HA *info= get_MARIA_HA_from_REDO_record(rec); + if (info == NULL) + goto end; + enlarge_buffer(rec); + if (log_record_buffer.str == NULL || + translog_read_record(rec->lsn, 0, rec->record_length, + log_record_buffer.str, NULL) != + rec->record_length) + { + fprintf(tracef, "Failed to read record\n"); + goto end; + } + buff= log_record_buffer.str; + if (_ma_apply_redo_insert_row_head_or_tail(info, current_group_end_lsn, + TAIL_PAGE, + buff + FILEID_STORE_SIZE, + buff + + FILEID_STORE_SIZE + + PAGE_STORE_SIZE + + DIRPOS_STORE_SIZE, + rec->record_length - + (FILEID_STORE_SIZE + + PAGE_STORE_SIZE + + DIRPOS_STORE_SIZE))) + goto end; + error= 0; + +end: + return error; +} + + +prototype_exec_hook(REDO_PURGE_ROW_HEAD) +{ + int error= 1; + MARIA_HA *info= get_MARIA_HA_from_REDO_record(rec); + if (info == NULL) + goto end; + if (_ma_apply_redo_purge_row_head_or_tail(info, current_group_end_lsn, + HEAD_PAGE, + rec->header + FILEID_STORE_SIZE)) + goto end; + error= 0; +end: + return error; +} + + +prototype_exec_hook(REDO_PURGE_ROW_TAIL) +{ + int error= 1; + MARIA_HA *info= get_MARIA_HA_from_REDO_record(rec); + if (info == NULL) + goto end; + if (_ma_apply_redo_purge_row_head_or_tail(info, current_group_end_lsn, + TAIL_PAGE, + rec->header + FILEID_STORE_SIZE)) + goto end; + error= 0; +end: + return error; +} + + +prototype_exec_hook(REDO_PURGE_BLOCKS) +{ + int error= 1; + MARIA_HA *info= get_MARIA_HA_from_REDO_record(rec); + if (info == NULL) + goto end; + if (_ma_apply_redo_purge_blocks(info, current_group_end_lsn, + rec->header + FILEID_STORE_SIZE)) + goto end; + error= 0; +end: + return error; +} + + +prototype_exec_hook(REDO_DELETE_ALL) +{ + int error= 1; + MARIA_HA *info= get_MARIA_HA_from_REDO_record(rec); + if (info == NULL) + goto end; + fprintf(tracef, " deleting all %lu rows\n", + (ulong)info->s->state.state.records); + if (maria_delete_all_rows(info)) + goto end; + error= 0; +end: + return error; +} + + +prototype_exec_hook(UNDO_ROW_INSERT) +{ + int error= 1; + MARIA_HA *info= get_MARIA_HA_from_UNDO_record(rec); + if (info == NULL) + goto end; + all_active_trans[rec->short_trid].undo_lsn= rec->lsn; + /* + todo: instead of above, call write_hook_for_undo, it will also set + first_undo_lsn + */ + /* + in an upcoming patch ("recovery of the state"), we introduce + state.is_of_lsn. For now, we just assume the state is old (true when we + recreate tables from scratch - but not idempotent). + */ + { + fprintf(tracef, " state older than record, updating rows' count\n"); + info->s->state.state.records++; + } + fprintf(tracef, " rows' count %lu\n", (ulong)info->s->state.state.records); + error= 0; +end: + return error; +} + + +prototype_exec_hook(UNDO_ROW_DELETE) +{ + int error= 1; + MARIA_HA *info= get_MARIA_HA_from_UNDO_record(rec); + if (info == NULL) + goto end; + all_active_trans[rec->short_trid].undo_lsn= rec->lsn; + /* + todo: instead of above, call write_hook_for_undo, it will also set + first_undo_lsn + */ + { + fprintf(tracef, " state older than record, updating rows' count\n"); + info->s->state.state.records--; + } + fprintf(tracef, " rows' count %lu\n", (ulong)info->s->state.state.records); + error= 0; +end: + return error; +} + + +prototype_exec_hook(UNDO_ROW_PURGE) +{ + int error= 1; + MARIA_HA *info= get_MARIA_HA_from_UNDO_record(rec); + if (info == NULL) + goto end; + /* this a bit broken, but this log record type will be deleted soon */ + all_active_trans[rec->short_trid].undo_lsn= rec->lsn; + /* + todo: instead of above, call write_hook_for_undo, it will also set + first_undo_lsn + */ + { + fprintf(tracef, " state older than record, updating rows' count\n"); + info->s->state.state.records--; + } + fprintf(tracef, " rows' count %lu\n", (ulong)info->s->state.state.records); + error= 0; +end: + return error; +} + + +prototype_exec_hook(COMMIT) +{ + uint16 sid= rec->short_trid; + TrID long_trid= all_active_trans[sid].long_trid; + LSN gslsn= all_active_trans[sid].group_start_lsn; + char llbuf[22]; + if (long_trid == 0) + { + fprintf(tracef, "We don't know about transaction with short_trid %u;" + "it probably committed long ago, forget it\n", sid); + return 0; + } + llstr(long_trid, llbuf); + fprintf(tracef, "Transaction long_trid %s short_trid %u committed", llbuf, sid); + if (gslsn != LSN_IMPOSSIBLE) + { + /* + It's not an error, it may be that trn got a disk error when writing to a + table, so an unfinished group staid in the log. + */ + fprintf(tracef, ", with group at LSN (%lu,0x%lx) short_trid %u aborted\n", + (ulong) LSN_FILE_NO(gslsn), (ulong) LSN_OFFSET(gslsn), sid); + all_active_trans[sid].group_start_lsn= LSN_IMPOSSIBLE; + } + else + fprintf(tracef, "\n"); + bzero(&all_active_trans[sid], sizeof(all_active_trans[sid])); +#ifdef MARIA_VERSIONING /* - Earlier or here, create true transactions in TM. - If done earlier, note that TM should not wake up the delete/update handler - when it receives a commit info, as existing REDO for purge may exist in - the log, and so the delete/update handler may do changes which conflict - with these REDOs. - Even if done here, better to not wake it up now as we're going to free the - page cache. + if real recovery: + transaction was committed, move it to some separate list for later + purging (but don't purge now! purging may have been started before, we + may find REDO_PURGE records soon). + */ +#endif + return 0; +} + + +/* Just to inform about any aborted groups or unfinished transactions */ +static int end_of_redo_phase() +{ + uint sid, unfinished= 0, error= 0; + for (sid= 0; sid <= SHORT_TRID_MAX; sid++) + { + TrID long_trid= all_active_trans[sid].long_trid; + LSN gslsn= all_active_trans[sid].group_start_lsn; + if (all_active_trans[sid].undo_lsn != LSN_IMPOSSIBLE) + { + char llbuf[22]; + llstr(long_trid, llbuf); + fprintf(tracef, "Transaction long_trid %s short_trid %u unfinished\n", + llbuf, sid); + unfinished++; + } + if (gslsn != LSN_IMPOSSIBLE) + { + fprintf(tracef, "Group at LSN (%lu,0x%lx) short_trid %u aborted\n", + (ulong) LSN_FILE_NO(gslsn), (ulong) LSN_OFFSET(gslsn), sid); + ALERT_USER(); + } + /* If real recovery: roll back unfinished transaction */ +#ifdef MARIA_VERSIONING + /* + If real recovery: transaction was committed, move it to some separate + list for soon purging. Create TRNs. + */ +#endif + } + /* + We don't close tables if there are some unfinished transactions, because + closing tables normally requires that all unfinished transactions on them + be rolled back. Unfinished transactions are symptom of a crash, we + reproduce the crash. + For example, closing will soon write the state to disk and when doing that + it will think this is a committed state, but it may not be. + */ + if (unfinished > 0) + fprintf(tracef, "WARNING: %u unfinished transactions; some tables may be" + " left inconsistent!\n", unfinished); + for (sid= 0; sid <= SHARE_ID_MAX; sid++) + { + MARIA_HA *info= all_tables[sid]; + if (info != NULL) + { + /* if error, still close other tables */ + error|= close_recovered_table(info); + } + } + return error; +} + +static int close_recovered_table(MARIA_HA *info) +{ + int error; + MARIA_SHARE *share= info->s; + fprintf(tracef, " Closing table '%s'\n", share->open_file_name); + _ma_reenable_logging_for_table(share); + /* + Recovery normally corrected problems, don't scare user with "table was not + closed properly" in CHECK TABLE and don't automatically check table at + next open (when we have --maria-recover). + */ + share->state.open_count= share->global_changed ? 1 : 0; + /* this var is set only by non-recovery operations (mi_write() etc) */ + DBUG_ASSERT(!share->global_changed); + if ((error= maria_close(info))) + fprintf(tracef, "Failed to close table\n"); + return error; +} + + +static MARIA_HA *get_MARIA_HA_from_REDO_record(const + TRANSLOG_HEADER_BUFFER *rec) +{ + uint16 sid; + ulonglong page; + MARIA_HA *info; + char llbuf[22]; + + sid= fileid_korr(rec->header); + page= page_korr(rec->header + FILEID_STORE_SIZE); + /* BUG not correct for REDO_PURGE_BLOCKS, page is not at this pos */ + llstr(page, llbuf); + fprintf(tracef, " For page %s of table of short id %u", llbuf, sid); + info= all_tables[sid]; + if (info == NULL) + { + fprintf(tracef, ", table skipped, so skipping record\n"); + return NULL; + } + fprintf(tracef, ", '%s'", info->s->open_file_name); + /* detect if an open instance of a dropped table (internal bug) */ + DBUG_ASSERT(info->s->last_version != 0); + if (cmp_translog_addr(info->s->state.create_rename_lsn, rec->lsn) >= 0) + { + fprintf(tracef, ", has create_rename_lsn (%lu,0x%lx) more recent than log" + " record\n", + (ulong) LSN_FILE_NO(rec->lsn), (ulong) LSN_OFFSET(rec->lsn)); + return NULL; + } + fprintf(tracef, ", applying record\n"); + return info; + /* + Soon we will also skip the page depending on the rec_lsn for this page in + the checkpoint record, but this is not absolutely needed for now (just + assume we have made no checkpoint). Btw rec_lsn and bitmap's recovery is a + an unsolved problem (rec_lsn is to ignore a REDO without reading the data + page and to do so we need to be sure the corresponding bitmap page does + not need a _ma_bitmap_set()). + */ +} + + +static MARIA_HA *get_MARIA_HA_from_UNDO_record(const + TRANSLOG_HEADER_BUFFER *rec) +{ + uint16 sid; + MARIA_HA *info; + + sid= fileid_korr(rec->header + LSN_STORE_SIZE); + fprintf(tracef, " For table of short id %u", sid); + info= all_tables[sid]; + if (info == NULL) + { + fprintf(tracef, ", table skipped, so skipping record\n"); + return NULL; + } + fprintf(tracef, ", '%s'", info->s->open_file_name); + DBUG_ASSERT(info->s->last_version != 0); + if (cmp_translog_addr(info->s->state.create_rename_lsn, rec->lsn) >= 0) + { + fprintf(tracef, ", has create_rename_lsn (%lu,0x%lx) more recent than log" + " record\n", + (ulong) LSN_FILE_NO(rec->lsn), (ulong) LSN_OFFSET(rec->lsn)); + return NULL; + } + fprintf(tracef, ", applying record\n"); + return info; + /* + Soon we will also skip the page depending on the rec_lsn for this page in + the checkpoint record, but this is not absolutely needed for now (just + assume we have made no checkpoint). + */ +} + + + + +/* some comments and pseudo-code which we keep for later */ +#if 0 + /* MikaelR suggests: support checkpoints during REDO phase too: do checkpoint after a certain amount of log records have been executed. This helps against repeated crashes. Those checkpoints could not be user-requested @@ -214,8 +1131,7 @@ int recovery() /**** UNDO PHASE *****/ - print_information_to_error_log(nb of trans to roll back, nb of prepared trans); - + print_information_to_error_log(nb of trans to roll back, nb of prepared trans /* Launch one or more threads to do the background rollback. Don't wait for them to complete their rollback (background rollback; for debugging, we @@ -265,3 +1181,4 @@ pthread_handler_decl rollback_background_thread() unlock_mutex(rollback_threads); pthread_exit(); } +#endif diff --git a/storage/maria/ma_recovery.h b/storage/maria/ma_recovery.h index 42c5071babd..0b576efc95f 100644 --- a/storage/maria/ma_recovery.h +++ b/storage/maria/ma_recovery.h @@ -22,4 +22,8 @@ /* This is the interface of this module. */ /* Performs recovery of the engine at start */ -int recovery(); + +C_MODE_START +int maria_recover(); +int maria_apply_log(LSN lsn, my_bool applyn, FILE *trace_file); +C_MODE_END diff --git a/storage/maria/ma_rename.c b/storage/maria/ma_rename.c index 9dd75705229..6250b781a68 100644 --- a/storage/maria/ma_rename.c +++ b/storage/maria/ma_rename.c @@ -62,8 +62,8 @@ int maria_rename(const char *old_name, const char *new_name) this is important; make sure transactionality has been re-enabled. */ DBUG_ASSERT(share->now_transactional == share->base.born_transactional); - sync_dir= (share->now_transactional && !share->temporary) ? - MY_SYNC_DIR : 0; + sync_dir= (share->now_transactional && !share->temporary && + !maria_in_recovery) ? MY_SYNC_DIR : 0; if (sync_dir) { uchar log_data[2 + 2]; diff --git a/storage/maria/ma_test2.c b/storage/maria/ma_test2.c index 1839efd0813..2673f331140 100644 --- a/storage/maria/ma_test2.c +++ b/storage/maria/ma_test2.c @@ -47,7 +47,7 @@ static void copy_key(struct st_maria_info *info,uint inx, static int verbose=0,testflag=0, first_key=0,async_io=0,pagecacheing=0,write_cacheing=0,locking=0, rec_pointer_size=0,pack_fields=1,silent=0, - opt_quick_mode=0, transactional= 0; + opt_quick_mode=0, transactional= 0, skip_update= 0; static int pack_seg=HA_SPACE_PACK,pack_type=HA_PACK_KEY,remove_count=-1; static int create_flag= 0, srand_arg= 0; static ulong pagecache_size=IO_SIZE*16; @@ -84,7 +84,24 @@ int main(int argc, char *argv[]) if (! async_io) my_disable_async_io=1; - maria_init(); + maria_data_root= "."; + /* Maria requires that we always have a page cache */ + if (maria_init() || + (init_pagecache(maria_pagecache, pagecache_size, 0, 0, + maria_block_size) == 0) || + ma_control_file_create_or_open(TRUE) || + (init_pagecache(maria_log_pagecache, + TRANSLOG_PAGECACHE_SIZE, 0, 0, + TRANSLOG_PAGE_SIZE) == 0) || + translog_init(maria_data_root, TRANSLOG_FILE_SIZE, + 0, 0, maria_log_pagecache, + TRANSLOG_DEFAULT_FLAGS) || + (transactional && trnman_init())) + { + fprintf(stderr, "Error in initialization"); + exit(1); + } + reclength=STANDARD_LENGTH+60+(use_blob ? 8 : 0); blob_pos=STANDARD_LENGTH+60; keyinfo[0].seg= &glob_keyseg[0][0]; @@ -220,22 +237,6 @@ int main(int argc, char *argv[]) goto err; if (!silent) printf("- Writing key:s\n"); - maria_data_root= "."; - /* Maria requires that we always have a page cache */ - if ((init_pagecache(maria_pagecache, pagecache_size, 0, 0, - maria_block_size) == 0) || - ma_control_file_create_or_open(TRUE) || - (init_pagecache(maria_log_pagecache, - TRANSLOG_PAGECACHE_SIZE, 0, 0, - TRANSLOG_PAGE_SIZE) == 0) || - translog_init(maria_data_root, TRANSLOG_FILE_SIZE, - 0, 0, maria_log_pagecache, - TRANSLOG_DEFAULT_FLAGS)) - { - fprintf(stderr, "Error in initialization"); - exit(1); - } - if (locking) maria_lock_database(file,F_WRLCK); if (write_cacheing) @@ -246,6 +247,14 @@ int main(int argc, char *argv[]) for (i=0 ; i < recant ; i++) { ulong blob_length; +#if 0 + /* + Starting from i==72, there was a difference between runtime and + log-appplying. This is now fixed, by not using non_header_data_len in + log-applying. + */ + if (i == 72) goto end; +#endif n1=rnd(1000); n2=rnd(100); n3=rnd(5000); sprintf(record,"%6d:%4d:%8d:Pos: %4d ",n1,n2,n3,write_count); int4store(record+STANDARD_LENGTH-4,(long) i); @@ -260,7 +269,7 @@ int main(int argc, char *argv[]) printf("Error: %d in write at record: %d\n",my_errno,i); goto err; } - if (verbose) printf(" Double key: %d\n",n3); + if (verbose) printf(" Double key: %d at record# %d\n", n3, i); } else { @@ -294,7 +303,7 @@ int main(int argc, char *argv[]) if (maria_extra(file,HA_EXTRA_NO_CACHE,0)) { puts("got error from maria_extra(HA_EXTRA_NO_CACHE)"); - goto end; + goto err; } } #ifdef REMOVE_WHEN_WE_HAVE_RESIZE @@ -376,6 +385,8 @@ int main(int argc, char *argv[]) else bmove(record+blob_pos,read_record+blob_pos,8); } + if (skip_update) + continue; if (maria_update(file,read_record,record2)) { if (my_errno != HA_ERR_FOUND_DUPP_KEY || key3[n3] == 0) @@ -423,7 +434,7 @@ int main(int argc, char *argv[]) if (memcmp(read_record,read_record2,reclength) != 0) { printf("maria_rsame didn't find same record\n"); - goto end; + goto err; } info.recpos=maria_position(file); if (maria_rfirst(file,read_record2,0) || @@ -431,7 +442,7 @@ int main(int argc, char *argv[]) memcmp(read_record,read_record2,reclength) != 0) { printf("maria_rsame_with_pos didn't find same record\n"); - goto end; + goto err; } { info.recpos= maria_position(file); @@ -442,7 +453,7 @@ int main(int argc, char *argv[]) info.recpos != maria_position(file)) { printf("maria_rsame_with_pos lost position\n"); - goto end; + goto err; } } ant=1; @@ -451,7 +462,7 @@ int main(int argc, char *argv[]) if (ant != dupp_keys) { printf("next: Found: %d keys of %d\n",ant,dupp_keys); - goto end; + goto err; } ant=0; while (maria_rprev(file,read_record3,0) == 0 && @@ -459,7 +470,7 @@ int main(int argc, char *argv[]) if (ant != dupp_keys) { printf("prev: Found: %d records of %d\n",ant,dupp_keys); - goto end; + goto err; } /* Check of maria_rnext_same */ @@ -471,7 +482,7 @@ int main(int argc, char *argv[]) if (ant != dupp_keys || my_errno != HA_ERR_END_OF_FILE) { printf("maria_rnext_same: Found: %d records of %d\n",ant,dupp_keys); - goto end; + goto err; } } @@ -482,7 +493,7 @@ int main(int argc, char *argv[]) if (maria_rfirst(file,read_record,0)) { printf("Can't find first record\n"); - goto end; + goto err; } while ((error=maria_rnext(file,read_record3,0)) == 0 && ant < write_count+10) ant++; @@ -490,7 +501,7 @@ int main(int argc, char *argv[]) { printf("next: I found: %d records of %d (error: %d)\n", ant, write_count - opt_delete, error); - goto end; + goto err; } if (maria_rlast(file,read_record2,0) || bcmp(read_record2,read_record3,reclength)) @@ -498,7 +509,7 @@ int main(int argc, char *argv[]) printf("Can't find last record\n"); DBUG_DUMP("record2",(byte*) read_record2,reclength); DBUG_DUMP("record3",(byte*) read_record3,reclength); - goto end; + goto err; } ant=1; while (maria_rprev(file,read_record3,0) == 0 && ant < write_count+10) @@ -506,12 +517,12 @@ int main(int argc, char *argv[]) if (ant != write_count - opt_delete) { printf("prev: I found: %d records of %d\n",ant,write_count); - goto end; + goto err; } if (bcmp(read_record,read_record3,reclength)) { printf("Can't find first record\n"); - goto end; + goto err; } if (!silent) @@ -552,7 +563,7 @@ int main(int argc, char *argv[]) if (bcmp(read_record+start,key,(uint) i)) { puts("Didn't find right record"); - goto end; + goto err; } } if (dupp_keys > 2) @@ -570,7 +581,7 @@ int main(int argc, char *argv[]) if (ant != dupp_keys-1) { printf("next: I can only find: %d keys of %d\n",ant,dupp_keys-1); - goto end; + goto err; } } if (dupp_keys>4) @@ -588,7 +599,7 @@ int main(int argc, char *argv[]) if (ant != dupp_keys-2) { printf("next: I can only find: %d keys of %d\n",ant,dupp_keys-2); - goto end; + goto err; } } if (dupp_keys > 6) @@ -607,7 +618,7 @@ int main(int argc, char *argv[]) if (ant != dupp_keys-3) { printf("next: I can only find: %d keys of %d\n",ant,dupp_keys-3); - goto end; + goto err; } if (!silent) @@ -622,7 +633,7 @@ int main(int argc, char *argv[]) if (ant != dupp_keys-4) { printf("next: I can only find: %d keys of %d\n",ant,dupp_keys-4); - goto end; + goto err; } } @@ -655,7 +666,7 @@ int main(int argc, char *argv[]) if (bcmp(read_record,read_record2,reclength) != 0) { printf("maria_rsame didn't find same record\n"); - goto end; + goto err; } } if (!silent) @@ -682,7 +693,7 @@ int main(int argc, char *argv[]) { printf("maria_records_range returned %ld; Should be about %ld\n", (long) range_records,(long) info.records); - goto end; + goto err; } if (verbose) { @@ -719,7 +730,7 @@ int main(int argc, char *argv[]) { printf("maria_records_range for key: %d returned %lu; Should be about %lu\n", i, (ulong) range_records, (ulong) records); - goto end; + goto err; } if (verbose && records) { @@ -740,6 +751,7 @@ int main(int argc, char *argv[]) puts("Wrong info from maria_info"); printf("Got: records: %lu delete: %lu i_keys: %d\n", (ulong) info.records, (ulong) info.deleted, info.keys); + goto err; } if (verbose) { @@ -764,7 +776,7 @@ int main(int argc, char *argv[]) if (locking || (!use_blob && !pack_fields)) { puts("got error from maria_extra(HA_EXTRA_CACHE)"); - goto end; + goto err; } } ant=0; @@ -777,12 +789,12 @@ int main(int argc, char *argv[]) { printf("scan with cache: I can only find: %d records of %d\n", ant,write_count-opt_delete); - goto end; + goto err; } if (maria_extra(file,HA_EXTRA_NO_CACHE,0)) { puts("got error from maria_extra(HA_EXTRA_NO_CACHE)"); - goto end; + goto err; } ant=0; @@ -794,7 +806,7 @@ int main(int argc, char *argv[]) { printf("scan with cache: I can only find: %d records of %d\n", ant,write_count-opt_delete); - goto end; + goto err; } if (testflag == 4) @@ -852,6 +864,15 @@ int main(int argc, char *argv[]) goto err; } opt_delete++; +#if 0 + / + /* + 179 is ok, 180 causes a difference between runtime and log-applying. + This is now fixed (we zero the last directory entry during + log-applying, just to eliminate this irrelevant difference). + */ + if (opt_delete==180) goto end; +#endif } else found_parts++; @@ -1021,6 +1042,9 @@ static void get_options(int argc, char **argv) case 'D': create_flag|=HA_CREATE_DELAY_KEY_WRITE; break; + case 'g': + skip_update= TRUE; + break; case '?': case 'I': case 'V': diff --git a/storage/maria/ma_test_all.sh b/storage/maria/ma_test_all.sh index a6786315afe..e8b9f1cef9a 100755 --- a/storage/maria/ma_test_all.sh +++ b/storage/maria/ma_test_all.sh @@ -6,6 +6,9 @@ # If you want to run this in Valgrind, you should use --trace-children=yes, # so that it detects problems in ma_test* and not in the shell script +# Running in a "shared memory" disk is 10 times faster; you can do +# mkdir /dev/shm/test; cd /dev/shm/test; maria_path= + # Remove # from following line if you need some more information #set -x -v -e @@ -21,6 +24,7 @@ fi # Delete temporary files rm -f *.TMD +rm -f maria_log* run_tests() { @@ -211,8 +215,14 @@ echo "$maria_path/maria_chk$suffix -sm test2 will warn that 'Datafile is almost $maria_path/maria_chk$suffix -sm test2 >ma_test2_message.txt 2>&1 cat ma_test2_message.txt grep "warning: Datafile is almost full" ma_test2_message.txt >/dev/null +rm -f ma_test2_message.txt $maria_path/maria_chk$suffix -ssm test2 +# +# Test that removing tables and applying the log leads to identical tables +# +/bin/sh $maria_path/ma_test_recovery + # # Some timing tests # diff --git a/storage/maria/ma_test_recovery b/storage/maria/ma_test_recovery new file mode 100644 index 00000000000..3393b932e18 --- /dev/null +++ b/storage/maria/ma_test_recovery @@ -0,0 +1,37 @@ +set -e + +if [ -z "$maria_path" ] +then + maria_path="." +fi + +echo "MARIA RECOVERY TESTS - success is if exit code is 0" + +# runs a program inserting/deleting rows, then moves the resulting table +# elsewhere; applies the log and checks that the data file is +# identical to the saved original. +# Does not test the index file as we don't have logging for it yet. + +rm -f maria_log* +prog="$maria_path/ma_test1 -M -T --skip-update" +echo "TEST WITH $prog" +$prog +mv -f test1.MAD test1.MAD.good +rm test1.MAI +echo "applying log" +$maria_path/maria_read_log -a > /dev/null +cmp test1.MAD test1.MAD.good +rm -f test1.* + +rm -f maria_log* +prog="$maria_path/ma_test2 -s -L -K -W -P -M -T -g" +echo "TEST WITH $prog" +$prog +mv -f test2.MAD test2.MAD.good +rm test2.MAI +echo "applying log" +$maria_path/maria_read_log -a > /dev/null +cmp test2.MAD test2.MAD.good +rm -f test2.* + +echo "ALL RECOVERY TESTS OK" diff --git a/storage/maria/maria_read_log.c b/storage/maria/maria_read_log.c index 2d664e08662..f8bfeb24826 100644 --- a/storage/maria/maria_read_log.c +++ b/storage/maria/maria_read_log.c @@ -14,7 +14,7 @@ Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ #include "maria_def.h" -#include +#include "ma_recovery.h" #include #define PCACHE_SIZE (1024*1024*10) @@ -32,60 +32,6 @@ const char *default_dbug_option= "d:t:i:o,/tmp/maria_read_log.trace"; #endif /* DBUG_OFF */ static my_bool opt_only_display, opt_display_and_apply; -struct TRN_FOR_RECOVERY -{ - LSN group_start_lsn, undo_lsn; - TrID long_trid; -}; - -struct TRN_FOR_RECOVERY all_active_trans[SHORT_TRID_MAX + 1]; -MARIA_HA *all_tables[SHORT_TRID_MAX + 1]; -LSN current_group_end_lsn= LSN_IMPOSSIBLE; - -static void end_of_redo_phase(); -static void display_record_position(const LOG_DESC *log_desc, - const TRANSLOG_HEADER_BUFFER *rec, - uint number); -static int display_and_apply_record(const LOG_DESC *log_desc, - const TRANSLOG_HEADER_BUFFER *rec); -#define prototype_exec_hook(R) \ -static int exec_LOGREC_ ## R(const TRANSLOG_HEADER_BUFFER *rec) -prototype_exec_hook(LONG_TRANSACTION_ID); -#ifdef MARIA_CHECKPOINT -prototype_exec_hook(CHECKPOINT); -#endif -prototype_exec_hook(REDO_CREATE_TABLE); -prototype_exec_hook(FILE_ID); -prototype_exec_hook(REDO_INSERT_ROW_HEAD); -prototype_exec_hook(REDO_INSERT_ROW_TAIL); -prototype_exec_hook(REDO_PURGE_ROW_HEAD); -prototype_exec_hook(REDO_PURGE_ROW_TAIL); -prototype_exec_hook(UNDO_ROW_INSERT); -prototype_exec_hook(UNDO_ROW_DELETE); -prototype_exec_hook(COMMIT); - - -/* - TODO: Avoid mallocs in exec. - - Proposed fix: - Add either a context/buffer argument to all exec_hook functions - or add 'record_buffer' and 'record_buffer_length' to - TRANSLOG_HEADER_BUFFER. - With this we could use my_realloc() instead of my_malloc() to - allocate data and save some mallocs. -*/ - -/* - To implement REDO_DROP_TABLE and REDO_RENAME_TABLE, we would need to go - through the all_tables[] array, find all open instances of the - table-to-drop-or-rename, and remove them from the array. - We however know that in real Recovery, we don't have to handle those log - records at all, same for REDO_CREATE_TABLE. - So for now, we can use this program to replay/debug a sequence of CREATE + - DMLs, but not DROP/RENAME; it is probably enough for a start. -*/ - int main(int argc, char **argv) { LSN lsn; @@ -97,6 +43,7 @@ int main(int argc, char **argv) get_options(&argc, &argv); maria_data_root= "."; + maria_in_recovery= TRUE; if (maria_init()) { @@ -114,6 +61,8 @@ int main(int argc, char **argv) fprintf(stderr, "Can't find any log\n"); goto err; } + /* same page cache for log and data; assumes same page size... */ + DBUG_ASSERT(maria_block_size == TRANSLOG_PAGE_SIZE); if (init_pagecache(maria_pagecache, PCACHE_SIZE, 0, 0, TRANSLOG_PAGE_SIZE) == 0) { @@ -133,147 +82,22 @@ int main(int argc, char **argv) goto err; } - /* install hooks for execution */ -#define install_exec_hook(R) \ - log_record_type_descriptor[LOGREC_ ## R].record_execute_in_redo_phase= \ - exec_LOGREC_ ## R; - install_exec_hook(LONG_TRANSACTION_ID); -#ifdef MARIA_CHECKPOINT - install_exec_hook(CHECKPOINT); -#endif - install_exec_hook(REDO_CREATE_TABLE); - install_exec_hook(FILE_ID); - install_exec_hook(REDO_INSERT_ROW_HEAD); - install_exec_hook(REDO_INSERT_ROW_TAIL); - install_exec_hook(REDO_PURGE_ROW_HEAD); - install_exec_hook(REDO_PURGE_ROW_TAIL); - install_exec_hook(UNDO_ROW_INSERT); - install_exec_hook(UNDO_ROW_DELETE); - install_exec_hook(COMMIT); - if (opt_only_display) printf("You are using --only-display, NOTHING will be written to disk\n"); - lsn= first_lsn_in_log(); /*could also be last_checkpoint_lsn */ - - TRANSLOG_HEADER_BUFFER rec; - struct st_translog_scanner_data scanner; - uint i= 1; + lsn= first_lsn_in_log(); /* LSN could be also --start-from-lsn=# */ - translog_size_t len= translog_read_record_header(lsn, &rec); - - if (len == (TRANSLOG_RECORD_HEADER_MAX_SIZE + 1)) - { - printf("EOF on the log\n"); - goto end; - } - - if (translog_init_scanner(lsn, 1, &scanner)) - { - fprintf(stderr, "Scanner init failed\n"); + fprintf(stdout, "TRACE of the last maria_read_log\n"); + if (maria_apply_log(lsn, opt_display_and_apply, stdout)) goto err; - } - for (;;i++) - { - uint16 sid= rec.short_trid; - const LOG_DESC *log_desc= &log_record_type_descriptor[rec.type]; - display_record_position(log_desc, &rec, i); + fprintf(stdout, "SUCCESS\n"); - /* - A complete group is a set of log records with an "end mark" record - (e.g. a set of REDOs for an operation, terminated by an UNDO for this - operation); if there is no "end mark" record the group is incomplete - and won't be executed. - There are pitfalls: if a table write failed, the transaction may have - put an incomplete group in the log and then a COMMIT record, that will - make a complete group which is wrong. We say that we should mark the - table corrupted if such error happens (what if it cannot be marked?). - */ - if (log_desc->record_ends_group) - { - if (all_active_trans[sid].group_start_lsn != LSN_IMPOSSIBLE) - { - /* - There is a complete group for this transaction, containing more than - this event. - */ - printf(" ends a group:\n"); - struct st_translog_scanner_data scanner2; - TRANSLOG_HEADER_BUFFER rec2; - len= - translog_read_record_header(all_active_trans[sid].group_start_lsn, &rec2); - if (len == (TRANSLOG_RECORD_HEADER_MAX_SIZE + 1)) - { - fprintf(stderr, "Cannot find record where it should be\n"); - goto err; - } - if (translog_init_scanner(rec2.lsn, 1, &scanner2)) - { - fprintf(stderr, "Scanner2 init failed\n"); - goto err; - } - current_group_end_lsn= rec.lsn; - do - { - if (rec2.short_trid == sid) /* it's in our group */ - { - const LOG_DESC *log_desc2= &log_record_type_descriptor[rec2.type]; - display_record_position(log_desc2, &rec2, 0); - if (display_and_apply_record(log_desc2, &rec2)) - goto err; - } - len= translog_read_next_record_header(&scanner2, &rec2); - if (len == (TRANSLOG_RECORD_HEADER_MAX_SIZE + 1)) - { - fprintf(stderr, "Cannot find record where it should be\n"); - goto err; - } - } - while (rec2.lsn < rec.lsn); - translog_free_record_header(&rec2); - /* group finished */ - all_active_trans[sid].group_start_lsn= LSN_IMPOSSIBLE; - current_group_end_lsn= LSN_IMPOSSIBLE; /* for debugging */ - } - if (display_and_apply_record(log_desc, &rec)) - goto err; - } - else /* record does not end group */ - { - /* just record the fact, can't know if can execute yet */ - if (all_active_trans[sid].group_start_lsn == LSN_IMPOSSIBLE) - { - /* group not yet started */ - all_active_trans[sid].group_start_lsn= rec.lsn; - } - } - len= translog_read_next_record_header(&scanner, &rec); - if (len == (TRANSLOG_RECORD_HEADER_MAX_SIZE + 1)) - { - printf("EOF on the log\n"); - goto end; - } - } - translog_free_record_header(&rec); - - /* - So we have applied all REDOs. - We may now have unfinished transactions. - I don't think it's this program's job to roll them back: - to roll back and at the same time stay idempotent, it needs to write log - records (without CLRs, 2nd rollback would hit the effects of first - rollback and fail). But this standalone tool is not allowed to write to - the server's transaction log. So we do not roll back anything. - In the real Recovery code, or the code to do "recover after online - backup", yes we will roll back. - */ - end_of_redo_phase(); goto end; err: /* don't touch anything more, in case we hit a bug */ exit(1); end: - maria_panic(HA_PANIC_CLOSE); + maria_end(); free_defaults(default_argv); my_end(0); exit(0); @@ -355,629 +179,3 @@ static void get_options(int *argc,char ***argv) exit(1); } } - - -/* very basic info about the record's header */ -static void display_record_position(const LOG_DESC *log_desc, - const TRANSLOG_HEADER_BUFFER *rec, - uint number) -{ - /* - if number==0, we're going over records which we had already seen and which - form a group, so we indent below the group's end record - */ - printf("%sRec#%u LSN (%lu,0x%lx) short_trid %u %s(num_type:%u) len %lu\n", - number ? "" : " ", number, - (ulong) LSN_FILE_NO(rec->lsn), (ulong) LSN_OFFSET(rec->lsn), - rec->short_trid, log_desc->name, rec->type, - (ulong)rec->record_length); -} - - -static int display_and_apply_record(const LOG_DESC *log_desc, - const TRANSLOG_HEADER_BUFFER *rec) -{ - int error; - if (opt_only_display) - return 0; - if (log_desc->record_execute_in_redo_phase == NULL) - { - /* die on all not-yet-handled records :) */ - DBUG_ASSERT("one more hook" == "to write"); - } - if ((error= (*log_desc->record_execute_in_redo_phase)(rec))) - fprintf(stderr, "Got error when executing record\n"); - return error; -} - - -prototype_exec_hook(LONG_TRANSACTION_ID) -{ - uint16 sid= rec->short_trid; - TrID long_trid= all_active_trans[sid].long_trid; - /* abort group of this trn (must be of before a crash) */ - LSN gslsn= all_active_trans[sid].group_start_lsn; - char llbuf[22]; - if (gslsn != LSN_IMPOSSIBLE) - { - printf("Group at LSN (%lu,0x%lx) short_trid %u aborted\n", - (ulong) LSN_FILE_NO(gslsn), (ulong) LSN_OFFSET(gslsn), sid); - all_active_trans[sid].group_start_lsn= LSN_IMPOSSIBLE; - } - if (long_trid != 0) - { - LSN ulsn= all_active_trans[sid].undo_lsn; - if (ulsn != LSN_IMPOSSIBLE) - { - llstr(long_trid, llbuf); - fprintf(stderr, "Found an old transaction long_trid %s short_trid %u" - " with same short id as this new transaction, and has neither" - " committed nor rollback (undo_lsn: (%lu,0x%lx))\n", llbuf, - sid, (ulong) LSN_FILE_NO(ulsn), (ulong) LSN_OFFSET(ulsn)); - goto err; - } - } - long_trid= uint6korr(rec->header); - all_active_trans[sid].long_trid= long_trid; - llstr(long_trid, llbuf); - printf("Transaction long_trid %s short_trid %u starts\n", llbuf, sid); - goto end; -err: - DBUG_ASSERT(0); - return 1; -end: - return 0; -} - - -#ifdef MARIA_CHECKPOINT -prototype_exec_hook(CHECKPOINT) -{ - /* the only checkpoint we care about was found via control file, ignore */ - return 0; -} -#endif - - -prototype_exec_hook(REDO_CREATE_TABLE) -{ - File dfile= -1, kfile= -1; - char *linkname_ptr, filename[FN_REFLEN]; - char *name, *ptr; - myf create_flag; - uint flags; - int error, create_mode= O_RDWR | O_TRUNC; - MARIA_HA *info= NULL; - if (((name= my_malloc(rec->record_length, MYF(MY_WME))) == NULL) || - (translog_read_record(rec->lsn, 0, rec->record_length, name, NULL) != - rec->record_length)) - { - fprintf(stderr, "Failed to read record\n"); - goto err; - } - printf("Table '%s'", name); - /* we try hard to get create_rename_lsn, to avoid mistakes if possible */ - info= maria_open(name, O_RDONLY, HA_OPEN_FOR_REPAIR); - if (info) - { - MARIA_SHARE *share= info->s; - /* check that we're not already using it */ - DBUG_ASSERT(share->reopen == 1); - DBUG_ASSERT(share->now_transactional == share->base.born_transactional); - if (!share->base.born_transactional) - { - /* - could be that transactional table was later dropped, and a non-trans - one was renamed to its name, thus create_rename_lsn is 0 and should - not be trusted. - */ - printf(", is not transactional\n"); - DBUG_ASSERT(0); /* I want to know this */ - goto end; - } - if (cmp_translog_addr(share->state.create_rename_lsn, rec->lsn) >= 0) - { - printf(", has create_rename_lsn (%lu,0x%lx) is more recent than record", - (ulong) LSN_FILE_NO(rec->lsn), - (ulong) LSN_OFFSET(rec->lsn)); - goto end; - } - if (maria_is_crashed(info)) - { - printf(", is crashed, overwriting it"); - DBUG_ASSERT(0); /* I want to know this */ - } - maria_close(info); - info= NULL; - } - /* if does not exist, is older, or its header is corrupted, overwrite it */ - // TODO symlinks - ptr= name + strlen(name) + 1; - if ((flags= ptr[0] ? HA_DONT_TOUCH_DATA : 0)) - printf(", we will only touch index file"); - fn_format(filename, name, "", MARIA_NAME_IEXT, - (MY_UNPACK_FILENAME | - (flags & HA_DONT_TOUCH_DATA) ? MY_RETURN_REAL_PATH : 0) | - MY_APPEND_EXT); - linkname_ptr= NULL; - create_flag= MY_DELETE_OLD; - printf(", creating as '%s'", filename); - if ((kfile= my_create_with_symlink(linkname_ptr, filename, 0, create_mode, - MYF(MY_WME|create_flag))) < 0) - { - fprintf(stderr, "Failed to create index file\n"); - goto err; - } - ptr++; - uint kfile_size_before_extension= uint2korr(ptr); - ptr+= 2; - uint keystart= uint2korr(ptr); - ptr+= 2; - /* set create_rename_lsn (for maria_read_log to be idempotent) */ - lsn_store(ptr + sizeof(info->s->state.header) + 2, rec->lsn); - if (my_pwrite(kfile, ptr, - kfile_size_before_extension, 0, MYF(MY_NABP|MY_WME)) || - my_chsize(kfile, keystart, 0, MYF(MY_WME))) - { - fprintf(stderr, "Failed to write to index file\n"); - goto err; - } - if (!(flags & HA_DONT_TOUCH_DATA)) - { - fn_format(filename,name,"", MARIA_NAME_DEXT, - MY_UNPACK_FILENAME | MY_APPEND_EXT); - linkname_ptr= NULL; - create_flag=MY_DELETE_OLD; - if ((dfile= - my_create_with_symlink(linkname_ptr, filename, 0, create_mode, - MYF(MY_WME | create_flag))) < 0) - { - fprintf(stderr, "Failed to create data file\n"); - goto err; - } - /* - we now have an empty data file. To be able to - _ma_initialize_data_file() we need some pieces of the share to be - correctly filled. So we just open the table (fortunately, an empty - data file does not preclude this). - */ - if (((info= maria_open(name, O_RDONLY, 0)) == NULL) || - _ma_initialize_data_file(info->s, dfile)) - { - fprintf(stderr, "Failed to open new table or write to data file\n"); - goto err; - } - } - error= 0; - goto end; -err: - DBUG_ASSERT(0); - error= 1; -end: - printf("\n"); - if (kfile >= 0) - error|= my_close(kfile, MYF(MY_WME)); - if (dfile >= 0) - error|= my_close(dfile, MYF(MY_WME)); - if (info != NULL) - error|= maria_close(info); - my_free(name, MYF(MY_ALLOW_ZERO_PTR)); - return 0; -} - - -prototype_exec_hook(FILE_ID) -{ - uint16 sid; - int error; - char *name, *buff; - MARIA_HA *info= NULL; - MARIA_SHARE *share; - if (((buff= my_malloc(rec->record_length, MYF(MY_WME))) == NULL) || - (translog_read_record(rec->lsn, 0, rec->record_length, buff, NULL) != - rec->record_length)) - { - fprintf(stderr, "Failed to read record\n"); - goto err; - } - sid= fileid_korr(buff); - name= buff + FILEID_STORE_SIZE; - printf("Table '%s', id %u", name, sid); - info= all_tables[sid]; - if (info != NULL) - { - printf(", closing table '%s'", info->s->open_file_name); - all_tables[sid]= NULL; - _ma_reenable_logging_for_table(info->s); /* put back the truth */ - if (maria_close(info)) - { - fprintf(stderr, "Failed to close table\n"); - goto err; - } - } - info= maria_open(name, O_RDWR, HA_OPEN_FOR_REPAIR); - if (info == NULL) - { - printf(", is absent (must have been dropped later?)" - " or its header is so corrupted that we cannot open it;" - " we skip it\n"); - goto end; - } - if (maria_is_crashed(info)) - { - fprintf(stderr, "Table is crashed, can't apply log records to it\n"); - goto err; - } - share= info->s; - /* check that we're not already using it */ - DBUG_ASSERT(share->reopen == 1); - DBUG_ASSERT(share->now_transactional == share->base.born_transactional); - if (!share->base.born_transactional) - { - printf(", is not transactional\n"); - DBUG_ASSERT(0); /* I want to know this */ - goto end; - } - all_tables[sid]= info; - /* don't log any records for this work */ - _ma_tmp_disable_logging_for_table(share); - printf(", opened\n"); - error= 0; - goto end; -err: - DBUG_ASSERT(0); - error= 1; - if (info != NULL) - error|= maria_close(info); -end: - my_free(buff, MYF(MY_ALLOW_ZERO_PTR)); - return 0; -} - - -prototype_exec_hook(REDO_INSERT_ROW_HEAD) -{ - uint16 sid; - ulonglong page; - MARIA_HA *info; - char llbuf[22]; - byte *buff= 0; - - sid= fileid_korr(rec->header); - page= page_korr(rec->header + FILEID_STORE_SIZE); - llstr(page, llbuf); - printf("For page %s of table of short id %u", llbuf, sid); - info= all_tables[sid]; - if (info == NULL) - { - printf(", table skipped, so skipping record\n"); - goto end; - } - printf(", '%s'", info->s->open_file_name); - if (cmp_translog_addr(info->s->state.create_rename_lsn, rec->lsn) >= 0) - { - printf(", has create_rename_lsn (%lu,0x%lx) is more recent than log" - " record\n", - (ulong) LSN_FILE_NO(rec->lsn), (ulong) LSN_OFFSET(rec->lsn)); - goto end; - } - /* - Soon we will also skip the page depending on the rec_lsn for this page in - the checkpoint record, but this is not absolutely needed for now (just - assume we have made no checkpoint). - */ - printf(", applying record\n"); - /* - If REDO's LSN is > page's LSN (read from disk), we are going to modify the - page and change its LSN. The normal runtime code stores the UNDO's LSN - into the page. Here storing the REDO's LSN (rec->lsn) would work - (we are not writing to the log here, so don't have to "flush up to UNDO's - LSN"). But in a test scenario where we do updates at runtime, then remove - tables, apply the log and check that this results in the same table as at - runtime, putting the same LSN as runtime had done will decrease - differences. So we use the UNDO's LSN which is current_group_end_lsn. - */ - - if ((!(buff= (byte*) my_malloc(rec->record_length, MYF(MY_WME)))) || - (translog_read_record(rec->lsn, 0, rec->record_length, buff, NULL) != - rec->record_length)) - { - fprintf(stderr, "Failed to read record\n"); - goto end; - } - if (_ma_apply_redo_insert_row_head_or_tail(info, rec->lsn, HEAD_PAGE, - rec->header + FILEID_STORE_SIZE, - buff + (rec->record_length - - rec->non_header_data_len), - rec->non_header_data_len)) - goto end; - my_free(buff, MYF(0)); - return 0; - -end: - /* as we don't have apply working: */ - my_free(buff, MYF(MY_ALLOW_ZERO_PTR)); - return 1; -} - - -prototype_exec_hook(REDO_INSERT_ROW_TAIL) -{ - uint16 sid; - ulonglong page; - MARIA_HA *info; - char llbuf[22]; - byte *buff= 0; - - sid= fileid_korr(rec->header); - page= page_korr(rec->header + FILEID_STORE_SIZE); - llstr(page, llbuf); - printf("For page %s of table of short id %u", llbuf, sid); - info= all_tables[sid]; - if (info == NULL) - { - printf(", table skipped, so skipping record\n"); - goto end; - } - printf(", '%s'", info->s->open_file_name); - if (cmp_translog_addr(info->s->state.create_rename_lsn, rec->lsn) >= 0) - { - printf(", has create_rename_lsn (%lu,0x%lx) is more recent than log" - " record\n", - (ulong) LSN_FILE_NO(rec->lsn), (ulong) LSN_OFFSET(rec->lsn)); - goto end; - } - /* - Soon we will also skip the page depending on the rec_lsn for this page in - the checkpoint record, but this is not absolutely needed for now (just - assume we have made no checkpoint). - */ - printf(", applying record\n"); - /* - If REDO's LSN is > page's LSN (read from disk), we are going to modify the - page and change its LSN. The normal runtime code stores the UNDO's LSN - into the page. Here storing the REDO's LSN (rec->lsn) would work - (we are not writing to the log here, so don't have to "flush up to UNDO's - LSN"). But in a test scenario where we do updates at runtime, then remove - tables, apply the log and check that this results in the same table as at - runtime, putting the same LSN as runtime had done will decrease - differences. So we use the UNDO's LSN which is current_group_end_lsn. - */ - - if ((!(buff= (byte*) my_malloc(rec->record_length, MYF(MY_WME)))) || - (translog_read_record(rec->lsn, 0, rec->record_length, buff, NULL) != - rec->record_length)) - { - fprintf(stderr, "Failed to read record\n"); - goto end; - } - if (_ma_apply_redo_insert_row_head_or_tail(info, rec->lsn, TAIL_PAGE, - rec->header + FILEID_STORE_SIZE, - buff + (rec->record_length - - rec->non_header_data_len), - rec->non_header_data_len)) - goto end; - - my_free(buff, MYF(0)); - return 0; - -end: - /* as we don't have apply working: */ - my_free(buff, MYF(MY_ALLOW_ZERO_PTR)); - return 1; -} - - -prototype_exec_hook(REDO_PURGE_ROW_HEAD) -{ - uint16 sid; - ulonglong page; - MARIA_HA *info; - char llbuf[22]; - - sid= fileid_korr(rec->header); - page= page_korr(rec->header + FILEID_STORE_SIZE); - llstr(page, llbuf); - printf("For page %s of table of short id %u", llbuf, sid); - info= all_tables[sid]; - if (info == NULL) - { - printf(", table skipped, so skipping record\n"); - goto end; - } - printf(", '%s'", info->s->open_file_name); - if (cmp_translog_addr(info->s->state.create_rename_lsn, rec->lsn) >= 0) - { - printf(", has create_rename_lsn (%lu,0x%lx) is more recent than log" - " record\n", - (ulong) LSN_FILE_NO(rec->lsn), (ulong) LSN_OFFSET(rec->lsn)); - goto end; - } - /* - Soon we will also skip the page depending on the rec_lsn for this page in - the checkpoint record, but this is not absolutely needed for now (just - assume we have made no checkpoint). - */ - printf(", applying record\n"); - /* - If REDO's LSN is > page's LSN (read from disk), we are going to modify the - page and change its LSN. The normal runtime code stores the UNDO's LSN - into the page. Here storing the REDO's LSN (rec->lsn) would work - (we are not writing to the log here, so don't have to "flush up to UNDO's - LSN"). But in a test scenario where we do updates at runtime, then remove - tables, apply the log and check that this results in the same table as at - runtime, putting the same LSN as runtime had done will decrease - differences. So we use the UNDO's LSN which is current_group_end_lsn. - */ - - if (_ma_apply_redo_purge_row_head_or_tail(info, rec->lsn, HEAD_PAGE, - rec->header + FILEID_STORE_SIZE)) - goto end; - - return 0; - -end: - /* as we don't have apply working: */ - return 1; -} - - -prototype_exec_hook(REDO_PURGE_ROW_TAIL) -{ - uint16 sid; - ulonglong page; - MARIA_HA *info; - char llbuf[22]; - - sid= fileid_korr(rec->header); - page= page_korr(rec->header + FILEID_STORE_SIZE); - llstr(page, llbuf); - printf("For page %s of table of short id %u", llbuf, sid); - info= all_tables[sid]; - if (info == NULL) - { - printf(", table skipped, so skipping record\n"); - goto end; - } - printf(", '%s'", info->s->open_file_name); - if (cmp_translog_addr(info->s->state.create_rename_lsn, rec->lsn) >= 0) - { - printf(", has create_rename_lsn (%lu,0x%lx) is more recent than log" - " record\n", - (ulong) LSN_FILE_NO(rec->lsn), (ulong) LSN_OFFSET(rec->lsn)); - goto end; - } - /* - Soon we will also skip the page depending on the rec_lsn for this page in - the checkpoint record, but this is not absolutely needed for now (just - assume we have made no checkpoint). - */ - printf(", applying record\n"); - /* - If REDO's LSN is > page's LSN (read from disk), we are going to modify the - page and change its LSN. The normal runtime code stores the UNDO's LSN - into the page. Here storing the REDO's LSN (rec->lsn) would work - (we are not writing to the log here, so don't have to "flush up to UNDO's - LSN"). But in a test scenario where we do updates at runtime, then remove - tables, apply the log and check that this results in the same table as at - runtime, putting the same LSN as runtime had done will decrease - differences. So we use the UNDO's LSN which is current_group_end_lsn. - */ - - if (_ma_apply_redo_purge_row_head_or_tail(info, rec->lsn, TAIL_PAGE, - rec->header + FILEID_STORE_SIZE)) - goto end; - - return 0; - -end: - /* as we don't have apply working: */ - return 1; -} - - -static int exec_LOGREC_UNDO_ROW_INSERT(const TRANSLOG_HEADER_BUFFER *rec - __attribute__((unused))) -{ - /* Ignore this during the redo phase */ - return 0; -} - -static int exec_LOGREC_UNDO_ROW_DELETE(const TRANSLOG_HEADER_BUFFER *rec - __attribute__((unused))) -{ - /* Ignore this during the redo phase */ - return 0; -} - - - -prototype_exec_hook(COMMIT) -{ - uint16 sid= rec->short_trid; - TrID long_trid= all_active_trans[sid].long_trid; - LSN gslsn= all_active_trans[sid].group_start_lsn; - char llbuf[22]; - if (long_trid == 0) - { - printf("We don't know about transaction short_trid %u;" - "it probably committed long ago, forget it\n", sid); - return 0; - } - llstr(long_trid, llbuf); - printf("Transaction long_trid %s short_trid %u committed", llbuf, sid); - if (gslsn != LSN_IMPOSSIBLE) - { - /* - It's not an error, it may be that trn got a disk error when writing to a - table, so an unfinished group staid in the log. - */ - printf(", with group at LSN (%lu,0x%lx) short_trid %u aborted\n", - (ulong) LSN_FILE_NO(gslsn), (ulong) LSN_OFFSET(gslsn), sid); - all_active_trans[sid].group_start_lsn= LSN_IMPOSSIBLE; - } - else - printf("\n"); - all_active_trans[sid].long_trid= 0; -#ifdef MARIA_VERSIONING - /* - if real recovery: - transaction was committed, move it to some separate list for later - purging (but don't purge now! purging may have been started before, we - may find REDO_PURGE records soon). - */ -#endif - return 0; -} - - -/* Just to inform about any aborted groups or unfinished transactions */ -static void end_of_redo_phase() -{ - uint sid, unfinished= 0; - for (sid= 0; sid <= SHORT_TRID_MAX; sid++) - { - TrID long_trid= all_active_trans[sid].long_trid; - LSN gslsn= all_active_trans[sid].group_start_lsn; - if (long_trid == 0) - continue; - if (all_active_trans[sid].undo_lsn != LSN_IMPOSSIBLE) - { - char llbuf[22]; - llstr(long_trid, llbuf); - printf("Transaction long_trid %s short_trid %u unfinished\n", - llbuf, sid); - } - if (gslsn != LSN_IMPOSSIBLE) - { - printf("Group at LSN (%lu,0x%lx) short_trid %u aborted\n", - (ulong) LSN_FILE_NO(gslsn), (ulong) LSN_OFFSET(gslsn), sid); - } - /* If real recovery: roll back unfinished transaction */ -#ifdef MARIA_VERSIONING - /* - If real recovery: transaction was committed, move it to some separate - list for soon purging. - */ -#endif - } - /* - We don't close tables if there are some unfinished transactions, because - closing tables normally requires that all unfinished transactions on them - be rolled back. - For example, closing will soon write the state to disk and when doing that - it will think this is a committed state, but it may not be. - */ - if (unfinished == 0) - { - for (sid= 0; sid <= SHORT_TRID_MAX; sid++) - { - MARIA_HA *info= all_tables[sid]; - if (info != NULL) - { - _ma_reenable_logging_for_table(info->s); /* put back the truth */ - maria_close(info); - } - } - } -} -- cgit v1.2.1 From 662002fc8f6301ec78d2eca14c5f991212d4f67f Mon Sep 17 00:00:00 2001 From: unknown Date: Thu, 26 Jul 2007 17:51:49 +0200 Subject: post-merge fixes, and fixes for some of the 16 compiler warnings found in pushbuild on sapsrv1. Some not fixed as not repeatable on my machine (32/64 bit issue?). Fixes for some test failures: - "maria-connect" now passes; - "maria": after fixing the obvious reasons for failures, the test went further and hit a more complex issues: difference in the output of EXPLAIN output; not fixed; - "ps_maria" still crashes in assertion mysqld: ha_maria.cc:1627: virtual int ha_maria::index_read(uchar*, const uchar*, uint, ha_rkey_function): Ass ertion `inited == INDEX' failed, as already observable in pushbuild. All this might just be due to an incomplete merge of MyISAM changes into Maria when 5.1 was last merged to mysql-maria. include/my_global.h: temporary fix until next merge of 5.1; without this it does not build mysql-test/r/maria-connect.result: position changed mysql-test/t/maria-connect.test: If one wants to use the binlog it has to ask for it. 1582 is not used for dup entry error anymore (it was in older 5.1). Size of first event in binlog was increased by 4 (when the new type of event "gap" was added). mysql-test/t/maria.test: 1582 not used anymore in this case storage/maria/ha_maria.cc: engine now has to say what binlogging it supports storage/maria/ma_blockrec.c: fix for compiler warnings ("comparison is always true" or "always false") storage/maria/ma_loghandler.c: fix for compiler warnings (comparing char* to uchar*) storage/maria/ma_packrec.c: fix for compiler warning (fix simply merged from MyISAM) storage/maria/ma_pagecache.c: info_check_pin() was not used so gave a compiler warning. storage/maria/ma_pagecache.h: fixing typo from the last 5.1->maria merge. storage/maria/ma_recovery.c: my_free() has a void* argument, so why cast. byte->uchar. storage/maria/ma_search.c: fix for compiler warning (fix simply merged from MyISAM) storage/maria/maria_read_log.c: gptr->uchar* storage/maria/trnman.c: probable fix for warning found in pushbuild (but not on my machine): storage/maria/trnman.c: 142 passing argument 6 of \u2018lf_hash_init\u2019 from incompatible pointer type on sapsrv1. --- include/my_global.h | 7 +++++++ mysql-test/r/maria-connect.result | 8 ++++---- mysql-test/t/maria-connect.test | 5 +++-- mysql-test/t/maria.test | 4 ++-- storage/maria/ha_maria.cc | 1 + storage/maria/ma_blockrec.c | 10 +++++----- storage/maria/ma_loghandler.c | 2 +- storage/maria/ma_packrec.c | 3 ++- storage/maria/ma_pagecache.c | 3 ++- storage/maria/ma_pagecache.h | 2 +- storage/maria/ma_recovery.c | 8 ++++---- storage/maria/ma_search.c | 4 ++-- storage/maria/maria_read_log.c | 6 +++--- storage/maria/trnman.c | 2 +- 14 files changed, 38 insertions(+), 27 deletions(-) diff --git a/include/my_global.h b/include/my_global.h index af9c31cf34f..ce74490ed38 100644 --- a/include/my_global.h +++ b/include/my_global.h @@ -77,6 +77,13 @@ #endif #endif /* _WIN32... */ +/* + STACK_DIRECTION was removed from 5.1 and then that was merged into Maria; + then it was added back into 5.1 but not yet merged into Maria. + When merge done, remove this. +*/ +#define STACK_DIRECTION -1 + /* Make it easier to add conditionl code for windows */ #ifdef __WIN__ #define IF_WIN(A,B) (A) diff --git a/mysql-test/r/maria-connect.result b/mysql-test/r/maria-connect.result index e232f564d10..8fe483c8467 100644 --- a/mysql-test/r/maria-connect.result +++ b/mysql-test/r/maria-connect.result @@ -14,10 +14,10 @@ a 2 3 4 -SHOW BINLOG EVENTS FROM 102; +SHOW BINLOG EVENTS FROM 106; Log_name Pos Event_type Server_id End_log_pos Info -master-bin.000001 102 Query 1 200 use `test`; CREATE TABLE t1 (a int primary key) -master-bin.000001 200 Query 1 291 use `test`; insert t1 values (1),(2),(3) -master-bin.000001 291 Query 1 382 use `test`; insert t1 values (4),(2),(5) +master-bin.000001 106 Query 1 204 use `test`; CREATE TABLE t1 (a int primary key) +master-bin.000001 204 Query 1 295 use `test`; insert t1 values (1),(2),(3) +master-bin.000001 295 Query 1 386 use `test`; insert t1 values (4),(2),(5) drop table t1; set binlog_format=default; diff --git a/mysql-test/t/maria-connect.test b/mysql-test/t/maria-connect.test index aedfa92e278..8fa23909ecb 100644 --- a/mysql-test/t/maria-connect.test +++ b/mysql-test/t/maria-connect.test @@ -3,6 +3,7 @@ # -- source include/have_maria.inc +-- source include/have_log_bin.inc let $default=`select @@global.storage_engine`; set global storage_engine=maria; @@ -23,10 +24,10 @@ RESET MASTER; set binlog_format=statement; CREATE TABLE t1 (a int primary key); insert t1 values (1),(2),(3); ---error 1582 +--error ER_DUP_ENTRY insert t1 values (4),(2),(5); select * from t1; -SHOW BINLOG EVENTS FROM 102; +SHOW BINLOG EVENTS FROM 106; drop table t1; set binlog_format=default; diff --git a/mysql-test/t/maria.test b/mysql-test/t/maria.test index f03d744f850..26637f1f651 100644 --- a/mysql-test/t/maria.test +++ b/mysql-test/t/maria.test @@ -470,9 +470,9 @@ create table t1 (a int not null auto_increment primary key, b text not null, uni insert into t1 (b) values ('a'),('b'),('c'); select concat(b,'.') from t1; update t1 set b='b ' where a=2; ---error 1582 +--error ER_DUP_ENTRY update t1 set b='b ' where a > 1; ---error 1582 +--error ER_DUP_ENTRY insert into t1 (b) values ('b'); select * from t1; delete from t1 where b='b'; diff --git a/storage/maria/ha_maria.cc b/storage/maria/ha_maria.cc index 8a2b8ad99ac..05b79d6f10c 100644 --- a/storage/maria/ha_maria.cc +++ b/storage/maria/ha_maria.cc @@ -486,6 +486,7 @@ void _ma_check_print_warning(HA_CHECK *param, const char *fmt, ...) ha_maria::ha_maria(handlerton *hton, TABLE_SHARE *table_arg): handler(hton, table_arg), file(0), int_table_flags(HA_NULL_IN_KEY | HA_CAN_FULLTEXT | HA_CAN_SQL_HANDLER | + HA_BINLOG_ROW_CAPABLE | HA_BINLOG_STMT_CAPABLE | HA_DUPLICATE_POS | HA_CAN_INDEX_BLOBS | HA_AUTO_PART_KEY | HA_FILE_BASED | HA_CAN_GEOMETRY | MARIA_CANNOT_ROLLBACK | HA_CAN_BIT_FIELD | HA_CAN_RTREEKEYS | diff --git a/storage/maria/ma_blockrec.c b/storage/maria/ma_blockrec.c index c89f7465f26..37400560197 100644 --- a/storage/maria/ma_blockrec.c +++ b/storage/maria/ma_blockrec.c @@ -1742,7 +1742,7 @@ static my_bool write_block_record(MARIA_HA *info, int2store(page_buff + EMPTY_SPACE_OFFSET, row_pos->empty_space); /* Mark in bitmaps how the current page was actually used */ head_block->empty_space= row_pos->empty_space; - if (page_buff[DIR_COUNT_OFFSET] == (char) MAX_ROWS_PER_PAGE) + if (page_buff[DIR_COUNT_OFFSET] == MAX_ROWS_PER_PAGE) head_block->empty_space= 0; /* Page is full */ head_block->used= BLOCKUSED_USED; } @@ -4394,13 +4394,13 @@ uint _ma_apply_redo_purge_row_head_or_tail(MARIA_HA *info, LSN lsn, */ uint _ma_apply_redo_purge_blocks(MARIA_HA *info, - LSN lsn, const byte *header) + LSN lsn, const uchar *header) { MARIA_SHARE *share= info->s; ulonglong page; uint page_range; uint res; - byte *buff= info->keyread_buff; + uchar *buff= info->keyread_buff; uint block_size= share->block_size; DBUG_ENTER("_ma_apply_redo_purge_blocks"); @@ -4441,8 +4441,8 @@ uint _ma_apply_redo_purge_blocks(MARIA_HA *info, */ { uint rownr= ((uint) ((uchar *) buff)[DIR_COUNT_OFFSET]) - 1; - byte *dir= (buff + block_size - DIR_ENTRY_SIZE * rownr - - DIR_ENTRY_SIZE - PAGE_SUFFIX_SIZE); + uchar *dir= (buff + block_size - DIR_ENTRY_SIZE * rownr - + DIR_ENTRY_SIZE - PAGE_SUFFIX_SIZE); dir[0]= dir[1]= 0; /* Delete entry */ } diff --git a/storage/maria/ma_loghandler.c b/storage/maria/ma_loghandler.c index fa604f71b4d..fd007e2a5ae 100644 --- a/storage/maria/ma_loghandler.c +++ b/storage/maria/ma_loghandler.c @@ -3541,7 +3541,7 @@ static my_bool translog_relative_LSN_encode(struct st_translog_parts *parts, uchar *dst_ptr= compressed_LSNs + (MAX_NUMBER_OF_LSNS_PER_RECORD * COMPRESSED_LSN_MAX_STORE_SIZE); for (src_ptr= buffer + lsns_len - LSN_STORE_SIZE; - src_ptr >= buffer; + src_ptr >= (uchar *)buffer; src_ptr-= LSN_STORE_SIZE) { ref= lsn_korr(src_ptr); diff --git a/storage/maria/ma_packrec.c b/storage/maria/ma_packrec.c index 5d54e42ac4f..ae3920dbb3c 100644 --- a/storage/maria/ma_packrec.c +++ b/storage/maria/ma_packrec.c @@ -170,7 +170,8 @@ static my_bool _ma_read_pack_info(MARIA_SHARE *share, File file, uint i,trees,huff_tree_bits,rec_reflength,length; uint16 *decode_table,*tmp_buff; ulong elements,intervall_length; - char *disk_cache,*intervall_buff; + char *disk_cache; + uchar *intervall_buff; uchar header[HEAD_LENGTH]; MARIA_BIT_BUFF bit_buff; DBUG_ENTER("_ma_read_pack_info"); diff --git a/storage/maria/ma_pagecache.c b/storage/maria/ma_pagecache.c index eb939ba9eb0..12e561c69bc 100755 --- a/storage/maria/ma_pagecache.c +++ b/storage/maria/ma_pagecache.c @@ -320,6 +320,8 @@ struct st_pagecache_block_link #ifndef DBUG_OFF /* debug checks */ + +#ifdef NOT_USED static my_bool info_check_pin(PAGECACHE_BLOCK_LINK *block, enum pagecache_page_pin mode __attribute__((unused))) @@ -380,7 +382,6 @@ static my_bool info_check_pin(PAGECACHE_BLOCK_LINK *block, 1 - Error */ -#ifdef NOT_USED static my_bool info_check_lock(PAGECACHE_BLOCK_LINK *block, enum pagecache_page_lock lock, enum pagecache_page_pin pin) diff --git a/storage/maria/ma_pagecache.h b/storage/maria/ma_pagecache.h index 86426c5b4bc..2f244c572d0 100644 --- a/storage/maria/ma_pagecache.h +++ b/storage/maria/ma_pagecache.h @@ -95,7 +95,7 @@ typedef struct st_pagecache_hash_link PAGECACHE_HASH_LINK; #include -typedef my_bool (*pagecache_disk_read_validator)(uchar *page, uchar** data); +typedef my_bool (*pagecache_disk_read_validator)(uchar *page, uchar *data); #define PAGECACHE_CHANGED_BLOCKS_HASH 128 /* must be power of 2 */ diff --git a/storage/maria/ma_recovery.c b/storage/maria/ma_recovery.c index eb802969bce..b8a1206cdb6 100644 --- a/storage/maria/ma_recovery.c +++ b/storage/maria/ma_recovery.c @@ -321,8 +321,8 @@ err: error= 1; fprintf(tracef, "Recovery of tables with transaction logs FAILED\n"); end: - my_free((gptr)all_tables, MYF(MY_ALLOW_ZERO_PTR)); - my_free((gptr)all_active_trans, MYF(MY_ALLOW_ZERO_PTR)); + my_free(all_tables, MYF(MY_ALLOW_ZERO_PTR)); + my_free(all_active_trans, MYF(MY_ALLOW_ZERO_PTR)); my_free(log_record_buffer.str, MYF(MY_ALLOW_ZERO_PTR)); log_record_buffer.str= NULL; log_record_buffer.length= 0; @@ -701,7 +701,7 @@ end: prototype_exec_hook(REDO_INSERT_ROW_HEAD) { int error= 1; - byte *buff= NULL; + uchar *buff= NULL; MARIA_HA *info= get_MARIA_HA_from_REDO_record(rec); if (info == NULL) goto end; @@ -746,7 +746,7 @@ end: prototype_exec_hook(REDO_INSERT_ROW_TAIL) { int error= 1; - byte *buff= NULL; + uchar *buff= NULL; MARIA_HA *info= get_MARIA_HA_from_REDO_record(rec); if (info == NULL) goto end; diff --git a/storage/maria/ma_search.c b/storage/maria/ma_search.c index 1a7263e6a5a..d9bc5cd6264 100644 --- a/storage/maria/ma_search.c +++ b/storage/maria/ma_search.c @@ -834,7 +834,7 @@ uint _ma_get_pack_key(register MARIA_KEYDEF *keyinfo, uint nod_flag, tot_length=rest_length+length; /* If the stored length has changed, we must move the key */ - if (tot_length >= 255 && *start != (char) 255) + if (tot_length >= 255 && *start != 255) { /* length prefix changed from a length of one to a length of 3 */ bmove_upp((char*) key+length+3,(char*) key+length+1,length); @@ -842,7 +842,7 @@ uint _ma_get_pack_key(register MARIA_KEYDEF *keyinfo, uint nod_flag, mi_int2store(key+1,tot_length); key+=3+length; } - else if (tot_length < 255 && *start == (char) 255) + else if (tot_length < 255 && *start == 255) { bmove(key+1,key+3,length); *key=tot_length; diff --git a/storage/maria/maria_read_log.c b/storage/maria/maria_read_log.c index f8bfeb24826..3ac3809ec04 100644 --- a/storage/maria/maria_read_log.c +++ b/storage/maria/maria_read_log.c @@ -108,11 +108,11 @@ end: static struct my_option my_long_options[] = { {"only-display", 'o', "display brief info about records's header", - (gptr*) &opt_only_display, (gptr*) &opt_only_display, 0, GET_BOOL, NO_ARG, - 0, 0, 0, 0, 0, 0}, + (uchar **) &opt_only_display, (uchar **) &opt_only_display, 0, GET_BOOL, + NO_ARG,0, 0, 0, 0, 0, 0}, {"display-and-apply", 'a', "like --only-display but displays more info and modifies tables", - (gptr*) &opt_display_and_apply, (gptr*) &opt_display_and_apply, 0, + (uchar **) &opt_display_and_apply, (uchar **) &opt_display_and_apply, 0, GET_BOOL, NO_ARG, 0, 0, 0, 0, 0, 0}, #ifndef DBUG_OFF {"debug", '#', "Output debug log. Often this is 'd:t:o,filename'.", diff --git a/storage/maria/trnman.c b/storage/maria/trnman.c index 75226588912..177ee2a7a70 100644 --- a/storage/maria/trnman.c +++ b/storage/maria/trnman.c @@ -94,7 +94,7 @@ static TRN *short_trid_to_TRN(uint16 short_trid) } #endif -static uchar *trn_get_hash_key(const uchar *trn, uint* len, +static uchar *trn_get_hash_key(const uchar *trn, size_t *len, my_bool unused __attribute__ ((unused))) { *len= sizeof(TrID); -- cgit v1.2.1 From a3d2ae4648d739a7ec7820e22c05373fde65b770 Mon Sep 17 00:00:00 2001 From: unknown Date: Fri, 27 Jul 2007 12:06:39 +0200 Subject: merging MyISAM changes into Maria (not done in 5.1->maria merge of Jul 7th). "maria.test" and "ps_maria.test" still fail; "ma_test_all" starts failing (MyISAM has the same issue see BUG#30094). include/maria.h: merging MyISAM changes into Maria mysys/mf_keycache.c: mi_test_all showed "floating point exception", this was already fixed in the latest 5.1, importing fix. sql/item_xmlfunc.cc: compiler warning (already fixed in latest 5.1) storage/maria/ha_maria.cc: merging MyISAM changes into Maria. See #ifdef ASK_MONTY. storage/maria/ha_maria.h: merging MyISAM changes into Maria storage/maria/ma_cache.c: merging MyISAM changes into Maria storage/maria/ma_check.c: merging MyISAM changes into Maria storage/maria/ma_create.c: merging MyISAM changes into Maria storage/maria/ma_dynrec.c: merging MyISAM changes into Maria storage/maria/ma_extra.c: merging MyISAM changes into Maria storage/maria/ma_ft_boolean_search.c: merging MyISAM changes into Maria storage/maria/ma_ft_nlq_search.c: merging MyISAM changes into Maria storage/maria/ma_info.c: merging MyISAM changes into Maria storage/maria/ma_key.c: merging MyISAM changes into Maria storage/maria/ma_loghandler.c: compiler warning (part->length is size_t) storage/maria/ma_open.c: merging MyISAM changes into Maria storage/maria/ma_preload.c: merging MyISAM changes into Maria storage/maria/ma_range.c: merging MyISAM changes into Maria storage/maria/ma_rkey.c: merging MyISAM changes into Maria storage/maria/ma_rt_index.c: merging MyISAM changes into Maria storage/maria/ma_rt_key.c: merging MyISAM changes into Maria storage/maria/ma_rt_split.c: merging MyISAM changes into Maria storage/maria/ma_search.c: merging MyISAM changes into Maria storage/maria/ma_sort.c: merging MyISAM changes into Maria storage/maria/maria_def.h: merging MyISAM changes into Maria --- include/maria.h | 4 +- mysys/mf_keycache.c | 14 ++- sql/item_xmlfunc.cc | 2 +- storage/maria/ha_maria.cc | 198 ++++++++++++++++++++++++++--------- storage/maria/ha_maria.h | 13 ++- storage/maria/ma_cache.c | 4 +- storage/maria/ma_check.c | 16 +-- storage/maria/ma_create.c | 26 +++-- storage/maria/ma_dynrec.c | 14 +-- storage/maria/ma_extra.c | 10 +- storage/maria/ma_ft_boolean_search.c | 51 +++++---- storage/maria/ma_ft_nlq_search.c | 8 +- storage/maria/ma_info.c | 6 +- storage/maria/ma_key.c | 64 ++++------- storage/maria/ma_loghandler.c | 4 +- storage/maria/ma_open.c | 48 ++++----- storage/maria/ma_preload.c | 13 ++- storage/maria/ma_range.c | 63 ++++++++--- storage/maria/ma_rkey.c | 15 +-- storage/maria/ma_rt_index.c | 107 ++++++++++++++----- storage/maria/ma_rt_key.c | 18 +++- storage/maria/ma_rt_split.c | 12 ++- storage/maria/ma_search.c | 42 ++++++-- storage/maria/ma_sort.c | 17 +-- storage/maria/maria_def.h | 5 +- 25 files changed, 508 insertions(+), 266 deletions(-) diff --git a/include/maria.h b/include/maria.h index abe6463edd0..e39498149a9 100644 --- a/include/maria.h +++ b/include/maria.h @@ -265,8 +265,8 @@ extern struct st_maria_info *maria_clone(struct st_maria_share *share, int mode) extern int maria_panic(enum ha_panic_function function); extern int maria_rfirst(struct st_maria_info *file, uchar *buf, int inx); extern int maria_rkey(struct st_maria_info *file, uchar *buf, int inx, - const uchar *key, - uint key_len, enum ha_rkey_function search_flag); + const uchar *key, key_part_map keypart_map, + enum ha_rkey_function search_flag); extern int maria_rlast(struct st_maria_info *file, uchar *buf, int inx); extern int maria_rnext(struct st_maria_info *file, uchar *buf, int inx); extern int maria_rnext_same(struct st_maria_info *info, uchar *buf); diff --git a/mysys/mf_keycache.c b/mysys/mf_keycache.c index 6845e63dc33..4a8ab5b6719 100644 --- a/mysys/mf_keycache.c +++ b/mysys/mf_keycache.c @@ -2522,10 +2522,8 @@ uchar *key_cache_read(KEY_CACHE *keycache, int error=0; uchar *start= buff; DBUG_ENTER("key_cache_read"); - DBUG_PRINT("enter", ("fd: %u pos: %lu page: %lu length: %u", - (uint) file, (ulong) filepos, - (ulong) (filepos / keycache->key_cache_block_size), - length)); + DBUG_PRINT("enter", ("fd: %u pos: %lu length: %u", + (uint) file, (ulong) filepos, length)); if (keycache->key_cache_inited) { @@ -2979,10 +2977,10 @@ int key_cache_write(KEY_CACHE *keycache, int error=0; DBUG_ENTER("key_cache_write"); DBUG_PRINT("enter", - ("fd: %u pos: %lu page: %lu length: %u block_length: %u", - (uint) file, (ulong) filepos, - (ulong) (filepos / keycache->key_cache_block_size), - length, block_length)); + ("fd: %u pos: %lu length: %u block_length: %u key_block_length: + %u", + (uint) file, (ulong) filepos, length, block_length, + keycache ? keycache->key_cache_block_size : 0)); if (!dont_write) { diff --git a/sql/item_xmlfunc.cc b/sql/item_xmlfunc.cc index 6bc73f49a5a..f8457a1ae50 100644 --- a/sql/item_xmlfunc.cc +++ b/sql/item_xmlfunc.cc @@ -2767,7 +2767,7 @@ String *Item_xml_str_func::parse_xml(String *raw_xml, String *parsed_xml_buf) if ((rc= my_xml_parse(&p, raw_xml->ptr(), raw_xml->length())) != MY_XML_OK) { char buf[128]; - my_snprintf(buf, sizeof(buf)-1, "parse error at line %d pos %u: %s", + my_snprintf(buf, sizeof(buf)-1, "parse error at line %d pos %lu: %s", my_xml_error_lineno(&p) + 1, my_xml_error_pos(&p) + 1, my_xml_error_string(&p)); diff --git a/storage/maria/ha_maria.cc b/storage/maria/ha_maria.cc index 05b79d6f10c..d0888278cca 100644 --- a/storage/maria/ha_maria.cc +++ b/storage/maria/ha_maria.cc @@ -114,6 +114,14 @@ static void _ma_check_print_msg(HA_CHECK *param, const char *msg_type, } length= (uint) (strxmov(name, param->db_name, ".", param->table_name, NullS) - name); + /* + TODO: switch from protocol to push_warning here. The main reason we didn't + it yet is parallel repair. Due to following trace: + ma_check_print_msg/push_warning/sql_alloc/my_pthread_getspecific_ptr. + + Also we likely need to lock mutex here (in both cases with protocol and + push_warning). + */ protocol->prepare_for_resend(); protocol->store(name, length, system_charset_info); protocol->store(param->op_name, system_charset_info); @@ -353,10 +361,11 @@ int table2maria(TABLE *table_arg, MARIA_KEYDEF **keydef_out, 0 - Equal definitions. 1 - Different definitions. - NOTES - This is currently not used. In MyISAM the corresponding function - (myisam_check_definition()) is used only by MERGE tables - (in ha_myisammrg.cc). + TODO + - compare FULLTEXT keys; + - compare SPATIAL keys; + - compare FIELD_SKIP_ZERO which is converted to FIELD_NORMAL correctly + (should be corretly detected in table2maria). */ int maria_check_definition(MARIA_KEYDEF *t1_keyinfo, MARIA_COLUMNDEF *t1_recinfo, @@ -383,6 +392,28 @@ int maria_check_definition(MARIA_KEYDEF *t1_keyinfo, { HA_KEYSEG *t1_keysegs= t1_keyinfo[i].seg; HA_KEYSEG *t2_keysegs= t2_keyinfo[i].seg; + if (t1_keyinfo[i].flag & HA_FULLTEXT && t2_keyinfo[i].flag & HA_FULLTEXT) + continue; + else if (t1_keyinfo[i].flag & HA_FULLTEXT || + t2_keyinfo[i].flag & HA_FULLTEXT) + { + DBUG_PRINT("error", ("Key %d has different definition", i)); + DBUG_PRINT("error", ("t1_fulltext= %d, t2_fulltext=%d", + test(t1_keyinfo[i].flag & HA_FULLTEXT), + test(t2_keyinfo[i].flag & HA_FULLTEXT))); + DBUG_RETURN(1); + } + if (t1_keyinfo[i].flag & HA_SPATIAL && t2_keyinfo[i].flag & HA_SPATIAL) + continue; + else if (t1_keyinfo[i].flag & HA_SPATIAL || + t2_keyinfo[i].flag & HA_SPATIAL) + { + DBUG_PRINT("error", ("Key %d has different definition", i)); + DBUG_PRINT("error", ("t1_spatial= %d, t2_spatial=%d", + test(t1_keyinfo[i].flag & HA_SPATIAL), + test(t2_keyinfo[i].flag & HA_SPATIAL))); + DBUG_RETURN(1); + } if (t1_keyinfo[i].keysegs != t2_keyinfo[i].keysegs || t1_keyinfo[i].key_alg != t2_keyinfo[i].key_alg) { @@ -419,7 +450,14 @@ int maria_check_definition(MARIA_KEYDEF *t1_keyinfo, { MARIA_COLUMNDEF *t1_rec= &t1_recinfo[i]; MARIA_COLUMNDEF *t2_rec= &t2_recinfo[i]; - if (t1_rec->type != t2_rec->type || + /* + FIELD_SKIP_ZERO can be changed to FIELD_NORMAL in maria_create, + see NOTE1 in ma_create.c + */ + if ((t1_rec->type != t2_rec->type && + !(t1_rec->type == (int) FIELD_SKIP_ZERO && + t1_rec->length == 1 && + t2_rec->type == (int) FIELD_NORMAL)) || t1_rec->length != t2_rec->length || t1_rec->null_bit != t2_rec->null_bit) { @@ -602,7 +640,7 @@ int ha_maria::dump(THD * thd, int fd) my_seek(data_fd, 0L, MY_SEEK_SET, MYF(MY_WME)); for (; bytes_to_read > 0;) { - uint bytes= my_read(data_fd, buf, block_size, MYF(MY_WME)); + size_t bytes= my_read(data_fd, buf, block_size, MYF(MY_WME)); if (bytes == MY_FILE_ERROR) { error= errno; @@ -644,7 +682,8 @@ err: bool ha_maria::check_if_locking_is_allowed(uint sql_command, ulong type, TABLE *table, - uint count, + uint count, uint current, + uint *system_count, bool called_by_privileged_thread) { /* @@ -653,10 +692,13 @@ bool ha_maria::check_if_locking_is_allowed(uint sql_command, we have to disallow write-locking of these tables with any other tables. */ if (table->s->system_table && - table->reginfo.lock_type >= TL_WRITE_ALLOW_WRITE && count != 1) + table->reginfo.lock_type >= TL_WRITE_ALLOW_WRITE) + (*system_count)++; + + /* 'current' is an index, that's why '<=' below. */ + if (*system_count > 0 && *system_count <= current) { - my_error(ER_WRONG_LOCK_OF_SYSTEM_TABLE, MYF(0), table->s->db.str, - table->s->table_name.str); + my_error(ER_WRONG_LOCK_OF_SYSTEM_TABLE, MYF(0)); return FALSE; } @@ -676,6 +718,9 @@ bool ha_maria::check_if_locking_is_allowed(uint sql_command, int ha_maria::open(const char *name, int mode, uint test_if_locked) { + MARIA_KEYDEF *keyinfo; + MARIA_COLUMNDEF *recinfo= 0; + uint recs; uint i; #ifdef NOT_USED @@ -701,6 +746,36 @@ int ha_maria::open(const char *name, int mode, uint test_if_locked) if (!(file= maria_open(name, mode, test_if_locked | HA_OPEN_FROM_SQL_LAYER))) return (my_errno ? my_errno : -1); +#ifdef ASK_MONTY + /* + This is a protection for the case of a frm and MAI containing incompatible + table definitions (as in BUG#25908). This was merged from MyISAM. + But it breaks maria.test and ps_maria.test ("incorrect key file") if the + table is BLOCK_RECORD (does it have to do with column reordering done in + ma_create.c ?). + */ + if (!table->s->tmp_table) /* No need to perform a check for tmp table */ + { + if ((my_errno= table2maria(table, &keyinfo, &recinfo, &recs))) + { + /* purecov: begin inspected */ + DBUG_PRINT("error", ("Failed to convert TABLE object to Maria " + "key and column definition")); + goto err; + /* purecov: end */ + } + if (maria_check_definition(keyinfo, recinfo, table->s->keys, recs, + file->s->keyinfo, file->s->columndef, + file->s->base.keys, file->s->base.fields, true)) + { + /* purecov: begin inspected */ + my_errno= HA_ERR_CRASHED; + goto err; + /* purecov: end */ + } + } +#endif + if (test_if_locked & (HA_OPEN_IGNORE_IF_LOCKED | HA_OPEN_TMP_TABLE)) VOID(maria_extra(file, HA_EXTRA_NO_WAIT_LOCK, 0)); @@ -731,7 +806,18 @@ int ha_maria::open(const char *name, int mode, uint test_if_locked) (struct st_mysql_ftparser *)plugin_decl(parser)->info; table->key_info[i].block_size= file->s->keyinfo[i].block_length; } - return (0); + my_errno= 0; + goto end; + err: + this->close(); + end: + /* + Both recinfo and keydef are allocated by my_multi_malloc(), thus only + recinfo must be freed. + */ + if (recinfo) + my_free((uchar*) recinfo, MYF(0)); + return my_errno; } @@ -745,7 +831,7 @@ int ha_maria::close(void) int ha_maria::write_row(uchar * buf) { - statistic_increment(table->in_use->status_var.ha_write_count, &LOCK_status); + ha_statistic_increment(&SSV::ha_write_count); /* If we have a timestamp column, update it to the current time */ if (table->timestamp_field_type & TIMESTAMP_AUTO_SET_ON_INSERT) @@ -1064,8 +1150,8 @@ int ha_maria::optimize(THD * thd, HA_CHECK_OPT *check_opt) param.sort_buffer_length= check_opt->sort_buffer_size; if ((error= repair(thd, param, 1)) && param.retry_repair) { - sql_print_warning("Warning: Optimize table got errno %d, retrying", - my_errno); + sql_print_warning("Warning: Optimize table got errno %d on %s.%s, retrying", + my_errno, param.db_name, param.table_name); param.testflag &= ~T_REP_BY_SORT; error= repair(thd, param, 1); } @@ -1084,6 +1170,22 @@ int ha_maria::repair(THD *thd, HA_CHECK ¶m, bool do_optimize) ha_rows rows= file->state->records; DBUG_ENTER("ha_maria::repair"); + /* + Normally this method is entered with a properly opened table. If the + repair fails, it can be repeated with more elaborate options. Under + special circumstances it can happen that a repair fails so that it + closed the data file and cannot re-open it. In this case file->dfile + is set to -1. We must not try another repair without an open data + file. (Bug #25289) + */ + if (file->dfile.file == -1) + { + sql_print_information("Retrying repair of: '%s' failed. " + "Please try REPAIR EXTENDED or maria_chk", + table->s->path.str); + DBUG_RETURN(HA_ADMIN_FAILED); + } + param.db_name= table->s->db.str; param.table_name= table->alias; param.tmpfile_createflag= O_RDWR | O_TRUNC; @@ -1222,7 +1324,7 @@ int ha_maria::assign_to_keycache(THD * thd, HA_CHECK_OPT *check_opt) PAGECACHE *new_pagecache= check_opt->pagecache; const char *errmsg= 0; int error= HA_ADMIN_OK; - ulonglong map= ~(ulonglong) 0; + ulonglong map; TABLE_LIST *table_list= table->pos_in_table_list; DBUG_ENTER("ha_maria::assign_to_keycache"); @@ -1268,9 +1370,10 @@ int ha_maria::preload_keys(THD * thd, HA_CHECK_OPT *check_opt) { int error; const char *errmsg; - ulonglong map= ~(ulonglong) 0; + ulonglong map; TABLE_LIST *table_list= table->pos_in_table_list; my_bool ignore_leaves= table_list->ignore_leaves; + char buf[ERRMSGSIZE+20]; DBUG_ENTER("ha_maria::preload_keys"); @@ -1298,7 +1401,6 @@ int ha_maria::preload_keys(THD * thd, HA_CHECK_OPT *check_opt) errmsg= "Failed to allocate buffer"; break; default: - char buf[ERRMSGSIZE + 20]; my_snprintf(buf, ERRMSGSIZE, "Failed to read from index file (errno: %d)", my_errno); errmsg= buf; @@ -1427,13 +1529,13 @@ int ha_maria::enable_indexes(uint mode) T_CREATE_MISSING_KEYS); param.myf_rw &= ~MY_WAIT_IF_FULL; param.sort_buffer_length= thd->variables.maria_sort_buff_size; - param.stats_method= (enum_handler_stats_method) thd->variables. - maria_stats_method; + param.stats_method= + (enum_handler_stats_method) thd->variables.maria_stats_method; param.tmpdir= &mysql_tmpdir_list; if ((error= (repair(thd, param, 0) != HA_ADMIN_OK)) && param.retry_repair) { - sql_print_warning("Warning: Enabling keys got errno %d, retrying", - my_errno); + sql_print_warning("Warning: Enabling keys got errno %d on %s.%s, retrying", + my_errno, param.db_name, param.table_name); /* Repairing by sort failed. Now try standard repair method. */ param.testflag &= ~(T_REP_BY_SORT | T_QUICK); error= (repair(thd, param, 0) != HA_ADMIN_OK); @@ -1442,8 +1544,10 @@ int ha_maria::enable_indexes(uint mode) might have been set by the first repair. They can still be seen with SHOW WARNINGS then. */ +#ifndef EMBEDDED_LIBRARY if (!error) thd->clear_error(); +#endif /* EMBEDDED_LIBRARY */ } info(HA_STATUS_CONST); thd->proc_info= save_proc_info; @@ -1607,7 +1711,7 @@ bool ha_maria::is_crashed() const int ha_maria::update_row(const uchar * old_data, uchar * new_data) { - statistic_increment(table->in_use->status_var.ha_update_count, &LOCK_status); + ha_statistic_increment(&SSV::ha_update_count); if (table->timestamp_field_type & TIMESTAMP_AUTO_SET_ON_UPDATE) table->timestamp_field->set_time(); return maria_update(file, old_data, new_data); @@ -1616,41 +1720,41 @@ int ha_maria::update_row(const uchar * old_data, uchar * new_data) int ha_maria::delete_row(const uchar * buf) { - statistic_increment(table->in_use->status_var.ha_delete_count, &LOCK_status); + ha_statistic_increment(&SSV::ha_delete_count); return maria_delete(file, buf); } int ha_maria::index_read(uchar * buf, const uchar * key, - uint key_len, enum ha_rkey_function find_flag) + key_part_map keypart_map, + enum ha_rkey_function find_flag) { DBUG_ASSERT(inited == INDEX); - statistic_increment(table->in_use->status_var.ha_read_key_count, - &LOCK_status); - int error= maria_rkey(file, buf, active_index, key, key_len, find_flag); + ha_statistic_increment(&SSV::ha_read_key_count); + int error= maria_rkey(file, buf, active_index, key, keypart_map, find_flag); table->status= error ? STATUS_NOT_FOUND : 0; return error; } int ha_maria::index_read_idx(uchar * buf, uint index, const uchar * key, - uint key_len, enum ha_rkey_function find_flag) + key_part_map keypart_map, + enum ha_rkey_function find_flag) { - statistic_increment(table->in_use->status_var.ha_read_key_count, - &LOCK_status); - int error= maria_rkey(file, buf, index, key, key_len, find_flag); + ha_statistic_increment(&SSV::ha_read_key_count); + int error= maria_rkey(file, buf, index, key, keypart_map, find_flag); table->status= error ? STATUS_NOT_FOUND : 0; return error; } -int ha_maria::index_read_last(uchar * buf, const uchar * key, uint key_len) +int ha_maria::index_read_last(uchar * buf, const uchar * key, + key_part_map keypart_map) { DBUG_ENTER("ha_maria::index_read_last"); DBUG_ASSERT(inited == INDEX); - statistic_increment(table->in_use->status_var.ha_read_key_count, - &LOCK_status); - int error= maria_rkey(file, buf, active_index, key, key_len, + ha_statistic_increment(&SSV::ha_read_key_count); + int error= maria_rkey(file, buf, active_index, key, keypart_map, HA_READ_PREFIX_LAST); table->status= error ? STATUS_NOT_FOUND : 0; DBUG_RETURN(error); @@ -1660,8 +1764,7 @@ int ha_maria::index_read_last(uchar * buf, const uchar * key, uint key_len) int ha_maria::index_next(uchar * buf) { DBUG_ASSERT(inited == INDEX); - statistic_increment(table->in_use->status_var.ha_read_next_count, - &LOCK_status); + ha_statistic_increment(&SSV::ha_read_next_count); int error= maria_rnext(file, buf, active_index); table->status= error ? STATUS_NOT_FOUND : 0; return error; @@ -1671,8 +1774,7 @@ int ha_maria::index_next(uchar * buf) int ha_maria::index_prev(uchar * buf) { DBUG_ASSERT(inited == INDEX); - statistic_increment(table->in_use->status_var.ha_read_prev_count, - &LOCK_status); + ha_statistic_increment(&SSV::ha_read_prev_count); int error= maria_rprev(file, buf, active_index); table->status= error ? STATUS_NOT_FOUND : 0; return error; @@ -1682,8 +1784,7 @@ int ha_maria::index_prev(uchar * buf) int ha_maria::index_first(uchar * buf) { DBUG_ASSERT(inited == INDEX); - statistic_increment(table->in_use->status_var.ha_read_first_count, - &LOCK_status); + ha_statistic_increment(&SSV::ha_read_first_count); int error= maria_rfirst(file, buf, active_index); table->status= error ? STATUS_NOT_FOUND : 0; return error; @@ -1693,8 +1794,7 @@ int ha_maria::index_first(uchar * buf) int ha_maria::index_last(uchar * buf) { DBUG_ASSERT(inited == INDEX); - statistic_increment(table->in_use->status_var.ha_read_last_count, - &LOCK_status); + ha_statistic_increment(&SSV::ha_read_last_count); int error= maria_rlast(file, buf, active_index); table->status= error ? STATUS_NOT_FOUND : 0; return error; @@ -1706,8 +1806,7 @@ int ha_maria::index_next_same(uchar * buf, uint length __attribute__ ((unused))) { DBUG_ASSERT(inited == INDEX); - statistic_increment(table->in_use->status_var.ha_read_next_count, - &LOCK_status); + ha_statistic_increment(&SSV::ha_read_next_count); int error= maria_rnext_same(file, buf); table->status= error ? STATUS_NOT_FOUND : 0; return error; @@ -1732,8 +1831,7 @@ int ha_maria::rnd_end() int ha_maria::rnd_next(uchar *buf) { - statistic_increment(table->in_use->status_var.ha_read_rnd_next_count, - &LOCK_status); + ha_statistic_increment(&SSV::ha_read_rnd_next_count); int error= maria_scan(file, buf); table->status= error ? STATUS_NOT_FOUND : 0; return error; @@ -1748,8 +1846,7 @@ int ha_maria::restart_rnd_next(uchar *buf, uchar *pos) int ha_maria::rnd_pos(uchar * buf, uchar *pos) { - statistic_increment(table->in_use->status_var.ha_read_rnd_count, - &LOCK_status); + ha_statistic_increment(&SSV::ha_read_rnd_count); int error= maria_rrnd(file, buf, my_get_ptr(pos, ref_length)); table->status= error ? STATUS_NOT_FOUND : 0; return error; @@ -2111,7 +2208,8 @@ void ha_maria::get_auto_increment(ulonglong offset, ulonglong increment, table->key_info + table->s->next_number_index, table->s->next_number_key_offset); error= maria_rkey(file, table->record[1], (int) table->s->next_number_index, - key, table->s->next_number_key_offset, HA_READ_PREFIX_LAST); + key, make_prev_keypart_map(table->s->next_number_keypart), + HA_READ_PREFIX_LAST); if (error) nr= 1; else diff --git a/storage/maria/ha_maria.h b/storage/maria/ha_maria.h index c97919ab86a..7675778ab5b 100644 --- a/storage/maria/ha_maria.h +++ b/storage/maria/ha_maria.h @@ -68,18 +68,21 @@ public: virtual bool check_if_locking_is_allowed(uint sql_command, ulong type, TABLE * table, - uint count, + uint count, uint current, + uint *system_count, bool called_by_logger_thread); int open(const char *name, int mode, uint test_if_locked); int close(void); int write_row(uchar * buf); int update_row(const uchar * old_data, uchar * new_data); int delete_row(const uchar * buf); - int index_read(uchar * buf, const uchar * key, - uint key_len, enum ha_rkey_function find_flag); + int index_read(uchar * buf, const uchar * key, key_part_map keypart_map, + enum ha_rkey_function find_flag); int index_read_idx(uchar * buf, uint idx, const uchar * key, - uint key_len, enum ha_rkey_function find_flag); - int index_read_last(uchar * buf, const uchar * key, uint key_len); + key_part_map keypart_map, + enum ha_rkey_function find_flag); + int index_read_last(uchar * buf, const uchar * key, + key_part_map keypart_map); int index_next(uchar * buf); int index_prev(uchar * buf); int index_first(uchar * buf); diff --git a/storage/maria/ma_cache.c b/storage/maria/ma_cache.c index 44aa7a15058..6b1f9ec3fae 100644 --- a/storage/maria/ma_cache.c +++ b/storage/maria/ma_cache.c @@ -40,7 +40,7 @@ int _ma_read_cache(IO_CACHE *info, uchar *buff, my_off_t pos, uint length, { uint read_length,in_buff_length; my_off_t offset; - char *in_buff_pos; + uchar *in_buff_pos; DBUG_ENTER("_ma_read_cache"); if (pos < info->pos_in_file) @@ -61,7 +61,7 @@ int _ma_read_cache(IO_CACHE *info, uchar *buff, my_off_t pos, uint length, (my_off_t) (info->read_end - info->request_pos)) { in_buff_pos=info->request_pos+(uint) offset; - in_buff_length= min(length,(uint) ((char*)(info->read_end)-in_buff_pos)); + in_buff_length= min(length,(size_t) (info->read_end-in_buff_pos)); memcpy(buff,info->request_pos+(uint) offset,(size_t) in_buff_length); if (!(length-=in_buff_length)) DBUG_RETURN(0); diff --git a/storage/maria/ma_check.c b/storage/maria/ma_check.c index 7d936409170..ce4a610cec9 100644 --- a/storage/maria/ma_check.c +++ b/storage/maria/ma_check.c @@ -344,7 +344,7 @@ int maria_chk_size(HA_CHECK *param, register MARIA_HA *info) flush_pagecache_blocks(info->s->pagecache, &info->s->kfile, FLUSH_FORCE_WRITE); - size= my_seek(info->s->kfile.file, 0L, MY_SEEK_END, MYF(0)); + size= my_seek(info->s->kfile.file, 0L, MY_SEEK_END, MYF(MY_THREADSAFE)); if ((skr=(my_off_t) info->state->key_file_length) != size) { /* Don't give error if file generated by mariapack */ @@ -539,7 +539,7 @@ int maria_chk_key(HA_CHECK *param, register MARIA_HA *info) maria_extra(info,HA_EXTRA_KEYREAD,0); bzero(info->lastkey,keyinfo->seg->length); if (!maria_rkey(info, info->rec_buff, key, (const uchar*) info->lastkey, - keyinfo->seg->length, HA_READ_KEY_EXACT)) + (key_part_map)1, HA_READ_KEY_EXACT)) { /* Don't count this as a real warning, as maria_chk can't correct it */ uint save=param->warning_printed; @@ -603,7 +603,8 @@ static int chk_index_down(HA_CHECK *param, MARIA_HA *info, { /* purecov: begin tested */ /* Give it a chance to fit in the real file size. */ - my_off_t max_length= my_seek(info->s->kfile.file, 0L, MY_SEEK_END, MYF(0)); + my_off_t max_length= my_seek(info->s->kfile.file, 0L, MY_SEEK_END, + MYF(MY_THREADSAFE)); _ma_check_print_error(param, "Invalid key block position: %s " "key block size: %u file_length: %s", llstr(page, llbuff), keyinfo->block_length, @@ -4772,10 +4773,11 @@ int maria_test_if_almost_full(MARIA_HA *info) { if (info->s->options & HA_OPTION_COMPRESS_RECORD) return 0; - return (my_seek(info->s->kfile.file, 0L, MY_SEEK_END, MYF(0))/10*9 > - (my_off_t) (info->s->base.max_key_file_length) || - my_seek(info->dfile.file, 0L, MY_SEEK_END, MYF(0)) / 10 * 9 > - (my_off_t) info->s->base.max_data_file_length); + return my_seek(info->s->kfile.file, 0L, MY_SEEK_END, + MYF(MY_THREADSAFE))/10*9 > + (my_off_t) info->s->base.max_key_file_length || + my_seek(info->dfile.file, 0L, MY_SEEK_END, MYF(0)) / 10 * 9 > + (my_off_t) info->s->base.max_data_file_length; } /* Recreate table with bigger more alloced record-data */ diff --git a/storage/maria/ma_create.c b/storage/maria/ma_create.c index 1736e24a7b6..201c5603c25 100644 --- a/storage/maria/ma_create.c +++ b/storage/maria/ma_create.c @@ -239,6 +239,10 @@ int maria_create(const char *name, enum data_file_type datafile_type, column--; if (column->type == (int) FIELD_SKIP_ZERO && column->length == 1) { + /* + NOTE1: here we change a field type FIELD_SKIP_ZERO -> + FIELD_NORMAL + */ column->type=(int) FIELD_NORMAL; column->empty_pos= 0; column->empty_bit= 0; @@ -701,6 +705,10 @@ int maria_create(const char *name, enum data_file_type datafile_type, pthread_mutex_lock(&THR_LOCK_maria); + /* + NOTE: For test_if_reopen() we need a real path name. Hence we need + MY_RETURN_REAL_PATH for every fn_format(filename, ...). + */ if (ci->index_file_name) { char *iext= strrchr(ci->index_file_name, '.'); @@ -712,13 +720,14 @@ int maria_create(const char *name, enum data_file_type datafile_type, if ((path= strrchr(ci->index_file_name, FN_LIBCHAR))) *path= '\0'; fn_format(filename, name, ci->index_file_name, MARIA_NAME_IEXT, - MY_REPLACE_DIR | MY_UNPACK_FILENAME | MY_APPEND_EXT); + MY_REPLACE_DIR | MY_UNPACK_FILENAME | + MY_RETURN_REAL_PATH | MY_APPEND_EXT); } else { fn_format(filename, ci->index_file_name, "", MARIA_NAME_IEXT, - MY_UNPACK_FILENAME | (have_iext ? MY_REPLACE_EXT : - MY_APPEND_EXT)); + MY_UNPACK_FILENAME | MY_RETURN_REAL_PATH | + (have_iext ? MY_REPLACE_EXT : MY_APPEND_EXT)); } fn_format(linkname, name, "", MARIA_NAME_IEXT, MY_UNPACK_FILENAME|MY_APPEND_EXT); @@ -734,10 +743,11 @@ int maria_create(const char *name, enum data_file_type datafile_type, } else { + char *iext= strrchr(name, '.'); + int have_iext= iext && !strcmp(iext, MARIA_NAME_IEXT); fn_format(filename, name, "", MARIA_NAME_IEXT, - (MY_UNPACK_FILENAME | - (flags & HA_DONT_TOUCH_DATA) ? MY_RETURN_REAL_PATH : 0) | - MY_APPEND_EXT); + MY_UNPACK_FILENAME | MY_RETURN_REAL_PATH | + (have_iext ? MY_REPLACE_EXT : MY_APPEND_EXT)); linkname_ptr= NullS; /* Replace the current file. @@ -752,6 +762,10 @@ int maria_create(const char *name, enum data_file_type datafile_type, A TRUNCATE command checks for the table in the cache only and could be fooled to believe, the table is not open. Pull the emergency brake in this situation. (Bug #8306) + + + NOTE: The filename is compared against unique_file_name of every + open table. Hence we need a real path here. */ if (_ma_test_if_reopen(filename)) { diff --git a/storage/maria/ma_dynrec.c b/storage/maria/ma_dynrec.c index 28b970ef589..246f9787b09 100644 --- a/storage/maria/ma_dynrec.c +++ b/storage/maria/ma_dynrec.c @@ -887,7 +887,7 @@ uint _ma_rec_pack(MARIA_HA *info, register uchar *to, register const uchar *from) { uint length,new_length,flag,bit,i; - char *pos,*end,*startpos,*packpos; + uchar *pos,*end,*startpos,*packpos; enum en_fieldtype type; reg3 MARIA_COLUMNDEF *column; MARIA_BLOB *blob; @@ -941,7 +941,7 @@ uint _ma_rec_pack(MARIA_HA *info, register uchar *to, pos= (uchar*) from; end= (uchar*) from + length; if (type == FIELD_SKIP_ENDSPACE) { /* Pack trailing spaces */ - while (end > (char*) from && *(end-1) == ' ') + while (end > from && *(end-1) == ' ') end--; } else @@ -1007,8 +1007,8 @@ uint _ma_rec_pack(MARIA_HA *info, register uchar *to, *packpos= (char) (uchar) flag; if (info->s->calc_checksum) *to++= (uchar) info->cur_row.checksum; - DBUG_PRINT("exit",("packed length: %d",(int) ((char*)to-startpos))); - DBUG_RETURN((uint) ((char*)to-startpos)); + DBUG_PRINT("exit",("packed length: %d",(int) (to-startpos))); + DBUG_RETURN((uint) (to-startpos)); } /* _ma_rec_pack */ @@ -1018,12 +1018,12 @@ uint _ma_rec_pack(MARIA_HA *info, register uchar *to, Returns 0 if record is ok. */ -my_bool _ma_rec_check(MARIA_HA *info,const char *record, uchar *rec_buff, +my_bool _ma_rec_check(MARIA_HA *info,const uchar *record, uchar *rec_buff, ulong packed_length, my_bool with_checksum, ha_checksum checksum) { uint length,new_length,flag,bit,i; - char *pos,*end,*packpos,*to; + uchar *pos,*end,*packpos,*to; enum en_fieldtype type; reg3 MARIA_COLUMNDEF *column; DBUG_ENTER("_ma_rec_check"); @@ -1123,7 +1123,7 @@ my_bool _ma_rec_check(MARIA_HA *info,const char *record, uchar *rec_buff, else to+= length; } - if (packed_length != (uint) (to - (char*) rec_buff) + + if (packed_length != (uint) (to - rec_buff) + test(info->s->calc_checksum) || (bit != 1 && (flag & ~(bit - 1)))) goto err; if (with_checksum && ((uchar) checksum != (uchar) *to)) diff --git a/storage/maria/ma_extra.c b/storage/maria/ma_extra.c index 2fc4873d535..9ee3b1a8870 100644 --- a/storage/maria/ma_extra.c +++ b/storage/maria/ma_extra.c @@ -379,11 +379,13 @@ int maria_extra(MARIA_HA *info, enum ha_extra_function function, break; /* Not supported */ pthread_mutex_lock(&share->intern_lock); /* - Memory map the data file if it is not already mapped and if there - are no other threads using this table. intern_lock prevents other - threads from starting to use the table while we are mapping it. + Memory map the data file if it is not already mapped. It is safe + to memory map a file while other threads are using file I/O on it. + Assigning a new address to a function pointer is an atomic + operation. intern_lock prevents that two or more mappings are done + at the same time. */ - if (!share->file_map && (share->tot_locks == 1)) + if (!share->file_map) { if (_ma_dynmap_file(info, share->state.state.data_file_length)) { diff --git a/storage/maria/ma_ft_boolean_search.c b/storage/maria/ma_ft_boolean_search.c index 41661d1c288..e09a076ceaa 100644 --- a/storage/maria/ma_ft_boolean_search.c +++ b/storage/maria/ma_ft_boolean_search.c @@ -286,8 +286,8 @@ static int ftb_parse_query_internal(MYSQL_FTPARSER_PARAM *param, } -static void _ftb_parse_query(FTB *ftb, uchar *query, uint len, - struct st_mysql_ftparser *parser) +static int _ftb_parse_query(FTB *ftb, uchar *query, uint len, + struct st_mysql_ftparser *parser) { MYSQL_FTPARSER_PARAM *param; MY_FTB_PARAM ftb_param; @@ -295,9 +295,9 @@ static void _ftb_parse_query(FTB *ftb, uchar *query, uint len, DBUG_ASSERT(parser); if (ftb->state != UNINITIALIZED) - DBUG_VOID_RETURN; + DBUG_RETURN(0); if (! (param= maria_ftparser_call_initializer(ftb->info, ftb->keynr, 0))) - DBUG_VOID_RETURN; + DBUG_RETURN(1); ftb_param.ftb= ftb; ftb_param.depth= 0; @@ -312,8 +312,7 @@ static void _ftb_parse_query(FTB *ftb, uchar *query, uint len, param->length= len; param->flags= 0; param->mode= MYSQL_FTPARSER_FULL_BOOLEAN_INFO; - parser->parse(param); - DBUG_VOID_RETURN; + DBUG_RETURN(parser->parse(param)); } @@ -537,9 +536,10 @@ FT_INFO * maria_ft_init_boolean_search(MARIA_HA *info, uint keynr, uchar *query, ftbe->phrase= NULL; ftbe->document= 0; ftb->root=ftbe; - _ftb_parse_query(ftb, query, query_len, keynr == NO_SUCH_KEY ? - &ft_default_parser : - info->s->keyinfo[keynr].parser); + if (unlikely(_ftb_parse_query(ftb, query, query_len, + keynr == NO_SUCH_KEY ? &ft_default_parser : + info->s->keyinfo[keynr].parser))) + goto err; /* Hack: instead of init_queue, we'll use reinit queue to be able to alloc queue with alloc_root() @@ -621,7 +621,7 @@ static int ftb_check_phrase_internal(MYSQL_FTPARSER_PARAM *param, { param->mysql_add_word(param, word.pos, word.len, 0); if (phrase_param->match) - return 1; + break; } return 0; } @@ -639,6 +639,7 @@ static int ftb_check_phrase_internal(MYSQL_FTPARSER_PARAM *param, RETURN VALUE 1 is returned if phrase found, 0 else. + -1 is returned if error occurs. */ static int _ftb_check_phrase(FTB *ftb, const uchar *document, uint len, @@ -666,12 +667,13 @@ static int _ftb_check_phrase(FTB *ftb, const uchar *document, uint len, param->length= len; param->flags= 0; param->mode= MYSQL_FTPARSER_WITH_STOPWORDS; - parser->parse(param); + if (unlikely(parser->parse(param))) + return -1; DBUG_RETURN(ftb_param.match ? 1 : 0); } -static void _ftb_climb_the_tree(FTB *ftb, FTB_WORD *ftbw, FT_SEG_ITERATOR *ftsi_orig) +static int _ftb_climb_the_tree(FTB *ftb, FTB_WORD *ftbw, FT_SEG_ITERATOR *ftsi_orig) { FT_SEG_ITERATOR ftsi; FTB_EXPR *ftbe; @@ -703,17 +705,19 @@ static void _ftb_climb_the_tree(FTB *ftb, FTB_WORD *ftbw, FT_SEG_ITERATOR *ftsi_ weight=ftbe->cur_weight*ftbe->weight; if (mode && ftbe->phrase) { - int not_found=1; + int found= 0; memcpy(&ftsi, ftsi_orig, sizeof(ftsi)); - while (_ma_ft_segiterator(&ftsi) && not_found) + while (_ma_ft_segiterator(&ftsi) && !found) { if (!ftsi.pos) continue; - not_found = ! _ftb_check_phrase(ftb, ftsi.pos, ftsi.len, - ftbe, parser); + found= _ftb_check_phrase(ftb, ftsi.pos, ftsi.len, ftbe, parser); + if (unlikely(found < 0)) + return 1; } - if (not_found) break; + if (!found) + break; } /* ftbe->quot */ } else @@ -745,6 +749,7 @@ static void _ftb_climb_the_tree(FTB *ftb, FTB_WORD *ftbw, FT_SEG_ITERATOR *ftsi_ weight*= ftbe->weight; } } + return 0; } @@ -777,7 +782,11 @@ int maria_ft_boolean_read_next(FT_INFO *ftb, char *record) { while (curdoc == (ftbw=(FTB_WORD *)queue_top(& ftb->queue))->docid[0]) { - _ftb_climb_the_tree(ftb, ftbw, 0); + if (unlikely(_ftb_climb_the_tree(ftb, ftbw, 0))) + { + my_errno= HA_ERR_OUT_OF_MEM; + goto err; + } /* update queue */ _ft2_search(ftb, ftbw, 0); @@ -853,7 +862,8 @@ static int ftb_find_relevance_add_word(MYSQL_FTPARSER_PARAM *param, if (ftbw->docid[1] == ftb->info->cur_row.lastpos) continue; ftbw->docid[1]= ftb->info->cur_row.lastpos; - _ftb_climb_the_tree(ftb, ftbw, ftb_param->ftsi); + if (unlikely(_ftb_climb_the_tree(ftb, ftbw, ftb_param->ftsi))) + return 1; } return(0); } @@ -926,7 +936,8 @@ float maria_ft_boolean_find_relevance(FT_INFO *ftb, uchar *record, uint length) continue; param->doc= (uchar *)ftsi.pos; param->length= ftsi.len; - parser->parse(param); + if (unlikely(parser->parse(param))) + return 0; } ftbe=ftb->root; if (ftbe->docid[1]==docid && ftbe->cur_weight>0 && diff --git a/storage/maria/ma_ft_nlq_search.c b/storage/maria/ma_ft_nlq_search.c index cad5238d4a5..18b101f0e05 100644 --- a/storage/maria/ma_ft_nlq_search.c +++ b/storage/maria/ma_ft_nlq_search.c @@ -258,8 +258,12 @@ FT_INFO *maria_ft_init_nlq_search(MARIA_HA *info, uint keynr, uchar *query, { info->update|= HA_STATE_AKTIV; ftparser_param->flags= MYSQL_FTFLAGS_NEED_COPY; - _ma_ft_parse(&wtree, info, keynr, record, ftparser_param, - &wtree.mem_root); + if (unlikely(_ma_ft_parse(&wtree, info, keynr, record, ftparser_param, + &wtree.mem_root))) + { + delete_queue(&best); + goto err; + } } } delete_queue(&best); diff --git a/storage/maria/ma_info.c b/storage/maria/ma_info.c index cfb4580a72f..4aecc33f816 100644 --- a/storage/maria/ma_info.c +++ b/storage/maria/ma_info.c @@ -57,9 +57,9 @@ int maria_status(MARIA_HA *info, register MARIA_INFO *x, uint flag) x->keys = share->state.header.keys; x->check_time = share->state.check_time; - x->mean_reclength = info->state->records ? - (ulong) ((info->state->data_file_length-info->state->empty)/ - info->state->records) : (ulong) share->min_pack_length; + x->mean_reclength = x->records ? + (ulong) ((x->data_file_length - x->delete_length) /x ->records) : + (ulong) share->min_pack_length; } if (flag & HA_STATUS_ERRKEY) { diff --git a/storage/maria/ma_key.c b/storage/maria/ma_key.c index 83ba6853330..96b8d2af0eb 100644 --- a/storage/maria/ma_key.c +++ b/storage/maria/ma_key.c @@ -206,7 +206,7 @@ uint _ma_make_key(register MARIA_HA *info, uint keynr, uchar *key, uint keynr key number key Store packed key here old Not packed key - k_length Length of 'old' to use + keypart_map bitmap of used keyparts last_used_keyseg out parameter. May be NULL RETURN @@ -216,34 +216,37 @@ uint _ma_make_key(register MARIA_HA *info, uint keynr, uchar *key, */ uint _ma_pack_key(register MARIA_HA *info, uint keynr, uchar *key, - const uchar *old, uint k_length, HA_KEYSEG **last_used_keyseg) + const uchar *old, key_part_map keypart_map, + HA_KEYSEG **last_used_keyseg) { uchar *start_key=key; HA_KEYSEG *keyseg; my_bool is_ft= info->s->keyinfo[keynr].flag & HA_FULLTEXT; DBUG_ENTER("_ma_pack_key"); - for (keyseg=info->s->keyinfo[keynr].seg ; - keyseg->type && (int) k_length > 0; - old+=keyseg->length, keyseg++) + /* "one part" rtree key is 2*SPDIMS part key in Maria */ + if (info->s->keyinfo[keynr].key_alg == HA_KEY_ALG_RTREE) + keypart_map= (((key_part_map)1) << (2*SPDIMS)) - 1; + + /* only key prefixes are supported */ + DBUG_ASSERT(((keypart_map+1) & keypart_map) == 0); + + for (keyseg=info->s->keyinfo[keynr].seg ; keyseg->type && keypart_map; + old+= keyseg->length, keyseg++) { - enum ha_base_keytype type=(enum ha_base_keytype) keyseg->type; - uint length=min((uint) keyseg->length,(uint) k_length); + enum ha_base_keytype type= (enum ha_base_keytype) keyseg->type; + uint length= keyseg->length; uint char_length; const uchar *pos; CHARSET_INFO *cs=keyseg->charset; + keypart_map>>= 1; if (keyseg->null_bit) { - k_length--; if (!(*key++= (char) 1-*old++)) /* Copy null marker */ { - k_length-=length; if (keyseg->flag & (HA_VAR_LENGTH_PART | HA_BLOB_PART)) - { - k_length-=2; /* Skip length */ old+= 2; - } continue; /* Found NULL */ } } @@ -253,17 +256,16 @@ uint _ma_pack_key(register MARIA_HA *info, uint keynr, uchar *key, if (keyseg->flag & HA_SPACE_PACK) { const uchar *end= pos + length; - if (type != HA_KEYTYPE_NUM) - { - while (end > pos && end[-1] == ' ') - end--; - } - else + if (type == HA_KEYTYPE_NUM) { while (pos < end && pos[0] == ' ') pos++; } - k_length-=length; + else if (type != HA_KEYTYPE_BINARY) + { + while (end > pos && end[-1] == ' ') + end--; + } length=(uint) (end-pos); FIX_LENGTH(cs, pos, length, char_length); store_key_length_inc(key,char_length); @@ -275,7 +277,6 @@ uint _ma_pack_key(register MARIA_HA *info, uint keynr, uchar *key, { /* Length of key-part used with maria_rkey() always 2 */ uint tmp_length=uint2korr(pos); - k_length-= 2+length; pos+=2; set_if_smaller(length,tmp_length); /* Safety */ FIX_LENGTH(cs, pos, length, char_length); @@ -288,11 +289,8 @@ uint _ma_pack_key(register MARIA_HA *info, uint keynr, uchar *key, else if (keyseg->flag & HA_SWAP_KEY) { /* Numerical column */ pos+=length; - k_length-=length; while (length--) - { *key++ = *--pos; - } continue; } FIX_LENGTH(cs, pos, length, char_length); @@ -300,30 +298,10 @@ uint _ma_pack_key(register MARIA_HA *info, uint keynr, uchar *key, if (length > char_length) cs->cset->fill(cs, (char*) key+char_length, length-char_length, ' '); key+= length; - k_length-=length; } if (last_used_keyseg) *last_used_keyseg= keyseg; -#ifdef NOT_USED - if (keyseg->type) - { - /* Part-key ; fill with ASCII 0 for easier searching */ - length= (uint) -k_length; /* unused part of last key */ - do - { - if (keyseg->flag & HA_NULL_PART) - length++; - if (keyseg->flag & HA_SPACE_PACK) - length+=2; - else - length+= keyseg->length; - keyseg++; - } while (keyseg->type); - bzero(key,length); - key+=length; - } -#endif DBUG_PRINT("exit", ("length: %u", (uint) (key-start_key))); DBUG_RETURN((uint) (key-start_key)); } /* _ma_pack_key */ diff --git a/storage/maria/ma_loghandler.c b/storage/maria/ma_loghandler.c index fd007e2a5ae..9d1514a67c3 100644 --- a/storage/maria/ma_loghandler.c +++ b/storage/maria/ma_loghandler.c @@ -3553,8 +3553,8 @@ static my_bool translog_relative_LSN_encode(struct st_translog_parts *parts, COMPRESSED_LSN_MAX_STORE_SIZE)) - dst_ptr); parts->record_length-= (economy= lsns_len - part->length); - DBUG_PRINT("info", ("new length of LSNs: %u economy: %d", - part->length, economy)); + DBUG_PRINT("info", ("new length of LSNs: %lu economy: %d", + (ulong)part->length, economy)); parts->total_record_length-= economy; part->str= (char*)dst_ptr; } diff --git a/storage/maria/ma_open.c b/storage/maria/ma_open.c index e6df213609b..67a76144c26 100644 --- a/storage/maria/ma_open.c +++ b/storage/maria/ma_open.c @@ -392,7 +392,7 @@ MARIA_HA *maria_open(const char *name, int mode, uint open_flags) key_parts+=fulltext_keys*FT_SEGS; if (share->base.max_key_length > maria_max_key_length() || - keys > MARIA_MAX_KEY || key_parts >= MARIA_MAX_KEY * HA_MAX_KEY_SEG) + keys > MARIA_MAX_KEY || key_parts > MARIA_MAX_KEY * HA_MAX_KEY_SEG) { DBUG_PRINT("error",("Wrong key info: Max_key_length: %d keys: %d key_parts: %d", share->base.max_key_length, keys, key_parts)); my_errno=HA_ERR_UNSUPPORTED; @@ -667,22 +667,6 @@ MARIA_HA *maria_open(const char *name, int mode, uint open_flags) _ma_setup_functions(share); if ((*share->once_init)(share, info.dfile.file)) goto err; - if (open_flags & HA_OPEN_MMAP) - { - info.s= share; - if (_ma_dynmap_file(&info, share->state.state.data_file_length)) - { - /* purecov: begin inspected */ - /* Ignore if mmap fails. Use file I/O instead. */ - DBUG_PRINT("warning", ("mmap failed: errno: %d", errno)); - /* purecov: end */ - } - else - { - share->file_read= _ma_mmap_pread; - share->file_write= _ma_mmap_pwrite; - } - } share->is_log_table= FALSE; if (open_flags & HA_OPEN_TMP_TABLE) share->options|= HA_OPTION_TMP_TABLE; @@ -721,6 +705,14 @@ MARIA_HA *maria_open(const char *name, int mode, uint open_flags) } } #endif + /* + Memory mapping can only be requested after initializing intern_lock. + */ + if (open_flags & HA_OPEN_MMAP) + { + info.s= share; + maria_extra(&info, HA_EXTRA_MMAP, 0); + } } else { @@ -1022,10 +1014,10 @@ uint _ma_state_info_write(File file, MARIA_STATE_INFO *state, uint pWrite) } if (pWrite & 1) - DBUG_RETURN(my_pwrite(file,(char*) buff, (uint) (ptr-buff), 0L, - MYF(MY_NABP | MY_THREADSAFE))); - DBUG_RETURN(my_write(file, (char*) buff, (uint) (ptr-buff), - MYF(MY_NABP))); + DBUG_RETURN(my_pwrite(file, buff, (size_t) (ptr-buff), 0L, + MYF(MY_NABP | MY_THREADSAFE)) != 0); + DBUG_RETURN(my_write(file, buff, (size_t) (ptr-buff), + MYF(MY_NABP)) != 0); } @@ -1089,10 +1081,10 @@ uint _ma_state_info_read_dsk(File file, MARIA_STATE_INFO *state, my_bool pRead) if (pRead) { if (my_pread(file, buff, state->state_length,0L, MYF(MY_NABP))) - return (MY_FILE_ERROR); + return 1; } else if (my_read(file, buff, state->state_length,MYF(MY_NABP))) - return (MY_FILE_ERROR); + return 1; _ma_state_info_read(buff, state); } return 0; @@ -1143,7 +1135,7 @@ uint _ma_base_info_write(File file, MARIA_BASE_INFO *base) *ptr++= base->extra_alloc_procent; bzero(ptr,16); ptr+= 16; /* extra */ DBUG_ASSERT((ptr - buff) == MARIA_BASE_INFO_SIZE); - return my_write(file,(char*) buff, (uint) (ptr-buff), MYF(MY_NABP)); + return my_write(file, buff, (size_t) (ptr-buff), MYF(MY_NABP)) != 0; } @@ -1204,7 +1196,7 @@ uint _ma_keydef_write(File file, MARIA_KEYDEF *keydef) mi_int2store(ptr,keydef->keylength); ptr+= 2; mi_int2store(ptr,keydef->minlength); ptr+= 2; mi_int2store(ptr,keydef->maxlength); ptr+= 2; - return my_write(file,(char*) buff, (uint) (ptr-buff), MYF(MY_NABP)); + return my_write(file, buff, (size_t) (ptr-buff), MYF(MY_NABP)) != 0; } char *_ma_keydef_read(char *ptr, MARIA_KEYDEF *keydef) @@ -1247,7 +1239,7 @@ int _ma_keyseg_write(File file, const HA_KEYSEG *keyseg) mi_int4store(ptr, pos); ptr+=4; - return my_write(file,(char*) buff, (uint) (ptr-buff), MYF(MY_NABP)); + return my_write(file, buff, (size_t) (ptr-buff), MYF(MY_NABP)) != 0; } @@ -1287,7 +1279,7 @@ uint _ma_uniquedef_write(File file, MARIA_UNIQUEDEF *def) *ptr++= (uchar) def->key; *ptr++ = (uchar) def->null_are_equal; - return my_write(file,(char*) buff, (uint) (ptr-buff), MYF(MY_NABP)); + return my_write(file, buff, (size_t) (ptr-buff), MYF(MY_NABP)) != 0; } char *_ma_uniquedef_read(char *ptr, MARIA_UNIQUEDEF *def) @@ -1315,7 +1307,7 @@ uint _ma_columndef_write(File file, MARIA_COLUMNDEF *columndef) mi_int2store(ptr,columndef->empty_pos); ptr+= 2; (*ptr++)= columndef->null_bit; (*ptr++)= columndef->empty_bit; - return my_write(file,(char*) buff, (uint) (ptr-buff), MYF(MY_NABP)); + return my_write(file, buff, (size_t) (ptr-buff), MYF(MY_NABP)) != 0; } char *_ma_columndef_read(char *ptr, MARIA_COLUMNDEF *columndef) diff --git a/storage/maria/ma_preload.c b/storage/maria/ma_preload.c index 35ae8868ee7..138bb94f7d0 100644 --- a/storage/maria/ma_preload.c +++ b/storage/maria/ma_preload.c @@ -55,12 +55,17 @@ int maria_preload(MARIA_HA *info, ulonglong key_map, my_bool ignore_leaves) block_length= keyinfo[0].block_length; - /* Check whether all indexes use the same block size */ - for (i= 1 ; i < keys ; i++) + if (ignore_leaves) { - if (keyinfo[i].block_length != block_length) - DBUG_RETURN(my_errno= HA_ERR_NON_UNIQUE_BLOCK_SIZE); + /* Check whether all indexes use the same block size */ + for (i= 1 ; i < keys ; i++) + { + if (keyinfo[i].block_length != block_length) + DBUG_RETURN(my_errno= HA_ERR_NON_UNIQUE_BLOCK_SIZE); + } } + else + block_length= share->pagecache->block_size; length= info->preload_buff_size/block_length * block_length; set_if_bigger(length, block_length); diff --git a/storage/maria/ma_range.c b/storage/maria/ma_range.c index 7250e6796d2..70dd522b8a1 100644 --- a/storage/maria/ma_range.c +++ b/storage/maria/ma_range.c @@ -21,12 +21,11 @@ #include "maria_def.h" #include "ma_rt_index.h" -static ha_rows _ma_record_pos(MARIA_HA *info,const uchar *key,uint key_len, - enum ha_rkey_function search_flag); -static double _ma_search_pos(MARIA_HA *info,MARIA_KEYDEF *keyinfo, uchar *key, - uint key_len,uint nextflag, my_off_t pos); -static uint _ma_keynr(MARIA_HA *info, MARIA_KEYDEF *keyinfo, uchar *page, - uchar *keypos, uint *ret_max_key); +static ha_rows _ma_record_pos(MARIA_HA *,const uchar *, key_part_map, + enum ha_rkey_function); +static double _ma_search_pos(MARIA_HA *, MARIA_KEYDEF *, uchar *, + uint, uint, my_off_t); +static uint _ma_keynr(MARIA_HA *, MARIA_KEYDEF *, uchar *, uchar *, uint *); /** @@ -84,7 +83,7 @@ ha_rows maria_records_in_range(MARIA_HA *info, int inx, key_range *min_key, } key_buff= info->lastkey+info->s->base.max_key_length; start_key_len= _ma_pack_key(info,inx, key_buff, - min_key->key, min_key->length, + min_key->key, min_key->keypart_map, (HA_KEYSEG**) 0); res= maria_rtree_estimate(info, inx, key_buff, start_key_len, maria_read_vec[min_key->flag]); @@ -95,13 +94,13 @@ ha_rows maria_records_in_range(MARIA_HA *info, int inx, key_range *min_key, case HA_KEY_ALG_BTREE: default: start_pos= (min_key ? - _ma_record_pos(info, min_key->key, min_key->length, + _ma_record_pos(info, min_key->key, min_key->keypart_map, min_key->flag) : (ha_rows) 0); end_pos= (max_key ? - _ma_record_pos(info, max_key->key, max_key->length, + _ma_record_pos(info, max_key->key, max_key->keypart_map, max_key->flag) : - info->state->records+ (ha_rows) 1); + info->state->records + (ha_rows) 1); res= (end_pos < start_pos ? (ha_rows) 0 : (end_pos == start_pos ? (ha_rows) 1 : end_pos-start_pos)); if (start_pos == HA_POS_ERROR || end_pos == HA_POS_ERROR) @@ -126,20 +125,22 @@ ha_rows maria_records_in_range(MARIA_HA *info, int inx, key_range *min_key, /* Find relative position (in records) for key in index-tree */ -static ha_rows _ma_record_pos(MARIA_HA *info, const uchar *key, uint key_len, +static ha_rows _ma_record_pos(MARIA_HA *info, const uchar *key, + key_part_map keypart_map, enum ha_rkey_function search_flag) { - uint inx=(uint) info->lastinx, nextflag; + uint inx=(uint) info->lastinx, nextflag, key_len; MARIA_KEYDEF *keyinfo=info->s->keyinfo+inx; uchar *key_buff; double pos; DBUG_ENTER("_ma_record_pos"); DBUG_PRINT("enter",("search_flag: %d",search_flag)); + DBUG_ASSERT(keypart_map); if (key_len == 0) key_len= USE_WHOLE_KEY; key_buff=info->lastkey+info->s->base.max_key_length; - key_len= _ma_pack_key(info, inx, key_buff, key, key_len, + key_len= _ma_pack_key(info, inx, key_buff, key, keypart_map, (HA_KEYSEG**) 0); DBUG_EXECUTE("key", _ma_print_key(DBUG_FILE, keyinfo->seg, key_buff, key_len);); @@ -147,8 +148,42 @@ static ha_rows _ma_record_pos(MARIA_HA *info, const uchar *key, uint key_len, if (!(nextflag & (SEARCH_FIND | SEARCH_NO_FIND | SEARCH_LAST))) key_len=USE_WHOLE_KEY; + /* + my_handler.c:mi_compare_text() has a flag 'skip_end_space'. + This is set in my_handler.c:ha_key_cmp() in dependence on the + compare flags 'nextflag' and the column type. + + TEXT columns are of type HA_KEYTYPE_VARTEXT. In this case the + condition is skip_end_space= ((nextflag & (SEARCH_FIND | + SEARCH_UPDATE)) == SEARCH_FIND). + + SEARCH_FIND is used for an exact key search. The combination + SEARCH_FIND | SEARCH_UPDATE is used in write/update/delete + operations with a comment like "Not real duplicates", whatever this + means. From the condition above we can see that 'skip_end_space' is + always false for these operations. The result is that trailing space + counts in key comparison and hence, emtpy strings ('', string length + zero, but not NULL) compare less that strings starting with control + characters and these in turn compare less than strings starting with + blanks. + + When estimating the number of records in a key range, we request an + exact search for the minimum key. This translates into a plain + SEARCH_FIND flag. Using this alone would lead to a 'skip_end_space' + compare. Empty strings would be expected above control characters. + Their keys would not be found because they are located below control + characters. + + This is the reason that we add the SEARCH_UPDATE flag here. It makes + the key estimation compare in the same way like key write operations + do. Olny so we will find the keys where they have been inserted. + + Adding the flag unconditionally does not hurt as it is used in the + above mentioned condition only. So it can safely be used together + with other flags. + */ pos= _ma_search_pos(info,keyinfo, key_buff, key_len, - nextflag | SEARCH_SAVE_BUFF, + nextflag | SEARCH_SAVE_BUFF | SEARCH_UPDATE, info->s->state.key_root[inx]); if (pos >= 0.0) { diff --git a/storage/maria/ma_rkey.c b/storage/maria/ma_rkey.c index ef8b8468f1f..c9653d30110 100644 --- a/storage/maria/ma_rkey.c +++ b/storage/maria/ma_rkey.c @@ -22,7 +22,7 @@ /* Ordinary search_flag is 0 ; Give error if no record with key */ int maria_rkey(MARIA_HA *info, uchar *buf, int inx, const uchar *key, - uint key_len, enum ha_rkey_function search_flag) + key_part_map keypart_map, enum ha_rkey_function search_flag) { uchar *key_buff; MARIA_SHARE *share=info->s; @@ -47,20 +47,21 @@ int maria_rkey(MARIA_HA *info, uchar *buf, int inx, const uchar *key, key is already packed!; This happens when we are using a MERGE TABLE */ key_buff= info->lastkey+info->s->base.max_key_length; - pack_key_length= key_len; - bmove(key_buff,key,key_len); - last_used_keyseg= 0; + pack_key_length= keypart_map; + bmove(key_buff, key, pack_key_length); + last_used_keyseg= info->s->keyinfo[inx].seg + info->last_used_keyseg; } else { - if (key_len == 0) - key_len=USE_WHOLE_KEY; + DBUG_ASSERT(keypart_map); /* Save the packed key for later use in the second buffer of lastkey. */ key_buff=info->lastkey+info->s->base.max_key_length; pack_key_length= _ma_pack_key(info,(uint) inx, key_buff, key, - key_len, &last_used_keyseg); + keypart_map, &last_used_keyseg); /* Save packed_key_length for use by the MERGE engine. */ info->pack_key_length= pack_key_length; + info->last_used_keyseg= (uint16) (last_used_keyseg - + info->s->keyinfo[inx].seg); DBUG_EXECUTE("key", _ma_print_key(DBUG_FILE, keyinfo->seg, key_buff, pack_key_length);); } diff --git a/storage/maria/ma_rt_index.c b/storage/maria/ma_rt_index.c index 4d99eade9b5..4980233fc11 100644 --- a/storage/maria/ma_rt_index.c +++ b/storage/maria/ma_rt_index.c @@ -187,6 +187,7 @@ int maria_rtree_find_first(MARIA_HA *info, uint keynr, uchar *key, /* Save searched key, include data pointer. The data pointer is required if the search_flag contains MBR_DATA. + (minimum bounding rectangle) */ memcpy(info->first_mbr_key, key, keyinfo->keylength); info->last_rkey_length = key_length; @@ -546,16 +547,19 @@ static int maria_rtree_insert_req(MARIA_HA *info, MARIA_KEYDEF *keyinfo, uint nod_flag; int res; uchar *page_buf, *k; + DBUG_ENTER("maria_rtree_insert_req"); if (!(page_buf= (uchar*) my_alloca((uint)keyinfo->block_length + HA_MAX_KEY_BUFF))) { my_errno = HA_ERR_OUT_OF_MEM; - return -1; + DBUG_RETURN(-1); /* purecov: inspected */ } if (!_ma_fetch_keypage(info, keyinfo, page, DFLT_INIT_HITS, page_buf, 0)) goto err1; nod_flag = _ma_test_if_nod(page_buf); + DBUG_PRINT("rtree", ("page: %lu level: %d ins_level: %d nod_flag: %u", + (ulong) page, level, ins_level, nod_flag)); if ((ins_level == -1 && nod_flag) || /* key: go down to leaf */ (ins_level > -1 && ins_level > level)) /* branch: go down to ins_level */ @@ -609,11 +613,11 @@ static int maria_rtree_insert_req(MARIA_HA *info, MARIA_KEYDEF *keyinfo, ok: my_afree(page_buf); - return res; + DBUG_RETURN(res); err1: my_afree(page_buf); - return -1; + DBUG_RETURN(-1); /* purecov: inspected */ } @@ -633,18 +637,19 @@ static int maria_rtree_insert_level(MARIA_HA *info, uint keynr, uchar *key, MARIA_KEYDEF *keyinfo = info->s->keyinfo + keynr; int res; my_off_t new_page; + DBUG_ENTER("maria_rtree_insert_level"); if ((old_root = info->s->state.key_root[keynr]) == HA_OFFSET_ERROR) { if ((old_root = _ma_new(info, keyinfo, DFLT_INIT_HITS)) == HA_OFFSET_ERROR) - return -1; + DBUG_RETURN(-1); info->keyread_buff_used = 1; maria_putint(info->buff, 2, 0); res = maria_rtree_add_key(info, keyinfo, key, key_length, info->buff, NULL); if (_ma_write_keypage(info, keyinfo, old_root, DFLT_INIT_HITS, info->buff)) - return 1; + DBUG_RETURN(1); info->s->state.key_root[keynr] = old_root; - return res; + DBUG_RETURN(res); } switch ((res = maria_rtree_insert_req(info, keyinfo, key, key_length, @@ -660,11 +665,12 @@ static int maria_rtree_insert_level(MARIA_HA *info, uint keynr, uchar *key, my_off_t new_root; uint nod_flag = info->s->base.key_reflength; + DBUG_PRINT("rtree", ("root was split, grow a new root")); if (!(new_root_buf= (uchar*) my_alloca((uint)keyinfo->block_length + HA_MAX_KEY_BUFF))) { my_errno = HA_ERR_OUT_OF_MEM; - return -1; + DBUG_RETURN(-1); /* purecov: inspected */ } maria_putint(new_root_buf, 2, nod_flag); @@ -694,12 +700,15 @@ static int maria_rtree_insert_level(MARIA_HA *info, uint keynr, uchar *key, DFLT_INIT_HITS, new_root_buf)) goto err1; info->s->state.key_root[keynr] = new_root; + DBUG_PRINT("rtree", ("new root page: %lu level: %d nod_flag: %u", + (ulong) new_root, 0, + _ma_test_if_nod(new_root_buf))); my_afree((uchar*)new_root_buf); break; err1: my_afree((uchar*)new_root_buf); - return -1; + DBUG_RETURN(-1); /* purecov: inspected */ } default: case -1: /* error */ @@ -707,7 +716,7 @@ err1: break; } } - return res; + DBUG_RETURN(res); } @@ -721,9 +730,10 @@ err1: int maria_rtree_insert(MARIA_HA *info, uint keynr, uchar *key, uint key_length) { - return (!key_length || - (maria_rtree_insert_level(info, keynr, key, key_length, -1) == -1)) ? - -1 : 0; + DBUG_ENTER("maria_rtree_insert"); + DBUG_RETURN((!key_length || + (maria_rtree_insert_level(info, keynr, key, key_length, -1) == -1)) ? + -1 : 0); } @@ -738,6 +748,8 @@ int maria_rtree_insert(MARIA_HA *info, uint keynr, uchar *key, uint key_length) static int maria_rtree_fill_reinsert_list(stPageList *ReinsertList, my_off_t page, int level) { + DBUG_ENTER("maria_rtree_fill_reinsert_list"); + DBUG_PRINT("rtree", ("page: %lu level: %d", (ulong) page, level)); if (ReinsertList->n_pages == ReinsertList->m_pages) { ReinsertList->m_pages += REINSERT_BUFFER_INC; @@ -749,10 +761,10 @@ static int maria_rtree_fill_reinsert_list(stPageList *ReinsertList, my_off_t pag ReinsertList->pages[ReinsertList->n_pages].offs = page; ReinsertList->pages[ReinsertList->n_pages].level = level; ReinsertList->n_pages++; - return 0; + DBUG_RETURN(0); err1: - return -1; + DBUG_RETURN(-1); /* purecov: inspected */ } @@ -776,15 +788,18 @@ static int maria_rtree_delete_req(MARIA_HA *info, MARIA_KEYDEF *keyinfo, uint nod_flag; int res; uchar *page_buf, *last, *k; + DBUG_ENTER("maria_rtree_delete_req"); if (!(page_buf = (uchar*) my_alloca((uint)keyinfo->block_length))) { my_errno = HA_ERR_OUT_OF_MEM; - return -1; + DBUG_RETURN(-1); /* purecov: inspected */ } if (!_ma_fetch_keypage(info, keyinfo, page, DFLT_INIT_HITS, page_buf, 0)) goto err1; nod_flag = _ma_test_if_nod(page_buf); + DBUG_PRINT("rtree", ("page: %lu level: %d nod_flag: %u", + (ulong) page, level, nod_flag)); k = rt_PAGE_FIRST_KEY(page_buf, nod_flag); last = rt_PAGE_END(page_buf); @@ -806,6 +821,7 @@ static int maria_rtree_delete_req(MARIA_HA *info, MARIA_KEYDEF *keyinfo, rt_PAGE_MIN_SIZE(keyinfo->block_length)) { /* OK */ + /* Calculate a new key value (MBR) for the shrinked block. */ if (maria_rtree_set_key_mbr(info, keyinfo, k, key_length, _ma_kpos(nod_flag, k))) goto err1; @@ -815,11 +831,24 @@ static int maria_rtree_delete_req(MARIA_HA *info, MARIA_KEYDEF *keyinfo, } else { - /* too small: delete key & add it descendant to reinsert list */ + /* + Too small: delete key & add it descendant to reinsert list. + Store position and level of the block so that it can be + accessed later for inserting the remaining keys. + */ + DBUG_PRINT("rtree", ("too small. move block to reinsert list")); if (maria_rtree_fill_reinsert_list(ReinsertList, _ma_kpos(nod_flag, k), level + 1)) goto err1; + /* + Delete the key that references the block. This makes the + block disappear from the index. Hence we need to insert + its remaining keys later. Note: if the block is a branch + block, we do not only remove this block, but the whole + subtree. So we need to re-insert its keys on the same + level later to reintegrate the subtrees. + */ maria_rtree_delete_key(info, page_buf, k, key_length, nod_flag); if (_ma_write_keypage(info, keyinfo, page, DFLT_INIT_HITS, page_buf)) @@ -879,11 +908,11 @@ static int maria_rtree_delete_req(MARIA_HA *info, MARIA_KEYDEF *keyinfo, ok: my_afree((uchar*)page_buf); - return res; + DBUG_RETURN(res); err1: my_afree((uchar*)page_buf); - return -1; + DBUG_RETURN(-1); /* purecov: inspected */ } @@ -901,12 +930,15 @@ int maria_rtree_delete(MARIA_HA *info, uint keynr, uchar *key, uint key_length) stPageList ReinsertList; my_off_t old_root; MARIA_KEYDEF *keyinfo = info->s->keyinfo + keynr; + DBUG_ENTER("maria_rtree_delete"); if ((old_root = info->s->state.key_root[keynr]) == HA_OFFSET_ERROR) { my_errno= HA_ERR_END_OF_FILE; - return -1; + DBUG_RETURN(-1); /* purecov: inspected */ } + DBUG_PRINT("rtree", ("starting deletion at root page: %lu", + (ulong) old_root)); ReinsertList.pages = NULL; ReinsertList.n_pages = 0; @@ -915,12 +947,12 @@ int maria_rtree_delete(MARIA_HA *info, uint keynr, uchar *key, uint key_length) switch (maria_rtree_delete_req(info, keyinfo, key, key_length, old_root, &page_size, &ReinsertList, 0)) { - case 2: + case 2: /* empty */ { info->s->state.key_root[keynr] = HA_OFFSET_ERROR; - return 0; + DBUG_RETURN(0); } - case 0: + case 0: /* deleted */ { uint nod_flag; ulong i; @@ -937,16 +969,35 @@ int maria_rtree_delete(MARIA_HA *info, uint keynr, uchar *key, uint key_length) DFLT_INIT_HITS, page_buf, 0)) goto err1; nod_flag = _ma_test_if_nod(page_buf); + DBUG_PRINT("rtree", ("reinserting keys from " + "page: %lu level: %d nod_flag: %u", + (ulong) ReinsertList.pages[i].offs, + ReinsertList.pages[i].level, nod_flag)); + k = rt_PAGE_FIRST_KEY(page_buf, nod_flag); last = rt_PAGE_END(page_buf); for (; k < last; k = rt_PAGE_NEXT_KEY(k, key_length, nod_flag)) { - if (maria_rtree_insert_level(info, keynr, k, key_length, - ReinsertList.pages[i].level) == -1) + int res; + if ((res= + maria_rtree_insert_level(info, keynr, k, key_length, + ReinsertList.pages[i].level)) == -1) { my_afree(page_buf); goto err1; } + if (res) + { + ulong j; + DBUG_PRINT("rtree", ("root has been split, adjust levels")); + for (j= i; j < ReinsertList.n_pages; j++) + { + ReinsertList.pages[j].level++; + DBUG_PRINT("rtree", ("keys from page: %lu now level: %d", + (ulong) ReinsertList.pages[i].offs, + ReinsertList.pages[i].level)); + } + } } my_afree(page_buf); if (_ma_dispose(info, keyinfo, ReinsertList.pages[i].offs, @@ -973,19 +1024,19 @@ int maria_rtree_delete(MARIA_HA *info, uint keynr, uchar *key, uint key_length) info->s->state.key_root[keynr] = new_root; } info->update= HA_STATE_DELETED; - return 0; + DBUG_RETURN(0); err1: - return -1; + DBUG_RETURN(-1); /* purecov: inspected */ } case 1: /* not found */ { my_errno = HA_ERR_KEY_NOT_FOUND; - return -1; + DBUG_RETURN(-1); /* purecov: inspected */ } default: case -1: /* error */ - return -1; + DBUG_RETURN(-1); /* purecov: inspected */ } } diff --git a/storage/maria/ma_rt_key.c b/storage/maria/ma_rt_key.c index 1b9f246081d..b74d5d06690 100644 --- a/storage/maria/ma_rt_key.c +++ b/storage/maria/ma_rt_key.c @@ -34,6 +34,7 @@ int maria_rtree_add_key(MARIA_HA *info, MARIA_KEYDEF *keyinfo, uchar *key, { uint page_size = maria_data_on_page(page_buf); uint nod_flag = _ma_test_if_nod(page_buf); + DBUG_ENTER("maria_rtree_add_key"); if (page_size + key_length + info->s->base.rec_reflength <= keyinfo->block_length) @@ -42,22 +43,27 @@ int maria_rtree_add_key(MARIA_HA *info, MARIA_KEYDEF *keyinfo, uchar *key, if (nod_flag) { /* save key */ + DBUG_ASSERT(_ma_kpos(nod_flag, key) < info->state->key_file_length); memcpy(rt_PAGE_END(page_buf), key - nod_flag, key_length + nod_flag); page_size += key_length + nod_flag; } else { /* save key */ + DBUG_ASSERT(_ma_dpos(info, nod_flag, key + key_length + + info->s->base.rec_reflength) < + info->state->data_file_length + + info->s->base.pack_reclength); memcpy(rt_PAGE_END(page_buf), key, key_length + info->s->base.rec_reflength); page_size += key_length + info->s->base.rec_reflength; } maria_putint(page_buf, page_size, nod_flag); - return 0; + DBUG_RETURN(0); } - return (maria_rtree_split_page(info, keyinfo, page_buf, key, key_length, - new_page) ? -1 : 1); + DBUG_RETURN(maria_rtree_split_page(info, keyinfo, page_buf, key, key_length, + new_page) ? -1 : 1); } @@ -91,11 +97,13 @@ int maria_rtree_delete_key(MARIA_HA *info, uchar *page_buf, uchar *key, int maria_rtree_set_key_mbr(MARIA_HA *info, MARIA_KEYDEF *keyinfo, uchar *key, uint key_length, my_off_t child_page) { + DBUG_ENTER("maria_rtree_set_key_mbr"); if (!_ma_fetch_keypage(info, keyinfo, child_page, DFLT_INIT_HITS, info->buff, 0)) - return -1; + DBUG_RETURN(-1); - return maria_rtree_page_mbr(info, keyinfo->seg, info->buff, key, key_length); + DBUG_RETURN(maria_rtree_page_mbr(info, keyinfo->seg, + info->buff, key, key_length)); } #endif /*HAVE_RTREE_KEYS*/ diff --git a/storage/maria/ma_rt_split.c b/storage/maria/ma_rt_split.c index 9d195d802c1..a91eaa47bea 100644 --- a/storage/maria/ma_rt_split.c +++ b/storage/maria/ma_rt_split.c @@ -266,13 +266,15 @@ int maria_rtree_split_page(MARIA_HA *info, MARIA_KEYDEF *keyinfo, uint full_length= key_length + (nod_flag ? nod_flag : info->s->base.rec_reflength); int max_keys= (maria_data_on_page(page)-2) / (full_length); + DBUG_ENTER("maria_rtree_split_page"); + DBUG_PRINT("rtree", ("splitting block")); n_dim = keyinfo->keysegs / 2; if (!(coord_buf= (double*) my_alloca(n_dim * 2 * sizeof(double) * (max_keys + 1 + 4) + sizeof(SplitStruct) * (max_keys + 1)))) - return -1; + DBUG_RETURN(-1); /* purecov: inspected */ task= (SplitStruct *)(coord_buf + n_dim * 2 * (max_keys + 1 + 4)); @@ -343,12 +345,18 @@ int maria_rtree_split_page(MARIA_HA *info, MARIA_KEYDEF *keyinfo, else err_code= _ma_write_keypage(info, keyinfo, *new_page_offs, DFLT_INIT_HITS, new_page); + DBUG_PRINT("rtree", ("split new block: %lu", (ulong) *new_page_offs)); my_afree((uchar*)new_page); split_err: + /** + @todo the cast below is useless (coord_buf is uchar*); at the moment we + changed all "byte" to "uchar", some casts became useless and should be + removed. + */ my_afree((uchar*) coord_buf); - return err_code; + DBUG_RETURN(err_code); } #endif /*HAVE_RTREE_KEYS*/ diff --git a/storage/maria/ma_search.c b/storage/maria/ma_search.c index d9bc5cd6264..8cb3e56e646 100644 --- a/storage/maria/ma_search.c +++ b/storage/maria/ma_search.c @@ -927,11 +927,16 @@ uint _ma_get_binary_pack_key(register MARIA_KEYDEF *keyinfo, uint nod_flag, /* Keys are compressed the following way: - prefix length Packed length of prefix for the prev key. (1 or 3 bytes) + prefix length Packed length of prefix common with prev key. (1 or 3 bytes) for each key segment: [is null] Null indicator if can be null (1 byte, zero means null) [length] Packed length if varlength (1 or 3 bytes) + key segment 'length' bytes of key segment value pointer Reference to the data file (last_keyseg->length). + + get_key_length() is a macro. It gets the prefix length from 'page' + and puts it into 'length'. It increments 'page' by 1 or 3, depending + on the packed length of the prefix length. */ get_key_length(length,page); if (length) @@ -946,34 +951,44 @@ uint _ma_get_binary_pack_key(register MARIA_KEYDEF *keyinfo, uint nod_flag, my_errno=HA_ERR_CRASHED; DBUG_RETURN(0); /* Wrong key */ } - from=key; from_end=key+length; + /* Key is packed against prev key, take prefix from prev key. */ + from= key; + from_end= key + length; } else { - from=page; from_end=page_end; /* Not packed key */ + /* Key is not packed against prev key, take all from page buffer. */ + from= page; + from_end= page_end; } /* - The trouble is that key is split in two parts: - The first part is in from ...from_end-1. - The second part starts at page + The trouble is that key can be split in two parts: + The first part (prefix) is in from .. from_end - 1. + The second part starts at page. + The split can be at every byte position. So we need to check for + the end of the first part before using every byte. */ for (keyseg=keyinfo->seg ; keyseg->type ;keyseg++) { if (keyseg->flag & HA_NULL_PART) { + /* If prefix is used up, switch to rest. */ if (from == from_end) { from=page; from_end=page_end; } if (!(*key++ = *from++)) continue; /* Null part */ } if (keyseg->flag & (HA_VAR_LENGTH_PART | HA_BLOB_PART | HA_SPACE_PACK)) { - /* Get length of dynamic length key part */ + /* If prefix is used up, switch to rest. */ if (from == from_end) { from=page; from_end=page_end; } + /* Get length of dynamic length key part */ if ((length= (uint) (uchar) (*key++ = *from++)) == 255) { + /* If prefix is used up, switch to rest. */ if (from == from_end) { from=page; from_end=page_end; } length= ((uint) (uchar) ((*key++ = *from++))) << 8; + /* If prefix is used up, switch to rest. */ if (from == from_end) { from=page; from_end=page_end; } length+= (uint) (uchar) ((*key++ = *from++)); } @@ -994,14 +1009,26 @@ uint _ma_get_binary_pack_key(register MARIA_KEYDEF *keyinfo, uint nod_flag, key+=length; from+=length; } + /* + Last segment (type == 0) contains length of data pointer. + If we have mixed key blocks with data pointer and key block pointer, + we have to copy both. + */ length=keyseg->length+nod_flag; if ((tmp=(uint) (from_end-from)) <= length) { + /* Remaining length is less or equal max possible length. */ memcpy(key+tmp,page,length-tmp); /* Get last part of key */ *page_pos= page+length-tmp; } else { + /* + Remaining length is greater than max possible length. + This can happen only if we switched to the new key bytes already. + 'page_end' is calculated with MI_MAX_KEY_BUFF. So it can be far + behind the real end of the key. + */ if (from_end != page_end) { DBUG_PRINT("error",("Error when unpacking key")); @@ -1009,6 +1036,7 @@ uint _ma_get_binary_pack_key(register MARIA_KEYDEF *keyinfo, uint nod_flag, my_errno=HA_ERR_CRASHED; DBUG_RETURN(0); /* Error */ } + /* Copy data pointer and, if appropriate, key block pointer. */ memcpy((uchar*) key,(uchar*) from,(size_t) length); *page_pos= from+length; } diff --git a/storage/maria/ma_sort.c b/storage/maria/ma_sort.c index bc2b75807d9..2851a3a09dd 100644 --- a/storage/maria/ma_sort.c +++ b/storage/maria/ma_sort.c @@ -138,8 +138,9 @@ int _ma_create_index_by_sort(MARIA_SORT_PARAM *info, my_bool no_messages, while (memavl >= MIN_SORT_MEMORY) { - if ((my_off_t) (records+1)*(sort_length+sizeof(char*)) <= - (my_off_t) memavl) + if ((records < UINT_MAX32) && + ((my_off_t) (records + 1) * + (sort_length + sizeof(char*)) <= (my_off_t) memavl)) keys= records+1; else do @@ -151,7 +152,7 @@ int _ma_create_index_by_sort(MARIA_SORT_PARAM *info, my_bool no_messages, keys < (uint) maxbuffer) { _ma_check_print_error(info->sort_info->param, - "sort_buffer_size is to small"); + "maria_sort_buffer_size is too small"); goto err; } } @@ -175,7 +176,8 @@ int _ma_create_index_by_sort(MARIA_SORT_PARAM *info, my_bool no_messages, } if (memavl < MIN_SORT_MEMORY) { - _ma_check_print_error(info->sort_info->param,"Sort buffer to small"); /* purecov: tested */ + _ma_check_print_error(info->sort_info->param, "Maria sort buffer" + " too small"); /* purecov: tested */ goto err; /* purecov: tested */ } (*info->lock_in_memory)(info->sort_info->param);/* Everything is allocated */ @@ -369,7 +371,7 @@ pthread_handler_t _ma_thr_find_all_keys(void *arg) keys < (uint) maxbuffer) { _ma_check_print_error(sort_param->sort_info->param, - "sort_buffer_size is to small"); + "maria_sort_buffer_size is too small"); goto err; } } @@ -397,7 +399,7 @@ pthread_handler_t _ma_thr_find_all_keys(void *arg) if (memavl < MIN_SORT_MEMORY) { _ma_check_print_error(sort_param->sort_info->param, - "Sort buffer too small"); + "Maria sort buffer too small"); goto err; /* purecov: tested */ } @@ -775,7 +777,7 @@ static int NEAR_F merge_many_buff(MARIA_SORT_PARAM *info, uint keys, { if (merge_buffers(info,keys,from_file,to_file,sort_keys,lastbuff++, buffpek+i,buffpek+i+MERGEBUFF-1)) - break; /* purecov: inspected */ + goto cleanup; } if (merge_buffers(info,keys,from_file,to_file,sort_keys,lastbuff++, buffpek+i,buffpek+ *maxbuffer)) @@ -785,6 +787,7 @@ static int NEAR_F merge_many_buff(MARIA_SORT_PARAM *info, uint keys, temp=from_file; from_file=to_file; to_file=temp; *maxbuffer= (int) (lastbuff-buffpek)-1; } +cleanup: close_cached_file(to_file); /* This holds old result */ if (to_file == t_file) *t_file=t_file2; /* Copy result file */ diff --git a/storage/maria/maria_def.h b/storage/maria/maria_def.h index 62f37077ceb..e2dc7d3be86 100644 --- a/storage/maria/maria_def.h +++ b/storage/maria/maria_def.h @@ -442,6 +442,7 @@ struct st_maria_info enum ha_rkey_function last_key_func; /* CONTAIN, OVERLAP, etc */ uint save_lastkey_length; uint pack_key_length; /* For MARIAMRG */ + uint16 last_used_keyseg; /* For MARIAMRG */ int errkey; /* Got last error on this key */ int lock_type; /* How database was locked */ int tmp_lock_type; /* When locked by readinfo */ @@ -749,7 +750,7 @@ extern my_off_t _ma_new(MARIA_HA *info, MARIA_KEYDEF *keyinfo, int level); extern uint _ma_make_key(MARIA_HA *info, uint keynr, uchar *key, const uchar *record, MARIA_RECORD_POS filepos); extern uint _ma_pack_key(MARIA_HA *info, uint keynr, uchar *key, - const uchar *old, uint key_length, + const uchar *old, key_part_map keypart_map, HA_KEYSEG ** last_used_keyseg); extern int _ma_read_key_record(MARIA_HA *info, uchar *buf, MARIA_RECORD_POS); extern int _ma_read_cache(IO_CACHE *info, uchar *buff, MARIA_RECORD_POS pos, @@ -760,7 +761,7 @@ extern my_bool _ma_alloc_buffer(uchar **old_addr, size_t *old_size, size_t new_size); extern ulong _ma_rec_unpack(MARIA_HA *info, uchar *to, uchar *from, ulong reclength); -extern my_bool _ma_rec_check(MARIA_HA *info, const char *record, +extern my_bool _ma_rec_check(MARIA_HA *info, const uchar *record, uchar *packpos, ulong packed_length, my_bool with_checkum, ha_checksum checksum); extern int _ma_write_part_record(MARIA_HA *info, my_off_t filepos, -- cgit v1.2.1 From 9b1e83dba732896f606181b21becc3b0144a91ec Mon Sep 17 00:00:00 2001 From: unknown Date: Fri, 27 Jul 2007 16:11:40 +0200 Subject: porting Serg's fix for BUG#30094 to Maria. Now ma_test_all passes. maria.test and ps_maria.test still fail. mysys/mf_keycache.c: split string annoys some compilers storage/maria/ha_maria.cc: fix for compiler warnings storage/maria/ma_test1.c: porting Serg's fix for BUG#30094 to Maria storage/maria/ma_test2.c: porting Serg's fix for BUG#30094 to Maria storage/maria/ma_test3.c: porting Serg's fix for BUG#30094 to Maria storage/maria/ma_test_recovery: don't print ma_test1's messages if no problem --- mysys/mf_keycache.c | 4 ++-- storage/maria/ha_maria.cc | 6 ++++-- storage/maria/ma_test1.c | 5 +++-- storage/maria/ma_test2.c | 23 +++++++++++++---------- storage/maria/ma_test3.c | 4 ++-- storage/maria/ma_test_recovery | 6 +++--- 6 files changed, 27 insertions(+), 21 deletions(-) diff --git a/mysys/mf_keycache.c b/mysys/mf_keycache.c index 4a8ab5b6719..e865d152633 100644 --- a/mysys/mf_keycache.c +++ b/mysys/mf_keycache.c @@ -2977,8 +2977,8 @@ int key_cache_write(KEY_CACHE *keycache, int error=0; DBUG_ENTER("key_cache_write"); DBUG_PRINT("enter", - ("fd: %u pos: %lu length: %u block_length: %u key_block_length: - %u", + ("fd: %u pos: %lu length: %u block_length: %u" + " key_block_length: %u", (uint) file, (ulong) filepos, length, block_length, keycache ? keycache->key_cache_block_size : 0)); diff --git a/storage/maria/ha_maria.cc b/storage/maria/ha_maria.cc index d0888278cca..3c764bbc827 100644 --- a/storage/maria/ha_maria.cc +++ b/storage/maria/ha_maria.cc @@ -746,7 +746,6 @@ int ha_maria::open(const char *name, int mode, uint test_if_locked) if (!(file= maria_open(name, mode, test_if_locked | HA_OPEN_FROM_SQL_LAYER))) return (my_errno ? my_errno : -1); -#ifdef ASK_MONTY /* This is a protection for the case of a frm and MAI containing incompatible table definitions (as in BUG#25908). This was merged from MyISAM. @@ -764,9 +763,13 @@ int ha_maria::open(const char *name, int mode, uint test_if_locked) goto err; /* purecov: end */ } +#ifdef ASK_MONTY if (maria_check_definition(keyinfo, recinfo, table->s->keys, recs, file->s->keyinfo, file->s->columndef, file->s->base.keys, file->s->base.fields, true)) +#else + if (0) +#endif { /* purecov: begin inspected */ my_errno= HA_ERR_CRASHED; @@ -774,7 +777,6 @@ int ha_maria::open(const char *name, int mode, uint test_if_locked) /* purecov: end */ } } -#endif if (test_if_locked & (HA_OPEN_IGNORE_IF_LOCKED | HA_OPEN_TMP_TABLE)) VOID(maria_extra(file, HA_EXTRA_NO_WAIT_LOCK, 0)); diff --git a/storage/maria/ma_test1.c b/storage/maria/ma_test1.c index 7d7a975a641..44ec6af5d8e 100644 --- a/storage/maria/ma_test1.c +++ b/storage/maria/ma_test1.c @@ -288,7 +288,8 @@ static int run_test(const char *filename) continue; create_key(key,j); my_errno=0; - if ((error = maria_rkey(file,read_record,0,key,0,HA_READ_KEY_EXACT))) + if ((error = maria_rkey(file, read_record, 0, key, + HA_WHOLE_KEY, HA_READ_KEY_EXACT))) { if (verbose || (flags[j] >= 1 || (error && my_errno != HA_ERR_KEY_NOT_FOUND))) @@ -316,7 +317,7 @@ static int run_test(const char *filename) { create_key(key,i); my_errno=0; - error=maria_rkey(file,read_record,0,key,0,HA_READ_KEY_EXACT); + error=maria_rkey(file,read_record,0,key,HA_WHOLE_KEY,HA_READ_KEY_EXACT); if (verbose || (error == 0 && flags[i] == 0 && unique_key) || (error && (flags[i] != 0 || my_errno != HA_ERR_KEY_NOT_FOUND))) diff --git a/storage/maria/ma_test2.c b/storage/maria/ma_test2.c index 8ab22c60d40..2e3884bf6ce 100644 --- a/storage/maria/ma_test2.c +++ b/storage/maria/ma_test2.c @@ -288,7 +288,7 @@ int main(int argc, char *argv[]) if (!j) for (j=999 ; j>0 && key1[j] == 0 ; j--) ; sprintf(key,"%6d",j); - if (maria_rkey(file,read_record,0,key,0,HA_READ_KEY_EXACT)) + if (maria_rkey(file,read_record,0,key,HA_WHOLE_KEY,HA_READ_KEY_EXACT)) { printf("Test in loop: Can't find key: \"%s\"\n",key); goto err; @@ -321,7 +321,7 @@ int main(int argc, char *argv[]) if (j != 0) { sprintf(key,"%6d",j); - if (maria_rkey(file,read_record,0,key,0,HA_READ_KEY_EXACT)) + if (maria_rkey(file,read_record,0,key,HA_WHOLE_KEY,HA_READ_KEY_EXACT)) { printf("can't find key1: \"%s\"\n",key); goto err; @@ -364,7 +364,7 @@ int main(int argc, char *argv[]) if (j != 0) { sprintf(key,"%6d",j); - if (maria_rkey(file,read_record,0,key,0,HA_READ_KEY_EXACT)) + if (maria_rkey(file,read_record,0,key,HA_WHOLE_KEY,HA_READ_KEY_EXACT)) { printf("can't find key1: \"%s\"\n",key); goto err; @@ -427,7 +427,7 @@ int main(int argc, char *argv[]) DBUG_PRINT("progpos",("first - next -> last - prev -> first")); if (verbose) printf(" Using key: \"%s\" Keys: %d\n",key,dupp_keys); - if (maria_rkey(file,read_record,0,key,0,HA_READ_KEY_EXACT)) + if (maria_rkey(file,read_record,0,key,HA_WHOLE_KEY,HA_READ_KEY_EXACT)) goto err; if (maria_rsame(file,read_record2,-1)) goto err; @@ -474,7 +474,7 @@ int main(int argc, char *argv[]) } /* Check of maria_rnext_same */ - if (maria_rkey(file,read_record,0,key,0,HA_READ_KEY_EXACT)) + if (maria_rkey(file,read_record,0,key,HA_WHOLE_KEY,HA_READ_KEY_EXACT)) goto err; ant=1; while (!maria_rnext_same(file,read_record3) && ant < dupp_keys+10) @@ -548,7 +548,7 @@ int main(int argc, char *argv[]) goto err; if (bcmp(read_record2,read_record3,reclength)) printf("Can't find last record\n"); - +#ifdef NOT_ANYMORE if (!silent) puts("- Test read key-part"); strmov(key2,key); @@ -566,12 +566,14 @@ int main(int argc, char *argv[]) goto err; } } +#endif if (dupp_keys > 2) { if (!silent) printf("- Read key (first) - next - delete - next -> last\n"); DBUG_PRINT("progpos",("first - next - delete - next -> last")); - if (maria_rkey(file,read_record,0,key,0,HA_READ_KEY_EXACT)) goto err; + if (maria_rkey(file,read_record,0,key,HA_WHOLE_KEY,HA_READ_KEY_EXACT)) + goto err; if (maria_rnext(file,read_record3,0)) goto err; if (maria_delete(file,read_record3)) goto err; opt_delete++; @@ -607,7 +609,8 @@ int main(int argc, char *argv[]) if (!silent) printf("- Read first - delete - next -> last\n"); DBUG_PRINT("progpos",("first - delete - next -> last")); - if (maria_rkey(file,read_record3,0,key,0,HA_READ_KEY_EXACT)) goto err; + if (maria_rkey(file,read_record3,0,key,HA_WHOLE_KEY,HA_READ_KEY_EXACT)) + goto err; if (maria_delete(file,read_record3)) goto err; opt_delete++; ant=1; @@ -681,10 +684,10 @@ int main(int argc, char *argv[]) copy_key(file,(uint) i,(uchar*) read_record,(uchar*) key); copy_key(file,(uint) i,(uchar*) read_record2,(uchar*) key2); min_key.key= key; - min_key.length= USE_WHOLE_KEY; + min_key.keypart_map= HA_WHOLE_KEY; min_key.flag= HA_READ_KEY_EXACT; max_key.key= key2; - max_key.length= USE_WHOLE_KEY; + max_key.keypart_map= HA_WHOLE_KEY; max_key.flag= HA_READ_AFTER_KEY; range_records= maria_records_in_range(file,(int) i, &min_key, &max_key); diff --git a/storage/maria/ma_test3.c b/storage/maria/ma_test3.c index 948ba09aa24..c25dd5dcdc6 100644 --- a/storage/maria/ma_test3.c +++ b/storage/maria/ma_test3.c @@ -243,7 +243,7 @@ int test_read(MARIA_HA *file,int id) { find=rnd(100000); if (!maria_rkey(file,record.id,1,(uchar*) &find, - sizeof(find),HA_READ_KEY_EXACT)) + HA_WHOLE_KEY,HA_READ_KEY_EXACT)) found++; else { @@ -426,7 +426,7 @@ int test_update(MARIA_HA *file,int id,int lock_type) tmp=rnd(100000); int4store(find,tmp); if (!maria_rkey(file,record.id,1,(uchar*) find, - sizeof(find),HA_READ_KEY_EXACT)) + HA_WHOLE_KEY,HA_READ_KEY_EXACT)) found++; else { diff --git a/storage/maria/ma_test_recovery b/storage/maria/ma_test_recovery index 3393b932e18..65b8ac0838a 100644 --- a/storage/maria/ma_test_recovery +++ b/storage/maria/ma_test_recovery @@ -1,5 +1,5 @@ set -e - +silent="-s" if [ -z "$maria_path" ] then maria_path="." @@ -13,7 +13,7 @@ echo "MARIA RECOVERY TESTS - success is if exit code is 0" # Does not test the index file as we don't have logging for it yet. rm -f maria_log* -prog="$maria_path/ma_test1 -M -T --skip-update" +prog="$maria_path/ma_test1 $silent -M -T --skip-update" echo "TEST WITH $prog" $prog mv -f test1.MAD test1.MAD.good @@ -24,7 +24,7 @@ cmp test1.MAD test1.MAD.good rm -f test1.* rm -f maria_log* -prog="$maria_path/ma_test2 -s -L -K -W -P -M -T -g" +prog="$maria_path/ma_test2 $silent -L -K -W -P -M -T -g" echo "TEST WITH $prog" $prog mv -f test2.MAD test2.MAD.good -- cgit v1.2.1 From 9554b40df2a2c1ad3be7b79cf8a5abe6ca4e211e Mon Sep 17 00:00:00 2001 From: unknown Date: Mon, 30 Jul 2007 12:01:52 +0200 Subject: Maria: it is allowed to change unknown type to any legal type of page in the pagecache: fixing wrong assertion and a test case. maria.test and ps_maria.test still fail. mysql-test/r/maria.result: result update mysql-test/t/maria.test: added test for incorrect assert on page's type storage/maria/ma_pagecache.c: It is allowed to change unknown type to any legal type of page in the pagecache. --- mysql-test/r/maria.result | 4 ++++ mysql-test/t/maria.test | 4 ++++ storage/maria/ma_pagecache.c | 4 +++- 3 files changed, 11 insertions(+), 1 deletion(-) diff --git a/mysql-test/r/maria.result b/mysql-test/r/maria.result index 6303f498e36..90b406cbb9d 100644 --- a/mysql-test/r/maria.result +++ b/mysql-test/r/maria.result @@ -33,6 +33,10 @@ test.t1 repair status OK check table t1; Table Op Msg_type Msg_text test.t1 check status OK +flush table t1; +repair table t1; +Table Op Msg_type Msg_text +test.t1 repair status OK drop table t1; create table t1 (a int not null auto_increment, b int not null, primary key (a), index(b)); insert into t1 (b) values (1),(2),(2),(2),(2); diff --git a/mysql-test/t/maria.test b/mysql-test/t/maria.test index 26637f1f651..5eddf0dd300 100644 --- a/mysql-test/t/maria.test +++ b/mysql-test/t/maria.test @@ -57,6 +57,10 @@ delete from t1 where (a & 1); check table t1; repair table t1; check table t1; + +# FLUSH + REPAIR used to cause assertion failure in page cache +flush table t1; +repair table t1; drop table t1; # diff --git a/storage/maria/ma_pagecache.c b/storage/maria/ma_pagecache.c index 12e561c69bc..5c803b3dd83 100755 --- a/storage/maria/ma_pagecache.c +++ b/storage/maria/ma_pagecache.c @@ -2865,7 +2865,9 @@ restart: (pin == PAGECACHE_PIN)), &page_st); DBUG_ASSERT(block->type == PAGECACHE_EMPTY_PAGE || - block->type == type || type == PAGECACHE_READ_UNKNOWN_PAGE); + block->type == type || + type == PAGECACHE_READ_UNKNOWN_PAGE || + block->type == PAGECACHE_READ_UNKNOWN_PAGE); if (type != PAGECACHE_READ_UNKNOWN_PAGE || block->type == PAGECACHE_EMPTY_PAGE) block->type= type; -- cgit v1.2.1 From 2cccfcd8dd728e2ad862a115ae43bc3b62d7529b Mon Sep 17 00:00:00 2001 From: unknown Date: Mon, 30 Jul 2007 15:05:43 +0200 Subject: Applying Sanja's patch which makes the log handler not issue errors when reading a log record which has a 0-length header (like LOGREC_REDO_DROP_TABLE). storage/maria/ma_loghandler.c: Functions reading record's header now don't use 0 to indicate error, as some valid records have a 0-length header (like REDO_DROP_TABLE). Instead, negative values are used for EOF and error. storage/maria/ma_loghandler.h: functions to read record's header now return an int (either the length of this header (>=0) or some negative values for EOF or error). storage/maria/ma_recovery.c: update to the new log handler's behaviour. Note the @todo. storage/maria/maria_read_log.c: inform when program failed storage/maria/unittest/ma_test_loghandler-t.c: update to new log handler's API storage/maria/unittest/ma_test_loghandler_multigroup-t.c: update to new log handler's API storage/maria/unittest/ma_test_loghandler_multithread-t.c: update to new log handler's API --- storage/maria/ma_loghandler.c | 263 +++++++++------------ storage/maria/ma_loghandler.h | 13 +- storage/maria/ma_recovery.c | 26 +- storage/maria/maria_read_log.c | 3 +- storage/maria/unittest/ma_test_loghandler-t.c | 26 +- .../unittest/ma_test_loghandler_multigroup-t.c | 28 +-- .../unittest/ma_test_loghandler_multithread-t.c | 13 +- 7 files changed, 177 insertions(+), 195 deletions(-) diff --git a/storage/maria/ma_loghandler.c b/storage/maria/ma_loghandler.c index 9d1514a67c3..4158884afb7 100644 --- a/storage/maria/ma_loghandler.c +++ b/storage/maria/ma_loghandler.c @@ -373,19 +373,8 @@ static LOG_DESC INIT_LOGREC_REDO_RENAME_TABLE= {LOGRECTYPE_VARIABLE_LENGTH, 0, 0, NULL, NULL, NULL, 0, "redo_rename_table", LOGREC_IS_GROUP_ITSELF, NULL, NULL}; -/** - @todo LOG BUG - the "1" below is a hack to overcome a bug in the log handler where a 0-byte - header is considered a read failure: - translog_read_record() calls translog_init_reader_data() which calls - translog_read_record_header_scan() which calls - translog_read_record_header_from_buffer() which calls - translog_variable_length_header() which returns 0 (normal); - translog_init_reader_data() considers this 0 as a problem, - and thus translog_read_record() fails. -*/ static LOG_DESC INIT_LOGREC_REDO_DROP_TABLE= -{LOGRECTYPE_VARIABLE_LENGTH, 0, 1, NULL, NULL, NULL, 0, +{LOGRECTYPE_VARIABLE_LENGTH, 0, 0, NULL, NULL, NULL, 0, "redo_drop_table", LOGREC_IS_GROUP_ITSELF, NULL, NULL}; static LOG_DESC INIT_LOGREC_REDO_DELETE_ALL= @@ -4437,25 +4426,23 @@ static uchar *translog_relative_LSN_decode(LSN base_lsn, return src; } -/* - Get header of fixed/pseudo length record and call hook for it processing +/** + @brief Get header of fixed/pseudo length record and call hook for + it processing - SYNOPSIS - translog_fixed_length_header() - page Pointer to the buffer with page where LSN chunk is - placed - page_offset Offset of the first chunk in the page - buff Buffer to be filled with header data + @param page Pointer to the buffer with page where LSN chunk is + placed + @param page_offset Offset of the first chunk in the page + @param buff Buffer to be filled with header data - RETURN - 0 error - # number of bytes in TRANSLOG_HEADER_BUFFER::header where stored decoded - part of the header + @return Length of header or operation status + @retval # number of bytes in TRANSLOG_HEADER_BUFFER::header where + stored decoded part of the header */ -translog_size_t translog_fixed_length_header(uchar *page, - translog_size_t page_offset, - TRANSLOG_HEADER_BUFFER *buff) +static int translog_fixed_length_header(uchar *page, + translog_size_t page_offset, + TRANSLOG_HEADER_BUFFER *buff) { struct st_log_record_type_descriptor *desc= log_record_type_descriptor + buff->type; @@ -4609,6 +4596,7 @@ my_bool translog_init_scanner(LSN lsn, 1 End of the Log 0 OK */ + static my_bool translog_scanner_eol(TRANSLOG_SCANNER_DATA *scanner) { DBUG_ENTER("translog_scanner_eol"); @@ -4652,6 +4640,7 @@ static my_bool translog_scanner_eol(TRANSLOG_SCANNER_DATA *scanner) 1 End of the Page 0 OK */ + static my_bool translog_scanner_eop(TRANSLOG_SCANNER_DATA *scanner) { DBUG_ENTER("translog_scanner_eop"); @@ -4672,6 +4661,7 @@ static my_bool translog_scanner_eop(TRANSLOG_SCANNER_DATA *scanner) 1 End of the File 0 OK */ + static my_bool translog_scanner_eof(TRANSLOG_SCANNER_DATA *scanner) { DBUG_ENTER("translog_scanner_eof"); @@ -4763,29 +4753,25 @@ translog_get_next_chunk(TRANSLOG_SCANNER_DATA *scanner) } -/* - Get header of variable length record and call hook for it processing - - SYNOPSIS - translog_variable_length_header() - page Pointer to the buffer with page where LSN chunk is - placed - page_offset Offset of the first chunk in the page - buff Buffer to be filled with header data - scanner If present should be moved to the header page if +/** + @brief Get header of variable length record and call hook for it processing + + @param page Pointer to the buffer with page where LSN chunk is + placed + @param page_offset Offset of the first chunk in the page + @param buff Buffer to be filled with header data + @param scanner If present should be moved to the header page if it differ from LSN page - - RETURN - 0 error - # number of bytes in TRANSLOG_HEADER_BUFFER::header where stored decoded - part of the header + @return Length of header or operation status + @retval RECHEADER_READ_ERROR error + @retval # number of bytes in + TRANSLOG_HEADER_BUFFER::header where + stored decoded part of the header */ -translog_size_t translog_variable_length_header(uchar *page, - translog_size_t page_offset, - TRANSLOG_HEADER_BUFFER *buff, - TRANSLOG_SCANNER_DATA - *scanner) +int translog_variable_length_header(uchar *page, translog_size_t page_offset, + TRANSLOG_HEADER_BUFFER *buff, + TRANSLOG_SCANNER_DATA *scanner) { struct st_log_record_type_descriptor *desc= (log_record_type_descriptor + buff->type); @@ -4827,7 +4813,7 @@ translog_size_t translog_variable_length_header(uchar *page, if (!(buff->groups= (TRANSLOG_GROUP*) my_malloc(sizeof(TRANSLOG_GROUP) * grp_no, MYF(0)))) - DBUG_RETURN(0); + DBUG_RETURN(RECHEADER_READ_ERROR); DBUG_PRINT("info", ("Groups: %u", (uint) grp_no)); src+= (2 + 2); page_rest= TRANSLOG_PAGE_SIZE - (src - page); @@ -4882,9 +4868,11 @@ translog_size_t translog_variable_length_header(uchar *page, { DBUG_PRINT("info", ("use internal scanner for header reading")); scanner= &internal_scanner; - translog_init_scanner(buff->lsn, 1, scanner); + if (translog_init_scanner(buff->lsn, 1, scanner)) + DBUG_RETURN(RECHEADER_READ_ERROR); } - translog_get_next_chunk(scanner); + if (translog_get_next_chunk(scanner)) + DBUG_RETURN(RECHEADER_READ_ERROR); page= scanner->page; page_offset= scanner->page_offset; src= page + page_offset + header_to_skip; @@ -4938,24 +4926,27 @@ translog_size_t translog_variable_length_header(uchar *page, } -/* - Read record header from the given buffer - - SYNOPSIS - translog_read_record_header_from_buffer() - page page content buffer - page_offset offset of the chunk in the page - buff destination buffer - scanner If this is set the scanner will be moved to the - record header page (differ from LSN page in case of - multi-group records) +/** + @brief Read record header from the given buffer + + @param page page content buffer + @param page_offset offset of the chunk in the page + @param buff destination buffer + @param scanner If this is set the scanner will be moved to the + record header page (differ from LSN page in case of + multi-group records) + + @return Length of header or operation status + @retval RECHEADER_READ_ERROR error + @retval # number of bytes in + TRANSLOG_HEADER_BUFFER::header where + stored decoded part of the header */ -translog_size_t -translog_read_record_header_from_buffer(uchar *page, - uint16 page_offset, - TRANSLOG_HEADER_BUFFER *buff, - TRANSLOG_SCANNER_DATA *scanner) +int translog_read_record_header_from_buffer(uchar *page, + uint16 page_offset, + TRANSLOG_HEADER_BUFFER *buff, + TRANSLOG_SCANNER_DATA *scanner) { translog_size_t res; DBUG_ENTER("translog_read_record_header_from_buffer"); @@ -4981,36 +4972,32 @@ translog_read_record_header_from_buffer(uchar *page, break; default: DBUG_ASSERT(0); - res= 0; + res= RECHEADER_READ_ERROR; } DBUG_RETURN(res); } -/* - Read record header and some fixed part of a record (the part depend on - record type). - - SYNOPSIS - translog_read_record_header() - lsn log record serial number (address of the record) - buff log record header buffer - - NOTE - - Some type of record can be read completely by this call - - "Decoded" header stored in TRANSLOG_HEADER_BUFFER::header (relative - LSN can be translated to absolute one), some fields can be added - (like actual header length in the record if the header has variable - length) - - RETURN - 0 error - # number of bytes in TRANSLOG_HEADER_BUFFER::header where stored decoded - part of the header +/** + @brief Read record header and some fixed part of a record (the part depend + on record type). + + @param lsn log record serial number (address of the record) + @param buff log record header buffer + + @note Some type of record can be read completely by this call + @note "Decoded" header stored in TRANSLOG_HEADER_BUFFER::header (relative + LSN can be translated to absolute one), some fields can be added (like + actual header length in the record if the header has variable length) + + @return Length of header or operation status + @retval RECHEADER_READ_ERROR error + @retval # number of bytes in + TRANSLOG_HEADER_BUFFER::header where + stored decoded part of the header */ -translog_size_t translog_read_record_header(LSN lsn, - TRANSLOG_HEADER_BUFFER *buff) +int translog_read_record_header(LSN lsn, TRANSLOG_HEADER_BUFFER *buff) { uchar buffer[TRANSLOG_PAGE_SIZE], *page; translog_size_t res, page_offset= LSN_OFFSET(lsn) % TRANSLOG_PAGE_SIZE; @@ -5027,40 +5014,35 @@ translog_size_t translog_read_record_header(LSN lsn, data.was_recovered= 0; addr= lsn; addr-= page_offset; /* offset decreasing */ - res= (!(page= translog_get_page(&data, buffer))) ? 0 : + res= (!(page= translog_get_page(&data, buffer))) ? RECHEADER_READ_ERROR : translog_read_record_header_from_buffer(page, page_offset, buff, 0); DBUG_RETURN(res); } -/* - Read record header and some fixed part of a record (the part depend on - record type). - - SYNOPSIS - translog_read_record_header_scan() - scan scanner position to read - buff log record header buffer - move_scanner request to move scanner to the header position - - NOTE - - Some type of record can be read completely by this call - - "Decoded" header stored in TRANSLOG_HEADER_BUFFER::header (relative - LSN can be translated to absolute one), some fields can be added - (like actual header length in the record if the header has variable - length) - - RETURN - 0 error - # number of bytes in TRANSLOG_HEADER_BUFFER::header where stored decoded - part of the header +/** + @brief Read record header and some fixed part of a record (the part depend + on record type). + + @param scan scanner position to read + @param buff log record header buffer + @param move_scanner request to move scanner to the header position + + @note Some type of record can be read completely by this call + @note "Decoded" header stored in TRANSLOG_HEADER_BUFFER::header (relative + LSN can be translated to absolute one), some fields can be added (like + actual header length in the record if the header has variable length) + + @return Length of header or operation status + @retval RECHEADER_READ_ERROR error + @retval # number of bytes in + TRANSLOG_HEADER_BUFFER::header where stored + decoded part of the header */ -translog_size_t -translog_read_record_header_scan(TRANSLOG_SCANNER_DATA - *scanner, - TRANSLOG_HEADER_BUFFER *buff, - my_bool move_scanner) +int translog_read_record_header_scan(TRANSLOG_SCANNER_DATA *scanner, + TRANSLOG_HEADER_BUFFER *buff, + my_bool move_scanner) { translog_size_t res; DBUG_ENTER("translog_read_record_header_scan"); @@ -5086,35 +5068,26 @@ translog_read_record_header_scan(TRANSLOG_SCANNER_DATA } -/* - Read record header and some fixed part of the next record (the part - depend on record type). - - SYNOPSIS - translog_read_next_record_header() - scanner data for scanning if lsn is NULL scanner data - will be used for continue scanning. - The scanner can be NULL. - buff log record header buffer - - NOTE - - it is like translog_read_record_header, but read next record, so see - its NOTES. - - in case of end of the log buff->lsn will be set to - (LSN_IMPOSSIBLE) - - RETURN - 0 error - TRANSLOG_RECORD_HEADER_MAX_SIZE + 1 End of the log - # number of bytes in - TRANSLOG_HEADER_BUFFER::header - where stored decoded - part of the header +/** + @brief Read record header and some fixed part of the next record (the part + depend on record type). + + @param scanner data for scanning if lsn is NULL scanner data + will be used for continue scanning. + The scanner can be NULL. + + @param buff log record header buffer + + @return Length of header or operation status + @retval RECHEADER_READ_ERROR error + @retval RECHEADER_READ_EOF EOF + @retval # number of bytes in + TRANSLOG_HEADER_BUFFER::header where + stored decoded part of the header */ -translog_size_t translog_read_next_record_header(TRANSLOG_SCANNER_DATA - *scanner, - TRANSLOG_HEADER_BUFFER *buff) +int translog_read_next_record_header(TRANSLOG_SCANNER_DATA *scanner, + TRANSLOG_HEADER_BUFFER *buff) { uint8 chunk_type; translog_size_t res; @@ -5136,7 +5109,7 @@ translog_size_t translog_read_next_record_header(TRANSLOG_SCANNER_DATA do { if (translog_get_next_chunk(scanner)) - DBUG_RETURN(0); + DBUG_RETURN(RECHEADER_READ_ERROR); chunk_type= scanner->page[scanner->page_offset] & TRANSLOG_CHUNK_TYPE; DBUG_PRINT("info", ("type: %x byte: %x", (uint) chunk_type, (uint) scanner->page[scanner->page_offset])); @@ -5148,7 +5121,7 @@ translog_size_t translog_read_next_record_header(TRANSLOG_SCANNER_DATA /* Last record was read */ buff->lsn= LSN_IMPOSSIBLE; /* Return 'end of log' marker */ - res= TRANSLOG_RECORD_HEADER_MAX_SIZE + 1; + res= RECHEADER_READ_EOF; } else res= translog_read_record_header_scan(scanner, buff, 0); diff --git a/storage/maria/ma_loghandler.h b/storage/maria/ma_loghandler.h index 011b8f4cf83..40e87d1d99d 100644 --- a/storage/maria/ma_loghandler.h +++ b/storage/maria/ma_loghandler.h @@ -30,6 +30,9 @@ #define TRANSLOG_FLAGS_NUM ((TRANSLOG_PAGE_CRC | TRANSLOG_SECTOR_PROTECTION | \ TRANSLOG_RECORD_CRC) + 1) +#define RECHEADER_READ_ERROR -1 +#define RECHEADER_READ_EOF -2 + /* Page size in transaction log It should be Power of 2 and multiple of DISK_DRIVE_SECTOR_SIZE @@ -228,9 +231,7 @@ translog_write_record(LSN *lsn, enum translog_record_type type, extern void translog_destroy(); -extern translog_size_t translog_read_record_header(LSN lsn, - TRANSLOG_HEADER_BUFFER - *buff); +extern int translog_read_record_header(LSN lsn, TRANSLOG_HEADER_BUFFER *buff); extern void translog_free_record_header(TRANSLOG_HEADER_BUFFER *buff); @@ -247,10 +248,8 @@ extern my_bool translog_init_scanner(LSN lsn, my_bool fixed_horizon, struct st_translog_scanner_data *scanner); -extern translog_size_t translog_read_next_record_header(TRANSLOG_SCANNER_DATA - *scanner, - TRANSLOG_HEADER_BUFFER - *buff); +extern int translog_read_next_record_header(TRANSLOG_SCANNER_DATA *scanner, + TRANSLOG_HEADER_BUFFER *buff); extern my_bool translog_lock(); extern my_bool translog_unlock(); extern void translog_lock_assert_owner(); diff --git a/storage/maria/ma_recovery.c b/storage/maria/ma_recovery.c index b8a1206cdb6..6ed47533fef 100644 --- a/storage/maria/ma_recovery.c +++ b/storage/maria/ma_recovery.c @@ -194,13 +194,13 @@ int maria_apply_log(LSN lsn, my_bool apply, FILE *trace_file) struct st_translog_scanner_data scanner; uint i= 1; - translog_size_t len= translog_read_record_header(lsn, &rec); + int len= translog_read_record_header(lsn, &rec); - /** @todo translog_read_record_header() should be fixed for 0-byte headers */ - if (len == 0) /* means error, but apparently EOF too */ + /** @todo EOF should be detected */ + if (len == RECHEADER_READ_ERROR) { - fprintf(tracef, "empty log\n"); - goto end; + fprintf(tracef, "Cannot find a first record\n"); + goto err; } if (translog_init_scanner(lsn, 1, &scanner)) @@ -246,7 +246,7 @@ int maria_apply_log(LSN lsn, my_bool apply, FILE *trace_file) TRANSLOG_HEADER_BUFFER rec2; len= translog_read_record_header(all_active_trans[sid].group_start_lsn, &rec2); - if (len == (TRANSLOG_RECORD_HEADER_MAX_SIZE + 1)) + if (len < 0) /* EOF or error */ { fprintf(tracef, "Cannot find record where it should be\n"); goto err; @@ -267,7 +267,7 @@ int maria_apply_log(LSN lsn, my_bool apply, FILE *trace_file) goto err; } len= translog_read_next_record_header(&scanner2, &rec2); - if (len == (TRANSLOG_RECORD_HEADER_MAX_SIZE + 1)) + if (len < 0) /* EOF or error */ { fprintf(tracef, "Cannot find record where it should be\n"); goto err; @@ -294,9 +294,17 @@ int maria_apply_log(LSN lsn, my_bool apply, FILE *trace_file) } } len= translog_read_next_record_header(&scanner, &rec); - if (len == (TRANSLOG_RECORD_HEADER_MAX_SIZE + 1)) + if (len < 0) { - fprintf(tracef, "EOF on the log\n"); + switch (len) + { + case RECHEADER_READ_EOF: + fprintf(tracef, "EOF on the log\n"); + break; + case RECHEADER_READ_ERROR: + fprintf(stderr, "Error reading log\n"); + goto err; + } break; } } diff --git a/storage/maria/maria_read_log.c b/storage/maria/maria_read_log.c index 3ac3809ec04..c594fe20490 100644 --- a/storage/maria/maria_read_log.c +++ b/storage/maria/maria_read_log.c @@ -90,11 +90,12 @@ int main(int argc, char **argv) fprintf(stdout, "TRACE of the last maria_read_log\n"); if (maria_apply_log(lsn, opt_display_and_apply, stdout)) goto err; - fprintf(stdout, "SUCCESS\n"); + fprintf(stdout, "%s: SUCCESS\n", my_progname); goto end; err: /* don't touch anything more, in case we hit a bug */ + fprintf(stderr, "%s: FAILED\n", my_progname); exit(1); end: maria_end(); diff --git a/storage/maria/unittest/ma_test_loghandler-t.c b/storage/maria/unittest/ma_test_loghandler-t.c index a6bd53e949d..04459adeac8 100644 --- a/storage/maria/unittest/ma_test_loghandler-t.c +++ b/storage/maria/unittest/ma_test_loghandler-t.c @@ -360,8 +360,8 @@ int main(int argc __attribute__((unused)), char *argv[]) rc= 1; { - translog_size_t len= translog_read_record_header(first_lsn, &rec); - if (len == 0) + int len= translog_read_record_header(first_lsn, &rec); + if (len == RECHEADER_READ_ERROR) { fprintf(stderr, "translog_read_record_header failed (%d)\n", errno); goto err; @@ -392,13 +392,13 @@ int main(int argc __attribute__((unused)), char *argv[]) for (i= 1;; i++) { len= translog_read_next_record_header(&scanner, &rec); - if (len == 0) + if (len == RECHEADER_READ_ERROR) { fprintf(stderr, "1-%d translog_read_next_record_header failed (%d)\n", i, errno); goto err; } - if (rec.lsn == LSN_IMPOSSIBLE) + if (len == RECHEADER_READ_EOF) { if (i != ITERATIONS) { @@ -471,13 +471,13 @@ int main(int argc __attribute__((unused)), char *argv[]) translog_free_record_header(&rec); len= translog_read_next_record_header(&scanner, &rec); - if (len == 0) + if (len == RECHEADER_READ_ERROR) { fprintf(stderr, "1-%d translog_read_next_record_header (var) " "failed (%d)\n", i, errno); goto err; } - if (rec.lsn == LSN_IMPOSSIBLE) + if (len == RECHEADER_READ_EOF) { fprintf(stderr, "EOL met at the middle of iteration (first var) %u " "instead of beginning of %u\n", i, ITERATIONS); @@ -542,12 +542,12 @@ int main(int argc __attribute__((unused)), char *argv[]) { fprintf(stderr, "Incorrect LOGREC_VARIABLE_RECORD_2LSN_EXAMPLE " "data read(%d) " - "type %u, strid %u, len %lu != %lu + 14, hdr len: %u, " + "type %u, strid %u, len %lu != %lu + 14, hdr len: %d, " "ref1(%lu,0x%lx), ref2(%lu,0x%lx), " "lsn(%lu,0x%lx)\n", i, (uint) rec.type, (uint) rec.short_trid, (ulong) rec.record_length, (ulong) rec_len, - (uint) len, + len, (ulong) LSN_FILE_NO(ref1), (ulong) LSN_OFFSET(ref1), (ulong) LSN_FILE_NO(ref2), (ulong) LSN_OFFSET(ref2), (ulong) LSN_FILE_NO(rec.lsn), (ulong) LSN_OFFSET(rec.lsn)); @@ -566,13 +566,13 @@ int main(int argc __attribute__((unused)), char *argv[]) translog_free_record_header(&rec); len= translog_read_next_record_header(&scanner, &rec); - if (len == 0) + if (len == RECHEADER_READ_ERROR) { fprintf(stderr, "1-%d translog_read_next_record_header failed (%d)\n", i, errno); goto err; } - if (rec.lsn == LSN_IMPOSSIBLE) + if (len == RECHEADER_READ_EOF) { fprintf(stderr, "EOL met at the middle of iteration %u " "instead of beginning of %u\n", i, ITERATIONS); @@ -604,15 +604,15 @@ int main(int argc __attribute__((unused)), char *argv[]) if (rec.type != LOGREC_VARIABLE_RECORD_0LSN_EXAMPLE || rec.short_trid != (i % 0xFFFF) || rec.record_length != rec_len || - len != 9 || check_content(rec.header, len)) + len != 9 || check_content(rec.header, (uint)len)) { fprintf(stderr, "Incorrect LOGREC_VARIABLE_RECORD_0LSN_EXAMPLE " "data read(%d) " - "type %u, strid %u, len %lu != %lu, hdr len: %u, " + "type %u, strid %u, len %lu != %lu, hdr len: %d, " "lsn(%lu,0x%lx)\n", i, (uint) rec.type, (uint) rec.short_trid, (ulong) rec.record_length, (ulong) rec_len, - (uint) len, + len, (ulong) LSN_FILE_NO(rec.lsn), (ulong) LSN_OFFSET(rec.lsn)); goto err; } diff --git a/storage/maria/unittest/ma_test_loghandler_multigroup-t.c b/storage/maria/unittest/ma_test_loghandler_multigroup-t.c index 4c534ad4e05..cec7198ef3b 100644 --- a/storage/maria/unittest/ma_test_loghandler_multigroup-t.c +++ b/storage/maria/unittest/ma_test_loghandler_multigroup-t.c @@ -349,8 +349,8 @@ int main(int argc __attribute__((unused)), char *argv[]) rc= 1; { - translog_size_t len= translog_read_record_header(first_lsn, &rec); - if (len == 0) + int len= translog_read_record_header(first_lsn, &rec); + if (len == RECHEADER_READ_ERROR) { fprintf(stderr, "translog_read_record_header failed (%d)\n", errno); translog_free_record_header(&rec); @@ -383,14 +383,14 @@ int main(int argc __attribute__((unused)), char *argv[]) for (i= 1;; i++) { len= translog_read_next_record_header(&scanner, &rec); - if (len == 0) + if (len == RECHEADER_READ_ERROR) { fprintf(stderr, "1-%d translog_read_next_record_header failed (%d)\n", i, errno); translog_free_record_header(&rec); goto err; } - if (rec.lsn == LSN_IMPOSSIBLE) + if (len == RECHEADER_READ_EOF) { if (i != ITERATIONS) { @@ -464,13 +464,13 @@ int main(int argc __attribute__((unused)), char *argv[]) translog_free_record_header(&rec); len= translog_read_next_record_header(&scanner, &rec); - if (len == 0) + if (len == RECHEADER_READ_ERROR) { fprintf(stderr, "1-%d translog_read_next_record_header (var) " "failed (%d)\n", i, errno); goto err; } - if (rec.lsn == LSN_IMPOSSIBLE) + if (len == RECHEADER_READ_EOF) { fprintf(stderr, "EOL met at the middle of iteration (first var) %u " "instead of beginning of %u\n", i, ITERATIONS); @@ -490,7 +490,7 @@ int main(int argc __attribute__((unused)), char *argv[]) fprintf(stderr, "Incorrect LOGREC_VARIABLE_RECORD_1LSN_EXAMPLE " "data read(%d)" "type %u (%d), strid %u (%d), len %lu, %lu + 7 (%d), " - "hdr len: %u (%d), " + "hdr len: %d (%d), " "ref(%lu,0x%lx), lsn(%lu,0x%lx) (%d), content: %d\n", i, (uint) rec.type, rec.type !=LOGREC_VARIABLE_RECORD_1LSN_EXAMPLE, @@ -498,7 +498,7 @@ int main(int argc __attribute__((unused)), char *argv[]) rec.short_trid != (i % 0xFFFF), (ulong) rec.record_length, (ulong) rec_len, rec.record_length != rec_len + LSN_STORE_SIZE, - (uint) len, + len, len != 12, (ulong) LSN_FILE_NO(ref), (ulong) LSN_OFFSET(ref), (ulong) LSN_FILE_NO(rec.lsn), (ulong) LSN_OFFSET(rec.lsn), @@ -535,12 +535,12 @@ int main(int argc __attribute__((unused)), char *argv[]) { fprintf(stderr, "Incorrect LOGREC_VARIABLE_RECORD_2LSN_EXAMPLE " " data read(%d) " - "type %u, strid %u, len %lu != %lu + 14, hdr len: %u, " + "type %u, strid %u, len %lu != %lu + 14, hdr len: %d, " "ref1(%lu,0x%lx), ref2(%lu,0x%lx), " "lsn(%lu,0x%lx)\n", i, (uint) rec.type, (uint) rec.short_trid, (ulong) rec.record_length, (ulong) rec_len, - (uint) len, + len, (ulong) LSN_FILE_NO(ref1), (ulong) LSN_OFFSET(ref1), (ulong) LSN_FILE_NO(ref2), (ulong) LSN_OFFSET(ref2), (ulong) LSN_FILE_NO(rec.lsn), (ulong) LSN_OFFSET(rec.lsn)); @@ -561,14 +561,14 @@ int main(int argc __attribute__((unused)), char *argv[]) translog_free_record_header(&rec); len= translog_read_next_record_header(&scanner, &rec); - if (len == 0) + if (len == RECHEADER_READ_ERROR) { fprintf(stderr, "1-%d translog_read_next_record_header failed (%d)\n", i, errno); translog_free_record_header(&rec); goto err; } - if (rec.lsn == LSN_IMPOSSIBLE) + if (len == RECHEADER_READ_EOF) { fprintf(stderr, "EOL met at the middle of iteration %u " "instead of beginning of %u\n", i, ITERATIONS); @@ -606,11 +606,11 @@ int main(int argc __attribute__((unused)), char *argv[]) { fprintf(stderr, "Incorrect LOGREC_VARIABLE_RECORD_0LSN_EXAMPLE " "data read(%d) " - "type %u, strid %u, len %lu != %lu, hdr len: %u, " + "type %u, strid %u, len %lu != %lu, hdr len: %d, " "lsn(%lu,0x%lx)\n", i, (uint) rec.type, (uint) rec.short_trid, (ulong) rec.record_length, (ulong) rec_len, - (uint) len, + len, (ulong) LSN_FILE_NO(rec.lsn), (ulong) LSN_OFFSET(rec.lsn)); translog_free_record_header(&rec); goto err; diff --git a/storage/maria/unittest/ma_test_loghandler_multithread-t.c b/storage/maria/unittest/ma_test_loghandler_multithread-t.c index 7bb4a5aba77..86e66daca52 100644 --- a/storage/maria/unittest/ma_test_loghandler_multithread-t.c +++ b/storage/maria/unittest/ma_test_loghandler_multithread-t.c @@ -363,7 +363,8 @@ int main(int argc __attribute__((unused)), { uint indeces[WRITERS]; - uint index, len, stage; + uint index, stage; + int len; bzero(indeces, sizeof(uint) * WRITERS); bzero(indeces, sizeof(indeces)); @@ -377,14 +378,14 @@ int main(int argc __attribute__((unused)), { len= translog_read_next_record_header(&scanner, &rec); - if (len == 0) + if (len == RECHEADER_READ_ERROR) { fprintf(stderr, "1-%d translog_read_next_record_header failed (%d)\n", i, errno); translog_free_record_header(&rec); goto err; } - if (rec.lsn == LSN_IMPOSSIBLE) + if (len == RECHEADER_READ_EOF) { if (i != WRITERS * ITERATIONS * 2) { @@ -427,18 +428,18 @@ int main(int argc __attribute__((unused)), len != 9 || rec.record_length != lens[rec.short_trid][index] || cmp_translog_addr(lsns2[rec.short_trid][index], rec.lsn) != 0 || - check_content(rec.header, len)) + check_content(rec.header, (uint)len)) { fprintf(stderr, "Incorrect LOGREC_VARIABLE_RECORD_0LSN_EXAMPLE " "data read(%d) " "thread: %d, iteration %d, stage %d\n" - "type %u (%d), len %u, length %lu %lu (%d) " + "type %u (%d), len %d, length %lu %lu (%d) " "lsn(%lu,0x%lx) (%lu,0x%lx)\n", i, (uint) rec.short_trid, index, stage, (uint) rec.type, (rec.type != LOGREC_VARIABLE_RECORD_0LSN_EXAMPLE), - (uint) len, + len, (ulong) rec.record_length, lens[rec.short_trid][index], (rec.record_length != lens[rec.short_trid][index]), (ulong) LSN_FILE_NO(rec.lsn), -- cgit v1.2.1 From 562923e85f94d1cf9f0cbc1e20315c50c2040488 Mon Sep 17 00:00:00 2001 From: unknown Date: Tue, 31 Jul 2007 19:01:23 +0200 Subject: updating result file of ps_maria, it is now identical to the result of ps_2myisam.result. Thanks Serg for the help. mysql-test/r/ps_maria.result: Result file was probably out-of-date; it thus becomes identical to the result of ps_2myisam.result --- mysql-test/r/ps_maria.result | 537 ++++++++++++++++++++++--------------------- 1 file changed, 272 insertions(+), 265 deletions(-) diff --git a/mysql-test/r/ps_maria.result b/mysql-test/r/ps_maria.result index b1ea905c406..8416a670228 100644 --- a/mysql-test/r/ps_maria.result +++ b/mysql-test/r/ps_maria.result @@ -1158,7 +1158,7 @@ def table 253 64 2 Y 0 31 8 def type 253 10 3 Y 0 31 8 def possible_keys 253 4096 0 Y 0 31 8 def key 253 64 0 Y 0 31 8 -def key_len 253 4096 0 Y 128 31 63 +def key_len 253 1365 0 Y 0 31 8 def ref 253 1024 0 Y 0 31 8 def rows 8 10 1 Y 32928 0 63 def Extra 253 255 0 N 1 31 8 @@ -1917,38 +1917,38 @@ from t9 where c1= 1 ; 1 1 1 1 1 1 1 1 1 1 1.0000 1.0000 2004-02-29 2004-02-29 11:11:11 2004-02-29 11:11:11 11:11:11 2004 1 1 a 123456789a 123456789a123456789b123456789c tinyblob tinytext blob text mediumblob mediumtext longblob longtext one monday execute full_info ; Catalog Database Table Table_alias Column Column_alias Type Length Max length Is_null Flags Decimals Charsetnr -def @arg01 253 20 1 Y 128 0 63 -def @arg02 253 20 1 Y 128 0 63 -def @arg03 253 20 1 Y 128 0 63 -def @arg04 253 20 1 Y 128 0 63 -def @arg05 253 20 1 Y 128 0 63 -def @arg06 253 20 1 Y 128 0 63 -def @arg07 253 23 1 Y 128 31 63 -def @arg08 253 23 1 Y 128 31 63 -def @arg09 253 23 1 Y 128 31 63 -def @arg10 253 23 1 Y 128 31 63 -def @arg11 253 67 6 Y 128 30 63 -def @arg12 253 67 6 Y 128 30 63 -def @arg13 253 16777216 10 Y 128 31 63 -def @arg14 253 16777216 19 Y 128 31 63 -def @arg15 253 16777216 19 Y 128 31 63 -def @arg16 253 16777216 8 Y 128 31 63 -def @arg17 253 20 4 Y 128 0 63 -def @arg18 253 20 1 Y 128 0 63 -def @arg19 253 20 1 Y 128 0 63 -def @arg20 253 16777216 1 Y 0 31 8 -def @arg21 253 16777216 10 Y 0 31 8 -def @arg22 253 16777216 30 Y 0 31 8 -def @arg23 253 16777216 8 Y 128 31 63 -def @arg24 253 16777216 8 Y 0 31 8 -def @arg25 253 16777216 4 Y 128 31 63 -def @arg26 253 16777216 4 Y 0 31 8 -def @arg27 253 16777216 10 Y 128 31 63 -def @arg28 253 16777216 10 Y 0 31 8 -def @arg29 253 16777216 8 Y 128 31 63 -def @arg30 253 16777216 8 Y 0 31 8 -def @arg31 253 16777216 3 Y 0 31 8 -def @arg32 253 16777216 6 Y 0 31 8 +def @arg01 8 20 1 Y 32896 0 63 +def @arg02 8 20 1 Y 32896 0 63 +def @arg03 8 20 1 Y 32896 0 63 +def @arg04 8 20 1 Y 32896 0 63 +def @arg05 8 20 1 Y 32896 0 63 +def @arg06 8 20 1 Y 32896 0 63 +def @arg07 5 23 1 Y 32896 31 63 +def @arg08 5 23 1 Y 32896 31 63 +def @arg09 5 23 1 Y 32896 31 63 +def @arg10 5 23 1 Y 32896 31 63 +def @arg11 246 83 6 Y 128 30 63 +def @arg12 246 83 6 Y 128 30 63 +def @arg13 251 16777216 10 Y 128 31 63 +def @arg14 251 16777216 19 Y 128 31 63 +def @arg15 251 16777216 19 Y 128 31 63 +def @arg16 251 16777216 8 Y 128 31 63 +def @arg17 8 20 4 Y 32928 0 63 +def @arg18 8 20 1 Y 32896 0 63 +def @arg19 8 20 1 Y 32896 0 63 +def @arg20 251 16777216 1 Y 0 31 8 +def @arg21 251 16777216 10 Y 0 31 8 +def @arg22 251 16777216 30 Y 0 31 8 +def @arg23 251 16777216 8 Y 128 31 63 +def @arg24 251 16777216 8 Y 0 31 8 +def @arg25 251 16777216 4 Y 128 31 63 +def @arg26 251 16777216 4 Y 0 31 8 +def @arg27 251 16777216 10 Y 128 31 63 +def @arg28 251 16777216 10 Y 0 31 8 +def @arg29 251 16777216 8 Y 128 31 63 +def @arg30 251 16777216 8 Y 0 31 8 +def @arg31 251 16777216 3 Y 0 31 8 +def @arg32 251 16777216 6 Y 0 31 8 @arg01 @arg02 @arg03 @arg04 @arg05 @arg06 @arg07 @arg08 @arg09 @arg10 @arg11 @arg12 @arg13 @arg14 @arg15 @arg16 @arg17 @arg18 @arg19 @arg20 @arg21 @arg22 @arg23 @arg24 @arg25 @arg26 @arg27 @arg28 @arg29 @arg30 @arg31 @arg32 1 1 1 1 1 1 1 1 1 1 1.0000 1.0000 2004-02-29 2004-02-29 11:11:11 2004-02-29 11:11:11 11:11:11 2004 1 1 a 123456789a 123456789a123456789b123456789c tinyblob tinytext blob text mediumblob mediumtext longblob longtext one monday select @arg01:= c1, @arg02:= c2, @arg03:= c3, @arg04:= c4, @@ -1964,38 +1964,38 @@ from t9 where c1= 0 ; 0 NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL 1991-01-01 01:01:01 NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL execute full_info ; Catalog Database Table Table_alias Column Column_alias Type Length Max length Is_null Flags Decimals Charsetnr -def @arg01 253 20 1 Y 128 0 63 -def @arg02 253 20 0 Y 128 0 63 -def @arg03 253 20 0 Y 128 0 63 -def @arg04 253 20 0 Y 128 0 63 -def @arg05 253 20 0 Y 128 0 63 -def @arg06 253 20 0 Y 128 0 63 -def @arg07 253 23 0 Y 128 31 63 -def @arg08 253 23 0 Y 128 31 63 -def @arg09 253 23 0 Y 128 31 63 -def @arg10 253 23 0 Y 128 31 63 -def @arg11 253 67 0 Y 128 30 63 -def @arg12 253 67 0 Y 128 30 63 -def @arg13 253 16777216 0 Y 128 31 63 -def @arg14 253 16777216 0 Y 128 31 63 -def @arg15 253 16777216 19 Y 128 31 63 -def @arg16 253 16777216 0 Y 128 31 63 -def @arg17 253 20 0 Y 128 0 63 -def @arg18 253 20 0 Y 128 0 63 -def @arg19 253 20 0 Y 128 0 63 -def @arg20 253 16777216 0 Y 0 31 8 -def @arg21 253 16777216 0 Y 0 31 8 -def @arg22 253 16777216 0 Y 0 31 8 -def @arg23 253 16777216 0 Y 128 31 63 -def @arg24 253 16777216 0 Y 0 31 8 -def @arg25 253 16777216 0 Y 128 31 63 -def @arg26 253 16777216 0 Y 0 31 8 -def @arg27 253 16777216 0 Y 128 31 63 -def @arg28 253 16777216 0 Y 0 31 8 -def @arg29 253 16777216 0 Y 128 31 63 -def @arg30 253 16777216 0 Y 0 31 8 -def @arg31 253 16777216 0 Y 0 31 8 -def @arg32 253 16777216 0 Y 0 31 8 +def @arg01 8 20 1 Y 32896 0 63 +def @arg02 8 20 0 Y 32896 0 63 +def @arg03 8 20 0 Y 32896 0 63 +def @arg04 8 20 0 Y 32896 0 63 +def @arg05 8 20 0 Y 32896 0 63 +def @arg06 8 20 0 Y 32896 0 63 +def @arg07 5 23 0 Y 32896 31 63 +def @arg08 5 23 0 Y 32896 31 63 +def @arg09 5 23 0 Y 32896 31 63 +def @arg10 5 23 0 Y 32896 31 63 +def @arg11 246 83 0 Y 128 30 63 +def @arg12 246 83 0 Y 128 30 63 +def @arg13 251 16777216 0 Y 128 31 63 +def @arg14 251 16777216 0 Y 128 31 63 +def @arg15 251 16777216 19 Y 128 31 63 +def @arg16 251 16777216 0 Y 128 31 63 +def @arg17 8 20 0 Y 32928 0 63 +def @arg18 8 20 0 Y 32896 0 63 +def @arg19 8 20 0 Y 32896 0 63 +def @arg20 251 16777216 0 Y 0 31 8 +def @arg21 251 16777216 0 Y 0 31 8 +def @arg22 251 16777216 0 Y 0 31 8 +def @arg23 251 16777216 0 Y 128 31 63 +def @arg24 251 16777216 0 Y 0 31 8 +def @arg25 251 16777216 0 Y 128 31 63 +def @arg26 251 16777216 0 Y 0 31 8 +def @arg27 251 16777216 0 Y 128 31 63 +def @arg28 251 16777216 0 Y 0 31 8 +def @arg29 251 16777216 0 Y 128 31 63 +def @arg30 251 16777216 0 Y 0 31 8 +def @arg31 251 16777216 0 Y 0 31 8 +def @arg32 251 16777216 0 Y 0 31 8 @arg01 @arg02 @arg03 @arg04 @arg05 @arg06 @arg07 @arg08 @arg09 @arg10 @arg11 @arg12 @arg13 @arg14 @arg15 @arg16 @arg17 @arg18 @arg19 @arg20 @arg21 @arg22 @arg23 @arg24 @arg25 @arg26 @arg27 @arg28 @arg29 @arg30 @arg31 @arg32 0 NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL 1991-01-01 01:01:01 NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL prepare stmt1 from "select @@ -2014,38 +2014,38 @@ execute stmt1 using @my_key ; 1 1 1 1 1 1 1 1 1 1 1.0000 1.0000 2004-02-29 2004-02-29 11:11:11 2004-02-29 11:11:11 11:11:11 2004 1 1 a 123456789a 123456789a123456789b123456789c tinyblob tinytext blob text mediumblob mediumtext longblob longtext one monday execute full_info ; Catalog Database Table Table_alias Column Column_alias Type Length Max length Is_null Flags Decimals Charsetnr -def @arg01 253 20 1 Y 128 0 63 -def @arg02 253 20 1 Y 128 0 63 -def @arg03 253 20 1 Y 128 0 63 -def @arg04 253 20 1 Y 128 0 63 -def @arg05 253 20 1 Y 128 0 63 -def @arg06 253 20 1 Y 128 0 63 -def @arg07 253 23 1 Y 128 31 63 -def @arg08 253 23 1 Y 128 31 63 -def @arg09 253 23 1 Y 128 31 63 -def @arg10 253 23 1 Y 128 31 63 -def @arg11 253 67 6 Y 128 30 63 -def @arg12 253 67 6 Y 128 30 63 -def @arg13 253 16777216 10 Y 128 31 63 -def @arg14 253 16777216 19 Y 128 31 63 -def @arg15 253 16777216 19 Y 128 31 63 -def @arg16 253 16777216 8 Y 128 31 63 -def @arg17 253 20 4 Y 128 0 63 -def @arg18 253 20 1 Y 128 0 63 -def @arg19 253 20 1 Y 128 0 63 -def @arg20 253 16777216 1 Y 0 31 8 -def @arg21 253 16777216 10 Y 0 31 8 -def @arg22 253 16777216 30 Y 0 31 8 -def @arg23 253 16777216 8 Y 128 31 63 -def @arg24 253 16777216 8 Y 0 31 8 -def @arg25 253 16777216 4 Y 128 31 63 -def @arg26 253 16777216 4 Y 0 31 8 -def @arg27 253 16777216 10 Y 128 31 63 -def @arg28 253 16777216 10 Y 0 31 8 -def @arg29 253 16777216 8 Y 128 31 63 -def @arg30 253 16777216 8 Y 0 31 8 -def @arg31 253 16777216 3 Y 0 31 8 -def @arg32 253 16777216 6 Y 0 31 8 +def @arg01 8 20 1 Y 32896 0 63 +def @arg02 8 20 1 Y 32896 0 63 +def @arg03 8 20 1 Y 32896 0 63 +def @arg04 8 20 1 Y 32896 0 63 +def @arg05 8 20 1 Y 32896 0 63 +def @arg06 8 20 1 Y 32896 0 63 +def @arg07 5 23 1 Y 32896 31 63 +def @arg08 5 23 1 Y 32896 31 63 +def @arg09 5 23 1 Y 32896 31 63 +def @arg10 5 23 1 Y 32896 31 63 +def @arg11 246 83 6 Y 128 30 63 +def @arg12 246 83 6 Y 128 30 63 +def @arg13 251 16777216 10 Y 128 31 63 +def @arg14 251 16777216 19 Y 128 31 63 +def @arg15 251 16777216 19 Y 128 31 63 +def @arg16 251 16777216 8 Y 128 31 63 +def @arg17 8 20 4 Y 32928 0 63 +def @arg18 8 20 1 Y 32896 0 63 +def @arg19 8 20 1 Y 32896 0 63 +def @arg20 251 16777216 1 Y 0 31 8 +def @arg21 251 16777216 10 Y 0 31 8 +def @arg22 251 16777216 30 Y 0 31 8 +def @arg23 251 16777216 8 Y 128 31 63 +def @arg24 251 16777216 8 Y 0 31 8 +def @arg25 251 16777216 4 Y 128 31 63 +def @arg26 251 16777216 4 Y 0 31 8 +def @arg27 251 16777216 10 Y 128 31 63 +def @arg28 251 16777216 10 Y 0 31 8 +def @arg29 251 16777216 8 Y 128 31 63 +def @arg30 251 16777216 8 Y 0 31 8 +def @arg31 251 16777216 3 Y 0 31 8 +def @arg32 251 16777216 6 Y 0 31 8 @arg01 @arg02 @arg03 @arg04 @arg05 @arg06 @arg07 @arg08 @arg09 @arg10 @arg11 @arg12 @arg13 @arg14 @arg15 @arg16 @arg17 @arg18 @arg19 @arg20 @arg21 @arg22 @arg23 @arg24 @arg25 @arg26 @arg27 @arg28 @arg29 @arg30 @arg31 @arg32 1 1 1 1 1 1 1 1 1 1 1.0000 1.0000 2004-02-29 2004-02-29 11:11:11 2004-02-29 11:11:11 11:11:11 2004 1 1 a 123456789a 123456789a123456789b123456789c tinyblob tinytext blob text mediumblob mediumtext longblob longtext one monday set @my_key= 0 ; @@ -2054,38 +2054,38 @@ execute stmt1 using @my_key ; 0 NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL 1991-01-01 01:01:01 NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL execute full_info ; Catalog Database Table Table_alias Column Column_alias Type Length Max length Is_null Flags Decimals Charsetnr -def @arg01 253 20 1 Y 128 0 63 -def @arg02 253 20 0 Y 128 0 63 -def @arg03 253 20 0 Y 128 0 63 -def @arg04 253 20 0 Y 128 0 63 -def @arg05 253 20 0 Y 128 0 63 -def @arg06 253 20 0 Y 128 0 63 -def @arg07 253 23 0 Y 128 31 63 -def @arg08 253 23 0 Y 128 31 63 -def @arg09 253 23 0 Y 128 31 63 -def @arg10 253 23 0 Y 128 31 63 -def @arg11 253 67 0 Y 128 30 63 -def @arg12 253 67 0 Y 128 30 63 -def @arg13 253 16777216 0 Y 128 31 63 -def @arg14 253 16777216 0 Y 128 31 63 -def @arg15 253 16777216 19 Y 128 31 63 -def @arg16 253 16777216 0 Y 128 31 63 -def @arg17 253 20 0 Y 128 0 63 -def @arg18 253 20 0 Y 128 0 63 -def @arg19 253 20 0 Y 128 0 63 -def @arg20 253 16777216 0 Y 0 31 8 -def @arg21 253 16777216 0 Y 0 31 8 -def @arg22 253 16777216 0 Y 0 31 8 -def @arg23 253 16777216 0 Y 128 31 63 -def @arg24 253 16777216 0 Y 0 31 8 -def @arg25 253 16777216 0 Y 128 31 63 -def @arg26 253 16777216 0 Y 0 31 8 -def @arg27 253 16777216 0 Y 128 31 63 -def @arg28 253 16777216 0 Y 0 31 8 -def @arg29 253 16777216 0 Y 128 31 63 -def @arg30 253 16777216 0 Y 0 31 8 -def @arg31 253 16777216 0 Y 0 31 8 -def @arg32 253 16777216 0 Y 0 31 8 +def @arg01 8 20 1 Y 32896 0 63 +def @arg02 8 20 0 Y 32896 0 63 +def @arg03 8 20 0 Y 32896 0 63 +def @arg04 8 20 0 Y 32896 0 63 +def @arg05 8 20 0 Y 32896 0 63 +def @arg06 8 20 0 Y 32896 0 63 +def @arg07 5 23 0 Y 32896 31 63 +def @arg08 5 23 0 Y 32896 31 63 +def @arg09 5 23 0 Y 32896 31 63 +def @arg10 5 23 0 Y 32896 31 63 +def @arg11 246 83 0 Y 128 30 63 +def @arg12 246 83 0 Y 128 30 63 +def @arg13 251 16777216 0 Y 128 31 63 +def @arg14 251 16777216 0 Y 128 31 63 +def @arg15 251 16777216 19 Y 128 31 63 +def @arg16 251 16777216 0 Y 128 31 63 +def @arg17 8 20 0 Y 32928 0 63 +def @arg18 8 20 0 Y 32896 0 63 +def @arg19 8 20 0 Y 32896 0 63 +def @arg20 251 16777216 0 Y 0 31 8 +def @arg21 251 16777216 0 Y 0 31 8 +def @arg22 251 16777216 0 Y 0 31 8 +def @arg23 251 16777216 0 Y 128 31 63 +def @arg24 251 16777216 0 Y 0 31 8 +def @arg25 251 16777216 0 Y 128 31 63 +def @arg26 251 16777216 0 Y 0 31 8 +def @arg27 251 16777216 0 Y 128 31 63 +def @arg28 251 16777216 0 Y 0 31 8 +def @arg29 251 16777216 0 Y 128 31 63 +def @arg30 251 16777216 0 Y 0 31 8 +def @arg31 251 16777216 0 Y 0 31 8 +def @arg32 251 16777216 0 Y 0 31 8 @arg01 @arg02 @arg03 @arg04 @arg05 @arg06 @arg07 @arg08 @arg09 @arg10 @arg11 @arg12 @arg13 @arg14 @arg15 @arg16 @arg17 @arg18 @arg19 @arg20 @arg21 @arg22 @arg23 @arg24 @arg25 @arg26 @arg27 @arg28 @arg29 @arg30 @arg31 @arg32 0 NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL 1991-01-01 01:01:01 NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL prepare stmt1 from "select ? := c1 from t9 where c1= 1" ; @@ -2102,38 +2102,38 @@ into @arg01, @arg02, @arg03, @arg04, @arg05, @arg06, @arg07, @arg08, from t9 where c1= 1 ; execute full_info ; Catalog Database Table Table_alias Column Column_alias Type Length Max length Is_null Flags Decimals Charsetnr -def @arg01 253 20 1 Y 128 0 63 -def @arg02 253 20 1 Y 128 0 63 -def @arg03 253 20 1 Y 128 0 63 -def @arg04 253 20 1 Y 128 0 63 -def @arg05 253 20 1 Y 128 0 63 -def @arg06 253 20 1 Y 128 0 63 -def @arg07 253 23 1 Y 128 31 63 -def @arg08 253 23 1 Y 128 31 63 -def @arg09 253 23 1 Y 128 31 63 -def @arg10 253 23 1 Y 128 31 63 -def @arg11 253 67 6 Y 128 30 63 -def @arg12 253 67 6 Y 128 30 63 -def @arg13 253 16777216 10 Y 128 31 63 -def @arg14 253 16777216 19 Y 128 31 63 -def @arg15 253 16777216 19 Y 128 31 63 -def @arg16 253 16777216 8 Y 128 31 63 -def @arg17 253 20 4 Y 128 0 63 -def @arg18 253 20 1 Y 128 0 63 -def @arg19 253 20 1 Y 128 0 63 -def @arg20 253 16777216 1 Y 0 31 8 -def @arg21 253 16777216 10 Y 0 31 8 -def @arg22 253 16777216 30 Y 0 31 8 -def @arg23 253 16777216 8 Y 128 31 63 -def @arg24 253 16777216 8 Y 0 31 8 -def @arg25 253 16777216 4 Y 128 31 63 -def @arg26 253 16777216 4 Y 0 31 8 -def @arg27 253 16777216 10 Y 128 31 63 -def @arg28 253 16777216 10 Y 0 31 8 -def @arg29 253 16777216 8 Y 128 31 63 -def @arg30 253 16777216 8 Y 0 31 8 -def @arg31 253 16777216 3 Y 0 31 8 -def @arg32 253 16777216 6 Y 0 31 8 +def @arg01 8 20 1 Y 32896 0 63 +def @arg02 8 20 1 Y 32896 0 63 +def @arg03 8 20 1 Y 32896 0 63 +def @arg04 8 20 1 Y 32896 0 63 +def @arg05 8 20 1 Y 32896 0 63 +def @arg06 8 20 1 Y 32896 0 63 +def @arg07 5 23 1 Y 32896 31 63 +def @arg08 5 23 1 Y 32896 31 63 +def @arg09 5 23 1 Y 32896 31 63 +def @arg10 5 23 1 Y 32896 31 63 +def @arg11 246 83 6 Y 128 30 63 +def @arg12 246 83 6 Y 128 30 63 +def @arg13 251 16777216 10 Y 128 31 63 +def @arg14 251 16777216 19 Y 128 31 63 +def @arg15 251 16777216 19 Y 128 31 63 +def @arg16 251 16777216 8 Y 128 31 63 +def @arg17 8 20 4 Y 32928 0 63 +def @arg18 8 20 1 Y 32896 0 63 +def @arg19 8 20 1 Y 32896 0 63 +def @arg20 251 16777216 1 Y 0 31 8 +def @arg21 251 16777216 10 Y 0 31 8 +def @arg22 251 16777216 30 Y 0 31 8 +def @arg23 251 16777216 8 Y 128 31 63 +def @arg24 251 16777216 8 Y 0 31 8 +def @arg25 251 16777216 4 Y 128 31 63 +def @arg26 251 16777216 4 Y 0 31 8 +def @arg27 251 16777216 10 Y 128 31 63 +def @arg28 251 16777216 10 Y 0 31 8 +def @arg29 251 16777216 8 Y 128 31 63 +def @arg30 251 16777216 8 Y 0 31 8 +def @arg31 251 16777216 3 Y 0 31 8 +def @arg32 251 16777216 6 Y 0 31 8 @arg01 @arg02 @arg03 @arg04 @arg05 @arg06 @arg07 @arg08 @arg09 @arg10 @arg11 @arg12 @arg13 @arg14 @arg15 @arg16 @arg17 @arg18 @arg19 @arg20 @arg21 @arg22 @arg23 @arg24 @arg25 @arg26 @arg27 @arg28 @arg29 @arg30 @arg31 @arg32 1 1 1 1 1 1 1 1 1 1 1.0000 1.0000 2004-02-29 2004-02-29 11:11:11 2004-02-29 11:11:11 11:11:11 2004 1 1 a 123456789a 123456789a123456789b123456789c tinyblob tinytext blob text mediumblob mediumtext longblob longtext one monday select c1, c2, c3, c4, c5, c6, c7, c8, c9, c10, c11, c12, @@ -2146,38 +2146,38 @@ into @arg01, @arg02, @arg03, @arg04, @arg05, @arg06, @arg07, @arg08, from t9 where c1= 0 ; execute full_info ; Catalog Database Table Table_alias Column Column_alias Type Length Max length Is_null Flags Decimals Charsetnr -def @arg01 253 20 1 Y 128 0 63 -def @arg02 253 20 0 Y 128 0 63 -def @arg03 253 20 0 Y 128 0 63 -def @arg04 253 20 0 Y 128 0 63 -def @arg05 253 20 0 Y 128 0 63 -def @arg06 253 20 0 Y 128 0 63 -def @arg07 253 23 0 Y 128 31 63 -def @arg08 253 23 0 Y 128 31 63 -def @arg09 253 23 0 Y 128 31 63 -def @arg10 253 23 0 Y 128 31 63 -def @arg11 253 67 0 Y 128 30 63 -def @arg12 253 67 0 Y 128 30 63 -def @arg13 253 16777216 0 Y 128 31 63 -def @arg14 253 16777216 0 Y 128 31 63 -def @arg15 253 16777216 19 Y 128 31 63 -def @arg16 253 16777216 0 Y 128 31 63 -def @arg17 253 20 0 Y 128 0 63 -def @arg18 253 20 0 Y 128 0 63 -def @arg19 253 20 0 Y 128 0 63 -def @arg20 253 16777216 0 Y 0 31 8 -def @arg21 253 16777216 0 Y 0 31 8 -def @arg22 253 16777216 0 Y 0 31 8 -def @arg23 253 16777216 0 Y 128 31 63 -def @arg24 253 16777216 0 Y 0 31 8 -def @arg25 253 16777216 0 Y 128 31 63 -def @arg26 253 16777216 0 Y 0 31 8 -def @arg27 253 16777216 0 Y 128 31 63 -def @arg28 253 16777216 0 Y 0 31 8 -def @arg29 253 16777216 0 Y 128 31 63 -def @arg30 253 16777216 0 Y 0 31 8 -def @arg31 253 16777216 0 Y 0 31 8 -def @arg32 253 16777216 0 Y 0 31 8 +def @arg01 8 20 1 Y 32896 0 63 +def @arg02 8 20 0 Y 32896 0 63 +def @arg03 8 20 0 Y 32896 0 63 +def @arg04 8 20 0 Y 32896 0 63 +def @arg05 8 20 0 Y 32896 0 63 +def @arg06 8 20 0 Y 32896 0 63 +def @arg07 5 23 0 Y 32896 31 63 +def @arg08 5 23 0 Y 32896 31 63 +def @arg09 5 23 0 Y 32896 31 63 +def @arg10 5 23 0 Y 32896 31 63 +def @arg11 246 83 0 Y 128 30 63 +def @arg12 246 83 0 Y 128 30 63 +def @arg13 251 16777216 0 Y 128 31 63 +def @arg14 251 16777216 0 Y 128 31 63 +def @arg15 251 16777216 19 Y 128 31 63 +def @arg16 251 16777216 0 Y 128 31 63 +def @arg17 8 20 0 Y 32928 0 63 +def @arg18 8 20 0 Y 32896 0 63 +def @arg19 8 20 0 Y 32896 0 63 +def @arg20 251 16777216 0 Y 0 31 8 +def @arg21 251 16777216 0 Y 0 31 8 +def @arg22 251 16777216 0 Y 0 31 8 +def @arg23 251 16777216 0 Y 128 31 63 +def @arg24 251 16777216 0 Y 0 31 8 +def @arg25 251 16777216 0 Y 128 31 63 +def @arg26 251 16777216 0 Y 0 31 8 +def @arg27 251 16777216 0 Y 128 31 63 +def @arg28 251 16777216 0 Y 0 31 8 +def @arg29 251 16777216 0 Y 128 31 63 +def @arg30 251 16777216 0 Y 0 31 8 +def @arg31 251 16777216 0 Y 0 31 8 +def @arg32 251 16777216 0 Y 0 31 8 @arg01 @arg02 @arg03 @arg04 @arg05 @arg06 @arg07 @arg08 @arg09 @arg10 @arg11 @arg12 @arg13 @arg14 @arg15 @arg16 @arg17 @arg18 @arg19 @arg20 @arg21 @arg22 @arg23 @arg24 @arg25 @arg26 @arg27 @arg28 @arg29 @arg30 @arg31 @arg32 0 NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL 1991-01-01 01:01:01 NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL prepare stmt1 from "select c1, c2, c3, c4, c5, c6, c7, c8, c9, c10, c11, c12, @@ -2192,76 +2192,76 @@ set @my_key= 1 ; execute stmt1 using @my_key ; execute full_info ; Catalog Database Table Table_alias Column Column_alias Type Length Max length Is_null Flags Decimals Charsetnr -def @arg01 253 20 1 Y 128 0 63 -def @arg02 253 20 1 Y 128 0 63 -def @arg03 253 20 1 Y 128 0 63 -def @arg04 253 20 1 Y 128 0 63 -def @arg05 253 20 1 Y 128 0 63 -def @arg06 253 20 1 Y 128 0 63 -def @arg07 253 23 1 Y 128 31 63 -def @arg08 253 23 1 Y 128 31 63 -def @arg09 253 23 1 Y 128 31 63 -def @arg10 253 23 1 Y 128 31 63 -def @arg11 253 67 6 Y 128 30 63 -def @arg12 253 67 6 Y 128 30 63 -def @arg13 253 16777216 10 Y 128 31 63 -def @arg14 253 16777216 19 Y 128 31 63 -def @arg15 253 16777216 19 Y 128 31 63 -def @arg16 253 16777216 8 Y 128 31 63 -def @arg17 253 20 4 Y 128 0 63 -def @arg18 253 20 1 Y 128 0 63 -def @arg19 253 20 1 Y 128 0 63 -def @arg20 253 16777216 1 Y 0 31 8 -def @arg21 253 16777216 10 Y 0 31 8 -def @arg22 253 16777216 30 Y 0 31 8 -def @arg23 253 16777216 8 Y 128 31 63 -def @arg24 253 16777216 8 Y 0 31 8 -def @arg25 253 16777216 4 Y 128 31 63 -def @arg26 253 16777216 4 Y 0 31 8 -def @arg27 253 16777216 10 Y 128 31 63 -def @arg28 253 16777216 10 Y 0 31 8 -def @arg29 253 16777216 8 Y 128 31 63 -def @arg30 253 16777216 8 Y 0 31 8 -def @arg31 253 16777216 3 Y 0 31 8 -def @arg32 253 16777216 6 Y 0 31 8 +def @arg01 8 20 1 Y 32896 0 63 +def @arg02 8 20 1 Y 32896 0 63 +def @arg03 8 20 1 Y 32896 0 63 +def @arg04 8 20 1 Y 32896 0 63 +def @arg05 8 20 1 Y 32896 0 63 +def @arg06 8 20 1 Y 32896 0 63 +def @arg07 5 23 1 Y 32896 31 63 +def @arg08 5 23 1 Y 32896 31 63 +def @arg09 5 23 1 Y 32896 31 63 +def @arg10 5 23 1 Y 32896 31 63 +def @arg11 246 83 6 Y 128 30 63 +def @arg12 246 83 6 Y 128 30 63 +def @arg13 251 16777216 10 Y 128 31 63 +def @arg14 251 16777216 19 Y 128 31 63 +def @arg15 251 16777216 19 Y 128 31 63 +def @arg16 251 16777216 8 Y 128 31 63 +def @arg17 8 20 4 Y 32928 0 63 +def @arg18 8 20 1 Y 32896 0 63 +def @arg19 8 20 1 Y 32896 0 63 +def @arg20 251 16777216 1 Y 0 31 8 +def @arg21 251 16777216 10 Y 0 31 8 +def @arg22 251 16777216 30 Y 0 31 8 +def @arg23 251 16777216 8 Y 128 31 63 +def @arg24 251 16777216 8 Y 0 31 8 +def @arg25 251 16777216 4 Y 128 31 63 +def @arg26 251 16777216 4 Y 0 31 8 +def @arg27 251 16777216 10 Y 128 31 63 +def @arg28 251 16777216 10 Y 0 31 8 +def @arg29 251 16777216 8 Y 128 31 63 +def @arg30 251 16777216 8 Y 0 31 8 +def @arg31 251 16777216 3 Y 0 31 8 +def @arg32 251 16777216 6 Y 0 31 8 @arg01 @arg02 @arg03 @arg04 @arg05 @arg06 @arg07 @arg08 @arg09 @arg10 @arg11 @arg12 @arg13 @arg14 @arg15 @arg16 @arg17 @arg18 @arg19 @arg20 @arg21 @arg22 @arg23 @arg24 @arg25 @arg26 @arg27 @arg28 @arg29 @arg30 @arg31 @arg32 1 1 1 1 1 1 1 1 1 1 1.0000 1.0000 2004-02-29 2004-02-29 11:11:11 2004-02-29 11:11:11 11:11:11 2004 1 1 a 123456789a 123456789a123456789b123456789c tinyblob tinytext blob text mediumblob mediumtext longblob longtext one monday set @my_key= 0 ; execute stmt1 using @my_key ; execute full_info ; Catalog Database Table Table_alias Column Column_alias Type Length Max length Is_null Flags Decimals Charsetnr -def @arg01 253 20 1 Y 128 0 63 -def @arg02 253 20 0 Y 128 0 63 -def @arg03 253 20 0 Y 128 0 63 -def @arg04 253 20 0 Y 128 0 63 -def @arg05 253 20 0 Y 128 0 63 -def @arg06 253 20 0 Y 128 0 63 -def @arg07 253 23 0 Y 128 31 63 -def @arg08 253 23 0 Y 128 31 63 -def @arg09 253 23 0 Y 128 31 63 -def @arg10 253 23 0 Y 128 31 63 -def @arg11 253 67 0 Y 128 30 63 -def @arg12 253 67 0 Y 128 30 63 -def @arg13 253 16777216 0 Y 128 31 63 -def @arg14 253 16777216 0 Y 128 31 63 -def @arg15 253 16777216 19 Y 128 31 63 -def @arg16 253 16777216 0 Y 128 31 63 -def @arg17 253 20 0 Y 128 0 63 -def @arg18 253 20 0 Y 128 0 63 -def @arg19 253 20 0 Y 128 0 63 -def @arg20 253 16777216 0 Y 0 31 8 -def @arg21 253 16777216 0 Y 0 31 8 -def @arg22 253 16777216 0 Y 0 31 8 -def @arg23 253 16777216 0 Y 128 31 63 -def @arg24 253 16777216 0 Y 0 31 8 -def @arg25 253 16777216 0 Y 128 31 63 -def @arg26 253 16777216 0 Y 0 31 8 -def @arg27 253 16777216 0 Y 128 31 63 -def @arg28 253 16777216 0 Y 0 31 8 -def @arg29 253 16777216 0 Y 128 31 63 -def @arg30 253 16777216 0 Y 0 31 8 -def @arg31 253 16777216 0 Y 0 31 8 -def @arg32 253 16777216 0 Y 0 31 8 +def @arg01 8 20 1 Y 32896 0 63 +def @arg02 8 20 0 Y 32896 0 63 +def @arg03 8 20 0 Y 32896 0 63 +def @arg04 8 20 0 Y 32896 0 63 +def @arg05 8 20 0 Y 32896 0 63 +def @arg06 8 20 0 Y 32896 0 63 +def @arg07 5 23 0 Y 32896 31 63 +def @arg08 5 23 0 Y 32896 31 63 +def @arg09 5 23 0 Y 32896 31 63 +def @arg10 5 23 0 Y 32896 31 63 +def @arg11 246 83 0 Y 128 30 63 +def @arg12 246 83 0 Y 128 30 63 +def @arg13 251 16777216 0 Y 128 31 63 +def @arg14 251 16777216 0 Y 128 31 63 +def @arg15 251 16777216 19 Y 128 31 63 +def @arg16 251 16777216 0 Y 128 31 63 +def @arg17 8 20 0 Y 32928 0 63 +def @arg18 8 20 0 Y 32896 0 63 +def @arg19 8 20 0 Y 32896 0 63 +def @arg20 251 16777216 0 Y 0 31 8 +def @arg21 251 16777216 0 Y 0 31 8 +def @arg22 251 16777216 0 Y 0 31 8 +def @arg23 251 16777216 0 Y 128 31 63 +def @arg24 251 16777216 0 Y 0 31 8 +def @arg25 251 16777216 0 Y 128 31 63 +def @arg26 251 16777216 0 Y 0 31 8 +def @arg27 251 16777216 0 Y 128 31 63 +def @arg28 251 16777216 0 Y 0 31 8 +def @arg29 251 16777216 0 Y 128 31 63 +def @arg30 251 16777216 0 Y 0 31 8 +def @arg31 251 16777216 0 Y 0 31 8 +def @arg32 251 16777216 0 Y 0 31 8 @arg01 @arg02 @arg03 @arg04 @arg05 @arg06 @arg07 @arg08 @arg09 @arg10 @arg11 @arg12 @arg13 @arg14 @arg15 @arg16 @arg17 @arg18 @arg19 @arg20 @arg21 @arg22 @arg23 @arg24 @arg25 @arg26 @arg27 @arg28 @arg29 @arg30 @arg31 @arg32 0 NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL 1991-01-01 01:01:01 NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL NULL prepare stmt1 from "select c1 into ? from t9 where c1= 1" ; @@ -2961,20 +2961,26 @@ delete from t9 ; test_sequence -- insert into date/time columns -- Warnings: +Note 1265 Data truncated for column 'c13' at row 1 Warning 1265 Data truncated for column 'c17' at row 1 Warnings: +Note 1265 Data truncated for column 'c13' at row 1 Warning 1265 Data truncated for column 'c17' at row 1 Warnings: +Note 1265 Data truncated for column 'c13' at row 1 Warning 1265 Data truncated for column 'c17' at row 1 Warnings: +Note 1265 Data truncated for column 'c13' at row 1 Warning 1265 Data truncated for column 'c17' at row 1 Warnings: Warning 1265 Data truncated for column 'c17' at row 1 Warnings: +Note 1265 Data truncated for column 'c13' at row 1 Warning 1265 Data truncated for column 'c17' at row 1 Warnings: Warning 1265 Data truncated for column 'c17' at row 1 Warnings: +Note 1265 Data truncated for column 'c13' at row 1 Warning 1265 Data truncated for column 'c17' at row 1 Warnings: Warning 1264 Out of range value for column 'c13' at row 1 @@ -3005,6 +3011,7 @@ Warning 1265 Data truncated for column 'c15' at row 1 Warning 1264 Out of range value for column 'c16' at row 1 Warning 1264 Out of range value for column 'c17' at row 1 Warnings: +Note 1265 Data truncated for column 'c13' at row 1 Warning 1265 Data truncated for column 'c15' at row 1 Warning 1264 Out of range value for column 'c16' at row 1 Warning 1264 Out of range value for column 'c17' at row 1 @@ -3046,31 +3053,31 @@ test_sequence -- select .. where date/time column = .. -- set @arg00= '1991-01-01 01:01:01' ; select 'true' as found from t9 -where c1= 20 and c13= '1991-01-01 01:01:01' and c14= '1991-01-01 01:01:01' and +where c1= 20 and c13= CAST('1991-01-01 01:01:01' AS DATE) and c14= '1991-01-01 01:01:01' and c15= '1991-01-01 01:01:01' and c16= '1991-01-01 01:01:01' and c17= '1991-01-01 01:01:01' ; found true select 'true' as found from t9 -where c1= 20 and c13= @arg00 and c14= @arg00 and c15= @arg00 and c16= @arg00 +where c1= 20 and c13= CAST(@arg00 AS DATE) and c14= @arg00 and c15= @arg00 and c16= @arg00 and c17= @arg00 ; found true prepare stmt1 from "select 'true' as found from t9 -where c1= 20 and c13= '1991-01-01 01:01:01' and c14= '1991-01-01 01:01:01' and +where c1= 20 and c13= CAST('1991-01-01 01:01:01' AS DATE) and c14= '1991-01-01 01:01:01' and c15= '1991-01-01 01:01:01' and c16= '1991-01-01 01:01:01' and c17= '1991-01-01 01:01:01'" ; execute stmt1 ; found true prepare stmt1 from "select 'true' as found from t9 -where c1= 20 and c13= ? and c14= ? and c15= ? and c16= ? and c17= ?" ; +where c1= 20 and c13= CAST(? AS DATE) and c14= ? and c15= ? and c16= ? and c17= ?" ; execute stmt1 using @arg00, @arg00, @arg00, @arg00, @arg00 ; found true set @arg00= CAST('1991-01-01 01:01:01' as datetime) ; select 'true' as found from t9 -where c1= 20 and c13= CAST('1991-01-01 01:01:01' as datetime) and +where c1= 20 and c13= CAST('1991-01-01 00:00:00' as datetime) and c14= CAST('1991-01-01 01:01:01' as datetime) and c15= CAST('1991-01-01 01:01:01' as datetime) and c16= CAST('1991-01-01 01:01:01' as datetime) and @@ -3078,12 +3085,12 @@ c17= CAST('1991-01-01 01:01:01' as datetime) ; found true select 'true' as found from t9 -where c1= 20 and c13= @arg00 and c14= @arg00 and c15= @arg00 and c16= @arg00 +where c1= 20 and c13= CAST(@arg00 AS DATE) and c14= @arg00 and c15= @arg00 and c16= @arg00 and c17= @arg00 ; found true prepare stmt1 from "select 'true' as found from t9 -where c1= 20 and c13= CAST('1991-01-01 01:01:01' as datetime) and +where c1= 20 and c13= CAST('1991-01-01 00:00:00' as datetime) and c14= CAST('1991-01-01 01:01:01' as datetime) and c15= CAST('1991-01-01 01:01:01' as datetime) and c16= CAST('1991-01-01 01:01:01' as datetime) and @@ -3092,7 +3099,7 @@ execute stmt1 ; found true prepare stmt1 from "select 'true' as found from t9 -where c1= 20 and c13= ? and c14= ? and c15= ? and c16= ? and c17= ?" ; +where c1= 20 and c13= CAST(? AS DATE) and c14= ? and c15= ? and c16= ? and c17= ?" ; execute stmt1 using @arg00, @arg00, @arg00, @arg00, @arg00 ; found true -- cgit v1.2.1 From 8dcd83d0daae46f345202b583b8a1b1478857fb7 Mon Sep 17 00:00:00 2001 From: unknown Date: Tue, 31 Jul 2007 19:05:19 +0200 Subject: testcase for a bug not yet fixed. Once fixed, the test should fail in EXPLAIN's output. mysql-test/t/maria.test: adding a test for "incorrect key file" error in UPDATE with row_format=dynamic (does not happen with row_format=page) --- mysql-test/t/maria.test | 11 +++++++++++ 1 file changed, 11 insertions(+) diff --git a/mysql-test/t/maria.test b/mysql-test/t/maria.test index 5eddf0dd300..b3e0787472b 100644 --- a/mysql-test/t/maria.test +++ b/mysql-test/t/maria.test @@ -431,6 +431,17 @@ update t1 set a='bbb' where a='aaa'; select concat(a,'.') from t1; drop table t1; +# test again but with dynamic format +create table t1 ( a text not null, key a (a(20))) row_format=dynamic; +insert into t1 values ('aaa '),('aaa'),('aa'); +check table t1; +repair table t1; +select concat(a,'.') from t1 where a='aaa'; +select concat(a,'.') from t1 where binary a='aaa'; +update t1 set a='bbb' where a='aaa'; +select concat(a,'.') from t1; +drop table t1; + # # More space testing # -- cgit v1.2.1 From 372b4ed49a896df3e00244dadbdcedd976e1bd1b Mon Sep 17 00:00:00 2001 From: unknown Date: Wed, 1 Aug 2007 15:52:57 +0200 Subject: WL#3072 Maria recovery: preparation to enable recoverability of the table's state, more exactly info->state->checksum. This will require that info->state->checksum be updated in a inwrite_hook when writing an UNDO record, thus info->cur_row.checksum needs to be accessible to inwrite_hook, so we make translog_write_record() accept a MARIA_HA* (info) instead of MARIA_SHARE* (info->s); with this, we will be able to access info->cur_row.checksum. Old code which needed the MARIA_SHARE can derive it from MARIA_HA. Fix for typos and compiler warnings. storage/maria/ma_blockrec.c: fix for new loghandler API. Removing strange lines (how could gcc accept that?) storage/maria/ma_check.c: fix for new loghandler API storage/maria/ma_delete_all.c: fix for new loghandler API storage/maria/ma_loghandler.c: functions now take a MARIA_HA in argument, this is more powerful than a MARIA_SHARE (MARIA_SHARE can be derived from MARIA_HA, not the other way around). MARIA_HA will be needed to allow recoverability of the table's state. Fixing wrong DBUG_PRINT ('i' is not the id). When writing the LOGREC_FILE_ID, we don't have a MARIA_HA around, so we cannot ask translog_write_record() to store the id for us; we thus store the file's id by ourselves. Alternative would have been to pass MARIA_HA to translog_assign_id_to_share() but I didn't like it. storage/maria/ma_loghandler.h: new loghandler API storage/maria/tablockman.c: fix for compiler warning (intptr is int on my machine) --- storage/maria/ma_blockrec.c | 26 +++++++++--------- storage/maria/ma_check.c | 2 +- storage/maria/ma_delete_all.c | 2 +- storage/maria/ma_loghandler.c | 61 +++++++++++++++++++++++-------------------- storage/maria/ma_loghandler.h | 8 +++--- storage/maria/tablockman.c | 4 +-- 6 files changed, 53 insertions(+), 50 deletions(-) diff --git a/storage/maria/ma_blockrec.c b/storage/maria/ma_blockrec.c index 37400560197..ed146ff1952 100644 --- a/storage/maria/ma_blockrec.c +++ b/storage/maria/ma_blockrec.c @@ -1143,7 +1143,7 @@ static my_bool write_tail(MARIA_HA *info, MARIA_BITMAP_BLOCK *block, uchar *row_part, uint length) { - MARIA_SHARE *share= share= info->s; + MARIA_SHARE *share= info->s; MARIA_PINNED_PAGE page_link; uint block_size= share->block_size, empty_space; struct st_row_pos_info row_pos; @@ -1179,7 +1179,7 @@ static my_bool write_tail(MARIA_HA *info, log_array[TRANSLOG_INTERNAL_PARTS + 1].str= (char*) row_pos.data; log_array[TRANSLOG_INTERNAL_PARTS + 1].length= length; if (translog_write_record(&lsn, LOGREC_REDO_INSERT_ROW_TAIL, - info->trn, share, sizeof(log_data) + length, + info->trn, info, sizeof(log_data) + length, TRANSLOG_INTERNAL_PARTS + 2, log_array, log_data)) DBUG_RETURN(1); @@ -1261,7 +1261,7 @@ static my_bool write_full_pages(MARIA_HA *info, uchar *data, ulong length) { my_off_t page; - MARIA_SHARE *share= share= info->s; + MARIA_SHARE *share= info->s; uint block_size= share->block_size; uint data_size= FULL_PAGE_SIZE(block_size); uchar *buff= info->keyread_buff; @@ -1433,7 +1433,7 @@ static my_bool free_full_pages(MARIA_HA *info, MARIA_ROW *row) log_array[TRANSLOG_INTERNAL_PARTS + 1].str= row->extents; log_array[TRANSLOG_INTERNAL_PARTS + 1].length= extents_length; if (translog_write_record(&lsn, LOGREC_REDO_PURGE_BLOCKS, info->trn, - info->s, sizeof(log_data) + extents_length, + info, sizeof(log_data) + extents_length, TRANSLOG_INTERNAL_PARTS + 2, log_array, log_data)) DBUG_RETURN(1); @@ -1479,7 +1479,7 @@ static my_bool free_full_page_range(MARIA_HA *info, ulonglong page, uint count) log_array[TRANSLOG_INTERNAL_PARTS + 0].length= sizeof(log_data); if (translog_write_record(&lsn, LOGREC_REDO_PURGE_BLOCKS, - info->trn, info->s, sizeof(log_data), + info->trn, info, sizeof(log_data), TRANSLOG_INTERNAL_PARTS + 1, log_array, log_data)) res= 1; @@ -1994,7 +1994,7 @@ static my_bool write_block_record(MARIA_HA *info, log_array[TRANSLOG_INTERNAL_PARTS + 1].str= (char*) row_pos->data; log_array[TRANSLOG_INTERNAL_PARTS + 1].length= data_length; if (translog_write_record(&lsn, LOGREC_REDO_INSERT_ROW_HEAD, info->trn, - share, sizeof(log_data) + data_length, + info, sizeof(log_data) + data_length, TRANSLOG_INTERNAL_PARTS + 2, log_array, log_data)) goto disk_err; @@ -2111,7 +2111,7 @@ static my_bool write_block_record(MARIA_HA *info, /* trn->rec_lsn is already set earlier in this function */ error= translog_write_record(&lsn, LOGREC_REDO_INSERT_ROW_BLOBS, - info->trn, share, log_entry_length, + info->trn, info, log_entry_length, (uint) (log_array_pos - log_array), log_array, log_data); if (log_array != tmp_log_array) @@ -2142,7 +2142,7 @@ static my_bool write_block_record(MARIA_HA *info, { /* Write UNDO log record for the INSERT */ if (translog_write_record(&lsn, LOGREC_UNDO_ROW_INSERT, - info->trn, share, sizeof(log_data), + info->trn, info, sizeof(log_data), TRANSLOG_INTERNAL_PARTS + 1, log_array, log_data + LSN_STORE_SIZE)) goto disk_err; @@ -2157,7 +2157,7 @@ static my_bool write_block_record(MARIA_HA *info, TRANSLOG_INTERNAL_PARTS + 1, &row_parts_count); if (translog_write_record(&lsn, LOGREC_UNDO_ROW_UPDATE, info->trn, - share, sizeof(log_data) + row_length, + info, sizeof(log_data) + row_length, TRANSLOG_INTERNAL_PARTS + 1 + row_parts_count, log_array, log_data + LSN_STORE_SIZE)) goto disk_err; @@ -2376,7 +2376,7 @@ my_bool _ma_write_abort_block_record(MARIA_HA *info) log_array[TRANSLOG_INTERNAL_PARTS + 0].str= (char*) log_data; log_array[TRANSLOG_INTERNAL_PARTS + 0].length= sizeof(log_data); if (translog_write_record(&lsn, LOGREC_UNDO_ROW_PURGE, - info->trn, info->s, sizeof(log_data), + info->trn, info, sizeof(log_data), TRANSLOG_INTERNAL_PARTS + 1, log_array, log_data + LSN_STORE_SIZE)) res= 1; @@ -2643,7 +2643,7 @@ static my_bool delete_head_or_tail(MARIA_HA *info, log_array[TRANSLOG_INTERNAL_PARTS + 0].length= sizeof(log_data); if (translog_write_record(&lsn, (head ? LOGREC_REDO_PURGE_ROW_HEAD : LOGREC_REDO_PURGE_ROW_TAIL), - info->trn, share, sizeof(log_data), + info->trn, info, sizeof(log_data), TRANSLOG_INTERNAL_PARTS + 1, log_array, log_data)) DBUG_RETURN(1); @@ -2670,7 +2670,7 @@ static my_bool delete_head_or_tail(MARIA_HA *info, log_array[TRANSLOG_INTERNAL_PARTS + 0].str= (char*) log_data; log_array[TRANSLOG_INTERNAL_PARTS + 0].length= sizeof(log_data); if (translog_write_record(&lsn, LOGREC_REDO_PURGE_BLOCKS, - info->trn, share, sizeof(log_data), + info->trn, info, sizeof(log_data), TRANSLOG_INTERNAL_PARTS + 1, log_array, log_data)) DBUG_RETURN(1); @@ -2775,7 +2775,7 @@ my_bool _ma_delete_block_record(MARIA_HA *info, const uchar *record) &row_parts_count); if (translog_write_record(&lsn, LOGREC_UNDO_ROW_DELETE, info->trn, - info->s, sizeof(log_data) + row_length, + info, sizeof(log_data) + row_length, TRANSLOG_INTERNAL_PARTS + 1 + row_parts_count, info->log_row_parts, log_data + LSN_STORE_SIZE)) goto err; diff --git a/storage/maria/ma_check.c b/storage/maria/ma_check.c index ce4a610cec9..8350fb86660 100644 --- a/storage/maria/ma_check.c +++ b/storage/maria/ma_check.c @@ -5536,7 +5536,7 @@ static int write_log_record_for_repair(const HA_CHECK *param, MARIA_HA *info) int4store(log_data + FILEID_STORE_SIZE, param->testflag); if (unlikely(translog_write_record(&share->state.create_rename_lsn, LOGREC_REDO_REPAIR_TABLE, - &dummy_transaction_object, share, + &dummy_transaction_object, info, log_array[TRANSLOG_INTERNAL_PARTS + 0].length, sizeof(log_array)/sizeof(log_array[0]), diff --git a/storage/maria/ma_delete_all.c b/storage/maria/ma_delete_all.c index 42e7fb3c2f9..14afb8ea870 100644 --- a/storage/maria/ma_delete_all.c +++ b/storage/maria/ma_delete_all.c @@ -62,7 +62,7 @@ int maria_delete_all_rows(MARIA_HA *info) log_array[TRANSLOG_INTERNAL_PARTS + 0].str= (char*) log_data; log_array[TRANSLOG_INTERNAL_PARTS + 0].length= sizeof(log_data); if (unlikely(translog_write_record(&lsn, LOGREC_REDO_DELETE_ALL, - info->trn, share, 0, + info->trn, info, 0, sizeof(log_array)/sizeof(log_array[0]), log_array, log_data) || translog_flush(lsn))) diff --git a/storage/maria/ma_loghandler.c b/storage/maria/ma_loghandler.c index 4158884afb7..cc38341d70b 100644 --- a/storage/maria/ma_loghandler.c +++ b/storage/maria/ma_loghandler.c @@ -181,10 +181,10 @@ static MARIA_SHARE **id_to_share= NULL; static my_atomic_rwlock_t LOCK_id_to_share; static my_bool write_hook_for_redo(enum translog_record_type type, - TRN *trn, MARIA_SHARE *share, LSN *lsn, + TRN *trn, MARIA_HA *tbl_info, LSN *lsn, struct st_translog_parts *parts); static my_bool write_hook_for_undo(enum translog_record_type type, - TRN *trn, MARIA_SHARE *share, LSN *lsn, + TRN *trn, MARIA_HA *tbl_info, LSN *lsn, struct st_translog_parts *parts); /* @@ -3049,7 +3049,7 @@ static translog_size_t translog_get_current_group_size() static my_bool translog_write_variable_record_1group(LSN *lsn, enum translog_record_type type, - MARIA_SHARE *share, + MARIA_HA *tbl_info, SHORT_TRANSACTION_ID short_trid, struct st_translog_parts *parts, struct st_translog_buffer @@ -3067,7 +3067,7 @@ translog_write_variable_record_1group(LSN *lsn, *lsn= horizon= log_descriptor.horizon; if (log_record_type_descriptor[type].inwrite_hook && - (*log_record_type_descriptor[type].inwrite_hook)(type, trn, share, + (*log_record_type_descriptor[type].inwrite_hook)(type, trn, tbl_info, lsn, parts)) { translog_unlock(); @@ -3205,7 +3205,7 @@ translog_write_variable_record_1group(LSN *lsn, static my_bool translog_write_variable_record_1chunk(LSN *lsn, enum translog_record_type type, - MARIA_SHARE *share, + MARIA_HA *tbl_info, SHORT_TRANSACTION_ID short_trid, struct st_translog_parts *parts, struct st_translog_buffer @@ -3221,7 +3221,7 @@ translog_write_variable_record_1chunk(LSN *lsn, *lsn= log_descriptor.horizon; if (log_record_type_descriptor[type].inwrite_hook && - (*log_record_type_descriptor[type].inwrite_hook)(type, trn, share, + (*log_record_type_descriptor[type].inwrite_hook)(type, trn, tbl_info, lsn, parts)) { translog_unlock(); @@ -3574,7 +3574,7 @@ static my_bool translog_relative_LSN_encode(struct st_translog_parts *parts, static my_bool translog_write_variable_record_mgroup(LSN *lsn, enum translog_record_type type, - MARIA_SHARE *share, + MARIA_HA *tbl_info, SHORT_TRANSACTION_ID short_trid, struct st_translog_parts *parts, struct st_translog_buffer @@ -3917,7 +3917,8 @@ translog_write_variable_record_mgroup(LSN *lsn, first_chunk0= 0; *lsn= horizon; if (log_record_type_descriptor[type].inwrite_hook && - (*log_record_type_descriptor[type].inwrite_hook) (type, trn, share, + (*log_record_type_descriptor[type].inwrite_hook) (type, trn, + tbl_info, lsn, parts)) goto err; } @@ -4003,7 +4004,7 @@ err: static my_bool translog_write_variable_record(LSN *lsn, enum translog_record_type type, - MARIA_SHARE *share, + MARIA_HA *tbl_info, SHORT_TRANSACTION_ID short_trid, struct st_translog_parts *parts, TRN *trn) @@ -4081,7 +4082,7 @@ static my_bool translog_write_variable_record(LSN *lsn, if (page_rest >= parts->record_length + header_length1) { /* following function makes translog_unlock(); */ - res= translog_write_variable_record_1chunk(lsn, type, share, + res= translog_write_variable_record_1chunk(lsn, type, tbl_info, short_trid, parts, buffer_to_flush, header_length1, trn); @@ -4093,14 +4094,14 @@ static my_bool translog_write_variable_record(LSN *lsn, if (buffer_rest >= parts->record_length + header_length1 - page_rest) { /* following function makes translog_unlock(); */ - res= translog_write_variable_record_1group(lsn, type, share, + res= translog_write_variable_record_1group(lsn, type, tbl_info, short_trid, parts, buffer_to_flush, header_length1, trn); DBUG_RETURN(res); } /* following function makes translog_unlock(); */ - res= translog_write_variable_record_mgroup(lsn, type, share, + res= translog_write_variable_record_mgroup(lsn, type, tbl_info, short_trid, parts, buffer_to_flush, header_length1, @@ -4128,7 +4129,7 @@ static my_bool translog_write_variable_record(LSN *lsn, static my_bool translog_write_fixed_record(LSN *lsn, enum translog_record_type type, - MARIA_SHARE *share, + MARIA_HA *tbl_info, SHORT_TRANSACTION_ID short_trid, struct st_translog_parts *parts, TRN *trn) @@ -4181,7 +4182,7 @@ static my_bool translog_write_fixed_record(LSN *lsn, *lsn= log_descriptor.horizon; if (log_record_type_descriptor[type].inwrite_hook && - (*log_record_type_descriptor[type].inwrite_hook) (type, trn, share, + (*log_record_type_descriptor[type].inwrite_hook) (type, trn, tbl_info, lsn, parts)) { rc= 1; @@ -4247,16 +4248,16 @@ err: @param type the log record type @param trn Transaction structure pointer for hooks by record log type, for short_id - @param share MARIA_SHARE of table or NULL + @param tbl_info MARIA_HA of table or NULL @param rec_len record length or 0 (count it) @param part_no number of parts or 0 (count it) @param parts_data zero ended (in case of number of parts is 0) array of LEX_STRINGs (parts), first TRANSLOG_INTERNAL_PARTS positions in the log should be unused (need for loghandler) - @param store_share_id if share!=NULL then share's id will automatically - be stored in the two first bytes pointed (so - pointer is assumed to be !=NULL) + @param store_share_id if tbl_info!=NULL then share's id will + automatically be stored in the two first bytes + pointed (so pointer is assumed to be !=NULL) @return Operation status @retval 0 OK @retval 1 Error @@ -4264,7 +4265,7 @@ err: my_bool translog_write_record(LSN *lsn, enum translog_record_type type, - TRN *trn, struct st_maria_share *share, + TRN *trn, MARIA_HA *tbl_info, translog_size_t rec_len, uint part_no, LEX_STRING *parts_data, @@ -4278,8 +4279,9 @@ my_bool translog_write_record(LSN *lsn, DBUG_PRINT("enter", ("type: %u ShortTrID: %u", (uint) type, (uint)short_trid)); - if (share) + if (tbl_info) { + MARIA_SHARE *share= tbl_info->s; if (!share->now_transactional) { DBUG_PRINT("info", ("It is not transactional table")); @@ -4375,17 +4377,17 @@ my_bool translog_write_record(LSN *lsn, /* process this parts */ if (!(rc= (log_record_type_descriptor[type].prewrite_hook && (*log_record_type_descriptor[type].prewrite_hook) (type, trn, - share, + tbl_info, &parts)))) { switch (log_record_type_descriptor[type].class) { case LOGRECTYPE_VARIABLE_LENGTH: - rc= translog_write_variable_record(lsn, type, share, + rc= translog_write_variable_record(lsn, type, tbl_info, short_trid, &parts, trn); break; case LOGRECTYPE_PSEUDOFIXEDLENGTH: case LOGRECTYPE_FIXEDLENGTH: - rc= translog_write_fixed_record(lsn, type, share, + rc= translog_write_fixed_record(lsn, type, tbl_info, short_trid, &parts, trn); break; case LOGRECTYPE_NOT_ALLOWED: @@ -5619,7 +5621,7 @@ my_bool translog_flush(LSN lsn) static my_bool write_hook_for_redo(enum translog_record_type type __attribute__ ((unused)), - TRN *trn, MARIA_SHARE *share + TRN *trn, MARIA_HA *tbl_info __attribute__ ((unused)), LSN *lsn, struct st_translog_parts *parts @@ -5657,7 +5659,7 @@ static my_bool write_hook_for_redo(enum translog_record_type type static my_bool write_hook_for_undo(enum translog_record_type type __attribute__ ((unused)), - TRN *trn, MARIA_SHARE *share + TRN *trn, MARIA_HA *tbl_info __attribute__ ((unused)), LSN *lsn, struct st_translog_parts *parts @@ -5683,7 +5685,7 @@ static my_bool write_hook_for_undo(enum translog_record_type type open MARIA_SHAREs), give it one and record this assignment in the log (LOGREC_FILE_ID log record). - @param share table + @param share table @param trn calling transaction @return Operation status @@ -5722,10 +5724,11 @@ int translog_assign_id_to_share(MARIA_SHARE *share, TRN *trn) my_atomic_rwlock_wrunlock(&LOCK_id_to_share); i= 1; /* scan the whole array */ } while (share->id == 0); - DBUG_PRINT("info", ("id_to_share: 0x%lx -> %u", (ulong)share, i)); + DBUG_PRINT("info", ("id_to_share: 0x%lx -> %u", (ulong)share, share->id)); LSN lsn; LEX_STRING log_array[TRANSLOG_INTERNAL_PARTS + 2]; uchar log_data[FILEID_STORE_SIZE]; + fileid_store(log_data, share->id); log_array[TRANSLOG_INTERNAL_PARTS + 0].str= (char*) log_data; log_array[TRANSLOG_INTERNAL_PARTS + 0].length= sizeof(log_data); /* @@ -5740,12 +5743,12 @@ int translog_assign_id_to_share(MARIA_SHARE *share, TRN *trn) */ log_array[TRANSLOG_INTERNAL_PARTS + 1].length= strlen(share->open_file_name) + 1; - if (unlikely(translog_write_record(&lsn, LOGREC_FILE_ID, trn, share, + if (unlikely(translog_write_record(&lsn, LOGREC_FILE_ID, trn, NULL, sizeof(log_data) + log_array[TRANSLOG_INTERNAL_PARTS + 1].length, sizeof(log_array)/sizeof(log_array[0]), - log_array, log_data))) + log_array, NULL))) return 1; } pthread_mutex_unlock(&share->intern_lock); diff --git a/storage/maria/ma_loghandler.h b/storage/maria/ma_loghandler.h index 40e87d1d99d..db3d43e39f4 100644 --- a/storage/maria/ma_loghandler.h +++ b/storage/maria/ma_loghandler.h @@ -46,7 +46,7 @@ /* short transaction ID type */ typedef uint16 SHORT_TRANSACTION_ID; -struct st_maria_share; +struct st_maria_info; /* Length of CRC at end of pages */ #define CRC_LENGTH 4 @@ -225,7 +225,7 @@ extern my_bool translog_init(const char *directory, uint32 log_file_max_size, extern my_bool translog_write_record(LSN *lsn, enum translog_record_type type, struct st_transaction *trn, - struct st_maria_share *share, + struct st_maria_info *tbl_info, translog_size_t rec_len, uint part_no, LEX_STRING *parts_data, uchar *store_share_id); @@ -284,11 +284,11 @@ struct st_translog_parts }; typedef my_bool(*prewrite_rec_hook) (enum translog_record_type type, - TRN *trn, struct st_maria_share *share, + TRN *trn, struct st_maria_info *tbl_info, struct st_translog_parts *parts); typedef my_bool(*inwrite_rec_hook) (enum translog_record_type type, - TRN *trn, struct st_maria_share *share, + TRN *trn, struct st_maria_info *tbl_info, LSN *lsn, struct st_translog_parts *parts); diff --git a/storage/maria/tablockman.c b/storage/maria/tablockman.c index b7e2d62e1ab..eb8da1d6865 100644 --- a/storage/maria/tablockman.c +++ b/storage/maria/tablockman.c @@ -663,11 +663,11 @@ void tablockman_print_tlo(TABLE_LOCK_OWNER *lo) printf("lo%d>", lo->loid); if ((lock= lo->waiting_lock)) - printf(" (%s.0x%lx)", lock2str[lock->lock_type], (intptr)lock->table); + printf(" (%s.0x%lx)", lock2str[lock->lock_type], (ulong)lock->table); for (lock= lo->active_locks; lock && lock != lock->next_in_lo; lock= lock->next_in_lo) - printf(" %s.0x%lx", lock2str[lock->lock_type], (intptr)lock->table); + printf(" %s.0x%lx", lock2str[lock->lock_type], (ulong)lock->table); if (lock && lock == lock->next_in_lo) printf("!"); printf("\n"); -- cgit v1.2.1 From 91f03a4d31f295acd4e63df6216a1c6c9260d7f1 Mon Sep 17 00:00:00 2001 From: unknown Date: Thu, 2 Aug 2007 12:42:49 +0200 Subject: WL#3072 Maria checkpoint Preparation: for the two-checkpoint rule we will flush all pages which have block->rec_lsn <= last_checkpoint_lsn. Pages with rec_lsn not yet set (i.e. ==0) were dirtied very recently and so should not be flushed (otherwise it's not efficient: we should try to let a page be modified several times in memory before we flush it to disk). To make this easy, "block->rec_lsn not yet set" is now expressed with block->rec_lsn==LSN_MAX, not block->rec_lsn==0 anymore. It is easier this way because LSN_MAX>last_checkpoint_lsn whereas 0<=last_checkpoint_lsn. storage/maria/ma_blockrec.c: typo storage/maria/ma_loghandler.c: typo storage/maria/ma_loghandler_lsn.h: LSN_MAX storage/maria/ma_pagecache.c: "block->rec_lsn not yet set" is now expressed by block->rec_lsn==LSN_MAX, not block_rec_lsn==0. --- storage/maria/ma_blockrec.c | 2 +- storage/maria/ma_loghandler.c | 2 +- storage/maria/ma_loghandler_lsn.h | 8 ++++++++ storage/maria/ma_pagecache.c | 26 ++++++++++++++++++-------- 4 files changed, 28 insertions(+), 10 deletions(-) diff --git a/storage/maria/ma_blockrec.c b/storage/maria/ma_blockrec.c index ed146ff1952..e51b06efbba 100644 --- a/storage/maria/ma_blockrec.c +++ b/storage/maria/ma_blockrec.c @@ -565,7 +565,7 @@ static my_bool check_if_zero(uchar *pos, uint length) We unpin pages in the reverse order as they where pinned; This may not be strictly necessary but may simplify things in the future. - info->s->rec_lsn contains the lsn for the first REDO + info->trn->rec_lsn contains the lsn for the first REDO RETURN 0 ok diff --git a/storage/maria/ma_loghandler.c b/storage/maria/ma_loghandler.c index cc38341d70b..1b1589fcbc7 100644 --- a/storage/maria/ma_loghandler.c +++ b/storage/maria/ma_loghandler.c @@ -5685,7 +5685,7 @@ static my_bool write_hook_for_undo(enum translog_record_type type open MARIA_SHAREs), give it one and record this assignment in the log (LOGREC_FILE_ID log record). - @param share table + @param share table @param trn calling transaction @return Operation status diff --git a/storage/maria/ma_loghandler_lsn.h b/storage/maria/ma_loghandler_lsn.h index 387fe2763d5..9e1c4632fb0 100644 --- a/storage/maria/ma_loghandler_lsn.h +++ b/storage/maria/ma_loghandler_lsn.h @@ -81,4 +81,12 @@ typedef LSN LSN_WITH_FLAGS; #define FILENO_IMPOSSIBLE 0 /**< log file's numbering starts at 1 */ #define LOG_OFFSET_IMPOSSIBLE 0 /**< log always has a header */ #define LSN_IMPOSSIBLE 0 + +/** + @brief the maximum valid LSN. + Unlike ULONGLONG_MAX, it can be safely used in comparison with valid LSNs + (ULONGLONG_MAX is too big for correctness of cmp_translog_address()). +*/ +#define LSN_MAX (LSN)ULL(0x00FFFFFFFFFFFFFF) + #endif diff --git a/storage/maria/ma_pagecache.c b/storage/maria/ma_pagecache.c index 5c803b3dd83..c5d9fbdca4d 100755 --- a/storage/maria/ma_pagecache.c +++ b/storage/maria/ma_pagecache.c @@ -314,7 +314,8 @@ struct st_pagecache_block_link enum pagecache_page_type type; /* type of the block */ uint hits_left; /* number of hits left until promotion */ ulonglong last_hit_time; /* timestamp of the last hit */ - LSN rec_lsn; /**< LSN when first became dirty */ + /** @brief LSN when first became dirty; LSN_MAX means "not yet set" */ + LSN rec_lsn; KEYCACHE_CONDVAR *condvar; /* condition variable for 'no readers' event */ }; @@ -1120,7 +1121,7 @@ static void link_to_file_list(PAGECACHE *pagecache, if (block->status & PCBLOCK_CHANGED) { block->status&= ~PCBLOCK_CHANGED; - block->rec_lsn= 0; + block->rec_lsn= LSN_MAX; pagecache->blocks_changed--; pagecache->global_blocks_changed--; } @@ -1892,6 +1893,7 @@ restart: block->temperature= PCBLOCK_COLD; block->hits_left= init_hits_left; block->last_hit_time= 0; + block->rec_lsn= LSN_MAX; link_to_file_list(pagecache, block, file, 0); block->hash_link= hash_link; hash_link->block= block; @@ -2537,8 +2539,12 @@ void pagecache_unlock(PAGECACHE *pagecache, { DBUG_ASSERT(lock == PAGECACHE_LOCK_WRITE_UNLOCK); DBUG_ASSERT(pin == PAGECACHE_UNPIN); - if (block->rec_lsn == 0) + if (block->rec_lsn == LSN_MAX) block->rec_lsn= first_REDO_LSN_for_page; + else + DBUG_ASSERT(cmp_translog_addr(block->rec_lsn, + first_REDO_LSN_for_page) <= 0); + } if (lsn != LSN_IMPOSSIBLE) check_and_set_lsn(pagecache, lsn, block); @@ -2695,8 +2701,11 @@ void pagecache_unlock_by_link(PAGECACHE *pagecache, DBUG_ASSERT(lock == PAGECACHE_LOCK_WRITE_UNLOCK || lock == PAGECACHE_LOCK_READ_UNLOCK); DBUG_ASSERT(pin == PAGECACHE_UNPIN); - if (block->rec_lsn == 0) + if (block->rec_lsn == LSN_MAX) block->rec_lsn= first_REDO_LSN_for_page; + else + DBUG_ASSERT(cmp_translog_addr(block->rec_lsn, + first_REDO_LSN_for_page) <= 0); } if (lsn != LSN_IMPOSSIBLE) check_and_set_lsn(pagecache, lsn, block); @@ -3377,7 +3386,7 @@ static void free_block(PAGECACHE *pagecache, PAGECACHE_BLOCK_LINK *block) #ifndef DBUG_OFF block->type= PAGECACHE_EMPTY_PAGE; #endif - block->rec_lsn= 0; + block->rec_lsn= LSN_MAX; KEYCACHE_THREAD_TRACE("free block"); KEYCACHE_DBUG_PRINT("free_block", ("block is freed")); @@ -3850,7 +3859,7 @@ my_bool pagecache_collect_changed_blocks_with_lsn(PAGECACHE *pagecache, ulong stored_list_size= 0; uint file_hash; char *ptr; - LSN minimum_rec_lsn= ULONGLONG_MAX, maximum_rec_lsn= 0; + LSN minimum_rec_lsn= LSN_MAX, maximum_rec_lsn= 0; DBUG_ENTER("pagecache_collect_changed_blocks_with_LSN"); DBUG_ASSERT(NULL == str->str); @@ -3919,13 +3928,14 @@ my_bool pagecache_collect_changed_blocks_with_lsn(PAGECACHE *pagecache, ptr+= 4; lsn_store(ptr, block->rec_lsn); ptr+= LSN_STORE_SIZE; - if (block->rec_lsn != LSN_IMPOSSIBLE) + if (block->rec_lsn != LSN_MAX) { + DBUG_ASSERT(LSN_VALID(block->rec_lsn)); if (cmp_translog_addr(block->rec_lsn, minimum_rec_lsn) < 0) minimum_rec_lsn= block->rec_lsn; if (cmp_translog_addr(block->rec_lsn, maximum_rec_lsn) > 0) maximum_rec_lsn= block->rec_lsn; - } /* otherwise, some trn->rec_lsn should hold the info */ + } /* otherwise, some trn->rec_lsn should hold the correct info */ } } end: -- cgit v1.2.1 From 3d0f42a94c35eaf2f62677f618963b2af9166338 Mon Sep 17 00:00:00 2001 From: unknown Date: Fri, 3 Aug 2007 15:58:50 +0200 Subject: Log handler's unit test were broken by me on June 22, fixing this: don't auto-log LOGREC_LONG_TRANSACTION_ID in log handler's unit tests, as they read their log and expect to find only the records they asked for, and so not LOGREC_LONG_TRANSACTION_ID. All log handler's unit tests pass. A way to run them faster: when their LONG_BUFFER_SIZE is too big (1GB) I divide it by ten and then I can run them on /dev/shm - they are then "instant". By the way, pushbuild was not displaying anymore "unit: failed" in the list of all pushes, which participates to the reasons why I didn't notice the breakage earlier. The other reason being that I was too lazy to run log handler unit tests on my machine as they took long (hadn't yet thought about the /dev/shm idea) and so I relied on pushbuild; Danny has now quickly fixed pushbuild - thanks :) storage/maria/unittest/ma_test_loghandler-t.c: don't auto-log LOGREC_LONG_TRANSACTION_ID in log handler's unit tests, as they read their log and expect to find only the records they asked for storage/maria/unittest/ma_test_loghandler_multigroup-t.c: don't auto-log LOGREC_LONG_TRANSACTION_ID in log handler's unit tests, as they read their log and expect to find only the records they asked for storage/maria/unittest/ma_test_loghandler_multithread-t.c: don't auto-log LOGREC_LONG_TRANSACTION_ID in log handler's unit tests, as they read their log and expect to find only the records they asked for storage/maria/unittest/ma_test_loghandler_pagecache-t.c: don't auto-log LOGREC_LONG_TRANSACTION_ID in log handler's unit tests, as they read their log and expect to find only the records they asked for --- storage/maria/unittest/ma_test_loghandler-t.c | 1 + storage/maria/unittest/ma_test_loghandler_multigroup-t.c | 1 + storage/maria/unittest/ma_test_loghandler_multithread-t.c | 2 ++ storage/maria/unittest/ma_test_loghandler_pagecache-t.c | 1 + 4 files changed, 5 insertions(+) diff --git a/storage/maria/unittest/ma_test_loghandler-t.c b/storage/maria/unittest/ma_test_loghandler-t.c index 04459adeac8..3aecb724c6d 100644 --- a/storage/maria/unittest/ma_test_loghandler-t.c +++ b/storage/maria/unittest/ma_test_loghandler-t.c @@ -193,6 +193,7 @@ int main(int argc __attribute__((unused)), char *argv[]) parts[TRANSLOG_INTERNAL_PARTS + 0].str= (char*)long_tr_id; parts[TRANSLOG_INTERNAL_PARTS + 0].length= 6; trn->short_id= 0; + trn->first_undo_lsn= TRANSACTION_LOGGED_LONG_ID; if (translog_write_record(&lsn, LOGREC_FIXED_RECORD_0LSN_EXAMPLE, trn, NULL, diff --git a/storage/maria/unittest/ma_test_loghandler_multigroup-t.c b/storage/maria/unittest/ma_test_loghandler_multigroup-t.c index cec7198ef3b..bf4cfe110e3 100644 --- a/storage/maria/unittest/ma_test_loghandler_multigroup-t.c +++ b/storage/maria/unittest/ma_test_loghandler_multigroup-t.c @@ -190,6 +190,7 @@ int main(int argc __attribute__((unused)), char *argv[]) parts[TRANSLOG_INTERNAL_PARTS + 0].str= (char*)long_tr_id; parts[TRANSLOG_INTERNAL_PARTS + 0].length= 6; trn->short_id= 0; + trn->first_undo_lsn= TRANSACTION_LOGGED_LONG_ID; if (translog_write_record(&lsn, LOGREC_FIXED_RECORD_0LSN_EXAMPLE, trn, NULL, 6, TRANSLOG_INTERNAL_PARTS + 1, parts, NULL)) diff --git a/storage/maria/unittest/ma_test_loghandler_multithread-t.c b/storage/maria/unittest/ma_test_loghandler_multithread-t.c index 86e66daca52..0f56ef5384c 100644 --- a/storage/maria/unittest/ma_test_loghandler_multithread-t.c +++ b/storage/maria/unittest/ma_test_loghandler_multithread-t.c @@ -124,6 +124,7 @@ void writer(int num) uint i; trn.short_id= num; + trn.first_undo_lsn= TRANSACTION_LOGGED_LONG_ID; for (i= 0; i < ITERATIONS; i++) { uint len= get_len(); @@ -299,6 +300,7 @@ int main(int argc __attribute__((unused)), parts[TRANSLOG_INTERNAL_PARTS + 0].str= (char*)long_tr_id; parts[TRANSLOG_INTERNAL_PARTS + 0].length= 6; + dummy_transaction_object.first_undo_lsn= TRANSACTION_LOGGED_LONG_ID; if (translog_write_record(&first_lsn, LOGREC_FIXED_RECORD_0LSN_EXAMPLE, &dummy_transaction_object, NULL, 6, diff --git a/storage/maria/unittest/ma_test_loghandler_pagecache-t.c b/storage/maria/unittest/ma_test_loghandler_pagecache-t.c index 804dd961fbc..7dfdc32234e 100644 --- a/storage/maria/unittest/ma_test_loghandler_pagecache-t.c +++ b/storage/maria/unittest/ma_test_loghandler_pagecache-t.c @@ -90,6 +90,7 @@ int main(int argc __attribute__((unused)), char *argv[]) int4store(long_tr_id, 0); parts[TRANSLOG_INTERNAL_PARTS + 0].str= (char*)long_tr_id; parts[TRANSLOG_INTERNAL_PARTS + 0].length= 6; + dummy_transaction_object.first_undo_lsn= TRANSACTION_LOGGED_LONG_ID; if (translog_write_record(&lsn, LOGREC_FIXED_RECORD_0LSN_EXAMPLE, &dummy_transaction_object, NULL, 6, -- cgit v1.2.1 From ec547ae830f759aef707de5e783f075568440ec1 Mon Sep 17 00:00:00 2001 From: unknown Date: Mon, 6 Aug 2007 16:13:42 +0200 Subject: fixes of bad merge (probably). Less duplication in ma_test_recovery. storage/maria/ma_range.c: old code, which is wrong now (key_len is now a local variable, not initialized). Fixes some problems in running ma_test2. storage/maria/ma_test2.c: keypart_map should be used instead of length (fix of bad merge probably) storage/maria/ma_test_recovery: less duplication in this script (one loop instead). --- storage/maria/ma_range.c | 2 -- storage/maria/ma_test2.c | 4 ++-- storage/maria/ma_test_recovery | 46 +++++++++++++++++++++++------------------- 3 files changed, 27 insertions(+), 25 deletions(-) diff --git a/storage/maria/ma_range.c b/storage/maria/ma_range.c index 70dd522b8a1..02616d8ac5c 100644 --- a/storage/maria/ma_range.c +++ b/storage/maria/ma_range.c @@ -137,8 +137,6 @@ static ha_rows _ma_record_pos(MARIA_HA *info, const uchar *key, DBUG_PRINT("enter",("search_flag: %d",search_flag)); DBUG_ASSERT(keypart_map); - if (key_len == 0) - key_len= USE_WHOLE_KEY; key_buff=info->lastkey+info->s->base.max_key_length; key_len= _ma_pack_key(info, inx, key_buff, key, keypart_map, (HA_KEYSEG**) 0); diff --git a/storage/maria/ma_test2.c b/storage/maria/ma_test2.c index 2e3884bf6ce..e820907dccd 100644 --- a/storage/maria/ma_test2.c +++ b/storage/maria/ma_test2.c @@ -719,10 +719,10 @@ int main(int argc, char *argv[]) sprintf(key2,"%6d",k); min_key.key= key; - min_key.length= USE_WHOLE_KEY; + min_key.keypart_map= HA_WHOLE_KEY; min_key.flag= HA_READ_AFTER_KEY; max_key.key= key2; - max_key.length= USE_WHOLE_KEY; + max_key.keypart_map= HA_WHOLE_KEY; max_key.flag= HA_READ_BEFORE_KEY; range_records= maria_records_in_range(file, 0, &min_key, &max_key); records=0; diff --git a/storage/maria/ma_test_recovery b/storage/maria/ma_test_recovery index 65b8ac0838a..87850cd18f2 100644 --- a/storage/maria/ma_test_recovery +++ b/storage/maria/ma_test_recovery @@ -12,26 +12,30 @@ echo "MARIA RECOVERY TESTS - success is if exit code is 0" # identical to the saved original. # Does not test the index file as we don't have logging for it yet. -rm -f maria_log* -prog="$maria_path/ma_test1 $silent -M -T --skip-update" -echo "TEST WITH $prog" -$prog -mv -f test1.MAD test1.MAD.good -rm test1.MAI -echo "applying log" -$maria_path/maria_read_log -a > /dev/null -cmp test1.MAD test1.MAD.good -rm -f test1.* - -rm -f maria_log* -prog="$maria_path/ma_test2 $silent -L -K -W -P -M -T -g" -echo "TEST WITH $prog" -$prog -mv -f test2.MAD test2.MAD.good -rm test2.MAI -echo "applying log" -$maria_path/maria_read_log -a > /dev/null -cmp test2.MAD test2.MAD.good -rm -f test2.* +for prog in "$maria_path/ma_test1 $silent -M -T --skip-update -c" "$maria_path/ma_test2 $silent -L -K -W -P -M -T -g -c" +do + rm -f maria_log* + echo "TEST WITH $prog" + $prog + # derive table's name from program's name + table=`echo $prog | sed -e 's;.*ma_\(test[0-9]\).*;\1;' ` + $maria_path/maria_chk -dvv $table > maria_chk_message.good.txt 2>&1 + mv -f $table.MAD $table.MAD.good + rm $table.MAI + echo "applying log" + $maria_path/maria_read_log -a > maria_read_log_$table.txt + cmp $table.MAD $table.MAD.good + $maria_path/maria_chk -dvv $table > maria_chk_message.txt 2>&1 +# When "recovery of the table's state" is ready, we can test it like this: +# diff maria_chk_message.good.txt maria_chk_message.txt >maria_chk_diff.txt || true +# if [ -s maria_chk_diff.txt ] +# then +# echo "Differences in maria_chk -dvv, recovery not yet perfect !" +# echo "========DIFF START=======" +# cat maria_chk_diff.txt +# echo "========DIFF END=======" +# fi + rm -f $table.* maria_chk_*.txt maria_read_log_$table.txt +done echo "ALL RECOVERY TESTS OK" -- cgit v1.2.1 From 1b47e3dabddd64f56b804721a37bd5defc94d23f Mon Sep 17 00:00:00 2001 From: unknown Date: Mon, 6 Aug 2007 20:17:38 +0200 Subject: Make test does not fail if run with --mysqld=--default-storage-engine=X where X does not support INSERT DELAYED (like Maria) mysql-test/r/alter_table.result: result update mysql-test/t/alter_table.test: so that test does not fail if run with --mysqld=--default-storage-engine=X where X does not support INSERT DELAYED (like Maria) --- mysql-test/r/alter_table.result | 2 +- mysql-test/t/alter_table.test | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/mysql-test/r/alter_table.result b/mysql-test/r/alter_table.result index cc93eab0cd5..e6c99cf0cf3 100644 --- a/mysql-test/r/alter_table.result +++ b/mysql-test/r/alter_table.result @@ -298,7 +298,7 @@ t1 0 a 1 a A 3 NULL NULL YES BTREE t1 0 a 2 b A 300 NULL NULL YES BTREE t1 1 b 1 b A 100 NULL NULL YES BTREE drop table t1; -CREATE TABLE t1 (i int(10), index(i) ); +CREATE TABLE t1 (i int(10), index(i) ) ENGINE=MyISAM; ALTER TABLE t1 DISABLE KEYS; INSERT DELAYED INTO t1 VALUES(1),(2),(3); ALTER TABLE t1 ENABLE KEYS; diff --git a/mysql-test/t/alter_table.test b/mysql-test/t/alter_table.test index fa50d75b8d2..bfec0d08679 100644 --- a/mysql-test/t/alter_table.test +++ b/mysql-test/t/alter_table.test @@ -159,7 +159,7 @@ drop table t1; # Test of ALTER TABLE DELAYED # -CREATE TABLE t1 (i int(10), index(i) ); +CREATE TABLE t1 (i int(10), index(i) ) ENGINE=MyISAM; ALTER TABLE t1 DISABLE KEYS; INSERT DELAYED INTO t1 VALUES(1),(2),(3); ALTER TABLE t1 ENABLE KEYS; -- cgit v1.2.1 From 72c3c369e4e86c89bc98bb9346fd4cd38ead4064 Mon Sep 17 00:00:00 2001 From: unknown Date: Tue, 7 Aug 2007 16:06:42 +0200 Subject: Fix for three bugs: number 1: "./mtr --mysqld=--default-storage-engine=maria backup" restored no rows (forgot to flush data pages before my_copy(), and also the maria_repair() used by ha_maria::restore() needed a correct data_file_length to not miss rows). [note that BACKUP TABLE will be removed anyway in 5.2] number 2: "./mtr --mysqld=--default-storage-engine=maria bootstrap" caused segfault (uninitialized variable) number 3: "./mtr --mysqld=--default-storage-engine=maria check" showed warning in CHECK TABLE (maria_create() created a non-empty data file with data_file_length==0). storage/maria/ha_maria.cc: in ha_maria::backup, need to flush the data file before copying it, otherwise data misses from the copy (bug 1) storage/maria/ma_bitmap.c: when allocating data at the end of the bitmap, best_data is at "end", should not be left to 0 (bug 2) storage/maria/ma_check.c: _ma_scan_block_record() is used in QUICK repair. It relies on data_file_length. RESTORE TABLE mixes the MAI of an empty table (so, data_file_length==0) with an non-empty MAD, and does a QUICK repair; that got fooled (thought it had hit EOF immediately, so found no records) (bug 1) storage/maria/ma_create.c: At the end of maria_create() we have, in the index file, data_file_length==0, while the data file has a bitmap page (8192). This inconsistency makes CHECK TABLE rightly complain. Fixed by not creating a first bitmap page during maria_create() (also saves disk space) (bug 3) Question for Monty. storage/maria/ma_extra.c: A function to flush the data and index files before one can use OS syscalls (reads, writes) on those files. For example, ha_maria::backup() does a my_copy() of the data file and so all cached pieces of this file must be sent to the OS (bug 1) This function will have to be used elsewhere in Maria, several places have not been updated when we added pagecache-ing of the data file (they still only flush the index file), they are probable bugs. storage/maria/maria_def.h: new function. Needs to be visible from ha_maria::backup. --- storage/maria/ha_maria.cc | 7 ++++++ storage/maria/ma_bitmap.c | 1 + storage/maria/ma_check.c | 10 +++++++- storage/maria/ma_create.c | 15 +++++++++++- storage/maria/ma_extra.c | 60 +++++++++++++++++++++++++++++++++++++++++++++++ storage/maria/maria_def.h | 5 ++++ 6 files changed, 96 insertions(+), 2 deletions(-) diff --git a/storage/maria/ha_maria.cc b/storage/maria/ha_maria.cc index 3c764bbc827..9a2637e0cc4 100644 --- a/storage/maria/ha_maria.cc +++ b/storage/maria/ha_maria.cc @@ -1063,6 +1063,13 @@ int ha_maria::backup(THD * thd, HA_CHECK_OPT *check_opt) } strxmov(src_path, table->s->normalized_path.str, MARIA_NAME_DEXT, NullS); + if (_ma_flush_table_files(file, MARIA_FLUSH_DATA, FLUSH_FORCE_WRITE, + FLUSH_KEEP)) + { + error= HA_ADMIN_FAILED; + errmsg= "Failed in flush (Error %d)"; + goto err; + } if (my_copy(src_path, dst_path, MYF(MY_WME | MY_HOLD_ORIGINAL_MODES | MY_DONT_OVERWRITE_FILE))) { diff --git a/storage/maria/ma_bitmap.c b/storage/maria/ma_bitmap.c index 6bb4d3c95f3..ca9657128e4 100644 --- a/storage/maria/ma_bitmap.c +++ b/storage/maria/ma_bitmap.c @@ -837,6 +837,7 @@ static my_bool allocate_tail(MARIA_FILE_BITMAP *bitmap, uint size, if (bitmap->used_size == bitmap->total_size) DBUG_RETURN(1); /* Allocate data at end of bitmap */ + best_data= end; bitmap->used_size+= 6; best_pos= best_bits= 0; } diff --git a/storage/maria/ma_check.c b/storage/maria/ma_check.c index 8350fb86660..b7298abeaa0 100644 --- a/storage/maria/ma_check.c +++ b/storage/maria/ma_check.c @@ -3753,7 +3753,15 @@ static int sort_get_next_record(MARIA_SORT_PARAM *sort_param) } else { - /* Scan on clean table */ + /* + Scan on clean table. + It requires a reliable data_file_length so we set it. + */ + my_off_t dfile_len= my_seek(info->dfile.file, 0, SEEK_END, + MYF(MY_WME)); + if (dfile_len == MY_FILEPOS_ERROR) + DBUG_RETURN(my_errno); + info->state->data_file_length= dfile_len; flag= _ma_scan_block_record(info, sort_param->record, info->cur_row.nextpos, 1); } diff --git a/storage/maria/ma_create.c b/storage/maria/ma_create.c index 201c5603c25..014f0189bbb 100644 --- a/storage/maria/ma_create.c +++ b/storage/maria/ma_create.c @@ -1045,15 +1045,28 @@ int maria_create(const char *name, enum data_file_type datafile_type, log record - data file must be created after log record, so that "missing log record" implies "unusable table"). + When we wrote the state, we hadn't called ma_initialize_data_file(), so + the data_file_length is 0! Thus, we below create a 8192-byte data file, but its recorded size is 0, so next time we read the bitmap (a maria_write() for example) we'll overwrite the bitmap we just created below. - It's not very efficient. Though there is no bug. + It's not very efficient. + It also makes maria_chk_size() print + Size of datafile is: 8192 Should be: 0 + on a freshly created table (run "check.test" with a Maria table). + Why do we absolutely want to create a 8192-byte page for a freshly created, empty table? Why don't we leave the data file empty? + Removing the call below at least removes the maria_chk_size() issue. + + Monty wrote on IRC, about a size of 0: + "This basically ok; The first block is a bitmap that may or may not + exists", but later he asked that the first block always exists.??? */ +#ifdef ASK_MONTY if (_ma_initialize_data_file(&share, dfile)) goto err; +#endif } /* Enlarge files */ diff --git a/storage/maria/ma_extra.c b/storage/maria/ma_extra.c index 9ee3b1a8870..0ee5990844b 100644 --- a/storage/maria/ma_extra.c +++ b/storage/maria/ma_extra.c @@ -17,6 +17,7 @@ #ifdef HAVE_SYS_MMAN_H #include #endif +#include "ma_blockrec.h" static void maria_extra_keyflag(MARIA_HA *info, enum ha_extra_function function); @@ -279,6 +280,8 @@ int maria_extra(MARIA_HA *info, enum ha_extra_function function, Don't we wait for all instances to be closed before dropping the table? Do we ever do something useful here? BUG? + FLUSH_IGNORE_CHANGED: we are also throwing away unique index blocks? + Does ENABLE KEYS rebuild them too? */ if (flush_pagecache_blocks(share->pagecache, &share->kfile, FLUSH_IGNORE_CHANGED)) @@ -488,3 +491,60 @@ int _ma_sync_table_files(const MARIA_HA *info) return (my_sync(info->dfile.file, MYF(0)) || my_sync(info->s->kfile.file, MYF(0))); } + + +/** + @brief flushes the data and/or index file of a table + + This is useful when one wants to read a table using OS syscalls (like + my_copy()) and first wants to be sure that MySQL-level caches go down to + the OS so that OS syscalls can see all data. It can flush rec_cache, + bitmap, pagecache of data file, pagecache of index file. + + @param info table + @param flush_data_or_index one or two of these flags: + MARIA_FLUSH_DATA, MARIA_FLUSH_INDEX + @param flush_type_for_data + @param flush_type_for_index + + @note does not sync files (@see _ma_sync_table_files()). + @note Progressively this function will be used in all places where we flush + the index but not the data file (probable bugs). + + @return Operation status + @retval 0 OK + @retval 1 Error +*/ + +int _ma_flush_table_files(MARIA_HA *info, uint flush_data_or_index, + enum flush_type flush_type_for_data, + enum flush_type flush_type_for_index) +{ + MARIA_SHARE *share= info->s; + /* flush data file first because it's more critical */ + if (flush_data_or_index & MARIA_FLUSH_DATA) + { + if (info->opt_flag & WRITE_CACHE_USED) + { + if (end_io_cache(&info->rec_cache)) + goto err; + info->opt_flag&= ~WRITE_CACHE_USED; + } + if (share->data_file_type == BLOCK_RECORD) + { + if(_ma_flush_bitmap(share) || + flush_pagecache_blocks(share->pagecache, &info->dfile, + flush_type_for_data)) + goto err; + } + } + if ((flush_data_or_index & MARIA_FLUSH_INDEX) && + flush_pagecache_blocks(share->pagecache, &share->kfile, + flush_type_for_index)) + goto err; + return 0; +err: + maria_print_error(info->s, HA_ERR_CRASHED); + maria_mark_crashed(info); + return 1; +} diff --git a/storage/maria/maria_def.h b/storage/maria/maria_def.h index e2dc7d3be86..dfaea4ab727 100644 --- a/storage/maria/maria_def.h +++ b/storage/maria/maria_def.h @@ -902,6 +902,11 @@ MARIA_RECORD_POS _ma_write_init_default(MARIA_HA *info, const uchar *record); my_bool _ma_write_abort_default(MARIA_HA *info); C_MODE_START +#define MARIA_FLUSH_DATA 1 +#define MARIA_FLUSH_INDEX 2 +int _ma_flush_table_files(MARIA_HA *info, uint flush_data_or_index, + enum flush_type flush_type_for_data, + enum flush_type flush_type_for_index); /* Functions needed by _ma_check (are overrided in MySQL) */ volatile int *_ma_killed_ptr(HA_CHECK *param); void _ma_check_print_error _VARARGS((HA_CHECK *param, const char *fmt, ...)); -- cgit v1.2.1 From 1ad3a05dd7353cc7106d57de1acf777cbd0368c0 Mon Sep 17 00:00:00 2001 From: unknown Date: Tue, 7 Aug 2007 18:23:49 +0200 Subject: Fix for errors during: "./mtr --mysqld=--default-storage-engine=maria mysqldump". First problem was use of INSERT DELAYED and MERGE tables without specifying that the tables to create should always be MyISAM. After fixing this, no rows were returned by the final SELECT of the "BUG 19025" portion of the test. Simplified problem was: LOCK TABLES `t1` WRITE; /*!40000 ALTER TABLE `t1` DISABLE KEYS */; INSERT INTO `t1` VALUES ('bla',1000),('bla',1001),('bla',1002); /*!40000 ALTER TABLE `t1` ENABLE KEYS */; UNLOCK TABLES; select * from t1; The SELECT would find no rows. Reason: ENABLE KEYS does a maria_repair(); but data pages are still in the page cache and not on disk (because they were not flushed because maria_lock_database(F_UNLCK) was not called at the end of INSERT because under LOCK TABLES). At start of maria_repair(), sort_info.filelength is set to the physical size of the data file (=> too small because pages are in cache and not on disk). Then in sort_get_next_record(), when seeing end-of-file, this is done: sort_param->max_pos= sort_info->filelength; Further in maria_repair(), this is done: info->state->data_file_length= sort_param.max_pos; and so data_file_length is smaller (0) than reality (16384). This makes SELECT think EOF is where it is not, and thus find no rows. This is fixed by flushing all data pages at the start of maria_repair() (no performance problem is introduced as in common cases where ALTER TABLE is not under LOCK TABLES, the previous statement did this flush anyway). Another reason to do this flush is that, if not doing it, old cached pages might go down onto the repaired data file at a later point and thus corrupt it (assume a REPAIR non-QUICK). A similar bug is fixed: LOCK TABLES WRITE; INSERT; CHECK TABLE; reports "Size of datafile is: 0 Should be: 16384" again because the physical size was read without a preliminary page cache flush. mysql-test/r/maria.result: result update mysql-test/r/mysqldump.result: result update mysql-test/t/maria.test: adding test for fixed bug in LOCK TABLES + CHECK TABLE + block format. Disabling portion which hits "incorrect key file" but still letting it make the test fail (Monty to fix). mysql-test/t/mysqldump.test: in places where test expects engine to support INSERT DELAYED and be includable in a MERGE table, i.e. be MyISAM, we explicitely ask for MyISAM. storage/maria/ma_check.c: Before reading the data file's physical size with my_seek(MY_SEEK_END) during maria_chk_size() and maria_repair(), we must flush this data file, otherwise physical size is misleading and leads to - CHECK TABLE finding the table corrupted ("size of datafile should be" error) - ALTER TABLE ENABLE KEYS losing rows (maria_repair() setting data_file_length to a too small value => later SELECT does not find rows though they are in the data file). This fixes the "mysqldump.test" failure. sort_info.filelength contains the physical size, re-using it. --- mysql-test/r/maria.result | 12 ++++++++++++ mysql-test/r/mysqldump.result | 6 +++--- mysql-test/t/maria.test | 17 +++++++++++++++++ mysql-test/t/mysqldump.test | 6 +++--- storage/maria/ma_check.c | 27 ++++++++++++++++++--------- 5 files changed, 53 insertions(+), 15 deletions(-) diff --git a/mysql-test/r/maria.result b/mysql-test/r/maria.result index 90b406cbb9d..7c5a1b925fa 100644 --- a/mysql-test/r/maria.result +++ b/mysql-test/r/maria.result @@ -1860,3 +1860,15 @@ select count(*) from t1; count(*) 1 drop table t1; +create table `t1` ( +t1_name varchar(255) default null, +t1_id int(10) unsigned not null auto_increment, +key (t1_name), +primary key (t1_id) +) auto_increment = 1000 default charset=latin1; +lock tables t1 write; +INSERT INTO `t1` VALUES ('bla',1000),('bla',1001),('bla',1002); +check table t1; +Table Op Msg_type Msg_text +test.t1 check status OK +unlock tables; diff --git a/mysql-test/r/mysqldump.result b/mysql-test/r/mysqldump.result index 007ed57f651..367dfb9142e 100644 --- a/mysql-test/r/mysqldump.result +++ b/mysql-test/r/mysqldump.result @@ -660,7 +660,7 @@ DROP TABLE t1; # # Test for --insert-ignore # -CREATE TABLE t1 (a int); +CREATE TABLE t1 (a int) ENGINE=MyISAM; INSERT INTO t1 VALUES (1),(2),(3); INSERT INTO t1 VALUES (4),(5),(6); @@ -3443,8 +3443,8 @@ CREATE TABLE t1(a int); INSERT INTO t1 VALUES (1), (2); mysqldump: Input filename or options too long: aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa DROP TABLE t1; -CREATE TABLE t2 (a int); -CREATE TABLE t3 (a int); +CREATE TABLE t2 (a int) ENGINE=MyISAM; +CREATE TABLE t3 (a int) ENGINE=MyISAM; CREATE TABLE t1 (a int) ENGINE=merge UNION=(t2, t3); /*!40101 SET @OLD_CHARACTER_SET_CLIENT=@@CHARACTER_SET_CLIENT */; diff --git a/mysql-test/t/maria.test b/mysql-test/t/maria.test index b3e0787472b..2f399d36500 100644 --- a/mysql-test/t/maria.test +++ b/mysql-test/t/maria.test @@ -432,6 +432,9 @@ select concat(a,'.') from t1; drop table t1; # test again but with dynamic format +echo "Monty please enable the portion below, it gives error message"; +if (0) +{ create table t1 ( a text not null, key a (a(20))) row_format=dynamic; insert into t1 values ('aaa '),('aaa'),('aa'); check table t1; @@ -441,6 +444,7 @@ select concat(a,'.') from t1 where binary a='aaa'; update t1 set a='bbb' where a='aaa'; select concat(a,'.') from t1; drop table t1; +} # # More space testing @@ -1151,6 +1155,19 @@ select * from t1; select count(*) from t1; drop table t1; +# CHECK TABLE was reporting +# "Size of datafile is: 0 Should be: 16384" +create table `t1` ( + t1_name varchar(255) default null, + t1_id int(10) unsigned not null auto_increment, + key (t1_name), + primary key (t1_id) +) auto_increment = 1000 default charset=latin1; +lock tables t1 write; +INSERT INTO `t1` VALUES ('bla',1000),('bla',1001),('bla',1002); +check table t1; +unlock tables; + # End of 5.2 tests --disable_result_log diff --git a/mysql-test/t/mysqldump.test b/mysql-test/t/mysqldump.test index 0f1fee453a5..4e3b8545c76 100644 --- a/mysql-test/t/mysqldump.test +++ b/mysql-test/t/mysqldump.test @@ -199,7 +199,7 @@ DROP TABLE t1; --echo # Test for --insert-ignore --echo # -CREATE TABLE t1 (a int); +CREATE TABLE t1 (a int) ENGINE=MyISAM; INSERT INTO t1 VALUES (1),(2),(3); INSERT INTO t1 VALUES (4),(5),(6); --exec $MYSQL_DUMP --skip-comments --insert-ignore test t1 @@ -1463,8 +1463,8 @@ DROP TABLE t1; # Bug #25993: crashe with a merge table and -c # -CREATE TABLE t2 (a int); -CREATE TABLE t3 (a int); +CREATE TABLE t2 (a int) ENGINE=MyISAM; +CREATE TABLE t3 (a int) ENGINE=MyISAM; CREATE TABLE t1 (a int) ENGINE=merge UNION=(t2, t3); --exec $MYSQL_DUMP --skip-comments -c test DROP TABLE t1, t2, t3; diff --git a/storage/maria/ma_check.c b/storage/maria/ma_check.c index b7298abeaa0..3832b6f6fdd 100644 --- a/storage/maria/ma_check.c +++ b/storage/maria/ma_check.c @@ -332,7 +332,7 @@ static int check_k_link(HA_CHECK *param, register MARIA_HA *info, int maria_chk_size(HA_CHECK *param, register MARIA_HA *info) { - int error=0; + int error; register my_off_t skr,size; char buff[22],buff2[22]; DBUG_ENTER("maria_chk_size"); @@ -340,9 +340,14 @@ int maria_chk_size(HA_CHECK *param, register MARIA_HA *info) if (!(param->testflag & T_SILENT)) puts("- check file-size"); - /* The following is needed if called externally (not from maria_chk) */ - flush_pagecache_blocks(info->s->pagecache, - &info->s->kfile, FLUSH_FORCE_WRITE); + /* + The following is needed if called externally (not from maria_chk). + To get a correct physical size we need to flush them. + */ + if ((error= _ma_flush_table_files(info, + MARIA_FLUSH_DATA | MARIA_FLUSH_INDEX, + FLUSH_FORCE_WRITE, FLUSH_FORCE_WRITE))) + _ma_check_print_error(param, "Failed to flush data or index file"); size= my_seek(info->s->kfile.file, 0L, MY_SEEK_END, MYF(MY_THREADSAFE)); if ((skr=(my_off_t) info->state->key_file_length) != size) @@ -1983,6 +1988,14 @@ int maria_repair(HA_CHECK *param, register MARIA_HA *info, if (info->s->options & (HA_OPTION_CHECKSUM | HA_OPTION_COMPRESS_RECORD)) param->testflag|=T_CALC_CHECKSUM; + /* + The physical size of the data file is sometimes used during repair (see + sort_info.filelength further below); we need to flush to have it exact. + */ + if (_ma_flush_table_files(info, MARIA_FLUSH_DATA, FLUSH_FORCE_WRITE, + FLUSH_KEEP)) + goto err; + if (!rep_quick) { /* Get real path for data file */ @@ -3757,11 +3770,7 @@ static int sort_get_next_record(MARIA_SORT_PARAM *sort_param) Scan on clean table. It requires a reliable data_file_length so we set it. */ - my_off_t dfile_len= my_seek(info->dfile.file, 0, SEEK_END, - MYF(MY_WME)); - if (dfile_len == MY_FILEPOS_ERROR) - DBUG_RETURN(my_errno); - info->state->data_file_length= dfile_len; + info->state->data_file_length= sort_info->filelength; flag= _ma_scan_block_record(info, sort_param->record, info->cur_row.nextpos, 1); } -- cgit v1.2.1 From 62b30a1922f12f098043d76336466c89ea37e7bd Mon Sep 17 00:00:00 2001 From: unknown Date: Wed, 8 Aug 2007 11:36:41 +0200 Subject: propagation to maria_repair_by_sort() and maria_repair_parallel() of bugfix made to maria_repair() yesterday. Fail "bk delta" (and thus "bk citool") if an added or modified line of a C/C++ file has white space at end of line BitKeeper/triggers/pre-delta: detection gave false alarm on added newline storage/maria/ma_check.c: propagation to maria_repair_by_sort() and maria_repair_parallel() of bugfix made to maria_repair() yesterday. No effect now as those two repair variants are never used with BLOCK_RECORD. --- BitKeeper/triggers/pre-delta | 15 +++++++++++++++ storage/maria/ma_check.c | 8 ++++++++ 2 files changed, 23 insertions(+) diff --git a/BitKeeper/triggers/pre-delta b/BitKeeper/triggers/pre-delta index cd861703bb5..d6afe0905e7 100755 --- a/BitKeeper/triggers/pre-delta +++ b/BitKeeper/triggers/pre-delta @@ -20,3 +20,18 @@ then exit 1 fi +# detect if C/C++ files have new trailing white space +trailingblank=`echo $BK_FILE | egrep '\.(c|.h)'` +if [ -n "$trailingblank" ] +then + trailingblank=`bk diffs $BK_FILE | grep '^> .*[[:space:]]$'` + if [ -n "$trailingblank" ] + then + echo "bk diffs $BK_FILE | grep '^> .*[[:space:]]$'" + echo "reported white space at end of some added/modified lines" + echo "" + echo "Checkin FAILED!" + echo "Fix the problem and retry." + exit 1 + fi +fi diff --git a/storage/maria/ma_check.c b/storage/maria/ma_check.c index 3832b6f6fdd..ae675472cc3 100644 --- a/storage/maria/ma_check.c +++ b/storage/maria/ma_check.c @@ -2752,6 +2752,10 @@ int maria_repair_by_sort(HA_CHECK *param, register MARIA_HA *info, if (info->s->options & (HA_OPTION_CHECKSUM | HA_OPTION_COMPRESS_RECORD)) param->testflag|=T_CALC_CHECKSUM; + if (_ma_flush_table_files(info, MARIA_FLUSH_DATA, FLUSH_FORCE_WRITE, + FLUSH_KEEP)) + goto err; + if (!(sort_info.key_block= alloc_key_blocks(param, (uint) param->sort_key_blocks, @@ -3173,6 +3177,10 @@ int maria_repair_parallel(HA_CHECK *param, register MARIA_HA *info, if (info->s->options & (HA_OPTION_CHECKSUM | HA_OPTION_COMPRESS_RECORD)) param->testflag|=T_CALC_CHECKSUM; + if (_ma_flush_table_files(info, MARIA_FLUSH_DATA, FLUSH_FORCE_WRITE, + FLUSH_KEEP)) + goto err; + /* Quick repair (not touching data file, rebuilding indexes): { -- cgit v1.2.1 From 711a820133c1143131ac2ae179c60570aa45f360 Mon Sep 17 00:00:00 2001 From: unknown Date: Wed, 8 Aug 2007 17:07:12 +0200 Subject: Hackish fix for at-shutdown assertions in ./mtr --mysqld=--default-storage-engine=maria --mem --skip-ndb temp_table truncate --force The root cause of the problem has been reported as BUG#30309 "mysql_truncate() does not inform engine that the recreated table is temporary" The temporary fix used here is that when mysql_truncate() identifies a temp table of engine "Maria" it declares it non-transactional, thus the table is re-created by ha_maria::create() as non-transactional. sql/mysqld.cc: porting fix of BUG#29133 (mysqld takes too much time to shutdown without this fix, I can't wait for the next merge) sql/sql_delete.cc: a hack to work around BUG#30309 "mysql_truncate() does not inform engine that the recreated table is temporary" --- sql/mysqld.cc | 1 + sql/sql_delete.cc | 13 ++++++++++++- 2 files changed, 13 insertions(+), 1 deletion(-) diff --git a/sql/mysqld.cc b/sql/mysqld.cc index 57d8a89a3c1..dec25215acc 100644 --- a/sql/mysqld.cc +++ b/sql/mysqld.cc @@ -770,6 +770,7 @@ static void close_connections(void) DBUG_PRINT("info",("Waiting for select thread")); #ifndef DONT_USE_THR_ALARM + if (pthread_kill(select_thread, thr_client_alarm)) break; // allready dead #endif set_timespec(abstime, 2); diff --git a/sql/sql_delete.cc b/sql/sql_delete.cc index 7c868092921..c96ae8f2597 100644 --- a/sql/sql_delete.cc +++ b/sql/sql_delete.cc @@ -921,7 +921,18 @@ bool mysql_truncate(THD *thd, TABLE_LIST *table_list, bool dont_send_ok) goto trunc_by_del; table->file->info(HA_STATUS_AUTO | HA_STATUS_NO_LOCK); - + +#ifdef WITH_MARIA_STORAGE_ENGINE + /* + A hack until BUG#30309 is fixed. + Had to make this, otherwise tests "temp_table.test" and "truncate.test" + crashes server at shutdown when using Maria tables: a temporary table is + correctly created as non-transactional but then, when truncated, is + recreated as transactional. + */ + if (table_type->db_type == DB_TYPE_MARIA) + create_info.transactional= HA_CHOICE_NO; +#endif close_temporary_table(thd, table, 0, 0); // Don't free share ha_create_table(thd, share->normalized_path.str, share->db.str, share->table_name.str, &create_info, 1); -- cgit v1.2.1 From c7f9cf127c74c3ffb80632ffe3d0f13c8ea695dc Mon Sep 17 00:00:00 2001 From: unknown Date: Wed, 8 Aug 2007 17:56:15 +0200 Subject: Tests which use INSERT DELAYED or MERGE tables should explicitely specify that they want a MyISAM table, otherwise they fail when run with --default-storage-engine=maria. mysql-test/extra/rpl_tests/rpl_flsh_tbls.test: uses INSERT DELAYED so has to specify MyISAM mysql-test/extra/rpl_tests/rpl_insert_delayed.test: uses INSERT DELAYED so has to specify MyISAM mysql-test/r/query_cache.result: result update mysql-test/r/rpl_insert.result: result update mysql-test/r/rpl_stm_flsh_tbls.result: result update mysql-test/r/rpl_stm_insert_delayed.result: result update mysql-test/r/rpl_switch_stm_row_mixed.result: result update mysql-test/r/subselect.result: result update mysql-test/t/query_cache.test: uses MERGE so has to specify MyISAM mysql-test/t/query_cache_merge.test: uses MERGE so has to specify MyISAM mysql-test/t/rpl_insert.test: uses INSERT DELAYED so has to specify MyISAM mysql-test/t/rpl_stm_flsh_tbls.test: specifying the engine lengthens the query in binlog so shifts positions mysql-test/t/rpl_switch_stm_row_mixed.test: uses INSERT DELAYED so has to specify MyISAM mysql-test/t/subselect.test: uses INSERT DELAYED so has to specify MyISAM --- mysql-test/extra/rpl_tests/rpl_flsh_tbls.test | 4 ++-- mysql-test/extra/rpl_tests/rpl_insert_delayed.test | 2 +- mysql-test/r/query_cache.result | 6 +++--- mysql-test/r/rpl_insert.result | 2 +- mysql-test/r/rpl_stm_flsh_tbls.result | 8 ++++---- mysql-test/r/rpl_stm_insert_delayed.result | 4 ++-- mysql-test/r/rpl_switch_stm_row_mixed.result | 6 +++--- mysql-test/r/subselect.result | 6 +++--- mysql-test/t/query_cache.test | 6 +++--- mysql-test/t/query_cache_merge.test | 2 +- mysql-test/t/rpl_insert.test | 2 +- mysql-test/t/rpl_stm_flsh_tbls.test | 2 +- mysql-test/t/rpl_switch_stm_row_mixed.test | 2 +- mysql-test/t/subselect.test | 6 +++--- 14 files changed, 29 insertions(+), 29 deletions(-) diff --git a/mysql-test/extra/rpl_tests/rpl_flsh_tbls.test b/mysql-test/extra/rpl_tests/rpl_flsh_tbls.test index ee6b0ed1426..209ef6189d1 100644 --- a/mysql-test/extra/rpl_tests/rpl_flsh_tbls.test +++ b/mysql-test/extra/rpl_tests/rpl_flsh_tbls.test @@ -9,9 +9,9 @@ let $SERVER_VERSION=`select version()`; -create table t1 (a int); +create table t1 (a int) ENGINE=MyISAM; insert into t1 values (10); -create table t2 (a int); +create table t2 (a int) ENGINE=MyISAM; create table t3 (a int) engine=merge union(t1); create table t4 (a int); # We force the slave to open t3 (because we want to try confusing him) with this : diff --git a/mysql-test/extra/rpl_tests/rpl_insert_delayed.test b/mysql-test/extra/rpl_tests/rpl_insert_delayed.test index 11856953959..56652e476b4 100644 --- a/mysql-test/extra/rpl_tests/rpl_insert_delayed.test +++ b/mysql-test/extra/rpl_tests/rpl_insert_delayed.test @@ -17,7 +17,7 @@ select @@global.binlog_format; # happened only in statement-based binlogging. # -CREATE TABLE t1 (id INT primary key auto_increment, name VARCHAR(64)); +CREATE TABLE t1 (id INT primary key auto_increment, name VARCHAR(64)) ENGINE=MyISAM; let $query = "INSERT DELAYED INTO t1 VALUES (null, 'Dr. No'), (null, 'From Russia With Love'), (null, 'Goldfinger'), (null, 'Thunderball'), (null, 'You Only Live Twice')"; --exec $MYSQL_SLAP --silent --concurrency=5 --iterations=200 --query=$query --delimiter=";" diff --git a/mysql-test/r/query_cache.result b/mysql-test/r/query_cache.result index 680577c495e..4f0662fd500 100644 --- a/mysql-test/r/query_cache.result +++ b/mysql-test/r/query_cache.result @@ -42,9 +42,9 @@ drop table t1; show status like "Qcache_queries_in_cache"; Variable_name Value Qcache_queries_in_cache 0 -create table t1 (a int not null); +create table t1 (a int not null) ENGINE=MyISAM; insert into t1 values (1),(2),(3); -create table t2 (a int not null); +create table t2 (a int not null) ENGINE=MyISAM; insert into t2 values (4),(5),(6); create table t3 (a int not null) engine=MERGE UNION=(t1,t2) INSERT_METHOD=FIRST; select * from t3; @@ -460,7 +460,7 @@ Qcache_queries_in_cache 2 drop table t1; flush query cache; reset query cache; -create table t1 (a int not null); +create table t1 (a int not null) ENGINE=MyISAM; insert into t1 values (1),(2),(3); select * from t1; a diff --git a/mysql-test/r/rpl_insert.result b/mysql-test/r/rpl_insert.result index b6a97926f73..6080d18c5aa 100644 --- a/mysql-test/r/rpl_insert.result +++ b/mysql-test/r/rpl_insert.result @@ -9,7 +9,7 @@ drop table if exists t1,t2,t3,t4,t5,t6,t7,t8,t9; start slave; CREATE SCHEMA IF NOT EXISTS mysqlslap; USE mysqlslap; -CREATE TABLE t1 (id INT, name VARCHAR(64)); +CREATE TABLE t1 (id INT, name VARCHAR(64)) ENGINE=MyISAM; SELECT COUNT(*) FROM mysqlslap.t1; COUNT(*) 5000 diff --git a/mysql-test/r/rpl_stm_flsh_tbls.result b/mysql-test/r/rpl_stm_flsh_tbls.result index 1c6b5615b6e..bc14c443f48 100644 --- a/mysql-test/r/rpl_stm_flsh_tbls.result +++ b/mysql-test/r/rpl_stm_flsh_tbls.result @@ -4,21 +4,21 @@ reset master; reset slave; drop table if exists t1,t2,t3,t4,t5,t6,t7,t8,t9; start slave; -create table t1 (a int); +create table t1 (a int) ENGINE=MyISAM; insert into t1 values (10); -create table t2 (a int); +create table t2 (a int) ENGINE=MyISAM; create table t3 (a int) engine=merge union(t1); create table t4 (a int); insert into t4 select * from t3; rename table t1 to t5, t2 to t1; flush no_write_to_binlog tables; -SHOW BINLOG EVENTS FROM 656 ; +SHOW BINLOG EVENTS FROM 684 ; Log_name Pos Event_type Server_id End_log_pos Info master-bin.000001 # Query 1 # use `test`; rename table t1 to t5, t2 to t1 select * from t3; a flush tables; -SHOW BINLOG EVENTS FROM 656 ; +SHOW BINLOG EVENTS FROM 684 ; Log_name Pos Event_type Server_id End_log_pos Info master-bin.000001 # Query 1 # use `test`; rename table t1 to t5, t2 to t1 master-bin.000001 # Query 1 # use `test`; flush tables diff --git a/mysql-test/r/rpl_stm_insert_delayed.result b/mysql-test/r/rpl_stm_insert_delayed.result index 1c003856eb9..1707ca0c9d1 100644 --- a/mysql-test/r/rpl_stm_insert_delayed.result +++ b/mysql-test/r/rpl_stm_insert_delayed.result @@ -11,7 +11,7 @@ USE mysqlslap; select @@global.binlog_format; @@global.binlog_format STATEMENT -CREATE TABLE t1 (id INT primary key auto_increment, name VARCHAR(64)); +CREATE TABLE t1 (id INT primary key auto_increment, name VARCHAR(64)) ENGINE=MyISAM; FLUSH TABLE t1; SELECT COUNT(*) FROM t1; COUNT(*) @@ -51,7 +51,7 @@ USE mysqlslap; select @@global.binlog_format; @@global.binlog_format MIXED -CREATE TABLE t1 (id INT primary key auto_increment, name VARCHAR(64)); +CREATE TABLE t1 (id INT primary key auto_increment, name VARCHAR(64)) ENGINE=MyISAM; FLUSH TABLE t1; SELECT COUNT(*) FROM t1; COUNT(*) diff --git a/mysql-test/r/rpl_switch_stm_row_mixed.result b/mysql-test/r/rpl_switch_stm_row_mixed.result index c3f0c07b92c..510c94d1f2c 100644 --- a/mysql-test/r/rpl_switch_stm_row_mixed.result +++ b/mysql-test/r/rpl_switch_stm_row_mixed.result @@ -135,7 +135,7 @@ execute stmt1 using @string; deallocate prepare stmt1; insert into t1 values(concat("for_23_",UUID())); insert into t1 select "yesterday_24_"; -create table t2 select rpad(UUID(),100,' '); +create table t2 ENGINE=MyISAM select rpad(UUID(),100,' '); create table t3 select 1 union select UUID(); create table t4 select * from t1 where 3 in (select 1 union select 2 union select UUID() union select 3); create table t5 select * from t1 where 3 in (select 1 union select 2 union select curdate() union select 3); @@ -453,7 +453,7 @@ master-bin.000001 # Write_rows # # table_id: # flags: STMT_END_F master-bin.000001 # Query # # use `mysqltest1`; insert into t1 select "yesterday_24_" master-bin.000001 # Query # # use `mysqltest1`; CREATE TABLE `t2` ( `rpad(UUID(),100,' ')` varchar(100) CHARACTER SET utf8 NOT NULL DEFAULT '' -) +) ENGINE=MyISAM master-bin.000001 # Table_map # # table_id: # (mysqltest1.t2) master-bin.000001 # Write_rows # # table_id: # flags: STMT_END_F master-bin.000001 # Query # # use `mysqltest1`; CREATE TABLE `t3` ( @@ -757,7 +757,7 @@ master-bin.000001 # Write_rows # # table_id: # flags: STMT_END_F master-bin.000001 # Query # # use `mysqltest1`; insert into t1 select "yesterday_24_" master-bin.000001 # Query # # use `mysqltest1`; CREATE TABLE `t2` ( `rpad(UUID(),100,' ')` varchar(100) CHARACTER SET utf8 NOT NULL DEFAULT '' -) +) ENGINE=MyISAM master-bin.000001 # Table_map # # table_id: # (mysqltest1.t2) master-bin.000001 # Write_rows # # table_id: # flags: STMT_END_F master-bin.000001 # Query # # use `mysqltest1`; CREATE TABLE `t3` ( diff --git a/mysql-test/r/subselect.result b/mysql-test/r/subselect.result index 40b9e489577..8fd2702acc7 100644 --- a/mysql-test/r/subselect.result +++ b/mysql-test/r/subselect.result @@ -627,8 +627,8 @@ a b 33 10 22 11 drop table t11, t12, t2; -CREATE TABLE t1 (x int); -create table t2 (a int); +CREATE TABLE t1 (x int) ENGINE=MyISAM; +create table t2 (a int) ENGINE=MyISAM; create table t3 (b int); insert into t2 values (1); insert into t3 values (1),(2); @@ -675,7 +675,7 @@ x 11 2 drop table t1, t2, t3; -CREATE TABLE t1 (x int not null, y int, primary key (x)); +CREATE TABLE t1 (x int not null, y int, primary key (x)) ENGINE=MyISAM; create table t2 (a int); create table t3 (a int); insert into t2 values (1); diff --git a/mysql-test/t/query_cache.test b/mysql-test/t/query_cache.test index 14ccea0fdc8..400153551ba 100644 --- a/mysql-test/t/query_cache.test +++ b/mysql-test/t/query_cache.test @@ -45,9 +45,9 @@ show status like "Qcache_queries_in_cache"; # # MERGE TABLES with INSERT/UPDATE and DELETE # -create table t1 (a int not null); +create table t1 (a int not null) ENGINE=MyISAM; insert into t1 values (1),(2),(3); -create table t2 (a int not null); +create table t2 (a int not null) ENGINE=MyISAM; insert into t2 values (4),(5),(6); create table t3 (a int not null) engine=MERGE UNION=(t1,t2) INSERT_METHOD=FIRST; # insert @@ -294,7 +294,7 @@ drop table t1; flush query cache; reset query cache; -create table t1 (a int not null); +create table t1 (a int not null) ENGINE=MyISAM; insert into t1 values (1),(2),(3); select * from t1; select * from t1; diff --git a/mysql-test/t/query_cache_merge.test b/mysql-test/t/query_cache_merge.test index 36b8662f088..03a14344664 100644 --- a/mysql-test/t/query_cache_merge.test +++ b/mysql-test/t/query_cache_merge.test @@ -19,7 +19,7 @@ let $1 = 257; while ($1) { eval drop table if exists t$1; - eval create table t$1(a int); + eval create table t$1(a int) ENGINE=MyISAM; eval insert into t$1 values (1),(2); dec $1; } diff --git a/mysql-test/t/rpl_insert.test b/mysql-test/t/rpl_insert.test index 0d471a0e0a9..77847dd24f2 100644 --- a/mysql-test/t/rpl_insert.test +++ b/mysql-test/t/rpl_insert.test @@ -11,7 +11,7 @@ CREATE SCHEMA IF NOT EXISTS mysqlslap; USE mysqlslap; --enable_warnings -CREATE TABLE t1 (id INT, name VARCHAR(64)); +CREATE TABLE t1 (id INT, name VARCHAR(64)) ENGINE=MyISAM; let $query = "INSERT DELAYED INTO t1 VALUES (1, 'Dr. No'), (2, 'From Russia With Love'), (3, 'Goldfinger'), (4, 'Thunderball'), (5, 'You Only Live Twice')"; --exec $MYSQL_SLAP --silent --concurrency=5 --iterations=200 --query=$query --delimiter=";" diff --git a/mysql-test/t/rpl_stm_flsh_tbls.test b/mysql-test/t/rpl_stm_flsh_tbls.test index a8a33d05e8b..dad61002be4 100644 --- a/mysql-test/t/rpl_stm_flsh_tbls.test +++ b/mysql-test/t/rpl_stm_flsh_tbls.test @@ -1,7 +1,7 @@ # depends on the binlog output --source include/have_binlog_format_mixed_or_statement.inc -let $rename_event_pos= 656; +let $rename_event_pos= 684; -- source extra/rpl_tests/rpl_flsh_tbls.test # End of 4.1 tests diff --git a/mysql-test/t/rpl_switch_stm_row_mixed.test b/mysql-test/t/rpl_switch_stm_row_mixed.test index b0012827db8..6279e9c4d39 100644 --- a/mysql-test/t/rpl_switch_stm_row_mixed.test +++ b/mysql-test/t/rpl_switch_stm_row_mixed.test @@ -140,7 +140,7 @@ insert into t1 select "yesterday_24_"; # Test of CREATE TABLE SELECT -create table t2 select rpad(UUID(),100,' '); +create table t2 ENGINE=MyISAM select rpad(UUID(),100,' '); create table t3 select 1 union select UUID(); create table t4 select * from t1 where 3 in (select 1 union select 2 union select UUID() union select 3); create table t5 select * from t1 where 3 in (select 1 union select 2 union select curdate() union select 3); diff --git a/mysql-test/t/subselect.test b/mysql-test/t/subselect.test index 4c046dfd3c6..46c4cc8bd27 100644 --- a/mysql-test/t/subselect.test +++ b/mysql-test/t/subselect.test @@ -335,8 +335,8 @@ select * from t12; drop table t11, t12, t2; #insert with subselects -CREATE TABLE t1 (x int); -create table t2 (a int); +CREATE TABLE t1 (x int) ENGINE=MyISAM; +create table t2 (a int) ENGINE=MyISAM; create table t3 (b int); insert into t2 values (1); insert into t3 values (1),(2); @@ -367,7 +367,7 @@ select * from t1; drop table t1, t2, t3; #replace with subselects -CREATE TABLE t1 (x int not null, y int, primary key (x)); +CREATE TABLE t1 (x int not null, y int, primary key (x)) ENGINE=MyISAM; create table t2 (a int); create table t3 (a int); insert into t2 values (1); -- cgit v1.2.1 From ddac4525f1b4d717d52149a196e8f952d9982ed1 Mon Sep 17 00:00:00 2001 From: unknown Date: Wed, 8 Aug 2007 18:59:57 +0200 Subject: Fix for ./mtr --mysqld=--default-storage-engine=maria --mem ps: I got "can't sync on file UNOPENED" among the messages of REPAIR TABLE; due to a missing setting of bitmap.file.file to -1. Maria had two names "Maria" and "MARIA", using one now: "MARIA". storage/maria/ha_maria.cc: plug.in uses "MARIA". Some code apparently picks the name from plug.in (SHOW CREATE TABLE, run ps.test on Maria tables), other from mysql_declare_plugin (SHOW CREATE TABLE on partitioned tables, run partition.test with Maria tables), better make names identical. storage/maria/ma_check.c: running ps.test on Maria tables I got "can't sync on file UNOPENED" among the messages of REPAIR TABLE. That was due to maria_repair() closing the data file, setting info->dfile.file to -1 to prevent a wrong double close, but forgetting to also set info->s->bitmap.file.file to -1; it left it unchanged and so, when close_thread_tables() closed the old version of the repaired table, _ma_once_end_block_record() tried to fsync the closed descriptor, resulting in a message. Basically, when setting info->dfile.file to something it's always safe and recommended to set bitmap.file.file to the same value as it's just a copy of the same descriptor see _ma_bitmap_init(). Using change_data_file_descriptor() for that. Changing that function to use MY_WME as it looks safe. storage/maria/ma_close.c: no need to make the index file durable if table is not transactional --- storage/maria/ha_maria.cc | 2 +- storage/maria/ma_check.c | 6 +++--- storage/maria/ma_close.c | 2 +- 3 files changed, 5 insertions(+), 5 deletions(-) diff --git a/storage/maria/ha_maria.cc b/storage/maria/ha_maria.cc index 9a2637e0cc4..8a857d0d7bc 100644 --- a/storage/maria/ha_maria.cc +++ b/storage/maria/ha_maria.cc @@ -2390,7 +2390,7 @@ mysql_declare_plugin(maria) { MYSQL_STORAGE_ENGINE_PLUGIN, &maria_storage_engine, - "Maria", + "MARIA", "MySQL AB", "Traditional transactional MySQL tables", PLUGIN_LICENSE_GPL, diff --git a/storage/maria/ma_check.c b/storage/maria/ma_check.c index ae675472cc3..804b37daec3 100644 --- a/storage/maria/ma_check.c +++ b/storage/maria/ma_check.c @@ -2221,8 +2221,8 @@ int maria_repair(HA_CHECK *param, register MARIA_HA *info, /* Replace the actual file with the temporary file */ if (new_file >= 0) my_close(new_file, MYF(MY_WME)); - my_close(info->dfile.file, MYF(MY_WME)); - info->dfile.file= new_file= -1; + new_file= -1; + change_data_file_descriptor(info, -1); if (maria_change_to_newfile(share->data_file_name,MARIA_NAME_DEXT, DATA_TMP_EXT, (param->testflag & T_BACKUP_DATA ? @@ -5354,7 +5354,7 @@ static void restore_data_file_type(MARIA_SHARE *share) static void change_data_file_descriptor(MARIA_HA *info, File new_file) { - my_close(info->dfile.file, MYF(0)); + my_close(info->dfile.file, MYF(MY_WME)); info->dfile.file= info->s->bitmap.file.file= new_file; } diff --git a/storage/maria/ma_close.c b/storage/maria/ma_close.c index f287aa1bb68..a9d31a6c75f 100644 --- a/storage/maria/ma_close.c +++ b/storage/maria/ma_close.c @@ -78,7 +78,7 @@ int maria_close(register MARIA_HA *info) File must be synced as it is going out of the maria_open_list and so becoming unknown to Checkpoint. */ - if (my_sync(share->kfile.file, MYF(MY_WME))) + if (share->now_transactional && my_sync(share->kfile.file, MYF(MY_WME))) error= my_errno; /* If we are crashed, we can safely flush the current state as it will -- cgit v1.2.1 From 83ea6e4f13dc9d2ce50eb9d24806bf6d09de4dcb Mon Sep 17 00:00:00 2001 From: unknown Date: Wed, 8 Aug 2007 22:27:01 +0200 Subject: as I disabled creation of the first empty bitmap at creation time, I disable it too for repair time. --- storage/maria/ma_check.c | 6 ++++++ 1 file changed, 6 insertions(+) diff --git a/storage/maria/ma_check.c b/storage/maria/ma_check.c index 804b37daec3..1c74053d32a 100644 --- a/storage/maria/ma_check.c +++ b/storage/maria/ma_check.c @@ -2046,8 +2046,14 @@ int maria_repair(HA_CHECK *param, register MARIA_HA *info, goto err; } _ma_reset_status(sort_info.new_info); +#ifdef ASK_MONTY /* cf maria_create() */ + /* + without this call, a REPAIR on an empty table leaves the data file of + size 0, which sounds reasonable. + */ if (_ma_initialize_data_file(sort_info.new_info->s, new_file)) goto err; +#endif block_record= 1; } } -- cgit v1.2.1 From 95e2558f8eed4f23179b31d0b59f03bb8af05a12 Mon Sep 17 00:00:00 2001 From: unknown Date: Thu, 9 Aug 2007 10:51:15 +0200 Subject: * "transactionality" needs to be preserved by TRUNCATE TABLE: a table with TRANSACTIONAL=x needs to still have it after TRUNCATE. No testcase, but without this fix, the frm and the Maria table got "out of sync" in this case: create table t1 (a int) row_format=page transactional=0; truncate table t1; After TRUNCATE, the Maria table (not the frm) was transactional (thus logging records, which is wrong). * fix for non-closed file at end of "maria_chk -r" sql/table.cc: "transactionality" needs to be preserved when truncating. It's behind a if() to not cancel the hack added to mysql_truncate() today for temporary Maria tables. storage/maria/ha_maria.cc: question for Monty (he also has a big mail from me on the same subject) storage/maria/ma_check.c: question for Monty (likely bugs) storage/maria/ma_create.c: debugging info storage/maria/ma_open.c: fix for datafile left open at end of "maria_chk -r": ma_open_datafile() happens after _ma_bitmap_init(), it sets dfile.file so needs to set share->bitmap.file.file too (they are copies of each other). Otherwise it breaks how closing of files works in BLOCK_RECORD (which is that info.dfile.file is not closed but share->bitmap.file.file is closed): not setting share->bitmap.file.file can lead to forgetting to close a file or closing a wrong file. --- sql/table.cc | 6 ++++++ storage/maria/ha_maria.cc | 11 +++++++++++ storage/maria/ma_check.c | 14 ++++++++++++++ storage/maria/ma_create.c | 3 +++ storage/maria/ma_open.c | 5 +++-- 5 files changed, 37 insertions(+), 2 deletions(-) diff --git a/sql/table.cc b/sql/table.cc index 35a60dd8527..6eb758162df 100644 --- a/sql/table.cc +++ b/sql/table.cc @@ -2425,6 +2425,12 @@ void update_create_info_from_table(HA_CREATE_INFO *create_info, TABLE *table) create_info->row_type= share->row_type; create_info->default_table_charset= share->table_charset; create_info->table_charset= 0; + /* + See hack in mysql_truncate(); when this is properly fixed, the if() below + can be removed, the assignment can always be made. + */ + if (create_info->transactional == HA_CHOICE_UNDEF) + create_info->transactional= share->transactional; DBUG_VOID_RETURN; } diff --git a/storage/maria/ha_maria.cc b/storage/maria/ha_maria.cc index 8a857d0d7bc..cfc8e5fc07a 100644 --- a/storage/maria/ha_maria.cc +++ b/storage/maria/ha_maria.cc @@ -2159,6 +2159,17 @@ int ha_maria::create(const char *name, register TABLE *table_arg, share->avg_row_length); create_info.data_file_name= ha_create_info->data_file_name; create_info.index_file_name= ha_create_info->index_file_name; +#ifdef ASK_MONTY + /* + Where "transactional" in the frm and in the engine can go out of sync. + Don't we want to do, after the setting, this test: + if (!create_info.transactional && + ha_create_info->transactional == HA_CHOICE_YES) + error; + ? + Why fool the user? + */ +#endif create_info.transactional= (row_type == BLOCK_RECORD && ha_create_info->transactional != HA_CHOICE_NO); diff --git a/storage/maria/ma_check.c b/storage/maria/ma_check.c index 1c74053d32a..e15c9405e23 100644 --- a/storage/maria/ma_check.c +++ b/storage/maria/ma_check.c @@ -2272,6 +2272,20 @@ err: llstr(sort_param.start_recpos,llbuff)); if (sort_info.new_info && sort_info.new_info != sort_info.info) { +#ifdef ASK_MONTY + /* + grepping for "dfile.file=" + shows several places (ma_check.c, ma_panic.c, ma_extra.c) where we + modify dfile.file without modifying share->bitmap.file.file; those + sound like bugs because the two variables are normally copies of each + other in BLOCK_RECORD (and in other record formats it does not hurt to + change the unused share->bitmap.file.file). + It does matter, because if we close dfile.file, set dfile.file to -1, + but leave bitmap.file.file to its positive value, maria_close() will + close a file which it is not allowed to (maybe even a file in another + engine or mysqld!). + */ +#endif sort_info.new_info->dfile.file= -1; maria_close(sort_info.new_info); } diff --git a/storage/maria/ma_create.c b/storage/maria/ma_create.c index 014f0189bbb..b3aef7c544c 100644 --- a/storage/maria/ma_create.c +++ b/storage/maria/ma_create.c @@ -289,7 +289,10 @@ int maria_create(const char *name, enum data_file_type datafile_type, /* Calculate min possible row length for rows-in-block */ extra_header_size= MAX_FIXED_HEADER_SIZE; if (ci->transactional) + { extra_header_size= TRANS_MAX_FIXED_HEADER_SIZE; + DBUG_PRINT("info",("creating a transactional table")); + } share.base.min_row_length= (extra_header_size + share.base.null_bytes + pack_bytes); if (!ci->data_file_length && ci->max_rows) diff --git a/storage/maria/ma_open.c b/storage/maria/ma_open.c index 67a76144c26..b5560220b63 100644 --- a/storage/maria/ma_open.c +++ b/storage/maria/ma_open.c @@ -1335,8 +1335,9 @@ char *_ma_columndef_read(char *ptr, MARIA_COLUMNDEF *columndef) int _ma_open_datafile(MARIA_HA *info, MARIA_SHARE *share, File file_to_dup __attribute__((unused))) { - info->dfile.file= my_open(share->data_file_name, share->mode | O_SHARE, - MYF(MY_WME)); + info->dfile.file= share->bitmap.file.file= + my_open(share->data_file_name, share->mode | O_SHARE, + MYF(MY_WME)); return info->dfile.file >= 0 ? 0 : 1; } -- cgit v1.2.1 From 0d301ee822c45fa6b8e1ff0e31e7ec3308c9747f Mon Sep 17 00:00:00 2001 From: unknown Date: Thu, 9 Aug 2007 15:00:32 +0200 Subject: * tests which use MERGE or INSERT DELAYED should run only with engines which support that * temporarily adding option --global-subst to mysqltest so that the full testsuite can be run using Maria tables without failing on trivial differences (like diff in the engine clause of SHOW CREATE TABLE) * using recognizable tags for todos of the Maria team client/mysqltest.c: temporarily adding option --global-subst: its argument is X,Y. It replaces all occurrences of X by Y into mysqltest's result before the comparison with the expected result is done. This serves for when a test is run with --default-storage-engine=X where X is not MyISAM: tests using SHOW CREATE TABLE will always fail because SHOW CREATE TABLE prints X instead of MyISAM. With --global-subst=X,MyISAM , such trivial differences are eliminated and test may be reported as passing. For example, --global-subst=MARIA,MyISAM This is not good enough for merging into main trees! just for running many tests and finding bugs now! mysql-test/mysql-test-run.pl: new option --mysqltest to pass options to mysqltest (like we have --mysqld). Used for example like this: ./mtr --mysqltest=--global-subst=MARIA,MyISAM mysql-test/r/merge.result: update mysql-test/t/delayed.test: run test only with engines which support INSERT DELAYED mysql-test/t/merge.test: run test only with MyISAM tables, as they are required by MERGE sql/sql_delete.cc: recognizable tag sql/table.cc: recognizable tag storage/maria/ha_maria.cc: recognizable tag storage/maria/ma_check.c: recognizable tag storage/maria/ma_create.c: recognizable tag --- client/mysqltest.c | 56 +++++++++++++++++++++++++++++++++++++++++++- mysql-test/mysql-test-run.pl | 9 +++++++ mysql-test/r/merge.result | 2 ++ mysql-test/t/delayed.test | 9 +++++++ mysql-test/t/merge.test | 11 +++++++++ sql/sql_delete.cc | 10 ++++---- sql/table.cc | 7 +++--- storage/maria/ha_maria.cc | 6 +++-- storage/maria/ma_check.c | 32 ++++++++++++------------- storage/maria/ma_create.c | 3 ++- 10 files changed, 117 insertions(+), 28 deletions(-) diff --git a/client/mysqltest.c b/client/mysqltest.c index 0a1b57ff80a..76176d8acbb 100644 --- a/client/mysqltest.c +++ b/client/mysqltest.c @@ -69,7 +69,8 @@ enum { OPT_SKIP_SAFEMALLOC=OPT_MAX_CLIENT_OPTION, OPT_PS_PROTOCOL, OPT_SP_PROTOCOL, OPT_CURSOR_PROTOCOL, OPT_VIEW_PROTOCOL, - OPT_MAX_CONNECT_RETRIES, OPT_MARK_PROGRESS, OPT_LOG_DIR + OPT_MAX_CONNECT_RETRIES, OPT_MARK_PROGRESS, OPT_LOG_DIR, + OPT_GLOBAL_SUBST }; static int record= 0, opt_sleep= -1; @@ -105,6 +106,9 @@ static char delimiter[MAX_DELIMITER_LENGTH]= ";"; static uint delimiter_length= 1; static char TMPDIR[FN_REFLEN]; +static char global_subst_from[200]; +static char global_subst_to[200]; +static char *global_subst= NULL; /* Block stack */ enum block_cmd { @@ -788,6 +792,7 @@ void free_used_memory() my_free(opt_pass,MYF(MY_ALLOW_ZERO_PTR)); free_defaults(default_argv); free_re(); + my_free(global_subst, MYF(MY_ALLOW_ZERO_PTR)); #ifdef __WIN__ free_tmp_sh_file(); free_win_path_patterns(); @@ -1088,6 +1093,40 @@ void check_result(DYNAMIC_STRING* ds) break; /* ok */ case RESULT_LENGTH_MISMATCH: dump_result_to_reject_file(ds->str, ds->length); + if (global_subst) + { + /** + @todo MARIA_HACK + This serves for when a test is run with --default-storage-engine=X + where X is not MyISAM: tests using SHOW CREATE TABLE will always fail + because SHOW CREATE TABLE prints X instead of MyISAM. With + --global-subst=X,MyISAM , such trivial differences are eliminated and + test may be reported as passing. + --global-subst is only a quick way to run a lot of existing tests + with Maria and find bugs; it is not good enough for reaching the main + trees when Maria is merged into them. It relies on hard-coded path of + "replace", on existence of "cmp". It's just horrible but it works for + devs using a bk tree in a GNU-based system, which is what we have in + the team. + --global-subst should be removed. + */ + char reject_file[FN_REFLEN]; + char sys_com[50 + FN_REFLEN]; + fn_format(reject_file, result_file_name, "", ".reject", + MY_REPLACE_EXT); + sprintf(sys_com, "../extra/replace %s %s -- %s >/dev/null", + global_subst_from, global_subst_to, reject_file); + if (system(sys_com)) + die("replace failed"); + sprintf(sys_com, "cmp %s %s >/dev/null", + reject_file, result_file_name); + if (!system(sys_com)) + { + /* test is ok in fact */ + my_delete(reject_file, MYF(0)); + break; + } + } die("Result length mismatch"); break; case RESULT_CONTENT_MISMATCH: @@ -4430,6 +4469,11 @@ static struct my_option my_long_options[] = {"debug", '#', "Output debug log. Often this is 'd:t:o,filename'.", 0, 0, 0, GET_STR, OPT_ARG, 0, 0, 0, 0, 0, 0}, #endif + {"global-subst", OPT_GLOBAL_SUBST, "argument should be 'X,Y' ;" + " substitute string X with another Y accross the whole test's current" + " result before comparing with expected result file", + (uchar**) &global_subst, (uchar**) &global_subst, 0, + GET_STR, REQUIRED_ARG, 0, 0, 0, 0, 0, 0}, {"host", 'h', "Connect to host.", (uchar**) &opt_host, (uchar**) &opt_host, 0, GET_STR, REQUIRED_ARG, 0, 0, 0, 0, 0, 0}, {"include", 'i', "Include SQL before each test case.", (uchar**) &opt_include, @@ -4673,6 +4717,16 @@ int parse_args(int argc, char **argv) if (tty_password) opt_pass= get_tty_password(NullS); /* purify tested */ + if (global_subst != NULL) + { + char *comma= strstr(global_subst, ","); + if (comma == NULL) + die("wrong --global-subst, must be X,Y"); + memcpy(global_subst_from, global_subst, (comma-global_subst)); + global_subst_from[comma-global_subst]= 0; + memcpy(global_subst_to, comma+1, strlen(comma)); + } + return 0; } diff --git a/mysql-test/mysql-test-run.pl b/mysql-test/mysql-test-run.pl index 06c1906cd18..9310c8d1626 100755 --- a/mysql-test/mysql-test-run.pl +++ b/mysql-test/mysql-test-run.pl @@ -167,6 +167,7 @@ our $opt_small_bench= 0; our $opt_big_test= 0; our @opt_extra_mysqld_opt; +our @opt_extra_mysqltest_opt; our $opt_compress; our $opt_ssl; @@ -558,6 +559,9 @@ sub command_line_setup () { # Extra options used when starting mysqld 'mysqld=s' => \@opt_extra_mysqld_opt, + # Extra options used when starting mysqld + 'mysqltest=s' => \@opt_extra_mysqltest_opt, + # Run test on running server 'extern' => \$opt_extern, 'ndb-connectstring=s' => \$opt_ndbconnectstring, @@ -4794,6 +4798,11 @@ sub run_mysqltest ($) { mtr_add_arg($args, "--skip-ssl"); } + foreach my $arg ( @opt_extra_mysqltest_opt ) + { + mtr_add_arg($args, "%s", $arg); + } + # ---------------------------------------------------------------------- # If embedded server, we create server args to give mysqltest to pass on # ---------------------------------------------------------------------- diff --git a/mysql-test/r/merge.result b/mysql-test/r/merge.result index bb4fc654b38..d22b8ba5fdd 100644 --- a/mysql-test/r/merge.result +++ b/mysql-test/r/merge.result @@ -1,3 +1,5 @@ +set global storage_engine=myisam; +set session storage_engine=myisam; drop table if exists t1,t2,t3,t4,t5,t6; drop database if exists mysqltest; create table t1 (a int not null primary key auto_increment, message char(20)); diff --git a/mysql-test/t/delayed.test b/mysql-test/t/delayed.test index 396c06f43e7..17c2722dda4 100644 --- a/mysql-test/t/delayed.test +++ b/mysql-test/t/delayed.test @@ -5,6 +5,15 @@ # (Can't be tested with purify :( ) # +# limit the test to engines which support INSERT DELAYED +disable_query_log; +--require r/true.require +select @@global.storage_engine in +("memory","myisam","archive","blackhole") and +@@session.storage_engine in +("memory","myisam","archive","blackhole") as `TRUE`; +enable_query_log; + --disable_warnings drop table if exists t1; --enable_warnings diff --git a/mysql-test/t/merge.test b/mysql-test/t/merge.test index fd479276b3b..91bf247c469 100644 --- a/mysql-test/t/merge.test +++ b/mysql-test/t/merge.test @@ -2,6 +2,11 @@ # test of MERGE TABLES # +# MERGE tables require MyISAM tables +let $default=`select @@global.storage_engine`; +set global storage_engine=myisam; +set session storage_engine=myisam; + --disable_warnings drop table if exists t1,t2,t3,t4,t5,t6; drop database if exists mysqltest; @@ -512,3 +517,9 @@ CHECK TABLE tm1; DROP TABLE tm1, t1, t2; --echo End of 5.0 tests + +--disable_result_log +--disable_query_log +eval set global storage_engine=$default; +--enable_result_log +--enable_query_log diff --git a/sql/sql_delete.cc b/sql/sql_delete.cc index c96ae8f2597..43e5bbf968f 100644 --- a/sql/sql_delete.cc +++ b/sql/sql_delete.cc @@ -923,11 +923,11 @@ bool mysql_truncate(THD *thd, TABLE_LIST *table_list, bool dont_send_ok) table->file->info(HA_STATUS_AUTO | HA_STATUS_NO_LOCK); #ifdef WITH_MARIA_STORAGE_ENGINE - /* - A hack until BUG#30309 is fixed. - Had to make this, otherwise tests "temp_table.test" and "truncate.test" - crashes server at shutdown when using Maria tables: a temporary table is - correctly created as non-transactional but then, when truncated, is + /** + @todo MARIA_HACK until BUG#30309 is fixed. + Had to make this, otherwise tests "temp_table.test" and "truncate.test" + crashes server at shutdown when using Maria tables: a temporary table + is correctly created as non-transactional but then, when truncated, is recreated as transactional. */ if (table_type->db_type == DB_TYPE_MARIA) diff --git a/sql/table.cc b/sql/table.cc index 6eb758162df..e8add38640c 100644 --- a/sql/table.cc +++ b/sql/table.cc @@ -2425,9 +2425,10 @@ void update_create_info_from_table(HA_CREATE_INFO *create_info, TABLE *table) create_info->row_type= share->row_type; create_info->default_table_charset= share->table_charset; create_info->table_charset= 0; - /* - See hack in mysql_truncate(); when this is properly fixed, the if() below - can be removed, the assignment can always be made. + /** + @todo MARIA_HACK + See hack in mysql_truncate(); when this is properly fixed, the if() below + can be removed, the assignment can always be made. */ if (create_info->transactional == HA_CHOICE_UNDEF) create_info->transactional= share->transactional; diff --git a/storage/maria/ha_maria.cc b/storage/maria/ha_maria.cc index cfc8e5fc07a..99b92c1bcfc 100644 --- a/storage/maria/ha_maria.cc +++ b/storage/maria/ha_maria.cc @@ -746,7 +746,8 @@ int ha_maria::open(const char *name, int mode, uint test_if_locked) if (!(file= maria_open(name, mode, test_if_locked | HA_OPEN_FROM_SQL_LAYER))) return (my_errno ? my_errno : -1); - /* + /** + @todo ASK_MONTY This is a protection for the case of a frm and MAI containing incompatible table definitions (as in BUG#25908). This was merged from MyISAM. But it breaks maria.test and ps_maria.test ("incorrect key file") if the @@ -2160,7 +2161,8 @@ int ha_maria::create(const char *name, register TABLE *table_arg, create_info.data_file_name= ha_create_info->data_file_name; create_info.index_file_name= ha_create_info->index_file_name; #ifdef ASK_MONTY - /* + /** + @todo ASK_MONTY Where "transactional" in the frm and in the engine can go out of sync. Don't we want to do, after the setting, this test: if (!create_info.transactional && diff --git a/storage/maria/ma_check.c b/storage/maria/ma_check.c index e15c9405e23..a68a21d0180 100644 --- a/storage/maria/ma_check.c +++ b/storage/maria/ma_check.c @@ -2047,9 +2047,10 @@ int maria_repair(HA_CHECK *param, register MARIA_HA *info, } _ma_reset_status(sort_info.new_info); #ifdef ASK_MONTY /* cf maria_create() */ - /* - without this call, a REPAIR on an empty table leaves the data file of - size 0, which sounds reasonable. + /** + @todo ASK_MONTY + without this call, a REPAIR on an empty table leaves the data file of + size 0, which sounds reasonable. */ if (_ma_initialize_data_file(sort_info.new_info->s, new_file)) goto err; @@ -2272,20 +2273,19 @@ err: llstr(sort_param.start_recpos,llbuff)); if (sort_info.new_info && sort_info.new_info != sort_info.info) { -#ifdef ASK_MONTY - /* - grepping for "dfile.file=" - shows several places (ma_check.c, ma_panic.c, ma_extra.c) where we - modify dfile.file without modifying share->bitmap.file.file; those - sound like bugs because the two variables are normally copies of each - other in BLOCK_RECORD (and in other record formats it does not hurt to - change the unused share->bitmap.file.file). - It does matter, because if we close dfile.file, set dfile.file to -1, - but leave bitmap.file.file to its positive value, maria_close() will - close a file which it is not allowed to (maybe even a file in another - engine or mysqld!). + /** + @todo ASK_MONTY + grepping for "dfile.file=" + shows several places (ma_check.c, ma_panic.c, ma_extra.c) where we + modify dfile.file without modifying share->bitmap.file.file; those + sound like bugs because the two variables are normally copies of each + other in BLOCK_RECORD (and in other record formats it does not hurt + to change the unused share->bitmap.file.file). + It does matter, because if we close dfile.file, set dfile.file to -1, + but leave bitmap.file.file to its positive value, maria_close() will + close a file which it is not allowed to (maybe even a file in another + engine or mysqld!). */ -#endif sort_info.new_info->dfile.file= -1; maria_close(sort_info.new_info); } diff --git a/storage/maria/ma_create.c b/storage/maria/ma_create.c index b3aef7c544c..f944b9d8bf7 100644 --- a/storage/maria/ma_create.c +++ b/storage/maria/ma_create.c @@ -1041,7 +1041,8 @@ int maria_create(const char *name, enum data_file_type datafile_type, goto err; errpos=3; - /* + /** + @todo ASK_MONTY QQ: this sets data_file_length from 0 to 8192, but we wrote the state already to the index file (because: - log record is built from index header so state must be written before -- cgit v1.2.1 From dd20a9a7017c3c2167c57b1fa1755e6d31452c35 Mon Sep 17 00:00:00 2001 From: unknown Date: Fri, 10 Aug 2007 15:53:41 +0200 Subject: code of the "mysqltest --global-subst" hack made a bit less horrible. client/mysqltest.c: making code a bit less horrible (no system()). Removing bad free(). --- client/mysqltest.c | 66 +++++++++++++++++++++++++----------------------------- 1 file changed, 30 insertions(+), 36 deletions(-) diff --git a/client/mysqltest.c b/client/mysqltest.c index 76176d8acbb..b08898f4194 100644 --- a/client/mysqltest.c +++ b/client/mysqltest.c @@ -173,6 +173,10 @@ static void init_re(void); static int match_re(my_regex_t *, char *); static void free_re(void); +static int replace(DYNAMIC_STRING *ds_str, + const char *search_str, ulong search_len, + const char *replace_str, ulong replace_len); + DYNAMIC_ARRAY q_lines; #include "sslopt-vars.h" @@ -792,7 +796,6 @@ void free_used_memory() my_free(opt_pass,MYF(MY_ALLOW_ZERO_PTR)); free_defaults(default_argv); free_re(); - my_free(global_subst, MYF(MY_ALLOW_ZERO_PTR)); #ifdef __WIN__ free_tmp_sh_file(); free_win_path_patterns(); @@ -1084,49 +1087,40 @@ err: void check_result(DYNAMIC_STRING* ds) { + int res; DBUG_ENTER("check_result"); DBUG_ASSERT(result_file_name); - switch (dyn_string_cmp(ds, result_file_name)) + res= dyn_string_cmp(ds, result_file_name); + if (global_subst && res != RESULT_OK) + { + /** + @todo MARIA_HACK + This serves for when a test is run with --default-storage-engine=X + where X is not MyISAM: tests using SHOW CREATE TABLE will always fail + because SHOW CREATE TABLE prints X instead of MyISAM. With + --global-subst=X,MyISAM , such trivial differences are eliminated and + test may be reported as passing. + --global-subst is only a quick way to run a lot of existing tests + with Maria and find bugs; it is not good enough for reaching the main + trees when Maria is merged into them. + --global-subst should be removed. + */ + uint global_subst_from_len= strlen(global_subst_from); + uint global_subst_to_len= strlen(global_subst_to); + while (replace(ds, + global_subst_from, global_subst_from_len, + global_subst_to, global_subst_to_len) == 0) + /* do nothing */ ; + /* let's compare again to see if it is ok now */ + res= dyn_string_cmp(ds, result_file_name); + } + switch(res) { case RESULT_OK: break; /* ok */ case RESULT_LENGTH_MISMATCH: dump_result_to_reject_file(ds->str, ds->length); - if (global_subst) - { - /** - @todo MARIA_HACK - This serves for when a test is run with --default-storage-engine=X - where X is not MyISAM: tests using SHOW CREATE TABLE will always fail - because SHOW CREATE TABLE prints X instead of MyISAM. With - --global-subst=X,MyISAM , such trivial differences are eliminated and - test may be reported as passing. - --global-subst is only a quick way to run a lot of existing tests - with Maria and find bugs; it is not good enough for reaching the main - trees when Maria is merged into them. It relies on hard-coded path of - "replace", on existence of "cmp". It's just horrible but it works for - devs using a bk tree in a GNU-based system, which is what we have in - the team. - --global-subst should be removed. - */ - char reject_file[FN_REFLEN]; - char sys_com[50 + FN_REFLEN]; - fn_format(reject_file, result_file_name, "", ".reject", - MY_REPLACE_EXT); - sprintf(sys_com, "../extra/replace %s %s -- %s >/dev/null", - global_subst_from, global_subst_to, reject_file); - if (system(sys_com)) - die("replace failed"); - sprintf(sys_com, "cmp %s %s >/dev/null", - reject_file, result_file_name); - if (!system(sys_com)) - { - /* test is ok in fact */ - my_delete(reject_file, MYF(0)); - break; - } - } die("Result length mismatch"); break; case RESULT_CONTENT_MISMATCH: -- cgit v1.2.1 From 7c32eac40a9ef033f98dc3854c003f99ab540f88 Mon Sep 17 00:00:00 2001 From: unknown Date: Mon, 13 Aug 2007 15:17:49 +0300 Subject: Ability to read unflushed data added (only problem with CRC left and have to be fixed). Already written pages injection to the cache fixed. storage/maria/ma_loghandler.c: Ability to read unflushed data added. storage/maria/ma_page.c: Parameters added storage/maria/ma_pagecache.c: Already written pages injection to the cache fixed. Validator for case of page content injection added. storage/maria/ma_pagecache.h: Validator for case of page content injection added. storage/maria/unittest/Makefile.am: Test of reading unflushed data storage/maria/unittest/ma_test_loghandler-t.c: Define fixed. Restart of the log removed. storage/maria/unittest/ma_test_loghandler_noflush-t.c: New BitKeeper file ``storage/maria/unittest/ma_test_loghandler_noflush-t.c'' --- storage/maria/ma_loghandler.c | 344 ++++++++++++++++----- storage/maria/ma_page.c | 2 +- storage/maria/ma_pagecache.c | 16 +- storage/maria/ma_pagecache.h | 12 +- storage/maria/unittest/Makefile.am | 4 +- storage/maria/unittest/ma_test_loghandler-t.c | 28 +- .../maria/unittest/ma_test_loghandler_noflush-t.c | 132 ++++++++ 7 files changed, 432 insertions(+), 106 deletions(-) create mode 100644 storage/maria/unittest/ma_test_loghandler_noflush-t.c diff --git a/storage/maria/ma_loghandler.c b/storage/maria/ma_loghandler.c index d5c4d59c45f..25163b8cbec 100644 --- a/storage/maria/ma_loghandler.c +++ b/storage/maria/ma_loghandler.c @@ -67,6 +67,11 @@ struct st_translog_buffer LSN last_lsn; /* This buffer offset in the file */ TRANSLOG_ADDRESS offset; + /* + Next buffer offset in the file (it is not always offset + size, + in case of flush by LSN it can be offset + size - TRANSLOG_PAGE_SIZE) + */ + TRANSLOG_ADDRESS next_buffer_offset; /* How much written (or will be written when copy_to_buffer_in_progress become 0) to this buffer @@ -150,7 +155,10 @@ struct st_translog_descriptor /* Last flushed LSN */ LSN flushed; + /* Last LSN sent to the disk (but maybe not written yet) */ LSN sent_to_file; + /* All what is after this addess is not sent to disk yet */ + TRANSLOG_ADDRESS in_buffers_only; pthread_mutex_t sent_to_file_lock; }; @@ -187,6 +195,8 @@ static my_bool write_hook_for_undo(enum translog_record_type type, TRN *trn, LSN *lsn, struct st_translog_parts *parts); +static my_bool translog_page_validator(uchar *page_addr, uchar* data_ptr); + /* Initialize log_record_type_descriptors @@ -672,7 +682,6 @@ my_bool translog_read_file_header(LOGHANDLER_FILE_INFO *desc) static my_bool translog_buffer_init(struct st_translog_buffer *buffer) { DBUG_ENTER("translog_buffer_init"); - /* This buffer offset */ buffer->last_lsn= LSN_IMPOSSIBLE; /* This Buffer File */ buffer->file= -1; @@ -966,7 +975,8 @@ static void translog_put_sector_protection(uchar *page, static uint32 translog_crc(uchar *area, uint length) { - return crc32(0L, (unsigned char*) area, length); + DBUG_ENTER("translog_crc"); + DBUG_RETURN(crc32(0L, (unsigned char*) area, length)); } @@ -1180,6 +1190,7 @@ static void translog_start_buffer(struct st_translog_buffer *buffer, DBUG_ASSERT(buffer_no == buffer->buffer_no); buffer->last_lsn= LSN_IMPOSSIBLE; buffer->offset= log_descriptor.horizon; + buffer->next_buffer_offset= LSN_IMPOSSIBLE; buffer->file= log_descriptor.log_file_num[0]; buffer->overlay= 0; buffer->size= 0; @@ -1254,45 +1265,117 @@ static my_bool translog_buffer_next(TRANSLOG_ADDRESS *horizon, translog_cursor_init(cursor, new_buffer, new_buffer_no); else translog_start_buffer(new_buffer, cursor, new_buffer_no); + log_descriptor.buffers[old_buffer_no].next_buffer_offset= new_buffer->offset; translog_new_page_header(horizon, cursor); DBUG_RETURN(0); } /* - Set max LSN sent to file + Sets max LSN sent to file, and address from which data is only in the buffer SYNOPSIS translog_set_sent_to_file() lsn LSN to assign + in_buffers to assign to in_buffers_only + + TODO: use atomic operations if possible (64bit architectures?) */ -static void translog_set_sent_to_file(LSN *lsn) +static void translog_set_sent_to_file(LSN lsn, TRANSLOG_ADDRESS in_buffers) { DBUG_ENTER("translog_set_sent_to_file"); pthread_mutex_lock(&log_descriptor.sent_to_file_lock); - DBUG_ASSERT(cmp_translog_addr(*lsn, log_descriptor.sent_to_file) >= 0); - log_descriptor.sent_to_file= *lsn; + DBUG_PRINT("enter", ("lsn: (%lu,0x%lx) in_buffers: (%lu,0x%lx) " + "in_buffers_only: (%lu,0x%lx)", + (ulong) LSN_FILE_NO(lsn), + (ulong) LSN_OFFSET(lsn), + (ulong) LSN_FILE_NO(in_buffers), + (ulong) LSN_OFFSET(in_buffers), + (ulong) LSN_FILE_NO(log_descriptor.in_buffers_only), + (ulong) LSN_OFFSET(log_descriptor.in_buffers_only))); + DBUG_ASSERT(cmp_translog_addr(lsn, log_descriptor.sent_to_file) >= 0); + log_descriptor.sent_to_file= lsn; + /* LSN_IMPOSSIBLE == 0 => it will work for very first time */ + if (cmp_translog_addr(in_buffers, log_descriptor.in_buffers_only) > 0) + { + log_descriptor.in_buffers_only= in_buffers; + DBUG_PRINT("info", ("set new in_buffers_only")); + } pthread_mutex_unlock(&log_descriptor.sent_to_file_lock); DBUG_VOID_RETURN; } /* - Get max LSN send to file + Sets address from which data is only in the buffer + + SYNOPSIS + translog_set_only_in_buffers() + lsn LSN to assign + in_buffers to assign to in_buffers_only +*/ + +static void translog_set_only_in_buffers(TRANSLOG_ADDRESS in_buffers) +{ + DBUG_ENTER("translog_set_only_in_buffers"); + pthread_mutex_lock(&log_descriptor.sent_to_file_lock); + DBUG_PRINT("enter", ("in_buffers: (%lu,0x%lx) " + "in_buffers_only: (%lu,0x%lx)", + (ulong) LSN_FILE_NO(in_buffers), + (ulong) LSN_OFFSET(in_buffers), + (ulong) LSN_FILE_NO(log_descriptor.in_buffers_only), + (ulong) LSN_OFFSET(log_descriptor.in_buffers_only))); + /* LSN_IMPOSSIBLE == 0 => it will work for very first time */ + if (cmp_translog_addr(in_buffers, log_descriptor.in_buffers_only) > 0) + { + log_descriptor.in_buffers_only= in_buffers; + DBUG_PRINT("info", ("set new in_buffers_only")); + } + pthread_mutex_unlock(&log_descriptor.sent_to_file_lock); + DBUG_VOID_RETURN; +} + + +/* + Gets address from which data is only in the buffer + + SYNOPSIS + translog_only_in_buffers() + + RETURN + address from which data is only in the buffer +*/ + +static TRANSLOG_ADDRESS translog_only_in_buffers() +{ + register TRANSLOG_ADDRESS addr; + DBUG_ENTER("translog_only_in_buffers"); + pthread_mutex_lock(&log_descriptor.sent_to_file_lock); + addr= log_descriptor.in_buffers_only; + pthread_mutex_unlock(&log_descriptor.sent_to_file_lock); + DBUG_RETURN(addr); +} + + +/* + Get max LSN sent to file SYNOPSIS translog_get_sent_to_file() - lsn LSN to value + + RETURN + max LSN send to file */ -static void translog_get_sent_to_file(LSN *lsn) +static LSN translog_get_sent_to_file() { + register LSN lsn; DBUG_ENTER("translog_get_sent_to_file"); pthread_mutex_lock(&log_descriptor.sent_to_file_lock); - *lsn= log_descriptor.sent_to_file; + lsn= log_descriptor.sent_to_file; pthread_mutex_unlock(&log_descriptor.sent_to_file_lock); - DBUG_VOID_RETURN; + DBUG_RETURN(lsn); } @@ -1532,16 +1615,20 @@ static my_bool translog_buffer_flush(struct st_translog_buffer *buffer) file.file= buffer->file; for (i= 0; i < buffer->size; i+= TRANSLOG_PAGE_SIZE) { + TRANSLOG_ADDRESS addr= (buffer->offset + i); + TRANSLOG_VALIDATOR_DATA data; + data.addr= &addr; DBUG_ASSERT(log_descriptor.pagecache->block_size == TRANSLOG_PAGE_SIZE); DBUG_ASSERT(i + TRANSLOG_PAGE_SIZE <= buffer->size); - if (pagecache_write(log_descriptor.pagecache, + if (pagecache_inject(log_descriptor.pagecache, &file, (LSN_OFFSET(buffer->offset) + i) / TRANSLOG_PAGE_SIZE, 3, buffer->buffer + i, PAGECACHE_PLAIN_PAGE, PAGECACHE_LOCK_LEFT_UNLOCKED, - PAGECACHE_PIN_LEFT_UNPINNED, PAGECACHE_WRITE_DONE, 0)) + PAGECACHE_PIN_LEFT_UNPINNED, 0, + &translog_page_validator, (uchar*) &data)) { UNRECOVERABLE_ERROR(("Can't write page (%lu,0x%lx) to pagecache", (ulong) buffer->file, @@ -1559,9 +1646,12 @@ static my_bool translog_buffer_flush(struct st_translog_buffer *buffer) (ulong) buffer->size, errno)); DBUG_RETURN(1); } - if (LSN_OFFSET(buffer->last_lsn) != 0) /* if buffer->last_lsn is set */ - translog_set_sent_to_file(&buffer->last_lsn); + if (LSN_OFFSET(buffer->last_lsn) != 0) /* if buffer->last_lsn is set */ + translog_set_sent_to_file(buffer->last_lsn, + buffer->next_buffer_offset); + else + translog_set_only_in_buffers(buffer->next_buffer_offset); /* Free buffer */ buffer->file= -1; buffer->overlay= 0; @@ -1747,6 +1837,59 @@ static my_bool translog_page_validator(uchar *page_addr, uchar* data_ptr) } +/* + Lock the loghandler + + SYNOPSIS + translog_lock() + + RETURN + 0 OK + 1 Error +*/ + +my_bool translog_lock() +{ + struct st_translog_buffer *current_buffer; + DBUG_ENTER("translog_lock"); + + /* + Locking the loghandler mean locking current buffer, but it can change + during locking, so we should check it + */ + for (;;) + { + current_buffer= log_descriptor.bc.buffer; + if (translog_buffer_lock(current_buffer)) + DBUG_RETURN(1); + if (log_descriptor.bc.buffer == current_buffer) + break; + translog_buffer_unlock(current_buffer); + } + DBUG_RETURN(0); +} + + +/* + Unlock the loghandler + + SYNOPSIS + translog_unlock() + + RETURN + 0 OK + 1 Error +*/ + +my_bool translog_unlock() +{ + DBUG_ENTER("translog_unlock"); + translog_buffer_unlock(log_descriptor.bc.buffer); + + DBUG_RETURN(0); +} + + /* Get log page by file number and offset of the beginning of the page @@ -1763,7 +1906,7 @@ static my_bool translog_page_validator(uchar *page_addr, uchar* data_ptr) static uchar *translog_get_page(TRANSLOG_VALIDATOR_DATA *data, uchar *buffer) { - TRANSLOG_ADDRESS addr= *(data->addr); + TRANSLOG_ADDRESS addr= *(data->addr), in_buffers; uint cache_index; uint32 file_no= LSN_FILE_NO(addr); DBUG_ENTER("translog_get_page"); @@ -1775,6 +1918,107 @@ static uchar *translog_get_page(TRANSLOG_VALIDATOR_DATA *data, uchar *buffer) /* it is really page address */ DBUG_ASSERT(LSN_OFFSET(addr) % TRANSLOG_PAGE_SIZE == 0); + in_buffers= translog_only_in_buffers(); + DBUG_PRINT("info", ("in_buffers: (%lu,0x%lx)", + (ulong) LSN_FILE_NO(in_buffers), + (ulong) LSN_OFFSET(in_buffers))); + if (in_buffers != LSN_IMPOSSIBLE && + cmp_translog_addr(addr, in_buffers) >= 0) + { + translog_lock(); + /* recheck with locked loghandler */ + in_buffers= translog_only_in_buffers(); + if (cmp_translog_addr(addr, in_buffers) >= 0) + { + uint16 buffer_no= log_descriptor.bc.buffer_no; + uint16 buffer_start= buffer_no; + struct st_translog_buffer *buffer_unlock= log_descriptor.bc.buffer; + struct st_translog_buffer *curr_buffer= log_descriptor.bc.buffer; + for (;;) + { + /* + if the page is in the buffer and it is the last version of the + page (in case of devision the page bu buffer flush + */ + if (curr_buffer->file != -1 && + cmp_translog_addr(addr, curr_buffer->offset) >= 0 && + cmp_translog_addr(addr, + (curr_buffer->next_buffer_offset ? + curr_buffer->next_buffer_offset: + curr_buffer->offset + curr_buffer->size)) < 0) + { + int is_last_unfinished_page; + uint last_protected_sector= 0; + uchar *from, *table; + translog_wait_for_writers(curr_buffer); + DBUG_ASSERT(LSN_FILE_NO(addr) == LSN_FILE_NO(curr_buffer->offset)); + from= curr_buffer->buffer + (addr - curr_buffer->offset); + memcpy(buffer, from, TRANSLOG_PAGE_SIZE); + is_last_unfinished_page= ((log_descriptor.bc.buffer == + curr_buffer) && + (log_descriptor.bc.ptr >= from) && + (log_descriptor.bc.ptr < + from + TRANSLOG_PAGE_SIZE)); + if (is_last_unfinished_page && + (buffer[TRANSLOG_PAGE_FLAGS] & TRANSLOG_SECTOR_PROTECTION)) + { + last_protected_sector= ((log_descriptor.bc.previous_offset - 1) / + DISK_DRIVE_SECTOR_SIZE); + table= buffer + log_descriptor.page_overhead - + (TRANSLOG_PAGE_SIZE / DISK_DRIVE_SECTOR_SIZE) * 2; + } + + DBUG_ASSERT(buffer_unlock == curr_buffer); + translog_buffer_unlock(buffer_unlock); + if (is_last_unfinished_page) + { + uint i; + /* + This is last unfinished page => we should not check CRC and + remove only that protection which already installed (no need + to check it) + + We do not check the flag of sector protection, because if + (buffer[TRANSLOG_PAGE_FLAGS] & TRANSLOG_SECTOR_PROTECTION) is + not set then last_protected_sector will be 0 so following loop + will be never executed + */ + DBUG_PRINT("info", ("This is last unfinished page, " + "last protected sector %u", + last_protected_sector)); + for (i= 1; i <= last_protected_sector; i++) + { + uint index= i * 2; + uint offset= i * DISK_DRIVE_SECTOR_SIZE; + DBUG_PRINT("info", ("Sector %u: 0x%02x%02x <- 0x%02x%02x", + i, buffer[offset], buffer[offset + 1], + table[index], table[index + 1])); + buffer[offset]= table[index]; + buffer[offset + 1]= table[index + 1]; + } + } + else + { + /* + This IF should be true because we use in-memory data which + supposed to be correct. + */ + if (translog_page_validator((uchar*) buffer, (uchar*) data)) + buffer= NULL; + } + DBUG_RETURN(buffer); + } + buffer_no= (buffer_no + 1) % TRANSLOG_BUFFERS_NO; + curr_buffer= log_descriptor.buffers + buffer_no; + translog_buffer_lock(curr_buffer); + translog_buffer_unlock(buffer_unlock); + buffer_unlock= curr_buffer; + /* we can't make full circle */ + DBUG_ASSERT(buffer_start != buffer_no); + } + } + translog_unlock(); + } if ((cache_index= LSN_FILE_NO(log_descriptor.horizon) - file_no) < OPENED_FILES_NUM) { @@ -2000,6 +2244,7 @@ my_bool translog_init(const char *directory, DBUG_RETURN(1); } + log_descriptor.in_buffers_only= LSN_IMPOSSIBLE; /* max size of one log size (for new logs creation) */ log_descriptor.log_file_max_size= log_file_max_size - (log_file_max_size % TRANSLOG_PAGE_SIZE); @@ -2269,7 +2514,9 @@ my_bool translog_init(const char *directory, } /* all LSNs that are on disk are flushed */ - log_descriptor.sent_to_file= log_descriptor.flushed= log_descriptor.horizon; + log_descriptor.sent_to_file= + log_descriptor.flushed= log_descriptor.horizon; + log_descriptor.in_buffers_only= log_descriptor.bc.buffer->offset; /* horizon is (potentially) address of the next LSN we need decrease it to signal that all LSNs before it are flushed @@ -2366,57 +2613,6 @@ void translog_destroy() } -/* - Lock the loghandler - - SYNOPSIS - translog_lock() - - RETURN - 0 OK - 1 Error -*/ - -my_bool translog_lock() -{ - struct st_translog_buffer *current_buffer; - DBUG_ENTER("translog_lock"); - - /* - Locking the loghandler mean locking current buffer, but it can change - during locking, so we should check it - */ - for (;;) - { - current_buffer= log_descriptor.bc.buffer; - if (translog_buffer_lock(current_buffer)) - DBUG_RETURN(1); - if (log_descriptor.bc.buffer == current_buffer) - break; - translog_buffer_unlock(current_buffer); - } - DBUG_RETURN(0); -} - - -/* - Unlock the loghandler - - SYNOPSIS - translog_unlock() - - RETURN - 0 OK - 1 Error -*/ - -my_bool translog_unlock() -{ - DBUG_ENTER("translog_unlock"); - translog_buffer_unlock(log_descriptor.bc.buffer); - - DBUG_RETURN(0); -} #define translog_buffer_lock_assert_owner(B) \ @@ -2923,6 +3119,7 @@ static my_bool translog_advance_pointer(uint pages, uint16 last_page_data) log_descriptor.horizon+= min_offset; /* offset increasing */ } translog_start_buffer(new_buffer, &log_descriptor.bc, new_buffer_no); + old_buffer->next_buffer_offset= new_buffer->offset; if (translog_buffer_unlock(old_buffer)) DBUG_RETURN(1); offset-= min_offset; @@ -3523,7 +3720,7 @@ static my_bool translog_relative_LSN_encode(struct st_translog_parts *parts, uchar *dst_ptr= compressed_LSNs + (MAX_NUMBER_OF_LSNS_PER_RECORD * COMPRESSED_LSN_MAX_STORE_SIZE); for (src_ptr= buffer + lsns_len - LSN_STORE_SIZE; - src_ptr >= buffer; + src_ptr >= (uchar*) buffer; src_ptr-= LSN_STORE_SIZE) { ref= lsn_korr(src_ptr); @@ -3536,7 +3733,7 @@ static my_bool translog_relative_LSN_encode(struct st_translog_parts *parts, dst_ptr); parts->record_length-= (economy= lsns_len - part->length); DBUG_PRINT("info", ("new length of LSNs: %u economy: %d", - part->length, economy)); + (uint) part->length, economy)); parts->total_record_length-= economy; part->str= (char*)dst_ptr; } @@ -5363,7 +5560,7 @@ static void translog_force_current_buffer_to_finish() struct st_translog_buffer *new_buffer= (log_descriptor.buffers + new_buffer_no); struct st_translog_buffer *old_buffer= log_descriptor.bc.buffer; - uchar *data= log_descriptor.bc.ptr -log_descriptor.bc.current_page_fill; + uchar *data= log_descriptor.bc.ptr - log_descriptor.bc.current_page_fill; uint16 left= TRANSLOG_PAGE_SIZE - log_descriptor.bc.current_page_fill; uint16 current_page_fill, write_counter, previous_offset; DBUG_ENTER("translog_force_current_buffer_to_finish"); @@ -5458,13 +5655,14 @@ static void translog_force_current_buffer_to_finish() if (left) { memcpy(new_buffer->buffer, data, current_page_fill); - log_descriptor.bc.ptr +=current_page_fill; + log_descriptor.bc.ptr+= current_page_fill; log_descriptor.bc.buffer->size= log_descriptor.bc.current_page_fill= current_page_fill; new_buffer->overlay= old_buffer; } else translog_new_page_header(&log_descriptor.horizon, &log_descriptor.bc); + old_buffer->next_buffer_offset= new_buffer->offset; DBUG_VOID_RETURN; } @@ -5527,7 +5725,7 @@ my_bool translog_flush(LSN lsn) DBUG_RETURN(0); } /* send to the file if it is not sent */ - translog_get_sent_to_file(&sent_to_file); + sent_to_file= translog_get_sent_to_file(); if (cmp_translog_addr(sent_to_file, lsn) >= 0) break; diff --git a/storage/maria/ma_page.c b/storage/maria/ma_page.c index 9e57898f4a1..f749414474f 100644 --- a/storage/maria/ma_page.c +++ b/storage/maria/ma_page.c @@ -141,7 +141,7 @@ int _ma_dispose(register MARIA_HA *info, MARIA_KEYDEF *keyinfo, my_off_t pos, PAGECACHE_LOCK_LEFT_UNLOCKED, PAGECACHE_PIN_LEFT_UNPINNED, PAGECACHE_WRITE_DELAY, 0, - offset, sizeof(buff))); + offset, sizeof(buff), 0, 0)); } /* _ma_dispose */ diff --git a/storage/maria/ma_pagecache.c b/storage/maria/ma_pagecache.c index eb939ba9eb0..4918b2fb762 100755 --- a/storage/maria/ma_pagecache.c +++ b/storage/maria/ma_pagecache.c @@ -3167,7 +3167,9 @@ my_bool pagecache_write_part(PAGECACHE *pagecache, enum pagecache_page_pin pin, enum pagecache_write_mode write_mode, PAGECACHE_PAGE_LINK *link, - uint offset, uint size) + uint offset, uint size, + pagecache_disk_read_validator validator, + uchar* validator_data) { PAGECACHE_BLOCK_LINK *block= NULL; PAGECACHE_PAGE_LINK fake_link; @@ -3253,7 +3255,7 @@ restart: if (write_mode == PAGECACHE_WRITE_DONE) { - if ((block->status & PCBLOCK_ERROR) && page_st != PAGE_READ) + if (!(block->status & PCBLOCK_ERROR)) { /* Copy data from buff */ if (!(size & 511)) @@ -3261,8 +3263,15 @@ restart: else memcpy(block->buffer + offset, buff, size); block->status= (PCBLOCK_READ | (block->status & PCBLOCK_WRLOCK)); + /* + The validator can change the page content (removing page + protection) so it have to be called + */ + if (validator != NULL && + (*validator)(block->buffer, validator_data)) + block->status|= PCBLOCK_ERROR; KEYCACHE_DBUG_PRINT("key_cache_insert", - ("primary request: new page in cache")); + ("Page injection")); #ifdef THREAD /* Signal that all pending requests for this now can be processed. */ if (block->wqueue[COND_FOR_REQUESTED].last_thread) @@ -3272,6 +3281,7 @@ restart: } else { + DBUG_ASSERT(validator == 0 && validator_data == 0); if (! (block->status & PCBLOCK_CHANGED)) link_to_changed_list(pagecache, block); diff --git a/storage/maria/ma_pagecache.h b/storage/maria/ma_pagecache.h index 86426c5b4bc..468da02425e 100644 --- a/storage/maria/ma_pagecache.h +++ b/storage/maria/ma_pagecache.h @@ -95,7 +95,7 @@ typedef struct st_pagecache_hash_link PAGECACHE_HASH_LINK; #include -typedef my_bool (*pagecache_disk_read_validator)(uchar *page, uchar** data); +typedef my_bool (*pagecache_disk_read_validator)(uchar *page, uchar* data); #define PAGECACHE_CHANGED_BLOCKS_HASH 128 /* must be power of 2 */ @@ -190,7 +190,11 @@ extern uchar *pagecache_valid_read(PAGECACHE *pagecache, uchar* validator_data); #define pagecache_write(P,F,N,L,B,T,O,I,M,K) \ - pagecache_write_part(P,F,N,L,B,T,O,I,M,K,0,(P)->block_size) + pagecache_write_part(P,F,N,L,B,T,O,I,M,K,0,(P)->block_size,0,0) + +#define pagecache_inject(P,F,N,L,B,T,O,I,K,V,D) \ + pagecache_write_part(P,F,N,L,B,T,O,I,PAGECACHE_WRITE_DONE, \ + K,0,(P)->block_size,V,D) extern my_bool pagecache_write_part(PAGECACHE *pagecache, PAGECACHE_FILE *file, @@ -203,7 +207,9 @@ extern my_bool pagecache_write_part(PAGECACHE *pagecache, enum pagecache_write_mode write_mode, PAGECACHE_PAGE_LINK *link, uint offset, - uint size); + uint size, + pagecache_disk_read_validator validator, + uchar* validator_data); extern void pagecache_unlock(PAGECACHE *pagecache, PAGECACHE_FILE *file, pgcache_page_no_t pageno, diff --git a/storage/maria/unittest/Makefile.am b/storage/maria/unittest/Makefile.am index b63cb60c059..2110a7a5d04 100644 --- a/storage/maria/unittest/Makefile.am +++ b/storage/maria/unittest/Makefile.am @@ -42,7 +42,8 @@ noinst_PROGRAMS = ma_control_file-t trnman-t lockman2-t \ ma_test_loghandler_multigroup-t \ ma_test_loghandler_multithread-t \ ma_test_loghandler_pagecache-t \ - ma_test_loghandler_long-t-big + ma_test_loghandler_long-t-big \ + ma_test_loghandler_noflush-t ma_test_loghandler_t_SOURCES = ma_test_loghandler-t.c ma_maria_log_cleanup.c ma_test_loghandler_multigroup_t_SOURCES = ma_test_loghandler_multigroup-t.c ma_maria_log_cleanup.c @@ -50,6 +51,7 @@ ma_test_loghandler_multithread_t_SOURCES = ma_test_loghandler_multithread-t.c ma ma_test_loghandler_pagecache_t_SOURCES = ma_test_loghandler_pagecache-t.c ma_maria_log_cleanup.c ma_test_loghandler_long_t_big_SOURCES = ma_test_loghandler-t.c ma_maria_log_cleanup.c ma_test_loghandler_long_t_big_CPPFLAGS = -DLONG_LOG_TEST +ma_test_loghandler_noflush_t_SOURCES = ma_test_loghandler_noflush-t.c ma_maria_log_cleanup.c ma_pagecache_single_src = ma_pagecache_single.c test_file.c ma_pagecache_consist_src = ma_pagecache_consist.c test_file.c diff --git a/storage/maria/unittest/ma_test_loghandler-t.c b/storage/maria/unittest/ma_test_loghandler-t.c index a6bd53e949d..9b5055bac4e 100644 --- a/storage/maria/unittest/ma_test_loghandler-t.c +++ b/storage/maria/unittest/ma_test_loghandler-t.c @@ -19,8 +19,9 @@ static TRN *trn= &dummy_transaction_object; #define LOG_FLAGS 0 #define LOG_FILE_SIZE (1024L*1024L) #define ITERATIONS (1600*4) + #else -#define LOG_FLAGS TRANSLOG_SECTOR_PROTECTION | TRANSLOG_PAGE_CRC +#define LOG_FLAGS (TRANSLOG_SECTOR_PROTECTION | TRANSLOG_PAGE_CRC) #define LOG_FILE_SIZE (1024L*1024L*3L) #define ITERATIONS 1600 #endif @@ -331,32 +332,8 @@ int main(int argc __attribute__((unused)), char *argv[]) ok(1, "flush"); } - translog_destroy(); - end_pagecache(&pagecache, 1); - ma_control_file_end(); - - - if (ma_control_file_create_or_open(TRUE)) - { - fprintf(stderr, "pass2: Can't init control file (%d)\n", errno); - exit(1); - } - if ((pagen= init_pagecache(&pagecache, PCACHE_SIZE, 0, 0, - TRANSLOG_PAGE_SIZE)) == 0) - { - fprintf(stderr, "pass2: Got error: init_pagecache() (errno: %d)\n", errno); - exit(1); - } - if (translog_init(".", LOG_FILE_SIZE, 50112, 0, &pagecache, LOG_FLAGS)) - { - fprintf(stderr, "pass2: Can't init loghandler (%d)\n", errno); - translog_destroy(); - exit(1); - } - example_loghandler_init(); srandom(122334817L); - rc= 1; { @@ -639,5 +616,6 @@ err: if (maria_log_remove()) exit(1); + return(test(exit_status())); } diff --git a/storage/maria/unittest/ma_test_loghandler_noflush-t.c b/storage/maria/unittest/ma_test_loghandler_noflush-t.c new file mode 100644 index 00000000000..fed66da72a4 --- /dev/null +++ b/storage/maria/unittest/ma_test_loghandler_noflush-t.c @@ -0,0 +1,132 @@ +#include "../maria_def.h" +#include +#include +#include +#include "../trnman.h" + +extern my_bool maria_log_remove(); + +#ifndef DBUG_OFF +static const char *default_dbug_option; +#endif + +#define PCACHE_SIZE (1024*1024*10) +#define PCACHE_PAGE TRANSLOG_PAGE_SIZE +#define LOG_FILE_SIZE (1024L*1024L*1024L + 1024L*1024L*512) +#define LOG_FLAGS 0 + +static char *first_translog_file= (char*)"maria_log.00000001"; + +int main(int argc __attribute__((unused)), char *argv[]) +{ + uint pagen; + int rc= 1; + uchar long_tr_id[6]; + PAGECACHE pagecache; + LSN first_lsn; + MY_STAT st; + TRANSLOG_HEADER_BUFFER rec; + LEX_STRING parts[TRANSLOG_INTERNAL_PARTS + 1]; + + MY_INIT(argv[0]); + + plan(1); + + bzero(&pagecache, sizeof(pagecache)); + maria_data_root= "."; + if (maria_log_remove()) + exit(1); + /* be sure that we have no logs in the directory*/ + if (my_stat(CONTROL_FILE_BASE_NAME, &st, MYF(0))) + my_delete(CONTROL_FILE_BASE_NAME, MYF(0)); + if (my_stat(first_translog_file, &st, MYF(0))) + my_delete(first_translog_file, MYF(0)); + + bzero(long_tr_id, 6); +#ifndef DBUG_OFF +#if defined(__WIN__) + default_dbug_option= "d:t:i:O,\\ma_test_loghandler.trace"; +#else + default_dbug_option= "d:t:i:o,/tmp/ma_test_loghandler.trace"; +#endif + if (argc > 1) + { + DBUG_SET(default_dbug_option); + DBUG_SET_INITIAL(default_dbug_option); + } +#endif + + if (ma_control_file_create_or_open(TRUE)) + { + fprintf(stderr, "Can't init control file (%d)\n", errno); + exit(1); + } + if ((pagen= init_pagecache(&pagecache, PCACHE_SIZE, 0, 0, + PCACHE_PAGE)) == 0) + { + fprintf(stderr, "Got error: init_pagecache() (errno: %d)\n", errno); + exit(1); + } + if (translog_init(".", LOG_FILE_SIZE, 50112, 0, &pagecache, LOG_FLAGS)) + { + fprintf(stderr, "Can't init loghandler (%d)\n", errno); + translog_destroy(); + exit(1); + } + example_loghandler_init(); + + + int4store(long_tr_id, 0); + long_tr_id[5]= 0xff; + parts[TRANSLOG_INTERNAL_PARTS + 0].str= (char*)long_tr_id; + parts[TRANSLOG_INTERNAL_PARTS + 0].length= 6; + if (translog_write_record(&first_lsn, + LOGREC_FIXED_RECORD_0LSN_EXAMPLE, + &dummy_transaction_object, NULL, 6, + TRANSLOG_INTERNAL_PARTS + 1, + parts, NULL)) + { + fprintf(stderr, "Can't write record #%lu\n", (ulong) 0); + translog_destroy(); + exit(1); + } + + translog_size_t len= translog_read_record_header(first_lsn, &rec); + if (len == 0) + { + fprintf(stderr, "translog_read_record_header failed (%d)\n", errno); + goto err; + } + if (rec.type !=LOGREC_FIXED_RECORD_0LSN_EXAMPLE || rec.short_trid != 0 || + rec.record_length != 6 || uint4korr(rec.header) != 0 || + ((uchar)rec.header[4]) != 0 || ((uchar)rec.header[5]) != 0xFF || + first_lsn != rec.lsn) + { + fprintf(stderr, "Incorrect LOGREC_FIXED_RECORD_0LSN_EXAMPLE " + "data read(0)\n" + "type: %u (%d) strid: %u (%d) len: %u (%d) i: %u (%d), " + "4: %u (%d) 5: %u (%d) " + "lsn(%lu,0x%lx) (%d)\n", + (uint) rec.type, (rec.type !=LOGREC_FIXED_RECORD_0LSN_EXAMPLE), + (uint) rec.short_trid, (rec.short_trid != 0), + (uint) rec.record_length, (rec.record_length != 6), + (uint) uint4korr(rec.header), (uint4korr(rec.header) != 0), + (uint) rec.header[4], (((uchar)rec.header[4]) != 0), + (uint) rec.header[5], (((uchar)rec.header[5]) != 0xFF), + (ulong) LSN_FILE_NO(rec.lsn), (ulong) LSN_OFFSET(rec.lsn), + (first_lsn != rec.lsn)); + goto err; + } + + ok(1, "read OK"); + rc= 0; + +err: + translog_destroy(); + end_pagecache(&pagecache, 1); + ma_control_file_end(); + my_delete(CONTROL_FILE_BASE_NAME, MYF(0)); + my_delete(first_translog_file, MYF(0)); + + exit(rc); +} -- cgit v1.2.1 From 4cf6756eb0f29618646e2eadbd62804d81ff6a79 Mon Sep 17 00:00:00 2001 From: unknown Date: Mon, 13 Aug 2007 22:54:29 +0300 Subject: First LSN calls added for transaction log. storage/maria/ma_checkpoint.c: Definitions of LSN should be collected in the one file (ma_loghandler_lsn.h) storage/maria/ma_loghandler.c: New calls to get first theoretical and first real LSN. storage/maria/ma_loghandler.h: New calls to get first theoretical and first real LSN. storage/maria/ma_loghandler_lsn.h: Defined yet another impossible LSN to indicate error. storage/maria/ma_recovery.c: The first LSN call changed. storage/maria/maria_read_log.c: The first LSN call changed. storage/maria/unittest/Makefile.am: New unittest added. storage/maria/unittest/ma_test_loghandler_first_lsn-t.c: New BitKeeper file ``storage/maria/unittest/ma_test_loghandler_first_lsn-t.c'' --- storage/maria/ma_checkpoint.c | 2 +- storage/maria/ma_loghandler.c | 120 ++++++++++++++++- storage/maria/ma_loghandler.h | 3 +- storage/maria/ma_loghandler_lsn.h | 2 + storage/maria/ma_recovery.c | 14 +- storage/maria/maria_read_log.c | 7 +- storage/maria/unittest/Makefile.am | 4 +- .../unittest/ma_test_loghandler_first_lsn-t.c | 150 +++++++++++++++++++++ 8 files changed, 294 insertions(+), 8 deletions(-) create mode 100644 storage/maria/unittest/ma_test_loghandler_first_lsn-t.c diff --git a/storage/maria/ma_checkpoint.c b/storage/maria/ma_checkpoint.c index b23d077897f..42b6c961b41 100644 --- a/storage/maria/ma_checkpoint.c +++ b/storage/maria/ma_checkpoint.c @@ -38,8 +38,8 @@ #include "transaction.h" #include "share.h" #include "log.h" +#include "ma_loghandler_lsn.h" -#define LSN_IMPOSSIBLE ((LSN)0) /* could also be called LSN_ERROR */ #define LSN_MAX ((LSN)ULONGLONG_MAX) /* diff --git a/storage/maria/ma_loghandler.c b/storage/maria/ma_loghandler.c index 22c49d61c91..1bd6d803064 100644 --- a/storage/maria/ma_loghandler.c +++ b/storage/maria/ma_loghandler.c @@ -5976,13 +5976,127 @@ void translog_deassign_id_from_share(MARIA_SHARE *share) } +/** + @brief check if such log file exists + + @param file_no number of the file to test + + @retval 0 no such file + @retval 1 there is file with such number +*/ + +my_bool translog_is_file(uint file_no) +{ + MY_STAT stat_buff; + char path[FN_REFLEN]; + return (test(my_stat(translog_filename_by_fileno(file_no, path), + &stat_buff, MYF(MY_WME)))); +} + + /** @brief returns the LSN of the first record starting in this log - @note so far works only for the very first log created on this system + @retval LSN_ERROR Error + @retval LSN_IMPOSSIBLE no log + @retval # LSN of the first record */ -LSN first_lsn_in_log() +LSN translog_first_lsn_in_log() { - return MAKE_LSN(1, TRANSLOG_PAGE_SIZE + log_descriptor.page_overhead); + TRANSLOG_ADDRESS addr, horizon= translog_get_horizon(); + TRANSLOG_VALIDATOR_DATA data; + uint min_file= 1, max_file= LSN_FILE_NO(horizon); + uint chunk_type; + uint16 chunk_offset; + uchar *page; + TRANSLOG_SCANNER_DATA scanner; + DBUG_ENTER("translog_first_lsn_in_log"); + DBUG_PRINT("info", ("Horizon: (%lu,0x%lx)", + LSN_FILE_NO(addr), LSN_OFFSET(addr))); + + if (addr == MAKE_LSN(1, TRANSLOG_PAGE_SIZE)) + { + /* there is no first page yet */ + DBUG_RETURN(LSN_IMPOSSIBLE); + } + + + /*TODO: lock loghandler purger when it will be created */ + /* binary search for last file */ + while (min_file != max_file && min_file != (max_file - 1)) + { + uint test= (min_file + max_file) / 2; + DBUG_PRINT("info", ("min_file: %u test: %u max_file: %u", + min_file, test, max_file)); + if (test == max_file) + test--; + if (translog_is_file(test)) + max_file= test; + else + min_file= test; + } + + addr= MAKE_LSN(max_file, TRANSLOG_PAGE_SIZE); /* the first page of the file */ + data.addr= &addr; + if ((page= translog_get_page(&data, scanner.buffer)) == NULL || + (chunk_offset= translog_get_first_chunk_offset(page)) == 0) + DBUG_RETURN(LSN_ERROR); + addr+= chunk_offset; + if (addr == horizon) + DBUG_RETURN(LSN_IMPOSSIBLE); + translog_init_scanner(addr, 0, &scanner); + + chunk_type= scanner.page[scanner.page_offset] & TRANSLOG_CHUNK_TYPE; + DBUG_PRINT("info", ("type: %x byte: %x", (uint) chunk_type, + (uint) scanner.page[scanner.page_offset])); + while (chunk_type != TRANSLOG_CHUNK_LSN && + chunk_type != TRANSLOG_CHUNK_FIXED && + scanner.page[scanner.page_offset] != 0) + { + if (translog_get_next_chunk(&scanner)) + DBUG_RETURN(LSN_ERROR); + chunk_type= scanner.page[scanner.page_offset] & TRANSLOG_CHUNK_TYPE; + DBUG_PRINT("info", ("type: %x byte: %x", (uint) chunk_type, + (uint) scanner.page[scanner.page_offset])); + } + if (scanner.page[scanner.page_offset] == 0) + DBUG_RETURN(LSN_IMPOSSIBLE); /* reached page filler */ + DBUG_RETURN(scanner.page_addr + scanner.page_offset); +} + + +/** + @brief returns theoretical first LSN if first log is present + + @retval LSN_ERROR Error + @retval LSN_IMPOSSIBLE no log + @retval # LSN of the first record +*/ + +LSN translog_first_theoretical_lsn() +{ + TRANSLOG_ADDRESS addr= translog_get_horizon(); + uchar buffer[TRANSLOG_PAGE_SIZE], *page; + TRANSLOG_VALIDATOR_DATA data; + DBUG_ENTER("translog_first_theoretical_lsn"); + DBUG_PRINT("info", ("Horizon: (%lu,0x%lx)", + LSN_FILE_NO(addr), LSN_OFFSET(addr))); + + if (!translog_is_file(1)) + DBUG_RETURN(LSN_IMPOSSIBLE); + if (addr == MAKE_LSN(1, TRANSLOG_PAGE_SIZE)) + { + /* there is no first page yet */ + DBUG_RETURN(MAKE_LSN(1, TRANSLOG_PAGE_SIZE + + log_descriptor.page_overhead)); + } + + addr= MAKE_LSN(1, TRANSLOG_PAGE_SIZE); /* the first page of the file */ + data.addr= &addr; + if ((page= translog_get_page(&data, buffer)) == NULL) + DBUG_RETURN(LSN_ERROR); + + DBUG_RETURN(MAKE_LSN(1, TRANSLOG_PAGE_SIZE + + page_overhead[page[TRANSLOG_PAGE_FLAGS]])); } diff --git a/storage/maria/ma_loghandler.h b/storage/maria/ma_loghandler.h index db3d43e39f4..4bc4ed1fff9 100644 --- a/storage/maria/ma_loghandler.h +++ b/storage/maria/ma_loghandler.h @@ -266,7 +266,8 @@ extern my_bool translog_inited; #define SHARE_ID_MAX 65535 /* array's size */ -extern LSN first_lsn_in_log(); +extern LSN translog_first_lsn_in_log(); +extern LSN translog_first_theoretical_lsn(); /* record parts descriptor */ struct st_translog_parts diff --git a/storage/maria/ma_loghandler_lsn.h b/storage/maria/ma_loghandler_lsn.h index 9e1c4632fb0..df41ceec7c8 100644 --- a/storage/maria/ma_loghandler_lsn.h +++ b/storage/maria/ma_loghandler_lsn.h @@ -81,6 +81,8 @@ typedef LSN LSN_WITH_FLAGS; #define FILENO_IMPOSSIBLE 0 /**< log file's numbering starts at 1 */ #define LOG_OFFSET_IMPOSSIBLE 0 /**< log always has a header */ #define LSN_IMPOSSIBLE 0 +/* following LSN also is impossible */ +#define LSN_ERROR 1 /** @brief the maximum valid LSN. diff --git a/storage/maria/ma_recovery.c b/storage/maria/ma_recovery.c index 6ed47533fef..5ad8115be46 100644 --- a/storage/maria/ma_recovery.c +++ b/storage/maria/ma_recovery.c @@ -96,7 +96,19 @@ int maria_recover() maria_in_recovery= TRUE; if (last_checkpoint_lsn == LSN_IMPOSSIBLE) - from_lsn= first_lsn_in_log(); + { + from_lsn= translog_first_theoretical_lsn(); + /* + as far as we have not yet any checkpoint then the very first + log file should be present. + */ + DBUG_ASSERT(from_lsn != LSN_IMPOSSIBLE); + /* + @todo process eroror of getting checkpoint + if (from_lsn == ERROR_LSN) + ... + */ + } else { DBUG_ASSERT(0); /* not yet implemented */ diff --git a/storage/maria/maria_read_log.c b/storage/maria/maria_read_log.c index c594fe20490..7c344d5f25d 100644 --- a/storage/maria/maria_read_log.c +++ b/storage/maria/maria_read_log.c @@ -85,7 +85,12 @@ int main(int argc, char **argv) if (opt_only_display) printf("You are using --only-display, NOTHING will be written to disk\n"); - lsn= first_lsn_in_log(); /* LSN could be also --start-from-lsn=# */ + /* LSN could be also --start-from-lsn=# */ + lsn= translog_first_theoretical_lsn(); + /* + @todo process LSN_IMPOSSIBLE and LSN_ERROR values of + translog_first_theoretical_lsn() + */ fprintf(stdout, "TRACE of the last maria_read_log\n"); if (maria_apply_log(lsn, opt_display_and_apply, stdout)) diff --git a/storage/maria/unittest/Makefile.am b/storage/maria/unittest/Makefile.am index 2110a7a5d04..d544ecadb7f 100644 --- a/storage/maria/unittest/Makefile.am +++ b/storage/maria/unittest/Makefile.am @@ -43,7 +43,8 @@ noinst_PROGRAMS = ma_control_file-t trnman-t lockman2-t \ ma_test_loghandler_multithread-t \ ma_test_loghandler_pagecache-t \ ma_test_loghandler_long-t-big \ - ma_test_loghandler_noflush-t + ma_test_loghandler_noflush-t \ + ma_test_loghandler_first_lsn-t ma_test_loghandler_t_SOURCES = ma_test_loghandler-t.c ma_maria_log_cleanup.c ma_test_loghandler_multigroup_t_SOURCES = ma_test_loghandler_multigroup-t.c ma_maria_log_cleanup.c @@ -52,6 +53,7 @@ ma_test_loghandler_pagecache_t_SOURCES = ma_test_loghandler_pagecache-t.c ma_mar ma_test_loghandler_long_t_big_SOURCES = ma_test_loghandler-t.c ma_maria_log_cleanup.c ma_test_loghandler_long_t_big_CPPFLAGS = -DLONG_LOG_TEST ma_test_loghandler_noflush_t_SOURCES = ma_test_loghandler_noflush-t.c ma_maria_log_cleanup.c +ma_test_loghandler_first_lsn_t_SOURCES = ma_test_loghandler_first_lsn-t.c ma_maria_log_cleanup.c ma_pagecache_single_src = ma_pagecache_single.c test_file.c ma_pagecache_consist_src = ma_pagecache_consist.c test_file.c diff --git a/storage/maria/unittest/ma_test_loghandler_first_lsn-t.c b/storage/maria/unittest/ma_test_loghandler_first_lsn-t.c new file mode 100644 index 00000000000..6f9354ec94b --- /dev/null +++ b/storage/maria/unittest/ma_test_loghandler_first_lsn-t.c @@ -0,0 +1,150 @@ +#include "../maria_def.h" +#include +#include +#include +#include "../trnman.h" + +extern my_bool maria_log_remove(); + +#ifndef DBUG_OFF +static const char *default_dbug_option; +#endif + +#define PCACHE_SIZE (1024*1024*10) +#define PCACHE_PAGE TRANSLOG_PAGE_SIZE +#define LOG_FILE_SIZE (1024L*1024L*1024L + 1024L*1024L*512) +#define LOG_FLAGS 0 + +static char *first_translog_file= (char*)"maria_log.00000001"; + +int main(int argc __attribute__((unused)), char *argv[]) +{ + uint pagen; + uchar long_tr_id[6]; + PAGECACHE pagecache; + LSN lsn, first_lsn, theor_lsn; + MY_STAT st; + LEX_STRING parts[TRANSLOG_INTERNAL_PARTS + 1]; + + MY_INIT(argv[0]); + + plan(2); + + bzero(&pagecache, sizeof(pagecache)); + maria_data_root= "."; + if (maria_log_remove()) + exit(1); + /* be sure that we have no logs in the directory*/ + if (my_stat(CONTROL_FILE_BASE_NAME, &st, MYF(0))) + my_delete(CONTROL_FILE_BASE_NAME, MYF(0)); + if (my_stat(first_translog_file, &st, MYF(0))) + my_delete(first_translog_file, MYF(0)); + + bzero(long_tr_id, 6); +#ifndef DBUG_OFF +#if defined(__WIN__) + default_dbug_option= "d:t:i:O,\\ma_test_loghandler.trace"; +#else + default_dbug_option= "d:t:i:o,/tmp/ma_test_loghandler.trace"; +#endif + if (argc > 1) + { + DBUG_SET(default_dbug_option); + DBUG_SET_INITIAL(default_dbug_option); + } +#endif + + if (ma_control_file_create_or_open(TRUE)) + { + fprintf(stderr, "Can't init control file (%d)\n", errno); + exit(1); + } + if ((pagen= init_pagecache(&pagecache, PCACHE_SIZE, 0, 0, + PCACHE_PAGE)) == 0) + { + fprintf(stderr, "Got error: init_pagecache() (errno: %d)\n", errno); + exit(1); + } + if (translog_init(".", LOG_FILE_SIZE, 50112, 0, &pagecache, LOG_FLAGS)) + { + fprintf(stderr, "Can't init loghandler (%d)\n", errno); + translog_destroy(); + exit(1); + } + example_loghandler_init(); + + theor_lsn= translog_first_theoretical_lsn(); + if (theor_lsn == 1) + { + fprintf(stderr, "Error reading the first log file."); + translog_destroy(); + exit(1); + } + if (theor_lsn == LSN_IMPOSSIBLE) + { + fprintf(stderr, "There is no first log file."); + translog_destroy(); + exit(1); + } + first_lsn= translog_first_lsn_in_log(); + if (first_lsn != LSN_IMPOSSIBLE) + { + fprintf(stderr, "Incorrect first lsn responce (%lu,0x%lx).", + (ulong) LSN_FILE_NO(first_lsn), + (ulong) LSN_OFFSET(first_lsn)); + translog_destroy(); + exit(1); + } + ok(1, "Empty log response"); + + + int4store(long_tr_id, 0); + parts[TRANSLOG_INTERNAL_PARTS + 0].str= (char*)long_tr_id; + parts[TRANSLOG_INTERNAL_PARTS + 0].length= 6; + if (translog_write_record(&lsn, + LOGREC_FIXED_RECORD_0LSN_EXAMPLE, + &dummy_transaction_object, NULL, 6, + TRANSLOG_INTERNAL_PARTS + 1, + parts, NULL)) + { + fprintf(stderr, "Can't write record #%lu\n", (ulong) 0); + translog_destroy(); + exit(1); + } + + theor_lsn= translog_first_theoretical_lsn(); + if (theor_lsn == 1) + { + fprintf(stderr, "Error reading the first log file\n"); + translog_destroy(); + exit(1); + } + if (theor_lsn == LSN_IMPOSSIBLE) + { + fprintf(stderr, "There is no first log file\n"); + translog_destroy(); + exit(1); + } + first_lsn= translog_first_lsn_in_log(); + if (first_lsn != theor_lsn) + { + fprintf(stderr, "Incorrect first lsn: (%lu,0x%lx) " + " theoretical first: (%lu,0x%lx)\n", + (ulong) LSN_FILE_NO(first_lsn), + (ulong) LSN_OFFSET(first_lsn), + (ulong) LSN_FILE_NO(theor_lsn), + (ulong) LSN_OFFSET(theor_lsn)); + translog_destroy(); + exit(1); + } + + ok(1, "Full log response"); + + translog_destroy(); + end_pagecache(&pagecache, 1); + ma_control_file_end(); + my_delete(CONTROL_FILE_BASE_NAME, MYF(0)); + my_delete(first_translog_file, MYF(0)); + + exit(0); +} -- cgit v1.2.1 From d0398f729849da3c616be34af24c8aee75b12857 Mon Sep 17 00:00:00 2001 From: unknown Date: Sun, 19 Aug 2007 22:27:43 +0300 Subject: Storing/getting maximum LSN of the record which parts written in the file to the file header. storage/maria/ma_loghandler.h: Getting maximum LSN of the record which parts written in the file to the file header. storage/maria/unittest/Makefile.am: Test suite for getting max LSN added. storage/maria/unittest/ma_test_loghandler_first_lsn-t.c: Spelling fixed. Cleanup fixed. storage/maria/unittest/ma_test_loghandler_noflush-t.c: Cleanup fixed. storage/maria/unittest/ma_test_loghandler_max_lsn-t.c: New BitKeeper file ``storage/maria/unittest/ma_test_loghandler_max_lsn-t.c'' --- storage/maria/ma_loghandler.c | 378 +++++++++++++++++++-- storage/maria/ma_loghandler.h | 1 + storage/maria/unittest/Makefile.am | 4 +- .../unittest/ma_test_loghandler_first_lsn-t.c | 7 +- .../maria/unittest/ma_test_loghandler_max_lsn-t.c | 143 ++++++++ .../maria/unittest/ma_test_loghandler_noflush-t.c | 4 +- 6 files changed, 505 insertions(+), 32 deletions(-) create mode 100644 storage/maria/unittest/ma_test_loghandler_max_lsn-t.c diff --git a/storage/maria/ma_loghandler.c b/storage/maria/ma_loghandler.c index 1bd6d803064..25971e863f6 100644 --- a/storage/maria/ma_loghandler.c +++ b/storage/maria/ma_loghandler.c @@ -152,6 +152,8 @@ struct st_translog_descriptor TRANSLOG_ADDRESS horizon; /* horizon buffer cursor */ struct st_buffer_cursor bc; + /* maximum LSN of the current (not finished) file */ + LSN max_lsn; /* Last flushed LSN */ LSN flushed; @@ -160,6 +162,16 @@ struct st_translog_descriptor /* All what is after this addess is not sent to disk yet */ TRANSLOG_ADDRESS in_buffers_only; pthread_mutex_t sent_to_file_lock; + + /* Protects changing of headers of finished files (max_lsn) */ + pthread_mutex_t file_header_lock; + + /* + Sorted array (with protection) of files where we started writing process + and so we can't give last LSN yet + */ + pthread_mutex_t unfinished_files_lock; + DYNAMIC_ARRAY unfinished_files; }; static struct st_translog_descriptor log_descriptor; @@ -524,7 +536,7 @@ static void translog_check_cursor(struct st_buffer_cursor *cursor) static char *translog_filename_by_fileno(uint32 file_no, char *path) { - char file_name[10 + 8 + 1]; /* See my_sprintf */ + char file_name[10 + 8 + 1]; /* See fallowing my_sprintf() call */ char *res; DBUG_ENTER("translog_filename_by_fileno"); DBUG_ASSERT(file_no <= 0xfffffff); @@ -614,12 +626,47 @@ static my_bool translog_write_file_header() /* file number */ int3store(page, LSN_FILE_NO(log_descriptor.horizon)); page+= 3; + /* + Here should be max lsn storing for current file (which is LSN_IPOSSIBLE): + lsn_store(page, LSN_IPOSSIBLE); + page+= LSN_STORE_SIZE; + But it is zeros so we can rely on bzero() in this case + */ bzero(page, sizeof(page_buff) - (page- page_buff)); DBUG_RETURN(my_pwrite(log_descriptor.log_file_num[0], page_buff, sizeof(page_buff), 0, log_write_flags) != 0); } +/* + @brief write the new LSN on the given file header + + @param file The file descriptor + @param lsn That LSN which should be written + + @retval 0 OK + @retval 1 Error +*/ + +static my_bool translog_max_lsn_to_header(File file, LSN lsn) +{ + uchar lsn_buff[LSN_STORE_SIZE]; + DBUG_ENTER("translog_max_lsn_to_header"); + DBUG_PRINT("enter", ("File descriptor: %ld " + "lsn: (%lu,0x%lx)", + (long) file, + (ulong) LSN_FILE_NO(lsn),(ulong) LSN_OFFSET(lsn))); + + lsn_store(lsn_buff, lsn); + + DBUG_RETURN(my_pwrite(file, lsn_buff, + LSN_STORE_SIZE, + (sizeof(maria_trans_file_magic) + + 8 + 4 + 4 + 4 + 2 + 3), + log_write_flags) != 0 || + my_sync(file, MYF(MY_WME)) != 0); +} + /* Information from transaction log file header @@ -627,6 +674,11 @@ static my_bool translog_write_file_header() typedef struct st_loghandler_file_info { + /* + LSN_IPOSSIBLE for current file and max LSN which parts stored in the + file for all other (finished) files. + */ + LSN max_lsn; ulonglong timestamp; /* Time stamp */ ulong maria_version; /* Version of maria loghandler */ ulong mysql_versiob; /* Version of mysql server */ @@ -636,20 +688,25 @@ typedef struct st_loghandler_file_info } LOGHANDLER_FILE_INFO; /* - @brief Read hander file information from last opened loghandler file + @brief Read hander file information from loghandler file @param desc header information descriptor to be filled with information + @param file file descriptor to read @retval 0 OK @retval 1 Error */ -my_bool translog_read_file_header(LOGHANDLER_FILE_INFO *desc) +#define LOG_HEADER_DATA_SIZE (sizeof(maria_trans_file_magic) + \ + 8 + 4 + 4 + 4 + 2 + 3 + \ + LSN_STORE_SIZE) + +my_bool translog_read_file_header(LOGHANDLER_FILE_INFO *desc, File file) { - uchar page_buff[TRANSLOG_PAGE_SIZE], *ptr; + uchar page_buff[LOG_HEADER_DATA_SIZE], *ptr; DBUG_ENTER("translog_read_file_header"); - if (my_pread(log_descriptor.log_file_num[0], page_buff, + if (my_pread(file, page_buff, sizeof(page_buff), 0, MYF(MY_FNABP | MY_WME))) { DBUG_PRINT("info", ("log read fail error: %d", my_errno)); @@ -663,14 +720,249 @@ my_bool translog_read_file_header(LOGHANDLER_FILE_INFO *desc) desc->mysql_versiob= uint4korr(ptr); ptr+= 4; desc->server_id= uint4korr(ptr); - ptr+= 2; + ptr+= 4; desc->page_size= uint2korr(ptr); ptr+= 2; desc->file_number= uint3korr(ptr); + ptr+=3; + desc->max_lsn= lsn_korr(ptr); + DBUG_RETURN(0); +} + + +/* + @brief set the lsn to the files from_file - to_file if it is greater + then written in the file + + @param from_file first file number (min) + @param to_file last file number (max) + @param lsn the lsn for writing + @param is_locked true if current thread locked the log handler + + @retval 0 OK + @retval 1 Error +*/ + +static my_bool translog_set_lsn_for_files(ulong from_file, ulong to_file, + LSN lsn, my_bool is_locked) +{ + ulong file; + DBUG_ENTER("translog_set_lsn_for_files"); + DBUG_PRINT("enter", ("From: %lu to: %lu lsn: (%lu,0x%lx) locked: %d", + from_file, to_file, + (ulong) LSN_FILE_NO(lsn), (ulong) LSN_OFFSET(lsn), + is_locked)); + DBUG_ASSERT(from_file <= to_file); + DBUG_ASSERT(from_file > 0); /* we have not file 0 */ + + /* Checks the current file (not finished yet file) */ + if (!is_locked) + translog_lock(); + if (to_file == (ulong) LSN_FILE_NO(log_descriptor.horizon)) + { + if (likely(cmp_translog_addr(lsn, log_descriptor.max_lsn) > 0)) + log_descriptor.max_lsn= lsn; + to_file--; + } + if (!is_locked) + translog_unlock(); + + /* Checks finished files if they are */ + pthread_mutex_lock(&log_descriptor.file_header_lock); + for (file= from_file; file <= to_file; file++) + { + LOGHANDLER_FILE_INFO info; + File fd= open_logfile_by_number_no_cache(file); + if (fd < 0 || + translog_read_file_header(&info, fd) || + (cmp_translog_addr(lsn, info.max_lsn) > 0 && + translog_max_lsn_to_header(fd, lsn))) + DBUG_RETURN(1); + } + pthread_mutex_unlock(&log_descriptor.file_header_lock); + DBUG_RETURN(0); } +/* descriptor of file in unfinished_files */ +struct st_file_counter +{ + ulong file; /* file number */ + ulong counter; /* counter for started writes */ +}; + + +/* + @brief mark file "in progress" (for multi-group records) + + @param file log file number +*/ + +static void translog_mark_file_unfinished(ulong file) +{ + int place, i; + struct st_file_counter fc, *fc_ptr; + fc.file= file; fc.counter= 1; + + DBUG_ENTER("translog_mark_file_unfinished"); + DBUG_PRINT("enter", ("file: %lu", file)); + + pthread_mutex_lock(&log_descriptor.unfinished_files_lock); + + if (log_descriptor.unfinished_files.elements == 0) + { + insert_dynamic(&log_descriptor.unfinished_files, (uchar*) &fc); + DBUG_PRINT("info", ("The first element inserted")); + goto end; + } + + for (place= log_descriptor.unfinished_files.elements; + place >= 0; + place--) + { + fc_ptr= dynamic_element(&log_descriptor.unfinished_files, + place, struct st_file_counter *); + if (fc_ptr->file <= file) + break; + } + + if (place >= 0 && fc_ptr->file == file) + { + fc_ptr->counter++; + DBUG_PRINT("info", ("counter increased")); + goto end; + } + + if (place == (int)log_descriptor.unfinished_files.elements) + { + insert_dynamic(&log_descriptor.unfinished_files, (uchar*) &fc); + DBUG_PRINT("info", ("The last element inserted")); + goto end; + } + /* shift and assign new element */ + insert_dynamic(&log_descriptor.unfinished_files, + (uchar*) + dynamic_element(&log_descriptor.unfinished_files, + log_descriptor.unfinished_files.elements- 1, + struct st_file_counter *)); + for(i= log_descriptor.unfinished_files.elements - 1; i > place; i--) + { + /* we do not use set_dynamic() to avoid unneeded checks */ + memcpy(dynamic_element(&log_descriptor.unfinished_files, + i, struct st_file_counter *), + dynamic_element(&log_descriptor.unfinished_files, + i + 1, struct st_file_counter *), + sizeof(struct st_file_counter)); + } + memcpy(dynamic_element(&log_descriptor.unfinished_files, + place + 1, struct st_file_counter *), + &fc, sizeof(struct st_file_counter)); +end: + pthread_mutex_unlock(&log_descriptor.unfinished_files_lock); + DBUG_VOID_RETURN; +} + + + +/* + @brief remove file mark "in progress" (for multi-group records) + + @param file log file number +*/ + +static void translog_mark_file_finished(ulong file) +{ + int i; + struct st_file_counter *fc_ptr; + + DBUG_ENTER("translog_mark_file_finished"); + DBUG_PRINT("enter", ("file: %lu", file)); + + pthread_mutex_lock(&log_descriptor.unfinished_files_lock); + + DBUG_ASSERT(log_descriptor.unfinished_files.elements > 0); + for (i= 0; + i < (int) log_descriptor.unfinished_files.elements; + i++) + { + fc_ptr= dynamic_element(&log_descriptor.unfinished_files, + i, struct st_file_counter *); + if (fc_ptr->file == file) + { + break; + } + } + DBUG_ASSERT(i < (int) log_descriptor.unfinished_files.elements); + + if (! --fc_ptr->counter) + delete_dynamic_element(&log_descriptor.unfinished_files, i); + pthread_mutex_unlock(&log_descriptor.unfinished_files_lock); + DBUG_VOID_RETURN; +} + + +/* + @brief get max LSN of the record which parts stored in this file + + @param file file number + + @return requested LSN or LSN_IMPOSSIBLE/LSN_ERROR + @retval LSN_IMPOSSIBLE File is still not finished + @retval LSN_ERROR Error opening file + @retval # LSN of the record which parts stored in this file +*/ + +LSN translog_get_file_max_lsn_stored(ulong file) +{ + ulong limit= FILENO_IMPOSSIBLE; + DBUG_ENTER("translog_get_file_max_lsn_stored"); + DBUG_PRINT("enter", ("file: %lu", file)); + + pthread_mutex_lock(&log_descriptor.unfinished_files_lock); + + /* find file with minimum file number "in progress" */ + if (log_descriptor.unfinished_files.elements > 0) + { + struct st_file_counter *fc_ptr; + fc_ptr= dynamic_element(&log_descriptor.unfinished_files, + 0, struct st_file_counter *); + limit= fc_ptr->file; /* minimal file number "in progress" */ + } + pthread_mutex_unlock(&log_descriptor.unfinished_files_lock); + + /* + if there is no "in progress file" then unfinished file is in progress + for sure + */ + if (limit == FILENO_IMPOSSIBLE) + { + TRANSLOG_ADDRESS horizon= translog_get_horizon(); + limit= LSN_FILE_NO(horizon); + } + + if (file >= limit) + { + DBUG_PRINT("info", ("The file in in progress")); + DBUG_RETURN(LSN_IMPOSSIBLE); + } + + { + LOGHANDLER_FILE_INFO info; + File fd= open_logfile_by_number_no_cache(file); + if (fd < 0 || + translog_read_file_header(&info, fd)) + { + DBUG_PRINT("error", ("Can't read file header")); + DBUG_RETURN(LSN_ERROR); + } + DBUG_PRINT("error", ("Max lsn: (%lu,0x%lx)", + (ulong) LSN_FILE_NO(info.max_lsn), + (ulong) LSN_OFFSET(info.max_lsn))); + DBUG_RETURN(info.max_lsn); + } +} + /* Initialize transaction log file buffer @@ -752,6 +1044,14 @@ static my_bool translog_create_new_file() uint32 file_no= LSN_FILE_NO(log_descriptor.horizon); DBUG_ENTER("translog_create_new_file"); + /* + Writes max_lsn to the file header before finishing it (it is no need to + lock file header buffer because it is still unfinished file) + */ + translog_max_lsn_to_header(log_descriptor.log_file_num[0], + log_descriptor.max_lsn); + log_descriptor.max_lsn= LSN_IMPOSSIBLE; + if (log_descriptor.log_file_num[OPENED_FILES_NUM - 1] != -1 && translog_close_log_file(log_descriptor.log_file_num[OPENED_FILES_NUM - 1])) @@ -1255,6 +1555,7 @@ static my_bool translog_buffer_next(TRANSLOG_ADDRESS *horizon, #endif if (new_file) { + /* move the horizon to the next file and its header page */ (*horizon)+= LSN_ONE_FILE; (*horizon)= LSN_REPLACE_OFFSET(*horizon, TRANSLOG_PAGE_SIZE); @@ -2234,7 +2535,14 @@ my_bool translog_init(const char *directory, loghandler_init(); /* Safe to do many times */ if (pthread_mutex_init(&log_descriptor.sent_to_file_lock, - MY_MUTEX_INIT_FAST)) + MY_MUTEX_INIT_FAST) || + pthread_mutex_init(&log_descriptor.file_header_lock, + MY_MUTEX_INIT_FAST) || + pthread_mutex_init(&log_descriptor.unfinished_files_lock, + MY_MUTEX_INIT_FAST) || + init_dynamic_array(&log_descriptor.unfinished_files, + sizeof(struct st_file_counter), + 10, 10 CALLER_INFO)) DBUG_RETURN(1); /* Directory to store files */ @@ -2476,7 +2784,7 @@ my_bool translog_init(const char *directory, if (!old_log_was_recovered && old_flags == flags) { LOGHANDLER_FILE_INFO info; - if (translog_read_file_header(&info)) + if (translog_read_file_header(&info, log_descriptor.log_file_num[0])) DBUG_RETURN(1); version_changed= (info.maria_version != TRANSLOG_VERSION_ID); } @@ -2521,6 +2829,7 @@ my_bool translog_init(const char *directory, log_descriptor.sent_to_file= log_descriptor.flushed= log_descriptor.horizon; log_descriptor.in_buffers_only= log_descriptor.bc.buffer->offset; + log_descriptor.max_lsn= LSN_IMPOSSIBLE; /* set to 0 */ /* horizon is (potentially) address of the next LSN we need decrease it to signal that all LSNs before it are flushed @@ -2589,7 +2898,7 @@ void translog_destroy() { uint i; DBUG_ENTER("translog_destroy"); - + if (translog_inited) { if (log_descriptor.bc.buffer->file != -1) @@ -2608,6 +2917,10 @@ void translog_destroy() translog_close_log_file(log_descriptor.log_file_num[i]); } pthread_mutex_destroy(&log_descriptor.sent_to_file_lock); + pthread_mutex_destroy(&log_descriptor.file_header_lock); + pthread_mutex_destroy(&log_descriptor.unfinished_files_lock); + delete_dynamic(&log_descriptor.unfinished_files); + my_close(log_descriptor.directory_fd, MYF(MY_WME)); my_atomic_rwlock_destroy(&LOCK_id_to_share); my_free((uchar*)(id_to_share + 1), MYF(MY_ALLOW_ZERO_PTR)); @@ -3263,9 +3576,11 @@ translog_write_variable_record_1group(LSN *lsn, DBUG_ENTER("translog_write_variable_record_1group"); *lsn= horizon= log_descriptor.horizon; - if (log_record_type_descriptor[type].inwrite_hook && - (*log_record_type_descriptor[type].inwrite_hook)(type, trn, tbl_info, - lsn, parts)) + if (translog_set_lsn_for_files(LSN_FILE_NO(*lsn), LSN_FILE_NO(*lsn), + *lsn, TRUE) || + (log_record_type_descriptor[type].inwrite_hook && + (*log_record_type_descriptor[type].inwrite_hook)(type, trn, tbl_info, + lsn, parts))) { translog_unlock(); DBUG_RETURN(1); @@ -3417,9 +3732,11 @@ translog_write_variable_record_1chunk(LSN *lsn, header_length, chunk0_header); *lsn= log_descriptor.horizon; - if (log_record_type_descriptor[type].inwrite_hook && - (*log_record_type_descriptor[type].inwrite_hook)(type, trn, tbl_info, - lsn, parts)) + if (translog_set_lsn_for_files(LSN_FILE_NO(*lsn), LSN_FILE_NO(*lsn), + *lsn, TRUE) || + (log_record_type_descriptor[type].inwrite_hook && + (*log_record_type_descriptor[type].inwrite_hook)(type, trn, tbl_info, + lsn, parts))) { translog_unlock(); DBUG_RETURN(1); @@ -3797,6 +4114,7 @@ translog_write_variable_record_mgroup(LSN *lsn, uchar chunk2_header[1]; uint header_fixed_part= header_length + 2; uint groups_per_page= (page_capacity - header_fixed_part) / (7 + 1); + uint file_of_the_first_group; DBUG_ENTER("translog_write_variable_record_mgroup"); chunk2_header[0]= TRANSLOG_CHUNK_NOHDR; @@ -3820,6 +4138,8 @@ translog_write_variable_record_mgroup(LSN *lsn, DBUG_ASSERT(record_rest >= buffer_rest); } + file_of_the_first_group= LSN_FILE_NO(log_descriptor.horizon); + translog_mark_file_unfinished(file_of_the_first_group); do { group.addr= horizon= log_descriptor.horizon; @@ -4171,6 +4491,12 @@ translog_write_variable_record_mgroup(LSN *lsn, translog_buffer_decrease_writers(cursor.buffer); rc|= translog_buffer_unlock(cursor.buffer); + if (translog_set_lsn_for_files(file_of_the_first_group, LSN_FILE_NO(*lsn), + *lsn, FALSE)) + goto err; + translog_mark_file_finished(file_of_the_first_group); + + delete_dynamic(&groups); DBUG_RETURN(rc); @@ -4378,9 +4704,11 @@ static my_bool translog_write_fixed_record(LSN *lsn, } *lsn= log_descriptor.horizon; - if (log_record_type_descriptor[type].inwrite_hook && - (*log_record_type_descriptor[type].inwrite_hook) (type, trn, tbl_info, - lsn, parts)) + if (translog_set_lsn_for_files(LSN_FILE_NO(*lsn), LSN_FILE_NO(*lsn), + *lsn, TRUE) || + (log_record_type_descriptor[type].inwrite_hook && + (*log_record_type_descriptor[type].inwrite_hook) (type, trn, tbl_info, + lsn, parts))) { rc= 1; goto err; @@ -4954,7 +5282,7 @@ translog_get_next_chunk(TRANSLOG_SCANNER_DATA *scanner) /** @brief Get header of variable length record and call hook for it processing - + @param page Pointer to the buffer with page where LSN chunk is placed @param page_offset Offset of the first chunk in the page @@ -5180,10 +5508,10 @@ int translog_read_record_header_from_buffer(uchar *page, /** @brief Read record header and some fixed part of a record (the part depend on record type). - + @param lsn log record serial number (address of the record) @param buff log record header buffer - + @note Some type of record can be read completely by this call @note "Decoded" header stored in TRANSLOG_HEADER_BUFFER::header (relative LSN can be translated to absolute one), some fields can be added (like @@ -5222,11 +5550,11 @@ int translog_read_record_header(LSN lsn, TRANSLOG_HEADER_BUFFER *buff) /** @brief Read record header and some fixed part of a record (the part depend on record type). - + @param scan scanner position to read @param buff log record header buffer @param move_scanner request to move scanner to the header position - + @note Some type of record can be read completely by this call @note "Decoded" header stored in TRANSLOG_HEADER_BUFFER::header (relative LSN can be translated to absolute one), some fields can be added (like @@ -6013,7 +6341,7 @@ LSN translog_first_lsn_in_log() TRANSLOG_SCANNER_DATA scanner; DBUG_ENTER("translog_first_lsn_in_log"); DBUG_PRINT("info", ("Horizon: (%lu,0x%lx)", - LSN_FILE_NO(addr), LSN_OFFSET(addr))); + (ulong) LSN_FILE_NO(addr), (ulong) LSN_OFFSET(addr))); if (addr == MAKE_LSN(1, TRANSLOG_PAGE_SIZE)) { @@ -6081,7 +6409,7 @@ LSN translog_first_theoretical_lsn() TRANSLOG_VALIDATOR_DATA data; DBUG_ENTER("translog_first_theoretical_lsn"); DBUG_PRINT("info", ("Horizon: (%lu,0x%lx)", - LSN_FILE_NO(addr), LSN_OFFSET(addr))); + (ulong) LSN_FILE_NO(addr), (ulong) LSN_OFFSET(addr))); if (!translog_is_file(1)) DBUG_RETURN(LSN_IMPOSSIBLE); diff --git a/storage/maria/ma_loghandler.h b/storage/maria/ma_loghandler.h index 4bc4ed1fff9..789057d7e1f 100644 --- a/storage/maria/ma_loghandler.h +++ b/storage/maria/ma_loghandler.h @@ -250,6 +250,7 @@ extern my_bool translog_init_scanner(LSN lsn, extern int translog_read_next_record_header(TRANSLOG_SCANNER_DATA *scanner, TRANSLOG_HEADER_BUFFER *buff); +extern LSN translog_get_file_max_lsn_stored(ulong file); extern my_bool translog_lock(); extern my_bool translog_unlock(); extern void translog_lock_assert_owner(); diff --git a/storage/maria/unittest/Makefile.am b/storage/maria/unittest/Makefile.am index d544ecadb7f..20278399ab8 100644 --- a/storage/maria/unittest/Makefile.am +++ b/storage/maria/unittest/Makefile.am @@ -44,7 +44,8 @@ noinst_PROGRAMS = ma_control_file-t trnman-t lockman2-t \ ma_test_loghandler_pagecache-t \ ma_test_loghandler_long-t-big \ ma_test_loghandler_noflush-t \ - ma_test_loghandler_first_lsn-t + ma_test_loghandler_first_lsn-t \ + ma_test_loghandler_max_lsn-t ma_test_loghandler_t_SOURCES = ma_test_loghandler-t.c ma_maria_log_cleanup.c ma_test_loghandler_multigroup_t_SOURCES = ma_test_loghandler_multigroup-t.c ma_maria_log_cleanup.c @@ -54,6 +55,7 @@ ma_test_loghandler_long_t_big_SOURCES = ma_test_loghandler-t.c ma_maria_log_clea ma_test_loghandler_long_t_big_CPPFLAGS = -DLONG_LOG_TEST ma_test_loghandler_noflush_t_SOURCES = ma_test_loghandler_noflush-t.c ma_maria_log_cleanup.c ma_test_loghandler_first_lsn_t_SOURCES = ma_test_loghandler_first_lsn-t.c ma_maria_log_cleanup.c +ma_test_loghandler_max_lsn_t_SOURCES = ma_test_loghandler_max_lsn-t.c ma_maria_log_cleanup.c ma_pagecache_single_src = ma_pagecache_single.c test_file.c ma_pagecache_consist_src = ma_pagecache_consist.c test_file.c diff --git a/storage/maria/unittest/ma_test_loghandler_first_lsn-t.c b/storage/maria/unittest/ma_test_loghandler_first_lsn-t.c index 6f9354ec94b..81e000a9181 100644 --- a/storage/maria/unittest/ma_test_loghandler_first_lsn-t.c +++ b/storage/maria/unittest/ma_test_loghandler_first_lsn-t.c @@ -89,7 +89,7 @@ int main(int argc __attribute__((unused)), char *argv[]) first_lsn= translog_first_lsn_in_log(); if (first_lsn != LSN_IMPOSSIBLE) { - fprintf(stderr, "Incorrect first lsn responce (%lu,0x%lx).", + fprintf(stderr, "Incorrect first lsn response (%lu,0x%lx).", (ulong) LSN_FILE_NO(first_lsn), (ulong) LSN_OFFSET(first_lsn)); translog_destroy(); @@ -143,8 +143,7 @@ int main(int argc __attribute__((unused)), char *argv[]) translog_destroy(); end_pagecache(&pagecache, 1); ma_control_file_end(); - my_delete(CONTROL_FILE_BASE_NAME, MYF(0)); - my_delete(first_translog_file, MYF(0)); - + if (maria_log_remove()) + exit(1); exit(0); } diff --git a/storage/maria/unittest/ma_test_loghandler_max_lsn-t.c b/storage/maria/unittest/ma_test_loghandler_max_lsn-t.c new file mode 100644 index 00000000000..8bd05e82bbf --- /dev/null +++ b/storage/maria/unittest/ma_test_loghandler_max_lsn-t.c @@ -0,0 +1,143 @@ +#include "../maria_def.h" +#include +#include +#include +#include "../trnman.h" + +extern my_bool maria_log_remove(); + +#ifndef DBUG_OFF +static const char *default_dbug_option; +#endif + +#define PCACHE_SIZE (1024*1024*10) +#define PCACHE_PAGE TRANSLOG_PAGE_SIZE +#define LOG_FILE_SIZE (4*1024L*1024L) +#define LOG_FLAGS 0 + + +int main(int argc __attribute__((unused)), char *argv[]) +{ + ulong i; + uint pagen; + uchar long_tr_id[6]; + PAGECACHE pagecache; + LSN lsn, max_lsn, last_lsn= LSN_IMPOSSIBLE; + MY_STAT st; + LEX_STRING parts[TRANSLOG_INTERNAL_PARTS + 1]; + + MY_INIT(argv[0]); + + plan(2); + + bzero(&pagecache, sizeof(pagecache)); + maria_data_root= "."; + if (maria_log_remove()) + exit(1); + + bzero(long_tr_id, 6); +#ifndef DBUG_OFF +#if defined(__WIN__) + default_dbug_option= "d:t:i:O,\\ma_test_loghandler.trace"; +#else + default_dbug_option= "d:t:i:o,/tmp/ma_test_loghandler.trace"; +#endif + if (argc > 1) + { + DBUG_SET(default_dbug_option); + DBUG_SET_INITIAL(default_dbug_option); + } +#endif + + if (ma_control_file_create_or_open(TRUE)) + { + fprintf(stderr, "Can't init control file (%d)\n", errno); + exit(1); + } + if ((pagen= init_pagecache(&pagecache, PCACHE_SIZE, 0, 0, + PCACHE_PAGE)) == 0) + { + fprintf(stderr, "Got error: init_pagecache() (errno: %d)\n", errno); + exit(1); + } + if (translog_init(".", LOG_FILE_SIZE, 50112, 0, &pagecache, LOG_FLAGS)) + { + fprintf(stderr, "Can't init loghandler (%d)\n", errno); + translog_destroy(); + exit(1); + } + example_loghandler_init(); + + max_lsn= translog_get_file_max_lsn_stored(1); + if (max_lsn == 1) + { + fprintf(stderr, "Error reading the first log file."); + translog_destroy(); + exit(1); + } + if (max_lsn != LSN_IMPOSSIBLE) + { + fprintf(stderr, "Incorrect first lsn response (%lu,0x%lx).", + (ulong) LSN_FILE_NO(max_lsn), + (ulong) LSN_OFFSET(max_lsn)); + translog_destroy(); + exit(1); + } + ok(1, "Empty log response"); + + + /* write more then 1 file */ + int4store(long_tr_id, 0); + parts[TRANSLOG_INTERNAL_PARTS + 0].str= (char*)long_tr_id; + parts[TRANSLOG_INTERNAL_PARTS + 0].length= 6; + for(i= 0; i < LOG_FILE_SIZE/6; i++) + { + if (translog_write_record(&lsn, + LOGREC_FIXED_RECORD_0LSN_EXAMPLE, + &dummy_transaction_object, NULL, 6, + TRANSLOG_INTERNAL_PARTS + 1, + parts, NULL)) + { + fprintf(stderr, "Can't write record #%lu\n", (ulong) 0); + translog_destroy(); + exit(1); + } + if (LSN_FILE_NO(lsn) == 1) + last_lsn= lsn; + } + + + max_lsn= translog_get_file_max_lsn_stored(1); + if (max_lsn == 1) + { + fprintf(stderr, "Error reading the first log file\n"); + translog_destroy(); + exit(1); + } + if (max_lsn == LSN_IMPOSSIBLE) + { + fprintf(stderr, "Isn't first file still finished?!!\n"); + translog_destroy(); + exit(1); + } + if (max_lsn != last_lsn) + { + fprintf(stderr, "Incorrect max lsn: (%lu,0x%lx) " + " last lsn on first file: (%lu,0x%lx)\n", + (ulong) LSN_FILE_NO(max_lsn), + (ulong) LSN_OFFSET(max_lsn), + (ulong) LSN_FILE_NO(last_lsn), + (ulong) LSN_OFFSET(last_lsn)); + translog_destroy(); + exit(1); + } + + ok(1, "First file max LSN"); + + translog_destroy(); + end_pagecache(&pagecache, 1); + ma_control_file_end(); + if (maria_log_remove()) + exit(1); + exit(0); +} diff --git a/storage/maria/unittest/ma_test_loghandler_noflush-t.c b/storage/maria/unittest/ma_test_loghandler_noflush-t.c index fed66da72a4..c924536dde6 100644 --- a/storage/maria/unittest/ma_test_loghandler_noflush-t.c +++ b/storage/maria/unittest/ma_test_loghandler_noflush-t.c @@ -125,8 +125,8 @@ err: translog_destroy(); end_pagecache(&pagecache, 1); ma_control_file_end(); - my_delete(CONTROL_FILE_BASE_NAME, MYF(0)); - my_delete(first_translog_file, MYF(0)); + if (maria_log_remove()) + exit(1); exit(rc); } -- cgit v1.2.1 From a8d94b4ab3826aeaa18d47ce862103591259e140 Mon Sep 17 00:00:00 2001 From: unknown Date: Tue, 21 Aug 2007 20:54:11 +0300 Subject: Fixes for bugs found by maria.test and event*tests: Fixed bug when doing rnd_read followed by update. Don't duplicate error messages in log Initialize transaction object properly. Fixed failure in event*tests when running with maria mysql-test/mysql-test-run.pl: Removed warning when running with --external mysql-test/r/maria.result: Added back disabled test Verified that the result changes are correct mysql-test/t/maria.test: Added back disabled test sql/handler.cc: More debugging. Simple style change sql/sql_class.cc: Initialize transaction object properly. Fixed failure in event*tests when running with maria storage/maria/ha_maria.cc: More dbug info storage/maria/ma_blockrec.c: Removed not needed line storage/maria/ma_rrnd.c: Removed not used code Ensure that cur_row.lastpos is always set when reading record with rnd. (Fixes failure in maria.test) storage/maria/maria_def.h: Don't call maria_print_error() except with EXTRA_DEBUG (Removes duplicate error messages when somethings goes wrong) --- mysql-test/mysql-test-run.pl | 7 +++++-- mysql-test/r/maria.result | 24 +++++++++++++++++++++++- mysql-test/t/maria.test | 8 +++----- sql/handler.cc | 6 +++--- sql/sql_class.cc | 1 + storage/maria/ha_maria.cc | 1 + storage/maria/ma_blockrec.c | 1 - storage/maria/ma_rrnd.c | 10 +--------- storage/maria/maria_def.h | 5 +++++ 9 files changed, 42 insertions(+), 21 deletions(-) diff --git a/mysql-test/mysql-test-run.pl b/mysql-test/mysql-test-run.pl index 9310c8d1626..967133e6b11 100755 --- a/mysql-test/mysql-test-run.pl +++ b/mysql-test/mysql-test-run.pl @@ -883,7 +883,7 @@ sub command_line_setup () { # -------------------------------------------------------------------------- if ($opt_extern) { - mtr_report("Disable instance manager when running with extern mysqld"); + # mtr_report("Disable instance manager when running with extern mysqld"); $opt_skip_im= 1; } elsif ( $mysql_version_id < 50000 ) @@ -2037,7 +2037,10 @@ sub environment_setup () { $ENV{'MYSQL_FIX_SYSTEM_TABLES'}= $cmdline_mysql_fix_system_tables; } - $ENV{'MYSQL_FIX_PRIVILEGE_TABLES'}= $file_mysql_fix_privilege_tables; + if ( !$opt_extern ) + { + $ENV{'MYSQL_FIX_PRIVILEGE_TABLES'}= $file_mysql_fix_privilege_tables; + } # ---------------------------------------------------- # Setup env so childs can execute my_print_defaults diff --git a/mysql-test/r/maria.result b/mysql-test/r/maria.result index 7c5a1b925fa..53063c9c7ab 100644 --- a/mysql-test/r/maria.result +++ b/mysql-test/r/maria.result @@ -435,6 +435,28 @@ bbb. bbb. aa. drop table t1; +create table t1 ( a text not null, key a (a(20))) row_format=dynamic; +insert into t1 values ('aaa '),('aaa'),('aa'); +check table t1; +Table Op Msg_type Msg_text +test.t1 check status OK +repair table t1; +Table Op Msg_type Msg_text +test.t1 repair status OK +select concat(a,'.') from t1 where a='aaa'; +concat(a,'.') +aaa . +aaa. +select concat(a,'.') from t1 where binary a='aaa'; +concat(a,'.') +aaa. +update t1 set a='bbb' where a='aaa'; +select concat(a,'.') from t1; +concat(a,'.') +bbb. +bbb. +aa. +drop table t1; create table t1(a text not null, b text not null, c text not null, index (a(10),b(10),c(10))); insert into t1 values('807780', '477', '165'); insert into t1 values('807780', '477', '162'); @@ -1045,7 +1067,7 @@ id select_type table type possible_keys key key_len ref rows Extra 1 SIMPLE t1 ref c c 11 const # Using where; Using index explain select count(*) from t1 where t='a '; id select_type table type possible_keys key key_len ref rows Extra -1 SIMPLE t1 range t t 13 NULL # Using where +1 SIMPLE t1 ref t t 13 const # Using where explain select count(*) from t1 where v like 'a%'; id select_type table type possible_keys key key_len ref rows Extra 1 SIMPLE t1 range v v 13 NULL # Using where; Using index diff --git a/mysql-test/t/maria.test b/mysql-test/t/maria.test index 2f399d36500..aca85b1442e 100644 --- a/mysql-test/t/maria.test +++ b/mysql-test/t/maria.test @@ -431,10 +431,9 @@ update t1 set a='bbb' where a='aaa'; select concat(a,'.') from t1; drop table t1; -# test again but with dynamic format -echo "Monty please enable the portion below, it gives error message"; -if (0) -{ +# +# Test again but with dynamic format +# create table t1 ( a text not null, key a (a(20))) row_format=dynamic; insert into t1 values ('aaa '),('aaa'),('aa'); check table t1; @@ -444,7 +443,6 @@ select concat(a,'.') from t1 where binary a='aaa'; update t1 set a='bbb' where a='aaa'; select concat(a,'.') from t1; drop table t1; -} # # More space testing diff --git a/sql/handler.cc b/sql/handler.cc index 4b1938fe569..7d03599fe8b 100644 --- a/sql/handler.cc +++ b/sql/handler.cc @@ -2523,10 +2523,10 @@ int handler::ha_repair(THD* thd, HA_CHECK_OPT* check_opt) int ha_enable_transaction(THD *thd, bool on) { int error=0; - DBUG_ENTER("ha_enable_transaction"); - thd->transaction.on= on; - if (on) + DBUG_PRINT("info", ("on: %d", (int) on)); + + if ((thd->transaction.on= on)) { /* Now all storage engines should have transaction handling enabled. diff --git a/sql/sql_class.cc b/sql/sql_class.cc index 40b37ed7405..53651f92001 100644 --- a/sql/sql_class.cc +++ b/sql/sql_class.cc @@ -407,6 +407,7 @@ THD::THD() cleanup_done= abort_on_warning= no_warnings_for_error= 0; peer_port= 0; // For SHOW PROCESSLIST transaction.m_pending_rows_event= 0; + transaction.on= 1; #ifdef SIGNAL_WITH_VIO_CLOSE active_vio = 0; #endif diff --git a/storage/maria/ha_maria.cc b/storage/maria/ha_maria.cc index 99b92c1bcfc..da9f1fa5014 100644 --- a/storage/maria/ha_maria.cc +++ b/storage/maria/ha_maria.cc @@ -2004,6 +2004,7 @@ int ha_maria::external_lock(THD *thd, int lock_type) tons of archived logs to roll-forward, we could then not disable REDOs/UNDOs in this case. */ + DBUG_PRINT("info", ("Disabling logging for table")); _ma_tmp_disable_logging_for_table(file->s); } if (!trn) /* no transaction yet - open it now */ diff --git a/storage/maria/ma_blockrec.c b/storage/maria/ma_blockrec.c index e51b06efbba..bc1d1faab0c 100644 --- a/storage/maria/ma_blockrec.c +++ b/storage/maria/ma_blockrec.c @@ -3357,7 +3357,6 @@ int _ma_read_block_record(MARIA_HA *info, uchar *record, DBUG_ENTER("_ma_read_block_record"); DBUG_PRINT("enter", ("rowid: %lu", (long) record_pos)); - info->cur_row.lastpos= record_pos; offset= ma_recordpos_to_dir_entry(record_pos); DBUG_ASSERT(info->s->pagecache->block_size == block_size); diff --git a/storage/maria/ma_rrnd.c b/storage/maria/ma_rrnd.c index 4f5c2fb06cf..24c4bfdd467 100644 --- a/storage/maria/ma_rrnd.c +++ b/storage/maria/ma_rrnd.c @@ -33,20 +33,12 @@ int maria_rrnd(MARIA_HA *info, uchar *buf, MARIA_RECORD_POS filepos) DBUG_ENTER("maria_rrnd"); DBUG_ASSERT(filepos != HA_OFFSET_ERROR); -#ifdef NOT_USED - if (filepos == HA_OFFSET_ERROR) - { - if (info->cur_row.lastpos == HA_OFFSET_ERROR) /* First read ? */ - filepos= info->s->pack.header_length; /* Read first record */ - else - filepos= info->cur_row.nextpos; - } -#endif /* Init all but update-flag */ info->update&= (HA_STATE_CHANGED | HA_STATE_ROW_CHANGED); if (info->opt_flag & WRITE_CACHE_USED && flush_io_cache(&info->rec_cache)) DBUG_RETURN(my_errno); + info->cur_row.lastpos= filepos; /* Remember for update */ DBUG_RETURN((*info->s->read_record)(info, buf, filepos)); } diff --git a/storage/maria/maria_def.h b/storage/maria/maria_def.h index dfaea4ab727..7dd98bfe1c7 100644 --- a/storage/maria/maria_def.h +++ b/storage/maria/maria_def.h @@ -517,8 +517,13 @@ struct st_maria_info }while(0) #define maria_is_crashed(x) ((x)->s->state.changed & STATE_CRASHED) #define maria_is_crashed_on_repair(x) ((x)->s->state.changed & STATE_CRASHED_ON_REPAIR) +#ifdef EXTRA_DEBUG #define maria_print_error(SHARE, ERRNO) \ _ma_report_error((ERRNO), (SHARE)->index_file_name) +#else +#define maria_print_error(SHARE, ERRNO) while (0) +#endif + /* Functions to store length of space packed keys, VARCHAR or BLOB keys */ -- cgit v1.2.1 From d430e5bfc1327de723911aa22f26eb83b46c6592 Mon Sep 17 00:00:00 2001 From: unknown Date: Wed, 22 Aug 2007 10:56:10 +0300 Subject: Fixed compiler warnings Fixed wrong hash function prototype (causes failure on 64 bit systems) mysql-test/r/rpl_events.result: Removed wrong merge (result file is now identical as in 5.1 tree) mysys/lf_hash.c: Fixed compiler warning mysys/my_safehash.c: Fixed wrong hash function prototype (causes failure on 64 bit systems) storage/maria/unittest/ma_test_loghandler_max_lsn-t.c: Fixed compiler warning --- mysql-test/r/rpl_events.result | 6 ++---- mysys/lf_hash.c | 8 ++++---- mysys/my_safehash.c | 2 +- storage/maria/unittest/ma_test_loghandler_max_lsn-t.c | 1 - 4 files changed, 7 insertions(+), 10 deletions(-) diff --git a/mysql-test/r/rpl_events.result b/mysql-test/r/rpl_events.result index 2cb9a4ab62f..18fe72a9879 100644 --- a/mysql-test/r/rpl_events.result +++ b/mysql-test/r/rpl_events.result @@ -122,15 +122,13 @@ SELECT * FROM t1 ORDER BY id; id c ts 1 manually TIMESTAMP 2 from justonce TIMESTAMP -4 from er TIMESTAMP -affected rows: 3 +affected rows: 2 "Checking event data on the slave" SELECT * FROM t1 ORDER BY id; id c ts 1 manually TIMESTAMP 2 from justonce TIMESTAMP -4 from er TIMESTAMP -affected rows: 3 +affected rows: 2 "Checking event is inactive on slave" SELECT db, name, status, originator FROM mysql.event WHERE db = 'test' AND name = 'justonce'; db name status originator diff --git a/mysys/lf_hash.c b/mysys/lf_hash.c index 3f6b9082ab9..322f04cdc41 100644 --- a/mysys/lf_hash.c +++ b/mysys/lf_hash.c @@ -32,10 +32,10 @@ LF_REQUIRE_PINS(3); /* An element of the list */ typedef struct { - intptr volatile link; /* a pointer to the next element in a listand a flag */ - uint32 hashnr; /* reversed hash number, for sorting */ + intptr volatile link; /* a pointer to the next element in a listand a flag */ + uint32 hashnr; /* reversed hash number, for sorting */ const uchar *key; - uint keylen; + size_t keylen; /* data is stored here, directly after the keylen. thus the pointer to data is (void*)(slist_element_ptr+1) @@ -272,7 +272,7 @@ static LF_SLIST *lsearch(LF_SLIST * volatile *head, CHARSET_INFO *cs, } static inline const uchar* hash_key(const LF_HASH *hash, - const uchar *record, uint *length) + const uchar *record, size_t *length) { if (hash->get_key) return (*hash->get_key)(record, length, 0); diff --git a/mysys/my_safehash.c b/mysys/my_safehash.c index b34ad5f16ff..b3d6439793c 100644 --- a/mysys/my_safehash.c +++ b/mysys/my_safehash.c @@ -70,7 +70,7 @@ static void safe_hash_entry_free(SAFE_HASH_ENTRY *entry) # reference on the key */ -static uchar *safe_hash_entry_get(SAFE_HASH_ENTRY *entry, uint *length, +static uchar *safe_hash_entry_get(SAFE_HASH_ENTRY *entry, size_t *length, my_bool not_used __attribute__((unused))) { *length= entry->length; diff --git a/storage/maria/unittest/ma_test_loghandler_max_lsn-t.c b/storage/maria/unittest/ma_test_loghandler_max_lsn-t.c index 8bd05e82bbf..c50681434e3 100644 --- a/storage/maria/unittest/ma_test_loghandler_max_lsn-t.c +++ b/storage/maria/unittest/ma_test_loghandler_max_lsn-t.c @@ -23,7 +23,6 @@ int main(int argc __attribute__((unused)), char *argv[]) uchar long_tr_id[6]; PAGECACHE pagecache; LSN lsn, max_lsn, last_lsn= LSN_IMPOSSIBLE; - MY_STAT st; LEX_STRING parts[TRANSLOG_INTERNAL_PARTS + 1]; MY_INIT(argv[0]); -- cgit v1.2.1 From 7c273b82a665e35171d7ea0cb07b20054d8c3256 Mon Sep 17 00:00:00 2001 From: unknown Date: Mon, 27 Aug 2007 21:55:39 +0300 Subject: Very simple log files purger added. File number types fixed. storage/maria/unittest/Makefile.am: Test of purger added. storage/maria/unittest/ma_test_loghandler_purge-t.c: New BitKeeper file ``storage/maria/unittest/ma_test_loghandler_purge-t.c'' --- storage/maria/ma_loghandler.c | 173 +++++++++++++++----- storage/maria/ma_loghandler.h | 4 +- storage/maria/unittest/Makefile.am | 4 +- .../maria/unittest/ma_test_loghandler_purge-t.c | 174 +++++++++++++++++++++ 4 files changed, 318 insertions(+), 37 deletions(-) create mode 100644 storage/maria/unittest/ma_test_loghandler_purge-t.c diff --git a/storage/maria/ma_loghandler.c b/storage/maria/ma_loghandler.c index 25971e863f6..b400b37d9e5 100644 --- a/storage/maria/ma_loghandler.c +++ b/storage/maria/ma_loghandler.c @@ -172,6 +172,13 @@ struct st_translog_descriptor */ pthread_mutex_t unfinished_files_lock; DYNAMIC_ARRAY unfinished_files; + + /* Purger data: minimum file in the log (or 0 if unknown) */ + uint32 min_file_number; + /* Protect purger from many calls and it's data */ + pthread_mutex_t purger_lock; + /* last low water mark checked */ + LSN last_lsn_checked; }; static struct st_translog_descriptor log_descriptor; @@ -743,13 +750,13 @@ my_bool translog_read_file_header(LOGHANDLER_FILE_INFO *desc, File file) @retval 1 Error */ -static my_bool translog_set_lsn_for_files(ulong from_file, ulong to_file, +static my_bool translog_set_lsn_for_files(uint32 from_file, uint32 to_file, LSN lsn, my_bool is_locked) { - ulong file; + uint32 file; DBUG_ENTER("translog_set_lsn_for_files"); DBUG_PRINT("enter", ("From: %lu to: %lu lsn: (%lu,0x%lx) locked: %d", - from_file, to_file, + (ulong) from_file, (ulong) to_file, (ulong) LSN_FILE_NO(lsn), (ulong) LSN_OFFSET(lsn), is_locked)); DBUG_ASSERT(from_file <= to_file); @@ -758,7 +765,7 @@ static my_bool translog_set_lsn_for_files(ulong from_file, ulong to_file, /* Checks the current file (not finished yet file) */ if (!is_locked) translog_lock(); - if (to_file == (ulong) LSN_FILE_NO(log_descriptor.horizon)) + if (to_file == (uint32) LSN_FILE_NO(log_descriptor.horizon)) { if (likely(cmp_translog_addr(lsn, log_descriptor.max_lsn) > 0)) log_descriptor.max_lsn= lsn; @@ -788,8 +795,8 @@ static my_bool translog_set_lsn_for_files(ulong from_file, ulong to_file, /* descriptor of file in unfinished_files */ struct st_file_counter { - ulong file; /* file number */ - ulong counter; /* counter for started writes */ + uint32 file; /* file number */ + uint32 counter; /* counter for started writes */ }; @@ -799,14 +806,14 @@ struct st_file_counter @param file log file number */ -static void translog_mark_file_unfinished(ulong file) +static void translog_mark_file_unfinished(uint32 file) { int place, i; struct st_file_counter fc, *fc_ptr; fc.file= file; fc.counter= 1; DBUG_ENTER("translog_mark_file_unfinished"); - DBUG_PRINT("enter", ("file: %lu", file)); + DBUG_PRINT("enter", ("file: %lu", (ulong) file)); pthread_mutex_lock(&log_descriptor.unfinished_files_lock); @@ -871,13 +878,13 @@ end: @param file log file number */ -static void translog_mark_file_finished(ulong file) +static void translog_mark_file_finished(uint32 file) { int i; struct st_file_counter *fc_ptr; DBUG_ENTER("translog_mark_file_finished"); - DBUG_PRINT("enter", ("file: %lu", file)); + DBUG_PRINT("enter", ("file: %lu", (ulong) file)); pthread_mutex_lock(&log_descriptor.unfinished_files_lock); @@ -913,11 +920,11 @@ static void translog_mark_file_finished(ulong file) @retval # LSN of the record which parts stored in this file */ -LSN translog_get_file_max_lsn_stored(ulong file) +LSN translog_get_file_max_lsn_stored(uint32 file) { - ulong limit= FILENO_IMPOSSIBLE; + uint32 limit= FILENO_IMPOSSIBLE; DBUG_ENTER("translog_get_file_max_lsn_stored"); - DBUG_PRINT("enter", ("file: %lu", file)); + DBUG_PRINT("enter", ("file: %lu", (ulong)file)); pthread_mutex_lock(&log_descriptor.unfinished_files_lock); @@ -2540,10 +2547,14 @@ my_bool translog_init(const char *directory, MY_MUTEX_INIT_FAST) || pthread_mutex_init(&log_descriptor.unfinished_files_lock, MY_MUTEX_INIT_FAST) || + pthread_mutex_init(&log_descriptor.purger_lock, + MY_MUTEX_INIT_FAST) || init_dynamic_array(&log_descriptor.unfinished_files, sizeof(struct st_file_counter), 10, 10 CALLER_INFO)) DBUG_RETURN(1); + log_descriptor.min_file_number= 0; + log_descriptor.last_lsn_checked= LSN_IMPOSSIBLE; /* Directory to store files */ unpack_dirname(log_descriptor.directory, directory); @@ -2919,6 +2930,7 @@ void translog_destroy() pthread_mutex_destroy(&log_descriptor.sent_to_file_lock); pthread_mutex_destroy(&log_descriptor.file_header_lock); pthread_mutex_destroy(&log_descriptor.unfinished_files_lock); + pthread_mutex_destroy(&log_descriptor.purger_lock); delete_dynamic(&log_descriptor.unfinished_files); my_close(log_descriptor.directory_fd, MYF(MY_WME)); @@ -6318,39 +6330,45 @@ my_bool translog_is_file(uint file_no) MY_STAT stat_buff; char path[FN_REFLEN]; return (test(my_stat(translog_filename_by_fileno(file_no, path), - &stat_buff, MYF(MY_WME)))); + &stat_buff, MYF(0)))); } /** - @brief returns the LSN of the first record starting in this log + @brief returns minimum log file number - @retval LSN_ERROR Error - @retval LSN_IMPOSSIBLE no log - @retval # LSN of the first record + @param horizon the end of the log + @param is_protected true if it is under purge_log protection + + @retval minimum file number + @retval 0 no files found */ -LSN translog_first_lsn_in_log() +static uint32 translog_first_file(TRANSLOG_ADDRESS horizon, int is_protected) { - TRANSLOG_ADDRESS addr, horizon= translog_get_horizon(); - TRANSLOG_VALIDATOR_DATA data; - uint min_file= 1, max_file= LSN_FILE_NO(horizon); - uint chunk_type; - uint16 chunk_offset; - uchar *page; - TRANSLOG_SCANNER_DATA scanner; - DBUG_ENTER("translog_first_lsn_in_log"); - DBUG_PRINT("info", ("Horizon: (%lu,0x%lx)", - (ulong) LSN_FILE_NO(addr), (ulong) LSN_OFFSET(addr))); + TRANSLOG_ADDRESS addr; + uint min_file= 1, max_file; + DBUG_ENTER("translog_first_file"); + if (!is_protected) + pthread_mutex_lock(&log_descriptor.purger_lock); + if (log_descriptor.min_file_number && + translog_is_file(log_descriptor.min_file_number)) + { + DBUG_PRINT("info", ("cached %lu", + (ulong) log_descriptor.min_file_number)); + if (!is_protected) + pthread_mutex_unlock(&log_descriptor.purger_lock); + DBUG_RETURN(log_descriptor.min_file_number); + } - if (addr == MAKE_LSN(1, TRANSLOG_PAGE_SIZE)) + max_file= LSN_FILE_NO(horizon); + + if (MAKE_LSN(1, TRANSLOG_PAGE_SIZE) >= horizon) { /* there is no first page yet */ - DBUG_RETURN(LSN_IMPOSSIBLE); + DBUG_RETURN(0); } - - /*TODO: lock loghandler purger when it will be created */ /* binary search for last file */ while (min_file != max_file && min_file != (max_file - 1)) { @@ -6364,8 +6382,41 @@ LSN translog_first_lsn_in_log() else min_file= test; } + log_descriptor.min_file_number= max_file; + if (!is_protected) + pthread_mutex_unlock(&log_descriptor.purger_lock); + DBUG_RETURN(max_file); +} + + +/** + @brief returns the LSN of the first record starting in this log + + @retval LSN_ERROR Error + @retval LSN_IMPOSSIBLE no log + @retval # LSN of the first record +*/ + +LSN translog_first_lsn_in_log() +{ + TRANSLOG_ADDRESS addr, horizon= translog_get_horizon(); + TRANSLOG_VALIDATOR_DATA data; + uint file; + uint chunk_type; + uint16 chunk_offset; + uchar *page; + TRANSLOG_SCANNER_DATA scanner; + DBUG_ENTER("translog_first_lsn_in_log"); + DBUG_PRINT("info", ("Horizon: (%lu,0x%lx)", + (ulong) LSN_FILE_NO(addr), (ulong) LSN_OFFSET(addr))); + + if (!(file= translog_first_file(horizon, 0))) + { + /* log has no records yet */ + DBUG_RETURN(LSN_IMPOSSIBLE); + } - addr= MAKE_LSN(max_file, TRANSLOG_PAGE_SIZE); /* the first page of the file */ + addr= MAKE_LSN(file, TRANSLOG_PAGE_SIZE); /* the first page of the file */ data.addr= &addr; if ((page= translog_get_page(&data, scanner.buffer)) == NULL || (chunk_offset= translog_get_first_chunk_offset(page)) == 0) @@ -6415,7 +6466,7 @@ LSN translog_first_theoretical_lsn() DBUG_RETURN(LSN_IMPOSSIBLE); if (addr == MAKE_LSN(1, TRANSLOG_PAGE_SIZE)) { - /* there is no first page yet */ + /* log has no records yet */ DBUG_RETURN(MAKE_LSN(1, TRANSLOG_PAGE_SIZE + log_descriptor.page_overhead)); } @@ -6428,3 +6479,55 @@ LSN translog_first_theoretical_lsn() DBUG_RETURN(MAKE_LSN(1, TRANSLOG_PAGE_SIZE + page_overhead[page[TRANSLOG_PAGE_FLAGS]])); } + + +/** + @brief Check given low water mark and purge files if it is need + + @param low the last (minimum) LSN which is need + + @retval 0 OK + @retval 1 Error +*/ + +my_bool translog_purge(LSN low) +{ + uint32 last_need_file= LSN_FILE_NO(low); + TRANSLOG_ADDRESS horizon= translog_get_horizon(); + int rc= 0; + DBUG_ENTER("translog_purge"); + DBUG_PRINT("enter", ("low: (%lu,0x%lx)", + (ulong)LSN_FILE_NO(low), + (ulong)LSN_OFFSET(low))); + + pthread_mutex_lock(&log_descriptor.purger_lock); + if (LSN_FILE_NO(log_descriptor.last_lsn_checked) < last_need_file) + { + uint32 i; + uint32 min_file= translog_first_file(horizon, 1); + DBUG_ASSERT(min_file != 0); /* log is already started */ + + for(i= min_file; i < last_need_file && rc == 0; i++) + { + LSN lsn= translog_get_file_max_lsn_stored(i); + if (lsn == LSN_IMPOSSIBLE) + break; /* files are still in writing */ + if (lsn == LSN_ERROR) + { + rc= 1; + break; + } + if (cmp_translog_addr(lsn, low) >= 0) + break; + DBUG_PRINT("info", ("purge file %lu", (ulong) i)); + { + char path[FN_REFLEN], *file_name; + file_name= translog_filename_by_fileno(i, path); + rc= test(my_delete(file_name, MYF(MY_WME))); + } + } + } + + pthread_mutex_unlock(&log_descriptor.purger_lock); + DBUG_RETURN(rc); +} diff --git a/storage/maria/ma_loghandler.h b/storage/maria/ma_loghandler.h index 789057d7e1f..ce38eb2ea8d 100644 --- a/storage/maria/ma_loghandler.h +++ b/storage/maria/ma_loghandler.h @@ -250,7 +250,9 @@ extern my_bool translog_init_scanner(LSN lsn, extern int translog_read_next_record_header(TRANSLOG_SCANNER_DATA *scanner, TRANSLOG_HEADER_BUFFER *buff); -extern LSN translog_get_file_max_lsn_stored(ulong file); +extern LSN translog_get_file_max_lsn_stored(uint32 file); +extern my_bool translog_purge(LSN low); +extern my_bool translog_is_file(uint file_no); extern my_bool translog_lock(); extern my_bool translog_unlock(); extern void translog_lock_assert_owner(); diff --git a/storage/maria/unittest/Makefile.am b/storage/maria/unittest/Makefile.am index 20278399ab8..73d903294ce 100644 --- a/storage/maria/unittest/Makefile.am +++ b/storage/maria/unittest/Makefile.am @@ -45,7 +45,8 @@ noinst_PROGRAMS = ma_control_file-t trnman-t lockman2-t \ ma_test_loghandler_long-t-big \ ma_test_loghandler_noflush-t \ ma_test_loghandler_first_lsn-t \ - ma_test_loghandler_max_lsn-t + ma_test_loghandler_max_lsn-t \ + ma_test_loghandler_purge-t ma_test_loghandler_t_SOURCES = ma_test_loghandler-t.c ma_maria_log_cleanup.c ma_test_loghandler_multigroup_t_SOURCES = ma_test_loghandler_multigroup-t.c ma_maria_log_cleanup.c @@ -56,6 +57,7 @@ ma_test_loghandler_long_t_big_CPPFLAGS = -DLONG_LOG_TEST ma_test_loghandler_noflush_t_SOURCES = ma_test_loghandler_noflush-t.c ma_maria_log_cleanup.c ma_test_loghandler_first_lsn_t_SOURCES = ma_test_loghandler_first_lsn-t.c ma_maria_log_cleanup.c ma_test_loghandler_max_lsn_t_SOURCES = ma_test_loghandler_max_lsn-t.c ma_maria_log_cleanup.c +ma_test_loghandler_purge_t_SOURCES = ma_test_loghandler_purge-t.c ma_maria_log_cleanup.c ma_pagecache_single_src = ma_pagecache_single.c test_file.c ma_pagecache_consist_src = ma_pagecache_consist.c test_file.c diff --git a/storage/maria/unittest/ma_test_loghandler_purge-t.c b/storage/maria/unittest/ma_test_loghandler_purge-t.c new file mode 100644 index 00000000000..1beeb442f8f --- /dev/null +++ b/storage/maria/unittest/ma_test_loghandler_purge-t.c @@ -0,0 +1,174 @@ +#include "../maria_def.h" +#include +#include +#include +#include "../trnman.h" + +extern my_bool maria_log_remove(); + +#ifndef DBUG_OFF +static const char *default_dbug_option; +#endif + +#define PCACHE_SIZE (1024*1024*10) +#define PCACHE_PAGE TRANSLOG_PAGE_SIZE +#define LOG_FILE_SIZE (4*1024L*1024L) +#define LOG_FLAGS 0 +#define LONG_BUFFER_SIZE (LOG_FILE_SIZE + LOG_FILE_SIZE / 2) + + +int main(int argc __attribute__((unused)), char *argv[]) +{ + ulong i; + uint pagen; + uchar long_tr_id[6]; + PAGECACHE pagecache; + LSN lsn; + LEX_STRING parts[TRANSLOG_INTERNAL_PARTS + 1]; + uchar *long_buffer= malloc(LONG_BUFFER_SIZE); + + MY_INIT(argv[0]); + + plan(4); + + bzero(&pagecache, sizeof(pagecache)); + bzero(long_buffer, LONG_BUFFER_SIZE); + maria_data_root= "."; + if (maria_log_remove()) + exit(1); + + bzero(long_tr_id, 6); +#ifndef DBUG_OFF +#if defined(__WIN__) + default_dbug_option= "d:t:i:O,\\ma_test_loghandler.trace"; +#else + default_dbug_option= "d:t:i:o,/tmp/ma_test_loghandler.trace"; +#endif + if (argc > 1) + { + DBUG_SET(default_dbug_option); + DBUG_SET_INITIAL(default_dbug_option); + } +#endif + + if (ma_control_file_create_or_open(TRUE)) + { + fprintf(stderr, "Can't init control file (%d)\n", errno); + exit(1); + } + if ((pagen= init_pagecache(&pagecache, PCACHE_SIZE, 0, 0, + PCACHE_PAGE)) == 0) + { + fprintf(stderr, "Got error: init_pagecache() (errno: %d)\n", errno); + exit(1); + } + if (translog_init(".", LOG_FILE_SIZE, 50112, 0, &pagecache, LOG_FLAGS)) + { + fprintf(stderr, "Can't init loghandler (%d)\n", errno); + translog_destroy(); + exit(1); + } + example_loghandler_init(); + + /* write more then 1 file */ + int4store(long_tr_id, 0); + parts[TRANSLOG_INTERNAL_PARTS + 0].str= (char*)long_tr_id; + parts[TRANSLOG_INTERNAL_PARTS + 0].length= 6; + if (translog_write_record(&lsn, + LOGREC_FIXED_RECORD_0LSN_EXAMPLE, + &dummy_transaction_object, NULL, 6, + TRANSLOG_INTERNAL_PARTS + 1, + parts, NULL)) + { + fprintf(stderr, "Can't write record #%lu\n", (ulong) 0); + translog_destroy(); + exit(1); + } + + translog_purge(lsn); + if (!translog_is_file(1)) + { + fprintf(stderr, "First file was removed after first record\n"); + translog_destroy(); + exit(1); + } + ok(1, "First is not removed"); + + for(i= 0; i < LOG_FILE_SIZE/6 && LSN_FILE_NO(lsn) == 1; i++) + { + if (translog_write_record(&lsn, + LOGREC_FIXED_RECORD_0LSN_EXAMPLE, + &dummy_transaction_object, NULL, 6, + TRANSLOG_INTERNAL_PARTS + 1, + parts, NULL)) + { + fprintf(stderr, "Can't write record #%lu\n", (ulong) 0); + translog_destroy(); + exit(1); + } + } + + translog_purge(lsn); + if (translog_is_file(1)) + { + fprintf(stderr, "First file was not removed.\n"); + translog_destroy(); + exit(1); + } + + ok(1, "First file is removed"); + + parts[TRANSLOG_INTERNAL_PARTS + 0].str= (char*)long_buffer; + parts[TRANSLOG_INTERNAL_PARTS + 0].length= LONG_BUFFER_SIZE; + if (translog_write_record(&lsn, + LOGREC_VARIABLE_RECORD_0LSN_EXAMPLE, + &dummy_transaction_object, NULL, LONG_BUFFER_SIZE, + TRANSLOG_INTERNAL_PARTS + 1, parts, NULL)) + { + fprintf(stderr, "Can't write variable record\n"); + translog_destroy(); + exit(1); + } + + translog_purge(lsn); + if (!translog_is_file(2) || !translog_is_file(3)) + { + fprintf(stderr, "Second file (%d) or third file (%d) is not present.\n", + translog_is_file(2), translog_is_file(3)); + translog_destroy(); + exit(1); + } + + ok(1, "Second and third files are not removed"); + + int4store(long_tr_id, 0); + parts[TRANSLOG_INTERNAL_PARTS + 0].str= (char*)long_tr_id; + parts[TRANSLOG_INTERNAL_PARTS + 0].length= 6; + if (translog_write_record(&lsn, + LOGREC_FIXED_RECORD_0LSN_EXAMPLE, + &dummy_transaction_object, NULL, 6, + TRANSLOG_INTERNAL_PARTS + 1, + parts, NULL)) + { + fprintf(stderr, "Can't write last record\n"); + translog_destroy(); + exit(1); + } + + translog_purge(lsn); + if (translog_is_file(2)) + { + fprintf(stderr, "Second file is not removed\n"); + translog_destroy(); + exit(1); + } + + ok(1, "Second file is removed"); + + translog_destroy(); + end_pagecache(&pagecache, 1); + ma_control_file_end(); + if (maria_log_remove()) + exit(1); + exit(0); +} -- cgit v1.2.1 From f7b766c029e087900792fa4abd60330f681f20ff Mon Sep 17 00:00:00 2001 From: unknown Date: Wed, 29 Aug 2007 09:03:10 +0300 Subject: Added maria_commit() and maria_begin() to be used with external tests Now ma_test1 -M -T and ma_test2 -M -T produces readable, applyable logs Note: The .MAD file is not binary identical after applying redo compare to a an original file. (This is becasue we don't have full information which function called PURGE_REDO_BLOCKS). To verify if a file was correctly applied, we now instead compare row checksums BitKeeper/etc/ignore: added storage/maria/tmp/* include/maria.h: Added maria_commit() and maria_begin() to be used with external tests storage/maria/ha_maria.cc: Ensure maria_def. is read in C mode storage/maria/ma_blockrec.c: Fixed redo handling. _ma_apply_redo_purge_blocks() updated to handle any number of purged blocks Removed code to make data file idenitcal after redo (can't easily be done). See changeset comments Now ma_test1 -M -T and ma_test2 -M -T produces readable, applyable logs storage/maria/ma_commit.c: More DBUG statements Moved variable declaration to start of function (portability fix) Added helper functions 'maria_commit()' and 'maria_begin()' storage/maria/ma_loghandler.c: Fixed wrong REDO_PURGE_BLOCKS initialization storage/maria/ma_recovery.c: Added UNDO_ROW_UPDATE Removed wrong setting of lsn (there was no lsn at the used position) Fixed REDO_PURGE_BLOCKS to handle any number of blocks storage/maria/ma_test1.c: Added transaction support (via maria_begin() & maria_commit()) to get a log that can be applied with maria_read_log storage/maria/ma_test2.c: Added transaction support (via maria_begin() & maria_commit()) to get a log that can be applied with maria_read_log storage/maria/ma_test_recovery: Create temporary files in maria/tmp Verify files with checksums instead of byte comparisons storage/maria/maria_chk.c: When using with -dss we only get filename, records and checksum. This is useful to do a quick comparision if a files is identical to another one. storage/maria/maria_def.h: Added ma_commit() storage/maria/maria_read_log.c: Added --help --- .bzrignore | 3 + include/maria.h | 3 +- storage/maria/ha_maria.cc | 3 +- storage/maria/ma_blockrec.c | 176 +++++++++++++++++++---------------------- storage/maria/ma_commit.c | 72 ++++++++++++++--- storage/maria/ma_loghandler.c | 14 ++-- storage/maria/ma_recovery.c | 39 +++++++-- storage/maria/ma_test1.c | 17 +++- storage/maria/ma_test2.c | 6 +- storage/maria/ma_test_recovery | 41 +++++++--- storage/maria/maria_chk.c | 15 +++- storage/maria/maria_def.h | 1 + storage/maria/maria_read_log.c | 5 ++ 13 files changed, 259 insertions(+), 136 deletions(-) mode change 100644 => 100755 storage/maria/ma_test_recovery diff --git a/.bzrignore b/.bzrignore index bcdb50c37b4..396d3920e39 100644 --- a/.bzrignore +++ b/.bzrignore @@ -3045,3 +3045,6 @@ win/vs71cache.txt win/vs8cache.txt zlib/*.ds? zlib/*.vcproj +maria_log_control +storage/maria/tmp/* +storage/maria/tmp/* diff --git a/include/maria.h b/include/maria.h index e39498149a9..1d7778f7c3d 100644 --- a/include/maria.h +++ b/include/maria.h @@ -300,7 +300,8 @@ extern ha_rows maria_records_in_range(struct st_maria_info *info, int inx, extern int maria_is_changed(struct st_maria_info *info); extern int maria_delete_all_rows(struct st_maria_info *info); extern uint maria_get_pointer_length(ulonglong file_length, uint def); - +extern int maria_commit(struct st_maria_info *info); +extern int maria_begin(struct st_maria_info *info); /* this is used to pass to mysql_mariachk_table */ diff --git a/storage/maria/ha_maria.cc b/storage/maria/ha_maria.cc index da9f1fa5014..59f97c8e1e5 100644 --- a/storage/maria/ha_maria.cc +++ b/storage/maria/ha_maria.cc @@ -27,10 +27,11 @@ #include "ha_maria.h" #include "trnman_public.h" +C_MODE_START #include "maria_def.h" #include "ma_rt_index.h" #include "ma_blockrec.h" -#include "ma_commit.h" +C_MODE_END #define MARIA_CANNOT_ROLLBACK HA_NO_TRANSACTIONS #ifdef MARIA_CANNOT_ROLLBACK diff --git a/storage/maria/ma_blockrec.c b/storage/maria/ma_blockrec.c index bc1d1faab0c..d8f65c7b367 100644 --- a/storage/maria/ma_blockrec.c +++ b/storage/maria/ma_blockrec.c @@ -530,6 +530,21 @@ static inline uint start_of_next_entry(uchar *dir) } +static inline uint end_of_previous_entry(uchar *dir, uchar *end) +{ + uchar *pos; + for (pos= dir + DIR_ENTRY_SIZE ; pos < end ; pos+= DIR_ENTRY_SIZE) + { + uint offset; + if ((offset= uint2korr(pos))) + { + return offset + uint2korr(pos+2); + } + } + return PAGE_HEADER_SIZE; +} + + /* Check that a region is all zero @@ -1438,7 +1453,7 @@ static my_bool free_full_pages(MARIA_HA *info, MARIA_ROW *row) log_data)) DBUG_RETURN(1); - DBUG_RETURN (_ma_bitmap_free_full_pages(info, row->extents, + DBUG_RETURN(_ma_bitmap_free_full_pages(info, row->extents, row->extents_count)); } @@ -1457,6 +1472,7 @@ static my_bool free_full_pages(MARIA_HA *info, MARIA_ROW *row) static my_bool free_full_page_range(MARIA_HA *info, ulonglong page, uint count) { my_bool res= 0; + DBUG_ENTER("free_full_page_range"); if (pagecache_delete_pages(info->s->pagecache, &info->dfile, page, count, PAGECACHE_LOCK_WRITE, 0)) @@ -1490,7 +1506,7 @@ static my_bool free_full_page_range(MARIA_HA *info, ulonglong page, uint count) count)) res= 1; pthread_mutex_unlock(&info->s->bitmap.bitmap_lock); - return res; + DBUG_RETURN(res); } @@ -2470,7 +2486,7 @@ my_bool _ma_update_block_record(MARIA_HA *info, MARIA_RECORD_POS record_pos, /* Update cur_row, if someone calls update at once again */ cur_row->head_length= new_row->total_length; - if (free_full_pages(info, cur_row)) + if (cur_row->extents_count && free_full_pages(info, cur_row)) goto err; DBUG_RETURN(write_block_record(info, oldrec, record, new_row, blocks, 1, &row_pos)); @@ -2492,9 +2508,9 @@ my_bool _ma_update_block_record(MARIA_HA *info, MARIA_RECORD_POS record_pos, } /* Delete old row */ - if (delete_tails(info, cur_row->tail_positions)) + if (*cur_row->tail_positions && delete_tails(info, cur_row->tail_positions)) goto err; - if (free_full_pages(info, cur_row)) + if (cur_row->extents_count && free_full_pages(info, cur_row)) goto err; if (_ma_bitmap_find_new_place(info, new_row, page, head_length, blocks)) goto err; @@ -4207,7 +4223,7 @@ uint _ma_apply_redo_insert_row_head_or_tail(MARIA_HA *info, LSN lsn, PAGE_SUFFIX_SIZE); empty_space= uint2korr(buff + EMPTY_SPACE_OFFSET); - if (max_entry >= rownr) + if (max_entry <= rownr) { /* Add directory entry first in directory and data last on page */ DBUG_ASSERT(max_entry == rownr); @@ -4232,40 +4248,27 @@ uint _ma_apply_redo_insert_row_head_or_tail(MARIA_HA *info, LSN lsn, } else { - /* reuse old empty entry */ - uchar *pos, *end, *end_data; - DBUG_ASSERT(uint2korr(dir) == 0); - if (uint2korr(dir)) - goto err; /* Should have been empty */ - - /* Find start of where we can put data */ - end= (buff + block_size - DIR_ENTRY_SIZE * max_entry - - PAGE_SUFFIX_SIZE); - for (pos= dir ; pos >= end ; pos-= DIR_ENTRY_SIZE) - { - if ((rec_offset= uint2korr(pos))) - { - rec_offset+= uint2korr(pos+2); - break; - } - } - DBUG_ASSERT(pos >= end); - if (pos < end) /* Wrong directory */ - goto err; + /* + reuse old entry. This is empty if the command was an insert and + possible used if the command was an update. + */ + uchar *end_data; + uint rec_end; + + /* Add back space if we are reusing entry */ + empty_space+= uint2korr(dir+2); + + /* Find first possible position where to put new data */ + end_data= (buff + block_size - PAGE_SUFFIX_SIZE - + DIR_ENTRY_SIZE * max_entry); + rec_offset= end_of_previous_entry(dir, end_data); + if (rownr != max_entry -1) + rec_end= start_of_next_entry(dir); + else + rec_end= (uint) (buff - end_data); + DBUG_ASSERT(rec_end > rec_offset); - /* find end data */ - end_data= end; /* Start of directory */ - end= (buff + block_size - PAGE_SUFFIX_SIZE); - for (pos= dir ; pos < end ; pos+= DIR_ENTRY_SIZE) - { - uint offset; - if ((offset= uint2korr(pos))) - { - end_data= buff + offset; - break; - } - } - if ((uint) (end_data - (buff + rec_offset)) < data_length) + if ((uint) (rec_end - rec_offset) < data_length) { uint length; /* Not enough continues space, compact page to get more */ @@ -4397,70 +4400,53 @@ uint _ma_apply_redo_purge_blocks(MARIA_HA *info, { MARIA_SHARE *share= info->s; ulonglong page; - uint page_range; - uint res; + uint page_range, ranges; + uint res= 0; uchar *buff= info->keyread_buff; - uint block_size= share->block_size; DBUG_ENTER("_ma_apply_redo_purge_blocks"); info->keyread_buff_used= 1; - page_range= pagerange_korr(header); - /* works only for a one-page range for now */ - DBUG_ASSERT(page_range == 1); // for now + ranges= pagerange_korr(header); header+= PAGERANGE_STORE_SIZE; - page= page_korr(header); - header+= PAGE_STORE_SIZE; - page_range= pagerange_korr(header); - DBUG_ASSERT(page_range == 1); // for now - if (!(buff= pagecache_read(share->pagecache, - &info->dfile, - page, 0, - buff, PAGECACHE_PLAIN_PAGE, - PAGECACHE_LOCK_LEFT_UNLOCKED, 0))) - DBUG_RETURN(my_errno); - - if (lsn_korr(buff) >= lsn) + while (ranges--) { - /* Already applied */ - goto mark_free_in_bitmap; - } + uint i; + page= page_korr(header); + header+= PAGE_STORE_SIZE; + page_range= pagerange_korr(header); + header+= PAGERANGE_STORE_SIZE; - buff[PAGE_TYPE_OFFSET]= UNALLOCATED_PAGE; + for (i= 0; i < page_range ; i++) + { + if (!(buff= pagecache_read(share->pagecache, + &info->dfile, + page+i, 0, + buff, PAGECACHE_PLAIN_PAGE, + PAGECACHE_LOCK_LEFT_UNLOCKED, 0))) + DBUG_RETURN(my_errno); - /* - Strictly speaking, we don't need to zero the last directory entry of this - page; setting the directory's count to zero is enough (it makes the last - directory entry invisible, irrelevant). - But as the "runtime" code (delete_head_or_tail()) called - delete_dir_entry() which zeroed the entry, if we don't do it here, we get - a difference between runtime and log-applying. Irrelevant, but it's - time-consuming to differentiate irrelevant differences from relevant - ones. So we remove the difference by zeroing the entry. - */ - { - uint rownr= ((uint) ((uchar *) buff)[DIR_COUNT_OFFSET]) - 1; - uchar *dir= (buff + block_size - DIR_ENTRY_SIZE * rownr - - DIR_ENTRY_SIZE - PAGE_SUFFIX_SIZE); - dir[0]= dir[1]= 0; /* Delete entry */ + if (lsn_korr(buff) >= lsn) + { + /* Already applied */ + continue; + } + buff[PAGE_TYPE_OFFSET]= UNALLOCATED_PAGE; + lsn_store(buff, lsn); + if (pagecache_write(share->pagecache, + &info->dfile, page+i, 0, + buff, PAGECACHE_PLAIN_PAGE, + PAGECACHE_LOCK_LEFT_UNLOCKED, + PAGECACHE_PIN_LEFT_UNPINNED, + PAGECACHE_WRITE_DELAY, 0)) + DBUG_RETURN(my_errno); + } + /** @todo leave bitmap lock to the bitmap code... */ + pthread_mutex_lock(&share->bitmap.bitmap_lock); + res= _ma_reset_full_page_bits(info, &share->bitmap, page, page_range); + pthread_mutex_unlock(&share->bitmap.bitmap_lock); + if (res) + DBUG_RETURN(res); } - - buff[DIR_COUNT_OFFSET]= 0; - - lsn_store(buff, lsn); - if (pagecache_write(share->pagecache, - &info->dfile, page, 0, - buff, PAGECACHE_PLAIN_PAGE, - PAGECACHE_LOCK_LEFT_UNLOCKED, - PAGECACHE_PIN_LEFT_UNPINNED, - PAGECACHE_WRITE_DELAY, 0)) - DBUG_RETURN(my_errno); - -mark_free_in_bitmap: - /** @todo leave bitmap lock to the bitmap code... */ - pthread_mutex_lock(&share->bitmap.bitmap_lock); - res= _ma_reset_full_page_bits(info, &share->bitmap, page, 1); - pthread_mutex_unlock(&share->bitmap.bitmap_lock); - - DBUG_RETURN(res); + DBUG_RETURN(0); } diff --git a/storage/maria/ma_commit.c b/storage/maria/ma_commit.c index 88aaee0509f..c8c37ae67db 100644 --- a/storage/maria/ma_commit.c +++ b/storage/maria/ma_commit.c @@ -28,8 +28,13 @@ int ma_commit(TRN *trn) { + int res; + LSN commit_lsn; + LEX_STRING log_array[TRANSLOG_INTERNAL_PARTS]; + DBUG_ENTER("ma_commit"); + if (trn->undo_lsn == 0) /* no work done, rollback (cheaper than commit) */ - return trnman_rollback_trn(trn); + DBUG_RETURN(trnman_rollback_trn(trn)); /* - if COMMIT record is written before trnman_commit_trn(): if Checkpoint comes in the middle it will see trn is not committed, @@ -45,27 +50,76 @@ int ma_commit(TRN *trn) issue (transaction's updates were made visible to other transactions). So we need to go the first way. */ + /** @todo RECOVERY share's state is written to disk only in maria_lock_database(), so COMMIT record is not the last record of the transaction! It is probably an issue. Recovery of the state is a problem not yet solved. */ - LSN commit_lsn; - LEX_STRING log_array[TRANSLOG_INTERNAL_PARTS]; /* We do not store "thd->transaction.xid_state.xid" for now, it will be needed only when we support XA. */ - return - translog_write_record(&commit_lsn, LOGREC_COMMIT, - trn, NULL, 0, - sizeof(log_array)/sizeof(log_array[0]), - log_array, NULL) || - translog_flush(commit_lsn) || trnman_commit_trn(trn); + res= (translog_write_record(&commit_lsn, LOGREC_COMMIT, + trn, NULL, 0, + sizeof(log_array)/sizeof(log_array[0]), + log_array, NULL) || + translog_flush(commit_lsn) || + trnman_commit_trn(trn)); + trn->undo_lsn= 0; /* Note: if trnman_commit_trn() fails above, we have already written the COMMIT record, so Checkpoint and Recovery will see the transaction as committed. */ + DBUG_RETURN(res); +} + + +/** + @brief Writes a COMMIT record for a transaciton associated with a file + + @param info Maria handler + + @return Operation status + @retval 0 ok + @retval # error (disk error or out of memory) +*/ + +int maria_commit(MARIA_HA *info) +{ + return info->s->now_transactional ? ma_commit(info->trn) : 0; +} + + +/** + @brief Starts a transaction on a file handle + + @param info Maria handler + + @return Operation status + @retval 0 ok + @retval # Error code. +*/ + + +int maria_begin(MARIA_HA *info) +{ + DBUG_ENTER("maria_begin"); + + if (info->s->now_transactional) + { + TRN *trn; + struct st_my_thread_var *mysys_var= my_thread_var; + trn= trnman_new_trn(&mysys_var->mutex, + &mysys_var->suspend, + (char*) &mysys_var + STACK_DIRECTION *1024*128); + if (unlikely(!trn)) + DBUG_RETURN(HA_ERR_OUT_OF_MEM); + + DBUG_PRINT("info", ("TRN set to 0x%lx", (ulong) trn)); + info->trn= trn; + } + DBUG_RETURN(0); } diff --git a/storage/maria/ma_loghandler.c b/storage/maria/ma_loghandler.c index 25971e863f6..ed34629b263 100644 --- a/storage/maria/ma_loghandler.c +++ b/storage/maria/ma_loghandler.c @@ -299,13 +299,9 @@ static LOG_DESC INIT_LOGREC_REDO_PURGE_ROW_TAIL= NULL, write_hook_for_redo, NULL, 0, "redo_purge_row_tail", LOGREC_NOT_LAST_IN_GROUP, NULL, NULL}; -/* QQQ: TODO: variable and fixed size??? */ static LOG_DESC INIT_LOGREC_REDO_PURGE_BLOCKS= -{LOGRECTYPE_VARIABLE_LENGTH, - FILEID_STORE_SIZE + PAGERANGE_STORE_SIZE + - PAGE_STORE_SIZE + PAGERANGE_STORE_SIZE, - FILEID_STORE_SIZE + PAGERANGE_STORE_SIZE + - PAGE_STORE_SIZE + PAGERANGE_STORE_SIZE, +{LOGRECTYPE_VARIABLE_LENGTH, 0, + FILEID_STORE_SIZE + PAGERANGE_STORE_SIZE, NULL, write_hook_for_redo, NULL, 0, "redo_purge_blocks", LOGREC_NOT_LAST_IN_GROUP, NULL, NULL}; @@ -5288,8 +5284,9 @@ translog_get_next_chunk(TRANSLOG_SCANNER_DATA *scanner) @param page_offset Offset of the first chunk in the page @param buff Buffer to be filled with header data @param scanner If present should be moved to the header page if - it differ from LSN page - @return Length of header or operation status + it differ from LSN page + + @return Length of header or operation status @retval RECHEADER_READ_ERROR error @retval # number of bytes in TRANSLOG_HEADER_BUFFER::header where @@ -5311,7 +5308,6 @@ int translog_variable_length_header(uchar *page, translog_size_t page_offset, uint16 buffer_length= length; uint16 body_len; TRANSLOG_SCANNER_DATA internal_scanner; - DBUG_ENTER("translog_variable_length_header"); buff->record_length= translog_variable_record_1group_decode_len(&src); diff --git a/storage/maria/ma_recovery.c b/storage/maria/ma_recovery.c index 5ad8115be46..c583a0cdd74 100644 --- a/storage/maria/ma_recovery.c +++ b/storage/maria/ma_recovery.c @@ -53,6 +53,7 @@ prototype_exec_hook(REDO_PURGE_BLOCKS); prototype_exec_hook(REDO_DELETE_ALL); prototype_exec_hook(UNDO_ROW_INSERT); prototype_exec_hook(UNDO_ROW_DELETE); +prototype_exec_hook(UNDO_ROW_UPDATE); prototype_exec_hook(UNDO_ROW_PURGE); prototype_exec_hook(COMMIT); static int end_of_redo_phase(); @@ -197,6 +198,7 @@ int maria_apply_log(LSN lsn, my_bool apply, FILE *trace_file) install_exec_hook(REDO_DELETE_ALL); install_exec_hook(UNDO_ROW_INSERT); install_exec_hook(UNDO_ROW_DELETE); + install_exec_hook(UNDO_ROW_UPDATE); install_exec_hook(UNDO_ROW_PURGE); install_exec_hook(COMMIT); @@ -512,9 +514,6 @@ prototype_exec_hook(REDO_CREATE_TABLE) ptr+= 2; /* set create_rename_lsn (for maria_read_log to be idempotent) */ lsn_store(ptr + sizeof(info->s->state.header) + 2, rec->lsn); - /* we also set is_of_lsn, like maria_create() does */ - lsn_store(ptr + sizeof(info->s->state.header) + 2 + LSN_STORE_SIZE, - rec->lsn); if (my_pwrite(kfile, ptr, kfile_size_before_extension, 0, MYF(MY_NABP|MY_WME)) || my_chsize(kfile, keystart, 0, MYF(MY_WME))) @@ -766,7 +765,7 @@ end: prototype_exec_hook(REDO_INSERT_ROW_TAIL) { int error= 1; - uchar *buff= NULL; + uchar *buff; MARIA_HA *info= get_MARIA_HA_from_REDO_record(rec); if (info == NULL) goto end; @@ -834,11 +833,24 @@ end: prototype_exec_hook(REDO_PURGE_BLOCKS) { int error= 1; + uchar *buff; MARIA_HA *info= get_MARIA_HA_from_REDO_record(rec); if (info == NULL) goto end; + enlarge_buffer(rec); + + if (log_record_buffer.str == NULL || + translog_read_record(rec->lsn, 0, rec->record_length, + log_record_buffer.str, NULL) != + rec->record_length) + { + fprintf(tracef, "Failed to read record\n"); + goto end; + } + + buff= log_record_buffer.str; if (_ma_apply_redo_purge_blocks(info, current_group_end_lsn, - rec->header + FILEID_STORE_SIZE)) + buff + FILEID_STORE_SIZE)) goto end; error= 0; end: @@ -911,6 +923,23 @@ end: } +prototype_exec_hook(UNDO_ROW_UPDATE) +{ + int error= 1; + MARIA_HA *info= get_MARIA_HA_from_UNDO_record(rec); + if (info == NULL) + goto end; + all_active_trans[rec->short_trid].undo_lsn= rec->lsn; + /* + todo: instead of above, call write_hook_for_undo, it will also set + first_undo_lsn + */ + error= 0; +end: + return error; +} + + prototype_exec_hook(UNDO_ROW_PURGE) { int error= 1; diff --git a/storage/maria/ma_test1.c b/storage/maria/ma_test1.c index 44ec6af5d8e..b25fc72bebd 100644 --- a/storage/maria/ma_test1.c +++ b/storage/maria/ma_test1.c @@ -66,7 +66,8 @@ int main(int argc,char *argv[]) TRANSLOG_PAGE_SIZE) == 0) || translog_init(maria_data_root, TRANSLOG_FILE_SIZE, 0, 0, maria_log_pagecache, - TRANSLOG_DEFAULT_FLAGS)) + TRANSLOG_DEFAULT_FLAGS) || + (transactional && trnman_init())) { fprintf(stderr, "Error in initialization"); exit(1); @@ -180,6 +181,8 @@ static int run_test(const char *filename) if (!silent) printf("- Writing key:s\n"); + if (maria_begin(file)) + goto err; my_errno=0; row_count=deleted=0; for (i=49 ; i>=1 ; i-=2 ) @@ -266,8 +269,14 @@ static int run_test(const char *filename) if (!silent) printf("- Reopening file\n"); - if (maria_close(file)) goto err; - if (!(file=maria_open(filename,2,HA_OPEN_ABORT_IF_LOCKED))) goto err; + if (maria_commit(file)) + goto err; + if (maria_close(file)) + goto err; + if (!(file=maria_open(filename,2,HA_OPEN_ABORT_IF_LOCKED))) + goto err; + if (maria_begin(file)) + goto err; if (!skip_delete) { if (!silent) @@ -354,6 +363,8 @@ static int run_test(const char *filename) i-1,error,my_errno,read_record+1); } } + if (maria_commit(file)) + goto err; if (maria_close(file)) goto err; maria_end(); diff --git a/storage/maria/ma_test2.c b/storage/maria/ma_test2.c index e820907dccd..00a7fc33cca 100644 --- a/storage/maria/ma_test2.c +++ b/storage/maria/ma_test2.c @@ -244,13 +244,15 @@ int main(int argc, char *argv[]) if (opt_quick_mode) maria_extra(file,HA_EXTRA_QUICK,0); + maria_begin(file); + for (i=0 ; i < recant ; i++) { ulong blob_length; #if 0 /* Starting from i==72, there was a difference between runtime and - log-appplying. This is now fixed, by not using non_header_data_len in + log-applying. This is now fixed, by not using non_header_data_len in log-applying. */ if (i == 72) goto end; @@ -890,6 +892,8 @@ int main(int argc, char *argv[]) goto err; } end: + if (maria_commit(file)) + goto err; if (maria_close(file)) goto err; maria_panic(HA_PANIC_CLOSE); /* Should close log */ diff --git a/storage/maria/ma_test_recovery b/storage/maria/ma_test_recovery old mode 100644 new mode 100755 index 87850cd18f2..7fb1a302a79 --- a/storage/maria/ma_test_recovery +++ b/storage/maria/ma_test_recovery @@ -1,3 +1,5 @@ +#!/bin/sh + set -e silent="-s" if [ -z "$maria_path" ] @@ -5,6 +7,13 @@ then maria_path="." fi +tmp=$maria_path/tmp + +if test '!' -d $tmp +then + mkdir $tmp +fi + echo "MARIA RECOVERY TESTS - success is if exit code is 0" # runs a program inserting/deleting rows, then moves the resulting table @@ -14,28 +23,40 @@ echo "MARIA RECOVERY TESTS - success is if exit code is 0" for prog in "$maria_path/ma_test1 $silent -M -T --skip-update -c" "$maria_path/ma_test2 $silent -L -K -W -P -M -T -g -c" do - rm -f maria_log* + rm -f maria_log.* maria_log_control echo "TEST WITH $prog" $prog # derive table's name from program's name table=`echo $prog | sed -e 's;.*ma_\(test[0-9]\).*;\1;' ` - $maria_path/maria_chk -dvv $table > maria_chk_message.good.txt 2>&1 - mv -f $table.MAD $table.MAD.good + $maria_path/maria_chk -dvv $table > $tmp/maria_chk_message.good.txt 2>&1 + checksum=`$maria_path/maria_chk -dss $table` + mv -f $table.MAD $tmp/$table.MAD.good rm $table.MAI echo "applying log" - $maria_path/maria_read_log -a > maria_read_log_$table.txt - cmp $table.MAD $table.MAD.good - $maria_path/maria_chk -dvv $table > maria_chk_message.txt 2>&1 + $maria_path/maria_read_log -a > $tmp/maria_read_log_$table.txt + $maria_path/maria_chk -dvv $table > $tmp/maria_chk_message.txt 2>&1 + + # QQ: Remove the following line when we also can recovert the index file + $maria_path/maria_chk -s -r $table + + $maria_path/maria_chk -s -e $table + checksum2=`$maria_path/maria_chk -dss $table` + if test "$checksum" != "$checksum2" + then + echo "checksum differs for $table before and after recovery" + exit 1; + fi +# cmp $table.MAD $tmp/$table.MAD.good # When "recovery of the table's state" is ready, we can test it like this: -# diff maria_chk_message.good.txt maria_chk_message.txt >maria_chk_diff.txt || true -# if [ -s maria_chk_diff.txt ] +# diff $tmp/maria_chk_message.good.txt $tmp/maria_chk_message.txt > $tmp/maria_chk_diff.txt || true +# if [ -s $tmp/maria_chk_diff.txt ] # then # echo "Differences in maria_chk -dvv, recovery not yet perfect !" # echo "========DIFF START=======" -# cat maria_chk_diff.txt +# cat $tmp/maria_chk_diff.txt # echo "========DIFF END=======" # fi - rm -f $table.* maria_chk_*.txt maria_read_log_$table.txt + rm -f $table.* $tmp/maria_chk_*.txt $tmp/maria_read_log_$table.txt done echo "ALL RECOVERY TESTS OK" diff --git a/storage/maria/maria_chk.c b/storage/maria/maria_chk.c index 58953caa57c..43da7698f87 100644 --- a/storage/maria/maria_chk.c +++ b/storage/maria/maria_chk.c @@ -115,7 +115,7 @@ int main(int argc, char **argv) (!(check_param.testflag & (T_REP | T_REP_BY_SORT | T_SORT_RECORDS | T_SORT_INDEX)))) { - uint old_testflag=check_param.testflag; + ulonglong old_testflag=check_param.testflag; if (!(check_param.testflag & T_REP)) check_param.testflag|= T_REP_BY_SORT; check_param.testflag&= ~T_EXTEND; /* Don't needed */ @@ -126,7 +126,8 @@ int main(int argc, char **argv) } else error|=new_error; - if (argc && (!(check_param.testflag & T_SILENT) || check_param.testflag & T_INFO)) + if (argc && (!(check_param.testflag & T_SILENT) || + check_param.testflag & T_INFO)) { puts("\n---------\n"); VOID(fflush(stdout)); @@ -1236,6 +1237,16 @@ static void descript(HA_CHECK *param, register MARIA_HA *info, char *name) char llbuff[22],llbuff2[22]; DBUG_ENTER("describe"); + if (param->testflag & T_VERY_SILENT) + { + longlong checksum= info->state->checksum; + if (!(share->options & (HA_OPTION_CHECKSUM | HA_OPTION_COMPRESS_RECORD))) + checksum= 0; + printf("%s %s %s\n", name, llstr(info->state->records,llbuff), + llstr(checksum, llbuff2)); + DBUG_VOID_RETURN; + } + printf("\nMARIA file: %s\n",name); printf("Record format: %s\n", record_formats[share->data_file_type]); printf("Character set: %s (%d)\n", diff --git a/storage/maria/maria_def.h b/storage/maria/maria_def.h index 7dd98bfe1c7..bf8d7a5971a 100644 --- a/storage/maria/maria_def.h +++ b/storage/maria/maria_def.h @@ -894,6 +894,7 @@ void _ma_restore_status(void *param); void _ma_copy_status(void *to, void *from); my_bool _ma_check_status(void *param); void _ma_reset_status(MARIA_HA *maria); +int ma_commit(struct st_transaction *trn); extern MARIA_HA *_ma_test_if_reopen(char *filename); my_bool _ma_check_table_is_closed(const char *name, const char *where); diff --git a/storage/maria/maria_read_log.c b/storage/maria/maria_read_log.c index 7c344d5f25d..b6bcb2040d6 100644 --- a/storage/maria/maria_read_log.c +++ b/storage/maria/maria_read_log.c @@ -113,6 +113,8 @@ end: static struct my_option my_long_options[] = { + {"help", '?', "Display this help and exit.", + 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, {"only-display", 'o', "display brief info about records's header", (uchar **) &opt_only_display, (uchar **) &opt_only_display, 0, GET_BOOL, NO_ARG,0, 0, 0, 0, 0, 0}, @@ -161,6 +163,9 @@ get_one_option(int optid __attribute__((unused)), char *argument __attribute__((unused))) { switch (optid) { + case '?': + usage(); + exit(0); #ifndef DBUG_OFF case '#': DBUG_SET_INITIAL(argument ? argument : default_dbug_option); -- cgit v1.2.1 From e27890cab0a9155b38df57748af9d20dfcccb590 Mon Sep 17 00:00:00 2001 From: unknown Date: Wed, 29 Aug 2007 16:43:01 +0200 Subject: WL#3072 Maria recovery * create page cache before initializing engine and not after, because Maria's recovery needs a page cache * make the creation of a bitmap page more crash-resistent * bugfix (see ma_blockrec.c) * back to old way: create an 8k bitmap page when creating table * preparations for the UNDO phase: recreate TRNs * preparations for Checkpoint: list of dirty pages, testing of rec_lsn to know if page should be skipped during Recovery (unused in this patch as no Checkpoint module pushed yet) * maria_chk tags repaired table with a special LSN * reworking all around in ma_recovery.c (less duplication) mysys/my_realloc.c: noted an issue in my_realloc() sql/mysqld.cc: page cache needs to be created before engines are initialized, because Maria's initialization may do a recovery which needs the page cache. storage/maria/ha_maria.cc: update to new prototype storage/maria/ma_bitmap.c: when creating the first bitmap page we used chsize to 8192 bytes then pwrite (overwrite) the last 2 bytes (8191-8192). If crash between the two operations, this leaves a bitmap page full without its end marker. A later recovery may try to read this page and find it exists and misses a marker and conclude it's corrupted and fail. Changing the chsize to only 8190 bytes: recovery will then find the page is too short and recreate it entirely. storage/maria/ma_blockrec.c: Fix for a bug: when executing a REDO, if the data page is created, data_file_length was increased before _ma_bitmap_set(): _ma_bitmap_set() called _ma_read_bitmap_page() which, due to the increased data_file_length, expected to find a bitmap page on disk with a correct end marker; if the bitmap page didn't exist already in fact, this failed. Fixed by increasing data_file_length only after _ma_read_bitmap_page() has created the new bitmap page correctly. This bug could happen every time a REDO is about creating a new bitmap page. storage/maria/ma_check.c: empty data file has a bitmap page storage/maria/ma_control_file.c: useless parameter to ma_control_file_create_or_open(), just test if this is recovery. storage/maria/ma_control_file.h: new prototype storage/maria/ma_create.c: Back to how it was before: maria_create() creates an 8k bitmap page. Thus (bugfix) data_file_length needs to reflect this instead of being 0. storage/maria/ma_loghandler.c: as ma_test1 and ma_test2 now use real transactions and not dummy_transaction_object, REDO for INSERT/UPDATE/DELETE are always about real transactions, can assert this. A function for Recovery to assign a short id to a table. storage/maria/ma_loghandler.h: new function storage/maria/ma_loghandler_lsn.h: maria_chk tags repaired tables with this LSN storage/maria/ma_open.c: * enforce that DMLs on transactional tables use real transactions and not dummy_transaction_object. * test if table was repaired with maria_chk (which has to been seen as an import of an external table into the server), test validity of create_rename_lsn (header corruption detection) * comments. storage/maria/ma_recovery.c: * preparations for the UNDO phase: recreate TRNs * preparations for Checkpoint: list of dirty pages, testing of rec_lsn to know if page should be skipped during Recovery (unused in this patch as no Checkpoint module pushed yet) * reworking all around (less duplication) storage/maria/ma_recovery.h: a parameter to say if the UNDO phase should be skipped storage/maria/maria_chk.c: tag repaired tables with a special LSN storage/maria/maria_read_log.c: * update to new prototype * no UNDO phase in maria_read_log for now storage/maria/trnman.c: * a function for Recovery to create a transaction (TRN), needed in the UNDO phase * a function for Recovery to grab an existing transaction, needed in the UNDO phase (rollback all existing transactions) storage/maria/trnman_public.h: new functions --- mysys/my_realloc.c | 11 + sql/mysqld.cc | 17 +- storage/maria/ha_maria.cc | 6 +- storage/maria/ma_bitmap.c | 13 +- storage/maria/ma_blockrec.c | 13 +- storage/maria/ma_check.c | 7 - storage/maria/ma_control_file.c | 13 +- storage/maria/ma_control_file.h | 2 +- storage/maria/ma_create.c | 36 +- storage/maria/ma_loghandler.c | 15 +- storage/maria/ma_loghandler.h | 3 + storage/maria/ma_loghandler_lsn.h | 3 + storage/maria/ma_open.c | 60 ++- storage/maria/ma_recovery.c | 962 +++++++++++++++++++++++++------------- storage/maria/ma_recovery.h | 3 +- storage/maria/maria_chk.c | 2 +- storage/maria/maria_read_log.c | 5 +- storage/maria/trnman.c | 27 ++ storage/maria/trnman_public.h | 2 + 19 files changed, 806 insertions(+), 394 deletions(-) diff --git a/mysys/my_realloc.c b/mysys/my_realloc.c index c7cf1323cd4..a55282e03a0 100644 --- a/mysys/my_realloc.c +++ b/mysys/my_realloc.c @@ -22,6 +22,16 @@ /* My memory re allocator */ +/** + @brief wrapper around realloc() + + @param oldpoint pointer to currently allocated area + @param size new size requested, must be >0 + @param my_flags flags + + @note if size==0 realloc() may return NULL; my_realloc() treats this as an + error which is not the intention of realloc() +*/ void* my_realloc(void* oldpoint, size_t size, myf my_flags) { void *point; @@ -29,6 +39,7 @@ void* my_realloc(void* oldpoint, size_t size, myf my_flags) DBUG_PRINT("my",("ptr: 0x%lx size: %lu my_flags: %d", (long) oldpoint, (ulong) size, my_flags)); + DBUG_ASSERT(size > 0); if (!oldpoint && (my_flags & MY_ALLOW_ZERO_PTR)) DBUG_RETURN(my_malloc(size,my_flags)); #ifdef USE_HALLOC diff --git a/sql/mysqld.cc b/sql/mysqld.cc index dec25215acc..4b403fd85b6 100644 --- a/sql/mysqld.cc +++ b/sql/mysqld.cc @@ -3418,6 +3418,17 @@ server."); using_update_log=1; } + /* call ha_init_key_cache() on all key caches to init them */ + process_key_caches(&ha_init_key_cache); + /* + Maria's pagecache needs to be ready before Maria engine (Recovery uses + pagecache, and Checkpoint may happen at startup). Maria engine is taken up + in plugin_init(). + */ +#ifdef WITH_MARIA_STORAGE_ENGINE + process_pagecaches(&ha_init_pagecache); +#endif /* WITH_MARIA_STORAGE_ENGINE */ + /* Allow storage engine to give real error messages */ if (ha_init_errors()) DBUG_RETURN(1); @@ -3588,12 +3599,6 @@ server."); if (opt_myisam_log) (void) mi_log(1); - /* call ha_init_key_cache() on all key caches to init them */ - process_key_caches(&ha_init_key_cache); -#ifdef WITH_MARIA_STORAGE_ENGINE - process_pagecaches(&ha_init_pagecache); -#endif /* WITH_MARIA_STORAGE_ENGINE */ - #if defined(HAVE_MLOCKALL) && defined(MCL_CURRENT) && !defined(EMBEDDED_LIBRARY) if (locked_in_memory && !getuid()) { diff --git a/storage/maria/ha_maria.cc b/storage/maria/ha_maria.cc index 99b92c1bcfc..7910e093e13 100644 --- a/storage/maria/ha_maria.cc +++ b/storage/maria/ha_maria.cc @@ -32,6 +32,10 @@ #include "ma_blockrec.h" #include "ma_commit.h" +/* + Note that in future versions, only *transactional* Maria tables can + rollback, so this flag should be up or down conditionally. +*/ #define MARIA_CANNOT_ROLLBACK HA_NO_TRANSACTIONS #ifdef MARIA_CANNOT_ROLLBACK #define trans_register_ha(A, B, C) do { /* nothing */ } while(0) @@ -2383,7 +2387,7 @@ static int ha_maria_init(void *p) maria_hton->flags= HTON_CAN_RECREATE | HTON_SUPPORT_LOG_TABLES; bzero(maria_log_pagecache, sizeof(*maria_log_pagecache)); maria_data_root= mysql_real_data_home; - res= maria_init() || ma_control_file_create_or_open(TRUE) || + res= maria_init() || ma_control_file_create_or_open() || (init_pagecache(maria_log_pagecache, TRANSLOG_PAGECACHE_SIZE, 0, 0, TRANSLOG_PAGE_SIZE) == 0) || diff --git a/storage/maria/ma_bitmap.c b/storage/maria/ma_bitmap.c index ca9657128e4..66377172877 100644 --- a/storage/maria/ma_bitmap.c +++ b/storage/maria/ma_bitmap.c @@ -512,15 +512,19 @@ static my_bool _ma_read_bitmap_page(MARIA_SHARE *share, MARIA_FILE_BITMAP *bitmap, ulonglong page) { - my_off_t position= page * bitmap->block_size; + my_off_t end_of_page= (page + 1) * bitmap->block_size; my_bool res; DBUG_ENTER("_ma_read_bitmap_page"); DBUG_ASSERT(page % bitmap->pages_covered == 0); bitmap->page= page; - if (position >= share->state.state.data_file_length) + if (end_of_page > share->state.state.data_file_length) { - share->state.state.data_file_length= position + bitmap->block_size; + /* + Inexistent or half-created page (could be crash in the middle of + _ma_bitmap_create_first(), before appending maria_bitmap_marker). + */ + share->state.state.data_file_length= end_of_page; bzero(bitmap->map, bitmap->block_size); memcpy(bitmap->map + bitmap->block_size - sizeof(maria_bitmap_marker), maria_bitmap_marker, sizeof(maria_bitmap_marker)); @@ -2047,7 +2051,8 @@ int _ma_bitmap_create_first(MARIA_SHARE *share) { uint block_size= share->bitmap.block_size; File file= share->bitmap.file.file; - if (my_chsize(file, block_size, 0, MYF(MY_WME)) || + if (my_chsize(file, block_size - sizeof(maria_bitmap_marker), + 0, MYF(MY_WME)) || my_pwrite(file, maria_bitmap_marker, sizeof(maria_bitmap_marker), block_size - sizeof(maria_bitmap_marker), MYF(MY_NABP | MY_WME))) diff --git a/storage/maria/ma_blockrec.c b/storage/maria/ma_blockrec.c index e51b06efbba..3fef3812f3d 100644 --- a/storage/maria/ma_blockrec.c +++ b/storage/maria/ma_blockrec.c @@ -4163,9 +4163,6 @@ uint _ma_apply_redo_insert_row_head_or_tail(MARIA_HA *info, LSN lsn, empty_space= (block_size - PAGE_OVERHEAD_SIZE); rec_offset= PAGE_HEADER_SIZE; dir= buff+ block_size - PAGE_SUFFIX_SIZE - DIR_ENTRY_SIZE; - - /* Update that file is extended */ - info->state->data_file_length= (page + 1) * info->s->block_size; } else { @@ -4302,6 +4299,16 @@ uint _ma_apply_redo_insert_row_head_or_tail(MARIA_HA *info, LSN lsn, if (_ma_bitmap_set(info, page, page_type == HEAD_PAGE, empty_space)) DBUG_RETURN(my_errno); + /* + Data page and bitmap page are in place, we can update data_file_length in + case we extended the file. We could not do it earlier: bitmap code tests + data_file_length to know if it has to create a new page or not. + */ + { + my_off_t end_of_page= (page + 1) * info->s->block_size; + set_if_bigger(info->state->data_file_length, end_of_page); + } + DBUG_RETURN(0); err: diff --git a/storage/maria/ma_check.c b/storage/maria/ma_check.c index a68a21d0180..1ac1fb3454f 100644 --- a/storage/maria/ma_check.c +++ b/storage/maria/ma_check.c @@ -2046,15 +2046,8 @@ int maria_repair(HA_CHECK *param, register MARIA_HA *info, goto err; } _ma_reset_status(sort_info.new_info); -#ifdef ASK_MONTY /* cf maria_create() */ - /** - @todo ASK_MONTY - without this call, a REPAIR on an empty table leaves the data file of - size 0, which sounds reasonable. - */ if (_ma_initialize_data_file(sort_info.new_info->s, new_file)) goto err; -#endif block_record= 1; } } diff --git a/storage/maria/ma_control_file.c b/storage/maria/ma_control_file.c index 4174a0e797e..3816830d9e1 100644 --- a/storage/maria/ma_control_file.c +++ b/storage/maria/ma_control_file.c @@ -41,6 +41,10 @@ #define CONTROL_FILE_SIZE (CONTROL_FILE_FILENO_OFFSET + CONTROL_FILE_FILENO_SIZE) /* This module owns these two vars. */ +/** + This LSN serves for the two-checkpoint rule, and also to find the + checkpoint record when doing a recovery. +*/ LSN last_checkpoint_lsn= LSN_IMPOSSIBLE; uint32 last_logno= FILENO_IMPOSSIBLE; @@ -68,8 +72,6 @@ static int control_file_fd= -1; the last_checkpoint_lsn and last_logno global variables. Called at engine's start. - @param create_if_missing - @note The format of the control file is: 4 bytes: magic string @@ -78,11 +80,13 @@ static int control_file_fd= -1; 4 bytes: offset in log where last checkpoint is 4 bytes: number of last log + @note If in recovery, file is not created + @return Operation status @retval 0 OK @retval 1 Error (in which case the file is left closed) */ -CONTROL_FILE_ERROR ma_control_file_create_or_open(my_bool create_if_missing) +CONTROL_FILE_ERROR ma_control_file_create_or_open() { char buffer[CONTROL_FILE_SIZE]; char name[FN_REFLEN]; @@ -111,7 +115,8 @@ CONTROL_FILE_ERROR ma_control_file_create_or_open(my_bool create_if_missing) if (create_file) { - if (!create_if_missing) + /* in a recovery, we expect to find a control file */ + if (maria_in_recovery) DBUG_RETURN(CONTROL_FILE_MISSING); if ((control_file_fd= my_create(name, 0, open_flags, MYF(MY_SYNC_DIR))) < 0) diff --git a/storage/maria/ma_control_file.h b/storage/maria/ma_control_file.h index d69f221abb8..88a1780543a 100644 --- a/storage/maria/ma_control_file.h +++ b/storage/maria/ma_control_file.h @@ -61,7 +61,7 @@ extern "C" { If present, reads it to find out last checkpoint's LSN and last log. Called at engine's start. */ -CONTROL_FILE_ERROR ma_control_file_create_or_open(my_bool); +CONTROL_FILE_ERROR ma_control_file_create_or_open(); /* Write information durably to the control file. Called when we have created a new log (after syncing this log's creation) diff --git a/storage/maria/ma_create.c b/storage/maria/ma_create.c index f944b9d8bf7..ba66bdb8ffb 100644 --- a/storage/maria/ma_create.c +++ b/storage/maria/ma_create.c @@ -664,6 +664,14 @@ int maria_create(const char *name, enum data_file_type datafile_type, share.base.keystart = share.state.state.key_file_length= MY_ALIGN(info_length, maria_block_size); + if (share.data_file_type == BLOCK_RECORD) + { + /* + we are going to create a first bitmap page, set data_file_length + to reflect this, before the state goes to disk + */ + share.state.state.data_file_length= maria_block_size; + } share.base.max_key_block_length= maria_block_size; share.base.max_key_length=ALIGN_SIZE(max_key_length+4); share.base.records=ci->max_rows; @@ -1041,36 +1049,8 @@ int maria_create(const char *name, enum data_file_type datafile_type, goto err; errpos=3; - /** - @todo ASK_MONTY - QQ: this sets data_file_length from 0 to 8192, but we wrote the state - already to the index file (because: - - log record is built from index header so state must be written before - log record - - data file must be created after log record, so that "missing log - record" implies "unusable table"). - When we wrote the state, we hadn't called ma_initialize_data_file(), so - the data_file_length is 0! - Thus, we below create a 8192-byte data file, but its recorded size is 0, - so next time we read the bitmap (a maria_write() for example) we'll - overwrite the bitmap we just created below. - It's not very efficient. - It also makes maria_chk_size() print - Size of datafile is: 8192 Should be: 0 - on a freshly created table (run "check.test" with a Maria table). - - Why do we absolutely want to create a 8192-byte page for a freshly - created, empty table? Why don't we leave the data file empty? - Removing the call below at least removes the maria_chk_size() issue. - - Monty wrote on IRC, about a size of 0: - "This basically ok; The first block is a bitmap that may or may not - exists", but later he asked that the first block always exists.??? - */ -#ifdef ASK_MONTY if (_ma_initialize_data_file(&share, dfile)) goto err; -#endif } /* Enlarge files */ diff --git a/storage/maria/ma_loghandler.c b/storage/maria/ma_loghandler.c index 1b1589fcbc7..0dc55796711 100644 --- a/storage/maria/ma_loghandler.c +++ b/storage/maria/ma_loghandler.c @@ -5636,7 +5636,7 @@ static my_bool write_hook_for_redo(enum translog_record_type type non-transactional log records (REPAIR, CREATE, RENAME, DROP) should not call this hook; we trust them but verify ;) */ - DBUG_ASSERT(!(maria_multi_threaded && (trn->trid == 0))); + DBUG_ASSERT(trn->trid != 0); /* If the hook stays so simple, it would be faster to pass !trn->rec_lsn ? trn->rec_lsn : some_dummy_lsn @@ -5665,7 +5665,7 @@ static my_bool write_hook_for_undo(enum translog_record_type type struct st_translog_parts *parts __attribute__ ((unused))) { - DBUG_ASSERT(!(maria_multi_threaded && (trn->trid == 0))); + DBUG_ASSERT(trn->trid != 0); trn->undo_lsn= *lsn; if (unlikely(LSN_WITH_FLAGS_TO_LSN(trn->first_undo_lsn) == 0)) trn->first_undo_lsn= @@ -5778,6 +5778,17 @@ void translog_deassign_id_from_share(MARIA_SHARE *share) } +void translog_assign_id_to_share_from_recovery(MARIA_SHARE *share, + uint16 id) +{ + DBUG_ASSERT(maria_in_recovery && !maria_multi_threaded); + DBUG_ASSERT(share->data_file_type == BLOCK_RECORD); + DBUG_ASSERT(share->id == 0); + DBUG_ASSERT(id_to_share[id] == NULL); + id_to_share[share->id= id]= share; +} + + /** @brief returns the LSN of the first record starting in this log diff --git a/storage/maria/ma_loghandler.h b/storage/maria/ma_loghandler.h index db3d43e39f4..b151bb6299b 100644 --- a/storage/maria/ma_loghandler.h +++ b/storage/maria/ma_loghandler.h @@ -257,6 +257,9 @@ extern TRANSLOG_ADDRESS translog_get_horizon(); extern int translog_assign_id_to_share(struct st_maria_share *share, struct st_transaction *trn); extern void translog_deassign_id_from_share(struct st_maria_share *share); +extern void +translog_assign_id_to_share_from_recovery(struct st_maria_share *share, + uint16 id); extern my_bool translog_inited; /* diff --git a/storage/maria/ma_loghandler_lsn.h b/storage/maria/ma_loghandler_lsn.h index 9e1c4632fb0..c1334934e39 100644 --- a/storage/maria/ma_loghandler_lsn.h +++ b/storage/maria/ma_loghandler_lsn.h @@ -82,6 +82,9 @@ typedef LSN LSN_WITH_FLAGS; #define LOG_OFFSET_IMPOSSIBLE 0 /**< log always has a header */ #define LSN_IMPOSSIBLE 0 +/** @brief some impossible LSN serve as markers */ +#define LSN_REPAIRED_BY_MARIA_CHK ((LSN)1) + /** @brief the maximum valid LSN. Unlike ULONGLONG_MAX, it can be safely used in comparison with valid LSNs diff --git a/storage/maria/ma_open.c b/storage/maria/ma_open.c index b5560220b63..5ee6931f69f 100644 --- a/storage/maria/ma_open.c +++ b/storage/maria/ma_open.c @@ -171,7 +171,8 @@ static MARIA_HA *maria_clone_internal(MARIA_SHARE *share, int mode, share->delay_key_write=1; info.state= &share->state.state; /* Change global values by default */ - info.trn= &dummy_transaction_object; + if (!share->base.born_transactional) /* but for transactional ones ... */ + info.trn= &dummy_transaction_object; /* ... force crash if no trn given */ pthread_mutex_unlock(&share->intern_lock); /* Allocate buffer for one record */ @@ -601,15 +602,30 @@ MARIA_HA *maria_open(const char *name, int mode, uint open_flags) { share->page_type= PAGECACHE_LSN_PAGE; share->base_length+= TRANS_ROW_EXTRA_HEADER_SIZE; - if (unlikely((share->state.create_rename_lsn == (LSN)ULONGLONG_MAX) && - (open_flags & HA_OPEN_FROM_SQL_LAYER))) + if (share->state.create_rename_lsn == LSN_REPAIRED_BY_MARIA_CHK) { /* - This table was repaired with maria_chk. Past log records should be - ignored, future log records should not: we define the present. + Was repaired with maria_chk, maybe later maria_pack-ed. Some sort of + import into the server. It starts its existence (from the point of + view of the server, including server's recovery) now. */ - share->state.create_rename_lsn= translog_get_horizon(); - _ma_update_create_rename_lsn_on_disk(share, TRUE); + if ((open_flags & HA_OPEN_FROM_SQL_LAYER) || maria_in_recovery) + { + share->state.create_rename_lsn= translog_get_horizon(); + _ma_update_create_rename_lsn_on_disk(share, TRUE); + } + } + else if (!LSN_VALID(share->state.create_rename_lsn) && + !(open_flags & HA_OPEN_FOR_REPAIR)) + { + /* + If in Recovery, it will not work. If LSN is invalid and not + LSN_REPAIRED_BY_MARIA_CHK, header must be corrupted. + In both cases, must repair. + */ + my_errno=((share->state.changed & STATE_CRASHED_ON_REPAIR) ? + HA_ERR_CRASHED_ON_REPAIR : HA_ERR_CRASHED_ON_USAGE); + goto err; } } else @@ -699,6 +715,14 @@ MARIA_HA *maria_open(const char *name, int mode, uint open_flags) { share->lock.get_status=_ma_get_status; share->lock.copy_status=_ma_copy_status; + /** + @todo RECOVERY + INSERT DELAYED and concurrent inserts are currently disabled for + transactional tables; when enabled again, we should re-evaluate + what problems the call to _ma_update_status() by + thr_reschedule_write_lock() can do (it may hurt Checkpoint as it + would be without intern_lock, and it modifies the state). + */ share->lock.update_status=_ma_update_status; share->lock.restore_status=_ma_restore_status; share->lock.check_status=_ma_check_status; @@ -958,6 +982,7 @@ uint _ma_state_info_write(File file, MARIA_STATE_INFO *state, uint pWrite) uchar buff[MARIA_STATE_INFO_SIZE + MARIA_STATE_EXTRA_SIZE]; uchar *ptr=buff; uint i, keys= (uint) state->header.keys; + size_t res; DBUG_ENTER("_ma_state_info_write"); memcpy_fixed(ptr,&state->header,sizeof(state->header)); @@ -1013,11 +1038,12 @@ uint _ma_state_info_write(File file, MARIA_STATE_INFO *state, uint pWrite) } } - if (pWrite & 1) - DBUG_RETURN(my_pwrite(file, buff, (size_t) (ptr-buff), 0L, - MYF(MY_NABP | MY_THREADSAFE)) != 0); - DBUG_RETURN(my_write(file, buff, (size_t) (ptr-buff), - MYF(MY_NABP)) != 0); + res= (pWrite & 1) ? + my_pwrite(file, buff, (size_t) (ptr-buff), 0L, + MYF(MY_NABP | MY_THREADSAFE)) : + my_write(file, buff, (size_t) (ptr-buff), + MYF(MY_NABP)); + DBUG_RETURN(res != 0); } @@ -1072,6 +1098,16 @@ uchar *_ma_state_info_read(uchar *ptr, MARIA_STATE_INFO *state) } +/** + @brief Fills the state by reading its copy on disk. + + @note Does nothing in single user mode. + + @param file file to read from + @param state state which will be filled + @param pRead if true, use my_pread(), otherwise my_read() +*/ + uint _ma_state_info_read_dsk(File file, MARIA_STATE_INFO *state, my_bool pRead) { char buff[MARIA_STATE_INFO_SIZE + MARIA_STATE_EXTRA_SIZE]; diff --git a/storage/maria/ma_recovery.c b/storage/maria/ma_recovery.c index 6ed47533fef..c6bb6306771 100644 --- a/storage/maria/ma_recovery.c +++ b/storage/maria/ma_recovery.c @@ -23,25 +23,39 @@ #include "maria_def.h" #include "ma_recovery.h" #include "ma_blockrec.h" +#include "trnman.h" -struct TRN_FOR_RECOVERY +struct st_trn_for_recovery /* used only in the REDO phase */ { - LSN group_start_lsn, undo_lsn; + LSN group_start_lsn, undo_lsn, first_undo_lsn; TrID long_trid; }; - +struct st_dirty_page /* used only in the REDO phase */ +{ + uint64 file_and_page_id; + LSN rec_lsn; +}; +struct st_table_for_recovery /* used in the REDO and UNDO phase */ +{ + MARIA_HA *info; + File org_kfile, org_dfile; /**< OS descriptors when Checkpoint saw table */ +}; /* Variables used by all functions of this module. Ok as single-threaded */ -static struct TRN_FOR_RECOVERY *all_active_trans; -static MARIA_HA **all_tables; -static LSN current_group_end_lsn; -FILE *tracef; /**< trace file for debugging */ - -#define prototype_exec_hook(R) \ -static int exec_LOGREC_ ## R(const TRANSLOG_HEADER_BUFFER *rec) +static struct st_trn_for_recovery *all_active_trans; +static struct st_table_for_recovery *all_tables; +static HASH all_dirty_pages; +static struct st_dirty_page *dirty_pages_pool; +static LSN current_group_end_lsn, + checkpoint_start= LSN_IMPOSSIBLE; +static FILE *tracef; /**< trace file for debugging */ + +#define prototype_exec_hook(R) \ + static int exec_LOGREC_ ## R(const TRANSLOG_HEADER_BUFFER *rec) +#define prototype_exec_hook_dummy(R) \ + static int exec_LOGREC_ ## R(const TRANSLOG_HEADER_BUFFER *rec \ + __attribute ((unused))) prototype_exec_hook(LONG_TRANSACTION_ID); -#ifdef MARIA_CHECKPOINT -prototype_exec_hook(CHECKPOINT); -#endif +prototype_exec_hook_dummy(CHECKPOINT); prototype_exec_hook(REDO_CREATE_TABLE); prototype_exec_hook(REDO_DROP_TABLE); prototype_exec_hook(FILE_ID); @@ -55,7 +69,9 @@ prototype_exec_hook(UNDO_ROW_INSERT); prototype_exec_hook(UNDO_ROW_DELETE); prototype_exec_hook(UNDO_ROW_PURGE); prototype_exec_hook(COMMIT); -static int end_of_redo_phase(); +static int run_redo_phase(LSN lsn, my_bool apply); +static uint end_of_redo_phase(my_bool prepare_for_undo_phase); +static int run_undo_phase(uint unfinished); static void display_record_position(const LOG_DESC *log_desc, const TRANSLOG_HEADER_BUFFER *rec, uint number); @@ -65,71 +81,57 @@ static MARIA_HA *get_MARIA_HA_from_REDO_record(const TRANSLOG_HEADER_BUFFER *rec); static MARIA_HA *get_MARIA_HA_from_UNDO_record(const TRANSLOG_HEADER_BUFFER *rec); -static int close_recovered_table(MARIA_HA *info); - +static void prepare_table_for_close(MARIA_HA *info, LSN at_lsn); +static int parse_checkpoint_record(LSN lsn); +static void new_transaction(uint16 sid, TrID long_id, LSN undo_lsn, + LSN first_undo_lsn); +static int new_table(uint16 sid, const char *name, + File org_kfile, File org_dfile, LSN lsn); +static int new_page(File fileid, pgcache_page_no_t pageid, LSN rec_lsn, + struct st_dirty_page *dirty_page); +static int close_all_tables(); /** @brief global [out] buffer for translog_read_record(); never shrinks */ static LEX_STRING log_record_buffer; #define enlarge_buffer(rec) \ - if (log_record_buffer.length < rec->record_length) \ + if (log_record_buffer.length < (rec)->record_length) \ { \ - log_record_buffer.length= rec->record_length; \ + log_record_buffer.length= (rec)->record_length; \ log_record_buffer.str= my_realloc(log_record_buffer.str, \ - rec->record_length, MYF(MY_WME)); \ + (rec)->record_length, MYF(MY_WME)); \ } #define ALERT_USER() DBUG_ASSERT(0) +#define LSN_IN_HEX(L) (ulong)LSN_FILE_NO(L),(ulong)LSN_OFFSET(L) /** - @brief Recovers from the last checkpoint + @brief Recovers from the last checkpoint. + + Runs the REDO phase using special structures, then sets up the playground + of runtime: recreates transactions inside trnman, open tables with their + two-byte-id mapping; takes a checkpoint and runs the UNDO phase. Closes all + tables. */ int maria_recover() { - my_bool res= TRUE; - LSN from_lsn; + int res= 1; FILE *trace_file; DBUG_ENTER("maria_recover"); DBUG_ASSERT(!maria_in_recovery); maria_in_recovery= TRUE; - if (last_checkpoint_lsn == LSN_IMPOSSIBLE) - from_lsn= first_lsn_in_log(); - else - { - DBUG_ASSERT(0); /* not yet implemented */ - /** - @todo read the checkpoint record, fill structures - and use the minimum of checkpoint_start_lsn, rec_lsn of trns, rec_lsn - of dirty pages. - */ - //from_lsn= something; - } - - /* - mysqld has not yet initialized any page cache. Let's create a dedicated - one for recovery. - */ if ((trace_file= fopen("maria_recovery.trace", "w"))) { fprintf(trace_file, "TRACE of the last MARIA recovery from mysqld\n"); - res= (init_pagecache(maria_pagecache, - /** @todo what size? */ - 1024*1024, - 0, 0, - maria_block_size) == 0) || - maria_apply_log(from_lsn, TRUE, trace_file); - end_pagecache(maria_pagecache, TRUE); + DBUG_ASSERT(maria_pagecache->inited); + res= maria_apply_log(LSN_IMPOSSIBLE, TRUE, trace_file, TRUE); if (!res) fprintf(trace_file, "SUCCESS\n"); fclose(trace_file); } - /** - @todo take checkpoint if log applying did some work. - Be sure to not checkpoint if no work. - */ maria_in_recovery= FALSE; DBUG_RETURN(res); } @@ -138,7 +140,8 @@ int maria_recover() /** @brief Displays and/or applies the log - @param lsn LSN from which log reading/applying should start + @param from_lsn LSN from which log reading/applying should start; + LSN_IMPOSSIBLE means "use last checkpoint" @param apply if log records should be applied or not @param trace_file trace file where progress/debug messages will go @@ -151,189 +154,81 @@ int maria_recover() @retval !=0 Error */ -int maria_apply_log(LSN lsn, my_bool apply, FILE *trace_file) +int maria_apply_log(LSN from_lsn, my_bool apply, FILE *trace_file, + my_bool should_run_undo_phase) { int error= 0; DBUG_ENTER("maria_apply_log"); + DBUG_ASSERT(apply || !should_run_undo_phase); DBUG_ASSERT(!maria_multi_threaded); - all_active_trans= (struct TRN_FOR_RECOVERY *) - my_malloc((SHORT_TRID_MAX + 1) * sizeof(struct TRN_FOR_RECOVERY), + all_active_trans= (struct st_trn_for_recovery *) + my_malloc((SHORT_TRID_MAX + 1) * sizeof(struct st_trn_for_recovery), + MYF(MY_ZEROFILL)); + all_tables= (struct st_table_for_recovery *) + my_malloc((SHARE_ID_MAX + 1) * sizeof(struct st_table_for_recovery), MYF(MY_ZEROFILL)); - all_tables= (MARIA_HA **)my_malloc((SHARE_ID_MAX + 1) * sizeof(MARIA_HA *), - MYF(MY_ZEROFILL)); if (!all_active_trans || !all_tables) goto err; tracef= trace_file; - /* install hooks for execution */ -#define install_exec_hook(R) \ - log_record_type_descriptor[LOGREC_ ## R].record_execute_in_redo_phase= \ - exec_LOGREC_ ## R; - install_exec_hook(LONG_TRANSACTION_ID); -#ifdef MARIA_CHECKPOINT - install_exec_hook(CHECKPOINT); -#endif - install_exec_hook(REDO_CREATE_TABLE); - install_exec_hook(REDO_DROP_TABLE); - install_exec_hook(FILE_ID); - install_exec_hook(REDO_INSERT_ROW_HEAD); - install_exec_hook(REDO_INSERT_ROW_TAIL); - install_exec_hook(REDO_PURGE_ROW_HEAD); - install_exec_hook(REDO_PURGE_ROW_TAIL); - install_exec_hook(REDO_PURGE_BLOCKS); - install_exec_hook(REDO_DELETE_ALL); - install_exec_hook(UNDO_ROW_INSERT); - install_exec_hook(UNDO_ROW_DELETE); - install_exec_hook(UNDO_ROW_PURGE); - install_exec_hook(COMMIT); - current_group_end_lsn= LSN_IMPOSSIBLE; - - TRANSLOG_HEADER_BUFFER rec; - struct st_translog_scanner_data scanner; - uint i= 1; - - int len= translog_read_record_header(lsn, &rec); - - /** @todo EOF should be detected */ - if (len == RECHEADER_READ_ERROR) + if (from_lsn == LSN_IMPOSSIBLE) { - fprintf(tracef, "Cannot find a first record\n"); - goto err; + if (last_checkpoint_lsn == LSN_IMPOSSIBLE) + from_lsn= first_lsn_in_log(); + else + { + DBUG_ASSERT(0); /* not yet implemented */ + from_lsn= parse_checkpoint_record(last_checkpoint_lsn); + if (from_lsn == LSN_IMPOSSIBLE) + goto err; + } } - if (translog_init_scanner(lsn, 1, &scanner)) - { - fprintf(tracef, "Scanner init failed\n"); + if (run_redo_phase(from_lsn, apply)) goto err; - } - for (;;i++) - { - uint16 sid= rec.short_trid; - const LOG_DESC *log_desc= &log_record_type_descriptor[rec.type]; - display_record_position(log_desc, &rec, i); - /* - A complete group is a set of log records with an "end mark" record - (e.g. a set of REDOs for an operation, terminated by an UNDO for this - operation); if there is no "end mark" record the group is incomplete - and won't be executed. - */ - if ((log_desc->record_in_group == LOGREC_IS_GROUP_ITSELF) || - (log_desc->record_in_group == LOGREC_LAST_IN_GROUP)) - { - if (all_active_trans[sid].group_start_lsn != LSN_IMPOSSIBLE) - { - if (log_desc->record_in_group == LOGREC_IS_GROUP_ITSELF) - { - /* - can happen if the transaction got a table write error, then - unlocked tables thus wrote a COMMIT record. - */ - fprintf(tracef, "\nDiscarding unfinished group before this record\n"); - ALERT_USER(); - all_active_trans[sid].group_start_lsn= LSN_IMPOSSIBLE; - } - else - { - /* - There is a complete group for this transaction, containing more - than this event. - */ - fprintf(tracef, " ends a group:\n"); - struct st_translog_scanner_data scanner2; - TRANSLOG_HEADER_BUFFER rec2; - len= - translog_read_record_header(all_active_trans[sid].group_start_lsn, &rec2); - if (len < 0) /* EOF or error */ - { - fprintf(tracef, "Cannot find record where it should be\n"); - goto err; - } - if (translog_init_scanner(rec2.lsn, 1, &scanner2)) - { - fprintf(tracef, "Scanner2 init failed\n"); - goto err; - } - current_group_end_lsn= rec.lsn; - do - { - if (rec2.short_trid == sid) /* it's in our group */ - { - const LOG_DESC *log_desc2= &log_record_type_descriptor[rec2.type]; - display_record_position(log_desc2, &rec2, 0); - if (apply && display_and_apply_record(log_desc2, &rec2)) - goto err; - } - len= translog_read_next_record_header(&scanner2, &rec2); - if (len < 0) /* EOF or error */ - { - fprintf(tracef, "Cannot find record where it should be\n"); - goto err; - } - } - while (rec2.lsn < rec.lsn); - translog_free_record_header(&rec2); - /* group finished */ - all_active_trans[sid].group_start_lsn= LSN_IMPOSSIBLE; - current_group_end_lsn= LSN_IMPOSSIBLE; /* for debugging */ - display_record_position(log_desc, &rec, 0); - } - } - if (apply && display_and_apply_record(log_desc, &rec)) - goto err; - } - else /* record does not end group */ - { - /* just record the fact, can't know if can execute yet */ - if (all_active_trans[sid].group_start_lsn == LSN_IMPOSSIBLE) - { - /* group not yet started */ - all_active_trans[sid].group_start_lsn= rec.lsn; - } - } - len= translog_read_next_record_header(&scanner, &rec); - if (len < 0) - { - switch (len) - { - case RECHEADER_READ_EOF: - fprintf(tracef, "EOF on the log\n"); - break; - case RECHEADER_READ_ERROR: - fprintf(stderr, "Error reading log\n"); - goto err; - } - break; - } + uint unfinished_trans= end_of_redo_phase(should_run_undo_phase); + if (unfinished_trans == (uint)-1) + goto err; + if (should_run_undo_phase) + { + if (run_undo_phase(unfinished_trans)) + return 1; } - translog_free_record_header(&rec); + else if (unfinished_trans > 0) + fprintf(tracef, "WARNING: %u unfinished transactions; some tables may be" + " left inconsistent!\n", unfinished_trans); /* - So we have applied all REDOs. - We may now have unfinished transactions. - I don't think it's this program's job to roll them back: - to roll back and at the same time stay idempotent, it needs to write log - records (without CLRs, 2nd rollback would hit the effects of first - rollback and fail). But this standalone tool is not allowed to write to - the server's transaction log. So we do not roll back anything. - In the real Recovery code, or the code to do "recover after online - backup", yes we will roll back. + we don't use maria_panic() because it would maria_end(), and Recovery does + not want that (we want to keep modules initialized for runtime). */ - if (end_of_redo_phase()) + if (close_all_tables()) goto err; + /* + At this stage, end of recovery, trnman is left initialized. This is for + the future, when we have an online UNDO phase or prepared transactions. + */ goto end; err: error= 1; fprintf(tracef, "Recovery of tables with transaction logs FAILED\n"); end: + hash_free(&all_dirty_pages); + bzero(&all_dirty_pages, sizeof(all_dirty_pages)); + my_free(dirty_pages_pool, MYF(MY_ALLOW_ZERO_PTR)); + dirty_pages_pool= NULL; my_free(all_tables, MYF(MY_ALLOW_ZERO_PTR)); + all_tables= NULL; my_free(all_active_trans, MYF(MY_ALLOW_ZERO_PTR)); + all_active_trans= NULL; my_free(log_record_buffer.str, MYF(MY_ALLOW_ZERO_PTR)); log_record_buffer.str= NULL; log_record_buffer.length= 0; + /* we don't cleanly close tables if we hit some error (may corrupt them) */ DBUG_RETURN(error); } @@ -348,9 +243,8 @@ static void display_record_position(const LOG_DESC *log_desc, form a group, so we indent below the group's end record */ fprintf(tracef, "%sRec#%u LSN (%lu,0x%lx) short_trid %u %s(num_type:%u) len %lu\n", - number ? "" : " ", number, - (ulong) LSN_FILE_NO(rec->lsn), (ulong) LSN_OFFSET(rec->lsn), - rec->short_trid, log_desc->name, rec->type, + number ? "" : " ", number, LSN_IN_HEX(rec->lsn), + rec->short_trid, log_desc->name, rec->type, (ulong)rec->record_length); } @@ -377,11 +271,10 @@ prototype_exec_hook(LONG_TRANSACTION_ID) TrID long_trid= all_active_trans[sid].long_trid; /* abort group of this trn (must be of before a crash) */ LSN gslsn= all_active_trans[sid].group_start_lsn; - char llbuf[22]; if (gslsn != LSN_IMPOSSIBLE) { fprintf(tracef, "Group at LSN (%lu,0x%lx) short_trid %u aborted\n", - (ulong) LSN_FILE_NO(gslsn), (ulong) LSN_OFFSET(gslsn), sid); + LSN_IN_HEX(gslsn), sid); all_active_trans[sid].group_start_lsn= LSN_IMPOSSIBLE; } if (long_trid != 0) @@ -389,18 +282,17 @@ prototype_exec_hook(LONG_TRANSACTION_ID) LSN ulsn= all_active_trans[sid].undo_lsn; if (ulsn != LSN_IMPOSSIBLE) { + char llbuf[22]; llstr(long_trid, llbuf); fprintf(tracef, "Found an old transaction long_trid %s short_trid %u" " with same short id as this new transaction, and has neither" " committed nor rollback (undo_lsn: (%lu,0x%lx))\n", llbuf, - sid, (ulong) LSN_FILE_NO(ulsn), (ulong) LSN_OFFSET(ulsn)); + sid, LSN_IN_HEX(ulsn)); goto err; } } long_trid= uint6korr(rec->header); - all_active_trans[sid].long_trid= long_trid; - llstr(long_trid, llbuf); - fprintf(tracef, "Transaction long_trid %s short_trid %u starts\n", llbuf, sid); + new_transaction(sid, long_trid, LSN_IMPOSSIBLE, LSN_IMPOSSIBLE); goto end; err: ALERT_USER(); @@ -410,13 +302,24 @@ end: } -#ifdef MARIA_CHECKPOINT -prototype_exec_hook(CHECKPOINT) +static void new_transaction(uint16 sid, TrID long_id, LSN undo_lsn, + LSN first_undo_lsn) +{ + char llbuf[22]; + all_active_trans[sid].long_trid= long_id; + llstr(long_id, llbuf); + fprintf(tracef, "Transaction long_trid %s short_trid %u starts\n", + llbuf, sid); + all_active_trans[sid].undo_lsn= undo_lsn; + all_active_trans[sid].first_undo_lsn= first_undo_lsn; +} + + +prototype_exec_hook_dummy(CHECKPOINT) { /* the only checkpoint we care about was found via control file, ignore */ return 0; } -#endif prototype_exec_hook(REDO_CREATE_TABLE) @@ -461,9 +364,9 @@ prototype_exec_hook(REDO_CREATE_TABLE) } if (cmp_translog_addr(share->state.create_rename_lsn, rec->lsn) >= 0) { - fprintf(tracef, ", has create_rename_lsn (%lu,0x%lx) more recent than record", - (ulong) LSN_FILE_NO(rec->lsn), - (ulong) LSN_OFFSET(rec->lsn)); + fprintf(tracef, ", has create_rename_lsn (%lu,0x%lx) more recent than" + " record, ignoring", + LSN_IN_HEX(share->state.create_rename_lsn)); error= 0; goto end; } @@ -476,7 +379,7 @@ prototype_exec_hook(REDO_CREATE_TABLE) info= NULL; } /* if does not exist, is older, or its header is corrupted, overwrite it */ - // TODO symlinks + /** @todo symlinks */ ptr= name + strlen(name) + 1; if ((flags= ptr[0] ? HA_DONT_TOUCH_DATA : 0)) fprintf(tracef, ", we will only touch index file"); @@ -500,9 +403,6 @@ prototype_exec_hook(REDO_CREATE_TABLE) ptr+= 2; /* set create_rename_lsn (for maria_read_log to be idempotent) */ lsn_store(ptr + sizeof(info->s->state.header) + 2, rec->lsn); - /* we also set is_of_lsn, like maria_create() does */ - lsn_store(ptr + sizeof(info->s->state.header) + 2 + LSN_STORE_SIZE, - rec->lsn); if (my_pwrite(kfile, ptr, kfile_size_before_extension, 0, MYF(MY_NABP|MY_WME)) || my_chsize(kfile, keystart, 0, MYF(MY_WME))) @@ -581,9 +481,9 @@ prototype_exec_hook(REDO_DROP_TABLE) } if (cmp_translog_addr(share->state.create_rename_lsn, rec->lsn) >= 0) { - fprintf(tracef, ", has create_rename_lsn (%lu,0x%lx) more recent than record", - (ulong) LSN_FILE_NO(rec->lsn), - (ulong) LSN_OFFSET(rec->lsn)); + fprintf(tracef, ", has create_rename_lsn (%lu,0x%lx) more recent than" + " record, ignoring", + LSN_IN_HEX(share->state.create_rename_lsn)); error= 0; goto end; } @@ -622,9 +522,15 @@ prototype_exec_hook(FILE_ID) { uint16 sid; int error= 1; - char *name, *buff; - MARIA_HA *info= NULL; - MARIA_SHARE *share; + const char *name; + MARIA_HA *info; + + if (cmp_translog_addr(rec->lsn, checkpoint_start) < 0) + { + fprintf(tracef, "ignoring because before checkpoint\n"); + return 0; + } + enlarge_buffer(rec); if (log_record_buffer.str == NULL || translog_read_record(rec->lsn, 0, rec->record_length, @@ -634,21 +540,40 @@ prototype_exec_hook(FILE_ID) fprintf(tracef, "Failed to read record\n"); goto end; } - buff= log_record_buffer.str; - sid= fileid_korr(buff); - name= buff + FILEID_STORE_SIZE; - info= all_tables[sid]; + sid= fileid_korr(log_record_buffer.str); + info= all_tables[sid].info; if (info != NULL) { - all_tables[sid]= NULL; - if (close_recovered_table(info)) + fprintf(tracef, " Closing table '%s'\n", info->s->open_file_name); + prepare_table_for_close(info, rec->lsn); + if (maria_close(info)) { fprintf(tracef, "Failed to close table\n"); goto end; } + all_tables[sid].info= NULL; } + name= log_record_buffer.str + FILEID_STORE_SIZE; + if (new_table(sid, name, -1, -1, rec->lsn)) + goto end; + error= 0; +end: + return error; +} + + +static int new_table(uint16 sid, const char *name, + File org_kfile, File org_dfile, LSN lsn) +{ + /* + -1 (skip table): close table and return 0; + 1 (error): close table and return 1; + 0 (success): leave table open and return 0. + */ + int error= 1; + fprintf(tracef, "Table '%s', id %u", name, sid); - info= maria_open(name, O_RDWR, HA_OPEN_FOR_REPAIR); + MARIA_HA *info= maria_open(name, O_RDWR, HA_OPEN_FOR_REPAIR); if (info == NULL) { fprintf(tracef, ", is absent (must have been dropped later?)" @@ -666,7 +591,7 @@ prototype_exec_hook(FILE_ID) execute them, we should not reject the crashed table here. */ } - share= info->s; + MARIA_SHARE *share= info->s; /* check that we're not already using it */ DBUG_ASSERT(share->reopen == 1); DBUG_ASSERT(share->now_transactional == share->base.born_transactional); @@ -674,10 +599,17 @@ prototype_exec_hook(FILE_ID) { fprintf(tracef, ", is not transactional\n"); ALERT_USER(); - error= 0; + error= -1; + goto end; + } + if (cmp_translog_addr(lsn, share->state.create_rename_lsn) <= 0) + { + fprintf(tracef, ", has create_rename_lsn (%lu,0x%lx) more recent than" + " record, ignoring", + LSN_IN_HEX(share->state.create_rename_lsn)); + error= -1; goto end; } - all_tables[sid]= info; /* don't log any records for this work */ _ma_tmp_disable_logging_for_table(share); /* execution of some REDO records relies on data_file_length */ @@ -691,17 +623,25 @@ prototype_exec_hook(FILE_ID) } share->state.state.data_file_length= dfile_len; share->state.state.key_file_length= kfile_len; - if ((dfile_len == 0) || ((dfile_len % share->block_size) > 0)) + if ((dfile_len % share->block_size) > 0) { fprintf(tracef, ", has too short last page\n"); /* Recovery will fix this, no error */ ALERT_USER(); } + all_tables[sid].info= info; + all_tables[sid].org_kfile= org_kfile; + all_tables[sid].org_dfile= org_dfile; fprintf(tracef, ", opened\n"); error= 0; end: - if (error && info != NULL) - error|= maria_close(info); + if (error) + { + if (info != NULL) + maria_close(info); + if (error == -1) + error= 0; + } return error; } @@ -850,17 +790,18 @@ end: } +#define set_undo_lsn_for_active_trans(I, L) do { \ + all_active_trans[I].undo_lsn= L; \ + if (all_active_trans[I].first_undo_lsn == LSN_IMPOSSIBLE) \ + all_active_trans[I].first_undo_lsn= L; } while (0) + prototype_exec_hook(UNDO_ROW_INSERT) { int error= 1; MARIA_HA *info= get_MARIA_HA_from_UNDO_record(rec); if (info == NULL) goto end; - all_active_trans[rec->short_trid].undo_lsn= rec->lsn; - /* - todo: instead of above, call write_hook_for_undo, it will also set - first_undo_lsn - */ + set_undo_lsn_for_active_trans(rec->short_trid, rec->lsn); /* in an upcoming patch ("recovery of the state"), we introduce state.is_of_lsn. For now, we just assume the state is old (true when we @@ -869,6 +810,7 @@ prototype_exec_hook(UNDO_ROW_INSERT) { fprintf(tracef, " state older than record, updating rows' count\n"); info->s->state.state.records++; + /** @todo RECOVERY BUG Also update the table's checksum */ } fprintf(tracef, " rows' count %lu\n", (ulong)info->s->state.state.records); error= 0; @@ -883,11 +825,7 @@ prototype_exec_hook(UNDO_ROW_DELETE) MARIA_HA *info= get_MARIA_HA_from_UNDO_record(rec); if (info == NULL) goto end; - all_active_trans[rec->short_trid].undo_lsn= rec->lsn; - /* - todo: instead of above, call write_hook_for_undo, it will also set - first_undo_lsn - */ + set_undo_lsn_for_active_trans(rec->short_trid, rec->lsn); { fprintf(tracef, " state older than record, updating rows' count\n"); info->s->state.state.records--; @@ -906,11 +844,7 @@ prototype_exec_hook(UNDO_ROW_PURGE) if (info == NULL) goto end; /* this a bit broken, but this log record type will be deleted soon */ - all_active_trans[rec->short_trid].undo_lsn= rec->lsn; - /* - todo: instead of above, call write_hook_for_undo, it will also set - first_undo_lsn - */ + set_undo_lsn_for_active_trans(rec->short_trid, rec->lsn); { fprintf(tracef, " state older than record, updating rows' count\n"); info->s->state.state.records--; @@ -961,77 +895,284 @@ prototype_exec_hook(COMMIT) } -/* Just to inform about any aborted groups or unfinished transactions */ -static int end_of_redo_phase() +static int run_redo_phase(LSN lsn, my_bool apply) +{ + /* install hooks for execution */ +#define install_exec_hook(R) \ + log_record_type_descriptor[LOGREC_ ## R].record_execute_in_redo_phase= \ + exec_LOGREC_ ## R; + install_exec_hook(LONG_TRANSACTION_ID); + install_exec_hook(CHECKPOINT); + install_exec_hook(REDO_CREATE_TABLE); + install_exec_hook(REDO_DROP_TABLE); + install_exec_hook(FILE_ID); + install_exec_hook(REDO_INSERT_ROW_HEAD); + install_exec_hook(REDO_INSERT_ROW_TAIL); + install_exec_hook(REDO_PURGE_ROW_HEAD); + install_exec_hook(REDO_PURGE_ROW_TAIL); + install_exec_hook(REDO_PURGE_BLOCKS); + install_exec_hook(REDO_DELETE_ALL); + install_exec_hook(UNDO_ROW_INSERT); + install_exec_hook(UNDO_ROW_DELETE); + install_exec_hook(UNDO_ROW_PURGE); + install_exec_hook(COMMIT); + + current_group_end_lsn= LSN_IMPOSSIBLE; + + TRANSLOG_HEADER_BUFFER rec; + /* + instead of this block below we will soon use + translog_first_lsn_in_log()... + */ + int len= translog_read_record_header(lsn, &rec); + + /** @todo EOF should be detected */ + if (len == RECHEADER_READ_ERROR) + { + fprintf(tracef, "Cannot find a first record\n"); + return 1; + } + struct st_translog_scanner_data scanner; + if (translog_init_scanner(lsn, 1, &scanner)) + { + fprintf(tracef, "Scanner init failed\n"); + return 1; + } + uint i; + for (i= 1;;i++) + { + uint16 sid= rec.short_trid; + const LOG_DESC *log_desc= &log_record_type_descriptor[rec.type]; + display_record_position(log_desc, &rec, i); + + /* + A complete group is a set of log records with an "end mark" record + (e.g. a set of REDOs for an operation, terminated by an UNDO for this + operation); if there is no "end mark" record the group is incomplete + and won't be executed. + */ + if ((log_desc->record_in_group == LOGREC_IS_GROUP_ITSELF) || + (log_desc->record_in_group == LOGREC_LAST_IN_GROUP)) + { + if (all_active_trans[sid].group_start_lsn != LSN_IMPOSSIBLE) + { + if (log_desc->record_in_group == LOGREC_IS_GROUP_ITSELF) + { + /* + can happen if the transaction got a table write error, then + unlocked tables thus wrote a COMMIT record. + */ + fprintf(tracef, "\nDiscarding unfinished group before this record\n"); + ALERT_USER(); + all_active_trans[sid].group_start_lsn= LSN_IMPOSSIBLE; + } + else + { + /* + There is a complete group for this transaction, containing more + than this event. + */ + fprintf(tracef, " ends a group:\n"); + struct st_translog_scanner_data scanner2; + TRANSLOG_HEADER_BUFFER rec2; + len= + translog_read_record_header(all_active_trans[sid].group_start_lsn, &rec2); + if (len < 0) /* EOF or error */ + { + fprintf(tracef, "Cannot find record where it should be\n"); + return 1; + } + if (translog_init_scanner(rec2.lsn, 1, &scanner2)) + { + fprintf(tracef, "Scanner2 init failed\n"); + return 1; + } + current_group_end_lsn= rec.lsn; + do + { + if (rec2.short_trid == sid) /* it's in our group */ + { + const LOG_DESC *log_desc2= &log_record_type_descriptor[rec2.type]; + display_record_position(log_desc2, &rec2, 0); + if (apply && display_and_apply_record(log_desc2, &rec2)) + return 1; + } + len= translog_read_next_record_header(&scanner2, &rec2); + if (len < 0) /* EOF or error */ + { + fprintf(tracef, "Cannot find record where it should be\n"); + return 1; + } + } + while (rec2.lsn < rec.lsn); + translog_free_record_header(&rec2); + /* group finished */ + all_active_trans[sid].group_start_lsn= LSN_IMPOSSIBLE; + current_group_end_lsn= LSN_IMPOSSIBLE; /* for debugging */ + display_record_position(log_desc, &rec, 0); + } + } + if (apply && display_and_apply_record(log_desc, &rec)) + return 1; + } + else /* record does not end group */ + { + /* just record the fact, can't know if can execute yet */ + if (all_active_trans[sid].group_start_lsn == LSN_IMPOSSIBLE) + { + /* group not yet started */ + all_active_trans[sid].group_start_lsn= rec.lsn; + } + } + len= translog_read_next_record_header(&scanner, &rec); + if (len < 0) + { + switch (len) + { + case RECHEADER_READ_EOF: + fprintf(tracef, "EOF on the log\n"); + break; + case RECHEADER_READ_ERROR: + fprintf(stderr, "Error reading log\n"); + return 1; + } + break; + } + } + translog_free_record_header(&rec); + return 0; +} + + +/** + @brief Informs about any aborted groups or unfinished transactions, + prepares for the UNDO phase if needed. + + @param prepare_for_undo_phase + + @note Observe that it may init trnman. +*/ +static uint end_of_redo_phase(my_bool prepare_for_undo_phase) { - uint sid, unfinished= 0, error= 0; + uint sid, unfinished= 0; + + hash_free(&all_dirty_pages); + /* + hash_free() can be called multiple times probably, but be safe it that + changes + */ + bzero(&all_dirty_pages, sizeof(all_dirty_pages)); + my_free(dirty_pages_pool, MYF(MY_ALLOW_ZERO_PTR)); + dirty_pages_pool= NULL; + + if (prepare_for_undo_phase && trnman_init()) + return -1; + for (sid= 0; sid <= SHORT_TRID_MAX; sid++) { TrID long_trid= all_active_trans[sid].long_trid; LSN gslsn= all_active_trans[sid].group_start_lsn; + TRN *trn; + if (gslsn != LSN_IMPOSSIBLE) + { + fprintf(tracef, "Group at LSN (%lu,0x%lx) short_trid %u aborted\n", + (ulong) LSN_FILE_NO(gslsn), (ulong) LSN_OFFSET(gslsn), sid); + ALERT_USER(); + } if (all_active_trans[sid].undo_lsn != LSN_IMPOSSIBLE) { char llbuf[22]; llstr(long_trid, llbuf); fprintf(tracef, "Transaction long_trid %s short_trid %u unfinished\n", llbuf, sid); + /* dummy_transaction_object serves only for DDLs */ + DBUG_ASSERT(long_trid != 0); + if (prepare_for_undo_phase) + { + if ((trn= trnman_recreate_trn_from_recovery(sid, long_trid)) == NULL) + return -1; + trn->undo_lsn= all_active_trans[sid].undo_lsn; + } + /* otherwise we will just warn about it */ unfinished++; } - if (gslsn != LSN_IMPOSSIBLE) - { - fprintf(tracef, "Group at LSN (%lu,0x%lx) short_trid %u aborted\n", - (ulong) LSN_FILE_NO(gslsn), (ulong) LSN_OFFSET(gslsn), sid); - ALERT_USER(); - } - /* If real recovery: roll back unfinished transaction */ #ifdef MARIA_VERSIONING /* - If real recovery: transaction was committed, move it to some separate - list for soon purging. Create TRNs. + If real recovery: if transaction was committed, move it to some separate + list for soon purging. */ #endif } - /* - We don't close tables if there are some unfinished transactions, because - closing tables normally requires that all unfinished transactions on them - be rolled back. Unfinished transactions are symptom of a crash, we - reproduce the crash. - For example, closing will soon write the state to disk and when doing that - it will think this is a committed state, but it may not be. + + my_free(all_active_trans, MYF(MY_ALLOW_ZERO_PTR)); + all_active_trans= NULL; + + /* + The UNDO phase uses some normal run-time code of ROLLBACK: generates log + records, etc; prepare tables for that */ - if (unfinished > 0) - fprintf(tracef, "WARNING: %u unfinished transactions; some tables may be" - " left inconsistent!\n", unfinished); + LSN addr= translog_get_horizon(); for (sid= 0; sid <= SHARE_ID_MAX; sid++) { - MARIA_HA *info= all_tables[sid]; + MARIA_HA *info= all_tables[sid].info; if (info != NULL) { - /* if error, still close other tables */ - error|= close_recovered_table(info); + prepare_table_for_close(info, addr); + /* + But we don't close it; we leave it available for the UNDO phase; + it's likely that the UNDO phase will need it. + */ + if (prepare_for_undo_phase) + translog_assign_id_to_share_from_recovery(info->s, sid); } } - return error; + + /* we don't need all_tables anymore, maria_open_list is enough */ + my_free(all_tables, MYF(MY_ALLOW_ZERO_PTR)); + all_tables= NULL; + + /* + We could take a checkpoint here, in case of a crash during the UNDO + phase. The drawback is that a page which got a REDO (thus, flushed + by this would-be checkpoint) is likely to have an UNDO executed on it + soon. And so, the flush was probably lost time. + So for now we prefer to do recovery with maximum speed and take a + checkpoint only at the end of the UNDO phase. + */ + + return unfinished; } -static int close_recovered_table(MARIA_HA *info) +static int run_undo_phase(uint unfinished) +{ + if (unfinished > 0) + { + fprintf(tracef, "%u transactions will be rolled back\n", unfinished); + for( ; unfinished-- ; ) + { + char llbuf[22]; + TRN *trn= trnman_get_any_trn(); + DBUG_ASSERT(trn != NULL); + llstr(trn->trid, llbuf); + fprintf(tracef, "Rolling back transaction of long id %s\n", llbuf); + /* of course we miss execution of UNDOs here */ + if (trnman_rollback_trn(trn)) + return 1; + /* We could want to span a few threads (4?) instead of 1 */ + /* In the future, we want to have this phase *online* */ + } + } + return 0; +} + + +static void prepare_table_for_close(MARIA_HA *info, + LSN at_lsn __attribute__ ((unused))) { - int error; MARIA_SHARE *share= info->s; - fprintf(tracef, " Closing table '%s'\n", share->open_file_name); + /* we will soon use at_lsn here */ _ma_reenable_logging_for_table(share); - /* - Recovery normally corrected problems, don't scare user with "table was not - closed properly" in CHECK TABLE and don't automatically check table at - next open (when we have --maria-recover). - */ - share->state.open_count= share->global_changed ? 1 : 0; - /* this var is set only by non-recovery operations (mi_write() etc) */ - DBUG_ASSERT(!share->global_changed); - if ((error= maria_close(info))) - fprintf(tracef, "Failed to close table\n"); - return error; } @@ -1039,16 +1180,22 @@ static MARIA_HA *get_MARIA_HA_from_REDO_record(const TRANSLOG_HEADER_BUFFER *rec) { uint16 sid; - ulonglong page; + pgcache_page_no_t page; MARIA_HA *info; char llbuf[22]; sid= fileid_korr(rec->header); page= page_korr(rec->header + FILEID_STORE_SIZE); - /* BUG not correct for REDO_PURGE_BLOCKS, page is not at this pos */ + /** + @todo RECOVERY BUG + - for REDO_PURGE_BLOCKS, page is not at this pos + - for DELETE_ALL, record ends here! buffer overrun! + Solution: caller should pass a param enum { i_am_about_data_file, + i_am_about_index_file, none }. + */ llstr(page, llbuf); fprintf(tracef, " For page %s of table of short id %u", llbuf, sid); - info= all_tables[sid]; + info= all_tables[sid].info; if (info == NULL) { fprintf(tracef, ", table skipped, so skipping record\n"); @@ -1057,23 +1204,38 @@ static MARIA_HA *get_MARIA_HA_from_REDO_record(const fprintf(tracef, ", '%s'", info->s->open_file_name); /* detect if an open instance of a dropped table (internal bug) */ DBUG_ASSERT(info->s->last_version != 0); - if (cmp_translog_addr(info->s->state.create_rename_lsn, rec->lsn) >= 0) + if (cmp_translog_addr(rec->lsn, checkpoint_start) < 0) { - fprintf(tracef, ", has create_rename_lsn (%lu,0x%lx) more recent than log" - " record\n", - (ulong) LSN_FILE_NO(rec->lsn), (ulong) LSN_OFFSET(rec->lsn)); - return NULL; + /** + @todo RECOVERY BUG always assuming this is REDO for data file, but it + could soon be index file + */ + uint64 file_and_page_id= + (((uint64)all_tables[sid].org_dfile) << 32) | page; + struct st_dirty_page *dirty_page= (struct st_dirty_page *) + hash_search(&all_dirty_pages, + (uchar *)&file_and_page_id, sizeof(file_and_page_id)); + if ((dirty_page == NULL) || + cmp_translog_addr(rec->lsn, dirty_page->rec_lsn) < 0) + { + fprintf(tracef, ", ignoring because of dirty_pages list\n"); + return NULL; + } } - fprintf(tracef, ", applying record\n"); - return info; + /* - Soon we will also skip the page depending on the rec_lsn for this page in - the checkpoint record, but this is not absolutely needed for now (just - assume we have made no checkpoint). Btw rec_lsn and bitmap's recovery is a - an unsolved problem (rec_lsn is to ignore a REDO without reading the data - page and to do so we need to be sure the corresponding bitmap page does - not need a _ma_bitmap_set()). + So we are going to read the page, and if its LSN is older than the + record's we will modify the page */ + fprintf(tracef, ", applying record\n"); + /* A future CHECK/OPTIMIZE/REPAIR should not be fooled: */ + /** + @todo but the ones about keys should be set only if REDO for keys. Same + in ..._from_UNDO_record + */ + info->s->state.changed|= STATE_CHANGED | STATE_NOT_ANALYZED | + STATE_NOT_OPTIMIZED_KEYS | STATE_NOT_SORTED_PAGES; + return info; } @@ -1085,7 +1247,7 @@ static MARIA_HA *get_MARIA_HA_from_UNDO_record(const sid= fileid_korr(rec->header + LSN_STORE_SIZE); fprintf(tracef, " For table of short id %u", sid); - info= all_tables[sid]; + info= all_tables[sid].info; if (info == NULL) { fprintf(tracef, ", table skipped, so skipping record\n"); @@ -1093,24 +1255,180 @@ static MARIA_HA *get_MARIA_HA_from_UNDO_record(const } fprintf(tracef, ", '%s'", info->s->open_file_name); DBUG_ASSERT(info->s->last_version != 0); - if (cmp_translog_addr(info->s->state.create_rename_lsn, rec->lsn) >= 0) - { - fprintf(tracef, ", has create_rename_lsn (%lu,0x%lx) more recent than log" - " record\n", - (ulong) LSN_FILE_NO(rec->lsn), (ulong) LSN_OFFSET(rec->lsn)); - return NULL; - } fprintf(tracef, ", applying record\n"); + /* execution of UNDOs may increment the records' count: */ + info->s->state.changed|= STATE_CHANGED | STATE_NOT_ANALYZED | + STATE_NOT_OPTIMIZED_KEYS | STATE_NOT_SORTED_PAGES; return info; +} + + +static int parse_checkpoint_record(LSN lsn) +{ + uint i; + TRANSLOG_HEADER_BUFFER rec; + + fprintf(tracef, "Loading data from checkpoint record\n"); + int len= translog_read_record_header(lsn, &rec); + + /** @todo EOF should be detected */ + if (len == RECHEADER_READ_ERROR) + { + fprintf(tracef, "Cannot find checkpoint record where it should be\n"); + return 1; + } + + enlarge_buffer(&rec); + if (log_record_buffer.str == NULL || + translog_read_record(rec.lsn, 0, rec.record_length, + log_record_buffer.str, NULL) != + rec.record_length) + { + fprintf(tracef, "Failed to read record\n"); + return 1; + } + + char *ptr= log_record_buffer.str; + checkpoint_start= lsn_korr(ptr); + ptr+= LSN_STORE_SIZE; + + /* transactions */ + uint nb_active_transactions= uint2korr(ptr); + ptr+= 2; + fprintf(tracef, "%u active transactions\n", nb_active_transactions); + LSN minimum_rec_lsn_of_active_transactions= lsn_korr(ptr); + ptr+= LSN_STORE_SIZE; + + /* + how much brain juice and discussions there was to come to writing this + line + */ + set_if_smaller(checkpoint_start, minimum_rec_lsn_of_active_transactions); + + for (i= 0; i < nb_active_transactions; i++) + { + uint16 sid= uint2korr(ptr); + ptr+= 2; + TrID long_id= uint6korr(ptr); + ptr+= 6; + DBUG_ASSERT(sid > 0 && long_id > 0); + LSN undo_lsn= lsn_korr(ptr); + ptr+= LSN_STORE_SIZE; + LSN first_undo_lsn= lsn_korr(ptr); + ptr+= LSN_STORE_SIZE; + new_transaction(sid, long_id, undo_lsn, first_undo_lsn); + } + uint nb_committed_transactions= uint4korr(ptr); + ptr+= 4; + fprintf(tracef, "%lu committed transactions\n", + (ulong)nb_committed_transactions); + /* no purging => committed transactions are not important */ + ptr+= (6 + LSN_STORE_SIZE) * nb_committed_transactions; + + /* tables */ + uint nb_tables= uint4korr(ptr); + fprintf(tracef, "%u open tables\n", nb_tables); + for (i= 0; i< nb_tables; i++) + { + char name[FN_REFLEN]; + uint16 sid= uint2korr(ptr); + ptr+= 2; + DBUG_ASSERT(sid > 0); + File kfile= uint4korr(ptr); + ptr+= 4; + File dfile= uint4korr(ptr); + ptr+= 4; + LSN first_log_write_lsn= lsn_korr(ptr); + ptr+= LSN_STORE_SIZE; + uint name_len= strlen(ptr) + 1; + ptr+= name_len; + strnmov(name, ptr, sizeof(name)); + if (new_table(sid, name, kfile, dfile, first_log_write_lsn)) + return 1; + } + + /* dirty pages */ + uint nb_dirty_pages= uint4korr(ptr); + ptr+= 4; + if (hash_init(&all_dirty_pages, &my_charset_bin, nb_dirty_pages, + offsetof(struct st_dirty_page, file_and_page_id), + sizeof(((struct st_dirty_page *)NULL)->file_and_page_id), + NULL, NULL, 0)) + return 1; + dirty_pages_pool= + (struct st_dirty_page *)my_malloc(nb_dirty_pages * + sizeof(struct st_dirty_page), + MYF(MY_WME)); + if (unlikely(dirty_pages_pool == NULL)) + return 1; + struct st_dirty_page *next_dirty_page_in_pool= dirty_pages_pool; + LSN minimum_rec_lsn_of_dirty_pages= LSN_MAX; + for (i= 0; i < nb_dirty_pages ; i++) + { + File fileid= uint4korr(ptr); + ptr+= 4; + pgcache_page_no_t pageid= uint4korr(ptr); + ptr+= 4; + LSN rec_lsn= lsn_korr(ptr); + ptr+= LSN_STORE_SIZE; + if (new_page(fileid, pageid, rec_lsn, next_dirty_page_in_pool++)) + return 1; + set_if_smaller(minimum_rec_lsn_of_dirty_pages, rec_lsn); + } + /* after that, there will be no insert/delete into the hash */ /* - Soon we will also skip the page depending on the rec_lsn for this page in - the checkpoint record, but this is not absolutely needed for now (just - assume we have made no checkpoint). + sanity check on record (did we screw up with all those "ptr+=", did the + checkpoint write code and checkpoint read code go out of sync?). + */ + /** + @todo This probably presently and hopefully detects that + first_log_write_lsn is not written by the checkpoint record; we need + to add MARIA_SHARE::first_log_write_lsn, fill it with a inwrite-hook of + LOGREC_FILE_ID (note that when we write this record we hold intern_lock, + so Checkpoint will read the LSN correctly), and store it in the + checkpoint record. */ + if (ptr != (log_record_buffer.str + log_record_buffer.length)) + { + fprintf(tracef, "checkpoint record corrupted\n"); + return 1; + } + set_if_smaller(checkpoint_start, minimum_rec_lsn_of_dirty_pages); + + return 0; } +static int new_page(File fileid, pgcache_page_no_t pageid, LSN rec_lsn, + struct st_dirty_page *dirty_page) +{ + /* serves as hash key */ + dirty_page->file_and_page_id= (((uint64)fileid) << 32) | pageid; + dirty_page->rec_lsn= rec_lsn; + return my_hash_insert(&all_dirty_pages, (uchar *)dirty_page); +} +static int close_all_tables() +{ + int error= 0; + LIST *list_element, *next_open; + MARIA_HA *info; + pthread_mutex_lock(&THR_LOCK_maria); + if (maria_open_list == NULL) + goto end; + fprintf(tracef, "Closing all tables\n"); + for (list_element= maria_open_list ; list_element ; list_element= next_open) + { + next_open= list_element->next; + info= (MARIA_HA*)list_element->data; + pthread_mutex_unlock(&THR_LOCK_maria); /* ok, UNDO phase not online yet */ + error|= maria_close(info); + pthread_mutex_lock(&THR_LOCK_maria); + } +end: + pthread_mutex_unlock(&THR_LOCK_maria); + return error; +} /* some comments and pseudo-code which we keep for later */ #if 0 diff --git a/storage/maria/ma_recovery.h b/storage/maria/ma_recovery.h index 0b576efc95f..9a5a2b3099e 100644 --- a/storage/maria/ma_recovery.h +++ b/storage/maria/ma_recovery.h @@ -25,5 +25,6 @@ C_MODE_START int maria_recover(); -int maria_apply_log(LSN lsn, my_bool applyn, FILE *trace_file); +int maria_apply_log(LSN lsn, my_bool apply, FILE *trace_file, + my_bool execute_undo_phase); C_MODE_END diff --git a/storage/maria/maria_chk.c b/storage/maria/maria_chk.c index 58953caa57c..e97298b9a2e 100644 --- a/storage/maria/maria_chk.c +++ b/storage/maria/maria_chk.c @@ -1034,7 +1034,7 @@ static int maria_chk(HA_CHECK *param, char *filename) that it will have to find and store it. */ if (share->base.born_transactional) - share->state.create_rename_lsn= (LSN)ULONGLONG_MAX; + share->state.create_rename_lsn= LSN_REPAIRED_BY_MARIA_CHK; if ((param->testflag & (T_REP_BY_SORT | T_REP_PARALLEL)) && (maria_is_any_key_active(share->state.key_map) || (rep_quick && !param->keys_in_use && !recreate)) && diff --git a/storage/maria/maria_read_log.c b/storage/maria/maria_read_log.c index c594fe20490..15f61b53f60 100644 --- a/storage/maria/maria_read_log.c +++ b/storage/maria/maria_read_log.c @@ -51,7 +51,7 @@ int main(int argc, char **argv) goto err; } /* we don't want to create a control file, it MUST exist */ - if (ma_control_file_create_or_open(FALSE)) + if (ma_control_file_create_or_open()) { fprintf(stderr, "Can't open control file (%d)\n", errno); goto err; @@ -88,7 +88,8 @@ int main(int argc, char **argv) lsn= first_lsn_in_log(); /* LSN could be also --start-from-lsn=# */ fprintf(stdout, "TRACE of the last maria_read_log\n"); - if (maria_apply_log(lsn, opt_display_and_apply, stdout)) + /* Until we have UNDO records, no UNDO phase */ + if (maria_apply_log(lsn, opt_display_and_apply, stdout, FALSE)) goto err; fprintf(stdout, "%s: SUCCESS\n", my_progname); diff --git a/storage/maria/trnman.c b/storage/maria/trnman.c index 177ee2a7a70..b0550085863 100644 --- a/storage/maria/trnman.c +++ b/storage/maria/trnman.c @@ -18,6 +18,7 @@ #include #include #include "trnman.h" +#include "ma_control_file.h" /* status variables: @@ -708,3 +709,29 @@ end: pthread_mutex_unlock(&LOCK_trn_list); DBUG_RETURN(error); } + + +TRN *trnman_recreate_trn_from_recovery(uint16 shortid, TrID longid) +{ + TrID old_trid_generator= global_trid_generator; + TRN *trn; + DBUG_ASSERT(maria_in_recovery && !maria_multi_threaded); + if (unlikely((trn= trnman_new_trn(NULL, NULL, NULL)) == NULL)) + return NULL; + /* deallocate excessive allocations of trnman_new_trn() */ + global_trid_generator= old_trid_generator; + set_if_bigger(global_trid_generator, longid); + short_trid_to_active_trn[trn->short_id]= 0; + DBUG_ASSERT(short_trid_to_active_trn[shortid] == NULL); + short_trid_to_active_trn[shortid]= trn; + trn->trid= longid; + trn->short_id= shortid; + return trn; +} + + +TRN *trnman_get_any_trn() +{ + TRN *trn= active_list_min.next; + return (trn != &active_list_max) ? trn : NULL; +} diff --git a/storage/maria/trnman_public.h b/storage/maria/trnman_public.h index e1891466c4d..10dcb479530 100644 --- a/storage/maria/trnman_public.h +++ b/storage/maria/trnman_public.h @@ -53,6 +53,8 @@ uint trnman_increment_locked_tables(TRN *trn); uint trnman_decrement_locked_tables(TRN *trn); my_bool trnman_has_locked_tables(TRN *trn); void trnman_reset_locked_tables(TRN *trn); +TRN *trnman_recreate_trn_from_recovery(uint16 shortid, TrID longid); +TRN *trnman_get_any_trn(); C_MODE_END #endif -- cgit v1.2.1 From 4201d2784945da87c4274eae1a5ea1b79f34ab12 Mon Sep 17 00:00:00 2001 From: unknown Date: Wed, 29 Aug 2007 17:28:44 +0200 Subject: cleanups storage/maria/ma_commit.c: theoretically unneeded, and could cause problems (when trnman_commit_trn() ends the TRN may have been recycled and be in use by another thread already, we cannot touch it). storage/maria/maria_def.h: just include the existing file --- storage/maria/ma_commit.c | 1 - storage/maria/maria_def.h | 2 +- 2 files changed, 1 insertion(+), 2 deletions(-) diff --git a/storage/maria/ma_commit.c b/storage/maria/ma_commit.c index c8c37ae67db..36ea2f6e6e4 100644 --- a/storage/maria/ma_commit.c +++ b/storage/maria/ma_commit.c @@ -67,7 +67,6 @@ int ma_commit(TRN *trn) log_array, NULL) || translog_flush(commit_lsn) || trnman_commit_trn(trn)); - trn->undo_lsn= 0; /* Note: if trnman_commit_trn() fails above, we have already written the COMMIT record, so Checkpoint and Recovery will see the diff --git a/storage/maria/maria_def.h b/storage/maria/maria_def.h index bf8d7a5971a..ab2546b72f3 100644 --- a/storage/maria/maria_def.h +++ b/storage/maria/maria_def.h @@ -894,7 +894,7 @@ void _ma_restore_status(void *param); void _ma_copy_status(void *to, void *from); my_bool _ma_check_status(void *param); void _ma_reset_status(MARIA_HA *maria); -int ma_commit(struct st_transaction *trn); +#include "ma_commit.h" extern MARIA_HA *_ma_test_if_reopen(char *filename); my_bool _ma_check_table_is_closed(const char *name, const char *where); -- cgit v1.2.1 From 90b63bf754f510519c3f66376a6eb849963a434c Mon Sep 17 00:00:00 2001 From: unknown Date: Wed, 29 Aug 2007 22:02:30 +0200 Subject: WL#3072 Maria recovery manual merge of ma_recovery.c (too big conflict to resolve in fmtool); the merged Monty's code allows correct replaying of REDO_PURGE_BLOCKS and was originally in monty@mysql.com/narttu.mysql.fi|ChangeSet|20070829060310|44058 storage/maria/ma_recovery.c: * manually merging Monty's and Sanja's changes of the two last weeks to my massively modified version of this file. The merged Monty's code allows correct replaying of REDO_PURGE_BLOCKS and was originally in monty@mysql.com/narttu.mysql.fi|ChangeSet|20070829060310|44058 . * Setting the state to "STATE_CHANGED|etc" in Recovery is more logically done when we update the state in memory (for example records++). --- storage/maria/ma_recovery.c | 67 ++++++++++++++++++++++++++++++++++++--------- 1 file changed, 54 insertions(+), 13 deletions(-) diff --git a/storage/maria/ma_recovery.c b/storage/maria/ma_recovery.c index c6bb6306771..b45346725e6 100644 --- a/storage/maria/ma_recovery.c +++ b/storage/maria/ma_recovery.c @@ -67,6 +67,7 @@ prototype_exec_hook(REDO_PURGE_BLOCKS); prototype_exec_hook(REDO_DELETE_ALL); prototype_exec_hook(UNDO_ROW_INSERT); prototype_exec_hook(UNDO_ROW_DELETE); +prototype_exec_hook(UNDO_ROW_UPDATE); prototype_exec_hook(UNDO_ROW_PURGE); prototype_exec_hook(COMMIT); static int run_redo_phase(LSN lsn, my_bool apply); @@ -176,7 +177,16 @@ int maria_apply_log(LSN from_lsn, my_bool apply, FILE *trace_file, if (from_lsn == LSN_IMPOSSIBLE) { if (last_checkpoint_lsn == LSN_IMPOSSIBLE) - from_lsn= first_lsn_in_log(); + { + from_lsn= translog_first_theoretical_lsn(); + /* + as far as we have not yet any checkpoint then the very first + log file should be present. + */ + if (unlikely((from_lsn == LSN_IMPOSSIBLE) || + (from_lsn == LSN_ERROR))) + goto err; + } else { DBUG_ASSERT(0); /* not yet implemented */ @@ -694,7 +704,7 @@ end: prototype_exec_hook(REDO_INSERT_ROW_TAIL) { int error= 1; - uchar *buff= NULL; + uchar *buff; MARIA_HA *info= get_MARIA_HA_from_REDO_record(rec); if (info == NULL) goto end; @@ -762,11 +772,24 @@ end: prototype_exec_hook(REDO_PURGE_BLOCKS) { int error= 1; + uchar *buff; MARIA_HA *info= get_MARIA_HA_from_REDO_record(rec); if (info == NULL) goto end; + enlarge_buffer(rec); + + if (log_record_buffer.str == NULL || + translog_read_record(rec->lsn, 0, rec->record_length, + log_record_buffer.str, NULL) != + rec->record_length) + { + fprintf(tracef, "Failed to read record\n"); + goto end; + } + + buff= log_record_buffer.str; if (_ma_apply_redo_purge_blocks(info, current_group_end_lsn, - rec->header + FILEID_STORE_SIZE)) + buff + FILEID_STORE_SIZE)) goto end; error= 0; end: @@ -811,6 +834,12 @@ prototype_exec_hook(UNDO_ROW_INSERT) fprintf(tracef, " state older than record, updating rows' count\n"); info->s->state.state.records++; /** @todo RECOVERY BUG Also update the table's checksum */ + /** + @todo some bits below will rather be set when executing UNDOs related + to keys + */ + info->s->state.changed|= STATE_CHANGED | STATE_NOT_ANALYZED | + STATE_NOT_OPTIMIZED_KEYS | STATE_NOT_SORTED_PAGES; } fprintf(tracef, " rows' count %lu\n", (ulong)info->s->state.state.records); error= 0; @@ -829,6 +858,8 @@ prototype_exec_hook(UNDO_ROW_DELETE) { fprintf(tracef, " state older than record, updating rows' count\n"); info->s->state.state.records--; + info->s->state.changed|= STATE_CHANGED | STATE_NOT_ANALYZED | + STATE_NOT_OPTIMIZED_KEYS | STATE_NOT_SORTED_PAGES; } fprintf(tracef, " rows' count %lu\n", (ulong)info->s->state.state.records); error= 0; @@ -837,6 +868,23 @@ end: } +prototype_exec_hook(UNDO_ROW_UPDATE) +{ + int error= 1; + MARIA_HA *info= get_MARIA_HA_from_UNDO_record(rec); + if (info == NULL) + goto end; + set_undo_lsn_for_active_trans(rec->short_trid, rec->lsn); + { + info->s->state.changed|= STATE_CHANGED | STATE_NOT_ANALYZED | + STATE_NOT_OPTIMIZED_KEYS | STATE_NOT_SORTED_PAGES; + } + error= 0; +end: + return error; +} + + prototype_exec_hook(UNDO_ROW_PURGE) { int error= 1; @@ -848,6 +896,8 @@ prototype_exec_hook(UNDO_ROW_PURGE) { fprintf(tracef, " state older than record, updating rows' count\n"); info->s->state.state.records--; + info->s->state.changed|= STATE_CHANGED | STATE_NOT_ANALYZED | + STATE_NOT_OPTIMIZED_KEYS | STATE_NOT_SORTED_PAGES; } fprintf(tracef, " rows' count %lu\n", (ulong)info->s->state.state.records); error= 0; @@ -914,6 +964,7 @@ static int run_redo_phase(LSN lsn, my_bool apply) install_exec_hook(REDO_DELETE_ALL); install_exec_hook(UNDO_ROW_INSERT); install_exec_hook(UNDO_ROW_DELETE); + install_exec_hook(UNDO_ROW_UPDATE); install_exec_hook(UNDO_ROW_PURGE); install_exec_hook(COMMIT); @@ -1228,13 +1279,6 @@ static MARIA_HA *get_MARIA_HA_from_REDO_record(const record's we will modify the page */ fprintf(tracef, ", applying record\n"); - /* A future CHECK/OPTIMIZE/REPAIR should not be fooled: */ - /** - @todo but the ones about keys should be set only if REDO for keys. Same - in ..._from_UNDO_record - */ - info->s->state.changed|= STATE_CHANGED | STATE_NOT_ANALYZED | - STATE_NOT_OPTIMIZED_KEYS | STATE_NOT_SORTED_PAGES; return info; } @@ -1256,9 +1300,6 @@ static MARIA_HA *get_MARIA_HA_from_UNDO_record(const fprintf(tracef, ", '%s'", info->s->open_file_name); DBUG_ASSERT(info->s->last_version != 0); fprintf(tracef, ", applying record\n"); - /* execution of UNDOs may increment the records' count: */ - info->s->state.changed|= STATE_CHANGED | STATE_NOT_ANALYZED | - STATE_NOT_OPTIMIZED_KEYS | STATE_NOT_SORTED_PAGES; return info; } -- cgit v1.2.1 From 4b3029662af1b8cc0d7fbc26d01070c6e2de82de Mon Sep 17 00:00:00 2001 From: unknown Date: Fri, 31 Aug 2007 09:22:15 +0300 Subject: Fixed bug in log "in progress" marking. storage/maria/ma_loghandler.c: Comments fixed. Fixed loop starting value. --- storage/maria/ma_loghandler.c | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/storage/maria/ma_loghandler.c b/storage/maria/ma_loghandler.c index 273ed8b55d0..71db026b0a5 100644 --- a/storage/maria/ma_loghandler.c +++ b/storage/maria/ma_loghandler.c @@ -159,7 +159,7 @@ struct st_translog_descriptor LSN flushed; /* Last LSN sent to the disk (but maybe not written yet) */ LSN sent_to_file; - /* All what is after this addess is not sent to disk yet */ + /* All what is after this address is not sent to disk yet */ TRANSLOG_ADDRESS in_buffers_only; pthread_mutex_t sent_to_file_lock; @@ -671,7 +671,7 @@ static my_bool translog_max_lsn_to_header(File file, LSN lsn) typedef struct st_loghandler_file_info { /* - LSN_IPOSSIBLE for current file and max LSN which parts stored in the + LSN_IMPOSSIBLE for current file and max LSN which parts stored in the file for all other (finished) files. */ LSN max_lsn; @@ -813,7 +813,7 @@ static void translog_mark_file_unfinished(ulong file) goto end; } - for (place= log_descriptor.unfinished_files.elements; + for (place= log_descriptor.unfinished_files.elements - 1; place >= 0; place--) { -- cgit v1.2.1 From 2cc2f3e75793246a7f1eb582c9931b2d2e5a9162 Mon Sep 17 00:00:00 2001 From: unknown Date: Fri, 31 Aug 2007 10:19:54 +0300 Subject: Generalized the way update and redo extends the size of a directory record. storage/maria/ma_blockrec.c: Generalized the way update and redo extends the size of a directory record. This will (for now) ensure that data files are idenitical after normal run and after a apply-log run. storage/maria/ma_open.c: Disabled reservation of transid on rows (for now) as these are not yet used. (I had to disable this as otherwise update thougth rows had grown in size when they hadn't and we had thus different row sizes on update and redo, which caused different block information) storage/maria/ma_test1.c: Added comment storage/maria/ma_test2.c: Do commit on error/abort storage/maria/ma_test_all.sh: Some more testing (to cover a bug that was not found in previous runs) storage/maria/ma_test_recovery: More tests --- storage/maria/ma_blockrec.c | 244 +++++++++++++++++++++++++++-------------- storage/maria/ma_open.c | 2 + storage/maria/ma_test1.c | 2 +- storage/maria/ma_test2.c | 8 ++ storage/maria/ma_test_all.sh | 4 + storage/maria/ma_test_recovery | 6 +- 6 files changed, 182 insertions(+), 84 deletions(-) diff --git a/storage/maria/ma_blockrec.c b/storage/maria/ma_blockrec.c index d8f65c7b367..6e8495bac23 100644 --- a/storage/maria/ma_blockrec.c +++ b/storage/maria/ma_blockrec.c @@ -504,11 +504,12 @@ void _ma_end_block_record(MARIA_HA *info) ****************************************************************************/ /* - Return the next used uchar on the page after a directory entry. + Return the next unused postion on the page after a directory entry. SYNOPSIS start_of_next_entry() - dir Directory entry to be used + dir Directory entry to be used. This can not be the + the last entry on the page! RETURN # Position in page where next entry starts. @@ -530,6 +531,20 @@ static inline uint start_of_next_entry(uchar *dir) } +/* + Return the offset where the previous entry ends (before on page) + + SYNOPSIS + end_of_previous_entry() + dir Address for current directory entry + end Address to last directory entry + + RETURN + # Position where previous entry ends (smallest address on page) + Everything between # and current entry are free to be used. +*/ + + static inline uint end_of_previous_entry(uchar *dir, uchar *end) { uchar *pos; @@ -537,14 +552,108 @@ static inline uint end_of_previous_entry(uchar *dir, uchar *end) { uint offset; if ((offset= uint2korr(pos))) - { return offset + uint2korr(pos+2); - } } return PAGE_HEADER_SIZE; } +/** + @brief Extend a record area to fit a given size block + + @fn extend_area_on_page() + @param buff Page buffer + @param dir Pointer to dir entry in buffer + @param rownr Row number we working on + @param block_size Block size of buffer + @param request_length How much data we want to put at [dir] + @param empty_space Total empty space in buffer + + IMPLEMENTATION + The logic is as follows (same as in _ma_update_block_record()) + - If new data fits in old block, use old block. + - Extend block with empty space before block. If enough, use it. + - Extend block with empty space after block. If enough, use it. + - Use compact_page() to get all empty space at dir. + + RETURN + @retval 0 ok + @retval ret_offset Pointer to store offset to found area + @retval ret_length Pointer to store length of found area + @retval [dir] rec_offset is store here too + + @retval 1 error (wrong info in block) +*/ + +static my_bool extend_area_on_page(uchar *buff, uchar *dir, + uint rownr, uint block_size, + uint request_length, + uint *empty_space, uint *ret_offset, + uint *ret_length) +{ + uint rec_offset, length; + DBUG_ENTER("extend_area_on_page"); + + rec_offset= uint2korr(dir); + length= uint2korr(dir + 2); + DBUG_PRINT("enter", ("rec_offset: %u length: %u request_length: %u", + rec_offset, length, request_length)); + + *empty_space+= length; + if (length < request_length) + { + uint max_entry= (uint) ((uchar*) buff)[DIR_COUNT_OFFSET]; + uint old_rec_offset; + /* + New data did not fit in old position. + Find first possible position where to put new data. + */ + old_rec_offset= rec_offset; + rec_offset= end_of_previous_entry(dir, buff + block_size - + PAGE_SUFFIX_SIZE); + length+= (uint) (old_rec_offset - rec_offset); + /* + old_rec_offset is 0 if we are doing an insert into a not allocated block. + This can only happen during REDO of INSERT + */ + if (!old_rec_offset || length < request_length) + { + /* + Did not fit in current block + empty space. Extend with + empty space after block. + */ + if (rownr == max_entry - 1) + { + /* Last entry; Everything is free between this and directory */ + length= ((block_size - PAGE_SUFFIX_SIZE - DIR_ENTRY_SIZE * max_entry) - + rec_offset); + } + else + length= start_of_next_entry(dir) - rec_offset; + DBUG_ASSERT((int) length > 0); + if (length < request_length) + { + /* Not enough continues space, compact page to get more */ + int2store(dir, rec_offset); + compact_page(buff, block_size, rownr, 1); + rec_offset= uint2korr(dir); + length= uint2korr(dir+2); + if (length < request_length) + DBUG_RETURN(1); /* Error in block */ + *empty_space= length; /* All space is here */ + } + } + } + int2store(dir, rec_offset); + *ret_offset= rec_offset; + *ret_length= length; + DBUG_RETURN(0); +} + + + + + /* Check that a region is all zero @@ -1610,7 +1719,7 @@ static my_bool write_block_record(MARIA_HA *info, row->head_length)) DBUG_RETURN(1); - tmp_data_used= 0; /* Either 0 or last used uchar in 'data' */ + tmp_data_used= 0; /* Either 0 or last used uchar in 'data' */ tmp_data= data; if (row_extents_in_use) @@ -2447,7 +2556,7 @@ my_bool _ma_update_block_record(MARIA_HA *info, MARIA_RECORD_POS record_pos, if ((org_empty_size + cur_row->head_length) >= new_row->total_length) { - uint empty, offset, length; + uint rec_offset, length; MARIA_BITMAP_BLOCK block; /* @@ -2456,27 +2565,18 @@ my_bool _ma_update_block_record(MARIA_HA *info, MARIA_RECORD_POS record_pos, */ block.org_bitmap_value= _ma_free_size_to_head_pattern(&share->bitmap, org_empty_size); - offset= uint2korr(dir); - length= uint2korr(dir + 2); - empty= 0; - if (new_row->total_length > length) - { - /* See if there is empty space after */ - if (rownr != (uint) ((uchar *) buff)[DIR_COUNT_OFFSET] - 1) - empty= start_of_next_entry(dir) - (offset + length); - if (new_row->total_length > length + empty) - { - compact_page(buff, share->block_size, rownr, 1); - org_empty_size= 0; - length= uint2korr(dir + 2); - } - } + + if (extend_area_on_page(buff, dir, rownr, share->block_size, + new_row->total_length, &org_empty_size, + &rec_offset, &length)) + DBUG_RETURN(1); + row_pos.buff= buff; row_pos.rownr= rownr; - row_pos.empty_space= org_empty_size + length; + row_pos.empty_space= org_empty_size; row_pos.dir= dir; - row_pos.data= buff + uint2korr(dir); - row_pos.length= length + empty; + row_pos.data= buff + rec_offset; + row_pos.length= length; blocks->block= █ blocks->count= 1; block.page= page; @@ -2545,13 +2645,13 @@ err: 0 ok 1 Page is now empty */ - + static int delete_dir_entry(uchar *buff, uint block_size, uint record_number, uint *empty_space_res) { uint number_of_records= (uint) ((uchar *) buff)[DIR_COUNT_OFFSET]; uint length, empty_space; - uchar *dir; + uchar *dir, *org_dir; DBUG_ENTER("delete_dir_entry"); #ifdef SANITY_CHECKS @@ -2567,9 +2667,8 @@ static int delete_dir_entry(uchar *buff, uint block_size, uint record_number, #endif empty_space= uint2korr(buff + EMPTY_SPACE_OFFSET); - dir= (buff + block_size - DIR_ENTRY_SIZE * record_number - - DIR_ENTRY_SIZE - PAGE_SUFFIX_SIZE); - dir[0]= dir[1]= 0; /* Delete entry */ + org_dir= dir= (buff + block_size - DIR_ENTRY_SIZE * record_number - + DIR_ENTRY_SIZE - PAGE_SUFFIX_SIZE); length= uint2korr(dir + 2); if (record_number == number_of_records - 1) @@ -2582,21 +2681,24 @@ static int delete_dir_entry(uchar *buff, uint block_size, uint record_number, dir+= DIR_ENTRY_SIZE; empty_space+= DIR_ENTRY_SIZE; } while (dir < end && dir[0] == 0 && dir[1] == 0); + + if (number_of_records == 0) + { + buff[PAGE_TYPE_OFFSET]= UNALLOCATED_PAGE; + *empty_space_res= block_size; + DBUG_RETURN(1); + } buff[DIR_COUNT_OFFSET]= (uchar) number_of_records; } empty_space+= length; - if (number_of_records != 0) - { - /* Update directory */ - int2store(buff + EMPTY_SPACE_OFFSET, empty_space); - buff[PAGE_TYPE_OFFSET]|= (uchar) PAGE_CAN_BE_COMPACTED; - *empty_space_res= empty_space; - DBUG_RETURN(0); - } - buff[PAGE_TYPE_OFFSET]= UNALLOCATED_PAGE; - *empty_space_res= block_size; - DBUG_RETURN(1); + /* Update directory */ + org_dir[0]= org_dir[1]= 0; org_dir[2]= org_dir[3]= 0; /* Delete entry */ + int2store(buff + EMPTY_SPACE_OFFSET, empty_space); + buff[PAGE_TYPE_OFFSET]|= (uchar) PAGE_CAN_BE_COMPACTED; + + *empty_space_res= empty_space; + DBUG_RETURN(0); } @@ -2654,7 +2756,7 @@ static my_bool delete_head_or_tail(MARIA_HA *info, page_store(log_data+ FILEID_STORE_SIZE, page); dirpos_store(log_data+ FILEID_STORE_SIZE + PAGE_STORE_SIZE, record_number); - + log_array[TRANSLOG_INTERNAL_PARTS + 0].str= (char*) log_data; log_array[TRANSLOG_INTERNAL_PARTS + 0].length= sizeof(log_data); if (translog_write_record(&lsn, (head ? LOGREC_REDO_PURGE_ROW_HEAD : @@ -3972,7 +4074,7 @@ static size_t fill_update_undo_parts(MARIA_HA *info, const uchar *oldrec, */ start_field_data= field_data= info->update_field_data + 4; log_parts++; - + if (memcmp(oldrec, newrec, share->base.null_bytes)) { /* Store changed null bits */ @@ -3993,7 +4095,7 @@ static size_t fill_update_undo_parts(MARIA_HA *info, const uchar *oldrec, if (memcmp(oldrec + column->offset, newrec + column->offset, column->length)) { - field_data= ma_store_length(field_data, + field_data= ma_store_length(field_data, (uint) (column - share->columndef)); field_count++; log_parts->str= (char*) oldrec + column->offset; @@ -4027,7 +4129,7 @@ static size_t fill_update_undo_parts(MARIA_HA *info, const uchar *oldrec, continue; /* Both are empty; skip */ /* Store null length column */ - field_data= ma_store_length(field_data, + field_data= ma_store_length(field_data, (uint) (column - share->columndef)); field_data= ma_store_length(field_data, 0); field_count++; @@ -4098,7 +4200,7 @@ static size_t fill_update_undo_parts(MARIA_HA *info, const uchar *oldrec, if (new_column_is_empty || new_column_length != old_column_length || memcmp(old_column_pos, new_column_pos, new_column_length)) { - field_data= ma_store_length(field_data, + field_data= ma_store_length(field_data, (uint) (column - share->columndef)); field_data= ma_store_length(field_data, old_column_length); field_count++; @@ -4133,7 +4235,7 @@ static size_t fill_update_undo_parts(MARIA_HA *info, const uchar *oldrec, SYNOPSIS _ma_apply_redo_insert_row_head_or_tail() info Maria handler - lsn LSN to put on page + lsn LSN to put on page page_type HEAD_PAGE or TAIL_PAGE header Header (without FILEID) data Data to be put on page @@ -4157,11 +4259,15 @@ uint _ma_apply_redo_insert_row_head_or_tail(MARIA_HA *info, LSN lsn, uint rec_offset; uchar *buff= info->keyread_buff, *dir; DBUG_ENTER("_ma_apply_redo_insert_row_head_or_tail"); - + info->keyread_buff_used= 1; page= page_korr(header); rownr= dirpos_korr(header+PAGE_STORE_SIZE); + DBUG_PRINT("enter", ("rowid: %lu page: %lu rownr: %u data_length: %u", + (ulong) ma_recordpos(page, rownr), + (ulong) page, rownr, (uint) data_length)); + if (((page + 1) * info->s->block_size) > info->state->data_file_length) { /* @@ -4222,7 +4328,7 @@ uint _ma_apply_redo_insert_row_head_or_tail(MARIA_HA *info, LSN lsn, dir= (buff + block_size - DIR_ENTRY_SIZE * (rownr + 1) - PAGE_SUFFIX_SIZE); empty_space= uint2korr(buff + EMPTY_SPACE_OFFSET); - + if (max_entry <= rownr) { /* Add directory entry first in directory and data last on page */ @@ -4248,39 +4354,15 @@ uint _ma_apply_redo_insert_row_head_or_tail(MARIA_HA *info, LSN lsn, } else { + uint length; /* - reuse old entry. This is empty if the command was an insert and + Reuse old entry. This is empty if the command was an insert and possible used if the command was an update. */ - uchar *end_data; - uint rec_end; - - /* Add back space if we are reusing entry */ - empty_space+= uint2korr(dir+2); - - /* Find first possible position where to put new data */ - end_data= (buff + block_size - PAGE_SUFFIX_SIZE - - DIR_ENTRY_SIZE * max_entry); - rec_offset= end_of_previous_entry(dir, end_data); - if (rownr != max_entry -1) - rec_end= start_of_next_entry(dir); - else - rec_end= (uint) (buff - end_data); - DBUG_ASSERT(rec_end > rec_offset); - - if ((uint) (rec_end - rec_offset) < data_length) - { - uint length; - /* Not enough continues space, compact page to get more */ - int2store(dir, rec_offset); - compact_page(buff, block_size, rownr, 1); - rec_offset= uint2korr(dir); - length= uint2korr(dir+2); - DBUG_ASSERT(length >= data_length); - if (length < data_length) - goto err; - empty_space= length; - } + if (extend_area_on_page(buff, dir, rownr, block_size, + data_length, &empty_space, + &rec_offset, &length)) + goto err; } } } @@ -4317,7 +4399,7 @@ err: SYNOPSIS _ma_apply_redo_purge_row_head_or_tail() info Maria handler - lsn LSN to put on page + lsn LSN to put on page page_type HEAD_PAGE or TAIL_PAGE header Header (without FILEID) @@ -4339,7 +4421,7 @@ uint _ma_apply_redo_purge_row_head_or_tail(MARIA_HA *info, LSN lsn, uint block_size= share->block_size; uchar *buff= info->keyread_buff; DBUG_ENTER("_ma_apply_redo_purge_row_head_or_tail"); - + info->keyread_buff_used= 1; page= page_korr(header); record_number= dirpos_korr(header+PAGE_STORE_SIZE); @@ -4405,7 +4487,7 @@ uint _ma_apply_redo_purge_blocks(MARIA_HA *info, uchar *buff= info->keyread_buff; DBUG_ENTER("_ma_apply_redo_purge_blocks"); - info->keyread_buff_used= 1; + info->keyread_buff_used= 1; ranges= pagerange_korr(header); header+= PAGERANGE_STORE_SIZE; diff --git a/storage/maria/ma_open.c b/storage/maria/ma_open.c index b5560220b63..64f232ca73c 100644 --- a/storage/maria/ma_open.c +++ b/storage/maria/ma_open.c @@ -600,7 +600,9 @@ MARIA_HA *maria_open(const char *name, int mode, uint open_flags) if (share->base.born_transactional) { share->page_type= PAGECACHE_LSN_PAGE; +#ifdef ENABLE_WHEN_WE_HAVE_TRANS_ROW_ID /* QQ */ share->base_length+= TRANS_ROW_EXTRA_HEADER_SIZE; +#endif if (unlikely((share->state.create_rename_lsn == (LSN)ULONGLONG_MAX) && (open_flags & HA_OPEN_FROM_SQL_LAYER))) { diff --git a/storage/maria/ma_test1.c b/storage/maria/ma_test1.c index b25fc72bebd..e5485c43f23 100644 --- a/storage/maria/ma_test1.c +++ b/storage/maria/ma_test1.c @@ -633,7 +633,7 @@ static struct my_option my_long_options[] = 0, 0, 0, 0, 0, 0}, {"unique", 'C', "Undocumented", (uchar**) &opt_unique, (uchar**) &opt_unique, 0, GET_BOOL, NO_ARG, 0, 0, 0, 0, 0, 0}, - {"update-rows", 'u', "Undocumented", (uchar**) &update_count, + {"update-rows", 'u', "Max number of rows to update", (uchar**) &update_count, (uchar**) &update_count, 0, GET_UINT, REQUIRED_ARG, 1000, 0, 0, 0, 0, 0}, {"verbose", 'v', "Be more verbose", (uchar**) &verbose, (uchar**) &verbose, 0, GET_BOOL, NO_ARG, 0, 0, 0, 0, 0, 0}, diff --git a/storage/maria/ma_test2.c b/storage/maria/ma_test2.c index 00a7fc33cca..19816606720 100644 --- a/storage/maria/ma_test2.c +++ b/storage/maria/ma_test2.c @@ -895,7 +895,11 @@ end: if (maria_commit(file)) goto err; if (maria_close(file)) + { + file= 0; goto err; + } + file= 0; maria_panic(HA_PANIC_CLOSE); /* Should close log */ if (!silent) { @@ -937,7 +941,11 @@ reads: %10lu\n", err: printf("got error: %d when using MARIA-database\n",my_errno); if (file) + { + if (maria_commit(file)) + goto err; VOID(maria_close(file)); + } maria_end(); return(1); } /* main */ diff --git a/storage/maria/ma_test_all.sh b/storage/maria/ma_test_all.sh index e8b9f1cef9a..108dffd7df7 100755 --- a/storage/maria/ma_test_all.sh +++ b/storage/maria/ma_test_all.sh @@ -143,6 +143,10 @@ run_repair_tests() $maria_path/maria_chk$suffix -se test2 $maria_path/maria_chk$suffix -s --parallel-recover --quick test2 $maria_path/maria_chk$suffix -se test2 + $maria_path/ma_test2$suffix $silent -c $row_type + $maria_path/maria_chk$suffix -se test2 + $maria_path/maria_chk$suffix -sr test2 + $maria_path/maria_chk$suffix -se test2 } run_pack_tests() diff --git a/storage/maria/ma_test_recovery b/storage/maria/ma_test_recovery index 7fb1a302a79..4e88824197e 100755 --- a/storage/maria/ma_test_recovery +++ b/storage/maria/ma_test_recovery @@ -21,7 +21,7 @@ echo "MARIA RECOVERY TESTS - success is if exit code is 0" # identical to the saved original. # Does not test the index file as we don't have logging for it yet. -for prog in "$maria_path/ma_test1 $silent -M -T --skip-update -c" "$maria_path/ma_test2 $silent -L -K -W -P -M -T -g -c" +for prog in "$maria_path/ma_test1 $silent -M -T -c" "$maria_path/ma_test2 $silent -L -K -W -P -M -T -c" "$maria_path/ma_test2 $silent -M -T -c -b" do rm -f maria_log.* maria_log_control echo "TEST WITH $prog" @@ -36,6 +36,8 @@ do $maria_path/maria_read_log -a > $tmp/maria_read_log_$table.txt $maria_path/maria_chk -dvv $table > $tmp/maria_chk_message.txt 2>&1 + cmp $table.MAD $tmp/$table.MAD.good + # QQ: Remove the following line when we also can recovert the index file $maria_path/maria_chk -s -r $table @@ -46,7 +48,7 @@ do echo "checksum differs for $table before and after recovery" exit 1; fi -# cmp $table.MAD $tmp/$table.MAD.good + # When "recovery of the table's state" is ready, we can test it like this: # diff $tmp/maria_chk_message.good.txt $tmp/maria_chk_message.txt > $tmp/maria_chk_diff.txt || true # if [ -s $tmp/maria_chk_diff.txt ] -- cgit v1.2.1 From 5183a4b00b0cadf74b9ed5c92734617ec3c9270b Mon Sep 17 00:00:00 2001 From: unknown Date: Mon, 3 Sep 2007 12:05:17 +0300 Subject: Fixed several bugs found by running *.test with maria engine Renamed HA_EXTRA_PREAPRE_FOR_DELETE to HA_EXTRA_PERPARE_FOR_DROP Added HA_EXTRA_PREPARE_FOR_RENAME (as we in the code before used HA_EXTRA_PREPARE_FOR_DELETE also for renames which confused things) Allow multiple write locks for same page by same file handle Don't write table state if table is not changed include/my_base.h: Renamed HA_EXTRA_PREAPRE_FOR_DELETE to HA_EXTRA_PERPARE_FOR_DROP Added HA_EXTRA_PREPARE_FOR_RENAME (as we in the code before used HA_EXTRA_PREPARE_FOR_DELETE also for renames which confused things) mysql-test/r/maria.result: More tests of things that failed in other tests mysql-test/t/maria.test: More tests of things that failed in other tests sql/ha_partition.cc: HA_EXTRA_PREPARE_FOR_DELETE -> HA_EXTRA_PREPARE_FOR_DROP Use HA_EXTRA_PREPARE_FOR_RENAME for renames sql/ha_partition.h: HA_EXTRA_PREPARE_FOR_DELETE -> HA_EXTRA_PREPARE_FOR_DROP Use HA_EXTRA_PREPARE_FOR_RENAME for renames sql/lock.cc: Fixed comment sql/sql_table.cc: Fixed wrong usage of HA_EXTRA_PREAPRE_FOR_DELETE storage/maria/ha_maria.cc: Added missing _ma_renable_logging_for_table() (When using with ALTER TABLE + repair index) Enabled fast generation of index storage/maria/ma_bitmap.c: Fixed bug when resetting full pages when page was a tail page storage/maria/ma_blockrec.c: Fixed several bugs found by running *.test with maria engine: During update we keep old changed pages locked with a write lock to be able to reuse them. - Fixed bug with allocated but not used tail part - Fixed bug with blob that only had tail part - Fixed bug when update reused a page (needed multiple write locks for same page) - Fixed bug when first extent was a tail block storage/maria/ma_check.c: Better error message when bitmap is destroyed storage/maria/ma_close.c: Only write status if file was changed. Fixed bug when maria_chk -e file_name changed the file. storage/maria/ma_dynrec.c: Removed not used argument to _ma_state_info_read_dsk storage/maria/ma_extra.c: HA_EXTRA_PREPARE_FOR_DELETE -> HA_EXTRA_PREPARE_FOR_DROP Use HA_EXTRA_PREPARE_FOR_RENAME for renames Only ignore flushing of pages for DROP (not rename) storage/maria/ma_locking.c: Removed not used argument to _ma_state_info_read_dsk storage/maria/ma_open.c: Removed not used argument to _ma_state_info_read_dsk storage/maria/ma_pagecache.c: Allow multiple write locks for same page by same file handle (Not yet complete, Sanja will fix) storage/maria/ma_recovery.c: HA_EXTRA_PREPARE_FOR_DELETE -> HA_EXTRA_PREPARE_FOR_DROP storage/maria/maria_def.h: Removed not used argument to _ma_state_info_read_dsk storage/myisam/mi_extra.c: HA_EXTRA_PREPARE_FOR_DELETE -> HA_EXTRA_PREPARE_FOR_DROP Use HA_EXTRA_PREPARE_FOR_RENAME for renames Only ignore flushing of pages for DROP (not rename) storage/myisammrg/ha_myisammrg.cc: HA_EXTRA_PREPARE_FOR_DELETE -> HA_EXTRA_PREPARE_FOR_DROP Use HA_EXTRA_PREPARE_FOR_RENAME for renames --- include/my_base.h | 6 ++- mysql-test/r/maria.result | 74 +++++++++++++++++++++++++++++- mysql-test/t/maria.test | 90 +++++++++++++++++++++++++++++++++++- sql/ha_partition.cc | 20 ++++---- sql/ha_partition.h | 2 +- sql/lock.cc | 4 +- sql/sql_table.cc | 7 +-- storage/maria/ha_maria.cc | 14 +++--- storage/maria/ma_bitmap.c | 20 +++++--- storage/maria/ma_blockrec.c | 96 +++++++++++++++++++++++++++------------ storage/maria/ma_check.c | 7 ++- storage/maria/ma_close.c | 3 +- storage/maria/ma_dynrec.c | 2 +- storage/maria/ma_extra.c | 7 +-- storage/maria/ma_locking.c | 6 +-- storage/maria/ma_open.c | 9 +--- storage/maria/ma_pagecache.c | 59 ++++++++++++++---------- storage/maria/ma_recovery.c | 2 +- storage/maria/maria_def.h | 3 +- storage/myisam/mi_extra.c | 7 +-- storage/myisammrg/ha_myisammrg.cc | 3 +- 21 files changed, 332 insertions(+), 109 deletions(-) diff --git a/include/my_base.h b/include/my_base.h index 57446e6136f..473ed46ff19 100644 --- a/include/my_base.h +++ b/include/my_base.h @@ -135,7 +135,7 @@ enum ha_extra_function { HA_EXTRA_RESET_STATE, /* Reset positions */ HA_EXTRA_IGNORE_DUP_KEY, /* Dup keys don't rollback everything*/ HA_EXTRA_NO_IGNORE_DUP_KEY, - HA_EXTRA_PREPARE_FOR_DELETE, + HA_EXTRA_PREPARE_FOR_DROP, HA_EXTRA_PREPARE_FOR_UPDATE, /* Remove read cache if problems */ HA_EXTRA_PRELOAD_BUFFER_SIZE, /* Set buffer size for preloading */ /* @@ -185,7 +185,9 @@ enum ha_extra_function { Inform handler that an "INSERT...ON DUPLICATE KEY UPDATE" will be executed. This condition is unset by HA_EXTRA_NO_IGNORE_DUP_KEY. */ - HA_EXTRA_INSERT_WITH_UPDATE + HA_EXTRA_INSERT_WITH_UPDATE, + /* Inform handler that we will do a rename */ + HA_EXTRA_PREPARE_FOR_RENAME }; /* The following is parameter to ha_panic() */ diff --git a/mysql-test/r/maria.result b/mysql-test/r/maria.result index 53063c9c7ab..68ae0cb267b 100644 --- a/mysql-test/r/maria.result +++ b/mysql-test/r/maria.result @@ -1598,6 +1598,9 @@ insert into t1 values (1),(2),(3),(4),(NULL),(NULL),(NULL),(NULL); analyze table t1; Table Op Msg_type Msg_text test.t1 analyze status OK +analyze table t1; +Table Op Msg_type Msg_text +test.t1 analyze status Table is already up to date show keys from t1; Table Non_unique Key_name Seq_in_index Column_name Collation Cardinality Sub_part Packed Null Index_type Comment t1 1 a 1 a A 8 NULL NULL YES BTREE @@ -1887,10 +1890,79 @@ t1_name varchar(255) default null, t1_id int(10) unsigned not null auto_increment, key (t1_name), primary key (t1_id) -) auto_increment = 1000 default charset=latin1; +) engine=maria auto_increment = 1000 default charset=latin1; lock tables t1 write; INSERT INTO `t1` VALUES ('bla',1000),('bla',1001),('bla',1002); check table t1; Table Op Msg_type Msg_text test.t1 check status OK unlock tables; +create table t2 like t1; +insert into t2 select * from t1; +analyze table t2; +Table Op Msg_type Msg_text +test.t2 analyze status OK +delete from t2; +insert into t2 select * from t1; +analyze table t2; +Table Op Msg_type Msg_text +test.t2 analyze status OK +drop table t1,t2; +create table t1 (a bigint auto_increment, primary key(a), b char(255), c varchar(20000)); +update t1 set b=repeat('a',100) where a between 1 and 100; +check table t1; +Table Op Msg_type Msg_text +test.t1 check status OK +update t1 set c=repeat('a',8192*2) where a between 200 and 202; +check table t1; +Table Op Msg_type Msg_text +test.t1 check status OK +drop table t1; +CREATE TABLE t1 ( +auto int(5) unsigned NOT NULL auto_increment, +string char(10) default "hello", +tiny tinyint(4) DEFAULT '0' NOT NULL , +short smallint(6) DEFAULT '1' NOT NULL , +medium mediumint(8) DEFAULT '0' NOT NULL, +long_int int(11) DEFAULT '0' NOT NULL, +longlong bigint(13) DEFAULT '0' NOT NULL, +real_float float(13,1) DEFAULT 0.0 NOT NULL, +real_double double(16,4), +utiny tinyint(3) unsigned DEFAULT '0' NOT NULL, +ushort smallint(5) unsigned zerofill DEFAULT '00000' NOT NULL, +umedium mediumint(8) unsigned DEFAULT '0' NOT NULL, +ulong int(11) unsigned DEFAULT '0' NOT NULL, +ulonglong bigint(13) unsigned DEFAULT '0' NOT NULL, +time_stamp timestamp, +date_field date, +time_field time, +date_time datetime, +blob_col blob, +tinyblob_col tinyblob, +mediumblob_col mediumblob not null default '', +longblob_col longblob not null default '', +options enum('one','two','tree') not null , +flags set('one','two','tree') not null default '', +PRIMARY KEY (auto), +KEY (utiny), +KEY (tiny), +KEY (short), +KEY any_name (medium), +KEY (longlong), +KEY (real_float), +KEY (ushort), +KEY (umedium), +KEY (ulong), +KEY (ulonglong,ulong), +KEY (options,flags) +) engine=maria; +insert into t1 values (10,1,1,1,1,1,1,1,1,1,1,1,1,1,NULL,0,0,0,1,1,1,1,'one','one'); +create table t2 (primary key (auto)) engine=maria row_format=page select auto+1 as auto,1 as t1, 'a' as t2, repeat('a',256) as t3, binary repeat('b',256) as t4, repeat('a',4096) as t5, binary repeat('b',4096) as t6, '' as t7, binary '' as t8 from t1; +check table t1,t2; +Table Op Msg_type Msg_text +test.t1 check status OK +test.t2 check status OK +select t1,t2,length(t3),length(t4),length(t5),length(t6),t7,t8 from t2; +t1 t2 length(t3) length(t4) length(t5) length(t6) t7 t8 +1 a 256 256 4096 4096 +drop table t1,t2; diff --git a/mysql-test/t/maria.test b/mysql-test/t/maria.test index aca85b1442e..d45f4c2ec6b 100644 --- a/mysql-test/t/maria.test +++ b/mysql-test/t/maria.test @@ -711,6 +711,7 @@ analyze table t1; show index from t1; set maria_stats_method=DEFAULT; + drop table t1; # @@ -952,6 +953,7 @@ create table t1 (a int, key(a)); insert into t1 values (1),(2),(3),(4),(NULL),(NULL),(NULL),(NULL); analyze table t1; +analyze table t1; show keys from t1; alter table t1 disable keys; @@ -1155,17 +1157,103 @@ drop table t1; # CHECK TABLE was reporting # "Size of datafile is: 0 Should be: 16384" +# + create table `t1` ( t1_name varchar(255) default null, t1_id int(10) unsigned not null auto_increment, key (t1_name), primary key (t1_id) -) auto_increment = 1000 default charset=latin1; +) engine=maria auto_increment = 1000 default charset=latin1; lock tables t1 write; INSERT INTO `t1` VALUES ('bla',1000),('bla',1001),('bla',1002); check table t1; unlock tables; +# +# Check that an empty table uses fast recreate of index when we fill it +# with insert ... select. + +create table t2 like t1; +insert into t2 select * from t1; + +# This should say that the table is already up to date +analyze table t2; +delete from t2; +insert into t2 select * from t1; +analyze table t2; + +drop table t1,t2; + +# +# Test when expanding a row so that it doesn't fit into the same page +# + +create table t1 (a bigint auto_increment, primary key(a), b char(255), c varchar(20000)); + +let $1=1000; +--disable_query_log +--disable_warnings +while ($1) +{ + insert into t1 () values(); + dec $1; +} +--enable_query_log +update t1 set b=repeat('a',100) where a between 1 and 100; +check table t1; +update t1 set c=repeat('a',8192*2) where a between 200 and 202; +check table t1; +drop table t1; + +# +# Test tail pages for blobs +# + +CREATE TABLE t1 ( + auto int(5) unsigned NOT NULL auto_increment, + string char(10) default "hello", + tiny tinyint(4) DEFAULT '0' NOT NULL , + short smallint(6) DEFAULT '1' NOT NULL , + medium mediumint(8) DEFAULT '0' NOT NULL, + long_int int(11) DEFAULT '0' NOT NULL, + longlong bigint(13) DEFAULT '0' NOT NULL, + real_float float(13,1) DEFAULT 0.0 NOT NULL, + real_double double(16,4), + utiny tinyint(3) unsigned DEFAULT '0' NOT NULL, + ushort smallint(5) unsigned zerofill DEFAULT '00000' NOT NULL, + umedium mediumint(8) unsigned DEFAULT '0' NOT NULL, + ulong int(11) unsigned DEFAULT '0' NOT NULL, + ulonglong bigint(13) unsigned DEFAULT '0' NOT NULL, + time_stamp timestamp, + date_field date, + time_field time, + date_time datetime, + blob_col blob, + tinyblob_col tinyblob, + mediumblob_col mediumblob not null default '', + longblob_col longblob not null default '', + options enum('one','two','tree') not null , + flags set('one','two','tree') not null default '', + PRIMARY KEY (auto), + KEY (utiny), + KEY (tiny), + KEY (short), + KEY any_name (medium), + KEY (longlong), + KEY (real_float), + KEY (ushort), + KEY (umedium), + KEY (ulong), + KEY (ulonglong,ulong), + KEY (options,flags) +) engine=maria; +insert into t1 values (10,1,1,1,1,1,1,1,1,1,1,1,1,1,NULL,0,0,0,1,1,1,1,'one','one'); +create table t2 (primary key (auto)) engine=maria row_format=page select auto+1 as auto,1 as t1, 'a' as t2, repeat('a',256) as t3, binary repeat('b',256) as t4, repeat('a',4096) as t5, binary repeat('b',4096) as t6, '' as t7, binary '' as t8 from t1; +check table t1,t2; +select t1,t2,length(t3),length(t4),length(t5),length(t6),t7,t8 from t2; +drop table t1,t2; + # End of 5.2 tests --disable_result_log diff --git a/sql/ha_partition.cc b/sql/ha_partition.cc index e4924e8e8f2..15d7353c03c 100644 --- a/sql/ha_partition.cc +++ b/sql/ha_partition.cc @@ -4621,11 +4621,14 @@ void ha_partition::get_dynamic_partition_info(PARTITION_INFO *stat_info, about this call). We pass this along to all underlying MyISAM handlers and ignore it for the rest. - HA_EXTRA_PREPARE_FOR_DELETE: + HA_EXTRA_PREPARE_FOR_DROP: Only used by MyISAM, called in preparation for a DROP TABLE. It's used mostly by Windows that cannot handle dropping an open file. On other platforms it has the same effect as HA_EXTRA_FORCE_REOPEN. + HA_EXTRA_PREPARE_FOR_RENAME: + Informs the handler we are about to attempt a rename of the table. + HA_EXTRA_READCHECK: HA_EXTRA_NO_READCHECK: Only one call to HA_EXTRA_NO_READCHECK from ha_open where it says that @@ -4751,14 +4754,15 @@ int ha_partition::extra(enum ha_extra_function operation) } /* Category 3), used by MyISAM handlers */ - case HA_EXTRA_PREPARE_FOR_DELETE: - DBUG_RETURN(prepare_for_delete()); + case HA_EXTRA_PREPARE_FOR_RENAME: + DBUG_RETURN(prepare_for_rename()); break; case HA_EXTRA_NORMAL: case HA_EXTRA_QUICK: case HA_EXTRA_NO_READCHECK: case HA_EXTRA_PREPARE_FOR_UPDATE: case HA_EXTRA_FORCE_REOPEN: + case HA_EXTRA_PREPARE_FOR_DROP: case HA_EXTRA_FLUSH_CACHE: { if (m_myisam) @@ -4910,24 +4914,24 @@ void ha_partition::prepare_extra_cache(uint cachesize) 0 Success */ -int ha_partition::prepare_for_delete() +int ha_partition::prepare_for_rename() { int result= 0, tmp; handler **file; - DBUG_ENTER("ha_partition::prepare_for_delete()"); + DBUG_ENTER("ha_partition::prepare_for_rename()"); if (m_new_file != NULL) { for (file= m_new_file; *file; file++) - if ((tmp= (*file)->extra(HA_EXTRA_PREPARE_FOR_DELETE))) + if ((tmp= (*file)->extra(HA_EXTRA_PREPARE_FOR_RENAME))) result= tmp; for (file= m_reorged_file; *file; file++) - if ((tmp= (*file)->extra(HA_EXTRA_PREPARE_FOR_DELETE))) + if ((tmp= (*file)->extra(HA_EXTRA_PREPARE_FOR_RENAME))) result= tmp; DBUG_RETURN(result); } - DBUG_RETURN(loop_extra(HA_EXTRA_PREPARE_FOR_DELETE)); + DBUG_RETURN(loop_extra(HA_EXTRA_PREPARE_FOR_RENAME)); } /* diff --git a/sql/ha_partition.h b/sql/ha_partition.h index 895f001fa6a..bc4747be85b 100644 --- a/sql/ha_partition.h +++ b/sql/ha_partition.h @@ -211,7 +211,7 @@ public: } virtual void change_table_ptr(TABLE *table_arg, TABLE_SHARE *share); private: - int prepare_for_delete(); + int prepare_for_rename(); int copy_partitions(ulonglong *copied, ulonglong *deleted); void cleanup_new_partition(uint part_count); int prepare_new_partition(TABLE *table, HA_CREATE_INFO *create_info, diff --git a/sql/lock.cc b/sql/lock.cc index e129da27005..12b69edc3d3 100644 --- a/sql/lock.cc +++ b/sql/lock.cc @@ -31,8 +31,8 @@ This is followed by a call to thr_multi_lock() for all tables. - When statement is done, we call mysql_unlock_tables(). - This will call thr_multi_unlock() followed by - table_handler->external_lock(thd, F_UNLCK) for each table. + table_handler->external_lock(thd, F_UNLCK) followed by + thr_multi_unlock() for each table. - Note that mysql_unlock_tables() may be called several times as MySQL in some cases can free some tables earlier than others. diff --git a/sql/sql_table.cc b/sql/sql_table.cc index 6fc1c235ebf..158f415317c 100644 --- a/sql/sql_table.cc +++ b/sql/sql_table.cc @@ -3694,8 +3694,9 @@ mysql_rename_table(handlerton *base, const char *old_db, wait_while_table_is_used() thd Thread handler table Table to remove from cache - function HA_EXTRA_PREPARE_FOR_DELETE if table is to be deleted + function HA_EXTRA_PREPARE_FOR_DROP if table is to be deleted HA_EXTRA_FORCE_REOPEN if table is not be used + HA_EXTRA_PREPARE_FOR_REANME if table is to be renamed NOTES When returning, the table will be unusable for other threads until the table is closed. @@ -3745,7 +3746,7 @@ void close_cached_table(THD *thd, TABLE *table) { DBUG_ENTER("close_cached_table"); - wait_while_table_is_used(thd, table, HA_EXTRA_PREPARE_FOR_DELETE); + wait_while_table_is_used(thd, table, HA_EXTRA_FORCE_REOPEN); /* Close lock if this is not got with LOCK TABLES */ if (thd->lock) { @@ -6532,7 +6533,7 @@ view_err: if (lower_case_table_names) my_casedn_str(files_charset_info, old_name); - wait_while_table_is_used(thd, table, HA_EXTRA_PREPARE_FOR_DELETE); + wait_while_table_is_used(thd, table, HA_EXTRA_PREPARE_FOR_RENAME); close_data_files_and_morph_locks(thd, db, table_name); error=0; diff --git a/storage/maria/ha_maria.cc b/storage/maria/ha_maria.cc index c2fa0ec14b1..55ce800d596 100644 --- a/storage/maria/ha_maria.cc +++ b/storage/maria/ha_maria.cc @@ -1324,7 +1324,10 @@ int ha_maria::repair(THD *thd, HA_CHECK ¶m, bool do_optimize) } thd->proc_info= old_proc_info; if (!thd->locked_tables) + { + _ma_reenable_logging_for_table(file->s); maria_lock_database(file, F_UNLCK); + } DBUG_RETURN(error ? HA_ADMIN_FAILED : !optimize_done ? HA_ADMIN_ALREADY_DONE : HA_ADMIN_OK); } @@ -1624,10 +1627,8 @@ void ha_maria::start_bulk_insert(ha_rows rows) if (!rows || (rows > MARIA_MIN_ROWS_TO_USE_WRITE_CACHE)) maria_extra(file, HA_EXTRA_WRITE_CACHE, (void*) &size); - can_enable_indexes= maria_is_all_keys_active(file->s->state.key_map, - file->s->base.keys); - /* TODO: Remove when we have repair() working */ - can_enable_indexes= 0; + can_enable_indexes= (maria_is_all_keys_active(file->s->state.key_map, + file->s->base.keys)); if (!(specialflag & SPECIAL_SAFE_MODE)) { @@ -1640,9 +1641,8 @@ void ha_maria::start_bulk_insert(ha_rows rows) if (file->state->records == 0 && can_enable_indexes && (!rows || rows >= MARIA_MIN_ROWS_TO_DISABLE_INDEXES)) maria_disable_non_unique_index(file, rows); - else - if (!file->bulk_insert && - (!rows || rows >= MARIA_MIN_ROWS_TO_USE_BULK_INSERT)) + else if (!file->bulk_insert && + (!rows || rows >= MARIA_MIN_ROWS_TO_USE_BULK_INSERT)) { maria_init_bulk_insert(file, thd->variables.bulk_insert_buff_size, rows); } diff --git a/storage/maria/ma_bitmap.c b/storage/maria/ma_bitmap.c index 66377172877..2a2308637b6 100644 --- a/storage/maria/ma_bitmap.c +++ b/storage/maria/ma_bitmap.c @@ -451,6 +451,10 @@ static void _ma_print_bitmap(MARIA_FILE_BITMAP *bitmap) fprintf(DBUG_FILE,"\nBitmap page changes at page %lu\n", (ulong) bitmap->page); + DBUG_ASSERT(memcmp(bitmap->map + bitmap->block_size - + sizeof(maria_bitmap_marker), + maria_bitmap_marker, sizeof(maria_bitmap_marker)) == 0); + page= (ulong) bitmap->page+1; for (pos= bitmap->map, org_pos= bitmap->map + bitmap->block_size ; pos < end ; @@ -536,14 +540,14 @@ static my_bool _ma_read_bitmap_page(MARIA_SHARE *share, } bitmap->used_size= bitmap->total_size; DBUG_ASSERT(share->pagecache->block_size == bitmap->block_size); - res= (pagecache_read(share->pagecache, + res= ((pagecache_read(share->pagecache, (PAGECACHE_FILE*)&bitmap->file, page, 0, (uchar*) bitmap->map, PAGECACHE_PLAIN_PAGE, - PAGECACHE_LOCK_LEFT_UNLOCKED, 0) == NULL) | - memcmp(bitmap->map + bitmap->block_size - - sizeof(maria_bitmap_marker), - maria_bitmap_marker, sizeof(maria_bitmap_marker)); + PAGECACHE_LOCK_LEFT_UNLOCKED, 0) == NULL) || + memcmp(bitmap->map + bitmap->block_size - + sizeof(maria_bitmap_marker), + maria_bitmap_marker, sizeof(maria_bitmap_marker))); #ifndef DBUG_OFF if (!res) memcpy(bitmap->map + bitmap->block_size, bitmap->map, bitmap->block_size); @@ -1838,11 +1842,15 @@ my_bool _ma_bitmap_release_unused(MARIA_HA *info, MARIA_BITMAP_BLOCKS *blocks) /* Handle all full pages and tail pages (for head page and blob) */ for (block++; block < end; block++) { + uint page_count; if (!block->page_count) continue; /* Skip 'filler blocks' */ + page_count= block->page_count; if (block->used & BLOCKUSED_TAIL) { + /* The bitmap page is only one page */ + page_count= 1; if (block->used & BLOCKUSED_USED) { DBUG_PRINT("info", ("tail empty_space: %u", block->empty_space)); @@ -1861,7 +1869,7 @@ my_bool _ma_bitmap_release_unused(MARIA_HA *info, MARIA_BITMAP_BLOCKS *blocks) } if (!(block->used & BLOCKUSED_USED) && _ma_reset_full_page_bits(info, bitmap, - block->page, block->page_count)) + block->page, page_count)) goto err; } pthread_mutex_unlock(&info->s->bitmap.bitmap_lock); diff --git a/storage/maria/ma_blockrec.c b/storage/maria/ma_blockrec.c index 555949dfa84..7f29f075463 100644 --- a/storage/maria/ma_blockrec.c +++ b/storage/maria/ma_blockrec.c @@ -288,8 +288,10 @@ typedef struct st_maria_extent_cursor uint extent_count; /* <> 0 if current extent is a tail page; Set while using cursor */ uint tail; + /* Position for tail on tail page */ + uint tail_row_nr; /* - <> 1 if we are working on the first extent (i.e., the one that is store in + == 1 if we are working on the first extent (i.e., the one that is stored in the row header, not an extent that is stored as part of the row data). */ my_bool first_extent; @@ -299,7 +301,7 @@ typedef struct st_maria_extent_cursor static my_bool delete_tails(MARIA_HA *info, MARIA_RECORD_POS *tails); static my_bool delete_head_or_tail(MARIA_HA *info, ulonglong page, uint record_number, - my_bool head); + my_bool head, my_bool from_update); static void _ma_print_directory(uchar *buff, uint block_size); static void compact_page(uchar *buff, uint block_size, uint rownr, my_bool extend_block); @@ -461,7 +463,7 @@ my_bool _ma_init_block_record(MARIA_HA *info) /* The following should be big enough for all purposes */ if (my_init_dynamic_array(&info->pinned_pages, sizeof(MARIA_PINNED_PAGE), - max(info->s->base.blobs + 2, + max(info->s->base.blobs*2 + 4, MARIA_MAX_TREE_LEVELS*2), 16)) goto err; row->base_length= new_row->base_length= info->s->base_length; @@ -840,6 +842,7 @@ static void calc_record_size(MARIA_HA *info, const uchar *record, MARIA_COLUMNDEF *column, *end_column; uint *null_field_lengths= row->null_field_lengths; ulong *blob_lengths= row->blob_lengths; + DBUG_ENTER("calc_record_size"); row->normal_length= row->char_length= row->varchar_length= row->blob_length= row->extents_count= 0; @@ -968,6 +971,9 @@ static void calc_record_size(MARIA_HA *info, const uchar *record, row->total_length= (row->head_length + row->blob_length); if (row->total_length < share->base.min_row_length) row->total_length= share->base.min_row_length; + DBUG_PRINT("exit", ("head_length: %lu total_length: %lu", + (ulong) row->head_length, (ulong) row->total_length)); + DBUG_VOID_RETURN; } @@ -1395,6 +1401,7 @@ static my_bool write_full_pages(MARIA_HA *info, DBUG_PRINT("enter", ("length: %lu page: %lu page_count: %lu", (ulong) length, (ulong) block->page, (ulong) block->page_count)); + DBUG_ASSERT((block->page_count & TAIL_BIT) == 0); info->keyread_buff_used= 1; page= block->page; @@ -1938,13 +1945,10 @@ static my_bool write_block_record(MARIA_HA *info, ulong length; ulong data_length= (tmp_data - info->rec_buff); -#ifdef MONTY_WILL_KNOW #ifdef SANITY_CHECKS - if (cur_block->sub_blocks == 1) + if (head_block->sub_blocks == 1) goto crashed; /* no reserved full or tails */ #endif -#endif - /* Find out where to write tail for non-blob fields. @@ -2073,6 +2077,11 @@ static my_bool write_block_record(MARIA_HA *info, length)) goto disk_err; tmp_data-= length; /* Remove the tail */ + if (tmp_data == info->rec_buff) + { + /* We have no full blocks to write for the head part */ + tmp_data_used= 0; + } /* Store the tail position for the non-blob fields */ if (head_tail_block == head_block + 1) @@ -2319,8 +2328,8 @@ static my_bool write_block_record(MARIA_HA *info, if (block[block->sub_blocks - 1].used & BLOCKUSED_TAIL) blob_length-= (blob_length % FULL_PAGE_SIZE(block_size)); - if (write_full_pages(info, info->trn->undo_lsn, block, - blob_pos, blob_length)) + if (blob_length && write_full_pages(info, info->trn->undo_lsn, block, + blob_pos, blob_length)) goto disk_err; block+= block->sub_blocks; } @@ -2356,8 +2365,11 @@ disk_err: @todo RECOVERY we should distinguish below between log write error and table write error. The former should stop Maria immediately, the latter should mark the table corrupted. - */ - /* Unpin all pinned pages to not cause problems for disk cache */ + */ + /* + Unpin all pinned pages to not cause problems for disk cache. This is + safe to call even if we already called _ma_unpin_all_pages() above. + */ _ma_unpin_all_pages(info, 0); DBUG_RETURN(1); @@ -2445,7 +2457,8 @@ my_bool _ma_write_abort_block_record(MARIA_HA *info) if (delete_head_or_tail(info, ma_recordpos_to_page(info->cur_row.lastpos), - ma_recordpos_to_dir_entry(info->cur_row.lastpos), 1)) + ma_recordpos_to_dir_entry(info->cur_row.lastpos), 1, + 0)) res= 1; for (block= blocks->block + 1, end= block + blocks->count - 1; block < end; block++) @@ -2457,7 +2470,7 @@ my_bool _ma_write_abort_block_record(MARIA_HA *info) write_block_record() */ if (delete_head_or_tail(info, block->page, block->page_count & ~TAIL_BIT, - 0)) + 0, 0)) res= 1; } else if (block->used & BLOCKUSED_USED) @@ -2533,7 +2546,6 @@ my_bool _ma_update_block_record(MARIA_HA *info, MARIA_RECORD_POS record_pos, ulonglong page; struct st_row_pos_info row_pos; MARIA_SHARE *share= info->s; - my_bool res; DBUG_ENTER("_ma_update_block_record"); DBUG_PRINT("enter", ("rowid: %lu", (long) record_pos)); @@ -2621,8 +2633,8 @@ my_bool _ma_update_block_record(MARIA_HA *info, MARIA_RECORD_POS record_pos, row_pos.dir= dir; row_pos.data= buff + uint2korr(dir); row_pos.length= head_length; - res= write_block_record(info, oldrec, record, new_row, blocks, 1, &row_pos); - DBUG_RETURN(res); + DBUG_RETURN(write_block_record(info, oldrec, record, new_row, blocks, 1, + &row_pos)); err: _ma_unpin_all_pages(info, 0); @@ -2710,6 +2722,9 @@ static int delete_dir_entry(uchar *buff, uint block_size, uint record_number, info Maria handler page Page (not file offset!) on which the row is head 1 if this is a head page + from_update 1 if we are called from update. In this case we + leave the page as write locked as we may put + the new row into the old position. NOTES Uses info->keyread_buff @@ -2721,7 +2736,7 @@ static int delete_dir_entry(uchar *buff, uint block_size, uint record_number, static my_bool delete_head_or_tail(MARIA_HA *info, ulonglong page, uint record_number, - my_bool head) + my_bool head, my_bool from_update) { MARIA_SHARE *share= info->s; uint empty_space; @@ -2730,6 +2745,7 @@ static my_bool delete_head_or_tail(MARIA_HA *info, LSN lsn; MARIA_PINNED_PAGE page_link; int res; + enum pagecache_page_lock lock_at_write, lock_at_unpin; DBUG_ENTER("delete_head_or_tail"); info->keyread_buff_used= 1; @@ -2743,6 +2759,17 @@ static my_bool delete_head_or_tail(MARIA_HA *info, page_link.unlock= PAGECACHE_LOCK_WRITE_UNLOCK; push_dynamic(&info->pinned_pages, (void*) &page_link); + if (from_update) + { + lock_at_write= PAGECACHE_LOCK_LEFT_WRITELOCKED; + lock_at_unpin= PAGECACHE_LOCK_WRITE_UNLOCK; + } + else + { + lock_at_write= PAGECACHE_LOCK_WRITE_TO_READ; + lock_at_unpin= PAGECACHE_LOCK_READ_UNLOCK; + } + res= delete_dir_entry(buff, block_size, record_number, &empty_space); if (res < 0) DBUG_RETURN(1); @@ -2769,7 +2796,7 @@ static my_bool delete_head_or_tail(MARIA_HA *info, if (pagecache_write(share->pagecache, &info->dfile, page, 0, buff, share->page_type, - PAGECACHE_LOCK_WRITE_TO_READ, + lock_at_write, PAGECACHE_PIN_LEFT_PINNED, PAGECACHE_WRITE_DELAY, &page_link.link)) DBUG_RETURN(1); @@ -2797,15 +2824,15 @@ static my_bool delete_head_or_tail(MARIA_HA *info, if (pagecache_write(share->pagecache, &info->dfile, page, 0, buff, share->page_type, - PAGECACHE_LOCK_WRITE_TO_READ, + lock_at_write, PAGECACHE_PIN_LEFT_PINNED, PAGECACHE_WRITE_DELAY, &page_link.link)) DBUG_RETURN(1); DBUG_ASSERT(empty_space >= info->s->bitmap.sizes[0]); } - /* Change the lock used when we read the page */ - page_link.unlock= PAGECACHE_LOCK_READ_UNLOCK; + /* The page is pinned with a read lock */ + page_link.unlock= lock_at_unpin; set_dynamic(&info->pinned_pages, (void*) &page_link, info->pinned_pages.elements-1); @@ -2838,7 +2865,7 @@ static my_bool delete_tails(MARIA_HA *info, MARIA_RECORD_POS *tails) { if (delete_head_or_tail(info, ma_recordpos_to_page(*tails), - ma_recordpos_to_dir_entry(*tails), 0)) + ma_recordpos_to_dir_entry(*tails), 0, 1)) res= 1; } DBUG_RETURN(res); @@ -2863,7 +2890,7 @@ my_bool _ma_delete_block_record(MARIA_HA *info, const uchar *record) page= ma_recordpos_to_page(info->cur_row.lastpos); record_number= ma_recordpos_to_dir_entry(info->cur_row.lastpos); - if (delete_head_or_tail(info, page, record_number, 1) || + if (delete_head_or_tail(info, page, record_number, 1, 0) || delete_tails(info, info->cur_row.tail_positions)) goto err; @@ -2987,8 +3014,14 @@ static void init_extent(MARIA_EXTENT_CURSOR *extent, uchar *extent_info, extent->extent_count= extents; extent->page= page_korr(extent_info); /* First extent */ page_count= uint2korr(extent_info + ROW_EXTENT_PAGE_SIZE); - extent->page_count= page_count & ~TAIL_BIT; extent->tail= page_count & TAIL_BIT; + if (extent->tail) + { + extent->page_count= 1; + extent->tail_row_nr= page_count & ~TAIL_BIT; + } + else + extent->page_count= page_count; extent->tail_positions= tail_positions; } @@ -3030,12 +3063,15 @@ static uchar *read_next_extent(MARIA_HA *info, MARIA_EXTENT_CURSOR *extent, if (!page_count) goto crashed; extent->tail= page_count & TAIL_BIT; - extent->page_count= (page_count & ~TAIL_BIT); - extent->first_extent= 0; + if (extent->tail) + extent->tail_row_nr= page_count & ~TAIL_BIT; + else + extent->page_count= page_count; DBUG_PRINT("info",("New extent. Page: %lu page_count: %u tail_flag: %d", (ulong) extent->page, extent->page_count, extent->tail != 0)); } + extent->first_extent= 0; DBUG_ASSERT(share->pagecache->block_size == share->block_size); if (!(buff= pagecache_read(share->pagecache, @@ -3059,16 +3095,16 @@ static uchar *read_next_extent(MARIA_HA *info, MARIA_EXTENT_CURSOR *extent, info->cur_row.full_page_count++; /* For maria_chk */ DBUG_RETURN(extent->data_start= buff + LSN_SIZE + PAGE_TYPE_SIZE); } - /* Found tail. page_count is in this case the position in the tail page */ + /* Found tail */ if ((buff[PAGE_TYPE_OFFSET] & PAGE_TYPE_MASK) != TAIL_PAGE) goto crashed; *(extent->tail_positions++)= ma_recordpos(extent->page, - extent->page_count); + extent->tail_row_nr); info->cur_row.tail_count++; /* For maria_chk */ if (!(data= get_record_position(buff, share->block_size, - extent->page_count, + extent->tail_row_nr, end_of_data))) goto crashed; extent->data_start= data; @@ -3124,7 +3160,7 @@ static my_bool read_long_data(MARIA_HA *info, uchar *to, ulong length, This may change in the future, which is why we have the loop written the way it's written. */ - if (length > (ulong) (*end_of_data - *data)) + if (extent->first_extent && length > (ulong) (*end_of_data - *data)) *end_of_data= *data; for(;;) diff --git a/storage/maria/ma_check.c b/storage/maria/ma_check.c index 1ac1fb3454f..a5e64cb555c 100644 --- a/storage/maria/ma_check.c +++ b/storage/maria/ma_check.c @@ -1706,8 +1706,13 @@ static int check_block_record(HA_CHECK *param, MARIA_HA *info, int extend, full_dir ? 0 : empty_space, &bitmap_pattern)) { + if (bitmap_pattern == ~(uint) 0) + _ma_check_print_error(param, + "Page: %9s: Wrong bitmap for data on page", + llstr(pos, llbuff)); + else _ma_check_print_error(param, - "Page %9s: Wrong data in bitmap. Page_type: %d empty_space: %u Bitmap: %d", + "Page %9s: Wrong data in bitmap. Page_type: %d empty_space: %u Bitmap-bits: %d", llstr(pos, llbuff), page_type, empty_space, bitmap_pattern); if (param->err_count++ > MAXERR || !(param->testflag & T_VERBOSE)) diff --git a/storage/maria/ma_close.c b/storage/maria/ma_close.c index a9d31a6c75f..cc9f0005a4d 100644 --- a/storage/maria/ma_close.c +++ b/storage/maria/ma_close.c @@ -65,6 +65,7 @@ int maria_close(register MARIA_HA *info) if (flag) { + /* Last close of file; Flush everything */ if (share->kfile.file >= 0) { if ((*share->once_end)(share)) @@ -87,7 +88,7 @@ int maria_close(register MARIA_HA *info) may be using the file at this point IF using --external-locking, which does not apply to Maria. */ - if (share->mode != O_RDONLY) + if (share->changed) _ma_state_info_write(share->kfile.file, &share->state, 1); if (my_close(share->kfile.file, MYF(0))) error= my_errno; diff --git a/storage/maria/ma_dynrec.c b/storage/maria/ma_dynrec.c index 246f9787b09..52ade04db98 100644 --- a/storage/maria/ma_dynrec.c +++ b/storage/maria/ma_dynrec.c @@ -1694,7 +1694,7 @@ int _ma_read_rnd_dynamic_record(MARIA_HA *info, { /* Check if changed */ info_read=1; info->rec_cache.seek_not_done=1; - if (_ma_state_info_read_dsk(share->kfile.file, &share->state, 1)) + if (_ma_state_info_read_dsk(share->kfile.file, &share->state)) goto panic; } if (filepos >= info->state->data_file_length) diff --git a/storage/maria/ma_extra.c b/storage/maria/ma_extra.c index 0ee5990844b..a3fb9569290 100644 --- a/storage/maria/ma_extra.c +++ b/storage/maria/ma_extra.c @@ -264,8 +264,8 @@ int maria_extra(MARIA_HA *info, enum ha_extra_function function, share->last_version= 0L; /* Impossible version */ pthread_mutex_unlock(&THR_LOCK_maria); break; - case HA_EXTRA_PREPARE_FOR_DELETE: - /* QQ: suggest to rename it to "PREPARE_FOR_DROP" */ + case HA_EXTRA_PREPARE_FOR_DROP: + case HA_EXTRA_PREPARE_FOR_RENAME: pthread_mutex_lock(&THR_LOCK_maria); share->last_version= 0L; /* Impossible version */ #ifdef __WIN__ @@ -284,7 +284,8 @@ int maria_extra(MARIA_HA *info, enum ha_extra_function function, Does ENABLE KEYS rebuild them too? */ if (flush_pagecache_blocks(share->pagecache, &share->kfile, - FLUSH_IGNORE_CHANGED)) + (function == HA_EXTRA_PREPARE_FOR_DROP ? + FLUSH_IGNORE_CHANGED : FLUSH_RELEASE))) { error=my_errno; share->changed=1; diff --git a/storage/maria/ma_locking.c b/storage/maria/ma_locking.c index f709d7e5759..dad4071edf8 100644 --- a/storage/maria/ma_locking.c +++ b/storage/maria/ma_locking.c @@ -153,7 +153,7 @@ int maria_lock_database(MARIA_HA *info, int lock_type) } if (!share->r_locks && !share->w_locks) { - if (_ma_state_info_read_dsk(share->kfile.file, &share->state, 1)) + if (_ma_state_info_read_dsk(share->kfile.file, &share->state)) { error=my_errno; break; @@ -181,7 +181,7 @@ int maria_lock_database(MARIA_HA *info, int lock_type) { if (!share->r_locks) { - if (_ma_state_info_read_dsk(share->kfile.file, &share->state, 1)) + if (_ma_state_info_read_dsk(share->kfile.file, &share->state)) { error=my_errno; break; @@ -364,7 +364,7 @@ int _ma_readinfo(register MARIA_HA *info, int lock_type, int check_keybuffer) MARIA_SHARE *share=info->s; if (!share->tot_locks) { - if (_ma_state_info_read_dsk(share->kfile.file, &share->state, 1)) + if (_ma_state_info_read_dsk(share->kfile.file, &share->state)) { int error=my_errno ? my_errno : -1; my_errno=error; diff --git a/storage/maria/ma_open.c b/storage/maria/ma_open.c index 6cdfadf5d6d..4c623ac56f3 100644 --- a/storage/maria/ma_open.c +++ b/storage/maria/ma_open.c @@ -1110,18 +1110,13 @@ uchar *_ma_state_info_read(uchar *ptr, MARIA_STATE_INFO *state) @param pRead if true, use my_pread(), otherwise my_read() */ -uint _ma_state_info_read_dsk(File file, MARIA_STATE_INFO *state, my_bool pRead) +uint _ma_state_info_read_dsk(File file, MARIA_STATE_INFO *state) { char buff[MARIA_STATE_INFO_SIZE + MARIA_STATE_EXTRA_SIZE]; if (!maria_single_user) { - if (pRead) - { - if (my_pread(file, buff, state->state_length,0L, MYF(MY_NABP))) - return 1; - } - else if (my_read(file, buff, state->state_length,MYF(MY_NABP))) + if (my_pread(file, buff, state->state_length, 0L, MYF(MY_NABP))) return 1; _ma_state_info_read(buff, state); } diff --git a/storage/maria/ma_pagecache.c b/storage/maria/ma_pagecache.c index eb1d1ad9b0a..735f1aeb504 100755 --- a/storage/maria/ma_pagecache.c +++ b/storage/maria/ma_pagecache.c @@ -300,23 +300,24 @@ struct st_pagecache_block_link *next_changed, **prev_changed; /* for lists of file dirty/clean blocks */ struct st_pagecache_hash_link *hash_link; /* backward ptr to referring hash_link */ +#ifndef DBUG_OFF + PAGECACHE_PIN_INFO *pin_list; + PAGECACHE_LOCK_INFO *lock_list; +#endif + KEYCACHE_CONDVAR *condvar; /* condition variable for 'no readers' event */ + uchar *buffer; /* buffer for the block page */ + PAGECACHE_FILE *write_locker; + ulonglong last_hit_time; /* timestamp of the last hit */ WQUEUE wqueue[COND_SIZE]; /* queues on waiting requests for new/old pages */ uint requests; /* number of requests for the block */ - uchar *buffer; /* buffer for the block page */ uint status; /* state of the block */ uint pins; /* pin counter */ -#ifndef DBUG_OFF - PAGECACHE_PIN_INFO *pin_list; - PAGECACHE_LOCK_INFO *lock_list; -#endif enum PCBLOCK_TEMPERATURE temperature; /* block temperature: cold, warm, hot */ enum pagecache_page_type type; /* type of the block */ uint hits_left; /* number of hits left until promotion */ - ulonglong last_hit_time; /* timestamp of the last hit */ /** @brief LSN when first became dirty; LSN_MAX means "not yet set" */ LSN rec_lsn; - KEYCACHE_CONDVAR *condvar; /* condition variable for 'no readers' event */ }; #ifndef DBUG_OFF @@ -2147,6 +2148,9 @@ static void info_change_lock(PAGECACHE_BLOCK_LINK *block, my_bool wl) get_wrlock() pagecache pointer to a page cache data structure block the block to work with + user_file Unique handler per handler file. Used to check if + we request many write locks withing the same + statement RETURN 0 - OK @@ -2154,7 +2158,8 @@ static void info_change_lock(PAGECACHE_BLOCK_LINK *block, my_bool wl) */ static my_bool get_wrlock(PAGECACHE *pagecache, - PAGECACHE_BLOCK_LINK *block) + PAGECACHE_BLOCK_LINK *block, + PAGECACHE_FILE *user_file) { PAGECACHE_FILE file= block->hash_link->file; pgcache_page_no_t pageno= block->hash_link->pageno; @@ -2165,7 +2170,7 @@ static my_bool get_wrlock(PAGECACHE *pagecache, file.file, block->hash_link->file.file, pageno, block->hash_link->pageno)); PCBLOCK_INFO(block); - while (block->status & PCBLOCK_WRLOCK) + while ((block->status & PCBLOCK_WRLOCK) && block->write_locker != user_file) { /* Lock failed we will wait */ #ifdef THREAD @@ -2197,9 +2202,9 @@ static my_bool get_wrlock(PAGECACHE *pagecache, DBUG_RETURN(1); } } - DBUG_ASSERT(block->pins == 0); /* we are doing it by global cache mutex protection, so it is OK */ block->status|= PCBLOCK_WRLOCK; + block->write_locker= user_file; DBUG_PRINT("info", ("WR lock set, block 0x%lx", (ulong)block)); DBUG_RETURN(0); } @@ -2223,6 +2228,8 @@ static void release_wrlock(PAGECACHE_BLOCK_LINK *block) PCBLOCK_INFO(block); DBUG_ASSERT(block->status & PCBLOCK_WRLOCK); DBUG_ASSERT(block->pins > 0); + if (block->pins > 1) + DBUG_VOID_RETURN; /* Multiple write locked */ block->status&= ~PCBLOCK_WRLOCK; DBUG_PRINT("info", ("WR lock reset, block 0x%lx", (ulong)block)); #ifdef THREAD @@ -2244,6 +2251,7 @@ static void release_wrlock(PAGECACHE_BLOCK_LINK *block) block the block to work with lock lock change mode pin pinchange mode + file File handler requesting pin RETURN 0 - OK @@ -2253,7 +2261,8 @@ static void release_wrlock(PAGECACHE_BLOCK_LINK *block) static my_bool make_lock_and_pin(PAGECACHE *pagecache, PAGECACHE_BLOCK_LINK *block, enum pagecache_page_lock lock, - enum pagecache_page_pin pin) + enum pagecache_page_pin pin, + PAGECACHE_FILE *file) { DBUG_ENTER("make_lock_and_pin"); @@ -2274,7 +2283,7 @@ static my_bool make_lock_and_pin(PAGECACHE *pagecache, switch (lock) { case PAGECACHE_LOCK_WRITE: /* free -> write */ /* Writelock and pin the buffer */ - if (get_wrlock(pagecache, block)) + if (get_wrlock(pagecache, block, file)) { /* can't lock => need retry */ goto retry; @@ -2291,6 +2300,7 @@ static my_bool make_lock_and_pin(PAGECACHE *pagecache, implementation) */ release_wrlock(block); + /* fall through */ case PAGECACHE_LOCK_READ_UNLOCK: /* read -> free */ case PAGECACHE_LOCK_LEFT_READLOCKED: /* read -> read */ if (pin == PAGECACHE_UNPIN) @@ -2549,7 +2559,7 @@ void pagecache_unlock(PAGECACHE *pagecache, if (lsn != LSN_IMPOSSIBLE) check_and_set_lsn(pagecache, lsn, block); - if (make_lock_and_pin(pagecache, block, lock, pin)) + if (make_lock_and_pin(pagecache, block, lock, pin, file)) { DBUG_ASSERT(0); /* should not happend */ } @@ -2617,7 +2627,7 @@ void pagecache_unpin(PAGECACHE *pagecache, */ if (make_lock_and_pin(pagecache, block, PAGECACHE_LOCK_LEFT_READLOCKED, - PAGECACHE_UNPIN)) + PAGECACHE_UNPIN, file)) DBUG_ASSERT(0); /* should not happend */ remove_reader(block); @@ -2678,7 +2688,7 @@ void pagecache_unlock_by_link(PAGECACHE *pagecache, lock == PAGECACHE_LOCK_READ_UNLOCK) { /* block do not need here so we do not provide it */ - if (make_lock_and_pin(pagecache, 0, lock, pin)) + if (make_lock_and_pin(pagecache, 0, lock, pin, 0)) DBUG_ASSERT(0); /* should not happend */ DBUG_VOID_RETURN; } @@ -2710,7 +2720,7 @@ void pagecache_unlock_by_link(PAGECACHE *pagecache, if (lsn != LSN_IMPOSSIBLE) check_and_set_lsn(pagecache, lsn, block); - if (make_lock_and_pin(pagecache, block, lock, pin)) + if (make_lock_and_pin(pagecache, block, lock, pin, 0)) DBUG_ASSERT(0); /* should not happend */ /* @@ -2772,7 +2782,7 @@ void pagecache_unpin_by_link(PAGECACHE *pagecache, */ if (make_lock_and_pin(pagecache, block, PAGECACHE_LOCK_LEFT_READLOCKED, - PAGECACHE_UNPIN)) + PAGECACHE_UNPIN, 0)) DBUG_ASSERT(0); /* should not happend */ /* @@ -2889,7 +2899,7 @@ restart: validator, validator_data); DBUG_PRINT("info", ("read is done")); } - if (make_lock_and_pin(pagecache, block, lock, pin)) + if (make_lock_and_pin(pagecache, block, lock, pin, file)) { /* We failed to write lock the block, cache is unlocked, @@ -3009,7 +3019,7 @@ restart: if (pin == PAGECACHE_PIN) reg_requests(pagecache, block, 1); DBUG_ASSERT(block != 0); - if (make_lock_and_pin(pagecache, block, lock, pin)) + if (make_lock_and_pin(pagecache, block, lock, pin, file)) { /* We failed to writelock the block, cache is unlocked, and last write @@ -3059,7 +3069,7 @@ restart: /* Cache is locked, so we can relese page before freeing it */ make_lock_and_pin(pagecache, block, PAGECACHE_LOCK_WRITE_UNLOCK, - PAGECACHE_UNPIN); + PAGECACHE_UNPIN, file); DBUG_ASSERT(link->requests > 0); link->requests--; /* See NOTE for pagecache_unlock about registering requests. */ @@ -3254,7 +3264,7 @@ restart: write_lock_change_table[lock].new_lock, (need_lock_change ? write_pin_change_table[pin].new_pin : - pin))) + pin), file)) { /* We failed to writelock the block, cache is unlocked, and last write @@ -3307,7 +3317,6 @@ restart: } } - if (need_lock_change) { /* @@ -3316,7 +3325,7 @@ restart: */ if (make_lock_and_pin(pagecache, block, write_lock_change_table[lock].unlock_lock, - write_pin_change_table[pin].unlock_pin)) + write_pin_change_table[pin].unlock_pin, file)) DBUG_ASSERT(0); } @@ -3474,7 +3483,7 @@ static int flush_cached_blocks(PAGECACHE *pagecache, DBUG_ASSERT((block->status & PCBLOCK_WRLOCK) == 0); DBUG_ASSERT(block->pins == 0); if (make_lock_and_pin(pagecache, block, - PAGECACHE_LOCK_WRITE, PAGECACHE_PIN)) + PAGECACHE_LOCK_WRITE, PAGECACHE_PIN, 0)) DBUG_ASSERT(0); KEYCACHE_DBUG_PRINT("flush_cached_blocks", @@ -3497,7 +3506,7 @@ static int flush_cached_blocks(PAGECACHE *pagecache, make_lock_and_pin(pagecache, block, PAGECACHE_LOCK_WRITE_UNLOCK, - PAGECACHE_UNPIN); + PAGECACHE_UNPIN, 0); pagecache->global_cache_write++; if (error) diff --git a/storage/maria/ma_recovery.c b/storage/maria/ma_recovery.c index b45346725e6..d5d758da4d6 100644 --- a/storage/maria/ma_recovery.c +++ b/storage/maria/ma_recovery.c @@ -507,7 +507,7 @@ prototype_exec_hook(REDO_DROP_TABLE) this table should not be used anymore, and (only on Windows) to close open files so they can be deleted */ - if (maria_extra(info, HA_EXTRA_PREPARE_FOR_DELETE, NULL) || + if (maria_extra(info, HA_EXTRA_PREPARE_FOR_DROP, NULL) || maria_close(info)) goto end; info= NULL; diff --git a/storage/maria/maria_def.h b/storage/maria/maria_def.h index ab2546b72f3..f5e143fe6f8 100644 --- a/storage/maria/maria_def.h +++ b/storage/maria/maria_def.h @@ -864,8 +864,7 @@ extern uint _ma_nommap_pwrite(MARIA_HA *info, uchar *Buffer, uint _ma_state_info_write(File file, MARIA_STATE_INFO *state, uint pWrite); uchar *_ma_state_info_read(uchar *ptr, MARIA_STATE_INFO *state); -uint _ma_state_info_read_dsk(File file, MARIA_STATE_INFO *state, - my_bool pRead); +uint _ma_state_info_read_dsk(File file, MARIA_STATE_INFO *state); uint _ma_base_info_write(File file, MARIA_BASE_INFO *base); int _ma_keyseg_write(File file, const HA_KEYSEG *keyseg); char *_ma_keyseg_read(char *ptr, HA_KEYSEG *keyseg); diff --git a/storage/myisam/mi_extra.c b/storage/myisam/mi_extra.c index 72c40741a22..f425d41a27e 100644 --- a/storage/myisam/mi_extra.c +++ b/storage/myisam/mi_extra.c @@ -255,15 +255,16 @@ int mi_extra(MI_INFO *info, enum ha_extra_function function, void *extra_arg) share->last_version= 0L; /* Impossible version */ pthread_mutex_unlock(&THR_LOCK_myisam); break; - case HA_EXTRA_PREPARE_FOR_DELETE: + case HA_EXTRA_PREPARE_FOR_RENAME: + case HA_EXTRA_PREPARE_FOR_DROP: pthread_mutex_lock(&THR_LOCK_myisam); share->last_version= 0L; /* Impossible version */ #ifdef __WIN__ /* Close the isam and data files as Win32 can't drop an open table */ pthread_mutex_lock(&share->intern_lock); if (flush_key_blocks(share->key_cache, share->kfile, - (function == HA_EXTRA_FORCE_REOPEN ? - FLUSH_RELEASE : FLUSH_IGNORE_CHANGED))) + (function == HA_EXTRA_PREPARE_FOR_DROP ? + FLUSH_IGNORE_CHANGED : FLUSH_RELEASE))) { error=my_errno; share->changed=1; diff --git a/storage/myisammrg/ha_myisammrg.cc b/storage/myisammrg/ha_myisammrg.cc index f3df1e82c4b..601c2479473 100644 --- a/storage/myisammrg/ha_myisammrg.cc +++ b/storage/myisammrg/ha_myisammrg.cc @@ -392,7 +392,8 @@ int ha_myisammrg::extra(enum ha_extra_function operation) /* As this is just a mapping, we don't have to force the underlying tables to be closed */ if (operation == HA_EXTRA_FORCE_REOPEN || - operation == HA_EXTRA_PREPARE_FOR_DELETE) + operation == HA_EXTRA_PREPARE_FOR_DROP || + operation == HA_EXTRA_PREPARE_FOR_RENAME) return 0; return myrg_extra(file,operation,0); } -- cgit v1.2.1 From 459943d7978abe34abd2c37b159b45b13cd891b6 Mon Sep 17 00:00:00 2001 From: unknown Date: Mon, 3 Sep 2007 16:55:46 +0300 Subject: fixed possible problem with multiply write locking (write counter added). --- storage/maria/ma_pagecache.c | 36 ++++++++++++++++++------------------ 1 file changed, 18 insertions(+), 18 deletions(-) diff --git a/storage/maria/ma_pagecache.c b/storage/maria/ma_pagecache.c index 735f1aeb504..ac341890296 100755 --- a/storage/maria/ma_pagecache.c +++ b/storage/maria/ma_pagecache.c @@ -96,7 +96,7 @@ #define PCBLOCK_INFO(B) \ DBUG_PRINT("info", \ ("block: 0x%lx file: %lu page: %lu s: %0x hshL: 0x%lx req: %u/%u " \ - "wrlock: %c", \ + "wrlocks: %u", \ (ulong)(B), \ (ulong)((B)->hash_link ? \ (B)->hash_link->file.file : \ @@ -110,7 +110,7 @@ (uint)((B)->hash_link ? \ (B)->hash_link->requests : \ 0), \ - ((block->status & PCBLOCK_WRLOCK)?'Y':'N'))) + block->wlocks)) /* TODO: put it to my_static.c */ my_bool my_disable_flush_pagecache_blocks= 0; @@ -160,7 +160,6 @@ struct st_pagecache_hash_link #define PCBLOCK_REASSIGNED 8 /* block does not accept requests for old page */ #define PCBLOCK_IN_FLUSH 16 /* block is in flush operation */ #define PCBLOCK_CHANGED 32 /* block buffer contains a dirty page */ -#define PCBLOCK_WRLOCK 64 /* write locked block */ /* page status, returned by find_block */ #define PAGE_READ 0 @@ -313,6 +312,7 @@ struct st_pagecache_block_link uint requests; /* number of requests for the block */ uint status; /* state of the block */ uint pins; /* pin counter */ + uint wlocks; /* write locks counter */ enum PCBLOCK_TEMPERATURE temperature; /* block temperature: cold, warm, hot */ enum pagecache_page_type type; /* type of the block */ uint hits_left; /* number of hits left until promotion */ @@ -1884,7 +1884,7 @@ restart: pagecache->blocks_used++; } pagecache->blocks_unused--; - DBUG_ASSERT((block->status & PCBLOCK_WRLOCK) == 0); + DBUG_ASSERT(block->wlocks == 0); DBUG_ASSERT(block->pins == 0); block->status= 0; #ifndef DBUG_OFF @@ -1949,16 +1949,16 @@ restart: hash_link->block= block; } PCBLOCK_INFO(block); - DBUG_ASSERT((block->status & PCBLOCK_WRLOCK) == 0); + DBUG_ASSERT(block->wlocks == 0); DBUG_ASSERT(block->pins == 0); if (block->hash_link != hash_link && ! (block->status & PCBLOCK_IN_SWITCH) ) { /* this is a primary request for a new page */ - DBUG_ASSERT((block->status & PCBLOCK_WRLOCK) == 0); + DBUG_ASSERT(block->wlocks == 0); DBUG_ASSERT(block->pins == 0); - block->status|= (PCBLOCK_IN_SWITCH | PCBLOCK_WRLOCK); + block->status|= PCBLOCK_IN_SWITCH; KEYCACHE_DBUG_PRINT("find_block", ("got block %u for new page", @@ -2170,7 +2170,7 @@ static my_bool get_wrlock(PAGECACHE *pagecache, file.file, block->hash_link->file.file, pageno, block->hash_link->pageno)); PCBLOCK_INFO(block); - while ((block->status & PCBLOCK_WRLOCK) && block->write_locker != user_file) + while (block->wlocks && block->write_locker != user_file) { /* Lock failed we will wait */ #ifdef THREAD @@ -2203,7 +2203,7 @@ static my_bool get_wrlock(PAGECACHE *pagecache, } } /* we are doing it by global cache mutex protection, so it is OK */ - block->status|= PCBLOCK_WRLOCK; + block->wlocks++; block->write_locker= user_file; DBUG_PRINT("info", ("WR lock set, block 0x%lx", (ulong)block)); DBUG_RETURN(0); @@ -2226,11 +2226,11 @@ static void release_wrlock(PAGECACHE_BLOCK_LINK *block) { DBUG_ENTER("release_wrlock"); PCBLOCK_INFO(block); - DBUG_ASSERT(block->status & PCBLOCK_WRLOCK); + DBUG_ASSERT(block->wlocks > 0); DBUG_ASSERT(block->pins > 0); - if (block->pins > 1) + block->wlocks--; + if (block->wlocks > 0) DBUG_VOID_RETURN; /* Multiple write locked */ - block->status&= ~PCBLOCK_WRLOCK; DBUG_PRINT("info", ("WR lock reset, block 0x%lx", (ulong)block)); #ifdef THREAD /* release all threads waiting for write lock */ @@ -2270,9 +2270,9 @@ static my_bool make_lock_and_pin(PAGECACHE *pagecache, #ifndef DBUG_OFF if (block) { - DBUG_PRINT("enter", ("block: 0x%lx (%u) wrlock: %c pins: %u lock: %s pin: %s", + DBUG_PRINT("enter", ("block: 0x%lx (%u) wrlocks: %u pins: %u lock: %s pin: %s", (ulong)block, PCBLOCK_NUMBER(pagecache, block), - ((block->status & PCBLOCK_WRLOCK)?'Y':'N'), + block->wlocks, block->pins, page_cache_page_lock_str[lock], page_cache_page_pin_str[pin])); @@ -2406,7 +2406,7 @@ static void read_block(PAGECACHE *pagecache, if (got_length < pagecache->block_size) block->status|= PCBLOCK_ERROR; else - block->status= (PCBLOCK_READ | (block->status & PCBLOCK_WRLOCK)); + block->status= PCBLOCK_READ; if (validator != NULL && (*validator)(block->buffer, validator_data)) @@ -3284,7 +3284,7 @@ restart: bmove512(block->buffer + offset, buff, size); else memcpy(block->buffer + offset, buff, size); - block->status= (PCBLOCK_READ | (block->status & PCBLOCK_WRLOCK)); + block->status= PCBLOCK_READ; /* The validator can change the page content (removing page protection) so it have to be called @@ -3399,7 +3399,7 @@ static void free_block(PAGECACHE *pagecache, PAGECACHE_BLOCK_LINK *block) } unlink_changed(block); - DBUG_ASSERT((block->status & PCBLOCK_WRLOCK) == 0); + DBUG_ASSERT(block->wlocks == 0); DBUG_ASSERT(block->pins == 0); block->status= 0; #ifndef DBUG_OFF @@ -3480,7 +3480,7 @@ static int flush_cached_blocks(PAGECACHE *pagecache, continue; } /* if the block is not pinned then it is not write locked */ - DBUG_ASSERT((block->status & PCBLOCK_WRLOCK) == 0); + DBUG_ASSERT(block->wlocks == 0); DBUG_ASSERT(block->pins == 0); if (make_lock_and_pin(pagecache, block, PAGECACHE_LOCK_WRITE, PAGECACHE_PIN, 0)) -- cgit v1.2.1 From fe8920365b4450a03b7bb3f12636852aecd93806 Mon Sep 17 00:00:00 2001 From: unknown Date: Tue, 4 Sep 2007 08:38:52 +0300 Subject: Spelling of comments fixed. --- storage/maria/ma_pagecache.c | 8 ++++---- 1 file changed, 4 insertions(+), 4 deletions(-) diff --git a/storage/maria/ma_pagecache.c b/storage/maria/ma_pagecache.c index ac341890296..17981840cc8 100755 --- a/storage/maria/ma_pagecache.c +++ b/storage/maria/ma_pagecache.c @@ -3458,7 +3458,7 @@ static int flush_cached_blocks(PAGECACHE *pagecache, pagecache_pthread_mutex_unlock(&pagecache->cache_lock); /* As all blocks referred in 'cache' are marked by PCBLOCK_IN_FLUSH - we are guarunteed no thread will change them + we are guarantied no thread will change them */ qsort((uchar*) cache, count, sizeof(*cache), (qsort_cmp) cmp_sec_link); @@ -3633,7 +3633,7 @@ restart: /* Mark the block with BLOCK_IN_FLUSH in order not to let other threads to use it for new pages and interfere with - our sequence ot flushing dirty file pages + our sequence of flushing dirty file pages */ block->status|= PCBLOCK_IN_FLUSH; @@ -3790,7 +3790,7 @@ restart: flush_type type of the flush RETURN - 0 ok + 0 OK 1 error */ @@ -3907,7 +3907,7 @@ my_bool pagecache_collect_changed_blocks_with_lsn(PAGECACHE *pagecache, block= block->next_changed) { /* - Q: is there somthing subtle with block->hash_link: can it be NULL? + Q: is there something subtle with block->hash_link: can it be NULL? does it have to be == hash_link->block... ? */ DBUG_ASSERT(block->hash_link != NULL); -- cgit v1.2.1 From 03437ea043f4f57d8910bea98b3a0c0412b91385 Mon Sep 17 00:00:00 2001 From: unknown Date: Tue, 4 Sep 2007 10:53:52 +0300 Subject: Added UNDO handling of insert during recovery storage/maria/ma_blockrec.c: Added UNDO handling of insert during recovery To do this, I also had to add write locking of tail pages during undo phase (As we need to access the same page twice if extents are split over two pages) Another way to handle the undo of insert would be to store the extent information as part of the UNDO_INSERT block. storage/maria/ma_blockrec.h: Added new prototype storage/maria/ma_loghandler.c: Changed type of CLR_END (to avoid crash in log handler) Removed not used variable storage/maria/ma_loghandler.h: Added TRN argument to record_execute_in_undo_phase() storage/maria/ma_pagecache.c: Hack for undo phase of recovery. During REDO we work with PLAIN pages, but UNDO works with LSN pages, which caused an abort when trying to access a cached page. storage/maria/ma_recovery.c: Added execution of UNDO_ROW_INSERT storage/maria/ma_test1.c: Added option --test-undo for testing recovery with undo storage/maria/maria_read_log.c: Added processing of undos --- storage/maria/ma_blockrec.c | 187 +++++++++++++++++++++++++++++++++++++++-- storage/maria/ma_blockrec.h | 2 + storage/maria/ma_loghandler.c | 4 +- storage/maria/ma_loghandler.h | 6 +- storage/maria/ma_pagecache.c | 1 + storage/maria/ma_recovery.c | 178 +++++++++++++++++++++++++-------------- storage/maria/ma_test1.c | 37 +++++++- storage/maria/maria_read_log.c | 3 +- 8 files changed, 337 insertions(+), 81 deletions(-) diff --git a/storage/maria/ma_blockrec.c b/storage/maria/ma_blockrec.c index 7f29f075463..20c53d0c9fc 100644 --- a/storage/maria/ma_blockrec.c +++ b/storage/maria/ma_blockrec.c @@ -281,9 +281,11 @@ typedef struct st_maria_extent_cursor /* Position to all tails in the row. Updated when reading a row */ MARIA_RECORD_POS *tail_positions; /* Current page */ - my_off_t page; + ulonglong page; /* How many pages in the page region */ uint page_count; + /* What kind of lock to use for tail pages */ + enum pagecache_page_lock lock_for_tail_pages; /* Total number of extents (i.e., entries in the 'extent' slot) */ uint extent_count; /* <> 0 if current extent is a tail page; Set while using cursor */ @@ -2435,7 +2437,7 @@ my_bool _ma_write_block_record(MARIA_HA *info __attribute__ ((unused)), /** - @brief Remove row written by _ma_write_block_record() + @brief Remove row written by _ma_write_block_record() and log undo @param info Maria handler @@ -2466,8 +2468,8 @@ my_bool _ma_write_abort_block_record(MARIA_HA *info) if (block->used & BLOCKUSED_TAIL) { /* - block->page_count is set to the tail directory entry number in - write_block_record() + block->page_count is set to the tail directory entry number in + write_block_record() */ if (delete_head_or_tail(info, block->page, block->page_count & ~TAIL_BIT, 0, 0)) @@ -2894,8 +2896,6 @@ my_bool _ma_delete_block_record(MARIA_HA *info, const uchar *record) delete_tails(info, info->cur_row.tail_positions)) goto err; - info->s->state.split--; - if (info->cur_row.extents && free_full_pages(info, &info->cur_row)) goto err; @@ -3023,6 +3023,7 @@ static void init_extent(MARIA_EXTENT_CURSOR *extent, uchar *extent_info, else extent->page_count= page_count; extent->tail_positions= tail_positions; + extent->lock_for_tail_pages= PAGECACHE_LOCK_LEFT_UNLOCKED; } @@ -3050,6 +3051,8 @@ static uchar *read_next_extent(MARIA_HA *info, MARIA_EXTENT_CURSOR *extent, { MARIA_SHARE *share= info->s; uchar *buff, *data; + MARIA_PINNED_PAGE page_link; + enum pagecache_page_lock lock; DBUG_ENTER("read_next_extent"); if (!extent->page_count) @@ -3073,17 +3076,22 @@ static uchar *read_next_extent(MARIA_HA *info, MARIA_EXTENT_CURSOR *extent, } extent->first_extent= 0; + lock= PAGECACHE_LOCK_LEFT_UNLOCKED; + if (extent->tail) + lock= extent->lock_for_tail_pages; + DBUG_ASSERT(share->pagecache->block_size == share->block_size); if (!(buff= pagecache_read(share->pagecache, &info->dfile, extent->page, 0, info->buff, share->page_type, - PAGECACHE_LOCK_LEFT_UNLOCKED, 0))) + lock, &page_link.link))) { /* check if we tried to read over end of file (ie: bad data in record) */ if ((extent->page + 1) * share->block_size > info->state->data_file_length) goto crashed; DBUG_RETURN(0); } + if (!extent->tail) { /* Full data page */ @@ -3095,7 +3103,14 @@ static uchar *read_next_extent(MARIA_HA *info, MARIA_EXTENT_CURSOR *extent, info->cur_row.full_page_count++; /* For maria_chk */ DBUG_RETURN(extent->data_start= buff + LSN_SIZE + PAGE_TYPE_SIZE); } + /* Found tail */ + if (lock != PAGECACHE_LOCK_LEFT_UNLOCKED) + { + /* Read during redo */ + page_link.unlock= PAGECACHE_LOCK_WRITE_UNLOCK; + push_dynamic(&info->pinned_pages, (void*) &page_link); + } if ((buff[PAGE_TYPE_OFFSET] & PAGE_TYPE_MASK) != TAIL_PAGE) goto crashed; @@ -3492,6 +3507,105 @@ err: } +/** @brief Read positions to tail blocks and full blocks + + @fn read_row_extent_info() + @param info Handler + + @notes + This function is a simpler version of _ma_read_block_record2() + The data about the used pages is stored in info->cur_row. + + @return + @retval 0 ok + @retval 1 Error. my_errno contains error number +*/ + +static my_bool read_row_extent_info(MARIA_HA *info, uchar *buff, + uint record_number) +{ + MARIA_SHARE *share= info->s; + uchar *data, *end_of_data; + uint flag, row_extents, field_lengths; + MARIA_EXTENT_CURSOR extent; + DBUG_ENTER("read_row_extent_info"); + + if (!(data= get_record_position(buff, share->block_size, + record_number, &end_of_data))) + DBUG_RETURN(1); /* Wrong in record */ + + flag= (uint) (uchar) data[0]; + /* Skip trans header */ + data+= total_header_size[(flag & PRECALC_HEADER_BITMASK)]; + + row_extents= 0; + if (flag & ROW_FLAG_EXTENTS) + { + uint row_extent_size; + /* + Record is split over many data pages. + Get number of extents and first extent + */ + get_key_length(row_extents, data); + row_extent_size= row_extents * ROW_EXTENT_SIZE; + if (info->cur_row.extents_buffer_length < row_extent_size && + _ma_alloc_buffer(&info->cur_row.extents, + &info->cur_row.extents_buffer_length, + row_extent_size)) + DBUG_RETURN(1); + memcpy(info->cur_row.extents, data, ROW_EXTENT_SIZE); + data+= ROW_EXTENT_SIZE; + init_extent(&extent, info->cur_row.extents, row_extents, + info->cur_row.tail_positions); + extent.first_extent= 1; + } + else + (*info->cur_row.tail_positions)= 0; + info->cur_row.extents_count= row_extents; + + if (share->base.max_field_lengths) + get_key_length(field_lengths, data); + + if (share->calc_checksum) + info->cur_row.checksum= (uint) (uchar) *data++; + if (row_extents > 1) + { + MARIA_RECORD_POS *tail_pos; + uchar *extents, *end; + + data+= share->base.null_bytes; + data+= share->base.pack_bytes; + data+= share->base.field_offsets * FIELD_OFFSET_SIZE; + + /* + Read row extents (note that first extent was already read into + info->cur_row.extents above) + Lock tails with write lock as we will delete them later. + */ + extent.lock_for_tail_pages= PAGECACHE_LOCK_LEFT_WRITELOCKED; + if (read_long_data(info, info->cur_row.extents + ROW_EXTENT_SIZE, + (row_extents - 1) * ROW_EXTENT_SIZE, + &extent, &data, &end_of_data)) + DBUG_RETURN(1); + + /* Update tail_positions with pointer to tails */ + tail_pos= info->cur_row.tail_positions; + for (extents= info->cur_row.extents, end= extents+ row_extents; + extents < end; + extents += ROW_EXTENT_SIZE) + { + ulonglong page= uint5korr(extents); + uint page_count= uint2korr(extents + ROW_EXTENT_PAGE_SIZE); + if (page_count & TAIL_BIT) + *(tail_pos++)= ma_recordpos(page, (page_count & ~TAIL_BIT)); + } + *tail_pos= 0; /* End marker */ + } + DBUG_RETURN(0); +} + + + /* Read a record based on record position @@ -4575,3 +4689,62 @@ uint _ma_apply_redo_purge_blocks(MARIA_HA *info, } DBUG_RETURN(0); } + +/**************************************************************************** + Applying of UNDO entries +****************************************************************************/ + +my_bool _ma_apply_undo_row_insert(MARIA_HA *info, LSN undo_lsn, + const uchar *header) +{ + ulonglong page; + uint record_number; + LEX_STRING log_array[TRANSLOG_INTERNAL_PARTS + 1]; + uchar log_data[LSN_STORE_SIZE + FILEID_STORE_SIZE], *buff; + my_bool res= 1; + MARIA_PINNED_PAGE page_link; + LSN lsn; + DBUG_ENTER("_ma_apply_undo_row_insert"); + + page= page_korr(header); + record_number= dirpos_korr(header + PAGE_STORE_SIZE); + DBUG_PRINT("enter", ("Page: %lu record_number: %u", (ulong) page, + record_number)); + + if (!(buff= pagecache_read(info->s->pagecache, + &info->dfile, page, 0, + info->buff, info->s->page_type, + PAGECACHE_LOCK_WRITE, + &page_link.link))) + DBUG_RETURN(1); + + + page_link.unlock= PAGECACHE_LOCK_WRITE_UNLOCK; + push_dynamic(&info->pinned_pages, (void*) &page_link); + + if (read_row_extent_info(info, buff, record_number)) + DBUG_RETURN(1); + + if (delete_head_or_tail(info, page, record_number, 1, 1) || + delete_tails(info, info->cur_row.tail_positions)) + goto err; + + if (info->cur_row.extents && free_full_pages(info, &info->cur_row)) + goto err; + + lsn_store(log_data + FILEID_STORE_SIZE, undo_lsn); + log_array[TRANSLOG_INTERNAL_PARTS + 0].str= (char*) log_data; + log_array[TRANSLOG_INTERNAL_PARTS + 0].length= sizeof(log_data); + + if (translog_write_record(&lsn, LOGREC_CLR_END, + info->trn, info, sizeof(log_data), + TRANSLOG_INTERNAL_PARTS + 1, log_array, + log_data)) + goto err; + + info->s->state.state.records--; + res= 0; +err: + _ma_unpin_all_pages(info, lsn); + DBUG_RETURN(res); +} diff --git a/storage/maria/ma_blockrec.h b/storage/maria/ma_blockrec.h index 71feb33cabb..eff99355d62 100644 --- a/storage/maria/ma_blockrec.h +++ b/storage/maria/ma_blockrec.h @@ -187,3 +187,5 @@ uint _ma_apply_redo_purge_row_head_or_tail(MARIA_HA *info, LSN lsn, const uchar *header); uint _ma_apply_redo_purge_blocks(MARIA_HA *info, LSN lsn, const uchar *header); +my_bool _ma_apply_undo_row_insert(MARIA_HA *info, LSN undo_lsn, + const uchar *header); diff --git a/storage/maria/ma_loghandler.c b/storage/maria/ma_loghandler.c index 548e22f0ce6..f2afedaf662 100644 --- a/storage/maria/ma_loghandler.c +++ b/storage/maria/ma_loghandler.c @@ -329,7 +329,7 @@ static LOG_DESC INIT_LOGREC_REDO_UNDELETE_ROW= "redo_undelete_row", LOGREC_NOT_LAST_IN_GROUP, NULL, NULL}; static LOG_DESC INIT_LOGREC_CLR_END= -{LOGRECTYPE_PSEUDOFIXEDLENGTH, 5, 5, NULL, write_hook_for_redo, NULL, 1, +{LOGRECTYPE_FIXEDLENGTH, 9, 9, NULL, write_hook_for_redo, NULL, 0, "clr_end", LOGREC_LAST_IN_GROUP, NULL, NULL}; static LOG_DESC INIT_LOGREC_PURGE_END= @@ -6211,7 +6211,6 @@ static my_bool write_hook_for_undo(enum translog_record_type type */ } - /** @brief Gives a 2-byte-id to MARIA_SHARE and logs this fact @@ -6353,7 +6352,6 @@ my_bool translog_is_file(uint file_no) static uint32 translog_first_file(TRANSLOG_ADDRESS horizon, int is_protected) { - TRANSLOG_ADDRESS addr; uint min_file= 1, max_file; DBUG_ENTER("translog_first_file"); if (!is_protected) diff --git a/storage/maria/ma_loghandler.h b/storage/maria/ma_loghandler.h index d2393627964..a4065fedf66 100644 --- a/storage/maria/ma_loghandler.h +++ b/storage/maria/ma_loghandler.h @@ -342,8 +342,8 @@ typedef struct st_log_record_type_descriptor /* HOOK for reading headers */ read_rec_hook read_hook; /* - For pseudo fixed records number of compressed LSNs followed by - system header + For pseudo fixed records number of compressed LSNs followed by + system header */ int16 compressed_LSN; /* the rest is for maria_read_log & Recovery */ @@ -353,7 +353,7 @@ typedef struct st_log_record_type_descriptor /* a function to execute when we see the record during the REDO phase */ int (*record_execute_in_redo_phase)(const TRANSLOG_HEADER_BUFFER *); /* a function to execute when we see the record during the UNDO phase */ - int (*record_execute_in_undo_phase)(const TRANSLOG_HEADER_BUFFER *); + int (*record_execute_in_undo_phase)(const TRANSLOG_HEADER_BUFFER *, TRN *); } LOG_DESC; extern LOG_DESC log_record_type_descriptor[LOGREC_NUMBER_OF_TYPES]; diff --git a/storage/maria/ma_pagecache.c b/storage/maria/ma_pagecache.c index 735f1aeb504..59ed8ed8e09 100755 --- a/storage/maria/ma_pagecache.c +++ b/storage/maria/ma_pagecache.c @@ -2885,6 +2885,7 @@ restart: &page_st); DBUG_ASSERT(block->type == PAGECACHE_EMPTY_PAGE || block->type == type || + type == PAGECACHE_LSN_PAGE || type == PAGECACHE_READ_UNKNOWN_PAGE || block->type == PAGECACHE_READ_UNKNOWN_PAGE); if (type != PAGECACHE_READ_UNKNOWN_PAGE || diff --git a/storage/maria/ma_recovery.c b/storage/maria/ma_recovery.c index d5d758da4d6..2d7a0ad7642 100644 --- a/storage/maria/ma_recovery.c +++ b/storage/maria/ma_recovery.c @@ -49,27 +49,34 @@ static LSN current_group_end_lsn, checkpoint_start= LSN_IMPOSSIBLE; static FILE *tracef; /**< trace file for debugging */ -#define prototype_exec_hook(R) \ - static int exec_LOGREC_ ## R(const TRANSLOG_HEADER_BUFFER *rec) -#define prototype_exec_hook_dummy(R) \ - static int exec_LOGREC_ ## R(const TRANSLOG_HEADER_BUFFER *rec \ +#define prototype_redo_exec_hook(R) \ + static int exec_REDO_LOGREC_ ## R(const TRANSLOG_HEADER_BUFFER *rec) + +#define prototype_redo_exec_hook_dummy(R) \ + static int exec_REDO_LOGREC_ ## R(const TRANSLOG_HEADER_BUFFER *rec \ __attribute ((unused))) -prototype_exec_hook(LONG_TRANSACTION_ID); -prototype_exec_hook_dummy(CHECKPOINT); -prototype_exec_hook(REDO_CREATE_TABLE); -prototype_exec_hook(REDO_DROP_TABLE); -prototype_exec_hook(FILE_ID); -prototype_exec_hook(REDO_INSERT_ROW_HEAD); -prototype_exec_hook(REDO_INSERT_ROW_TAIL); -prototype_exec_hook(REDO_PURGE_ROW_HEAD); -prototype_exec_hook(REDO_PURGE_ROW_TAIL); -prototype_exec_hook(REDO_PURGE_BLOCKS); -prototype_exec_hook(REDO_DELETE_ALL); -prototype_exec_hook(UNDO_ROW_INSERT); -prototype_exec_hook(UNDO_ROW_DELETE); -prototype_exec_hook(UNDO_ROW_UPDATE); -prototype_exec_hook(UNDO_ROW_PURGE); -prototype_exec_hook(COMMIT); + +#define prototype_undo_exec_hook(R) \ + static int exec_UNDO_LOGREC_ ## R(const TRANSLOG_HEADER_BUFFER *rec, TRN *trn) + +prototype_redo_exec_hook(LONG_TRANSACTION_ID); +prototype_redo_exec_hook_dummy(CHECKPOINT); +prototype_redo_exec_hook(REDO_CREATE_TABLE); +prototype_redo_exec_hook(REDO_DROP_TABLE); +prototype_redo_exec_hook(FILE_ID); +prototype_redo_exec_hook(REDO_INSERT_ROW_HEAD); +prototype_redo_exec_hook(REDO_INSERT_ROW_TAIL); +prototype_redo_exec_hook(REDO_PURGE_ROW_HEAD); +prototype_redo_exec_hook(REDO_PURGE_ROW_TAIL); +prototype_redo_exec_hook(REDO_PURGE_BLOCKS); +prototype_redo_exec_hook(REDO_DELETE_ALL); +prototype_redo_exec_hook(UNDO_ROW_INSERT); +prototype_redo_exec_hook(UNDO_ROW_DELETE); +prototype_redo_exec_hook(UNDO_ROW_UPDATE); +prototype_redo_exec_hook(UNDO_ROW_PURGE); +prototype_redo_exec_hook(COMMIT); +prototype_undo_exec_hook(UNDO_ROW_INSERT); + static int run_redo_phase(LSN lsn, my_bool apply); static uint end_of_redo_phase(my_bool prepare_for_undo_phase); static int run_undo_phase(uint unfinished); @@ -159,6 +166,7 @@ int maria_apply_log(LSN from_lsn, my_bool apply, FILE *trace_file, my_bool should_run_undo_phase) { int error= 0; + uint unfinished_trans; DBUG_ENTER("maria_apply_log"); DBUG_ASSERT(apply || !should_run_undo_phase); @@ -199,7 +207,7 @@ int maria_apply_log(LSN from_lsn, my_bool apply, FILE *trace_file, if (run_redo_phase(from_lsn, apply)) goto err; - uint unfinished_trans= end_of_redo_phase(should_run_undo_phase); + unfinished_trans= end_of_redo_phase(should_run_undo_phase); if (unfinished_trans == (uint)-1) goto err; if (should_run_undo_phase) @@ -270,12 +278,12 @@ static int display_and_apply_record(const LOG_DESC *log_desc, return 1; } if ((error= (*log_desc->record_execute_in_redo_phase)(rec))) - fprintf(tracef, "Got error when executing record\n"); + fprintf(tracef, "Got error when executing redo on record\n"); return error; } -prototype_exec_hook(LONG_TRANSACTION_ID) +prototype_redo_exec_hook(LONG_TRANSACTION_ID) { uint16 sid= rec->short_trid; TrID long_trid= all_active_trans[sid].long_trid; @@ -325,14 +333,14 @@ static void new_transaction(uint16 sid, TrID long_id, LSN undo_lsn, } -prototype_exec_hook_dummy(CHECKPOINT) +prototype_redo_exec_hook_dummy(CHECKPOINT) { /* the only checkpoint we care about was found via control file, ignore */ return 0; } -prototype_exec_hook(REDO_CREATE_TABLE) +prototype_redo_exec_hook(REDO_CREATE_TABLE) { File dfile= -1, kfile= -1; char *linkname_ptr, filename[FN_REFLEN]; @@ -458,7 +466,7 @@ end: } -prototype_exec_hook(REDO_DROP_TABLE) +prototype_redo_exec_hook(REDO_DROP_TABLE) { char *name; int error= 1; @@ -528,7 +536,7 @@ end: } -prototype_exec_hook(FILE_ID) +prototype_redo_exec_hook(FILE_ID) { uint16 sid; int error= 1; @@ -656,7 +664,7 @@ end: } -prototype_exec_hook(REDO_INSERT_ROW_HEAD) +prototype_redo_exec_hook(REDO_INSERT_ROW_HEAD) { int error= 1; uchar *buff= NULL; @@ -701,7 +709,7 @@ end: } -prototype_exec_hook(REDO_INSERT_ROW_TAIL) +prototype_redo_exec_hook(REDO_INSERT_ROW_TAIL) { int error= 1; uchar *buff; @@ -737,7 +745,7 @@ end: } -prototype_exec_hook(REDO_PURGE_ROW_HEAD) +prototype_redo_exec_hook(REDO_PURGE_ROW_HEAD) { int error= 1; MARIA_HA *info= get_MARIA_HA_from_REDO_record(rec); @@ -753,7 +761,7 @@ end: } -prototype_exec_hook(REDO_PURGE_ROW_TAIL) +prototype_redo_exec_hook(REDO_PURGE_ROW_TAIL) { int error= 1; MARIA_HA *info= get_MARIA_HA_from_REDO_record(rec); @@ -769,7 +777,7 @@ end: } -prototype_exec_hook(REDO_PURGE_BLOCKS) +prototype_redo_exec_hook(REDO_PURGE_BLOCKS) { int error= 1; uchar *buff; @@ -797,7 +805,7 @@ end: } -prototype_exec_hook(REDO_DELETE_ALL) +prototype_redo_exec_hook(REDO_DELETE_ALL) { int error= 1; MARIA_HA *info= get_MARIA_HA_from_REDO_record(rec); @@ -813,12 +821,12 @@ end: } -#define set_undo_lsn_for_active_trans(I, L) do { \ - all_active_trans[I].undo_lsn= L; \ - if (all_active_trans[I].first_undo_lsn == LSN_IMPOSSIBLE) \ - all_active_trans[I].first_undo_lsn= L; } while (0) +#define set_undo_lsn_for_active_trans(TRID, LSN) do { \ + all_active_trans[TRID].undo_lsn= LSN; \ + if (all_active_trans[TRID].first_undo_lsn == LSN_IMPOSSIBLE) \ + all_active_trans[TRID].first_undo_lsn= LSN; } while (0) -prototype_exec_hook(UNDO_ROW_INSERT) +prototype_redo_exec_hook(UNDO_ROW_INSERT) { int error= 1; MARIA_HA *info= get_MARIA_HA_from_UNDO_record(rec); @@ -843,12 +851,13 @@ prototype_exec_hook(UNDO_ROW_INSERT) } fprintf(tracef, " rows' count %lu\n", (ulong)info->s->state.state.records); error= 0; + end: return error; } -prototype_exec_hook(UNDO_ROW_DELETE) +prototype_redo_exec_hook(UNDO_ROW_DELETE) { int error= 1; MARIA_HA *info= get_MARIA_HA_from_UNDO_record(rec); @@ -868,7 +877,7 @@ end: } -prototype_exec_hook(UNDO_ROW_UPDATE) +prototype_redo_exec_hook(UNDO_ROW_UPDATE) { int error= 1; MARIA_HA *info= get_MARIA_HA_from_UNDO_record(rec); @@ -885,7 +894,7 @@ end: } -prototype_exec_hook(UNDO_ROW_PURGE) +prototype_redo_exec_hook(UNDO_ROW_PURGE) { int error= 1; MARIA_HA *info= get_MARIA_HA_from_UNDO_record(rec); @@ -906,7 +915,7 @@ end: } -prototype_exec_hook(COMMIT) +prototype_redo_exec_hook(COMMIT) { uint16 sid= rec->short_trid; TrID long_trid= all_active_trans[sid].long_trid; @@ -945,28 +954,55 @@ prototype_exec_hook(COMMIT) } +prototype_undo_exec_hook(UNDO_ROW_INSERT) +{ + my_bool error; + MARIA_HA *info= get_MARIA_HA_from_UNDO_record(rec); + + if (info == NULL) + return 1; + + info->s->state.changed|= STATE_CHANGED | STATE_NOT_ANALYZED | + STATE_NOT_OPTIMIZED_KEYS | STATE_NOT_SORTED_PAGES; + + /* Set undo to point to previous undo record */ + info->trn= trn; + info->trn->undo_lsn= lsn_korr(rec->header); + + error= _ma_apply_undo_row_insert(info, rec->lsn, + rec->header + LSN_STORE_SIZE + + FILEID_STORE_SIZE); + info->trn= 0; + return error; +} + + static int run_redo_phase(LSN lsn, my_bool apply) { /* install hooks for execution */ -#define install_exec_hook(R) \ +#define install_redo_exec_hook(R) \ log_record_type_descriptor[LOGREC_ ## R].record_execute_in_redo_phase= \ - exec_LOGREC_ ## R; - install_exec_hook(LONG_TRANSACTION_ID); - install_exec_hook(CHECKPOINT); - install_exec_hook(REDO_CREATE_TABLE); - install_exec_hook(REDO_DROP_TABLE); - install_exec_hook(FILE_ID); - install_exec_hook(REDO_INSERT_ROW_HEAD); - install_exec_hook(REDO_INSERT_ROW_TAIL); - install_exec_hook(REDO_PURGE_ROW_HEAD); - install_exec_hook(REDO_PURGE_ROW_TAIL); - install_exec_hook(REDO_PURGE_BLOCKS); - install_exec_hook(REDO_DELETE_ALL); - install_exec_hook(UNDO_ROW_INSERT); - install_exec_hook(UNDO_ROW_DELETE); - install_exec_hook(UNDO_ROW_UPDATE); - install_exec_hook(UNDO_ROW_PURGE); - install_exec_hook(COMMIT); + exec_REDO_LOGREC_ ## R; +#define install_undo_exec_hook(R) \ + log_record_type_descriptor[LOGREC_ ## R].record_execute_in_undo_phase= \ + exec_UNDO_LOGREC_ ## R; + install_redo_exec_hook(LONG_TRANSACTION_ID); + install_redo_exec_hook(CHECKPOINT); + install_redo_exec_hook(REDO_CREATE_TABLE); + install_redo_exec_hook(REDO_DROP_TABLE); + install_redo_exec_hook(FILE_ID); + install_redo_exec_hook(REDO_INSERT_ROW_HEAD); + install_redo_exec_hook(REDO_INSERT_ROW_TAIL); + install_redo_exec_hook(REDO_PURGE_ROW_HEAD); + install_redo_exec_hook(REDO_PURGE_ROW_TAIL); + install_redo_exec_hook(REDO_PURGE_BLOCKS); + install_redo_exec_hook(REDO_DELETE_ALL); + install_redo_exec_hook(UNDO_ROW_INSERT); + install_redo_exec_hook(UNDO_ROW_DELETE); + install_redo_exec_hook(UNDO_ROW_UPDATE); + install_redo_exec_hook(UNDO_ROW_PURGE); + install_redo_exec_hook(COMMIT); + install_undo_exec_hook(UNDO_ROW_INSERT); current_group_end_lsn= LSN_IMPOSSIBLE; @@ -1178,10 +1214,6 @@ static uint end_of_redo_phase(my_bool prepare_for_undo_phase) } } - /* we don't need all_tables anymore, maria_open_list is enough */ - my_free(all_tables, MYF(MY_ALLOW_ZERO_PTR)); - all_tables= NULL; - /* We could take a checkpoint here, in case of a crash during the UNDO phase. The drawback is that a page which got a REDO (thus, flushed @@ -1207,7 +1239,23 @@ static int run_undo_phase(uint unfinished) DBUG_ASSERT(trn != NULL); llstr(trn->trid, llbuf); fprintf(tracef, "Rolling back transaction of long id %s\n", llbuf); - /* of course we miss execution of UNDOs here */ + + /* Execute all undo entries */ + while (trn->undo_lsn) + { + TRANSLOG_HEADER_BUFFER rec; + LOG_DESC *log_desc; + if (translog_read_record_header(trn->undo_lsn, &rec) == + RECHEADER_READ_ERROR) + return 1; + log_desc= &log_record_type_descriptor[rec.type]; + if (log_desc->record_execute_in_undo_phase(&rec, trn)) + { + fprintf(tracef, "Got error when executing undo\n"); + return 1; + } + } + if (trnman_rollback_trn(trn)) return 1; /* We could want to span a few threads (4?) instead of 1 */ diff --git a/storage/maria/ma_test1.c b/storage/maria/ma_test1.c index e5485c43f23..507e51ced35 100644 --- a/storage/maria/ma_test1.c +++ b/storage/maria/ma_test1.c @@ -38,7 +38,7 @@ static uint insert_count, update_count, remove_count; static uint pack_keys=0, pack_seg=0, key_length; static uint unique_key=HA_NOSAME; static my_bool pagecacheing, null_fields, silent, skip_update, opt_unique, - verbose, skip_delete, transactional; + verbose, skip_delete, transactional, die_in_middle_of_transaction; static MARIA_COLUMNDEF recinfo[4]; static MARIA_KEYDEF keyinfo[10]; static HA_KEYSEG keyseg[10]; @@ -50,6 +50,19 @@ static void create_key(char *key,uint rownr); static void create_record(char *record,uint rownr); static void update_record(char *record); + +/* + These are here only for testing of recovery with undo. We are not + including maria_def.h here as this test is also to be an example of + how to use maria outside of the maria directory +*/ + +extern int _ma_flush_table_files(MARIA_HA *info, uint flush_data_or_index, + enum flush_type flush_type_for_data, + enum flush_type flush_type_for_index); +#define MARIA_FLUSH_DATA 1 + + int main(int argc,char *argv[]) { MY_INIT(argv[0]); @@ -86,6 +99,9 @@ static int run_test(const char *filename) MARIA_UNIQUEDEF uniquedef; MARIA_CREATE_INFO create_info; + if (die_in_middle_of_transaction) + null_fields= 1; + bzero((char*) recinfo,sizeof(recinfo)); bzero((char*) &create_info,sizeof(create_info)); @@ -198,6 +214,9 @@ static int run_test(const char *filename) printf("J= %2d maria_write: %d errno: %d\n", j,error,my_errno); } + if (maria_commit(file) || maria_begin(file)) + goto err; + /* Insert 2 rows with null values */ if (null_fields) { @@ -215,6 +234,17 @@ static int run_test(const char *filename) flags[0]=2; } + if (die_in_middle_of_transaction) + { + /* + Ensure we get changed pages and log to disk + As commit record is not done, the undo entries needs to be rolled back. + */ + _ma_flush_table_files(file, MARIA_FLUSH_DATA, FLUSH_RELEASE, + FLUSH_RELEASE); + exit(1); + } + if (!skip_update) { if (opt_unique) @@ -627,6 +657,11 @@ static struct my_option my_long_options[] = (uchar**) &skip_delete, 0, GET_BOOL, NO_ARG, 0, 0, 0, 0, 0, 0}, {"skip-update", 'D', "Don't test updates", (uchar**) &skip_update, (uchar**) &skip_update, 0, GET_BOOL, NO_ARG, 0, 0, 0, 0, 0, 0}, + {"test-undo", 'A', + "Abort hard after doing inserts. Used for testing recovery with undo", + (uchar**) &die_in_middle_of_transaction, + (uchar**) &die_in_middle_of_transaction, + 0, GET_BOOL, NO_ARG, 0, 0, 0, 0, 0, 0}, {"transactional", 'T', "Test in transactional mode. (Only works with block format)", (uchar**) &transactional, (uchar**) &transactional, 0, GET_BOOL, NO_ARG, diff --git a/storage/maria/maria_read_log.c b/storage/maria/maria_read_log.c index e487847b486..4057fd51e85 100644 --- a/storage/maria/maria_read_log.c +++ b/storage/maria/maria_read_log.c @@ -93,8 +93,7 @@ int main(int argc, char **argv) */ fprintf(stdout, "TRACE of the last maria_read_log\n"); - /* Until we have UNDO records, no UNDO phase */ - if (maria_apply_log(lsn, opt_display_and_apply, stdout, FALSE)) + if (maria_apply_log(lsn, opt_display_and_apply, stdout, TRUE)) goto err; fprintf(stdout, "%s: SUCCESS\n", my_progname); -- cgit v1.2.1 From 73b073fff675e9cc27e880f4236613b6b55b4568 Mon Sep 17 00:00:00 2001 From: unknown Date: Tue, 4 Sep 2007 22:52:32 +0300 Subject: Check of transaction log descriptor table consistance added. Small fixes made. storage/maria/ma_loghandler.c: Check of transaction log descriptor table consistance added.\ Incorrect record description fixed. Compiler warning fixed. storage/maria/ma_loghandler.h: fixed ident. storage/maria/unittest/ma_test_loghandler-t.c: Suppressing of automatic record writing storage/maria/unittest/ma_test_loghandler_first_lsn-t.c: Suppressing of automatic record writing storage/maria/unittest/ma_test_loghandler_max_lsn-t.c: Suppressing of automatic record writing storage/maria/unittest/ma_test_loghandler_multigroup-t.c: Suppressing of automatic record writing storage/maria/unittest/ma_test_loghandler_multithread-t.c: Suppressing of automatic record writing storage/maria/unittest/ma_test_loghandler_noflush-t.c: Suppressing of automatic record writing storage/maria/unittest/ma_test_loghandler_pagecache-t.c: Suppressing of automatic record writing storage/maria/unittest/ma_test_loghandler_purge-t.c: Suppressing of automatic record writing --- storage/maria/ma_loghandler.c | 96 +++++++++++++++++++++- storage/maria/ma_loghandler.h | 2 +- storage/maria/unittest/ma_test_loghandler-t.c | 2 + .../unittest/ma_test_loghandler_first_lsn-t.c | 2 + .../maria/unittest/ma_test_loghandler_max_lsn-t.c | 2 + .../unittest/ma_test_loghandler_multigroup-t.c | 2 + .../unittest/ma_test_loghandler_multithread-t.c | 2 + .../maria/unittest/ma_test_loghandler_noflush-t.c | 3 +- .../unittest/ma_test_loghandler_pagecache-t.c | 2 + .../maria/unittest/ma_test_loghandler_purge-t.c | 2 + 10 files changed, 111 insertions(+), 4 deletions(-) diff --git a/storage/maria/ma_loghandler.c b/storage/maria/ma_loghandler.c index f2afedaf662..56c0e1aaef7 100644 --- a/storage/maria/ma_loghandler.c +++ b/storage/maria/ma_loghandler.c @@ -224,6 +224,84 @@ static my_bool translog_page_validator(uchar *page_addr, uchar* data_ptr); LOG_DESC log_record_type_descriptor[LOGREC_NUMBER_OF_TYPES]; + +#ifndef DBUG_OFF +/** + @brief check the description table validity + + @param num how many records should be filled +*/ + +static void check_translog_description_table(int num) +{ + int i; + DBUG_ENTER("check_translog_description_table"); + DBUG_PRINT("enter", ("last record: %d", num)); + DBUG_ASSERT(num > 0); + /* last is reserved for extending the table */ + DBUG_ASSERT(num < LOGREC_NUMBER_OF_TYPES - 1); + DBUG_PRINT("info", ("records number: OK")); + DBUG_PRINT("info", + ("record type: %d class: %d fixed: %u header: %u LSNs: %u " + "name: %s", + 0, + log_record_type_descriptor[0].class, + (uint)log_record_type_descriptor[0].fixed_length, + (uint)log_record_type_descriptor[0].read_header_len, + (uint)log_record_type_descriptor[0].compressed_LSN, + log_record_type_descriptor[0].name)); + DBUG_ASSERT(log_record_type_descriptor[0].class == LOGRECTYPE_NOT_ALLOWED); + DBUG_PRINT("info", ("record type 0: OK")); + for (i= 1; i <= num; i++) + { + DBUG_PRINT("info", + ("record type: %d class: %d fixed: %u header: %u LSNs: %u " + "name: %s", + i, log_record_type_descriptor[i].class, + (uint)log_record_type_descriptor[i].fixed_length, + (uint)log_record_type_descriptor[i].read_header_len, + (uint)log_record_type_descriptor[i].compressed_LSN, + log_record_type_descriptor[i].name)); + switch (log_record_type_descriptor[i].class) { + case LOGRECTYPE_NOT_ALLOWED: + DBUG_ASSERT(0); + break; + case LOGRECTYPE_VARIABLE_LENGTH: + DBUG_ASSERT(log_record_type_descriptor[i].fixed_length == 0); + DBUG_ASSERT((log_record_type_descriptor[i].compressed_LSN == 0) || + ((log_record_type_descriptor[i].compressed_LSN == 1) && + (log_record_type_descriptor[i].read_header_len >= + LSN_STORE_SIZE)) || + ((log_record_type_descriptor[i].compressed_LSN == 2) && + (log_record_type_descriptor[i].read_header_len >= + LSN_STORE_SIZE * 2))); + break; + case LOGRECTYPE_PSEUDOFIXEDLENGTH: + DBUG_ASSERT(log_record_type_descriptor[i].fixed_length == + log_record_type_descriptor[i].read_header_len); + DBUG_ASSERT(log_record_type_descriptor[i].compressed_LSN > 0); + DBUG_ASSERT(log_record_type_descriptor[i].compressed_LSN <= 2); + break; + case LOGRECTYPE_FIXEDLENGTH: + DBUG_ASSERT(log_record_type_descriptor[i].fixed_length == + log_record_type_descriptor[i].read_header_len); + DBUG_ASSERT(log_record_type_descriptor[i].compressed_LSN == 0); + break; + default: + DBUG_ASSERT(0); + } + DBUG_PRINT("info", ("record type %d: OK", i)); + } + DBUG_PRINT("info", ("All filled records are OK")); + for (i= num + 1; i < LOGREC_NUMBER_OF_TYPES; i++) + { + DBUG_ASSERT(log_record_type_descriptor[i].class == LOGRECTYPE_NOT_ALLOWED); + DBUG_PRINT("info", ("record type %d: OK", i)); + } + DBUG_VOID_RETURN; +} +#endif + static LOG_DESC INIT_LOGREC_FIXED_RECORD_0LSN_EXAMPLE= {LOGRECTYPE_FIXEDLENGTH, 6, 6, NULL, NULL, NULL, 0, "fixed0example", LOGREC_NOT_LAST_IN_GROUP, NULL, NULL}; @@ -251,6 +329,7 @@ static LOG_DESC INIT_LOGREC_VARIABLE_RECORD_2LSN_EXAMPLE= void example_loghandler_init() { + int i; log_record_type_descriptor[LOGREC_FIXED_RECORD_0LSN_EXAMPLE]= INIT_LOGREC_FIXED_RECORD_0LSN_EXAMPLE; log_record_type_descriptor[LOGREC_VARIABLE_RECORD_0LSN_EXAMPLE]= @@ -263,6 +342,12 @@ void example_loghandler_init() INIT_LOGREC_FIXED_RECORD_2LSN_EXAMPLE; log_record_type_descriptor[LOGREC_VARIABLE_RECORD_2LSN_EXAMPLE]= INIT_LOGREC_VARIABLE_RECORD_2LSN_EXAMPLE; + for (i= LOGREC_VARIABLE_RECORD_2LSN_EXAMPLE + 1; + i < LOGREC_NUMBER_OF_TYPES; + i++) + log_record_type_descriptor[i].class= LOGRECTYPE_NOT_ALLOWED; + DBUG_EXECUTE("info", + check_translog_description_table(LOGREC_VARIABLE_RECORD_2LSN_EXAMPLE);); } @@ -374,7 +459,7 @@ static LOG_DESC INIT_LOGREC_PREPARE= "prepare", LOGREC_IS_GROUP_ITSELF, NULL, NULL}; static LOG_DESC INIT_LOGREC_PREPARE_WITH_UNDO_PURGE= -{LOGRECTYPE_VARIABLE_LENGTH, 0, 5, NULL, NULL, NULL, 1, +{LOGRECTYPE_VARIABLE_LENGTH, 0, LSN_STORE_SIZE, NULL, NULL, NULL, 1, "prepare_with_undo_purge", LOGREC_IS_GROUP_ITSELF, NULL, NULL}; static LOG_DESC INIT_LOGREC_COMMIT= @@ -424,6 +509,7 @@ const myf log_write_flags= MY_WME | MY_NABP | MY_WAIT_IF_FULL; static void loghandler_init() { + int i; log_record_type_descriptor[LOGREC_RESERVED_FOR_CHUNKS23]= INIT_LOGREC_RESERVED_FOR_CHUNKS23; log_record_type_descriptor[LOGREC_REDO_INSERT_ROW_HEAD]= @@ -488,6 +574,12 @@ static void loghandler_init() INIT_LOGREC_FILE_ID; log_record_type_descriptor[LOGREC_LONG_TRANSACTION_ID]= INIT_LOGREC_LONG_TRANSACTION_ID; + for (i= LOGREC_LONG_TRANSACTION_ID + 1; + i < LOGREC_NUMBER_OF_TYPES; + i++) + log_record_type_descriptor[i].class= LOGRECTYPE_NOT_ALLOWED; + DBUG_EXECUTE("info", + check_translog_description_table(LOGREC_LONG_TRANSACTION_ID);); }; @@ -2257,7 +2349,7 @@ static uchar *translog_get_page(TRANSLOG_VALIDATOR_DATA *data, uchar *buffer) { int is_last_unfinished_page; uint last_protected_sector= 0; - uchar *from, *table; + uchar *from, *table= NULL; translog_wait_for_writers(curr_buffer); DBUG_ASSERT(LSN_FILE_NO(addr) == LSN_FILE_NO(curr_buffer->offset)); from= curr_buffer->buffer + (addr - curr_buffer->offset); diff --git a/storage/maria/ma_loghandler.h b/storage/maria/ma_loghandler.h index a4065fedf66..5c014fe05af 100644 --- a/storage/maria/ma_loghandler.h +++ b/storage/maria/ma_loghandler.h @@ -159,7 +159,7 @@ typedef struct st_translog_header_buffer /* in multi-group number of chunk0 pages (valid only if groups_no > 0) */ uint chunk0_pages; /* type of the read record */ - enum translog_record_type type; + enum translog_record_type type; /* chunk 0 data address (valid only if groups_no > 0) */ TRANSLOG_ADDRESS chunk0_data_addr; /* diff --git a/storage/maria/unittest/ma_test_loghandler-t.c b/storage/maria/unittest/ma_test_loghandler-t.c index cd399637b9d..170efd6c90f 100644 --- a/storage/maria/unittest/ma_test_loghandler-t.c +++ b/storage/maria/unittest/ma_test_loghandler-t.c @@ -183,6 +183,8 @@ int main(int argc __attribute__((unused)), char *argv[]) exit(1); } example_loghandler_init(); + /* Suppressing of automatic record writing */ + trn->first_undo_lsn|= TRANSACTION_LOGGED_LONG_ID; plan(((ITERATIONS - 1) * 4 + 1)*2 + ITERATIONS - 1); diff --git a/storage/maria/unittest/ma_test_loghandler_first_lsn-t.c b/storage/maria/unittest/ma_test_loghandler_first_lsn-t.c index 81e000a9181..17a41f1ad3e 100644 --- a/storage/maria/unittest/ma_test_loghandler_first_lsn-t.c +++ b/storage/maria/unittest/ma_test_loghandler_first_lsn-t.c @@ -72,6 +72,8 @@ int main(int argc __attribute__((unused)), char *argv[]) exit(1); } example_loghandler_init(); + /* Suppressing of automatic record writing */ + dummy_transaction_object.first_undo_lsn|= TRANSACTION_LOGGED_LONG_ID; theor_lsn= translog_first_theoretical_lsn(); if (theor_lsn == 1) diff --git a/storage/maria/unittest/ma_test_loghandler_max_lsn-t.c b/storage/maria/unittest/ma_test_loghandler_max_lsn-t.c index c50681434e3..08f838ebb65 100644 --- a/storage/maria/unittest/ma_test_loghandler_max_lsn-t.c +++ b/storage/maria/unittest/ma_test_loghandler_max_lsn-t.c @@ -66,6 +66,8 @@ int main(int argc __attribute__((unused)), char *argv[]) exit(1); } example_loghandler_init(); + /* Suppressing of automatic record writing */ + dummy_transaction_object.first_undo_lsn|= TRANSACTION_LOGGED_LONG_ID; max_lsn= translog_get_file_max_lsn_stored(1); if (max_lsn == 1) diff --git a/storage/maria/unittest/ma_test_loghandler_multigroup-t.c b/storage/maria/unittest/ma_test_loghandler_multigroup-t.c index bf4cfe110e3..aa4b7b473cf 100644 --- a/storage/maria/unittest/ma_test_loghandler_multigroup-t.c +++ b/storage/maria/unittest/ma_test_loghandler_multigroup-t.c @@ -179,6 +179,8 @@ int main(int argc __attribute__((unused)), char *argv[]) exit(1); } example_loghandler_init(); + /* Suppressing of automatic record writing */ + trn->first_undo_lsn|= TRANSACTION_LOGGED_LONG_ID; plan(((ITERATIONS - 1) * 4 + 1) * 2); diff --git a/storage/maria/unittest/ma_test_loghandler_multithread-t.c b/storage/maria/unittest/ma_test_loghandler_multithread-t.c index 0f56ef5384c..d526dd933a1 100644 --- a/storage/maria/unittest/ma_test_loghandler_multithread-t.c +++ b/storage/maria/unittest/ma_test_loghandler_multithread-t.c @@ -289,6 +289,8 @@ int main(int argc __attribute__((unused)), exit(1); } example_loghandler_init(); + /* Suppressing of automatic record writing */ + dummy_transaction_object.first_undo_lsn|= TRANSACTION_LOGGED_LONG_ID; srandom(122334817L); { diff --git a/storage/maria/unittest/ma_test_loghandler_noflush-t.c b/storage/maria/unittest/ma_test_loghandler_noflush-t.c index c924536dde6..901bf588197 100644 --- a/storage/maria/unittest/ma_test_loghandler_noflush-t.c +++ b/storage/maria/unittest/ma_test_loghandler_noflush-t.c @@ -74,7 +74,8 @@ int main(int argc __attribute__((unused)), char *argv[]) exit(1); } example_loghandler_init(); - + /* Suppressing of automatic record writing */ + dummy_transaction_object.first_undo_lsn|= TRANSACTION_LOGGED_LONG_ID; int4store(long_tr_id, 0); long_tr_id[5]= 0xff; diff --git a/storage/maria/unittest/ma_test_loghandler_pagecache-t.c b/storage/maria/unittest/ma_test_loghandler_pagecache-t.c index 7dfdc32234e..276640dfd17 100644 --- a/storage/maria/unittest/ma_test_loghandler_pagecache-t.c +++ b/storage/maria/unittest/ma_test_loghandler_pagecache-t.c @@ -74,6 +74,8 @@ int main(int argc __attribute__((unused)), char *argv[]) exit(1); } example_loghandler_init(); + /* Suppressing of automatic record writing */ + dummy_transaction_object.first_undo_lsn|= TRANSACTION_LOGGED_LONG_ID; if ((stat= my_stat(first_translog_file, &st, MYF(0))) == 0) { diff --git a/storage/maria/unittest/ma_test_loghandler_purge-t.c b/storage/maria/unittest/ma_test_loghandler_purge-t.c index 1beeb442f8f..c638aa85ac6 100644 --- a/storage/maria/unittest/ma_test_loghandler_purge-t.c +++ b/storage/maria/unittest/ma_test_loghandler_purge-t.c @@ -69,6 +69,8 @@ int main(int argc __attribute__((unused)), char *argv[]) exit(1); } example_loghandler_init(); + /* Suppressing of automatic record writing */ + dummy_transaction_object.first_undo_lsn|= TRANSACTION_LOGGED_LONG_ID; /* write more then 1 file */ int4store(long_tr_id, 0); -- cgit v1.2.1 From 044c4103ba4c2a99ac1024cac536d38704b27c15 Mon Sep 17 00:00:00 2001 From: unknown Date: Wed, 5 Sep 2007 02:57:53 +0300 Subject: Added undo of deleted row Added part of undo of update row Extended ma_test1 for recovery testing Some bug fixes storage/maria/ha_maria.cc: Ignore 'state.split' in case of block records storage/maria/ma_bitmap.c: Added return value for _ma_bitmap_find_place() for how much data we should put on head page storage/maria/ma_blockrec.c: Added undo of deleted row. - Added logging of CLR_END records in write_block_record() - Split ma_write_init_block_record() to two functions to get better code reuse - Added _ma_apply_undo_row_delete() - Added ma_get_length() Added 'empty' prototype for undo_row_update() Fixed bug when moving data withing a head/tail page. Fixed bug when reading a page with bigger LSN but of different type than was expected. Store undo_lsn first in CLR_END record Simplified some code by adding local variables. Changed log format for UNDO_ROW_DELETE to store total length of used blobs storage/maria/ma_blockrec.h: Added prototypes for undo code. storage/maria/ma_pagecache.c: Allow plain page to change to LSN page (needed in recovery to apply UNDO) storage/maria/ma_recovery.c: Added undo handling of UNDO_ROW_DELETE and UNDO_ROW_UPDATE storage/maria/ma_test1.c: Extended --test-undo option to allow us to die after insert or after delete. Fixed bug in printing key values when using -v storage/maria/maria_def.h: Moved some variables around to be getter alignment Added length_buff buffer to be used during undo handling --- storage/maria/ha_maria.cc | 4 +- storage/maria/ma_bitmap.c | 5 + storage/maria/ma_blockrec.c | 505 ++++++++++++++++++++++++++++++++++--------- storage/maria/ma_blockrec.h | 4 + storage/maria/ma_pagecache.c | 4 +- storage/maria/ma_recovery.c | 86 ++++++++ storage/maria/ma_test1.c | 39 +++- storage/maria/maria_def.h | 8 +- 8 files changed, 543 insertions(+), 112 deletions(-) diff --git a/storage/maria/ha_maria.cc b/storage/maria/ha_maria.cc index 55ce800d596..7613ada0919 100644 --- a/storage/maria/ha_maria.cc +++ b/storage/maria/ha_maria.cc @@ -1219,7 +1219,9 @@ int ha_maria::repair(THD *thd, HA_CHECK ¶m, bool do_optimize) } if (!do_optimize || - ((file->state->del || share->state.split != file->state->records) && + ((file->state->del || + ((file->s->data_file_type != BLOCK_RECORD) && + share->state.split != file->state->records)) && (!(param.testflag & T_QUICK) || (share->state.changed & (STATE_NOT_OPTIMIZED_KEYS | STATE_NOT_OPTIMIZED_ROWS))))) diff --git a/storage/maria/ma_bitmap.c b/storage/maria/ma_bitmap.c index 2a2308637b6..37c1a77aa44 100644 --- a/storage/maria/ma_bitmap.c +++ b/storage/maria/ma_bitmap.c @@ -1421,6 +1421,8 @@ static my_bool write_rest_of_head(MARIA_HA *info, uint position, RETURN 0 ok + row->space_on_head_page contains minimum number of bytes we + expect to put on the head page. 1 error */ @@ -1457,6 +1459,7 @@ my_bool _ma_bitmap_find_place(MARIA_HA *info, MARIA_ROW *row, position= ELEMENTS_RESERVED_FOR_MAIN_PART - 1; if (find_head(info, (uint) row->total_length, position)) goto abort; + row->space_on_head_page= row->total_length; goto end; } @@ -1474,6 +1477,7 @@ my_bool _ma_bitmap_find_place(MARIA_HA *info, MARIA_ROW *row, position= ELEMENTS_RESERVED_FOR_MAIN_PART - 1; if (find_head(info, head_length, position)) goto abort; + row->space_on_head_page= head_length; goto end; } @@ -1490,6 +1494,7 @@ my_bool _ma_bitmap_find_place(MARIA_HA *info, MARIA_ROW *row, position= ELEMENTS_RESERVED_FOR_MAIN_PART -2; /* Only head and tail */ if (find_head(info, row_length, position)) goto abort; + row->space_on_head_page= row_length; rest_length= head_length - row_length; if (write_rest_of_head(info, position, rest_length)) goto abort; diff --git a/storage/maria/ma_blockrec.c b/storage/maria/ma_blockrec.c index 20c53d0c9fc..55dc72b1d02 100644 --- a/storage/maria/ma_blockrec.c +++ b/storage/maria/ma_blockrec.c @@ -449,7 +449,7 @@ my_bool _ma_init_block_record(MARIA_HA *info) &info->log_row_parts, sizeof(*info->log_row_parts) * (TRANSLOG_INTERNAL_PARTS + 2 + - info->s->base.fields + 2), + info->s->base.fields + 3), &info->update_field_data, (info->s->base.fields * 4 + info->s->base.max_field_lengths + 1 + 4), @@ -655,9 +655,6 @@ static my_bool extend_area_on_page(uchar *buff, uchar *dir, } - - - /* Check that a region is all zero @@ -726,6 +723,23 @@ void _ma_unpin_all_pages(MARIA_HA *info, LSN undo_lsn) } +#ifdef NOT_YET_NEEDED +/* Calculate empty space on a page */ + +static uint empty_space_on_page(uchar *buff, uint block_size) +{ + enum en_page_type; + page_type= (enum en_page_type) (buff[PAGE_TYPE_OFFSET] & + ~(uchar) PAGE_CAN_BE_COMPACTED); + if (page_type == UNALLOCATED_PAGE) + return block_size; + if ((uint) page_type <= TAIL_PAGE) + return uint2korr(buff+EMPTY_SPACE_OFFSET); + return 0; /* Blob page */ +} +#endif + + /* Find free position in directory @@ -917,8 +931,8 @@ static void calc_record_size(MARIA_HA *info, const uchar *record, { uint length, field_length_data_length; const uchar *field_pos= record + column->offset; - /* 256 is correct as this includes the length uchar */ + /* 256 is correct as this includes the length uchar */ field_length_data[0]= field_pos[0]; if (column->length <= 256) { @@ -1174,9 +1188,9 @@ static void make_empty_page(uchar *buff, uint block_size, uint page_type) struct st_row_pos_info { - uchar *buff; /* page buffer */ - uchar *data; /* Place for data */ - uchar *dir; /* Directory */ + uchar *buff; /* page buffer */ + uchar *data; /* Place for data */ + uchar *dir; /* Directory */ uint length; /* Length for data */ uint rownr; /* Offset in directory */ uint empty_space; /* Space left on page */ @@ -1228,16 +1242,20 @@ static my_bool get_head_or_tail_page(MARIA_HA *info, if (res->length < length) { - if (res->empty_space + res->length < length) + if (res->empty_space + res->length >= length) { compact_page(res->buff, block_size, res->rownr, 1); /* All empty space are now after current position */ dir= (res->buff + block_size - DIR_ENTRY_SIZE * res->rownr - - PAGE_SUFFIX_SIZE); + DIR_ENTRY_SIZE - PAGE_SUFFIX_SIZE); res->length= res->empty_space= uint2korr(dir+2); } if (res->length < length) + { + DBUG_PRINT("error", ("length: %u res->length: %u empty_space: %u", + length, res->length, res->empty_space)); goto crashed; /* Wrong bitmap information */ + } } res->dir= dir; res->data= res->buff + uint2korr(dir); @@ -1631,6 +1649,7 @@ static my_bool free_full_page_range(MARIA_HA *info, ulonglong page, uint count) /** @brief Write a record to a (set of) pages + @fn write_block_record() @param info Maria handler @param old_record Original record in case of update; NULL in case of insert @@ -1640,6 +1659,7 @@ static my_bool free_full_page_range(MARIA_HA *info, ulonglong page, uint count) @param map_blocks On which pages the record should be stored @param row_pos Position on head page where to put head part of record + @param undo_lsn <> 0 if we are in UNDO @note On return all pinned pages are released. @@ -1654,7 +1674,8 @@ static my_bool write_block_record(MARIA_HA *info, MARIA_ROW *row, MARIA_BITMAP_BLOCKS *bitmap_blocks, my_bool head_block_is_read, - struct st_row_pos_info *row_pos) + struct st_row_pos_info *row_pos, + LSN undo_lsn) { uchar *data, *end_of_data, *tmp_data_used, *tmp_data; uchar *row_extents_first_part, *row_extents_second_part; @@ -1862,7 +1883,7 @@ static my_bool write_block_record(MARIA_HA *info, { /* Update page directory */ uint length= (uint) (data - row_pos->data); - DBUG_PRINT("info", ("head length: %u", length)); + DBUG_PRINT("info", ("Used head length on page: %u", length)); if (length < info->s->base.min_row_length) { uint diff_length= info->s->base.min_row_length - length; @@ -2256,51 +2277,70 @@ static my_bool write_block_record(MARIA_HA *info, goto disk_err; } - /* Write UNDO record */ + /* Write UNDO or CLR record */ + lsn= 0; if (share->now_transactional) { - uchar log_data[LSN_STORE_SIZE + FILEID_STORE_SIZE + - PAGE_STORE_SIZE + DIRPOS_STORE_SIZE]; LEX_STRING *log_array= info->log_row_parts; - /* LOGREC_UNDO_ROW_INSERT & LOGREC_UNDO_ROW_INSERT share same header */ - lsn_store(log_data, info->trn->undo_lsn); - page_store(log_data+ LSN_STORE_SIZE + FILEID_STORE_SIZE, - head_block->page); - dirpos_store(log_data+ LSN_STORE_SIZE + FILEID_STORE_SIZE + - PAGE_STORE_SIZE, - row_pos->rownr); + if (undo_lsn) + { + uchar log_data[LSN_STORE_SIZE + FILEID_STORE_SIZE]; - log_array[TRANSLOG_INTERNAL_PARTS + 0].str= (char*) log_data; - log_array[TRANSLOG_INTERNAL_PARTS + 0].length= sizeof(log_data); + /* undo_lsn must be first for compression to work */ + lsn_store(log_data, undo_lsn); + log_array[TRANSLOG_INTERNAL_PARTS + 0].str= (char*) log_data; + log_array[TRANSLOG_INTERNAL_PARTS + 0].length= sizeof(log_data); - if (!old_record) - { - /* Write UNDO log record for the INSERT */ - if (translog_write_record(&lsn, LOGREC_UNDO_ROW_INSERT, + if (translog_write_record(&lsn, LOGREC_CLR_END, info->trn, info, sizeof(log_data), TRANSLOG_INTERNAL_PARTS + 1, log_array, - log_data + LSN_STORE_SIZE)) + log_data+ FILEID_STORE_SIZE)) goto disk_err; } else { - /* Write UNDO log record for the UPDATE */ - size_t row_length; - uint row_parts_count; - row_length= fill_update_undo_parts(info, old_record, record, - info->log_row_parts + - TRANSLOG_INTERNAL_PARTS + 1, - &row_parts_count); - if (translog_write_record(&lsn, LOGREC_UNDO_ROW_UPDATE, info->trn, - info, sizeof(log_data) + row_length, - TRANSLOG_INTERNAL_PARTS + 1 + row_parts_count, - log_array, log_data + LSN_STORE_SIZE)) - goto disk_err; + uchar log_data[LSN_STORE_SIZE + FILEID_STORE_SIZE + + PAGE_STORE_SIZE + DIRPOS_STORE_SIZE]; + + /* LOGREC_UNDO_ROW_INSERT & LOGREC_UNDO_ROW_INSERT share same header */ + lsn_store(log_data, info->trn->undo_lsn); + page_store(log_data+ LSN_STORE_SIZE + FILEID_STORE_SIZE, + head_block->page); + dirpos_store(log_data+ LSN_STORE_SIZE + FILEID_STORE_SIZE + + PAGE_STORE_SIZE, + row_pos->rownr); + + log_array[TRANSLOG_INTERNAL_PARTS + 0].str= (char*) log_data; + log_array[TRANSLOG_INTERNAL_PARTS + 0].length= sizeof(log_data); + + if (!old_record) + { + /* Write UNDO log record for the INSERT */ + if (translog_write_record(&lsn, LOGREC_UNDO_ROW_INSERT, + info->trn, info, sizeof(log_data), + TRANSLOG_INTERNAL_PARTS + 1, log_array, + log_data + LSN_STORE_SIZE)) + goto disk_err; + } + else + { + /* Write UNDO log record for the UPDATE */ + size_t row_length; + uint row_parts_count; + row_length= fill_update_undo_parts(info, old_record, record, + info->log_row_parts + + TRANSLOG_INTERNAL_PARTS + 1, + &row_parts_count); + if (translog_write_record(&lsn, LOGREC_UNDO_ROW_UPDATE, info->trn, + info, sizeof(log_data) + row_length, + TRANSLOG_INTERNAL_PARTS + 1 + row_parts_count, + log_array, log_data + LSN_STORE_SIZE)) + goto disk_err; + } } } - - _ma_unpin_all_pages(info, info->trn->undo_lsn); + _ma_unpin_all_pages(info, lsn); if (tmp_data_used) { @@ -2379,7 +2419,49 @@ disk_err: /* - Write a record (to get the row id for it) + @brief Write a record + + @fn allocate_and_write_block_record() + @param info Maria handler + @param record Record to write + @param row Information about fields in 'record' + @param undo_lsn <> 0 if in undo + + @return + @retval 0 ok + @retval 1 Error +*/ + +static my_bool allocate_and_write_block_record(MARIA_HA *info, + const uchar *record, + MARIA_ROW *row, + LSN undo_lsn) +{ + struct st_row_pos_info row_pos; + MARIA_BITMAP_BLOCKS *blocks= &row->insert_blocks; + DBUG_ENTER("allocate_and_write_block_record"); + + if (_ma_bitmap_find_place(info, row, blocks)) + DBUG_RETURN(1); /* Error reading bitmap */ + /* page will be pinned & locked by get_head_or_tail_page */ + if (get_head_or_tail_page(info, blocks->block, info->buff, + row->space_on_head_page, HEAD_PAGE, + PAGECACHE_LOCK_WRITE, &row_pos)) + DBUG_RETURN(1); + row->lastpos= ma_recordpos(blocks->block->page, row_pos.rownr); + if (info->s->calc_checksum) + row->checksum= (info->s->calc_checksum)(info,record); + if (write_block_record(info, (uchar*) 0, record, row, + blocks, blocks->block->org_bitmap_value != 0, + &row_pos, undo_lsn)) + DBUG_RETURN(1); /* Error reading bitmap */ + DBUG_PRINT("exit", ("Rowid: %lu", (ulong) row->lastpos)); + DBUG_RETURN(0); +} + + +/* + Write a record and return rowid for it SYNOPSIS _ma_write_init_block_record() @@ -2397,27 +2479,11 @@ disk_err: MARIA_RECORD_POS _ma_write_init_block_record(MARIA_HA *info, const uchar *record) { - MARIA_BITMAP_BLOCKS *blocks= &info->cur_row.insert_blocks; - struct st_row_pos_info row_pos; DBUG_ENTER("_ma_write_init_block_record"); calc_record_size(info, record, &info->cur_row); - if (_ma_bitmap_find_place(info, &info->cur_row, blocks)) - DBUG_RETURN(HA_OFFSET_ERROR); /* Error reading bitmap */ - /* page will be pinned & locked by get_head_or_tail_page */ - if (get_head_or_tail_page(info, blocks->block, info->buff, - info->s->base.min_row_length, HEAD_PAGE, - PAGECACHE_LOCK_WRITE, &row_pos)) + if (allocate_and_write_block_record(info, record, &info->cur_row, 0)) DBUG_RETURN(HA_OFFSET_ERROR); - info->cur_row.lastpos= ma_recordpos(blocks->block->page, row_pos.rownr); - if (info->s->calc_checksum) - info->cur_row.checksum= (info->s->calc_checksum)(info,record); - if (write_block_record(info, (uchar*) 0, record, &info->cur_row, - blocks, blocks->block->org_bitmap_value != 0, - &row_pos)) - DBUG_RETURN(HA_OFFSET_ERROR); /* Error reading bitmap */ - DBUG_PRINT("exit", ("Rowid: %lu", (ulong) info->cur_row.lastpos)); - info->s->state.split++; DBUG_RETURN(info->cur_row.lastpos); } @@ -2603,7 +2669,7 @@ my_bool _ma_update_block_record(MARIA_HA *info, MARIA_RECORD_POS record_pos, if (cur_row->extents_count && free_full_pages(info, cur_row)) goto err; DBUG_RETURN(write_block_record(info, oldrec, record, new_row, blocks, - 1, &row_pos)); + 1, &row_pos, 0)); } /* Allocate all size in block for record @@ -2636,7 +2702,7 @@ my_bool _ma_update_block_record(MARIA_HA *info, MARIA_RECORD_POS record_pos, row_pos.data= buff + uint2korr(dir); row_pos.length= head_length; DBUG_RETURN(write_block_record(info, oldrec, record, new_row, blocks, 1, - &row_pos)); + &row_pos, 0)); err: _ma_unpin_all_pages(info, 0); @@ -3218,6 +3284,7 @@ static my_bool read_long_data(MARIA_HA *info, uchar *to, ulong length, cur_row.extents_counts contains number of extents cur_row.empty_bits is set to empty bits cur_row.field_lengths contains packed length of all fields + cur_row.blob_length contains total length of all blobs. RETURN 0 ok @@ -3233,6 +3300,7 @@ int _ma_read_block_record2(MARIA_HA *info, uchar *record, my_bool found_blob= 0; MARIA_EXTENT_CURSOR extent; MARIA_COLUMNDEF *column, *end_column; + MARIA_ROW *cur_row= &info->cur_row; DBUG_ENTER("_ma_read_block_record2"); LINT_INIT(field_length_data); @@ -3242,8 +3310,9 @@ int _ma_read_block_record2(MARIA_HA *info, uchar *record, flag= (uint) (uchar) data[0]; cur_null_bytes= share->base.original_null_bytes; null_bytes= share->base.null_bytes; - info->cur_row.head_length= (uint) (end_of_data - data); - info->cur_row.full_page_count= info->cur_row.tail_count= 0; + cur_row->head_length= (uint) (end_of_data - data); + cur_row->full_page_count= cur_row->tail_count= 0; + cur_row->blob_length= 0; /* Skip trans header (for now, until we have MVCC csupport) */ data+= total_header_size[(flag & PRECALC_HEADER_BITMASK)]; @@ -3259,22 +3328,22 @@ int _ma_read_block_record2(MARIA_HA *info, uchar *record, Get number of extents and first extent */ get_key_length(row_extents, data); - info->cur_row.extents_count= row_extents; + cur_row->extents_count= row_extents; row_extent_size= row_extents * ROW_EXTENT_SIZE; - if (info->cur_row.extents_buffer_length < row_extent_size && - _ma_alloc_buffer(&info->cur_row.extents, - &info->cur_row.extents_buffer_length, + if (cur_row->extents_buffer_length < row_extent_size && + _ma_alloc_buffer(&cur_row->extents, + &cur_row->extents_buffer_length, row_extent_size)) DBUG_RETURN(my_errno); - memcpy(info->cur_row.extents, data, ROW_EXTENT_SIZE); + memcpy(cur_row->extents, data, ROW_EXTENT_SIZE); data+= ROW_EXTENT_SIZE; - init_extent(&extent, info->cur_row.extents, row_extents, - info->cur_row.tail_positions); + init_extent(&extent, cur_row->extents, row_extents, + cur_row->tail_positions); } else { - info->cur_row.extents_count= 0; - (*info->cur_row.tail_positions)= 0; + cur_row->extents_count= 0; + (*cur_row->tail_positions)= 0; extent.page_count= 0; extent.extent_count= 1; } @@ -3284,7 +3353,7 @@ int _ma_read_block_record2(MARIA_HA *info, uchar *record, if (share->base.max_field_lengths) { get_key_length(field_lengths, data); - info->cur_row.field_lengths_length= field_lengths; + cur_row->field_lengths_length= field_lengths; #ifdef SANITY_CHECKS if (field_lengths > share->base.max_field_lengths) goto err; @@ -3292,7 +3361,7 @@ int _ma_read_block_record2(MARIA_HA *info, uchar *record, } if (share->calc_checksum) - info->cur_row.checksum= (uint) (uchar) *data++; + cur_row->checksum= (uint) (uchar) *data++; /* data now points on null bits */ memcpy(record, data, cur_null_bytes); if (unlikely(cur_null_bytes != null_bytes)) @@ -3305,7 +3374,7 @@ int _ma_read_block_record2(MARIA_HA *info, uchar *record, } data+= null_bytes; /* We copy the empty bits to be able to use them for delete/update */ - memcpy(info->cur_row.empty_bits, data, share->base.pack_bytes); + memcpy(cur_row->empty_bits, data, share->base.pack_bytes); data+= share->base.pack_bytes; /* TODO: Use field offsets, instead of just skipping them */ @@ -3313,11 +3382,11 @@ int _ma_read_block_record2(MARIA_HA *info, uchar *record, /* Read row extents (note that first extent was already read into - info->cur_row.extents above) + cur_row->extents above) */ if (row_extents > 1) { - if (read_long_data(info, info->cur_row.extents + ROW_EXTENT_SIZE, + if (read_long_data(info, cur_row->extents + ROW_EXTENT_SIZE, (row_extents - 1) * ROW_EXTENT_SIZE, &extent, &data, &end_of_data)) DBUG_RETURN(my_errno); @@ -3342,7 +3411,7 @@ int _ma_read_block_record2(MARIA_HA *info, uchar *record, /* Read array of field lengths. This may be stored in several extents */ if (field_lengths) { - field_length_data= info->cur_row.field_lengths; + field_length_data= cur_row->field_lengths; if (read_long_data(info, field_length_data, field_lengths, &extent, &data, &end_of_data)) DBUG_RETURN(my_errno); @@ -3356,7 +3425,7 @@ int _ma_read_block_record2(MARIA_HA *info, uchar *record, uchar *field_pos= record + column->offset; /* First check if field is present in record */ if ((record[column->null_pos] & column->null_bit) || - (info->cur_row.empty_bits[column->empty_pos] & column->empty_bit)) + (cur_row->empty_bits[column->empty_pos] & column->empty_bit)) { if (type == FIELD_SKIP_ENDSPACE) bfill(record + column->offset, column->length, ' '); @@ -3432,13 +3501,14 @@ int _ma_read_block_record2(MARIA_HA *info, uchar *record, { uint size_length; if ((record[blob_field->null_pos] & blob_field->null_bit) || - (info->cur_row.empty_bits[blob_field->empty_pos] & + (cur_row->empty_bits[blob_field->empty_pos] & blob_field->empty_bit)) continue; size_length= blob_field->length - portable_sizeof_char_ptr; blob_lengths+= _ma_calc_blob_length(size_length, length_data); length_data+= size_length; } + cur_row->blob_length= blob_lengths; DBUG_PRINT("info", ("Total blob length: %lu", blob_lengths)); if (_ma_alloc_buffer(&info->rec_buff, &info->rec_buff_size, blob_lengths)) @@ -4012,6 +4082,37 @@ uint ma_calc_length_for_store_length(ulong nr) } +/* Retrive a stored number */ + +static ulong ma_get_length(uchar **packet) +{ + reg1 uchar *pos= *packet; + if (*pos < 251) + { + (*packet)++; + return (ulong) *pos; + } + if (*pos == 251) + { + (*packet)+= 2; + return (ulong) pos[1]; + } + if (*pos == 252) + { + (*packet)+= 3; + return (ulong) uint2korr(pos+1); + } + if (*pos == 253) + { + (*packet)+= 4; + return (ulong) uint3korr(pos+1); + } + DBUG_ASSERT(*pos == 254); + (*packet)+= 5; + return (ulong) uint4korr(pos+1); +} + + /* Fill array with pointers to field parts to be stored in log for insert @@ -4058,7 +4159,7 @@ static size_t fill_insert_undo_parts(MARIA_HA *info, const uchar *record, if (share->base.max_field_lengths) { - /* Store field lenghts, with a prefix of number of bytes */ + /* Store length of all not empty char, varchar and blob fields */ log_parts->str= field_lengths-2; log_parts->length= info->cur_row.field_lengths_length+2; int2store(log_parts->str, info->cur_row.field_lengths_length); @@ -4066,6 +4167,17 @@ static size_t fill_insert_undo_parts(MARIA_HA *info, const uchar *record, log_parts++; } + if (share->base.blobs) + { + /* Store total blob length to make buffer allocation easier during undo */ + log_parts->str= info->length_buff; + log_parts->length= (uint) (ma_store_length(log_parts->str, + info->cur_row.blob_length) - + (uchar*) log_parts->str); + row_length+= log_parts->length; + log_parts++; + } + /* Handle constant length fields that are always present */ for (column= share->columndef, end_column= column+ share->base.fixed_not_null_fields; @@ -4574,14 +4686,18 @@ uint _ma_apply_redo_purge_row_head_or_tail(MARIA_HA *info, LSN lsn, { MARIA_SHARE *share= info->s; ulonglong page; - uint record_number, empty_space; + uint rownr, empty_space; uint block_size= share->block_size; uchar *buff= info->keyread_buff; DBUG_ENTER("_ma_apply_redo_purge_row_head_or_tail"); - info->keyread_buff_used= 1; page= page_korr(header); - record_number= dirpos_korr(header+PAGE_STORE_SIZE); + rownr= dirpos_korr(header+PAGE_STORE_SIZE); + DBUG_PRINT("enter", ("rowid: %lu page: %lu rownr: %u", + (ulong) ma_recordpos(page, rownr), + (ulong) page, rownr)); + + info->keyread_buff_used= 1; if (!(buff= pagecache_read(share->pagecache, &info->dfile, @@ -4589,18 +4705,27 @@ uint _ma_apply_redo_purge_row_head_or_tail(MARIA_HA *info, LSN lsn, buff, PAGECACHE_PLAIN_PAGE, PAGECACHE_LOCK_LEFT_UNLOCKED, 0))) DBUG_RETURN(my_errno); - DBUG_ASSERT((buff[PAGE_TYPE_OFFSET] & PAGE_TYPE_MASK) == (uchar) page_type); if (lsn_korr(buff) >= lsn) { - /* Already applied */ - empty_space= uint2korr(buff + EMPTY_SPACE_OFFSET); - if (_ma_bitmap_set(info, page, page_type == HEAD_PAGE, empty_space)) - DBUG_RETURN(my_errno); + /* + Already applied + Note that in case the page is not anymore a head or tail page + a future redo will fix the bitmap. + */ + if ((buff[PAGE_TYPE_OFFSET] & PAGE_TYPE_MASK) == page_type) + { + empty_space= uint2korr(buff+EMPTY_SPACE_OFFSET); + if (_ma_bitmap_set(info, page, page_type == HEAD_PAGE, + empty_space)) + DBUG_RETURN(my_errno); + } DBUG_RETURN(0); } - if (delete_dir_entry(buff, block_size, record_number, &empty_space) < 0) + DBUG_ASSERT((buff[PAGE_TYPE_OFFSET] & PAGE_TYPE_MASK) == (uchar) page_type); + + if (delete_dir_entry(buff, block_size, rownr, &empty_space) < 0) DBUG_RETURN(HA_ERR_WRONG_IN_RECORD); lsn_store(buff, lsn); @@ -4698,7 +4823,7 @@ my_bool _ma_apply_undo_row_insert(MARIA_HA *info, LSN undo_lsn, const uchar *header) { ulonglong page; - uint record_number; + uint rownr; LEX_STRING log_array[TRANSLOG_INTERNAL_PARTS + 1]; uchar log_data[LSN_STORE_SIZE + FILEID_STORE_SIZE], *buff; my_bool res= 1; @@ -4707,9 +4832,8 @@ my_bool _ma_apply_undo_row_insert(MARIA_HA *info, LSN undo_lsn, DBUG_ENTER("_ma_apply_undo_row_insert"); page= page_korr(header); - record_number= dirpos_korr(header + PAGE_STORE_SIZE); - DBUG_PRINT("enter", ("Page: %lu record_number: %u", (ulong) page, - record_number)); + rownr= dirpos_korr(header + PAGE_STORE_SIZE); + DBUG_PRINT("enter", ("Page: %lu rownr: %u", (ulong) page, rownr)); if (!(buff= pagecache_read(info->s->pagecache, &info->dfile, page, 0, @@ -4722,24 +4846,25 @@ my_bool _ma_apply_undo_row_insert(MARIA_HA *info, LSN undo_lsn, page_link.unlock= PAGECACHE_LOCK_WRITE_UNLOCK; push_dynamic(&info->pinned_pages, (void*) &page_link); - if (read_row_extent_info(info, buff, record_number)) + if (read_row_extent_info(info, buff, rownr)) DBUG_RETURN(1); - if (delete_head_or_tail(info, page, record_number, 1, 1) || + if (delete_head_or_tail(info, page, rownr, 1, 1) || delete_tails(info, info->cur_row.tail_positions)) goto err; if (info->cur_row.extents && free_full_pages(info, &info->cur_row)) goto err; - lsn_store(log_data + FILEID_STORE_SIZE, undo_lsn); + /* undo_lsn must be first for compression to work */ + lsn_store(log_data, undo_lsn); log_array[TRANSLOG_INTERNAL_PARTS + 0].str= (char*) log_data; log_array[TRANSLOG_INTERNAL_PARTS + 0].length= sizeof(log_data); if (translog_write_record(&lsn, LOGREC_CLR_END, info->trn, info, sizeof(log_data), TRANSLOG_INTERNAL_PARTS + 1, log_array, - log_data)) + log_data+ FILEID_STORE_SIZE)) goto err; info->s->state.state.records--; @@ -4748,3 +4873,183 @@ err: _ma_unpin_all_pages(info, lsn); DBUG_RETURN(res); } + + +/* Execute undo of a row delete (insert the row back somewhere) */ + +my_bool _ma_apply_undo_row_delete(MARIA_HA *info, LSN undo_lsn, + const uchar *header, size_t length) +{ + uchar *record; + const uchar *null_bits, *field_length_data; + MARIA_SHARE *share= info->s; + MARIA_ROW row; + uint *null_field_lengths; + ulong *blob_lengths; + MARIA_COLUMNDEF *column, *end_column; + DBUG_ENTER("_ma_apply_undo_row_delete"); + + /* + Use cur row as a base; We need to make a copy as we will change + some buffers to point directly to 'header' + */ + memcpy(&row, &info->cur_row, sizeof(row)); + null_field_lengths= row.null_field_lengths; + blob_lengths= row.blob_lengths; + + /* + Fill in info->cur_row with information about the row, like in + calc_record_size(), to be used by write_block_record() + */ + + row.normal_length= row.char_length= row.varchar_length= + row.blob_length= row.extents_count= row.field_lengths_length= 0; + + null_bits= header; + header+= share->base.null_bytes; + row.empty_bits= (uchar*) header; + header+= share->base.pack_bytes; + if (share->base.max_field_lengths) + { + row.field_lengths_length= uint2korr(header); + row.field_lengths= (uchar*) header + 2 ; + header+= 2 + row.field_lengths_length; + } + if (share->base.blobs) + row.blob_length= ma_get_length((uchar**) &header); + + /* We need to build up a record (without blobs) in rec_buff */ + if (_ma_alloc_buffer(&info->rec_buff, &info->rec_buff_size, + length - row.blob_length)) + DBUG_RETURN(1); + + record= info->rec_buff; + memcpy(record, null_bits, share->base.null_bytes); + + /* Copy field information from header to record */ + + /* Handle constant length fields that are always present */ + for (column= share->columndef, + end_column= column+ share->base.fixed_not_null_fields; + column < end_column; + column++) + { + memcpy(record + column->offset, header, column->length); + header+= column->length; + } + + /* Handle NULL fields and CHAR/VARCHAR fields */ + field_length_data= row.field_lengths; + for (end_column= share->columndef + share->base.fields; + column < end_column; + column++, null_field_lengths++) + { + if ((record[column->null_pos] & column->null_bit) || + row.empty_bits[column->empty_pos] & column->empty_bit) + { + if (column->type != FIELD_BLOB) + *null_field_lengths= 0; + else + *blob_lengths++= 0; + if (share->calc_checksum) + bzero(record + column->offset, column->length); + continue; + } + switch ((enum en_fieldtype) column->type) { + case FIELD_CHECK: + case FIELD_NORMAL: /* Fixed length field */ + case FIELD_ZERO: + case FIELD_SKIP_PRESPACE: /* Not packed */ + case FIELD_SKIP_ZERO: /* Fixed length field */ + row.normal_length+= column->length; + *null_field_lengths= column->length; + memcpy(record + column->offset, header, column->length); + header+= column->length; + break; + case FIELD_SKIP_ENDSPACE: /* CHAR */ + if (column->length <= 255) + length= (uint) *field_length_data++; + else + { + length= uint2korr(field_length_data); + field_length_data+= 2; + } + row.char_length+= length; + *null_field_lengths= length; + memcpy(record + column->offset, header, length); + if (share->calc_checksum) + bfill(record + column->offset + length, (column->length - length), + ' '); + header+= length; + break; + case FIELD_VARCHAR: + { + uint length; + uchar *field_pos= record + column->offset; + + /* 256 is correct as this includes the length uchar */ + if (column->length <= 256) + { + field_pos[0]= *field_length_data; + length= (uint) *field_length_data++; + } + else + { + field_pos[0]= field_length_data[0]; + field_pos[1]= field_length_data[1]; + length= uint2korr(field_length_data); + field_length_data+= 2; + } + row.varchar_length+= length; + *null_field_lengths= length; + memcpy(record + column->offset, header, length); + header+= length; + break; + } + case FIELD_BLOB: + { + /* Copy length of blob and pointer to blob data to record */ + uchar *field_pos= record + column->offset; + uint size_length= column->length - portable_sizeof_char_ptr; + ulong blob_length= _ma_calc_blob_length(size_length, field_length_data); + + memcpy(field_pos, field_length_data, size_length); + field_length_data+= size_length; + memcpy(field_pos + size_length, &header, sizeof(&header)); + header+= blob_length; + *blob_lengths++= blob_length; + row.blob_length+= blob_length; + break; + } + default: + DBUG_ASSERT(0); + } + } + row.head_length= (row.base_length + + share->base.fixed_not_null_fields_length + + row.field_lengths_length + + size_to_store_key_length(row.field_lengths_length) + + row.normal_length + + row.char_length + row.varchar_length); + row.total_length= (row.head_length + row.blob_length); + if (row.total_length < share->base.min_row_length) + row.total_length= share->base.min_row_length; + + /* Row is now up to date. Time to insert the record */ + + DBUG_RETURN(allocate_and_write_block_record(info, record, &row, undo_lsn)); +} + + +/* Execute undo of a row update */ + +my_bool _ma_apply_undo_row_update(MARIA_HA *info __attribute__ ((unused)), + LSN undo_lsn __attribute__ ((unused)), + const uchar *header __attribute__ ((unused)), + size_t length __attribute__ ((unused))) +{ + DBUG_ENTER("_ma_apply_undo_row_update"); + fprintf(stderr, "Undo of row update is not yet done\n"); + exit(1); + DBUG_RETURN(0); +} diff --git a/storage/maria/ma_blockrec.h b/storage/maria/ma_blockrec.h index eff99355d62..cdaf8b9d124 100644 --- a/storage/maria/ma_blockrec.h +++ b/storage/maria/ma_blockrec.h @@ -189,3 +189,7 @@ uint _ma_apply_redo_purge_blocks(MARIA_HA *info, LSN lsn, const uchar *header); my_bool _ma_apply_undo_row_insert(MARIA_HA *info, LSN undo_lsn, const uchar *header); +my_bool _ma_apply_undo_row_delete(MARIA_HA *info, LSN undo_lsn, + const uchar *header, size_t length); +my_bool _ma_apply_undo_row_update(MARIA_HA *info, LSN undo_lsn, + const uchar *header, size_t length); diff --git a/storage/maria/ma_pagecache.c b/storage/maria/ma_pagecache.c index 7c0cc88d70a..d76d59f32ba 100755 --- a/storage/maria/ma_pagecache.c +++ b/storage/maria/ma_pagecache.c @@ -3258,7 +3258,9 @@ restart: DBUG_ASSERT(block->type == PAGECACHE_EMPTY_PAGE || block->type == PAGECACHE_READ_UNKNOWN_PAGE || - block->type == type); + block->type == type || + (block->type == PAGECACHE_PLAIN_PAGE && + type == PAGECACHE_LSN_PAGE)); block->type= type; if (make_lock_and_pin(pagecache, block, diff --git a/storage/maria/ma_recovery.c b/storage/maria/ma_recovery.c index 2d7a0ad7642..5810f8e06fe 100644 --- a/storage/maria/ma_recovery.c +++ b/storage/maria/ma_recovery.c @@ -76,6 +76,8 @@ prototype_redo_exec_hook(UNDO_ROW_UPDATE); prototype_redo_exec_hook(UNDO_ROW_PURGE); prototype_redo_exec_hook(COMMIT); prototype_undo_exec_hook(UNDO_ROW_INSERT); +prototype_undo_exec_hook(UNDO_ROW_DELETE); +prototype_undo_exec_hook(UNDO_ROW_UPDATE); static int run_redo_phase(LSN lsn, my_bool apply); static uint end_of_redo_phase(my_bool prepare_for_undo_phase); @@ -977,6 +979,88 @@ prototype_undo_exec_hook(UNDO_ROW_INSERT) } +prototype_undo_exec_hook(UNDO_ROW_DELETE) +{ + my_bool error; + MARIA_HA *info= get_MARIA_HA_from_UNDO_record(rec); + + if (info == NULL) + return 1; + + info->s->state.changed|= STATE_CHANGED | STATE_NOT_ANALYZED | + STATE_NOT_OPTIMIZED_KEYS | STATE_NOT_SORTED_PAGES; + + enlarge_buffer(rec); + if (log_record_buffer.str == NULL || + translog_read_record(rec->lsn, 0, rec->record_length, + log_record_buffer.str, NULL) != + rec->record_length) + { + fprintf(tracef, "Failed to read record\n"); + return 1; + } + + /* Set undo to point to previous undo record */ + info->trn= trn; + info->trn->undo_lsn= lsn_korr(rec->header); + + /* + For now we skip the page and directory entry. This is to be used + later when we mark rows as deleted. + */ + error= _ma_apply_undo_row_delete(info, rec->lsn, + log_record_buffer.str + LSN_STORE_SIZE + + FILEID_STORE_SIZE + PAGE_STORE_SIZE + + DIRPOS_STORE_SIZE, + rec->record_length - + (LSN_STORE_SIZE + FILEID_STORE_SIZE + + PAGE_STORE_SIZE + DIRPOS_STORE_SIZE)); + info->trn= 0; + return error; +} + + +prototype_undo_exec_hook(UNDO_ROW_UPDATE) +{ + my_bool error; + MARIA_HA *info= get_MARIA_HA_from_UNDO_record(rec); + + if (info == NULL) + return 1; + + info->s->state.changed|= STATE_CHANGED | STATE_NOT_ANALYZED | + STATE_NOT_OPTIMIZED_KEYS | STATE_NOT_SORTED_PAGES; + + enlarge_buffer(rec); + if (log_record_buffer.str == NULL || + translog_read_record(rec->lsn, 0, rec->record_length, + log_record_buffer.str, NULL) != + rec->record_length) + { + fprintf(tracef, "Failed to read record\n"); + return 1; + } + + /* Set undo to point to previous undo record */ + info->trn= trn; + info->trn->undo_lsn= lsn_korr(rec->header); + + /* + For now we skip the page and directory entry. This is to be used + later when we mark rows as deleted. + */ + error= _ma_apply_undo_row_update(info, rec->lsn, + log_record_buffer.str + LSN_STORE_SIZE + + FILEID_STORE_SIZE + PAGE_STORE_SIZE + + DIRPOS_STORE_SIZE, + rec->record_length - + (LSN_STORE_SIZE + FILEID_STORE_SIZE + + PAGE_STORE_SIZE + DIRPOS_STORE_SIZE)); + info->trn= 0; + return error; +} + + static int run_redo_phase(LSN lsn, my_bool apply) { /* install hooks for execution */ @@ -1003,6 +1087,8 @@ static int run_redo_phase(LSN lsn, my_bool apply) install_redo_exec_hook(UNDO_ROW_PURGE); install_redo_exec_hook(COMMIT); install_undo_exec_hook(UNDO_ROW_INSERT); + install_undo_exec_hook(UNDO_ROW_DELETE); + install_undo_exec_hook(UNDO_ROW_UPDATE); current_group_end_lsn= LSN_IMPOSSIBLE; diff --git a/storage/maria/ma_test1.c b/storage/maria/ma_test1.c index 507e51ced35..4435de0bbdb 100644 --- a/storage/maria/ma_test1.c +++ b/storage/maria/ma_test1.c @@ -37,8 +37,9 @@ static enum data_file_type record_type= DYNAMIC_RECORD; static uint insert_count, update_count, remove_count; static uint pack_keys=0, pack_seg=0, key_length; static uint unique_key=HA_NOSAME; -static my_bool pagecacheing, null_fields, silent, skip_update, opt_unique, - verbose, skip_delete, transactional, die_in_middle_of_transaction; +static uint die_in_middle_of_transaction; +static my_bool pagecacheing, null_fields, silent, skip_update, opt_unique; +static my_bool verbose, skip_delete, transactional; static MARIA_COLUMNDEF recinfo[4]; static MARIA_KEYDEF keyinfo[10]; static HA_KEYSEG keyseg[10]; @@ -94,6 +95,7 @@ static int run_test(const char *filename) { MARIA_HA *file; int i,j,error,deleted,rec_length,uniques=0; + uint offset_to_key; ha_rows found,row_count; char record[MAX_REC_LENGTH],key[MAX_REC_LENGTH],read_record[MAX_REC_LENGTH]; MARIA_UNIQUEDEF uniquedef; @@ -182,6 +184,10 @@ static int run_test(const char *filename) else uniques=0; + offset_to_key= test(null_fields); + if (key_field == FIELD_BLOB) + offset_to_key+= 2; + if (!silent) printf("- Creating maria file\n"); create_info.max_rows=(ulong) (rec_pointer_size ? @@ -234,7 +240,7 @@ static int run_test(const char *filename) flags[0]=2; } - if (die_in_middle_of_transaction) + if (die_in_middle_of_transaction == 1) { /* Ensure we get changed pages and log to disk @@ -242,6 +248,7 @@ static int run_test(const char *filename) */ _ma_flush_table_files(file, MARIA_FLUSH_DATA, FLUSH_RELEASE, FLUSH_RELEASE); + printf("Dying on request after insert without maria_close()\n"); exit(1); } @@ -333,14 +340,14 @@ static int run_test(const char *filename) if (verbose || (flags[j] >= 1 || (error && my_errno != HA_ERR_KEY_NOT_FOUND))) printf("key: '%.*s' maria_rkey: %3d errno: %3d\n", - (int) key_length,key+test(null_fields),error,my_errno); + (int) key_length,key+offset_to_key,error,my_errno); } else { error=maria_delete(file,read_record); if (verbose || error) printf("key: '%.*s' maria_delete: %3d errno: %3d\n", - (int) key_length, key+test(null_fields), error, my_errno); + (int) key_length, key+offset_to_key, error, my_errno); if (! error) { deleted++; @@ -348,6 +355,18 @@ static int run_test(const char *filename) } } } + + if (die_in_middle_of_transaction == 2) + { + /* + Ensure we get changed pages and log to disk + As commit record is not done, the undo entries needs to be rolled back. + */ + _ma_flush_table_files(file, MARIA_FLUSH_DATA, FLUSH_RELEASE, + FLUSH_RELEASE); + printf("Dying on request after delete without maria_close()\n"); + exit(1); + } } if (!silent) printf("- Reading rows with key\n"); @@ -362,7 +381,7 @@ static int run_test(const char *filename) (error && (flags[i] != 0 || my_errno != HA_ERR_KEY_NOT_FOUND))) { printf("key: '%.*s' maria_rkey: %3d errno: %3d record: %s\n", - (int) key_length,key+test(null_fields),error,my_errno,record+1); + (int) key_length,key+offset_to_key,error,my_errno,record+1); } } @@ -661,7 +680,7 @@ static struct my_option my_long_options[] = "Abort hard after doing inserts. Used for testing recovery with undo", (uchar**) &die_in_middle_of_transaction, (uchar**) &die_in_middle_of_transaction, - 0, GET_BOOL, NO_ARG, 0, 0, 0, 0, 0, 0}, + 0, GET_INT, OPT_ARG, 0, 0, 0, 0, 0, 0}, {"transactional", 'T', "Test in transactional mode. (Only works with block format)", (uchar**) &transactional, (uchar**) &transactional, 0, GET_BOOL, NO_ARG, @@ -749,6 +768,12 @@ get_one_option(int optid, const struct my_option *opt __attribute__((unused)), case 'K': /* Use key cacheing */ pagecacheing=1; break; + case 'A': + if (!argument) + die_in_middle_of_transaction= 1; + else + die_in_middle_of_transaction= atoi(argument); + break; case 'V': printf("test1 Ver 1.2 \n"); exit(0); diff --git a/storage/maria/maria_def.h b/storage/maria/maria_def.h index f5e143fe6f8..e8b53757a53 100644 --- a/storage/maria/maria_def.h +++ b/storage/maria/maria_def.h @@ -369,10 +369,11 @@ typedef struct st_maria_row ulong *blob_lengths; /* Length for each blob */ ulong base_length, normal_length, char_length, varchar_length, blob_length; ulong head_length, total_length; - size_t extents_buffer_length; /* Size of 'extents' buffer */ + size_t extents_buffer_length; /* Size of 'extents' buffer */ uint field_lengths_length; /* Length of data in field_lengths */ uint extents_count; /* number of extents in 'extents' */ uint full_page_count, tail_count; /* For maria_chk */ + uint space_on_head_page; } MARIA_ROW; /* Data to scan row in blocked format */ @@ -434,6 +435,8 @@ struct st_maria_info ulong packed_length, blob_length; /* Length of found, packed record */ size_t rec_buff_size; PAGECACHE_FILE dfile; /* The datafile */ + IO_CACHE rec_cache; /* When cacheing records */ + LIST open_list; uint opt_flag; /* Optim. for space/speed */ uint update; /* If file changed since open */ int lastinx; /* Last used index */ @@ -449,8 +452,6 @@ struct st_maria_info uint data_changed; /* Somebody has changed data */ uint save_update; /* When using KEY_READ */ int save_lastinx; - LIST open_list; - IO_CACHE rec_cache; /* When cacheing records */ uint preload_buff_size; /* When preloading indexes */ myf lock_wait; /* is 0 or MY_DONT_WAIT */ my_bool was_locked; /* Was locked in panic */ @@ -468,6 +469,7 @@ struct st_maria_info THR_LOCK_DATA lock; #endif uchar *maria_rtree_recursion_state; /* For RTREE */ + uchar length_buff[5]; /* temp buff to store blob lengths */ int maria_rtree_recursion_depth; }; -- cgit v1.2.1 From ac4ad9bdba4082c184443858da11bf4b61d582ff Mon Sep 17 00:00:00 2001 From: unknown Date: Thu, 6 Sep 2007 16:04:36 +0200 Subject: WL#3072 Maria Recovery misc fixes of execution of UNDOs in the UNDO phase: - into the CLR_END, store the LSN of the _previous_ UNDO (we debated what was best, so far we're going with "previous"; later we can change to "current" if needed), and store the type of record which is being undone (needed to know how to update state.records when we see the CLR_END during the REDO phase). - declaring all UNDOs and CLR_END as "compressed" - when executing an UNDO in the UNDO phase, state.records is updated as a hook when writing CLR_END (needed for "recovery of the state"), and so is trn->undo_lsn (needed for when we have checkpoints). - bugfix (execution of UNDO_ROW_DELETE didn't store the correct checksum into the re-inserted row, maria_chk -r thus threw the row away). - modifications of ma_test1: where to stop is now driven by --testflag; --test-undo just tells how to stop (flush data, flush log, nothing). - ma_test_recovery: testing of the UNDO phase, more testing of the REDO phase, identification of a bug. storage/maria/ma_blockrec.c: - bugfix: execution of UNDO_ROW_DELETE didn't store the correct checksum into the row (leading to "maria_chk -r" eliminating the re-inserted row, net effect was that rollback appeared to have rolled back no deletion). Reason was that write_block_record() used info->cur_row.checksum, while "row" can be != &info->cur_row (case of UNDO_ROW_DELETE). After fixing this, problems with _ma_update_block_record() appeared; indeed checksum was computed by allocate_and_write_block_record() while _ma_update_block_record() directly calls write_block_record(). Solution is to compute checksum in write_block_record() instead. - when executing an UNDO, we now pass the LSN of the _previous_ UNDO to block_format functions. This LSN can be 0 (if the being-executed UNDO was the transaction's first UNDO), so "undo_lsn==0" cannot work anymore to indicate "this is not UNDO work". Using undo_lsn==LSN_ERROR instead (this is an impossible LSN). - store into CLR_END the type of log record which was undone (INSERT/UPDATE/DELETE); needed for Recovery to know if/how it has to update state.records if it sees this CLR_END in the REDO phase. - when writing the CLR_END in _ma_apply_undo_row_insert(), the place to store file's id is log_data+LSN_STORE_SIZE. - in _ma_apply_undo_row_insert(), the records-- is moved to a hook when writing the CLR_END (this way it is under log's mutex which is needed for "recovery of the state") storage/maria/ma_loghandler.c: - all UNDOs, and CLR_END, start with the LSN of another UNDO; so we can declare them "compressed". - write_hook_for_clr_end() to set trn->undo_lsn (to the previous UNDO's LSN) under log's lock (like UNDOs set trn->undo_lsn under log's lock), and also update, if appropriate, state.records. - reset share->id to 0 when deassigning; not useful for now but sounds logical. storage/maria/ma_recovery.c: - if no table is found for a REDO, it's not an error; for an UNDO, it is - in the REDO phase, when we see a CLR_END we must update trn->undo_lsn and sometimes state.records. - in the UNDO phase, when we execute an UNDO_ROW_INSERT: * update trn->undo_lsn only after executing the record * store the _previous_ undo_lsn into the CLR_END - at the end of the REDO phase, when we recreate TRN objects, they have already their long id in the log (either via a LOGREC_LONG_TRANSACTION_ID, or in a checkpoint record), don't write a new, useless LOGREC_LONG_TRANSACTION_ID for them. storage/maria/ma_test1.c: * where to stop execution is now driven by --testflag and not --test-undo (ma_test2 already has --testflag for the same purpose). This allows us to do a clean stop (with commit) at any point. * --test-undo=# tells how to abort (flush all pages (which implies flushing log) or only log or nothing); all such "ways of crashing" are tested in ma_test_recovery storage/maria/ma_test_recovery: * Testing execution of UNDOs, with and without BLOBs. * Testing idempotency of REDOs. * See @todo for a probable bug with BLOBs. * maria_chk -rq instead of -r, as with -q it nicely stops on any problem in the data file (like the checksum bug see comment of ma_blockrec.c). * Testing if log was written by UNDO phase (often expected), not written by REDO phase (always expected). * Less output on the screen, compares with expected output in the end. * some shell thingies like "set --" and $# are courtesy of Danny and Pekka. storage/maria/maria_read_log.c: when only displaying the records, don't do an UNDO phase storage/maria/ma_test_recovery.expected: This is the expected output of a great part of ma_test_recovery. ma_test_recovery compares its output to the expected output and tells if different. If we look at this file it mentions differences in checksum (normal, it's not recovered yet) and in records count (getting a correct records' count when recovery starts on an already existing table, like when testing rollback, is coded but not yet pushed). --- storage/maria/ma_blockrec.c | 38 +- storage/maria/ma_loghandler.c | 66 +++- storage/maria/ma_recovery.c | 135 ++++--- storage/maria/ma_test1.c | 90 +++-- storage/maria/ma_test_recovery | 205 ++++++++-- storage/maria/ma_test_recovery.expected | 675 ++++++++++++++++++++++++++++++++ storage/maria/maria_read_log.c | 3 +- 7 files changed, 1085 insertions(+), 127 deletions(-) create mode 100644 storage/maria/ma_test_recovery.expected diff --git a/storage/maria/ma_blockrec.c b/storage/maria/ma_blockrec.c index 55dc72b1d02..66cf58efffa 100644 --- a/storage/maria/ma_blockrec.c +++ b/storage/maria/ma_blockrec.c @@ -1659,7 +1659,7 @@ static my_bool free_full_page_range(MARIA_HA *info, ulonglong page, uint count) @param map_blocks On which pages the record should be stored @param row_pos Position on head page where to put head part of record - @param undo_lsn <> 0 if we are in UNDO + @param undo_lsn <> LSN_ERROR if we are executing an UNDO @note On return all pinned pages are released. @@ -1729,7 +1729,10 @@ static my_bool write_block_record(MARIA_HA *info, if (share->base.pack_fields) store_key_length_inc(data, row->field_lengths_length); if (share->calc_checksum) - *(data++)= (uchar) info->cur_row.checksum; + { + row->checksum= (info->s->calc_checksum)(info, record); + *(data++)= (uchar) (row->checksum); /* store least significant byte */ + } memcpy(data, record, share->base.null_bytes); data+= share->base.null_bytes; memcpy(data, row->empty_bits, share->base.pack_bytes); @@ -2283,19 +2286,25 @@ static my_bool write_block_record(MARIA_HA *info, { LEX_STRING *log_array= info->log_row_parts; - if (undo_lsn) + if (undo_lsn != LSN_ERROR) { - uchar log_data[LSN_STORE_SIZE + FILEID_STORE_SIZE]; - + uchar log_data[LSN_STORE_SIZE + FILEID_STORE_SIZE + 1]; /* undo_lsn must be first for compression to work */ lsn_store(log_data, undo_lsn); + /* + Store if this CLR is about an UNDO_INSERT, UNDO_DELETE or UNDO_UPDATE; + in the first/second case, Recovery, when it sees the CLR_END in the + REDO phase, may decrement/increment the records' count. + */ + /** @todo when Monty has UNDO_UPDATE coded, revisit this */ + log_data[LSN_STORE_SIZE + FILEID_STORE_SIZE]= LOGREC_UNDO_ROW_DELETE; log_array[TRANSLOG_INTERNAL_PARTS + 0].str= (char*) log_data; log_array[TRANSLOG_INTERNAL_PARTS + 0].length= sizeof(log_data); if (translog_write_record(&lsn, LOGREC_CLR_END, info->trn, info, sizeof(log_data), TRANSLOG_INTERNAL_PARTS + 1, log_array, - log_data+ FILEID_STORE_SIZE)) + log_data + LSN_STORE_SIZE)) goto disk_err; } else @@ -2425,7 +2434,7 @@ disk_err: @param info Maria handler @param record Record to write @param row Information about fields in 'record' - @param undo_lsn <> 0 if in undo + @param undo_lsn <> LSN_ERROR if we are executing an UNDO @return @retval 0 ok @@ -2449,8 +2458,6 @@ static my_bool allocate_and_write_block_record(MARIA_HA *info, PAGECACHE_LOCK_WRITE, &row_pos)) DBUG_RETURN(1); row->lastpos= ma_recordpos(blocks->block->page, row_pos.rownr); - if (info->s->calc_checksum) - row->checksum= (info->s->calc_checksum)(info,record); if (write_block_record(info, (uchar*) 0, record, row, blocks, blocks->block->org_bitmap_value != 0, &row_pos, undo_lsn)) @@ -2482,7 +2489,8 @@ MARIA_RECORD_POS _ma_write_init_block_record(MARIA_HA *info, DBUG_ENTER("_ma_write_init_block_record"); calc_record_size(info, record, &info->cur_row); - if (allocate_and_write_block_record(info, record, &info->cur_row, 0)) + if (allocate_and_write_block_record(info, record, + &info->cur_row, LSN_ERROR)) DBUG_RETURN(HA_OFFSET_ERROR); DBUG_RETURN(info->cur_row.lastpos); } @@ -2669,7 +2677,7 @@ my_bool _ma_update_block_record(MARIA_HA *info, MARIA_RECORD_POS record_pos, if (cur_row->extents_count && free_full_pages(info, cur_row)) goto err; DBUG_RETURN(write_block_record(info, oldrec, record, new_row, blocks, - 1, &row_pos, 0)); + 1, &row_pos, LSN_ERROR)); } /* Allocate all size in block for record @@ -2702,7 +2710,7 @@ my_bool _ma_update_block_record(MARIA_HA *info, MARIA_RECORD_POS record_pos, row_pos.data= buff + uint2korr(dir); row_pos.length= head_length; DBUG_RETURN(write_block_record(info, oldrec, record, new_row, blocks, 1, - &row_pos, 0)); + &row_pos, LSN_ERROR)); err: _ma_unpin_all_pages(info, 0); @@ -4825,7 +4833,7 @@ my_bool _ma_apply_undo_row_insert(MARIA_HA *info, LSN undo_lsn, ulonglong page; uint rownr; LEX_STRING log_array[TRANSLOG_INTERNAL_PARTS + 1]; - uchar log_data[LSN_STORE_SIZE + FILEID_STORE_SIZE], *buff; + uchar log_data[LSN_STORE_SIZE + FILEID_STORE_SIZE + 1], *buff; my_bool res= 1; MARIA_PINNED_PAGE page_link; LSN lsn; @@ -4858,16 +4866,16 @@ my_bool _ma_apply_undo_row_insert(MARIA_HA *info, LSN undo_lsn, /* undo_lsn must be first for compression to work */ lsn_store(log_data, undo_lsn); + log_data[LSN_STORE_SIZE + FILEID_STORE_SIZE]= LOGREC_UNDO_ROW_INSERT; log_array[TRANSLOG_INTERNAL_PARTS + 0].str= (char*) log_data; log_array[TRANSLOG_INTERNAL_PARTS + 0].length= sizeof(log_data); if (translog_write_record(&lsn, LOGREC_CLR_END, info->trn, info, sizeof(log_data), TRANSLOG_INTERNAL_PARTS + 1, log_array, - log_data+ FILEID_STORE_SIZE)) + log_data + LSN_STORE_SIZE)) goto err; - info->s->state.state.records--; res= 0; err: _ma_unpin_all_pages(info, lsn); diff --git a/storage/maria/ma_loghandler.c b/storage/maria/ma_loghandler.c index 56c0e1aaef7..f556193b147 100644 --- a/storage/maria/ma_loghandler.c +++ b/storage/maria/ma_loghandler.c @@ -213,6 +213,9 @@ static my_bool write_hook_for_redo(enum translog_record_type type, static my_bool write_hook_for_undo(enum translog_record_type type, TRN *trn, MARIA_HA *tbl_info, LSN *lsn, struct st_translog_parts *parts); +static my_bool write_hook_for_clr_end(enum translog_record_type type, + TRN *trn, MARIA_HA *tbl_info, LSN *lsn, + struct st_translog_parts *parts); static my_bool translog_page_validator(uchar *page_addr, uchar* data_ptr); @@ -414,7 +417,8 @@ static LOG_DESC INIT_LOGREC_REDO_UNDELETE_ROW= "redo_undelete_row", LOGREC_NOT_LAST_IN_GROUP, NULL, NULL}; static LOG_DESC INIT_LOGREC_CLR_END= -{LOGRECTYPE_FIXEDLENGTH, 9, 9, NULL, write_hook_for_redo, NULL, 0, +{LOGRECTYPE_PSEUDOFIXEDLENGTH, LSN_STORE_SIZE + FILEID_STORE_SIZE + 1, + LSN_STORE_SIZE + FILEID_STORE_SIZE + 1, NULL, write_hook_for_clr_end, NULL, 1, "clr_end", LOGREC_LAST_IN_GROUP, NULL, NULL}; static LOG_DESC INIT_LOGREC_PURGE_END= @@ -422,16 +426,16 @@ static LOG_DESC INIT_LOGREC_PURGE_END= "purge_end", LOGREC_LAST_IN_GROUP, NULL, NULL}; static LOG_DESC INIT_LOGREC_UNDO_ROW_INSERT= -{LOGRECTYPE_FIXEDLENGTH, +{LOGRECTYPE_PSEUDOFIXEDLENGTH, LSN_STORE_SIZE + FILEID_STORE_SIZE + PAGE_STORE_SIZE + DIRPOS_STORE_SIZE, LSN_STORE_SIZE + FILEID_STORE_SIZE + PAGE_STORE_SIZE + DIRPOS_STORE_SIZE, - NULL, write_hook_for_undo, NULL, 0, + NULL, write_hook_for_undo, NULL, 1, "undo_row_insert", LOGREC_LAST_IN_GROUP, NULL, NULL}; static LOG_DESC INIT_LOGREC_UNDO_ROW_DELETE= {LOGRECTYPE_VARIABLE_LENGTH, 0, LSN_STORE_SIZE + FILEID_STORE_SIZE + PAGE_STORE_SIZE + DIRPOS_STORE_SIZE, - NULL, write_hook_for_undo, NULL, 0, + NULL, write_hook_for_undo, NULL, 1, "undo_row_delete", LOGREC_LAST_IN_GROUP, NULL, NULL}; static LOG_DESC INIT_LOGREC_UNDO_ROW_UPDATE= @@ -451,8 +455,8 @@ static LOG_DESC INIT_LOGREC_UNDO_KEY_INSERT= "undo_key_insert", LOGREC_LAST_IN_GROUP, NULL, NULL}; static LOG_DESC INIT_LOGREC_UNDO_KEY_DELETE= -{LOGRECTYPE_VARIABLE_LENGTH, 0, 15, NULL, write_hook_for_undo, NULL, 0, - "undo_key_delete", LOGREC_LAST_IN_GROUP, NULL, NULL}; // QQ: why not compressed? +{LOGRECTYPE_VARIABLE_LENGTH, 0, 15, NULL, write_hook_for_undo, NULL, 1, + "undo_key_delete", LOGREC_LAST_IN_GROUP, NULL, NULL}; static LOG_DESC INIT_LOGREC_PREPARE= {LOGRECTYPE_VARIABLE_LENGTH, 0, 0, NULL, NULL, NULL, 0, @@ -6303,6 +6307,46 @@ static my_bool write_hook_for_undo(enum translog_record_type type */ } + +/** + @brief Sets transaction's undo_lsn, first_undo_lsn if needed + + @todo move it to a separate file + + @return Operation status, always 0 (success) +*/ + +static my_bool write_hook_for_clr_end(enum translog_record_type type + __attribute__ ((unused)), + TRN *trn, MARIA_HA *tbl_info + __attribute__ ((unused)), + LSN *lsn + __attribute__ ((unused)), + struct st_translog_parts *parts) +{ + char *ptr= parts->parts[TRANSLOG_INTERNAL_PARTS + 0].str; + enum translog_record_type undone_record_type= + ptr[LSN_STORE_SIZE + FILEID_STORE_SIZE]; + + DBUG_ASSERT(trn->trid != 0); + /** @todo depending on what we are undoing, update "records" or not */ + trn->undo_lsn= lsn_korr(ptr); + switch (undone_record_type) { + case LOGREC_UNDO_ROW_DELETE: + tbl_info->s->state.state.records++; + break; + case LOGREC_UNDO_ROW_INSERT: + tbl_info->s->state.state.records--; + break; + default: + DBUG_ASSERT(0); + } + if (trn->undo_lsn == LSN_IMPOSSIBLE) /* has fully rolled back */ + trn->first_undo_lsn= LSN_WITH_FLAGS_TO_FLAGS(trn->first_undo_lsn); + return 0; +} + + /** @brief Gives a 2-byte-id to MARIA_SHARE and logs this fact @@ -6375,6 +6419,15 @@ int translog_assign_id_to_share(MARIA_SHARE *share, TRN *trn) sizeof(log_array)/sizeof(log_array[0]), log_array, NULL))) return 1; + /* + Note that we first set share->id then write the record. The checkpoint + record does not include any share with id==0; this is ok because: + checkpoint_start_log_horizon is either before or after the above + record. If before, ok to not include the share, as the record will be + seen for sure during the REDO phase. If after, Checkpoint will see all + data as it was after this record was written, including the id!=0, so + share will be included. + */ } pthread_mutex_unlock(&share->intern_lock); return 0; @@ -6400,6 +6453,7 @@ void translog_deassign_id_from_share(MARIA_SHARE *share) my_atomic_rwlock_rdlock(&LOCK_id_to_share); my_atomic_storeptr((void **)&id_to_share[share->id], 0); my_atomic_rwlock_rdunlock(&LOCK_id_to_share); + share->id= 0; } diff --git a/storage/maria/ma_recovery.c b/storage/maria/ma_recovery.c index 5810f8e06fe..d65202f045e 100644 --- a/storage/maria/ma_recovery.c +++ b/storage/maria/ma_recovery.c @@ -75,6 +75,7 @@ prototype_redo_exec_hook(UNDO_ROW_DELETE); prototype_redo_exec_hook(UNDO_ROW_UPDATE); prototype_redo_exec_hook(UNDO_ROW_PURGE); prototype_redo_exec_hook(COMMIT); +prototype_redo_exec_hook(CLR_END); prototype_undo_exec_hook(UNDO_ROW_INSERT); prototype_undo_exec_hook(UNDO_ROW_DELETE); prototype_undo_exec_hook(UNDO_ROW_UPDATE); @@ -385,7 +386,7 @@ prototype_redo_exec_hook(REDO_CREATE_TABLE) if (cmp_translog_addr(share->state.create_rename_lsn, rec->lsn) >= 0) { fprintf(tracef, ", has create_rename_lsn (%lu,0x%lx) more recent than" - " record, ignoring", + " record, ignoring creation", LSN_IN_HEX(share->state.create_rename_lsn)); error= 0; goto end; @@ -502,7 +503,7 @@ prototype_redo_exec_hook(REDO_DROP_TABLE) if (cmp_translog_addr(share->state.create_rename_lsn, rec->lsn) >= 0) { fprintf(tracef, ", has create_rename_lsn (%lu,0x%lx) more recent than" - " record, ignoring", + " record, ignoring removal", LSN_IN_HEX(share->state.create_rename_lsn)); error= 0; goto end; @@ -625,7 +626,7 @@ static int new_table(uint16 sid, const char *name, if (cmp_translog_addr(lsn, share->state.create_rename_lsn) <= 0) { fprintf(tracef, ", has create_rename_lsn (%lu,0x%lx) more recent than" - " record, ignoring", + " record, ignoring open request", LSN_IN_HEX(share->state.create_rename_lsn)); error= -1; goto end; @@ -652,9 +653,10 @@ static int new_table(uint16 sid, const char *name, all_tables[sid].info= info; all_tables[sid].org_kfile= org_kfile; all_tables[sid].org_dfile= org_dfile; - fprintf(tracef, ", opened\n"); + fprintf(tracef, ", opened"); error= 0; end: + fprintf(tracef, "\n"); if (error) { if (info != NULL) @@ -672,7 +674,14 @@ prototype_redo_exec_hook(REDO_INSERT_ROW_HEAD) uchar *buff= NULL; MARIA_HA *info= get_MARIA_HA_from_REDO_record(rec); if (info == NULL) - goto end; + { + /* + Table was skipped at open time (because later dropped/renamed, not + transactional, or create_rename_lsn newer than LOGREC_FILE_ID); it is + not an error. + */ + return 0; + } /* If REDO's LSN is > page's LSN (read from disk), we are going to modify the page and change its LSN. The normal runtime code stores the UNDO's LSN @@ -717,7 +726,7 @@ prototype_redo_exec_hook(REDO_INSERT_ROW_TAIL) uchar *buff; MARIA_HA *info= get_MARIA_HA_from_REDO_record(rec); if (info == NULL) - goto end; + return 0; enlarge_buffer(rec); if (log_record_buffer.str == NULL || translog_read_record(rec->lsn, 0, rec->record_length, @@ -752,7 +761,7 @@ prototype_redo_exec_hook(REDO_PURGE_ROW_HEAD) int error= 1; MARIA_HA *info= get_MARIA_HA_from_REDO_record(rec); if (info == NULL) - goto end; + return 0; if (_ma_apply_redo_purge_row_head_or_tail(info, current_group_end_lsn, HEAD_PAGE, rec->header + FILEID_STORE_SIZE)) @@ -768,7 +777,7 @@ prototype_redo_exec_hook(REDO_PURGE_ROW_TAIL) int error= 1; MARIA_HA *info= get_MARIA_HA_from_REDO_record(rec); if (info == NULL) - goto end; + return 0; if (_ma_apply_redo_purge_row_head_or_tail(info, current_group_end_lsn, TAIL_PAGE, rec->header + FILEID_STORE_SIZE)) @@ -785,7 +794,7 @@ prototype_redo_exec_hook(REDO_PURGE_BLOCKS) uchar *buff; MARIA_HA *info= get_MARIA_HA_from_REDO_record(rec); if (info == NULL) - goto end; + return 0; enlarge_buffer(rec); if (log_record_buffer.str == NULL || @@ -812,7 +821,7 @@ prototype_redo_exec_hook(REDO_DELETE_ALL) int error= 1; MARIA_HA *info= get_MARIA_HA_from_REDO_record(rec); if (info == NULL) - goto end; + return 0; fprintf(tracef, " deleting all %lu rows\n", (ulong)info->s->state.state.records); if (maria_delete_all_rows(info)) @@ -830,10 +839,9 @@ end: prototype_redo_exec_hook(UNDO_ROW_INSERT) { - int error= 1; MARIA_HA *info= get_MARIA_HA_from_UNDO_record(rec); if (info == NULL) - goto end; + return 0; set_undo_lsn_for_active_trans(rec->short_trid, rec->lsn); /* in an upcoming patch ("recovery of the state"), we introduce @@ -852,19 +860,15 @@ prototype_redo_exec_hook(UNDO_ROW_INSERT) STATE_NOT_OPTIMIZED_KEYS | STATE_NOT_SORTED_PAGES; } fprintf(tracef, " rows' count %lu\n", (ulong)info->s->state.state.records); - error= 0; - -end: - return error; + return 0; } prototype_redo_exec_hook(UNDO_ROW_DELETE) { - int error= 1; MARIA_HA *info= get_MARIA_HA_from_UNDO_record(rec); if (info == NULL) - goto end; + return 0; set_undo_lsn_for_active_trans(rec->short_trid, rec->lsn); { fprintf(tracef, " state older than record, updating rows' count\n"); @@ -873,35 +877,29 @@ prototype_redo_exec_hook(UNDO_ROW_DELETE) STATE_NOT_OPTIMIZED_KEYS | STATE_NOT_SORTED_PAGES; } fprintf(tracef, " rows' count %lu\n", (ulong)info->s->state.state.records); - error= 0; -end: - return error; + return 0; } prototype_redo_exec_hook(UNDO_ROW_UPDATE) { - int error= 1; MARIA_HA *info= get_MARIA_HA_from_UNDO_record(rec); if (info == NULL) - goto end; + return 0; set_undo_lsn_for_active_trans(rec->short_trid, rec->lsn); { info->s->state.changed|= STATE_CHANGED | STATE_NOT_ANALYZED | STATE_NOT_OPTIMIZED_KEYS | STATE_NOT_SORTED_PAGES; } - error= 0; -end: - return error; + return 0; } prototype_redo_exec_hook(UNDO_ROW_PURGE) { - int error= 1; MARIA_HA *info= get_MARIA_HA_from_UNDO_record(rec); if (info == NULL) - goto end; + return 0; /* this a bit broken, but this log record type will be deleted soon */ set_undo_lsn_for_active_trans(rec->short_trid, rec->lsn); { @@ -911,9 +909,7 @@ prototype_redo_exec_hook(UNDO_ROW_PURGE) STATE_NOT_OPTIMIZED_KEYS | STATE_NOT_SORTED_PAGES; } fprintf(tracef, " rows' count %lu\n", (ulong)info->s->state.state.records); - error= 0; -end: - return error; + return 0; } @@ -956,25 +952,67 @@ prototype_redo_exec_hook(COMMIT) } +prototype_redo_exec_hook(CLR_END) +{ + MARIA_HA *info= get_MARIA_HA_from_UNDO_record(rec); + if (info == NULL) + return 0; + LSN previous_undo_lsn= lsn_korr(rec->header); + enum translog_record_type undone_record_type= + (rec->header)[LSN_STORE_SIZE + FILEID_STORE_SIZE]; + const LOG_DESC *log_desc= &log_record_type_descriptor[undone_record_type]; + + set_undo_lsn_for_active_trans(rec->short_trid, previous_undo_lsn); + fprintf(tracef, " CLR_END was about %s, undo_lsn now LSN (%lu,0x%lx)\n", + log_desc->name, LSN_IN_HEX(previous_undo_lsn)); + { + fprintf(tracef, " state older than record, updating rows' count\n"); + switch (undone_record_type) { + case LOGREC_UNDO_ROW_DELETE: + info->s->state.state.records++; + break; + case LOGREC_UNDO_ROW_INSERT: + info->s->state.state.records--; + break; + default: + DBUG_ASSERT(0); + } + info->s->state.changed|= STATE_CHANGED | STATE_NOT_ANALYZED | + STATE_NOT_OPTIMIZED_KEYS | STATE_NOT_SORTED_PAGES; + } + fprintf(tracef, " rows' count %lu\n", (ulong)info->s->state.state.records); + return 0; +} + + prototype_undo_exec_hook(UNDO_ROW_INSERT) { my_bool error; MARIA_HA *info= get_MARIA_HA_from_UNDO_record(rec); + LSN previous_undo_lsn= lsn_korr(rec->header); if (info == NULL) + { + /* + Unlike for REDOs, if the table was skipped it is abnormal; we have a + transaction to rollback which used this table, as it is not rolled back + it was supposed to hold this table and so the table should still be + there. + */ return 1; - + } info->s->state.changed|= STATE_CHANGED | STATE_NOT_ANALYZED | STATE_NOT_OPTIMIZED_KEYS | STATE_NOT_SORTED_PAGES; - /* Set undo to point to previous undo record */ info->trn= trn; - info->trn->undo_lsn= lsn_korr(rec->header); - - error= _ma_apply_undo_row_insert(info, rec->lsn, + error= _ma_apply_undo_row_insert(info, previous_undo_lsn, rec->header + LSN_STORE_SIZE + FILEID_STORE_SIZE); info->trn= 0; + /* trn->undo_lsn is updated in an inwrite_hook when writing the CLR_END */ + fprintf(tracef, " rows' count %lu\n", (ulong)info->s->state.state.records); + fprintf(tracef, " undo_lsn now LSN (%lu,0x%lx)\n", + LSN_IN_HEX(previous_undo_lsn)); return error; } @@ -983,6 +1021,7 @@ prototype_undo_exec_hook(UNDO_ROW_DELETE) { my_bool error; MARIA_HA *info= get_MARIA_HA_from_UNDO_record(rec); + LSN previous_undo_lsn= lsn_korr(rec->header); if (info == NULL) return 1; @@ -1000,15 +1039,12 @@ prototype_undo_exec_hook(UNDO_ROW_DELETE) return 1; } - /* Set undo to point to previous undo record */ info->trn= trn; - info->trn->undo_lsn= lsn_korr(rec->header); - /* For now we skip the page and directory entry. This is to be used later when we mark rows as deleted. */ - error= _ma_apply_undo_row_delete(info, rec->lsn, + error= _ma_apply_undo_row_delete(info, previous_undo_lsn, log_record_buffer.str + LSN_STORE_SIZE + FILEID_STORE_SIZE + PAGE_STORE_SIZE + DIRPOS_STORE_SIZE, @@ -1016,6 +1052,9 @@ prototype_undo_exec_hook(UNDO_ROW_DELETE) (LSN_STORE_SIZE + FILEID_STORE_SIZE + PAGE_STORE_SIZE + DIRPOS_STORE_SIZE)); info->trn= 0; + fprintf(tracef, " rows' count %lu\n", (ulong)info->s->state.state.records); + fprintf(tracef, " undo_lsn now LSN (%lu,0x%lx)\n", + LSN_IN_HEX(previous_undo_lsn)); return error; } @@ -1024,6 +1063,7 @@ prototype_undo_exec_hook(UNDO_ROW_UPDATE) { my_bool error; MARIA_HA *info= get_MARIA_HA_from_UNDO_record(rec); + LSN previous_undo_lsn= lsn_korr(rec->header); if (info == NULL) return 1; @@ -1041,15 +1081,12 @@ prototype_undo_exec_hook(UNDO_ROW_UPDATE) return 1; } - /* Set undo to point to previous undo record */ info->trn= trn; - info->trn->undo_lsn= lsn_korr(rec->header); - /* For now we skip the page and directory entry. This is to be used later when we mark rows as deleted. */ - error= _ma_apply_undo_row_update(info, rec->lsn, + error= _ma_apply_undo_row_update(info, previous_undo_lsn, log_record_buffer.str + LSN_STORE_SIZE + FILEID_STORE_SIZE + PAGE_STORE_SIZE + DIRPOS_STORE_SIZE, @@ -1057,6 +1094,8 @@ prototype_undo_exec_hook(UNDO_ROW_UPDATE) (LSN_STORE_SIZE + FILEID_STORE_SIZE + PAGE_STORE_SIZE + DIRPOS_STORE_SIZE)); info->trn= 0; + fprintf(tracef, " undo_lsn now LSN (%lu,0x%lx)\n", + LSN_IN_HEX(previous_undo_lsn)); return error; } @@ -1086,6 +1125,7 @@ static int run_redo_phase(LSN lsn, my_bool apply) install_redo_exec_hook(UNDO_ROW_UPDATE); install_redo_exec_hook(UNDO_ROW_PURGE); install_redo_exec_hook(COMMIT); + install_redo_exec_hook(CLR_END); install_undo_exec_hook(UNDO_ROW_INSERT); install_undo_exec_hook(UNDO_ROW_DELETE); install_undo_exec_hook(UNDO_ROW_UPDATE); @@ -1265,6 +1305,8 @@ static uint end_of_redo_phase(my_bool prepare_for_undo_phase) if ((trn= trnman_recreate_trn_from_recovery(sid, long_trid)) == NULL) return -1; trn->undo_lsn= all_active_trans[sid].undo_lsn; + trn->first_undo_lsn= all_active_trans[sid].first_undo_lsn | + TRANSACTION_LOGGED_LONG_ID; /* because trn is known in log */ } /* otherwise we will just warn about it */ unfinished++; @@ -1335,6 +1377,7 @@ static int run_undo_phase(uint unfinished) RECHEADER_READ_ERROR) return 1; log_desc= &log_record_type_descriptor[rec.type]; + display_record_position(log_desc, &rec, 0); if (log_desc->record_execute_in_undo_phase(&rec, trn)) { fprintf(tracef, "Got error when executing undo\n"); @@ -1413,6 +1456,8 @@ static MARIA_HA *get_MARIA_HA_from_REDO_record(const record's we will modify the page */ fprintf(tracef, ", applying record\n"); + /* to flush data/index pages and state on close: */ + info->s->changed= 1; return info; } @@ -1434,6 +1479,8 @@ static MARIA_HA *get_MARIA_HA_from_UNDO_record(const fprintf(tracef, ", '%s'", info->s->open_file_name); DBUG_ASSERT(info->s->last_version != 0); fprintf(tracef, ", applying record\n"); + /* to flush data/index pages and state on close: */ + info->s->changed= 1; return info; } diff --git a/storage/maria/ma_test1.c b/storage/maria/ma_test1.c index 4435de0bbdb..6360153a171 100644 --- a/storage/maria/ma_test1.c +++ b/storage/maria/ma_test1.c @@ -15,11 +15,12 @@ /* Testing of the basic functions of a MARIA table */ -#include "maria.h" +#include "maria_def.h" #include #include #include "ma_control_file.h" #include "ma_loghandler.h" +#include "trnman.h" extern PAGECACHE *maria_log_pagecache; extern const char *maria_data_root; @@ -28,7 +29,7 @@ extern const char *maria_data_root; static void usage(); -static int rec_pointer_size=0, flags[50]; +static int rec_pointer_size=0, flags[50], testflag; static int key_field=FIELD_SKIP_PRESPACE,extra_field=FIELD_SKIP_ENDSPACE; static int key_type=HA_KEYTYPE_NUM; static int create_flag=0; @@ -223,6 +224,9 @@ static int run_test(const char *filename) if (maria_commit(file) || maria_begin(file)) goto err; + if (testflag == 1) + goto end; + /* Insert 2 rows with null values */ if (null_fields) { @@ -240,16 +244,10 @@ static int run_test(const char *filename) flags[0]=2; } - if (die_in_middle_of_transaction == 1) + if (testflag == 2) { - /* - Ensure we get changed pages and log to disk - As commit record is not done, the undo entries needs to be rolled back. - */ - _ma_flush_table_files(file, MARIA_FLUSH_DATA, FLUSH_RELEASE, - FLUSH_RELEASE); - printf("Dying on request after insert without maria_close()\n"); - exit(1); + printf("terminating after inserts\n"); + goto end; } if (!skip_update) @@ -304,6 +302,8 @@ static int run_test(const char *filename) maria_scan_end(file); } + if (testflag == 3) + goto end; if (!silent) printf("- Reopening file\n"); if (maria_commit(file)) @@ -321,6 +321,12 @@ static int run_test(const char *filename) for (i=0 ; i <= 10 ; i++) { + /* + If you want to debug the problem in ma_test_recovery with BLOBs + (see @todo there), you can break out of the loop after just one + delete, it is enough, like this: + if (i==1) break; + */ /* testing */ if (remove_count-- == 0) { @@ -355,19 +361,14 @@ static int run_test(const char *filename) } } } + } - if (die_in_middle_of_transaction == 2) - { - /* - Ensure we get changed pages and log to disk - As commit record is not done, the undo entries needs to be rolled back. - */ - _ma_flush_table_files(file, MARIA_FLUSH_DATA, FLUSH_RELEASE, - FLUSH_RELEASE); - printf("Dying on request after delete without maria_close()\n"); - exit(1); - } + if (testflag == 4) + { + printf("terminating after deletes\n"); + goto end; } + if (!silent) printf("- Reading rows with key\n"); record[1]= 0; /* For nicer printf */ @@ -412,6 +413,39 @@ static int run_test(const char *filename) i-1,error,my_errno,read_record+1); } } + +end: + if (die_in_middle_of_transaction) + { + /* As commit record is not done, UNDO entries needs to be rolled back */ + switch (die_in_middle_of_transaction) { + case 1: + /* + Flush changed pages go to disk. That will also flush log. Recovery + will skip REDOs and apply UNDOs. + */ + _ma_flush_table_files(file, MARIA_FLUSH_DATA, FLUSH_RELEASE, + FLUSH_RELEASE); + break; + case 2: + /* + Just flush log. Pages are likely to not be on disk. Recovery will + then execute REDOs and UNDOs. + */ + if (translog_flush(file->trn->undo_lsn)) + goto err; + break; + case 3: + /* + Flush nothing. Pages and log are likely to not be on disk. Recovery + will then do nothing. + */ + break; + } + printf("Dying on request without maria_commit()/maria_close()\n"); + exit(0); + } + if (maria_commit(file)) goto err; if (maria_close(file)) @@ -676,11 +710,13 @@ static struct my_option my_long_options[] = (uchar**) &skip_delete, 0, GET_BOOL, NO_ARG, 0, 0, 0, 0, 0, 0}, {"skip-update", 'D', "Don't test updates", (uchar**) &skip_update, (uchar**) &skip_update, 0, GET_BOOL, NO_ARG, 0, 0, 0, 0, 0, 0}, + {"testflag", 't', "Stop test at specified stage", (uchar**) &testflag, + (uchar**) &testflag, 0, GET_INT, REQUIRED_ARG, 0, 0, 0, 0, 0, 0}, {"test-undo", 'A', - "Abort hard after doing inserts. Used for testing recovery with undo", + "Abort hard. Used for testing recovery with undo", (uchar**) &die_in_middle_of_transaction, (uchar**) &die_in_middle_of_transaction, - 0, GET_INT, OPT_ARG, 0, 0, 0, 0, 0, 0}, + 0, GET_INT, REQUIRED_ARG, 0, 0, 0, 0, 0, 0}, {"transactional", 'T', "Test in transactional mode. (Only works with block format)", (uchar**) &transactional, (uchar**) &transactional, 0, GET_BOOL, NO_ARG, @@ -768,12 +804,6 @@ get_one_option(int optid, const struct my_option *opt __attribute__((unused)), case 'K': /* Use key cacheing */ pagecacheing=1; break; - case 'A': - if (!argument) - die_in_middle_of_transaction= 1; - else - die_in_middle_of_transaction= atoi(argument); - break; case 'V': printf("test1 Ver 1.2 \n"); exit(0); diff --git a/storage/maria/ma_test_recovery b/storage/maria/ma_test_recovery index 4e88824197e..b2d2bab7a2e 100755 --- a/storage/maria/ma_test_recovery +++ b/storage/maria/ma_test_recovery @@ -7,58 +7,201 @@ then maria_path="." fi -tmp=$maria_path/tmp +# test data is always put in the current directory or a tmp subdirectory of it +tmp="./tmp" if test '!' -d $tmp then mkdir $tmp fi -echo "MARIA RECOVERY TESTS - success is if exit code is 0" +echo "MARIA RECOVERY TESTS" +check_table_is_same() +{ + # Computes checksum of new table and compares to checksum of old table + # Shows any difference in table's state (info from the index's header) + + $maria_path/maria_chk -dvv $table | grep -v "Creation time:" > $tmp/maria_chk_message.txt 2>&1 + + # save the index file (because we want to test idempotency afterwards) + cp $table.MAI tmp/ + # In the repair below it's good to use -q because it will die on any + # incorrectness of the data file if UNDO was badly applied. + # QQ: Remove the following line when we also can recover the index file + $maria_path/maria_chk -s -rq $table + + $maria_path/maria_chk -s -e $table + checksum2=`$maria_path/maria_chk -dss $table` + if test "$checksum" != "$checksum2" + then + echo "checksum differs for $table before and after recovery" + return 1; + fi + + diff $tmp/maria_chk_message.good.txt $tmp/maria_chk_message.txt > $tmp/maria_chk_diff.txt || true + if [ -s $tmp/maria_chk_diff.txt ] + then + echo "Differences in maria_chk -dvv, recovery not yet perfect !" + echo "========DIFF START=======" + cat $tmp/maria_chk_diff.txt + echo "========DIFF END=======" + fi + mv tmp/$table.MAI . +} + +apply_log() +{ + # applies log, can verify if applying did write to log or not + + shouldchangelog=$1 + if [ "$shouldchangelog" != "shouldnotchangelog" ] && + [ "$shouldchangelog" != "shouldchangelog" ] && + [ "$shouldchangelog" != "dontknow" ] + then + echo "bad argument '$shouldchangelog'" + return 1 + fi + log_md5=`md5sum maria_log.*` + echo "applying log" + $maria_path/maria_read_log -a > $tmp/maria_read_log_$table.txt + log_md5_2=`md5sum maria_log.*` + if [ "$log_md5" != "$log_md5_2" ] + then + if [ "$shouldchangelog" == "shouldnotchangelog" ] + then + echo "maria_read_log should not have modified the log" + return 1 + fi + else + if [ "$shouldchangelog" == "shouldchangelog" ] + then + echo "maria_read_log should have modified the log" + return 1 + fi + fi +} + +# To not flood the screen, we redirect all the commands below to a text file +# and just give a final error if their output is not as expected + +( + +# this message is to remember about the problem with -b (see @todo below) +echo "!!!!!!!! REMEMBER to FIX this BLOB issue !!!!!!!" + +echo "Testing the REDO PHASE ALONE" # runs a program inserting/deleting rows, then moves the resulting table # elsewhere; applies the log and checks that the data file is # identical to the saved original. # Does not test the index file as we don't have logging for it yet. -for prog in "$maria_path/ma_test1 $silent -M -T -c" "$maria_path/ma_test2 $silent -L -K -W -P -M -T -c" "$maria_path/ma_test2 $silent -M -T -c -b" +set -- "$maria_path/ma_test1 $silent -M -T -c" "$maria_path/ma_test2 $silent -L -K -W -P -M -T -c" "$maria_path/ma_test2 $silent -M -T -c -b" +while [ $# != 0 ] do - rm -f maria_log.* maria_log_control + prog=$1 + rm maria_log.* maria_log_control echo "TEST WITH $prog" $prog # derive table's name from program's name table=`echo $prog | sed -e 's;.*ma_\(test[0-9]\).*;\1;' ` - $maria_path/maria_chk -dvv $table > $tmp/maria_chk_message.good.txt 2>&1 + $maria_path/maria_chk -dvv $table | grep -v "Creation time:"> $tmp/maria_chk_message.good.txt 2>&1 checksum=`$maria_path/maria_chk -dss $table` - mv -f $table.MAD $tmp/$table.MAD.good + mv $table.MAD $tmp/$table.MAD.good rm $table.MAI - echo "applying log" - $maria_path/maria_read_log -a > $tmp/maria_read_log_$table.txt - $maria_path/maria_chk -dvv $table > $tmp/maria_chk_message.txt 2>&1 - + apply_log "shouldnotchangelog" + cmp $table.MAD $tmp/$table.MAD.good + check_table_is_same + echo "testing idempotency" + apply_log "shouldnotchangelog" cmp $table.MAD $tmp/$table.MAD.good + check_table_is_same + shift +done - # QQ: Remove the following line when we also can recovert the index file - $maria_path/maria_chk -s -r $table - - $maria_path/maria_chk -s -e $table - checksum2=`$maria_path/maria_chk -dss $table` - if test "$checksum" != "$checksum2" - then - echo "checksum differs for $table before and after recovery" - exit 1; - fi - -# When "recovery of the table's state" is ready, we can test it like this: -# diff $tmp/maria_chk_message.good.txt $tmp/maria_chk_message.txt > $tmp/maria_chk_diff.txt || true -# if [ -s $tmp/maria_chk_diff.txt ] -# then -# echo "Differences in maria_chk -dvv, recovery not yet perfect !" -# echo "========DIFF START=======" -# cat $tmp/maria_chk_diff.txt -# echo "========DIFF END=======" -# fi - rm -f $table.* $tmp/maria_chk_*.txt $tmp/maria_read_log_$table.txt +echo "Testing the REDO AND UNDO PHASE" +# The test programs look like: +# work; commit (time T1); work; exit-without-commit (time T2) +# We first run the test program and let it exit after T1's commit. +# Then we run it again and let it exit at T2. Then we compare +# and expect identity. + +for blobs in "" "-b" # we test table without blobs and then table with blobs +do + for test_undo in 1 2 3 + do + # first iteration tests rollback of insert, second tests rollback of delete + set -- "$maria_path/ma_test1 $silent -M -T -c -N $blobs" "--testflag=1" "--testflag=2" "$maria_path/ma_test1 $silent -M -T -c -N --debug=d:t:i:o,/tmp/ma_test1.trace $blobs" "--testflag=3" "--testflag=4" + # -N (create NULL fields) is needed because --test-undo adds it anyway + while [ $# != 0 ] + do + prog=$1 + commit_run_args=$2 + abort_run_args=$3; + rm maria_log.* maria_log_control + echo "TEST WITH $prog $commit_run_args (commit at end)" + $prog $commit_run_args + # derive table's name from program's name + table=`echo $prog | sed -e 's;.*ma_\(test[0-9]\).*;\1;' ` + $maria_path/maria_chk -dvv $table | grep -v "Creation time:"> $tmp/maria_chk_message.good.txt 2>&1 + checksum=`$maria_path/maria_chk -dss $table` + mv $table.MAD $tmp/$table.MAD.good + rm $table.MAI + rm maria_log.* maria_log_control + echo "TEST WITH $prog $abort_run_args --test-undo=$test_undo (additional aborted work)" + $prog $abort_run_args --test-undo=$test_undo + cp $table.MAD $tmp/$table.MAD.before_undo + if [ $test_undo -lt 3 ] + then + apply_log "shouldchangelog" # should undo aborted work + else + # probably nothing to undo went to log or data file + apply_log "dontknow" + fi + cp $table.MAD $tmp/$table.MAD.after_undo + + # It is impossible to do a "cmp" between .good and .after_undo, + # because the UNDO phase generated log + # records whose LSN tagged pages. Another reason is that rolling back + # INSERT only marks the rows free, does not empty them (optimization), so + # traces of the INSERT+rollback remain. + + check_table_is_same + echo "testing idempotency" + apply_log "shouldnotchangelog" + cmp $table.MAD $tmp/$table.MAD.after_undo + check_table_is_same + echo "testing applying of CLRs to recreate table" + rm $table.MA? + apply_log "shouldnotchangelog" + # the cmp below fails with blobs! @todo RECOVERY BUG find out why. + # It is probably serious; REDOs shouldn't place rows in different + # positions from what the run-time code did. Indeed it may lead to + # more or less free space... + # Execution of UNDO re-inserted rows at different positions than + # originally. This generated REDOs which do not insert at the same + # positions as the execution of UNDOs, but at the same positions + # as before the row was originally deleted. + if [ "$blobs" == "" ] + then + cmp $table.MAD $tmp/$table.MAD.after_undo + fi + check_table_is_same + shift 3 + done done +done +rm -f $table.* $tmp/$table* $tmp/maria_chk_*.txt $tmp/maria_read_log_$table.txt + +) > $tmp/ma_test_recovery.output +diff $maria_path/ma_test_recovery.expected $tmp/ma_test_recovery.output > /dev/null || diff_failed=1 +if [ "$diff_failed" == "1" ] + then + echo "UNEXPECTED OUTPUT OF TESTS, FAILED" + echo "For more info, do diff $maria_path/ma_test_recovery.expected $tmp/ma_test_recovery.output" + exit 1 + fi echo "ALL RECOVERY TESTS OK" +# this message is to remember about the problem with -b (see @todo above) +echo "!!!!!!!! BUT REMEMBER to FIX this BLOB issue !!!!!!!" diff --git a/storage/maria/ma_test_recovery.expected b/storage/maria/ma_test_recovery.expected new file mode 100644 index 00000000000..0ba9ae83775 --- /dev/null +++ b/storage/maria/ma_test_recovery.expected @@ -0,0 +1,675 @@ +!!!!!!!! REMEMBER to FIX this BLOB issue !!!!!!! +Testing the REDO PHASE ALONE +TEST WITH /m/mysql-maria-tmp/storage/maria//ma_test1 -s -M -T -c +applying log +Differences in maria_chk -dvv, recovery not yet perfect ! +========DIFF START======= +7c7 +< Checksum: 3757530372 +--- +> Checksum: 0 +11c11 +< Datafile length: 16384 Keyfile length: 16384 +--- +> Datafile length: 16384 Keyfile length: 8192 +18c18 +< 1 2 6 unique number 1 8192 8192 +--- +> 1 2 6 unique number 1 8192 +========DIFF END======= +testing idempotency +applying log +Differences in maria_chk -dvv, recovery not yet perfect ! +========DIFF START======= +7,8c7,8 +< Checksum: 3757530372 +< Data records: 15 Deleted blocks: 0 +--- +> Checksum: 0 +> Data records: 30 Deleted blocks: 0 +11c11 +< Datafile length: 16384 Keyfile length: 16384 +--- +> Datafile length: 16384 Keyfile length: 8192 +18c18 +< 1 2 6 unique number 1 8192 8192 +--- +> 1 2 6 unique number 1 8192 +========DIFF END======= +TEST WITH /m/mysql-maria-tmp/storage/maria//ma_test2 -s -L -K -W -P -M -T -c +applying log +Differences in maria_chk -dvv, recovery not yet perfect ! +========DIFF START======= +11c11 +< Datafile length: 90112 Keyfile length: 212992 +--- +> Datafile length: 90112 Keyfile length: 8192 +========DIFF END======= +testing idempotency +applying log +Differences in maria_chk -dvv, recovery not yet perfect ! +========DIFF START======= +11c11 +< Datafile length: 90112 Keyfile length: 212992 +--- +> Datafile length: 90112 Keyfile length: 8192 +========DIFF END======= +TEST WITH /m/mysql-maria-tmp/storage/maria//ma_test2 -s -M -T -c -b +applying log +Differences in maria_chk -dvv, recovery not yet perfect ! +========DIFF START======= +11c11 +< Datafile length: 81920 Keyfile length: 172032 +--- +> Datafile length: 81920 Keyfile length: 8192 +========DIFF END======= +testing idempotency +applying log +Differences in maria_chk -dvv, recovery not yet perfect ! +========DIFF START======= +11c11 +< Datafile length: 81920 Keyfile length: 172032 +--- +> Datafile length: 81920 Keyfile length: 8192 +========DIFF END======= +Testing the REDO AND UNDO PHASE +TEST WITH /m/mysql-maria-tmp/storage/maria//ma_test1 -s -M -T -c -N --testflag=1 (commit at end) +TEST WITH /m/mysql-maria-tmp/storage/maria//ma_test1 -s -M -T -c -N --testflag=2 --test-undo=1 (additional aborted work) +terminating after inserts +Dying on request without maria_commit()/maria_close() +applying log +Differences in maria_chk -dvv, recovery not yet perfect ! +========DIFF START======= +6,8c6,8 +< Status: changed +< Checksum: 221293111 +< Data records: 25 Deleted blocks: 0 +--- +> Status: open,changed +> Checksum: 2134133517 +> Data records: 52 Deleted blocks: 0 +11c11 +< Datafile length: 16384 Keyfile length: 16384 +--- +> Datafile length: 16384 Keyfile length: 8192 +========DIFF END======= +testing idempotency +applying log +Differences in maria_chk -dvv, recovery not yet perfect ! +========DIFF START======= +6,8c6,8 +< Status: changed +< Checksum: 221293111 +< Data records: 25 Deleted blocks: 0 +--- +> Status: open,changed +> Checksum: 2134133517 +> Data records: 77 Deleted blocks: 0 +11c11 +< Datafile length: 16384 Keyfile length: 16384 +--- +> Datafile length: 16384 Keyfile length: 8192 +========DIFF END======= +testing applying of CLRs to recreate table +applying log +Differences in maria_chk -dvv, recovery not yet perfect ! +========DIFF START======= +7c7 +< Checksum: 221293111 +--- +> Checksum: 0 +11c11 +< Datafile length: 16384 Keyfile length: 16384 +--- +> Datafile length: 16384 Keyfile length: 8192 +18c18 +< 1 2 6 unique number NULL 0 8192 8192 +--- +> 1 2 6 unique number NULL 0 8192 +========DIFF END======= +TEST WITH /m/mysql-maria-tmp/storage/maria//ma_test1 -s -M -T -c -N --debug=d:t:i:o,/tmp/ma_test1.trace --testflag=3 (commit at end) +TEST WITH /m/mysql-maria-tmp/storage/maria//ma_test1 -s -M -T -c -N --debug=d:t:i:o,/tmp/ma_test1.trace --testflag=4 --test-undo=1 (additional aborted work) +terminating after deletes +Dying on request without maria_commit()/maria_close() +applying log +Differences in maria_chk -dvv, recovery not yet perfect ! +========DIFF START======= +6,8c6,8 +< Status: changed +< Checksum: 3536469224 +< Data records: 27 Deleted blocks: 0 +--- +> Status: open,changed +> Checksum: 2633446536 +> Data records: 43 Deleted blocks: 0 +========DIFF END======= +testing idempotency +applying log +Differences in maria_chk -dvv, recovery not yet perfect ! +========DIFF START======= +6,8c6,8 +< Status: changed +< Checksum: 3536469224 +< Data records: 27 Deleted blocks: 0 +--- +> Status: open,changed +> Checksum: 2633446536 +> Data records: 70 Deleted blocks: 0 +========DIFF END======= +testing applying of CLRs to recreate table +applying log +Differences in maria_chk -dvv, recovery not yet perfect ! +========DIFF START======= +7c7 +< Checksum: 3536469224 +--- +> Checksum: 0 +11c11 +< Datafile length: 16384 Keyfile length: 16384 +--- +> Datafile length: 16384 Keyfile length: 8192 +18c18 +< 1 2 6 unique number NULL 0 8192 8192 +--- +> 1 2 6 unique number NULL 0 8192 +========DIFF END======= +TEST WITH /m/mysql-maria-tmp/storage/maria//ma_test1 -s -M -T -c -N --testflag=1 (commit at end) +TEST WITH /m/mysql-maria-tmp/storage/maria//ma_test1 -s -M -T -c -N --testflag=2 --test-undo=2 (additional aborted work) +terminating after inserts +Dying on request without maria_commit()/maria_close() +applying log +Differences in maria_chk -dvv, recovery not yet perfect ! +========DIFF START======= +6,8c6,8 +< Status: changed +< Checksum: 221293111 +< Data records: 25 Deleted blocks: 0 +--- +> Status: open,changed +> Checksum: 2134133517 +> Data records: 52 Deleted blocks: 0 +11c11 +< Datafile length: 16384 Keyfile length: 16384 +--- +> Datafile length: 16384 Keyfile length: 8192 +========DIFF END======= +testing idempotency +applying log +Differences in maria_chk -dvv, recovery not yet perfect ! +========DIFF START======= +6,8c6,8 +< Status: changed +< Checksum: 221293111 +< Data records: 25 Deleted blocks: 0 +--- +> Status: open,changed +> Checksum: 2134133517 +> Data records: 77 Deleted blocks: 0 +11c11 +< Datafile length: 16384 Keyfile length: 16384 +--- +> Datafile length: 16384 Keyfile length: 8192 +========DIFF END======= +testing applying of CLRs to recreate table +applying log +Differences in maria_chk -dvv, recovery not yet perfect ! +========DIFF START======= +7c7 +< Checksum: 221293111 +--- +> Checksum: 0 +11c11 +< Datafile length: 16384 Keyfile length: 16384 +--- +> Datafile length: 16384 Keyfile length: 8192 +18c18 +< 1 2 6 unique number NULL 0 8192 8192 +--- +> 1 2 6 unique number NULL 0 8192 +========DIFF END======= +TEST WITH /m/mysql-maria-tmp/storage/maria//ma_test1 -s -M -T -c -N --debug=d:t:i:o,/tmp/ma_test1.trace --testflag=3 (commit at end) +TEST WITH /m/mysql-maria-tmp/storage/maria//ma_test1 -s -M -T -c -N --debug=d:t:i:o,/tmp/ma_test1.trace --testflag=4 --test-undo=2 (additional aborted work) +terminating after deletes +Dying on request without maria_commit()/maria_close() +applying log +Differences in maria_chk -dvv, recovery not yet perfect ! +========DIFF START======= +6,8c6,8 +< Status: changed +< Checksum: 3536469224 +< Data records: 27 Deleted blocks: 0 +--- +> Status: open,changed +> Checksum: 2633446536 +> Data records: 43 Deleted blocks: 0 +========DIFF END======= +testing idempotency +applying log +Differences in maria_chk -dvv, recovery not yet perfect ! +========DIFF START======= +6,8c6,8 +< Status: changed +< Checksum: 3536469224 +< Data records: 27 Deleted blocks: 0 +--- +> Status: open,changed +> Checksum: 2633446536 +> Data records: 70 Deleted blocks: 0 +========DIFF END======= +testing applying of CLRs to recreate table +applying log +Differences in maria_chk -dvv, recovery not yet perfect ! +========DIFF START======= +7c7 +< Checksum: 3536469224 +--- +> Checksum: 0 +11c11 +< Datafile length: 16384 Keyfile length: 16384 +--- +> Datafile length: 16384 Keyfile length: 8192 +18c18 +< 1 2 6 unique number NULL 0 8192 8192 +--- +> 1 2 6 unique number NULL 0 8192 +========DIFF END======= +TEST WITH /m/mysql-maria-tmp/storage/maria//ma_test1 -s -M -T -c -N --testflag=1 (commit at end) +TEST WITH /m/mysql-maria-tmp/storage/maria//ma_test1 -s -M -T -c -N --testflag=2 --test-undo=3 (additional aborted work) +terminating after inserts +Dying on request without maria_commit()/maria_close() +applying log +Differences in maria_chk -dvv, recovery not yet perfect ! +========DIFF START======= +6,8c6,8 +< Status: changed +< Checksum: 221293111 +< Data records: 25 Deleted blocks: 0 +--- +> Status: open,changed +> Checksum: 2134133517 +> Data records: 52 Deleted blocks: 0 +11c11 +< Datafile length: 16384 Keyfile length: 16384 +--- +> Datafile length: 16384 Keyfile length: 8192 +========DIFF END======= +testing idempotency +applying log +Differences in maria_chk -dvv, recovery not yet perfect ! +========DIFF START======= +6,8c6,8 +< Status: changed +< Checksum: 221293111 +< Data records: 25 Deleted blocks: 0 +--- +> Status: open,changed +> Checksum: 2134133517 +> Data records: 77 Deleted blocks: 0 +11c11 +< Datafile length: 16384 Keyfile length: 16384 +--- +> Datafile length: 16384 Keyfile length: 8192 +========DIFF END======= +testing applying of CLRs to recreate table +applying log +Differences in maria_chk -dvv, recovery not yet perfect ! +========DIFF START======= +7c7 +< Checksum: 221293111 +--- +> Checksum: 0 +11c11 +< Datafile length: 16384 Keyfile length: 16384 +--- +> Datafile length: 16384 Keyfile length: 8192 +18c18 +< 1 2 6 unique number NULL 0 8192 8192 +--- +> 1 2 6 unique number NULL 0 8192 +========DIFF END======= +TEST WITH /m/mysql-maria-tmp/storage/maria//ma_test1 -s -M -T -c -N --debug=d:t:i:o,/tmp/ma_test1.trace --testflag=3 (commit at end) +TEST WITH /m/mysql-maria-tmp/storage/maria//ma_test1 -s -M -T -c -N --debug=d:t:i:o,/tmp/ma_test1.trace --testflag=4 --test-undo=3 (additional aborted work) +terminating after deletes +Dying on request without maria_commit()/maria_close() +applying log +Differences in maria_chk -dvv, recovery not yet perfect ! +========DIFF START======= +6,8c6,8 +< Status: changed +< Checksum: 3536469224 +< Data records: 27 Deleted blocks: 0 +--- +> Status: open,changed +> Checksum: 2633446536 +> Data records: 43 Deleted blocks: 0 +========DIFF END======= +testing idempotency +applying log +Differences in maria_chk -dvv, recovery not yet perfect ! +========DIFF START======= +6,8c6,8 +< Status: changed +< Checksum: 3536469224 +< Data records: 27 Deleted blocks: 0 +--- +> Status: open,changed +> Checksum: 2633446536 +> Data records: 70 Deleted blocks: 0 +========DIFF END======= +testing applying of CLRs to recreate table +applying log +Differences in maria_chk -dvv, recovery not yet perfect ! +========DIFF START======= +7c7 +< Checksum: 3536469224 +--- +> Checksum: 0 +11c11 +< Datafile length: 16384 Keyfile length: 16384 +--- +> Datafile length: 16384 Keyfile length: 8192 +18c18 +< 1 2 6 unique number NULL 0 8192 8192 +--- +> 1 2 6 unique number NULL 0 8192 +========DIFF END======= +TEST WITH /m/mysql-maria-tmp/storage/maria//ma_test1 -s -M -T -c -N -b --testflag=1 (commit at end) +TEST WITH /m/mysql-maria-tmp/storage/maria//ma_test1 -s -M -T -c -N -b --testflag=2 --test-undo=1 (additional aborted work) +terminating after inserts +Dying on request without maria_commit()/maria_close() +applying log +Differences in maria_chk -dvv, recovery not yet perfect ! +========DIFF START======= +6,8c6,8 +< Status: changed +< Checksum: 411409161 +< Data records: 25 Deleted blocks: 0 +--- +> Status: open,changed +> Checksum: 515581293 +> Data records: 52 Deleted blocks: 0 +11c11 +< Datafile length: 49152 Keyfile length: 16384 +--- +> Datafile length: 57344 Keyfile length: 8192 +========DIFF END======= +testing idempotency +applying log +Differences in maria_chk -dvv, recovery not yet perfect ! +========DIFF START======= +6,8c6,8 +< Status: changed +< Checksum: 411409161 +< Data records: 25 Deleted blocks: 0 +--- +> Status: open,changed +> Checksum: 515581293 +> Data records: 77 Deleted blocks: 0 +11c11 +< Datafile length: 49152 Keyfile length: 16384 +--- +> Datafile length: 57344 Keyfile length: 8192 +========DIFF END======= +testing applying of CLRs to recreate table +applying log +Differences in maria_chk -dvv, recovery not yet perfect ! +========DIFF START======= +7c7 +< Checksum: 411409161 +--- +> Checksum: 0 +11c11 +< Datafile length: 49152 Keyfile length: 16384 +--- +> Datafile length: 57344 Keyfile length: 8192 +18c18 +< 1 2 6 unique varchar BLOB NULL 0 8192 8192 +--- +> 1 2 6 unique varchar BLOB NULL 0 8192 +========DIFF END======= +TEST WITH /m/mysql-maria-tmp/storage/maria//ma_test1 -s -M -T -c -N --debug=d:t:i:o,/tmp/ma_test1.trace -b --testflag=3 (commit at end) +TEST WITH /m/mysql-maria-tmp/storage/maria//ma_test1 -s -M -T -c -N --debug=d:t:i:o,/tmp/ma_test1.trace -b --testflag=4 --test-undo=1 (additional aborted work) +terminating after deletes +Dying on request without maria_commit()/maria_close() +applying log +Differences in maria_chk -dvv, recovery not yet perfect ! +========DIFF START======= +6,8c6,8 +< Status: changed +< Checksum: 1984748106 +< Data records: 27 Deleted blocks: 0 +--- +> Status: open,changed +> Checksum: 3236097623 +> Data records: 43 Deleted blocks: 0 +========DIFF END======= +testing idempotency +applying log +Differences in maria_chk -dvv, recovery not yet perfect ! +========DIFF START======= +6,8c6,8 +< Status: changed +< Checksum: 1984748106 +< Data records: 27 Deleted blocks: 0 +--- +> Status: open,changed +> Checksum: 3236097623 +> Data records: 70 Deleted blocks: 0 +========DIFF END======= +testing applying of CLRs to recreate table +applying log +Differences in maria_chk -dvv, recovery not yet perfect ! +========DIFF START======= +7c7 +< Checksum: 1984748106 +--- +> Checksum: 0 +11c11 +< Datafile length: 57344 Keyfile length: 16384 +--- +> Datafile length: 57344 Keyfile length: 8192 +18c18 +< 1 2 6 unique varchar BLOB NULL 0 8192 8192 +--- +> 1 2 6 unique varchar BLOB NULL 0 8192 +========DIFF END======= +TEST WITH /m/mysql-maria-tmp/storage/maria//ma_test1 -s -M -T -c -N -b --testflag=1 (commit at end) +TEST WITH /m/mysql-maria-tmp/storage/maria//ma_test1 -s -M -T -c -N -b --testflag=2 --test-undo=2 (additional aborted work) +terminating after inserts +Dying on request without maria_commit()/maria_close() +applying log +Differences in maria_chk -dvv, recovery not yet perfect ! +========DIFF START======= +6,8c6,8 +< Status: changed +< Checksum: 411409161 +< Data records: 25 Deleted blocks: 0 +--- +> Status: open,changed +> Checksum: 515581293 +> Data records: 52 Deleted blocks: 0 +11c11 +< Datafile length: 49152 Keyfile length: 16384 +--- +> Datafile length: 57344 Keyfile length: 8192 +========DIFF END======= +testing idempotency +applying log +Differences in maria_chk -dvv, recovery not yet perfect ! +========DIFF START======= +6,8c6,8 +< Status: changed +< Checksum: 411409161 +< Data records: 25 Deleted blocks: 0 +--- +> Status: open,changed +> Checksum: 515581293 +> Data records: 77 Deleted blocks: 0 +11c11 +< Datafile length: 49152 Keyfile length: 16384 +--- +> Datafile length: 57344 Keyfile length: 8192 +========DIFF END======= +testing applying of CLRs to recreate table +applying log +Differences in maria_chk -dvv, recovery not yet perfect ! +========DIFF START======= +7c7 +< Checksum: 411409161 +--- +> Checksum: 0 +11c11 +< Datafile length: 49152 Keyfile length: 16384 +--- +> Datafile length: 57344 Keyfile length: 8192 +18c18 +< 1 2 6 unique varchar BLOB NULL 0 8192 8192 +--- +> 1 2 6 unique varchar BLOB NULL 0 8192 +========DIFF END======= +TEST WITH /m/mysql-maria-tmp/storage/maria//ma_test1 -s -M -T -c -N --debug=d:t:i:o,/tmp/ma_test1.trace -b --testflag=3 (commit at end) +TEST WITH /m/mysql-maria-tmp/storage/maria//ma_test1 -s -M -T -c -N --debug=d:t:i:o,/tmp/ma_test1.trace -b --testflag=4 --test-undo=2 (additional aborted work) +terminating after deletes +Dying on request without maria_commit()/maria_close() +applying log +Differences in maria_chk -dvv, recovery not yet perfect ! +========DIFF START======= +6,8c6,8 +< Status: changed +< Checksum: 1984748106 +< Data records: 27 Deleted blocks: 0 +--- +> Status: open,changed +> Checksum: 3236097623 +> Data records: 43 Deleted blocks: 0 +========DIFF END======= +testing idempotency +applying log +Differences in maria_chk -dvv, recovery not yet perfect ! +========DIFF START======= +6,8c6,8 +< Status: changed +< Checksum: 1984748106 +< Data records: 27 Deleted blocks: 0 +--- +> Status: open,changed +> Checksum: 3236097623 +> Data records: 70 Deleted blocks: 0 +========DIFF END======= +testing applying of CLRs to recreate table +applying log +Differences in maria_chk -dvv, recovery not yet perfect ! +========DIFF START======= +7c7 +< Checksum: 1984748106 +--- +> Checksum: 0 +11c11 +< Datafile length: 57344 Keyfile length: 16384 +--- +> Datafile length: 57344 Keyfile length: 8192 +18c18 +< 1 2 6 unique varchar BLOB NULL 0 8192 8192 +--- +> 1 2 6 unique varchar BLOB NULL 0 8192 +========DIFF END======= +TEST WITH /m/mysql-maria-tmp/storage/maria//ma_test1 -s -M -T -c -N -b --testflag=1 (commit at end) +TEST WITH /m/mysql-maria-tmp/storage/maria//ma_test1 -s -M -T -c -N -b --testflag=2 --test-undo=3 (additional aborted work) +terminating after inserts +Dying on request without maria_commit()/maria_close() +applying log +Differences in maria_chk -dvv, recovery not yet perfect ! +========DIFF START======= +6,8c6,8 +< Status: changed +< Checksum: 411409161 +< Data records: 25 Deleted blocks: 0 +--- +> Status: open,changed +> Checksum: 515581293 +> Data records: 52 Deleted blocks: 0 +11c11 +< Datafile length: 49152 Keyfile length: 16384 +--- +> Datafile length: 49152 Keyfile length: 8192 +========DIFF END======= +testing idempotency +applying log +Differences in maria_chk -dvv, recovery not yet perfect ! +========DIFF START======= +6,8c6,8 +< Status: changed +< Checksum: 411409161 +< Data records: 25 Deleted blocks: 0 +--- +> Status: open,changed +> Checksum: 515581293 +> Data records: 77 Deleted blocks: 0 +11c11 +< Datafile length: 49152 Keyfile length: 16384 +--- +> Datafile length: 49152 Keyfile length: 8192 +========DIFF END======= +testing applying of CLRs to recreate table +applying log +Differences in maria_chk -dvv, recovery not yet perfect ! +========DIFF START======= +7c7 +< Checksum: 411409161 +--- +> Checksum: 0 +11c11 +< Datafile length: 49152 Keyfile length: 16384 +--- +> Datafile length: 49152 Keyfile length: 8192 +18c18 +< 1 2 6 unique varchar BLOB NULL 0 8192 8192 +--- +> 1 2 6 unique varchar BLOB NULL 0 8192 +========DIFF END======= +TEST WITH /m/mysql-maria-tmp/storage/maria//ma_test1 -s -M -T -c -N --debug=d:t:i:o,/tmp/ma_test1.trace -b --testflag=3 (commit at end) +TEST WITH /m/mysql-maria-tmp/storage/maria//ma_test1 -s -M -T -c -N --debug=d:t:i:o,/tmp/ma_test1.trace -b --testflag=4 --test-undo=3 (additional aborted work) +terminating after deletes +Dying on request without maria_commit()/maria_close() +applying log +Differences in maria_chk -dvv, recovery not yet perfect ! +========DIFF START======= +6,8c6,8 +< Status: changed +< Checksum: 1984748106 +< Data records: 27 Deleted blocks: 0 +--- +> Status: open,changed +> Checksum: 3236097623 +> Data records: 43 Deleted blocks: 0 +========DIFF END======= +testing idempotency +applying log +Differences in maria_chk -dvv, recovery not yet perfect ! +========DIFF START======= +6,8c6,8 +< Status: changed +< Checksum: 1984748106 +< Data records: 27 Deleted blocks: 0 +--- +> Status: open,changed +> Checksum: 3236097623 +> Data records: 70 Deleted blocks: 0 +========DIFF END======= +testing applying of CLRs to recreate table +applying log +Differences in maria_chk -dvv, recovery not yet perfect ! +========DIFF START======= +7c7 +< Checksum: 1984748106 +--- +> Checksum: 0 +11c11 +< Datafile length: 57344 Keyfile length: 16384 +--- +> Datafile length: 57344 Keyfile length: 8192 +18c18 +< 1 2 6 unique varchar BLOB NULL 0 8192 8192 +--- +> 1 2 6 unique varchar BLOB NULL 0 8192 +========DIFF END======= diff --git a/storage/maria/maria_read_log.c b/storage/maria/maria_read_log.c index 4057fd51e85..d22df34f14c 100644 --- a/storage/maria/maria_read_log.c +++ b/storage/maria/maria_read_log.c @@ -93,7 +93,8 @@ int main(int argc, char **argv) */ fprintf(stdout, "TRACE of the last maria_read_log\n"); - if (maria_apply_log(lsn, opt_display_and_apply, stdout, TRUE)) + if (maria_apply_log(lsn, opt_display_and_apply, stdout, + opt_display_and_apply)) goto err; fprintf(stdout, "%s: SUCCESS\n", my_progname); -- cgit v1.2.1 From d53991853ec660de38c97629e54f84b15d3d64c4 Mon Sep 17 00:00:00 2001 From: unknown Date: Thu, 6 Sep 2007 16:53:26 +0200 Subject: - speed optimization: minimize writes to transactional Maria tables: don't write data pages, state, and open_count at the end of each statement. Data pages will be written by a background thread periodically. State will be written by Checkpoint periodically. open_count serves to detect when a table is potentially damaged due to an unclean mysqld stop, but thanks to recovery an unclean mysqld stop will be corrected and so open_count becomes useless. As state is written less often, it is often obsolete on disk, we thus should avoid to read it from disk. - by removing the data page writes above, it is necessary to put it back at the start of some statements like check, repair and delete_all. It was already necessary in fact (see ma_delete_all.c). - disabling CACHE INDEX on Maria tables for now (fixes crash of test 'key_cache' when run with --default-storage-engine=maria). - correcting some fishy code in maria_extra.c (we possibly could lose index pages when doing a DROP TABLE under Windows, in theory). storage/maria/ha_maria.cc: disable CACHE INDEX in Maria for now (there is a single cache for now), it crashes and it's not a priority storage/maria/ma_bitmap.c: debug message storage/maria/ma_check.c: The statement before maria_repair() may not flush state, so it needs to be done by maria_repair() (indeed this function uses maria_open(HA_OPEN_COPY) so reads state from disk, so needs to find it up-to-date on disk). For safety (but normally this is not needed) we remove index blocks out of the cache before repairing. _ma_flush_blocks() becomes _ma_flush_table_files_after_repair(): it now additionally flushes the data file and state and syncs files. As a side effect, the assertion "no WRITE_CACHE_USED" from _ma_flush_table_files() fired so we move all end_io_cache() done at the end of repair to before the calls to _ma_flush_table_files_after_repair(). storage/maria/ma_close.c: when closing a transactional table, we fsync it. But we need to do this only after writing its state. We need to write the state at close time only for transactional tables (the other tables do that at last unlock). Putting back the O_RDONLY||crashed condition which I had removed earlier. Unmap the file before syncing it (does not matter now as Maria does not use mmap) storage/maria/ma_delete_all.c: need to flush data pages before chsize-ing it. Was needed even when we flushed data pages at the end of each statement, because we didn't anyway do it if under LOCK TABLES: the change here thus fixes this bug: create table t(a int) engine=maria;lock tables t write; insert into t values(1);delete from t;unlock tables;check table t; "Size of datafile is: 16384 Should be: 8192" (an obsolete page went to disk after the chsize(), at unlock time). storage/maria/ma_extra.c: When doing share->last_version=0, we make the MARIA_SHARE-in-memory invisible to future openers, so need to have an up-to-date state on disk for them. The same way, future openers will reopen the data and index file, so they will not find our cached blocks, so we need to flush them to disk. In HA_EXTRA_FORCE_REOPEN, this probably happens naturally as all tables normally get closed, we however add a safety flush. In HA_EXTRA_PREPARE_FOR_RENAME, we need to do the flushing. On Windows we additionally need to close files. In HA_EXTRA_PREPARE_FOR_DROP, we don't need to flush anything but remove dirty cached blocks from memory. On Windows we need to close files. Closing files forces us to sync them before (requirement for transactional tables). For mutex reasons (don't lock intern_lock twice), we move maria_lock_database() and _ma_decrement_open_count() first in the list of operations. Flush also data file in HA_EXTRA_FLUSH. storage/maria/ma_locking.c: For transactional tables: - don't write data pages / state at unlock time; as a consequence, "share->changed=0" cannot be done. - don't write state in _ma_writeinfo() - don't maintain open_count on disk (Recovery corrects the table in case of crash anyway, and we gain speed by not writing open_count to disk), For non-transactional tables, flush the state at unlock only if the table was changed (optimization). Code which read the state from disk is relevant only with external locking, we disable it (if want to re-enable it, it shouldn't for transactional tables as state on disk may be obsolete (such tables does not flush state at unlock anymore). The comment "We have to flush the write cache" is now wrong because maria_lock_database(F_UNLCK) now happens before thr_unlock(), and we are not using external locking. storage/maria/ma_open.c: _ma_state_info_read() is only used in ma_open.c, making it static storage/maria/ma_recovery.c: set MARIA_SHARE::changed to TRUE when we are going to apply a REDO/UNDO, so that the state gets flushed at close. storage/maria/ma_test_recovery.expected: Changes introduced by this patch: - good: the "open" (table open, not properly closed) is gone, it was pointless for a recovered table - bad: stemming from different moments of writing the index's state probably (_ma_writeinfo() used to write the state after every row write in ma_test* programs, doesn't anymore as the table is transactional): some differences in indexes (not relevant as we don't yet have recovery for them); some differences in count of records (changed from a wrong value to another wrong value) (not relevant as we don't recover this count correctly yet anyway, though a patch will be pushed soon). storage/maria/ma_test_recovery: for repeatable output, no names of varying directories. storage/maria/maria_chk.c: function renamed storage/maria/maria_def.h: Function became local to ma_open.c. Function renamed. --- storage/maria/ha_maria.cc | 3 + storage/maria/ma_bitmap.c | 3 +- storage/maria/ma_check.c | 75 ++++---- storage/maria/ma_close.c | 28 +-- storage/maria/ma_delete_all.c | 10 +- storage/maria/ma_extra.c | 122 +++++++++---- storage/maria/ma_locking.c | 98 +++++++---- storage/maria/ma_open.c | 9 +- storage/maria/ma_recovery.c | 6 +- storage/maria/ma_test_recovery | 10 +- storage/maria/ma_test_recovery.expected | 294 ++++++++++++++------------------ storage/maria/maria_chk.c | 5 +- storage/maria/maria_def.h | 4 +- 13 files changed, 369 insertions(+), 298 deletions(-) diff --git a/storage/maria/ha_maria.cc b/storage/maria/ha_maria.cc index 7613ada0919..eae2704688d 100644 --- a/storage/maria/ha_maria.cc +++ b/storage/maria/ha_maria.cc @@ -1348,6 +1348,9 @@ int ha_maria::assign_to_keycache(THD * thd, HA_CHECK_OPT *check_opt) TABLE_LIST *table_list= table->pos_in_table_list; DBUG_ENTER("ha_maria::assign_to_keycache"); + /* for now, it is disabled */ + DBUG_RETURN(HA_ADMIN_NOT_IMPLEMENTED); + table->keys_in_use_for_query.clear_all(); if (table_list->process_index_hints(table)) diff --git a/storage/maria/ma_bitmap.c b/storage/maria/ma_bitmap.c index 37c1a77aa44..684f5e16ffa 100644 --- a/storage/maria/ma_bitmap.c +++ b/storage/maria/ma_bitmap.c @@ -265,6 +265,7 @@ my_bool _ma_bitmap_end(MARIA_SHARE *share) my_bool _ma_flush_bitmap(MARIA_SHARE *share) { my_bool res= 0; + DBUG_ENTER("_ma_flush_bitmap"); if (share->bitmap.changed) { pthread_mutex_lock(&share->bitmap.bitmap_lock); @@ -275,7 +276,7 @@ my_bool _ma_flush_bitmap(MARIA_SHARE *share) } pthread_mutex_unlock(&share->bitmap.bitmap_lock); } - return res; + DBUG_RETURN(res); } diff --git a/storage/maria/ma_check.c b/storage/maria/ma_check.c index a5e64cb555c..02bce28ca7c 100644 --- a/storage/maria/ma_check.c +++ b/storage/maria/ma_check.c @@ -1996,9 +1996,12 @@ int maria_repair(HA_CHECK *param, register MARIA_HA *info, /* The physical size of the data file is sometimes used during repair (see sort_info.filelength further below); we need to flush to have it exact. + We flush the state because our maria_open(HA_OPEN_COPY) will want to read + it from disk. Index file will be recreated. */ - if (_ma_flush_table_files(info, MARIA_FLUSH_DATA, FLUSH_FORCE_WRITE, - FLUSH_KEEP)) + if (_ma_flush_table_files(info, MARIA_FLUSH_DATA | MARIA_FLUSH_INDEX, + FLUSH_FORCE_WRITE, FLUSH_IGNORE_CHANGED) || + _ma_state_info_write(share->kfile.file, &share->state, 1|2)) goto err; if (!rep_quick) @@ -2025,13 +2028,9 @@ int maria_repair(HA_CHECK *param, register MARIA_HA *info, share->state.header.org_data_file_type == BLOCK_RECORD)) { MARIA_HA *new_info; - /** - @todo RECOVERY it's a bit worrying to have two MARIA_SHARE on the - same index file: - - Checkpoint will see them as two tables - - are we sure that new_info never flushes an in-progress state - to the index file? And how to prevent Checkpoint from doing that? - - in the close future maria_close() will write the state... + /* + It's ok for Recovery to have two MARIA_SHARE on the same index file + because the one below is not transactional */ if (!(sort_info.new_info= maria_open(info->s->open_file_name, O_RDWR, HA_OPEN_COPY | HA_OPEN_FOR_REPAIR))) @@ -2264,6 +2263,11 @@ err: if (scan_inited) maria_scan_end(sort_info.info); + VOID(end_io_cache(¶m->read_cache)); + info->opt_flag&= ~(READ_CACHE_USED | WRITE_CACHE_USED); + /* this below could fail, shouldn't we detect error? */ + VOID(end_io_cache(&info->rec_cache)); + got_error|= _ma_flush_table_files_after_repair(param, info); if (got_error) { if (! param->error_printed) @@ -2298,10 +2302,6 @@ err: my_free(sort_param.rec_buff, MYF(MY_ALLOW_ZERO_PTR)); my_free(sort_param.record,MYF(MY_ALLOW_ZERO_PTR)); my_free(sort_info.buff,MYF(MY_ALLOW_ZERO_PTR)); - VOID(end_io_cache(¶m->read_cache)); - info->opt_flag&= ~(READ_CACHE_USED | WRITE_CACHE_USED); - VOID(end_io_cache(&info->rec_cache)); - got_error|=_ma_flush_blocks(param, share->pagecache, &share->kfile); if (!got_error && (param->testflag & T_UNPACK)) restore_data_file_type(share); share->state.changed|= (STATE_NOT_OPTIMIZED_KEYS | STATE_NOT_SORTED_PAGES | @@ -2443,18 +2443,31 @@ void maria_lock_memory(HA_CHECK *param __attribute__((unused))) } /* maria_lock_memory */ - /* Flush all changed blocks to disk */ +/** + Flush all changed blocks to disk so that we can say "at the end of repair, + the table is fully ok on disk". + + It is a requirement for transactional tables. + We release blocks as it's unlikely that they would all be needed soon. -int _ma_flush_blocks(HA_CHECK *param, PAGECACHE *pagecache, - PAGECACHE_FILE *file) + @param param description of the repair operation + @param info table +*/ + +int _ma_flush_table_files_after_repair(HA_CHECK *param, MARIA_HA *info) { - if (flush_pagecache_blocks(pagecache, file, FLUSH_RELEASE)) + MARIA_SHARE *share= info->s; + if (_ma_flush_table_files(info, MARIA_FLUSH_DATA | MARIA_FLUSH_INDEX, + FLUSH_RELEASE, FLUSH_RELEASE) || + _ma_state_info_write(share->kfile.file, &share->state, 1) || + (share->now_transactional && !share->temporary + && _ma_sync_table_files(info))) { _ma_check_print_error(param,"%d when trying to write bufferts",my_errno); - return(1); + return 1; } return 0; -} /* _ma_flush_blocks */ +} /* _ma_flush_table_files_after_repair */ /* Sort index for more efficent reads */ @@ -3064,8 +3077,10 @@ int maria_repair_by_sort(HA_CHECK *param, register MARIA_HA *info, memcpy( &share->state.state, info->state, sizeof(*info->state)); err: - got_error|= _ma_flush_blocks(param, share->pagecache, &share->kfile); VOID(end_io_cache(&info->rec_cache)); + VOID(end_io_cache(¶m->read_cache)); + info->opt_flag&= ~(READ_CACHE_USED | WRITE_CACHE_USED); + got_error|= _ma_flush_table_files_after_repair(param, info); if (!got_error) { /* Replace the actual file with the temporary file */ @@ -3105,8 +3120,6 @@ err: my_free((uchar*) sort_info.key_block,MYF(MY_ALLOW_ZERO_PTR)); my_free((uchar*) sort_info.ft_buf, MYF(MY_ALLOW_ZERO_PTR)); my_free(sort_info.buff,MYF(MY_ALLOW_ZERO_PTR)); - VOID(end_io_cache(¶m->read_cache)); - info->opt_flag&= ~(READ_CACHE_USED | WRITE_CACHE_USED); if (!got_error && (param->testflag & T_UNPACK)) restore_data_file_type(share); DBUG_RETURN(got_error); @@ -3581,13 +3594,14 @@ int maria_repair_parallel(HA_CHECK *param, register MARIA_HA *info, memcpy(&share->state.state, info->state, sizeof(*info->state)); err: - got_error|= _ma_flush_blocks(param, share->pagecache, &share->kfile); /* Destroy the write cache. The master thread did already detach from the share by remove_io_thread() or it was not yet started (if the error happend before creating the thread). */ VOID(end_io_cache(&info->rec_cache)); + VOID(end_io_cache(¶m->read_cache)); + info->opt_flag&= ~(READ_CACHE_USED | WRITE_CACHE_USED); /* Destroy the new data cache in case of non-quick repair. All slave threads did either detach from the share by remove_io_thread() @@ -3596,6 +3610,7 @@ err: */ if (!rep_quick) VOID(end_io_cache(&new_data_cache)); + got_error|= _ma_flush_table_files_after_repair(param, info); if (!got_error) { /* Replace the actual file with the temporary file */ @@ -3637,8 +3652,6 @@ err: my_free((uchar*) sort_info.key_block,MYF(MY_ALLOW_ZERO_PTR)); my_free((uchar*) sort_param,MYF(MY_ALLOW_ZERO_PTR)); my_free(sort_info.buff,MYF(MY_ALLOW_ZERO_PTR)); - VOID(end_io_cache(¶m->read_cache)); - info->opt_flag&= ~(READ_CACHE_USED | WRITE_CACHE_USED); if (!got_error && (param->testflag & T_UNPACK)) restore_data_file_type(share); DBUG_RETURN(got_error); @@ -5587,13 +5600,13 @@ static int write_log_record_for_repair(const HA_CHECK *param, MARIA_HA *info) translog_flush(share->state.create_rename_lsn))) return 1; /* - But this piece is really needed, to have the new table's content durable - and to not apply old REDOs to the new table. The table's existence was - made durable earlier (MY_SYNC_DIR passed to maria_change_to_newfile()). + The table's existence was made durable earlier (MY_SYNC_DIR passed to + maria_change_to_newfile()). + _ma_flush_table_files_after_repair() is later called by maria_repair(), + and makes sure to flush the data, index and state and sync, so + create_rename_lsn reaches disk, thus we won't apply old REDOs to the new + table. */ - DBUG_ASSERT(info->dfile.file >= 0); - return (_ma_update_create_rename_lsn_on_disk(share, FALSE) || - _ma_sync_table_files(info)); } return 0; } diff --git a/storage/maria/ma_close.c b/storage/maria/ma_close.c index cc9f0005a4d..508cbb6f672 100644 --- a/storage/maria/ma_close.c +++ b/storage/maria/ma_close.c @@ -75,12 +75,10 @@ int maria_close(register MARIA_HA *info) FLUSH_IGNORE_CHANGED : FLUSH_RELEASE))) error= my_errno; - /* - File must be synced as it is going out of the maria_open_list and so - becoming unknown to Checkpoint. - */ - if (share->now_transactional && my_sync(share->kfile.file, MYF(MY_WME))) - error= my_errno; +#ifdef HAVE_MMAP + if (share->file_map) + _ma_unmap_file(info); +#endif /* If we are crashed, we can safely flush the current state as it will not change the crashed state. @@ -88,15 +86,21 @@ int maria_close(register MARIA_HA *info) may be using the file at this point IF using --external-locking, which does not apply to Maria. */ - if (share->changed) - _ma_state_info_write(share->kfile.file, &share->state, 1); + if ((share->changed && share->base.born_transactional) || + (share->mode != O_RDONLY && maria_is_crashed(info))) + { + /* + File must be synced as it is going out of the maria_open_list and so + becoming unknown to Checkpoint. State must be written to file as + it was not done at table's unlocking. + */ + if (_ma_state_info_write(share->kfile.file, &share->state, 1) || + my_sync(share->kfile.file, MYF(MY_WME))) + error= my_errno; + } if (my_close(share->kfile.file, MYF(0))) error= my_errno; } -#ifdef HAVE_MMAP - if (share->file_map) - _ma_unmap_file(info); -#endif #ifdef THREAD thr_lock_delete(&share->lock); VOID(pthread_mutex_destroy(&share->intern_lock)); diff --git a/storage/maria/ma_delete_all.c b/storage/maria/ma_delete_all.c index 14afb8ea870..c46ca48d2c6 100644 --- a/storage/maria/ma_delete_all.c +++ b/storage/maria/ma_delete_all.c @@ -73,11 +73,13 @@ int maria_delete_all_rows(MARIA_HA *info) /* If we are using delayed keys or if the user has done changes to the tables - since it was locked then there may be key blocks in the key cache + since it was locked then there may be key blocks in the page cache. Or + there may be data blocks there. We need to throw them away or they may + re-enter the emptied table later. */ - flush_pagecache_blocks(share->pagecache, &share->kfile, - FLUSH_IGNORE_CHANGED); - if (my_chsize(info->dfile.file, 0, 0, MYF(MY_WME)) || + if (_ma_flush_table_files(info, MARIA_FLUSH_DATA|MARIA_FLUSH_INDEX, + FLUSH_IGNORE_CHANGED, FLUSH_IGNORE_CHANGED) || + my_chsize(info->dfile.file, 0, 0, MYF(MY_WME)) || my_chsize(share->kfile.file, share->base.keystart, 0, MYF(MY_WME)) ) goto err; diff --git a/storage/maria/ma_extra.c b/storage/maria/ma_extra.c index a3fb9569290..d1b78a11c82 100644 --- a/storage/maria/ma_extra.c +++ b/storage/maria/ma_extra.c @@ -256,60 +256,107 @@ int maria_extra(MARIA_HA *info, enum ha_extra_function function, } } share->state.state= *info->state; + /* + That state write to disk must be done, even for transactional tables; + indeed the table's share is going to be lost (there was a + HA_EXTRA_FORCE_REOPEN before, which set share->last_version to + 0), and so the only way it leaves information (share->state.key_map) + for the posterity is by writing it to disk. + */ error=_ma_state_info_write(share->kfile.file, &share->state, (1 | 2)); } break; case HA_EXTRA_FORCE_REOPEN: + /* + Normally MySQL uses this case when it is going to close all open + instances of the table, thus going to flush all data/index/state. + We however do a flush here for additional safety. + */ + /** @todo consider porting these flush-es to MyISAM */ + error= _ma_flush_table_files(info, MARIA_FLUSH_DATA | MARIA_FLUSH_INDEX, + FLUSH_FORCE_WRITE, FLUSH_FORCE_WRITE) || + _ma_state_info_write(share->kfile.file, &share->state, 1 | 2) || + (share->changed= 0); pthread_mutex_lock(&THR_LOCK_maria); + /* this makes the share not be re-used next time the table is opened */ share->last_version= 0L; /* Impossible version */ pthread_mutex_unlock(&THR_LOCK_maria); break; case HA_EXTRA_PREPARE_FOR_DROP: case HA_EXTRA_PREPARE_FOR_RENAME: + { + my_bool do_flush= test(function != HA_EXTRA_PREPARE_FOR_DROP); pthread_mutex_lock(&THR_LOCK_maria); share->last_version= 0L; /* Impossible version */ -#ifdef __WIN__ - /* Close the isam and data files as Win32 can't drop an open table */ - pthread_mutex_lock(&share->intern_lock); /* - If this is Windows we remove blocks from pagecache. If not Windows we - don't do it, so these pages stay in the pagecache? So they may later be - flushed to a wrong file? - Or is it that this flush_pagecache_blocks() never finds any blocks? Then - why do we do it on Windows? - Don't we wait for all instances to be closed before dropping the table? - Do we ever do something useful here? - BUG? - FLUSH_IGNORE_CHANGED: we are also throwing away unique index blocks? - Does ENABLE KEYS rebuild them too? + This share, having last_version=0, needs to save all its data/index + blocks to disk if this is not for a DROP TABLE. Otherwise they would be + invisible to future openers; and they could even go to disk late and + cancel the work of future openers. + On Windows, which cannot delete an open file (cannot drop an open table) + we have to close the table's files. */ - if (flush_pagecache_blocks(share->pagecache, &share->kfile, - (function == HA_EXTRA_PREPARE_FOR_DROP ? - FLUSH_IGNORE_CHANGED : FLUSH_RELEASE))) + if (info->lock_type != F_UNLCK && !info->was_locked) + { + info->was_locked= info->lock_type; + if (maria_lock_database(info, F_UNLCK)) + error= my_errno; + info->lock_type= F_UNLCK; + } + if (share->kfile.file >= 0) + _ma_decrement_open_count(info); + pthread_mutex_lock(&share->intern_lock); + enum flush_type type= do_flush ? FLUSH_RELEASE : FLUSH_IGNORE_CHANGED; + if (_ma_flush_table_files(info, MARIA_FLUSH_DATA | MARIA_FLUSH_INDEX, + type, type)) { error=my_errno; share->changed=1; - maria_print_error(info->s, HA_ERR_CRASHED); - maria_mark_crashed(info); /* Fatal error found */ } if (info->opt_flag & (READ_CACHE_USED | WRITE_CACHE_USED)) { info->opt_flag&= ~(READ_CACHE_USED | WRITE_CACHE_USED); - error=end_io_cache(&info->rec_cache); - } - if (info->lock_type != F_UNLCK && ! info->was_locked) - { - info->was_locked=info->lock_type; - if (maria_lock_database(info,F_UNLCK)) - error=my_errno; - info->lock_type = F_UNLCK; + if (end_io_cache(&info->rec_cache)) + error= 1; } if (share->kfile.file >= 0) { - _ma_decrement_open_count(info); - if (my_close(share->kfile,MYF(0))) + if (do_flush) + { + /* + Save the state so that others can find it from disk. + We have to sync now, as on Windows we are going to close the file + (so cannot sync later). + */ + if (_ma_state_info_write(share->kfile.file, &share->state, 1 | 2) || + my_sync(share->kfile.file, MYF(0))) + error= my_errno; + else + share->changed= 0; + } + else + { + /* be sure that state is not tried for write as file may be closed */ + share->changed= 0; + } +#ifdef __WIN__ + if (my_close(share->kfile, MYF(0))) error=my_errno; + share->kfile.file= -1; +#endif } + if (share->data_file_type == BLOCK_RECORD && + share->bitmap.file.file >= 0) + { + if (do_flush && my_sync(share->bitmap.file.file, MYF(0))) + error= my_errno; +#ifdef __WIN__ + if (my_close(share->bitmap.file.file, MYF(0))) + error= my_errno; + share->bitmap.file.file= -1; +#endif + } +#ifdef __WIN__ { LIST *list_element ; for (list_element=maria_open_list ; @@ -319,24 +366,23 @@ int maria_extra(MARIA_HA *info, enum ha_extra_function function, MARIA_HA *tmpinfo=(MARIA_HA*) list_element->data; if (tmpinfo->s == info->s) { - /** - @todo RECOVERY BUG: flush of bitmap and sync of dfile are missing - */ - if (tmpinfo->dfile.file >= 0 && + if (share->data_file_type != BLOCK_RECORD && + tmpinfo->dfile.file >= 0 && my_close(tmpinfo->dfile.file, MYF(0))) error = my_errno; tmpinfo->dfile.file= -1; } } } - share->kfile.file= -1; /* Files aren't open anymore */ - pthread_mutex_unlock(&share->intern_lock); #endif + pthread_mutex_unlock(&share->intern_lock); pthread_mutex_unlock(&THR_LOCK_maria); break; + } case HA_EXTRA_FLUSH: if (!share->temporary) - flush_pagecache_blocks(share->pagecache, &share->kfile, FLUSH_KEEP); + error= _ma_flush_table_files(info, MARIA_FLUSH_DATA | MARIA_FLUSH_INDEX, + FLUSH_KEEP, FLUSH_KEEP); #ifdef HAVE_PWRITE _ma_decrement_open_count(info); #endif @@ -489,8 +535,8 @@ int maria_reset(MARIA_HA *info) int _ma_sync_table_files(const MARIA_HA *info) { - return (my_sync(info->dfile.file, MYF(0)) || - my_sync(info->s->kfile.file, MYF(0))); + return (my_sync(info->dfile.file, MYF(MY_WME)) || + my_sync(info->s->kfile.file, MYF(MY_WME))); } @@ -527,6 +573,8 @@ int _ma_flush_table_files(MARIA_HA *info, uint flush_data_or_index, { if (info->opt_flag & WRITE_CACHE_USED) { + /* normally any code which creates a WRITE_CACHE destroys it later */ + DBUG_ASSERT(0); if (end_io_cache(&info->rec_cache)) goto err; info->opt_flag&= ~WRITE_CACHE_USED; diff --git a/storage/maria/ma_locking.c b/storage/maria/ma_locking.c index dad4071edf8..a69ed5f8a76 100644 --- a/storage/maria/ma_locking.c +++ b/storage/maria/ma_locking.c @@ -76,14 +76,11 @@ int maria_lock_database(MARIA_HA *info, int lock_type) /* Mark that table must be checked */ maria_mark_crashed(info); } - if (share->data_file_type == BLOCK_RECORD && - flush_pagecache_blocks(share->pagecache, &info->dfile, FLUSH_KEEP)) - { + /* pages of transactional tables get flushed at Checkpoint */ + if (!share->base.born_transactional && + _ma_flush_table_files(info, MARIA_FLUSH_DATA, + FLUSH_KEEP, FLUSH_KEEP)) error= my_errno; - maria_print_error(info->s, HA_ERR_CRASHED); - /* Mark that table must be checked */ - maria_mark_crashed(info); - } } if (info->opt_flag & (READ_CACHE_USED | WRITE_CACHE_USED)) { @@ -116,9 +113,17 @@ int maria_lock_database(MARIA_HA *info, int lock_type) share->state.process= share->last_process=share->this_process; share->state.unique= info->last_unique= info->this_unique; share->state.update_count= info->last_loop= ++info->this_loop; - if (_ma_state_info_write(share->kfile.file, &share->state, 1)) - error=my_errno; - share->changed=0; + /* transactional tables rather flush their state at Checkpoint */ + if (!share->base.born_transactional) + { + if (_ma_state_info_write(share->kfile.file, &share->state, 1)) + error= my_errno; + else + { + /* A value of 0 means below means "state flushed" */ + share->changed= 0; + } + } if (maria_flush) { if (_ma_sync_table_files(info)) @@ -135,6 +140,7 @@ int maria_lock_database(MARIA_HA *info, int lock_type) } info->opt_flag&= ~(READ_CACHE_USED | WRITE_CACHE_USED); info->lock_type= F_UNLCK; + /* verify that user of the table cleaned up after itself */ DBUG_ASSERT(share->now_transactional == share->base.born_transactional); break; case F_RDLCK: @@ -151,14 +157,17 @@ int maria_lock_database(MARIA_HA *info, int lock_type) info->lock_type=lock_type; break; } +#ifdef MARIA_EXTERNAL_LOCKING if (!share->r_locks && !share->w_locks) { + /* note that a transactional table should not do this */ if (_ma_state_info_read_dsk(share->kfile.file, &share->state)) { error=my_errno; break; } } +#endif VOID(_ma_test_if_changed(info)); share->r_locks++; share->tot_locks++; @@ -175,12 +184,29 @@ int maria_lock_database(MARIA_HA *info, int lock_type) break; } } +#ifdef MARIA_EXTERNAL_LOCKING if (!(share->options & HA_OPTION_READ_ONLY_DATA)) { if (!share->w_locks) { if (!share->r_locks) { + /* + Note that transactional tables should not do this. + If we enabled this code, we should make sure to skip it if + born_transactional is true. We should not test + now_transactional to decide if we can call + _ma_state_info_read_dsk(), because it can temporarily be 0 + (TRUNCATE on a partitioned table) and thus it would make a state + modification below without mutex, confusing a concurrent + checkpoint running. + Even if this code was enabled only for non-transactional tables: + in scenario LOCK TABLE t1 WRITE; INSERT INTO t1; DELETE FROM t1; + state on disk read by DELETE is obsolete as it was not flushed + at the end of INSERT. MyISAM same. It however causes no issue as + maria_delete_all_rows() calls _ma_reset_status() thus is not + influenced by the obsolete read values. + */ if (_ma_state_info_read_dsk(share->kfile.file, &share->state)) { error=my_errno; @@ -189,6 +215,7 @@ int maria_lock_database(MARIA_HA *info, int lock_type) } } } +#endif /* defined(MARIA_EXTERNAL_LOCKING) */ VOID(_ma_test_if_changed(info)); info->lock_type=lock_type; @@ -278,24 +305,15 @@ void _ma_update_status(void* param) (long) info->s->state.state.key_file_length, (long) info->s->state.state.data_file_length)); #endif + /* + we are going to modify the state without lock's log, this would break + recovery if done with a transactional table. + */ + DBUG_ASSERT(!info->s->base.born_transactional); info->s->state.state= *info->state; info->state= &info->s->state.state; } info->append_insert_at_end= 0; - - /* - We have to flush the write cache here as other threads may start - reading the table before maria_lock_database() is called - */ - if (info->opt_flag & WRITE_CACHE_USED) - { - if (end_io_cache(&info->rec_cache)) - { - maria_print_error(info->s, HA_ERR_CRASHED); - maria_mark_crashed(info); - } - info->opt_flag&= ~WRITE_CACHE_USED; - } } @@ -355,8 +373,11 @@ my_bool _ma_check_status(void *param) ** functions to read / write the state ****************************************************************************/ -int _ma_readinfo(register MARIA_HA *info, int lock_type, int check_keybuffer) +int _ma_readinfo(register MARIA_HA *info __attribute__ ((unused)), + int lock_type __attribute__ ((unused)), + int check_keybuffer __attribute__ ((unused))) { +#ifdef MARIA_EXTERNAL_LOCKING DBUG_ENTER("_ma_readinfo"); if (info->lock_type == F_UNLCK) @@ -364,6 +385,7 @@ int _ma_readinfo(register MARIA_HA *info, int lock_type, int check_keybuffer) MARIA_SHARE *share=info->s; if (!share->tot_locks) { + /* should not be done for transactional tables */ if (_ma_state_info_read_dsk(share->kfile.file, &share->state)) { int error=my_errno ? my_errno : -1; @@ -381,6 +403,9 @@ int _ma_readinfo(register MARIA_HA *info, int lock_type, int check_keybuffer) DBUG_RETURN(-1); /* when have read_lock() */ } DBUG_RETURN(0); +#else + return 0; +#endif /* defined(MARIA_EXTERNAL_LOCKING) */ } /* _ma_readinfo */ @@ -398,8 +423,9 @@ int _ma_writeinfo(register MARIA_HA *info, uint operation) share->tot_locks)); error=0; - if (share->tot_locks == 0) + if (share->tot_locks == 0 && !share->base.born_transactional) { + /* transactional tables flush their state at Checkpoint */ if (operation) { /* Two threads can't be here */ olderror= my_errno; /* Remember last error */ @@ -459,7 +485,7 @@ int _ma_test_if_changed(register MARIA_HA *info) state.open_count in the .MYI file is used the following way: - For the first change of the .MYI file in this process open_count is - incremented by maria_mark_file_change(). (We have a write lock on the file + incremented by _ma_mark_file_changed(). (We have a write lock on the file when this happens) - In maria_close() it's decremented by _ma_decrement_open_count() if it was incremented in the same process. @@ -467,6 +493,8 @@ int _ma_test_if_changed(register MARIA_HA *info) This mean that if we are the only process using the file, the open_count tells us if the MARIA file wasn't properly closed. (This is true if my_disable_locking is set). + + open_count is not maintained on disk for transactional or temporary tables. */ @@ -485,7 +513,12 @@ int _ma_mark_file_changed(MARIA_HA *info) share->global_changed=1; share->state.open_count++; } - if (!share->temporary) + /* + temp tables don't need an open_count as they are removed on crash; + transactional tables are fixed by log-based recovery, so don't need an + open_count either (and we thus avoid the disk write below). + */ + if (!(share->temporary | share->base.born_transactional)) { mi_int2store(buff,share->state.open_count); buff[2]=1; /* Mark that it's changed */ @@ -517,10 +550,13 @@ int _ma_decrement_open_count(MARIA_HA *info) if (share->state.open_count > 0) { share->state.open_count--; - mi_int2store(buff,share->state.open_count); - write_error= my_pwrite(share->kfile.file, buff, sizeof(buff), - sizeof(share->state.header), + if (!(share->temporary | share->base.born_transactional)) + { + mi_int2store(buff,share->state.open_count); + write_error= my_pwrite(share->kfile.file, buff, sizeof(buff), + sizeof(share->state.header), MYF(MY_NABP)); + } } if (!lock_error) lock_error=maria_lock_database(info,old_lock); diff --git a/storage/maria/ma_open.c b/storage/maria/ma_open.c index 4c623ac56f3..a8bb7b444e9 100644 --- a/storage/maria/ma_open.c +++ b/storage/maria/ma_open.c @@ -39,6 +39,7 @@ static void maria_scan_end_dummy(MARIA_HA *info); static my_bool maria_once_init_dummy(MARIA_SHARE *, File); static my_bool maria_once_end_dummy(MARIA_SHARE *); static uchar *_ma_base_info_read(uchar *ptr, MARIA_BASE_INFO *base); +static uchar *_ma_state_info_read(uchar *ptr, MARIA_STATE_INFO *state); #define get_next_element(to,pos,size) { memcpy((char*) to,pos,(size_t) size); \ pos+=size;} @@ -1049,7 +1050,7 @@ uint _ma_state_info_write(File file, MARIA_STATE_INFO *state, uint pWrite) } -uchar *_ma_state_info_read(uchar *ptr, MARIA_STATE_INFO *state) +static uchar *_ma_state_info_read(uchar *ptr, MARIA_STATE_INFO *state) { uint i,keys,key_parts; memcpy_fixed(&state->header,ptr, sizeof(state->header)); @@ -1103,7 +1104,9 @@ uchar *_ma_state_info_read(uchar *ptr, MARIA_STATE_INFO *state) /** @brief Fills the state by reading its copy on disk. - @note Does nothing in single user mode. + Should not be called for transactional tables, as their state on disk is + rarely current and so is often misleading for a reader. + Does nothing in single user mode. @param file file to read from @param state state which will be filled @@ -1114,6 +1117,8 @@ uint _ma_state_info_read_dsk(File file, MARIA_STATE_INFO *state) { char buff[MARIA_STATE_INFO_SIZE + MARIA_STATE_EXTRA_SIZE]; + /* trick to detect transactional tables */ + DBUG_ASSERT(state->create_rename_lsn == LSN_IMPOSSIBLE); if (!maria_single_user) { if (my_pread(file, buff, state->state_length, 0L, MYF(MY_NABP))) diff --git a/storage/maria/ma_recovery.c b/storage/maria/ma_recovery.c index d65202f045e..4f4681303f4 100644 --- a/storage/maria/ma_recovery.c +++ b/storage/maria/ma_recovery.c @@ -1456,8 +1456,7 @@ static MARIA_HA *get_MARIA_HA_from_REDO_record(const record's we will modify the page */ fprintf(tracef, ", applying record\n"); - /* to flush data/index pages and state on close: */ - info->s->changed= 1; + _ma_writeinfo(info, WRITEINFO_UPDATE_KEYFILE); /* to flush state on close */ return info; } @@ -1476,11 +1475,10 @@ static MARIA_HA *get_MARIA_HA_from_UNDO_record(const fprintf(tracef, ", table skipped, so skipping record\n"); return NULL; } + _ma_writeinfo(info, WRITEINFO_UPDATE_KEYFILE); /* to flush state on close */ fprintf(tracef, ", '%s'", info->s->open_file_name); DBUG_ASSERT(info->s->last_version != 0); fprintf(tracef, ", applying record\n"); - /* to flush data/index pages and state on close: */ - info->s->changed= 1; return info; } diff --git a/storage/maria/ma_test_recovery b/storage/maria/ma_test_recovery index b2d2bab7a2e..fdf54736404 100755 --- a/storage/maria/ma_test_recovery +++ b/storage/maria/ma_test_recovery @@ -96,13 +96,13 @@ echo "Testing the REDO PHASE ALONE" # identical to the saved original. # Does not test the index file as we don't have logging for it yet. -set -- "$maria_path/ma_test1 $silent -M -T -c" "$maria_path/ma_test2 $silent -L -K -W -P -M -T -c" "$maria_path/ma_test2 $silent -M -T -c -b" +set -- "ma_test1 $silent -M -T -c" "ma_test2 $silent -L -K -W -P -M -T -c" "ma_test2 $silent -M -T -c -b" while [ $# != 0 ] do prog=$1 rm maria_log.* maria_log_control echo "TEST WITH $prog" - $prog + $maria_path/$prog # derive table's name from program's name table=`echo $prog | sed -e 's;.*ma_\(test[0-9]\).*;\1;' ` $maria_path/maria_chk -dvv $table | grep -v "Creation time:"> $tmp/maria_chk_message.good.txt 2>&1 @@ -131,7 +131,7 @@ do for test_undo in 1 2 3 do # first iteration tests rollback of insert, second tests rollback of delete - set -- "$maria_path/ma_test1 $silent -M -T -c -N $blobs" "--testflag=1" "--testflag=2" "$maria_path/ma_test1 $silent -M -T -c -N --debug=d:t:i:o,/tmp/ma_test1.trace $blobs" "--testflag=3" "--testflag=4" + set -- "ma_test1 $silent -M -T -c -N $blobs" "--testflag=1" "--testflag=2" "ma_test1 $silent -M -T -c -N --debug=d:t:i:o,/tmp/ma_test1.trace $blobs" "--testflag=3" "--testflag=4" # -N (create NULL fields) is needed because --test-undo adds it anyway while [ $# != 0 ] do @@ -140,7 +140,7 @@ do abort_run_args=$3; rm maria_log.* maria_log_control echo "TEST WITH $prog $commit_run_args (commit at end)" - $prog $commit_run_args + $maria_path/$prog $commit_run_args # derive table's name from program's name table=`echo $prog | sed -e 's;.*ma_\(test[0-9]\).*;\1;' ` $maria_path/maria_chk -dvv $table | grep -v "Creation time:"> $tmp/maria_chk_message.good.txt 2>&1 @@ -149,7 +149,7 @@ do rm $table.MAI rm maria_log.* maria_log_control echo "TEST WITH $prog $abort_run_args --test-undo=$test_undo (additional aborted work)" - $prog $abort_run_args --test-undo=$test_undo + $maria_path/$prog $abort_run_args --test-undo=$test_undo cp $table.MAD $tmp/$table.MAD.before_undo if [ $test_undo -lt 3 ] then diff --git a/storage/maria/ma_test_recovery.expected b/storage/maria/ma_test_recovery.expected index 0ba9ae83775..67be4940bb1 100644 --- a/storage/maria/ma_test_recovery.expected +++ b/storage/maria/ma_test_recovery.expected @@ -1,6 +1,6 @@ !!!!!!!! REMEMBER to FIX this BLOB issue !!!!!!! Testing the REDO PHASE ALONE -TEST WITH /m/mysql-maria-tmp/storage/maria//ma_test1 -s -M -T -c +TEST WITH ma_test1 -s -M -T -c applying log Differences in maria_chk -dvv, recovery not yet perfect ! ========DIFF START======= @@ -36,7 +36,7 @@ Differences in maria_chk -dvv, recovery not yet perfect ! --- > 1 2 6 unique number 1 8192 ========DIFF END======= -TEST WITH /m/mysql-maria-tmp/storage/maria//ma_test2 -s -L -K -W -P -M -T -c +TEST WITH ma_test2 -s -L -K -W -P -M -T -c applying log Differences in maria_chk -dvv, recovery not yet perfect ! ========DIFF START======= @@ -54,7 +54,7 @@ Differences in maria_chk -dvv, recovery not yet perfect ! --- > Datafile length: 90112 Keyfile length: 8192 ========DIFF END======= -TEST WITH /m/mysql-maria-tmp/storage/maria//ma_test2 -s -M -T -c -b +TEST WITH ma_test2 -s -M -T -c -b applying log Differences in maria_chk -dvv, recovery not yet perfect ! ========DIFF START======= @@ -73,42 +73,44 @@ Differences in maria_chk -dvv, recovery not yet perfect ! > Datafile length: 81920 Keyfile length: 8192 ========DIFF END======= Testing the REDO AND UNDO PHASE -TEST WITH /m/mysql-maria-tmp/storage/maria//ma_test1 -s -M -T -c -N --testflag=1 (commit at end) -TEST WITH /m/mysql-maria-tmp/storage/maria//ma_test1 -s -M -T -c -N --testflag=2 --test-undo=1 (additional aborted work) +TEST WITH ma_test1 -s -M -T -c -N --testflag=1 (commit at end) +TEST WITH ma_test1 -s -M -T -c -N --testflag=2 --test-undo=1 (additional aborted work) terminating after inserts Dying on request without maria_commit()/maria_close() applying log Differences in maria_chk -dvv, recovery not yet perfect ! ========DIFF START======= -6,8c6,8 -< Status: changed +7c7 < Checksum: 221293111 -< Data records: 25 Deleted blocks: 0 --- -> Status: open,changed -> Checksum: 2134133517 -> Data records: 52 Deleted blocks: 0 +> Checksum: 0 11c11 < Datafile length: 16384 Keyfile length: 16384 --- > Datafile length: 16384 Keyfile length: 8192 +18c18 +< 1 2 6 unique number NULL 0 8192 8192 +--- +> 1 2 6 unique number NULL 0 8192 ========DIFF END======= testing idempotency applying log Differences in maria_chk -dvv, recovery not yet perfect ! ========DIFF START======= -6,8c6,8 -< Status: changed +7,8c7,8 < Checksum: 221293111 < Data records: 25 Deleted blocks: 0 --- -> Status: open,changed -> Checksum: 2134133517 -> Data records: 77 Deleted blocks: 0 +> Checksum: 0 +> Data records: 50 Deleted blocks: 0 11c11 < Datafile length: 16384 Keyfile length: 16384 --- > Datafile length: 16384 Keyfile length: 8192 +18c18 +< 1 2 6 unique number NULL 0 8192 8192 +--- +> 1 2 6 unique number NULL 0 8192 ========DIFF END======= testing applying of CLRs to recreate table applying log @@ -127,34 +129,26 @@ Differences in maria_chk -dvv, recovery not yet perfect ! --- > 1 2 6 unique number NULL 0 8192 ========DIFF END======= -TEST WITH /m/mysql-maria-tmp/storage/maria//ma_test1 -s -M -T -c -N --debug=d:t:i:o,/tmp/ma_test1.trace --testflag=3 (commit at end) -TEST WITH /m/mysql-maria-tmp/storage/maria//ma_test1 -s -M -T -c -N --debug=d:t:i:o,/tmp/ma_test1.trace --testflag=4 --test-undo=1 (additional aborted work) +TEST WITH ma_test1 -s -M -T -c -N --debug=d:t:i:o,/tmp/ma_test1.trace --testflag=3 (commit at end) +TEST WITH ma_test1 -s -M -T -c -N --debug=d:t:i:o,/tmp/ma_test1.trace --testflag=4 --test-undo=1 (additional aborted work) terminating after deletes Dying on request without maria_commit()/maria_close() applying log Differences in maria_chk -dvv, recovery not yet perfect ! ========DIFF START======= -6,8c6,8 -< Status: changed -< Checksum: 3536469224 +8c8 < Data records: 27 Deleted blocks: 0 --- -> Status: open,changed -> Checksum: 2633446536 -> Data records: 43 Deleted blocks: 0 +> Data records: 54 Deleted blocks: 0 ========DIFF END======= testing idempotency applying log Differences in maria_chk -dvv, recovery not yet perfect ! ========DIFF START======= -6,8c6,8 -< Status: changed -< Checksum: 3536469224 +8c8 < Data records: 27 Deleted blocks: 0 --- -> Status: open,changed -> Checksum: 2633446536 -> Data records: 70 Deleted blocks: 0 +> Data records: 81 Deleted blocks: 0 ========DIFF END======= testing applying of CLRs to recreate table applying log @@ -173,42 +167,44 @@ Differences in maria_chk -dvv, recovery not yet perfect ! --- > 1 2 6 unique number NULL 0 8192 ========DIFF END======= -TEST WITH /m/mysql-maria-tmp/storage/maria//ma_test1 -s -M -T -c -N --testflag=1 (commit at end) -TEST WITH /m/mysql-maria-tmp/storage/maria//ma_test1 -s -M -T -c -N --testflag=2 --test-undo=2 (additional aborted work) +TEST WITH ma_test1 -s -M -T -c -N --testflag=1 (commit at end) +TEST WITH ma_test1 -s -M -T -c -N --testflag=2 --test-undo=2 (additional aborted work) terminating after inserts Dying on request without maria_commit()/maria_close() applying log Differences in maria_chk -dvv, recovery not yet perfect ! ========DIFF START======= -6,8c6,8 -< Status: changed +7c7 < Checksum: 221293111 -< Data records: 25 Deleted blocks: 0 --- -> Status: open,changed -> Checksum: 2134133517 -> Data records: 52 Deleted blocks: 0 +> Checksum: 0 11c11 < Datafile length: 16384 Keyfile length: 16384 --- > Datafile length: 16384 Keyfile length: 8192 +18c18 +< 1 2 6 unique number NULL 0 8192 8192 +--- +> 1 2 6 unique number NULL 0 8192 ========DIFF END======= testing idempotency applying log Differences in maria_chk -dvv, recovery not yet perfect ! ========DIFF START======= -6,8c6,8 -< Status: changed +7,8c7,8 < Checksum: 221293111 < Data records: 25 Deleted blocks: 0 --- -> Status: open,changed -> Checksum: 2134133517 -> Data records: 77 Deleted blocks: 0 +> Checksum: 0 +> Data records: 50 Deleted blocks: 0 11c11 < Datafile length: 16384 Keyfile length: 16384 --- > Datafile length: 16384 Keyfile length: 8192 +18c18 +< 1 2 6 unique number NULL 0 8192 8192 +--- +> 1 2 6 unique number NULL 0 8192 ========DIFF END======= testing applying of CLRs to recreate table applying log @@ -227,34 +223,26 @@ Differences in maria_chk -dvv, recovery not yet perfect ! --- > 1 2 6 unique number NULL 0 8192 ========DIFF END======= -TEST WITH /m/mysql-maria-tmp/storage/maria//ma_test1 -s -M -T -c -N --debug=d:t:i:o,/tmp/ma_test1.trace --testflag=3 (commit at end) -TEST WITH /m/mysql-maria-tmp/storage/maria//ma_test1 -s -M -T -c -N --debug=d:t:i:o,/tmp/ma_test1.trace --testflag=4 --test-undo=2 (additional aborted work) +TEST WITH ma_test1 -s -M -T -c -N --debug=d:t:i:o,/tmp/ma_test1.trace --testflag=3 (commit at end) +TEST WITH ma_test1 -s -M -T -c -N --debug=d:t:i:o,/tmp/ma_test1.trace --testflag=4 --test-undo=2 (additional aborted work) terminating after deletes Dying on request without maria_commit()/maria_close() applying log Differences in maria_chk -dvv, recovery not yet perfect ! ========DIFF START======= -6,8c6,8 -< Status: changed -< Checksum: 3536469224 +8c8 < Data records: 27 Deleted blocks: 0 --- -> Status: open,changed -> Checksum: 2633446536 -> Data records: 43 Deleted blocks: 0 +> Data records: 54 Deleted blocks: 0 ========DIFF END======= testing idempotency applying log Differences in maria_chk -dvv, recovery not yet perfect ! ========DIFF START======= -6,8c6,8 -< Status: changed -< Checksum: 3536469224 +8c8 < Data records: 27 Deleted blocks: 0 --- -> Status: open,changed -> Checksum: 2633446536 -> Data records: 70 Deleted blocks: 0 +> Data records: 81 Deleted blocks: 0 ========DIFF END======= testing applying of CLRs to recreate table applying log @@ -273,42 +261,44 @@ Differences in maria_chk -dvv, recovery not yet perfect ! --- > 1 2 6 unique number NULL 0 8192 ========DIFF END======= -TEST WITH /m/mysql-maria-tmp/storage/maria//ma_test1 -s -M -T -c -N --testflag=1 (commit at end) -TEST WITH /m/mysql-maria-tmp/storage/maria//ma_test1 -s -M -T -c -N --testflag=2 --test-undo=3 (additional aborted work) +TEST WITH ma_test1 -s -M -T -c -N --testflag=1 (commit at end) +TEST WITH ma_test1 -s -M -T -c -N --testflag=2 --test-undo=3 (additional aborted work) terminating after inserts Dying on request without maria_commit()/maria_close() applying log Differences in maria_chk -dvv, recovery not yet perfect ! ========DIFF START======= -6,8c6,8 -< Status: changed +7c7 < Checksum: 221293111 -< Data records: 25 Deleted blocks: 0 --- -> Status: open,changed -> Checksum: 2134133517 -> Data records: 52 Deleted blocks: 0 +> Checksum: 0 11c11 < Datafile length: 16384 Keyfile length: 16384 --- > Datafile length: 16384 Keyfile length: 8192 +18c18 +< 1 2 6 unique number NULL 0 8192 8192 +--- +> 1 2 6 unique number NULL 0 8192 ========DIFF END======= testing idempotency applying log Differences in maria_chk -dvv, recovery not yet perfect ! ========DIFF START======= -6,8c6,8 -< Status: changed +7,8c7,8 < Checksum: 221293111 < Data records: 25 Deleted blocks: 0 --- -> Status: open,changed -> Checksum: 2134133517 -> Data records: 77 Deleted blocks: 0 +> Checksum: 0 +> Data records: 50 Deleted blocks: 0 11c11 < Datafile length: 16384 Keyfile length: 16384 --- > Datafile length: 16384 Keyfile length: 8192 +18c18 +< 1 2 6 unique number NULL 0 8192 8192 +--- +> 1 2 6 unique number NULL 0 8192 ========DIFF END======= testing applying of CLRs to recreate table applying log @@ -327,34 +317,26 @@ Differences in maria_chk -dvv, recovery not yet perfect ! --- > 1 2 6 unique number NULL 0 8192 ========DIFF END======= -TEST WITH /m/mysql-maria-tmp/storage/maria//ma_test1 -s -M -T -c -N --debug=d:t:i:o,/tmp/ma_test1.trace --testflag=3 (commit at end) -TEST WITH /m/mysql-maria-tmp/storage/maria//ma_test1 -s -M -T -c -N --debug=d:t:i:o,/tmp/ma_test1.trace --testflag=4 --test-undo=3 (additional aborted work) +TEST WITH ma_test1 -s -M -T -c -N --debug=d:t:i:o,/tmp/ma_test1.trace --testflag=3 (commit at end) +TEST WITH ma_test1 -s -M -T -c -N --debug=d:t:i:o,/tmp/ma_test1.trace --testflag=4 --test-undo=3 (additional aborted work) terminating after deletes Dying on request without maria_commit()/maria_close() applying log Differences in maria_chk -dvv, recovery not yet perfect ! ========DIFF START======= -6,8c6,8 -< Status: changed -< Checksum: 3536469224 +8c8 < Data records: 27 Deleted blocks: 0 --- -> Status: open,changed -> Checksum: 2633446536 -> Data records: 43 Deleted blocks: 0 +> Data records: 54 Deleted blocks: 0 ========DIFF END======= testing idempotency applying log Differences in maria_chk -dvv, recovery not yet perfect ! ========DIFF START======= -6,8c6,8 -< Status: changed -< Checksum: 3536469224 +8c8 < Data records: 27 Deleted blocks: 0 --- -> Status: open,changed -> Checksum: 2633446536 -> Data records: 70 Deleted blocks: 0 +> Data records: 81 Deleted blocks: 0 ========DIFF END======= testing applying of CLRs to recreate table applying log @@ -373,42 +355,44 @@ Differences in maria_chk -dvv, recovery not yet perfect ! --- > 1 2 6 unique number NULL 0 8192 ========DIFF END======= -TEST WITH /m/mysql-maria-tmp/storage/maria//ma_test1 -s -M -T -c -N -b --testflag=1 (commit at end) -TEST WITH /m/mysql-maria-tmp/storage/maria//ma_test1 -s -M -T -c -N -b --testflag=2 --test-undo=1 (additional aborted work) +TEST WITH ma_test1 -s -M -T -c -N -b --testflag=1 (commit at end) +TEST WITH ma_test1 -s -M -T -c -N -b --testflag=2 --test-undo=1 (additional aborted work) terminating after inserts Dying on request without maria_commit()/maria_close() applying log Differences in maria_chk -dvv, recovery not yet perfect ! ========DIFF START======= -6,8c6,8 -< Status: changed +7c7 < Checksum: 411409161 -< Data records: 25 Deleted blocks: 0 --- -> Status: open,changed -> Checksum: 515581293 -> Data records: 52 Deleted blocks: 0 +> Checksum: 0 11c11 < Datafile length: 49152 Keyfile length: 16384 --- > Datafile length: 57344 Keyfile length: 8192 +18c18 +< 1 2 6 unique varchar BLOB NULL 0 8192 8192 +--- +> 1 2 6 unique varchar BLOB NULL 0 8192 ========DIFF END======= testing idempotency applying log Differences in maria_chk -dvv, recovery not yet perfect ! ========DIFF START======= -6,8c6,8 -< Status: changed +7,8c7,8 < Checksum: 411409161 < Data records: 25 Deleted blocks: 0 --- -> Status: open,changed -> Checksum: 515581293 -> Data records: 77 Deleted blocks: 0 +> Checksum: 0 +> Data records: 50 Deleted blocks: 0 11c11 < Datafile length: 49152 Keyfile length: 16384 --- > Datafile length: 57344 Keyfile length: 8192 +18c18 +< 1 2 6 unique varchar BLOB NULL 0 8192 8192 +--- +> 1 2 6 unique varchar BLOB NULL 0 8192 ========DIFF END======= testing applying of CLRs to recreate table applying log @@ -427,34 +411,26 @@ Differences in maria_chk -dvv, recovery not yet perfect ! --- > 1 2 6 unique varchar BLOB NULL 0 8192 ========DIFF END======= -TEST WITH /m/mysql-maria-tmp/storage/maria//ma_test1 -s -M -T -c -N --debug=d:t:i:o,/tmp/ma_test1.trace -b --testflag=3 (commit at end) -TEST WITH /m/mysql-maria-tmp/storage/maria//ma_test1 -s -M -T -c -N --debug=d:t:i:o,/tmp/ma_test1.trace -b --testflag=4 --test-undo=1 (additional aborted work) +TEST WITH ma_test1 -s -M -T -c -N --debug=d:t:i:o,/tmp/ma_test1.trace -b --testflag=3 (commit at end) +TEST WITH ma_test1 -s -M -T -c -N --debug=d:t:i:o,/tmp/ma_test1.trace -b --testflag=4 --test-undo=1 (additional aborted work) terminating after deletes Dying on request without maria_commit()/maria_close() applying log Differences in maria_chk -dvv, recovery not yet perfect ! ========DIFF START======= -6,8c6,8 -< Status: changed -< Checksum: 1984748106 +8c8 < Data records: 27 Deleted blocks: 0 --- -> Status: open,changed -> Checksum: 3236097623 -> Data records: 43 Deleted blocks: 0 +> Data records: 54 Deleted blocks: 0 ========DIFF END======= testing idempotency applying log Differences in maria_chk -dvv, recovery not yet perfect ! ========DIFF START======= -6,8c6,8 -< Status: changed -< Checksum: 1984748106 +8c8 < Data records: 27 Deleted blocks: 0 --- -> Status: open,changed -> Checksum: 3236097623 -> Data records: 70 Deleted blocks: 0 +> Data records: 81 Deleted blocks: 0 ========DIFF END======= testing applying of CLRs to recreate table applying log @@ -473,42 +449,44 @@ Differences in maria_chk -dvv, recovery not yet perfect ! --- > 1 2 6 unique varchar BLOB NULL 0 8192 ========DIFF END======= -TEST WITH /m/mysql-maria-tmp/storage/maria//ma_test1 -s -M -T -c -N -b --testflag=1 (commit at end) -TEST WITH /m/mysql-maria-tmp/storage/maria//ma_test1 -s -M -T -c -N -b --testflag=2 --test-undo=2 (additional aborted work) +TEST WITH ma_test1 -s -M -T -c -N -b --testflag=1 (commit at end) +TEST WITH ma_test1 -s -M -T -c -N -b --testflag=2 --test-undo=2 (additional aborted work) terminating after inserts Dying on request without maria_commit()/maria_close() applying log Differences in maria_chk -dvv, recovery not yet perfect ! ========DIFF START======= -6,8c6,8 -< Status: changed +7c7 < Checksum: 411409161 -< Data records: 25 Deleted blocks: 0 --- -> Status: open,changed -> Checksum: 515581293 -> Data records: 52 Deleted blocks: 0 +> Checksum: 0 11c11 < Datafile length: 49152 Keyfile length: 16384 --- > Datafile length: 57344 Keyfile length: 8192 +18c18 +< 1 2 6 unique varchar BLOB NULL 0 8192 8192 +--- +> 1 2 6 unique varchar BLOB NULL 0 8192 ========DIFF END======= testing idempotency applying log Differences in maria_chk -dvv, recovery not yet perfect ! ========DIFF START======= -6,8c6,8 -< Status: changed +7,8c7,8 < Checksum: 411409161 < Data records: 25 Deleted blocks: 0 --- -> Status: open,changed -> Checksum: 515581293 -> Data records: 77 Deleted blocks: 0 +> Checksum: 0 +> Data records: 50 Deleted blocks: 0 11c11 < Datafile length: 49152 Keyfile length: 16384 --- > Datafile length: 57344 Keyfile length: 8192 +18c18 +< 1 2 6 unique varchar BLOB NULL 0 8192 8192 +--- +> 1 2 6 unique varchar BLOB NULL 0 8192 ========DIFF END======= testing applying of CLRs to recreate table applying log @@ -527,34 +505,26 @@ Differences in maria_chk -dvv, recovery not yet perfect ! --- > 1 2 6 unique varchar BLOB NULL 0 8192 ========DIFF END======= -TEST WITH /m/mysql-maria-tmp/storage/maria//ma_test1 -s -M -T -c -N --debug=d:t:i:o,/tmp/ma_test1.trace -b --testflag=3 (commit at end) -TEST WITH /m/mysql-maria-tmp/storage/maria//ma_test1 -s -M -T -c -N --debug=d:t:i:o,/tmp/ma_test1.trace -b --testflag=4 --test-undo=2 (additional aborted work) +TEST WITH ma_test1 -s -M -T -c -N --debug=d:t:i:o,/tmp/ma_test1.trace -b --testflag=3 (commit at end) +TEST WITH ma_test1 -s -M -T -c -N --debug=d:t:i:o,/tmp/ma_test1.trace -b --testflag=4 --test-undo=2 (additional aborted work) terminating after deletes Dying on request without maria_commit()/maria_close() applying log Differences in maria_chk -dvv, recovery not yet perfect ! ========DIFF START======= -6,8c6,8 -< Status: changed -< Checksum: 1984748106 +8c8 < Data records: 27 Deleted blocks: 0 --- -> Status: open,changed -> Checksum: 3236097623 -> Data records: 43 Deleted blocks: 0 +> Data records: 54 Deleted blocks: 0 ========DIFF END======= testing idempotency applying log Differences in maria_chk -dvv, recovery not yet perfect ! ========DIFF START======= -6,8c6,8 -< Status: changed -< Checksum: 1984748106 +8c8 < Data records: 27 Deleted blocks: 0 --- -> Status: open,changed -> Checksum: 3236097623 -> Data records: 70 Deleted blocks: 0 +> Data records: 81 Deleted blocks: 0 ========DIFF END======= testing applying of CLRs to recreate table applying log @@ -573,42 +543,44 @@ Differences in maria_chk -dvv, recovery not yet perfect ! --- > 1 2 6 unique varchar BLOB NULL 0 8192 ========DIFF END======= -TEST WITH /m/mysql-maria-tmp/storage/maria//ma_test1 -s -M -T -c -N -b --testflag=1 (commit at end) -TEST WITH /m/mysql-maria-tmp/storage/maria//ma_test1 -s -M -T -c -N -b --testflag=2 --test-undo=3 (additional aborted work) +TEST WITH ma_test1 -s -M -T -c -N -b --testflag=1 (commit at end) +TEST WITH ma_test1 -s -M -T -c -N -b --testflag=2 --test-undo=3 (additional aborted work) terminating after inserts Dying on request without maria_commit()/maria_close() applying log Differences in maria_chk -dvv, recovery not yet perfect ! ========DIFF START======= -6,8c6,8 -< Status: changed +7c7 < Checksum: 411409161 -< Data records: 25 Deleted blocks: 0 --- -> Status: open,changed -> Checksum: 515581293 -> Data records: 52 Deleted blocks: 0 +> Checksum: 0 11c11 < Datafile length: 49152 Keyfile length: 16384 --- > Datafile length: 49152 Keyfile length: 8192 +18c18 +< 1 2 6 unique varchar BLOB NULL 0 8192 8192 +--- +> 1 2 6 unique varchar BLOB NULL 0 8192 ========DIFF END======= testing idempotency applying log Differences in maria_chk -dvv, recovery not yet perfect ! ========DIFF START======= -6,8c6,8 -< Status: changed +7,8c7,8 < Checksum: 411409161 < Data records: 25 Deleted blocks: 0 --- -> Status: open,changed -> Checksum: 515581293 -> Data records: 77 Deleted blocks: 0 +> Checksum: 0 +> Data records: 50 Deleted blocks: 0 11c11 < Datafile length: 49152 Keyfile length: 16384 --- > Datafile length: 49152 Keyfile length: 8192 +18c18 +< 1 2 6 unique varchar BLOB NULL 0 8192 8192 +--- +> 1 2 6 unique varchar BLOB NULL 0 8192 ========DIFF END======= testing applying of CLRs to recreate table applying log @@ -627,34 +599,26 @@ Differences in maria_chk -dvv, recovery not yet perfect ! --- > 1 2 6 unique varchar BLOB NULL 0 8192 ========DIFF END======= -TEST WITH /m/mysql-maria-tmp/storage/maria//ma_test1 -s -M -T -c -N --debug=d:t:i:o,/tmp/ma_test1.trace -b --testflag=3 (commit at end) -TEST WITH /m/mysql-maria-tmp/storage/maria//ma_test1 -s -M -T -c -N --debug=d:t:i:o,/tmp/ma_test1.trace -b --testflag=4 --test-undo=3 (additional aborted work) +TEST WITH ma_test1 -s -M -T -c -N --debug=d:t:i:o,/tmp/ma_test1.trace -b --testflag=3 (commit at end) +TEST WITH ma_test1 -s -M -T -c -N --debug=d:t:i:o,/tmp/ma_test1.trace -b --testflag=4 --test-undo=3 (additional aborted work) terminating after deletes Dying on request without maria_commit()/maria_close() applying log Differences in maria_chk -dvv, recovery not yet perfect ! ========DIFF START======= -6,8c6,8 -< Status: changed -< Checksum: 1984748106 +8c8 < Data records: 27 Deleted blocks: 0 --- -> Status: open,changed -> Checksum: 3236097623 -> Data records: 43 Deleted blocks: 0 +> Data records: 54 Deleted blocks: 0 ========DIFF END======= testing idempotency applying log Differences in maria_chk -dvv, recovery not yet perfect ! ========DIFF START======= -6,8c6,8 -< Status: changed -< Checksum: 1984748106 +8c8 < Data records: 27 Deleted blocks: 0 --- -> Status: open,changed -> Checksum: 3236097623 -> Data records: 70 Deleted blocks: 0 +> Data records: 81 Deleted blocks: 0 ========DIFF END======= testing applying of CLRs to recreate table applying log diff --git a/storage/maria/maria_chk.c b/storage/maria/maria_chk.c index bea9e487314..9f51b0caf64 100644 --- a/storage/maria/maria_chk.c +++ b/storage/maria/maria_chk.c @@ -1142,7 +1142,7 @@ static int maria_chk(HA_CHECK *param, char *filename) if ((info->s->data_file_type != STATIC_RECORD) || (param->testflag & (T_EXTEND | T_MEDIUM))) error|=maria_chk_data_link(param, info, param->testflag & T_EXTEND); - error|=_ma_flush_blocks(param, share->pagecache, &share->kfile); + error|= _ma_flush_table_files_after_repair(param, info); VOID(end_io_cache(¶m->read_cache)); } if (!error) @@ -1658,8 +1658,7 @@ err: my_free(sort_info.buff,MYF(MY_ALLOW_ZERO_PTR)); sort_info.buff=0; share->state.sortkey=sort_key; - DBUG_RETURN(_ma_flush_blocks(param, share->pagecache, &share->kfile) | - got_error); + DBUG_RETURN(_ma_flush_table_files_after_repair(param, info) | got_error); } /* sort_records */ diff --git a/storage/maria/maria_def.h b/storage/maria/maria_def.h index e8b53757a53..368764d7a17 100644 --- a/storage/maria/maria_def.h +++ b/storage/maria/maria_def.h @@ -865,7 +865,6 @@ extern uint _ma_nommap_pwrite(MARIA_HA *info, uchar *Buffer, uint Count, my_off_t offset, myf MyFlags); uint _ma_state_info_write(File file, MARIA_STATE_INFO *state, uint pWrite); -uchar *_ma_state_info_read(uchar *ptr, MARIA_STATE_INFO *state); uint _ma_state_info_read_dsk(File file, MARIA_STATE_INFO *state); uint _ma_base_info_write(File file, MARIA_BASE_INFO *base); int _ma_keyseg_write(File file, const HA_KEYSEG *keyseg); @@ -927,8 +926,7 @@ int _ma_thr_write_keys(MARIA_SORT_PARAM *sort_param); #ifdef THREAD pthread_handler_t _ma_thr_find_all_keys(void *arg); #endif -int _ma_flush_blocks(HA_CHECK *param, PAGECACHE *pagecache, - PAGECACHE_FILE *file); +int _ma_flush_table_files_after_repair(HA_CHECK *param, MARIA_HA *info); int _ma_sort_write_record(MARIA_SORT_PARAM *sort_param); int _ma_create_index_by_sort(MARIA_SORT_PARAM *info, my_bool no_messages, -- cgit v1.2.1 From 2291f932b2f26af3329ce48e08eccaaa5a41aef3 Mon Sep 17 00:00:00 2001 From: unknown Date: Fri, 7 Sep 2007 15:02:30 +0200 Subject: - WL#3072 Maria Recovery: Recovery of state.records (the count of records which is stored into the header of the index file). For that, state.is_of_lsn is introduced; logic is explained in ma_recovery.c (look for "Recovery of the state"). The net gain is that in case of crash, we now recover state.records, and it is idempotent (ma_test_recovery tests it). state.checksum is not recovered yet, mail sent for discussion. - WL#3071 Maria Checkpoint: preparation for it, by protecting all modifications of the state in memory or on disk with intern_lock (with the exception of the really-often-modified state.records, which is now protected with the log's lock, see ma_recovery.c (look for "Recovery of the state"). Also, if maria_close() sees that Checkpoint is looking at this table it will not my_free() the share. - don't compute row's checksum twice in case of UPDATE (correction to a bugfix I made yesterday). storage/maria/ha_maria.cc: protect state write with intern_lock (against Checkpoint) storage/maria/ma_blockrec.c: * don't reset trn->rec_lsn in _ma_unpin_all_pages(), because it should wait until we have corrected the allocation in the bitmap (as the REDO can serve to correct the allocation during Recovery); introducing _ma_finalize_row() for that. * In a changeset yesterday I moved computation of the checksum into write_block_record(), to fix a bug in UPDATE. Now I notice that maria_update() already computes the checksum, it's just that it puts it into info->cur_row while _ma_update_block_record() uses info->new_row; so, removing the checksum computation from write_block_record(), putting it back into allocate_and_write_block_record() (which is called only by INSERT and UNDO_DELETE), and copying cur_row->checksum into new_row->checksum in _ma_update_block_record(). storage/maria/ma_check.c: new prototypes, they will take intern_lock when writing the state; also take intern_lock when changing share->kfile. In both cases this is to protect against Checkpoint reading/writing the state or reading kfile at the same time. Not updating create_rename_lsn directly at end of write_log_record_for_repair() as it wouldn't have intern_lock. storage/maria/ma_close.c: Checkpoint builds a list of shares (under THR_LOCK_maria), then it handles each such share (under intern_lock) (doing flushing etc); if maria_close() freed this share between the two, Checkpoint would see a bad pointer. To avoid this, when building the list Checkpoint marks each share, so that maria_close() knows it should not free it and Checkpoint will free it itself. Extending the zone covered by intern_lock to protect against Checkpoint reading kfile, writing state. storage/maria/ma_create.c: When we update create_rename_lsn, we also update is_of_lsn to the same value: it is logical, and allows us to test in maria_open() that the former is not bigger than the latter (the contrary is a sign of index header corruption, or severe logging bug which hinders Recovery, table needs a repair). _ma_update_create_rename_lsn_on_disk() also writes is_of_lsn; it now operates under intern_lock (protect against Checkpoint), a shortcut function is available for cases where acquiring intern_lock is not needed (table's creation or first open). storage/maria/ma_delete.c: if table is transactional, "records" is already decremented when logging UNDO_ROW_DELETE. storage/maria/ma_delete_all.c: comments storage/maria/ma_extra.c: Protect modifications of the state, in memory and/or on disk, with intern_lock, against a concurrent Checkpoint. When state goes to disk, update it's is_of_lsn (by calling the new _ma_state_info_write()). In HA_EXTRA_FORCE_REOPEN, don't set share->changed to 0 (undoing a change I made a few days ago) and ASK_MONTY storage/maria/ma_locking.c: no real code change here. storage/maria/ma_loghandler.c: Log-write-hooks for updating "state.records" under log's mutex when writing/updating/deleting a row or deleting all rows. storage/maria/ma_loghandler_lsn.h: merge (make LSN_ERROR and LSN_REPAIRED_BY_MARIA_CHK different) storage/maria/ma_open.c: When opening a table verify that is_of_lsn >= create_rename_lsn; if false the header must be corrupted. _ma_state_info_write() is split in two: _ma_state_info_write_sub() which is the old _ma_state_info_write(), and _ma_state_info_write() which additionally takes intern_lock if requested (to protect against Checkpoint) and updates is_of_lsn. _ma_open_keyfile() should change kfile.file under intern_lock to protect Checkpoint from reading a wrong kfile.file. storage/maria/ma_recovery.c: Recovery of state.records: when the REDO phase sees UNDO_ROW_INSERT which has a LSN > state.is_of_lsn it increments state.records. Same for UNDO_ROW_DELETE and UNDO_ROW_PURGE. When closing a table during Recovery, we know its state is at least as new as the current log record we are looking at, so increase is_of_lsn to the LSN of the current log record. storage/maria/ma_rename.c: update for new behaviour of _ma_update_create_rename_lsn_on_disk(). storage/maria/ma_test1.c: update to new prototype storage/maria/ma_test2.c: update to new prototype (actually prototype was changed days ago, but compiler does not complain about the extra argument??) storage/maria/ma_test_recovery.expected: new result file of ma_test_recovery. Improvements: record count read from index's header is now always correct. storage/maria/ma_test_recovery: "rm" fails if file does not exist. Redirect stderr of script. storage/maria/ma_write.c: if table is transactional, "records" is already incremented when logging UNDO_ROW_INSERT. Comments. storage/maria/maria_chk.c: update is_of_lsn too storage/maria/maria_def.h: - MARIA_STATE_INFO::is_of_lsn which is used by Recovery. It is stored into the index file's header. - Checkpoint can now mark a table as "don't free this", and maria_close() can reply "ok then you will free it". - new functions storage/maria/maria_pack.c: update for new name --- storage/maria/ha_maria.cc | 2 + storage/maria/ma_blockrec.c | 40 ++++++--- storage/maria/ma_check.c | 28 ++++--- storage/maria/ma_close.c | 43 +++++++--- storage/maria/ma_create.c | 57 ++++++++++--- storage/maria/ma_delete.c | 2 +- storage/maria/ma_delete_all.c | 8 ++ storage/maria/ma_extra.c | 29 ++++++- storage/maria/ma_locking.c | 18 ++-- storage/maria/ma_loghandler.c | 106 +++++++++++++++++++++++- storage/maria/ma_loghandler_lsn.h | 2 +- storage/maria/ma_open.c | 89 ++++++++++++++++---- storage/maria/ma_recovery.c | 141 ++++++++++++++++++++++++++++++-- storage/maria/ma_rename.c | 8 +- storage/maria/ma_test1.c | 2 +- storage/maria/ma_test2.c | 2 +- storage/maria/ma_test_recovery | 6 +- storage/maria/ma_test_recovery.expected | 116 ++------------------------ storage/maria/ma_write.c | 10 ++- storage/maria/maria_chk.c | 3 +- storage/maria/maria_def.h | 14 +++- storage/maria/maria_pack.c | 5 +- 22 files changed, 516 insertions(+), 215 deletions(-) diff --git a/storage/maria/ha_maria.cc b/storage/maria/ha_maria.cc index eae2704688d..78d2bdc1da5 100644 --- a/storage/maria/ha_maria.cc +++ b/storage/maria/ha_maria.cc @@ -1287,6 +1287,7 @@ int ha_maria::repair(THD *thd, HA_CHECK ¶m, bool do_optimize) } } thd->proc_info= "Saving state"; + pthread_mutex_lock(&share->intern_lock); if (!error) { if ((share->state.changed & STATE_CHANGED) || maria_is_crashed(file)) @@ -1324,6 +1325,7 @@ int ha_maria::repair(THD *thd, HA_CHECK ¶m, bool do_optimize) file->update |= HA_STATE_CHANGED | HA_STATE_ROW_CHANGED; maria_update_state_info(¶m, file, 0); } + pthread_mutex_unlock(&share->intern_lock); thd->proc_info= old_proc_info; if (!thd->locked_tables) { diff --git a/storage/maria/ma_blockrec.c b/storage/maria/ma_blockrec.c index 66cf58efffa..31f2c30e058 100644 --- a/storage/maria/ma_blockrec.c +++ b/storage/maria/ma_blockrec.c @@ -690,8 +690,6 @@ static my_bool check_if_zero(uchar *pos, uint length) We unpin pages in the reverse order as they where pinned; This may not be strictly necessary but may simplify things in the future. - info->trn->rec_lsn contains the lsn for the first REDO - RETURN 0 ok 1 error (fatal disk error) @@ -717,7 +715,6 @@ void _ma_unpin_all_pages(MARIA_HA *info, LSN undo_lsn) pinned_page->unlock, PAGECACHE_UNPIN, info->trn->rec_lsn, undo_lsn); - info->trn->rec_lsn= 0; info->pinned_pages.elements= 0; DBUG_VOID_RETURN; } @@ -739,6 +736,22 @@ static uint empty_space_on_page(uchar *buff, uint block_size) } #endif +/** + When we have finished the write/update/delete of a row, we have cleanups to + do. For now it is signalling to Checkpoint that all dirtied pages have + their rec_lsn set and page LSN set (_ma_unpin_all_pages() has been called), + and that bitmap pages are correct (_ma_bitmap_release_unused() has been + called). +*/ +#define _ma_finalize_row(info) \ + do { info->trn->rec_lsn= LSN_IMPOSSIBLE; } while(0) +/** unpinning is often the last operation before finalizing: */ +#define _ma_unpin_all_pages_and_finalize_row(info,undo_lsn) do \ + { \ + _ma_unpin_all_pages(info, undo_lsn); \ + _ma_finalize_row(info); \ + } while(0) + /* Find free position in directory @@ -1729,10 +1742,7 @@ static my_bool write_block_record(MARIA_HA *info, if (share->base.pack_fields) store_key_length_inc(data, row->field_lengths_length); if (share->calc_checksum) - { - row->checksum= (info->s->calc_checksum)(info, record); *(data++)= (uchar) (row->checksum); /* store least significant byte */ - } memcpy(data, record, share->base.null_bytes); data+= share->base.null_bytes; memcpy(data, row->empty_bits, share->base.pack_bytes); @@ -2387,6 +2397,8 @@ static my_bool write_block_record(MARIA_HA *info, /* Release not used space in used pages */ if (_ma_bitmap_release_unused(info, bitmap_blocks)) goto disk_err; + + _ma_finalize_row(info); DBUG_RETURN(0); crashed: @@ -2421,7 +2433,7 @@ disk_err: Unpin all pinned pages to not cause problems for disk cache. This is safe to call even if we already called _ma_unpin_all_pages() above. */ - _ma_unpin_all_pages(info, 0); + _ma_unpin_all_pages_and_finalize_row(info, 0); DBUG_RETURN(1); } @@ -2458,6 +2470,8 @@ static my_bool allocate_and_write_block_record(MARIA_HA *info, PAGECACHE_LOCK_WRITE, &row_pos)) DBUG_RETURN(1); row->lastpos= ma_recordpos(blocks->block->page, row_pos.rownr); + if (info->s->calc_checksum) + row->checksum= (info->s->calc_checksum)(info, record); if (write_block_record(info, (uchar*) 0, record, row, blocks, blocks->block->org_bitmap_value != 0, &row_pos, undo_lsn)) @@ -2595,7 +2609,7 @@ my_bool _ma_write_abort_block_record(MARIA_HA *info) log_data + LSN_STORE_SIZE)) res= 1; } - _ma_unpin_all_pages(info, info->trn->undo_lsn); + _ma_unpin_all_pages_and_finalize_row(info, info->trn->undo_lsn); DBUG_RETURN(res); } @@ -2625,6 +2639,8 @@ my_bool _ma_update_block_record(MARIA_HA *info, MARIA_RECORD_POS record_pos, DBUG_ENTER("_ma_update_block_record"); DBUG_PRINT("enter", ("rowid: %lu", (long) record_pos)); + /* checksum was computed by maria_update() already and put into cur_row */ + new_row->checksum= cur_row->checksum; calc_record_size(info, record, new_row); page= ma_recordpos_to_page(record_pos); @@ -2713,7 +2729,7 @@ my_bool _ma_update_block_record(MARIA_HA *info, MARIA_RECORD_POS record_pos, &row_pos, LSN_ERROR)); err: - _ma_unpin_all_pages(info, 0); + _ma_unpin_all_pages_and_finalize_row(info, 0); DBUG_RETURN(1); } @@ -3001,11 +3017,11 @@ my_bool _ma_delete_block_record(MARIA_HA *info, const uchar *record) } - _ma_unpin_all_pages(info, info->trn->undo_lsn); + _ma_unpin_all_pages_and_finalize_row(info, info->trn->undo_lsn); DBUG_RETURN(0); err: - _ma_unpin_all_pages(info, 0); + _ma_unpin_all_pages_and_finalize_row(info, 0); DBUG_RETURN(1); } @@ -4878,7 +4894,7 @@ my_bool _ma_apply_undo_row_insert(MARIA_HA *info, LSN undo_lsn, res= 0; err: - _ma_unpin_all_pages(info, lsn); + _ma_unpin_all_pages_and_finalize_row(info, lsn); DBUG_RETURN(res); } diff --git a/storage/maria/ma_check.c b/storage/maria/ma_check.c index 02bce28ca7c..52390de9690 100644 --- a/storage/maria/ma_check.c +++ b/storage/maria/ma_check.c @@ -2001,7 +2001,7 @@ int maria_repair(HA_CHECK *param, register MARIA_HA *info, */ if (_ma_flush_table_files(info, MARIA_FLUSH_DATA | MARIA_FLUSH_INDEX, FLUSH_FORCE_WRITE, FLUSH_IGNORE_CHANGED) || - _ma_state_info_write(share->kfile.file, &share->state, 1|2)) + _ma_state_info_write(share, 1|2|4)) goto err; if (!rep_quick) @@ -2459,9 +2459,8 @@ int _ma_flush_table_files_after_repair(HA_CHECK *param, MARIA_HA *info) MARIA_SHARE *share= info->s; if (_ma_flush_table_files(info, MARIA_FLUSH_DATA | MARIA_FLUSH_INDEX, FLUSH_RELEASE, FLUSH_RELEASE) || - _ma_state_info_write(share->kfile.file, &share->state, 1) || - (share->now_transactional && !share->temporary - && _ma_sync_table_files(info))) + _ma_state_info_write(share, 1|4) || + (share->base.born_transactional && _ma_sync_table_files(info))) { _ma_check_print_error(param,"%d when trying to write bufferts",my_errno); return 1; @@ -2540,8 +2539,10 @@ int maria_sort_index(HA_CHECK *param, register MARIA_HA *info, char *name) /* Put same locks as old file */ share->r_locks= share->w_locks= share->tot_locks= 0; (void) _ma_writeinfo(info,WRITEINFO_UPDATE_KEYFILE); + pthread_mutex_lock(&share->intern_lock); VOID(my_close(share->kfile.file, MYF(MY_WME))); share->kfile.file = -1; + pthread_mutex_unlock(&share->intern_lock); VOID(my_close(new_file,MYF(MY_WME))); if (maria_change_to_newfile(share->index_file_name, MARIA_NAME_IEXT, INDEX_TMP_EXT, sync_dir) || @@ -5087,7 +5088,7 @@ int maria_update_state_info(HA_CHECK *param, MARIA_HA *info,uint update) */ if (info->lock_type == F_WRLCK) share->state.state= *info->state; - if (_ma_state_info_write(share->kfile.file, &share->state, 1 + 2)) + if (_ma_state_info_write(share, 1|2)) goto err; share->changed=0; } @@ -5540,6 +5541,7 @@ read_next_page: /** @brief Writes a LOGREC_REPAIR_TABLE record and updates create_rename_lsn + and is_of_lsn REPAIR/OPTIMIZE have replaced the data/index file with a new file and so, in this scenario: @@ -5560,8 +5562,8 @@ read_next_page: static int write_log_record_for_repair(const HA_CHECK *param, MARIA_HA *info) { - MARIA_SHARE *share= info->s; - if (translog_inited) /* test it in case this is maria_chk */ + /* in case this is maria_chk or recovery... */ + if (translog_inited && !maria_in_recovery) { /* For now this record is only informative. It could serve when applying @@ -5582,6 +5584,7 @@ static int write_log_record_for_repair(const HA_CHECK *param, MARIA_HA *info) */ LEX_STRING log_array[TRANSLOG_INTERNAL_PARTS + 1]; uchar log_data[LSN_STORE_SIZE]; + LSN lsn; compile_time_assert(LSN_STORE_SIZE >= (FILEID_STORE_SIZE + 4)); log_array[TRANSLOG_INTERNAL_PARTS + 0].str= (char*) log_data; log_array[TRANSLOG_INTERNAL_PARTS + 0].length= FILEID_STORE_SIZE + 4; @@ -5590,22 +5593,21 @@ static int write_log_record_for_repair(const HA_CHECK *param, MARIA_HA *info) or not: did it touch the data file or not?). */ int4store(log_data + FILEID_STORE_SIZE, param->testflag); - if (unlikely(translog_write_record(&share->state.create_rename_lsn, - LOGREC_REDO_REPAIR_TABLE, + if (unlikely(translog_write_record(&lsn, LOGREC_REDO_REPAIR_TABLE, &dummy_transaction_object, info, log_array[TRANSLOG_INTERNAL_PARTS + 0].length, sizeof(log_array)/sizeof(log_array[0]), log_array, log_data) || - translog_flush(share->state.create_rename_lsn))) + translog_flush(lsn))) return 1; /* The table's existence was made durable earlier (MY_SYNC_DIR passed to maria_change_to_newfile()). _ma_flush_table_files_after_repair() is later called by maria_repair(), - and makes sure to flush the data, index and state and sync, so - create_rename_lsn reaches disk, thus we won't apply old REDOs to the new - table. + and makes sure to flush the data, index, update is_of_lsn, flush state + and sync, so create_rename_lsn reaches disk, thus we won't apply old + REDOs to the new table. */ } return 0; diff --git a/storage/maria/ma_close.c b/storage/maria/ma_close.c index 508cbb6f672..9b654803945 100644 --- a/storage/maria/ma_close.c +++ b/storage/maria/ma_close.c @@ -25,6 +25,7 @@ int maria_close(register MARIA_HA *info) { int error=0,flag; + my_bool share_can_be_freed= FALSE; MARIA_SHARE *share=info->s; DBUG_ENTER("maria_close"); DBUG_PRINT("enter",("base: 0x%lx reopen: %u locks: %u", @@ -58,7 +59,6 @@ int maria_close(register MARIA_HA *info) } flag= !--share->reopen; maria_open_list=list_delete(maria_open_list,&info->open_list); - pthread_mutex_unlock(&share->intern_lock); my_free(info->rec_buff, MYF(MY_ALLOW_ZERO_PTR)); (*share->end)(info); @@ -90,20 +90,23 @@ int maria_close(register MARIA_HA *info) (share->mode != O_RDONLY && maria_is_crashed(info))) { /* - File must be synced as it is going out of the maria_open_list and so - becoming unknown to Checkpoint. State must be written to file as - it was not done at table's unlocking. + State must be written to file as it was not done at table's + unlocking. */ - if (_ma_state_info_write(share->kfile.file, &share->state, 1) || - my_sync(share->kfile.file, MYF(MY_WME))) + if (_ma_state_info_write(share, 1)) error= my_errno; } + /* + File must be synced as it is going out of the maria_open_list and so + becoming unknown to future Checkpoints. + */ + if (my_sync(share->kfile.file, MYF(MY_WME))) + error= my_errno; if (my_close(share->kfile.file, MYF(0))) error= my_errno; } #ifdef THREAD thr_lock_delete(&share->lock); - VOID(pthread_mutex_destroy(&share->intern_lock)); { int i,keys; keys = share->state.header.keys; @@ -114,16 +117,36 @@ int maria_close(register MARIA_HA *info) } #endif DBUG_ASSERT(share->now_transactional == share->base.born_transactional); - my_free((uchar*) share, MYF(0)); + if (share->in_checkpoint == MARIA_CHECKPOINT_LOOKS_AT_ME) + { + share->kfile.file= -1; /* because Checkpoint does not need to flush */ + /* we cannot my_free() the share, Checkpoint would see a bad pointer */ + share->in_checkpoint|= MARIA_CHECKPOINT_SHOULD_FREE_ME; + } + else + share_can_be_freed= TRUE; } pthread_mutex_unlock(&THR_LOCK_maria); + pthread_mutex_unlock(&share->intern_lock); + if (share_can_be_freed) + { + VOID(pthread_mutex_destroy(&share->intern_lock)); + my_free((uchar *)share, MYF(0)); + } if (info->ftparser_param) { my_free((uchar*)info->ftparser_param, MYF(0)); info->ftparser_param= 0; } - if (info->dfile.file >= 0 && my_close(info->dfile.file, MYF(0))) - error = my_errno; + if (info->dfile.file >= 0) + { + /* + This is outside of mutex so would confuse a concurrent + Checkpoint. Fortunately in BLOCK_RECORD we close earlier under mutex. + */ + if (my_close(info->dfile.file, MYF(0))) + error = my_errno; + } my_free((uchar*) info,MYF(0)); diff --git a/storage/maria/ma_create.c b/storage/maria/ma_create.c index ba66bdb8ffb..7f26a7777c0 100644 --- a/storage/maria/ma_create.c +++ b/storage/maria/ma_create.c @@ -634,7 +634,7 @@ int maria_create(const char *name, enum data_file_type datafile_type, share.state.dellink = HA_OFFSET_ERROR; share.state.first_bitmap_with_space= 0; - share.state.create_rename_lsn= LSN_IMPOSSIBLE; + share.state.create_rename_lsn= share.state.is_of_lsn= LSN_IMPOSSIBLE; share.state.process= (ulong) getpid(); share.state.unique= (ulong) 0; share.state.update_count=(ulong) 0; @@ -792,7 +792,7 @@ int maria_create(const char *name, enum data_file_type datafile_type, errpos=1; DBUG_PRINT("info", ("write state info and base info")); - if (_ma_state_info_write(file, &share.state, 2) || + if (_ma_state_info_write_sub(file, &share.state, 2) || _ma_base_info_write(file, &share.base)) goto err; DBUG_PRINT("info", ("base_pos: %d base_info_size: %d", @@ -933,6 +933,7 @@ int maria_create(const char *name, enum data_file_type datafile_type, LEX_STRING log_array[TRANSLOG_INTERNAL_PARTS + 4]; uint total_rec_length= 0; uint i; + LSN lsn; log_array[TRANSLOG_INTERNAL_PARTS + 1].length= 1 + 2 + 2 + kfile_size_before_extension; /* we are needing maybe 64 kB, so don't use the stack */ @@ -991,20 +992,20 @@ int maria_create(const char *name, enum data_file_type datafile_type, called external_lock(), so have no TRN. It does not matter, as all these operations are non-transactional and sync their files. */ - if (unlikely(translog_write_record(&share.state.create_rename_lsn, + if (unlikely(translog_write_record(&lsn, LOGREC_REDO_CREATE_TABLE, &dummy_transaction_object, NULL, total_rec_length, sizeof(log_array)/sizeof(log_array[0]), log_array, NULL) || - translog_flush(share.state.create_rename_lsn))) + translog_flush(lsn))) goto err; /* store LSN into file, needed for Recovery to not be confused if a DROP+CREATE happened (applying REDOs to the wrong table). */ share.kfile.file= file; - if (_ma_update_create_rename_lsn_on_disk(&share, FALSE)) + if (_ma_update_create_rename_lsn_on_disk_sub(&share, lsn, FALSE)) goto err; my_free(log_data, MYF(0)); } @@ -1205,13 +1206,14 @@ int _ma_initialize_data_file(MARIA_SHARE *share, File dfile) /** - @brief Writes create_rename_lsn to disk, optionally forces + @brief Writes create_rename_lsn and is_of_lsn to disk, optionally forces. This is for special cases where: - we don't want to write the full state to disk (so, not call _ma_state_info_write()) because some parts of the state may be currently inconsistent, or because it would be overkill - - we must sync this LSN immediately for correctness. + - we must sync these LSNs immediately for correctness. + It acquires intern_lock to protect the two LSNs and state write. @param share table's share @param do_sync if the write should be forced to disk @@ -1221,13 +1223,42 @@ int _ma_initialize_data_file(MARIA_SHARE *share, File dfile) @retval 1 error (disk problem) */ -int _ma_update_create_rename_lsn_on_disk(MARIA_SHARE *share, my_bool do_sync) +int _ma_update_create_rename_lsn_on_disk(MARIA_SHARE *share, + LSN lsn, my_bool do_sync) { - char buf[LSN_STORE_SIZE]; + int res; + pthread_mutex_lock(&share->intern_lock); + res= _ma_update_create_rename_lsn_on_disk_sub(share, lsn, do_sync); + pthread_mutex_unlock(&share->intern_lock); + return res; +} + + +/** + @brief Writes create_rename_lsn and is_of_lsn to disk, optionally forces. + + Shortcut of _ma_update_create_rename_lsn_on_disk() when we know that + intern_lock is not needed (when creating a table or opening it for the + first time). + + @param share table's share + @param do_sync if the write should be forced to disk + + @return Operation status + @retval 0 ok + @retval 1 error (disk problem) +*/ + +int _ma_update_create_rename_lsn_on_disk_sub(MARIA_SHARE *share, + LSN lsn, my_bool do_sync) +{ + char buf[LSN_STORE_SIZE*2], *ptr; File file= share->kfile.file; DBUG_ASSERT(file >= 0); - lsn_store(buf, share->state.create_rename_lsn); - return (my_pwrite(file, buf, sizeof(buf), - sizeof(share->state.header) + 2, MYF(MY_NABP)) || - (do_sync && my_sync(file, MYF(0)))); + for (ptr= buf; ptr < (buf + sizeof(buf)); ptr+= LSN_STORE_SIZE) + lsn_store(ptr, lsn); + share->state.is_of_lsn= share->state.create_rename_lsn= lsn; + return my_pwrite(file, buf, sizeof(buf), + sizeof(share->state.header) + 2, MYF(MY_NABP)) || + (do_sync && my_sync(file, MYF(0))); } diff --git a/storage/maria/ma_delete.c b/storage/maria/ma_delete.c index 8dafd1c4f17..56da6fd3ed3 100644 --- a/storage/maria/ma_delete.c +++ b/storage/maria/ma_delete.c @@ -103,7 +103,7 @@ int maria_delete(MARIA_HA *info,const uchar *record) } info->update= HA_STATE_CHANGED+HA_STATE_DELETED+HA_STATE_ROW_CHANGED; - info->state->records--; + info->state->records-= !share->now_transactional; share->state.changed|= STATE_NOT_OPTIMIZED_ROWS; mi_sizestore(lastpos, info->cur_row.lastpos); diff --git a/storage/maria/ma_delete_all.c b/storage/maria/ma_delete_all.c index c46ca48d2c6..8cb4fdb8a3e 100644 --- a/storage/maria/ma_delete_all.c +++ b/storage/maria/ma_delete_all.c @@ -69,6 +69,10 @@ int maria_delete_all_rows(MARIA_HA *info) goto err; } + /* + For recovery it matters that this is called after writing the log record, + so that resetting state.records actually happens under log's mutex. + */ _ma_reset_status(info); /* @@ -143,6 +147,10 @@ void _ma_reset_status(MARIA_HA *info) info->state->key_file_length= share->base.keystart; info->state->data_file_length= 0; info->state->empty= info->state->key_empty= 0; + /** + @todo RECOVERY BUG + the line below must happen under log's mutex when writing the REDO + */ info->state->checksum= 0; /* Drop the delete key chain. */ diff --git a/storage/maria/ma_extra.c b/storage/maria/ma_extra.c index d1b78a11c82..f72a92c7506 100644 --- a/storage/maria/ma_extra.c +++ b/storage/maria/ma_extra.c @@ -227,8 +227,11 @@ int maria_extra(MARIA_HA *info, enum ha_extra_function function, info->lock_wait=MY_DONT_WAIT; break; case HA_EXTRA_NO_KEYS: + /* we're going to modify pieces of the state, stall Checkpoint */ + pthread_mutex_lock(&share->intern_lock); if (info->lock_type == F_UNLCK) { + pthread_mutex_unlock(&share->intern_lock); error=1; /* Not possibly if not lock */ break; } @@ -263,8 +266,10 @@ int maria_extra(MARIA_HA *info, enum ha_extra_function function, 0), and so the only way it leaves information (share->state.key_map) for the posterity is by writing it to disk. */ - error=_ma_state_info_write(share->kfile.file, &share->state, (1 | 2)); + DBUG_ASSERT(!maria_in_recovery); + error= _ma_state_info_write(share, 1|2); } + pthread_mutex_unlock(&share->intern_lock); break; case HA_EXTRA_FORCE_REOPEN: /* @@ -275,8 +280,22 @@ int maria_extra(MARIA_HA *info, enum ha_extra_function function, /** @todo consider porting these flush-es to MyISAM */ error= _ma_flush_table_files(info, MARIA_FLUSH_DATA | MARIA_FLUSH_INDEX, FLUSH_FORCE_WRITE, FLUSH_FORCE_WRITE) || - _ma_state_info_write(share->kfile.file, &share->state, 1 | 2) || - (share->changed= 0); + _ma_state_info_write(share, 1|2|4); +#ifdef ASK_MONTY + || (share->changed= 0); +#endif + /** + @todo RECOVERY BUG + Though we flushed the state, IF some other thread may have the same + table (same MARIA_SHARE) open at this time then it may have a + more recent state to flush when it closes, thus we don't set + share->changed to 0 here. On the other hand, this means that when our + thread closes its table, it will flush the state again, then it would + overwrite any state written by yet another thread which may have opened + the table (new MARIA_SHARE) and done some updates. + ASK_MONTY about the IF above. See also same tag in + HA_EXTRA_PREPARE_FOR_DROP|RENAME. + */ pthread_mutex_lock(&THR_LOCK_maria); /* this makes the share not be re-used next time the table is opened */ share->last_version= 0L; /* Impossible version */ @@ -328,11 +347,13 @@ int maria_extra(MARIA_HA *info, enum ha_extra_function function, We have to sync now, as on Windows we are going to close the file (so cannot sync later). */ - if (_ma_state_info_write(share->kfile.file, &share->state, 1 | 2) || + if (_ma_state_info_write(share, 1 | 2) || my_sync(share->kfile.file, MYF(0))) error= my_errno; +#ifdef ASK_MONTY /* see same tag in HA_EXTRA_FORCE_REOPEN */ else share->changed= 0; +#endif } else { diff --git a/storage/maria/ma_locking.c b/storage/maria/ma_locking.c index a69ed5f8a76..70ae7fe4202 100644 --- a/storage/maria/ma_locking.c +++ b/storage/maria/ma_locking.c @@ -116,7 +116,7 @@ int maria_lock_database(MARIA_HA *info, int lock_type) /* transactional tables rather flush their state at Checkpoint */ if (!share->base.born_transactional) { - if (_ma_state_info_write(share->kfile.file, &share->state, 1)) + if (_ma_state_info_write_sub(share->kfile.file, &share->state, 1)) error= my_errno; else { @@ -287,6 +287,7 @@ void _ma_get_status(void* param, int concurrent_insert) void _ma_update_status(void* param) { MARIA_HA *info=(MARIA_HA*) param; + MARIA_SHARE *share= info->s; /* Because someone may have closed the table we point at, we only update the state if its our own state. This isn't a problem as @@ -299,19 +300,19 @@ void _ma_update_status(void* param) DBUG_PRINT("info",("updating status: key_file: %ld data_file: %ld", (long) info->state->key_file_length, (long) info->state->data_file_length)); - if (info->state->key_file_length < info->s->state.state.key_file_length || - info->state->data_file_length < info->s->state.state.data_file_length) + if (info->state->key_file_length < share->state.state.key_file_length || + info->state->data_file_length < share->state.state.data_file_length) DBUG_PRINT("warning",("old info: key_file: %ld data_file: %ld", - (long) info->s->state.state.key_file_length, - (long) info->s->state.state.data_file_length)); + (long) share->state.state.key_file_length, + (long) share->state.state.data_file_length)); #endif /* we are going to modify the state without lock's log, this would break recovery if done with a transactional table. */ DBUG_ASSERT(!info->s->base.born_transactional); - info->s->state.state= *info->state; - info->state= &info->s->state.state; + share->state.state= *info->state; + info->state= &share->state.state; } info->append_insert_at_end= 0; } @@ -432,7 +433,8 @@ int _ma_writeinfo(register MARIA_HA *info, uint operation) share->state.process= share->last_process= share->this_process; share->state.unique= info->last_unique= info->this_unique; share->state.update_count= info->last_loop= ++info->this_loop; - if ((error= _ma_state_info_write(share->kfile.file, &share->state, 1))) + if ((error= _ma_state_info_write_sub(share->kfile.file, + &share->state, 1))) olderror=my_errno; #ifdef __WIN__ if (maria_flush) diff --git a/storage/maria/ma_loghandler.c b/storage/maria/ma_loghandler.c index f556193b147..74408c53662 100644 --- a/storage/maria/ma_loghandler.c +++ b/storage/maria/ma_loghandler.c @@ -213,6 +213,22 @@ static my_bool write_hook_for_redo(enum translog_record_type type, static my_bool write_hook_for_undo(enum translog_record_type type, TRN *trn, MARIA_HA *tbl_info, LSN *lsn, struct st_translog_parts *parts); +static my_bool write_hook_for_redo_delete_all(enum translog_record_type type, + TRN *trn, MARIA_HA *tbl_info, + LSN *lsn, + struct st_translog_parts *parts); +static my_bool write_hook_for_undo_row_insert(enum translog_record_type type, + TRN *trn, MARIA_HA *tbl_info, + LSN *lsn, + struct st_translog_parts *parts); +static my_bool write_hook_for_undo_row_delete(enum translog_record_type type, + TRN *trn, MARIA_HA *tbl_info, + LSN *lsn, + struct st_translog_parts *parts); +static my_bool write_hook_for_undo_row_purge(enum translog_record_type type, + TRN *trn, MARIA_HA *tbl_info, + LSN *lsn, + struct st_translog_parts *parts); static my_bool write_hook_for_clr_end(enum translog_record_type type, TRN *trn, MARIA_HA *tbl_info, LSN *lsn, struct st_translog_parts *parts); @@ -429,13 +445,13 @@ static LOG_DESC INIT_LOGREC_UNDO_ROW_INSERT= {LOGRECTYPE_PSEUDOFIXEDLENGTH, LSN_STORE_SIZE + FILEID_STORE_SIZE + PAGE_STORE_SIZE + DIRPOS_STORE_SIZE, LSN_STORE_SIZE + FILEID_STORE_SIZE + PAGE_STORE_SIZE + DIRPOS_STORE_SIZE, - NULL, write_hook_for_undo, NULL, 1, + NULL, write_hook_for_undo_row_insert, NULL, 1, "undo_row_insert", LOGREC_LAST_IN_GROUP, NULL, NULL}; static LOG_DESC INIT_LOGREC_UNDO_ROW_DELETE= {LOGRECTYPE_VARIABLE_LENGTH, 0, LSN_STORE_SIZE + FILEID_STORE_SIZE + PAGE_STORE_SIZE + DIRPOS_STORE_SIZE, - NULL, write_hook_for_undo, NULL, 1, + NULL, write_hook_for_undo_row_delete, NULL, 1, "undo_row_delete", LOGREC_LAST_IN_GROUP, NULL, NULL}; static LOG_DESC INIT_LOGREC_UNDO_ROW_UPDATE= @@ -447,7 +463,7 @@ static LOG_DESC INIT_LOGREC_UNDO_ROW_UPDATE= static LOG_DESC INIT_LOGREC_UNDO_ROW_PURGE= {LOGRECTYPE_PSEUDOFIXEDLENGTH, LSN_STORE_SIZE + FILEID_STORE_SIZE, LSN_STORE_SIZE + FILEID_STORE_SIZE, - NULL, write_hook_for_undo, NULL, 1, + NULL, write_hook_for_undo_row_purge, NULL, 1, "undo_row_purge", LOGREC_LAST_IN_GROUP, NULL, NULL}; static LOG_DESC INIT_LOGREC_UNDO_KEY_INSERT= @@ -493,7 +509,7 @@ static LOG_DESC INIT_LOGREC_REDO_DROP_TABLE= static LOG_DESC INIT_LOGREC_REDO_DELETE_ALL= {LOGRECTYPE_FIXEDLENGTH, FILEID_STORE_SIZE, FILEID_STORE_SIZE, - NULL, write_hook_for_redo, NULL, 0, + NULL, write_hook_for_redo_delete_all, NULL, 0, "redo_delete_all", LOGREC_IS_GROUP_ITSELF, NULL, NULL}; static LOG_DESC INIT_LOGREC_REDO_REPAIR_TABLE= @@ -6308,6 +6324,88 @@ static my_bool write_hook_for_undo(enum translog_record_type type } +/** + @brief Sets the table's records count to 0, then calls the generic REDO + hook. + + @todo move it to a separate file + + @return Operation status, always 0 (success) +*/ + +static my_bool write_hook_for_redo_delete_all(enum translog_record_type type + __attribute__ ((unused)), + TRN *trn, MARIA_HA *tbl_info + __attribute__ ((unused)), + LSN *lsn, + struct st_translog_parts *parts + __attribute__ ((unused))) +{ + tbl_info->s->state.state.records= 0; + return write_hook_for_redo(type, trn, tbl_info, lsn, parts); +} + + +/** + @brief Upates "records" and calls the generic UNDO hook + + @todo move it to a separate file + + @return Operation status, always 0 (success) +*/ + +static my_bool write_hook_for_undo_row_insert(enum translog_record_type type + __attribute__ ((unused)), + TRN *trn, MARIA_HA *tbl_info, + LSN *lsn, + struct st_translog_parts *parts + __attribute__ ((unused))) +{ + tbl_info->s->state.state.records++; + return write_hook_for_undo(type, trn, tbl_info, lsn, parts); +} + + +/** + @brief Upates "records" and calls the generic UNDO hook + + @todo move it to a separate file + + @return Operation status, always 0 (success) +*/ + +static my_bool write_hook_for_undo_row_delete(enum translog_record_type type + __attribute__ ((unused)), + TRN *trn, MARIA_HA *tbl_info, + LSN *lsn, + struct st_translog_parts *parts + __attribute__ ((unused))) +{ + tbl_info->s->state.state.records--; + return write_hook_for_undo(type, trn, tbl_info, lsn, parts); +} + + +/** + @brief Upates "records" and calls the generic UNDO hook + + @todo we will get rid of this record soon. + + @return Operation status, always 0 (success) +*/ + +static my_bool write_hook_for_undo_row_purge(enum translog_record_type type + __attribute__ ((unused)), + TRN *trn, MARIA_HA *tbl_info, + LSN *lsn, + struct st_translog_parts *parts + __attribute__ ((unused))) +{ + tbl_info->s->state.state.records--; + return write_hook_for_undo(type, trn, tbl_info, lsn, parts); +} + + /** @brief Sets transaction's undo_lsn, first_undo_lsn if needed diff --git a/storage/maria/ma_loghandler_lsn.h b/storage/maria/ma_loghandler_lsn.h index b106a4ab30e..5658d8d03e3 100644 --- a/storage/maria/ma_loghandler_lsn.h +++ b/storage/maria/ma_loghandler_lsn.h @@ -85,7 +85,7 @@ typedef LSN LSN_WITH_FLAGS; #define LSN_ERROR 1 /** @brief some impossible LSN serve as markers */ -#define LSN_REPAIRED_BY_MARIA_CHK ((LSN)1) +#define LSN_REPAIRED_BY_MARIA_CHK ((LSN)2) /** @brief the maximum valid LSN. diff --git a/storage/maria/ma_open.c b/storage/maria/ma_open.c index a8bb7b444e9..542dddf243b 100644 --- a/storage/maria/ma_open.c +++ b/storage/maria/ma_open.c @@ -613,12 +613,14 @@ MARIA_HA *maria_open(const char *name, int mode, uint open_flags) view of the server, including server's recovery) now. */ if ((open_flags & HA_OPEN_FROM_SQL_LAYER) || maria_in_recovery) - { - share->state.create_rename_lsn= translog_get_horizon(); - _ma_update_create_rename_lsn_on_disk(share, TRUE); - } + _ma_update_create_rename_lsn_on_disk_sub(share, + translog_get_horizon(), + TRUE); } - else if (!LSN_VALID(share->state.create_rename_lsn) && + else if ((!LSN_VALID(share->state.create_rename_lsn) || + !LSN_VALID(share->state.is_of_lsn) || + (cmp_translog_addr(share->state.create_rename_lsn, + share->state.is_of_lsn) > 0)) && !(open_flags & HA_OPEN_FOR_REPAIR)) { /* @@ -968,18 +970,64 @@ static void setup_key_functions(register MARIA_KEYDEF *keyinfo) /** @brief Function to save and store the header in the index file (.MYI) + Operates under MARIA_SHARE::intern_lock if requested. + Sets MARIA_SHARE::MARIA_STATE_INFO::is_of_lsn if table is transactional. + Then calls _ma_state_info_write_sub(). + + @param share table + @param pWrite bitmap: if 1 is set my_pwrite() is used otherwise + my_write(); if 2 is set, info about keys is written + (should only be needed after ALTER TABLE + ENABLE/DISABLE KEYS, and REPAIR/OPTIMIZE); if 4 is + set, MARIA_SHARE::intern_lock is taken. + + @return Operation status + @retval 0 OK + @retval 1 Error +*/ + +uint _ma_state_info_write(MARIA_SHARE *share, uint pWrite) +{ + uint res= 0; + if (pWrite & 4) + pthread_mutex_lock(&share->intern_lock); + else if (maria_multi_threaded) + safe_mutex_assert_owner(&share->intern_lock); + if (share->base.born_transactional && translog_inited && + !maria_in_recovery) + { + /* + In a recovery, we want to set is_of_lsn to the LSN of the last + record executed by Recovery, not the current EOF of the log (which + is too new). Recovery does it by itself. + */ + share->state.is_of_lsn= translog_get_horizon(); + } + res= _ma_state_info_write_sub(share->kfile.file, &share->state, pWrite); + if (pWrite & 4) + pthread_mutex_unlock(&share->intern_lock); + return res; +} + + +/** + @brief Function to save and store the header in the index file (.MYI). + + Shortcut to use instead of _ma_state_info_write() when appropriate. + @param file descriptor of the index file to write @param state state information to write to the file - @param pWrite bitmap (determines the amount of information to - write, and if my_write() or my_pwrite() should be - used) + @param pWrite bitmap: if 1 is set my_pwrite() is used otherwise + my_write(); if 2 is set, info about keys is written + (should only be needed after ALTER TABLE + ENABLE/DISABLE KEYS, and REPAIR/OPTIMIZE). @return Operation status @retval 0 OK @retval 1 Error */ -uint _ma_state_info_write(File file, MARIA_STATE_INFO *state, uint pWrite) +uint _ma_state_info_write_sub(File file, MARIA_STATE_INFO *state, uint pWrite) { /** @todo RECOVERY write it only at checkpoint time */ uchar buff[MARIA_STATE_INFO_SIZE + MARIA_STATE_EXTRA_SIZE]; @@ -994,10 +1042,11 @@ uint _ma_state_info_write(File file, MARIA_STATE_INFO *state, uint pWrite) /* open_count must be first because of _ma_mark_file_changed ! */ mi_int2store(ptr,state->open_count); ptr+= 2; /* - if you change the offset of this LSN inside the file, fix - ma_create + ma_rename + ma_delete_all + backward-compatibility. + if you change the offset of create_rename_lsn/is_of_lsn inside the file, + fix ma_create + ma_rename + ma_delete_all + backward-compatibility. */ lsn_store(ptr, state->create_rename_lsn); ptr+= LSN_STORE_SIZE; + lsn_store(ptr, state->is_of_lsn); ptr+= LSN_STORE_SIZE; *ptr++= (uchar)state->changed; *ptr++= state->sortkey; mi_rowstore(ptr,state->state.records); ptr+= 8; @@ -1022,7 +1071,7 @@ uint _ma_state_info_write(File file, MARIA_STATE_INFO *state, uint pWrite) { mi_sizestore(ptr,state->key_root[i]); ptr+= 8; } - /** @todo RECOVERY key_del is a problem for recovery */ + /** @todo RECOVERY BUG key_del is a problem for recovery */ mi_sizestore(ptr,state->key_del); ptr+= 8; if (pWrite & 2) /* From maria_chk */ { @@ -1060,6 +1109,7 @@ static uchar *_ma_state_info_read(uchar *ptr, MARIA_STATE_INFO *state) state->open_count = mi_uint2korr(ptr); ptr+= 2; state->create_rename_lsn= lsn_korr(ptr); ptr+= LSN_STORE_SIZE; + state->is_of_lsn= lsn_korr(ptr); ptr+= LSN_STORE_SIZE; state->changed= (my_bool) *ptr++; state->sortkey= (uint) *ptr++; state->state.records= mi_rowkorr(ptr); ptr+= 8; @@ -1382,11 +1432,16 @@ int _ma_open_datafile(MARIA_HA *info, MARIA_SHARE *share, int _ma_open_keyfile(MARIA_SHARE *share) { - if ((share->kfile.file= my_open(share->unique_file_name, - share->mode | O_SHARE, - MYF(MY_WME))) < 0) - return 1; - return 0; + /* + Modifications to share->kfile should be under intern_lock to protect + against a concurrent checkpoint. + */ + pthread_mutex_lock(&share->intern_lock); + share->kfile.file= my_open(share->unique_file_name, + share->mode | O_SHARE, + MYF(MY_WME)); + pthread_mutex_unlock(&share->intern_lock); + return (share->kfile.file < 0); } diff --git a/storage/maria/ma_recovery.c b/storage/maria/ma_recovery.c index 4f4681303f4..e9b1ec90de7 100644 --- a/storage/maria/ma_recovery.c +++ b/storage/maria/ma_recovery.c @@ -143,6 +143,9 @@ int maria_recover() fprintf(trace_file, "SUCCESS\n"); fclose(trace_file); } + // @todo set global_trid_generator from checkpoint or default value of 1/0, + // and also update it when seeing LOGREC_LONG_TRANSACTION_ID + // suggestion: add an arg to trnman_init maria_in_recovery= FALSE; DBUG_RETURN(res); } @@ -224,7 +227,7 @@ int maria_apply_log(LSN from_lsn, my_bool apply, FILE *trace_file, /* we don't use maria_panic() because it would maria_end(), and Recovery does - not want that (we want to keep modules initialized for runtime). + not want that (we want to keep some modules initialized for runtime). */ if (close_all_tables()) goto err; @@ -333,6 +336,10 @@ static void new_transaction(uint16 sid, TrID long_id, LSN undo_lsn, llbuf, sid); all_active_trans[sid].undo_lsn= undo_lsn; all_active_trans[sid].first_undo_lsn= first_undo_lsn; + // @todo set_if_bigger(global_trid_generator, long_id) + // indeed not only uncommitted transactions should bump generator, + // committed ones too (those not seen by undo phase so not + // into trnman_recreate) } @@ -424,6 +431,9 @@ prototype_redo_exec_hook(REDO_CREATE_TABLE) ptr+= 2; /* set create_rename_lsn (for maria_read_log to be idempotent) */ lsn_store(ptr + sizeof(info->s->state.header) + 2, rec->lsn); + /* we also set is_of_lsn, like maria_create() does */ + lsn_store(ptr + sizeof(info->s->state.header) + 2 + LSN_STORE_SIZE, + rec->lsn); if (my_pwrite(kfile, ptr, kfile_size_before_extension, 0, MYF(MY_NABP|MY_WME)) || my_chsize(kfile, keystart, 0, MYF(MY_WME))) @@ -843,11 +853,7 @@ prototype_redo_exec_hook(UNDO_ROW_INSERT) if (info == NULL) return 0; set_undo_lsn_for_active_trans(rec->short_trid, rec->lsn); - /* - in an upcoming patch ("recovery of the state"), we introduce - state.is_of_lsn. For now, we just assume the state is old (true when we - recreate tables from scratch - but not idempotent). - */ + if (cmp_translog_addr(rec->lsn, info->s->state.is_of_lsn) > 0) { fprintf(tracef, " state older than record, updating rows' count\n"); info->s->state.state.records++; @@ -870,6 +876,7 @@ prototype_redo_exec_hook(UNDO_ROW_DELETE) if (info == NULL) return 0; set_undo_lsn_for_active_trans(rec->short_trid, rec->lsn); + if (cmp_translog_addr(rec->lsn, info->s->state.is_of_lsn) > 0) { fprintf(tracef, " state older than record, updating rows' count\n"); info->s->state.state.records--; @@ -887,6 +894,7 @@ prototype_redo_exec_hook(UNDO_ROW_UPDATE) if (info == NULL) return 0; set_undo_lsn_for_active_trans(rec->short_trid, rec->lsn); + if (cmp_translog_addr(rec->lsn, info->s->state.is_of_lsn) > 0) { info->s->state.changed|= STATE_CHANGED | STATE_NOT_ANALYZED | STATE_NOT_OPTIMIZED_KEYS | STATE_NOT_SORTED_PAGES; @@ -902,6 +910,7 @@ prototype_redo_exec_hook(UNDO_ROW_PURGE) return 0; /* this a bit broken, but this log record type will be deleted soon */ set_undo_lsn_for_active_trans(rec->short_trid, rec->lsn); + if (cmp_translog_addr(rec->lsn, info->s->state.is_of_lsn) > 0) { fprintf(tracef, " state older than record, updating rows' count\n"); info->s->state.state.records--; @@ -965,6 +974,7 @@ prototype_redo_exec_hook(CLR_END) set_undo_lsn_for_active_trans(rec->short_trid, previous_undo_lsn); fprintf(tracef, " CLR_END was about %s, undo_lsn now LSN (%lu,0x%lx)\n", log_desc->name, LSN_IN_HEX(previous_undo_lsn)); + if (cmp_translog_addr(rec->lsn, info->s->state.is_of_lsn) > 0) { fprintf(tracef, " state older than record, updating rows' count\n"); switch (undone_record_type) { @@ -1395,11 +1405,23 @@ static int run_undo_phase(uint unfinished) } -static void prepare_table_for_close(MARIA_HA *info, - LSN at_lsn __attribute__ ((unused))) +/** + @brief re-enables transactionality, updates is_of_lsn + + @param info table + @param at_lsn LSN to set is_of_lsn +*/ + +static void prepare_table_for_close(MARIA_HA *info, LSN at_lsn) { MARIA_SHARE *share= info->s; - /* we will soon use at_lsn here */ + /* + State is now at least as new as the LSN of the current record. It may be + newer, in case we are seeing a LOGREC_FILE_ID which tells us to close a + table, but that table was later modified further in the log. + */ + if (cmp_translog_addr(share->state.is_of_lsn, at_lsn) < 0) + share->state.is_of_lsn= at_lsn; _ma_reenable_logging_for_table(share); } @@ -1637,11 +1659,18 @@ static int close_all_tables() if (maria_open_list == NULL) goto end; fprintf(tracef, "Closing all tables\n"); + /* + Since the end of end_of_redo_phase(), we may have written new records + (if UNDO phase ran) and thus the state is newer than at + end_of_redo_phase(), we need to bump is_of_lsn again. + */ + LSN addr= translog_get_horizon(); for (list_element= maria_open_list ; list_element ; list_element= next_open) { next_open= list_element->next; info= (MARIA_HA*)list_element->data; pthread_mutex_unlock(&THR_LOCK_maria); /* ok, UNDO phase not online yet */ + prepare_table_for_close(info, addr); error|= maria_close(info); pthread_mutex_lock(&THR_LOCK_maria); } @@ -1650,6 +1679,100 @@ end: return error; } +#ifdef MARIA_EXTERNAL_LOCKING +#error Maria's Recovery is really not ready for it +#endif + +/* +Recovery of the state : how it works +===================================== + +Ignoring Checkpoints for a start. + +The state (MARIA_HA::MARIA_SHARE::MARIA_STATE_INFO) is updated in +memory frequently (at least at every row write/update/delete) but goes +to disk at few moments: maria_close() when closing the last open +instance, and a few rare places like CHECK/REPAIR/ALTER +(non-transactional tables also do it at maria_lock_database() but we +needn't cover them here). + +In case of crash, state on disk is likely to be older than what it was +in memory, the REDO phase needs to recreate the state as it was in +memory at the time of crash. When we say Recovery here we will always +mean "REDO phase". + +For example MARIA_STATUS_INFO::records (count of records). It is updated at +the end of every row write/update/delete/delete_all. When Recovery sees the +sign of such row operation (UNDO or REDO), it may need to update the records' +count if that count does not reflect that operation (is older). How to know +the age of the state compared to the log record: every time the state +goes to disk at runtime, its member "is_of_lsn" is updated to the +current end-of-log LSN. So Recovery just needs to compare is_of_lsn +and the record's LSN to know if it should modify "records". + +Other operations like ALTER TABLE DISABLE KEYS update the state but +don't write log records, thus the REDO phase cannot repeat their +effect on the state in case of crash. But we make them sync the state +as soon as they have finished. This reduces the window for a problem. + +It looks like only one thread at a time updates the state in memory or +on disk. However there is not 100% certainty when it comes to +HA_EXTRA_(FORCE_REOPEN|PREPARE_FOR_RENAME): can they read the state +from memory while some other thread is updating "records" in memory? +If yes, they may write a corrupted state to disk. +We assume that no for now: ASK_MONTY. + +With checkpoints +================ + +Checkpoint module needs to read the state in memory and write it to +disk. This may happen while some other thread is modifying the state +in memory or on disk. Checkpoint thus may be reading changing data, it +needs a mutex to not have it corrupted, and concurrent modifiers of +the state need that mutex too for the same reason. +"records" is modified for every row write/update/delete, we don't want +to add a mutex lock/unlock there. So we re-use the mutex lock/unlock +which is already present in these moments, namely the log's mutex which is +taken when UNDO_ROW_INSERT|UPDATE|DELETE is written: we update "records" in +under-log-mutex hooks when writing these records (thus "records" is +not updated at the end of maria_write/update/delete() anymore). +Thus Checkpoint takes the log's lock and can read "records" from +memory an write it to disk and release log's lock. +We however want to avoid having the disk write under the log's +lock. So it has to be under another mutex, natural choice is +intern_lock (as Checkpoint needs it anyway to read MARIA_SHARE::kfile, +and as maria_close() takes it too). All state writes to disk are +changed to be protected with intern_lock. +So Checkpoint takes intern_lock, log's lock, reads "records" from +memory, releases log's lock, updates is_of_lsn and writes "records" to +disk, release intern_lock. +In practice, not only "records" needs to be written but the full +state. So, Checkpoint reads the full state from memory. Some other +thread may at this moment be modifying in memory some pieces of the +state which are not protected by the lock's log (see ma_extra.c +HA_EXTRA_NO_KEYS), and Checkpoint would be reading a corrupted state +from memory; to guard against that we extend the intern_lock-zone to +changes done to the state in memory by HA_EXTRA_NO_KEYS et al, and +also any change made in memory to create_rename_lsn/state_is_of_lsn. +Last, we don't want in Checkpoint to do + log lock; read state from memory; release log lock; +for each table, it may hold the log's lock too much in total. +So, we instead do + log lock; read N states from memory; release log lock; +Thus, the sequence above happens outside of any intern_lock. +But this re-introduces the problem that some other thread may be changing the +state in memory and on disk under intern_lock, without log's lock, like +HA_EXTRA_NO_KEYS, while we read the N states. However, when Checkpoint later +comes to handling the table under intern_lock, which is serialized with +HA_EXTRA_NO_KEYS, it can see that is_of_lsn is higher then when the state was +read from memory under log's lock, and thus can decide to not flush the +obsolete state it has, knowing that the other thread flushed a more recent +state already. If on the other hand is_of_lsn is not higher, the read state is +current and can be flushed. So we have a per-table sequence: + lock intern_lock; test if is_of_lsn is higher than when we read the state + under log's lock; if no then flush the read state to disk. +*/ + /* some comments and pseudo-code which we keep for later */ #if 0 /* diff --git a/storage/maria/ma_rename.c b/storage/maria/ma_rename.c index 6250b781a68..ba05d745195 100644 --- a/storage/maria/ma_rename.c +++ b/storage/maria/ma_rename.c @@ -66,6 +66,7 @@ int maria_rename(const char *old_name, const char *new_name) !maria_in_recovery) ? MY_SYNC_DIR : 0; if (sync_dir) { + LSN lsn; uchar log_data[2 + 2]; LEX_STRING log_array[TRANSLOG_INTERNAL_PARTS + 3]; uint old_name_len= strlen(old_name), new_name_len= strlen(new_name); @@ -85,13 +86,12 @@ int maria_rename(const char *old_name, const char *new_name) under THR_LOCK_maria or not...), how to use it in Recovery. For now it can serve to apply logs to a backup so we sync it. */ - if (unlikely(translog_write_record(&share->state.create_rename_lsn, - LOGREC_REDO_RENAME_TABLE, + if (unlikely(translog_write_record(&lsn, LOGREC_REDO_RENAME_TABLE, &dummy_transaction_object, NULL, 2 + 2 + old_name_len + new_name_len, sizeof(log_array)/sizeof(log_array[0]), log_array, NULL) || - translog_flush(share->state.create_rename_lsn))) + translog_flush(lsn))) { maria_close(info); DBUG_RETURN(1); @@ -100,7 +100,7 @@ int maria_rename(const char *old_name, const char *new_name) store LSN into file, needed for Recovery to not be confused if a RENAME happened (applying REDOs to the wrong table). */ - if (_ma_update_create_rename_lsn_on_disk(share, TRUE)) + if (_ma_update_create_rename_lsn_on_disk(share, lsn, TRUE)) { maria_close(info); DBUG_RETURN(1); diff --git a/storage/maria/ma_test1.c b/storage/maria/ma_test1.c index 6360153a171..df8b19859fd 100644 --- a/storage/maria/ma_test1.c +++ b/storage/maria/ma_test1.c @@ -75,7 +75,7 @@ int main(int argc,char *argv[]) if (maria_init() || (init_pagecache(maria_pagecache, IO_SIZE*16, 0, 0, maria_block_size) == 0) || - ma_control_file_create_or_open(TRUE) || + ma_control_file_create_or_open() || (init_pagecache(maria_log_pagecache, TRANSLOG_PAGECACHE_SIZE, 0, 0, TRANSLOG_PAGE_SIZE) == 0) || diff --git a/storage/maria/ma_test2.c b/storage/maria/ma_test2.c index 19816606720..a69fa97d51b 100644 --- a/storage/maria/ma_test2.c +++ b/storage/maria/ma_test2.c @@ -89,7 +89,7 @@ int main(int argc, char *argv[]) if (maria_init() || (init_pagecache(maria_pagecache, pagecache_size, 0, 0, maria_block_size) == 0) || - ma_control_file_create_or_open(TRUE) || + ma_control_file_create_or_open() || (init_pagecache(maria_log_pagecache, TRANSLOG_PAGECACHE_SIZE, 0, 0, TRANSLOG_PAGE_SIZE) == 0) || diff --git a/storage/maria/ma_test_recovery b/storage/maria/ma_test_recovery index fdf54736404..a88814ade7f 100755 --- a/storage/maria/ma_test_recovery +++ b/storage/maria/ma_test_recovery @@ -100,7 +100,7 @@ set -- "ma_test1 $silent -M -T -c" "ma_test2 $silent -L -K -W -P -M -T -c" "ma_t while [ $# != 0 ] do prog=$1 - rm maria_log.* maria_log_control + rm -f maria_log.* maria_log_control echo "TEST WITH $prog" $maria_path/$prog # derive table's name from program's name @@ -138,7 +138,7 @@ do prog=$1 commit_run_args=$2 abort_run_args=$3; - rm maria_log.* maria_log_control + rm -f maria_log.* maria_log_control echo "TEST WITH $prog $commit_run_args (commit at end)" $maria_path/$prog $commit_run_args # derive table's name from program's name @@ -193,7 +193,7 @@ done done rm -f $table.* $tmp/$table* $tmp/maria_chk_*.txt $tmp/maria_read_log_$table.txt -) > $tmp/ma_test_recovery.output +) 2>&1 > $tmp/ma_test_recovery.output diff $maria_path/ma_test_recovery.expected $tmp/ma_test_recovery.output > /dev/null || diff_failed=1 if [ "$diff_failed" == "1" ] diff --git a/storage/maria/ma_test_recovery.expected b/storage/maria/ma_test_recovery.expected index 67be4940bb1..3ded68e7f56 100644 --- a/storage/maria/ma_test_recovery.expected +++ b/storage/maria/ma_test_recovery.expected @@ -21,12 +21,10 @@ testing idempotency applying log Differences in maria_chk -dvv, recovery not yet perfect ! ========DIFF START======= -7,8c7,8 +7c7 < Checksum: 3757530372 -< Data records: 15 Deleted blocks: 0 --- > Checksum: 0 -> Data records: 30 Deleted blocks: 0 11c11 < Datafile length: 16384 Keyfile length: 16384 --- @@ -41,7 +39,7 @@ applying log Differences in maria_chk -dvv, recovery not yet perfect ! ========DIFF START======= 11c11 -< Datafile length: 90112 Keyfile length: 212992 +< Datafile length: 90112 Keyfile length: 204800 --- > Datafile length: 90112 Keyfile length: 8192 ========DIFF END======= @@ -50,7 +48,7 @@ applying log Differences in maria_chk -dvv, recovery not yet perfect ! ========DIFF START======= 11c11 -< Datafile length: 90112 Keyfile length: 212992 +< Datafile length: 90112 Keyfile length: 204800 --- > Datafile length: 90112 Keyfile length: 8192 ========DIFF END======= @@ -97,12 +95,10 @@ testing idempotency applying log Differences in maria_chk -dvv, recovery not yet perfect ! ========DIFF START======= -7,8c7,8 +7c7 < Checksum: 221293111 -< Data records: 25 Deleted blocks: 0 --- > Checksum: 0 -> Data records: 50 Deleted blocks: 0 11c11 < Datafile length: 16384 Keyfile length: 16384 --- @@ -134,22 +130,8 @@ TEST WITH ma_test1 -s -M -T -c -N --debug=d:t:i:o,/tmp/ma_test1.trace --testfla terminating after deletes Dying on request without maria_commit()/maria_close() applying log -Differences in maria_chk -dvv, recovery not yet perfect ! -========DIFF START======= -8c8 -< Data records: 27 Deleted blocks: 0 ---- -> Data records: 54 Deleted blocks: 0 -========DIFF END======= testing idempotency applying log -Differences in maria_chk -dvv, recovery not yet perfect ! -========DIFF START======= -8c8 -< Data records: 27 Deleted blocks: 0 ---- -> Data records: 81 Deleted blocks: 0 -========DIFF END======= testing applying of CLRs to recreate table applying log Differences in maria_chk -dvv, recovery not yet perfect ! @@ -191,12 +173,10 @@ testing idempotency applying log Differences in maria_chk -dvv, recovery not yet perfect ! ========DIFF START======= -7,8c7,8 +7c7 < Checksum: 221293111 -< Data records: 25 Deleted blocks: 0 --- > Checksum: 0 -> Data records: 50 Deleted blocks: 0 11c11 < Datafile length: 16384 Keyfile length: 16384 --- @@ -228,22 +208,8 @@ TEST WITH ma_test1 -s -M -T -c -N --debug=d:t:i:o,/tmp/ma_test1.trace --testfla terminating after deletes Dying on request without maria_commit()/maria_close() applying log -Differences in maria_chk -dvv, recovery not yet perfect ! -========DIFF START======= -8c8 -< Data records: 27 Deleted blocks: 0 ---- -> Data records: 54 Deleted blocks: 0 -========DIFF END======= testing idempotency applying log -Differences in maria_chk -dvv, recovery not yet perfect ! -========DIFF START======= -8c8 -< Data records: 27 Deleted blocks: 0 ---- -> Data records: 81 Deleted blocks: 0 -========DIFF END======= testing applying of CLRs to recreate table applying log Differences in maria_chk -dvv, recovery not yet perfect ! @@ -285,12 +251,10 @@ testing idempotency applying log Differences in maria_chk -dvv, recovery not yet perfect ! ========DIFF START======= -7,8c7,8 +7c7 < Checksum: 221293111 -< Data records: 25 Deleted blocks: 0 --- > Checksum: 0 -> Data records: 50 Deleted blocks: 0 11c11 < Datafile length: 16384 Keyfile length: 16384 --- @@ -322,22 +286,8 @@ TEST WITH ma_test1 -s -M -T -c -N --debug=d:t:i:o,/tmp/ma_test1.trace --testfla terminating after deletes Dying on request without maria_commit()/maria_close() applying log -Differences in maria_chk -dvv, recovery not yet perfect ! -========DIFF START======= -8c8 -< Data records: 27 Deleted blocks: 0 ---- -> Data records: 54 Deleted blocks: 0 -========DIFF END======= testing idempotency applying log -Differences in maria_chk -dvv, recovery not yet perfect ! -========DIFF START======= -8c8 -< Data records: 27 Deleted blocks: 0 ---- -> Data records: 81 Deleted blocks: 0 -========DIFF END======= testing applying of CLRs to recreate table applying log Differences in maria_chk -dvv, recovery not yet perfect ! @@ -379,12 +329,10 @@ testing idempotency applying log Differences in maria_chk -dvv, recovery not yet perfect ! ========DIFF START======= -7,8c7,8 +7c7 < Checksum: 411409161 -< Data records: 25 Deleted blocks: 0 --- > Checksum: 0 -> Data records: 50 Deleted blocks: 0 11c11 < Datafile length: 49152 Keyfile length: 16384 --- @@ -416,22 +364,8 @@ TEST WITH ma_test1 -s -M -T -c -N --debug=d:t:i:o,/tmp/ma_test1.trace -b --testf terminating after deletes Dying on request without maria_commit()/maria_close() applying log -Differences in maria_chk -dvv, recovery not yet perfect ! -========DIFF START======= -8c8 -< Data records: 27 Deleted blocks: 0 ---- -> Data records: 54 Deleted blocks: 0 -========DIFF END======= testing idempotency applying log -Differences in maria_chk -dvv, recovery not yet perfect ! -========DIFF START======= -8c8 -< Data records: 27 Deleted blocks: 0 ---- -> Data records: 81 Deleted blocks: 0 -========DIFF END======= testing applying of CLRs to recreate table applying log Differences in maria_chk -dvv, recovery not yet perfect ! @@ -473,12 +407,10 @@ testing idempotency applying log Differences in maria_chk -dvv, recovery not yet perfect ! ========DIFF START======= -7,8c7,8 +7c7 < Checksum: 411409161 -< Data records: 25 Deleted blocks: 0 --- > Checksum: 0 -> Data records: 50 Deleted blocks: 0 11c11 < Datafile length: 49152 Keyfile length: 16384 --- @@ -510,22 +442,8 @@ TEST WITH ma_test1 -s -M -T -c -N --debug=d:t:i:o,/tmp/ma_test1.trace -b --testf terminating after deletes Dying on request without maria_commit()/maria_close() applying log -Differences in maria_chk -dvv, recovery not yet perfect ! -========DIFF START======= -8c8 -< Data records: 27 Deleted blocks: 0 ---- -> Data records: 54 Deleted blocks: 0 -========DIFF END======= testing idempotency applying log -Differences in maria_chk -dvv, recovery not yet perfect ! -========DIFF START======= -8c8 -< Data records: 27 Deleted blocks: 0 ---- -> Data records: 81 Deleted blocks: 0 -========DIFF END======= testing applying of CLRs to recreate table applying log Differences in maria_chk -dvv, recovery not yet perfect ! @@ -567,12 +485,10 @@ testing idempotency applying log Differences in maria_chk -dvv, recovery not yet perfect ! ========DIFF START======= -7,8c7,8 +7c7 < Checksum: 411409161 -< Data records: 25 Deleted blocks: 0 --- > Checksum: 0 -> Data records: 50 Deleted blocks: 0 11c11 < Datafile length: 49152 Keyfile length: 16384 --- @@ -604,22 +520,8 @@ TEST WITH ma_test1 -s -M -T -c -N --debug=d:t:i:o,/tmp/ma_test1.trace -b --testf terminating after deletes Dying on request without maria_commit()/maria_close() applying log -Differences in maria_chk -dvv, recovery not yet perfect ! -========DIFF START======= -8c8 -< Data records: 27 Deleted blocks: 0 ---- -> Data records: 54 Deleted blocks: 0 -========DIFF END======= testing idempotency applying log -Differences in maria_chk -dvv, recovery not yet perfect ! -========DIFF START======= -8c8 -< Data records: 27 Deleted blocks: 0 ---- -> Data records: 81 Deleted blocks: 0 -========DIFF END======= testing applying of CLRs to recreate table applying log Differences in maria_chk -dvv, recovery not yet perfect ! diff --git a/storage/maria/ma_write.c b/storage/maria/ma_write.c index cb15280fc6e..4a6ace13741 100644 --- a/storage/maria/ma_write.c +++ b/storage/maria/ma_write.c @@ -162,12 +162,20 @@ int maria_write(MARIA_HA *info, uchar *record) rw_unlock(&share->key_root_lock[i]); } } + /** + @todo RECOVERY BUG + this += must happen under log's mutex when writing the UNDO + */ if (share->calc_write_checksum) info->cur_row.checksum= (*share->calc_write_checksum)(info,record); if (filepos != HA_OFFSET_ERROR) { if ((*share->write_record)(info,record)) goto err; + /** + @todo when we enable multiple writers, we will have to protect + 'records' and 'checksum' somehow. + */ info->state->checksum+= info->cur_row.checksum; } if (share->base.auto_key) @@ -175,7 +183,7 @@ int maria_write(MARIA_HA *info, uchar *record) ma_retrieve_auto_increment(info, record)); info->update= (HA_STATE_CHANGED | HA_STATE_AKTIV | HA_STATE_WRITTEN | HA_STATE_ROW_CHANGED); - info->state->records++; + info->state->records+= !share->now_transactional; /*otherwise already done*/ info->cur_row.lastpos= filepos; VOID(_ma_writeinfo(info, WRITEINFO_UPDATE_KEYFILE)); if (info->invalidator != 0) diff --git a/storage/maria/maria_chk.c b/storage/maria/maria_chk.c index 9f51b0caf64..edd99d01629 100644 --- a/storage/maria/maria_chk.c +++ b/storage/maria/maria_chk.c @@ -1035,7 +1035,8 @@ static int maria_chk(HA_CHECK *param, char *filename) that it will have to find and store it. */ if (share->base.born_transactional) - share->state.create_rename_lsn= LSN_REPAIRED_BY_MARIA_CHK; + share->state.create_rename_lsn= share->state.is_of_lsn= + LSN_REPAIRED_BY_MARIA_CHK; if ((param->testflag & (T_REP_BY_SORT | T_REP_PARALLEL)) && (maria_is_any_key_active(share->state.key_map) || (rep_quick && !param->keys_in_use && !recreate)) && diff --git a/storage/maria/maria_def.h b/storage/maria/maria_def.h index 368764d7a17..7fc502d0771 100644 --- a/storage/maria/maria_def.h +++ b/storage/maria/maria_def.h @@ -95,6 +95,7 @@ typedef struct st_maria_state_info uint open_count; uint8 changed; /* Changed since mariachk */ LSN create_rename_lsn; /**< LSN when table was last created/renamed */ + LSN is_of_lsn; /**< LSN when state was last updated on disk */ /* the following isn't saved on disk */ uint state_diff_length; /* Should be 0 */ @@ -104,7 +105,7 @@ typedef struct st_maria_state_info #define MARIA_STATE_INFO_SIZE \ - (24 + LSN_STORE_SIZE + 4 + 11*8 + 4*4 + 8 + 3*4 + 5*8) + (24 + LSN_STORE_SIZE*2 + 4 + 11*8 + 4*4 + 8 + 3*4 + 5*8) #define MARIA_STATE_KEY_SIZE 8 #define MARIA_STATE_KEYBLOCK_SIZE 8 #define MARIA_STATE_KEYSEG_SIZE 4 @@ -214,6 +215,8 @@ typedef struct st_maria_file_bitmap ulong pages_covered; /* Pages covered by bitmap + 1 */ } MARIA_FILE_BITMAP; +#define MARIA_CHECKPOINT_LOOKS_AT_ME 1 +#define MARIA_CHECKPOINT_SHOULD_FREE_ME 2 typedef struct st_maria_share { /* Shared between opens */ @@ -300,6 +303,7 @@ typedef struct st_maria_share myf write_flag; enum data_file_type data_file_type; enum pagecache_page_type page_type; /* value depending transactional */ + uint8 in_checkpoint; /**< if Checkpoint looking at table */ my_bool temporary; /* Below flag is needed to make log tables work with concurrent insert */ my_bool is_log_table; @@ -864,7 +868,8 @@ extern uint _ma_nommap_pread(MARIA_HA *info, uchar *Buffer, extern uint _ma_nommap_pwrite(MARIA_HA *info, uchar *Buffer, uint Count, my_off_t offset, myf MyFlags); -uint _ma_state_info_write(File file, MARIA_STATE_INFO *state, uint pWrite); +uint _ma_state_info_write(MARIA_SHARE *share, uint pWrite); +uint _ma_state_info_write_sub(File file, MARIA_STATE_INFO *state, uint pWrite); uint _ma_state_info_read_dsk(File file, MARIA_STATE_INFO *state); uint _ma_base_info_write(File file, MARIA_BASE_INFO *base); int _ma_keyseg_write(File file, const HA_KEYSEG *keyseg); @@ -933,7 +938,10 @@ int _ma_create_index_by_sort(MARIA_SORT_PARAM *info, my_bool no_messages, ulong); int _ma_sync_table_files(const MARIA_HA *info); int _ma_initialize_data_file(MARIA_SHARE *share, File dfile); -int _ma_update_create_rename_lsn_on_disk(MARIA_SHARE *share, my_bool do_sync); +int _ma_update_create_rename_lsn_on_disk(MARIA_SHARE *share, + LSN lsn, my_bool do_sync); +int _ma_update_create_rename_lsn_on_disk_sub(MARIA_SHARE *share, + LSN lsn, my_bool do_sync); void _ma_unpin_all_pages(MARIA_HA *info, LSN undo_lsn); #define _ma_tmp_disable_logging_for_table(S) \ diff --git a/storage/maria/maria_pack.c b/storage/maria/maria_pack.c index 987711a270d..83f88fcb0dc 100644 --- a/storage/maria/maria_pack.c +++ b/storage/maria/maria_pack.c @@ -3000,7 +3000,8 @@ static int save_state(MARIA_HA *isam_file,PACK_MRG_INFO *mrg, VOID(my_chsize(share->kfile.file, share->base.keystart, 0, MYF(0))); if (share->base.keys) isamchk_neaded=1; - DBUG_RETURN(_ma_state_info_write(share->kfile.file, &share->state, (1 + 2))); + DBUG_RETURN(_ma_state_info_write_sub(share->kfile.file, + &share->state, (1 + 2))); } @@ -3033,7 +3034,7 @@ static int save_state_mrg(File file,PACK_MRG_INFO *mrg,my_off_t new_length, if (isam_file->s->base.keys) isamchk_neaded=1; state.changed=STATE_CHANGED | STATE_NOT_ANALYZED; /* Force check of table */ - DBUG_RETURN (_ma_state_info_write(file,&state,1+2)); + DBUG_RETURN (_ma_state_info_write_sub(file,&state,1+2)); } -- cgit v1.2.1 From 69d7db7758fec5a711602b775358aa7bf4a3e744 Mon Sep 17 00:00:00 2001 From: unknown Date: Fri, 7 Sep 2007 15:52:25 +0200 Subject: WL#3072 - Maria Recovery At the end of recovery, we initialize the transaction manager's trid generator with the maximum trid seen during the REDO phase. This ensures that trids always grow (needed for versioning), even after a crash. This patch is only preparation, as ma_recover() is not called from ha_maria yet. storage/maria/ha_maria.cc: trnman_init() needs argument now (soon trnman_init() will rather be done via ma_recover() and thus it will not be 0) storage/maria/ma_recovery.c: During the REDO phase, remember the max long trid of transactions which we have seen (both in the checkpoint record and the LOGREC_LONG_TRANSACTION_ID records) storage/maria/ma_test1.c: trnman_init() needs argument now storage/maria/ma_test2.c: trnman_init() needs argument now storage/maria/trnman.c: new argument to trnman_init() so that caller can decide which value the generator of trids starts from. storage/maria/trnman_public.h: trnman_init() needs argument now storage/maria/unittest/trnman-t.c: trnman_init() needs argument now --- storage/maria/ha_maria.cc | 2 +- storage/maria/ma_recovery.c | 14 ++++++-------- storage/maria/ma_test1.c | 2 +- storage/maria/ma_test2.c | 2 +- storage/maria/trnman.c | 15 +++++++++++++-- storage/maria/trnman_public.h | 2 +- storage/maria/unittest/trnman-t.c | 2 +- 7 files changed, 24 insertions(+), 15 deletions(-) diff --git a/storage/maria/ha_maria.cc b/storage/maria/ha_maria.cc index 78d2bdc1da5..ab3c9a9310c 100644 --- a/storage/maria/ha_maria.cc +++ b/storage/maria/ha_maria.cc @@ -2403,7 +2403,7 @@ static int ha_maria_init(void *p) translog_init(maria_data_root, TRANSLOG_FILE_SIZE, MYSQL_VERSION_ID, server_id, maria_log_pagecache, TRANSLOG_DEFAULT_FLAGS) || - trnman_init(); + trnman_init(0); maria_multi_threaded= TRUE; return res; } diff --git a/storage/maria/ma_recovery.c b/storage/maria/ma_recovery.c index e9b1ec90de7..ccfab16d0a8 100644 --- a/storage/maria/ma_recovery.c +++ b/storage/maria/ma_recovery.c @@ -47,6 +47,7 @@ static HASH all_dirty_pages; static struct st_dirty_page *dirty_pages_pool; static LSN current_group_end_lsn, checkpoint_start= LSN_IMPOSSIBLE; +static TrID max_long_trid= 0; /**< max long trid seen by REDO phase */ static FILE *tracef; /**< trace file for debugging */ #define prototype_redo_exec_hook(R) \ @@ -143,9 +144,6 @@ int maria_recover() fprintf(trace_file, "SUCCESS\n"); fclose(trace_file); } - // @todo set global_trid_generator from checkpoint or default value of 1/0, - // and also update it when seeing LOGREC_LONG_TRANSACTION_ID - // suggestion: add an arg to trnman_init maria_in_recovery= FALSE; DBUG_RETURN(res); } @@ -336,10 +334,7 @@ static void new_transaction(uint16 sid, TrID long_id, LSN undo_lsn, llbuf, sid); all_active_trans[sid].undo_lsn= undo_lsn; all_active_trans[sid].first_undo_lsn= first_undo_lsn; - // @todo set_if_bigger(global_trid_generator, long_id) - // indeed not only uncommitted transactions should bump generator, - // committed ones too (those not seen by undo phase so not - // into trnman_recreate) + set_if_bigger(max_long_trid, long_id); } @@ -1278,6 +1273,7 @@ static int run_redo_phase(LSN lsn, my_bool apply) static uint end_of_redo_phase(my_bool prepare_for_undo_phase) { uint sid, unfinished= 0; + char llbuf[22]; hash_free(&all_dirty_pages); /* @@ -1288,7 +1284,9 @@ static uint end_of_redo_phase(my_bool prepare_for_undo_phase) my_free(dirty_pages_pool, MYF(MY_ALLOW_ZERO_PTR)); dirty_pages_pool= NULL; - if (prepare_for_undo_phase && trnman_init()) + llstr(max_long_trid, llbuf); + printf("Maximum transaction long id seen: %s\n", llbuf); + if (prepare_for_undo_phase && trnman_init(max_long_trid)) return -1; for (sid= 0; sid <= SHORT_TRID_MAX; sid++) diff --git a/storage/maria/ma_test1.c b/storage/maria/ma_test1.c index df8b19859fd..dd74f6b0f07 100644 --- a/storage/maria/ma_test1.c +++ b/storage/maria/ma_test1.c @@ -82,7 +82,7 @@ int main(int argc,char *argv[]) translog_init(maria_data_root, TRANSLOG_FILE_SIZE, 0, 0, maria_log_pagecache, TRANSLOG_DEFAULT_FLAGS) || - (transactional && trnman_init())) + (transactional && trnman_init(0))) { fprintf(stderr, "Error in initialization"); exit(1); diff --git a/storage/maria/ma_test2.c b/storage/maria/ma_test2.c index a69fa97d51b..596374eef80 100644 --- a/storage/maria/ma_test2.c +++ b/storage/maria/ma_test2.c @@ -96,7 +96,7 @@ int main(int argc, char *argv[]) translog_init(maria_data_root, TRANSLOG_FILE_SIZE, 0, 0, maria_log_pagecache, TRANSLOG_DEFAULT_FLAGS) || - (transactional && trnman_init())) + (transactional && trnman_init(0))) { fprintf(stderr, "Error in initialization"); exit(1); diff --git a/storage/maria/trnman.c b/storage/maria/trnman.c index b0550085863..d2d4549a895 100644 --- a/storage/maria/trnman.c +++ b/storage/maria/trnman.c @@ -102,7 +102,18 @@ static uchar *trn_get_hash_key(const uchar *trn, size_t *len, return (uchar *) & ((*((TRN **)trn))->trid); } -int trnman_init() + +/** + @brief Initializes transaction manager. + + @param initial_trid Generated TrIDs will start from initial_trid+1. + + @return Operation status + @retval 0 OK + @retval !=0 Error +*/ + +int trnman_init(TrID initial_trid) { DBUG_ENTER("trnman_init"); @@ -138,7 +149,7 @@ int trnman_init() trnman_allocated_transactions= 0; pool= 0; - global_trid_generator= 0; /* set later by the recovery code */ + global_trid_generator= initial_trid; lf_hash_init(&trid_to_committed_trn, sizeof(TRN*), LF_HASH_UNIQUE, 0, 0, trn_get_hash_key, 0); DBUG_PRINT("info", ("pthread_mutex_init LOCK_trn_list")); diff --git a/storage/maria/trnman_public.h b/storage/maria/trnman_public.h index 10dcb479530..97b492c3a57 100644 --- a/storage/maria/trnman_public.h +++ b/storage/maria/trnman_public.h @@ -34,7 +34,7 @@ typedef struct st_transaction TRN; extern uint trnman_active_transactions, trnman_allocated_transactions; extern TRN dummy_transaction_object; -int trnman_init(void); +int trnman_init(TrID); void trnman_destroy(void); TRN *trnman_new_trn(pthread_mutex_t *, pthread_cond_t *, void *); int trnman_end_trn(TRN *trn, my_bool commit); diff --git a/storage/maria/unittest/trnman-t.c b/storage/maria/unittest/trnman-t.c index b0a087370f2..db137cf088c 100644 --- a/storage/maria/unittest/trnman-t.c +++ b/storage/maria/unittest/trnman-t.c @@ -174,7 +174,7 @@ int main() #define CYCLES 10000 #define THREADS 10 - trnman_init(); + trnman_init(0); test_trnman_read_from(); run_test("trnman", test_trnman, THREADS, CYCLES); -- cgit v1.2.1 From 64da2c3f6affbb818fbb76457a6a5d3f080f65f8 Mon Sep 17 00:00:00 2001 From: unknown Date: Fri, 7 Sep 2007 16:59:12 +0200 Subject: If Maria engine is not compiled in, don't use page caches (fix for compiler errors in pushbuild). Small bugfix. sql/handler.h: don't use pagecaches if no Maria storage/maria/ma_check.c: correcting mistake in previous push; need to call this function otherwise create_rename_lsn would not be updated at end of REPAIR. --- sql/handler.h | 6 ++++++ storage/maria/ma_check.c | 4 ++++ 2 files changed, 10 insertions(+) diff --git a/sql/handler.h b/sql/handler.h index 6805876ff49..feb551670a3 100644 --- a/sql/handler.h +++ b/sql/handler.h @@ -23,7 +23,9 @@ #include #include #include +#ifdef WITH_MARIA_STORAGE_ENGINE #include "../storage/maria/ma_pagecache.h" +#endif #ifndef NO_HASH #define NO_HASH /* Not yet implemented */ @@ -859,7 +861,9 @@ typedef struct st_ha_check_opt uint flags; /* isam layer flags (e.g. for myisamchk) */ uint sql_flags; /* sql layer flags - for something myisamchk cannot do */ KEY_CACHE *key_cache; /* new key cache when changing key cache */ +#ifdef WITH_MARIA_STORAGE_ENGINE PAGECACHE *pagecache; /* new pagecache when changing pagecache */ +#endif void init(); } HA_CHECK_OPT; @@ -1818,6 +1822,7 @@ int ha_resize_key_cache(KEY_CACHE *key_cache); int ha_change_key_cache_param(KEY_CACHE *key_cache); int ha_change_key_cache(KEY_CACHE *old_key_cache, KEY_CACHE *new_key_cache); int ha_end_key_cache(KEY_CACHE *key_cache); +#ifdef WITH_MARIA_STORAGE_ENGINE /* pagecache */ int ha_init_pagecache(const char *name, PAGECACHE *pagecache); /* @@ -1827,6 +1832,7 @@ int ha_resize_pagecache(PAGECACHE *pagecache); int ha_change_pagecache_param(PAGECACHE *pagecache); int ha_change_pagecache(PAGECACHE *old_pagecache, PAGECACHE *new_pagecache); int ha_end_pagecache(PAGECACHE *pagecache); +#endif /* report to InnoDB that control passes to the client */ int ha_release_temporary_latches(THD *thd); diff --git a/storage/maria/ma_check.c b/storage/maria/ma_check.c index 52390de9690..6ed4a3f2757 100644 --- a/storage/maria/ma_check.c +++ b/storage/maria/ma_check.c @@ -5604,6 +5604,10 @@ static int write_log_record_for_repair(const HA_CHECK *param, MARIA_HA *info) /* The table's existence was made durable earlier (MY_SYNC_DIR passed to maria_change_to_newfile()). + */ + if (_ma_update_create_rename_lsn_on_disk(info->s, lsn, FALSE)) + return 1; + /* _ma_flush_table_files_after_repair() is later called by maria_repair(), and makes sure to flush the data, index, update is_of_lsn, flush state and sync, so create_rename_lsn reaches disk, thus we won't apply old -- cgit v1.2.1 From 02fd80bacb76a87574046ae028e41fc06d7aab9c Mon Sep 17 00:00:00 2001 From: unknown Date: Fri, 7 Sep 2007 17:03:36 +0200 Subject: Fix for pushbuild maria.test failure, where directory syncing failed at the end of translog_flush() when datadir was in /dev/shm. storage/maria/ma_loghandler.c: directory syncing can fail on shared memory devices (/dev/shm on Linux in this case); see my_sync_dir(). --- storage/maria/ma_loghandler.c | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/storage/maria/ma_loghandler.c b/storage/maria/ma_loghandler.c index 74408c53662..7c5c4e4842f 100644 --- a/storage/maria/ma_loghandler.c +++ b/storage/maria/ma_loghandler.c @@ -6243,7 +6243,7 @@ my_bool translog_flush(LSN lsn) } log_descriptor.flushed= sent_to_file; /** @todo LOG decide if syncing of directory is needed */ - rc|= my_sync(log_descriptor.directory_fd, MYF(MY_WME)); + rc|= my_sync(log_descriptor.directory_fd, MYF(MY_WME | MY_IGNORE_BADFD)); translog_unlock(); DBUG_RETURN(rc); } -- cgit v1.2.1 From 602e13cb055b7c58c4f81c971c74d82cfc8c946b Mon Sep 17 00:00:00 2001 From: unknown Date: Fri, 7 Sep 2007 19:45:33 +0200 Subject: enable --with-maria-storage-engine --- storage/maria/plug.in | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/storage/maria/plug.in b/storage/maria/plug.in index 198f5d8c289..1ce64f6e2bb 100644 --- a/storage/maria/plug.in +++ b/storage/maria/plug.in @@ -1,4 +1,4 @@ -MYSQL_STORAGE_ENGINE(maria, no, [Maria Storage Engine], +MYSQL_STORAGE_ENGINE(maria,, [Maria Storage Engine], [Traditional transactional MySQL tables], [max,max-no-ndb]) MYSQL_PLUGIN_DIRECTORY(maria, [storage/maria]) MYSQL_PLUGIN_ACTIONS(maria, [AC_CONFIG_FILES(storage/maria/unittest/Makefile)]) -- cgit v1.2.1 From 155193a6e27b22e5557e536dd9189e26bbb8fb3a Mon Sep 17 00:00:00 2001 From: unknown Date: Sun, 9 Sep 2007 19:15:10 +0300 Subject: Added applying of undo for updates Fixed bug in duplicate key handling for block records during repair All read-row methods now return error number in case of error Don't calculate checksum for null fields Fixed bug when running maria_read_log with -o BUILD/SETUP.sh: Added STACK_DIRECTION BUILD/compile-pentium-debug-max: Moved STACK_DIRECTION to SETUP include/myisam.h: Added extra parameter to write_key storage/maria/ma_blockrec.c: Added applying of undo for updates Fixed indentation Removed some not needed casts Fixed wrong logging of CLR record Split ma_update_block_record to two functions to be able to reuse it from undo-applying Simplify filling of packed fields ma_record_block_record) now returns error number on failure Sligtly changed log record information for undo-update storage/maria/ma_check.c: Fixed bug in duplicate key handling for block records during repair storage/maria/ma_checksum.c: Don't calculate checksum for null fields storage/maria/ma_dynrec.c: _ma_read_dynamic_reocrd() now returns error number on error Rest of the changes are code simplification and indentation fixes storage/maria/ma_locking.c: Added comment storage/maria/ma_loghandler.c: More debugging Removed printing of total_record_length as this was always same as record_length storage/maria/ma_open.c: Allocate bitmap for changed fields storage/maria/ma_packrec.c: read_record now returns error number on error storage/maria/ma_recovery.c: Fixed wrong arguments to undo_row_update storage/maria/ma_statrec.c: read_record now returns error number on error (not 1) Code simplification storage/maria/ma_test1.c: Added exit possibility after update phase (to test undo of updates) storage/maria/maria_def.h: Include bitmap header file storage/maria/maria_read_log.c: Fixed bug when running with -o --- BUILD/SETUP.sh | 6 +- BUILD/compile-pentium-debug-max | 2 +- include/myisam.h | 2 +- storage/maria/ma_blockrec.c | 309 ++++++++++++++++++++++++++++++++-------- storage/maria/ma_check.c | 14 +- storage/maria/ma_checksum.c | 7 +- storage/maria/ma_dynrec.c | 191 +++++++++++++------------ storage/maria/ma_locking.c | 3 + storage/maria/ma_loghandler.c | 8 +- storage/maria/ma_open.c | 5 + storage/maria/ma_packrec.c | 12 +- storage/maria/ma_recovery.c | 10 +- storage/maria/ma_statrec.c | 21 +-- storage/maria/ma_test1.c | 19 ++- storage/maria/maria_def.h | 4 +- storage/maria/maria_read_log.c | 3 +- 16 files changed, 418 insertions(+), 198 deletions(-) diff --git a/BUILD/SETUP.sh b/BUILD/SETUP.sh index 6b3708f475f..429c0cd85b4 100755 --- a/BUILD/SETUP.sh +++ b/BUILD/SETUP.sh @@ -170,10 +170,10 @@ max_configs="$SSL_LIBRARY --with-plugins=max --with-embedded-server" # CPU and platform specific compilation flags. # alpha_cflags="$check_cpu_cflags -Wa,-m$cpu_flag" -amd64_cflags="$check_cpu_cflags" +amd64_cflags="$check_cpu_cflags -DSTACK_DIRECTION=-1" amd64_cxxflags="" # If dropping '--with-big-tables', add here "-DBIG_TABLES" -pentium_cflags="$check_cpu_cflags" -pentium64_cflags="$check_cpu_cflags -m64" +pentium_cflags="$check_cpu_cflags -DSTACK_DIRECTION=-1" +pentium64_cflags="$check_cpu_cflags -m64 -DSTACK_DIRECTION=-1" ppc_cflags="$check_cpu_cflags" sparc_cflags="" diff --git a/BUILD/compile-pentium-debug-max b/BUILD/compile-pentium-debug-max index 39d12a8320a..941a63a209f 100755 --- a/BUILD/compile-pentium-debug-max +++ b/BUILD/compile-pentium-debug-max @@ -4,7 +4,7 @@ path=`dirname $0` set -- "$@" --with-debug=full . "$path/SETUP.sh" -extra_flags="$pentium_cflags $debug_cflags -DSTACK_DIRECTION=-1" +extra_flags="$pentium_cflags $debug_cflags" extra_configs="$pentium_configs $debug_configs $max_configs $error_inject --with-experimental-collations" . "$path/FINISH.sh" diff --git a/include/myisam.h b/include/myisam.h index 154a3d95b41..75c56e100c9 100644 --- a/include/myisam.h +++ b/include/myisam.h @@ -364,7 +364,7 @@ typedef struct st_mi_sort_param NEAR int (*write_keys)(struct st_mi_sort_param *, register uchar **, uint , struct st_buffpek *, IO_CACHE *); NEAR uint (*read_to_buffer)(IO_CACHE *,struct st_buffpek *, uint); - NEAR int (*write_key)(struct st_mi_sort_param *, IO_CACHE *,char *, + NEAR int (*write_key)(struct st_mi_sort_param *, IO_CACHE *,uchar *, uint, uint); } MI_SORT_PARAM; diff --git a/storage/maria/ma_blockrec.c b/storage/maria/ma_blockrec.c index 55dc72b1d02..9dfdab4688d 100644 --- a/storage/maria/ma_blockrec.c +++ b/storage/maria/ma_blockrec.c @@ -677,6 +677,46 @@ static my_bool check_if_zero(uchar *pos, uint length) } +/* + @brief Copy not changed fields from 'from' to 'to' + + @notes + Assumption is that most fields are not changed! + (Which is why we don't test if all bits are set for some bytes in bitmap) +*/ + +void copy_not_changed_fields(MARIA_HA *info, MY_BITMAP *changed_fields, + uchar *to, uchar *from) +{ + MARIA_COLUMNDEF *column, *end_column; + uchar *bitmap= (uchar*) changed_fields->bitmap; + MARIA_SHARE *share= info->s; + uint bit= 1; + + for (column= share->columndef, end_column= column+ share->base.fields; + column < end_column; column++) + { + if (!(*bitmap & bit)) + { + uint field_length= column->length; + if (column->type == FIELD_VARCHAR) + { + if (column->fill_length == 1) + field_length= (uint) from[column->offset] + 1; + else + field_length= uint2korr(from + column->offset) + 2; + } + memcpy(to + column->offset, from + column->offset, field_length); + } + if ((bit= (bit << 1)) == 256) + { + bitmap++; + bit= 1; + } + } +} + + /* Unpin all pinned pages @@ -878,7 +918,7 @@ static void calc_record_size(MARIA_HA *info, const uchar *record, *blob_lengths++= 0; continue; } - switch ((enum en_fieldtype) column->type) { + switch (column->type) { case FIELD_CHECK: case FIELD_NORMAL: /* Fixed length field */ case FIELD_ZERO: @@ -1321,8 +1361,8 @@ static my_bool write_tail(MARIA_HA *info, LSN lsn; /* Log REDO changes of tail page */ - page_store(log_data+ FILEID_STORE_SIZE, block->page); - dirpos_store(log_data+ FILEID_STORE_SIZE + PAGE_STORE_SIZE, + page_store(log_data + FILEID_STORE_SIZE, block->page); + dirpos_store(log_data + FILEID_STORE_SIZE + PAGE_STORE_SIZE, row_pos.rownr); log_array[TRANSLOG_INTERNAL_PARTS + 0].str= (char*) log_data; log_array[TRANSLOG_INTERNAL_PARTS + 0].length= sizeof(log_data); @@ -1804,7 +1844,7 @@ static my_bool write_block_record(MARIA_HA *info, continue; field_pos= record + column->offset; - switch ((enum en_fieldtype) column->type) { + switch (column->type) { case FIELD_NORMAL: /* Fixed length field */ case FIELD_SKIP_PRESPACE: case FIELD_SKIP_ZERO: /* Fixed length field */ @@ -2295,7 +2335,7 @@ static my_bool write_block_record(MARIA_HA *info, if (translog_write_record(&lsn, LOGREC_CLR_END, info->trn, info, sizeof(log_data), TRANSLOG_INTERNAL_PARTS + 1, log_array, - log_data+ FILEID_STORE_SIZE)) + log_data + LSN_STORE_SIZE)) goto disk_err; } else @@ -2305,9 +2345,9 @@ static my_bool write_block_record(MARIA_HA *info, /* LOGREC_UNDO_ROW_INSERT & LOGREC_UNDO_ROW_INSERT share same header */ lsn_store(log_data, info->trn->undo_lsn); - page_store(log_data+ LSN_STORE_SIZE + FILEID_STORE_SIZE, + page_store(log_data + LSN_STORE_SIZE + FILEID_STORE_SIZE, head_block->page); - dirpos_store(log_data+ LSN_STORE_SIZE + FILEID_STORE_SIZE + + dirpos_store(log_data + LSN_STORE_SIZE + FILEID_STORE_SIZE + PAGE_STORE_SIZE, row_pos->rownr); @@ -2329,12 +2369,13 @@ static my_bool write_block_record(MARIA_HA *info, size_t row_length; uint row_parts_count; row_length= fill_update_undo_parts(info, old_record, record, - info->log_row_parts + + log_array + TRANSLOG_INTERNAL_PARTS + 1, &row_parts_count); if (translog_write_record(&lsn, LOGREC_UNDO_ROW_UPDATE, info->trn, info, sizeof(log_data) + row_length, - TRANSLOG_INTERNAL_PARTS + 1 + row_parts_count, + TRANSLOG_INTERNAL_PARTS + 1 + + row_parts_count, log_array, log_data + LSN_STORE_SIZE)) goto disk_err; } @@ -2601,8 +2642,11 @@ my_bool _ma_write_abort_block_record(MARIA_HA *info) for rows split into many extents. */ -my_bool _ma_update_block_record(MARIA_HA *info, MARIA_RECORD_POS record_pos, - const uchar *oldrec, const uchar *record) +static my_bool _ma_update_block_record2(MARIA_HA *info, + MARIA_RECORD_POS record_pos, + const uchar *oldrec, + const uchar *record, + LSN undo_lsn) { MARIA_BITMAP_BLOCKS *blocks= &info->cur_row.insert_blocks; uchar *buff; @@ -2614,9 +2658,14 @@ my_bool _ma_update_block_record(MARIA_HA *info, MARIA_RECORD_POS record_pos, ulonglong page; struct st_row_pos_info row_pos; MARIA_SHARE *share= info->s; - DBUG_ENTER("_ma_update_block_record"); + DBUG_ENTER("_ma_update_block_record2"); DBUG_PRINT("enter", ("rowid: %lu", (long) record_pos)); +#ifdef ENABLE_IF_PROBLEM_WITH_UPDATE + DBUG_DUMP("oldrec", oldrec, share->base.reclength); + DBUG_DUMP("newrec", record, share->base.reclength); +#endif + calc_record_size(info, record, new_row); page= ma_recordpos_to_page(record_pos); @@ -2669,11 +2718,12 @@ my_bool _ma_update_block_record(MARIA_HA *info, MARIA_RECORD_POS record_pos, if (cur_row->extents_count && free_full_pages(info, cur_row)) goto err; DBUG_RETURN(write_block_record(info, oldrec, record, new_row, blocks, - 1, &row_pos, 0)); + 1, &row_pos, undo_lsn)); } /* Allocate all size in block for record - QQ: Need to improve this to do compact if we can fit one more blob into + TODO: + Need to improve this to do compact if we can fit one more blob into the head page */ head_length= uint2korr(dir + 2); @@ -2702,7 +2752,7 @@ my_bool _ma_update_block_record(MARIA_HA *info, MARIA_RECORD_POS record_pos, row_pos.data= buff + uint2korr(dir); row_pos.length= head_length; DBUG_RETURN(write_block_record(info, oldrec, record, new_row, blocks, 1, - &row_pos, 0)); + &row_pos, undo_lsn)); err: _ma_unpin_all_pages(info, 0); @@ -2710,6 +2760,16 @@ err: } +/* Wrapper for _ma_update_block_record2() used by ma_update() */ + + +my_bool _ma_update_block_record(MARIA_HA *info, MARIA_RECORD_POS record_pos, + const uchar *orig_rec, const uchar *new_rec) +{ + return _ma_update_block_record2(info, record_pos, orig_rec, new_rec, 0); +} + + /* Delete a directory entry @@ -2848,8 +2908,8 @@ static my_bool delete_head_or_tail(MARIA_HA *info, if (info->s->now_transactional) { /* Log REDO data */ - page_store(log_data+ FILEID_STORE_SIZE, page); - dirpos_store(log_data+ FILEID_STORE_SIZE + PAGE_STORE_SIZE, + page_store(log_data + FILEID_STORE_SIZE, page); + dirpos_store(log_data + FILEID_STORE_SIZE + PAGE_STORE_SIZE, record_number); log_array[TRANSLOG_INTERNAL_PARTS + 0].str= (char*) log_data; @@ -2877,7 +2937,7 @@ static my_bool delete_head_or_tail(MARIA_HA *info, PAGE_STORE_SIZE + PAGERANGE_STORE_SIZE]; LEX_STRING log_array[TRANSLOG_INTERNAL_PARTS + 1]; pagerange_store(log_data + FILEID_STORE_SIZE, 1); - page_store(log_data+ FILEID_STORE_SIZE + PAGERANGE_STORE_SIZE, page); + page_store(log_data + FILEID_STORE_SIZE + PAGERANGE_STORE_SIZE, page); pagerange_store(log_data + FILEID_STORE_SIZE + PAGERANGE_STORE_SIZE + PAGE_STORE_SIZE, 1); log_array[TRANSLOG_INTERNAL_PARTS + 0].str= (char*) log_data; @@ -2975,8 +3035,8 @@ my_bool _ma_delete_block_record(MARIA_HA *info, const uchar *record) /* Write UNDO record */ lsn_store(log_data, info->trn->undo_lsn); - page_store(log_data+ LSN_STORE_SIZE + FILEID_STORE_SIZE, page); - dirpos_store(log_data+ LSN_STORE_SIZE + FILEID_STORE_SIZE + + page_store(log_data + LSN_STORE_SIZE + FILEID_STORE_SIZE, page); + dirpos_store(log_data + LSN_STORE_SIZE + FILEID_STORE_SIZE + PAGE_STORE_SIZE, record_number); info->log_row_parts[TRANSLOG_INTERNAL_PARTS].str= (char*) log_data; @@ -3421,16 +3481,14 @@ int _ma_read_block_record2(MARIA_HA *info, uchar *record, for (end_column= share->columndef + share->base.fields; column < end_column; column++) { - enum en_fieldtype type= (enum en_fieldtype) column->type; + enum en_fieldtype type= column->type; uchar *field_pos= record + column->offset; /* First check if field is present in record */ if ((record[column->null_pos] & column->null_bit) || (cur_row->empty_bits[column->empty_pos] & column->empty_bit)) { - if (type == FIELD_SKIP_ENDSPACE) - bfill(record + column->offset, column->length, ' '); - else - bzero(record + column->offset, column->fill_length); + bfill(record + column->offset, column->fill_length, + type == FIELD_SKIP_ENDSPACE ? ' ' : 0); continue; } switch (type) { @@ -3586,7 +3644,7 @@ err: This function is a simpler version of _ma_read_block_record2() The data about the used pages is stored in info->cur_row. - @return + @return Status @retval 0 ok @retval 1 Error. my_errno contains error number */ @@ -3679,11 +3737,14 @@ static my_bool read_row_extent_info(MARIA_HA *info, uchar *buff, /* Read a record based on record position - SYNOPSIS - _ma_read_block_record() - info Maria handler - record Store record here - record_pos Record position + @fn _ma_read_block_record() + @param info Maria handler + @param record Store record here + @param record_pos Record position + + @return Status + @retval 0 ok + @retval # Error number */ int _ma_read_block_record(MARIA_HA *info, uchar *record, @@ -3702,13 +3763,13 @@ int _ma_read_block_record(MARIA_HA *info, uchar *record, &info->dfile, ma_recordpos_to_page(record_pos), 0, info->buff, info->s->page_type, PAGECACHE_LOCK_LEFT_UNLOCKED, 0))) - DBUG_RETURN(1); + DBUG_RETURN(my_errno); DBUG_ASSERT((buff[PAGE_TYPE_OFFSET] & PAGE_TYPE_MASK) == HEAD_PAGE); if (!(data= get_record_position(buff, block_size, offset, &end_of_data))) { - my_errno= HA_ERR_WRONG_IN_RECORD; /* File crashed */ DBUG_PRINT("error", ("Wrong directory entry in data block")); - DBUG_RETURN(1); + my_errno= HA_ERR_WRONG_IN_RECORD; /* File crashed */ + DBUG_RETURN(HA_ERR_WRONG_IN_RECORD); } DBUG_RETURN(_ma_read_block_record2(info, record, data, end_of_data)); } @@ -4204,7 +4265,7 @@ static size_t fill_insert_undo_parts(MARIA_HA *info, const uchar *record, column_pos= record+ column->offset; column_length= column->length; - switch ((enum en_fieldtype) column->type) { + switch (column->type) { case FIELD_CHECK: case FIELD_NORMAL: /* Fixed length field */ case FIELD_ZERO: @@ -4288,7 +4349,7 @@ static size_t fill_insert_undo_parts(MARIA_HA *info, const uchar *record, Fields are stored in same order as the field array. - Number of changed fields (packed) + Offset to changed field data (packed) For each changed field Fieldnumber (packed) @@ -4323,7 +4384,7 @@ static size_t fill_update_undo_parts(MARIA_HA *info, const uchar *oldrec, uchar *old_field_lengths= old_row->field_lengths; uchar *new_field_lengths= new_row->field_lengths; size_t row_length= 0; - uint field_count= 0; + uint field_lengths; LEX_STRING *start_log_parts; my_bool new_column_is_empty; DBUG_ENTER("fill_update_undo_parts"); @@ -4341,7 +4402,6 @@ static size_t fill_update_undo_parts(MARIA_HA *info, const uchar *oldrec, { /* Store changed null bits */ *field_data++= (uchar) 255; /* Special case */ - field_count++; log_parts->str= (char*) oldrec; log_parts->length= share->base.null_bytes; row_length= log_parts->length; @@ -4359,10 +4419,9 @@ static size_t fill_update_undo_parts(MARIA_HA *info, const uchar *oldrec, { field_data= ma_store_length(field_data, (uint) (column - share->columndef)); - field_count++; log_parts->str= (char*) oldrec + column->offset; log_parts->length= column->length; - row_length+= log_parts->length; + row_length+= column->length; log_parts++; } } @@ -4394,7 +4453,6 @@ static size_t fill_update_undo_parts(MARIA_HA *info, const uchar *oldrec, field_data= ma_store_length(field_data, (uint) (column - share->columndef)); field_data= ma_store_length(field_data, 0); - field_count++; continue; } /* @@ -4409,7 +4467,7 @@ static size_t fill_update_undo_parts(MARIA_HA *info, const uchar *oldrec, new_column_pos= newrec + column->offset; old_column_length= new_column_length= column->length; - switch ((enum en_fieldtype) column->type) { + switch (column->type) { case FIELD_CHECK: case FIELD_NORMAL: /* Fixed length field */ case FIELD_ZERO: @@ -4418,6 +4476,8 @@ static size_t fill_update_undo_parts(MARIA_HA *info, const uchar *oldrec, break; case FIELD_VARCHAR: new_column_length--; /* Skip length prefix */ + old_column_pos+= column->fill_length; + new_column_pos+= column->fill_length; /* Fall through */ case FIELD_SKIP_ENDSPACE: /* CHAR */ { @@ -4465,22 +4525,22 @@ static size_t fill_update_undo_parts(MARIA_HA *info, const uchar *oldrec, field_data= ma_store_length(field_data, (uint) (column - share->columndef)); field_data= ma_store_length(field_data, old_column_length); - field_count++; log_parts->str= (char*) old_column_pos; log_parts->length= old_column_length; - row_length+= log_parts->length; + row_length+= old_column_length; log_parts++; } } *log_parts_count= (log_parts - start_log_parts); - /* Store number of fields before the field/field_lengths */ + /* Store length of field length data before the field/field_lengths */ + field_lengths= (field_data - start_field_data); start_log_parts->str= ((char*) (start_field_data - - ma_calc_length_for_store_length(field_count))); - ma_store_length(start_log_parts->str, field_count); + ma_calc_length_for_store_length(field_lengths))); + ma_store_length(start_log_parts->str, field_lengths); start_log_parts->length= (size_t) ((char*) field_data - start_log_parts->str); row_length+= start_log_parts->length; @@ -4864,7 +4924,7 @@ my_bool _ma_apply_undo_row_insert(MARIA_HA *info, LSN undo_lsn, if (translog_write_record(&lsn, LOGREC_CLR_END, info->trn, info, sizeof(log_data), TRANSLOG_INTERNAL_PARTS + 1, log_array, - log_data+ FILEID_STORE_SIZE)) + log_data + FILEID_STORE_SIZE)) goto err; info->s->state.state.records--; @@ -4952,10 +5012,11 @@ my_bool _ma_apply_undo_row_delete(MARIA_HA *info, LSN undo_lsn, else *blob_lengths++= 0; if (share->calc_checksum) - bzero(record + column->offset, column->length); + bfill(record + column->offset, column->fill_length, + column->type == FIELD_SKIP_ENDSPACE ? ' ' : 0); continue; } - switch ((enum en_fieldtype) column->type) { + switch (column->type) { case FIELD_CHECK: case FIELD_NORMAL: /* Fixed length field */ case FIELD_ZERO: @@ -5041,15 +5102,147 @@ my_bool _ma_apply_undo_row_delete(MARIA_HA *info, LSN undo_lsn, } -/* Execute undo of a row update */ +/* + Execute undo of a row update + + @fn _ma_apply_undo_row_update() + + @return Operation status + @retval 0 OK + @retval 1 Error +*/ -my_bool _ma_apply_undo_row_update(MARIA_HA *info __attribute__ ((unused)), - LSN undo_lsn __attribute__ ((unused)), - const uchar *header __attribute__ ((unused)), - size_t length __attribute__ ((unused))) +my_bool _ma_apply_undo_row_update(MARIA_HA *info, LSN undo_lsn, + const uchar *header, + size_t header_length __attribute__((unused))) { + ulonglong page; + uint rownr, field_length_header; + MARIA_SHARE *share= info->s; + const uchar *field_length_data, *field_length_data_end; + uchar *current_record, *orig_record; + int error= 1; + MARIA_RECORD_POS record_pos; DBUG_ENTER("_ma_apply_undo_row_update"); - fprintf(stderr, "Undo of row update is not yet done\n"); - exit(1); - DBUG_RETURN(0); + + page= page_korr(header); + rownr= dirpos_korr(header + PAGE_STORE_SIZE); + record_pos= ma_recordpos(page, rownr); + DBUG_PRINT("enter", ("Page: %lu rownr: %u", (ulong) page, rownr)); + + /* + Set header to point to old field values, generated by + fill_update_undo_parts() + */ + header+= PAGE_STORE_SIZE + DIRPOS_STORE_SIZE; + field_length_header= ma_get_length((uchar**) &header); + field_length_data= header; + header+= field_length_header; + field_length_data_end= header; + + /* Allocate buffer for current row & original row */ + if (!(current_record= my_malloc(share->base.reclength * 2, MYF(MY_WME)))) + DBUG_RETURN(1); + orig_record= current_record+ share->base.reclength; + + /* Read current record */ + if (_ma_read_block_record(info, current_record, record_pos)) + goto err; + + if (*field_length_data == 255) + { + /* Bitmap changed */ + field_length_data++; + memcpy(orig_record, header, share->base.null_bytes); + header+= share->base.null_bytes; + } + else + memcpy(orig_record, current_record, share->base.null_bytes); + bitmap_clear_all(&info->changed_fields); + + while (field_length_data < field_length_data_end) + { + uint field_nr= ma_get_length((uchar**) &field_length_data), field_length; + MARIA_COLUMNDEF *column= share->columndef + field_nr; + uchar *orig_field_pos= orig_record + column->offset; + + bitmap_set_bit(&info->changed_fields, field_nr); + if (field_nr >= share->base.fixed_not_null_fields) + { + if (!(field_length= ma_get_length((uchar**) &field_length_data))) + { + /* Null field or empty field */ + bfill(orig_field_pos, column->fill_length, + column->type == FIELD_SKIP_ENDSPACE ? ' ' : 0); + continue; + } + } + else + field_length= column->length; + + switch (column->type) { + case FIELD_CHECK: + case FIELD_NORMAL: /* Fixed length field */ + case FIELD_ZERO: + case FIELD_SKIP_PRESPACE: /* Not packed */ + memcpy(orig_field_pos, header, column->length); + header+= column->length; + break; + case FIELD_SKIP_ZERO: /* Number */ + case FIELD_SKIP_ENDSPACE: /* CHAR */ + { + uint diff; + memcpy(orig_field_pos, header, field_length); + if ((diff= (column->length - field_length))) + bfill(orig_field_pos + column->length - diff, diff, + column->type == FIELD_SKIP_ENDSPACE ? ' ' : 0); + header+= field_length; + } + break; + case FIELD_VARCHAR: + if (column->length <= 256) + { + *orig_field_pos++= (uchar) field_length; + } + else + { + int2store(orig_field_pos, field_length); + orig_field_pos+= 2; + } + memcpy(orig_field_pos, header, field_length); + header+= field_length; + break; + case FIELD_BLOB: + { + uint size_length= column->length - portable_sizeof_char_ptr; + _ma_store_blob_length(orig_field_pos, size_length, field_length); + memcpy_fixed(orig_field_pos + size_length, &header, sizeof(header)); + header+= field_length; + break; + } + default: + DBUG_ASSERT(0); + } + } + copy_not_changed_fields(info, &info->changed_fields, + orig_record, current_record); + + if (share->calc_checksum) + { + info->cur_row.checksum= (*share->calc_checksum)(info, orig_record); + info->state->checksum+= (info->cur_row.checksum - + (*share->calc_checksum)(info, current_record)); + } + + /* + Now records are up to date, execute the update to original values + */ + if (_ma_update_block_record2(info, record_pos, current_record, orig_record, + undo_lsn)) + goto err; + + error= 0; +err: + my_free(current_record, MYF(0)); + DBUG_RETURN(error); } diff --git a/storage/maria/ma_check.c b/storage/maria/ma_check.c index a5e64cb555c..f051dc4518e 100644 --- a/storage/maria/ma_check.c +++ b/storage/maria/ma_check.c @@ -2165,9 +2165,19 @@ int maria_repair(HA_CHECK *param, register MARIA_HA *info, param->error_printed=1; goto err; } - continue; + /* purecov: begin tested */ + if (block_record) + { + sort_info.new_info->state->records--; + if ((*sort_info.new_info->s->write_record_abort)(sort_info.new_info)) + { + _ma_check_print_error(param,"Couldn't delete duplicate row"); + goto err; + } + continue; + } + /* purecov: end */ } - if (!block_record && _ma_sort_write_record(&sort_param)) goto err; } diff --git a/storage/maria/ma_checksum.c b/storage/maria/ma_checksum.c index 30921ad8213..9076b3ebb86 100644 --- a/storage/maria/ma_checksum.c +++ b/storage/maria/ma_checksum.c @@ -31,6 +31,9 @@ ha_checksum _ma_checksum(MARIA_HA *info, const uchar *record) const uchar *pos= record + column->offset; ulong length; + if (record[column->null_pos] & column->null_bit) + continue; /* Null field */ + switch (column->type) { case FIELD_BLOB: { @@ -45,12 +48,12 @@ ha_checksum _ma_checksum(MARIA_HA *info, const uchar *record) } case FIELD_VARCHAR: { - uint pack_length= HA_VARCHAR_PACKLENGTH(column->length-1); + uint pack_length= column->fill_length; if (pack_length == 1) length= (ulong) *(uchar*) pos; else length= uint2korr(pos); - pos+= pack_length; + pos+= pack_length; /* Skip length information */ break; } default: diff --git a/storage/maria/ma_dynrec.c b/storage/maria/ma_dynrec.c index 52ade04db98..6e13fbcecb6 100644 --- a/storage/maria/ma_dynrec.c +++ b/storage/maria/ma_dynrec.c @@ -1138,10 +1138,14 @@ err: } +/* + @brief Unpacks a record - /* Unpacks a record */ - /* Returns -1 and my_errno =HA_ERR_RECORD_DELETED if reclength isn't */ - /* right. Returns reclength (>0) if ok */ + @return Recordlength + @retval >0 ok + @retval MY_FILE_ERROR (== -1) Error. + my_errno is set to HA_ERR_WRONG_IN_RECORD +*/ ulong _ma_rec_unpack(register MARIA_HA *info, register uchar *to, uchar *from, ulong found_length) @@ -1369,9 +1373,10 @@ void _ma_store_blob_length(uchar *pos,uint pack_length,uint length) part of the record. RETURN - 0 OK - 1 Error + 0 OK + # Error number */ + int _ma_read_dynamic_record(MARIA_HA *info, uchar *buf, MARIA_RECORD_POS filepos) { @@ -1379,103 +1384,102 @@ int _ma_read_dynamic_record(MARIA_HA *info, uchar *buf, uint b_type; MARIA_BLOCK_INFO block_info; File file; + uchar *to; + uint left_length; DBUG_ENTER("_ma_read_dynamic_record"); - if (filepos != HA_OFFSET_ERROR) + if (filepos == HA_OFFSET_ERROR) + goto err; + + LINT_INIT(to); + LINT_INIT(left_length); + file= info->dfile.file; + block_of_record= 0; /* First block of record is numbered as zero. */ + block_info.second_read= 0; + do { - uchar *to; - uint left_length; - - LINT_INIT(to); - LINT_INIT(left_length); - file= info->dfile.file; - block_of_record= 0; /* First block of record is numbered as zero. */ - block_info.second_read= 0; - do - { - /* A corrupted table can have wrong pointers. (Bug# 19835) */ - if (filepos == HA_OFFSET_ERROR) + /* A corrupted table can have wrong pointers. (Bug# 19835) */ + if (filepos == HA_OFFSET_ERROR) + goto panic; + if (info->opt_flag & WRITE_CACHE_USED && + (info->rec_cache.pos_in_file < filepos + + MARIA_BLOCK_INFO_HEADER_LENGTH) && + flush_io_cache(&info->rec_cache)) + goto err; + info->rec_cache.seek_not_done=1; + if ((b_type= _ma_get_block_info(&block_info, file, filepos)) & + (BLOCK_DELETED | BLOCK_ERROR | BLOCK_SYNC_ERROR | + BLOCK_FATAL_ERROR)) + { + if (b_type & (BLOCK_SYNC_ERROR | BLOCK_DELETED)) + my_errno=HA_ERR_RECORD_DELETED; + goto err; + } + if (block_of_record++ == 0) /* First block */ + { + if (block_info.rec_len > (uint) info->s->base.max_pack_length) goto panic; - if (info->opt_flag & WRITE_CACHE_USED && - (info->rec_cache.pos_in_file < filepos + - MARIA_BLOCK_INFO_HEADER_LENGTH) && - flush_io_cache(&info->rec_cache)) - goto err; - info->rec_cache.seek_not_done=1; - if ((b_type= _ma_get_block_info(&block_info, file, filepos)) & - (BLOCK_DELETED | BLOCK_ERROR | BLOCK_SYNC_ERROR | - BLOCK_FATAL_ERROR)) - { - if (b_type & (BLOCK_SYNC_ERROR | BLOCK_DELETED)) - my_errno=HA_ERR_RECORD_DELETED; - goto err; - } - if (block_of_record++ == 0) /* First block */ - { - if (block_info.rec_len > (uint) info->s->base.max_pack_length) - goto panic; - if (info->s->base.blobs) - { - if (_ma_alloc_buffer(&info->rec_buff, &info->rec_buff_size, - block_info.rec_len + - info->s->base.extra_rec_buff_size)) - goto err; - } - to= info->rec_buff; - left_length=block_info.rec_len; - } - if (left_length < block_info.data_len || ! block_info.data_len) - goto panic; /* Wrong linked record */ - /* copy information that is already read */ + if (info->s->base.blobs) { - uint offset= (uint) (block_info.filepos - filepos); - uint prefetch_len= (sizeof(block_info.header) - offset); - filepos+= sizeof(block_info.header); - - if (prefetch_len > block_info.data_len) - prefetch_len= block_info.data_len; - if (prefetch_len) - { - memcpy((uchar*) to, block_info.header + offset, prefetch_len); - block_info.data_len-= prefetch_len; - left_length-= prefetch_len; - to+= prefetch_len; - } + if (_ma_alloc_buffer(&info->rec_buff, &info->rec_buff_size, + block_info.rec_len + + info->s->base.extra_rec_buff_size)) + goto err; } - /* read rest of record from file */ - if (block_info.data_len) + to= info->rec_buff; + left_length=block_info.rec_len; + } + if (left_length < block_info.data_len || ! block_info.data_len) + goto panic; /* Wrong linked record */ + /* copy information that is already read */ + { + uint offset= (uint) (block_info.filepos - filepos); + uint prefetch_len= (sizeof(block_info.header) - offset); + filepos+= sizeof(block_info.header); + + if (prefetch_len > block_info.data_len) + prefetch_len= block_info.data_len; + if (prefetch_len) { - if (info->opt_flag & WRITE_CACHE_USED && - info->rec_cache.pos_in_file < filepos + block_info.data_len && - flush_io_cache(&info->rec_cache)) - goto err; - /* - What a pity that this method is not called 'file_pread' and that - there is no equivalent without seeking. We are at the right - position already. :( - */ - if (info->s->file_read(info, (uchar*) to, block_info.data_len, - filepos, MYF(MY_NABP))) - goto panic; - left_length-=block_info.data_len; - to+=block_info.data_len; + memcpy((uchar*) to, block_info.header + offset, prefetch_len); + block_info.data_len-= prefetch_len; + left_length-= prefetch_len; + to+= prefetch_len; } - filepos= block_info.next_filepos; - } while (left_length); + } + /* read rest of record from file */ + if (block_info.data_len) + { + if (info->opt_flag & WRITE_CACHE_USED && + info->rec_cache.pos_in_file < filepos + block_info.data_len && + flush_io_cache(&info->rec_cache)) + goto err; + /* + What a pity that this method is not called 'file_pread' and that + there is no equivalent without seeking. We are at the right + position already. :( + */ + if (info->s->file_read(info, (uchar*) to, block_info.data_len, + filepos, MYF(MY_NABP))) + goto panic; + left_length-=block_info.data_len; + to+=block_info.data_len; + } + filepos= block_info.next_filepos; + } while (left_length); - info->update|= HA_STATE_AKTIV; /* We have a aktive record */ - fast_ma_writeinfo(info); - DBUG_RETURN(_ma_rec_unpack(info,buf,info->rec_buff,block_info.rec_len) != - MY_FILE_ERROR ? 0 : 1); - } + info->update|= HA_STATE_AKTIV; /* We have a aktive record */ + fast_ma_writeinfo(info); + DBUG_RETURN(_ma_rec_unpack(info,buf,info->rec_buff,block_info.rec_len) != + MY_FILE_ERROR ? 0 : my_errno); + +err: fast_ma_writeinfo(info); - DBUG_RETURN(1); /* Wrong data to read */ + DBUG_RETURN(my_errno); panic: my_errno=HA_ERR_WRONG_IN_RECORD; -err: - VOID(_ma_writeinfo(info,0)); - DBUG_RETURN(1); + goto err; } /* compare unique constraint between stored rows */ @@ -1655,7 +1659,7 @@ err: RETURN 0 OK - != 0 Error + != 0 Error number */ int _ma_read_rnd_dynamic_record(MARIA_HA *info, @@ -1663,7 +1667,7 @@ int _ma_read_rnd_dynamic_record(MARIA_HA *info, MARIA_RECORD_POS filepos, my_bool skip_deleted_blocks) { - int block_of_record, info_read, save_errno; + int block_of_record, info_read; uint left_len,b_type; uchar *to; MARIA_BLOCK_INFO block_info; @@ -1827,9 +1831,8 @@ int _ma_read_rnd_dynamic_record(MARIA_HA *info, panic: my_errno=HA_ERR_WRONG_IN_RECORD; /* Something is fatal wrong */ err: - save_errno=my_errno; - VOID(_ma_writeinfo(info,0)); - DBUG_RETURN(my_errno=save_errno); + fast_ma_writeinfo(info); + DBUG_RETURN(my_errno); } diff --git a/storage/maria/ma_locking.c b/storage/maria/ma_locking.c index dad4071edf8..190356af76b 100644 --- a/storage/maria/ma_locking.c +++ b/storage/maria/ma_locking.c @@ -387,6 +387,9 @@ int _ma_readinfo(register MARIA_HA *info, int lock_type, int check_keybuffer) /* Every isam-function that uppdates the isam-database MUST end with this request + + NOTES + my_errno is not changed if this succeeds! */ int _ma_writeinfo(register MARIA_HA *info, uint operation) diff --git a/storage/maria/ma_loghandler.c b/storage/maria/ma_loghandler.c index 56c0e1aaef7..24ec2b52e8d 100644 --- a/storage/maria/ma_loghandler.c +++ b/storage/maria/ma_loghandler.c @@ -4901,8 +4901,8 @@ my_bool translog_write_record(LSN *lsn, int rc; uint short_trid= trn->short_id; DBUG_ENTER("translog_write_record"); - DBUG_PRINT("enter", ("type: %u ShortTrID: %u", - (uint) type, (uint)short_trid)); + DBUG_PRINT("enter", ("type: %u ShortTrID: %u rec_len: %lu", + (uint) type, (uint) short_trid, (ulong) rec_len)); if (tbl_info) { @@ -4995,9 +4995,7 @@ my_bool translog_write_record(LSN *lsn, be add */ parts.total_record_length= parts.record_length; - DBUG_PRINT("info", ("record length: %lu %lu", - (ulong) parts.record_length, - (ulong) parts.total_record_length)); + DBUG_PRINT("info", ("record length: %lu", (ulong) parts.record_length)); /* process this parts */ if (!(rc= (log_record_type_descriptor[type].prewrite_hook && diff --git a/storage/maria/ma_open.c b/storage/maria/ma_open.c index 4c623ac56f3..a4085a27b08 100644 --- a/storage/maria/ma_open.c +++ b/storage/maria/ma_open.c @@ -94,6 +94,7 @@ static MARIA_HA *maria_clone_internal(MARIA_SHARE *share, int mode, int save_errno; uint errpos; MARIA_HA info,*m_info; + my_bitmap_map *changed_fields_bitmap; DBUG_ENTER("maria_clone_internal"); errpos= 0; @@ -120,6 +121,8 @@ static MARIA_HA *maria_clone_internal(MARIA_SHARE *share, int mode, &info.first_mbr_key, share->base.max_key_length, &info.maria_rtree_recursion_state, share->have_rtree ? 1024 : 0, + &changed_fields_bitmap, + bitmap_buffer_size(share->base.fields), NullS)) goto err; errpos= 6; @@ -144,6 +147,8 @@ static MARIA_HA *maria_clone_internal(MARIA_SHARE *share, int mode, info.errkey= -1; info.page_changed=1; info.keyread_buff= info.buff + share->base.max_key_block_length; + bitmap_init(&info.changed_fields, changed_fields_bitmap, + share->base.fields, 0); if ((*share->init)(&info)) goto err; diff --git a/storage/maria/ma_packrec.c b/storage/maria/ma_packrec.c index ae3920dbb3c..173fafaf73f 100644 --- a/storage/maria/ma_packrec.c +++ b/storage/maria/ma_packrec.c @@ -728,8 +728,8 @@ static uint find_longest_bitstream(uint16 *table, uint16 *end) buf RETURN The buffer to receive the record. RETURN - 0 on success - HA_ERR_WRONG_IN_RECORD or -1 on error + 0 On success + # Error number */ int _ma_read_pack_record(MARIA_HA *info, uchar *buf, MARIA_RECORD_POS filepos) @@ -739,7 +739,7 @@ int _ma_read_pack_record(MARIA_HA *info, uchar *buf, MARIA_RECORD_POS filepos) DBUG_ENTER("maria_read_pack_record"); if (filepos == HA_OFFSET_ERROR) - DBUG_RETURN(-1); /* _search() didn't find record */ + DBUG_RETURN(my_errno); /* _search() didn't find record */ file= info->dfile.file; if (_ma_pack_get_block_info(info, &info->bit_buff, &block_info, @@ -755,7 +755,7 @@ int _ma_read_pack_record(MARIA_HA *info, uchar *buf, MARIA_RECORD_POS filepos) panic: my_errno=HA_ERR_WRONG_IN_RECORD; err: - DBUG_RETURN(-1); + DBUG_RETURN(my_errno); } @@ -1598,14 +1598,14 @@ static int _ma_read_mempack_record(MARIA_HA *info, uchar *buf, DBUG_ENTER("maria_read_mempack_record"); if (filepos == HA_OFFSET_ERROR) - DBUG_RETURN(-1); /* _search() didn't find record */ + DBUG_RETURN(my_errno); /* _search() didn't find record */ if (!(pos= (uchar*) _ma_mempack_get_block_info(info, &info->bit_buff, &block_info, &info->rec_buff, &info->rec_buff_size, (uchar*) share->file_map+ filepos))) - DBUG_RETURN(-1); + DBUG_RETURN(my_errno); DBUG_RETURN(_ma_pack_rec_unpack(info, &info->bit_buff, buf, pos, block_info.rec_len)); } diff --git a/storage/maria/ma_recovery.c b/storage/maria/ma_recovery.c index 5810f8e06fe..d7c67d4bd8a 100644 --- a/storage/maria/ma_recovery.c +++ b/storage/maria/ma_recovery.c @@ -1045,17 +1045,11 @@ prototype_undo_exec_hook(UNDO_ROW_UPDATE) info->trn= trn; info->trn->undo_lsn= lsn_korr(rec->header); - /* - For now we skip the page and directory entry. This is to be used - later when we mark rows as deleted. - */ error= _ma_apply_undo_row_update(info, rec->lsn, log_record_buffer.str + LSN_STORE_SIZE + - FILEID_STORE_SIZE + PAGE_STORE_SIZE + - DIRPOS_STORE_SIZE, + FILEID_STORE_SIZE, rec->record_length - - (LSN_STORE_SIZE + FILEID_STORE_SIZE + - PAGE_STORE_SIZE + DIRPOS_STORE_SIZE)); + (LSN_STORE_SIZE + FILEID_STORE_SIZE)); info->trn= 0; return error; } diff --git a/storage/maria/ma_statrec.c b/storage/maria/ma_statrec.c index b04b858c685..ebfab4fad76 100644 --- a/storage/maria/ma_statrec.c +++ b/storage/maria/ma_statrec.c @@ -183,26 +183,25 @@ int _ma_read_static_record(register MARIA_HA *info, register uchar *record, if (info->opt_flag & WRITE_CACHE_USED && info->rec_cache.pos_in_file <= pos && flush_io_cache(&info->rec_cache)) - return(-1); + return(my_errno); info->rec_cache.seek_not_done=1; /* We have done a seek */ error=info->s->file_read(info,(char*) record,info->s->base.reclength, - pos, MYF(MY_NABP)) != 0; - fast_ma_writeinfo(info); + pos, MYF(MY_NABP)); if (! error) { + fast_ma_writeinfo(info); if (!*record) { - my_errno=HA_ERR_RECORD_DELETED; - return(1); /* Record is deleted */ + /* Record is deleted */ + return ((my_errno=HA_ERR_RECORD_DELETED)); } info->update|= HA_STATE_AKTIV; /* Record is read */ return(0); } - return(-1); /* Error on read */ } fast_ma_writeinfo(info); /* No such record */ - return(-1); + return(my_errno); } @@ -264,13 +263,7 @@ int _ma_read_rnd_static_record(MARIA_HA *info, uchar *buf, if (! cache_read) /* No cacheing */ { - if ((error= _ma_read_static_record(info, buf, filepos))) - { - if (error > 0) - error=my_errno=HA_ERR_RECORD_DELETED; - else - error=my_errno; - } + error= _ma_read_static_record(info, buf, filepos); DBUG_RETURN(error); } diff --git a/storage/maria/ma_test1.c b/storage/maria/ma_test1.c index 4435de0bbdb..3209bbd6975 100644 --- a/storage/maria/ma_test1.c +++ b/storage/maria/ma_test1.c @@ -252,6 +252,9 @@ static int run_test(const char *filename) exit(1); } + if (maria_commit(file) || maria_begin(file)) + goto err; + if (!skip_update) { if (opt_unique) @@ -289,7 +292,7 @@ static int run_test(const char *filename) found=0; while ((error= maria_scan(file,read_record)) == 0) { - if (update_count-- == 0) { VOID(maria_close(file)) ; exit(0) ; } + if (--update_count == 0) { VOID(maria_close(file)) ; exit(0) ; } memcpy(record,read_record,rec_length); update_record(record); if (maria_update(file,read_record,record)) @@ -304,6 +307,18 @@ static int run_test(const char *filename) maria_scan_end(file); } + if (die_in_middle_of_transaction == 2) + { + /* + Ensure we get changed pages and log to disk + As commit record is not done, the undo entries needs to be rolled back. + */ + _ma_flush_table_files(file, MARIA_FLUSH_DATA, FLUSH_RELEASE, + FLUSH_RELEASE); + printf("Dying on request after update without maria_close()\n"); + exit(1); + } + if (!silent) printf("- Reopening file\n"); if (maria_commit(file)) @@ -356,7 +371,7 @@ static int run_test(const char *filename) } } - if (die_in_middle_of_transaction == 2) + if (die_in_middle_of_transaction == 3) { /* Ensure we get changed pages and log to disk diff --git a/storage/maria/maria_def.h b/storage/maria/maria_def.h index e8b53757a53..77a3f55e58f 100644 --- a/storage/maria/maria_def.h +++ b/storage/maria/maria_def.h @@ -16,8 +16,9 @@ /* This file is included by all internal maria files */ #include "maria.h" /* Structs & some defines */ -#include "myisampack.h" /* packing of keys */ +#include /* packing of keys */ #include +#include #ifdef THREAD #include #include @@ -437,6 +438,7 @@ struct st_maria_info PAGECACHE_FILE dfile; /* The datafile */ IO_CACHE rec_cache; /* When cacheing records */ LIST open_list; + MY_BITMAP changed_fields; uint opt_flag; /* Optim. for space/speed */ uint update; /* If file changed since open */ int lastinx; /* Last used index */ diff --git a/storage/maria/maria_read_log.c b/storage/maria/maria_read_log.c index 4057fd51e85..d22df34f14c 100644 --- a/storage/maria/maria_read_log.c +++ b/storage/maria/maria_read_log.c @@ -93,7 +93,8 @@ int main(int argc, char **argv) */ fprintf(stdout, "TRACE of the last maria_read_log\n"); - if (maria_apply_log(lsn, opt_display_and_apply, stdout, TRUE)) + if (maria_apply_log(lsn, opt_display_and_apply, stdout, + opt_display_and_apply)) goto err; fprintf(stdout, "%s: SUCCESS\n", my_progname); -- cgit v1.2.1 From 3785d845ed83e359873749c4e433382f9c6d80fa Mon Sep 17 00:00:00 2001 From: unknown Date: Mon, 10 Sep 2007 20:19:27 +0200 Subject: include maria in pushbuild's 'make dist' --- BUILD/compile-dist | 1 + 1 file changed, 1 insertion(+) diff --git a/BUILD/compile-dist b/BUILD/compile-dist index 0ecb386bf71..8065c88e2ed 100755 --- a/BUILD/compile-dist +++ b/BUILD/compile-dist @@ -50,6 +50,7 @@ fi --with-blackhole-storage-engine \ --with-csv-storage-engine \ --with-example-storage-engine \ + --with-maria-storage-engine \ --with-federated-storage-engine \ --with-innodb \ --with-ssl \ -- cgit v1.2.1 From daf62687dc66cd11d47dc9cb3edd29311934b06e Mon Sep 17 00:00:00 2001 From: unknown Date: Mon, 10 Sep 2007 21:14:46 +0200 Subject: fix a typo in #ifdef --- storage/maria/maria_read_log.c | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/storage/maria/maria_read_log.c b/storage/maria/maria_read_log.c index d22df34f14c..e47068f50dd 100644 --- a/storage/maria/maria_read_log.c +++ b/storage/maria/maria_read_log.c @@ -171,8 +171,8 @@ get_one_option(int optid __attribute__((unused)), case '#': DBUG_SET_INITIAL(argument ? argument : default_dbug_option); break; - } #endif + } return 0; } -- cgit v1.2.1 From daa83508cc9f8130d81638744dd143629ac41c8f Mon Sep 17 00:00:00 2001 From: unknown Date: Mon, 10 Sep 2007 23:20:39 +0200 Subject: fix for pushbuild failure, include trnman_public.h in source tarball (make dist) storage/maria/Makefile.am: include trnman_public.h in source tarball --- storage/maria/Makefile.am | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/storage/maria/Makefile.am b/storage/maria/Makefile.am index 6e15b1df056..03d1cc75347 100644 --- a/storage/maria/Makefile.am +++ b/storage/maria/Makefile.am @@ -61,7 +61,7 @@ noinst_HEADERS = maria_def.h ma_rt_index.h ma_rt_key.h ma_rt_mbr.h \ ma_ft_eval.h trnman.h lockman.h tablockman.h \ ma_control_file.h ha_maria.h ma_blockrec.h \ ma_loghandler.h ma_loghandler_lsn.h ma_pagecache.h \ - ma_recovery.h ma_commit.h + ma_recovery.h ma_commit.h trnman_public.h ma_test1_DEPENDENCIES= $(LIBRARIES) ma_test1_LDADD= @CLIENT_EXTRA_LDFLAGS@ libmaria.a \ $(top_builddir)/storage/myisam/libmyisam.a \ -- cgit v1.2.1 From 6aef814d98ee2c8c4f3199e9505a988f7609e3a7 Mon Sep 17 00:00:00 2001 From: unknown Date: Tue, 11 Sep 2007 01:58:15 +0300 Subject: Fixed some bugs when using undo of VARCHAR fields Fixed bug in undo_delete Fixed wrong error output from maria_check include/my_base.h: Added marker if we have null fields in table mysql-test/r/maria.result: checksum in maria now ignore null fields that are null sql/sql_table.cc: Ignore null fields that are now (Before enabling this, we have to change MyISAM to also skip null fields) storage/maria/ma_blockrec.c: More logging After merge fixes Fixed some bugs when using undo of VARCHAR fields Fixed bug in undo_delete (We can't use info->rec_buff here as this is used in write_block_record()) storage/maria/ma_blockrec.h: ma_recordpos_to_dir_entry changed to return uint storage/maria/ma_check.c: Fixed wrong output in case of errors storage/maria/ma_create.c: Set share.base.pack_reclength more correct for block record Delete support for RAID storage/maria/ma_open.c: Don't calculate checksum fields with value NULL storage/maria/ma_test1.c: Fixed output from -v for VARCHAR keys storage/maria/ma_test_recovery.expected: Update results after adding new printf New checksums (because we now ignore nulls) Some file lengths are different, but think they are ok (didn't have time to investigate) storage/myisam/ha_myisam.cc: Fixed comment storage/myisam/mi_test1.c: Fixed bug --- include/my_base.h | 1 + mysql-test/r/maria.result | 4 ++-- sql/sql_table.cc | 4 ++++ storage/maria/ma_blockrec.c | 38 ++++++++++++++++------------- storage/maria/ma_blockrec.h | 4 ++-- storage/maria/ma_check.c | 4 ++-- storage/maria/ma_create.c | 27 +++++++++++---------- storage/maria/ma_open.c | 11 ++++++--- storage/maria/ma_test1.c | 7 +++--- storage/maria/ma_test_recovery.expected | 42 +++++++++++++++++++-------------- storage/myisam/ha_myisam.cc | 3 ++- storage/myisam/mi_test1.c | 7 ++++-- 12 files changed, 89 insertions(+), 63 deletions(-) diff --git a/include/my_base.h b/include/my_base.h index 473ed46ff19..2ab1123ce76 100644 --- a/include/my_base.h +++ b/include/my_base.h @@ -292,6 +292,7 @@ enum ha_base_keytype { #define HA_OPTION_NO_PACK_KEYS 128 /* Reserved for MySQL */ #define HA_OPTION_CREATE_FROM_ENGINE 256 #define HA_OPTION_RELIES_ON_SQL_LAYER 512 +#define HA_OPTION_NULL_FIELDS 1024 #define HA_OPTION_TEMP_COMPRESS_RECORD ((uint) 16384) /* set by isamchk */ #define HA_OPTION_READ_ONLY_DATA ((uint) 32768) /* Set by isamchk */ diff --git a/mysql-test/r/maria.result b/mysql-test/r/maria.result index 68ae0cb267b..710cb5ec094 100644 --- a/mysql-test/r/maria.result +++ b/mysql-test/r/maria.result @@ -588,14 +588,14 @@ insert t1 values (1, "aaa", "bbb"), (NULL, "", "ccccc"), (0, NULL, ""); insert t2 select * from t1; checksum table t1, t2, t3 quick; Table Checksum -test.t1 2948697075 +test.t1 3442722830 test.t2 NULL test.t3 NULL Warnings: Error 1146 Table 'test.t3' doesn't exist checksum table t1, t2, t3; Table Checksum -test.t1 2948697075 +test.t1 3442722830 test.t2 2948697075 test.t3 NULL Warnings: diff --git a/sql/sql_table.cc b/sql/sql_table.cc index 158f415317c..22b1d7701e7 100644 --- a/sql/sql_table.cc +++ b/sql/sql_table.cc @@ -7082,6 +7082,10 @@ bool mysql_checksum_table(THD *thd, TABLE_LIST *tables, for (uint i= 0; i < t->s->fields; i++ ) { Field *f= t->field[i]; +#ifdef NOT_YET + if (f->is_real_null(0)) + continue; +#endif if ((f->type() == MYSQL_TYPE_BLOB) || (f->type() == MYSQL_TYPE_VARCHAR)) { diff --git a/storage/maria/ma_blockrec.c b/storage/maria/ma_blockrec.c index 40fa5364a44..394ff7bc6ae 100644 --- a/storage/maria/ma_blockrec.c +++ b/storage/maria/ma_blockrec.c @@ -1937,6 +1937,7 @@ static my_bool write_block_record(MARIA_HA *info, /* Update page directory */ uint length= (uint) (data - row_pos->data); DBUG_PRINT("info", ("Used head length on page: %u", length)); + DBUG_ASSERT(data <= end_of_data); if (length < info->s->base.min_row_length) { uint diff_length= info->s->base.min_row_length - length; @@ -2517,7 +2518,9 @@ static my_bool allocate_and_write_block_record(MARIA_HA *info, blocks, blocks->block->org_bitmap_value != 0, &row_pos, undo_lsn)) DBUG_RETURN(1); /* Error reading bitmap */ - DBUG_PRINT("exit", ("Rowid: %lu", (ulong) row->lastpos)); + DBUG_PRINT("exit", ("Rowid: %lu (%lu:%u)", (ulong) row->lastpos, + (ulong) ma_recordpos_to_page(row->lastpos), + ma_recordpos_to_dir_entry(row->lastpos))); DBUG_RETURN(0); } @@ -2790,7 +2793,8 @@ err: my_bool _ma_update_block_record(MARIA_HA *info, MARIA_RECORD_POS record_pos, const uchar *orig_rec, const uchar *new_rec) { - return _ma_update_block_record2(info, record_pos, orig_rec, new_rec, 0); + return _ma_update_block_record2(info, record_pos, orig_rec, new_rec, + LSN_ERROR); } @@ -3041,6 +3045,8 @@ my_bool _ma_delete_block_record(MARIA_HA *info, const uchar *record) page= ma_recordpos_to_page(info->cur_row.lastpos); record_number= ma_recordpos_to_dir_entry(info->cur_row.lastpos); + DBUG_PRINT("enter", ("Rowid: %lu (%lu:%u)", (ulong) info->cur_row.lastpos, + (ulong) page, record_number)); if (delete_head_or_tail(info, page, record_number, 1, 0) || delete_tails(info, info->cur_row.tail_positions)) @@ -4309,16 +4315,12 @@ static size_t fill_insert_undo_parts(MARIA_HA *info, const uchar *record, } case FIELD_VARCHAR: { - if (column->length <= 256) - { + if (column->fill_length == 1) column_length= *field_lengths; - field_lengths++; - } else - { column_length= uint2korr(field_lengths); - field_lengths+= 2; - } + field_lengths+= column->fill_length; + column_pos+= column->fill_length; break; } default: @@ -4971,6 +4973,7 @@ my_bool _ma_apply_undo_row_delete(MARIA_HA *info, LSN undo_lsn, uint *null_field_lengths; ulong *blob_lengths; MARIA_COLUMNDEF *column, *end_column; + my_bool res; DBUG_ENTER("_ma_apply_undo_row_delete"); /* @@ -5003,11 +5006,9 @@ my_bool _ma_apply_undo_row_delete(MARIA_HA *info, LSN undo_lsn, row.blob_length= ma_get_length((uchar**) &header); /* We need to build up a record (without blobs) in rec_buff */ - if (_ma_alloc_buffer(&info->rec_buff, &info->rec_buff_size, - length - row.blob_length)) + if (!(record= my_malloc(share->base.reclength, MYF(MY_WME)))) DBUG_RETURN(1); - record= info->rec_buff; memcpy(record, null_bits, share->base.null_bytes); /* Copy field information from header to record */ @@ -5073,21 +5074,22 @@ my_bool _ma_apply_undo_row_delete(MARIA_HA *info, LSN undo_lsn, uchar *field_pos= record + column->offset; /* 256 is correct as this includes the length uchar */ - if (column->length <= 256) + if (column->fill_length == 1) { field_pos[0]= *field_length_data; - length= (uint) *field_length_data++; + length= (uint) *field_length_data; } else { field_pos[0]= field_length_data[0]; field_pos[1]= field_length_data[1]; length= uint2korr(field_length_data); - field_length_data+= 2; } + field_length_data+= column->fill_length; + field_pos+= column->fill_length; row.varchar_length+= length; *null_field_lengths= length; - memcpy(record + column->offset, header, length); + memcpy(field_pos, header, length); header+= length; break; } @@ -5122,7 +5124,9 @@ my_bool _ma_apply_undo_row_delete(MARIA_HA *info, LSN undo_lsn, /* Row is now up to date. Time to insert the record */ - DBUG_RETURN(allocate_and_write_block_record(info, record, &row, undo_lsn)); + res= allocate_and_write_block_record(info, record, &row, undo_lsn); + my_free(record, MYF(0)); + DBUG_RETURN(res); } diff --git a/storage/maria/ma_blockrec.h b/storage/maria/ma_blockrec.h index cdaf8b9d124..30dffe1c0c0 100644 --- a/storage/maria/ma_blockrec.h +++ b/storage/maria/ma_blockrec.h @@ -118,9 +118,9 @@ static inline my_off_t ma_recordpos_to_page(MARIA_RECORD_POS record_pos) return record_pos >> 8; } -static inline my_off_t ma_recordpos_to_dir_entry(MARIA_RECORD_POS record_pos) +static inline uint ma_recordpos_to_dir_entry(MARIA_RECORD_POS record_pos) { - return record_pos & 255; + return (uint) (record_pos & 255); } /* ma_blockrec.c */ diff --git a/storage/maria/ma_check.c b/storage/maria/ma_check.c index 93d4e2aa492..354af51fbfd 100644 --- a/storage/maria/ma_check.c +++ b/storage/maria/ma_check.c @@ -3836,7 +3836,7 @@ static int sort_get_next_record(MARIA_SORT_PARAM *sort_param) if (param->testflag & T_VERBOSE) { char llbuff[22]; - record_pos_to_txt(info, sort_param->filepos, llbuff); + record_pos_to_txt(info, info->cur_row.lastpos, llbuff); _ma_check_print_info(param, "Found record with wrong checksum at %s", llbuff); @@ -3846,7 +3846,7 @@ static int sort_get_next_record(MARIA_SORT_PARAM *sort_param) info->cur_row.checksum= checksum; param->glob_crc+= checksum; } - sort_param->filepos= info->cur_row.lastpos; + sort_param->start_recpos= sort_param->filepos= info->cur_row.lastpos; DBUG_RETURN(0); } if (flag == HA_ERR_END_OF_FILE) diff --git a/storage/maria/ma_create.c b/storage/maria/ma_create.c index 7f26a7777c0..b31460d24d2 100644 --- a/storage/maria/ma_create.c +++ b/storage/maria/ma_create.c @@ -131,6 +131,8 @@ int maria_create(const char *name, enum data_file_type datafile_type, column->empty_pos= 0; column->empty_bit= 0; column->fill_length= column->length; + if (column->null_bit) + options|= HA_OPTION_NULL_FIELDS; reclength+= column->length; type= column->type; @@ -664,14 +666,6 @@ int maria_create(const char *name, enum data_file_type datafile_type, share.base.keystart = share.state.state.key_file_length= MY_ALIGN(info_length, maria_block_size); - if (share.data_file_type == BLOCK_RECORD) - { - /* - we are going to create a first bitmap page, set data_file_length - to reflect this, before the state goes to disk - */ - share.state.state.data_file_length= maria_block_size; - } share.base.max_key_block_length= maria_block_size; share.base.max_key_length=ALIGN_SIZE(max_key_length+4); share.base.records=ci->max_rows; @@ -683,11 +677,18 @@ int maria_create(const char *name, enum data_file_type datafile_type, share.base.pack_bytes= pack_bytes; share.base.fields= columns; share.base.pack_fields= packed; -#ifdef USE_RAID - share.base.raid_type=ci->raid_type; - share.base.raid_chunks=ci->raid_chunks; - share.base.raid_chunksize=ci->raid_chunksize; -#endif + + if (share.data_file_type == BLOCK_RECORD) + { + /* + we are going to create a first bitmap page, set data_file_length + to reflect this, before the state goes to disk + */ + share.state.state.data_file_length= maria_block_size; + /* Add length of packed fields + length */ + share.base.pack_reclength+= share.base.max_field_lengths+3; + + } /* max_data_file_length and max_key_file_length are recalculated on open */ if (tmp_table) diff --git a/storage/maria/ma_open.c b/storage/maria/ma_open.c index f0962404ac4..cd57f6f0b11 100644 --- a/storage/maria/ma_open.c +++ b/storage/maria/ma_open.c @@ -314,7 +314,7 @@ MARIA_HA *maria_open(const char *name, int mode, uint open_flags) HA_OPTION_COMPRESS_RECORD | HA_OPTION_READ_ONLY_DATA | HA_OPTION_TEMP_COMPRESS_RECORD | HA_OPTION_CHECKSUM | HA_OPTION_TMP_TABLE | HA_OPTION_DELAY_KEY_WRITE | - HA_OPTION_RELIES_ON_SQL_LAYER)) + HA_OPTION_RELIES_ON_SQL_LAYER | HA_OPTION_NULL_FIELDS)) { DBUG_PRINT("error",("wrong options: 0x%lx", share->options)); my_errno=HA_ERR_OLD_FILE; @@ -847,7 +847,8 @@ void _ma_setup_functions(register MARIA_SHARE *share) Calculate checksum according to data in the original, not compressed, row. */ - if (share->state.header.org_data_file_type == STATIC_RECORD) + if (share->state.header.org_data_file_type == STATIC_RECORD && + ! (share->options & HA_OPTION_NULL_FIELDS)) share->calc_checksum= _ma_static_checksum; else share->calc_checksum= _ma_checksum; @@ -881,7 +882,11 @@ void _ma_setup_functions(register MARIA_SHARE *share) share->update_record= _ma_update_static_record; share->write_record= _ma_write_static_record; share->compare_unique= _ma_cmp_static_unique; - share->calc_checksum= share->calc_write_checksum= _ma_static_checksum; + if (share->state.header.org_data_file_type == STATIC_RECORD && + ! (share->options & HA_OPTION_NULL_FIELDS)) + share->calc_checksum= _ma_static_checksum; + else + share->calc_checksum= _ma_checksum; break; case BLOCK_RECORD: share->once_init= _ma_once_init_block_record; diff --git a/storage/maria/ma_test1.c b/storage/maria/ma_test1.c index 45dffd279a0..e7cde8a7e28 100644 --- a/storage/maria/ma_test1.c +++ b/storage/maria/ma_test1.c @@ -119,6 +119,8 @@ static int run_test(const char *filename) recinfo[1].length= (extra_field == FIELD_BLOB ? 4 + portable_sizeof_char_ptr : 24); if (extra_field == FIELD_VARCHAR) recinfo[1].length+= HA_VARCHAR_PACKLENGTH(recinfo[1].length); + recinfo[1].null_bit= null_fields ? 2 : 0; + if (opt_unique) { recinfo[2].type=FIELD_CHECK; @@ -186,7 +188,7 @@ static int run_test(const char *filename) uniques=0; offset_to_key= test(null_fields); - if (key_field == FIELD_BLOB) + if (key_field == FIELD_BLOB || key_field == FIELD_VARCHAR) offset_to_key+= 2; if (!silent) @@ -338,8 +340,7 @@ static int run_test(const char *filename) { fprintf(stderr, "delete-rows number of rows deleted; Going down hard!\n"); - VOID(maria_close(file)); - exit(0) ; + goto end; } j=i*2; if (!flags[j]) diff --git a/storage/maria/ma_test_recovery.expected b/storage/maria/ma_test_recovery.expected index 3ded68e7f56..55fe04cffdb 100644 --- a/storage/maria/ma_test_recovery.expected +++ b/storage/maria/ma_test_recovery.expected @@ -126,6 +126,7 @@ Differences in maria_chk -dvv, recovery not yet perfect ! > 1 2 6 unique number NULL 0 8192 ========DIFF END======= TEST WITH ma_test1 -s -M -T -c -N --debug=d:t:i:o,/tmp/ma_test1.trace --testflag=3 (commit at end) +Terminating after update TEST WITH ma_test1 -s -M -T -c -N --debug=d:t:i:o,/tmp/ma_test1.trace --testflag=4 --test-undo=1 (additional aborted work) terminating after deletes Dying on request without maria_commit()/maria_close() @@ -137,7 +138,7 @@ applying log Differences in maria_chk -dvv, recovery not yet perfect ! ========DIFF START======= 7c7 -< Checksum: 3536469224 +< Checksum: 3697324514 --- > Checksum: 0 11c11 @@ -204,6 +205,7 @@ Differences in maria_chk -dvv, recovery not yet perfect ! > 1 2 6 unique number NULL 0 8192 ========DIFF END======= TEST WITH ma_test1 -s -M -T -c -N --debug=d:t:i:o,/tmp/ma_test1.trace --testflag=3 (commit at end) +Terminating after update TEST WITH ma_test1 -s -M -T -c -N --debug=d:t:i:o,/tmp/ma_test1.trace --testflag=4 --test-undo=2 (additional aborted work) terminating after deletes Dying on request without maria_commit()/maria_close() @@ -215,7 +217,7 @@ applying log Differences in maria_chk -dvv, recovery not yet perfect ! ========DIFF START======= 7c7 -< Checksum: 3536469224 +< Checksum: 3697324514 --- > Checksum: 0 11c11 @@ -282,6 +284,7 @@ Differences in maria_chk -dvv, recovery not yet perfect ! > 1 2 6 unique number NULL 0 8192 ========DIFF END======= TEST WITH ma_test1 -s -M -T -c -N --debug=d:t:i:o,/tmp/ma_test1.trace --testflag=3 (commit at end) +Terminating after update TEST WITH ma_test1 -s -M -T -c -N --debug=d:t:i:o,/tmp/ma_test1.trace --testflag=4 --test-undo=3 (additional aborted work) terminating after deletes Dying on request without maria_commit()/maria_close() @@ -293,7 +296,7 @@ applying log Differences in maria_chk -dvv, recovery not yet perfect ! ========DIFF START======= 7c7 -< Checksum: 3536469224 +< Checksum: 3697324514 --- > Checksum: 0 11c11 @@ -319,7 +322,7 @@ Differences in maria_chk -dvv, recovery not yet perfect ! 11c11 < Datafile length: 49152 Keyfile length: 16384 --- -> Datafile length: 57344 Keyfile length: 8192 +> Datafile length: 49152 Keyfile length: 8192 18c18 < 1 2 6 unique varchar BLOB NULL 0 8192 8192 --- @@ -336,7 +339,7 @@ Differences in maria_chk -dvv, recovery not yet perfect ! 11c11 < Datafile length: 49152 Keyfile length: 16384 --- -> Datafile length: 57344 Keyfile length: 8192 +> Datafile length: 49152 Keyfile length: 8192 18c18 < 1 2 6 unique varchar BLOB NULL 0 8192 8192 --- @@ -353,13 +356,14 @@ Differences in maria_chk -dvv, recovery not yet perfect ! 11c11 < Datafile length: 49152 Keyfile length: 16384 --- -> Datafile length: 57344 Keyfile length: 8192 +> Datafile length: 49152 Keyfile length: 8192 18c18 < 1 2 6 unique varchar BLOB NULL 0 8192 8192 --- > 1 2 6 unique varchar BLOB NULL 0 8192 ========DIFF END======= TEST WITH ma_test1 -s -M -T -c -N --debug=d:t:i:o,/tmp/ma_test1.trace -b --testflag=3 (commit at end) +Terminating after update TEST WITH ma_test1 -s -M -T -c -N --debug=d:t:i:o,/tmp/ma_test1.trace -b --testflag=4 --test-undo=1 (additional aborted work) terminating after deletes Dying on request without maria_commit()/maria_close() @@ -371,13 +375,13 @@ applying log Differences in maria_chk -dvv, recovery not yet perfect ! ========DIFF START======= 7c7 -< Checksum: 1984748106 +< Checksum: 4024695312 --- > Checksum: 0 11c11 -< Datafile length: 57344 Keyfile length: 16384 +< Datafile length: 49152 Keyfile length: 16384 --- -> Datafile length: 57344 Keyfile length: 8192 +> Datafile length: 49152 Keyfile length: 8192 18c18 < 1 2 6 unique varchar BLOB NULL 0 8192 8192 --- @@ -397,7 +401,7 @@ Differences in maria_chk -dvv, recovery not yet perfect ! 11c11 < Datafile length: 49152 Keyfile length: 16384 --- -> Datafile length: 57344 Keyfile length: 8192 +> Datafile length: 49152 Keyfile length: 8192 18c18 < 1 2 6 unique varchar BLOB NULL 0 8192 8192 --- @@ -414,7 +418,7 @@ Differences in maria_chk -dvv, recovery not yet perfect ! 11c11 < Datafile length: 49152 Keyfile length: 16384 --- -> Datafile length: 57344 Keyfile length: 8192 +> Datafile length: 49152 Keyfile length: 8192 18c18 < 1 2 6 unique varchar BLOB NULL 0 8192 8192 --- @@ -431,13 +435,14 @@ Differences in maria_chk -dvv, recovery not yet perfect ! 11c11 < Datafile length: 49152 Keyfile length: 16384 --- -> Datafile length: 57344 Keyfile length: 8192 +> Datafile length: 49152 Keyfile length: 8192 18c18 < 1 2 6 unique varchar BLOB NULL 0 8192 8192 --- > 1 2 6 unique varchar BLOB NULL 0 8192 ========DIFF END======= TEST WITH ma_test1 -s -M -T -c -N --debug=d:t:i:o,/tmp/ma_test1.trace -b --testflag=3 (commit at end) +Terminating after update TEST WITH ma_test1 -s -M -T -c -N --debug=d:t:i:o,/tmp/ma_test1.trace -b --testflag=4 --test-undo=2 (additional aborted work) terminating after deletes Dying on request without maria_commit()/maria_close() @@ -449,13 +454,13 @@ applying log Differences in maria_chk -dvv, recovery not yet perfect ! ========DIFF START======= 7c7 -< Checksum: 1984748106 +< Checksum: 4024695312 --- > Checksum: 0 11c11 -< Datafile length: 57344 Keyfile length: 16384 +< Datafile length: 49152 Keyfile length: 16384 --- -> Datafile length: 57344 Keyfile length: 8192 +> Datafile length: 49152 Keyfile length: 8192 18c18 < 1 2 6 unique varchar BLOB NULL 0 8192 8192 --- @@ -516,6 +521,7 @@ Differences in maria_chk -dvv, recovery not yet perfect ! > 1 2 6 unique varchar BLOB NULL 0 8192 ========DIFF END======= TEST WITH ma_test1 -s -M -T -c -N --debug=d:t:i:o,/tmp/ma_test1.trace -b --testflag=3 (commit at end) +Terminating after update TEST WITH ma_test1 -s -M -T -c -N --debug=d:t:i:o,/tmp/ma_test1.trace -b --testflag=4 --test-undo=3 (additional aborted work) terminating after deletes Dying on request without maria_commit()/maria_close() @@ -527,13 +533,13 @@ applying log Differences in maria_chk -dvv, recovery not yet perfect ! ========DIFF START======= 7c7 -< Checksum: 1984748106 +< Checksum: 4024695312 --- > Checksum: 0 11c11 -< Datafile length: 57344 Keyfile length: 16384 +< Datafile length: 49152 Keyfile length: 16384 --- -> Datafile length: 57344 Keyfile length: 8192 +> Datafile length: 49152 Keyfile length: 8192 18c18 < 1 2 6 unique varchar BLOB NULL 0 8192 8192 --- diff --git a/storage/myisam/ha_myisam.cc b/storage/myisam/ha_myisam.cc index 4f02705c18c..a9a9f44049f 100644 --- a/storage/myisam/ha_myisam.cc +++ b/storage/myisam/ha_myisam.cc @@ -252,7 +252,8 @@ int table2myisam(TABLE *table_arg, MI_KEYDEF **keydef_out, DBUG_PRINT("loop", ("found: 0x%lx recpos: %d minpos: %d length: %d", (long) found, recpos, minpos, length)); if (recpos != minpos) - { // Reserved space (Null bits?) + { + /* reserve space for null bits */ bzero((char*) recinfo_pos, sizeof(*recinfo_pos)); recinfo_pos->type= (int) FIELD_NORMAL; recinfo_pos++->length= (uint16) (minpos - recpos); diff --git a/storage/myisam/mi_test1.c b/storage/myisam/mi_test1.c index 1ceedf7f86a..4137b33fe42 100644 --- a/storage/myisam/mi_test1.c +++ b/storage/myisam/mi_test1.c @@ -79,6 +79,8 @@ static int run_test(const char *filename) recinfo[2].length= (extra_field == FIELD_BLOB ? 4 + portable_sizeof_char_ptr : 24); if (extra_field == FIELD_VARCHAR) recinfo[2].length+= HA_VARCHAR_PACKLENGTH(recinfo[2].length); + recinfo[1].null_bit= null_fields ? 2 : 0; + if (opt_unique) { recinfo[3].type=FIELD_CHECK; @@ -258,7 +260,8 @@ static int run_test(const char *filename) continue; create_key(key,j); my_errno=0; - if ((error = mi_rkey(file,read_record,0,key,0,HA_READ_KEY_EXACT))) + if ((error = mi_rkey(file,read_record,0,key,HA_WHOLE_KEY, + HA_READ_KEY_EXACT))) { if (verbose || (flags[j] >= 1 || (error && my_errno != HA_ERR_KEY_NOT_FOUND))) @@ -285,7 +288,7 @@ static int run_test(const char *filename) { create_key(key,i); my_errno=0; - error=mi_rkey(file,read_record,0,key,0,HA_READ_KEY_EXACT); + error=mi_rkey(file,read_record,0,key,HA_WHOLE_KEY,HA_READ_KEY_EXACT); if (verbose || (error == 0 && flags[i] == 0 && unique_key) || (error && (flags[i] != 0 || my_errno != HA_ERR_KEY_NOT_FOUND))) -- cgit v1.2.1 From 48189ca420930b0a3fdd1a66adf4725e9349d2c5 Mon Sep 17 00:00:00 2001 From: unknown Date: Tue, 11 Sep 2007 09:37:17 +0300 Subject: Absence of test_file.h fixed. --- storage/maria/unittest/Makefile.am | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/storage/maria/unittest/Makefile.am b/storage/maria/unittest/Makefile.am index 73d903294ce..4631b436b0b 100644 --- a/storage/maria/unittest/Makefile.am +++ b/storage/maria/unittest/Makefile.am @@ -59,8 +59,8 @@ ma_test_loghandler_first_lsn_t_SOURCES = ma_test_loghandler_first_lsn-t.c ma_mar ma_test_loghandler_max_lsn_t_SOURCES = ma_test_loghandler_max_lsn-t.c ma_maria_log_cleanup.c ma_test_loghandler_purge_t_SOURCES = ma_test_loghandler_purge-t.c ma_maria_log_cleanup.c -ma_pagecache_single_src = ma_pagecache_single.c test_file.c -ma_pagecache_consist_src = ma_pagecache_consist.c test_file.c +ma_pagecache_single_src = ma_pagecache_single.c test_file.c test_file.h +ma_pagecache_consist_src = ma_pagecache_consist.c test_file.c test_file.h ma_pagecache_common_cppflags = -DEXTRA_DEBUG -DPAGECACHE_DEBUG -DMAIN ma_pagecache_single_1k_t_SOURCES = $(ma_pagecache_single_src) -- cgit v1.2.1 From 0b2ba820c3f97f83f53571548481aca4a112b3b0 Mon Sep 17 00:00:00 2001 From: unknown Date: Tue, 11 Sep 2007 11:11:22 +0200 Subject: WL#3072 Maria recovery * testing of execution of UNDO_ROW_UPDATE * when executing an UNDO_ROW_UPDATE, store "UNDO_ROW_UPDATE" as "type of undone record" into the CLR_END record. storage/maria/ma_blockrec.c: When logging a CLR_END in write_block_record(), it can be for a DELETE or for an UPDATE (now that Monty has coded execution of UNDO_UPDATE) storage/maria/ma_loghandler.c: UNDO_ROW_UPDATE's execution coded, so no crash storage/maria/ma_recovery.c: UNDO_ROW_UPDATE's execution now coded, so no crash storage/maria/ma_test1.c: upper case letter storage/maria/ma_test_recovery.expected: output of testing execution of UNDO_ROW_UPDATE. Table's checksum not recovered (known issue not specific to UPDATE). storage/maria/ma_test_recovery: Test execution of UNDO_ROW_UPDATE: first we stop ma_test1 after deletes and commit, then we stop ma_test1 after updates and abort; we verify that updates are rolled back by comparing tables --- storage/maria/ma_blockrec.c | 4 +- storage/maria/ma_loghandler.c | 2 + storage/maria/ma_recovery.c | 2 + storage/maria/ma_test1.c | 6 +- storage/maria/ma_test_recovery | 2 +- storage/maria/ma_test_recovery.expected | 366 ++++++++++++++++++++++++++++++-- 6 files changed, 358 insertions(+), 24 deletions(-) diff --git a/storage/maria/ma_blockrec.c b/storage/maria/ma_blockrec.c index 394ff7bc6ae..76104d00673 100644 --- a/storage/maria/ma_blockrec.c +++ b/storage/maria/ma_blockrec.c @@ -2347,8 +2347,8 @@ static my_bool write_block_record(MARIA_HA *info, in the first/second case, Recovery, when it sees the CLR_END in the REDO phase, may decrement/increment the records' count. */ - /** @todo when Monty has UNDO_UPDATE coded, revisit this */ - log_data[LSN_STORE_SIZE + FILEID_STORE_SIZE]= LOGREC_UNDO_ROW_DELETE; + log_data[LSN_STORE_SIZE + FILEID_STORE_SIZE]= old_record ? + LOGREC_UNDO_ROW_UPDATE : LOGREC_UNDO_ROW_DELETE; log_array[TRANSLOG_INTERNAL_PARTS + 0].str= (char*) log_data; log_array[TRANSLOG_INTERNAL_PARTS + 0].length= sizeof(log_data); diff --git a/storage/maria/ma_loghandler.c b/storage/maria/ma_loghandler.c index b9233666dbc..55635bdab93 100644 --- a/storage/maria/ma_loghandler.c +++ b/storage/maria/ma_loghandler.c @@ -6434,6 +6434,8 @@ static my_bool write_hook_for_clr_end(enum translog_record_type type case LOGREC_UNDO_ROW_INSERT: tbl_info->s->state.state.records--; break; + case LOGREC_UNDO_ROW_UPDATE: + break; default: DBUG_ASSERT(0); } diff --git a/storage/maria/ma_recovery.c b/storage/maria/ma_recovery.c index c1b877411a9..baee227e7ca 100644 --- a/storage/maria/ma_recovery.c +++ b/storage/maria/ma_recovery.c @@ -979,6 +979,8 @@ prototype_redo_exec_hook(CLR_END) case LOGREC_UNDO_ROW_INSERT: info->s->state.state.records--; break; + case LOGREC_UNDO_ROW_UPDATE: + break; default: DBUG_ASSERT(0); } diff --git a/storage/maria/ma_test1.c b/storage/maria/ma_test1.c index e7cde8a7e28..80bd3c348a7 100644 --- a/storage/maria/ma_test1.c +++ b/storage/maria/ma_test1.c @@ -248,7 +248,7 @@ static int run_test(const char *filename) if (testflag == 2) { - printf("terminating after inserts\n"); + printf("Terminating after inserts\n"); goto end; } @@ -309,7 +309,7 @@ static int run_test(const char *filename) if (testflag == 3) { - printf("Terminating after update\n"); + printf("Terminating after updates\n"); goto end; } if (!silent) @@ -372,7 +372,7 @@ static int run_test(const char *filename) if (testflag == 4) { - printf("terminating after deletes\n"); + printf("Terminating after deletes\n"); goto end; } diff --git a/storage/maria/ma_test_recovery b/storage/maria/ma_test_recovery index a88814ade7f..7067d79a49d 100755 --- a/storage/maria/ma_test_recovery +++ b/storage/maria/ma_test_recovery @@ -131,7 +131,7 @@ do for test_undo in 1 2 3 do # first iteration tests rollback of insert, second tests rollback of delete - set -- "ma_test1 $silent -M -T -c -N $blobs" "--testflag=1" "--testflag=2" "ma_test1 $silent -M -T -c -N --debug=d:t:i:o,/tmp/ma_test1.trace $blobs" "--testflag=3" "--testflag=4" + set -- "ma_test1 $silent -M -T -c -N $blobs" "--testflag=1" "--testflag=2" "ma_test1 $silent -M -T -c -N --debug=d:t:i:o,/tmp/ma_test1.trace $blobs" "--testflag=3" "--testflag=4" "ma_test1 $silent -M -T -c -N --debug=d:t:i:o,/tmp/ma_test1.trace $blobs" "--testflag=2" "--testflag=3" # -N (create NULL fields) is needed because --test-undo adds it anyway while [ $# != 0 ] do diff --git a/storage/maria/ma_test_recovery.expected b/storage/maria/ma_test_recovery.expected index 55fe04cffdb..87bc32e3a70 100644 --- a/storage/maria/ma_test_recovery.expected +++ b/storage/maria/ma_test_recovery.expected @@ -73,7 +73,7 @@ Differences in maria_chk -dvv, recovery not yet perfect ! Testing the REDO AND UNDO PHASE TEST WITH ma_test1 -s -M -T -c -N --testflag=1 (commit at end) TEST WITH ma_test1 -s -M -T -c -N --testflag=2 --test-undo=1 (additional aborted work) -terminating after inserts +Terminating after inserts Dying on request without maria_commit()/maria_close() applying log Differences in maria_chk -dvv, recovery not yet perfect ! @@ -126,9 +126,9 @@ Differences in maria_chk -dvv, recovery not yet perfect ! > 1 2 6 unique number NULL 0 8192 ========DIFF END======= TEST WITH ma_test1 -s -M -T -c -N --debug=d:t:i:o,/tmp/ma_test1.trace --testflag=3 (commit at end) -Terminating after update +Terminating after updates TEST WITH ma_test1 -s -M -T -c -N --debug=d:t:i:o,/tmp/ma_test1.trace --testflag=4 --test-undo=1 (additional aborted work) -terminating after deletes +Terminating after deletes Dying on request without maria_commit()/maria_close() applying log testing idempotency @@ -150,9 +150,64 @@ Differences in maria_chk -dvv, recovery not yet perfect ! --- > 1 2 6 unique number NULL 0 8192 ========DIFF END======= +TEST WITH ma_test1 -s -M -T -c -N --debug=d:t:i:o,/tmp/ma_test1.trace --testflag=2 (commit at end) +Terminating after inserts +TEST WITH ma_test1 -s -M -T -c -N --debug=d:t:i:o,/tmp/ma_test1.trace --testflag=3 --test-undo=1 (additional aborted work) +Terminating after updates +Dying on request without maria_commit()/maria_close() +applying log +Differences in maria_chk -dvv, recovery not yet perfect ! +========DIFF START======= +7c7 +< Checksum: 2428948025 +--- +> Checksum: 3026590807 +11c11 +< Datafile length: 16384 Keyfile length: 16384 +--- +> Datafile length: 16384 Keyfile length: 8192 +18c18 +< 1 2 6 unique number NULL 0 8192 8192 +--- +> 1 2 6 unique number NULL 0 8192 +========DIFF END======= +testing idempotency +applying log +Differences in maria_chk -dvv, recovery not yet perfect ! +========DIFF START======= +7c7 +< Checksum: 2428948025 +--- +> Checksum: 3026590807 +11c11 +< Datafile length: 16384 Keyfile length: 16384 +--- +> Datafile length: 16384 Keyfile length: 8192 +18c18 +< 1 2 6 unique number NULL 0 8192 8192 +--- +> 1 2 6 unique number NULL 0 8192 +========DIFF END======= +testing applying of CLRs to recreate table +applying log +Differences in maria_chk -dvv, recovery not yet perfect ! +========DIFF START======= +7c7 +< Checksum: 2428948025 +--- +> Checksum: 0 +11c11 +< Datafile length: 16384 Keyfile length: 16384 +--- +> Datafile length: 16384 Keyfile length: 8192 +18c18 +< 1 2 6 unique number NULL 0 8192 8192 +--- +> 1 2 6 unique number NULL 0 8192 +========DIFF END======= TEST WITH ma_test1 -s -M -T -c -N --testflag=1 (commit at end) TEST WITH ma_test1 -s -M -T -c -N --testflag=2 --test-undo=2 (additional aborted work) -terminating after inserts +Terminating after inserts Dying on request without maria_commit()/maria_close() applying log Differences in maria_chk -dvv, recovery not yet perfect ! @@ -205,9 +260,9 @@ Differences in maria_chk -dvv, recovery not yet perfect ! > 1 2 6 unique number NULL 0 8192 ========DIFF END======= TEST WITH ma_test1 -s -M -T -c -N --debug=d:t:i:o,/tmp/ma_test1.trace --testflag=3 (commit at end) -Terminating after update +Terminating after updates TEST WITH ma_test1 -s -M -T -c -N --debug=d:t:i:o,/tmp/ma_test1.trace --testflag=4 --test-undo=2 (additional aborted work) -terminating after deletes +Terminating after deletes Dying on request without maria_commit()/maria_close() applying log testing idempotency @@ -229,9 +284,64 @@ Differences in maria_chk -dvv, recovery not yet perfect ! --- > 1 2 6 unique number NULL 0 8192 ========DIFF END======= +TEST WITH ma_test1 -s -M -T -c -N --debug=d:t:i:o,/tmp/ma_test1.trace --testflag=2 (commit at end) +Terminating after inserts +TEST WITH ma_test1 -s -M -T -c -N --debug=d:t:i:o,/tmp/ma_test1.trace --testflag=3 --test-undo=2 (additional aborted work) +Terminating after updates +Dying on request without maria_commit()/maria_close() +applying log +Differences in maria_chk -dvv, recovery not yet perfect ! +========DIFF START======= +7c7 +< Checksum: 2428948025 +--- +> Checksum: 3026590807 +11c11 +< Datafile length: 16384 Keyfile length: 16384 +--- +> Datafile length: 16384 Keyfile length: 8192 +18c18 +< 1 2 6 unique number NULL 0 8192 8192 +--- +> 1 2 6 unique number NULL 0 8192 +========DIFF END======= +testing idempotency +applying log +Differences in maria_chk -dvv, recovery not yet perfect ! +========DIFF START======= +7c7 +< Checksum: 2428948025 +--- +> Checksum: 3026590807 +11c11 +< Datafile length: 16384 Keyfile length: 16384 +--- +> Datafile length: 16384 Keyfile length: 8192 +18c18 +< 1 2 6 unique number NULL 0 8192 8192 +--- +> 1 2 6 unique number NULL 0 8192 +========DIFF END======= +testing applying of CLRs to recreate table +applying log +Differences in maria_chk -dvv, recovery not yet perfect ! +========DIFF START======= +7c7 +< Checksum: 2428948025 +--- +> Checksum: 0 +11c11 +< Datafile length: 16384 Keyfile length: 16384 +--- +> Datafile length: 16384 Keyfile length: 8192 +18c18 +< 1 2 6 unique number NULL 0 8192 8192 +--- +> 1 2 6 unique number NULL 0 8192 +========DIFF END======= TEST WITH ma_test1 -s -M -T -c -N --testflag=1 (commit at end) TEST WITH ma_test1 -s -M -T -c -N --testflag=2 --test-undo=3 (additional aborted work) -terminating after inserts +Terminating after inserts Dying on request without maria_commit()/maria_close() applying log Differences in maria_chk -dvv, recovery not yet perfect ! @@ -284,9 +394,9 @@ Differences in maria_chk -dvv, recovery not yet perfect ! > 1 2 6 unique number NULL 0 8192 ========DIFF END======= TEST WITH ma_test1 -s -M -T -c -N --debug=d:t:i:o,/tmp/ma_test1.trace --testflag=3 (commit at end) -Terminating after update +Terminating after updates TEST WITH ma_test1 -s -M -T -c -N --debug=d:t:i:o,/tmp/ma_test1.trace --testflag=4 --test-undo=3 (additional aborted work) -terminating after deletes +Terminating after deletes Dying on request without maria_commit()/maria_close() applying log testing idempotency @@ -308,9 +418,64 @@ Differences in maria_chk -dvv, recovery not yet perfect ! --- > 1 2 6 unique number NULL 0 8192 ========DIFF END======= +TEST WITH ma_test1 -s -M -T -c -N --debug=d:t:i:o,/tmp/ma_test1.trace --testflag=2 (commit at end) +Terminating after inserts +TEST WITH ma_test1 -s -M -T -c -N --debug=d:t:i:o,/tmp/ma_test1.trace --testflag=3 --test-undo=3 (additional aborted work) +Terminating after updates +Dying on request without maria_commit()/maria_close() +applying log +Differences in maria_chk -dvv, recovery not yet perfect ! +========DIFF START======= +7c7 +< Checksum: 2428948025 +--- +> Checksum: 0 +11c11 +< Datafile length: 16384 Keyfile length: 16384 +--- +> Datafile length: 16384 Keyfile length: 8192 +18c18 +< 1 2 6 unique number NULL 0 8192 8192 +--- +> 1 2 6 unique number NULL 0 8192 +========DIFF END======= +testing idempotency +applying log +Differences in maria_chk -dvv, recovery not yet perfect ! +========DIFF START======= +7c7 +< Checksum: 2428948025 +--- +> Checksum: 0 +11c11 +< Datafile length: 16384 Keyfile length: 16384 +--- +> Datafile length: 16384 Keyfile length: 8192 +18c18 +< 1 2 6 unique number NULL 0 8192 8192 +--- +> 1 2 6 unique number NULL 0 8192 +========DIFF END======= +testing applying of CLRs to recreate table +applying log +Differences in maria_chk -dvv, recovery not yet perfect ! +========DIFF START======= +7c7 +< Checksum: 2428948025 +--- +> Checksum: 0 +11c11 +< Datafile length: 16384 Keyfile length: 16384 +--- +> Datafile length: 16384 Keyfile length: 8192 +18c18 +< 1 2 6 unique number NULL 0 8192 8192 +--- +> 1 2 6 unique number NULL 0 8192 +========DIFF END======= TEST WITH ma_test1 -s -M -T -c -N -b --testflag=1 (commit at end) TEST WITH ma_test1 -s -M -T -c -N -b --testflag=2 --test-undo=1 (additional aborted work) -terminating after inserts +Terminating after inserts Dying on request without maria_commit()/maria_close() applying log Differences in maria_chk -dvv, recovery not yet perfect ! @@ -363,9 +528,9 @@ Differences in maria_chk -dvv, recovery not yet perfect ! > 1 2 6 unique varchar BLOB NULL 0 8192 ========DIFF END======= TEST WITH ma_test1 -s -M -T -c -N --debug=d:t:i:o,/tmp/ma_test1.trace -b --testflag=3 (commit at end) -Terminating after update +Terminating after updates TEST WITH ma_test1 -s -M -T -c -N --debug=d:t:i:o,/tmp/ma_test1.trace -b --testflag=4 --test-undo=1 (additional aborted work) -terminating after deletes +Terminating after deletes Dying on request without maria_commit()/maria_close() applying log testing idempotency @@ -387,9 +552,64 @@ Differences in maria_chk -dvv, recovery not yet perfect ! --- > 1 2 6 unique varchar BLOB NULL 0 8192 ========DIFF END======= +TEST WITH ma_test1 -s -M -T -c -N --debug=d:t:i:o,/tmp/ma_test1.trace -b --testflag=2 (commit at end) +Terminating after inserts +TEST WITH ma_test1 -s -M -T -c -N --debug=d:t:i:o,/tmp/ma_test1.trace -b --testflag=3 --test-undo=1 (additional aborted work) +Terminating after updates +Dying on request without maria_commit()/maria_close() +applying log +Differences in maria_chk -dvv, recovery not yet perfect ! +========DIFF START======= +7c7 +< Checksum: 529753687 +--- +> Checksum: 800025671 +11c11 +< Datafile length: 49152 Keyfile length: 16384 +--- +> Datafile length: 49152 Keyfile length: 8192 +18c18 +< 1 2 6 unique varchar BLOB NULL 0 8192 8192 +--- +> 1 2 6 unique varchar BLOB NULL 0 8192 +========DIFF END======= +testing idempotency +applying log +Differences in maria_chk -dvv, recovery not yet perfect ! +========DIFF START======= +7c7 +< Checksum: 529753687 +--- +> Checksum: 800025671 +11c11 +< Datafile length: 49152 Keyfile length: 16384 +--- +> Datafile length: 49152 Keyfile length: 8192 +18c18 +< 1 2 6 unique varchar BLOB NULL 0 8192 8192 +--- +> 1 2 6 unique varchar BLOB NULL 0 8192 +========DIFF END======= +testing applying of CLRs to recreate table +applying log +Differences in maria_chk -dvv, recovery not yet perfect ! +========DIFF START======= +7c7 +< Checksum: 529753687 +--- +> Checksum: 0 +11c11 +< Datafile length: 49152 Keyfile length: 16384 +--- +> Datafile length: 49152 Keyfile length: 8192 +18c18 +< 1 2 6 unique varchar BLOB NULL 0 8192 8192 +--- +> 1 2 6 unique varchar BLOB NULL 0 8192 +========DIFF END======= TEST WITH ma_test1 -s -M -T -c -N -b --testflag=1 (commit at end) TEST WITH ma_test1 -s -M -T -c -N -b --testflag=2 --test-undo=2 (additional aborted work) -terminating after inserts +Terminating after inserts Dying on request without maria_commit()/maria_close() applying log Differences in maria_chk -dvv, recovery not yet perfect ! @@ -442,9 +662,9 @@ Differences in maria_chk -dvv, recovery not yet perfect ! > 1 2 6 unique varchar BLOB NULL 0 8192 ========DIFF END======= TEST WITH ma_test1 -s -M -T -c -N --debug=d:t:i:o,/tmp/ma_test1.trace -b --testflag=3 (commit at end) -Terminating after update +Terminating after updates TEST WITH ma_test1 -s -M -T -c -N --debug=d:t:i:o,/tmp/ma_test1.trace -b --testflag=4 --test-undo=2 (additional aborted work) -terminating after deletes +Terminating after deletes Dying on request without maria_commit()/maria_close() applying log testing idempotency @@ -466,9 +686,64 @@ Differences in maria_chk -dvv, recovery not yet perfect ! --- > 1 2 6 unique varchar BLOB NULL 0 8192 ========DIFF END======= +TEST WITH ma_test1 -s -M -T -c -N --debug=d:t:i:o,/tmp/ma_test1.trace -b --testflag=2 (commit at end) +Terminating after inserts +TEST WITH ma_test1 -s -M -T -c -N --debug=d:t:i:o,/tmp/ma_test1.trace -b --testflag=3 --test-undo=2 (additional aborted work) +Terminating after updates +Dying on request without maria_commit()/maria_close() +applying log +Differences in maria_chk -dvv, recovery not yet perfect ! +========DIFF START======= +7c7 +< Checksum: 529753687 +--- +> Checksum: 800025671 +11c11 +< Datafile length: 49152 Keyfile length: 16384 +--- +> Datafile length: 49152 Keyfile length: 8192 +18c18 +< 1 2 6 unique varchar BLOB NULL 0 8192 8192 +--- +> 1 2 6 unique varchar BLOB NULL 0 8192 +========DIFF END======= +testing idempotency +applying log +Differences in maria_chk -dvv, recovery not yet perfect ! +========DIFF START======= +7c7 +< Checksum: 529753687 +--- +> Checksum: 800025671 +11c11 +< Datafile length: 49152 Keyfile length: 16384 +--- +> Datafile length: 49152 Keyfile length: 8192 +18c18 +< 1 2 6 unique varchar BLOB NULL 0 8192 8192 +--- +> 1 2 6 unique varchar BLOB NULL 0 8192 +========DIFF END======= +testing applying of CLRs to recreate table +applying log +Differences in maria_chk -dvv, recovery not yet perfect ! +========DIFF START======= +7c7 +< Checksum: 529753687 +--- +> Checksum: 0 +11c11 +< Datafile length: 49152 Keyfile length: 16384 +--- +> Datafile length: 49152 Keyfile length: 8192 +18c18 +< 1 2 6 unique varchar BLOB NULL 0 8192 8192 +--- +> 1 2 6 unique varchar BLOB NULL 0 8192 +========DIFF END======= TEST WITH ma_test1 -s -M -T -c -N -b --testflag=1 (commit at end) TEST WITH ma_test1 -s -M -T -c -N -b --testflag=2 --test-undo=3 (additional aborted work) -terminating after inserts +Terminating after inserts Dying on request without maria_commit()/maria_close() applying log Differences in maria_chk -dvv, recovery not yet perfect ! @@ -521,9 +796,9 @@ Differences in maria_chk -dvv, recovery not yet perfect ! > 1 2 6 unique varchar BLOB NULL 0 8192 ========DIFF END======= TEST WITH ma_test1 -s -M -T -c -N --debug=d:t:i:o,/tmp/ma_test1.trace -b --testflag=3 (commit at end) -Terminating after update +Terminating after updates TEST WITH ma_test1 -s -M -T -c -N --debug=d:t:i:o,/tmp/ma_test1.trace -b --testflag=4 --test-undo=3 (additional aborted work) -terminating after deletes +Terminating after deletes Dying on request without maria_commit()/maria_close() applying log testing idempotency @@ -545,3 +820,58 @@ Differences in maria_chk -dvv, recovery not yet perfect ! --- > 1 2 6 unique varchar BLOB NULL 0 8192 ========DIFF END======= +TEST WITH ma_test1 -s -M -T -c -N --debug=d:t:i:o,/tmp/ma_test1.trace -b --testflag=2 (commit at end) +Terminating after inserts +TEST WITH ma_test1 -s -M -T -c -N --debug=d:t:i:o,/tmp/ma_test1.trace -b --testflag=3 --test-undo=3 (additional aborted work) +Terminating after updates +Dying on request without maria_commit()/maria_close() +applying log +Differences in maria_chk -dvv, recovery not yet perfect ! +========DIFF START======= +7c7 +< Checksum: 529753687 +--- +> Checksum: 0 +11c11 +< Datafile length: 49152 Keyfile length: 16384 +--- +> Datafile length: 49152 Keyfile length: 8192 +18c18 +< 1 2 6 unique varchar BLOB NULL 0 8192 8192 +--- +> 1 2 6 unique varchar BLOB NULL 0 8192 +========DIFF END======= +testing idempotency +applying log +Differences in maria_chk -dvv, recovery not yet perfect ! +========DIFF START======= +7c7 +< Checksum: 529753687 +--- +> Checksum: 0 +11c11 +< Datafile length: 49152 Keyfile length: 16384 +--- +> Datafile length: 49152 Keyfile length: 8192 +18c18 +< 1 2 6 unique varchar BLOB NULL 0 8192 8192 +--- +> 1 2 6 unique varchar BLOB NULL 0 8192 +========DIFF END======= +testing applying of CLRs to recreate table +applying log +Differences in maria_chk -dvv, recovery not yet perfect ! +========DIFF START======= +7c7 +< Checksum: 529753687 +--- +> Checksum: 0 +11c11 +< Datafile length: 49152 Keyfile length: 16384 +--- +> Datafile length: 49152 Keyfile length: 8192 +18c18 +< 1 2 6 unique varchar BLOB NULL 0 8192 8192 +--- +> 1 2 6 unique varchar BLOB NULL 0 8192 +========DIFF END======= -- cgit v1.2.1 From cec8ac3e078041fb603e7ab49e4f64501bc02679 Mon Sep 17 00:00:00 2001 From: unknown Date: Wed, 12 Sep 2007 11:27:34 +0200 Subject: WL#3071 Maria checkpoint Finally this is the real checkpoint code. It however exhibits unstabilities when a checkpoint runs concurrently with data-modifying clients (table corruption, transaction log's assertions) so for now a checkpoint is taken only at startup after recovery and at shutdown, i.e. not in concurrent situations. Later we will let it run periodically, as well as flush dirty pages periodically (almost all needed code is there already, only pagecache code is written but not committed). WL#3072 Maria recovery * replacing UNDO_ROW_PURGE with CLR_END; testing of those CLR_END via ma_test2 which has INSERTs failing with duplicate keys. * replaying of REDO_RENAME_TABLE Now, off to test Recovery in ha_maria :) BitKeeper/deleted/.del-ma_least_recently_dirtied.c: Delete: storage/maria/ma_least_recently_dirtied.c BitKeeper/deleted/.del-ma_least_recently_dirtied.h: Delete: storage/maria/ma_least_recently_dirtied.h storage/maria/Makefile.am: compile Checkpoint module storage/maria/ha_maria.cc: When ha_maria starts, do a recovery from last checkpoint. Take a checkpoint when that recovery has ended and when ha_maria shuts down cleanly. storage/maria/ma_blockrec.c: * even if my_sync() fails we have to my_close() (otherwise we leak a descriptor) * UNDO_ROW_PURGE is replaced by a simple CLR_END for UNDO_ROW_INSERT, as promised in the old comment; it gives us skipping during the UNDO phase. storage/maria/ma_check.c: All REDOs before create_rename_lsn are ignored by Recovery. So create_rename_lsn must be set only after all data/index has been flushed and forced to disk. We thus move write_log_record_for_repair() to after _ma_flush_tables_files_after_repair(). storage/maria/ma_checkpoint.c: Checkpoint module. storage/maria/ma_checkpoint.h: optional argument if caller wants a thread to periodically take checkpoints and flush dirty pages. storage/maria/ma_create.c: * no need to init some vars as the initial bzero(share) takes care of this. * update to new function's name * even if we fail in my_sync() we have to my_close() storage/maria/ma_extra.c: Checkpoint reads share->last_version under intern_lock, so we make maria_extra() update it under intern_lock. THR_LOCK_maria still needed because of _ma_test_if_reopen(). storage/maria/ma_init.c: destroy checkpoint module when Maria shuts down. storage/maria/ma_loghandler.c: * UNDO_ROW_PURGE gone (see ma_blockrec.c) * we need to remember the LSN of the LOGREC_FILE_ID for a share, because this LSN is needed into the checkpoint record (Recovery wants to know the validity domain of an id->name mapping) * translog_get_horizon_no_lock() needed for Checkpoint * comment about failing assertion (Sanja knows) * translog_init_reader_data() thought that translog_read_record_header_scan() returns 0 in case of error, but 0 just means "0-length header". * translog_assign_id_to_share() now needs the MARIA_HA because LOGREC_FILE_ID uses a log-write hook. * Verify that (de)assignment of share->id happens only under intern_lock, as Checkpoint reads this id with intern_lock. * translog_purge() can accept TRANSLOG_ADDRESS, not necessarily a real LSN. storage/maria/ma_loghandler.h: prototype updates storage/maria/ma_open.c: no need to initialize "res" storage/maria/ma_pagecache.c: When taking a checkpoint, we don't need to know the maximum rec_lsn of dirty pages; this LSN was intended to be used in the two-checkpoint rule, but last_checkpoint_lsn is as good. 4 bytes for stored_list_size is enough as PAGECACHE::blocks (number of blocks which the pagecache can contain) is int. storage/maria/ma_pagecache.h: new prototype storage/maria/ma_recovery.c: * added replaying of REDO_RENAME_TABLE * UNDO_ROW_PURGE gone (see ma_blockrec.c), replaced by CLR_END * Recovery from the last checkpoint record now possible * In new_table() we skip the table if the id->name mapping is older than create_rename_lsn (mapping dates from lsn_of_file_id). * in get_MARIA_HA_from_REDO_record() we skip the record if the id->name mapping is newer than the record (can happen if processing a record which is before the checkpoint record). * parse_checkpoint_record() has to return a LSN, that's what caller expects storage/maria/ma_rename.c: new function's name; log end zeroes of tables' names (ease recovery) storage/maria/ma_test2.c: * equivalent of ma_test1's --test-undo added (named -u here). * -t=1 now stops right after creating the table, so that we can test undoing of INSERTs with duplicate keys (which tests the CLR_END logged by _ma_write_abort_block_record()). storage/maria/ma_test_recovery.expected: Result of testing undoing of INSERTs with duplicate keys; there are some differences in maria_chk -dvv but they are normal (removing records does not shrink data/index file, does not put back the "analyzed, optimized keys"(etc) index state. storage/maria/ma_test_recovery: Test undoing of INSERTs with duplicate keys, using ma_test2; when such INSERT happens, it logs REDO_INSERT, UNDO_INSERT, REDO_DELETE, CLR_END; we abort after that, and test that CLR_END causes recovery to jump over UNDO_INSERT. storage/maria/ma_write.c: comment storage/maria/maria_chk.c: comment storage/maria/maria_def.h: * a new bit in MARIA_SHARE::in_checkpoint, used to build a list of unique shares during Checkpoint. * MARIA_SHARE::lsn_of_file_id added: the LSN of the last LOGREC_FILE_ID for this share; needed to know to which LSN domain the mappings found in the Checkpoint record apply (new mappings should not apply to old REDOs). storage/maria/trnman.c: * small changes to how trnman_collect_transactions() fills its buffer; it also uses a non-dummy lsn_read_non_atomic() found in ma_checkpoint.h --- storage/maria/Makefile.am | 5 +- storage/maria/ha_maria.cc | 8 +- storage/maria/ma_blockrec.c | 64 +- storage/maria/ma_check.c | 25 +- storage/maria/ma_checkpoint.c | 1291 ++++++++++++++++++++++------- storage/maria/ma_checkpoint.h | 2 +- storage/maria/ma_create.c | 38 +- storage/maria/ma_extra.c | 7 +- storage/maria/ma_init.c | 2 + storage/maria/ma_least_recently_dirtied.c | 105 --- storage/maria/ma_least_recently_dirtied.h | 25 - storage/maria/ma_loghandler.c | 125 +-- storage/maria/ma_loghandler.h | 6 +- storage/maria/ma_open.c | 26 +- storage/maria/ma_pagecache.c | 19 +- storage/maria/ma_pagecache.h | 3 +- storage/maria/ma_recovery.c | 296 ++++--- storage/maria/ma_rename.c | 21 +- storage/maria/ma_test2.c | 54 +- storage/maria/ma_test_recovery | 13 +- storage/maria/ma_test_recovery.expected | 246 ++++++ storage/maria/ma_write.c | 10 + storage/maria/maria_chk.c | 4 +- storage/maria/maria_def.h | 13 +- storage/maria/trnman.c | 41 +- 25 files changed, 1699 insertions(+), 750 deletions(-) delete mode 100644 storage/maria/ma_least_recently_dirtied.c delete mode 100644 storage/maria/ma_least_recently_dirtied.h diff --git a/storage/maria/Makefile.am b/storage/maria/Makefile.am index 03d1cc75347..2bd9b7db922 100644 --- a/storage/maria/Makefile.am +++ b/storage/maria/Makefile.am @@ -61,7 +61,8 @@ noinst_HEADERS = maria_def.h ma_rt_index.h ma_rt_key.h ma_rt_mbr.h \ ma_ft_eval.h trnman.h lockman.h tablockman.h \ ma_control_file.h ha_maria.h ma_blockrec.h \ ma_loghandler.h ma_loghandler_lsn.h ma_pagecache.h \ - ma_recovery.h ma_commit.h trnman_public.h + ma_checkpoint.h ma_recovery.h ma_commit.h \ + trnman_public.h ma_test1_DEPENDENCIES= $(LIBRARIES) ma_test1_LDADD= @CLIENT_EXTRA_LDFLAGS@ libmaria.a \ $(top_builddir)/storage/myisam/libmyisam.a \ @@ -120,7 +121,7 @@ libmaria_a_SOURCES = ma_init.c ma_open.c ma_extra.c ma_info.c ma_rkey.c \ ma_rt_index.c ma_rt_key.c ma_rt_mbr.c ma_rt_split.c \ ma_sp_key.c ma_control_file.c ma_loghandler.c \ ma_pagecache.c ma_pagecaches.c \ - ma_recovery.c ma_commit.c + ma_checkpoint.c ma_recovery.c ma_commit.c CLEANFILES = test?.MA? FT?.MA? isam.log ma_test_all ma_rt_test.MA? sp_test.MA? SUFFIXES = .sh diff --git a/storage/maria/ha_maria.cc b/storage/maria/ha_maria.cc index ab3c9a9310c..678b88063db 100644 --- a/storage/maria/ha_maria.cc +++ b/storage/maria/ha_maria.cc @@ -31,6 +31,8 @@ C_MODE_START #include "maria_def.h" #include "ma_rt_index.h" #include "ma_blockrec.h" +#include "ma_checkpoint.h" +#include "ma_recovery.h" C_MODE_END /* @@ -2344,6 +2346,7 @@ bool ha_maria::check_if_incompatible_data(HA_CREATE_INFO *info, static int maria_hton_panic(handlerton *hton, ha_panic_function flag) { + ma_checkpoint_execute(CHECKPOINT_FULL, FALSE); /* can't catch error */ return maria_panic(flag); } @@ -2403,7 +2406,10 @@ static int ha_maria_init(void *p) translog_init(maria_data_root, TRANSLOG_FILE_SIZE, MYSQL_VERSION_ID, server_id, maria_log_pagecache, TRANSLOG_DEFAULT_FLAGS) || - trnman_init(0); + maria_recover() || + ma_checkpoint_init(FALSE) || + /* One checkpoint after Recovery */ + ma_checkpoint_execute(CHECKPOINT_FULL, FALSE); maria_multi_threaded= TRUE; return res; } diff --git a/storage/maria/ma_blockrec.c b/storage/maria/ma_blockrec.c index 76104d00673..2558450b663 100644 --- a/storage/maria/ma_blockrec.c +++ b/storage/maria/ma_blockrec.c @@ -402,9 +402,10 @@ my_bool _ma_once_end_block_record(MARIA_SHARE *share) File must be synced as it is going out of the maria_open_list and so becoming unknown to Checkpoint. */ - if ((share->now_transactional && - my_sync(share->bitmap.file.file, MYF(MY_WME))) || - my_close(share->bitmap.file.file, MYF(MY_WME))) + if (share->now_transactional && + my_sync(share->bitmap.file.file, MYF(MY_WME))) + res= 1; + if (my_close(share->bitmap.file.file, MYF(MY_WME))) res= 1; /* Trivial assignment to guard against multiple invocations @@ -2587,7 +2588,8 @@ my_bool _ma_write_abort_block_record(MARIA_HA *info) my_bool res= 0; MARIA_BITMAP_BLOCKS *blocks= &info->cur_row.insert_blocks; MARIA_BITMAP_BLOCK *block, *end; - DBUG_ENTER("_ma_abort_write_block_record"); + LSN lsn= LSN_IMPOSSIBLE; + DBUG_ENTER("_ma_write_abort_block_record"); if (delete_head_or_tail(info, ma_recordpos_to_page(info->cur_row.lastpos), @@ -2616,44 +2618,42 @@ my_bool _ma_write_abort_block_record(MARIA_HA *info) if (info->s->now_transactional) { - LSN lsn; + LSN previous_undo_lsn; + TRANSLOG_HEADER_BUFFER rec; LEX_STRING log_array[TRANSLOG_INTERNAL_PARTS + 1]; - uchar log_data[LSN_STORE_SIZE + FILEID_STORE_SIZE]; - + uchar log_data[LSN_STORE_SIZE + FILEID_STORE_SIZE + 1]; + int len; /* - Write UNDO record - This entry is just an end marker for the abort_insert as we will never - really undo a failed insert. Note that this UNDO will cause recover - to ignore the LOGREC_UNDO_ROW_INSERT that is the previous entry - in the UNDO chain. + We do need the code above (delete_head_or_tail() etc) for + non-transactional tables. + For transactional tables we could skip this code above and just execute + the UNDO_INSERT, but we try to have one code path. + Write CLR record, because we are somehow undoing UNDO_ROW_INSERT. + When we have logging for keys: as maria_write() first writes the row + then the keys, and if failure, deletes the keys then the rows, + info->trn->undo_lsn below will properly point to the UNDO of the + UNDO_ROW_INSERT for this row. */ - /** - @todo RECOVERY BUG - We do need the code above (delete_head_or_tail() etc) for - non-transactional tables. - For transactional tables we can either also use it or execute the - UNDO_INSERT. If we crash before this - _ma_write_abort_block_record(), Recovery will do the work of this - function by executing UNDO_INSERT. - For transactional tables, we will remove this LOGREC_UNDO_PURGE and - replace it with a LOGREC_CLR_END: we should go back the UNDO chain - until we reach the UNDO which inserted the row into the data file, and - use its previous_undo_lsn. - Same logic for when we remove inserted keys (in case of error in - maria_write(): we come to the present function only after removing the - inserted keys... as long as we unpin the key pages only after writing - the CLR_END, this would be recovery-safe...). - */ - lsn_store(log_data, info->trn->undo_lsn); + if ((len= translog_read_record_header(info->trn->undo_lsn, &rec)) == + RECHEADER_READ_ERROR) + { + res= 1; + goto end; + } + DBUG_ASSERT(rec.type == LOGREC_UNDO_ROW_INSERT); + previous_undo_lsn= lsn_korr(rec.header); + lsn_store(log_data, previous_undo_lsn); + log_data[LSN_STORE_SIZE + FILEID_STORE_SIZE]= LOGREC_UNDO_ROW_INSERT; log_array[TRANSLOG_INTERNAL_PARTS + 0].str= (char*) log_data; log_array[TRANSLOG_INTERNAL_PARTS + 0].length= sizeof(log_data); - if (translog_write_record(&lsn, LOGREC_UNDO_ROW_PURGE, + if (translog_write_record(&lsn, LOGREC_CLR_END, info->trn, info, sizeof(log_data), TRANSLOG_INTERNAL_PARTS + 1, log_array, log_data + LSN_STORE_SIZE)) res= 1; } - _ma_unpin_all_pages_and_finalize_row(info, info->trn->undo_lsn); +end: + _ma_unpin_all_pages_and_finalize_row(info, lsn); DBUG_RETURN(res); } diff --git a/storage/maria/ma_check.c b/storage/maria/ma_check.c index 354af51fbfd..fa1c812daf7 100644 --- a/storage/maria/ma_check.c +++ b/storage/maria/ma_check.c @@ -2264,7 +2264,7 @@ int maria_repair(HA_CHECK *param, register MARIA_HA *info, llstr(sort_info.dupp,llbuff)); } - got_error= sync_dir ? write_log_record_for_repair(param, info) : 0; + got_error= 0; /* If invoked by external program that uses thr_lock */ if (&share->state.state != info->state) memcpy( &share->state.state, info->state, sizeof(*info->state)); @@ -2309,6 +2309,14 @@ err: } maria_mark_crashed_on_repair(info); } + else if (sync_dir) + { + /* + Now that we have flushed and forced everything, we can bump + create_rename_lsn: + */ + write_log_record_for_repair(param, info); + } my_free(sort_param.rec_buff, MYF(MY_ALLOW_ZERO_PTR)); my_free(sort_param.record,MYF(MY_ALLOW_ZERO_PTR)); my_free(sort_info.buff,MYF(MY_ALLOW_ZERO_PTR)); @@ -5551,7 +5559,7 @@ read_next_page: /** @brief Writes a LOGREC_REPAIR_TABLE record and updates create_rename_lsn - and is_of_lsn + and is_of_horizon REPAIR/OPTIMIZE have replaced the data/index file with a new file and so, in this scenario: @@ -5572,6 +5580,7 @@ read_next_page: static int write_log_record_for_repair(const HA_CHECK *param, MARIA_HA *info) { + MARIA_SHARE *share= info->s; /* in case this is maria_chk or recovery... */ if (translog_inited && !maria_in_recovery) { @@ -5613,16 +5622,12 @@ static int write_log_record_for_repair(const HA_CHECK *param, MARIA_HA *info) return 1; /* The table's existence was made durable earlier (MY_SYNC_DIR passed to - maria_change_to_newfile()). + maria_change_to_newfile()). _ma_flush_table_files_after_repair() was + called earlier, flushed and forced data+index+state. Old REDOs should + not be applied to the table: */ - if (_ma_update_create_rename_lsn_on_disk(info->s, lsn, FALSE)) + if (_ma_update_create_rename_lsn(share, lsn, TRUE)) return 1; - /* - _ma_flush_table_files_after_repair() is later called by maria_repair(), - and makes sure to flush the data, index, update is_of_lsn, flush state - and sync, so create_rename_lsn reaches disk, thus we won't apply old - REDOs to the new table. - */ } return 0; } diff --git a/storage/maria/ma_checkpoint.c b/storage/maria/ma_checkpoint.c index 42b6c961b41..caeb5ec45d6 100644 --- a/storage/maria/ma_checkpoint.c +++ b/storage/maria/ma_checkpoint.c @@ -1,4 +1,4 @@ -/* Copyright (C) 2006 MySQL AB & MySQL Finland AB & TCX DataKonsult AB +/* Copyright (C) 2006,2007 MySQL AB 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 @@ -16,413 +16,1084 @@ /* WL#3071 Maria checkpoint First version written by Guilhem Bichot on 2006-04-27. - Does not compile yet. */ /* Here is the implementation of this module */ +/** + @todo RECOVERY BUG this is unreviewed code, but used in safe conditions: + ha_maria takes a checkpoint at end of recovery and one at clean shutdown, + that's all. So there never are open tables, dirty pages, transactions. +*/ /* Summary: - - there are asynchronous checkpoints (a writer to the log notices that it's - been a long time since we last checkpoint-ed, so posts a request for a - background thread to do a checkpoint; does not care about the success of the - checkpoint). Then the checkpoint is done by the checkpoint thread, at an - unspecified moment ("later") (==soon, of course). - - there are synchronous checkpoints: a thread requests a checkpoint to - happen now and wants to know when it finishes and if it succeeded; then the - checkpoint is done by that same thread. + checkpoints are done either by a background thread (checkpoint every Nth + second) or by a client. + In ha_maria, it's not made available to clients, and will soon be done by a + background thread (periodically taking checkpoints and flushing dirty + pages). */ -#include "page_cache.h" -#include "least_recently_dirtied.h" -#include "transaction.h" -#include "share.h" -#include "log.h" +#include "maria_def.h" +#include "ma_pagecache.h" +#include "trnman.h" +#include "ma_blockrec.h" +#include "ma_checkpoint.h" #include "ma_loghandler_lsn.h" -#define LSN_MAX ((LSN)ULONGLONG_MAX) /* - this transaction is used for any system work (purge, checkpoint writing - etc), that is, background threads. It will not be declared/initialized here - in the final version. + Checkpoints currently happen only at ha_maria's startup (after recovery) and + at shutdown, always when there is no open tables. + Background page flushing is not used. + So, needed pagecache functions for doing this flushing are not yet pushed. */ -st_transaction system_trans= {0 /* long trans id */, 0 /* short trans id */,0,...}; - -/* those three are protected by the log's mutex */ -/* - The maximum rec_lsn in the LRD when last checkpoint was run, serves for the - MEDIUM checkpoint. +#define flush_pagecache_blocks_with_filter(A,B,C,D,E) (((int)D) * 0) +/** + filter has to return 0, 1 or 2: 0 means "don't flush this page", 1 means + "flush it", 2 means "don't flush this page and following pages". + Will move to ma_pagecache.h */ -LSN max_rec_lsn_at_last_checkpoint= 0; -/* last submitted checkpoint request; cleared when starts */ -CHECKPOINT_LEVEL next_asynchronous_checkpoint_to_do= NONE; -CHECKPOINT_LEVEL checkpoint_in_progress= NONE; +typedef int (*PAGECACHE_FILTER)(enum pagecache_page_type type, + pgcache_page_no_t page, + LSN rec_lsn, void *arg); -static inline ulonglong read_non_atomic(ulonglong volatile *x); -/* - Used by MySQL client threads requesting a checkpoint (like "ALTER MARIA - ENGINE DO CHECKPOINT"), and probably by maria_panic(), and at the end of the - UNDO recovery phase. -*/ -my_bool execute_synchronous_checkpoint(CHECKPOINT_LEVEL level) +/** @brief type of checkpoint currently running */ +static CHECKPOINT_LEVEL checkpoint_in_progress= CHECKPOINT_NONE; +/** @brief protects checkpoint_in_progress */ +static pthread_mutex_t LOCK_checkpoint; +/** @brief for killing the background checkpoint thread */ +static pthread_cond_t COND_checkpoint; +/** @brief if checkpoint module was inited or not */ +static my_bool checkpoint_inited= FALSE; +/** @brief 'kill' flag for the background checkpoint thread */ +static int checkpoint_thread_die; +/* is ulong like pagecache->blocks_changed */ +static ulong pages_to_flush_before_next_checkpoint; +static PAGECACHE_FILE *dfiles, /**< data files to flush in background */ + *dfiles_end; /**< list of data files ends here */ +static PAGECACHE_FILE *kfiles, /**< index files to flush in background */ + *kfiles_end; /**< list of index files ends here */ +/* those two statistics below could serve in SHOW GLOBAL STATUS */ +static uint checkpoints_total= 0, /**< all checkpoint requests made */ + checkpoints_ok_total= 0; /**< all checkpoints which succeeded */ + +struct st_filter_param { - my_bool result; - DBUG_ENTER("execute_synchronous_checkpoint"); - DBUG_ASSERT(level > NONE); + my_bool is_data_file; /**< is the file about data or index */ + LSN up_to_lsn; /**< only pages with rec_lsn < this LSN */ + ulong pages_covered_by_bitmap; /**< to know which page is a bitmap page */ + uint max_pages; /**< stop after flushing this number pages */ +}; /**< information to determine which dirty pages should be flushed */ - lock(log_mutex); - while (checkpoint_in_progress != NONE) - wait_on_checkpoint_done_cond(); +static int filter_flush_data_file_medium(enum pagecache_page_type type, + pgcache_page_no_t page, + LSN rec_lsn, void *arg); +static int filter_flush_data_file_full(enum pagecache_page_type type, + pgcache_page_no_t page, + LSN rec_lsn, void *arg); +static int filter_flush_data_file_indirect(enum pagecache_page_type type, + pgcache_page_no_t page, + LSN rec_lsn, void *arg); +static int filter_flush_data_file_evenly(enum pagecache_page_type type, + pgcache_page_no_t pageno, + LSN rec_lsn, void *arg); +static int really_execute_checkpoint(); +pthread_handler_t ma_checkpoint_background(void *arg); +static int collect_tables(); - result= execute_checkpoint(level); - DBUG_RETURN(result); -} +/** + @brief Does a checkpoint -/* - If no checkpoint is running, and there is a pending asynchronous checkpoint - request, executes it. - Is safe if multiple threads call it, though in first version only one will. - It's intended to be used by a thread which regularly calls this function; - this is why, if there is a request, it does not wait in a loop for - synchronous checkpoints to be finished, but just exits (because the thread - may want to do something useful meanwhile (flushing dirty pages for example) - instead of waiting). -*/ -my_bool execute_asynchronous_checkpoint_if_any() -{ - my_bool result; - CHECKPOINT_LEVEL level; - DBUG_ENTER("execute_asynchronous_checkpoint"); - - /* first check without mutex, ok to see old data */ - if (likely((next_asynchronous_checkpoint_to_do == NONE) || - (checkpoint_in_progress != NONE))) - DBUG_RETURN(FALSE); - - lock(log_mutex); - if (likely((next_asynchronous_checkpoint_to_do == NONE) || - (checkpoint_in_progress != NONE))) - { - unlock(log_mutex); - DBUG_RETURN(FALSE); - } + @param level what level of checkpoint to do + @param no_wait if another checkpoint of same or stronger level + is already running, consider our job done - result= execute_checkpoint(next_asynchronous_checkpoint_to_do); - DBUG_RETURN(result); -} + @note In ha_maria, there can never be two threads trying a checkpoint at + the same time. - -/* - Does the actual checkpointing. Called by - execute_synchronous_checkpoint() and - execute_asynchronous_checkpoint_if_any(). + @return Operation status + @retval 0 ok + @retval !=0 error */ -my_bool execute_checkpoint(CHECKPOINT_LEVEL level) + +int ma_checkpoint_execute(CHECKPOINT_LEVEL level, my_bool no_wait) { - my_bool result; - DBUG_ENTER("execute_checkpoint"); + int result= 0; + DBUG_ENTER("ma_checkpoint_execute"); - safemutex_assert_owner(log_mutex); - if (next_asynchronous_checkpoint_to_do <= level) - next_asynchronous_checkpoint_to_do= NONE; - checkpoint_in_progress= level; + DBUG_ASSERT(checkpoint_inited); + DBUG_ASSERT(level > CHECKPOINT_NONE); - if (unlikely(level > INDIRECT)) + /* look for already running checkpoints */ + pthread_mutex_lock(&LOCK_checkpoint); + while (checkpoint_in_progress != CHECKPOINT_NONE) { - LSN copy_of_max_rec_lsn_at_last_checkpoint= - max_rec_lsn_at_last_checkpoint; - /* much I/O work to do, release log mutex */ - unlock(log_mutex); - - switch (level) + if (no_wait && (checkpoint_in_progress >= level)) { - case FULL: - /* flush all pages up to the current end of the LRD */ - flush_all_LRD_to_lsn(LSN_MAX); - /* this will go full speed (normal scheduling, no sleep) */ - break; - case MEDIUM: /* - flush all pages which were already dirty at last checkpoint: - ensures that recovery will never start from before the next-to-last - checkpoint (two-checkpoint rule). + If we are the checkpoint background thread, we don't wait (it's + smarter to flush pages instead of waiting here while the other thread + finishes its checkpoint). */ - flush_all_LRD_to_lsn(copy_of_max_rec_lsn_at_last_checkpoint); - /* this will go full speed (normal scheduling, no sleep) */ - break; + pthread_mutex_unlock(&LOCK_checkpoint); + goto end; } - lock(log_mutex); + pthread_cond_wait(&COND_checkpoint, &LOCK_checkpoint); } - result= execute_checkpoint_indirect(); - checkpoint_in_progress= NONE; - unlock(log_mutex); - broadcast(checkpoint_done_cond); + checkpoint_in_progress= level; + pthread_mutex_unlock(&LOCK_checkpoint); + /* from then on, we are sure to be and stay the only checkpointer */ + + result= really_execute_checkpoint(); + pthread_cond_broadcast(&COND_checkpoint); +end: DBUG_RETURN(result); } -/* - Does an indirect checpoint (collects data from data structures, writes into - a checkpoint log record). - Starts and ends while having log's mutex (released in the middle). +/** + @brief Does a checkpoint, really; expects no other checkpoints + running. + + Checkpoint level requested is read from checkpoint_in_progress. + + @return Operation status + @retval 0 ok + @retval !=0 error */ -my_bool execute_checkpoint_indirect() + +static int really_execute_checkpoint() { - int error= 0, i; - /* checkpoint record data: */ - LSN checkpoint_start_lsn; - char checkpoint_start_lsn_char[8]; - LEX_STRING strings[6]= - {checkpoint_start_lsn_char, 8}, {0,0}, {0,0}, {0,0}, {0,0}, {0,0} }; + uint i, error= 0; + /** @brief checkpoint_start_log_horizon will be stored there */ char *ptr; - LSN checkpoint_lsn; - LSN candidate_max_rec_lsn_at_last_checkpoint; - DBUG_ENTER("execute_checkpoint_indirect"); + LEX_STRING record_pieces[4]; /**< only malloc-ed pieces */ + LSN min_page_rec_lsn, min_trn_rec_lsn, min_first_undo_lsn; + TRANSLOG_ADDRESS checkpoint_start_log_horizon; + uchar checkpoint_start_log_horizon_char[LSN_STORE_SIZE]; + DBUG_ENTER("really_execute_checkpoint"); + bzero(&record_pieces, sizeof(record_pieces)); - DBUG_ASSERT(sizeof(uchar *) <= 8); - DBUG_ASSERT(sizeof(LSN) <= 8); + /* + STEP 1: record current end-of-log position using log's lock. It is + critical for the correctness of Checkpoint (related to memory visibility + rules, the log's lock is a mutex). + "Horizon" is a lower bound of the LSN of the next log record. + */ + /** + @todo RECOVERY BUG + this is an horizon, but it is used as a LSN (REDO phase may start from + there! probably log handler would refuse to read then; + Sanja proposed to make a loghandler's function which finds the LSN after + this horizon. + */ + checkpoint_start_log_horizon= translog_get_horizon(); +#define LSN_IN_HEX(L) (ulong)LSN_FILE_NO(L),(ulong)LSN_OFFSET(L) + DBUG_PRINT("info",("checkpoint_start_log_horizon (%lu,0x%lx)", + LSN_IN_HEX(checkpoint_start_log_horizon))); + lsn_store(checkpoint_start_log_horizon_char, checkpoint_start_log_horizon); - safemutex_assert_owner(log_mutex); - /* STEP 1: record current end-of-log LSN */ - checkpoint_start_lsn= log_read_end_lsn(); - if (LSN_IMPOSSIBLE == checkpoint_start_lsn) /* error */ - DBUG_RETURN(TRUE); - unlock(log_mutex); + /* + STEP 2: fetch information about transactions. + We must fetch transactions before dirty pages. Indeed, a transaction + first sets its rec_lsn then sets the page's rec_lsn then sets its rec_lsn + to 0. If we fetched pages first, we may see no dirty page yet, then we + fetch transactions but the transaction has already reset its rec_lsn to 0 + so we miss rec_lsn again. + For a similar reason (over-allocated bitmap pages) we have to fetch + transactions before flushing bitmap pages. - DBUG_PRINT("info",("checkpoint_start_lsn %lu", checkpoint_start_lsn)); - int8store(strings[0].str, checkpoint_start_lsn); + min_trn_rec_lsn will serve to lower the starting point of the REDO phase + (down from checkpoint_start_log_horizon). + */ + if (unlikely(trnman_collect_transactions(&record_pieces[0], + &record_pieces[1], + &min_trn_rec_lsn, + &min_first_undo_lsn))) + goto err; - /* STEP 2: fetch information about dirty pages */ - if (pagecache_collect_changed_blocks_with_LSN(pagecache, &strings[1], - &candidate_max_rec_lsn_at_last_checkpoint)) + /* STEP 3: fetch information about table files */ + if (unlikely(collect_tables(&record_pieces[2]))) goto err; - /* STEP 3: fetch information about transactions */ - if (trnman_collect_transactions(&strings[2], &strings[3])) + + /* STEP 4: fetch information about dirty pages */ + /* + It's better to do it _after_ having flushed some data pages (which + collect_tables() may have done), because those are now non-dirty and so we + have a more up-to-date dirty pages list to put into the checkpoint record, + and thus we will have less work at Recovery. + */ + /* Using default pagecache for now */ + if (unlikely(pagecache_collect_changed_blocks_with_lsn(maria_pagecache, + &record_pieces[3], + &min_page_rec_lsn))) goto err; - /* STEP 4: fetch information about table files */ + /* LAST STEP: now write the checkpoint log record */ { - /* This global mutex is in fact THR_LOCK_maria (see ma_open()) */ - lock(global_share_list_mutex); - strings[4].length= 8+(8+8)*share_list->count; - if (NULL == (strings[4].str= my_malloc(strings[4].length))) - goto err; - ptr= string3.str; + LSN lsn; + uint total_rec_length; /* - Note that maria_open_list is a list of MARIA_HA*, while we would prefer - a list of MARIA_SHARE* here (we are interested in the short id, - unique file name, members of MARIA_SHARE*, and in file descriptors, - which will in the end be in MARIA_SHARE*). + the log handler is allowed to modify "str" and "length" (but not "*str") + of its argument, so we must not pass it record_pieces directly, + otherwise we would later not know what memory pieces to my_free(). */ - for (iterate on the maria_open_list) + LEX_STRING log_array[TRANSLOG_INTERNAL_PARTS + 5]; + log_array[TRANSLOG_INTERNAL_PARTS + 0].str= + checkpoint_start_log_horizon_char; + log_array[TRANSLOG_INTERNAL_PARTS + 0].length= total_rec_length= + sizeof(checkpoint_start_log_horizon_char); + for (i= 0; i < (sizeof(record_pieces)/sizeof(record_pieces[0])); i++) { - /* latch each MARIA_SHARE, one by one, like this: */ - pthread_mutex_lock(&share->intern_lock); - /* - TODO: - we need to prevent the share from going away while we later flush and - force it without holding THR_LOCK_maria. For example if the share is - free()d by maria_close() we'll have a problem. Or if the share's file - descriptor is closed by maria_close() we will not be able to my_sync() - it. - */ - pthread_mutex_unlock(&share->intern_lock); - store the share pointer into a private array; + log_array[TRANSLOG_INTERNAL_PARTS + 1 + i]= record_pieces[i]; + total_rec_length+= record_pieces[i].length; } - unlock(global_share_list_mutex); - /* work on copy */ - int8store(ptr, elements_in_array); - ptr+= 8; - for (el in array) - { - int8store(ptr, array[...].short_id); - ptr+= 8; - memcpy(ptr, array[...].unique_file_name[_length], ...); - ptr+= ...; - /* maybe we need to lock share->intern_lock here */ - /* - these two are long ops (involving disk I/O) that's why we copied the - list, to not keep the list locked for long: - */ - flush_bitmap_pages(el); - /* TODO: and also autoinc counter, logical file end, free page list */ + if (unlikely(translog_write_record(&lsn, LOGREC_CHECKPOINT, + &dummy_transaction_object, NULL, + total_rec_length, + sizeof(log_array)/sizeof(log_array[0]), + log_array, NULL) || + translog_flush(lsn))) + goto err; - /* - fsyncs the fd, that's the loooong operation (e.g. max 150 fsync per - second, so if you have touched 1000 files it's 7 seconds). - */ - force_file(el); + translog_lock(); + /* + This cannot be done as a inwrite_rec_hook of LOGREC_CHECKPOINT, because + such hook would be called before translog_flush (and we must be sure + that log was flushed before we write to the control file). + */ + if (unlikely(ma_control_file_write_and_force(lsn, FILENO_IMPOSSIBLE, + CONTROL_FILE_UPDATE_ONLY_LSN))) + { + translog_unlock(); + goto err; } + translog_unlock(); } - /* LAST STEP: now write the checkpoint log record */ - - checkpoint_lsn= log_write_record(LOGREC_CHECKPOINT, - &system_trans, strings); - - /* - Do nothing between the log write and the control file write, for the - "repair control file" tool to be possible one day. - */ - - if (LSN_IMPOSSIBLE == checkpoint_lsn) - goto err; - - if (0 != control_file_write_and_force(checkpoint_lsn, NULL)) - goto err; - /* Note that we should not alter memory structures until we have successfully written the checkpoint record and control file. - Btw, a log write failure is serious: - - if we know how many bytes we managed to write, we should try to write - more, keeping the log's mutex (MY_FULL_IO) - - if we don't know, this log record is corrupted and we have no way to - "de-corrupt" it, so it will stay corrupted, and as the log is sequential, - any log record written after it will not be reachable (for example if we - would write UNDOs and crash, we would not be able to read the log and so - not be able to rollback), so we should stop the engine now (holding the - log's mutex) and do a recovery. */ + /* checkpoint succeeded */ + ptr= record_pieces[3].str; + pages_to_flush_before_next_checkpoint= uint4korr(ptr); + DBUG_PRINT("info",("%u pages to flush before next checkpoint", + (uint)pages_to_flush_before_next_checkpoint)); + + /* compute log's low-water mark */ + TRANSLOG_ADDRESS log_low_water_mark= min_page_rec_lsn; + set_if_smaller(log_low_water_mark, min_trn_rec_lsn); + set_if_smaller(log_low_water_mark, min_first_undo_lsn); + set_if_smaller(log_low_water_mark, checkpoint_start_log_horizon); + /** + Now purge unneeded logs. + As some systems have an unreliable fsync (drive lying), we could try to + be robust against that: remember a few previous checkpoints in the + control file, and not purge logs immediately... Think about it. + */ +#if 0 /* purging/keeping will be an option */ + if (translog_purge(log_low_water_mark)) + fprintf(stderr, "Maria engine: log purge failed\n"); /* not deadly */ +#endif + goto end; err: - print_error_to_error_log(the_error_message); - candidate_max_rec_lsn_at_last_checkpoint= LSN_IMPOSSIBLE; + error= 1; + fprintf(stderr, "Maria engine: checkpoint failed\n"); /* TODO: improve ;) */ + /* we were possibly not able to determine what pages to flush */ + pages_to_flush_before_next_checkpoint= 0; end: + for (i= 0; i < (sizeof(record_pieces)/sizeof(record_pieces[0])); i++) + my_free(record_pieces[i].str, MYF(MY_ALLOW_ZERO_PTR)); + pthread_mutex_lock(&LOCK_checkpoint); + checkpoint_in_progress= CHECKPOINT_NONE; + checkpoints_total++; + checkpoints_ok_total+= !error; + pthread_mutex_unlock(&LOCK_checkpoint); + DBUG_RETURN(error); +} - for (i= 1; i<6; i++) - my_free(strings[i].str, MYF(MY_ALLOW_ZERO_PTR)); - /* - this portion cannot be done as a hook in write_log_record() for the - LOGREC_CHECKPOINT type because: - - at that moment we still have not written to the control file so cannot - mark the request as done; this could be solved by writing to the control - file in the hook but that would be an I/O under the log's mutex, bad. - - it would not be nice organisation of code (I tried it :). - */ - if (candidate_max_rec_lsn_at_last_checkpoint != LSN_IMPOSSIBLE) +/** + @brief Initializes the checkpoint module + + @param create_background_thread If one wants the module to now create a + thread which will periodically do + checkpoints, and flush dirty pages, in the + background. + + @return Operation status + @retval 0 ok + @retval !=0 error +*/ + +int ma_checkpoint_init(my_bool create_background_thread) +{ + pthread_t th; + int res= 0; + DBUG_ENTER("ma_checkpoint_init"); + checkpoint_inited= TRUE; + checkpoint_thread_die= 2; /* not yet born == dead */ + if (pthread_mutex_init(&LOCK_checkpoint, MY_MUTEX_INIT_SLOW) || + pthread_cond_init(&COND_checkpoint, 0)) + res= 1; + else if (create_background_thread) { - /* checkpoint succeeded */ - /* - TODO: compute log's low water mark (how to do that with our fuzzy - ARIES-like reads of data structures? TODO think about it :). - */ - lock(log_mutex); - /* That LSN is used for the "two-checkpoint rule" (MEDIUM checkpoints) */ - maximum_rec_lsn_last_checkpoint= candidate_max_rec_lsn_at_last_checkpoint; - DBUG_RETURN(FALSE); + if (!(res= pthread_create(&th, NULL, ma_checkpoint_background, NULL))) + checkpoint_thread_die= 0; /* thread lives, will have to be killed */ } - lock(log_mutex); - DBUG_RETURN(TRUE); - /* - keep mutex locked upon exit because callers will want to clear - mutex-protected status variables - */ + DBUG_RETURN(res); } - -/* - Here's what should be put in log_write_record() in the log handler: +/** + @brief Destroys the checkpoint module */ -log_write_record(...) + +void ma_checkpoint_end() { - ...; - lock(log_mutex); - ...; - write_to_log(length); - written_since_last_checkpoint+= length; - if (written_since_last_checkpoint > - MAX_LOG_BYTES_WRITTEN_BETWEEN_CHECKPOINTS) + DBUG_ENTER("ma_checkpoint_end"); + if (checkpoint_inited) { - /* - ask one system thread (the "LRD background flusher and checkpointer - thread" WL#3261) to do a checkpoint - */ - request_asynchronous_checkpoint(INDIRECT); - /* prevent similar redundant requests */ - written_since_last_checkpoint= (my_off_t)0; + pthread_mutex_lock(&LOCK_checkpoint); + if (checkpoint_thread_die != 2) /* thread was started ok */ + { + DBUG_PRINT("info",("killing Maria background checkpoint thread")); + checkpoint_thread_die= 1; /* kill it */ + do /* and wait for it to be dead */ + { + /* wake it up if it was in a sleep */ + pthread_cond_broadcast(&COND_checkpoint); + DBUG_PRINT("info",("waiting for Maria background checkpoint thread" + " to die")); + pthread_cond_wait(&COND_checkpoint, &LOCK_checkpoint); + } + while (checkpoint_thread_die != 2); + } + pthread_mutex_unlock(&LOCK_checkpoint); + my_free((uchar *)dfiles, MYF(MY_ALLOW_ZERO_PTR)); + my_free((uchar *)kfiles, MYF(MY_ALLOW_ZERO_PTR)); + pthread_mutex_destroy(&LOCK_checkpoint); + pthread_cond_destroy(&COND_checkpoint); + checkpoint_inited= FALSE; } - ...; - unlock(log_mutex); - ...; + DBUG_VOID_RETURN; } -/* - Requests a checkpoint from the background thread, *asynchronously* - (requestor does not wait for completion, and does not even later check the - result). - In real life it will be called by log_write_record(). + +/** + @brief dirty-page filtering criteria for MEDIUM checkpoint. + + We flush data/index pages which have been dirty since the previous + checkpoint (this is the two-checkpoint rule: the REDO phase will not have + to start from earlier than the next-to-last checkpoint), and all dirty + bitmap pages. + + @param type Page's type + @param pageno Page's number + @param rec_lsn Page's rec_lsn + @param arg filter_param + + @return Operation status + @retval 0 don't flush the page + @retval 1 flush the page +*/ + +static int filter_flush_data_file_medium(enum pagecache_page_type type, + pgcache_page_no_t pageno, + LSN rec_lsn, void *arg) +{ + struct st_filter_param *param= (struct st_filter_param *)arg; + return ((type == PAGECACHE_LSN_PAGE) && + (cmp_translog_addr(rec_lsn, param->up_to_lsn) <= 0)) || + (param->is_data_file && + ((pageno % param->pages_covered_by_bitmap) == 0)); +} + + +/** + @brief dirty-page filtering criteria for FULL checkpoint. + + We flush all dirty data/index pages and all dirty bitmap pages. + + @param type Page's type + @param pageno Page's number + @param rec_lsn Page's rec_lsn + @param arg filter_param + + @return Operation status + @retval 0 don't flush the page + @retval 1 flush the page */ -void request_asynchronous_checkpoint(CHECKPOINT_LEVEL level); + +static int filter_flush_data_file_full(enum pagecache_page_type type, + pgcache_page_no_t pageno, + LSN rec_lsn + __attribute__ ((unused)), + void *arg) +{ + struct st_filter_param *param= (struct st_filter_param *)arg; + return (type == PAGECACHE_LSN_PAGE) || + (param->is_data_file && + ((pageno % param->pages_covered_by_bitmap) == 0)); +} + + +/** + @brief dirty-page filtering criteria for INDIRECT checkpoint. + + We flush all dirty bitmap pages. + + @param type Page's type + @param pageno Page's number + @param rec_lsn Page's rec_lsn + @param arg filter_param + + @return Operation status + @retval 0 don't flush the page + @retval 1 flush the page +*/ + +static int filter_flush_data_file_indirect(enum pagecache_page_type type + __attribute__ ((unused)), + pgcache_page_no_t pageno, + LSN rec_lsn + __attribute__ ((unused)), + void *arg) { - safemutex_assert_owner(log_mutex); + struct st_filter_param *param= (struct st_filter_param *)arg; + return + (param->is_data_file && + ((pageno % param->pages_covered_by_bitmap) == 0)); +} + + +/** + @brief dirty-page filtering criteria for background flushing thread. + + We flush data pages which have been dirty since the previous checkpoint + (this is the two-checkpoint rule: the REDO phase will not have to start + from earlier than the next-to-last checkpoint), and all dirty bitmap + pages. But we flush no more than a certain number of pages (to have an + even flushing, no write burst). - DBUG_ASSERT(level > NONE); - if ((next_asynchronous_checkpoint_to_do < level) && - (checkpoint_in_progress < level)) + @param type Page's type + @param pageno Page's number + @param rec_lsn Page's rec_lsn + @param arg filter_param + + @return Operation status + @retval 0 don't flush the page + @retval 1 flush the page + @retval 2 don't flush the page and following pages +*/ + +static int filter_flush_data_file_evenly(enum pagecache_page_type type, + pgcache_page_no_t pageno + __attribute__ ((unused)), + LSN rec_lsn, void *arg) +{ + struct st_filter_param *param= (struct st_filter_param *)arg; + if (unlikely(param->max_pages == 0)) /* all flushed already */ + return 2; + if ((type == PAGECACHE_LSN_PAGE) && + (cmp_translog_addr(rec_lsn, param->up_to_lsn) <= 0)) { - /* no equal or stronger running or to run, we post request */ - /* - We just don't broacast a cond, the checkpoint thread - (see ma_least_recently_dirtied.c) will notice our request in max a few - seconds. - */ - next_asynchronous_checkpoint_to_do= level; /* post request */ + param->max_pages--; + return 1; } + return 0; +} + + +/** + @brief Background thread which does checkpoints and flushes periodically. + + Takes a checkpoint every 30th second. After taking a checkpoint, all pages + dirty at the time of that checkpoint are flushed evenly until it is time to + take another checkpoint (30 seconds later). This ensures that the REDO + phase starts at earliest (in LSN time) at the next-to-last checkpoint + record ("two-checkpoint rule"). + + @note MikaelR questioned why the same thread does two different jobs, the + risk could be that while a checkpoint happens no LRD flushing happens. + + @note MikaelR noted that he observed that Linux's file cache may never + fsync to disk until this cache is full, at which point it decides to empty + the cache, making the machine very slow. A solution was to fsync after + writing 2 MB. +*/ + +pthread_handler_t ma_checkpoint_background(void *arg __attribute__((unused))) +{ + const uint sleep_unit= 1 /* 1 second */, + time_between_checkpoints= 30; /* 30 sleep units */ + uint sleeps= 0; + my_thread_init(); + DBUG_PRINT("info",("Maria background checkpoint thread starts")); + for(;;) + { +#if 0 /* good for testing, to do a lot of checkpoints, finds a lot of bugs */ + sleeps=0; +#endif + uint pages_bunch_size; + struct st_filter_param filter_param; + PAGECACHE_FILE *dfile; /**< data file currently being flushed */ + PAGECACHE_FILE *kfile; /**< index file currently being flushed */ + TRANSLOG_ADDRESS log_horizon_at_last_checkpoint= LSN_IMPOSSIBLE; + ulonglong pagecache_flushes_at_last_checkpoint= 0; + struct timespec abstime; + switch((sleeps++) % time_between_checkpoints) + { + case 0: + /* + With background flushing evenly distributed over the time + between two checkpoints, we should have only little flushing to do + in the checkpoint. + */ + /* + No checkpoint if no work of interest for recovery was done + since last checkpoint. Such work includes log writing (lengthens + recovery, checkpoint would shorten it), page flushing (checkpoint + would decrease the amount of read pages in recovery). + */ + if ((translog_get_horizon() == log_horizon_at_last_checkpoint) && + (pagecache_flushes_at_last_checkpoint == + maria_pagecache->global_cache_write)) + { + /* safety against errors during flush by this thread: */ + pages_to_flush_before_next_checkpoint= 0; + break; + } + ma_checkpoint_execute(CHECKPOINT_MEDIUM, TRUE); + /* + Snapshot this kind of "state" of the engine. Note that the value below + is possibly greater than last_checkpoint_lsn. + */ + log_horizon_at_last_checkpoint= translog_get_horizon(); + pagecache_flushes_at_last_checkpoint= + maria_pagecache->global_cache_write; + /* + If the checkpoint above succeeded it has set d|kfiles and + d|kfiles_end. If is has failed, it has set + pages_to_flush_before_next_checkpoint to 0 so we will skip flushing + and sleep until the next checkpoint. + */ + break; + case 1: + /* set up parameters for background page flushing */ + filter_param.up_to_lsn= last_checkpoint_lsn; + pages_bunch_size= pages_to_flush_before_next_checkpoint / + time_between_checkpoints; + dfile= dfiles; + kfile= kfiles; + /* fall through */ + default: + if (pages_bunch_size > 0) + { + /* flush a bunch of dirty pages */ + filter_param.max_pages= pages_bunch_size; + filter_param.is_data_file= TRUE; + while (dfile != dfiles_end) + { + int res= + flush_pagecache_blocks_with_filter(maria_pagecache, + dfile, FLUSH_KEEP, + filter_flush_data_file_evenly, + &filter_param); + /* note that it may just be a pinned page */ + if (unlikely(res)) + fprintf(stderr, "Maria engine: warning - background page flush" + " failed\n"); + if (filter_param.max_pages == 0) /* bunch all flushed, sleep */ + break; /* and we will continue with the same file */ + dfile++; /* otherwise all this file is flushed, move to next file */ + } + filter_param.is_data_file= FALSE; + while (kfile != kfiles_end) + { + int res= + flush_pagecache_blocks_with_filter(maria_pagecache, + dfile, FLUSH_KEEP, + filter_flush_data_file_evenly, + &filter_param); + if (unlikely(res)) + fprintf(stderr, "Maria engine: warning - background page flush" + " failed\n"); + if (filter_param.max_pages == 0) /* bunch all flushed, sleep */ + break; /* and we will continue with the same file */ + kfile++; /* otherwise all this file is flushed, move to next file */ + } + } + } + pthread_mutex_lock(&LOCK_checkpoint); + if (checkpoint_thread_die == 1) + break; +#if 0 /* good for testing, to do a lot of checkpoints, finds a lot of bugs */ + pthread_mutex_unlock(&LOCK_checkpoint); + my_sleep(100000); // a tenth of a second + pthread_mutex_lock(&LOCK_checkpoint); +#else + /* To have a killable sleep, we use timedwait like our SQL GET_LOCK() */ + set_timespec(abstime, sleep_unit); + pthread_cond_timedwait(&COND_checkpoint, &LOCK_checkpoint, &abstime); +#endif + if (checkpoint_thread_die == 1) + break; + pthread_mutex_unlock(&LOCK_checkpoint); + } + pthread_mutex_unlock(&LOCK_checkpoint); + DBUG_PRINT("info",("Maria background checkpoint thread ends")); /* - If there was an error, only an error - message to the error log will say it; normal, for a checkpoint triggered - by a log write, we probably don't want the client's log write to throw an - error, as the log write succeeded and a checkpoint failure is not - critical: the failure in this case is more for the DBA to know than for - the end user. + A last checkpoint, now that all tables should be closed; to have instant + recovery later. We always do it, because the test above about number of + log records or flushed pages is only approximative. For example, some log + records may have been written while ma_checkpoint_execute() above was + running, or some pages may have been flushed during this time. Thus it + could be that, while nothing has changed since that checkpoint's *end*, if + we recovered from that checkpoint we would have a non-empty dirty pages + list, REDOs to execute, and we don't want that, we want a clean shutdown + to have an empty recovery (simplifies upgrade/backups: one can just do a + clean shutdown, copy its tables to another system without copying the log + or control file and it will work because recovery will not need those). + Another reason why it's approximative is that a log record may have been + written above between ma_checkpoint_execute() and the + tranlog_get_horizon() which follows. + So, we have at least two checkpoints per start/stop of the engine, and + only two if the engine stays idle. */ + ma_checkpoint_execute(CHECKPOINT_FULL, FALSE); + pthread_mutex_lock(&LOCK_checkpoint); + checkpoint_thread_die= 2; /* indicate that we are dead */ + /* wake up ma_checkpoint_end() which may be waiting for our death */ + pthread_cond_broadcast(&COND_checkpoint); + /* broadcast was inside unlock because ma_checkpoint_end() destroys mutex */ + pthread_mutex_unlock(&LOCK_checkpoint); + my_thread_end(); + return 0; } -/* - If a 64-bit variable transitions from both halves being zero to both halves - being non-zero, and never changes after that (like the transaction's - first_undo_lsn), this function can be used to do a read of it (without - mutex, without atomic load) which always produces a correct (though maybe - slightly old) value (even on 32-bit CPUs). - The prototype will change with Sanja's new LSN type. +/** + @brief Allocates buffer and stores in it some info about open tables, + does some flushing on those. + + Does the allocation because the caller cannot know the size itself. + Memory freeing is to be done by the caller (if the "str" member of the + LEX_STRING is not NULL). + The caller is taking a checkpoint. + + @param[out] str pointer to where the allocated buffer, + and its size, will be put; buffer will be filled + with info about open tables + @param checkpoint_start_log_horizon Of the in-progress checkpoint + record. + + @return Operation status + @retval 0 OK + @retval 1 Error */ -static inline ulonglong read_non_atomic(ulonglong volatile *x) + +static int collect_tables(LEX_STRING *str, LSN checkpoint_start_log_horizon) { -#if ( SIZEOF_CHARP >= 8 ) - /* 64-bit CPU (right?), 64-bit reads are atomic */ - return *x; -#else + MARIA_SHARE **distinct_shares= NULL; + char *ptr; + uint error= 1, sync_error= 0, nb, nb_stored, i; + my_bool unmark_tables= TRUE; + uint total_names_length; + LIST *pos; /**< to iterate over open tables */ + struct st_state_copy { + uint index; + MARIA_STATE_INFO state; + }; + struct st_state_copy *state_copies= NULL, /**< fixed-size cache of states */ + *state_copies_end, /**< cache ends here */ + *state_copy; /**< iterator in cache */ + TRANSLOG_ADDRESS state_copies_horizon; /**< horizon of states' _copies_ */ + DBUG_ENTER("collect_tables"); + + /* let's make a list of distinct shares */ + pthread_mutex_lock(&THR_LOCK_maria); + for (nb= 0, pos= maria_open_list; pos; pos= pos->next) + { + MARIA_HA *info= (MARIA_HA*)pos->data; + MARIA_SHARE *share= info->s; + /* the first three variables below can never change */ + if (share->base.born_transactional && !share->temporary && + share->mode != O_RDONLY && + !(share->in_checkpoint & MARIA_CHECKPOINT_SEEN_IN_LOOP)) + { + /* + Why we didn't take intern_lock above: table had in_checkpoint==0 so no + thread could set in_checkpoint. And no thread needs to know that we + are setting in_checkpoint, because only maria_close() needs it and + cannot run now as we hold THR_LOCK_maria. + */ + /* + This table is relevant for checkpoint and not already seen. Mark it, + so that it is not seen again in the loop. + */ + nb++; + DBUG_ASSERT(share->in_checkpoint == 0); + /* This flag ensures that we count only _distinct_ shares. */ + share->in_checkpoint= MARIA_CHECKPOINT_SEEN_IN_LOOP; + } + } + if (unlikely((distinct_shares= + (MARIA_SHARE **)my_malloc(nb * sizeof(MARIA_SHARE *), + MYF(MY_WME))) == NULL)) + goto err; + for (total_names_length= 0, i= 0, pos= maria_open_list; pos; pos= pos->next) + { + MARIA_HA *info= (MARIA_HA*)pos->data; + MARIA_SHARE *share= info->s; + if (share->in_checkpoint & MARIA_CHECKPOINT_SEEN_IN_LOOP) + { + distinct_shares[i++]= share; + /* + With this we prevent the share from going away while we later flush + and force it without holding THR_LOCK_maria. For example if the share + could be my_free()d by maria_close() we would have a problem when we + access it to flush the table. We "pin" the share pointer. + And we also take down MARIA_CHECKPOINT_SEEN_IN_LOOP, so that it is + not seen again in the loop. + */ + share->in_checkpoint= MARIA_CHECKPOINT_LOOKS_AT_ME; + /** @todo avoid strlen() */ + total_names_length+= strlen(share->open_file_name); + } + } + + DBUG_ASSERT(i == nb); + pthread_mutex_unlock(&THR_LOCK_maria); + DBUG_PRINT("info",("found %u table shares", nb)); + + str->length= + 4 + /* number of tables */ + (2 + /* short id */ + 4 + /* kfile */ + 4 + /* dfile */ + LSN_STORE_SIZE + /* first_log_write_at_lsn */ + 1 /* end-of-name 0 */ + ) * nb + total_names_length; + if (unlikely((str->str= my_malloc(str->length, MYF(MY_WME))) == NULL)) + goto err; + + ptr= str->str; + ptr+= 4; /* real number of stored tables is not yet know */ + + struct st_filter_param filter_param; + /* only possible checkpointer, so can do the read below without mutex */ + filter_param.up_to_lsn= last_checkpoint_lsn; + PAGECACHE_FILTER filter; + switch(checkpoint_in_progress) + { + case CHECKPOINT_MEDIUM: + filter= &filter_flush_data_file_medium; + break; + case CHECKPOINT_FULL: + filter= &filter_flush_data_file_full; + break; + case CHECKPOINT_INDIRECT: + filter= &filter_flush_data_file_indirect; + break; + default: + DBUG_ASSERT(0); + goto err; + } + + /* + The principle of reading/writing the state below is explained in + ma_recovery.c, look for "Recovery of the state". + */ +#define STATE_COPIES 1024 + state_copies= (struct st_state_copy *) + my_malloc(STATE_COPIES * sizeof(struct st_state_copy), MYF(MY_WME)); + dfiles= (PAGECACHE_FILE *)my_realloc((uchar *)dfiles, + /* avoid size of 0 for my_realloc */ + max(1, nb) * sizeof(PAGECACHE_FILE), + MYF(MY_WME)); + kfiles= (PAGECACHE_FILE *)my_realloc((uchar *)kfiles, + /* avoid size of 0 for my_realloc */ + max(1, nb) * sizeof(PAGECACHE_FILE), + MYF(MY_WME)); + if (unlikely((state_copies == NULL) || + (dfiles == NULL) || (kfiles == NULL))) + goto err; + state_copy= state_copies_end= NULL; + dfiles_end= dfiles; + kfiles_end= kfiles; + + for (nb_stored= 0, i= 0; i < nb; i++) + { + MARIA_SHARE *share= distinct_shares[i]; + PAGECACHE_FILE kfile, dfile; + if (!(share->in_checkpoint & MARIA_CHECKPOINT_LOOKS_AT_ME)) + { + /* No need for a mutex to read the above, only us can write this flag */ + continue; + } + DBUG_PRINT("info",("looking at table '%s'", share->open_file_name)); + if (state_copy == state_copies_end) /* we have no more cached states */ + { + /* + Collect and cache a bunch of states. We do this for many states at a + time, to not lock/unlock the log's lock too often. + */ + uint j, bound= min(nb, i + STATE_COPIES); + state_copy= state_copies; + /* part of the state is protected by log's lock */ + translog_lock(); + state_copies_horizon= translog_get_horizon_no_lock(); + for (j= i; j < bound; j++) + { + MARIA_SHARE *share2= distinct_shares[j]; + if (!(share2->in_checkpoint & MARIA_CHECKPOINT_LOOKS_AT_ME)) + continue; + state_copy->index= j; + state_copy->state= share2->state; /* we copy the state */ + state_copy++; + /* + data_file_length is not updated under log's lock by the bitmap + code, but writing a wrong data_file_length is ok: a next + maria_close() will correct it; if we crash before, Recovery will + set it to the true physical size. + */ + } + translog_unlock(); + state_copies_end= state_copy; + state_copy= state_copies; + /* so now we have cached states */ + } + + /* locate our state among these cached ones */ + for ( ; state_copy->index != i; state_copy++) + DBUG_ASSERT(state_copy < state_copies_end); + + filter_param.pages_covered_by_bitmap= share->bitmap.pages_covered; + /* OS file descriptors are ints which we stored in 4 bytes */ + compile_time_assert(sizeof(int) == 4); + pthread_mutex_lock(&share->intern_lock); + /* + Tables in a normal state have their two file descriptors open. + In some rare cases like REPAIR, some descriptor may be closed or even + -1. If that happened, the _ma_state_info_write() may fail. This is + prevented by enclosing all all places which close/change kfile.file with + intern_lock. + */ + kfile= share->kfile; + dfile= share->bitmap.file; + /* + Ignore table which has no logged writes (all its future log records will + be found naturally by Recovery). Ignore obsolete shares (_before_ + setting themselves to last_version=0 they already did all flush and + sync; if we flush their state now we may be flushing an obsolete state + onto a newer one (assuming the table has been reopened with a different + share but of course same physical index file). + */ + if ((share->id != 0) && (share->last_version != 0)) + { + /** @todo avoid strlen */ + uint open_file_name_len= strlen(share->open_file_name) + 1; + /* remember the descriptors for background flush */ + *(dfiles_end++)= dfile; + *(kfiles_end++)= kfile; + /* we will store this table in the record */ + nb_stored++; + int2store(ptr, share->id); + ptr+= 2; + /* + We must store the OS file descriptors, because the pagecache, which + tells us the list of dirty pages, refers to these pages by OS file + descriptors. An alternative is to make the page cache aware of the + 2-byte id and of the location of a page ("is it a data file page or an + index file page?"). + If one descriptor is -1, normally there should be no dirty pages + collected for this file, it's ok to store -1, it will not be used. + */ + int4store(ptr, kfile.file); + ptr+= 4; + int4store(ptr, dfile.file); + ptr+= 4; + lsn_store(ptr, share->lsn_of_file_id); + ptr+= LSN_STORE_SIZE; + /* + first_bitmap_with_space is not updated under log's lock, and is + important. We would need the bitmap's lock to get it right. Recovery + of this is not clear, so we just play safe: write it out as + unknown: if crash, _ma_bitmap_init() at next open (for example in + Recovery) will convert it to 0 and thus the first insertion will + search for free space from the file's first bitmap (0) - + under-optimal but safe. + If no crash, maria_close() will write the exact value. + */ + state_copy->state.first_bitmap_with_space= ~(ulonglong)0; + memcpy(ptr, share->open_file_name, open_file_name_len); + ptr+= open_file_name_len; + if (cmp_translog_addr(share->state.is_of_horizon, + checkpoint_start_log_horizon) >= 0) + { + /* + State was flushed recently, it does not hold down the log's + low-water mark and will not give avoidable work to Recovery. So we + needn't flush it. Also, it is possible that while we copied the + state above (under log's lock, without intern_lock) it was being + modified in memory or flushed to disk (without log's lock, under + intern_lock, like in maria_extra()), so our copy may be incorrect + and we should not flush it. + It may also be a share which got last_version==0 since we checked + last_version; in this case, it flushed its state and the LSN test + above will catch it. + */ + } + else + { + /* + We could do the state flush only if share->changed, but it's + tricky. + Consider a maria_write() which has written REDO,UNDO, and before it + calls _ma_writeinfo() (setting share->changed=1), checkpoint + happens and sees share->changed=0, does not flush state. It is + possible that Recovery does not start from before the REDO and thus + the state is not recovered. A solution may be to set + share->changed=1 under log mutex when writing log records. + But as anyway we have another problem below, this optimization would + be of little use. + */ + /** @todo flush state only if changed since last checkpoint */ + DBUG_ASSERT(share->last_version != 0); + state_copy->state.is_of_horizon= share->state.is_of_horizon= + state_copies_horizon; + if (kfile.file >= 0) + sync_error|= + _ma_state_info_write_sub(kfile.file, &state_copy->state, 1); + /* + We don't set share->changed=0 because it may interfere with a + concurrent _ma_writeinfo() doing share->changed=1 (cancel its + effect). The sad consequence is that we will flush the same state at + each checkpoint if the table was once written and then not anymore. + */ + } + sync_error|= + _ma_flush_bitmap(share); /* after that, all is in page cache */ + DBUG_ASSERT(share->pagecache == maria_pagecache); + } + if (share->in_checkpoint & MARIA_CHECKPOINT_SHOULD_FREE_ME) + { + /* maria_close() left us to free the share */ + pthread_mutex_unlock(&share->intern_lock); + pthread_mutex_destroy(&share->intern_lock); + my_free((uchar *)share, MYF(0)); + } + else + { + /* share goes back to normal state */ + share->in_checkpoint= 0; + pthread_mutex_unlock(&share->intern_lock); + } + + /* + We do the big disk writes out of intern_lock to not block other + users of this table (intern_lock is taken at the start and end of + every statement). This means that file descriptors may be invalid + (files may have been closed for example by HA_EXTRA_PREPARE_FOR_* + under Windows, or REPAIR). This should not be a problem as we use + MY_IGNORE_BADFD. Descriptors may even point to other files but then + the old blocks (of before the close) must have been flushed for sure, + so our flush will flush new blocks (of after the latest open) and that + should do no harm. + */ + /* + If CHECKPOINT_MEDIUM, this big flush below may result in a + serious write burst. Realize that all pages dirtied between the + last checkpoint and the one we are doing now, will be flushed at + next checkpoint, except those evicted by LRU eviction (depending on + the size of the page cache compared to the size of the working data + set, eviction may be rare or frequent). + We avoid that burst by anticipating: those pages are flushed + in bunches spanned regularly over the time interval between now and + the next checkpoint, by a background thread. Thus the next checkpoint + will have only little flushing to do (CHECKPOINT_MEDIUM should thus be + only a little slower than CHECKPOINT_INDIRECT). + */ + + /** + @todo we ignore the error because it may be just due a pinned page; + we should rather fix the function below to distinguish between + pinned page and write error. Then we can turn the warning into an + error. + */ + if (((filter_param.is_data_file= TRUE), + flush_pagecache_blocks_with_filter(maria_pagecache, + &dfile, FLUSH_KEEP, + filter, &filter_param)) || + ((filter_param.is_data_file= FALSE), + flush_pagecache_blocks_with_filter(maria_pagecache, + &kfile, FLUSH_KEEP, + filter, &filter_param))) + fprintf(stderr, "Maria engine: warning - checkpoint page flush" + " failed\n"); /** @todo improve */ + /* + fsyncs the fd, that's the loooong operation (e.g. max 150 fsync + per second, so if you have touched 1000 files it's 7 seconds). + */ + sync_error|= + my_sync(dfile.file, MYF(MY_WME | MY_IGNORE_BADFD)) | + my_sync(kfile.file, MYF(MY_WME | MY_IGNORE_BADFD)); + /* + in case of error, we continue because writing other tables to disk is + still useful. + */ + } + + if (sync_error) + goto err; + /* We maybe over-estimated (due to share->id==0 or last_version==0) */ + DBUG_ASSERT(str->length >= (uint)(ptr - str->str)); + str->length= (uint)(ptr - str->str); /* - 32-bit CPU, 64-bit reads may give a mixed of old half and new half (old - low bits and new high bits, or the contrary). - As the variable we read transitions from both halves being zero to both - halves being non-zero, and never changes then, we can detect atomicity - problems: + As we support max 65k tables open at a time (2-byte short id), we + assume uint is enough for the cumulated length of table names; and + LEX_STRING::length is uint. */ - ulonglong y; - for (;;) /* loop until no atomicity problems */ + int4store(str->str, nb_stored); + error= unmark_tables= 0; + +err: + if (unlikely(unmark_tables)) { - y= *x; - if (likely(((0 == y) || - ((0 != (y >> 32)) && (0 != (y << 32))))) - return y; - /* Worth seeing it! */ - DBUG_PRINT("info",("atomicity problem")); + /* maria_close() uses THR_LOCK_maria from start to end */ + pthread_mutex_lock(&THR_LOCK_maria); + for (i= 0; i < nb; i++) + { + MARIA_SHARE *share= distinct_shares[i]; + if (share->in_checkpoint & MARIA_CHECKPOINT_SHOULD_FREE_ME) + { + /* maria_close() left us to free the share */ + pthread_mutex_destroy(&share->intern_lock); + my_free((uchar *)share, MYF(0)); + } + else + { + /* share goes back to normal state */ + share->in_checkpoint= 0; + } + } + pthread_mutex_unlock(&THR_LOCK_maria); } -#endif + my_free((uchar *)distinct_shares, MYF(MY_ALLOW_ZERO_PTR)); + my_free((uchar *)state_copies, MYF(MY_ALLOW_ZERO_PTR)); + DBUG_RETURN(error); } diff --git a/storage/maria/ma_checkpoint.h b/storage/maria/ma_checkpoint.h index c011c8234b7..60bbff0b295 100644 --- a/storage/maria/ma_checkpoint.h +++ b/storage/maria/ma_checkpoint.h @@ -32,7 +32,7 @@ typedef enum enum_ma_checkpoint_level { } CHECKPOINT_LEVEL; C_MODE_START -int ma_checkpoint_init(); +int ma_checkpoint_init(my_bool create_background_thread); void ma_checkpoint_end(); int ma_checkpoint_execute(CHECKPOINT_LEVEL level, my_bool no_wait); C_MODE_END diff --git a/storage/maria/ma_create.c b/storage/maria/ma_create.c index b31460d24d2..ba1d9a13b42 100644 --- a/storage/maria/ma_create.c +++ b/storage/maria/ma_create.c @@ -636,7 +636,6 @@ int maria_create(const char *name, enum data_file_type datafile_type, share.state.dellink = HA_OFFSET_ERROR; share.state.first_bitmap_with_space= 0; - share.state.create_rename_lsn= share.state.is_of_lsn= LSN_IMPOSSIBLE; share.state.process= (ulong) getpid(); share.state.unique= (ulong) 0; share.state.update_count=(ulong) 0; @@ -1006,7 +1005,7 @@ int maria_create(const char *name, enum data_file_type datafile_type, DROP+CREATE happened (applying REDOs to the wrong table). */ share.kfile.file= file; - if (_ma_update_create_rename_lsn_on_disk_sub(&share, lsn, FALSE)) + if (_ma_update_create_rename_lsn_sub(&share, lsn, FALSE)) goto err; my_free(log_data, MYF(0)); } @@ -1070,7 +1069,9 @@ int maria_create(const char *name, enum data_file_type datafile_type, if (my_chsize(dfile,share.base.min_pack_length*ci->reloc_rows,0,MYF(0))) goto err; #endif - if ((sync_dir && my_sync(dfile, MYF(0))) || my_close(dfile,MYF(0))) + if (sync_dir && my_sync(dfile, MYF(0))) + goto err; + if (my_close(dfile,MYF(0))) goto err; } pthread_mutex_unlock(&THR_LOCK_maria); @@ -1207,7 +1208,7 @@ int _ma_initialize_data_file(MARIA_SHARE *share, File dfile) /** - @brief Writes create_rename_lsn and is_of_lsn to disk, optionally forces. + @brief Writes create_rename_lsn and is_of_horizon to disk, can force. This is for special cases where: - we don't want to write the full state to disk (so, not call @@ -1224,21 +1225,21 @@ int _ma_initialize_data_file(MARIA_SHARE *share, File dfile) @retval 1 error (disk problem) */ -int _ma_update_create_rename_lsn_on_disk(MARIA_SHARE *share, - LSN lsn, my_bool do_sync) +int _ma_update_create_rename_lsn(MARIA_SHARE *share, + LSN lsn, my_bool do_sync) { int res; pthread_mutex_lock(&share->intern_lock); - res= _ma_update_create_rename_lsn_on_disk_sub(share, lsn, do_sync); + res= _ma_update_create_rename_lsn_sub(share, lsn, do_sync); pthread_mutex_unlock(&share->intern_lock); return res; } /** - @brief Writes create_rename_lsn and is_of_lsn to disk, optionally forces. + @brief Writes create_rename_lsn and is_of_horizon to disk, can force. - Shortcut of _ma_update_create_rename_lsn_on_disk() when we know that + Shortcut of _ma_update_create_rename_lsn() when we know that intern_lock is not needed (when creating a table or opening it for the first time). @@ -1250,15 +1251,28 @@ int _ma_update_create_rename_lsn_on_disk(MARIA_SHARE *share, @retval 1 error (disk problem) */ -int _ma_update_create_rename_lsn_on_disk_sub(MARIA_SHARE *share, - LSN lsn, my_bool do_sync) +int _ma_update_create_rename_lsn_sub(MARIA_SHARE *share, + LSN lsn, my_bool do_sync) { char buf[LSN_STORE_SIZE*2], *ptr; File file= share->kfile.file; DBUG_ASSERT(file >= 0); for (ptr= buf; ptr < (buf + sizeof(buf)); ptr+= LSN_STORE_SIZE) lsn_store(ptr, lsn); - share->state.is_of_lsn= share->state.create_rename_lsn= lsn; + share->state.is_of_horizon= share->state.create_rename_lsn= lsn; + if (share->id != 0) + { + /* + If OP is the operation which is calling us, if table is later written, + we could see in the log: + FILE_ID ... REDO_OP ... REDO_INSERT. + (that can happen in real life at least with OP=REPAIR). + As FILE_ID will be ignored by Recovery because it is < + create_rename_lsn, REDO_INSERT would be ignored too, wrongly. + To avoid that, we force a LOGREC_FILE_ID to be logged at next write: + */ + translog_deassign_id_from_share(share); + } return my_pwrite(file, buf, sizeof(buf), sizeof(share->state.header) + 2, MYF(MY_NABP)) || (do_sync && my_sync(file, MYF(0))); diff --git a/storage/maria/ma_extra.c b/storage/maria/ma_extra.c index f72a92c7506..4f1634756ab 100644 --- a/storage/maria/ma_extra.c +++ b/storage/maria/ma_extra.c @@ -297,8 +297,10 @@ int maria_extra(MARIA_HA *info, enum ha_extra_function function, HA_EXTRA_PREPARE_FOR_DROP|RENAME. */ pthread_mutex_lock(&THR_LOCK_maria); + pthread_mutex_lock(&share->intern_lock); /* protect against Checkpoint */ /* this makes the share not be re-used next time the table is opened */ share->last_version= 0L; /* Impossible version */ + pthread_mutex_unlock(&share->intern_lock); pthread_mutex_unlock(&THR_LOCK_maria); break; case HA_EXTRA_PREPARE_FOR_DROP: @@ -306,9 +308,8 @@ int maria_extra(MARIA_HA *info, enum ha_extra_function function, { my_bool do_flush= test(function != HA_EXTRA_PREPARE_FOR_DROP); pthread_mutex_lock(&THR_LOCK_maria); - share->last_version= 0L; /* Impossible version */ /* - This share, having last_version=0, needs to save all its data/index + This share, to have last_version=0, needs to save all its data/index blocks to disk if this is not for a DROP TABLE. Otherwise they would be invisible to future openers; and they could even go to disk late and cancel the work of future openers. @@ -396,6 +397,8 @@ int maria_extra(MARIA_HA *info, enum ha_extra_function function, } } #endif + /* For protection against Checkpoint, we set under intern_lock: */ + share->last_version= 0L; /* Impossible version */ pthread_mutex_unlock(&share->intern_lock); pthread_mutex_unlock(&THR_LOCK_maria); break; diff --git a/storage/maria/ma_init.c b/storage/maria/ma_init.c index 1cd82720260..fb8efddd778 100644 --- a/storage/maria/ma_init.c +++ b/storage/maria/ma_init.c @@ -19,6 +19,7 @@ #include #include "ma_blockrec.h" #include "trnman_public.h" +#include "ma_checkpoint.h" my_bool maria_inited= FALSE; pthread_mutex_t THR_LOCK_maria; @@ -56,6 +57,7 @@ void maria_end(void) { maria_inited= maria_multi_threaded= FALSE; ft_free_stopwords(); + ma_checkpoint_end(); trnman_destroy(); translog_destroy(); end_pagecache(maria_log_pagecache, TRUE); diff --git a/storage/maria/ma_least_recently_dirtied.c b/storage/maria/ma_least_recently_dirtied.c deleted file mode 100644 index 3d2c85bbf98..00000000000 --- a/storage/maria/ma_least_recently_dirtied.c +++ /dev/null @@ -1,105 +0,0 @@ -/* Copyright (C) 2006 MySQL AB & MySQL Finland AB & TCX DataKonsult AB - - 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ - -/* - WL#3261 Maria - background flushing of the least-recently-dirtied pages - First version written by Guilhem Bichot on 2006-04-27. - Does not compile yet. -*/ - -/* - To be part of the page cache. - The pseudocode below is dependent on the page cache - which is being designed WL#3134. It is not clear if I need to do page - copies, as the page cache already keeps page copies. - So, this code will move to the page cache and take inspiration from its - methods. Below is just to give the idea of what could be done. - And I should compare my imaginations to WL#3134. -*/ - -/* Here is the implementation of this module */ - -#include "page_cache.h" -#include "least_recently_dirtied.h" - -/* - This thread does background flush of pieces of the LRD, and serves - requests for asynchronous checkpoints. - Just launch it when engine starts. - MikaelR questioned why the same thread does two different jobs, the risk - could be that while a checkpoint happens no LRD flushing happens. - For now, we only do checkpoints - no LRD flushing (to be done when the - second version of the page cache is ready WL#3077). - Reasons to delay: - - Recovery will work (just slower) - - new page cache may be different, why do then re-do - - current pagecache probably has issues with flushing when somebody is - writing to the table being flushed - better avoid that. -*/ -pthread_handler_decl background_flush_and_checkpoint_thread() -{ - while (this_thread_not_killed) - { - /* note that we don't care of the checkpoint's success */ - (void)execute_asynchronous_checkpoint_if_any(); - sleep(5); - /* - in the final version, we will not sleep but call flush_pages_from_LRD() - repeatedly. If there are no dirty pages, we'll make sure to not have a - tight loop probing for checkpoint requests. - */ - } -} - -/* The rest of this file will not serve in first version */ - -/* - flushes only the first pages of the LRD. - max_this_number could be FLUSH_CACHE (of mf_pagecache.c) for example. -*/ -flush_pages_from_LRD(uint max_this_number, LSN max_this_lsn) -{ - /* - One rule to better observe is "page must be flushed to disk before it is - removed from LRD" (otherwise checkpoint is incomplete info, corruption). - */ - - /* - Build a list of pages to flush: - changed_blocks[i] is roughly sorted by descending rec_lsn, - so we could do a merge sort of changed_blocks[] lists, stopping after we - have the max_this_number first elements or after we have found a page with - rec_lsn > max_this_lsn. - Then do like pagecache_flush_blocks_int() does (beware! this time we are - not alone on the file! there may be dangers! TODO: sort this out). - */ - - /* - MikaelR noted that he observed that Linux's file cache may never fsync to - disk until this cache is full, at which point it decides to empty the - cache, making the machine very slow. A solution was to fsync after writing - 2 MB. - */ -} - -/* - Note that when we flush all page from LRD up to rec_lsn>=max_lsn, - this is approximate because the LRD list may - not be exactly sorted by rec_lsn (because for a big row, all pages of the - row are inserted into the LRD with rec_lsn being the LSN of the REDO for the - first page, so if there are concurrent insertions, the last page of the big - row may have a smaller rec_lsn than the previous pages inserted by - concurrent inserters). -*/ diff --git a/storage/maria/ma_least_recently_dirtied.h b/storage/maria/ma_least_recently_dirtied.h deleted file mode 100644 index 1d57f3596f8..00000000000 --- a/storage/maria/ma_least_recently_dirtied.h +++ /dev/null @@ -1,25 +0,0 @@ -/* Copyright (C) 2006 MySQL AB & MySQL Finland AB & TCX DataKonsult AB - - 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ - -/* - WL#3261 Maria - background flushing of the least-recently-dirtied pages - First version written by Guilhem Bichot on 2006-04-27. - Does not compile yet. -*/ - -/* This is the interface of this module. */ - -/* flushes all page from LRD up to approximately rec_lsn>=max_lsn */ -int flush_all_LRD_to_lsn(LSN max_lsn); diff --git a/storage/maria/ma_loghandler.c b/storage/maria/ma_loghandler.c index 55635bdab93..f42af62d202 100644 --- a/storage/maria/ma_loghandler.c +++ b/storage/maria/ma_loghandler.c @@ -225,13 +225,12 @@ static my_bool write_hook_for_undo_row_delete(enum translog_record_type type, TRN *trn, MARIA_HA *tbl_info, LSN *lsn, struct st_translog_parts *parts); -static my_bool write_hook_for_undo_row_purge(enum translog_record_type type, - TRN *trn, MARIA_HA *tbl_info, - LSN *lsn, - struct st_translog_parts *parts); static my_bool write_hook_for_clr_end(enum translog_record_type type, TRN *trn, MARIA_HA *tbl_info, LSN *lsn, struct st_translog_parts *parts); +static my_bool write_hook_for_file_id(enum translog_record_type type, + TRN *trn, MARIA_HA *tbl_info, LSN *lsn, + struct st_translog_parts *parts); static my_bool translog_page_validator(uchar *page_addr, uchar* data_ptr); @@ -386,10 +385,12 @@ static LOG_DESC INIT_LOGREC_REDO_INSERT_ROW_TAIL= write_hook_for_redo, NULL, 0, "redo_insert_row_tail", LOGREC_NOT_LAST_IN_GROUP, NULL, NULL}; +/** @todo RECOVERY BUG unused, remove? */ static LOG_DESC INIT_LOGREC_REDO_INSERT_ROW_BLOB= {LOGRECTYPE_VARIABLE_LENGTH, 0, 8, NULL, write_hook_for_redo, NULL, 0, "redo_insert_row_blob", LOGREC_NOT_LAST_IN_GROUP, NULL, NULL}; +/** @todo RECOVERY BUG handle it in recovery */ /*QQQ:TODO:header???*/ static LOG_DESC INIT_LOGREC_REDO_INSERT_ROW_BLOBS= {LOGRECTYPE_VARIABLE_LENGTH, 0, FILEID_STORE_SIZE, NULL, @@ -416,10 +417,12 @@ static LOG_DESC INIT_LOGREC_REDO_PURGE_BLOCKS= NULL, write_hook_for_redo, NULL, 0, "redo_purge_blocks", LOGREC_NOT_LAST_IN_GROUP, NULL, NULL}; +/* not yet used; for when we have versioning */ static LOG_DESC INIT_LOGREC_REDO_DELETE_ROW= {LOGRECTYPE_FIXEDLENGTH, 16, 16, NULL, write_hook_for_redo, NULL, 0, "redo_delete_row", LOGREC_NOT_LAST_IN_GROUP, NULL, NULL}; +/** @todo RECOVERY BUG unused, remove? */ static LOG_DESC INIT_LOGREC_REDO_UPDATE_ROW_HEAD= {LOGRECTYPE_VARIABLE_LENGTH, 0, 9, NULL, write_hook_for_redo, NULL, 0, "redo_update_row_head", LOGREC_NOT_LAST_IN_GROUP, NULL, NULL}; @@ -460,12 +463,6 @@ static LOG_DESC INIT_LOGREC_UNDO_ROW_UPDATE= NULL, write_hook_for_undo, NULL, 1, "undo_row_update", LOGREC_LAST_IN_GROUP, NULL, NULL}; -static LOG_DESC INIT_LOGREC_UNDO_ROW_PURGE= -{LOGRECTYPE_PSEUDOFIXEDLENGTH, LSN_STORE_SIZE + FILEID_STORE_SIZE, - LSN_STORE_SIZE + FILEID_STORE_SIZE, - NULL, write_hook_for_undo_row_purge, NULL, 1, - "undo_row_purge", LOGREC_LAST_IN_GROUP, NULL, NULL}; - static LOG_DESC INIT_LOGREC_UNDO_KEY_INSERT= {LOGRECTYPE_VARIABLE_LENGTH, 0, 10, NULL, write_hook_for_undo, NULL, 1, "undo_key_insert", LOGREC_LAST_IN_GROUP, NULL, NULL}; @@ -518,7 +515,7 @@ static LOG_DESC INIT_LOGREC_REDO_REPAIR_TABLE= "redo_repair_table", LOGREC_IS_GROUP_ITSELF, NULL, NULL}; static LOG_DESC INIT_LOGREC_FILE_ID= -{LOGRECTYPE_VARIABLE_LENGTH, 0, 2, NULL, NULL, NULL, 0, +{LOGRECTYPE_VARIABLE_LENGTH, 0, 2, NULL, write_hook_for_file_id, NULL, 0, "file_id", LOGREC_IS_GROUP_ITSELF, NULL, NULL}; static LOG_DESC INIT_LOGREC_LONG_TRANSACTION_ID= @@ -564,8 +561,6 @@ static void loghandler_init() INIT_LOGREC_UNDO_ROW_DELETE; log_record_type_descriptor[LOGREC_UNDO_ROW_UPDATE]= INIT_LOGREC_UNDO_ROW_UPDATE; - log_record_type_descriptor[LOGREC_UNDO_ROW_PURGE]= - INIT_LOGREC_UNDO_ROW_PURGE; log_record_type_descriptor[LOGREC_UNDO_KEY_INSERT]= INIT_LOGREC_UNDO_KEY_INSERT; log_record_type_descriptor[LOGREC_UNDO_KEY_DELETE]= @@ -4941,7 +4936,7 @@ my_bool translog_write_record(LSN *lsn, log records are written; for example SELECT FOR UPDATE takes locks but writes no log record. */ - if (unlikely(translog_assign_id_to_share(share, trn))) + if (unlikely(translog_assign_id_to_share(tbl_info, trn))) DBUG_RETURN(1); } fileid_store(store_share_id, share->id); @@ -5156,6 +5151,20 @@ TRANSLOG_ADDRESS translog_get_horizon() } +/** + @brief Returns the current horizon at the end of the current log, caller is + assumed to already hold the lock + + @return Horizon +*/ + +TRANSLOG_ADDRESS translog_get_horizon_no_lock() +{ + translog_lock_assert_owner(); + return log_descriptor.horizon; +} + + /* Set last page in the scanner data structure @@ -5616,7 +5625,9 @@ int translog_read_record_header_from_buffer(uchar *page, res= translog_fixed_length_header(page, page_offset, buff); break; default: - DBUG_ASSERT(0); +#ifdef ASK_SANJA + DBUG_ASSERT(0); /* fails on empty log (Sanja knows) */ +#endif res= RECHEADER_READ_ERROR; } DBUG_RETURN(res); @@ -5877,13 +5888,14 @@ static my_bool translog_record_read_next_chunk(struct st_translog_reader_data static my_bool translog_init_reader_data(LSN lsn, struct st_translog_reader_data *data) { + int read_header; DBUG_ENTER("translog_init_reader_data"); if (translog_init_scanner(lsn, 1, &data->scanner) || - !(data->read_header= - translog_read_record_header_scan(&data->scanner, &data->header, 1))) - { + ((read_header= + translog_read_record_header_scan(&data->scanner, &data->header, 1)) + == RECHEADER_READ_ERROR)) DBUG_RETURN(1); - } + data->read_header= read_header; data->body_offset= data->header.non_header_data_start_offset; data->chunk_size= data->header.non_header_data_len; data->current_offset= data->read_header; @@ -6384,26 +6396,6 @@ static my_bool write_hook_for_undo_row_delete(enum translog_record_type type } -/** - @brief Upates "records" and calls the generic UNDO hook - - @todo we will get rid of this record soon. - - @return Operation status, always 0 (success) -*/ - -static my_bool write_hook_for_undo_row_purge(enum translog_record_type type - __attribute__ ((unused)), - TRN *trn, MARIA_HA *tbl_info, - LSN *lsn, - struct st_translog_parts *parts - __attribute__ ((unused))) -{ - tbl_info->s->state.state.records--; - return write_hook_for_undo(type, trn, tbl_info, lsn, parts); -} - - /** @brief Sets transaction's undo_lsn, first_undo_lsn if needed @@ -6425,7 +6417,6 @@ static my_bool write_hook_for_clr_end(enum translog_record_type type ptr[LSN_STORE_SIZE + FILEID_STORE_SIZE]; DBUG_ASSERT(trn->trid != 0); - /** @todo depending on what we are undoing, update "records" or not */ trn->undo_lsn= lsn_korr(ptr); switch (undone_record_type) { case LOGREC_UNDO_ROW_DELETE: @@ -6445,6 +6436,30 @@ static my_bool write_hook_for_clr_end(enum translog_record_type type } +/** + @brief Updates table's lsn_of_file_id. + + @todo move it to a separate file + + @return Operation status, always 0 (success) +*/ + +static my_bool write_hook_for_file_id(enum translog_record_type type + __attribute__ ((unused)), + TRN *trn + __attribute__ ((unused)), + MARIA_HA *tbl_info, + LSN *lsn + __attribute__ ((unused)), + struct st_translog_parts *parts + __attribute__ ((unused))) +{ + DBUG_ASSERT(cmp_translog_addr(tbl_info->s->lsn_of_file_id, *lsn) < 0); + tbl_info->s->lsn_of_file_id= *lsn; + return 0; +} + + /** @brief Gives a 2-byte-id to MARIA_SHARE and logs this fact @@ -6452,7 +6467,7 @@ static my_bool write_hook_for_clr_end(enum translog_record_type type open MARIA_SHAREs), give it one and record this assignment in the log (LOGREC_FILE_ID log record). - @param share table + @param tbl_info table @param trn calling transaction @return Operation status @@ -6462,8 +6477,9 @@ static my_bool write_hook_for_clr_end(enum translog_record_type type @note Can be called even if share already has an id (then will do nothing) */ -int translog_assign_id_to_share(MARIA_SHARE *share, TRN *trn) +int translog_assign_id_to_share(MARIA_HA *tbl_info, TRN *trn) { + MARIA_SHARE *share= tbl_info->s; /* If you give an id to a non-BLOCK_RECORD table, you also need to release this id somewhere. Then you can change the assertion. @@ -6495,7 +6511,6 @@ int translog_assign_id_to_share(MARIA_SHARE *share, TRN *trn) LSN lsn; LEX_STRING log_array[TRANSLOG_INTERNAL_PARTS + 2]; uchar log_data[FILEID_STORE_SIZE]; - fileid_store(log_data, share->id); log_array[TRANSLOG_INTERNAL_PARTS + 0].str= (char*) log_data; log_array[TRANSLOG_INTERNAL_PARTS + 0].length= sizeof(log_data); /* @@ -6510,22 +6525,13 @@ int translog_assign_id_to_share(MARIA_SHARE *share, TRN *trn) */ log_array[TRANSLOG_INTERNAL_PARTS + 1].length= strlen(share->open_file_name) + 1; - if (unlikely(translog_write_record(&lsn, LOGREC_FILE_ID, trn, NULL, + if (unlikely(translog_write_record(&lsn, LOGREC_FILE_ID, trn, tbl_info, sizeof(log_data) + log_array[TRANSLOG_INTERNAL_PARTS + 1].length, sizeof(log_array)/sizeof(log_array[0]), - log_array, NULL))) + log_array, log_data))) return 1; - /* - Note that we first set share->id then write the record. The checkpoint - record does not include any share with id==0; this is ok because: - checkpoint_start_log_horizon is either before or after the above - record. If before, ok to not include the share, as the record will be - seen for sure during the REDO phase. If after, Checkpoint will see all - data as it was after this record was written, including the id!=0, so - share will be included. - */ } pthread_mutex_unlock(&share->intern_lock); return 0; @@ -6546,12 +6552,17 @@ void translog_deassign_id_from_share(MARIA_SHARE *share) (ulong)share, share->id)); /* We don't need any mutex as we are called only when closing the last - instance of the table: no writes can be happening. + instance of the table or at the end of REPAIR: no writes can be + happening. But a Checkpoint may be reading share->id, so we require this + mutex: */ + safe_mutex_assert_owner(&share->intern_lock); my_atomic_rwlock_rdlock(&LOCK_id_to_share); my_atomic_storeptr((void **)&id_to_share[share->id], 0); my_atomic_rwlock_rdunlock(&LOCK_id_to_share); share->id= 0; + /* useless but safety: */ + share->lsn_of_file_id= LSN_IMPOSSIBLE; } @@ -6733,13 +6744,13 @@ LSN translog_first_theoretical_lsn() /** @brief Check given low water mark and purge files if it is need - @param low the last (minimum) LSN which is need + @param low the last (minimum) address which is need @retval 0 OK @retval 1 Error */ -my_bool translog_purge(LSN low) +my_bool translog_purge(TRANSLOG_ADDRESS low) { uint32 last_need_file= LSN_FILE_NO(low); TRANSLOG_ADDRESS horizon= translog_get_horizon(); diff --git a/storage/maria/ma_loghandler.h b/storage/maria/ma_loghandler.h index 5c014fe05af..b7e3be18fb3 100644 --- a/storage/maria/ma_loghandler.h +++ b/storage/maria/ma_loghandler.h @@ -105,7 +105,6 @@ enum translog_record_type LOGREC_UNDO_ROW_INSERT, LOGREC_UNDO_ROW_DELETE, LOGREC_UNDO_ROW_UPDATE, - LOGREC_UNDO_ROW_PURGE, LOGREC_UNDO_KEY_INSERT, LOGREC_UNDO_KEY_DELETE, LOGREC_PREPARE, @@ -251,13 +250,14 @@ extern my_bool translog_init_scanner(LSN lsn, extern int translog_read_next_record_header(TRANSLOG_SCANNER_DATA *scanner, TRANSLOG_HEADER_BUFFER *buff); extern LSN translog_get_file_max_lsn_stored(uint32 file); -extern my_bool translog_purge(LSN low); +extern my_bool translog_purge(TRANSLOG_ADDRESS low); extern my_bool translog_is_file(uint file_no); extern my_bool translog_lock(); extern my_bool translog_unlock(); extern void translog_lock_assert_owner(); extern TRANSLOG_ADDRESS translog_get_horizon(); -extern int translog_assign_id_to_share(struct st_maria_share *share, +extern TRANSLOG_ADDRESS translog_get_horizon_no_lock(); +extern int translog_assign_id_to_share(struct st_maria_info *tbl_info, struct st_transaction *trn); extern void translog_deassign_id_from_share(struct st_maria_share *share); extern void diff --git a/storage/maria/ma_open.c b/storage/maria/ma_open.c index cd57f6f0b11..9b665cfb958 100644 --- a/storage/maria/ma_open.c +++ b/storage/maria/ma_open.c @@ -618,14 +618,13 @@ MARIA_HA *maria_open(const char *name, int mode, uint open_flags) view of the server, including server's recovery) now. */ if ((open_flags & HA_OPEN_FROM_SQL_LAYER) || maria_in_recovery) - _ma_update_create_rename_lsn_on_disk_sub(share, - translog_get_horizon(), - TRUE); + _ma_update_create_rename_lsn_sub(share, translog_get_horizon(), + TRUE); } else if ((!LSN_VALID(share->state.create_rename_lsn) || - !LSN_VALID(share->state.is_of_lsn) || + !LSN_VALID(share->state.is_of_horizon) || (cmp_translog_addr(share->state.create_rename_lsn, - share->state.is_of_lsn) > 0)) && + share->state.is_of_horizon) > 0)) && !(open_flags & HA_OPEN_FOR_REPAIR)) { /* @@ -981,7 +980,7 @@ static void setup_key_functions(register MARIA_KEYDEF *keyinfo) @brief Function to save and store the header in the index file (.MYI) Operates under MARIA_SHARE::intern_lock if requested. - Sets MARIA_SHARE::MARIA_STATE_INFO::is_of_lsn if table is transactional. + Sets MARIA_SHARE::MARIA_STATE_INFO::is_of_horizon if transactional table. Then calls _ma_state_info_write_sub(). @param share table @@ -998,7 +997,7 @@ static void setup_key_functions(register MARIA_KEYDEF *keyinfo) uint _ma_state_info_write(MARIA_SHARE *share, uint pWrite) { - uint res= 0; + uint res; if (pWrite & 4) pthread_mutex_lock(&share->intern_lock); else if (maria_multi_threaded) @@ -1007,11 +1006,11 @@ uint _ma_state_info_write(MARIA_SHARE *share, uint pWrite) !maria_in_recovery) { /* - In a recovery, we want to set is_of_lsn to the LSN of the last + In a recovery, we want to set is_of_horizon to the LSN of the last record executed by Recovery, not the current EOF of the log (which is too new). Recovery does it by itself. */ - share->state.is_of_lsn= translog_get_horizon(); + share->state.is_of_horizon= translog_get_horizon(); } res= _ma_state_info_write_sub(share->kfile.file, &share->state, pWrite); if (pWrite & 4) @@ -1052,11 +1051,12 @@ uint _ma_state_info_write_sub(File file, MARIA_STATE_INFO *state, uint pWrite) /* open_count must be first because of _ma_mark_file_changed ! */ mi_int2store(ptr,state->open_count); ptr+= 2; /* - if you change the offset of create_rename_lsn/is_of_lsn inside the file, - fix ma_create + ma_rename + ma_delete_all + backward-compatibility. + if you change the offset of create_rename_lsn/is_of_horizon inside the + index file's header, fix ma_create + ma_rename + ma_delete_all + + backward-compatibility. */ lsn_store(ptr, state->create_rename_lsn); ptr+= LSN_STORE_SIZE; - lsn_store(ptr, state->is_of_lsn); ptr+= LSN_STORE_SIZE; + lsn_store(ptr, state->is_of_horizon); ptr+= LSN_STORE_SIZE; *ptr++= (uchar)state->changed; *ptr++= state->sortkey; mi_rowstore(ptr,state->state.records); ptr+= 8; @@ -1119,7 +1119,7 @@ static uchar *_ma_state_info_read(uchar *ptr, MARIA_STATE_INFO *state) state->open_count = mi_uint2korr(ptr); ptr+= 2; state->create_rename_lsn= lsn_korr(ptr); ptr+= LSN_STORE_SIZE; - state->is_of_lsn= lsn_korr(ptr); ptr+= LSN_STORE_SIZE; + state->is_of_horizon= lsn_korr(ptr); ptr+= LSN_STORE_SIZE; state->changed= (my_bool) *ptr++; state->sortkey= (uint) *ptr++; state->state.records= mi_rowkorr(ptr); ptr+= 8; diff --git a/storage/maria/ma_pagecache.c b/storage/maria/ma_pagecache.c index d76d59f32ba..792f0d645ab 100755 --- a/storage/maria/ma_pagecache.c +++ b/storage/maria/ma_pagecache.c @@ -3865,8 +3865,6 @@ int reset_pagecache_counters(const char *name, PAGECACHE *pagecache) its size, will be put @param[out] min_rec_lsn pointer to where the minimum rec_lsn of all relevant dirty pages will be put - @param[out] max_rec_lsn pointer to where the maximum rec_lsn of all - relevant dirty pages will be put @return Operation status @retval 0 OK @retval 1 Error @@ -3874,14 +3872,13 @@ int reset_pagecache_counters(const char *name, PAGECACHE *pagecache) my_bool pagecache_collect_changed_blocks_with_lsn(PAGECACHE *pagecache, LEX_STRING *str, - LSN *min_rec_lsn, - LSN *max_rec_lsn) + LSN *min_rec_lsn) { my_bool error= 0; - ulong stored_list_size= 0; + uint stored_list_size= 0; uint file_hash; char *ptr; - LSN minimum_rec_lsn= LSN_MAX, maximum_rec_lsn= 0; + LSN minimum_rec_lsn= LSN_MAX; DBUG_ENTER("pagecache_collect_changed_blocks_with_LSN"); DBUG_ASSERT(NULL == str->str); @@ -3921,7 +3918,8 @@ my_bool pagecache_collect_changed_blocks_with_lsn(PAGECACHE *pagecache, } } - str->length= 8 + /* number of dirty pages */ + compile_time_assert(sizeof(pagecache->blocks == 4)); + str->length= 4 + /* number of dirty pages */ (4 + /* file */ 4 + /* pageno */ LSN_STORE_SIZE /* rec_lsn */ @@ -3929,8 +3927,8 @@ my_bool pagecache_collect_changed_blocks_with_lsn(PAGECACHE *pagecache, if (NULL == (str->str= my_malloc(str->length, MYF(MY_WME)))) goto err; ptr= str->str; - int8store(ptr, stored_list_size); - ptr+= 8; + int4store(ptr, stored_list_size); + ptr+= 4; if (!stored_list_size) goto end; for (file_hash= 0; file_hash < PAGECACHE_CHANGED_BLOCKS_HASH; file_hash++) @@ -3955,15 +3953,12 @@ my_bool pagecache_collect_changed_blocks_with_lsn(PAGECACHE *pagecache, DBUG_ASSERT(LSN_VALID(block->rec_lsn)); if (cmp_translog_addr(block->rec_lsn, minimum_rec_lsn) < 0) minimum_rec_lsn= block->rec_lsn; - if (cmp_translog_addr(block->rec_lsn, maximum_rec_lsn) > 0) - maximum_rec_lsn= block->rec_lsn; } /* otherwise, some trn->rec_lsn should hold the correct info */ } } end: pagecache_pthread_mutex_unlock(&pagecache->cache_lock); *min_rec_lsn= minimum_rec_lsn; - *max_rec_lsn= maximum_rec_lsn; DBUG_RETURN(error); err: diff --git a/storage/maria/ma_pagecache.h b/storage/maria/ma_pagecache.h index 78dd555776d..0e2aff3644d 100644 --- a/storage/maria/ma_pagecache.h +++ b/storage/maria/ma_pagecache.h @@ -247,8 +247,7 @@ extern my_bool pagecache_delete_pages(PAGECACHE *pagecache, extern void end_pagecache(PAGECACHE *keycache, my_bool cleanup); extern my_bool pagecache_collect_changed_blocks_with_lsn(PAGECACHE *pagecache, LEX_STRING *str, - LSN *min_lsn, - LSN *max_lsn); + LSN *min_lsn); extern int reset_pagecache_counters(const char *name, PAGECACHE *pagecache); diff --git a/storage/maria/ma_recovery.c b/storage/maria/ma_recovery.c index baee227e7ca..508615bc65f 100644 --- a/storage/maria/ma_recovery.c +++ b/storage/maria/ma_recovery.c @@ -63,6 +63,7 @@ static FILE *tracef; /**< trace file for debugging */ prototype_redo_exec_hook(LONG_TRANSACTION_ID); prototype_redo_exec_hook_dummy(CHECKPOINT); prototype_redo_exec_hook(REDO_CREATE_TABLE); +prototype_redo_exec_hook(REDO_RENAME_TABLE); prototype_redo_exec_hook(REDO_DROP_TABLE); prototype_redo_exec_hook(FILE_ID); prototype_redo_exec_hook(REDO_INSERT_ROW_HEAD); @@ -74,7 +75,6 @@ prototype_redo_exec_hook(REDO_DELETE_ALL); prototype_redo_exec_hook(UNDO_ROW_INSERT); prototype_redo_exec_hook(UNDO_ROW_DELETE); prototype_redo_exec_hook(UNDO_ROW_UPDATE); -prototype_redo_exec_hook(UNDO_ROW_PURGE); prototype_redo_exec_hook(COMMIT); prototype_redo_exec_hook(CLR_END); prototype_undo_exec_hook(UNDO_ROW_INSERT); @@ -93,12 +93,13 @@ static MARIA_HA *get_MARIA_HA_from_REDO_record(const TRANSLOG_HEADER_BUFFER *rec); static MARIA_HA *get_MARIA_HA_from_UNDO_record(const TRANSLOG_HEADER_BUFFER *rec); -static void prepare_table_for_close(MARIA_HA *info, LSN at_lsn); -static int parse_checkpoint_record(LSN lsn); +static void prepare_table_for_close(MARIA_HA *info, TRANSLOG_ADDRESS horizon); +static LSN parse_checkpoint_record(LSN lsn); static void new_transaction(uint16 sid, TrID long_id, LSN undo_lsn, LSN first_undo_lsn); static int new_table(uint16 sid, const char *name, - File org_kfile, File org_dfile, LSN lsn); + File org_kfile, File org_dfile, + LSN lsn_of_file_id); static int new_page(File fileid, pgcache_page_no_t pageid, LSN rec_lsn, struct st_dirty_page *dirty_page); static int close_all_tables(); @@ -124,6 +125,10 @@ static LEX_STRING log_record_buffer; of runtime: recreates transactions inside trnman, open tables with their two-byte-id mapping; takes a checkpoint and runs the UNDO phase. Closes all tables. + + @return Operation status + @retval 0 OK + @retval !=0 Error */ int maria_recover() @@ -201,7 +206,6 @@ int maria_apply_log(LSN from_lsn, my_bool apply, FILE *trace_file, } else { - DBUG_ASSERT(0); /* not yet implemented */ from_lsn= parse_checkpoint_record(last_checkpoint_lsn); if (from_lsn == LSN_IMPOSSIBLE) goto err; @@ -230,10 +234,7 @@ int maria_apply_log(LSN from_lsn, my_bool apply, FILE *trace_file, if (close_all_tables()) goto err; - /* - At this stage, end of recovery, trnman is left initialized. This is for - the future, when we have an online UNDO phase or prepared transactions. - */ + /* If inside ha_maria, a checkpoint will soon be taken and save our work */ goto end; err: error= 1; @@ -426,7 +427,7 @@ prototype_redo_exec_hook(REDO_CREATE_TABLE) ptr+= 2; /* set create_rename_lsn (for maria_read_log to be idempotent) */ lsn_store(ptr + sizeof(info->s->state.header) + 2, rec->lsn); - /* we also set is_of_lsn, like maria_create() does */ + /* we also set is_of_horizon, like maria_create() does */ lsn_store(ptr + sizeof(info->s->state.header) + 2 + LSN_STORE_SIZE, rec->lsn); if (my_pwrite(kfile, ptr, @@ -474,6 +475,77 @@ end: } +prototype_redo_exec_hook(REDO_RENAME_TABLE) +{ + char *old_name, *new_name; + int error= 1; + MARIA_HA *info= NULL; + enlarge_buffer(rec); + if (log_record_buffer.str == NULL || + translog_read_record(rec->lsn, 0, rec->record_length, + log_record_buffer.str, NULL) != + rec->record_length) + { + fprintf(tracef, "Failed to read record\n"); + goto end; + } + old_name= log_record_buffer.str; + new_name= old_name + strlen(old_name) + 1; + fprintf(tracef, "Table '%s' to rename to '%s'", old_name, new_name); + info= maria_open(old_name, O_RDONLY, HA_OPEN_FOR_REPAIR); + if (info) + { + MARIA_SHARE *share= info->s; + /* + We may have open instances on this table. But it does not matter, the + maria_extra() below will take care of them. + */ + if (!share->base.born_transactional) + { + fprintf(tracef, ", is not transactional\n"); + ALERT_USER(); + error= 0; + goto end; + } + if (cmp_translog_addr(share->state.create_rename_lsn, rec->lsn) >= 0) + { + fprintf(tracef, ", has create_rename_lsn (%lu,0x%lx) more recent than" + " record, ignoring renaming", + LSN_IN_HEX(share->state.create_rename_lsn)); + error= 0; + goto end; + } + if (maria_is_crashed(info)) + { + fprintf(tracef, ", is crashed, can't rename it"); + ALERT_USER(); + goto end; + } + /* + This maria_extra() call serves to signal that old open instances of + this table should not be used anymore, and (only on Windows) to close + open files so they can be renamed + */ + if (maria_extra(info, HA_EXTRA_PREPARE_FOR_RENAME, NULL) || + maria_close(info)) + goto end; + info= NULL; + } + fprintf(tracef, ", renaming '%s'", old_name); + if (maria_rename(old_name, new_name)) + { + fprintf(tracef, "Failed to rename table\n"); + goto end; + } + error= 0; +end: + fprintf(tracef, "\n"); + if (info != NULL) + error|= maria_close(info); + return error; +} + + prototype_redo_exec_hook(REDO_DROP_TABLE) { char *name; @@ -553,6 +625,12 @@ prototype_redo_exec_hook(FILE_ID) if (cmp_translog_addr(rec->lsn, checkpoint_start) < 0) { + /* + If that mapping was still true at checkpoint time, it was found in + checkpoint record, no need to recreate it. If that mapping had ended at + checkpoint time (table was closed or repaired), a flush and force + happened and so mapping is not needed. + */ fprintf(tracef, "ignoring because before checkpoint\n"); return 0; } @@ -589,7 +667,8 @@ end: static int new_table(uint16 sid, const char *name, - File org_kfile, File org_dfile, LSN lsn) + File org_kfile, File org_dfile, + LSN lsn_of_file_id) { /* -1 (skip table): close table and return 0; @@ -628,11 +707,12 @@ static int new_table(uint16 sid, const char *name, error= -1; goto end; } - if (cmp_translog_addr(lsn, share->state.create_rename_lsn) <= 0) + if (cmp_translog_addr(lsn_of_file_id, share->state.create_rename_lsn) <= 0) { fprintf(tracef, ", has create_rename_lsn (%lu,0x%lx) more recent than" - " record, ignoring open request", - LSN_IN_HEX(share->state.create_rename_lsn)); + " LOGREC_FILE_ID's LSN (%lu,0x%lx), ignoring open request", + LSN_IN_HEX(share->state.create_rename_lsn), + LSN_IN_HEX(lsn_of_file_id)); error= -1; goto end; } @@ -655,6 +735,16 @@ static int new_table(uint16 sid, const char *name, /* Recovery will fix this, no error */ ALERT_USER(); } + /* + This LSN serves in this situation; assume log is: + FILE_ID(6->"t2") REDO_INSERT(6) FILE_ID(6->"t1") CHECKPOINT(6->"t1") + then crash, checkpoint record is parsed and opens "t1" with id 6; assume + REDO phase starts from the REDO_INSERT above: it will wrongly try to + update a page of "t1". With this LSN below, REDO_INSERT can realize the + mapping is newer than itself, and not execute. + Same example is possible with UNDO_INSERT (update of the state). + */ + info->s->lsn_of_file_id= lsn_of_file_id; all_tables[sid].info= info; all_tables[sid].org_kfile= org_kfile; all_tables[sid].org_dfile= org_dfile; @@ -848,7 +938,7 @@ prototype_redo_exec_hook(UNDO_ROW_INSERT) if (info == NULL) return 0; set_undo_lsn_for_active_trans(rec->short_trid, rec->lsn); - if (cmp_translog_addr(rec->lsn, info->s->state.is_of_lsn) > 0) + if (cmp_translog_addr(rec->lsn, info->s->state.is_of_horizon) > 0) { fprintf(tracef, " state older than record, updating rows' count\n"); info->s->state.state.records++; @@ -871,7 +961,7 @@ prototype_redo_exec_hook(UNDO_ROW_DELETE) if (info == NULL) return 0; set_undo_lsn_for_active_trans(rec->short_trid, rec->lsn); - if (cmp_translog_addr(rec->lsn, info->s->state.is_of_lsn) > 0) + if (cmp_translog_addr(rec->lsn, info->s->state.is_of_horizon) > 0) { fprintf(tracef, " state older than record, updating rows' count\n"); info->s->state.state.records--; @@ -889,30 +979,11 @@ prototype_redo_exec_hook(UNDO_ROW_UPDATE) if (info == NULL) return 0; set_undo_lsn_for_active_trans(rec->short_trid, rec->lsn); - if (cmp_translog_addr(rec->lsn, info->s->state.is_of_lsn) > 0) - { - info->s->state.changed|= STATE_CHANGED | STATE_NOT_ANALYZED | - STATE_NOT_OPTIMIZED_KEYS | STATE_NOT_SORTED_PAGES; - } - return 0; -} - - -prototype_redo_exec_hook(UNDO_ROW_PURGE) -{ - MARIA_HA *info= get_MARIA_HA_from_UNDO_record(rec); - if (info == NULL) - return 0; - /* this a bit broken, but this log record type will be deleted soon */ - set_undo_lsn_for_active_trans(rec->short_trid, rec->lsn); - if (cmp_translog_addr(rec->lsn, info->s->state.is_of_lsn) > 0) + if (cmp_translog_addr(rec->lsn, info->s->state.is_of_horizon) > 0) { - fprintf(tracef, " state older than record, updating rows' count\n"); - info->s->state.state.records--; info->s->state.changed|= STATE_CHANGED | STATE_NOT_ANALYZED | STATE_NOT_OPTIMIZED_KEYS | STATE_NOT_SORTED_PAGES; } - fprintf(tracef, " rows' count %lu\n", (ulong)info->s->state.state.records); return 0; } @@ -969,7 +1040,7 @@ prototype_redo_exec_hook(CLR_END) set_undo_lsn_for_active_trans(rec->short_trid, previous_undo_lsn); fprintf(tracef, " CLR_END was about %s, undo_lsn now LSN (%lu,0x%lx)\n", log_desc->name, LSN_IN_HEX(previous_undo_lsn)); - if (cmp_translog_addr(rec->lsn, info->s->state.is_of_lsn) > 0) + if (cmp_translog_addr(rec->lsn, info->s->state.is_of_horizon) > 0) { fprintf(tracef, " state older than record, updating rows' count\n"); switch (undone_record_type) { @@ -1113,6 +1184,7 @@ static int run_redo_phase(LSN lsn, my_bool apply) install_redo_exec_hook(LONG_TRANSACTION_ID); install_redo_exec_hook(CHECKPOINT); install_redo_exec_hook(REDO_CREATE_TABLE); + install_redo_exec_hook(REDO_RENAME_TABLE); install_redo_exec_hook(REDO_DROP_TABLE); install_redo_exec_hook(FILE_ID); install_redo_exec_hook(REDO_INSERT_ROW_HEAD); @@ -1124,7 +1196,6 @@ static int run_redo_phase(LSN lsn, my_bool apply) install_redo_exec_hook(UNDO_ROW_INSERT); install_redo_exec_hook(UNDO_ROW_DELETE); install_redo_exec_hook(UNDO_ROW_UPDATE); - install_redo_exec_hook(UNDO_ROW_PURGE); install_redo_exec_hook(COMMIT); install_redo_exec_hook(CLR_END); install_undo_exec_hook(UNDO_ROW_INSERT); @@ -1134,17 +1205,14 @@ static int run_redo_phase(LSN lsn, my_bool apply) current_group_end_lsn= LSN_IMPOSSIBLE; TRANSLOG_HEADER_BUFFER rec; - /* - instead of this block below we will soon use - translog_first_lsn_in_log()... - */ + int len= translog_read_record_header(lsn, &rec); /** @todo EOF should be detected */ if (len == RECHEADER_READ_ERROR) { - fprintf(tracef, "Cannot find a first record\n"); - return 1; + fprintf(tracef, "Cannot find a first record, empty log, nothing to do\n"); + return 0; } struct st_translog_scanner_data scanner; if (translog_init_scanner(lsn, 1, &scanner)) @@ -1247,7 +1315,7 @@ static int run_redo_phase(LSN lsn, my_bool apply) fprintf(tracef, "EOF on the log\n"); break; case RECHEADER_READ_ERROR: - fprintf(stderr, "Error reading log\n"); + fprintf(tracef, "Error reading log\n"); return 1; } break; @@ -1281,7 +1349,7 @@ static uint end_of_redo_phase(my_bool prepare_for_undo_phase) dirty_pages_pool= NULL; llstr(max_long_trid, llbuf); - printf("Maximum transaction long id seen: %s\n", llbuf); + fprintf(tracef, "Maximum transaction long id seen: %s\n", llbuf); if (prepare_for_undo_phase && trnman_init(max_long_trid)) return -1; @@ -1400,22 +1468,30 @@ static int run_undo_phase(uint unfinished) /** - @brief re-enables transactionality, updates is_of_lsn + @brief re-enables transactionality, updates is_of_horizon @param info table - @param at_lsn LSN to set is_of_lsn + @param horizon address to set is_of_horizon */ -static void prepare_table_for_close(MARIA_HA *info, LSN at_lsn) +static void prepare_table_for_close(MARIA_HA *info, TRANSLOG_ADDRESS horizon) { MARIA_SHARE *share= info->s; /* - State is now at least as new as the LSN of the current record. It may be + In a fully-forward REDO phase (no checkpoint record), + state is now at least as new as the LSN of the current record. It may be newer, in case we are seeing a LOGREC_FILE_ID which tells us to close a table, but that table was later modified further in the log. + But if we parsed a checkpoint record, it may be this way in the log: + FILE_ID(6->t2)... FILE_ID(6->t1)... CHECKPOINT(6->t1) + Checkpoint parsing opened t1 with id 6; first FILE_ID above is going to + make t1 close; the first condition below is however false (when checkpoint + was taken it increased is_of_horizon) and so it works. For safety we + add the second condition. */ - if (cmp_translog_addr(share->state.is_of_lsn, at_lsn) < 0) - share->state.is_of_lsn= at_lsn; + if (cmp_translog_addr(share->state.is_of_horizon, horizon) < 0 && + cmp_translog_addr(share->lsn_of_file_id, horizon) < 0) + share->state.is_of_horizon= horizon; _ma_reenable_logging_for_table(share); } @@ -1446,6 +1522,22 @@ static MARIA_HA *get_MARIA_HA_from_REDO_record(const return NULL; } fprintf(tracef, ", '%s'", info->s->open_file_name); + if (cmp_translog_addr(rec->lsn, info->s->lsn_of_file_id) <= 0) + { + /* + This can happen only if processing a record before the checkpoint + record. + id->name mapping is newer than REDO record: for sure the table subject + of the REDO has been flushed and forced (id re-assignment implies this); + REDO can be ignored (and must be, as we don't know what this subject + table was). + */ + DBUG_ASSERT(cmp_translog_addr(rec->lsn, checkpoint_start) < 0); + fprintf(tracef, ", table's LOGREC_FILE_ID has LSN (%lu,0x%lx) more recent" + " than record, skipping record", + LSN_IN_HEX(info->s->lsn_of_file_id)); + return NULL; + } /* detect if an open instance of a dropped table (internal bug) */ DBUG_ASSERT(info->s->last_version != 0); if (cmp_translog_addr(rec->lsn, checkpoint_start) < 0) @@ -1491,27 +1583,45 @@ static MARIA_HA *get_MARIA_HA_from_UNDO_record(const fprintf(tracef, ", table skipped, so skipping record\n"); return NULL; } - _ma_writeinfo(info, WRITEINFO_UPDATE_KEYFILE); /* to flush state on close */ fprintf(tracef, ", '%s'", info->s->open_file_name); + if (cmp_translog_addr(rec->lsn, info->s->lsn_of_file_id) <= 0) + { + fprintf(tracef, ", table's LOGREC_FILE_ID has LSN (%lu,0x%lx) more recent" + " than record, skipping record", + LSN_IN_HEX(info->s->lsn_of_file_id)); + return NULL; + } DBUG_ASSERT(info->s->last_version != 0); + _ma_writeinfo(info, WRITEINFO_UPDATE_KEYFILE); /* to flush state on close */ fprintf(tracef, ", applying record\n"); return info; } -static int parse_checkpoint_record(LSN lsn) +/** + @brief Parses checkpoint record. + + Builds from it the dirty_pages list (a hash), opens tables and maps them to + their 2-byte IDs, recreates transactions (not real TRNs though). + + @return From where in the log the REDO phase should start + @retval LSN_IMPOSSIBLE error + @retval other ok +*/ + +static LSN parse_checkpoint_record(LSN lsn) { uint i; TRANSLOG_HEADER_BUFFER rec; - fprintf(tracef, "Loading data from checkpoint record\n"); + fprintf(tracef, "Loading data from checkpoint record at LSN (%lu,0x%lx)\n", + LSN_IN_HEX(lsn)); int len= translog_read_record_header(lsn, &rec); - /** @todo EOF should be detected */ if (len == RECHEADER_READ_ERROR) { fprintf(tracef, "Cannot find checkpoint record where it should be\n"); - return 1; + return LSN_IMPOSSIBLE; } enlarge_buffer(&rec); @@ -1521,7 +1631,7 @@ static int parse_checkpoint_record(LSN lsn) rec.record_length) { fprintf(tracef, "Failed to read record\n"); - return 1; + return LSN_IMPOSSIBLE; } char *ptr= log_record_buffer.str; @@ -1563,6 +1673,7 @@ static int parse_checkpoint_record(LSN lsn) /* tables */ uint nb_tables= uint4korr(ptr); + ptr+= 4; fprintf(tracef, "%u open tables\n", nb_tables); for (i= 0; i< nb_tables; i++) { @@ -1580,23 +1691,24 @@ static int parse_checkpoint_record(LSN lsn) ptr+= name_len; strnmov(name, ptr, sizeof(name)); if (new_table(sid, name, kfile, dfile, first_log_write_lsn)) - return 1; + return LSN_IMPOSSIBLE; } /* dirty pages */ uint nb_dirty_pages= uint4korr(ptr); ptr+= 4; + fprintf(tracef, "%u dirty pages\n", nb_dirty_pages); if (hash_init(&all_dirty_pages, &my_charset_bin, nb_dirty_pages, offsetof(struct st_dirty_page, file_and_page_id), sizeof(((struct st_dirty_page *)NULL)->file_and_page_id), NULL, NULL, 0)) - return 1; + return LSN_IMPOSSIBLE; dirty_pages_pool= (struct st_dirty_page *)my_malloc(nb_dirty_pages * sizeof(struct st_dirty_page), MYF(MY_WME)); if (unlikely(dirty_pages_pool == NULL)) - return 1; + return LSN_IMPOSSIBLE; struct st_dirty_page *next_dirty_page_in_pool= dirty_pages_pool; LSN minimum_rec_lsn_of_dirty_pages= LSN_MAX; for (i= 0; i < nb_dirty_pages ; i++) @@ -1608,7 +1720,7 @@ static int parse_checkpoint_record(LSN lsn) LSN rec_lsn= lsn_korr(ptr); ptr+= LSN_STORE_SIZE; if (new_page(fileid, pageid, rec_lsn, next_dirty_page_in_pool++)) - return 1; + return LSN_IMPOSSIBLE; set_if_smaller(minimum_rec_lsn_of_dirty_pages, rec_lsn); } /* after that, there will be no insert/delete into the hash */ @@ -1627,11 +1739,11 @@ static int parse_checkpoint_record(LSN lsn) if (ptr != (log_record_buffer.str + log_record_buffer.length)) { fprintf(tracef, "checkpoint record corrupted\n"); - return 1; + return LSN_IMPOSSIBLE; } set_if_smaller(checkpoint_start, minimum_rec_lsn_of_dirty_pages); - return 0; + return checkpoint_start; } static int new_page(File fileid, pgcache_page_no_t pageid, LSN rec_lsn, @@ -1656,9 +1768,9 @@ static int close_all_tables() /* Since the end of end_of_redo_phase(), we may have written new records (if UNDO phase ran) and thus the state is newer than at - end_of_redo_phase(), we need to bump is_of_lsn again. + end_of_redo_phase(), we need to bump is_of_horizon again. */ - LSN addr= translog_get_horizon(); + TRANSLOG_ADDRESS addr= translog_get_horizon(); for (list_element= maria_open_list ; list_element ; list_element= next_open) { next_open= list_element->next; @@ -1674,14 +1786,14 @@ end: } #ifdef MARIA_EXTERNAL_LOCKING -#error Maria's Recovery is really not ready for it +#error Maria's Checkpoint and Recovery are really not ready for it #endif /* Recovery of the state : how it works ===================================== -Ignoring Checkpoints for a start. +Here we ignore Checkpoints for a start. The state (MARIA_HA::MARIA_SHARE::MARIA_STATE_INFO) is updated in memory frequently (at least at every row write/update/delete) but goes @@ -1700,8 +1812,8 @@ the end of every row write/update/delete/delete_all. When Recovery sees the sign of such row operation (UNDO or REDO), it may need to update the records' count if that count does not reflect that operation (is older). How to know the age of the state compared to the log record: every time the state -goes to disk at runtime, its member "is_of_lsn" is updated to the -current end-of-log LSN. So Recovery just needs to compare is_of_lsn +goes to disk at runtime, its member "is_of_horizon" is updated to the +current end-of-log horizon. So Recovery just needs to compare is_of_horizon and the record's LSN to know if it should modify "records". Other operations like ALTER TABLE DISABLE KEYS update the state but @@ -1738,7 +1850,7 @@ intern_lock (as Checkpoint needs it anyway to read MARIA_SHARE::kfile, and as maria_close() takes it too). All state writes to disk are changed to be protected with intern_lock. So Checkpoint takes intern_lock, log's lock, reads "records" from -memory, releases log's lock, updates is_of_lsn and writes "records" to +memory, releases log's lock, updates is_of_horizon and writes "records" to disk, release intern_lock. In practice, not only "records" needs to be written but the full state. So, Checkpoint reads the full state from memory. Some other @@ -1747,7 +1859,7 @@ state which are not protected by the lock's log (see ma_extra.c HA_EXTRA_NO_KEYS), and Checkpoint would be reading a corrupted state from memory; to guard against that we extend the intern_lock-zone to changes done to the state in memory by HA_EXTRA_NO_KEYS et al, and -also any change made in memory to create_rename_lsn/state_is_of_lsn. +also any change made in memory to create_rename_lsn/state_is_of_horizon. Last, we don't want in Checkpoint to do log lock; read state from memory; release log lock; for each table, it may hold the log's lock too much in total. @@ -1758,12 +1870,12 @@ But this re-introduces the problem that some other thread may be changing the state in memory and on disk under intern_lock, without log's lock, like HA_EXTRA_NO_KEYS, while we read the N states. However, when Checkpoint later comes to handling the table under intern_lock, which is serialized with -HA_EXTRA_NO_KEYS, it can see that is_of_lsn is higher then when the state was -read from memory under log's lock, and thus can decide to not flush the +HA_EXTRA_NO_KEYS, it can see that is_of_horizon is higher then when the state +was read from memory under log's lock, and thus can decide to not flush the obsolete state it has, knowing that the other thread flushed a more recent -state already. If on the other hand is_of_lsn is not higher, the read state is -current and can be flushed. So we have a per-table sequence: - lock intern_lock; test if is_of_lsn is higher than when we read the state +state already. If on the other hand is_of_horizon is not higher, the read +state is current and can be flushed. So we have a per-table sequence: + lock intern_lock; test if is_of_horizon is higher than when we read the state under log's lock; if no then flush the read state to disk. */ @@ -1794,7 +1906,6 @@ current and can be flushed. So we have a per-table sequence: /**** UNDO PHASE *****/ - print_information_to_error_log(nb of trans to roll back, nb of prepared trans /* Launch one or more threads to do the background rollback. Don't wait for them to complete their rollback (background rollback; for debugging, we @@ -1815,33 +1926,4 @@ current and can be flushed. So we have a per-table sequence: /* mark that checkpoint requests are now allowed. */ -} - -pthread_handler_decl rollback_background_thread() -{ - /* - execute the normal runtime-rollback code for a bunch of transactions. - */ - while (trans in list_of_trans_to_rollback_by_this_thread) - { - while (trans->undo_lsn != 0) - { - /* this is the normal runtime-rollback code: */ - record= log_read_record(trans->undo_lsn); - execute_log_record_in_undo_phase(record); - trans->undo_lsn= record.prev_undo_lsn; - } - /* remove trans from list */ - } - lock_mutex(rollback_threads); /* or atomic counter */ - if (--total_of_rollback_threads == 0) - { - /* - All rollback threads are done. Print "rollback finished" to the error - log and take a full checkpoint. - */ - } - unlock_mutex(rollback_threads); - pthread_exit(); -} #endif diff --git a/storage/maria/ma_rename.c b/storage/maria/ma_rename.c index ba05d745195..44cd60711da 100644 --- a/storage/maria/ma_rename.c +++ b/storage/maria/ma_rename.c @@ -67,17 +67,12 @@ int maria_rename(const char *old_name, const char *new_name) if (sync_dir) { LSN lsn; - uchar log_data[2 + 2]; - LEX_STRING log_array[TRANSLOG_INTERNAL_PARTS + 3]; - uint old_name_len= strlen(old_name), new_name_len= strlen(new_name); - int2store(log_data, old_name_len); - int2store(log_data + 2, new_name_len); - log_array[TRANSLOG_INTERNAL_PARTS + 0].str= log_data; - log_array[TRANSLOG_INTERNAL_PARTS + 0].length= sizeof(log_data); - log_array[TRANSLOG_INTERNAL_PARTS + 1].str= (char *)old_name; - log_array[TRANSLOG_INTERNAL_PARTS + 1].length= old_name_len; - log_array[TRANSLOG_INTERNAL_PARTS + 2].str= (char *)new_name; - log_array[TRANSLOG_INTERNAL_PARTS + 2].length= new_name_len; + LEX_STRING log_array[TRANSLOG_INTERNAL_PARTS + 2]; + uint old_name_len= strlen(old_name)+1, new_name_len= strlen(new_name)+1; + log_array[TRANSLOG_INTERNAL_PARTS + 0].str= (char *)old_name; + log_array[TRANSLOG_INTERNAL_PARTS + 0].length= old_name_len; + log_array[TRANSLOG_INTERNAL_PARTS + 1].str= (char *)new_name; + log_array[TRANSLOG_INTERNAL_PARTS + 1].length= new_name_len; /* For this record to be of any use for Recovery, we need the upper MySQL layer to be crash-safe, which it is not now (that would require @@ -88,7 +83,7 @@ int maria_rename(const char *old_name, const char *new_name) */ if (unlikely(translog_write_record(&lsn, LOGREC_REDO_RENAME_TABLE, &dummy_transaction_object, NULL, - 2 + 2 + old_name_len + new_name_len, + old_name_len + new_name_len, sizeof(log_array)/sizeof(log_array[0]), log_array, NULL) || translog_flush(lsn))) @@ -100,7 +95,7 @@ int maria_rename(const char *old_name, const char *new_name) store LSN into file, needed for Recovery to not be confused if a RENAME happened (applying REDOs to the wrong table). */ - if (_ma_update_create_rename_lsn_on_disk(share, lsn, TRUE)) + if (_ma_update_create_rename_lsn(share, lsn, TRUE)) { maria_close(info); DBUG_RETURN(1); diff --git a/storage/maria/ma_test2.c b/storage/maria/ma_test2.c index 596374eef80..935be09850c 100644 --- a/storage/maria/ma_test2.c +++ b/storage/maria/ma_test2.c @@ -25,6 +25,7 @@ #define SAFEMALLOC #endif #include "maria_def.h" +#include "trnman.h" #include #include @@ -47,7 +48,8 @@ static void copy_key(struct st_maria_info *info,uint inx, static int verbose=0,testflag=0, first_key=0,async_io=0,pagecacheing=0,write_cacheing=0,locking=0, rec_pointer_size=0,pack_fields=1,silent=0, - opt_quick_mode=0, transactional= 0, skip_update= 0; + opt_quick_mode=0, transactional= 0, skip_update= 0, + die_in_middle_of_transaction= 0; static int pack_seg=HA_SPACE_PACK,pack_type=HA_PACK_KEY,remove_count=-1; static int create_flag= 0, srand_arg= 0; static ulong pagecache_size=IO_SIZE*16; @@ -235,6 +237,9 @@ int main(int argc, char *argv[]) goto err; if (!(file=maria_open(filename,2,HA_OPEN_ABORT_IF_LOCKED))) goto err; + maria_begin(file); + if (testflag == 1) + goto end; if (!silent) printf("- Writing key:s\n"); if (locking) @@ -244,8 +249,6 @@ int main(int argc, char *argv[]) if (opt_quick_mode) maria_extra(file,HA_EXTRA_QUICK,0); - maria_begin(file); - for (i=0 ; i < recant ; i++) { ulong blob_length; @@ -297,7 +300,7 @@ int main(int argc, char *argv[]) } } } - if (testflag == 1) + if (testflag == 2) goto end; if (write_cacheing) @@ -348,7 +351,7 @@ int main(int argc, char *argv[]) else puts("Warning: Skipping delete test because no dupplicate keys"); } - if (testflag == 2) + if (testflag == 3) goto end; if (!silent) @@ -409,7 +412,7 @@ int main(int argc, char *argv[]) } } } - if (testflag == 3) + if (testflag == 4) goto end; for (i=999, dupp_keys=j=0 ; i>0 ; i--) @@ -814,7 +817,7 @@ int main(int argc, char *argv[]) goto err; } - if (testflag == 4) + if (testflag == 5) goto end; if (!silent) @@ -892,6 +895,36 @@ int main(int argc, char *argv[]) goto err; } end: + if (die_in_middle_of_transaction) + { + /* As commit record is not done, UNDO entries needs to be rolled back */ + switch (die_in_middle_of_transaction) { + case 1: + /* + Flush changed pages go to disk. That will also flush log. Recovery + will skip REDOs and apply UNDOs. + */ + _ma_flush_table_files(file, MARIA_FLUSH_DATA, FLUSH_RELEASE, + FLUSH_RELEASE); + break; + case 2: + /* + Just flush log. Pages are likely to not be on disk. Recovery will + then execute REDOs and UNDOs. + */ + if (translog_flush(file->trn->undo_lsn)) + goto err; + break; + case 3: + /* + Flush nothing. Pages and log are likely to not be on disk. Recovery + will then do nothing. + */ + break; + } + printf("Dying on request without maria_commit()/maria_close()\n"); + exit(0); + } if (maria_commit(file)) goto err; if (maria_close(file)) @@ -998,9 +1031,9 @@ static void get_options(int argc, char **argv) verbose=1; break; case 'm': /* records */ - if ((recant=atoi(++pos)) < 10 && testflag > 1) + if ((recant=atoi(++pos)) < 10 && testflag > 2) { - fprintf(stderr,"record count must be >= 10 (if testflag != 1)\n"); + fprintf(stderr,"record count must be >= 10 (if testflag > 2)\n"); exit(1); } break; @@ -1048,6 +1081,9 @@ static void get_options(int argc, char **argv) case 'T': transactional= 1; break; + case 'u': + die_in_middle_of_transaction= atoi(++pos); + break; case 'q': opt_quick_mode=1; break; diff --git a/storage/maria/ma_test_recovery b/storage/maria/ma_test_recovery index 7067d79a49d..23f65c7e764 100755 --- a/storage/maria/ma_test_recovery +++ b/storage/maria/ma_test_recovery @@ -131,7 +131,7 @@ do for test_undo in 1 2 3 do # first iteration tests rollback of insert, second tests rollback of delete - set -- "ma_test1 $silent -M -T -c -N $blobs" "--testflag=1" "--testflag=2" "ma_test1 $silent -M -T -c -N --debug=d:t:i:o,/tmp/ma_test1.trace $blobs" "--testflag=3" "--testflag=4" "ma_test1 $silent -M -T -c -N --debug=d:t:i:o,/tmp/ma_test1.trace $blobs" "--testflag=2" "--testflag=3" + set -- "ma_test1 $silent -M -T -c -N $blobs" "--testflag=1" "--testflag=2 --test-undo=" "ma_test1 $silent -M -T -c -N --debug=d:t:i:o,/tmp/ma_test1.trace $blobs" "--testflag=3" "--testflag=4 --test-undo=" "ma_test1 $silent -M -T -c -N --debug=d:t:i:o,/tmp/ma_test1.trace $blobs" "--testflag=2" "--testflag=3 --test-undo=" "ma_test2 $silent -L -K -W -P -M -T -c $blobs" "-t1" "-t2 -u" # -N (create NULL fields) is needed because --test-undo adds it anyway while [ $# != 0 ] do @@ -148,8 +148,8 @@ do mv $table.MAD $tmp/$table.MAD.good rm $table.MAI rm maria_log.* maria_log_control - echo "TEST WITH $prog $abort_run_args --test-undo=$test_undo (additional aborted work)" - $maria_path/$prog $abort_run_args --test-undo=$test_undo + echo "TEST WITH $prog $abort_run_args$test_undo (additional aborted work)" + $maria_path/$prog $abort_run_args$test_undo cp $table.MAD $tmp/$table.MAD.before_undo if [ $test_undo -lt 3 ] then @@ -174,7 +174,7 @@ do echo "testing applying of CLRs to recreate table" rm $table.MA? apply_log "shouldnotchangelog" - # the cmp below fails with blobs! @todo RECOVERY BUG find out why. + # the cmp below fails with ma_test1+blobs! @todo RECOVERY BUG why? # It is probably serious; REDOs shouldn't place rows in different # positions from what the run-time code did. Indeed it may lead to # more or less free space... @@ -189,12 +189,15 @@ do check_table_is_same shift 3 done + rm -f $table.* $tmp/$table* $tmp/maria_chk_*.txt $tmp/maria_read_log_$table.txt done done -rm -f $table.* $tmp/$table* $tmp/maria_chk_*.txt $tmp/maria_read_log_$table.txt ) 2>&1 > $tmp/ma_test_recovery.output +# also note that maria_chk -dvv shows differences for ma_test2 in UNDO phase, +# this is normal: removing records does not shrink the data/key file, +# does not put back the "analyzed,optimized keys"(etc) index state. diff $maria_path/ma_test_recovery.expected $tmp/ma_test_recovery.output > /dev/null || diff_failed=1 if [ "$diff_failed" == "1" ] then diff --git a/storage/maria/ma_test_recovery.expected b/storage/maria/ma_test_recovery.expected index 87bc32e3a70..253471af802 100644 --- a/storage/maria/ma_test_recovery.expected +++ b/storage/maria/ma_test_recovery.expected @@ -205,6 +205,47 @@ Differences in maria_chk -dvv, recovery not yet perfect ! --- > 1 2 6 unique number NULL 0 8192 ========DIFF END======= +TEST WITH ma_test2 -s -L -K -W -P -M -T -c -t1 (commit at end) +TEST WITH ma_test2 -s -L -K -W -P -M -T -c -t2 -u1 (additional aborted work) +Dying on request without maria_commit()/maria_close() +applying log +Differences in maria_chk -dvv, recovery not yet perfect ! +========DIFF START======= +6c6 +< Status: checked,analyzed,optimized keys,sorted index pages +--- +> Status: changed +11c11 +< Datafile length: 8192 Keyfile length: 8192 +--- +> Datafile length: 90112 Keyfile length: 204800 +========DIFF END======= +testing idempotency +applying log +Differences in maria_chk -dvv, recovery not yet perfect ! +========DIFF START======= +6c6 +< Status: checked,analyzed,optimized keys,sorted index pages +--- +> Status: changed +11c11 +< Datafile length: 8192 Keyfile length: 8192 +--- +> Datafile length: 90112 Keyfile length: 204800 +========DIFF END======= +testing applying of CLRs to recreate table +applying log +Differences in maria_chk -dvv, recovery not yet perfect ! +========DIFF START======= +6c6 +< Status: checked,analyzed,optimized keys,sorted index pages +--- +> Status: changed +11c11 +< Datafile length: 8192 Keyfile length: 8192 +--- +> Datafile length: 90112 Keyfile length: 8192 +========DIFF END======= TEST WITH ma_test1 -s -M -T -c -N --testflag=1 (commit at end) TEST WITH ma_test1 -s -M -T -c -N --testflag=2 --test-undo=2 (additional aborted work) Terminating after inserts @@ -339,6 +380,47 @@ Differences in maria_chk -dvv, recovery not yet perfect ! --- > 1 2 6 unique number NULL 0 8192 ========DIFF END======= +TEST WITH ma_test2 -s -L -K -W -P -M -T -c -t1 (commit at end) +TEST WITH ma_test2 -s -L -K -W -P -M -T -c -t2 -u2 (additional aborted work) +Dying on request without maria_commit()/maria_close() +applying log +Differences in maria_chk -dvv, recovery not yet perfect ! +========DIFF START======= +6c6 +< Status: checked,analyzed,optimized keys,sorted index pages +--- +> Status: changed +11c11 +< Datafile length: 8192 Keyfile length: 8192 +--- +> Datafile length: 90112 Keyfile length: 204800 +========DIFF END======= +testing idempotency +applying log +Differences in maria_chk -dvv, recovery not yet perfect ! +========DIFF START======= +6c6 +< Status: checked,analyzed,optimized keys,sorted index pages +--- +> Status: changed +11c11 +< Datafile length: 8192 Keyfile length: 8192 +--- +> Datafile length: 90112 Keyfile length: 204800 +========DIFF END======= +testing applying of CLRs to recreate table +applying log +Differences in maria_chk -dvv, recovery not yet perfect ! +========DIFF START======= +6c6 +< Status: checked,analyzed,optimized keys,sorted index pages +--- +> Status: changed +11c11 +< Datafile length: 8192 Keyfile length: 8192 +--- +> Datafile length: 90112 Keyfile length: 8192 +========DIFF END======= TEST WITH ma_test1 -s -M -T -c -N --testflag=1 (commit at end) TEST WITH ma_test1 -s -M -T -c -N --testflag=2 --test-undo=3 (additional aborted work) Terminating after inserts @@ -473,6 +555,47 @@ Differences in maria_chk -dvv, recovery not yet perfect ! --- > 1 2 6 unique number NULL 0 8192 ========DIFF END======= +TEST WITH ma_test2 -s -L -K -W -P -M -T -c -t1 (commit at end) +TEST WITH ma_test2 -s -L -K -W -P -M -T -c -t2 -u3 (additional aborted work) +Dying on request without maria_commit()/maria_close() +applying log +Differences in maria_chk -dvv, recovery not yet perfect ! +========DIFF START======= +6c6 +< Status: checked,analyzed,optimized keys,sorted index pages +--- +> Status: changed +11c11 +< Datafile length: 8192 Keyfile length: 8192 +--- +> Datafile length: 90112 Keyfile length: 204800 +========DIFF END======= +testing idempotency +applying log +Differences in maria_chk -dvv, recovery not yet perfect ! +========DIFF START======= +6c6 +< Status: checked,analyzed,optimized keys,sorted index pages +--- +> Status: changed +11c11 +< Datafile length: 8192 Keyfile length: 8192 +--- +> Datafile length: 90112 Keyfile length: 204800 +========DIFF END======= +testing applying of CLRs to recreate table +applying log +Differences in maria_chk -dvv, recovery not yet perfect ! +========DIFF START======= +6c6 +< Status: checked,analyzed,optimized keys,sorted index pages +--- +> Status: changed +11c11 +< Datafile length: 8192 Keyfile length: 8192 +--- +> Datafile length: 90112 Keyfile length: 8192 +========DIFF END======= TEST WITH ma_test1 -s -M -T -c -N -b --testflag=1 (commit at end) TEST WITH ma_test1 -s -M -T -c -N -b --testflag=2 --test-undo=1 (additional aborted work) Terminating after inserts @@ -607,6 +730,47 @@ Differences in maria_chk -dvv, recovery not yet perfect ! --- > 1 2 6 unique varchar BLOB NULL 0 8192 ========DIFF END======= +TEST WITH ma_test2 -s -L -K -W -P -M -T -c -b -t1 (commit at end) +TEST WITH ma_test2 -s -L -K -W -P -M -T -c -b -t2 -u1 (additional aborted work) +Dying on request without maria_commit()/maria_close() +applying log +Differences in maria_chk -dvv, recovery not yet perfect ! +========DIFF START======= +6c6 +< Status: checked,analyzed,optimized keys,sorted index pages +--- +> Status: changed +11c11 +< Datafile length: 8192 Keyfile length: 8192 +--- +> Datafile length: 81920 Keyfile length: 212992 +========DIFF END======= +testing idempotency +applying log +Differences in maria_chk -dvv, recovery not yet perfect ! +========DIFF START======= +6c6 +< Status: checked,analyzed,optimized keys,sorted index pages +--- +> Status: changed +11c11 +< Datafile length: 8192 Keyfile length: 8192 +--- +> Datafile length: 81920 Keyfile length: 212992 +========DIFF END======= +testing applying of CLRs to recreate table +applying log +Differences in maria_chk -dvv, recovery not yet perfect ! +========DIFF START======= +6c6 +< Status: checked,analyzed,optimized keys,sorted index pages +--- +> Status: changed +11c11 +< Datafile length: 8192 Keyfile length: 8192 +--- +> Datafile length: 81920 Keyfile length: 8192 +========DIFF END======= TEST WITH ma_test1 -s -M -T -c -N -b --testflag=1 (commit at end) TEST WITH ma_test1 -s -M -T -c -N -b --testflag=2 --test-undo=2 (additional aborted work) Terminating after inserts @@ -741,6 +905,47 @@ Differences in maria_chk -dvv, recovery not yet perfect ! --- > 1 2 6 unique varchar BLOB NULL 0 8192 ========DIFF END======= +TEST WITH ma_test2 -s -L -K -W -P -M -T -c -b -t1 (commit at end) +TEST WITH ma_test2 -s -L -K -W -P -M -T -c -b -t2 -u2 (additional aborted work) +Dying on request without maria_commit()/maria_close() +applying log +Differences in maria_chk -dvv, recovery not yet perfect ! +========DIFF START======= +6c6 +< Status: checked,analyzed,optimized keys,sorted index pages +--- +> Status: changed +11c11 +< Datafile length: 8192 Keyfile length: 8192 +--- +> Datafile length: 81920 Keyfile length: 212992 +========DIFF END======= +testing idempotency +applying log +Differences in maria_chk -dvv, recovery not yet perfect ! +========DIFF START======= +6c6 +< Status: checked,analyzed,optimized keys,sorted index pages +--- +> Status: changed +11c11 +< Datafile length: 8192 Keyfile length: 8192 +--- +> Datafile length: 81920 Keyfile length: 212992 +========DIFF END======= +testing applying of CLRs to recreate table +applying log +Differences in maria_chk -dvv, recovery not yet perfect ! +========DIFF START======= +6c6 +< Status: checked,analyzed,optimized keys,sorted index pages +--- +> Status: changed +11c11 +< Datafile length: 8192 Keyfile length: 8192 +--- +> Datafile length: 81920 Keyfile length: 8192 +========DIFF END======= TEST WITH ma_test1 -s -M -T -c -N -b --testflag=1 (commit at end) TEST WITH ma_test1 -s -M -T -c -N -b --testflag=2 --test-undo=3 (additional aborted work) Terminating after inserts @@ -875,3 +1080,44 @@ Differences in maria_chk -dvv, recovery not yet perfect ! --- > 1 2 6 unique varchar BLOB NULL 0 8192 ========DIFF END======= +TEST WITH ma_test2 -s -L -K -W -P -M -T -c -b -t1 (commit at end) +TEST WITH ma_test2 -s -L -K -W -P -M -T -c -b -t2 -u3 (additional aborted work) +Dying on request without maria_commit()/maria_close() +applying log +Differences in maria_chk -dvv, recovery not yet perfect ! +========DIFF START======= +6c6 +< Status: checked,analyzed,optimized keys,sorted index pages +--- +> Status: changed +11c11 +< Datafile length: 8192 Keyfile length: 8192 +--- +> Datafile length: 81920 Keyfile length: 212992 +========DIFF END======= +testing idempotency +applying log +Differences in maria_chk -dvv, recovery not yet perfect ! +========DIFF START======= +6c6 +< Status: checked,analyzed,optimized keys,sorted index pages +--- +> Status: changed +11c11 +< Datafile length: 8192 Keyfile length: 8192 +--- +> Datafile length: 81920 Keyfile length: 212992 +========DIFF END======= +testing applying of CLRs to recreate table +applying log +Differences in maria_chk -dvv, recovery not yet perfect ! +========DIFF START======= +6c6 +< Status: checked,analyzed,optimized keys,sorted index pages +--- +> Status: changed +11c11 +< Datafile length: 8192 Keyfile length: 8192 +--- +> Datafile length: 81920 Keyfile length: 8192 +========DIFF END======= diff --git a/storage/maria/ma_write.c b/storage/maria/ma_write.c index 4a6ace13741..b034d71ef9d 100644 --- a/storage/maria/ma_write.c +++ b/storage/maria/ma_write.c @@ -222,6 +222,12 @@ err: maria_flush_bulk_insert(info, j); } info->errkey= (int) i; + /* + We delete keys in the reverse order of insertion. This is the order that + a rollback would do and is important for CLR_ENDs generated by + _ma_ft|ck_delete() and write_record_abort() to work (with any other + order they would cause wrong jumps in the chain). + */ while ( i-- > 0) { if (maria_is_key_active(share->state.key_map, i)) @@ -231,6 +237,10 @@ err: is_tree_inited(&info->bulk_insert[i]))); if (local_lock_tree) rw_wrlock(&share->key_root_lock[i]); + /** + @todo RECOVERY BUG + The key deletes below should generate CLR_ENDs + */ if (share->keyinfo[i].flag & HA_FULLTEXT) { if (_ma_ft_del(info,i,(char*) buff,record,filepos)) diff --git a/storage/maria/maria_chk.c b/storage/maria/maria_chk.c index edd99d01629..f9ed249817e 100644 --- a/storage/maria/maria_chk.c +++ b/storage/maria/maria_chk.c @@ -1033,9 +1033,11 @@ static int maria_chk(HA_CHECK *param, char *filename) Tell the server's Recovery to ignore old REDOs on this table; we don't know what the log's end LSN is now, so we just let the server know that it will have to find and store it. + This is the only case where create_rename_lsn can be a horizon and not + a LSN. */ if (share->base.born_transactional) - share->state.create_rename_lsn= share->state.is_of_lsn= + share->state.create_rename_lsn= share->state.is_of_horizon= LSN_REPAIRED_BY_MARIA_CHK; if ((param->testflag & (T_REP_BY_SORT | T_REP_PARALLEL)) && (maria_is_any_key_active(share->state.key_map) || diff --git a/storage/maria/maria_def.h b/storage/maria/maria_def.h index 8f5b3a68d16..09852f4dc86 100644 --- a/storage/maria/maria_def.h +++ b/storage/maria/maria_def.h @@ -96,7 +96,8 @@ typedef struct st_maria_state_info uint open_count; uint8 changed; /* Changed since mariachk */ LSN create_rename_lsn; /**< LSN when table was last created/renamed */ - LSN is_of_lsn; /**< LSN when state was last updated on disk */ + /** @brief Log horizon when state was last updated on disk */ + TRANSLOG_ADDRESS is_of_horizon; /* the following isn't saved on disk */ uint state_diff_length; /* Should be 0 */ @@ -218,6 +219,7 @@ typedef struct st_maria_file_bitmap #define MARIA_CHECKPOINT_LOOKS_AT_ME 1 #define MARIA_CHECKPOINT_SHOULD_FREE_ME 2 +#define MARIA_CHECKPOINT_SEEN_IN_LOOP 4 typedef struct st_maria_share { /* Shared between opens */ @@ -331,6 +333,7 @@ typedef struct st_maria_share non-mmaped area */ MARIA_FILE_BITMAP bitmap; rw_lock_t mmap_lock; + LSN lsn_of_file_id; /**< LSN of its last LOGREC_FILE_ID */ } MARIA_SHARE; @@ -940,10 +943,10 @@ int _ma_create_index_by_sort(MARIA_SORT_PARAM *info, my_bool no_messages, ulong); int _ma_sync_table_files(const MARIA_HA *info); int _ma_initialize_data_file(MARIA_SHARE *share, File dfile); -int _ma_update_create_rename_lsn_on_disk(MARIA_SHARE *share, - LSN lsn, my_bool do_sync); -int _ma_update_create_rename_lsn_on_disk_sub(MARIA_SHARE *share, - LSN lsn, my_bool do_sync); +int _ma_update_create_rename_lsn(MARIA_SHARE *share, + LSN lsn, my_bool do_sync); +int _ma_update_create_rename_lsn_sub(MARIA_SHARE *share, + LSN lsn, my_bool do_sync); void _ma_unpin_all_pages(MARIA_HA *info, LSN undo_lsn); #define _ma_tmp_disable_logging_for_table(S) \ diff --git a/storage/maria/trnman.c b/storage/maria/trnman.c index d2d4549a895..03d11db3b5b 100644 --- a/storage/maria/trnman.c +++ b/storage/maria/trnman.c @@ -18,6 +18,7 @@ #include #include #include "trnman.h" +#include "ma_checkpoint.h" #include "ma_control_file.h" /* @@ -587,27 +588,25 @@ my_bool trnman_collect_transactions(LEX_STRING *str_act, LEX_STRING *str_com, TRN *trn; char *ptr; uint stored_transactions= 0; - LSN minimum_rec_lsn= ULONGLONG_MAX, minimum_first_undo_lsn= ULONGLONG_MAX; + LSN minimum_rec_lsn= LSN_MAX, minimum_first_undo_lsn= LSN_MAX; DBUG_ENTER("trnman_collect_transactions"); DBUG_ASSERT((NULL == str_act->str) && (NULL == str_com->str)); /* validate the use of read_non_atomic() in general: */ compile_time_assert((sizeof(LSN) == 8) && (sizeof(LSN_WITH_FLAGS) == 8)); - - DBUG_PRINT("info", ("pthread_mutex_lock LOCK_trn_list")); pthread_mutex_lock(&LOCK_trn_list); str_act->length= 2 + /* number of active transactions */ LSN_STORE_SIZE + /* minimum of their rec_lsn */ - (6 + /* long id */ - 2 + /* short id */ + (2 + /* short id */ + 6 + /* long id */ LSN_STORE_SIZE + /* undo_lsn */ #ifdef MARIA_VERSIONING /* not enabled yet */ LSN_STORE_SIZE + /* undo_purge_lsn */ #endif LSN_STORE_SIZE /* first_undo_lsn */ ) * trnman_active_transactions; - str_com->length= 8 + /* number of committed transactions */ + str_com->length= 4 + /* number of committed transactions */ (6 + /* long id */ #ifdef MARIA_VERSIONING /* not enabled yet */ LSN_STORE_SIZE + /* undo_purge_lsn */ @@ -638,13 +637,6 @@ my_bool trnman_collect_transactions(LEX_STRING *str_act, LEX_STRING *str_com, */ continue; } -#ifndef MARIA_CHECKPOINT -/* - in the checkpoint patch (not yet ready) we will have a real implementation - of lsn_read_non_atomic(); for now it's not needed -*/ -#define lsn_read_non_atomic(A) (A) -#endif /* needed for low-water mark calculation */ if (((rec_lsn= lsn_read_non_atomic(trn->rec_lsn)) > 0) && (cmp_translog_addr(rec_lsn, minimum_rec_lsn) < 0)) @@ -656,23 +648,23 @@ my_bool trnman_collect_transactions(LEX_STRING *str_act, LEX_STRING *str_com, if ((undo_lsn= trn->undo_lsn) == 0) /* trn can be forgotten */ continue; stored_transactions++; - int6store(ptr, trn->trid); - ptr+= 6; int2store(ptr, sid); ptr+= 2; + int6store(ptr, trn->trid); + ptr+= 6; lsn_store(ptr, undo_lsn); /* needed for rollback */ ptr+= LSN_STORE_SIZE; -#ifdef MARIA_VERSIONING /* not enabled yet */ - /* to know where purging should start (last delete of this trn) */ - lsn_store(ptr, trn->undo_purge_lsn); - ptr+= LSN_STORE_SIZE; -#endif /* needed for low-water mark calculation */ if (((first_undo_lsn= lsn_read_non_atomic(trn->first_undo_lsn)) > 0) && (cmp_translog_addr(first_undo_lsn, minimum_first_undo_lsn) < 0)) minimum_first_undo_lsn= first_undo_lsn; lsn_store(ptr, first_undo_lsn); ptr+= LSN_STORE_SIZE; +#ifdef MARIA_VERSIONING /* not enabled yet */ + /* to know where purging should start (last delete of this trn) */ + lsn_store(ptr, trn->undo_purge_lsn); + ptr+= LSN_STORE_SIZE; +#endif /** @todo RECOVERY: add a comment explaining why we can dirtily read some vars, inspired by the text of "assumption 8" in WL#3072 @@ -680,6 +672,8 @@ my_bool trnman_collect_transactions(LEX_STRING *str_act, LEX_STRING *str_com, } str_act->length= ptr - str_act->str; /* as we maybe over-estimated */ ptr= str_act->str; + DBUG_PRINT("info",("collected %u active transactions", + (uint)stored_transactions)); int2store(ptr, stored_transactions); ptr+= 2; /* this LSN influences how REDOs for any page can be ignored by Recovery */ @@ -687,8 +681,10 @@ my_bool trnman_collect_transactions(LEX_STRING *str_act, LEX_STRING *str_com, /* one day there will also be a list of prepared transactions */ /* do the same for committed ones */ ptr= str_com->str; - int8store(ptr, (ulonglong)trnman_committed_transactions); - ptr+= 8; + int4store(ptr, trnman_committed_transactions); + ptr+= 4; + DBUG_PRINT("info",("collected %u committed transactions", + (uint)trnman_committed_transactions)); for (trn= committed_list_min.next; trn != &committed_list_max; trn= trn->next) { @@ -716,7 +712,6 @@ my_bool trnman_collect_transactions(LEX_STRING *str_act, LEX_STRING *str_com, err: error= 1; end: - DBUG_PRINT("info", ("pthread_mutex_unlock LOCK_trn_list")); pthread_mutex_unlock(&LOCK_trn_list); DBUG_RETURN(error); } -- cgit v1.2.1 From a5f4e79db962ad7b6e2e5e7241c4789d5630e395 Mon Sep 17 00:00:00 2001 From: unknown Date: Wed, 12 Sep 2007 12:39:04 +0200 Subject: WL#3072 Maria Recovery * added replaying of REDO_REPAIR_TABLE, but disabled it as mysterious linker errors appear. * after replaying RENAME/REPAIR, we must bump create_rename_lsn for idempotency of maria_read_log. sql/mysqld.cc: typo storage/maria/ma_checkpoint.c: silence compiler warning storage/maria/ma_recovery.c: * added replaying of REDO_REPAIR_TABLE, but disabled it as mysterious linker errors appear. * after replaying RENAME/REPAIR, we must bump create_rename_lsn for idempotency of maria_read_log. --- sql/mysqld.cc | 2 +- storage/maria/ma_checkpoint.c | 2 +- storage/maria/ma_recovery.c | 68 +++++++++++++++++++++++++++++++++++-------- 3 files changed, 58 insertions(+), 14 deletions(-) diff --git a/sql/mysqld.cc b/sql/mysqld.cc index 4b403fd85b6..fdfc7336658 100644 --- a/sql/mysqld.cc +++ b/sql/mysqld.cc @@ -6240,7 +6240,7 @@ The minimum value for this variable is 4096.", (uchar**) 0, 0, (GET_ULONG | GET_ASK_ADDR), REQUIRED_ARG, 300, 100, ~0L, 0, 100, 0}, {"pagecache_buffer_size", OPT_KEY_BUFFER_SIZE, - "The size of the buffer used for index blocks for MyISAM tables. Increase this to get better index handling (for all reads and multiple writes) to as much as you can afford; 64M on a 256M machine that mainly runs MySQL is quite common.", + "The size of the buffer used for index blocks for Maria tables. Increase this to get better index handling (for all reads and multiple writes) to as much as you can afford; 64M on a 256M machine that mainly runs MySQL is quite common.", (uchar**) &maria_pagecache_var.param_buff_size, (uchar**) 0, 0, (GET_ULL | GET_ASK_ADDR), REQUIRED_ARG, KEY_CACHE_SIZE, MALLOC_OVERHEAD, ~(ulong) 0, MALLOC_OVERHEAD, IO_SIZE, 0}, diff --git a/storage/maria/ma_checkpoint.c b/storage/maria/ma_checkpoint.c index caeb5ec45d6..b50874e004f 100644 --- a/storage/maria/ma_checkpoint.c +++ b/storage/maria/ma_checkpoint.c @@ -48,7 +48,7 @@ Background page flushing is not used. So, needed pagecache functions for doing this flushing are not yet pushed. */ -#define flush_pagecache_blocks_with_filter(A,B,C,D,E) (((int)D) * 0) +#define flush_pagecache_blocks_with_filter(A,B,C,D,E) (int)(((ulong)D) * 0) /** filter has to return 0, 1 or 2: 0 means "don't flush this page", 1 means "flush it", 2 means "don't flush this page and following pages". diff --git a/storage/maria/ma_recovery.c b/storage/maria/ma_recovery.c index 508615bc65f..3d388518583 100644 --- a/storage/maria/ma_recovery.c +++ b/storage/maria/ma_recovery.c @@ -64,6 +64,7 @@ prototype_redo_exec_hook(LONG_TRANSACTION_ID); prototype_redo_exec_hook_dummy(CHECKPOINT); prototype_redo_exec_hook(REDO_CREATE_TABLE); prototype_redo_exec_hook(REDO_RENAME_TABLE); +prototype_redo_exec_hook(REDO_REPAIR_TABLE); prototype_redo_exec_hook(REDO_DROP_TABLE); prototype_redo_exec_hook(FILE_ID); prototype_redo_exec_hook(REDO_INSERT_ROW_HEAD); @@ -537,6 +538,17 @@ prototype_redo_exec_hook(REDO_RENAME_TABLE) fprintf(tracef, "Failed to rename table\n"); goto end; } + info= maria_open(new_name, O_RDONLY, 0); + if (info == NULL) + { + fprintf(tracef, "Failed to open renamed table\n"); + goto end; + } + if (_ma_update_create_rename_lsn(info->s, rec->lsn, TRUE)) + goto end; + if (maria_close(info)) + goto end; + info= NULL; error= 0; end: fprintf(tracef, "\n"); @@ -546,6 +558,44 @@ end: } +/* + The record may come from REPAIR, ALTER TABLE ENABLE KEYS, OPTIMIZE. +*/ +prototype_redo_exec_hook(REDO_REPAIR_TABLE) +{ + int error= 1; + MARIA_HA *info= get_MARIA_HA_from_REDO_record(rec); + if (info == NULL) + return 0; + /* + Otherwise, the mapping is newer than the table, and our record is newer + than the mapping, so we can repair. + */ + fprintf(tracef, " repairing...\n"); + /** + @todo RECOVERY BUG fix this: + the maria_chk_init() call causes a heap of linker errors in ha_maria.cc! + */ +#if 0 + HA_CHECK param; + maria_chk_init(¶m); + param.isam_file_name= info->s->open_file_name; + param.testflag= uint4korr(rec->header); + if (maria_repair(¶m, info, info->s->open_file_name, + param.testflag & T_QUICK)) + goto end; + if (_ma_update_create_rename_lsn(info->s, rec->lsn, TRUE)) + goto end; + error= 0; +end: + return error; +#else + DBUG_ASSERT("fix this table repairing" == NULL); + return error; +#endif +} + + prototype_redo_exec_hook(REDO_DROP_TABLE) { char *name; @@ -691,10 +741,6 @@ static int new_table(uint16 sid, const char *name, { fprintf(tracef, "Table is crashed, can't apply log records to it\n"); goto end; - /* - we should make an exception for REDO_REPAIR_TABLE records: if we want to - execute them, we should not reject the crashed table here. - */ } MARIA_SHARE *share= info->s; /* check that we're not already using it */ @@ -748,6 +794,11 @@ static int new_table(uint16 sid, const char *name, all_tables[sid].info= info; all_tables[sid].org_kfile= org_kfile; all_tables[sid].org_dfile= org_dfile; + /* + We don't set info->s->id, it would be useless (no logging in REDO phase); + if you change that, know that some records in REDO phase call + _ma_update_create_rename_lsn() which resets info->s->id. + */ fprintf(tracef, ", opened"); error= 0; end: @@ -1185,6 +1236,7 @@ static int run_redo_phase(LSN lsn, my_bool apply) install_redo_exec_hook(CHECKPOINT); install_redo_exec_hook(REDO_CREATE_TABLE); install_redo_exec_hook(REDO_RENAME_TABLE); + install_redo_exec_hook(REDO_REPAIR_TABLE); install_redo_exec_hook(REDO_DROP_TABLE); install_redo_exec_hook(FILE_ID); install_redo_exec_hook(REDO_INSERT_ROW_HEAD); @@ -1728,14 +1780,6 @@ static LSN parse_checkpoint_record(LSN lsn) sanity check on record (did we screw up with all those "ptr+=", did the checkpoint write code and checkpoint read code go out of sync?). */ - /** - @todo This probably presently and hopefully detects that - first_log_write_lsn is not written by the checkpoint record; we need - to add MARIA_SHARE::first_log_write_lsn, fill it with a inwrite-hook of - LOGREC_FILE_ID (note that when we write this record we hold intern_lock, - so Checkpoint will read the LSN correctly), and store it in the - checkpoint record. - */ if (ptr != (log_record_buffer.str + log_record_buffer.length)) { fprintf(tracef, "checkpoint record corrupted\n"); -- cgit v1.2.1 From 20d871e5de798e5f12db39fab6194ceb6e526e42 Mon Sep 17 00:00:00 2001 From: unknown Date: Wed, 12 Sep 2007 14:46:05 +0200 Subject: fix for pushbuild test failure (my_realloc() failed => checkpoint failed => Maria didn't start => tables were created as MyISAM). storage/maria/ma_checkpoint.c: safemalloc complains if my_realloc() is passed NULL and MY_ALLOW_ZERO_PTR is not used. --- storage/maria/ma_checkpoint.c | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/storage/maria/ma_checkpoint.c b/storage/maria/ma_checkpoint.c index b50874e004f..ef09b650820 100644 --- a/storage/maria/ma_checkpoint.c +++ b/storage/maria/ma_checkpoint.c @@ -824,11 +824,11 @@ static int collect_tables(LEX_STRING *str, LSN checkpoint_start_log_horizon) dfiles= (PAGECACHE_FILE *)my_realloc((uchar *)dfiles, /* avoid size of 0 for my_realloc */ max(1, nb) * sizeof(PAGECACHE_FILE), - MYF(MY_WME)); + MYF(MY_WME | MY_ALLOW_ZERO_PTR)); kfiles= (PAGECACHE_FILE *)my_realloc((uchar *)kfiles, /* avoid size of 0 for my_realloc */ max(1, nb) * sizeof(PAGECACHE_FILE), - MYF(MY_WME)); + MYF(MY_WME | MY_ALLOW_ZERO_PTR)); if (unlikely((state_copies == NULL) || (dfiles == NULL) || (kfiles == NULL))) goto err; -- cgit v1.2.1 From 5becb1e4040884cde2f6537db7c3daef0b5e5eb9 Mon Sep 17 00:00:00 2001 From: unknown Date: Wed, 12 Sep 2007 15:10:41 +0200 Subject: fix for pushbuild test failures mysql-test/r/rpl_row_flsh_tbls.result: result update mysql-test/r/rpl_row_insert_delayed.result: result update mysql-test/t/rpl_row_flsh_tbls.test: CREATE TABLE statement got an ENGINE clause so became longer --- mysql-test/r/rpl_row_flsh_tbls.result | 8 ++++---- mysql-test/r/rpl_row_insert_delayed.result | 2 +- mysql-test/t/rpl_row_flsh_tbls.test | 2 +- 3 files changed, 6 insertions(+), 6 deletions(-) diff --git a/mysql-test/r/rpl_row_flsh_tbls.result b/mysql-test/r/rpl_row_flsh_tbls.result index 942a6b83bf6..b2fd110973f 100644 --- a/mysql-test/r/rpl_row_flsh_tbls.result +++ b/mysql-test/r/rpl_row_flsh_tbls.result @@ -4,21 +4,21 @@ reset master; reset slave; drop table if exists t1,t2,t3,t4,t5,t6,t7,t8,t9; start slave; -create table t1 (a int); +create table t1 (a int) ENGINE=MyISAM; insert into t1 values (10); -create table t2 (a int); +create table t2 (a int) ENGINE=MyISAM; create table t3 (a int) engine=merge union(t1); create table t4 (a int); insert into t4 select * from t3; rename table t1 to t5, t2 to t1; flush no_write_to_binlog tables; -SHOW BINLOG EVENTS FROM 619 ; +SHOW BINLOG EVENTS FROM 647 ; Log_name Pos Event_type Server_id End_log_pos Info master-bin.000001 # Query 1 # use `test`; rename table t1 to t5, t2 to t1 select * from t3; a flush tables; -SHOW BINLOG EVENTS FROM 619 ; +SHOW BINLOG EVENTS FROM 647 ; Log_name Pos Event_type Server_id End_log_pos Info master-bin.000001 # Query 1 # use `test`; rename table t1 to t5, t2 to t1 master-bin.000001 # Query 1 # use `test`; flush tables diff --git a/mysql-test/r/rpl_row_insert_delayed.result b/mysql-test/r/rpl_row_insert_delayed.result index 2044672f49d..bec410cc3fa 100644 --- a/mysql-test/r/rpl_row_insert_delayed.result +++ b/mysql-test/r/rpl_row_insert_delayed.result @@ -11,7 +11,7 @@ USE mysqlslap; select @@global.binlog_format; @@global.binlog_format ROW -CREATE TABLE t1 (id INT primary key auto_increment, name VARCHAR(64)); +CREATE TABLE t1 (id INT primary key auto_increment, name VARCHAR(64)) ENGINE=MyISAM; FLUSH TABLE t1; SELECT COUNT(*) FROM t1; COUNT(*) diff --git a/mysql-test/t/rpl_row_flsh_tbls.test b/mysql-test/t/rpl_row_flsh_tbls.test index a2f9e31fc5d..b96bb50b3cc 100644 --- a/mysql-test/t/rpl_row_flsh_tbls.test +++ b/mysql-test/t/rpl_row_flsh_tbls.test @@ -1,7 +1,7 @@ # depends on the binlog output -- source include/have_binlog_format_row.inc -let $rename_event_pos= 619; +let $rename_event_pos= 647; # Bug#18326: Do not lock table for writing during prepare of statement # The use of the ps protocol causes extra table maps in the binlog, so -- cgit v1.2.1 From 9b2663926b749845dd43c0b96ec3d03aaf00ac01 Mon Sep 17 00:00:00 2001 From: unknown Date: Wed, 12 Sep 2007 19:18:52 +0200 Subject: MY_ALLOW_ZERO_PTR in my_realloc() to fix safemalloc errors in pushbuild storage/maria/ma_recovery.c: MY_ALLOW_ZERO_PTR needed as log_record_buffer.str is initially NULL. --- storage/maria/ma_recovery.c | 15 +++++++++------ 1 file changed, 9 insertions(+), 6 deletions(-) diff --git a/storage/maria/ma_recovery.c b/storage/maria/ma_recovery.c index 3d388518583..653887d7ae8 100644 --- a/storage/maria/ma_recovery.c +++ b/storage/maria/ma_recovery.c @@ -107,13 +107,16 @@ static int close_all_tables(); /** @brief global [out] buffer for translog_read_record(); never shrinks */ static LEX_STRING log_record_buffer; -#define enlarge_buffer(rec) \ - if (log_record_buffer.length < (rec)->record_length) \ - { \ - log_record_buffer.length= (rec)->record_length; \ - log_record_buffer.str= my_realloc(log_record_buffer.str, \ - (rec)->record_length, MYF(MY_WME)); \ +static void enlarge_buffer(const TRANSLOG_HEADER_BUFFER *rec) +{ + if (log_record_buffer.length < rec->record_length) + { + log_record_buffer.length= rec->record_length; + log_record_buffer.str= my_realloc(log_record_buffer.str, + rec->record_length, + MYF(MY_WME | MY_ALLOW_ZERO_PTR)); } +} #define ALERT_USER() DBUG_ASSERT(0) #define LSN_IN_HEX(L) (ulong)LSN_FILE_NO(L),(ulong)LSN_OFFSET(L) -- cgit v1.2.1 From a303f5b2c8f7e93e6ce25db654fbcbee4ea593a2 Mon Sep 17 00:00:00 2001 From: unknown Date: Thu, 13 Sep 2007 10:37:51 +0300 Subject: Fixes of the empty log problem. storage/maria/ma_checkpoint.c: The new macro for easier printing LSN added. storage/maria/ma_loghandler.c: The assertion returned. The new macro for easier printing LSN added. storage/maria/ma_loghandler_lsn.h: The new macro for easier printing LSN added. storage/maria/ma_pagecache.c: The new macro for easier printing LSN added. storage/maria/ma_recovery.c: Recovery checks empty log state. RECHEADER_READ_ERROR means some real error. storage/maria/maria_read_log.c: Read log starts from real beggining of the log and precess error and empty log states. The new macro for easier printing LSN added. storage/maria/unittest/ma_test_loghandler-t.c: The new macro for easier printing LSN added. storage/maria/unittest/ma_test_loghandler_first_lsn-t.c: The new macro for easier printing LSN added. storage/maria/unittest/ma_test_loghandler_max_lsn-t.c: The new macro for easier printing LSN added. storage/maria/unittest/ma_test_loghandler_multigroup-t.c: The new macro for easier printing LSN added. storage/maria/unittest/ma_test_loghandler_multithread-t.c: The new macro for easier printing LSN added. storage/maria/unittest/ma_test_loghandler_noflush-t.c: The new macro for easier printing LSN added. --- storage/maria/ma_checkpoint.c | 3 +- storage/maria/ma_loghandler.c | 217 +++++++-------------- storage/maria/ma_loghandler_lsn.h | 3 + storage/maria/ma_pagecache.c | 3 +- storage/maria/ma_recovery.c | 45 +++-- storage/maria/maria_read_log.c | 17 +- storage/maria/unittest/ma_test_loghandler-t.c | 35 ++-- .../unittest/ma_test_loghandler_first_lsn-t.c | 8 +- .../maria/unittest/ma_test_loghandler_max_lsn-t.c | 8 +- .../unittest/ma_test_loghandler_multigroup-t.c | 29 ++- .../unittest/ma_test_loghandler_multithread-t.c | 15 +- .../maria/unittest/ma_test_loghandler_noflush-t.c | 3 +- 12 files changed, 150 insertions(+), 236 deletions(-) diff --git a/storage/maria/ma_checkpoint.c b/storage/maria/ma_checkpoint.c index ef09b650820..8c3f2c0a2e2 100644 --- a/storage/maria/ma_checkpoint.c +++ b/storage/maria/ma_checkpoint.c @@ -191,9 +191,8 @@ static int really_execute_checkpoint() this horizon. */ checkpoint_start_log_horizon= translog_get_horizon(); -#define LSN_IN_HEX(L) (ulong)LSN_FILE_NO(L),(ulong)LSN_OFFSET(L) DBUG_PRINT("info",("checkpoint_start_log_horizon (%lu,0x%lx)", - LSN_IN_HEX(checkpoint_start_log_horizon))); + LSN_IN_PARTS(checkpoint_start_log_horizon))); lsn_store(checkpoint_start_log_horizon_char, checkpoint_start_log_horizon); diff --git a/storage/maria/ma_loghandler.c b/storage/maria/ma_loghandler.c index f42af62d202..3470e1d408b 100644 --- a/storage/maria/ma_loghandler.c +++ b/storage/maria/ma_loghandler.c @@ -765,7 +765,7 @@ static my_bool translog_max_lsn_to_header(File file, LSN lsn) DBUG_PRINT("enter", ("File descriptor: %ld " "lsn: (%lu,0x%lx)", (long) file, - (ulong) LSN_FILE_NO(lsn),(ulong) LSN_OFFSET(lsn))); + LSN_IN_PARTS(lsn))); lsn_store(lsn_buff, lsn); @@ -860,7 +860,7 @@ static my_bool translog_set_lsn_for_files(uint32 from_file, uint32 to_file, DBUG_ENTER("translog_set_lsn_for_files"); DBUG_PRINT("enter", ("From: %lu to: %lu lsn: (%lu,0x%lx) locked: %d", (ulong) from_file, (ulong) to_file, - (ulong) LSN_FILE_NO(lsn), (ulong) LSN_OFFSET(lsn), + LSN_IN_PARTS(lsn), is_locked)); DBUG_ASSERT(from_file <= to_file); DBUG_ASSERT(from_file > 0); /* we have not file 0 */ @@ -1067,8 +1067,7 @@ LSN translog_get_file_max_lsn_stored(uint32 file) DBUG_RETURN(LSN_ERROR); } DBUG_PRINT("error", ("Max lsn: (%lu,0x%lx)", - (ulong) LSN_FILE_NO(info.max_lsn), - (ulong) LSN_OFFSET(info.max_lsn))); + LSN_IN_PARTS(info.max_lsn))); DBUG_RETURN(info.max_lsn); } } @@ -1280,8 +1279,7 @@ static void translog_new_page_header(TRANSLOG_ADDRESS *horizon, { #ifndef DBUG_OFF DBUG_PRINT("info", ("write 0x11223344 CRC to (%lu,0x%lx)", - (ulong) LSN_FILE_NO(*horizon), - (ulong) LSN_OFFSET(*horizon))); + LSN_IN_PARTS(*horizon))); /* This will be overwritten by real CRC; This is just for debugging */ int4store(ptr, 0x11223344); #endif @@ -1414,8 +1412,7 @@ static void translog_finish_page(TRANSLOG_ADDRESS *horizon, "Page addr: (%lu,0x%lx) " "size:%lu (%lu) Pg:%u left:%u", (uint) cursor->buffer_no, (ulong) cursor->buffer, - (ulong) LSN_FILE_NO(cursor->buffer->offset), - (ulong) LSN_OFFSET(cursor->buffer->offset), + LSN_IN_PARTS(cursor->buffer->offset), (ulong) LSN_FILE_NO(*horizon), (ulong) (LSN_OFFSET(*horizon) - cursor->current_page_fill), @@ -1647,8 +1644,7 @@ static my_bool translog_buffer_next(TRANSLOG_ADDRESS *horizon, DBUG_ENTER("translog_buffer_next"); DBUG_PRINT("info", ("horizon: (%lu,0x%lx) chasing: %d", - (ulong) LSN_FILE_NO(log_descriptor.horizon), - (ulong) LSN_OFFSET(log_descriptor.horizon), chasing)); + LSN_IN_PARTS(log_descriptor.horizon), chasing)); DBUG_ASSERT(cmp_translog_addr(log_descriptor.horizon, *horizon) >= 0); @@ -1703,12 +1699,9 @@ static void translog_set_sent_to_file(LSN lsn, TRANSLOG_ADDRESS in_buffers) pthread_mutex_lock(&log_descriptor.sent_to_file_lock); DBUG_PRINT("enter", ("lsn: (%lu,0x%lx) in_buffers: (%lu,0x%lx) " "in_buffers_only: (%lu,0x%lx)", - (ulong) LSN_FILE_NO(lsn), - (ulong) LSN_OFFSET(lsn), - (ulong) LSN_FILE_NO(in_buffers), - (ulong) LSN_OFFSET(in_buffers), - (ulong) LSN_FILE_NO(log_descriptor.in_buffers_only), - (ulong) LSN_OFFSET(log_descriptor.in_buffers_only))); + LSN_IN_PARTS(lsn), + LSN_IN_PARTS(in_buffers), + LSN_IN_PARTS(log_descriptor.in_buffers_only))); DBUG_ASSERT(cmp_translog_addr(lsn, log_descriptor.sent_to_file) >= 0); log_descriptor.sent_to_file= lsn; /* LSN_IMPOSSIBLE == 0 => it will work for very first time */ @@ -1737,10 +1730,8 @@ static void translog_set_only_in_buffers(TRANSLOG_ADDRESS in_buffers) pthread_mutex_lock(&log_descriptor.sent_to_file_lock); DBUG_PRINT("enter", ("in_buffers: (%lu,0x%lx) " "in_buffers_only: (%lu,0x%lx)", - (ulong) LSN_FILE_NO(in_buffers), - (ulong) LSN_OFFSET(in_buffers), - (ulong) LSN_FILE_NO(log_descriptor.in_buffers_only), - (ulong) LSN_OFFSET(log_descriptor.in_buffers_only))); + LSN_IN_PARTS(in_buffers), + LSN_IN_PARTS(log_descriptor.in_buffers_only))); /* LSN_IMPOSSIBLE == 0 => it will work for very first time */ if (cmp_translog_addr(in_buffers, log_descriptor.in_buffers_only) > 0) { @@ -2010,8 +2001,7 @@ static my_bool translog_buffer_flush(struct st_translog_buffer *buffer) "file: %d offset: (%lu,0x%lx) size: %lu", (uint) buffer->buffer_no, (ulong) buffer, buffer->file, - (ulong) LSN_FILE_NO(buffer->offset), - (ulong) LSN_OFFSET(buffer->offset), + LSN_IN_PARTS(buffer->offset), (ulong) buffer->size)); DBUG_ASSERT(buffer->file != -1); @@ -2174,7 +2164,7 @@ static my_bool translog_page_validator(uchar *page_addr, uchar* data_ptr) UNRECOVERABLE_ERROR(("Page (%lu,0x%lx): " "page address written in the page is incorrect: " "File %lu instead of %lu or page %lu instead of %lu", - (ulong) LSN_FILE_NO(addr), (ulong) LSN_OFFSET(addr), + LSN_IN_PARTS(addr), (ulong) uint3korr(page + 3), (ulong) LSN_FILE_NO(addr), (ulong) uint3korr(page), (ulong) LSN_OFFSET(addr) / TRANSLOG_PAGE_SIZE)); @@ -2187,8 +2177,7 @@ static my_bool translog_page_validator(uchar *page_addr, uchar* data_ptr) { UNRECOVERABLE_ERROR(("Page (%lu,0x%lx): " "Garbage in the page flags field detected : %x", - (ulong) LSN_FILE_NO(addr), (ulong) LSN_OFFSET(addr), - (uint) flags)); + LSN_IN_PARTS(addr), (uint) flags)); DBUG_RETURN(1); } page_pos= page + (3 + 3 + 1); @@ -2201,7 +2190,7 @@ static my_bool translog_page_validator(uchar *page_addr, uchar* data_ptr) { UNRECOVERABLE_ERROR(("Page (%lu,0x%lx): " "CRC mismatch: calculated: %lx on the page %lx", - (ulong) LSN_FILE_NO(addr), (ulong) LSN_OFFSET(addr), + LSN_IN_PARTS(addr), (ulong) crc, (ulong) uint4korr(page_pos))); DBUG_RETURN(1); } @@ -2335,8 +2324,7 @@ static uchar *translog_get_page(TRANSLOG_VALIDATOR_DATA *data, uchar *buffer) in_buffers= translog_only_in_buffers(); DBUG_PRINT("info", ("in_buffers: (%lu,0x%lx)", - (ulong) LSN_FILE_NO(in_buffers), - (ulong) LSN_OFFSET(in_buffers))); + LSN_IN_PARTS(in_buffers))); if (in_buffers != LSN_IMPOSSIBLE && cmp_translog_addr(addr, in_buffers) >= 0) { @@ -2983,8 +2971,7 @@ static void translog_buffer_destroy(struct st_translog_buffer *buffer) ("Buffer #%u: 0x%lx file: %d offset: (%lu,0x%lx) size: %lu", (uint) buffer->buffer_no, (ulong) buffer, buffer->file, - (ulong) LSN_FILE_NO(buffer->offset), - (ulong) LSN_OFFSET(buffer->offset), + LSN_IN_PARTS(buffer->offset), (ulong) buffer->size)); DBUG_ASSERT(buffer->waiting_filling_buffer.last_thread == 0); if (buffer->file != -1) @@ -3240,8 +3227,7 @@ static my_bool translog_write_parts_on_page(TRANSLOG_ADDRESS *horizon, } while (left); DBUG_PRINT("info", ("Horizon: (%lu,0x%lx) Length %lu(0x%lx)", - (ulong) LSN_FILE_NO(*horizon), - (ulong) LSN_OFFSET(*horizon), + LSN_IN_PARTS(*horizon), (ulong) length, (ulong) length)); parts->current= cur; (*horizon)+= length; /* offset increasing */ @@ -3254,8 +3240,7 @@ static my_bool translog_write_parts_on_page(TRANSLOG_ADDRESS *horizon, (uint) cursor->buffer->buffer_no, (ulong) cursor->buffer, cursor->chaser, (ulong) cursor->buffer->size, (ulong) (cursor->ptr - cursor->buffer->buffer), - (ulong) LSN_FILE_NO(*horizon), - (ulong) LSN_OFFSET(*horizon), + LSN_IN_PARTS(*horizon), (ulong) (LSN_OFFSET(cursor->buffer->offset) + cursor->buffer->size))); DBUG_EXECUTE("info", translog_check_cursor(cursor);); @@ -3473,8 +3458,7 @@ static my_bool translog_advance_pointer(uint pages, uint16 last_page_data) translog_size_t buffer_end_offset, file_end_offset, min_offset; DBUG_ENTER("translog_advance_pointer"); DBUG_PRINT("enter", ("Pointer: (%lu, 0x%lx) + %u + %u pages + %u + %u", - (ulong) LSN_FILE_NO(log_descriptor.horizon), - (ulong) LSN_OFFSET(log_descriptor.horizon), + LSN_IN_PARTS(log_descriptor.horizon), (uint) (TRANSLOG_PAGE_SIZE - log_descriptor.bc.current_page_fill), pages, (uint) log_descriptor.page_overhead, @@ -3576,8 +3560,7 @@ static my_bool translog_advance_pointer(uint pages, uint16 last_page_data) (uint) last_page_offset)); DBUG_PRINT("info", ("pointer moved to: (%lu, 0x%lx)", - (ulong) LSN_FILE_NO(log_descriptor.horizon), - (ulong) LSN_OFFSET(log_descriptor.horizon))); + LSN_IN_PARTS(log_descriptor.horizon))); DBUG_EXECUTE("info", translog_check_cursor(&log_descriptor.bc);); log_descriptor.bc.protected= 0; DBUG_RETURN(0); @@ -3755,10 +3738,8 @@ translog_write_variable_record_1group(LSN *lsn, DBUG_PRINT("info", ("absolute horizon: (%lu,0x%lx) local: (%lu,0x%lx)", - (ulong) LSN_FILE_NO(log_descriptor.horizon), - (ulong) LSN_OFFSET(log_descriptor.horizon), - (ulong) LSN_FILE_NO(horizon), - (ulong) LSN_OFFSET(horizon))); + LSN_IN_PARTS(log_descriptor.horizon), + LSN_IN_PARTS(horizon))); for (i= 0; i < full_pages; i++) { @@ -3766,10 +3747,8 @@ translog_write_variable_record_1group(LSN *lsn, DBUG_RETURN(1); DBUG_PRINT("info", ("absolute horizon: (%lu,0x%lx) local: (%lu,0x%lx)", - (ulong) LSN_FILE_NO(log_descriptor.horizon), - (ulong) LSN_OFFSET(log_descriptor.horizon), - (ulong) LSN_FILE_NO(horizon), - (ulong) LSN_OFFSET(horizon))); + LSN_IN_PARTS(log_descriptor.horizon), + LSN_IN_PARTS(horizon))); } if (additional_chunk3_page) @@ -3780,10 +3759,8 @@ translog_write_variable_record_1group(LSN *lsn, &horizon, &cursor)) DBUG_RETURN(1); DBUG_PRINT("info", ("absolute horizon: (%lu,0x%lx) local: (%lu,0x%lx)", - (ulong) LSN_FILE_NO(log_descriptor.horizon), - (ulong) LSN_OFFSET(log_descriptor.horizon), - (ulong) LSN_FILE_NO(horizon), - (ulong) LSN_OFFSET(horizon))); + LSN_IN_PARTS(log_descriptor.horizon), + LSN_IN_PARTS(horizon))); DBUG_ASSERT(cursor.current_page_fill == TRANSLOG_PAGE_SIZE); } @@ -3909,10 +3886,8 @@ static uchar *translog_put_LSN_diff(LSN base_lsn, LSN lsn, uchar *dst) { DBUG_ENTER("translog_put_LSN_diff"); DBUG_PRINT("enter", ("Base: (0x%lu,0x%lx) val: (0x%lu,0x%lx) dst: 0x%lx", - (ulong) LSN_FILE_NO(base_lsn), - (ulong) LSN_OFFSET(base_lsn), - (ulong) LSN_FILE_NO(lsn), - (ulong) LSN_OFFSET(lsn), (ulong) dst)); + LSN_IN_PARTS(base_lsn), LSN_IN_PARTS(lsn), + (ulong) dst)); if (LSN_FILE_NO(base_lsn) == LSN_FILE_NO(lsn)) { uint32 diff; @@ -4022,9 +3997,7 @@ static uchar *translog_get_LSN_from_diff(LSN base_lsn, uchar *src, uchar *dst) uint8 code; DBUG_ENTER("translog_get_LSN_from_diff"); DBUG_PRINT("enter", ("Base: (0x%lx,0x%lx) src: 0x%lx dst 0x%lx", - (ulong) LSN_FILE_NO(base_lsn), - (ulong) LSN_OFFSET(base_lsn), - (ulong) src, (ulong) dst)); + LSN_IN_PARTS(base_lsn), (ulong) src, (ulong) dst)); first_byte= *((uint8*) src); code= first_byte >> 6; /* Length is in 2 most significant bits */ first_byte&= 0x3F; @@ -4312,10 +4285,8 @@ translog_write_variable_record_mgroup(LSN *lsn, translog_write_parts_on_page(&horizon, &cursor, first_page - 1, parts); DBUG_PRINT("info", ("absolute horizon: (%lu,0x%lx) local: (%lu,0x%lx) " "Left %lu", - (ulong) LSN_FILE_NO(log_descriptor.horizon), - (ulong) LSN_OFFSET(log_descriptor.horizon), - (ulong) LSN_FILE_NO(horizon), - (ulong) LSN_OFFSET(horizon), + LSN_IN_PARTS(log_descriptor.horizon), + LSN_IN_PARTS(horizon), (ulong) (parts->record_length - (first_page - 1) - done))); @@ -4327,10 +4298,8 @@ translog_write_variable_record_mgroup(LSN *lsn, DBUG_PRINT("info", ("absolute horizon: (%lu,0x%lx) " "local: (%lu,0x%lx) " "Left: %lu", - (ulong) LSN_FILE_NO(log_descriptor.horizon), - (ulong) LSN_OFFSET(log_descriptor.horizon), - (ulong) LSN_FILE_NO(horizon), - (ulong) LSN_OFFSET(horizon), + LSN_IN_PARTS(log_descriptor.horizon), + LSN_IN_PARTS(horizon), (ulong) (parts->record_length - (first_page - 1) - i * log_descriptor.page_capacity_chunk_2 - done))); @@ -4456,10 +4425,8 @@ translog_write_variable_record_mgroup(LSN *lsn, translog_write_parts_on_page(&horizon, &cursor, first_page - 1, parts); DBUG_PRINT("info", ("absolute horizon: (%lu,0x%lx) local: (%lu,0x%lx) " "Left: %lu", - (ulong) LSN_FILE_NO(log_descriptor.horizon), - (ulong) LSN_OFFSET(log_descriptor.horizon), - (ulong) LSN_FILE_NO(horizon), - (ulong) LSN_OFFSET(horizon), + LSN_IN_PARTS(log_descriptor.horizon), + LSN_IN_PARTS(horizon), (ulong) (parts->record_length - (first_page - 1) - done))); } @@ -4475,10 +4442,8 @@ translog_write_variable_record_mgroup(LSN *lsn, translog_write_parts_on_page(&horizon, &cursor, chunk3_size, parts); DBUG_PRINT("info", ("absolute horizon: (%lu,0x%lx) local: (%lu,0x%lx) " "Left: %lu", - (ulong) LSN_FILE_NO(log_descriptor.horizon), - (ulong) LSN_OFFSET(log_descriptor.horizon), - (ulong) LSN_FILE_NO(horizon), - (ulong) LSN_OFFSET(horizon), + LSN_IN_PARTS(log_descriptor.horizon), + LSN_IN_PARTS(horizon), (ulong) (parts->record_length - chunk3_size - done))); } else @@ -4495,10 +4460,8 @@ translog_write_variable_record_mgroup(LSN *lsn, DBUG_PRINT("info", ("absolute horizon: (%lu,0x%lx) local: (%lu,0x%lx) " "Left: %lu", - (ulong) LSN_FILE_NO(log_descriptor.horizon), - (ulong) LSN_OFFSET(log_descriptor.horizon), - (ulong) LSN_FILE_NO(horizon), - (ulong) LSN_OFFSET(horizon), + LSN_IN_PARTS(log_descriptor.horizon), + LSN_IN_PARTS(horizon), (ulong) (parts->record_length - (first_page - 1) - i * log_descriptor.page_capacity_chunk_2 - done))); @@ -4510,11 +4473,8 @@ translog_write_variable_record_mgroup(LSN *lsn, &horizon, &cursor)) goto err; DBUG_PRINT("info", ("absolute horizon: (%lu,0x%lx) local: (%lu,0x%lx)", - (ulong) LSN_FILE_NO(log_descriptor.horizon), - (ulong) LSN_OFFSET(log_descriptor.horizon), - (ulong) LSN_FILE_NO(horizon), - (ulong) LSN_OFFSET(horizon))); - + LSN_IN_PARTS(log_descriptor.horizon), + LSN_IN_PARTS(horizon))); *chunk0_header= (uchar) (type |TRANSLOG_CHUNK_LSN); int2store(chunk0_header + 1, short_trid); @@ -4660,8 +4620,7 @@ static my_bool translog_write_variable_record(LSN *lsn, translog_lock(); DBUG_PRINT("info", ("horizon: (%lu,0x%lx)", - (ulong) LSN_FILE_NO(log_descriptor.horizon), - (ulong) LSN_OFFSET(log_descriptor.horizon))); + LSN_IN_PARTS(log_descriptor.horizon))); page_rest= TRANSLOG_PAGE_SIZE - log_descriptor.bc.current_page_fill; DBUG_PRINT("info", ("header length: %u page_rest: %u", header_length1, page_rest)); @@ -4791,8 +4750,7 @@ static my_bool translog_write_fixed_record(LSN *lsn, translog_lock(); DBUG_PRINT("info", ("horizon: (%lu,0x%lx)", - (ulong) LSN_FILE_NO(log_descriptor.horizon), - (ulong) LSN_OFFSET(log_descriptor.horizon))); + LSN_IN_PARTS(log_descriptor.horizon))); DBUG_ASSERT(log_descriptor.bc.current_page_fill <= TRANSLOG_PAGE_SIZE); DBUG_PRINT("info", @@ -5035,8 +4993,7 @@ my_bool translog_write_record(LSN *lsn, } } - DBUG_PRINT("info", ("LSN: (%lu,0x%lx)", (ulong) LSN_FILE_NO(*lsn), - (ulong) LSN_OFFSET(*lsn))); + DBUG_PRINT("info", ("LSN: (%lu,0x%lx)", LSN_IN_PARTS(*lsn))); DBUG_RETURN(rc); } @@ -5207,9 +5164,7 @@ my_bool translog_init_scanner(LSN lsn, { TRANSLOG_VALIDATOR_DATA data; DBUG_ENTER("translog_init_scanner"); - DBUG_PRINT("enter", ("LSN: (0x%lu,0x%lx)", - (ulong) LSN_FILE_NO(lsn), - (ulong) LSN_OFFSET(lsn)); + DBUG_PRINT("enter", ("LSN: (0x%lu,0x%lx)", LSN_IN_PARTS(lsn)); DBUG_ASSERT(LSN_OFFSET(lsn) % TRANSLOG_PAGE_SIZE != 0); data.addr= &scanner->page_addr; @@ -5221,8 +5176,7 @@ my_bool translog_init_scanner(LSN lsn, scanner->horizon= translog_get_horizon(); DBUG_PRINT("info", ("horizon: (0x%lu,0x%lx)", - (ulong) LSN_FILE_NO(scanner->horizon), - (ulong) LSN_OFFSET(scanner->horizon))); + LSN_IN_PARTS(scanner->horizon))); /* lsn < horizon */ DBUG_ASSERT(lsn < scanner->horizon)); @@ -5256,10 +5210,8 @@ static my_bool translog_scanner_eol(TRANSLOG_SCANNER_DATA *scanner) DBUG_ENTER("translog_scanner_eol"); DBUG_PRINT("enter", ("Horizon: (%lu, 0x%lx) Current: (%lu, 0x%lx+0x%x=0x%lx)", - (ulong) LSN_FILE_NO(scanner->horizon), - (ulong) LSN_OFFSET(scanner->horizon), - (ulong) LSN_FILE_NO(scanner->page_addr), - (ulong) LSN_OFFSET(scanner->page_addr), + LSN_IN_PARTS(scanner->horizon), + LSN_IN_PARTS(scanner->page_addr), (uint) scanner->page_offset, (ulong) (LSN_OFFSET(scanner->page_addr) + scanner->page_offset))); if (scanner->horizon > (scanner->page_addr + @@ -5371,10 +5323,8 @@ translog_get_next_chunk(TRANSLOG_SCANNER_DATA *scanner) if (translog_scanner_eof(scanner)) { DBUG_PRINT("info", ("horizon: (%lu,0x%lx) pageaddr: (%lu,0x%lx)", - (ulong) LSN_FILE_NO(scanner->horizon), - (ulong) LSN_OFFSET(scanner->horizon), - (ulong) LSN_FILE_NO(scanner->page_addr), - (ulong) LSN_OFFSET(scanner->page_addr))); + LSN_IN_PARTS(scanner->horizon), + LSN_IN_PARTS(scanner->page_addr))); /* if it is log end it have to be caught before */ DBUG_ASSERT(LSN_FILE_NO(scanner->horizon) > LSN_FILE_NO(scanner->page_addr)); @@ -5492,8 +5442,7 @@ int translog_variable_length_header(uchar *page, translog_size_t page_offset, buff->groups[curr].num= src[i * (7 + 1) + 7]; DBUG_PRINT("info", ("group #%u (%lu,0x%lx) chunks: %u", curr, - (ulong) LSN_FILE_NO(buff->groups[curr].addr), - (ulong) LSN_OFFSET(buff->groups[curr].addr), + LSN_IN_PARTS(buff->groups[curr].addr), (uint) buff->groups[curr].num)); } grp_no-= read; @@ -5513,8 +5462,7 @@ int translog_variable_length_header(uchar *page, translog_size_t page_offset, } buff->chunk0_data_len= chunk_len - 2 - read * (7 + 1); DBUG_PRINT("info", ("Data address: (%lu,0x%lx) len: %u", - (ulong) LSN_FILE_NO(buff->chunk0_data_addr), - (ulong) LSN_OFFSET(buff->chunk0_data_addr), + LSN_IN_PARTS(buff->chunk0_data_addr), buff->chunk0_data_len)); break; } @@ -5612,8 +5560,7 @@ int translog_read_record_header_from_buffer(uchar *page, buff->short_trid= uint2korr(page + page_offset + 1); DBUG_PRINT("info", ("Type %u, Short TrID %u, LSN (%lu,0x%lx)", (uint) buff->type, (uint)buff->short_trid, - (ulong) LSN_FILE_NO(buff->lsn), - (ulong) LSN_OFFSET(buff->lsn))); + LSN_IN_PARTS(buff->lsn))); /* Read required bytes from the header and call hook */ switch (log_record_type_descriptor[buff->type].class) { case LOGRECTYPE_VARIABLE_LENGTH: @@ -5625,9 +5572,7 @@ int translog_read_record_header_from_buffer(uchar *page, res= translog_fixed_length_header(page, page_offset, buff); break; default: -#ifdef ASK_SANJA - DBUG_ASSERT(0); /* fails on empty log (Sanja knows) */ -#endif + DBUG_ASSERT(0); /* we read some junk (got no LSN) */ res= RECHEADER_READ_ERROR; } DBUG_RETURN(res); @@ -5660,8 +5605,7 @@ int translog_read_record_header(LSN lsn, TRANSLOG_HEADER_BUFFER *buff) TRANSLOG_ADDRESS addr; TRANSLOG_VALIDATOR_DATA data; DBUG_ENTER("translog_read_record_header"); - DBUG_PRINT("enter", ("LSN: (0x%lu,0x%lx)", - (ulong) LSN_FILE_NO(lsn), (ulong) LSN_OFFSET(lsn))); + DBUG_PRINT("enter", ("LSN: (0x%lu,0x%lx)", LSN_IN_PARTS(lsn))); DBUG_ASSERT(LSN_OFFSET(lsn) % TRANSLOG_PAGE_SIZE != 0); buff->lsn= lsn; @@ -5704,12 +5648,9 @@ int translog_read_record_header_scan(TRANSLOG_SCANNER_DATA *scanner, DBUG_ENTER("translog_read_record_header_scan"); DBUG_PRINT("enter", ("Scanner: Cur: (%lu,0x%lx) Hrz: (%lu,0x%lx) " "Lst: (%lu,0x%lx) Offset: %u(%x) fixed %d", - (ulong) LSN_FILE_NO(scanner->page_addr), - (ulong) LSN_OFFSET(scanner->page_addr), - (ulong) LSN_FILE_NO(scanner->horizon), - (ulong) LSN_OFFSET(scanner->horizon), - (ulong) LSN_FILE_NO(scanner->last_file_page), - (ulong) LSN_OFFSET(scanner->last_file_page), + LSN_IN_PARTS(scanner->page_addr), + LSN_IN_PARTS(scanner->horizon), + LSN_IN_PARTS(scanner->last_file_page), (uint) scanner->page_offset, (uint) scanner->page_offset, scanner->fixed_horizon)); buff->groups_no= 0; @@ -5753,12 +5694,9 @@ int translog_read_next_record_header(TRANSLOG_SCANNER_DATA *scanner, DBUG_PRINT("enter", ("scanner: 0x%lx", (ulong) scanner)); DBUG_PRINT("info", ("Scanner: Cur: (%lu,0x%lx) Hrz: (%lu,0x%lx) " "Lst: (%lu,0x%lx) Offset: %u(%x) fixed: %d", - (ulong) LSN_FILE_NO(scanner->page_addr), - (ulong) LSN_OFFSET(scanner->page_addr), - (ulong) LSN_FILE_NO(scanner->horizon), - (ulong) LSN_OFFSET(scanner->horizon), - (ulong) LSN_FILE_NO(scanner->last_file_page), - (ulong) LSN_OFFSET(scanner->last_file_page), + LSN_IN_PARTS(scanner->page_addr), + LSN_IN_PARTS(scanner->horizon), + LSN_IN_PARTS(scanner->last_file_page), (uint) scanner->page_offset, (uint) scanner->page_offset, scanner->fixed_horizon)); @@ -5954,12 +5892,9 @@ translog_size_t translog_read_record(LSN lsn, "Scanner: Cur: (%lu,0x%lx) Hrz: (%lu,0x%lx) " "Lst: (%lu,0x%lx) Offset: %u(%x) fixed: %d", (ulong) offset, (ulong) length, - (ulong) LSN_FILE_NO(data->scanner.page_addr), - (ulong) LSN_OFFSET(data->scanner.page_addr), - (ulong) LSN_FILE_NO(data->scanner.horizon), - (ulong) LSN_OFFSET(data->scanner.horizon), - (ulong) LSN_FILE_NO(data->scanner.last_file_page), - (ulong) LSN_OFFSET(data->scanner.last_file_page), + LSN_IN_PARTS(data->scanner.page_addr), + LSN_IN_PARTS(data->scanner.horizon), + LSN_IN_PARTS(data->scanner.last_file_page), (uint) data->scanner.page_offset, (uint) data->scanner.page_offset, data->scanner.fixed_horizon)); @@ -6039,8 +5974,7 @@ static void translog_force_current_buffer_to_finish() "size: %lu (%lu) Pg: %u left: %u", (uint) log_descriptor.bc.buffer_no, (ulong) log_descriptor.bc.buffer, - (ulong) LSN_FILE_NO(log_descriptor.bc.buffer->offset), - (ulong) LSN_OFFSET(log_descriptor.bc.buffer->offset), + LSN_IN_PARTS(log_descriptor.bc.buffer->offset), (ulong) LSN_FILE_NO(log_descriptor.horizon), (ulong) (LSN_OFFSET(log_descriptor.horizon) - log_descriptor.bc.current_page_fill), @@ -6171,9 +6105,7 @@ my_bool translog_flush(LSN lsn) uint i; my_bool full_circle= 0; DBUG_ENTER("translog_flush"); - DBUG_PRINT("enter", ("Flush up to LSN: (%lu,0x%lx)", - (ulong) LSN_FILE_NO(lsn), - (ulong) LSN_OFFSET(lsn))); + DBUG_PRINT("enter", ("Flush up to LSN: (%lu,0x%lx)", LSN_IN_PARTS(lsn))); translog_lock(); old_flushed= log_descriptor.flushed; @@ -6188,8 +6120,7 @@ my_bool translog_flush(LSN lsn) if (cmp_translog_addr(log_descriptor.flushed, lsn) >= 0) { DBUG_PRINT("info", ("already flushed: (%lu,0x%lx)", - (ulong) LSN_FILE_NO(log_descriptor.flushed), - (ulong) LSN_OFFSET(log_descriptor.flushed))); + LSN_IN_PARTS(log_descriptor.flushed))); translog_unlock(); DBUG_RETURN(0); } @@ -6653,7 +6584,7 @@ static uint32 translog_first_file(TRANSLOG_ADDRESS horizon, int is_protected) @brief returns the LSN of the first record starting in this log @retval LSN_ERROR Error - @retval LSN_IMPOSSIBLE no log + @retval LSN_IMPOSSIBLE no log or the log is empty @retval # LSN of the first record */ @@ -6667,8 +6598,7 @@ LSN translog_first_lsn_in_log() uchar *page; TRANSLOG_SCANNER_DATA scanner; DBUG_ENTER("translog_first_lsn_in_log"); - DBUG_PRINT("info", ("Horizon: (%lu,0x%lx)", - (ulong) LSN_FILE_NO(addr), (ulong) LSN_OFFSET(addr))); + DBUG_PRINT("info", ("Horizon: (%lu,0x%lx)", LSN_IN_PARTS(addr))); if (!(file= translog_first_file(horizon, 0))) { @@ -6719,8 +6649,7 @@ LSN translog_first_theoretical_lsn() uchar buffer[TRANSLOG_PAGE_SIZE], *page; TRANSLOG_VALIDATOR_DATA data; DBUG_ENTER("translog_first_theoretical_lsn"); - DBUG_PRINT("info", ("Horizon: (%lu,0x%lx)", - (ulong) LSN_FILE_NO(addr), (ulong) LSN_OFFSET(addr))); + DBUG_PRINT("info", ("Horizon: (%lu,0x%lx)", LSN_IN_PARTS(addr))); if (!translog_is_file(1)) DBUG_RETURN(LSN_IMPOSSIBLE); @@ -6756,9 +6685,7 @@ my_bool translog_purge(TRANSLOG_ADDRESS low) TRANSLOG_ADDRESS horizon= translog_get_horizon(); int rc= 0; DBUG_ENTER("translog_purge"); - DBUG_PRINT("enter", ("low: (%lu,0x%lx)", - (ulong)LSN_FILE_NO(low), - (ulong)LSN_OFFSET(low))); + DBUG_PRINT("enter", ("low: (%lu,0x%lx)", LSN_IN_PARTS(low))); pthread_mutex_lock(&log_descriptor.purger_lock); if (LSN_FILE_NO(log_descriptor.last_lsn_checked) < last_need_file) diff --git a/storage/maria/ma_loghandler_lsn.h b/storage/maria/ma_loghandler_lsn.h index 5658d8d03e3..e019be16fd2 100644 --- a/storage/maria/ma_loghandler_lsn.h +++ b/storage/maria/ma_loghandler_lsn.h @@ -41,6 +41,9 @@ typedef TRANSLOG_ADDRESS LSN; /* Gets raw file number part of a LSN/log address */ #define LSN_FILE_NO_PART(L) ((L) & ((int64)0xFFFFFF00000000LL)) +/* Parts of LSN for printing */ +#define LSN_IN_PARTS(L) (ulong)LSN_FILE_NO(L),(ulong)LSN_OFFSET(L) + /* Gets record offset of a LSN/log address */ #define LSN_OFFSET(L) ((L) & 0xFFFFFFFFL) diff --git a/storage/maria/ma_pagecache.c b/storage/maria/ma_pagecache.c index 792f0d645ab..9f450d25c50 100755 --- a/storage/maria/ma_pagecache.c +++ b/storage/maria/ma_pagecache.c @@ -2469,8 +2469,7 @@ static void check_and_set_lsn(PAGECACHE *pagecache, DBUG_ASSERT(block->type == PAGECACHE_LSN_PAGE); old= lsn_korr(block->buffer + PAGE_LSN_OFFSET); DBUG_PRINT("info", ("old lsn: (%lu, 0x%lx) new lsn: (%lu, 0x%lx)", - (ulong)LSN_FILE_NO(old), (ulong)LSN_OFFSET(old), - (ulong)LSN_FILE_NO(lsn), (ulong)LSN_OFFSET(lsn))); + LSN_IN_PARTS(old), LSN_IN_PARTS(lsn))); if (cmp_translog_addr(lsn, old) > 0) { diff --git a/storage/maria/ma_recovery.c b/storage/maria/ma_recovery.c index 653887d7ae8..b9bfdecf9f1 100644 --- a/storage/maria/ma_recovery.c +++ b/storage/maria/ma_recovery.c @@ -119,7 +119,6 @@ static void enlarge_buffer(const TRANSLOG_HEADER_BUFFER *rec) } #define ALERT_USER() DBUG_ASSERT(0) -#define LSN_IN_HEX(L) (ulong)LSN_FILE_NO(L),(ulong)LSN_OFFSET(L) /** @@ -270,7 +269,7 @@ static void display_record_position(const LOG_DESC *log_desc, form a group, so we indent below the group's end record */ fprintf(tracef, "%sRec#%u LSN (%lu,0x%lx) short_trid %u %s(num_type:%u) len %lu\n", - number ? "" : " ", number, LSN_IN_HEX(rec->lsn), + number ? "" : " ", number, LSN_IN_PARTS(rec->lsn), rec->short_trid, log_desc->name, rec->type, (ulong)rec->record_length); } @@ -301,7 +300,7 @@ prototype_redo_exec_hook(LONG_TRANSACTION_ID) if (gslsn != LSN_IMPOSSIBLE) { fprintf(tracef, "Group at LSN (%lu,0x%lx) short_trid %u aborted\n", - LSN_IN_HEX(gslsn), sid); + LSN_IN_PARTS(gslsn), sid); all_active_trans[sid].group_start_lsn= LSN_IMPOSSIBLE; } if (long_trid != 0) @@ -314,7 +313,7 @@ prototype_redo_exec_hook(LONG_TRANSACTION_ID) fprintf(tracef, "Found an old transaction long_trid %s short_trid %u" " with same short id as this new transaction, and has neither" " committed nor rollback (undo_lsn: (%lu,0x%lx))\n", llbuf, - sid, LSN_IN_HEX(ulsn)); + sid, LSN_IN_PARTS(ulsn)); goto err; } } @@ -394,7 +393,7 @@ prototype_redo_exec_hook(REDO_CREATE_TABLE) { fprintf(tracef, ", has create_rename_lsn (%lu,0x%lx) more recent than" " record, ignoring creation", - LSN_IN_HEX(share->state.create_rename_lsn)); + LSN_IN_PARTS(share->state.create_rename_lsn)); error= 0; goto end; } @@ -515,7 +514,7 @@ prototype_redo_exec_hook(REDO_RENAME_TABLE) { fprintf(tracef, ", has create_rename_lsn (%lu,0x%lx) more recent than" " record, ignoring renaming", - LSN_IN_HEX(share->state.create_rename_lsn)); + LSN_IN_PARTS(share->state.create_rename_lsn)); error= 0; goto end; } @@ -634,7 +633,7 @@ prototype_redo_exec_hook(REDO_DROP_TABLE) { fprintf(tracef, ", has create_rename_lsn (%lu,0x%lx) more recent than" " record, ignoring removal", - LSN_IN_HEX(share->state.create_rename_lsn)); + LSN_IN_PARTS(share->state.create_rename_lsn)); error= 0; goto end; } @@ -760,8 +759,8 @@ static int new_table(uint16 sid, const char *name, { fprintf(tracef, ", has create_rename_lsn (%lu,0x%lx) more recent than" " LOGREC_FILE_ID's LSN (%lu,0x%lx), ignoring open request", - LSN_IN_HEX(share->state.create_rename_lsn), - LSN_IN_HEX(lsn_of_file_id)); + LSN_IN_PARTS(share->state.create_rename_lsn), + LSN_IN_PARTS(lsn_of_file_id)); error= -1; goto end; } @@ -1063,7 +1062,7 @@ prototype_redo_exec_hook(COMMIT) table, so an unfinished group staid in the log. */ fprintf(tracef, ", with group at LSN (%lu,0x%lx) short_trid %u aborted\n", - (ulong) LSN_FILE_NO(gslsn), (ulong) LSN_OFFSET(gslsn), sid); + LSN_IN_PARTS(gslsn), sid); all_active_trans[sid].group_start_lsn= LSN_IMPOSSIBLE; } else @@ -1093,7 +1092,7 @@ prototype_redo_exec_hook(CLR_END) set_undo_lsn_for_active_trans(rec->short_trid, previous_undo_lsn); fprintf(tracef, " CLR_END was about %s, undo_lsn now LSN (%lu,0x%lx)\n", - log_desc->name, LSN_IN_HEX(previous_undo_lsn)); + log_desc->name, LSN_IN_PARTS(previous_undo_lsn)); if (cmp_translog_addr(rec->lsn, info->s->state.is_of_horizon) > 0) { fprintf(tracef, " state older than record, updating rows' count\n"); @@ -1144,7 +1143,7 @@ prototype_undo_exec_hook(UNDO_ROW_INSERT) /* trn->undo_lsn is updated in an inwrite_hook when writing the CLR_END */ fprintf(tracef, " rows' count %lu\n", (ulong)info->s->state.state.records); fprintf(tracef, " undo_lsn now LSN (%lu,0x%lx)\n", - LSN_IN_HEX(previous_undo_lsn)); + LSN_IN_PARTS(previous_undo_lsn)); return error; } @@ -1186,7 +1185,7 @@ prototype_undo_exec_hook(UNDO_ROW_DELETE) info->trn= 0; fprintf(tracef, " rows' count %lu\n", (ulong)info->s->state.state.records); fprintf(tracef, " undo_lsn now LSN (%lu,0x%lx)\n", - LSN_IN_HEX(previous_undo_lsn)); + LSN_IN_PARTS(previous_undo_lsn)); return error; } @@ -1221,7 +1220,7 @@ prototype_undo_exec_hook(UNDO_ROW_UPDATE) (LSN_STORE_SIZE + FILEID_STORE_SIZE)); info->trn= 0; fprintf(tracef, " undo_lsn now LSN (%lu,0x%lx)\n", - LSN_IN_HEX(previous_undo_lsn)); + LSN_IN_PARTS(previous_undo_lsn)); return error; } @@ -1261,13 +1260,19 @@ static int run_redo_phase(LSN lsn, my_bool apply) TRANSLOG_HEADER_BUFFER rec; + if (unlikely(lsn == translog_get_horizon())) + { + fprintf(tracef, "Cannot find a first record, empty log, nothing to do.\n"); + return 0; + } + int len= translog_read_record_header(lsn, &rec); /** @todo EOF should be detected */ if (len == RECHEADER_READ_ERROR) { - fprintf(tracef, "Cannot find a first record, empty log, nothing to do\n"); - return 0; + fprintf(tracef, "Failed to read header of the first record.\n"); + return 1; } struct st_translog_scanner_data scanner; if (translog_init_scanner(lsn, 1, &scanner)) @@ -1416,7 +1421,7 @@ static uint end_of_redo_phase(my_bool prepare_for_undo_phase) if (gslsn != LSN_IMPOSSIBLE) { fprintf(tracef, "Group at LSN (%lu,0x%lx) short_trid %u aborted\n", - (ulong) LSN_FILE_NO(gslsn), (ulong) LSN_OFFSET(gslsn), sid); + LSN_IN_PARTS(gslsn), sid); ALERT_USER(); } if (all_active_trans[sid].undo_lsn != LSN_IMPOSSIBLE) @@ -1590,7 +1595,7 @@ static MARIA_HA *get_MARIA_HA_from_REDO_record(const DBUG_ASSERT(cmp_translog_addr(rec->lsn, checkpoint_start) < 0); fprintf(tracef, ", table's LOGREC_FILE_ID has LSN (%lu,0x%lx) more recent" " than record, skipping record", - LSN_IN_HEX(info->s->lsn_of_file_id)); + LSN_IN_PARTS(info->s->lsn_of_file_id)); return NULL; } /* detect if an open instance of a dropped table (internal bug) */ @@ -1643,7 +1648,7 @@ static MARIA_HA *get_MARIA_HA_from_UNDO_record(const { fprintf(tracef, ", table's LOGREC_FILE_ID has LSN (%lu,0x%lx) more recent" " than record, skipping record", - LSN_IN_HEX(info->s->lsn_of_file_id)); + LSN_IN_PARTS(info->s->lsn_of_file_id)); return NULL; } DBUG_ASSERT(info->s->last_version != 0); @@ -1670,7 +1675,7 @@ static LSN parse_checkpoint_record(LSN lsn) TRANSLOG_HEADER_BUFFER rec; fprintf(tracef, "Loading data from checkpoint record at LSN (%lu,0x%lx)\n", - LSN_IN_HEX(lsn)); + LSN_IN_PARTS(lsn)); int len= translog_read_record_header(lsn, &rec); if (len == RECHEADER_READ_ERROR) diff --git a/storage/maria/maria_read_log.c b/storage/maria/maria_read_log.c index e47068f50dd..dc537695739 100644 --- a/storage/maria/maria_read_log.c +++ b/storage/maria/maria_read_log.c @@ -86,11 +86,18 @@ int main(int argc, char **argv) printf("You are using --only-display, NOTHING will be written to disk\n"); /* LSN could be also --start-from-lsn=# */ - lsn= translog_first_theoretical_lsn(); - /* - @todo process LSN_IMPOSSIBLE and LSN_ERROR values of - translog_first_theoretical_lsn() - */ + lsn= translog_first_lsn_in_log(); + if (lsn == LSN_ERROR) + { + fprintf(stderr, "Opening transaction log failed\n"); + goto end; + } + if (lsn == LSN_IMPOSSIBLE) + { + fprintf(stdout, "The transaction log is empty\n"); + } + fprintf(stdout, "The transaction log starts from lsn (%lu,0x%lx)\n", + LSN_IN_PARTS(lsn)); fprintf(stdout, "TRACE of the last maria_read_log\n"); if (maria_apply_log(lsn, opt_display_and_apply, stdout, diff --git a/storage/maria/unittest/ma_test_loghandler-t.c b/storage/maria/unittest/ma_test_loghandler-t.c index 170efd6c90f..6ea45f80433 100644 --- a/storage/maria/unittest/ma_test_loghandler-t.c +++ b/storage/maria/unittest/ma_test_loghandler-t.c @@ -88,8 +88,7 @@ void read_ok(TRANSLOG_HEADER_BUFFER *rec) { char buff[80]; snprintf(buff, sizeof(buff), "read record type: %u LSN: (%lu,0x%lx)", - rec->type, (ulong) LSN_FILE_NO(rec->lsn), - (ulong) LSN_OFFSET(rec->lsn)); + rec->type, LSN_IN_PARTS(rec->lsn)); ok(1, buff); } @@ -358,7 +357,7 @@ int main(int argc __attribute__((unused)), char *argv[]) (uint) rec.type, (uint) rec.short_trid, (uint) rec.record_length, (uint) uint4korr(rec.header), (uint) rec.header[4], (uint) rec.header[5], - (ulong) LSN_FILE_NO(rec.lsn), (ulong) LSN_OFFSET(rec.lsn)); + LSN_IN_PARTS(rec.lsn)); goto err; } read_ok(&rec); @@ -403,9 +402,8 @@ int main(int argc __attribute__((unused)), char *argv[]) "lsn(%lu,0x%lx)\n", i, (uint) rec.type, (uint) rec.short_trid, (uint) rec.record_length, - (ulong) LSN_FILE_NO(ref), (ulong) LSN_OFFSET(ref), - (ulong) LSN_FILE_NO(lsn), (ulong) LSN_OFFSET(lsn), - (ulong) LSN_FILE_NO(rec.lsn), (ulong) LSN_OFFSET(rec.lsn)); + LSN_IN_PARTS(ref), LSN_IN_PARTS(lsn), + LSN_IN_PARTS(rec.lsn)); goto err; } } @@ -436,14 +434,13 @@ int main(int argc __attribute__((unused)), char *argv[]) "lsn(%lu,0x%lx)\n", i, (uint) rec.type, (uint) rec.short_trid, (uint) rec.record_length, - (ulong) LSN_FILE_NO(ref1), (ulong) LSN_OFFSET(ref1), - (ulong) LSN_FILE_NO(ref2), (ulong) LSN_OFFSET(ref2), + LSN_IN_PARTS(ref1), LSN_IN_PARTS(ref2), (uint) rec.header[14], (uint) rec.header[15], (uint) rec.header[16], (uint) rec.header[17], (uint) rec.header[18], (uint) rec.header[19], (uint) rec.header[20], (uint) rec.header[21], (uint) rec.header[22], - (ulong) LSN_FILE_NO(rec.lsn), (ulong) LSN_OFFSET(rec.lsn)); + LSN_IN_PARTS(rec.lsn)); goto err; } } @@ -488,8 +485,7 @@ int main(int argc __attribute__((unused)), char *argv[]) rec.record_length != rec_len + LSN_STORE_SIZE, (uint) len, len != 12, - (ulong) LSN_FILE_NO(ref), (ulong) LSN_OFFSET(ref), - (ulong) LSN_FILE_NO(rec.lsn), (ulong) LSN_OFFSET(rec.lsn), + LSN_IN_PARTS(ref), LSN_IN_PARTS(rec.lsn), (len != 12 || ref != lsn), check_content(rec.header + LSN_STORE_SIZE, len - LSN_STORE_SIZE)); @@ -500,7 +496,7 @@ int main(int argc __attribute__((unused)), char *argv[]) fprintf(stderr, "Incorrect LOGREC_VARIABLE_RECORD_1LSN_EXAMPLE " "in whole rec read lsn(%lu,0x%lx)\n", - (ulong) LSN_FILE_NO(rec.lsn), (ulong) LSN_OFFSET(rec.lsn)); + LSN_IN_PARTS(rec.lsn)); goto err; } } @@ -527,10 +523,8 @@ int main(int argc __attribute__((unused)), char *argv[]) "lsn(%lu,0x%lx)\n", i, (uint) rec.type, (uint) rec.short_trid, (ulong) rec.record_length, (ulong) rec_len, - len, - (ulong) LSN_FILE_NO(ref1), (ulong) LSN_OFFSET(ref1), - (ulong) LSN_FILE_NO(ref2), (ulong) LSN_OFFSET(ref2), - (ulong) LSN_FILE_NO(rec.lsn), (ulong) LSN_OFFSET(rec.lsn)); + len, LSN_IN_PARTS(ref1), LSN_IN_PARTS(ref2), + LSN_IN_PARTS(rec.lsn)); goto err; } if (read_and_check_content(&rec, long_buffer, LSN_STORE_SIZE * 2)) @@ -538,7 +532,7 @@ int main(int argc __attribute__((unused)), char *argv[]) fprintf(stderr, "Incorrect LOGREC_VARIABLE_RECORD_2LSN_EXAMPLE " "in whole rec read lsn(%lu,0x%lx)\n", - (ulong) LSN_FILE_NO(rec.lsn), (ulong) LSN_OFFSET(rec.lsn)); + LSN_IN_PARTS(rec.lsn)); goto err; } } @@ -571,7 +565,7 @@ int main(int argc __attribute__((unused)), char *argv[]) (uint) rec.record_length, (uint) uint4korr(rec.header), (uint) rec.header[4], (uint) rec.header[5], - (ulong) LSN_FILE_NO(rec.lsn), (ulong) LSN_OFFSET(rec.lsn)); + LSN_IN_PARTS(rec.lsn)); goto err; } lsn= rec.lsn; @@ -592,8 +586,7 @@ int main(int argc __attribute__((unused)), char *argv[]) "lsn(%lu,0x%lx)\n", i, (uint) rec.type, (uint) rec.short_trid, (ulong) rec.record_length, (ulong) rec_len, - len, - (ulong) LSN_FILE_NO(rec.lsn), (ulong) LSN_OFFSET(rec.lsn)); + len, LSN_IN_PARTS(rec.lsn)); goto err; } if (read_and_check_content(&rec, long_buffer, 0)) @@ -601,7 +594,7 @@ int main(int argc __attribute__((unused)), char *argv[]) fprintf(stderr, "Incorrect LOGREC_VARIABLE_RECORD_2LSN_EXAMPLE " "in whole rec read lsn(%lu,0x%lx)\n", - (ulong) LSN_FILE_NO(rec.lsn), (ulong) LSN_OFFSET(rec.lsn)); + LSN_IN_PARTS(rec.lsn)); goto err; } read_ok(&rec); diff --git a/storage/maria/unittest/ma_test_loghandler_first_lsn-t.c b/storage/maria/unittest/ma_test_loghandler_first_lsn-t.c index 17a41f1ad3e..28233ae04cb 100644 --- a/storage/maria/unittest/ma_test_loghandler_first_lsn-t.c +++ b/storage/maria/unittest/ma_test_loghandler_first_lsn-t.c @@ -92,8 +92,7 @@ int main(int argc __attribute__((unused)), char *argv[]) if (first_lsn != LSN_IMPOSSIBLE) { fprintf(stderr, "Incorrect first lsn response (%lu,0x%lx).", - (ulong) LSN_FILE_NO(first_lsn), - (ulong) LSN_OFFSET(first_lsn)); + LSN_IN_PARTS(first_lsn)); translog_destroy(); exit(1); } @@ -132,10 +131,7 @@ int main(int argc __attribute__((unused)), char *argv[]) { fprintf(stderr, "Incorrect first lsn: (%lu,0x%lx) " " theoretical first: (%lu,0x%lx)\n", - (ulong) LSN_FILE_NO(first_lsn), - (ulong) LSN_OFFSET(first_lsn), - (ulong) LSN_FILE_NO(theor_lsn), - (ulong) LSN_OFFSET(theor_lsn)); + LSN_IN_PARTS(first_lsn), LSN_IN_PARTS(theor_lsn)); translog_destroy(); exit(1); } diff --git a/storage/maria/unittest/ma_test_loghandler_max_lsn-t.c b/storage/maria/unittest/ma_test_loghandler_max_lsn-t.c index 08f838ebb65..d6f0bde7a8e 100644 --- a/storage/maria/unittest/ma_test_loghandler_max_lsn-t.c +++ b/storage/maria/unittest/ma_test_loghandler_max_lsn-t.c @@ -79,8 +79,7 @@ int main(int argc __attribute__((unused)), char *argv[]) if (max_lsn != LSN_IMPOSSIBLE) { fprintf(stderr, "Incorrect first lsn response (%lu,0x%lx).", - (ulong) LSN_FILE_NO(max_lsn), - (ulong) LSN_OFFSET(max_lsn)); + LSN_IN_PARTS(max_lsn)); translog_destroy(); exit(1); } @@ -125,10 +124,7 @@ int main(int argc __attribute__((unused)), char *argv[]) { fprintf(stderr, "Incorrect max lsn: (%lu,0x%lx) " " last lsn on first file: (%lu,0x%lx)\n", - (ulong) LSN_FILE_NO(max_lsn), - (ulong) LSN_OFFSET(max_lsn), - (ulong) LSN_FILE_NO(last_lsn), - (ulong) LSN_OFFSET(last_lsn)); + LSN_IN_PARTS(max_lsn), LSN_IN_PARTS(last_lsn)); translog_destroy(); exit(1); } diff --git a/storage/maria/unittest/ma_test_loghandler_multigroup-t.c b/storage/maria/unittest/ma_test_loghandler_multigroup-t.c index aa4b7b473cf..d5f00bdb6fd 100644 --- a/storage/maria/unittest/ma_test_loghandler_multigroup-t.c +++ b/storage/maria/unittest/ma_test_loghandler_multigroup-t.c @@ -371,7 +371,7 @@ int main(int argc __attribute__((unused)), char *argv[]) (uint) rec.type, (uint) rec.short_trid, (uint) rec.record_length, (uint)uint4korr(rec.header), (uint) rec.header[4], (uint) rec.header[5], - (ulong) LSN_FILE_NO(rec.lsn), (ulong) LSN_OFFSET(rec.lsn)); + LSN_IN_PARTS(rec.lsn)); translog_free_record_header(&rec); goto err; } @@ -418,8 +418,7 @@ int main(int argc __attribute__((unused)), char *argv[]) "type %u, strid %u, len %u, ref(%lu,0x%lx), lsn(%lu,0x%lx)\n", i, (uint) rec.type, (uint) rec.short_trid, (uint) rec.record_length, - (ulong) LSN_FILE_NO(ref), (ulong) LSN_OFFSET(ref), - (ulong) LSN_FILE_NO(rec.lsn), (ulong) LSN_OFFSET(rec.lsn)); + LSN_IN_PARTS(ref), LSN_IN_PARTS(rec.lsn)); translog_free_record_header(&rec); goto err; } @@ -451,14 +450,13 @@ int main(int argc __attribute__((unused)), char *argv[]) "lsn(%lu,0x%lx)\n", i, (uint) rec.type, (uint) rec.short_trid, (uint) rec.record_length, - (ulong) LSN_FILE_NO(ref1), (ulong) LSN_OFFSET(ref1), - (ulong) LSN_FILE_NO(ref2), (ulong) LSN_OFFSET(ref2), + LSN_IN_PARTS(ref1), LSN_IN_PARTS(ref2), (uint) rec.header[14], (uint) rec.header[15], (uint) rec.header[16], (uint) rec.header[17], (uint) rec.header[18], (uint) rec.header[19], (uint) rec.header[20], (uint) rec.header[21], (uint) rec.header[22], - (ulong) LSN_FILE_NO(rec.lsn), (ulong) LSN_OFFSET(rec.lsn)); + LSN_IN_PARTS(rec.lsn)); translog_free_record_header(&rec); goto err; } @@ -503,8 +501,7 @@ int main(int argc __attribute__((unused)), char *argv[]) rec.record_length != rec_len + LSN_STORE_SIZE, len, len != 12, - (ulong) LSN_FILE_NO(ref), (ulong) LSN_OFFSET(ref), - (ulong) LSN_FILE_NO(rec.lsn), (ulong) LSN_OFFSET(rec.lsn), + LSN_IN_PARTS(ref), LSN_IN_PARTS(rec.lsn), (ref != lsn), check_content(rec.header + LSN_STORE_SIZE, len - LSN_STORE_SIZE)); @@ -516,7 +513,7 @@ int main(int argc __attribute__((unused)), char *argv[]) fprintf(stderr, "Incorrect LOGREC_VARIABLE_RECORD_1LSN_EXAMPLE " "in whole rec read lsn(%lu,0x%lx)\n", - (ulong) LSN_FILE_NO(rec.lsn), (ulong) LSN_OFFSET(rec.lsn)); + LSN_IN_PARTS(rec.lsn)); translog_free_record_header(&rec); goto err; } @@ -544,9 +541,8 @@ int main(int argc __attribute__((unused)), char *argv[]) i, (uint) rec.type, (uint) rec.short_trid, (ulong) rec.record_length, (ulong) rec_len, len, - (ulong) LSN_FILE_NO(ref1), (ulong) LSN_OFFSET(ref1), - (ulong) LSN_FILE_NO(ref2), (ulong) LSN_OFFSET(ref2), - (ulong) LSN_FILE_NO(rec.lsn), (ulong) LSN_OFFSET(rec.lsn)); + LSN_IN_PARTS(ref1), LSN_IN_PARTS(ref2), + LSN_IN_PARTS(rec.lsn)); translog_free_record_header(&rec); goto err; } @@ -555,7 +551,7 @@ int main(int argc __attribute__((unused)), char *argv[]) fprintf(stderr, "Incorrect LOGREC_VARIABLE_RECORD_2LSN_EXAMPLE " "in whole rec read lsn(%lu,0x%lx)\n", - (ulong) LSN_FILE_NO(rec.lsn), (ulong) LSN_OFFSET(rec.lsn)); + LSN_IN_PARTS(rec.lsn)); translog_free_record_header(&rec); goto err; } @@ -591,7 +587,7 @@ int main(int argc __attribute__((unused)), char *argv[]) (uint) rec.record_length, (uint)uint4korr(rec.header), (uint) rec.header[4], (uint) rec.header[5], - (ulong) LSN_FILE_NO(rec.lsn), (ulong) LSN_OFFSET(rec.lsn)); + LSN_IN_PARTS(rec.lsn)); translog_free_record_header(&rec); goto err; } @@ -613,8 +609,7 @@ int main(int argc __attribute__((unused)), char *argv[]) "lsn(%lu,0x%lx)\n", i, (uint) rec.type, (uint) rec.short_trid, (ulong) rec.record_length, (ulong) rec_len, - len, - (ulong) LSN_FILE_NO(rec.lsn), (ulong) LSN_OFFSET(rec.lsn)); + len, LSN_IN_PARTS(rec.lsn)); translog_free_record_header(&rec); goto err; } @@ -623,7 +618,7 @@ int main(int argc __attribute__((unused)), char *argv[]) fprintf(stderr, "Incorrect LOGREC_VARIABLE_RECORD_2LSN_EXAMPLE " "in whole rec read lsn(%lu,0x%lx)\n", - (ulong) LSN_FILE_NO(rec.lsn), (ulong) LSN_OFFSET(rec.lsn)); + LSN_IN_PARTS(rec.lsn)); translog_free_record_header(&rec); goto err; } diff --git a/storage/maria/unittest/ma_test_loghandler_multithread-t.c b/storage/maria/unittest/ma_test_loghandler_multithread-t.c index d526dd933a1..6255c11db89 100644 --- a/storage/maria/unittest/ma_test_loghandler_multithread-t.c +++ b/storage/maria/unittest/ma_test_loghandler_multithread-t.c @@ -418,10 +418,8 @@ int main(int argc __attribute__((unused)), (uint) rec.short_trid, (uint) uint2korr(rec.header), (uint) rec.record_length, (uint) index, (uint) uint4korr(rec.header + 2), - (ulong) LSN_FILE_NO(rec.lsn), - (ulong) LSN_OFFSET(rec.lsn), - (ulong) LSN_FILE_NO(lsns1[rec.short_trid][index]), - (ulong) LSN_OFFSET(lsns1[rec.short_trid][index])); + LSN_IN_PARTS(rec.lsn), + LSN_IN_PARTS(lsns1[rec.short_trid][index])); translog_free_record_header(&rec); goto err; } @@ -446,10 +444,8 @@ int main(int argc __attribute__((unused)), len, (ulong) rec.record_length, lens[rec.short_trid][index], (rec.record_length != lens[rec.short_trid][index]), - (ulong) LSN_FILE_NO(rec.lsn), - (ulong) LSN_OFFSET(rec.lsn), - (ulong) LSN_FILE_NO(lsns2[rec.short_trid][index]), - (ulong) LSN_OFFSET(lsns2[rec.short_trid][index])); + LSN_IN_PARTS(rec.lsn), + LSN_IN_PARTS(lsns2[rec.short_trid][index])); translog_free_record_header(&rec); goto err; } @@ -458,8 +454,7 @@ int main(int argc __attribute__((unused)), fprintf(stderr, "Incorrect LOGREC_VARIABLE_RECORD_0LSN_EXAMPLE " "in whole rec read lsn(%lu,0x%lx)\n", - (ulong) LSN_FILE_NO(rec.lsn), - (ulong) LSN_OFFSET(rec.lsn)); + LSN_IN_PARTS(rec.lsn)); translog_free_record_header(&rec); goto err; } diff --git a/storage/maria/unittest/ma_test_loghandler_noflush-t.c b/storage/maria/unittest/ma_test_loghandler_noflush-t.c index 901bf588197..2c3afb9a76b 100644 --- a/storage/maria/unittest/ma_test_loghandler_noflush-t.c +++ b/storage/maria/unittest/ma_test_loghandler_noflush-t.c @@ -114,8 +114,7 @@ int main(int argc __attribute__((unused)), char *argv[]) (uint) uint4korr(rec.header), (uint4korr(rec.header) != 0), (uint) rec.header[4], (((uchar)rec.header[4]) != 0), (uint) rec.header[5], (((uchar)rec.header[5]) != 0xFF), - (ulong) LSN_FILE_NO(rec.lsn), (ulong) LSN_OFFSET(rec.lsn), - (first_lsn != rec.lsn)); + LSN_IN_PARTS(rec.lsn), (first_lsn != rec.lsn)); goto err; } -- cgit v1.2.1 From ce3bf146285022f5cdef375803d107a39ce77bf0 Mon Sep 17 00:00:00 2001 From: unknown Date: Thu, 13 Sep 2007 21:05:17 +0200 Subject: Bugfix: --pagecache* options were ignored; with --key-cache*=# --pagecache*=#, the latter wrote into the settings of the key cache! Problem was that --pagecache* options reused OPT_KEY_CACHE* enum values. sql/mysqld.cc: --pagecache* options should not reuse the OPT_KEY_CACHE* enum values, it confuses my_getopt. Making handling of page caches similar to handling of key caches, in mysqld.cc. --- sql/mysqld.cc | 42 +++++++++++++++++++++++++++--------------- 1 file changed, 27 insertions(+), 15 deletions(-) diff --git a/sql/mysqld.cc b/sql/mysqld.cc index fdfc7336658..2b47be2cddc 100644 --- a/sql/mysqld.cc +++ b/sql/mysqld.cc @@ -2802,16 +2802,6 @@ static int init_common_variables(const char *conf_file_name, int argc, get_options(&defaults_argc, defaults_argv); set_server_version(); -#ifdef WITH_MARIA_STORAGE_ENGINE - if (!(maria_pagecache= get_or_create_pagecache(maria_pagecache_base.str, - maria_pagecache_base.length))) - exit(1); -/* - maria_pagecache->param_buff_size= maria_pagecache_var.param_buff_size; - maria_pagecache->param_block_size= maria_block_size; -*/ -#endif - DBUG_PRINT("info",("%s Ver %s for %s on %s\n",my_progname, server_version, SYSTEM_TYPE,MACHINE_TYPE)); @@ -5063,6 +5053,8 @@ enum options_mysqld OPT_MARIA_MAX_SORT_FILE_SIZE, OPT_MARIA_SORT_BUFFER_SIZE, OPT_MARIA_USE_MMAP, OPT_MARIA_REPAIR_THREADS, OPT_MARIA_STATS_METHOD, + OPT_PAGECACHE_BUFFER_SIZE, + OPT_PAGECACHE_DIVISION_LIMIT, OPT_PAGECACHE_AGE_THRESHOLD, OPT_NET_BUFFER_LENGTH, OPT_NET_RETRY_COUNT, OPT_NET_READ_TIMEOUT, OPT_NET_WRITE_TIMEOUT, @@ -6234,17 +6226,17 @@ The minimum value for this variable is 4096.", (uchar**) &max_system_variables.optimizer_search_depth, 0, GET_ULONG, OPT_ARG, MAX_TABLES+1, 0, MAX_TABLES+2, 0, 1, 0}, #ifdef WITH_MARIA_STORAGE_ENGINE - {"pagecache_age_threshold", OPT_KEY_CACHE_AGE_THRESHOLD, + {"pagecache_age_threshold", OPT_PAGECACHE_AGE_THRESHOLD, "This characterizes the number of hits a hot block has to be untouched until it is considered aged enough to be downgraded to a warm block. This specifies the percentage ratio of that number of hits to the total number of blocks in key cache", (uchar**) &maria_pagecache_var.param_age_threshold, (uchar**) 0, 0, (GET_ULONG | GET_ASK_ADDR), REQUIRED_ARG, 300, 100, ~0L, 0, 100, 0}, - {"pagecache_buffer_size", OPT_KEY_BUFFER_SIZE, + {"pagecache_buffer_size", OPT_PAGECACHE_BUFFER_SIZE, "The size of the buffer used for index blocks for Maria tables. Increase this to get better index handling (for all reads and multiple writes) to as much as you can afford; 64M on a 256M machine that mainly runs MySQL is quite common.", (uchar**) &maria_pagecache_var.param_buff_size, (uchar**) 0, 0, (GET_ULL | GET_ASK_ADDR), REQUIRED_ARG, KEY_CACHE_SIZE, MALLOC_OVERHEAD, ~(ulong) 0, MALLOC_OVERHEAD, IO_SIZE, 0}, - {"pagecache_division_limit", OPT_KEY_CACHE_DIVISION_LIMIT, + {"pagecache_division_limit", OPT_PAGECACHE_DIVISION_LIMIT, "The minimum percentage of warm blocks in key cache", (uchar**) &maria_pagecache_var.param_division_limit, (uchar**) 0, @@ -7186,8 +7178,10 @@ static void mysql_init_variables(void) /* set key_cache_hash.default_value = dflt_key_cache */ multi_keycache_init(); - #ifdef WITH_MARIA_STORAGE_ENGINE + if (!(maria_pagecache= get_or_create_pagecache(maria_pagecache_base.str, + maria_pagecache_base.length))) + exit(1); /* set pagecache_hash.default_value = maria_pagecache */ multi_pagecache_init(); #endif @@ -7846,8 +7840,26 @@ mysql_getopt_value(const char *keyname, uint key_length, return (uchar**) &key_cache->param_age_threshold; } } +#ifdef WITH_MARIA_STORAGE_ENGINE + case OPT_PAGECACHE_BUFFER_SIZE: + case OPT_PAGECACHE_DIVISION_LIMIT: + case OPT_PAGECACHE_AGE_THRESHOLD: + { + PAGECACHE *pagecache; + if (!(pagecache= get_or_create_pagecache(keyname, key_length))) + exit(1); + switch (option->id) { + case OPT_PAGECACHE_BUFFER_SIZE: + return (uchar**) &pagecache->param_buff_size; + case OPT_PAGECACHE_DIVISION_LIMIT: + return (uchar**) &pagecache->param_division_limit; + case OPT_PAGECACHE_AGE_THRESHOLD: + return (uchar**) &pagecache->param_age_threshold; + } + } +#endif } - return option->value; + return option->value; } -- cgit v1.2.1 From f77e2969efb1a48fba04ec0b12e71a7fb3646afe Mon Sep 17 00:00:00 2001 From: unknown Date: Fri, 14 Sep 2007 09:17:36 +0300 Subject: Check of log initialization added. storage/maria/ma_loghandler.c: Check of log initialization added. Function descriptions fixed. --- storage/maria/ma_loghandler.c | 23 ++++++++++++++++++++--- 1 file changed, 20 insertions(+), 3 deletions(-) diff --git a/storage/maria/ma_loghandler.c b/storage/maria/ma_loghandler.c index 3470e1d408b..03d89b640c5 100644 --- a/storage/maria/ma_loghandler.c +++ b/storage/maria/ma_loghandler.c @@ -1028,6 +1028,7 @@ LSN translog_get_file_max_lsn_stored(uint32 file) uint32 limit= FILENO_IMPOSSIBLE; DBUG_ENTER("translog_get_file_max_lsn_stored"); DBUG_PRINT("enter", ("file: %lu", (ulong)file)); + DBUG_ASSERT(translog_inited == 1); pthread_mutex_lock(&log_descriptor.unfinished_files_lock); @@ -2629,6 +2630,7 @@ my_bool translog_init(const char *directory, TRANSLOG_ADDRESS sure_page, last_page, last_valid_page; my_bool version_changed= 0; DBUG_ENTER("translog_init"); + DBUG_ASSERT(translog_inited == 0); loghandler_init(); /* Safe to do many times */ @@ -4876,6 +4878,7 @@ my_bool translog_write_record(LSN *lsn, DBUG_ENTER("translog_write_record"); DBUG_PRINT("enter", ("type: %u ShortTrID: %u rec_len: %lu", (uint) type, (uint) short_trid, (ulong) rec_len)); + DBUG_ASSERT(translog_inited == 1); if (tbl_info) { @@ -5083,6 +5086,7 @@ static int translog_fixed_length_header(uchar *page, void translog_free_record_header(TRANSLOG_HEADER_BUFFER *buff) { DBUG_ENTER("translog_free_record_header"); + DBUG_ASSERT(translog_inited == 1); if (buff->groups_no != 0) { my_free((uchar*) buff->groups, MYF(0)); @@ -5101,6 +5105,7 @@ void translog_free_record_header(TRANSLOG_HEADER_BUFFER *buff) TRANSLOG_ADDRESS translog_get_horizon() { TRANSLOG_ADDRESS res; + DBUG_ASSERT(translog_inited == 1); translog_lock(); res= log_descriptor.horizon; translog_unlock(); @@ -5117,6 +5122,7 @@ TRANSLOG_ADDRESS translog_get_horizon() TRANSLOG_ADDRESS translog_get_horizon_no_lock() { + DBUG_ASSERT(translog_inited == 1); translog_lock_assert_owner(); return log_descriptor.horizon; } @@ -5166,6 +5172,7 @@ my_bool translog_init_scanner(LSN lsn, DBUG_ENTER("translog_init_scanner"); DBUG_PRINT("enter", ("LSN: (0x%lu,0x%lx)", LSN_IN_PARTS(lsn)); DBUG_ASSERT(LSN_OFFSET(lsn) % TRANSLOG_PAGE_SIZE != 0); + DBUG_ASSERT(translog_inited == 1); data.addr= &scanner->page_addr; data.was_recovered= 0; @@ -5374,9 +5381,10 @@ translog_get_next_chunk(TRANSLOG_SCANNER_DATA *scanner) stored decoded part of the header */ -int translog_variable_length_header(uchar *page, translog_size_t page_offset, - TRANSLOG_HEADER_BUFFER *buff, - TRANSLOG_SCANNER_DATA *scanner) +static int +translog_variable_length_header(uchar *page, translog_size_t page_offset, + TRANSLOG_HEADER_BUFFER *buff, + TRANSLOG_SCANNER_DATA *scanner) { struct st_log_record_type_descriptor *desc= (log_record_type_descriptor + buff->type); @@ -5556,6 +5564,7 @@ int translog_read_record_header_from_buffer(uchar *page, TRANSLOG_CHUNK_LSN || (page[page_offset] & TRANSLOG_CHUNK_TYPE) == TRANSLOG_CHUNK_FIXED); + DBUG_ASSERT(translog_inited == 1); buff->type= (page[page_offset] & TRANSLOG_REC_TYPE); buff->short_trid= uint2korr(page + page_offset + 1); DBUG_PRINT("info", ("Type %u, Short TrID %u, LSN (%lu,0x%lx)", @@ -5607,6 +5616,7 @@ int translog_read_record_header(LSN lsn, TRANSLOG_HEADER_BUFFER *buff) DBUG_ENTER("translog_read_record_header"); DBUG_PRINT("enter", ("LSN: (0x%lu,0x%lx)", LSN_IN_PARTS(lsn))); DBUG_ASSERT(LSN_OFFSET(lsn) % TRANSLOG_PAGE_SIZE != 0); + DBUG_ASSERT(translog_inited == 1); buff->lsn= lsn; buff->groups_no= 0; @@ -5653,6 +5663,7 @@ int translog_read_record_header_scan(TRANSLOG_SCANNER_DATA *scanner, LSN_IN_PARTS(scanner->last_file_page), (uint) scanner->page_offset, (uint) scanner->page_offset, scanner->fixed_horizon)); + DBUG_ASSERT(translog_inited == 1); buff->groups_no= 0; buff->lsn= scanner->page_addr; buff->lsn+= scanner->page_offset; /* offset increasing */ @@ -5699,6 +5710,7 @@ int translog_read_next_record_header(TRANSLOG_SCANNER_DATA *scanner, LSN_IN_PARTS(scanner->last_file_page), (uint) scanner->page_offset, (uint) scanner->page_offset, scanner->fixed_horizon)); + DBUG_ASSERT(translog_inited == 1); do { @@ -5875,6 +5887,7 @@ translog_size_t translog_read_record(LSN lsn, translog_size_t end= offset + length; struct st_translog_reader_data internal_data; DBUG_ENTER("translog_read_record"); + DBUG_ASSERT(translog_inited == 1); if (data == NULL) { @@ -6106,6 +6119,7 @@ my_bool translog_flush(LSN lsn) my_bool full_circle= 0; DBUG_ENTER("translog_flush"); DBUG_PRINT("enter", ("Flush up to LSN: (%lu,0x%lx)", LSN_IN_PARTS(lsn))); + DBUG_ASSERT(translog_inited == 1); translog_lock(); old_flushed= log_descriptor.flushed; @@ -6599,6 +6613,7 @@ LSN translog_first_lsn_in_log() TRANSLOG_SCANNER_DATA scanner; DBUG_ENTER("translog_first_lsn_in_log"); DBUG_PRINT("info", ("Horizon: (%lu,0x%lx)", LSN_IN_PARTS(addr))); + DBUG_ASSERT(translog_inited == 1); if (!(file= translog_first_file(horizon, 0))) { @@ -6650,6 +6665,7 @@ LSN translog_first_theoretical_lsn() TRANSLOG_VALIDATOR_DATA data; DBUG_ENTER("translog_first_theoretical_lsn"); DBUG_PRINT("info", ("Horizon: (%lu,0x%lx)", LSN_IN_PARTS(addr))); + DBUG_ASSERT(translog_inited == 1); if (!translog_is_file(1)) DBUG_RETURN(LSN_IMPOSSIBLE); @@ -6686,6 +6702,7 @@ my_bool translog_purge(TRANSLOG_ADDRESS low) int rc= 0; DBUG_ENTER("translog_purge"); DBUG_PRINT("enter", ("low: (%lu,0x%lx)", LSN_IN_PARTS(low))); + DBUG_ASSERT(translog_inited == 1); pthread_mutex_lock(&log_descriptor.purger_lock); if (LSN_FILE_NO(log_descriptor.last_lsn_checked) < last_need_file) -- cgit v1.2.1 From 19b75b6c73e99a8436263e209b7cbc790bbed22d Mon Sep 17 00:00:00 2001 From: unknown Date: Fri, 14 Sep 2007 15:01:44 +0300 Subject: Fixes problem with getting not LSN address gotten from horizon addres. storage/maria/ma_loghandler.c: New function to get correct LSN from chunk address. storage/maria/ma_loghandler.h: New function to get correct LSN from chunk address. --- storage/maria/ma_loghandler.c | 74 +++++++++++++++++++++++++++++-------------- storage/maria/ma_loghandler.h | 1 + storage/maria/ma_recovery.c | 12 +++++-- 3 files changed, 61 insertions(+), 26 deletions(-) diff --git a/storage/maria/ma_loghandler.c b/storage/maria/ma_loghandler.c index 03d89b640c5..e5b68056673 100644 --- a/storage/maria/ma_loghandler.c +++ b/storage/maria/ma_loghandler.c @@ -5679,7 +5679,7 @@ int translog_read_record_header_scan(TRANSLOG_SCANNER_DATA *scanner, /** @brief Read record header and some fixed part of the next record (the part depend on record type). - + @param scanner data for scanning if lsn is NULL scanner data will be used for continue scanning. The scanner can be NULL. @@ -6594,6 +6594,49 @@ static uint32 translog_first_file(TRANSLOG_ADDRESS horizon, int is_protected) } +/** + @brief returns the most close LSN higher the given chunk address + + @param addr the chunk address to start from + @param horizon the horizon if it is known or LSN_IMPOSSIBLE + + @retval LSN_ERROR Error + @retval LSN_IMPOSSIBLE no LSNs after the address + @retval # LSN of the most close LSN higher the given chunk address +*/ + +LSN translog_next_LSN(TRANSLOG_ADDRESS addr, TRANSLOG_ADDRESS horizon) +{ + uint chunk_type; + TRANSLOG_SCANNER_DATA scanner; + DBUG_ENTER("translog_next_LSN"); + + if (horizon == LSN_IMPOSSIBLE) + horizon= translog_get_horizon(); + + if (addr == horizon) + DBUG_RETURN(LSN_IMPOSSIBLE); + + translog_init_scanner(addr, 0, &scanner); + + chunk_type= scanner.page[scanner.page_offset] & TRANSLOG_CHUNK_TYPE; + DBUG_PRINT("info", ("type: %x byte: %x", (uint) chunk_type, + (uint) scanner.page[scanner.page_offset])); + while (chunk_type != TRANSLOG_CHUNK_LSN && + chunk_type != TRANSLOG_CHUNK_FIXED && + scanner.page[scanner.page_offset] != 0) + { + if (translog_get_next_chunk(&scanner)) + DBUG_RETURN(LSN_ERROR); + chunk_type= scanner.page[scanner.page_offset] & TRANSLOG_CHUNK_TYPE; + DBUG_PRINT("info", ("type: %x byte: %x", (uint) chunk_type, + (uint) scanner.page[scanner.page_offset])); + } + if (scanner.page[scanner.page_offset] == 0) + DBUG_RETURN(LSN_IMPOSSIBLE); /* reached page filler */ + DBUG_RETURN(scanner.page_addr + scanner.page_offset); +} + /** @brief returns the LSN of the first record starting in this log @@ -6607,10 +6650,8 @@ LSN translog_first_lsn_in_log() TRANSLOG_ADDRESS addr, horizon= translog_get_horizon(); TRANSLOG_VALIDATOR_DATA data; uint file; - uint chunk_type; uint16 chunk_offset; uchar *page; - TRANSLOG_SCANNER_DATA scanner; DBUG_ENTER("translog_first_lsn_in_log"); DBUG_PRINT("info", ("Horizon: (%lu,0x%lx)", LSN_IN_PARTS(addr))); DBUG_ASSERT(translog_inited == 1); @@ -6623,30 +6664,15 @@ LSN translog_first_lsn_in_log() addr= MAKE_LSN(file, TRANSLOG_PAGE_SIZE); /* the first page of the file */ data.addr= &addr; - if ((page= translog_get_page(&data, scanner.buffer)) == NULL || - (chunk_offset= translog_get_first_chunk_offset(page)) == 0) - DBUG_RETURN(LSN_ERROR); - addr+= chunk_offset; - if (addr == horizon) - DBUG_RETURN(LSN_IMPOSSIBLE); - translog_init_scanner(addr, 0, &scanner); - - chunk_type= scanner.page[scanner.page_offset] & TRANSLOG_CHUNK_TYPE; - DBUG_PRINT("info", ("type: %x byte: %x", (uint) chunk_type, - (uint) scanner.page[scanner.page_offset])); - while (chunk_type != TRANSLOG_CHUNK_LSN && - chunk_type != TRANSLOG_CHUNK_FIXED && - scanner.page[scanner.page_offset] != 0) { - if (translog_get_next_chunk(&scanner)) + uchar buffer[TRANSLOG_PAGE_SIZE]; + if ((page= translog_get_page(&data, buffer)) == NULL || + (chunk_offset= translog_get_first_chunk_offset(page)) == 0) DBUG_RETURN(LSN_ERROR); - chunk_type= scanner.page[scanner.page_offset] & TRANSLOG_CHUNK_TYPE; - DBUG_PRINT("info", ("type: %x byte: %x", (uint) chunk_type, - (uint) scanner.page[scanner.page_offset])); } - if (scanner.page[scanner.page_offset] == 0) - DBUG_RETURN(LSN_IMPOSSIBLE); /* reached page filler */ - DBUG_RETURN(scanner.page_addr + scanner.page_offset); + addr+= chunk_offset; + + DBUG_RETURN(translog_next_LSN(addr, horizon)); } diff --git a/storage/maria/ma_loghandler.h b/storage/maria/ma_loghandler.h index b7e3be18fb3..164ff013b10 100644 --- a/storage/maria/ma_loghandler.h +++ b/storage/maria/ma_loghandler.h @@ -274,6 +274,7 @@ extern my_bool translog_inited; extern LSN translog_first_lsn_in_log(); extern LSN translog_first_theoretical_lsn(); +extern LSN translog_next_LSN(TRANSLOG_ADDRESS addr, TRANSLOG_ADDRESS horizon); /* record parts descriptor */ struct st_translog_parts diff --git a/storage/maria/ma_recovery.c b/storage/maria/ma_recovery.c index b9bfdecf9f1..0f831ae63b1 100644 --- a/storage/maria/ma_recovery.c +++ b/storage/maria/ma_recovery.c @@ -212,6 +212,13 @@ int maria_apply_log(LSN from_lsn, my_bool apply, FILE *trace_file, from_lsn= parse_checkpoint_record(last_checkpoint_lsn); if (from_lsn == LSN_IMPOSSIBLE) goto err; + from_lsn= translog_next_LSN(from_lsn, LSN_IMPOSSIBLE); + if (from_lsn == LSN_ERROR) + goto err; + /* + from_lsn LSN_IMPOSSIBLE will be correctly processed + by run_redo_phase() + */ } } @@ -1260,9 +1267,10 @@ static int run_redo_phase(LSN lsn, my_bool apply) TRANSLOG_HEADER_BUFFER rec; - if (unlikely(lsn == translog_get_horizon())) + if (unlikely(lsn == LSN_IMPOSSIBLE || lsn == translog_get_horizon())) { - fprintf(tracef, "Cannot find a first record, empty log, nothing to do.\n"); + fprintf(tracef, "checkpoint address refers to the log end log or " + "log is empty, nothing to do.\n"); return 0; } -- cgit v1.2.1 From 9c2ff270fa725954d91f6f3d13b0aeb9b3960f47 Mon Sep 17 00:00:00 2001 From: unknown Date: Sat, 15 Sep 2007 14:45:26 +0200 Subject: WL#3072 Maria Recovery * recovery from ha_maria now skips replaying DDLs (too dangerous) * maria_read_log still replays DDLs, print warning about issues * fixes to replaying of REDO_RENAME * don't replay DDLs on corrupted tables (safer) * print a one-line message when really doing a recovery (applies to ha_maria, not maria_read_log) i.e. some REDOs or UNDOs are read. storage/maria/ma_checkpoint.c: fix for assertion failure storage/maria/ma_recovery.c: * Recovery from ha_maria now skips replaying DDLs (as the initial plan said) as this is unsafe in case of crashes during the DDL; applying the records may do harm (destroy important files) so we prefer to leave the "mess" of files untouched. A proper recovery of DDLs requires very careful thinking, probably testing separately the existence of the data and index file instead of using maria_open() which tests the existence of both, and maybe storing create_rename_lsn in the data file too. * maria_read_log still replays DDLs, we print a warning about dangers (due to ALTER TABLE not logging insertions into the tmp table; we will maybe need an option to have logging of those insertions). * fixes to replaying of REDO_RENAME (test create_rename_lsn of 'new_name' table if it exists; if that table exists and is more recent than the record, remove the 'old_name' table). * don't replay DDLs on corrupted tables (play safe) * fail also in non-debug builds if table is open when it should not be (when creating it for example, it should not be already open). * when the trace file is not stdout (i.e. when this is ha_maria), if really doing a recovery (reading REDOs or UNDOs), print a one-line message to stderr to inform about start and end of recovery (useful to know what mysqld is doing, especially if it takes long or crashes). storage/maria/ma_recovery.h: parameter to replay DDLs or not storage/maria/maria_read_log.c: replay DDLs in maria_read_log, to be able to recreate tables from scratch. --- storage/maria/ma_checkpoint.c | 8 +- storage/maria/ma_recovery.c | 240 +++++++++++++++++++++++++++++++++++++---- storage/maria/ma_recovery.h | 2 +- storage/maria/maria_read_log.c | 2 +- 4 files changed, 227 insertions(+), 25 deletions(-) diff --git a/storage/maria/ma_checkpoint.c b/storage/maria/ma_checkpoint.c index 8c3f2c0a2e2..aa291fe6c97 100644 --- a/storage/maria/ma_checkpoint.c +++ b/storage/maria/ma_checkpoint.c @@ -123,7 +123,13 @@ int ma_checkpoint_execute(CHECKPOINT_LEVEL level, my_bool no_wait) int result= 0; DBUG_ENTER("ma_checkpoint_execute"); - DBUG_ASSERT(checkpoint_inited); + if (!checkpoint_inited) + { + /* + If ha_maria failed to start, maria_panic_hton is called, we come here. + */ + DBUG_RETURN(0); + } DBUG_ASSERT(level > CHECKPOINT_NONE); /* look for already running checkpoints */ diff --git a/storage/maria/ma_recovery.c b/storage/maria/ma_recovery.c index 0f831ae63b1..2f951b0b776 100644 --- a/storage/maria/ma_recovery.c +++ b/storage/maria/ma_recovery.c @@ -49,6 +49,7 @@ static LSN current_group_end_lsn, checkpoint_start= LSN_IMPOSSIBLE; static TrID max_long_trid= 0; /**< max long trid seen by REDO phase */ static FILE *tracef; /**< trace file for debugging */ +static my_bool skip_DDLs; /**< if REDO phase should skip DDL records */ #define prototype_redo_exec_hook(R) \ static int exec_REDO_LOGREC_ ## R(const TRANSLOG_HEADER_BUFFER *rec) @@ -117,7 +118,23 @@ static void enlarge_buffer(const TRANSLOG_HEADER_BUFFER *rec) MYF(MY_WME | MY_ALLOW_ZERO_PTR)); } } - +static my_bool recovery_message_printed; +static inline void print_recovery_message() +{ + /* + If we're really doing a recovery (reading REDOs or UNDOs), we print a + one-line message when we start it and when we end it. It goes to stderr, + not tracef, so that it is visible in the error log (soon we should maybe + use sql_print_error). We don't print if if tracef is stdout as stdout will + be seen by the user and thus convey sufficient info already. + */ + if (!recovery_message_printed && (tracef != stdout)) + { + recovery_message_printed= TRUE; + /** @todo RECOVERY BUG all prints to stderr should go to error log */ + fprintf(stderr, "Maria engine: starting recovery\n"); + } +} #define ALERT_USER() DBUG_ASSERT(0) @@ -147,7 +164,7 @@ int maria_recover() { fprintf(trace_file, "TRACE of the last MARIA recovery from mysqld\n"); DBUG_ASSERT(maria_pagecache->inited); - res= maria_apply_log(LSN_IMPOSSIBLE, TRUE, trace_file, TRUE); + res= maria_apply_log(LSN_IMPOSSIBLE, TRUE, trace_file, TRUE, TRUE); if (!res) fprintf(trace_file, "SUCCESS\n"); fclose(trace_file); @@ -164,6 +181,8 @@ int maria_recover() LSN_IMPOSSIBLE means "use last checkpoint" @param apply if log records should be applied or not @param trace_file trace file where progress/debug messages will go + @param skip_DDLs Should DDL records (CREATE/RENAME/DROP/REPAIR) + be skipped by the REDO phase or not @todo This trace_file thing is primitive; soon we will make it similar to ma_check_print_warning() etc, and a successful recovery does not need to @@ -175,7 +194,7 @@ int maria_recover() */ int maria_apply_log(LSN from_lsn, my_bool apply, FILE *trace_file, - my_bool should_run_undo_phase) + my_bool should_run_undo_phase, my_bool skip_DDLs_arg) { int error= 0; uint unfinished_trans; @@ -192,7 +211,33 @@ int maria_apply_log(LSN from_lsn, my_bool apply, FILE *trace_file, if (!all_active_trans || !all_tables) goto err; + recovery_message_printed= FALSE; tracef= trace_file; + if (!(skip_DDLs= skip_DDLs_arg)) + { + /* + Example of what can go wrong when replaying DDLs: + CREATE TABLE t (logged); INSERT INTO t VALUES(1) (logged); + ALTER TABLE t ... which does + CREATE a temporary table #sql... (logged) + INSERT data from t into #sql... (not logged) + RENAME #sql TO t (logged) + Removing tables by hand and replaying the log will leave in the + end an empty table "t": missing records. If after the RENAME an INSERT + into t was done, that row had number 1 in its page, executing the + REDO_INSERT_ROW_HEAD on the recreated empty t will fail (assertion + failure in _ma_apply_redo_insert_row_head_or_tail(): new data page is + created whereas rownr is not 0). + Another issue is that replaying of DDLs is not correct enough to work if + there was a crash during a DDL (see comment in execution of + REDO_RENAME_TABLE ). + */ + fprintf(tracef, "WARNING: MySQL server currently disables log records" + " about insertion of data by ALTER TABLE" + " (copy_data_between_tables()), applying of log records may" + " well not work. Additionally, applying of DDL records will" + " cause damage if there are tables left by a crash of a DDL.\n"); + } if (from_lsn == LSN_IMPOSSIBLE) { @@ -245,6 +290,8 @@ int maria_apply_log(LSN from_lsn, my_bool apply, FILE *trace_file, goto err; /* If inside ha_maria, a checkpoint will soon be taken and save our work */ + if (recovery_message_printed && (tracef != stdout)) + fprintf(stderr, "Maria engine: finished recovery\n"); goto end; err: error= 1; @@ -365,6 +412,11 @@ prototype_redo_exec_hook(REDO_CREATE_TABLE) uint flags; int error= 1, create_mode= O_RDWR | O_TRUNC; MARIA_HA *info= NULL; + if (skip_DDLs) + { + fprintf(tracef, "we skip DDLs\n"); + return 0; + } enlarge_buffer(rec); if (log_record_buffer.str == NULL || translog_read_record(rec->lsn, 0, rec->record_length, @@ -382,7 +434,12 @@ prototype_redo_exec_hook(REDO_CREATE_TABLE) { MARIA_SHARE *share= info->s; /* check that we're not already using it */ - DBUG_ASSERT(share->reopen == 1); + if (share->reopen != 1) + { + fprintf(tracef, ", is already open (reopen=%u)\n", share->reopen); + ALERT_USER(); + goto end; + } DBUG_ASSERT(share->now_transactional == share->base.born_transactional); if (!share->base.born_transactional) { @@ -391,7 +448,7 @@ prototype_redo_exec_hook(REDO_CREATE_TABLE) one was renamed to its name, thus create_rename_lsn is 0 and should not be trusted. */ - fprintf(tracef, ", is not transactional\n"); + fprintf(tracef, ", is not transactional, ignoring creation\n"); ALERT_USER(); error= 0; goto end; @@ -406,13 +463,16 @@ prototype_redo_exec_hook(REDO_CREATE_TABLE) } if (maria_is_crashed(info)) { - fprintf(tracef, ", is crashed, overwriting it"); + fprintf(tracef, ", is crashed, can't recreate it"); ALERT_USER(); + goto end; } maria_close(info); info= NULL; } - /* if does not exist, is older, or its header is corrupted, overwrite it */ + else /* one or two files absent, or header corrupted... */ + fprintf(tracef, "can't be opened, probably does not exist"); + /* if does not exist, or is older, overwrite it */ /** @todo symlinks */ ptr= name + strlen(name) + 1; if ((flags= ptr[0] ? HA_DONT_TOUCH_DATA : 0)) @@ -490,6 +550,11 @@ prototype_redo_exec_hook(REDO_RENAME_TABLE) char *old_name, *new_name; int error= 1; MARIA_HA *info= NULL; + if (skip_DDLs) + { + fprintf(tracef, "we skip DDLs\n"); + return 0; + } enlarge_buffer(rec); if (log_record_buffer.str == NULL || translog_read_record(rec->lsn, 0, rec->record_length, @@ -501,7 +566,36 @@ prototype_redo_exec_hook(REDO_RENAME_TABLE) } old_name= log_record_buffer.str; new_name= old_name + strlen(old_name) + 1; - fprintf(tracef, "Table '%s' to rename to '%s'", old_name, new_name); + fprintf(tracef, "Table '%s' to rename to '%s'; old-name table ", old_name, + new_name); + /* + Here is why we skip CREATE/DROP/RENAME when doing a recovery from + ha_maria (whereas we do when called from maria_read_log). Consider: + CREATE TABLE t; + RENAME TABLE t to u; + DROP TABLE u; + RENAME TABLE v to u; # crash between index rename and data rename. + And do a Recovery (not removing tables beforehand). + Recovery replays CREATE, then RENAME: the maria_open("t") works, + maria_open("u") does not (no data file) so table "u" is considered + inexistent and so maria_rename() is done which overwrites u's index file, + which is lost. Ok, the data file (v.MAD) is still available, but only a + REPAIR USE_FRM can rebuild the index, which is unsafe and downtime. + So it is preferrable to not execute RENAME, and leave the "mess" of files, + rather than possibly destroy a file. DBA will manually rename files. + A safe recovery method would probably require checking the existence of + the index file and of the data file separately (not via maria_open()), and + maybe also to store a create_rename_lsn in the data file too + For now, all we risk is to leave the mess (half-renamed files) left by the + crash. We however sync files and directories at each file rename. The SQL + layer is anyway not crash-safe for DDLs (except the repartioning-related + ones). + We replay DDLs in maria_read_log to be able to recreate tables from + scratch. It means that "maria_read_log -a" should not be used on a + database which just crashed during a DDL. And also ALTER TABLE does not + log insertions of records into the temporary table, so replaying may + fail (see comment and warning in maria_apply_log()). + */ info= maria_open(old_name, O_RDONLY, HA_OPEN_FOR_REPAIR); if (info) { @@ -512,7 +606,7 @@ prototype_redo_exec_hook(REDO_RENAME_TABLE) */ if (!share->base.born_transactional) { - fprintf(tracef, ", is not transactional\n"); + fprintf(tracef, ", is not transactional, ignoring renaming\n"); ALERT_USER(); error= 0; goto end; @@ -540,7 +634,76 @@ prototype_redo_exec_hook(REDO_RENAME_TABLE) maria_close(info)) goto end; info= NULL; + fprintf(tracef, ", is ok for renaming; new-name table "); + } + else /* one or two files absent, or header corrupted... */ + { + fprintf(tracef, ", can't be opened, probably does not exist"); + error= 0; + goto end; } + /* + We must also check the create_rename_lsn of the 'new_name' table if it + exists: otherwise we may, with our rename which overwrites, destroy + another table. For example: + CREATE TABLE t; + RENAME t to u; + DROP TABLE u; + RENAME v to u; # v is an old table, its creation/insertions not in log + And start executing the log (without removing tables beforehand): creates + t, renames it to u (if not testing create_rename_lsn) thus overwriting + old-named v, drops u, and we are stuck, we have lost data. + */ + info= maria_open(new_name, O_RDONLY, HA_OPEN_FOR_REPAIR); + if (info) + { + MARIA_SHARE *share= info->s; + /* We should not have open instances on this table. */ + if (share->reopen != 1) + { + fprintf(tracef, ", is already open (reopen=%u)\n", share->reopen); + ALERT_USER(); + goto end; + } + if (!share->base.born_transactional) + { + fprintf(tracef, ", is not transactional, ignoring renaming\n"); + ALERT_USER(); + goto drop; + } + if (cmp_translog_addr(share->state.create_rename_lsn, rec->lsn) >= 0) + { + fprintf(tracef, ", has create_rename_lsn (%lu,0x%lx) more recent than" + " record, ignoring renaming", + LSN_IN_PARTS(share->state.create_rename_lsn)); + /* + We have to drop the old_name table. Consider: + CREATE TABLE t; + CREATE TABLE v; + RENAME TABLE t to u; + DROP TABLE u; + RENAME TABLE v to u; + and apply the log without removing tables beforehand. t will be + created, v too; in REDO_RENAME u will be more recent, but we still + have to drop t otherwise it stays. + */ + goto drop; + } + if (maria_is_crashed(info)) + { + fprintf(tracef, ", is crashed, can't rename it"); + ALERT_USER(); + goto end; + } + if (maria_close(info)) + goto end; + info= NULL; + /* abnormal situation */ + fprintf(tracef, ", exists but is older than record, can't rename it"); + goto end; + } + else /* one or two files absent, or header corrupted... */ + fprintf(tracef, ", can't be opened, probably does not exist"); fprintf(tracef, ", renaming '%s'", old_name); if (maria_rename(old_name, new_name)) { @@ -559,6 +722,16 @@ prototype_redo_exec_hook(REDO_RENAME_TABLE) goto end; info= NULL; error= 0; + goto end; +drop: + fprintf(tracef, ", only dropping '%s'", old_name); + if (maria_delete_table(old_name)) + { + fprintf(tracef, "Failed to drop table\n"); + goto end; + } + error= 0; + goto end; end: fprintf(tracef, "\n"); if (info != NULL) @@ -573,8 +746,17 @@ end: prototype_redo_exec_hook(REDO_REPAIR_TABLE) { int error= 1; - MARIA_HA *info= get_MARIA_HA_from_REDO_record(rec); - if (info == NULL) + MARIA_HA *info; + if (skip_DDLs) + { + /* + REPAIR is not exactly a DDL, but it manipulates files without logging + insertions into them. + */ + fprintf(tracef, "we skip DDLs\n"); + return 0; + } + if ((info= get_MARIA_HA_from_REDO_record(rec)) == NULL) return 0; /* Otherwise, the mapping is newer than the table, and our record is newer @@ -610,6 +792,11 @@ prototype_redo_exec_hook(REDO_DROP_TABLE) char *name; int error= 1; MARIA_HA *info= NULL; + if (skip_DDLs) + { + fprintf(tracef, "we skip DDLs\n"); + return 0; + } enlarge_buffer(rec); if (log_record_buffer.str == NULL || translog_read_record(rec->lsn, 0, rec->record_length, @@ -631,7 +818,7 @@ prototype_redo_exec_hook(REDO_DROP_TABLE) */ if (!share->base.born_transactional) { - fprintf(tracef, ", is not transactional\n"); + fprintf(tracef, ", is not transactional, ignoring removal\n"); ALERT_USER(); error= 0; goto end; @@ -646,8 +833,9 @@ prototype_redo_exec_hook(REDO_DROP_TABLE) } if (maria_is_crashed(info)) { - fprintf(tracef, ", is crashed, dropping it"); + fprintf(tracef, ", is crashed, can't drop it"); ALERT_USER(); + goto end; } /* This maria_extra() call serves to signal that old open instances of @@ -658,14 +846,16 @@ prototype_redo_exec_hook(REDO_DROP_TABLE) maria_close(info)) goto end; info= NULL; + /* if it is older, or its header is corrupted, drop it */ + fprintf(tracef, ", dropping '%s'", name); + if (maria_delete_table(name)) + { + fprintf(tracef, "Failed to drop table\n"); + goto end; + } } - /* if does not exist, is older, or its header is corrupted, drop it */ - fprintf(tracef, ", dropping '%s'", name); - if (maria_delete_table(name)) - { - fprintf(tracef, "Failed to drop table\n"); - goto end; - } + else /* one or two files absent, or header corrupted... */ + fprintf(tracef,", can't be opened, probably does not exist"); error= 0; end: fprintf(tracef, "\n"); @@ -753,7 +943,12 @@ static int new_table(uint16 sid, const char *name, } MARIA_SHARE *share= info->s; /* check that we're not already using it */ - DBUG_ASSERT(share->reopen == 1); + if (share->reopen != 1) + { + fprintf(tracef, ", is already open (reopen=%u)\n", share->reopen); + ALERT_USER(); + goto end; + } DBUG_ASSERT(share->now_transactional == share->base.born_transactional); if (!share->base.born_transactional) { @@ -1294,7 +1489,6 @@ static int run_redo_phase(LSN lsn, my_bool apply) uint16 sid= rec.short_trid; const LOG_DESC *log_desc= &log_record_type_descriptor[rec.type]; display_record_position(log_desc, &rec, i); - /* A complete group is a set of log records with an "end mark" record (e.g. a set of REDOs for an operation, terminated by an UNDO for this @@ -1572,6 +1766,7 @@ static MARIA_HA *get_MARIA_HA_from_REDO_record(const MARIA_HA *info; char llbuf[22]; + print_recovery_message(); sid= fileid_korr(rec->header); page= page_korr(rec->header + FILEID_STORE_SIZE); /** @@ -1643,6 +1838,7 @@ static MARIA_HA *get_MARIA_HA_from_UNDO_record(const uint16 sid; MARIA_HA *info; + print_recovery_message(); sid= fileid_korr(rec->header + LSN_STORE_SIZE); fprintf(tracef, " For table of short id %u", sid); info= all_tables[sid].info; diff --git a/storage/maria/ma_recovery.h b/storage/maria/ma_recovery.h index 9a5a2b3099e..6a4b359be4d 100644 --- a/storage/maria/ma_recovery.h +++ b/storage/maria/ma_recovery.h @@ -26,5 +26,5 @@ C_MODE_START int maria_recover(); int maria_apply_log(LSN lsn, my_bool apply, FILE *trace_file, - my_bool execute_undo_phase); + my_bool execute_undo_phase, my_bool skip_DDLs); C_MODE_END diff --git a/storage/maria/maria_read_log.c b/storage/maria/maria_read_log.c index dc537695739..a7a6370b1c4 100644 --- a/storage/maria/maria_read_log.c +++ b/storage/maria/maria_read_log.c @@ -101,7 +101,7 @@ int main(int argc, char **argv) fprintf(stdout, "TRACE of the last maria_read_log\n"); if (maria_apply_log(lsn, opt_display_and_apply, stdout, - opt_display_and_apply)) + opt_display_and_apply, FALSE)) goto err; fprintf(stdout, "%s: SUCCESS\n", my_progname); -- cgit v1.2.1 From be382b4220656eca2642943abca1f8ffcb1faaa2 Mon Sep 17 00:00:00 2001 From: unknown Date: Thu, 20 Sep 2007 10:31:25 +0200 Subject: Fix for segmentation fault when updating a record having a small BLOB whose size didn't change. Fix for probably impossible problem in Recovery. mysql-test/r/maria.result: result for new test mysql-test/t/maria.test: testcase for a bug (used to segfault) storage/maria/ma_blockrec.c: When writing a record, we put BLOBs into the head part if there is room for them. "Is there room" was first decided by !(tmp_data + length > end_of_data) (line 1894) but then was tested again as *blob_lengths < (ulong)(end_of_data - data). We see that in case of equality, the first condition was true but the second was not, so it was inconsistent and crashed later. storage/maria/ma_recovery.c: When wondering if recovery should update the state (like state.records): if table was closed, its is_of_horizon was set to X, then table was reopened and a REDO was written. If this REDO had LSN X (as horizon is just a lower bound of the LSN of the next record), we have to apply it. In practice this equality probably could not happen because of LOGREC_FILE_ID would be written before the REDO. --- mysql-test/r/maria.result | 7 +++++++ mysql-test/t/maria.test | 8 ++++++++ storage/maria/ma_blockrec.c | 3 ++- storage/maria/ma_recovery.c | 8 ++++---- 4 files changed, 21 insertions(+), 5 deletions(-) diff --git a/mysql-test/r/maria.result b/mysql-test/r/maria.result index 710cb5ec094..8c1cb5f3a2f 100644 --- a/mysql-test/r/maria.result +++ b/mysql-test/r/maria.result @@ -1966,3 +1966,10 @@ select t1,t2,length(t3),length(t4),length(t5),length(t6),t7,t8 from t2; t1 t2 length(t3) length(t4) length(t5) length(t6) t7 t8 1 a 256 256 4096 4096 drop table t1,t2; +CREATE TABLE t1 (seq int, s1 int, s2 blob); +insert into t1 values (1, 1, MD5(1)); +update t1 set s1=2 where seq=1; +check table t1 extended; +Table Op Msg_type Msg_text +test.t1 check status OK +drop table t1; diff --git a/mysql-test/t/maria.test b/mysql-test/t/maria.test index d45f4c2ec6b..a98e2a8acf4 100644 --- a/mysql-test/t/maria.test +++ b/mysql-test/t/maria.test @@ -1254,6 +1254,14 @@ check table t1,t2; select t1,t2,length(t3),length(t4),length(t5),length(t6),t7,t8 from t2; drop table t1,t2; +# Test UPDATE with small BLOB which fits on head page + +CREATE TABLE t1 (seq int, s1 int, s2 blob); +insert into t1 values (1, 1, MD5(1)); +update t1 set s1=2 where seq=1; +check table t1 extended; +drop table t1; + # End of 5.2 tests --disable_result_log diff --git a/storage/maria/ma_blockrec.c b/storage/maria/ma_blockrec.c index 2558450b663..96e8663bfc8 100644 --- a/storage/maria/ma_blockrec.c +++ b/storage/maria/ma_blockrec.c @@ -1909,7 +1909,8 @@ static my_bool write_block_record(MARIA_HA *info, { /* Still room on page; Copy as many blobs we can into this page */ data= tmp_data; - for (; column < end_column && *blob_lengths < (ulong) (end_of_data - data); + for (; column < end_column && + *blob_lengths <= (ulong)(end_of_data - data); column++, blob_lengths++) { uchar *tmp_pos; diff --git a/storage/maria/ma_recovery.c b/storage/maria/ma_recovery.c index 2f951b0b776..5989adfd0d9 100644 --- a/storage/maria/ma_recovery.c +++ b/storage/maria/ma_recovery.c @@ -1193,7 +1193,7 @@ prototype_redo_exec_hook(UNDO_ROW_INSERT) if (info == NULL) return 0; set_undo_lsn_for_active_trans(rec->short_trid, rec->lsn); - if (cmp_translog_addr(rec->lsn, info->s->state.is_of_horizon) > 0) + if (cmp_translog_addr(rec->lsn, info->s->state.is_of_horizon) >= 0) { fprintf(tracef, " state older than record, updating rows' count\n"); info->s->state.state.records++; @@ -1216,7 +1216,7 @@ prototype_redo_exec_hook(UNDO_ROW_DELETE) if (info == NULL) return 0; set_undo_lsn_for_active_trans(rec->short_trid, rec->lsn); - if (cmp_translog_addr(rec->lsn, info->s->state.is_of_horizon) > 0) + if (cmp_translog_addr(rec->lsn, info->s->state.is_of_horizon) >= 0) { fprintf(tracef, " state older than record, updating rows' count\n"); info->s->state.state.records--; @@ -1234,7 +1234,7 @@ prototype_redo_exec_hook(UNDO_ROW_UPDATE) if (info == NULL) return 0; set_undo_lsn_for_active_trans(rec->short_trid, rec->lsn); - if (cmp_translog_addr(rec->lsn, info->s->state.is_of_horizon) > 0) + if (cmp_translog_addr(rec->lsn, info->s->state.is_of_horizon) >= 0) { info->s->state.changed|= STATE_CHANGED | STATE_NOT_ANALYZED | STATE_NOT_OPTIMIZED_KEYS | STATE_NOT_SORTED_PAGES; @@ -1295,7 +1295,7 @@ prototype_redo_exec_hook(CLR_END) set_undo_lsn_for_active_trans(rec->short_trid, previous_undo_lsn); fprintf(tracef, " CLR_END was about %s, undo_lsn now LSN (%lu,0x%lx)\n", log_desc->name, LSN_IN_PARTS(previous_undo_lsn)); - if (cmp_translog_addr(rec->lsn, info->s->state.is_of_horizon) > 0) + if (cmp_translog_addr(rec->lsn, info->s->state.is_of_horizon) >= 0) { fprintf(tracef, " state older than record, updating rows' count\n"); switch (undone_record_type) { -- cgit v1.2.1 From 95420b947e2e11050c480c8bda67e29b14f413b1 Mon Sep 17 00:00:00 2001 From: unknown Date: Thu, 20 Sep 2007 16:11:46 +0200 Subject: fix for non-debug compilation errors. Note that non-debug build fails in log handler functions, mail sent. storage/maria/ma_blockrec.c: fix for compiler warning storage/maria/ma_checkpoint.c: Debug build does not catch this situation static int f(); ... f(2); ... static int f(int a, int b); Maybe this is because it believes the declaration is K&R. Non-debug build catches it. Adding (void) as an habit to avoid such errors. storage/maria/ma_checkpoint.h: adding (void) storage/maria/ma_recovery.c: adding (void) storage/maria/ma_recovery.h: adding (void) --- storage/maria/ma_blockrec.c | 2 ++ storage/maria/ma_checkpoint.c | 11 ++++++----- storage/maria/ma_checkpoint.h | 2 +- storage/maria/ma_recovery.c | 8 ++++---- storage/maria/ma_recovery.h | 2 +- 5 files changed, 14 insertions(+), 11 deletions(-) diff --git a/storage/maria/ma_blockrec.c b/storage/maria/ma_blockrec.c index 96e8663bfc8..b12035c9cfa 100644 --- a/storage/maria/ma_blockrec.c +++ b/storage/maria/ma_blockrec.c @@ -304,7 +304,9 @@ static my_bool delete_tails(MARIA_HA *info, MARIA_RECORD_POS *tails); static my_bool delete_head_or_tail(MARIA_HA *info, ulonglong page, uint record_number, my_bool head, my_bool from_update); +#ifndef DBUG_OFF static void _ma_print_directory(uchar *buff, uint block_size); +#endif static void compact_page(uchar *buff, uint block_size, uint rownr, my_bool extend_block); static uchar *store_page_range(uchar *to, MARIA_BITMAP_BLOCK *block, diff --git a/storage/maria/ma_checkpoint.c b/storage/maria/ma_checkpoint.c index aa291fe6c97..0e8b558e7c5 100644 --- a/storage/maria/ma_checkpoint.c +++ b/storage/maria/ma_checkpoint.c @@ -99,9 +99,9 @@ static int filter_flush_data_file_indirect(enum pagecache_page_type type, static int filter_flush_data_file_evenly(enum pagecache_page_type type, pgcache_page_no_t pageno, LSN rec_lsn, void *arg); -static int really_execute_checkpoint(); +static int really_execute_checkpoint(void); pthread_handler_t ma_checkpoint_background(void *arg); -static int collect_tables(); +static int collect_tables(LEX_STRING *str, LSN checkpoint_start_log_horizon); /** @brief Does a checkpoint @@ -171,7 +171,7 @@ end: @retval !=0 error */ -static int really_execute_checkpoint() +static int really_execute_checkpoint(void) { uint i, error= 0; /** @brief checkpoint_start_log_horizon will be stored there */ @@ -223,7 +223,8 @@ static int really_execute_checkpoint() /* STEP 3: fetch information about table files */ - if (unlikely(collect_tables(&record_pieces[2]))) + if (unlikely(collect_tables(&record_pieces[2], + checkpoint_start_log_horizon))) goto err; @@ -366,7 +367,7 @@ int ma_checkpoint_init(my_bool create_background_thread) @brief Destroys the checkpoint module */ -void ma_checkpoint_end() +void ma_checkpoint_end(void) { DBUG_ENTER("ma_checkpoint_end"); if (checkpoint_inited) diff --git a/storage/maria/ma_checkpoint.h b/storage/maria/ma_checkpoint.h index 60bbff0b295..86f3779ca7a 100644 --- a/storage/maria/ma_checkpoint.h +++ b/storage/maria/ma_checkpoint.h @@ -33,7 +33,7 @@ typedef enum enum_ma_checkpoint_level { C_MODE_START int ma_checkpoint_init(my_bool create_background_thread); -void ma_checkpoint_end(); +void ma_checkpoint_end(void); int ma_checkpoint_execute(CHECKPOINT_LEVEL level, my_bool no_wait); C_MODE_END diff --git a/storage/maria/ma_recovery.c b/storage/maria/ma_recovery.c index 5989adfd0d9..23e5ffe2a29 100644 --- a/storage/maria/ma_recovery.c +++ b/storage/maria/ma_recovery.c @@ -104,7 +104,7 @@ static int new_table(uint16 sid, const char *name, LSN lsn_of_file_id); static int new_page(File fileid, pgcache_page_no_t pageid, LSN rec_lsn, struct st_dirty_page *dirty_page); -static int close_all_tables(); +static int close_all_tables(void); /** @brief global [out] buffer for translog_read_record(); never shrinks */ static LEX_STRING log_record_buffer; @@ -119,7 +119,7 @@ static void enlarge_buffer(const TRANSLOG_HEADER_BUFFER *rec) } } static my_bool recovery_message_printed; -static inline void print_recovery_message() +static inline void print_recovery_message(void) { /* If we're really doing a recovery (reading REDOs or UNDOs), we print a @@ -151,7 +151,7 @@ static inline void print_recovery_message() @retval !=0 Error */ -int maria_recover() +int maria_recover(void) { int res= 1; FILE *trace_file; @@ -2012,7 +2012,7 @@ static int new_page(File fileid, pgcache_page_no_t pageid, LSN rec_lsn, } -static int close_all_tables() +static int close_all_tables(void) { int error= 0; LIST *list_element, *next_open; diff --git a/storage/maria/ma_recovery.h b/storage/maria/ma_recovery.h index 6a4b359be4d..e3864d6022b 100644 --- a/storage/maria/ma_recovery.h +++ b/storage/maria/ma_recovery.h @@ -24,7 +24,7 @@ /* Performs recovery of the engine at start */ C_MODE_START -int maria_recover(); +int maria_recover(void); int maria_apply_log(LSN lsn, my_bool apply, FILE *trace_file, my_bool execute_undo_phase, my_bool skip_DDLs); C_MODE_END -- cgit v1.2.1 From 59259dd1f2adda72da253576334967b0d0fedc46 Mon Sep 17 00:00:00 2001 From: unknown Date: Thu, 20 Sep 2007 18:02:36 +0200 Subject: In non-debug builds, the log handler failed to read any log record (for example in ma_test_loghandler-t). Reason was wrongly matched () in DBUG. storage/maria/ma_loghandler.c: Wrongly matched parenthesis: DBUG_PRINT(keyword, argslist) expands to roughly _db_doprnt arglist; So DBUG_PRINT("enter",(a); b; c); expands to roughly _db_doprnt(a);b;c; which is valid code. Except that in non-debug builds, DBUG_PRINT( expands to nothing so the wrongly "included" code is thrown away, leading to some members of "scanner" to not be initialized. --- storage/maria/ma_loghandler.c | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/storage/maria/ma_loghandler.c b/storage/maria/ma_loghandler.c index e5b68056673..308176740ac 100644 --- a/storage/maria/ma_loghandler.c +++ b/storage/maria/ma_loghandler.c @@ -5170,7 +5170,7 @@ my_bool translog_init_scanner(LSN lsn, { TRANSLOG_VALIDATOR_DATA data; DBUG_ENTER("translog_init_scanner"); - DBUG_PRINT("enter", ("LSN: (0x%lu,0x%lx)", LSN_IN_PARTS(lsn)); + DBUG_PRINT("enter", ("LSN: (0x%lu,0x%lx)", LSN_IN_PARTS(lsn))); DBUG_ASSERT(LSN_OFFSET(lsn) % TRANSLOG_PAGE_SIZE != 0); DBUG_ASSERT(translog_inited == 1); @@ -5186,7 +5186,7 @@ my_bool translog_init_scanner(LSN lsn, LSN_IN_PARTS(scanner->horizon))); /* lsn < horizon */ - DBUG_ASSERT(lsn < scanner->horizon)); + DBUG_ASSERT(lsn < scanner->horizon); scanner->page_addr= lsn; scanner->page_addr-= scanner->page_offset; /*decrease offset */ -- cgit v1.2.1 From e88aa0c84e69f126ffa9bffb954629288dd3ea62 Mon Sep 17 00:00:00 2001 From: unknown Date: Fri, 21 Sep 2007 13:48:57 +0300 Subject: Transaction log flush serialization. --- storage/maria/ma_loghandler.c | 27 +++++++++++++++++++++------ 1 file changed, 21 insertions(+), 6 deletions(-) diff --git a/storage/maria/ma_loghandler.c b/storage/maria/ma_loghandler.c index e5b68056673..766265c225f 100644 --- a/storage/maria/ma_loghandler.c +++ b/storage/maria/ma_loghandler.c @@ -162,6 +162,7 @@ struct st_translog_descriptor /* All what is after this address is not sent to disk yet */ TRANSLOG_ADDRESS in_buffers_only; pthread_mutex_t sent_to_file_lock; + pthread_mutex_t log_flush_lock; /* Protects changing of headers of finished files (max_lsn) */ pthread_mutex_t file_header_lock; @@ -2642,6 +2643,8 @@ my_bool translog_init(const char *directory, MY_MUTEX_INIT_FAST) || pthread_mutex_init(&log_descriptor.purger_lock, MY_MUTEX_INIT_FAST) || + pthread_mutex_init(&log_descriptor.log_flush_lock, + MY_MUTEX_INIT_FAST) || init_dynamic_array(&log_descriptor.unfinished_files, sizeof(struct st_file_counter), 10, 10 CALLER_INFO)) @@ -3023,6 +3026,7 @@ void translog_destroy() pthread_mutex_destroy(&log_descriptor.file_header_lock); pthread_mutex_destroy(&log_descriptor.unfinished_files_lock); pthread_mutex_destroy(&log_descriptor.purger_lock); + pthread_mutex_destroy(&log_descriptor.log_flush_lock); delete_dynamic(&log_descriptor.unfinished_files); my_close(log_descriptor.directory_fd, MYF(MY_WME)); @@ -6070,6 +6074,10 @@ static void translog_force_current_buffer_to_finish() if (left) { + /* + TODO: do not copy begining of the page if we have no CRC or sector + checks on + */ memcpy(new_buffer->buffer, data, current_page_fill); log_descriptor.bc.ptr+= current_page_fill; log_descriptor.bc.buffer->size= log_descriptor.bc.current_page_fill= @@ -6109,6 +6117,8 @@ static void translog_force_current_buffer_to_finish() a log flush fails (we however don't want to crash the entire mysqld, but stopping all engine's operations immediately would make sense). Same applies to translog_write_record(). + + @todo: remove serialization and make group commit. */ my_bool translog_flush(LSN lsn) @@ -6121,6 +6131,7 @@ my_bool translog_flush(LSN lsn) DBUG_PRINT("enter", ("Flush up to LSN: (%lu,0x%lx)", LSN_IN_PARTS(lsn))); DBUG_ASSERT(translog_inited == 1); + pthread_mutex_lock(&log_descriptor.log_flush_lock); translog_lock(); old_flushed= log_descriptor.flushed; for (;;) @@ -6135,8 +6146,7 @@ my_bool translog_flush(LSN lsn) { DBUG_PRINT("info", ("already flushed: (%lu,0x%lx)", LSN_IN_PARTS(log_descriptor.flushed))); - translog_unlock(); - DBUG_RETURN(0); + goto out; } /* send to the file if it is not sent */ sent_to_file= translog_get_sent_to_file(); @@ -6163,12 +6173,15 @@ my_bool translog_flush(LSN lsn) } } while ((buffer_start != buffer_no) && cmp_translog_addr(log_descriptor.flushed, lsn) < 0); - if (buffer_unlock != NULL) + if (buffer_unlock != NULL && buffer_unlock != buffer) translog_buffer_unlock(buffer_unlock); rc= translog_buffer_flush(buffer); translog_buffer_unlock(buffer); if (rc) - DBUG_RETURN(1); + { + rc= 1; + goto out; + } if (!full_circle) translog_lock(); } @@ -6187,8 +6200,8 @@ my_bool translog_flush(LSN lsn) if ((log_descriptor.log_file_num[cache_index]= open_logfile_by_number_no_cache(i)) == -1) { - translog_unlock(); - DBUG_RETURN(1); + rc= 1; + goto out; } } file= log_descriptor.log_file_num[cache_index]; @@ -6199,7 +6212,9 @@ my_bool translog_flush(LSN lsn) log_descriptor.flushed= sent_to_file; /** @todo LOG decide if syncing of directory is needed */ rc|= my_sync(log_descriptor.directory_fd, MYF(MY_WME | MY_IGNORE_BADFD)); +out: translog_unlock(); + pthread_mutex_unlock(&log_descriptor.log_flush_lock); DBUG_RETURN(rc); } -- cgit v1.2.1 From e57cccc65ca6d6a7da8f1fa7e7efb9826defebd1 Mon Sep 17 00:00:00 2001 From: unknown Date: Mon, 24 Sep 2007 10:24:51 +0200 Subject: Optimization (useful at least for the Maria engine): we disable logging of insertions made by CREATE SELECT. sql/sql_insert.cc: If error during the CREATE SELECT we drop the table, so no need for engines to do logging of the insertions (optimization). Engines require that disabling is done before locking and re-enabling is done before unlocking; as table creation and locking is done as one function (create_table_from_items()) we disable before calling this function and re-enable before unlocking, in send_eof() (called if success) and abort() (called if error). Question for reviewer: would it be better to do the disabling between creation and locking, so inside create_table_from_items(), given that this function is used only by CREATE SELECT? --- sql/sql_insert.cc | 10 ++++++++++ 1 file changed, 10 insertions(+) diff --git a/sql/sql_insert.cc b/sql/sql_insert.cc index 11db88d8f5e..d7267e6ecf6 100644 --- a/sql/sql_insert.cc +++ b/sql/sql_insert.cc @@ -3458,6 +3458,13 @@ select_create::prepare(List &values, SELECT_LEX_UNIT *u) thd->binlog_start_trans_and_stmt(); } + /* + If error during the CREATE SELECT we drop the table, so no need for + engines to do logging of insertions (optimization). + */ + if (ha_enable_transaction(thd, FALSE)) + DBUG_RETURN(-1); + if (!(table= create_table_from_items(thd, create_info, create_table, alter_info, &values, &thd->extra_lock, hook_ptr))) @@ -3602,6 +3609,7 @@ bool select_create::send_eof() table->file->extra(HA_EXTRA_NO_IGNORE_DUP_KEY); table->file->extra(HA_EXTRA_WRITE_CANNOT_REPLACE); + ha_enable_transaction(thd, TRUE); if (thd->extra_lock) { mysql_unlock_tables(thd, thd->extra_lock); @@ -3640,6 +3648,8 @@ void select_create::abort() if (thd->current_stmt_binlog_row_based) ha_rollback_stmt(thd); + ha_enable_transaction(thd, TRUE); + if (thd->extra_lock) { mysql_unlock_tables(thd, thd->extra_lock); -- cgit v1.2.1 From 8b5dddbc006afe8f6dae8408cea7481c17dade72 Mon Sep 17 00:00:00 2001 From: unknown Date: Tue, 25 Sep 2007 11:54:35 +0200 Subject: WL#3072 Maria recovery Progress reports on stderr if doing recovery from ha_maria; don't do checkpoints if activity since last checkpoint < 2MB (no change in fact as background thread is disabled for now); recovery trace is only if EXTRA_DEBUG now (better for benchmarks). storage/maria/ma_checkpoint.c: don't do checkpoints if activity (log writes plus page flushes) since last checkpoint was < 2MB. storage/maria/ma_recovery.c: progress reports in recovery (10%, transactions left to rollback etc); that is only if from ha_maria and is displayed on stderr. Recovery trace is now created only if EXTRA_DEBUG. storage/maria/ma_test_recovery.expected: update (--debug gone) storage/maria/ma_test_recovery: don't use --debug, as it can absent from binary --- storage/maria/ma_checkpoint.c | 15 +- storage/maria/ma_recovery.c | 472 ++++++++++++++++++-------------- storage/maria/ma_test_recovery | 2 +- storage/maria/ma_test_recovery.expected | 48 ++-- 4 files changed, 302 insertions(+), 235 deletions(-) diff --git a/storage/maria/ma_checkpoint.c b/storage/maria/ma_checkpoint.c index 0e8b558e7c5..4446285fce9 100644 --- a/storage/maria/ma_checkpoint.c +++ b/storage/maria/ma_checkpoint.c @@ -544,7 +544,9 @@ static int filter_flush_data_file_evenly(enum pagecache_page_type type, pthread_handler_t ma_checkpoint_background(void *arg __attribute__((unused))) { const uint sleep_unit= 1 /* 1 second */, - time_between_checkpoints= 30; /* 30 sleep units */ + time_between_checkpoints= 30, /* 30 sleep units */ + /** @brief At least this of log/page bytes written between checkpoints */ + checkpoint_min_activity= 2*1024*1024; uint sleeps= 0; my_thread_init(); @@ -570,16 +572,17 @@ pthread_handler_t ma_checkpoint_background(void *arg __attribute__((unused))) in the checkpoint. */ /* - No checkpoint if no work of interest for recovery was done + No checkpoint if little work of interest for recovery was done since last checkpoint. Such work includes log writing (lengthens recovery, checkpoint would shorten it), page flushing (checkpoint would decrease the amount of read pages in recovery). */ - if ((translog_get_horizon() == log_horizon_at_last_checkpoint) && - (pagecache_flushes_at_last_checkpoint == - maria_pagecache->global_cache_write)) + if (((translog_get_horizon() - log_horizon_at_last_checkpoint) + + (maria_pagecache->global_cache_write - + pagecache_flushes_at_last_checkpoint) * + maria_pagecache->block_size) < checkpoint_min_activity) { - /* safety against errors during flush by this thread: */ + /* don't take checkpoint, so don't know what to flush */ pages_to_flush_before_next_checkpoint= 0; break; } diff --git a/storage/maria/ma_recovery.c b/storage/maria/ma_recovery.c index 23e5ffe2a29..e740e334b5f 100644 --- a/storage/maria/ma_recovery.c +++ b/storage/maria/ma_recovery.c @@ -105,6 +105,7 @@ static int new_table(uint16 sid, const char *name, static int new_page(File fileid, pgcache_page_no_t pageid, LSN rec_lsn, struct st_dirty_page *dirty_page); static int close_all_tables(void); +static void print_redo_phase_progress(TRANSLOG_ADDRESS addr); /** @brief global [out] buffer for translog_read_record(); never shrinks */ static LEX_STRING log_record_buffer; @@ -118,23 +119,19 @@ static void enlarge_buffer(const TRANSLOG_HEADER_BUFFER *rec) MYF(MY_WME | MY_ALLOW_ZERO_PTR)); } } -static my_bool recovery_message_printed; -static inline void print_recovery_message(void) +static my_bool redo_phase_message_printed; +/** @brief Prints to a trace file if it is not NULL */ +void tprint(FILE *trace_file, const char *format, ...) + ATTRIBUTE_FORMAT(printf, 2, 3); +void tprint(FILE *trace_file, const char *format, ...) { - /* - If we're really doing a recovery (reading REDOs or UNDOs), we print a - one-line message when we start it and when we end it. It goes to stderr, - not tracef, so that it is visible in the error log (soon we should maybe - use sql_print_error). We don't print if if tracef is stdout as stdout will - be seen by the user and thus convey sufficient info already. - */ - if (!recovery_message_printed && (tracef != stdout)) - { - recovery_message_printed= TRUE; - /** @todo RECOVERY BUG all prints to stderr should go to error log */ - fprintf(stderr, "Maria engine: starting recovery\n"); - } + va_list args; + va_start(args, format); + if (trace_file != NULL) + vfprintf(trace_file, format, args); + va_end(args); } + #define ALERT_USER() DBUG_ASSERT(0) @@ -160,15 +157,18 @@ int maria_recover(void) DBUG_ASSERT(!maria_in_recovery); maria_in_recovery= TRUE; - if ((trace_file= fopen("maria_recovery.trace", "w"))) - { - fprintf(trace_file, "TRACE of the last MARIA recovery from mysqld\n"); - DBUG_ASSERT(maria_pagecache->inited); - res= maria_apply_log(LSN_IMPOSSIBLE, TRUE, trace_file, TRUE, TRUE); - if (!res) - fprintf(trace_file, "SUCCESS\n"); +#if !defined(DBUG_OFF) && defined(EXTRA_DEBUG) + trace_file= fopen("maria_recovery.trace", "w"); +#else + trace_file= NULL; /* no trace file for being fast */ +#endif + tprint(trace_file, "TRACE of the last MARIA recovery from mysqld\n"); + DBUG_ASSERT(maria_pagecache->inited); + res= maria_apply_log(LSN_IMPOSSIBLE, TRUE, trace_file, TRUE, TRUE); + if (!res) + tprint(trace_file, "SUCCESS\n"); + if (trace_file) fclose(trace_file); - } maria_in_recovery= FALSE; DBUG_RETURN(res); } @@ -211,7 +211,7 @@ int maria_apply_log(LSN from_lsn, my_bool apply, FILE *trace_file, if (!all_active_trans || !all_tables) goto err; - recovery_message_printed= FALSE; + redo_phase_message_printed= FALSE; tracef= trace_file; if (!(skip_DDLs= skip_DDLs_arg)) { @@ -232,11 +232,11 @@ int maria_apply_log(LSN from_lsn, my_bool apply, FILE *trace_file, there was a crash during a DDL (see comment in execution of REDO_RENAME_TABLE ). */ - fprintf(tracef, "WARNING: MySQL server currently disables log records" - " about insertion of data by ALTER TABLE" - " (copy_data_between_tables()), applying of log records may" - " well not work. Additionally, applying of DDL records will" - " cause damage if there are tables left by a crash of a DDL.\n"); + tprint(tracef, "WARNING: MySQL server currently disables log records" + " about insertion of data by ALTER TABLE" + " (copy_data_between_tables()), applying of log records may" + " well not work. Additionally, applying of DDL records will" + " cause damage if there are tables left by a crash of a DDL.\n"); } if (from_lsn == LSN_IMPOSSIBLE) @@ -279,8 +279,8 @@ int maria_apply_log(LSN from_lsn, my_bool apply, FILE *trace_file, return 1; } else if (unfinished_trans > 0) - fprintf(tracef, "WARNING: %u unfinished transactions; some tables may be" - " left inconsistent!\n", unfinished_trans); + tprint(tracef, "WARNING: %u unfinished transactions; some tables may be" + " left inconsistent!\n", unfinished_trans); /* we don't use maria_panic() because it would maria_end(), and Recovery does @@ -290,12 +290,10 @@ int maria_apply_log(LSN from_lsn, my_bool apply, FILE *trace_file, goto err; /* If inside ha_maria, a checkpoint will soon be taken and save our work */ - if (recovery_message_printed && (tracef != stdout)) - fprintf(stderr, "Maria engine: finished recovery\n"); goto end; err: error= 1; - fprintf(tracef, "Recovery of tables with transaction logs FAILED\n"); + tprint(tracef, "Recovery of tables with transaction logs FAILED\n"); end: hash_free(&all_dirty_pages); bzero(&all_dirty_pages, sizeof(all_dirty_pages)); @@ -308,6 +306,11 @@ end: my_free(log_record_buffer.str, MYF(MY_ALLOW_ZERO_PTR)); log_record_buffer.str= NULL; log_record_buffer.length= 0; + if (tracef != stdout && redo_phase_message_printed) + { + /** @todo RECOVERY BUG all prints to stderr should go to error log */ + fprintf(stderr, "\n"); + } /* we don't cleanly close tables if we hit some error (may corrupt them) */ DBUG_RETURN(error); } @@ -322,9 +325,9 @@ static void display_record_position(const LOG_DESC *log_desc, if number==0, we're going over records which we had already seen and which form a group, so we indent below the group's end record */ - fprintf(tracef, "%sRec#%u LSN (%lu,0x%lx) short_trid %u %s(num_type:%u) len %lu\n", - number ? "" : " ", number, LSN_IN_PARTS(rec->lsn), - rec->short_trid, log_desc->name, rec->type, + tprint(tracef, "%sRec#%u LSN (%lu,0x%lx) short_trid %u %s(num_type:%u) len %lu\n", + number ? "" : " ", number, LSN_IN_PARTS(rec->lsn), + rec->short_trid, log_desc->name, rec->type, (ulong)rec->record_length); } @@ -340,7 +343,7 @@ static int display_and_apply_record(const LOG_DESC *log_desc, return 1; } if ((error= (*log_desc->record_execute_in_redo_phase)(rec))) - fprintf(tracef, "Got error when executing redo on record\n"); + tprint(tracef, "Got error when executing redo on record\n"); return error; } @@ -353,8 +356,8 @@ prototype_redo_exec_hook(LONG_TRANSACTION_ID) LSN gslsn= all_active_trans[sid].group_start_lsn; if (gslsn != LSN_IMPOSSIBLE) { - fprintf(tracef, "Group at LSN (%lu,0x%lx) short_trid %u aborted\n", - LSN_IN_PARTS(gslsn), sid); + tprint(tracef, "Group at LSN (%lu,0x%lx) short_trid %u aborted\n", + LSN_IN_PARTS(gslsn), sid); all_active_trans[sid].group_start_lsn= LSN_IMPOSSIBLE; } if (long_trid != 0) @@ -364,10 +367,10 @@ prototype_redo_exec_hook(LONG_TRANSACTION_ID) { char llbuf[22]; llstr(long_trid, llbuf); - fprintf(tracef, "Found an old transaction long_trid %s short_trid %u" - " with same short id as this new transaction, and has neither" - " committed nor rollback (undo_lsn: (%lu,0x%lx))\n", llbuf, - sid, LSN_IN_PARTS(ulsn)); + tprint(tracef, "Found an old transaction long_trid %s short_trid %u" + " with same short id as this new transaction, and has neither" + " committed nor rollback (undo_lsn: (%lu,0x%lx))\n", llbuf, + sid, LSN_IN_PARTS(ulsn)); goto err; } } @@ -388,8 +391,8 @@ static void new_transaction(uint16 sid, TrID long_id, LSN undo_lsn, char llbuf[22]; all_active_trans[sid].long_trid= long_id; llstr(long_id, llbuf); - fprintf(tracef, "Transaction long_trid %s short_trid %u starts\n", - llbuf, sid); + tprint(tracef, "Transaction long_trid %s short_trid %u starts\n", + llbuf, sid); all_active_trans[sid].undo_lsn= undo_lsn; all_active_trans[sid].first_undo_lsn= first_undo_lsn; set_if_bigger(max_long_trid, long_id); @@ -414,7 +417,7 @@ prototype_redo_exec_hook(REDO_CREATE_TABLE) MARIA_HA *info= NULL; if (skip_DDLs) { - fprintf(tracef, "we skip DDLs\n"); + tprint(tracef, "we skip DDLs\n"); return 0; } enlarge_buffer(rec); @@ -423,11 +426,11 @@ prototype_redo_exec_hook(REDO_CREATE_TABLE) log_record_buffer.str, NULL) != rec->record_length) { - fprintf(tracef, "Failed to read record\n"); + tprint(tracef, "Failed to read record\n"); goto end; } name= log_record_buffer.str; - fprintf(tracef, "Table '%s'", name); + tprint(tracef, "Table '%s'", name); /* we try hard to get create_rename_lsn, to avoid mistakes if possible */ info= maria_open(name, O_RDONLY, HA_OPEN_FOR_REPAIR); if (info) @@ -436,7 +439,7 @@ prototype_redo_exec_hook(REDO_CREATE_TABLE) /* check that we're not already using it */ if (share->reopen != 1) { - fprintf(tracef, ", is already open (reopen=%u)\n", share->reopen); + tprint(tracef, ", is already open (reopen=%u)\n", share->reopen); ALERT_USER(); goto end; } @@ -448,22 +451,22 @@ prototype_redo_exec_hook(REDO_CREATE_TABLE) one was renamed to its name, thus create_rename_lsn is 0 and should not be trusted. */ - fprintf(tracef, ", is not transactional, ignoring creation\n"); + tprint(tracef, ", is not transactional, ignoring creation\n"); ALERT_USER(); error= 0; goto end; } if (cmp_translog_addr(share->state.create_rename_lsn, rec->lsn) >= 0) { - fprintf(tracef, ", has create_rename_lsn (%lu,0x%lx) more recent than" - " record, ignoring creation", - LSN_IN_PARTS(share->state.create_rename_lsn)); + tprint(tracef, ", has create_rename_lsn (%lu,0x%lx) more recent than" + " record, ignoring creation", + LSN_IN_PARTS(share->state.create_rename_lsn)); error= 0; goto end; } if (maria_is_crashed(info)) { - fprintf(tracef, ", is crashed, can't recreate it"); + tprint(tracef, ", is crashed, can't recreate it"); ALERT_USER(); goto end; } @@ -471,23 +474,23 @@ prototype_redo_exec_hook(REDO_CREATE_TABLE) info= NULL; } else /* one or two files absent, or header corrupted... */ - fprintf(tracef, "can't be opened, probably does not exist"); + tprint(tracef, "can't be opened, probably does not exist"); /* if does not exist, or is older, overwrite it */ /** @todo symlinks */ ptr= name + strlen(name) + 1; if ((flags= ptr[0] ? HA_DONT_TOUCH_DATA : 0)) - fprintf(tracef, ", we will only touch index file"); + tprint(tracef, ", we will only touch index file"); fn_format(filename, name, "", MARIA_NAME_IEXT, (MY_UNPACK_FILENAME | (flags & HA_DONT_TOUCH_DATA) ? MY_RETURN_REAL_PATH : 0) | MY_APPEND_EXT); linkname_ptr= NULL; create_flag= MY_DELETE_OLD; - fprintf(tracef, ", creating as '%s'", filename); + tprint(tracef, ", creating as '%s'", filename); if ((kfile= my_create_with_symlink(linkname_ptr, filename, 0, create_mode, MYF(MY_WME|create_flag))) < 0) { - fprintf(tracef, "Failed to create index file\n"); + tprint(tracef, "Failed to create index file\n"); goto end; } ptr++; @@ -504,7 +507,7 @@ prototype_redo_exec_hook(REDO_CREATE_TABLE) kfile_size_before_extension, 0, MYF(MY_NABP|MY_WME)) || my_chsize(kfile, keystart, 0, MYF(MY_WME))) { - fprintf(tracef, "Failed to write to index file\n"); + tprint(tracef, "Failed to write to index file\n"); goto end; } if (!(flags & HA_DONT_TOUCH_DATA)) @@ -518,7 +521,7 @@ prototype_redo_exec_hook(REDO_CREATE_TABLE) MYF(MY_WME | create_flag))) < 0) || my_close(dfile, MYF(MY_WME))) { - fprintf(tracef, "Failed to create data file\n"); + tprint(tracef, "Failed to create data file\n"); goto end; } /* @@ -530,13 +533,13 @@ prototype_redo_exec_hook(REDO_CREATE_TABLE) if (((info= maria_open(name, O_RDONLY, 0)) == NULL) || _ma_initialize_data_file(info->s, info->dfile.file)) { - fprintf(tracef, "Failed to open new table or write to data file\n"); + tprint(tracef, "Failed to open new table or write to data file\n"); goto end; } } error= 0; end: - fprintf(tracef, "\n"); + tprint(tracef, "\n"); if (kfile >= 0) error|= my_close(kfile, MYF(MY_WME)); if (info != NULL) @@ -552,7 +555,7 @@ prototype_redo_exec_hook(REDO_RENAME_TABLE) MARIA_HA *info= NULL; if (skip_DDLs) { - fprintf(tracef, "we skip DDLs\n"); + tprint(tracef, "we skip DDLs\n"); return 0; } enlarge_buffer(rec); @@ -561,13 +564,13 @@ prototype_redo_exec_hook(REDO_RENAME_TABLE) log_record_buffer.str, NULL) != rec->record_length) { - fprintf(tracef, "Failed to read record\n"); + tprint(tracef, "Failed to read record\n"); goto end; } old_name= log_record_buffer.str; new_name= old_name + strlen(old_name) + 1; - fprintf(tracef, "Table '%s' to rename to '%s'; old-name table ", old_name, - new_name); + tprint(tracef, "Table '%s' to rename to '%s'; old-name table ", old_name, + new_name); /* Here is why we skip CREATE/DROP/RENAME when doing a recovery from ha_maria (whereas we do when called from maria_read_log). Consider: @@ -606,22 +609,22 @@ prototype_redo_exec_hook(REDO_RENAME_TABLE) */ if (!share->base.born_transactional) { - fprintf(tracef, ", is not transactional, ignoring renaming\n"); + tprint(tracef, ", is not transactional, ignoring renaming\n"); ALERT_USER(); error= 0; goto end; } if (cmp_translog_addr(share->state.create_rename_lsn, rec->lsn) >= 0) { - fprintf(tracef, ", has create_rename_lsn (%lu,0x%lx) more recent than" - " record, ignoring renaming", - LSN_IN_PARTS(share->state.create_rename_lsn)); + tprint(tracef, ", has create_rename_lsn (%lu,0x%lx) more recent than" + " record, ignoring renaming", + LSN_IN_PARTS(share->state.create_rename_lsn)); error= 0; goto end; } if (maria_is_crashed(info)) { - fprintf(tracef, ", is crashed, can't rename it"); + tprint(tracef, ", is crashed, can't rename it"); ALERT_USER(); goto end; } @@ -634,11 +637,11 @@ prototype_redo_exec_hook(REDO_RENAME_TABLE) maria_close(info)) goto end; info= NULL; - fprintf(tracef, ", is ok for renaming; new-name table "); + tprint(tracef, ", is ok for renaming; new-name table "); } else /* one or two files absent, or header corrupted... */ { - fprintf(tracef, ", can't be opened, probably does not exist"); + tprint(tracef, ", can't be opened, probably does not exist"); error= 0; goto end; } @@ -661,21 +664,21 @@ prototype_redo_exec_hook(REDO_RENAME_TABLE) /* We should not have open instances on this table. */ if (share->reopen != 1) { - fprintf(tracef, ", is already open (reopen=%u)\n", share->reopen); + tprint(tracef, ", is already open (reopen=%u)\n", share->reopen); ALERT_USER(); goto end; } if (!share->base.born_transactional) { - fprintf(tracef, ", is not transactional, ignoring renaming\n"); + tprint(tracef, ", is not transactional, ignoring renaming\n"); ALERT_USER(); goto drop; } if (cmp_translog_addr(share->state.create_rename_lsn, rec->lsn) >= 0) { - fprintf(tracef, ", has create_rename_lsn (%lu,0x%lx) more recent than" - " record, ignoring renaming", - LSN_IN_PARTS(share->state.create_rename_lsn)); + tprint(tracef, ", has create_rename_lsn (%lu,0x%lx) more recent than" + " record, ignoring renaming", + LSN_IN_PARTS(share->state.create_rename_lsn)); /* We have to drop the old_name table. Consider: CREATE TABLE t; @@ -691,7 +694,7 @@ prototype_redo_exec_hook(REDO_RENAME_TABLE) } if (maria_is_crashed(info)) { - fprintf(tracef, ", is crashed, can't rename it"); + tprint(tracef, ", is crashed, can't rename it"); ALERT_USER(); goto end; } @@ -699,21 +702,21 @@ prototype_redo_exec_hook(REDO_RENAME_TABLE) goto end; info= NULL; /* abnormal situation */ - fprintf(tracef, ", exists but is older than record, can't rename it"); + tprint(tracef, ", exists but is older than record, can't rename it"); goto end; } else /* one or two files absent, or header corrupted... */ - fprintf(tracef, ", can't be opened, probably does not exist"); - fprintf(tracef, ", renaming '%s'", old_name); + tprint(tracef, ", can't be opened, probably does not exist"); + tprint(tracef, ", renaming '%s'", old_name); if (maria_rename(old_name, new_name)) { - fprintf(tracef, "Failed to rename table\n"); + tprint(tracef, "Failed to rename table\n"); goto end; } info= maria_open(new_name, O_RDONLY, 0); if (info == NULL) { - fprintf(tracef, "Failed to open renamed table\n"); + tprint(tracef, "Failed to open renamed table\n"); goto end; } if (_ma_update_create_rename_lsn(info->s, rec->lsn, TRUE)) @@ -724,16 +727,16 @@ prototype_redo_exec_hook(REDO_RENAME_TABLE) error= 0; goto end; drop: - fprintf(tracef, ", only dropping '%s'", old_name); + tprint(tracef, ", only dropping '%s'", old_name); if (maria_delete_table(old_name)) { - fprintf(tracef, "Failed to drop table\n"); + tprint(tracef, "Failed to drop table\n"); goto end; } error= 0; goto end; end: - fprintf(tracef, "\n"); + tprint(tracef, "\n"); if (info != NULL) error|= maria_close(info); return error; @@ -753,7 +756,7 @@ prototype_redo_exec_hook(REDO_REPAIR_TABLE) REPAIR is not exactly a DDL, but it manipulates files without logging insertions into them. */ - fprintf(tracef, "we skip DDLs\n"); + tprint(tracef, "we skip DDLs\n"); return 0; } if ((info= get_MARIA_HA_from_REDO_record(rec)) == NULL) @@ -762,7 +765,7 @@ prototype_redo_exec_hook(REDO_REPAIR_TABLE) Otherwise, the mapping is newer than the table, and our record is newer than the mapping, so we can repair. */ - fprintf(tracef, " repairing...\n"); + tprint(tracef, " repairing...\n"); /** @todo RECOVERY BUG fix this: the maria_chk_init() call causes a heap of linker errors in ha_maria.cc! @@ -794,7 +797,7 @@ prototype_redo_exec_hook(REDO_DROP_TABLE) MARIA_HA *info= NULL; if (skip_DDLs) { - fprintf(tracef, "we skip DDLs\n"); + tprint(tracef, "we skip DDLs\n"); return 0; } enlarge_buffer(rec); @@ -803,11 +806,11 @@ prototype_redo_exec_hook(REDO_DROP_TABLE) log_record_buffer.str, NULL) != rec->record_length) { - fprintf(tracef, "Failed to read record\n"); + tprint(tracef, "Failed to read record\n"); goto end; } name= log_record_buffer.str; - fprintf(tracef, "Table '%s'", name); + tprint(tracef, "Table '%s'", name); info= maria_open(name, O_RDONLY, HA_OPEN_FOR_REPAIR); if (info) { @@ -818,22 +821,22 @@ prototype_redo_exec_hook(REDO_DROP_TABLE) */ if (!share->base.born_transactional) { - fprintf(tracef, ", is not transactional, ignoring removal\n"); + tprint(tracef, ", is not transactional, ignoring removal\n"); ALERT_USER(); error= 0; goto end; } if (cmp_translog_addr(share->state.create_rename_lsn, rec->lsn) >= 0) { - fprintf(tracef, ", has create_rename_lsn (%lu,0x%lx) more recent than" - " record, ignoring removal", - LSN_IN_PARTS(share->state.create_rename_lsn)); + tprint(tracef, ", has create_rename_lsn (%lu,0x%lx) more recent than" + " record, ignoring removal", + LSN_IN_PARTS(share->state.create_rename_lsn)); error= 0; goto end; } if (maria_is_crashed(info)) { - fprintf(tracef, ", is crashed, can't drop it"); + tprint(tracef, ", is crashed, can't drop it"); ALERT_USER(); goto end; } @@ -847,18 +850,18 @@ prototype_redo_exec_hook(REDO_DROP_TABLE) goto end; info= NULL; /* if it is older, or its header is corrupted, drop it */ - fprintf(tracef, ", dropping '%s'", name); + tprint(tracef, ", dropping '%s'", name); if (maria_delete_table(name)) { - fprintf(tracef, "Failed to drop table\n"); + tprint(tracef, "Failed to drop table\n"); goto end; } } else /* one or two files absent, or header corrupted... */ - fprintf(tracef,", can't be opened, probably does not exist"); + tprint(tracef,", can't be opened, probably does not exist"); error= 0; end: - fprintf(tracef, "\n"); + tprint(tracef, "\n"); if (info != NULL) error|= maria_close(info); return error; @@ -880,7 +883,7 @@ prototype_redo_exec_hook(FILE_ID) checkpoint time (table was closed or repaired), a flush and force happened and so mapping is not needed. */ - fprintf(tracef, "ignoring because before checkpoint\n"); + tprint(tracef, "ignoring because before checkpoint\n"); return 0; } @@ -890,18 +893,18 @@ prototype_redo_exec_hook(FILE_ID) log_record_buffer.str, NULL) != rec->record_length) { - fprintf(tracef, "Failed to read record\n"); + tprint(tracef, "Failed to read record\n"); goto end; } sid= fileid_korr(log_record_buffer.str); info= all_tables[sid].info; if (info != NULL) { - fprintf(tracef, " Closing table '%s'\n", info->s->open_file_name); + tprint(tracef, " Closing table '%s'\n", info->s->open_file_name); prepare_table_for_close(info, rec->lsn); if (maria_close(info)) { - fprintf(tracef, "Failed to close table\n"); + tprint(tracef, "Failed to close table\n"); goto end; } all_tables[sid].info= NULL; @@ -926,11 +929,11 @@ static int new_table(uint16 sid, const char *name, */ int error= 1; - fprintf(tracef, "Table '%s', id %u", name, sid); + tprint(tracef, "Table '%s', id %u", name, sid); MARIA_HA *info= maria_open(name, O_RDWR, HA_OPEN_FOR_REPAIR); if (info == NULL) { - fprintf(tracef, ", is absent (must have been dropped later?)" + tprint(tracef, ", is absent (must have been dropped later?)" " or its header is so corrupted that we cannot open it;" " we skip it\n"); error= 0; @@ -938,31 +941,31 @@ static int new_table(uint16 sid, const char *name, } if (maria_is_crashed(info)) { - fprintf(tracef, "Table is crashed, can't apply log records to it\n"); + tprint(tracef, "Table is crashed, can't apply log records to it\n"); goto end; } MARIA_SHARE *share= info->s; /* check that we're not already using it */ if (share->reopen != 1) { - fprintf(tracef, ", is already open (reopen=%u)\n", share->reopen); + tprint(tracef, ", is already open (reopen=%u)\n", share->reopen); ALERT_USER(); goto end; } DBUG_ASSERT(share->now_transactional == share->base.born_transactional); if (!share->base.born_transactional) { - fprintf(tracef, ", is not transactional\n"); + tprint(tracef, ", is not transactional\n"); ALERT_USER(); error= -1; goto end; } if (cmp_translog_addr(lsn_of_file_id, share->state.create_rename_lsn) <= 0) { - fprintf(tracef, ", has create_rename_lsn (%lu,0x%lx) more recent than" - " LOGREC_FILE_ID's LSN (%lu,0x%lx), ignoring open request", - LSN_IN_PARTS(share->state.create_rename_lsn), - LSN_IN_PARTS(lsn_of_file_id)); + tprint(tracef, ", has create_rename_lsn (%lu,0x%lx) more recent than" + " LOGREC_FILE_ID's LSN (%lu,0x%lx), ignoring open request", + LSN_IN_PARTS(share->state.create_rename_lsn), + LSN_IN_PARTS(lsn_of_file_id)); error= -1; goto end; } @@ -974,14 +977,14 @@ static int new_table(uint16 sid, const char *name, if ((dfile_len == MY_FILEPOS_ERROR) || (kfile_len == MY_FILEPOS_ERROR)) { - fprintf(tracef, ", length unknown\n"); + tprint(tracef, ", length unknown\n"); goto end; } share->state.state.data_file_length= dfile_len; share->state.state.key_file_length= kfile_len; if ((dfile_len % share->block_size) > 0) { - fprintf(tracef, ", has too short last page\n"); + tprint(tracef, ", has too short last page\n"); /* Recovery will fix this, no error */ ALERT_USER(); } @@ -1003,10 +1006,10 @@ static int new_table(uint16 sid, const char *name, if you change that, know that some records in REDO phase call _ma_update_create_rename_lsn() which resets info->s->id. */ - fprintf(tracef, ", opened"); + tprint(tracef, ", opened"); error= 0; end: - fprintf(tracef, "\n"); + tprint(tracef, "\n"); if (error) { if (info != NULL) @@ -1043,12 +1046,16 @@ prototype_redo_exec_hook(REDO_INSERT_ROW_HEAD) differences. So we use the UNDO's LSN which is current_group_end_lsn. */ enlarge_buffer(rec); - if (log_record_buffer.str == NULL || - translog_read_record(rec->lsn, 0, rec->record_length, + if (log_record_buffer.str == NULL) + { + tprint(tracef, "Failed to read allocate buffer for record\n"); + goto end; + } + if (translog_read_record(rec->lsn, 0, rec->record_length, log_record_buffer.str, NULL) != - rec->record_length) + rec->record_length) { - fprintf(tracef, "Failed to read record\n"); + tprint(tracef, "Failed to read record\n"); goto end; } buff= log_record_buffer.str; @@ -1083,7 +1090,7 @@ prototype_redo_exec_hook(REDO_INSERT_ROW_TAIL) log_record_buffer.str, NULL) != rec->record_length) { - fprintf(tracef, "Failed to read record\n"); + tprint(tracef, "Failed to read record\n"); goto end; } buff= log_record_buffer.str; @@ -1152,7 +1159,7 @@ prototype_redo_exec_hook(REDO_PURGE_BLOCKS) log_record_buffer.str, NULL) != rec->record_length) { - fprintf(tracef, "Failed to read record\n"); + tprint(tracef, "Failed to read record\n"); goto end; } @@ -1172,7 +1179,7 @@ prototype_redo_exec_hook(REDO_DELETE_ALL) MARIA_HA *info= get_MARIA_HA_from_REDO_record(rec); if (info == NULL) return 0; - fprintf(tracef, " deleting all %lu rows\n", + tprint(tracef, " deleting all %lu rows\n", (ulong)info->s->state.state.records); if (maria_delete_all_rows(info)) goto end; @@ -1195,7 +1202,7 @@ prototype_redo_exec_hook(UNDO_ROW_INSERT) set_undo_lsn_for_active_trans(rec->short_trid, rec->lsn); if (cmp_translog_addr(rec->lsn, info->s->state.is_of_horizon) >= 0) { - fprintf(tracef, " state older than record, updating rows' count\n"); + tprint(tracef, " state older than record, updating rows' count\n"); info->s->state.state.records++; /** @todo RECOVERY BUG Also update the table's checksum */ /** @@ -1205,7 +1212,7 @@ prototype_redo_exec_hook(UNDO_ROW_INSERT) info->s->state.changed|= STATE_CHANGED | STATE_NOT_ANALYZED | STATE_NOT_OPTIMIZED_KEYS | STATE_NOT_SORTED_PAGES; } - fprintf(tracef, " rows' count %lu\n", (ulong)info->s->state.state.records); + tprint(tracef, " rows' count %lu\n", (ulong)info->s->state.state.records); return 0; } @@ -1218,12 +1225,12 @@ prototype_redo_exec_hook(UNDO_ROW_DELETE) set_undo_lsn_for_active_trans(rec->short_trid, rec->lsn); if (cmp_translog_addr(rec->lsn, info->s->state.is_of_horizon) >= 0) { - fprintf(tracef, " state older than record, updating rows' count\n"); + tprint(tracef, " state older than record, updating rows' count\n"); info->s->state.state.records--; info->s->state.changed|= STATE_CHANGED | STATE_NOT_ANALYZED | STATE_NOT_OPTIMIZED_KEYS | STATE_NOT_SORTED_PAGES; } - fprintf(tracef, " rows' count %lu\n", (ulong)info->s->state.state.records); + tprint(tracef, " rows' count %lu\n", (ulong)info->s->state.state.records); return 0; } @@ -1251,24 +1258,24 @@ prototype_redo_exec_hook(COMMIT) char llbuf[22]; if (long_trid == 0) { - fprintf(tracef, "We don't know about transaction with short_trid %u;" + tprint(tracef, "We don't know about transaction with short_trid %u;" "it probably committed long ago, forget it\n", sid); return 0; } llstr(long_trid, llbuf); - fprintf(tracef, "Transaction long_trid %s short_trid %u committed", llbuf, sid); + tprint(tracef, "Transaction long_trid %s short_trid %u committed", llbuf, sid); if (gslsn != LSN_IMPOSSIBLE) { /* It's not an error, it may be that trn got a disk error when writing to a table, so an unfinished group staid in the log. */ - fprintf(tracef, ", with group at LSN (%lu,0x%lx) short_trid %u aborted\n", - LSN_IN_PARTS(gslsn), sid); + tprint(tracef, ", with group at LSN (%lu,0x%lx) short_trid %u aborted\n", + LSN_IN_PARTS(gslsn), sid); all_active_trans[sid].group_start_lsn= LSN_IMPOSSIBLE; } else - fprintf(tracef, "\n"); + tprint(tracef, "\n"); bzero(&all_active_trans[sid], sizeof(all_active_trans[sid])); #ifdef MARIA_VERSIONING /* @@ -1293,11 +1300,11 @@ prototype_redo_exec_hook(CLR_END) const LOG_DESC *log_desc= &log_record_type_descriptor[undone_record_type]; set_undo_lsn_for_active_trans(rec->short_trid, previous_undo_lsn); - fprintf(tracef, " CLR_END was about %s, undo_lsn now LSN (%lu,0x%lx)\n", - log_desc->name, LSN_IN_PARTS(previous_undo_lsn)); + tprint(tracef, " CLR_END was about %s, undo_lsn now LSN (%lu,0x%lx)\n", + log_desc->name, LSN_IN_PARTS(previous_undo_lsn)); if (cmp_translog_addr(rec->lsn, info->s->state.is_of_horizon) >= 0) { - fprintf(tracef, " state older than record, updating rows' count\n"); + tprint(tracef, " state older than record, updating rows' count\n"); switch (undone_record_type) { case LOGREC_UNDO_ROW_DELETE: info->s->state.state.records++; @@ -1313,7 +1320,7 @@ prototype_redo_exec_hook(CLR_END) info->s->state.changed|= STATE_CHANGED | STATE_NOT_ANALYZED | STATE_NOT_OPTIMIZED_KEYS | STATE_NOT_SORTED_PAGES; } - fprintf(tracef, " rows' count %lu\n", (ulong)info->s->state.state.records); + tprint(tracef, " rows' count %lu\n", (ulong)info->s->state.state.records); return 0; } @@ -1343,9 +1350,9 @@ prototype_undo_exec_hook(UNDO_ROW_INSERT) FILEID_STORE_SIZE); info->trn= 0; /* trn->undo_lsn is updated in an inwrite_hook when writing the CLR_END */ - fprintf(tracef, " rows' count %lu\n", (ulong)info->s->state.state.records); - fprintf(tracef, " undo_lsn now LSN (%lu,0x%lx)\n", - LSN_IN_PARTS(previous_undo_lsn)); + tprint(tracef, " rows' count %lu\n", (ulong)info->s->state.state.records); + tprint(tracef, " undo_lsn now LSN (%lu,0x%lx)\n", + LSN_IN_PARTS(previous_undo_lsn)); return error; } @@ -1368,7 +1375,7 @@ prototype_undo_exec_hook(UNDO_ROW_DELETE) log_record_buffer.str, NULL) != rec->record_length) { - fprintf(tracef, "Failed to read record\n"); + tprint(tracef, "Failed to read record\n"); return 1; } @@ -1385,9 +1392,9 @@ prototype_undo_exec_hook(UNDO_ROW_DELETE) (LSN_STORE_SIZE + FILEID_STORE_SIZE + PAGE_STORE_SIZE + DIRPOS_STORE_SIZE)); info->trn= 0; - fprintf(tracef, " rows' count %lu\n", (ulong)info->s->state.state.records); - fprintf(tracef, " undo_lsn now LSN (%lu,0x%lx)\n", - LSN_IN_PARTS(previous_undo_lsn)); + tprint(tracef, " rows' count %lu\n undo_lsn now LSN (%lu,0x%lx)\n", + (ulong)info->s->state.state.records, + LSN_IN_PARTS(previous_undo_lsn)); return error; } @@ -1410,7 +1417,7 @@ prototype_undo_exec_hook(UNDO_ROW_UPDATE) log_record_buffer.str, NULL) != rec->record_length) { - fprintf(tracef, "Failed to read record\n"); + tprint(tracef, "Failed to read record\n"); return 1; } @@ -1421,8 +1428,8 @@ prototype_undo_exec_hook(UNDO_ROW_UPDATE) rec->record_length - (LSN_STORE_SIZE + FILEID_STORE_SIZE)); info->trn= 0; - fprintf(tracef, " undo_lsn now LSN (%lu,0x%lx)\n", - LSN_IN_PARTS(previous_undo_lsn)); + tprint(tracef, " undo_lsn now LSN (%lu,0x%lx)\n", + LSN_IN_PARTS(previous_undo_lsn)); return error; } @@ -1464,8 +1471,8 @@ static int run_redo_phase(LSN lsn, my_bool apply) if (unlikely(lsn == LSN_IMPOSSIBLE || lsn == translog_get_horizon())) { - fprintf(tracef, "checkpoint address refers to the log end log or " - "log is empty, nothing to do.\n"); + tprint(tracef, "checkpoint address refers to the log end log or " + "log is empty, nothing to do.\n"); return 0; } @@ -1474,13 +1481,13 @@ static int run_redo_phase(LSN lsn, my_bool apply) /** @todo EOF should be detected */ if (len == RECHEADER_READ_ERROR) { - fprintf(tracef, "Failed to read header of the first record.\n"); + tprint(tracef, "Failed to read header of the first record.\n"); return 1; } struct st_translog_scanner_data scanner; if (translog_init_scanner(lsn, 1, &scanner)) { - fprintf(tracef, "Scanner init failed\n"); + tprint(tracef, "Scanner init failed\n"); return 1; } uint i; @@ -1506,7 +1513,7 @@ static int run_redo_phase(LSN lsn, my_bool apply) can happen if the transaction got a table write error, then unlocked tables thus wrote a COMMIT record. */ - fprintf(tracef, "\nDiscarding unfinished group before this record\n"); + tprint(tracef, "\nDiscarding unfinished group before this record\n"); ALERT_USER(); all_active_trans[sid].group_start_lsn= LSN_IMPOSSIBLE; } @@ -1516,19 +1523,19 @@ static int run_redo_phase(LSN lsn, my_bool apply) There is a complete group for this transaction, containing more than this event. */ - fprintf(tracef, " ends a group:\n"); + tprint(tracef, " ends a group:\n"); struct st_translog_scanner_data scanner2; TRANSLOG_HEADER_BUFFER rec2; len= translog_read_record_header(all_active_trans[sid].group_start_lsn, &rec2); if (len < 0) /* EOF or error */ { - fprintf(tracef, "Cannot find record where it should be\n"); + tprint(tracef, "Cannot find record where it should be\n"); return 1; } if (translog_init_scanner(rec2.lsn, 1, &scanner2)) { - fprintf(tracef, "Scanner2 init failed\n"); + tprint(tracef, "Scanner2 init failed\n"); return 1; } current_group_end_lsn= rec.lsn; @@ -1544,7 +1551,7 @@ static int run_redo_phase(LSN lsn, my_bool apply) len= translog_read_next_record_header(&scanner2, &rec2); if (len < 0) /* EOF or error */ { - fprintf(tracef, "Cannot find record where it should be\n"); + tprint(tracef, "Cannot find record where it should be\n"); return 1; } } @@ -1574,10 +1581,10 @@ static int run_redo_phase(LSN lsn, my_bool apply) switch (len) { case RECHEADER_READ_EOF: - fprintf(tracef, "EOF on the log\n"); + tprint(tracef, "EOF on the log\n"); break; case RECHEADER_READ_ERROR: - fprintf(tracef, "Error reading log\n"); + tprint(tracef, "Error reading log\n"); return 1; } break; @@ -1611,7 +1618,7 @@ static uint end_of_redo_phase(my_bool prepare_for_undo_phase) dirty_pages_pool= NULL; llstr(max_long_trid, llbuf); - fprintf(tracef, "Maximum transaction long id seen: %s\n", llbuf); + tprint(tracef, "Maximum transaction long id seen: %s\n", llbuf); if (prepare_for_undo_phase && trnman_init(max_long_trid)) return -1; @@ -1622,15 +1629,15 @@ static uint end_of_redo_phase(my_bool prepare_for_undo_phase) TRN *trn; if (gslsn != LSN_IMPOSSIBLE) { - fprintf(tracef, "Group at LSN (%lu,0x%lx) short_trid %u aborted\n", - LSN_IN_PARTS(gslsn), sid); + tprint(tracef, "Group at LSN (%lu,0x%lx) short_trid %u aborted\n", + LSN_IN_PARTS(gslsn), sid); ALERT_USER(); } if (all_active_trans[sid].undo_lsn != LSN_IMPOSSIBLE) { char llbuf[22]; llstr(long_trid, llbuf); - fprintf(tracef, "Transaction long_trid %s short_trid %u unfinished\n", + tprint(tracef, "Transaction long_trid %s short_trid %u unfinished\n", llbuf, sid); /* dummy_transaction_object serves only for DDLs */ DBUG_ASSERT(long_trid != 0); @@ -1676,14 +1683,22 @@ static uint end_of_redo_phase(my_bool prepare_for_undo_phase) } } - /* - We could take a checkpoint here, in case of a crash during the UNDO - phase. The drawback is that a page which got a REDO (thus, flushed - by this would-be checkpoint) is likely to have an UNDO executed on it - soon. And so, the flush was probably lost time. - So for now we prefer to do recovery with maximum speed and take a - checkpoint only at the end of the UNDO phase. - */ +#if 0 /* will be enabled soon */ + if (prepare_for_undo_phase) + { + /* + We take a checkpoint as it can save future recovery work if we crash + soon. But we don't flush pages, as UNDOs would change them again + probably. + */ + if (ma_checkpoint_init(FALSE)) + return -1; + int res= ma_checkpoint_execute(CHECKPOINT_INDIRECT, FALSE); + ma_checkpoint_end(); + if (res) + unfinished= -1; + } +#endif return unfinished; } @@ -1693,14 +1708,23 @@ static int run_undo_phase(uint unfinished) { if (unfinished > 0) { - fprintf(tracef, "%u transactions will be rolled back\n", unfinished); - for( ; unfinished-- ; ) + if (tracef != stdout) { + /** @todo RECOVERY BUG all prints to stderr should go to error log */ + fprintf(stderr, " 100%%; transactions to roll back:"); + } + tprint(tracef, "%u transactions will be rolled back\n", unfinished); + for( ; ; ) + { + if (tracef != stdout) + fprintf(stderr, " %u", unfinished); + if ((unfinished--) == 0) + break; char llbuf[22]; TRN *trn= trnman_get_any_trn(); DBUG_ASSERT(trn != NULL); llstr(trn->trid, llbuf); - fprintf(tracef, "Rolling back transaction of long id %s\n", llbuf); + tprint(tracef, "Rolling back transaction of long id %s\n", llbuf); /* Execute all undo entries */ while (trn->undo_lsn) @@ -1714,7 +1738,7 @@ static int run_undo_phase(uint unfinished) display_record_position(log_desc, &rec, 0); if (log_desc->record_execute_in_undo_phase(&rec, trn)) { - fprintf(tracef, "Got error when executing undo\n"); + tprint(tracef, "Got error when executing undo\n"); return 1; } } @@ -1766,7 +1790,7 @@ static MARIA_HA *get_MARIA_HA_from_REDO_record(const MARIA_HA *info; char llbuf[22]; - print_recovery_message(); + print_redo_phase_progress(rec->lsn); sid= fileid_korr(rec->header); page= page_korr(rec->header + FILEID_STORE_SIZE); /** @@ -1777,14 +1801,14 @@ static MARIA_HA *get_MARIA_HA_from_REDO_record(const i_am_about_index_file, none }. */ llstr(page, llbuf); - fprintf(tracef, " For page %s of table of short id %u", llbuf, sid); + tprint(tracef, " For page %s of table of short id %u", llbuf, sid); info= all_tables[sid].info; if (info == NULL) { - fprintf(tracef, ", table skipped, so skipping record\n"); + tprint(tracef, ", table skipped, so skipping record\n"); return NULL; } - fprintf(tracef, ", '%s'", info->s->open_file_name); + tprint(tracef, ", '%s'", info->s->open_file_name); if (cmp_translog_addr(rec->lsn, info->s->lsn_of_file_id) <= 0) { /* @@ -1796,9 +1820,9 @@ static MARIA_HA *get_MARIA_HA_from_REDO_record(const table was). */ DBUG_ASSERT(cmp_translog_addr(rec->lsn, checkpoint_start) < 0); - fprintf(tracef, ", table's LOGREC_FILE_ID has LSN (%lu,0x%lx) more recent" - " than record, skipping record", - LSN_IN_PARTS(info->s->lsn_of_file_id)); + tprint(tracef, ", table's LOGREC_FILE_ID has LSN (%lu,0x%lx) more recent" + " than record, skipping record", + LSN_IN_PARTS(info->s->lsn_of_file_id)); return NULL; } /* detect if an open instance of a dropped table (internal bug) */ @@ -1817,7 +1841,7 @@ static MARIA_HA *get_MARIA_HA_from_REDO_record(const if ((dirty_page == NULL) || cmp_translog_addr(rec->lsn, dirty_page->rec_lsn) < 0) { - fprintf(tracef, ", ignoring because of dirty_pages list\n"); + tprint(tracef, ", ignoring because of dirty_pages list\n"); return NULL; } } @@ -1826,7 +1850,7 @@ static MARIA_HA *get_MARIA_HA_from_REDO_record(const So we are going to read the page, and if its LSN is older than the record's we will modify the page */ - fprintf(tracef, ", applying record\n"); + tprint(tracef, ", applying record\n"); _ma_writeinfo(info, WRITEINFO_UPDATE_KEYFILE); /* to flush state on close */ return info; } @@ -1838,26 +1862,25 @@ static MARIA_HA *get_MARIA_HA_from_UNDO_record(const uint16 sid; MARIA_HA *info; - print_recovery_message(); sid= fileid_korr(rec->header + LSN_STORE_SIZE); - fprintf(tracef, " For table of short id %u", sid); + tprint(tracef, " For table of short id %u", sid); info= all_tables[sid].info; if (info == NULL) { - fprintf(tracef, ", table skipped, so skipping record\n"); + tprint(tracef, ", table skipped, so skipping record\n"); return NULL; } - fprintf(tracef, ", '%s'", info->s->open_file_name); + tprint(tracef, ", '%s'", info->s->open_file_name); if (cmp_translog_addr(rec->lsn, info->s->lsn_of_file_id) <= 0) { - fprintf(tracef, ", table's LOGREC_FILE_ID has LSN (%lu,0x%lx) more recent" - " than record, skipping record", - LSN_IN_PARTS(info->s->lsn_of_file_id)); + tprint(tracef, ", table's LOGREC_FILE_ID has LSN (%lu,0x%lx) more recent" + " than record, skipping record", + LSN_IN_PARTS(info->s->lsn_of_file_id)); return NULL; } DBUG_ASSERT(info->s->last_version != 0); _ma_writeinfo(info, WRITEINFO_UPDATE_KEYFILE); /* to flush state on close */ - fprintf(tracef, ", applying record\n"); + tprint(tracef, ", applying record\n"); return info; } @@ -1878,13 +1901,13 @@ static LSN parse_checkpoint_record(LSN lsn) uint i; TRANSLOG_HEADER_BUFFER rec; - fprintf(tracef, "Loading data from checkpoint record at LSN (%lu,0x%lx)\n", - LSN_IN_PARTS(lsn)); + tprint(tracef, "Loading data from checkpoint record at LSN (%lu,0x%lx)\n", + LSN_IN_PARTS(lsn)); int len= translog_read_record_header(lsn, &rec); if (len == RECHEADER_READ_ERROR) { - fprintf(tracef, "Cannot find checkpoint record where it should be\n"); + tprint(tracef, "Cannot find checkpoint record where it should be\n"); return LSN_IMPOSSIBLE; } @@ -1894,7 +1917,7 @@ static LSN parse_checkpoint_record(LSN lsn) log_record_buffer.str, NULL) != rec.record_length) { - fprintf(tracef, "Failed to read record\n"); + tprint(tracef, "Failed to read record\n"); return LSN_IMPOSSIBLE; } @@ -1905,7 +1928,7 @@ static LSN parse_checkpoint_record(LSN lsn) /* transactions */ uint nb_active_transactions= uint2korr(ptr); ptr+= 2; - fprintf(tracef, "%u active transactions\n", nb_active_transactions); + tprint(tracef, "%u active transactions\n", nb_active_transactions); LSN minimum_rec_lsn_of_active_transactions= lsn_korr(ptr); ptr+= LSN_STORE_SIZE; @@ -1930,15 +1953,15 @@ static LSN parse_checkpoint_record(LSN lsn) } uint nb_committed_transactions= uint4korr(ptr); ptr+= 4; - fprintf(tracef, "%lu committed transactions\n", - (ulong)nb_committed_transactions); + tprint(tracef, "%lu committed transactions\n", + (ulong)nb_committed_transactions); /* no purging => committed transactions are not important */ ptr+= (6 + LSN_STORE_SIZE) * nb_committed_transactions; /* tables */ uint nb_tables= uint4korr(ptr); ptr+= 4; - fprintf(tracef, "%u open tables\n", nb_tables); + tprint(tracef, "%u open tables\n", nb_tables); for (i= 0; i< nb_tables; i++) { char name[FN_REFLEN]; @@ -1961,7 +1984,7 @@ static LSN parse_checkpoint_record(LSN lsn) /* dirty pages */ uint nb_dirty_pages= uint4korr(ptr); ptr+= 4; - fprintf(tracef, "%u dirty pages\n", nb_dirty_pages); + tprint(tracef, "%u dirty pages\n", nb_dirty_pages); if (hash_init(&all_dirty_pages, &my_charset_bin, nb_dirty_pages, offsetof(struct st_dirty_page, file_and_page_id), sizeof(((struct st_dirty_page *)NULL)->file_and_page_id), @@ -1994,7 +2017,7 @@ static LSN parse_checkpoint_record(LSN lsn) */ if (ptr != (log_record_buffer.str + log_record_buffer.length)) { - fprintf(tracef, "checkpoint record corrupted\n"); + tprint(tracef, "checkpoint record corrupted\n"); return LSN_IMPOSSIBLE; } set_if_smaller(checkpoint_start, minimum_rec_lsn_of_dirty_pages); @@ -2020,7 +2043,13 @@ static int close_all_tables(void) pthread_mutex_lock(&THR_LOCK_maria); if (maria_open_list == NULL) goto end; - fprintf(tracef, "Closing all tables\n"); + tprint(tracef, "Closing all tables\n"); + if (tracef != stdout && redo_phase_message_printed) + { + /** @todo RECOVERY BUG all prints to stderr should go to error log */ + fprintf(stderr, "; flushing tables"); + } + /* Since the end of end_of_redo_phase(), we may have written new records (if UNDO phase ran) and thus the state is newer than at @@ -2041,6 +2070,41 @@ end: return error; } +static void print_redo_phase_progress(TRANSLOG_ADDRESS addr) +{ + static int end_logno= FILENO_IMPOSSIBLE, end_offset, percentage_printed= 0; + static ulonglong initial_remainder= -1; + if (tracef == stdout) + return; + if (!redo_phase_message_printed) + { + /** @todo RECOVERY BUG all prints to stderr should go to error log */ + fprintf(stderr, "Maria engine: starting recovery; recovered pages: 0%%"); + redo_phase_message_printed= TRUE; + } + if (end_logno == FILENO_IMPOSSIBLE) + { + LSN end_addr= translog_get_horizon(); + end_logno= LSN_FILE_NO(end_addr); + end_offset= LSN_OFFSET(end_addr); + } + int cur_logno= LSN_FILE_NO(addr); + int cur_offset= LSN_OFFSET(addr); + ulonglong remainder; + remainder= (cur_logno == end_logno) ? (end_offset - cur_offset) : + (TRANSLOG_FILE_SIZE - cur_offset + + max(end_logno - cur_logno - 1, 0) * TRANSLOG_FILE_SIZE + end_offset); + if (initial_remainder == (ulonglong)(-1)) + initial_remainder= remainder; + int percentage_done= + (initial_remainder - remainder) * ULL(100) / initial_remainder; + if ((percentage_done - percentage_printed) >= 10) + { + percentage_printed= percentage_done; + fprintf(stderr, " %d%%", percentage_done); + } +} + #ifdef MARIA_EXTERNAL_LOCKING #error Maria's Checkpoint and Recovery are really not ready for it #endif diff --git a/storage/maria/ma_test_recovery b/storage/maria/ma_test_recovery index 23f65c7e764..7c45af1e206 100755 --- a/storage/maria/ma_test_recovery +++ b/storage/maria/ma_test_recovery @@ -131,7 +131,7 @@ do for test_undo in 1 2 3 do # first iteration tests rollback of insert, second tests rollback of delete - set -- "ma_test1 $silent -M -T -c -N $blobs" "--testflag=1" "--testflag=2 --test-undo=" "ma_test1 $silent -M -T -c -N --debug=d:t:i:o,/tmp/ma_test1.trace $blobs" "--testflag=3" "--testflag=4 --test-undo=" "ma_test1 $silent -M -T -c -N --debug=d:t:i:o,/tmp/ma_test1.trace $blobs" "--testflag=2" "--testflag=3 --test-undo=" "ma_test2 $silent -L -K -W -P -M -T -c $blobs" "-t1" "-t2 -u" + set -- "ma_test1 $silent -M -T -c -N $blobs" "--testflag=1" "--testflag=2 --test-undo=" "ma_test1 $silent -M -T -c -N $blobs" "--testflag=3" "--testflag=4 --test-undo=" "ma_test1 $silent -M -T -c -N $blobs" "--testflag=2" "--testflag=3 --test-undo=" "ma_test2 $silent -L -K -W -P -M -T -c $blobs" "-t1" "-t2 -u" # -N (create NULL fields) is needed because --test-undo adds it anyway while [ $# != 0 ] do diff --git a/storage/maria/ma_test_recovery.expected b/storage/maria/ma_test_recovery.expected index 253471af802..926943b11b3 100644 --- a/storage/maria/ma_test_recovery.expected +++ b/storage/maria/ma_test_recovery.expected @@ -125,9 +125,9 @@ Differences in maria_chk -dvv, recovery not yet perfect ! --- > 1 2 6 unique number NULL 0 8192 ========DIFF END======= -TEST WITH ma_test1 -s -M -T -c -N --debug=d:t:i:o,/tmp/ma_test1.trace --testflag=3 (commit at end) +TEST WITH ma_test1 -s -M -T -c -N --testflag=3 (commit at end) Terminating after updates -TEST WITH ma_test1 -s -M -T -c -N --debug=d:t:i:o,/tmp/ma_test1.trace --testflag=4 --test-undo=1 (additional aborted work) +TEST WITH ma_test1 -s -M -T -c -N --testflag=4 --test-undo=1 (additional aborted work) Terminating after deletes Dying on request without maria_commit()/maria_close() applying log @@ -150,9 +150,9 @@ Differences in maria_chk -dvv, recovery not yet perfect ! --- > 1 2 6 unique number NULL 0 8192 ========DIFF END======= -TEST WITH ma_test1 -s -M -T -c -N --debug=d:t:i:o,/tmp/ma_test1.trace --testflag=2 (commit at end) +TEST WITH ma_test1 -s -M -T -c -N --testflag=2 (commit at end) Terminating after inserts -TEST WITH ma_test1 -s -M -T -c -N --debug=d:t:i:o,/tmp/ma_test1.trace --testflag=3 --test-undo=1 (additional aborted work) +TEST WITH ma_test1 -s -M -T -c -N --testflag=3 --test-undo=1 (additional aborted work) Terminating after updates Dying on request without maria_commit()/maria_close() applying log @@ -300,9 +300,9 @@ Differences in maria_chk -dvv, recovery not yet perfect ! --- > 1 2 6 unique number NULL 0 8192 ========DIFF END======= -TEST WITH ma_test1 -s -M -T -c -N --debug=d:t:i:o,/tmp/ma_test1.trace --testflag=3 (commit at end) +TEST WITH ma_test1 -s -M -T -c -N --testflag=3 (commit at end) Terminating after updates -TEST WITH ma_test1 -s -M -T -c -N --debug=d:t:i:o,/tmp/ma_test1.trace --testflag=4 --test-undo=2 (additional aborted work) +TEST WITH ma_test1 -s -M -T -c -N --testflag=4 --test-undo=2 (additional aborted work) Terminating after deletes Dying on request without maria_commit()/maria_close() applying log @@ -325,9 +325,9 @@ Differences in maria_chk -dvv, recovery not yet perfect ! --- > 1 2 6 unique number NULL 0 8192 ========DIFF END======= -TEST WITH ma_test1 -s -M -T -c -N --debug=d:t:i:o,/tmp/ma_test1.trace --testflag=2 (commit at end) +TEST WITH ma_test1 -s -M -T -c -N --testflag=2 (commit at end) Terminating after inserts -TEST WITH ma_test1 -s -M -T -c -N --debug=d:t:i:o,/tmp/ma_test1.trace --testflag=3 --test-undo=2 (additional aborted work) +TEST WITH ma_test1 -s -M -T -c -N --testflag=3 --test-undo=2 (additional aborted work) Terminating after updates Dying on request without maria_commit()/maria_close() applying log @@ -475,9 +475,9 @@ Differences in maria_chk -dvv, recovery not yet perfect ! --- > 1 2 6 unique number NULL 0 8192 ========DIFF END======= -TEST WITH ma_test1 -s -M -T -c -N --debug=d:t:i:o,/tmp/ma_test1.trace --testflag=3 (commit at end) +TEST WITH ma_test1 -s -M -T -c -N --testflag=3 (commit at end) Terminating after updates -TEST WITH ma_test1 -s -M -T -c -N --debug=d:t:i:o,/tmp/ma_test1.trace --testflag=4 --test-undo=3 (additional aborted work) +TEST WITH ma_test1 -s -M -T -c -N --testflag=4 --test-undo=3 (additional aborted work) Terminating after deletes Dying on request without maria_commit()/maria_close() applying log @@ -500,9 +500,9 @@ Differences in maria_chk -dvv, recovery not yet perfect ! --- > 1 2 6 unique number NULL 0 8192 ========DIFF END======= -TEST WITH ma_test1 -s -M -T -c -N --debug=d:t:i:o,/tmp/ma_test1.trace --testflag=2 (commit at end) +TEST WITH ma_test1 -s -M -T -c -N --testflag=2 (commit at end) Terminating after inserts -TEST WITH ma_test1 -s -M -T -c -N --debug=d:t:i:o,/tmp/ma_test1.trace --testflag=3 --test-undo=3 (additional aborted work) +TEST WITH ma_test1 -s -M -T -c -N --testflag=3 --test-undo=3 (additional aborted work) Terminating after updates Dying on request without maria_commit()/maria_close() applying log @@ -650,9 +650,9 @@ Differences in maria_chk -dvv, recovery not yet perfect ! --- > 1 2 6 unique varchar BLOB NULL 0 8192 ========DIFF END======= -TEST WITH ma_test1 -s -M -T -c -N --debug=d:t:i:o,/tmp/ma_test1.trace -b --testflag=3 (commit at end) +TEST WITH ma_test1 -s -M -T -c -N -b --testflag=3 (commit at end) Terminating after updates -TEST WITH ma_test1 -s -M -T -c -N --debug=d:t:i:o,/tmp/ma_test1.trace -b --testflag=4 --test-undo=1 (additional aborted work) +TEST WITH ma_test1 -s -M -T -c -N -b --testflag=4 --test-undo=1 (additional aborted work) Terminating after deletes Dying on request without maria_commit()/maria_close() applying log @@ -675,9 +675,9 @@ Differences in maria_chk -dvv, recovery not yet perfect ! --- > 1 2 6 unique varchar BLOB NULL 0 8192 ========DIFF END======= -TEST WITH ma_test1 -s -M -T -c -N --debug=d:t:i:o,/tmp/ma_test1.trace -b --testflag=2 (commit at end) +TEST WITH ma_test1 -s -M -T -c -N -b --testflag=2 (commit at end) Terminating after inserts -TEST WITH ma_test1 -s -M -T -c -N --debug=d:t:i:o,/tmp/ma_test1.trace -b --testflag=3 --test-undo=1 (additional aborted work) +TEST WITH ma_test1 -s -M -T -c -N -b --testflag=3 --test-undo=1 (additional aborted work) Terminating after updates Dying on request without maria_commit()/maria_close() applying log @@ -825,9 +825,9 @@ Differences in maria_chk -dvv, recovery not yet perfect ! --- > 1 2 6 unique varchar BLOB NULL 0 8192 ========DIFF END======= -TEST WITH ma_test1 -s -M -T -c -N --debug=d:t:i:o,/tmp/ma_test1.trace -b --testflag=3 (commit at end) +TEST WITH ma_test1 -s -M -T -c -N -b --testflag=3 (commit at end) Terminating after updates -TEST WITH ma_test1 -s -M -T -c -N --debug=d:t:i:o,/tmp/ma_test1.trace -b --testflag=4 --test-undo=2 (additional aborted work) +TEST WITH ma_test1 -s -M -T -c -N -b --testflag=4 --test-undo=2 (additional aborted work) Terminating after deletes Dying on request without maria_commit()/maria_close() applying log @@ -850,9 +850,9 @@ Differences in maria_chk -dvv, recovery not yet perfect ! --- > 1 2 6 unique varchar BLOB NULL 0 8192 ========DIFF END======= -TEST WITH ma_test1 -s -M -T -c -N --debug=d:t:i:o,/tmp/ma_test1.trace -b --testflag=2 (commit at end) +TEST WITH ma_test1 -s -M -T -c -N -b --testflag=2 (commit at end) Terminating after inserts -TEST WITH ma_test1 -s -M -T -c -N --debug=d:t:i:o,/tmp/ma_test1.trace -b --testflag=3 --test-undo=2 (additional aborted work) +TEST WITH ma_test1 -s -M -T -c -N -b --testflag=3 --test-undo=2 (additional aborted work) Terminating after updates Dying on request without maria_commit()/maria_close() applying log @@ -1000,9 +1000,9 @@ Differences in maria_chk -dvv, recovery not yet perfect ! --- > 1 2 6 unique varchar BLOB NULL 0 8192 ========DIFF END======= -TEST WITH ma_test1 -s -M -T -c -N --debug=d:t:i:o,/tmp/ma_test1.trace -b --testflag=3 (commit at end) +TEST WITH ma_test1 -s -M -T -c -N -b --testflag=3 (commit at end) Terminating after updates -TEST WITH ma_test1 -s -M -T -c -N --debug=d:t:i:o,/tmp/ma_test1.trace -b --testflag=4 --test-undo=3 (additional aborted work) +TEST WITH ma_test1 -s -M -T -c -N -b --testflag=4 --test-undo=3 (additional aborted work) Terminating after deletes Dying on request without maria_commit()/maria_close() applying log @@ -1025,9 +1025,9 @@ Differences in maria_chk -dvv, recovery not yet perfect ! --- > 1 2 6 unique varchar BLOB NULL 0 8192 ========DIFF END======= -TEST WITH ma_test1 -s -M -T -c -N --debug=d:t:i:o,/tmp/ma_test1.trace -b --testflag=2 (commit at end) +TEST WITH ma_test1 -s -M -T -c -N -b --testflag=2 (commit at end) Terminating after inserts -TEST WITH ma_test1 -s -M -T -c -N --debug=d:t:i:o,/tmp/ma_test1.trace -b --testflag=3 --test-undo=3 (additional aborted work) +TEST WITH ma_test1 -s -M -T -c -N -b --testflag=3 --test-undo=3 (additional aborted work) Terminating after updates Dying on request without maria_commit()/maria_close() applying log -- cgit v1.2.1 From 301a31fddfd5f08e3e38915a9c74517116028301 Mon Sep 17 00:00:00 2001 From: unknown Date: Tue, 25 Sep 2007 15:52:01 +0200 Subject: WL#3072 Maria recovery Minor change: decouple the writing of a recovery trace from compilation with --with-debug; that writing happens if and only if EXTRA_DEBUG. So one can enable (by defining EXTRA_DEBUG) a recovery trace in a non-debug build. storage/maria/ma_recovery.c: Small optimization. Decouple recovery trace from --with-debug. --- storage/maria/ma_recovery.c | 7 +++++-- 1 file changed, 5 insertions(+), 2 deletions(-) diff --git a/storage/maria/ma_recovery.c b/storage/maria/ma_recovery.c index e740e334b5f..2f9390b0036 100644 --- a/storage/maria/ma_recovery.c +++ b/storage/maria/ma_recovery.c @@ -123,13 +123,16 @@ static my_bool redo_phase_message_printed; /** @brief Prints to a trace file if it is not NULL */ void tprint(FILE *trace_file, const char *format, ...) ATTRIBUTE_FORMAT(printf, 2, 3); -void tprint(FILE *trace_file, const char *format, ...) +void tprint(FILE *trace_file __attribute__ ((unused)), + const char *format __attribute__ ((unused)), ...) { +#ifdef EXTRA_DEBUG va_list args; va_start(args, format); if (trace_file != NULL) vfprintf(trace_file, format, args); va_end(args); +#endif } #define ALERT_USER() DBUG_ASSERT(0) @@ -157,7 +160,7 @@ int maria_recover(void) DBUG_ASSERT(!maria_in_recovery); maria_in_recovery= TRUE; -#if !defined(DBUG_OFF) && defined(EXTRA_DEBUG) +#ifdef EXTRA_DEBUG trace_file= fopen("maria_recovery.trace", "w"); #else trace_file= NULL; /* no trace file for being fast */ -- cgit v1.2.1 From c9c58163a936ccf2882aae1230018117744546b0 Mon Sep 17 00:00:00 2001 From: unknown Date: Thu, 27 Sep 2007 14:18:28 +0300 Subject: Remove SAFE_MODE for opt_range as it disables UPDATE to use keys REDO optimization (Bascily avoid moving blocks from/to pagecache) More command line arguments to maria_read_log Fixed recovery bug when recreating table sql/opt_range.cc: Remove SAFE_MODE for opt_range as it disables UPDATE to use keys storage/maria/ma_blockrec.c: REDO optimization Use new interface for pagecache_reads to avoid copying page buffers storage/maria/ma_loghandler.c: Patch from Sanja: - Added new parameter to translog_get_page to use direct links to pagecache - Changed scanner to be able to use direct links This avoids a lot of calls to bmove512() in page cache. storage/maria/ma_loghandler.h: Added direct link to pagecache objects storage/maria/ma_open.c: Added const to parameter Added missing braces storage/maria/ma_pagecache.c: From Sanja: - Added direct links to pagecache (from pagecache_read()) Dirrect link means that on pagecache_read we get back a pointer to the pagecache buffer From Monty: - Fixed arguments to init_page_cache to handle big page caches - Fixed compiler warnings - Replaced PAGECACHE_PAGE_LINK with PAGECACHE_BLOCK_LINK * to catch errors storage/maria/ma_pagecache.h: Changed block numbers from int to long to be able to handle big page caches Changed some PAGECACHE_PAGE_LINK to PAGECACHE_BLOCK_LINK storage/maria/ma_recovery.c: Fixed recovery bug when recreating table (table was kept open) Moved some variables to function start (portability) Added space to some print messages storage/maria/maria_chk.c: key_buffer_size -> page_buffer_size storage/maria/maria_def.h: Changed default page_buffer_size to 10M storage/maria/maria_read_log.c: Added more startup options: --version --undo (apply undo) --page_cache_size (to run with big cache sizes) --silent (to not get any output from --apply) storage/maria/unittest/ma_control_file-t.c: Fixed compiler warning storage/maria/unittest/ma_test_loghandler-t.c: Added new argument to translog_init_scanner() storage/maria/unittest/ma_test_loghandler_multigroup-t.c: Added new argument to translog_init_scanner() storage/maria/unittest/ma_test_loghandler_multithread-t.c: Added new argument to translog_init_scanner() --- sql/opt_range.cc | 3 - storage/maria/ma_blockrec.c | 78 +++++--- storage/maria/ma_loghandler.c | 200 ++++++++++++++++----- storage/maria/ma_loghandler.h | 8 +- storage/maria/ma_open.c | 4 +- storage/maria/ma_pagecache.c | 123 ++++++++----- storage/maria/ma_pagecache.h | 42 ++--- storage/maria/ma_recovery.c | 77 ++++++-- storage/maria/maria_chk.c | 4 +- storage/maria/maria_def.h | 4 +- storage/maria/maria_read_log.c | 47 +++-- storage/maria/unittest/ma_control_file-t.c | 2 +- storage/maria/unittest/ma_test_loghandler-t.c | 2 +- .../unittest/ma_test_loghandler_multigroup-t.c | 2 +- .../unittest/ma_test_loghandler_multithread-t.c | 2 +- 15 files changed, 415 insertions(+), 183 deletions(-) diff --git a/sql/opt_range.cc b/sql/opt_range.cc index 3ee7e8ee813..fbb1a2be487 100644 --- a/sql/opt_range.cc +++ b/sql/opt_range.cc @@ -2130,9 +2130,6 @@ int SQL_SELECT::test_quick_select(THD *thd, key_map keys_to_use, quick=0; needed_reg.clear_all(); quick_keys.clear_all(); - if ((specialflag & SPECIAL_SAFE_MODE) && ! force_quick_range || - !limit) - DBUG_RETURN(0); /* purecov: inspected */ if (keys_to_use.is_clear_all()) DBUG_RETURN(0); records= head->file->stats.records; diff --git a/storage/maria/ma_blockrec.c b/storage/maria/ma_blockrec.c index b12035c9cfa..3ec18d229c2 100644 --- a/storage/maria/ma_blockrec.c +++ b/storage/maria/ma_blockrec.c @@ -4609,6 +4609,9 @@ uint _ma_apply_redo_insert_row_head_or_tail(MARIA_HA *info, LSN lsn, uint block_size= share->block_size; uint rec_offset; uchar *buff= info->keyread_buff, *dir; + MARIA_PINNED_PAGE page_link; + enum pagecache_page_lock unlock_method; + enum pagecache_page_pin unpin_method; DBUG_ENTER("_ma_apply_redo_insert_row_head_or_tail"); info->keyread_buff_used= 1; @@ -4635,26 +4638,31 @@ uint _ma_apply_redo_insert_row_head_or_tail(MARIA_HA *info, LSN lsn, empty_space= (block_size - PAGE_OVERHEAD_SIZE); rec_offset= PAGE_HEADER_SIZE; dir= buff+ block_size - PAGE_SUFFIX_SIZE - DIR_ENTRY_SIZE; + unlock_method= PAGECACHE_LOCK_LEFT_UNLOCKED; + unpin_method= PAGECACHE_PIN_LEFT_UNPINNED; } else { uint max_entry; - if (!(buff= pagecache_read(share->pagecache, - &info->dfile, - page, 0, - buff, PAGECACHE_PLAIN_PAGE, - PAGECACHE_LOCK_LEFT_UNLOCKED, 0))) + if (!(buff= pagecache_read(share->pagecache, &info->dfile, + page, 0, 0, + PAGECACHE_PLAIN_PAGE, PAGECACHE_LOCK_WRITE, + &page_link.link))) DBUG_RETURN(my_errno); - if (lsn_korr(buff) >= lsn) + if (lsn_korr(buff) >= lsn) /* Test if already applied */ { - /* Already applied */ - + pagecache_unlock_by_link(share->pagecache, page_link.link, + PAGECACHE_LOCK_WRITE_UNLOCK, + PAGECACHE_UNPIN, LSN_IMPOSSIBLE, + LSN_IMPOSSIBLE); /* Fix bitmap, just in case */ empty_space= uint2korr(buff + EMPTY_SPACE_OFFSET); if (_ma_bitmap_set(info, page, page_type == HEAD_PAGE, empty_space)) DBUG_RETURN(my_errno); DBUG_RETURN(0); } + unlock_method= PAGECACHE_LOCK_WRITE_UNLOCK; + unpin_method= PAGECACHE_UNPIN; max_entry= (uint) ((uchar*) buff)[DIR_COUNT_OFFSET]; if (((buff[PAGE_TYPE_OFFSET] & PAGE_TYPE_MASK) != page_type)) @@ -4725,8 +4733,7 @@ uint _ma_apply_redo_insert_row_head_or_tail(MARIA_HA *info, LSN lsn, if (pagecache_write(share->pagecache, &info->dfile, page, 0, buff, PAGECACHE_PLAIN_PAGE, - PAGECACHE_LOCK_LEFT_UNLOCKED, - PAGECACHE_PIN_LEFT_UNPINNED, + unlock_method, unpin_method, PAGECACHE_WRITE_DELAY, 0)) DBUG_RETURN(my_errno); @@ -4747,6 +4754,11 @@ uint _ma_apply_redo_insert_row_head_or_tail(MARIA_HA *info, LSN lsn, DBUG_RETURN(0); err: + if (unlock_method == PAGECACHE_LOCK_WRITE_UNLOCK) + pagecache_unlock_by_link(share->pagecache, page_link.link, + PAGECACHE_LOCK_WRITE_UNLOCK, + PAGECACHE_UNPIN, LSN_IMPOSSIBLE, + LSN_IMPOSSIBLE); DBUG_RETURN(HA_ERR_WRONG_IN_RECORD); } @@ -4778,6 +4790,8 @@ uint _ma_apply_redo_purge_row_head_or_tail(MARIA_HA *info, LSN lsn, uint rownr, empty_space; uint block_size= share->block_size; uchar *buff= info->keyread_buff; + int result; + MARIA_PINNED_PAGE page_link; DBUG_ENTER("_ma_apply_redo_purge_row_head_or_tail"); page= page_korr(header); @@ -4788,11 +4802,10 @@ uint _ma_apply_redo_purge_row_head_or_tail(MARIA_HA *info, LSN lsn, info->keyread_buff_used= 1; - if (!(buff= pagecache_read(share->pagecache, - &info->dfile, - page, 0, - buff, PAGECACHE_PLAIN_PAGE, - PAGECACHE_LOCK_LEFT_UNLOCKED, 0))) + if (!(buff= pagecache_read(share->pagecache, &info->dfile, + page, 0, 0, + PAGECACHE_PLAIN_PAGE, PAGECACHE_LOCK_WRITE, + &page_link.link))) DBUG_RETURN(my_errno); if (lsn_korr(buff) >= lsn) @@ -4802,6 +4815,11 @@ uint _ma_apply_redo_purge_row_head_or_tail(MARIA_HA *info, LSN lsn, Note that in case the page is not anymore a head or tail page a future redo will fix the bitmap. */ + pagecache_unlock_by_link(share->pagecache, page_link.link, + PAGECACHE_LOCK_WRITE_UNLOCK, + PAGECACHE_UNPIN, LSN_IMPOSSIBLE, + LSN_IMPOSSIBLE); + if ((buff[PAGE_TYPE_OFFSET] & PAGE_TYPE_MASK) == page_type) { empty_space= uint2korr(buff+EMPTY_SPACE_OFFSET); @@ -4815,22 +4833,30 @@ uint _ma_apply_redo_purge_row_head_or_tail(MARIA_HA *info, LSN lsn, DBUG_ASSERT((buff[PAGE_TYPE_OFFSET] & PAGE_TYPE_MASK) == (uchar) page_type); if (delete_dir_entry(buff, block_size, rownr, &empty_space) < 0) - DBUG_RETURN(HA_ERR_WRONG_IN_RECORD); + goto err; lsn_store(buff, lsn); + result= 0; if (pagecache_write(share->pagecache, &info->dfile, page, 0, buff, PAGECACHE_PLAIN_PAGE, - PAGECACHE_LOCK_LEFT_UNLOCKED, - PAGECACHE_PIN_LEFT_UNPINNED, + PAGECACHE_LOCK_WRITE_UNLOCK, PAGECACHE_UNPIN, PAGECACHE_WRITE_DELAY, 0)) - DBUG_RETURN(my_errno); + result= my_errno; /* This will work even if the page was marked as UNALLOCATED_PAGE */ if (_ma_bitmap_set(info, page, page_type == HEAD_PAGE, empty_space)) - DBUG_RETURN(my_errno); + result= my_errno; + + DBUG_RETURN(result); + +err: + pagecache_unlock_by_link(share->pagecache, page_link.link, + PAGECACHE_LOCK_WRITE_UNLOCK, + PAGECACHE_UNPIN, LSN_IMPOSSIBLE, + LSN_IMPOSSIBLE); + DBUG_RETURN(HA_ERR_WRONG_IN_RECORD); - DBUG_RETURN(0); } @@ -4872,16 +4898,21 @@ uint _ma_apply_redo_purge_blocks(MARIA_HA *info, for (i= 0; i < page_range ; i++) { + MARIA_PINNED_PAGE page_link; if (!(buff= pagecache_read(share->pagecache, &info->dfile, page+i, 0, buff, PAGECACHE_PLAIN_PAGE, - PAGECACHE_LOCK_LEFT_UNLOCKED, 0))) + PAGECACHE_LOCK_WRITE, &page_link.link))) DBUG_RETURN(my_errno); if (lsn_korr(buff) >= lsn) { /* Already applied */ + pagecache_unlock_by_link(share->pagecache, page_link.link, + PAGECACHE_LOCK_WRITE_UNLOCK, + PAGECACHE_UNPIN, LSN_IMPOSSIBLE, + LSN_IMPOSSIBLE); continue; } buff[PAGE_TYPE_OFFSET]= UNALLOCATED_PAGE; @@ -4889,8 +4920,7 @@ uint _ma_apply_redo_purge_blocks(MARIA_HA *info, if (pagecache_write(share->pagecache, &info->dfile, page+i, 0, buff, PAGECACHE_PLAIN_PAGE, - PAGECACHE_LOCK_LEFT_UNLOCKED, - PAGECACHE_PIN_LEFT_UNPINNED, + PAGECACHE_LOCK_WRITE_UNLOCK, PAGECACHE_UNPIN, PAGECACHE_WRITE_DELAY, 0)) DBUG_RETURN(my_errno); } diff --git a/storage/maria/ma_loghandler.c b/storage/maria/ma_loghandler.c index f3c90ceb1f5..e30c8c6627f 100644 --- a/storage/maria/ma_loghandler.c +++ b/storage/maria/ma_loghandler.c @@ -2296,21 +2296,21 @@ my_bool translog_unlock() } -/* - Get log page by file number and offset of the beginning of the page +/** + @brief Get log page by file number and offset of the beginning of the page - SYNOPSIS - translog_get_page() - data validator data, which contains the page address - buffer buffer for page placing + @param data validator data, which contains the page address + @param buffer buffer for page placing (might not be used in some cache implementations) + @param direct_link if it is not NULL then caller can accept direct + link to the page cache - RETURN - NULL - Error - # pointer to the page cache which should be used to read this page + @retval NULL Error + @retval # pointer to the page cache which should be used to read this page */ -static uchar *translog_get_page(TRANSLOG_VALIDATOR_DATA *data, uchar *buffer) +static uchar *translog_get_page(TRANSLOG_VALIDATOR_DATA *data, uchar *buffer, + PAGECACHE_PAGE_LINK *direct_link) { TRANSLOG_ADDRESS addr= *(data->addr), in_buffers; uint cache_index; @@ -2324,6 +2324,9 @@ static uchar *translog_get_page(TRANSLOG_VALIDATOR_DATA *data, uchar *buffer) /* it is really page address */ DBUG_ASSERT(LSN_OFFSET(addr) % TRANSLOG_PAGE_SIZE == 0); + if (direct_link) + *direct_link= NULL; + in_buffers= translog_only_in_buffers(); DBUG_PRINT("info", ("in_buffers: (%lu,0x%lx)", LSN_IN_PARTS(in_buffers))); @@ -2336,7 +2339,9 @@ static uchar *translog_get_page(TRANSLOG_VALIDATOR_DATA *data, uchar *buffer) if (cmp_translog_addr(addr, in_buffers) >= 0) { uint16 buffer_no= log_descriptor.bc.buffer_no; +#ifndef DBUG_OFF uint16 buffer_start= buffer_no; +#endif struct st_translog_buffer *buffer_unlock= log_descriptor.bc.buffer; struct st_translog_buffer *curr_buffer= log_descriptor.bc.buffer; for (;;) @@ -2437,13 +2442,23 @@ static uchar *translog_get_page(TRANSLOG_VALIDATOR_DATA *data, uchar *buffer) } file.file= log_descriptor.log_file_num[cache_index]; - buffer= (uchar*) - pagecache_valid_read(log_descriptor.pagecache, &file, - LSN_OFFSET(addr) / TRANSLOG_PAGE_SIZE, - 3, (char*) buffer, - PAGECACHE_PLAIN_PAGE, - PAGECACHE_LOCK_LEFT_UNLOCKED, 0, - &translog_page_validator, (uchar*) data); + buffer= + (uchar*) (direct_link ? + pagecache_valid_read(log_descriptor.pagecache, &file, + LSN_OFFSET(addr) / TRANSLOG_PAGE_SIZE, + 3, NULL, + PAGECACHE_PLAIN_PAGE, + PAGECACHE_LOCK_READ, direct_link, + &translog_page_validator, (uchar*) data) : + pagecache_valid_read(log_descriptor.pagecache, &file, + LSN_OFFSET(addr) / TRANSLOG_PAGE_SIZE, + 3, (char*) buffer, + PAGECACHE_PLAIN_PAGE, + PAGECACHE_LOCK_LEFT_UNLOCKED, direct_link, + &translog_page_validator, (uchar*) data)); + DBUG_PRINT("info", ("Direct link is assigned to : 0x%lx * 0x%lx", + (ulong) direct_link, + (ulong)(direct_link ? *direct_link : NULL))); } else { @@ -2468,6 +2483,24 @@ static uchar *translog_get_page(TRANSLOG_VALIDATOR_DATA *data, uchar *buffer) DBUG_RETURN(buffer); } +/** + @brief free direct log page link + + @param direct_link the direct log page link to be freed + +*/ + +static void translog_free_link(PAGECACHE_PAGE_LINK *direct_link) +{ + DBUG_ENTER("translog_free_link"); + DBUG_PRINT("info", ("Direct link: 0x%lx", + (ulong) direct_link)); + if (direct_link) + pagecache_unlock_by_link(log_descriptor.pagecache, direct_link, + PAGECACHE_LOCK_READ_UNLOCK, PAGECACHE_UNPIN, + LSN_IMPOSSIBLE, LSN_IMPOSSIBLE); + DBUG_VOID_RETURN; +} /* Finds last page of the given log file @@ -2796,7 +2829,7 @@ my_bool translog_init(const char *directory, TRANSLOG_VALIDATOR_DATA data; uchar buffer[TRANSLOG_PAGE_SIZE], *page; data.addr= ¤t_page; - if ((page= translog_get_page(&data, buffer)) == NULL) + if ((page= translog_get_page(&data, buffer, NULL)) == NULL) DBUG_RETURN(1); if (data.was_recovered) { @@ -2848,7 +2881,7 @@ my_bool translog_init(const char *directory, /* continue old log */ DBUG_ASSERT(LSN_FILE_NO(last_valid_page)== LSN_FILE_NO(log_descriptor.horizon)); - if ((page= translog_get_page(&data, buffer)) == NULL || + if ((page= translog_get_page(&data, buffer, NULL)) == NULL || (chunk_offset= translog_get_first_chunk_offset(page)) == 0) DBUG_RETURN(1); @@ -5153,24 +5186,54 @@ static my_bool translog_scanner_set_last_page(TRANSLOG_SCANNER_DATA } -/* - Initialize reader scanner +/** + @brief Get page from page cache according to requested method - SYNOPSIS - translog_init_scanner() - lsn LSN with which it have to be inited - fixed_horizon true if it is OK do not read records which was written + @param scanner The scanner data + + @return operation status + @retval 0 OK + @retval 1 Error +*/ + +static my_bool +translog_scanner_get_page(TRANSLOG_SCANNER_DATA *scanner) +{ + TRANSLOG_VALIDATOR_DATA data; + DBUG_ENTER("translog_scanner_get_page"); + data.addr= &scanner->page_addr; + data.was_recovered= 0; + DBUG_RETURN((scanner->page= + translog_get_page(&data, scanner->buffer, + (scanner->use_direct_link ? + &scanner->direct_link : + NULL))) == + NULL); +} + + +/** + @brief Initialize reader scanner. + + @param lsn LSN with which it have to be inited + @param fixed_horizon true if it is OK do not read records which was written after scanning beginning - scanner scanner which have to be inited + @param scanner scanner which have to be inited + @param use_direct prefer using direct lings from page handler + where it is possible. - RETURN - 0 OK - 1 Error + @note If direct link was used translog_destroy_scanner should be + called after it using + + @return status of the operation + @retval 0 OK + @retval 1 Error */ my_bool translog_init_scanner(LSN lsn, my_bool fixed_horizon, - struct st_translog_scanner_data *scanner) + TRANSLOG_SCANNER_DATA *scanner, + my_bool use_direct) { TRANSLOG_VALIDATOR_DATA data; DBUG_ENTER("translog_init_scanner"); @@ -5184,6 +5247,8 @@ my_bool translog_init_scanner(LSN lsn, scanner->page_offset= LSN_OFFSET(lsn) % TRANSLOG_PAGE_SIZE; scanner->fixed_horizon= fixed_horizon; + scanner->use_direct_link= use_direct; + scanner->direct_link= NULL; scanner->horizon= translog_get_horizon(); DBUG_PRINT("info", ("horizon: (0x%lu,0x%lx)", @@ -5198,12 +5263,24 @@ my_bool translog_init_scanner(LSN lsn, if (translog_scanner_set_last_page(scanner)) DBUG_RETURN(1); - if ((scanner->page= translog_get_page(&data, scanner->buffer)) == NULL) + if (translog_scanner_get_page(scanner)) DBUG_RETURN(1); DBUG_RETURN(0); } +/** + @brief Destroy scanner object; + + @param scanner The scanner object to destroy +*/ + +void translog_destroy_scanner(TRANSLOG_SCANNER_DATA *scanner) +{ + translog_free_link(scanner->direct_link); +} + + /* Checks End of the Log @@ -5298,7 +5375,6 @@ static my_bool translog_scanner_eof(TRANSLOG_SCANNER_DATA *scanner) scanner->last_file_page); } - /* Move scanner to the next chunk @@ -5315,7 +5391,6 @@ static my_bool translog_get_next_chunk(TRANSLOG_SCANNER_DATA *scanner) { uint16 len; - TRANSLOG_VALIDATOR_DATA data; DBUG_ENTER("translog_get_next_chunk"); if ((len= translog_get_total_chunk_length(scanner->page, @@ -5331,6 +5406,8 @@ translog_get_next_chunk(TRANSLOG_SCANNER_DATA *scanner) } if (translog_scanner_eop(scanner)) { + /* before reading next page we should unpin current one if it was pinned */ + translog_free_link(scanner->direct_link); if (translog_scanner_eof(scanner)) { DBUG_PRINT("info", ("horizon: (%lu,0x%lx) pageaddr: (%lu,0x%lx)", @@ -5350,9 +5427,7 @@ translog_get_next_chunk(TRANSLOG_SCANNER_DATA *scanner) scanner->page_addr+= TRANSLOG_PAGE_SIZE; /* offset increased */ } - data.addr= &scanner->page_addr; - data.was_recovered= 0; - if ((scanner->page= translog_get_page(&data, scanner->buffer)) == NULL) + if (translog_scanner_get_page(scanner)) DBUG_RETURN(1); scanner->page_offset= translog_get_first_chunk_offset(scanner->page); @@ -5482,7 +5557,7 @@ translog_variable_length_header(uchar *page, translog_size_t page_offset, { DBUG_PRINT("info", ("use internal scanner for header reading")); scanner= &internal_scanner; - if (translog_init_scanner(buff->lsn, 1, scanner)) + if (translog_init_scanner(buff->lsn, 1, scanner, 0)) DBUG_RETURN(RECHEADER_READ_ERROR); } if (translog_get_next_chunk(scanner)) @@ -5502,13 +5577,15 @@ translog_variable_length_header(uchar *page, translog_size_t page_offset, } base_lsn= buff->groups[0].addr; - translog_init_scanner(base_lsn, 1, scanner); + translog_init_scanner(base_lsn, 1, scanner, scanner == &internal_scanner); /* first group chunk is always chunk type 2 */ page= scanner->page; page_offset= scanner->page_offset; src= page + page_offset + 1; page_rest= TRANSLOG_PAGE_SIZE - (src - page); body_len= page_rest; + if (scanner == &internal_scanner) + translog_destroy_scanner(scanner); } if (lsns) { @@ -5615,6 +5692,7 @@ int translog_read_record_header(LSN lsn, TRANSLOG_HEADER_BUFFER *buff) { uchar buffer[TRANSLOG_PAGE_SIZE], *page; translog_size_t res, page_offset= LSN_OFFSET(lsn) % TRANSLOG_PAGE_SIZE; + PAGECACHE_PAGE_LINK direct_link; TRANSLOG_ADDRESS addr; TRANSLOG_VALIDATOR_DATA data; DBUG_ENTER("translog_read_record_header"); @@ -5628,8 +5706,10 @@ int translog_read_record_header(LSN lsn, TRANSLOG_HEADER_BUFFER *buff) data.was_recovered= 0; addr= lsn; addr-= page_offset; /* offset decreasing */ - res= (!(page= translog_get_page(&data, buffer))) ? RECHEADER_READ_ERROR : + res= (!(page= translog_get_page(&data, buffer, &direct_link))) ? + RECHEADER_READ_ERROR : translog_read_record_header_from_buffer(page, page_offset, buff, 0); + translog_free_link(direct_link); DBUG_RETURN(res); } @@ -5774,8 +5854,9 @@ static my_bool translog_record_read_next_chunk(struct st_translog_reader_data data->current_group++; data->current_chunk= 0; DBUG_PRINT("info", ("skip to group: #%u", data->current_group)); + translog_destroy_scanner(&data->scanner); translog_init_scanner(data->header.groups[data->current_group].addr, - 1, &data->scanner); + 1, &data->scanner, 1); } else { @@ -5794,7 +5875,8 @@ static my_bool translog_record_read_next_chunk(struct st_translog_reader_data DBUG_ASSERT(data->header.groups_no - 1 == data->current_group); DBUG_ASSERT(data->header.lsn == data->scanner.page_addr + data->scanner.page_offset); - translog_init_scanner(data->header.chunk0_data_addr, 1, &data->scanner); + translog_destroy_scanner(&data->scanner); + translog_init_scanner(data->header.chunk0_data_addr, 1, &data->scanner, 1); data->chunk_size= data->header.chunk0_data_len; data->body_offset= data->scanner.page_offset; data->current_offset= new_current_offset; @@ -5844,7 +5926,7 @@ static my_bool translog_init_reader_data(LSN lsn, { int read_header; DBUG_ENTER("translog_init_reader_data"); - if (translog_init_scanner(lsn, 1, &data->scanner) || + if (translog_init_scanner(lsn, 1, &data->scanner, 1) || ((read_header= translog_read_record_header_scan(&data->scanner, &data->header, 1)) == RECHEADER_READ_ERROR)) @@ -5865,6 +5947,16 @@ static my_bool translog_init_reader_data(LSN lsn, } +/** + @brief Destroy reader data object +*/ + +static void translog_destroy_reader_data(struct st_translog_reader_data *data) +{ + translog_destroy_scanner(&data->scanner); +} + + /* Read a part of the record. @@ -5924,7 +6016,10 @@ translog_size_t translog_read_record(LSN lsn, memcpy(buffer, data->header.header + offset, len); length-= len; if (length == 0) + { + translog_destroy_reader_data(data); DBUG_RETURN(requested_length); + } offset+= len; buffer+= len; DBUG_PRINT("info", @@ -5952,7 +6047,10 @@ translog_size_t translog_read_record(LSN lsn, (offset - data->current_offset), len); length-= len; if (length == 0) + { + translog_destroy_reader_data(data); DBUG_RETURN(requested_length); + } offset+= len; buffer+= len; DBUG_PRINT("info", @@ -5961,7 +6059,10 @@ translog_size_t translog_read_record(LSN lsn, (ulong) length)); } if (translog_record_read_next_chunk(data)) + { + translog_destroy_reader_data(data); DBUG_RETURN(requested_length - length); + } } } @@ -6624,6 +6725,7 @@ LSN translog_next_LSN(TRANSLOG_ADDRESS addr, TRANSLOG_ADDRESS horizon) { uint chunk_type; TRANSLOG_SCANNER_DATA scanner; + LSN result; DBUG_ENTER("translog_next_LSN"); if (horizon == LSN_IMPOSSIBLE) @@ -6632,7 +6734,7 @@ LSN translog_next_LSN(TRANSLOG_ADDRESS addr, TRANSLOG_ADDRESS horizon) if (addr == horizon) DBUG_RETURN(LSN_IMPOSSIBLE); - translog_init_scanner(addr, 0, &scanner); + translog_init_scanner(addr, 0, &scanner, 1); chunk_type= scanner.page[scanner.page_offset] & TRANSLOG_CHUNK_TYPE; DBUG_PRINT("info", ("type: %x byte: %x", (uint) chunk_type, @@ -6647,9 +6749,13 @@ LSN translog_next_LSN(TRANSLOG_ADDRESS addr, TRANSLOG_ADDRESS horizon) DBUG_PRINT("info", ("type: %x byte: %x", (uint) chunk_type, (uint) scanner.page[scanner.page_offset])); } + if (scanner.page[scanner.page_offset] == 0) - DBUG_RETURN(LSN_IMPOSSIBLE); /* reached page filler */ - DBUG_RETURN(scanner.page_addr + scanner.page_offset); + result= LSN_IMPOSSIBLE; /* reached page filler */ + else + result= scanner.page_addr + scanner.page_offset; + translog_destroy_scanner(&scanner); + DBUG_RETURN(result); } /** @@ -6681,7 +6787,7 @@ LSN translog_first_lsn_in_log() data.addr= &addr; { uchar buffer[TRANSLOG_PAGE_SIZE]; - if ((page= translog_get_page(&data, buffer)) == NULL || + if ((page= translog_get_page(&data, buffer, NULL)) == NULL || (chunk_offset= translog_get_first_chunk_offset(page)) == 0) DBUG_RETURN(LSN_ERROR); } @@ -6719,7 +6825,7 @@ LSN translog_first_theoretical_lsn() addr= MAKE_LSN(1, TRANSLOG_PAGE_SIZE); /* the first page of the file */ data.addr= &addr; - if ((page= translog_get_page(&data, buffer)) == NULL) + if ((page= translog_get_page(&data, buffer, NULL)) == NULL) DBUG_RETURN(LSN_ERROR); DBUG_RETURN(MAKE_LSN(1, TRANSLOG_PAGE_SIZE + diff --git a/storage/maria/ma_loghandler.h b/storage/maria/ma_loghandler.h index 164ff013b10..c53132916ec 100644 --- a/storage/maria/ma_loghandler.h +++ b/storage/maria/ma_loghandler.h @@ -182,10 +182,14 @@ typedef struct st_translog_scanner_data TRANSLOG_ADDRESS horizon; TRANSLOG_ADDRESS last_file_page; /* Last page on in this file */ uchar *page; /* page content pointer */ + /* direct link on the current page or NULL if it is not supported/requested */ + PAGECACHE_PAGE_LINK direct_link; /* offset of the chunk in the page */ translog_size_t page_offset; /* set horizon only once at init */ my_bool fixed_horizon; + /* try to get direct link on the page if it is possible */ + my_bool use_direct_link; } TRANSLOG_SCANNER_DATA; @@ -245,7 +249,9 @@ extern my_bool translog_flush(LSN lsn); extern my_bool translog_init_scanner(LSN lsn, my_bool fixed_horizon, - struct st_translog_scanner_data *scanner); + struct st_translog_scanner_data *scanner, + my_bool use_direct_link); +extern void translog_destroy_scanner(TRANSLOG_SCANNER_DATA *scanner); extern int translog_read_next_record_header(TRANSLOG_SCANNER_DATA *scanner, TRANSLOG_HEADER_BUFFER *buff); diff --git a/storage/maria/ma_open.c b/storage/maria/ma_open.c index 9b665cfb958..6852d5b4a6d 100644 --- a/storage/maria/ma_open.c +++ b/storage/maria/ma_open.c @@ -58,7 +58,7 @@ if (pos > end_pos) \ ** In MySQL the server will handle version issues. ******************************************************************************/ -MARIA_HA *_ma_test_if_reopen(char *filename) +MARIA_HA *_ma_test_if_reopen(const char *filename) { LIST *pos; @@ -1001,7 +1001,9 @@ uint _ma_state_info_write(MARIA_SHARE *share, uint pWrite) if (pWrite & 4) pthread_mutex_lock(&share->intern_lock); else if (maria_multi_threaded) + { safe_mutex_assert_owner(&share->intern_lock); + } if (share->base.born_transactional && translog_inited && !maria_in_recovery) { diff --git a/storage/maria/ma_pagecache.c b/storage/maria/ma_pagecache.c index 9f450d25c50..27c9ceaa0f1 100755 --- a/storage/maria/ma_pagecache.c +++ b/storage/maria/ma_pagecache.c @@ -96,7 +96,7 @@ #define PCBLOCK_INFO(B) \ DBUG_PRINT("info", \ ("block: 0x%lx file: %lu page: %lu s: %0x hshL: 0x%lx req: %u/%u " \ - "wrlocks: %u", \ + "wrlocks: %u pins: %u", \ (ulong)(B), \ (ulong)((B)->hash_link ? \ (B)->hash_link->file.file : \ @@ -110,7 +110,8 @@ (uint)((B)->hash_link ? \ (B)->hash_link->requests : \ 0), \ - block->wlocks)) + block->wlocks, \ + (uint)(B)->pins)) /* TODO: put it to my_static.c */ my_bool my_disable_flush_pagecache_blocks= 0; @@ -457,8 +458,10 @@ error: #define FLUSH_CACHE 2000 /* sort this many blocks at once */ static void free_block(PAGECACHE *pagecache, PAGECACHE_BLOCK_LINK *block); +#ifndef DBUG_OFF static void test_key_cache(PAGECACHE *pagecache, const char *where, my_bool lock); +#endif #define PAGECACHE_HASH(p, f, pos) (((ulong) (pos) + \ (ulong) (f).file) & (p->hash_entries-1)) @@ -655,11 +658,11 @@ static inline uint next_power(uint value) */ -int init_pagecache(PAGECACHE *pagecache, size_t use_mem, - uint division_limit, uint age_threshold, - uint block_size) +ulong init_pagecache(PAGECACHE *pagecache, size_t use_mem, + uint division_limit, uint age_threshold, + uint block_size) { - uint blocks, hash_links, length; + ulong blocks, hash_links, length; int error; DBUG_ENTER("init_pagecache"); DBUG_ASSERT(block_size >= 512); @@ -689,10 +692,10 @@ int init_pagecache(PAGECACHE *pagecache, size_t use_mem, block_size)); DBUG_ASSERT(((uint)(1 << pagecache->shift)) == block_size); - blocks= (int) (use_mem / (sizeof(PAGECACHE_BLOCK_LINK) + - 2 * sizeof(PAGECACHE_HASH_LINK) + - sizeof(PAGECACHE_HASH_LINK*) * - 5/4 + block_size)); + blocks= (ulong) (use_mem / (sizeof(PAGECACHE_BLOCK_LINK) + + 2 * sizeof(PAGECACHE_HASH_LINK) + + sizeof(PAGECACHE_HASH_LINK*) * + 5/4 + block_size)); /* We need to support page cache with just one block to be able to do scanning of rows-in-block files @@ -714,7 +717,7 @@ int init_pagecache(PAGECACHE *pagecache, size_t use_mem, ALIGN_SIZE(hash_links * sizeof(PAGECACHE_HASH_LINK)) + ALIGN_SIZE(sizeof(PAGECACHE_HASH_LINK*) * pagecache->hash_entries))) + - (((ulong) blocks) << pagecache->shift) > use_mem) + (blocks << pagecache->shift) > use_mem) blocks--; /* Allocate memory for cache page buffers */ if ((pagecache->block_mem= @@ -726,8 +729,7 @@ int init_pagecache(PAGECACHE *pagecache, size_t use_mem, For each block 2 hash links are allocated */ if ((pagecache->block_root= - (PAGECACHE_BLOCK_LINK*) my_malloc((uint) length, - MYF(0)))) + (PAGECACHE_BLOCK_LINK*) my_malloc((size_t) length, MYF(0)))) break; my_large_free(pagecache->block_mem, MYF(0)); pagecache->block_mem= 0; @@ -739,8 +741,8 @@ int init_pagecache(PAGECACHE *pagecache, size_t use_mem, } blocks= blocks / 4*3; } - pagecache->blocks_unused= (ulong) blocks; - pagecache->disk_blocks= (int) blocks; + pagecache->blocks_unused= blocks; + pagecache->disk_blocks= (long) blocks; pagecache->hash_links= hash_links; pagecache->hash_root= (PAGECACHE_HASH_LINK**) ((char*) pagecache->block_root + @@ -782,8 +784,8 @@ int init_pagecache(PAGECACHE *pagecache, size_t use_mem, pagecache->waiting_for_hash_link.last_thread= NULL; pagecache->waiting_for_block.last_thread= NULL; DBUG_PRINT("exit", - ("disk_blocks: %d block_root: 0x%lx hash_entries: %d\ - hash_root: 0x%lx hash_links: %d hash_link_root: 0x%lx", + ("disk_blocks: %ld block_root: 0x%lx hash_entries: %ld\ + hash_root: 0x%lx hash_links: %ld hash_link_root: 0x%lx", pagecache->disk_blocks, (long) pagecache->block_root, pagecache->hash_entries, (long) pagecache->hash_root, pagecache->hash_links, (long) pagecache->hash_link_root)); @@ -796,7 +798,7 @@ int init_pagecache(PAGECACHE *pagecache, size_t use_mem, } pagecache->blocks= pagecache->disk_blocks > 0 ? pagecache->disk_blocks : 0; - DBUG_RETURN((int) pagecache->disk_blocks); + DBUG_RETURN((ulong) pagecache->disk_blocks); err: error= my_errno; @@ -887,11 +889,11 @@ static int flush_all_key_blocks(PAGECACHE *pagecache) So we disable it for now. */ #if NOT_USED /* keep disabled until code is fixed see above !! */ -int resize_pagecache(PAGECACHE *pagecache, - size_t use_mem, uint division_limit, - uint age_threshold) +ulong resize_pagecache(PAGECACHE *pagecache, + size_t use_mem, uint division_limit, + uint age_threshold) { - int blocks; + ulong blocks; #ifdef THREAD struct st_my_thread_var *thread; WQUEUE *wqueue; @@ -1282,8 +1284,10 @@ static void unlink_block(PAGECACHE *pagecache, PAGECACHE_BLOCK_LINK *block) DBUG_ENTER("unlink_block"); DBUG_PRINT("unlink_block", ("unlink 0x%lx", (ulong)block)); if (block->next_used == block) + { /* The list contains only one member */ pagecache->used_last= pagecache->used_ins= NULL; + } else { block->next_used->prev_used= block->prev_used; @@ -2661,13 +2665,12 @@ void pagecache_unpin(PAGECACHE *pagecache, */ void pagecache_unlock_by_link(PAGECACHE *pagecache, - PAGECACHE_PAGE_LINK *link, + PAGECACHE_BLOCK_LINK *block, enum pagecache_page_lock lock, enum pagecache_page_pin pin, LSN first_REDO_LSN_for_page, LSN lsn) { - PAGECACHE_BLOCK_LINK *block= (PAGECACHE_BLOCK_LINK *)link; DBUG_ENTER("pagecache_unlock_by_link"); DBUG_PRINT("enter", ("block: 0x%lx fd: %u page: %lu %s %s", (ulong) block, @@ -2820,16 +2823,28 @@ void pagecache_unpin_by_link(PAGECACHE *pagecache, Pin will be chosen according to lock parameter (see lock_to_pin) */ -static enum pagecache_page_pin lock_to_pin[]= +static enum pagecache_page_pin lock_to_pin[2][8]= { - PAGECACHE_PIN_LEFT_UNPINNED /*PAGECACHE_LOCK_LEFT_UNLOCKED*/, - PAGECACHE_PIN_LEFT_UNPINNED /*PAGECACHE_LOCK_LEFT_READLOCKED*/, - PAGECACHE_PIN_LEFT_PINNED /*PAGECACHE_LOCK_LEFT_WRITELOCKED*/, - PAGECACHE_PIN_LEFT_UNPINNED /*PAGECACHE_LOCK_READ*/, - PAGECACHE_PIN /*PAGECACHE_LOCK_WRITE*/, - PAGECACHE_PIN_LEFT_UNPINNED /*PAGECACHE_LOCK_READ_UNLOCK*/, - PAGECACHE_UNPIN /*PAGECACHE_LOCK_WRITE_UNLOCK*/, - PAGECACHE_UNPIN /*PAGECACHE_LOCK_WRITE_TO_READ*/ + { + PAGECACHE_PIN_LEFT_UNPINNED /*PAGECACHE_LOCK_LEFT_UNLOCKED*/, + PAGECACHE_PIN_LEFT_UNPINNED /*PAGECACHE_LOCK_LEFT_READLOCKED*/, + PAGECACHE_PIN_LEFT_PINNED /*PAGECACHE_LOCK_LEFT_WRITELOCKED*/, + PAGECACHE_PIN_LEFT_UNPINNED /*PAGECACHE_LOCK_READ*/, + PAGECACHE_PIN /*PAGECACHE_LOCK_WRITE*/, + PAGECACHE_PIN_LEFT_UNPINNED /*PAGECACHE_LOCK_READ_UNLOCK*/, + PAGECACHE_UNPIN /*PAGECACHE_LOCK_WRITE_UNLOCK*/, + PAGECACHE_UNPIN /*PAGECACHE_LOCK_WRITE_TO_READ*/ + }, + { + PAGECACHE_PIN_LEFT_UNPINNED /*PAGECACHE_LOCK_LEFT_UNLOCKED*/, + PAGECACHE_PIN_LEFT_PINNED /*PAGECACHE_LOCK_LEFT_READLOCKED*/, + PAGECACHE_PIN_LEFT_PINNED /*PAGECACHE_LOCK_LEFT_WRITELOCKED*/, + PAGECACHE_PIN /*PAGECACHE_LOCK_READ*/, + PAGECACHE_PIN /*PAGECACHE_LOCK_WRITE*/, + PAGECACHE_PIN_LEFT_UNPINNED /*PAGECACHE_LOCK_READ_UNLOCK*/, + PAGECACHE_UNPIN /*PAGECACHE_LOCK_WRITE_UNLOCK*/, + PAGECACHE_PIN_LEFT_PINNED /*PAGECACHE_LOCK_WRITE_TO_READ*/ + } }; uchar *pagecache_valid_read(PAGECACHE *pagecache, @@ -2839,24 +2854,27 @@ uchar *pagecache_valid_read(PAGECACHE *pagecache, uchar *buff, enum pagecache_page_type type, enum pagecache_page_lock lock, - PAGECACHE_PAGE_LINK *link, + PAGECACHE_BLOCK_LINK **link, pagecache_disk_read_validator validator, uchar* validator_data) { int error= 0; - enum pagecache_page_pin pin= lock_to_pin[lock]; - PAGECACHE_PAGE_LINK fake_link; + enum pagecache_page_pin pin= lock_to_pin[test(buff==0)][lock]; + PAGECACHE_BLOCK_LINK *fake_link; DBUG_ENTER("pagecache_valid_read"); - DBUG_PRINT("enter", ("fd: %u page: %lu level: %u t:%s %s %s", - (uint) file->file, (ulong) pageno, level, + DBUG_PRINT("enter", ("fd: %u page: %lu buffer: 0x%lx level: %u " + "t:%s %s %s", + (uint) file->file, (ulong) pageno, + (ulong) buff, level, page_cache_page_type_str[type], page_cache_page_lock_str[lock], page_cache_page_pin_str[pin])); + DBUG_ASSERT(buff != 0 || (buff == 0 && (pin == PAGECACHE_PIN || + pin == PAGECACHE_PIN_LEFT_PINNED))); if (!link) link= &fake_link; - else - *link= 0; + *link= 0; /* Catch errors */ restart: @@ -2910,19 +2928,25 @@ restart: goto restart; } - if (! ((status= block->status) & PCBLOCK_ERROR)) + status= block->status; + if (!buff) + buff= block->buffer; + else { + if (!(status & PCBLOCK_ERROR)) + { #if !defined(SERIALIZED_READ_FROM_CACHE) - pagecache_pthread_mutex_unlock(&pagecache->cache_lock); + pagecache_pthread_mutex_unlock(&pagecache->cache_lock); #endif - DBUG_ASSERT((pagecache->block_size & 511) == 0); - /* Copy data from the cache buffer */ - bmove512(buff, block->buffer, pagecache->block_size); + DBUG_ASSERT((pagecache->block_size & 511) == 0); + /* Copy data from the cache buffer */ + bmove512(buff, block->buffer, pagecache->block_size); #if !defined(SERIALIZED_READ_FROM_CACHE) - pagecache_pthread_mutex_lock(&pagecache->cache_lock); + pagecache_pthread_mutex_lock(&pagecache->cache_lock); #endif + } } remove_reader(block); @@ -2934,7 +2958,7 @@ restart: if (pin == PAGECACHE_PIN_LEFT_UNPINNED || pin == PAGECACHE_UNPIN) unreg_request(pagecache, block, 1); else - *link= (PAGECACHE_PAGE_LINK)block; + *link= block; dec_counter_for_resize_op(pagecache); @@ -2983,7 +3007,7 @@ my_bool pagecache_delete(PAGECACHE *pagecache, my_bool flush) { int error= 0; - enum pagecache_page_pin pin= lock_to_pin[lock]; + enum pagecache_page_pin pin= lock_to_pin[0][lock]; DBUG_ENTER("pagecache_delete"); DBUG_PRINT("enter", ("fd: %u page: %lu %s %s", (uint) file->file, (ulong) pageno, @@ -3830,7 +3854,8 @@ int flush_pagecache_blocks(PAGECACHE *pagecache, 0 on success (always because it can't fail) */ -int reset_pagecache_counters(const char *name, PAGECACHE *pagecache) +int reset_pagecache_counters(const char *name __attribute__((unused)), + PAGECACHE *pagecache) { DBUG_ENTER("reset_pagecache_counters"); if (!pagecache->inited) diff --git a/storage/maria/ma_pagecache.h b/storage/maria/ma_pagecache.h index 0e2aff3644d..7318cc08040 100644 --- a/storage/maria/ma_pagecache.h +++ b/storage/maria/ma_pagecache.h @@ -73,8 +73,6 @@ enum pagecache_write_mode PAGECACHE_WRITE_DONE }; -typedef void *PAGECACHE_PAGE_LINK; - /* file descriptor for Maria */ typedef struct st_pagecache_file { @@ -93,6 +91,8 @@ typedef struct st_pagecache_page PAGECACHE_PAGE; struct st_pagecache_hash_link; typedef struct st_pagecache_hash_link PAGECACHE_HASH_LINK; +typedef PAGECACHE_BLOCK_LINK * PAGECACHE_PAGE_LINK; /* To be removed */ + #include typedef my_bool (*pagecache_disk_read_validator)(uchar *page, uchar *data); @@ -106,25 +106,22 @@ typedef my_bool (*pagecache_disk_read_validator)(uchar *page, uchar *data); typedef struct st_pagecache { - my_bool inited; - my_bool resize_in_flush; /* true during flush of resize operation */ - my_bool can_be_used; /* usage of cache for read/write is allowed */ - uint shift; /* block size = 2 ^ shift */ - size_t mem_size; /* specified size of the cache memory */ - uint32 block_size; /* size of the page buffer of a cache block */ + size_t mem_size; /* specified size of the cache memory */ ulong min_warm_blocks; /* min number of warm blocks; */ ulong age_threshold; /* age threshold for hot blocks */ ulonglong time; /* total number of block link operations */ - uint hash_entries; /* max number of entries in the hash table */ - int hash_links; /* max number of hash links */ - int hash_links_used; /* number of hash links taken from free links pool */ - int disk_blocks; /* max number of blocks in the cache */ + ulong hash_entries; /* max number of entries in the hash table */ + long hash_links; /* max number of hash links */ + long hash_links_used; /* number of hash links taken from free links pool */ + long disk_blocks; /* max number of blocks in the cache */ ulong blocks_used; /* maximum number of concurrently used blocks */ ulong blocks_unused; /* number of currently unused blocks */ ulong blocks_changed; /* number of currently dirty blocks */ ulong warm_blocks; /* number of blocks in warm sub-chain */ ulong cnt_for_resize_op; /* counter to block resize operation */ ulong blocks_available; /* number of blocks available in the LRU chain */ + long blocks; /* max number of blocks in the cache */ + uint32 block_size; /* size of the page buffer of a cache block */ PAGECACHE_HASH_LINK **hash_root;/* arr. of entries into hash table buckets */ PAGECACHE_HASH_LINK *hash_link_root;/* memory for hash table links */ PAGECACHE_HASH_LINK *free_hash_list;/* list of free hash links */ @@ -159,19 +156,22 @@ typedef struct st_pagecache ulonglong global_cache_r_requests;/* number of read requests (read hits) */ ulonglong global_cache_read; /* number of reads from files to cache */ - int blocks; /* max number of blocks in the cache */ + uint shift; /* block size = 2 ^ shift */ + my_bool inited; + my_bool resize_in_flush; /* true during flush of resize operation */ + my_bool can_be_used; /* usage of cache for read/write is allowed */ my_bool in_init; /* Set to 1 in MySQL during init/resize */ } PAGECACHE; /* The default key cache */ extern PAGECACHE dflt_pagecache_var, *dflt_pagecache; -extern int init_pagecache(PAGECACHE *pagecache, size_t use_mem, - uint division_limit, uint age_threshold, - uint block_size); -extern int resize_pagecache(PAGECACHE *pagecache, - size_t use_mem, uint division_limit, - uint age_threshold); +extern ulong init_pagecache(PAGECACHE *pagecache, size_t use_mem, + uint division_limit, uint age_threshold, + uint block_size); +extern ulong resize_pagecache(PAGECACHE *pagecache, + size_t use_mem, uint division_limit, + uint age_threshold); extern void change_pagecache_param(PAGECACHE *pagecache, uint division_limit, uint age_threshold); @@ -185,7 +185,7 @@ extern uchar *pagecache_valid_read(PAGECACHE *pagecache, uchar *buff, enum pagecache_page_type type, enum pagecache_page_lock lock, - PAGECACHE_PAGE_LINK *link, + PAGECACHE_BLOCK_LINK **link, pagecache_disk_read_validator validator, uchar* validator_data); @@ -218,7 +218,7 @@ extern void pagecache_unlock(PAGECACHE *pagecache, LSN first_REDO_LSN_for_page, LSN lsn); extern void pagecache_unlock_by_link(PAGECACHE *pagecache, - PAGECACHE_PAGE_LINK *link, + PAGECACHE_BLOCK_LINK *block, enum pagecache_page_lock lock, enum pagecache_page_pin pin, LSN first_REDO_LSN_for_page, diff --git a/storage/maria/ma_recovery.c b/storage/maria/ma_recovery.c index e740e334b5f..bb9852779c8 100644 --- a/storage/maria/ma_recovery.c +++ b/storage/maria/ma_recovery.c @@ -105,6 +105,7 @@ static int new_table(uint16 sid, const char *name, static int new_page(File fileid, pgcache_page_no_t pageid, LSN rec_lsn, struct st_dirty_page *dirty_page); static int close_all_tables(void); +static my_bool close_one_table(const char *name, LSN addr); static void print_redo_phase_progress(TRANSLOG_ADDRESS addr); /** @brief global [out] buffer for translog_read_record(); never shrinks */ @@ -415,6 +416,8 @@ prototype_redo_exec_hook(REDO_CREATE_TABLE) uint flags; int error= 1, create_mode= O_RDWR | O_TRUNC; MARIA_HA *info= NULL; + uint kfile_size_before_extension, keystart; + if (skip_DDLs) { tprint(tracef, "we skip DDLs\n"); @@ -431,6 +434,12 @@ prototype_redo_exec_hook(REDO_CREATE_TABLE) } name= log_record_buffer.str; tprint(tracef, "Table '%s'", name); + if (close_one_table(name, rec->lsn)) + { + tprint(tracef, " got error %d on close\n", my_errno); + ALERT_USER(); + goto end; + } /* we try hard to get create_rename_lsn, to avoid mistakes if possible */ info= maria_open(name, O_RDONLY, HA_OPEN_FOR_REPAIR); if (info) @@ -474,7 +483,7 @@ prototype_redo_exec_hook(REDO_CREATE_TABLE) info= NULL; } else /* one or two files absent, or header corrupted... */ - tprint(tracef, "can't be opened, probably does not exist"); + tprint(tracef, " can't be opened, probably does not exist"); /* if does not exist, or is older, overwrite it */ /** @todo symlinks */ ptr= name + strlen(name) + 1; @@ -490,13 +499,13 @@ prototype_redo_exec_hook(REDO_CREATE_TABLE) if ((kfile= my_create_with_symlink(linkname_ptr, filename, 0, create_mode, MYF(MY_WME|create_flag))) < 0) { - tprint(tracef, "Failed to create index file\n"); + tprint(tracef, " Failed to create index file\n"); goto end; } ptr++; - uint kfile_size_before_extension= uint2korr(ptr); + kfile_size_before_extension= uint2korr(ptr); ptr+= 2; - uint keystart= uint2korr(ptr); + keystart= uint2korr(ptr); ptr+= 2; /* set create_rename_lsn (for maria_read_log to be idempotent) */ lsn_store(ptr + sizeof(info->s->state.header) + 2, rec->lsn); @@ -507,7 +516,7 @@ prototype_redo_exec_hook(REDO_CREATE_TABLE) kfile_size_before_extension, 0, MYF(MY_NABP|MY_WME)) || my_chsize(kfile, keystart, 0, MYF(MY_WME))) { - tprint(tracef, "Failed to write to index file\n"); + tprint(tracef, " Failed to write to index file\n"); goto end; } if (!(flags & HA_DONT_TOUCH_DATA)) @@ -521,7 +530,7 @@ prototype_redo_exec_hook(REDO_CREATE_TABLE) MYF(MY_WME | create_flag))) < 0) || my_close(dfile, MYF(MY_WME))) { - tprint(tracef, "Failed to create data file\n"); + tprint(tracef, " Failed to create data file\n"); goto end; } /* @@ -533,7 +542,7 @@ prototype_redo_exec_hook(REDO_CREATE_TABLE) if (((info= maria_open(name, O_RDONLY, 0)) == NULL) || _ma_initialize_data_file(info->s, info->dfile.file)) { - tprint(tracef, "Failed to open new table or write to data file\n"); + tprint(tracef, " Failed to open new table or write to data file\n"); goto end; } } @@ -1436,6 +1445,11 @@ prototype_undo_exec_hook(UNDO_ROW_UPDATE) static int run_redo_phase(LSN lsn, my_bool apply) { + TRANSLOG_HEADER_BUFFER rec; + struct st_translog_scanner_data scanner; + int len; + uint i; + /* install hooks for execution */ #define install_redo_exec_hook(R) \ log_record_type_descriptor[LOGREC_ ## R].record_execute_in_redo_phase= \ @@ -1467,8 +1481,6 @@ static int run_redo_phase(LSN lsn, my_bool apply) current_group_end_lsn= LSN_IMPOSSIBLE; - TRANSLOG_HEADER_BUFFER rec; - if (unlikely(lsn == LSN_IMPOSSIBLE || lsn == translog_get_horizon())) { tprint(tracef, "checkpoint address refers to the log end log or " @@ -1476,7 +1488,7 @@ static int run_redo_phase(LSN lsn, my_bool apply) return 0; } - int len= translog_read_record_header(lsn, &rec); + len= translog_read_record_header(lsn, &rec); /** @todo EOF should be detected */ if (len == RECHEADER_READ_ERROR) @@ -1484,13 +1496,11 @@ static int run_redo_phase(LSN lsn, my_bool apply) tprint(tracef, "Failed to read header of the first record.\n"); return 1; } - struct st_translog_scanner_data scanner; - if (translog_init_scanner(lsn, 1, &scanner)) + if (translog_init_scanner(lsn, 1, &scanner, 0)) { tprint(tracef, "Scanner init failed\n"); return 1; } - uint i; for (i= 1;;i++) { uint16 sid= rec.short_trid; @@ -1533,7 +1543,7 @@ static int run_redo_phase(LSN lsn, my_bool apply) tprint(tracef, "Cannot find record where it should be\n"); return 1; } - if (translog_init_scanner(rec2.lsn, 1, &scanner2)) + if (translog_init_scanner(rec2.lsn, 1, &scanner2, 0)) { tprint(tracef, "Scanner2 init failed\n"); return 1; @@ -1607,6 +1617,7 @@ static uint end_of_redo_phase(my_bool prepare_for_undo_phase) { uint sid, unfinished= 0; char llbuf[22]; + LSN addr; hash_free(&all_dirty_pages); /* @@ -1667,7 +1678,7 @@ static uint end_of_redo_phase(my_bool prepare_for_undo_phase) The UNDO phase uses some normal run-time code of ROLLBACK: generates log records, etc; prepare tables for that */ - LSN addr= translog_get_horizon(); + addr= translog_get_horizon(); for (sid= 0; sid <= SHARE_ID_MAX; sid++) { MARIA_HA *info= all_tables[sid].info; @@ -2070,6 +2081,42 @@ end: return error; } + +/* Close one table during redo phase */ + +static my_bool close_one_table(const char *open_name, LSN addr) +{ + my_bool res= 0; + LIST *pos; + /* There are no other threads using the tables, so we don't need any locks */ + for (pos=maria_open_list ; pos ;) + { + MARIA_HA *info= (MARIA_HA*) pos->data; + MARIA_SHARE *share= info->s; + pos= pos->next; + if (!strcmp(share->open_file_name, open_name)) + { + struct st_table_for_recovery *internal_table, *end; + + for (internal_table= all_tables, end= internal_table + SHARE_ID_MAX + 1; + internal_table < end ; + internal_table++) + { + if (internal_table->info == info) + { + internal_table->info= 0; + break; + } + } + prepare_table_for_close(info, addr); + if (maria_close(info)) + res= 1; + } + } + return res; +} + + static void print_redo_phase_progress(TRANSLOG_ADDRESS addr) { static int end_logno= FILENO_IMPOSSIBLE, end_offset, percentage_printed= 0; diff --git a/storage/maria/maria_chk.c b/storage/maria/maria_chk.c index f9ed249817e..e6ac0dcfc50 100644 --- a/storage/maria/maria_chk.c +++ b/storage/maria/maria_chk.c @@ -154,7 +154,7 @@ int main(int argc, char **argv) enum options_mc { OPT_CHARSETS_DIR=256, OPT_SET_COLLATION,OPT_START_CHECK_POS, - OPT_CORRECT_CHECKSUM, OPT_KEY_BUFFER_SIZE, + OPT_CORRECT_CHECKSUM, OPT_PAGE_BUFFER_SIZE, OPT_KEY_CACHE_BLOCK_SIZE, OPT_MARIA_BLOCK_SIZE, OPT_READ_BUFFER_SIZE, OPT_WRITE_BUFFER_SIZE, OPT_SORT_BUFFER_SIZE, OPT_SORT_KEY_BLOCKS, OPT_DECODE_BITS, OPT_FT_MIN_WORD_LEN, @@ -296,7 +296,7 @@ static struct my_option my_long_options[] = {"wait", 'w', "Wait if table is locked.", 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, - { "key_buffer_size", OPT_KEY_BUFFER_SIZE, "", + { "page_buffer_size", OPT_PAGE_BUFFER_SIZE, "", (uchar**) &check_param.use_buffers, (uchar**) &check_param.use_buffers, 0, GET_ULONG, REQUIRED_ARG, (long) USE_BUFFER_INIT, (long) MALLOC_OVERHEAD, (long) ~0L, (long) MALLOC_OVERHEAD, (long) IO_SIZE, 0}, diff --git a/storage/maria/maria_def.h b/storage/maria/maria_def.h index 09852f4dc86..adcb35a5e0e 100644 --- a/storage/maria/maria_def.h +++ b/storage/maria/maria_def.h @@ -842,7 +842,7 @@ typedef struct st_maria_block_info #define UPDATE_AUTO_INC 8 #define UPDATE_OPEN_COUNT 16 -#define USE_BUFFER_INIT (((1024L*512L-MALLOC_OVERHEAD)/IO_SIZE)*IO_SIZE) +#define USE_BUFFER_INIT (((1024L*1024L*10-MALLOC_OVERHEAD)/8192)*8192) #define READ_BUFFER_INIT (1024L*256L-MALLOC_OVERHEAD) #define SORT_BUFFER_INIT (2048L*1024L-MALLOC_OVERHEAD) #define MIN_SORT_BUFFER (4096-MALLOC_OVERHEAD) @@ -906,7 +906,7 @@ my_bool _ma_check_status(void *param); void _ma_reset_status(MARIA_HA *maria); #include "ma_commit.h" -extern MARIA_HA *_ma_test_if_reopen(char *filename); +extern MARIA_HA *_ma_test_if_reopen(const char *filename); my_bool _ma_check_table_is_closed(const char *name, const char *where); int _ma_open_datafile(MARIA_HA *info, MARIA_SHARE *share, File file_to_dup); int _ma_open_keyfile(MARIA_SHARE *share); diff --git a/storage/maria/maria_read_log.c b/storage/maria/maria_read_log.c index a7a6370b1c4..ec1f7697a23 100644 --- a/storage/maria/maria_read_log.c +++ b/storage/maria/maria_read_log.c @@ -17,7 +17,6 @@ #include "ma_recovery.h" #include -#define PCACHE_SIZE (1024*1024*10) #define LOG_FLAGS 0 #define LOG_FILE_SIZE (1024L*1024L) @@ -30,7 +29,8 @@ const char *default_dbug_option= "d:t:i:O,\\maria_read_log.trace"; const char *default_dbug_option= "d:t:i:o,/tmp/maria_read_log.trace"; #endif #endif /* DBUG_OFF */ -static my_bool opt_only_display, opt_display_and_apply; +static my_bool opt_only_display, opt_apply, opt_apply_undo, opt_silent; +static ulong opt_page_buffer_size; int main(int argc, char **argv) { @@ -63,7 +63,7 @@ int main(int argc, char **argv) } /* same page cache for log and data; assumes same page size... */ DBUG_ASSERT(maria_block_size == TRANSLOG_PAGE_SIZE); - if (init_pagecache(maria_pagecache, PCACHE_SIZE, 0, 0, + if (init_pagecache(maria_pagecache, opt_page_buffer_size, 0, 0, TRANSLOG_PAGE_SIZE) == 0) { fprintf(stderr, "Got error in init_pagecache() (errno: %d)\n", errno); @@ -100,8 +100,8 @@ int main(int argc, char **argv) LSN_IN_PARTS(lsn)); fprintf(stdout, "TRACE of the last maria_read_log\n"); - if (maria_apply_log(lsn, opt_display_and_apply, stdout, - opt_display_and_apply, FALSE)) + if (maria_apply_log(lsn, opt_apply, opt_silent ? NULL : stdout, + opt_apply_undo, FALSE)) goto err; fprintf(stdout, "%s: SUCCESS\n", my_progname); @@ -121,19 +121,32 @@ end: static struct my_option my_long_options[] = { + {"apply", 'a', + "Apply log to tables. Will display a lot of information if not run with --silent", + (uchar **) &opt_apply, (uchar **) &opt_apply, 0, + GET_BOOL, NO_ARG, 0, 0, 0, 0, 0, 0}, +#ifndef DBUG_OFF + {"debug", '#', "Output debug log. Often the argument is 'd:t:o,filename'.", + 0, 0, 0, GET_STR, OPT_ARG, 0, 0, 0, 0, 0, 0}, +#endif {"help", '?', "Display this help and exit.", 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, {"only-display", 'o', "display brief info about records's header", (uchar **) &opt_only_display, (uchar **) &opt_only_display, 0, GET_BOOL, NO_ARG,0, 0, 0, 0, 0, 0}, - {"display-and-apply", 'a', - "like --only-display but displays more info and modifies tables", - (uchar **) &opt_display_and_apply, (uchar **) &opt_display_and_apply, 0, + { "page_buffer_size", 'P', "", + (uchar**) &opt_page_buffer_size, (uchar**) &opt_page_buffer_size, 0, + GET_ULONG, REQUIRED_ARG, (long) USE_BUFFER_INIT, + (long) MALLOC_OVERHEAD, (long) ~(ulong) 0, (long) MALLOC_OVERHEAD, + (long) IO_SIZE, 0}, + {"silent", 's', "Print less information during apply/undo phase", + (uchar **) &opt_silent, (uchar **) &opt_silent, 0, GET_BOOL, NO_ARG, 0, 0, 0, 0, 0, 0}, -#ifndef DBUG_OFF - {"debug", '#', "Output debug log. Often this is 'd:t:o,filename'.", - 0, 0, 0, GET_STR, OPT_ARG, 0, 0, 0, 0, 0, 0}, -#endif + {"undo", 'u', "Apply undos to tables. (disable with --disable-undo)", + (uchar **) &opt_apply_undo, (uchar **) &opt_apply_undo, 0, + GET_BOOL, NO_ARG, 1, 0, 0, 0, 0, 0}, + {"version", 'V', "Print version and exit.", + 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, { 0, 0, 0, 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0} }; @@ -141,7 +154,7 @@ static struct my_option my_long_options[] = static void print_version(void) { - VOID(printf("%s Ver 1.0 for %s on %s\n", + VOID(printf("%s Ver 1.1 for %s on %s\n", my_progname, SYSTEM_TYPE, MACHINE_TYPE)); NETWARE_SET_SCREEN_MODE(1); } @@ -174,6 +187,9 @@ get_one_option(int optid __attribute__((unused)), case '?': usage(); exit(0); + case 'V': + print_version(); + exit(0); #ifndef DBUG_OFF case '#': DBUG_SET_INITIAL(argument ? argument : default_dbug_option); @@ -192,7 +208,10 @@ static void get_options(int *argc,char ***argv) if ((ho_error=handle_options(argc, argv, my_long_options, get_one_option))) exit(ho_error); - if ((opt_only_display + opt_display_and_apply) != 1) + if (opt_apply_undo) + opt_apply= 1; + + if ((opt_only_display + opt_apply) != 1) { usage(); exit(1); diff --git a/storage/maria/unittest/ma_control_file-t.c b/storage/maria/unittest/ma_control_file-t.c index a7472361dad..db0a1690ab8 100644 --- a/storage/maria/unittest/ma_control_file-t.c +++ b/storage/maria/unittest/ma_control_file-t.c @@ -407,7 +407,7 @@ static void version() static my_bool get_one_option(int optid, const struct my_option *opt __attribute__((unused)), - char *argument) + char *argument __attribute__((unused))) { switch(optid) { case 'V': diff --git a/storage/maria/unittest/ma_test_loghandler-t.c b/storage/maria/unittest/ma_test_loghandler-t.c index 6ea45f80433..fa7267e31d4 100644 --- a/storage/maria/unittest/ma_test_loghandler-t.c +++ b/storage/maria/unittest/ma_test_loghandler-t.c @@ -363,7 +363,7 @@ int main(int argc __attribute__((unused)), char *argv[]) read_ok(&rec); translog_free_record_header(&rec); lsn= first_lsn; - if (translog_init_scanner(first_lsn, 1, &scanner)) + if (translog_init_scanner(first_lsn, 1, &scanner, 0)) { fprintf(stderr, "scanner init failed\n"); goto err; diff --git a/storage/maria/unittest/ma_test_loghandler_multigroup-t.c b/storage/maria/unittest/ma_test_loghandler_multigroup-t.c index d5f00bdb6fd..559cd17638f 100644 --- a/storage/maria/unittest/ma_test_loghandler_multigroup-t.c +++ b/storage/maria/unittest/ma_test_loghandler_multigroup-t.c @@ -378,7 +378,7 @@ int main(int argc __attribute__((unused)), char *argv[]) ok(1, "read record"); translog_free_record_header(&rec); lsn= first_lsn; - if (translog_init_scanner(first_lsn, 1, &scanner)) + if (translog_init_scanner(first_lsn, 1, &scanner, 0)) { fprintf(stderr, "scanner init failed\n"); goto err; diff --git a/storage/maria/unittest/ma_test_loghandler_multithread-t.c b/storage/maria/unittest/ma_test_loghandler_multithread-t.c index 6255c11db89..a3af67c8ac2 100644 --- a/storage/maria/unittest/ma_test_loghandler_multithread-t.c +++ b/storage/maria/unittest/ma_test_loghandler_multithread-t.c @@ -373,7 +373,7 @@ int main(int argc __attribute__((unused)), bzero(indeces, sizeof(indeces)); - if (translog_init_scanner(first_lsn, 1, &scanner)) + if (translog_init_scanner(first_lsn, 1, &scanner, 0)) { fprintf(stderr, "scanner init failed\n"); goto err; -- cgit v1.2.1 From 7492e2ca8e4005495963f9b5e42b1c0038c86972 Mon Sep 17 00:00:00 2001 From: unknown Date: Thu, 27 Sep 2007 17:36:03 +0300 Subject: Use direct memory access for the log scan BUILD/SETUP.sh: compile maria by default --- BUILD/SETUP.sh | 1 + storage/maria/ma_recovery.c | 32 +++++++++++++++++++++----------- 2 files changed, 22 insertions(+), 11 deletions(-) diff --git a/BUILD/SETUP.sh b/BUILD/SETUP.sh index 429c0cd85b4..a07f294be28 100755 --- a/BUILD/SETUP.sh +++ b/BUILD/SETUP.sh @@ -145,6 +145,7 @@ base_configs="--prefix=$prefix --enable-assembler " base_configs="$base_configs --with-extra-charsets=complex " base_configs="$base_configs --enable-thread-safe-client " base_configs="$base_configs --with-big-tables" +base_configs="$base_configs --with-plugin-maria" if test -d "$path/../cmd-line-utils/readline" then diff --git a/storage/maria/ma_recovery.c b/storage/maria/ma_recovery.c index bb9852779c8..d3aaa64bbda 100644 --- a/storage/maria/ma_recovery.c +++ b/storage/maria/ma_recovery.c @@ -1496,7 +1496,7 @@ static int run_redo_phase(LSN lsn, my_bool apply) tprint(tracef, "Failed to read header of the first record.\n"); return 1; } - if (translog_init_scanner(lsn, 1, &scanner, 0)) + if (translog_init_scanner(lsn, 1, &scanner, 1)) { tprint(tracef, "Scanner init failed\n"); return 1; @@ -1529,24 +1529,25 @@ static int run_redo_phase(LSN lsn, my_bool apply) } else { + struct st_translog_scanner_data scanner2; + TRANSLOG_HEADER_BUFFER rec2; /* There is a complete group for this transaction, containing more than this event. */ tprint(tracef, " ends a group:\n"); - struct st_translog_scanner_data scanner2; - TRANSLOG_HEADER_BUFFER rec2; len= - translog_read_record_header(all_active_trans[sid].group_start_lsn, &rec2); + translog_read_record_header(all_active_trans[sid].group_start_lsn, + &rec2); if (len < 0) /* EOF or error */ { tprint(tracef, "Cannot find record where it should be\n"); - return 1; + goto err; } - if (translog_init_scanner(rec2.lsn, 1, &scanner2, 0)) + if (translog_init_scanner(rec2.lsn, 1, &scanner2, 1)) { tprint(tracef, "Scanner2 init failed\n"); - return 1; + goto err; } current_group_end_lsn= rec.lsn; do @@ -1556,13 +1557,16 @@ static int run_redo_phase(LSN lsn, my_bool apply) const LOG_DESC *log_desc2= &log_record_type_descriptor[rec2.type]; display_record_position(log_desc2, &rec2, 0); if (apply && display_and_apply_record(log_desc2, &rec2)) - return 1; + { + translog_destroy(&scanner2); + goto err; + } } len= translog_read_next_record_header(&scanner2, &rec2); if (len < 0) /* EOF or error */ { tprint(tracef, "Cannot find record where it should be\n"); - return 1; + goto err; } } while (rec2.lsn < rec.lsn); @@ -1571,10 +1575,11 @@ static int run_redo_phase(LSN lsn, my_bool apply) all_active_trans[sid].group_start_lsn= LSN_IMPOSSIBLE; current_group_end_lsn= LSN_IMPOSSIBLE; /* for debugging */ display_record_position(log_desc, &rec, 0); + translog_destroy_scanner(&scanner2); } } if (apply && display_and_apply_record(log_desc, &rec)) - return 1; + goto err; } else /* record does not end group */ { @@ -1595,13 +1600,18 @@ static int run_redo_phase(LSN lsn, my_bool apply) break; case RECHEADER_READ_ERROR: tprint(tracef, "Error reading log\n"); - return 1; + goto err; } break; } } + translog_destroy_scanner(&scanner); translog_free_record_header(&rec); return 0; + +err: + translog_destroy_scanner(&scanner); + return 1; } -- cgit v1.2.1 From 0de4c5562339eb0af9303eb33526a6dde717f5a7 Mon Sep 17 00:00:00 2001 From: unknown Date: Thu, 27 Sep 2007 17:41:21 +0300 Subject: Faster translog_filename_by_fileno Fixed compiler warning PAGECACHE_PAGE_LINK -> PAGECACHE_BLOCK_LINK* storage/maria/ma_loghandler.h: PAGECACHE_PAGE_LINK -> PAGECACHE_BLOCK_LINK* storage/maria/ma_pagecache.c: PAGECACHE_PAGE_LINK -> PAGECACHE_BLOCK_LINK* storage/maria/ma_pagecache.h: PAGECACHE_PAGE_LINK -> PAGECACHE_BLOCK_LINK* storage/maria/maria_def.h: PAGECACHE_PAGE_LINK -> PAGECACHE_BLOCK_LINK* --- storage/maria/ma_loghandler.c | 30 +++++++++++++++++------------- storage/maria/ma_loghandler.h | 4 ++-- storage/maria/ma_pagecache.c | 13 ++++++------- storage/maria/ma_pagecache.h | 6 ++---- storage/maria/maria_def.h | 2 +- 5 files changed, 28 insertions(+), 27 deletions(-) diff --git a/storage/maria/ma_loghandler.c b/storage/maria/ma_loghandler.c index e30c8c6627f..e52be8be427 100644 --- a/storage/maria/ma_loghandler.c +++ b/storage/maria/ma_loghandler.c @@ -640,22 +640,25 @@ static void translog_check_cursor(struct st_buffer_cursor *cursor) translog_filename_by_fileno() file_no Number of the log we want to open path Pointer to buffer where file name will be - stored (must be FN_REFLEN bytes at least + stored (must be FN_REFLEN bytes at least) RETURN pointer to path */ static char *translog_filename_by_fileno(uint32 file_no, char *path) { - char file_name[10 + 8 + 1]; /* See fallowing my_sprintf() call */ - char *res; + char buff[11], *end; + uint length; DBUG_ENTER("translog_filename_by_fileno"); DBUG_ASSERT(file_no <= 0xfffffff); - my_sprintf(file_name, (file_name, "maria_log.%08u", file_no)); - res= fn_format(path, file_name, log_descriptor.directory, "", MYF(MY_WME)); - DBUG_PRINT("info", ("Path: '%s' path: 0x%lx res: 0x%lx", - res, (ulong) path, (ulong) res)); - DBUG_RETURN(res); + + /* log_descriptor.directory is already formated */ + end= strxmov(path, log_descriptor.directory, "maria_log.0000000", NullS); + length= (uint) (int10_to_str(file_no, buff, 10) - buff); + strmov(end-length+1, buff); + + DBUG_PRINT("info", ("Path: '%s' path: 0x%lx", path, (ulong) res)); + DBUG_RETURN(path); } @@ -986,10 +989,11 @@ static void translog_mark_file_finished(uint32 file) { int i; struct st_file_counter *fc_ptr; - DBUG_ENTER("translog_mark_file_finished"); DBUG_PRINT("enter", ("file: %lu", (ulong) file)); + LINT_INIT(fc_ptr); + pthread_mutex_lock(&log_descriptor.unfinished_files_lock); DBUG_ASSERT(log_descriptor.unfinished_files.elements > 0); @@ -2310,7 +2314,7 @@ my_bool translog_unlock() */ static uchar *translog_get_page(TRANSLOG_VALIDATOR_DATA *data, uchar *buffer, - PAGECACHE_PAGE_LINK *direct_link) + PAGECACHE_BLOCK_LINK **direct_link) { TRANSLOG_ADDRESS addr= *(data->addr), in_buffers; uint cache_index; @@ -2490,7 +2494,7 @@ static uchar *translog_get_page(TRANSLOG_VALIDATOR_DATA *data, uchar *buffer, */ -static void translog_free_link(PAGECACHE_PAGE_LINK *direct_link) +static void translog_free_link(PAGECACHE_BLOCK_LINK *direct_link) { DBUG_ENTER("translog_free_link"); DBUG_PRINT("info", ("Direct link: 0x%lx", @@ -5692,7 +5696,7 @@ int translog_read_record_header(LSN lsn, TRANSLOG_HEADER_BUFFER *buff) { uchar buffer[TRANSLOG_PAGE_SIZE], *page; translog_size_t res, page_offset= LSN_OFFSET(lsn) % TRANSLOG_PAGE_SIZE; - PAGECACHE_PAGE_LINK direct_link; + PAGECACHE_BLOCK_LINK *direct_link; TRANSLOG_ADDRESS addr; TRANSLOG_VALIDATOR_DATA data; DBUG_ENTER("translog_read_record_header"); @@ -5963,7 +5967,7 @@ static void translog_destroy_reader_data(struct st_translog_reader_data *data) SYNOPSIS translog_read_record_header() lsn log record serial number (address of the record) - offset From the beginning of the record beginning (read§ + offset From the beginning of the record beginning (read by translog_read_record_header). length Length of record part which have to be read. buffer Buffer where to read the record part (have to be at diff --git a/storage/maria/ma_loghandler.h b/storage/maria/ma_loghandler.h index c53132916ec..b13984d88ef 100644 --- a/storage/maria/ma_loghandler.h +++ b/storage/maria/ma_loghandler.h @@ -182,8 +182,8 @@ typedef struct st_translog_scanner_data TRANSLOG_ADDRESS horizon; TRANSLOG_ADDRESS last_file_page; /* Last page on in this file */ uchar *page; /* page content pointer */ - /* direct link on the current page or NULL if it is not supported/requested */ - PAGECACHE_PAGE_LINK direct_link; + /* direct link on the current page or NULL if not supported/requested */ + PAGECACHE_BLOCK_LINK *direct_link; /* offset of the chunk in the page */ translog_size_t page_offset; /* set horizon only once at init */ diff --git a/storage/maria/ma_pagecache.c b/storage/maria/ma_pagecache.c index 27c9ceaa0f1..4b0ccd2c9c8 100755 --- a/storage/maria/ma_pagecache.c +++ b/storage/maria/ma_pagecache.c @@ -2755,10 +2755,9 @@ void pagecache_unlock_by_link(PAGECACHE *pagecache, */ void pagecache_unpin_by_link(PAGECACHE *pagecache, - PAGECACHE_PAGE_LINK *link, + PAGECACHE_BLOCK_LINK *block, LSN lsn) { - PAGECACHE_BLOCK_LINK *block= (PAGECACHE_BLOCK_LINK *)link; DBUG_ENTER("pagecache_unpin_by_link"); DBUG_PRINT("enter", ("block: 0x%lx fd: %u page: %lu", (ulong) block, @@ -3212,13 +3211,13 @@ my_bool pagecache_write_part(PAGECACHE *pagecache, enum pagecache_page_lock lock, enum pagecache_page_pin pin, enum pagecache_write_mode write_mode, - PAGECACHE_PAGE_LINK *link, + PAGECACHE_BLOCK_LINK **link, uint offset, uint size, pagecache_disk_read_validator validator, uchar* validator_data) { PAGECACHE_BLOCK_LINK *block= NULL; - PAGECACHE_PAGE_LINK fake_link; + PAGECACHE_BLOCK_LINK *fake_link; int error= 0; int need_lock_change= write_lock_change_table[lock].need_lock_change; DBUG_ENTER("pagecache_write_part"); @@ -3234,10 +3233,10 @@ my_bool pagecache_write_part(PAGECACHE *pagecache, DBUG_ASSERT(lock != PAGECACHE_LOCK_LEFT_READLOCKED); DBUG_ASSERT(lock != PAGECACHE_LOCK_READ_UNLOCK); DBUG_ASSERT(offset + size <= pagecache->block_size); + if (!link) link= &fake_link; - else - *link= 0; + *link= 0; restart: @@ -3362,7 +3361,7 @@ restart: if (pin == PAGECACHE_PIN_LEFT_UNPINNED || pin == PAGECACHE_UNPIN) unreg_request(pagecache, block, 1); else - *link= (PAGECACHE_PAGE_LINK)block; + *link= block; if (block->status & PCBLOCK_ERROR) error= 1; diff --git a/storage/maria/ma_pagecache.h b/storage/maria/ma_pagecache.h index 7318cc08040..23a443a5b97 100644 --- a/storage/maria/ma_pagecache.h +++ b/storage/maria/ma_pagecache.h @@ -91,8 +91,6 @@ typedef struct st_pagecache_page PAGECACHE_PAGE; struct st_pagecache_hash_link; typedef struct st_pagecache_hash_link PAGECACHE_HASH_LINK; -typedef PAGECACHE_BLOCK_LINK * PAGECACHE_PAGE_LINK; /* To be removed */ - #include typedef my_bool (*pagecache_disk_read_validator)(uchar *page, uchar *data); @@ -205,7 +203,7 @@ extern my_bool pagecache_write_part(PAGECACHE *pagecache, enum pagecache_page_lock lock, enum pagecache_page_pin pin, enum pagecache_write_mode write_mode, - PAGECACHE_PAGE_LINK *link, + PAGECACHE_BLOCK_LINK **link, uint offset, uint size, pagecache_disk_read_validator validator, @@ -228,7 +226,7 @@ extern void pagecache_unpin(PAGECACHE *pagecache, pgcache_page_no_t pageno, LSN lsn); extern void pagecache_unpin_by_link(PAGECACHE *pagecache, - PAGECACHE_PAGE_LINK *link, + PAGECACHE_BLOCK_LINK *link, LSN lsn); extern int flush_pagecache_blocks(PAGECACHE *keycache, PAGECACHE_FILE *file, diff --git a/storage/maria/maria_def.h b/storage/maria/maria_def.h index adcb35a5e0e..bac83db4f51 100644 --- a/storage/maria/maria_def.h +++ b/storage/maria/maria_def.h @@ -632,7 +632,7 @@ typedef struct st_maria_s_param /* Used to store reference to pinned page */ typedef struct st_pinned_page { - PAGECACHE_PAGE_LINK link; + PAGECACHE_BLOCK_LINK *link; enum pagecache_page_lock unlock; } MARIA_PINNED_PAGE; -- cgit v1.2.1 From 6f296a01ac39f7d10774fd3d93b5633f31ce99c7 Mon Sep 17 00:00:00 2001 From: unknown Date: Thu, 27 Sep 2007 20:00:40 +0300 Subject: Fixed merge problems. BitKeeper/etc/ignore: Added libmysql_r/client_settings.h libmysqld/scheduler.cc libmysqld/sql_connect.cc libmysqld/sql_tablespace.cc to the ignore list --- .bzrignore | 4 ++++ client/mysqltest.c | 1 - sql/slave.cc | 2 -- sql/sql_yacc.yy | 7 ------- storage/maria/ha_maria.cc | 35 ----------------------------------- storage/maria/ha_maria.h | 5 ----- 6 files changed, 4 insertions(+), 50 deletions(-) diff --git a/.bzrignore b/.bzrignore index 6001873606c..18b01f4714a 100644 --- a/.bzrignore +++ b/.bzrignore @@ -3056,3 +3056,7 @@ win/vs71cache.txt win/vs8cache.txt zlib/*.ds? zlib/*.vcproj +libmysql_r/client_settings.h +libmysqld/scheduler.cc +libmysqld/sql_connect.cc +libmysqld/sql_tablespace.cc diff --git a/client/mysqltest.c b/client/mysqltest.c index e530c235b41..fdee137fa98 100644 --- a/client/mysqltest.c +++ b/client/mysqltest.c @@ -64,7 +64,6 @@ enum { OPT_SKIP_SAFEMALLOC=OPT_MAX_CLIENT_OPTION, OPT_PS_PROTOCOL, OPT_SP_PROTOCOL, OPT_CURSOR_PROTOCOL, OPT_VIEW_PROTOCOL, OPT_MAX_CONNECT_RETRIES, OPT_MARK_PROGRESS, OPT_LOG_DIR, OPT_TAIL_LINES, - OPT_MAX_CONNECT_RETRIES, OPT_MARK_PROGRESS, OPT_LOG_DIR, OPT_GLOBAL_SUBST }; diff --git a/sql/slave.cc b/sql/slave.cc index 4c73853f5e8..fcbd4eb841b 100644 --- a/sql/slave.cc +++ b/sql/slave.cc @@ -29,8 +29,6 @@ #include #include -static Log_event* next_event(RELAY_LOG_INFO* rli); - #ifdef HAVE_REPLICATION #include "rpl_tblmap.h" diff --git a/sql/sql_yacc.yy b/sql/sql_yacc.yy index a674a6702ed..92d938c6b79 100644 --- a/sql/sql_yacc.yy +++ b/sql/sql_yacc.yy @@ -4499,13 +4499,6 @@ create_table_option: Lex->create_info.used_fields|= HA_CREATE_USED_KEY_BLOCK_SIZE; Lex->create_info.key_block_size= $3; } - | TRANSACTIONAL_SYM opt_equal ulong_num - { - Lex->create_info.used_fields|= HA_CREATE_USED_TRANSACTIONAL; - Lex->create_info.transactional= ($3 != 0 ? HA_CHOICE_YES : - HA_CHOICE_NO); - } - | TRANSACTIONAL_SYM opt_equal ulong_num { Lex->create_info.used_fields|= HA_CREATE_USED_TRANSACTIONAL; diff --git a/storage/maria/ha_maria.cc b/storage/maria/ha_maria.cc index 678b88063db..362e89bb088 100644 --- a/storage/maria/ha_maria.cc +++ b/storage/maria/ha_maria.cc @@ -686,41 +686,6 @@ err: } #endif /* HAVE_REPLICATION */ - -bool ha_maria::check_if_locking_is_allowed(uint sql_command, - ulong type, TABLE *table, - uint count, uint current, - uint *system_count, - bool called_by_privileged_thread) -{ - /* - To be able to open and lock for reading system tables like 'mysql.proc', - when we already have some tables opened and locked, and avoid deadlocks - we have to disallow write-locking of these tables with any other tables. - */ - if (table->s->system_table && - table->reginfo.lock_type >= TL_WRITE_ALLOW_WRITE) - (*system_count)++; - - /* 'current' is an index, that's why '<=' below. */ - if (*system_count > 0 && *system_count <= current) - { - my_error(ER_WRONG_LOCK_OF_SYSTEM_TABLE, MYF(0)); - return FALSE; - } - - /* - Deny locking of the log tables, which is incompatible with - concurrent insert. Unless called from a logger THD (general_log_thd - or slow_log_thd) or by a privileged thread. - */ - if (!called_by_privileged_thread) - return check_if_log_table_locking_is_allowed(sql_command, type, table); - - return TRUE; -} - - /* Name is here without an extension */ int ha_maria::open(const char *name, int mode, uint test_if_locked) diff --git a/storage/maria/ha_maria.h b/storage/maria/ha_maria.h index 7675778ab5b..2813e2ce037 100644 --- a/storage/maria/ha_maria.h +++ b/storage/maria/ha_maria.h @@ -66,11 +66,6 @@ public: uint checksum() const; virtual double scan_time(); - virtual bool check_if_locking_is_allowed(uint sql_command, - ulong type, TABLE * table, - uint count, uint current, - uint *system_count, - bool called_by_logger_thread); int open(const char *name, int mode, uint test_if_locked); int close(void); int write_row(uchar * buf); -- cgit v1.2.1 From d097806323618aef56bcfb1968b036be2aa8491a Mon Sep 17 00:00:00 2001 From: unknown Date: Thu, 27 Sep 2007 21:45:45 +0300 Subject: fix typo --- storage/maria/ma_loghandler.c | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/storage/maria/ma_loghandler.c b/storage/maria/ma_loghandler.c index e52be8be427..3ac25879619 100644 --- a/storage/maria/ma_loghandler.c +++ b/storage/maria/ma_loghandler.c @@ -657,7 +657,7 @@ static char *translog_filename_by_fileno(uint32 file_no, char *path) length= (uint) (int10_to_str(file_no, buff, 10) - buff); strmov(end-length+1, buff); - DBUG_PRINT("info", ("Path: '%s' path: 0x%lx", path, (ulong) res)); + DBUG_PRINT("info", ("Path: '%s' path: 0x%lx", path, (ulong) path)); DBUG_RETURN(path); } -- cgit v1.2.1 From e71e5893ee1b8cf0328234667123d42727d9bf01 Mon Sep 17 00:00:00 2001 From: unknown Date: Mon, 1 Oct 2007 09:59:05 +0300 Subject: Incorrect function call fixed. --- storage/maria/ma_recovery.c | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/storage/maria/ma_recovery.c b/storage/maria/ma_recovery.c index d3aaa64bbda..72247728fab 100644 --- a/storage/maria/ma_recovery.c +++ b/storage/maria/ma_recovery.c @@ -1558,7 +1558,7 @@ static int run_redo_phase(LSN lsn, my_bool apply) display_record_position(log_desc2, &rec2, 0); if (apply && display_and_apply_record(log_desc2, &rec2)) { - translog_destroy(&scanner2); + translog_destroy_scanner(&scanner2); goto err; } } -- cgit v1.2.1 From 3c3e3648db433e11a783ea561792f523151d8b37 Mon Sep 17 00:00:00 2001 From: unknown Date: Mon, 1 Oct 2007 18:48:20 +0300 Subject: fater way to detect last page address for the last log file. gprof build for amd64. storage/maria/ma_loghandler.c: fater way to detect last page address for the last log file. BUILD/compile-amd64-gprof-no-ndb: New BitKeeper file ``BUILD/compile-amd64-gprof-no-ndb'' --- BUILD/compile-amd64-gprof-no-ndb | 7 +++++++ storage/maria/ma_loghandler.c | 8 ++++++++ 2 files changed, 15 insertions(+) create mode 100755 BUILD/compile-amd64-gprof-no-ndb diff --git a/BUILD/compile-amd64-gprof-no-ndb b/BUILD/compile-amd64-gprof-no-ndb new file mode 100755 index 00000000000..9fd4c67155c --- /dev/null +++ b/BUILD/compile-amd64-gprof-no-ndb @@ -0,0 +1,7 @@ +#! /bin/sh +path=`dirname $0` +. "$path/SETUP.sh" +extra_flags="$amd64_cflags -pg -g" +extra_configs="$amd64_configs $max_no_ndb_configs --disable-shared $static_link" + +. "$path/FINISH.sh" diff --git a/storage/maria/ma_loghandler.c b/storage/maria/ma_loghandler.c index 3ac25879619..0b064c5cfe2 100644 --- a/storage/maria/ma_loghandler.c +++ b/storage/maria/ma_loghandler.c @@ -5185,6 +5185,14 @@ static my_bool translog_scanner_set_last_page(TRANSLOG_SCANNER_DATA *scanner) { my_bool page_ok; + if (LSN_FILE_NO(scanner->page_addr) == LSN_FILE_NO(scanner->horizon)) + { + /* It is last file => we can easy find last page address by horizon */ + uint pagegrest= LSN_OFFSET(scanner->horizon) % TRANSLOG_PAGE_SIZE; + scanner->last_file_page= (scanner->horizon - + (pagegrest ? pagegrest : TRANSLOG_PAGE_SIZE)); + return (0); + } scanner->last_file_page= scanner->page_addr; return (translog_get_last_page_addr(&scanner->last_file_page, &page_ok)); } -- cgit v1.2.1 From ca12435fe11e558e548b5178ff69339336b7acc3 Mon Sep 17 00:00:00 2001 From: unknown Date: Mon, 1 Oct 2007 19:39:16 +0200 Subject: Fix for "innodb_mysql" and "events" failures: we can disable transactionality in CREATE SELECT only if the table is not temporary (because re-enabling causes a commit). In the future we should disable again for temporary tables; that will probably require changing ha_enable_transaction(). sql/sql_insert.cc: When we disable transactionality in CREATE SELECT, we re-enable it at the end and this causes a commit (inside ha_enable_transaction()); but this is undesired if the created table is temporary (we don't want CREATE TEMPORARY TABLE SELECT to commit all previous statements). So we disable logging only if the table is not temporary. Ideally in the future we would want to lift this restriction which sounds stupid, but for Maria it does not matter now (temporary tables are not transactional yet). --- sql/sql_insert.cc | 18 +++++++++++------- 1 file changed, 11 insertions(+), 7 deletions(-) diff --git a/sql/sql_insert.cc b/sql/sql_insert.cc index d7267e6ecf6..ecaab5638cc 100644 --- a/sql/sql_insert.cc +++ b/sql/sql_insert.cc @@ -3460,10 +3460,12 @@ select_create::prepare(List &values, SELECT_LEX_UNIT *u) /* If error during the CREATE SELECT we drop the table, so no need for - engines to do logging of insertions (optimization). + engines to do logging of insertions (optimization). We don't do it for + temporary tables (yet) as re-enabling causes an undesirable commit. */ - if (ha_enable_transaction(thd, FALSE)) - DBUG_RETURN(-1); + if (((thd->lex->create_info.options & HA_LEX_CREATE_TMP_TABLE) == 0) && + ha_enable_transaction(thd, FALSE)) + DBUG_RETURN(-1); if (!(table= create_table_from_items(thd, create_info, create_table, alter_info, &values, @@ -3605,11 +3607,12 @@ bool select_create::send_eof() nevertheless. */ if (!table->s->tmp_table) + { + ha_enable_transaction(thd, TRUE); ha_commit(thd); // Can fail, but we proceed anyway - + } table->file->extra(HA_EXTRA_NO_IGNORE_DUP_KEY); table->file->extra(HA_EXTRA_WRITE_CANNOT_REPLACE); - ha_enable_transaction(thd, TRUE); if (thd->extra_lock) { mysql_unlock_tables(thd, thd->extra_lock); @@ -3632,6 +3635,9 @@ void select_create::abort() select_insert::abort(); reenable_binlog(thd); + if (table && !table->s->tmp_table) + ha_enable_transaction(thd, TRUE); + /* We roll back the statement, including truncating the transaction cache of the binary log, if the statement failed. @@ -3648,8 +3654,6 @@ void select_create::abort() if (thd->current_stmt_binlog_row_based) ha_rollback_stmt(thd); - ha_enable_transaction(thd, TRUE); - if (thd->extra_lock) { mysql_unlock_tables(thd, thd->extra_lock); -- cgit v1.2.1 From 48ed54c1403a1b1d5aacb68978aa59d60b4015ba Mon Sep 17 00:00:00 2001 From: unknown Date: Mon, 1 Oct 2007 23:51:57 +0300 Subject: Fixed problem with scanner destruction (unpinned page in the cache). storage/maria/ma_loghandler.c: Debug output adder to control scanner freeing. Fixed problem with scanner destruction (unpinned page in the cache). --- storage/maria/ma_loghandler.c | 11 +++++++++-- 1 file changed, 9 insertions(+), 2 deletions(-) diff --git a/storage/maria/ma_loghandler.c b/storage/maria/ma_loghandler.c index 0b064c5cfe2..3a94be3a16a 100644 --- a/storage/maria/ma_loghandler.c +++ b/storage/maria/ma_loghandler.c @@ -5249,7 +5249,8 @@ my_bool translog_init_scanner(LSN lsn, { TRANSLOG_VALIDATOR_DATA data; DBUG_ENTER("translog_init_scanner"); - DBUG_PRINT("enter", ("LSN: (0x%lu,0x%lx)", LSN_IN_PARTS(lsn))); + DBUG_PRINT("enter", ("Scanner: 0x%lx LSN: (0x%lu,0x%lx)", + (ulong) scanner, LSN_IN_PARTS(lsn))); DBUG_ASSERT(LSN_OFFSET(lsn) % TRANSLOG_PAGE_SIZE != 0); DBUG_ASSERT(translog_inited == 1); @@ -5289,7 +5290,10 @@ my_bool translog_init_scanner(LSN lsn, void translog_destroy_scanner(TRANSLOG_SCANNER_DATA *scanner) { + DBUG_ENTER("translog_destroy_scanner"); + DBUG_PRINT("enter", ("Scanner: 0x%lx", (ulong)scanner)); translog_free_link(scanner->direct_link); + DBUG_VOID_RETURN; } @@ -5587,7 +5591,10 @@ translog_variable_length_header(uchar *page, translog_size_t page_offset, DBUG_PRINT("info", ("use internal scanner")); scanner= &internal_scanner; } - + else + { + translog_destroy_scanner(scanner); + } base_lsn= buff->groups[0].addr; translog_init_scanner(base_lsn, 1, scanner, scanner == &internal_scanner); /* first group chunk is always chunk type 2 */ -- cgit v1.2.1 From d0b9387b883dc30b6c89f4e769c74338e22cfc58 Mon Sep 17 00:00:00 2001 From: unknown Date: Tue, 2 Oct 2007 18:02:09 +0200 Subject: WL#3072 - Maria recovery. * Recovery of the table's live checksum (CREATE TABLE ... CHECKSUM=1) is achieved in this patch. The table's live checksum (info->s->state.state.checksum) is updated in inwrite_rec_hook's under the log mutex when writing UNDO_ROW_INSERT|UPDATE|DELETE and REDO_DELETE_ALL. The checksum variation caused by the operation is stored in these UNDOs, so that the REDO phase, when it sees such UNDOs, can update the live checksum if it is older (state.is_of_lsn is lower) than the record. It is also used, as a nice add-on with no cost, to do less row checksum computation during the UNDO phase (as we have it in the record already). Doing this work, it became pressing to move in-write hooks (write_hook_for_redo() et al) to ma_blockrec.c. The 'parts' argument of inwrite_rec_hook is unpredictable (it comes mangled at this stage, for example by LSN compression) so it is replaced by a 'void* hook_arg', which is used to pass down information, currently only to write_hook_for_clr_end() (previous undo_lsn and type of undone record). * If from ha_maria, we print to stderr how many seconds (with one fractional digit) the REDO phase took, same for UNDO phase and for final table close. Just to give an indication for debugging and maybe also for Support. storage/maria/ha_maria.cc: question for Monty storage/maria/ma_blockrec.c: * log in-write hooks (write_hook_for_redo() etc) move from ma_loghandler.c to here; this is natural: the hooks are coupled to their callers (functions in ma_blockrec.c). * translog_write_record() now has a new argument "hook_arg"; using it to pass down to write_hook_for_clr_end() the transaction's previous_undo_lsn and the type of the being undone record, and also to pass down to all UNDOs the live checksum variation caused by the operation. * If table has live checksum, store in UNDO_ROW_INSERT|UPDATE|DELETE and in CLR_END the checksum variation ("delta") caused by the operation. For example if a DELETE caused the table's live checksum to change from 123 to 456, we store in the UNDO_ROW_DELETE, in 4 bytes, the value 333 (456-123). * Instead of hard-coded "1" as length of the place where we store the undone record's type in CLR_END, use a symbol CLR_TYPE_STORE_SIZE; use macros clr_type_store and clr_type_korr. * write_block_record() has a new parameter 'old_record_checksum' which is the pre-computed checksum of old_record; that value is used to update the table's live checksum when writing UNDO_ROW_UPDATE|CLR_END. * In allocate_write_block_record(), if we are executing UNDO_ROW_DELETE the row's checksum is already computed. * _ma_update_block_record2() now expect the new row's checksum into cur_row.checksum (was already true) and the old row's checksum into new_row.checksum (that's new). Its two callers, maria_update() and _ma_apply_undo_row_update(), honour this. * When executing an UNDO_ROW_INSERT|UPDATE|DELETE in UNDO phase, pick up the checksum delta from the log record. It is then used to update the table's live checksum when writing CLR_END, and saves us a computation of record. storage/maria/ma_blockrec.h: in-write hooks move from ma_loghandler.c storage/maria/ma_check.c: more straightforward size of buffer storage/maria/ma_checkpoint.c: <= is enough storage/maria/ma_commit.c: new prototype of translog_write_record() storage/maria/ma_create.c: new prototype of translog_write_record() storage/maria/ma_delete.c: The row's checksum must be computed before calling(*delete_record)(), not after, because it must be known inside _ma_delete_block_record() (to update the table's live checksum when writing UNDO_ROW_DELETE). If deleting from a transactional table, live checksum was already updated when writing UNDO_ROW_DELETE. storage/maria/ma_delete_all.c: @todo is now done (in ma_loghandler.c) storage/maria/ma_delete_table.c: new prototype of translog_write_record() storage/maria/ma_loghandler.c: * in-write hooks move to ma_blockrec.c. * translog_write_record() gets a new argument 'hook_arg' which is passed down to pre|inwrite_rec_hook. It is more useful that 'parts' for those hooks, because when those hooks are called, 'parts' has possibly been mangled (like with LSN compression) and is so unpredictable. * fix for compiler warning (unused buffer_start when compiling without debug support) * Because checksum delta is stored into UNDO_ROW_INSERT|UPDATE|DELETE and CLR_END, but only if the table has live checksum, these records are not PSEUDOFIXEDLENGTH anymore, they are now VARIABLE_LENGTH (their length is X if no live checksum and X+4 otherwise). * add an inwrite_rec_hook for UNDO_ROW_UPDATE, which updates the table's live checksum. Update it also in hooks of UNDO_ROW_INSERT| DELETE and REDO_DELETE_ALL and CLR_END. * Bugfix: when reading a record in translog_read_record(), it happened that "length" became negative, because the function assumed that the record extended beyond the page's end, whereas it may be shorter. storage/maria/ma_loghandler.h: * Instead of hard-coded "1" and "4", use symbols and macros to store/retrieve the type of record which the CLR_END corresponds to, and the checksum variation caused by the operation which logs the record * translog_write_record() gets a new argument 'hook_arg' which is passed down to pre|inwrite_rec_hook. It is more useful that 'parts' for those hooks, because when those hooks are called, 'parts' has possibly been mangled (like with LSN compression) and is so unpredictable. storage/maria/ma_open.c: fix for "empty body in if() statement" (when compiling without safemutex) storage/maria/ma_pagecache.c: <= is enough storage/maria/ma_recovery.c: * print the time that each recovery phase (REDO/UNDO/flush) took; this is enabled only when recovering from ha_maria. Is it printed n seconds with a fractional part of one digit (like 123.4 seconds). * In the REDO phase, update the table's live checksum by using the checksum delta stored in UNDO_ROW_INSERT|DELETE|UPDATE and CLR_END. Update it too when seeing REDO_DELETE_ALL. * In the UNDO phase, when executing UNDO_ROW_INSERT, if the table does not have live checksum then reading the record's header (as done by the master loop of run_undo_phase()) is enough; otherwise we do a translog_read_record() to have the checksum delta ready for _ma_apply_undo_row_insert(). * When at the end of the REDO phase we notice that there is an unfinished group of REDOs, don't assert in debug binaries, as I verified that it can happen in real life (with kill -9) * removing ' in #error as it confuses gcc3 storage/maria/ma_rename.c: new prototype of translog_write_record() storage/maria/ma_test_recovery.expected: Change in output of ma_test_recovery: now all live checksums of original tables equal those of tables recreated by the REDO phase and those of tables fixed by the UNDO phase. I.e. recovery of the live checksum looks like working (which was after all the only goal of this changeset). I checked by hand that it's not just all live checksums which are now 0 and that's why they match. They are the old values like 3757530372. maria.test has hard-coded checksum values in its result file so checks this too. storage/maria/ma_update.c: * It's useless to put up HA_STATE_CHANGED in 'key_changed', as we put up HA_STATE_CHANGED in info->update anyway. * We need to compute the old and new rows' checksum before calling (*update_record)(), as checksum delta must be known when logging UNDO_ROW_UPDATE which is done by _ma_update_block_record(). Note that some functions change the 'newrec' record (at least _ma_check_unique() does) so we cannot move the checksum computation too early in the function. storage/maria/ma_write.c: If inserting into a transactional table, live's checksum was already updated when writing UNDO_ROW_INSERT. The multiplication is a trick to save an if(). storage/maria/unittest/ma_test_loghandler-t.c: new prototype of translog_write_record() storage/maria/unittest/ma_test_loghandler_first_lsn-t.c: new prototype of translog_write_record() storage/maria/unittest/ma_test_loghandler_max_lsn-t.c: new prototype of translog_write_record() storage/maria/unittest/ma_test_loghandler_multigroup-t.c: new prototype of translog_write_record() storage/maria/unittest/ma_test_loghandler_multithread-t.c: new prototype of translog_write_record() storage/maria/unittest/ma_test_loghandler_noflush-t.c: new prototype of translog_write_record() storage/maria/unittest/ma_test_loghandler_pagecache-t.c: new prototype of translog_write_record() storage/maria/unittest/ma_test_loghandler_purge-t.c: new prototype of translog_write_record() storage/myisam/sort.c: fix for compiler warnings in pushbuild (write_merge_key* functions didn't have their declaration match MARIA_HA::write_key). --- storage/maria/ha_maria.cc | 3 + storage/maria/ma_blockrec.c | 463 ++++++++++++++++++--- storage/maria/ma_blockrec.h | 25 ++ storage/maria/ma_check.c | 10 +- storage/maria/ma_checkpoint.c | 6 +- storage/maria/ma_commit.c | 2 +- storage/maria/ma_create.c | 2 +- storage/maria/ma_delete.c | 41 +- storage/maria/ma_delete_all.c | 9 +- storage/maria/ma_delete_table.c | 2 +- storage/maria/ma_loghandler.c | 424 +++++-------------- storage/maria/ma_loghandler.h | 16 +- storage/maria/ma_open.c | 2 +- storage/maria/ma_pagecache.c | 6 +- storage/maria/ma_recovery.c | 145 +++++-- storage/maria/ma_rename.c | 2 +- storage/maria/ma_test_recovery.expected | 176 -------- storage/maria/ma_update.c | 51 +-- storage/maria/ma_write.c | 7 +- storage/maria/unittest/ma_test_loghandler-t.c | 21 +- .../unittest/ma_test_loghandler_first_lsn-t.c | 2 +- .../maria/unittest/ma_test_loghandler_max_lsn-t.c | 2 +- .../unittest/ma_test_loghandler_multigroup-t.c | 24 +- .../unittest/ma_test_loghandler_multithread-t.c | 6 +- .../maria/unittest/ma_test_loghandler_noflush-t.c | 2 +- .../unittest/ma_test_loghandler_pagecache-t.c | 2 +- .../maria/unittest/ma_test_loghandler_purge-t.c | 8 +- storage/myisam/sort.c | 15 +- 28 files changed, 755 insertions(+), 719 deletions(-) diff --git a/storage/maria/ha_maria.cc b/storage/maria/ha_maria.cc index 678b88063db..2397eac068d 100644 --- a/storage/maria/ha_maria.cc +++ b/storage/maria/ha_maria.cc @@ -2185,6 +2185,9 @@ int ha_maria::create(const char *name, register TABLE *table_arg, error; ? Why fool the user? + Shouldn't this test be pushed down to maria_create()? Because currently, + ma_test1 -T crashes: it creates a table with DYNAMIC_RECORD but has + born_transactional==1, which confuses some recovery-related code. */ #endif create_info.transactional= (row_type == BLOCK_RECORD && diff --git a/storage/maria/ma_blockrec.c b/storage/maria/ma_blockrec.c index 3ec18d229c2..0d1a530dbfc 100644 --- a/storage/maria/ma_blockrec.c +++ b/storage/maria/ma_blockrec.c @@ -300,6 +300,32 @@ typedef struct st_maria_extent_cursor } MARIA_EXTENT_CURSOR; +/** + @brief Structure for passing down info to write_hook_for_clr_end(). + This hooks needs to know the variation of the live checksum caused by the + current operation to update state.checksum under log's mutex, + needs to know the transaction's previous undo_lsn to set + trn->undo_lsn under log mutex, and needs to know the type of UNDO being + undone now to modify state.records under log mutex. +*/ +struct st_msg_to_write_hook_for_clr_end +{ + LSN previous_undo_lsn; + enum translog_record_type undone_record_type; + ha_checksum checksum_delta; +}; +/** S:share,D:checksum_delta,E:expression,P:pointer_into_record,L:length */ +#define store_checksum_in_rec(S,D,E,P,L) do \ + { \ + D= 0; \ + if ((S)->calc_checksum != NULL) \ + { \ + D= (E); \ + ha_checksum_store(P, D); \ + L+= HA_CHECKSUM_STORE_SIZE; \ + } \ + } while (0) + static my_bool delete_tails(MARIA_HA *info, MARIA_RECORD_POS *tails); static my_bool delete_head_or_tail(MARIA_HA *info, ulonglong page, uint record_number, @@ -1387,7 +1413,7 @@ static my_bool write_tail(MARIA_HA *info, if (translog_write_record(&lsn, LOGREC_REDO_INSERT_ROW_TAIL, info->trn, info, sizeof(log_data) + length, TRANSLOG_INTERNAL_PARTS + 2, log_array, - log_data)) + log_data, NULL)) DBUG_RETURN(1); } @@ -1642,7 +1668,7 @@ static my_bool free_full_pages(MARIA_HA *info, MARIA_ROW *row) if (translog_write_record(&lsn, LOGREC_REDO_PURGE_BLOCKS, info->trn, info, sizeof(log_data) + extents_length, TRANSLOG_INTERNAL_PARTS + 2, log_array, - log_data)) + log_data, NULL)) DBUG_RETURN(1); DBUG_RETURN(_ma_bitmap_free_full_pages(info, row->extents, @@ -1689,7 +1715,7 @@ static my_bool free_full_page_range(MARIA_HA *info, ulonglong page, uint count) if (translog_write_record(&lsn, LOGREC_REDO_PURGE_BLOCKS, info->trn, info, sizeof(log_data), TRANSLOG_INTERNAL_PARTS + 1, log_array, - log_data)) + log_data, NULL)) res= 1; } @@ -1716,6 +1742,9 @@ static my_bool free_full_page_range(MARIA_HA *info, ulonglong page, uint count) @param row_pos Position on head page where to put head part of record @param undo_lsn <> LSN_ERROR if we are executing an UNDO + @param old_record_checksum Checksum of old_record: ignored if table does + not have live checksum; otherwise if + old_record==NULL it must be 0. @note On return all pinned pages are released. @@ -1731,7 +1760,8 @@ static my_bool write_block_record(MARIA_HA *info, MARIA_BITMAP_BLOCKS *bitmap_blocks, my_bool head_block_is_read, struct st_row_pos_info *row_pos, - LSN undo_lsn) + LSN undo_lsn, + ha_checksum old_record_checksum) { uchar *data, *end_of_data, *tmp_data_used, *tmp_data; uchar *row_extents_first_part, *row_extents_second_part; @@ -1785,7 +1815,10 @@ static my_bool write_block_record(MARIA_HA *info, if (share->base.pack_fields) store_key_length_inc(data, row->field_lengths_length); if (share->calc_checksum) + { *(data++)= (uchar) (row->checksum); /* store least significant byte */ + DBUG_ASSERT(!((old_record_checksum != 0) && (old_record == NULL))); + } memcpy(data, record, share->base.null_bytes); data+= share->base.null_bytes; memcpy(data, row->empty_bits, share->base.pack_bytes); @@ -2211,7 +2244,7 @@ static my_bool write_block_record(MARIA_HA *info, if (translog_write_record(&lsn, LOGREC_REDO_INSERT_ROW_HEAD, info->trn, info, sizeof(log_data) + data_length, TRANSLOG_INTERNAL_PARTS + 2, log_array, - log_data)) + log_data, NULL)) goto disk_err; } @@ -2328,7 +2361,7 @@ static my_bool write_block_record(MARIA_HA *info, error= translog_write_record(&lsn, LOGREC_REDO_INSERT_ROW_BLOBS, info->trn, info, log_entry_length, (uint) (log_array_pos - log_array), - log_array, log_data); + log_array, log_data, NULL); if (log_array != tmp_log_array) my_free((uchar*) log_array, MYF(0)); if (error) @@ -2343,31 +2376,44 @@ static my_bool write_block_record(MARIA_HA *info, if (undo_lsn != LSN_ERROR) { - uchar log_data[LSN_STORE_SIZE + FILEID_STORE_SIZE + 1]; + uchar log_data[LSN_STORE_SIZE + FILEID_STORE_SIZE + + CLR_TYPE_STORE_SIZE + HA_CHECKSUM_STORE_SIZE]; + struct st_msg_to_write_hook_for_clr_end msg; /* undo_lsn must be first for compression to work */ lsn_store(log_data, undo_lsn); /* - Store if this CLR is about an UNDO_INSERT, UNDO_DELETE or UNDO_UPDATE; - in the first/second case, Recovery, when it sees the CLR_END in the - REDO phase, may decrement/increment the records' count. + Store if this CLR is about UNDO_DELETE or UNDO_UPDATE; + in the first case, Recovery, when it sees the CLR_END in the + REDO phase, may decrement the records' count. */ - log_data[LSN_STORE_SIZE + FILEID_STORE_SIZE]= old_record ? - LOGREC_UNDO_ROW_UPDATE : LOGREC_UNDO_ROW_DELETE; log_array[TRANSLOG_INTERNAL_PARTS + 0].str= (char*) log_data; - log_array[TRANSLOG_INTERNAL_PARTS + 0].length= sizeof(log_data); - + log_array[TRANSLOG_INTERNAL_PARTS + 0].length= + sizeof(log_data) - HA_CHECKSUM_STORE_SIZE; + msg.undone_record_type= + old_record ? LOGREC_UNDO_ROW_UPDATE : LOGREC_UNDO_ROW_DELETE; + clr_type_store(log_data + LSN_STORE_SIZE + FILEID_STORE_SIZE, + msg.undone_record_type); + msg.previous_undo_lsn= undo_lsn; + store_checksum_in_rec(share, msg.checksum_delta, + row->checksum - old_record_checksum, + log_data + LSN_STORE_SIZE + FILEID_STORE_SIZE + + CLR_TYPE_STORE_SIZE, + log_array[TRANSLOG_INTERNAL_PARTS + 0].length); if (translog_write_record(&lsn, LOGREC_CLR_END, - info->trn, info, sizeof(log_data), + info->trn, info, + log_array[TRANSLOG_INTERNAL_PARTS + 0].length, TRANSLOG_INTERNAL_PARTS + 1, log_array, - log_data + LSN_STORE_SIZE)) + log_data + LSN_STORE_SIZE, &msg)) goto disk_err; } else { uchar log_data[LSN_STORE_SIZE + FILEID_STORE_SIZE + - PAGE_STORE_SIZE + DIRPOS_STORE_SIZE]; + PAGE_STORE_SIZE + DIRPOS_STORE_SIZE + + HA_CHECKSUM_STORE_SIZE]; + ha_checksum checksum_delta; - /* LOGREC_UNDO_ROW_INSERT & LOGREC_UNDO_ROW_INSERT share same header */ + /* LOGREC_UNDO_ROW_INSERT & LOGREC_UNDO_ROW_UPDATE share same header */ lsn_store(log_data, info->trn->undo_lsn); page_store(log_data + LSN_STORE_SIZE + FILEID_STORE_SIZE, head_block->page); @@ -2376,15 +2422,24 @@ static my_bool write_block_record(MARIA_HA *info, row_pos->rownr); log_array[TRANSLOG_INTERNAL_PARTS + 0].str= (char*) log_data; - log_array[TRANSLOG_INTERNAL_PARTS + 0].length= sizeof(log_data); + log_array[TRANSLOG_INTERNAL_PARTS + 0].length= + sizeof(log_data) - HA_CHECKSUM_STORE_SIZE; + store_checksum_in_rec(share, checksum_delta, + row->checksum - old_record_checksum, + log_data + LSN_STORE_SIZE + FILEID_STORE_SIZE + + PAGE_STORE_SIZE + DIRPOS_STORE_SIZE, + log_array[TRANSLOG_INTERNAL_PARTS + 0].length); + compile_time_assert(sizeof(ha_checksum) == HA_CHECKSUM_STORE_SIZE); if (!old_record) { /* Write UNDO log record for the INSERT */ if (translog_write_record(&lsn, LOGREC_UNDO_ROW_INSERT, - info->trn, info, sizeof(log_data), + info->trn, info, + log_array[TRANSLOG_INTERNAL_PARTS + + 0].length, TRANSLOG_INTERNAL_PARTS + 1, log_array, - log_data + LSN_STORE_SIZE)) + log_data + LSN_STORE_SIZE, &checksum_delta)) goto disk_err; } else @@ -2397,10 +2452,11 @@ static my_bool write_block_record(MARIA_HA *info, TRANSLOG_INTERNAL_PARTS + 1, &row_parts_count); if (translog_write_record(&lsn, LOGREC_UNDO_ROW_UPDATE, info->trn, - info, sizeof(log_data) + row_length, + info, log_array[TRANSLOG_INTERNAL_PARTS + + 0].length + row_length, TRANSLOG_INTERNAL_PARTS + 1 + - row_parts_count, - log_array, log_data + LSN_STORE_SIZE)) + row_parts_count, log_array, + log_data + LSN_STORE_SIZE, &checksum_delta)) goto disk_err; } } @@ -2517,10 +2573,18 @@ static my_bool allocate_and_write_block_record(MARIA_HA *info, DBUG_RETURN(1); row->lastpos= ma_recordpos(blocks->block->page, row_pos.rownr); if (info->s->calc_checksum) - row->checksum= (info->s->calc_checksum)(info, record); + { + if (undo_lsn == LSN_ERROR) + row->checksum= (info->s->calc_checksum)(info, record); + else + { + /* _ma_apply_undo_row_delete() already set row's checksum. Verify it. */ + DBUG_ASSERT(row->checksum == (info->s->calc_checksum)(info, record)); + } + } if (write_block_record(info, (uchar*) 0, record, row, blocks, blocks->block->org_bitmap_value != 0, - &row_pos, undo_lsn)) + &row_pos, undo_lsn, 0)) DBUG_RETURN(1); /* Error reading bitmap */ DBUG_PRINT("exit", ("Rowid: %lu (%lu:%u)", (ulong) row->lastpos, (ulong) ma_recordpos_to_page(row->lastpos), @@ -2592,6 +2656,7 @@ my_bool _ma_write_abort_block_record(MARIA_HA *info) MARIA_BITMAP_BLOCKS *blocks= &info->cur_row.insert_blocks; MARIA_BITMAP_BLOCK *block, *end; LSN lsn= LSN_IMPOSSIBLE; + MARIA_SHARE *share= info->s; DBUG_ENTER("_ma_write_abort_block_record"); if (delete_head_or_tail(info, @@ -2619,13 +2684,14 @@ my_bool _ma_write_abort_block_record(MARIA_HA *info) } } - if (info->s->now_transactional) + if (share->now_transactional) { - LSN previous_undo_lsn; TRANSLOG_HEADER_BUFFER rec; LEX_STRING log_array[TRANSLOG_INTERNAL_PARTS + 1]; - uchar log_data[LSN_STORE_SIZE + FILEID_STORE_SIZE + 1]; + uchar log_data[LSN_STORE_SIZE + FILEID_STORE_SIZE + + CLR_TYPE_STORE_SIZE + HA_CHECKSUM_STORE_SIZE]; int len; + struct st_msg_to_write_hook_for_clr_end msg; /* We do need the code above (delete_head_or_tail() etc) for non-transactional tables. @@ -2644,15 +2710,24 @@ my_bool _ma_write_abort_block_record(MARIA_HA *info) goto end; } DBUG_ASSERT(rec.type == LOGREC_UNDO_ROW_INSERT); - previous_undo_lsn= lsn_korr(rec.header); - lsn_store(log_data, previous_undo_lsn); - log_data[LSN_STORE_SIZE + FILEID_STORE_SIZE]= LOGREC_UNDO_ROW_INSERT; + memcpy(log_data, rec.header, LSN_STORE_SIZE); /* previous UNDO LSN */ + msg.previous_undo_lsn= lsn_korr(rec.header); + msg.undone_record_type= LOGREC_UNDO_ROW_INSERT; + clr_type_store(log_data + LSN_STORE_SIZE + FILEID_STORE_SIZE, + LOGREC_UNDO_ROW_INSERT); log_array[TRANSLOG_INTERNAL_PARTS + 0].str= (char*) log_data; - log_array[TRANSLOG_INTERNAL_PARTS + 0].length= sizeof(log_data); + log_array[TRANSLOG_INTERNAL_PARTS + 0].length= + sizeof(log_data) - HA_CHECKSUM_STORE_SIZE; + store_checksum_in_rec(share, msg.checksum_delta, + - info->cur_row.checksum, + log_data + LSN_STORE_SIZE + FILEID_STORE_SIZE + + CLR_TYPE_STORE_SIZE, + log_array[TRANSLOG_INTERNAL_PARTS + 0].length); if (translog_write_record(&lsn, LOGREC_CLR_END, - info->trn, info, sizeof(log_data), + info->trn, info, + log_array[TRANSLOG_INTERNAL_PARTS + 0].length, TRANSLOG_INTERNAL_PARTS + 1, log_array, - log_data + LSN_STORE_SIZE)) + log_data + LSN_STORE_SIZE, &msg)) res= 1; } end: @@ -2685,6 +2760,8 @@ static my_bool _ma_update_block_record2(MARIA_HA *info, uchar *dir; ulonglong page; struct st_row_pos_info row_pos; + my_bool res; + ha_checksum old_checksum; MARIA_SHARE *share= info->s; DBUG_ENTER("_ma_update_block_record2"); DBUG_PRINT("enter", ("rowid: %lu", (long) record_pos)); @@ -2694,7 +2771,11 @@ static my_bool _ma_update_block_record2(MARIA_HA *info, DBUG_DUMP("newrec", record, share->base.reclength); #endif - /* checksum was computed by maria_update() already and put into cur_row */ + /* + Checksums of new and old rows were computed by callers already; new + row's was put into cur_row, old row's was put into new_row. + */ + old_checksum= new_row->checksum; new_row->checksum= cur_row->checksum; calc_record_size(info, record, new_row); page= ma_recordpos_to_page(record_pos); @@ -2747,8 +2828,9 @@ static my_bool _ma_update_block_record2(MARIA_HA *info, if (cur_row->extents_count && free_full_pages(info, cur_row)) goto err; - DBUG_RETURN(write_block_record(info, oldrec, record, new_row, blocks, - 1, &row_pos, undo_lsn)); + res= write_block_record(info, oldrec, record, new_row, blocks, + 1, &row_pos, undo_lsn, old_checksum); + DBUG_RETURN(res); } /* Allocate all size in block for record @@ -2781,8 +2863,9 @@ static my_bool _ma_update_block_record2(MARIA_HA *info, row_pos.dir= dir; row_pos.data= buff + uint2korr(dir); row_pos.length= head_length; - DBUG_RETURN(write_block_record(info, oldrec, record, new_row, blocks, 1, - &row_pos, undo_lsn)); + res= write_block_record(info, oldrec, record, new_row, blocks, 1, + &row_pos, undo_lsn, old_checksum); + DBUG_RETURN(res); err: _ma_unpin_all_pages_and_finalize_row(info, 0); @@ -2949,7 +3032,7 @@ static my_bool delete_head_or_tail(MARIA_HA *info, LOGREC_REDO_PURGE_ROW_TAIL), info->trn, info, sizeof(log_data), TRANSLOG_INTERNAL_PARTS + 1, log_array, - log_data)) + log_data, NULL)) DBUG_RETURN(1); } if (pagecache_write(share->pagecache, @@ -2976,7 +3059,7 @@ static my_bool delete_head_or_tail(MARIA_HA *info, if (translog_write_record(&lsn, LOGREC_REDO_PURGE_BLOCKS, info->trn, info, sizeof(log_data), TRANSLOG_INTERNAL_PARTS + 1, log_array, - log_data)) + log_data, NULL)) DBUG_RETURN(1); } /* Write the empty page (needed only for REPAIR to work) */ @@ -3044,6 +3127,7 @@ my_bool _ma_delete_block_record(MARIA_HA *info, const uchar *record) { ulonglong page; uint record_number; + MARIA_SHARE *share= info->s; DBUG_ENTER("_ma_delete_block_record"); page= ma_recordpos_to_page(info->cur_row.lastpos); @@ -3058,13 +3142,14 @@ my_bool _ma_delete_block_record(MARIA_HA *info, const uchar *record) if (info->cur_row.extents && free_full_pages(info, &info->cur_row)) goto err; - if (info->s->now_transactional) + if (share->now_transactional) { LSN lsn; uchar log_data[LSN_STORE_SIZE + FILEID_STORE_SIZE + PAGE_STORE_SIZE + - DIR_COUNT_SIZE]; + DIRPOS_STORE_SIZE + HA_CHECKSUM_STORE_SIZE]; size_t row_length; uint row_parts_count; + ha_checksum checksum_delta; /* Write UNDO record */ lsn_store(log_data, info->trn->undo_lsn); @@ -3073,15 +3158,26 @@ my_bool _ma_delete_block_record(MARIA_HA *info, const uchar *record) PAGE_STORE_SIZE, record_number); info->log_row_parts[TRANSLOG_INTERNAL_PARTS].str= (char*) log_data; - info->log_row_parts[TRANSLOG_INTERNAL_PARTS].length= sizeof(log_data); + info->log_row_parts[TRANSLOG_INTERNAL_PARTS].length= + sizeof(log_data) - HA_CHECKSUM_STORE_SIZE; + store_checksum_in_rec(share, checksum_delta, + - info->cur_row.checksum, + log_data + LSN_STORE_SIZE + FILEID_STORE_SIZE + + PAGE_STORE_SIZE + DIRPOS_STORE_SIZE, + info->log_row_parts[TRANSLOG_INTERNAL_PARTS + + 0].length); + row_length= fill_insert_undo_parts(info, record, info->log_row_parts + TRANSLOG_INTERNAL_PARTS + 1, &row_parts_count); if (translog_write_record(&lsn, LOGREC_UNDO_ROW_DELETE, info->trn, - info, sizeof(log_data) + row_length, + info, + info->log_row_parts[TRANSLOG_INTERNAL_PARTS + + 0].length + row_length, TRANSLOG_INTERNAL_PARTS + 1 + row_parts_count, - info->log_row_parts, log_data + LSN_STORE_SIZE)) + info->log_row_parts, log_data + LSN_STORE_SIZE, + &checksum_delta)) goto err; } @@ -3377,7 +3473,8 @@ static my_bool read_long_data(MARIA_HA *info, uchar *to, ulong length, cur_row.extents_counts contains number of extents cur_row.empty_bits is set to empty bits cur_row.field_lengths contains packed length of all fields - cur_row.blob_length contains total length of all blobs. + cur_row.blob_length contains total length of all blobs + cur_row.checksum contains checksum of read record. RETURN 0 ok @@ -4576,6 +4673,211 @@ static size_t fill_update_undo_parts(MARIA_HA *info, const uchar *oldrec, DBUG_RETURN(row_length); } +/*************************************************************************** + In-write hooks called under log's lock when log record is written +***************************************************************************/ + +/** + @brief Sets transaction's rec_lsn if needed + + A transaction sometimes writes a REDO even before the page is in the + pagecache (example: brand new head or tail pages; full pages). So, if + Checkpoint happens just after the REDO write, it needs to know that the + REDO phase must start before this REDO. Scanning the pagecache cannot + tell that as the page is not in the cache. So, transaction sets its rec_lsn + to the REDO's LSN or somewhere before, and Checkpoint reads the + transaction's rec_lsn. + + @return Operation status, always 0 (success) +*/ + +my_bool write_hook_for_redo(enum translog_record_type type + __attribute__ ((unused)), + TRN *trn, MARIA_HA *tbl_info + __attribute__ ((unused)), + LSN *lsn, void *hook_arg + __attribute__ ((unused))) +{ + /* + Users of dummy_transaction_object must keep this TRN clean as it + is used by many threads (like those manipulating non-transactional + tables). It might be dangerous if one user sets rec_lsn or some other + member and it is picked up by another user (like putting this rec_lsn into + a page of a non-transactional table); it's safer if all members stay 0. So + non-transactional log records (REPAIR, CREATE, RENAME, DROP) should not + call this hook; we trust them but verify ;) + */ + DBUG_ASSERT(trn->trid != 0); + /* + If the hook stays so simple, it would be faster to pass + !trn->rec_lsn ? trn->rec_lsn : some_dummy_lsn + to translog_write_record(), like Monty did in his original code, and not + have a hook. For now we keep it like this. + */ + if (trn->rec_lsn == 0) + trn->rec_lsn= *lsn; + return 0; +} + + +/** + @brief Sets transaction's undo_lsn, first_undo_lsn if needed + + @return Operation status, always 0 (success) +*/ + +my_bool write_hook_for_undo(enum translog_record_type type + __attribute__ ((unused)), + TRN *trn, MARIA_HA *tbl_info + __attribute__ ((unused)), + LSN *lsn, void *hook_arg + __attribute__ ((unused))) +{ + DBUG_ASSERT(trn->trid != 0); + trn->undo_lsn= *lsn; + if (unlikely(LSN_WITH_FLAGS_TO_LSN(trn->first_undo_lsn) == 0)) + trn->first_undo_lsn= + trn->undo_lsn | LSN_WITH_FLAGS_TO_FLAGS(trn->first_undo_lsn); + DBUG_ASSERT(tbl_info->state == &tbl_info->s->state.state); + return 0; + /* + when we implement purging, we will specialize this hook: UNDO_PURGE + records will additionally set trn->undo_purge_lsn + */ +} + + +/** + @brief Sets the table's records count and checksum to 0, then calls the + generic REDO hook. + + @return Operation status, always 0 (success) +*/ + +my_bool write_hook_for_redo_delete_all(enum translog_record_type type + __attribute__ ((unused)), + TRN *trn, MARIA_HA *tbl_info + __attribute__ ((unused)), + LSN *lsn, void *hook_arg) +{ + MARIA_SHARE *share= tbl_info->s; + DBUG_ASSERT(tbl_info->state == &tbl_info->s->state.state); + share->state.state.records= share->state.state.checksum= 0; + return write_hook_for_redo(type, trn, tbl_info, lsn, hook_arg); +} + + +/** + @brief Upates "records" and "checksum" and calls the generic UNDO hook + + @return Operation status, always 0 (success) +*/ + +my_bool write_hook_for_undo_row_insert(enum translog_record_type type + __attribute__ ((unused)), + TRN *trn, MARIA_HA *tbl_info, + LSN *lsn, void *hook_arg) +{ + MARIA_SHARE *share= tbl_info->s; + share->state.state.records++; + share->state.state.checksum+= *(ha_checksum *)hook_arg; + return write_hook_for_undo(type, trn, tbl_info, lsn, hook_arg); +} + + +/** + @brief Upates "records" and calls the generic UNDO hook + + @return Operation status, always 0 (success) +*/ + +my_bool write_hook_for_undo_row_delete(enum translog_record_type type + __attribute__ ((unused)), + TRN *trn, MARIA_HA *tbl_info, + LSN *lsn, void *hook_arg) +{ + MARIA_SHARE *share= tbl_info->s; + share->state.state.records--; + share->state.state.checksum+= *(ha_checksum *)hook_arg; + return write_hook_for_undo(type, trn, tbl_info, lsn, hook_arg); +} + + +/** + @brief Upates "records" and "checksum" and calls the generic UNDO hook + + @return Operation status, always 0 (success) +*/ + +my_bool write_hook_for_undo_row_update(enum translog_record_type type + __attribute__ ((unused)), + TRN *trn, MARIA_HA *tbl_info, + LSN *lsn, void *hook_arg) +{ + MARIA_SHARE *share= tbl_info->s; + share->state.state.checksum+= *(ha_checksum *)hook_arg; + return write_hook_for_undo(type, trn, tbl_info, lsn, hook_arg); +} + + +/** + @brief Sets transaction's undo_lsn, first_undo_lsn if needed + + @return Operation status, always 0 (success) +*/ + +my_bool write_hook_for_clr_end(enum translog_record_type type + __attribute__ ((unused)), + TRN *trn, MARIA_HA *tbl_info + __attribute__ ((unused)), + LSN *lsn __attribute__ ((unused)), + void *hook_arg) +{ + MARIA_SHARE *share= tbl_info->s; + struct st_msg_to_write_hook_for_clr_end *msg= + (struct st_msg_to_write_hook_for_clr_end *)hook_arg; + DBUG_ASSERT(trn->trid != 0); + trn->undo_lsn= msg->previous_undo_lsn; + share->state.state.checksum+= msg->checksum_delta; + + switch (msg->undone_record_type) { + case LOGREC_UNDO_ROW_DELETE: + share->state.state.records++; + break; + case LOGREC_UNDO_ROW_INSERT: + share->state.state.records--; + break; + case LOGREC_UNDO_ROW_UPDATE: + break; + default: + DBUG_ASSERT(0); + } + if (trn->undo_lsn == LSN_IMPOSSIBLE) /* has fully rolled back */ + trn->first_undo_lsn= LSN_WITH_FLAGS_TO_FLAGS(trn->first_undo_lsn); + return 0; +} + + +/** + @brief Updates table's lsn_of_file_id. + + @return Operation status, always 0 (success) +*/ + +my_bool write_hook_for_file_id(enum translog_record_type type + __attribute__ ((unused)), + TRN *trn + __attribute__ ((unused)), + MARIA_HA *tbl_info, + LSN *lsn __attribute__ ((unused)), + void *hook_arg + __attribute__ ((unused))) +{ + DBUG_ASSERT(cmp_translog_addr(tbl_info->s->lsn_of_file_id, *lsn) < 0); + tbl_info->s->lsn_of_file_id= *lsn; + return 0; +} + /*************************************************************************** Applying of REDO log records ***************************************************************************/ @@ -4944,19 +5246,25 @@ my_bool _ma_apply_undo_row_insert(MARIA_HA *info, LSN undo_lsn, ulonglong page; uint rownr; LEX_STRING log_array[TRANSLOG_INTERNAL_PARTS + 1]; - uchar log_data[LSN_STORE_SIZE + FILEID_STORE_SIZE + 1], *buff; + uchar log_data[LSN_STORE_SIZE + FILEID_STORE_SIZE + + CLR_TYPE_STORE_SIZE + HA_CHECKSUM_STORE_SIZE], + *buff; my_bool res= 1; MARIA_PINNED_PAGE page_link; LSN lsn; + MARIA_SHARE *share= info->s; + struct st_msg_to_write_hook_for_clr_end msg; DBUG_ENTER("_ma_apply_undo_row_insert"); page= page_korr(header); - rownr= dirpos_korr(header + PAGE_STORE_SIZE); + header+= PAGE_STORE_SIZE; + rownr= dirpos_korr(header); + header+= DIRPOS_STORE_SIZE; DBUG_PRINT("enter", ("Page: %lu rownr: %u", (ulong) page, rownr)); - if (!(buff= pagecache_read(info->s->pagecache, + if (!(buff= pagecache_read(share->pagecache, &info->dfile, page, 0, - info->buff, info->s->page_type, + info->buff, share->page_type, PAGECACHE_LOCK_WRITE, &page_link.link))) DBUG_RETURN(1); @@ -4977,14 +5285,24 @@ my_bool _ma_apply_undo_row_insert(MARIA_HA *info, LSN undo_lsn, /* undo_lsn must be first for compression to work */ lsn_store(log_data, undo_lsn); - log_data[LSN_STORE_SIZE + FILEID_STORE_SIZE]= LOGREC_UNDO_ROW_INSERT; + clr_type_store(log_data + LSN_STORE_SIZE + FILEID_STORE_SIZE, + LOGREC_UNDO_ROW_INSERT); + log_array[TRANSLOG_INTERNAL_PARTS + 0].length= + sizeof(log_data) - HA_CHECKSUM_STORE_SIZE; + msg.undone_record_type= LOGREC_UNDO_ROW_INSERT; + msg.previous_undo_lsn= undo_lsn; + store_checksum_in_rec(share, msg.checksum_delta, + - ha_checksum_korr(header), + log_data + LSN_STORE_SIZE + FILEID_STORE_SIZE + + CLR_TYPE_STORE_SIZE, + log_array[TRANSLOG_INTERNAL_PARTS + 0].length); log_array[TRANSLOG_INTERNAL_PARTS + 0].str= (char*) log_data; - log_array[TRANSLOG_INTERNAL_PARTS + 0].length= sizeof(log_data); if (translog_write_record(&lsn, LOGREC_CLR_END, - info->trn, info, sizeof(log_data), + info->trn, info, log_array[TRANSLOG_INTERNAL_PARTS + + 0].length, TRANSLOG_INTERNAL_PARTS + 1, log_array, - log_data + LSN_STORE_SIZE)) + log_data + LSN_STORE_SIZE, &msg)) goto err; res= 0; @@ -5014,6 +5332,16 @@ my_bool _ma_apply_undo_row_delete(MARIA_HA *info, LSN undo_lsn, some buffers to point directly to 'header' */ memcpy(&row, &info->cur_row, sizeof(row)); + if (share->calc_checksum) + { + /* + We extract the checksum delta here, saving a recomputation in + allocate_and_write_block_record(). It's only an optimization. + */ + row.checksum= - ha_checksum_korr(header); + header+= HA_CHECKSUM_STORE_SIZE; + } + null_field_lengths= row.null_field_lengths; blob_lengths= row.blob_lengths; @@ -5184,18 +5512,25 @@ my_bool _ma_apply_undo_row_update(MARIA_HA *info, LSN undo_lsn, uchar *current_record, *orig_record; int error= 1; MARIA_RECORD_POS record_pos; + ha_checksum checksum_delta; DBUG_ENTER("_ma_apply_undo_row_update"); page= page_korr(header); - rownr= dirpos_korr(header + PAGE_STORE_SIZE); + header+= PAGE_STORE_SIZE; + rownr= dirpos_korr(header); + header+= DIRPOS_STORE_SIZE; record_pos= ma_recordpos(page, rownr); DBUG_PRINT("enter", ("Page: %lu rownr: %u", (ulong) page, rownr)); + if (share->calc_checksum) + { + checksum_delta= ha_checksum_korr(header); + header+= HA_CHECKSUM_STORE_SIZE; + } /* Set header to point to old field values, generated by fill_update_undo_parts() */ - header+= PAGE_STORE_SIZE + DIRPOS_STORE_SIZE; field_length_header= ma_get_length((uchar**) &header); field_length_data= header; header+= field_length_header; @@ -5290,14 +5625,14 @@ my_bool _ma_apply_undo_row_update(MARIA_HA *info, LSN undo_lsn, if (share->calc_checksum) { - info->cur_row.checksum= (*share->calc_checksum)(info, orig_record); - info->state->checksum+= (info->cur_row.checksum - - (*share->calc_checksum)(info, current_record)); + info->new_row.checksum= checksum_delta + + (info->cur_row.checksum= (*share->calc_checksum)(info, orig_record)); + /* verify that record's content is sane */ + DBUG_ASSERT(info->new_row.checksum == + (*share->calc_checksum)(info, current_record)); } - /* - Now records are up to date, execute the update to original values - */ + /* Now records are up to date, execute the update to original values */ if (_ma_update_block_record2(info, record_pos, current_record, orig_record, undo_lsn)) goto err; diff --git a/storage/maria/ma_blockrec.h b/storage/maria/ma_blockrec.h index 30dffe1c0c0..0bce4015daf 100644 --- a/storage/maria/ma_blockrec.h +++ b/storage/maria/ma_blockrec.h @@ -193,3 +193,28 @@ my_bool _ma_apply_undo_row_delete(MARIA_HA *info, LSN undo_lsn, const uchar *header, size_t length); my_bool _ma_apply_undo_row_update(MARIA_HA *info, LSN undo_lsn, const uchar *header, size_t length); + +my_bool write_hook_for_redo(enum translog_record_type type, + TRN *trn, MARIA_HA *tbl_info, LSN *lsn, + void *hook_arg); +my_bool write_hook_for_undo(enum translog_record_type type, + TRN *trn, MARIA_HA *tbl_info, LSN *lsn, + void *hook_arg); +my_bool write_hook_for_redo_delete_all(enum translog_record_type type, + TRN *trn, MARIA_HA *tbl_info, + LSN *lsn, void *hook_arg); +my_bool write_hook_for_undo_row_insert(enum translog_record_type type, + TRN *trn, MARIA_HA *tbl_info, + LSN *lsn, void *hook_arg); +my_bool write_hook_for_undo_row_delete(enum translog_record_type type, + TRN *trn, MARIA_HA *tbl_info, + LSN *lsn, void *hook_arg); +my_bool write_hook_for_undo_row_update(enum translog_record_type type, + TRN *trn, MARIA_HA *tbl_info, + LSN *lsn, void *hook_arg); +my_bool write_hook_for_clr_end(enum translog_record_type type, + TRN *trn, MARIA_HA *tbl_info, LSN *lsn, + void *hook_arg); +my_bool write_hook_for_file_id(enum translog_record_type type, + TRN *trn, MARIA_HA *tbl_info, LSN *lsn, + void *hook_arg); diff --git a/storage/maria/ma_check.c b/storage/maria/ma_check.c index fa1c812daf7..8ef8d6a7e76 100644 --- a/storage/maria/ma_check.c +++ b/storage/maria/ma_check.c @@ -5602,11 +5602,10 @@ static int write_log_record_for_repair(const HA_CHECK *param, MARIA_HA *info) record). */ LEX_STRING log_array[TRANSLOG_INTERNAL_PARTS + 1]; - uchar log_data[LSN_STORE_SIZE]; + uchar log_data[FILEID_STORE_SIZE + 4]; LSN lsn; - compile_time_assert(LSN_STORE_SIZE >= (FILEID_STORE_SIZE + 4)); log_array[TRANSLOG_INTERNAL_PARTS + 0].str= (char*) log_data; - log_array[TRANSLOG_INTERNAL_PARTS + 0].length= FILEID_STORE_SIZE + 4; + log_array[TRANSLOG_INTERNAL_PARTS + 0].length= sizeof(log_data); /* testflag gives an idea of what REPAIR did (in particular T_QUICK or not: did it touch the data file or not?). @@ -5614,10 +5613,9 @@ static int write_log_record_for_repair(const HA_CHECK *param, MARIA_HA *info) int4store(log_data + FILEID_STORE_SIZE, param->testflag); if (unlikely(translog_write_record(&lsn, LOGREC_REDO_REPAIR_TABLE, &dummy_transaction_object, info, - log_array[TRANSLOG_INTERNAL_PARTS + - 0].length, + sizeof(log_data), sizeof(log_array)/sizeof(log_array[0]), - log_array, log_data) || + log_array, log_data, NULL) || translog_flush(lsn))) return 1; /* diff --git a/storage/maria/ma_checkpoint.c b/storage/maria/ma_checkpoint.c index 4446285fce9..76eacaede0d 100644 --- a/storage/maria/ma_checkpoint.c +++ b/storage/maria/ma_checkpoint.c @@ -266,7 +266,7 @@ static int really_execute_checkpoint(void) &dummy_transaction_object, NULL, total_rec_length, sizeof(log_array)/sizeof(log_array[0]), - log_array, NULL) || + log_array, NULL, NULL) || translog_flush(lsn))) goto err; @@ -652,7 +652,7 @@ pthread_handler_t ma_checkpoint_background(void *arg __attribute__((unused))) break; #if 0 /* good for testing, to do a lot of checkpoints, finds a lot of bugs */ pthread_mutex_unlock(&LOCK_checkpoint); - my_sleep(100000); // a tenth of a second + my_sleep(100000); /* a tenth of a second */ pthread_mutex_lock(&LOCK_checkpoint); #else /* To have a killable sleep, we use timedwait like our SQL GET_LOCK() */ @@ -893,7 +893,7 @@ static int collect_tables(LEX_STRING *str, LSN checkpoint_start_log_horizon) filter_param.pages_covered_by_bitmap= share->bitmap.pages_covered; /* OS file descriptors are ints which we stored in 4 bytes */ - compile_time_assert(sizeof(int) == 4); + compile_time_assert(sizeof(int) <= 4); pthread_mutex_lock(&share->intern_lock); /* Tables in a normal state have their two file descriptors open. diff --git a/storage/maria/ma_commit.c b/storage/maria/ma_commit.c index 36ea2f6e6e4..8a0a4c136bb 100644 --- a/storage/maria/ma_commit.c +++ b/storage/maria/ma_commit.c @@ -64,7 +64,7 @@ int ma_commit(TRN *trn) res= (translog_write_record(&commit_lsn, LOGREC_COMMIT, trn, NULL, 0, sizeof(log_array)/sizeof(log_array[0]), - log_array, NULL) || + log_array, NULL, NULL) || translog_flush(commit_lsn) || trnman_commit_trn(trn)); /* diff --git a/storage/maria/ma_create.c b/storage/maria/ma_create.c index ba1d9a13b42..f3cee8b26e2 100644 --- a/storage/maria/ma_create.c +++ b/storage/maria/ma_create.c @@ -997,7 +997,7 @@ int maria_create(const char *name, enum data_file_type datafile_type, &dummy_transaction_object, NULL, total_rec_length, sizeof(log_array)/sizeof(log_array[0]), - log_array, NULL) || + log_array, NULL, NULL) || translog_flush(lsn))) goto err; /* diff --git a/storage/maria/ma_delete.c b/storage/maria/ma_delete.c index 56da6fd3ed3..c229e6c55d0 100644 --- a/storage/maria/ma_delete.c +++ b/storage/maria/ma_delete.c @@ -44,10 +44,10 @@ int maria_delete(MARIA_HA *info,const uchar *record) /* Test if record is in datafile */ DBUG_EXECUTE_IF("maria_pretend_crashed_table_on_usage", - maria_print_error(info->s, HA_ERR_CRASHED); + maria_print_error(share, HA_ERR_CRASHED); DBUG_RETURN(my_errno= HA_ERR_CRASHED);); DBUG_EXECUTE_IF("my_error_test_undefined_error", - maria_print_error(info->s, INT_MAX); + maria_print_error(share, INT_MAX); DBUG_RETURN(my_errno= INT_MAX);); if (!(info->update & HA_STATE_AKTIV)) { @@ -70,17 +70,17 @@ int maria_delete(MARIA_HA *info,const uchar *record) old_key= info->lastkey2; for (i=0 ; i < share->base.keys ; i++ ) { - if (maria_is_key_active(info->s->state.key_map, i)) + if (maria_is_key_active(share->state.key_map, i)) { - info->s->keyinfo[i].version++; - if (info->s->keyinfo[i].flag & HA_FULLTEXT ) + share->keyinfo[i].version++; + if (share->keyinfo[i].flag & HA_FULLTEXT ) { if (_ma_ft_del(info,i,(char*) old_key,record,info->cur_row.lastpos)) goto err; } else { - if (info->s->keyinfo[i].ck_delete(info,i,old_key, + if (share->keyinfo[i].ck_delete(info,i,old_key, _ma_make_key(info,i,old_key,record,info->cur_row.lastpos))) goto err; } @@ -89,19 +89,20 @@ int maria_delete(MARIA_HA *info,const uchar *record) } } - if ((*share->delete_record)(info, record)) - goto err; /* Remove record from database */ - - /* - We can't use the row based checksum as this doesn't have enough - precision. - */ - if (info->s->calc_checksum) + if (share->calc_checksum) { - info->cur_row.checksum= (*info->s->calc_checksum)(info,record); - info->state->checksum-= info->cur_row.checksum; + /* + We can't use the row based checksum as this doesn't have enough + precision. + */ + info->cur_row.checksum= (*share->calc_checksum)(info, record); } + if ((*share->delete_record)(info, record)) + goto err; /* Remove record from database */ + + info->state->checksum+= - !share->now_transactional * + info->cur_row.checksum; info->update= HA_STATE_CHANGED+HA_STATE_DELETED+HA_STATE_ROW_CHANGED; info->state->records-= !share->now_transactional; share->state.changed|= STATE_NOT_OPTIMIZED_ROWS; @@ -111,8 +112,8 @@ int maria_delete(MARIA_HA *info,const uchar *record) allow_break(); /* Allow SIGHUP & SIGINT */ if (info->invalidator != 0) { - DBUG_PRINT("info", ("invalidator... '%s' (delete)", info->s->open_file_name)); - (*info->invalidator)(info->s->open_file_name); + DBUG_PRINT("info", ("invalidator... '%s' (delete)", share->open_file_name)); + (*info->invalidator)(share->open_file_name); info->invalidator=0; } DBUG_RETURN(0); @@ -122,7 +123,7 @@ err: mi_sizestore(lastpos, info->cur_row.lastpos); if (save_errno != HA_ERR_RECORD_CHANGED) { - maria_print_error(info->s, HA_ERR_CRASHED); + maria_print_error(share, HA_ERR_CRASHED); maria_mark_crashed(info); /* mark table crashed */ } VOID(_ma_writeinfo(info,WRITEINFO_UPDATE_KEYFILE)); @@ -131,7 +132,7 @@ err: my_errno=save_errno; if (save_errno == HA_ERR_KEY_NOT_FOUND) { - maria_print_error(info->s, HA_ERR_CRASHED); + maria_print_error(share, HA_ERR_CRASHED); my_errno=HA_ERR_CRASHED; } diff --git a/storage/maria/ma_delete_all.c b/storage/maria/ma_delete_all.c index 8cb4fdb8a3e..408ca31b3c6 100644 --- a/storage/maria/ma_delete_all.c +++ b/storage/maria/ma_delete_all.c @@ -64,14 +64,15 @@ int maria_delete_all_rows(MARIA_HA *info) if (unlikely(translog_write_record(&lsn, LOGREC_REDO_DELETE_ALL, info->trn, info, 0, sizeof(log_array)/sizeof(log_array[0]), - log_array, log_data) || + log_array, log_data, NULL) || translog_flush(lsn))) goto err; } /* For recovery it matters that this is called after writing the log record, - so that resetting state.records actually happens under log's mutex. + so that resetting state.records and state.checksum actually happens under + log's mutex. */ _ma_reset_status(info); @@ -147,10 +148,6 @@ void _ma_reset_status(MARIA_HA *info) info->state->key_file_length= share->base.keystart; info->state->data_file_length= 0; info->state->empty= info->state->key_empty= 0; - /** - @todo RECOVERY BUG - the line below must happen under log's mutex when writing the REDO - */ info->state->checksum= 0; /* Drop the delete key chain. */ diff --git a/storage/maria/ma_delete_table.c b/storage/maria/ma_delete_table.c index 693c68c7e5f..e6cbd961b7a 100644 --- a/storage/maria/ma_delete_table.c +++ b/storage/maria/ma_delete_table.c @@ -92,7 +92,7 @@ int maria_delete_table(const char *name) log_array[TRANSLOG_INTERNAL_PARTS + 0].length, sizeof(log_array)/sizeof(log_array[0]), - log_array, NULL) || + log_array, NULL, NULL) || translog_flush(lsn))) DBUG_RETURN(1); } diff --git a/storage/maria/ma_loghandler.c b/storage/maria/ma_loghandler.c index 3a94be3a16a..a4e5404815d 100644 --- a/storage/maria/ma_loghandler.c +++ b/storage/maria/ma_loghandler.c @@ -14,8 +14,8 @@ Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ #include "maria_def.h" -#include "ma_blockrec.h" -#include "trnman.h" +#include "ma_blockrec.h" /* for some constants and in-write hooks */ +#include "trnman.h" /* for access to members of TRN */ /** @file @@ -208,31 +208,6 @@ static MARIA_SHARE **id_to_share= NULL; /* lock for id_to_share */ static my_atomic_rwlock_t LOCK_id_to_share; -static my_bool write_hook_for_redo(enum translog_record_type type, - TRN *trn, MARIA_HA *tbl_info, LSN *lsn, - struct st_translog_parts *parts); -static my_bool write_hook_for_undo(enum translog_record_type type, - TRN *trn, MARIA_HA *tbl_info, LSN *lsn, - struct st_translog_parts *parts); -static my_bool write_hook_for_redo_delete_all(enum translog_record_type type, - TRN *trn, MARIA_HA *tbl_info, - LSN *lsn, - struct st_translog_parts *parts); -static my_bool write_hook_for_undo_row_insert(enum translog_record_type type, - TRN *trn, MARIA_HA *tbl_info, - LSN *lsn, - struct st_translog_parts *parts); -static my_bool write_hook_for_undo_row_delete(enum translog_record_type type, - TRN *trn, MARIA_HA *tbl_info, - LSN *lsn, - struct st_translog_parts *parts); -static my_bool write_hook_for_clr_end(enum translog_record_type type, - TRN *trn, MARIA_HA *tbl_info, LSN *lsn, - struct st_translog_parts *parts); -static my_bool write_hook_for_file_id(enum translog_record_type type, - TRN *trn, MARIA_HA *tbl_info, LSN *lsn, - struct st_translog_parts *parts); - static my_bool translog_page_validator(uchar *page_addr, uchar* data_ptr); /* @@ -437,8 +412,8 @@ static LOG_DESC INIT_LOGREC_REDO_UNDELETE_ROW= "redo_undelete_row", LOGREC_NOT_LAST_IN_GROUP, NULL, NULL}; static LOG_DESC INIT_LOGREC_CLR_END= -{LOGRECTYPE_PSEUDOFIXEDLENGTH, LSN_STORE_SIZE + FILEID_STORE_SIZE + 1, - LSN_STORE_SIZE + FILEID_STORE_SIZE + 1, NULL, write_hook_for_clr_end, NULL, 1, +{LOGRECTYPE_VARIABLE_LENGTH, 0, LSN_STORE_SIZE + FILEID_STORE_SIZE + + CLR_TYPE_STORE_SIZE, NULL, write_hook_for_clr_end, NULL, 1, "clr_end", LOGREC_LAST_IN_GROUP, NULL, NULL}; static LOG_DESC INIT_LOGREC_PURGE_END= @@ -446,8 +421,7 @@ static LOG_DESC INIT_LOGREC_PURGE_END= "purge_end", LOGREC_LAST_IN_GROUP, NULL, NULL}; static LOG_DESC INIT_LOGREC_UNDO_ROW_INSERT= -{LOGRECTYPE_PSEUDOFIXEDLENGTH, - LSN_STORE_SIZE + FILEID_STORE_SIZE + PAGE_STORE_SIZE + DIRPOS_STORE_SIZE, +{LOGRECTYPE_VARIABLE_LENGTH, 0, LSN_STORE_SIZE + FILEID_STORE_SIZE + PAGE_STORE_SIZE + DIRPOS_STORE_SIZE, NULL, write_hook_for_undo_row_insert, NULL, 1, "undo_row_insert", LOGREC_LAST_IN_GROUP, NULL, NULL}; @@ -461,7 +435,7 @@ static LOG_DESC INIT_LOGREC_UNDO_ROW_DELETE= static LOG_DESC INIT_LOGREC_UNDO_ROW_UPDATE= {LOGRECTYPE_VARIABLE_LENGTH, 0, LSN_STORE_SIZE + FILEID_STORE_SIZE + PAGE_STORE_SIZE + DIRPOS_STORE_SIZE, - NULL, write_hook_for_undo, NULL, 1, + NULL, write_hook_for_undo_row_update, NULL, 1, "undo_row_update", LOGREC_LAST_IN_GROUP, NULL, NULL}; static LOG_DESC INIT_LOGREC_UNDO_KEY_INSERT= @@ -3678,23 +3652,23 @@ static translog_size_t translog_get_current_group_size() } -/* - Write variable record in 1 group +/** + @brief Write variable record in 1 group. - SYNOPSIS - translog_write_variable_record_1group() - lsn LSN of the record will be written here - type the log record type - short_trid Short transaction ID or 0 if it has no sense - parts Descriptor of record source parts - buffer_to_flush Buffer which have to be flushed if it is not 0 - header_length Calculated header length of chunk type 0 - trn Transaction structure pointer for hooks by - record log type, for short_id + @param lsn LSN of the record will be written here + @param type the log record type + @param short_trid Short transaction ID or 0 if it has no sense + @param parts Descriptor of record source parts + @param buffer_to_flush Buffer which have to be flushed if it is not 0 + @param header_length Calculated header length of chunk type 0 + @param trn Transaction structure pointer for hooks by + record log type, for short_id + @param hook_arg Argument which will be passed to pre-write and + in-write hooks of this record. - RETURN - 0 OK - 1 Error + @return Operation status + @retval 0 OK + @retval 1 Error */ static my_bool @@ -3705,7 +3679,7 @@ translog_write_variable_record_1group(LSN *lsn, struct st_translog_parts *parts, struct st_translog_buffer *buffer_to_flush, uint16 header_length, - TRN *trn) + TRN *trn, void *hook_arg) { TRANSLOG_ADDRESS horizon; struct st_buffer_cursor cursor; @@ -3721,7 +3695,7 @@ translog_write_variable_record_1group(LSN *lsn, *lsn, TRUE) || (log_record_type_descriptor[type].inwrite_hook && (*log_record_type_descriptor[type].inwrite_hook)(type, trn, tbl_info, - lsn, parts))) + lsn, hook_arg))) { translog_unlock(); DBUG_RETURN(1); @@ -3830,23 +3804,23 @@ translog_write_variable_record_1group(LSN *lsn, } -/* - Write variable record in 1 chunk +/** + @brief Write variable record in 1 chunk. - SYNOPSIS - translog_write_variable_record_1chunk() - lsn LSN of the record will be written here - type the log record type - short_trid Short transaction ID or 0 if it has no sense - parts Descriptor of record source parts - buffer_to_flush Buffer which have to be flushed if it is not 0 - header_length Calculated header length of chunk type 0 - trn Transaction structure pointer for hooks by - record log type, for short_id + @param lsn LSN of the record will be written here + @param type the log record type + @param short_trid Short transaction ID or 0 if it has no sense + @param parts Descriptor of record source parts + @param buffer_to_flush Buffer which have to be flushed if it is not 0 + @param header_length Calculated header length of chunk type 0 + @param trn Transaction structure pointer for hooks by + record log type, for short_id + @param hook_arg Argument which will be passed to pre-write and + in-write hooks of this record. - RETURN - 0 OK - 1 Error + @return Operation status + @retval 0 OK + @retval 1 Error */ static my_bool @@ -3857,7 +3831,7 @@ translog_write_variable_record_1chunk(LSN *lsn, struct st_translog_parts *parts, struct st_translog_buffer *buffer_to_flush, uint16 header_length, - TRN *trn) + TRN *trn, void *hook_arg) { int rc; uchar chunk0_header[1 + 2 + 5 + 2]; @@ -3871,7 +3845,7 @@ translog_write_variable_record_1chunk(LSN *lsn, *lsn, TRUE) || (log_record_type_descriptor[type].inwrite_hook && (*log_record_type_descriptor[type].inwrite_hook)(type, trn, tbl_info, - lsn, parts))) + lsn, hook_arg))) { translog_unlock(); DBUG_RETURN(1); @@ -4196,24 +4170,24 @@ static my_bool translog_relative_LSN_encode(struct st_translog_parts *parts, } -/* - Write multi-group variable-size record +/** + @brief Write multi-group variable-size record. - SYNOPSIS - translog_write_variable_record_mgroup() - lsn LSN of the record will be written here - type the log record type - short_trid Short transaction ID or 0 if it has no sense - parts Descriptor of record source parts - buffer_to_flush Buffer which have to be flushed if it is not 0 - header_length Header length calculated for 1 group - buffer_rest Beginning from which we plan to write in full pages - trn Transaction structure pointer for hooks by - record log type, for short_id + @param lsn LSN of the record will be written here + @param type the log record type + @param short_trid Short transaction ID or 0 if it has no sense + @param parts Descriptor of record source parts + @param buffer_to_flush Buffer which have to be flushed if it is not 0 + @param header_length Header length calculated for 1 group + @param buffer_rest Beginning from which we plan to write in full pages + @param trn Transaction structure pointer for hooks by + record log type, for short_id + @param hook_arg Argument which will be passed to pre-write and + in-write hooks of this record. - RETURN - 0 OK - 1 Error + @return Operation status + @retval 0 OK + @retval 1 Error */ static my_bool @@ -4226,7 +4200,7 @@ translog_write_variable_record_mgroup(LSN *lsn, *buffer_to_flush, uint16 header_length, translog_size_t buffer_rest, - TRN *trn) + TRN *trn, void *hook_arg) { TRANSLOG_ADDRESS horizon; struct st_buffer_cursor cursor; @@ -4554,7 +4528,7 @@ translog_write_variable_record_mgroup(LSN *lsn, if (log_record_type_descriptor[type].inwrite_hook && (*log_record_type_descriptor[type].inwrite_hook) (type, trn, tbl_info, - lsn, parts)) + lsn, hook_arg)) goto err; } @@ -4626,21 +4600,21 @@ err: } -/* - Write the variable length log record +/** + @brief Write the variable length log record. - SYNOPSIS - translog_write_variable_record() - lsn LSN of the record will be written here - type the log record type - short_trid Short transaction ID or 0 if it has no sense - parts Descriptor of record source parts - trn Transaction structure pointer for hooks by - record log type, for short_id + @param lsn LSN of the record will be written here + @param type the log record type + @param short_trid Short transaction ID or 0 if it has no sense + @param parts Descriptor of record source parts + @param trn Transaction structure pointer for hooks by + record log type, for short_id + @param hook_arg Argument which will be passed to pre-write and + in-write hooks of this record. - RETURN - 0 OK - 1 Error + @return Operation status + @retval 0 OK + @retval 1 Error */ static my_bool translog_write_variable_record(LSN *lsn, @@ -4648,7 +4622,7 @@ static my_bool translog_write_variable_record(LSN *lsn, MARIA_HA *tbl_info, SHORT_TRANSACTION_ID short_trid, struct st_translog_parts *parts, - TRN *trn) + TRN *trn, void *hook_arg) { struct st_translog_buffer *buffer_to_flush= NULL; uint header_length1= 1 + 2 + 2 + @@ -4725,7 +4699,7 @@ static my_bool translog_write_variable_record(LSN *lsn, res= translog_write_variable_record_1chunk(lsn, type, tbl_info, short_trid, parts, buffer_to_flush, - header_length1, trn); + header_length1, trn, hook_arg); DBUG_RETURN(res); } @@ -4737,7 +4711,7 @@ static my_bool translog_write_variable_record(LSN *lsn, res= translog_write_variable_record_1group(lsn, type, tbl_info, short_trid, parts, buffer_to_flush, - header_length1, trn); + header_length1, trn, hook_arg); DBUG_RETURN(res); } /* following function makes translog_unlock(); */ @@ -4745,26 +4719,26 @@ static my_bool translog_write_variable_record(LSN *lsn, short_trid, parts, buffer_to_flush, header_length1, - buffer_rest, trn); + buffer_rest, trn, hook_arg); DBUG_RETURN(res); } -/* - Write the fixed and pseudo-fixed log record +/** + @brief Write the fixed and pseudo-fixed log record. - SYNOPSIS - translog_write_fixed_record() - lsn LSN of the record will be written here - type the log record type - short_trid Short transaction ID or 0 if it has no sense - parts Descriptor of record source parts - trn Transaction structure pointer for hooks by - record log type, for short_id + @param lsn LSN of the record will be written here + @param type the log record type + @param short_trid Short transaction ID or 0 if it has no sense + @param parts Descriptor of record source parts + @param trn Transaction structure pointer for hooks by + record log type, for short_id + @param hook_arg Argument which will be passed to pre-write and + in-write hooks of this record. - RETURN - 0 OK - 1 Error + @return Operation status + @retval 0 OK + @retval 1 Error */ static my_bool translog_write_fixed_record(LSN *lsn, @@ -4772,7 +4746,7 @@ static my_bool translog_write_fixed_record(LSN *lsn, MARIA_HA *tbl_info, SHORT_TRANSACTION_ID short_trid, struct st_translog_parts *parts, - TRN *trn) + TRN *trn, void *hook_arg) { struct st_translog_buffer *buffer_to_flush= NULL; uchar chunk1_header[1 + 2]; @@ -4824,7 +4798,7 @@ static my_bool translog_write_fixed_record(LSN *lsn, *lsn, TRUE) || (log_record_type_descriptor[type].inwrite_hook && (*log_record_type_descriptor[type].inwrite_hook) (type, trn, tbl_info, - lsn, parts))) + lsn, hook_arg))) { rc= 1; goto err; @@ -4899,6 +4873,9 @@ err: @param store_share_id if tbl_info!=NULL then share's id will automatically be stored in the two first bytes pointed (so pointer is assumed to be !=NULL) + @param hook_arg argument which will be passed to pre-write and + in-write hooks of this record. + @return Operation status @retval 0 OK @retval 1 Error @@ -4910,7 +4887,8 @@ my_bool translog_write_record(LSN *lsn, translog_size_t rec_len, uint part_no, LEX_STRING *parts_data, - uchar *store_share_id) + uchar *store_share_id, + void *hook_arg) { struct st_translog_parts parts; LEX_STRING *part; @@ -4955,7 +4933,7 @@ my_bool translog_write_record(LSN *lsn, if (unlikely(translog_write_record(&dummy_lsn, LOGREC_LONG_TRANSACTION_ID, trn, NULL, sizeof(log_data), sizeof(log_array)/sizeof(log_array[0]), - log_array, NULL))) + log_array, NULL, NULL))) DBUG_RETURN(1); } @@ -5018,17 +4996,17 @@ my_bool translog_write_record(LSN *lsn, if (!(rc= (log_record_type_descriptor[type].prewrite_hook && (*log_record_type_descriptor[type].prewrite_hook) (type, trn, tbl_info, - &parts)))) + hook_arg)))) { switch (log_record_type_descriptor[type].class) { case LOGRECTYPE_VARIABLE_LENGTH: rc= translog_write_variable_record(lsn, type, tbl_info, - short_trid, &parts, trn); + short_trid, &parts, trn, hook_arg); break; case LOGRECTYPE_PSEUDOFIXEDLENGTH: case LOGRECTYPE_FIXEDLENGTH: rc= translog_write_fixed_record(lsn, type, tbl_info, - short_trid, &parts, trn); + short_trid, &parts, trn, hook_arg); break; case LOGRECTYPE_NOT_ALLOWED: default: @@ -6060,6 +6038,7 @@ translog_size_t translog_read_record(LSN lsn, if (offset < page_end) { uint len= page_end - offset; + set_if_smaller(len, length); /* in case we read beyond record's end */ DBUG_ASSERT(offset >= data->current_offset); memcpy(buffer, data->scanner.page + data->body_offset + @@ -6339,207 +6318,6 @@ out: } -/** - @brief Sets transaction's rec_lsn if needed - - A transaction sometimes writes a REDO even before the page is in the - pagecache (example: brand new head or tail pages; full pages). So, if - Checkpoint happens just after the REDO write, it needs to know that the - REDO phase must start before this REDO. Scanning the pagecache cannot - tell that as the page is not in the cache. So, transaction sets its rec_lsn - to the REDO's LSN or somewhere before, and Checkpoint reads the - transaction's rec_lsn. - - @todo move it to a separate file - - @return Operation status, always 0 (success) -*/ - -static my_bool write_hook_for_redo(enum translog_record_type type - __attribute__ ((unused)), - TRN *trn, MARIA_HA *tbl_info - __attribute__ ((unused)), - LSN *lsn, - struct st_translog_parts *parts - __attribute__ ((unused))) -{ - /* - Users of dummy_transaction_object must keep this TRN clean as it - is used by many threads (like those manipulating non-transactional - tables). It might be dangerous if one user sets rec_lsn or some other - member and it is picked up by another user (like putting this rec_lsn into - a page of a non-transactional table); it's safer if all members stay 0. So - non-transactional log records (REPAIR, CREATE, RENAME, DROP) should not - call this hook; we trust them but verify ;) - */ - DBUG_ASSERT(trn->trid != 0); - /* - If the hook stays so simple, it would be faster to pass - !trn->rec_lsn ? trn->rec_lsn : some_dummy_lsn - to translog_write_record(), like Monty did in his original code, and not - have a hook. For now we keep it like this. - */ - if (trn->rec_lsn == 0) - trn->rec_lsn= *lsn; - return 0; -} - - -/** - @brief Sets transaction's undo_lsn, first_undo_lsn if needed - - @todo move it to a separate file - - @return Operation status, always 0 (success) -*/ - -static my_bool write_hook_for_undo(enum translog_record_type type - __attribute__ ((unused)), - TRN *trn, MARIA_HA *tbl_info - __attribute__ ((unused)), - LSN *lsn, - struct st_translog_parts *parts - __attribute__ ((unused))) -{ - DBUG_ASSERT(trn->trid != 0); - trn->undo_lsn= *lsn; - if (unlikely(LSN_WITH_FLAGS_TO_LSN(trn->first_undo_lsn) == 0)) - trn->first_undo_lsn= - trn->undo_lsn | LSN_WITH_FLAGS_TO_FLAGS(trn->first_undo_lsn); - return 0; - /* - when we implement purging, we will specialize this hook: UNDO_PURGE - records will additionally set trn->undo_purge_lsn - */ -} - - -/** - @brief Sets the table's records count to 0, then calls the generic REDO - hook. - - @todo move it to a separate file - - @return Operation status, always 0 (success) -*/ - -static my_bool write_hook_for_redo_delete_all(enum translog_record_type type - __attribute__ ((unused)), - TRN *trn, MARIA_HA *tbl_info - __attribute__ ((unused)), - LSN *lsn, - struct st_translog_parts *parts - __attribute__ ((unused))) -{ - tbl_info->s->state.state.records= 0; - return write_hook_for_redo(type, trn, tbl_info, lsn, parts); -} - - -/** - @brief Upates "records" and calls the generic UNDO hook - - @todo move it to a separate file - - @return Operation status, always 0 (success) -*/ - -static my_bool write_hook_for_undo_row_insert(enum translog_record_type type - __attribute__ ((unused)), - TRN *trn, MARIA_HA *tbl_info, - LSN *lsn, - struct st_translog_parts *parts - __attribute__ ((unused))) -{ - tbl_info->s->state.state.records++; - return write_hook_for_undo(type, trn, tbl_info, lsn, parts); -} - - -/** - @brief Upates "records" and calls the generic UNDO hook - - @todo move it to a separate file - - @return Operation status, always 0 (success) -*/ - -static my_bool write_hook_for_undo_row_delete(enum translog_record_type type - __attribute__ ((unused)), - TRN *trn, MARIA_HA *tbl_info, - LSN *lsn, - struct st_translog_parts *parts - __attribute__ ((unused))) -{ - tbl_info->s->state.state.records--; - return write_hook_for_undo(type, trn, tbl_info, lsn, parts); -} - - -/** - @brief Sets transaction's undo_lsn, first_undo_lsn if needed - - @todo move it to a separate file - - @return Operation status, always 0 (success) -*/ - -static my_bool write_hook_for_clr_end(enum translog_record_type type - __attribute__ ((unused)), - TRN *trn, MARIA_HA *tbl_info - __attribute__ ((unused)), - LSN *lsn - __attribute__ ((unused)), - struct st_translog_parts *parts) -{ - char *ptr= parts->parts[TRANSLOG_INTERNAL_PARTS + 0].str; - enum translog_record_type undone_record_type= - ptr[LSN_STORE_SIZE + FILEID_STORE_SIZE]; - - DBUG_ASSERT(trn->trid != 0); - trn->undo_lsn= lsn_korr(ptr); - switch (undone_record_type) { - case LOGREC_UNDO_ROW_DELETE: - tbl_info->s->state.state.records++; - break; - case LOGREC_UNDO_ROW_INSERT: - tbl_info->s->state.state.records--; - break; - case LOGREC_UNDO_ROW_UPDATE: - break; - default: - DBUG_ASSERT(0); - } - if (trn->undo_lsn == LSN_IMPOSSIBLE) /* has fully rolled back */ - trn->first_undo_lsn= LSN_WITH_FLAGS_TO_FLAGS(trn->first_undo_lsn); - return 0; -} - - -/** - @brief Updates table's lsn_of_file_id. - - @todo move it to a separate file - - @return Operation status, always 0 (success) -*/ - -static my_bool write_hook_for_file_id(enum translog_record_type type - __attribute__ ((unused)), - TRN *trn - __attribute__ ((unused)), - MARIA_HA *tbl_info, - LSN *lsn - __attribute__ ((unused)), - struct st_translog_parts *parts - __attribute__ ((unused))) -{ - DBUG_ASSERT(cmp_translog_addr(tbl_info->s->lsn_of_file_id, *lsn) < 0); - tbl_info->s->lsn_of_file_id= *lsn; - return 0; -} - - /** @brief Gives a 2-byte-id to MARIA_SHARE and logs this fact @@ -6610,7 +6388,7 @@ int translog_assign_id_to_share(MARIA_HA *tbl_info, TRN *trn) log_array[TRANSLOG_INTERNAL_PARTS + 1].length, sizeof(log_array)/sizeof(log_array[0]), - log_array, log_data))) + log_array, log_data, NULL))) return 1; } pthread_mutex_unlock(&share->intern_lock); diff --git a/storage/maria/ma_loghandler.h b/storage/maria/ma_loghandler.h index b13984d88ef..9ea3bfca263 100644 --- a/storage/maria/ma_loghandler.h +++ b/storage/maria/ma_loghandler.h @@ -48,6 +48,7 @@ typedef uint16 SHORT_TRANSACTION_ID; struct st_maria_info; +/* Changing one of the "SIZE" below will break backward-compatibility! */ /* Length of CRC at end of pages */ #define CRC_LENGTH 4 /* Size of file id in logs */ @@ -57,16 +58,23 @@ struct st_maria_info; /* Size of page ranges in log */ #define PAGERANGE_STORE_SIZE ROW_EXTENT_COUNT_SIZE #define DIRPOS_STORE_SIZE 1 +#define CLR_TYPE_STORE_SIZE 1 +/* If table has live checksum we store its changes in UNDOs */ +#define HA_CHECKSUM_STORE_SIZE 4 /* Store methods to match the above sizes */ #define fileid_store(T,A) int2store(T,A) #define page_store(T,A) int5store(T,A) #define dirpos_store(T,A) ((*(uchar*) (T)) = A) #define pagerange_store(T,A) int2store(T,A) +#define clr_type_store(T,A) ((*(uchar*) (T)) = A) +#define ha_checksum_store(T,A) int4store(T,A) #define fileid_korr(P) uint2korr(P) #define page_korr(P) uint5korr(P) #define dirpos_korr(P) ((P)[0]) #define pagerange_korr(P) uint2korr(P) +#define clr_type_korr(P) ((P)[0]) +#define ha_checksum_korr(P) uint4korr(P) /* Length of disk drive sector size (we assume that writing it @@ -230,7 +238,8 @@ translog_write_record(LSN *lsn, enum translog_record_type type, struct st_transaction *trn, struct st_maria_info *tbl_info, translog_size_t rec_len, uint part_no, - LEX_STRING *parts_data, uchar *store_share_id); + LEX_STRING *parts_data, uchar *store_share_id, + void *hook_arg); extern void translog_destroy(); @@ -299,12 +308,11 @@ struct st_translog_parts typedef my_bool(*prewrite_rec_hook) (enum translog_record_type type, TRN *trn, struct st_maria_info *tbl_info, - struct st_translog_parts *parts); + void *hook_arg); typedef my_bool(*inwrite_rec_hook) (enum translog_record_type type, TRN *trn, struct st_maria_info *tbl_info, - LSN *lsn, - struct st_translog_parts *parts); + LSN *lsn, void *hook_arg); typedef uint16(*read_rec_hook) (enum translog_record_type type, uint16 read_length, uchar *read_buff, diff --git a/storage/maria/ma_open.c b/storage/maria/ma_open.c index 6852d5b4a6d..b736fc6e9dc 100644 --- a/storage/maria/ma_open.c +++ b/storage/maria/ma_open.c @@ -177,7 +177,7 @@ static MARIA_HA *maria_clone_internal(MARIA_SHARE *share, int mode, share->delay_key_write=1; info.state= &share->state.state; /* Change global values by default */ - if (!share->base.born_transactional) /* but for transactional ones ... */ + if (!share->base.born_transactional) /* For transactional ones ... */ info.trn= &dummy_transaction_object; /* ... force crash if no trn given */ pthread_mutex_unlock(&share->intern_lock); diff --git a/storage/maria/ma_pagecache.c b/storage/maria/ma_pagecache.c index 4b0ccd2c9c8..4a3fc555a8a 100755 --- a/storage/maria/ma_pagecache.c +++ b/storage/maria/ma_pagecache.c @@ -3941,7 +3941,7 @@ my_bool pagecache_collect_changed_blocks_with_lsn(PAGECACHE *pagecache, } } - compile_time_assert(sizeof(pagecache->blocks == 4)); + compile_time_assert(sizeof(pagecache->blocks) <= 4); str->length= 4 + /* number of dirty pages */ (4 + /* file */ 4 + /* pageno */ @@ -3963,8 +3963,8 @@ my_bool pagecache_collect_changed_blocks_with_lsn(PAGECACHE *pagecache, { if (block->type != PAGECACHE_LSN_PAGE) continue; /* no need to store it in the checkpoint record */ - compile_time_assert((4 == sizeof(block->hash_link->file.file))); - compile_time_assert((4 == sizeof(block->hash_link->pageno))); + compile_time_assert(sizeof(block->hash_link->file.file) <= 4); + compile_time_assert(sizeof(block->hash_link->pageno) <= 4); int4store(ptr, block->hash_link->file.file); ptr+= 4; int4store(ptr, block->hash_link->pageno); diff --git a/storage/maria/ma_recovery.c b/storage/maria/ma_recovery.c index 6967571655e..1ae2244159c 100644 --- a/storage/maria/ma_recovery.c +++ b/storage/maria/ma_recovery.c @@ -50,6 +50,7 @@ static LSN current_group_end_lsn, static TrID max_long_trid= 0; /**< max long trid seen by REDO phase */ static FILE *tracef; /**< trace file for debugging */ static my_bool skip_DDLs; /**< if REDO phase should skip DDL records */ +static ulonglong now; /**< for tracking execution time of phases */ #define prototype_redo_exec_hook(R) \ static int exec_REDO_LOGREC_ ## R(const TRANSLOG_HEADER_BUFFER *rec) @@ -312,8 +313,12 @@ end: log_record_buffer.length= 0; if (tracef != stdout && redo_phase_message_printed) { + ulonglong old_now= now; + now= my_getsystime(); + float previous_phase_took= (now - old_now)/10000000.0; /** @todo RECOVERY BUG all prints to stderr should go to error log */ - fprintf(stderr, "\n"); + /** @todo RECOVERY BUG all prints to stderr should go to error log */ + fprintf(stderr, " (%.1f seconds)\n", previous_phase_took); } /* we don't cleanly close tables if we hit some error (may corrupt them) */ DBUG_RETURN(error); @@ -1211,12 +1216,25 @@ prototype_redo_exec_hook(UNDO_ROW_INSERT) MARIA_HA *info= get_MARIA_HA_from_UNDO_record(rec); if (info == NULL) return 0; + MARIA_SHARE *share= info->s; set_undo_lsn_for_active_trans(rec->short_trid, rec->lsn); - if (cmp_translog_addr(rec->lsn, info->s->state.is_of_horizon) >= 0) + if (cmp_translog_addr(rec->lsn, share->state.is_of_horizon) >= 0) { tprint(tracef, " state older than record, updating rows' count\n"); - info->s->state.state.records++; - /** @todo RECOVERY BUG Also update the table's checksum */ + share->state.state.records++; + if (share->calc_checksum) + { + uchar buff[HA_CHECKSUM_STORE_SIZE]; + if (translog_read_record(rec->lsn, LSN_STORE_SIZE + FILEID_STORE_SIZE + + PAGE_STORE_SIZE + DIRPOS_STORE_SIZE, + HA_CHECKSUM_STORE_SIZE, buff, NULL) != + HA_CHECKSUM_STORE_SIZE) + { + tprint(tracef, "Failed to read record\n"); + return 1; + } + share->state.state.checksum+= ha_checksum_korr(buff); + } /** @todo some bits below will rather be set when executing UNDOs related to keys @@ -1234,15 +1252,29 @@ prototype_redo_exec_hook(UNDO_ROW_DELETE) MARIA_HA *info= get_MARIA_HA_from_UNDO_record(rec); if (info == NULL) return 0; + MARIA_SHARE *share= info->s; set_undo_lsn_for_active_trans(rec->short_trid, rec->lsn); - if (cmp_translog_addr(rec->lsn, info->s->state.is_of_horizon) >= 0) + if (cmp_translog_addr(rec->lsn, share->state.is_of_horizon) >= 0) { tprint(tracef, " state older than record, updating rows' count\n"); - info->s->state.state.records--; - info->s->state.changed|= STATE_CHANGED | STATE_NOT_ANALYZED | + share->state.state.records--; + if (share->calc_checksum) + { + uchar buff[HA_CHECKSUM_STORE_SIZE]; + if (translog_read_record(rec->lsn, LSN_STORE_SIZE + FILEID_STORE_SIZE + + PAGE_STORE_SIZE + DIRPOS_STORE_SIZE, + HA_CHECKSUM_STORE_SIZE, buff, NULL) != + HA_CHECKSUM_STORE_SIZE) + { + tprint(tracef, "Failed to read record\n"); + return 1; + } + share->state.state.checksum+= ha_checksum_korr(buff); + } + share->state.changed|= STATE_CHANGED | STATE_NOT_ANALYZED | STATE_NOT_OPTIMIZED_KEYS | STATE_NOT_SORTED_PAGES; } - tprint(tracef, " rows' count %lu\n", (ulong)info->s->state.state.records); + tprint(tracef, " rows' count %lu\n", (ulong)share->state.state.records); return 0; } @@ -1252,10 +1284,24 @@ prototype_redo_exec_hook(UNDO_ROW_UPDATE) MARIA_HA *info= get_MARIA_HA_from_UNDO_record(rec); if (info == NULL) return 0; + MARIA_SHARE *share= info->s; set_undo_lsn_for_active_trans(rec->short_trid, rec->lsn); - if (cmp_translog_addr(rec->lsn, info->s->state.is_of_horizon) >= 0) + if (cmp_translog_addr(rec->lsn, share->state.is_of_horizon) >= 0) { - info->s->state.changed|= STATE_CHANGED | STATE_NOT_ANALYZED | + if (share->calc_checksum) + { + uchar buff[HA_CHECKSUM_STORE_SIZE]; + if (translog_read_record(rec->lsn, LSN_STORE_SIZE + FILEID_STORE_SIZE + + PAGE_STORE_SIZE + DIRPOS_STORE_SIZE, + HA_CHECKSUM_STORE_SIZE, buff, NULL) != + HA_CHECKSUM_STORE_SIZE) + { + tprint(tracef, "Failed to read record\n"); + return 1; + } + share->state.state.checksum+= ha_checksum_korr(buff); + } + share->state.changed|= STATE_CHANGED | STATE_NOT_ANALYZED | STATE_NOT_OPTIMIZED_KEYS | STATE_NOT_SORTED_PAGES; } return 0; @@ -1306,33 +1352,46 @@ prototype_redo_exec_hook(CLR_END) MARIA_HA *info= get_MARIA_HA_from_UNDO_record(rec); if (info == NULL) return 0; + MARIA_SHARE *share= info->s; LSN previous_undo_lsn= lsn_korr(rec->header); enum translog_record_type undone_record_type= - (rec->header)[LSN_STORE_SIZE + FILEID_STORE_SIZE]; + clr_type_korr(rec->header + LSN_STORE_SIZE + FILEID_STORE_SIZE); const LOG_DESC *log_desc= &log_record_type_descriptor[undone_record_type]; set_undo_lsn_for_active_trans(rec->short_trid, previous_undo_lsn); tprint(tracef, " CLR_END was about %s, undo_lsn now LSN (%lu,0x%lx)\n", log_desc->name, LSN_IN_PARTS(previous_undo_lsn)); - if (cmp_translog_addr(rec->lsn, info->s->state.is_of_horizon) >= 0) + if (cmp_translog_addr(rec->lsn, share->state.is_of_horizon) >= 0) { tprint(tracef, " state older than record, updating rows' count\n"); + if (share->calc_checksum) + { + uchar buff[HA_CHECKSUM_STORE_SIZE]; + if (translog_read_record(rec->lsn, LSN_STORE_SIZE + FILEID_STORE_SIZE + + CLR_TYPE_STORE_SIZE, HA_CHECKSUM_STORE_SIZE, + buff, NULL) != HA_CHECKSUM_STORE_SIZE) + { + tprint(tracef, "Failed to read record\n"); + return 1; + } + share->state.state.checksum+= ha_checksum_korr(buff); + } switch (undone_record_type) { case LOGREC_UNDO_ROW_DELETE: - info->s->state.state.records++; + share->state.state.records++; break; case LOGREC_UNDO_ROW_INSERT: - info->s->state.state.records--; + share->state.state.records--; break; case LOGREC_UNDO_ROW_UPDATE: break; default: DBUG_ASSERT(0); } - info->s->state.changed|= STATE_CHANGED | STATE_NOT_ANALYZED | + share->state.changed|= STATE_CHANGED | STATE_NOT_ANALYZED | STATE_NOT_OPTIMIZED_KEYS | STATE_NOT_SORTED_PAGES; } - tprint(tracef, " rows' count %lu\n", (ulong)info->s->state.state.records); + tprint(tracef, " rows' count %lu\n", (ulong)share->state.state.records); return 0; } @@ -1353,12 +1412,33 @@ prototype_undo_exec_hook(UNDO_ROW_INSERT) */ return 1; } - info->s->state.changed|= STATE_CHANGED | STATE_NOT_ANALYZED | + MARIA_SHARE *share= info->s; + share->state.changed|= STATE_CHANGED | STATE_NOT_ANALYZED | STATE_NOT_OPTIMIZED_KEYS | STATE_NOT_SORTED_PAGES; + const uchar *record_ptr= rec->header; + if (share->calc_checksum) + { + /* + We need to read more of the record to put the checksum into the record + buffer used by _ma_apply_undo_row_insert(). + If the table has no live checksum, rec->header will be enough. + */ + enlarge_buffer(rec); + if (log_record_buffer.str == NULL || + translog_read_record(rec->lsn, 0, rec->record_length, + log_record_buffer.str, NULL) != + rec->record_length) + { + tprint(tracef, "Failed to read record\n"); + return 1; + } + record_ptr= log_record_buffer.str; + } + info->trn= trn; error= _ma_apply_undo_row_insert(info, previous_undo_lsn, - rec->header + LSN_STORE_SIZE + + record_ptr + LSN_STORE_SIZE + FILEID_STORE_SIZE); info->trn= 0; /* trn->undo_lsn is updated in an inwrite_hook when writing the CLR_END */ @@ -1378,7 +1458,8 @@ prototype_undo_exec_hook(UNDO_ROW_DELETE) if (info == NULL) return 1; - info->s->state.changed|= STATE_CHANGED | STATE_NOT_ANALYZED | + MARIA_SHARE *share= info->s; + share->state.changed|= STATE_CHANGED | STATE_NOT_ANALYZED | STATE_NOT_OPTIMIZED_KEYS | STATE_NOT_SORTED_PAGES; enlarge_buffer(rec); @@ -1405,8 +1486,7 @@ prototype_undo_exec_hook(UNDO_ROW_DELETE) PAGE_STORE_SIZE + DIRPOS_STORE_SIZE)); info->trn= 0; tprint(tracef, " rows' count %lu\n undo_lsn now LSN (%lu,0x%lx)\n", - (ulong)info->s->state.state.records, - LSN_IN_PARTS(previous_undo_lsn)); + (ulong)share->state.state.records, LSN_IN_PARTS(previous_undo_lsn)); return error; } @@ -1419,8 +1499,8 @@ prototype_undo_exec_hook(UNDO_ROW_UPDATE) if (info == NULL) return 1; - - info->s->state.changed|= STATE_CHANGED | STATE_NOT_ANALYZED | + MARIA_SHARE *share= info->s; + share->state.changed|= STATE_CHANGED | STATE_NOT_ANALYZED | STATE_NOT_OPTIMIZED_KEYS | STATE_NOT_SORTED_PAGES; enlarge_buffer(rec); @@ -1634,7 +1714,7 @@ static uint end_of_redo_phase(my_bool prepare_for_undo_phase) hash_free(&all_dirty_pages); /* - hash_free() can be called multiple times probably, but be safe it that + hash_free() can be called multiple times probably, but be safe if that changes */ bzero(&all_dirty_pages, sizeof(all_dirty_pages)); @@ -1652,11 +1732,8 @@ static uint end_of_redo_phase(my_bool prepare_for_undo_phase) LSN gslsn= all_active_trans[sid].group_start_lsn; TRN *trn; if (gslsn != LSN_IMPOSSIBLE) - { tprint(tracef, "Group at LSN (%lu,0x%lx) short_trid %u aborted\n", LSN_IN_PARTS(gslsn), sid); - ALERT_USER(); - } if (all_active_trans[sid].undo_lsn != LSN_IMPOSSIBLE) { char llbuf[22]; @@ -1734,8 +1811,12 @@ static int run_undo_phase(uint unfinished) { if (tracef != stdout) { + ulonglong old_now= now; + now= my_getsystime(); + float previous_phase_took= (now - old_now)/10000000.0; /** @todo RECOVERY BUG all prints to stderr should go to error log */ - fprintf(stderr, " 100%%; transactions to roll back:"); + fprintf(stderr, " 100%% (%.1f seconds); transactions to roll back:", + previous_phase_took); } tprint(tracef, "%u transactions will be rolled back\n", unfinished); for( ; ; ) @@ -2070,8 +2151,11 @@ static int close_all_tables(void) tprint(tracef, "Closing all tables\n"); if (tracef != stdout && redo_phase_message_printed) { + ulonglong old_now= now; + now= my_getsystime(); + float previous_phase_took= (now - old_now)/10000000.0; /** @todo RECOVERY BUG all prints to stderr should go to error log */ - fprintf(stderr, "; flushing tables"); + fprintf(stderr, " (%.1f seconds); flushing tables", previous_phase_took); } /* @@ -2141,6 +2225,7 @@ static void print_redo_phase_progress(TRANSLOG_ADDRESS addr) /** @todo RECOVERY BUG all prints to stderr should go to error log */ fprintf(stderr, "Maria engine: starting recovery; recovered pages: 0%%"); redo_phase_message_printed= TRUE; + now= my_getsystime(); } if (end_logno == FILENO_IMPOSSIBLE) { @@ -2166,7 +2251,7 @@ static void print_redo_phase_progress(TRANSLOG_ADDRESS addr) } #ifdef MARIA_EXTERNAL_LOCKING -#error Maria's Checkpoint and Recovery are really not ready for it +#error Marias Checkpoint and Recovery are really not ready for it #endif /* diff --git a/storage/maria/ma_rename.c b/storage/maria/ma_rename.c index 44cd60711da..57d35f6c066 100644 --- a/storage/maria/ma_rename.c +++ b/storage/maria/ma_rename.c @@ -85,7 +85,7 @@ int maria_rename(const char *old_name, const char *new_name) &dummy_transaction_object, NULL, old_name_len + new_name_len, sizeof(log_array)/sizeof(log_array[0]), - log_array, NULL) || + log_array, NULL, NULL) || translog_flush(lsn))) { maria_close(info); diff --git a/storage/maria/ma_test_recovery.expected b/storage/maria/ma_test_recovery.expected index 926943b11b3..70339fc69b6 100644 --- a/storage/maria/ma_test_recovery.expected +++ b/storage/maria/ma_test_recovery.expected @@ -4,10 +4,6 @@ TEST WITH ma_test1 -s -M -T -c applying log Differences in maria_chk -dvv, recovery not yet perfect ! ========DIFF START======= -7c7 -< Checksum: 3757530372 ---- -> Checksum: 0 11c11 < Datafile length: 16384 Keyfile length: 16384 --- @@ -21,10 +17,6 @@ testing idempotency applying log Differences in maria_chk -dvv, recovery not yet perfect ! ========DIFF START======= -7c7 -< Checksum: 3757530372 ---- -> Checksum: 0 11c11 < Datafile length: 16384 Keyfile length: 16384 --- @@ -78,10 +70,6 @@ Dying on request without maria_commit()/maria_close() applying log Differences in maria_chk -dvv, recovery not yet perfect ! ========DIFF START======= -7c7 -< Checksum: 221293111 ---- -> Checksum: 0 11c11 < Datafile length: 16384 Keyfile length: 16384 --- @@ -95,10 +83,6 @@ testing idempotency applying log Differences in maria_chk -dvv, recovery not yet perfect ! ========DIFF START======= -7c7 -< Checksum: 221293111 ---- -> Checksum: 0 11c11 < Datafile length: 16384 Keyfile length: 16384 --- @@ -112,10 +96,6 @@ testing applying of CLRs to recreate table applying log Differences in maria_chk -dvv, recovery not yet perfect ! ========DIFF START======= -7c7 -< Checksum: 221293111 ---- -> Checksum: 0 11c11 < Datafile length: 16384 Keyfile length: 16384 --- @@ -137,10 +117,6 @@ testing applying of CLRs to recreate table applying log Differences in maria_chk -dvv, recovery not yet perfect ! ========DIFF START======= -7c7 -< Checksum: 3697324514 ---- -> Checksum: 0 11c11 < Datafile length: 16384 Keyfile length: 16384 --- @@ -158,10 +134,6 @@ Dying on request without maria_commit()/maria_close() applying log Differences in maria_chk -dvv, recovery not yet perfect ! ========DIFF START======= -7c7 -< Checksum: 2428948025 ---- -> Checksum: 3026590807 11c11 < Datafile length: 16384 Keyfile length: 16384 --- @@ -175,10 +147,6 @@ testing idempotency applying log Differences in maria_chk -dvv, recovery not yet perfect ! ========DIFF START======= -7c7 -< Checksum: 2428948025 ---- -> Checksum: 3026590807 11c11 < Datafile length: 16384 Keyfile length: 16384 --- @@ -192,10 +160,6 @@ testing applying of CLRs to recreate table applying log Differences in maria_chk -dvv, recovery not yet perfect ! ========DIFF START======= -7c7 -< Checksum: 2428948025 ---- -> Checksum: 0 11c11 < Datafile length: 16384 Keyfile length: 16384 --- @@ -253,10 +217,6 @@ Dying on request without maria_commit()/maria_close() applying log Differences in maria_chk -dvv, recovery not yet perfect ! ========DIFF START======= -7c7 -< Checksum: 221293111 ---- -> Checksum: 0 11c11 < Datafile length: 16384 Keyfile length: 16384 --- @@ -270,10 +230,6 @@ testing idempotency applying log Differences in maria_chk -dvv, recovery not yet perfect ! ========DIFF START======= -7c7 -< Checksum: 221293111 ---- -> Checksum: 0 11c11 < Datafile length: 16384 Keyfile length: 16384 --- @@ -287,10 +243,6 @@ testing applying of CLRs to recreate table applying log Differences in maria_chk -dvv, recovery not yet perfect ! ========DIFF START======= -7c7 -< Checksum: 221293111 ---- -> Checksum: 0 11c11 < Datafile length: 16384 Keyfile length: 16384 --- @@ -312,10 +264,6 @@ testing applying of CLRs to recreate table applying log Differences in maria_chk -dvv, recovery not yet perfect ! ========DIFF START======= -7c7 -< Checksum: 3697324514 ---- -> Checksum: 0 11c11 < Datafile length: 16384 Keyfile length: 16384 --- @@ -333,10 +281,6 @@ Dying on request without maria_commit()/maria_close() applying log Differences in maria_chk -dvv, recovery not yet perfect ! ========DIFF START======= -7c7 -< Checksum: 2428948025 ---- -> Checksum: 3026590807 11c11 < Datafile length: 16384 Keyfile length: 16384 --- @@ -350,10 +294,6 @@ testing idempotency applying log Differences in maria_chk -dvv, recovery not yet perfect ! ========DIFF START======= -7c7 -< Checksum: 2428948025 ---- -> Checksum: 3026590807 11c11 < Datafile length: 16384 Keyfile length: 16384 --- @@ -367,10 +307,6 @@ testing applying of CLRs to recreate table applying log Differences in maria_chk -dvv, recovery not yet perfect ! ========DIFF START======= -7c7 -< Checksum: 2428948025 ---- -> Checksum: 0 11c11 < Datafile length: 16384 Keyfile length: 16384 --- @@ -428,10 +364,6 @@ Dying on request without maria_commit()/maria_close() applying log Differences in maria_chk -dvv, recovery not yet perfect ! ========DIFF START======= -7c7 -< Checksum: 221293111 ---- -> Checksum: 0 11c11 < Datafile length: 16384 Keyfile length: 16384 --- @@ -445,10 +377,6 @@ testing idempotency applying log Differences in maria_chk -dvv, recovery not yet perfect ! ========DIFF START======= -7c7 -< Checksum: 221293111 ---- -> Checksum: 0 11c11 < Datafile length: 16384 Keyfile length: 16384 --- @@ -462,10 +390,6 @@ testing applying of CLRs to recreate table applying log Differences in maria_chk -dvv, recovery not yet perfect ! ========DIFF START======= -7c7 -< Checksum: 221293111 ---- -> Checksum: 0 11c11 < Datafile length: 16384 Keyfile length: 16384 --- @@ -487,10 +411,6 @@ testing applying of CLRs to recreate table applying log Differences in maria_chk -dvv, recovery not yet perfect ! ========DIFF START======= -7c7 -< Checksum: 3697324514 ---- -> Checksum: 0 11c11 < Datafile length: 16384 Keyfile length: 16384 --- @@ -508,10 +428,6 @@ Dying on request without maria_commit()/maria_close() applying log Differences in maria_chk -dvv, recovery not yet perfect ! ========DIFF START======= -7c7 -< Checksum: 2428948025 ---- -> Checksum: 0 11c11 < Datafile length: 16384 Keyfile length: 16384 --- @@ -525,10 +441,6 @@ testing idempotency applying log Differences in maria_chk -dvv, recovery not yet perfect ! ========DIFF START======= -7c7 -< Checksum: 2428948025 ---- -> Checksum: 0 11c11 < Datafile length: 16384 Keyfile length: 16384 --- @@ -542,10 +454,6 @@ testing applying of CLRs to recreate table applying log Differences in maria_chk -dvv, recovery not yet perfect ! ========DIFF START======= -7c7 -< Checksum: 2428948025 ---- -> Checksum: 0 11c11 < Datafile length: 16384 Keyfile length: 16384 --- @@ -603,10 +511,6 @@ Dying on request without maria_commit()/maria_close() applying log Differences in maria_chk -dvv, recovery not yet perfect ! ========DIFF START======= -7c7 -< Checksum: 411409161 ---- -> Checksum: 0 11c11 < Datafile length: 49152 Keyfile length: 16384 --- @@ -620,10 +524,6 @@ testing idempotency applying log Differences in maria_chk -dvv, recovery not yet perfect ! ========DIFF START======= -7c7 -< Checksum: 411409161 ---- -> Checksum: 0 11c11 < Datafile length: 49152 Keyfile length: 16384 --- @@ -637,10 +537,6 @@ testing applying of CLRs to recreate table applying log Differences in maria_chk -dvv, recovery not yet perfect ! ========DIFF START======= -7c7 -< Checksum: 411409161 ---- -> Checksum: 0 11c11 < Datafile length: 49152 Keyfile length: 16384 --- @@ -662,10 +558,6 @@ testing applying of CLRs to recreate table applying log Differences in maria_chk -dvv, recovery not yet perfect ! ========DIFF START======= -7c7 -< Checksum: 4024695312 ---- -> Checksum: 0 11c11 < Datafile length: 49152 Keyfile length: 16384 --- @@ -683,10 +575,6 @@ Dying on request without maria_commit()/maria_close() applying log Differences in maria_chk -dvv, recovery not yet perfect ! ========DIFF START======= -7c7 -< Checksum: 529753687 ---- -> Checksum: 800025671 11c11 < Datafile length: 49152 Keyfile length: 16384 --- @@ -700,10 +588,6 @@ testing idempotency applying log Differences in maria_chk -dvv, recovery not yet perfect ! ========DIFF START======= -7c7 -< Checksum: 529753687 ---- -> Checksum: 800025671 11c11 < Datafile length: 49152 Keyfile length: 16384 --- @@ -717,10 +601,6 @@ testing applying of CLRs to recreate table applying log Differences in maria_chk -dvv, recovery not yet perfect ! ========DIFF START======= -7c7 -< Checksum: 529753687 ---- -> Checksum: 0 11c11 < Datafile length: 49152 Keyfile length: 16384 --- @@ -778,10 +658,6 @@ Dying on request without maria_commit()/maria_close() applying log Differences in maria_chk -dvv, recovery not yet perfect ! ========DIFF START======= -7c7 -< Checksum: 411409161 ---- -> Checksum: 0 11c11 < Datafile length: 49152 Keyfile length: 16384 --- @@ -795,10 +671,6 @@ testing idempotency applying log Differences in maria_chk -dvv, recovery not yet perfect ! ========DIFF START======= -7c7 -< Checksum: 411409161 ---- -> Checksum: 0 11c11 < Datafile length: 49152 Keyfile length: 16384 --- @@ -812,10 +684,6 @@ testing applying of CLRs to recreate table applying log Differences in maria_chk -dvv, recovery not yet perfect ! ========DIFF START======= -7c7 -< Checksum: 411409161 ---- -> Checksum: 0 11c11 < Datafile length: 49152 Keyfile length: 16384 --- @@ -837,10 +705,6 @@ testing applying of CLRs to recreate table applying log Differences in maria_chk -dvv, recovery not yet perfect ! ========DIFF START======= -7c7 -< Checksum: 4024695312 ---- -> Checksum: 0 11c11 < Datafile length: 49152 Keyfile length: 16384 --- @@ -858,10 +722,6 @@ Dying on request without maria_commit()/maria_close() applying log Differences in maria_chk -dvv, recovery not yet perfect ! ========DIFF START======= -7c7 -< Checksum: 529753687 ---- -> Checksum: 800025671 11c11 < Datafile length: 49152 Keyfile length: 16384 --- @@ -875,10 +735,6 @@ testing idempotency applying log Differences in maria_chk -dvv, recovery not yet perfect ! ========DIFF START======= -7c7 -< Checksum: 529753687 ---- -> Checksum: 800025671 11c11 < Datafile length: 49152 Keyfile length: 16384 --- @@ -892,10 +748,6 @@ testing applying of CLRs to recreate table applying log Differences in maria_chk -dvv, recovery not yet perfect ! ========DIFF START======= -7c7 -< Checksum: 529753687 ---- -> Checksum: 0 11c11 < Datafile length: 49152 Keyfile length: 16384 --- @@ -953,10 +805,6 @@ Dying on request without maria_commit()/maria_close() applying log Differences in maria_chk -dvv, recovery not yet perfect ! ========DIFF START======= -7c7 -< Checksum: 411409161 ---- -> Checksum: 0 11c11 < Datafile length: 49152 Keyfile length: 16384 --- @@ -970,10 +818,6 @@ testing idempotency applying log Differences in maria_chk -dvv, recovery not yet perfect ! ========DIFF START======= -7c7 -< Checksum: 411409161 ---- -> Checksum: 0 11c11 < Datafile length: 49152 Keyfile length: 16384 --- @@ -987,10 +831,6 @@ testing applying of CLRs to recreate table applying log Differences in maria_chk -dvv, recovery not yet perfect ! ========DIFF START======= -7c7 -< Checksum: 411409161 ---- -> Checksum: 0 11c11 < Datafile length: 49152 Keyfile length: 16384 --- @@ -1012,10 +852,6 @@ testing applying of CLRs to recreate table applying log Differences in maria_chk -dvv, recovery not yet perfect ! ========DIFF START======= -7c7 -< Checksum: 4024695312 ---- -> Checksum: 0 11c11 < Datafile length: 49152 Keyfile length: 16384 --- @@ -1033,10 +869,6 @@ Dying on request without maria_commit()/maria_close() applying log Differences in maria_chk -dvv, recovery not yet perfect ! ========DIFF START======= -7c7 -< Checksum: 529753687 ---- -> Checksum: 0 11c11 < Datafile length: 49152 Keyfile length: 16384 --- @@ -1050,10 +882,6 @@ testing idempotency applying log Differences in maria_chk -dvv, recovery not yet perfect ! ========DIFF START======= -7c7 -< Checksum: 529753687 ---- -> Checksum: 0 11c11 < Datafile length: 49152 Keyfile length: 16384 --- @@ -1067,10 +895,6 @@ testing applying of CLRs to recreate table applying log Differences in maria_chk -dvv, recovery not yet perfect ! ========DIFF START======= -7c7 -< Checksum: 529753687 ---- -> Checksum: 0 11c11 < Datafile length: 49152 Keyfile length: 16384 --- diff --git a/storage/maria/ma_update.c b/storage/maria/ma_update.c index 0cb2e2b648b..1d138c13273 100644 --- a/storage/maria/ma_update.c +++ b/storage/maria/ma_update.c @@ -27,11 +27,9 @@ int maria_update(register MARIA_HA *info, const uchar *oldrec, uchar *newrec) bool auto_key_changed=0; ulonglong changed; MARIA_SHARE *share=info->s; - ha_checksum old_checksum; DBUG_ENTER("maria_update"); LINT_INIT(new_key); LINT_INIT(changed); - LINT_INIT(old_checksum); DBUG_EXECUTE_IF("maria_pretend_crashed_table_on_usage", maria_print_error(info->s, HA_ERR_CRASHED); @@ -59,16 +57,6 @@ int maria_update(register MARIA_HA *info, const uchar *oldrec, uchar *newrec) goto err_end; /* Record has changed */ } - if (share->calc_checksum) - { - /* - We can't use the row based checksum as this doesn't have enough - precision. - */ - if (info->s->calc_checksum) - old_checksum= (*info->s->calc_checksum)(info, oldrec); - } - /* Calculate and check all unique constraints */ key_changed=0; for (i=0 ; i < share->state.header.uniques ; i++) @@ -137,19 +125,23 @@ int maria_update(register MARIA_HA *info, const uchar *oldrec, uchar *newrec) } } } - /* - If we are running with external locking, we must update the index file - that something has changed. - */ - if (changed || !my_disable_locking) - key_changed|= HA_STATE_CHANGED; if (share->calc_checksum) { - info->cur_row.checksum= (*share->calc_checksum)(info,newrec); - info->state->checksum+= (info->cur_row.checksum - old_checksum); - /* Store new checksum in index file header */ - key_changed|= HA_STATE_CHANGED; + /* + We can't use the row based checksum as this doesn't have enough + precision (one byte, while the table's is more bytes). + At least _ma_check_unique() modifies the 'newrec' record, so checksum + has to be computed _after_ it. Nobody apparently modifies 'oldrec'. + We need to pass the old row's checksum down to (*update_record)(), we do + this via info->new_row.checksum (not intuitive but existing code + mandated that cur_row is the new row). + If (*update_record)() fails, table will be marked corrupted so no need + to revert the live checksum change. + */ + info->state->checksum+= !share->now_transactional * + ((info->cur_row.checksum= (*share->calc_checksum)(info, newrec)) - + (info->new_row.checksum= (*share->calc_checksum)(info, oldrec))); } { /* @@ -165,14 +157,9 @@ int maria_update(register MARIA_HA *info, const uchar *oldrec, uchar *newrec) org_delete_link= share->state.dellink; if ((*share->update_record)(info, pos, oldrec, newrec)) goto err; - if (!key_changed && - (memcmp((char*) &state, (char*) info->state, sizeof(state)) || - org_split != share->state.split || - org_delete_link != share->state.dellink)) - key_changed|= HA_STATE_CHANGED; /* Must update index file */ } if (auto_key_changed) - set_if_bigger(info->s->state.auto_increment, + set_if_bigger(share->state.auto_increment, ma_retrieve_auto_increment(info, newrec)); /* @@ -195,8 +182,8 @@ int maria_update(register MARIA_HA *info, const uchar *oldrec, uchar *newrec) allow_break(); /* Allow SIGHUP & SIGINT */ if (info->invalidator != 0) { - DBUG_PRINT("info", ("invalidator... '%s' (update)", info->s->open_file_name)); - (*info->invalidator)(info->s->open_file_name); + DBUG_PRINT("info", ("invalidator... '%s' (update)", share->open_file_name)); + (*info->invalidator)(share->open_file_name); info->invalidator=0; } DBUG_RETURN(0); @@ -232,7 +219,7 @@ err: } else { - maria_print_error(info->s, HA_ERR_CRASHED); + maria_print_error(share, HA_ERR_CRASHED); maria_mark_crashed(info); } info->update= (HA_STATE_CHANGED | HA_STATE_AKTIV | HA_STATE_ROW_CHANGED | @@ -243,7 +230,7 @@ err: allow_break(); /* Allow SIGHUP & SIGINT */ if (save_errno == HA_ERR_KEY_NOT_FOUND) { - maria_print_error(info->s, HA_ERR_CRASHED); + maria_print_error(share, HA_ERR_CRASHED); save_errno=HA_ERR_CRASHED; } DBUG_RETURN(my_errno=save_errno); diff --git a/storage/maria/ma_write.c b/storage/maria/ma_write.c index b034d71ef9d..18892475229 100644 --- a/storage/maria/ma_write.c +++ b/storage/maria/ma_write.c @@ -162,10 +162,6 @@ int maria_write(MARIA_HA *info, uchar *record) rw_unlock(&share->key_root_lock[i]); } } - /** - @todo RECOVERY BUG - this += must happen under log's mutex when writing the UNDO - */ if (share->calc_write_checksum) info->cur_row.checksum= (*share->calc_write_checksum)(info,record); if (filepos != HA_OFFSET_ERROR) @@ -176,7 +172,8 @@ int maria_write(MARIA_HA *info, uchar *record) @todo when we enable multiple writers, we will have to protect 'records' and 'checksum' somehow. */ - info->state->checksum+= info->cur_row.checksum; + info->state->checksum+= !share->now_transactional * + info->cur_row.checksum; } if (share->base.auto_key) set_if_bigger(info->s->state.auto_increment, diff --git a/storage/maria/unittest/ma_test_loghandler-t.c b/storage/maria/unittest/ma_test_loghandler-t.c index fa7267e31d4..d9627409c80 100644 --- a/storage/maria/unittest/ma_test_loghandler-t.c +++ b/storage/maria/unittest/ma_test_loghandler-t.c @@ -198,8 +198,8 @@ int main(int argc __attribute__((unused)), char *argv[]) trn->first_undo_lsn= TRANSACTION_LOGGED_LONG_ID; if (translog_write_record(&lsn, LOGREC_FIXED_RECORD_0LSN_EXAMPLE, - trn, NULL, - 6, TRANSLOG_INTERNAL_PARTS + 1, parts, NULL)) + trn, NULL, 6, TRANSLOG_INTERNAL_PARTS + 1, + parts, NULL, NULL)) { fprintf(stderr, "Can't write record #%lu\n", (ulong) 0); translog_destroy(); @@ -220,8 +220,8 @@ int main(int argc __attribute__((unused)), char *argv[]) /* check auto-count feature */ parts[TRANSLOG_INTERNAL_PARTS + 1].str= NULL; parts[TRANSLOG_INTERNAL_PARTS + 1].length= 0; - if (translog_write_record(&lsn, LOGREC_FIXED_RECORD_1LSN_EXAMPLE, - trn, NULL, LSN_STORE_SIZE, 0, parts, NULL)) + if (translog_write_record(&lsn, LOGREC_FIXED_RECORD_1LSN_EXAMPLE, trn, + NULL, LSN_STORE_SIZE, 0, parts, NULL, NULL)) { fprintf(stderr, "1 Can't write reference defore record #%lu\n", (ulong) i); @@ -241,7 +241,7 @@ int main(int argc __attribute__((unused)), char *argv[]) if (translog_write_record(&lsn, LOGREC_VARIABLE_RECORD_1LSN_EXAMPLE, trn, NULL, 0, TRANSLOG_INTERNAL_PARTS + 2, - parts, NULL)) + parts, NULL, NULL)) { fprintf(stderr, "1 Can't write var reference defore record #%lu\n", (ulong) i); @@ -259,8 +259,8 @@ int main(int argc __attribute__((unused)), char *argv[]) parts[TRANSLOG_INTERNAL_PARTS + 0].length= 23; if (translog_write_record(&lsn, LOGREC_FIXED_RECORD_2LSN_EXAMPLE, - trn, NULL, - 23, TRANSLOG_INTERNAL_PARTS + 1, parts, NULL)) + trn, NULL, 23, TRANSLOG_INTERNAL_PARTS + 1, + parts, NULL, NULL)) { fprintf(stderr, "0 Can't write reference defore record #%lu\n", (ulong) i); @@ -280,7 +280,8 @@ int main(int argc __attribute__((unused)), char *argv[]) if (translog_write_record(&lsn, LOGREC_VARIABLE_RECORD_2LSN_EXAMPLE, trn, NULL, 14 + rec_len, - TRANSLOG_INTERNAL_PARTS + 2, parts, NULL)) + TRANSLOG_INTERNAL_PARTS + 2, parts, NULL, + NULL)) { fprintf(stderr, "0 Can't write var reference defore record #%lu\n", (ulong) i); @@ -297,7 +298,7 @@ int main(int argc __attribute__((unused)), char *argv[]) LOGREC_FIXED_RECORD_0LSN_EXAMPLE, trn, NULL, 6, TRANSLOG_INTERNAL_PARTS + 1, - parts, NULL)) + parts, NULL, NULL)) { fprintf(stderr, "Can't write record #%lu\n", (ulong) i); translog_destroy(); @@ -316,7 +317,7 @@ int main(int argc __attribute__((unused)), char *argv[]) LOGREC_VARIABLE_RECORD_0LSN_EXAMPLE, trn, NULL, rec_len, TRANSLOG_INTERNAL_PARTS + 1, - parts, NULL)) + parts, NULL, NULL)) { fprintf(stderr, "Can't write variable record #%lu\n", (ulong) i); translog_destroy(); diff --git a/storage/maria/unittest/ma_test_loghandler_first_lsn-t.c b/storage/maria/unittest/ma_test_loghandler_first_lsn-t.c index 28233ae04cb..845e33b63a5 100644 --- a/storage/maria/unittest/ma_test_loghandler_first_lsn-t.c +++ b/storage/maria/unittest/ma_test_loghandler_first_lsn-t.c @@ -106,7 +106,7 @@ int main(int argc __attribute__((unused)), char *argv[]) LOGREC_FIXED_RECORD_0LSN_EXAMPLE, &dummy_transaction_object, NULL, 6, TRANSLOG_INTERNAL_PARTS + 1, - parts, NULL)) + parts, NULL, NULL)) { fprintf(stderr, "Can't write record #%lu\n", (ulong) 0); translog_destroy(); diff --git a/storage/maria/unittest/ma_test_loghandler_max_lsn-t.c b/storage/maria/unittest/ma_test_loghandler_max_lsn-t.c index d6f0bde7a8e..68561443a02 100644 --- a/storage/maria/unittest/ma_test_loghandler_max_lsn-t.c +++ b/storage/maria/unittest/ma_test_loghandler_max_lsn-t.c @@ -96,7 +96,7 @@ int main(int argc __attribute__((unused)), char *argv[]) LOGREC_FIXED_RECORD_0LSN_EXAMPLE, &dummy_transaction_object, NULL, 6, TRANSLOG_INTERNAL_PARTS + 1, - parts, NULL)) + parts, NULL, NULL)) { fprintf(stderr, "Can't write record #%lu\n", (ulong) 0); translog_destroy(); diff --git a/storage/maria/unittest/ma_test_loghandler_multigroup-t.c b/storage/maria/unittest/ma_test_loghandler_multigroup-t.c index 559cd17638f..e5afa0b40db 100644 --- a/storage/maria/unittest/ma_test_loghandler_multigroup-t.c +++ b/storage/maria/unittest/ma_test_loghandler_multigroup-t.c @@ -194,8 +194,8 @@ int main(int argc __attribute__((unused)), char *argv[]) trn->short_id= 0; trn->first_undo_lsn= TRANSACTION_LOGGED_LONG_ID; if (translog_write_record(&lsn, LOGREC_FIXED_RECORD_0LSN_EXAMPLE, - trn, NULL, - 6, TRANSLOG_INTERNAL_PARTS + 1, parts, NULL)) + trn, NULL, 6, TRANSLOG_INTERNAL_PARTS + 1, parts, + NULL, NULL)) { fprintf(stderr, "Can't write record #%lu\n", (ulong) 0); translog_destroy(); @@ -214,10 +214,9 @@ int main(int argc __attribute__((unused)), char *argv[]) parts[TRANSLOG_INTERNAL_PARTS + 0].length= LSN_STORE_SIZE; trn->short_id= i % 0xFFFF; if (translog_write_record(&lsn, - LOGREC_FIXED_RECORD_1LSN_EXAMPLE, - trn, NULL, - LSN_STORE_SIZE, - TRANSLOG_INTERNAL_PARTS + 1, parts, NULL)) + LOGREC_FIXED_RECORD_1LSN_EXAMPLE, trn, NULL, + LSN_STORE_SIZE, TRANSLOG_INTERNAL_PARTS + 1, + parts, NULL, NULL)) { fprintf(stderr, "1 Can't write reference before record #%lu\n", (ulong) i); @@ -237,7 +236,7 @@ int main(int argc __attribute__((unused)), char *argv[]) LOGREC_VARIABLE_RECORD_1LSN_EXAMPLE, trn, NULL, LSN_STORE_SIZE + rec_len, TRANSLOG_INTERNAL_PARTS + 2, - parts, NULL)) + parts, NULL, NULL)) { fprintf(stderr, "1 Can't write var reference before record #%lu\n", (ulong) i); @@ -256,9 +255,8 @@ int main(int argc __attribute__((unused)), char *argv[]) trn->short_id= i % 0xFFFF; if (translog_write_record(&lsn, LOGREC_FIXED_RECORD_2LSN_EXAMPLE, - trn, NULL, 23, - TRANSLOG_INTERNAL_PARTS + 1, - parts, NULL)) + trn, NULL, 23, TRANSLOG_INTERNAL_PARTS + 1, + parts, NULL, NULL)) { fprintf(stderr, "0 Can't write reference before record #%lu\n", (ulong) i); @@ -279,7 +277,7 @@ int main(int argc __attribute__((unused)), char *argv[]) LOGREC_VARIABLE_RECORD_2LSN_EXAMPLE, trn, NULL, LSN_STORE_SIZE * 2 + rec_len, TRANSLOG_INTERNAL_PARTS + 2, - parts, NULL)) + parts, NULL, NULL)) { fprintf(stderr, "0 Can't write var reference before record #%lu\n", (ulong) i); @@ -296,7 +294,7 @@ int main(int argc __attribute__((unused)), char *argv[]) if (translog_write_record(&lsn, LOGREC_FIXED_RECORD_0LSN_EXAMPLE, trn, NULL, 6, - TRANSLOG_INTERNAL_PARTS + 1, parts, NULL)) + TRANSLOG_INTERNAL_PARTS + 1, parts, NULL, NULL)) { fprintf(stderr, "Can't write record #%lu\n", (ulong) i); translog_destroy(); @@ -314,7 +312,7 @@ int main(int argc __attribute__((unused)), char *argv[]) if (translog_write_record(&lsn, LOGREC_VARIABLE_RECORD_0LSN_EXAMPLE, trn, NULL, rec_len, - TRANSLOG_INTERNAL_PARTS + 1, parts, NULL)) + TRANSLOG_INTERNAL_PARTS + 1, parts, NULL, NULL)) { fprintf(stderr, "Can't write variable record #%lu\n", (ulong) i); translog_destroy(); diff --git a/storage/maria/unittest/ma_test_loghandler_multithread-t.c b/storage/maria/unittest/ma_test_loghandler_multithread-t.c index a3af67c8ac2..0a81aeac55f 100644 --- a/storage/maria/unittest/ma_test_loghandler_multithread-t.c +++ b/storage/maria/unittest/ma_test_loghandler_multithread-t.c @@ -138,7 +138,7 @@ void writer(int num) if (translog_write_record(&lsn, LOGREC_FIXED_RECORD_0LSN_EXAMPLE, &trn, NULL, 6, TRANSLOG_INTERNAL_PARTS + 1, - parts, NULL)) + parts, NULL, NULL)) { fprintf(stderr, "Can't write LOGREC_FIXED_RECORD_0LSN_EXAMPLE record #%lu " "thread %i\n", (ulong) i, num); @@ -155,7 +155,7 @@ void writer(int num) LOGREC_VARIABLE_RECORD_0LSN_EXAMPLE, &trn, NULL, len, TRANSLOG_INTERNAL_PARTS + 1, - parts, NULL)) + parts, NULL, NULL)) { fprintf(stderr, "Can't write variable record #%lu\n", (ulong) i); translog_destroy(); @@ -307,7 +307,7 @@ int main(int argc __attribute__((unused)), LOGREC_FIXED_RECORD_0LSN_EXAMPLE, &dummy_transaction_object, NULL, 6, TRANSLOG_INTERNAL_PARTS + 1, - parts, NULL)) + parts, NULL, NULL)) { fprintf(stderr, "Can't write the first record\n"); translog_destroy(); diff --git a/storage/maria/unittest/ma_test_loghandler_noflush-t.c b/storage/maria/unittest/ma_test_loghandler_noflush-t.c index 2c3afb9a76b..a53e6257314 100644 --- a/storage/maria/unittest/ma_test_loghandler_noflush-t.c +++ b/storage/maria/unittest/ma_test_loghandler_noflush-t.c @@ -85,7 +85,7 @@ int main(int argc __attribute__((unused)), char *argv[]) LOGREC_FIXED_RECORD_0LSN_EXAMPLE, &dummy_transaction_object, NULL, 6, TRANSLOG_INTERNAL_PARTS + 1, - parts, NULL)) + parts, NULL, NULL)) { fprintf(stderr, "Can't write record #%lu\n", (ulong) 0); translog_destroy(); diff --git a/storage/maria/unittest/ma_test_loghandler_pagecache-t.c b/storage/maria/unittest/ma_test_loghandler_pagecache-t.c index 276640dfd17..fa6fcd544a3 100644 --- a/storage/maria/unittest/ma_test_loghandler_pagecache-t.c +++ b/storage/maria/unittest/ma_test_loghandler_pagecache-t.c @@ -97,7 +97,7 @@ int main(int argc __attribute__((unused)), char *argv[]) LOGREC_FIXED_RECORD_0LSN_EXAMPLE, &dummy_transaction_object, NULL, 6, TRANSLOG_INTERNAL_PARTS + 1, - parts, NULL)) + parts, NULL, NULL)) { fprintf(stderr, "Can't write record #%lu\n", (ulong) 0); translog_destroy(); diff --git a/storage/maria/unittest/ma_test_loghandler_purge-t.c b/storage/maria/unittest/ma_test_loghandler_purge-t.c index c638aa85ac6..d9001b9d3ad 100644 --- a/storage/maria/unittest/ma_test_loghandler_purge-t.c +++ b/storage/maria/unittest/ma_test_loghandler_purge-t.c @@ -80,7 +80,7 @@ int main(int argc __attribute__((unused)), char *argv[]) LOGREC_FIXED_RECORD_0LSN_EXAMPLE, &dummy_transaction_object, NULL, 6, TRANSLOG_INTERNAL_PARTS + 1, - parts, NULL)) + parts, NULL, NULL)) { fprintf(stderr, "Can't write record #%lu\n", (ulong) 0); translog_destroy(); @@ -102,7 +102,7 @@ int main(int argc __attribute__((unused)), char *argv[]) LOGREC_FIXED_RECORD_0LSN_EXAMPLE, &dummy_transaction_object, NULL, 6, TRANSLOG_INTERNAL_PARTS + 1, - parts, NULL)) + parts, NULL, NULL)) { fprintf(stderr, "Can't write record #%lu\n", (ulong) 0); translog_destroy(); @@ -125,7 +125,7 @@ int main(int argc __attribute__((unused)), char *argv[]) if (translog_write_record(&lsn, LOGREC_VARIABLE_RECORD_0LSN_EXAMPLE, &dummy_transaction_object, NULL, LONG_BUFFER_SIZE, - TRANSLOG_INTERNAL_PARTS + 1, parts, NULL)) + TRANSLOG_INTERNAL_PARTS + 1, parts, NULL, NULL)) { fprintf(stderr, "Can't write variable record\n"); translog_destroy(); @@ -150,7 +150,7 @@ int main(int argc __attribute__((unused)), char *argv[]) LOGREC_FIXED_RECORD_0LSN_EXAMPLE, &dummy_transaction_object, NULL, 6, TRANSLOG_INTERNAL_PARTS + 1, - parts, NULL)) + parts, NULL, NULL)) { fprintf(stderr, "Can't write last record\n"); translog_destroy(); diff --git a/storage/myisam/sort.c b/storage/myisam/sort.c index 3ab478682c6..3d7e460eea1 100644 --- a/storage/myisam/sort.c +++ b/storage/myisam/sort.c @@ -78,10 +78,10 @@ static int NEAR_F write_keys_varlen(MI_SORT_PARAM *info,uchar **sort_keys, static uint NEAR_F read_to_buffer_varlen(IO_CACHE *fromfile,BUFFPEK *buffpek, uint sort_length); static int NEAR_F write_merge_key(MI_SORT_PARAM *info, IO_CACHE *to_file, - char *key, uint sort_length, uint count); + uchar *key, uint sort_length, uint count); static int NEAR_F write_merge_key_varlen(MI_SORT_PARAM *info, IO_CACHE *to_file, - char* key, uint sort_length, + uchar* key, uint sort_length, uint count); static inline int my_var_write(MI_SORT_PARAM *info, IO_CACHE *to_file, uchar *bufs); @@ -858,16 +858,15 @@ static uint NEAR_F read_to_buffer_varlen(IO_CACHE *fromfile, BUFFPEK *buffpek, static int NEAR_F write_merge_key_varlen(MI_SORT_PARAM *info, - IO_CACHE *to_file,char* key, + IO_CACHE *to_file, uchar* key, uint sort_length, uint count) { uint idx; - - char *bufs = key; + uchar *bufs = key; for (idx=1;idx<=count;idx++) { int err; - if ((err= my_var_write(info,to_file, (uchar*) bufs))) + if ((err= my_var_write(info, to_file, bufs))) return (err); bufs=bufs+sort_length; } @@ -876,10 +875,10 @@ static int NEAR_F write_merge_key_varlen(MI_SORT_PARAM *info, static int NEAR_F write_merge_key(MI_SORT_PARAM *info __attribute__((unused)), - IO_CACHE *to_file, char* key, + IO_CACHE *to_file, uchar* key, uint sort_length, uint count) { - return my_b_write(to_file,(uchar*) key,(uint) sort_length*count); + return my_b_write(to_file, key, (uint) sort_length * count); } /* -- cgit v1.2.1 From e81426170095017c9fe9727ca6274ef2e3e4819d Mon Sep 17 00:00:00 2001 From: unknown Date: Tue, 2 Oct 2007 19:16:20 +0200 Subject: Fix for pushbuild failure (noticable only on 64-bit) storage/maria/ma_pagecache.c: pagecache->blocks is now long, takes 8 bytes on some platforms. The cast to ulonglong in int8store is for those platforms where ulong is 32-bit and int8store uses some << shifts, if there are (x<<40 is undefined if x is 32-bit). storage/maria/ma_recovery.c: this change corresponds to the one done in ma_pagecache.c: number of dirty pages is stored in 8 bytes. --- storage/maria/ma_pagecache.c | 10 +++++----- storage/maria/ma_recovery.c | 8 ++++---- 2 files changed, 9 insertions(+), 9 deletions(-) diff --git a/storage/maria/ma_pagecache.c b/storage/maria/ma_pagecache.c index 4a3fc555a8a..aa21a8bfacb 100755 --- a/storage/maria/ma_pagecache.c +++ b/storage/maria/ma_pagecache.c @@ -3898,7 +3898,7 @@ my_bool pagecache_collect_changed_blocks_with_lsn(PAGECACHE *pagecache, LSN *min_rec_lsn) { my_bool error= 0; - uint stored_list_size= 0; + ulong stored_list_size= 0; uint file_hash; char *ptr; LSN minimum_rec_lsn= LSN_MAX; @@ -3941,8 +3941,8 @@ my_bool pagecache_collect_changed_blocks_with_lsn(PAGECACHE *pagecache, } } - compile_time_assert(sizeof(pagecache->blocks) <= 4); - str->length= 4 + /* number of dirty pages */ + compile_time_assert(sizeof(pagecache->blocks) <= 8); + str->length= 8 + /* number of dirty pages */ (4 + /* file */ 4 + /* pageno */ LSN_STORE_SIZE /* rec_lsn */ @@ -3950,8 +3950,8 @@ my_bool pagecache_collect_changed_blocks_with_lsn(PAGECACHE *pagecache, if (NULL == (str->str= my_malloc(str->length, MYF(MY_WME)))) goto err; ptr= str->str; - int4store(ptr, stored_list_size); - ptr+= 4; + int8store(ptr, (ulonglong)stored_list_size); + ptr+= 8; if (!stored_list_size) goto end; for (file_hash= 0; file_hash < PAGECACHE_CHANGED_BLOCKS_HASH; file_hash++) diff --git a/storage/maria/ma_recovery.c b/storage/maria/ma_recovery.c index 1ae2244159c..c994860ddfd 100644 --- a/storage/maria/ma_recovery.c +++ b/storage/maria/ma_recovery.c @@ -2003,7 +2003,7 @@ static MARIA_HA *get_MARIA_HA_from_UNDO_record(const static LSN parse_checkpoint_record(LSN lsn) { - uint i; + ulong i; TRANSLOG_HEADER_BUFFER rec; tprint(tracef, "Loading data from checkpoint record at LSN (%lu,0x%lx)\n", @@ -2087,9 +2087,9 @@ static LSN parse_checkpoint_record(LSN lsn) } /* dirty pages */ - uint nb_dirty_pages= uint4korr(ptr); - ptr+= 4; - tprint(tracef, "%u dirty pages\n", nb_dirty_pages); + ulong nb_dirty_pages= uint8korr(ptr); + ptr+= 8; + tprint(tracef, "%lu dirty pages\n", nb_dirty_pages); if (hash_init(&all_dirty_pages, &my_charset_bin, nb_dirty_pages, offsetof(struct st_dirty_page, file_and_page_id), sizeof(((struct st_dirty_page *)NULL)->file_and_page_id), -- cgit v1.2.1 From 24db7ed7e13c603488c0a5e05ad9c87754b2ffaf Mon Sep 17 00:00:00 2001 From: unknown Date: Wed, 3 Oct 2007 18:10:32 +0200 Subject: WL#3072 Maria recovery. * Thanks to Serg's tip, we fix here the compilation issue of REDO_REPAIR_TABLE's execution, by defining versions of _ma_killed_ptr() and _ma_check_print_info|warning|error() in maria_read_log.c (we move those of maria_chk.c into an include file and include it in maria_chk.c and maria_read_log.c). Execution of such record looks like working from my tests (it only happens in maria_read_log; recovery-from-mysqld skips DDLs and REPAIR is considered DDL here as it bypasses logging): tested ALTER TABLE ENABLE KEYS and then remove table, apply log: that did a repair. * Recent changes broke maria_read_log a bit: -a became default and -o caused error; fixing this. storage/maria/Makefile.am: addind new file storage/maria/ma_recovery.c: enable execution of REDO_REPAIR_TABLE by maria_read_log now that it compiles. Now reason to keep only T_QUICK from testflag. storage/maria/maria_chk.c: moving these functions to ma_check_standalone.h for reusability storage/maria/maria_def.h: comment storage/maria/maria_read_log.c: ma_check_standalone.h needs my_progname_short. Fixing bug where "maria_read_log" would default to -a and "maria_read_log -o" would throw an error. Implemented behaviour is: - no options: usage() - -a : applys, applys UNDOs by default unless --disable-undo - -o : only prints storage/maria/ma_check_standalone.h: All standalone programs which need to use functions from ma_check.c (like maria_repair()) must define their version of _ma_killed_ptr() and _ma_check_print_info|warning|error(). Indeed, linking with ma_check.o brings in the dependencies of ma_check.o which are definitions of the above functions; if the program does not define them then the ones of ha_maria.o are used i.e. ha_maria.o is linked into the program, and this brings dependencies of ha_maria.o on mysqld.o into the program's linking which thus fails, as the program is not linked with mysqld.o. We put in this file the functions which maria_chk.c uses, so that they can be reused by maria_read_log (when it replays REDO_REPAIR_TABLE) as they are good enough (they just print to stdout/stderr like maria_read_log already does). --- storage/maria/Makefile.am | 2 +- storage/maria/ma_check_standalone.h | 106 ++++++++++++++++++++++++++++++++++++ storage/maria/ma_recovery.c | 12 +--- storage/maria/maria_chk.c | 82 +--------------------------- storage/maria/maria_def.h | 5 +- storage/maria/maria_read_log.c | 19 ++++--- 6 files changed, 124 insertions(+), 102 deletions(-) create mode 100644 storage/maria/ma_check_standalone.h diff --git a/storage/maria/Makefile.am b/storage/maria/Makefile.am index 2bd9b7db922..795784f31aa 100644 --- a/storage/maria/Makefile.am +++ b/storage/maria/Makefile.am @@ -62,7 +62,7 @@ noinst_HEADERS = maria_def.h ma_rt_index.h ma_rt_key.h ma_rt_mbr.h \ ma_control_file.h ha_maria.h ma_blockrec.h \ ma_loghandler.h ma_loghandler_lsn.h ma_pagecache.h \ ma_checkpoint.h ma_recovery.h ma_commit.h \ - trnman_public.h + trnman_public.h ma_check_standalone.h ma_test1_DEPENDENCIES= $(LIBRARIES) ma_test1_LDADD= @CLIENT_EXTRA_LDFLAGS@ libmaria.a \ $(top_builddir)/storage/myisam/libmyisam.a \ diff --git a/storage/maria/ma_check_standalone.h b/storage/maria/ma_check_standalone.h new file mode 100644 index 00000000000..3874d722d6c --- /dev/null +++ b/storage/maria/ma_check_standalone.h @@ -0,0 +1,106 @@ +/* Copyright (C) 2007 MySQL AB + + 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ + +/* + All standalone programs which need to use functions from ma_check.c + (like maria_repair()) must define their version of _ma_killed_ptr() + and _ma_check_print_info|warning|error(). Indeed, linking with ma_check.o + brings in the dependencies of ma_check.o which are definitions of the above + functions; if the program does not define them then the ones of + ha_maria.o are used i.e. ha_maria.o is linked into the program, and this + brings dependencies of ha_maria.o on mysqld.o into the program's linking + which thus fails, as the program is not linked with mysqld.o. + This file contains the versions of these functions used by maria_chk and + maria_read_log. +*/ + +/* + Check if check/repair operation was killed by a signal +*/ + +static int not_killed= 0; + +volatile int *_ma_killed_ptr(HA_CHECK *param __attribute__((unused))) +{ + return ¬_killed; /* always NULL */ +} + + /* print warnings and errors */ + /* VARARGS */ + +void _ma_check_print_info(HA_CHECK *param __attribute__((unused)), + const char *fmt,...) +{ + va_list args; + DBUG_ENTER("_ma_check_print_info"); + DBUG_PRINT("enter", ("format: %s", fmt)); + + va_start(args,fmt); + VOID(vfprintf(stdout, fmt, args)); + VOID(fputc('\n',stdout)); + va_end(args); + DBUG_VOID_RETURN; +} + +/* VARARGS */ + +void _ma_check_print_warning(HA_CHECK *param, const char *fmt,...) +{ + va_list args; + DBUG_ENTER("_ma_check_print_warning"); + DBUG_PRINT("enter", ("format: %s", fmt)); + + fflush(stdout); + if (!param->warning_printed && !param->error_printed) + { + if (param->testflag & T_SILENT) + fprintf(stderr,"%s: MARIA file %s\n",my_progname_short, + param->isam_file_name); + param->out_flag|= O_DATA_LOST; + } + param->warning_printed=1; + va_start(args,fmt); + fprintf(stderr,"%s: warning: ",my_progname_short); + VOID(vfprintf(stderr, fmt, args)); + VOID(fputc('\n',stderr)); + fflush(stderr); + va_end(args); + DBUG_VOID_RETURN; +} + +/* VARARGS */ + +void _ma_check_print_error(HA_CHECK *param, const char *fmt,...) +{ + va_list args; + DBUG_ENTER("_ma_check_print_error"); + DBUG_PRINT("enter", ("format: %s", fmt)); + + fflush(stdout); + if (!param->warning_printed && !param->error_printed) + { + if (param->testflag & T_SILENT) + fprintf(stderr,"%s: MARIA file %s\n",my_progname_short,param->isam_file_name); + param->out_flag|= O_DATA_LOST; + } + param->error_printed|=1; + va_start(args,fmt); + fprintf(stderr,"%s: error: ",my_progname_short); + VOID(vfprintf(stderr, fmt, args)); + VOID(fputc('\n',stderr)); + fflush(stderr); + va_end(args); + DBUG_VOID_RETURN; +} diff --git a/storage/maria/ma_recovery.c b/storage/maria/ma_recovery.c index c994860ddfd..631a10b8025 100644 --- a/storage/maria/ma_recovery.c +++ b/storage/maria/ma_recovery.c @@ -783,27 +783,17 @@ prototype_redo_exec_hook(REDO_REPAIR_TABLE) than the mapping, so we can repair. */ tprint(tracef, " repairing...\n"); - /** - @todo RECOVERY BUG fix this: - the maria_chk_init() call causes a heap of linker errors in ha_maria.cc! - */ -#if 0 HA_CHECK param; maria_chk_init(¶m); param.isam_file_name= info->s->open_file_name; param.testflag= uint4korr(rec->header); - if (maria_repair(¶m, info, info->s->open_file_name, - param.testflag & T_QUICK)) + if (maria_repair(¶m, info, info->s->open_file_name, param.testflag)) goto end; if (_ma_update_create_rename_lsn(info->s, rec->lsn, TRUE)) goto end; error= 0; end: return error; -#else - DBUG_ASSERT("fix this table repairing" == NULL); - return error; -#endif } diff --git a/storage/maria/maria_chk.c b/storage/maria/maria_chk.c index e6ac0dcfc50..639334c3033 100644 --- a/storage/maria/maria_chk.c +++ b/storage/maria/maria_chk.c @@ -1758,84 +1758,4 @@ err: } /* sort_record_index */ - -/* - Check if maria_chk was killed by a signal - This is overloaded by other programs that want to be able to abort - sorting -*/ - -static int not_killed= 0; - -volatile int *_ma_killed_ptr(HA_CHECK *param __attribute__((unused))) -{ - return ¬_killed; /* always NULL */ -} - - /* print warnings and errors */ - /* VARARGS */ - -void _ma_check_print_info(HA_CHECK *param __attribute__((unused)), - const char *fmt,...) -{ - va_list args; - DBUG_ENTER("_ma_check_print_info"); - DBUG_PRINT("enter", ("format: %s", fmt)); - - va_start(args,fmt); - VOID(vfprintf(stdout, fmt, args)); - VOID(fputc('\n',stdout)); - va_end(args); - DBUG_VOID_RETURN; -} - -/* VARARGS */ - -void _ma_check_print_warning(HA_CHECK *param, const char *fmt,...) -{ - va_list args; - DBUG_ENTER("_ma_check_print_warning"); - DBUG_PRINT("enter", ("format: %s", fmt)); - - fflush(stdout); - if (!param->warning_printed && !param->error_printed) - { - if (param->testflag & T_SILENT) - fprintf(stderr,"%s: MARIA file %s\n",my_progname_short, - param->isam_file_name); - param->out_flag|= O_DATA_LOST; - } - param->warning_printed=1; - va_start(args,fmt); - fprintf(stderr,"%s: warning: ",my_progname_short); - VOID(vfprintf(stderr, fmt, args)); - VOID(fputc('\n',stderr)); - fflush(stderr); - va_end(args); - DBUG_VOID_RETURN; -} - -/* VARARGS */ - -void _ma_check_print_error(HA_CHECK *param, const char *fmt,...) -{ - va_list args; - DBUG_ENTER("_ma_check_print_error"); - DBUG_PRINT("enter", ("format: %s", fmt)); - - fflush(stdout); - if (!param->warning_printed && !param->error_printed) - { - if (param->testflag & T_SILENT) - fprintf(stderr,"%s: MARIA file %s\n",my_progname_short,param->isam_file_name); - param->out_flag|= O_DATA_LOST; - } - param->error_printed|=1; - va_start(args,fmt); - fprintf(stderr,"%s: error: ",my_progname_short); - VOID(vfprintf(stderr, fmt, args)); - VOID(fputc('\n',stderr)); - fflush(stderr); - va_end(args); - DBUG_VOID_RETURN; -} +#include "ma_check_standalone.h" diff --git a/storage/maria/maria_def.h b/storage/maria/maria_def.h index bac83db4f51..c1f21a4ef67 100644 --- a/storage/maria/maria_def.h +++ b/storage/maria/maria_def.h @@ -923,7 +923,10 @@ C_MODE_START int _ma_flush_table_files(MARIA_HA *info, uint flush_data_or_index, enum flush_type flush_type_for_data, enum flush_type flush_type_for_index); -/* Functions needed by _ma_check (are overrided in MySQL) */ +/* + Functions needed by _ma_check (are overridden in MySQL/ha_maria.cc). + See ma_check_standalone.h . +*/ volatile int *_ma_killed_ptr(HA_CHECK *param); void _ma_check_print_error _VARARGS((HA_CHECK *param, const char *fmt, ...)); void _ma_check_print_warning _VARARGS((HA_CHECK *param, const char *fmt, ...)); diff --git a/storage/maria/maria_read_log.c b/storage/maria/maria_read_log.c index ec1f7697a23..59882f81d9a 100644 --- a/storage/maria/maria_read_log.c +++ b/storage/maria/maria_read_log.c @@ -31,12 +31,14 @@ const char *default_dbug_option= "d:t:i:o,/tmp/maria_read_log.trace"; #endif /* DBUG_OFF */ static my_bool opt_only_display, opt_apply, opt_apply_undo, opt_silent; static ulong opt_page_buffer_size; +static const char *my_progname_short; int main(int argc, char **argv) { LSN lsn; char **default_argv; MY_INIT(argv[0]); + my_progname_short= my_progname+dirname_length(my_progname); load_defaults("my", load_default_groups, &argc, &argv); default_argv= argv; @@ -103,12 +105,12 @@ int main(int argc, char **argv) if (maria_apply_log(lsn, opt_apply, opt_silent ? NULL : stdout, opt_apply_undo, FALSE)) goto err; - fprintf(stdout, "%s: SUCCESS\n", my_progname); + fprintf(stdout, "%s: SUCCESS\n", my_progname_short); goto end; err: /* don't touch anything more, in case we hit a bug */ - fprintf(stderr, "%s: FAILED\n", my_progname); + fprintf(stderr, "%s: FAILED\n", my_progname_short); exit(1); end: maria_end(); @@ -119,6 +121,9 @@ end: } +#include "ma_check_standalone.h" + + static struct my_option my_long_options[] = { {"apply", 'a', @@ -155,7 +160,7 @@ static struct my_option my_long_options[] = static void print_version(void) { VOID(printf("%s Ver 1.1 for %s on %s\n", - my_progname, SYSTEM_TYPE, MACHINE_TYPE)); + my_progname_short, SYSTEM_TYPE, MACHINE_TYPE)); NETWARE_SET_SCREEN_MODE(1); } @@ -169,7 +174,7 @@ static void usage(void) puts("Display and apply log records from a MARIA transaction log"); puts("found in the current directory (for now)"); - VOID(printf("\nUsage: %s OPTIONS\n", my_progname)); + VOID(printf("\nUsage: %s OPTIONS\n", my_progname_short)); puts("You need to use one of -o or -a"); my_print_help(my_long_options); print_defaults("my", load_default_groups); @@ -203,13 +208,11 @@ static void get_options(int *argc,char ***argv) { int ho_error; - my_progname= argv[0][0]; - if ((ho_error=handle_options(argc, argv, my_long_options, get_one_option))) exit(ho_error); - if (opt_apply_undo) - opt_apply= 1; + if (!opt_apply) + opt_apply_undo= FALSE; if ((opt_only_display + opt_apply) != 1) { -- cgit v1.2.1 From 4eb383e06c021e89244cbf7d24abecb7b3bde01b Mon Sep 17 00:00:00 2001 From: unknown Date: Thu, 4 Oct 2007 20:33:42 +0300 Subject: After merge fixes. Applied changes made in myisam manually to maria. storage/maria/ma_test1.c: After merge fixes. Applied changes made in myisam manually to maria. Fixed memory leak. storage/maria/ma_test2.c: After merge fixes. Applied changes made in myisam manually to maria. Fixed memory leak. --- include/maria.h | 8 ++-- mysql-test/r/maria.result | 8 ++-- storage/maria/CMakeLists.txt | 65 ++++++++++++++++++++++++++- storage/maria/ha_maria.cc | 101 +++++++++++++++++++++++++++++++++++++----- storage/maria/ha_maria.h | 21 ++++++--- storage/maria/ma_check.c | 13 +++--- storage/maria/ma_create.c | 5 ++- storage/maria/ma_delete.c | 12 ++--- storage/maria/ma_dynrec.c | 36 +++++++-------- storage/maria/ma_extra.c | 45 +++---------------- storage/maria/ma_init.c | 1 + storage/maria/ma_key.c | 5 ++- storage/maria/ma_keycache.c | 3 +- storage/maria/ma_locking.c | 4 +- storage/maria/ma_open.c | 21 ++++----- storage/maria/ma_packrec.c | 8 ++-- storage/maria/ma_page.c | 5 ++- storage/maria/ma_pagecache.c | 3 +- storage/maria/ma_rkey.c | 83 +++++++++++++++++++++------------- storage/maria/ma_search.c | 2 +- storage/maria/ma_sort.c | 103 +++++++++++++++++++++---------------------- storage/maria/ma_sp_key.c | 73 +++++++++++++----------------- storage/maria/ma_sp_test.c | 34 +++++++------- storage/maria/ma_statrec.c | 34 +++++++------- storage/maria/ma_test1.c | 74 ++++++++++++++++--------------- storage/maria/ma_test2.c | 46 ++++++++++--------- storage/maria/ma_test3.c | 20 ++++----- storage/maria/ma_update.c | 6 +-- storage/maria/ma_write.c | 8 ++-- storage/maria/maria_chk.c | 4 +- storage/maria/maria_def.h | 28 ++++++------ storage/maria/maria_pack.c | 6 +-- 32 files changed, 510 insertions(+), 375 deletions(-) diff --git a/include/maria.h b/include/maria.h index 1d7778f7c3d..dfd0596ba7e 100644 --- a/include/maria.h +++ b/include/maria.h @@ -375,11 +375,11 @@ typedef struct st_maria_sort_param int (*key_read)(struct st_maria_sort_param *, uchar *); int (*key_write)(struct st_maria_sort_param *, const uchar *); void (*lock_in_memory)(HA_CHECK *); - NEAR int (*write_keys)(struct st_maria_sort_param *, register uchar **, + int (*write_keys)(struct st_maria_sort_param *, register uchar **, uint , struct st_buffpek *, IO_CACHE *); - NEAR uint (*read_to_buffer)(IO_CACHE *,struct st_buffpek *, uint); - NEAR int (*write_key)(struct st_maria_sort_param *, IO_CACHE *,char *, - uint, uint); + uint (*read_to_buffer)(IO_CACHE *,struct st_buffpek *, uint); + int (*write_key)(struct st_maria_sort_param *, IO_CACHE *,uchar *, + uint, uint); } MARIA_SORT_PARAM; diff --git a/mysql-test/r/maria.result b/mysql-test/r/maria.result index 8c1cb5f3a2f..d49d65e6dc2 100644 --- a/mysql-test/r/maria.result +++ b/mysql-test/r/maria.result @@ -860,7 +860,7 @@ _id DELETE FROM t1 WHERE _id < 8; SHOW TABLE STATUS LIKE 't1'; Name Engine Version Row_format Rows Avg_row_length Data_length Max_data_length Index_length Data_free Auto_increment Create_time Update_time Check_time Collation Checksum Create_options Comment -t1 MARIA 10 Page 2 # # # # 0 # # # # # # +t1 MARIA 10 Paged 2 # # # # 0 # # # # # # CHECK TABLE t1 EXTENDED; Table Op Msg_type Msg_text test.t1 check status OK @@ -872,7 +872,7 @@ Table Op Msg_type Msg_text test.t1 check status OK SHOW TABLE STATUS LIKE 't1'; Name Engine Version Row_format Rows Avg_row_length Data_length Max_data_length Index_length Data_free Auto_increment Create_time Update_time Check_time Collation Checksum Create_options Comment -t1 MARIA 10 Page 2 # # # # 0 # # # # # # +t1 MARIA 10 Paged 2 # # # # 0 # # # # # # SELECT _id FROM t1; _id 8 @@ -919,7 +919,7 @@ _id DELETE FROM t1 WHERE _id < 8; SHOW TABLE STATUS LIKE 't1'; Name Engine Version Row_format Rows Avg_row_length Data_length Max_data_length Index_length Data_free Auto_increment Create_time Update_time Check_time Collation Checksum Create_options Comment -t1 MARIA 10 Page 2 # # # # 0 # # # # # # +t1 MARIA 10 Paged 2 # # # # 0 # # # # # # CHECK TABLE t1 EXTENDED; Table Op Msg_type Msg_text test.t1 check status OK @@ -931,7 +931,7 @@ Table Op Msg_type Msg_text test.t1 check status OK SHOW TABLE STATUS LIKE 't1'; Name Engine Version Row_format Rows Avg_row_length Data_length Max_data_length Index_length Data_free Auto_increment Create_time Update_time Check_time Collation Checksum Create_options Comment -t1 MARIA 10 Page 2 # # # # 0 # # # # # # +t1 MARIA 10 Paged 2 # # # # 0 # # # # # # SELECT _id FROM t1; _id 8 diff --git a/storage/maria/CMakeLists.txt b/storage/maria/CMakeLists.txt index cfe23054e2f..17d3aa53839 100644 --- a/storage/maria/CMakeLists.txt +++ b/storage/maria/CMakeLists.txt @@ -1 +1,64 @@ -# empty for the moment; will fill it when we build under Windows +# Copyright (C) 2007 MySQL AB +# +# 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 St, Fifth Floor, Boston, MA 02110-1301 USA +INCLUDE("${PROJECT_SOURCE_DIR}/win/mysql_manifest.cmake") + +SET(CMAKE_CXX_FLAGS_DEBUG "${CMAKE_CXX_FLAGS_DEBUG} -DSAFEMALLOC -DSAFE_MUTEX") +SET(CMAKE_C_FLAGS_DEBUG "${CMAKE_C_FLAGS_DEBUG} -DSAFEMALLOC -DSAFE_MUTEX") + +INCLUDE_DIRECTORIES(${CMAKE_SOURCE_DIR}/include ${CMAKE_SOURCE_DIR}/zlib + ${CMAKE_SOURCE_DIR}/sql + ${CMAKE_SOURCE_DIR}/regex + ${CMAKE_SOURCE_DIR}/extra/yassl/include) +SET(MARIA_SOURCES ft_maria.c ha_maria.cc lockman.c ma_bitmap.c ma_blockrec.c + ma_cache.c ma_changed.c ma_check.c ma_checkpoint.c ma_checksum.c + ma_close.c ma_commit.c ma_control_file.c ma_create.c ma_dbug.c + ma_delete.c ma_delete_all.c ma_delete_table.c ma_dynrec.c + ma_extra.c ma_ft_boolean_search.c ma_ft_eval.c ma_ft_nlq_search.c + ma_ft_parser.c ma_ft_stem.c ma_ft_test1.c ma_ft_update.c ma_info.c + ma_init.c ma_key.c ma_keycache.c ma_least_recently_dirtied.c + ma_locking.c ma_loghandler.c ma_open.c ma_packrec.c ma_page.c + ma_pagecache.c ma_pagecaches.c ma_panic.c ma_preload.c ma_range.c + ma_recovery.c ma_rename.c ma_rfirst.c ma_rkey.c ma_rlast.c + ma_rnext.c ma_rnext_same.c ma_rprev.c ma_rrnd.c ma_rsame.c + ma_rsamepos.c ma_rt_index.c ma_rt_key.c ma_rt_mbr.c ma_rt_split.c + ma_rt_test.c ma_scan.c ma_search.c ma_sort.c ma_sp_key.c + ma_sp_test.c ma_static.c ma_statrec.c + ma_unique.c ma_update.c ma_write.c tablockman.c trnman.c + ha_maria.h maria_def.h) + +IF(NOT SOURCE_SUBLIBS) + + ADD_LIBRARY(maria ${MARIA_SOURCES}) + +ADD_EXECUTABLE(maria_ftdump maria_ftdump.c) +TARGET_LINK_LIBRARIES(maria_ftdump maria myisam mysys debug dbug strings zlib wsock32) + +ADD_EXECUTABLE(maria_chk maria_chk.c) +TARGET_LINK_LIBRARIES(maria_chk maria myisam mysys debug dbug strings zlib wsock32) + +ADD_EXECUTABLE(maria_read_log maria_read_log.c) +TARGET_LINK_LIBRARIES(maria_read_log maria myisam mysys debug dbug strings zlib wsock32) + +ADD_EXECUTABLE(maria_pack maria_pack.c) +TARGET_LINK_LIBRARIES(maria_pack maria myisam mysys debug dbug strings zlib wsock32) + +IF(EMBED_MANIFESTS) + MYSQL_EMBED_MANIFEST("maria_ftdump" "asInvoker") + MYSQL_EMBED_MANIFEST("maria_chk" "asInvoker") + MYSQL_EMBED_MANIFEST("maria_read_log" "asInvoker") + MYSQL_EMBED_MANIFEST("maria_pack" "asInvoker") +ENDIF(EMBED_MANIFESTS) + +ENDIF(NOT SOURCE_SUBLIBS) diff --git a/storage/maria/ha_maria.cc b/storage/maria/ha_maria.cc index 362e89bb088..94ea9656e8b 100644 --- a/storage/maria/ha_maria.cc +++ b/storage/maria/ha_maria.cc @@ -289,7 +289,8 @@ int table2maria(TABLE *table_arg, MARIA_KEYDEF **keydef_out, DBUG_PRINT("loop", ("found: 0x%lx recpos: %d minpos: %d length: %d", (long) found, recpos, minpos, length)); if (recpos != minpos) - { // Reserved space (Null bits?) + { + /* reserve space for null bits */ bzero((char*) recinfo_pos, sizeof(*recinfo_pos)); recinfo_pos->type= FIELD_NORMAL; recinfo_pos++->length= (uint16) (minpos - recpos); @@ -1593,7 +1594,7 @@ void ha_maria::start_bulk_insert(ha_rows rows) DBUG_ENTER("ha_maria::start_bulk_insert"); THD *thd= current_thd; ulong size= min(thd->variables.read_buff_size, - table->s->avg_row_length * rows); + (ulong) (table->s->avg_row_length * rows)); DBUG_PRINT("info", ("start_bulk_insert: rows %lu size %lu", (ulong) rows, size)); @@ -1714,9 +1715,9 @@ int ha_maria::delete_row(const uchar * buf) } -int ha_maria::index_read(uchar * buf, const uchar * key, - key_part_map keypart_map, - enum ha_rkey_function find_flag) +int ha_maria::index_read_map(uchar * buf, const uchar * key, + key_part_map keypart_map, + enum ha_rkey_function find_flag) { DBUG_ASSERT(inited == INDEX); ha_statistic_increment(&SSV::ha_read_key_count); @@ -1726,9 +1727,9 @@ int ha_maria::index_read(uchar * buf, const uchar * key, } -int ha_maria::index_read_idx(uchar * buf, uint index, const uchar * key, - key_part_map keypart_map, - enum ha_rkey_function find_flag) +int ha_maria::index_read_idx_map(uchar * buf, uint index, const uchar * key, + key_part_map keypart_map, + enum ha_rkey_function find_flag) { ha_statistic_increment(&SSV::ha_read_key_count); int error= maria_rkey(file, buf, index, key, keypart_map, find_flag); @@ -1737,10 +1738,10 @@ int ha_maria::index_read_idx(uchar * buf, uint index, const uchar * key, } -int ha_maria::index_read_last(uchar * buf, const uchar * key, - key_part_map keypart_map) +int ha_maria::index_read_last_map(uchar * buf, const uchar * key, + key_part_map keypart_map) { - DBUG_ENTER("ha_maria::index_read_last"); + DBUG_ENTER("ha_maria::index_read_last_map"); DBUG_ASSERT(inited == INDEX); ha_statistic_increment(&SSV::ha_read_key_count); int error= maria_rkey(file, buf, active_index, key, keypart_map, @@ -2157,6 +2158,8 @@ int ha_maria::create(const char *name, register TABLE *table_arg, if (ha_create_info->options & HA_LEX_CREATE_TMP_TABLE) create_flags|= HA_CREATE_TMP_TABLE; + if (ha_create_info->options & HA_CREATE_KEEP_FILES) + create_flags|= HA_CREATE_KEEP_FILES; if (options & HA_OPTION_PACK_RECORD) create_flags|= HA_PACK_RECORD; if (options & HA_OPTION_CHECKSUM) @@ -2380,6 +2383,82 @@ static int ha_maria_init(void *p) } +#ifdef HAVE_QUERY_CACHE +/** + @brief Register a named table with a call back function to the query cache. + + @param thd The thread handle + @param table_key A pointer to the table name in the table cache + @param key_length The length of the table name + @param[out] engine_callback The pointer to the storage engine call back + function, currently 0 + @param[out] engine_data Engine data will be set to 0. + + @note Despite the name of this function, it is used to check each statement + before it is cached and not to register a table or callback function. + + @see handler::register_query_cache_table + + @return The error code. The engine_data and engine_callback will be set to 0. + @retval TRUE Success + @retval FALSE An error occured +*/ + +my_bool ha_maria::register_query_cache_table(THD *thd, char *table_name, + uint table_name_len, + qc_engine_callback + *engine_callback, + ulonglong *engine_data) +{ + /* + No call back function is needed to determine if a cached statement + is valid or not. + */ + *engine_callback= 0; + + /* + No engine data is needed. + */ + *engine_data= 0; + + /* + If a concurrent INSERT has happened just before the currently processed + SELECT statement, the total size of the table is unknown. + + To determine if the table size is known, the current thread's snap shot of + the table size with the actual table size are compared. + + If the table size is unknown the SELECT statement can't be cached. + */ + ulonglong actual_data_file_length; + ulonglong current_data_file_length; + + /* + POSIX visibility rules specify that "2. Whatever memory values a + thread can see when it unlocks a mutex <...> can also be seen by any + thread that later locks the same mutex". In this particular case, + concurrent insert thread had modified the data_file_length in + MYISAM_SHARE before it has unlocked (or even locked) + structure_guard_mutex. So, here we're guaranteed to see at least that + value after we've locked the same mutex. We can see a later value + (modified by some other thread) though, but it's ok, as we only want + to know if the variable was changed, the actual new value doesn't matter + */ + actual_data_file_length= file->s->state.state.data_file_length; + current_data_file_length= file->save_state.data_file_length; + + if (current_data_file_length != actual_data_file_length) + { + /* Don't cache current statement. */ + return FALSE; + } + + /* It is ok to try to cache current statement. */ + return TRUE; +} +#endif + + struct st_mysql_storage_engine maria_storage_engine= { MYSQL_HANDLERTON_INTERFACE_VERSION }; diff --git a/storage/maria/ha_maria.h b/storage/maria/ha_maria.h index 2813e2ce037..21b08979bb3 100644 --- a/storage/maria/ha_maria.h +++ b/storage/maria/ha_maria.h @@ -71,13 +71,13 @@ public: int write_row(uchar * buf); int update_row(const uchar * old_data, uchar * new_data); int delete_row(const uchar * buf); - int index_read(uchar * buf, const uchar * key, key_part_map keypart_map, - enum ha_rkey_function find_flag); - int index_read_idx(uchar * buf, uint idx, const uchar * key, - key_part_map keypart_map, - enum ha_rkey_function find_flag); - int index_read_last(uchar * buf, const uchar * key, - key_part_map keypart_map); + int index_read_map(uchar * buf, const uchar * key, key_part_map keypart_map, + enum ha_rkey_function find_flag); + int index_read_idx_map(uchar * buf, uint idx, const uchar * key, + key_part_map keypart_map, + enum ha_rkey_function find_flag); + int index_read_last_map(uchar * buf, const uchar * key, + key_part_map keypart_map); int index_next(uchar * buf); int index_prev(uchar * buf); int index_first(uchar * buf); @@ -143,4 +143,11 @@ public: int dump(THD * thd, int fd); int net_read_dump(NET * net); #endif +#ifdef HAVE_QUERY_CACHE + my_bool register_query_cache_table(THD *thd, char *table_key, + uint key_length, + qc_engine_callback + *engine_callback, + ulonglong *engine_data); +#endif }; diff --git a/storage/maria/ma_check.c b/storage/maria/ma_check.c index fa1c812daf7..465c57a21a4 100644 --- a/storage/maria/ma_check.c +++ b/storage/maria/ma_check.c @@ -187,7 +187,7 @@ int maria_chk_del(HA_CHECK *param, register MARIA_HA *info, uint test_flag) printf(" %9s",llstr(next_link,buff)); if (next_link >= info->state->data_file_length) goto wrong; - if (my_pread(info->dfile.file, (char*) buff, delete_link_length, + if (my_pread(info->dfile.file, (uchar*) buff, delete_link_length, next_link,MYF(MY_NABP))) { if (test_flag & T_VERBOSE) puts(""); @@ -264,7 +264,8 @@ static int check_k_link(HA_CHECK *param, register MARIA_HA *info, { uint block_size= info->s->block_size; ha_rows records; - char llbuff[21], llbuff2[21], *buff; + char llbuff[21], llbuff2[21]; + uchar *buff; DBUG_ENTER("check_k_link"); records= (ha_rows) (info->state->key_file_length / block_size); @@ -2347,7 +2348,7 @@ static int writekeys(MARIA_SORT_PARAM *sort_param) { if (info->s->keyinfo[i].flag & HA_FULLTEXT ) { - if (_ma_ft_add(info,i,(char*) key,buff,filepos)) + if (_ma_ft_add(info,i, key,buff,filepos)) goto err; } #ifdef HAVE_SPATIAL @@ -2378,7 +2379,7 @@ static int writekeys(MARIA_SORT_PARAM *sort_param) { if (info->s->keyinfo[i].flag & HA_FULLTEXT) { - if (_ma_ft_del(info,i,(char*) key,buff,filepos)) + if (_ma_ft_del(info,i,key,buff,filepos)) break; } else @@ -3421,7 +3422,7 @@ int maria_repair_parallel(HA_CHECK *param, register MARIA_HA *info, sort_param[i].filepos=new_header_length; sort_param[i].max_pos=sort_param[i].pos=share->pack.header_length; - sort_param[i].record= (((char *)(sort_param+share->base.keys))+ + sort_param[i].record= (((uchar *)(sort_param+share->base.keys))+ (share->base.pack_reclength * i)); if (_ma_alloc_buffer(&sort_param[i].rec_buff, &sort_param[i].rec_buff_size, share->base.default_rec_buff_size)) @@ -5051,7 +5052,7 @@ int maria_write_data_suffix(MARIA_SORT_INFO *sort_info, my_bool fix_datafile) if (info->s->data_file_type == COMPRESSED_RECORD && fix_datafile) { - char buff[MEMMAP_EXTRA_MARGIN]; + uchar buff[MEMMAP_EXTRA_MARGIN]; bzero(buff,sizeof(buff)); if (my_b_write(&info->rec_cache,buff,sizeof(buff))) { diff --git a/storage/maria/ma_create.c b/storage/maria/ma_create.c index ba1d9a13b42..43e596708b9 100644 --- a/storage/maria/ma_create.c +++ b/storage/maria/ma_create.c @@ -764,7 +764,8 @@ int maria_create(const char *name, enum data_file_type datafile_type, Replace the current file. Don't sync dir now if the data file has the same path. */ - create_flag= MY_DELETE_OLD | (!ci->data_file_name ? 0 : sync_dir); + create_flag= (flags & HA_CREATE_KEEP_FILES) ? 0 : MY_DELETE_OLD; + create_flag|= (!ci->data_file_name ? 0 : sync_dir); } /* @@ -1042,7 +1043,7 @@ int maria_create(const char *name, enum data_file_type datafile_type, fn_format(filename,name,"", MARIA_NAME_DEXT, MY_UNPACK_FILENAME | MY_APPEND_EXT); linkname_ptr= NullS; - create_flag=MY_DELETE_OLD; + create_flag= (flags & HA_CREATE_KEEP_FILES) ? 0 : MY_DELETE_OLD; } if ((dfile= my_create_with_symlink(linkname_ptr, filename, 0, create_mode, diff --git a/storage/maria/ma_delete.c b/storage/maria/ma_delete.c index 56da6fd3ed3..f711102e491 100644 --- a/storage/maria/ma_delete.c +++ b/storage/maria/ma_delete.c @@ -75,13 +75,15 @@ int maria_delete(MARIA_HA *info,const uchar *record) info->s->keyinfo[i].version++; if (info->s->keyinfo[i].flag & HA_FULLTEXT ) { - if (_ma_ft_del(info,i,(char*) old_key,record,info->cur_row.lastpos)) + if (_ma_ft_del(info, i, old_key, record, info->cur_row.lastpos)) goto err; } else { if (info->s->keyinfo[i].ck_delete(info,i,old_key, - _ma_make_key(info,i,old_key,record,info->cur_row.lastpos))) + _ma_make_key(info, i, old_key, + record, + info->cur_row.lastpos))) goto err; } /* The above changed info->lastkey2. Inform maria_rnext_same(). */ @@ -255,7 +257,7 @@ static int d_search(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, if (info->ft1_to_ft2) { /* we're in ft1->ft2 conversion mode. Saving key data */ - insert_dynamic(info->ft1_to_ft2, (char*) (lastkey+off)); + insert_dynamic(info->ft1_to_ft2, (lastkey+off)); } else { @@ -806,7 +808,7 @@ static uint remove_key(MARIA_KEYDEF *keyinfo, uint nod_flag, if (next_length > prev_length) { /* We have to copy data from the current key to the next key */ - bmove_upp((char*) keypos,(char*) (lastkey+next_length), + bmove_upp(keypos, (lastkey+next_length), (next_length-prev_length)); keypos-=(next_length-prev_length)+prev_pack_length; store_key_length(keypos,prev_length); @@ -853,7 +855,7 @@ static uint remove_key(MARIA_KEYDEF *keyinfo, uint nod_flag, if (next_length >= prev_length) { /* Key after is based on deleted key */ uint pack_length,tmp; - bmove_upp((char*) keypos,(char*) (lastkey+next_length), + bmove_upp(keypos, (lastkey+next_length), tmp=(next_length-prev_length)); rest_length+=tmp; pack_length= prev_length ? get_pack_length(rest_length): 0; diff --git a/storage/maria/ma_dynrec.c b/storage/maria/ma_dynrec.c index 6e13fbcecb6..c631594671f 100644 --- a/storage/maria/ma_dynrec.c +++ b/storage/maria/ma_dynrec.c @@ -88,7 +88,7 @@ my_bool _ma_dynmap_file(MARIA_HA *info, my_off_t size) DBUG_RETURN(1); } #if defined(HAVE_MADVISE) - madvise(info->s->file_map, size, MADV_RANDOM); + madvise((char*) info->s->file_map, size, MADV_RANDOM); #endif info->s->mmaped_length= size; DBUG_RETURN(0); @@ -109,7 +109,7 @@ void _ma_remap_file(MARIA_HA *info, my_off_t size) { if (info->s->file_map) { - VOID(my_munmap(info->s->file_map, + VOID(my_munmap((char*) info->s->file_map, (size_t) info->s->mmaped_length + MEMMAP_EXTRA_MARGIN)); _ma_dynmap_file(info, size); } @@ -132,8 +132,8 @@ void _ma_remap_file(MARIA_HA *info, my_off_t size) 0 ok */ -uint _ma_mmap_pread(MARIA_HA *info, uchar *Buffer, - uint Count, my_off_t offset, myf MyFlags) +size_t _ma_mmap_pread(MARIA_HA *info, uchar *Buffer, + size_t Count, my_off_t offset, myf MyFlags) { DBUG_PRINT("info", ("maria_read with mmap %d\n", info->dfile.file)); if (info->s->concurrent_insert) @@ -164,8 +164,8 @@ uint _ma_mmap_pread(MARIA_HA *info, uchar *Buffer, /* wrapper for my_pread in case if mmap isn't used */ -uint _ma_nommap_pread(MARIA_HA *info, uchar *Buffer, - uint Count, my_off_t offset, myf MyFlags) +size_t _ma_nommap_pread(MARIA_HA *info, uchar *Buffer, + size_t Count, my_off_t offset, myf MyFlags) { return my_pread(info->dfile.file, Buffer, Count, offset, MyFlags); } @@ -187,8 +187,8 @@ uint _ma_nommap_pread(MARIA_HA *info, uchar *Buffer, !=0 error. In this case return error from pwrite */ -uint _ma_mmap_pwrite(MARIA_HA *info, uchar *Buffer, - uint Count, my_off_t offset, myf MyFlags) +size_t _ma_mmap_pwrite(MARIA_HA *info, const uchar *Buffer, + size_t Count, my_off_t offset, myf MyFlags) { DBUG_PRINT("info", ("maria_write with mmap %d\n", info->dfile.file)); if (info->s->concurrent_insert) @@ -221,8 +221,8 @@ uint _ma_mmap_pwrite(MARIA_HA *info, uchar *Buffer, /* wrapper for my_pwrite in case if mmap isn't used */ -uint _ma_nommap_pwrite(MARIA_HA *info, uchar *Buffer, - uint Count, my_off_t offset, myf MyFlags) +size_t _ma_nommap_pwrite(MARIA_HA *info, const uchar *Buffer, + size_t Count, my_off_t offset, myf MyFlags) { return my_pwrite(info->dfile.file, Buffer, Count, offset, MyFlags); } @@ -428,7 +428,7 @@ static bool unlink_deleted_block(MARIA_HA *info, MARIA_BLOCK_INFO *block_info) & BLOCK_DELETED)) DBUG_RETURN(1); /* Something is wrong */ mi_sizestore(tmp.header+4,block_info->next_filepos); - if (info->s->file_write(info,(char*) tmp.header+4,8, + if (info->s->file_write(info, tmp.header+4,8, block_info->prev_filepos+4, MYF(MY_NABP))) DBUG_RETURN(1); /* Unlink block from next block */ @@ -439,7 +439,7 @@ static bool unlink_deleted_block(MARIA_HA *info, MARIA_BLOCK_INFO *block_info) & BLOCK_DELETED)) DBUG_RETURN(1); /* Something is wrong */ mi_sizestore(tmp.header+12,block_info->prev_filepos); - if (info->s->file_write(info,(char*) tmp.header+12,8, + if (info->s->file_write(info, tmp.header+12,8, block_info->next_filepos+12, MYF(MY_NABP))) DBUG_RETURN(1); @@ -489,7 +489,7 @@ static my_bool update_backward_delete_link(MARIA_HA *info, if (_ma_get_block_info(&block_info, info->dfile.file, delete_block) & BLOCK_DELETED) { - char buff[8]; + uchar buff[8]; mi_sizestore(buff,filepos); if (info->s->file_write(info,buff, 8, delete_block+12, MYF(MY_NABP))) DBUG_RETURN(1); /* Error on write */ @@ -1610,7 +1610,7 @@ static my_bool _ma_cmp_buffer(File file, const uchar *buff, my_off_t filepos, uint length) { uint next_length; - char temp_buff[IO_SIZE*2]; + uchar temp_buff[IO_SIZE*2]; DBUG_ENTER("_ma_cmp_buffer"); next_length= IO_SIZE*2 - (uint) (filepos & (IO_SIZE-1)); @@ -1618,7 +1618,7 @@ static my_bool _ma_cmp_buffer(File file, const uchar *buff, my_off_t filepos, while (length > IO_SIZE*2) { if (my_pread(file,temp_buff,next_length,filepos, MYF(MY_NABP)) || - memcmp((uchar*) buff,temp_buff,next_length)) + memcmp(buff, temp_buff, next_length)) goto err; filepos+=next_length; buff+=next_length; @@ -1627,7 +1627,7 @@ static my_bool _ma_cmp_buffer(File file, const uchar *buff, my_off_t filepos, } if (my_pread(file,temp_buff,length,filepos,MYF(MY_NABP))) goto err; - DBUG_RETURN(memcmp((uchar*) buff,temp_buff,length) != 0); + DBUG_RETURN(memcmp(buff, temp_buff, length) != 0); err: DBUG_RETURN(1); } @@ -1851,11 +1851,11 @@ uint _ma_get_block_info(MARIA_BLOCK_INFO *info, File file, my_off_t filepos) my_pread() may leave the file pointer untouched. */ VOID(my_seek(file,filepos,MY_SEEK_SET,MYF(0))); - if (my_read(file,(char*) header,sizeof(info->header),MYF(0)) != + if (my_read(file, header, sizeof(info->header),MYF(0)) != sizeof(info->header)) goto err; } - DBUG_DUMP("header",(uchar*) header,MARIA_BLOCK_INFO_HEADER_LENGTH); + DBUG_DUMP("header",header,MARIA_BLOCK_INFO_HEADER_LENGTH); if (info->second_read) { if (info->header[0] <= 6 || info->header[0] == 13) diff --git a/storage/maria/ma_extra.c b/storage/maria/ma_extra.c index 4f1634756ab..b0b4ce12652 100644 --- a/storage/maria/ma_extra.c +++ b/storage/maria/ma_extra.c @@ -85,7 +85,7 @@ int maria_extra(MARIA_HA *info, enum ha_extra_function function, if (_ma_memmap_file(info)) { /* We don't nead MADV_SEQUENTIAL if small file */ - madvise(share->file_map,share->state.state.data_file_length, + madvise((char*) share->file_map, share->state.state.data_file_length, share->state.state.data_file_length <= RECORD_CACHE_SIZE*16 ? MADV_RANDOM : MADV_SEQUENTIAL); pthread_mutex_unlock(&share->intern_lock); @@ -167,7 +167,8 @@ int maria_extra(MARIA_HA *info, enum ha_extra_function function, } #if defined(HAVE_MMAP) && defined(HAVE_MADVISE) if (info->opt_flag & MEMMAP_USED) - madvise(share->file_map,share->state.state.data_file_length,MADV_RANDOM); + madvise((char*) share->file_map, share->state.state.data_file_length, + MADV_RANDOM); #endif break; case HA_EXTRA_FLUSH_CACHE: @@ -313,8 +314,6 @@ int maria_extra(MARIA_HA *info, enum ha_extra_function function, blocks to disk if this is not for a DROP TABLE. Otherwise they would be invisible to future openers; and they could even go to disk late and cancel the work of future openers. - On Windows, which cannot delete an open file (cannot drop an open table) - we have to close the table's files. */ if (info->lock_type != F_UNLCK && !info->was_locked) { @@ -343,11 +342,7 @@ int maria_extra(MARIA_HA *info, enum ha_extra_function function, { if (do_flush) { - /* - Save the state so that others can find it from disk. - We have to sync now, as on Windows we are going to close the file - (so cannot sync later). - */ + /* Save the state so that others can find it from disk. */ if (_ma_state_info_write(share, 1 | 2) || my_sync(share->kfile.file, MYF(0))) error= my_errno; @@ -361,42 +356,13 @@ int maria_extra(MARIA_HA *info, enum ha_extra_function function, /* be sure that state is not tried for write as file may be closed */ share->changed= 0; } -#ifdef __WIN__ - if (my_close(share->kfile, MYF(0))) - error=my_errno; - share->kfile.file= -1; -#endif } if (share->data_file_type == BLOCK_RECORD && share->bitmap.file.file >= 0) { if (do_flush && my_sync(share->bitmap.file.file, MYF(0))) error= my_errno; -#ifdef __WIN__ - if (my_close(share->bitmap.file.file, MYF(0))) - error= my_errno; - share->bitmap.file.file= -1; -#endif - } -#ifdef __WIN__ - { - LIST *list_element ; - for (list_element=maria_open_list ; - list_element ; - list_element=list_element->next) - { - MARIA_HA *tmpinfo=(MARIA_HA*) list_element->data; - if (tmpinfo->s == info->s) - { - if (share->data_file_type != BLOCK_RECORD && - tmpinfo->dfile.file >= 0 && - my_close(tmpinfo->dfile.file, MYF(0))) - error = my_errno; - tmpinfo->dfile.file= -1; - } - } } -#endif /* For protection against Checkpoint, we set under intern_lock: */ share->last_version= 0L; /* Impossible version */ pthread_mutex_unlock(&share->intern_lock); @@ -544,7 +510,8 @@ int maria_reset(MARIA_HA *info) } #if defined(HAVE_MMAP) && defined(HAVE_MADVISE) if (info->opt_flag & MEMMAP_USED) - madvise(share->file_map,share->state.state.data_file_length,MADV_RANDOM); + madvise((char*) share->file_map, share->state.state.data_file_length, + MADV_RANDOM); #endif info->opt_flag&= ~(KEY_READ_USED | REMEMBER_OLD_POS); info->quick_mode=0; diff --git a/storage/maria/ma_init.c b/storage/maria/ma_init.c index fb8efddd778..7785ac7b8df 100644 --- a/storage/maria/ma_init.c +++ b/storage/maria/ma_init.c @@ -61,6 +61,7 @@ void maria_end(void) trnman_destroy(); translog_destroy(); end_pagecache(maria_log_pagecache, TRUE); + end_pagecache(maria_pagecache, TRUE); ma_control_file_end(); pthread_mutex_destroy(&THR_LOCK_maria); } diff --git a/storage/maria/ma_key.c b/storage/maria/ma_key.c index 96b8d2af0eb..9640daafa84 100644 --- a/storage/maria/ma_key.c +++ b/storage/maria/ma_key.c @@ -108,7 +108,7 @@ uint _ma_make_key(register MARIA_HA *info, uint keynr, uchar *key, { if (type != HA_KEYTYPE_NUM) { - length= cs->cset->lengthsp(cs, pos, length); + length= cs->cset->lengthsp(cs, (char*) pos, length); } else { @@ -383,7 +383,8 @@ static int _ma_put_key_in_record(register MARIA_HA *info, uint keynr, { memcpy(pos,key,(size_t) length); keyseg->charset->cset->fill(keyseg->charset, - pos + length, keyseg->length - length, + (char*) pos + length, + keyseg->length - length, ' '); } else diff --git a/storage/maria/ma_keycache.c b/storage/maria/ma_keycache.c index 7a2a56488e6..9295904dbcf 100644 --- a/storage/maria/ma_keycache.c +++ b/storage/maria/ma_keycache.c @@ -105,7 +105,8 @@ int maria_assign_to_pagecache(MARIA_HA *info, share->pagecache= pagecache; /* store the key cache in the global hash structure for future opens */ - if (multi_pagecache_set(share->unique_file_name, share->unique_name_length, + if (multi_pagecache_set((uchar*) share->unique_file_name, + share->unique_name_length, share->pagecache)) error= my_errno; pthread_mutex_unlock(&share->intern_lock); diff --git a/storage/maria/ma_locking.c b/storage/maria/ma_locking.c index 01d59ed56df..178ced3906f 100644 --- a/storage/maria/ma_locking.c +++ b/storage/maria/ma_locking.c @@ -505,7 +505,7 @@ int _ma_test_if_changed(register MARIA_HA *info) int _ma_mark_file_changed(MARIA_HA *info) { - char buff[3]; + uchar buff[3]; register MARIA_SHARE *share=info->s; DBUG_ENTER("_ma_mark_file_changed"); @@ -543,7 +543,7 @@ int _ma_mark_file_changed(MARIA_HA *info) int _ma_decrement_open_count(MARIA_HA *info) { - char buff[2]; + uchar buff[2]; register MARIA_SHARE *share=info->s; int lock_error=0,write_error=0; if (share->global_changed) diff --git a/storage/maria/ma_open.c b/storage/maria/ma_open.c index 9b665cfb958..8a002832092 100644 --- a/storage/maria/ma_open.c +++ b/storage/maria/ma_open.c @@ -249,7 +249,7 @@ MARIA_HA *maria_open(const char *name, int mode, uint open_flags) key_parts,unique_key_parts,fulltext_keys,uniques; char name_buff[FN_REFLEN], org_name[FN_REFLEN], index_name[FN_REFLEN], data_name[FN_REFLEN]; - char *disk_cache, *disk_pos, *end_pos; + uchar *disk_cache, *disk_pos, *end_pos; MARIA_HA info,*m_info,*old_info; MARIA_SHARE share_buff,*share; ulong rec_per_key_part[HA_MAX_POSSIBLE_KEY*HA_MAX_KEY_SEG]; @@ -275,7 +275,8 @@ MARIA_HA *maria_open(const char *name, int mode, uint open_flags) bzero((uchar*) &share_buff,sizeof(share_buff)); share_buff.state.rec_per_key_part=rec_per_key_part; share_buff.state.key_root=key_root; - share_buff.pagecache= multi_pagecache_search(name_buff, strlen(name_buff), + share_buff.pagecache= multi_pagecache_search((uchar*) name_buff, + strlen(name_buff), maria_pagecache); DBUG_EXECUTE_IF("maria_pretend_crashed_table_on_open", @@ -293,7 +294,7 @@ MARIA_HA *maria_open(const char *name, int mode, uint open_flags) } share->mode=open_mode; errpos= 1; - if (my_read(kfile,(char*) share->state.header.file_version,head_length, + if (my_read(kfile,share->state.header.file_version, head_length, MYF(MY_NABP))) { my_errno= HA_ERR_NOT_A_TABLE; @@ -337,7 +338,7 @@ MARIA_HA *maria_open(const char *name, int mode, uint open_flags) info_length=mi_uint2korr(share->state.header.header_length); base_pos= mi_uint2korr(share->state.header.base_pos); - if (!(disk_cache=(char*) my_alloca(info_length+128))) + if (!(disk_cache= (uchar*) my_alloca(info_length+128))) { my_errno=ENOMEM; goto err; @@ -688,7 +689,7 @@ MARIA_HA *maria_open(const char *name, int mode, uint open_flags) (keys ? MARIA_INDEX_BLOCK_MARGIN * share->block_size * keys : 0)); share->block_size= share->base.block_size; - my_afree((uchar*) disk_cache); + my_afree(disk_cache); _ma_setup_functions(share); if ((*share->once_init)(share, info.dfile.file)) goto err; @@ -1175,7 +1176,7 @@ static uchar *_ma_state_info_read(uchar *ptr, MARIA_STATE_INFO *state) uint _ma_state_info_read_dsk(File file, MARIA_STATE_INFO *state) { - char buff[MARIA_STATE_INFO_SIZE + MARIA_STATE_EXTRA_SIZE]; + uchar buff[MARIA_STATE_INFO_SIZE + MARIA_STATE_EXTRA_SIZE]; /* trick to detect transactional tables */ DBUG_ASSERT(state->create_rename_lsn == LSN_IMPOSSIBLE); @@ -1297,7 +1298,7 @@ uint _ma_keydef_write(File file, MARIA_KEYDEF *keydef) return my_write(file, buff, (size_t) (ptr-buff), MYF(MY_NABP)) != 0; } -char *_ma_keydef_read(char *ptr, MARIA_KEYDEF *keydef) +uchar *_ma_keydef_read(uchar *ptr, MARIA_KEYDEF *keydef) { keydef->keysegs = (uint) *ptr++; keydef->key_alg = *ptr++; /* Rtree or Btree */ @@ -1341,7 +1342,7 @@ int _ma_keyseg_write(File file, const HA_KEYSEG *keyseg) } -char *_ma_keyseg_read(char *ptr, HA_KEYSEG *keyseg) +uchar *_ma_keyseg_read(uchar *ptr, HA_KEYSEG *keyseg) { keyseg->type = *ptr++; keyseg->language = *ptr++; @@ -1380,7 +1381,7 @@ uint _ma_uniquedef_write(File file, MARIA_UNIQUEDEF *def) return my_write(file, buff, (size_t) (ptr-buff), MYF(MY_NABP)) != 0; } -char *_ma_uniquedef_read(char *ptr, MARIA_UNIQUEDEF *def) +uchar *_ma_uniquedef_read(uchar *ptr, MARIA_UNIQUEDEF *def) { def->keysegs = mi_uint2korr(ptr); def->key = ptr[2]; @@ -1408,7 +1409,7 @@ uint _ma_columndef_write(File file, MARIA_COLUMNDEF *columndef) return my_write(file, buff, (size_t) (ptr-buff), MYF(MY_NABP)) != 0; } -char *_ma_columndef_read(char *ptr, MARIA_COLUMNDEF *columndef) +uchar *_ma_columndef_read(uchar *ptr, MARIA_COLUMNDEF *columndef) { columndef->offset= mi_uint6korr(ptr); ptr+= 6; columndef->type= mi_sint2korr(ptr); ptr+= 2; diff --git a/storage/maria/ma_packrec.c b/storage/maria/ma_packrec.c index 173fafaf73f..446981d9fb9 100644 --- a/storage/maria/ma_packrec.c +++ b/storage/maria/ma_packrec.c @@ -170,7 +170,7 @@ static my_bool _ma_read_pack_info(MARIA_SHARE *share, File file, uint i,trees,huff_tree_bits,rec_reflength,length; uint16 *decode_table,*tmp_buff; ulong elements,intervall_length; - char *disk_cache; + uchar *disk_cache; uchar *intervall_buff; uchar header[HEAD_LENGTH]; MARIA_BIT_BUFF bit_buff; @@ -257,7 +257,7 @@ static my_bool _ma_read_pack_info(MARIA_SHARE *share, File file, goto err2; huff_tree_bits=max_bit(trees ? trees-1 : 0); - init_bit_buffer(&bit_buff, (uchar*) disk_cache, + init_bit_buffer(&bit_buff, disk_cache, (uint) (share->pack.header_length-sizeof(header))); /* Read new info for each field */ for (i=0 ; i < share->base.fields ; i++) @@ -1416,7 +1416,7 @@ uint _ma_pack_get_block_info(MARIA_HA *maria, MARIA_BIT_BUFF *bit_buff, position is ok */ VOID(my_seek(file,filepos,MY_SEEK_SET,MYF(0))); - if (my_read(file,(char*) header,ref_length,MYF(MY_NABP))) + if (my_read(file, header,ref_length,MYF(MY_NABP))) return BLOCK_FATAL_ERROR; DBUG_DUMP("header",(uchar*) header,ref_length); } @@ -1559,7 +1559,7 @@ my_bool _ma_memmap_file(MARIA_HA *info) void _ma_unmap_file(MARIA_HA *info) { - VOID(my_munmap(info->s->file_map, + VOID(my_munmap((char*) info->s->file_map, (size_t) info->s->mmaped_length + MEMMAP_EXTRA_MARGIN)); } diff --git a/storage/maria/ma_page.c b/storage/maria/ma_page.c index f749414474f..ac9e33f896a 100644 --- a/storage/maria/ma_page.c +++ b/storage/maria/ma_page.c @@ -116,7 +116,7 @@ int _ma_dispose(register MARIA_HA *info, MARIA_KEYDEF *keyinfo, my_off_t pos, int level) { my_off_t old_link; - char buff[8]; + uchar buff[8]; uint offset; pgcache_page_no_t page_no; DBUG_ENTER("_ma_dispose"); @@ -166,7 +166,7 @@ my_off_t _ma_new(register MARIA_HA *info, MARIA_KEYDEF *keyinfo, int level) } else { - buff= alloca(info->s->block_size); + buff= my_alloca(info->s->block_size); DBUG_ASSERT(info->s->pagecache->block_size == keyinfo->block_length && info->s->pagecache->block_size == info->s->block_size); /* @@ -181,6 +181,7 @@ my_off_t _ma_new(register MARIA_HA *info, MARIA_KEYDEF *keyinfo, int level) pos= HA_OFFSET_ERROR; else info->s->state.key_del= mi_sizekorr(buff); + my_afree(buff); } info->s->state.changed|= STATE_NOT_SORTED_PAGES; DBUG_PRINT("exit",("Pos: %ld",(long) pos)); diff --git a/storage/maria/ma_pagecache.c b/storage/maria/ma_pagecache.c index 9f450d25c50..702d8c6d857 100755 --- a/storage/maria/ma_pagecache.c +++ b/storage/maria/ma_pagecache.c @@ -726,8 +726,7 @@ int init_pagecache(PAGECACHE *pagecache, size_t use_mem, For each block 2 hash links are allocated */ if ((pagecache->block_root= - (PAGECACHE_BLOCK_LINK*) my_malloc((uint) length, - MYF(0)))) + (PAGECACHE_BLOCK_LINK*) my_malloc((uint) length, MYF(0)))) break; my_large_free(pagecache->block_mem, MYF(0)); pagecache->block_mem= 0; diff --git a/storage/maria/ma_rkey.c b/storage/maria/ma_rkey.c index c9653d30110..ff1110e5344 100644 --- a/storage/maria/ma_rkey.c +++ b/storage/maria/ma_rkey.c @@ -45,6 +45,7 @@ int maria_rkey(MARIA_HA *info, uchar *buf, int inx, const uchar *key, info->once_flags&= ~USE_PACKED_KEYS; /* Reset flag */ /* key is already packed!; This happens when we are using a MERGE TABLE + In this key 'key_part_map' is the length of the key ! */ key_buff= info->lastkey+info->s->base.max_key_length; pack_key_length= keypart_map; @@ -95,43 +96,63 @@ int maria_rkey(MARIA_HA *info, uchar *buf, int inx, const uchar *key, share->concurrent_insert) { /* - If we searching for a partial key (or using >, >=, < or <=) and - the data is outside of the data file, we need to continue searching - for the first key inside the data file + Found a key, but it might not be usable. We cannot use rows that + are inserted by other threads after we got our table lock + ("concurrent inserts"). The record may not even be present yet. + Keys are inserted into the index(es) before the record is + inserted into the data file. When we got our table lock, we + saved the current data_file_length. Concurrent inserts always go + to the end of the file. So we can test if the found key + references a new record. */ - if (info->cur_row.lastpos >= info->state->data_file_length && - (search_flag != HA_READ_KEY_EXACT || - last_used_keyseg != keyinfo->seg + keyinfo->keysegs)) + if (info->cur_row.lastpos >= info->state->data_file_length) { - do + /* The key references a concurrently inserted record. */ + if (search_flag == HA_READ_KEY_EXACT && + last_used_keyseg == keyinfo->seg + keyinfo->keysegs) + { + /* Simply ignore the key if it matches exactly. (Bug #29838) */ + my_errno= HA_ERR_KEY_NOT_FOUND; + info->cur_row.lastpos= HA_OFFSET_ERROR; + } + else { - uint not_used[2]; - /* - Skip rows that are inserted by other threads since we got a lock - Note that this can only happen if we are not searching after an - full length exact key, because the keys are sorted - according to position - */ - if (_ma_search_next(info, keyinfo, info->lastkey, - info->lastkey_length, - maria_readnext_vec[search_flag], - info->s->state.key_root[inx])) - break; /* - Check that the found key does still match the search. - _ma_search_next() delivers the next key regardless of its - value. + If searching for a partial key (or using >, >=, < or <=) and + the data is outside of the data file, we need to continue + searching for the first key inside the data file. */ - if (search_flag == HA_READ_KEY_EXACT && - ha_key_cmp(keyinfo->seg, (uchar*) key_buff, - (uchar*) info->lastkey, use_key_length, - SEARCH_FIND, not_used)) + do { - my_errno= HA_ERR_KEY_NOT_FOUND; - info->cur_row.lastpos= HA_OFFSET_ERROR; - break; - } - } while (info->cur_row.lastpos >= info->state->data_file_length); + uint not_used[2]; + /* + Skip rows that are inserted by other threads since we got + a lock. Note that this can only happen if we are not + searching after a full length exact key, because the keys + are sorted according to position. + */ + if (_ma_search_next(info, keyinfo, info->lastkey, + info->lastkey_length, + maria_readnext_vec[search_flag], + info->s->state.key_root[inx])) + break; /* purecov: inspected */ + /* + Check that the found key does still match the search. + _ma_search_next() delivers the next key regardless of its + value. + */ + if (search_flag == HA_READ_KEY_EXACT && + ha_key_cmp(keyinfo->seg, key_buff, info->lastkey, + use_key_length, SEARCH_FIND, not_used)) + { + /* purecov: begin inspected */ + my_errno= HA_ERR_KEY_NOT_FOUND; + info->cur_row.lastpos= HA_OFFSET_ERROR; + break; + /* purecov: end */ + } + } while (info->cur_row.lastpos >= info->state->data_file_length); + } } } } diff --git a/storage/maria/ma_search.c b/storage/maria/ma_search.c index 8cb3e56e646..7bbf461096c 100644 --- a/storage/maria/ma_search.c +++ b/storage/maria/ma_search.c @@ -837,7 +837,7 @@ uint _ma_get_pack_key(register MARIA_KEYDEF *keyinfo, uint nod_flag, if (tot_length >= 255 && *start != 255) { /* length prefix changed from a length of one to a length of 3 */ - bmove_upp((char*) key+length+3,(char*) key+length+1,length); + bmove_upp(key+length+3, key+length+1, length); *key=255; mi_int2store(key+1,tot_length); key+=3+length; diff --git a/storage/maria/ma_sort.c b/storage/maria/ma_sort.c index 2851a3a09dd..e367f8c20e3 100644 --- a/storage/maria/ma_sort.c +++ b/storage/maria/ma_sort.c @@ -47,42 +47,41 @@ extern void print_error _VARARGS((const char *fmt,...)); /* Functions defined in this file */ -static ha_rows NEAR_F find_all_keys(MARIA_SORT_PARAM *info,uint keys, - uchar **sort_keys, - DYNAMIC_ARRAY *buffpek,int *maxbuffer, - IO_CACHE *tempfile, - IO_CACHE *tempfile_for_exceptions); -static int NEAR_F write_keys(MARIA_SORT_PARAM *info, uchar **sort_keys, - uint count, BUFFPEK *buffpek,IO_CACHE *tempfile); -static int NEAR_F write_key(MARIA_SORT_PARAM *info, uchar *key, - IO_CACHE *tempfile); -static int NEAR_F write_index(MARIA_SORT_PARAM *info, uchar **sort_keys, - uint count); -static int NEAR_F merge_many_buff(MARIA_SORT_PARAM *info,uint keys, - uchar **sort_keys, - BUFFPEK *buffpek,int *maxbuffer, - IO_CACHE *t_file); -static uint NEAR_F read_to_buffer(IO_CACHE *fromfile,BUFFPEK *buffpek, - uint sort_length); -static int NEAR_F merge_buffers(MARIA_SORT_PARAM *info,uint keys, - IO_CACHE *from_file, IO_CACHE *to_file, - uchar **sort_keys, BUFFPEK *lastbuff, - BUFFPEK *Fb, BUFFPEK *Tb); -static int NEAR_F merge_index(MARIA_SORT_PARAM *,uint, uchar **,BUFFPEK *, int, - IO_CACHE *); +static ha_rows find_all_keys(MARIA_SORT_PARAM *info,uint keys, + uchar **sort_keys, + DYNAMIC_ARRAY *buffpek,int *maxbuffer, + IO_CACHE *tempfile, + IO_CACHE *tempfile_for_exceptions); +static int write_keys(MARIA_SORT_PARAM *info, uchar **sort_keys, + uint count, BUFFPEK *buffpek,IO_CACHE *tempfile); +static int write_key(MARIA_SORT_PARAM *info, uchar *key, + IO_CACHE *tempfile); +static int write_index(MARIA_SORT_PARAM *info, uchar **sort_keys, + uint count); +static int merge_many_buff(MARIA_SORT_PARAM *info,uint keys, + uchar **sort_keys, + BUFFPEK *buffpek,int *maxbuffer, + IO_CACHE *t_file); +static uint read_to_buffer(IO_CACHE *fromfile,BUFFPEK *buffpek, + uint sort_length); +static int merge_buffers(MARIA_SORT_PARAM *info,uint keys, + IO_CACHE *from_file, IO_CACHE *to_file, + uchar **sort_keys, BUFFPEK *lastbuff, + BUFFPEK *Fb, BUFFPEK *Tb); +static int merge_index(MARIA_SORT_PARAM *,uint, uchar **,BUFFPEK *, int, + IO_CACHE *); static int flush_maria_ft_buf(MARIA_SORT_PARAM *info); -static int NEAR_F write_keys_varlen(MARIA_SORT_PARAM *info, uchar **sort_keys, - uint count, BUFFPEK *buffpek, - IO_CACHE *tempfile); -static uint NEAR_F read_to_buffer_varlen(IO_CACHE *fromfile,BUFFPEK *buffpek, - uint sort_length); -static int NEAR_F write_merge_key(MARIA_SORT_PARAM *info, IO_CACHE *to_file, - char *key, uint sort_length, uint count); -static int NEAR_F write_merge_key_varlen(MARIA_SORT_PARAM *info, - IO_CACHE *to_file, - char* key, uint sort_length, - uint count); +static int write_keys_varlen(MARIA_SORT_PARAM *info, uchar **sort_keys, + uint count, BUFFPEK *buffpek, + IO_CACHE *tempfile); +static uint read_to_buffer_varlen(IO_CACHE *fromfile,BUFFPEK *buffpek, + uint sort_length); +static int write_merge_key(MARIA_SORT_PARAM *info, IO_CACHE *to_file, + uchar *key, uint sort_length, uint count); +static int write_merge_key_varlen(MARIA_SORT_PARAM *info, + IO_CACHE *to_file, uchar *key, + uint sort_length, uint count); static inline int my_var_write(MARIA_SORT_PARAM *info, IO_CACHE *to_file, uchar *bufs); @@ -141,7 +140,7 @@ int _ma_create_index_by_sort(MARIA_SORT_PARAM *info, my_bool no_messages, if ((records < UINT_MAX32) && ((my_off_t) (records + 1) * (sort_length + sizeof(char*)) <= (my_off_t) memavl)) - keys= records+1; + keys= (uint)records+1; else do { @@ -257,7 +256,7 @@ err: /* Search after all keys and place them in a temp. file */ -static ha_rows NEAR_F find_all_keys(MARIA_SORT_PARAM *info, uint keys, +static ha_rows find_all_keys(MARIA_SORT_PARAM *info, uint keys, uchar **sort_keys, DYNAMIC_ARRAY *buffpek, int *maxbuffer, IO_CACHE *tempfile, IO_CACHE *tempfile_for_exceptions) @@ -350,7 +349,7 @@ pthread_handler_t _ma_thr_find_all_keys(void *arg) bzero((char*) &sort_param->unique, sizeof(sort_param->unique)); memavl= max(sort_param->sortbuff_size, MIN_SORT_MEMORY); - idx= sort_param->sort_info->max_records; + idx= (uint)sort_param->sort_info->max_records; sort_length= sort_param->key_length; maxbuffer= 1; @@ -685,7 +684,7 @@ my_var_write(MARIA_SORT_PARAM *info, IO_CACHE *to_file, uchar *bufs) } -static int NEAR_F write_keys_varlen(MARIA_SORT_PARAM *info, +static int write_keys_varlen(MARIA_SORT_PARAM *info, register uchar **sort_keys, uint count, BUFFPEK *buffpek, IO_CACHE *tempfile) @@ -712,7 +711,7 @@ static int NEAR_F write_keys_varlen(MARIA_SORT_PARAM *info, } /* write_keys_varlen */ -static int NEAR_F write_key(MARIA_SORT_PARAM *info, uchar *key, +static int write_key(MARIA_SORT_PARAM *info, uchar *key, IO_CACHE *tempfile) { uint key_length=info->real_key_length; @@ -732,7 +731,7 @@ static int NEAR_F write_key(MARIA_SORT_PARAM *info, uchar *key, /* Write index */ -static int NEAR_F write_index(MARIA_SORT_PARAM *info, +static int write_index(MARIA_SORT_PARAM *info, register uchar **sort_keys, register uint count) { @@ -751,7 +750,7 @@ static int NEAR_F write_index(MARIA_SORT_PARAM *info, /* Merge buffers to make < MERGEBUFF2 buffers */ -static int NEAR_F merge_many_buff(MARIA_SORT_PARAM *info, uint keys, +static int merge_many_buff(MARIA_SORT_PARAM *info, uint keys, uchar **sort_keys, BUFFPEK *buffpek, int *maxbuffer, IO_CACHE *t_file) { @@ -809,7 +808,7 @@ cleanup: -1 Error */ -static uint NEAR_F read_to_buffer(IO_CACHE *fromfile, BUFFPEK *buffpek, +static uint read_to_buffer(IO_CACHE *fromfile, BUFFPEK *buffpek, uint sort_length) { register uint count; @@ -828,7 +827,7 @@ static uint NEAR_F read_to_buffer(IO_CACHE *fromfile, BUFFPEK *buffpek, return (count*sort_length); } /* read_to_buffer */ -static uint NEAR_F read_to_buffer_varlen(IO_CACHE *fromfile, BUFFPEK *buffpek, +static uint read_to_buffer_varlen(IO_CACHE *fromfile, BUFFPEK *buffpek, uint sort_length) { register uint count; @@ -860,17 +859,17 @@ static uint NEAR_F read_to_buffer_varlen(IO_CACHE *fromfile, BUFFPEK *buffpek, } /* read_to_buffer_varlen */ -static int NEAR_F write_merge_key_varlen(MARIA_SORT_PARAM *info, - IO_CACHE *to_file,char* key, - uint sort_length, uint count) +static int write_merge_key_varlen(MARIA_SORT_PARAM *info, + IO_CACHE *to_file, uchar* key, + uint sort_length, uint count) { uint idx; + uchar *bufs = key; - char *bufs = key; for (idx=1;idx<=count;idx++) { int err; - if ((err= my_var_write(info,to_file, (uchar*) bufs))) + if ((err= my_var_write(info, to_file, bufs))) return (err); bufs=bufs+sort_length; } @@ -878,11 +877,11 @@ static int NEAR_F write_merge_key_varlen(MARIA_SORT_PARAM *info, } -static int NEAR_F write_merge_key(MARIA_SORT_PARAM *info __attribute__((unused)), - IO_CACHE *to_file, char* key, +static int write_merge_key(MARIA_SORT_PARAM *info __attribute__((unused)), + IO_CACHE *to_file, uchar *key, uint sort_length, uint count) { - return my_b_write(to_file,(uchar*) key,(uint) sort_length*count); + return my_b_write(to_file, key, (size_t) sort_length*count); } /* @@ -927,7 +926,7 @@ merge_buffers(MARIA_SORT_PARAM *info, uint keys, IO_CACHE *from_file, sort_length)); if (error == -1) goto err; /* purecov: inspected */ - queue_insert(&queue,(char*) buffpek); + queue_insert(&queue,(uchar*) buffpek); } while (queue.elements > 1) diff --git a/storage/maria/ma_sp_key.c b/storage/maria/ma_sp_key.c index 1ea9b410ab6..5f83b432228 100644 --- a/storage/maria/ma_sp_key.c +++ b/storage/maria/ma_sp_key.c @@ -31,11 +31,6 @@ static int sp_get_geometry_mbr(uchar *(*wkb), uchar *end, uint n_dims, double *mbr, int top); static int sp_mbr_from_wkb(uchar (*wkb), uint size, uint n_dims, double *mbr); -static void get_double(double *d, const uchar *pos) -{ - float8get(*d, pos); -} - uint _ma_sp_make_key(register MARIA_HA *info, uint keynr, uchar *key, const uchar *record, my_off_t filepos) { @@ -62,48 +57,40 @@ uint _ma_sp_make_key(register MARIA_HA *info, uint keynr, uchar *key, for (i = 0, keyseg = keyinfo->seg; keyseg->type; keyseg++, i++) { - uint length = keyseg->length; + uint length = keyseg->length, start= keyseg->start; + double val; - pos = ((uchar*)mbr) + keyseg->start; - if (keyseg->flag & HA_SWAP_KEY) - { + DBUG_ASSERT(length == sizeof(double)); + DBUG_ASSERT(!(start % sizeof(double))); + DBUG_ASSERT(start < sizeof(mbr)); + DBUG_ASSERT(keyseg->type == HA_KEYTYPE_DOUBLE); + + val= mbr[start / sizeof (double)]; #ifdef HAVE_ISNAN - if (keyseg->type == HA_KEYTYPE_FLOAT) - { - float nr; - float4get(nr, pos); - if (isnan(nr)) - { - /* Replace NAN with zero */ - bzero(key, length); - key+= length; - continue; - } - } - else if (keyseg->type == HA_KEYTYPE_DOUBLE) - { - double nr; - get_double(&nr, pos); - if (isnan(nr)) - { - bzero(key, length); - key+= length; - continue; - } - } + if (isnan(val)) + { + bzero(key, length); + key+= length; + len+= length; + continue; + } #endif - pos += length; - while (length--) - { - *key++ = *--pos; - } + + if (keyseg->flag & HA_SWAP_KEY) + { + uchar buf[sizeof(double)]; + + float8store(buf, val); + pos= &buf[length]; + while (pos > buf) + *key++ = *--pos; } else { - memcpy((uchar*)key, pos, length); - key += keyseg->length; + float8store((uchar *)key, val); + key += length; } - len += keyseg->length; + len+= length; } _ma_dpointer(info, key, filepos); return len; @@ -141,13 +128,13 @@ static int sp_add_point_to_mbr(uchar *(*wkb), uchar *end, uint n_dims, { if ((*wkb) > end - 8) return -1; - get_double(&ord, (const uchar*) *wkb); + float8get(ord, (const uchar*) *wkb); (*wkb)+= 8; if (ord < *mbr) - float8store((char*) mbr, ord); + *mbr= ord; mbr++; if (ord > *mbr) - float8store((char*) mbr, ord); + *mbr= ord; mbr++; } return 0; diff --git a/storage/maria/ma_sp_test.c b/storage/maria/ma_sp_test.c index 7a413f68135..b8c00753acb 100644 --- a/storage/maria/ma_sp_test.c +++ b/storage/maria/ma_sp_test.c @@ -24,11 +24,11 @@ #define MAX_REC_LENGTH 1024 #define KEYALG HA_KEY_ALG_RTREE -static void create_linestring(char *record,uint rownr); -static void print_record(char * record,my_off_t offs,const char * tail); +static void create_linestring(uchar *record,uint rownr); +static void print_record(uchar * record,my_off_t offs,const char * tail); -static void create_key(char *key,uint rownr); -static void print_key(const char *key,const char * tail); +static void create_key(uchar *key,uint rownr); +static void print_key(const uchar *key,const char * tail); static int run_test(const char *filename); static int read_with_pos(MARIA_HA * file, int silent); @@ -65,9 +65,9 @@ int run_test(const char *filename) int i; int error; int row_count=0; - char record[MAX_REC_LENGTH]; - char key[MAX_REC_LENGTH]; - char read_record[MAX_REC_LENGTH]; + uchar record[MAX_REC_LENGTH]; + uchar key[MAX_REC_LENGTH]; + uchar read_record[MAX_REC_LENGTH]; int upd=10; ha_rows hrows; @@ -277,7 +277,7 @@ static int read_with_pos (MARIA_HA * file,int silent) { int error; int i; - char read_record[MAX_REC_LENGTH]; + uchar read_record[MAX_REC_LENGTH]; int rows=0; if (!silent) @@ -305,7 +305,7 @@ static int read_with_pos (MARIA_HA * file,int silent) #ifdef NOT_USED -static void bprint_record(char * record, +static void bprint_record(uchar * record, my_off_t offs __attribute__((unused)), const char * tail) { @@ -324,9 +324,9 @@ static void bprint_record(char * record, #endif -static void print_record(char * record, my_off_t offs,const char * tail) +static void print_record(uchar * record, my_off_t offs,const char * tail) { - char *pos; + uchar *pos; char *ptr; uint len; @@ -346,7 +346,7 @@ static void print_record(char * record, my_off_t offs,const char * tail) #ifdef NOT_USED -static void create_point(char *record,uint rownr) +static void create_point(uchar *record,uint rownr) { uint tmp; char *ptr; @@ -373,11 +373,11 @@ static void create_point(char *record,uint rownr) #endif -static void create_linestring(char *record,uint rownr) +static void create_linestring(uchar *record,uint rownr) { uint tmp; char *ptr; - char *pos=record; + uchar *pos= record; double x[200]; int i,j; int npoints=2; @@ -401,10 +401,10 @@ static void create_linestring(char *record,uint rownr) } -static void create_key(char *key,uint rownr) +static void create_key(uchar *key,uint rownr) { double c=rownr; - char *pos; + uchar *pos; uint i; bzero(key,MAX_REC_LENGTH); @@ -415,7 +415,7 @@ static void create_key(char *key,uint rownr) } } -static void print_key(const char *key,const char * tail) +static void print_key(const uchar *key,const char * tail) { double c; uint i; diff --git a/storage/maria/ma_statrec.c b/storage/maria/ma_statrec.c index ebfab4fad76..03ebc781104 100644 --- a/storage/maria/ma_statrec.c +++ b/storage/maria/ma_statrec.c @@ -26,16 +26,15 @@ my_bool _ma_write_static_record(MARIA_HA *info, const uchar *record) { my_off_t filepos=info->s->state.dellink; info->rec_cache.seek_not_done=1; /* We have done a seek */ - if (info->s->file_read(info,(char*) &temp[0],info->s->base.rec_reflength, + if (info->s->file_read(info, &temp[0],info->s->base.rec_reflength, info->s->state.dellink+1, MYF(MY_NABP))) goto err; info->s->state.dellink= _ma_rec_pos(info->s,temp); info->state->del--; info->state->empty-=info->s->base.pack_reclength; - if (info->s->file_write(info, (char*) record, info->s->base.reclength, - filepos, - MYF(MY_NABP))) + if (info->s->file_write(info, record, info->s->base.reclength, + filepos, MYF(MY_NABP))) goto err; } else @@ -48,29 +47,29 @@ my_bool _ma_write_static_record(MARIA_HA *info, const uchar *record) } if (info->opt_flag & WRITE_CACHE_USED) { /* Cash in use */ - if (my_b_write(&info->rec_cache, (uchar*) record, + if (my_b_write(&info->rec_cache, record, info->s->base.reclength)) goto err; if (info->s->base.pack_reclength != info->s->base.reclength) { uint length=info->s->base.pack_reclength - info->s->base.reclength; - bzero((char*) temp,length); - if (my_b_write(&info->rec_cache, (uchar*) temp,length)) + bzero(temp,length); + if (my_b_write(&info->rec_cache, temp,length)) goto err; } } else { info->rec_cache.seek_not_done=1; /* We have done a seek */ - if (info->s->file_write(info,(char*) record,info->s->base.reclength, + if (info->s->file_write(info, record, info->s->base.reclength, info->state->data_file_length, info->s->write_flag)) goto err; if (info->s->base.pack_reclength != info->s->base.reclength) { uint length=info->s->base.pack_reclength - info->s->base.reclength; - bzero((char*) temp,length); - if (info->s->file_write(info, (uchar*) temp,length, + bzero(temp,length); + if (info->s->file_write(info, temp,length, info->state->data_file_length+ info->s->base.reclength, info->s->write_flag)) @@ -91,7 +90,7 @@ my_bool _ma_update_static_record(MARIA_HA *info, MARIA_RECORD_POS pos, { info->rec_cache.seek_not_done=1; /* We have done a seek */ return (info->s->file_write(info, - (char*) record,info->s->base.reclength, + record, info->s->base.reclength, pos, MYF(MY_NABP)) != 0); } @@ -132,13 +131,10 @@ my_bool _ma_cmp_static_record(register MARIA_HA *info, if ((info->opt_flag & READ_CHECK_USED)) { /* If check isn't disabled */ info->rec_cache.seek_not_done=1; /* We have done a seek */ - if (info->s->file_read(info, (char*) info->rec_buff, - info->s->base.reclength, - info->cur_row.lastpos, - MYF(MY_NABP))) + if (info->s->file_read(info, info->rec_buff, info->s->base.reclength, + info->cur_row.lastpos, MYF(MY_NABP))) DBUG_RETURN(1); - if (memcmp((uchar*) info->rec_buff, (uchar*) old, - (uint) info->s->base.reclength)) + if (memcmp(info->rec_buff, old, (uint) info->s->base.reclength)) { DBUG_DUMP("read",old,info->s->base.reclength); DBUG_DUMP("disk",info->rec_buff,info->s->base.reclength); @@ -156,7 +152,7 @@ my_bool _ma_cmp_static_unique(MARIA_HA *info, MARIA_UNIQUEDEF *def, DBUG_ENTER("_ma_cmp_static_unique"); info->rec_cache.seek_not_done=1; /* We have done a seek */ - if (info->s->file_read(info, (char*) info->rec_buff, info->s->base.reclength, + if (info->s->file_read(info, info->rec_buff, info->s->base.reclength, pos, MYF(MY_NABP))) DBUG_RETURN(1); DBUG_RETURN(_ma_unique_comp(def, record, (uchar*) info->rec_buff, @@ -186,7 +182,7 @@ int _ma_read_static_record(register MARIA_HA *info, register uchar *record, return(my_errno); info->rec_cache.seek_not_done=1; /* We have done a seek */ - error=info->s->file_read(info,(char*) record,info->s->base.reclength, + error=info->s->file_read(info, record,info->s->base.reclength, pos, MYF(MY_NABP)); if (! error) { diff --git a/storage/maria/ma_test1.c b/storage/maria/ma_test1.c index 80bd3c348a7..fbb91256bc2 100644 --- a/storage/maria/ma_test1.c +++ b/storage/maria/ma_test1.c @@ -48,9 +48,9 @@ static HA_KEYSEG uniqueseg[10]; static int run_test(const char *filename); static void get_options(int argc, char *argv[]); -static void create_key(char *key,uint rownr); -static void create_record(char *record,uint rownr); -static void update_record(char *record); +static void create_key(uchar *key,uint rownr); +static void create_record(uchar *record,uint rownr); +static void update_record(uchar *record); /* @@ -98,7 +98,7 @@ static int run_test(const char *filename) int i,j,error,deleted,rec_length,uniques=0; uint offset_to_key; ha_rows found,row_count; - char record[MAX_REC_LENGTH],key[MAX_REC_LENGTH],read_record[MAX_REC_LENGTH]; + uchar record[MAX_REC_LENGTH],key[MAX_REC_LENGTH],read_record[MAX_REC_LENGTH]; MARIA_UNIQUEDEF uniquedef; MARIA_CREATE_INFO create_info; @@ -420,6 +420,7 @@ static int run_test(const char *filename) i-1,error,my_errno,read_record+1); } } + maria_scan_end(file); end: if (die_in_middle_of_transaction) @@ -467,20 +468,20 @@ err: } -static void create_key_part(char *key,uint rownr) +static void create_key_part(uchar *key,uint rownr) { if (!unique_key) rownr&=7; /* Some identical keys */ if (keyinfo[0].seg[0].type == HA_KEYTYPE_NUM) { - sprintf(key,"%*d",keyinfo[0].seg[0].length,rownr); + sprintf((char*) key,"%*d",keyinfo[0].seg[0].length,rownr); } else if (keyinfo[0].seg[0].type == HA_KEYTYPE_VARTEXT1 || keyinfo[0].seg[0].type == HA_KEYTYPE_VARTEXT2) { /* Alpha record */ /* Create a key that may be easily packed */ bfill(key,keyinfo[0].seg[0].length,rownr < 10 ? 'A' : 'B'); - sprintf(key+keyinfo[0].seg[0].length-2,"%-2d",rownr); + sprintf((char*) key+keyinfo[0].seg[0].length-2,"%-2d",rownr); if ((rownr & 7) == 0) { /* Change the key to force a unpack of the next key */ @@ -490,12 +491,12 @@ static void create_key_part(char *key,uint rownr) else { /* Alpha record */ if (keyinfo[0].seg[0].flag & HA_SPACE_PACK) - sprintf(key,"%-*d",keyinfo[0].seg[0].length,rownr); + sprintf((char*) key,"%-*d",keyinfo[0].seg[0].length,rownr); else { /* Create a key that may be easily packed */ bfill(key,keyinfo[0].seg[0].length,rownr < 10 ? 'A' : 'B'); - sprintf(key+keyinfo[0].seg[0].length-2,"%-2d",rownr); + sprintf((char*) key+keyinfo[0].seg[0].length-2,"%-2d",rownr); if ((rownr & 7) == 0) { /* Change the key to force a unpack of the next key */ @@ -506,7 +507,7 @@ static void create_key_part(char *key,uint rownr) } -static void create_key(char *key,uint rownr) +static void create_key(uchar *key,uint rownr) { if (keyinfo[0].seg[0].null_bit) { @@ -522,7 +523,7 @@ static void create_key(char *key,uint rownr) { uint tmp; create_key_part(key+2,rownr); - tmp=strlen(key+2); + tmp=strlen((char*) key+2); int2store(key,tmp); } else @@ -530,13 +531,13 @@ static void create_key(char *key,uint rownr) } -static char blob_key[MAX_REC_LENGTH]; -static char blob_record[MAX_REC_LENGTH+20*20]; +static uchar blob_key[MAX_REC_LENGTH]; +static uchar blob_record[MAX_REC_LENGTH+20*20]; -static void create_record(char *record,uint rownr) +static void create_record(uchar *record,uint rownr) { - char *pos; + uchar *pos; bzero((char*) record,MAX_REC_LENGTH); record[0]=1; /* delete marker */ if (rownr == 0 && keyinfo[0].seg[0].null_bit) @@ -546,9 +547,9 @@ static void create_record(char *record,uint rownr) if (recinfo[0].type == FIELD_BLOB) { uint tmp; - char *ptr; + uchar *ptr; create_key_part(blob_key,rownr); - tmp=strlen(blob_key); + tmp=strlen((char*) blob_key); int4store(pos,tmp); ptr=blob_key; memcpy_fixed(pos+4,&ptr,sizeof(char*)); @@ -558,7 +559,7 @@ static void create_record(char *record,uint rownr) { uint tmp, pack_length= HA_VARCHAR_PACKLENGTH(recinfo[0].length-1); create_key_part(pos+pack_length,rownr); - tmp= strlen(pos+pack_length); + tmp= strlen((char*) pos+pack_length); if (pack_length == 1) *(uchar*) pos= (uchar) tmp; else @@ -573,10 +574,10 @@ static void create_record(char *record,uint rownr) if (recinfo[1].type == FIELD_BLOB) { uint tmp; - char *ptr;; - sprintf(blob_record,"... row: %d", rownr); - strappend(blob_record,max(MAX_REC_LENGTH-rownr,10),' '); - tmp=strlen(blob_record); + uchar *ptr;; + sprintf((char*) blob_record,"... row: %d", rownr); + strappend((char*) blob_record,max(MAX_REC_LENGTH-rownr,10),' '); + tmp=strlen((char*) blob_record); int4store(pos,tmp); ptr=blob_record; memcpy_fixed(pos+4,&ptr,sizeof(char*)); @@ -584,28 +585,28 @@ static void create_record(char *record,uint rownr) else if (recinfo[1].type == FIELD_VARCHAR) { uint tmp, pack_length= HA_VARCHAR_PACKLENGTH(recinfo[1].length-1); - sprintf(pos+pack_length, "... row: %d", rownr); - tmp= strlen(pos+pack_length); + sprintf((char*) pos+pack_length, "... row: %d", rownr); + tmp= strlen((char*) pos+pack_length); if (pack_length == 1) - *(uchar*) pos= (uchar) tmp; + pos= (uchar) tmp; else int2store(pos,tmp); } else { - sprintf(pos,"... row: %d", rownr); - strappend(pos,recinfo[1].length,' '); + sprintf((char*) pos,"... row: %d", rownr); + strappend((char*) pos,recinfo[1].length,' '); } } /* change row to test re-packing of rows and reallocation of keys */ -static void update_record(char *record) +static void update_record(uchar *record) { - char *pos=record+1; + uchar *pos=record+1; if (recinfo[0].type == FIELD_BLOB) { - char *column,*ptr; + uchar *column,*ptr; int length; length=uint4korr(pos); /* Long blob */ memcpy_fixed(&column,pos+4,sizeof(char*)); @@ -614,7 +615,8 @@ static void update_record(char *record) memcpy_fixed(pos+4,&ptr,sizeof(char*)); /* Store pointer to new key */ if (keyinfo[0].seg[0].type != HA_KEYTYPE_NUM) default_charset_info->cset->casedn(default_charset_info, - blob_key, length, blob_key, length); + (char*) blob_key, length, + (char*) blob_key, length); pos+=recinfo[0].length; } else if (recinfo[0].type == FIELD_VARCHAR) @@ -622,22 +624,22 @@ static void update_record(char *record) uint pack_length= HA_VARCHAR_PACKLENGTH(recinfo[0].length-1); uint length= pack_length == 1 ? (uint) *(uchar*) pos : uint2korr(pos); default_charset_info->cset->casedn(default_charset_info, - pos + pack_length, length, - pos + pack_length, length); + (char*) pos + pack_length, length, + (char*) pos + pack_length, length); pos+=recinfo[0].length; } else { if (keyinfo[0].seg[0].type != HA_KEYTYPE_NUM) default_charset_info->cset->casedn(default_charset_info, - pos, keyinfo[0].seg[0].length, - pos, keyinfo[0].seg[0].length); + (char*) pos, keyinfo[0].seg[0].length, + (char*) pos, keyinfo[0].seg[0].length); pos+=recinfo[0].length; } if (recinfo[1].type == FIELD_BLOB) { - char *column; + uchar *column; int length; length=uint4korr(pos); memcpy_fixed(&column,pos+4,sizeof(char*)); diff --git a/storage/maria/ma_test2.c b/storage/maria/ma_test2.c index 935be09850c..d9bd9bca9e0 100644 --- a/storage/maria/ma_test2.c +++ b/storage/maria/ma_test2.c @@ -40,7 +40,7 @@ static void get_options(int argc, char *argv[]); static uint rnd(uint max_value); static void fix_length(uchar *record,uint length); -static void put_blob_in_record(char *blob_pos,char **blob_buffer, +static void put_blob_in_record(uchar *blob_pos,char **blob_buffer, ulong *length); static void copy_key(struct st_maria_info *info,uint inx, uchar *record,uchar *key); @@ -58,8 +58,8 @@ static enum data_file_type record_type= DYNAMIC_RECORD; static uint keys=MARIA_KEYS,recant=1000; static uint use_blob=0; static uint16 key1[1001],key3[5000]; -static char record[300],record2[300],key[100],key2[100], - read_record[300],read_record2[300],read_record3[300]; +static uchar record[300],record2[300],key[100],key2[100]; +static uchar read_record[300],read_record2[300],read_record3[300]; static HA_KEYSEG glob_keyseg[MARIA_KEYS][MAX_PARTS]; /* Test program */ @@ -261,7 +261,7 @@ int main(int argc, char *argv[]) if (i == 72) goto end; #endif n1=rnd(1000); n2=rnd(100); n3=rnd(5000); - sprintf(record,"%6d:%4d:%8d:Pos: %4d ",n1,n2,n3,write_count); + sprintf((char*) record,"%6d:%4d:%8d:Pos: %4d ",n1,n2,n3,write_count); int4store(record+STANDARD_LENGTH-4,(long) i); fix_length(record,(uint) STANDARD_LENGTH+rnd(60)); put_blob_in_record(record+blob_pos,&blob_buffer, &blob_length); @@ -292,7 +292,7 @@ int main(int argc, char *argv[]) for (j=rnd(1000)+1 ; j>0 && key1[j] == 0 ; j--) ; if (!j) for (j=999 ; j>0 && key1[j] == 0 ; j--) ; - sprintf(key,"%6d",j); + sprintf((char*) key,"%6d",j); if (maria_rkey(file,read_record,0,key,HA_WHOLE_KEY,HA_READ_KEY_EXACT)) { printf("Test in loop: Can't find key: \"%s\"\n",key); @@ -325,7 +325,7 @@ int main(int argc, char *argv[]) for (j=rnd(1000)+1 ; j>0 && key1[j] == 0 ; j--) ; if (j != 0) { - sprintf(key,"%6d",j); + sprintf((char*) key,"%6d",j); if (maria_rkey(file,read_record,0,key,HA_WHOLE_KEY,HA_READ_KEY_EXACT)) { printf("can't find key1: \"%s\"\n",key); @@ -345,8 +345,8 @@ int main(int argc, char *argv[]) goto err; } opt_delete++; - key1[atoi(read_record+keyinfo[0].seg[0].start)]--; - key3[atoi(read_record+keyinfo[2].seg[0].start)]=0; + key1[atoi((char*) read_record+keyinfo[0].seg[0].start)]--; + key3[atoi((char*) read_record+keyinfo[2].seg[0].start)]=0; } else puts("Warning: Skipping delete test because no dupplicate keys"); @@ -361,17 +361,17 @@ int main(int argc, char *argv[]) for (i=0 ; i0 && key1[j] == 0 ; j--) ; if (j != 0) { - sprintf(key,"%6d",j); + sprintf((char*) key,"%6d",j); if (maria_rkey(file,read_record,0,key,HA_WHOLE_KEY,HA_READ_KEY_EXACT)) { - printf("can't find key1: \"%s\"\n",key); + printf("can't find key1: \"%s\"\n", (char*) key); goto err; } if (bcmp(read_record+keyinfo[0].seg[0].start, @@ -405,8 +405,8 @@ int main(int argc, char *argv[]) } else { - key1[atoi(read_record+keyinfo[0].seg[0].start)]--; - key3[atoi(read_record+keyinfo[2].seg[0].start)]=0; + key1[atoi((char*) read_record+keyinfo[0].seg[0].start)]--; + key3[atoi((char*) read_record+keyinfo[2].seg[0].start)]=0; key1[n1]++; key3[n3]=1; update++; } @@ -422,7 +422,7 @@ int main(int argc, char *argv[]) dupp_keys=key1[i]; j=i; } } - sprintf(key,"%6d",j); + sprintf((char*) key,"%6d",j); start=keyinfo[0].seg[0].start; length=keyinfo[0].seg[0].length; if (dupp_keys) @@ -720,8 +720,8 @@ int main(int argc, char *argv[]) key_range min_key, max_key; if (j > k) swap_variables(int, j, k); - sprintf(key,"%6d",j); - sprintf(key2,"%6d",k); + sprintf((char*) key,"%6d",j); + sprintf((char*) key2,"%6d",k); min_key.key= key; min_key.keypart_map= HA_WHOLE_KEY; @@ -797,13 +797,16 @@ int main(int argc, char *argv[]) { printf("scan with cache: I can only find: %d records of %d\n", ant,write_count-opt_delete); + maria_scan_end(file); goto err; } if (maria_extra(file,HA_EXTRA_NO_CACHE,0)) { puts("got error from maria_extra(HA_EXTRA_NO_CACHE)"); + maria_scan_end(file); goto err; } + maria_scan_end(file); ant=0; maria_scan_init(file); @@ -814,8 +817,10 @@ int main(int argc, char *argv[]) { printf("scan with cache: I can only find: %d records of %d\n", ant,write_count-opt_delete); + maria_scan_end(file); goto err; } + maria_scan_end(file); if (testflag == 5) goto end; @@ -885,16 +890,17 @@ int main(int argc, char *argv[]) else found_parts++; } - maria_scan_end(file); if (my_errno != HA_ERR_END_OF_FILE && my_errno != HA_ERR_RECORD_DELETED) printf("error: %d from maria_rrnd\n",my_errno); if (write_count != opt_delete) { printf("Deleted only %d of %d records (%d parts)\n",opt_delete,write_count, found_parts); + maria_scan_end(file); goto err; } end: + maria_scan_end(file); if (die_in_middle_of_transaction) { /* As commit record is not done, UNDO entries needs to be rolled back */ @@ -967,7 +973,7 @@ reads: %10lu\n", (ulong) maria_pagecache->global_cache_r_requests, (ulong) maria_pagecache->global_cache_read); } - end_pagecache(maria_pagecache,1); + maria_end(); my_free(blob_buffer, MYF(MY_ALLOW_ZERO_PTR)); my_end(silent ? MY_CHECK_ERROR : MY_CHECK_ERROR | MY_GIVE_INFO); return(0); @@ -1130,13 +1136,13 @@ static void fix_length(uchar *rec, uint length) bmove(rec+STANDARD_LENGTH, "0123456789012345678901234567890123456789012345678901234567890", length-STANDARD_LENGTH); - strfill(rec+length,STANDARD_LENGTH+60-length,' '); + strfill((char*) rec+length,STANDARD_LENGTH+60-length,' '); } /* fix_length */ /* Put maybe a blob in record */ -static void put_blob_in_record(char *blob_pos, char **blob_buffer, +static void put_blob_in_record(uchar *blob_pos, char **blob_buffer, ulong *blob_length) { ulong i,length; diff --git a/storage/maria/ma_test3.c b/storage/maria/ma_test3.c index c25dd5dcdc6..6d45e391e65 100644 --- a/storage/maria/ma_test3.c +++ b/storage/maria/ma_test3.c @@ -48,9 +48,9 @@ int test_read(MARIA_HA *,int),test_write(MARIA_HA *,int,int), test_update(MARIA_HA *,int,int),test_rrnd(MARIA_HA *,int); struct record { - char id[8]; - char nr[4]; - char text[10]; + uchar id[8]; + uchar nr[4]; + uchar text[10]; } record; @@ -242,8 +242,8 @@ int test_read(MARIA_HA *file,int id) for (i=0 ; i < 100 ; i++) { find=rnd(100000); - if (!maria_rkey(file,record.id,1,(uchar*) &find, - HA_WHOLE_KEY,HA_READ_KEY_EXACT)) + if (!maria_rkey(file,record.id,1,(uchar*) &find, HA_WHOLE_KEY, + HA_READ_KEY_EXACT)) found++; else { @@ -361,8 +361,8 @@ int test_write(MARIA_HA *file,int id,int lock_type) maria_extra(file,HA_EXTRA_WRITE_CACHE,0); } - sprintf(record.id,"%7d",getpid()); - strnmov(record.text,"Testing...", sizeof(record.text)); + sprintf((char*) record.id,"%7d",getpid()); + strnmov((char*) record.text,"Testing...", sizeof(record.text)); tries=(uint) rnd(100)+10; for (i=count=0 ; i < tries ; i++) @@ -418,15 +418,15 @@ int test_update(MARIA_HA *file,int id,int lock_type) } } bzero((char*) &new_record,sizeof(new_record)); - strmov(new_record.text,"Updated"); + strmov((char*) new_record.text,"Updated"); found=next=prev=update=0; for (i=0 ; i < 100 ; i++) { tmp=rnd(100000); int4store(find,tmp); - if (!maria_rkey(file,record.id,1,(uchar*) find, - HA_WHOLE_KEY,HA_READ_KEY_EXACT)) + if (!maria_rkey(file,record.id,1,(uchar*) find, HA_WHOLE_KEY, + HA_READ_KEY_EXACT)) found++; else { diff --git a/storage/maria/ma_update.c b/storage/maria/ma_update.c index 0cb2e2b648b..759f94e6450 100644 --- a/storage/maria/ma_update.c +++ b/storage/maria/ma_update.c @@ -110,7 +110,7 @@ int maria_update(register MARIA_HA *info, const uchar *oldrec, uchar *newrec) key_changed|=HA_STATE_WRITTEN; } changed|=((ulonglong) 1 << i); - if (_ma_ft_update(info,i,(char*) old_key,oldrec,newrec,pos)) + if (_ma_ft_update(info,i,old_key,oldrec,newrec,pos)) goto err; } } @@ -215,8 +215,8 @@ err: { if (share->keyinfo[i].flag & HA_FULLTEXT) { - if ((flag++ && _ma_ft_del(info,i,(char*) new_key,newrec,pos)) || - _ma_ft_add(info,i,(char*) old_key,oldrec,pos)) + if ((flag++ && _ma_ft_del(info,i,new_key,newrec,pos)) || + _ma_ft_add(info,i,old_key,oldrec,pos)) break; } else diff --git a/storage/maria/ma_write.c b/storage/maria/ma_write.c index b034d71ef9d..9982d770c81 100644 --- a/storage/maria/ma_write.c +++ b/storage/maria/ma_write.c @@ -134,7 +134,7 @@ int maria_write(MARIA_HA *info, uchar *record) } if (share->keyinfo[i].flag & HA_FULLTEXT ) { - if (_ma_ft_add(info,i,(char*) buff,record,filepos)) + if (_ma_ft_add(info,i, buff,record,filepos)) { if (local_lock_tree) rw_unlock(&share->key_root_lock[i]); @@ -243,7 +243,7 @@ err: */ if (share->keyinfo[i].flag & HA_FULLTEXT) { - if (_ma_ft_del(info,i,(char*) buff,record,filepos)) + if (_ma_ft_del(info,i,buff,record,filepos)) { if (local_lock_tree) rw_unlock(&share->key_root_lock[i]); @@ -615,7 +615,7 @@ int _ma_insert(register MARIA_HA *info, register MARIA_KEYDEF *keyinfo, we cannot easily dispatch an empty page here */ b+=blen+ft2len+2; for (a=anc_buff+a_length ; b < a ; b+=ft2len+2) - insert_dynamic(info->ft1_to_ft2, (char*) b); + insert_dynamic(info->ft1_to_ft2, b); /* fixing the page's length - it contains only one key now */ maria_putint(anc_buff,2+blen+ft2len+2,0); @@ -1041,7 +1041,7 @@ int maria_init_bulk_insert(MARIA_HA *info, ulong cache_size, ha_rows rows) DBUG_RETURN(0); if (rows && rows*total_keylength < cache_size) - cache_size=rows; + cache_size= (ulong)rows; else cache_size/=total_keylength*16; diff --git a/storage/maria/maria_chk.c b/storage/maria/maria_chk.c index f9ed249817e..227eb9727e4 100644 --- a/storage/maria/maria_chk.c +++ b/storage/maria/maria_chk.c @@ -332,7 +332,7 @@ static struct my_option my_long_options[] = (uchar**) &ft_stopword_file, (uchar**) &ft_stopword_file, 0, GET_STR, REQUIRED_ARG, 0, 0, 0, 0, 0, 0}, {"stats_method", OPT_STATS_METHOD, - "Specifies how index statistics collection code should threat NULLs. " + "Specifies how index statistics collection code should treat NULLs. " "Possible values of name are \"nulls_unequal\" (default behavior for 4.1/5.0), " "\"nulls_equal\" (emulate 4.0 behavior), and \"nulls_ignored\".", (uchar**) &maria_stats_method_str, (uchar**) &maria_stats_method_str, 0, @@ -447,7 +447,7 @@ static void usage(void) MySQL faster. You can check the calculated distribution\n\ by using '--description --verbose table_name'.\n\ --stats_method=name Specifies how index statistics collection code should\n\ - threat NULLs. Possible values of name are \"nulls_unequal\"\n\ + treat NULLs. Possible values of name are \"nulls_unequal\"\n\ (default for 4.1/5.0), \"nulls_equal\" (emulate 4.0), and \n\ \"nulls_ignored\".\n\ -d, --description Prints some information about table.\n\ diff --git a/storage/maria/maria_def.h b/storage/maria/maria_def.h index 09852f4dc86..cf8ab10e22f 100644 --- a/storage/maria/maria_def.h +++ b/storage/maria/maria_def.h @@ -281,8 +281,8 @@ typedef struct st_maria_share my_bool (*compare_unique)(struct st_maria_info *, MARIA_UNIQUEDEF *, const uchar *record, MARIA_RECORD_POS pos); /* Mapings to read/write the data file */ - uint (*file_read)(MARIA_HA *, uchar *, uint, my_off_t, myf); - uint (*file_write)(MARIA_HA *, uchar *, uint, my_off_t, myf); + uint (*file_read)(MARIA_HA *, uchar *, size_t, my_off_t, myf); + uint (*file_write)(MARIA_HA *, const uchar *, size_t, my_off_t, myf); invalidator_by_filename invalidator; /* query cache invalidator */ ulong this_process; /* processid */ ulong last_process; /* For table-change-check */ @@ -864,27 +864,27 @@ extern uint _ma_save_pack_length(uint version, uchar * block_buff, ulong length); extern uint _ma_calc_pack_length(uint version, ulong length); extern ulong _ma_calc_blob_length(uint length, const uchar *pos); -extern uint _ma_mmap_pread(MARIA_HA *info, uchar *Buffer, - uint Count, my_off_t offset, myf MyFlags); -extern uint _ma_mmap_pwrite(MARIA_HA *info, uchar *Buffer, - uint Count, my_off_t offset, myf MyFlags); -extern uint _ma_nommap_pread(MARIA_HA *info, uchar *Buffer, - uint Count, my_off_t offset, myf MyFlags); -extern uint _ma_nommap_pwrite(MARIA_HA *info, uchar *Buffer, - uint Count, my_off_t offset, myf MyFlags); +extern size_t _ma_mmap_pread(MARIA_HA *info, uchar *Buffer, + size_t Count, my_off_t offset, myf MyFlags); +extern size_t _ma_mmap_pwrite(MARIA_HA *info, const uchar *Buffer, + size_t Count, my_off_t offset, myf MyFlags); +extern size_t _ma_nommap_pread(MARIA_HA *info, uchar *Buffer, + size_t Count, my_off_t offset, myf MyFlags); +extern size_t _ma_nommap_pwrite(MARIA_HA *info, const uchar *Buffer, + size_t Count, my_off_t offset, myf MyFlags); uint _ma_state_info_write(MARIA_SHARE *share, uint pWrite); uint _ma_state_info_write_sub(File file, MARIA_STATE_INFO *state, uint pWrite); uint _ma_state_info_read_dsk(File file, MARIA_STATE_INFO *state); uint _ma_base_info_write(File file, MARIA_BASE_INFO *base); int _ma_keyseg_write(File file, const HA_KEYSEG *keyseg); -char *_ma_keyseg_read(char *ptr, HA_KEYSEG *keyseg); +uchar *_ma_keyseg_read(uchar *ptr, HA_KEYSEG *keyseg); uint _ma_keydef_write(File file, MARIA_KEYDEF *keydef); -char *_ma_keydef_read(char *ptr, MARIA_KEYDEF *keydef); +uchar *_ma_keydef_read(uchar *ptr, MARIA_KEYDEF *keydef); uint _ma_uniquedef_write(File file, MARIA_UNIQUEDEF *keydef); -char *_ma_uniquedef_read(char *ptr, MARIA_UNIQUEDEF *keydef); +uchar *_ma_uniquedef_read(uchar *ptr, MARIA_UNIQUEDEF *keydef); uint _ma_columndef_write(File file, MARIA_COLUMNDEF *columndef); -char *_ma_columndef_read(char *ptr, MARIA_COLUMNDEF *columndef); +uchar *_ma_columndef_read(uchar *ptr, MARIA_COLUMNDEF *columndef); ulong _ma_calc_total_blob_length(MARIA_HA *info, const uchar *record); ha_checksum _ma_checksum(MARIA_HA *info, const uchar *buf); ha_checksum _ma_static_checksum(MARIA_HA *info, const uchar *buf); diff --git a/storage/maria/maria_pack.c b/storage/maria/maria_pack.c index 83f88fcb0dc..88df230b5d4 100644 --- a/storage/maria/maria_pack.c +++ b/storage/maria/maria_pack.c @@ -526,14 +526,14 @@ static int compress(PACK_MRG_INFO *mrg,char *result_table) { /* Make a new indexfile based on first file in list */ uint length; - char *buff; + uchar *buff; strmov(org_name,result_table); /* Fix error messages */ VOID(fn_format(new_name,result_table,"",MARIA_NAME_IEXT,2)); if ((join_isam_file=my_create(new_name,0,tmpfile_createflag,MYF(MY_WME))) < 0) goto err; length=(uint) share->base.keystart; - if (!(buff=my_malloc(length,MYF(MY_WME)))) + if (!(buff= (uchar*) my_malloc(length,MYF(MY_WME)))) goto err; if (my_pread(share->kfile.file, buff, length, 0L, MYF(MY_WME | MY_NABP)) || my_write(join_isam_file,buff,length, @@ -657,7 +657,7 @@ static int compress(PACK_MRG_INFO *mrg,char *result_table) new_length=file_buffer.pos_in_file; if (!error && !test_only) { - char buff[MEMMAP_EXTRA_MARGIN]; /* End marginal for memmap */ + uchar buff[MEMMAP_EXTRA_MARGIN]; /* End marginal for memmap */ bzero(buff,sizeof(buff)); error=my_write(file_buffer.file,buff,sizeof(buff), MYF(MY_WME | MY_NABP | MY_WAIT_IF_FULL)) != 0; -- cgit v1.2.1 From a1bb4afa7ed8c77122c508bf2536e7a3f93ae1df Mon Sep 17 00:00:00 2001 From: unknown Date: Sat, 6 Oct 2007 11:39:26 +0200 Subject: after-merge fixes for 64bit storage/maria/ma_test1.c: typo (?) fixed --- storage/maria/ma_test1.c | 2 +- storage/maria/maria_def.h | 4 ++-- storage/myisam/myisamdef.h | 12 ++++++------ 3 files changed, 9 insertions(+), 9 deletions(-) diff --git a/storage/maria/ma_test1.c b/storage/maria/ma_test1.c index fbb91256bc2..7c7305279aa 100644 --- a/storage/maria/ma_test1.c +++ b/storage/maria/ma_test1.c @@ -588,7 +588,7 @@ static void create_record(uchar *record,uint rownr) sprintf((char*) pos+pack_length, "... row: %d", rownr); tmp= strlen((char*) pos+pack_length); if (pack_length == 1) - pos= (uchar) tmp; + *pos= (uchar) tmp; else int2store(pos,tmp); } diff --git a/storage/maria/maria_def.h b/storage/maria/maria_def.h index bd5746cbb94..e4c8c3efcdc 100644 --- a/storage/maria/maria_def.h +++ b/storage/maria/maria_def.h @@ -281,8 +281,8 @@ typedef struct st_maria_share my_bool (*compare_unique)(struct st_maria_info *, MARIA_UNIQUEDEF *, const uchar *record, MARIA_RECORD_POS pos); /* Mapings to read/write the data file */ - uint (*file_read)(MARIA_HA *, uchar *, size_t, my_off_t, myf); - uint (*file_write)(MARIA_HA *, const uchar *, size_t, my_off_t, myf); + size_t (*file_read)(MARIA_HA *, uchar *, size_t, my_off_t, myf); + size_t (*file_write)(MARIA_HA *, const uchar *, size_t, my_off_t, myf); invalidator_by_filename invalidator; /* query cache invalidator */ ulong this_process; /* processid */ ulong last_process; /* For table-change-check */ diff --git a/storage/myisam/myisamdef.h b/storage/myisam/myisamdef.h index b0b6da03015..fde80172643 100644 --- a/storage/myisam/myisamdef.h +++ b/storage/myisam/myisamdef.h @@ -185,8 +185,8 @@ typedef struct st_mi_isam_share ha_checksum(*calc_checksum) (struct st_myisam_info *, const uchar*); int(*compare_unique) (struct st_myisam_info *, MI_UNIQUEDEF *, const uchar *record, my_off_t pos); - uint(*file_read) (MI_INFO *, uchar *, size_t, my_off_t, myf); - uint(*file_write) (MI_INFO *, const uchar *, size_t, my_off_t, myf); + size_t (*file_read) (MI_INFO *, uchar *, size_t, my_off_t, myf); + size_t (*file_write) (MI_INFO *, const uchar *, size_t, my_off_t, myf); invalidator_by_filename invalidator; /* query cache invalidator */ ulong this_process; /* processid */ ulong last_process; /* For table-change-check */ @@ -664,13 +664,13 @@ extern void _mi_unmap_file(MI_INFO *info); extern uint save_pack_length(uint version, uchar *block_buff, ulong length); extern uint calc_pack_length(uint version, ulong length); extern size_t mi_mmap_pread(MI_INFO *info, uchar *Buffer, - uint Count, my_off_t offset, myf MyFlags); + size_t Count, my_off_t offset, myf MyFlags); extern size_t mi_mmap_pwrite(MI_INFO *info, const uchar *Buffer, - uint Count, my_off_t offset, myf MyFlags); + size_t Count, my_off_t offset, myf MyFlags); extern size_t mi_nommap_pread(MI_INFO *info, uchar *Buffer, - uint Count, my_off_t offset, myf MyFlags); + size_t Count, my_off_t offset, myf MyFlags); extern size_t mi_nommap_pwrite(MI_INFO *info, const uchar *Buffer, - uint Count, my_off_t offset, myf MyFlags); + size_t Count, my_off_t offset, myf MyFlags); uint mi_state_info_write(File file, MI_STATE_INFO *state, uint pWrite); uchar *mi_state_info_read(uchar *ptr, MI_STATE_INFO *state); -- cgit v1.2.1 From 6c7a278c836c086d59a84589988536d654fd0a57 Mon Sep 17 00:00:00 2001 From: unknown Date: Sat, 6 Oct 2007 17:56:52 +0200 Subject: remove maria_* variables from the server, declare them in ha_maria.cc note: this does *not* move pagecache* variables. sql/mysqld.cc: remove maria_* variables from the server, declare them in ha_maria.cc sql/set_var.cc: remove maria_* variables from the server, declare them in ha_maria.cc sql/sql_class.h: remove maria_* variables from the server, declare them in ha_maria.cc sql/sql_plugin.cc: bugfix: support for ENUM/SET thd local plugin variables storage/maria/ha_maria.cc: remove maria_* variables from the server, declare them in ha_maria.cc --- sql/mysqld.cc | 50 ++++------------------------------ sql/set_var.cc | 25 ++--------------- sql/sql_class.h | 68 ++++++++++++++++++++++------------------------- sql/sql_plugin.cc | 2 ++ storage/maria/ha_maria.cc | 59 +++++++++++++++++++++++++++++++--------- 5 files changed, 87 insertions(+), 117 deletions(-) diff --git a/sql/mysqld.cc b/sql/mysqld.cc index c79975fa508..296cdfdce12 100644 --- a/sql/mysqld.cc +++ b/sql/mysqld.cc @@ -506,7 +506,6 @@ char *mysqld_unix_port, *opt_mysql_tmpdir; const char **errmesg; /* Error messages */ const char *myisam_recover_options_str="OFF"; const char *myisam_stats_method_str="nulls_unequal"; -const char *maria_stats_method_str="nulls_unequal"; /* name of reference on left espression in rewritten IN subquery */ const char *in_left_expr_name= ""; @@ -550,7 +549,7 @@ MY_LOCALE *my_default_lc_time_names; SHOW_COMP_OPTION have_ssl, have_symlink, have_dlopen, have_query_cache; SHOW_COMP_OPTION have_geometry, have_rtree_keys; -SHOW_COMP_OPTION have_crypt, have_compress, have_maria_db; +SHOW_COMP_OPTION have_crypt, have_compress; /* Thread specific variables */ @@ -2208,7 +2207,7 @@ or misconfigured. This error can also be caused by malfunctioning hardware.\n", We will try our best to scrape up some info that will hopefully help diagnose\n\ the problem, but since we have already crashed, something is definitely wrong\n\ and this may fail.\n\n"); - fprintf(stderr, "key_buffer_size=%lu\n", + fprintf(stderr, "key_buffer_size=%lu\n", (ulong) dflt_key_cache->key_cache_mem_size); #ifdef WITH_MARIA_STORAGE_ENGINE fprintf(stderr, "page_buffer_size=%lu\n", @@ -5059,10 +5058,6 @@ enum options_mysqld OPT_MYISAM_USE_MMAP, OPT_MYISAM_REPAIR_THREADS, OPT_MYISAM_STATS_METHOD, - OPT_MARIA_BLOCK_SIZE, - OPT_MARIA_MAX_SORT_FILE_SIZE, OPT_MARIA_SORT_BUFFER_SIZE, - OPT_MARIA_USE_MMAP, OPT_MARIA_REPAIR_THREADS, - OPT_MARIA_STATS_METHOD, OPT_PAGECACHE_BUFFER_SIZE, OPT_PAGECACHE_DIVISION_LIMIT, OPT_PAGECACHE_AGE_THRESHOLD, @@ -5079,7 +5074,7 @@ enum options_mysqld OPT_SORT_BUFFER, OPT_TABLE_OPEN_CACHE, OPT_TABLE_DEF_CACHE, OPT_THREAD_CONCURRENCY, OPT_THREAD_CACHE_SIZE, OPT_TMP_TABLE_SIZE, OPT_THREAD_STACK, - OPT_WAIT_TIMEOUT, + OPT_WAIT_TIMEOUT, OPT_ERROR_LOG_FILE, OPT_DEFAULT_WEEK_FORMAT, OPT_RANGE_ALLOC_BLOCK_SIZE, OPT_ALLOW_SUSPICIOUS_UDFS, @@ -6046,40 +6041,6 @@ log and this option does nothing anymore.", 0 #endif , 0, 2, 0, 1, 0}, -#ifdef WITH_MARIA_STORAGE_ENGINE - {"maria_block_size", OPT_MARIA_BLOCK_SIZE, - "Block size to be used for MARIA index pages.", - (uchar**) &maria_block_size, - (uchar**) &maria_block_size, 0, GET_ULONG, REQUIRED_ARG, - MARIA_KEY_BLOCK_LENGTH, MARIA_MIN_KEY_BLOCK_LENGTH, - MARIA_MAX_KEY_BLOCK_LENGTH, - 0, MARIA_MIN_KEY_BLOCK_LENGTH, 0}, - {"maria_max_sort_file_size", OPT_MARIA_MAX_SORT_FILE_SIZE, - "Don't use the fast sort index method to created index if the temporary " - "file would get bigger than this.", - (uchar**) &global_system_variables.maria_max_sort_file_size, - (uchar**) &max_system_variables.maria_max_sort_file_size, 0, - GET_ULL, REQUIRED_ARG, (longlong) LONG_MAX, 0, (ulonglong) MAX_FILE_SIZE, - 0, 1024*1024, 0}, - {"maria_repair_threads", OPT_MARIA_REPAIR_THREADS, - "Number of threads to use when repairing maria tables. The value of 1 " - "disables parallel repair.", - (uchar**) &global_system_variables.maria_repair_threads, - (uchar**) &max_system_variables.maria_repair_threads, 0, - GET_ULONG, REQUIRED_ARG, 1, 1, ~0L, 0, 1, 0}, - {"maria_sort_buffer_size", OPT_MARIA_SORT_BUFFER_SIZE, - "The buffer that is allocated when sorting the index when doing a REPAIR " - "or when creating indexes with CREATE INDEX or ALTER TABLE.", - (uchar**) &global_system_variables.maria_sort_buff_size, - (uchar**) &max_system_variables.maria_sort_buff_size, 0, - GET_ULONG, REQUIRED_ARG, 8192*1024, 4, ~0L, 0, 1, 0}, - {"maria_stats_method", OPT_MARIA_STATS_METHOD, - "Specifies how maria index statistics collection code should threat NULLs. " - "Possible values of name are \"nulls_unequal\" (default behavior for 4.1/5.0), " - "\"nulls_equal\" (emulate 4.0 behavior), and \"nulls_ignored\".", - (uchar**) &maria_stats_method_str, (uchar**) &maria_stats_method_str, 0, - GET_STR, REQUIRED_ARG, 0, 0, 0, 0, 0, 0}, -#endif {"max_allowed_packet", OPT_MAX_ALLOWED_PACKET, "Max packetlength to send/receive from to server.", (uchar**) &global_system_variables.max_allowed_packet, @@ -7194,7 +7155,7 @@ static void mysql_init_variables(void) global_query_id= thread_id= 1L; strmov(server_version, MYSQL_SERVER_VERSION); myisam_recover_options_str= sql_mode_str= "OFF"; - myisam_stats_method_str= maria_stats_method_str= "nulls_unequal"; + myisam_stats_method_str= "nulls_unequal"; my_bind_addr = htonl(INADDR_ANY); threads.empty(); thread_cache.empty(); @@ -7229,7 +7190,7 @@ static void mysql_init_variables(void) master_password= master_host= 0; master_info_file= (char*) "master.info", relay_log_info_file= (char*) "relay-log.info"; - master_ssl_key= master_ssl_cert= master_ssl_ca= + master_ssl_key= master_ssl_cert= master_ssl_ca= master_ssl_capath= master_ssl_cipher= 0; report_user= report_password = report_host= 0; /* TO BE DELETED */ opt_relay_logname= opt_relaylog_index_name= 0; @@ -7257,7 +7218,6 @@ static void mysql_init_variables(void) when collecting index statistics for MyISAM tables. */ global_system_variables.myisam_stats_method= MI_STATS_METHOD_NULLS_NOT_EQUAL; - global_system_variables.maria_stats_method= MI_STATS_METHOD_NULLS_NOT_EQUAL; /* Variables that depends on compile options */ #ifndef DBUG_OFF diff --git a/sql/set_var.cc b/sql/set_var.cc index 29cb66f16a6..95bdc54a5e9 100644 --- a/sql/set_var.cc +++ b/sql/set_var.cc @@ -116,7 +116,6 @@ static void fix_max_join_size(THD *thd, enum_var_type type); static void fix_query_cache_size(THD *thd, enum_var_type type); static void fix_query_cache_min_res_unit(THD *thd, enum_var_type type); static void fix_myisam_max_sort_file_size(THD *thd, enum_var_type type); -static void fix_maria_max_sort_file_size(THD *thd, enum_var_type type); static void fix_max_binlog_size(THD *thd, enum_var_type type); static void fix_max_relay_log_size(THD *thd, enum_var_type type); static void fix_max_connections(THD *thd, enum_var_type type); @@ -258,7 +257,7 @@ static sys_var_bool_ptr sys_local_infile(&vars, "local_infile", static sys_var_trust_routine_creators sys_trust_routine_creators(&vars, "log_bin_trust_routine_creators", &trust_function_creators); -static sys_var_bool_ptr +static sys_var_bool_ptr sys_trust_function_creators(&vars, "log_bin_trust_function_creators", &trust_function_creators); static sys_var_bool_ptr @@ -341,20 +340,10 @@ static sys_var_thd_ulong sys_myisam_repair_threads(&vars, "myisam_repair_t static sys_var_thd_ulong sys_myisam_sort_buffer_size(&vars, "myisam_sort_buffer_size", &SV::myisam_sort_buff_size); static sys_var_bool_ptr sys_myisam_use_mmap(&vars, "myisam_use_mmap", &opt_myisam_use_mmap); - static sys_var_thd_enum sys_myisam_stats_method(&vars, "myisam_stats_method", &SV::myisam_stats_method, &myisam_stats_method_typelib, NULL); - -static sys_var_thd_ulonglong sys_maria_max_sort_file_size(&vars, "maria_max_sort_file_size", &SV::maria_max_sort_file_size, fix_maria_max_sort_file_size, 1); -static sys_var_thd_ulong sys_maria_repair_threads(&vars, "maria_repair_threads", &SV::maria_repair_threads); -static sys_var_thd_ulong sys_maria_sort_buffer_size(&vars, "maria_sort_buffer_size", &SV::maria_sort_buff_size); -static sys_var_thd_enum sys_maria_stats_method(&vars, "maria_stats_method", - &SV::maria_stats_method, - &myisam_stats_method_typelib, - NULL); - static sys_var_thd_ulong sys_net_buffer_length(&vars, "net_buffer_length", &SV::net_buffer_length); static sys_var_thd_ulong sys_net_read_timeout(&vars, "net_read_timeout", @@ -651,7 +640,7 @@ static sys_var_have_plugin sys_have_csv(&vars, "have_csv", C_STRING_WITH_LEN("cs static sys_var_have_variable sys_have_dlopen(&vars, "have_dynamic_loading", &have_dlopen); static sys_var_have_variable sys_have_geometry(&vars, "have_geometry", &have_geometry); static sys_var_have_plugin sys_have_innodb(&vars, "have_innodb", C_STRING_WITH_LEN("innodb"), MYSQL_STORAGE_ENGINE_PLUGIN); -static sys_var_have_variable sys_have_maria_db(&vars, "have_maria", &have_maria_db); +static sys_var_have_plugin sys_have_maria(&vars, "have_maria", C_STRING_WITH_LEN("maria"), MYSQL_STORAGE_ENGINE_PLUGIN); static sys_var_have_plugin sys_have_ndbcluster(&vars, "have_ndbcluster", C_STRING_WITH_LEN("ndbcluster"), MYSQL_STORAGE_ENGINE_PLUGIN); static sys_var_have_variable sys_have_openssl(&vars, "have_openssl", &have_ssl); static sys_var_have_variable sys_have_ssl(&vars, "have_ssl", &have_ssl); @@ -867,16 +856,6 @@ fix_myisam_max_sort_file_size(THD *thd, enum_var_type type) (my_off_t) global_system_variables.myisam_max_sort_file_size; } -static void -fix_maria_max_sort_file_size(THD *thd, enum_var_type type) -{ -#ifdef WITH_MARIA_STORAGE_ENGINE - maria_max_temp_length= - (my_off_t) global_system_variables.myisam_max_sort_file_size; -#endif -} - - /* Set the OPTION_BIG_SELECTS flag if max_join_size == HA_POS_ERROR */ diff --git a/sql/sql_class.h b/sql/sql_class.h index f892abb69ee..f65e5c3b77c 100644 --- a/sql/sql_class.h +++ b/sql/sql_class.h @@ -232,20 +232,19 @@ struct system_variables { /* How dynamically allocated system variables are handled: - + The global_system_variables and max_system_variables are "authoritative" They both should have the same 'version' and 'size'. When attempting to access a dynamic variable, if the session version is out of date, then the session version is updated and realloced if neccessary and bytes copied from global to make up for missing data. - */ + */ ulong dynamic_variables_version; char* dynamic_variables_ptr; uint dynamic_variables_head; /* largest valid variable offset */ uint dynamic_variables_size; /* how many bytes are in use */ - + ulonglong myisam_max_extra_sort_file_size; - ulonglong maria_max_sort_file_size; ulonglong myisam_max_sort_file_size; ulonglong max_heap_table_size; ulonglong tmp_table_size; @@ -262,9 +261,6 @@ struct system_variables ulong max_tmp_tables; ulong max_insert_delayed_threads; ulong min_examined_row_limit; - ulong maria_repair_threads; - ulong maria_sort_buff_size; - ulong maria_stats_method; ulong multi_range_count; ulong myisam_repair_threads; ulong myisam_sort_buff_size; @@ -312,9 +308,9 @@ struct system_variables my_bool low_priority_updates; my_bool new_mode; - /* + /* compatibility option: - - index usage hints (USE INDEX without a FOR clause) behave as in 5.0 + - index usage hints (USE INDEX without a FOR clause) behave as in 5.0 */ my_bool old_mode; my_bool query_cache_wlock_invalidate; @@ -538,7 +534,7 @@ class Server_side_cursor; - prepared, that is, contain placeholders, - opened as cursors. We maintain 1 to 1 relationship between statement and cursor - if user wants to create another cursor for his - query, we create another statement for it. + query, we create another statement for it. To perform some action with statement we reset THD part to the state of that statement, do the action, and then save back modified state from THD to the statement. It will be changed in near future, and Statement will @@ -589,7 +585,7 @@ public: it. We will see the query_length field as either 0, or the right value for it. Assuming that the write and read of an n-bit memory field in an n-bit - computer is atomic, we can avoid races in the above way. + computer is atomic, we can avoid races in the above way. This printing is needed at least in SHOW PROCESSLIST and SHOW INNODB STATUS. */ @@ -741,7 +737,7 @@ public: { return (*priv_host ? priv_host : (char *)"%"); } - + bool set_user(char *user_arg); #ifndef NO_EMBEDDED_ACCESS_CHECKS @@ -1084,7 +1080,7 @@ public: /* One thread can hold up to one named user-level lock. This variable points to a lock object if the lock is present. See item_func.cc and - chapter 'Miscellaneous functions', for functions GET_LOCK, RELEASE_LOCK. + chapter 'Miscellaneous functions', for functions GET_LOCK, RELEASE_LOCK. */ User_level_lock *ull; #ifndef DBUG_OFF @@ -1103,7 +1099,7 @@ public: time_t start_time, user_time; ulonglong connect_utime, thr_create_utime; // track down slow pthread_create ulonglong start_utime, utime_after_lock; - + thr_lock_type update_lock_default; Delayed_insert *di; @@ -1471,7 +1467,7 @@ public: bool substitute_null_with_insert_id; bool in_lock_tables; bool query_error, bootstrap, cleanup_done; - + /** is set if some thread specific value(s) used in a statement. */ bool thread_specific_used; bool charset_is_system_charset, charset_is_collation_connection; @@ -1503,10 +1499,10 @@ public: ulong ulong_value; ulonglong ulonglong_value; } sys_var_tmp; - + struct { - /* - If true, mysql_bin_log::write(Log_event) call will not write events to + /* + If true, mysql_bin_log::write(Log_event) call will not write events to binlog, and maintain 2 below variables instead (use mysql_bin_log.start_union_events to turn this on) */ @@ -1517,13 +1513,13 @@ public: */ bool unioned_events; /* - If TRUE, at least one mysql_bin_log::write(Log_event e), where - e.cache_stmt == TRUE call has been made after last + If TRUE, at least one mysql_bin_log::write(Log_event e), where + e.cache_stmt == TRUE call has been made after last mysql_bin_log.start_union_events() call. */ bool unioned_events_trans; - - /* + + /* 'queries' (actually SP statements) that run under inside this binlog union have thd->query_id >= first_query_id. */ @@ -1554,7 +1550,7 @@ public: killing mysqld) where it's vital to not allocate excessive and not used memory. Note, that we still don't return error from init_for_queries(): if preallocation fails, we should notice that at the first call to - alloc_root. + alloc_root. */ void init_for_queries(); void change_user(void); @@ -1584,12 +1580,12 @@ public: The query can be logged row-based or statement-based */ ROW_QUERY_TYPE, - + /* The query has to be logged statement-based */ STMT_QUERY_TYPE, - + /* The query represents a change to a table in the "mysql" database and is currently mapped to ROW_QUERY_TYPE. @@ -1597,7 +1593,7 @@ public: MYSQL_QUERY_TYPE, QUERY_TYPE_COUNT }; - + int binlog_query(enum_binlog_query_type qtype, char const *query, ulong query_len, bool is_trans, bool suppress_use, @@ -1819,7 +1815,7 @@ public: if ((temporary_tables == NULL) && (in_sub_stmt == 0) && (system_thread != SYSTEM_THREAD_NDBCLUSTER_BINLOG)) { - current_stmt_binlog_row_based= + current_stmt_binlog_row_based= test(variables.binlog_format == BINLOG_FORMAT_ROW); } } @@ -2165,8 +2161,8 @@ public: #include -/* - Param to create temporary tables when doing SELECT:s +/* + Param to create temporary tables when doing SELECT:s NOTE This structure is copied using memcpy as a part of JOIN. */ @@ -2194,8 +2190,8 @@ public: uint quick_group; bool using_indirect_summary_function; /* If >0 convert all blob fields to varchar(convert_blob_length) */ - uint convert_blob_length; - CHARSET_INFO *table_charset; + uint convert_blob_length; + CHARSET_INFO *table_charset; bool schema_table; /* True if GROUP BY and its aggregate functions are already computed @@ -2329,7 +2325,7 @@ public: else db= db_arg; } - inline Table_ident(LEX_STRING table_arg) + inline Table_ident(LEX_STRING table_arg) :table(table_arg), sel((SELECT_LEX_UNIT *)0) { db.str=0; @@ -2375,7 +2371,7 @@ class user_var_entry }; /* - Unique -- class for unique (removing of duplicates). + Unique -- class for unique (removing of duplicates). Puts all values to the TREE. If the tree becomes too big, it's dumped to the file. User can request sorted values, or just iterate through them. In the last case tree merging is performed in @@ -2409,9 +2405,9 @@ public: } bool get(TABLE *table); - static double get_use_cost(uint *buffer, uint nkeys, uint key_size, + static double get_use_cost(uint *buffer, uint nkeys, uint key_size, ulonglong max_in_memory_size); - inline static int get_cost_calc_buff_size(ulong nkeys, uint key_size, + inline static int get_cost_calc_buff_size(ulong nkeys, uint key_size, ulonglong max_in_memory_size) { register ulonglong max_elems_in_tree= @@ -2466,7 +2462,7 @@ class multi_update :public select_result_interceptor uint table_count; /* List of tables referenced in the CHECK OPTION condition of - the updated view excluding the updated table. + the updated view excluding the updated table. */ List unupdated_check_opt_tables; Copy_field *copy_field; diff --git a/sql/sql_plugin.cc b/sql/sql_plugin.cc index 2d33da29b77..fa480f2d588 100644 --- a/sql/sql_plugin.cc +++ b/sql/sql_plugin.cc @@ -2168,9 +2168,11 @@ static st_bookmark *register_var(const char *plugin, const char *name, size= sizeof(int); break; case PLUGIN_VAR_LONG: + case PLUGIN_VAR_ENUM: size= sizeof(long); break; case PLUGIN_VAR_LONGLONG: + case PLUGIN_VAR_SET: size= sizeof(ulonglong); break; case PLUGIN_VAR_STR: diff --git a/storage/maria/ha_maria.cc b/storage/maria/ha_maria.cc index 62c3968c0ca..6a3282b0454 100644 --- a/storage/maria/ha_maria.cc +++ b/storage/maria/ha_maria.cc @@ -78,6 +78,32 @@ TYPELIB maria_stats_method_typelib= maria_stats_method_names, NULL }; +static MYSQL_SYSVAR_ULONG(block_size, maria_block_size, + PLUGIN_VAR_RQCMDARG | PLUGIN_VAR_READONLY, + "Block size to be used for MARIA index pages.", 0, 0, + MARIA_KEY_BLOCK_LENGTH, MARIA_MIN_KEY_BLOCK_LENGTH, + MARIA_MAX_KEY_BLOCK_LENGTH, MARIA_MIN_KEY_BLOCK_LENGTH); + +static MYSQL_SYSVAR_ULONGLONG(max_sort_file_size, + maria_max_temp_length, PLUGIN_VAR_RQCMDARG, + "Don't use the fast sort index method to created index if the " + "temporary file would get bigger than this.", + 0, 0, MAX_FILE_SIZE, 0, MAX_FILE_SIZE, 1024*1024); + +static MYSQL_THDVAR_ULONG(repair_threads, PLUGIN_VAR_RQCMDARG, + "Number of threads to use when repairing maria tables. The value of 1 " + "disables parallel repair.", + 0, 0, 1, 1, ~0L, 1); + +static MYSQL_THDVAR_ULONG(sort_buffer_size, PLUGIN_VAR_RQCMDARG, + "The buffer that is allocated when sorting the index when doing a " + "REPAIR or when creating indexes with CREATE INDEX or ALTER TABLE.", + 0, 0, 8192*1024, 4, ~0L, 1); + +static MYSQL_THDVAR_ENUM(stats_method, PLUGIN_VAR_RQCMDARG, + "Specifies how maria index statistics collection code should threat " + "NULLs. Possible values of name are \"nulls_unequal\", \"nulls_equal\", " + "and \"nulls_ignored\".", 0, 0, 0, &maria_stats_method_typelib); /***************************************************************************** ** MARIA tables @@ -843,8 +869,7 @@ int ha_maria::check(THD * thd, HA_CHECK_OPT * check_opt) param.db_name= table->s->db.str; param.table_name= table->alias; param.testflag= check_opt->flags | T_CHECK | T_SILENT; - param.stats_method= (enum_handler_stats_method) thd->variables. - maria_stats_method; + param.stats_method= (enum_handler_stats_method)THDVAR(thd,stats_method); if (!(table->db_stat & HA_READ_ONLY)) param.testflag |= T_STATISTICS; @@ -935,8 +960,7 @@ int ha_maria::analyze(THD *thd, HA_CHECK_OPT * check_opt) param.testflag= (T_FAST | T_CHECK | T_SILENT | T_STATISTICS | T_DONT_CHECK_CHECKSUM); param.using_global_keycache= 1; - param.stats_method= (enum_handler_stats_method) thd->variables. - maria_stats_method; + param.stats_method= (enum_handler_stats_method)THDVAR(thd,stats_method); if (!(share->state.changed & STATE_NOT_ANALYZED)) return HA_ADMIN_ALREADY_DONE; @@ -1205,7 +1229,7 @@ int ha_maria::repair(THD *thd, HA_CHECK ¶m, bool do_optimize) param.testflag |= T_STATISTICS; // We get this for free statistics_done= 1; /* TODO: Remove BLOCK_RECORD test when parallel works with blocks */ - if (thd->variables.maria_repair_threads > 1 && + if (THDVAR(thd,repair_threads) > 1 && file->s->data_file_type != BLOCK_RECORD) { char buf[40]; @@ -1521,9 +1545,8 @@ int ha_maria::enable_indexes(uint mode) param.testflag= (T_SILENT | T_REP_BY_SORT | T_QUICK | T_CREATE_MISSING_KEYS); param.myf_rw &= ~MY_WAIT_IF_FULL; - param.sort_buffer_length= thd->variables.maria_sort_buff_size; - param.stats_method= - (enum_handler_stats_method) thd->variables.maria_stats_method; + param.sort_buffer_length= THDVAR(thd,sort_buffer_size); + param.stats_method= (enum_handler_stats_method)THDVAR(thd,stats_method); param.tmpdir= &mysql_tmpdir_list; if ((error= (repair(thd, param, 0) != HA_ADMIN_OK)) && param.retry_repair) { @@ -2461,6 +2484,16 @@ my_bool ha_maria::register_query_cache_table(THD *thd, char *table_name, } #endif +static struct st_mysql_sys_var* system_variables[]= { + MYSQL_SYSVAR(block_size), + MYSQL_SYSVAR(max_sort_file_size), + MYSQL_SYSVAR(repair_threads), + MYSQL_SYSVAR(sort_buffer_size), + MYSQL_SYSVAR(stats_method), + NULL +}; + + struct st_mysql_storage_engine maria_storage_engine= { MYSQL_HANDLERTON_INTERFACE_VERSION }; @@ -2473,11 +2506,11 @@ mysql_declare_plugin(maria) "MySQL AB", "Traditional transactional MySQL tables", PLUGIN_LICENSE_GPL, - ha_maria_init, /* Plugin Init */ - NULL, /* Plugin Deinit */ - 0x0100, /* 1.0 */ + ha_maria_init, /* Plugin Init */ + NULL, /* Plugin Deinit */ + 0x0100, /* 1.0 */ NULL, /* status variables */ - NULL, /* system variables */ - NULL /* config options */ + system_variables, /* system variables */ + NULL } mysql_declare_plugin_end; -- cgit v1.2.1